Skip to content
Snippets Groups Projects
Commit cbd04604 authored by Mirko Wächter's avatar Mirko Wächter
Browse files

refactored HSV color utils

parent 835831b0
No related branches found
No related tags found
No related merge requests found
...@@ -52,11 +52,15 @@ module armarx ...@@ -52,11 +52,15 @@ module armarx
byte b; byte b;
}; };
struct HsvColor struct HsvColor
{ {
byte h; //! 0-360
byte s; float h;
byte v; //! 0 - 1
float s;
//! 0 - 1
float v;
}; };
......
...@@ -30,103 +30,126 @@ namespace armarx ...@@ -30,103 +30,126 @@ namespace armarx
namespace colorutils namespace colorutils
{ {
DrawColor24Bit HsvToRgb(const HsvColor& in)
DrawColor24Bit HsvToRgb(const HsvColor& hsv)
{ {
DrawColor24Bit rgb; double hh, p, q, t, ff;
unsigned char region, remainder, p, q, t; long i;
double r, g, b;
if (hsv.s == 0) if (in.s <= 0.0) // < is bogus, just shuts up warnings
{ {
rgb.r = hsv.v; r = in.v;
rgb.g = hsv.v; g = in.v;
rgb.b = hsv.v; b = in.v;
return rgb; return DrawColor24Bit {(Ice::Byte)(r * 255), (Ice::Byte)(g * 255), (Ice::Byte)(b * 255)};
} }
constexpr float by43 = 1.0f / 43.0f; hh = in.h;
region = hsv.h * by43; if (hh >= 360.0)
remainder = (hsv.h - (region * 43)) * 6; {
hh = 0.0;
p = (hsv.v * (255 - hsv.s)) >> 8; }
q = (hsv.v * (255 - ((hsv.s * remainder) >> 8))) >> 8; hh /= 60.0;
t = (hsv.v * (255 - ((hsv.s * (255 - remainder)) >> 8))) >> 8; i = (long)hh;
ff = hh - i;
switch (region) p = in.v * (1.0 - in.s);
q = in.v * (1.0 - (in.s * ff));
t = in.v * (1.0 - (in.s * (1.0 - ff)));
switch (i)
{ {
case 0: case 0:
rgb.r = hsv.v; r = in.v;
rgb.g = t; g = t;
rgb.b = p; b = p;
break; break;
case 1: case 1:
rgb.r = q; r = q;
rgb.g = hsv.v; g = in.v;
rgb.b = p; b = p;
break; break;
case 2: case 2:
rgb.r = p; r = p;
rgb.g = hsv.v; g = in.v;
rgb.b = t; b = t;
break; break;
case 3: case 3:
rgb.r = p; r = p;
rgb.g = q; g = q;
rgb.b = hsv.v; b = in.v;
break; break;
case 4: case 4:
rgb.r = t; r = t;
rgb.g = p; g = p;
rgb.b = hsv.v; b = in.v;
break; break;
case 5:
default: default:
rgb.r = hsv.v; r = in.v;
rgb.g = p; g = p;
rgb.b = q; b = q;
break; break;
} }
return rgb; return DrawColor24Bit {(Ice::Byte)(r * 255), (Ice::Byte)(g * 255), (Ice::Byte)(b * 255)};
} }
HsvColor RgbToHsv(const DrawColor24Bit& rgb) HsvColor RgbToHsv(const DrawColor24Bit& in)
{ {
HsvColor hsv; double r = in.r * 0.00392156862;
unsigned char rgbMin, rgbMax; double g = in.g * 0.00392156862;
double b = in.b * 0.00392156862;
HsvColor out;
double min, max, delta;
rgbMin = rgb.r < rgb.g ? (rgb.r < rgb.b ? rgb.r : rgb.b) : (rgb.g < rgb.b ? rgb.g : rgb.b); min = r < g ? r : g;
rgbMax = rgb.r > rgb.g ? (rgb.r > rgb.b ? rgb.r : rgb.b) : (rgb.g > rgb.b ? rgb.g : rgb.b); min = min < b ? min : b;
hsv.v = rgbMax; max = r > g ? r : g;
if (hsv.v == 0) max = max > b ? max : b;
out.v = max; // v
delta = max - min;
if (delta < 0.00001)
{ {
hsv.h = 0; out.s = 0;
hsv.s = 0; out.h = 0; // undefined, maybe nan?
return hsv; return out;
} }
if (max > 0.0) // NOTE: if Max is == 0, this divide would cause a crash
hsv.s = 255 * long(rgbMax - rgbMin) / hsv.v;
if (hsv.s == 0)
{ {
hsv.h = 0; out.s = (delta / max); // s
return hsv;
} }
else
if (rgbMax == rgb.r) {
// if max is 0, then r = g = b = 0
// s = 0, h is undefined
out.s = 0.0;
out.h = 0; // its now undefined
return out;
}
if (r >= max) // > is bogus, just keeps compilor happy
{ {
hsv.h = 0 + 43 * (rgb.g - rgb.b) / (rgbMax - rgbMin); out.h = (g - b) / delta; // between yellow & magenta
} }
else if (rgbMax == rgb.g) else if (g >= max)
{ {
hsv.h = 85 + 43 * (rgb.b - rgb.r) / (rgbMax - rgbMin); out.h = 2.0 + (b - r) / delta; // between cyan & yellow
} }
else else
{ {
hsv.h = 171 + 43 * (rgb.r - rgb.g) / (rgbMax - rgbMin); out.h = 4.0 + (r - g) / delta; // between magenta & cyan
} }
return hsv; out.h *= 60.0; // degrees
if (out.h < 0.0)
{
out.h += 360.0;
}
return out;
} }
/** /**
...@@ -137,8 +160,8 @@ namespace armarx ...@@ -137,8 +160,8 @@ namespace armarx
HsvColor HeatMapColor(float percentage) HsvColor HeatMapColor(float percentage)
{ {
percentage = math::MathUtils::LimitMinMax(0.0f, 1.0f, percentage); percentage = math::MathUtils::LimitMinMax(0.0f, 1.0f, percentage);
constexpr float factor = 240.0 * (255.0 / 360.0);
return HsvColor {(unsigned char)((1.0 - percentage) * factor), 255, 255}; return HsvColor {((1.0f - percentage) * 240.f), 1.0f, 1.0f};
} }
DrawColor24Bit HeatMapRGBColor(float percentage) DrawColor24Bit HeatMapRGBColor(float percentage)
......
...@@ -25,6 +25,7 @@ ...@@ -25,6 +25,7 @@
#include <RobotAPI/Test.h> #include <RobotAPI/Test.h>
#include <ArmarXCore/core/test/IceTestHelper.h> #include <ArmarXCore/core/test/IceTestHelper.h>
#include "../math/MathUtils.h" #include "../math/MathUtils.h"
#include "../math/ColorUtils.h"
using namespace armarx; using namespace armarx;
void check_close(float a, float b, float tolerance) void check_close(float a, float b, float tolerance)
...@@ -60,3 +61,16 @@ BOOST_AUTO_TEST_CASE(fmodTest) ...@@ -60,3 +61,16 @@ BOOST_AUTO_TEST_CASE(fmodTest)
} }
} }
BOOST_AUTO_TEST_CASE(ColorUtilTest)
{
auto testColor = [](DrawColor24Bit rgb)
{
auto hsv = colorutils::RgbToHsv(rgb);
ARMARX_INFO << VAROUT(rgb.r) << VAROUT(rgb.g) << VAROUT(rgb.b) << " --> " << VAROUT(hsv.h) << VAROUT(hsv.s) << VAROUT(hsv.v);
};
testColor({255, 0, 0});
testColor({0, 255, 0});
testColor({0, 0, 255});
testColor({0, 20, 0});
}
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment