2014-06-11 18:13:55 +02:00
|
|
|
/*
|
2016-02-13 23:00:05 +01:00
|
|
|
* Copyright (C) 2014 - 2016 Freie Universität Berlin
|
2014-06-11 18:13:55 +02:00
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
|
|
|
* Public License v2.1. See the file LICENSE in the top level directory for more
|
|
|
|
* details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup sys_color
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Implementation of color module
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
2016-02-13 23:00:05 +01:00
|
|
|
* @author Cenk Gündoğan <mail@cgundogan.de>
|
2016-03-15 18:01:12 +01:00
|
|
|
* @author Simon Brummer <brummer.simon@googlemail.com>
|
2014-06-11 18:13:55 +02:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "color.h"
|
|
|
|
|
2020-10-22 11:35:22 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2016-02-13 23:00:05 +01:00
|
|
|
#include "debug.h"
|
2014-06-11 18:13:55 +02:00
|
|
|
|
|
|
|
void color_rgb2hsv(color_rgb_t *rgb, color_hsv_t *hsv)
|
|
|
|
{
|
2023-05-31 20:49:57 +02:00
|
|
|
float rd, gd, bd, delta, min, max;
|
|
|
|
int imax, imin, sector;
|
2014-06-11 18:13:55 +02:00
|
|
|
|
2023-05-31 20:49:57 +02:00
|
|
|
/* catch special case grey first */
|
|
|
|
if (rgb->r == rgb->g && rgb->r == rgb->b) {
|
|
|
|
hsv->v = (float)rgb->r * (1 / 255.0f);
|
|
|
|
hsv->s = 0.0f;
|
|
|
|
hsv->h = 0.0f; /* hue might be anything for grey, but it is not uncommon to use 0 */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* normalize RGB colors to the range [0 - 1.0] */
|
|
|
|
/* multiplication is often faster than division -> using compile time constant */
|
|
|
|
rd = (float)rgb->r * (1 / 255.0f);
|
|
|
|
gd = (float)rgb->g * (1 / 255.0f);
|
|
|
|
bd = (float)rgb->b * (1 / 255.0f);
|
2014-06-11 18:13:55 +02:00
|
|
|
|
2023-05-31 20:49:57 +02:00
|
|
|
/* find maximum of the three colors and sector */
|
|
|
|
/* using the comparing faster integer color value */
|
|
|
|
imax = rgb->r;
|
|
|
|
max = rd;
|
|
|
|
sector = 0;
|
|
|
|
if (rgb->g > imax) {
|
|
|
|
imax = rgb->g;
|
|
|
|
max = gd;
|
|
|
|
sector = 1;
|
2014-06-11 18:13:55 +02:00
|
|
|
}
|
2023-05-31 20:49:57 +02:00
|
|
|
if (rgb->b > imax) {
|
|
|
|
imax = rgb->b;
|
|
|
|
max = bd;
|
|
|
|
sector = 2;
|
2014-06-11 18:13:55 +02:00
|
|
|
}
|
2023-05-31 20:49:57 +02:00
|
|
|
/* value is maximum*/
|
|
|
|
hsv->v = max;
|
2014-06-11 18:13:55 +02:00
|
|
|
|
2023-05-31 20:49:57 +02:00
|
|
|
/* find of minimum the three RGB colors */
|
|
|
|
imin = rgb->r;
|
|
|
|
min = rd;
|
|
|
|
if (rgb->g < imin) {
|
|
|
|
imin = rgb->g;
|
|
|
|
min = gd;
|
2014-06-11 18:13:55 +02:00
|
|
|
}
|
2023-05-31 20:49:57 +02:00
|
|
|
if (rgb->b < imin) {
|
|
|
|
imin = rgb->b;
|
|
|
|
min = bd;
|
2014-06-11 18:13:55 +02:00
|
|
|
}
|
2023-05-31 20:49:57 +02:00
|
|
|
/* compute delta from value and minimum*/
|
|
|
|
delta = hsv->v - min;
|
2014-06-11 18:13:55 +02:00
|
|
|
|
|
|
|
/* find the saturation from value and delta */
|
2023-05-31 20:49:57 +02:00
|
|
|
/* special case gray r == g == b ^= min == max */
|
|
|
|
hsv->s = delta / max;
|
2014-06-11 18:13:55 +02:00
|
|
|
|
|
|
|
/* compute hue */
|
2023-05-31 20:49:57 +02:00
|
|
|
float p = 60.0f / delta;
|
|
|
|
switch (sector){
|
|
|
|
case 0:
|
|
|
|
hsv->h = (gd - bd) * p;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
hsv->h = 120.0f + (bd - rd) * p;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
hsv->h = 240.0f + (rd - gd) * p;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (hsv->h < 0.0f) {
|
|
|
|
hsv->h += 360.0f;
|
2014-06-11 18:13:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void color_hsv2rgb(color_hsv_t *hsv, color_rgb_t *rgb)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
float aa, bb, cc, f, h;
|
|
|
|
|
|
|
|
if (hsv->s == 0.0f) {
|
|
|
|
rgb->r = (uint8_t)(hsv->v * 255.0f);
|
|
|
|
rgb->g = (uint8_t)(hsv->v * 255.0f);
|
|
|
|
rgb->b = (uint8_t)(hsv->v * 255.0f);
|
2014-08-11 19:51:23 +02:00
|
|
|
return;
|
2014-06-11 18:13:55 +02:00
|
|
|
}
|
|
|
|
|
2018-09-28 09:32:26 +02:00
|
|
|
h = (hsv->h == 360.0f) ? 0.0f : hsv->h;
|
2014-08-11 19:51:23 +02:00
|
|
|
h /= 60.0f;
|
|
|
|
i = (int)h;
|
|
|
|
f = h - i;
|
|
|
|
aa = hsv->v * (1.0f - hsv->s);
|
|
|
|
bb = hsv->v * (1.0f - (hsv->s * f));
|
|
|
|
cc = hsv->v * (1.0f - (hsv->s * (1.0f - f)));
|
|
|
|
|
|
|
|
switch (i) {
|
|
|
|
case 0:
|
|
|
|
rgb->r = (uint8_t)(hsv->v * 255.0f);
|
|
|
|
rgb->g = (uint8_t)(cc * 255.0f);
|
|
|
|
rgb->b = (uint8_t)(aa * 255.0f);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
rgb->r = (uint8_t)(bb * 255.0f);
|
|
|
|
rgb->g = (uint8_t)(hsv->v * 255.0f);
|
|
|
|
rgb->b = (uint8_t)(aa * 255.0f);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
rgb->r = (uint8_t)(aa * 255.0f);
|
|
|
|
rgb->g = (uint8_t)(hsv->v * 255.0f);
|
|
|
|
rgb->b = (uint8_t)(cc * 255.0f);
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
rgb->r = (uint8_t)(aa * 255.0f);
|
|
|
|
rgb->g = (uint8_t)(bb * 255.0f);
|
|
|
|
rgb->b = (uint8_t)(hsv->v * 255.0f);
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
rgb->r = (uint8_t)(cc * 255.0f);
|
|
|
|
rgb->g = (uint8_t)(aa * 255.0f);
|
|
|
|
rgb->b = (uint8_t)(hsv->v * 255.0f);
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
rgb->r = (uint8_t)(hsv->v * 255.0f);
|
|
|
|
rgb->g = (uint8_t)(aa * 255.0f);
|
|
|
|
rgb->b = (uint8_t)(bb * 255.0f);
|
|
|
|
break;
|
2014-06-11 18:13:55 +02:00
|
|
|
}
|
|
|
|
}
|
2016-02-13 23:00:05 +01:00
|
|
|
|
|
|
|
void color_hex2rgb(const uint32_t hex, color_rgb_t *rgb)
|
|
|
|
{
|
|
|
|
rgb->r = ((hex >> 16UL) & 0xFF);
|
|
|
|
rgb->g = ((hex >> 8UL) & 0xFF);
|
|
|
|
rgb->b = (hex & 0xFF);
|
|
|
|
}
|
|
|
|
|
|
|
|
void color_rgb2hex(const color_rgb_t *rgb, uint32_t *hex)
|
|
|
|
{
|
|
|
|
*hex = (((uint32_t) rgb->r) << 16UL) | (rgb->g << 8UL) | (rgb->b);
|
|
|
|
}
|
|
|
|
|
|
|
|
void color_str2rgb(const char* str, color_rgb_t *rgb)
|
|
|
|
{
|
|
|
|
rgb->r = (((str[0] > '9') ? (str[0] &~ 0x20) - 'A' + 10 : (str[0] - '0')) << 4) | /* R */
|
|
|
|
(((str[1] > '9') ? (str[1] &~ 0x20) - 'A' + 10 : (str[1] - '0')) << 0) ; /* R */
|
|
|
|
rgb->g = (((str[2] > '9') ? (str[2] &~ 0x20) - 'A' + 10 : (str[2] - '0')) << 4) | /* G */
|
|
|
|
(((str[3] > '9') ? (str[3] &~ 0x20) - 'A' + 10 : (str[3] - '0')) << 0) ; /* G */
|
|
|
|
rgb->b = (((str[4] > '9') ? (str[4] &~ 0x20) - 'A' + 10 : (str[4] - '0')) << 4) | /* B */
|
|
|
|
(((str[5] > '9') ? (str[5] &~ 0x20) - 'A' + 10 : (str[5] - '0')) << 0) ; /* B */
|
|
|
|
}
|
|
|
|
|
|
|
|
void color_rgb2str(const color_rgb_t *rgb, char* str)
|
|
|
|
{
|
|
|
|
uint8_t tmp;
|
|
|
|
|
|
|
|
/* RR */
|
|
|
|
tmp = rgb->r >> 4;
|
|
|
|
str[0] = (tmp > 9) ? ('A' - 10 + tmp) : ('0' + tmp);
|
|
|
|
tmp = rgb->r & 0x0F;
|
|
|
|
str[1] = (tmp > 9) ? ('A' - 10 + tmp) : ('0' + tmp);
|
|
|
|
|
|
|
|
/* GG */
|
|
|
|
tmp = rgb->g >> 4;
|
|
|
|
str[2] = (tmp > 9) ? ('A' - 10 + tmp) : ('0' + tmp);
|
|
|
|
tmp = rgb->g & 0x0F;
|
|
|
|
str[3] = (tmp > 9) ? ('A' - 10 + tmp) : ('0' + tmp);
|
|
|
|
|
|
|
|
/* BB */
|
|
|
|
tmp = rgb->b >> 4;
|
|
|
|
str[4] = (tmp > 9) ? ('A' - 10 + tmp) : ('0' + tmp);
|
|
|
|
tmp = rgb->b & 0x0F;
|
|
|
|
str[5] = (tmp > 9) ? ('A' - 10 + tmp) : ('0' + tmp);
|
|
|
|
}
|
2016-03-15 18:01:12 +01:00
|
|
|
|
|
|
|
void color_rgb_complementary(const color_rgb_t *rgb, color_rgb_t *comp_rgb)
|
|
|
|
{
|
|
|
|
uint8_t max = 0;
|
|
|
|
uint8_t min = 0;
|
|
|
|
uint16_t val = 0;
|
|
|
|
|
|
|
|
max = (rgb->r > rgb->g) ? rgb->r : rgb->g;
|
|
|
|
max = (rgb->b > max) ? rgb->b : max;
|
|
|
|
min = (rgb->r < rgb->g) ? rgb->r : rgb->g;
|
|
|
|
min = (rgb->b < min) ? rgb->b : min;
|
|
|
|
val = max + min;
|
|
|
|
|
|
|
|
comp_rgb->r = val - rgb->r;
|
|
|
|
comp_rgb->g = val - rgb->g;
|
|
|
|
comp_rgb->b = val - rgb->b;
|
|
|
|
}
|