css-to-tailwind-react 0.2.2 → 0.3.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/dist/cssParser.d.ts +3 -0
- package/dist/cssParser.js +78 -6
- package/dist/tailwindMapper.d.ts +38 -0
- package/dist/tailwindMapper.js +993 -80
- package/dist/utils/propertyMapper.d.ts +2 -0
- package/dist/utils/propertyMapper.js +133 -6
- package/dist/utils/variableRegistry.d.ts +52 -0
- package/dist/utils/variableRegistry.js +196 -0
- package/package.json +1 -1
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.VariableRegistry = void 0;
|
|
4
|
+
exports.parseVarExpression = parseVarExpression;
|
|
5
|
+
exports.isCssVariable = isCssVariable;
|
|
6
|
+
exports.isVarExpression = isVarExpression;
|
|
7
|
+
exports.createGlobalScope = createGlobalScope;
|
|
8
|
+
exports.createSelectorScope = createSelectorScope;
|
|
9
|
+
const specificityCalculator_1 = require("./specificityCalculator");
|
|
10
|
+
const logger_1 = require("./logger");
|
|
11
|
+
function createScopeKey(scope) {
|
|
12
|
+
if (scope.type === 'global') {
|
|
13
|
+
return ':root';
|
|
14
|
+
}
|
|
15
|
+
return scope.selector || '';
|
|
16
|
+
}
|
|
17
|
+
class VariableRegistry {
|
|
18
|
+
constructor() {
|
|
19
|
+
this.variables = new Map();
|
|
20
|
+
this.resolutionCache = new Map();
|
|
21
|
+
this.resolutionStack = new Set();
|
|
22
|
+
}
|
|
23
|
+
register(variable) {
|
|
24
|
+
const key = variable.name;
|
|
25
|
+
if (!this.variables.has(key)) {
|
|
26
|
+
this.variables.set(key, []);
|
|
27
|
+
}
|
|
28
|
+
const definitions = this.variables.get(key);
|
|
29
|
+
definitions.push(variable);
|
|
30
|
+
definitions.sort((a, b) => {
|
|
31
|
+
const specCompare = (0, specificityCalculator_1.compareSpecificity)(b.specificity, a.specificity);
|
|
32
|
+
if (specCompare !== 0)
|
|
33
|
+
return specCompare;
|
|
34
|
+
return b.sourceOrder - a.sourceOrder;
|
|
35
|
+
});
|
|
36
|
+
this.resolutionCache.clear();
|
|
37
|
+
}
|
|
38
|
+
resolve(varName, context, fallback) {
|
|
39
|
+
const cacheKey = `${varName}:${context.selector}:${context.variants.join(',')}`;
|
|
40
|
+
if (this.resolutionCache.has(cacheKey)) {
|
|
41
|
+
return { value: this.resolutionCache.get(cacheKey), resolved: true, source: 'cache' };
|
|
42
|
+
}
|
|
43
|
+
if (this.resolutionStack.has(varName)) {
|
|
44
|
+
logger_1.logger.warn(`Circular reference detected for variable: ${varName}`);
|
|
45
|
+
return { value: fallback || '', resolved: false, source: 'circular' };
|
|
46
|
+
}
|
|
47
|
+
this.resolutionStack.add(varName);
|
|
48
|
+
try {
|
|
49
|
+
const definitions = this.variables.get(varName);
|
|
50
|
+
if (!definitions || definitions.length === 0) {
|
|
51
|
+
if (fallback !== undefined) {
|
|
52
|
+
const resolvedFallback = this.resolveValue(fallback, context);
|
|
53
|
+
return { value: resolvedFallback.value, resolved: true, source: 'fallback' };
|
|
54
|
+
}
|
|
55
|
+
logger_1.logger.warn(`Undefined CSS variable: ${varName}`);
|
|
56
|
+
return { value: '', resolved: false, source: 'undefined' };
|
|
57
|
+
}
|
|
58
|
+
const applicableDefs = definitions.filter(def => this.isApplicable(def, context));
|
|
59
|
+
if (applicableDefs.length === 0) {
|
|
60
|
+
if (fallback !== undefined) {
|
|
61
|
+
const resolvedFallback = this.resolveValue(fallback, context);
|
|
62
|
+
return { value: resolvedFallback.value, resolved: true, source: 'fallback' };
|
|
63
|
+
}
|
|
64
|
+
logger_1.logger.warn(`No applicable definition for variable: ${varName} in context: ${context.selector}`);
|
|
65
|
+
return { value: '', resolved: false, source: 'no-match' };
|
|
66
|
+
}
|
|
67
|
+
const bestMatch = applicableDefs[0];
|
|
68
|
+
const resolvedValue = this.resolveValue(bestMatch.value, context);
|
|
69
|
+
if (resolvedValue.isCircular) {
|
|
70
|
+
return {
|
|
71
|
+
value: resolvedValue.value,
|
|
72
|
+
resolved: false,
|
|
73
|
+
source: 'circular'
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
const result = {
|
|
77
|
+
value: resolvedValue.value,
|
|
78
|
+
resolved: !resolvedValue.hasUnresolved,
|
|
79
|
+
source: resolvedValue.hasUnresolved ? 'unresolved' : 'resolved'
|
|
80
|
+
};
|
|
81
|
+
this.resolutionCache.set(cacheKey, result.value);
|
|
82
|
+
return result;
|
|
83
|
+
}
|
|
84
|
+
finally {
|
|
85
|
+
this.resolutionStack.delete(varName);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
resolveValue(value, context) {
|
|
89
|
+
const varPattern = /var\s*\(\s*(--[a-zA-Z0-9_-]+)\s*(?:,\s*([^)]+))?\s*\)/g;
|
|
90
|
+
let result = value;
|
|
91
|
+
let hasUnresolved = false;
|
|
92
|
+
let isCircular = false;
|
|
93
|
+
let match;
|
|
94
|
+
let iterations = 0;
|
|
95
|
+
const maxIterations = 10;
|
|
96
|
+
while ((match = varPattern.exec(result)) !== null) {
|
|
97
|
+
if (++iterations > maxIterations) {
|
|
98
|
+
logger_1.logger.warn(`Max variable resolution depth reached for: ${value}`);
|
|
99
|
+
hasUnresolved = true;
|
|
100
|
+
break;
|
|
101
|
+
}
|
|
102
|
+
const fullMatch = match[0];
|
|
103
|
+
const varName = match[1];
|
|
104
|
+
const fallback = match[2]?.trim();
|
|
105
|
+
const resolved = this.resolve(varName, context, fallback);
|
|
106
|
+
if (resolved.source === 'circular') {
|
|
107
|
+
isCircular = true;
|
|
108
|
+
hasUnresolved = true;
|
|
109
|
+
result = resolved.value || '';
|
|
110
|
+
break;
|
|
111
|
+
}
|
|
112
|
+
if (!resolved.resolved) {
|
|
113
|
+
hasUnresolved = true;
|
|
114
|
+
result = resolved.value;
|
|
115
|
+
}
|
|
116
|
+
else {
|
|
117
|
+
result = result.replace(fullMatch, resolved.value);
|
|
118
|
+
}
|
|
119
|
+
varPattern.lastIndex = 0;
|
|
120
|
+
}
|
|
121
|
+
return { value: result, hasUnresolved, isCircular };
|
|
122
|
+
}
|
|
123
|
+
isApplicable(def, context) {
|
|
124
|
+
if (def.scope.type === 'global') {
|
|
125
|
+
if (def.variants.length === 0) {
|
|
126
|
+
return true;
|
|
127
|
+
}
|
|
128
|
+
return this.variantsMatch(def.variants, context.variants);
|
|
129
|
+
}
|
|
130
|
+
const scopeSelector = def.scope.selector || '';
|
|
131
|
+
const contextSelector = context.selector;
|
|
132
|
+
if (scopeSelector === contextSelector) {
|
|
133
|
+
return this.variantsMatch(def.variants, context.variants);
|
|
134
|
+
}
|
|
135
|
+
if (this.selectorContains(contextSelector, scopeSelector)) {
|
|
136
|
+
return this.variantsMatch(def.variants, context.variants);
|
|
137
|
+
}
|
|
138
|
+
return false;
|
|
139
|
+
}
|
|
140
|
+
variantsMatch(defVariants, contextVariants) {
|
|
141
|
+
if (defVariants.length === 0) {
|
|
142
|
+
return true;
|
|
143
|
+
}
|
|
144
|
+
if (contextVariants.length === 0) {
|
|
145
|
+
return false;
|
|
146
|
+
}
|
|
147
|
+
return defVariants.every(v => contextVariants.includes(v));
|
|
148
|
+
}
|
|
149
|
+
selectorContains(outer, inner) {
|
|
150
|
+
const innerClasses = inner.match(/\.[a-zA-Z_-][a-zA-Z0-9_-]*/g) || [];
|
|
151
|
+
const outerClasses = outer.match(/\.[a-zA-Z_-][a-zA-Z0-9_-]*/g) || [];
|
|
152
|
+
return innerClasses.every(cls => outerClasses.includes(cls));
|
|
153
|
+
}
|
|
154
|
+
hasVariable(name) {
|
|
155
|
+
return this.variables.has(name);
|
|
156
|
+
}
|
|
157
|
+
getVariableDefinitions(name) {
|
|
158
|
+
return this.variables.get(name) || [];
|
|
159
|
+
}
|
|
160
|
+
clear() {
|
|
161
|
+
this.variables.clear();
|
|
162
|
+
this.resolutionCache.clear();
|
|
163
|
+
this.resolutionStack.clear();
|
|
164
|
+
}
|
|
165
|
+
getRegisteredVariables() {
|
|
166
|
+
return Array.from(this.variables.keys());
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
exports.VariableRegistry = VariableRegistry;
|
|
170
|
+
function parseVarExpression(value) {
|
|
171
|
+
const varPattern = /^var\s*\(\s*(--[a-zA-Z0-9_-]+)\s*(?:,\s*(.+)\s*)?\)$/;
|
|
172
|
+
const match = value.match(varPattern);
|
|
173
|
+
if (match) {
|
|
174
|
+
const fallback = match[2]?.trim();
|
|
175
|
+
return {
|
|
176
|
+
hasVar: true,
|
|
177
|
+
varName: match[1],
|
|
178
|
+
fallback: fallback && fallback.length > 0 ? fallback : undefined,
|
|
179
|
+
rawValue: value
|
|
180
|
+
};
|
|
181
|
+
}
|
|
182
|
+
return { hasVar: false, rawValue: value };
|
|
183
|
+
}
|
|
184
|
+
function isCssVariable(property) {
|
|
185
|
+
return property.startsWith('--');
|
|
186
|
+
}
|
|
187
|
+
function isVarExpression(value) {
|
|
188
|
+
return /var\s*\(\s*--[a-zA-Z0-9_-]+\s*(?:,[^)]+)?\s*\)/.test(value);
|
|
189
|
+
}
|
|
190
|
+
function createGlobalScope() {
|
|
191
|
+
return { type: 'global' };
|
|
192
|
+
}
|
|
193
|
+
function createSelectorScope(selector, selectorType = 'class') {
|
|
194
|
+
return { type: 'selector', selector, selectorType };
|
|
195
|
+
}
|
|
196
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"variableRegistry.js","sourceRoot":"","sources":["../../src/utils/variableRegistry.ts"],"names":[],"mappings":"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","sourcesContent":["import { Specificity, ZERO_SPECIFICITY, compareSpecificity } from './specificityCalculator';\nimport { logger } from './logger';\n\nexport interface VariableDefinition {\n  name: string;\n  value: string;\n  scope: VariableScope;\n  specificity: Specificity;\n  sourceOrder: number;\n  variants: string[];\n}\n\nexport interface VariableScope {\n  type: 'global' | 'selector';\n  selector?: string;\n  selectorType?: 'class' | 'element' | 'root';\n}\n\nexport interface ResolutionContext {\n  selector: string;\n  specificity: Specificity;\n  variants: string[];\n}\n\ninterface VariableKey {\n  name: string;\n  scopeKey: string;\n}\n\nfunction createScopeKey(scope: VariableScope): string {\n  if (scope.type === 'global') {\n    return ':root';\n  }\n  return scope.selector || '';\n}\n\nexport class VariableRegistry {\n  private variables: Map<string, VariableDefinition[]> = new Map();\n  private resolutionCache: Map<string, string> = new Map();\n  private resolutionStack: Set<string> = new Set();\n\n  register(variable: VariableDefinition): void {\n    const key = variable.name;\n    \n    if (!this.variables.has(key)) {\n      this.variables.set(key, []);\n    }\n    \n    const definitions = this.variables.get(key)!;\n    definitions.push(variable);\n    \n    definitions.sort((a, b) => {\n      const specCompare = compareSpecificity(b.specificity, a.specificity);\n      if (specCompare !== 0) return specCompare;\n      return b.sourceOrder - a.sourceOrder;\n    });\n    \n    this.resolutionCache.clear();\n  }\n\n  resolve(\n    varName: string,\n    context: ResolutionContext,\n    fallback?: string\n  ): { value: string; resolved: boolean; source: string } {\n    const cacheKey = `${varName}:${context.selector}:${context.variants.join(',')}`;\n    \n    if (this.resolutionCache.has(cacheKey)) {\n      return { value: this.resolutionCache.get(cacheKey)!, resolved: true, source: 'cache' };\n    }\n    \n    if (this.resolutionStack.has(varName)) {\n      logger.warn(`Circular reference detected for variable: ${varName}`);\n      return { value: fallback || '', resolved: false, source: 'circular' };\n    }\n    \n    this.resolutionStack.add(varName);\n    \n    try {\n      const definitions = this.variables.get(varName);\n      \n      if (!definitions || definitions.length === 0) {\n        if (fallback !== undefined) {\n          const resolvedFallback = this.resolveValue(fallback, context);\n          return { value: resolvedFallback.value, resolved: true, source: 'fallback' };\n        }\n        logger.warn(`Undefined CSS variable: ${varName}`);\n        return { value: '', resolved: false, source: 'undefined' };\n      }\n      \n      const applicableDefs = definitions.filter(def => \n        this.isApplicable(def, context)\n      );\n      \n      if (applicableDefs.length === 0) {\n        if (fallback !== undefined) {\n          const resolvedFallback = this.resolveValue(fallback, context);\n          return { value: resolvedFallback.value, resolved: true, source: 'fallback' };\n        }\n        logger.warn(`No applicable definition for variable: ${varName} in context: ${context.selector}`);\n        return { value: '', resolved: false, source: 'no-match' };\n      }\n      \n      const bestMatch = applicableDefs[0];\n      const resolvedValue = this.resolveValue(bestMatch.value, context);\n      \n      if (resolvedValue.isCircular) {\n        return { \n          value: resolvedValue.value, \n          resolved: false, \n          source: 'circular' \n        };\n      }\n      \n      const result = { \n        value: resolvedValue.value, \n        resolved: !resolvedValue.hasUnresolved, \n        source: resolvedValue.hasUnresolved ? 'unresolved' : 'resolved' \n      };\n      \n      this.resolutionCache.set(cacheKey, result.value);\n      return result;\n      \n    } finally {\n      this.resolutionStack.delete(varName);\n    }\n  }\n\n  resolveValue(value: string, context: ResolutionContext): { value: string; hasUnresolved: boolean; isCircular?: boolean } {\n    const varPattern = /var\\s*\\(\\s*(--[a-zA-Z0-9_-]+)\\s*(?:,\\s*([^)]+))?\\s*\\)/g;\n    let result = value;\n    let hasUnresolved = false;\n    let isCircular = false;\n    let match;\n    let iterations = 0;\n    const maxIterations = 10;\n    \n    while ((match = varPattern.exec(result)) !== null) {\n      if (++iterations > maxIterations) {\n        logger.warn(`Max variable resolution depth reached for: ${value}`);\n        hasUnresolved = true;\n        break;\n      }\n      \n      const fullMatch = match[0];\n      const varName = match[1];\n      const fallback = match[2]?.trim();\n      \n      const resolved = this.resolve(varName, context, fallback);\n      \n      if (resolved.source === 'circular') {\n        isCircular = true;\n        hasUnresolved = true;\n        result = resolved.value || '';\n        break;\n      }\n      \n      if (!resolved.resolved) {\n        hasUnresolved = true;\n        result = resolved.value;\n      } else {\n        result = result.replace(fullMatch, resolved.value);\n      }\n      varPattern.lastIndex = 0;\n    }\n    \n    return { value: result, hasUnresolved, isCircular };\n  }\n\n  private isApplicable(def: VariableDefinition, context: ResolutionContext): boolean {\n    if (def.scope.type === 'global') {\n      if (def.variants.length === 0) {\n        return true;\n      }\n      return this.variantsMatch(def.variants, context.variants);\n    }\n    \n    const scopeSelector = def.scope.selector || '';\n    const contextSelector = context.selector;\n    \n    if (scopeSelector === contextSelector) {\n      return this.variantsMatch(def.variants, context.variants);\n    }\n    \n    if (this.selectorContains(contextSelector, scopeSelector)) {\n      return this.variantsMatch(def.variants, context.variants);\n    }\n    \n    return false;\n  }\n\n  private variantsMatch(defVariants: string[], contextVariants: string[]): boolean {\n    if (defVariants.length === 0) {\n      return true;\n    }\n    \n    if (contextVariants.length === 0) {\n      return false;\n    }\n    \n    return defVariants.every(v => contextVariants.includes(v));\n  }\n\n  private selectorContains(outer: string, inner: string): boolean {\n    const innerClasses: string[] = inner.match(/\\.[a-zA-Z_-][a-zA-Z0-9_-]*/g) || [];\n    const outerClasses: string[] = outer.match(/\\.[a-zA-Z_-][a-zA-Z0-9_-]*/g) || [];\n    \n    return innerClasses.every(cls => outerClasses.includes(cls));\n  }\n\n  hasVariable(name: string): boolean {\n    return this.variables.has(name);\n  }\n\n  getVariableDefinitions(name: string): VariableDefinition[] {\n    return this.variables.get(name) || [];\n  }\n\n  clear(): void {\n    this.variables.clear();\n    this.resolutionCache.clear();\n    this.resolutionStack.clear();\n  }\n\n  getRegisteredVariables(): string[] {\n    return Array.from(this.variables.keys());\n  }\n}\n\nexport function parseVarExpression(value: string): { \n  hasVar: boolean; \n  varName?: string; \n  fallback?: string;\n  rawValue: string;\n} {\n  const varPattern = /^var\\s*\\(\\s*(--[a-zA-Z0-9_-]+)\\s*(?:,\\s*(.+)\\s*)?\\)$/;\n  const match = value.match(varPattern);\n  \n  if (match) {\n    const fallback = match[2]?.trim();\n    return {\n      hasVar: true,\n      varName: match[1],\n      fallback: fallback && fallback.length > 0 ? fallback : undefined,\n      rawValue: value\n    };\n  }\n  \n  return { hasVar: false, rawValue: value };\n}\n\nexport function isCssVariable(property: string): boolean {\n  return property.startsWith('--');\n}\n\nexport function isVarExpression(value: string): boolean {\n  return /var\\s*\\(\\s*--[a-zA-Z0-9_-]+\\s*(?:,[^)]+)?\\s*\\)/.test(value);\n}\n\nexport function createGlobalScope(): VariableScope {\n  return { type: 'global' };\n}\n\nexport function createSelectorScope(selector: string, selectorType: 'class' | 'element' | 'root' = 'class'): VariableScope {\n  return { type: 'selector', selector, selectorType };\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "css-to-tailwind-react",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.3.1",
|
|
4
4
|
"description": "Convert traditional CSS (inline, internal, and external) into Tailwind CSS utility classes for React-based frameworks",
|
|
5
5
|
"main": "dist/index.js",
|
|
6
6
|
"module": "dist/index.mjs",
|