padding-line-between-blocks.js 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. /**
  2. * @fileoverview Require or disallow padding lines between blocks
  3. * @author Yosuke Ota
  4. */
  5. 'use strict'
  6. const utils = require('../utils')
  7. /**
  8. * Split the source code into multiple lines based on the line delimiters.
  9. * @param {string} text Source code as a string.
  10. * @returns {string[]} Array of source code lines.
  11. */
  12. function splitLines(text) {
  13. return text.split(/\r\n|[\r\n\u2028\u2029]/gu)
  14. }
  15. /**
  16. * Check and report blocks for `never` configuration.
  17. * This autofix removes blank lines between the given 2 blocks.
  18. * @param {RuleContext} context The rule context to report.
  19. * @param {VElement} prevBlock The previous block to check.
  20. * @param {VElement} nextBlock The next block to check.
  21. * @param {Token[]} betweenTokens The array of tokens between blocks.
  22. * @returns {void}
  23. * @private
  24. */
  25. function verifyForNever(context, prevBlock, nextBlock, betweenTokens) {
  26. if (prevBlock.loc.end.line === nextBlock.loc.start.line) {
  27. // same line
  28. return
  29. }
  30. const tokenOrNodes = [...betweenTokens, nextBlock]
  31. /** @type {ASTNode | Token} */
  32. let prev = prevBlock
  33. /** @type {[ASTNode | Token, ASTNode | Token][]} */
  34. const paddingLines = []
  35. for (const tokenOrNode of tokenOrNodes) {
  36. const numOfLineBreaks = tokenOrNode.loc.start.line - prev.loc.end.line
  37. if (numOfLineBreaks > 1) {
  38. paddingLines.push([prev, tokenOrNode])
  39. }
  40. prev = tokenOrNode
  41. }
  42. if (paddingLines.length === 0) {
  43. return
  44. }
  45. context.report({
  46. node: nextBlock,
  47. messageId: 'never',
  48. *fix(fixer) {
  49. for (const [prevToken, nextToken] of paddingLines) {
  50. const start = prevToken.range[1]
  51. const end = nextToken.range[0]
  52. const paddingText = context.getSourceCode().text.slice(start, end)
  53. const lastSpaces = splitLines(paddingText).pop()
  54. yield fixer.replaceTextRange([start, end], `\n${lastSpaces}`)
  55. }
  56. }
  57. })
  58. }
  59. /**
  60. * Check and report blocks for `always` configuration.
  61. * This autofix inserts a blank line between the given 2 blocks.
  62. * @param {RuleContext} context The rule context to report.
  63. * @param {VElement} prevBlock The previous block to check.
  64. * @param {VElement} nextBlock The next block to check.
  65. * @param {Token[]} betweenTokens The array of tokens between blocks.
  66. * @returns {void}
  67. * @private
  68. */
  69. function verifyForAlways(context, prevBlock, nextBlock, betweenTokens) {
  70. const tokenOrNodes = [...betweenTokens, nextBlock]
  71. /** @type {ASTNode | Token} */
  72. let prev = prevBlock
  73. /** @type {ASTNode | Token | undefined} */
  74. let linebreak
  75. for (const tokenOrNode of tokenOrNodes) {
  76. const numOfLineBreaks = tokenOrNode.loc.start.line - prev.loc.end.line
  77. if (numOfLineBreaks > 1) {
  78. // Already padded.
  79. return
  80. }
  81. if (!linebreak && numOfLineBreaks > 0) {
  82. linebreak = prev
  83. }
  84. prev = tokenOrNode
  85. }
  86. context.report({
  87. node: nextBlock,
  88. messageId: 'always',
  89. fix(fixer) {
  90. if (linebreak) {
  91. return fixer.insertTextAfter(linebreak, '\n')
  92. }
  93. return fixer.insertTextAfter(prevBlock, '\n\n')
  94. }
  95. })
  96. }
  97. /**
  98. * Types of blank lines.
  99. * `never` and `always` are defined.
  100. * Those have `verify` method to check and report statements.
  101. * @private
  102. */
  103. const PaddingTypes = {
  104. never: { verify: verifyForNever },
  105. always: { verify: verifyForAlways }
  106. }
  107. module.exports = {
  108. meta: {
  109. type: 'layout',
  110. docs: {
  111. description: 'require or disallow padding lines between blocks',
  112. categories: undefined,
  113. url: 'https://eslint.vuejs.org/rules/padding-line-between-blocks.html'
  114. },
  115. fixable: 'whitespace',
  116. schema: [
  117. {
  118. enum: Object.keys(PaddingTypes)
  119. }
  120. ],
  121. messages: {
  122. never: 'Unexpected blank line before this block.',
  123. always: 'Expected blank line before this block.'
  124. }
  125. },
  126. /** @param {RuleContext} context */
  127. create(context) {
  128. if (!context.parserServices.getDocumentFragment) {
  129. return {}
  130. }
  131. const df = context.parserServices.getDocumentFragment()
  132. if (!df) {
  133. return {}
  134. }
  135. const documentFragment = df
  136. /** @type {'always' | 'never'} */
  137. const option = context.options[0] || 'always'
  138. const paddingType = PaddingTypes[option]
  139. /** @type {Token[]} */
  140. let tokens
  141. /**
  142. * @returns {VElement[]}
  143. */
  144. function getTopLevelHTMLElements() {
  145. return documentFragment.children.filter(utils.isVElement)
  146. }
  147. /**
  148. * @param {VElement} prev
  149. * @param {VElement} next
  150. */
  151. function getTokenAndCommentsBetween(prev, next) {
  152. // When there is no <template>, tokenStore.getTokensBetween cannot be used.
  153. if (!tokens) {
  154. tokens = [
  155. ...documentFragment.tokens.filter(
  156. (token) => token.type !== 'HTMLWhitespace'
  157. ),
  158. ...documentFragment.comments
  159. ].sort((a, b) => {
  160. if (a.range[0] > b.range[0]) return 1
  161. return a.range[0] < b.range[0] ? -1 : 0
  162. })
  163. }
  164. let token = tokens.shift()
  165. const results = []
  166. while (token) {
  167. if (prev.range[1] <= token.range[0]) {
  168. if (next.range[0] <= token.range[0]) {
  169. tokens.unshift(token)
  170. break
  171. } else {
  172. results.push(token)
  173. }
  174. }
  175. token = tokens.shift()
  176. }
  177. return results
  178. }
  179. return utils.defineTemplateBodyVisitor(
  180. context,
  181. {},
  182. {
  183. /** @param {Program} node */
  184. Program(node) {
  185. if (utils.hasInvalidEOF(node)) {
  186. return
  187. }
  188. const elements = [...getTopLevelHTMLElements()]
  189. let prev = elements.shift()
  190. for (const element of elements) {
  191. if (!prev) {
  192. return
  193. }
  194. const betweenTokens = getTokenAndCommentsBetween(prev, element)
  195. paddingType.verify(context, prev, element, betweenTokens)
  196. prev = element
  197. }
  198. }
  199. }
  200. )
  201. }
  202. }