代码拉取完成,页面将自动刷新
const fs = require("fs");
const path = require("path");
const glob = require("glob");
const postcss = require("postcss");
const less = require("less");
const bundle = require("less-bundle-promise");
const hash = require("hash.js");
const NpmImportPlugin = require('less-plugin-npm-import');
const colorsOnly = require('postcss-colors-only');
const options = {
withoutGrey: true, // set to true to remove rules that only have grey colors
withoutMonochrome: true, // set to true to remove rules that only have grey, black, or white colors
};
let hashCache = "";
let cssCache = "";
function randomColor() {
return '#' + (Math.random() * 0xFFFFFF << 0).toString(16);
}
/*
Recursively get the color code assigned to a variable e.g.
@primary-color: #1890ff;
@link-color: @primary-color;
@link-color -> @primary-color -> #1890ff
Which means
@link-color: #1890ff
*/
function getColor(varName, mappings) {
const color = mappings[varName];
if (color in mappings) {
return getColor(color, mappings);
} else {
return color;
}
}
/*
Read following files and generate color variables and color codes mapping
- Ant design color.less, themes/default.less
- Your own variables.less
It will generate map like this
{
'@primary-color': '#00375B',
'@info-color': '#1890ff',
'@success-color': '#52c41a',
'@error-color': '#f5222d',
'@normal-color': '#d9d9d9',
'@primary-6': '#1890ff',
'@heading-color': '#fa8c16',
'@text-color': '#cccccc',
....
}
*/
function generateColorMap(content) {
return content
.split("\n")
.filter(line => line.startsWith("@") && line.indexOf(":") > -1)
.reduce((prev, next) => {
try {
const matches = next.match(
/(?=\S*['-])([@a-zA-Z0-9'-]+).*:[ ]{1,}(.*);/
);
if (!matches) {
return prev;
}
let [, varName, color] = matches;
if (color && color.startsWith("@")) {
color = getColor(color, prev);
if (!isValidColor(color)) return prev;
prev[varName] = color;
} else if (isValidColor(color)) {
prev[varName] = color;
}
return prev;
} catch (e) {
console.log("e", e);
return prev;
}
}, {});
}
/*
This plugin will remove all css rules except those are related to colors
e.g.
Input:
.body {
font-family: 'Lato';
background: #cccccc;
color: #000;
padding: 0;
pargin: 0
}
Output:
.body {
background: #cccccc;
color: #000;
}
*/
const reducePlugin = postcss.plugin("reducePlugin", () => {
const cleanRule = rule => {
if (rule.selector.startsWith(".main-color .palatte-")) {
rule.remove();
return;
}
let removeRule = true;
rule.walkDecls(decl => {
if (
!decl.prop.includes("color") &&
!decl.prop.includes("background") &&
!decl.prop.includes("border") &&
!decl.prop.includes("box-shadow")
) {
decl.remove();
} else {
removeRule = false;
}
});
if (removeRule) {
rule.remove();
}
};
return css => {
css.walkAtRules(atRule => {
atRule.remove();
});
css.walkRules(cleanRule);
css.walkComments(c => c.remove());
};
});
function getMatches(string, regex) {
const matches = {};
let match;
while ((match = regex.exec(string))) {
if (match[2].startsWith("rgba") || match[2].startsWith("#")) {
matches[`@${match[1]}`] = match[2];
}
}
return matches;
}
/*
This function takes less input as string and compiles into css.
*/
function render(text, paths) {
return less.render.call(less, text, {
paths: paths,
javascriptEnabled: true,
plugins: [new NpmImportPlugin({ prefix: '~' })]
});
}
/*
This funtion reads a less file and create an object with keys as variable names
and values as variables respective values. e.g.
//variabables.less
@primary-color : #1890ff;
@heading-color : #fa8c16;
@text-color : #cccccc;
to
{
'@primary-color' : '#1890ff',
'@heading-color' : '#fa8c16',
'@text-color' : '#cccccc'
}
*/
function getLessVars(filtPath) {
const sheet = fs.readFileSync(filtPath).toString();
const lessVars = {};
const matches = sheet.match(/@(.*:[^;]*)/g) || [];
matches.forEach(variable => {
const definition = variable.split(/:\s*/);
const varName = definition[0].replace(/['"]+/g, "").trim();
lessVars[varName] = definition.splice(1).join(":");
});
return lessVars;
}
/*
This function take primary color palette name and returns @primary-color dependent value
.e.g
Input: @primary-1
Output: color(~`colorPalette("@{primary-color}", ' 1 ')`)
*/
function getShade(varName) {
let [, className, number] = varName.match(/(.*)-(\d)/);
if (/primary-\d/.test(varName)) className = '@primary-color';
return 'color(~`colorPalette("@{' + className.replace('@', '') + '}", ' + number + ")`)";
}
/*
This function takes color string as input and return true if string is a valid color otherwise returns false.
e.g.
isValidColor('#ffffff'); //true
isValidColor('#fff'); //true
isValidColor('rgba(0, 0, 0, 0.5)'); //true
isValidColor('20px'); //false
*/
function isValidColor(color) {
if (!color || color.match(/px/g)) return false;
if (color.match(/colorPalette|fade/g)) return true;
if (color.charAt(0) === "#") {
color = color.substring(1);
return (
[3, 4, 6, 8].indexOf(color.length) > -1 && !isNaN(parseInt(color, 16))
);
}
return /^(rgb|hsl|hsv)a?\((\d+%?(deg|rad|grad|turn)?[,\s]+){2,3}[\s\/]*[\d\.]+%?\)$/i.test(
color
);
}
function getCssModulesStyles(stylesDir, antdStylesDir) {
const styles = glob.sync(path.join(stylesDir, './**/*.less'));
return Promise.all(
styles.map(p =>
less
.render(fs.readFileSync(p).toString(), {
paths: [
stylesDir,
antdStylesDir,
],
filename: path.resolve(p),
javascriptEnabled: true,
plugins: [new NpmImportPlugin({ prefix: '~' })],
})
.catch(() => '\n')
)
)
.then(csss => csss.map(c => c.css).join('\n'))
.catch(err => {
console.log('Error', err);
return '';
});
}
/*
This is main function which call all other functions to generate color.less file which contins all color
related css rules based on Ant Design styles and your own custom styles
By default color.less will be generated in /public directory
*/
function generateTheme({
antDir,
antdStylesDir,
stylesDir,
mainLessFile,
varFile,
outputFilePath,
cssModules = false,
themeVariables = ['@primary-color']
}) {
return new Promise((resolve, reject) => {
/*
Ant Design Specific Files (Change according to your project structure)
You can even use different less based css framework and create color.less for that
- antDir - ant design instalation path
- entry - Ant Design less main file / entry file
- styles - Ant Design less styles for each component
*/
let antdPath;
if (antdStylesDir) {
antdPath = antdStylesDir;
} else {
// antdPath = path.join(antDir, 'lib');
antdPath = antDir;
}
const entry = path.join(antdPath, './style/index.less');
const styles = glob.sync(path.join(antdPath, './*/style/index.less'));
/*
You own custom styles (Change according to your project structure)
- stylesDir - styles directory containing all less files
- mainLessFile - less main file which imports all other custom styles
- varFile - variable file containing ant design specific and your own custom variables
*/
varFile = varFile || path.join(antdPath, "./style/themes/default.less");
let content = fs.readFileSync(entry).toString();
content += "\n";
styles.forEach(style => {
content += `@import "${style}";\n`;
});
if (mainLessFile) {
const customStyles = fs.readFileSync(mainLessFile).toString();
content += `\n${customStyles}`;
}
const hashCode = hash.sha256().update(content).digest('hex');
if(hashCode === hashCache){
resolve(cssCache);
return;
}
hashCache = hashCode;
let themeCompiledVars = {};
let themeVars = themeVariables || ["@primary-color"];
const lessPaths = [
path.join(antdPath, "./style"),
stylesDir
];
return bundle({
src: varFile
})
.then(colorsLess => {
const mappings = Object.assign(generateColorMap(colorsLess), generateColorMap(mainLessFile));
return [mappings, colorsLess];
})
.then(([mappings, colorsLess]) => {
let css = "";
themeVars = themeVars.filter(name => name in mappings);
themeVars.forEach(varName => {
const color = mappings[varName];
css = `.${varName.replace("@", "")} { color: ${color}; }\n ${css}`;
});
themeVars.forEach(varName => {
[1, 2, 3, 4, 5, 7].forEach(key => {
let name = varName === '@primary-color' ? `@primary-${key}` : `${varName}-${key}`;
css = `.${name.replace("@", "")} { color: ${getShade(name)}; }\n ${css}`;
});
});
css = `${colorsLess}\n${css}`;
return render(css, lessPaths).then(({ css }) => [
css,
mappings,
colorsLess
]);
})
.then(([css, mappings, colorsLess]) => {
css = css.replace(/(\/.*\/)/g, "");
const regex = /.(?=\S*['-])([.a-zA-Z0-9'-]+)\ {\n\ \ color:\ (.*);/g;
themeCompiledVars = getMatches(css, regex);
content = `${content}\n${colorsLess}`;
return render(content, lessPaths).then(({ css }) => {
return getCssModulesStyles(stylesDir, antdStylesDir).then(customCss => {
return [
`${customCss}\n${css}`,
mappings,
colorsLess
];
})
});
})
.then(([css, mappings, colorsLess]) => {
return postcss([reducePlugin])
// return postcss.use(colorsOnly(options))
.process(css, {
parser: less.parser,
from: entry
})
.then(({ css }) => [css, mappings, colorsLess]);
})
.then(([css, mappings, colorsLess]) => {
Object.keys(themeCompiledVars).forEach(varName => {
let color;
if (/(.*)-(\d)/.test(varName)) {
color = themeCompiledVars[varName];
varName = getShade(varName);
} else {
color = themeCompiledVars[varName];
}
color = color.replace('(', '\\(').replace(')', '\\)');
// css = css.replace(new RegExp(`${color}`, "g"), varName); // Fixed bug https://github.com/mzohaibqc/antd-theme-webpack-plugin/issues/25
css = css.replace(new RegExp(`${color}` + ' *;', "g"), `${varName};`);
});
css = `${colorsLess}\n${css}`;
themeVars.reverse().forEach(varName => {
css = css.replace(new RegExp(`${varName}(\ *):(.*);`, 'g'), '');
css = `${varName}: ${mappings[varName]};\n${css}\n`;
});
css = css.replace(/\\9/g, '');
if (outputFilePath) {
fs.writeFileSync(outputFilePath, css);
console.log(
`🌈 Theme generated successfully. OutputFile: ${outputFilePath}`
);
} else {
console.log(`Theme generated successfully`);
}
cssCache = css;
return resolve(css);
})
.catch(err => {
console.log("Error", err);
reject(err);
});
});
}
module.exports = {
generateTheme,
isValidColor,
getLessVars,
randomColor,
renderLessContent: render
};
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。