| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258 | /** * @author Yosuke ota * See LICENSE file in root directory for full license. */'use strict'// -----------------------------------------------------------------------------// Requirements// -----------------------------------------------------------------------------const htmlComments = require('../utils/html-comments')// ------------------------------------------------------------------------------// Helpers// ------------------------------------------------------------------------------/** * Normalize options. * @param {number|"tab"|undefined} type The type of indentation. * @returns { { indentChar: string, indentSize: number, indentText: string } } Normalized options. */function parseOptions(type) {  const ret = {    indentChar: ' ',    indentSize: 2,    indentText: ''  }  if (Number.isSafeInteger(type)) {    ret.indentSize = Number(type)  } else if (type === 'tab') {    ret.indentChar = '\t'    ret.indentSize = 1  }  ret.indentText = ret.indentChar.repeat(ret.indentSize)  return ret}/** * @param {string} s * @param {string} [unitChar] */function toDisplay(s, unitChar) {  if (s.length === 0 && unitChar) {    return `0 ${toUnit(unitChar)}s`  }  const char = s[0]  if (char === ' ' || char === '\t') {    if (s.split('').every((c) => c === char)) {      return `${s.length} ${toUnit(char)}${s.length === 1 ? '' : 's'}`    }  }  return JSON.stringify(s)}/** @param {string} char */function toUnit(char) {  if (char === '\t') {    return 'tab'  }  if (char === ' ') {    return 'space'  }  return JSON.stringify(char)}// ------------------------------------------------------------------------------// Rule Definition// ------------------------------------------------------------------------------module.exports = {  meta: {    type: 'layout',    docs: {      description: 'enforce consistent indentation in HTML comments',      categories: undefined,      url: 'https://eslint.vuejs.org/rules/html-comment-indent.html'    },    fixable: 'whitespace',    schema: [      {        anyOf: [{ type: 'integer', minimum: 0 }, { enum: ['tab'] }]      }    ],    messages: {      unexpectedBaseIndentation:        'Expected base point indentation of {{expected}}, but found {{actual}}.',      missingBaseIndentation:        'Expected base point indentation of {{expected}}, but not found.',      unexpectedIndentationCharacter:        'Expected {{expected}} character, but found {{actual}} character.',      unexpectedIndentation:        'Expected indentation of {{expected}} but found {{actual}}.',      unexpectedRelativeIndentation:        'Expected relative indentation of {{expected}} but found {{actual}}.'    }  },  /** @param {RuleContext} context */  create(context) {    const options = parseOptions(context.options[0])    const sourceCode = context.getSourceCode()    return htmlComments.defineVisitor(      context,      null,      (comment) => {        const baseIndentText = getLineIndentText(comment.open.loc.start.line)        let endLine        if (comment.value) {          const startLine = comment.value.loc.start.line          endLine = comment.value.loc.end.line          const checkStartLine =            comment.open.loc.end.line === startLine ? startLine + 1 : startLine          for (let line = checkStartLine; line <= endLine; line++) {            validateIndentForLine(line, baseIndentText, 1)          }        } else {          endLine = comment.open.loc.end.line        }        if (endLine < comment.close.loc.start.line) {          // `-->`          validateIndentForLine(comment.close.loc.start.line, baseIndentText, 0)        }      },      { includeDirectives: true }    )    /**     * Checks whether the given line is a blank line.     * @param {number} line The number of line. Begins with 1.     * @returns {boolean} `true` if the given line is a blank line     */    function isEmptyLine(line) {      const lineText = sourceCode.getLines()[line - 1]      return !lineText.trim()    }    /**     * Get the actual indentation of the given line.     * @param {number} line The number of line. Begins with 1.     * @returns {string} The actual indentation text     */    function getLineIndentText(line) {      const lineText = sourceCode.getLines()[line - 1]      const charIndex = lineText.search(/\S/)      // already checked      // if (charIndex < 0) {      //   return lineText      // }      return lineText.slice(0, charIndex)    }    /**     * Define the function which fixes the problem.     * @param {number} line The number of line.     * @param {string} actualIndentText The actual indentation text.     * @param {string} expectedIndentText The expected indentation text.     * @returns { (fixer: RuleFixer) => Fix } The defined function.     */    function defineFix(line, actualIndentText, expectedIndentText) {      return (fixer) => {        const start = sourceCode.getIndexFromLoc({          line,          column: 0        })        return fixer.replaceTextRange(          [start, start + actualIndentText.length],          expectedIndentText        )      }    }    /**     * Validate the indentation of a line.     * @param {number} line The number of line. Begins with 1.     * @param {string} baseIndentText The expected base indentation text.     * @param {number} offset The number of the indentation offset.     */    function validateIndentForLine(line, baseIndentText, offset) {      if (isEmptyLine(line)) {        return      }      const actualIndentText = getLineIndentText(line)      const expectedOffsetIndentText = options.indentText.repeat(offset)      const expectedIndentText = baseIndentText + expectedOffsetIndentText      // validate base indent      if (        baseIndentText &&        (actualIndentText.length < baseIndentText.length ||          !actualIndentText.startsWith(baseIndentText))      ) {        context.report({          loc: {            start: { line, column: 0 },            end: { line, column: actualIndentText.length }          },          messageId: actualIndentText            ? 'unexpectedBaseIndentation'            : 'missingBaseIndentation',          data: {            expected: toDisplay(baseIndentText),            actual: toDisplay(actualIndentText.slice(0, baseIndentText.length))          },          fix: defineFix(line, actualIndentText, expectedIndentText)        })        return      }      const actualOffsetIndentText = actualIndentText.slice(        baseIndentText.length      )      // validate indent charctor      for (let i = 0; i < actualOffsetIndentText.length; ++i) {        if (actualOffsetIndentText[i] !== options.indentChar) {          context.report({            loc: {              start: { line, column: baseIndentText.length + i },              end: { line, column: baseIndentText.length + i + 1 }            },            messageId: 'unexpectedIndentationCharacter',            data: {              expected: toUnit(options.indentChar),              actual: toUnit(actualOffsetIndentText[i])            },            fix: defineFix(line, actualIndentText, expectedIndentText)          })          return        }      }      // validate indent length      if (actualOffsetIndentText.length !== expectedOffsetIndentText.length) {        context.report({          loc: {            start: { line, column: baseIndentText.length },            end: { line, column: actualIndentText.length }          },          messageId: baseIndentText            ? 'unexpectedRelativeIndentation'            : 'unexpectedIndentation',          data: {            expected: toDisplay(expectedOffsetIndentText, options.indentChar),            actual: toDisplay(actualOffsetIndentText, options.indentChar)          },          fix: defineFix(line, actualIndentText, expectedIndentText)        })      }    }  }}
 |