Patrick Marsceill b7b0d0d7bf
Initial commit
2017-03-09 13:16:08 -05:00

105 lines
3.3 KiB
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