UNPKG

@stylistic/stylelint-plugin

Version:
106 lines (82 loc) 3.21 kB
import stylelint from "stylelint" import { addNamespace } from "../../utils/addNamespace/index.js" import { declarationValueIndex } from "../../utils/declarationValueIndex/index.js" import { getDeclarationValue } from "../../utils/getDeclarationValue/index.js" import { getRuleDocUrl } from "../../utils/getRuleDocUrl/index.js" import { setDeclarationValue } from "../../utils/setDeclarationValue/index.js" import { valueListCommaWhitespaceChecker } from "../../utils/valueListCommaWhitespaceChecker/index.js" import { whitespaceChecker } from "../../utils/whitespaceChecker/index.js" let { utils: { ruleMessages, validateOptions } } = stylelint let shortName = `value-list-comma-newline-after` export let ruleName = addNamespace(shortName) export let messages = ruleMessages(ruleName, { expectedAfter: () => `Expected newline after ","`, expectedAfterMultiLine: () => `Expected newline after "," in a multi-line list`, rejectedAfterMultiLine: () => `Unexpected whitespace after "," in a multi-line list`, }) export let meta = { url: getRuleDocUrl(shortName), fixable: true, } /** @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-multi-line`, `never-multi-line`], }) if (!validOptions) { return } /** @type {Map<import('postcss').Declaration, number[]> | undefined} */ let fixData valueListCommaWhitespaceChecker({ root, result, locationChecker: checker.afterOneOnly, checkedRuleName: ruleName, fix: (declNode, index) => { let valueIndex = declarationValueIndex(declNode) if (index <= valueIndex) { return false } fixData = fixData || (new Map) let commaIndices = fixData.get(declNode) || [] commaIndices.push(index) fixData.set(declNode, commaIndices) return true }, determineIndex: (declString, match) => { let nextChars = declString.substring(match.endIndex, declString.length) // If there's a // comment, that means there has to be a newline // ending the comment so we're fine if ((/^[ \t]*\/\//).test(nextChars)) { return false } // If there are spaces and then a comment begins, look for the newline return (/^[ \t]*\/\*/).test(nextChars) ? declString.indexOf(`*/`, match.endIndex) + 1 : match.startIndex }, }) if (fixData) { for (let [decl, commaIndices] of fixData.entries()) { for (let index of commaIndices.sort((a, b) => a - b).reverse()) { let value = getDeclarationValue(decl) let valueIndex = index - declarationValueIndex(decl) let beforeValue = value.slice(0, valueIndex + 1) let afterValue = value.slice(valueIndex + 1) if (primary.startsWith(`always`)) { afterValue = context.newline + afterValue } else if (primary.startsWith(`never-multi-line`)) { afterValue = afterValue.replace(/^\s*/, ``) } setDeclarationValue(decl, beforeValue + afterValue) } } } } } rule.ruleName = ruleName rule.messages = messages rule.meta = meta export default rule