1
0
mirror of https://github.com/thangisme/notes.git synced 2024-09-28 12:56:03 -04:00
notes/node_modules/stylelint/lib/rules/declaration-empty-line-before/index.js
Patrick Marsceill b7b0d0d7bf
Initial commit
2017-03-09 13:16:08 -05:00

136 lines
3.7 KiB
JavaScript

"use strict"
const blockString = require("../../utils/blockString")
const hasEmptyLine = require("../../utils/hasEmptyLine")
const isCustomProperty = require("../../utils/isCustomProperty")
const isSingleLineString = require("../../utils/isSingleLineString")
const isStandardSyntaxDeclaration = require("../../utils/isStandardSyntaxDeclaration")
const optionsMatches = require("../../utils/optionsMatches")
const report = require("../../utils/report")
const ruleMessages = require("../../utils/ruleMessages")
const validateOptions = require("../../utils/validateOptions")
const ruleName = "declaration-empty-line-before"
const messages = ruleMessages(ruleName, {
expected: "Expected empty line before declaration",
rejected: "Unexpected empty line before declaration",
})
const rule = function (expectation, options) {
return (root, result) => {
const validOptions = validateOptions(result, ruleName, {
actual: expectation,
possible: [
"always",
"never",
],
}, {
actual: options,
possible: {
except: [
"first-nested",
"after-comment",
"after-declaration",
],
ignore: [
"after-comment",
"after-declaration",
"inside-single-line-block",
],
},
optional: true,
})
if (!validOptions) {
return
}
root.walkDecls(decl => {
const prop = decl.prop,
parent = decl.parent
if (!isStandardSyntaxDeclaration(decl)) {
return
}
if (isCustomProperty(prop)) {
return
}
// Optionally ignore the node if a comment precedes it
if (
optionsMatches(options, "ignore", "after-comment")
&& decl.prev()
&& decl.prev().type === "comment"
) {
return
}
// Optionally ignore the node if a declaration precedes it
if (
optionsMatches(options, "ignore", "after-declaration")
&& decl.prev()
&& decl.prev().type === "decl"
) {
return
}
// Optionally ignore nodes inside single-line blocks
if (
optionsMatches(options, "ignore", "inside-single-line-block")
&& isSingleLineString(blockString(parent))
) {
return
}
let expectEmptyLineBefore = expectation === "always"
? true
: false
// Optionally reverse the expectation for the first nested node
if (
optionsMatches(options, "except", "first-nested")
&& decl === parent.first
) {
expectEmptyLineBefore = !expectEmptyLineBefore
}
// Optionally reverse the expectation if a comment precedes this node
if (
optionsMatches(options, "except", "after-comment")
&& decl.prev()
&& decl.prev().type === "comment"
) {
expectEmptyLineBefore = !expectEmptyLineBefore
}
// Optionally reverse the expectation if a declaration precedes this node
if (
optionsMatches(options, "except", "after-declaration")
&& decl.prev()
&& decl.prev().prop
&& isStandardSyntaxDeclaration(decl.prev())
&& !isCustomProperty(decl.prev().prop)
) {
expectEmptyLineBefore = !expectEmptyLineBefore
}
// Check for at least one empty line
const hasEmptyLineBefore = hasEmptyLine(decl.raws.before)
// Return if the expectation is met
if (expectEmptyLineBefore === hasEmptyLineBefore) {
return
}
const message = expectEmptyLineBefore
? messages.expected
: messages.rejected
report({ message, node: decl, result, ruleName })
})
}
}
rule.ruleName = ruleName
rule.messages = messages
module.exports = rule