epiphany/node_modules/stylelint/lib/rules/hue-degree-notation/index.js
2023-12-09 22:48:07 -08:00

151 lines
3.5 KiB
JavaScript

'use strict';
const valueParser = require('postcss-value-parser');
const declarationValueIndex = require('../../utils/declarationValueIndex');
const getDeclarationValue = require('../../utils/getDeclarationValue');
const isStandardSyntaxValue = require('../../utils/isStandardSyntaxValue');
const report = require('../../utils/report');
const ruleMessages = require('../../utils/ruleMessages');
const setDeclarationValue = require('../../utils/setDeclarationValue');
const validateOptions = require('../../utils/validateOptions');
const ruleName = 'hue-degree-notation';
const messages = ruleMessages(ruleName, {
expected: (unfixed, fixed) => `Expected "${unfixed}" to be "${fixed}"`,
});
const meta = {
url: 'https://stylelint.io/user-guide/rules/hue-degree-notation',
fixable: true,
};
const HUE_FIRST_ARG_FUNCS = ['hsl', 'hsla', 'hwb'];
const HUE_THIRD_ARG_FUNCS = ['lch', 'oklch'];
const HUE_FUNCS = new Set([...HUE_FIRST_ARG_FUNCS, ...HUE_THIRD_ARG_FUNCS]);
const HAS_HUE_COLOR_FUNC = new RegExp(`\\b(?:${[...HUE_FUNCS].join('|')})\\(`, 'i');
/** @type {import('stylelint').Rule} */
const rule = (primary, _secondaryOptions, context) => {
return (root, result) => {
const validOptions = validateOptions(result, ruleName, {
actual: primary,
possible: ['angle', 'number'],
});
if (!validOptions) return;
root.walkDecls((decl) => {
if (!HAS_HUE_COLOR_FUNC.test(decl.value)) return;
let needsFix = false;
const parsedValue = valueParser(getDeclarationValue(decl));
parsedValue.walk((node) => {
if (node.type !== 'function') return;
if (!HUE_FUNCS.has(node.value.toLowerCase())) return;
const hue = findHue(node);
if (!hue) return;
const { value } = hue;
if (!isStandardSyntaxValue(value)) return;
if (!isDegree(value) && !isNumber(value)) return;
if (primary === 'angle' && isDegree(value)) return;
if (primary === 'number' && isNumber(value)) return;
const fixed = primary === 'angle' ? asDegree(value) : asNumber(value);
const unfixed = value;
if (context.fix) {
hue.value = fixed;
needsFix = true;
return;
}
const valueIndex = declarationValueIndex(decl);
report({
message: messages.expected,
messageArgs: [unfixed, fixed],
node: decl,
index: valueIndex + hue.sourceIndex,
endIndex: valueIndex + hue.sourceEndIndex,
result,
ruleName,
});
});
if (needsFix) {
setDeclarationValue(decl, parsedValue.toString());
}
});
};
};
/**
* @param {string} value
*/
function asDegree(value) {
return `${value}deg`;
}
/**
* @param {string} value
*/
function asNumber(value) {
const dimension = valueParser.unit(value);
if (dimension) return dimension.number;
throw new TypeError(`The "${value}" value must have a unit`);
}
/**
* @param {import('postcss-value-parser').FunctionNode} node
*/
function findHue(node) {
const args = node.nodes.filter(({ type }) => type === 'word' || type === 'function');
const value = node.value.toLowerCase();
if (HUE_FIRST_ARG_FUNCS.includes(value)) {
return args[0];
}
if (HUE_THIRD_ARG_FUNCS.includes(value)) {
return args[2];
}
return undefined;
}
/**
* @param {string} value
*/
function isDegree(value) {
const dimension = valueParser.unit(value);
return dimension && dimension.unit.toLowerCase() === 'deg';
}
/**
* @param {string} value
*/
function isNumber(value) {
const dimension = valueParser.unit(value);
return dimension && dimension.unit === '';
}
rule.ruleName = ruleName;
rule.messages = messages;
rule.meta = meta;
module.exports = rule;