eslint-plugin-css-modules
Version:
Checks that you are using the existent css/scss/less classes, no more no less
135 lines (126 loc) • 16.1 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports["default"] = void 0;
var _fp = _interopRequireDefault(require("lodash/fp"));
var _lodash = _interopRequireDefault(require("lodash"));
var _path = _interopRequireDefault(require("path"));
var _core = require("../core");
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }
var _default = exports["default"] = {
meta: {
docs: {
description: 'Checks that you are using all css/scss/less classes',
recommended: true
},
schema: [{
type: 'object',
properties: {
camelCase: {
"enum": [true, 'dashes', 'only', 'dashes-only']
},
markAsUsed: {
type: 'array'
}
}
}]
},
create: function create(context) {
var markAsUsed = _lodash["default"].get(context, 'options[0].markAsUsed');
var camelCase = _lodash["default"].get(context, 'options[0].camelCase');
/*
maps variable name to property Object
map = {
[variableName]: {
classes: { foo: false, 'foo-bar': false },
classesMap: { foo: 'foo', fooBar: 'foo-bar', 'foo-bar': 'foo-bar' },
node: {...}
}
}
example:
import s from './foo.scss';
s is variable name
property Object has two keys
1. classes: an object with className as key and a boolean as value. The boolean is marked if it is used in file
2. classesMap: an object with propertyName as key and its className as value
3. node: node that correspond to s (see example above)
*/
var map = {};
return {
ImportDeclaration: function ImportDeclaration(node) {
var styleImportNodeData = (0, _core.getStyleImportNodeData)(node);
if (!styleImportNodeData) {
return;
}
var importName = styleImportNodeData.importName,
styleFilePath = styleImportNodeData.styleFilePath,
importNode = styleImportNodeData.importNode;
var styleFileAbsolutePath = (0, _core.getFilePath)(context, styleFilePath);
var classes = {};
var classesMap = {};
if ((0, _core.fileExists)(styleFileAbsolutePath)) {
// this will be used to mark s.foo as used in MemberExpression
var ast = (0, _core.getAST)(styleFileAbsolutePath);
classes = ast && (0, _core.getStyleClasses)(ast);
classesMap = classes && (0, _core.getClassesMap)(classes, camelCase);
}
_lodash["default"].set(map, "".concat(importName, ".classes"), classes);
_lodash["default"].set(map, "".concat(importName, ".classesMap"), classesMap);
// save node for reporting unused styles
_lodash["default"].set(map, "".concat(importName, ".node"), importNode);
// save file path for reporting unused styles
_lodash["default"].set(map, "".concat(importName, ".filePath"), styleFilePath);
},
MemberExpression: function MemberExpression(node) {
/*
Check if property exists in css/scss file as class
*/
var objectName = node.object.name;
var propertyName = (0, _core.getPropertyName)(node, camelCase);
if (!propertyName) {
return;
}
var className = _lodash["default"].get(map, "".concat(objectName, ".classesMap.").concat(propertyName));
if (className == null) {
return;
}
// mark this property has used
_lodash["default"].set(map, "".concat(objectName, ".classes.").concat(className), true);
},
'Program:exit': function ProgramExit() {
/*
Check if all classes defined in css/scss file are used
*/
/*
we are looping over each import style node in program
example:
```
import s from './foo.css';
import x from './bar.scss';
```
then the loop will be run 2 times
*/
_lodash["default"].forIn(map, function (o) {
var classes = o.classes,
node = o.node,
filePath = o.filePath;
/*
if option is passed to mark a class as used, example:
eslint css-modules/no-unused-class: [2, { markAsUsed: ['container'] }]
*/
_lodash["default"].forEach(markAsUsed, function (usedClass) {
classes[usedClass] = true;
});
// classNames not marked as true are unused
var unusedClasses = _fp["default"].compose(_fp["default"].keys, _fp["default"].omitBy(_fp["default"].identity) // omit truthy values
)(classes);
if (!_lodash["default"].isEmpty(unusedClasses)) {
context.report(node, "Unused classes found in ".concat(_path["default"].basename(filePath), ": ").concat(unusedClasses.join(', ')));
}
});
}
};
}
};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_fp","_interopRequireDefault","require","_lodash","_path","_core","obj","__esModule","_default","exports","meta","docs","description","recommended","schema","type","properties","camelCase","markAsUsed","create","context","_","get","map","ImportDeclaration","node","styleImportNodeData","getStyleImportNodeData","importName","styleFilePath","importNode","styleFileAbsolutePath","getFilePath","classes","classesMap","fileExists","ast","getAST","getStyleClasses","getClassesMap","set","concat","MemberExpression","objectName","object","name","propertyName","getPropertyName","className","ProgramExit","forIn","o","filePath","forEach","usedClass","unusedClasses","fp","compose","keys","omitBy","identity","isEmpty","report","path","basename","join"],"sources":["../../lib/rules/no-unused-class.js"],"sourcesContent":["/* @flow */\nimport fp from 'lodash/fp';\nimport _ from 'lodash';\nimport path from 'path';\n\nimport {\n  getStyleImportNodeData,\n  getStyleClasses,\n  getPropertyName,\n  getClassesMap,\n  getFilePath,\n  getAST,\n  fileExists,\n} from '../core';\n\nimport type { JsNode } from '../types';\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 = _.get(context, 'options[0].markAsUsed');\n    const camelCase = _.get(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        _.set(map, `${importName}.classes`, classes);\n        _.set(map, `${importName}.classesMap`, classesMap);\n\n        // save node for reporting unused styles\n        _.set(map, `${importName}.node`, importNode);\n\n        // save file path for reporting unused styles\n        _.set(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 = _.get(map, `${objectName}.classesMap.${propertyName}`);\n\n        if (className == null) {\n          return;\n        }\n\n        // mark this property has used\n        _.set(map, `${objectName}.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        _.forIn(map, (o) => {\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          _.forEach(markAsUsed, (usedClass) => {\n            classes[usedClass] = true;\n          });\n\n          // classNames not marked as true are unused\n          const unusedClasses = fp.compose(\n            fp.keys,\n            fp.omitBy(fp.identity), // omit truthy values\n          )(classes);\n\n          if (!_.isEmpty(unusedClasses)) {\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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"}