UNPKG

stylelint-scss

Version:

A collection of SCSS-specific rules for Stylelint

101 lines (80 loc) 2.92 kB
import resolveNestedSelector from "postcss-resolve-nested-selector"; import stylelint from "stylelint"; import { isRegExp, isString } from "../../utils/validateTypes.js"; import hasInterpolatingAmpersand from "../../utils/hasInterpolatingAmpersand.js"; import isStandardRule from "../../utils/isStandardRule.js"; import isStandardSelector from "../../utils/isStandardSelector.js"; import parseSelector from "../../utils/parseSelector.js"; import namespace from "../../utils/namespace.js"; import ruleUrl from "../../utils/ruleUrl.js"; const { utils } = stylelint; const ruleName = namespace("percent-placeholder-pattern"); const messages = utils.ruleMessages(ruleName, { expected: (placeholderName, pattern) => `Expected "${placeholderName}" to match pattern "${pattern}"` }); const meta = { url: ruleUrl(ruleName) }; function rule(pattern) { return (root, result) => { const validOptions = utils.validateOptions(result, ruleName, { actual: pattern, possible: [isRegExp, isString] }); if (!validOptions) { return; } const regexpPattern = isString(pattern) ? new RegExp(pattern) : pattern; // Checking placeholder definitions (looking among regular rules) root.walkRules(rule => { const { selector } = rule; // Just a shorthand for calling `parseSelector` function parse(selector) { parseSelector(selector, result, rule, s => checkSelector(s, rule)); } // If it's a custom prop or a less mixin if (!isStandardRule(rule)) { return; } // If the selector has interpolation if (!isStandardSelector(selector)) { return; } // Nested selectors are processed in steps, as nesting levels are resolved. // Here we skip processing intermediate parts of selectors (to process only fully resolved selectors) // if (rule.nodes.some(node => node.type === "rule" || node.type === "atrule")) { return } // Only resolve selectors that have an interpolating "&" if (hasInterpolatingAmpersand(selector)) { resolveNestedSelector(selector, rule).forEach(parse); } else { parse(selector); } }); function checkSelector(fullSelector, rule) { // postcss-selector-parser gives %placeholders' nodes a "tag" type fullSelector.walkTags(compoundSelector => { const { value } = compoundSelector; if (value[0] !== "%") { return; } const placeholderName = value.slice(1); if (regexpPattern.test(placeholderName)) { return; } utils.report({ result, ruleName, message: messages.expected, messageArgs: [placeholderName, pattern], node: rule, word: value }); }); } }; } rule.ruleName = ruleName; rule.messages = messages; rule.meta = meta; export default rule;