rnwind 0.0.2 → 0.0.3

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 (80) hide show
  1. package/lib/cjs/core/parser/color.cjs +53 -24
  2. package/lib/cjs/core/parser/color.cjs.map +1 -1
  3. package/lib/cjs/core/parser/layout-dispatcher.cjs +20 -0
  4. package/lib/cjs/core/parser/layout-dispatcher.cjs.map +1 -1
  5. package/lib/cjs/core/parser/length.cjs +20 -6
  6. package/lib/cjs/core/parser/length.cjs.map +1 -1
  7. package/lib/cjs/core/parser/length.d.ts +6 -3
  8. package/lib/cjs/core/parser/shorthand.cjs +37 -5
  9. package/lib/cjs/core/parser/shorthand.cjs.map +1 -1
  10. package/lib/cjs/core/parser/shorthand.d.ts +11 -5
  11. package/lib/cjs/core/parser/theme-vars.cjs +53 -0
  12. package/lib/cjs/core/parser/theme-vars.cjs.map +1 -1
  13. package/lib/cjs/core/parser/theme-vars.d.ts +21 -0
  14. package/lib/cjs/core/parser/tokens.cjs +183 -1
  15. package/lib/cjs/core/parser/tokens.cjs.map +1 -1
  16. package/lib/cjs/core/parser/tw-parser.cjs +140 -27
  17. package/lib/cjs/core/parser/tw-parser.cjs.map +1 -1
  18. package/lib/cjs/core/parser/tw-parser.d.ts +21 -5
  19. package/lib/cjs/core/parser/typography-dispatcher.cjs +16 -1
  20. package/lib/cjs/core/parser/typography-dispatcher.cjs.map +1 -1
  21. package/lib/cjs/core/style-builder/build-style.cjs +73 -26
  22. package/lib/cjs/core/style-builder/build-style.cjs.map +1 -1
  23. package/lib/cjs/metro/state.cjs +52 -2
  24. package/lib/cjs/metro/state.cjs.map +1 -1
  25. package/lib/cjs/metro/state.d.ts +17 -1
  26. package/lib/cjs/metro/transform-ast.cjs +238 -21
  27. package/lib/cjs/metro/transform-ast.cjs.map +1 -1
  28. package/lib/cjs/metro/transform-ast.d.ts +15 -0
  29. package/lib/cjs/metro/transformer.cjs +29 -2
  30. package/lib/cjs/metro/transformer.cjs.map +1 -1
  31. package/lib/cjs/metro/with-config.cjs +1 -1
  32. package/lib/cjs/metro/with-config.cjs.map +1 -1
  33. package/lib/cjs/metro/with-config.d.ts +22 -0
  34. package/lib/esm/core/parser/color.mjs +53 -24
  35. package/lib/esm/core/parser/color.mjs.map +1 -1
  36. package/lib/esm/core/parser/layout-dispatcher.mjs +20 -0
  37. package/lib/esm/core/parser/layout-dispatcher.mjs.map +1 -1
  38. package/lib/esm/core/parser/length.d.ts +6 -3
  39. package/lib/esm/core/parser/length.mjs +20 -6
  40. package/lib/esm/core/parser/length.mjs.map +1 -1
  41. package/lib/esm/core/parser/shorthand.d.ts +11 -5
  42. package/lib/esm/core/parser/shorthand.mjs +37 -5
  43. package/lib/esm/core/parser/shorthand.mjs.map +1 -1
  44. package/lib/esm/core/parser/theme-vars.d.ts +21 -0
  45. package/lib/esm/core/parser/theme-vars.mjs +53 -1
  46. package/lib/esm/core/parser/theme-vars.mjs.map +1 -1
  47. package/lib/esm/core/parser/tokens.mjs +183 -1
  48. package/lib/esm/core/parser/tokens.mjs.map +1 -1
  49. package/lib/esm/core/parser/tw-parser.d.ts +21 -5
  50. package/lib/esm/core/parser/tw-parser.mjs +141 -28
  51. package/lib/esm/core/parser/tw-parser.mjs.map +1 -1
  52. package/lib/esm/core/parser/typography-dispatcher.mjs +16 -1
  53. package/lib/esm/core/parser/typography-dispatcher.mjs.map +1 -1
  54. package/lib/esm/core/style-builder/build-style.mjs +73 -26
  55. package/lib/esm/core/style-builder/build-style.mjs.map +1 -1
  56. package/lib/esm/metro/state.d.ts +17 -1
  57. package/lib/esm/metro/state.mjs +51 -3
  58. package/lib/esm/metro/state.mjs.map +1 -1
  59. package/lib/esm/metro/transform-ast.d.ts +15 -0
  60. package/lib/esm/metro/transform-ast.mjs +238 -21
  61. package/lib/esm/metro/transform-ast.mjs.map +1 -1
  62. package/lib/esm/metro/transformer.mjs +30 -3
  63. package/lib/esm/metro/transformer.mjs.map +1 -1
  64. package/lib/esm/metro/with-config.d.ts +22 -0
  65. package/lib/esm/metro/with-config.mjs +1 -1
  66. package/lib/esm/metro/with-config.mjs.map +1 -1
  67. package/package.json +2 -1
  68. package/src/core/parser/color.ts +52 -18
  69. package/src/core/parser/layout-dispatcher.ts +19 -0
  70. package/src/core/parser/length.ts +20 -6
  71. package/src/core/parser/shorthand.ts +35 -5
  72. package/src/core/parser/theme-vars.ts +53 -0
  73. package/src/core/parser/tokens.ts +171 -1
  74. package/src/core/parser/tw-parser.ts +147 -28
  75. package/src/core/parser/typography-dispatcher.ts +15 -1
  76. package/src/core/style-builder/build-style.ts +84 -26
  77. package/src/metro/state.ts +49 -1
  78. package/src/metro/transform-ast.ts +249 -18
  79. package/src/metro/transformer.ts +28 -3
  80. package/src/metro/with-config.ts +23 -1
