2022-02-20 13:39:06 +02:00
|
|
|
const stylelint = require('stylelint');
|
2022-04-19 15:11:48 +02:00
|
|
|
// eslint-disable-next-line import/no-unresolved -- TS BUG
|
2022-02-20 13:39:06 +02:00
|
|
|
const postcss = require('postcss');
|
|
|
|
|
|
|
|
const ruleName = 'plugin/whole-pixel';
|
|
|
|
|
|
|
|
const isString = (s) => typeof s === 'string';
|
|
|
|
|
|
|
|
const messages = stylelint.utils.ruleMessages(ruleName, {
|
|
|
|
expected: (unfixed, fixed) => `Expected "${unfixed}" to be "${fixed}"`,
|
|
|
|
});
|
|
|
|
|
|
|
|
const PX_PER_REM = 16;
|
|
|
|
const unitRegex = /(px|rem)$/;
|
|
|
|
const numberRegex = /^([-0-9.]+)/;
|
|
|
|
|
|
|
|
module.exports = stylelint.createPlugin(ruleName, (primaryOption, secondaryOptionObject, context) => {
|
|
|
|
const secondaryOptions = secondaryOptionObject || {};
|
|
|
|
return (root, result) => {
|
|
|
|
const validOptions = stylelint.utils.validateOptions(
|
|
|
|
result,
|
|
|
|
ruleName,
|
|
|
|
{
|
|
|
|
actual: secondaryOptions,
|
|
|
|
possible: {
|
|
|
|
pxPerRem: (value) => value % 1 === 0,
|
|
|
|
ignoreList: [isString],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!validOptions) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const pxPerRem = Number(secondaryOptions.pxPerRem) || PX_PER_REM;
|
|
|
|
const ignoreList = secondaryOptions.ignoreList || [];
|
|
|
|
|
|
|
|
const isAutoFixing = Boolean(context.fix);
|
|
|
|
|
|
|
|
const isValid = (value, unit) => {
|
|
|
|
if (unit === 'px') return Number.isInteger(value);
|
|
|
|
if (unit === 'rem') return Number.isInteger(value * pxPerRem);
|
2022-04-19 15:11:48 +02:00
|
|
|
return false;
|
2022-02-20 13:39:06 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
const suggestFix = (value, unit) => {
|
|
|
|
if (unit === 'px') return `${Math.round(value)}px`;
|
|
|
|
if (unit === 'rem') return `${Math.round(value * pxPerRem) / pxPerRem}rem`;
|
2022-04-19 15:11:48 +02:00
|
|
|
return undefined;
|
2022-02-20 13:39:06 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
const handleValue = (decl, value) => {
|
|
|
|
if (!unitRegex.test(value)) return;
|
|
|
|
const matched = value.match(numberRegex);
|
|
|
|
if (!matched) return;
|
|
|
|
const valueNumberString = matched[0];
|
|
|
|
const valueNumber = parseFloat(valueNumberString);
|
|
|
|
const unit = value.replace(valueNumberString, '');
|
|
|
|
|
|
|
|
if (isValid(valueNumber, unit)) return;
|
|
|
|
const suggestedValue = suggestFix(valueNumber, unit);
|
|
|
|
if (isAutoFixing) {
|
|
|
|
decl.value = decl.value.replace(value, suggestedValue);
|
|
|
|
} else {
|
|
|
|
stylelint.utils.report({
|
|
|
|
ruleName,
|
|
|
|
result,
|
|
|
|
node: decl,
|
|
|
|
message: messages.expected(value, suggestedValue),
|
|
|
|
word: value,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-04-19 15:11:48 +02:00
|
|
|
root.walkDecls((decl) => {
|
2022-02-20 13:39:06 +02:00
|
|
|
if (!decl.value || ignoreList.includes(decl.prop)) return;
|
|
|
|
const values = postcss.list.space(decl.value);
|
|
|
|
if (!values?.length) return;
|
|
|
|
values.forEach((value) => handleValue(decl, value));
|
|
|
|
});
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
module.exports.ruleName = ruleName;
|
|
|
|
module.exports.messages = messages;
|