Add components
This commit is contained in:
186
slider/node_modules/eslint-plugin-vue/lib/rules/require-explicit-slots.js
generated
vendored
Normal file
186
slider/node_modules/eslint-plugin-vue/lib/rules/require-explicit-slots.js
generated
vendored
Normal file
@@ -0,0 +1,186 @@
|
||||
/**
|
||||
* @author Mussin Benarbia
|
||||
* See LICENSE file in root directory for full license.
|
||||
*/
|
||||
'use strict'
|
||||
|
||||
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)
|
||||
}
|
||||
}
|
||||
})
|
||||
)
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user