UNPKG

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
"use strict"; 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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"}