'use strict'; const isStandardSyntaxRule = require('../utils/isStandardSyntaxRule'); const parseSelector = require('../utils/parseSelector'); const report = require('../utils/report'); const styleSearch = require('style-search'); /** * @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) | null, * }} options * @returns {void} */ module.exports = function selectorAttributeOperatorSpaceChecker(options) { options.root.walkRules((rule) => { if (!isStandardSyntaxRule(rule)) { return; } if (!rule.selector.includes('[') || !rule.selector.includes('=')) { return; } let hasFixed = false; const selector = rule.raws.selector ? rule.raws.selector.raw : rule.selector; const fixedSelector = parseSelector(selector, options.result, rule, (selectorTree) => { selectorTree.walkAttributes((attributeNode) => { const operator = attributeNode.operator; if (!operator) { return; } const attributeNodeString = attributeNode.toString(); styleSearch({ source: attributeNodeString, target: operator }, (match) => { const 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) => { if (options.fix && options.fix(attributeNode)) { hasFixed = true; return; } report({ message: msg.replace( options.checkBeforeOperator ? operator[0] : operator[operator.length - 1], operator, ), node, index: attributeNode.sourceIndex + index, result: options.result, ruleName: options.checkedRuleName, }); }, }); } }); };