stylelint
Version:
A mighty CSS linter that helps you avoid errors and enforce conventions.
66 lines (50 loc) • 2.2 kB
JavaScript
import valueParser from 'postcss-value-parser';
import { assert, isRegExp, isString } from './validateTypes.mjs';
/** @typedef {(expression: string, expressionIndex: number, funcNode: valueParser.FunctionNode, parsedValue: valueParser.ParsedValue) => void} Callback */
/**
* Search a CSS string for functions by name.
* For every match, invoke the callback, passing the function's
* "argument(s) string" (whatever is inside the parentheses)
* as an argument.
*
* Callback will be called once for every matching function found,
* with the function's "argument(s) string" and its starting index
* as the arguments.
*
* @param {string} source
* @param {string | RegExp} functionName
* @param {Callback} callback
* @returns {valueParser.ParsedValue}
*/
export default function functionArgumentsSearch(source, functionName, callback) {
const parsedValue = valueParser(source);
return parsedValue.walk((node) => {
if (node.type !== 'function') return;
const { value } = node;
if (isString(functionName) && value !== functionName) return;
if (isRegExp(functionName) && !functionName.test(node.value)) return;
const parenLength = 1; // == '('
const expressionIndex = node.sourceIndex + value.length + parenLength;
let expression;
if (node.value.toLowerCase() === 'url') {
// Re-parse url tokens as component values to support interpolation patterns in non-standard CSS
const [reParsedValue] = valueParser(
// Slice the source value at the opening parenthesis `url(foo) trailing` -> `(foo) trailing`
// When parsed as a list of component values the first node will be a simple block
source.slice(node.sourceIndex + value.length),
).nodes;
assert(reParsedValue);
const endIndex =
expressionIndex +
reParsedValue.sourceEndIndex -
1 -
(source[expressionIndex + reParsedValue.sourceEndIndex - 2] === ')' ? parenLength : 0);
expression = source.slice(expressionIndex, endIndex);
} else {
const endIndex =
node.sourceEndIndex - (source[node.sourceEndIndex - 1] === ')' ? parenLength : 0);
expression = source.slice(expressionIndex, endIndex);
}
callback(expression, expressionIndex, node, parsedValue);
});
}