import { TinyColor } from '@ctrl/tinycolor'; const hueStep = 2; // 色相阶梯 const saturationStep = 0.16; // 饱和度阶梯,浅色部分 const saturationStep2 = 0.05; // 饱和度阶梯,深色部分 const brightnessStep1 = 0.05; // 亮度阶梯,浅色部分 const brightnessStep2 = 0.15; // 亮度阶梯,深色部分 const lightColorCount = 5; // 浅色数量,主色上 const darkColorCount = 4; // 深色数量,主色下 // 暗色主题颜色映射关系表 const darkColorMap = [ { index: 7, opacity: 0.15 }, { index: 6, opacity: 0.25 }, { index: 5, opacity: 0.3 }, { index: 5, opacity: 0.45 }, { index: 5, opacity: 0.65 }, { index: 5, opacity: 0.85 }, { index: 4, opacity: 0.9 }, { index: 3, opacity: 0.95 }, { index: 2, opacity: 0.97 }, { index: 1, opacity: 0.98 }, ]; interface HsvObject { h: number; s: number; v: number; } function getHue(hsv: HsvObject, i: number, light?: boolean): number { let hue: number; // 根据色相不同,色相转向不同 if (Math.round(hsv.h) >= 60 && Math.round(hsv.h) <= 240) { hue = light ? Math.round(hsv.h) - hueStep * i : Math.round(hsv.h) + hueStep * i; } else { hue = light ? Math.round(hsv.h) + hueStep * i : Math.round(hsv.h) - hueStep * i; } if (hue < 0) { hue += 360; } else if (hue >= 360) { hue -= 360; } return hue; } function getSaturation(hsv: HsvObject, i: number, light?: boolean): number { // grey color don't change saturation if (hsv.h === 0 && hsv.s === 0) { return hsv.s; } let saturation: number; if (light) { saturation = hsv.s - saturationStep * i; } else if (i === darkColorCount) { saturation = hsv.s + saturationStep; } else { saturation = hsv.s + saturationStep2 * i; } // 边界值修正 if (saturation > 1) { saturation = 1; } // 第一格的 s 限制在 0.06-0.1 之间 if (light && i === lightColorCount && saturation > 0.1) { saturation = 0.1; } if (saturation < 0.06) { saturation = 0.06; } return Number(saturation.toFixed(2)); } function getValue(hsv: HsvObject, i: number, light?: boolean): number { let value: number; if (light) { value = hsv.v + brightnessStep1 * i; } else { value = hsv.v - brightnessStep2 * i; } if (value > 1) { value = 1; } return Number(value.toFixed(2)); } interface Opts { theme?: 'dark' | 'default'; backgroundColor?: string; } /** * 根据输入颜色生成色阶 * @param color : 颜色; * @param opts * @returns */ export default function generateColorGradation(color: string, opts: Opts = {}): string[] { const patterns: Array<string> = []; const pColor = new TinyColor(color); for (let i = lightColorCount; i > 0; i -= 1) { const hsv = pColor.toHsv(); const colorString: string = new TinyColor({ h: getHue(hsv, i, true), s: getSaturation(hsv, i, true), v: getValue(hsv, i, true), }).toHexString(); patterns.push(colorString); } patterns.push(pColor.toHexString()); for (let i = 1; i <= darkColorCount; i += 1) { const hsv = pColor.toHsv(); const colorString: string = new TinyColor({ h: getHue(hsv, i), s: getSaturation(hsv, i), v: getValue(hsv, i), }).toHexString(); patterns.push(colorString); } // dark theme patterns if (opts.theme === 'dark') { return darkColorMap.map(({ index, opacity }) => { const darkColorString: string = new TinyColor(opts.backgroundColor || '#141414') .mix(patterns[index], opacity * 100) .toHexString(); return darkColorString; }); } return patterns; }