forked from mystiq/hydrogen-web
89 lines
3.1 KiB
JavaScript
89 lines
3.1 KiB
JavaScript
const valueParser = require("postcss-value-parser");
|
|
|
|
let aliasMap;
|
|
let resolvedMap;
|
|
|
|
function getValueFromAlias(alias, variables) {
|
|
const derivedVariable = aliasMap.get(`--${alias}`);
|
|
return variables[derivedVariable] ?? resolvedMap.get(`--${derivedVariable}`); // what if we haven't resolved this variable yet?
|
|
}
|
|
|
|
function parseDeclarationValue(value) {
|
|
const parsed = valueParser(value);
|
|
const variables = [];
|
|
parsed.walk(node => {
|
|
if (node.type !== "function" && node.value !== "var") {
|
|
return;
|
|
}
|
|
const variable = node.nodes[0];
|
|
variables.push(variable.value);
|
|
});
|
|
return variables;
|
|
}
|
|
|
|
function resolveDerivedVariable(decl, {variables, derive}) {
|
|
const RE_VARIABLE_VALUE = /--(.+)--(.+)-(.+)/;
|
|
const variableCollection = parseDeclarationValue(decl.value);
|
|
for (const variable of variableCollection) {
|
|
const matches = variable.match(RE_VARIABLE_VALUE);
|
|
if (matches) {
|
|
const [wholeVariable, baseVariable, operation, argument] = matches;
|
|
if (!variables[baseVariable]) {
|
|
// hmm.. baseVariable should be in config..., maybe this is an alias?
|
|
if (!aliasMap.get(`--${baseVariable}`)) {
|
|
throw new Error(`Cannot derive from ${baseVariable} because it is neither defined in config nor is it an alias!`);
|
|
}
|
|
}
|
|
const value = variables[baseVariable] ?? getValueFromAlias(baseVariable, variables);
|
|
const derivedValue = derive(value, operation, argument);
|
|
resolvedMap.set(wholeVariable, derivedValue);
|
|
}
|
|
}
|
|
}
|
|
|
|
function extractAlias(decl) {
|
|
const wholeVariable = decl.value.match(/var\(--(.+)\)/)?.[1];
|
|
if (decl.prop.startsWith("--") && wholeVariable) {
|
|
aliasMap.set(decl.prop, wholeVariable);
|
|
}
|
|
}
|
|
|
|
function addResolvedVariablesToRootSelector(root, variables, { Rule, Declaration }) {
|
|
const newRule = new Rule({ selector: ":root", source: root.source });
|
|
// Add base css variables to :root
|
|
for (const [key, value] of Object.entries(variables)) {
|
|
const declaration = new Declaration({ prop: `--${key}`, value });
|
|
newRule.append(declaration);
|
|
}
|
|
// Add derived css variables to :root
|
|
resolvedMap.forEach((value, key) => {
|
|
const declaration = new Declaration({ prop: key, value });
|
|
newRule.append(declaration);
|
|
});
|
|
root.append(newRule);
|
|
}
|
|
|
|
/* *
|
|
* @type {import('postcss').PluginCreator}
|
|
*/
|
|
module.exports = (opts = {}) => {
|
|
aliasMap = new Map();
|
|
resolvedMap = new Map();
|
|
const {variables} = opts;
|
|
return {
|
|
postcssPlugin: "postcss-compile-variables",
|
|
|
|
Once(root, { Rule, Declaration }) {
|
|
/*
|
|
Go through the CSS file once to extract all aliases.
|
|
We use the extracted alias when resolving derived variables
|
|
later.
|
|
*/
|
|
root.walkDecls(decl => extractAlias(decl));
|
|
root.walkDecls(decl => resolveDerivedVariable(decl, opts));
|
|
addResolvedVariablesToRootSelector(root, variables, { Rule, Declaration });
|
|
},
|
|
};
|
|
};
|
|
|
|
module.exports.postcss = true;
|