stylelint
Version:
A mighty, modern CSS linter.
121 lines (100 loc) • 3.54 kB
JavaScript
"use strict";
const declarationValueIndex = require("../../utils/declarationValueIndex");
const isSingleLineString = require("../../utils/isSingleLineString");
const isStandardSyntaxFunction = require("../../utils/isStandardSyntaxFunction");
const report = require("../../utils/report");
const ruleMessages = require("../../utils/ruleMessages");
const validateOptions = require("../../utils/validateOptions");
const valueParser = require("postcss-value-parser");
const ruleName = "function-parentheses-newline-inside";
const messages = ruleMessages(ruleName, {
expectedOpening: 'Expected newline after "("',
expectedClosing: 'Expected newline before ")"',
expectedOpeningMultiLine:
'Expected newline after "(" in a multi-line function',
rejectedOpeningMultiLine:
'Unexpected whitespace after "(" in a multi-line function',
expectedClosingMultiLine:
'Expected newline before ")" in a multi-line function',
rejectedClosingMultiLine:
'Unexpected whitespace before ")" in a multi-line function'
});
const rule = function(expectation) {
return (root, result) => {
const validOptions = validateOptions(result, ruleName, {
actual: expectation,
possible: ["always", "always-multi-line", "never-multi-line"]
});
if (!validOptions) {
return;
}
root.walkDecls(decl => {
if (decl.value.indexOf("(") === -1) {
return;
}
valueParser(decl.value).walk(valueNode => {
if (valueNode.type !== "function") {
return;
}
if (!isStandardSyntaxFunction(valueNode)) {
return;
}
const functionString = valueParser.stringify(valueNode);
const isMultiLine = !isSingleLineString(functionString);
function containsNewline(str) {
return str.indexOf("\n") !== -1;
}
// Check opening ...
const openingIndex = valueNode.sourceIndex + valueNode.value.length + 1;
if (expectation === "always" && !containsNewline(valueNode.before)) {
complain(messages.expectedOpening, openingIndex);
}
if (
isMultiLine &&
expectation === "always-multi-line" &&
!containsNewline(valueNode.before)
) {
complain(messages.expectedOpeningMultiLine, openingIndex);
}
if (
isMultiLine &&
expectation === "never-multi-line" &&
valueNode.before !== ""
) {
complain(messages.rejectedOpeningMultiLine, openingIndex);
}
// Check closing ...
const closingIndex = valueNode.sourceIndex + functionString.length - 2;
if (expectation === "always" && !containsNewline(valueNode.after)) {
complain(messages.expectedClosing, closingIndex);
}
if (
isMultiLine &&
expectation === "always-multi-line" &&
!containsNewline(valueNode.after)
) {
complain(messages.expectedClosingMultiLine, closingIndex);
}
if (
isMultiLine &&
expectation === "never-multi-line" &&
valueNode.after !== ""
) {
complain(messages.rejectedClosingMultiLine, closingIndex);
}
});
function complain(message, offset) {
report({
ruleName,
result,
message,
node: decl,
index: declarationValueIndex(decl) + offset
});
}
});
};
};
rule.ruleName = ruleName;
rule.messages = messages;
module.exports = rule;