stylelint
Version:
A mighty, modern CSS linter.
60 lines (50 loc) • 1.53 kB
JavaScript
const { isRoot, isAtRule, isRule } = require('./typeGuards');
/** @typedef {import('postcss').Root} Root */
/** @typedef {import('postcss').Root} Document */
/** @typedef {import('postcss').Node} PostcssNode */
/** @typedef {import('postcss').ContainerBase} PostcssContainerNode */
/**
* @param {PostcssNode} node
* @returns {node is PostcssContainerNode}
*/
function isContainerNode(node) {
return isRule(node) || isAtRule(node) || isRoot(node);
}
/**
* In order to accommodate nested blocks (postcss-nested),
* we need to run a shallow loop (instead of eachDecl() or eachRule(),
* which loop recursively) and allow each nested block to accumulate
* its own list of properties -- so that a property in a nested rule
* does not conflict with the same property in the parent rule
* executes a provided function once for each declaration block.
*
* @param {Root | Document} root - root element of file.
* @param {function} cb - Function to execute for each declaration block
*
* @returns {void}
*/
module.exports = function (root, cb) {
/**
* @param {PostcssNode} statement
*
* @returns {void}
*/
function each(statement) {
if (!isContainerNode(statement)) return;
if (statement.nodes && statement.nodes.length) {
/** @type {PostcssNode[]} */
const decls = [];
statement.nodes.forEach((node) => {
if (node.type === 'decl') {
decls.push(node);
}
each(node);
});
if (decls.length) {
cb(decls.forEach.bind(decls));
}
}
}
each(root);
};
;