mirror of
https://github.com/mountain-loop/yaak.git
synced 2026-01-16 14:06:49 +01:00
95 lines
2.2 KiB
TypeScript
95 lines
2.2 KiB
TypeScript
import parseColor from 'parse-color';
|
|
|
|
export class Color {
|
|
private theme: 'dark' | 'light' = 'light';
|
|
|
|
private hue: number = 0;
|
|
private saturation: number = 0;
|
|
private lightness: number = 0;
|
|
private alpha: number = 1;
|
|
|
|
constructor(cssColor: string, theme: 'dark' | 'light') {
|
|
try {
|
|
const { hsla } = parseColor(cssColor || '');
|
|
this.hue = hsla[0];
|
|
this.saturation = hsla[1];
|
|
this.lightness = hsla[2];
|
|
this.alpha = hsla[3] ?? 1;
|
|
this.theme = theme;
|
|
} catch (err) {
|
|
console.log('Failed to parse CSS color', cssColor, err);
|
|
}
|
|
}
|
|
|
|
static transparent(): Color {
|
|
return new Color('rgb(0,0,0)', 'light').translucify(1);
|
|
}
|
|
|
|
static white(): Color {
|
|
return new Color('rgb(0,0,0)', 'light').lower(1);
|
|
}
|
|
|
|
static black(): Color {
|
|
return new Color('rgb(0,0,0)', 'light').lift(1);
|
|
}
|
|
|
|
private clone(): Color {
|
|
return new Color(this.css(), this.theme);
|
|
}
|
|
|
|
lower(mod: number): Color {
|
|
return this.theme === 'dark' ? this._darken(mod) : this._lighten(mod);
|
|
}
|
|
|
|
lift(mod: number): Color {
|
|
return this.theme === 'dark' ? this._lighten(mod) : this._darken(mod);
|
|
}
|
|
|
|
translucify(mod: number): Color {
|
|
const c = this.clone();
|
|
c.alpha = c.alpha - c.alpha * mod;
|
|
return c;
|
|
}
|
|
|
|
desaturate(mod: number): Color {
|
|
const c = this.clone();
|
|
c.saturation = c.saturation - c.saturation * mod;
|
|
return c;
|
|
}
|
|
|
|
saturate(mod: number): Color {
|
|
const c = this.clone();
|
|
c.saturation = this.saturation + (100 - this.saturation) * mod;
|
|
return c;
|
|
}
|
|
|
|
lighterThan(c: Color): boolean {
|
|
return this.lightness > c.lightness;
|
|
}
|
|
|
|
css(): string {
|
|
// If opacity is 1, allow for Tailwind modification
|
|
const h = Math.round(this.hue);
|
|
const s = Math.round(this.saturation);
|
|
const l = Math.round(this.lightness);
|
|
const a = Math.round(this.alpha * 100) / 100;
|
|
return `hsla(${h}, ${s}%, ${l}%, ${a})`;
|
|
}
|
|
|
|
hex(): string {
|
|
return parseColor(this.css()).hex;
|
|
}
|
|
|
|
private _lighten(mod: number): Color {
|
|
const c = this.clone();
|
|
c.lightness = this.lightness + (100 - this.lightness) * mod;
|
|
return c;
|
|
}
|
|
|
|
private _darken(mod: number): Color {
|
|
const c = this.clone();
|
|
c.lightness = this.lightness - this.lightness * mod;
|
|
return c;
|
|
}
|
|
}
|