UNPKG

@stylistic/stylelint-plugin

Version:
134 lines (108 loc) 3.93 kB
import stylelint from "stylelint" import { addNamespace } from "../../utils/addNamespace/index.js" import { blockString } from "../../utils/blockString/index.js" import { getRuleDocUrl } from "../../utils/getRuleDocUrl/index.js" import { hasBlock } from "../../utils/hasBlock/index.js" import { optionsMatches } from "../../utils/optionsMatches/index.js" import { rawNodeString } from "../../utils/rawNodeString/index.js" import { isString } from "../../utils/validateTypes/index.js" import { whitespaceChecker } from "../../utils/whitespaceChecker/index.js" let { utils: { report, ruleMessages, validateOptions } } = stylelint let shortName = `block-closing-brace-newline-after` export let ruleName = addNamespace(shortName) export let messages = ruleMessages(ruleName, { expectedAfter: () => `Expected newline after "}"`, expectedAfterSingleLine: () => `Expected newline after "}" of a single-line block`, rejectedAfterSingleLine: () => `Unexpected whitespace after "}" of a single-line block`, expectedAfterMultiLine: () => `Expected newline after "}" of a multi-line block`, rejectedAfterMultiLine: () => `Unexpected whitespace after "}" of a multi-line block`, }) export let meta = { url: getRuleDocUrl(shortName), fixable: true, } /** * Requires a newline or disallows whitespace after the closing brace of blocks. * @type {import('stylelint').Rule} */ function rule (primary, secondaryOptions, context) { let checker = whitespaceChecker(`newline`, primary, messages) return (root, result) => { let validOptions = validateOptions( result, ruleName, { actual: primary, possible: [ `always`, `always-single-line`, `never-single-line`, `always-multi-line`, `never-multi-line`, ], }, { actual: secondaryOptions, possible: { ignoreAtRules: [isString], }, optional: true, }, ) if (!validOptions) return // Check both kinds of statements: rules and at-rules root.walkRules(check) root.walkAtRules(check) /** * Checks a statement for closing brace newline after violations. * @param {import('postcss').Rule | import('postcss').AtRule} statement - The rule or at-rule to check. */ function check (statement) { if (!hasBlock(statement)) return if (statement.type === `atrule` && optionsMatches(secondaryOptions, `ignoreAtRules`, statement.name)) return let nextNode = statement.next() if (!nextNode) return // Allow an end-of-line comment x spaces after the brace let nextNodeIsSingleLineComment = nextNode.type === `comment` && !(/[^ ]/u).test(nextNode.raws.before || ``) && !nextNode.toString().includes(`\n`) let nodeToCheck = nextNodeIsSingleLineComment ? nextNode.next() : nextNode if (!nodeToCheck) return let reportIndex = statement.toString().length let source = rawNodeString(nodeToCheck) // Skip a semicolon at the beginning, if any if (source && source.startsWith(`;`)) { source = source.slice(1) reportIndex += 1 } // Only check one after, because there might be other // spaces handled by the indentation rule checker.afterOneOnly({ source, index: -1, lineCheckStr: blockString(statement), err: (msg) => { report({ message: msg, node: statement, index: reportIndex, endIndex: reportIndex, result, ruleName, fix () { let nodeToCheckRaws = nodeToCheck.raws if (typeof nodeToCheckRaws.before !== `string`) return if (primary.startsWith(`always`)) { let index = nodeToCheckRaws.before.search(/\r?\n/u) nodeToCheckRaws.before = index >= 0 ? nodeToCheckRaws.before.slice(index) : context.newline + nodeToCheckRaws.before } else if (primary.startsWith(`never`)) nodeToCheckRaws.before = `` }, }) }, }) } } } rule.ruleName = ruleName rule.messages = messages rule.meta = meta export default rule