2021-01-29 06:31:58 -05:00
|
|
|
//
|
|
|
|
// FILE: Kelvin2RGB.cpp
|
|
|
|
// AUTHOR: Rob Tillaart
|
2022-11-14 14:29:43 -05:00
|
|
|
// VERSION: 0.1.5
|
2021-01-29 06:31:58 -05:00
|
|
|
// DATE: 2018-01-31
|
|
|
|
// PURPOSE: Arduino library for converting temperature to RGB values
|
|
|
|
// URL: https://github.com/RobTillaart/Kelvin2RGB
|
2022-11-14 14:29:43 -05:00
|
|
|
//
|
|
|
|
// HISTORY: see changelog.md
|
2021-12-20 11:09:35 -05:00
|
|
|
|
2021-06-02 07:46:05 -04:00
|
|
|
|
2021-01-29 06:31:58 -05:00
|
|
|
#include "Kelvin2RGB.h"
|
|
|
|
|
2021-06-02 07:46:05 -04:00
|
|
|
|
|
|
|
#define DIVIDE_255 (0.0039215686274509803921568627451)
|
|
|
|
|
2021-01-29 06:31:58 -05:00
|
|
|
|
|
|
|
Kelvin2RGB::Kelvin2RGB()
|
|
|
|
{
|
2022-11-14 14:29:43 -05:00
|
|
|
reset();
|
2021-01-29 06:31:58 -05:00
|
|
|
}
|
|
|
|
|
2022-11-14 14:29:43 -05:00
|
|
|
// empty function for now, remove?
|
2021-01-29 06:31:58 -05:00
|
|
|
void Kelvin2RGB::begin()
|
2022-11-14 14:29:43 -05:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Kelvin2RGB::reset()
|
2021-01-29 06:31:58 -05:00
|
|
|
{
|
2021-11-06 10:56:44 -04:00
|
|
|
_temperature = 0;
|
2022-11-14 14:29:43 -05:00
|
|
|
_brightness = 0; // default = darkness
|
2021-11-06 10:56:44 -04:00
|
|
|
_red = 0;
|
|
|
|
_green = 0;
|
|
|
|
_blue = 0;
|
|
|
|
_rgb = 0;
|
2021-01-29 06:31:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-11-14 14:29:43 -05:00
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// Tanner Helland formulas
|
|
|
|
//
|
2021-01-29 06:31:58 -05:00
|
|
|
void Kelvin2RGB::convert_TH(float temperature, float brightness)
|
|
|
|
{
|
|
|
|
_temperature = constrain(temperature, 0, 65500);
|
|
|
|
_brightness = constrain(brightness, 0, 100);
|
|
|
|
|
2021-11-06 10:56:44 -04:00
|
|
|
_red = _green = _blue = 0;
|
2021-01-29 06:31:58 -05:00
|
|
|
float t = _temperature * 0.01;
|
|
|
|
|
|
|
|
if (t <= 66)
|
|
|
|
{
|
|
|
|
_red = 255;
|
|
|
|
_green = (99.4708025861 * log(t)) - 161.1195681661;
|
|
|
|
if (t > 19)
|
|
|
|
{
|
|
|
|
_blue = (138.5177312231 * log(t - 10)) - 305.0447927307;
|
|
|
|
}
|
|
|
|
else _blue = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_red = 329.698727466 * pow(t - 60, -0.1332047592);
|
|
|
|
_green = 288.1221695283 * pow(t - 60, -0.0755148492);
|
|
|
|
_blue = 255;
|
|
|
|
}
|
2021-06-02 07:46:05 -04:00
|
|
|
|
|
|
|
_normalize();
|
2021-01-29 06:31:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-11-14 14:29:43 -05:00
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// Neil Bartlett formulas
|
|
|
|
//
|
2021-01-29 06:31:58 -05:00
|
|
|
void Kelvin2RGB::convert_NB(float temperature, float brightness)
|
|
|
|
{
|
|
|
|
_temperature = constrain(temperature, 0, 65500);
|
|
|
|
_brightness = constrain(brightness, 0, 100);
|
|
|
|
|
|
|
|
_red = _green = _blue = 0;
|
|
|
|
float t = _temperature * 0.01;
|
|
|
|
|
|
|
|
if (t <= 66)
|
|
|
|
{
|
|
|
|
_red = 255;
|
|
|
|
_green = t - 2;
|
|
|
|
_green = -155.25485562709179 - 0.44596950469579133 * _green + 104.49216199393888 * log(_green);
|
|
|
|
_blue = 0;
|
|
|
|
if (t > 20)
|
|
|
|
{
|
|
|
|
_blue = t - 10;
|
|
|
|
_blue = -254.76935184120902 + 0.8274096064007395 * _blue + 115.67994401066147 * log(_blue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_red = t - 55.0;
|
|
|
|
_red = 351.97690566805693 + 0.114206453784165 * _red - 40.25366309332127 * log(_red);
|
|
|
|
_green = t - 50.0;
|
|
|
|
_green = 325.4494125711974 + 0.07943456536662342 * _green - 28.0852963507957 * log(_green);
|
|
|
|
_blue = 255;
|
|
|
|
}
|
|
|
|
|
2021-06-02 07:46:05 -04:00
|
|
|
_normalize();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-11-14 14:29:43 -05:00
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// Other functions
|
|
|
|
//
|
|
|
|
|
|
|
|
float Kelvin2RGB::temperature()
|
|
|
|
{
|
|
|
|
return _temperature;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
float Kelvin2RGB::brightness()
|
|
|
|
{
|
|
|
|
return _brightness;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
float Kelvin2RGB::red()
|
|
|
|
{
|
|
|
|
return _red;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
float Kelvin2RGB::green()
|
|
|
|
{
|
|
|
|
return _green;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
float Kelvin2RGB::blue()
|
|
|
|
{
|
|
|
|
return _blue;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2021-11-06 10:56:44 -04:00
|
|
|
uint32_t Kelvin2RGB::setRGB(float red, float green, float blue, float brightness)
|
|
|
|
{
|
|
|
|
_brightness = brightness;
|
|
|
|
_red = red * 255;
|
|
|
|
_green = green * 255;
|
|
|
|
_blue = blue * 255;
|
|
|
|
_normalize();
|
|
|
|
return _rgb;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-11-14 14:29:43 -05:00
|
|
|
//
|
|
|
|
// 32 bit colour (only 3 bytes used)
|
|
|
|
//
|
|
|
|
uint32_t Kelvin2RGB::RGB()
|
|
|
|
{
|
|
|
|
return _rgb;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// 16 bit colour - of last conversion.
|
|
|
|
//
|
2021-06-02 07:46:05 -04:00
|
|
|
uint16_t Kelvin2RGB::RGB565()
|
|
|
|
{
|
|
|
|
uint16_t val = 0;
|
|
|
|
val = uint8_t(_red * 32);
|
|
|
|
val <<= 6;
|
|
|
|
val |= uint8_t(_green * 64);
|
|
|
|
val <<= 5;
|
|
|
|
val |= uint8_t(_blue * 32);
|
2021-11-06 10:56:44 -04:00
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t Kelvin2RGB::CMYK()
|
|
|
|
{
|
|
|
|
float k = _red;
|
|
|
|
if (k < _green) k = _green;
|
|
|
|
if (k < _blue) k = _blue;
|
|
|
|
k = 1 - k;
|
|
|
|
uint32_t c = 255 * (1 - _red - k) / (1 - k);
|
|
|
|
uint32_t m = 255 * (1 - _green - k) / (1 - k);
|
|
|
|
uint32_t y = 255 * (1 - _blue - k) / (1 - k);
|
|
|
|
|
|
|
|
return (c << 24) + (m << 16) + (y << 8) + (k * 255);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t Kelvin2RGB::BGR()
|
|
|
|
{
|
2022-11-14 14:29:43 -05:00
|
|
|
return round(255 * _blue) * 65536UL + round(255 * _green) * 256UL + round(255 * _red);
|
2021-06-02 07:46:05 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// private
|
|
|
|
//
|
2021-11-06 10:56:44 -04:00
|
|
|
|
|
|
|
// expects _red, _green, _blue in the 0..255 range.
|
2021-06-02 07:46:05 -04:00
|
|
|
void Kelvin2RGB::_normalize()
|
|
|
|
{
|
2021-01-29 06:31:58 -05:00
|
|
|
float f = 0.01 * _brightness;
|
2021-06-02 07:46:05 -04:00
|
|
|
|
2021-01-29 06:31:58 -05:00
|
|
|
_red = constrain(f * _red, 0, 255);
|
|
|
|
_green = constrain(f * _green, 0, 255);
|
|
|
|
_blue = constrain(f * _blue, 0, 255);
|
2021-06-02 07:46:05 -04:00
|
|
|
|
2021-01-29 06:31:58 -05:00
|
|
|
_rgb = round(_red) * 65536UL + round(_green) * 256UL + round(_blue);
|
2021-11-06 10:56:44 -04:00
|
|
|
|
2021-01-29 06:31:58 -05:00
|
|
|
// divide by 255 to get factors between 0..1
|
|
|
|
_red *= DIVIDE_255;
|
|
|
|
_green *= DIVIDE_255;
|
|
|
|
_blue *= DIVIDE_255;
|
|
|
|
}
|
|
|
|
|
2021-06-02 07:46:05 -04:00
|
|
|
|
2021-01-29 06:31:58 -05:00
|
|
|
// -- END OF FILE --
|
2021-12-20 11:09:35 -05:00
|
|
|
|