|
|
|
|
|
|
|
//
|
|
|
|
// color.cc
|
|
|
|
//
|
|
|
|
// Copyright (c) 2010 LearnBoost <tj@learnboost.com>
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "color.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Consume whitespace.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define WHITESPACE \
|
|
|
|
while (' ' == *str) ++str;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Parse color channel value
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define CHANNEL(NAME) \
|
|
|
|
if (*str >= '0' && *str <= '9') { \
|
|
|
|
do { \
|
|
|
|
NAME *= 10; \
|
|
|
|
NAME += *str++ - '0'; \
|
|
|
|
} while (*str >= '0' && *str <= '9'); \
|
|
|
|
} else { \
|
|
|
|
return 0; \
|
|
|
|
} \
|
|
|
|
while (' ' == *str || ',' == *str) str++;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Named colors.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static struct named_color {
|
|
|
|
const char *name;
|
|
|
|
uint32_t val;
|
|
|
|
} named_colors[] = {
|
|
|
|
{"transparent", 0xffffff00}
|
|
|
|
, {"aliceblue", 0xf0f8ffff}
|
|
|
|
, {"antiquewhite", 0xfaebd7ff}
|
|
|
|
, {"aqua", 0x00ffffff}
|
|
|
|
, {"aquamarine", 0x7fffd4ff}
|
|
|
|
, {"azure", 0xf0ffffff}
|
|
|
|
, {"beige", 0xf5f5dcff}
|
|
|
|
, {"bisque", 0xffe4c4ff}
|
|
|
|
, {"black", 0x000000ff}
|
|
|
|
, {"blanchedalmond", 0xffebcdff}
|
|
|
|
, {"blue", 0x0000ffff}
|
|
|
|
, {"blueviolet", 0x8a2be2ff}
|
|
|
|
, {"brown", 0xa52a2aff}
|
|
|
|
, {"burlywood", 0xdeb887ff}
|
|
|
|
, {"cadetblue", 0x5f9ea0ff}
|
|
|
|
, {"chartreuse", 0x7fff00ff}
|
|
|
|
, {"chocolate", 0xd2691eff}
|
|
|
|
, {"coral", 0xff7f50ff}
|
|
|
|
, {"cornflowerblue", 0x6495edff}
|
|
|
|
, {"cornsilk", 0xfff8dcff}
|
|
|
|
, {"crimson", 0xdc143cff}
|
|
|
|
, {"cyan", 0x00ffffff}
|
|
|
|
, {"darkblue", 0x00008bff}
|
|
|
|
, {"darkcyan", 0x008b8bff}
|
|
|
|
, {"darkgoldenrod", 0xb8860bff}
|
|
|
|
, {"darkgray", 0xa9a9a9ff}
|
|
|
|
, {"darkgreen", 0x006400ff}
|
|
|
|
, {"darkkhaki", 0xbdb76bff}
|
|
|
|
, {"darkmagenta", 0x8b008bff}
|
|
|
|
, {"darkolivegreen", 0x556b2fff}
|
|
|
|
, {"darkorange", 0xff8c00ff}
|
|
|
|
, {"darkorchid", 0x9932ccff}
|
|
|
|
, {"darkred", 0x8b0000ff}
|
|
|
|
, {"darksalmon", 0xe9967aff}
|
|
|
|
, {"darkseagreen", 0x8fbc8fff}
|
|
|
|
, {"darkslateblue", 0x483d8bff}
|
|
|
|
, {"darkslategray", 0x2f4f4fff}
|
|
|
|
, {"darkturquoise", 0x00ced1ff}
|
|
|
|
, {"darkviolet", 0x9400d3ff}
|
|
|
|
, {"deeppink", 0xff1493ff}
|
|
|
|
, {"deepskyblue", 0x00bfffff}
|
|
|
|
, {"dimgray", 0x696969ff}
|
|
|
|
, {"dodgerblue", 0x1e90ffff}
|
|
|
|
, {"feldspar", 0xd19275ff}
|
|
|
|
, {"firebrick", 0xb22222ff}
|
|
|
|
, {"floralwhite", 0xfffaf0ff}
|
|
|
|
, {"forestgreen", 0x228b22ff}
|
|
|
|
, {"fuchsia", 0xff00ffff}
|
|
|
|
, {"gainsboro", 0xdcdcdcff}
|
|
|
|
, {"ghostwhite", 0xf8f8ffff}
|
|
|
|
, {"gold", 0xffd700ff}
|
|
|
|
, {"goldenrod", 0xdaa520ff}
|
|
|
|
, {"gray", 0x808080ff}
|
|
|
|
, {"green", 0x008000ff}
|
|
|
|
, {"greenyellow", 0xadff2fff}
|
|
|
|
, {"honeydew", 0xf0fff0ff}
|
|
|
|
, {"hotpink", 0xff69b4ff}
|
|
|
|
, {"indianred ", 0xcd5c5cff}
|
|
|
|
, {"indigo ", 0x4b0082ff}
|
|
|
|
, {"ivory", 0xfffff0ff}
|
|
|
|
, {"khaki", 0xf0e68cff}
|
|
|
|
, {"lavender", 0xe6e6faff}
|
|
|
|
, {"lavenderblush", 0xfff0f5ff}
|
|
|
|
, {"lawngreen", 0x7cfc00ff}
|
|
|
|
, {"lemonchiffon", 0xfffacdff}
|
|
|
|
, {"lightblue", 0xadd8e6ff}
|
|
|
|
, {"lightcoral", 0xf08080ff}
|
|
|
|
, {"lightcyan", 0xe0ffffff}
|
|
|
|
, {"lightgoldenrodyellow", 0xfafa}
|
|
|
|
, {"lightgrey", 0xd3d3d3ff}
|
|
|
|
, {"lightgreen", 0x90ee90ff}
|
|
|
|
, {"lightpink", 0xffb6c1ff}
|
|
|
|
, {"lightsalmon", 0xffa07aff}
|
|
|
|
, {"lightseagreen", 0x20b2aaff}
|
|
|
|
, {"lightskyblue", 0x87cefaff}
|
|
|
|
, {"lightslateblue", 0x8470ffff}
|
|
|
|
, {"lightslategray", 0x778899ff}
|
|
|
|
, {"lightsteelblue", 0xb0c4deff}
|
|
|
|
, {"lightyellow", 0xffffe0ff}
|
|
|
|
, {"lime", 0x00ff00ff}
|
|
|
|
, {"limegreen", 0x32cd32ff}
|
|
|
|
, {"linen", 0xfaf0e6ff}
|
|
|
|
, {"magenta", 0xff00ffff}
|
|
|
|
, {"maroon", 0x800000ff}
|
|
|
|
, {"mediumaquamarine", 0x66cdaaff}
|
|
|
|
, {"mediumblue", 0x0000cdff}
|
|
|
|
, {"mediumorchid", 0xba55d3ff}
|
|
|
|
, {"mediumpurple", 0x9370d8ff}
|
|
|
|
, {"mediumseagreen", 0x3cb371ff}
|
|
|
|
, {"mediumslateblue", 0x7b68eeff}
|
|
|
|
, {"mediumspringgreen", 0x00fa9af}
|
|
|
|
, {"mediumturquoise", 0x48d1ccff}
|
|
|
|
, {"mediumvioletred", 0xc71585ff}
|
|
|
|
, {"midnightblue", 0x191970ff}
|
|
|
|
, {"mintcream", 0xf5fffaff}
|
|
|
|
, {"mistyrose", 0xffe4e1ff}
|
|
|
|
, {"moccasin", 0xffe4b5ff}
|
|
|
|
, {"navajowhite", 0xffdeadff}
|
|
|
|
, {"navy", 0x000080ff}
|
|
|
|
, {"oldlace", 0xfdf5e6ff}
|
|
|
|
, {"olive", 0x808000ff}
|
|
|
|
, {"olivedrab", 0x6b8e23ff}
|
|
|
|
, {"orange", 0xffa500ff}
|
|
|
|
, {"orangered", 0xff4500ff}
|
|
|
|
, {"orchid", 0xda70d6ff}
|
|
|
|
, {"palegoldenrod", 0xeee8aaff}
|
|
|
|
, {"palegreen", 0x98fb98ff}
|
|
|
|
, {"paleturquoise", 0xafeeeeff}
|
|
|
|
, {"palevioletred", 0xd87093ff}
|
|
|
|
, {"papayawhip", 0xffefd5ff}
|
|
|
|
, {"peachpuff", 0xffdab9ff}
|
|
|
|
, {"peru", 0xcd853fff}
|
|
|
|
, {"pink", 0xffc0cbff}
|
|
|
|
, {"plum", 0xdda0ddff}
|
|
|
|
, {"powderblue", 0xb0e0e6ff}
|
|
|
|
, {"purple", 0x800080ff}
|
|
|
|
, {"red", 0xff0000ff}
|
|
|
|
, {"rosybrown", 0xbc8f8fff}
|
|
|
|
, {"royalblue", 0x4169e1ff}
|
|
|
|
, {"saddlebrown", 0x8b4513ff}
|
|
|
|
, {"salmon", 0xfa8072ff}
|
|
|
|
, {"sandybrown", 0xf4a460ff}
|
|
|
|
, {"seagreen", 0x2e8b57ff}
|
|
|
|
, {"seashell", 0xfff5eeff}
|
|
|
|
, {"sienna", 0xa0522dff}
|
|
|
|
, {"silver", 0xc0c0c0ff}
|
|
|
|
, {"skyblue", 0x87ceebff}
|
|
|
|
, {"slateblue", 0x6a5acdff}
|
|
|
|
, {"slategray", 0x708090ff}
|
|
|
|
, {"snow", 0xfffafaff}
|
|
|
|
, {"springgreen", 0x00ff7fff}
|
|
|
|
, {"steelblue", 0x4682b4ff}
|
|
|
|
, {"tan", 0xd2b48cff}
|
|
|
|
, {"teal", 0x008080ff}
|
|
|
|
, {"thistle", 0xd8bfd8ff}
|
|
|
|
, {"tomato", 0xff6347ff}
|
|
|
|
, {"turquoise", 0x40e0d0ff}
|
|
|
|
, {"violet", 0xee82eeff}
|
|
|
|
, {"violetred", 0xd02090ff}
|
|
|
|
, {"wheat", 0xf5deb3ff}
|
|
|
|
, {"white", 0xffffffff}
|
|
|
|
, {"whitesmoke", 0xf5f5f5ff}
|
|
|
|
, {"yellow", 0xffff00ff}
|
|
|
|
, {"yellowgreen", 0x9acd32ff}
|
|
|
|
, {NULL, NULL}
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Hex digit int val.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int
|
|
|
|
h(char c) {
|
|
|
|
switch (c) {
|
|
|
|
case '0':
|
|
|
|
case '1':
|
|
|
|
case '2':
|
|
|
|
case '3':
|
|
|
|
case '4':
|
|
|
|
case '5':
|
|
|
|
case '6':
|
|
|
|
case '7':
|
|
|
|
case '8':
|
|
|
|
case '9':
|
|
|
|
return c - '0';
|
|
|
|
case 'a':
|
|
|
|
case 'b':
|
|
|
|
case 'c':
|
|
|
|
case 'd':
|
|
|
|
case 'e':
|
|
|
|
case 'f':
|
|
|
|
return (c - 'a') + 10;
|
|
|
|
case 'A':
|
|
|
|
case 'B':
|
|
|
|
case 'C':
|
|
|
|
case 'D':
|
|
|
|
case 'E':
|
|
|
|
case 'F':
|
|
|
|
return (c - 'A') + 10;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return rgba_t from rgba.
|
|
|
|
*/
|
|
|
|
|
|
|
|
rgba_t
|
|
|
|
rgba_create(uint32_t rgba) {
|
|
|
|
rgba_t color;
|
|
|
|
color.r = (double) (rgba >> 24) / 255;
|
|
|
|
color.g = (double) ((rgba & 0x00ff0000) >> 16) / 255;
|
|
|
|
color.b = (double) ((rgba & 0x0000ff00) >> 8) / 255;
|
|
|
|
color.a = (double) (rgba & 0xff) / 255;
|
|
|
|
return color;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return a string representation of the color.
|
|
|
|
*/
|
|
|
|
|
|
|
|
char *
|
|
|
|
rgba_to_string(rgba_t rgba) {
|
|
|
|
char *buf = (char *) malloc(64);
|
|
|
|
if (1 == rgba.a) {
|
|
|
|
sprintf(buf, "#%.2X%.2X%.2X"
|
|
|
|
, (int) (rgba.r * 255)
|
|
|
|
, (int) (rgba.g * 255)
|
|
|
|
, (int) (rgba.b * 255));
|
|
|
|
} else {
|
|
|
|
sprintf(buf, "rgba(%d, %d, %d, %.2f)"
|
|
|
|
, (int) (rgba.r * 255)
|
|
|
|
, (int) (rgba.g * 255)
|
|
|
|
, (int) (rgba.b * 255)
|
|
|
|
, rgba.a);
|
|
|
|
}
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return rgba from (r,g,b,a).
|
|
|
|
*/
|
|
|
|
|
|
|
|
static inline int32_t
|
|
|
|
rgba_from_rgba(uint8_t r, uint8_t g, uint8_t b, uint8_t a) {
|
|
|
|
return
|
|
|
|
r << 24
|
|
|
|
| g << 16
|
|
|
|
| b << 8
|
|
|
|
| a;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return rgba from (r,g,b).
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int32_t
|
|
|
|
rgba_from_rgb(uint8_t r, uint8_t g, uint8_t b) {
|
|
|
|
return rgba_from_rgba(r, g, b, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return rgb from "#RRGGBB".
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int32_t
|
|
|
|
rgba_from_hex6_string(const char *str) {
|
|
|
|
return rgba_from_rgb(
|
|
|
|
(h(str[0]) << 4) + h(str[1])
|
|
|
|
, (h(str[2]) << 4) + h(str[3])
|
|
|
|
, (h(str[4]) << 4) + h(str[5])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return rgb from "#RGB"
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int32_t
|
|
|
|
rgba_from_hex3_string(const char *str) {
|
|
|
|
return rgba_from_rgb(
|
|
|
|
(h(str[0]) << 4) + h(str[0])
|
|
|
|
, (h(str[1]) << 4) + h(str[1])
|
|
|
|
, (h(str[2]) << 4) + h(str[2])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return rgb from "rgb()"
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int32_t
|
|
|
|
rgba_from_rgb_string(const char *str, short *ok) {
|
|
|
|
if (str == strnstr(str, "rgb(", 4)) {
|
|
|
|
str += 4;
|
|
|
|
WHITESPACE;
|
|
|
|
uint8_t r = 0, g = 0, b = 0;
|
|
|
|
CHANNEL(r);
|
|
|
|
CHANNEL(g);
|
|
|
|
CHANNEL(b);
|
|
|
|
return *ok = 1, rgba_from_rgb(r, g, b);
|
|
|
|
}
|
|
|
|
return *ok = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return rgb from "rgba()"
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int32_t
|
|
|
|
rgba_from_rgba_string(const char *str, short *ok) {
|
|
|
|
if (str == strnstr(str, "rgba(", 5)) {
|
|
|
|
str += 5;
|
|
|
|
WHITESPACE;
|
|
|
|
uint8_t r = 0, g = 0, b = 0;
|
|
|
|
float a = 0;
|
|
|
|
CHANNEL(r);
|
|
|
|
CHANNEL(g);
|
|
|
|
CHANNEL(b);
|
|
|
|
// TODO: less strict
|
|
|
|
if ('1' == *str) {
|
|
|
|
a = 1;
|
|
|
|
} else {
|
|
|
|
if ('0' == *str) a = 0, ++str;
|
|
|
|
if ('.' == *str) {
|
|
|
|
++str;
|
|
|
|
float n = .1;
|
|
|
|
while (*str >= '0' && *str <= '9') {
|
|
|
|
a += (*str++ - '0') * n;
|
|
|
|
n *= .1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return *ok = 1, rgba_from_rgba(r, g, b, a * 255);
|
|
|
|
}
|
|
|
|
return *ok = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return rgb from:
|
|
|
|
*
|
|
|
|
* - "#RGB"
|
|
|
|
* - "#RRGGBB"
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int32_t
|
|
|
|
rgba_from_hex_string(const char *str, short *ok) {
|
|
|
|
size_t len = strlen(str);
|
|
|
|
*ok = 1;
|
|
|
|
if (6 == len) return rgba_from_hex6_string(str);
|
|
|
|
if (3 == len) return rgba_from_hex3_string(str);
|
|
|
|
return *ok = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return named color value.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int32_t
|
|
|
|
rgba_from_name_string(const char *str, short *ok) {
|
|
|
|
int i = 0;
|
|
|
|
struct named_color color;
|
|
|
|
while ((color = named_colors[i++]).name) {
|
|
|
|
if (*str == *color.name && 0 == strcmp(str, color.name))
|
|
|
|
return *ok = 1, color.val;
|
|
|
|
}
|
|
|
|
return *ok = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return rgb from:
|
|
|
|
*
|
|
|
|
* - #RGB
|
|
|
|
* - #RRGGBB
|
|
|
|
* - rgb(r,g,b)
|
|
|
|
* - rgba(r,g,b,a)
|
|
|
|
* - name
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
int32_t
|
|
|
|
rgba_from_string(const char *str, short *ok) {
|
|
|
|
if ('#' == str[0])
|
|
|
|
return rgba_from_hex_string(++str, ok);
|
|
|
|
if (str == strnstr(str, "rgba", 4))
|
|
|
|
return rgba_from_rgba_string(str, ok);
|
|
|
|
if (str == strnstr(str, "rgb", 3))
|
|
|
|
return rgba_from_rgb_string(str, ok);
|
|
|
|
return rgba_from_name_string(str, ok);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Inspect the given rgba color.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
rgba_inspect(int32_t rgba) {
|
|
|
|
printf("rgba(%d,%d,%d,%d)\n"
|
|
|
|
, rgba >> 24 & 0xFF
|
|
|
|
, rgba >> 16 & 0xFF
|
|
|
|
, rgba >> 8 & 0xFF
|
|
|
|
, rgba & 0xFF
|
|
|
|
);
|
|
|
|
}
|