eslint-plugin-vue
Version:
Official ESLint plugin for Vue.js
187 lines (168 loc) • 5.13 kB
JavaScript
/**
* @author Mussin Benarbia
* See LICENSE file in root directory for full license.
*/
const utils = require('../utils')
/**
* @typedef {import('@typescript-eslint/types').TSESTree.TypeNode} TypeNode
* @typedef {import('@typescript-eslint/types').TSESTree.TypeElement} TypeElement
*/
/**
* @param {TypeElement} node
* @return {string | null}
*/
function getSlotsName(node) {
if (
node.type !== 'TSMethodSignature' &&
node.type !== 'TSPropertySignature'
) {
return null
}
const key = node.key
if (key.type === 'Literal') {
return typeof key.value === 'string' ? key.value : null
}
if (key.type === 'Identifier') {
return key.name
}
return null
}
/**
* @param {VElement} node
* @return {VAttribute | VDirective | undefined}
*/
function getSlotNameNode(node) {
return node.startTag.attributes.find(
(node) =>
(!node.directive && node.key.name === 'name') ||
(node.directive &&
node.key.name.name === 'bind' &&
node.key.argument?.type === 'VIdentifier' &&
node.key.argument?.name === 'name')
)
}
module.exports = {
meta: {
type: 'problem',
docs: {
description: 'require slots to be explicitly defined',
categories: undefined,
url: 'https://eslint.vuejs.org/rules/require-explicit-slots.html'
},
fixable: null,
schema: [],
messages: {
requireExplicitSlots: 'Slots must be explicitly defined.',
alreadyDefinedSlot: 'Slot {{slotName}} is already defined.'
}
},
/** @param {RuleContext} context */
create(context) {
const sourceCode = context.getSourceCode()
const documentFragment =
sourceCode.parserServices.getDocumentFragment &&
sourceCode.parserServices.getDocumentFragment()
if (!documentFragment) {
return {}
}
const scripts = documentFragment.children.filter(
/** @returns {element is VElement} */
(element) => utils.isVElement(element) && element.name === 'script'
)
if (scripts.every((script) => !utils.hasAttribute(script, 'lang', 'ts'))) {
return {}
}
const slotsDefined = new Set()
/**
* @param {VElement} node
* @param {string | undefined} slotName
*/
function reportMissingSlot(node, slotName) {
if (!slotsDefined.has(slotName)) {
context.report({
node,
messageId: 'requireExplicitSlots'
})
}
}
return utils.compositingVisitors(
utils.defineScriptSetupVisitor(context, {
onDefineSlotsEnter(_node, slots) {
for (const slot of slots) {
if (!slot.slotName) {
continue
}
if (slotsDefined.has(slot.slotName)) {
context.report({
node: slot.node,
messageId: 'alreadyDefinedSlot',
data: {
slotName: slot.slotName
}
})
} else {
slotsDefined.add(slot.slotName)
}
}
}
}),
utils.executeOnVue(context, (obj) => {
const slotsProperty = utils.findProperty(obj, 'slots')
if (!slotsProperty) return
const slotsTypeHelper =
slotsProperty.value.type === 'TSAsExpression' &&
slotsProperty.value.typeAnnotation?.typeName.name === 'SlotsType' &&
slotsProperty.value.typeAnnotation
if (!slotsTypeHelper) return
const typeArguments =
'typeArguments' in slotsTypeHelper
? slotsTypeHelper.typeArguments
: slotsTypeHelper.typeParameters
const param = /** @type {TypeNode|undefined} */ (
typeArguments?.params[0]
)
if (!param) return
if (param.type === 'TSTypeLiteral') {
for (const memberNode of param.members) {
const slotName = getSlotsName(memberNode)
if (!slotName) continue
if (slotsDefined.has(slotName)) {
context.report({
node: memberNode,
messageId: 'alreadyDefinedSlot',
data: {
slotName
}
})
} else {
slotsDefined.add(slotName)
}
}
}
}),
utils.defineTemplateBodyVisitor(context, {
/** @param {VElement} node */
"VElement[name='slot']"(node) {
const nameNode = getSlotNameNode(node)
// if no slot name is declared, default to 'default'
if (!nameNode) {
reportMissingSlot(node, 'default')
return
}
if (nameNode.directive) {
const expression = nameNode.value?.expression
// ignore attribute binding except string literal
if (!expression || !utils.isStringLiteral(expression)) {
return
}
const name = utils.getStringLiteralValue(expression) || undefined
reportMissingSlot(node, name)
} else {
reportMissingSlot(node, nameNode.value?.value)
}
}
})
)
}
}