css-to-tailwind-react 0.1.0 → 0.1.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,9 +1,14 @@
1
1
  import { TailwindMapper, CSSProperty } from './tailwindMapper';
2
+ export interface UtilityWithVariant {
3
+ value: string;
4
+ variants: string[];
5
+ }
2
6
  export interface CSSRule {
3
7
  selector: string;
4
8
  className: string;
5
9
  declarations: CSSProperty[];
6
10
  convertedClasses: string[];
11
+ utilities: UtilityWithVariant[];
7
12
  skipped: boolean;
8
13
  fullyConverted: boolean;
9
14
  partialConversion: boolean;
@@ -21,7 +26,10 @@ export interface CSSUsageMap {
21
26
  }
22
27
  export declare class CSSParser {
23
28
  private mapper;
24
- constructor(mapper: TailwindMapper);
29
+ private breakpoints;
30
+ constructor(mapper: TailwindMapper, screens?: Record<string, string | [string, string]>);
31
+ private convertDeclarations;
32
+ private processRuleWithVariants;
25
33
  parse(css: string, filePath: string): Promise<CSSParseResult>;
26
34
  parseInternalStyle(html: string): {
27
35
  styles: Array<{
package/dist/cssParser.js CHANGED
@@ -7,9 +7,93 @@ exports.CSSParser = void 0;
7
7
  const postcss_1 = __importDefault(require("postcss"));
8
8
  const postcss_safe_parser_1 = __importDefault(require("postcss-safe-parser"));
9
9
  const logger_1 = require("./utils/logger");
10
+ const breakpointResolver_1 = require("./utils/breakpointResolver");
11
+ const pseudoSelectorResolver_1 = require("./utils/pseudoSelectorResolver");
12
+ const variantAssembler_1 = require("./utils/variantAssembler");
10
13
  class CSSParser {
11
- constructor(mapper) {
14
+ constructor(mapper, screens) {
12
15
  this.mapper = mapper;
16
+ this.breakpoints = screens
17
+ ? (0, breakpointResolver_1.resolveBreakpointsFromConfig)(screens)
18
+ : (0, breakpointResolver_1.getBreakpoints)();
19
+ }
20
+ convertDeclarations(declarations) {
21
+ const conversionResults = [];
22
+ const conversionWarnings = [];
23
+ declarations.forEach(decl => {
24
+ const result = this.mapper.convertProperty(decl.property, decl.value);
25
+ conversionResults.push({
26
+ declaration: decl,
27
+ converted: !result.skipped && result.className !== null,
28
+ className: result.className
29
+ });
30
+ if (result.skipped && result.reason) {
31
+ conversionWarnings.push(result.reason);
32
+ }
33
+ });
34
+ const utilities = conversionResults
35
+ .filter(r => r.converted && r.className)
36
+ .map(r => ({
37
+ value: r.className,
38
+ variants: []
39
+ }));
40
+ return { utilities, conversionResults, conversionWarnings };
41
+ }
42
+ processRuleWithVariants(rule, additionalVariants = []) {
43
+ const selector = rule.selector;
44
+ const parsedSelectors = (0, pseudoSelectorResolver_1.parseMultipleSelectors)(selector);
45
+ const validSelectors = parsedSelectors.filter(s => !s.isComplex && s.baseClass);
46
+ if (validSelectors.length === 0) {
47
+ return null;
48
+ }
49
+ const declarations = [];
50
+ rule.walkDecls((decl) => {
51
+ if (decl.prop.startsWith('--')) {
52
+ return;
53
+ }
54
+ if (decl.value.includes('calc(')) {
55
+ return;
56
+ }
57
+ declarations.push({
58
+ property: decl.prop,
59
+ value: decl.value
60
+ });
61
+ });
62
+ if (declarations.length === 0) {
63
+ return null;
64
+ }
65
+ const { utilities, conversionResults, conversionWarnings } = this.convertDeclarations(declarations);
66
+ const utilitiesWithVariants = utilities.map(u => ({
67
+ value: u.value,
68
+ variants: (0, variantAssembler_1.normalizeVariantOrder)([...u.variants, ...additionalVariants])
69
+ }));
70
+ const cssRules = [];
71
+ const allConversionResults = [];
72
+ for (const parsed of validSelectors) {
73
+ const pseudoVariants = parsed.pseudos || [];
74
+ const allVariants = (0, variantAssembler_1.normalizeVariantOrder)([...pseudoVariants, ...additionalVariants]);
75
+ const utilitiesForSelector = utilities.map(u => ({
76
+ value: u.value,
77
+ variants: allVariants
78
+ }));
79
+ const convertedClasses = (0, variantAssembler_1.assembleUtilities)(utilitiesForSelector);
80
+ const allDeclarationsConverted = conversionResults.every(r => r.converted);
81
+ const someDeclarationsConverted = convertedClasses.length > 0;
82
+ const cssRule = {
83
+ selector: selector,
84
+ className: parsed.baseClass,
85
+ declarations,
86
+ convertedClasses,
87
+ utilities: utilitiesForSelector,
88
+ skipped: !someDeclarationsConverted,
89
+ fullyConverted: allDeclarationsConverted,
90
+ partialConversion: someDeclarationsConverted && !allDeclarationsConverted,
91
+ reason: !someDeclarationsConverted ? 'No convertible declarations' : undefined
92
+ };
93
+ cssRules.push(cssRule);
94
+ allConversionResults.push(conversionResults);
95
+ }
96
+ return { cssRules, conversionResults: allConversionResults, conversionWarnings };
13
97
  }
14
98
  async parse(css, filePath) {
15
99
  const rules = [];
@@ -20,105 +104,97 @@ class CSSParser {
20
104
  parser: postcss_safe_parser_1.default,
21
105
  from: filePath
22
106
  }).then(result => result.root);
23
- // Process each rule
24
- root.walkRules((rule) => {
25
- // Skip rules inside @media, @supports, etc.
26
- if (rule.parent && rule.parent.type === 'atrule') {
27
- warnings.push(`Skipped rule with at-rule parent: ${rule.selector}`);
28
- logger_1.logger.verbose(`Skipping at-rule: ${rule.selector}`);
29
- return;
30
- }
31
- // Skip pseudo-selectors
32
- if (rule.selector.includes(':')) {
33
- warnings.push(`Skipped pseudo-selector: ${rule.selector}`);
34
- logger_1.logger.verbose(`Skipping pseudo-selector: ${rule.selector}`);
107
+ root.walkAtRules((atRule) => {
108
+ if (atRule.name !== 'media') {
35
109
  return;
36
110
  }
37
- // Only process simple class selectors
38
- const classNameMatch = rule.selector.match(/^\.([a-zA-Z_-][a-zA-Z0-9_-]*)$/);
39
- if (!classNameMatch) {
40
- warnings.push(`Skipped complex selector: ${rule.selector}`);
41
- logger_1.logger.verbose(`Skipping complex selector: ${rule.selector}`);
111
+ const mediaResult = (0, breakpointResolver_1.processMediaQuery)(atRule.params, this.breakpoints);
112
+ if (mediaResult.skipped) {
113
+ warnings.push(mediaResult.reason || `Skipped media query: ${atRule.params}`);
42
114
  return;
43
115
  }
44
- const className = classNameMatch[1];
45
- const declarations = [];
46
- rule.walkDecls((decl) => {
47
- // Skip CSS variables
48
- if (decl.prop.startsWith('--')) {
49
- warnings.push(`Skipped CSS variable: ${decl.prop}`);
50
- return;
116
+ const responsiveVariant = mediaResult.breakpoint;
117
+ const nestedRules = [];
118
+ atRule.walkRules((rule) => {
119
+ nestedRules.push(rule);
120
+ });
121
+ for (const rule of nestedRules) {
122
+ const result = this.processRuleWithVariants(rule, [responsiveVariant]);
123
+ if (result) {
124
+ rules.push(...result.cssRules);
125
+ warnings.push(...result.conversionWarnings);
126
+ const anyConverted = result.cssRules.some(r => r.convertedClasses.length > 0);
127
+ if (anyConverted) {
128
+ hasChanges = true;
129
+ const allFullyConverted = result.cssRules.every(r => r.fullyConverted);
130
+ if (allFullyConverted) {
131
+ rule.remove();
132
+ const classNames = result.cssRules.map(r => r.className).join(', .');
133
+ logger_1.logger.verbose(`Removed rule .${classNames} in @media (min-width) → ${responsiveVariant}`);
134
+ }
135
+ else {
136
+ for (const cr of result.conversionResults.flat()) {
137
+ if (cr.converted) {
138
+ rule.walkDecls((decl) => {
139
+ if (decl.prop === cr.declaration.property && decl.value === cr.declaration.value) {
140
+ decl.remove();
141
+ }
142
+ });
143
+ }
144
+ }
145
+ logger_1.logger.verbose(`Partial conversion in @media → ${responsiveVariant}`);
146
+ }
147
+ }
51
148
  }
52
- // Skip calc()
53
- if (decl.value.includes('calc(')) {
54
- warnings.push(`Skipped calc() value: ${decl.value}`);
55
- return;
149
+ else {
150
+ warnings.push(`Skipped rule in @media: ${rule.selector}`);
56
151
  }
57
- declarations.push({
58
- property: decl.prop,
59
- value: decl.value
60
- });
61
- });
62
- if (declarations.length === 0) {
152
+ }
153
+ if (atRule.nodes && atRule.nodes.length === 0) {
154
+ atRule.remove();
155
+ logger_1.logger.verbose(`Removed empty @media rule`);
156
+ }
157
+ });
158
+ root.walkRules((rule) => {
159
+ if (rule.parent && rule.parent.type === 'atrule') {
63
160
  return;
64
161
  }
65
- // Convert to Tailwind classes - track which specific declarations were converted
66
- const conversionResults = [];
67
- const conversionWarnings = [];
68
- declarations.forEach(decl => {
69
- const result = this.mapper.convertProperty(decl.property, decl.value);
70
- conversionResults.push({
71
- declaration: decl,
72
- converted: !result.skipped && result.className !== null,
73
- className: result.className
74
- });
75
- if (result.skipped && result.reason) {
76
- conversionWarnings.push(result.reason);
162
+ const result = this.processRuleWithVariants(rule);
163
+ if (!result) {
164
+ const parsedSelectors = (0, pseudoSelectorResolver_1.parseMultipleSelectors)(rule.selector);
165
+ const allComplex = parsedSelectors.every(s => s.isComplex);
166
+ if (allComplex) {
167
+ const reasons = parsedSelectors.map(s => s.reason).filter(Boolean);
168
+ warnings.push(...reasons);
169
+ logger_1.logger.verbose(`Skipping complex selector: ${rule.selector}`);
77
170
  }
78
- });
79
- const convertedClasses = conversionResults
80
- .filter(r => r.converted && r.className)
81
- .map(r => r.className);
82
- const allDeclarationsConverted = conversionResults.every(r => r.converted);
83
- const someDeclarationsConverted = convertedClasses.length > 0;
84
- const cssRule = {
85
- selector: rule.selector,
86
- className,
87
- declarations,
88
- convertedClasses,
89
- skipped: !someDeclarationsConverted,
90
- fullyConverted: allDeclarationsConverted,
91
- partialConversion: someDeclarationsConverted && !allDeclarationsConverted,
92
- reason: !someDeclarationsConverted ? 'No convertible declarations' : undefined
93
- };
94
- rules.push(cssRule);
95
- warnings.push(...conversionWarnings);
96
- // CRITICAL FIX: Only remove declarations that were successfully converted
97
- // Never remove the entire rule unless ALL declarations are converted
98
- if (someDeclarationsConverted) {
171
+ return;
172
+ }
173
+ rules.push(...result.cssRules);
174
+ warnings.push(...result.conversionWarnings);
175
+ const anyConverted = result.cssRules.some(r => r.convertedClasses.length > 0);
176
+ if (anyConverted) {
99
177
  hasChanges = true;
100
- if (allDeclarationsConverted) {
101
- // All declarations converted - safe to remove entire rule
178
+ const allFullyConverted = result.cssRules.every(r => r.fullyConverted);
179
+ if (allFullyConverted) {
102
180
  rule.remove();
103
- logger_1.logger.verbose(`Removed rule .${className} (all ${declarations.length} declarations converted)`);
181
+ const classNames = result.cssRules.map(r => r.className).join(', .');
182
+ logger_1.logger.verbose(`Removed rule .${classNames} (all declarations converted)`);
104
183
  }
105
184
  else {
106
- // Partial conversion - only remove the converted declarations
107
- let removedCount = 0;
108
- rule.walkDecls((decl) => {
109
- const wasConverted = conversionResults.some(r => r.converted &&
110
- r.declaration.property === decl.prop &&
111
- r.declaration.value === decl.value);
112
- if (wasConverted) {
113
- decl.remove();
114
- removedCount++;
185
+ for (const cr of result.conversionResults.flat()) {
186
+ if (cr.converted) {
187
+ rule.walkDecls((decl) => {
188
+ if (decl.prop === cr.declaration.property && decl.value === cr.declaration.value) {
189
+ decl.remove();
190
+ }
191
+ });
115
192
  }
116
- });
117
- logger_1.logger.verbose(`Partial conversion of .${className}: removed ${removedCount}/${declarations.length} declarations`);
193
+ }
194
+ logger_1.logger.verbose(`Partial conversion of rule`);
118
195
  }
119
196
  }
120
197
  });
121
- // Clean up empty at-rules
122
198
  root.walkAtRules((atRule) => {
123
199
  if (atRule.nodes && atRule.nodes.length === 0) {
124
200
  atRule.remove();
@@ -142,7 +218,6 @@ class CSSParser {
142
218
  parseInternalStyle(html) {
143
219
  const styles = [];
144
220
  const warnings = [];
145
- // Simple regex to find style tags (this is safe for finding tags, not for parsing content)
146
221
  const styleRegex = /<style[^>]*>([\s\S]*?)<\/style>/gi;
147
222
  let match;
148
223
  while ((match = styleRegex.exec(html)) !== null) {
@@ -160,7 +235,6 @@ class CSSParser {
160
235
  let modifiedHtml = html;
161
236
  let hasChanges = false;
162
237
  const { styles } = this.parseInternalStyle(html);
163
- // Process styles in reverse order to preserve indices
164
238
  for (let i = styles.length - 1; i >= 0; i--) {
165
239
  const style = styles[i];
166
240
  try {
@@ -170,11 +244,9 @@ class CSSParser {
170
244
  if (result.hasChanges) {
171
245
  hasChanges = true;
172
246
  if (result.canDelete || result.css.trim() === '') {
173
- // Remove entire style tag
174
247
  modifiedHtml = modifiedHtml.slice(0, style.start) + modifiedHtml.slice(style.end);
175
248
  }
176
249
  else {
177
- // Replace style content
178
250
  const before = modifiedHtml.slice(0, style.start);
179
251
  const after = modifiedHtml.slice(style.end);
180
252
  const tagStart = html.slice(style.start).match(/<style[^>]*>/)?.[0] || '<style>';
@@ -197,13 +269,11 @@ class CSSParser {
197
269
  }
198
270
  extractImportPaths(code) {
199
271
  const imports = [];
200
- // Match CSS imports
201
272
  const importRegex = /import\s+['"]([^'"]+\.css)['"];?/g;
202
273
  let match;
203
274
  while ((match = importRegex.exec(code)) !== null) {
204
275
  imports.push(match[1]);
205
276
  }
206
- // Match require statements
207
277
  const requireRegex = /require\s*\(\s*['"]([^'"]+\.css)['"]\s*\)/g;
208
278
  while ((match = requireRegex.exec(code)) !== null) {
209
279
  imports.push(match[1]);
@@ -212,4 +282,4 @@ class CSSParser {
212
282
  }
213
283
  }
214
284
  exports.CSSParser = CSSParser;
215
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cssParser.js","sourceRoot":"","sources":["../src/cssParser.ts"],"names":[],"mappings":"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","sourcesContent":["import postcss, { Root, Rule, Declaration, AtRule } from 'postcss';\nimport safeParser from 'postcss-safe-parser';\nimport { TailwindMapper, CSSProperty } from './tailwindMapper';\nimport { logger } from './utils/logger';\n\nexport interface CSSRule {\n  selector: string;\n  className: string;\n  declarations: CSSProperty[];\n  convertedClasses: string[];\n  skipped: boolean;\n  fullyConverted: boolean; // NEW: true if ALL declarations in this rule were converted\n  partialConversion: boolean; // NEW: true if SOME but not all declarations were converted\n  reason?: string;\n}\n\nexport interface CSSParseResult {\n  css: string;\n  rules: CSSRule[];\n  hasChanges: boolean;\n  canDelete: boolean;\n  warnings: string[];\n}\n\nexport interface CSSUsageMap {\n  [className: string]: string[]; // className -> file paths\n}\n\nexport class CSSParser {\n  private mapper: TailwindMapper;\n\n  constructor(mapper: TailwindMapper) {\n    this.mapper = mapper;\n  }\n\n  async parse(css: string, filePath: string): Promise<CSSParseResult> {\n    const rules: CSSRule[] = [];\n    const warnings: string[] = [];\n    let hasChanges = false;\n\n    try {\n      const root = await postcss().process(css, {\n        parser: safeParser,\n        from: filePath\n      }).then(result => result.root);\n\n      // Process each rule\n      root.walkRules((rule) => {\n        // Skip rules inside @media, @supports, etc.\n        if (rule.parent && rule.parent.type === 'atrule') {\n          warnings.push(`Skipped rule with at-rule parent: ${rule.selector}`);\n          logger.verbose(`Skipping at-rule: ${rule.selector}`);\n          return;\n        }\n\n        // Skip pseudo-selectors\n        if (rule.selector.includes(':')) {\n          warnings.push(`Skipped pseudo-selector: ${rule.selector}`);\n          logger.verbose(`Skipping pseudo-selector: ${rule.selector}`);\n          return;\n        }\n\n        // Only process simple class selectors\n        const classNameMatch = rule.selector.match(/^\\.([a-zA-Z_-][a-zA-Z0-9_-]*)$/);\n        if (!classNameMatch) {\n          warnings.push(`Skipped complex selector: ${rule.selector}`);\n          logger.verbose(`Skipping complex selector: ${rule.selector}`);\n          return;\n        }\n\n        const className = classNameMatch[1];\n        const declarations: CSSProperty[] = [];\n\n        rule.walkDecls((decl) => {\n          // Skip CSS variables\n          if (decl.prop.startsWith('--')) {\n            warnings.push(`Skipped CSS variable: ${decl.prop}`);\n            return;\n          }\n\n          // Skip calc()\n          if (decl.value.includes('calc(')) {\n            warnings.push(`Skipped calc() value: ${decl.value}`);\n            return;\n          }\n\n          declarations.push({\n            property: decl.prop,\n            value: decl.value\n          });\n        });\n\n        if (declarations.length === 0) {\n          return;\n        }\n\n        // Convert to Tailwind classes - track which specific declarations were converted\n        const conversionResults: Array<{\n          declaration: CSSProperty;\n          converted: boolean;\n          className: string | null;\n        }> = [];\n        const conversionWarnings: string[] = [];\n\n        declarations.forEach(decl => {\n          const result = this.mapper.convertProperty(decl.property, decl.value);\n          conversionResults.push({\n            declaration: decl,\n            converted: !result.skipped && result.className !== null,\n            className: result.className\n          });\n          if (result.skipped && result.reason) {\n            conversionWarnings.push(result.reason);\n          }\n        });\n\n        const convertedClasses = conversionResults\n          .filter(r => r.converted && r.className)\n          .map(r => r.className!);\n        \n        const allDeclarationsConverted = conversionResults.every(r => r.converted);\n        const someDeclarationsConverted = convertedClasses.length > 0;\n\n        const cssRule: CSSRule = {\n          selector: rule.selector,\n          className,\n          declarations,\n          convertedClasses,\n          skipped: !someDeclarationsConverted,\n          fullyConverted: allDeclarationsConverted,\n          partialConversion: someDeclarationsConverted && !allDeclarationsConverted,\n          reason: !someDeclarationsConverted ? 'No convertible declarations' : undefined\n        };\n\n        rules.push(cssRule);\n        warnings.push(...conversionWarnings);\n\n        // CRITICAL FIX: Only remove declarations that were successfully converted\n        // Never remove the entire rule unless ALL declarations are converted\n        if (someDeclarationsConverted) {\n          hasChanges = true;\n          \n          if (allDeclarationsConverted) {\n            // All declarations converted - safe to remove entire rule\n            rule.remove();\n            logger.verbose(`Removed rule .${className} (all ${declarations.length} declarations converted)`);\n          } else {\n            // Partial conversion - only remove the converted declarations\n            let removedCount = 0;\n            rule.walkDecls((decl) => {\n              const wasConverted = conversionResults.some(r => \n                r.converted && \n                r.declaration.property === decl.prop && \n                r.declaration.value === decl.value\n              );\n              \n              if (wasConverted) {\n                decl.remove();\n                removedCount++;\n              }\n            });\n            \n            logger.verbose(`Partial conversion of .${className}: removed ${removedCount}/${declarations.length} declarations`);\n          }\n        }\n      });\n\n      // Clean up empty at-rules\n      root.walkAtRules((atRule) => {\n        if (atRule.nodes && atRule.nodes.length === 0) {\n          atRule.remove();\n        }\n      });\n\n      const canDelete = root.nodes.length === 0;\n      const newCss = root.toString();\n\n      return {\n        css: newCss,\n        rules,\n        hasChanges,\n        canDelete,\n        warnings\n      };\n\n    } catch (error) {\n      logger.error(`Failed to parse CSS ${filePath}:`, error);\n      throw new Error(`CSS parsing failed: ${error}`);\n    }\n  }\n\n  parseInternalStyle(html: string): { \n    styles: Array<{ content: string; start: number; end: number }>;\n    warnings: string[];\n  } {\n    const styles: Array<{ content: string; start: number; end: number }> = [];\n    const warnings: string[] = [];\n\n    // Simple regex to find style tags (this is safe for finding tags, not for parsing content)\n    const styleRegex = /<style[^>]*>([\\s\\S]*?)<\\/style>/gi;\n    let match;\n\n    while ((match = styleRegex.exec(html)) !== null) {\n      styles.push({\n        content: match[1].trim(),\n        start: match.index,\n        end: match.index + match[0].length\n      });\n    }\n\n    return { styles, warnings };\n  }\n\n  async parseInternalCSS(html: string, filePath: string): Promise<{\n    html: string;\n    rules: CSSRule[];\n    hasChanges: boolean;\n    warnings: string[];\n  }> {\n    const allRules: CSSRule[] = [];\n    const allWarnings: string[] = [];\n    let modifiedHtml = html;\n    let hasChanges = false;\n\n    const { styles } = this.parseInternalStyle(html);\n\n    // Process styles in reverse order to preserve indices\n    for (let i = styles.length - 1; i >= 0; i--) {\n      const style = styles[i];\n      \n      try {\n        const result = await this.parse(style.content, filePath);\n        \n        allRules.push(...result.rules);\n        allWarnings.push(...result.warnings);\n\n        if (result.hasChanges) {\n          hasChanges = true;\n\n          if (result.canDelete || result.css.trim() === '') {\n            // Remove entire style tag\n            modifiedHtml = modifiedHtml.slice(0, style.start) + modifiedHtml.slice(style.end);\n          } else {\n            // Replace style content\n            const before = modifiedHtml.slice(0, style.start);\n            const after = modifiedHtml.slice(style.end);\n            const tagStart = html.slice(style.start).match(/<style[^>]*>/)?.[0] || '<style>';\n            const tagEnd = '</style>';\n            modifiedHtml = before + tagStart + '\\n' + result.css + '\\n' + tagEnd + after;\n          }\n        }\n      } catch (error) {\n        logger.warn(`Failed to parse internal style block: ${error}`);\n        allWarnings.push(`Failed to parse internal style: ${error}`);\n      }\n    }\n\n    return {\n      html: modifiedHtml,\n      rules: allRules,\n      hasChanges,\n      warnings: allWarnings\n    };\n  }\n\n  extractImportPaths(code: string): string[] {\n    const imports: string[] = [];\n    \n    // Match CSS imports\n    const importRegex = /import\\s+['\"]([^'\"]+\\.css)['\"];?/g;\n    let match;\n    \n    while ((match = importRegex.exec(code)) !== null) {\n      imports.push(match[1]);\n    }\n\n    // Match require statements\n    const requireRegex = /require\\s*\\(\\s*['\"]([^'\"]+\\.css)['\"]\\s*\\)/g;\n    while ((match = requireRegex.exec(code)) !== null) {\n      imports.push(match[1]);\n    }\n\n    return imports;\n  }\n}\n"]}
285
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cssParser.js","sourceRoot":"","sources":["../src/cssParser.ts"],"names":[],"mappings":"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","sourcesContent":["import postcss, { Root, Rule, Declaration, AtRule } from 'postcss';\nimport safeParser from 'postcss-safe-parser';\nimport { TailwindMapper, CSSProperty } from './tailwindMapper';\nimport { logger } from './utils/logger';\nimport { \n  Breakpoint, \n  getBreakpoints, \n  resolveBreakpointsFromConfig,\n  processMediaQuery \n} from './utils/breakpointResolver';\nimport {\n  processPseudoSelector,\n  parseMultipleSelectors,\n  ParsedSelector\n} from './utils/pseudoSelectorResolver';\nimport {\n  assembleUtility,\n  assembleUtilities,\n  MergedUtility,\n  mergeUtilities,\n  normalizeVariantOrder\n} from './utils/variantAssembler';\n\nexport interface UtilityWithVariant {\n  value: string;\n  variants: string[];\n}\n\nexport interface CSSRule {\n  selector: string;\n  className: string;\n  declarations: CSSProperty[];\n  convertedClasses: string[];\n  utilities: UtilityWithVariant[];\n  skipped: boolean;\n  fullyConverted: boolean;\n  partialConversion: boolean;\n  reason?: string;\n}\n\nexport interface CSSParseResult {\n  css: string;\n  rules: CSSRule[];\n  hasChanges: boolean;\n  canDelete: boolean;\n  warnings: string[];\n}\n\nexport interface CSSUsageMap {\n  [className: string]: string[];\n}\n\nexport class CSSParser {\n  private mapper: TailwindMapper;\n  private breakpoints: Breakpoint[];\n\n  constructor(mapper: TailwindMapper, screens?: Record<string, string | [string, string]>) {\n    this.mapper = mapper;\n    this.breakpoints = screens \n      ? resolveBreakpointsFromConfig(screens) \n      : getBreakpoints();\n  }\n\n  private convertDeclarations(declarations: CSSProperty[]): {\n    utilities: UtilityWithVariant[];\n    conversionResults: Array<{ declaration: CSSProperty; converted: boolean; className: string | null }>;\n    conversionWarnings: string[];\n  } {\n    const conversionResults: Array<{\n      declaration: CSSProperty;\n      converted: boolean;\n      className: string | null;\n    }> = [];\n    const conversionWarnings: string[] = [];\n\n    declarations.forEach(decl => {\n      const result = this.mapper.convertProperty(decl.property, decl.value);\n      conversionResults.push({\n        declaration: decl,\n        converted: !result.skipped && result.className !== null,\n        className: result.className\n      });\n      if (result.skipped && result.reason) {\n        conversionWarnings.push(result.reason);\n      }\n    });\n\n    const utilities: UtilityWithVariant[] = conversionResults\n      .filter(r => r.converted && r.className)\n      .map(r => ({\n        value: r.className!,\n        variants: []\n      }));\n\n    return { utilities, conversionResults, conversionWarnings };\n  }\n\n  private processRuleWithVariants(\n    rule: Rule,\n    additionalVariants: string[] = []\n  ): { cssRules: CSSRule[]; conversionResults: Array<{ declaration: CSSProperty; converted: boolean; className: string | null }>[]; conversionWarnings: string[] } | null {\n    const selector = rule.selector;\n    const parsedSelectors = parseMultipleSelectors(selector);\n    \n    const validSelectors = parsedSelectors.filter(s => !s.isComplex && s.baseClass);\n    \n    if (validSelectors.length === 0) {\n      return null;\n    }\n\n    const declarations: CSSProperty[] = [];\n\n    rule.walkDecls((decl) => {\n      if (decl.prop.startsWith('--')) {\n        return;\n      }\n\n      if (decl.value.includes('calc(')) {\n        return;\n      }\n\n      declarations.push({\n        property: decl.prop,\n        value: decl.value\n      });\n    });\n\n    if (declarations.length === 0) {\n      return null;\n    }\n\n    const { utilities, conversionResults, conversionWarnings } = this.convertDeclarations(declarations);\n    \n    const utilitiesWithVariants = utilities.map(u => ({\n      value: u.value,\n      variants: normalizeVariantOrder([...u.variants, ...additionalVariants])\n    }));\n\n    const cssRules: CSSRule[] = [];\n    const allConversionResults: Array<{ declaration: CSSProperty; converted: boolean; className: string | null }>[] = [];\n\n    for (const parsed of validSelectors) {\n      const pseudoVariants = parsed.pseudos || [];\n      const allVariants = normalizeVariantOrder([...pseudoVariants, ...additionalVariants]);\n      \n      const utilitiesForSelector = utilities.map(u => ({\n        value: u.value,\n        variants: allVariants\n      }));\n\n      const convertedClasses = assembleUtilities(utilitiesForSelector);\n\n      const allDeclarationsConverted = conversionResults.every(r => r.converted);\n      const someDeclarationsConverted = convertedClasses.length > 0;\n\n      const cssRule: CSSRule = {\n        selector: selector,\n        className: parsed.baseClass,\n        declarations,\n        convertedClasses,\n        utilities: utilitiesForSelector,\n        skipped: !someDeclarationsConverted,\n        fullyConverted: allDeclarationsConverted,\n        partialConversion: someDeclarationsConverted && !allDeclarationsConverted,\n        reason: !someDeclarationsConverted ? 'No convertible declarations' : undefined\n      };\n\n      cssRules.push(cssRule);\n      allConversionResults.push(conversionResults);\n    }\n\n    return { cssRules, conversionResults: allConversionResults, conversionWarnings };\n  }\n\n  async parse(css: string, filePath: string): Promise<CSSParseResult> {\n    const rules: CSSRule[] = [];\n    const warnings: string[] = [];\n    let hasChanges = false;\n\n    try {\n      const root = await postcss().process(css, {\n        parser: safeParser,\n        from: filePath\n      }).then(result => result.root);\n\n      root.walkAtRules((atRule) => {\n        if (atRule.name !== 'media') {\n          return;\n        }\n\n        const mediaResult = processMediaQuery(atRule.params, this.breakpoints);\n\n        if (mediaResult.skipped) {\n          warnings.push(mediaResult.reason || `Skipped media query: ${atRule.params}`);\n          return;\n        }\n\n        const responsiveVariant = mediaResult.breakpoint!;\n\n        const nestedRules: Rule[] = [];\n        atRule.walkRules((rule) => {\n          nestedRules.push(rule);\n        });\n\n        for (const rule of nestedRules) {\n          const result = this.processRuleWithVariants(rule, [responsiveVariant]);\n          if (result) {\n            rules.push(...result.cssRules);\n            warnings.push(...result.conversionWarnings);\n\n            const anyConverted = result.cssRules.some(r => r.convertedClasses.length > 0);\n            if (anyConverted) {\n              hasChanges = true;\n\n              const allFullyConverted = result.cssRules.every(r => r.fullyConverted);\n              if (allFullyConverted) {\n                rule.remove();\n                const classNames = result.cssRules.map(r => r.className).join(', .');\n                logger.verbose(`Removed rule .${classNames} in @media (min-width) → ${responsiveVariant}`);\n              } else {\n                for (const cr of result.conversionResults.flat()) {\n                  if (cr.converted) {\n                    rule.walkDecls((decl) => {\n                      if (decl.prop === cr.declaration.property && decl.value === cr.declaration.value) {\n                        decl.remove();\n                      }\n                    });\n                  }\n                }\n                logger.verbose(`Partial conversion in @media → ${responsiveVariant}`);\n              }\n            }\n          } else {\n            warnings.push(`Skipped rule in @media: ${rule.selector}`);\n          }\n        }\n\n        if (atRule.nodes && atRule.nodes.length === 0) {\n          atRule.remove();\n          logger.verbose(`Removed empty @media rule`);\n        }\n      });\n\n      root.walkRules((rule) => {\n        if (rule.parent && rule.parent.type === 'atrule') {\n          return;\n        }\n\n        const result = this.processRuleWithVariants(rule);\n        \n        if (!result) {\n          const parsedSelectors = parseMultipleSelectors(rule.selector);\n          const allComplex = parsedSelectors.every(s => s.isComplex);\n          \n          if (allComplex) {\n            const reasons = parsedSelectors.map(s => s.reason).filter(Boolean);\n            warnings.push(...reasons as string[]);\n            logger.verbose(`Skipping complex selector: ${rule.selector}`);\n          }\n          return;\n        }\n\n        rules.push(...result.cssRules);\n        warnings.push(...result.conversionWarnings);\n\n        const anyConverted = result.cssRules.some(r => r.convertedClasses.length > 0);\n        if (anyConverted) {\n          hasChanges = true;\n\n          const allFullyConverted = result.cssRules.every(r => r.fullyConverted);\n          if (allFullyConverted) {\n            rule.remove();\n            const classNames = result.cssRules.map(r => r.className).join(', .');\n            logger.verbose(`Removed rule .${classNames} (all declarations converted)`);\n          } else {\n            for (const cr of result.conversionResults.flat()) {\n              if (cr.converted) {\n                rule.walkDecls((decl) => {\n                  if (decl.prop === cr.declaration.property && decl.value === cr.declaration.value) {\n                    decl.remove();\n                  }\n                });\n              }\n            }\n            logger.verbose(`Partial conversion of rule`);\n          }\n        }\n      });\n\n      root.walkAtRules((atRule) => {\n        if (atRule.nodes && atRule.nodes.length === 0) {\n          atRule.remove();\n        }\n      });\n\n      const canDelete = root.nodes.length === 0;\n      const newCss = root.toString();\n\n      return {\n        css: newCss,\n        rules,\n        hasChanges,\n        canDelete,\n        warnings\n      };\n\n    } catch (error) {\n      logger.error(`Failed to parse CSS ${filePath}:`, error);\n      throw new Error(`CSS parsing failed: ${error}`);\n    }\n  }\n\n  parseInternalStyle(html: string): { \n    styles: Array<{ content: string; start: number; end: number }>;\n    warnings: string[];\n  } {\n    const styles: Array<{ content: string; start: number; end: number }> = [];\n    const warnings: string[] = [];\n\n    const styleRegex = /<style[^>]*>([\\s\\S]*?)<\\/style>/gi;\n    let match;\n\n    while ((match = styleRegex.exec(html)) !== null) {\n      styles.push({\n        content: match[1].trim(),\n        start: match.index,\n        end: match.index + match[0].length\n      });\n    }\n\n    return { styles, warnings };\n  }\n\n  async parseInternalCSS(html: string, filePath: string): Promise<{\n    html: string;\n    rules: CSSRule[];\n    hasChanges: boolean;\n    warnings: string[];\n  }> {\n    const allRules: CSSRule[] = [];\n    const allWarnings: string[] = [];\n    let modifiedHtml = html;\n    let hasChanges = false;\n\n    const { styles } = this.parseInternalStyle(html);\n\n    for (let i = styles.length - 1; i >= 0; i--) {\n      const style = styles[i];\n      \n      try {\n        const result = await this.parse(style.content, filePath);\n        \n        allRules.push(...result.rules);\n        allWarnings.push(...result.warnings);\n\n        if (result.hasChanges) {\n          hasChanges = true;\n\n          if (result.canDelete || result.css.trim() === '') {\n            modifiedHtml = modifiedHtml.slice(0, style.start) + modifiedHtml.slice(style.end);\n          } else {\n            const before = modifiedHtml.slice(0, style.start);\n            const after = modifiedHtml.slice(style.end);\n            const tagStart = html.slice(style.start).match(/<style[^>]*>/)?.[0] || '<style>';\n            const tagEnd = '</style>';\n            modifiedHtml = before + tagStart + '\\n' + result.css + '\\n' + tagEnd + after;\n          }\n        }\n      } catch (error) {\n        logger.warn(`Failed to parse internal style block: ${error}`);\n        allWarnings.push(`Failed to parse internal style: ${error}`);\n      }\n    }\n\n    return {\n      html: modifiedHtml,\n      rules: allRules,\n      hasChanges,\n      warnings: allWarnings\n    };\n  }\n\n  extractImportPaths(code: string): string[] {\n    const imports: string[] = [];\n    \n    const importRegex = /import\\s+['\"]([^'\"]+\\.css)['\"];?/g;\n    let match;\n    \n    while ((match = importRegex.exec(code)) !== null) {\n      imports.push(match[1]);\n    }\n\n    const requireRegex = /require\\s*\\(\\s*['\"]([^'\"]+\\.css)['\"]\\s*\\)/g;\n    while ((match = requireRegex.exec(code)) !== null) {\n      imports.push(match[1]);\n    }\n\n    return imports;\n  }\n}"]}
@@ -0,0 +1,15 @@
1
+ import { TailwindMapper } from './tailwindMapper';
2
+ export interface HTMLParseResult {
3
+ html: string;
4
+ hasChanges: boolean;
5
+ conversions: number;
6
+ warnings: string[];
7
+ }
8
+ export declare class HTMLParser {
9
+ private mapper;
10
+ constructor(mapper: TailwindMapper);
11
+ parse(html: string, filePath: string): HTMLParseResult;
12
+ private parseInlineStyle;
13
+ private mergeClasses;
14
+ extractStylesheets(html: string): string[];
15
+ }
@@ -0,0 +1,103 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.HTMLParser = void 0;
4
+ class HTMLParser {
5
+ constructor(mapper) {
6
+ this.mapper = mapper;
7
+ }
8
+ parse(html, filePath) {
9
+ const warnings = [];
10
+ let hasChanges = false;
11
+ let conversions = 0;
12
+ // Parse inline styles: style="display: flex; padding: 16px;"
13
+ // Convert to: class="flex p-4"
14
+ const styleRegex = /style="([^"]*)"/g;
15
+ let modifiedHtml = html;
16
+ let match;
17
+ while ((match = styleRegex.exec(html)) !== null) {
18
+ const fullMatch = match[0];
19
+ const styleValue = match[1];
20
+ // Parse CSS declarations from inline style
21
+ const declarations = this.parseInlineStyle(styleValue);
22
+ if (declarations.length === 0) {
23
+ continue;
24
+ }
25
+ // Convert to Tailwind classes
26
+ const { classes, warnings: convWarnings } = this.mapper.convertMultiple(declarations);
27
+ if (classes.length === 0) {
28
+ warnings.push(...convWarnings);
29
+ continue;
30
+ }
31
+ // Find existing class attribute
32
+ const beforeStyle = html.substring(0, match.index);
33
+ const tagMatch = beforeStyle.match(/<([a-z][a-z0-9]*)[^>]*$/i);
34
+ if (!tagMatch) {
35
+ warnings.push(`Could not find tag for inline style`);
36
+ continue;
37
+ }
38
+ // Check if there's an existing class attribute
39
+ const tagEnd = html.indexOf('>', match.index);
40
+ const tagContent = html.substring(match.index, tagEnd);
41
+ const existingClassMatch = tagContent.match(/class="([^"]*)"/);
42
+ let replacement;
43
+ if (existingClassMatch && existingClassMatch.index !== undefined) {
44
+ // Merge with existing class
45
+ const existingClasses = existingClassMatch[1];
46
+ const mergedClasses = this.mergeClasses(existingClasses, classes);
47
+ // Replace class attribute and remove style
48
+ const beforeClass = modifiedHtml.substring(0, match.index + existingClassMatch.index);
49
+ const afterStyle = modifiedHtml.substring(match.index + fullMatch.length);
50
+ // This is complex - need to handle both class and style replacement
51
+ // For now, simplified version:
52
+ replacement = `class="${mergedClasses}"`;
53
+ modifiedHtml = modifiedHtml.replace(fullMatch, replacement);
54
+ }
55
+ else {
56
+ // Add class attribute, remove style
57
+ replacement = `class="${classes.join(' ')}"`;
58
+ modifiedHtml = modifiedHtml.replace(fullMatch, replacement);
59
+ }
60
+ conversions += classes.length;
61
+ hasChanges = true;
62
+ warnings.push(...convWarnings);
63
+ }
64
+ return {
65
+ html: modifiedHtml,
66
+ hasChanges,
67
+ conversions,
68
+ warnings
69
+ };
70
+ }
71
+ parseInlineStyle(styleValue) {
72
+ const declarations = [];
73
+ // Split by semicolon
74
+ const props = styleValue.split(';').filter(s => s.trim());
75
+ props.forEach(prop => {
76
+ const colonIndex = prop.indexOf(':');
77
+ if (colonIndex === -1)
78
+ return;
79
+ const property = prop.substring(0, colonIndex).trim();
80
+ const value = prop.substring(colonIndex + 1).trim();
81
+ if (property && value) {
82
+ declarations.push({ property, value });
83
+ }
84
+ });
85
+ return declarations;
86
+ }
87
+ mergeClasses(existing, newClasses) {
88
+ const existingSet = new Set(existing.split(/\s+/).filter(Boolean));
89
+ newClasses.forEach(cls => existingSet.add(cls));
90
+ return Array.from(existingSet).join(' ');
91
+ }
92
+ extractStylesheets(html) {
93
+ const stylesheets = [];
94
+ const linkRegex = /<link[^>]*rel="stylesheet"[^>]*href="([^"]*)"[^>]*>/gi;
95
+ let match;
96
+ while ((match = linkRegex.exec(html)) !== null) {
97
+ stylesheets.push(match[1]);
98
+ }
99
+ return stylesheets;
100
+ }
101
+ }
102
+ exports.HTMLParser = HTMLParser;
103
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"htmlParser.js","sourceRoot":"","sources":["../src/htmlParser.ts"],"names":[],"mappings":"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","sourcesContent":["import { TailwindMapper } from './tailwindMapper';\nimport { logger } from './utils/logger';\n\nexport interface HTMLParseResult {\n  html: string;\n  hasChanges: boolean;\n  conversions: number;\n  warnings: string[];\n}\n\nexport class HTMLParser {\n  private mapper: TailwindMapper;\n\n  constructor(mapper: TailwindMapper) {\n    this.mapper = mapper;\n  }\n\n  parse(html: string, filePath: string): HTMLParseResult {\n    const warnings: string[] = [];\n    let hasChanges = false;\n    let conversions = 0;\n\n    // Parse inline styles: style=\"display: flex; padding: 16px;\"\n    // Convert to: class=\"flex p-4\"\n    const styleRegex = /style=\"([^\"]*)\"/g;\n    let modifiedHtml = html;\n    let match;\n\n    while ((match = styleRegex.exec(html)) !== null) {\n      const fullMatch = match[0];\n      const styleValue = match[1];\n      \n      // Parse CSS declarations from inline style\n      const declarations = this.parseInlineStyle(styleValue);\n      \n      if (declarations.length === 0) {\n        continue;\n      }\n\n      // Convert to Tailwind classes\n      const { classes, warnings: convWarnings } = this.mapper.convertMultiple(declarations);\n\n      if (classes.length === 0) {\n        warnings.push(...convWarnings);\n        continue;\n      }\n\n      // Find existing class attribute\n      const beforeStyle = html.substring(0, match.index);\n      const tagMatch = beforeStyle.match(/<([a-z][a-z0-9]*)[^>]*$/i);\n      \n      if (!tagMatch) {\n        warnings.push(`Could not find tag for inline style`);\n        continue;\n      }\n\n      // Check if there's an existing class attribute\n      const tagEnd = html.indexOf('>', match.index);\n      const tagContent = html.substring(match.index, tagEnd);\n      const existingClassMatch = tagContent.match(/class=\"([^\"]*)\"/);\n\n      let replacement: string;\n      \n        if (existingClassMatch && existingClassMatch.index !== undefined) {\n        // Merge with existing class\n        const existingClasses = existingClassMatch[1];\n        const mergedClasses = this.mergeClasses(existingClasses, classes);\n        \n        // Replace class attribute and remove style\n        const beforeClass = modifiedHtml.substring(0, match.index + existingClassMatch.index);\n        const afterStyle = modifiedHtml.substring(match.index + fullMatch.length);\n        \n        // This is complex - need to handle both class and style replacement\n        // For now, simplified version:\n        replacement = `class=\"${mergedClasses}\"`;\n        modifiedHtml = modifiedHtml.replace(fullMatch, replacement);\n      } else {\n        // Add class attribute, remove style\n        replacement = `class=\"${classes.join(' ')}\"`;\n        modifiedHtml = modifiedHtml.replace(fullMatch, replacement);\n      }\n\n      conversions += classes.length;\n      hasChanges = true;\n      warnings.push(...convWarnings);\n    }\n\n    return {\n      html: modifiedHtml,\n      hasChanges,\n      conversions,\n      warnings\n    };\n  }\n\n  private parseInlineStyle(styleValue: string): Array<{property: string; value: string}> {\n    const declarations: Array<{property: string; value: string}> = [];\n    \n    // Split by semicolon\n    const props = styleValue.split(';').filter(s => s.trim());\n    \n    props.forEach(prop => {\n      const colonIndex = prop.indexOf(':');\n      if (colonIndex === -1) return;\n      \n      const property = prop.substring(0, colonIndex).trim();\n      const value = prop.substring(colonIndex + 1).trim();\n      \n      if (property && value) {\n        declarations.push({ property, value });\n      }\n    });\n    \n    return declarations;\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  extractStylesheets(html: string): string[] {\n    const stylesheets: string[] = [];\n    const linkRegex = /<link[^>]*rel=\"stylesheet\"[^>]*href=\"([^\"]*)\"[^>]*>/gi;\n    let match;\n\n    while ((match = linkRegex.exec(html)) !== null) {\n      stylesheets.push(match[1]);\n    }\n\n    return stylesheets;\n  }\n}\n"]}
package/dist/index.d.ts CHANGED
@@ -2,7 +2,10 @@ export { scanProject, ScannedFile } from './scanner';
2
2
  export { transformFiles, TransformOptions, TransformResults } from './transformer';
3
3
  export { TailwindMapper, CSSProperty, ConversionResult } from './tailwindMapper';
4
4
  export { JSXParser, JSXTransformation, JSXParseResult } from './jsxParser';
5
- export { CSSParser, CSSRule, CSSParseResult } from './cssParser';
5
+ export { CSSParser, CSSRule, CSSParseResult, UtilityWithVariant } from './cssParser';
6
6
  export { FileWriter, FileWriteOptions } from './fileWriter';
7
7
  export { loadTailwindConfig, TailwindConfig } from './utils/config';
8
8
  export { logger } from './utils/logger';
9
+ export { Breakpoint, MediaQueryInfo, getDefaultBreakpoints, resolveBreakpointsFromConfig, parseMediaQuery, findBreakpointForMinWidth, processMediaQuery, prefixWithBreakpoint } from './utils/breakpointResolver';
10
+ export { ParsedSelector, PSEUDO_TO_VARIANT, SUPPORTED_PSEUDOS, parseSelector, mapPseudoToVariant, processPseudoSelector, parseMultipleSelectors } from './utils/pseudoSelectorResolver';
11
+ export { VARIANT_ORDER, isResponsiveVariant, isPseudoVariant, sortVariants, deduplicateVariants, normalizeVariantOrder, assembleUtility, assembleUtilities, mergeUtilities, MergedUtility } from './utils/variantAssembler';