| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511 | /** * @author Yosuke Ota * See LICENSE file in root directory for full license. */'use strict'/** * @typedef {import('../utils').ComponentArrayEmit} ComponentArrayEmit * @typedef {import('../utils').ComponentObjectEmit} ComponentObjectEmit * @typedef {import('../utils').ComponentArrayProp} ComponentArrayProp * @typedef {import('../utils').ComponentObjectProp} ComponentObjectProp * @typedef {import('../utils').VueObjectData} VueObjectData */// ------------------------------------------------------------------------------// Requirements// ------------------------------------------------------------------------------const { findVariable } = require('eslint-utils')const utils = require('../utils')const { capitalize } = require('../utils/casing')// ------------------------------------------------------------------------------// Helpers// ------------------------------------------------------------------------------const FIX_EMITS_AFTER_OPTIONS = [  'setup',  'data',  'computed',  'watch',  'methods',  'template',  'render',  'renderError',  // lifecycle hooks  'beforeCreate',  'created',  'beforeMount',  'mounted',  'beforeUpdate',  'updated',  'activated',  'deactivated',  'beforeUnmount',  'unmounted',  'beforeDestroy',  'destroyed',  'renderTracked',  'renderTriggered',  'errorCaptured']/** * Check whether the given token is a left brace. * @param {Token} token The token to check. * @returns {boolean} `true` if the token is a left brace. */function isLeftBrace(token) {  return token != null && token.type === 'Punctuator' && token.value === '{'}/** * Check whether the given token is a right brace. * @param {Token} token The token to check. * @returns {boolean} `true` if the token is a right brace. */function isRightBrace(token) {  return token != null && token.type === 'Punctuator' && token.value === '}'}/** * Check whether the given token is a left bracket. * @param {Token} token The token to check. * @returns {boolean} `true` if the token is a left bracket. */function isLeftBracket(token) {  return token != null && token.type === 'Punctuator' && token.value === '['}// ------------------------------------------------------------------------------// Rule Definition// ------------------------------------------------------------------------------module.exports = {  meta: {    type: 'suggestion',    docs: {      description: 'require `emits` option with name triggered by `$emit()`',      categories: ['vue3-strongly-recommended'],      url: 'https://eslint.vuejs.org/rules/require-explicit-emits.html'    },    fixable: null,    schema: [      {        type: 'object',        properties: {          allowProps: {            type: 'boolean'          }        },        additionalProperties: false      }    ],    messages: {      missing:        'The "{{name}}" event has been triggered but not declared on `emits` option.',      addOneOption: 'Add the "{{name}}" to `emits` option.',      addArrayEmitsOption:        'Add the `emits` option with array syntax and define "{{name}}" event.',      addObjectEmitsOption:        'Add the `emits` option with object syntax and define "{{name}}" event.'    }  },  /** @param {RuleContext} context */  create(context) {    const options = context.options[0] || {}    const allowProps = !!options.allowProps    /** @type {Map<ObjectExpression, { contextReferenceIds: Set<Identifier>, emitReferenceIds: Set<Identifier> }>} */    const setupContexts = new Map()    /** @type {Map<ObjectExpression, (ComponentArrayEmit | ComponentObjectEmit)[]>} */    const vueEmitsDeclarations = new Map()    /** @type {Map<ObjectExpression, (ComponentArrayProp | ComponentObjectProp)[]>} */    const vuePropsDeclarations = new Map()    /**     * @typedef {object} VueTemplateObjectData     * @property {'export' | 'mark' | 'definition'} type     * @property {ObjectExpression} object     * @property {(ComponentArrayEmit | ComponentObjectEmit)[]} emits     * @property {(ComponentArrayProp | ComponentObjectProp)[]} props     */    /** @type {VueTemplateObjectData | null} */    let vueTemplateObjectData = null    /**     * @param {(ComponentArrayEmit | ComponentObjectEmit)[]} emits     * @param {(ComponentArrayProp | ComponentObjectProp)[]} props     * @param {Literal} nameLiteralNode     * @param {ObjectExpression} vueObjectNode     */    function verifyEmit(emits, props, nameLiteralNode, vueObjectNode) {      const name = `${nameLiteralNode.value}`      if (emits.some((e) => e.emitName === name)) {        return      }      if (allowProps) {        const key = `on${capitalize(name)}`        if (props.some((e) => e.propName === key)) {          return        }      }      context.report({        node: nameLiteralNode,        messageId: 'missing',        data: {          name        },        suggest: buildSuggest(vueObjectNode, emits, nameLiteralNode, context)      })    }    return utils.defineTemplateBodyVisitor(      context,      {        /** @param { CallExpression & { argument: [Literal, ...Expression] } } node */        'CallExpression[arguments.0.type=Literal]'(node) {          const callee = utils.skipChainExpression(node.callee)          const nameLiteralNode = /** @type {Literal} */ (node.arguments[0])          if (!nameLiteralNode || typeof nameLiteralNode.value !== 'string') {            // cannot check            return          }          if (!vueTemplateObjectData) {            return          }          if (callee.type === 'Identifier' && callee.name === '$emit') {            verifyEmit(              vueTemplateObjectData.emits,              vueTemplateObjectData.props,              nameLiteralNode,              vueTemplateObjectData.object            )          }        }      },      utils.defineVueVisitor(context, {        onVueObjectEnter(node) {          vueEmitsDeclarations.set(node, utils.getComponentEmits(node))          if (allowProps) {            vuePropsDeclarations.set(node, utils.getComponentProps(node))          }        },        onSetupFunctionEnter(node, { node: vueNode }) {          const contextParam = node.params[1]          if (!contextParam) {            // no arguments            return          }          if (contextParam.type === 'RestElement') {            // cannot check            return          }          if (contextParam.type === 'ArrayPattern') {            // cannot check            return          }          /** @type {Set<Identifier>} */          const contextReferenceIds = new Set()          /** @type {Set<Identifier>} */          const emitReferenceIds = new Set()          if (contextParam.type === 'ObjectPattern') {            const emitProperty = utils.findAssignmentProperty(              contextParam,              'emit'            )            if (!emitProperty) {              return            }            const emitParam = emitProperty.value            // `setup(props, {emit})`            const variable =              emitParam.type === 'Identifier'                ? findVariable(context.getScope(), emitParam)                : null            if (!variable) {              return            }            for (const reference of variable.references) {              if (!reference.isRead()) {                continue              }              emitReferenceIds.add(reference.identifier)            }          } else if (contextParam.type === 'Identifier') {            // `setup(props, context)`            const variable = findVariable(context.getScope(), contextParam)            if (!variable) {              return            }            for (const reference of variable.references) {              if (!reference.isRead()) {                continue              }              contextReferenceIds.add(reference.identifier)            }          }          setupContexts.set(vueNode, {            contextReferenceIds,            emitReferenceIds          })        },        /**         * @param {CallExpression & { arguments: [Literal, ...Expression] }} node         * @param {VueObjectData} data         */        'CallExpression[arguments.0.type=Literal]'(node, { node: vueNode }) {          const callee = utils.skipChainExpression(node.callee)          const nameLiteralNode = node.arguments[0]          if (!nameLiteralNode || typeof nameLiteralNode.value !== 'string') {            // cannot check            return          }          const emitsDeclarations = vueEmitsDeclarations.get(vueNode)          if (!emitsDeclarations) {            return          }          let emit          if (callee.type === 'MemberExpression') {            const name = utils.getStaticPropertyName(callee)            if (name === 'emit' || name === '$emit') {              emit = { name, member: callee }            }          }          // verify setup context          const setupContext = setupContexts.get(vueNode)          if (setupContext) {            const { contextReferenceIds, emitReferenceIds } = setupContext            if (callee.type === 'Identifier' && emitReferenceIds.has(callee)) {              // verify setup(props,{emit}) {emit()}              verifyEmit(                emitsDeclarations,                vuePropsDeclarations.get(vueNode) || [],                nameLiteralNode,                vueNode              )            } else if (emit && emit.name === 'emit') {              const memObject = utils.skipChainExpression(emit.member.object)              if (                memObject.type === 'Identifier' &&                contextReferenceIds.has(memObject)              ) {                // verify setup(props,context) {context.emit()}                verifyEmit(                  emitsDeclarations,                  vuePropsDeclarations.get(vueNode) || [],                  nameLiteralNode,                  vueNode                )              }            }          }          // verify $emit          if (emit && emit.name === '$emit') {            const memObject = utils.skipChainExpression(emit.member.object)            if (utils.isThis(memObject, context)) {              // verify this.$emit()              verifyEmit(                emitsDeclarations,                vuePropsDeclarations.get(vueNode) || [],                nameLiteralNode,                vueNode              )            }          }        },        onVueObjectExit(node, { type }) {          const emits = vueEmitsDeclarations.get(node)          if (            !vueTemplateObjectData ||            vueTemplateObjectData.type !== 'export'          ) {            if (              emits &&              (type === 'mark' || type === 'export' || type === 'definition')            ) {              vueTemplateObjectData = {                type,                object: node,                emits,                props: vuePropsDeclarations.get(node) || []              }            }          }          setupContexts.delete(node)          vueEmitsDeclarations.delete(node)          vuePropsDeclarations.delete(node)        }      })    )  }}/** * @param {ObjectExpression} object * @param {(ComponentArrayEmit | ComponentObjectEmit)[]} emits * @param {Literal} nameNode * @param {RuleContext} context * @returns {Rule.SuggestionReportDescriptor[]} */function buildSuggest(object, emits, nameNode, context) {  const certainEmits = emits.filter((e) => e.key)  if (certainEmits.length) {    const last = certainEmits[certainEmits.length - 1]    return [      {        messageId: 'addOneOption',        data: { name: `${nameNode.value}` },        fix(fixer) {          if (last.value === null) {            // Array            return fixer.insertTextAfter(last.node, `, '${nameNode.value}'`)          } else {            // Object            return fixer.insertTextAfter(              last.node,              `, '${nameNode.value}': null`            )          }        }      }    ]  }  const propertyNodes = object.properties.filter(utils.isProperty)  const emitsOption = propertyNodes.find(    (p) => utils.getStaticPropertyName(p) === 'emits'  )  if (emitsOption) {    const sourceCode = context.getSourceCode()    const emitsOptionValue = emitsOption.value    if (emitsOptionValue.type === 'ArrayExpression') {      const leftBracket = /** @type {Token} */ (sourceCode.getFirstToken(        emitsOptionValue,        isLeftBracket      ))      return [        {          messageId: 'addOneOption',          data: { name: `${nameNode.value}` },          fix(fixer) {            return fixer.insertTextAfter(              leftBracket,              `'${nameNode.value}'${                emitsOptionValue.elements.length ? ',' : ''              }`            )          }        }      ]    } else if (emitsOptionValue.type === 'ObjectExpression') {      const leftBrace = /** @type {Token} */ (sourceCode.getFirstToken(        emitsOptionValue,        isLeftBrace      ))      return [        {          messageId: 'addOneOption',          data: { name: `${nameNode.value}` },          fix(fixer) {            return fixer.insertTextAfter(              leftBrace,              `'${nameNode.value}': null${                emitsOptionValue.properties.length ? ',' : ''              }`            )          }        }      ]    }    return []  }  const sourceCode = context.getSourceCode()  const afterOptionNode = propertyNodes.find((p) =>    FIX_EMITS_AFTER_OPTIONS.includes(utils.getStaticPropertyName(p) || '')  )  return [    {      messageId: 'addArrayEmitsOption',      data: { name: `${nameNode.value}` },      fix(fixer) {        if (afterOptionNode) {          return fixer.insertTextAfter(            sourceCode.getTokenBefore(afterOptionNode),            `\nemits: ['${nameNode.value}'],`          )        } else if (object.properties.length) {          const before =            propertyNodes[propertyNodes.length - 1] ||            object.properties[object.properties.length - 1]          return fixer.insertTextAfter(            before,            `,\nemits: ['${nameNode.value}']`          )        } else {          const objectLeftBrace = /** @type {Token} */ (sourceCode.getFirstToken(            object,            isLeftBrace          ))          const objectRightBrace = /** @type {Token} */ (sourceCode.getLastToken(            object,            isRightBrace          ))          return fixer.insertTextAfter(            objectLeftBrace,            `\nemits: ['${nameNode.value}']${              objectLeftBrace.loc.end.line < objectRightBrace.loc.start.line                ? ''                : '\n'            }`          )        }      }    },    {      messageId: 'addObjectEmitsOption',      data: { name: `${nameNode.value}` },      fix(fixer) {        if (afterOptionNode) {          return fixer.insertTextAfter(            sourceCode.getTokenBefore(afterOptionNode),            `\nemits: {'${nameNode.value}': null},`          )        } else if (object.properties.length) {          const before =            propertyNodes[propertyNodes.length - 1] ||            object.properties[object.properties.length - 1]          return fixer.insertTextAfter(            before,            `,\nemits: {'${nameNode.value}': null}`          )        } else {          const objectLeftBrace = /** @type {Token} */ (sourceCode.getFirstToken(            object,            isLeftBrace          ))          const objectRightBrace = /** @type {Token} */ (sourceCode.getLastToken(            object,            isRightBrace          ))          return fixer.insertTextAfter(            objectLeftBrace,            `\nemits: {'${nameNode.value}': null}${              objectLeftBrace.loc.end.line < objectRightBrace.loc.start.line                ? ''                : '\n'            }`          )        }      }    }  ]}
 |