css-to-tailwind-react 0.3.4 ā 0.4.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/dist/cli.js +107 -24
- package/dist/fileWriter.d.ts +11 -3
- package/dist/fileWriter.js +34 -39
- package/dist/index.d.ts +4 -2
- package/dist/index.js +11 -2
- package/dist/transformer.d.ts +6 -0
- package/dist/transformer.js +131 -56
- package/dist/utils/diff.d.ts +26 -0
- package/dist/utils/diff.js +197 -0
- package/dist/utils/logger.d.ts +2 -0
- package/dist/utils/logger.js +21 -7
- package/dist/utils/reporter.d.ts +50 -0
- package/dist/utils/reporter.js +215 -0
- package/package.json +1 -1
package/dist/cli.js
CHANGED
|
@@ -1,4 +1,37 @@
|
|
|
1
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
|
+
})();
|
|
2
35
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
36
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
37
|
};
|
|
@@ -6,8 +39,10 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
6
39
|
const commander_1 = require("commander");
|
|
7
40
|
const scanner_1 = require("./scanner");
|
|
8
41
|
const transformer_1 = require("./transformer");
|
|
42
|
+
const fileWriter_1 = require("./fileWriter");
|
|
9
43
|
const logger_1 = require("./utils/logger");
|
|
10
44
|
const config_1 = require("./utils/config");
|
|
45
|
+
const reporter_1 = require("./utils/reporter");
|
|
11
46
|
const path_1 = __importDefault(require("path"));
|
|
12
47
|
const program = new commander_1.Command();
|
|
13
48
|
program
|
|
@@ -15,7 +50,10 @@ program
|
|
|
15
50
|
.description('Convert traditional CSS into Tailwind CSS utility classes for React')
|
|
16
51
|
.version('1.0.0')
|
|
17
52
|
.argument('<directory>', 'Target directory to scan and transform')
|
|
18
|
-
.option('--dry-run', 'Show changes without modifying files')
|
|
53
|
+
.option('--dry-run, --preview', 'Show changes without modifying files')
|
|
54
|
+
.option('--diff', 'Print unified diff for each modified file')
|
|
55
|
+
.option('--silent', 'Suppress per-file logs, show only summary')
|
|
56
|
+
.option('--json-report', 'Output structured JSON summary')
|
|
19
57
|
.option('--verbose', 'Show detailed output')
|
|
20
58
|
.option('--delete-css', 'Delete CSS files when all rules are converted')
|
|
21
59
|
.option('--skip-external', 'Skip external CSS files (imports)')
|
|
@@ -23,18 +61,23 @@ program
|
|
|
23
61
|
.option('--skip-internal', 'Skip internal <style> blocks')
|
|
24
62
|
.action(async (directory, options) => {
|
|
25
63
|
const startTime = Date.now();
|
|
64
|
+
const isDryRun = options.dryRun || options.preview || false;
|
|
65
|
+
const showDiff = options.diff || false;
|
|
66
|
+
const isSilent = options.silent || options.jsonReport || false;
|
|
67
|
+
const isJsonReport = options.jsonReport || false;
|
|
26
68
|
try {
|
|
27
|
-
// Set verbose mode for logger
|
|
28
69
|
logger_1.logger.setVerbose(options.verbose || false);
|
|
70
|
+
logger_1.logger.setSilent(isSilent);
|
|
29
71
|
logger_1.logger.info('š CSS to Tailwind React Converter');
|
|
30
72
|
logger_1.logger.info(`š Target directory: ${path_1.default.resolve(directory)}`);
|
|
31
|
-
if (
|
|
73
|
+
if (isDryRun) {
|
|
32
74
|
logger_1.logger.info('š Dry run mode - no files will be modified');
|
|
33
75
|
}
|
|
34
|
-
|
|
76
|
+
if (showDiff) {
|
|
77
|
+
logger_1.logger.info('š Diff mode enabled - showing changes');
|
|
78
|
+
}
|
|
35
79
|
logger_1.logger.info('āļø Loading Tailwind configuration...');
|
|
36
80
|
const tailwindConfig = await (0, config_1.loadTailwindConfig)(directory);
|
|
37
|
-
// Scan project files
|
|
38
81
|
logger_1.logger.info('š Scanning project files...');
|
|
39
82
|
const files = await (0, scanner_1.scanProject)(directory);
|
|
40
83
|
logger_1.logger.success(`Found ${files.length} files to process`);
|
|
@@ -42,9 +85,8 @@ program
|
|
|
42
85
|
logger_1.logger.warn('No supported files found in the target directory');
|
|
43
86
|
process.exit(0);
|
|
44
87
|
}
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
dryRun: options.dryRun || false,
|
|
88
|
+
const { fileResults, stats } = await (0, transformer_1.transformFilesDetailed)(files, {
|
|
89
|
+
dryRun: isDryRun,
|
|
48
90
|
deleteCss: options.deleteCss || false,
|
|
49
91
|
skipExternal: options.skipExternal || false,
|
|
50
92
|
skipInline: options.skipInline || false,
|
|
@@ -52,22 +94,63 @@ program
|
|
|
52
94
|
tailwindConfig,
|
|
53
95
|
projectRoot: path_1.default.resolve(directory)
|
|
54
96
|
});
|
|
55
|
-
|
|
97
|
+
const reporter = new reporter_1.Reporter({
|
|
98
|
+
showDiff,
|
|
99
|
+
silent: isSilent,
|
|
100
|
+
verbose: options.verbose || false,
|
|
101
|
+
dryRun: isDryRun,
|
|
102
|
+
projectRoot: path_1.default.resolve(directory)
|
|
103
|
+
});
|
|
104
|
+
for (const result of fileResults) {
|
|
105
|
+
reporter.addFileResult(result);
|
|
106
|
+
}
|
|
107
|
+
reporter.addWarning(stats.warnings);
|
|
108
|
+
if (!isDryRun) {
|
|
109
|
+
logger_1.logger.info('\nš¾ Writing changes...');
|
|
110
|
+
const writtenCount = await (0, fileWriter_1.writeFiles)(fileResults, {
|
|
111
|
+
dryRun: false,
|
|
112
|
+
projectRoot: path_1.default.resolve(directory)
|
|
113
|
+
});
|
|
114
|
+
if (options.deleteCss) {
|
|
115
|
+
for (const result of fileResults) {
|
|
116
|
+
if (result.hasChanges && result.filePath.endsWith('.css')) {
|
|
117
|
+
const cssContent = result.newContent.trim();
|
|
118
|
+
if (cssContent === '' || cssContent === '/* All CSS converted to Tailwind */') {
|
|
119
|
+
const fs = await Promise.resolve().then(() => __importStar(require('fs')));
|
|
120
|
+
const backupDir = path_1.default.join(path_1.default.resolve(directory), '.css-to-tailwind-backups');
|
|
121
|
+
const relativePath = path_1.default.relative(path_1.default.resolve(directory), result.filePath);
|
|
122
|
+
const backupPath = path_1.default.join(backupDir, relativePath);
|
|
123
|
+
if (!fs.existsSync(backupDir)) {
|
|
124
|
+
fs.mkdirSync(backupDir, { recursive: true });
|
|
125
|
+
}
|
|
126
|
+
const backupDirPath = path_1.default.dirname(backupPath);
|
|
127
|
+
if (!fs.existsSync(backupDirPath)) {
|
|
128
|
+
fs.mkdirSync(backupDirPath, { recursive: true });
|
|
129
|
+
}
|
|
130
|
+
fs.writeFileSync(backupPath, result.originalContent, 'utf-8');
|
|
131
|
+
fs.unlinkSync(result.filePath);
|
|
132
|
+
logger_1.logger.info(`šļø Deleted ${path_1.default.basename(result.filePath)} (all rules converted)`);
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
logger_1.logger.success(`Wrote ${writtenCount} files`);
|
|
138
|
+
}
|
|
56
139
|
const duration = ((Date.now() - startTime) / 1000).toFixed(2);
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
process.exit(
|
|
140
|
+
if (isJsonReport) {
|
|
141
|
+
const jsonOutput = reporter.toJSON();
|
|
142
|
+
const output = {
|
|
143
|
+
...jsonOutput,
|
|
144
|
+
duration: `${duration}s`
|
|
145
|
+
};
|
|
146
|
+
console.log(JSON.stringify(output, null, 2));
|
|
147
|
+
}
|
|
148
|
+
else {
|
|
149
|
+
reporter.print();
|
|
150
|
+
console.log(`\nā±ļø Duration: ${duration}s`);
|
|
151
|
+
}
|
|
152
|
+
if (stats.errors > 0) {
|
|
153
|
+
process.exit(1);
|
|
71
154
|
}
|
|
72
155
|
process.exit(0);
|
|
73
156
|
}
|
|
@@ -77,4 +160,4 @@ program
|
|
|
77
160
|
}
|
|
78
161
|
});
|
|
79
162
|
program.parse();
|
|
80
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command } from 'commander';\nimport { scanProject } from './scanner';\nimport { transformFiles } from './transformer';\nimport { logger } from './utils/logger';\nimport { loadTailwindConfig } from './utils/config';\nimport path from 'path';\n\ninterface CLIOptions {\n  dryRun?: boolean;\n  verbose?: boolean;\n  deleteCss?: boolean;\n  skipExternal?: boolean;\n  skipInline?: boolean;\n  skipInternal?: boolean;\n}\n\nconst program = new Command();\n\nprogram\n  .name('css-to-tailwind-react')\n  .description('Convert traditional CSS into Tailwind CSS utility classes for React')\n  .version('1.0.0')\n  .argument('<directory>', 'Target directory to scan and transform')\n  .option('--dry-run', 'Show changes without modifying files')\n  .option('--verbose', 'Show detailed output')\n  .option('--delete-css', 'Delete CSS files when all rules are converted')\n  .option('--skip-external', 'Skip external CSS files (imports)')\n  .option('--skip-inline', 'Skip inline styles')\n  .option('--skip-internal', 'Skip internal <style> blocks')\n  .action(async (directory: string, options: CLIOptions) => {\n    const startTime = Date.now();\n    \n    try {\n      // Set verbose mode for logger\n      logger.setVerbose(options.verbose || false);\n      \n      logger.info('🚀 CSS to Tailwind React Converter');\n      logger.info(`📁 Target directory: ${path.resolve(directory)}`);\n      \n      if (options.dryRun) {\n        logger.info('🔍 Dry run mode - no files will be modified');\n      }\n\n      // Load Tailwind config\n      logger.info('⚙️  Loading Tailwind configuration...');\n      const tailwindConfig = await loadTailwindConfig(directory);\n      \n      // Scan project files\n      logger.info('🔎 Scanning project files...');\n      const files = await scanProject(directory);\n      \n      logger.success(`Found ${files.length} files to process`);\n      \n      if (files.length === 0) {\n        logger.warn('No supported files found in the target directory');\n        process.exit(0);\n      }\n\n      // Transform files\n      const results = await transformFiles(files, {\n        dryRun: options.dryRun || false,\n        deleteCss: options.deleteCss || false,\n        skipExternal: options.skipExternal || false,\n        skipInline: options.skipInline || false,\n        skipInternal: options.skipInternal || false,\n        tailwindConfig,\n        projectRoot: path.resolve(directory)\n      });\n\n      // Print summary\n      const duration = ((Date.now() - startTime) / 1000).toFixed(2);\n      \n      console.log('\\n' + '='.repeat(50));\n      logger.success('✨ Transformation Complete!');\n      console.log('='.repeat(50));\n      console.log(`📊 Summary:`);\n      console.log(`   Files scanned:      ${results.filesScanned}`);\n      console.log(`   Files modified:     ${results.filesModified}`);\n      console.log(`   Styles converted:   ${results.stylesConverted}`);\n      console.log(`   Classes replaced:   ${results.classesReplaced}`);\n      console.log(`   Warnings:           ${results.warnings}`);\n      console.log(`   Duration:           ${duration}s`);\n      console.log('='.repeat(50));\n\n      if (results.warnings > 0) {\n        logger.warn('Some styles could not be converted. Run with --verbose for details.');\n        process.exit(0);\n      }\n\n      process.exit(0);\n    } catch (error) {\n      logger.error('❌ Transformation failed:', error);\n      process.exit(1);\n    }\n  });\n\nprogram.parse();\n"]}
|
|
163
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command } from 'commander';\nimport { scanProject } from './scanner';\nimport { transformFilesDetailed } from './transformer';\nimport { writeFiles } from './fileWriter';\nimport { logger } from './utils/logger';\nimport { loadTailwindConfig } from './utils/config';\nimport { Reporter } from './utils/reporter';\nimport path from 'path';\n\ninterface CLIOptions {\n  dryRun?: boolean;\n  preview?: boolean;\n  diff?: boolean;\n  silent?: boolean;\n  jsonReport?: boolean;\n  verbose?: boolean;\n  deleteCss?: boolean;\n  skipExternal?: boolean;\n  skipInline?: boolean;\n  skipInternal?: boolean;\n}\n\nconst program = new Command();\n\nprogram\n  .name('css-to-tailwind-react')\n  .description('Convert traditional CSS into Tailwind CSS utility classes for React')\n  .version('1.0.0')\n  .argument('<directory>', 'Target directory to scan and transform')\n  .option('--dry-run, --preview', 'Show changes without modifying files')\n  .option('--diff', 'Print unified diff for each modified file')\n  .option('--silent', 'Suppress per-file logs, show only summary')\n  .option('--json-report', 'Output structured JSON summary')\n  .option('--verbose', 'Show detailed output')\n  .option('--delete-css', 'Delete CSS files when all rules are converted')\n  .option('--skip-external', 'Skip external CSS files (imports)')\n  .option('--skip-inline', 'Skip inline styles')\n  .option('--skip-internal', 'Skip internal <style> blocks')\n  .action(async (directory: string, options: CLIOptions) => {\n    const startTime = Date.now();\n    \n    const isDryRun = options.dryRun || options.preview || false;\n    const showDiff = options.diff || false;\n    const isSilent = options.silent || options.jsonReport || false;\n    const isJsonReport = options.jsonReport || false;\n    \n    try {\n      logger.setVerbose(options.verbose || false);\n      logger.setSilent(isSilent);\n      \n      logger.info('🚀 CSS to Tailwind React Converter');\n      logger.info(`📁 Target directory: ${path.resolve(directory)}`);\n      \n      if (isDryRun) {\n        logger.info('🔍 Dry run mode - no files will be modified');\n      }\n      \n      if (showDiff) {\n        logger.info('📋 Diff mode enabled - showing changes');\n      }\n\n      logger.info('⚙️  Loading Tailwind configuration...');\n      const tailwindConfig = await loadTailwindConfig(directory);\n      \n      logger.info('🔎 Scanning project files...');\n      const files = await scanProject(directory);\n      \n      logger.success(`Found ${files.length} files to process`);\n      \n      if (files.length === 0) {\n        logger.warn('No supported files found in the target directory');\n        process.exit(0);\n      }\n\n      const { fileResults, stats } = await transformFilesDetailed(files, {\n        dryRun: isDryRun,\n        deleteCss: options.deleteCss || false,\n        skipExternal: options.skipExternal || false,\n        skipInline: options.skipInline || false,\n        skipInternal: options.skipInternal || false,\n        tailwindConfig,\n        projectRoot: path.resolve(directory)\n      });\n\n      const reporter = new Reporter({\n        showDiff,\n        silent: isSilent,\n        verbose: options.verbose || false,\n        dryRun: isDryRun,\n        projectRoot: path.resolve(directory)\n      });\n\n      for (const result of fileResults) {\n        reporter.addFileResult(result);\n      }\n      reporter.addWarning(stats.warnings);\n\n      if (!isDryRun) {\n        logger.info('\\n💾 Writing changes...');\n        const writtenCount = await writeFiles(fileResults, {\n          dryRun: false,\n          projectRoot: path.resolve(directory)\n        });\n        \n        if (options.deleteCss) {\n          for (const result of fileResults) {\n            if (result.hasChanges && result.filePath.endsWith('.css')) {\n              const cssContent = result.newContent.trim();\n              if (cssContent === '' || cssContent === '/* All CSS converted to Tailwind */') {\n                const fs = await import('fs');\n                const backupDir = path.join(path.resolve(directory), '.css-to-tailwind-backups');\n                const relativePath = path.relative(path.resolve(directory), result.filePath);\n                const backupPath = path.join(backupDir, relativePath);\n                \n                if (!fs.existsSync(backupDir)) {\n                  fs.mkdirSync(backupDir, { recursive: true });\n                }\n                const backupDirPath = path.dirname(backupPath);\n                if (!fs.existsSync(backupDirPath)) {\n                  fs.mkdirSync(backupDirPath, { recursive: true });\n                }\n                fs.writeFileSync(backupPath, result.originalContent, 'utf-8');\n                \n                fs.unlinkSync(result.filePath);\n                logger.info(`🗑️  Deleted ${path.basename(result.filePath)} (all rules converted)`);\n              }\n            }\n          }\n        }\n        \n        logger.success(`Wrote ${writtenCount} files`);\n      }\n\n      const duration = ((Date.now() - startTime) / 1000).toFixed(2);\n\n      if (isJsonReport) {\n        const jsonOutput = reporter.toJSON();\n        const output = {\n          ...jsonOutput,\n          duration: `${duration}s`\n        };\n        console.log(JSON.stringify(output, null, 2));\n      } else {\n        reporter.print();\n        console.log(`\\n⏱️  Duration: ${duration}s`);\n      }\n\n      if (stats.errors > 0) {\n        process.exit(1);\n      }\n\n      process.exit(0);\n    } catch (error) {\n      logger.error('❌ Transformation failed:', error);\n      process.exit(1);\n    }\n  });\n\nprogram.parse();\n"]}
|
package/dist/fileWriter.d.ts
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import { FileResult } from './utils/reporter';
|
|
1
2
|
export interface FileWriteOptions {
|
|
2
3
|
dryRun: boolean;
|
|
3
4
|
backup?: boolean;
|
|
@@ -5,10 +6,17 @@ export interface FileWriteOptions {
|
|
|
5
6
|
export declare class FileWriter {
|
|
6
7
|
private dryRun;
|
|
7
8
|
private backupDir;
|
|
8
|
-
|
|
9
|
+
private projectRoot;
|
|
10
|
+
constructor(options: FileWriteOptions & {
|
|
11
|
+
projectRoot?: string;
|
|
12
|
+
});
|
|
9
13
|
writeFile(filePath: string, content: string, originalContent: string): Promise<boolean>;
|
|
10
14
|
deleteFile(filePath: string): Promise<boolean>;
|
|
15
|
+
writeResults(results: FileResult[]): Promise<number>;
|
|
11
16
|
private createBackup;
|
|
12
|
-
|
|
13
|
-
static restoreBackups(): void;
|
|
17
|
+
static restoreBackups(projectRoot?: string): void;
|
|
14
18
|
}
|
|
19
|
+
export declare function writeFiles(results: FileResult[], options: {
|
|
20
|
+
dryRun: boolean;
|
|
21
|
+
projectRoot: string;
|
|
22
|
+
}): Promise<number>;
|
package/dist/fileWriter.js
CHANGED
|
@@ -4,26 +4,24 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
|
4
4
|
};
|
|
5
5
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
6
|
exports.FileWriter = void 0;
|
|
7
|
+
exports.writeFiles = writeFiles;
|
|
7
8
|
const fs_1 = __importDefault(require("fs"));
|
|
8
9
|
const path_1 = __importDefault(require("path"));
|
|
9
|
-
const chalk_1 = __importDefault(require("chalk"));
|
|
10
10
|
const logger_1 = require("./utils/logger");
|
|
11
11
|
class FileWriter {
|
|
12
12
|
constructor(options) {
|
|
13
13
|
this.dryRun = options.dryRun;
|
|
14
|
-
this.
|
|
14
|
+
this.projectRoot = options.projectRoot || process.cwd();
|
|
15
|
+
this.backupDir = path_1.default.join(this.projectRoot, '.css-to-tailwind-backups');
|
|
15
16
|
}
|
|
16
17
|
async writeFile(filePath, content, originalContent) {
|
|
17
18
|
if (this.dryRun) {
|
|
18
|
-
this.showDiff(filePath, originalContent, content);
|
|
19
19
|
return true;
|
|
20
20
|
}
|
|
21
21
|
try {
|
|
22
|
-
// Create backup
|
|
23
22
|
await this.createBackup(filePath, originalContent);
|
|
24
|
-
// Write file
|
|
25
23
|
fs_1.default.writeFileSync(filePath, content, 'utf-8');
|
|
26
|
-
logger_1.logger.success(`āļø Modified: ${path_1.default.relative(
|
|
24
|
+
logger_1.logger.success(`āļø Modified: ${path_1.default.relative(this.projectRoot, filePath)}`);
|
|
27
25
|
return true;
|
|
28
26
|
}
|
|
29
27
|
catch (error) {
|
|
@@ -33,15 +31,14 @@ class FileWriter {
|
|
|
33
31
|
}
|
|
34
32
|
async deleteFile(filePath) {
|
|
35
33
|
if (this.dryRun) {
|
|
36
|
-
logger_1.logger.info(`šļø Would delete: ${path_1.default.relative(
|
|
34
|
+
logger_1.logger.info(`šļø Would delete: ${path_1.default.relative(this.projectRoot, filePath)}`);
|
|
37
35
|
return true;
|
|
38
36
|
}
|
|
39
37
|
try {
|
|
40
|
-
// Create backup before deletion
|
|
41
38
|
const content = fs_1.default.readFileSync(filePath, 'utf-8');
|
|
42
39
|
await this.createBackup(filePath, content);
|
|
43
40
|
fs_1.default.unlinkSync(filePath);
|
|
44
|
-
logger_1.logger.success(`šļø Deleted: ${path_1.default.relative(
|
|
41
|
+
logger_1.logger.success(`šļø Deleted: ${path_1.default.relative(this.projectRoot, filePath)}`);
|
|
45
42
|
return true;
|
|
46
43
|
}
|
|
47
44
|
catch (error) {
|
|
@@ -49,20 +46,30 @@ class FileWriter {
|
|
|
49
46
|
return false;
|
|
50
47
|
}
|
|
51
48
|
}
|
|
49
|
+
async writeResults(results) {
|
|
50
|
+
let written = 0;
|
|
51
|
+
for (const result of results) {
|
|
52
|
+
if (result.status === 'error')
|
|
53
|
+
continue;
|
|
54
|
+
if (!result.hasChanges)
|
|
55
|
+
continue;
|
|
56
|
+
const success = await this.writeFile(result.filePath, result.newContent, result.originalContent);
|
|
57
|
+
if (success)
|
|
58
|
+
written++;
|
|
59
|
+
}
|
|
60
|
+
return written;
|
|
61
|
+
}
|
|
52
62
|
async createBackup(filePath, content) {
|
|
53
63
|
try {
|
|
54
|
-
// Create backup directory if it doesn't exist
|
|
55
64
|
if (!fs_1.default.existsSync(this.backupDir)) {
|
|
56
65
|
fs_1.default.mkdirSync(this.backupDir, { recursive: true });
|
|
57
66
|
}
|
|
58
|
-
const relativePath = path_1.default.relative(
|
|
67
|
+
const relativePath = path_1.default.relative(this.projectRoot, filePath);
|
|
59
68
|
const backupPath = path_1.default.join(this.backupDir, relativePath);
|
|
60
69
|
const backupDir = path_1.default.dirname(backupPath);
|
|
61
|
-
// Create subdirectory structure
|
|
62
70
|
if (!fs_1.default.existsSync(backupDir)) {
|
|
63
71
|
fs_1.default.mkdirSync(backupDir, { recursive: true });
|
|
64
72
|
}
|
|
65
|
-
// Write backup
|
|
66
73
|
fs_1.default.writeFileSync(backupPath, content, 'utf-8');
|
|
67
74
|
logger_1.logger.verbose(`Created backup: ${backupPath}`);
|
|
68
75
|
}
|
|
@@ -70,35 +77,13 @@ class FileWriter {
|
|
|
70
77
|
logger_1.logger.warn(`Failed to create backup for ${filePath}:`, error);
|
|
71
78
|
}
|
|
72
79
|
}
|
|
73
|
-
|
|
74
|
-
const
|
|
75
|
-
logger_1.logger.info(`\nš ${relativePath} (dry-run)`);
|
|
76
|
-
const originalLines = original.split('\n');
|
|
77
|
-
const modifiedLines = modified.split('\n');
|
|
78
|
-
// Simple diff - show first few changed lines
|
|
79
|
-
const maxLines = Math.max(originalLines.length, modifiedLines.length);
|
|
80
|
-
let changes = 0;
|
|
81
|
-
for (let i = 0; i < maxLines && changes < 10; i++) {
|
|
82
|
-
const orig = originalLines[i] || '';
|
|
83
|
-
const mod = modifiedLines[i] || '';
|
|
84
|
-
if (orig !== mod) {
|
|
85
|
-
changes++;
|
|
86
|
-
console.log(chalk_1.default.red(` - ${orig}`));
|
|
87
|
-
console.log(chalk_1.default.green(` + ${mod}`));
|
|
88
|
-
}
|
|
89
|
-
}
|
|
90
|
-
if (changes === 0) {
|
|
91
|
-
logger_1.logger.verbose(' (no visible changes)');
|
|
92
|
-
}
|
|
93
|
-
}
|
|
94
|
-
static restoreBackups() {
|
|
95
|
-
const backupDir = path_1.default.join(process.cwd(), '.css-to-tailwind-backups');
|
|
80
|
+
static restoreBackups(projectRoot = process.cwd()) {
|
|
81
|
+
const backupDir = path_1.default.join(projectRoot, '.css-to-tailwind-backups');
|
|
96
82
|
if (!fs_1.default.existsSync(backupDir)) {
|
|
97
83
|
logger_1.logger.warn('No backups found to restore');
|
|
98
84
|
return;
|
|
99
85
|
}
|
|
100
86
|
logger_1.logger.info('š Restoring files from backup...');
|
|
101
|
-
// Recursively restore files
|
|
102
87
|
const restoreRecursive = (dir) => {
|
|
103
88
|
const items = fs_1.default.readdirSync(dir);
|
|
104
89
|
for (const item of items) {
|
|
@@ -109,7 +94,7 @@ class FileWriter {
|
|
|
109
94
|
}
|
|
110
95
|
else {
|
|
111
96
|
const relativePath = path_1.default.relative(backupDir, fullPath);
|
|
112
|
-
const originalPath = path_1.default.join(
|
|
97
|
+
const originalPath = path_1.default.join(projectRoot, relativePath);
|
|
113
98
|
try {
|
|
114
99
|
fs_1.default.copyFileSync(fullPath, originalPath);
|
|
115
100
|
logger_1.logger.success(`Restored: ${relativePath}`);
|
|
@@ -125,4 +110,14 @@ class FileWriter {
|
|
|
125
110
|
}
|
|
126
111
|
}
|
|
127
112
|
exports.FileWriter = FileWriter;
|
|
128
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fileWriter.js","sourceRoot":"","sources":["../src/fileWriter.ts"],"names":[],"mappings":"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","sourcesContent":["import fs from 'fs';\nimport path from 'path';\nimport chalk from 'chalk';\nimport { logger } from './utils/logger';\n\nexport interface FileWriteOptions {\n  dryRun: boolean;\n  backup?: boolean;\n}\n\nexport class FileWriter {\n  private dryRun: boolean;\n  private backupDir: string;\n\n  constructor(options: FileWriteOptions) {\n    this.dryRun = options.dryRun;\n    this.backupDir = path.join(process.cwd(), '.css-to-tailwind-backups');\n  }\n\n  async writeFile(filePath: string, content: string, originalContent: string): Promise<boolean> {\n    if (this.dryRun) {\n      this.showDiff(filePath, originalContent, content);\n      return true;\n    }\n\n    try {\n      // Create backup\n      await this.createBackup(filePath, originalContent);\n\n      // Write file\n      fs.writeFileSync(filePath, content, 'utf-8');\n      logger.success(`✏️  Modified: ${path.relative(process.cwd(), filePath)}`);\n      return true;\n    } catch (error) {\n      logger.error(`Failed to write ${filePath}:`, error);\n      return false;\n    }\n  }\n\n  async deleteFile(filePath: string): Promise<boolean> {\n    if (this.dryRun) {\n      logger.info(`🗑️  Would delete: ${path.relative(process.cwd(), filePath)}`);\n      return true;\n    }\n\n    try {\n      // Create backup before deletion\n      const content = fs.readFileSync(filePath, 'utf-8');\n      await this.createBackup(filePath, content);\n\n      fs.unlinkSync(filePath);\n      logger.success(`🗑️  Deleted: ${path.relative(process.cwd(), filePath)}`);\n      return true;\n    } catch (error) {\n      logger.error(`Failed to delete ${filePath}:`, error);\n      return false;\n    }\n  }\n\n  private async createBackup(filePath: string, content: string): Promise<void> {\n    try {\n      // Create backup directory if it doesn't exist\n      if (!fs.existsSync(this.backupDir)) {\n        fs.mkdirSync(this.backupDir, { recursive: true });\n      }\n\n      const relativePath = path.relative(process.cwd(), filePath);\n      const backupPath = path.join(this.backupDir, relativePath);\n      const backupDir = path.dirname(backupPath);\n\n      // Create subdirectory structure\n      if (!fs.existsSync(backupDir)) {\n        fs.mkdirSync(backupDir, { recursive: true });\n      }\n\n      // Write backup\n      fs.writeFileSync(backupPath, content, 'utf-8');\n      logger.verbose(`Created backup: ${backupPath}`);\n    } catch (error) {\n      logger.warn(`Failed to create backup for ${filePath}:`, error);\n    }\n  }\n\n  private showDiff(filePath: string, original: string, modified: string): void {\n    const relativePath = path.relative(process.cwd(), filePath);\n    logger.info(`\\n📄 ${relativePath} (dry-run)`);\n    \n    const originalLines = original.split('\\n');\n    const modifiedLines = modified.split('\\n');\n\n    // Simple diff - show first few changed lines\n    const maxLines = Math.max(originalLines.length, modifiedLines.length);\n    let changes = 0;\n\n    for (let i = 0; i < maxLines && changes < 10; i++) {\n      const orig = originalLines[i] || '';\n      const mod = modifiedLines[i] || '';\n\n      if (orig !== mod) {\n        changes++;\n        console.log(chalk.red(`  - ${orig}`));\n        console.log(chalk.green(`  + ${mod}`));\n      }\n    }\n\n    if (changes === 0) {\n      logger.verbose('  (no visible changes)');\n    }\n  }\n\n  static restoreBackups(): void {\n    const backupDir = path.join(process.cwd(), '.css-to-tailwind-backups');\n    \n    if (!fs.existsSync(backupDir)) {\n      logger.warn('No backups found to restore');\n      return;\n    }\n\n    logger.info('🔄 Restoring files from backup...');\n    \n    // Recursively restore files\n    const restoreRecursive = (dir: string) => {\n      const items = fs.readdirSync(dir);\n\n      for (const item of items) {\n        const fullPath = path.join(dir, item);\n        const stat = fs.statSync(fullPath);\n\n        if (stat.isDirectory()) {\n          restoreRecursive(fullPath);\n        } else {\n          const relativePath = path.relative(backupDir, fullPath);\n          const originalPath = path.join(process.cwd(), relativePath);\n          \n          try {\n            fs.copyFileSync(fullPath, originalPath);\n            logger.success(`Restored: ${relativePath}`);\n          } catch (error) {\n            logger.error(`Failed to restore ${relativePath}:`, error);\n          }\n        }\n      }\n    };\n\n    restoreRecursive(backupDir);\n    logger.success('✅ Restore complete');\n  }\n}\n"]}
|
|
113
|
+
async function writeFiles(results, options) {
|
|
114
|
+
if (options.dryRun) {
|
|
115
|
+
return results.filter(r => r.hasChanges && r.status !== 'error').length;
|
|
116
|
+
}
|
|
117
|
+
const writer = new FileWriter({
|
|
118
|
+
dryRun: false,
|
|
119
|
+
projectRoot: options.projectRoot
|
|
120
|
+
});
|
|
121
|
+
return writer.writeResults(results);
|
|
122
|
+
}
|
|
123
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fileWriter.js","sourceRoot":"","sources":["../src/fileWriter.ts"],"names":[],"mappings":"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","sourcesContent":["import fs from 'fs';\nimport path from 'path';\nimport { logger } from './utils/logger';\nimport { FileResult } from './utils/reporter';\n\nexport interface FileWriteOptions {\n  dryRun: boolean;\n  backup?: boolean;\n}\n\nexport class FileWriter {\n  private dryRun: boolean;\n  private backupDir: string;\n  private projectRoot: string;\n\n  constructor(options: FileWriteOptions & { projectRoot?: string }) {\n    this.dryRun = options.dryRun;\n    this.projectRoot = options.projectRoot || process.cwd();\n    this.backupDir = path.join(this.projectRoot, '.css-to-tailwind-backups');\n  }\n\n  async writeFile(filePath: string, content: string, originalContent: string): Promise<boolean> {\n    if (this.dryRun) {\n      return true;\n    }\n\n    try {\n      await this.createBackup(filePath, originalContent);\n      fs.writeFileSync(filePath, content, 'utf-8');\n      logger.success(`✏️  Modified: ${path.relative(this.projectRoot, filePath)}`);\n      return true;\n    } catch (error) {\n      logger.error(`Failed to write ${filePath}:`, error);\n      return false;\n    }\n  }\n\n  async deleteFile(filePath: string): Promise<boolean> {\n    if (this.dryRun) {\n      logger.info(`🗑️  Would delete: ${path.relative(this.projectRoot, filePath)}`);\n      return true;\n    }\n\n    try {\n      const content = fs.readFileSync(filePath, 'utf-8');\n      await this.createBackup(filePath, content);\n      fs.unlinkSync(filePath);\n      logger.success(`🗑️  Deleted: ${path.relative(this.projectRoot, filePath)}`);\n      return true;\n    } catch (error) {\n      logger.error(`Failed to delete ${filePath}:`, error);\n      return false;\n    }\n  }\n\n  async writeResults(results: FileResult[]): Promise<number> {\n    let written = 0;\n    \n    for (const result of results) {\n      if (result.status === 'error') continue;\n      if (!result.hasChanges) continue;\n      \n      const success = await this.writeFile(\n        result.filePath,\n        result.newContent,\n        result.originalContent\n      );\n      \n      if (success) written++;\n    }\n    \n    return written;\n  }\n\n  private async createBackup(filePath: string, content: string): Promise<void> {\n    try {\n      if (!fs.existsSync(this.backupDir)) {\n        fs.mkdirSync(this.backupDir, { recursive: true });\n      }\n\n      const relativePath = path.relative(this.projectRoot, filePath);\n      const backupPath = path.join(this.backupDir, relativePath);\n      const backupDir = path.dirname(backupPath);\n\n      if (!fs.existsSync(backupDir)) {\n        fs.mkdirSync(backupDir, { recursive: true });\n      }\n\n      fs.writeFileSync(backupPath, content, 'utf-8');\n      logger.verbose(`Created backup: ${backupPath}`);\n    } catch (error) {\n      logger.warn(`Failed to create backup for ${filePath}:`, error);\n    }\n  }\n\n  static restoreBackups(projectRoot: string = process.cwd()): void {\n    const backupDir = path.join(projectRoot, '.css-to-tailwind-backups');\n    \n    if (!fs.existsSync(backupDir)) {\n      logger.warn('No backups found to restore');\n      return;\n    }\n\n    logger.info('🔄 Restoring files from backup...');\n    \n    const restoreRecursive = (dir: string) => {\n      const items = fs.readdirSync(dir);\n\n      for (const item of items) {\n        const fullPath = path.join(dir, item);\n        const stat = fs.statSync(fullPath);\n\n        if (stat.isDirectory()) {\n          restoreRecursive(fullPath);\n        } else {\n          const relativePath = path.relative(backupDir, fullPath);\n          const originalPath = path.join(projectRoot, relativePath);\n          \n          try {\n            fs.copyFileSync(fullPath, originalPath);\n            logger.success(`Restored: ${relativePath}`);\n          } catch (error) {\n            logger.error(`Failed to restore ${relativePath}:`, error);\n          }\n        }\n      }\n    };\n\n    restoreRecursive(backupDir);\n    logger.success('✅ Restore complete');\n  }\n}\n\nexport async function writeFiles(\n  results: FileResult[],\n  options: { dryRun: boolean; projectRoot: string }\n): Promise<number> {\n  if (options.dryRun) {\n    return results.filter(r => r.hasChanges && r.status !== 'error').length;\n  }\n  \n  const writer = new FileWriter({ \n    dryRun: false, \n    projectRoot: options.projectRoot \n  });\n  \n  return writer.writeResults(results);\n}\n"]}
|
package/dist/index.d.ts
CHANGED
|
@@ -1,11 +1,13 @@
|
|
|
1
1
|
export { scanProject, ScannedFile } from './scanner';
|
|
2
|
-
export { transformFiles, TransformOptions, TransformResults } from './transformer';
|
|
2
|
+
export { transformFiles, transformFilesDetailed, TransformOptions, TransformResults, DetailedTransformResults } from './transformer';
|
|
3
3
|
export { TailwindMapper, CSSProperty, ConversionResult } from './tailwindMapper';
|
|
4
4
|
export { JSXParser, JSXTransformation, JSXParseResult } from './jsxParser';
|
|
5
5
|
export { CSSParser, CSSRule, CSSParseResult, UtilityWithVariant, SelectorTarget } from './cssParser';
|
|
6
|
-
export { FileWriter, FileWriteOptions } from './fileWriter';
|
|
6
|
+
export { FileWriter, FileWriteOptions, writeFiles } from './fileWriter';
|
|
7
7
|
export { loadTailwindConfig, TailwindConfig } from './utils/config';
|
|
8
8
|
export { logger } from './utils/logger';
|
|
9
|
+
export { Reporter, FileResult, SummaryStats, ReporterOptions } from './utils/reporter';
|
|
10
|
+
export { computeUnifiedDiff, formatDiff, computeAndFormatDiff, getChangeStats, DiffResult, DiffHunk, DiffLine } from './utils/diff';
|
|
9
11
|
export { Breakpoint, MediaQueryInfo, getDefaultBreakpoints, resolveBreakpointsFromConfig, parseMediaQuery, findBreakpointForMinWidth, processMediaQuery, prefixWithBreakpoint } from './utils/breakpointResolver';
|
|
10
12
|
export { ParsedSelector, PSEUDO_TO_VARIANT, SUPPORTED_PSEUDOS, parseSelector, mapPseudoToVariant, processPseudoSelector, parseMultipleSelectors } from './utils/pseudoSelectorResolver';
|
|
11
13
|
export { VARIANT_ORDER, isResponsiveVariant, isPseudoVariant, sortVariants, deduplicateVariants, normalizeVariantOrder, assembleUtility, assembleUtilities, mergeUtilities, MergedUtility } from './utils/variantAssembler';
|
package/dist/index.js
CHANGED
|
@@ -1,11 +1,12 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.groupDescendantRulesByParent = exports.transformDescendantSelectors = exports.isHtmlElement = exports.processDescendantSelector = exports.isSimpleSelector = exports.isDescendantSelector = exports.parseDescendantSelector = exports.mergeUtilities = exports.assembleUtilities = exports.assembleUtility = exports.normalizeVariantOrder = exports.deduplicateVariants = exports.sortVariants = exports.isPseudoVariant = exports.isResponsiveVariant = exports.VARIANT_ORDER = exports.parseMultipleSelectors = exports.processPseudoSelector = exports.mapPseudoToVariant = exports.parseSelector = exports.SUPPORTED_PSEUDOS = exports.PSEUDO_TO_VARIANT = exports.prefixWithBreakpoint = exports.processMediaQuery = exports.findBreakpointForMinWidth = exports.parseMediaQuery = exports.resolveBreakpointsFromConfig = exports.getDefaultBreakpoints = exports.logger = exports.loadTailwindConfig = exports.FileWriter = exports.CSSParser = exports.JSXParser = exports.TailwindMapper = exports.transformFiles = exports.scanProject = void 0;
|
|
3
|
+
exports.groupDescendantRulesByParent = exports.transformDescendantSelectors = exports.isHtmlElement = exports.processDescendantSelector = exports.isSimpleSelector = exports.isDescendantSelector = exports.parseDescendantSelector = exports.mergeUtilities = exports.assembleUtilities = exports.assembleUtility = exports.normalizeVariantOrder = exports.deduplicateVariants = exports.sortVariants = exports.isPseudoVariant = exports.isResponsiveVariant = exports.VARIANT_ORDER = exports.parseMultipleSelectors = exports.processPseudoSelector = exports.mapPseudoToVariant = exports.parseSelector = exports.SUPPORTED_PSEUDOS = exports.PSEUDO_TO_VARIANT = exports.prefixWithBreakpoint = exports.processMediaQuery = exports.findBreakpointForMinWidth = exports.parseMediaQuery = exports.resolveBreakpointsFromConfig = exports.getDefaultBreakpoints = exports.getChangeStats = exports.computeAndFormatDiff = exports.formatDiff = exports.computeUnifiedDiff = exports.Reporter = exports.logger = exports.loadTailwindConfig = exports.writeFiles = exports.FileWriter = exports.CSSParser = exports.JSXParser = exports.TailwindMapper = exports.transformFilesDetailed = exports.transformFiles = exports.scanProject = void 0;
|
|
4
4
|
// Export public API
|
|
5
5
|
var scanner_1 = require("./scanner");
|
|
6
6
|
Object.defineProperty(exports, "scanProject", { enumerable: true, get: function () { return scanner_1.scanProject; } });
|
|
7
7
|
var transformer_1 = require("./transformer");
|
|
8
8
|
Object.defineProperty(exports, "transformFiles", { enumerable: true, get: function () { return transformer_1.transformFiles; } });
|
|
9
|
+
Object.defineProperty(exports, "transformFilesDetailed", { enumerable: true, get: function () { return transformer_1.transformFilesDetailed; } });
|
|
9
10
|
var tailwindMapper_1 = require("./tailwindMapper");
|
|
10
11
|
Object.defineProperty(exports, "TailwindMapper", { enumerable: true, get: function () { return tailwindMapper_1.TailwindMapper; } });
|
|
11
12
|
var jsxParser_1 = require("./jsxParser");
|
|
@@ -14,10 +15,18 @@ var cssParser_1 = require("./cssParser");
|
|
|
14
15
|
Object.defineProperty(exports, "CSSParser", { enumerable: true, get: function () { return cssParser_1.CSSParser; } });
|
|
15
16
|
var fileWriter_1 = require("./fileWriter");
|
|
16
17
|
Object.defineProperty(exports, "FileWriter", { enumerable: true, get: function () { return fileWriter_1.FileWriter; } });
|
|
18
|
+
Object.defineProperty(exports, "writeFiles", { enumerable: true, get: function () { return fileWriter_1.writeFiles; } });
|
|
17
19
|
var config_1 = require("./utils/config");
|
|
18
20
|
Object.defineProperty(exports, "loadTailwindConfig", { enumerable: true, get: function () { return config_1.loadTailwindConfig; } });
|
|
19
21
|
var logger_1 = require("./utils/logger");
|
|
20
22
|
Object.defineProperty(exports, "logger", { enumerable: true, get: function () { return logger_1.logger; } });
|
|
23
|
+
var reporter_1 = require("./utils/reporter");
|
|
24
|
+
Object.defineProperty(exports, "Reporter", { enumerable: true, get: function () { return reporter_1.Reporter; } });
|
|
25
|
+
var diff_1 = require("./utils/diff");
|
|
26
|
+
Object.defineProperty(exports, "computeUnifiedDiff", { enumerable: true, get: function () { return diff_1.computeUnifiedDiff; } });
|
|
27
|
+
Object.defineProperty(exports, "formatDiff", { enumerable: true, get: function () { return diff_1.formatDiff; } });
|
|
28
|
+
Object.defineProperty(exports, "computeAndFormatDiff", { enumerable: true, get: function () { return diff_1.computeAndFormatDiff; } });
|
|
29
|
+
Object.defineProperty(exports, "getChangeStats", { enumerable: true, get: function () { return diff_1.getChangeStats; } });
|
|
21
30
|
var breakpointResolver_1 = require("./utils/breakpointResolver");
|
|
22
31
|
Object.defineProperty(exports, "getDefaultBreakpoints", { enumerable: true, get: function () { return breakpointResolver_1.getDefaultBreakpoints; } });
|
|
23
32
|
Object.defineProperty(exports, "resolveBreakpointsFromConfig", { enumerable: true, get: function () { return breakpointResolver_1.resolveBreakpointsFromConfig; } });
|
|
@@ -51,4 +60,4 @@ Object.defineProperty(exports, "isHtmlElement", { enumerable: true, get: functio
|
|
|
51
60
|
var jsxDescendantTransformer_1 = require("./jsxDescendantTransformer");
|
|
52
61
|
Object.defineProperty(exports, "transformDescendantSelectors", { enumerable: true, get: function () { return jsxDescendantTransformer_1.transformDescendantSelectors; } });
|
|
53
62
|
Object.defineProperty(exports, "groupDescendantRulesByParent", { enumerable: true, get: function () { return jsxDescendantTransformer_1.groupDescendantRulesByParent; } });
|
|
54
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
63
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsb0JBQW9CO0FBQ3BCLHFDQUFxRDtBQUE1QyxzR0FBQSxXQUFXLE9BQUE7QUFDcEIsNkNBQXFJO0FBQTVILDZHQUFBLGNBQWMsT0FBQTtBQUFFLHFIQUFBLHNCQUFzQixPQUFBO0FBQy9DLG1EQUFpRjtBQUF4RSxnSEFBQSxjQUFjLE9BQUE7QUFDdkIseUNBQTJFO0FBQWxFLHNHQUFBLFNBQVMsT0FBQTtBQUNsQix5Q0FBcUc7QUFBNUYsc0dBQUEsU0FBUyxPQUFBO0FBQ2xCLDJDQUF3RTtBQUEvRCx3R0FBQSxVQUFVLE9BQUE7QUFBb0Isd0dBQUEsVUFBVSxPQUFBO0FBQ2pELHlDQUFvRTtBQUEzRCw0R0FBQSxrQkFBa0IsT0FBQTtBQUMzQix5Q0FBd0M7QUFBL0IsZ0dBQUEsTUFBTSxPQUFBO0FBQ2YsNkNBQXVGO0FBQTlFLG9HQUFBLFFBQVEsT0FBQTtBQUNqQixxQ0FRc0I7QUFQcEIsMEdBQUEsa0JBQWtCLE9BQUE7QUFDbEIsa0dBQUEsVUFBVSxPQUFBO0FBQ1YsNEdBQUEsb0JBQW9CLE9BQUE7QUFDcEIsc0dBQUEsY0FBYyxPQUFBO0FBS2hCLGlFQVNvQztBQU5sQywySEFBQSxxQkFBcUIsT0FBQTtBQUNyQixrSUFBQSw0QkFBNEIsT0FBQTtBQUM1QixxSEFBQSxlQUFlLE9BQUE7QUFDZiwrSEFBQSx5QkFBeUIsT0FBQTtBQUN6Qix1SEFBQSxpQkFBaUIsT0FBQTtBQUNqQiwwSEFBQSxvQkFBb0IsT0FBQTtBQUV0Qix5RUFRd0M7QUFOdEMsMkhBQUEsaUJBQWlCLE9BQUE7QUFDakIsMkhBQUEsaUJBQWlCLE9BQUE7QUFDakIsdUhBQUEsYUFBYSxPQUFBO0FBQ2IsNEhBQUEsa0JBQWtCLE9BQUE7QUFDbEIsK0hBQUEscUJBQXFCLE9BQUE7QUFDckIsZ0lBQUEsc0JBQXNCLE9BQUE7QUFFeEIsNkRBV2tDO0FBVmhDLGlIQUFBLGFBQWEsT0FBQTtBQUNiLHVIQUFBLG1CQUFtQixPQUFBO0FBQ25CLG1IQUFBLGVBQWUsT0FBQTtBQUNmLGdIQUFBLFlBQVksT0FBQTtBQUNaLHVIQUFBLG1CQUFtQixPQUFBO0FBQ25CLHlIQUFBLHFCQUFxQixPQUFBO0FBQ3JCLG1IQUFBLGVBQWUsT0FBQTtBQUNmLHFIQUFBLGlCQUFpQixPQUFBO0FBQ2pCLGtIQUFBLGNBQWMsT0FBQTtBQUdoQixpRkFTNEM7QUFMMUMscUlBQUEsdUJBQXVCLE9BQUE7QUFDdkIsa0lBQUEsb0JBQW9CLE9BQUE7QUFDcEIsOEhBQUEsZ0JBQWdCLE9BQUE7QUFDaEIsdUlBQUEseUJBQXlCLE9BQUE7QUFDekIsMkhBQUEsYUFBYSxPQUFBO0FBRWYsdUVBSW9DO0FBSGxDLHdJQUFBLDRCQUE0QixPQUFBO0FBRTVCLHdJQUFBLDRCQUE0QixPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLy8gRXhwb3J0IHB1YmxpYyBBUElcbmV4cG9ydCB7IHNjYW5Qcm9qZWN0LCBTY2FubmVkRmlsZSB9IGZyb20gJy4vc2Nhbm5lcic7XG5leHBvcnQgeyB0cmFuc2Zvcm1GaWxlcywgdHJhbnNmb3JtRmlsZXNEZXRhaWxlZCwgVHJhbnNmb3JtT3B0aW9ucywgVHJhbnNmb3JtUmVzdWx0cywgRGV0YWlsZWRUcmFuc2Zvcm1SZXN1bHRzIH0gZnJvbSAnLi90cmFuc2Zvcm1lcic7XG5leHBvcnQgeyBUYWlsd2luZE1hcHBlciwgQ1NTUHJvcGVydHksIENvbnZlcnNpb25SZXN1bHQgfSBmcm9tICcuL3RhaWx3aW5kTWFwcGVyJztcbmV4cG9ydCB7IEpTWFBhcnNlciwgSlNYVHJhbnNmb3JtYXRpb24sIEpTWFBhcnNlUmVzdWx0IH0gZnJvbSAnLi9qc3hQYXJzZXInO1xuZXhwb3J0IHsgQ1NTUGFyc2VyLCBDU1NSdWxlLCBDU1NQYXJzZVJlc3VsdCwgVXRpbGl0eVdpdGhWYXJpYW50LCBTZWxlY3RvclRhcmdldCB9IGZyb20gJy4vY3NzUGFyc2VyJztcbmV4cG9ydCB7IEZpbGVXcml0ZXIsIEZpbGVXcml0ZU9wdGlvbnMsIHdyaXRlRmlsZXMgfSBmcm9tICcuL2ZpbGVXcml0ZXInO1xuZXhwb3J0IHsgbG9hZFRhaWx3aW5kQ29uZmlnLCBUYWlsd2luZENvbmZpZyB9IGZyb20gJy4vdXRpbHMvY29uZmlnJztcbmV4cG9ydCB7IGxvZ2dlciB9IGZyb20gJy4vdXRpbHMvbG9nZ2VyJztcbmV4cG9ydCB7IFJlcG9ydGVyLCBGaWxlUmVzdWx0LCBTdW1tYXJ5U3RhdHMsIFJlcG9ydGVyT3B0aW9ucyB9IGZyb20gJy4vdXRpbHMvcmVwb3J0ZXInO1xuZXhwb3J0IHsgXG4gIGNvbXB1dGVVbmlmaWVkRGlmZiwgXG4gIGZvcm1hdERpZmYsIFxuICBjb21wdXRlQW5kRm9ybWF0RGlmZiwgXG4gIGdldENoYW5nZVN0YXRzLFxuICBEaWZmUmVzdWx0LFxuICBEaWZmSHVuayxcbiAgRGlmZkxpbmVcbn0gZnJvbSAnLi91dGlscy9kaWZmJztcbmV4cG9ydCB7XG4gIEJyZWFrcG9pbnQsXG4gIE1lZGlhUXVlcnlJbmZvLFxuICBnZXREZWZhdWx0QnJlYWtwb2ludHMsXG4gIHJlc29sdmVCcmVha3BvaW50c0Zyb21Db25maWcsXG4gIHBhcnNlTWVkaWFRdWVyeSxcbiAgZmluZEJyZWFrcG9pbnRGb3JNaW5XaWR0aCxcbiAgcHJvY2Vzc01lZGlhUXVlcnksXG4gIHByZWZpeFdpdGhCcmVha3BvaW50XG59IGZyb20gJy4vdXRpbHMvYnJlYWtwb2ludFJlc29sdmVyJztcbmV4cG9ydCB7XG4gIFBhcnNlZFNlbGVjdG9yLFxuICBQU0VVRE9fVE9fVkFSSUFOVCxcbiAgU1VQUE9SVEVEX1BTRVVET1MsXG4gIHBhcnNlU2VsZWN0b3IsXG4gIG1hcFBzZXVkb1RvVmFyaWFudCxcbiAgcHJvY2Vzc1BzZXVkb1NlbGVjdG9yLFxuICBwYXJzZU11bHRpcGxlU2VsZWN0b3JzXG59IGZyb20gJy4vdXRpbHMvcHNldWRvU2VsZWN0b3JSZXNvbHZlcic7XG5leHBvcnQge1xuICBWQVJJQU5UX09SREVSLFxuICBpc1Jlc3BvbnNpdmVWYXJpYW50LFxuICBpc1BzZXVkb1ZhcmlhbnQsXG4gIHNvcnRWYXJpYW50cyxcbiAgZGVkdXBsaWNhdGVWYXJpYW50cyxcbiAgbm9ybWFsaXplVmFyaWFudE9yZGVyLFxuICBhc3NlbWJsZVV0aWxpdHksXG4gIGFzc2VtYmxlVXRpbGl0aWVzLFxuICBtZXJnZVV0aWxpdGllcyxcbiAgTWVyZ2VkVXRpbGl0eVxufSBmcm9tICcuL3V0aWxzL3ZhcmlhbnRBc3NlbWJsZXInO1xuZXhwb3J0IHtcbiAgRGVzY2VuZGFudFNlbGVjdG9yLFxuICBTZWxlY3RvclBhcnQsXG4gIFNlbGVjdG9yVHlwZSxcbiAgcGFyc2VEZXNjZW5kYW50U2VsZWN0b3IsXG4gIGlzRGVzY2VuZGFudFNlbGVjdG9yLFxuICBpc1NpbXBsZVNlbGVjdG9yLFxuICBwcm9jZXNzRGVzY2VuZGFudFNlbGVjdG9yLFxuICBpc0h0bWxFbGVtZW50XG59IGZyb20gJy4vdXRpbHMvZGVzY2VuZGFudFNlbGVjdG9yUmVzb2x2ZXInO1xuZXhwb3J0IHtcbiAgdHJhbnNmb3JtRGVzY2VuZGFudFNlbGVjdG9ycyxcbiAgRGVzY2VuZGFudFRyYW5zZm9ybVJlc3VsdCxcbiAgZ3JvdXBEZXNjZW5kYW50UnVsZXNCeVBhcmVudFxufSBmcm9tICcuL2pzeERlc2NlbmRhbnRUcmFuc2Zvcm1lcic7XG4iXX0=
|
package/dist/transformer.d.ts
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { ScannedFile } from './scanner';
|
|
2
2
|
import { TailwindConfig } from './utils/config';
|
|
3
|
+
import { FileResult, SummaryStats } from './utils/reporter';
|
|
3
4
|
export interface TransformOptions {
|
|
4
5
|
dryRun: boolean;
|
|
5
6
|
deleteCss: boolean;
|
|
@@ -16,4 +17,9 @@ export interface TransformResults {
|
|
|
16
17
|
classesReplaced: number;
|
|
17
18
|
warnings: number;
|
|
18
19
|
}
|
|
20
|
+
export interface DetailedTransformResults {
|
|
21
|
+
fileResults: FileResult[];
|
|
22
|
+
stats: SummaryStats;
|
|
23
|
+
}
|
|
19
24
|
export declare function transformFiles(files: ScannedFile[], options: TransformOptions): Promise<TransformResults>;
|
|
25
|
+
export declare function transformFilesDetailed(files: ScannedFile[], options: TransformOptions): Promise<DetailedTransformResults>;
|