@dorsk/tsumikit 0.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.
Files changed (99) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +165 -0
  3. package/dist/autoresize.d.ts +11 -0
  4. package/dist/autoresize.js +24 -0
  5. package/dist/components/atoms/Badge.svelte +72 -0
  6. package/dist/components/atoms/Badge.svelte.d.ts +12 -0
  7. package/dist/components/atoms/Button.svelte +156 -0
  8. package/dist/components/atoms/Button.svelte.d.ts +13 -0
  9. package/dist/components/atoms/Card.svelte +46 -0
  10. package/dist/components/atoms/Card.svelte.d.ts +11 -0
  11. package/dist/components/atoms/Checkbox.svelte +99 -0
  12. package/dist/components/atoms/Checkbox.svelte.d.ts +10 -0
  13. package/dist/components/atoms/Chip.svelte +53 -0
  14. package/dist/components/atoms/Chip.svelte.d.ts +11 -0
  15. package/dist/components/atoms/Heading.svelte +66 -0
  16. package/dist/components/atoms/Heading.svelte.d.ts +13 -0
  17. package/dist/components/atoms/Icon.svelte +151 -0
  18. package/dist/components/atoms/Icon.svelte.d.ts +18 -0
  19. package/dist/components/atoms/Input.svelte +42 -0
  20. package/dist/components/atoms/Input.svelte.d.ts +10 -0
  21. package/dist/components/atoms/Link.svelte +31 -0
  22. package/dist/components/atoms/Link.svelte.d.ts +10 -0
  23. package/dist/components/atoms/Progress.svelte +59 -0
  24. package/dist/components/atoms/Progress.svelte.d.ts +9 -0
  25. package/dist/components/atoms/Select.svelte +95 -0
  26. package/dist/components/atoms/Select.svelte.d.ts +11 -0
  27. package/dist/components/atoms/Slider.svelte +136 -0
  28. package/dist/components/atoms/Slider.svelte.d.ts +14 -0
  29. package/dist/components/atoms/Switch.svelte +64 -0
  30. package/dist/components/atoms/Switch.svelte.d.ts +8 -0
  31. package/dist/components/atoms/Text.svelte +127 -0
  32. package/dist/components/atoms/Text.svelte.d.ts +16 -0
  33. package/dist/components/atoms/Textarea.svelte +62 -0
  34. package/dist/components/atoms/Textarea.svelte.d.ts +11 -0
  35. package/dist/components/layouts/AppShell.svelte +304 -0
  36. package/dist/components/layouts/AppShell.svelte.d.ts +21 -0
  37. package/dist/components/layouts/AutoGrid.svelte +36 -0
  38. package/dist/components/layouts/AutoGrid.svelte.d.ts +12 -0
  39. package/dist/components/layouts/Cluster.svelte +45 -0
  40. package/dist/components/layouts/Cluster.svelte.d.ts +14 -0
  41. package/dist/components/layouts/Container.svelte +40 -0
  42. package/dist/components/layouts/Container.svelte.d.ts +13 -0
  43. package/dist/components/layouts/NavItem.svelte +95 -0
  44. package/dist/components/layouts/NavItem.svelte.d.ts +14 -0
  45. package/dist/components/layouts/Stack.svelte +44 -0
  46. package/dist/components/layouts/Stack.svelte.d.ts +13 -0
  47. package/dist/components/molecules/Accordion.svelte +94 -0
  48. package/dist/components/molecules/Accordion.svelte.d.ts +16 -0
  49. package/dist/components/molecules/CodeBlock.svelte +119 -0
  50. package/dist/components/molecules/CodeBlock.svelte.d.ts +17 -0
  51. package/dist/components/molecules/CopyButton.svelte +80 -0
  52. package/dist/components/molecules/CopyButton.svelte.d.ts +13 -0
  53. package/dist/components/molecules/Dropzone.svelte +140 -0
  54. package/dist/components/molecules/Dropzone.svelte.d.ts +13 -0
  55. package/dist/components/molecules/Field.svelte +57 -0
  56. package/dist/components/molecules/Field.svelte.d.ts +12 -0
  57. package/dist/components/molecules/FileButton.svelte +68 -0
  58. package/dist/components/molecules/FileButton.svelte.d.ts +14 -0
  59. package/dist/components/molecules/FontScalePicker.svelte +21 -0
  60. package/dist/components/molecules/FontScalePicker.svelte.d.ts +6 -0
  61. package/dist/components/molecules/IconButton.svelte +36 -0
  62. package/dist/components/molecules/IconButton.svelte.d.ts +13 -0
  63. package/dist/components/molecules/Menu.svelte +120 -0
  64. package/dist/components/molecules/Menu.svelte.d.ts +17 -0
  65. package/dist/components/molecules/Modal.svelte +263 -0
  66. package/dist/components/molecules/Modal.svelte.d.ts +13 -0
  67. package/dist/components/molecules/OptionButton.svelte +76 -0
  68. package/dist/components/molecules/OptionButton.svelte.d.ts +10 -0
  69. package/dist/components/molecules/Popover.svelte +125 -0
  70. package/dist/components/molecules/Popover.svelte.d.ts +18 -0
  71. package/dist/components/molecules/RadioGroup.svelte +110 -0
  72. package/dist/components/molecules/RadioGroup.svelte.d.ts +16 -0
  73. package/dist/components/molecules/SelectButton.svelte +52 -0
  74. package/dist/components/molecules/SelectButton.svelte.d.ts +15 -0
  75. package/dist/components/molecules/Tabs.svelte +119 -0
  76. package/dist/components/molecules/Tabs.svelte.d.ts +15 -0
  77. package/dist/components/molecules/ThemePicker.svelte +22 -0
  78. package/dist/components/molecules/ThemePicker.svelte.d.ts +6 -0
  79. package/dist/components/molecules/Toaster.svelte +73 -0
  80. package/dist/components/molecules/Toaster.svelte.d.ts +18 -0
  81. package/dist/components/molecules/Toggle.svelte +68 -0
  82. package/dist/components/molecules/Toggle.svelte.d.ts +11 -0
  83. package/dist/components/molecules/Tooltip.svelte +106 -0
  84. package/dist/components/molecules/Tooltip.svelte.d.ts +10 -0
  85. package/dist/components/organisms/DataTable.svelte +145 -0
  86. package/dist/components/organisms/DataTable.svelte.d.ts +43 -0
  87. package/dist/env.d.ts +1 -0
  88. package/dist/env.js +4 -0
  89. package/dist/index.d.ts +46 -0
  90. package/dist/index.js +56 -0
  91. package/dist/stores/fontscale.svelte.d.ts +15 -0
  92. package/dist/stores/fontscale.svelte.js +49 -0
  93. package/dist/stores/theme.svelte.d.ts +96 -0
  94. package/dist/stores/theme.svelte.js +71 -0
  95. package/dist/stores/toast.svelte.d.ts +19 -0
  96. package/dist/stores/toast.svelte.js +26 -0
  97. package/dist/styles/app.css +522 -0
  98. package/dist/styles/variables.css +651 -0
  99. package/package.json +71 -0
