index.js 2.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. 'use strict';
  2. const isStandardSyntaxRule = require('../../utils/isStandardSyntaxRule');
  3. const parseSelector = require('../../utils/parseSelector');
  4. const report = require('../../utils/report');
  5. const ruleMessages = require('../../utils/ruleMessages');
  6. const validateOptions = require('../../utils/validateOptions');
  7. const ruleName = 'selector-descendant-combinator-no-non-space';
  8. const messages = ruleMessages(ruleName, {
  9. rejected: (nonSpaceCharacter) => `Unexpected "${nonSpaceCharacter}"`,
  10. });
  11. const meta = {
  12. url: 'https://stylelint.io/user-guide/rules/list/selector-descendant-combinator-no-non-space',
  13. fixable: true,
  14. };
  15. /** @type {import('stylelint').Rule} */
  16. const rule = (primary, _secondaryOptions, context) => {
  17. return (root, result) => {
  18. const validOptions = validateOptions(result, ruleName, {
  19. actual: primary,
  20. });
  21. if (!validOptions) {
  22. return;
  23. }
  24. root.walkRules((ruleNode) => {
  25. if (!isStandardSyntaxRule(ruleNode)) {
  26. return;
  27. }
  28. let hasFixed = false;
  29. const selector = ruleNode.raws.selector ? ruleNode.raws.selector.raw : ruleNode.selector;
  30. // Return early for selectors containing comments
  31. // TODO: renable when parser and stylelint are compatible
  32. if (selector.includes('/*')) return;
  33. const fixedSelector = parseSelector(selector, result, ruleNode, (fullSelector) => {
  34. fullSelector.walkCombinators((combinatorNode) => {
  35. if (combinatorNode.value !== ' ') {
  36. return;
  37. }
  38. const value = combinatorNode.toString();
  39. if (
  40. value.includes(' ') ||
  41. value.includes('\t') ||
  42. value.includes('\n') ||
  43. value.includes('\r')
  44. ) {
  45. if (context.fix && /^\s+$/.test(value)) {
  46. hasFixed = true;
  47. if (!combinatorNode.raws) combinatorNode.raws = {};
  48. combinatorNode.raws.value = ' ';
  49. combinatorNode.rawSpaceBefore = combinatorNode.rawSpaceBefore.replace(/^\s+/, '');
  50. combinatorNode.rawSpaceAfter = combinatorNode.rawSpaceAfter.replace(/\s+$/, '');
  51. return;
  52. }
  53. report({
  54. result,
  55. ruleName,
  56. message: messages.rejected(value),
  57. node: ruleNode,
  58. index: combinatorNode.sourceIndex,
  59. });
  60. }
  61. });
  62. });
  63. if (hasFixed && fixedSelector) {
  64. if (!ruleNode.raws.selector) {
  65. ruleNode.selector = fixedSelector;
  66. } else {
  67. ruleNode.raws.selector.raw = fixedSelector;
  68. }
  69. }
  70. });
  71. };
  72. };
  73. rule.ruleName = ruleName;
  74. rule.messages = messages;
  75. rule.meta = meta;
  76. module.exports = rule;