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,259 @@
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.transformFiles = transformFiles;
7
+ const fs_1 = __importDefault(require("fs"));
8
+ const path_1 = __importDefault(require("path"));
9
+ const tailwindMapper_1 = require("./tailwindMapper");
10
+ const jsxParser_1 = require("./jsxParser");
11
+ const cssParser_1 = require("./cssParser");
12
+ const fileWriter_1 = require("./fileWriter");
13
+ const logger_1 = require("./utils/logger");
14
+ async function transformFiles(files, options) {
15
+ const results = {
16
+ filesScanned: files.length,
17
+ filesModified: 0,
18
+ stylesConverted: 0,
19
+ classesReplaced: 0,
20
+ warnings: 0
21
+ };
22
+ const mapper = new tailwindMapper_1.TailwindMapper(options.tailwindConfig || {});
23
+ const jsxParser = new jsxParser_1.JSXParser(mapper);
24
+ const cssParser = new cssParser_1.CSSParser(mapper);
25
+ const fileWriter = new fileWriter_1.FileWriter({ dryRun: options.dryRun });
26
+ // PASS 1: Analyze all files WITHOUT modifying anything
27
+ // Collect CSS mappings and gather info about what can be safely converted
28
+ const cssClassMap = {};
29
+ const cssFileResults = new Map();
30
+ logger_1.logger.info('\n🔍 Phase 1: Analyzing files...');
31
+ // Analyze CSS files
32
+ if (!options.skipExternal) {
33
+ for (const file of files.filter(f => f.type === 'css')) {
34
+ try {
35
+ const content = fs_1.default.readFileSync(file.path, 'utf-8');
36
+ const result = await cssParser.parse(content, file.path);
37
+ // Check if ALL rules in this file are FULLY converted (all declarations)
38
+ const totalRules = result.rules.length;
39
+ const fullyConvertedRules = result.rules.filter(r => r.fullyConverted).length;
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
+ const fullyConvertible = totalRules > 0 && totalRules === fullyConvertedRules && partiallyConvertedRules === 0;
43
+ // Build class map (only for fully converted classes - partial conversions keep the CSS)
44
+ result.rules.forEach(rule => {
45
+ if (rule.fullyConverted) {
46
+ cssClassMap[rule.className] = {
47
+ tailwindClasses: rule.convertedClasses,
48
+ sourceFile: file.path,
49
+ fullyConvertible: true
50
+ };
51
+ results.stylesConverted += rule.declarations.length;
52
+ }
53
+ else if (rule.partialConversion) {
54
+ // For partial conversions, we converted some declarations but keep the CSS rule
55
+ // Count the converted declarations
56
+ results.stylesConverted += rule.convertedClasses.length;
57
+ logger_1.logger.verbose(` Rule .${rule.className}: partial conversion (${rule.convertedClasses.length}/${rule.declarations.length} declarations)`);
58
+ }
59
+ });
60
+ cssFileResults.set(file.path, {
61
+ content,
62
+ newContent: result.css,
63
+ rules: result.rules,
64
+ canDelete: result.canDelete,
65
+ hasChanges: result.hasChanges,
66
+ fullyConvertible
67
+ });
68
+ results.warnings += result.warnings.length;
69
+ // Log analysis
70
+ logger_1.logger.verbose(`Analyzed ${file.path}:`);
71
+ logger_1.logger.verbose(` - Total rules: ${totalRules}`);
72
+ logger_1.logger.verbose(` - Fully converted rules: ${fullyConvertedRules}`);
73
+ logger_1.logger.verbose(` - Partially converted rules: ${partiallyConvertedRules}`);
74
+ logger_1.logger.verbose(` - Fully convertible: ${fullyConvertible}`);
75
+ // Log warnings
76
+ result.warnings.forEach(warning => {
77
+ logger_1.logger.verbose(`⚠️ ${file.path}: ${warning}`);
78
+ });
79
+ }
80
+ catch (error) {
81
+ logger_1.logger.error(`Failed to analyze ${file.path}:`, error);
82
+ results.warnings++;
83
+ }
84
+ }
85
+ }
86
+ // PASS 2: Transform JSX/TSX files
87
+ logger_1.logger.info('\n⚛️ Phase 2: Transforming React components...');
88
+ const jsxFileResults = new Map();
89
+ for (const file of files.filter(f => f.type === 'jsx')) {
90
+ try {
91
+ let content = fs_1.default.readFileSync(file.path, 'utf-8');
92
+ const originalContent = content;
93
+ let hasChanges = false;
94
+ let fileWarnings = [];
95
+ // Process inline styles
96
+ if (!options.skipInline) {
97
+ try {
98
+ const jsxResult = jsxParser.parse(content, file.path);
99
+ if (jsxResult.hasChanges) {
100
+ content = jsxResult.code;
101
+ hasChanges = true;
102
+ results.stylesConverted += jsxResult.transformations.reduce((sum, t) => sum + t.classes.length, 0);
103
+ fileWarnings.push(...jsxResult.warnings);
104
+ }
105
+ }
106
+ catch (error) {
107
+ logger_1.logger.warn(`Failed to parse JSX inline styles in ${file.path}: ${error}`);
108
+ fileWarnings.push(`JSX parse error: ${error}`);
109
+ }
110
+ }
111
+ // Process internal CSS
112
+ if (!options.skipInternal) {
113
+ try {
114
+ const internalResult = await cssParser.parseInternalCSS(content, file.path);
115
+ if (internalResult.hasChanges) {
116
+ content = internalResult.html;
117
+ hasChanges = true;
118
+ // Build class map from internal styles
119
+ internalResult.rules.forEach(rule => {
120
+ if (rule.convertedClasses.length > 0) {
121
+ cssClassMap[rule.className] = {
122
+ tailwindClasses: rule.convertedClasses,
123
+ sourceFile: file.path,
124
+ fullyConvertible: true
125
+ };
126
+ results.stylesConverted += rule.declarations.length;
127
+ }
128
+ });
129
+ fileWarnings.push(...internalResult.warnings);
130
+ }
131
+ }
132
+ catch (error) {
133
+ logger_1.logger.warn(`Failed to parse internal CSS in ${file.path}: ${error}`);
134
+ fileWarnings.push(`Internal CSS parse error: ${error}`);
135
+ }
136
+ }
137
+ jsxFileResults.set(file.path, {
138
+ content: originalContent,
139
+ newContent: content,
140
+ hasChanges
141
+ });
142
+ results.warnings += fileWarnings.length;
143
+ // Log warnings
144
+ fileWarnings.forEach(warning => {
145
+ logger_1.logger.verbose(`⚠️ ${file.path}: ${warning}`);
146
+ });
147
+ }
148
+ catch (error) {
149
+ logger_1.logger.error(`Failed to process ${file.path}:`, error);
150
+ results.warnings++;
151
+ }
152
+ }
153
+ // PASS 3: Replace className references from external CSS
154
+ // This must happen after all JSX files are parsed
155
+ if (Object.keys(cssClassMap).length > 0) {
156
+ logger_1.logger.info('\n🔄 Phase 3: Replacing className references...');
157
+ for (const [filePath, fileResult] of jsxFileResults) {
158
+ let content = fileResult.newContent;
159
+ let hasChanges = fileResult.hasChanges;
160
+ // Replace className references
161
+ const replacementResult = replaceClassNameReferences(content, cssClassMap);
162
+ if (replacementResult.hasChanges) {
163
+ content = replacementResult.code;
164
+ hasChanges = true;
165
+ results.classesReplaced += replacementResult.replacements;
166
+ logger_1.logger.verbose(`Replaced ${replacementResult.replacements} class references in ${path_1.default.basename(filePath)}`);
167
+ }
168
+ // Update the result
169
+ jsxFileResults.set(filePath, {
170
+ ...fileResult,
171
+ newContent: content,
172
+ hasChanges
173
+ });
174
+ }
175
+ }
176
+ // PASS 4: Write all changes
177
+ logger_1.logger.info('\n💾 Phase 4: Writing changes...');
178
+ // Write JSX files
179
+ for (const [filePath, fileResult] of jsxFileResults) {
180
+ if (fileResult.hasChanges) {
181
+ const success = await fileWriter.writeFile(filePath, fileResult.newContent, fileResult.content);
182
+ if (success) {
183
+ results.filesModified++;
184
+ }
185
+ }
186
+ }
187
+ // Write CSS files (SAFETY: Only modify if fully convertible or explicitly allowed)
188
+ if (!options.skipExternal) {
189
+ for (const [filePath, fileResult] of cssFileResults) {
190
+ if (!fileResult.hasChanges)
191
+ 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
+ if (!fileResult.fullyConvertible) {
195
+ logger_1.logger.warn(`⏭️ Skipping ${path_1.default.basename(filePath)} - not fully convertible (would break styles)`);
196
+ logger_1.logger.warn(` Convertible: ${fileResult.rules.filter(r => r.convertedClasses.length > 0).length}/${fileResult.rules.length} rules`);
197
+ continue;
198
+ }
199
+ // SAFETY RULE 2: Only delete if ALL rules converted AND --delete-css flag used
200
+ if (fileResult.canDelete && options.deleteCss) {
201
+ const success = await fileWriter.deleteFile(filePath);
202
+ if (success) {
203
+ results.filesModified++;
204
+ logger_1.logger.info(`🗑️ Deleted ${path_1.default.basename(filePath)} (all rules converted)`);
205
+ }
206
+ }
207
+ else if (fileResult.canDelete && !options.deleteCss) {
208
+ // File is empty but don't delete without permission
209
+ logger_1.logger.info(`ℹ️ ${path_1.default.basename(filePath)} is now empty (use --delete-css to remove)`);
210
+ }
211
+ else {
212
+ // Write modified CSS (only if fully convertible)
213
+ const success = await fileWriter.writeFile(filePath, fileResult.newContent, fileResult.content);
214
+ if (success) {
215
+ results.filesModified++;
216
+ }
217
+ }
218
+ }
219
+ }
220
+ return results;
221
+ }
222
+ function replaceClassNameReferences(code, classMap) {
223
+ let hasChanges = false;
224
+ let replacements = 0;
225
+ let modifiedCode = code;
226
+ Object.entries(classMap).forEach(([oldClass, info]) => {
227
+ // Skip if not fully convertible
228
+ if (!info.fullyConvertible) {
229
+ return;
230
+ }
231
+ const tailwindClassString = info.tailwindClasses.join(' ');
232
+ // Pattern 1: className="oldClass" (simple string)
233
+ const pattern1 = new RegExp(`className="${oldClass}"`, 'g');
234
+ if (pattern1.test(modifiedCode)) {
235
+ modifiedCode = modifiedCode.replace(pattern1, `className="${tailwindClassString}"`);
236
+ hasChanges = true;
237
+ replacements++;
238
+ logger_1.logger.verbose(`Replaced className="${oldClass}" with "${tailwindClassString}"`);
239
+ }
240
+ // Pattern 2: className={"oldClass"} (expression with string)
241
+ const pattern2 = new RegExp(`className=\\{"${oldClass}"\\}`, 'g');
242
+ if (pattern2.test(modifiedCode)) {
243
+ modifiedCode = modifiedCode.replace(pattern2, `className="${tailwindClassString}"`);
244
+ hasChanges = true;
245
+ replacements++;
246
+ logger_1.logger.verbose(`Replaced className={"${oldClass}"} with "${tailwindClassString}"`);
247
+ }
248
+ // Pattern 3: className={`oldClass`} (simple template literal)
249
+ const pattern3 = new RegExp(`className=\\{\`\\s*${oldClass}\\s*\`\\}`, 'g');
250
+ if (pattern3.test(modifiedCode)) {
251
+ modifiedCode = modifiedCode.replace(pattern3, `className="${tailwindClassString}"`);
252
+ hasChanges = true;
253
+ replacements++;
254
+ logger_1.logger.verbose(`Replaced className={\`${oldClass}\`} with "${tailwindClassString}"`);
255
+ }
256
+ });
257
+ return { code: modifiedCode, hasChanges, replacements };
258
+ }
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"]}
@@ -0,0 +1,14 @@
1
+ export interface TailwindConfig {
2
+ theme?: {
3
+ extend?: Record<string, any>;
4
+ spacing?: Record<string, string>;
5
+ colors?: Record<string, any>;
6
+ fontSize?: Record<string, any>;
7
+ fontWeight?: Record<string, string>;
8
+ borderRadius?: Record<string, string>;
9
+ [key: string]: any;
10
+ };
11
+ content?: string[];
12
+ [key: string]: any;
13
+ }
14
+ export declare function loadTailwindConfig(projectRoot: string): Promise<TailwindConfig | null>;
@@ -0,0 +1,139 @@
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.loadTailwindConfig = loadTailwindConfig;
7
+ const path_1 = __importDefault(require("path"));
8
+ const fs_1 = __importDefault(require("fs"));
9
+ const resolve_1 = __importDefault(require("resolve"));
10
+ async function loadTailwindConfig(projectRoot) {
11
+ const configPaths = [
12
+ 'tailwind.config.js',
13
+ 'tailwind.config.ts',
14
+ 'tailwind.config.mjs',
15
+ 'tailwind.config.cjs'
16
+ ];
17
+ for (const configPath of configPaths) {
18
+ const fullPath = path_1.default.join(projectRoot, configPath);
19
+ if (fs_1.default.existsSync(fullPath)) {
20
+ try {
21
+ // Clear require cache for hot reloading
22
+ delete require.cache[require.resolve(fullPath)];
23
+ // Load the config
24
+ let config;
25
+ if (configPath.endsWith('.ts')) {
26
+ // For TypeScript configs, we need to use a dynamic import
27
+ // But for CLI usage, we'll use a simpler approach
28
+ // Try to resolve tailwindcss and use its config resolution
29
+ const tailwindcssPath = resolve_1.default.sync('tailwindcss', { basedir: projectRoot });
30
+ const tailwindcss = require(tailwindcssPath);
31
+ if (tailwindcss.resolveConfig) {
32
+ const userConfig = require(fullPath);
33
+ config = tailwindcss.resolveConfig(userConfig.default || userConfig);
34
+ }
35
+ else {
36
+ config = require(fullPath);
37
+ }
38
+ }
39
+ else {
40
+ const userConfig = require(fullPath);
41
+ config = userConfig.default || userConfig;
42
+ }
43
+ return config;
44
+ }
45
+ catch (error) {
46
+ console.warn(`Failed to load Tailwind config at ${fullPath}:`, error);
47
+ continue;
48
+ }
49
+ }
50
+ }
51
+ // Return default Tailwind-like config
52
+ return {
53
+ theme: {
54
+ spacing: {
55
+ '0': '0px',
56
+ '1': '0.25rem',
57
+ '2': '0.5rem',
58
+ '3': '0.75rem',
59
+ '4': '1rem',
60
+ '5': '1.25rem',
61
+ '6': '1.5rem',
62
+ '8': '2rem',
63
+ '10': '2.5rem',
64
+ '12': '3rem',
65
+ '16': '4rem',
66
+ '20': '5rem',
67
+ '24': '6rem',
68
+ '32': '8rem',
69
+ '40': '10rem',
70
+ '48': '12rem',
71
+ '56': '14rem',
72
+ '64': '16rem'
73
+ },
74
+ colors: {
75
+ transparent: 'transparent',
76
+ current: 'currentColor',
77
+ black: '#000000',
78
+ white: '#ffffff',
79
+ gray: {
80
+ 50: '#f9fafb',
81
+ 100: '#f3f4f6',
82
+ 200: '#e5e7eb',
83
+ 300: '#d1d5db',
84
+ 400: '#9ca3af',
85
+ 500: '#6b7280',
86
+ 600: '#4b5563',
87
+ 700: '#374151',
88
+ 800: '#1f2937',
89
+ 900: '#111827'
90
+ },
91
+ red: {
92
+ 500: '#ef4444',
93
+ 600: '#dc2626'
94
+ },
95
+ blue: {
96
+ 500: '#3b82f6',
97
+ 600: '#2563eb'
98
+ },
99
+ green: {
100
+ 500: '#22c55e',
101
+ 600: '#16a34a'
102
+ }
103
+ },
104
+ fontSize: {
105
+ 'xs': ['0.75rem', { lineHeight: '1rem' }],
106
+ 'sm': ['0.875rem', { lineHeight: '1.25rem' }],
107
+ 'base': ['1rem', { lineHeight: '1.5rem' }],
108
+ 'lg': ['1.125rem', { lineHeight: '1.75rem' }],
109
+ 'xl': ['1.25rem', { lineHeight: '1.75rem' }],
110
+ '2xl': ['1.5rem', { lineHeight: '2rem' }],
111
+ '3xl': ['1.875rem', { lineHeight: '2.25rem' }],
112
+ '4xl': ['2.25rem', { lineHeight: '2.5rem' }]
113
+ },
114
+ fontWeight: {
115
+ thin: '100',
116
+ extralight: '200',
117
+ light: '300',
118
+ normal: '400',
119
+ medium: '500',
120
+ semibold: '600',
121
+ bold: '700',
122
+ extrabold: '800',
123
+ black: '900'
124
+ },
125
+ borderRadius: {
126
+ 'none': '0px',
127
+ 'sm': '0.125rem',
128
+ 'DEFAULT': '0.25rem',
129
+ 'md': '0.375rem',
130
+ 'lg': '0.5rem',
131
+ 'xl': '0.75rem',
132
+ '2xl': '1rem',
133
+ '3xl': '1.5rem',
134
+ 'full': '9999px'
135
+ }
136
+ }
137
+ };
138
+ }
139
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config.js","sourceRoot":"","sources":["../../src/utils/config.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path';\nimport fs from 'fs';\nimport resolve from 'resolve';\n\nexport interface TailwindConfig {\n  theme?: {\n    extend?: Record<string, any>;\n    spacing?: Record<string, string>;\n    colors?: Record<string, any>;\n    fontSize?: Record<string, any>;\n    fontWeight?: Record<string, string>;\n    borderRadius?: Record<string, string>;\n    [key: string]: any;\n  };\n  content?: string[];\n  [key: string]: any;\n}\n\nexport async function loadTailwindConfig(projectRoot: string): Promise<TailwindConfig | null> {\n  const configPaths = [\n    'tailwind.config.js',\n    'tailwind.config.ts',\n    'tailwind.config.mjs',\n    'tailwind.config.cjs'\n  ];\n\n  for (const configPath of configPaths) {\n    const fullPath = path.join(projectRoot, configPath);\n    \n    if (fs.existsSync(fullPath)) {\n      try {\n        // Clear require cache for hot reloading\n        delete require.cache[require.resolve(fullPath)];\n        \n        // Load the config\n        let config: TailwindConfig;\n        \n        if (configPath.endsWith('.ts')) {\n          // For TypeScript configs, we need to use a dynamic import\n          // But for CLI usage, we'll use a simpler approach\n          // Try to resolve tailwindcss and use its config resolution\n          const tailwindcssPath = resolve.sync('tailwindcss', { basedir: projectRoot });\n          const tailwindcss = require(tailwindcssPath);\n          \n          if (tailwindcss.resolveConfig) {\n            const userConfig = require(fullPath);\n            config = tailwindcss.resolveConfig(userConfig.default || userConfig);\n          } else {\n            config = require(fullPath);\n          }\n        } else {\n          const userConfig = require(fullPath);\n          config = userConfig.default || userConfig;\n        }\n\n        return config;\n      } catch (error) {\n        console.warn(`Failed to load Tailwind config at ${fullPath}:`, error);\n        continue;\n      }\n    }\n  }\n\n  // Return default Tailwind-like config\n  return {\n    theme: {\n      spacing: {\n        '0': '0px',\n        '1': '0.25rem',\n        '2': '0.5rem',\n        '3': '0.75rem',\n        '4': '1rem',\n        '5': '1.25rem',\n        '6': '1.5rem',\n        '8': '2rem',\n        '10': '2.5rem',\n        '12': '3rem',\n        '16': '4rem',\n        '20': '5rem',\n        '24': '6rem',\n        '32': '8rem',\n        '40': '10rem',\n        '48': '12rem',\n        '56': '14rem',\n        '64': '16rem'\n      },\n      colors: {\n        transparent: 'transparent',\n        current: 'currentColor',\n        black: '#000000',\n        white: '#ffffff',\n        gray: {\n          50: '#f9fafb',\n          100: '#f3f4f6',\n          200: '#e5e7eb',\n          300: '#d1d5db',\n          400: '#9ca3af',\n          500: '#6b7280',\n          600: '#4b5563',\n          700: '#374151',\n          800: '#1f2937',\n          900: '#111827'\n        },\n        red: {\n          500: '#ef4444',\n          600: '#dc2626'\n        },\n        blue: {\n          500: '#3b82f6',\n          600: '#2563eb'\n        },\n        green: {\n          500: '#22c55e',\n          600: '#16a34a'\n        }\n      },\n      fontSize: {\n        'xs': ['0.75rem', { lineHeight: '1rem' }],\n        'sm': ['0.875rem', { lineHeight: '1.25rem' }],\n        'base': ['1rem', { lineHeight: '1.5rem' }],\n        'lg': ['1.125rem', { lineHeight: '1.75rem' }],\n        'xl': ['1.25rem', { lineHeight: '1.75rem' }],\n        '2xl': ['1.5rem', { lineHeight: '2rem' }],\n        '3xl': ['1.875rem', { lineHeight: '2.25rem' }],\n        '4xl': ['2.25rem', { lineHeight: '2.5rem' }]\n      },\n      fontWeight: {\n        thin: '100',\n        extralight: '200',\n        light: '300',\n        normal: '400',\n        medium: '500',\n        semibold: '600',\n        bold: '700',\n        extrabold: '800',\n        black: '900'\n      },\n      borderRadius: {\n        'none': '0px',\n        'sm': '0.125rem',\n        'DEFAULT': '0.25rem',\n        'md': '0.375rem',\n        'lg': '0.5rem',\n        'xl': '0.75rem',\n        '2xl': '1rem',\n        '3xl': '1.5rem',\n        'full': '9999px'\n      }\n    }\n  };\n}\n"]}
@@ -0,0 +1,14 @@
1
+ declare class Logger {
2
+ private isVerbose;
3
+ setVerbose(verbose: boolean): void;
4
+ info(message: string, ...args: any[]): void;
5
+ success(message: string, ...args: any[]): void;
6
+ warn(message: string, ...args: any[]): void;
7
+ error(message: string, error?: any): void;
8
+ verbose(message: string, ...args: any[]): void;
9
+ debug(message: string, ...args: any[]): void;
10
+ file(message: string, filePath: string): void;
11
+ diff(original: string, converted: string): void;
12
+ }
13
+ export declare const logger: Logger;
14
+ export {};
@@ -0,0 +1,56 @@
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.logger = void 0;
7
+ const chalk_1 = __importDefault(require("chalk"));
8
+ class Logger {
9
+ constructor() {
10
+ this.isVerbose = false;
11
+ }
12
+ setVerbose(verbose) {
13
+ this.isVerbose = verbose;
14
+ }
15
+ info(message, ...args) {
16
+ console.log(chalk_1.default.blue(message), ...args);
17
+ }
18
+ success(message, ...args) {
19
+ console.log(chalk_1.default.green(message), ...args);
20
+ }
21
+ warn(message, ...args) {
22
+ console.log(chalk_1.default.yellow(`⚠️ ${message}`), ...args);
23
+ }
24
+ error(message, error) {
25
+ console.error(chalk_1.default.red(`❌ ${message}`));
26
+ if (error && this.isVerbose) {
27
+ if (error instanceof Error) {
28
+ console.error(chalk_1.default.red(error.stack || error.message));
29
+ }
30
+ else {
31
+ console.error(chalk_1.default.red(error));
32
+ }
33
+ }
34
+ }
35
+ verbose(message, ...args) {
36
+ if (this.isVerbose) {
37
+ console.log(chalk_1.default.gray(`[verbose] ${message}`), ...args);
38
+ }
39
+ }
40
+ debug(message, ...args) {
41
+ if (this.isVerbose) {
42
+ console.log(chalk_1.default.cyan(`[debug] ${message}`), ...args);
43
+ }
44
+ }
45
+ file(message, filePath) {
46
+ console.log(chalk_1.default.magenta(`${message}:`), chalk_1.default.white(filePath));
47
+ }
48
+ diff(original, converted) {
49
+ if (this.isVerbose) {
50
+ console.log(chalk_1.default.red(' -'), original);
51
+ console.log(chalk_1.default.green(' +'), converted);
52
+ }
53
+ }
54
+ }
55
+ exports.logger = new Logger();
56
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nZ2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3V0aWxzL2xvZ2dlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7QUFBQSxrREFBMEI7QUFFMUIsTUFBTSxNQUFNO0lBQVo7UUFDVSxjQUFTLEdBQVksS0FBSyxDQUFDO0lBbURyQyxDQUFDO0lBakRDLFVBQVUsQ0FBQyxPQUFnQjtRQUN6QixJQUFJLENBQUMsU0FBUyxHQUFHLE9BQU8sQ0FBQztJQUMzQixDQUFDO0lBRUQsSUFBSSxDQUFDLE9BQWUsRUFBRSxHQUFHLElBQVc7UUFDbEMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxlQUFLLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxFQUFFLEdBQUcsSUFBSSxDQUFDLENBQUM7SUFDNUMsQ0FBQztJQUVELE9BQU8sQ0FBQyxPQUFlLEVBQUUsR0FBRyxJQUFXO1FBQ3JDLE9BQU8sQ0FBQyxHQUFHLENBQUMsZUFBSyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsRUFBRSxHQUFHLElBQUksQ0FBQyxDQUFDO0lBQzdDLENBQUM7SUFFRCxJQUFJLENBQUMsT0FBZSxFQUFFLEdBQUcsSUFBVztRQUNsQyxPQUFPLENBQUMsR0FBRyxDQUFDLGVBQUssQ0FBQyxNQUFNLENBQUMsT0FBTyxPQUFPLEVBQUUsQ0FBQyxFQUFFLEdBQUcsSUFBSSxDQUFDLENBQUM7SUFDdkQsQ0FBQztJQUVELEtBQUssQ0FBQyxPQUFlLEVBQUUsS0FBVztRQUNoQyxPQUFPLENBQUMsS0FBSyxDQUFDLGVBQUssQ0FBQyxHQUFHLENBQUMsS0FBSyxPQUFPLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDekMsSUFBSSxLQUFLLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO1lBQzVCLElBQUksS0FBSyxZQUFZLEtBQUssRUFBRSxDQUFDO2dCQUMzQixPQUFPLENBQUMsS0FBSyxDQUFDLGVBQUssQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLEtBQUssSUFBSSxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQztZQUN6RCxDQUFDO2lCQUFNLENBQUM7Z0JBQ04sT0FBTyxDQUFDLEtBQUssQ0FBQyxlQUFLLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7WUFDbEMsQ0FBQztRQUNILENBQUM7SUFDSCxDQUFDO0lBRUQsT0FBTyxDQUFDLE9BQWUsRUFBRSxHQUFHLElBQVc7UUFDckMsSUFBSSxJQUFJLENBQUMsU0FBUyxFQUFFLENBQUM7WUFDbkIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxlQUFLLENBQUMsSUFBSSxDQUFDLGFBQWEsT0FBTyxFQUFFLENBQUMsRUFBRSxHQUFHLElBQUksQ0FBQyxDQUFDO1FBQzNELENBQUM7SUFDSCxDQUFDO0lBRUQsS0FBSyxDQUFDLE9BQWUsRUFBRSxHQUFHLElBQVc7UUFDbkMsSUFBSSxJQUFJLENBQUMsU0FBUyxFQUFFLENBQUM7WUFDbkIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxlQUFLLENBQUMsSUFBSSxDQUFDLFdBQVcsT0FBTyxFQUFFLENBQUMsRUFBRSxHQUFHLElBQUksQ0FBQyxDQUFDO1FBQ3pELENBQUM7SUFDSCxDQUFDO0lBRUQsSUFBSSxDQUFDLE9BQWUsRUFBRSxRQUFnQjtRQUNwQyxPQUFPLENBQUMsR0FBRyxDQUFDLGVBQUssQ0FBQyxPQUFPLENBQUMsR0FBRyxPQUFPLEdBQUcsQ0FBQyxFQUFFLGVBQUssQ0FBQyxLQUFLLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQztJQUNuRSxDQUFDO0lBRUQsSUFBSSxDQUFDLFFBQWdCLEVBQUUsU0FBaUI7UUFDdEMsSUFBSSxJQUFJLENBQUMsU0FBUyxFQUFFLENBQUM7WUFDbkIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxlQUFLLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxFQUFFLFFBQVEsQ0FBQyxDQUFDO1lBQ3hDLE9BQU8sQ0FBQyxHQUFHLENBQUMsZUFBSyxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsRUFBRSxTQUFTLENBQUMsQ0FBQztRQUM3QyxDQUFDO0lBQ0gsQ0FBQztDQUNGO0FBRVksUUFBQSxNQUFNLEdBQUcsSUFBSSxNQUFNLEVBQUUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBjaGFsayBmcm9tICdjaGFsayc7XG5cbmNsYXNzIExvZ2dlciB7XG4gIHByaXZhdGUgaXNWZXJib3NlOiBib29sZWFuID0gZmFsc2U7XG5cbiAgc2V0VmVyYm9zZSh2ZXJib3NlOiBib29sZWFuKTogdm9pZCB7XG4gICAgdGhpcy5pc1ZlcmJvc2UgPSB2ZXJib3NlO1xuICB9XG5cbiAgaW5mbyhtZXNzYWdlOiBzdHJpbmcsIC4uLmFyZ3M6IGFueVtdKTogdm9pZCB7XG4gICAgY29uc29sZS5sb2coY2hhbGsuYmx1ZShtZXNzYWdlKSwgLi4uYXJncyk7XG4gIH1cblxuICBzdWNjZXNzKG1lc3NhZ2U6IHN0cmluZywgLi4uYXJnczogYW55W10pOiB2b2lkIHtcbiAgICBjb25zb2xlLmxvZyhjaGFsay5ncmVlbihtZXNzYWdlKSwgLi4uYXJncyk7XG4gIH1cblxuICB3YXJuKG1lc3NhZ2U6IHN0cmluZywgLi4uYXJnczogYW55W10pOiB2b2lkIHtcbiAgICBjb25zb2xlLmxvZyhjaGFsay55ZWxsb3coYOKaoO+4jyAgJHttZXNzYWdlfWApLCAuLi5hcmdzKTtcbiAgfVxuXG4gIGVycm9yKG1lc3NhZ2U6IHN0cmluZywgZXJyb3I/OiBhbnkpOiB2b2lkIHtcbiAgICBjb25zb2xlLmVycm9yKGNoYWxrLnJlZChg4p2MICR7bWVzc2FnZX1gKSk7XG4gICAgaWYgKGVycm9yICYmIHRoaXMuaXNWZXJib3NlKSB7XG4gICAgICBpZiAoZXJyb3IgaW5zdGFuY2VvZiBFcnJvcikge1xuICAgICAgICBjb25zb2xlLmVycm9yKGNoYWxrLnJlZChlcnJvci5zdGFjayB8fCBlcnJvci5tZXNzYWdlKSk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBjb25zb2xlLmVycm9yKGNoYWxrLnJlZChlcnJvcikpO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIHZlcmJvc2UobWVzc2FnZTogc3RyaW5nLCAuLi5hcmdzOiBhbnlbXSk6IHZvaWQge1xuICAgIGlmICh0aGlzLmlzVmVyYm9zZSkge1xuICAgICAgY29uc29sZS5sb2coY2hhbGsuZ3JheShgW3ZlcmJvc2VdICR7bWVzc2FnZX1gKSwgLi4uYXJncyk7XG4gICAgfVxuICB9XG5cbiAgZGVidWcobWVzc2FnZTogc3RyaW5nLCAuLi5hcmdzOiBhbnlbXSk6IHZvaWQge1xuICAgIGlmICh0aGlzLmlzVmVyYm9zZSkge1xuICAgICAgY29uc29sZS5sb2coY2hhbGsuY3lhbihgW2RlYnVnXSAke21lc3NhZ2V9YCksIC4uLmFyZ3MpO1xuICAgIH1cbiAgfVxuXG4gIGZpbGUobWVzc2FnZTogc3RyaW5nLCBmaWxlUGF0aDogc3RyaW5nKTogdm9pZCB7XG4gICAgY29uc29sZS5sb2coY2hhbGsubWFnZW50YShgJHttZXNzYWdlfTpgKSwgY2hhbGsud2hpdGUoZmlsZVBhdGgpKTtcbiAgfVxuXG4gIGRpZmYob3JpZ2luYWw6IHN0cmluZywgY29udmVydGVkOiBzdHJpbmcpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5pc1ZlcmJvc2UpIHtcbiAgICAgIGNvbnNvbGUubG9nKGNoYWxrLnJlZCgnICAtJyksIG9yaWdpbmFsKTtcbiAgICAgIGNvbnNvbGUubG9nKGNoYWxrLmdyZWVuKCcgICsnKSwgY29udmVydGVkKTtcbiAgICB9XG4gIH1cbn1cblxuZXhwb3J0IGNvbnN0IGxvZ2dlciA9IG5ldyBMb2dnZXIoKTtcbiJdfQ==
package/package.json ADDED
@@ -0,0 +1,73 @@
1
+ {
2
+ "name": "css-to-tailwind-react",
3
+ "version": "0.1.0",
4
+ "description": "Convert traditional CSS (inline, internal, and external) into Tailwind CSS utility classes for React-based frameworks",
5
+ "main": "dist/index.js",
6
+ "module": "dist/index.mjs",
7
+ "types": "dist/index.d.ts",
8
+ "exports": {
9
+ ".": {
10
+ "import": "./dist/index.mjs",
11
+ "require": "./dist/index.js",
12
+ "types": "./dist/index.d.ts"
13
+ }
14
+ },
15
+ "bin": {
16
+ "css-to-tailwind-react": "./bin/index.js"
17
+ },
18
+ "files": [
19
+ "dist/**/*",
20
+ "bin/**/*",
21
+ "README.md"
22
+ ],
23
+ "scripts": {
24
+ "build": "tsc && tsc-alias",
25
+ "build:watch": "tsc --watch",
26
+ "clean": "rm -rf dist",
27
+ "prepublishOnly": "npm run clean && npm run build",
28
+ "test": "echo \"No tests yet\" && exit 0",
29
+ "lint": "tsc --noEmit"
30
+ },
31
+ "keywords": [
32
+ "tailwindcss",
33
+ "css",
34
+ "react",
35
+ "jsx",
36
+ "tsx",
37
+ "converter",
38
+ "utility-classes",
39
+ "ast",
40
+ "babel",
41
+ "postcss"
42
+ ],
43
+ "author": "Rudra Patel",
44
+ "license": "MIT",
45
+ "engines": {
46
+ "node": ">=16.0.0"
47
+ },
48
+ "dependencies": {
49
+ "@babel/generator": "^7.23.6",
50
+ "@babel/parser": "^7.23.6",
51
+ "@babel/traverse": "^7.23.7",
52
+ "@babel/types": "^7.23.6",
53
+ "chalk": "^4.1.2",
54
+ "commander": "^11.1.0",
55
+ "fast-glob": "^3.3.2",
56
+ "postcss": "^8.4.32",
57
+ "postcss-safe-parser": "^6.0.0",
58
+ "resolve": "^1.22.8",
59
+ "tailwindcss": ">=3.0.0"
60
+ },
61
+ "devDependencies": {
62
+ "@types/babel__generator": "^7.6.8",
63
+ "@types/babel__traverse": "^7.20.5",
64
+ "@types/node": "^20.10.6",
65
+ "@types/postcss-safe-parser": "^5.0.4",
66
+ "@types/resolve": "^1.20.6",
67
+ "tsc-alias": "^1.8.8",
68
+ "typescript": "^5.3.3"
69
+ },
70
+ "peerDependencies": {
71
+ "tailwindcss": ">=3.0.0"
72
+ }
73
+ }