block-order.js 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. /**
  2. * @author Yosuke Ota
  3. * issue https://github.com/vuejs/eslint-plugin-vue/issues/140
  4. */
  5. 'use strict'
  6. const utils = require('../utils')
  7. const { parseSelector } = require('../utils/selector')
  8. /**
  9. * @typedef {import('../utils/selector').VElementSelector} VElementSelector
  10. */
  11. const DEFAULT_ORDER = Object.freeze([['script', 'template'], 'style'])
  12. /**
  13. * @param {VElement} element
  14. * @return {string}
  15. */
  16. function getAttributeString(element) {
  17. return element.startTag.attributes
  18. .map((attribute) => {
  19. if (attribute.value && attribute.value.type !== 'VLiteral') {
  20. return ''
  21. }
  22. return `${attribute.key.name}${
  23. attribute.value && attribute.value.value
  24. ? `=${attribute.value.value}`
  25. : ''
  26. }`
  27. })
  28. .join(' ')
  29. }
  30. module.exports = {
  31. meta: {
  32. type: 'suggestion',
  33. docs: {
  34. description: 'enforce order of component top-level elements',
  35. categories: undefined,
  36. url: 'https://eslint.vuejs.org/rules/block-order.html'
  37. },
  38. fixable: 'code',
  39. schema: [
  40. {
  41. type: 'object',
  42. properties: {
  43. order: {
  44. type: 'array',
  45. items: {
  46. anyOf: [
  47. { type: 'string' },
  48. { type: 'array', items: { type: 'string' }, uniqueItems: true }
  49. ]
  50. },
  51. uniqueItems: true,
  52. additionalItems: false
  53. }
  54. },
  55. additionalProperties: false
  56. }
  57. ],
  58. messages: {
  59. unexpected:
  60. "'<{{elementName}}{{elementAttributes}}>' should be above '<{{firstUnorderedName}}{{firstUnorderedAttributes}}>' on line {{line}}."
  61. }
  62. },
  63. /**
  64. * @param {RuleContext} context - The rule context.
  65. * @returns {RuleListener} AST event handlers.
  66. */
  67. create(context) {
  68. /**
  69. * @typedef {object} OrderElement
  70. * @property {string} selectorText
  71. * @property {VElementSelector} selector
  72. * @property {number} index
  73. */
  74. /** @type {OrderElement[]} */
  75. const orders = []
  76. /** @type {(string|string[])[]} */
  77. const orderOptions =
  78. (context.options[0] && context.options[0].order) || DEFAULT_ORDER
  79. for (const [index, selectorOrSelectors] of orderOptions.entries()) {
  80. if (Array.isArray(selectorOrSelectors)) {
  81. for (const selector of selectorOrSelectors) {
  82. orders.push({
  83. selectorText: selector,
  84. selector: parseSelector(selector, context),
  85. index
  86. })
  87. }
  88. } else {
  89. orders.push({
  90. selectorText: selectorOrSelectors,
  91. selector: parseSelector(selectorOrSelectors, context),
  92. index
  93. })
  94. }
  95. }
  96. /**
  97. * @param {VElement} element
  98. */
  99. function getOrderElement(element) {
  100. return orders.find((o) => o.selector.test(element))
  101. }
  102. const documentFragment =
  103. context.parserServices.getDocumentFragment &&
  104. context.parserServices.getDocumentFragment()
  105. function getTopLevelHTMLElements() {
  106. if (documentFragment) {
  107. return documentFragment.children.filter(utils.isVElement)
  108. }
  109. return []
  110. }
  111. return {
  112. Program(node) {
  113. if (utils.hasInvalidEOF(node)) {
  114. return
  115. }
  116. const elements = getTopLevelHTMLElements()
  117. const elementsWithOrder = elements.flatMap((element) => {
  118. const order = getOrderElement(element)
  119. return order ? [{ order, element }] : []
  120. })
  121. const sourceCode = context.getSourceCode()
  122. for (const [index, elementWithOrders] of elementsWithOrder.entries()) {
  123. const { order: expected, element } = elementWithOrders
  124. const firstUnordered = elementsWithOrder
  125. .slice(0, index)
  126. .filter(({ order }) => expected.index < order.index)
  127. .sort((e1, e2) => e1.order.index - e2.order.index)[0]
  128. if (firstUnordered) {
  129. const firstUnorderedAttributes = getAttributeString(
  130. firstUnordered.element
  131. )
  132. const elementAttributes = getAttributeString(element)
  133. context.report({
  134. node: element,
  135. loc: element.loc,
  136. messageId: 'unexpected',
  137. data: {
  138. elementName: element.name,
  139. elementAttributes: elementAttributes
  140. ? ` ${elementAttributes}`
  141. : '',
  142. firstUnorderedName: firstUnordered.element.name,
  143. firstUnorderedAttributes: firstUnorderedAttributes
  144. ? ` ${firstUnorderedAttributes}`
  145. : '',
  146. line: firstUnordered.element.loc.start.line
  147. },
  148. *fix(fixer) {
  149. // insert element before firstUnordered
  150. const fixedElements = elements.flatMap((it) => {
  151. if (it === firstUnordered.element) {
  152. return [element, it]
  153. } else if (it === element) {
  154. return []
  155. }
  156. return [it]
  157. })
  158. for (let i = elements.length - 1; i >= 0; i--) {
  159. if (elements[i] !== fixedElements[i]) {
  160. yield fixer.replaceTextRange(
  161. elements[i].range,
  162. sourceCode.text.slice(...fixedElements[i].range)
  163. )
  164. }
  165. }
  166. }
  167. })
  168. }
  169. }
  170. }
  171. }
  172. }
  173. }