123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109 |
- /**
- * @author Toru Nagashima
- * @copyright 2016 Toru Nagashima. All rights reserved.
- * See LICENSE file in root directory for full license.
- */
- 'use strict'
- const utils = require('../utils')
- /**
- * @param {number} lineBreaks
- */
- function getPhrase(lineBreaks) {
- switch (lineBreaks) {
- case 0: {
- return 'no line breaks'
- }
- case 1: {
- return '1 line break'
- }
- default: {
- return `${lineBreaks} line breaks`
- }
- }
- }
- module.exports = {
- meta: {
- type: 'layout',
- docs: {
- description:
- "require or disallow a line break before tag's closing brackets",
- categories: ['vue3-strongly-recommended', 'strongly-recommended'],
- url: 'https://eslint.vuejs.org/rules/html-closing-bracket-newline.html'
- },
- fixable: 'whitespace',
- schema: [
- {
- type: 'object',
- properties: {
- singleline: { enum: ['always', 'never'] },
- multiline: { enum: ['always', 'never'] }
- },
- additionalProperties: false
- }
- ],
- messages: {
- expectedBeforeClosingBracket:
- 'Expected {{expected}} before closing bracket, but {{actual}} found.'
- }
- },
- /** @param {RuleContext} context */
- create(context) {
- const options = Object.assign(
- {},
- {
- singleline: 'never',
- multiline: 'always'
- },
- context.options[0] || {}
- )
- const template =
- context.parserServices.getTemplateBodyTokenStore &&
- context.parserServices.getTemplateBodyTokenStore()
- return utils.defineDocumentVisitor(context, {
- /** @param {VStartTag | VEndTag} node */
- 'VStartTag, VEndTag'(node) {
- const closingBracketToken = template.getLastToken(node)
- if (
- closingBracketToken.type !== 'HTMLSelfClosingTagClose' &&
- closingBracketToken.type !== 'HTMLTagClose'
- ) {
- return
- }
- const prevToken = template.getTokenBefore(closingBracketToken)
- const type =
- node.loc.start.line === prevToken.loc.end.line
- ? 'singleline'
- : 'multiline'
- const expectedLineBreaks = options[type] === 'always' ? 1 : 0
- const actualLineBreaks =
- closingBracketToken.loc.start.line - prevToken.loc.end.line
- if (actualLineBreaks !== expectedLineBreaks) {
- context.report({
- node,
- loc: {
- start: prevToken.loc.end,
- end: closingBracketToken.loc.start
- },
- messageId: 'expectedBeforeClosingBracket',
- data: {
- expected: getPhrase(expectedLineBreaks),
- actual: getPhrase(actualLineBreaks)
- },
- fix(fixer) {
- /** @type {Range} */
- const range = [prevToken.range[1], closingBracketToken.range[0]]
- const text = '\n'.repeat(expectedLineBreaks)
- return fixer.replaceTextRange(range, text)
- }
- })
- }
- }
- })
- }
- }
|