css-to-tailwind-react 0.1.0
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/README.md +303 -0
- package/bin/index.js +11 -0
- package/dist/cli.d.ts +1 -0
- package/dist/cli.js +80 -0
- package/dist/cssParser.d.ts +41 -0
- package/dist/cssParser.js +215 -0
- package/dist/fileWriter.d.ts +14 -0
- package/dist/fileWriter.js +128 -0
- package/dist/index.d.ts +8 -0
- package/dist/index.js +21 -0
- package/dist/jsxParser.d.ts +26 -0
- package/dist/jsxParser.js +273 -0
- package/dist/scanner.d.ts +5 -0
- package/dist/scanner.js +55 -0
- package/dist/tailwindMapper.d.ts +35 -0
- package/dist/tailwindMapper.js +428 -0
- package/dist/transformer.d.ts +19 -0
- package/dist/transformer.js +259 -0
- package/dist/utils/config.d.ts +14 -0
- package/dist/utils/config.js +139 -0
- package/dist/utils/logger.d.ts +14 -0
- package/dist/utils/logger.js +56 -0
- package/package.json +73 -0
|
@@ -0,0 +1,273 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
exports.JSXParser = void 0;
|
|
40
|
+
const parser_1 = require("@babel/parser");
|
|
41
|
+
const traverse_1 = __importDefault(require("@babel/traverse"));
|
|
42
|
+
const generator_1 = __importDefault(require("@babel/generator"));
|
|
43
|
+
const t = __importStar(require("@babel/types"));
|
|
44
|
+
const logger_1 = require("./utils/logger");
|
|
45
|
+
class JSXParser {
|
|
46
|
+
constructor(mapper) {
|
|
47
|
+
this.mapper = mapper;
|
|
48
|
+
}
|
|
49
|
+
parse(code, filePath) {
|
|
50
|
+
const transformations = [];
|
|
51
|
+
const warnings = [];
|
|
52
|
+
let hasChanges = false;
|
|
53
|
+
try {
|
|
54
|
+
const ast = (0, parser_1.parse)(code, {
|
|
55
|
+
sourceType: 'module',
|
|
56
|
+
allowImportExportEverywhere: true,
|
|
57
|
+
allowReturnOutsideFunction: true,
|
|
58
|
+
plugins: [
|
|
59
|
+
'jsx',
|
|
60
|
+
'typescript',
|
|
61
|
+
'decorators-legacy',
|
|
62
|
+
'classProperties',
|
|
63
|
+
'optionalChaining',
|
|
64
|
+
'nullishCoalescingOperator',
|
|
65
|
+
'dynamicImport'
|
|
66
|
+
]
|
|
67
|
+
});
|
|
68
|
+
(0, traverse_1.default)(ast, {
|
|
69
|
+
JSXOpeningElement: (path) => {
|
|
70
|
+
const elementName = this.getElementName(path.node);
|
|
71
|
+
// Find style attribute
|
|
72
|
+
const styleAttrIndex = path.node.attributes.findIndex(attr => t.isJSXAttribute(attr) &&
|
|
73
|
+
t.isJSXIdentifier(attr.name) &&
|
|
74
|
+
attr.name.name === 'style');
|
|
75
|
+
if (styleAttrIndex === -1) {
|
|
76
|
+
return;
|
|
77
|
+
}
|
|
78
|
+
const styleAttr = path.node.attributes[styleAttrIndex];
|
|
79
|
+
// Check if style value is dynamic
|
|
80
|
+
if (!this.isStaticStyle(styleAttr)) {
|
|
81
|
+
warnings.push(`Skipped dynamic style in ${elementName} (line ${path.node.loc?.start.line})`);
|
|
82
|
+
logger_1.logger.warn(`Dynamic style detected in ${filePath} - skipping`);
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
// Extract CSS properties from style
|
|
86
|
+
const cssProperties = this.extractCSSProperties(styleAttr);
|
|
87
|
+
if (cssProperties.length === 0) {
|
|
88
|
+
return;
|
|
89
|
+
}
|
|
90
|
+
// Convert to Tailwind classes
|
|
91
|
+
const { classes, warnings: conversionWarnings } = this.mapper.convertMultiple(cssProperties);
|
|
92
|
+
if (classes.length === 0) {
|
|
93
|
+
warnings.push(...conversionWarnings);
|
|
94
|
+
return;
|
|
95
|
+
}
|
|
96
|
+
// Find existing className
|
|
97
|
+
const classNameAttrIndex = path.node.attributes.findIndex(attr => t.isJSXAttribute(attr) &&
|
|
98
|
+
t.isJSXIdentifier(attr.name) &&
|
|
99
|
+
attr.name.name === 'className');
|
|
100
|
+
const originalStyle = (0, generator_1.default)(styleAttr).code;
|
|
101
|
+
if (classNameAttrIndex !== -1) {
|
|
102
|
+
// Merge with existing className
|
|
103
|
+
const classNameAttr = path.node.attributes[classNameAttrIndex];
|
|
104
|
+
if (!this.isStaticClassName(classNameAttr)) {
|
|
105
|
+
warnings.push(`Skipped dynamic className in ${elementName} (line ${path.node.loc?.start.line})`);
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
108
|
+
const existingClasses = this.extractClassNameValue(classNameAttr);
|
|
109
|
+
const mergedClasses = this.mergeClasses(existingClasses, classes);
|
|
110
|
+
// Update className attribute
|
|
111
|
+
classNameAttr.value = t.stringLiteral(mergedClasses);
|
|
112
|
+
}
|
|
113
|
+
else {
|
|
114
|
+
// Create new className attribute
|
|
115
|
+
const newClassNameAttr = t.jsxAttribute(t.jsxIdentifier('className'), t.stringLiteral(classes.join(' ')));
|
|
116
|
+
path.node.attributes.push(newClassNameAttr);
|
|
117
|
+
}
|
|
118
|
+
// Remove style attribute
|
|
119
|
+
path.node.attributes.splice(styleAttrIndex, 1);
|
|
120
|
+
// Record transformation
|
|
121
|
+
const newCode = (0, generator_1.default)(path.node).code;
|
|
122
|
+
transformations.push({
|
|
123
|
+
original: originalStyle,
|
|
124
|
+
converted: newCode,
|
|
125
|
+
classes,
|
|
126
|
+
warnings: conversionWarnings
|
|
127
|
+
});
|
|
128
|
+
hasChanges = true;
|
|
129
|
+
warnings.push(...conversionWarnings);
|
|
130
|
+
}
|
|
131
|
+
});
|
|
132
|
+
// Generate new code
|
|
133
|
+
const output = (0, generator_1.default)(ast, {
|
|
134
|
+
retainLines: true,
|
|
135
|
+
retainFunctionParens: true,
|
|
136
|
+
comments: true
|
|
137
|
+
});
|
|
138
|
+
return {
|
|
139
|
+
code: output.code,
|
|
140
|
+
hasChanges,
|
|
141
|
+
transformations,
|
|
142
|
+
warnings
|
|
143
|
+
};
|
|
144
|
+
}
|
|
145
|
+
catch (error) {
|
|
146
|
+
logger_1.logger.error(`Failed to parse ${filePath}:`, error);
|
|
147
|
+
throw new Error(`Parsing failed: ${error}`);
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
getElementName(node) {
|
|
151
|
+
if (t.isJSXIdentifier(node.name)) {
|
|
152
|
+
return node.name.name;
|
|
153
|
+
}
|
|
154
|
+
if (t.isJSXMemberExpression(node.name)) {
|
|
155
|
+
return this.getMemberExpressionName(node.name);
|
|
156
|
+
}
|
|
157
|
+
return 'unknown';
|
|
158
|
+
}
|
|
159
|
+
getMemberExpressionName(node) {
|
|
160
|
+
if (t.isJSXIdentifier(node.object) && t.isJSXIdentifier(node.property)) {
|
|
161
|
+
return `${node.object.name}.${node.property.name}`;
|
|
162
|
+
}
|
|
163
|
+
return 'unknown';
|
|
164
|
+
}
|
|
165
|
+
isStaticStyle(attr) {
|
|
166
|
+
if (!attr.value) {
|
|
167
|
+
return false;
|
|
168
|
+
}
|
|
169
|
+
// Handle style={{ ... }} (object expression)
|
|
170
|
+
if (t.isJSXExpressionContainer(attr.value)) {
|
|
171
|
+
const expression = attr.value.expression;
|
|
172
|
+
// Direct object: style={{ color: 'red' }}
|
|
173
|
+
if (t.isObjectExpression(expression)) {
|
|
174
|
+
return true;
|
|
175
|
+
}
|
|
176
|
+
// Check if it's a simple object without variables
|
|
177
|
+
if (t.isIdentifier(expression) || t.isMemberExpression(expression)) {
|
|
178
|
+
return false; // Variable reference - dynamic
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
return false;
|
|
182
|
+
}
|
|
183
|
+
isStaticClassName(attr) {
|
|
184
|
+
if (!attr.value) {
|
|
185
|
+
return true;
|
|
186
|
+
}
|
|
187
|
+
if (t.isStringLiteral(attr.value)) {
|
|
188
|
+
return true;
|
|
189
|
+
}
|
|
190
|
+
if (t.isJSXExpressionContainer(attr.value)) {
|
|
191
|
+
const expression = attr.value.expression;
|
|
192
|
+
// String literal in expression: className={"container"}
|
|
193
|
+
if (t.isStringLiteral(expression)) {
|
|
194
|
+
return true;
|
|
195
|
+
}
|
|
196
|
+
// Template literal without expressions: className={`container`}
|
|
197
|
+
if (t.isTemplateLiteral(expression) && expression.expressions.length === 0) {
|
|
198
|
+
return true;
|
|
199
|
+
}
|
|
200
|
+
// Anything else is dynamic
|
|
201
|
+
return false;
|
|
202
|
+
}
|
|
203
|
+
return false;
|
|
204
|
+
}
|
|
205
|
+
extractCSSProperties(attr) {
|
|
206
|
+
const properties = [];
|
|
207
|
+
if (!attr.value || !t.isJSXExpressionContainer(attr.value)) {
|
|
208
|
+
return properties;
|
|
209
|
+
}
|
|
210
|
+
const expression = attr.value.expression;
|
|
211
|
+
if (!t.isObjectExpression(expression)) {
|
|
212
|
+
return properties;
|
|
213
|
+
}
|
|
214
|
+
expression.properties.forEach(prop => {
|
|
215
|
+
if (!t.isObjectProperty(prop)) {
|
|
216
|
+
return;
|
|
217
|
+
}
|
|
218
|
+
let propertyName = null;
|
|
219
|
+
let propertyValue = null;
|
|
220
|
+
// Get property name
|
|
221
|
+
if (t.isIdentifier(prop.key)) {
|
|
222
|
+
propertyName = prop.key.name;
|
|
223
|
+
}
|
|
224
|
+
else if (t.isStringLiteral(prop.key)) {
|
|
225
|
+
propertyName = prop.key.value;
|
|
226
|
+
}
|
|
227
|
+
// Get property value
|
|
228
|
+
if (t.isStringLiteral(prop.value)) {
|
|
229
|
+
propertyValue = prop.value.value;
|
|
230
|
+
}
|
|
231
|
+
else if (t.isNumericLiteral(prop.value)) {
|
|
232
|
+
propertyValue = `${prop.value.value}px`;
|
|
233
|
+
}
|
|
234
|
+
// Handle camelCase to kebab-case conversion for CSS properties
|
|
235
|
+
if (propertyName && propertyValue) {
|
|
236
|
+
const cssProperty = this.camelToKebab(propertyName);
|
|
237
|
+
properties.push({
|
|
238
|
+
property: cssProperty,
|
|
239
|
+
value: propertyValue
|
|
240
|
+
});
|
|
241
|
+
}
|
|
242
|
+
});
|
|
243
|
+
return properties;
|
|
244
|
+
}
|
|
245
|
+
extractClassNameValue(attr) {
|
|
246
|
+
if (!attr.value) {
|
|
247
|
+
return '';
|
|
248
|
+
}
|
|
249
|
+
if (t.isStringLiteral(attr.value)) {
|
|
250
|
+
return attr.value.value;
|
|
251
|
+
}
|
|
252
|
+
if (t.isJSXExpressionContainer(attr.value)) {
|
|
253
|
+
const expression = attr.value.expression;
|
|
254
|
+
if (t.isStringLiteral(expression)) {
|
|
255
|
+
return expression.value;
|
|
256
|
+
}
|
|
257
|
+
if (t.isTemplateLiteral(expression) && expression.quasis.length > 0) {
|
|
258
|
+
return expression.quasis[0].value.raw;
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
return '';
|
|
262
|
+
}
|
|
263
|
+
mergeClasses(existing, newClasses) {
|
|
264
|
+
const existingSet = new Set(existing.split(/\s+/).filter(Boolean));
|
|
265
|
+
newClasses.forEach(cls => existingSet.add(cls));
|
|
266
|
+
return Array.from(existingSet).join(' ');
|
|
267
|
+
}
|
|
268
|
+
camelToKebab(str) {
|
|
269
|
+
return str.replace(/([a-z0-9])([A-Z])/g, '$1-$2').toLowerCase();
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
exports.JSXParser = JSXParser;
|
|
273
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"jsxParser.js","sourceRoot":"","sources":["../src/jsxParser.ts"],"names":[],"mappings":"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","sourcesContent":["import { parse } from '@babel/parser';\nimport traverse from '@babel/traverse';\nimport generate from '@babel/generator';\nimport * as t from '@babel/types';\nimport { TailwindMapper, CSSProperty } from './tailwindMapper';\nimport { logger } from './utils/logger';\n\nexport interface JSXTransformation {\n  original: string;\n  converted: string;\n  classes: string[];\n  warnings: string[];\n}\n\nexport interface JSXParseResult {\n  code: string;\n  hasChanges: boolean;\n  transformations: JSXTransformation[];\n  warnings: string[];\n}\n\nexport class JSXParser {\n  private mapper: TailwindMapper;\n\n  constructor(mapper: TailwindMapper) {\n    this.mapper = mapper;\n  }\n\n  parse(code: string, filePath: string): JSXParseResult {\n    const transformations: JSXTransformation[] = [];\n    const warnings: string[] = [];\n    let hasChanges = false;\n\n    try {\n      const ast = parse(code, {\n        sourceType: 'module',\n        allowImportExportEverywhere: true,\n        allowReturnOutsideFunction: true,\n        plugins: [\n          'jsx',\n          'typescript',\n          'decorators-legacy',\n          'classProperties',\n          'optionalChaining',\n          'nullishCoalescingOperator',\n          'dynamicImport'\n        ]\n      });\n\n      traverse(ast, {\n        JSXOpeningElement: (path) => {\n          const elementName = this.getElementName(path.node);\n          \n          // Find style attribute\n          const styleAttrIndex = path.node.attributes.findIndex(\n            attr => t.isJSXAttribute(attr) && \n                    t.isJSXIdentifier(attr.name) && \n                    attr.name.name === 'style'\n          );\n\n          if (styleAttrIndex === -1) {\n            return;\n          }\n\n          const styleAttr = path.node.attributes[styleAttrIndex] as t.JSXAttribute;\n          \n          // Check if style value is dynamic\n          if (!this.isStaticStyle(styleAttr)) {\n            warnings.push(`Skipped dynamic style in ${elementName} (line ${path.node.loc?.start.line})`);\n            logger.warn(`Dynamic style detected in ${filePath} - skipping`);\n            return;\n          }\n\n          // Extract CSS properties from style\n          const cssProperties = this.extractCSSProperties(styleAttr);\n          \n          if (cssProperties.length === 0) {\n            return;\n          }\n\n          // Convert to Tailwind classes\n          const { classes, warnings: conversionWarnings } = this.mapper.convertMultiple(cssProperties);\n\n          if (classes.length === 0) {\n            warnings.push(...conversionWarnings);\n            return;\n          }\n\n          // Find existing className\n          const classNameAttrIndex = path.node.attributes.findIndex(\n            attr => t.isJSXAttribute(attr) && \n                    t.isJSXIdentifier(attr.name) && \n                    attr.name.name === 'className'\n          );\n\n          const originalStyle = generate(styleAttr).code;\n\n          if (classNameAttrIndex !== -1) {\n            // Merge with existing className\n            const classNameAttr = path.node.attributes[classNameAttrIndex] as t.JSXAttribute;\n            \n            if (!this.isStaticClassName(classNameAttr)) {\n              warnings.push(`Skipped dynamic className in ${elementName} (line ${path.node.loc?.start.line})`);\n              return;\n            }\n\n            const existingClasses = this.extractClassNameValue(classNameAttr);\n            const mergedClasses = this.mergeClasses(existingClasses, classes);\n\n            // Update className attribute\n            classNameAttr.value = t.stringLiteral(mergedClasses);\n          } else {\n            // Create new className attribute\n            const newClassNameAttr = t.jsxAttribute(\n              t.jsxIdentifier('className'),\n              t.stringLiteral(classes.join(' '))\n            );\n            path.node.attributes.push(newClassNameAttr);\n          }\n\n          // Remove style attribute\n          path.node.attributes.splice(styleAttrIndex, 1);\n\n          // Record transformation\n          const newCode = generate(path.node).code;\n          transformations.push({\n            original: originalStyle,\n            converted: newCode,\n            classes,\n            warnings: conversionWarnings\n          });\n\n          hasChanges = true;\n          warnings.push(...conversionWarnings);\n        }\n      });\n\n      // Generate new code\n      const output = generate(ast, {\n        retainLines: true,\n        retainFunctionParens: true,\n        comments: true\n      });\n\n      return {\n        code: output.code,\n        hasChanges,\n        transformations,\n        warnings\n      };\n\n    } catch (error) {\n      logger.error(`Failed to parse ${filePath}:`, error);\n      throw new Error(`Parsing failed: ${error}`);\n    }\n  }\n\n  private getElementName(node: t.JSXOpeningElement): string {\n    if (t.isJSXIdentifier(node.name)) {\n      return node.name.name;\n    }\n    if (t.isJSXMemberExpression(node.name)) {\n      return this.getMemberExpressionName(node.name);\n    }\n    return 'unknown';\n  }\n\n  private getMemberExpressionName(node: t.JSXMemberExpression): string {\n    if (t.isJSXIdentifier(node.object) && t.isJSXIdentifier(node.property)) {\n      return `${node.object.name}.${node.property.name}`;\n    }\n    return 'unknown';\n  }\n\n  private isStaticStyle(attr: t.JSXAttribute): boolean {\n    if (!attr.value) {\n      return false;\n    }\n\n    // Handle style={{ ... }} (object expression)\n    if (t.isJSXExpressionContainer(attr.value)) {\n      const expression = attr.value.expression;\n      \n      // Direct object: style={{ color: 'red' }}\n      if (t.isObjectExpression(expression)) {\n        return true;\n      }\n\n      // Check if it's a simple object without variables\n      if (t.isIdentifier(expression) || t.isMemberExpression(expression)) {\n        return false; // Variable reference - dynamic\n      }\n    }\n\n    return false;\n  }\n\n  private isStaticClassName(attr: t.JSXAttribute): boolean {\n    if (!attr.value) {\n      return true;\n    }\n\n    if (t.isStringLiteral(attr.value)) {\n      return true;\n    }\n\n    if (t.isJSXExpressionContainer(attr.value)) {\n      const expression = attr.value.expression;\n      \n      // String literal in expression: className={\"container\"}\n      if (t.isStringLiteral(expression)) {\n        return true;\n      }\n\n      // Template literal without expressions: className={`container`}\n      if (t.isTemplateLiteral(expression) && expression.expressions.length === 0) {\n        return true;\n      }\n\n      // Anything else is dynamic\n      return false;\n    }\n\n    return false;\n  }\n\n  private extractCSSProperties(attr: t.JSXAttribute): CSSProperty[] {\n    const properties: CSSProperty[] = [];\n\n    if (!attr.value || !t.isJSXExpressionContainer(attr.value)) {\n      return properties;\n    }\n\n    const expression = attr.value.expression;\n\n    if (!t.isObjectExpression(expression)) {\n      return properties;\n    }\n\n    expression.properties.forEach(prop => {\n      if (!t.isObjectProperty(prop)) {\n        return;\n      }\n\n      let propertyName: string | null = null;\n      let propertyValue: string | null = null;\n\n      // Get property name\n      if (t.isIdentifier(prop.key)) {\n        propertyName = prop.key.name;\n      } else if (t.isStringLiteral(prop.key)) {\n        propertyName = prop.key.value;\n      }\n\n      // Get property value\n      if (t.isStringLiteral(prop.value)) {\n        propertyValue = prop.value.value;\n      } else if (t.isNumericLiteral(prop.value)) {\n        propertyValue = `${prop.value.value}px`;\n      }\n\n      // Handle camelCase to kebab-case conversion for CSS properties\n      if (propertyName && propertyValue) {\n        const cssProperty = this.camelToKebab(propertyName);\n        properties.push({\n          property: cssProperty,\n          value: propertyValue\n        });\n      }\n    });\n\n    return properties;\n  }\n\n  private extractClassNameValue(attr: t.JSXAttribute): string {\n    if (!attr.value) {\n      return '';\n    }\n\n    if (t.isStringLiteral(attr.value)) {\n      return attr.value.value;\n    }\n\n    if (t.isJSXExpressionContainer(attr.value)) {\n      const expression = attr.value.expression;\n      \n      if (t.isStringLiteral(expression)) {\n        return expression.value;\n      }\n\n      if (t.isTemplateLiteral(expression) && expression.quasis.length > 0) {\n        return expression.quasis[0].value.raw;\n      }\n    }\n\n    return '';\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  private camelToKebab(str: string): string {\n    return str.replace(/([a-z0-9])([A-Z])/g, '$1-$2').toLowerCase();\n  }\n}\n"]}
|
package/dist/scanner.js
ADDED
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.scanProject = scanProject;
|
|
7
|
+
const fast_glob_1 = __importDefault(require("fast-glob"));
|
|
8
|
+
const path_1 = __importDefault(require("path"));
|
|
9
|
+
async function scanProject(directory) {
|
|
10
|
+
const absoluteDir = path_1.default.resolve(directory);
|
|
11
|
+
// Patterns for React components
|
|
12
|
+
const jsxPatterns = [
|
|
13
|
+
'**/*.{js,jsx,ts,tsx}',
|
|
14
|
+
];
|
|
15
|
+
// Patterns for CSS files
|
|
16
|
+
const cssPatterns = [
|
|
17
|
+
'**/*.css',
|
|
18
|
+
];
|
|
19
|
+
// Ignore patterns
|
|
20
|
+
const ignorePatterns = [
|
|
21
|
+
'**/node_modules/**',
|
|
22
|
+
'**/.next/**',
|
|
23
|
+
'**/dist/**',
|
|
24
|
+
'**/build/**',
|
|
25
|
+
'**/.git/**',
|
|
26
|
+
'**/coverage/**',
|
|
27
|
+
'**/*.d.ts',
|
|
28
|
+
'**/.*' // hidden files
|
|
29
|
+
];
|
|
30
|
+
try {
|
|
31
|
+
// Scan for JSX/TSX files
|
|
32
|
+
const jsxFiles = await (0, fast_glob_1.default)(jsxPatterns, {
|
|
33
|
+
cwd: absoluteDir,
|
|
34
|
+
ignore: ignorePatterns,
|
|
35
|
+
absolute: true,
|
|
36
|
+
onlyFiles: true
|
|
37
|
+
});
|
|
38
|
+
// Scan for CSS files
|
|
39
|
+
const cssFiles = await (0, fast_glob_1.default)(cssPatterns, {
|
|
40
|
+
cwd: absoluteDir,
|
|
41
|
+
ignore: ignorePatterns,
|
|
42
|
+
absolute: true,
|
|
43
|
+
onlyFiles: true
|
|
44
|
+
});
|
|
45
|
+
const files = [
|
|
46
|
+
...jsxFiles.map(file => ({ path: file, type: 'jsx' })),
|
|
47
|
+
...cssFiles.map(file => ({ path: file, type: 'css' }))
|
|
48
|
+
];
|
|
49
|
+
return files.sort((a, b) => a.path.localeCompare(b.path));
|
|
50
|
+
}
|
|
51
|
+
catch (error) {
|
|
52
|
+
throw new Error(`Failed to scan directory: ${error}`);
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2Nhbm5lci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9zY2FubmVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBUUEsa0NBbURDO0FBM0RELDBEQUE2QjtBQUM3QixnREFBd0I7QUFPakIsS0FBSyxVQUFVLFdBQVcsQ0FBQyxTQUFpQjtJQUNqRCxNQUFNLFdBQVcsR0FBRyxjQUFJLENBQUMsT0FBTyxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBRTVDLGdDQUFnQztJQUNoQyxNQUFNLFdBQVcsR0FBRztRQUNsQixzQkFBc0I7S0FDdkIsQ0FBQztJQUVGLHlCQUF5QjtJQUN6QixNQUFNLFdBQVcsR0FBRztRQUNsQixVQUFVO0tBQ1gsQ0FBQztJQUVGLGtCQUFrQjtJQUNsQixNQUFNLGNBQWMsR0FBRztRQUNyQixvQkFBb0I7UUFDcEIsYUFBYTtRQUNiLFlBQVk7UUFDWixhQUFhO1FBQ2IsWUFBWTtRQUNaLGdCQUFnQjtRQUNoQixXQUFXO1FBQ1gsT0FBTyxDQUFDLGVBQWU7S0FDeEIsQ0FBQztJQUVGLElBQUksQ0FBQztRQUNILHlCQUF5QjtRQUN6QixNQUFNLFFBQVEsR0FBRyxNQUFNLElBQUEsbUJBQUksRUFBQyxXQUFXLEVBQUU7WUFDdkMsR0FBRyxFQUFFLFdBQVc7WUFDaEIsTUFBTSxFQUFFLGNBQWM7WUFDdEIsUUFBUSxFQUFFLElBQUk7WUFDZCxTQUFTLEVBQUUsSUFBSTtTQUNoQixDQUFDLENBQUM7UUFFSCxxQkFBcUI7UUFDckIsTUFBTSxRQUFRLEdBQUcsTUFBTSxJQUFBLG1CQUFJLEVBQUMsV0FBVyxFQUFFO1lBQ3ZDLEdBQUcsRUFBRSxXQUFXO1lBQ2hCLE1BQU0sRUFBRSxjQUFjO1lBQ3RCLFFBQVEsRUFBRSxJQUFJO1lBQ2QsU0FBUyxFQUFFLElBQUk7U0FDaEIsQ0FBQyxDQUFDO1FBRUgsTUFBTSxLQUFLLEdBQWtCO1lBQzNCLEdBQUcsUUFBUSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLENBQUMsRUFBRSxJQUFJLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxLQUFjLEVBQUUsQ0FBQyxDQUFDO1lBQy9ELEdBQUcsUUFBUSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLENBQUMsRUFBRSxJQUFJLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxLQUFjLEVBQUUsQ0FBQyxDQUFDO1NBQ2hFLENBQUM7UUFFRixPQUFPLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQztJQUM1RCxDQUFDO0lBQUMsT0FBTyxLQUFLLEVBQUUsQ0FBQztRQUNmLE1BQU0sSUFBSSxLQUFLLENBQUMsNkJBQTZCLEtBQUssRUFBRSxDQUFDLENBQUM7SUFDeEQsQ0FBQztBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgZ2xvYiBmcm9tICdmYXN0LWdsb2InO1xuaW1wb3J0IHBhdGggZnJvbSAncGF0aCc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2Nhbm5lZEZpbGUge1xuICBwYXRoOiBzdHJpbmc7XG4gIHR5cGU6ICdqc3gnIHwgJ2Nzcyc7XG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBzY2FuUHJvamVjdChkaXJlY3Rvcnk6IHN0cmluZyk6IFByb21pc2U8U2Nhbm5lZEZpbGVbXT4ge1xuICBjb25zdCBhYnNvbHV0ZURpciA9IHBhdGgucmVzb2x2ZShkaXJlY3RvcnkpO1xuICBcbiAgLy8gUGF0dGVybnMgZm9yIFJlYWN0IGNvbXBvbmVudHNcbiAgY29uc3QganN4UGF0dGVybnMgPSBbXG4gICAgJyoqLyoue2pzLGpzeCx0cyx0c3h9JyxcbiAgXTtcbiAgXG4gIC8vIFBhdHRlcm5zIGZvciBDU1MgZmlsZXNcbiAgY29uc3QgY3NzUGF0dGVybnMgPSBbXG4gICAgJyoqLyouY3NzJyxcbiAgXTtcbiAgXG4gIC8vIElnbm9yZSBwYXR0ZXJuc1xuICBjb25zdCBpZ25vcmVQYXR0ZXJucyA9IFtcbiAgICAnKiovbm9kZV9tb2R1bGVzLyoqJyxcbiAgICAnKiovLm5leHQvKionLFxuICAgICcqKi9kaXN0LyoqJyxcbiAgICAnKiovYnVpbGQvKionLFxuICAgICcqKi8uZ2l0LyoqJyxcbiAgICAnKiovY292ZXJhZ2UvKionLFxuICAgICcqKi8qLmQudHMnLFxuICAgICcqKi8uKicgLy8gaGlkZGVuIGZpbGVzXG4gIF07XG5cbiAgdHJ5IHtcbiAgICAvLyBTY2FuIGZvciBKU1gvVFNYIGZpbGVzXG4gICAgY29uc3QganN4RmlsZXMgPSBhd2FpdCBnbG9iKGpzeFBhdHRlcm5zLCB7XG4gICAgICBjd2Q6IGFic29sdXRlRGlyLFxuICAgICAgaWdub3JlOiBpZ25vcmVQYXR0ZXJucyxcbiAgICAgIGFic29sdXRlOiB0cnVlLFxuICAgICAgb25seUZpbGVzOiB0cnVlXG4gICAgfSk7XG5cbiAgICAvLyBTY2FuIGZvciBDU1MgZmlsZXNcbiAgICBjb25zdCBjc3NGaWxlcyA9IGF3YWl0IGdsb2IoY3NzUGF0dGVybnMsIHtcbiAgICAgIGN3ZDogYWJzb2x1dGVEaXIsXG4gICAgICBpZ25vcmU6IGlnbm9yZVBhdHRlcm5zLFxuICAgICAgYWJzb2x1dGU6IHRydWUsXG4gICAgICBvbmx5RmlsZXM6IHRydWVcbiAgICB9KTtcblxuICAgIGNvbnN0IGZpbGVzOiBTY2FubmVkRmlsZVtdID0gW1xuICAgICAgLi4uanN4RmlsZXMubWFwKGZpbGUgPT4gKHsgcGF0aDogZmlsZSwgdHlwZTogJ2pzeCcgYXMgY29uc3QgfSkpLFxuICAgICAgLi4uY3NzRmlsZXMubWFwKGZpbGUgPT4gKHsgcGF0aDogZmlsZSwgdHlwZTogJ2NzcycgYXMgY29uc3QgfSkpXG4gICAgXTtcblxuICAgIHJldHVybiBmaWxlcy5zb3J0KChhLCBiKSA9PiBhLnBhdGgubG9jYWxlQ29tcGFyZShiLnBhdGgpKTtcbiAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoYEZhaWxlZCB0byBzY2FuIGRpcmVjdG9yeTogJHtlcnJvcn1gKTtcbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { TailwindConfig } from './utils/config';
|
|
2
|
+
export interface CSSProperty {
|
|
3
|
+
property: string;
|
|
4
|
+
value: string;
|
|
5
|
+
}
|
|
6
|
+
export interface ConversionResult {
|
|
7
|
+
className: string | null;
|
|
8
|
+
skipped: boolean;
|
|
9
|
+
reason?: string;
|
|
10
|
+
}
|
|
11
|
+
export declare class TailwindMapper {
|
|
12
|
+
private config;
|
|
13
|
+
private spacingScale;
|
|
14
|
+
constructor(config: TailwindConfig);
|
|
15
|
+
private buildSpacingScale;
|
|
16
|
+
private pxToSpacing;
|
|
17
|
+
private extractPx;
|
|
18
|
+
convertProperty(property: string, value: string): ConversionResult;
|
|
19
|
+
private convertDisplay;
|
|
20
|
+
private convertMargin;
|
|
21
|
+
private convertPadding;
|
|
22
|
+
private convertFontWeight;
|
|
23
|
+
private convertFontSize;
|
|
24
|
+
private convertFlexbox;
|
|
25
|
+
private convertGap;
|
|
26
|
+
private convertWidth;
|
|
27
|
+
private convertHeight;
|
|
28
|
+
private convertBackgroundColor;
|
|
29
|
+
private convertTextColor;
|
|
30
|
+
private convertBorderRadius;
|
|
31
|
+
convertMultiple(properties: CSSProperty[]): {
|
|
32
|
+
classes: string[];
|
|
33
|
+
warnings: string[];
|
|
34
|
+
};
|
|
35
|
+
}
|