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.
129 lines
3.2 KiB
129 lines
3.2 KiB
6 months ago
|
import { ResolvedConfig } from 'vite';
|
||
|
import { createHash } from 'crypto';
|
||
|
import { ResolveSelector } from '.';
|
||
|
import { commentRE, cssBlockRE, ruleRE, cssValueRE, safeEmptyRE, importSafeRE } from './constants';
|
||
|
import CleanCSS from 'clean-css';
|
||
|
export function getVariablesReg(colors: string[]) {
|
||
|
return new RegExp(
|
||
|
colors
|
||
|
.map(
|
||
|
(i) =>
|
||
|
`(${i
|
||
|
.replace(/\s/g, ' ?')
|
||
|
.replace(/\(/g, `\\(`)
|
||
|
.replace(/\)/g, `\\)`)
|
||
|
.replace(/0?\./g, `0?\\.`)})`
|
||
|
)
|
||
|
.join('|')
|
||
|
);
|
||
|
}
|
||
|
|
||
|
export function combineRegs(decorator = '', joinString = '', ...args: any[]) {
|
||
|
const regString = args
|
||
|
.map((item) => {
|
||
|
const str = item.toString();
|
||
|
return `(${str.slice(1, str.length - 1)})`;
|
||
|
})
|
||
|
.join(joinString);
|
||
|
return new RegExp(regString, decorator);
|
||
|
}
|
||
|
|
||
|
export function formatCss(s: string) {
|
||
|
s = s.replace(/\s*([{}:;,])\s*/g, '$1');
|
||
|
s = s.replace(/;\s*;/g, ';');
|
||
|
s = s.replace(/,[\s.#\d]*{/g, '{');
|
||
|
s = s.replace(/([^\s])\{([^\s])/g, '$1 {\n\t$2');
|
||
|
s = s.replace(/([^\s])\}([^\n]*)/g, '$1\n}\n$2');
|
||
|
s = s.replace(/([^\s]);([^\s}])/g, '$1;\n\t$2');
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
export function createFileHash() {
|
||
|
return createHash('sha256').digest('hex').substr(0, 8);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Compress the generated code
|
||
|
*/
|
||
|
export async function minifyCSS(css: string, config: ResolvedConfig) {
|
||
|
const res = new CleanCSS({
|
||
|
rebase: false,
|
||
|
...config.build.cleanCssOptions,
|
||
|
}).minify(css);
|
||
|
|
||
|
if (res.errors && res.errors.length) {
|
||
|
console.error(`error when minifying css:\n${res.errors}`);
|
||
|
throw res.errors[0];
|
||
|
}
|
||
|
|
||
|
if (res.warnings && res.warnings.length) {
|
||
|
config.logger.warn(`warnings when minifying css:\n${res.warnings}`);
|
||
|
}
|
||
|
|
||
|
return res.styles;
|
||
|
}
|
||
|
|
||
|
// Used to extract relevant color configuration in css
|
||
|
export function extractVariable(
|
||
|
code: string,
|
||
|
colorVariables: string[],
|
||
|
resolveSelector?: ResolveSelector,
|
||
|
colorRE?: RegExp
|
||
|
) {
|
||
|
colorVariables = Array.from(new Set(colorVariables));
|
||
|
code = code.replace(commentRE, '');
|
||
|
|
||
|
const cssBlocks = code.match(cssBlockRE);
|
||
|
if (!cssBlocks || cssBlocks.length === 0) {
|
||
|
return '';
|
||
|
}
|
||
|
|
||
|
let allExtractedVariable = '';
|
||
|
|
||
|
const variableReg = getVariablesReg(colorVariables);
|
||
|
|
||
|
for (let index = 0; index < cssBlocks.length; index++) {
|
||
|
const cssBlock = cssBlocks[index];
|
||
|
if (!variableReg.test(cssBlock) || !cssBlock) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
const cssSelector = cssBlock.match(/[^{]*/)?.[0] ?? '';
|
||
|
if (!cssSelector) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (/^@.*keyframes/.test(cssSelector)) {
|
||
|
allExtractedVariable += `${cssSelector}{${extractVariable(
|
||
|
cssBlock.replace(/[^{]*\{/, '').replace(/}$/, ''),
|
||
|
colorVariables,
|
||
|
resolveSelector,
|
||
|
colorRE
|
||
|
)}}`;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
const colorReg = combineRegs(
|
||
|
'g',
|
||
|
'',
|
||
|
ruleRE,
|
||
|
cssValueRE,
|
||
|
safeEmptyRE,
|
||
|
variableReg,
|
||
|
importSafeRE
|
||
|
);
|
||
|
|
||
|
const colorReplaceTemplates = cssBlock.match(colorRE || colorReg);
|
||
|
|
||
|
if (!colorReplaceTemplates) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
allExtractedVariable += `${
|
||
|
resolveSelector ? resolveSelector(cssSelector) : cssSelector
|
||
|
} {${colorReplaceTemplates.join(';')}}`;
|
||
|
}
|
||
|
|
||
|
return allExtractedVariable;
|
||
|
}
|