| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143 | /** * @fileoverview Report used components * @author Michał Sajnóg */'use strict'// ------------------------------------------------------------------------------// Requirements// ------------------------------------------------------------------------------const utils = require('../utils')const casing = require('../utils/casing')// ------------------------------------------------------------------------------// Rule Definition// ------------------------------------------------------------------------------module.exports = {  meta: {    type: 'suggestion',    docs: {      description:        'disallow registering components that are not used inside templates',      categories: ['vue3-essential', 'essential'],      url: 'https://eslint.vuejs.org/rules/no-unused-components.html'    },    fixable: null,    schema: [      {        type: 'object',        properties: {          ignoreWhenBindingPresent: {            type: 'boolean'          }        },        additionalProperties: false      }    ]  },  /** @param {RuleContext} context */  create(context) {    const options = context.options[0] || {}    const ignoreWhenBindingPresent =      options.ignoreWhenBindingPresent !== undefined        ? options.ignoreWhenBindingPresent        : true    const usedComponents = new Set()    /** @type { { node: Property, name: string }[] } */    let registeredComponents = []    let ignoreReporting = false    /** @type {Position} */    let templateLocation    return utils.defineTemplateBodyVisitor(      context,      {        /** @param {VElement} node */        VElement(node) {          if (            (!utils.isHtmlElementNode(node) && !utils.isSvgElementNode(node)) ||            utils.isHtmlWellKnownElementName(node.rawName) ||            utils.isSvgWellKnownElementName(node.rawName)          ) {            return          }          usedComponents.add(node.rawName)        },        /** @param {VDirective} node */        "VAttribute[directive=true][key.name.name='bind'][key.argument.name='is'], VAttribute[directive=true][key.name.name='is']"(          node        ) {          if (            !node.value || // `<component :is>`            node.value.type !== 'VExpressionContainer' ||            !node.value.expression // `<component :is="">`          )            return          if (node.value.expression.type === 'Literal') {            usedComponents.add(node.value.expression.value)          } else if (ignoreWhenBindingPresent) {            ignoreReporting = true          }        },        /** @param {VAttribute} node */        "VAttribute[directive=false][key.name='is']"(node) {          if (!node.value) {            return          }          usedComponents.add(node.value.value)        },        /** @param {VElement} node */        "VElement[name='template']"(node) {          templateLocation = templateLocation || node.loc.start        },        /** @param {VElement} node */        "VElement[name='template']:exit"(node) {          if (            node.loc.start !== templateLocation ||            ignoreReporting ||            utils.hasAttribute(node, 'src')          )            return          registeredComponents            .filter(({ name }) => {              // If the component name is PascalCase or camelCase              // it can be used in various of ways inside template,              // like "theComponent", "The-component" etc.              // but except snake_case              if (casing.isPascalCase(name) || casing.isCamelCase(name)) {                return ![...usedComponents].some((n) => {                  return (                    n.indexOf('_') === -1 &&                    (name === casing.pascalCase(n) ||                      casing.camelCase(n) === name)                  )                })              } else {                // In any other case the used component name must exactly match                // the registered name                return !usedComponents.has(name)              }            })            .forEach(({ node, name }) =>              context.report({                node,                message:                  'The "{{name}}" component has been registered but not used.',                data: {                  name                }              })            )        }      },      utils.executeOnVue(context, (obj) => {        registeredComponents = utils.getRegisteredComponents(obj)      })    )  }}
 |