stylelint
Version:
A mighty, modern CSS linter.
141 lines (122 loc) • 3.81 kB
JavaScript
"use strict";
const _ = require("lodash");
const isCustomProperty = require("../../utils/isCustomProperty");
const isStandardSyntaxProperty = require("../../utils/isStandardSyntaxProperty");
const optionsMatches = require("../../utils/optionsMatches");
const report = require("../../utils/report");
const ruleMessages = require("../../utils/ruleMessages");
const validateOptions = require("../../utils/validateOptions");
const ruleName = "declaration-block-no-duplicate-properties";
const messages = ruleMessages(ruleName, {
rejected: property => `Unexpected duplicate "${property}"`
});
const rule = function(on, options) {
return (root, result) => {
const validOptions = validateOptions(
result,
ruleName,
{ actual: on },
{
actual: options,
possible: {
ignore: [
"consecutive-duplicates",
"consecutive-duplicates-with-different-values"
],
ignoreProperties: [_.isString]
},
optional: true
}
);
if (!validOptions) {
return;
}
// 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
root.each(node => {
if (node.type === "rule" || node.type === "atrule") {
checkRulesInNode(node);
}
});
function checkRulesInNode(node) {
const decls = [];
const values = [];
node.each(child => {
if (child.nodes && child.nodes.length) {
checkRulesInNode(child);
}
if (child.type !== "decl") {
return;
}
const prop = child.prop;
const value = child.value;
if (!isStandardSyntaxProperty(prop)) {
return;
}
if (isCustomProperty(prop)) {
return;
}
// Return early if the property is to be ignored
if (optionsMatches(options, "ignoreProperties", prop)) {
return;
}
// Ignore the src property as commonly duplicated in at-fontface
if (prop.toLowerCase() === "src") {
return;
}
const indexDuplicate = decls.indexOf(prop.toLowerCase());
if (indexDuplicate !== -1) {
if (
optionsMatches(
options,
"ignore",
"consecutive-duplicates-with-different-values"
)
) {
// if duplicates are not consecutive
if (indexDuplicate !== decls.length - 1) {
report({
message: messages.rejected(prop),
node: child,
result,
ruleName
});
return;
}
// if values of consecutive duplicates are equal
if (value === values[indexDuplicate]) {
report({
message: messages.rejected(value),
node: child,
result,
ruleName
});
return;
}
return;
}
if (
optionsMatches(options, "ignore", "consecutive-duplicates") &&
indexDuplicate === decls.length - 1
) {
return;
}
report({
message: messages.rejected(prop),
node: child,
result,
ruleName
});
}
decls.push(prop.toLowerCase());
values.push(value.toLowerCase());
});
}
};
};
rule.ruleName = ruleName;
rule.messages = messages;
module.exports = rule;