mirror of https://github.com/buggins/dlangui.git
Rewrote the decodeHexColor function (#656)
This commit is contained in:
parent
1d5df4d634
commit
f339555061
|
@ -20,7 +20,12 @@ module dlangui.graphics.colors;
|
|||
|
||||
import dlangui.core.types;
|
||||
|
||||
private import std.string : strip;
|
||||
import std.string;
|
||||
import std.algorithm;
|
||||
import std.traits;
|
||||
import std.conv;
|
||||
import std.range;
|
||||
import dimage.jpeg;
|
||||
|
||||
/// special color constant to identify value as not a color (to use default/parent value instead)
|
||||
immutable uint COLOR_UNSPECIFIED = 0xFFDEADFF;
|
||||
|
@ -312,44 +317,95 @@ bool isFullyTransparentColor(uint color) pure nothrow {
|
|||
return (color >> 24) == 0xFF;
|
||||
}
|
||||
|
||||
/// decode color string supported formats: #RGB #ARGB #RRGGBB #AARRGGBB
|
||||
uint decodeHexColor(string s, uint defValue = 0) pure {
|
||||
s = strip(s);
|
||||
switch (s) {
|
||||
case "@null":
|
||||
case "transparent":
|
||||
/// decode color string supported formats: #RGB, #ARGB, #RRGGBB, #AARRGGBB, rgb(r,g,b), rgba(r,g,b,a), rgba(r,g,b,a%)
|
||||
//TODO: the name doesn't match function
|
||||
uint decodeHexColor(string s, uint defValue = 0) { //pure
|
||||
s = s.strip.toLower;
|
||||
if (s.empty)
|
||||
return defValue;
|
||||
if (s == "@null" || s == "transparent")
|
||||
return COLOR_TRANSPARENT;
|
||||
case "black":
|
||||
return 0x000000;
|
||||
case "white":
|
||||
return 0xFFFFFF;
|
||||
case "red":
|
||||
return 0xFF0000;
|
||||
case "green":
|
||||
return 0x00FF00;
|
||||
case "blue":
|
||||
return 0x0000FF;
|
||||
case "gray":
|
||||
return 0x808080;
|
||||
case "lightgray":
|
||||
case "silver":
|
||||
return 0xC0C0C0;
|
||||
default:
|
||||
break;
|
||||
if (s.startsWith("#")) {
|
||||
if (s.length.among(4, 5, 7, 9)) { //#RGB #ARGB #RRGGBB #AARRGGBB
|
||||
s = s[1 .. $];
|
||||
auto color = parse!uint(s, 16); //RGB(A) by default
|
||||
if (s.length == 4)
|
||||
{ //ARGB
|
||||
color = ((color & 0xF00) >> 4) | ((color & 0xF0) << 8) | ((color & 0xF) << 20);
|
||||
}
|
||||
if (s.length != 4 && s.length != 5 && s.length != 7 && s.length != 9)
|
||||
return defValue;
|
||||
if (s[0] != '#')
|
||||
return defValue;
|
||||
uint value = 0;
|
||||
foreach(i; 1 .. s.length) {
|
||||
uint digit = parseHexDigit(s[i]);
|
||||
if (digit == uint.max)
|
||||
return defValue;
|
||||
value = (value << 4) | digit;
|
||||
if (s.length < 7) // double the same digit for short forms
|
||||
value = (value << 4) | digit;
|
||||
else if (s.length == 8)
|
||||
{ //AARRGGBB
|
||||
color = ((color & 0xFF00) >> 8) | ((color & 0xFF) << 24) | (
|
||||
(color & 0xFF0000) >> 8) | ((color & 0xFF000000) >> 24);
|
||||
}
|
||||
return value;
|
||||
return color;
|
||||
}
|
||||
return defValue;
|
||||
}
|
||||
else if (s.startsWith("rgba(") && s.endsWith(")"))
|
||||
{
|
||||
s = s[5 .. $ - 1];
|
||||
auto parts = s.split(",");
|
||||
if (parts.length != 4)
|
||||
return defValue;
|
||||
uint r = to!uint(parts[0].strip).clamp(0, 255);
|
||||
uint g = to!uint(parts[1].strip).clamp(0, 255);
|
||||
uint b = to!uint(parts[2].strip).clamp(0, 255);
|
||||
uint a = 255;
|
||||
auto ap = parts[3].strip;
|
||||
if (ap.endsWith("%")) { //rgba(r,g,b,a%)
|
||||
auto alpha = to!float(ap[0 .. $ - 1].strip);
|
||||
a = cast(uint)((alpha * 255.0 / 100.0) + 0.5).clamp(0, 255);
|
||||
}
|
||||
else { //rgba(r,g,b,a)
|
||||
auto alpha = to!float(parts[3].strip);
|
||||
a = cast(uint)(alpha * 255.0 + 0.5).clamp(0, 255);
|
||||
}
|
||||
if(a == 255)
|
||||
return (r << 16) | (g << 8) | b;
|
||||
return (a == 0) ? 0x00000000 : (a << 24) | (r << 16) | (g << 8) | b;
|
||||
}
|
||||
else if (s.startsWith("rgb(") && s.endsWith(")"))
|
||||
{
|
||||
s = s[4 .. $ - 1];
|
||||
auto parts = s.split(",");
|
||||
if (parts.length != 3)
|
||||
return defValue;
|
||||
uint r = to!uint(parts[0].strip).clamp(0, 255);
|
||||
uint g = to!uint(parts[1].strip).clamp(0, 255);
|
||||
uint b = to!uint(parts[2].strip).clamp(0, 255);
|
||||
return (r << 16) | (g << 8) | b;
|
||||
}
|
||||
foreach (color; __traits(allMembers, Color))
|
||||
{
|
||||
if (color == s)
|
||||
return color.to!Color;
|
||||
}
|
||||
return defValue;
|
||||
}
|
||||
|
||||
unittest
|
||||
{
|
||||
static assert(decodeHexColor("") == 0);
|
||||
static assert(decodeHexColor("@null") == COLOR_TRANSPARENT);
|
||||
static assert(decodeHexColor("trAnsParent") == COLOR_TRANSPARENT);
|
||||
static assert(decodeHexColor("grAy") == 0x808080);
|
||||
static assert(decodeHexColor("#8B008B") == 0x8B008B);
|
||||
static assert(decodeHexColor("#fFf") == 0xfff);
|
||||
static assert(decodeHexColor("#f0F0") == 0xf0f0);
|
||||
static assert(decodeHexColor("#80ff0000") == 0x80ff0000);
|
||||
static assert(decodeHexColor("rgba(255, 0, 0,.5 )") == 0x80ff0000);
|
||||
static assert(decodeHexColor("rgba(255,0, 0, 50%)") == 0x80ff0000);
|
||||
static assert(decodeHexColor("rgba(255,0, 0, 100%)") == 0xff0000);
|
||||
static assert(decodeHexColor("rgba(255,0, 0, 0%)") == 0x00000000);
|
||||
static assert(decodeHexColor("rgb(255,255, 255)") == 0xffffff);
|
||||
static assert(decodeHexColor("rgba(255,0, 0, 150%)") == 0xff0000); // invalid input
|
||||
static assert(decodeHexColor("rgba(255,0, 0, -34%)") == 0x00000000); // invalid input
|
||||
static assert(decodeHexColor("rgb(321,321,321)") == 0xffffff); // invalid input
|
||||
static assert(decodeHexColor("not_valid_color_name") == 0x00000000); // invalid input, return def value
|
||||
static assert(decodeHexColor("#80ff00000") == 0x000000000); // invalid input, return def value
|
||||
static assert(decodeHexColor("#f0") == 0x00000000); // invalid input, return def value
|
||||
static assert(decodeHexColor("rgba(255,255, 255, 10)") == 0xffffff); // invalid input
|
||||
static assert(decodeHexColor("rgba(444,0, 0, -5)") == 0x00000000); // invalid input
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue