You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
284 lines
7.8 KiB
284 lines
7.8 KiB
1 month ago
|
'use strict';
|
||
|
|
||
|
/**
|
||
|
* @typedef {import('css-tree').Rule} CsstreeRule
|
||
|
* @typedef {import('./types').Specificity} Specificity
|
||
|
* @typedef {import('./types').Stylesheet} Stylesheet
|
||
|
* @typedef {import('./types').StylesheetRule} StylesheetRule
|
||
|
* @typedef {import('./types').StylesheetDeclaration} StylesheetDeclaration
|
||
|
* @typedef {import('./types').ComputedStyles} ComputedStyles
|
||
|
* @typedef {import('./types').XastRoot} XastRoot
|
||
|
* @typedef {import('./types').XastElement} XastElement
|
||
|
* @typedef {import('./types').XastParent} XastParent
|
||
|
* @typedef {import('./types').XastChild} XastChild
|
||
|
*/
|
||
|
|
||
|
const stable = require('stable');
|
||
|
const csstree = require('css-tree');
|
||
|
// @ts-ignore not defined in @types/csso
|
||
|
const specificity = require('csso/lib/restructure/prepare/specificity');
|
||
|
const { visit, matches } = require('./xast.js');
|
||
|
const {
|
||
|
attrsGroups,
|
||
|
inheritableAttrs,
|
||
|
presentationNonInheritableGroupAttrs,
|
||
|
} = require('../plugins/_collections.js');
|
||
|
|
||
|
// @ts-ignore not defined in @types/csstree
|
||
|
const csstreeWalkSkip = csstree.walk.skip;
|
||
|
|
||
|
/**
|
||
|
* @type {(ruleNode: CsstreeRule, dynamic: boolean) => StylesheetRule}
|
||
|
*/
|
||
|
const parseRule = (ruleNode, dynamic) => {
|
||
|
let selectors;
|
||
|
let selectorsSpecificity;
|
||
|
/**
|
||
|
* @type {Array<StylesheetDeclaration>}
|
||
|
*/
|
||
|
const declarations = [];
|
||
|
csstree.walk(ruleNode, (cssNode) => {
|
||
|
if (cssNode.type === 'SelectorList') {
|
||
|
// compute specificity from original node to consider pseudo classes
|
||
|
selectorsSpecificity = specificity(cssNode);
|
||
|
const newSelectorsNode = csstree.clone(cssNode);
|
||
|
csstree.walk(newSelectorsNode, (pseudoClassNode, item, list) => {
|
||
|
if (pseudoClassNode.type === 'PseudoClassSelector') {
|
||
|
dynamic = true;
|
||
|
list.remove(item);
|
||
|
}
|
||
|
});
|
||
|
selectors = csstree.generate(newSelectorsNode);
|
||
|
return csstreeWalkSkip;
|
||
|
}
|
||
|
if (cssNode.type === 'Declaration') {
|
||
|
declarations.push({
|
||
|
name: cssNode.property,
|
||
|
value: csstree.generate(cssNode.value),
|
||
|
important: cssNode.important === true,
|
||
|
});
|
||
|
return csstreeWalkSkip;
|
||
|
}
|
||
|
});
|
||
|
if (selectors == null || selectorsSpecificity == null) {
|
||
|
throw Error('assert');
|
||
|
}
|
||
|
return {
|
||
|
dynamic,
|
||
|
selectors,
|
||
|
specificity: selectorsSpecificity,
|
||
|
declarations,
|
||
|
};
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* @type {(css: string, dynamic: boolean) => Array<StylesheetRule>}
|
||
|
*/
|
||
|
const parseStylesheet = (css, dynamic) => {
|
||
|
/**
|
||
|
* @type {Array<StylesheetRule>}
|
||
|
*/
|
||
|
const rules = [];
|
||
|
const ast = csstree.parse(css, {
|
||
|
parseValue: false,
|
||
|
parseAtrulePrelude: false,
|
||
|
});
|
||
|
csstree.walk(ast, (cssNode) => {
|
||
|
if (cssNode.type === 'Rule') {
|
||
|
rules.push(parseRule(cssNode, dynamic || false));
|
||
|
return csstreeWalkSkip;
|
||
|
}
|
||
|
if (cssNode.type === 'Atrule') {
|
||
|
if (cssNode.name === 'keyframes') {
|
||
|
return csstreeWalkSkip;
|
||
|
}
|
||
|
csstree.walk(cssNode, (ruleNode) => {
|
||
|
if (ruleNode.type === 'Rule') {
|
||
|
rules.push(parseRule(ruleNode, dynamic || true));
|
||
|
return csstreeWalkSkip;
|
||
|
}
|
||
|
});
|
||
|
return csstreeWalkSkip;
|
||
|
}
|
||
|
});
|
||
|
return rules;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* @type {(css: string) => Array<StylesheetDeclaration>}
|
||
|
*/
|
||
|
const parseStyleDeclarations = (css) => {
|
||
|
/**
|
||
|
* @type {Array<StylesheetDeclaration>}
|
||
|
*/
|
||
|
const declarations = [];
|
||
|
const ast = csstree.parse(css, {
|
||
|
context: 'declarationList',
|
||
|
parseValue: false,
|
||
|
});
|
||
|
csstree.walk(ast, (cssNode) => {
|
||
|
if (cssNode.type === 'Declaration') {
|
||
|
declarations.push({
|
||
|
name: cssNode.property,
|
||
|
value: csstree.generate(cssNode.value),
|
||
|
important: cssNode.important === true,
|
||
|
});
|
||
|
}
|
||
|
});
|
||
|
return declarations;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* @type {(stylesheet: Stylesheet, node: XastElement) => ComputedStyles}
|
||
|
*/
|
||
|
const computeOwnStyle = (stylesheet, node) => {
|
||
|
/**
|
||
|
* @type {ComputedStyles}
|
||
|
*/
|
||
|
const computedStyle = {};
|
||
|
const importantStyles = new Map();
|
||
|
|
||
|
// collect attributes
|
||
|
for (const [name, value] of Object.entries(node.attributes)) {
|
||
|
if (attrsGroups.presentation.includes(name)) {
|
||
|
computedStyle[name] = { type: 'static', inherited: false, value };
|
||
|
importantStyles.set(name, false);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// collect matching rules
|
||
|
for (const { selectors, declarations, dynamic } of stylesheet.rules) {
|
||
|
if (matches(node, selectors)) {
|
||
|
for (const { name, value, important } of declarations) {
|
||
|
const computed = computedStyle[name];
|
||
|
if (computed && computed.type === 'dynamic') {
|
||
|
continue;
|
||
|
}
|
||
|
if (dynamic) {
|
||
|
computedStyle[name] = { type: 'dynamic', inherited: false };
|
||
|
continue;
|
||
|
}
|
||
|
if (
|
||
|
computed == null ||
|
||
|
important === true ||
|
||
|
importantStyles.get(name) === false
|
||
|
) {
|
||
|
computedStyle[name] = { type: 'static', inherited: false, value };
|
||
|
importantStyles.set(name, important);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// collect inline styles
|
||
|
const styleDeclarations =
|
||
|
node.attributes.style == null
|
||
|
? []
|
||
|
: parseStyleDeclarations(node.attributes.style);
|
||
|
for (const { name, value, important } of styleDeclarations) {
|
||
|
const computed = computedStyle[name];
|
||
|
if (computed && computed.type === 'dynamic') {
|
||
|
continue;
|
||
|
}
|
||
|
if (
|
||
|
computed == null ||
|
||
|
important === true ||
|
||
|
importantStyles.get(name) === false
|
||
|
) {
|
||
|
computedStyle[name] = { type: 'static', inherited: false, value };
|
||
|
importantStyles.set(name, important);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return computedStyle;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Compares two selector specificities.
|
||
|
* extracted from https://github.com/keeganstreet/specificity/blob/master/specificity.js#L211
|
||
|
*
|
||
|
* @type {(a: Specificity, b: Specificity) => number}
|
||
|
*/
|
||
|
const compareSpecificity = (a, b) => {
|
||
|
for (var i = 0; i < 4; i += 1) {
|
||
|
if (a[i] < b[i]) {
|
||
|
return -1;
|
||
|
} else if (a[i] > b[i]) {
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* @type {(root: XastRoot) => Stylesheet}
|
||
|
*/
|
||
|
const collectStylesheet = (root) => {
|
||
|
/**
|
||
|
* @type {Array<StylesheetRule>}
|
||
|
*/
|
||
|
const rules = [];
|
||
|
/**
|
||
|
* @type {Map<XastElement, XastParent>}
|
||
|
*/
|
||
|
const parents = new Map();
|
||
|
visit(root, {
|
||
|
element: {
|
||
|
enter: (node, parentNode) => {
|
||
|
// store parents
|
||
|
parents.set(node, parentNode);
|
||
|
// find and parse all styles
|
||
|
if (node.name === 'style') {
|
||
|
const dynamic =
|
||
|
node.attributes.media != null && node.attributes.media !== 'all';
|
||
|
if (
|
||
|
node.attributes.type == null ||
|
||
|
node.attributes.type === '' ||
|
||
|
node.attributes.type === 'text/css'
|
||
|
) {
|
||
|
const children = node.children;
|
||
|
for (const child of children) {
|
||
|
if (child.type === 'text' || child.type === 'cdata') {
|
||
|
rules.push(...parseStylesheet(child.value, dynamic));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
},
|
||
|
});
|
||
|
// sort by selectors specificity
|
||
|
stable.inplace(rules, (a, b) =>
|
||
|
compareSpecificity(a.specificity, b.specificity)
|
||
|
);
|
||
|
return { rules, parents };
|
||
|
};
|
||
|
exports.collectStylesheet = collectStylesheet;
|
||
|
|
||
|
/**
|
||
|
* @type {(stylesheet: Stylesheet, node: XastElement) => ComputedStyles}
|
||
|
*/
|
||
|
const computeStyle = (stylesheet, node) => {
|
||
|
const { parents } = stylesheet;
|
||
|
// collect inherited styles
|
||
|
const computedStyles = computeOwnStyle(stylesheet, node);
|
||
|
let parent = parents.get(node);
|
||
|
while (parent != null && parent.type !== 'root') {
|
||
|
const inheritedStyles = computeOwnStyle(stylesheet, parent);
|
||
|
for (const [name, computed] of Object.entries(inheritedStyles)) {
|
||
|
if (
|
||
|
computedStyles[name] == null &&
|
||
|
// ignore not inheritable styles
|
||
|
inheritableAttrs.includes(name) === true &&
|
||
|
presentationNonInheritableGroupAttrs.includes(name) === false
|
||
|
) {
|
||
|
computedStyles[name] = { ...computed, inherited: true };
|
||
|
}
|
||
|
}
|
||
|
parent = parents.get(parent);
|
||
|
}
|
||
|
return computedStyles;
|
||
|
};
|
||
|
exports.computeStyle = computeStyle;
|