Fork of the original NeoPixel repository.
colorspace.cpp
- Committer:
- dstarke
- Date:
- 2020-04-18
- Revision:
- 1:c00a57df399b
- Parent:
- 0:a81364d9a67b
File content as of revision 1:c00a57df399b:
#include "mbed.h" #include "colorspace.h" #include "math.h" #define max(a,b) ((a)>(b)?(a):(b)) #define min(a,b) ((a)>(b)?(b):(a)) static float threeway_max(float a, float b, float c) { return max(a, max(b, c)); } static float threeway_min(float a, float b, float c) { return min(a, min(b, c)); } static float hue2rgb(float p, float q, float t) { if(t < 0) t += 1; if(t > 1) t -= 1; if(t < 1/6.0f) return p + (q - p) * 6 * t; if(t < 1/2.0f) return q; if(t < 2/3.0f) return p + (q - p) * (2/3.0f - t) * 6; return p; } void hsl2rgb(const FloatHSL *hsl, FloatRGB *rgb) { float h = hsl->h; float s = hsl->s; float l = hsl->l; if (s == 0) { rgb->r = rgb->g = rgb->b = l; // achromatic } else { float q = l < 0.5f ? l * (1 + s) : l + s - l * s; float p = 2 * l - q; rgb->r = hue2rgb(p, q, h + 1/3.0f); rgb->g = hue2rgb(p, q, h); rgb->b = hue2rgb(p, q, h - 1/3.0f); } } void hsv2rgb(const FloatHSV *hsv, FloatRGB *rgb) { float r, g, b; int i = floor(hsv->h * 6); float f = hsv->h * 6 - i; float p = hsv->v * (1 - hsv->s); float q = hsv->v * (1 - f * hsv->s); float t = hsv->v * (1 - (1 - f) * hsv->s); switch(i % 6) { case 0: r = hsv->v; g = t; b = p; break; case 1: r = q; g = hsv->v; b = p; break; case 2: r = p; g = hsv->v; b = t; break; case 3: r = p; g = q; b = hsv->v; break; case 4: r = t; g = p; b = hsv->v; break; case 5: r = hsv->v; g = p; b = q; break; } rgb->r = r; rgb->g = g; rgb->b = b; } void rgb2hsl(const FloatRGB *rgb, FloatHSL *hsl) { float rd = rgb->r; float gd = rgb->g; float bd = rgb->b; float max = threeway_max(rd, gd, bd); float min = threeway_min(rd, gd, bd); float h, s, l = (max + min) / 2.0f; if (max == min) { h = s = 0; // achromatic } else { float d = max - min; s = l > 0.5f ? d / (2 - max - min) : d / (max + min); if (max == rd) { h = (gd - bd) / d + (gd < bd ? 6 : 0); } else if (max == gd) { h = (bd - rd) / d + 2; } else if (max == bd) { h = (rd - gd) / d + 4; } h /= 6; } hsl->h = h; hsl->s = s; hsl->l = l; } /** Convert from HSV to HSL (this is possibly wrong) */ void hsv2hsl(const FloatHSV *hsv, FloatHSL *hsl) { float l = (2 - hsv->s) * hsv->v / 2.0f; float s = hsv->s; if (l != 0) { if (l == 1) { s = 0; } else if (l < 0.5f) { s = s * hsv->v / (l * 2); } else { s = s * hsv->v / (2 - l * 2); } } hsl->h = hsv->h; hsl->s = s; hsl->l = l; } /** Convert from HSL to HSV */ void hsl2hsv(const FloatHSL *hsl, FloatHSV *hsv) { float sat = hsv->s * ((hsl->l < 0.5f) ? hsl->l : (1 - hsl->l)); hsv->s = 2*sat / (hsl->l + sat); hsv->h = hsl->h; hsv->v = hsl->l + sat; } uint32_t hsl2hex(const FloatHSL *hsl) { FloatRGB rgb; hsl2rgb(hsl, &rgb); return rgb2hex(&rgb); } uint32_t hsv2hex(const FloatHSV *hsv) { FloatRGB rgb; hsv2rgb(hsv, &rgb); return rgb2hex(&rgb); } uint32_t rgb2hex(FloatRGB *rgb) { return ((int)floor(rgb->r*255 + 0.5f) << 16) | ((int)floor(rgb->g*255 + 0.5f) << 8) | ((int)floor(rgb->b*255 + 0.5f)); }