@tenphi/glaze 0.0.0-snapshot.78261ef → 0.0.0-snapshot.7dca259

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/docs/api.md ADDED
@@ -0,0 +1,1215 @@
1
+ # API Reference
2
+
3
+ Full reference for every public method, option, and type exported by `@tenphi/glaze`. Organized for lookup, not for reading top-to-bottom — see [methodology.md](methodology.md) for a guided walkthrough of how to use these primitives to build a real palette.
4
+
5
+ ## Contents
6
+
7
+ - [Theme creation](#theme-creation)
8
+ - [Theme methods](#theme-methods)
9
+ - [Color definitions](#color-definitions)
10
+ - [Standalone color tokens](#standalone-color-tokens)
11
+ - [Shadows](#shadows)
12
+ - [Mix colors](#mix-colors)
13
+ - [Palette](#palette)
14
+ - [Output formats](#output-formats)
15
+ - [Adaptation modes](#adaptation-modes)
16
+ - [Light / dark scheme mapping](#light--dark-scheme-mapping)
17
+ - [Configuration](#configuration)
18
+ - [Output modes](#output-modes)
19
+ - [Validation](#validation)
20
+ - [Color math utilities](#color-math-utilities)
21
+
22
+ ---
23
+
24
+ ## Theme creation
25
+
26
+ | Method | Description |
27
+ |---|---|
28
+ | `glaze(hue, saturation?, config?)` | Create a theme from hue (0–360) and saturation (0–100). Optional `config` overrides the global config for this theme. |
29
+ | `glaze({ hue, saturation }, config?)` | Create a theme from an options object, with optional per-theme config override. |
30
+ | `glaze.from(data)` | Create a theme from an exported configuration (`theme.export()` snapshot). |
31
+ | `glaze.fromHex(hex)` | Create a theme from a hex color (`#rgb` or `#rrggbb`). Extracts hue and saturation. |
32
+ | `glaze.fromRgb(r, g, b)` | Create a theme from RGB values (0–255). Extracts hue and saturation. |
33
+
34
+ ```ts
35
+ const a = glaze(280, 80);
36
+ const b = glaze({ hue: 280, saturation: 80 });
37
+ const c = glaze.fromHex('#7a4dbf');
38
+ const d = glaze.fromRgb(122, 77, 191);
39
+ const e = glaze.from(a.export());
40
+
41
+ // Per-theme config override:
42
+ const rawTheme = glaze(280, 80, { lightTone: false, darkTone: false });
43
+ ```
44
+
45
+ The optional `config` parameter is a `GlazeConfigOverride` — see [Per-instance config override](#per-instance-config-override).
46
+
47
+ ---
48
+
49
+ ## Theme methods
50
+
51
+ A `GlazeTheme` exposes:
52
+
53
+ | Method | Description |
54
+ |---|---|
55
+ | `theme.hue` (readonly) | The hue seed (0–360). |
56
+ | `theme.saturation` (readonly) | The saturation seed (0–100). |
57
+ | `theme.colors(defs)` | Add/replace colors (additive merge — adds new, overwrites existing by name, doesn't remove others). |
58
+ | `theme.color(name)` | Get a color definition by name. |
59
+ | `theme.color(name, def)` | Set a single color definition. |
60
+ | `theme.remove(name \| names[])` | Remove one or more color definitions. |
61
+ | `theme.has(name)` | Check if a color is defined. |
62
+ | `theme.list()` | List all defined color names. |
63
+ | `theme.reset()` | Clear all color definitions. |
64
+ | `theme.export()` | Export the theme configuration as a JSON-safe object. |
65
+ | `theme.extend(options)` | Create a child theme inheriting all color definitions (see [`extend`](#themeextendoptions) below). |
66
+ | `theme.resolve()` | Resolve all colors and return a `Map<string, ResolvedColor>`. |
67
+ | `theme.tokens(options?)` | Export as a flat token map grouped by scheme variant. |
68
+ | `theme.tasty(options?)` | Export as Tasty style-to-state bindings. |
69
+ | `theme.json(options?)` | Export as plain JSON. |
70
+ | `theme.css(options?)` | Export as CSS custom property declarations. |
71
+
72
+ ### `theme.colors(defs)`
73
+
74
+ ```ts
75
+ theme.colors({ surface: { tone: 97 } });
76
+ theme.colors({ text: { tone: 30 } });
77
+ // Both 'surface' and 'text' are now defined.
78
+ ```
79
+
80
+ ### `theme.color(name) / theme.color(name, def)`
81
+
82
+ ```ts
83
+ theme.color('surface', { tone: 97, saturation: 0.75 }); // set
84
+ const def = theme.color('surface'); // get
85
+ ```
86
+
87
+ ### `theme.extend(options)`
88
+
89
+ Creates a new theme inheriting all color definitions, optionally replacing the hue / saturation seed, color overrides, and config:
90
+
91
+ ```ts
92
+ const danger = primary.extend({
93
+ hue: 23,
94
+ colors: { 'accent-fill': { tone: 48, mode: 'fixed' } },
95
+ });
96
+
97
+ // Inherit parent's config override and widen the dark window further:
98
+ const highSat = base.extend({ config: { darkTone: [10, 100] } });
99
+ ```
100
+
101
+ `GlazeExtendOptions`:
102
+
103
+ | Field | Type | Description |
104
+ |---|---|---|
105
+ | `hue` | `number` | Replace the hue seed. Defaults to the parent's hue. |
106
+ | `saturation` | `number` | Replace the saturation seed. Defaults to the parent's saturation. |
107
+ | `colors` | `ColorMap` | Per-theme overrides (additive merge over the inherited map). |
108
+ | `config` | `GlazeConfigOverride` | Config override for the child. Shallow-merged with the parent's override — child fields win. |
109
+
110
+ Colors marked with `inherit: false` on the parent are **not** copied into the child.
111
+
112
+ ### `theme.tokens(options?)`
113
+
114
+ Flat token map grouped by scheme variant.
115
+
116
+ ```ts
117
+ theme.tokens()
118
+ // → { light: { surface: 'okhsl(...)' }, dark: { surface: 'okhsl(...)' } }
119
+ ```
120
+
121
+ `GlazeJsonOptions`:
122
+
123
+ | Option | Default | Description |
124
+ |---|---|---|
125
+ | `format` | `'okhsl'` | Output color format. One of `'okhsl' \| 'rgb' \| 'hsl' \| 'oklch'`. |
126
+ | `modes` | `{ dark: true, highContrast: false }` (or global config) | Which scheme variants to include. |
127
+
128
+ ### `theme.tasty(options?)`
129
+
130
+ Tasty style-to-state bindings for the [Tasty style system](https://tasty.style/docs). Uses `#name` color token keys and state aliases (`''`, `@dark`, etc.).
131
+
132
+ ```ts
133
+ theme.tasty()
134
+ // → {
135
+ // '#surface': { '': 'okhsl(...)', '@dark': 'okhsl(...)' },
136
+ // ...
137
+ // }
138
+ ```
139
+
140
+ `GlazeTokenOptions`:
141
+
142
+ | Option | Default | Description |
143
+ |---|---|---|
144
+ | `format` | `'okhsl'` | Output color format. |
145
+ | `modes` | global config | Which scheme variants to include. |
146
+ | `states.dark` | `'@dark'` (or global config) | State alias for dark mode tokens. |
147
+ | `states.highContrast` | `'@high-contrast'` (or global config) | State alias for high-contrast tokens. |
148
+ | `prefix` | (palette only) | See [Palette](#palette). |
149
+
150
+ When both `dark` and `highContrast` modes are enabled, dark high-contrast variants are emitted under the combined key `<dark> & <highContrast>` (e.g. `'@dark & @high-contrast'`).
151
+
152
+ ### `theme.json(options?)`
153
+
154
+ Per-color JSON map.
155
+
156
+ ```ts
157
+ theme.json()
158
+ // → {
159
+ // surface: { light: 'okhsl(...)', dark: 'okhsl(...)' },
160
+ // text: { light: 'okhsl(...)', dark: 'okhsl(...)' },
161
+ // }
162
+ ```
163
+
164
+ Same options as `tokens()`.
165
+
166
+ ### `theme.css(options?)`
167
+
168
+ CSS custom property declaration strings, grouped by scheme variant.
169
+
170
+ ```ts
171
+ theme.css();
172
+ // → {
173
+ // light: '--surface-color: rgb(...);\n--text-color: rgb(...);',
174
+ // dark: '--surface-color: rgb(...);\n--text-color: rgb(...);',
175
+ // lightContrast: '...',
176
+ // darkContrast: '...',
177
+ // }
178
+ ```
179
+
180
+ `GlazeCssOptions`:
181
+
182
+ | Option | Default | Description |
183
+ |---|---|---|
184
+ | `format` | `'rgb'` | Output color format. |
185
+ | `suffix` | `'-color'` | Suffix appended to each CSS property name. Pass `''` for bare property names. |
186
+
187
+ `GlazeCssResult` always contains all four keys (`light`, `dark`, `lightContrast`, `darkContrast`); empty if no colors are defined for that variant.
188
+
189
+ ### `theme.export()`
190
+
191
+ ```ts
192
+ const snapshot = theme.export();
193
+ // → { hue: 280, saturation: 80, colors: { surface: { ... }, ... } }
194
+
195
+ const restored = glaze.from(snapshot);
196
+ ```
197
+
198
+ The export contains only the configuration — not resolved color values. Resolved values are recomputed on demand.
199
+
200
+ ---
201
+
202
+ ## Color definitions
203
+
204
+ `ColorDef` is a discriminated union:
205
+
206
+ ```ts
207
+ type ColorDef = RegularColorDef | ShadowColorDef | MixColorDef;
208
+ ```
209
+
210
+ ### `RegularColorDef`
211
+
212
+ | Field | Type | Description |
213
+ |---|---|---|
214
+ | `tone` | `HCPair<ToneValue>` | Number = absolute (0–100, contrast-uniform). `'+N'`/`'-N'` = relative to base's tone (requires `base`). `'max'`/`'min'` = forced to the scheme's tone extreme (no `base`). Optional HC pair `[normal, hc]`. |
215
+ | `saturation` | `number` | Saturation factor applied to the seed saturation (0–1). Default: `1`. |
216
+ | `hue` | `number \| RelativeValue` | Number = absolute (0–360). String (`'+N'`/`'-N'`) = relative to the **theme seed hue** (never to a base color). |
217
+ | `base` | `string` | Name of another color in the same theme — makes this a *dependent* color. |
218
+ | `contrast` | `HCPair<ContrastSpec>` | Contrast floor against `base`. Requires `base`. See [`contrast`](#contrast-floor). |
219
+ | `mode` | `'auto' \| 'fixed' \| 'static'` | Adaptation mode. Default: `'auto'`. See [Adaptation modes](#adaptation-modes). |
220
+ | `flip` | `boolean` | Flip out-of-bounds results (relative `tone` overshoot / unmet `contrast`) to the opposite side instead of clamping. Default: the global `autoFlip` (`true`). See [`flip`](#flip). |
221
+ | `opacity` | `number` | Fixed alpha 0–1. Output includes alpha in the CSS value. Combining with `contrast` is not recommended (a `console.warn` is emitted). |
222
+ | `inherit` | `boolean` | Whether this color is inherited by child themes via `extend()`. Default: `true`. Set to `false` to make the color local to the current theme. |
223
+
224
+ #### Tone values
225
+
226
+ `tone` (0–100) replaces OKHSL lightness with a contrast-uniform axis — equal tone steps give equal WCAG contrast. See [`docs/okhst.md`](okhst.md) for the math. (To port old `lightness` values, see [migration.md](migration.md).)
227
+
228
+ | Form | Example | Meaning |
229
+ |---|---|---|
230
+ | Number (absolute) | `tone: 45` | Absolute tone 0–100. |
231
+ | String (relative) | `tone: '-52'` | Relative to base color's tone (requires `base`). |
232
+ | Extreme | `tone: 'max'` / `'min'` | Force to the scheme's highest (`'max'` = 100) or lowest (`'min'` = 0) tone. No `base` needed. |
233
+ | HC pair | `tone: ['-7', '-20']` | `[normal, high-contrast]`. A single value applies to both. |
234
+
235
+ **Absolute tone** on a dependent color (`base` set) positions the color independently. In dark mode it is tone-mapped (inverted + windowed) on its own. The `contrast` solver acts as a safety net.
236
+
237
+ **Relative tone** applies a signed delta to the base color's resolved tone. Because tone is contrast-uniform, a fixed delta yields a fixed contrast step. In dark mode with `mode: 'auto'`, the offset is anchored to the base's per-scheme tone. If `base + delta` falls outside `[0, 100]`, the result is clamped to the boundary, or — with `flip` (default on) — mirrored to the other side of the base.
238
+
239
+ **Extreme tone** (`'max'` / `'min'`) forces the color to the scheme's tone extreme without a contrast hack or a magic number. `'max'` resolves to author tone 100 and `'min'` to 0; both flow through scheme mapping like an absolute tone, so under `mode: 'auto'` they invert in dark (`'max'` is lightest in light, darkest in dark). Use `mode: 'static'` to pin the same extreme across schemes, or `mode: 'fixed'` to keep the same end without inverting. No `base` required.
240
+
241
+ A dependent color with `base` but no `tone` inherits the base's tone (equivalent to a delta of 0).
242
+
243
+ #### `flip`
244
+
245
+ `flip` governs what happens when a result would fall outside its valid range:
246
+
247
+ - **Relative `tone` overshoot:** when `base ± delta` exceeds `[0, 100]`, `flip` mirrors the delta to the other side of the base (e.g. `'+30'` becomes `'-30'`) instead of clamping to the boundary.
248
+ - **`contrast` direction:** when the requested tone direction can't meet the floor, `flip` lets the solver try the opposite side (the same behavior as the global `autoFlip`).
249
+
250
+ `flip` defaults to the global `autoFlip` (`true`). Set `flip: false` on a color to clamp instead of mirror — useful when you want a relative offset to stay on the authored side of the base, or to keep an unmet contrast pinned to one direction's extreme.
251
+
252
+ #### `contrast` (floor)
253
+
254
+ ```ts
255
+ type ContrastPreset = 'AA' | 'AAA' | 'AA-large' | 'AAA-large';
256
+ type ContrastSpec =
257
+ | number // bare WCAG ratio
258
+ | ContrastPreset // named WCAG preset
259
+ | { wcag: HCPair<number | ContrastPreset> }
260
+ | { apca: HCPair<number> }; // APCA Lc target
261
+ ```
262
+
263
+ | Preset | WCAG ratio |
264
+ |---|---|
265
+ | `'AA-large'` | 3 |
266
+ | `'AA'` | 4.5 |
267
+ | `'AAA-large'` | 4.5 |
268
+ | `'AAA'` | 7 |
269
+
270
+ A bare number or preset means **WCAG**. Use `{ wcag }` / `{ apca }` to pick the metric explicitly. The `[normal, highContrast]` pair may live at the outer level (`[4.5, 7]`, `[{ wcag: 4.5 }, { wcag: 7 }]`) or inside the metric (`{ wcag: [4.5, 7] }`, `{ apca: [45, 60] }`).
271
+
272
+ ```ts
273
+ contrast: 4.5 // WCAG 4.5
274
+ contrast: 'AAA' // WCAG 7
275
+ contrast: { wcag: 6 } // WCAG 6
276
+ contrast: { wcag: [4.5, 7] } // WCAG 4.5 normal / 7 high-contrast
277
+ contrast: { apca: 60 } // APCA Lc 60
278
+ contrast: { apca: [45, 60] } // APCA Lc 45 normal / 60 high-contrast
279
+ ```
280
+
281
+ The floor is applied independently per scheme — if the `tone` already satisfies it the tone is kept, otherwise the solver searches in tone (contrast-uniform → a closed-form WCAG seed and fast convergence) until the target is met.
282
+
283
+ By default, the solver crosses to the opposite side of the base color when the requested tone direction cannot satisfy the floor. This is controlled per-color by [`flip`](#flip) (which defaults to the global `autoFlip`). Set `glaze.configure({ autoFlip: false })` — or `flip: false` on a single color — to keep strict directionality: unmet colors pin to that direction's 0 or 100 tone extreme instead of falling back to the original requested value.
284
+
285
+ **Full tone spectrum in HC mode:** in high-contrast variants the `lightTone` and `darkTone` window constraints are bypassed entirely (the window is forced to `[0, 100]`). Colors can reach the full range, maximizing perceivable contrast.
286
+
287
+ **Chromatic drift (verification):** tone is contrast-uniform for grays. A chromatic swatch at a given tone shares its OKHSL lightness with the equivalent gray but drifts in real luminance, so a contrast-floored color may land slightly under its gray-tone expectation. Glaze measures the resolved result against the base and emits a deduped advisory `console.warn` when it drifts below the target. See [`docs/okhst.md`](okhst.md) §Verification.
288
+
289
+ #### Per-color hue override
290
+
291
+ ```ts
292
+ const theme = glaze(280, 80);
293
+ theme.colors({
294
+ surface: { tone: 97 },
295
+ gradientEnd: { tone: 90, hue: '+20' }, // 280 + 20 = 300
296
+ warning: { tone: 60, hue: 40 }, // absolute
297
+ });
298
+ ```
299
+
300
+ Relative hue is always relative to the **theme seed hue**, not to a base color.
301
+
302
+ ### `ShadowColorDef`
303
+
304
+ | Field | Type | Description |
305
+ |---|---|---|
306
+ | `type` | `'shadow'` | Discriminator. |
307
+ | `bg` | `string` | Background color name — must reference a non-shadow color in the same theme. |
308
+ | `fg` | `string` | Optional foreground color name for tinting and intensity modulation. Must reference a non-shadow color. Omit for an achromatic shadow at full user-specified intensity. |
309
+ | `intensity` | `HCPair<number>` | Shadow intensity, 0–100. Supports HC pairs. |
310
+ | `tuning` | `ShadowTuning` | Per-color tuning overrides. Merged field-by-field with the global `shadowTuning`. |
311
+ | `inherit` | `boolean` | Inheritance flag, default `true`. |
312
+
313
+ See [Shadows](#shadows) below for the algorithm and tuning details.
314
+
315
+ ### `MixColorDef`
316
+
317
+ | Field | Type | Description |
318
+ |---|---|---|
319
+ | `type` | `'mix'` | Discriminator. |
320
+ | `base` | `string` | "From" color name. |
321
+ | `target` | `string` | "To" color name. |
322
+ | `value` | `HCPair<number>` | Mix ratio 0–100 (0 = pure base, 100 = pure target). In `'transparent'` blend, this becomes the target's opacity. Supports HC pairs. |
323
+ | `blend` | `'opaque' \| 'transparent'` | Default `'opaque'`. |
324
+ | `space` | `'okhsl' \| 'srgb'` | Interpolation space for opaque blending. Default `'okhsl'`. Ignored for `'transparent'` (always composites in linear sRGB). |
325
+ | `contrast` | `HCPair<ContrastSpec>` | Optional contrast floor against `base` (WCAG or APCA — see [`contrast`](#contrast-floor)). The solver adjusts the mix ratio (opaque) or opacity (transparent). |
326
+ | `inherit` | `boolean` | Inheritance flag, default `true`. |
327
+
328
+ See [Mix colors](#mix-colors) below.
329
+
330
+ ---
331
+
332
+ ## Standalone color tokens
333
+
334
+ `glaze.color()` creates a single color token without a full theme.
335
+
336
+ ```ts
337
+ // arg1: the color (four shapes — see below)
338
+ // arg2: optional config override (GlazeConfigOverride — see below)
339
+ glaze.color(color: GlazeFromInput | GlazeColorInput | GlazeColorValue, config?: GlazeConfigOverride): GlazeColorToken;
340
+ ```
341
+
342
+ ### Input forms
343
+
344
+ `glaze.color()` accepts **four input shapes**, discriminated by structure:
345
+
346
+ | Shape | Example | Notes |
347
+ |---|---|---|
348
+ | **Bare string** | `'#26fcb2'` | Hex or CSS color function (`rgb()`, `hsl()`, `okhsl()`, `okhst()`, `oklch()`). |
349
+ | **Value object** | `{ h: 152, s: 0.95, l: 0.74 }` | OKHSL, OKHST (`{ h, s, t }`), `{ r, g, b }` (sRGB 0–255), or `{ l, c, h }` (OKLCh). |
350
+ | **`{ from, ...overrides }`** | `{ from: '#1a1a2e', base: bg, contrast: 'AA' }` | Value + color overrides in one object. |
351
+ | **Structured** | `{ hue: 152, saturation: 95, tone: 74 }` | Full theme-style token (hue/saturation in 0–100, tone in 0–100). |
352
+
353
+ `GlazeColorValue` (bare string or value-object forms) accepts:
354
+
355
+ | Form | Example | Notes |
356
+ |---|---|---|
357
+ | Hex | `'#26fcb2'`, `'#26fcb2ff'`, `'#abc'` | 3, 6, or 8 digits. Alpha is dropped with a `console.warn` — use `opacity` instead. |
358
+ | `rgb()` | `'rgb(38 252 178)'`, `'rgb(38 252 178 / 0.8)'` | Modern space syntax. Alpha dropped with warning. |
359
+ | `hsl()` | `'hsl(152 97% 57%)'` | Modern space syntax. Alpha dropped with warning. |
360
+ | `okhsl()` | `'okhsl(152 95% 74%)'` | Glaze's own emit format. Alpha dropped with warning. |
361
+ | `okhst()` | `'okhst(152 95% 70%)'` | OKHST tone input (third value is tone 0–100). **Input only** — never emitted. Alpha dropped with warning. |
362
+ | `oklch()` | `'oklch(0.85 0.18 152)'` | Glaze's own emit format. Alpha dropped with warning. |
363
+ | `OkhslColor` object | `{ h: 152, s: 0.95, l: 0.74 }` | OKHSL shape (h: 0–360, s/l: 0–1). Passing 0–100 for `s`/`l` throws with a hint to use the structured form. |
364
+ | `OkhstColor` object | `{ h: 152, s: 0.95, t: 0.70 }` | OKHST shape (h: 0–360, s/t: 0–1). The `t` key disambiguates it from `{ h, s, l }`. **Input only.** |
365
+ | `RgbColor` object | `{ r: 38, g: 252, b: 178 }` | sRGB 0–255. RGB tuple `[r, g, b]` is not supported — use this object form. |
366
+ | `OklchColor` object | `{ l: 0.85, c: 0.18, h: 152 }` | OKLCh (L/C: 0–1, H: degrees), same semantics as `oklch()` strings. |
367
+
368
+ `GlazeColorInput` (structured form) is `{ hue, saturation, tone, ... }`:
369
+
370
+ | Field | Type | Description |
371
+ |---|---|---|
372
+ | `hue` | `number` | 0–360. |
373
+ | `saturation` | `number` | 0–100. |
374
+ | `tone` | `HCPair<number \| ExtremeValue>` | 0–100 (contrast-uniform) or `'max'`/`'min'`, optional HC pair. |
375
+ | `saturationFactor` | `number` | Multiplier on the seed (0–1). Default: `1`. |
376
+ | `mode` | `AdaptationMode` | Default: `'auto'`. |
377
+ | `flip` | `boolean` | Flip out-of-bounds results instead of clamping. Default: global `autoFlip`. |
378
+ | `opacity` | `number` | Fixed alpha 0–1. |
379
+ | `base` | `GlazeColorToken \| GlazeColorValue` | Optional dependency. See [Pairing colors](#pairing-colors). |
380
+ | `contrast` | `HCPair<ContrastSpec>` | Contrast floor against `base` (WCAG or APCA). Without `base`, anchored to the literal seed. |
381
+ | `name` | `string` | Debug label for warnings; doesn't change output keys. Reserved names (`'value'`, `'seed'`, `'externalBase'`) are rejected. |
382
+
383
+ `GlazeFromInput` (from form) is `{ from: GlazeColorValue, ...colorOverrides }`:
384
+
385
+ | Field | Notes |
386
+ |---|---|
387
+ | `from` | **Required.** The source color value — same forms as `GlazeColorValue`. |
388
+ | `hue` | Number (absolute 0–360) or `'+N'`/`'-N'` (relative to seed, never to `base`). |
389
+ | `saturation` | Override seed saturation (0–100). |
390
+ | `tone` | Number (absolute 0–100), `'+N'`/`'-N'`, or `'max'`/`'min'`. Without `base`, relative anchors to the seed; with `base`, anchors to `base`'s tone per scheme. |
391
+ | `saturationFactor` | Multiplier on the seed (0–1). |
392
+ | `mode` | `'auto'` (default) / `'fixed'` / `'static'`. |
393
+ | `flip` | Flip out-of-bounds results instead of clamping. Default: global `autoFlip`. |
394
+ | `contrast` | Contrast floor (WCAG or APCA). Without `base`, anchored to the literal seed; with `base`, solved per scheme. |
395
+ | `base` | `GlazeColorToken` or raw `GlazeColorValue`. See [Pairing colors](#pairing-colors). |
396
+ | `opacity` | Fixed alpha 0–1. Combining with `contrast` is not recommended — `console.warn` is emitted. |
397
+ | `name` | Debug label only — surfaces in warnings/errors. Does not change output keys. |
398
+
399
+ Named CSS colors (`'red'`, `'blueviolet'`) are not supported.
400
+
401
+ ### Defaults
402
+
403
+ Every input form defaults to `mode: 'auto'` so the resolved token adapts between light and dark like an ordinary theme color. The config snapshot taken at create time differs by input form:
404
+
405
+ - **Value-shorthand** (bare strings, value objects, and `{ from, ...overrides }`):
406
+ - Light variant preserves the input tone exactly (`lightTone: false`).
407
+ - All other config fields (`darkTone`, `darkDesaturation`, `saturationTaper`, `autoFlip`) snapshot from `globalConfig` at create time.
408
+ - **Structured input** (`{ hue, saturation, tone, ... }`):
409
+ - Both tone windows snapshot from `globalConfig` at create time (same as a theme color).
410
+ - All fields are **snapshotted at color-creation time** — later `glaze.configure()` calls don't retroactively change existing tokens.
411
+
412
+ ```ts
413
+ // Bare string — adapts automatically
414
+ glaze.color('#26fcb2')
415
+
416
+ // Value-object — same behavior
417
+ glaze.color({ h: 152, s: 0.95, l: 0.74 })
418
+
419
+ // OKHST value-object — tone axis
420
+ glaze.color({ h: 152, s: 0.95, t: 0.70 })
421
+
422
+ // From form — value + color overrides
423
+ glaze.color({ from: '#1a1a2e', hue: '+20', contrast: 'AA' })
424
+
425
+ // Structured form — explicit hue/saturation/tone (0–100)
426
+ glaze.color({ hue: 152, saturation: 95, tone: 74 })
427
+ ```
428
+
429
+ ### Token methods
430
+
431
+ A `GlazeColorToken` exposes:
432
+
433
+ | Method | Description |
434
+ |---|---|
435
+ | `token.resolve()` | Resolve as a `ResolvedColor` (light/dark/lightContrast/darkContrast variants). |
436
+ | `token.token(options?)` | Flat token map (no color-name key). Options: `format`, `modes`, `states`. |
437
+ | `token.tasty(options?)` | Tasty state map (no color-name key). Same options as `token.token`. |
438
+ | `token.json(options?)` | JSON map (no color-name key). Options: `format`, `modes`. |
439
+ | `token.css({ name, format?, suffix? })` | CSS custom property declarations grouped by scheme variant. `name` is **required** and becomes the variable identifier (`'brand'` → `--brand-color`). Defaults: `format: 'rgb'`, `suffix: '-color'` (matches `theme.css`). |
440
+ | `token.export()` | JSON-safe snapshot — pass to `glaze.colorFrom(...)` to rehydrate. |
441
+
442
+ ### Per-instance config override
443
+
444
+ The optional `config` second argument (`GlazeConfigOverride`) overrides the resolve-relevant global config fields for a single token or theme. Fields that are omitted fall through to the live global config at create time (and are snapshotted). A tone window can be `[lo, hi]`, `{ lo, hi, eps }`, or `false` (disable clamping).
445
+
446
+ `GlazeConfigOverride`:
447
+
448
+ | Field | Default (from global) | Description |
449
+ |---|---|---|
450
+ | `lightTone` | `[13, 100]` | Light tone window: `[lo, hi]`, `{ lo, hi, eps }`, or `false` (disable clamping). |
451
+ | `darkTone` | `[10, 95]` | Dark tone window: `[lo, hi]`, `{ lo, hi, eps }`, or `false` (disable clamping). |
452
+ | `darkDesaturation` | `0.1` | Saturation reduction in dark scheme (0–1). |
453
+ | `saturationTaper` | `0.15` | Saturation taper strength toward the tone extremes (0–1). |
454
+ | `autoFlip` | `true` | Default for each color's `flip`: when solving `contrast` (or applying a relative `tone` that overshoots), allow crossing to the opposite side instead of clamping. |
455
+ | `shadowTuning` | `undefined` | Default shadow tuning (meaningful for themes; harmless on color tokens). |
456
+
457
+ Config overrides apply to both `glaze.color()` tokens and `glaze()` themes:
458
+
459
+ ```ts
460
+ // Standalone color — preserve raw tone in both schemes
461
+ glaze.color('#26fcb2', { darkTone: false })
462
+
463
+ // Restore the #000 → white dark flip (full dark range)
464
+ glaze.color('#000000', {
465
+ lightTone: false,
466
+ darkTone: [15, 100],
467
+ })
468
+
469
+ // Structured form with config override
470
+ glaze.color({ hue: 152, saturation: 95, tone: 74 }, { darkTone: false })
471
+
472
+ // Theme with config override
473
+ const rawTheme = glaze(280, 80, { lightTone: false })
474
+ ```
475
+
476
+ The override is **snapshotted at create time** so later `glaze.configure()` calls don't change already-created tokens or themes (for non-overridden fields, the snapshot captured the global value at creation time; for themes, non-overridden fields are re-read from the live global at resolve time — see [Theme config override](#theme-config-override)).
477
+
478
+ ### Theme config override
479
+
480
+ When a theme is created with a `GlazeConfigOverride`, the override is **merged over the live global config at resolve time**. This means:
481
+
482
+ - Fields you overrode are fixed — `glaze.configure()` can't change them for this theme.
483
+ - Fields you didn't override still react to later `glaze.configure()` calls.
484
+
485
+ ```ts
486
+ const t = glaze(280, 80, { lightTone: [0, 50] });
487
+ t.colors({ text: { tone: 50, saturation: 1 } });
488
+ // text.light lands inside the [0, 50] window — always, regardless of
489
+ // global lightTone changes.
490
+ // text.dark.s reacts to glaze.configure({ darkDesaturation }) since it's not overridden.
491
+ ```
492
+
493
+ `extend` inherits the parent's override and shallow-merges the child's:
494
+
495
+ ```ts
496
+ const child = t.extend({ config: { darkTone: false } });
497
+ // child: lightTone { lo: 0, hi: 50 } (inherited) + darkTone: false (added)
498
+ ```
499
+
500
+ `theme.export()` includes `config`; `glaze.from(data)` restores it.
501
+
502
+ ### `glaze.colorFrom(data)`
503
+
504
+ Inverse of `token.export()`. The exported snapshot includes the original input, all overrides (with any `base` token recursively serialized), and the full effective config — so later `glaze.configure()` calls don't change rehydrated tokens.
505
+
506
+ ```ts
507
+ const text = glaze.color({ from: '#1a1a1a', contrast: 'AA' });
508
+ const data = text.export();
509
+ const restored = glaze.colorFrom(data);
510
+ // restored.resolve() === text.resolve() byte-for-byte
511
+ ```
512
+
513
+ Both value-form and structured-form tokens round-trip.
514
+
515
+ ### Pairing colors
516
+
517
+ Set `base` to anchor a standalone color to another standalone color or raw value. The contrast solver and relative `tone` offsets switch their anchor from the literal seed to the base's resolved variant per scheme — so the same text color automatically lands at AA against its background in light, dark, and high-contrast modes.
518
+
519
+ ```ts
520
+ const bg = glaze.color('#1a1a2e');
521
+
522
+ // Text guaranteed AA against `bg` in every scheme.
523
+ const text = glaze.color({ from: '#ffffff', base: bg, contrast: 'AA' });
524
+
525
+ // Border 8 tone units lighter than `bg` in each scheme.
526
+ const border = glaze.color({ from: '#000000',
527
+ base: bg,
528
+ tone: '+8',
529
+ mode: 'fixed',
530
+ });
531
+
532
+ // Raw-value base — Glaze auto-wraps it via `glaze.color(value)`.
533
+ const text2 = glaze.color({ from: '#ffffff', base: '#1a1a2e', contrast: 'AA' });
534
+ ```
535
+
536
+ Behavior with `base`:
537
+
538
+ - `contrast` is solved per scheme against `base`'s resolved variant (light / dark / lightContrast / darkContrast).
539
+ - Relative `tone: '+N'` / `'-N'` is anchored to `base`'s tone per scheme (matches theme behavior).
540
+ - Relative `hue: '+N'` / `'-N'` still anchors to the **seed** (the value passed to `glaze.color()`), not the base.
541
+ - `mode` works as a per-pair knob.
542
+ - The base token's `.resolve()` is called lazily on the first resolve of the dependent and the result is captured by reference; later mutations to the base don't apply.
543
+ - **Structured bases are resolved at full range for linking math**: when a value/`from` color links to a base created via the structured form, the contrast/tone anchor uses the raw input tone (not the windowed output). This ensures the anchor matches what you intended, not what the light window remapped it to. The base's own `.resolve()` output is unaffected.
544
+ - When the contrast target is physically unreachable, `glaze` emits a single `console.warn` per `(name, scheme, target)` triple and returns the closest passing variant. Use the `name` override to make the warning identifiable.
545
+
546
+ Chains compose:
547
+
548
+ ```ts
549
+ const bg = glaze.color('#000000');
550
+ const surface = glaze.color({ from: '#222222', base: bg, contrast: 'AAA' });
551
+ const text = glaze.color({ from: '#ffffff', base: surface, contrast: 'AA' });
552
+ ```
553
+
554
+ ### `name` is a debug label
555
+
556
+ The `name` override appears in `console.warn` / Error messages but **does not** change output keys (`.token()`, `.tasty()`, `.json()`, `.css()` still use `''`, `light`, etc.). The CSS variable name comes from `css({ name })`, not from the override.
557
+
558
+ ---
559
+
560
+ ## Shadows
561
+
562
+ ### Defining shadow colors in a theme
563
+
564
+ ```ts
565
+ theme.colors({
566
+ surface: { tone: 95 },
567
+ text: { base: 'surface', tone: '-52', contrast: 'AAA' },
568
+
569
+ 'shadow-sm': { type: 'shadow', bg: 'surface', fg: 'text', intensity: 5 },
570
+ 'shadow-md': { type: 'shadow', bg: 'surface', fg: 'text', intensity: 10 },
571
+ 'shadow-lg': { type: 'shadow', bg: 'surface', fg: 'text', intensity: 20 },
572
+ });
573
+ ```
574
+
575
+ Shadow colors are included in all output methods (`tokens()`, `tasty()`, `css()`, `json()`) alongside regular colors and emit an alpha component:
576
+
577
+ ```
578
+ 'oklch(0.15 0.009 282 / 0.1)'
579
+ 'rgb(34 28 42 / 0.1)'
580
+ ```
581
+
582
+ ### How shadows work
583
+
584
+ 1. **Contrast weight** — when `fg` is provided, shadow strength scales with `|l_bg − l_fg|`. Dark text on a light background produces a strong shadow; near-background-lightness elements produce barely visible shadows.
585
+ 2. **Pigment color** — hue blended between fg and bg, low saturation, dark lightness.
586
+ 3. **Alpha** — computed via a `tanh` curve that saturates smoothly toward `alphaMax` (default `1.0`), ensuring well-separated shadow levels even on dark backgrounds.
587
+
588
+ Omit `fg` for an achromatic shadow at full user-specified intensity:
589
+
590
+ ```ts
591
+ theme.colors({
592
+ 'drop-shadow': { type: 'shadow', bg: 'surface', intensity: 12 },
593
+ });
594
+ ```
595
+
596
+ `intensity` supports `[normal, highContrast]` pairs:
597
+
598
+ ```ts
599
+ 'shadow-card': { type: 'shadow', bg: 'surface', fg: 'text', intensity: [10, 20] },
600
+ ```
601
+
602
+ ### `ShadowTuning`
603
+
604
+ Fine-tune behavior per-color or globally via `glaze.configure({ shadowTuning })`. Per-color `tuning` is merged field-by-field with the global one.
605
+
606
+ | Parameter | Default | Description |
607
+ |---|---|---|
608
+ | `saturationFactor` | `0.18` | Fraction of fg saturation kept in pigment. |
609
+ | `maxSaturation` | `0.25` | Upper clamp on pigment saturation. |
610
+ | `lightnessFactor` | `0.25` | Multiplier for bg lightness → pigment lightness. |
611
+ | `lightnessBounds` | `[0.05, 0.20]` | Clamp range for pigment lightness. |
612
+ | `minGapTarget` | `0.05` | Target minimum gap between pigment and bg lightness. |
613
+ | `alphaMax` | `1.0` | Asymptotic maximum alpha. |
614
+ | `bgHueBlend` | `0.2` | Blend weight pulling pigment hue toward bg hue. `0` = pure fg hue, `1` = pure bg hue. |
615
+
616
+ ```ts
617
+ theme.colors({
618
+ 'shadow-soft': {
619
+ type: 'shadow', bg: 'surface', intensity: 10,
620
+ tuning: { alphaMax: 0.3, saturationFactor: 0.1 },
621
+ },
622
+ });
623
+
624
+ glaze.configure({
625
+ shadowTuning: { alphaMax: 0.5, bgHueBlend: 0.3 },
626
+ });
627
+ ```
628
+
629
+ ### Standalone shadow computation
630
+
631
+ `glaze.shadow(input)` computes a shadow outside of a theme. `bg` and `fg` accept any `GlazeColorValue`:
632
+
633
+ ```ts
634
+ const v = glaze.shadow({
635
+ bg: '#f0eef5',
636
+ fg: '#1a1a2e',
637
+ intensity: 10,
638
+ });
639
+ // → { h: 280, s: 0.14, l: 0.2, alpha: 0.1 }
640
+
641
+ const css = glaze.format(v, 'oklch');
642
+ // → 'oklch(0.15 0.014 280 / 0.1)'
643
+ ```
644
+
645
+ `GlazeShadowInput`:
646
+
647
+ | Field | Type | Description |
648
+ |---|---|---|
649
+ | `bg` | `GlazeColorValue` | Background. Any `GlazeColorValue` form. Alpha components dropped with warning. |
650
+ | `fg` | `GlazeColorValue` | Optional foreground. Same forms as `bg`. |
651
+ | `intensity` | `number` | 0–100. |
652
+ | `tuning` | `ShadowTuning` | Optional. |
653
+
654
+ ### Fixed opacity (regular colors)
655
+
656
+ For a simple fixed-alpha color (no shadow algorithm), use `opacity` on a regular color:
657
+
658
+ ```ts
659
+ theme.colors({
660
+ overlay: { tone: 0, opacity: 0.5 },
661
+ });
662
+ // → 'oklch(0 0 0 / 0.5)'
663
+ ```
664
+
665
+ ---
666
+
667
+ ## Mix colors
668
+
669
+ ### Opaque mix
670
+
671
+ Produces a solid color by interpolating between `base` and `target`:
672
+
673
+ ```ts
674
+ theme.colors({
675
+ surface: { tone: 95 },
676
+ accent: { tone: 30 },
677
+ tint: { type: 'mix', base: 'surface', target: 'accent', value: 30 },
678
+ });
679
+ ```
680
+
681
+ - `value: 0` = pure base, `value: 100` = pure target.
682
+ - Result has alpha = 1.
683
+ - Adapts to light/dark/HC schemes automatically via the resolved base and target.
684
+
685
+ ### Transparent mix
686
+
687
+ Produces the target color with controlled opacity — useful for hover overlays:
688
+
689
+ ```ts
690
+ theme.colors({
691
+ surface: { tone: 95 },
692
+ black: { tone: 0, saturation: 0 },
693
+ hover: {
694
+ type: 'mix', base: 'surface', target: 'black',
695
+ value: 8, blend: 'transparent',
696
+ },
697
+ });
698
+ // hover → black with alpha = 0.08
699
+ ```
700
+
701
+ The output color has `h`, `s`, `l` from the target and `alpha = value / 100`.
702
+
703
+ ### Blend space (opaque only)
704
+
705
+ | `space` | Behavior | Best for |
706
+ |---|---|---|
707
+ | `'okhsl'` (default) | Perceptually uniform OKHSL interpolation. | Design token derivation. |
708
+ | `'srgb'` | Linear sRGB channel interpolation. | Matching browser compositing of CSS color-mix / overlay. |
709
+
710
+ Transparent blending always composites in linear sRGB (matches browser alpha compositing).
711
+
712
+ ### Contrast solving on mixes
713
+
714
+ Mix colors support the same `contrast` prop as regular colors. The solver adjusts the mix ratio (opaque) or opacity (transparent) to meet the WCAG target:
715
+
716
+ ```ts
717
+ 'tint': {
718
+ type: 'mix', base: 'surface', target: 'accent',
719
+ value: 10, contrast: 'AA',
720
+ },
721
+ 'overlay': {
722
+ type: 'mix', base: 'surface', target: 'accent',
723
+ value: 5, blend: 'transparent', contrast: 3,
724
+ },
725
+ ```
726
+
727
+ Both `value` and `contrast` support `[normal, highContrast]` pairs.
728
+
729
+ ### Achromatic colors
730
+
731
+ When mixing with achromatic colors (saturation near zero, e.g. white or black) in `okhsl` space, the hue comes from whichever color has saturation. Matches CSS `color-mix()` "missing component" behavior. For purely achromatic mixes prefer `space: 'srgb'` where hue is irrelevant.
732
+
733
+ ### Mix chaining
734
+
735
+ Mix colors can reference other mix colors:
736
+
737
+ ```ts
738
+ theme.colors({
739
+ white: { tone: 100, saturation: 0 },
740
+ black: { tone: 0, saturation: 0 },
741
+ gray: { type: 'mix', base: 'white', target: 'black', value: 50, space: 'srgb' },
742
+ lightGray: { type: 'mix', base: 'white', target: 'gray', value: 50, space: 'srgb' },
743
+ });
744
+ ```
745
+
746
+ Mix colors **cannot** reference shadow colors (same restriction as regular dependent colors).
747
+
748
+ ---
749
+
750
+ ## Palette
751
+
752
+ `glaze.palette(themes, options?)` composes multiple themes into a single token namespace.
753
+
754
+ ```ts
755
+ const palette = glaze.palette({ primary, danger, success, warning });
756
+ const palette = glaze.palette(
757
+ { primary, danger, success },
758
+ { primary: 'primary' },
759
+ );
760
+ ```
761
+
762
+ `GlazePaletteOptions`:
763
+
764
+ | Option | Description |
765
+ |---|---|
766
+ | `primary` | Name of the primary theme. The primary's tokens are duplicated **without** prefix in all exports, providing convenient short aliases alongside the prefixed versions. Throws if the name doesn't match any theme. |
767
+
768
+ A `GlazePalette` exposes:
769
+
770
+ | Method | Description |
771
+ |---|---|
772
+ | `palette.tokens(options?)` | Flat token map grouped by scheme variant. |
773
+ | `palette.tasty(options?)` | Tasty style-to-state bindings. |
774
+ | `palette.json(options?)` | Per-theme JSON map (no prefix needed — keyed by theme name). |
775
+ | `palette.css(options?)` | CSS custom property declaration strings. |
776
+
777
+ ### `GlazePaletteExportOptions`
778
+
779
+ Shared by `tokens`, `tasty`, and `css`:
780
+
781
+ | Option | Default | Description |
782
+ |---|---|---|
783
+ | `prefix` | `true` (= `"<themeName>-"`) | `false` disables prefixing. Or pass a custom map: `{ primary: 'brand-', danger: 'error-' }`. |
784
+ | `primary` | inherits from palette creation | `string` to override, `false` to disable for this call. |
785
+
786
+ Each export method also accepts its own format/options shape:
787
+
788
+ | Method | Additional options |
789
+ |---|---|
790
+ | `palette.tokens(options?)` | `format`, `modes` |
791
+ | `palette.tasty(options?)` | `format`, `modes`, `states` |
792
+ | `palette.css(options?)` | `format`, `suffix` |
793
+
794
+ `palette.css()` does not accept `modes`; it always returns all four CSS strings (`light`, `dark`, `lightContrast`, `darkContrast`).
795
+
796
+ ### Prefix behavior
797
+
798
+ By default all palette tokens are prefixed:
799
+
800
+ ```ts
801
+ palette.tokens();
802
+ // → {
803
+ // light: { 'primary-surface': 'okhsl(...)', 'danger-surface': 'okhsl(...)' },
804
+ // dark: { 'primary-surface': 'okhsl(...)', 'danger-surface': 'okhsl(...)' },
805
+ // }
806
+ ```
807
+
808
+ Custom map (any theme not listed falls back to `"<themeName>-"`):
809
+
810
+ ```ts
811
+ palette.tokens({ prefix: { primary: 'brand-', danger: 'error-' } });
812
+ ```
813
+
814
+ Disable prefixing:
815
+
816
+ ```ts
817
+ palette.tokens({ prefix: false });
818
+ ```
819
+
820
+ ### Collision detection
821
+
822
+ When two themes produce the same output key (via `prefix: false`, custom prefix maps, or primary unprefixed aliases), the **first-written value wins** and a `console.warn` is emitted:
823
+
824
+ ```
825
+ glaze: token "surface" from theme "b" collides with theme "a" — skipping.
826
+ ```
827
+
828
+ ### Primary theme aliases
829
+
830
+ The primary theme's tokens are duplicated without prefix:
831
+
832
+ ```ts
833
+ const palette = glaze.palette(
834
+ { primary, danger, success },
835
+ { primary: 'primary' },
836
+ );
837
+ palette.tokens();
838
+ // → {
839
+ // light: {
840
+ // 'primary-surface': 'okhsl(...)',
841
+ // 'danger-surface': 'okhsl(...)',
842
+ // 'success-surface': 'okhsl(...)',
843
+ // 'surface': 'okhsl(...)', // unprefixed alias
844
+ // },
845
+ // }
846
+ ```
847
+
848
+ Override per-export:
849
+
850
+ ```ts
851
+ palette.tokens({ primary: 'danger' });
852
+ palette.tokens({ primary: false });
853
+ ```
854
+
855
+ The primary alias works alongside any prefix mode — when using a custom map, primary tokens are still duplicated without prefix:
856
+
857
+ ```ts
858
+ palette.tokens({ prefix: { primary: 'p-', danger: 'd-' } });
859
+ // → 'p-surface' + 'surface' (alias) + 'd-surface'
860
+ ```
861
+
862
+ ### `palette.json()`
863
+
864
+ JSON export groups by theme name (no prefix needed):
865
+
866
+ ```ts
867
+ palette.json();
868
+ // → {
869
+ // primary: { surface: { light: 'okhsl(...)', dark: 'okhsl(...)' } },
870
+ // danger: { surface: { light: 'okhsl(...)', dark: 'okhsl(...)' } },
871
+ // }
872
+ ```
873
+
874
+ ### `palette.css()`
875
+
876
+ ```ts
877
+ const css = palette.css();
878
+ const stylesheet = `
879
+ :root { ${css.light} }
880
+ @media (prefers-color-scheme: dark) {
881
+ :root { ${css.dark} }
882
+ }
883
+ `;
884
+ ```
885
+
886
+ `palette.css()` accepts the same `GlazeCssOptions` as `theme.css()` plus `GlazePaletteExportOptions`.
887
+ It does not accept `modes`; all four result fields are always returned.
888
+
889
+ ---
890
+
891
+ ## Output formats
892
+
893
+ Control the color format with the `format` option on any export method:
894
+
895
+ | Format | Output (alpha = 1) | Output (alpha < 1) | Notes |
896
+ |---|---|---|---|
897
+ | `'okhsl'` (default for tokens/tasty/json) | `okhsl(H S% L%)` | `okhsl(H S% L% / A)` | Glaze's native format, not a CSS function. |
898
+ | `'rgb'` (default for css) | `rgb(R G B)` | `rgb(R G B / A)` | Rounded integers, modern space syntax. |
899
+ | `'hsl'` | `hsl(H S% L%)` | `hsl(H S% L% / A)` | Modern space syntax. |
900
+ | `'oklch'` | `oklch(L C H)` | `oklch(L C H / A)` | OKLab-based LCH. |
901
+
902
+ ```ts
903
+ theme.tokens(); // 'okhsl(280 60% 97%)'
904
+ theme.tokens({ format: 'rgb' }); // 'rgb(244 240 250)'
905
+ theme.tokens({ format: 'hsl' }); // 'hsl(270.5 45.2% 95.8%)'
906
+ theme.tokens({ format: 'oklch' }); // 'oklch(0.965 0.0123 280)'
907
+ ```
908
+
909
+ All numeric output strips trailing zeros for cleaner CSS (e.g. `95` not `95.0`).
910
+
911
+ The `format` option works on every export: `theme.tokens()`, `theme.tasty()`, `theme.json()`, `theme.css()`, the same on `palette`, and on `token.token()` / `.tasty()` / `.json()` / `.css()`.
912
+
913
+ ---
914
+
915
+ ## Adaptation modes
916
+
917
+ `mode` controls how a color adapts across schemes:
918
+
919
+ | Mode | Behavior |
920
+ |---|---|
921
+ | `'auto'` (default) | Full adaptation. Dark is a single tone inversion (`100 − t`) remapped into the dark window. High-contrast uses the full range. |
922
+ | `'fixed'` | Color stays recognizable. Tone is *mapped* (not inverted) into the dark window. Use for brand buttons, CTAs, status banners. |
923
+ | `'static'` | No adaptation. Same tone in every scheme. |
924
+
925
+ ### How relative tone adapts
926
+
927
+ **`auto`** — the offset is anchored to the base's per-scheme tone:
928
+
929
+ ```
930
+ Light: surface tone=97, text tone='-52' → tone 45 (dark text on light bg)
931
+ Dark: surface inverts to a low tone; the '-52' offset re-anchors to the
932
+ base's light tone and maps into the dark window (light text on dark bg)
933
+ ```
934
+
935
+ **`fixed`** — tone is mapped (not inverted), relative sign preserved:
936
+
937
+ ```
938
+ Light: accent-fill tone=52, accent-text tone='+20' → lighter than the fill
939
+ Dark: accent-fill maps into the dark window, sign preserved
940
+ ```
941
+
942
+ Offsets that would push past `[0, 100]` clamp to the boundary, or — with `flip` (default on) — mirror to the other side of the base. Set `flip: false` to keep the authored side and clamp instead.
943
+
944
+ **`static`** — no adaptation, same tone in every scheme.
945
+
946
+ ---
947
+
948
+ ## Light / dark scheme mapping
949
+
950
+ The mapping is now a single tone pipeline; there is no Möbius curve. See [`docs/okhst.md`](okhst.md) for the full math and the calibrated default constants.
951
+
952
+ ### Light scheme
953
+
954
+ An authored tone (0–100) is remapped into the `lightTone` window. The window's `lo`/`hi` are OKHSL-lightness endpoints (0–100); the tone is positioned within the window's tone interval and converted to a final OKHSL lightness. `static` mode and HC variants use the full range.
955
+
956
+ ```
957
+ window = lightTone // default [13, 100]
958
+ finalTone = remap(authorTone, window)
959
+ finalL = fromTone(finalTone) // OKHSL lightness
960
+ ```
961
+
962
+ ### Dark scheme
963
+
964
+ **`auto`** — invert the tone, then remap into the dark window:
965
+
966
+ ```
967
+ window = darkTone // default [10, 95]
968
+ inverted = 100 - authorTone
969
+ finalTone = remap(inverted, window)
970
+ ```
971
+
972
+ Because tone is contrast-uniform, the inversion preserves contrast steps without a fitted curve — the asymmetry between light and dark lives entirely in the two windows' `(lo, hi, eps)` scalars (`eps` defaults to the reference `0.05`).
973
+
974
+ **`fixed`** — remap into the dark window without inversion:
975
+
976
+ ```
977
+ finalTone = remap(authorTone, darkTone)
978
+ ```
979
+
980
+ In high-contrast variants both windows are bypassed (forced to the full `[0, 100]` range): `auto` still inverts, `fixed`/`static` do not.
981
+
982
+ ### Dark scheme — saturation
983
+
984
+ `darkDesaturation` reduces saturation for all colors in dark scheme:
985
+
986
+ ```ts
987
+ S_dark = S_light * (1 - darkDesaturation) // default: 0.1
988
+ ```
989
+
990
+ `static` mode skips desaturation.
991
+
992
+ ### Saturation taper
993
+
994
+ `saturationTaper` (default `0.15`) gently reduces saturation toward the tone extremes, where in-gamut chroma collapses. It is the *strength* (0–1) — the maximum fraction of saturation removed at the very edges — ramped in smoothly over the outer ~15% of tone on each end. Mid-tones are untouched; `0` disables it.
995
+
996
+ ---
997
+
998
+ ## Configuration
999
+
1000
+ ```ts
1001
+ glaze.configure({
1002
+ lightTone: [13, 100], // [lo, hi]; or { lo, hi, eps } / false to disable clamping
1003
+ darkTone: [10, 95], // [lo, hi]; or { lo, hi, eps } / false to disable clamping
1004
+ darkDesaturation: 0.1,
1005
+ saturationTaper: 0.15,
1006
+ states: {
1007
+ dark: '@dark',
1008
+ highContrast: '@high-contrast',
1009
+ },
1010
+ modes: {
1011
+ dark: true,
1012
+ highContrast: false,
1013
+ },
1014
+ shadowTuning: {
1015
+ alphaMax: 0.6,
1016
+ bgHueBlend: 0.2,
1017
+ },
1018
+ });
1019
+ ```
1020
+
1021
+ A `ToneWindow` is `[lo, hi]` (OKHSL-lightness endpoints, reference eps — the common form), `{ lo, hi, eps }` (advanced: explicit per-mode render eps), or `false` to disable clamping (full range `[0, 100]` at the reference eps). `false` removes the *boundaries*, not the contrast-uniform tone curve.
1022
+
1023
+ `GlazeConfig`:
1024
+
1025
+ | Field | Default | Description |
1026
+ |---|---|---|
1027
+ | `lightTone` | `[13, 100]` | Light scheme tone window: `[lo, hi]`, `{ lo, hi, eps }`, or `false` to disable clamping. Bypassed in HC. |
1028
+ | `darkTone` | `[10, 95]` | Dark scheme tone window: `[lo, hi]`, `{ lo, hi, eps }`, or `false` to disable clamping. Bypassed in HC. |
1029
+ | `darkDesaturation` | `0.1` | Saturation reduction in dark scheme (0–1). |
1030
+ | `saturationTaper` | `0.15` | Saturation taper strength toward the tone extremes (0–1). `0` disables. |
1031
+ | `states.dark` | `'@dark'` | State alias for dark mode tokens (Tasty export). |
1032
+ | `states.highContrast` | `'@high-contrast'` | State alias for HC tokens. |
1033
+ | `modes.dark` | `true` | Include dark variants in exports. |
1034
+ | `modes.highContrast` | `false` | Include HC variants. |
1035
+ | `shadowTuning` | `undefined` | Default tuning for all shadow colors. Per-color tuning merges field-by-field. |
1036
+ | `autoFlip` | `true` | Default for each color's `flip`. When solving `contrast` (or applying a relative `tone` that overshoots `[0, 100]`), allow crossing to the opposite side instead of clamping. With `false`, only the requested direction is considered; unmet contrasts pin the tone to that direction's extreme (and emit a warning) and overshooting offsets clamp to the boundary. Override per color via [`flip`](#flip). |
1037
+
1038
+ | Method | Description |
1039
+ |---|---|
1040
+ | `glaze.configure(config)` | Merge into the global config. Bumps a config version that invalidates theme caches. |
1041
+ | `glaze.getConfig()` | Snapshot the current resolved config (shallow copy). |
1042
+ | `glaze.resetConfig()` | Reset to defaults (also bumps the version counter). |
1043
+
1044
+ Standalone `glaze.color()` tokens snapshot the resolve-relevant fields at create time, so later `configure()` calls don't change already-created tokens. Themes merge the live global at resolve time for fields not overridden via `GlazeConfigOverride`.
1045
+
1046
+ ---
1047
+
1048
+ ## Output modes
1049
+
1050
+ Control which scheme variants appear in `tokens()` / `tasty()` / `json()` exports:
1051
+
1052
+ ```ts
1053
+ // Light only
1054
+ palette.tokens({ modes: { dark: false, highContrast: false } });
1055
+
1056
+ // Light + dark (default)
1057
+ palette.tokens({ modes: { highContrast: false } });
1058
+
1059
+ // All four variants
1060
+ palette.tokens({ modes: { dark: true, highContrast: true } });
1061
+ // → { light, dark, lightContrast, darkContrast }
1062
+ ```
1063
+
1064
+ Resolution priority (highest first):
1065
+
1066
+ 1. Per-call `modes` option on `tokens` / `tasty` / `json`.
1067
+ 2. `glaze.configure({ modes })` — global config.
1068
+ 3. Built-in default: `{ dark: true, highContrast: false }`.
1069
+
1070
+ ---
1071
+
1072
+ ## Validation
1073
+
1074
+ | Condition | Behavior |
1075
+ |---|---|
1076
+ | `contrast` without `base` in a **theme** color | Validation error |
1077
+ | Relative `tone` without `base` in a **theme** color | Validation error |
1078
+ | `contrast` without `base` in `glaze.color()` | Anchors against the literal seed (no error) |
1079
+ | Relative `tone` without `base` in `glaze.color()` | Anchors against the literal seed (no error) |
1080
+ | Relative `tone` overshoots `[0, 100]` | Mirror to the other side of the base (`flip` on, default), or clamp to the boundary (`flip` off) |
1081
+ | `tone` resolves outside 0–100 | Clamp silently |
1082
+ | `'max'` / `'min'` without `base` | Allowed — resolves to the scheme's tone extreme (root color) |
1083
+ | `saturation` outside 0–1 | Clamp silently |
1084
+ | Circular `base` references | Validation error |
1085
+ | `base` references non-existent name | Validation error |
1086
+ | Shadow `bg` references non-existent color | Validation error |
1087
+ | Shadow `fg` references non-existent color | Validation error |
1088
+ | Shadow `bg` references another shadow color | Validation error |
1089
+ | Shadow `fg` references another shadow color | Validation error |
1090
+ | Regular color `base` references a shadow color | Validation error |
1091
+ | Shadow `intensity` outside 0–100 | Clamp silently |
1092
+ | `contrast` + `opacity` combined | `console.warn` |
1093
+ | Mix `base` references non-existent color | Validation error |
1094
+ | Mix `target` references non-existent color | Validation error |
1095
+ | Mix `base` references a shadow color | Validation error |
1096
+ | Mix `target` references a shadow color | Validation error |
1097
+ | Mix `value` outside 0–100 | Clamp silently |
1098
+ | Circular references involving mix colors | Validation error |
1099
+ | Contrast target physically unreachable | `console.warn` (deduped per `(name, scheme, target)`); closest passing variant returned |
1100
+
1101
+ ---
1102
+
1103
+ ## Color math utilities
1104
+
1105
+ For advanced use, Glaze re-exports its internal color math.
1106
+
1107
+ ### Conversions
1108
+
1109
+ ```ts
1110
+ import {
1111
+ okhslToLinearSrgb,
1112
+ okhslToSrgb,
1113
+ okhslToOklab,
1114
+ oklabToOkhsl,
1115
+ srgbToOkhsl,
1116
+ hslToSrgb,
1117
+ parseHex,
1118
+ parseHexAlpha,
1119
+ relativeLuminanceFromLinearRgb,
1120
+ contrastRatioFromLuminance,
1121
+ gamutClampedLuminance,
1122
+ } from '@tenphi/glaze';
1123
+ ```
1124
+
1125
+ | Function | Description |
1126
+ |---|---|
1127
+ | `okhslToLinearSrgb(h, s, l)` | OKHSL (h: 0–360, s/l: 0–1) → linear sRGB tuple. |
1128
+ | `okhslToSrgb(h, s, l)` | OKHSL → gamma-encoded sRGB tuple (0–1 per channel). |
1129
+ | `okhslToOklab([h, s, l])` | OKHSL → OKLab `[L, a, b]`. |
1130
+ | `oklabToOkhsl([L, a, b])` | OKLab → OKHSL. |
1131
+ | `srgbToOkhsl([r, g, b])` | Gamma sRGB (0–1) → OKHSL. |
1132
+ | `hslToSrgb(h, s, l)` | CSS HSL → sRGB tuple. |
1133
+ | `parseHex(hex)` | Parse `#rgb` / `#rrggbb` to sRGB tuple. Returns `null` on invalid input. |
1134
+ | `parseHexAlpha(hex)` | Parse `#rgb` / `#rrggbb` / `#rrggbbaa`; returns `[r, g, b, a?]`. |
1135
+ | `relativeLuminanceFromLinearRgb(rgb)` | WCAG relative luminance from linear sRGB. |
1136
+ | `contrastRatioFromLuminance(yA, yB)` | WCAG contrast ratio from two luminances. |
1137
+ | `gamutClampedLuminance(linearRgb)` | Relative luminance with channel clamping for out-of-gamut colors. |
1138
+
1139
+ ### Format writers
1140
+
1141
+ ```ts
1142
+ import { formatOkhsl, formatRgb, formatHsl, formatOklch } from '@tenphi/glaze';
1143
+
1144
+ formatOkhsl(280, 60, 95); // 'okhsl(280 60% 95%)'
1145
+ formatRgb(280, 60, 95); // 'rgb(244 240 250)'
1146
+ formatHsl(280, 60, 95); // 'hsl(280 60% 95%)'
1147
+ formatOklch(280, 60, 95); // 'oklch(0.95 ... 280)'
1148
+ ```
1149
+
1150
+ To attach an alpha component, use `glaze.format(variant, format)` on a `ResolvedColorVariant` (which carries the `alpha` channel) instead of these raw writers.
1151
+
1152
+ ### OKHST tone utilities
1153
+
1154
+ ```ts
1155
+ import {
1156
+ toTone,
1157
+ fromTone,
1158
+ toneFromY,
1159
+ yFromTone,
1160
+ okhstToOkhsl,
1161
+ okhslToOkhst,
1162
+ variantToOkhsl,
1163
+ REF_EPS,
1164
+ } from '@tenphi/glaze';
1165
+ ```
1166
+
1167
+ | Function | Description |
1168
+ |---|---|
1169
+ | `toTone(l, eps?)` | OKHSL lightness (0–1) → tone (0–100). Defaults to `REF_EPS`. |
1170
+ | `fromTone(t, eps?)` | Tone (0–100) → OKHSL lightness (0–1). Inverse of `toTone`. |
1171
+ | `toneFromY(y, eps?)` / `yFromTone(t, eps?)` | Same transfer in luminance space (0–1). |
1172
+ | `okhstToOkhsl({ h, s, t })` | OKHST → OKHSL (`{ h, s, l }`). |
1173
+ | `okhslToOkhst({ h, s, l })` | OKHSL → OKHST (`{ h, s, t }`). |
1174
+ | `variantToOkhsl(variant)` | `ResolvedColorVariant` (stores `t`) → `{ h, s, l, alpha }` for rendering. |
1175
+ | `REF_EPS` | Reference epsilon (`0.05`) for the canonical tone axis. |
1176
+
1177
+ `ResolvedColorVariant` now stores `{ h, s, t, alpha }` (tone, not lightness). Use `variantToOkhsl(variant).l` to recover OKHSL lightness. See [`docs/okhst.md`](okhst.md) for the full model.
1178
+
1179
+ ### Contrast solver
1180
+
1181
+ ```ts
1182
+ import {
1183
+ findToneForContrast,
1184
+ findValueForMixContrast,
1185
+ resolveContrastForMode,
1186
+ resolveMinContrast,
1187
+ apcaContrast,
1188
+ } from '@tenphi/glaze';
1189
+ ```
1190
+
1191
+ | Function | Description |
1192
+ |---|---|
1193
+ | `findToneForContrast(opts)` | Binary-search for the tone (0–1) that meets a contrast floor (WCAG or APCA) against a base color. Returns `{ tone, contrast, met, branch, flipped? }`. |
1194
+ | `findValueForMixContrast(opts)` | Same, but searches for a mix `value` (0–1) that meets a contrast floor between a base and a target. |
1195
+ | `resolveContrastForMode(spec, isHC)` | Resolves a `ContrastSpec` to `{ metric: 'wcag' \| 'apca', target }` for the requested mode (picks the normal or HC entry of any pair). |
1196
+ | `resolveMinContrast(value)` | Resolves a `MinContrast` (WCAG preset or number) to a numeric ratio. |
1197
+ | `apcaContrast(yText, yBg)` | APCA Lc magnitude (0–106) for two relative luminances. |
1198
+
1199
+ `findToneForContrast` options:
1200
+
1201
+ | Option | Default | Description |
1202
+ |---|---|---|
1203
+ | `hue` | — | Candidate hue (0–360). |
1204
+ | `saturation` | — | Candidate saturation (0–1). |
1205
+ | `preferredTone` | — | Preferred candidate tone (0–1). Kept if it already meets the target. |
1206
+ | `baseLinearRgb` | — | Base color as linear sRGB tuple. |
1207
+ | `contrast` | — | `ResolvedContrast` (`{ metric, target }`). |
1208
+ | `toneRange` | `[0, 1]` | Search bounds in tone. |
1209
+ | `epsilon` | `1e-4` | Convergence threshold. |
1210
+ | `maxIterations` | `18` | Max binary-search iterations per branch. |
1211
+ | `initialDirection` | higher-contrast side | Direction to search first (`'lighter'` or `'darker'`). |
1212
+ | `flip` | `false` | When `true`, try the opposite direction if the initial one doesn't meet the target. When `false`, only the initial direction is searched — unmet contrasts pin the result to that direction's extreme. |
1213
+ | `saturationTaper` | `0` | When `> 0`, candidate saturation rolls off toward the tone extremes (same envelope the renderer applies), so the solved tone meets the floor at its rendered saturation. |
1214
+
1215
+ Result: `{ tone, contrast, met, branch: 'lighter' | 'darker' | 'preferred', flipped? }`. `flipped: true` indicates the initial direction failed and the opposite direction satisfied the target.