UNPKG

stylelint-scss

Version:

A collection of SCSS-specific rules for Stylelint

110 lines (94 loc) 2.99 kB
"use strict"; const { utils } = require("stylelint"); const namespace = require("../../utils/namespace"); const optionsHaveIgnored = require("../../utils/optionsHaveIgnored"); const ruleUrl = require("../../utils/ruleUrl"); const ruleName = namespace("media-feature-value-dollar-variable"); const messages = utils.ruleMessages(ruleName, { rejected: "Unexpected dollar-variable as a media feature value", expected: "Expected a dollar-variable (e.g. $var) to be used as a media feature value" }); const meta = { url: ruleUrl(ruleName) }; function rule(expectation, options) { return (root, result) => { const validOptions = utils.validateOptions( result, ruleName, { actual: expectation, possible: ["always", "never"] }, { actual: options, possible: { ignore: ["keywords"] }, optional: true } ); if (!validOptions) { return; } const valueRegex = /:\s*(\S.+?)(:?\s*)\)/; // In `(max-width: 10px )` find `: 10px )`. // Got to go with that (the global search doesn't remember parens' insides) // and parse it again afterwards to remove trailing junk const valueRegexGlobal = new RegExp(valueRegex.source, "g"); // `$var-name_sth` const variableRegex = /^(\w+\.)?\$[\w-]+$/; // `#{$var-name_sth}` const interpolationVarRegex = /^#{\s*(\w+\.)?\$\w+\s*}$/; // `none`, `dark` const keywordValueRegex = /^[a-z][a-z\d-]*$/; root.walkAtRules("media", atRule => { const found = atRule.params.match(valueRegexGlobal); // If there are no values if (!found || !found.length) { return; } found.forEach(found => { // ... parse `: 10px )` to `10px` const valueParsed = found.match(valueRegex)[1]; // Just a shorthand to stylelint.utils.report() function complain(message) { utils.report({ ruleName, result, node: atRule, word: valueParsed, message }); } // Keyword values, like `none`, should always be fine if keywords are // ignored. if ( keywordValueRegex.test(valueParsed) && optionsHaveIgnored(options, "keywords") ) { return; } // A value should be a single variable // or it should be a single variable inside Sass interpolation if ( expectation === "always" && !( variableRegex.test(valueParsed) || interpolationVarRegex.test(valueParsed) ) ) { complain(messages.expected); } else if (expectation === "never" && valueParsed.includes("$")) { // "Never" means no variables at all (functions allowed) complain(messages.rejected); } }); }); }; } rule.ruleName = ruleName; rule.messages = messages; rule.meta = meta; module.exports = rule;