require-slots-as-functions.js 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. /**
  2. * @author Yosuke Ota
  3. * See LICENSE file in root directory for full license.
  4. */
  5. 'use strict'
  6. const utils = require('../utils')
  7. const { findVariable } = require('@eslint-community/eslint-utils')
  8. module.exports = {
  9. meta: {
  10. type: 'problem',
  11. docs: {
  12. description: 'enforce properties of `$slots` to be used as a function',
  13. categories: ['vue3-essential'],
  14. url: 'https://eslint.vuejs.org/rules/require-slots-as-functions.html'
  15. },
  16. fixable: null,
  17. schema: [],
  18. messages: {
  19. unexpected: 'Property in `$slots` should be used as function.'
  20. }
  21. },
  22. /** @param {RuleContext} context */
  23. create(context) {
  24. /**
  25. * Verify the given node
  26. * @param {MemberExpression | Identifier | ChainExpression} node The node to verify
  27. * @param {Expression} reportNode The node to report
  28. */
  29. function verify(node, reportNode) {
  30. const parent = node.parent
  31. if (
  32. parent.type === 'VariableDeclarator' &&
  33. parent.id.type === 'Identifier'
  34. ) {
  35. // const children = this.$slots.foo
  36. verifyReferences(parent.id, reportNode)
  37. return
  38. }
  39. if (
  40. parent.type === 'AssignmentExpression' &&
  41. parent.right === node &&
  42. parent.left.type === 'Identifier'
  43. ) {
  44. // children = this.$slots.foo
  45. verifyReferences(parent.left, reportNode)
  46. return
  47. }
  48. if (parent.type === 'ChainExpression') {
  49. // (this.$slots?.foo).x
  50. verify(parent, reportNode)
  51. return
  52. }
  53. if (
  54. // this.$slots.foo.xxx
  55. parent.type === 'MemberExpression' ||
  56. // var [foo] = this.$slots.foo
  57. parent.type === 'VariableDeclarator' ||
  58. // [...this.$slots.foo]
  59. parent.type === 'SpreadElement' ||
  60. // [this.$slots.foo]
  61. parent.type === 'ArrayExpression'
  62. ) {
  63. context.report({
  64. node: reportNode,
  65. messageId: 'unexpected'
  66. })
  67. }
  68. }
  69. /**
  70. * Verify the references of the given node.
  71. * @param {Identifier} node The node to verify
  72. * @param {Expression} reportNode The node to report
  73. */
  74. function verifyReferences(node, reportNode) {
  75. const variable = findVariable(context.getScope(), node)
  76. if (!variable) {
  77. return
  78. }
  79. for (const reference of variable.references) {
  80. if (!reference.isRead()) {
  81. continue
  82. }
  83. /** @type {Identifier} */
  84. const id = reference.identifier
  85. verify(id, reportNode)
  86. }
  87. }
  88. return utils.defineVueVisitor(context, {
  89. /** @param {MemberExpression} node */
  90. MemberExpression(node) {
  91. const object = utils.skipChainExpression(node.object)
  92. if (object.type !== 'MemberExpression') {
  93. return
  94. }
  95. if (utils.getStaticPropertyName(object) !== '$slots') {
  96. return
  97. }
  98. if (!utils.isThis(object.object, context)) {
  99. return
  100. }
  101. if (node.property.type === 'PrivateIdentifier') {
  102. // Unreachable
  103. return
  104. }
  105. verify(node, node.property)
  106. }
  107. })
  108. }
  109. }