index.js 1.9 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879
  1. 'use strict';
  2. const isStandardSyntaxRule = require('../../utils/isStandardSyntaxRule');
  3. const matchesStringOrRegExp = require('../../utils/matchesStringOrRegExp');
  4. const parseSelector = require('../../utils/parseSelector');
  5. const report = require('../../utils/report');
  6. const ruleMessages = require('../../utils/ruleMessages');
  7. const validateOptions = require('../../utils/validateOptions');
  8. const vendor = require('../../utils/vendor');
  9. const { isRegExp, isString } = require('../../utils/validateTypes');
  10. const ruleName = 'selector-pseudo-element-allowed-list';
  11. const messages = ruleMessages(ruleName, {
  12. rejected: (selector) => `Unexpected pseudo-element "${selector}"`,
  13. });
  14. const meta = {
  15. url: 'https://stylelint.io/user-guide/rules/list/selector-pseudo-element-allowed-list',
  16. };
  17. /** @type {import('stylelint').Rule<string | RegExp | Array<string | RegExp>>} */
  18. const rule = (primary) => {
  19. return (root, result) => {
  20. const validOptions = validateOptions(result, ruleName, {
  21. actual: primary,
  22. possible: [isString, isRegExp],
  23. });
  24. if (!validOptions) {
  25. return;
  26. }
  27. root.walkRules((ruleNode) => {
  28. if (!isStandardSyntaxRule(ruleNode)) {
  29. return;
  30. }
  31. const selector = ruleNode.selector;
  32. if (!selector.includes('::')) {
  33. return;
  34. }
  35. parseSelector(selector, result, ruleNode, (selectorTree) => {
  36. selectorTree.walkPseudos((pseudoNode) => {
  37. const value = pseudoNode.value;
  38. // Ignore pseudo-classes
  39. if (value.charAt(1) !== ':') {
  40. return;
  41. }
  42. const name = value.slice(2);
  43. if (matchesStringOrRegExp(vendor.unprefixed(name), primary)) {
  44. return;
  45. }
  46. report({
  47. index: pseudoNode.sourceIndex,
  48. message: messages.rejected(value),
  49. node: ruleNode,
  50. word: value,
  51. result,
  52. ruleName,
  53. });
  54. });
  55. });
  56. });
  57. };
  58. };
  59. rule.primaryOptionArray = true;
  60. rule.ruleName = ruleName;
  61. rule.messages = messages;
  62. rule.meta = meta;
  63. module.exports = rule;