css-to-tailwind-react 0.1.1 → 0.1.2
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 -3
- package/dist/cssParser.js +100 -84
- package/dist/index.d.ts +2 -0
- package/dist/index.js +19 -2
- package/dist/transformer.js +20 -59
- package/dist/utils/pseudoSelectorResolver.d.ts +17 -0
- package/dist/utils/pseudoSelectorResolver.js +163 -0
- package/dist/utils/variantAssembler.d.ts +20 -0
- package/dist/utils/variantAssembler.js +114 -0
- package/package.json +1 -1
package/dist/cssParser.d.ts
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { TailwindMapper, CSSProperty } from './tailwindMapper';
|
|
2
2
|
export interface UtilityWithVariant {
|
|
3
3
|
value: string;
|
|
4
|
-
|
|
4
|
+
variants: string[];
|
|
5
5
|
}
|
|
6
6
|
export interface CSSRule {
|
|
7
7
|
selector: string;
|
|
@@ -9,7 +9,6 @@ export interface CSSRule {
|
|
|
9
9
|
declarations: CSSProperty[];
|
|
10
10
|
convertedClasses: string[];
|
|
11
11
|
utilities: UtilityWithVariant[];
|
|
12
|
-
breakpoint?: string;
|
|
13
12
|
skipped: boolean;
|
|
14
13
|
fullyConverted: boolean;
|
|
15
14
|
partialConversion: boolean;
|
|
@@ -29,7 +28,8 @@ export declare class CSSParser {
|
|
|
29
28
|
private mapper;
|
|
30
29
|
private breakpoints;
|
|
31
30
|
constructor(mapper: TailwindMapper, screens?: Record<string, string | [string, string]>);
|
|
32
|
-
private
|
|
31
|
+
private convertDeclarations;
|
|
32
|
+
private processRuleWithVariants;
|
|
33
33
|
parse(css: string, filePath: string): Promise<CSSParseResult>;
|
|
34
34
|
parseInternalStyle(html: string): {
|
|
35
35
|
styles: Array<{
|
package/dist/cssParser.js
CHANGED
|
@@ -8,6 +8,8 @@ const postcss_1 = __importDefault(require("postcss"));
|
|
|
8
8
|
const postcss_safe_parser_1 = __importDefault(require("postcss-safe-parser"));
|
|
9
9
|
const logger_1 = require("./utils/logger");
|
|
10
10
|
const breakpointResolver_1 = require("./utils/breakpointResolver");
|
|
11
|
+
const pseudoSelectorResolver_1 = require("./utils/pseudoSelectorResolver");
|
|
12
|
+
const variantAssembler_1 = require("./utils/variantAssembler");
|
|
11
13
|
class CSSParser {
|
|
12
14
|
constructor(mapper, screens) {
|
|
13
15
|
this.mapper = mapper;
|
|
@@ -15,15 +17,35 @@ class CSSParser {
|
|
|
15
17
|
? (0, breakpointResolver_1.resolveBreakpointsFromConfig)(screens)
|
|
16
18
|
: (0, breakpointResolver_1.getBreakpoints)();
|
|
17
19
|
}
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
20
|
+
convertDeclarations(declarations) {
|
|
21
|
+
const conversionResults = [];
|
|
22
|
+
const conversionWarnings = [];
|
|
23
|
+
declarations.forEach(decl => {
|
|
24
|
+
const result = this.mapper.convertProperty(decl.property, decl.value);
|
|
25
|
+
conversionResults.push({
|
|
26
|
+
declaration: decl,
|
|
27
|
+
converted: !result.skipped && result.className !== null,
|
|
28
|
+
className: result.className
|
|
29
|
+
});
|
|
30
|
+
if (result.skipped && result.reason) {
|
|
31
|
+
conversionWarnings.push(result.reason);
|
|
32
|
+
}
|
|
33
|
+
});
|
|
34
|
+
const utilities = conversionResults
|
|
35
|
+
.filter(r => r.converted && r.className)
|
|
36
|
+
.map(r => ({
|
|
37
|
+
value: r.className,
|
|
38
|
+
variants: []
|
|
39
|
+
}));
|
|
40
|
+
return { utilities, conversionResults, conversionWarnings };
|
|
41
|
+
}
|
|
42
|
+
processRuleWithVariants(rule, additionalVariants = []) {
|
|
43
|
+
const selector = rule.selector;
|
|
44
|
+
const parsedSelectors = (0, pseudoSelectorResolver_1.parseMultipleSelectors)(selector);
|
|
45
|
+
const validSelectors = parsedSelectors.filter(s => !s.isComplex && s.baseClass);
|
|
46
|
+
if (validSelectors.length === 0) {
|
|
24
47
|
return null;
|
|
25
48
|
}
|
|
26
|
-
const className = classNameMatch[1];
|
|
27
49
|
const declarations = [];
|
|
28
50
|
rule.walkDecls((decl) => {
|
|
29
51
|
if (decl.prop.startsWith('--')) {
|
|
@@ -40,41 +62,38 @@ class CSSParser {
|
|
|
40
62
|
if (declarations.length === 0) {
|
|
41
63
|
return null;
|
|
42
64
|
}
|
|
43
|
-
const conversionResults =
|
|
44
|
-
const
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
conversionResults.push({
|
|
48
|
-
declaration: decl,
|
|
49
|
-
converted: !result.skipped && result.className !== null,
|
|
50
|
-
className: result.className
|
|
51
|
-
});
|
|
52
|
-
if (result.skipped && result.reason) {
|
|
53
|
-
conversionWarnings.push(result.reason);
|
|
54
|
-
}
|
|
55
|
-
});
|
|
56
|
-
const utilities = conversionResults
|
|
57
|
-
.filter(r => r.converted && r.className)
|
|
58
|
-
.map(r => ({
|
|
59
|
-
value: r.className,
|
|
60
|
-
variant: breakpoint
|
|
65
|
+
const { utilities, conversionResults, conversionWarnings } = this.convertDeclarations(declarations);
|
|
66
|
+
const utilitiesWithVariants = utilities.map(u => ({
|
|
67
|
+
value: u.value,
|
|
68
|
+
variants: (0, variantAssembler_1.normalizeVariantOrder)([...u.variants, ...additionalVariants])
|
|
61
69
|
}));
|
|
62
|
-
const
|
|
63
|
-
const
|
|
64
|
-
const
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
70
|
+
const cssRules = [];
|
|
71
|
+
const allConversionResults = [];
|
|
72
|
+
for (const parsed of validSelectors) {
|
|
73
|
+
const pseudoVariants = parsed.pseudos || [];
|
|
74
|
+
const allVariants = (0, variantAssembler_1.normalizeVariantOrder)([...pseudoVariants, ...additionalVariants]);
|
|
75
|
+
const utilitiesForSelector = utilities.map(u => ({
|
|
76
|
+
value: u.value,
|
|
77
|
+
variants: allVariants
|
|
78
|
+
}));
|
|
79
|
+
const convertedClasses = (0, variantAssembler_1.assembleUtilities)(utilitiesForSelector);
|
|
80
|
+
const allDeclarationsConverted = conversionResults.every(r => r.converted);
|
|
81
|
+
const someDeclarationsConverted = convertedClasses.length > 0;
|
|
82
|
+
const cssRule = {
|
|
83
|
+
selector: selector,
|
|
84
|
+
className: parsed.baseClass,
|
|
85
|
+
declarations,
|
|
86
|
+
convertedClasses,
|
|
87
|
+
utilities: utilitiesForSelector,
|
|
88
|
+
skipped: !someDeclarationsConverted,
|
|
89
|
+
fullyConverted: allDeclarationsConverted,
|
|
90
|
+
partialConversion: someDeclarationsConverted && !allDeclarationsConverted,
|
|
91
|
+
reason: !someDeclarationsConverted ? 'No convertible declarations' : undefined
|
|
92
|
+
};
|
|
93
|
+
cssRules.push(cssRule);
|
|
94
|
+
allConversionResults.push(conversionResults);
|
|
95
|
+
}
|
|
96
|
+
return { cssRules, conversionResults: allConversionResults, conversionWarnings };
|
|
78
97
|
}
|
|
79
98
|
async parse(css, filePath) {
|
|
80
99
|
const rules = [];
|
|
@@ -94,24 +113,27 @@ class CSSParser {
|
|
|
94
113
|
warnings.push(mediaResult.reason || `Skipped media query: ${atRule.params}`);
|
|
95
114
|
return;
|
|
96
115
|
}
|
|
97
|
-
const
|
|
116
|
+
const responsiveVariant = mediaResult.breakpoint;
|
|
98
117
|
const nestedRules = [];
|
|
99
118
|
atRule.walkRules((rule) => {
|
|
100
119
|
nestedRules.push(rule);
|
|
101
120
|
});
|
|
102
121
|
for (const rule of nestedRules) {
|
|
103
|
-
const result = this.
|
|
122
|
+
const result = this.processRuleWithVariants(rule, [responsiveVariant]);
|
|
104
123
|
if (result) {
|
|
105
|
-
rules.push(result.
|
|
124
|
+
rules.push(...result.cssRules);
|
|
106
125
|
warnings.push(...result.conversionWarnings);
|
|
107
|
-
|
|
126
|
+
const anyConverted = result.cssRules.some(r => r.convertedClasses.length > 0);
|
|
127
|
+
if (anyConverted) {
|
|
108
128
|
hasChanges = true;
|
|
109
|
-
|
|
129
|
+
const allFullyConverted = result.cssRules.every(r => r.fullyConverted);
|
|
130
|
+
if (allFullyConverted) {
|
|
110
131
|
rule.remove();
|
|
111
|
-
|
|
132
|
+
const classNames = result.cssRules.map(r => r.className).join(', .');
|
|
133
|
+
logger_1.logger.verbose(`Removed rule .${classNames} in @media (min-width) → ${responsiveVariant}`);
|
|
112
134
|
}
|
|
113
135
|
else {
|
|
114
|
-
for (const cr of result.conversionResults) {
|
|
136
|
+
for (const cr of result.conversionResults.flat()) {
|
|
115
137
|
if (cr.converted) {
|
|
116
138
|
rule.walkDecls((decl) => {
|
|
117
139
|
if (decl.prop === cr.declaration.property && decl.value === cr.declaration.value) {
|
|
@@ -120,10 +142,13 @@ class CSSParser {
|
|
|
120
142
|
});
|
|
121
143
|
}
|
|
122
144
|
}
|
|
123
|
-
logger_1.logger.verbose(`Partial conversion
|
|
145
|
+
logger_1.logger.verbose(`Partial conversion in @media → ${responsiveVariant}`);
|
|
124
146
|
}
|
|
125
147
|
}
|
|
126
148
|
}
|
|
149
|
+
else {
|
|
150
|
+
warnings.push(`Skipped rule in @media: ${rule.selector}`);
|
|
151
|
+
}
|
|
127
152
|
}
|
|
128
153
|
if (atRule.nodes && atRule.nodes.length === 0) {
|
|
129
154
|
atRule.remove();
|
|
@@ -134,42 +159,39 @@ class CSSParser {
|
|
|
134
159
|
if (rule.parent && rule.parent.type === 'atrule') {
|
|
135
160
|
return;
|
|
136
161
|
}
|
|
137
|
-
|
|
138
|
-
warnings.push(`Skipped pseudo-selector: ${rule.selector}`);
|
|
139
|
-
logger_1.logger.verbose(`Skipping pseudo-selector: ${rule.selector}`);
|
|
140
|
-
return;
|
|
141
|
-
}
|
|
142
|
-
const classNameMatch = rule.selector.match(/^\.([a-zA-Z_-][a-zA-Z0-9_-]*)$/);
|
|
143
|
-
if (!classNameMatch) {
|
|
144
|
-
warnings.push(`Skipped complex selector: ${rule.selector}`);
|
|
145
|
-
logger_1.logger.verbose(`Skipping complex selector: ${rule.selector}`);
|
|
146
|
-
return;
|
|
147
|
-
}
|
|
148
|
-
const result = this.processRule(rule);
|
|
162
|
+
const result = this.processRuleWithVariants(rule);
|
|
149
163
|
if (!result) {
|
|
164
|
+
const parsedSelectors = (0, pseudoSelectorResolver_1.parseMultipleSelectors)(rule.selector);
|
|
165
|
+
const allComplex = parsedSelectors.every(s => s.isComplex);
|
|
166
|
+
if (allComplex) {
|
|
167
|
+
const reasons = parsedSelectors.map(s => s.reason).filter(Boolean);
|
|
168
|
+
warnings.push(...reasons);
|
|
169
|
+
logger_1.logger.verbose(`Skipping complex selector: ${rule.selector}`);
|
|
170
|
+
}
|
|
150
171
|
return;
|
|
151
172
|
}
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
if (
|
|
173
|
+
rules.push(...result.cssRules);
|
|
174
|
+
warnings.push(...result.conversionWarnings);
|
|
175
|
+
const anyConverted = result.cssRules.some(r => r.convertedClasses.length > 0);
|
|
176
|
+
if (anyConverted) {
|
|
156
177
|
hasChanges = true;
|
|
157
|
-
|
|
178
|
+
const allFullyConverted = result.cssRules.every(r => r.fullyConverted);
|
|
179
|
+
if (allFullyConverted) {
|
|
158
180
|
rule.remove();
|
|
159
|
-
|
|
181
|
+
const classNames = result.cssRules.map(r => r.className).join(', .');
|
|
182
|
+
logger_1.logger.verbose(`Removed rule .${classNames} (all declarations converted)`);
|
|
160
183
|
}
|
|
161
184
|
else {
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
removedCount++;
|
|
185
|
+
for (const cr of result.conversionResults.flat()) {
|
|
186
|
+
if (cr.converted) {
|
|
187
|
+
rule.walkDecls((decl) => {
|
|
188
|
+
if (decl.prop === cr.declaration.property && decl.value === cr.declaration.value) {
|
|
189
|
+
decl.remove();
|
|
190
|
+
}
|
|
191
|
+
});
|
|
170
192
|
}
|
|
171
|
-
}
|
|
172
|
-
logger_1.logger.verbose(`Partial conversion of
|
|
193
|
+
}
|
|
194
|
+
logger_1.logger.verbose(`Partial conversion of rule`);
|
|
173
195
|
}
|
|
174
196
|
}
|
|
175
197
|
});
|
|
@@ -196,7 +218,6 @@ class CSSParser {
|
|
|
196
218
|
parseInternalStyle(html) {
|
|
197
219
|
const styles = [];
|
|
198
220
|
const warnings = [];
|
|
199
|
-
// Simple regex to find style tags (this is safe for finding tags, not for parsing content)
|
|
200
221
|
const styleRegex = /<style[^>]*>([\s\S]*?)<\/style>/gi;
|
|
201
222
|
let match;
|
|
202
223
|
while ((match = styleRegex.exec(html)) !== null) {
|
|
@@ -214,7 +235,6 @@ class CSSParser {
|
|
|
214
235
|
let modifiedHtml = html;
|
|
215
236
|
let hasChanges = false;
|
|
216
237
|
const { styles } = this.parseInternalStyle(html);
|
|
217
|
-
// Process styles in reverse order to preserve indices
|
|
218
238
|
for (let i = styles.length - 1; i >= 0; i--) {
|
|
219
239
|
const style = styles[i];
|
|
220
240
|
try {
|
|
@@ -224,11 +244,9 @@ class CSSParser {
|
|
|
224
244
|
if (result.hasChanges) {
|
|
225
245
|
hasChanges = true;
|
|
226
246
|
if (result.canDelete || result.css.trim() === '') {
|
|
227
|
-
// Remove entire style tag
|
|
228
247
|
modifiedHtml = modifiedHtml.slice(0, style.start) + modifiedHtml.slice(style.end);
|
|
229
248
|
}
|
|
230
249
|
else {
|
|
231
|
-
// Replace style content
|
|
232
250
|
const before = modifiedHtml.slice(0, style.start);
|
|
233
251
|
const after = modifiedHtml.slice(style.end);
|
|
234
252
|
const tagStart = html.slice(style.start).match(/<style[^>]*>/)?.[0] || '<style>';
|
|
@@ -251,13 +269,11 @@ class CSSParser {
|
|
|
251
269
|
}
|
|
252
270
|
extractImportPaths(code) {
|
|
253
271
|
const imports = [];
|
|
254
|
-
// Match CSS imports
|
|
255
272
|
const importRegex = /import\s+['"]([^'"]+\.css)['"];?/g;
|
|
256
273
|
let match;
|
|
257
274
|
while ((match = importRegex.exec(code)) !== null) {
|
|
258
275
|
imports.push(match[1]);
|
|
259
276
|
}
|
|
260
|
-
// Match require statements
|
|
261
277
|
const requireRegex = /require\s*\(\s*['"]([^'"]+\.css)['"]\s*\)/g;
|
|
262
278
|
while ((match = requireRegex.exec(code)) !== null) {
|
|
263
279
|
imports.push(match[1]);
|
|
@@ -266,4 +282,4 @@ class CSSParser {
|
|
|
266
282
|
}
|
|
267
283
|
}
|
|
268
284
|
exports.CSSParser = CSSParser;
|
|
269
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cssParser.js","sourceRoot":"","sources":["../src/cssParser.ts"],"names":[],"mappings":"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","sourcesContent":["import postcss, { Root, Rule, Declaration, AtRule } from 'postcss';\nimport safeParser from 'postcss-safe-parser';\nimport { TailwindMapper, CSSProperty } from './tailwindMapper';\nimport { logger } from './utils/logger';\nimport { \n  Breakpoint, \n  getBreakpoints, \n  resolveBreakpointsFromConfig,\n  processMediaQuery, \n  prefixWithBreakpoint,\n  clearBreakpointCache \n} from './utils/breakpointResolver';\n\nexport interface UtilityWithVariant {\n  value: string;\n  variant?: string;\n}\n\nexport interface CSSRule {\n  selector: string;\n  className: string;\n  declarations: CSSProperty[];\n  convertedClasses: string[];\n  utilities: UtilityWithVariant[];\n  breakpoint?: string;\n  skipped: boolean;\n  fullyConverted: boolean;\n  partialConversion: boolean;\n  reason?: string;\n}\n\nexport interface CSSParseResult {\n  css: string;\n  rules: CSSRule[];\n  hasChanges: boolean;\n  canDelete: boolean;\n  warnings: string[];\n}\n\nexport interface CSSUsageMap {\n  [className: string]: string[]; // className -> file paths\n}\n\nexport class CSSParser {\n  private mapper: TailwindMapper;\n  private breakpoints: Breakpoint[];\n\n  constructor(mapper: TailwindMapper, screens?: Record<string, string | [string, string]>) {\n    this.mapper = mapper;\n    this.breakpoints = screens \n      ? resolveBreakpointsFromConfig(screens) \n      : getBreakpoints();\n  }\n\n  private processRule(\n    rule: Rule,\n    breakpoint?: string\n  ): { cssRule: CSSRule; conversionResults: Array<{ declaration: CSSProperty; converted: boolean; className: string | null }>; conversionWarnings: string[] } | null {\n    if (rule.selector.includes(':')) {\n      return null;\n    }\n\n    const classNameMatch = rule.selector.match(/^\\.([a-zA-Z_-][a-zA-Z0-9_-]*)$/);\n    if (!classNameMatch) {\n      return null;\n    }\n\n    const className = classNameMatch[1];\n    const declarations: CSSProperty[] = [];\n\n    rule.walkDecls((decl) => {\n      if (decl.prop.startsWith('--')) {\n        return;\n      }\n\n      if (decl.value.includes('calc(')) {\n        return;\n      }\n\n      declarations.push({\n        property: decl.prop,\n        value: decl.value\n      });\n    });\n\n    if (declarations.length === 0) {\n      return null;\n    }\n\n    const conversionResults: Array<{\n      declaration: CSSProperty;\n      converted: boolean;\n      className: string | null;\n    }> = [];\n    const conversionWarnings: string[] = [];\n\n    declarations.forEach(decl => {\n      const result = this.mapper.convertProperty(decl.property, decl.value);\n      conversionResults.push({\n        declaration: decl,\n        converted: !result.skipped && result.className !== null,\n        className: result.className\n      });\n      if (result.skipped && result.reason) {\n        conversionWarnings.push(result.reason);\n      }\n    });\n\n    const utilities: UtilityWithVariant[] = conversionResults\n      .filter(r => r.converted && r.className)\n      .map(r => ({\n        value: r.className!,\n        variant: breakpoint\n      }));\n\n    const convertedClasses = utilities.map(u => \n      u.variant ? prefixWithBreakpoint(u.value, u.variant) : u.value\n    );\n\n    const allDeclarationsConverted = conversionResults.every(r => r.converted);\n    const someDeclarationsConverted = convertedClasses.length > 0;\n\n    const cssRule: CSSRule = {\n      selector: rule.selector,\n      className,\n      declarations,\n      convertedClasses,\n      utilities,\n      breakpoint,\n      skipped: !someDeclarationsConverted,\n      fullyConverted: allDeclarationsConverted,\n      partialConversion: someDeclarationsConverted && !allDeclarationsConverted,\n      reason: !someDeclarationsConverted ? 'No convertible declarations' : undefined\n    };\n\n    return { cssRule, conversionResults, conversionWarnings };\n  }\n\n  async parse(css: string, filePath: string): Promise<CSSParseResult> {\n    const rules: CSSRule[] = [];\n    const warnings: string[] = [];\n    let hasChanges = false;\n\n    try {\n      const root = await postcss().process(css, {\n        parser: safeParser,\n        from: filePath\n      }).then(result => result.root);\n\n      root.walkAtRules((atRule) => {\n        if (atRule.name !== 'media') {\n          return;\n        }\n\n        const mediaResult = processMediaQuery(atRule.params, this.breakpoints);\n\n        if (mediaResult.skipped) {\n          warnings.push(mediaResult.reason || `Skipped media query: ${atRule.params}`);\n          return;\n        }\n\n        const breakpoint = mediaResult.breakpoint!;\n\n        const nestedRules: Rule[] = [];\n        atRule.walkRules((rule) => {\n          nestedRules.push(rule);\n        });\n\n        for (const rule of nestedRules) {\n          const result = this.processRule(rule, breakpoint);\n          if (result) {\n            rules.push(result.cssRule);\n            warnings.push(...result.conversionWarnings);\n\n            if (result.cssRule.convertedClasses.length > 0) {\n              hasChanges = true;\n\n              if (result.cssRule.fullyConverted) {\n                rule.remove();\n                logger.verbose(`Removed rule .${result.cssRule.className} in @media (min-width) → ${breakpoint}`);\n              } else {\n                for (const cr of result.conversionResults) {\n                  if (cr.converted) {\n                    rule.walkDecls((decl) => {\n                      if (decl.prop === cr.declaration.property && decl.value === cr.declaration.value) {\n                        decl.remove();\n                      }\n                    });\n                  }\n                }\n                logger.verbose(`Partial conversion of .${result.cssRule.className} in @media → ${breakpoint}`);\n              }\n            }\n          }\n        }\n\n        if (atRule.nodes && atRule.nodes.length === 0) {\n          atRule.remove();\n          logger.verbose(`Removed empty @media rule`);\n        }\n      });\n\n      root.walkRules((rule) => {\n        if (rule.parent && rule.parent.type === 'atrule') {\n          return;\n        }\n\n        if (rule.selector.includes(':')) {\n          warnings.push(`Skipped pseudo-selector: ${rule.selector}`);\n          logger.verbose(`Skipping pseudo-selector: ${rule.selector}`);\n          return;\n        }\n\n        const classNameMatch = rule.selector.match(/^\\.([a-zA-Z_-][a-zA-Z0-9_-]*)$/);\n        if (!classNameMatch) {\n          warnings.push(`Skipped complex selector: ${rule.selector}`);\n          logger.verbose(`Skipping complex selector: ${rule.selector}`);\n          return;\n        }\n\n        const result = this.processRule(rule);\n        if (!result) {\n          return;\n        }\n\n        const { cssRule, conversionResults, conversionWarnings } = result;\n        rules.push(cssRule);\n        warnings.push(...conversionWarnings);\n\n        if (cssRule.convertedClasses.length > 0) {\n          hasChanges = true;\n\n          if (cssRule.fullyConverted) {\n            rule.remove();\n            logger.verbose(`Removed rule .${cssRule.className} (all ${cssRule.declarations.length} declarations converted)`);\n          } else {\n            let removedCount = 0;\n            rule.walkDecls((decl) => {\n              const wasConverted = conversionResults.some(r =>\n                r.converted &&\n                r.declaration.property === decl.prop &&\n                r.declaration.value === decl.value\n              );\n\n              if (wasConverted) {\n                decl.remove();\n                removedCount++;\n              }\n            });\n\n            logger.verbose(`Partial conversion of .${cssRule.className}: removed ${removedCount}/${cssRule.declarations.length} declarations`);\n          }\n        }\n      });\n\n      root.walkAtRules((atRule) => {\n        if (atRule.nodes && atRule.nodes.length === 0) {\n          atRule.remove();\n        }\n      });\n\n      const canDelete = root.nodes.length === 0;\n      const newCss = root.toString();\n\n      return {\n        css: newCss,\n        rules,\n        hasChanges,\n        canDelete,\n        warnings\n      };\n\n    } catch (error) {\n      logger.error(`Failed to parse CSS ${filePath}:`, error);\n      throw new Error(`CSS parsing failed: ${error}`);\n    }\n  }\n\n  parseInternalStyle(html: string): { \n    styles: Array<{ content: string; start: number; end: number }>;\n    warnings: string[];\n  } {\n    const styles: Array<{ content: string; start: number; end: number }> = [];\n    const warnings: string[] = [];\n\n    // Simple regex to find style tags (this is safe for finding tags, not for parsing content)\n    const styleRegex = /<style[^>]*>([\\s\\S]*?)<\\/style>/gi;\n    let match;\n\n    while ((match = styleRegex.exec(html)) !== null) {\n      styles.push({\n        content: match[1].trim(),\n        start: match.index,\n        end: match.index + match[0].length\n      });\n    }\n\n    return { styles, warnings };\n  }\n\n  async parseInternalCSS(html: string, filePath: string): Promise<{\n    html: string;\n    rules: CSSRule[];\n    hasChanges: boolean;\n    warnings: string[];\n  }> {\n    const allRules: CSSRule[] = [];\n    const allWarnings: string[] = [];\n    let modifiedHtml = html;\n    let hasChanges = false;\n\n    const { styles } = this.parseInternalStyle(html);\n\n    // Process styles in reverse order to preserve indices\n    for (let i = styles.length - 1; i >= 0; i--) {\n      const style = styles[i];\n      \n      try {\n        const result = await this.parse(style.content, filePath);\n        \n        allRules.push(...result.rules);\n        allWarnings.push(...result.warnings);\n\n        if (result.hasChanges) {\n          hasChanges = true;\n\n          if (result.canDelete || result.css.trim() === '') {\n            // Remove entire style tag\n            modifiedHtml = modifiedHtml.slice(0, style.start) + modifiedHtml.slice(style.end);\n          } else {\n            // Replace style content\n            const before = modifiedHtml.slice(0, style.start);\n            const after = modifiedHtml.slice(style.end);\n            const tagStart = html.slice(style.start).match(/<style[^>]*>/)?.[0] || '<style>';\n            const tagEnd = '</style>';\n            modifiedHtml = before + tagStart + '\\n' + result.css + '\\n' + tagEnd + after;\n          }\n        }\n      } catch (error) {\n        logger.warn(`Failed to parse internal style block: ${error}`);\n        allWarnings.push(`Failed to parse internal style: ${error}`);\n      }\n    }\n\n    return {\n      html: modifiedHtml,\n      rules: allRules,\n      hasChanges,\n      warnings: allWarnings\n    };\n  }\n\n  extractImportPaths(code: string): string[] {\n    const imports: string[] = [];\n    \n    // Match CSS imports\n    const importRegex = /import\\s+['\"]([^'\"]+\\.css)['\"];?/g;\n    let match;\n    \n    while ((match = importRegex.exec(code)) !== null) {\n      imports.push(match[1]);\n    }\n\n    // Match require statements\n    const requireRegex = /require\\s*\\(\\s*['\"]([^'\"]+\\.css)['\"]\\s*\\)/g;\n    while ((match = requireRegex.exec(code)) !== null) {\n      imports.push(match[1]);\n    }\n\n    return imports;\n  }\n}\n"]}
|
|
285
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cssParser.js","sourceRoot":"","sources":["../src/cssParser.ts"],"names":[],"mappings":"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","sourcesContent":["import postcss, { Root, Rule, Declaration, AtRule } from 'postcss';\nimport safeParser from 'postcss-safe-parser';\nimport { TailwindMapper, CSSProperty } from './tailwindMapper';\nimport { logger } from './utils/logger';\nimport { \n  Breakpoint, \n  getBreakpoints, \n  resolveBreakpointsFromConfig,\n  processMediaQuery \n} from './utils/breakpointResolver';\nimport {\n  processPseudoSelector,\n  parseMultipleSelectors,\n  ParsedSelector\n} from './utils/pseudoSelectorResolver';\nimport {\n  assembleUtility,\n  assembleUtilities,\n  MergedUtility,\n  mergeUtilities,\n  normalizeVariantOrder\n} from './utils/variantAssembler';\n\nexport interface UtilityWithVariant {\n  value: string;\n  variants: string[];\n}\n\nexport interface CSSRule {\n  selector: string;\n  className: string;\n  declarations: CSSProperty[];\n  convertedClasses: string[];\n  utilities: UtilityWithVariant[];\n  skipped: boolean;\n  fullyConverted: boolean;\n  partialConversion: boolean;\n  reason?: string;\n}\n\nexport interface CSSParseResult {\n  css: string;\n  rules: CSSRule[];\n  hasChanges: boolean;\n  canDelete: boolean;\n  warnings: string[];\n}\n\nexport interface CSSUsageMap {\n  [className: string]: string[];\n}\n\nexport class CSSParser {\n  private mapper: TailwindMapper;\n  private breakpoints: Breakpoint[];\n\n  constructor(mapper: TailwindMapper, screens?: Record<string, string | [string, string]>) {\n    this.mapper = mapper;\n    this.breakpoints = screens \n      ? resolveBreakpointsFromConfig(screens) \n      : getBreakpoints();\n  }\n\n  private convertDeclarations(declarations: CSSProperty[]): {\n    utilities: UtilityWithVariant[];\n    conversionResults: Array<{ declaration: CSSProperty; converted: boolean; className: string | null }>;\n    conversionWarnings: string[];\n  } {\n    const conversionResults: Array<{\n      declaration: CSSProperty;\n      converted: boolean;\n      className: string | null;\n    }> = [];\n    const conversionWarnings: string[] = [];\n\n    declarations.forEach(decl => {\n      const result = this.mapper.convertProperty(decl.property, decl.value);\n      conversionResults.push({\n        declaration: decl,\n        converted: !result.skipped && result.className !== null,\n        className: result.className\n      });\n      if (result.skipped && result.reason) {\n        conversionWarnings.push(result.reason);\n      }\n    });\n\n    const utilities: UtilityWithVariant[] = conversionResults\n      .filter(r => r.converted && r.className)\n      .map(r => ({\n        value: r.className!,\n        variants: []\n      }));\n\n    return { utilities, conversionResults, conversionWarnings };\n  }\n\n  private processRuleWithVariants(\n    rule: Rule,\n    additionalVariants: string[] = []\n  ): { cssRules: CSSRule[]; conversionResults: Array<{ declaration: CSSProperty; converted: boolean; className: string | null }>[]; conversionWarnings: string[] } | null {\n    const selector = rule.selector;\n    const parsedSelectors = parseMultipleSelectors(selector);\n    \n    const validSelectors = parsedSelectors.filter(s => !s.isComplex && s.baseClass);\n    \n    if (validSelectors.length === 0) {\n      return null;\n    }\n\n    const declarations: CSSProperty[] = [];\n\n    rule.walkDecls((decl) => {\n      if (decl.prop.startsWith('--')) {\n        return;\n      }\n\n      if (decl.value.includes('calc(')) {\n        return;\n      }\n\n      declarations.push({\n        property: decl.prop,\n        value: decl.value\n      });\n    });\n\n    if (declarations.length === 0) {\n      return null;\n    }\n\n    const { utilities, conversionResults, conversionWarnings } = this.convertDeclarations(declarations);\n    \n    const utilitiesWithVariants = utilities.map(u => ({\n      value: u.value,\n      variants: normalizeVariantOrder([...u.variants, ...additionalVariants])\n    }));\n\n    const cssRules: CSSRule[] = [];\n    const allConversionResults: Array<{ declaration: CSSProperty; converted: boolean; className: string | null }>[] = [];\n\n    for (const parsed of validSelectors) {\n      const pseudoVariants = parsed.pseudos || [];\n      const allVariants = normalizeVariantOrder([...pseudoVariants, ...additionalVariants]);\n      \n      const utilitiesForSelector = utilities.map(u => ({\n        value: u.value,\n        variants: allVariants\n      }));\n\n      const convertedClasses = assembleUtilities(utilitiesForSelector);\n\n      const allDeclarationsConverted = conversionResults.every(r => r.converted);\n      const someDeclarationsConverted = convertedClasses.length > 0;\n\n      const cssRule: CSSRule = {\n        selector: selector,\n        className: parsed.baseClass,\n        declarations,\n        convertedClasses,\n        utilities: utilitiesForSelector,\n        skipped: !someDeclarationsConverted,\n        fullyConverted: allDeclarationsConverted,\n        partialConversion: someDeclarationsConverted && !allDeclarationsConverted,\n        reason: !someDeclarationsConverted ? 'No convertible declarations' : undefined\n      };\n\n      cssRules.push(cssRule);\n      allConversionResults.push(conversionResults);\n    }\n\n    return { cssRules, conversionResults: allConversionResults, conversionWarnings };\n  }\n\n  async parse(css: string, filePath: string): Promise<CSSParseResult> {\n    const rules: CSSRule[] = [];\n    const warnings: string[] = [];\n    let hasChanges = false;\n\n    try {\n      const root = await postcss().process(css, {\n        parser: safeParser,\n        from: filePath\n      }).then(result => result.root);\n\n      root.walkAtRules((atRule) => {\n        if (atRule.name !== 'media') {\n          return;\n        }\n\n        const mediaResult = processMediaQuery(atRule.params, this.breakpoints);\n\n        if (mediaResult.skipped) {\n          warnings.push(mediaResult.reason || `Skipped media query: ${atRule.params}`);\n          return;\n        }\n\n        const responsiveVariant = mediaResult.breakpoint!;\n\n        const nestedRules: Rule[] = [];\n        atRule.walkRules((rule) => {\n          nestedRules.push(rule);\n        });\n\n        for (const rule of nestedRules) {\n          const result = this.processRuleWithVariants(rule, [responsiveVariant]);\n          if (result) {\n            rules.push(...result.cssRules);\n            warnings.push(...result.conversionWarnings);\n\n            const anyConverted = result.cssRules.some(r => r.convertedClasses.length > 0);\n            if (anyConverted) {\n              hasChanges = true;\n\n              const allFullyConverted = result.cssRules.every(r => r.fullyConverted);\n              if (allFullyConverted) {\n                rule.remove();\n                const classNames = result.cssRules.map(r => r.className).join(', .');\n                logger.verbose(`Removed rule .${classNames} in @media (min-width) → ${responsiveVariant}`);\n              } else {\n                for (const cr of result.conversionResults.flat()) {\n                  if (cr.converted) {\n                    rule.walkDecls((decl) => {\n                      if (decl.prop === cr.declaration.property && decl.value === cr.declaration.value) {\n                        decl.remove();\n                      }\n                    });\n                  }\n                }\n                logger.verbose(`Partial conversion in @media → ${responsiveVariant}`);\n              }\n            }\n          } else {\n            warnings.push(`Skipped rule in @media: ${rule.selector}`);\n          }\n        }\n\n        if (atRule.nodes && atRule.nodes.length === 0) {\n          atRule.remove();\n          logger.verbose(`Removed empty @media rule`);\n        }\n      });\n\n      root.walkRules((rule) => {\n        if (rule.parent && rule.parent.type === 'atrule') {\n          return;\n        }\n\n        const result = this.processRuleWithVariants(rule);\n        \n        if (!result) {\n          const parsedSelectors = parseMultipleSelectors(rule.selector);\n          const allComplex = parsedSelectors.every(s => s.isComplex);\n          \n          if (allComplex) {\n            const reasons = parsedSelectors.map(s => s.reason).filter(Boolean);\n            warnings.push(...reasons as string[]);\n            logger.verbose(`Skipping complex selector: ${rule.selector}`);\n          }\n          return;\n        }\n\n        rules.push(...result.cssRules);\n        warnings.push(...result.conversionWarnings);\n\n        const anyConverted = result.cssRules.some(r => r.convertedClasses.length > 0);\n        if (anyConverted) {\n          hasChanges = true;\n\n          const allFullyConverted = result.cssRules.every(r => r.fullyConverted);\n          if (allFullyConverted) {\n            rule.remove();\n            const classNames = result.cssRules.map(r => r.className).join(', .');\n            logger.verbose(`Removed rule .${classNames} (all declarations converted)`);\n          } else {\n            for (const cr of result.conversionResults.flat()) {\n              if (cr.converted) {\n                rule.walkDecls((decl) => {\n                  if (decl.prop === cr.declaration.property && decl.value === cr.declaration.value) {\n                    decl.remove();\n                  }\n                });\n              }\n            }\n            logger.verbose(`Partial conversion of rule`);\n          }\n        }\n      });\n\n      root.walkAtRules((atRule) => {\n        if (atRule.nodes && atRule.nodes.length === 0) {\n          atRule.remove();\n        }\n      });\n\n      const canDelete = root.nodes.length === 0;\n      const newCss = root.toString();\n\n      return {\n        css: newCss,\n        rules,\n        hasChanges,\n        canDelete,\n        warnings\n      };\n\n    } catch (error) {\n      logger.error(`Failed to parse CSS ${filePath}:`, error);\n      throw new Error(`CSS parsing failed: ${error}`);\n    }\n  }\n\n  parseInternalStyle(html: string): { \n    styles: Array<{ content: string; start: number; end: number }>;\n    warnings: string[];\n  } {\n    const styles: Array<{ content: string; start: number; end: number }> = [];\n    const warnings: string[] = [];\n\n    const styleRegex = /<style[^>]*>([\\s\\S]*?)<\\/style>/gi;\n    let match;\n\n    while ((match = styleRegex.exec(html)) !== null) {\n      styles.push({\n        content: match[1].trim(),\n        start: match.index,\n        end: match.index + match[0].length\n      });\n    }\n\n    return { styles, warnings };\n  }\n\n  async parseInternalCSS(html: string, filePath: string): Promise<{\n    html: string;\n    rules: CSSRule[];\n    hasChanges: boolean;\n    warnings: string[];\n  }> {\n    const allRules: CSSRule[] = [];\n    const allWarnings: string[] = [];\n    let modifiedHtml = html;\n    let hasChanges = false;\n\n    const { styles } = this.parseInternalStyle(html);\n\n    for (let i = styles.length - 1; i >= 0; i--) {\n      const style = styles[i];\n      \n      try {\n        const result = await this.parse(style.content, filePath);\n        \n        allRules.push(...result.rules);\n        allWarnings.push(...result.warnings);\n\n        if (result.hasChanges) {\n          hasChanges = true;\n\n          if (result.canDelete || result.css.trim() === '') {\n            modifiedHtml = modifiedHtml.slice(0, style.start) + modifiedHtml.slice(style.end);\n          } else {\n            const before = modifiedHtml.slice(0, style.start);\n            const after = modifiedHtml.slice(style.end);\n            const tagStart = html.slice(style.start).match(/<style[^>]*>/)?.[0] || '<style>';\n            const tagEnd = '</style>';\n            modifiedHtml = before + tagStart + '\\n' + result.css + '\\n' + tagEnd + after;\n          }\n        }\n      } catch (error) {\n        logger.warn(`Failed to parse internal style block: ${error}`);\n        allWarnings.push(`Failed to parse internal style: ${error}`);\n      }\n    }\n\n    return {\n      html: modifiedHtml,\n      rules: allRules,\n      hasChanges,\n      warnings: allWarnings\n    };\n  }\n\n  extractImportPaths(code: string): string[] {\n    const imports: string[] = [];\n    \n    const importRegex = /import\\s+['\"]([^'\"]+\\.css)['\"];?/g;\n    let match;\n    \n    while ((match = importRegex.exec(code)) !== null) {\n      imports.push(match[1]);\n    }\n\n    const requireRegex = /require\\s*\\(\\s*['\"]([^'\"]+\\.css)['\"]\\s*\\)/g;\n    while ((match = requireRegex.exec(code)) !== null) {\n      imports.push(match[1]);\n    }\n\n    return imports;\n  }\n}"]}
|
package/dist/index.d.ts
CHANGED
|
@@ -7,3 +7,5 @@ export { FileWriter, FileWriteOptions } from './fileWriter';
|
|
|
7
7
|
export { loadTailwindConfig, TailwindConfig } from './utils/config';
|
|
8
8
|
export { logger } from './utils/logger';
|
|
9
9
|
export { Breakpoint, MediaQueryInfo, getDefaultBreakpoints, resolveBreakpointsFromConfig, parseMediaQuery, findBreakpointForMinWidth, processMediaQuery, prefixWithBreakpoint } from './utils/breakpointResolver';
|
|
10
|
+
export { ParsedSelector, PSEUDO_TO_VARIANT, SUPPORTED_PSEUDOS, parseSelector, mapPseudoToVariant, processPseudoSelector, parseMultipleSelectors } from './utils/pseudoSelectorResolver';
|
|
11
|
+
export { VARIANT_ORDER, isResponsiveVariant, isPseudoVariant, sortVariants, deduplicateVariants, normalizeVariantOrder, assembleUtility, assembleUtilities, mergeUtilities, MergedUtility } from './utils/variantAssembler';
|
package/dist/index.js
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.prefixWithBreakpoint = exports.processMediaQuery = exports.findBreakpointForMinWidth = exports.parseMediaQuery = exports.resolveBreakpointsFromConfig = exports.getDefaultBreakpoints = exports.logger = exports.loadTailwindConfig = exports.FileWriter = exports.CSSParser = exports.JSXParser = exports.TailwindMapper = exports.transformFiles = exports.scanProject = void 0;
|
|
3
|
+
exports.mergeUtilities = exports.assembleUtilities = exports.assembleUtility = exports.normalizeVariantOrder = exports.deduplicateVariants = exports.sortVariants = exports.isPseudoVariant = exports.isResponsiveVariant = exports.VARIANT_ORDER = exports.parseMultipleSelectors = exports.processPseudoSelector = exports.mapPseudoToVariant = exports.parseSelector = exports.SUPPORTED_PSEUDOS = exports.PSEUDO_TO_VARIANT = exports.prefixWithBreakpoint = exports.processMediaQuery = exports.findBreakpointForMinWidth = exports.parseMediaQuery = exports.resolveBreakpointsFromConfig = exports.getDefaultBreakpoints = exports.logger = exports.loadTailwindConfig = exports.FileWriter = exports.CSSParser = exports.JSXParser = exports.TailwindMapper = exports.transformFiles = exports.scanProject = void 0;
|
|
4
4
|
// Export public API
|
|
5
5
|
var scanner_1 = require("./scanner");
|
|
6
6
|
Object.defineProperty(exports, "scanProject", { enumerable: true, get: function () { return scanner_1.scanProject; } });
|
|
@@ -25,4 +25,21 @@ Object.defineProperty(exports, "parseMediaQuery", { enumerable: true, get: funct
|
|
|
25
25
|
Object.defineProperty(exports, "findBreakpointForMinWidth", { enumerable: true, get: function () { return breakpointResolver_1.findBreakpointForMinWidth; } });
|
|
26
26
|
Object.defineProperty(exports, "processMediaQuery", { enumerable: true, get: function () { return breakpointResolver_1.processMediaQuery; } });
|
|
27
27
|
Object.defineProperty(exports, "prefixWithBreakpoint", { enumerable: true, get: function () { return breakpointResolver_1.prefixWithBreakpoint; } });
|
|
28
|
-
|
|
28
|
+
var pseudoSelectorResolver_1 = require("./utils/pseudoSelectorResolver");
|
|
29
|
+
Object.defineProperty(exports, "PSEUDO_TO_VARIANT", { enumerable: true, get: function () { return pseudoSelectorResolver_1.PSEUDO_TO_VARIANT; } });
|
|
30
|
+
Object.defineProperty(exports, "SUPPORTED_PSEUDOS", { enumerable: true, get: function () { return pseudoSelectorResolver_1.SUPPORTED_PSEUDOS; } });
|
|
31
|
+
Object.defineProperty(exports, "parseSelector", { enumerable: true, get: function () { return pseudoSelectorResolver_1.parseSelector; } });
|
|
32
|
+
Object.defineProperty(exports, "mapPseudoToVariant", { enumerable: true, get: function () { return pseudoSelectorResolver_1.mapPseudoToVariant; } });
|
|
33
|
+
Object.defineProperty(exports, "processPseudoSelector", { enumerable: true, get: function () { return pseudoSelectorResolver_1.processPseudoSelector; } });
|
|
34
|
+
Object.defineProperty(exports, "parseMultipleSelectors", { enumerable: true, get: function () { return pseudoSelectorResolver_1.parseMultipleSelectors; } });
|
|
35
|
+
var variantAssembler_1 = require("./utils/variantAssembler");
|
|
36
|
+
Object.defineProperty(exports, "VARIANT_ORDER", { enumerable: true, get: function () { return variantAssembler_1.VARIANT_ORDER; } });
|
|
37
|
+
Object.defineProperty(exports, "isResponsiveVariant", { enumerable: true, get: function () { return variantAssembler_1.isResponsiveVariant; } });
|
|
38
|
+
Object.defineProperty(exports, "isPseudoVariant", { enumerable: true, get: function () { return variantAssembler_1.isPseudoVariant; } });
|
|
39
|
+
Object.defineProperty(exports, "sortVariants", { enumerable: true, get: function () { return variantAssembler_1.sortVariants; } });
|
|
40
|
+
Object.defineProperty(exports, "deduplicateVariants", { enumerable: true, get: function () { return variantAssembler_1.deduplicateVariants; } });
|
|
41
|
+
Object.defineProperty(exports, "normalizeVariantOrder", { enumerable: true, get: function () { return variantAssembler_1.normalizeVariantOrder; } });
|
|
42
|
+
Object.defineProperty(exports, "assembleUtility", { enumerable: true, get: function () { return variantAssembler_1.assembleUtility; } });
|
|
43
|
+
Object.defineProperty(exports, "assembleUtilities", { enumerable: true, get: function () { return variantAssembler_1.assembleUtilities; } });
|
|
44
|
+
Object.defineProperty(exports, "mergeUtilities", { enumerable: true, get: function () { return variantAssembler_1.mergeUtilities; } });
|
|
45
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsb0JBQW9CO0FBQ3BCLHFDQUFxRDtBQUE1QyxzR0FBQSxXQUFXLE9BQUE7QUFDcEIsNkNBQW1GO0FBQTFFLDZHQUFBLGNBQWMsT0FBQTtBQUN2QixtREFBaUY7QUFBeEUsZ0hBQUEsY0FBYyxPQUFBO0FBQ3ZCLHlDQUEyRTtBQUFsRSxzR0FBQSxTQUFTLE9BQUE7QUFDbEIseUNBQXFGO0FBQTVFLHNHQUFBLFNBQVMsT0FBQTtBQUNsQiwyQ0FBNEQ7QUFBbkQsd0dBQUEsVUFBVSxPQUFBO0FBQ25CLHlDQUFvRTtBQUEzRCw0R0FBQSxrQkFBa0IsT0FBQTtBQUMzQix5Q0FBd0M7QUFBL0IsZ0dBQUEsTUFBTSxPQUFBO0FBQ2YsaUVBU29DO0FBTmxDLDJIQUFBLHFCQUFxQixPQUFBO0FBQ3JCLGtJQUFBLDRCQUE0QixPQUFBO0FBQzVCLHFIQUFBLGVBQWUsT0FBQTtBQUNmLCtIQUFBLHlCQUF5QixPQUFBO0FBQ3pCLHVIQUFBLGlCQUFpQixPQUFBO0FBQ2pCLDBIQUFBLG9CQUFvQixPQUFBO0FBRXRCLHlFQVF3QztBQU50QywySEFBQSxpQkFBaUIsT0FBQTtBQUNqQiwySEFBQSxpQkFBaUIsT0FBQTtBQUNqQix1SEFBQSxhQUFhLE9BQUE7QUFDYiw0SEFBQSxrQkFBa0IsT0FBQTtBQUNsQiwrSEFBQSxxQkFBcUIsT0FBQTtBQUNyQixnSUFBQSxzQkFBc0IsT0FBQTtBQUV4Qiw2REFXa0M7QUFWaEMsaUhBQUEsYUFBYSxPQUFBO0FBQ2IsdUhBQUEsbUJBQW1CLE9BQUE7QUFDbkIsbUhBQUEsZUFBZSxPQUFBO0FBQ2YsZ0hBQUEsWUFBWSxPQUFBO0FBQ1osdUhBQUEsbUJBQW1CLE9BQUE7QUFDbkIseUhBQUEscUJBQXFCLE9BQUE7QUFDckIsbUhBQUEsZUFBZSxPQUFBO0FBQ2YscUhBQUEsaUJBQWlCLE9BQUE7QUFDakIsa0hBQUEsY0FBYyxPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLy8gRXhwb3J0IHB1YmxpYyBBUElcbmV4cG9ydCB7IHNjYW5Qcm9qZWN0LCBTY2FubmVkRmlsZSB9IGZyb20gJy4vc2Nhbm5lcic7XG5leHBvcnQgeyB0cmFuc2Zvcm1GaWxlcywgVHJhbnNmb3JtT3B0aW9ucywgVHJhbnNmb3JtUmVzdWx0cyB9IGZyb20gJy4vdHJhbnNmb3JtZXInO1xuZXhwb3J0IHsgVGFpbHdpbmRNYXBwZXIsIENTU1Byb3BlcnR5LCBDb252ZXJzaW9uUmVzdWx0IH0gZnJvbSAnLi90YWlsd2luZE1hcHBlcic7XG5leHBvcnQgeyBKU1hQYXJzZXIsIEpTWFRyYW5zZm9ybWF0aW9uLCBKU1hQYXJzZVJlc3VsdCB9IGZyb20gJy4vanN4UGFyc2VyJztcbmV4cG9ydCB7IENTU1BhcnNlciwgQ1NTUnVsZSwgQ1NTUGFyc2VSZXN1bHQsIFV0aWxpdHlXaXRoVmFyaWFudCB9IGZyb20gJy4vY3NzUGFyc2VyJztcbmV4cG9ydCB7IEZpbGVXcml0ZXIsIEZpbGVXcml0ZU9wdGlvbnMgfSBmcm9tICcuL2ZpbGVXcml0ZXInO1xuZXhwb3J0IHsgbG9hZFRhaWx3aW5kQ29uZmlnLCBUYWlsd2luZENvbmZpZyB9IGZyb20gJy4vdXRpbHMvY29uZmlnJztcbmV4cG9ydCB7IGxvZ2dlciB9IGZyb20gJy4vdXRpbHMvbG9nZ2VyJztcbmV4cG9ydCB7XG4gIEJyZWFrcG9pbnQsXG4gIE1lZGlhUXVlcnlJbmZvLFxuICBnZXREZWZhdWx0QnJlYWtwb2ludHMsXG4gIHJlc29sdmVCcmVha3BvaW50c0Zyb21Db25maWcsXG4gIHBhcnNlTWVkaWFRdWVyeSxcbiAgZmluZEJyZWFrcG9pbnRGb3JNaW5XaWR0aCxcbiAgcHJvY2Vzc01lZGlhUXVlcnksXG4gIHByZWZpeFdpdGhCcmVha3BvaW50XG59IGZyb20gJy4vdXRpbHMvYnJlYWtwb2ludFJlc29sdmVyJztcbmV4cG9ydCB7XG4gIFBhcnNlZFNlbGVjdG9yLFxuICBQU0VVRE9fVE9fVkFSSUFOVCxcbiAgU1VQUE9SVEVEX1BTRVVET1MsXG4gIHBhcnNlU2VsZWN0b3IsXG4gIG1hcFBzZXVkb1RvVmFyaWFudCxcbiAgcHJvY2Vzc1BzZXVkb1NlbGVjdG9yLFxuICBwYXJzZU11bHRpcGxlU2VsZWN0b3JzXG59IGZyb20gJy4vdXRpbHMvcHNldWRvU2VsZWN0b3JSZXNvbHZlcic7XG5leHBvcnQge1xuICBWQVJJQU5UX09SREVSLFxuICBpc1Jlc3BvbnNpdmVWYXJpYW50LFxuICBpc1BzZXVkb1ZhcmlhbnQsXG4gIHNvcnRWYXJpYW50cyxcbiAgZGVkdXBsaWNhdGVWYXJpYW50cyxcbiAgbm9ybWFsaXplVmFyaWFudE9yZGVyLFxuICBhc3NlbWJsZVV0aWxpdHksXG4gIGFzc2VtYmxlVXRpbGl0aWVzLFxuICBtZXJnZVV0aWxpdGllcyxcbiAgTWVyZ2VkVXRpbGl0eVxufSBmcm9tICcuL3V0aWxzL3ZhcmlhbnRBc3NlbWJsZXInO1xuIl19
|
package/dist/transformer.js
CHANGED
|
@@ -12,6 +12,7 @@ const cssParser_1 = require("./cssParser");
|
|
|
12
12
|
const fileWriter_1 = require("./fileWriter");
|
|
13
13
|
const logger_1 = require("./utils/logger");
|
|
14
14
|
const breakpointResolver_1 = require("./utils/breakpointResolver");
|
|
15
|
+
const variantAssembler_1 = require("./utils/variantAssembler");
|
|
15
16
|
async function transformFiles(files, options) {
|
|
16
17
|
const results = {
|
|
17
18
|
filesScanned: files.length,
|
|
@@ -26,24 +27,18 @@ async function transformFiles(files, options) {
|
|
|
26
27
|
const cssParser = new cssParser_1.CSSParser(mapper, screens);
|
|
27
28
|
const fileWriter = new fileWriter_1.FileWriter({ dryRun: options.dryRun });
|
|
28
29
|
(0, breakpointResolver_1.clearBreakpointCache)();
|
|
29
|
-
// PASS 1: Analyze all files WITHOUT modifying anything
|
|
30
|
-
// Collect CSS mappings and gather info about what can be safely converted
|
|
31
30
|
const cssClassMap = {};
|
|
32
31
|
const cssFileResults = new Map();
|
|
33
32
|
logger_1.logger.info('\n🔍 Phase 1: Analyzing files...');
|
|
34
|
-
// Analyze CSS files
|
|
35
33
|
if (!options.skipExternal) {
|
|
36
34
|
for (const file of files.filter(f => f.type === 'css')) {
|
|
37
35
|
try {
|
|
38
36
|
const content = fs_1.default.readFileSync(file.path, 'utf-8');
|
|
39
37
|
const result = await cssParser.parse(content, file.path);
|
|
40
|
-
// Check if ALL rules in this file are FULLY converted (all declarations)
|
|
41
38
|
const totalRules = result.rules.length;
|
|
42
39
|
const fullyConvertedRules = result.rules.filter(r => r.fullyConverted).length;
|
|
43
40
|
const partiallyConvertedRules = result.rules.filter(r => r.partialConversion).length;
|
|
44
|
-
// A file is only "fully convertible" if ALL rules are fully converted (no partial conversions)
|
|
45
41
|
const fullyConvertible = totalRules > 0 && totalRules === fullyConvertedRules && partiallyConvertedRules === 0;
|
|
46
|
-
// Build class map (only for fully converted classes - partial conversions keep the CSS)
|
|
47
42
|
result.rules.forEach(rule => {
|
|
48
43
|
if (rule.fullyConverted) {
|
|
49
44
|
const existing = cssClassMap[rule.className];
|
|
@@ -69,13 +64,11 @@ async function transformFiles(files, options) {
|
|
|
69
64
|
fullyConvertible
|
|
70
65
|
});
|
|
71
66
|
results.warnings += result.warnings.length;
|
|
72
|
-
// Log analysis
|
|
73
67
|
logger_1.logger.verbose(`Analyzed ${file.path}:`);
|
|
74
68
|
logger_1.logger.verbose(` - Total rules: ${totalRules}`);
|
|
75
69
|
logger_1.logger.verbose(` - Fully converted rules: ${fullyConvertedRules}`);
|
|
76
70
|
logger_1.logger.verbose(` - Partially converted rules: ${partiallyConvertedRules}`);
|
|
77
71
|
logger_1.logger.verbose(` - Fully convertible: ${fullyConvertible}`);
|
|
78
|
-
// Log warnings
|
|
79
72
|
result.warnings.forEach(warning => {
|
|
80
73
|
logger_1.logger.verbose(`⚠️ ${file.path}: ${warning}`);
|
|
81
74
|
});
|
|
@@ -86,7 +79,6 @@ async function transformFiles(files, options) {
|
|
|
86
79
|
}
|
|
87
80
|
}
|
|
88
81
|
}
|
|
89
|
-
// PASS 2: Transform JSX/TSX files
|
|
90
82
|
logger_1.logger.info('\n⚛️ Phase 2: Transforming React components...');
|
|
91
83
|
const jsxFileResults = new Map();
|
|
92
84
|
for (const file of files.filter(f => f.type === 'jsx')) {
|
|
@@ -95,7 +87,6 @@ async function transformFiles(files, options) {
|
|
|
95
87
|
const originalContent = content;
|
|
96
88
|
let hasChanges = false;
|
|
97
89
|
let fileWarnings = [];
|
|
98
|
-
// Process inline styles
|
|
99
90
|
if (!options.skipInline) {
|
|
100
91
|
try {
|
|
101
92
|
const jsxResult = jsxParser.parse(content, file.path);
|
|
@@ -111,14 +102,12 @@ async function transformFiles(files, options) {
|
|
|
111
102
|
fileWarnings.push(`JSX parse error: ${error}`);
|
|
112
103
|
}
|
|
113
104
|
}
|
|
114
|
-
// Process internal CSS
|
|
115
105
|
if (!options.skipInternal) {
|
|
116
106
|
try {
|
|
117
107
|
const internalResult = await cssParser.parseInternalCSS(content, file.path);
|
|
118
108
|
if (internalResult.hasChanges) {
|
|
119
109
|
content = internalResult.html;
|
|
120
110
|
hasChanges = true;
|
|
121
|
-
// Build class map from internal styles
|
|
122
111
|
internalResult.rules.forEach(rule => {
|
|
123
112
|
if (rule.convertedClasses.length > 0) {
|
|
124
113
|
const existing = cssClassMap[rule.className];
|
|
@@ -145,7 +134,6 @@ async function transformFiles(files, options) {
|
|
|
145
134
|
hasChanges
|
|
146
135
|
});
|
|
147
136
|
results.warnings += fileWarnings.length;
|
|
148
|
-
// Log warnings
|
|
149
137
|
fileWarnings.forEach(warning => {
|
|
150
138
|
logger_1.logger.verbose(`⚠️ ${file.path}: ${warning}`);
|
|
151
139
|
});
|
|
@@ -155,14 +143,11 @@ async function transformFiles(files, options) {
|
|
|
155
143
|
results.warnings++;
|
|
156
144
|
}
|
|
157
145
|
}
|
|
158
|
-
// PASS 3: Replace className references from external CSS
|
|
159
|
-
// This must happen after all JSX files are parsed
|
|
160
146
|
if (Object.keys(cssClassMap).length > 0) {
|
|
161
147
|
logger_1.logger.info('\n🔄 Phase 3: Replacing className references...');
|
|
162
148
|
for (const [filePath, fileResult] of jsxFileResults) {
|
|
163
149
|
let content = fileResult.newContent;
|
|
164
150
|
let hasChanges = fileResult.hasChanges;
|
|
165
|
-
// Replace className references
|
|
166
151
|
const replacementResult = replaceClassNameReferences(content, cssClassMap);
|
|
167
152
|
if (replacementResult.hasChanges) {
|
|
168
153
|
content = replacementResult.code;
|
|
@@ -170,7 +155,6 @@ async function transformFiles(files, options) {
|
|
|
170
155
|
results.classesReplaced += replacementResult.replacements;
|
|
171
156
|
logger_1.logger.verbose(`Replaced ${replacementResult.replacements} class references in ${path_1.default.basename(filePath)}`);
|
|
172
157
|
}
|
|
173
|
-
// Update the result
|
|
174
158
|
jsxFileResults.set(filePath, {
|
|
175
159
|
...fileResult,
|
|
176
160
|
newContent: content,
|
|
@@ -178,9 +162,7 @@ async function transformFiles(files, options) {
|
|
|
178
162
|
});
|
|
179
163
|
}
|
|
180
164
|
}
|
|
181
|
-
// PASS 4: Write all changes
|
|
182
165
|
logger_1.logger.info('\n💾 Phase 4: Writing changes...');
|
|
183
|
-
// Write JSX files
|
|
184
166
|
for (const [filePath, fileResult] of jsxFileResults) {
|
|
185
167
|
if (fileResult.hasChanges) {
|
|
186
168
|
const success = await fileWriter.writeFile(filePath, fileResult.newContent, fileResult.content);
|
|
@@ -189,19 +171,15 @@ async function transformFiles(files, options) {
|
|
|
189
171
|
}
|
|
190
172
|
}
|
|
191
173
|
}
|
|
192
|
-
// Write CSS files (SAFETY: Only modify if fully convertible or explicitly allowed)
|
|
193
174
|
if (!options.skipExternal) {
|
|
194
175
|
for (const [filePath, fileResult] of cssFileResults) {
|
|
195
176
|
if (!fileResult.hasChanges)
|
|
196
177
|
continue;
|
|
197
|
-
// SAFETY RULE 1: Never modify CSS files that aren't fully convertible
|
|
198
|
-
// unless they only have unconvertible rules (no changes needed)
|
|
199
178
|
if (!fileResult.fullyConvertible) {
|
|
200
179
|
logger_1.logger.warn(`⏭️ Skipping ${path_1.default.basename(filePath)} - not fully convertible (would break styles)`);
|
|
201
180
|
logger_1.logger.warn(` Convertible: ${fileResult.rules.filter(r => r.convertedClasses.length > 0).length}/${fileResult.rules.length} rules`);
|
|
202
181
|
continue;
|
|
203
182
|
}
|
|
204
|
-
// SAFETY RULE 2: Only delete if ALL rules converted AND --delete-css flag used
|
|
205
183
|
if (fileResult.canDelete && options.deleteCss) {
|
|
206
184
|
const success = await fileWriter.deleteFile(filePath);
|
|
207
185
|
if (success) {
|
|
@@ -210,11 +188,9 @@ async function transformFiles(files, options) {
|
|
|
210
188
|
}
|
|
211
189
|
}
|
|
212
190
|
else if (fileResult.canDelete && !options.deleteCss) {
|
|
213
|
-
// File is empty but don't delete without permission
|
|
214
191
|
logger_1.logger.info(`ℹ️ ${path_1.default.basename(filePath)} is now empty (use --delete-css to remove)`);
|
|
215
192
|
}
|
|
216
193
|
else {
|
|
217
|
-
// Write modified CSS (only if fully convertible)
|
|
218
194
|
const success = await fileWriter.writeFile(filePath, fileResult.newContent, fileResult.content);
|
|
219
195
|
if (success) {
|
|
220
196
|
results.filesModified++;
|
|
@@ -225,52 +201,37 @@ async function transformFiles(files, options) {
|
|
|
225
201
|
return results;
|
|
226
202
|
}
|
|
227
203
|
function buildClassInfoFromRule(rule, sourceFile) {
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
204
|
+
return {
|
|
205
|
+
utilities: rule.utilities.map(u => ({
|
|
206
|
+
value: u.value,
|
|
207
|
+
variants: (0, variantAssembler_1.normalizeVariantOrder)([...u.variants])
|
|
208
|
+
})),
|
|
231
209
|
sourceFile,
|
|
232
210
|
fullyConvertible: true
|
|
233
211
|
};
|
|
234
|
-
for (const utility of rule.utilities) {
|
|
235
|
-
if (utility.variant) {
|
|
236
|
-
const existing = info.responsiveClasses.get(utility.variant) || [];
|
|
237
|
-
existing.push(utility.value);
|
|
238
|
-
info.responsiveClasses.set(utility.variant, existing);
|
|
239
|
-
}
|
|
240
|
-
else {
|
|
241
|
-
info.baseClasses.push(utility.value);
|
|
242
|
-
}
|
|
243
|
-
}
|
|
244
|
-
return info;
|
|
245
212
|
}
|
|
246
213
|
function mergeRuleIntoClassInfo(info, rule) {
|
|
247
214
|
for (const utility of rule.utilities) {
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
existing.
|
|
252
|
-
|
|
215
|
+
const existing = info.utilities.find(u => u.value === utility.value);
|
|
216
|
+
if (existing) {
|
|
217
|
+
for (const variant of utility.variants) {
|
|
218
|
+
if (!existing.variants.includes(variant)) {
|
|
219
|
+
existing.variants.push(variant);
|
|
220
|
+
}
|
|
253
221
|
}
|
|
222
|
+
existing.variants = (0, variantAssembler_1.normalizeVariantOrder)(existing.variants);
|
|
254
223
|
}
|
|
255
224
|
else {
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
225
|
+
info.utilities.push({
|
|
226
|
+
value: utility.value,
|
|
227
|
+
variants: (0, variantAssembler_1.normalizeVariantOrder)([...utility.variants])
|
|
228
|
+
});
|
|
259
229
|
}
|
|
260
230
|
}
|
|
261
231
|
}
|
|
262
232
|
function assembleTailwindClasses(info) {
|
|
263
|
-
const
|
|
264
|
-
|
|
265
|
-
for (const bp of sortedBreakpoints) {
|
|
266
|
-
const bpClasses = info.responsiveClasses.get(bp);
|
|
267
|
-
if (bpClasses) {
|
|
268
|
-
for (const cls of bpClasses) {
|
|
269
|
-
classes.push(`${bp}:${cls}`);
|
|
270
|
-
}
|
|
271
|
-
}
|
|
272
|
-
}
|
|
273
|
-
return classes.join(' ');
|
|
233
|
+
const merged = (0, variantAssembler_1.mergeUtilities)(info.utilities);
|
|
234
|
+
return (0, variantAssembler_1.assembleUtilities)(merged).join(' ');
|
|
274
235
|
}
|
|
275
236
|
function replaceClassNameReferences(code, classMap) {
|
|
276
237
|
let hasChanges = false;
|
|
@@ -305,4 +266,4 @@ function replaceClassNameReferences(code, classMap) {
|
|
|
305
266
|
});
|
|
306
267
|
return { code: modifiedCode, hasChanges, replacements };
|
|
307
268
|
}
|
|
308
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transformer.js","sourceRoot":"","sources":["../src/transformer.ts"],"names":[],"mappings":"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","sourcesContent":["import fs from 'fs';\nimport path from 'path';\nimport { ScannedFile } from './scanner';\nimport { TailwindMapper } from './tailwindMapper';\nimport { JSXParser } from './jsxParser';\nimport { CSSParser, CSSRule, UtilityWithVariant } from './cssParser';\nimport { FileWriter } from './fileWriter';\nimport { TailwindConfig } from './utils/config';\nimport { logger } from './utils/logger';\nimport { clearBreakpointCache } from './utils/breakpointResolver';\n\nexport interface TransformOptions {\n  dryRun: boolean;\n  deleteCss: boolean;\n  skipExternal: boolean;\n  skipInline: boolean;\n  skipInternal: boolean;\n  tailwindConfig: TailwindConfig | null;\n  projectRoot: string;\n}\n\nexport interface TransformResults {\n  filesScanned: number;\n  filesModified: number;\n  stylesConverted: number;\n  classesReplaced: number;\n  warnings: number;\n}\n\ninterface ClassInfo {\n  baseClasses: string[];\n  responsiveClasses: Map<string, string[]>;\n  sourceFile: string;\n  fullyConvertible: boolean;\n}\n\ninterface CSSClassMap {\n  [className: string]: ClassInfo;\n}\n\nexport async function transformFiles(\n  files: ScannedFile[],\n  options: TransformOptions\n): Promise<TransformResults> {\n  const results: TransformResults = {\n    filesScanned: files.length,\n    filesModified: 0,\n    stylesConverted: 0,\n    classesReplaced: 0,\n    warnings: 0\n  };\n\n  const mapper = new TailwindMapper(options.tailwindConfig || {});\n  const jsxParser = new JSXParser(mapper);\n  const screens = options.tailwindConfig?.theme?.screens;\n  const cssParser = new CSSParser(mapper, screens);\n  const fileWriter = new FileWriter({ dryRun: options.dryRun });\n\n  clearBreakpointCache();\n\n  // PASS 1: Analyze all files WITHOUT modifying anything\n  // Collect CSS mappings and gather info about what can be safely converted\n  const cssClassMap: CSSClassMap = {};\n  const cssFileResults: Map<string, {\n    content: string;\n    newContent: string;\n    rules: CSSRule[];\n    canDelete: boolean;\n    hasChanges: boolean;\n    fullyConvertible: boolean;\n  }> = new Map();\n\n  logger.info('\\n🔍 Phase 1: Analyzing files...');\n\n  // Analyze CSS files\n  if (!options.skipExternal) {\n    for (const file of files.filter(f => f.type === 'css')) {\n      try {\n        const content = fs.readFileSync(file.path, 'utf-8');\n        const result = await cssParser.parse(content, file.path);\n\n        // Check if ALL rules in this file are FULLY converted (all declarations)\n        const totalRules = result.rules.length;\n        const fullyConvertedRules = result.rules.filter(r => r.fullyConverted).length;\n        const partiallyConvertedRules = result.rules.filter(r => r.partialConversion).length;\n        // A file is only \"fully convertible\" if ALL rules are fully converted (no partial conversions)\n        const fullyConvertible = totalRules > 0 && totalRules === fullyConvertedRules && partiallyConvertedRules === 0;\n\n        // Build class map (only for fully converted classes - partial conversions keep the CSS)\n        result.rules.forEach(rule => {\n          if (rule.fullyConverted) {\n            const existing = cssClassMap[rule.className];\n            if (existing) {\n              mergeRuleIntoClassInfo(existing, rule);\n            } else {\n              cssClassMap[rule.className] = buildClassInfoFromRule(rule, file.path);\n            }\n            results.stylesConverted += rule.declarations.length;\n          } else if (rule.partialConversion) {\n            results.stylesConverted += rule.convertedClasses.length;\n            logger.verbose(`  Rule .${rule.className}: partial conversion (${rule.convertedClasses.length}/${rule.declarations.length} declarations)`);\n          }\n        });\n\n        cssFileResults.set(file.path, {\n          content,\n          newContent: result.css,\n          rules: result.rules,\n          canDelete: result.canDelete,\n          hasChanges: result.hasChanges,\n          fullyConvertible\n        });\n\n        results.warnings += result.warnings.length;\n\n        // Log analysis\n        logger.verbose(`Analyzed ${file.path}:`);\n        logger.verbose(`  - Total rules: ${totalRules}`);\n        logger.verbose(`  - Fully converted rules: ${fullyConvertedRules}`);\n        logger.verbose(`  - Partially converted rules: ${partiallyConvertedRules}`);\n        logger.verbose(`  - Fully convertible: ${fullyConvertible}`);\n\n        // Log warnings\n        result.warnings.forEach(warning => {\n          logger.verbose(`⚠️  ${file.path}: ${warning}`);\n        });\n\n      } catch (error) {\n        logger.error(`Failed to analyze ${file.path}:`, error);\n        results.warnings++;\n      }\n    }\n  }\n\n  // PASS 2: Transform JSX/TSX files\n  logger.info('\\n⚛️  Phase 2: Transforming React components...');\n\n  const jsxFileResults: Map<string, {\n    content: string;\n    newContent: string;\n    hasChanges: boolean;\n  }> = new Map();\n\n  for (const file of files.filter(f => f.type === 'jsx')) {\n    try {\n      let content = fs.readFileSync(file.path, 'utf-8');\n      const originalContent = content;\n      let hasChanges = false;\n      let fileWarnings: string[] = [];\n\n      // Process inline styles\n      if (!options.skipInline) {\n        try {\n          const jsxResult = jsxParser.parse(content, file.path);\n          \n          if (jsxResult.hasChanges) {\n            content = jsxResult.code;\n            hasChanges = true;\n            results.stylesConverted += jsxResult.transformations.reduce(\n              (sum, t) => sum + t.classes.length, 0\n            );\n            fileWarnings.push(...jsxResult.warnings);\n          }\n        } catch (error) {\n          logger.warn(`Failed to parse JSX inline styles in ${file.path}: ${error}`);\n          fileWarnings.push(`JSX parse error: ${error}`);\n        }\n      }\n\n      // Process internal CSS\n      if (!options.skipInternal) {\n        try {\n          const internalResult = await cssParser.parseInternalCSS(content, file.path);\n          \n          if (internalResult.hasChanges) {\n            content = internalResult.html;\n            hasChanges = true;\n            \n            // Build class map from internal styles\n            internalResult.rules.forEach(rule => {\n              if (rule.convertedClasses.length > 0) {\n                const existing = cssClassMap[rule.className];\n                if (existing) {\n                  mergeRuleIntoClassInfo(existing, rule);\n                } else {\n                  cssClassMap[rule.className] = buildClassInfoFromRule(rule, file.path);\n                }\n                results.stylesConverted += rule.declarations.length;\n              }\n            });\n            \n            fileWarnings.push(...internalResult.warnings);\n          }\n        } catch (error) {\n          logger.warn(`Failed to parse internal CSS in ${file.path}: ${error}`);\n          fileWarnings.push(`Internal CSS parse error: ${error}`);\n        }\n      }\n\n      jsxFileResults.set(file.path, {\n        content: originalContent,\n        newContent: content,\n        hasChanges\n      });\n\n      results.warnings += fileWarnings.length;\n\n      // Log warnings\n      fileWarnings.forEach(warning => {\n        logger.verbose(`⚠️  ${file.path}: ${warning}`);\n      });\n\n    } catch (error) {\n      logger.error(`Failed to process ${file.path}:`, error);\n      results.warnings++;\n    }\n  }\n\n  // PASS 3: Replace className references from external CSS\n  // This must happen after all JSX files are parsed\n  if (Object.keys(cssClassMap).length > 0) {\n    logger.info('\\n🔄 Phase 3: Replacing className references...');\n\n    for (const [filePath, fileResult] of jsxFileResults) {\n      let content = fileResult.newContent;\n      let hasChanges = fileResult.hasChanges;\n\n      // Replace className references\n      const replacementResult = replaceClassNameReferences(content, cssClassMap);\n      if (replacementResult.hasChanges) {\n        content = replacementResult.code;\n        hasChanges = true;\n        results.classesReplaced += replacementResult.replacements;\n        \n        logger.verbose(`Replaced ${replacementResult.replacements} class references in ${path.basename(filePath)}`);\n      }\n\n      // Update the result\n      jsxFileResults.set(filePath, {\n        ...fileResult,\n        newContent: content,\n        hasChanges\n      });\n    }\n  }\n\n  // PASS 4: Write all changes\n  logger.info('\\n💾 Phase 4: Writing changes...');\n\n  // Write JSX files\n  for (const [filePath, fileResult] of jsxFileResults) {\n    if (fileResult.hasChanges) {\n      const success = await fileWriter.writeFile(filePath, fileResult.newContent, fileResult.content);\n      if (success) {\n        results.filesModified++;\n      }\n    }\n  }\n\n  // Write CSS files (SAFETY: Only modify if fully convertible or explicitly allowed)\n  if (!options.skipExternal) {\n    for (const [filePath, fileResult] of cssFileResults) {\n      if (!fileResult.hasChanges) continue;\n\n      // SAFETY RULE 1: Never modify CSS files that aren't fully convertible\n      // unless they only have unconvertible rules (no changes needed)\n      if (!fileResult.fullyConvertible) {\n        logger.warn(`⏭️  Skipping ${path.basename(filePath)} - not fully convertible (would break styles)`);\n        logger.warn(`   Convertible: ${fileResult.rules.filter(r => r.convertedClasses.length > 0).length}/${fileResult.rules.length} rules`);\n        continue;\n      }\n\n      // SAFETY RULE 2: Only delete if ALL rules converted AND --delete-css flag used\n      if (fileResult.canDelete && options.deleteCss) {\n        const success = await fileWriter.deleteFile(filePath);\n        if (success) {\n          results.filesModified++;\n          logger.info(`🗑️  Deleted ${path.basename(filePath)} (all rules converted)`);\n        }\n      } else if (fileResult.canDelete && !options.deleteCss) {\n        // File is empty but don't delete without permission\n        logger.info(`ℹ️  ${path.basename(filePath)} is now empty (use --delete-css to remove)`);\n      } else {\n        // Write modified CSS (only if fully convertible)\n        const success = await fileWriter.writeFile(filePath, fileResult.newContent, fileResult.content);\n        if (success) {\n          results.filesModified++;\n        }\n      }\n    }\n  }\n\n  return results;\n}\n\nfunction buildClassInfoFromRule(rule: CSSRule, sourceFile: string): ClassInfo {\n  const info: ClassInfo = {\n    baseClasses: [],\n    responsiveClasses: new Map(),\n    sourceFile,\n    fullyConvertible: true\n  };\n  \n  for (const utility of rule.utilities) {\n    if (utility.variant) {\n      const existing = info.responsiveClasses.get(utility.variant) || [];\n      existing.push(utility.value);\n      info.responsiveClasses.set(utility.variant, existing);\n    } else {\n      info.baseClasses.push(utility.value);\n    }\n  }\n  \n  return info;\n}\n\nfunction mergeRuleIntoClassInfo(info: ClassInfo, rule: CSSRule): void {\n  for (const utility of rule.utilities) {\n    if (utility.variant) {\n      const existing = info.responsiveClasses.get(utility.variant) || [];\n      if (!existing.includes(utility.value)) {\n        existing.push(utility.value);\n        info.responsiveClasses.set(utility.variant, existing);\n      }\n    } else {\n      if (!info.baseClasses.includes(utility.value)) {\n        info.baseClasses.push(utility.value);\n      }\n    }\n  }\n}\n\nfunction assembleTailwindClasses(info: ClassInfo): string {\n  const classes: string[] = [...info.baseClasses];\n  \n  const sortedBreakpoints = ['sm', 'md', 'lg', 'xl', '2xl'];\n  \n  for (const bp of sortedBreakpoints) {\n    const bpClasses = info.responsiveClasses.get(bp);\n    if (bpClasses) {\n      for (const cls of bpClasses) {\n        classes.push(`${bp}:${cls}`);\n      }\n    }\n  }\n  \n  return classes.join(' ');\n}\n\nfunction replaceClassNameReferences(\n  code: string, \n  classMap: CSSClassMap\n): { \n  code: string; \n  hasChanges: boolean; \n  replacements: number \n} {\n  let hasChanges = false;\n  let replacements = 0;\n  let modifiedCode = code;\n\n  Object.entries(classMap).forEach(([oldClass, info]) => {\n    if (!info.fullyConvertible) {\n      return;\n    }\n\n    const tailwindClassString = assembleTailwindClasses(info);\n    \n    const pattern1 = new RegExp(`className=\"${oldClass}\"`, 'g');\n    if (pattern1.test(modifiedCode)) {\n      modifiedCode = modifiedCode.replace(pattern1, `className=\"${tailwindClassString}\"`);\n      hasChanges = true;\n      replacements++;\n      logger.verbose(`Replaced className=\"${oldClass}\" with \"${tailwindClassString}\"`);\n    }\n\n    const pattern2 = new RegExp(`className=\\\\{\"${oldClass}\"\\\\}`, 'g');\n    if (pattern2.test(modifiedCode)) {\n      modifiedCode = modifiedCode.replace(pattern2, `className=\"${tailwindClassString}\"`);\n      hasChanges = true;\n      replacements++;\n      logger.verbose(`Replaced className={\"${oldClass}\"} with \"${tailwindClassString}\"`);\n    }\n\n    const pattern3 = new RegExp(`className=\\\\{\\`\\\\s*${oldClass}\\\\s*\\`\\\\}`, 'g');\n    if (pattern3.test(modifiedCode)) {\n      modifiedCode = modifiedCode.replace(pattern3, `className=\"${tailwindClassString}\"`);\n      hasChanges = true;\n      replacements++;\n      logger.verbose(`Replaced className={\\`${oldClass}\\`} with \"${tailwindClassString}\"`);\n    }\n  });\n\n  return { code: modifiedCode, hasChanges, replacements };\n}\n"]}
|
|
269
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transformer.js","sourceRoot":"","sources":["../src/transformer.ts"],"names":[],"mappings":"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","sourcesContent":["import fs from 'fs';\nimport path from 'path';\nimport { ScannedFile } from './scanner';\nimport { TailwindMapper } from './tailwindMapper';\nimport { JSXParser } from './jsxParser';\nimport { CSSParser, CSSRule, UtilityWithVariant } from './cssParser';\nimport { FileWriter } from './fileWriter';\nimport { TailwindConfig } from './utils/config';\nimport { logger } from './utils/logger';\nimport { clearBreakpointCache } from './utils/breakpointResolver';\nimport { \n  assembleUtilities, \n  mergeUtilities,\n  normalizeVariantOrder \n} from './utils/variantAssembler';\n\nexport interface TransformOptions {\n  dryRun: boolean;\n  deleteCss: boolean;\n  skipExternal: boolean;\n  skipInline: boolean;\n  skipInternal: boolean;\n  tailwindConfig: TailwindConfig | null;\n  projectRoot: string;\n}\n\nexport interface TransformResults {\n  filesScanned: number;\n  filesModified: number;\n  stylesConverted: number;\n  classesReplaced: number;\n  warnings: number;\n}\n\ninterface ClassInfo {\n  utilities: Array<{ value: string; variants: string[] }>;\n  sourceFile: string;\n  fullyConvertible: boolean;\n}\n\ninterface CSSClassMap {\n  [className: string]: ClassInfo;\n}\n\nexport async function transformFiles(\n  files: ScannedFile[],\n  options: TransformOptions\n): Promise<TransformResults> {\n  const results: TransformResults = {\n    filesScanned: files.length,\n    filesModified: 0,\n    stylesConverted: 0,\n    classesReplaced: 0,\n    warnings: 0\n  };\n\n  const mapper = new TailwindMapper(options.tailwindConfig || {});\n  const jsxParser = new JSXParser(mapper);\n  const screens = options.tailwindConfig?.theme?.screens;\n  const cssParser = new CSSParser(mapper, screens);\n  const fileWriter = new FileWriter({ dryRun: options.dryRun });\n\n  clearBreakpointCache();\n\n  const cssClassMap: CSSClassMap = {};\n  const cssFileResults: Map<string, {\n    content: string;\n    newContent: string;\n    rules: CSSRule[];\n    canDelete: boolean;\n    hasChanges: boolean;\n    fullyConvertible: boolean;\n  }> = new Map();\n\n  logger.info('\\n🔍 Phase 1: Analyzing files...');\n\n  if (!options.skipExternal) {\n    for (const file of files.filter(f => f.type === 'css')) {\n      try {\n        const content = fs.readFileSync(file.path, 'utf-8');\n        const result = await cssParser.parse(content, file.path);\n\n        const totalRules = result.rules.length;\n        const fullyConvertedRules = result.rules.filter(r => r.fullyConverted).length;\n        const partiallyConvertedRules = result.rules.filter(r => r.partialConversion).length;\n        const fullyConvertible = totalRules > 0 && totalRules === fullyConvertedRules && partiallyConvertedRules === 0;\n\n        result.rules.forEach(rule => {\n          if (rule.fullyConverted) {\n            const existing = cssClassMap[rule.className];\n            if (existing) {\n              mergeRuleIntoClassInfo(existing, rule);\n            } else {\n              cssClassMap[rule.className] = buildClassInfoFromRule(rule, file.path);\n            }\n            results.stylesConverted += rule.declarations.length;\n          } else if (rule.partialConversion) {\n            results.stylesConverted += rule.convertedClasses.length;\n            logger.verbose(`  Rule .${rule.className}: partial conversion (${rule.convertedClasses.length}/${rule.declarations.length} declarations)`);\n          }\n        });\n\n        cssFileResults.set(file.path, {\n          content,\n          newContent: result.css,\n          rules: result.rules,\n          canDelete: result.canDelete,\n          hasChanges: result.hasChanges,\n          fullyConvertible\n        });\n\n        results.warnings += result.warnings.length;\n\n        logger.verbose(`Analyzed ${file.path}:`);\n        logger.verbose(`  - Total rules: ${totalRules}`);\n        logger.verbose(`  - Fully converted rules: ${fullyConvertedRules}`);\n        logger.verbose(`  - Partially converted rules: ${partiallyConvertedRules}`);\n        logger.verbose(`  - Fully convertible: ${fullyConvertible}`);\n\n        result.warnings.forEach(warning => {\n          logger.verbose(`⚠️  ${file.path}: ${warning}`);\n        });\n\n      } catch (error) {\n        logger.error(`Failed to analyze ${file.path}:`, error);\n        results.warnings++;\n      }\n    }\n  }\n\n  logger.info('\\n⚛️  Phase 2: Transforming React components...');\n\n  const jsxFileResults: Map<string, {\n    content: string;\n    newContent: string;\n    hasChanges: boolean;\n  }> = new Map();\n\n  for (const file of files.filter(f => f.type === 'jsx')) {\n    try {\n      let content = fs.readFileSync(file.path, 'utf-8');\n      const originalContent = content;\n      let hasChanges = false;\n      let fileWarnings: string[] = [];\n\n      if (!options.skipInline) {\n        try {\n          const jsxResult = jsxParser.parse(content, file.path);\n          \n          if (jsxResult.hasChanges) {\n            content = jsxResult.code;\n            hasChanges = true;\n            results.stylesConverted += jsxResult.transformations.reduce(\n              (sum, t) => sum + t.classes.length, 0\n            );\n            fileWarnings.push(...jsxResult.warnings);\n          }\n        } catch (error) {\n          logger.warn(`Failed to parse JSX inline styles in ${file.path}: ${error}`);\n          fileWarnings.push(`JSX parse error: ${error}`);\n        }\n      }\n\n      if (!options.skipInternal) {\n        try {\n          const internalResult = await cssParser.parseInternalCSS(content, file.path);\n          \n          if (internalResult.hasChanges) {\n            content = internalResult.html;\n            hasChanges = true;\n            \n            internalResult.rules.forEach(rule => {\n              if (rule.convertedClasses.length > 0) {\n                const existing = cssClassMap[rule.className];\n                if (existing) {\n                  mergeRuleIntoClassInfo(existing, rule);\n                } else {\n                  cssClassMap[rule.className] = buildClassInfoFromRule(rule, file.path);\n                }\n                results.stylesConverted += rule.declarations.length;\n              }\n            });\n            \n            fileWarnings.push(...internalResult.warnings);\n          }\n        } catch (error) {\n          logger.warn(`Failed to parse internal CSS in ${file.path}: ${error}`);\n          fileWarnings.push(`Internal CSS parse error: ${error}`);\n        }\n      }\n\n      jsxFileResults.set(file.path, {\n        content: originalContent,\n        newContent: content,\n        hasChanges\n      });\n\n      results.warnings += fileWarnings.length;\n\n      fileWarnings.forEach(warning => {\n        logger.verbose(`⚠️  ${file.path}: ${warning}`);\n      });\n\n    } catch (error) {\n      logger.error(`Failed to process ${file.path}:`, error);\n      results.warnings++;\n    }\n  }\n\n  if (Object.keys(cssClassMap).length > 0) {\n    logger.info('\\n🔄 Phase 3: Replacing className references...');\n\n    for (const [filePath, fileResult] of jsxFileResults) {\n      let content = fileResult.newContent;\n      let hasChanges = fileResult.hasChanges;\n\n      const replacementResult = replaceClassNameReferences(content, cssClassMap);\n      if (replacementResult.hasChanges) {\n        content = replacementResult.code;\n        hasChanges = true;\n        results.classesReplaced += replacementResult.replacements;\n        \n        logger.verbose(`Replaced ${replacementResult.replacements} class references in ${path.basename(filePath)}`);\n      }\n\n      jsxFileResults.set(filePath, {\n        ...fileResult,\n        newContent: content,\n        hasChanges\n      });\n    }\n  }\n\n  logger.info('\\n💾 Phase 4: Writing changes...');\n\n  for (const [filePath, fileResult] of jsxFileResults) {\n    if (fileResult.hasChanges) {\n      const success = await fileWriter.writeFile(filePath, fileResult.newContent, fileResult.content);\n      if (success) {\n        results.filesModified++;\n      }\n    }\n  }\n\n  if (!options.skipExternal) {\n    for (const [filePath, fileResult] of cssFileResults) {\n      if (!fileResult.hasChanges) continue;\n\n      if (!fileResult.fullyConvertible) {\n        logger.warn(`⏭️  Skipping ${path.basename(filePath)} - not fully convertible (would break styles)`);\n        logger.warn(`   Convertible: ${fileResult.rules.filter(r => r.convertedClasses.length > 0).length}/${fileResult.rules.length} rules`);\n        continue;\n      }\n\n      if (fileResult.canDelete && options.deleteCss) {\n        const success = await fileWriter.deleteFile(filePath);\n        if (success) {\n          results.filesModified++;\n          logger.info(`🗑️  Deleted ${path.basename(filePath)} (all rules converted)`);\n        }\n      } else if (fileResult.canDelete && !options.deleteCss) {\n        logger.info(`ℹ️  ${path.basename(filePath)} is now empty (use --delete-css to remove)`);\n      } else {\n        const success = await fileWriter.writeFile(filePath, fileResult.newContent, fileResult.content);\n        if (success) {\n          results.filesModified++;\n        }\n      }\n    }\n  }\n\n  return results;\n}\n\nfunction buildClassInfoFromRule(rule: CSSRule, sourceFile: string): ClassInfo {\n  return {\n    utilities: rule.utilities.map(u => ({\n      value: u.value,\n      variants: normalizeVariantOrder([...u.variants])\n    })),\n    sourceFile,\n    fullyConvertible: true\n  };\n}\n\nfunction mergeRuleIntoClassInfo(info: ClassInfo, rule: CSSRule): void {\n  for (const utility of rule.utilities) {\n    const existing = info.utilities.find(u => u.value === utility.value);\n    if (existing) {\n      for (const variant of utility.variants) {\n        if (!existing.variants.includes(variant)) {\n          existing.variants.push(variant);\n        }\n      }\n      existing.variants = normalizeVariantOrder(existing.variants);\n    } else {\n      info.utilities.push({\n        value: utility.value,\n        variants: normalizeVariantOrder([...utility.variants])\n      });\n    }\n  }\n}\n\nfunction assembleTailwindClasses(info: ClassInfo): string {\n  const merged = mergeUtilities(info.utilities);\n  return assembleUtilities(merged).join(' ');\n}\n\nfunction replaceClassNameReferences(\n  code: string, \n  classMap: CSSClassMap\n): { \n  code: string; \n  hasChanges: boolean; \n  replacements: number \n} {\n  let hasChanges = false;\n  let replacements = 0;\n  let modifiedCode = code;\n\n  Object.entries(classMap).forEach(([oldClass, info]) => {\n    if (!info.fullyConvertible) {\n      return;\n    }\n\n    const tailwindClassString = assembleTailwindClasses(info);\n    \n    const pattern1 = new RegExp(`className=\"${oldClass}\"`, 'g');\n    if (pattern1.test(modifiedCode)) {\n      modifiedCode = modifiedCode.replace(pattern1, `className=\"${tailwindClassString}\"`);\n      hasChanges = true;\n      replacements++;\n      logger.verbose(`Replaced className=\"${oldClass}\" with \"${tailwindClassString}\"`);\n    }\n\n    const pattern2 = new RegExp(`className=\\\\{\"${oldClass}\"\\\\}`, 'g');\n    if (pattern2.test(modifiedCode)) {\n      modifiedCode = modifiedCode.replace(pattern2, `className=\"${tailwindClassString}\"`);\n      hasChanges = true;\n      replacements++;\n      logger.verbose(`Replaced className={\"${oldClass}\"} with \"${tailwindClassString}\"`);\n    }\n\n    const pattern3 = new RegExp(`className=\\\\{\\`\\\\s*${oldClass}\\\\s*\\`\\\\}`, 'g');\n    if (pattern3.test(modifiedCode)) {\n      modifiedCode = modifiedCode.replace(pattern3, `className=\"${tailwindClassString}\"`);\n      hasChanges = true;\n      replacements++;\n      logger.verbose(`Replaced className={\\`${oldClass}\\`} with \"${tailwindClassString}\"`);\n    }\n  });\n\n  return { code: modifiedCode, hasChanges, replacements };\n}"]}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
export interface ParsedSelector {
|
|
2
|
+
baseClass: string;
|
|
3
|
+
pseudos: string[];
|
|
4
|
+
isComplex: boolean;
|
|
5
|
+
reason?: string;
|
|
6
|
+
}
|
|
7
|
+
export declare const PSEUDO_TO_VARIANT: Record<string, string>;
|
|
8
|
+
export declare const SUPPORTED_PSEUDOS: Set<string>;
|
|
9
|
+
export declare function parseSelector(selector: string): ParsedSelector;
|
|
10
|
+
export declare function mapPseudoToVariant(pseudo: string): string | null;
|
|
11
|
+
export declare function processPseudoSelector(selector: string): {
|
|
12
|
+
baseClass: string | null;
|
|
13
|
+
variants: string[];
|
|
14
|
+
skipped: boolean;
|
|
15
|
+
reason?: string;
|
|
16
|
+
};
|
|
17
|
+
export declare function parseMultipleSelectors(selector: string): ParsedSelector[];
|
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.SUPPORTED_PSEUDOS = exports.PSEUDO_TO_VARIANT = void 0;
|
|
4
|
+
exports.parseSelector = parseSelector;
|
|
5
|
+
exports.mapPseudoToVariant = mapPseudoToVariant;
|
|
6
|
+
exports.processPseudoSelector = processPseudoSelector;
|
|
7
|
+
exports.parseMultipleSelectors = parseMultipleSelectors;
|
|
8
|
+
const logger_1 = require("./logger");
|
|
9
|
+
exports.PSEUDO_TO_VARIANT = {
|
|
10
|
+
'hover': 'hover',
|
|
11
|
+
'focus': 'focus',
|
|
12
|
+
'active': 'active',
|
|
13
|
+
'disabled': 'disabled',
|
|
14
|
+
'visited': 'visited',
|
|
15
|
+
'first-child': 'first',
|
|
16
|
+
'last-child': 'last',
|
|
17
|
+
'before': 'before',
|
|
18
|
+
'after': 'after'
|
|
19
|
+
};
|
|
20
|
+
exports.SUPPORTED_PSEUDOS = new Set(Object.keys(exports.PSEUDO_TO_VARIANT));
|
|
21
|
+
const UNSUPPORTED_PATTERNS = [
|
|
22
|
+
':nth-child',
|
|
23
|
+
':nth-of-type',
|
|
24
|
+
':not(',
|
|
25
|
+
':has(',
|
|
26
|
+
':is(',
|
|
27
|
+
':where(',
|
|
28
|
+
':first-of-type',
|
|
29
|
+
':last-of-type',
|
|
30
|
+
':only-child',
|
|
31
|
+
':only-of-type',
|
|
32
|
+
':empty',
|
|
33
|
+
':checked',
|
|
34
|
+
':indeterminate',
|
|
35
|
+
':default',
|
|
36
|
+
':required',
|
|
37
|
+
':valid',
|
|
38
|
+
':invalid',
|
|
39
|
+
':in-range',
|
|
40
|
+
':out-of-range',
|
|
41
|
+
':placeholder-shown',
|
|
42
|
+
':autofill',
|
|
43
|
+
':read-only',
|
|
44
|
+
':target',
|
|
45
|
+
':root',
|
|
46
|
+
':scope',
|
|
47
|
+
':lang(',
|
|
48
|
+
':dir('
|
|
49
|
+
];
|
|
50
|
+
function parseSelector(selector) {
|
|
51
|
+
const trimmed = selector.trim();
|
|
52
|
+
if (!trimmed.startsWith('.')) {
|
|
53
|
+
return {
|
|
54
|
+
baseClass: '',
|
|
55
|
+
pseudos: [],
|
|
56
|
+
isComplex: true,
|
|
57
|
+
reason: `Not a class selector: ${selector}`
|
|
58
|
+
};
|
|
59
|
+
}
|
|
60
|
+
for (const pattern of UNSUPPORTED_PATTERNS) {
|
|
61
|
+
if (trimmed.toLowerCase().includes(pattern.toLowerCase())) {
|
|
62
|
+
return {
|
|
63
|
+
baseClass: '',
|
|
64
|
+
pseudos: [],
|
|
65
|
+
isComplex: true,
|
|
66
|
+
reason: `Unsupported pseudo selector pattern: ${pattern}`
|
|
67
|
+
};
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
const pseudoMatches = [];
|
|
71
|
+
let remaining = trimmed.slice(1);
|
|
72
|
+
const pseudoRegex = /:([a-zA-Z-]+)/g;
|
|
73
|
+
let match;
|
|
74
|
+
let hasComplexPseudo = false;
|
|
75
|
+
while ((match = pseudoRegex.exec(trimmed)) !== null) {
|
|
76
|
+
const pseudo = match[1].toLowerCase();
|
|
77
|
+
pseudoMatches.push(pseudo);
|
|
78
|
+
if (!exports.SUPPORTED_PSEUDOS.has(pseudo)) {
|
|
79
|
+
hasComplexPseudo = true;
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
if (pseudoMatches.length > 1) {
|
|
83
|
+
return {
|
|
84
|
+
baseClass: '',
|
|
85
|
+
pseudos: [],
|
|
86
|
+
isComplex: true,
|
|
87
|
+
reason: `Skipped complex pseudo chain (${selector})`
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
if (hasComplexPseudo && pseudoMatches.some(p => !exports.SUPPORTED_PSEUDOS.has(p))) {
|
|
91
|
+
const unsupported = pseudoMatches.find(p => !exports.SUPPORTED_PSEUDOS.has(p));
|
|
92
|
+
return {
|
|
93
|
+
baseClass: '',
|
|
94
|
+
pseudos: [],
|
|
95
|
+
isComplex: true,
|
|
96
|
+
reason: `Unsupported pseudo selector :${unsupported}`
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
const baseClassMatch = remaining.match(/^([a-zA-Z_-][a-zA-Z0-9_-]*)/);
|
|
100
|
+
if (!baseClassMatch) {
|
|
101
|
+
return {
|
|
102
|
+
baseClass: '',
|
|
103
|
+
pseudos: [],
|
|
104
|
+
isComplex: true,
|
|
105
|
+
reason: `Invalid class name in selector: ${selector}`
|
|
106
|
+
};
|
|
107
|
+
}
|
|
108
|
+
const baseClass = baseClassMatch[1];
|
|
109
|
+
const expectedSelector = '.' + baseClass + pseudoMatches.map(p => `:${p}`).join('');
|
|
110
|
+
const hasMultipleSelectors = trimmed.includes(',');
|
|
111
|
+
const hasCombinators = /[>\s+~]/.test(trimmed.slice(baseClass.length + 1).replace(/:[a-zA-Z-]+/g, ''));
|
|
112
|
+
if (hasCombinators && !hasMultipleSelectors) {
|
|
113
|
+
return {
|
|
114
|
+
baseClass: '',
|
|
115
|
+
pseudos: [],
|
|
116
|
+
isComplex: true,
|
|
117
|
+
reason: `Complex selector with combinators: ${selector}`
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
const variants = pseudoMatches
|
|
121
|
+
.filter(p => exports.SUPPORTED_PSEUDOS.has(p))
|
|
122
|
+
.map(p => exports.PSEUDO_TO_VARIANT[p]);
|
|
123
|
+
return {
|
|
124
|
+
baseClass,
|
|
125
|
+
pseudos: variants,
|
|
126
|
+
isComplex: false
|
|
127
|
+
};
|
|
128
|
+
}
|
|
129
|
+
function mapPseudoToVariant(pseudo) {
|
|
130
|
+
const normalized = pseudo.toLowerCase().replace(/^:/, '');
|
|
131
|
+
return exports.PSEUDO_TO_VARIANT[normalized] || null;
|
|
132
|
+
}
|
|
133
|
+
function processPseudoSelector(selector) {
|
|
134
|
+
const parsed = parseSelector(selector);
|
|
135
|
+
if (parsed.isComplex) {
|
|
136
|
+
logger_1.logger.verbose(parsed.reason || `Skipped complex selector: ${selector}`);
|
|
137
|
+
return {
|
|
138
|
+
baseClass: null,
|
|
139
|
+
variants: [],
|
|
140
|
+
skipped: true,
|
|
141
|
+
reason: parsed.reason
|
|
142
|
+
};
|
|
143
|
+
}
|
|
144
|
+
if (parsed.pseudos.length > 0) {
|
|
145
|
+
logger_1.logger.verbose(`Converted pseudo selector :${parsed.pseudos.join(':')} → ${parsed.pseudos.join(':')}:`);
|
|
146
|
+
logger_1.logger.verbose(`Applied to class .${parsed.baseClass}`);
|
|
147
|
+
}
|
|
148
|
+
return {
|
|
149
|
+
baseClass: parsed.baseClass,
|
|
150
|
+
variants: parsed.pseudos,
|
|
151
|
+
skipped: false
|
|
152
|
+
};
|
|
153
|
+
}
|
|
154
|
+
function parseMultipleSelectors(selector) {
|
|
155
|
+
const parts = selector.split(',').map(s => s.trim()).filter(Boolean);
|
|
156
|
+
const results = [];
|
|
157
|
+
for (const part of parts) {
|
|
158
|
+
const parsed = parseSelector(part);
|
|
159
|
+
results.push(parsed);
|
|
160
|
+
}
|
|
161
|
+
return results;
|
|
162
|
+
}
|
|
163
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pseudoSelectorResolver.js","sourceRoot":"","sources":["../../src/utils/pseudoSelectorResolver.ts"],"names":[],"mappings":"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","sourcesContent":["import { logger } from './logger';\n\nexport interface ParsedSelector {\n  baseClass: string;\n  pseudos: string[];\n  isComplex: boolean;\n  reason?: string;\n}\n\nexport const PSEUDO_TO_VARIANT: Record<string, string> = {\n  'hover': 'hover',\n  'focus': 'focus',\n  'active': 'active',\n  'disabled': 'disabled',\n  'visited': 'visited',\n  'first-child': 'first',\n  'last-child': 'last',\n  'before': 'before',\n  'after': 'after'\n};\n\nexport const SUPPORTED_PSEUDOS = new Set(Object.keys(PSEUDO_TO_VARIANT));\n\nconst UNSUPPORTED_PATTERNS = [\n  ':nth-child',\n  ':nth-of-type',\n  ':not(',\n  ':has(',\n  ':is(',\n  ':where(',\n  ':first-of-type',\n  ':last-of-type',\n  ':only-child',\n  ':only-of-type',\n  ':empty',\n  ':checked',\n  ':indeterminate',\n  ':default',\n  ':required',\n  ':valid',\n  ':invalid',\n  ':in-range',\n  ':out-of-range',\n  ':placeholder-shown',\n  ':autofill',\n  ':read-only',\n  ':target',\n  ':root',\n  ':scope',\n  ':lang(',\n  ':dir('\n];\n\nexport function parseSelector(selector: string): ParsedSelector {\n  const trimmed = selector.trim();\n  \n  if (!trimmed.startsWith('.')) {\n    return {\n      baseClass: '',\n      pseudos: [],\n      isComplex: true,\n      reason: `Not a class selector: ${selector}`\n    };\n  }\n  \n  for (const pattern of UNSUPPORTED_PATTERNS) {\n    if (trimmed.toLowerCase().includes(pattern.toLowerCase())) {\n      return {\n        baseClass: '',\n        pseudos: [],\n        isComplex: true,\n        reason: `Unsupported pseudo selector pattern: ${pattern}`\n      };\n    }\n  }\n  \n  const pseudoMatches: string[] = [];\n  let remaining = trimmed.slice(1);\n  \n  const pseudoRegex = /:([a-zA-Z-]+)/g;\n  let match;\n  let hasComplexPseudo = false;\n  \n  while ((match = pseudoRegex.exec(trimmed)) !== null) {\n    const pseudo = match[1].toLowerCase();\n    pseudoMatches.push(pseudo);\n    \n    if (!SUPPORTED_PSEUDOS.has(pseudo)) {\n      hasComplexPseudo = true;\n    }\n  }\n  \n  if (pseudoMatches.length > 1) {\n    return {\n      baseClass: '',\n      pseudos: [],\n      isComplex: true,\n      reason: `Skipped complex pseudo chain (${selector})`\n    };\n  }\n  \n  if (hasComplexPseudo && pseudoMatches.some(p => !SUPPORTED_PSEUDOS.has(p))) {\n    const unsupported = pseudoMatches.find(p => !SUPPORTED_PSEUDOS.has(p));\n    return {\n      baseClass: '',\n      pseudos: [],\n      isComplex: true,\n      reason: `Unsupported pseudo selector :${unsupported}`\n    };\n  }\n  \n  const baseClassMatch = remaining.match(/^([a-zA-Z_-][a-zA-Z0-9_-]*)/);\n  if (!baseClassMatch) {\n    return {\n      baseClass: '',\n      pseudos: [],\n      isComplex: true,\n      reason: `Invalid class name in selector: ${selector}`\n    };\n  }\n  \n  const baseClass = baseClassMatch[1];\n  \n  const expectedSelector = '.' + baseClass + pseudoMatches.map(p => `:${p}`).join('');\n  const hasMultipleSelectors = trimmed.includes(',');\n  const hasCombinators = /[>\\s+~]/.test(trimmed.slice(baseClass.length + 1).replace(/:[a-zA-Z-]+/g, ''));\n  \n  if (hasCombinators && !hasMultipleSelectors) {\n    return {\n      baseClass: '',\n      pseudos: [],\n      isComplex: true,\n      reason: `Complex selector with combinators: ${selector}`\n    };\n  }\n  \n  const variants = pseudoMatches\n    .filter(p => SUPPORTED_PSEUDOS.has(p))\n    .map(p => PSEUDO_TO_VARIANT[p]);\n  \n  return {\n    baseClass,\n    pseudos: variants,\n    isComplex: false\n  };\n}\n\nexport function mapPseudoToVariant(pseudo: string): string | null {\n  const normalized = pseudo.toLowerCase().replace(/^:/, '');\n  return PSEUDO_TO_VARIANT[normalized] || null;\n}\n\nexport function processPseudoSelector(selector: string): {\n  baseClass: string | null;\n  variants: string[];\n  skipped: boolean;\n  reason?: string;\n} {\n  const parsed = parseSelector(selector);\n  \n  if (parsed.isComplex) {\n    logger.verbose(parsed.reason || `Skipped complex selector: ${selector}`);\n    return {\n      baseClass: null,\n      variants: [],\n      skipped: true,\n      reason: parsed.reason\n    };\n  }\n  \n  if (parsed.pseudos.length > 0) {\n    logger.verbose(`Converted pseudo selector :${parsed.pseudos.join(':')} → ${parsed.pseudos.join(':')}:`);\n    logger.verbose(`Applied to class .${parsed.baseClass}`);\n  }\n  \n  return {\n    baseClass: parsed.baseClass,\n    variants: parsed.pseudos,\n    skipped: false\n  };\n}\n\nexport function parseMultipleSelectors(selector: string): ParsedSelector[] {\n  const parts = selector.split(',').map(s => s.trim()).filter(Boolean);\n  const results: ParsedSelector[] = [];\n  \n  for (const part of parts) {\n    const parsed = parseSelector(part);\n    results.push(parsed);\n  }\n  \n  return results;\n}"]}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
export declare const VARIANT_ORDER: string[];
|
|
2
|
+
export declare function isResponsiveVariant(variant: string): boolean;
|
|
3
|
+
export declare function isPseudoVariant(variant: string): boolean;
|
|
4
|
+
export declare function sortVariants(variants: string[]): string[];
|
|
5
|
+
export declare function deduplicateVariants(variants: string[]): string[];
|
|
6
|
+
export declare function validateVariantOrder(variants: string[]): boolean;
|
|
7
|
+
export declare function normalizeVariantOrder(variants: string[]): string[];
|
|
8
|
+
export declare function assembleUtility(utility: string, variants?: string[]): string;
|
|
9
|
+
export declare function assembleUtilities(utilities: Array<{
|
|
10
|
+
value: string;
|
|
11
|
+
variants?: string[];
|
|
12
|
+
}>): string[];
|
|
13
|
+
export interface MergedUtility {
|
|
14
|
+
value: string;
|
|
15
|
+
variants: string[];
|
|
16
|
+
}
|
|
17
|
+
export declare function mergeUtilities(utilities: Array<{
|
|
18
|
+
value: string;
|
|
19
|
+
variants?: string[];
|
|
20
|
+
}>): MergedUtility[];
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.VARIANT_ORDER = void 0;
|
|
4
|
+
exports.isResponsiveVariant = isResponsiveVariant;
|
|
5
|
+
exports.isPseudoVariant = isPseudoVariant;
|
|
6
|
+
exports.sortVariants = sortVariants;
|
|
7
|
+
exports.deduplicateVariants = deduplicateVariants;
|
|
8
|
+
exports.validateVariantOrder = validateVariantOrder;
|
|
9
|
+
exports.normalizeVariantOrder = normalizeVariantOrder;
|
|
10
|
+
exports.assembleUtility = assembleUtility;
|
|
11
|
+
exports.assembleUtilities = assembleUtilities;
|
|
12
|
+
exports.mergeUtilities = mergeUtilities;
|
|
13
|
+
exports.VARIANT_ORDER = ['sm', 'md', 'lg', 'xl', '2xl', 'hover', 'focus', 'active', 'disabled', 'visited', 'first', 'last', 'before', 'after', 'dark', 'light'];
|
|
14
|
+
const RESPONSIVE_VARIANTS = new Set(['sm', 'md', 'lg', 'xl', '2xl']);
|
|
15
|
+
const PSEUDO_VARIANTS = new Set(['hover', 'focus', 'active', 'disabled', 'visited', 'first', 'last', 'before', 'after']);
|
|
16
|
+
function isResponsiveVariant(variant) {
|
|
17
|
+
return RESPONSIVE_VARIANTS.has(variant);
|
|
18
|
+
}
|
|
19
|
+
function isPseudoVariant(variant) {
|
|
20
|
+
return PSEUDO_VARIANTS.has(variant);
|
|
21
|
+
}
|
|
22
|
+
function sortVariants(variants) {
|
|
23
|
+
return [...variants].sort((a, b) => {
|
|
24
|
+
const aIndex = exports.VARIANT_ORDER.indexOf(a);
|
|
25
|
+
const bIndex = exports.VARIANT_ORDER.indexOf(b);
|
|
26
|
+
if (aIndex === -1 && bIndex === -1)
|
|
27
|
+
return a.localeCompare(b);
|
|
28
|
+
if (aIndex === -1)
|
|
29
|
+
return 1;
|
|
30
|
+
if (bIndex === -1)
|
|
31
|
+
return -1;
|
|
32
|
+
return aIndex - bIndex;
|
|
33
|
+
});
|
|
34
|
+
}
|
|
35
|
+
function deduplicateVariants(variants) {
|
|
36
|
+
const seen = new Set();
|
|
37
|
+
const result = [];
|
|
38
|
+
for (const variant of variants) {
|
|
39
|
+
if (!seen.has(variant)) {
|
|
40
|
+
seen.add(variant);
|
|
41
|
+
result.push(variant);
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
return result;
|
|
45
|
+
}
|
|
46
|
+
function validateVariantOrder(variants) {
|
|
47
|
+
let hasSeenPseudo = false;
|
|
48
|
+
for (const variant of variants) {
|
|
49
|
+
if (isPseudoVariant(variant)) {
|
|
50
|
+
hasSeenPseudo = true;
|
|
51
|
+
}
|
|
52
|
+
else if (isResponsiveVariant(variant) && hasSeenPseudo) {
|
|
53
|
+
return false;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
return true;
|
|
57
|
+
}
|
|
58
|
+
function normalizeVariantOrder(variants) {
|
|
59
|
+
const deduped = deduplicateVariants(variants);
|
|
60
|
+
const responsive = [];
|
|
61
|
+
const pseudo = [];
|
|
62
|
+
const other = [];
|
|
63
|
+
for (const variant of deduped) {
|
|
64
|
+
if (isResponsiveVariant(variant)) {
|
|
65
|
+
responsive.push(variant);
|
|
66
|
+
}
|
|
67
|
+
else if (isPseudoVariant(variant)) {
|
|
68
|
+
pseudo.push(variant);
|
|
69
|
+
}
|
|
70
|
+
else {
|
|
71
|
+
other.push(variant);
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
const sortedResponsive = sortVariants(responsive);
|
|
75
|
+
const sortedPseudo = sortVariants(pseudo);
|
|
76
|
+
const sortedOther = sortVariants(other);
|
|
77
|
+
return [...sortedResponsive, ...sortedPseudo, ...sortedOther];
|
|
78
|
+
}
|
|
79
|
+
function assembleUtility(utility, variants) {
|
|
80
|
+
if (!variants || variants.length === 0) {
|
|
81
|
+
return utility;
|
|
82
|
+
}
|
|
83
|
+
const normalized = normalizeVariantOrder(variants);
|
|
84
|
+
if (normalized.length === 0) {
|
|
85
|
+
return utility;
|
|
86
|
+
}
|
|
87
|
+
const prefix = normalized.join(':');
|
|
88
|
+
return `${prefix}:${utility}`;
|
|
89
|
+
}
|
|
90
|
+
function assembleUtilities(utilities) {
|
|
91
|
+
return utilities.map(u => assembleUtility(u.value, u.variants));
|
|
92
|
+
}
|
|
93
|
+
function mergeUtilities(utilities) {
|
|
94
|
+
const merged = new Map();
|
|
95
|
+
for (const utility of utilities) {
|
|
96
|
+
const key = utility.value;
|
|
97
|
+
const existing = merged.get(key) || new Set();
|
|
98
|
+
if (utility.variants) {
|
|
99
|
+
for (const v of utility.variants) {
|
|
100
|
+
existing.add(v);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
merged.set(key, existing);
|
|
104
|
+
}
|
|
105
|
+
const result = [];
|
|
106
|
+
for (const [value, variantSet] of merged.entries()) {
|
|
107
|
+
result.push({
|
|
108
|
+
value,
|
|
109
|
+
variants: Array.from(variantSet)
|
|
110
|
+
});
|
|
111
|
+
}
|
|
112
|
+
return result;
|
|
113
|
+
}
|
|
114
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"variantAssembler.js","sourceRoot":"","sources":["../../src/utils/variantAssembler.ts"],"names":[],"mappings":"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","sourcesContent":["import { logger } from './logger';\n\nexport const VARIANT_ORDER = ['sm', 'md', 'lg', 'xl', '2xl', 'hover', 'focus', 'active', 'disabled', 'visited', 'first', 'last', 'before', 'after', 'dark', 'light'];\n\nconst RESPONSIVE_VARIANTS = new Set(['sm', 'md', 'lg', 'xl', '2xl']);\nconst PSEUDO_VARIANTS = new Set(['hover', 'focus', 'active', 'disabled', 'visited', 'first', 'last', 'before', 'after']);\n\nexport function isResponsiveVariant(variant: string): boolean {\n  return RESPONSIVE_VARIANTS.has(variant);\n}\n\nexport function isPseudoVariant(variant: string): boolean {\n  return PSEUDO_VARIANTS.has(variant);\n}\n\nexport function sortVariants(variants: string[]): string[] {\n  return [...variants].sort((a, b) => {\n    const aIndex = VARIANT_ORDER.indexOf(a);\n    const bIndex = VARIANT_ORDER.indexOf(b);\n    \n    if (aIndex === -1 && bIndex === -1) return a.localeCompare(b);\n    if (aIndex === -1) return 1;\n    if (bIndex === -1) return -1;\n    \n    return aIndex - bIndex;\n  });\n}\n\nexport function deduplicateVariants(variants: string[]): string[] {\n  const seen = new Set<string>();\n  const result: string[] = [];\n  \n  for (const variant of variants) {\n    if (!seen.has(variant)) {\n      seen.add(variant);\n      result.push(variant);\n    }\n  }\n  \n  return result;\n}\n\nexport function validateVariantOrder(variants: string[]): boolean {\n  let hasSeenPseudo = false;\n  \n  for (const variant of variants) {\n    if (isPseudoVariant(variant)) {\n      hasSeenPseudo = true;\n    } else if (isResponsiveVariant(variant) && hasSeenPseudo) {\n      return false;\n    }\n  }\n  \n  return true;\n}\n\nexport function normalizeVariantOrder(variants: string[]): string[] {\n  const deduped = deduplicateVariants(variants);\n  const responsive: string[] = [];\n  const pseudo: string[] = [];\n  const other: string[] = [];\n  \n  for (const variant of deduped) {\n    if (isResponsiveVariant(variant)) {\n      responsive.push(variant);\n    } else if (isPseudoVariant(variant)) {\n      pseudo.push(variant);\n    } else {\n      other.push(variant);\n    }\n  }\n  \n  const sortedResponsive = sortVariants(responsive);\n  const sortedPseudo = sortVariants(pseudo);\n  const sortedOther = sortVariants(other);\n  \n  return [...sortedResponsive, ...sortedPseudo, ...sortedOther];\n}\n\nexport function assembleUtility(utility: string, variants?: string[]): string {\n  if (!variants || variants.length === 0) {\n    return utility;\n  }\n  \n  const normalized = normalizeVariantOrder(variants);\n  \n  if (normalized.length === 0) {\n    return utility;\n  }\n  \n  const prefix = normalized.join(':');\n  return `${prefix}:${utility}`;\n}\n\nexport function assembleUtilities(utilities: Array<{ value: string; variants?: string[] }>): string[] {\n  return utilities.map(u => assembleUtility(u.value, u.variants));\n}\n\nexport interface MergedUtility {\n  value: string;\n  variants: string[];\n}\n\nexport function mergeUtilities(utilities: Array<{ value: string; variants?: string[] }>): MergedUtility[] {\n  const merged = new Map<string, Set<string>>();\n  \n  for (const utility of utilities) {\n    const key = utility.value;\n    const existing = merged.get(key) || new Set<string>();\n    \n    if (utility.variants) {\n      for (const v of utility.variants) {\n        existing.add(v);\n      }\n    }\n    \n    merged.set(key, existing);\n  }\n  \n  const result: MergedUtility[] = [];\n  \n  for (const [value, variantSet] of merged.entries()) {\n    result.push({\n      value,\n      variants: Array.from(variantSet)\n    });\n  }\n  \n  return result;\n}"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "css-to-tailwind-react",
|
|
3
|
-
"version": "0.1.
|
|
3
|
+
"version": "0.1.2",
|
|
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",
|