UNPKG

stylelint

Version:

A mighty CSS linter that helps you avoid errors and enforce conventions.

76 lines (63 loc) 2.43 kB
import { isRegExp, isString } from '../../utils/validateTypes.mjs'; import findNodeUpToRoot from '../../utils/findNodeUpToRoot.mjs'; import { isAtRule } from '../../utils/typeGuards.mjs'; import isStandardSyntaxRule from '../../utils/isStandardSyntaxRule.mjs'; import matchesStringOrRegExp from '../../utils/matchesStringOrRegExp.mjs'; import report from '../../utils/report.mjs'; import ruleMessages from '../../utils/ruleMessages.mjs'; import validateOptions from '../../utils/validateOptions.mjs'; const ruleName = 'rule-nesting-at-rule-required-list'; const messages = ruleMessages(ruleName, { expected: (...atRulePatterns) => { const patterns = atRulePatterns.map((pattern) => `"${pattern}"`).join(', '); return `Expected rule to be nested in an at-rule matching pattern(s): ${patterns}`; }, }); const meta = { url: 'https://stylelint.io/user-guide/rules/rule-nesting-at-rule-required-list', }; /** @type {import('stylelint').CoreRules[ruleName]} */ const rule = (primary) => { return (root, result) => { // Validate that the primary option is a string, regex, or array of strings/regexes const validOptions = validateOptions(result, ruleName, { actual: primary, possible: [isString, isRegExp], }); if (!validOptions) return; const requiredAtRules = [primary].flat(); root.walkRules((ruleNode) => { if (!isStandardSyntaxRule(ruleNode)) return; // Skip if rule already has a required at-rule ancestor if (hasRequiredAtRuleAncestor(ruleNode, requiredAtRules)) return; report({ message: messages.expected, messageArgs: requiredAtRules, node: ruleNode, result, ruleName, }); }); }; }; /** * Check if rule has a required at-rule ancestor * Traverses up the PostCSS AST tree from the rule to find * if any parent node is an at-rule that matches the required patterns * @param {import('postcss').Rule} ruleNode - The CSS rule to check * @param {Array<string | RegExp>} requiredAtRules - Patterns to match at-rule names against * @returns {boolean} True if rule has required at-rule ancestor, false otherwise */ function hasRequiredAtRuleAncestor(ruleNode, requiredAtRules) { return Boolean( findNodeUpToRoot( ruleNode, (node) => isAtRule(node) && Boolean(matchesStringOrRegExp(node.name, requiredAtRules)), ), ); } rule.primaryOptionArray = true; rule.ruleName = ruleName; rule.messages = messages; rule.meta = meta; export default rule;