epiphany/node_modules/stylelint/lib/rules/media-query-no-invalid/index.js
2023-12-09 22:48:07 -08:00

139 lines
3.5 KiB
JavaScript

'use strict';
const { sourceIndices } = require('@csstools/css-parser-algorithms');
const {
isMediaQueryInvalid,
isGeneralEnclosed,
isMediaFeaturePlain,
isMediaFeatureRange,
isMediaFeatureBoolean,
} = require('@csstools/media-query-list-parser');
const atRuleParamIndex = require('../../utils/atRuleParamIndex');
const parseMediaQuery = require('../../utils/parseMediaQuery');
const report = require('../../utils/report');
const ruleMessages = require('../../utils/ruleMessages');
const validateOptions = require('../../utils/validateOptions');
const isCustomMediaQuery = require('../../utils/isCustomMediaQuery');
const { rangeTypeMediaFeatureNames } = require('../../reference/mediaFeatures');
const ruleName = 'media-query-no-invalid';
const messages = ruleMessages(ruleName, {
rejected: (query) => `Unexpected invalid media query "${query}"`,
});
const HAS_MIN_MAX_PREFIX = /^(?:min|max)-/i;
const meta = {
url: 'https://stylelint.io/user-guide/rules/media-query-no-invalid',
};
/** @type {import('stylelint').Rule} */
const rule = (primary) => {
return (root, result) => {
const validOptions = validateOptions(result, ruleName, { actual: primary });
if (!validOptions) {
return;
}
root.walkAtRules(/^media$/i, (atRule) => {
/** @type {Array<{tokens(): Array<import('@csstools/css-tokenizer').CSSToken>}>} */
let invalidNodes = [];
parseMediaQuery(atRule).forEach((mediaQuery) => {
if (isMediaQueryInvalid(mediaQuery)) {
// Queries that fail to parse are invalid.
invalidNodes.push(mediaQuery);
return;
}
mediaQuery.walk(({ node, parent }) => {
// All general enclosed nodes are invalid.
if (isGeneralEnclosed(node)) {
invalidNodes.push(node);
return;
}
// Invalid plain media features.
if (isMediaFeaturePlain(node)) {
const name = node.getName();
if (isCustomMediaQuery(name)) {
// In a plain context, custom media queries are invalid.
invalidNodes.push(parent);
return;
}
return;
}
// Invalid range media features.
if (isMediaFeatureRange(node)) {
const name = node.getName().toLowerCase();
if (isCustomMediaQuery(name)) {
// In a range context, custom media queries are invalid.
invalidNodes.push(parent);
return;
}
if (HAS_MIN_MAX_PREFIX.test(name)) {
// In a range context, min- and max- prefixed feature names are invalid.
invalidNodes.push(parent);
return;
}
if (!rangeTypeMediaFeatureNames.has(name)) {
// In a range context, non-range typed features are invalid.
invalidNodes.push(parent);
return;
}
return;
}
// Invalid boolean media features.
if (isMediaFeatureBoolean(node)) {
const name = node.getName().toLowerCase();
if (HAS_MIN_MAX_PREFIX.test(name)) {
// In a range context, min- and max- prefixed feature names are invalid
invalidNodes.push(parent);
}
}
});
});
if (invalidNodes.length === 0) return;
const atRuleParamIndexValue = atRuleParamIndex(atRule);
invalidNodes.forEach((invalidNode) => {
const [start, end] = sourceIndices(invalidNode);
report({
message: messages.rejected,
messageArgs: [invalidNode.toString()],
index: atRuleParamIndexValue + start,
endIndex: atRuleParamIndexValue + end + 1,
node: atRule,
ruleName,
result,
});
});
});
};
};
rule.ruleName = ruleName;
rule.messages = messages;
rule.meta = meta;
module.exports = rule;