css-to-tailwind-react 0.1.0 → 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 +9 -1
- package/dist/cssParser.js +160 -90
- package/dist/htmlParser.d.ts +15 -0
- package/dist/htmlParser.js +103 -0
- package/dist/index.d.ts +4 -1
- package/dist/index.js +26 -2
- package/dist/transformer.js +54 -44
- package/dist/utils/breakpointResolver.d.ts +21 -0
- package/dist/utils/breakpointResolver.js +154 -0
- package/dist/utils/config.d.ts +1 -0
- package/dist/utils/config.js +8 -1
- 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 +7 -2
package/dist/cssParser.d.ts
CHANGED
|
@@ -1,9 +1,14 @@
|
|
|
1
1
|
import { TailwindMapper, CSSProperty } from './tailwindMapper';
|
|
2
|
+
export interface UtilityWithVariant {
|
|
3
|
+
value: string;
|
|
4
|
+
variants: string[];
|
|
5
|
+
}
|
|
2
6
|
export interface CSSRule {
|
|
3
7
|
selector: string;
|
|
4
8
|
className: string;
|
|
5
9
|
declarations: CSSProperty[];
|
|
6
10
|
convertedClasses: string[];
|
|
11
|
+
utilities: UtilityWithVariant[];
|
|
7
12
|
skipped: boolean;
|
|
8
13
|
fullyConverted: boolean;
|
|
9
14
|
partialConversion: boolean;
|
|
@@ -21,7 +26,10 @@ export interface CSSUsageMap {
|
|
|
21
26
|
}
|
|
22
27
|
export declare class CSSParser {
|
|
23
28
|
private mapper;
|
|
24
|
-
|
|
29
|
+
private breakpoints;
|
|
30
|
+
constructor(mapper: TailwindMapper, screens?: Record<string, string | [string, string]>);
|
|
31
|
+
private convertDeclarations;
|
|
32
|
+
private processRuleWithVariants;
|
|
25
33
|
parse(css: string, filePath: string): Promise<CSSParseResult>;
|
|
26
34
|
parseInternalStyle(html: string): {
|
|
27
35
|
styles: Array<{
|
package/dist/cssParser.js
CHANGED
|
@@ -7,9 +7,93 @@ exports.CSSParser = void 0;
|
|
|
7
7
|
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
|
+
const breakpointResolver_1 = require("./utils/breakpointResolver");
|
|
11
|
+
const pseudoSelectorResolver_1 = require("./utils/pseudoSelectorResolver");
|
|
12
|
+
const variantAssembler_1 = require("./utils/variantAssembler");
|
|
10
13
|
class CSSParser {
|
|
11
|
-
constructor(mapper) {
|
|
14
|
+
constructor(mapper, screens) {
|
|
12
15
|
this.mapper = mapper;
|
|
16
|
+
this.breakpoints = screens
|
|
17
|
+
? (0, breakpointResolver_1.resolveBreakpointsFromConfig)(screens)
|
|
18
|
+
: (0, breakpointResolver_1.getBreakpoints)();
|
|
19
|
+
}
|
|
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) {
|
|
47
|
+
return null;
|
|
48
|
+
}
|
|
49
|
+
const declarations = [];
|
|
50
|
+
rule.walkDecls((decl) => {
|
|
51
|
+
if (decl.prop.startsWith('--')) {
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
if (decl.value.includes('calc(')) {
|
|
55
|
+
return;
|
|
56
|
+
}
|
|
57
|
+
declarations.push({
|
|
58
|
+
property: decl.prop,
|
|
59
|
+
value: decl.value
|
|
60
|
+
});
|
|
61
|
+
});
|
|
62
|
+
if (declarations.length === 0) {
|
|
63
|
+
return null;
|
|
64
|
+
}
|
|
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])
|
|
69
|
+
}));
|
|
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 };
|
|
13
97
|
}
|
|
14
98
|
async parse(css, filePath) {
|
|
15
99
|
const rules = [];
|
|
@@ -20,105 +104,97 @@ class CSSParser {
|
|
|
20
104
|
parser: postcss_safe_parser_1.default,
|
|
21
105
|
from: filePath
|
|
22
106
|
}).then(result => result.root);
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
// Skip rules inside @media, @supports, etc.
|
|
26
|
-
if (rule.parent && rule.parent.type === 'atrule') {
|
|
27
|
-
warnings.push(`Skipped rule with at-rule parent: ${rule.selector}`);
|
|
28
|
-
logger_1.logger.verbose(`Skipping at-rule: ${rule.selector}`);
|
|
29
|
-
return;
|
|
30
|
-
}
|
|
31
|
-
// Skip pseudo-selectors
|
|
32
|
-
if (rule.selector.includes(':')) {
|
|
33
|
-
warnings.push(`Skipped pseudo-selector: ${rule.selector}`);
|
|
34
|
-
logger_1.logger.verbose(`Skipping pseudo-selector: ${rule.selector}`);
|
|
107
|
+
root.walkAtRules((atRule) => {
|
|
108
|
+
if (atRule.name !== 'media') {
|
|
35
109
|
return;
|
|
36
110
|
}
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
warnings.push(`Skipped complex selector: ${rule.selector}`);
|
|
41
|
-
logger_1.logger.verbose(`Skipping complex selector: ${rule.selector}`);
|
|
111
|
+
const mediaResult = (0, breakpointResolver_1.processMediaQuery)(atRule.params, this.breakpoints);
|
|
112
|
+
if (mediaResult.skipped) {
|
|
113
|
+
warnings.push(mediaResult.reason || `Skipped media query: ${atRule.params}`);
|
|
42
114
|
return;
|
|
43
115
|
}
|
|
44
|
-
const
|
|
45
|
-
const
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
116
|
+
const responsiveVariant = mediaResult.breakpoint;
|
|
117
|
+
const nestedRules = [];
|
|
118
|
+
atRule.walkRules((rule) => {
|
|
119
|
+
nestedRules.push(rule);
|
|
120
|
+
});
|
|
121
|
+
for (const rule of nestedRules) {
|
|
122
|
+
const result = this.processRuleWithVariants(rule, [responsiveVariant]);
|
|
123
|
+
if (result) {
|
|
124
|
+
rules.push(...result.cssRules);
|
|
125
|
+
warnings.push(...result.conversionWarnings);
|
|
126
|
+
const anyConverted = result.cssRules.some(r => r.convertedClasses.length > 0);
|
|
127
|
+
if (anyConverted) {
|
|
128
|
+
hasChanges = true;
|
|
129
|
+
const allFullyConverted = result.cssRules.every(r => r.fullyConverted);
|
|
130
|
+
if (allFullyConverted) {
|
|
131
|
+
rule.remove();
|
|
132
|
+
const classNames = result.cssRules.map(r => r.className).join(', .');
|
|
133
|
+
logger_1.logger.verbose(`Removed rule .${classNames} in @media (min-width) → ${responsiveVariant}`);
|
|
134
|
+
}
|
|
135
|
+
else {
|
|
136
|
+
for (const cr of result.conversionResults.flat()) {
|
|
137
|
+
if (cr.converted) {
|
|
138
|
+
rule.walkDecls((decl) => {
|
|
139
|
+
if (decl.prop === cr.declaration.property && decl.value === cr.declaration.value) {
|
|
140
|
+
decl.remove();
|
|
141
|
+
}
|
|
142
|
+
});
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
logger_1.logger.verbose(`Partial conversion in @media → ${responsiveVariant}`);
|
|
146
|
+
}
|
|
147
|
+
}
|
|
51
148
|
}
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
warnings.push(`Skipped calc() value: ${decl.value}`);
|
|
55
|
-
return;
|
|
149
|
+
else {
|
|
150
|
+
warnings.push(`Skipped rule in @media: ${rule.selector}`);
|
|
56
151
|
}
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
}
|
|
62
|
-
|
|
152
|
+
}
|
|
153
|
+
if (atRule.nodes && atRule.nodes.length === 0) {
|
|
154
|
+
atRule.remove();
|
|
155
|
+
logger_1.logger.verbose(`Removed empty @media rule`);
|
|
156
|
+
}
|
|
157
|
+
});
|
|
158
|
+
root.walkRules((rule) => {
|
|
159
|
+
if (rule.parent && rule.parent.type === 'atrule') {
|
|
63
160
|
return;
|
|
64
161
|
}
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
className: result.className
|
|
74
|
-
});
|
|
75
|
-
if (result.skipped && result.reason) {
|
|
76
|
-
conversionWarnings.push(result.reason);
|
|
162
|
+
const result = this.processRuleWithVariants(rule);
|
|
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}`);
|
|
77
170
|
}
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
const
|
|
83
|
-
|
|
84
|
-
const cssRule = {
|
|
85
|
-
selector: rule.selector,
|
|
86
|
-
className,
|
|
87
|
-
declarations,
|
|
88
|
-
convertedClasses,
|
|
89
|
-
skipped: !someDeclarationsConverted,
|
|
90
|
-
fullyConverted: allDeclarationsConverted,
|
|
91
|
-
partialConversion: someDeclarationsConverted && !allDeclarationsConverted,
|
|
92
|
-
reason: !someDeclarationsConverted ? 'No convertible declarations' : undefined
|
|
93
|
-
};
|
|
94
|
-
rules.push(cssRule);
|
|
95
|
-
warnings.push(...conversionWarnings);
|
|
96
|
-
// CRITICAL FIX: Only remove declarations that were successfully converted
|
|
97
|
-
// Never remove the entire rule unless ALL declarations are converted
|
|
98
|
-
if (someDeclarationsConverted) {
|
|
171
|
+
return;
|
|
172
|
+
}
|
|
173
|
+
rules.push(...result.cssRules);
|
|
174
|
+
warnings.push(...result.conversionWarnings);
|
|
175
|
+
const anyConverted = result.cssRules.some(r => r.convertedClasses.length > 0);
|
|
176
|
+
if (anyConverted) {
|
|
99
177
|
hasChanges = true;
|
|
100
|
-
|
|
101
|
-
|
|
178
|
+
const allFullyConverted = result.cssRules.every(r => r.fullyConverted);
|
|
179
|
+
if (allFullyConverted) {
|
|
102
180
|
rule.remove();
|
|
103
|
-
|
|
181
|
+
const classNames = result.cssRules.map(r => r.className).join(', .');
|
|
182
|
+
logger_1.logger.verbose(`Removed rule .${classNames} (all declarations converted)`);
|
|
104
183
|
}
|
|
105
184
|
else {
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
decl.remove();
|
|
114
|
-
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
|
+
});
|
|
115
192
|
}
|
|
116
|
-
}
|
|
117
|
-
logger_1.logger.verbose(`Partial conversion of
|
|
193
|
+
}
|
|
194
|
+
logger_1.logger.verbose(`Partial conversion of rule`);
|
|
118
195
|
}
|
|
119
196
|
}
|
|
120
197
|
});
|
|
121
|
-
// Clean up empty at-rules
|
|
122
198
|
root.walkAtRules((atRule) => {
|
|
123
199
|
if (atRule.nodes && atRule.nodes.length === 0) {
|
|
124
200
|
atRule.remove();
|
|
@@ -142,7 +218,6 @@ class CSSParser {
|
|
|
142
218
|
parseInternalStyle(html) {
|
|
143
219
|
const styles = [];
|
|
144
220
|
const warnings = [];
|
|
145
|
-
// Simple regex to find style tags (this is safe for finding tags, not for parsing content)
|
|
146
221
|
const styleRegex = /<style[^>]*>([\s\S]*?)<\/style>/gi;
|
|
147
222
|
let match;
|
|
148
223
|
while ((match = styleRegex.exec(html)) !== null) {
|
|
@@ -160,7 +235,6 @@ class CSSParser {
|
|
|
160
235
|
let modifiedHtml = html;
|
|
161
236
|
let hasChanges = false;
|
|
162
237
|
const { styles } = this.parseInternalStyle(html);
|
|
163
|
-
// Process styles in reverse order to preserve indices
|
|
164
238
|
for (let i = styles.length - 1; i >= 0; i--) {
|
|
165
239
|
const style = styles[i];
|
|
166
240
|
try {
|
|
@@ -170,11 +244,9 @@ class CSSParser {
|
|
|
170
244
|
if (result.hasChanges) {
|
|
171
245
|
hasChanges = true;
|
|
172
246
|
if (result.canDelete || result.css.trim() === '') {
|
|
173
|
-
// Remove entire style tag
|
|
174
247
|
modifiedHtml = modifiedHtml.slice(0, style.start) + modifiedHtml.slice(style.end);
|
|
175
248
|
}
|
|
176
249
|
else {
|
|
177
|
-
// Replace style content
|
|
178
250
|
const before = modifiedHtml.slice(0, style.start);
|
|
179
251
|
const after = modifiedHtml.slice(style.end);
|
|
180
252
|
const tagStart = html.slice(style.start).match(/<style[^>]*>/)?.[0] || '<style>';
|
|
@@ -197,13 +269,11 @@ class CSSParser {
|
|
|
197
269
|
}
|
|
198
270
|
extractImportPaths(code) {
|
|
199
271
|
const imports = [];
|
|
200
|
-
// Match CSS imports
|
|
201
272
|
const importRegex = /import\s+['"]([^'"]+\.css)['"];?/g;
|
|
202
273
|
let match;
|
|
203
274
|
while ((match = importRegex.exec(code)) !== null) {
|
|
204
275
|
imports.push(match[1]);
|
|
205
276
|
}
|
|
206
|
-
// Match require statements
|
|
207
277
|
const requireRegex = /require\s*\(\s*['"]([^'"]+\.css)['"]\s*\)/g;
|
|
208
278
|
while ((match = requireRegex.exec(code)) !== null) {
|
|
209
279
|
imports.push(match[1]);
|
|
@@ -212,4 +282,4 @@ class CSSParser {
|
|
|
212
282
|
}
|
|
213
283
|
}
|
|
214
284
|
exports.CSSParser = CSSParser;
|
|
215
|
-
//# 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';\n\nexport interface CSSRule {\n  selector: string;\n  className: string;\n  declarations: CSSProperty[];\n  convertedClasses: string[];\n  skipped: boolean;\n  fullyConverted: boolean; // NEW: true if ALL declarations in this rule were converted\n  partialConversion: boolean; // NEW: true if SOME but not all declarations were converted\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\n  constructor(mapper: TailwindMapper) {\n    this.mapper = mapper;\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      // Process each rule\n      root.walkRules((rule) => {\n        // Skip rules inside @media, @supports, etc.\n        if (rule.parent && rule.parent.type === 'atrule') {\n          warnings.push(`Skipped rule with at-rule parent: ${rule.selector}`);\n          logger.verbose(`Skipping at-rule: ${rule.selector}`);\n          return;\n        }\n\n        // Skip pseudo-selectors\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        // Only process simple class selectors\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 className = classNameMatch[1];\n        const declarations: CSSProperty[] = [];\n\n        rule.walkDecls((decl) => {\n          // Skip CSS variables\n          if (decl.prop.startsWith('--')) {\n            warnings.push(`Skipped CSS variable: ${decl.prop}`);\n            return;\n          }\n\n          // Skip calc()\n          if (decl.value.includes('calc(')) {\n            warnings.push(`Skipped calc() value: ${decl.value}`);\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;\n        }\n\n        // Convert to Tailwind classes - track which specific declarations were converted\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 convertedClasses = conversionResults\n          .filter(r => r.converted && r.className)\n          .map(r => r.className!);\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          skipped: !someDeclarationsConverted,\n          fullyConverted: allDeclarationsConverted,\n          partialConversion: someDeclarationsConverted && !allDeclarationsConverted,\n          reason: !someDeclarationsConverted ? 'No convertible declarations' : undefined\n        };\n\n        rules.push(cssRule);\n        warnings.push(...conversionWarnings);\n\n        // CRITICAL FIX: Only remove declarations that were successfully converted\n        // Never remove the entire rule unless ALL declarations are converted\n        if (someDeclarationsConverted) {\n          hasChanges = true;\n          \n          if (allDeclarationsConverted) {\n            // All declarations converted - safe to remove entire rule\n            rule.remove();\n            logger.verbose(`Removed rule .${className} (all ${declarations.length} declarations converted)`);\n          } else {\n            // Partial conversion - only remove the converted declarations\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 .${className}: removed ${removedCount}/${declarations.length} declarations`);\n          }\n        }\n      });\n\n      // Clean up empty at-rules\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}"]}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { TailwindMapper } from './tailwindMapper';
|
|
2
|
+
export interface HTMLParseResult {
|
|
3
|
+
html: string;
|
|
4
|
+
hasChanges: boolean;
|
|
5
|
+
conversions: number;
|
|
6
|
+
warnings: string[];
|
|
7
|
+
}
|
|
8
|
+
export declare class HTMLParser {
|
|
9
|
+
private mapper;
|
|
10
|
+
constructor(mapper: TailwindMapper);
|
|
11
|
+
parse(html: string, filePath: string): HTMLParseResult;
|
|
12
|
+
private parseInlineStyle;
|
|
13
|
+
private mergeClasses;
|
|
14
|
+
extractStylesheets(html: string): string[];
|
|
15
|
+
}
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.HTMLParser = void 0;
|
|
4
|
+
class HTMLParser {
|
|
5
|
+
constructor(mapper) {
|
|
6
|
+
this.mapper = mapper;
|
|
7
|
+
}
|
|
8
|
+
parse(html, filePath) {
|
|
9
|
+
const warnings = [];
|
|
10
|
+
let hasChanges = false;
|
|
11
|
+
let conversions = 0;
|
|
12
|
+
// Parse inline styles: style="display: flex; padding: 16px;"
|
|
13
|
+
// Convert to: class="flex p-4"
|
|
14
|
+
const styleRegex = /style="([^"]*)"/g;
|
|
15
|
+
let modifiedHtml = html;
|
|
16
|
+
let match;
|
|
17
|
+
while ((match = styleRegex.exec(html)) !== null) {
|
|
18
|
+
const fullMatch = match[0];
|
|
19
|
+
const styleValue = match[1];
|
|
20
|
+
// Parse CSS declarations from inline style
|
|
21
|
+
const declarations = this.parseInlineStyle(styleValue);
|
|
22
|
+
if (declarations.length === 0) {
|
|
23
|
+
continue;
|
|
24
|
+
}
|
|
25
|
+
// Convert to Tailwind classes
|
|
26
|
+
const { classes, warnings: convWarnings } = this.mapper.convertMultiple(declarations);
|
|
27
|
+
if (classes.length === 0) {
|
|
28
|
+
warnings.push(...convWarnings);
|
|
29
|
+
continue;
|
|
30
|
+
}
|
|
31
|
+
// Find existing class attribute
|
|
32
|
+
const beforeStyle = html.substring(0, match.index);
|
|
33
|
+
const tagMatch = beforeStyle.match(/<([a-z][a-z0-9]*)[^>]*$/i);
|
|
34
|
+
if (!tagMatch) {
|
|
35
|
+
warnings.push(`Could not find tag for inline style`);
|
|
36
|
+
continue;
|
|
37
|
+
}
|
|
38
|
+
// Check if there's an existing class attribute
|
|
39
|
+
const tagEnd = html.indexOf('>', match.index);
|
|
40
|
+
const tagContent = html.substring(match.index, tagEnd);
|
|
41
|
+
const existingClassMatch = tagContent.match(/class="([^"]*)"/);
|
|
42
|
+
let replacement;
|
|
43
|
+
if (existingClassMatch && existingClassMatch.index !== undefined) {
|
|
44
|
+
// Merge with existing class
|
|
45
|
+
const existingClasses = existingClassMatch[1];
|
|
46
|
+
const mergedClasses = this.mergeClasses(existingClasses, classes);
|
|
47
|
+
// Replace class attribute and remove style
|
|
48
|
+
const beforeClass = modifiedHtml.substring(0, match.index + existingClassMatch.index);
|
|
49
|
+
const afterStyle = modifiedHtml.substring(match.index + fullMatch.length);
|
|
50
|
+
// This is complex - need to handle both class and style replacement
|
|
51
|
+
// For now, simplified version:
|
|
52
|
+
replacement = `class="${mergedClasses}"`;
|
|
53
|
+
modifiedHtml = modifiedHtml.replace(fullMatch, replacement);
|
|
54
|
+
}
|
|
55
|
+
else {
|
|
56
|
+
// Add class attribute, remove style
|
|
57
|
+
replacement = `class="${classes.join(' ')}"`;
|
|
58
|
+
modifiedHtml = modifiedHtml.replace(fullMatch, replacement);
|
|
59
|
+
}
|
|
60
|
+
conversions += classes.length;
|
|
61
|
+
hasChanges = true;
|
|
62
|
+
warnings.push(...convWarnings);
|
|
63
|
+
}
|
|
64
|
+
return {
|
|
65
|
+
html: modifiedHtml,
|
|
66
|
+
hasChanges,
|
|
67
|
+
conversions,
|
|
68
|
+
warnings
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
parseInlineStyle(styleValue) {
|
|
72
|
+
const declarations = [];
|
|
73
|
+
// Split by semicolon
|
|
74
|
+
const props = styleValue.split(';').filter(s => s.trim());
|
|
75
|
+
props.forEach(prop => {
|
|
76
|
+
const colonIndex = prop.indexOf(':');
|
|
77
|
+
if (colonIndex === -1)
|
|
78
|
+
return;
|
|
79
|
+
const property = prop.substring(0, colonIndex).trim();
|
|
80
|
+
const value = prop.substring(colonIndex + 1).trim();
|
|
81
|
+
if (property && value) {
|
|
82
|
+
declarations.push({ property, value });
|
|
83
|
+
}
|
|
84
|
+
});
|
|
85
|
+
return declarations;
|
|
86
|
+
}
|
|
87
|
+
mergeClasses(existing, newClasses) {
|
|
88
|
+
const existingSet = new Set(existing.split(/\s+/).filter(Boolean));
|
|
89
|
+
newClasses.forEach(cls => existingSet.add(cls));
|
|
90
|
+
return Array.from(existingSet).join(' ');
|
|
91
|
+
}
|
|
92
|
+
extractStylesheets(html) {
|
|
93
|
+
const stylesheets = [];
|
|
94
|
+
const linkRegex = /<link[^>]*rel="stylesheet"[^>]*href="([^"]*)"[^>]*>/gi;
|
|
95
|
+
let match;
|
|
96
|
+
while ((match = linkRegex.exec(html)) !== null) {
|
|
97
|
+
stylesheets.push(match[1]);
|
|
98
|
+
}
|
|
99
|
+
return stylesheets;
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
exports.HTMLParser = HTMLParser;
|
|
103
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"htmlParser.js","sourceRoot":"","sources":["../src/htmlParser.ts"],"names":[],"mappings":"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","sourcesContent":["import { TailwindMapper } from './tailwindMapper';\nimport { logger } from './utils/logger';\n\nexport interface HTMLParseResult {\n  html: string;\n  hasChanges: boolean;\n  conversions: number;\n  warnings: string[];\n}\n\nexport class HTMLParser {\n  private mapper: TailwindMapper;\n\n  constructor(mapper: TailwindMapper) {\n    this.mapper = mapper;\n  }\n\n  parse(html: string, filePath: string): HTMLParseResult {\n    const warnings: string[] = [];\n    let hasChanges = false;\n    let conversions = 0;\n\n    // Parse inline styles: style=\"display: flex; padding: 16px;\"\n    // Convert to: class=\"flex p-4\"\n    const styleRegex = /style=\"([^\"]*)\"/g;\n    let modifiedHtml = html;\n    let match;\n\n    while ((match = styleRegex.exec(html)) !== null) {\n      const fullMatch = match[0];\n      const styleValue = match[1];\n      \n      // Parse CSS declarations from inline style\n      const declarations = this.parseInlineStyle(styleValue);\n      \n      if (declarations.length === 0) {\n        continue;\n      }\n\n      // Convert to Tailwind classes\n      const { classes, warnings: convWarnings } = this.mapper.convertMultiple(declarations);\n\n      if (classes.length === 0) {\n        warnings.push(...convWarnings);\n        continue;\n      }\n\n      // Find existing class attribute\n      const beforeStyle = html.substring(0, match.index);\n      const tagMatch = beforeStyle.match(/<([a-z][a-z0-9]*)[^>]*$/i);\n      \n      if (!tagMatch) {\n        warnings.push(`Could not find tag for inline style`);\n        continue;\n      }\n\n      // Check if there's an existing class attribute\n      const tagEnd = html.indexOf('>', match.index);\n      const tagContent = html.substring(match.index, tagEnd);\n      const existingClassMatch = tagContent.match(/class=\"([^\"]*)\"/);\n\n      let replacement: string;\n      \n        if (existingClassMatch && existingClassMatch.index !== undefined) {\n        // Merge with existing class\n        const existingClasses = existingClassMatch[1];\n        const mergedClasses = this.mergeClasses(existingClasses, classes);\n        \n        // Replace class attribute and remove style\n        const beforeClass = modifiedHtml.substring(0, match.index + existingClassMatch.index);\n        const afterStyle = modifiedHtml.substring(match.index + fullMatch.length);\n        \n        // This is complex - need to handle both class and style replacement\n        // For now, simplified version:\n        replacement = `class=\"${mergedClasses}\"`;\n        modifiedHtml = modifiedHtml.replace(fullMatch, replacement);\n      } else {\n        // Add class attribute, remove style\n        replacement = `class=\"${classes.join(' ')}\"`;\n        modifiedHtml = modifiedHtml.replace(fullMatch, replacement);\n      }\n\n      conversions += classes.length;\n      hasChanges = true;\n      warnings.push(...convWarnings);\n    }\n\n    return {\n      html: modifiedHtml,\n      hasChanges,\n      conversions,\n      warnings\n    };\n  }\n\n  private parseInlineStyle(styleValue: string): Array<{property: string; value: string}> {\n    const declarations: Array<{property: string; value: string}> = [];\n    \n    // Split by semicolon\n    const props = styleValue.split(';').filter(s => s.trim());\n    \n    props.forEach(prop => {\n      const colonIndex = prop.indexOf(':');\n      if (colonIndex === -1) return;\n      \n      const property = prop.substring(0, colonIndex).trim();\n      const value = prop.substring(colonIndex + 1).trim();\n      \n      if (property && value) {\n        declarations.push({ property, value });\n      }\n    });\n    \n    return declarations;\n  }\n\n  private mergeClasses(existing: string, newClasses: string[]): string {\n    const existingSet = new Set(existing.split(/\\s+/).filter(Boolean));\n    newClasses.forEach(cls => existingSet.add(cls));\n    return Array.from(existingSet).join(' ');\n  }\n\n  extractStylesheets(html: string): string[] {\n    const stylesheets: string[] = [];\n    const linkRegex = /<link[^>]*rel=\"stylesheet\"[^>]*href=\"([^\"]*)\"[^>]*>/gi;\n    let match;\n\n    while ((match = linkRegex.exec(html)) !== null) {\n      stylesheets.push(match[1]);\n    }\n\n    return stylesheets;\n  }\n}\n"]}
|
package/dist/index.d.ts
CHANGED
|
@@ -2,7 +2,10 @@ export { scanProject, ScannedFile } from './scanner';
|
|
|
2
2
|
export { transformFiles, TransformOptions, TransformResults } from './transformer';
|
|
3
3
|
export { TailwindMapper, CSSProperty, ConversionResult } from './tailwindMapper';
|
|
4
4
|
export { JSXParser, JSXTransformation, JSXParseResult } from './jsxParser';
|
|
5
|
-
export { CSSParser, CSSRule, CSSParseResult } from './cssParser';
|
|
5
|
+
export { CSSParser, CSSRule, CSSParseResult, UtilityWithVariant } from './cssParser';
|
|
6
6
|
export { FileWriter, FileWriteOptions } from './fileWriter';
|
|
7
7
|
export { loadTailwindConfig, TailwindConfig } from './utils/config';
|
|
8
8
|
export { logger } from './utils/logger';
|
|
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';
|