mirror of https://github.com/adamdruppe/arsd.git
real sux
This commit is contained in:
parent
63c3cf8a8a
commit
49341201fe
97
color.d
97
color.d
|
@ -6,8 +6,8 @@ module arsd.color;
|
||||||
// importing phobos explodes the size of this code 10x, so not doing it.
|
// importing phobos explodes the size of this code 10x, so not doing it.
|
||||||
|
|
||||||
private {
|
private {
|
||||||
real toInternal(T)(scope const(char)[] s) {
|
double toInternal(T)(scope const(char)[] s) {
|
||||||
real accumulator = 0.0;
|
double accumulator = 0.0;
|
||||||
size_t i = s.length;
|
size_t i = s.length;
|
||||||
foreach(idx, c; s) {
|
foreach(idx, c; s) {
|
||||||
if(c >= '0' && c <= '9') {
|
if(c >= '0' && c <= '9') {
|
||||||
|
@ -17,21 +17,21 @@ private {
|
||||||
i = idx + 1;
|
i = idx + 1;
|
||||||
break;
|
break;
|
||||||
} else {
|
} else {
|
||||||
string wtfIsWrongWithThisStupidLanguageWithItsBrokenSafeAttribute = "bad char to make real from ";
|
string wtfIsWrongWithThisStupidLanguageWithItsBrokenSafeAttribute = "bad char to make double from ";
|
||||||
wtfIsWrongWithThisStupidLanguageWithItsBrokenSafeAttribute ~= s;
|
wtfIsWrongWithThisStupidLanguageWithItsBrokenSafeAttribute ~= s;
|
||||||
throw new Exception(wtfIsWrongWithThisStupidLanguageWithItsBrokenSafeAttribute);
|
throw new Exception(wtfIsWrongWithThisStupidLanguageWithItsBrokenSafeAttribute);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
real accumulator2 = 0.0;
|
double accumulator2 = 0.0;
|
||||||
real count = 1;
|
double count = 1;
|
||||||
foreach(c; s[i .. $]) {
|
foreach(c; s[i .. $]) {
|
||||||
if(c >= '0' && c <= '9') {
|
if(c >= '0' && c <= '9') {
|
||||||
accumulator2 *= 10;
|
accumulator2 *= 10;
|
||||||
accumulator2 += c - '0';
|
accumulator2 += c - '0';
|
||||||
count *= 10;
|
count *= 10;
|
||||||
} else {
|
} else {
|
||||||
string wtfIsWrongWithThisStupidLanguageWithItsBrokenSafeAttribute = "bad char to make real from ";
|
string wtfIsWrongWithThisStupidLanguageWithItsBrokenSafeAttribute = "bad char to make double from ";
|
||||||
wtfIsWrongWithThisStupidLanguageWithItsBrokenSafeAttribute ~= s;
|
wtfIsWrongWithThisStupidLanguageWithItsBrokenSafeAttribute ~= s;
|
||||||
throw new Exception(wtfIsWrongWithThisStupidLanguageWithItsBrokenSafeAttribute);
|
throw new Exception(wtfIsWrongWithThisStupidLanguageWithItsBrokenSafeAttribute);
|
||||||
}
|
}
|
||||||
|
@ -64,8 +64,8 @@ private {
|
||||||
|
|
||||||
return cast(string) ret;
|
return cast(string) ret;
|
||||||
}
|
}
|
||||||
string toInternal(T)(real a) {
|
string toInternal(T)(double a) {
|
||||||
// a simplifying assumption here is the fact that we only use this in one place: toInternal!string(cast(real) a / 255)
|
// a simplifying assumption here is the fact that we only use this in one place: toInternal!string(cast(double) a / 255)
|
||||||
// thus we know this will always be between 0.0 and 1.0, inclusive.
|
// thus we know this will always be between 0.0 and 1.0, inclusive.
|
||||||
if(a <= 0.0)
|
if(a <= 0.0)
|
||||||
return "0.0";
|
return "0.0";
|
||||||
|
@ -78,16 +78,16 @@ private {
|
||||||
}
|
}
|
||||||
|
|
||||||
nothrow @safe @nogc pure
|
nothrow @safe @nogc pure
|
||||||
real absInternal(real a) { return a < 0 ? -a : a; }
|
double absInternal(double a) { return a < 0 ? -a : a; }
|
||||||
nothrow @safe @nogc pure
|
nothrow @safe @nogc pure
|
||||||
real minInternal(real a, real b, real c) {
|
double minInternal(double a, double b, double c) {
|
||||||
auto m = a;
|
auto m = a;
|
||||||
if(b < m) m = b;
|
if(b < m) m = b;
|
||||||
if(c < m) m = c;
|
if(c < m) m = c;
|
||||||
return m;
|
return m;
|
||||||
}
|
}
|
||||||
nothrow @safe @nogc pure
|
nothrow @safe @nogc pure
|
||||||
real maxInternal(real a, real b, real c) {
|
double maxInternal(double a, double b, double c) {
|
||||||
auto m = a;
|
auto m = a;
|
||||||
if(b > m) m = b;
|
if(b > m) m = b;
|
||||||
if(c > m) m = c;
|
if(c > m) m = c;
|
||||||
|
@ -226,7 +226,7 @@ struct Color {
|
||||||
if(a == 255)
|
if(a == 255)
|
||||||
return "#" ~ toHexInternal(r) ~ toHexInternal(g) ~ toHexInternal(b);
|
return "#" ~ toHexInternal(r) ~ toHexInternal(g) ~ toHexInternal(b);
|
||||||
else {
|
else {
|
||||||
return "rgba("~toInternal!string(r)~", "~toInternal!string(g)~", "~toInternal!string(b)~", "~toInternal!string(cast(real)a / 255.0)~")";
|
return "rgba("~toInternal!string(r)~", "~toInternal!string(g)~", "~toInternal!string(b)~", "~toInternal!string(cast(double)a / 255.0)~")";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -277,15 +277,15 @@ struct Color {
|
||||||
assert(s[$-1] == ')');
|
assert(s[$-1] == ')');
|
||||||
s = s[s.startsWithInternal("hsl(") ? 4 : 5 .. $ - 1]; // the closing paren
|
s = s[s.startsWithInternal("hsl(") ? 4 : 5 .. $ - 1]; // the closing paren
|
||||||
|
|
||||||
real[3] hsl;
|
double[3] hsl;
|
||||||
ubyte a = 255;
|
ubyte a = 255;
|
||||||
|
|
||||||
auto parts = s.splitInternal(',');
|
auto parts = s.splitInternal(',');
|
||||||
foreach(i, part; parts) {
|
foreach(i, part; parts) {
|
||||||
if(i < 3)
|
if(i < 3)
|
||||||
hsl[i] = toInternal!real(part.stripInternal);
|
hsl[i] = toInternal!double(part.stripInternal);
|
||||||
else
|
else
|
||||||
a = clampToByte(cast(int) (toInternal!real(part.stripInternal) * 255));
|
a = clampToByte(cast(int) (toInternal!double(part.stripInternal) * 255));
|
||||||
}
|
}
|
||||||
|
|
||||||
c = .fromHsl(hsl);
|
c = .fromHsl(hsl);
|
||||||
|
@ -302,7 +302,7 @@ struct Color {
|
||||||
auto parts = s.splitInternal(',');
|
auto parts = s.splitInternal(',');
|
||||||
foreach(i, part; parts) {
|
foreach(i, part; parts) {
|
||||||
// lol the loop-switch pattern
|
// lol the loop-switch pattern
|
||||||
auto v = toInternal!real(part.stripInternal);
|
auto v = toInternal!double(part.stripInternal);
|
||||||
switch(i) {
|
switch(i) {
|
||||||
case 0: // red
|
case 0: // red
|
||||||
c.r = clampToByte(cast(int) v);
|
c.r = clampToByte(cast(int) v);
|
||||||
|
@ -353,7 +353,7 @@ struct Color {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// from hsl
|
/// from hsl
|
||||||
static Color fromHsl(real h, real s, real l) {
|
static Color fromHsl(double h, double s, double l) {
|
||||||
return .fromHsl(h, s, l);
|
return .fromHsl(h, s, l);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -460,22 +460,26 @@ private ubyte fromHexInternal(in char[] s) {
|
||||||
|
|
||||||
/// Converts hsl to rgb
|
/// Converts hsl to rgb
|
||||||
Color fromHsl(real[3] hsl) nothrow pure @safe @nogc {
|
Color fromHsl(real[3] hsl) nothrow pure @safe @nogc {
|
||||||
|
return fromHsl(cast(double) hsl[0], cast(double) hsl[1], cast(double) hsl[2]);
|
||||||
|
}
|
||||||
|
|
||||||
|
Color fromHsl(double[3] hsl) nothrow pure @safe @nogc {
|
||||||
return fromHsl(hsl[0], hsl[1], hsl[2]);
|
return fromHsl(hsl[0], hsl[1], hsl[2]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Converts hsl to rgb
|
/// Converts hsl to rgb
|
||||||
Color fromHsl(real h, real s, real l, real a = 255) nothrow pure @safe @nogc {
|
Color fromHsl(double h, double s, double l, double a = 255) nothrow pure @safe @nogc {
|
||||||
h = h % 360;
|
h = h % 360;
|
||||||
|
|
||||||
real C = (1 - absInternal(2 * l - 1)) * s;
|
double C = (1 - absInternal(2 * l - 1)) * s;
|
||||||
|
|
||||||
real hPrime = h / 60;
|
double hPrime = h / 60;
|
||||||
|
|
||||||
real X = C * (1 - absInternal(hPrime % 2 - 1));
|
double X = C * (1 - absInternal(hPrime % 2 - 1));
|
||||||
|
|
||||||
real r, g, b;
|
double r, g, b;
|
||||||
|
|
||||||
if(h is real.nan)
|
if(h is double.nan)
|
||||||
r = g = b = 0;
|
r = g = b = 0;
|
||||||
else if (hPrime >= 0 && hPrime < 1) {
|
else if (hPrime >= 0 && hPrime < 1) {
|
||||||
r = C;
|
r = C;
|
||||||
|
@ -503,7 +507,7 @@ Color fromHsl(real h, real s, real l, real a = 255) nothrow pure @safe @nogc {
|
||||||
b = X;
|
b = X;
|
||||||
}
|
}
|
||||||
|
|
||||||
real m = l - C / 2;
|
double m = l - C / 2;
|
||||||
|
|
||||||
r += m;
|
r += m;
|
||||||
g += m;
|
g += m;
|
||||||
|
@ -517,22 +521,23 @@ Color fromHsl(real h, real s, real l, real a = 255) nothrow pure @safe @nogc {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Converts an RGB color into an HSL triplet. useWeightedLightness will try to get a better value for luminosity for the human eye, which is more sensitive to green than red and more to red than blue. If it is false, it just does average of the rgb.
|
/// Converts an RGB color into an HSL triplet. useWeightedLightness will try to get a better value for luminosity for the human eye, which is more sensitive to green than red and more to red than blue. If it is false, it just does average of the rgb.
|
||||||
real[3] toHsl(Color c, bool useWeightedLightness = false) nothrow pure @trusted @nogc {
|
double[3] toHsl(Color c, bool useWeightedLightness = false) nothrow pure @trusted @nogc {
|
||||||
real r1 = cast(real) c.r / 255;
|
double r1 = cast(double) c.r / 255;
|
||||||
real g1 = cast(real) c.g / 255;
|
double g1 = cast(double) c.g / 255;
|
||||||
real b1 = cast(real) c.b / 255;
|
double b1 = cast(double) c.b / 255;
|
||||||
|
|
||||||
real maxColor = maxInternal(r1, g1, b1);
|
double maxColor = maxInternal(r1, g1, b1);
|
||||||
real minColor = minInternal(r1, g1, b1);
|
double minColor = minInternal(r1, g1, b1);
|
||||||
|
|
||||||
real L = (maxColor + minColor) / 2 ;
|
double L = (maxColor + minColor) / 2 ;
|
||||||
if(useWeightedLightness) {
|
if(useWeightedLightness) {
|
||||||
// the colors don't affect the eye equally
|
// the colors don't affect the eye equally
|
||||||
// this is a little more accurate than plain HSL numbers
|
// this is a little more accurate than plain HSL numbers
|
||||||
L = 0.2126*r1 + 0.7152*g1 + 0.0722*b1;
|
L = 0.2126*r1 + 0.7152*g1 + 0.0722*b1;
|
||||||
|
// maybe a better number is 299, 587, 114
|
||||||
}
|
}
|
||||||
real S = 0;
|
double S = 0;
|
||||||
real H = 0;
|
double H = 0;
|
||||||
if(maxColor != minColor) {
|
if(maxColor != minColor) {
|
||||||
if(L < 0.5) {
|
if(L < 0.5) {
|
||||||
S = (maxColor - minColor) / (maxColor + minColor);
|
S = (maxColor - minColor) / (maxColor + minColor);
|
||||||
|
@ -557,7 +562,7 @@ real[3] toHsl(Color c, bool useWeightedLightness = false) nothrow pure @trusted
|
||||||
}
|
}
|
||||||
|
|
||||||
/// .
|
/// .
|
||||||
Color lighten(Color c, real percentage) nothrow pure @safe @nogc {
|
Color lighten(Color c, double percentage) nothrow pure @safe @nogc {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[2] *= (1 + percentage);
|
hsl[2] *= (1 + percentage);
|
||||||
if(hsl[2] > 1)
|
if(hsl[2] > 1)
|
||||||
|
@ -566,7 +571,7 @@ Color lighten(Color c, real percentage) nothrow pure @safe @nogc {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// .
|
/// .
|
||||||
Color darken(Color c, real percentage) nothrow pure @safe @nogc {
|
Color darken(Color c, double percentage) nothrow pure @safe @nogc {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[2] *= (1 - percentage);
|
hsl[2] *= (1 - percentage);
|
||||||
return fromHsl(hsl);
|
return fromHsl(hsl);
|
||||||
|
@ -574,7 +579,7 @@ Color darken(Color c, real percentage) nothrow pure @safe @nogc {
|
||||||
|
|
||||||
/// for light colors, call darken. for dark colors, call lighten.
|
/// for light colors, call darken. for dark colors, call lighten.
|
||||||
/// The goal: get toward center grey.
|
/// The goal: get toward center grey.
|
||||||
Color moderate(Color c, real percentage) nothrow pure @safe @nogc {
|
Color moderate(Color c, double percentage) nothrow pure @safe @nogc {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
if(hsl[2] > 0.5)
|
if(hsl[2] > 0.5)
|
||||||
hsl[2] *= (1 - percentage);
|
hsl[2] *= (1 - percentage);
|
||||||
|
@ -590,7 +595,7 @@ Color moderate(Color c, real percentage) nothrow pure @safe @nogc {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// the opposite of moderate. Make darks darker and lights lighter
|
/// the opposite of moderate. Make darks darker and lights lighter
|
||||||
Color extremify(Color c, real percentage) nothrow pure @safe @nogc {
|
Color extremify(Color c, double percentage) nothrow pure @safe @nogc {
|
||||||
auto hsl = toHsl(c, true);
|
auto hsl = toHsl(c, true);
|
||||||
if(hsl[2] < 0.5)
|
if(hsl[2] < 0.5)
|
||||||
hsl[2] *= (1 - percentage);
|
hsl[2] *= (1 - percentage);
|
||||||
|
@ -626,7 +631,7 @@ Color makeTextColor(Color c) nothrow pure @safe @nogc {
|
||||||
|
|
||||||
// These provide functional access to hsl manipulation; useful if you need a delegate
|
// These provide functional access to hsl manipulation; useful if you need a delegate
|
||||||
|
|
||||||
Color setLightness(Color c, real lightness) nothrow pure @safe @nogc {
|
Color setLightness(Color c, double lightness) nothrow pure @safe @nogc {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[2] = lightness;
|
hsl[2] = lightness;
|
||||||
return fromHsl(hsl);
|
return fromHsl(hsl);
|
||||||
|
@ -634,28 +639,28 @@ Color setLightness(Color c, real lightness) nothrow pure @safe @nogc {
|
||||||
|
|
||||||
|
|
||||||
///
|
///
|
||||||
Color rotateHue(Color c, real degrees) nothrow pure @safe @nogc {
|
Color rotateHue(Color c, double degrees) nothrow pure @safe @nogc {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[0] += degrees;
|
hsl[0] += degrees;
|
||||||
return fromHsl(hsl);
|
return fromHsl(hsl);
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
Color setHue(Color c, real hue) nothrow pure @safe @nogc {
|
Color setHue(Color c, double hue) nothrow pure @safe @nogc {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[0] = hue;
|
hsl[0] = hue;
|
||||||
return fromHsl(hsl);
|
return fromHsl(hsl);
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
Color desaturate(Color c, real percentage) nothrow pure @safe @nogc {
|
Color desaturate(Color c, double percentage) nothrow pure @safe @nogc {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[1] *= (1 - percentage);
|
hsl[1] *= (1 - percentage);
|
||||||
return fromHsl(hsl);
|
return fromHsl(hsl);
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
Color saturate(Color c, real percentage) nothrow pure @safe @nogc {
|
Color saturate(Color c, double percentage) nothrow pure @safe @nogc {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[1] *= (1 + percentage);
|
hsl[1] *= (1 + percentage);
|
||||||
if(hsl[1] > 1)
|
if(hsl[1] > 1)
|
||||||
|
@ -664,7 +669,7 @@ Color saturate(Color c, real percentage) nothrow pure @safe @nogc {
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
Color setSaturation(Color c, real saturation) nothrow pure @safe @nogc {
|
Color setSaturation(Color c, double saturation) nothrow pure @safe @nogc {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[1] = saturation;
|
hsl[1] = saturation;
|
||||||
return fromHsl(hsl);
|
return fromHsl(hsl);
|
||||||
|
@ -785,9 +790,9 @@ void main() {
|
||||||
import browser.document;
|
import browser.document;
|
||||||
foreach(ele; document.querySelectorAll("input")) {
|
foreach(ele; document.querySelectorAll("input")) {
|
||||||
ele.addEventListener("change", {
|
ele.addEventListener("change", {
|
||||||
auto h = toInternal!real(document.querySelector("input[name=h]").value);
|
auto h = toInternal!double(document.querySelector("input[name=h]").value);
|
||||||
auto s = toInternal!real(document.querySelector("input[name=s]").value);
|
auto s = toInternal!double(document.querySelector("input[name=s]").value);
|
||||||
auto l = toInternal!real(document.querySelector("input[name=l]").value);
|
auto l = toInternal!double(document.querySelector("input[name=l]").value);
|
||||||
|
|
||||||
Color c = Color.fromHsl(h, s, l);
|
Color c = Color.fromHsl(h, s, l);
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue