index.js 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  1. 'use strict';
  2. const addEmptyLineBefore = require('../../utils/addEmptyLineBefore');
  3. const getPreviousNonSharedLineCommentNode = require('../../utils/getPreviousNonSharedLineCommentNode');
  4. const hasEmptyLine = require('../../utils/hasEmptyLine');
  5. const isAfterComment = require('../../utils/isAfterComment');
  6. const isBlocklessAtRuleAfterBlocklessAtRule = require('../../utils/isBlocklessAtRuleAfterBlocklessAtRule');
  7. const isBlocklessAtRuleAfterSameNameBlocklessAtRule = require('../../utils/isBlocklessAtRuleAfterSameNameBlocklessAtRule');
  8. const isFirstNested = require('../../utils/isFirstNested');
  9. const isFirstNodeOfRoot = require('../../utils/isFirstNodeOfRoot');
  10. const isStandardSyntaxAtRule = require('../../utils/isStandardSyntaxAtRule');
  11. const optionsMatches = require('../../utils/optionsMatches');
  12. const removeEmptyLinesBefore = require('../../utils/removeEmptyLinesBefore');
  13. const report = require('../../utils/report');
  14. const ruleMessages = require('../../utils/ruleMessages');
  15. const { isAtRule } = require('../../utils/typeGuards');
  16. const validateOptions = require('../../utils/validateOptions');
  17. const { isString } = require('../../utils/validateTypes');
  18. const ruleName = 'at-rule-empty-line-before';
  19. const messages = ruleMessages(ruleName, {
  20. expected: 'Expected empty line before at-rule',
  21. rejected: 'Unexpected empty line before at-rule',
  22. });
  23. const meta = {
  24. url: 'https://stylelint.io/user-guide/rules/list/at-rule-empty-line-before',
  25. fixable: true,
  26. };
  27. /** @type {import('stylelint').Rule} */
  28. const rule = (primary, secondaryOptions, context) => {
  29. return (root, result) => {
  30. const validOptions = validateOptions(
  31. result,
  32. ruleName,
  33. {
  34. actual: primary,
  35. possible: ['always', 'never'],
  36. },
  37. {
  38. actual: secondaryOptions,
  39. possible: {
  40. except: [
  41. 'after-same-name',
  42. 'inside-block',
  43. 'blockless-after-same-name-blockless',
  44. 'blockless-after-blockless',
  45. 'first-nested',
  46. ],
  47. ignore: [
  48. 'after-comment',
  49. 'first-nested',
  50. 'inside-block',
  51. 'blockless-after-same-name-blockless',
  52. 'blockless-after-blockless',
  53. ],
  54. ignoreAtRules: [isString],
  55. },
  56. optional: true,
  57. },
  58. );
  59. if (!validOptions) {
  60. return;
  61. }
  62. /** @type {'always' | 'never'} */
  63. const expectation = primary;
  64. root.walkAtRules((atRule) => {
  65. const isNested = atRule.parent && atRule.parent.type !== 'root';
  66. // Ignore the first node
  67. if (isFirstNodeOfRoot(atRule)) {
  68. return;
  69. }
  70. if (!isStandardSyntaxAtRule(atRule)) {
  71. return;
  72. }
  73. // Return early if at-rule is to be ignored
  74. if (optionsMatches(secondaryOptions, 'ignoreAtRules', atRule.name)) {
  75. return;
  76. }
  77. // Optionally ignore the expectation if the node is blockless
  78. if (
  79. optionsMatches(secondaryOptions, 'ignore', 'blockless-after-blockless') &&
  80. isBlocklessAtRuleAfterBlocklessAtRule(atRule)
  81. ) {
  82. return;
  83. }
  84. // Optionally ignore the node if it is the first nested
  85. if (optionsMatches(secondaryOptions, 'ignore', 'first-nested') && isFirstNested(atRule)) {
  86. return;
  87. }
  88. // Optionally ignore the expectation if the node is blockless
  89. // and following another blockless at-rule with the same name
  90. if (
  91. optionsMatches(secondaryOptions, 'ignore', 'blockless-after-same-name-blockless') &&
  92. isBlocklessAtRuleAfterSameNameBlocklessAtRule(atRule)
  93. ) {
  94. return;
  95. }
  96. // Optionally ignore the expectation if the node is inside a block
  97. if (optionsMatches(secondaryOptions, 'ignore', 'inside-block') && isNested) {
  98. return;
  99. }
  100. // Optionally ignore the expectation if a comment precedes this node
  101. if (optionsMatches(secondaryOptions, 'ignore', 'after-comment') && isAfterComment(atRule)) {
  102. return;
  103. }
  104. const hasEmptyLineBefore = hasEmptyLine(atRule.raws.before);
  105. let expectEmptyLineBefore = expectation === 'always';
  106. // Optionally reverse the expectation if any exceptions apply
  107. if (
  108. (optionsMatches(secondaryOptions, 'except', 'after-same-name') &&
  109. isAtRuleAfterSameNameAtRule(atRule)) ||
  110. (optionsMatches(secondaryOptions, 'except', 'inside-block') && isNested) ||
  111. (optionsMatches(secondaryOptions, 'except', 'first-nested') && isFirstNested(atRule)) ||
  112. (optionsMatches(secondaryOptions, 'except', 'blockless-after-blockless') &&
  113. isBlocklessAtRuleAfterBlocklessAtRule(atRule)) ||
  114. (optionsMatches(secondaryOptions, 'except', 'blockless-after-same-name-blockless') &&
  115. isBlocklessAtRuleAfterSameNameBlocklessAtRule(atRule))
  116. ) {
  117. expectEmptyLineBefore = !expectEmptyLineBefore;
  118. }
  119. // Return if the expectation is met
  120. if (expectEmptyLineBefore === hasEmptyLineBefore) {
  121. return;
  122. }
  123. // Fix
  124. if (context.fix && context.newline) {
  125. if (expectEmptyLineBefore) {
  126. addEmptyLineBefore(atRule, context.newline);
  127. } else {
  128. removeEmptyLinesBefore(atRule, context.newline);
  129. }
  130. return;
  131. }
  132. const message = expectEmptyLineBefore ? messages.expected : messages.rejected;
  133. report({ message, node: atRule, result, ruleName });
  134. });
  135. };
  136. };
  137. /**
  138. * @param {import('postcss').AtRule} atRule
  139. */
  140. function isAtRuleAfterSameNameAtRule(atRule) {
  141. const previousNode = getPreviousNonSharedLineCommentNode(atRule);
  142. return previousNode && isAtRule(previousNode) && previousNode.name === atRule.name;
  143. }
  144. rule.ruleName = ruleName;
  145. rule.messages = messages;
  146. rule.meta = meta;
  147. module.exports = rule;