UNPKG

stylelint-scss

Version:

A collection of SCSS-specific rules for Stylelint

143 lines (118 loc) 3.64 kB
"use strict"; const valueParser = require("postcss-value-parser"); const { utils } = require("stylelint"); const declarationValueIndex = require("../../utils/declarationValueIndex"); const isNativeCssFunction = require("../../utils/isNativeCssFunction"); const { isString } = require("../../utils/validateTypes"); const namespace = require("../../utils/namespace"); const optionsHaveIgnored = require("../../utils/optionsHaveIgnored"); const { parseFunctionArguments } = require("../../utils/parseFunctionArguments"); const ruleUrl = require("../../utils/ruleUrl"); const ruleName = namespace("at-function-named-arguments"); const messages = utils.ruleMessages(ruleName, { expected: "Expected a named parameter to be used in function call", rejected: "Unexpected a named parameter in function call" }); const meta = { url: ruleUrl(ruleName) }; const isScssVarRegExp = /^\$\S*/; function rule(expectation, options) { return (root, result) => { const validOptions = utils.validateOptions( result, ruleName, { actual: expectation, possible: ["always", "never"] }, { actual: options, possible: { ignore: ["single-argument"], ignoreFunctions: [isString] }, optional: true } ); if (!validOptions) { return; } const shouldIgnoreSingleArgument = optionsHaveIgnored( options, "single-argument" ); root.walkDecls(decl => { const declValueIndex = declarationValueIndex(decl); valueParser(decl.value).walk(node => { if ( node.type !== "function" || isNativeCssFunction(node.value) || node.value === "" ) { return; } const hasFuncIgnored = options && options.ignoreFunctions && options.ignoreFunctions.some(f => { const interpolationRegex = /^#{/; const funcName = node.value.replace(interpolationRegex, ""); const isRegex = /^\/.*\//.test(f); if (!isRegex) { return f === funcName; } const parts = f.split("/"); return new RegExp(parts[1], parts[2] || "").test(funcName); }); if (hasFuncIgnored) { return; } const args = parseFunctionArguments(valueParser.stringify(node)); const isSingleArgument = args.length === 1; if (isSingleArgument && shouldIgnoreSingleArgument) { return; } const baseIndex = declValueIndex + node.sourceIndex; args.forEach(({ key, index, endIndex }) => { switch (expectation) { case "never": { if (!key) { return; } utils.report({ message: messages.rejected, node: decl, result, ruleName, index: baseIndex + index, endIndex: baseIndex + endIndex }); break; } case "always": { if (key && isScssVarRegExp.test(key)) { return; } utils.report({ message: messages.expected, node: decl, result, ruleName, index: baseIndex + index, endIndex: baseIndex + endIndex }); break; } } }); }); }); }; } rule.ruleName = ruleName; rule.messages = messages; rule.meta = meta; module.exports = rule;