@stylistic/stylelint-plugin
Version:
A collection of stylistic/formatting Stylelint rules
96 lines (81 loc) • 2.57 kB
JavaScript
import stylelint from "stylelint"
import styleSearch from "style-search"
import { isStandardSyntaxRule } from "../isStandardSyntaxRule/index.js"
import { parseSelector } from "../parseSelector/index.js"
let { utils: { report } } = stylelint
/**
* @param {{
* root: import('postcss').Root,
* result: import('stylelint').PostcssResult,
* locationChecker: (opts: { source: string, index: number, err: (msg: string) => void }) => void,
* checkedRuleName: string,
* checkBeforeOperator: boolean,
* fix: ((attributeNode: import('postcss-selector-parser').Attribute) => boolean),
* }} options
* @returns {void}
*/
export function selectorAttributeOperatorSpaceChecker (options) {
options.root.walkRules((rule) => {
if (!isStandardSyntaxRule(rule)) {
return
}
if (!rule.selector.includes(`[`) || !rule.selector.includes(`=`)) {
return
}
let hasFixed = false
let selector = rule.raws.selector ? rule.raws.selector.raw : rule.selector
let fixedSelector = parseSelector(selector, options.result, rule, (selectorTree) => {
selectorTree.walkAttributes((attributeNode) => {
let operator = attributeNode.operator
if (!operator) {
return
}
let attributeNodeString = attributeNode.toString()
styleSearch({ source: attributeNodeString, target: operator }, (match) => {
let index = options.checkBeforeOperator ? match.startIndex : match.endIndex - 1
checkOperator(attributeNodeString, index, rule, attributeNode, operator)
})
})
})
if (hasFixed && fixedSelector) {
if (!rule.raws.selector) {
rule.selector = fixedSelector
} else {
rule.raws.selector.raw = fixedSelector
}
}
/**
* @param {string} source
* @param {number} index
* @param {import('postcss').Node} node
* @param {import('postcss-selector-parser').Attribute} attributeNode
* @param {string} operator
*/
function checkOperator (source, index, node, attributeNode, operator) {
options.locationChecker({
source,
index,
err: (msg) => {
const problemIndex = attributeNode.sourceIndex + index
report({
message: msg.replace(
options.checkBeforeOperator ? operator.charAt(0) : operator.charAt(operator.length - 1),
operator,
),
node,
index: problemIndex,
endIndex: problemIndex,
result: options.result,
ruleName: options.checkedRuleName,
fix: options.fix
? () => {
hasFixed = true
return options.fix(attributeNode)
}
: undefined,
})
},
})
}
})
}