@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 +142 -0
- package/build/core/index.js +138 -0
- package/build/core/traversalUtils.js +186 -0
- package/build/index.js +27 -0
- package/build/rules/index.js +7 -0
- package/build/rules/no-undef-class.js +86 -0
- package/build/rules/no-unused-class.js +132 -0
- package/build/types/index.js +2 -0
- package/package.json +66 -0
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
|
+

|
|
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
|
+
}
|