mirror of https://github.com/adamdruppe/arsd.git
toying with oklab colors. i dont love it
This commit is contained in:
parent
6cd77bb283
commit
f7e1a4e062
226
color.d
226
color.d
|
@ -218,10 +218,36 @@ struct Color {
|
||||||
Added July 18, 2022 (dub v10.9)
|
Added July 18, 2022 (dub v10.9)
|
||||||
+/
|
+/
|
||||||
nothrow pure @nogc
|
nothrow pure @nogc
|
||||||
this(ubyte[] components) {
|
this(scope ubyte[] components) {
|
||||||
this.components[] = components[0 .. 4];
|
this.components[] = components[0 .. 4];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
Constructs a color from floating-point rgba components, each between 0 and 1.0.
|
||||||
|
|
||||||
|
History:
|
||||||
|
Added December 1, 2022 (dub v10.10)
|
||||||
|
+/
|
||||||
|
this(float r, float g, float b, float a = 1.0) {
|
||||||
|
if(r < 0) r = 0;
|
||||||
|
if(g < 0) g = 0;
|
||||||
|
if(b < 0) b = 0;
|
||||||
|
if(r > 1) r = 1;
|
||||||
|
if(g > 1) g = 1;
|
||||||
|
if(b > 1) b = 1;
|
||||||
|
/*
|
||||||
|
import std.conv;
|
||||||
|
assert(r >= 0.0 && r <= 1.0, to!string(r));
|
||||||
|
assert(g >= 0.0 && g <= 1.0, to!string(g));
|
||||||
|
assert(b >= 0.0 && b <= 1.0, to!string(b));
|
||||||
|
assert(a >= 0.0 && a <= 1.0, to!string(a));
|
||||||
|
*/
|
||||||
|
this.r = cast(ubyte) (r * 255);
|
||||||
|
this.g = cast(ubyte) (g * 255);
|
||||||
|
this.b = cast(ubyte) (b * 255);
|
||||||
|
this.a = cast(ubyte) (a * 255);
|
||||||
|
}
|
||||||
|
|
||||||
/// Static convenience functions for common color names
|
/// Static convenience functions for common color names
|
||||||
nothrow pure @nogc
|
nothrow pure @nogc
|
||||||
static Color transparent() { return Color(0, 0, 0, 0); }
|
static Color transparent() { return Color(0, 0, 0, 0); }
|
||||||
|
@ -502,6 +528,204 @@ struct Color {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
OKLab colorspace conversions to/from [Color]. See: [https://bottosson.github.io/posts/oklab/]
|
||||||
|
|
||||||
|
L = perceived lightness. From 0 to 1.0.
|
||||||
|
|
||||||
|
a = how green/red the color is. Apparently supposed to be from -.233 to .276
|
||||||
|
|
||||||
|
b = how blue/yellow the color is. Apparently supposed to be from -.311 to 0.198.
|
||||||
|
|
||||||
|
History:
|
||||||
|
Added December 1, 2022 (dub v10.10)
|
||||||
|
|
||||||
|
Bugs:
|
||||||
|
Seems to be some but i might just not understand what the result is supposed to be.
|
||||||
|
+/
|
||||||
|
struct Lab {
|
||||||
|
float L = 0.0;
|
||||||
|
float a = 0.0;
|
||||||
|
float b = 0.0;
|
||||||
|
float alpha = 1.0;
|
||||||
|
|
||||||
|
float C() const {
|
||||||
|
import core.stdc.math;
|
||||||
|
return sqrtf(a * a + b * b);
|
||||||
|
}
|
||||||
|
|
||||||
|
float h() const {
|
||||||
|
import core.stdc.math;
|
||||||
|
return atan2f(b, a);
|
||||||
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
L's useful range is between 0 and 1.0
|
||||||
|
|
||||||
|
C's useful range is between 0 and 0.4
|
||||||
|
|
||||||
|
H can be 0 to 360 for degrees, or 0 to 2pi for radians.
|
||||||
|
+/
|
||||||
|
static Lab fromLChDegrees(float L, float C, float h, float alpha = 1.0) {
|
||||||
|
return fromLChRadians(L, C, h * 3.14159265358979323f / 180.0f, alpha);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// ditto
|
||||||
|
static Lab fromLChRadians(float L, float C, float h, float alpha = 1.0) {
|
||||||
|
import core.stdc.math;
|
||||||
|
// if(C > 0.4) C = 0.4;
|
||||||
|
return Lab(L, C * cosf(h), C * sinf(h), alpha);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// ditto
|
||||||
|
Lab toOklab(Color c) {
|
||||||
|
import core.stdc.math;
|
||||||
|
|
||||||
|
// this algorithm requires linear sRGB
|
||||||
|
|
||||||
|
float f(float w) {
|
||||||
|
w = srbgToLinear(w);
|
||||||
|
if(w < 0)
|
||||||
|
w = 0;
|
||||||
|
if(w > 1)
|
||||||
|
w = 1;
|
||||||
|
return w;
|
||||||
|
}
|
||||||
|
|
||||||
|
float r = f(cast(float) c.r / 255);
|
||||||
|
float g = f(cast(float) c.g / 255);
|
||||||
|
float b = f(cast(float) c.b / 255);
|
||||||
|
|
||||||
|
float l = 0.4122214708f * r + 0.5363325363f * g + 0.0514459929f * b;
|
||||||
|
float m = 0.2119034982f * r + 0.6806995451f * g + 0.1073969566f * b;
|
||||||
|
float s = 0.0883024619f * r + 0.2817188376f * g + 0.6299787005f * b;
|
||||||
|
|
||||||
|
float l_ = cbrtf(l);
|
||||||
|
float m_ = cbrtf(m);
|
||||||
|
float s_ = cbrtf(s);
|
||||||
|
|
||||||
|
return Lab(
|
||||||
|
0.2104542553f*l_ + 0.7936177850f*m_ - 0.0040720468f*s_,
|
||||||
|
1.9779984951f*l_ - 2.4285922050f*m_ + 0.4505937099f*s_,
|
||||||
|
0.0259040371f*l_ + 0.7827717662f*m_ - 0.8086757660f*s_,
|
||||||
|
cast(float) c.a / 255
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// ditto
|
||||||
|
Color fromOklab(Lab c) {
|
||||||
|
float l_ = c.L + 0.3963377774f * c.a + 0.2158037573f * c.b;
|
||||||
|
float m_ = c.L - 0.1055613458f * c.a - 0.0638541728f * c.b;
|
||||||
|
float s_ = c.L - 0.0894841775f * c.a - 1.2914855480f * c.b;
|
||||||
|
|
||||||
|
float l = l_*l_*l_;
|
||||||
|
float m = m_*m_*m_;
|
||||||
|
float s = s_*s_*s_;
|
||||||
|
|
||||||
|
float f(float w) {
|
||||||
|
w = linearToSrbg(w);
|
||||||
|
if(w < 0)
|
||||||
|
w = 0;
|
||||||
|
if(w > 1)
|
||||||
|
w = 1;
|
||||||
|
return w;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Color(
|
||||||
|
f(+4.0767416621f * l - 3.3077115913f * m + 0.2309699292f * s),
|
||||||
|
f(-1.2684380046f * l + 2.6097574011f * m - 0.3413193965f * s),
|
||||||
|
f(-0.0041960863f * l - 0.7034186147f * m + 1.7076147010f * s),
|
||||||
|
c.alpha
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// from https://bottosson.github.io/posts/colorwrong/#what-can-we-do%3F
|
||||||
|
float linearToSrbg(float x) { // aka f
|
||||||
|
import core.stdc.math;
|
||||||
|
if (x >= 0.0031308)
|
||||||
|
return (1.055) * powf(x, (1.0/2.4)) - 0.055;
|
||||||
|
else
|
||||||
|
return 12.92 * x;
|
||||||
|
}
|
||||||
|
|
||||||
|
float srbgToLinear(float x) { // aka f_inv
|
||||||
|
import core.stdc.math;
|
||||||
|
if (x >= 0.04045)
|
||||||
|
return powf((x + 0.055)/(1 + 0.055), 2.4);
|
||||||
|
else
|
||||||
|
return x / 12.92;
|
||||||
|
}
|
||||||
|
|
||||||
|
/+
|
||||||
|
float[3] colorToYCbCr(Color c) {
|
||||||
|
return matrixMultiply(
|
||||||
|
[
|
||||||
|
+0.2126, +0.7152, +0.0722,
|
||||||
|
-0.1146, -0.3854, +0.5000,
|
||||||
|
+0.5000, -0.4542, -0.0458
|
||||||
|
],
|
||||||
|
[float(c.r) / 255, float(c.g) / 255, float(c.b) / 255]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Color YCbCrToColor(float Y, float Cb, float Cr) {
|
||||||
|
|
||||||
|
/*
|
||||||
|
Y = Y * 255;
|
||||||
|
Cb = Cb * 255;
|
||||||
|
Cr = Cr * 255;
|
||||||
|
|
||||||
|
int r = cast(int) (Y + 1.40200 * (Cr - 0x80));
|
||||||
|
int g = cast(int) (Y - 0.34414 * (Cb - 0x80) - 0.71414 * (Cr - 0x80));
|
||||||
|
int b = cast(int) (Y + 1.77200 * (Cb - 0x80));
|
||||||
|
|
||||||
|
void clamp(ref int item, int min, int max) {
|
||||||
|
if(item < min) item = min;
|
||||||
|
if(item > max) item = max;
|
||||||
|
}
|
||||||
|
|
||||||
|
clamp(r, 0, 255);
|
||||||
|
clamp(g, 0, 255);
|
||||||
|
clamp(b, 0, 255);
|
||||||
|
return Color(r, g, b);
|
||||||
|
*/
|
||||||
|
|
||||||
|
float f(float w) {
|
||||||
|
if(w < 0 || w > 1)
|
||||||
|
return 0;
|
||||||
|
assert(w >= 0.0);
|
||||||
|
assert(w <= 1.0);
|
||||||
|
//w = linearToSrbg(w);
|
||||||
|
if(w < 0)
|
||||||
|
w = 0;
|
||||||
|
if(w > 1)
|
||||||
|
w = 1;
|
||||||
|
return w;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto rgb = matrixMultiply(
|
||||||
|
[
|
||||||
|
1, +0.0000, +1.5748,
|
||||||
|
1, -0.1873, -0.4681,
|
||||||
|
1, +1.8556, +0.0000
|
||||||
|
],
|
||||||
|
[Y, Cb, Cr]
|
||||||
|
);
|
||||||
|
|
||||||
|
return Color(f(rgb[0]), f(rgb[1]), f(rgb[2]));
|
||||||
|
}
|
||||||
|
|
||||||
|
private float[3] matrixMultiply(float[9] matrix, float[3] vector) {
|
||||||
|
return [
|
||||||
|
matrix[0] * vector[0] + matrix[1] * vector[1] + matrix[2] * vector[2],
|
||||||
|
matrix[3] * vector[0] + matrix[4] * vector[1] + matrix[5] * vector[2],
|
||||||
|
matrix[6] * vector[0] + matrix[7] * vector[1] + matrix[8] * vector[2],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
+/
|
||||||
|
|
||||||
void premultiplyBgra(ubyte[] bgra) pure @nogc @safe nothrow in { assert(bgra.length == 4); } do {
|
void premultiplyBgra(ubyte[] bgra) pure @nogc @safe nothrow in { assert(bgra.length == 4); } do {
|
||||||
auto a = bgra[3];
|
auto a = bgra[3];
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue