push
This commit is contained in:
128
build/plugin/theme/utils.ts
Normal file
128
build/plugin/theme/utils.ts
Normal file
@@ -0,0 +1,128 @@
|
||||
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;
|
||||
}
|
||||
Reference in New Issue
Block a user