import { nextAnimationFrame } from "./utils"; interface ColorRGB { r: number; g: number; b: number; } interface ColorHSV { h: number; s: number; v: number; } export enum ColorFmt { RGB, // rgb(R,G,B) HEX, // #RRGGBB } /* https://developer.mozilla.org/en-US/docs/Web/CSS/color_value $$('#colors_table tbody tr') .map((row) => { const [name, hex] = Array.from(row.children).slice(-3); return [name.innerText.trim().replace(/\s.+/g, ''), hex.innerText.trim()]; }).sort((a, b) => [a[0], b[0]].sort()[0] === a[0] ? -1 : 1) .map(([name, hex]) => `${name} : '${hex}',`) .join('\n') */ const namedColors = { aliceblue : '#f0f8ff', antiquewhite : '#faebd7', aqua : '#00ffff', aquamarine : '#7fffd4', azure : '#f0ffff', beige : '#f5f5dc', bisque : '#ffe4c4', black : '#000000', blanchedalmond : '#ffebcd', blue : '#0000ff', blueviolet : '#8a2be2', brown : '#a52a2a', burlywood : '#deb887', cadetblue : '#5f9ea0', chartreuse : '#7fff00', chocolate : '#d2691e', coral : '#ff7f50', cornflowerblue : '#6495ed', cornsilk : '#fff8dc', crimson : '#dc143c', cyan : '#00ffff', darkblue : '#00008b', darkcyan : '#008b8b', darkgoldenrod : '#b8860b', darkgray : '#a9a9a9', darkgreen : '#006400', darkgrey : '#a9a9a9', darkkhaki : '#bdb76b', darkmagenta : '#8b008b', darkolivegreen : '#556b2f', darkorange : '#ff8c00', darkorchid : '#9932cc', darkred : '#8b0000', darksalmon : '#e9967a', darkseagreen : '#8fbc8f', darkslateblue : '#483d8b', darkslategray : '#2f4f4f', darkslategrey : '#2f4f4f', darkturquoise : '#00ced1', darkviolet : '#9400d3', deeppink : '#ff1493', deepskyblue : '#00bfff', dimgray : '#696969', dimgrey : '#696969', dodgerblue : '#1e90ff', firebrick : '#b22222', floralwhite : '#fffaf0', forestgreen : '#228b22', fuchsia : '#ff00ff', gainsboro : '#dcdcdc', ghostwhite : '#f8f8ff', gold : '#ffd700', goldenrod : '#daa520', gray : '#808080', green : '#008000', greenyellow : '#adff2f', grey : '#808080', honeydew : '#f0fff0', hotpink : '#ff69b4', indianred : '#cd5c5c', indigo : '#4b0082', ivory : '#fffff0', khaki : '#f0e68c', lavender : '#e6e6fa', lavenderblush : '#fff0f5', lawngreen : '#7cfc00', lemonchiffon : '#fffacd', lightblue : '#add8e6', lightcoral : '#f08080', lightcyan : '#e0ffff', lightgoldenrodyellow : '#fafad2', lightgray : '#d3d3d3', lightgreen : '#90ee90', lightgrey : '#d3d3d3', lightpink : '#ffb6c1', lightsalmon : '#ffa07a', lightseagreen : '#20b2aa', lightskyblue : '#87cefa', lightslategray : '#778899', lightslategrey : '#778899', lightsteelblue : '#b0c4de', lightyellow : '#ffffe0', lime : '#00ff00', limegreen : '#32cd32', linen : '#faf0e6', magenta : '#ff00ff', maroon : '#800000', mediumaquamarine : '#66cdaa', mediumblue : '#0000cd', mediumorchid : '#ba55d3', mediumpurple : '#9370db', mediumseagreen : '#3cb371', mediumslateblue : '#7b68ee', mediumspringgreen : '#00fa9a', mediumturquoise : '#48d1cc', mediumvioletred : '#c71585', midnightblue : '#191970', mintcream : '#f5fffa', mistyrose : '#ffe4e1', moccasin : '#ffe4b5', navajowhite : '#ffdead', navy : '#000080', oldlace : '#fdf5e6', olive : '#808000', olivedrab : '#6b8e23', orange : '#ffa500', orangered : '#ff4500', orchid : '#da70d6', palegoldenrod : '#eee8aa', palegreen : '#98fb98', paleturquoise : '#afeeee', palevioletred : '#db7093', papayawhip : '#ffefd5', peachpuff : '#ffdab9', peru : '#cd853f', pink : '#ffc0cb', plum : '#dda0dd', powderblue : '#b0e0e6', purple : '#800080', rebeccapurple : '#663399', red : '#ff0000', rosybrown : '#bc8f8f', royalblue : '#4169e1', saddlebrown : '#8b4513', salmon : '#fa8072', sandybrown : '#f4a460', seagreen : '#2e8b57', seashell : '#fff5ee', sienna : '#a0522d', silver : '#c0c0c0', skyblue : '#87ceeb', slateblue : '#6a5acd', slategray : '#708090', slategrey : '#708090', snow : '#fffafa', springgreen : '#00ff7f', steelblue : '#4682b4', tan : '#d2b48c', teal : '#008080', thistle : '#d8bfd8', tomato : '#ff6347', turquoise : '#40e0d0', violet : '#ee82ee', wheat : '#f5deb3', white : '#ffffff', whitesmoke : '#f5f5f5', yellow : '#ffff00', yellowgreen : '#9acd32', } function hsvToRgb({ h, s, v }: ColorHSV): ColorRGB { const i = Math.floor(h * 6); const f = h * 6 - i; const p = v * (1 - s); const q = v * (1 - f * s); const t = v * (1 - (1 - f) * s); switch (i % 6) { case 0: return { r: v, g: t, b: p }; case 1: return { r: q, g: v, b: p }; case 2: return { r: p, g: v, b: t }; case 3: return { r: p, g: q, b: v }; case 4: return { r: t, g: p, b: v }; case 5: return { r: v, g: p, b: q }; default: throw new Error("unreacheable"); } } function rgbToHsv({ r, g, b }: ColorRGB): ColorHSV { const max = Math.max(r, g, b); const min = Math.min(r, g, b); const v = max; const d = max - min; const s = max === 0 ? 0 : d / max; let h; if (max === min) { h = 0; // achromatic } else { switch (max) { case r: h = (g - b) / d + (g < b ? 6 : 0); break; case g: h = (b - r) / d + 2; break; case b: h = (r - g) / d + 4; break; default: throw new Error("unreacheable"); } h /= 6; } return { h, s, v }; } // Hacky way to get RGB values FOR SURE! const nameToRGB = function self(name: string): ColorRGB { if ( namedColors[name] ) { return parseColor(namedColors[name]); } // Create fake div self.fakeDiv = self.fakeDiv || (() => { const tmp = document.createElement("div"); Object.assign(tmp.style, { position: 'fixed'; height: 0; width: 0; top: -99; left: -99; }); document.body.appendChild(tmp); return tmp; })(); self.fakeDiv.style.color = name; // Get color of div const cs = window.getComputedStyle(self.fakeDiv); const pv = cs.color; //TODO slow return parseColor(pv); } // If you also wonder "What the fuck", go here: // https://stackoverflow.com/questions/11068240/what-is-the-most-efficient-way-to-parse-a-css-color-in-javascript function parseColor(input: string): ColorRGB { // Hex format if (input[0] === "#") { const collen = (input.length - 1) / 3; const fact = [17, 1, 0.062272][collen - 1]; return { //TODO refactor with substring r: Math.round(parseInt(input.substr(1, collen), 16) * fact) / 256, g: Math.round(parseInt(input.substr(1 + collen, collen), 16) * fact) / 256, b: Math.round(parseInt(input.substr(1 + 2 * collen, collen), 16) * fact) / 256, }; } if (input.startsWith("rgb")) { // RGB() format const vals = input .split("(")[1] .split(")")[0] .split(",") .map((i) => parseInt(i, 10)) .map(Math.round); return { r: vals[0] / 256, g: vals[1] / 256, b: vals[2] / 256 }; } // Fuck, well.. normalize it and put it into an element and get back to me return nameToRGB(input); } function serializeColor(col: ColorRGB, format: ColorFmt): string { const r = Math.round(col.r * 255); const g = Math.round(col.g * 255); const b = Math.round(col.b * 255); switch (format) { case ColorFmt.RGB: return `rgb(${r}, ${g}, ${b})`; case ColorFmt.HEX: { const rhex = `00${r.toString(16)}`.slice(-2); const ghex = `00${g.toString(16)}`.slice(-2); const bhex = `00${b.toString(16)}`.slice(-2); return `#${rhex}${ghex}${bhex}`; } default: return "#000"; } } export function darken(color: string, format: ColorFmt): string { const col = parseColor(color); const hsl = rgbToHsv(col); if (hsl.s < 0.15) { hsl.h = 0.6; hsl.s = 0.5; hsl.v = Math.max(0.2, 1 - hsl.v); } else if (hsl.v > 0.5) { hsl.v = 0.4; if (hsl.s > 0.2) { hsl.s = Math.min(1, hsl.s + 0.2); } } const out = hsvToRgb(hsl); return serializeColor(out, format); } export function lighten(color: string, format: ColorFmt): string { const col = parseColor(color); const hsl = rgbToHsv(col); if (hsl.v < 0.5) { hsl.v = 0.8; } if (hsl.s > 0.7) { hsl.s = Math.min(1, hsl.s - 0.3); hsl.v = 1; } // Blue is shit to read, make it cyan if (Math.abs(hsl.h - 0.666) < 0.1) { hsl.h -= 0.13; hsl.v = Math.min(1, hsl.v + 0.1); } const out = hsvToRgb(hsl); return serializeColor(out, format); }