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/index.js
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
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; } });
|
|
@@ -18,4 +18,28 @@ var config_1 = require("./utils/config");
|
|
|
18
18
|
Object.defineProperty(exports, "loadTailwindConfig", { enumerable: true, get: function () { return config_1.loadTailwindConfig; } });
|
|
19
19
|
var logger_1 = require("./utils/logger");
|
|
20
20
|
Object.defineProperty(exports, "logger", { enumerable: true, get: function () { return logger_1.logger; } });
|
|
21
|
-
|
|
21
|
+
var breakpointResolver_1 = require("./utils/breakpointResolver");
|
|
22
|
+
Object.defineProperty(exports, "getDefaultBreakpoints", { enumerable: true, get: function () { return breakpointResolver_1.getDefaultBreakpoints; } });
|
|
23
|
+
Object.defineProperty(exports, "resolveBreakpointsFromConfig", { enumerable: true, get: function () { return breakpointResolver_1.resolveBreakpointsFromConfig; } });
|
|
24
|
+
Object.defineProperty(exports, "parseMediaQuery", { enumerable: true, get: function () { return breakpointResolver_1.parseMediaQuery; } });
|
|
25
|
+
Object.defineProperty(exports, "findBreakpointForMinWidth", { enumerable: true, get: function () { return breakpointResolver_1.findBreakpointForMinWidth; } });
|
|
26
|
+
Object.defineProperty(exports, "processMediaQuery", { enumerable: true, get: function () { return breakpointResolver_1.processMediaQuery; } });
|
|
27
|
+
Object.defineProperty(exports, "prefixWithBreakpoint", { enumerable: true, get: function () { return breakpointResolver_1.prefixWithBreakpoint; } });
|
|
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
|
@@ -11,6 +11,8 @@ const jsxParser_1 = require("./jsxParser");
|
|
|
11
11
|
const cssParser_1 = require("./cssParser");
|
|
12
12
|
const fileWriter_1 = require("./fileWriter");
|
|
13
13
|
const logger_1 = require("./utils/logger");
|
|
14
|
+
const breakpointResolver_1 = require("./utils/breakpointResolver");
|
|
15
|
+
const variantAssembler_1 = require("./utils/variantAssembler");
|
|
14
16
|
async function transformFiles(files, options) {
|
|
15
17
|
const results = {
|
|
16
18
|
filesScanned: files.length,
|
|
@@ -21,38 +23,34 @@ async function transformFiles(files, options) {
|
|
|
21
23
|
};
|
|
22
24
|
const mapper = new tailwindMapper_1.TailwindMapper(options.tailwindConfig || {});
|
|
23
25
|
const jsxParser = new jsxParser_1.JSXParser(mapper);
|
|
24
|
-
const
|
|
26
|
+
const screens = options.tailwindConfig?.theme?.screens;
|
|
27
|
+
const cssParser = new cssParser_1.CSSParser(mapper, screens);
|
|
25
28
|
const fileWriter = new fileWriter_1.FileWriter({ dryRun: options.dryRun });
|
|
26
|
-
|
|
27
|
-
// Collect CSS mappings and gather info about what can be safely converted
|
|
29
|
+
(0, breakpointResolver_1.clearBreakpointCache)();
|
|
28
30
|
const cssClassMap = {};
|
|
29
31
|
const cssFileResults = new Map();
|
|
30
32
|
logger_1.logger.info('\nš Phase 1: Analyzing files...');
|
|
31
|
-
// Analyze CSS files
|
|
32
33
|
if (!options.skipExternal) {
|
|
33
34
|
for (const file of files.filter(f => f.type === 'css')) {
|
|
34
35
|
try {
|
|
35
36
|
const content = fs_1.default.readFileSync(file.path, 'utf-8');
|
|
36
37
|
const result = await cssParser.parse(content, file.path);
|
|
37
|
-
// Check if ALL rules in this file are FULLY converted (all declarations)
|
|
38
38
|
const totalRules = result.rules.length;
|
|
39
39
|
const fullyConvertedRules = result.rules.filter(r => r.fullyConverted).length;
|
|
40
40
|
const partiallyConvertedRules = result.rules.filter(r => r.partialConversion).length;
|
|
41
|
-
// A file is only "fully convertible" if ALL rules are fully converted (no partial conversions)
|
|
42
41
|
const fullyConvertible = totalRules > 0 && totalRules === fullyConvertedRules && partiallyConvertedRules === 0;
|
|
43
|
-
// Build class map (only for fully converted classes - partial conversions keep the CSS)
|
|
44
42
|
result.rules.forEach(rule => {
|
|
45
43
|
if (rule.fullyConverted) {
|
|
46
|
-
cssClassMap[rule.className]
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
44
|
+
const existing = cssClassMap[rule.className];
|
|
45
|
+
if (existing) {
|
|
46
|
+
mergeRuleIntoClassInfo(existing, rule);
|
|
47
|
+
}
|
|
48
|
+
else {
|
|
49
|
+
cssClassMap[rule.className] = buildClassInfoFromRule(rule, file.path);
|
|
50
|
+
}
|
|
51
51
|
results.stylesConverted += rule.declarations.length;
|
|
52
52
|
}
|
|
53
53
|
else if (rule.partialConversion) {
|
|
54
|
-
// For partial conversions, we converted some declarations but keep the CSS rule
|
|
55
|
-
// Count the converted declarations
|
|
56
54
|
results.stylesConverted += rule.convertedClasses.length;
|
|
57
55
|
logger_1.logger.verbose(` Rule .${rule.className}: partial conversion (${rule.convertedClasses.length}/${rule.declarations.length} declarations)`);
|
|
58
56
|
}
|
|
@@ -66,13 +64,11 @@ async function transformFiles(files, options) {
|
|
|
66
64
|
fullyConvertible
|
|
67
65
|
});
|
|
68
66
|
results.warnings += result.warnings.length;
|
|
69
|
-
// Log analysis
|
|
70
67
|
logger_1.logger.verbose(`Analyzed ${file.path}:`);
|
|
71
68
|
logger_1.logger.verbose(` - Total rules: ${totalRules}`);
|
|
72
69
|
logger_1.logger.verbose(` - Fully converted rules: ${fullyConvertedRules}`);
|
|
73
70
|
logger_1.logger.verbose(` - Partially converted rules: ${partiallyConvertedRules}`);
|
|
74
71
|
logger_1.logger.verbose(` - Fully convertible: ${fullyConvertible}`);
|
|
75
|
-
// Log warnings
|
|
76
72
|
result.warnings.forEach(warning => {
|
|
77
73
|
logger_1.logger.verbose(`ā ļø ${file.path}: ${warning}`);
|
|
78
74
|
});
|
|
@@ -83,7 +79,6 @@ async function transformFiles(files, options) {
|
|
|
83
79
|
}
|
|
84
80
|
}
|
|
85
81
|
}
|
|
86
|
-
// PASS 2: Transform JSX/TSX files
|
|
87
82
|
logger_1.logger.info('\nāļø Phase 2: Transforming React components...');
|
|
88
83
|
const jsxFileResults = new Map();
|
|
89
84
|
for (const file of files.filter(f => f.type === 'jsx')) {
|
|
@@ -92,7 +87,6 @@ async function transformFiles(files, options) {
|
|
|
92
87
|
const originalContent = content;
|
|
93
88
|
let hasChanges = false;
|
|
94
89
|
let fileWarnings = [];
|
|
95
|
-
// Process inline styles
|
|
96
90
|
if (!options.skipInline) {
|
|
97
91
|
try {
|
|
98
92
|
const jsxResult = jsxParser.parse(content, file.path);
|
|
@@ -108,21 +102,21 @@ async function transformFiles(files, options) {
|
|
|
108
102
|
fileWarnings.push(`JSX parse error: ${error}`);
|
|
109
103
|
}
|
|
110
104
|
}
|
|
111
|
-
// Process internal CSS
|
|
112
105
|
if (!options.skipInternal) {
|
|
113
106
|
try {
|
|
114
107
|
const internalResult = await cssParser.parseInternalCSS(content, file.path);
|
|
115
108
|
if (internalResult.hasChanges) {
|
|
116
109
|
content = internalResult.html;
|
|
117
110
|
hasChanges = true;
|
|
118
|
-
// Build class map from internal styles
|
|
119
111
|
internalResult.rules.forEach(rule => {
|
|
120
112
|
if (rule.convertedClasses.length > 0) {
|
|
121
|
-
cssClassMap[rule.className]
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
113
|
+
const existing = cssClassMap[rule.className];
|
|
114
|
+
if (existing) {
|
|
115
|
+
mergeRuleIntoClassInfo(existing, rule);
|
|
116
|
+
}
|
|
117
|
+
else {
|
|
118
|
+
cssClassMap[rule.className] = buildClassInfoFromRule(rule, file.path);
|
|
119
|
+
}
|
|
126
120
|
results.stylesConverted += rule.declarations.length;
|
|
127
121
|
}
|
|
128
122
|
});
|
|
@@ -140,7 +134,6 @@ async function transformFiles(files, options) {
|
|
|
140
134
|
hasChanges
|
|
141
135
|
});
|
|
142
136
|
results.warnings += fileWarnings.length;
|
|
143
|
-
// Log warnings
|
|
144
137
|
fileWarnings.forEach(warning => {
|
|
145
138
|
logger_1.logger.verbose(`ā ļø ${file.path}: ${warning}`);
|
|
146
139
|
});
|
|
@@ -150,14 +143,11 @@ async function transformFiles(files, options) {
|
|
|
150
143
|
results.warnings++;
|
|
151
144
|
}
|
|
152
145
|
}
|
|
153
|
-
// PASS 3: Replace className references from external CSS
|
|
154
|
-
// This must happen after all JSX files are parsed
|
|
155
146
|
if (Object.keys(cssClassMap).length > 0) {
|
|
156
147
|
logger_1.logger.info('\nš Phase 3: Replacing className references...');
|
|
157
148
|
for (const [filePath, fileResult] of jsxFileResults) {
|
|
158
149
|
let content = fileResult.newContent;
|
|
159
150
|
let hasChanges = fileResult.hasChanges;
|
|
160
|
-
// Replace className references
|
|
161
151
|
const replacementResult = replaceClassNameReferences(content, cssClassMap);
|
|
162
152
|
if (replacementResult.hasChanges) {
|
|
163
153
|
content = replacementResult.code;
|
|
@@ -165,7 +155,6 @@ async function transformFiles(files, options) {
|
|
|
165
155
|
results.classesReplaced += replacementResult.replacements;
|
|
166
156
|
logger_1.logger.verbose(`Replaced ${replacementResult.replacements} class references in ${path_1.default.basename(filePath)}`);
|
|
167
157
|
}
|
|
168
|
-
// Update the result
|
|
169
158
|
jsxFileResults.set(filePath, {
|
|
170
159
|
...fileResult,
|
|
171
160
|
newContent: content,
|
|
@@ -173,9 +162,7 @@ async function transformFiles(files, options) {
|
|
|
173
162
|
});
|
|
174
163
|
}
|
|
175
164
|
}
|
|
176
|
-
// PASS 4: Write all changes
|
|
177
165
|
logger_1.logger.info('\nš¾ Phase 4: Writing changes...');
|
|
178
|
-
// Write JSX files
|
|
179
166
|
for (const [filePath, fileResult] of jsxFileResults) {
|
|
180
167
|
if (fileResult.hasChanges) {
|
|
181
168
|
const success = await fileWriter.writeFile(filePath, fileResult.newContent, fileResult.content);
|
|
@@ -184,19 +171,15 @@ async function transformFiles(files, options) {
|
|
|
184
171
|
}
|
|
185
172
|
}
|
|
186
173
|
}
|
|
187
|
-
// Write CSS files (SAFETY: Only modify if fully convertible or explicitly allowed)
|
|
188
174
|
if (!options.skipExternal) {
|
|
189
175
|
for (const [filePath, fileResult] of cssFileResults) {
|
|
190
176
|
if (!fileResult.hasChanges)
|
|
191
177
|
continue;
|
|
192
|
-
// SAFETY RULE 1: Never modify CSS files that aren't fully convertible
|
|
193
|
-
// unless they only have unconvertible rules (no changes needed)
|
|
194
178
|
if (!fileResult.fullyConvertible) {
|
|
195
179
|
logger_1.logger.warn(`āļø Skipping ${path_1.default.basename(filePath)} - not fully convertible (would break styles)`);
|
|
196
180
|
logger_1.logger.warn(` Convertible: ${fileResult.rules.filter(r => r.convertedClasses.length > 0).length}/${fileResult.rules.length} rules`);
|
|
197
181
|
continue;
|
|
198
182
|
}
|
|
199
|
-
// SAFETY RULE 2: Only delete if ALL rules converted AND --delete-css flag used
|
|
200
183
|
if (fileResult.canDelete && options.deleteCss) {
|
|
201
184
|
const success = await fileWriter.deleteFile(filePath);
|
|
202
185
|
if (success) {
|
|
@@ -205,11 +188,9 @@ async function transformFiles(files, options) {
|
|
|
205
188
|
}
|
|
206
189
|
}
|
|
207
190
|
else if (fileResult.canDelete && !options.deleteCss) {
|
|
208
|
-
// File is empty but don't delete without permission
|
|
209
191
|
logger_1.logger.info(`ā¹ļø ${path_1.default.basename(filePath)} is now empty (use --delete-css to remove)`);
|
|
210
192
|
}
|
|
211
193
|
else {
|
|
212
|
-
// Write modified CSS (only if fully convertible)
|
|
213
194
|
const success = await fileWriter.writeFile(filePath, fileResult.newContent, fileResult.content);
|
|
214
195
|
if (success) {
|
|
215
196
|
results.filesModified++;
|
|
@@ -219,17 +200,48 @@ async function transformFiles(files, options) {
|
|
|
219
200
|
}
|
|
220
201
|
return results;
|
|
221
202
|
}
|
|
203
|
+
function buildClassInfoFromRule(rule, sourceFile) {
|
|
204
|
+
return {
|
|
205
|
+
utilities: rule.utilities.map(u => ({
|
|
206
|
+
value: u.value,
|
|
207
|
+
variants: (0, variantAssembler_1.normalizeVariantOrder)([...u.variants])
|
|
208
|
+
})),
|
|
209
|
+
sourceFile,
|
|
210
|
+
fullyConvertible: true
|
|
211
|
+
};
|
|
212
|
+
}
|
|
213
|
+
function mergeRuleIntoClassInfo(info, rule) {
|
|
214
|
+
for (const utility of rule.utilities) {
|
|
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
|
+
}
|
|
221
|
+
}
|
|
222
|
+
existing.variants = (0, variantAssembler_1.normalizeVariantOrder)(existing.variants);
|
|
223
|
+
}
|
|
224
|
+
else {
|
|
225
|
+
info.utilities.push({
|
|
226
|
+
value: utility.value,
|
|
227
|
+
variants: (0, variantAssembler_1.normalizeVariantOrder)([...utility.variants])
|
|
228
|
+
});
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
function assembleTailwindClasses(info) {
|
|
233
|
+
const merged = (0, variantAssembler_1.mergeUtilities)(info.utilities);
|
|
234
|
+
return (0, variantAssembler_1.assembleUtilities)(merged).join(' ');
|
|
235
|
+
}
|
|
222
236
|
function replaceClassNameReferences(code, classMap) {
|
|
223
237
|
let hasChanges = false;
|
|
224
238
|
let replacements = 0;
|
|
225
239
|
let modifiedCode = code;
|
|
226
240
|
Object.entries(classMap).forEach(([oldClass, info]) => {
|
|
227
|
-
// Skip if not fully convertible
|
|
228
241
|
if (!info.fullyConvertible) {
|
|
229
242
|
return;
|
|
230
243
|
}
|
|
231
|
-
const tailwindClassString = info
|
|
232
|
-
// Pattern 1: className="oldClass" (simple string)
|
|
244
|
+
const tailwindClassString = assembleTailwindClasses(info);
|
|
233
245
|
const pattern1 = new RegExp(`className="${oldClass}"`, 'g');
|
|
234
246
|
if (pattern1.test(modifiedCode)) {
|
|
235
247
|
modifiedCode = modifiedCode.replace(pattern1, `className="${tailwindClassString}"`);
|
|
@@ -237,7 +249,6 @@ function replaceClassNameReferences(code, classMap) {
|
|
|
237
249
|
replacements++;
|
|
238
250
|
logger_1.logger.verbose(`Replaced className="${oldClass}" with "${tailwindClassString}"`);
|
|
239
251
|
}
|
|
240
|
-
// Pattern 2: className={"oldClass"} (expression with string)
|
|
241
252
|
const pattern2 = new RegExp(`className=\\{"${oldClass}"\\}`, 'g');
|
|
242
253
|
if (pattern2.test(modifiedCode)) {
|
|
243
254
|
modifiedCode = modifiedCode.replace(pattern2, `className="${tailwindClassString}"`);
|
|
@@ -245,7 +256,6 @@ function replaceClassNameReferences(code, classMap) {
|
|
|
245
256
|
replacements++;
|
|
246
257
|
logger_1.logger.verbose(`Replaced className={"${oldClass}"} with "${tailwindClassString}"`);
|
|
247
258
|
}
|
|
248
|
-
// Pattern 3: className={`oldClass`} (simple template literal)
|
|
249
259
|
const pattern3 = new RegExp(`className=\\{\`\\s*${oldClass}\\s*\`\\}`, 'g');
|
|
250
260
|
if (pattern3.test(modifiedCode)) {
|
|
251
261
|
modifiedCode = modifiedCode.replace(pattern3, `className="${tailwindClassString}"`);
|
|
@@ -256,4 +266,4 @@ function replaceClassNameReferences(code, classMap) {
|
|
|
256
266
|
});
|
|
257
267
|
return { code: modifiedCode, hasChanges, replacements };
|
|
258
268
|
}
|
|
259
|
-
//# 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 } from './cssParser';\nimport { FileWriter } from './fileWriter';\nimport { TailwindConfig } from './utils/config';\nimport { logger } from './utils/logger';\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 CSSClassMap {\n  [className: string]: {\n    tailwindClasses: string[];\n    sourceFile: string;\n    fullyConvertible: boolean;\n  };\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 cssParser = new CSSParser(mapper);\n  const fileWriter = new FileWriter({ dryRun: options.dryRun });\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            cssClassMap[rule.className] = {\n              tailwindClasses: rule.convertedClasses,\n              sourceFile: file.path,\n              fullyConvertible: true\n            };\n            results.stylesConverted += rule.declarations.length;\n          } else if (rule.partialConversion) {\n            // For partial conversions, we converted some declarations but keep the CSS rule\n            // Count the converted declarations\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                cssClassMap[rule.className] = {\n                  tailwindClasses: rule.convertedClasses,\n                  sourceFile: file.path,\n                  fullyConvertible: true\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 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    // Skip if not fully convertible\n    if (!info.fullyConvertible) {\n      return;\n    }\n\n    const tailwindClassString = info.tailwindClasses.join(' ');\n    \n    // Pattern 1: className=\"oldClass\" (simple string)\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    // Pattern 2: className={\"oldClass\"} (expression with string)\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    // Pattern 3: className={`oldClass`} (simple template literal)\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,21 @@
|
|
|
1
|
+
export interface Breakpoint {
|
|
2
|
+
name: string;
|
|
3
|
+
minWidth: number;
|
|
4
|
+
}
|
|
5
|
+
export interface MediaQueryInfo {
|
|
6
|
+
type: 'min-width' | 'max-width' | 'unsupported';
|
|
7
|
+
value?: number;
|
|
8
|
+
raw: string;
|
|
9
|
+
}
|
|
10
|
+
export declare function getDefaultBreakpoints(): Breakpoint[];
|
|
11
|
+
export declare function resolveBreakpointsFromConfig(screens: Record<string, string | [string, string]> | undefined): Breakpoint[];
|
|
12
|
+
export declare function getBreakpoints(screens?: Record<string, string | [string, string]>): Breakpoint[];
|
|
13
|
+
export declare function clearBreakpointCache(): void;
|
|
14
|
+
export declare function parseMediaQuery(params: string): MediaQueryInfo;
|
|
15
|
+
export declare function findBreakpointForMinWidth(minWidth: number, breakpoints: Breakpoint[]): string | null;
|
|
16
|
+
export declare function prefixWithBreakpoint(className: string, breakpoint: string): string;
|
|
17
|
+
export declare function processMediaQuery(params: string, breakpoints: Breakpoint[]): {
|
|
18
|
+
breakpoint: string | null;
|
|
19
|
+
skipped: boolean;
|
|
20
|
+
reason?: string;
|
|
21
|
+
};
|
|
@@ -0,0 +1,154 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.getDefaultBreakpoints = getDefaultBreakpoints;
|
|
4
|
+
exports.resolveBreakpointsFromConfig = resolveBreakpointsFromConfig;
|
|
5
|
+
exports.getBreakpoints = getBreakpoints;
|
|
6
|
+
exports.clearBreakpointCache = clearBreakpointCache;
|
|
7
|
+
exports.parseMediaQuery = parseMediaQuery;
|
|
8
|
+
exports.findBreakpointForMinWidth = findBreakpointForMinWidth;
|
|
9
|
+
exports.prefixWithBreakpoint = prefixWithBreakpoint;
|
|
10
|
+
exports.processMediaQuery = processMediaQuery;
|
|
11
|
+
const logger_1 = require("./logger");
|
|
12
|
+
const DEFAULT_BREAKPOINTS = [
|
|
13
|
+
{ name: 'sm', minWidth: 640 },
|
|
14
|
+
{ name: 'md', minWidth: 768 },
|
|
15
|
+
{ name: 'lg', minWidth: 1024 },
|
|
16
|
+
{ name: 'xl', minWidth: 1280 },
|
|
17
|
+
{ name: '2xl', minWidth: 1536 }
|
|
18
|
+
];
|
|
19
|
+
let cachedBreakpoints = null;
|
|
20
|
+
function getDefaultBreakpoints() {
|
|
21
|
+
return [...DEFAULT_BREAKPOINTS];
|
|
22
|
+
}
|
|
23
|
+
function resolveBreakpointsFromConfig(screens) {
|
|
24
|
+
if (!screens) {
|
|
25
|
+
return getDefaultBreakpoints();
|
|
26
|
+
}
|
|
27
|
+
const breakpoints = [];
|
|
28
|
+
for (const [name, value] of Object.entries(screens)) {
|
|
29
|
+
let minWidth = null;
|
|
30
|
+
if (typeof value === 'string') {
|
|
31
|
+
minWidth = parsePixelValue(value);
|
|
32
|
+
}
|
|
33
|
+
else if (Array.isArray(value)) {
|
|
34
|
+
minWidth = parsePixelValue(value[0]);
|
|
35
|
+
}
|
|
36
|
+
if (minWidth !== null) {
|
|
37
|
+
breakpoints.push({ name, minWidth });
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
breakpoints.sort((a, b) => a.minWidth - b.minWidth);
|
|
41
|
+
return breakpoints.length > 0 ? breakpoints : getDefaultBreakpoints();
|
|
42
|
+
}
|
|
43
|
+
function parsePixelValue(value) {
|
|
44
|
+
const pxMatch = value.match(/^([\d.]+)px$/);
|
|
45
|
+
if (pxMatch) {
|
|
46
|
+
return parseFloat(pxMatch[1]);
|
|
47
|
+
}
|
|
48
|
+
const remMatch = value.match(/^([\d.]+)rem$/);
|
|
49
|
+
if (remMatch) {
|
|
50
|
+
return parseFloat(remMatch[1]) * 16;
|
|
51
|
+
}
|
|
52
|
+
const emMatch = value.match(/^([\d.]+)em$/);
|
|
53
|
+
if (emMatch) {
|
|
54
|
+
return parseFloat(emMatch[1]) * 16;
|
|
55
|
+
}
|
|
56
|
+
return null;
|
|
57
|
+
}
|
|
58
|
+
function getBreakpoints(screens) {
|
|
59
|
+
if (cachedBreakpoints && !screens) {
|
|
60
|
+
return cachedBreakpoints;
|
|
61
|
+
}
|
|
62
|
+
const breakpoints = screens ? resolveBreakpointsFromConfig(screens) : getDefaultBreakpoints();
|
|
63
|
+
if (!screens) {
|
|
64
|
+
cachedBreakpoints = breakpoints;
|
|
65
|
+
}
|
|
66
|
+
return breakpoints;
|
|
67
|
+
}
|
|
68
|
+
function clearBreakpointCache() {
|
|
69
|
+
cachedBreakpoints = null;
|
|
70
|
+
}
|
|
71
|
+
function parseMediaQuery(params) {
|
|
72
|
+
const trimmed = params.trim().toLowerCase();
|
|
73
|
+
if (trimmed.includes('screen') && trimmed.includes('and')) {
|
|
74
|
+
return {
|
|
75
|
+
type: 'unsupported',
|
|
76
|
+
raw: params
|
|
77
|
+
};
|
|
78
|
+
}
|
|
79
|
+
if (trimmed.includes('orientation') || trimmed.includes('print')) {
|
|
80
|
+
return {
|
|
81
|
+
type: 'unsupported',
|
|
82
|
+
raw: params
|
|
83
|
+
};
|
|
84
|
+
}
|
|
85
|
+
const minMatch = trimmed.match(/\(\s*min-width\s*:\s*([\d.]+)(px|rem|em)\s*\)/);
|
|
86
|
+
if (minMatch) {
|
|
87
|
+
const value = parseFloat(minMatch[1]);
|
|
88
|
+
const unit = minMatch[2];
|
|
89
|
+
let pxValue = value;
|
|
90
|
+
if (unit === 'rem' || unit === 'em') {
|
|
91
|
+
pxValue = value * 16;
|
|
92
|
+
}
|
|
93
|
+
return {
|
|
94
|
+
type: 'min-width',
|
|
95
|
+
value: pxValue,
|
|
96
|
+
raw: params
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
const maxMatch = trimmed.match(/\(\s*max-width\s*:\s*([\d.]+)(px|rem|em)\s*\)/);
|
|
100
|
+
if (maxMatch) {
|
|
101
|
+
return {
|
|
102
|
+
type: 'max-width',
|
|
103
|
+
raw: params
|
|
104
|
+
};
|
|
105
|
+
}
|
|
106
|
+
return {
|
|
107
|
+
type: 'unsupported',
|
|
108
|
+
raw: params
|
|
109
|
+
};
|
|
110
|
+
}
|
|
111
|
+
function findBreakpointForMinWidth(minWidth, breakpoints) {
|
|
112
|
+
const exact = breakpoints.find(bp => bp.minWidth === minWidth);
|
|
113
|
+
if (exact) {
|
|
114
|
+
return exact.name;
|
|
115
|
+
}
|
|
116
|
+
const closest = breakpoints.reduce((closest, bp) => {
|
|
117
|
+
if (!closest)
|
|
118
|
+
return bp;
|
|
119
|
+
const currentDiff = Math.abs(bp.minWidth - minWidth);
|
|
120
|
+
const closestDiff = Math.abs(closest.minWidth - minWidth);
|
|
121
|
+
return currentDiff < closestDiff ? bp : closest;
|
|
122
|
+
}, null);
|
|
123
|
+
if (closest) {
|
|
124
|
+
const diff = Math.abs(closest.minWidth - minWidth);
|
|
125
|
+
const tolerance = minWidth * 0.05;
|
|
126
|
+
if (diff <= tolerance) {
|
|
127
|
+
logger_1.logger.verbose(`Matched min-width ${minWidth}px to closest breakpoint ${closest.name} (${closest.minWidth}px)`);
|
|
128
|
+
return closest.name;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
return null;
|
|
132
|
+
}
|
|
133
|
+
function prefixWithBreakpoint(className, breakpoint) {
|
|
134
|
+
return `${breakpoint}:${className}`;
|
|
135
|
+
}
|
|
136
|
+
function processMediaQuery(params, breakpoints) {
|
|
137
|
+
const info = parseMediaQuery(params);
|
|
138
|
+
if (info.type !== 'min-width' || info.value === undefined) {
|
|
139
|
+
const reason = info.type === 'max-width'
|
|
140
|
+
? `Skipped media query (max-width: ...) ā unsupported`
|
|
141
|
+
: `Skipped media query (${info.raw}) ā unsupported`;
|
|
142
|
+
logger_1.logger.verbose(reason);
|
|
143
|
+
return { breakpoint: null, skipped: true, reason };
|
|
144
|
+
}
|
|
145
|
+
const breakpoint = findBreakpointForMinWidth(info.value, breakpoints);
|
|
146
|
+
if (!breakpoint) {
|
|
147
|
+
const reason = `No matching breakpoint for min-width: ${info.value}px`;
|
|
148
|
+
logger_1.logger.verbose(reason);
|
|
149
|
+
return { breakpoint: null, skipped: true, reason };
|
|
150
|
+
}
|
|
151
|
+
logger_1.logger.verbose(`Converted media query (min-width: ${info.value}px) ā ${breakpoint}`);
|
|
152
|
+
return { breakpoint, skipped: false };
|
|
153
|
+
}
|
|
154
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"breakpointResolver.js","sourceRoot":"","sources":["../../src/utils/breakpointResolver.ts"],"names":[],"mappings":"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","sourcesContent":["import { logger } from './logger';\n\nexport interface Breakpoint {\n  name: string;\n  minWidth: number;\n}\n\nexport interface MediaQueryInfo {\n  type: 'min-width' | 'max-width' | 'unsupported';\n  value?: number;\n  raw: string;\n}\n\nconst DEFAULT_BREAKPOINTS: Breakpoint[] = [\n  { name: 'sm', minWidth: 640 },\n  { name: 'md', minWidth: 768 },\n  { name: 'lg', minWidth: 1024 },\n  { name: 'xl', minWidth: 1280 },\n  { name: '2xl', minWidth: 1536 }\n];\n\nlet cachedBreakpoints: Breakpoint[] | null = null;\n\nexport function getDefaultBreakpoints(): Breakpoint[] {\n  return [...DEFAULT_BREAKPOINTS];\n}\n\nexport function resolveBreakpointsFromConfig(screens: Record<string, string | [string, string]> | undefined): Breakpoint[] {\n  if (!screens) {\n    return getDefaultBreakpoints();\n  }\n\n  const breakpoints: Breakpoint[] = [];\n\n  for (const [name, value] of Object.entries(screens)) {\n    let minWidth: number | null = null;\n\n    if (typeof value === 'string') {\n      minWidth = parsePixelValue(value);\n    } else if (Array.isArray(value)) {\n      minWidth = parsePixelValue(value[0]);\n    }\n\n    if (minWidth !== null) {\n      breakpoints.push({ name, minWidth });\n    }\n  }\n\n  breakpoints.sort((a, b) => a.minWidth - b.minWidth);\n\n  return breakpoints.length > 0 ? breakpoints : getDefaultBreakpoints();\n}\n\nfunction parsePixelValue(value: string): number | null {\n  const pxMatch = value.match(/^([\\d.]+)px$/);\n  if (pxMatch) {\n    return parseFloat(pxMatch[1]);\n  }\n\n  const remMatch = value.match(/^([\\d.]+)rem$/);\n  if (remMatch) {\n    return parseFloat(remMatch[1]) * 16;\n  }\n\n  const emMatch = value.match(/^([\\d.]+)em$/);\n  if (emMatch) {\n    return parseFloat(emMatch[1]) * 16;\n  }\n\n  return null;\n}\n\nexport function getBreakpoints(screens?: Record<string, string | [string, string]>): Breakpoint[] {\n  if (cachedBreakpoints && !screens) {\n    return cachedBreakpoints;\n  }\n\n  const breakpoints = screens ? resolveBreakpointsFromConfig(screens) : getDefaultBreakpoints();\n  \n  if (!screens) {\n    cachedBreakpoints = breakpoints;\n  }\n  \n  return breakpoints;\n}\n\nexport function clearBreakpointCache(): void {\n  cachedBreakpoints = null;\n}\n\nexport function parseMediaQuery(params: string): MediaQueryInfo {\n  const trimmed = params.trim().toLowerCase();\n\n  if (trimmed.includes('screen') && trimmed.includes('and')) {\n    return {\n      type: 'unsupported',\n      raw: params\n    };\n  }\n\n  if (trimmed.includes('orientation') || trimmed.includes('print')) {\n    return {\n      type: 'unsupported',\n      raw: params\n    };\n  }\n\n  const minMatch = trimmed.match(/\\(\\s*min-width\\s*:\\s*([\\d.]+)(px|rem|em)\\s*\\)/);\n  if (minMatch) {\n    const value = parseFloat(minMatch[1]);\n    const unit = minMatch[2];\n\n    let pxValue = value;\n    if (unit === 'rem' || unit === 'em') {\n      pxValue = value * 16;\n    }\n\n    return {\n      type: 'min-width',\n      value: pxValue,\n      raw: params\n    };\n  }\n\n  const maxMatch = trimmed.match(/\\(\\s*max-width\\s*:\\s*([\\d.]+)(px|rem|em)\\s*\\)/);\n  if (maxMatch) {\n    return {\n      type: 'max-width',\n      raw: params\n    };\n  }\n\n  return {\n    type: 'unsupported',\n    raw: params\n  };\n}\n\nexport function findBreakpointForMinWidth(minWidth: number, breakpoints: Breakpoint[]): string | null {\n  const exact = breakpoints.find(bp => bp.minWidth === minWidth);\n  if (exact) {\n    return exact.name;\n  }\n\n  const closest = breakpoints.reduce<Breakpoint | null>((closest, bp) => {\n    if (!closest) return bp;\n\n    const currentDiff = Math.abs(bp.minWidth - minWidth);\n    const closestDiff = Math.abs(closest.minWidth - minWidth);\n\n    return currentDiff < closestDiff ? bp : closest;\n  }, null);\n\n  if (closest) {\n    const diff = Math.abs(closest.minWidth - minWidth);\n    const tolerance = minWidth * 0.05;\n\n    if (diff <= tolerance) {\n      logger.verbose(`Matched min-width ${minWidth}px to closest breakpoint ${closest.name} (${closest.minWidth}px)`);\n      return closest.name;\n    }\n  }\n\n  return null;\n}\n\nexport function prefixWithBreakpoint(className: string, breakpoint: string): string {\n  return `${breakpoint}:${className}`;\n}\n\nexport function processMediaQuery(params: string, breakpoints: Breakpoint[]): {\n  breakpoint: string | null;\n  skipped: boolean;\n  reason?: string;\n} {\n  const info = parseMediaQuery(params);\n\n  if (info.type !== 'min-width' || info.value === undefined) {\n    const reason = info.type === 'max-width'\n      ? `Skipped media query (max-width: ...) — unsupported`\n      : `Skipped media query (${info.raw}) — unsupported`;\n\n    logger.verbose(reason);\n    return { breakpoint: null, skipped: true, reason };\n  }\n\n  const breakpoint = findBreakpointForMinWidth(info.value, breakpoints);\n\n  if (!breakpoint) {\n    const reason = `No matching breakpoint for min-width: ${info.value}px`;\n    logger.verbose(reason);\n    return { breakpoint: null, skipped: true, reason };\n  }\n\n  logger.verbose(`Converted media query (min-width: ${info.value}px) → ${breakpoint}`);\n\n  return { breakpoint, skipped: false };\n}"]}
|