ltcai 4.0.1 → 4.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +28 -23
- package/desktop/electron/main.cjs +44 -0
- package/docs/CHANGELOG.md +42 -0
- package/docs/V4_1_FRONTEND_ARCHITECTURE_REVIEW.md +65 -0
- package/docs/V4_1_FRONTEND_MIGRATION_REPORT.md +70 -0
- package/docs/V4_1_VALIDATION_REPORT.md +47 -0
- package/docs/V4_DIGITAL_BRAIN_RECOVERY.md +26 -19
- package/frontend/index.html +24 -0
- package/frontend/openapi.json +14190 -0
- package/frontend/src/App.tsx +184 -0
- package/frontend/src/api/client.ts +317 -0
- package/frontend/src/api/openapi.ts +16637 -0
- package/frontend/src/components/primitives.tsx +204 -0
- package/frontend/src/components/ui/badge.tsx +27 -0
- package/frontend/src/components/ui/button.tsx +37 -0
- package/frontend/src/components/ui/card.tsx +22 -0
- package/frontend/src/components/ui/input.tsx +16 -0
- package/frontend/src/components/ui/textarea.tsx +16 -0
- package/frontend/src/lib/utils.ts +33 -0
- package/frontend/src/main.tsx +23 -0
- package/frontend/src/pages/Act.tsx +245 -0
- package/frontend/src/pages/Ask.tsx +200 -0
- package/frontend/src/pages/Brain.tsx +267 -0
- package/frontend/src/pages/Capture.tsx +158 -0
- package/frontend/src/pages/Library.tsx +187 -0
- package/frontend/src/pages/System.tsx +344 -0
- package/frontend/src/routes.ts +85 -0
- package/frontend/src/store/appStore.ts +54 -0
- package/frontend/src/styles.css +107 -0
- package/latticeai/__init__.py +1 -1
- package/latticeai/api/setup.py +5 -4
- package/latticeai/api/static_routes.py +4 -4
- package/latticeai/core/marketplace.py +1 -1
- package/latticeai/core/multi_agent.py +1 -1
- package/latticeai/core/workspace_os.py +1 -1
- package/package.json +54 -15
- package/scripts/build_frontend_assets.mjs +38 -0
- package/scripts/bump_version.py +1 -1
- package/scripts/export_openapi.py +31 -0
- package/scripts/lint_frontend.mjs +86 -0
- package/scripts/run_python.mjs +47 -0
- package/src-tauri/Cargo.lock +4833 -0
- package/src-tauri/Cargo.toml +19 -0
- package/src-tauri/build.rs +3 -0
- package/src-tauri/capabilities/default.json +7 -0
- package/src-tauri/src/main.rs +78 -0
- package/src-tauri/tauri.conf.json +36 -0
- package/static/app/asset-manifest.json +32 -0
- package/static/app/assets/core-CwxXejkd.js +2 -0
- package/static/app/assets/core-CwxXejkd.js.map +1 -0
- package/static/app/assets/index-CJRAzNnf.js +333 -0
- package/static/app/assets/index-CJRAzNnf.js.map +1 -0
- package/static/app/assets/index-CSwBBgf4.css +2 -0
- package/static/app/index.html +25 -0
- package/static/manifest.json +2 -2
- package/static/sw.js +4 -4
- package/scripts/build_v3_assets.mjs +0 -170
- package/scripts/lint_v3.mjs +0 -120
- package/static/v3/asset-manifest.json +0 -63
- package/static/v3/css/lattice.base.49deefb5.css +0 -128
- package/static/v3/css/lattice.base.css +0 -128
- package/static/v3/css/lattice.components.cde18231.css +0 -472
- package/static/v3/css/lattice.components.css +0 -472
- package/static/v3/css/lattice.shell.29d36d85.css +0 -452
- package/static/v3/css/lattice.shell.css +0 -452
- package/static/v3/css/lattice.tokens.304cbc40.css +0 -135
- package/static/v3/css/lattice.tokens.css +0 -135
- package/static/v3/css/lattice.views.0a18b6c5.css +0 -360
- package/static/v3/css/lattice.views.css +0 -360
- package/static/v3/index.html +0 -68
- package/static/v3/js/app.c5c80c46.js +0 -26
- package/static/v3/js/app.js +0 -26
- package/static/v3/js/core/api.ba0fbf14.js +0 -625
- package/static/v3/js/core/api.js +0 -625
- package/static/v3/js/core/components.f25b3b93.js +0 -230
- package/static/v3/js/core/components.js +0 -230
- package/static/v3/js/core/dom.a2773eb0.js +0 -148
- package/static/v3/js/core/dom.js +0 -148
- package/static/v3/js/core/i18n.880e1fec.js +0 -575
- package/static/v3/js/core/i18n.js +0 -575
- package/static/v3/js/core/router.584570f2.js +0 -37
- package/static/v3/js/core/router.js +0 -37
- package/static/v3/js/core/routes.37522821.js +0 -101
- package/static/v3/js/core/routes.js +0 -101
- package/static/v3/js/core/shell.e3f6bbfa.js +0 -420
- package/static/v3/js/core/shell.js +0 -420
- package/static/v3/js/core/store.7b2aa044.js +0 -123
- package/static/v3/js/core/store.js +0 -123
- package/static/v3/js/views/account.eff40715.js +0 -143
- package/static/v3/js/views/account.js +0 -143
- package/static/v3/js/views/activity.0d271ef9.js +0 -67
- package/static/v3/js/views/activity.js +0 -67
- package/static/v3/js/views/admin-audit.660a1fb1.js +0 -185
- package/static/v3/js/views/admin-audit.js +0 -185
- package/static/v3/js/views/admin-permissions.a7ae5f09.js +0 -177
- package/static/v3/js/views/admin-permissions.js +0 -177
- package/static/v3/js/views/admin-policies.3658fd86.js +0 -102
- package/static/v3/js/views/admin-policies.js +0 -102
- package/static/v3/js/views/admin-private-vpc.7d342d36.js +0 -135
- package/static/v3/js/views/admin-private-vpc.js +0 -135
- package/static/v3/js/views/admin-security.07c66b72.js +0 -180
- package/static/v3/js/views/admin-security.js +0 -180
- package/static/v3/js/views/admin-users.f7ac7b43.js +0 -166
- package/static/v3/js/views/admin-users.js +0 -166
- package/static/v3/js/views/agents.17c5288d.js +0 -564
- package/static/v3/js/views/agents.js +0 -564
- package/static/v3/js/views/chat.e250e2cc.js +0 -624
- package/static/v3/js/views/chat.js +0 -624
- package/static/v3/js/views/files.adad14c1.js +0 -365
- package/static/v3/js/views/files.js +0 -365
- package/static/v3/js/views/graph-canvas.17c15d65.js +0 -509
- package/static/v3/js/views/graph-canvas.js +0 -509
- package/static/v3/js/views/home.24f8b8ae.js +0 -200
- package/static/v3/js/views/home.js +0 -200
- package/static/v3/js/views/hooks.37895880.js +0 -220
- package/static/v3/js/views/hooks.js +0 -220
- package/static/v3/js/views/hybrid-search.2fb63ed9.js +0 -194
- package/static/v3/js/views/hybrid-search.js +0 -194
- package/static/v3/js/views/knowledge-graph.4d09c537.js +0 -529
- package/static/v3/js/views/knowledge-graph.js +0 -529
- package/static/v3/js/views/marketplace.ab0583d4.js +0 -141
- package/static/v3/js/views/marketplace.js +0 -141
- package/static/v3/js/views/mcp.99b5c6a7.js +0 -114
- package/static/v3/js/views/mcp.js +0 -114
- package/static/v3/js/views/memory.4ebdf474.js +0 -147
- package/static/v3/js/views/memory.js +0 -147
- package/static/v3/js/views/models.a1ffa147.js +0 -256
- package/static/v3/js/views/models.js +0 -256
- package/static/v3/js/views/my-computer.d9d9ae1c.js +0 -463
- package/static/v3/js/views/my-computer.js +0 -463
- package/static/v3/js/views/network.52a4f181.js +0 -97
- package/static/v3/js/views/network.js +0 -97
- package/static/v3/js/views/pipeline.c522f1ce.js +0 -157
- package/static/v3/js/views/pipeline.js +0 -157
- package/static/v3/js/views/planning.4876fd77.js +0 -174
- package/static/v3/js/views/planning.js +0 -174
- package/static/v3/js/views/runs.b63b2afa.js +0 -144
- package/static/v3/js/views/runs.js +0 -144
- package/static/v3/js/views/settings.b7140634.js +0 -317
- package/static/v3/js/views/settings.js +0 -317
- package/static/v3/js/views/skills.c6c2f965.js +0 -109
- package/static/v3/js/views/skills.js +0 -109
- package/static/v3/js/views/snapshots.6f5db095.js +0 -135
- package/static/v3/js/views/snapshots.js +0 -135
- package/static/v3/js/views/tools.e4f11276.js +0 -108
- package/static/v3/js/views/tools.js +0 -108
- package/static/v3/js/views/workflows.7752225a.js +0 -213
- package/static/v3/js/views/workflows.js +0 -213
- package/static/v3/js/views/workspace-admin.c466029b.js +0 -156
- package/static/v3/js/views/workspace-admin.js +0 -156
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
/*! tailwindcss v4.3.0 | MIT License | https://tailwindcss.com */
|
|
2
|
+
@layer properties{@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,:before,:after,::backdrop{--tw-scale-x:1;--tw-scale-y:1;--tw-scale-z:1;--tw-rotate-x:initial;--tw-rotate-y:initial;--tw-rotate-z:initial;--tw-skew-x:initial;--tw-skew-y:initial;--tw-pan-x:initial;--tw-pan-y:initial;--tw-pinch-zoom:initial;--tw-space-y-reverse:0;--tw-space-x-reverse:0;--tw-divide-x-reverse:0;--tw-border-style:solid;--tw-divide-y-reverse:0;--tw-leading:initial;--tw-font-weight:initial;--tw-tracking:initial;--tw-ordinal:initial;--tw-slashed-zero:initial;--tw-numeric-figure:initial;--tw-numeric-spacing:initial;--tw-numeric-fraction:initial;--tw-shadow:0 0 #0000;--tw-shadow-color:initial;--tw-shadow-alpha:100%;--tw-inset-shadow:0 0 #0000;--tw-inset-shadow-color:initial;--tw-inset-shadow-alpha:100%;--tw-ring-color:initial;--tw-ring-shadow:0 0 #0000;--tw-inset-ring-color:initial;--tw-inset-ring-shadow:0 0 #0000;--tw-ring-inset:initial;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-offset-shadow:0 0 #0000;--tw-outline-style:solid;--tw-blur:initial;--tw-brightness:initial;--tw-contrast:initial;--tw-grayscale:initial;--tw-hue-rotate:initial;--tw-invert:initial;--tw-opacity:initial;--tw-saturate:initial;--tw-sepia:initial;--tw-drop-shadow:initial;--tw-drop-shadow-color:initial;--tw-drop-shadow-alpha:100%;--tw-drop-shadow-size:initial;--tw-backdrop-blur:initial;--tw-backdrop-brightness:initial;--tw-backdrop-contrast:initial;--tw-backdrop-grayscale:initial;--tw-backdrop-hue-rotate:initial;--tw-backdrop-invert:initial;--tw-backdrop-opacity:initial;--tw-backdrop-saturate:initial;--tw-backdrop-sepia:initial;--tw-ease:initial}}}@layer theme{:root,:host{--font-sans:ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";--font-mono:ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;--color-amber-300:oklch(87.9% .169 91.605);--color-amber-500:oklch(76.9% .188 70.08);--color-emerald-300:oklch(84.5% .143 164.978);--color-emerald-500:oklch(69.6% .17 162.48);--spacing:.25rem;--container-xl:36rem;--container-3xl:48rem;--text-xs:.75rem;--text-xs--line-height:calc(1 / .75);--text-sm:.875rem;--text-sm--line-height:calc(1.25 / .875);--text-xl:1.25rem;--text-xl--line-height:calc(1.75 / 1.25);--text-2xl:1.5rem;--text-2xl--line-height:calc(2 / 1.5);--text-3xl:1.875rem;--text-3xl--line-height:calc(2.25 / 1.875);--font-weight-medium:500;--font-weight-semibold:600;--font-weight-bold:700;--tracking-normal:0em;--tracking-wide:.025em;--leading-relaxed:1.625;--radius-md:.375rem;--radius-lg:.5rem;--ease-in:cubic-bezier(.4, 0, 1, 1);--ease-out:cubic-bezier(0, 0, .2, 1);--ease-in-out:cubic-bezier(.4, 0, .2, 1);--animate-spin:spin 1s linear infinite;--blur-sm:8px;--default-transition-duration:.15s;--default-transition-timing-function:cubic-bezier(.4, 0, .2, 1);--default-font-family:var(--font-sans);--default-mono-font-family:var(--font-mono)}}@layer base{*,:after,:before,::backdrop{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}html,:host{-webkit-text-size-adjust:100%;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:var(--default-mono-font-family,ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}ol,ul,menu{list-style:none}img,svg,video,canvas,audio,iframe,embed,object{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,select,optgroup,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::placeholder{opacity:1}@supports (not ((-webkit-appearance:-apple-pay-button))) or (contain-intrinsic-size:1px){::placeholder{color:currentColor}@supports (color:color-mix(in lab, red, red)){::placeholder{color:color-mix(in oklab, currentcolor 50%, transparent)}}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit{padding-block:0}::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-day-field{padding-block:0}::-webkit-datetime-edit-hour-field{padding-block:0}::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-millisecond-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}::-webkit-calendar-picker-indicator{line-height:1}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){appearance:button}::file-selector-button{appearance:button}::-webkit-inner-spin-button{height:auto}::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}}@layer components;@layer utilities{.\@container{container-type:inline-size}.collapse{visibility:collapse}.invisible{visibility:hidden}.visible{visibility:visible}.sr-only{clip-path:inset(50%);white-space:nowrap;border-width:0;width:1px;height:1px;margin:-1px;padding:0;position:absolute;overflow:hidden}.not-sr-only{clip-path:none;white-space:normal;width:auto;height:auto;margin:0;padding:0;position:static;overflow:visible}.\!static{position:static!important}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.static{position:static}.sticky{position:sticky}.-inset-1{inset:calc(var(--spacing) * -1)}.inset-0{inset:calc(var(--spacing) * 0)}.top-0{top:calc(var(--spacing) * 0)}.isolate{isolation:isolate}.isolation-auto{isolation:auto}.z-8{z-index:8}.z-30{z-index:30}.z-40{z-index:40}.z-50{z-index:50}.\!container{width:100%!important}@media (width>=40rem){.\!container{max-width:40rem!important}}@media (width>=48rem){.\!container{max-width:48rem!important}}@media (width>=64rem){.\!container{max-width:64rem!important}}@media (width>=80rem){.\!container{max-width:80rem!important}}@media (width>=96rem){.\!container{max-width:96rem!important}}.container{width:100%}@media (width>=40rem){.container{max-width:40rem}}@media (width>=48rem){.container{max-width:48rem}}@media (width>=64rem){.container{max-width:64rem}}@media (width>=80rem){.container{max-width:80rem}}@media (width>=96rem){.container{max-width:96rem}}.m-2{margin:calc(var(--spacing) * 2)}.mx-auto{margin-inline:auto}.mt-1{margin-top:calc(var(--spacing) * 1)}.mt-2{margin-top:calc(var(--spacing) * 2)}.mt-16{margin-top:calc(var(--spacing) * 16)}.mb-1{margin-bottom:calc(var(--spacing) * 1)}.mb-2{margin-bottom:calc(var(--spacing) * 2)}.block{display:block}.contents{display:contents}.flex{display:flex}.flow-root{display:flow-root}.grid{display:grid}.hidden{display:none}.inline{display:inline}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.inline-grid{display:inline-grid}.inline-table{display:inline-table}.list-item{display:list-item}.table{display:table}.table-caption{display:table-caption}.table-cell{display:table-cell}.table-column{display:table-column}.table-column-group{display:table-column-group}.table-footer-group{display:table-footer-group}.table-header-group{display:table-header-group}.table-row{display:table-row}.table-row-group{display:table-row-group}.h-0{height:calc(var(--spacing) * 0)}.h-3\.5{height:calc(var(--spacing) * 3.5)}.h-4{height:calc(var(--spacing) * 4)}.h-5{height:calc(var(--spacing) * 5)}.h-6{height:calc(var(--spacing) * 6)}.h-7{height:calc(var(--spacing) * 7)}.h-8{height:calc(var(--spacing) * 8)}.h-9{height:calc(var(--spacing) * 9)}.h-16{height:calc(var(--spacing) * 16)}.h-\[440px\]{height:440px}.h-\[520px\]{height:520px}.h-full{height:100%}.max-h-80{max-height:calc(var(--spacing) * 80)}.max-h-96{max-height:calc(var(--spacing) * 96)}.min-h-6{min-height:calc(var(--spacing) * 6)}.min-h-14{min-height:calc(var(--spacing) * 14)}.min-h-24{min-height:calc(var(--spacing) * 24)}.min-h-28{min-height:calc(var(--spacing) * 28)}.min-h-44{min-height:calc(var(--spacing) * 44)}.min-h-\[40rem\]{min-height:40rem}.min-h-\[calc\(100vh-7rem\)\]{min-height:calc(100vh - 7rem)}.min-h-screen{min-height:100vh}.w-3\.5{width:calc(var(--spacing) * 3.5)}.w-4{width:calc(var(--spacing) * 4)}.w-5{width:calc(var(--spacing) * 5)}.w-7{width:calc(var(--spacing) * 7)}.w-9{width:calc(var(--spacing) * 9)}.w-38{width:calc(var(--spacing) * 38)}.w-64{width:calc(var(--spacing) * 64)}.w-full{width:100%}.max-w-3xl{max-width:var(--container-3xl)}.max-w-\[78\%\]{max-width:78%}.max-w-xl{max-width:var(--container-xl)}.min-w-0{min-width:calc(var(--spacing) * 0)}.flex-1{flex:1}.shrink{flex-shrink:1}.shrink-0{flex-shrink:0}.grow{flex-grow:1}.border-collapse{border-collapse:collapse}.translate-none{translate:none}.scale-3d{scale:var(--tw-scale-x) var(--tw-scale-y) var(--tw-scale-z)}.transform{transform:var(--tw-rotate-x,) var(--tw-rotate-y,) var(--tw-rotate-z,) var(--tw-skew-x,) var(--tw-skew-y,)}.transform\!{transform:var(--tw-rotate-x,) var(--tw-rotate-y,) var(--tw-rotate-z,) var(--tw-skew-x,) var(--tw-skew-y,)!important}.animate-spin{animation:var(--animate-spin)}.cursor-pointer{cursor:pointer}.touch-pinch-zoom{--tw-pinch-zoom:pinch-zoom;touch-action:var(--tw-pan-x,) var(--tw-pan-y,) var(--tw-pinch-zoom,)}.resize{resize:both}.resize-y{resize:vertical}.scrollbar-thin{scrollbar-width:thin}.grid-cols-\[minmax\(9rem\,0\.5fr\)_1fr\]{grid-template-columns:minmax(9rem,.5fr) 1fr}.flex-col{flex-direction:column}.flex-row{flex-direction:row}.flex-wrap{flex-wrap:wrap}.place-items-center{place-items:center}.items-center{align-items:center}.items-start{align-items:flex-start}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.justify-end{justify-content:flex-end}.justify-start{justify-content:flex-start}.gap-1{gap:calc(var(--spacing) * 1)}.gap-1\.5{gap:calc(var(--spacing) * 1.5)}.gap-2{gap:calc(var(--spacing) * 2)}.gap-3{gap:calc(var(--spacing) * 3)}.gap-4{gap:calc(var(--spacing) * 4)}:where(.space-y-1>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 1) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 1) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-2>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 2) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 2) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-3>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 3) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 3) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-4>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 4) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 4) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-reverse>:not(:last-child)){--tw-space-y-reverse:1}:where(.space-x-reverse>:not(:last-child)){--tw-space-x-reverse:1}:where(.divide-x>:not(:last-child)){--tw-divide-x-reverse:0;border-inline-style:var(--tw-border-style);border-inline-start-width:calc(1px * var(--tw-divide-x-reverse));border-inline-end-width:calc(1px * calc(1 - var(--tw-divide-x-reverse)))}:where(.divide-y>:not(:last-child)){--tw-divide-y-reverse:0;border-bottom-style:var(--tw-border-style);border-top-style:var(--tw-border-style);border-top-width:calc(1px * var(--tw-divide-y-reverse));border-bottom-width:calc(1px * calc(1 - var(--tw-divide-y-reverse)))}:where(.divide-y-reverse>:not(:last-child)){--tw-divide-y-reverse:1}:where(.divide-border>:not(:last-child)){border-color:hsl(var(--border))}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-auto{overflow:auto}.overflow-hidden{overflow:hidden}.rounded{border-radius:.25rem}.rounded-lg{border-radius:var(--radius-lg)}.rounded-md{border-radius:var(--radius-md)}.rounded-s{border-start-start-radius:.25rem;border-end-start-radius:.25rem}.rounded-ss{border-start-start-radius:.25rem}.rounded-e{border-start-end-radius:.25rem;border-end-end-radius:.25rem}.rounded-se{border-start-end-radius:.25rem}.rounded-ee{border-end-end-radius:.25rem}.rounded-es{border-end-start-radius:.25rem}.rounded-t{border-top-left-radius:.25rem;border-top-right-radius:.25rem}.rounded-l{border-top-left-radius:.25rem;border-bottom-left-radius:.25rem}.rounded-tl{border-top-left-radius:.25rem}.rounded-r{border-top-right-radius:.25rem;border-bottom-right-radius:.25rem}.rounded-tr{border-top-right-radius:.25rem}.rounded-b{border-bottom-right-radius:.25rem;border-bottom-left-radius:.25rem}.rounded-br{border-bottom-right-radius:.25rem}.rounded-bl{border-bottom-left-radius:.25rem}.border{border-style:var(--tw-border-style);border-width:1px}.border-x{border-inline-style:var(--tw-border-style);border-inline-width:1px}.border-y{border-block-style:var(--tw-border-style);border-block-width:1px}.border-s{border-inline-start-style:var(--tw-border-style);border-inline-start-width:1px}.border-e{border-inline-end-style:var(--tw-border-style);border-inline-end-width:1px}.border-bs{border-block-start-style:var(--tw-border-style);border-block-start-width:1px}.border-be{border-block-end-style:var(--tw-border-style);border-block-end-width:1px}.border-t{border-top-style:var(--tw-border-style);border-top-width:1px}.border-r{border-right-style:var(--tw-border-style);border-right-width:1px}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-l{border-left-style:var(--tw-border-style);border-left-width:1px}.border-dashed{--tw-border-style:dashed;border-style:dashed}.border-amber-500\/25{border-color:#f99c0040}@supports (color:color-mix(in lab, red, red)){.border-amber-500\/25{border-color:color-mix(in oklab, var(--color-amber-500) 25%, transparent)}}.border-border{border-color:hsl(var(--border))}.border-destructive\/30{border-color:hsl(var(--destructive))}@supports (color:color-mix(in lab, red, red)){.border-destructive\/30{border-color:color-mix(in oklab, hsl(var(--destructive)) 30%, transparent)}}.border-emerald-500\/25{border-color:#00bb7f40}@supports (color:color-mix(in lab, red, red)){.border-emerald-500\/25{border-color:color-mix(in oklab, var(--color-emerald-500) 25%, transparent)}}.border-input{border-color:hsl(var(--input))}.border-primary\/25{border-color:hsl(var(--primary))}@supports (color:color-mix(in lab, red, red)){.border-primary\/25{border-color:color-mix(in oklab, hsl(var(--primary)) 25%, transparent)}}.border-primary\/30{border-color:hsl(var(--primary))}@supports (color:color-mix(in lab, red, red)){.border-primary\/30{border-color:color-mix(in oklab, hsl(var(--primary)) 30%, transparent)}}.bg-amber-500\/12{background-color:#f99c001f}@supports (color:color-mix(in lab, red, red)){.bg-amber-500\/12{background-color:color-mix(in oklab, var(--color-amber-500) 12%, transparent)}}.bg-background,.bg-background\/70{background-color:hsl(var(--background))}@supports (color:color-mix(in lab, red, red)){.bg-background\/70{background-color:color-mix(in oklab, hsl(var(--background)) 70%, transparent)}}.bg-background\/80{background-color:hsl(var(--background))}@supports (color:color-mix(in lab, red, red)){.bg-background\/80{background-color:color-mix(in oklab, hsl(var(--background)) 80%, transparent)}}.bg-background\/95{background-color:hsl(var(--background))}@supports (color:color-mix(in lab, red, red)){.bg-background\/95{background-color:color-mix(in oklab, hsl(var(--background)) 95%, transparent)}}.bg-card{background-color:hsl(var(--card))}.bg-destructive,.bg-destructive\/12{background-color:hsl(var(--destructive))}@supports (color:color-mix(in lab, red, red)){.bg-destructive\/12{background-color:color-mix(in oklab, hsl(var(--destructive)) 12%, transparent)}}.bg-emerald-500\/12{background-color:#00bb7f1f}@supports (color:color-mix(in lab, red, red)){.bg-emerald-500\/12{background-color:color-mix(in oklab, var(--color-emerald-500) 12%, transparent)}}.bg-muted,.bg-muted\/30{background-color:hsl(var(--muted))}@supports (color:color-mix(in lab, red, red)){.bg-muted\/30{background-color:color-mix(in oklab, hsl(var(--muted)) 30%, transparent)}}.bg-muted\/40{background-color:hsl(var(--muted))}@supports (color:color-mix(in lab, red, red)){.bg-muted\/40{background-color:color-mix(in oklab, hsl(var(--muted)) 40%, transparent)}}.bg-primary,.bg-primary\/12{background-color:hsl(var(--primary))}@supports (color:color-mix(in lab, red, red)){.bg-primary\/12{background-color:color-mix(in oklab, hsl(var(--primary)) 12%, transparent)}}.bg-primary\/14{background-color:hsl(var(--primary))}@supports (color:color-mix(in lab, red, red)){.bg-primary\/14{background-color:color-mix(in oklab, hsl(var(--primary)) 14%, transparent)}}.bg-primary\/15{background-color:hsl(var(--primary))}@supports (color:color-mix(in lab, red, red)){.bg-primary\/15{background-color:color-mix(in oklab, hsl(var(--primary)) 15%, transparent)}}.bg-secondary{background-color:hsl(var(--secondary))}.bg-repeat{background-repeat:repeat}.mask-no-clip{-webkit-mask-clip:no-clip;mask-clip:no-clip}.mask-repeat{-webkit-mask-repeat:repeat;mask-repeat:repeat}.p-1{padding:calc(var(--spacing) * 1)}.p-2{padding:calc(var(--spacing) * 2)}.p-3{padding:calc(var(--spacing) * 3)}.p-4{padding:calc(var(--spacing) * 4)}.p-5{padding:calc(var(--spacing) * 5)}.p-6{padding:calc(var(--spacing) * 6)}.px-0{padding-inline:calc(var(--spacing) * 0)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-2\.5{padding-inline:calc(var(--spacing) * 2.5)}.px-3{padding-inline:calc(var(--spacing) * 3)}.px-4{padding-inline:calc(var(--spacing) * 4)}.py-0\.5{padding-block:calc(var(--spacing) * .5)}.py-2{padding-block:calc(var(--spacing) * 2)}.pt-0{padding-top:calc(var(--spacing) * 0)}.text-center{text-align:center}.text-left{text-align:left}.text-2xl{font-size:var(--text-2xl);line-height:var(--tw-leading,var(--text-2xl--line-height))}.text-3xl{font-size:var(--text-3xl);line-height:var(--tw-leading,var(--text-3xl--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.font-bold{--tw-font-weight:var(--font-weight-bold);font-weight:var(--font-weight-bold)}.font-medium{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-normal{--tw-tracking:var(--tracking-normal);letter-spacing:var(--tracking-normal)}.tracking-wide{--tw-tracking:var(--tracking-wide);letter-spacing:var(--tracking-wide)}.text-wrap{text-wrap:wrap}.break-words{overflow-wrap:break-word}.text-clip{text-overflow:clip}.text-ellipsis{text-overflow:ellipsis}.whitespace-pre-wrap{white-space:pre-wrap}.text-amber-300{color:var(--color-amber-300)}.text-card-foreground{color:hsl(var(--card-foreground))}.text-destructive{color:hsl(var(--destructive))}.text-destructive-foreground{color:hsl(var(--destructive-foreground))}.text-emerald-300{color:var(--color-emerald-300)}.text-foreground{color:hsl(var(--foreground))}.text-muted-foreground{color:hsl(var(--muted-foreground))}.text-primary{color:hsl(var(--primary))}.text-primary-foreground{color:hsl(var(--primary-foreground))}.text-secondary-foreground{color:hsl(var(--secondary-foreground))}.capitalize{text-transform:capitalize}.lowercase{text-transform:lowercase}.normal-case{text-transform:none}.uppercase{text-transform:uppercase}.italic{font-style:italic}.not-italic{font-style:normal}.diagonal-fractions{--tw-numeric-fraction:diagonal-fractions;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.lining-nums{--tw-numeric-figure:lining-nums;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.oldstyle-nums{--tw-numeric-figure:oldstyle-nums;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.ordinal{--tw-ordinal:ordinal;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.proportional-nums{--tw-numeric-spacing:proportional-nums;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.slashed-zero{--tw-slashed-zero:slashed-zero;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.stacked-fractions{--tw-numeric-fraction:stacked-fractions;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.tabular-nums{--tw-numeric-spacing:tabular-nums;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.normal-nums{font-variant-numeric:normal}.line-through{text-decoration-line:line-through}.no-underline{text-decoration-line:none}.overline{text-decoration-line:overline}.underline{text-decoration-line:underline}.antialiased{-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.subpixel-antialiased{-webkit-font-smoothing:auto;-moz-osx-font-smoothing:auto}.shadow,.shadow-sm{--tw-shadow:0 1px 3px 0 var(--tw-shadow-color,#0000001a), 0 1px 2px -1px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-xl{--tw-shadow:0 20px 25px -5px var(--tw-shadow-color,#0000001a), 0 8px 10px -6px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.ring{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.inset-ring{--tw-inset-ring-shadow:inset 0 0 0 1px var(--tw-inset-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.outline{outline-style:var(--tw-outline-style);outline-width:1px}.blur{--tw-blur:blur(8px);filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.blur\!{--tw-blur:blur(8px)!important;filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)!important}.drop-shadow{--tw-drop-shadow-size:drop-shadow(0 1px 2px var(--tw-drop-shadow-color,#0000001a)) drop-shadow(0 1px 1px var(--tw-drop-shadow-color,#0000000f));--tw-drop-shadow:drop-shadow(0 1px 2px #0000001a) drop-shadow(0 1px 1px #0000000f);filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.grayscale{--tw-grayscale:grayscale(100%);filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.invert{--tw-invert:invert(100%);filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.sepia{--tw-sepia:sepia(100%);filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.filter{filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.backdrop-blur{--tw-backdrop-blur:blur(8px);-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.backdrop-blur-sm{--tw-backdrop-blur:blur(var(--blur-sm));-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.backdrop-grayscale{--tw-backdrop-grayscale:grayscale(100%);-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.backdrop-invert{--tw-backdrop-invert:invert(100%);-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.backdrop-sepia{--tw-backdrop-sepia:sepia(100%);-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.backdrop-filter{-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.transition{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to,opacity,box-shadow,transform,translate,scale,rotate,filter,-webkit-backdrop-filter,backdrop-filter,display,content-visibility,overlay,pointer-events;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.ease-in{--tw-ease:var(--ease-in);transition-timing-function:var(--ease-in)}.ease-in-out{--tw-ease:var(--ease-in-out);transition-timing-function:var(--ease-in-out)}.ease-out{--tw-ease:var(--ease-out);transition-timing-function:var(--ease-out)}.outline-none{--tw-outline-style:none;outline-style:none}:where(.divide-x-reverse>:not(:last-child)){--tw-divide-x-reverse:1}.ring-inset{--tw-ring-inset:inset}.placeholder\:text-muted-foreground::placeholder{color:hsl(var(--muted-foreground))}@media (hover:hover){.hover\:bg-destructive\/90:hover{background-color:hsl(var(--destructive))}@supports (color:color-mix(in lab, red, red)){.hover\:bg-destructive\/90:hover{background-color:color-mix(in oklab, hsl(var(--destructive)) 90%, transparent)}}.hover\:bg-muted:hover{background-color:hsl(var(--muted))}.hover\:bg-primary\/90:hover{background-color:hsl(var(--primary))}@supports (color:color-mix(in lab, red, red)){.hover\:bg-primary\/90:hover{background-color:color-mix(in oklab, hsl(var(--primary)) 90%, transparent)}}.hover\:bg-secondary\/80:hover{background-color:hsl(var(--secondary))}@supports (color:color-mix(in lab, red, red)){.hover\:bg-secondary\/80:hover{background-color:color-mix(in oklab, hsl(var(--secondary)) 80%, transparent)}}.hover\:text-foreground:hover{color:hsl(var(--foreground))}}.focus\:ring-2:focus{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.focus\:ring-ring:focus{--tw-ring-color:hsl(var(--ring))}.focus-visible\:ring-2:focus-visible{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.focus-visible\:ring-ring:focus-visible{--tw-ring-color:hsl(var(--ring))}.focus-visible\:outline-none:focus-visible{--tw-outline-style:none;outline-style:none}.disabled\:pointer-events-none:disabled{pointer-events:none}.disabled\:cursor-not-allowed:disabled{cursor:not-allowed}.disabled\:opacity-45:disabled{opacity:.45}.disabled\:opacity-50:disabled{opacity:.5}@media (width>=40rem){.sm\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.sm\:flex-row{flex-direction:row}}@media (width>=64rem){.lg\:fixed{position:fixed}.lg\:inset-y-0{inset-block:calc(var(--spacing) * 0)}.lg\:left-0{left:calc(var(--spacing) * 0)}.lg\:block{display:block}.lg\:hidden{display:none}.lg\:p-6{padding:calc(var(--spacing) * 6)}.lg\:pl-64{padding-left:calc(var(--spacing) * 64)}}@media (width>=80rem){.xl\:col-span-2{grid-column:span 2/span 2}.xl\:col-span-3{grid-column:span 3/span 3}.xl\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.xl\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.xl\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.xl\:grid-cols-\[0\.8fr_1\.2fr\]{grid-template-columns:.8fr 1.2fr}.xl\:grid-cols-\[0\.9fr_1\.1fr\]{grid-template-columns:.9fr 1.1fr}.xl\:grid-cols-\[0\.75fr_1\.25fr\]{grid-template-columns:.75fr 1.25fr}.xl\:grid-cols-\[1\.1fr_0\.9fr\]{grid-template-columns:1.1fr .9fr}.xl\:grid-cols-\[1\.2fr_0\.8fr\]{grid-template-columns:1.2fr .8fr}.xl\:grid-cols-\[1\.4fr_0\.6fr\]{grid-template-columns:1.4fr .6fr}.xl\:grid-cols-\[1fr_1fr\]{grid-template-columns:1fr 1fr}.xl\:grid-cols-\[18rem_minmax\(0\,1fr\)_22rem\]{grid-template-columns:18rem minmax(0,1fr) 22rem}}}.react-flow{direction:ltr}.react-flow__container{width:100%;height:100%;position:absolute;top:0;left:0}.react-flow__pane{z-index:1;cursor:-webkit-grab;cursor:grab}.react-flow__pane.selection{cursor:pointer}.react-flow__pane.dragging{cursor:-webkit-grabbing;cursor:grabbing}.react-flow__viewport{transform-origin:0 0;z-index:2;pointer-events:none}.react-flow__renderer{z-index:4}.react-flow__selection{z-index:6}.react-flow__nodesselection-rect:focus,.react-flow__nodesselection-rect:focus-visible{outline:none}.react-flow .react-flow__edges{pointer-events:none;overflow:visible}.react-flow__edge-path,.react-flow__connection-path{stroke:#b1b1b7;stroke-width:1px;fill:none}.react-flow__edge{pointer-events:visibleStroke;cursor:pointer}.react-flow__edge.animated path{stroke-dasharray:5;animation:.5s linear infinite dashdraw}.react-flow__edge.animated path.react-flow__edge-interaction{stroke-dasharray:none;animation:none}.react-flow__edge.inactive{pointer-events:none}.react-flow__edge.selected,.react-flow__edge:focus,.react-flow__edge:focus-visible{outline:none}.react-flow__edge.selected .react-flow__edge-path,.react-flow__edge:focus .react-flow__edge-path,.react-flow__edge:focus-visible .react-flow__edge-path{stroke:#555}.react-flow__edge-textwrapper{pointer-events:all}.react-flow__edge-textbg{fill:#fff}.react-flow__edge .react-flow__edge-text{pointer-events:none;-webkit-user-select:none;user-select:none}.react-flow__connection{pointer-events:none}.react-flow__connection .animated{stroke-dasharray:5;animation:.5s linear infinite dashdraw}.react-flow__connectionline{z-index:1001}.react-flow__nodes{pointer-events:none;transform-origin:0 0}.react-flow__node{-webkit-user-select:none;user-select:none;pointer-events:all;transform-origin:0 0;box-sizing:border-box;cursor:-webkit-grab;cursor:grab;position:absolute}.react-flow__node.dragging{cursor:-webkit-grabbing;cursor:grabbing}.react-flow__nodesselection{z-index:3;transform-origin:0 0;pointer-events:none}.react-flow__nodesselection-rect{pointer-events:all;cursor:-webkit-grab;cursor:grab;position:absolute}.react-flow__handle{pointer-events:none;background:#1a192b;border:1px solid #fff;border-radius:100%;width:6px;min-width:5px;height:6px;min-height:5px;position:absolute}.react-flow__handle.connectionindicator{pointer-events:all;cursor:crosshair}.react-flow__handle-bottom{top:auto;bottom:-4px;left:50%;transform:translate(-50%)}.react-flow__handle-top{top:-4px;left:50%;transform:translate(-50%)}.react-flow__handle-left{top:50%;left:-4px;transform:translateY(-50%)}.react-flow__handle-right{top:50%;right:-4px;transform:translateY(-50%)}.react-flow__edgeupdater{cursor:move;pointer-events:all}.react-flow__panel{z-index:5;margin:15px;position:absolute}.react-flow__panel.top{top:0}.react-flow__panel.bottom{bottom:0}.react-flow__panel.left{left:0}.react-flow__panel.right{right:0}.react-flow__panel.center{left:50%;transform:translate(-50%)}.react-flow__attribution{background:#ffffff80;margin:0;padding:2px 3px;font-size:10px}.react-flow__attribution a{color:#999;text-decoration:none}@keyframes dashdraw{0%{stroke-dashoffset:10px}}.react-flow__edgelabel-renderer{pointer-events:none;-webkit-user-select:none;user-select:none;width:100%;height:100%;position:absolute}.react-flow__edge.updating .react-flow__edge-path{stroke:#777}.react-flow__edge-text{font-size:10px}.react-flow__node.selectable:focus,.react-flow__node.selectable:focus-visible{outline:none}.react-flow__node-default,.react-flow__node-input,.react-flow__node-output,.react-flow__node-group{color:#222;text-align:center;background-color:#fff;border:1px solid #1a192b;border-radius:3px;width:150px;padding:10px;font-size:12px}.react-flow__node-default.selectable:hover,.react-flow__node-input.selectable:hover,.react-flow__node-output.selectable:hover,.react-flow__node-group.selectable:hover{box-shadow:0 1px 4px 1px #00000014}.react-flow__node-default.selectable.selected,.react-flow__node-default.selectable:focus,.react-flow__node-default.selectable:focus-visible,.react-flow__node-input.selectable.selected,.react-flow__node-input.selectable:focus,.react-flow__node-input.selectable:focus-visible,.react-flow__node-output.selectable.selected,.react-flow__node-output.selectable:focus,.react-flow__node-output.selectable:focus-visible,.react-flow__node-group.selectable.selected,.react-flow__node-group.selectable:focus,.react-flow__node-group.selectable:focus-visible{box-shadow:0 0 0 .5px #1a192b}.react-flow__node-group{background-color:#f0f0f040}.react-flow__nodesselection-rect,.react-flow__selection{background:#0059dc14;border:1px dotted #0059dccc}.react-flow__nodesselection-rect:focus,.react-flow__nodesselection-rect:focus-visible,.react-flow__selection:focus,.react-flow__selection:focus-visible{outline:none}.react-flow__controls{box-shadow:0 0 2px 1px #00000014}.react-flow__controls-button{box-sizing:content-box;cursor:pointer;-webkit-user-select:none;user-select:none;background:#fefefe;border:none;border-bottom:1px solid #eee;justify-content:center;align-items:center;width:16px;height:16px;padding:5px;display:flex}.react-flow__controls-button:hover{background:#f4f4f4}.react-flow__controls-button svg{width:100%;max-width:12px;max-height:12px}.react-flow__controls-button:disabled{pointer-events:none}.react-flow__controls-button:disabled svg{fill-opacity:.4}.react-flow__minimap{background-color:#fff}.react-flow__minimap svg{display:block}.react-flow__resize-control{position:absolute}.react-flow__resize-control.left,.react-flow__resize-control.right{cursor:ew-resize}.react-flow__resize-control.top,.react-flow__resize-control.bottom{cursor:ns-resize}.react-flow__resize-control.top.left,.react-flow__resize-control.bottom.right{cursor:nwse-resize}.react-flow__resize-control.bottom.left,.react-flow__resize-control.top.right{cursor:nesw-resize}.react-flow__resize-control.handle{background-color:#3367d9;border:1px solid #fff;border-radius:1px;width:4px;height:4px;transform:translate(-50%,-50%)}.react-flow__resize-control.handle.left{top:50%;left:0}.react-flow__resize-control.handle.right{top:50%;left:100%}.react-flow__resize-control.handle.top{top:0;left:50%}.react-flow__resize-control.handle.bottom{top:100%;left:50%}.react-flow__resize-control.handle.top.left,.react-flow__resize-control.handle.bottom.left{left:0}.react-flow__resize-control.handle.top.right,.react-flow__resize-control.handle.bottom.right{left:100%}.react-flow__resize-control.line{border:0 solid #3367d9}.react-flow__resize-control.line.left,.react-flow__resize-control.line.right{width:1px;height:100%;top:0;transform:translate(-50%)}.react-flow__resize-control.line.left{border-left-width:1px;left:0}.react-flow__resize-control.line.right{border-right-width:1px;left:100%}.react-flow__resize-control.line.top,.react-flow__resize-control.line.bottom{width:100%;height:1px;left:0;transform:translateY(-50%)}.react-flow__resize-control.line.top{border-top-width:1px;top:0}.react-flow__resize-control.line.bottom{border-bottom-width:1px;top:100%}:root{--lightningcss-light: ;--lightningcss-dark:initial;color-scheme:dark;--background:220 18% 8%;--foreground:210 30% 96%;--card:220 16% 11%;--card-foreground:210 30% 96%;--muted:218 13% 18%;--muted-foreground:215 14% 66%;--primary:176 68% 48%;--primary-foreground:212 26% 8%;--secondary:221 14% 20%;--secondary-foreground:210 30% 96%;--destructive:355 78% 58%;--destructive-foreground:0 0% 100%;--border:218 13% 23%;--input:218 13% 24%;--ring:176 68% 48%;--brain:176 68% 48%;--ask:216 78% 66%;--capture:142 58% 56%;--act:33 92% 58%;--library:262 68% 70%;--system:350 72% 68%}:root[data-theme=light]{--lightningcss-light:initial;--lightningcss-dark: ;color-scheme:light;--background:210 30% 98%;--foreground:222 38% 9%;--card:0 0% 100%;--card-foreground:222 38% 9%;--muted:213 24% 92%;--muted-foreground:219 11% 38%;--primary:178 72% 32%;--primary-foreground:0 0% 100%;--secondary:211 24% 90%;--secondary-foreground:222 38% 9%;--destructive:355 72% 48%;--destructive-foreground:0 0% 100%;--border:214 20% 83%;--input:214 20% 78%;--ring:178 72% 32%}*{box-sizing:border-box}body{background:hsl(var(--background));min-width:320px;color:hsl(var(--foreground));margin:0;font-family:Inter,ui-sans-serif,system-ui,-apple-system,BlinkMacSystemFont,Segoe UI,sans-serif}button,input,textarea,select{font:inherit}.brain-grid{background-image:linear-gradient(hsl(var(--border) / .34) 1px, transparent 1px), linear-gradient(90deg, hsl(var(--border) / .34) 1px, transparent 1px);background-size:32px 32px}.scrollbar-thin{scrollbar-width:thin}.react-flow__node{border:1px solid hsl(var(--border));background:hsl(var(--card));color:hsl(var(--foreground));border-radius:8px;padding:8px 10px;font-size:12px}.react-flow__edge-path{stroke:hsl(var(--primary))}@property --tw-scale-x{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-y{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-z{syntax:"*";inherits:false;initial-value:1}@property --tw-rotate-x{syntax:"*";inherits:false}@property --tw-rotate-y{syntax:"*";inherits:false}@property --tw-rotate-z{syntax:"*";inherits:false}@property --tw-skew-x{syntax:"*";inherits:false}@property --tw-skew-y{syntax:"*";inherits:false}@property --tw-pan-x{syntax:"*";inherits:false}@property --tw-pan-y{syntax:"*";inherits:false}@property --tw-pinch-zoom{syntax:"*";inherits:false}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-space-x-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-divide-x-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-divide-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-leading{syntax:"*";inherits:false}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-ordinal{syntax:"*";inherits:false}@property --tw-slashed-zero{syntax:"*";inherits:false}@property --tw-numeric-figure{syntax:"*";inherits:false}@property --tw-numeric-spacing{syntax:"*";inherits:false}@property --tw-numeric-fraction{syntax:"*";inherits:false}@property --tw-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-shadow-color{syntax:"*";inherits:false}@property --tw-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-inset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-shadow-color{syntax:"*";inherits:false}@property --tw-inset-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-ring-color{syntax:"*";inherits:false}@property --tw-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-ring-color{syntax:"*";inherits:false}@property --tw-inset-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-ring-inset{syntax:"*";inherits:false}@property --tw-ring-offset-width{syntax:"<length>";inherits:false;initial-value:0}@property --tw-ring-offset-color{syntax:"*";inherits:false;initial-value:#fff}@property --tw-ring-offset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-outline-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-blur{syntax:"*";inherits:false}@property --tw-brightness{syntax:"*";inherits:false}@property --tw-contrast{syntax:"*";inherits:false}@property --tw-grayscale{syntax:"*";inherits:false}@property --tw-hue-rotate{syntax:"*";inherits:false}@property --tw-invert{syntax:"*";inherits:false}@property --tw-opacity{syntax:"*";inherits:false}@property --tw-saturate{syntax:"*";inherits:false}@property --tw-sepia{syntax:"*";inherits:false}@property --tw-drop-shadow{syntax:"*";inherits:false}@property --tw-drop-shadow-color{syntax:"*";inherits:false}@property --tw-drop-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-drop-shadow-size{syntax:"*";inherits:false}@property --tw-backdrop-blur{syntax:"*";inherits:false}@property --tw-backdrop-brightness{syntax:"*";inherits:false}@property --tw-backdrop-contrast{syntax:"*";inherits:false}@property --tw-backdrop-grayscale{syntax:"*";inherits:false}@property --tw-backdrop-hue-rotate{syntax:"*";inherits:false}@property --tw-backdrop-invert{syntax:"*";inherits:false}@property --tw-backdrop-opacity{syntax:"*";inherits:false}@property --tw-backdrop-saturate{syntax:"*";inherits:false}@property --tw-backdrop-sepia{syntax:"*";inherits:false}@property --tw-ease{syntax:"*";inherits:false}@keyframes spin{to{transform:rotate(360deg)}}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
<!doctype html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8" />
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0, viewport-fit=cover" />
|
|
6
|
+
<meta name="color-scheme" content="dark light" />
|
|
7
|
+
<title>Lattice AI · Digital Brain</title>
|
|
8
|
+
<link rel="manifest" href="/manifest.json" />
|
|
9
|
+
<link rel="icon" type="image/png" sizes="32x32" href="/icons/favicon-32.png" />
|
|
10
|
+
<script>
|
|
11
|
+
(function () {
|
|
12
|
+
try {
|
|
13
|
+
var theme = localStorage.getItem("lattice.theme");
|
|
14
|
+
if (theme === "light" || theme === "dark") document.documentElement.dataset.theme = theme;
|
|
15
|
+
} catch (e) {}
|
|
16
|
+
})();
|
|
17
|
+
</script>
|
|
18
|
+
<script type="module" crossorigin src="/static/app/assets/index-CJRAzNnf.js"></script>
|
|
19
|
+
<link rel="stylesheet" crossorigin href="/static/app/assets/index-CSwBBgf4.css">
|
|
20
|
+
</head>
|
|
21
|
+
<body>
|
|
22
|
+
<div id="root"></div>
|
|
23
|
+
<noscript>Lattice AI requires JavaScript for the local Digital Brain desktop shell.</noscript>
|
|
24
|
+
</body>
|
|
25
|
+
</html>
|
package/static/manifest.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "Lattice AI",
|
|
3
3
|
"short_name": "LatticeAI",
|
|
4
|
-
"description": "Local-first
|
|
4
|
+
"description": "Local-first Digital Brain desktop platform for knowledge graphs, durable memory, hybrid search, agents, and signed brain exchange",
|
|
5
5
|
"start_url": "/app",
|
|
6
6
|
"id": "/",
|
|
7
7
|
"display": "standalone",
|
|
@@ -28,7 +28,7 @@
|
|
|
28
28
|
{
|
|
29
29
|
"name": "새 대화",
|
|
30
30
|
"short_name": "대화",
|
|
31
|
-
"url": "/app#/
|
|
31
|
+
"url": "/app#/ask",
|
|
32
32
|
"icons": [{ "src": "/icons/icon-192.png", "sizes": "192x192" }]
|
|
33
33
|
}
|
|
34
34
|
]
|
package/static/sw.js
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
// Lattice Service Worker — PWA install + offline shell for the /app SPA.
|
|
2
|
-
// Strategy: precache the
|
|
2
|
+
// Strategy: precache the Vite app bundle from its asset manifest,
|
|
3
3
|
// cache-first for static assets, network-only for everything dynamic.
|
|
4
|
-
const CACHE = "lattice-
|
|
5
|
-
const MANIFEST_URL = "/static/
|
|
4
|
+
const CACHE = "lattice-v410";
|
|
5
|
+
const MANIFEST_URL = "/static/app/asset-manifest.json";
|
|
6
6
|
|
|
7
7
|
// Non-manifest assets the shell needs offline.
|
|
8
8
|
const SHELL = [
|
|
@@ -29,7 +29,7 @@ async function precache() {
|
|
|
29
29
|
const res = await fetch(MANIFEST_URL, { cache: "no-cache" });
|
|
30
30
|
const manifest = await res.json();
|
|
31
31
|
const entry = manifest.entrypoints || {};
|
|
32
|
-
manifestPaths = [entry.app, ...
|
|
32
|
+
manifestPaths = [entry.app, ...Object.values(manifest.assets || {})]
|
|
33
33
|
.filter(Boolean);
|
|
34
34
|
} catch (err) {
|
|
35
35
|
// Offline install: shell precache below still applies.
|
|
@@ -1,170 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
/*
|
|
3
|
-
* Build the v3 browser asset manifest.
|
|
4
|
-
*
|
|
5
|
-
* The source files stay importable in development. This script writes hashed
|
|
6
|
-
* siblings next to each runtime asset, rewrites ES-module imports to those
|
|
7
|
-
* hashed siblings, and emits static/v3/asset-manifest.json for /app.
|
|
8
|
-
*/
|
|
9
|
-
import { createHash } from "node:crypto";
|
|
10
|
-
import {
|
|
11
|
-
existsSync,
|
|
12
|
-
mkdirSync,
|
|
13
|
-
readdirSync,
|
|
14
|
-
readFileSync,
|
|
15
|
-
rmSync,
|
|
16
|
-
statSync,
|
|
17
|
-
writeFileSync,
|
|
18
|
-
} from "node:fs";
|
|
19
|
-
import { basename, dirname, extname, join, relative } from "node:path";
|
|
20
|
-
import { fileURLToPath } from "node:url";
|
|
21
|
-
|
|
22
|
-
const repoRoot = join(dirname(fileURLToPath(import.meta.url)), "..");
|
|
23
|
-
const staticRoot = join(repoRoot, "static");
|
|
24
|
-
const manifestPath = join(staticRoot, "v3", "asset-manifest.json");
|
|
25
|
-
|
|
26
|
-
// Version is sourced from package.json — the single source of truth for the
|
|
27
|
-
// release. Never hard-code a version string in the generated manifest.
|
|
28
|
-
const pkgVersion = JSON.parse(
|
|
29
|
-
readFileSync(join(repoRoot, "package.json"), "utf8"),
|
|
30
|
-
).version;
|
|
31
|
-
|
|
32
|
-
const cssSources = [
|
|
33
|
-
"static/css/tokens.css",
|
|
34
|
-
"static/v3/css/lattice.tokens.css",
|
|
35
|
-
"static/v3/css/lattice.base.css",
|
|
36
|
-
"static/v3/css/lattice.components.css",
|
|
37
|
-
"static/v3/css/lattice.shell.css",
|
|
38
|
-
"static/v3/css/lattice.views.css",
|
|
39
|
-
];
|
|
40
|
-
|
|
41
|
-
const moduleRoot = join(staticRoot, "v3", "js");
|
|
42
|
-
const entry = "static/v3/js/app.js";
|
|
43
|
-
|
|
44
|
-
function posix(p) {
|
|
45
|
-
return p.replaceAll("\\", "/");
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
function sha(text) {
|
|
49
|
-
return createHash("sha256").update(text).digest("hex").slice(0, 8);
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
function repoPath(abs) {
|
|
53
|
-
return posix(relative(repoRoot, abs));
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
function publicUrl(repoRel) {
|
|
57
|
-
return "/" + posix(repoRel);
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
function walk(dir) {
|
|
61
|
-
const out = [];
|
|
62
|
-
for (const name of readdirSync(dir)) {
|
|
63
|
-
const p = join(dir, name);
|
|
64
|
-
const st = statSync(p);
|
|
65
|
-
if (st.isDirectory()) out.push(...walk(p));
|
|
66
|
-
else if (name.endsWith(".js")) out.push(p);
|
|
67
|
-
}
|
|
68
|
-
return out;
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
function removeGenerated(dir, ext) {
|
|
72
|
-
if (!existsSync(dir)) return;
|
|
73
|
-
for (const name of readdirSync(dir)) {
|
|
74
|
-
const p = join(dir, name);
|
|
75
|
-
const st = statSync(p);
|
|
76
|
-
if (st.isDirectory()) removeGenerated(p, ext);
|
|
77
|
-
else if (new RegExp(`\\.[0-9a-f]{8}\\${ext}$`).test(name)) rmSync(p);
|
|
78
|
-
}
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
removeGenerated(join(staticRoot, "css"), ".css");
|
|
82
|
-
removeGenerated(join(staticRoot, "v3", "css"), ".css");
|
|
83
|
-
removeGenerated(moduleRoot, ".js");
|
|
84
|
-
|
|
85
|
-
const modules = new Map();
|
|
86
|
-
for (const abs of walk(moduleRoot)) {
|
|
87
|
-
modules.set(repoPath(abs), {
|
|
88
|
-
abs,
|
|
89
|
-
rel: repoPath(abs),
|
|
90
|
-
source: readFileSync(abs, "utf8"),
|
|
91
|
-
deps: [],
|
|
92
|
-
});
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
const importFromRe = /\b(?:import|export)\s+(?:[^"'()]*?\s+from\s*)?["']([^"']+\.js)["']/g;
|
|
96
|
-
for (const mod of modules.values()) {
|
|
97
|
-
const deps = [];
|
|
98
|
-
let match;
|
|
99
|
-
while ((match = importFromRe.exec(mod.source))) {
|
|
100
|
-
const spec = match[1];
|
|
101
|
-
if (!spec.startsWith(".")) continue;
|
|
102
|
-
const depRel = repoPath(join(dirname(mod.abs), spec));
|
|
103
|
-
if (modules.has(depRel)) deps.push(depRel);
|
|
104
|
-
}
|
|
105
|
-
mod.deps = deps;
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
const hashMemo = new Map();
|
|
109
|
-
function moduleHash(rel, stack = []) {
|
|
110
|
-
if (hashMemo.has(rel)) return hashMemo.get(rel);
|
|
111
|
-
if (stack.includes(rel)) return sha(modules.get(rel).source);
|
|
112
|
-
const mod = modules.get(rel);
|
|
113
|
-
const depHashes = mod.deps
|
|
114
|
-
.sort()
|
|
115
|
-
.map((dep) => `${dep}:${moduleHash(dep, [...stack, rel])}`)
|
|
116
|
-
.join("\n");
|
|
117
|
-
const h = sha(`${mod.source}\n/* dependency-hashes */\n${depHashes}`);
|
|
118
|
-
hashMemo.set(rel, h);
|
|
119
|
-
return h;
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
for (const rel of modules.keys()) moduleHash(rel);
|
|
123
|
-
|
|
124
|
-
function hashedRel(rel, hash) {
|
|
125
|
-
const ext = extname(rel);
|
|
126
|
-
return posix(join(dirname(rel), `${basename(rel, ext)}.${hash}${ext}`));
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
const assets = {};
|
|
130
|
-
|
|
131
|
-
for (const sourceRel of cssSources) {
|
|
132
|
-
const abs = join(repoRoot, sourceRel);
|
|
133
|
-
const source = readFileSync(abs, "utf8");
|
|
134
|
-
const outRel = hashedRel(sourceRel, sha(source));
|
|
135
|
-
writeFileSync(join(repoRoot, outRel), source, "utf8");
|
|
136
|
-
assets[sourceRel] = publicUrl(outRel);
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
function rewriteModule(mod) {
|
|
140
|
-
return mod.source.replace(importFromRe, (full, spec) => {
|
|
141
|
-
if (!spec.startsWith(".")) return full;
|
|
142
|
-
const depRel = repoPath(join(dirname(mod.abs), spec));
|
|
143
|
-
const depHash = hashMemo.get(depRel);
|
|
144
|
-
if (!depHash) return full;
|
|
145
|
-
const depOutAbs = join(repoRoot, hashedRel(depRel, depHash));
|
|
146
|
-
const nextSpec = posix(relative(dirname(join(repoRoot, hashedRel(mod.rel, hashMemo.get(mod.rel)))), depOutAbs));
|
|
147
|
-
const normalized = nextSpec.startsWith(".") ? nextSpec : `./${nextSpec}`;
|
|
148
|
-
return full.replace(spec, normalized);
|
|
149
|
-
});
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
for (const mod of modules.values()) {
|
|
153
|
-
const outRel = hashedRel(mod.rel, hashMemo.get(mod.rel));
|
|
154
|
-
mkdirSync(dirname(join(repoRoot, outRel)), { recursive: true });
|
|
155
|
-
writeFileSync(join(repoRoot, outRel), rewriteModule(mod), "utf8");
|
|
156
|
-
assets[mod.rel] = publicUrl(outRel);
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
const manifest = {
|
|
160
|
-
version: pkgVersion,
|
|
161
|
-
generated_at: "deterministic",
|
|
162
|
-
entrypoints: {
|
|
163
|
-
app: assets[entry],
|
|
164
|
-
styles: cssSources.map((rel) => assets[rel]),
|
|
165
|
-
},
|
|
166
|
-
assets,
|
|
167
|
-
};
|
|
168
|
-
|
|
169
|
-
writeFileSync(manifestPath, JSON.stringify(manifest, null, 2) + "\n", "utf8");
|
|
170
|
-
console.log(`wrote ${repoPath(manifestPath)} with ${Object.keys(assets).length} assets`);
|
package/scripts/lint_v3.mjs
DELETED
|
@@ -1,120 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
/* Lattice v3 frontend lint. Gates `npm run lint`:
|
|
3
|
-
* 1. Syntax-check every v3 ES module (node --check).
|
|
4
|
-
* 2. Design tokens: no raw hex/rgb colors in static/v3/css outside the two
|
|
5
|
-
* token files (lattice.tokens.css, tokens.css) — themed surfaces must use
|
|
6
|
-
* var(--…) tokens.
|
|
7
|
-
* 3. No inline style colors in view JS (style="…color: #…" or
|
|
8
|
-
* style.color = "#…" literals).
|
|
9
|
-
* 4. Privacy: zero CDN/external URLs in shipped static HTML/CSS/JS —
|
|
10
|
-
* fonts/icons/libs are vendored under static/vendor.
|
|
11
|
-
* 5. i18n: routes, shell, and new v4 parity views must use the SPA i18n
|
|
12
|
-
* runtime instead of inert localStorage-only language toggles.
|
|
13
|
-
* Exits non-zero on any failure. */
|
|
14
|
-
import { readdirSync, statSync, readFileSync } from "node:fs";
|
|
15
|
-
import { join, dirname, relative } from "node:path";
|
|
16
|
-
import { fileURLToPath } from "node:url";
|
|
17
|
-
import { spawnSync } from "node:child_process";
|
|
18
|
-
|
|
19
|
-
const repo = join(dirname(fileURLToPath(import.meta.url)), "..");
|
|
20
|
-
const v3js = join(repo, "static", "v3", "js");
|
|
21
|
-
const v3css = join(repo, "static", "v3", "css");
|
|
22
|
-
const staticRoot = join(repo, "static");
|
|
23
|
-
|
|
24
|
-
function walk(dir, ext) {
|
|
25
|
-
const out = [];
|
|
26
|
-
for (const name of readdirSync(dir)) {
|
|
27
|
-
const p = join(dir, name);
|
|
28
|
-
if (statSync(p).isDirectory()) out.push(...walk(p, ext));
|
|
29
|
-
else if (ext.some((e) => name.endsWith(e))) out.push(p);
|
|
30
|
-
}
|
|
31
|
-
return out;
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
let failed = 0;
|
|
35
|
-
const fail = (msg) => { failed++; console.error(`FAIL ${msg}`); };
|
|
36
|
-
|
|
37
|
-
// ── 1. syntax ────────────────────────────────────────────────────────────
|
|
38
|
-
const modules = walk(v3js, [".js"]).sort();
|
|
39
|
-
let syntaxOk = 0;
|
|
40
|
-
for (const file of modules) {
|
|
41
|
-
const r = spawnSync(process.execPath, ["--check", file], { encoding: "utf8" });
|
|
42
|
-
if (r.status === 0) syntaxOk++;
|
|
43
|
-
else fail(`${relative(repo, file)}\n${r.stderr || r.stdout}`);
|
|
44
|
-
}
|
|
45
|
-
console.log(`syntax: ${syntaxOk}/${modules.length} v3 modules pass`);
|
|
46
|
-
|
|
47
|
-
// ── 2. raw colors in v3 css (outside token files) ────────────────────────
|
|
48
|
-
const TOKEN_FILES = new Set(["lattice.tokens.css", "tokens.css"]);
|
|
49
|
-
const colorRe = /#[0-9a-fA-F]{3,8}\b|rgba?\(/;
|
|
50
|
-
let cssChecked = 0;
|
|
51
|
-
for (const file of walk(v3css, [".css"]).sort()) {
|
|
52
|
-
const base = file.split("/").pop();
|
|
53
|
-
if (TOKEN_FILES.has(base) || /\.[0-9a-f]{8}\.css$/.test(base)) continue; // tokens + hashed builds
|
|
54
|
-
cssChecked++;
|
|
55
|
-
const lines = readFileSync(file, "utf8").split("\n");
|
|
56
|
-
lines.forEach((line, i) => {
|
|
57
|
-
const code = line.split("/*")[0];
|
|
58
|
-
// mask-image gradients use #000/transparent as ALPHA values, not themed
|
|
59
|
-
// colors — they are theme-independent and exempt.
|
|
60
|
-
if (/mask-image|-webkit-mask/.test(code)) return;
|
|
61
|
-
if (colorRe.test(code)) fail(`${relative(repo, file)}:${i + 1} raw color (use a var(--…) token): ${line.trim().slice(0, 90)}`);
|
|
62
|
-
});
|
|
63
|
-
}
|
|
64
|
-
console.log(`tokens: ${cssChecked} non-token v3 css files scanned for raw colors`);
|
|
65
|
-
|
|
66
|
-
// ── 3. inline style colors in view JS ────────────────────────────────────
|
|
67
|
-
const inlineColorRe = /style\s*=\s*["'`][^"'`]*(?:color|background)\s*:\s*(#|rgb)/i;
|
|
68
|
-
const styleAssignRe = /\.style\.(color|background(?:Color)?)\s*=\s*["'`](#|rgb)/i;
|
|
69
|
-
let jsChecked = 0;
|
|
70
|
-
for (const file of modules) {
|
|
71
|
-
if (/\.[0-9a-f]{8}\.js$/.test(file)) continue; // hashed builds mirror sources
|
|
72
|
-
jsChecked++;
|
|
73
|
-
const lines = readFileSync(file, "utf8").split("\n");
|
|
74
|
-
lines.forEach((line, i) => {
|
|
75
|
-
if (inlineColorRe.test(line) || styleAssignRe.test(line)) {
|
|
76
|
-
fail(`${relative(repo, file)}:${i + 1} inline style color (use a token/class): ${line.trim().slice(0, 90)}`);
|
|
77
|
-
}
|
|
78
|
-
});
|
|
79
|
-
}
|
|
80
|
-
console.log(`inline-style: ${jsChecked} v3 source modules scanned`);
|
|
81
|
-
|
|
82
|
-
// ── 4. no CDN/external asset URLs in shipped static files ────────────────
|
|
83
|
-
const cdnRe = /https?:\/\/(fonts\.googleapis\.com|fonts\.gstatic\.com|cdn\.jsdelivr\.net|unpkg\.com|cdnjs\.cloudflare\.com)/;
|
|
84
|
-
let shippedChecked = 0;
|
|
85
|
-
for (const file of walk(staticRoot, [".html", ".css", ".js"]).sort()) {
|
|
86
|
-
if (file.includes(`${join("static", "vendor")}`)) continue; // vendored copies may cite origins in comments
|
|
87
|
-
shippedChecked++;
|
|
88
|
-
const lines = readFileSync(file, "utf8").split("\n");
|
|
89
|
-
lines.forEach((line, i) => {
|
|
90
|
-
if (cdnRe.test(line)) fail(`${relative(repo, file)}:${i + 1} CDN reference (vendor it under static/vendor): ${line.trim().slice(0, 90)}`);
|
|
91
|
-
});
|
|
92
|
-
}
|
|
93
|
-
console.log(`privacy: ${shippedChecked} shipped static files scanned for CDN URLs`);
|
|
94
|
-
|
|
95
|
-
// ── 5. i18n acceptance for T9 surfaces ──────────────────────────────────
|
|
96
|
-
const i18nRequired = [
|
|
97
|
-
"static/v3/js/core/routes.js",
|
|
98
|
-
"static/v3/js/core/shell.js",
|
|
99
|
-
"static/v3/js/views/account.js",
|
|
100
|
-
"static/v3/js/views/workspace-admin.js",
|
|
101
|
-
"static/v3/js/views/snapshots.js",
|
|
102
|
-
"static/v3/js/views/activity.js",
|
|
103
|
-
"static/v3/js/views/runs.js",
|
|
104
|
-
"static/v3/js/views/network.js",
|
|
105
|
-
];
|
|
106
|
-
let i18nChecked = 0;
|
|
107
|
-
for (const rel of i18nRequired) {
|
|
108
|
-
const file = join(repo, rel);
|
|
109
|
-
const text = readFileSync(file, "utf8");
|
|
110
|
-
i18nChecked++;
|
|
111
|
-
if (!/i18n\.js/.test(text)) fail(`${rel}: missing i18n runtime import`);
|
|
112
|
-
if (!/\bt\(/.test(text)) fail(`${rel}: no translation lookup found`);
|
|
113
|
-
}
|
|
114
|
-
console.log(`i18n: ${i18nChecked} route/shell/parity modules checked`);
|
|
115
|
-
|
|
116
|
-
if (failed) {
|
|
117
|
-
console.error(`\nv3 frontend lint: ${failed} failure(s)`);
|
|
118
|
-
process.exit(1);
|
|
119
|
-
}
|
|
120
|
-
console.log(`\nv3 frontend lint: all checks pass`);
|
|
@@ -1,63 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"version": "4.0.1",
|
|
3
|
-
"generated_at": "deterministic",
|
|
4
|
-
"entrypoints": {
|
|
5
|
-
"app": "/static/v3/js/app.c5c80c46.js",
|
|
6
|
-
"styles": [
|
|
7
|
-
"/static/css/tokens.3ba22e37.css",
|
|
8
|
-
"/static/v3/css/lattice.tokens.304cbc40.css",
|
|
9
|
-
"/static/v3/css/lattice.base.49deefb5.css",
|
|
10
|
-
"/static/v3/css/lattice.components.cde18231.css",
|
|
11
|
-
"/static/v3/css/lattice.shell.29d36d85.css",
|
|
12
|
-
"/static/v3/css/lattice.views.0a18b6c5.css"
|
|
13
|
-
]
|
|
14
|
-
},
|
|
15
|
-
"assets": {
|
|
16
|
-
"static/css/tokens.css": "/static/css/tokens.3ba22e37.css",
|
|
17
|
-
"static/v3/css/lattice.tokens.css": "/static/v3/css/lattice.tokens.304cbc40.css",
|
|
18
|
-
"static/v3/css/lattice.base.css": "/static/v3/css/lattice.base.49deefb5.css",
|
|
19
|
-
"static/v3/css/lattice.components.css": "/static/v3/css/lattice.components.cde18231.css",
|
|
20
|
-
"static/v3/css/lattice.shell.css": "/static/v3/css/lattice.shell.29d36d85.css",
|
|
21
|
-
"static/v3/css/lattice.views.css": "/static/v3/css/lattice.views.0a18b6c5.css",
|
|
22
|
-
"static/v3/js/app.js": "/static/v3/js/app.c5c80c46.js",
|
|
23
|
-
"static/v3/js/core/api.js": "/static/v3/js/core/api.ba0fbf14.js",
|
|
24
|
-
"static/v3/js/core/components.js": "/static/v3/js/core/components.f25b3b93.js",
|
|
25
|
-
"static/v3/js/core/dom.js": "/static/v3/js/core/dom.a2773eb0.js",
|
|
26
|
-
"static/v3/js/core/i18n.js": "/static/v3/js/core/i18n.880e1fec.js",
|
|
27
|
-
"static/v3/js/core/router.js": "/static/v3/js/core/router.584570f2.js",
|
|
28
|
-
"static/v3/js/core/routes.js": "/static/v3/js/core/routes.37522821.js",
|
|
29
|
-
"static/v3/js/core/shell.js": "/static/v3/js/core/shell.e3f6bbfa.js",
|
|
30
|
-
"static/v3/js/core/store.js": "/static/v3/js/core/store.7b2aa044.js",
|
|
31
|
-
"static/v3/js/views/account.js": "/static/v3/js/views/account.eff40715.js",
|
|
32
|
-
"static/v3/js/views/activity.js": "/static/v3/js/views/activity.0d271ef9.js",
|
|
33
|
-
"static/v3/js/views/admin-audit.js": "/static/v3/js/views/admin-audit.660a1fb1.js",
|
|
34
|
-
"static/v3/js/views/admin-permissions.js": "/static/v3/js/views/admin-permissions.a7ae5f09.js",
|
|
35
|
-
"static/v3/js/views/admin-policies.js": "/static/v3/js/views/admin-policies.3658fd86.js",
|
|
36
|
-
"static/v3/js/views/admin-private-vpc.js": "/static/v3/js/views/admin-private-vpc.7d342d36.js",
|
|
37
|
-
"static/v3/js/views/admin-security.js": "/static/v3/js/views/admin-security.07c66b72.js",
|
|
38
|
-
"static/v3/js/views/admin-users.js": "/static/v3/js/views/admin-users.f7ac7b43.js",
|
|
39
|
-
"static/v3/js/views/agents.js": "/static/v3/js/views/agents.17c5288d.js",
|
|
40
|
-
"static/v3/js/views/chat.js": "/static/v3/js/views/chat.e250e2cc.js",
|
|
41
|
-
"static/v3/js/views/files.js": "/static/v3/js/views/files.adad14c1.js",
|
|
42
|
-
"static/v3/js/views/graph-canvas.js": "/static/v3/js/views/graph-canvas.17c15d65.js",
|
|
43
|
-
"static/v3/js/views/home.js": "/static/v3/js/views/home.24f8b8ae.js",
|
|
44
|
-
"static/v3/js/views/hooks.js": "/static/v3/js/views/hooks.37895880.js",
|
|
45
|
-
"static/v3/js/views/hybrid-search.js": "/static/v3/js/views/hybrid-search.2fb63ed9.js",
|
|
46
|
-
"static/v3/js/views/knowledge-graph.js": "/static/v3/js/views/knowledge-graph.4d09c537.js",
|
|
47
|
-
"static/v3/js/views/marketplace.js": "/static/v3/js/views/marketplace.ab0583d4.js",
|
|
48
|
-
"static/v3/js/views/mcp.js": "/static/v3/js/views/mcp.99b5c6a7.js",
|
|
49
|
-
"static/v3/js/views/memory.js": "/static/v3/js/views/memory.4ebdf474.js",
|
|
50
|
-
"static/v3/js/views/models.js": "/static/v3/js/views/models.a1ffa147.js",
|
|
51
|
-
"static/v3/js/views/my-computer.js": "/static/v3/js/views/my-computer.d9d9ae1c.js",
|
|
52
|
-
"static/v3/js/views/network.js": "/static/v3/js/views/network.52a4f181.js",
|
|
53
|
-
"static/v3/js/views/pipeline.js": "/static/v3/js/views/pipeline.c522f1ce.js",
|
|
54
|
-
"static/v3/js/views/planning.js": "/static/v3/js/views/planning.4876fd77.js",
|
|
55
|
-
"static/v3/js/views/runs.js": "/static/v3/js/views/runs.b63b2afa.js",
|
|
56
|
-
"static/v3/js/views/settings.js": "/static/v3/js/views/settings.b7140634.js",
|
|
57
|
-
"static/v3/js/views/skills.js": "/static/v3/js/views/skills.c6c2f965.js",
|
|
58
|
-
"static/v3/js/views/snapshots.js": "/static/v3/js/views/snapshots.6f5db095.js",
|
|
59
|
-
"static/v3/js/views/tools.js": "/static/v3/js/views/tools.e4f11276.js",
|
|
60
|
-
"static/v3/js/views/workflows.js": "/static/v3/js/views/workflows.7752225a.js",
|
|
61
|
-
"static/v3/js/views/workspace-admin.js": "/static/v3/js/views/workspace-admin.c466029b.js"
|
|
62
|
-
}
|
|
63
|
-
}
|
|
@@ -1,128 +0,0 @@
|
|
|
1
|
-
/* ============================================================================
|
|
2
|
-
* Lattice AI v3 — Base layer (reset + element defaults + lattice backdrop)
|
|
3
|
-
* Token-native: no themed hex values, everything via var(--*).
|
|
4
|
-
* ========================================================================== */
|
|
5
|
-
|
|
6
|
-
*, *::before, *::after { box-sizing: border-box; }
|
|
7
|
-
|
|
8
|
-
html, body { height: 100%; }
|
|
9
|
-
|
|
10
|
-
body {
|
|
11
|
-
margin: 0;
|
|
12
|
-
font-family: var(--lt3-font-sans);
|
|
13
|
-
font-size: var(--lt3-text-md);
|
|
14
|
-
line-height: var(--lt3-leading-normal);
|
|
15
|
-
color: var(--text);
|
|
16
|
-
background: var(--bg);
|
|
17
|
-
-webkit-font-smoothing: antialiased;
|
|
18
|
-
text-rendering: optimizeLegibility;
|
|
19
|
-
overflow: hidden; /* shell owns scroll regions */
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
/* The signature lattice backdrop — a faint structural mesh of nodes+edges.
|
|
23
|
-
Sits behind the whole app; reinforces the "lattice" identity without noise. */
|
|
24
|
-
.lt3-app::before {
|
|
25
|
-
content: "";
|
|
26
|
-
position: fixed;
|
|
27
|
-
inset: 0;
|
|
28
|
-
z-index: 0;
|
|
29
|
-
pointer-events: none;
|
|
30
|
-
background:
|
|
31
|
-
radial-gradient(circle at center, var(--lt3-mesh-node) 0.9px, transparent 1.1px),
|
|
32
|
-
linear-gradient(var(--lt3-mesh-line) 1px, transparent 1px),
|
|
33
|
-
linear-gradient(90deg, var(--lt3-mesh-line) 1px, transparent 1px),
|
|
34
|
-
var(--app-bg);
|
|
35
|
-
background-size:
|
|
36
|
-
var(--lt3-mesh-size) var(--lt3-mesh-size),
|
|
37
|
-
var(--lt3-mesh-size) var(--lt3-mesh-size),
|
|
38
|
-
var(--lt3-mesh-size) var(--lt3-mesh-size),
|
|
39
|
-
cover;
|
|
40
|
-
background-position: center;
|
|
41
|
-
mask-image: radial-gradient(ellipse 120% 90% at 50% -10%, #000 55%, transparent 100%);
|
|
42
|
-
opacity: 0.9;
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
h1, h2, h3, h4, p, figure { margin: 0; }
|
|
46
|
-
|
|
47
|
-
a { color: inherit; text-decoration: none; }
|
|
48
|
-
|
|
49
|
-
button {
|
|
50
|
-
font: inherit;
|
|
51
|
-
color: inherit;
|
|
52
|
-
cursor: pointer;
|
|
53
|
-
border: none;
|
|
54
|
-
background: none;
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
input, select, textarea { font: inherit; color: inherit; }
|
|
58
|
-
|
|
59
|
-
textarea { resize: vertical; }
|
|
60
|
-
|
|
61
|
-
img, svg { display: block; max-width: 100%; }
|
|
62
|
-
|
|
63
|
-
code, pre, kbd, samp { font-family: var(--lt3-font-mono); }
|
|
64
|
-
|
|
65
|
-
:root[data-lt-icons="fallback"] .ti {
|
|
66
|
-
display: inline-grid;
|
|
67
|
-
place-items: center;
|
|
68
|
-
min-width: 1em;
|
|
69
|
-
min-height: 1em;
|
|
70
|
-
font-family: var(--lt3-font-mono);
|
|
71
|
-
font-size: 0.72em;
|
|
72
|
-
font-style: normal;
|
|
73
|
-
font-weight: 800;
|
|
74
|
-
line-height: 1;
|
|
75
|
-
text-transform: uppercase;
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
:root[data-lt-icons="fallback"] .ti::before {
|
|
79
|
-
content: attr(data-fallback);
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
:focus-visible {
|
|
83
|
-
outline: 2px solid var(--focus-ring);
|
|
84
|
-
outline-offset: 2px;
|
|
85
|
-
border-radius: var(--lt3-radius-xs);
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
::-webkit-scrollbar { width: 8px; height: 8px; }
|
|
89
|
-
::-webkit-scrollbar-track { background: transparent; }
|
|
90
|
-
::-webkit-scrollbar-thumb {
|
|
91
|
-
background: color-mix(in srgb, var(--border-strong) 60%, transparent);
|
|
92
|
-
border-radius: 99px;
|
|
93
|
-
border: 2px solid transparent;
|
|
94
|
-
background-clip: padding-box;
|
|
95
|
-
}
|
|
96
|
-
::-webkit-scrollbar-thumb:hover { background: var(--border-strong); background-clip: padding-box; }
|
|
97
|
-
|
|
98
|
-
/* Accessible visually-hidden utility */
|
|
99
|
-
.lt3-sr {
|
|
100
|
-
position: absolute !important;
|
|
101
|
-
width: 1px; height: 1px;
|
|
102
|
-
padding: 0; margin: -1px;
|
|
103
|
-
overflow: hidden; clip: rect(0,0,0,0);
|
|
104
|
-
white-space: nowrap; border: 0;
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
.lt3-skip {
|
|
108
|
-
position: fixed;
|
|
109
|
-
top: var(--lt3-space-3);
|
|
110
|
-
left: var(--lt3-space-3);
|
|
111
|
-
z-index: var(--lt3-z-toast);
|
|
112
|
-
padding: var(--lt3-space-2) var(--lt3-space-4);
|
|
113
|
-
background: var(--accent);
|
|
114
|
-
color: var(--lt3-on-accent);
|
|
115
|
-
border-radius: var(--lt3-radius-sm);
|
|
116
|
-
transform: translateY(-200%);
|
|
117
|
-
transition: transform var(--lt3-dur-2) var(--lt3-ease);
|
|
118
|
-
}
|
|
119
|
-
.lt3-skip:focus { transform: translateY(0); }
|
|
120
|
-
|
|
121
|
-
@media (prefers-reduced-motion: reduce) {
|
|
122
|
-
*, *::before, *::after {
|
|
123
|
-
animation-duration: 0.001ms !important;
|
|
124
|
-
animation-iteration-count: 1 !important;
|
|
125
|
-
transition-duration: 0.001ms !important;
|
|
126
|
-
scroll-behavior: auto !important;
|
|
127
|
-
}
|
|
128
|
-
}
|