@@ -0,0 +1,651 @@
1
+ /*
2
+ * THEME — single source of truth.
3
+ *
4
+ * Swapping the theme = replacing THIS ONE FILE. Every color, space, radius,
5
+ * font and size used anywhere in the app is a `var(--…)` defined here. No
6
+ * component hard-codes a color or pixel value.
7
+ *
8
+ * Default is dark; `[data-theme="…"]` blocks at the bottom flip palette tokens
9
+ * only (everything downstream is unchanged).
10
+ */
11
+
12
+ :root {
13
+ /* Tell the UA our default surface is dark so native widgets (form controls,
14
+ scrollbars, the caret, spellcheck underlines) render their dark variant.
15
+ Light themes flip this in their own block. */
16
+ color-scheme: dark;
17
+
18
+ /* ---- Palette (raw) ---- */
19
+ --c-bg: #0f1115;
20
+ --c-bg-elev: #171a21;
21
+ --c-bg-elev-2: #1f232c;
22
+ --c-surface: #21262f;
23
+ --c-border: #2c323d;
24
+ --c-border-strong: #3a414e;
25
+ --c-text: #e6e9ef;
26
+ --c-text-muted: #9aa3b2;
27
+ --c-text-faint: #6b7384;
28
+
29
+ --c-accent: #5ad6a0;
30
+ --c-accent-ink: #04130c;
31
+ --c-accent-dim: #2a6e54;
32
+
33
+ --c-blue: #5aa9ff;
34
+ --c-amber: #f0b454;
35
+ --c-red: #f0716b;
36
+ --c-green: #5ad6a0;
37
+ --c-violet: #b48ef0;
38
+ --c-gold: #f0b454;
39
+ --c-teal: #4fd6cf;
40
+
41
+ /* ---- Role colors (CCT-161) — uniform message theming, all derived from
42
+ palette tokens so each theme adapts. Used for role badges + bubble tints. */
43
+ --role-user: var(--c-green);
44
+ --role-assistant: var(--c-blue);
45
+ --role-system: var(--c-violet);
46
+ --role-tool: var(--c-gold);
47
+ --role-mcp: var(--c-teal);
48
+ /* compact/clear boundary — neutral "paper white" so it reads as structure */
49
+ --role-boundary: var(--c-text-muted);
50
+
51
+ /* ---- Markdown / syntax-highlight tokens (CCT-161) — themeable ---- */
52
+ --md-text: var(--c-text-muted); /* base prose: grayish */
53
+ --md-strong: var(--c-text); /* bold: bright white-ish */
54
+ --md-code: var(--c-blue); /* inline code: bold blue */
55
+ --md-code-bg: color-mix(in srgb, var(--c-blue) 14%, var(--c-bg));
56
+ --md-heading: var(--c-text);
57
+ --syn-keyword: var(--c-violet);
58
+ --syn-string: var(--c-green);
59
+ --syn-number: var(--c-amber);
60
+ --syn-comment: var(--c-text-faint);
61
+ --syn-function: var(--c-blue);
62
+ --syn-punct: var(--c-text-muted);
63
+
64
+ /* ---- Semantic ---- */
65
+ --bg: var(--c-bg);
66
+ --bg-elevated: var(--c-bg-elev);
67
+ --bg-elevated-2: var(--c-bg-elev-2);
68
+ --surface: var(--c-surface);
69
+ --border: var(--c-border);
70
+ --border-strong: var(--c-border-strong);
71
+
72
+ --text: var(--c-text);
73
+ --text-muted: var(--c-text-muted);
74
+ --text-faint: var(--c-text-faint);
75
+ --text-on-accent: var(--c-accent-ink);
76
+
77
+ --accent: var(--c-accent);
78
+ --accent-dim: var(--c-accent-dim);
79
+ --link: var(--c-blue);
80
+
81
+ --ok: var(--c-green);
82
+ --warn: var(--c-amber);
83
+ --danger: var(--c-red);
84
+ --info: var(--c-blue);
85
+
86
+ /* status dots */
87
+ --dot-active: var(--c-green);
88
+ --dot-stale: var(--c-amber);
89
+ --dot-dead: var(--c-text-faint);
90
+ --dot-hibernated: var(--c-red);
91
+
92
+ /* attention highlight (needs-input row) */
93
+ --attention-bg: color-mix(in srgb, var(--c-amber) 12%, transparent);
94
+ --attention-bar: var(--c-amber);
95
+
96
+ /* machine-badge tinting (CCT-271, fixed CCT-272) — the hue is per-machine
97
+ (name hash or operator override, --mh, set on the badge element), but the
98
+ saturation/lightness come from the theme as `<sat%> <light%>` PAIRS. The
99
+ badge composes `hsl(var(--mh) var(--mach-*-sl))` itself, so both vars
100
+ resolve on the element where --mh lives. (Composing the full hsl() here at
101
+ :root would break: --mh is undefined at :root, so the token is invalid and
102
+ inherits down invalid → white badges.) Dark themes default here; light
103
+ themes flip to dark-text-on-pale-tint in their [data-theme] block. */
104
+ --mach-bg-sl: 45% 22%;
105
+ --mach-fg-sl: 70% 80%;
106
+ --mach-border-sl: 45% 35%;
107
+
108
+ /* ---- Typography ---- */
109
+ --font-sans:
110
+ ui-sans-serif, system-ui, -apple-system, "Segoe UI", Roboto,
111
+ "Helvetica Neue", Arial, sans-serif;
112
+ --font-mono:
113
+ ui-monospace, "SF Mono", "JetBrains Mono", "Fira Code", Menlo, Consolas,
114
+ monospace;
115
+
116
+ /* Text-scale factor (CCT-305). The UI font control multiplies ONLY the
117
+ font-size tokens by this — not the root font-size — so bumping the size up
118
+ enlarges text (chat, session text, badges, labels) meaningfully while the
119
+ chrome (button heights, padding, spacing, the px-pinned header) stays put.
120
+ Previously the control scaled the document root font-size, which grew every
121
+ rem-based chrome dimension in lockstep so text "barely grew" relative to the
122
+ ballooning UI. Default 1 = no scaling. */
123
+ --fs-scale: 1;
124
+
125
+ --fs-xs: calc(0.75rem * var(--fs-scale)); /* 12 */
126
+ --fs-sm: calc(0.8125rem * var(--fs-scale)); /* 13 */
127
+ --fs-base: calc(0.9375rem * var(--fs-scale)); /* 15 */
128
+ --fs-md: calc(1rem * var(--fs-scale)); /* 16 */
129
+ --fs-lg: calc(1.125rem * var(--fs-scale)); /* 18 */
130
+ --fs-xl: calc(1.375rem * var(--fs-scale)); /* 22 */
131
+ --fs-2xl: calc(1.75rem * var(--fs-scale)); /* 28 */
132
+
133
+ --fw-normal: 400;
134
+ --fw-medium: 500;
135
+ --fw-semibold: 600;
136
+ --fw-bold: 700;
137
+
138
+ --lh-tight: 1.25;
139
+ --lh-normal: 1.5;
140
+
141
+ /* ---- Spacing (4px base) ---- */
142
+ --sp-0: 0;
143
+ --sp-1: 0.25rem;
144
+ --sp-2: 0.5rem;
145
+ --sp-3: 0.75rem;
146
+ --sp-4: 1rem;
147
+ --sp-5: 1.25rem;
148
+ --sp-6: 1.5rem;
149
+ --sp-8: 2rem;
150
+ --sp-10: 2.5rem;
151
+ --sp-12: 3rem;
152
+
153
+ /* ---- Radii ---- */
154
+ --r-sm: 6px;
155
+ --r-md: 10px;
156
+ --r-lg: 14px;
157
+ --r-pill: 999px;
158
+
159
+ /* ---- Elevation ---- */
160
+ --shadow-sm: 0 1px 2px rgba(0, 0, 0, 0.4);
161
+ --shadow-md: 0 6px 20px rgba(0, 0, 0, 0.45);
162
+ --shadow-lg: 0 12px 40px rgba(0, 0, 0, 0.55);
163
+
164
+ /* ---- Controls (CCT-250 item 1) — shared height for icon buttons + inputs
165
+ so toolbars/composers line up. `.btn-control` (app.css) consumes it. */
166
+ --control-height: 2.75rem;
167
+
168
+ /* ---- Layout ---- */
169
+ /* Fixed (px, not rem): the header is chrome that must NOT scale with the
170
+ UI font slider it hosts (CCT-264) — a rem header-h would grow under the
171
+ slider and reflow it mid-drag. Pinning here keeps the bar height, the
172
+ content clearance, and the sticky offsets that key off it all consistent
173
+ with the fixed toolbar. */
174
+ --header-h: 52px;
175
+ --nav-h: 3.5rem;
176
+ --content-max: 56rem;
177
+ --z-header: 50;
178
+ --z-nav: 50;
179
+ --z-drawer: 100;
180
+ --z-modal: 200;
181
+ --z-toast: 300;
182
+
183
+ /* safe-area insets (phones with notch / gesture bar / bottom address bar) */
184
+ --safe-top: env(safe-area-inset-top, 0px);
185
+ --safe-bottom: env(safe-area-inset-bottom, 0px);
186
+ --safe-left: env(safe-area-inset-left, 0px);
187
+ --safe-right: env(safe-area-inset-right, 0px);
188
+
189
+ --ease: cubic-bezier(0.2, 0.8, 0.2, 1);
190
+ }
191
+
192
+ [data-theme="light"] {
193
+ color-scheme: light;
194
+ --c-bg: #f6f7f9;
195
+ --c-bg-elev: #ffffff;
196
+ --c-bg-elev-2: #eef0f3;
197
+ --c-surface: #ffffff;
198
+ --c-border: #d9dde3;
199
+ --c-border-strong: #c2c8d1;
200
+ --c-text: #1a1d23;
201
+ --c-text-muted: #5b6472;
202
+ --c-text-faint: #8a93a3;
203
+ --c-accent: #1f9d6b;
204
+ --c-accent-ink: #ffffff;
205
+ --c-accent-dim: #9bd9c2;
206
+ --c-blue: #2f7ddb;
207
+ --c-amber: #c98a1d;
208
+ --c-red: #d6463f;
209
+ --c-green: #1f9d6b;
210
+ --c-violet: #7c3aed;
211
+ --c-gold: #c98a1d;
212
+ --c-teal: #0d9488;
213
+ --md-code-bg: color-mix(in srgb, var(--c-blue) 12%, var(--c-bg));
214
+ --shadow-sm: 0 1px 2px rgba(20, 24, 31, 0.08);
215
+ --shadow-md: 0 6px 20px rgba(20, 24, 31, 0.12);
216
+ --shadow-lg: 0 12px 40px rgba(20, 24, 31, 0.18);
217
+ /* light bg → pale tint + dark ink so badges stay legible */
218
+ --mach-bg-sl: 60% 90%;
219
+ --mach-fg-sl: 55% 30%;
220
+ --mach-border-sl: 45% 72%;
221
+ }
222
+
223
+ /* ---- Sepia / "Claude paper" theme (CCT-161 item 3) — warm cream paper,
224
+ brown ink, solarized-ish accents. Same variable contract as light/dark. */
225
+ [data-theme="sepia"] {
226
+ color-scheme: light;
227
+ --c-bg: #f4ecd8;
228
+ --c-bg-elev: #faf3e3;
229
+ --c-bg-elev-2: #ece1c6;
230
+ --c-surface: #faf3e3;
231
+ --c-border: #ddccab;
232
+ --c-border-strong: #c8b48c;
233
+ --c-text: #3b2f1e;
234
+ --c-text-muted: #6f5d42;
235
+ --c-text-faint: #9a8868;
236
+ --c-accent: #5a8250;
237
+ --c-accent-ink: #faf3e3;
238
+ --c-accent-dim: #aec79f;
239
+ --c-blue: #2c6f9b;
240
+ --c-amber: #b07a18;
241
+ --c-red: #b5403a;
242
+ --c-green: #5a8250;
243
+ --c-violet: #8a5fa8;
244
+ --c-gold: #a87d1f;
245
+ --c-teal: #2a8f86;
246
+ --md-code-bg: color-mix(in srgb, var(--c-blue) 14%, var(--c-bg));
247
+ --shadow-sm: 0 1px 2px rgba(80, 60, 30, 0.1);
248
+ --shadow-md: 0 6px 20px rgba(80, 60, 30, 0.14);
249
+ --shadow-lg: 0 12px 40px rgba(80, 60, 30, 0.2);
250
+ /* warm, slightly desaturated tint to sit with the cream paper */
251
+ --mach-bg-sl: 40% 84%;
252
+ --mach-fg-sl: 45% 32%;
253
+ --mach-border-sl: 35% 66%;
254
+ }
255
+
256
+ /* ---- Catppuccin Mocha - official Mocha base/surface/text/accent colors. */
257
+ [data-theme="mocha"] {
258
+ --c-bg: #1e1e2e;
259
+ --c-bg-elev: #242438;
260
+ --c-bg-elev-2: #313244;
261
+ --c-surface: #313244;
262
+ --c-border: #45475a;
263
+ --c-border-strong: #585b70;
264
+ --c-text: #cdd6f4;
265
+ --c-text-muted: #bac2de;
266
+ --c-text-faint: #7f849c;
267
+ --c-accent: #cba6f7;
268
+ --c-accent-ink: #1e1e2e;
269
+ --c-accent-dim: #6c7086;
270
+ --c-blue: #89b4fa;
271
+ --c-amber: #f9e2af;
272
+ --c-red: #f38ba8;
273
+ --c-green: #a6e3a1;
274
+ --c-violet: #cba6f7;
275
+ --c-gold: #f9e2af;
276
+ --c-teal: #94e2d5;
277
+ --shadow-sm: 0 1px 2px rgba(17, 17, 27, 0.36);
278
+ --shadow-md: 0 6px 20px rgba(17, 17, 27, 0.42);
279
+ --shadow-lg: 0 12px 40px rgba(17, 17, 27, 0.5);
280
+ /* soft pastel tint to match Catppuccin's low-saturation palette */
281
+ --mach-bg-sl: 35% 26%;
282
+ --mach-fg-sl: 55% 82%;
283
+ --mach-border-sl: 35% 40%;
284
+ }
285
+
286
+ /* ---- Dracula - official palette colors with UI contrast assignments. */
287
+ [data-theme="dracula"] {
288
+ --c-bg: #282a36;
289
+ --c-bg-elev: #343746;
290
+ --c-bg-elev-2: #44475a;
291
+ --c-surface: #343746;
292
+ --c-border: #44475a;
293
+ --c-border-strong: #6272a4;
294
+ --c-text: #f8f8f2;
295
+ --c-text-muted: #c7c9d1;
296
+ --c-text-faint: #6272a4;
297
+ --c-accent: #bd93f9;
298
+ --c-accent-ink: #1d1726;
299
+ --c-accent-dim: #6f558f;
300
+ --c-blue: #8be9fd;
301
+ --c-amber: #ffb86c;
302
+ --c-red: #ff5555;
303
+ --c-green: #50fa7b;
304
+ --c-violet: #bd93f9;
305
+ --c-gold: #f1fa8c;
306
+ --c-teal: #8be9fd;
307
+ --shadow-sm: 0 1px 2px rgba(20, 21, 27, 0.36);
308
+ --shadow-md: 0 6px 20px rgba(20, 21, 27, 0.44);
309
+ --shadow-lg: 0 12px 40px rgba(20, 21, 27, 0.52);
310
+ /* vivid, high-chroma to echo Dracula's neon accents */
311
+ --mach-bg-sl: 55% 24%;
312
+ --mach-fg-sl: 85% 82%;
313
+ --mach-border-sl: 55% 40%;
314
+ }
315
+
316
+ /* ---- Nord - Polar Night, Snow Storm, Frost, and Aurora palette tokens. */
317
+ [data-theme="nord"] {
318
+ --c-bg: #2e3440;
319
+ --c-bg-elev: #3b4252;
320
+ --c-bg-elev-2: #434c5e;
321
+ --c-surface: #3b4252;
322
+ --c-border: #4c566a;
323
+ --c-border-strong: #5e81ac;
324
+ --c-text: #eceff4;
325
+ --c-text-muted: #d8dee9;
326
+ --c-text-faint: #8fbcbb;
327
+ --c-accent: #88c0d0;
328
+ --c-accent-ink: #1f242d;
329
+ --c-accent-dim: #5e81ac;
330
+ --c-blue: #81a1c1;
331
+ --c-amber: #ebcb8b;
332
+ --c-red: #bf616a;
333
+ --c-green: #a3be8c;
334
+ --c-violet: #b48ead;
335
+ --c-gold: #ebcb8b;
336
+ --c-teal: #8fbcbb;
337
+ --shadow-sm: 0 1px 2px rgba(20, 25, 34, 0.34);
338
+ --shadow-md: 0 6px 20px rgba(20, 25, 34, 0.42);
339
+ --shadow-lg: 0 12px 40px rgba(20, 25, 34, 0.5);
340
+ /* muted, frost-cool tint in keeping with Nord's restrained palette */
341
+ --mach-bg-sl: 28% 30%;
342
+ --mach-fg-sl: 40% 84%;
343
+ --mach-border-sl: 28% 44%;
344
+ }
345
+
346
+ /* ---- Tokyo Night — deep blue/indigo base with purple + cyan accents. */
347
+ [data-theme="tokyonight"] {
348
+ --c-bg: #1a1b26;
349
+ --c-bg-elev: #1f2233;
350
+ --c-bg-elev-2: #24283b;
351
+ --c-surface: #1f2233;
352
+ --c-border: #2f344d;
353
+ --c-border-strong: #414868;
354
+ --c-text: #c0caf5;
355
+ --c-text-muted: #9aa5ce;
356
+ --c-text-faint: #565f89;
357
+ --c-accent: #7aa2f7;
358
+ --c-accent-ink: #11121d;
359
+ --c-accent-dim: #3d59a1;
360
+ --c-blue: #7aa2f7;
361
+ --c-amber: #e0af68;
362
+ --c-red: #f7768e;
363
+ --c-green: #9ece6a;
364
+ --c-violet: #bb9af7;
365
+ --c-gold: #e0af68;
366
+ --c-teal: #7dcfff;
367
+ --shadow-sm: 0 1px 2px rgba(13, 14, 22, 0.4);
368
+ --shadow-md: 0 6px 20px rgba(13, 14, 22, 0.48);
369
+ --shadow-lg: 0 12px 40px rgba(13, 14, 22, 0.56);
370
+ --mach-bg-sl: 48% 23%;
371
+ --mach-fg-sl: 75% 82%;
372
+ --mach-border-sl: 48% 38%;
373
+ }
374
+
375
+ /* ---- Gruvbox (dark, medium) — warm retro browns with orange/green accents. */
376
+ [data-theme="gruvbox"] {
377
+ --c-bg: #282828;
378
+ --c-bg-elev: #32302f;
379
+ --c-bg-elev-2: #3c3836;
380
+ --c-surface: #32302f;
381
+ --c-border: #504945;
382
+ --c-border-strong: #665c54;
383
+ --c-text: #ebdbb2;
384
+ --c-text-muted: #d5c4a1;
385
+ --c-text-faint: #a89984;
386
+ --c-accent: #fe8019;
387
+ --c-accent-ink: #1d1a14;
388
+ --c-accent-dim: #af3a03;
389
+ --c-blue: #83a598;
390
+ --c-amber: #fabd2f;
391
+ --c-red: #fb4934;
392
+ --c-green: #b8bb26;
393
+ --c-violet: #d3869b;
394
+ --c-gold: #fabd2f;
395
+ --c-teal: #8ec07c;
396
+ --shadow-sm: 0 1px 2px rgba(20, 19, 17, 0.4);
397
+ --shadow-md: 0 6px 20px rgba(20, 19, 17, 0.48);
398
+ --shadow-lg: 0 12px 40px rgba(20, 19, 17, 0.56);
399
+ --mach-bg-sl: 42% 25%;
400
+ --mach-fg-sl: 60% 80%;
401
+ --mach-border-sl: 42% 40%;
402
+ }
403
+
404
+ /* ---- Solarized Dark — Ethan Schoonover's base03 with the 16-color accents. */
405
+ [data-theme="solarized"] {
406
+ --c-bg: #002b36;
407
+ --c-bg-elev: #073642;
408
+ --c-bg-elev-2: #0a4350;
409
+ --c-surface: #073642;
410
+ --c-border: #0f4d5c;
411
+ --c-border-strong: #586e75;
412
+ --c-text: #eee8d5;
413
+ --c-text-muted: #93a1a1;
414
+ --c-text-faint: #657b83;
415
+ --c-accent: #2aa198;
416
+ --c-accent-ink: #002b36;
417
+ --c-accent-dim: #1a6b66;
418
+ --c-blue: #268bd2;
419
+ --c-amber: #b58900;
420
+ --c-red: #dc322f;
421
+ --c-green: #859900;
422
+ --c-violet: #6c71c4;
423
+ --c-gold: #b58900;
424
+ --c-teal: #2aa198;
425
+ --shadow-sm: 0 1px 2px rgba(0, 17, 22, 0.42);
426
+ --shadow-md: 0 6px 20px rgba(0, 17, 22, 0.5);
427
+ --shadow-lg: 0 12px 40px rgba(0, 17, 22, 0.58);
428
+ --mach-bg-sl: 50% 20%;
429
+ --mach-fg-sl: 55% 78%;
430
+ --mach-border-sl: 50% 34%;
431
+ }
432
+
433
+ /* ---- Rosé Pine — soho-vibe with muted rose, iris (purple) and pine. */
434
+ [data-theme="rosepine"] {
435
+ --c-bg: #191724;
436
+ --c-bg-elev: #1f1d2e;
437
+ --c-bg-elev-2: #26233a;
438
+ --c-surface: #1f1d2e;
439
+ --c-border: #2a2839;
440
+ --c-border-strong: #403d52;
441
+ --c-text: #e0def4;
442
+ --c-text-muted: #908caa;
443
+ --c-text-faint: #6e6a86;
444
+ --c-accent: #c4a7e7;
445
+ --c-accent-ink: #14121d;
446
+ --c-accent-dim: #6c5d8a;
447
+ --c-blue: #9ccfd8;
448
+ --c-amber: #f6c177;
449
+ --c-red: #eb6f92;
450
+ --c-green: #3e8fb0;
451
+ --c-violet: #c4a7e7;
452
+ --c-gold: #f6c177;
453
+ --c-teal: #9ccfd8;
454
+ --shadow-sm: 0 1px 2px rgba(13, 12, 20, 0.42);
455
+ --shadow-md: 0 6px 20px rgba(13, 12, 20, 0.5);
456
+ --shadow-lg: 0 12px 40px rgba(13, 12, 20, 0.58);
457
+ --mach-bg-sl: 38% 26%;
458
+ --mach-fg-sl: 58% 84%;
459
+ --mach-border-sl: 38% 42%;
460
+ }
461
+
462
+ /* ---- One Dark — Atom's classic palette, blue-leaning with warm accents. */
463
+ [data-theme="onedark"] {
464
+ --c-bg: #282c34;
465
+ --c-bg-elev: #2f343f;
466
+ --c-bg-elev-2: #353b45;
467
+ --c-surface: #2f343f;
468
+ --c-border: #3e4451;
469
+ --c-border-strong: #4b5263;
470
+ --c-text: #abb2bf;
471
+ --c-text-muted: #9da5b4;
472
+ --c-text-faint: #5c6370;
473
+ --c-accent: #61afef;
474
+ --c-accent-ink: #11151c;
475
+ --c-accent-dim: #2c6da3;
476
+ --c-blue: #61afef;
477
+ --c-amber: #e5c07b;
478
+ --c-red: #e06c75;
479
+ --c-green: #98c379;
480
+ --c-violet: #c678dd;
481
+ --c-gold: #e5c07b;
482
+ --c-teal: #56b6c2;
483
+ --shadow-sm: 0 1px 2px rgba(17, 19, 25, 0.4);
484
+ --shadow-md: 0 6px 20px rgba(17, 19, 25, 0.48);
485
+ --shadow-lg: 0 12px 40px rgba(17, 19, 25, 0.56);
486
+ --mach-bg-sl: 40% 26%;
487
+ --mach-fg-sl: 65% 80%;
488
+ --mach-border-sl: 40% 40%;
489
+ }
490
+
491
+ /* ---- Everforest (dark, medium) — soft green forest base, low-contrast. */
492
+ [data-theme="everforest"] {
493
+ --c-bg: #2d353b;
494
+ --c-bg-elev: #343f44;
495
+ --c-bg-elev-2: #3d484d;
496
+ --c-surface: #343f44;
497
+ --c-border: #475258;
498
+ --c-border-strong: #4f585e;
499
+ --c-text: #d3c6aa;
500
+ --c-text-muted: #a7b0a3;
501
+ --c-text-faint: #859289;
502
+ --c-accent: #a7c080;
503
+ --c-accent-ink: #1c2226;
504
+ --c-accent-dim: #5c6f49;
505
+ --c-blue: #7fbbb3;
506
+ --c-amber: #dbbc7f;
507
+ --c-red: #e67e80;
508
+ --c-green: #a7c080;
509
+ --c-violet: #d699b6;
510
+ --c-gold: #dbbc7f;
511
+ --c-teal: #83c092;
512
+ --shadow-sm: 0 1px 2px rgba(22, 27, 30, 0.4);
513
+ --shadow-md: 0 6px 20px rgba(22, 27, 30, 0.48);
514
+ --shadow-lg: 0 12px 40px rgba(22, 27, 30, 0.56);
515
+ --mach-bg-sl: 30% 28%;
516
+ --mach-fg-sl: 45% 80%;
517
+ --mach-border-sl: 30% 42%;
518
+ }
519
+
520
+ /* ---- Monokai — the vivid classic: lime, magenta, orange and cyan on slate. */
521
+ [data-theme="monokai"] {
522
+ --c-bg: #272822;
523
+ --c-bg-elev: #2f302a;
524
+ --c-bg-elev-2: #383830;
525
+ --c-surface: #2f302a;
526
+ --c-border: #49483e;
527
+ --c-border-strong: #5b5a4e;
528
+ --c-text: #f8f8f2;
529
+ --c-text-muted: #cfcfc2;
530
+ --c-text-faint: #75715e;
531
+ --c-accent: #a6e22e;
532
+ --c-accent-ink: #14160f;
533
+ --c-accent-dim: #5f7d18;
534
+ --c-blue: #66d9ef;
535
+ --c-amber: #fd971f;
536
+ --c-red: #f92672;
537
+ --c-green: #a6e22e;
538
+ --c-violet: #ae81ff;
539
+ --c-gold: #e6db74;
540
+ --c-teal: #66d9ef;
541
+ --shadow-sm: 0 1px 2px rgba(18, 19, 14, 0.42);
542
+ --shadow-md: 0 6px 20px rgba(18, 19, 14, 0.5);
543
+ --shadow-lg: 0 12px 40px rgba(18, 19, 14, 0.58);
544
+ --mach-bg-sl: 55% 22%;
545
+ --mach-fg-sl: 80% 80%;
546
+ --mach-border-sl: 55% 38%;
547
+ }
548
+
549
+ /* ---- AMOLED (CCT-309) — pure-black background, high-contrast white text for
550
+ maximum contrast / OLED power saving. Elevated surfaces step up in tiny
551
+ increments off true black (so cards/popovers/composer/code blocks stay
552
+ distinguishable without diluting the black page), borders are bright enough
553
+ to read as crisp dividers, and accents are vivid for legibility on black. */
554
+ [data-theme="amoled"] {
555
+ --c-bg: #000000;
556
+ --c-bg-elev: #0a0a0a;
557
+ --c-bg-elev-2: #141414;
558
+ --c-surface: #0d0d0d;
559
+ --c-border: #787878;
560
+ --c-border-strong: #c8c8c8;
561
+ --c-text: #ffffff;
562
+ --c-text-muted: #c8c8c8;
563
+ --c-text-faint: #8e8e8e;
564
+ --c-accent: #34e89e;
565
+ --c-accent-ink: #000000;
566
+ --c-accent-dim: #1f8f62;
567
+ --c-blue: #4db5ff;
568
+ --c-amber: #ffc24b;
569
+ --c-red: #ff6b6b;
570
+ --c-green: #34e89e;
571
+ --c-violet: #c79bff;
572
+ --c-gold: #ffc24b;
573
+ --c-teal: #45e0d8;
574
+ --md-code-bg: color-mix(in srgb, var(--c-blue) 18%, #000000);
575
+ --shadow-sm: 0 1px 2px rgba(0, 0, 0, 0.8);
576
+ --shadow-md: 0 6px 20px rgba(0, 0, 0, 0.85);
577
+ --shadow-lg: 0 12px 40px rgba(0, 0, 0, 0.9);
578
+ /* high-chroma tint to read sharply against pure black */
579
+ --mach-bg-sl: 60% 18%;
580
+ --mach-fg-sl: 90% 84%;
581
+ --mach-border-sl: 60% 42%;
582
+ }
583
+
584
+ /* ---- High Contrast Light (CCT-309) — pure-white background, near-black text.
585
+ The light-mode counterpart to AMOLED: maximum contrast for bright
586
+ environments. Strong borders and dark, saturated accents keep every surface
587
+ legible on white. */
588
+ [data-theme="highcontrast"] {
589
+ color-scheme: light;
590
+ --c-bg: #ffffff;
591
+ --c-bg-elev: #ffffff;
592
+ --c-bg-elev-2: #f0f0f0;
593
+ --c-surface: #ffffff;
594
+ --c-border: #b0b0b0;
595
+ --c-border-strong: #707070;
596
+ --c-text: #000000;
597
+ --c-text-muted: #2e2e2e;
598
+ --c-text-faint: #555555;
599
+ --c-accent: #0a7d4f;
600
+ --c-accent-ink: #ffffff;
601
+ --c-accent-dim: #6fbf9a;
602
+ --c-blue: #0b5fc4;
603
+ --c-amber: #9a6500;
604
+ --c-red: #c4150c;
605
+ --c-green: #0a7d4f;
606
+ --c-violet: #6d28d9;
607
+ --c-gold: #9a6500;
608
+ --c-teal: #0a7d7d;
609
+ --md-code-bg: color-mix(in srgb, var(--c-blue) 10%, #ffffff);
610
+ --shadow-sm: 0 1px 2px rgba(0, 0, 0, 0.18);
611
+ --shadow-md: 0 6px 20px rgba(0, 0, 0, 0.22);
612
+ --shadow-lg: 0 12px 40px rgba(0, 0, 0, 0.28);
613
+ /* pale tint + dark ink so badges stay legible on white */
614
+ --mach-bg-sl: 65% 88%;
615
+ --mach-fg-sl: 60% 26%;
616
+ --mach-border-sl: 50% 64%;
617
+ }
618
+
619
+ /* ---- Color-blind safe (Okabe-Ito) — the recognized qualitative palette that
620
+ stays distinguishable across protanopia, deuteranopia and tritanopia. Status
621
+ colors are mapped so ok/warn/danger/info never collide for any common CVD:
622
+ ok = bluish-green, warn = orange, danger = vermillion, info = sky-blue,
623
+ accent = blue. Pair with the text labels / icons the components already carry
624
+ so meaning is never conveyed by hue alone. Dark base for low glare. */
625
+ [data-theme="colorblind"] {
626
+ --c-bg: #16181d;
627
+ --c-bg-elev: #1d2027;
628
+ --c-bg-elev-2: #262a33;
629
+ --c-surface: #1d2027;
630
+ --c-border: #353a44;
631
+ --c-border-strong: #4a505c;
632
+ --c-text: #f2f4f8;
633
+ --c-text-muted: #b6bdc9;
634
+ --c-text-faint: #828b99;
635
+ --c-accent: #56b4e9; /* sky blue — distinct from the green "ok" */
636
+ --c-accent-ink: #06131c;
637
+ --c-accent-dim: #2c6f93;
638
+ --c-blue: #56b4e9; /* sky blue */
639
+ --c-amber: #e69f00; /* orange */
640
+ --c-red: #d55e00; /* vermillion */
641
+ --c-green: #009e73; /* bluish green */
642
+ --c-violet: #cc79a7; /* reddish purple */
643
+ --c-gold: #f0e442; /* yellow */
644
+ --c-teal: #009e73;
645
+ --shadow-sm: 0 1px 2px rgba(0, 0, 0, 0.45);
646
+ --shadow-md: 0 6px 20px rgba(0, 0, 0, 0.5);
647
+ --shadow-lg: 0 12px 40px rgba(0, 0, 0, 0.6);
648
+ --mach-bg-sl: 45% 24%;
649
+ --mach-fg-sl: 70% 82%;
650
+ --mach-border-sl: 45% 40%;
651
+ }