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.
@@ -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"]}
@@ -0,0 +1,5 @@
1
+ export interface ScannedFile {
2
+ path: string;
3
+ type: 'jsx' | 'css';
4
+ }
5
+ export declare function scanProject(directory: string): Promise<ScannedFile[]>;
@@ -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
+ }