/** * @author Marton Csordas * See LICENSE file in root directory for full license. */ 'use strict' const path = require('path') const casing = require('../utils/casing') const utils = require('../utils') const RESERVED_NAMES_IN_VUE3 = new Set( require('../utils/vue3-builtin-components') ) module.exports = { meta: { type: 'suggestion', docs: { description: 'require component names to be always multi-word', categories: ['vue3-essential', 'vue2-essential'], url: 'https://eslint.vuejs.org/rules/multi-word-component-names.html' }, schema: [ { type: 'object', properties: { ignores: { type: 'array', items: { type: 'string' }, uniqueItems: true, additionalItems: false } }, additionalProperties: false } ], messages: { unexpected: 'Component name "{{value}}" should always be multi-word.' } }, /** @param {RuleContext} context */ create(context) { /** @type {Set} */ const ignores = new Set() ignores.add('App') ignores.add('app') for (const ignore of (context.options[0] && context.options[0].ignores) || []) { ignores.add(ignore) if (casing.isPascalCase(ignore)) { // PascalCase ignores.add(casing.kebabCase(ignore)) } } let hasVue = utils.isScriptSetup(context) let hasName = false /** * Returns true if the given component name is valid, otherwise false. * @param {string} name * */ function isValidComponentName(name) { if (ignores.has(name) || RESERVED_NAMES_IN_VUE3.has(name)) { return true } const elements = casing.kebabCase(name).split('-') return elements.length > 1 } /** * @param {Expression | SpreadElement} nameNode */ function validateName(nameNode) { if (nameNode.type !== 'Literal') return const componentName = `${nameNode.value}` if (!isValidComponentName(componentName)) { context.report({ node: nameNode, messageId: 'unexpected', data: { value: componentName } }) } } return utils.compositingVisitors( utils.executeOnCallVueComponent(context, (node) => { hasVue = true if (node.arguments.length !== 2) return hasName = true validateName(node.arguments[0]) }), utils.executeOnVue(context, (obj) => { hasVue = true const node = utils.findProperty(obj, 'name') if (!node) return hasName = true validateName(node.value) }), utils.defineScriptSetupVisitor(context, { onDefineOptionsEnter(node) { if (node.arguments.length === 0) return const define = node.arguments[0] if (define.type !== 'ObjectExpression') return const nameNode = utils.findProperty(define, 'name') if (!nameNode) return hasName = true validateName(nameNode.value) } }), { /** @param {Program} node */ 'Program:exit'(node) { if (hasName) return if (!hasVue && node.body.length > 0) return const fileName = context.getFilename() const componentName = path.basename(fileName, path.extname(fileName)) if ( utils.isVueFile(fileName) && !isValidComponentName(componentName) ) { context.report({ messageId: 'unexpected', data: { value: componentName }, loc: { line: 1, column: 0 } }) } } } ) } }