component-name-in-template-casing.js 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. /**
  2. * @author Yosuke Ota
  3. * issue https://github.com/vuejs/eslint-plugin-vue/issues/250
  4. */
  5. 'use strict'
  6. const utils = require('../utils')
  7. const casing = require('../utils/casing')
  8. const { toRegExp } = require('../utils/regexp')
  9. const allowedCaseOptions = ['PascalCase', 'kebab-case']
  10. const defaultCase = 'PascalCase'
  11. /**
  12. * Checks whether the given variable is the type-only import object.
  13. * @param {Variable} variable
  14. * @returns {boolean} `true` if the given variable is the type-only import.
  15. */
  16. function isTypeOnlyImport(variable) {
  17. if (variable.defs.length === 0) return false
  18. return variable.defs.every((def) => {
  19. if (def.type !== 'ImportBinding') {
  20. return false
  21. }
  22. if (def.parent.importKind === 'type') {
  23. // check for `import type Foo from './xxx'`
  24. return true
  25. }
  26. if (def.node.type === 'ImportSpecifier' && def.node.importKind === 'type') {
  27. // check for `import { type Foo } from './xxx'`
  28. return true
  29. }
  30. return false
  31. })
  32. }
  33. module.exports = {
  34. meta: {
  35. type: 'suggestion',
  36. docs: {
  37. description:
  38. 'enforce specific casing for the component naming style in template',
  39. categories: undefined,
  40. url: 'https://eslint.vuejs.org/rules/component-name-in-template-casing.html'
  41. },
  42. fixable: 'code',
  43. schema: [
  44. {
  45. enum: allowedCaseOptions
  46. },
  47. {
  48. type: 'object',
  49. properties: {
  50. globals: {
  51. type: 'array',
  52. items: { type: 'string' },
  53. uniqueItems: true
  54. },
  55. ignores: {
  56. type: 'array',
  57. items: { type: 'string' },
  58. uniqueItems: true,
  59. additionalItems: false
  60. },
  61. registeredComponentsOnly: {
  62. type: 'boolean'
  63. }
  64. },
  65. additionalProperties: false
  66. }
  67. ],
  68. messages: {
  69. incorrectCase: 'Component name "{{name}}" is not {{caseType}}.'
  70. }
  71. },
  72. /** @param {RuleContext} context */
  73. create(context) {
  74. const caseOption = context.options[0]
  75. const options = context.options[1] || {}
  76. const caseType = allowedCaseOptions.includes(caseOption)
  77. ? caseOption
  78. : defaultCase
  79. /** @type {RegExp[]} */
  80. const ignores = (options.ignores || []).map(toRegExp)
  81. /** @type {string[]} */
  82. const globals = (options.globals || []).map(casing.pascalCase)
  83. const registeredComponentsOnly = options.registeredComponentsOnly !== false
  84. const tokens =
  85. context.parserServices.getTemplateBodyTokenStore &&
  86. context.parserServices.getTemplateBodyTokenStore()
  87. /** @type { Set<string> } */
  88. const registeredComponents = new Set(globals)
  89. if (utils.isScriptSetup(context)) {
  90. // For <script setup>
  91. const globalScope = context.getSourceCode().scopeManager.globalScope
  92. if (globalScope) {
  93. // Only check find the import module
  94. const moduleScope = globalScope.childScopes.find(
  95. (scope) => scope.type === 'module'
  96. )
  97. for (const variable of (moduleScope && moduleScope.variables) || []) {
  98. if (!isTypeOnlyImport(variable)) {
  99. registeredComponents.add(variable.name)
  100. }
  101. }
  102. }
  103. }
  104. /**
  105. * Checks whether the given node is the verification target node.
  106. * @param {VElement} node element node
  107. * @returns {boolean} `true` if the given node is the verification target node.
  108. */
  109. function isVerifyTarget(node) {
  110. if (ignores.some((re) => re.test(node.rawName))) {
  111. // ignore
  112. return false
  113. }
  114. if (
  115. (!utils.isHtmlElementNode(node) && !utils.isSvgElementNode(node)) ||
  116. utils.isHtmlWellKnownElementName(node.rawName) ||
  117. utils.isSvgWellKnownElementName(node.rawName)
  118. ) {
  119. return false
  120. }
  121. if (!registeredComponentsOnly) {
  122. // If the user specifies registeredComponentsOnly as false, it checks all component tags.
  123. return true
  124. }
  125. // We only verify the registered components.
  126. return registeredComponents.has(casing.pascalCase(node.rawName))
  127. }
  128. let hasInvalidEOF = false
  129. return utils.defineTemplateBodyVisitor(
  130. context,
  131. {
  132. VElement(node) {
  133. if (hasInvalidEOF) {
  134. return
  135. }
  136. if (!isVerifyTarget(node)) {
  137. return
  138. }
  139. const name = node.rawName
  140. if (!casing.getChecker(caseType)(name)) {
  141. const startTag = node.startTag
  142. const open = tokens.getFirstToken(startTag)
  143. const casingName = casing.getExactConverter(caseType)(name)
  144. context.report({
  145. node: open,
  146. loc: open.loc,
  147. messageId: 'incorrectCase',
  148. data: {
  149. name,
  150. caseType
  151. },
  152. *fix(fixer) {
  153. yield fixer.replaceText(open, `<${casingName}`)
  154. const endTag = node.endTag
  155. if (endTag) {
  156. const endTagOpen = tokens.getFirstToken(endTag)
  157. yield fixer.replaceText(endTagOpen, `</${casingName}`)
  158. }
  159. }
  160. })
  161. }
  162. }
  163. },
  164. {
  165. Program(node) {
  166. hasInvalidEOF = utils.hasInvalidEOF(node)
  167. },
  168. ...(registeredComponentsOnly
  169. ? utils.executeOnVue(context, (obj) => {
  170. for (const n of utils.getRegisteredComponents(obj)) {
  171. registeredComponents.add(n.name)
  172. }
  173. })
  174. : {})
  175. }
  176. )
  177. }
  178. }