123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148 |
- /**
- * @fileoverview disallow unused variable definitions of v-for directives or scope attributes.
- * @author 薛定谔的猫<hh_2013@foxmail.com>
- */
- 'use strict'
- const utils = require('../utils')
- /**
- * @typedef {VVariable['kind']} VariableKind
- */
- // ------------------------------------------------------------------------------
- // Helpers
- // ------------------------------------------------------------------------------
- /**
- * Groups variables by directive kind.
- * @param {VElement} node The element node
- * @returns { { [kind in VariableKind]?: VVariable[] } } The variables of grouped by directive kind.
- */
- function groupingVariables(node) {
- /** @type { { [kind in VariableKind]?: VVariable[] } } */
- const result = {}
- for (const variable of node.variables) {
- const vars = result[variable.kind] || (result[variable.kind] = [])
- vars.push(variable)
- }
- return result
- }
- /**
- * Checks if the given variable was defined by destructuring.
- * @param {VVariable} variable the given variable to check
- * @returns {boolean} `true` if the given variable was defined by destructuring.
- */
- function isDestructuringVar(variable) {
- const node = variable.id
- /** @type {ASTNode | null} */
- let parent = node.parent
- while (parent) {
- if (
- parent.type === 'VForExpression' ||
- parent.type === 'VSlotScopeExpression'
- ) {
- return false
- }
- if (parent.type === 'Property' || parent.type === 'ArrayPattern') {
- return true
- }
- parent = parent.parent
- }
- return false
- }
- // ------------------------------------------------------------------------------
- // Rule Definition
- // ------------------------------------------------------------------------------
- module.exports = {
- meta: {
- hasSuggestions: true,
- type: 'suggestion',
- docs: {
- description:
- 'disallow unused variable definitions of v-for directives or scope attributes',
- categories: ['vue3-essential', 'essential'],
- url: 'https://eslint.vuejs.org/rules/no-unused-vars.html'
- },
- fixable: null,
- schema: [
- {
- type: 'object',
- properties: {
- ignorePattern: {
- type: 'string'
- }
- },
- additionalProperties: false
- }
- ]
- },
- /** @param {RuleContext} context */
- create(context) {
- const option = context.options[0] || {}
- const ignorePattern = option.ignorePattern
- /** @type {RegExp | null} */
- let ignoreRegEx = null
- if (ignorePattern) {
- ignoreRegEx = new RegExp(ignorePattern, 'u')
- }
- return utils.defineTemplateBodyVisitor(context, {
- /**
- * @param {VElement} node
- */
- VElement(node) {
- const vars = groupingVariables(node)
- for (const variables of Object.values(vars)) {
- if (!variables) {
- continue
- }
- let hasAfterUsed = false
- for (let i = variables.length - 1; i >= 0; i--) {
- const variable = variables[i]
- if (variable.references.length) {
- hasAfterUsed = true
- continue
- }
- if (ignoreRegEx != null && ignoreRegEx.test(variable.id.name)) {
- continue
- }
- if (hasAfterUsed && !isDestructuringVar(variable)) {
- // If a variable after the variable is used, it will be skipped.
- continue
- }
- context.report({
- node: variable.id,
- loc: variable.id.loc,
- message: `'{{name}}' is defined but never used.`,
- data: {
- name: variable.id.name
- },
- suggest:
- ignorePattern === '^_'
- ? [
- {
- desc: `Replace the ${variable.id.name} with _${variable.id.name}`,
- fix(fixer) {
- return fixer.replaceText(
- variable.id,
- `_${variable.id.name}`
- )
- }
- }
- ]
- : []
- })
- }
- }
- }
- })
- }
- }
|