@@ -1 +1 @@
1
- {"version":3,"file":"tokens.mjs","sources":["../../../../../src/core/parser/tokens.ts"],"sourcesContent":["import type { Token, TokenOrValue } from 'lightningcss'\nimport { BARE_NUMBER_REGEX, CALC_MUL_REGEX, CALC_RATIO_REGEX, LENGTH_PX_REGEX, LENGTH_REM_REGEX, REM_TO_PX } from './constants'\nimport { cssColorToString } from './color'\nimport type { RNStyleValue } from './types'\n\n/**\n * Extract the fallback clause of a `var(--name, fallback)` by walking the\n * string with paren-depth tracking. Linear-time; safe on nested\n * `var(--a, var(--b, var(--c, 1rem)))` without regex backtracking.\n * @param text CSS value already trimmed.\n * @returns Fallback text, or `null` when `text` is not a `var(..., ...)`\n * with a fallback.\n */\nfunction extractVariableFallback(text: string): string | null {\n if (!text.startsWith('var(') || !text.endsWith(')')) return null\n const inner = text.slice(4, -1)\n let depth = 0\n for (let index = 0; index < inner.length; index += 1) {\n const ch = inner[index]\n if (ch === '(') depth += 1\n else if (ch === ')') depth -= 1\n else if (ch === ',' && depth === 0) return inner.slice(index + 1).trim()\n }\n return null\n}\n\n/**\n * Find the matching `)` for the opening `var(` whose body starts at\n * `start`. Returns `-1` when the parens are unbalanced.\n * @param text Source text.\n * @param start Index just past the opening `var(`.\n * @returns Index of the matching `)`, or `-1`.\n */\nfunction findBalancedParenEnd(text: string, start: number): number {\n let depth = 1\n for (let index = start; index < text.length; index += 1) {\n const ch = text[index]\n if (ch === '(') depth += 1\n else if (ch === ')') {\n depth -= 1\n if (depth === 0) return index\n }\n }\n return -1\n}\n\n/**\n * Resolve a `var(…)` body (the bit between parentheses). Reads the\n * variable name, then either returns the table lookup or recurses into\n * the fallback clause. When neither resolves, re-wraps as `var(…)` so\n * downstream coercion still sees a well-formed reference.\n * @param body Text between the outer parentheses of a `var()` call.\n * @param table var → value map.\n * @returns Substituted text fragment.\n */\nfunction resolveVariableBody(body: string, table: ReadonlyMap<string, string>): string {\n let depth = 0\n let commaIndex = -1\n for (const [index, ch] of [...body].entries()) {\n if (ch === '(') depth += 1\n else if (ch === ')') depth -= 1\n else if (ch === ',' && depth === 0) {\n commaIndex = index\n break\n }\n }\n const rawName = (commaIndex === -1 ? body : body.slice(0, commaIndex)).trim()\n const fallback = commaIndex === -1 ? null : body.slice(commaIndex + 1).trim()\n const resolved = table.get(rawName)\n if (resolved !== undefined) return resolved\n if (fallback !== null) return substitutePass(fallback, table)\n return `var(${body})`\n}\n\n/**\n * One pass of left-to-right `var(...)` substitution. Separate from the\n * fixed-point driver so the recursion depth stays bounded.\n * @param text Value text to scan.\n * @param table var → value map.\n * @returns Value text after one pass.\n */\nfunction substitutePass(text: string, table: ReadonlyMap<string, string>): string {\n let out = ''\n let index = 0\n while (index < text.length) {\n const head = text.indexOf('var(', index)\n if (head === -1) {\n out += text.slice(index)\n break\n }\n out += text.slice(index, head)\n const end = findBalancedParenEnd(text, head + 4)\n if (end === -1) {\n out += text.slice(head)\n break\n }\n const body = text.slice(head + 4, end)\n out += resolveVariableBody(body, table)\n index = end + 1\n }\n return out\n}\n\n/**\n * Serialize a list of `TokenOrValue` nodes into a CSS-ish value string.\n * Preserves the CSS form closely enough for downstream numeric coercion.\n * @param tokens Token list from an unparsed declaration or custom-property body.\n * @returns Concatenated CSS-value fragment.\n */\nexport function serializeTokens(tokens: readonly TokenOrValue[]): string {\n let out = ''\n for (const token of tokens) out += serializeToken(token)\n return out.trim()\n}\n\n/**\n * Serialize one `TokenOrValue` node back to CSS text. Handles the shapes\n * Tailwind v4 actually emits in utility-class bodies: raw tokens, `var()`\n * references, and numeric functions (`calc()`).\n * @param token One token node.\n * @returns CSS-value fragment.\n */\nexport function serializeToken(token: TokenOrValue): string {\n switch (token.type) {\n case 'token': {\n return serializeRawToken(token.value)\n }\n case 'var': {\n const head = token.value.name.ident\n const { fallback } = token.value\n if (!fallback || fallback.length === 0) return `var(${head})`\n return `var(${head}, ${serializeTokens(fallback)})`\n }\n case 'function': {\n return `${token.value.name}(${serializeTokens(token.value.arguments)})`\n }\n case 'length': {\n return `${token.value.value}${token.value.unit}`\n }\n case 'dashed-ident': {\n return token.value\n }\n case 'angle': {\n return `${token.value.value}${token.value.type}`\n }\n case 'time': {\n const unit = token.value.type === 'milliseconds' ? 'ms' : 's'\n return `${token.value.value}${unit}`\n }\n case 'resolution': {\n return `${token.value.value}${token.value.type}`\n }\n case 'color': {\n // Pre-resolved CSS color (`oklch(...)`, `rgb(...)`, etc.) — render\n // it back to a hex/rgba string RN can read.\n return cssColorToString(token.value)\n }\n case 'env':\n case 'unresolved-color':\n case 'url':\n case 'animation-name': {\n return ''\n }\n default: {\n return ''\n }\n }\n}\n\n/**\n * Serialize a raw `Token` back to CSS text. `TokenOrValue` with\n * `type === 'token'` wraps one of these; the discriminated union lets\n * TypeScript narrow per branch without casts.\n * @param token Raw token.\n * @returns CSS text fragment.\n */\nexport function serializeRawToken(token: Token): string {\n switch (token.type) {\n case 'ident':\n case 'at-keyword':\n case 'string':\n case 'unquoted-url':\n case 'function': {\n return token.value\n }\n case 'hash':\n case 'id-hash': {\n return `#${token.value}`\n }\n case 'number': {\n return String(token.value)\n }\n case 'percentage': {\n return `${token.value * 100}%`\n }\n case 'dimension': {\n return `${token.value}${token.unit}`\n }\n case 'white-space': {\n return ' '\n }\n case 'delim': {\n return token.value\n }\n case 'comma': {\n return ','\n }\n case 'colon':\n case 'semicolon':\n case 'parenthesis-block':\n case 'square-bracket-block':\n case 'curly-bracket-block':\n case 'cdo':\n case 'cdc':\n case 'include-match':\n case 'dash-match':\n case 'prefix-match':\n case 'suffix-match':\n case 'substring-match':\n case 'comment':\n case 'bad-url':\n case 'bad-string': {\n return ''\n }\n default: {\n return ''\n }\n }\n}\n\n/**\n * Coerce the flat serialization of an unparsed value into an RN scalar.\n * Handles the shapes Tailwind v4 actually emits: bare numbers, pixel /\n * rem lengths, calc ratios, and `var(--x, fallback)` where we recurse into\n * the fallback. Anything else passes through as a string.\n * @param text Serialized CSS value.\n * @returns Coerced primitive, or `null` when unrepresentable.\n */\nexport function coerceUnparsedValue(text: string): RNStyleValue | null {\n const trimmed = text.trim()\n if (trimmed.length === 0) return null\n if (BARE_NUMBER_REGEX.test(trimmed)) return Number(trimmed)\n const px = LENGTH_PX_REGEX.exec(trimmed)\n if (px) return Number(px[1])\n const rem = LENGTH_REM_REGEX.exec(trimmed)\n if (rem) return Number(rem[1]) * REM_TO_PX\n const fallback = extractVariableFallback(trimmed)\n if (fallback !== null) return coerceUnparsedValue(fallback)\n const calcRatio = CALC_RATIO_REGEX.exec(trimmed)\n if (calcRatio) {\n const right = Number(calcRatio[2])\n if (right === 0) return null\n return Number(calcRatio[1]) / right\n }\n const calcMul = CALC_MUL_REGEX.exec(trimmed)\n if (calcMul) {\n // Unit-aware multiply: `calc(0.5rem * 2)` → 16 (rem scaled to px).\n // Regex captures `(number)(unit?) * (number)` — the unit is implicit\n // in the match position; rebuild via the full match text.\n const unitMatch = /^calc\\(\\s*-?\\d+(?:\\.\\d+)?(rem|px)?/.exec(trimmed)\n const unit = unitMatch?.[1]\n const base = Number(calcMul[1]) * Number(calcMul[2])\n return unit === 'rem' ? base * REM_TO_PX : base\n }\n return trimmed\n}\n\n/**\n * Substitute every `var(--name [, fallback])` reference in `text` with\n * the value from `table` (or the fallback clause when the name misses).\n * Paren-balanced so nested `var(…)` refs don't confuse the scanner.\n * Iterates to a fixed point so multi-hop substitutions land in one call\n * (with a safety cap so a malformed self-referential token can't loop).\n * @param text Raw CSS value.\n * @param table var name → resolved value map.\n * @returns Substituted text.\n */\nexport function substituteThemeVars(text: string, table: ReadonlyMap<string, string>): string {\n let current = text\n for (let pass = 0; pass < 8; pass += 1) {\n const next = substitutePass(current, table)\n if (next === current) return next\n current = next\n }\n return current\n}\n"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"tokens.mjs","sources":["../../../../../src/core/parser/tokens.ts"],"sourcesContent":["import type { Token, TokenOrValue } from 'lightningcss'\nimport { rgb as culoriRgb } from 'culori'\nimport { BARE_NUMBER_REGEX, CALC_MUL_REGEX, CALC_RATIO_REGEX, LENGTH_PX_REGEX, LENGTH_REM_REGEX, REM_TO_PX } from './constants'\nimport { cssColorToString } from './color'\nimport type { RNStyleValue } from './types'\n\n/**\n * Extract the fallback clause of a `var(--name, fallback)` by walking the\n * string with paren-depth tracking. Linear-time; safe on nested\n * `var(--a, var(--b, var(--c, 1rem)))` without regex backtracking.\n * @param text CSS value already trimmed.\n * @returns Fallback text, or `null` when `text` is not a `var(..., ...)`\n * with a fallback.\n */\nfunction extractVariableFallback(text: string): string | null {\n if (!text.startsWith('var(') || !text.endsWith(')')) return null\n const inner = text.slice(4, -1)\n let depth = 0\n for (let index = 0; index < inner.length; index += 1) {\n const ch = inner[index]\n if (ch === '(') depth += 1\n else if (ch === ')') depth -= 1\n else if (ch === ',' && depth === 0) return inner.slice(index + 1).trim()\n }\n return null\n}\n\n/**\n * Find the matching `)` for the opening `var(` whose body starts at\n * `start`. Returns `-1` when the parens are unbalanced.\n * @param text Source text.\n * @param start Index just past the opening `var(`.\n * @returns Index of the matching `)`, or `-1`.\n */\nfunction findBalancedParenEnd(text: string, start: number): number {\n let depth = 1\n for (let index = start; index < text.length; index += 1) {\n const ch = text[index]\n if (ch === '(') depth += 1\n else if (ch === ')') {\n depth -= 1\n if (depth === 0) return index\n }\n }\n return -1\n}\n\n/**\n * Resolve a `var(…)` body (the bit between parentheses). Reads the\n * variable name, then either returns the table lookup or recurses into\n * the fallback clause. When neither resolves, re-wraps as `var(…)` so\n * downstream coercion still sees a well-formed reference.\n * @param body Text between the outer parentheses of a `var()` call.\n * @param table var → value map.\n * @returns Substituted text fragment.\n */\nfunction resolveVariableBody(body: string, table: ReadonlyMap<string, string>): string {\n let depth = 0\n let commaIndex = -1\n for (const [index, ch] of [...body].entries()) {\n if (ch === '(') depth += 1\n else if (ch === ')') depth -= 1\n else if (ch === ',' && depth === 0) {\n commaIndex = index\n break\n }\n }\n const rawName = (commaIndex === -1 ? body : body.slice(0, commaIndex)).trim()\n const fallback = commaIndex === -1 ? null : body.slice(commaIndex + 1).trim()\n const resolved = table.get(rawName)\n if (resolved !== undefined) return resolved\n if (fallback !== null) return substitutePass(fallback, table)\n return `var(${body})`\n}\n\n/**\n * One pass of left-to-right `var(...)` substitution. Separate from the\n * fixed-point driver so the recursion depth stays bounded.\n * @param text Value text to scan.\n * @param table var → value map.\n * @returns Value text after one pass.\n */\nfunction substitutePass(text: string, table: ReadonlyMap<string, string>): string {\n let out = ''\n let index = 0\n while (index < text.length) {\n const head = text.indexOf('var(', index)\n if (head === -1) {\n out += text.slice(index)\n break\n }\n out += text.slice(index, head)\n const end = findBalancedParenEnd(text, head + 4)\n if (end === -1) {\n out += text.slice(head)\n break\n }\n const body = text.slice(head + 4, end)\n out += resolveVariableBody(body, table)\n index = end + 1\n }\n return out\n}\n\n/**\n * Serialize a list of `TokenOrValue` nodes into a CSS-ish value string.\n * Preserves the CSS form closely enough for downstream numeric coercion.\n * @param tokens Token list from an unparsed declaration or custom-property body.\n * @returns Concatenated CSS-value fragment.\n */\nexport function serializeTokens(tokens: readonly TokenOrValue[]): string {\n let out = ''\n for (const token of tokens) out += serializeToken(token)\n return out.trim()\n}\n\n/**\n * Serialize one `TokenOrValue` node back to CSS text. Handles the shapes\n * Tailwind v4 actually emits in utility-class bodies: raw tokens, `var()`\n * references, and numeric functions (`calc()`).\n * @param token One token node.\n * @returns CSS-value fragment.\n */\nexport function serializeToken(token: TokenOrValue): string {\n switch (token.type) {\n case 'token': {\n return serializeRawToken(token.value)\n }\n case 'var': {\n const head = token.value.name.ident\n const { fallback } = token.value\n if (!fallback || fallback.length === 0) return `var(${head})`\n return `var(${head}, ${serializeTokens(fallback)})`\n }\n case 'function': {\n return `${token.value.name}(${serializeTokens(token.value.arguments)})`\n }\n case 'length': {\n return `${token.value.value}${token.value.unit}`\n }\n case 'dashed-ident': {\n return token.value\n }\n case 'angle': {\n return `${token.value.value}${token.value.type}`\n }\n case 'time': {\n const unit = token.value.type === 'milliseconds' ? 'ms' : 's'\n return `${token.value.value}${unit}`\n }\n case 'resolution': {\n return `${token.value.value}${token.value.type}`\n }\n case 'color': {\n // Pre-resolved CSS color (`oklch(...)`, `rgb(...)`, etc.) — render\n // it back to a hex/rgba string RN can read.\n return cssColorToString(token.value)\n }\n case 'env':\n case 'unresolved-color':\n case 'url':\n case 'animation-name': {\n return ''\n }\n default: {\n return ''\n }\n }\n}\n\n/**\n * Serialize a raw `Token` back to CSS text. `TokenOrValue` with\n * `type === 'token'` wraps one of these; the discriminated union lets\n * TypeScript narrow per branch without casts.\n * @param token Raw token.\n * @returns CSS text fragment.\n */\nexport function serializeRawToken(token: Token): string {\n switch (token.type) {\n case 'ident':\n case 'at-keyword':\n case 'string':\n case 'unquoted-url':\n case 'function': {\n return token.value\n }\n case 'hash':\n case 'id-hash': {\n return `#${token.value}`\n }\n case 'number': {\n return String(token.value)\n }\n case 'percentage': {\n return `${token.value * 100}%`\n }\n case 'dimension': {\n return `${token.value}${token.unit}`\n }\n case 'white-space': {\n return ' '\n }\n case 'delim': {\n return token.value\n }\n case 'comma': {\n return ','\n }\n case 'colon':\n case 'semicolon':\n case 'parenthesis-block':\n case 'square-bracket-block':\n case 'curly-bracket-block':\n case 'cdo':\n case 'cdc':\n case 'include-match':\n case 'dash-match':\n case 'prefix-match':\n case 'suffix-match':\n case 'substring-match':\n case 'comment':\n case 'bad-url':\n case 'bad-string': {\n return ''\n }\n default: {\n return ''\n }\n }\n}\n\n/**\n * Coerce the flat serialization of an unparsed value into an RN scalar.\n * Handles the shapes Tailwind v4 actually emits: bare numbers, pixel /\n * rem lengths, calc ratios, and `var(--x, fallback)` where we recurse into\n * the fallback. Anything else passes through as a string.\n * @param text Serialized CSS value.\n * @returns Coerced primitive, or `null` when unrepresentable.\n */\nexport function coerceUnparsedValue(text: string): RNStyleValue | null {\n const trimmed = text.trim()\n if (trimmed.length === 0) return null\n if (BARE_NUMBER_REGEX.test(trimmed)) return Number(trimmed)\n const px = LENGTH_PX_REGEX.exec(trimmed)\n if (px) return Number(px[1])\n const rem = LENGTH_REM_REGEX.exec(trimmed)\n if (rem) return Number(rem[1]) * REM_TO_PX\n const colorMix = evaluateColorMixWithTransparent(trimmed)\n if (colorMix !== null) return colorMix\n const fallback = extractVariableFallback(trimmed)\n if (fallback !== null) return coerceUnparsedValue(fallback)\n const calcRatio = CALC_RATIO_REGEX.exec(trimmed)\n if (calcRatio) {\n const right = Number(calcRatio[2])\n if (right === 0) return null\n return Number(calcRatio[1]) / right\n }\n const calcMul = CALC_MUL_REGEX.exec(trimmed)\n if (calcMul) {\n // Unit-aware multiply: `calc(0.5rem * 2)` → 16 (rem scaled to px).\n // Regex captures `(number)(unit?) * (number)` — the unit is implicit\n // in the match position; rebuild via the full match text.\n const unitMatch = /^calc\\(\\s*-?\\d+(?:\\.\\d+)?(rem|px)?/.exec(trimmed)\n const unit = unitMatch?.[1]\n const base = Number(calcMul[1]) * Number(calcMul[2])\n return unit === 'rem' ? base * REM_TO_PX : base\n }\n return unquoteCssString(trimmed)\n}\n\n/**\n * Evaluate the specific `color-mix(in <space>, <color> <pct>%, transparent)`\n * shape Tailwind v4 emits for opacity-suffixed themed colors (e.g.\n * `border-text/20`, `bg-on-background/30`). The result is the original\n * color with `alpha = originalAlpha * pct/100` — no actual color-space\n * conversion is needed because mixing a color with `transparent` only\n * changes its alpha, regardless of the named space (oklab / srgb /\n * lab / …) — the chrominance is preserved.\n *\n * Returns null when the expression isn't this shape (handed back to\n * the caller for the next coercion strategy).\n * @param text Trimmed CSS value.\n * @returns RN-compatible `rgba(...)` string, or null when unmatched.\n */\nfunction evaluateColorMixWithTransparent(text: string): string | null {\n const lower = text.toLowerCase()\n if (!lower.startsWith('color-mix(')) return null\n // Match the trailing `, transparent)` (allowing optional whitespace).\n const tail = /,\\s*transparent\\s*\\)\\s*$/.exec(text)\n if (!tail) return null\n // Skip the `in <space>` clause (everything up to the FIRST comma after\n // the opening paren). Walking by hand instead of regex because the\n // color slot may itself contain `(...)` (e.g. `rgb(...)`).\n const inComma = text.indexOf(',', 'color-mix('.length)\n if (inComma === -1 || inComma > tail.index) return null\n const middle = text.slice(inComma + 1, tail.index).trim()\n // `<color> <pct>%` — the `<num>%` token is at the END of `middle`.\n // Anchored, no backtracking — explicit `% ` then end.\n const pctMatch = COLOR_MIX_PCT_TAIL.exec(middle)\n if (!pctMatch) return null\n const pct = Number(pctMatch[1]) / 100\n if (!Number.isFinite(pct)) return null\n const colorText = middle.slice(0, pctMatch.index).trim()\n if (colorText.length === 0) return null\n return applyAlphaToCssColor(colorText, pct)\n}\n\n/** End-anchored `<num>%` matcher used to slice a color-mix percentage off the right of an expression. */\n// eslint-disable-next-line sonarjs/slow-regex -- end-anchored, atomic-style group; bounded backtracking is safe.\nconst COLOR_MIX_PCT_TAIL = /(-?\\d+(?:\\.\\d+)?)%$/\n\n/**\n * Multiply the alpha channel of a serialized CSS color by `multiplier`\n * (0…1). Recognises `#rgb` / `#rrggbb` / `#rrggbbaa` hex, named colors\n * (only `transparent` matters here), and `rgb(…)` / `rgba(…)` forms —\n * which is what theme tokens resolve to after substitution.\n * @param color CSS color text.\n * @param multiplier Alpha multiplier (0…1).\n * @returns `rgba(r, g, b, a)` string with the adjusted alpha.\n */\nfunction applyAlphaToCssColor(color: string, multiplier: number): string | null {\n const trimmed = color.trim()\n if (trimmed === 'transparent') return 'rgba(0, 0, 0, 0)'\n return alphaFromHex(trimmed, multiplier) ?? alphaFromRgbFunction(trimmed, multiplier) ?? alphaFromCulori(trimmed, multiplier)\n}\n\n/**\n * Apply the alpha multiplier to a hex literal, expanding 3/4/6/8-digit forms.\n * @param text\n * @param multiplier\n */\nfunction alphaFromHex(text: string, multiplier: number): string | null {\n const hexMatch = /^#([0-9a-fA-F]{3,8})$/.exec(text)\n if (!hexMatch) return null\n const expanded = expandHex(hexMatch[1]!)\n if (!expanded) return null\n return `rgba(${expanded.r}, ${expanded.g}, ${expanded.b}, ${expanded.alpha * multiplier})`\n}\n\n/**\n * Apply alpha to an `rgb(…)` / `rgba(…)` literal. Walks the channels by\n * hand instead of a multi-capture regex (the linter flags the regex\n * form as backtracking-prone).\n * @param text\n * @param multiplier\n */\nfunction alphaFromRgbFunction(text: string, multiplier: number): string | null {\n if (!text.startsWith('rgb(') && !text.startsWith('rgba(')) return null\n const inner = text.slice(text.indexOf('(') + 1, -1)\n const channels = inner.split(/\\s*[\\s,]\\s*/).filter((part) => part.length > 0)\n if (channels.length !== 3 && channels.length !== 4) return null\n const r = Math.round(Number(channels[0]))\n const g = Math.round(Number(channels[1]))\n const b = Math.round(Number(channels[2]))\n const baseAlpha = channels.length === 4 ? Number(channels[3]) : 1\n if (![r, g, b, baseAlpha].every((value) => Number.isFinite(value))) return null\n return `rgba(${r}, ${g}, ${b}, ${baseAlpha * multiplier})`\n}\n\n/**\n * Fallback for wide-gamut color forms (`oklch`, `oklab`, `lab`, `lch`,\n * `hsl`, …) — culori parses every CSS color shape and yields RGB. Lets\n * `color-mix(in oklab, oklch(...) 50%, transparent)` resolve when\n * Tailwind emits the source color in a wide-gamut space (every\n * built-in `bg-red-500` / `shadow-red-500` does, in v4).\n * @param text\n * @param multiplier\n */\nfunction alphaFromCulori(text: string, multiplier: number): string | null {\n try {\n const parsed = culoriRgb(text) as { r?: number; g?: number; b?: number; alpha?: number } | undefined\n if (!parsed) return null\n if (![parsed.r, parsed.g, parsed.b].every((v) => typeof v === 'number' && Number.isFinite(v))) return null\n const r = Math.round(Math.max(0, Math.min(1, parsed.r!)) * 255)\n const g = Math.round(Math.max(0, Math.min(1, parsed.g!)) * 255)\n const b = Math.round(Math.max(0, Math.min(1, parsed.b!)) * 255)\n const baseAlpha = typeof parsed.alpha === 'number' ? parsed.alpha : 1\n return `rgba(${r}, ${g}, ${b}, ${baseAlpha * multiplier})`\n } catch {\n // culori threw on an unrecognised CSS form — fall through.\n return null\n }\n}\n\n/**\n * Expand `#rgb` / `#rrggbb` / `#rrggbbaa` hex to its `{r, g, b, alpha}`\n * components. Returns null when the digit count doesn't match a CSS hex\n * shape.\n * @param digits Hex digits without the leading `#`.\n * @returns Decoded color or null.\n */\nfunction expandHex(digits: string): { r: number; g: number; b: number; alpha: number } | null {\n const {length} = digits\n if (length === 3 || length === 4) {\n const r = Number.parseInt(digits[0]! + digits[0]!, 16)\n const g = Number.parseInt(digits[1]! + digits[1]!, 16)\n const b = Number.parseInt(digits[2]! + digits[2]!, 16)\n const alpha = length === 4 ? Number.parseInt(digits[3]! + digits[3]!, 16) / 255 : 1\n return Number.isNaN(r) || Number.isNaN(g) || Number.isNaN(b) ? null : { r, g, b, alpha }\n }\n if (length === 6 || length === 8) {\n const r = Number.parseInt(digits.slice(0, 2), 16)\n const g = Number.parseInt(digits.slice(2, 4), 16)\n const b = Number.parseInt(digits.slice(4, 6), 16)\n const alpha = length === 8 ? Number.parseInt(digits.slice(6, 8), 16) / 255 : 1\n return Number.isNaN(r) || Number.isNaN(g) || Number.isNaN(b) ? null : { r, g, b, alpha }\n }\n return null\n}\n\n/**\n * Strip the matched outer quote characters from a CSS string literal.\n * `--font-sans: 'Inter-Medium'` flows through `var(--font-sans)`\n * substitution as the raw value text — quotes included. Without this\n * step `fontFamily` lands on the RN style as `\"'Inter-Medium'\"` (with\n * literal quote characters), which RN can't match against the registered\n * native font and silently falls back to the system face.\n *\n * Only strips when both ends agree (`'…'` or `\"…\"`) and there are no\n * other top-level quote chars — keeps multi-segment fallbacks like\n * `'Inter', sans-serif` untouched (those get split downstream).\n * @param text Trimmed CSS value.\n * @returns Same text with outer matching quotes removed, or unchanged.\n */\nfunction unquoteCssString(text: string): string {\n if (text.length < 2) return text\n const first = text.codePointAt(0)\n const last = text.codePointAt(text.length - 1)\n if (first === undefined || first !== last) return text\n if (first !== 34 && first !== 39) return text // \" or '\n const inner = text.slice(1, -1)\n // Don't unquote when the inner string itself contains an unescaped\n // matching quote — that means we'd be merging two adjacent literals.\n if (inner.includes(text[0]!)) return text\n return inner\n}\n\n/**\n * Substitute every `var(--name [, fallback])` reference in `text` with\n * the value from `table` (or the fallback clause when the name misses).\n * Paren-balanced so nested `var(…)` refs don't confuse the scanner.\n * Iterates to a fixed point so multi-hop substitutions land in one call\n * (with a safety cap so a malformed self-referential token can't loop).\n * @param text Raw CSS value.\n * @param table var name → resolved value map.\n * @returns Substituted text.\n */\nexport function substituteThemeVars(text: string, table: ReadonlyMap<string, string>): string {\n let current = text\n for (let pass = 0; pass < 8; pass += 1) {\n const next = substitutePass(current, table)\n if (next === current) return next\n current = next\n }\n return current\n}\n"],"names":["culoriRgb"],"mappings":"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"}
@@ -117,6 +117,13 @@ export declare class TailwindParser {
117
117
  private compiler;
118
118
  private readonly themeSchemes;
119
119
  private readonly schemeAliases;
120
+ /**
121
+ * Scheme names declared via `@custom-variant <name> …;`. A scheme
122
+ * listed here but absent from {@link themeSchemes} (no `@variant`
123
+ * override block) draws its values from the base `@theme` — the
124
+ * standard Tailwind v4 "light defaults + dark override" shape.
125
+ */
126
+ private readonly customVariantSchemes;
120
127
  /**
121
128
  * Memoise `resolveCandidates` results by candidate-list fingerprint.
122
129
  * Fast Refresh hits this on every save: oxide's scan is cheap, but
@@ -138,11 +145,20 @@ export declare class TailwindParser {
138
145
  */
139
146
  constructor(config: TailwindParserConfig);
140
147
  /**
141
- * Schemes declared by the user — either every `@variant <name>` block
142
- * found (in declaration order) or just `['base']` for themes without
143
- * any variants. Used to decide how many per-scheme buckets the
144
- * per-atom resolver fills. Exposed publicly so Metro integration can
145
- * hand the names to the `.d.ts` generator without a full parse.
148
+ * Schemes declared by the user — the union of every `@custom-variant
149
+ * <name>` declaration and every `@variant <name>` block, or just
150
+ * `['base']` for themes without any. Used to decide how many
151
+ * per-scheme buckets the per-atom resolver fills. Exposed publicly so
152
+ * Metro integration can hand the names to the `.d.ts` generator
153
+ * without a full parse.
154
+ *
155
+ * Both sources matter. `@variant` blocks alone miss the common
156
+ * Tailwind v4 shape where the light palette sits in the base `@theme`
157
+ * and only `@variant dark` overrides it: there `light` exists solely
158
+ * as a `@custom-variant` and would otherwise be dropped, collapsing
159
+ * every themed atom to a single bucket that can't switch.
160
+ * `@custom-variant` order wins (it's where users enumerate their
161
+ * schemes); any `@variant`-only scheme is appended after.
146
162
  * @returns Scheme names.
147
163
  */
148
164
  get declaredSchemes(): readonly string[];
@@ -8,7 +8,7 @@ import { detectHapticAtom } from './haptics.mjs';
8
8
  import { keyframesName, keyframeSelectorOffset, pickAnimationName } from './keyframes.mjs';
9
9
  import { serializeInitialValue } from './property.mjs';
10
10
  import { classNameFromSelector } from './selector.mjs';
11
- import { extractThemeVars, extractSchemeAliases, BASE_SCHEME, compileReadyTheme } from './theme-vars.mjs';
11
+ import { extractThemeVars, extractSchemeAliases, extractCustomVariantSchemes, BASE_SCHEME, compileReadyTheme } from './theme-vars.mjs';
12
12
  import { serializeTokens } from './tokens.mjs';
13
13
 
14
14
  /**
@@ -25,14 +25,18 @@ const DEFAULT_TRANSFORM_OPTIONS = {
25
25
  },
26
26
  include: Features.Nesting | Features.MediaQueries,
27
27
  exclude: Features.LogicalProperties | Features.DirSelector | Features.LightDark,
28
- targets: {
29
- // eslint-disable-next-line sonarjs/no-identical-expressions
30
- safari: (16 << 16) | (4 << 8),
31
- // eslint-disable-next-line camelcase, sonarjs/no-identical-expressions
32
- ios_saf: (16 << 16) | (4 << 8),
33
- firefox: 128 << 16,
34
- chrome: 111 << 16,
35
- },
28
+ // NOTE: deliberately no `targets`. With targets that include
29
+ // color-mix-supporting browsers (Safari 16.4+, Chrome 111+, …),
30
+ // lightningcss EVALUATES `color-mix(in oklab, var(--theme-color)
31
+ // <pct>%, transparent)` at parse time using whichever value of
32
+ // `--theme-color` it sees first in the cascade. Tailwind v4 emits
33
+ // exactly this shape for `<prop>-<themed>/<N>` utilities (e.g.
34
+ // `border-text/20`), so the resulting RGB color is locked to ONE
35
+ // scheme — every variant gets the same value. By dropping targets,
36
+ // lightningcss leaves color-mix as an unparsed function and our
37
+ // per-scheme `unparsedToEntries` substitution path runs instead,
38
+ // producing the right rgba(...) for each scheme. Targets in this
39
+ // pipeline are otherwise unused — we never re-emit CSS from the AST.
36
40
  };
37
41
  /**
38
42
  * Parses one source file's Tailwind usage into RN-ready style objects.
@@ -56,6 +60,13 @@ class TailwindParser {
56
60
  compiler;
57
61
  themeSchemes;
58
62
  schemeAliases;
63
+ /**
64
+ * Scheme names declared via `@custom-variant <name> …;`. A scheme
65
+ * listed here but absent from {@link themeSchemes} (no `@variant`
66
+ * override block) draws its values from the base `@theme` — the
67
+ * standard Tailwind v4 "light defaults + dark override" shape.
68
+ */
69
+ customVariantSchemes;
59
70
  /**
60
71
  * Memoise `resolveCandidates` results by candidate-list fingerprint.
61
72
  * Fast Refresh hits this on every save: oxide's scan is cheap, but
@@ -79,19 +90,42 @@ class TailwindParser {
79
90
  this.config = config;
80
91
  this.themeSchemes = extractThemeVars(config.themeCss);
81
92
  this.schemeAliases = extractSchemeAliases(config.themeCss);
93
+ this.customVariantSchemes = extractCustomVariantSchemes(config.themeCss);
82
94
  this.scanner = new Scanner({ sources: config.sources ? [...config.sources] : [] });
83
95
  }
84
96
  /**
85
- * Schemes declared by the user — either every `@variant <name>` block
86
- * found (in declaration order) or just `['base']` for themes without
87
- * any variants. Used to decide how many per-scheme buckets the
88
- * per-atom resolver fills. Exposed publicly so Metro integration can
89
- * hand the names to the `.d.ts` generator without a full parse.
97
+ * Schemes declared by the user — the union of every `@custom-variant
98
+ * <name>` declaration and every `@variant <name>` block, or just
99
+ * `['base']` for themes without any. Used to decide how many
100
+ * per-scheme buckets the per-atom resolver fills. Exposed publicly so
101
+ * Metro integration can hand the names to the `.d.ts` generator
102
+ * without a full parse.
103
+ *
104
+ * Both sources matter. `@variant` blocks alone miss the common
105
+ * Tailwind v4 shape where the light palette sits in the base `@theme`
106
+ * and only `@variant dark` overrides it: there `light` exists solely
107
+ * as a `@custom-variant` and would otherwise be dropped, collapsing
108
+ * every themed atom to a single bucket that can't switch.
109
+ * `@custom-variant` order wins (it's where users enumerate their
110
+ * schemes); any `@variant`-only scheme is appended after.
90
111
  * @returns Scheme names.
91
112
  */
92
113
  get declaredSchemes() {
93
- const variants = [...this.themeSchemes.keys()].filter((name) => name !== BASE_SCHEME);
94
- return variants.length > 0 ? variants : [BASE_SCHEME];
114
+ const ordered = [];
115
+ const seen = new Set();
116
+ for (const name of this.customVariantSchemes) {
117
+ if (seen.has(name))
118
+ continue;
119
+ seen.add(name);
120
+ ordered.push(name);
121
+ }
122
+ for (const name of this.themeSchemes.keys()) {
123
+ if (name === BASE_SCHEME || seen.has(name))
124
+ continue;
125
+ seen.add(name);
126
+ ordered.push(name);
127
+ }
128
+ return ordered.length > 0 ? ordered : [BASE_SCHEME];
95
129
  }
96
130
  /**
97
131
  * Build an effective var table for one scheme — base vars overridden by
@@ -184,6 +218,19 @@ class TailwindParser {
184
218
  catch (error) {
185
219
  throw wrapThemeError(error);
186
220
  }
221
+ // Tailwind v4 emits opacity-suffixed themed colors as a pre-resolved
222
+ // sRGB fallback PLUS a `@supports`-gated var()-based override:
223
+ // border-color: color-mix(in srgb, #0A0A0A 20%, transparent);
224
+ // @supports (color: color-mix(in lab, red, red)) {
225
+ // border-color: color-mix(in oklab, var(--color-text) 20%, transparent);
226
+ // }
227
+ // Lightningcss takes the OUTER fallback (locked to whichever scheme
228
+ // the compiler resolved first), and our per-scheme substitution
229
+ // never gets a chance. Unwrap the @supports so the var()-based
230
+ // declaration overrides the fallback in the same rule — lightningcss
231
+ // emits the override as `unparsed` and the parser's themeVars-aware
232
+ // path produces correct rgba per scheme.
233
+ css = unwrapColorMixSupports(css);
187
234
  // `compiler.build(candidates)` memoizes across calls — it returns CSS for
188
235
  // every candidate the compiler has EVER seen in this process. To keep
189
236
  // parser output pure per-call we restrict outputs to this call's
@@ -916,20 +963,24 @@ function parseFirstShadow(raw) {
916
963
  * @returns Pixel lengths + the remainder text (color expression).
917
964
  */
918
965
  function extractShadowLengths(head) {
919
- const lengthRegex = /(?<![A-Za-z(])-?\d*\.?\d+(?:px|rem|em|%)?/g;
966
+ // Take ONLY the leading run of length tokens, stopping at the first
967
+ // non-length token (the color). A previous global digit-regex scanned
968
+ // the whole string, so a <4-length shadow like `0 1px 1px rgb(0 0 0 /
969
+ // 0.05)` stole a digit out of the color expression — corrupting the
970
+ // alpha (opacity) or a digit-leading hex. Whitespace-splitting can't
971
+ // reach inside the color because we break as soon as a token isn't a
972
+ // bare/`px`/`rem`/`em`/`%` length.
973
+ // Unambiguous integer-or-decimal (no `\d*\.?\d+` overlap) so there's no
974
+ // super-linear backtracking on long digit runs.
975
+ const isLength = /^-?(?:\d+(?:\.\d+)?|\.\d+)(?:px|rem|em|%)?$/;
976
+ const parts = head.split(/\s+/);
920
977
  const lengths = [];
921
- const matches = [];
922
- let next = lengthRegex.exec(head);
923
- while (next !== null && lengths.length < 4) {
924
- matches.push({ text: next[0], index: next.index });
925
- lengths.push(parseLengthToken(next[0]));
926
- next = lengthRegex.exec(head);
927
- }
928
- let remainder = head;
929
- for (const { text, index } of matches.toReversed()) {
930
- remainder = `${remainder.slice(0, index)}${remainder.slice(index + text.length)}`;
978
+ let index = 0;
979
+ while (index < parts.length && lengths.length < 4 && isLength.test(parts[index])) {
980
+ lengths.push(parseLengthToken(parts[index]));
981
+ index += 1;
931
982
  }
932
- return { lengths, remainder };
983
+ return { lengths, remainder: parts.slice(index).join(' ') };
933
984
  }
934
985
  /**
935
986
  * Coerce one shadow length token into a pixel number. Accepts bare
@@ -1468,6 +1519,68 @@ function resolveAngleExpression(text) {
1468
1519
  * @param css Tailwind's compiled CSS.
1469
1520
  * @returns Map of custom-property name → resolved value.
1470
1521
  */
1522
+ /**
1523
+ * Strip `\@supports (color: color-mix(in lab, red, red)) { … }` wrappers
1524
+ * from Tailwind v4's compiled CSS, hoisting their inner declarations up
1525
+ * to the parent rule.
1526
+ *
1527
+ * Tailwind emits opacity-suffixed themed colors with both a pre-resolved
1528
+ * sRGB fallback AND a var()-based override gated behind the color-mix
1529
+ * `\@supports` clause. The OUTER fallback hard-codes a single scheme's
1530
+ * value of the theme token; the inner override is var()-based and
1531
+ * substitutes correctly per scheme. By unwrapping the gate, the inner
1532
+ * declaration becomes a sibling of the fallback in the same rule body —
1533
+ * lightningcss takes the LATER one (the var()-based unparsed form), and
1534
+ * the parser's themeVars-aware path produces correct rgba per scheme.
1535
+ * Modern RN-targeted browsers all support color-mix anyway, so dropping
1536
+ * the gating is safe.
1537
+ * @param css Tailwind-compiled CSS.
1538
+ * @returns CSS with the color-mix support gates unwrapped.
1539
+ */
1540
+ function unwrapColorMixSupports(css) {
1541
+ const guard = '@supports (color: color-mix(in lab, red, red))';
1542
+ let out = '';
1543
+ let cursor = 0;
1544
+ while (cursor < css.length) {
1545
+ const head = css.indexOf(guard, cursor);
1546
+ if (head === -1) {
1547
+ out += css.slice(cursor);
1548
+ break;
1549
+ }
1550
+ out += css.slice(cursor, head);
1551
+ const brace = css.indexOf('{', head);
1552
+ if (brace === -1) {
1553
+ out += css.slice(head);
1554
+ break;
1555
+ }
1556
+ const blockEnd = findMatchingClose(css, brace + 1);
1557
+ if (blockEnd === -1) {
1558
+ out += css.slice(head);
1559
+ break;
1560
+ }
1561
+ const inner = css.slice(brace + 1, blockEnd);
1562
+ // Only unwrap when the gated declaration substitutes a USER theme
1563
+ // token (`var(--color-…)`). Tailwind also gates `--tw-*` internal
1564
+ // composers (shadow color, ring color, …) on the same supports
1565
+ // clause; their outer fallback is the optimized hex/oklch value
1566
+ // the parser's own composed-prop pass needs (`applyComposedShadow`
1567
+ // reads `--tw-shadow-color` from the rule's local vars). Unwrapping
1568
+ // them would replace the resolvable color with an unresolvable
1569
+ // `color-mix(... var(--tw-shadow-alpha), transparent)` text and
1570
+ // break the composed-shadow path.
1571
+ // Keep the gate intact for non-themed colors — the outer fallback
1572
+ // wins, which is what Tailwind intended.
1573
+ out += inner.includes('var(--color-') ? inner : css.slice(head, blockEnd + 1);
1574
+ cursor = blockEnd + 1;
1575
+ }
1576
+ return out;
1577
+ }
1578
+ /**
1579
+ * Extract every `--name: value` declaration from the `:root` blocks in
1580
+ * Tailwind's compiled CSS into a flat map.
1581
+ * @param css Tailwind-compiled CSS.
1582
+ * @returns Map of custom-property name → resolved value.
1583
+ */
1471
1584
  function extractRootCustomProperties(css) {
1472
1585
  const out = new Map();
1473
1586
  let cursor = 0;