diff --git a/docs/rules/README.md b/docs/rules/README.md index f45d9c68a..8ea5ef8ba 100644 --- a/docs/rules/README.md +++ b/docs/rules/README.md @@ -153,6 +153,7 @@ For example: | [vue/keyword-spacing](./keyword-spacing.md) | enforce consistent spacing before and after keywords | :wrench: | | [vue/match-component-file-name](./match-component-file-name.md) | require component name property to match its file name | | | [vue/no-boolean-default](./no-boolean-default.md) | disallow boolean defaults | :wrench: | +| [vue/no-deprecated-slot-attribute](./no-deprecated-slot-attribute.md) | disallow deprecated `slot` attribute (in Vue.js 2.6.0+) | :wrench: | | [vue/no-deprecated-scope-attribute](./no-deprecated-scope-attribute.md) | disallow deprecated `scope` attribute (in Vue.js 2.5.0+) | :wrench: | | [vue/no-empty-pattern](./no-empty-pattern.md) | disallow empty destructuring patterns | | | [vue/no-reserved-component-names](./no-reserved-component-names.md) | disallow the use of reserved names in component definitions | | diff --git a/docs/rules/no-deprecated-slot-attribute.md b/docs/rules/no-deprecated-slot-attribute.md new file mode 100644 index 000000000..9da6225a7 --- /dev/null +++ b/docs/rules/no-deprecated-slot-attribute.md @@ -0,0 +1,44 @@ +--- +pageClass: rule-details +sidebarDepth: 0 +title: vue/no-deprecated-slot-attribute +description: disallow deprecated `slot` attribute (in Vue.js 2.6.0+) +--- +# vue/no-deprecated-slot-attribute +> disallow deprecated `slot` attribute (in Vue.js 2.6.0+) + +- :wrench: The `--fix` option on the [command line](https://eslint.org/docs/user-guide/command-line-interface#fixing-problems) can automatically fix some of the problems reported by this rule. + +## :book: Rule Details + +This rule reports deprecated `slot` attribute in Vue.js v2.6.0+. + + + +```vue + +``` + + + +## :books: Further reading + +- [API - slot](https://vuejs.org/v2/api/#slot-deprecated) + +## :mag: Implementation + +- [Rule source](https://github.com/vuejs/eslint-plugin-vue/blob/master/lib/rules/no-deprecated-slot-attribute.js) +- [Test source](https://github.com/vuejs/eslint-plugin-vue/blob/master/tests/lib/rules/no-deprecated-slot-attribute.js) diff --git a/lib/index.js b/lib/index.js index 9b6c34693..32ea7296e 100644 --- a/lib/index.js +++ b/lib/index.js @@ -37,6 +37,7 @@ module.exports = { 'no-async-in-computed-properties': require('./rules/no-async-in-computed-properties'), 'no-boolean-default': require('./rules/no-boolean-default'), 'no-confusing-v-for-v-if': require('./rules/no-confusing-v-for-v-if'), + 'no-deprecated-slot-attribute': require('./rules/no-deprecated-slot-attribute'), 'no-deprecated-scope-attribute': require('./rules/no-deprecated-scope-attribute'), 'no-dupe-keys': require('./rules/no-dupe-keys'), 'no-duplicate-attributes': require('./rules/no-duplicate-attributes'), diff --git a/lib/rules/no-deprecated-slot-attribute.js b/lib/rules/no-deprecated-slot-attribute.js new file mode 100644 index 000000000..cf3e7c6ba --- /dev/null +++ b/lib/rules/no-deprecated-slot-attribute.js @@ -0,0 +1,28 @@ +/** + * @author Yosuke Ota + * See LICENSE file in root directory for full license. + */ +'use strict' + +const utils = require('../utils') +const slotAttribute = require('./syntaxes/slot-attribute') + +module.exports = { + meta: { + type: 'suggestion', + docs: { + description: 'disallow deprecated `slot` attribute (in Vue.js 2.6.0+)', + category: undefined, + url: 'https://eslint.vuejs.org/rules/no-deprecated-slot-attribute.html' + }, + fixable: 'code', + schema: [], + messages: { + forbiddenSlotAttribute: '`slot` attributes are deprecated.' + } + }, + create (context) { + const templateBodyVisitor = slotAttribute.createTemplateBodyVisitor(context) + return utils.defineTemplateBodyVisitor(context, templateBodyVisitor) + } +} diff --git a/lib/rules/syntaxes/slot-attribute.js b/lib/rules/syntaxes/slot-attribute.js new file mode 100644 index 000000000..db91e24fb --- /dev/null +++ b/lib/rules/syntaxes/slot-attribute.js @@ -0,0 +1,128 @@ +/** + * @author Yosuke Ota + * See LICENSE file in root directory for full license. + */ +'use strict' +module.exports = { + deprecated: '2.6.0', + createTemplateBodyVisitor (context) { + const sourceCode = context.getSourceCode() + + /** + * Checks whether the given node can convert to the `v-slot`. + * @param {VAttribute} slotAttr node of `slot` + * @returns {boolean} `true` if the given node can convert to the `v-slot` + */ + function canConvertFromSlotToVSlot (slotAttr) { + if (slotAttr.parent.parent.name !== 'template') { + return false + } + if (!slotAttr.value) { + return true + } + const slotName = slotAttr.value.value + // If non-Latin characters are included it can not be converted. + return !/[^a-z]/i.test(slotName) + } + + /** + * Checks whether the given node can convert to the `v-slot`. + * @param {VAttribute} slotAttr node of `v-bind:slot` + * @returns {boolean} `true` if the given node can convert to the `v-slot` + */ + function canConvertFromVBindSlotToVSlot (slotAttr) { + if (slotAttr.parent.parent.name !== 'template') { + return false + } + + if (!slotAttr.value) { + return true + } + + if (!slotAttr.value.expression) { + // parse error or empty expression + return false + } + const slotName = sourceCode.getText(slotAttr.value.expression).trim() + // If non-Latin characters are included it can not be converted. + // It does not check the space only because `a>b?c:d` should be rejected. + return !/[^a-z]/i.test(slotName) + } + + /** + * Convert to `v-slot`. + * @param {object} fixer fixer + * @param {VAttribute} slotAttr node of `slot` + * @param {string | null} slotName name of `slot` + * @param {boolean} vBind `true` if `slotAttr` is `v-bind:slot` + * @returns {*} fix data + */ + function fixSlotToVSlot (fixer, slotAttr, slotName, vBind) { + const element = slotAttr.parent + const scopeAttr = element.attributes + .find(attr => attr.directive === true && attr.key.name && ( + attr.key.name.name === 'slot-scope' || + attr.key.name.name === 'scope' + )) + const nameArgument = slotName ? (vBind ? `:[${slotName}]` : `:${slotName}`) : '' + const scopeValue = scopeAttr && scopeAttr.value + ? `=${sourceCode.getText(scopeAttr.value)}` + : '' + + const replaceText = `v-slot${nameArgument}${scopeValue}` + const fixers = [ + fixer.replaceText(slotAttr || scopeAttr, replaceText) + ] + if (slotAttr && scopeAttr) { + fixers.push(fixer.remove(scopeAttr)) + } + return fixers + } + /** + * Reports `slot` node + * @param {VAttribute} slotAttr node of `slot` + * @returns {void} + */ + function reportSlot (slotAttr) { + context.report({ + node: slotAttr.key, + messageId: 'forbiddenSlotAttribute', + // fix to use `v-slot` + fix (fixer) { + if (!canConvertFromSlotToVSlot(slotAttr)) { + return null + } + const slotName = slotAttr.value && + slotAttr.value.value + return fixSlotToVSlot(fixer, slotAttr, slotName, false) + } + }) + } + /** + * Reports `v-bind:slot` node + * @param {VAttribute} slotAttr node of `v-bind:slot` + * @returns {void} + */ + function reportVBindSlot (slotAttr) { + context.report({ + node: slotAttr.key, + messageId: 'forbiddenSlotAttribute', + // fix to use `v-slot` + fix (fixer) { + if (!canConvertFromVBindSlotToVSlot(slotAttr)) { + return null + } + const slotName = slotAttr.value && + slotAttr.value.expression && + sourceCode.getText(slotAttr.value.expression).trim() + return fixSlotToVSlot(fixer, slotAttr, slotName, true) + } + }) + } + + return { + "VAttribute[directive=false][key.name='slot']": reportSlot, + "VAttribute[directive=true][key.name.name='bind'][key.argument.name='slot']": reportVBindSlot + } + } +} diff --git a/tests/lib/rules/no-deprecated-slot-attribute.js b/tests/lib/rules/no-deprecated-slot-attribute.js new file mode 100644 index 000000000..bb0dd9b54 --- /dev/null +++ b/tests/lib/rules/no-deprecated-slot-attribute.js @@ -0,0 +1,353 @@ +'use strict' + +const RuleTester = require('eslint').RuleTester +const rule = require('../../../lib/rules/no-deprecated-slot-attribute.js') + +const tester = new RuleTester({ + parser: 'vue-eslint-parser', + parserOptions: { + ecmaVersion: 2015 + } +}) + +tester.run('no-deprecated-slot-attribute', rule, { + valid: [ + ``, + ``, + ``, + ``, + ``, + ``, + `` + ], + invalid: [ + { + code: ` + `, + output: ` + `, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: ` + `, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: ` + `, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: ` + `, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: ` + `, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: null, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + }, + { + message: '`slot` attributes are deprecated.', + line: 5 + }, + { + message: '`slot` attributes are deprecated.', + line: 6 + }, + { + message: '`slot` attributes are deprecated.', + line: 7 + }, + { + message: '`slot` attributes are deprecated.', + line: 8 + } + ] + }, + { + code: ` + `, + output: ` + `, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: null, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: ` + `, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: null, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: null, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: null, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: null, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: ` + `, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: null, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + }, + { + code: ` + `, + output: null, + errors: [ + { + message: '`slot` attributes are deprecated.', + line: 4 + } + ] + } + ] +})