slot-scope-attribute.js 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. /**
  2. * @author Yosuke Ota
  3. * See LICENSE file in root directory for full license.
  4. */
  5. 'use strict'
  6. const canConvertToVSlotForElement = require('./utils/can-convert-to-v-slot')
  7. module.exports = {
  8. deprecated: '2.6.0',
  9. supported: '>=2.5.0 <3.0.0',
  10. /**
  11. * @param {RuleContext} context
  12. * @param {object} option
  13. * @param {boolean} [option.fixToUpgrade]
  14. * @returns {TemplateListener}
  15. */
  16. createTemplateBodyVisitor(context, { fixToUpgrade } = {}) {
  17. const sourceCode = context.getSourceCode()
  18. const tokenStore =
  19. context.parserServices.getTemplateBodyTokenStore &&
  20. context.parserServices.getTemplateBodyTokenStore()
  21. /**
  22. * Checks whether the given node can convert to the `v-slot`.
  23. * @param {VStartTag} startTag node of `<element v-slot ... >`
  24. * @returns {boolean} `true` if the given node can convert to the `v-slot`
  25. */
  26. function canConvertToVSlot(startTag) {
  27. if (
  28. !canConvertToVSlotForElement(startTag.parent, sourceCode, tokenStore)
  29. ) {
  30. return false
  31. }
  32. const slotAttr = startTag.attributes.find(
  33. (attr) => attr.directive === false && attr.key.name === 'slot'
  34. )
  35. if (slotAttr) {
  36. // if the element have `slot` it can not be converted.
  37. // Conversion of `slot` is done with `vue/no-deprecated-slot-attribute`.
  38. return false
  39. }
  40. const vBindSlotAttr = startTag.attributes.find(
  41. (attr) =>
  42. attr.directive === true &&
  43. attr.key.name.name === 'bind' &&
  44. attr.key.argument &&
  45. attr.key.argument.type === 'VIdentifier' &&
  46. attr.key.argument.name === 'slot'
  47. )
  48. if (vBindSlotAttr) {
  49. // if the element have `v-bind:slot` it can not be converted.
  50. // Conversion of `v-bind:slot` is done with `vue/no-deprecated-slot-attribute`.
  51. return false
  52. }
  53. return true
  54. }
  55. /**
  56. * Convert to `v-slot`.
  57. * @param {RuleFixer} fixer fixer
  58. * @param {VDirective} scopeAttr node of `slot-scope`
  59. * @returns {Fix} fix data
  60. */
  61. function fixSlotScopeToVSlot(fixer, scopeAttr) {
  62. const scopeValue =
  63. scopeAttr && scopeAttr.value
  64. ? `=${sourceCode.getText(scopeAttr.value)}`
  65. : ''
  66. const replaceText = `v-slot${scopeValue}`
  67. return fixer.replaceText(scopeAttr, replaceText)
  68. }
  69. /**
  70. * Reports `slot-scope` node
  71. * @param {VDirective} scopeAttr node of `slot-scope`
  72. * @returns {void}
  73. */
  74. function reportSlotScope(scopeAttr) {
  75. context.report({
  76. node: scopeAttr.key,
  77. messageId: 'forbiddenSlotScopeAttribute',
  78. fix(fixer) {
  79. if (!fixToUpgrade) {
  80. return null
  81. }
  82. // fix to use `v-slot`
  83. const startTag = scopeAttr.parent
  84. if (!canConvertToVSlot(startTag)) {
  85. return null
  86. }
  87. return fixSlotScopeToVSlot(fixer, scopeAttr)
  88. }
  89. })
  90. }
  91. return {
  92. "VAttribute[directive=true][key.name.name='slot-scope']": reportSlotScope
  93. }
  94. }
  95. }