123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287 |
- // src/utils/gen-color.ts
- // 定义颜色类型
- type RGB = {
- r: number;
- g: number;
- b: number;
- };
- type HSL = {
- h: number;
- s: number;
- l: number;
- };
- type HEX =
- | '0'
- | '1'
- | '2'
- | '3'
- | '4'
- | '5'
- | '6'
- | '7'
- | '8'
- | '9'
- | 'A'
- | 'B'
- | 'C'
- | 'D'
- | 'E'
- | 'F';
- const RGBUnit = 255;
- const HEX_MAP: Record<HEX, number> = {
- 0: 0,
- 1: 1,
- 2: 2,
- 3: 3,
- 4: 4,
- 5: 5,
- 6: 6,
- 7: 7,
- 8: 8,
- 9: 9,
- A: 10,
- B: 11,
- C: 12,
- D: 13,
- E: 14,
- F: 15,
- };
- const rgbWhite = {
- r: 255,
- g: 255,
- b: 255,
- };
- const rgbBlack = {
- r: 0,
- g: 0,
- b: 0,
- };
- /**
- * RGB颜色转HSL颜色值
- * @param r 红色值
- * @param g 绿色值
- * @param b 蓝色值
- * @returns { h: [0, 360]; s: [0, 1]; l: [0, 1] }
- */
- function rgbToHsl(rgb: RGB): HSL {
- let { r, g, b } = rgb;
- const hsl = {
- h: 0,
- s: 0,
- l: 0,
- };
- // 计算rgb基数 ∈ [0, 1]
- r /= RGBUnit;
- g /= RGBUnit;
- b /= RGBUnit;
- const max = Math.max(r, g, b);
- const min = Math.min(r, g, b);
- // 计算h
- if (max === min) {
- hsl.h = 0;
- } else if (max === r) {
- hsl.h = 60 * ((g - b) / (max - min)) + (g >= b ? 0 : 360);
- } else if (max === g) {
- hsl.h = 60 * ((b - r) / (max - min)) + 120;
- } else {
- hsl.h = 60 * ((r - g) / (max - min)) + 240;
- }
- hsl.h = hsl.h > 360 ? hsl.h - 360 : hsl.h;
- // 计算l
- hsl.l = (max + min) / 2;
- // 计算s
- if (hsl.l === 0 || max === min) {
- // 灰/白/黑
- hsl.s = 0;
- } else if (hsl.l > 0 && hsl.l <= 0.5) {
- hsl.s = (max - min) / (max + min);
- } else {
- hsl.s = (max - min) / (2 - (max + min));
- }
- return hsl;
- }
- /**
- * hsl -> rgb
- * @param h [0, 360]
- * @param s [0, 1]
- * @param l [0, 1]
- * @returns RGB
- */
- function hslToRgb(hsl: HSL): RGB {
- const { h, s, l } = hsl;
- const q = l < 0.5 ? l * (1 + s) : l + s - l * s;
- const p = 2 * l - q;
- const hUnit = h / 360; // 色相转换为 [0, 1]
- const Cr = fillCircleVal(hUnit + 1 / 3);
- const Cg = fillCircleVal(hUnit);
- const Cb = fillCircleVal(hUnit - 1 / 3);
- // 保持 [0, 1] 环状取值
- function fillCircleVal(val: number): number {
- return val < 0 ? val + 1 : val > 1 ? val - 1 : val;
- }
- function computedRgb(val: number): number {
- let colorVal: number;
- if (val < 1 / 6) {
- colorVal = p + (q - p) * 6 * val;
- } else if (val >= 1 / 6 && val < 1 / 2) {
- colorVal = q;
- } else if (val >= 1 / 2 && val < 2 / 3) {
- colorVal = p + (q - p) * 6 * (2 / 3 - val);
- } else {
- colorVal = p;
- }
- return colorVal * 255;
- }
- return {
- r: Number(computedRgb(Cr).toFixed(0)),
- g: Number(computedRgb(Cg).toFixed(0)),
- b: Number(computedRgb(Cb).toFixed(0)),
- };
- }
- /**
- * 16进制颜色转换RGB
- * @param color #rrggbb
- * @returns RGB
- */
- function hexToRGB(hex: string): RGB {
- hex = hex.toUpperCase();
- const hexRegExp = /^#([0-9A-F]{6})$/;
- if (!hexRegExp.test(hex)) {
- throw new Error('请传入合法的16进制颜色值,eg: #FF0000, input: ' + hex);
- }
- const hexValArr = (hexRegExp.exec(hex)?.[1] || '000000').split(
- ''
- ) as Array<HEX>;
- return {
- r: HEX_MAP[hexValArr[0]] * 16 + HEX_MAP[hexValArr[1]],
- g: HEX_MAP[hexValArr[2]] * 16 + HEX_MAP[hexValArr[3]],
- b: HEX_MAP[hexValArr[4]] * 16 + HEX_MAP[hexValArr[5]],
- };
- }
- /**
- * rgb 转 16进制
- * @param rgb RGB
- * @returns #HEX{6}
- */
- function rgbToHex(rgb: RGB): string {
- const HEX_MAP_REVERSE: Record<string, HEX> = {};
- for (const key in HEX_MAP) {
- HEX_MAP_REVERSE[HEX_MAP[key as HEX]] = key as HEX;
- }
- function getRemainderAndQuotient(val: number): string {
- val = Math.round(val);
- return `${HEX_MAP_REVERSE[Math.floor(val / 16)]}${
- HEX_MAP_REVERSE[val % 16]
- }`;
- }
- return `#${getRemainderAndQuotient(rgb.r)}${getRemainderAndQuotient(
- rgb.g
- )}${getRemainderAndQuotient(rgb.b)}`;
- }
- // hsl 转 16进制
- function hslToHex(hsl: HSL): string {
- return rgbToHex(hslToRgb(hsl));
- }
- // 16进制 转 hsl
- function hexToHsl(hex: string): HSL {
- return rgbToHsl(hexToRGB(hex));
- }
- // 生成混合色(混黑 + 混白)
- function genMixColor(base: string | RGB | HSL): {
- DEFAULT: string;
- dark: {
- 1: string;
- 2: string;
- 3: string;
- 4: string;
- 5: string;
- 6: string;
- 7: string;
- 8: string;
- 9: string;
- };
- light: {
- 1: string;
- 2: string;
- 3: string;
- 4: string;
- 5: string;
- 6: string;
- 7: string;
- 8: string;
- 9: string;
- };
- } {
- // 基准色统一转换为RGB,方便进行混合色生成
- if (typeof base === 'string') {
- base = hexToRGB(base);
- } else if ('h' in base) {
- base = hslToRgb(base);
- }
- // 混合色(这里参考Sass mix函数逻辑,本质上就是两种颜色的rbg各个值加权平均)
- function mix(color: RGB, mixColor: RGB, weight: number): RGB {
- return {
- r: color.r * (1 - weight) + mixColor.r * weight,
- g: color.g * (1 - weight) + mixColor.g * weight,
- b: color.b * (1 - weight) + mixColor.b * weight,
- };
- }
- return {
- DEFAULT: rgbToHex(base),
- dark: {
- 1: rgbToHex(mix(base, rgbBlack, 0.1)),
- 2: rgbToHex(mix(base, rgbBlack, 0.2)),
- 3: rgbToHex(mix(base, rgbBlack, 0.3)),
- 4: rgbToHex(mix(base, rgbBlack, 0.4)),
- 5: rgbToHex(mix(base, rgbBlack, 0.5)),
- 6: rgbToHex(mix(base, rgbBlack, 0.6)),
- 7: rgbToHex(mix(base, rgbBlack, 0.7)),
- 8: rgbToHex(mix(base, rgbBlack, 0.78)),
- 9: rgbToHex(mix(base, rgbBlack, 0.85)),
- },
- light: {
- 1: rgbToHex(mix(base, rgbWhite, 0.1)),
- 2: rgbToHex(mix(base, rgbWhite, 0.2)),
- 3: rgbToHex(mix(base, rgbWhite, 0.3)),
- 4: rgbToHex(mix(base, rgbWhite, 0.4)),
- 5: rgbToHex(mix(base, rgbWhite, 0.5)),
- 6: rgbToHex(mix(base, rgbWhite, 0.6)),
- 7: rgbToHex(mix(base, rgbWhite, 0.7)),
- 8: rgbToHex(mix(base, rgbWhite, 0.78)),
- 9: rgbToHex(mix(base, rgbWhite, 0.85)),
- },
- };
- }
- export {
- genMixColor,
- rgbToHsl,
- rgbToHex,
- hslToRgb,
- hslToHex,
- hexToRGB,
- hexToHsl,
- };
|