@pyreon/styler 0.11.10 → 0.12.1
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/lib/index.d.ts +11 -8
- package/lib/index.d.ts.map +1 -0
- package/lib/index.js +24 -15
- package/lib/index.js.map +1 -0
- package/package.json +4 -5
- package/src/__tests__/composition-chain.test.ts +1 -1
- package/src/__tests__/hybrid-injection.test.ts +24 -29
- package/src/__tests__/sheet-advanced.test.ts +12 -22
- package/src/__tests__/sheet-split-atrules.test.ts +16 -16
- package/src/__tests__/sheet.test.ts +7 -6
- package/src/__tests__/styled-ssr.test.ts +1 -1
- package/src/__tests__/styled.test.ts +3 -3
- package/src/sheet.ts +27 -13
- package/src/styled.tsx +8 -8
package/lib/index.d.ts
CHANGED
|
@@ -149,10 +149,13 @@ declare class StyleSheet {
|
|
|
149
149
|
* Deduplicates: same CSS text always produces the same class name and
|
|
150
150
|
* the rules are only injected once.
|
|
151
151
|
*
|
|
152
|
-
*
|
|
153
|
-
*
|
|
152
|
+
* @param cssText - CSS declarations to insert
|
|
153
|
+
* @param _unused - Reserved for backward compatibility (was `boost`)
|
|
154
|
+
* @param insertLayer - CSS @layer to wrap this rule in (e.g. 'rocketstyle').
|
|
155
|
+
* Used by rocketstyle to ensure wrapper styles override inner component styles
|
|
156
|
+
* via @layer order (base < rocketstyle) instead of specificity hacks.
|
|
154
157
|
*/
|
|
155
|
-
insert(cssText: string,
|
|
158
|
+
insert(cssText: string, _unused?: boolean, insertLayer?: string): string;
|
|
156
159
|
/** Insert a @keyframes rule. Deduplicates by animation name. */
|
|
157
160
|
insertKeyframes(name: string, body: string): void;
|
|
158
161
|
/**
|
|
@@ -178,7 +181,7 @@ declare class StyleSheet {
|
|
|
178
181
|
/**
|
|
179
182
|
* Compute className and full CSS rule text without injecting.
|
|
180
183
|
*/
|
|
181
|
-
prepare(cssText: string
|
|
184
|
+
prepare(cssText: string): {
|
|
182
185
|
className: string;
|
|
183
186
|
rules: string;
|
|
184
187
|
};
|
|
@@ -201,11 +204,11 @@ interface StyledOptions {
|
|
|
201
204
|
/** Custom prop filter. Return true to forward the prop to the DOM element. */
|
|
202
205
|
shouldForwardProp?: (prop: string) => boolean;
|
|
203
206
|
/**
|
|
204
|
-
*
|
|
205
|
-
*
|
|
206
|
-
*
|
|
207
|
+
* CSS @layer name. Rules are wrapped in `@layer <name> { ... }`.
|
|
208
|
+
* Used by rocketstyle to ensure wrapper styles override inner component
|
|
209
|
+
* styles via layer order (base < rocketstyle) instead of specificity hacks.
|
|
207
210
|
*/
|
|
208
|
-
|
|
211
|
+
layer?: string;
|
|
209
212
|
}
|
|
210
213
|
type TagTemplateFn = (strings: TemplateStringsArray, ...values: Interpolation[]) => ComponentFn;
|
|
211
214
|
type HtmlTags = 'a' | 'abbr' | 'address' | 'article' | 'aside' | 'audio' | 'b' | 'blockquote' | 'body' | 'br' | 'button' | 'canvas' | 'caption' | 'code' | 'col' | 'colgroup' | 'dd' | 'details' | 'div' | 'dl' | 'dt' | 'em' | 'fieldset' | 'figcaption' | 'figure' | 'footer' | 'form' | 'h1' | 'h2' | 'h3' | 'h4' | 'h5' | 'h6' | 'head' | 'header' | 'hr' | 'html' | 'i' | 'iframe' | 'img' | 'input' | 'label' | 'legend' | 'li' | 'link' | 'main' | 'mark' | 'menu' | 'meta' | 'nav' | 'ol' | 'optgroup' | 'option' | 'output' | 'p' | 'picture' | 'pre' | 'progress' | 'q' | 'section' | 'select' | 'small' | 'source' | 'span' | 'strong' | 'style' | 'sub' | 'summary' | 'sup' | 'svg' | 'table' | 'tbody' | 'td' | 'template' | 'textarea' | 'tfoot' | 'th' | 'thead' | 'time' | 'tr' | 'u' | 'ul' | 'video';
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index2.d.ts","names":[],"sources":["../../src/ThemeProvider.ts","../../src/resolve.ts","../../src/css.ts","../../src/forward.ts","../../src/globalStyle.ts","../../src/hash.ts","../../src/keyframes.ts","../../src/shared.ts","../../src/sheet.ts","../../src/styled.tsx","../../src/useCSS.ts"],"mappings":"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"}
|
package/lib/index.js
CHANGED
|
@@ -440,9 +440,14 @@ var StyleSheet = class {
|
|
|
440
440
|
document.head.appendChild(el);
|
|
441
441
|
this.sheet = el.sheet ?? null;
|
|
442
442
|
}
|
|
443
|
-
if (this.
|
|
444
|
-
|
|
445
|
-
|
|
443
|
+
if (this.sheet) {
|
|
444
|
+
try {
|
|
445
|
+
this.sheet.insertRule("@layer base, rocketstyle;", 0);
|
|
446
|
+
} catch {}
|
|
447
|
+
if (this.layer) try {
|
|
448
|
+
this.sheet.insertRule(`@layer ${this.layer};`, 1);
|
|
449
|
+
} catch {}
|
|
450
|
+
}
|
|
446
451
|
}
|
|
447
452
|
/** Extract className from a selector like ".pyr-abc" or ".pyr-abc.pyr-abc" → "pyr-abc" */
|
|
448
453
|
extractClassName(selectorText) {
|
|
@@ -542,11 +547,14 @@ var StyleSheet = class {
|
|
|
542
547
|
* Deduplicates: same CSS text always produces the same class name and
|
|
543
548
|
* the rules are only injected once.
|
|
544
549
|
*
|
|
545
|
-
*
|
|
546
|
-
*
|
|
550
|
+
* @param cssText - CSS declarations to insert
|
|
551
|
+
* @param _unused - Reserved for backward compatibility (was `boost`)
|
|
552
|
+
* @param insertLayer - CSS @layer to wrap this rule in (e.g. 'rocketstyle').
|
|
553
|
+
* Used by rocketstyle to ensure wrapper styles override inner component styles
|
|
554
|
+
* via @layer order (base < rocketstyle) instead of specificity hacks.
|
|
547
555
|
*/
|
|
548
|
-
insert(cssText,
|
|
549
|
-
const icKey =
|
|
556
|
+
insert(cssText, _unused = false, insertLayer) {
|
|
557
|
+
const icKey = insertLayer ? `${cssText}\0L:${insertLayer}` : cssText;
|
|
550
558
|
const icHit = this.insertCache.get(icKey);
|
|
551
559
|
if (icHit) return icHit;
|
|
552
560
|
const className = `${PREFIX}-${hash(cssText)}`;
|
|
@@ -556,12 +564,13 @@ var StyleSheet = class {
|
|
|
556
564
|
}
|
|
557
565
|
this.evictIfNeeded();
|
|
558
566
|
this.cache.set(className, className);
|
|
559
|
-
const selector =
|
|
567
|
+
const selector = `.${className}`;
|
|
560
568
|
const { base, atRules } = this.splitAtRules(cssText, selector);
|
|
561
569
|
const rules = [];
|
|
562
570
|
if (base) rules.push(`${selector}{${base}}`);
|
|
563
571
|
rules.push(...atRules);
|
|
564
|
-
const
|
|
572
|
+
const layerName = insertLayer ?? this.layer;
|
|
573
|
+
const finalRules = layerName ? rules.map((r) => `@layer ${layerName}{${r}}`) : rules;
|
|
565
574
|
if (this.isSSR) for (const rule of finalRules) this.ssrBuffer.push(rule);
|
|
566
575
|
else if (this.sheet) for (const rule of finalRules) try {
|
|
567
576
|
this.sheet.insertRule(rule, this.sheet.cssRules.length);
|
|
@@ -656,9 +665,9 @@ var StyleSheet = class {
|
|
|
656
665
|
/**
|
|
657
666
|
* Compute className and full CSS rule text without injecting.
|
|
658
667
|
*/
|
|
659
|
-
prepare(cssText
|
|
668
|
+
prepare(cssText) {
|
|
660
669
|
const className = `${PREFIX}-${hash(cssText)}`;
|
|
661
|
-
const selector =
|
|
670
|
+
const selector = `.${className}`;
|
|
662
671
|
const { base, atRules } = this.splitAtRules(cssText, selector);
|
|
663
672
|
const allRules = [];
|
|
664
673
|
if (base) allRules.push(`${selector}{${base}}`);
|
|
@@ -774,10 +783,10 @@ const createStyledComponent = (tag, strings, values, options) => {
|
|
|
774
783
|
}
|
|
775
784
|
const hasDynamicValues = values.length > 0 && values.some(isDynamic);
|
|
776
785
|
const customFilter = options ? options.shouldForwardProp : void 0;
|
|
777
|
-
const
|
|
786
|
+
const insertLayer = options?.layer;
|
|
778
787
|
if (!hasDynamicValues) {
|
|
779
788
|
const cssText = normalizeCSS(values.length === 0 ? strings[0] : resolve(strings, values, {}));
|
|
780
|
-
const staticClassName = cssText.length > 0 ? sheet.insert(cssText,
|
|
789
|
+
const staticClassName = cssText.length > 0 ? sheet.insert(cssText, false, insertLayer) : "";
|
|
781
790
|
const StaticStyled = (rawProps) => {
|
|
782
791
|
const finalTag = rawProps.as || tag;
|
|
783
792
|
return h(finalTag, buildProps(rawProps, staticClassName, typeof finalTag === "string", customFilter), ...Array.isArray(rawProps.children) ? rawProps.children : rawProps.children != null ? [rawProps.children] : []);
|
|
@@ -810,7 +819,7 @@ const createStyledComponent = (tag, strings, values, options) => {
|
|
|
810
819
|
...isReactiveState ? { $rocketstate: resolvedState } : {},
|
|
811
820
|
theme
|
|
812
821
|
}));
|
|
813
|
-
const initialClassName = cssText.length > 0 ? sheet.insert(cssText,
|
|
822
|
+
const initialClassName = cssText.length > 0 ? sheet.insert(cssText, false, insertLayer) : "";
|
|
814
823
|
const finalTag = rawProps.as || tag;
|
|
815
824
|
const isDOM = typeof finalTag === "string";
|
|
816
825
|
let el = null;
|
|
@@ -837,7 +846,7 @@ const createStyledComponent = (tag, strings, values, options) => {
|
|
|
837
846
|
$rocketstate: newState,
|
|
838
847
|
theme
|
|
839
848
|
}));
|
|
840
|
-
const newClass = newCss.length > 0 ? sheet.insert(newCss,
|
|
849
|
+
const newClass = newCss.length > 0 ? sheet.insert(newCss, false, insertLayer) : "";
|
|
841
850
|
if (el && newClass !== currentClassName) {
|
|
842
851
|
if (currentClassName) el.classList.remove(currentClassName);
|
|
843
852
|
if (newClass) el.classList.add(newClass);
|
package/lib/index.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","names":[],"sources":["../src/resolve.ts","../src/css.ts","../src/forward.ts","../src/shared.ts","../src/hash.ts","../src/sheet.ts","../src/ThemeProvider.ts","../src/globalStyle.ts","../src/keyframes.ts","../src/styled.tsx","../src/useCSS.ts"],"sourcesContent":["/**\n * Interpolation resolver: converts tagged template strings + values into a\n * final CSS string. Handles nested CSSResults, arrays, functions, and\n * primitive values.\n */\n\nimport type { DefaultTheme } from './ThemeProvider'\n\nexport type Interpolation =\n | string\n | number\n | boolean\n | null\n | undefined\n | CSSResult\n | Interpolation[]\n | ((props: { theme?: DefaultTheme & Record<string, any>; [key: string]: any }) => Interpolation)\n\n/**\n * Lazy representation of a `css` tagged template. Stores the raw template\n * strings and interpolation values without resolving them. Resolution is\n * deferred until a styled component renders (or until explicitly resolved).\n */\nexport class CSSResult {\n constructor(\n readonly strings: TemplateStringsArray,\n readonly values: Interpolation[],\n ) {}\n\n /** Resolve with empty props — useful for static templates, testing, and debugging. */\n toString(): string {\n return resolve(this.strings, this.values, {})\n }\n}\n\n/** Resolve a tagged template's strings + values into a final CSS string. */\nexport const resolve = (\n strings: TemplateStringsArray,\n values: Interpolation[],\n props: Record<string, any>,\n): string => {\n // Tagged templates guarantee strings.length === values.length + 1,\n // so strings[0] and strings[i+1] are always defined — no ?? needed.\n let result = strings[0] as string\n for (let i = 0; i < values.length; i++) {\n const v = values[i]\n const s = strings[i + 1] as string\n // Inline the most common value types to avoid function call overhead.\n if (typeof v === 'function') {\n const r = v(props)\n result +=\n (typeof r === 'string'\n ? r\n : r == null || r === false || r === true\n ? ''\n : resolveValue(r as Interpolation, props)) + s\n } else if (v == null || v === false || v === true) {\n result += s\n } else if (typeof v === 'string') {\n result += v + s\n } else if (typeof v === 'number') {\n result += v + s\n } else {\n result += resolveValue(v, props) + s\n }\n }\n return result\n}\n\n/**\n * Normalize resolved CSS text for strict `insertRule` compatibility.\n *\n * Single-pass scanner that handles all cleanup in one traversal:\n * - Strips block comments and line comments (preserves :// in URLs)\n * - Collapses whitespace to single spaces\n * - Removes redundant semicolons\n * - Trims leading/trailing whitespace\n */\nconst normCache = new Map<string, string>()\n/** Clear the normalizeCSS cache (called during HMR cleanup). */\nexport const clearNormCache = () => normCache.clear()\n\nexport const normalizeCSS = (css: string): string => {\n const cached = normCache.get(css)\n if (cached !== undefined) return cached\n\n const len = css.length\n let out = ''\n let space = false // pending space to emit before next non-whitespace char\n let last = 0 // charCode of last char written to output (0 = nothing yet)\n\n for (let i = 0; i < len; i++) {\n const c = css.charCodeAt(i)\n\n // /* block comment */\n if (c === 47 /* / */ && css.charCodeAt(i + 1) === 42 /* * */) {\n const end = css.indexOf('*/', i + 2)\n i = end === -1 ? len : end + 1\n space = true\n continue\n }\n\n // // line comment (but not :// in URLs)\n if (c === 47 /* / */ && css.charCodeAt(i + 1) === 47 /* / */ && last !== 58 /* : */) {\n const nl = css.indexOf('\\n', i + 2)\n i = nl === -1 ? len : nl\n space = true\n continue\n }\n\n // Whitespace → collapse\n if (c === 32 || c === 9 || c === 10 || c === 13 || c === 12) {\n space = true\n continue\n }\n\n // Semicolon → skip if redundant (after start, {, }, or another ;)\n if (c === 59 /* ; */) {\n if (last === 0 || last === 123 /* { */ || last === 125 /* } */ || last === 59 /* ; */) {\n continue\n }\n space = false\n out += ';'\n last = 59\n continue\n }\n\n // Regular char — emit pending space (but not at start of output)\n if (space && last !== 0) out += ' '\n space = false\n\n out += css[i]\n last = c\n }\n\n // Evict oldest ~10% to prevent memory leaks without cliff-edge drop\n if (normCache.size > 2000) {\n let count = 0\n for (const key of normCache.keys()) {\n if (count >= 200) break\n normCache.delete(key)\n count++\n }\n }\n normCache.set(css, out)\n\n return out\n}\n\nexport const resolveValue = (value: Interpolation, props: Record<string, any>): string => {\n // null, undefined, false, true → empty (enables conditional: ${cond && css`...`})\n if (value == null || value === false || value === true) return ''\n\n // function interpolation → call with props/theme context, resolve result\n if (typeof value === 'function') return resolveValue(value(props) as Interpolation, props)\n\n // nested CSSResult → recursively resolve\n if (value instanceof CSSResult) return resolve(value.strings, value.values, props)\n\n // array of results (e.g. from makeItResponsive's breakpoints.map())\n if (Array.isArray(value)) {\n let arrayResult = ''\n for (let i = 0; i < value.length; i++) {\n arrayResult += resolveValue(value[i], props)\n }\n return arrayResult\n }\n\n return String(value)\n}\n","import { CSSResult, type Interpolation } from './resolve'\n\n/**\n * Tagged template function for CSS. Captures the template strings and\n * interpolation values as a lazy CSSResult — resolution is deferred\n * until a styled component renders.\n *\n * Works as both a tagged template (`css\\`...\\``) and a regular function\n * call (`css(...args)`) since tagged templates are syntactic sugar for\n * function calls with (TemplateStringsArray, ...values).\n */\nexport const css = (strings: TemplateStringsArray, ...values: Interpolation[]): CSSResult =>\n new CSSResult(strings, values)\n","/**\n * HTML prop filtering. Prevents unknown props from being forwarded to DOM\n * elements (which causes warnings). Props starting with `$` are\n * transient (styling-only) and are always filtered out.\n */\n\n// Common HTML attributes, event handlers, and ARIA/data attributes\nconst HTML_PROPS = new Set([\n // Core props\n 'className',\n 'class',\n 'dangerouslySetInnerHTML',\n 'htmlFor',\n 'id',\n 'key',\n 'ref',\n 'style',\n 'tabIndex',\n 'role',\n // Event handlers\n 'onAbort',\n 'onAnimationEnd',\n 'onAnimationIteration',\n 'onAnimationStart',\n 'onBlur',\n 'onChange',\n 'onClick',\n 'onCompositionEnd',\n 'onCompositionStart',\n 'onCompositionUpdate',\n 'onContextMenu',\n 'onCopy',\n 'onCut',\n 'onDoubleClick',\n 'onDrag',\n 'onDragEnd',\n 'onDragEnter',\n 'onDragLeave',\n 'onDragOver',\n 'onDragStart',\n 'onDrop',\n 'onError',\n 'onFocus',\n 'onInput',\n 'onKeyDown',\n 'onKeyPress',\n 'onKeyUp',\n 'onLoad',\n 'onMouseDown',\n 'onMouseEnter',\n 'onMouseLeave',\n 'onMouseMove',\n 'onMouseOut',\n 'onMouseOver',\n 'onMouseUp',\n 'onPaste',\n 'onPointerCancel',\n 'onPointerDown',\n 'onPointerEnter',\n 'onPointerLeave',\n 'onPointerMove',\n 'onPointerOut',\n 'onPointerOver',\n 'onPointerUp',\n 'onScroll',\n 'onSelect',\n 'onSubmit',\n 'onTouchCancel',\n 'onTouchEnd',\n 'onTouchMove',\n 'onTouchStart',\n 'onTransitionEnd',\n 'onWheel',\n // HTML attributes\n 'accept',\n 'acceptCharset',\n 'accessKey',\n 'action',\n 'allow',\n 'allowFullScreen',\n 'alt',\n 'as',\n 'async',\n 'autoCapitalize',\n 'autoComplete',\n 'autoCorrect',\n 'autoFocus',\n 'autoPlay',\n 'capture',\n 'cellPadding',\n 'cellSpacing',\n 'charSet',\n 'checked',\n 'cite',\n 'cols',\n 'colSpan',\n 'content',\n 'contentEditable',\n 'controls',\n 'controlsList',\n 'coords',\n 'crossOrigin',\n 'dateTime',\n 'decoding',\n 'default',\n 'defaultChecked',\n 'defaultValue',\n 'defer',\n 'dir',\n 'disabled',\n 'disablePictureInPicture',\n 'disableRemotePlayback',\n 'download',\n 'draggable',\n 'encType',\n 'enterKeyHint',\n 'fetchPriority',\n 'form',\n 'formAction',\n 'formEncType',\n 'formMethod',\n 'formNoValidate',\n 'formTarget',\n 'frameBorder',\n 'headers',\n 'height',\n 'hidden',\n 'high',\n 'href',\n 'hrefLang',\n 'httpEquiv',\n 'inputMode',\n 'integrity',\n 'is',\n 'label',\n 'lang',\n 'list',\n 'loading',\n 'loop',\n 'low',\n 'max',\n 'maxLength',\n 'media',\n 'method',\n 'min',\n 'minLength',\n 'multiple',\n 'muted',\n 'name',\n 'noModule',\n 'noValidate',\n 'nonce',\n 'open',\n 'optimum',\n 'pattern',\n 'placeholder',\n 'playsInline',\n 'poster',\n 'preload',\n 'readOnly',\n 'referrerPolicy',\n 'rel',\n 'required',\n 'reversed',\n 'rows',\n 'rowSpan',\n 'sandbox',\n 'scope',\n 'scoped',\n 'scrolling',\n 'selected',\n 'shape',\n 'size',\n 'sizes',\n 'slot',\n 'span',\n 'spellCheck',\n 'src',\n 'srcDoc',\n 'srcLang',\n 'srcSet',\n 'start',\n 'step',\n 'summary',\n 'target',\n 'title',\n 'translate',\n 'type',\n 'useMap',\n 'value',\n 'width',\n 'wrap',\n])\n\n/**\n * Filters props for HTML elements. Keeps valid HTML attrs, data-*, aria-*.\n * Rejects unknown props and $-prefixed transient props.\n */\nexport const filterProps = (props: Record<string, unknown>): Record<string, unknown> => {\n const filtered: Record<string, unknown> = {}\n\n for (const key in props) {\n // Skip transient props ($-prefixed) — used for styling-only props\n if (key.charCodeAt(0) === 36) continue // '$'\n\n // Skip `as` prop — handled separately by styled\n if (key === 'as') continue\n\n // Keep data-* and aria-* attributes\n if (key.startsWith('data-') || key.startsWith('aria-')) {\n filtered[key] = props[key]\n continue\n }\n\n // Keep known HTML props\n if (HTML_PROPS.has(key)) {\n filtered[key] = props[key]\n }\n }\n\n return filtered\n}\n\n/**\n * Build final props for a styled component in a single pass.\n * Combines className merging, ref injection, and prop filtering into one\n * allocation and one iteration.\n */\nexport const buildProps = (\n rawProps: Record<string, any>,\n generatedCls: string,\n isDOM: boolean,\n customFilter?: (prop: string) => boolean,\n): Record<string, any> => {\n const result: Record<string, any> = {}\n\n // Merge generated + user className\n const userCls = rawProps.class || rawProps.className\n if (generatedCls) {\n result.class = userCls ? `${generatedCls} ${userCls}` : generatedCls\n } else if (userCls) {\n result.class = userCls\n }\n\n // Component target — forward all props except as/className/class and $-prefixed\n if (!isDOM) {\n for (const key in rawProps) {\n if (key === 'as' || key === 'className' || key === 'class') continue\n if (key.charCodeAt(0) === 36) continue // $-prefixed transient\n result[key] = rawProps[key]\n }\n return result\n }\n\n // DOM element with custom shouldForwardProp\n if (customFilter) {\n for (const key in rawProps) {\n if (key === 'as' || key === 'className' || key === 'class') continue\n if (customFilter(key)) result[key] = rawProps[key]\n }\n return result\n }\n\n // DOM element with default filtering\n for (const key in rawProps) {\n if (key === 'as' || key === 'className' || key === 'class') continue\n if (key.charCodeAt(0) === 36) continue // $-prefixed transient\n if (key.startsWith('data-') || key.startsWith('aria-')) {\n result[key] = rawProps[key]\n continue\n }\n if (HTML_PROPS.has(key)) result[key] = rawProps[key]\n }\n return result\n}\n","/**\n * Shared utilities used across multiple modules.\n */\nimport { CSSResult, type Interpolation } from './resolve'\n\n/** Check if an interpolation value is dynamic (contains functions or nested dynamic CSSResults). */\nexport const isDynamic = (v: Interpolation): boolean => {\n if (typeof v === 'function') return true\n if (Array.isArray(v)) return v.some(isDynamic)\n if (v instanceof CSSResult) return v.values.some(isDynamic)\n return false\n}\n","/**\n * Fast FNV-1a non-cryptographic hash. Returns base-36 string for compact class names.\n *\n * 32-bit hash space → ~4.3 billion unique values. Collision probability is\n * negligible for typical applications (< 10,000 unique CSS rules).\n */\n\n/** FNV-1a offset basis — starting state for streaming hash. */\nexport const HASH_INIT = 2166136261\n\nconst FNV_PRIME = 16777619\n\n/**\n * Feed a string segment into the running hash state.\n * Streaming: hashUpdate(hashUpdate(HASH_INIT, 'ab'), 'cd') === hash('abcd').\n */\nexport const hashUpdate = (init: number, str: string): number => {\n let h = init\n for (let i = 0; i < str.length; i++) {\n h ^= str.charCodeAt(i)\n h = Math.imul(h, FNV_PRIME)\n }\n return h\n}\n\n/** Finalize a hash state into a base-36 class name suffix. */\nexport const hashFinalize = (h: number): string => (h >>> 0).toString(36)\n\n/** Hash a complete string in one shot. Returns base-36 string. */\nexport const hash = (str: string): string => hashFinalize(hashUpdate(HASH_INIT, str))\n","/**\n * StyleSheet manager. Handles CSS rule injection, hash-based deduplication,\n * SSR buffering, client-side hydration, bounded cache, and @layer support.\n *\n * Media queries (@media), @supports, and @container blocks nested inside\n * component CSS are automatically extracted into separate top-level rules.\n */\nimport { hash } from './hash'\nimport { clearNormCache } from './resolve'\n\nconst PREFIX = 'pyr'\nconst ATTR = 'data-pyreon-styler'\nconst DEFAULT_MAX_CACHE_SIZE = 10000\n\nexport interface StyleSheetOptions {\n /** Maximum number of cached rules before eviction (default: 10000). */\n maxCacheSize?: number\n /** CSS @layer name to wrap scoped rules in. */\n layer?: string\n}\n\nexport class StyleSheet {\n private cache = new Map<string, string>()\n private insertCache = new Map<string, string>()\n private sheet: CSSStyleSheet | null = null\n private ssrBuffer: string[] = []\n private isSSR: boolean\n private maxCacheSize: number\n private layer: string | undefined\n\n constructor(options: StyleSheetOptions = {}) {\n this.maxCacheSize = options.maxCacheSize ?? DEFAULT_MAX_CACHE_SIZE\n this.layer = options.layer\n this.isSSR = typeof document === 'undefined'\n if (!this.isSSR) this.mount()\n }\n\n private mount() {\n // Reuse existing <style> tag from SSR hydration\n const existing = document.querySelector(`style[${ATTR}]`) as HTMLStyleElement | null\n\n if (existing) {\n this.sheet = existing.sheet ?? null\n this.hydrateFromTag(existing)\n } else {\n const el = document.createElement('style')\n el.setAttribute(ATTR, '')\n document.head.appendChild(el)\n this.sheet = el.sheet ?? null\n }\n\n // Inject @layer declarations.\n // 'base' is for plain styled() components, 'rocketstyle' is for\n // rocketstyle wrappers. Layer order ensures rocketstyle overrides base\n // without needing doubled selectors (boost).\n if (this.sheet) {\n try {\n this.sheet.insertRule('@layer base, rocketstyle;', 0)\n } catch {\n // @layer not supported — falls back to source order\n }\n if (this.layer) {\n try {\n this.sheet.insertRule(`@layer ${this.layer};`, 1)\n } catch {\n // skip\n }\n }\n }\n }\n\n /** Extract className from a selector like \".pyr-abc\" or \".pyr-abc.pyr-abc\" → \"pyr-abc\" */\n private extractClassName(selectorText: string): string | null {\n if (selectorText[0] !== '.') return null\n const dotIdx = selectorText.indexOf('.', 1)\n return dotIdx > 0 ? selectorText.slice(1, dotIdx) : selectorText.slice(1)\n }\n\n /** Parse existing rules from SSR-rendered <style> tag into cache. */\n private hydrateFromTag(el: HTMLStyleElement) {\n const sheet = el.sheet\n if (!sheet) return\n\n for (let i = 0; i < sheet.cssRules.length; i++) {\n const rule = sheet.cssRules[i]\n\n if (rule instanceof CSSStyleRule) {\n const className = this.extractClassName(rule.selectorText)\n if (className) this.cache.set(className, className)\n }\n\n // Handle split @media rules that wrap our selectors\n if (typeof CSSMediaRule !== 'undefined' && rule instanceof CSSMediaRule) {\n for (let j = 0; j < rule.cssRules.length; j++) {\n const inner = rule.cssRules[j]\n if (inner instanceof CSSStyleRule) {\n const className = this.extractClassName(inner.selectorText)\n if (className) this.cache.set(className, className)\n }\n }\n }\n }\n }\n\n /** Evict oldest entries when cache exceeds max size. */\n private evictIfNeeded() {\n if (this.cache.size <= this.maxCacheSize) return\n\n // Map iteration order is insertion order — delete oldest 10%\n const toDelete = Math.floor(this.maxCacheSize * 0.1)\n let count = 0\n for (const key of this.cache.keys()) {\n if (count >= toDelete) break\n this.cache.delete(key)\n count++\n }\n }\n\n /**\n * Extract nested at-rules (@media, @supports, @container) from CSS text\n * and wrap their content in the given selector as separate top-level rules.\n */\n private splitAtRules(cssText: string, selector: string): { base: string; atRules: string[] } {\n // Fast path: no at-rules to split\n if (cssText.indexOf('@') === -1) return { base: cssText, atRules: [] }\n\n const atRules: string[] = []\n const baseParts: string[] = []\n let depth = 0\n let atStart = -1\n let lastBase = 0\n\n for (let i = 0; i < cssText.length; i++) {\n const ch = cssText[i]\n\n if (ch === '{') {\n depth++\n } else if (ch === '}') {\n depth--\n if (depth === 0 && atStart >= 0) {\n // End of a tracked at-rule block — extract and wrap with selector\n const openBrace = cssText.indexOf('{', atStart)\n const atPrefix = cssText.slice(atStart, openBrace).trim()\n const innerCSS = cssText.slice(openBrace + 1, i).trim()\n if (innerCSS) {\n atRules.push(`${atPrefix}{${selector}{${innerCSS}}}`)\n }\n atStart = -1\n lastBase = i + 1\n }\n } else if (depth === 0 && ch === '@' && atStart < 0) {\n // Check if this starts a splittable at-rule (not @keyframes, @font-face, etc.)\n const remaining = cssText.slice(i, i + 20)\n if (/^@(?:media|supports|container)\\b/.test(remaining)) {\n // Save any base CSS that precedes this at-rule\n const baseBefore = cssText.slice(lastBase, i).trim()\n if (baseBefore) baseParts.push(baseBefore)\n atStart = i\n }\n }\n }\n\n // Collect remaining base CSS after the last at-rule\n if (lastBase < cssText.length && atStart < 0) {\n const remaining = cssText.slice(lastBase).trim()\n if (remaining) baseParts.push(remaining)\n }\n\n // If no at-rules were found, return original unchanged\n if (atRules.length === 0) return { base: cssText, atRules: [] }\n\n return { base: baseParts.join(' '), atRules }\n }\n\n /**\n * Compute a className from CSS text without injecting (pure function).\n */\n getClassName(cssText: string): string {\n const cached = this.insertCache.get(cssText)\n if (cached) return cached\n const h = hash(cssText)\n return `${PREFIX}-${h}`\n }\n\n /**\n * Insert CSS rules for a component. Returns the class name (deterministic, hash-based).\n * Deduplicates: same CSS text always produces the same class name and\n * the rules are only injected once.\n *\n * @param cssText - CSS declarations to insert\n * @param _unused - Reserved for backward compatibility (was `boost`)\n * @param insertLayer - CSS @layer to wrap this rule in (e.g. 'rocketstyle').\n * Used by rocketstyle to ensure wrapper styles override inner component styles\n * via @layer order (base < rocketstyle) instead of specificity hacks.\n */\n insert(cssText: string, _unused = false, insertLayer?: string): string {\n // Fast path: skip hash computation on repeated insertions of same CSS text\n const icKey = insertLayer ? `${cssText}\\0L:${insertLayer}` : cssText\n const icHit = this.insertCache.get(icKey)\n if (icHit) return icHit\n\n const h = hash(cssText)\n const className = `${PREFIX}-${h}`\n\n if (this.cache.has(className)) {\n this.insertCache.set(icKey, className)\n return className\n }\n\n this.evictIfNeeded()\n this.cache.set(className, className)\n\n const selector = `.${className}`\n\n // Split nested at-rules into separate top-level rules\n const { base, atRules } = this.splitAtRules(cssText, selector)\n\n const rules: string[] = []\n if (base) rules.push(`${selector}{${base}}`)\n rules.push(...atRules)\n\n // Apply @layer wrapping — per-insert layer takes precedence over sheet-level layer\n const layerName = insertLayer ?? this.layer\n const finalRules = layerName ? rules.map((r) => `@layer ${layerName}{${r}}`) : rules\n\n if (this.isSSR) {\n for (const rule of finalRules) {\n this.ssrBuffer.push(rule)\n }\n } else if (this.sheet) {\n for (const rule of finalRules) {\n try {\n this.sheet.insertRule(rule, this.sheet.cssRules.length)\n } catch (_e) {\n if (process.env.NODE_ENV !== 'production') {\n // oxlint-disable-next-line no-console\n console.warn('[styler] Failed to insert CSS rule:', rule, _e)\n }\n }\n }\n }\n\n this.insertCache.set(icKey, className)\n return className\n }\n\n /** Insert a @keyframes rule. Deduplicates by animation name. */\n insertKeyframes(name: string, body: string): void {\n if (this.cache.has(name)) return\n\n this.evictIfNeeded()\n this.cache.set(name, name)\n\n const rule = `@keyframes ${name}{${body}}`\n\n if (this.isSSR) {\n this.ssrBuffer.push(rule)\n } else if (this.sheet) {\n try {\n this.sheet.insertRule(rule, this.sheet.cssRules.length)\n } catch (_e) {\n if (process.env.NODE_ENV !== 'production') {\n // silently ignore invalid CSS rules in production\n }\n }\n }\n }\n\n /**\n * Split CSS text into individual top-level rules.\n * CSSStyleSheet.insertRule() only accepts one rule at a time.\n */\n private splitRules(cssText: string): string[] {\n const rules: string[] = []\n let depth = 0\n let start = 0\n\n for (let i = 0; i < cssText.length; i++) {\n const ch = cssText[i]\n if (ch === '{') depth++\n else if (ch === '}') {\n depth--\n if (depth === 0) {\n const rule = cssText.slice(start, i + 1).trim()\n if (rule) rules.push(rule)\n start = i + 1\n }\n }\n }\n\n return rules\n }\n\n /** Insert global CSS rules (no wrapper selector). Deduplicates by hash. */\n insertGlobal(cssText: string): void {\n const h = hash(cssText)\n const key = `global-${h}`\n\n if (this.cache.has(key)) return\n\n this.evictIfNeeded()\n this.cache.set(key, key)\n\n if (this.isSSR) {\n this.ssrBuffer.push(cssText)\n } else if (this.sheet) {\n const rules = this.splitRules(cssText)\n for (const rule of rules) {\n try {\n this.sheet.insertRule(rule, this.sheet.cssRules.length)\n } catch (_e) {\n if (process.env.NODE_ENV !== 'production') {\n // oxlint-disable-next-line no-console\n console.warn('[styler] Failed to insert global CSS rule:', rule, _e)\n }\n }\n }\n }\n }\n\n /** Returns collected CSS for SSR as a complete `<style>` tag string. */\n getStyleTag(): string {\n const css = this.ssrBuffer.join('').replace(/<\\/style/gi, '<\\\\/style')\n return `<style ${ATTR}=\"\">${css}</style>`\n }\n\n /** Returns collected CSS rules as a raw string (useful for streaming SSR). */\n getStyles(): string {\n return this.ssrBuffer.join('')\n }\n\n /** Reset SSR buffer and cache (call between server requests). */\n reset(): void {\n this.ssrBuffer = []\n this.cache.clear()\n this.insertCache.clear()\n }\n\n /** Clear the dedup cache. Useful for HMR / dev-time reloads. */\n clearCache(): void {\n this.cache.clear()\n this.insertCache.clear()\n clearNormCache()\n }\n\n /**\n * Full cleanup: clear cache and remove all CSS rules from the DOM.\n * Intended for HMR / dev-time reloads where stale styles must be purged.\n */\n clearAll(): void {\n this.cache.clear()\n this.insertCache.clear()\n clearNormCache()\n this.ssrBuffer = []\n if (this.sheet) {\n while (this.sheet.cssRules.length > 0) {\n this.sheet.deleteRule(0)\n }\n }\n }\n\n /**\n * Compute className and full CSS rule text without injecting.\n */\n prepare(cssText: string): { className: string; rules: string } {\n const h = hash(cssText)\n const className = `${PREFIX}-${h}`\n const selector = `.${className}`\n const { base, atRules } = this.splitAtRules(cssText, selector)\n\n const allRules: string[] = []\n if (base) allRules.push(`${selector}{${base}}`)\n allRules.push(...atRules)\n\n const finalRules = this.layer ? allRules.map((r) => `@layer ${this.layer}{${r}}`) : allRules\n\n return { className, rules: finalRules.join('') }\n }\n\n /** Check if a className is already in the cache. O(1) Map lookup. */\n has(className: string): boolean {\n return this.cache.has(className)\n }\n\n /** Current number of cached rules. */\n get cacheSize(): number {\n return this.cache.size\n }\n}\n\n/** Default singleton sheet for client-side use. */\nexport const sheet = new StyleSheet()\n\n/**\n * Factory for creating isolated StyleSheet instances.\n * Use in SSR to get per-request isolation.\n */\nexport const createSheet = (options?: StyleSheetOptions): StyleSheet => new StyleSheet(options)\n","/**\n * Theme context for styled components.\n *\n * Extensible theme interface. Consumers can augment this via module\n * declaration merging for full strict types:\n *\n * declare module '@pyreon/styler' {\n * interface DefaultTheme {\n * colors: { primary: string; secondary: string }\n * spacing: (n: number) => string\n * }\n * }\n */\nimport type { VNode, VNodeChild } from '@pyreon/core'\nimport { createContext, provide, useContext } from '@pyreon/core'\n\nexport interface DefaultTheme {}\n\ntype Theme = DefaultTheme & Record<string, unknown>\n\nexport const ThemeContext = createContext<Theme>({} as Theme)\n\n/** Hook to read the current theme from the nearest ThemeProvider. */\nexport const useTheme = <T extends object = Theme>(): T => useContext(ThemeContext) as T\n\n/**\n * @internal Low-level provider — use `PyreonUI` from `@pyreon/ui-core` instead.\n *\n * Provides a theme object to all nested styled components via Pyreon context.\n *\n * @deprecated Prefer `<PyreonUI theme={theme}>` which provides theme to\n * all three context layers (styler, core, mode) in one component.\n */\nexport function ThemeProvider({\n theme,\n children,\n}: {\n theme: Theme\n children?: VNodeChild\n}): VNode | null {\n provide(ThemeContext, theme)\n return (children ?? null) as VNode | null\n}\n","/**\n * createGlobalStyle() — tagged template function that injects global CSS\n * rules (not scoped to a class name). Returns a component function that\n * injects styles when called and supports dynamic interpolations via\n * props/theme.\n *\n * Usage:\n * const GlobalStyle = createGlobalStyle`\n * body { margin: 0; font-family: ${({ theme }) => theme.font}; }\n * *, *::before, *::after { box-sizing: border-box; }\n * `\n */\nimport type { ComponentFn } from '@pyreon/core'\nimport { type Interpolation, normalizeCSS, resolve } from './resolve'\nimport { isDynamic } from './shared'\nimport { sheet } from './sheet'\nimport { useTheme } from './ThemeProvider'\n\nexport const createGlobalStyle = (\n strings: TemplateStringsArray,\n ...values: Interpolation[]\n): ComponentFn => {\n const hasDynamicValues = values.some(isDynamic)\n\n // STATIC FAST PATH: compute once at creation time\n if (!hasDynamicValues) {\n const cssText = normalizeCSS(resolve(strings, values, {}))\n\n // Inject into sheet immediately\n if (cssText.trim()) sheet.insertGlobal(cssText)\n\n const StaticGlobal: ComponentFn = () => null\n return StaticGlobal\n }\n\n // DYNAMIC PATH: resolve on every render with theme/props\n const DynamicGlobal: ComponentFn = (props: Record<string, any>) => {\n const theme = useTheme()\n const allProps = { ...props, theme }\n const cssText = normalizeCSS(resolve(strings, values, allProps))\n\n if (cssText.trim()) sheet.insertGlobal(cssText)\n\n return null\n }\n\n return DynamicGlobal\n}\n","/**\n * keyframes() tagged template function. Creates a CSS @keyframes rule,\n * injects it into the stylesheet, and returns the generated animation name.\n *\n * Usage:\n * const fadeIn = keyframes`\n * from { opacity: 0; }\n * to { opacity: 1; }\n * `\n * // fadeIn === \"pyr-kf-abc123\" (deterministic, hash-based)\n */\nimport { hash } from './hash'\nimport { type Interpolation, normalizeCSS, resolve } from './resolve'\nimport { sheet } from './sheet'\n\nclass KeyframesResult {\n readonly name: string\n\n constructor(strings: TemplateStringsArray, values: Interpolation[]) {\n const body = normalizeCSS(resolve(strings, values, {}))\n const h = hash(body)\n this.name = `pyr-kf-${h}`\n\n sheet.insertKeyframes(this.name, body)\n }\n\n /** Returns the animation name when used in string context. */\n toString(): string {\n return this.name\n }\n}\n\nexport const keyframes = (\n strings: TemplateStringsArray,\n ...values: Interpolation[]\n): KeyframesResult => new KeyframesResult(strings, values)\n","/**\n * styled() component factory. Creates Pyreon components that inject CSS\n * class names from tagged template literals.\n *\n * Supports:\n * - styled('div')`...` and styled(Component)`...`\n * - styled.div`...` (via Proxy)\n * - `as` prop for polymorphic rendering\n * - $-prefixed transient props (not forwarded to DOM)\n * - Custom shouldForwardProp for per-component prop filtering\n * - Static path optimization (templates with no dynamic interpolations)\n * - Boost specificity via doubled selector\n *\n * CSS nesting (`&` selectors) works natively — the resolver passes CSS\n * through without transformation, so `&:hover`, `&::before`, etc. work\n * as-is in browsers supporting CSS Nesting (all modern browsers).\n */\nimport type { ComponentFn, VNode } from '@pyreon/core'\nimport { h } from '@pyreon/core'\nimport { effect, runUntracked } from '@pyreon/reactivity'\nimport { buildProps } from './forward'\nimport { type Interpolation, normalizeCSS, resolve } from './resolve'\nimport { isDynamic } from './shared'\nimport { sheet } from './sheet'\nimport { useTheme } from './ThemeProvider'\n\ntype Tag = string | ComponentFn<any>\n\nexport interface StyledOptions {\n /** Custom prop filter. Return true to forward the prop to the DOM element. */\n shouldForwardProp?: (prop: string) => boolean\n /**\n * CSS @layer name. Rules are wrapped in `@layer <name> { ... }`.\n * Used by rocketstyle to ensure wrapper styles override inner component\n * styles via layer order (base < rocketstyle) instead of specificity hacks.\n */\n layer?: string\n}\n\nconst getDisplayName = (tag: Tag): string =>\n typeof tag === 'string'\n ? tag\n : (tag as ComponentFn<any> & { displayName?: string }).displayName || tag.name || 'Component'\n\n// Component cache: same template literal + tag + no options → same component.\n// WeakMap on `strings` (TemplateStringsArray is object-identity per source location).\nconst staticComponentCache = new WeakMap<TemplateStringsArray, Map<Tag, ComponentFn>>()\n\n// Single-entry hot cache — just 3 reference comparisons, no Map/WeakMap overhead.\nlet _hotStrings: TemplateStringsArray | null = null\nlet _hotTag: Tag | null = null\nlet _hotComponent: ComponentFn | null = null\n\nconst createStyledComponent = (\n tag: Tag,\n strings: TemplateStringsArray,\n values: Interpolation[],\n options?: StyledOptions,\n): ComponentFn => {\n // Ultra-fast hot cache: 3 reference comparisons → return immediately\n if (values.length === 0 && !options) {\n if (strings === _hotStrings && tag === _hotTag) return _hotComponent as ComponentFn\n\n // WeakMap fallback for alternating patterns\n const tagMap = staticComponentCache.get(strings)\n if (tagMap) {\n const cached = tagMap.get(tag)\n if (cached) {\n _hotStrings = strings\n _hotTag = tag\n _hotComponent = cached\n return cached\n }\n }\n }\n\n // Fast check: no values means no dynamic interpolations — avoids .some() scan\n const hasDynamicValues = values.length > 0 && values.some(isDynamic)\n const customFilter = options ? options.shouldForwardProp : undefined\n const insertLayer = options?.layer\n\n // STATIC FAST PATH: no function interpolations → compute class once at creation time\n if (!hasDynamicValues) {\n // Inline resolve for the common no-values case\n const raw = values.length === 0 ? (strings[0] as string) : resolve(strings, values, {})\n const cssText = normalizeCSS(raw)\n const hasCss = cssText.length > 0\n\n const staticClassName = hasCss ? sheet.insert(cssText, false, insertLayer) : ''\n\n const StaticStyled: ComponentFn = (rawProps: Record<string, any>): VNode | null => {\n const finalTag = rawProps.as || tag\n const isDOM = typeof finalTag === 'string'\n const finalProps = buildProps(rawProps, staticClassName, isDOM, customFilter)\n\n return h(\n finalTag as string,\n finalProps,\n ...(Array.isArray(rawProps.children)\n ? rawProps.children\n : rawProps.children != null\n ? [rawProps.children]\n : []),\n )\n }\n\n ;(StaticStyled as ComponentFn & { displayName?: string }).displayName =\n `styled(${getDisplayName(tag)})`\n\n // Store in component cache + hot cache for future reuse\n if (!options && values.length === 0) {\n let tagMap = staticComponentCache.get(strings)\n if (!tagMap) {\n tagMap = new Map()\n staticComponentCache.set(strings, tagMap)\n }\n tagMap.set(tag, StaticStyled)\n _hotStrings = strings\n _hotTag = tag\n _hotComponent = StaticStyled\n }\n\n return StaticStyled\n }\n\n // DYNAMIC PATH: resolve CSS on every render with theme/props.\n // When $rocketstyle is a function accessor (from rocketstyle's\n // EnhancedComponent), we resolve it initially for the first class,\n // then set up an effect to reactively swap classes when mode changes.\n // This avoids VNode remounting — only classList updates on the DOM element.\n const DynamicStyled: ComponentFn = (rawProps: Record<string, any>): VNode | null => {\n const theme = useTheme()\n const $rs = rawProps.$rocketstyle\n const $rsState = rawProps.$rocketstate\n const isReactiveRS = typeof $rs === 'function'\n const isReactiveState = typeof $rsState === 'function'\n\n // Resolve initial accessor values — both $rocketstyle and $rocketstate\n // must be plain objects when passed to resolve(), because .styles()\n // interpolation functions destructure them directly:\n // const { hover, pressed } = $rocketstate.pseudo\n // const { hover: hoverStyles } = $rocketstyle\n const resolvedRS = isReactiveRS ? $rs() : $rs\n const resolvedState = isReactiveState ? $rsState() : $rsState\n const initialProps = {\n ...rawProps,\n ...(isReactiveRS ? { $rocketstyle: resolvedRS } : {}),\n ...(isReactiveState ? { $rocketstate: resolvedState } : {}),\n }\n const cssText = normalizeCSS(resolve(strings, values, { ...initialProps, theme }))\n const initialClassName = cssText.length > 0 ? sheet.insert(cssText, false, insertLayer) : ''\n\n const finalTag = rawProps.as || tag\n const isDOM = typeof finalTag === 'string'\n\n // Track mounted element for reactive class updates\n let el: Element | null = null\n let currentClassName = initialClassName\n\n const originalRef = rawProps.ref\n const refCallback = (node: Element | null) => {\n el = node\n if (originalRef) {\n if (typeof originalRef === 'function') originalRef(node)\n else if (originalRef && typeof originalRef === 'object') originalRef.current = node\n }\n }\n\n const finalProps = buildProps(\n { ...rawProps, ref: isReactiveRS ? refCallback : rawProps.ref },\n initialClassName,\n isDOM,\n customFilter,\n )\n\n // Set up reactive class swap when $rocketstyle is a function accessor.\n // CRITICAL: only $rs() is tracked. resolve() and DOM mutations run\n // inside runUntracked() to prevent subscribing to signals read by\n // interpolation functions (theme properties, context getters, etc.).\n // Without this, every styled component's effect subscribes to every\n // signal touched during CSS resolution — causing an exponential\n // cascade across 50+ components on any signal change.\n if (isReactiveRS) {\n effect(() => {\n const newRS = $rs() // TRACKED: subscribes to mode + dimension signals\n const newState = isReactiveState ? $rsState() : $rsState // TRACKED\n\n runUntracked(() => {\n // UNTRACKED: resolve + DOM mutation — no additional subscriptions\n const newResolvedProps = {\n ...rawProps,\n $rocketstyle: newRS,\n $rocketstate: newState,\n }\n const newCss = normalizeCSS(\n resolve(strings, values, { ...newResolvedProps, theme }),\n )\n const newClass = newCss.length > 0 ? sheet.insert(newCss, false, insertLayer) : ''\n\n if (el && newClass !== currentClassName) {\n if (currentClassName) el.classList.remove(currentClassName)\n if (newClass) el.classList.add(newClass)\n currentClassName = newClass\n }\n })\n })\n }\n\n // STATIC VNode — created once, never remounted on mode change\n return h(\n finalTag as string,\n finalProps,\n ...(Array.isArray(rawProps.children)\n ? rawProps.children\n : rawProps.children != null\n ? [rawProps.children]\n : []),\n )\n }\n\n ;(DynamicStyled as ComponentFn & { displayName?: string }).displayName =\n `styled(${getDisplayName(tag)})`\n return DynamicStyled\n}\n\n/** Factory function: styled(tag) returns a tagged template function. */\nconst styledFactory = (tag: Tag, options?: StyledOptions) => {\n const templateFn = (strings: TemplateStringsArray, ...values: Interpolation[]) =>\n createStyledComponent(tag, strings, values, options)\n\n return templateFn\n}\n\n/**\n * Main styled export. Supports both calling conventions:\n * - `styled('div')` or `styled(Component)` → returns tagged template function\n * - `styled('div', { shouldForwardProp })` → with custom prop filtering\n * - `styled.div` → shorthand via Proxy (no options)\n */\n// Cache template functions per tag to avoid closure allocation on every Proxy get\nconst proxyCache = new Map<string, (...args: any[]) => any>()\n\ntype TagTemplateFn = (strings: TemplateStringsArray, ...values: Interpolation[]) => ComponentFn\n\ntype HtmlTags =\n | 'a'\n | 'abbr'\n | 'address'\n | 'article'\n | 'aside'\n | 'audio'\n | 'b'\n | 'blockquote'\n | 'body'\n | 'br'\n | 'button'\n | 'canvas'\n | 'caption'\n | 'code'\n | 'col'\n | 'colgroup'\n | 'dd'\n | 'details'\n | 'div'\n | 'dl'\n | 'dt'\n | 'em'\n | 'fieldset'\n | 'figcaption'\n | 'figure'\n | 'footer'\n | 'form'\n | 'h1'\n | 'h2'\n | 'h3'\n | 'h4'\n | 'h5'\n | 'h6'\n | 'head'\n | 'header'\n | 'hr'\n | 'html'\n | 'i'\n | 'iframe'\n | 'img'\n | 'input'\n | 'label'\n | 'legend'\n | 'li'\n | 'link'\n | 'main'\n | 'mark'\n | 'menu'\n | 'meta'\n | 'nav'\n | 'ol'\n | 'optgroup'\n | 'option'\n | 'output'\n | 'p'\n | 'picture'\n | 'pre'\n | 'progress'\n | 'q'\n | 'section'\n | 'select'\n | 'small'\n | 'source'\n | 'span'\n | 'strong'\n | 'style'\n | 'sub'\n | 'summary'\n | 'sup'\n | 'svg'\n | 'table'\n | 'tbody'\n | 'td'\n | 'template'\n | 'textarea'\n | 'tfoot'\n | 'th'\n | 'thead'\n | 'time'\n | 'tr'\n | 'u'\n | 'ul'\n | 'video'\n\nexport type StyledFunction = ((tag: Tag, options?: StyledOptions) => TagTemplateFn) & {\n [K in HtmlTags]: TagTemplateFn\n}\n\n// Proxy is needed to support styled.div`...` syntax; the cast bridges\n// styledFactory's call signature to StyledFunction which adds HTML tag properties.\n// Proxy target uses `as any` because TS can't resolve Proxy<StyledFunction> with mapped types\nexport const styled: StyledFunction = new Proxy(styledFactory as any, {\n get(_target: unknown, prop: string) {\n if (prop === 'prototype' || prop === '$$typeof') return undefined\n // styled.div`...`, styled.span`...`, etc.\n let fn = proxyCache.get(prop)\n if (!fn) {\n fn = (strings: TemplateStringsArray, ...values: Interpolation[]) =>\n createStyledComponent(prop, strings, values)\n proxyCache.set(prop, fn)\n }\n return fn\n },\n})\n","/**\n * Hook that resolves a CSSResult template with props, injects CSS\n * into the shared stylesheet, and returns the className.\n *\n * Use this when you need computed CSS class names on plain elements\n * without the overhead of a styled component layer.\n */\nimport { type CSSResult, normalizeCSS, resolve } from './resolve'\nimport { sheet } from './sheet'\nimport { useTheme } from './ThemeProvider'\n\nexport function useCSS(template: CSSResult, props?: Record<string, any>, boost?: boolean): string {\n const theme = useTheme()\n const allProps = theme ? { ...props, theme } : (props ?? {})\n const cssText = normalizeCSS(resolve(template.strings, template.values, allProps))\n\n if (!cssText.trim()) return ''\n\n return sheet.insert(cssText, boost)\n}\n"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@pyreon/styler",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.12.1",
|
|
4
4
|
"description": "Lightweight CSS-in-JS engine for Pyreon",
|
|
5
5
|
"license": "MIT",
|
|
6
6
|
"repository": {
|
|
@@ -10,7 +10,6 @@
|
|
|
10
10
|
},
|
|
11
11
|
"files": [
|
|
12
12
|
"lib",
|
|
13
|
-
"!lib/**/*.map",
|
|
14
13
|
"!lib/analysis",
|
|
15
14
|
"README.md",
|
|
16
15
|
"LICENSE",
|
|
@@ -41,12 +40,12 @@
|
|
|
41
40
|
"typecheck": "tsc --noEmit"
|
|
42
41
|
},
|
|
43
42
|
"devDependencies": {
|
|
44
|
-
"@pyreon/typescript": "^0.
|
|
43
|
+
"@pyreon/typescript": "^0.12.1",
|
|
45
44
|
"@vitus-labs/tools-rolldown": "^1.15.3"
|
|
46
45
|
},
|
|
47
46
|
"peerDependencies": {
|
|
48
|
-
"@pyreon/core": "^0.
|
|
49
|
-
"@pyreon/reactivity": "^0.
|
|
47
|
+
"@pyreon/core": "^0.12.1",
|
|
48
|
+
"@pyreon/reactivity": "^0.12.1"
|
|
50
49
|
},
|
|
51
50
|
"engines": {
|
|
52
51
|
"node": ">= 22"
|
|
@@ -345,7 +345,7 @@ describe('resolve composition chain', () => {
|
|
|
345
345
|
describe('styled component composition', () => {
|
|
346
346
|
it('handles array of functions as single interpolation (calculateStyles pattern)', () => {
|
|
347
347
|
// This is EXACTLY what rocketstyle does:
|
|
348
|
-
// styled(component, {
|
|
348
|
+
// styled(component, { layer: 'rocketstyle' })`${calculateStyles(styles)};`
|
|
349
349
|
// calculateStyles returns an array of function results
|
|
350
350
|
|
|
351
351
|
const fn1 = (props: any) => `position: ${props.$rocketstyle?.position ?? 'static'};`
|
|
@@ -2,14 +2,14 @@
|
|
|
2
2
|
* Tests for the hybrid injection approach:
|
|
3
3
|
* - Client (jsdom): shared <style data-pyreon-styler> sheet
|
|
4
4
|
* - CSS rules present in the CSSOM sheet after insertion
|
|
5
|
-
* - `
|
|
5
|
+
* - `layer` option threaded from styled() through to the sheet
|
|
6
6
|
*
|
|
7
7
|
* Ported to VNode-level testing: we call the component function directly
|
|
8
8
|
* and inspect the returned VNode + the sheet's CSSOM.
|
|
9
9
|
*/
|
|
10
10
|
|
|
11
11
|
import type { VNode } from '@pyreon/core'
|
|
12
|
-
import { afterEach, describe, expect, it } from 'vitest'
|
|
12
|
+
import { afterEach, describe, expect, it, vi } from 'vitest'
|
|
13
13
|
import { createGlobalStyle } from '../globalStyle'
|
|
14
14
|
import { sheet } from '../sheet'
|
|
15
15
|
import { styled } from '../styled'
|
|
@@ -160,37 +160,38 @@ describe('hybrid injection — VNode output (no <style> in tree)', () => {
|
|
|
160
160
|
})
|
|
161
161
|
})
|
|
162
162
|
|
|
163
|
-
describe('hybrid injection —
|
|
163
|
+
describe('hybrid injection — layer option at component level', () => {
|
|
164
164
|
afterEach(() => {
|
|
165
165
|
sheet.clearAll()
|
|
166
166
|
})
|
|
167
167
|
|
|
168
|
-
it('static
|
|
169
|
-
const
|
|
168
|
+
it('static layered component generates className', () => {
|
|
169
|
+
const warnSpy = vi.spyOn(console, 'warn').mockImplementation(() => undefined)
|
|
170
|
+
const Comp = styled('div', { layer: 'rocketstyle' })`
|
|
170
171
|
color: red;
|
|
171
172
|
`
|
|
172
173
|
const vnode = Comp({}) as VNode
|
|
173
174
|
const className = vnode.props.class as string
|
|
174
175
|
|
|
175
|
-
|
|
176
|
-
expect(
|
|
177
|
-
|
|
178
|
-
expect(rules.some((r) => r.includes(`.${className}.${className}`))).toBe(true)
|
|
176
|
+
// className is generated regardless of CSSOM @layer support
|
|
177
|
+
expect(className).toMatch(/^pyr-/)
|
|
178
|
+
warnSpy.mockRestore()
|
|
179
179
|
})
|
|
180
180
|
|
|
181
|
-
it('dynamic
|
|
182
|
-
const
|
|
181
|
+
it('dynamic layered component generates className', () => {
|
|
182
|
+
const warnSpy = vi.spyOn(console, 'warn').mockImplementation(() => undefined)
|
|
183
|
+
const Comp = styled('div', { layer: 'rocketstyle' })`
|
|
183
184
|
color: ${(p: any) => p.$color};
|
|
184
185
|
`
|
|
185
186
|
const vnode = Comp({ $color: 'blue' }) as VNode
|
|
186
187
|
const className = vnode.props.class as string
|
|
187
188
|
|
|
188
|
-
|
|
189
|
-
expect(
|
|
190
|
-
|
|
189
|
+
// className is generated regardless of CSSOM @layer support
|
|
190
|
+
expect(className).toMatch(/^pyr-/)
|
|
191
|
+
warnSpy.mockRestore()
|
|
191
192
|
})
|
|
192
193
|
|
|
193
|
-
it('non-
|
|
194
|
+
it('non-layered component produces single selector without @layer', () => {
|
|
194
195
|
const Comp = styled('div')`
|
|
195
196
|
color: green;
|
|
196
197
|
`
|
|
@@ -199,17 +200,16 @@ describe('hybrid injection — boost option at component level', () => {
|
|
|
199
200
|
|
|
200
201
|
const rules = findRulesFor(className)
|
|
201
202
|
expect(rules.length).toBeGreaterThanOrEqual(1)
|
|
202
|
-
// Single selector: .pyr-abc { ... }
|
|
203
|
+
// Single selector: .pyr-abc { ... }
|
|
203
204
|
const baseRule = rules[0] as string
|
|
204
205
|
expect(baseRule).toContain(`.${className}`)
|
|
205
|
-
//
|
|
206
|
-
|
|
207
|
-
const occurrences = selectorPart.split(`.${className}`).length - 1
|
|
208
|
-
expect(occurrences).toBe(1)
|
|
206
|
+
// No @layer wrapping
|
|
207
|
+
expect(baseRule).not.toContain('@layer')
|
|
209
208
|
})
|
|
210
209
|
|
|
211
|
-
it('
|
|
212
|
-
const
|
|
210
|
+
it('layered component with @media generates className', () => {
|
|
211
|
+
const warnSpy = vi.spyOn(console, 'warn').mockImplementation(() => undefined)
|
|
212
|
+
const Comp = styled('div', { layer: 'rocketstyle' })`
|
|
213
213
|
color: red;
|
|
214
214
|
@media (min-width: 768px) {
|
|
215
215
|
font-size: 20px;
|
|
@@ -218,12 +218,7 @@ describe('hybrid injection — boost option at component level', () => {
|
|
|
218
218
|
const vnode = Comp({}) as VNode
|
|
219
219
|
const className = vnode.props.class as string
|
|
220
220
|
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
expect(rules.length).toBeGreaterThanOrEqual(2)
|
|
224
|
-
// Both base and media rule should use doubled selector
|
|
225
|
-
for (const rule of rules) {
|
|
226
|
-
expect(rule).toContain(`.${className}.${className}`)
|
|
227
|
-
}
|
|
221
|
+
expect(className).toMatch(/^pyr-/)
|
|
222
|
+
warnSpy.mockRestore()
|
|
228
223
|
})
|
|
229
224
|
})
|
|
@@ -221,25 +221,15 @@ describe('StyleSheet -- advanced features', () => {
|
|
|
221
221
|
})
|
|
222
222
|
})
|
|
223
223
|
|
|
224
|
-
describe('
|
|
225
|
-
it('
|
|
224
|
+
describe('prepare()', () => {
|
|
225
|
+
it('produces single selector', () => {
|
|
226
226
|
document.querySelectorAll('style[data-pyreon-styler]').forEach((el) => {
|
|
227
227
|
el.remove()
|
|
228
228
|
})
|
|
229
229
|
const s = createSheet()
|
|
230
|
-
const result = s.prepare('color: red;'
|
|
230
|
+
const result = s.prepare('color: red;')
|
|
231
231
|
expect(result.className).toMatch(/^pyr-/)
|
|
232
|
-
//
|
|
233
|
-
expect(result.rules).toContain(`.${result.className}.${result.className}`)
|
|
234
|
-
})
|
|
235
|
-
|
|
236
|
-
it('single selector when boost is false', () => {
|
|
237
|
-
document.querySelectorAll('style[data-pyreon-styler]').forEach((el) => {
|
|
238
|
-
el.remove()
|
|
239
|
-
})
|
|
240
|
-
const s = createSheet()
|
|
241
|
-
const result = s.prepare('color: red;', false)
|
|
242
|
-
// Non-boosted: single selector
|
|
232
|
+
// Single selector, no doubling
|
|
243
233
|
expect(result.rules).toContain(`.${result.className}{`)
|
|
244
234
|
expect(result.rules).not.toContain(`.${result.className}.${result.className}`)
|
|
245
235
|
})
|
|
@@ -275,23 +265,23 @@ describe('StyleSheet -- advanced features', () => {
|
|
|
275
265
|
})
|
|
276
266
|
})
|
|
277
267
|
|
|
278
|
-
describe('
|
|
268
|
+
describe('layer on insert()', () => {
|
|
279
269
|
beforeEach(() => {
|
|
280
270
|
document.querySelectorAll('style[data-pyreon-styler]').forEach((el) => {
|
|
281
271
|
el.remove()
|
|
282
272
|
})
|
|
283
273
|
})
|
|
284
274
|
|
|
285
|
-
it('inserts
|
|
275
|
+
it('inserts rule with layer on client side', () => {
|
|
286
276
|
const s = createSheet()
|
|
287
|
-
const cls = s.insert('color: red;',
|
|
277
|
+
const cls = s.insert('color: red;', false, 'rocketstyle')
|
|
288
278
|
expect(cls).toMatch(/^pyr-/)
|
|
289
279
|
})
|
|
290
280
|
|
|
291
|
-
it('deduplicates
|
|
281
|
+
it('deduplicates layered and non-layered separately via insertCache key', () => {
|
|
292
282
|
const s = createSheet()
|
|
293
283
|
const cls1 = s.insert('color: green;', false)
|
|
294
|
-
const cls2 = s.insert('color: green;',
|
|
284
|
+
const cls2 = s.insert('color: green;', false, 'rocketstyle')
|
|
295
285
|
// Same className (same hash) but both should work without error
|
|
296
286
|
expect(cls1).toBe(cls2)
|
|
297
287
|
})
|
|
@@ -606,11 +596,11 @@ describe('StyleSheet -- advanced features', () => {
|
|
|
606
596
|
expect(result.rules).toContain('color: red;')
|
|
607
597
|
})
|
|
608
598
|
|
|
609
|
-
it('prepare()
|
|
599
|
+
it('prepare() wraps in @layer with layer option', () => {
|
|
610
600
|
const s = createSheet({ layer: 'lib' })
|
|
611
|
-
const result = s.prepare('color: blue;'
|
|
601
|
+
const result = s.prepare('color: blue;')
|
|
612
602
|
expect(result.rules).toContain('@layer lib')
|
|
613
|
-
expect(result.rules).toContain(`.${result.className}
|
|
603
|
+
expect(result.rules).toContain(`.${result.className}`)
|
|
614
604
|
})
|
|
615
605
|
})
|
|
616
606
|
|
|
@@ -74,16 +74,16 @@ describe('StyleSheet -- at-rule splitting', () => {
|
|
|
74
74
|
expect(styles).toMatch(/@media \(min-width: 1024px\)\{\.pyr-[0-9a-z]+\{color: green;\}\}/)
|
|
75
75
|
})
|
|
76
76
|
|
|
77
|
-
it('
|
|
78
|
-
const s = createSheet()
|
|
79
|
-
s.insert('color: red; @media (min-width: 768px){color: blue;}'
|
|
77
|
+
it('layer wraps both base and media rules in @layer', () => {
|
|
78
|
+
const s = createSheet({ layer: 'rocketstyle' })
|
|
79
|
+
s.insert('color: red; @media (min-width: 768px){color: blue;}')
|
|
80
80
|
const styles = s.getStyles()
|
|
81
81
|
|
|
82
|
-
// Base: .pyr-xxx
|
|
83
|
-
expect(styles).toMatch(
|
|
84
|
-
// Media: @media (...){.pyr-xxx
|
|
82
|
+
// Base wrapped in @layer: @layer rocketstyle{.pyr-xxx{color: red;}}
|
|
83
|
+
expect(styles).toMatch(/@layer rocketstyle\{\.pyr-[0-9a-z]+\{color: red;\}\}/)
|
|
84
|
+
// Media wrapped in @layer: @layer rocketstyle{@media (...){.pyr-xxx{color: blue;}}}
|
|
85
85
|
expect(styles).toMatch(
|
|
86
|
-
/@media \(min-width: 768px\)\{\.pyr-[0-9a-z]
|
|
86
|
+
/@layer rocketstyle\{@media \(min-width: 768px\)\{\.pyr-[0-9a-z]+\{color: blue;\}\}\}/,
|
|
87
87
|
)
|
|
88
88
|
})
|
|
89
89
|
|
|
@@ -139,7 +139,7 @@ describe('StyleSheet -- at-rule splitting', () => {
|
|
|
139
139
|
'@media only screen and (min-width: 48em){bottom: 0; height: 40rem;} ' +
|
|
140
140
|
'@media only screen and (min-width: 62em){right: -6.25rem;} ' +
|
|
141
141
|
'@media only screen and (min-width: 100em){right: initial; left: 55%;}'
|
|
142
|
-
s.insert(cssStr
|
|
142
|
+
s.insert(cssStr)
|
|
143
143
|
const styles = s.getStyles()
|
|
144
144
|
|
|
145
145
|
// Base rule has position, bottom, right, height
|
|
@@ -213,27 +213,27 @@ describe('StyleSheet -- at-rule splitting', () => {
|
|
|
213
213
|
expect(hasMediaRule).toBe(true)
|
|
214
214
|
})
|
|
215
215
|
|
|
216
|
-
it('
|
|
216
|
+
it('single selector appears in both base and media rules', () => {
|
|
217
217
|
const s = createSheet()
|
|
218
|
-
const className = s.insert('color: red; @media (min-width: 768px){color: blue;}'
|
|
218
|
+
const className = s.insert('color: red; @media (min-width: 768px){color: blue;}')
|
|
219
219
|
|
|
220
220
|
const styleEl = document.querySelector('style[data-pyreon-styler]') as HTMLStyleElement
|
|
221
221
|
const sheet = styleEl.sheet
|
|
222
222
|
if (!sheet) throw new Error('expected sheet')
|
|
223
|
-
const
|
|
223
|
+
const singleSelector = `.${className}`
|
|
224
224
|
|
|
225
225
|
let baseFound = false
|
|
226
226
|
let mediaInnerFound = false
|
|
227
227
|
|
|
228
228
|
for (let i = 0; i < sheet.cssRules.length; i++) {
|
|
229
229
|
const rule = sheet.cssRules[i]
|
|
230
|
-
if (rule instanceof CSSStyleRule && rule.selectorText ===
|
|
230
|
+
if (rule instanceof CSSStyleRule && rule.selectorText === singleSelector) {
|
|
231
231
|
baseFound = true
|
|
232
232
|
}
|
|
233
233
|
if (rule instanceof CSSMediaRule) {
|
|
234
234
|
for (let j = 0; j < rule.cssRules.length; j++) {
|
|
235
235
|
const inner = rule.cssRules[j]
|
|
236
|
-
if (inner instanceof CSSStyleRule && inner.selectorText ===
|
|
236
|
+
if (inner instanceof CSSStyleRule && inner.selectorText === singleSelector) {
|
|
237
237
|
mediaInnerFound = true
|
|
238
238
|
}
|
|
239
239
|
}
|
|
@@ -271,16 +271,16 @@ describe('StyleSheet -- at-rule splitting', () => {
|
|
|
271
271
|
expect(s.cacheSize).toBeGreaterThanOrEqual(1)
|
|
272
272
|
})
|
|
273
273
|
|
|
274
|
-
it('hydrates className from
|
|
274
|
+
it('hydrates className from @layer wrapped selectors in media rules', () => {
|
|
275
275
|
const el = document.createElement('style')
|
|
276
276
|
el.setAttribute('data-pyreon-styler', '')
|
|
277
277
|
document.head.appendChild(el)
|
|
278
278
|
|
|
279
279
|
const className = `pyr-${hash('font-size: 1rem;')}`
|
|
280
280
|
|
|
281
|
-
el.sheet?.insertRule(`.${className}
|
|
281
|
+
el.sheet?.insertRule(`.${className}{font-size: 1rem;}`, 0)
|
|
282
282
|
el.sheet?.insertRule(
|
|
283
|
-
`@media (min-width: 768px){.${className}
|
|
283
|
+
`@media (min-width: 768px){.${className}{font-size: 1.5rem;}}`,
|
|
284
284
|
1,
|
|
285
285
|
)
|
|
286
286
|
|
|
@@ -40,8 +40,8 @@ describe('StyleSheet', () => {
|
|
|
40
40
|
expect(className).toMatch(/^pyr-[0-9a-z]+$/)
|
|
41
41
|
})
|
|
42
42
|
|
|
43
|
-
it('supports
|
|
44
|
-
const className = sheet.insert('color: red;',
|
|
43
|
+
it('supports layer mode (@layer wrapping)', () => {
|
|
44
|
+
const className = sheet.insert('color: red;', false, 'rocketstyle')
|
|
45
45
|
expect(className).toMatch(/^pyr-[0-9a-z]+$/)
|
|
46
46
|
})
|
|
47
47
|
})
|
|
@@ -107,10 +107,11 @@ describe('StyleSheet', () => {
|
|
|
107
107
|
expect(rules).toContain('color: red;')
|
|
108
108
|
})
|
|
109
109
|
|
|
110
|
-
it('
|
|
111
|
-
const { className, rules } = sheet.prepare('color: red;'
|
|
112
|
-
//
|
|
113
|
-
expect(rules).toContain(`.${className}
|
|
110
|
+
it('produces single selector (no boost)', () => {
|
|
111
|
+
const { className, rules } = sheet.prepare('color: red;')
|
|
112
|
+
// Single selector, no doubling
|
|
113
|
+
expect(rules).toContain(`.${className}{`)
|
|
114
|
+
expect(rules).not.toContain(`.${className}.${className}`)
|
|
114
115
|
})
|
|
115
116
|
})
|
|
116
117
|
|
|
@@ -34,7 +34,7 @@ describe('styled -- SSR mode', () => {
|
|
|
34
34
|
|
|
35
35
|
it('static: boost option in SSR', async () => {
|
|
36
36
|
const { styled } = await import('../styled')
|
|
37
|
-
const Comp = styled('div', {
|
|
37
|
+
const Comp = styled('div', { layer: 'rocketstyle' })`
|
|
38
38
|
color: blue;
|
|
39
39
|
`
|
|
40
40
|
expect((Comp as any).displayName).toBe('styled(div)')
|
|
@@ -286,9 +286,9 @@ describe('styled', () => {
|
|
|
286
286
|
})
|
|
287
287
|
})
|
|
288
288
|
|
|
289
|
-
describe('
|
|
290
|
-
it('accepts
|
|
291
|
-
const Comp = styled('div', {
|
|
289
|
+
describe('layer option', () => {
|
|
290
|
+
it('accepts layer option without error', () => {
|
|
291
|
+
const Comp = styled('div', { layer: 'rocketstyle' })`
|
|
292
292
|
color: red;
|
|
293
293
|
`
|
|
294
294
|
const vnode = Comp({}) as VNode
|
package/src/sheet.ts
CHANGED
|
@@ -49,12 +49,22 @@ export class StyleSheet {
|
|
|
49
49
|
this.sheet = el.sheet ?? null
|
|
50
50
|
}
|
|
51
51
|
|
|
52
|
-
// Inject @layer
|
|
53
|
-
|
|
52
|
+
// Inject @layer declarations.
|
|
53
|
+
// 'base' is for plain styled() components, 'rocketstyle' is for
|
|
54
|
+
// rocketstyle wrappers. Layer order ensures rocketstyle overrides base
|
|
55
|
+
// without needing doubled selectors (boost).
|
|
56
|
+
if (this.sheet) {
|
|
54
57
|
try {
|
|
55
|
-
this.sheet.insertRule(
|
|
58
|
+
this.sheet.insertRule('@layer base, rocketstyle;', 0)
|
|
56
59
|
} catch {
|
|
57
|
-
//
|
|
60
|
+
// @layer not supported — falls back to source order
|
|
61
|
+
}
|
|
62
|
+
if (this.layer) {
|
|
63
|
+
try {
|
|
64
|
+
this.sheet.insertRule(`@layer ${this.layer};`, 1)
|
|
65
|
+
} catch {
|
|
66
|
+
// skip
|
|
67
|
+
}
|
|
58
68
|
}
|
|
59
69
|
}
|
|
60
70
|
}
|
|
@@ -177,12 +187,15 @@ export class StyleSheet {
|
|
|
177
187
|
* Deduplicates: same CSS text always produces the same class name and
|
|
178
188
|
* the rules are only injected once.
|
|
179
189
|
*
|
|
180
|
-
*
|
|
181
|
-
*
|
|
190
|
+
* @param cssText - CSS declarations to insert
|
|
191
|
+
* @param _unused - Reserved for backward compatibility (was `boost`)
|
|
192
|
+
* @param insertLayer - CSS @layer to wrap this rule in (e.g. 'rocketstyle').
|
|
193
|
+
* Used by rocketstyle to ensure wrapper styles override inner component styles
|
|
194
|
+
* via @layer order (base < rocketstyle) instead of specificity hacks.
|
|
182
195
|
*/
|
|
183
|
-
insert(cssText: string,
|
|
196
|
+
insert(cssText: string, _unused = false, insertLayer?: string): string {
|
|
184
197
|
// Fast path: skip hash computation on repeated insertions of same CSS text
|
|
185
|
-
const icKey =
|
|
198
|
+
const icKey = insertLayer ? `${cssText}\0L:${insertLayer}` : cssText
|
|
186
199
|
const icHit = this.insertCache.get(icKey)
|
|
187
200
|
if (icHit) return icHit
|
|
188
201
|
|
|
@@ -197,7 +210,7 @@ export class StyleSheet {
|
|
|
197
210
|
this.evictIfNeeded()
|
|
198
211
|
this.cache.set(className, className)
|
|
199
212
|
|
|
200
|
-
const selector =
|
|
213
|
+
const selector = `.${className}`
|
|
201
214
|
|
|
202
215
|
// Split nested at-rules into separate top-level rules
|
|
203
216
|
const { base, atRules } = this.splitAtRules(cssText, selector)
|
|
@@ -206,8 +219,9 @@ export class StyleSheet {
|
|
|
206
219
|
if (base) rules.push(`${selector}{${base}}`)
|
|
207
220
|
rules.push(...atRules)
|
|
208
221
|
|
|
209
|
-
// Apply @layer wrapping
|
|
210
|
-
const
|
|
222
|
+
// Apply @layer wrapping — per-insert layer takes precedence over sheet-level layer
|
|
223
|
+
const layerName = insertLayer ?? this.layer
|
|
224
|
+
const finalRules = layerName ? rules.map((r) => `@layer ${layerName}{${r}}`) : rules
|
|
211
225
|
|
|
212
226
|
if (this.isSSR) {
|
|
213
227
|
for (const rule of finalRules) {
|
|
@@ -348,10 +362,10 @@ export class StyleSheet {
|
|
|
348
362
|
/**
|
|
349
363
|
* Compute className and full CSS rule text without injecting.
|
|
350
364
|
*/
|
|
351
|
-
prepare(cssText: string
|
|
365
|
+
prepare(cssText: string): { className: string; rules: string } {
|
|
352
366
|
const h = hash(cssText)
|
|
353
367
|
const className = `${PREFIX}-${h}`
|
|
354
|
-
const selector =
|
|
368
|
+
const selector = `.${className}`
|
|
355
369
|
const { base, atRules } = this.splitAtRules(cssText, selector)
|
|
356
370
|
|
|
357
371
|
const allRules: string[] = []
|
package/src/styled.tsx
CHANGED
|
@@ -30,11 +30,11 @@ export interface StyledOptions {
|
|
|
30
30
|
/** Custom prop filter. Return true to forward the prop to the DOM element. */
|
|
31
31
|
shouldForwardProp?: (prop: string) => boolean
|
|
32
32
|
/**
|
|
33
|
-
*
|
|
34
|
-
*
|
|
35
|
-
*
|
|
33
|
+
* CSS @layer name. Rules are wrapped in `@layer <name> { ... }`.
|
|
34
|
+
* Used by rocketstyle to ensure wrapper styles override inner component
|
|
35
|
+
* styles via layer order (base < rocketstyle) instead of specificity hacks.
|
|
36
36
|
*/
|
|
37
|
-
|
|
37
|
+
layer?: string
|
|
38
38
|
}
|
|
39
39
|
|
|
40
40
|
const getDisplayName = (tag: Tag): string =>
|
|
@@ -77,7 +77,7 @@ const createStyledComponent = (
|
|
|
77
77
|
// Fast check: no values means no dynamic interpolations — avoids .some() scan
|
|
78
78
|
const hasDynamicValues = values.length > 0 && values.some(isDynamic)
|
|
79
79
|
const customFilter = options ? options.shouldForwardProp : undefined
|
|
80
|
-
const
|
|
80
|
+
const insertLayer = options?.layer
|
|
81
81
|
|
|
82
82
|
// STATIC FAST PATH: no function interpolations → compute class once at creation time
|
|
83
83
|
if (!hasDynamicValues) {
|
|
@@ -86,7 +86,7 @@ const createStyledComponent = (
|
|
|
86
86
|
const cssText = normalizeCSS(raw)
|
|
87
87
|
const hasCss = cssText.length > 0
|
|
88
88
|
|
|
89
|
-
const staticClassName = hasCss ? sheet.insert(cssText,
|
|
89
|
+
const staticClassName = hasCss ? sheet.insert(cssText, false, insertLayer) : ''
|
|
90
90
|
|
|
91
91
|
const StaticStyled: ComponentFn = (rawProps: Record<string, any>): VNode | null => {
|
|
92
92
|
const finalTag = rawProps.as || tag
|
|
@@ -148,7 +148,7 @@ const createStyledComponent = (
|
|
|
148
148
|
...(isReactiveState ? { $rocketstate: resolvedState } : {}),
|
|
149
149
|
}
|
|
150
150
|
const cssText = normalizeCSS(resolve(strings, values, { ...initialProps, theme }))
|
|
151
|
-
const initialClassName = cssText.length > 0 ? sheet.insert(cssText,
|
|
151
|
+
const initialClassName = cssText.length > 0 ? sheet.insert(cssText, false, insertLayer) : ''
|
|
152
152
|
|
|
153
153
|
const finalTag = rawProps.as || tag
|
|
154
154
|
const isDOM = typeof finalTag === 'string'
|
|
@@ -195,7 +195,7 @@ const createStyledComponent = (
|
|
|
195
195
|
const newCss = normalizeCSS(
|
|
196
196
|
resolve(strings, values, { ...newResolvedProps, theme }),
|
|
197
197
|
)
|
|
198
|
-
const newClass = newCss.length > 0 ? sheet.insert(newCss,
|
|
198
|
+
const newClass = newCss.length > 0 ? sheet.insert(newCss, false, insertLayer) : ''
|
|
199
199
|
|
|
200
200
|
if (el && newClass !== currentClassName) {
|
|
201
201
|
if (currentClassName) el.classList.remove(currentClassName)
|