@bhollis/eslint-plugin-css-modules 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1,142 @@
1
+ # @bhollis/eslint-plugin-css-modules
2
+
3
+ This plugin intends to help you in tracking down problems when you are using css-modules. It tells if you are using a non-existent css/scss/less class in js or if you forgot to use some classes which you declared in css/scss/less.
4
+
5
+ This is a forked version of https://github.com/atfzl/eslint-plugin-css-modules with fixes to make it compatible with more recent versions of css-loader, especially with [`namedExports: true`](https://webpack.js.org/loaders/css-loader/#namedexport). It is also compatible with ESLint's flat config.
6
+
7
+ ## Rules
8
+
9
+ * `css-modules/no-unused-class`: You must use all the classes defined in css/scss/less file.
10
+
11
+ >If you still want to mark a class as used, then use this comment on top of your file
12
+ ```js
13
+ /* eslint css-modules/no-unused-class: [2, { markAsUsed: ['container'] }] */
14
+ ```
15
+ where container is the css class that you want to mark as used.
16
+ Add all such classes in the array.
17
+
18
+ >If you use the `camelCase` option of `css-loader`, you must also enabled it for this plugin
19
+ ```js
20
+ /* eslint css-modules/no-unused-class: [2, { camelCase: true }] */
21
+ ```
22
+
23
+ * `css-modules/no-undef-class`: You must not use a non existing class, or a property that hasn't been exported using the [:export keyword](https://github.com/css-modules/icss#export).
24
+
25
+ >If you use the `camelCase` option of `css-loader`, you must also enabled it for this plugin
26
+ ```js
27
+ /* eslint css-modules/no-undef-class: [2, { camelCase: true }] */
28
+ ```
29
+
30
+ ## Installation
31
+
32
+ ```
33
+ npm i --save-dev @bhollis/eslint-plugin-css-modules
34
+ ```
35
+
36
+ ## Usage:
37
+
38
+ ```js
39
+ // eslint.config.js
40
+ import { defineConfig } from "eslint/config";
41
+ import cssModules from "@bhollis/eslint-plugin-css-modules";
42
+
43
+ export default defineConfig([
44
+ cssModules.configs.recommended
45
+ {
46
+ files: ["**/*.jsx"], // any patterns you want to apply the config to
47
+ plugins: {
48
+ 'css-modules': cssModules,
49
+ },
50
+ extends: ["css-modules/recommended"],
51
+ },
52
+ ]);
53
+ ```
54
+
55
+ You may also tweak the rules individually. For instance, if you use the [camelCase](https://github.com/webpack-contrib/css-loader#camelcase) option of webpack's css-loader:
56
+
57
+ ```js
58
+ // eslint.config.js
59
+ import { defineConfig } from "eslint/config";
60
+ import cssModules from "@bhollis/eslint-plugin-css-modules";
61
+
62
+ export default defineConfig([
63
+ {
64
+ files: ["**/*.jsx"], // any patterns you want to apply the config to
65
+ plugins: {
66
+ 'css-modules': cssModules,
67
+ },
68
+ rules: {
69
+ "css-modules/no-unused-class": [2, { "camelCase": true }],
70
+ "css-modules/no-undef-class": [2, { "camelCase": true }]
71
+ },
72
+ },
73
+ ]);
74
+ ```
75
+
76
+ The camelCase option has 4 possible values, see [css-loader#camelCase](https://github.com/webpack-contrib/css-loader#camelcase) for description:
77
+ ```js
78
+ true | "dashes" | "only" | "dashes-only"
79
+ ```
80
+
81
+ ## Specifying base path
82
+
83
+ You can specify path for the base directory via plugin settings in eslint.config.js. This is used by the plugin to resolve absolute (S)CSS paths:
84
+
85
+ ```js
86
+ // eslint.config.js
87
+ import { defineConfig } from "eslint/config";
88
+ import cssModules from "@bhollis/eslint-plugin-css-modules";
89
+
90
+ export default defineConfig([
91
+ {
92
+ files: ["**/*.jsx"], // any patterns you want to apply the config to
93
+ plugins: {
94
+ 'css-modules': cssModules,
95
+ },
96
+ settings: {
97
+ 'css-modules': {
98
+ basePath: "app/scripts/..."
99
+ }
100
+ },
101
+ },
102
+ ]);
103
+ ```
104
+
105
+ ## Screen Shot
106
+
107
+ ![ScreenShot](https://raw.githubusercontent.com/bhollis/eslint-plugin-css-modules/master/screenshots/screenshot3.png)
108
+
109
+ ```
110
+ 1:8 error Unused classes found: container css-modules/no-unused-class
111
+ 5:17 error Class 'containr' not found css-modules/no-undef-class
112
+ 10:26 error Class 'foo' not found css-modules/no-undef-class
113
+ ```
114
+
115
+ scss:
116
+
117
+ ```scss
118
+ /* .head is global, will not be used in js */
119
+ :global(.head) {
120
+ color: green;
121
+ }
122
+
123
+ .container {
124
+ width: 116px;
125
+
126
+ i {
127
+ font-size: 2.2rem;
128
+ }
129
+
130
+ .button {
131
+ padding: 7px 0 0 5px;
132
+ }
133
+ }
134
+
135
+ .footer {
136
+ color: cyan;
137
+ }
138
+ ```
139
+
140
+ ## With Thanks
141
+
142
+ * This is forked from [`eslint-plugin-css-modules`](https://github.com/atfzl/eslint-plugin-css-modules).
@@ -0,0 +1,138 @@
1
+ import fs from 'node:fs';
2
+ import path from 'node:path';
3
+ import { camelCase } from 'es-toolkit';
4
+ import gonzales from 'gonzales-pe';
5
+ import { getRegularClassesMap, getComposesClassesMap, getExtendClassesMap, getParentSelectorClassesMap, getICSSExportPropsMap, eliminateGlobals } from './traversalUtils.js';
6
+ const styleExtensionRegex = /\.(s?css|less)$/;
7
+ function dashesCamelCase(str) {
8
+ return str.replace(/-+(\w)/g, function (match, firstLetter) {
9
+ return firstLetter.toUpperCase();
10
+ });
11
+ }
12
+ export const getFilePath = (context, styleFilePath) => {
13
+ const settings = context.settings['css-modules'];
14
+ const dirName = path.dirname(context.getFilename());
15
+ const basePath = settings && settings.basePath ? settings.basePath : '';
16
+ return styleFilePath.startsWith('.') ? path.resolve(dirName, styleFilePath) : path.resolve(basePath, styleFilePath);
17
+ };
18
+ export const getPropertyName = node => {
19
+ const propertyName = node.computed
20
+ /*
21
+ square braces eg s['header']
22
+ we won't use node.property.name because it is for cases like
23
+ s[abc] where abc is a variable
24
+ */ ? node.property.value
25
+ /* dot notation, eg s.header */ : node.property.name;
26
+
27
+ /*
28
+ skip property names starting with _
29
+ eg. special functions provided
30
+ by css modules like _getCss()
31
+ Tried to just skip function calls, but the parser
32
+ thinks of normal property access like s._getCss and
33
+ function calls like s._getCss() as same.
34
+ */
35
+ if (!propertyName || propertyName?.toString().startsWith('_')) {
36
+ return null;
37
+ }
38
+ return propertyName;
39
+ };
40
+ export const getClassesMap = (classes, camelCaseOption) => {
41
+ const classesMap = {};
42
+
43
+ // Unroll the loop because of performance!
44
+ // Remember that this function will run on every lint (e.g.: on file save)
45
+ switch (camelCaseOption) {
46
+ case true:
47
+ for (const className of Object.keys(classes)) {
48
+ classesMap[className] = className;
49
+ classesMap[camelCase(className)] = className;
50
+ }
51
+ break;
52
+ case 'dashes':
53
+ for (const className of Object.keys(classes)) {
54
+ classesMap[className] = className;
55
+ classesMap[dashesCamelCase(className)] = className;
56
+ }
57
+ break;
58
+ case 'only':
59
+ for (const className of Object.keys(classes)) {
60
+ classesMap[camelCase(className)] = className;
61
+ }
62
+ break;
63
+ case 'dashes-only':
64
+ for (const className of Object.keys(classes)) {
65
+ classesMap[dashesCamelCase(className)] = className;
66
+ }
67
+ break;
68
+ default:
69
+ for (const className of Object.keys(classes)) {
70
+ classesMap[className] = className;
71
+ }
72
+ }
73
+ return classesMap;
74
+ };
75
+ export const getStyleImportNodeData = node => {
76
+ // path from which it was imported
77
+ const styleFilePath = node?.source?.value;
78
+ if (styleFilePath && styleExtensionRegex.test(styleFilePath)) {
79
+ const importNode = node.specifiers?.find(specifier => specifier.type === 'ImportDefaultSpecifier' || specifier.type === 'ImportNamespaceSpecifier');
80
+ const importName = importNode?.local?.name;
81
+ if (importName) {
82
+ // it had a default or namespace import
83
+ return {
84
+ importName,
85
+ styleFilePath,
86
+ importNode
87
+ };
88
+ }
89
+ }
90
+ };
91
+ export const fileExists = filePath => {
92
+ try {
93
+ // check if file exists
94
+ fs.statSync(filePath);
95
+ return true;
96
+ } catch (e) {
97
+ return false;
98
+ }
99
+ };
100
+
101
+ /**
102
+ * @returns AST of the parsed file or null if parse failed
103
+ */
104
+ export const getAST = filePath => {
105
+ const fileContent = fs.readFileSync(filePath);
106
+ const syntax = path.extname(filePath).slice(1); // remove leading .
107
+
108
+ try {
109
+ return gonzales.parse(fileContent.toString(), {
110
+ syntax
111
+ });
112
+ } catch (e) {
113
+ return null;
114
+ }
115
+ };
116
+ export const getStyleClasses = ast => {
117
+ /*
118
+ mutates ast by removing :global scopes
119
+ */
120
+ eliminateGlobals(ast);
121
+ const classesMap = getRegularClassesMap(ast);
122
+ const composedClassesMap = getComposesClassesMap(ast);
123
+ const extendClassesMap = getExtendClassesMap(ast);
124
+ const parentSelectorClassesMap = getParentSelectorClassesMap(ast);
125
+ return {
126
+ ...classesMap,
127
+ ...composedClassesMap,
128
+ ...extendClassesMap,
129
+ ...parentSelectorClassesMap
130
+ };
131
+ };
132
+ export const getExportPropsMap = ast => {
133
+ const exportPropsMap = getICSSExportPropsMap(ast);
134
+ return {
135
+ ...exportPropsMap
136
+ };
137
+ };
138
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["fs","path","camelCase","gonzales","getRegularClassesMap","getComposesClassesMap","getExtendClassesMap","getParentSelectorClassesMap","getICSSExportPropsMap","eliminateGlobals","styleExtensionRegex","dashesCamelCase","str","replace","match","firstLetter","toUpperCase","getFilePath","context","styleFilePath","settings","dirName","dirname","getFilename","basePath","startsWith","resolve","getPropertyName","node","propertyName","computed","property","value","name","toString","getClassesMap","classes","camelCaseOption","classesMap","className","Object","keys","getStyleImportNodeData","source","test","importNode","specifiers","find","specifier","type","importName","local","fileExists","filePath","statSync","e","getAST","fileContent","readFileSync","syntax","extname","slice","parse","getStyleClasses","ast","composedClassesMap","extendClassesMap","parentSelectorClassesMap","getExportPropsMap","exportPropsMap"],"sources":["../../lib/core/index.js"],"sourcesContent":["// @flow\n\nimport fs from 'node:fs';\nimport path from 'node:path';\nimport { camelCase } from 'es-toolkit';\nimport gonzales from 'gonzales-pe';\n\nimport type { JsNode, gASTNode } from '../types/index.js';\n\nimport {\n  getRegularClassesMap,\n  getComposesClassesMap,\n  getExtendClassesMap,\n  getParentSelectorClassesMap,\n  getICSSExportPropsMap,\n  eliminateGlobals,\n} from './traversalUtils.js';\n\nconst styleExtensionRegex = /\\.(s?css|less)$/;\n\nfunction dashesCamelCase (str: string) {\n  return str.replace(/-+(\\w)/g, function (match, firstLetter) {\n    return firstLetter.toUpperCase();\n  });\n}\n\nexport const getFilePath = (context, styleFilePath) => {\n  const settings = context.settings['css-modules'];\n\n  const dirName = path.dirname(context.getFilename());\n  const basePath = (settings && settings.basePath) ? settings.basePath : '';\n\n  return styleFilePath.startsWith('.')\n    ? path.resolve(dirName, styleFilePath)\n    : path.resolve(basePath, styleFilePath);\n};\n\nexport const getPropertyName = (node: JsNode): ?string => {\n  const propertyName = node.computed\n    /*\n       square braces eg s['header']\n       we won't use node.property.name because it is for cases like\n       s[abc] where abc is a variable\n     */\n    ? node.property.value\n    /* dot notation, eg s.header */\n    : node.property.name;\n\n  /*\n     skip property names starting with _\n     eg. special functions provided\n     by css modules like _getCss()\n\n     Tried to just skip function calls, but the parser\n     thinks of normal property access like s._getCss and\n     function calls like s._getCss() as same.\n   */\n  if (!propertyName || propertyName?.toString().startsWith('_')) {\n    return null;\n  }\n\n  return propertyName;\n};\n\nexport const getClassesMap = (classes: Object, camelCaseOption: string|boolean): Object => {\n  const classesMap = {};\n\n  // Unroll the loop because of performance!\n  // Remember that this function will run on every lint (e.g.: on file save)\n  switch (camelCaseOption) {\n    case true:\n      for (const className of Object.keys(classes)) {\n        classesMap[className] = className;\n        classesMap[camelCase(className)] = className;\n      }\n      break;\n    case 'dashes':\n      for (const className of Object.keys(classes)) {\n        classesMap[className] = className;\n        classesMap[dashesCamelCase(className)] = className;\n      }\n      break;\n    case 'only':\n      for (const className of Object.keys(classes)) {\n        classesMap[camelCase(className)] = className;\n      }\n      break;\n    case 'dashes-only':\n      for (const className of Object.keys(classes)) {\n        classesMap[dashesCamelCase(className)] = className;\n      }\n      break;\n    default:\n      for (const className of Object.keys(classes)) {\n        classesMap[className] = className;\n      }\n  }\n\n  return classesMap;\n};\n\nexport const getStyleImportNodeData = (node: JsNode): ?Object => {\n  // path from which it was imported\n  const styleFilePath = node?.source?.value;\n\n  if (styleFilePath && styleExtensionRegex.test(styleFilePath)) {\n    const importNode = node.specifiers?.find(\n      (specifier) =>\n        specifier.type === 'ImportDefaultSpecifier' || specifier.type === 'ImportNamespaceSpecifier'\n    );\n    const importName = importNode?.local?.name;\n    if (importName) {\n      // it had a default or namespace import\n      return { importName, styleFilePath, importNode };\n    }\n  }\n};\n\nexport const fileExists = (filePath: string): boolean => {\n  try {\n    // check if file exists\n    fs.statSync(filePath);\n    return true;\n  } catch (e) {\n    return false;\n  }\n};\n\n/**\n * @returns AST of the parsed file or null if parse failed\n */\nexport const getAST = (filePath: string): gASTNode | null => {\n  const fileContent = fs.readFileSync(filePath);\n\n  const syntax = path.extname(filePath).slice(1); // remove leading .\n\n  try {\n    return gonzales.parse(fileContent.toString(), { syntax });\n  } catch (e) {\n    return null;\n  }\n};\n\nexport const getStyleClasses = (ast: gASTNode): ?Object => {\n  /*\n     mutates ast by removing :global scopes\n   */\n  eliminateGlobals(ast);\n\n  const classesMap = getRegularClassesMap(ast);\n  const composedClassesMap = getComposesClassesMap(ast);\n  const extendClassesMap = getExtendClassesMap(ast);\n  const parentSelectorClassesMap = getParentSelectorClassesMap(ast);\n\n  return {\n    ...classesMap,\n    ...composedClassesMap,\n    ...extendClassesMap,\n    ...parentSelectorClassesMap\n  };\n};\n\nexport const getExportPropsMap = (ast: gASTNode): ?Object => {\n  const exportPropsMap = getICSSExportPropsMap(ast);\n  return {\n    ...exportPropsMap\n  };\n};\n"],"mappings":"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"}
@@ -0,0 +1,186 @@
1
+ /* eslint-disable no-param-reassign */
2
+ export const getICSSExportPropsMap = ast => {
3
+ const ruleSets = [];
4
+ ast.traverseByType('ruleset', node => ruleSets.push(node));
5
+ return ruleSets.filter(ruleSet => {
6
+ const content = ruleSet.content || [];
7
+ return content.some(item => item.type === 'selector' && item.content?.some(selectorItem => selectorItem.type === 'pseudoClass' && selectorItem.content?.some(pseudoItem => pseudoItem.type === 'ident' && pseudoItem.content === 'export')));
8
+ }).reduce((result, ruleSet) => {
9
+ const declarations = (ruleSet.content || []).filter(item => item.type === 'block').flatMap(block => block.content || []).filter(item => item.type === 'declaration');
10
+ for (const declaration of declarations) {
11
+ const property = (declaration.content || []).find(item => item.type === 'property');
12
+ const propName = (property?.content || [])[0]?.content;
13
+ if (propName) {
14
+ result[propName] = propName;
15
+ }
16
+ }
17
+ return result;
18
+ }, {});
19
+ };
20
+ export const getRegularClassesMap = ast => {
21
+ const ruleSets = [];
22
+ ast.traverseByType('ruleset', node => ruleSets.push(node));
23
+ return ruleSets.reduce((result, ruleSet) => {
24
+ const selectors = ruleSet.content?.filter(item => item.type === 'selector') ?? [];
25
+ for (const selector of selectors) {
26
+ const classes = selector.content?.filter(item => item.type === 'class') ?? [];
27
+ for (const classNode of classes) {
28
+ const ident = classNode.content?.find(item => item.type === 'ident');
29
+ if (ident?.content) {
30
+ result[ident.content] = false;
31
+ }
32
+ }
33
+ }
34
+ return result;
35
+ }, {});
36
+ };
37
+ export const getComposesClassesMap = ast => {
38
+ const declarations = [];
39
+ ast.traverseByType('declaration', node => declarations.push(node));
40
+ return declarations.reduce((result, declaration) => {
41
+ const content = declaration.content;
42
+ const property = content?.find(item => item.type === 'property');
43
+ const hasComposes = property?.content?.some(item => item.type === 'ident' && item.content === 'composes');
44
+ if (!hasComposes) return result;
45
+ const value = content?.find(item => item.type === 'value');
46
+ const valueContent = value?.content;
47
+
48
+ // Reject classes composing from other files, e.g. `composes: foo from './other.css'`
49
+ if (valueContent?.some(item => item.type === 'ident' && item.content === 'from')) {
50
+ return result;
51
+ }
52
+
53
+ // Extract and mark composed classes
54
+ const composedClasses = valueContent?.filter(item => item.type === 'ident' && item.content) ?? [];
55
+ for (const item of composedClasses) {
56
+ result[item.content] = true;
57
+ }
58
+ return result;
59
+ }, {});
60
+ };
61
+ export const getExtendClassesMap = ast => {
62
+ const extendNodes = [];
63
+ ast.traverseByType('extend', node => extendNodes.push(node));
64
+ return extendNodes.reduce((result, extendNode) => {
65
+ const selector = extendNode.content?.find(item => item.type === 'selector');
66
+ const classNode = selector?.content?.find(item => item.type === 'class');
67
+ const ident = classNode?.content?.find(item => item.type === 'ident');
68
+ const className = ident?.content;
69
+ if (className) {
70
+ result[className] = true; // mark extend classes as true
71
+ }
72
+
73
+ return result;
74
+ }, {});
75
+ };
76
+
77
+ /**
78
+ * Resolves parent selectors to their full class names.
79
+ *
80
+ * E.g. `.foo { &_bar {color: blue } }` to `.foo_bar`.
81
+ */
82
+ export const getParentSelectorClassesMap = ast => {
83
+ const classesMap = {};
84
+
85
+ // Recursively traverses down the tree looking for parent selector
86
+ // extensions. Recursion is necessary as these selectors can be nested.
87
+ const getExtensions = nodeContent => {
88
+ const blockContent = nodeContent.filter(item => item.type === 'block').flatMap(item => item.content || []);
89
+ const rulesetChildren = blockContent.filter(item => item.type === 'ruleset');
90
+ const rulesetDescendants = blockContent.filter(item => item.type === 'include').flatMap(item => item.content || []).filter(subItem => subItem.type === 'block').flatMap(subItem => subItem.content || []).filter(subItem => subItem.type === 'ruleset');
91
+ const rulesetsContent = [...rulesetChildren, ...rulesetDescendants].flatMap(item => item.content || []);
92
+ const extensions = rulesetsContent.filter(item => item.type === 'selector').flatMap(item => item.content || []).filter(selectorItem => selectorItem.type === 'parentSelectorExtension').flatMap(selectorItem => selectorItem.content || []).filter(identItem => identItem.type === 'ident').map(identItem => identItem.content);
93
+ if (!extensions.length) return [];
94
+ const nestedExtensions = getExtensions(rulesetsContent);
95
+ const result = extensions;
96
+ if (nestedExtensions.length) {
97
+ for (const nestedExt of nestedExtensions) {
98
+ extensions.forEach(ext => {
99
+ result.push(ext + nestedExt);
100
+ });
101
+ }
102
+ }
103
+ return result;
104
+ };
105
+ ast.traverseByType('ruleset', node => {
106
+ const classNames = (node.content || []).filter(item => item.type === 'selector').flatMap(item => item.content || []).filter(item => item.type === 'class').flatMap(item => item.content || []).filter(item => item.type === 'ident' && item.content).map(item => item.content);
107
+ if (!classNames.length) return;
108
+ const extensions = getExtensions(node.content);
109
+ if (!extensions.length) return;
110
+ classNames.forEach(className => {
111
+ extensions.forEach(ext => {
112
+ classesMap[className + ext] = false;
113
+ });
114
+
115
+ // Ignore the base class if it only exists for nesting parent selectors
116
+ const hasDeclarations = node.content?.some(item => item.type === 'block' && item.content?.some(subItem => subItem.type === 'declaration'));
117
+ if (!hasDeclarations) classesMap[className] = true;
118
+ });
119
+ });
120
+ return classesMap;
121
+ };
122
+
123
+ /**
124
+ * Mutates the AST by removing `:global` instances.
125
+ *
126
+ * For the AST structure:
127
+ * @see https://github.com/css/gonzales/blob/master/doc/AST.CSSP.en.md
128
+ */
129
+ export const eliminateGlobals = ast => {
130
+ // Remove all :global/:global(...) in selectors
131
+ ast.traverseByType('selector', selectorNode => {
132
+ const selectorContent = selectorNode.content;
133
+ let hasGlobalWithNoArgs = false;
134
+ let i = 0;
135
+ let currNode = selectorContent[i];
136
+ while (currNode) {
137
+ if (currNode.is('pseudoClass')) {
138
+ // Remove all :global/:global(...) and trailing space
139
+ const identifierNode = currNode.content[0];
140
+ if (identifierNode && identifierNode.content === 'global') {
141
+ if (currNode.content.length === 1) hasGlobalWithNoArgs = true;
142
+ selectorNode.removeChild(i);
143
+ if (selectorContent[i] && selectorContent[i].is('space')) {
144
+ selectorNode.removeChild(i);
145
+ }
146
+ } else {
147
+ i++;
148
+ }
149
+ } else if (currNode.is('class') && hasGlobalWithNoArgs) {
150
+ // Remove all class after :global and their trailing space
151
+ selectorNode.removeChild(i);
152
+ if (selectorContent[i] && selectorContent[i].is('space')) {
153
+ selectorNode.removeChild(i);
154
+ }
155
+ } else {
156
+ i++;
157
+ }
158
+ currNode = selectorContent[i];
159
+ }
160
+ });
161
+
162
+ // Remove all ruleset with no selectors
163
+ ast.traverseByType('ruleset', (node, index, parent) => {
164
+ const rulesetContent = node.content;
165
+
166
+ // Remove empty selectors and trailing deliminator and space
167
+ let i = 0;
168
+ let currNode = rulesetContent[i];
169
+ while (currNode) {
170
+ if (currNode.is('selector') && currNode.content.length === 0) {
171
+ node.removeChild(i);
172
+ if (rulesetContent[i].is('delimiter')) node.removeChild(i);
173
+ if (rulesetContent[i].is('space')) node.removeChild(i);
174
+ } else {
175
+ i++;
176
+ }
177
+ currNode = rulesetContent[i];
178
+ }
179
+
180
+ // Remove the ruleset if no selectors
181
+ if (rulesetContent.filter(node => node.is('selector')).length === 0) {
182
+ parent.removeChild(index);
183
+ }
184
+ });
185
+ };
186
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["getICSSExportPropsMap","ast","ruleSets","traverseByType","node","push","filter","ruleSet","content","some","item","type","selectorItem","pseudoItem","reduce","result","declarations","flatMap","block","declaration","property","find","propName","getRegularClassesMap","selectors","selector","classes","classNode","ident","getComposesClassesMap","hasComposes","value","valueContent","composedClasses","getExtendClassesMap","extendNodes","extendNode","className","getParentSelectorClassesMap","classesMap","getExtensions","nodeContent","blockContent","rulesetChildren","rulesetDescendants","subItem","rulesetsContent","extensions","identItem","map","length","nestedExtensions","nestedExt","forEach","ext","classNames","hasDeclarations","eliminateGlobals","selectorNode","selectorContent","hasGlobalWithNoArgs","i","currNode","is","identifierNode","removeChild","index","parent","rulesetContent"],"sources":["../../lib/core/traversalUtils.js"],"sourcesContent":["// @flow\n/* eslint-disable no-param-reassign */\n\nimport type { gASTNode } from '../types/index.js';\n\ntype classMapType = {\n  [key: string]: boolean,\n}\n\nexport const getICSSExportPropsMap = (ast: gASTNode): classMapType => {\n  const ruleSets: Array<gASTNode> = [];\n  ast.traverseByType('ruleset', node => ruleSets.push(node));\n\n  return ruleSets\n    .filter(ruleSet => {\n      const content = ruleSet.content || [];\n      return content.some(item =>\n        item.type === 'selector' &&\n        item.content?.some(selectorItem =>\n          selectorItem.type === 'pseudoClass' &&\n          selectorItem.content?.some(pseudoItem =>\n            pseudoItem.type === 'ident' && pseudoItem.content === 'export'\n          )\n        )\n      );\n    })\n    .reduce((result, ruleSet) => {\n      const declarations = (ruleSet.content || [])\n        .filter(item => item.type === 'block')\n        .flatMap(block => (block.content || []))\n        .filter(item => item.type === 'declaration');\n\n      for (const declaration of declarations) {\n        const property = (declaration.content || []).find(item => item.type === 'property');\n        const propName = (property?.content || [])[0]?.content;\n        if (propName) {\n          result[propName] = propName;\n        }\n      }\n\n      return result;\n    }, {});\n};\n\nexport const getRegularClassesMap = (ast: gASTNode): classMapType => {\n  const ruleSets: Array<gASTNode> = [];\n  ast.traverseByType('ruleset', node => ruleSets.push(node));\n\n  return ruleSets\n    .reduce((result, ruleSet) => {\n      const selectors = ruleSet.content?.filter(item => item.type === 'selector') ?? [];\n      for (const selector of selectors) {\n        const classes = selector.content?.filter(item => item.type === 'class') ?? [];\n        for (const classNode of classes) {\n          const ident = classNode.content?.find(item => item.type === 'ident');\n          if (ident?.content) {\n            result[ident.content] = false;\n          }\n        }\n      }\n      return result;\n    }, {});\n};\n\nexport const getComposesClassesMap = (ast: gASTNode): classMapType => {\n  const declarations = [];\n  ast.traverseByType('declaration', node => declarations.push(node));\n\n  return declarations\n    .reduce((result, declaration) => {\n      const content = declaration.content;\n      const property = content?.find(item => item.type === 'property');\n      const hasComposes = property?.content?.some(item =>\n        item.type === 'ident' && item.content === 'composes'\n      );\n\n      if (!hasComposes) return result;\n\n      const value = content?.find(item => item.type === 'value');\n      const valueContent = value?.content;\n\n      // Reject classes composing from other files, e.g. `composes: foo from './other.css'`\n      if (valueContent?.some(item => item.type === 'ident' && item.content === 'from')) {\n        return result;\n      }\n\n      // Extract and mark composed classes\n      const composedClasses = valueContent?.filter(item => item.type === 'ident' && item.content) ?? [];\n      for (const item of composedClasses) {\n        result[item.content] = true;\n      }\n\n      return result;\n    }, {});\n};\n\nexport const getExtendClassesMap = (ast: gASTNode): classMapType => {\n  const extendNodes = [];\n  ast.traverseByType('extend', node => extendNodes.push(node));\n\n  return extendNodes.reduce((result, extendNode) => {\n    const selector = extendNode.content?.find(item => item.type === 'selector');\n    const classNode = selector?.content?.find(item => item.type === 'class');\n    const ident = classNode?.content?.find(item => item.type === 'ident');\n    const className = ident?.content;\n\n    if (className) {\n      result[className] = true; // mark extend classes as true\n    }\n\n    return result;\n  }, {});\n};\n\n/**\n * Resolves parent selectors to their full class names.\n *\n * E.g. `.foo { &_bar {color: blue } }` to `.foo_bar`.\n */\nexport const getParentSelectorClassesMap = (ast: gASTNode): classMapType => {\n  const classesMap: classMapType = {};\n\n  // Recursively traverses down the tree looking for parent selector\n  // extensions. Recursion is necessary as these selectors can be nested.\n  const getExtensions = nodeContent => {\n    const blockContent = nodeContent\n      .filter(item => item.type === 'block')\n      .flatMap(item => item.content || []);\n\n    const rulesetChildren = blockContent.filter(item => item.type === 'ruleset');\n\n    const rulesetDescendants = blockContent\n      .filter(item => item.type === 'include')\n      .flatMap(item => item.content || [])\n      .filter(subItem => subItem.type === 'block')\n      .flatMap(subItem => subItem.content || [])\n      .filter(subItem => subItem.type === 'ruleset');\n\n    const rulesetsContent = [...rulesetChildren, ...rulesetDescendants]\n      .flatMap(item => item.content || []);\n\n    const extensions = rulesetsContent\n      .filter(item => item.type === 'selector')\n      .flatMap(item => item.content || [])\n      .filter(selectorItem => selectorItem.type === 'parentSelectorExtension')\n      .flatMap(selectorItem => selectorItem.content || [])\n      .filter(identItem => identItem.type === 'ident')\n      .map(identItem => identItem.content);\n\n    if (!extensions.length) return [];\n\n    const nestedExtensions = getExtensions(rulesetsContent);\n    const result = extensions;\n    if (nestedExtensions.length) {\n      for (const nestedExt of nestedExtensions) {\n        extensions.forEach(ext => {\n          result.push(ext + nestedExt);\n        });\n      }\n    }\n\n    return result;\n  };\n\n  ast.traverseByType('ruleset', node => {\n    const classNames = (node.content || [])\n      .filter(item => item.type === 'selector')\n      .flatMap(item => item.content || [])\n      .filter(item => item.type === 'class')\n      .flatMap(item => item.content || [])\n      .filter(item => item.type === 'ident' && item.content)\n      .map(item => item.content);\n\n    if (!classNames.length) return;\n\n    const extensions = getExtensions(node.content);\n    if (!extensions.length) return;\n\n    classNames.forEach(className => {\n      extensions.forEach(ext => {\n        classesMap[className + ext] = false;\n      });\n\n      // Ignore the base class if it only exists for nesting parent selectors\n      const hasDeclarations = node.content?.some(item => item.type === 'block' &&\n          item.content?.some(subItem => subItem.type === 'declaration'));\n      if (!hasDeclarations) classesMap[className] = true;\n    });\n  });\n\n  return classesMap;\n};\n\n/**\n * Mutates the AST by removing `:global` instances.\n *\n * For the AST structure:\n * @see https://github.com/css/gonzales/blob/master/doc/AST.CSSP.en.md\n */\nexport const eliminateGlobals = (ast: gASTNode) => {\n  // Remove all :global/:global(...) in selectors\n  ast.traverseByType('selector', (selectorNode) => {\n    const selectorContent = selectorNode.content;\n    let hasGlobalWithNoArgs = false;\n    let i = 0;\n    let currNode = selectorContent[i];\n    while (currNode) {\n      if (currNode.is('pseudoClass')) {\n        // Remove all :global/:global(...) and trailing space\n        const identifierNode = currNode.content[0];\n        if (identifierNode && identifierNode.content === 'global') {\n          if (currNode.content.length === 1) hasGlobalWithNoArgs = true;\n          selectorNode.removeChild(i);\n          if (selectorContent[i] && selectorContent[i].is('space')) {\n            selectorNode.removeChild(i);\n          }\n        } else {\n          i++;\n        }\n      } else if (currNode.is('class') && hasGlobalWithNoArgs) {\n        // Remove all class after :global and their trailing space\n        selectorNode.removeChild(i);\n        if (selectorContent[i] && selectorContent[i].is('space')) {\n          selectorNode.removeChild(i);\n        }\n      } else {\n        i++;\n      }\n\n      currNode = selectorContent[i];\n    }\n  });\n\n  // Remove all ruleset with no selectors\n  ast.traverseByType('ruleset', (node, index, parent) => {\n    const rulesetContent = node.content;\n\n    // Remove empty selectors and trailing deliminator and space\n    let i = 0;\n    let currNode = rulesetContent[i];\n    while (currNode) {\n      if (currNode.is('selector') && currNode.content.length === 0) {\n        node.removeChild(i);\n        if (rulesetContent[i].is('delimiter')) node.removeChild(i);\n        if (rulesetContent[i].is('space')) node.removeChild(i);\n      } else {\n        i++;\n      }\n      currNode = rulesetContent[i];\n    }\n\n    // Remove the ruleset if no selectors\n    if (rulesetContent.filter((node) => node.is('selector')).length === 0) {\n      parent.removeChild(index);\n    }\n  });\n};\n"],"mappings":"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"}
package/build/index.js ADDED
@@ -0,0 +1,27 @@
1
+ import fs from 'node:fs';
2
+ import rules from './rules/index.js';
3
+ const pkg = JSON.parse(fs.readFileSync(new URL('../package.json', import.meta.url), 'utf8'));
4
+ const plugin = {
5
+ meta: {
6
+ name: pkg.name,
7
+ version: pkg.version,
8
+ namespace: 'css-modules'
9
+ },
10
+ configs: {},
11
+ rules
12
+ };
13
+ Object.assign(plugin.configs, {
14
+ recommended: [{
15
+ plugins: {
16
+ 'css-modules': plugin
17
+ },
18
+ rules: {
19
+ 'css-modules/no-unused-class': 2,
20
+ // error
21
+ 'css-modules/no-undef-class': 2 // error
22
+ }
23
+ }]
24
+ });
25
+
26
+ export default plugin;
27
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmcyIsInJ1bGVzIiwicGtnIiwiSlNPTiIsInBhcnNlIiwicmVhZEZpbGVTeW5jIiwiVVJMIiwiaW1wb3J0IiwibWV0YSIsInVybCIsInBsdWdpbiIsIm5hbWUiLCJ2ZXJzaW9uIiwibmFtZXNwYWNlIiwiY29uZmlncyIsIk9iamVjdCIsImFzc2lnbiIsInJlY29tbWVuZGVkIiwicGx1Z2lucyJdLCJzb3VyY2VzIjpbIi4uL2xpYi9pbmRleC5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgZnMgZnJvbSAnbm9kZTpmcyc7XG5pbXBvcnQgcnVsZXMgZnJvbSAnLi9ydWxlcy9pbmRleC5qcyc7XG5cbmNvbnN0IHBrZyA9IEpTT04ucGFyc2UoXG4gIGZzLnJlYWRGaWxlU3luYyhuZXcgVVJMKCcuLi9wYWNrYWdlLmpzb24nLCBpbXBvcnQubWV0YS51cmwpLCAndXRmOCcpLFxuKTtcblxuY29uc3QgcGx1Z2luID0ge1xuICBtZXRhOiB7XG4gICAgbmFtZTogcGtnLm5hbWUsXG4gICAgdmVyc2lvbjogcGtnLnZlcnNpb24sXG4gICAgbmFtZXNwYWNlOiAnY3NzLW1vZHVsZXMnLFxuICB9LFxuICBjb25maWdzOiB7fSxcbiAgcnVsZXMsXG59O1xuXG5PYmplY3QuYXNzaWduKHBsdWdpbi5jb25maWdzLCB7XG4gIHJlY29tbWVuZGVkOiBbXG4gICAge1xuICAgICAgcGx1Z2luczoge1xuICAgICAgICAnY3NzLW1vZHVsZXMnOiBwbHVnaW4sXG4gICAgICB9LFxuICAgICAgcnVsZXM6IHtcbiAgICAgICAgJ2Nzcy1tb2R1bGVzL25vLXVudXNlZC1jbGFzcyc6IDIsIC8vIGVycm9yXG4gICAgICAgICdjc3MtbW9kdWxlcy9uby11bmRlZi1jbGFzcyc6IDIsIC8vIGVycm9yXG4gICAgICB9LFxuICAgIH0sXG4gIF0sXG59KTtcblxuZXhwb3J0IGRlZmF1bHQgcGx1Z2luO1xuIl0sIm1hcHBpbmdzIjoiQUFBQSxPQUFPQSxFQUFFLE1BQU0sU0FBUztBQUN4QixPQUFPQyxLQUFLLE1BQU0sa0JBQWtCO0FBRXBDLE1BQU1DLEdBQUcsR0FBR0MsSUFBSSxDQUFDQyxLQUFLLENBQ3BCSixFQUFFLENBQUNLLFlBQVksQ0FBQyxJQUFJQyxHQUFHLENBQUMsaUJBQWlCLEVBQUVDLE1BQU0sQ0FBQ0MsSUFBSSxDQUFDQyxHQUFHLENBQUMsRUFBRSxNQUFNLENBQ3JFLENBQUM7QUFFRCxNQUFNQyxNQUFNLEdBQUc7RUFDYkYsSUFBSSxFQUFFO0lBQ0pHLElBQUksRUFBRVQsR0FBRyxDQUFDUyxJQUFJO0lBQ2RDLE9BQU8sRUFBRVYsR0FBRyxDQUFDVSxPQUFPO0lBQ3BCQyxTQUFTLEVBQUU7RUFDYixDQUFDO0VBQ0RDLE9BQU8sRUFBRSxDQUFDLENBQUM7RUFDWGI7QUFDRixDQUFDO0FBRURjLE1BQU0sQ0FBQ0MsTUFBTSxDQUFDTixNQUFNLENBQUNJLE9BQU8sRUFBRTtFQUM1QkcsV0FBVyxFQUFFLENBQ1g7SUFDRUMsT0FBTyxFQUFFO01BQ1AsYUFBYSxFQUFFUjtJQUNqQixDQUFDO0lBQ0RULEtBQUssRUFBRTtNQUNMLDZCQUE2QixFQUFFLENBQUM7TUFBRTtNQUNsQyw0QkFBNEIsRUFBRSxDQUFDLENBQUU7SUFDbkM7RUFDRixDQUFDO0FBRUwsQ0FBQyxDQUFDOztBQUVGLGVBQWVTLE1BQU0ifQ==
@@ -0,0 +1,7 @@
1
+ import noUnusedClass from './no-unused-class.js';
2
+ import noUndefClass from './no-undef-class.js';
3
+ export default {
4
+ 'no-unused-class': noUnusedClass,
5
+ 'no-undef-class': noUndefClass
6
+ };
7
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJub1VudXNlZENsYXNzIiwibm9VbmRlZkNsYXNzIl0sInNvdXJjZXMiOlsiLi4vLi4vbGliL3J1bGVzL2luZGV4LmpzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBub1VudXNlZENsYXNzIGZyb20gJy4vbm8tdW51c2VkLWNsYXNzLmpzJztcbmltcG9ydCBub1VuZGVmQ2xhc3MgZnJvbSAnLi9uby11bmRlZi1jbGFzcy5qcyc7XG5cbmV4cG9ydCBkZWZhdWx0IHtcbiAgJ25vLXVudXNlZC1jbGFzcyc6IG5vVW51c2VkQ2xhc3MsXG4gICduby11bmRlZi1jbGFzcyc6IG5vVW5kZWZDbGFzcyxcbn07XG4iXSwibWFwcGluZ3MiOiJBQUFBLE9BQU9BLGFBQWEsTUFBTSxzQkFBc0I7QUFDaEQsT0FBT0MsWUFBWSxNQUFNLHFCQUFxQjtBQUU5QyxlQUFlO0VBQ2IsaUJBQWlCLEVBQUVELGFBQWE7RUFDaEMsZ0JBQWdCLEVBQUVDO0FBQ3BCLENBQUMifQ==
@@ -0,0 +1,86 @@
1
+ import { getStyleImportNodeData, getAST, fileExists, getStyleClasses, getPropertyName, getClassesMap, getExportPropsMap, getFilePath } from '../core/index.js';
2
+ export default {
3
+ meta: {
4
+ docs: {
5
+ description: 'Checks that you are using the existent css/scss/less classes',
6
+ recommended: true
7
+ },
8
+ schema: [{
9
+ type: 'object',
10
+ properties: {
11
+ camelCase: {
12
+ enum: [true, 'dashes', 'only', 'dashes-only']
13
+ }
14
+ }
15
+ }]
16
+ },
17
+ create(context) {
18
+ const camelCase = context?.options[0]?.camelCase;
19
+
20
+ /*
21
+ maps variable name to property Object
22
+ map = {
23
+ [variableName]: {
24
+ classesMap: { foo: 'foo', fooBar: 'foo-bar', 'foo-bar': 'foo-bar' },
25
+ node: {...}
26
+ }
27
+ }
28
+ example:
29
+ import s from './foo.scss';
30
+ s is variable name
31
+ property Object has two keys
32
+ 1. classesMap: an object with propertyName as key and its className as value
33
+ 2. node: node that correspond to s (see example above)
34
+ */
35
+ const map = {};
36
+ return {
37
+ ImportDeclaration(node) {
38
+ const styleImportNodeData = getStyleImportNodeData(node);
39
+ if (!styleImportNodeData) {
40
+ return;
41
+ }
42
+ const {
43
+ importName,
44
+ styleFilePath,
45
+ importNode
46
+ } = styleImportNodeData;
47
+ const styleFileAbsolutePath = getFilePath(context, styleFilePath);
48
+ let classesMap = {};
49
+ let exportPropsMap = {};
50
+ if (fileExists(styleFileAbsolutePath)) {
51
+ const ast = getAST(styleFileAbsolutePath);
52
+ const classes = ast && getStyleClasses(ast);
53
+ classesMap = classes && getClassesMap(classes, camelCase);
54
+ exportPropsMap = ast && getExportPropsMap(ast);
55
+ }
56
+ map[importName] ??= {};
57
+
58
+ // this will be used to check if classes are defined
59
+ map[importName].classesMap = classesMap;
60
+
61
+ // this will be used to check if :export properties are defined
62
+ map[importName].exportPropsMap = exportPropsMap;
63
+
64
+ // save node for reporting unused styles
65
+ map[importName].node = importNode;
66
+ },
67
+ MemberExpression: node => {
68
+ /*
69
+ Check if property exists in css/scss file as class
70
+ */
71
+
72
+ const objectName = node.object.name;
73
+ const propertyName = getPropertyName(node, camelCase);
74
+ if (!propertyName) {
75
+ return;
76
+ }
77
+ const classesMap = map[objectName]?.classesMap;
78
+ const exportPropsMap = map[objectName]?.exportPropsMap;
79
+ if (classesMap && classesMap[propertyName] == null && exportPropsMap && exportPropsMap[propertyName] == null) {
80
+ context.report(node.property, `Class or exported property '${propertyName}' not found`);
81
+ }
82
+ }
83
+ };
84
+ }
85
+ };
86
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJnZXRTdHlsZUltcG9ydE5vZGVEYXRhIiwiZ2V0QVNUIiwiZmlsZUV4aXN0cyIsImdldFN0eWxlQ2xhc3NlcyIsImdldFByb3BlcnR5TmFtZSIsImdldENsYXNzZXNNYXAiLCJnZXRFeHBvcnRQcm9wc01hcCIsImdldEZpbGVQYXRoIiwibWV0YSIsImRvY3MiLCJkZXNjcmlwdGlvbiIsInJlY29tbWVuZGVkIiwic2NoZW1hIiwidHlwZSIsInByb3BlcnRpZXMiLCJjYW1lbENhc2UiLCJlbnVtIiwiY3JlYXRlIiwiY29udGV4dCIsIm9wdGlvbnMiLCJtYXAiLCJJbXBvcnREZWNsYXJhdGlvbiIsIm5vZGUiLCJzdHlsZUltcG9ydE5vZGVEYXRhIiwiaW1wb3J0TmFtZSIsInN0eWxlRmlsZVBhdGgiLCJpbXBvcnROb2RlIiwic3R5bGVGaWxlQWJzb2x1dGVQYXRoIiwiY2xhc3Nlc01hcCIsImV4cG9ydFByb3BzTWFwIiwiYXN0IiwiY2xhc3NlcyIsIk1lbWJlckV4cHJlc3Npb24iLCJvYmplY3ROYW1lIiwib2JqZWN0IiwibmFtZSIsInByb3BlcnR5TmFtZSIsInJlcG9ydCIsInByb3BlcnR5Il0sInNvdXJjZXMiOlsiLi4vLi4vbGliL3J1bGVzL25vLXVuZGVmLWNsYXNzLmpzIl0sInNvdXJjZXNDb250ZW50IjpbIi8qIEBmbG93ICovXG5pbXBvcnQge1xuICBnZXRTdHlsZUltcG9ydE5vZGVEYXRhLFxuICBnZXRBU1QsXG4gIGZpbGVFeGlzdHMsXG4gIGdldFN0eWxlQ2xhc3NlcyxcbiAgZ2V0UHJvcGVydHlOYW1lLFxuICBnZXRDbGFzc2VzTWFwLFxuICBnZXRFeHBvcnRQcm9wc01hcCxcbiAgZ2V0RmlsZVBhdGgsXG59IGZyb20gJy4uL2NvcmUvaW5kZXguanMnO1xuXG5pbXBvcnQgdHlwZSB7IEpzTm9kZSB9IGZyb20gJy4uL3R5cGVzL2luZGV4LmpzJztcblxuZXhwb3J0IGRlZmF1bHQge1xuICBtZXRhOiB7XG4gICAgZG9jczoge1xuICAgICAgZGVzY3JpcHRpb246ICdDaGVja3MgdGhhdCB5b3UgYXJlIHVzaW5nIHRoZSBleGlzdGVudCBjc3Mvc2Nzcy9sZXNzIGNsYXNzZXMnLFxuICAgICAgcmVjb21tZW5kZWQ6IHRydWUsXG4gICAgfSxcbiAgICBzY2hlbWE6IFtcbiAgICAgIHtcbiAgICAgICAgdHlwZTogJ29iamVjdCcsXG4gICAgICAgIHByb3BlcnRpZXM6IHtcbiAgICAgICAgICBjYW1lbENhc2U6IHsgZW51bTogW3RydWUsICdkYXNoZXMnLCAnb25seScsICdkYXNoZXMtb25seSddIH1cbiAgICAgICAgfSxcbiAgICAgIH1cbiAgICBdLFxuICB9LFxuICBjcmVhdGUgKGNvbnRleHQ6IE9iamVjdCkge1xuICAgIGNvbnN0IGNhbWVsQ2FzZSA9IGNvbnRleHQ/Lm9wdGlvbnNbMF0/LmNhbWVsQ2FzZTtcblxuICAgIC8qXG4gICAgICAgbWFwcyB2YXJpYWJsZSBuYW1lIHRvIHByb3BlcnR5IE9iamVjdFxuICAgICAgIG1hcCA9IHtcbiAgICAgICAgIFt2YXJpYWJsZU5hbWVdOiB7XG4gICAgICAgICAgIGNsYXNzZXNNYXA6IHsgZm9vOiAnZm9vJywgZm9vQmFyOiAnZm9vLWJhcicsICdmb28tYmFyJzogJ2Zvby1iYXInIH0sXG4gICAgICAgICAgIG5vZGU6IHsuLi59XG4gICAgICAgICB9XG4gICAgICAgfVxuXG4gICAgICAgZXhhbXBsZTpcbiAgICAgICBpbXBvcnQgcyBmcm9tICcuL2Zvby5zY3NzJztcbiAgICAgICBzIGlzIHZhcmlhYmxlIG5hbWVcblxuICAgICAgIHByb3BlcnR5IE9iamVjdCBoYXMgdHdvIGtleXNcbiAgICAgICAxLiBjbGFzc2VzTWFwOiBhbiBvYmplY3Qgd2l0aCBwcm9wZXJ0eU5hbWUgYXMga2V5IGFuZCBpdHMgY2xhc3NOYW1lIGFzIHZhbHVlXG4gICAgICAgMi4gbm9kZTogbm9kZSB0aGF0IGNvcnJlc3BvbmQgdG8gcyAoc2VlIGV4YW1wbGUgYWJvdmUpXG4gICAgICovXG4gICAgY29uc3QgbWFwID0ge307XG5cbiAgICByZXR1cm4ge1xuICAgICAgSW1wb3J0RGVjbGFyYXRpb24gKG5vZGU6IEpzTm9kZSkge1xuICAgICAgICBjb25zdCBzdHlsZUltcG9ydE5vZGVEYXRhID0gZ2V0U3R5bGVJbXBvcnROb2RlRGF0YShub2RlKTtcblxuICAgICAgICBpZiAoIXN0eWxlSW1wb3J0Tm9kZURhdGEpIHtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCB7XG4gICAgICAgICAgaW1wb3J0TmFtZSxcbiAgICAgICAgICBzdHlsZUZpbGVQYXRoLFxuICAgICAgICAgIGltcG9ydE5vZGUsXG4gICAgICAgIH0gPSBzdHlsZUltcG9ydE5vZGVEYXRhO1xuXG4gICAgICAgIGNvbnN0IHN0eWxlRmlsZUFic29sdXRlUGF0aCA9IGdldEZpbGVQYXRoKGNvbnRleHQsIHN0eWxlRmlsZVBhdGgpO1xuICAgICAgICBsZXQgY2xhc3Nlc01hcCA9IHt9O1xuICAgICAgICBsZXQgZXhwb3J0UHJvcHNNYXAgPSB7fTtcblxuICAgICAgICBpZiAoZmlsZUV4aXN0cyhzdHlsZUZpbGVBYnNvbHV0ZVBhdGgpKSB7XG4gICAgICAgICAgY29uc3QgYXN0ID0gZ2V0QVNUKHN0eWxlRmlsZUFic29sdXRlUGF0aCk7XG4gICAgICAgICAgY29uc3QgY2xhc3NlcyA9IGFzdCAmJiBnZXRTdHlsZUNsYXNzZXMoYXN0KTtcblxuICAgICAgICAgIGNsYXNzZXNNYXAgPSBjbGFzc2VzICYmIGdldENsYXNzZXNNYXAoY2xhc3NlcywgY2FtZWxDYXNlKTtcbiAgICAgICAgICBleHBvcnRQcm9wc01hcCA9IGFzdCAmJiBnZXRFeHBvcnRQcm9wc01hcChhc3QpO1xuICAgICAgICB9XG5cbiAgICAgICAgbWFwW2ltcG9ydE5hbWVdID8/PSB7fTtcblxuICAgICAgICAvLyB0aGlzIHdpbGwgYmUgdXNlZCB0byBjaGVjayBpZiBjbGFzc2VzIGFyZSBkZWZpbmVkXG4gICAgICAgIG1hcFtpbXBvcnROYW1lXS5jbGFzc2VzTWFwID0gY2xhc3Nlc01hcDtcblxuICAgICAgICAvLyB0aGlzIHdpbGwgYmUgdXNlZCB0byBjaGVjayBpZiA6ZXhwb3J0IHByb3BlcnRpZXMgYXJlIGRlZmluZWRcbiAgICAgICAgbWFwW2ltcG9ydE5hbWVdLmV4cG9ydFByb3BzTWFwID0gZXhwb3J0UHJvcHNNYXA7XG5cbiAgICAgICAgLy8gc2F2ZSBub2RlIGZvciByZXBvcnRpbmcgdW51c2VkIHN0eWxlc1xuICAgICAgICBtYXBbaW1wb3J0TmFtZV0ubm9kZSA9IGltcG9ydE5vZGU7XG4gICAgICB9LFxuICAgICAgTWVtYmVyRXhwcmVzc2lvbjogKG5vZGU6IEpzTm9kZSkgPT4ge1xuICAgICAgICAvKlxuICAgICAgICAgICBDaGVjayBpZiBwcm9wZXJ0eSBleGlzdHMgaW4gY3NzL3Njc3MgZmlsZSBhcyBjbGFzc1xuICAgICAgICAgKi9cblxuICAgICAgICBjb25zdCBvYmplY3ROYW1lID0gbm9kZS5vYmplY3QubmFtZTtcblxuICAgICAgICBjb25zdCBwcm9wZXJ0eU5hbWUgPSBnZXRQcm9wZXJ0eU5hbWUobm9kZSwgY2FtZWxDYXNlKTtcblxuICAgICAgICBpZiAoIXByb3BlcnR5TmFtZSkge1xuICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGNsYXNzZXNNYXAgPSBtYXBbb2JqZWN0TmFtZV0/LmNsYXNzZXNNYXA7XG4gICAgICAgIGNvbnN0IGV4cG9ydFByb3BzTWFwID0gbWFwW29iamVjdE5hbWVdPy5leHBvcnRQcm9wc01hcDtcblxuICAgICAgICBpZiAoY2xhc3Nlc01hcCAmJiBjbGFzc2VzTWFwW3Byb3BlcnR5TmFtZV0gPT0gbnVsbCAmJlxuICAgICAgICAgICAgZXhwb3J0UHJvcHNNYXAgJiYgZXhwb3J0UHJvcHNNYXBbcHJvcGVydHlOYW1lXSA9PSBudWxsKSB7XG4gICAgICAgICAgY29udGV4dC5yZXBvcnQobm9kZS5wcm9wZXJ0eSwgYENsYXNzIG9yIGV4cG9ydGVkIHByb3BlcnR5ICcke3Byb3BlcnR5TmFt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
@@ -0,0 +1,132 @@
1
+ import path from 'node:path';
2
+ import { getStyleImportNodeData, getStyleClasses, getPropertyName, getClassesMap, getFilePath, getAST, fileExists } from '../core/index.js';
3
+ export default {
4
+ meta: {
5
+ docs: {
6
+ description: 'Checks that you are using all css/scss/less classes',
7
+ recommended: true
8
+ },
9
+ schema: [{
10
+ type: 'object',
11
+ properties: {
12
+ camelCase: {
13
+ enum: [true, 'dashes', 'only', 'dashes-only']
14
+ },
15
+ markAsUsed: {
16
+ type: 'array'
17
+ }
18
+ }
19
+ }]
20
+ },
21
+ create(context) {
22
+ const markAsUsed = context?.options?.[0]?.markAsUsed;
23
+ const camelCase = context?.options?.[0]?.camelCase;
24
+
25
+ /*
26
+ maps variable name to property Object
27
+ map = {
28
+ [variableName]: {
29
+ classes: { foo: false, 'foo-bar': false },
30
+ classesMap: { foo: 'foo', fooBar: 'foo-bar', 'foo-bar': 'foo-bar' },
31
+ node: {...}
32
+ }
33
+ }
34
+ example:
35
+ import s from './foo.scss';
36
+ s is variable name
37
+ property Object has two keys
38
+ 1. classes: an object with className as key and a boolean as value. The boolean is marked if it is used in file
39
+ 2. classesMap: an object with propertyName as key and its className as value
40
+ 3. node: node that correspond to s (see example above)
41
+ */
42
+ const map = {};
43
+ return {
44
+ ImportDeclaration(node) {
45
+ const styleImportNodeData = getStyleImportNodeData(node);
46
+ if (!styleImportNodeData) {
47
+ return;
48
+ }
49
+ const {
50
+ importName,
51
+ styleFilePath,
52
+ importNode
53
+ } = styleImportNodeData;
54
+ const styleFileAbsolutePath = getFilePath(context, styleFilePath);
55
+ let classes = {};
56
+ let classesMap = {};
57
+ if (fileExists(styleFileAbsolutePath)) {
58
+ // this will be used to mark s.foo as used in MemberExpression
59
+ const ast = getAST(styleFileAbsolutePath);
60
+ classes = ast && getStyleClasses(ast);
61
+ classesMap = classes && getClassesMap(classes, camelCase);
62
+ }
63
+ map[importName] ??= {};
64
+ map[importName].classes = classes;
65
+ map[importName].classesMap = classesMap;
66
+
67
+ // save node for reporting unused styles
68
+ map[importName].node = importNode;
69
+
70
+ // save file path for reporting unused styles
71
+ map[importName].filePath = styleFilePath;
72
+ },
73
+ MemberExpression: node => {
74
+ /*
75
+ Check if property exists in css/scss file as class
76
+ */
77
+
78
+ const objectName = node.object.name;
79
+ const propertyName = getPropertyName(node, camelCase);
80
+ if (!propertyName) {
81
+ return;
82
+ }
83
+ const className = map[objectName]?.classesMap?.[propertyName];
84
+ if (className == null) {
85
+ return;
86
+ }
87
+
88
+ // mark this property has used
89
+ (map[objectName] ??= {
90
+ classes: {}
91
+ }).classes[className] = true;
92
+ },
93
+ 'Program:exit'() {
94
+ /*
95
+ Check if all classes defined in css/scss file are used
96
+ */
97
+
98
+ /*
99
+ we are looping over each import style node in program
100
+ example:
101
+ ```
102
+ import s from './foo.css';
103
+ import x from './bar.scss';
104
+ ```
105
+ then the loop will be run 2 times
106
+ */
107
+ for (const o of Object.values(map)) {
108
+ const {
109
+ classes,
110
+ node,
111
+ filePath
112
+ } = o;
113
+
114
+ /*
115
+ if option is passed to mark a class as used, example:
116
+ eslint css-modules/no-unused-class: [2, { markAsUsed: ['container'] }]
117
+ */
118
+ for (const usedClass of markAsUsed ?? []) {
119
+ classes[usedClass] = true;
120
+ }
121
+
122
+ // classNames not marked as true are unused
123
+ const unusedClasses = classes ? Object.entries(classes).filter(([_k, v]) => !v).map(([k, _v]) => k) : [];
124
+ if (unusedClasses.length > 0) {
125
+ context.report(node, `Unused classes found in ${path.basename(filePath)}: ${unusedClasses.join(', ')}`);
126
+ }
127
+ }
128
+ }
129
+ };
130
+ }
131
+ };
132
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["path","getStyleImportNodeData","getStyleClasses","getPropertyName","getClassesMap","getFilePath","getAST","fileExists","meta","docs","description","recommended","schema","type","properties","camelCase","enum","markAsUsed","create","context","options","map","ImportDeclaration","node","styleImportNodeData","importName","styleFilePath","importNode","styleFileAbsolutePath","classes","classesMap","ast","filePath","MemberExpression","objectName","object","name","propertyName","className","Program:exit","o","Object","values","usedClass","unusedClasses","entries","filter","_k","v","k","_v","length","report","basename","join"],"sources":["../../lib/rules/no-unused-class.js"],"sourcesContent":["/* @flow */\nimport path from 'node:path';\n\nimport {\n  getStyleImportNodeData,\n  getStyleClasses,\n  getPropertyName,\n  getClassesMap,\n  getFilePath,\n  getAST,\n  fileExists,\n} from '../core/index.js';\n\nimport type { JsNode } from '../types/index.js';\n\nexport default {\n  meta: {\n    docs: {\n      description: 'Checks that you are using all css/scss/less classes',\n      recommended: true,\n    },\n    schema: [\n      {\n        type: 'object',\n        properties: {\n          camelCase: { enum: [true, 'dashes', 'only', 'dashes-only'] },\n          markAsUsed: { type: 'array' },\n        },\n      }\n    ],\n  },\n  create (context: Object) {\n    const markAsUsed = context?.options?.[0]?.markAsUsed;\n    const camelCase = context?.options?.[0]?.camelCase;\n\n    /*\n       maps variable name to property Object\n       map = {\n         [variableName]: {\n           classes: { foo: false, 'foo-bar': false },\n           classesMap: { foo: 'foo', fooBar: 'foo-bar', 'foo-bar': 'foo-bar' },\n           node: {...}\n         }\n       }\n\n       example:\n       import s from './foo.scss';\n       s is variable name\n\n       property Object has two keys\n       1. classes: an object with className as key and a boolean as value. The boolean is marked if it is used in file\n       2. classesMap: an object with propertyName as key and its className as value\n       3. node: node that correspond to s (see example above)\n     */\n    const map = {};\n\n    return {\n      ImportDeclaration (node: JsNode) {\n        const styleImportNodeData = getStyleImportNodeData(node);\n\n        if (!styleImportNodeData) {\n          return;\n        }\n\n        const {\n          importName,\n          styleFilePath,\n          importNode,\n        } = styleImportNodeData;\n\n        const styleFileAbsolutePath = getFilePath(context, styleFilePath);\n\n        let classes = {};\n        let classesMap = {};\n\n        if (fileExists(styleFileAbsolutePath)) {\n          // this will be used to mark s.foo as used in MemberExpression\n          const ast = getAST(styleFileAbsolutePath);\n          classes = ast && getStyleClasses(ast);\n          classesMap = classes && getClassesMap(classes, camelCase);\n        }\n\n        map[importName] ??= {};\n\n        map[importName].classes = classes;\n        map[importName].classesMap = classesMap;\n\n        // save node for reporting unused styles\n        map[importName].node = importNode;\n\n        // save file path for reporting unused styles\n        map[importName].filePath = styleFilePath;\n      },\n      MemberExpression: (node: JsNode) => {\n        /*\n           Check if property exists in css/scss file as class\n         */\n\n        const objectName = node.object.name;\n        const propertyName = getPropertyName(node, camelCase);\n\n        if (!propertyName) {\n          return;\n        }\n\n        const className = map[objectName]?.classesMap?.[propertyName];\n\n        if (className == null) {\n          return;\n        }\n\n        // mark this property has used\n        (map[objectName] ??= { classes: {} }).classes[className] = true;\n      },\n      'Program:exit' () {\n        /*\n           Check if all classes defined in css/scss file are used\n         */\n\n        /*\n           we are looping over each import style node in program\n           example:\n           ```\n             import s from './foo.css';\n             import x from './bar.scss';\n           ```\n           then the loop will be run 2 times\n         */\n        for (const o of Object.values(map)) {\n          const { classes, node, filePath } = o;\n\n          /*\n             if option is passed to mark a class as used, example:\n             eslint css-modules/no-unused-class: [2, { markAsUsed: ['container'] }]\n           */\n          for (const usedClass of markAsUsed ?? []) {\n            classes[usedClass] = true;\n          }\n\n          // classNames not marked as true are unused\n          const unusedClasses = classes ? Object.entries(classes).filter(([_k, v]) => !v).map(([k, _v]) => k) : [];\n\n          if (unusedClasses.length > 0) {\n            context.report(node, `Unused classes found in ${path.basename(filePath)}: ${unusedClasses.join(', ')}`);\n          }\n        }\n      }\n    };\n  }\n};\n"],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+
2
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vLi4vbGliL3R5cGVzL2luZGV4LmpzIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIGpzIE5vZGVcbmV4cG9ydCB0eXBlIFBvc2l0aW9uID0ge1xuICBsaW5lOiBudW1iZXIsIC8vIDEgaW5kZXhlZFxuICBjb2x1bW46IG51bWJlciwgLy8gMCBpbmRleGVkXG59O1xuXG5leHBvcnQgdHlwZSBTb3VyY2VMb2NhdGlvbiA9IHtcbiAgc3RhcnQ6IFBvc2l0aW9uLFxuICBlbmQ6IFBvc2l0aW9uLFxuICBpZGVudGlmaWVyTmFtZT86IHN0cmluZyxcbn07XG5cbmV4cG9ydCB0eXBlIEpzTm9kZSA9IHtcbiAgdHlwZTogJ0ltcG9ydERlY2xhcmF0aW9uJyB8ICdJbXBvcnREZWZhdWx0U3BlY2lmaWVyJyxcbiAgc3RhcnQ6IG51bWJlcixcbiAgZW5kOiBudW1iZXIsXG4gIGxvYzogSnNOb2RlLFxuICBsb2NhbD86IEpzTm9kZSxcbiAgbmFtZT86IHN0cmluZyxcbiAgdmFsdWU/OiBzdHJpbmcsXG4gIHNwZWNpZmllcnM/OiBBcnJheTxKc05vZGU+LFxuICBpbXBvcnRLaW5kPzogJ3ZhbHVlJyxcbiAgZXh0cmE/OiB7XG4gICAgcmF3VmFsdWU6IHN0cmluZyxcbiAgICByYXc6IHN0cmluZyxcbiAgfSxcbiAgc291cmNlOiBKc05vZGUsXG4gIHJhbmdlOiBBcnJheTxudW1iZXI+LCAvLyBtb3N0IHByb2JhYmx5IGFycmF5IG9mIDIgbnVtYmVycyA/LFxuICBfYmFiZWxUeXBlOiBzdHJpbmcsXG4gIHBhcmVudDogSnNOb2RlLFxufTtcblxuLy8gZ29uemFsZXMgQVNUIE5vZGUgVHlwZVxuZXhwb3J0IHR5cGUgZ0FTVE5vZGUgPSB7XG4gIHRyYXZlcnNlQnlUeXBlOiBGdW5jdGlvbixcblxuICB0eXBlOiAnc3R5bGVzaGVldCdcbiAgICAgIHwgJ2lkZW50J1xuICAgICAgfCAnY2xhc3MnXG4gICAgICB8ICdzZWxlY3RvcidcbiAgICAgIHwgJ3ZhbHVlJ1xuICAgICAgfCAncHJvcGVydHknXG4gICAgICB8ICdydWxlc2V0J1xuICAgICAgfCAnZXh0ZW5kJ1xuICAgICAgfCAnZGVjbGFyYXRpb24nLFxuICBjb250ZW50OiBzdHJpbmcgfCBBcnJheTxnQVNUTm9kZT4sXG4gIHN5bnRheDogJ2NzcycgfCAnc2NzcycgfCAnbGVzcycsXG59O1xuIl0sIm1hcHBpbmdzIjoiIn0=
package/package.json ADDED
@@ -0,0 +1,66 @@
1
+ {
2
+ "name": "@bhollis/eslint-plugin-css-modules",
3
+ "version": "1.0.0",
4
+ "description": "Checks that you are using the existent css/scss/less classes, no more no less",
5
+ "main": "build/index.js",
6
+ "files": [
7
+ "build",
8
+ "packages"
9
+ ],
10
+ "type": "module",
11
+ "scripts": {
12
+ "watch": "babel lib -d build --watch",
13
+ "build": "rm -rf build && babel lib -d build",
14
+ "lint": "eslint lib test",
15
+ "test": "npm run build && rm -rf test-out && babel test -d test-out && cp -r test/files test-out/ && node --test test-out/**/*.test.js",
16
+ "my-pre-publish": "npm run test && npm run build",
17
+ "my-publish": "npm run my-pre-publish && yarn publish"
18
+ },
19
+ "engines": {
20
+ "node": ">=20.0.0"
21
+ },
22
+ "keywords": [
23
+ "eslint",
24
+ "eslintplugin",
25
+ "eslint-plugin",
26
+ "css-modules"
27
+ ],
28
+ "author": {
29
+ "name": "Atif Afzal",
30
+ "email": "atif5801@gmail.com",
31
+ "url": "http://atfzl.me"
32
+ },
33
+ "repository": {
34
+ "type": "git",
35
+ "url": "git://github.com/bhollis/eslint-plugin-css-modules.git"
36
+ },
37
+ "license": "MIT",
38
+ "bugs": {
39
+ "url": "https://github.com/bhollis/eslint-plugin-css-modules/issues"
40
+ },
41
+ "homepage": "https://github.com/bhollis/eslint-plugin-css-modules#readme",
42
+ "peerDependencies": {
43
+ "eslint": ">=2.0.0"
44
+ },
45
+ "devDependencies": {
46
+ "@babel/cli": "^7.23.0",
47
+ "@babel/core": "^7.23.0",
48
+ "@babel/eslint-parser": "^7.22.15",
49
+ "@babel/plugin-proposal-export-default-from": "^7.22.17",
50
+ "@babel/plugin-syntax-flow": "^7.22.5",
51
+ "@babel/plugin-transform-flow-strip-types": "^7.22.5",
52
+ "@babel/preset-env": "^7.22.20",
53
+ "@babel/register": "^7.22.15",
54
+ "eslint": "^8.50.0",
55
+ "eslint-config-standard": "^17.1.0",
56
+ "eslint-plugin-import": "^2.28.1",
57
+ "eslint-plugin-n": "^16.1.0",
58
+ "eslint-plugin-promise": "^6.1.1",
59
+ "flow-bin": "^0.36.0",
60
+ "nodemon": "^3.0.1"
61
+ },
62
+ "dependencies": {
63
+ "es-toolkit": "^1.42.0",
64
+ "gonzales-pe": "^4.3.0"
65
+ }
66
+ }