|
- #define FASTLED_INTERNAL
- #define __PROG_TYPES_COMPAT__
-
- #include <stdint.h>
- #include <math.h>
-
- #include "FastLED.h"
-
- FASTLED_NAMESPACE_BEGIN
-
-
-
- void fill_solid( struct CRGB * leds, int numToFill,
- const struct CRGB& color)
- {
- for( int i = 0; i < numToFill; i++) {
- leds[i] = color;
- }
- }
-
- void fill_solid( struct CHSV * targetArray, int numToFill,
- const struct CHSV& hsvColor)
- {
- for( int i = 0; i < numToFill; i++) {
- targetArray[i] = hsvColor;
- }
- }
-
-
- // void fill_solid( struct CRGB* targetArray, int numToFill,
- // const struct CHSV& hsvColor)
- // {
- // fill_solid<CRGB>( targetArray, numToFill, (CRGB) hsvColor);
- // }
-
- void fill_rainbow( struct CRGB * pFirstLED, int numToFill,
- uint8_t initialhue,
- uint8_t deltahue )
- {
- CHSV hsv;
- hsv.hue = initialhue;
- hsv.val = 255;
- hsv.sat = 240;
- for( int i = 0; i < numToFill; i++) {
- pFirstLED[i] = hsv;
- hsv.hue += deltahue;
- }
- }
-
- void fill_rainbow( struct CHSV * targetArray, int numToFill,
- uint8_t initialhue,
- uint8_t deltahue )
- {
- CHSV hsv;
- hsv.hue = initialhue;
- hsv.val = 255;
- hsv.sat = 240;
- for( int i = 0; i < numToFill; i++) {
- targetArray[i] = hsv;
- hsv.hue += deltahue;
- }
- }
-
-
- void fill_gradient_RGB( CRGB* leds,
- uint16_t startpos, CRGB startcolor,
- uint16_t endpos, CRGB endcolor )
- {
- // if the points are in the wrong order, straighten them
- if( endpos < startpos ) {
- uint16_t t = endpos;
- CRGB tc = endcolor;
- endcolor = startcolor;
- endpos = startpos;
- startpos = t;
- startcolor = tc;
- }
-
- saccum87 rdistance87;
- saccum87 gdistance87;
- saccum87 bdistance87;
-
- rdistance87 = (endcolor.r - startcolor.r) << 7;
- gdistance87 = (endcolor.g - startcolor.g) << 7;
- bdistance87 = (endcolor.b - startcolor.b) << 7;
-
- uint16_t pixeldistance = endpos - startpos;
- int16_t divisor = pixeldistance ? pixeldistance : 1;
-
- saccum87 rdelta87 = rdistance87 / divisor;
- saccum87 gdelta87 = gdistance87 / divisor;
- saccum87 bdelta87 = bdistance87 / divisor;
-
- rdelta87 *= 2;
- gdelta87 *= 2;
- bdelta87 *= 2;
-
- accum88 r88 = startcolor.r << 8;
- accum88 g88 = startcolor.g << 8;
- accum88 b88 = startcolor.b << 8;
- for( uint16_t i = startpos; i <= endpos; i++) {
- leds[i] = CRGB( r88 >> 8, g88 >> 8, b88 >> 8);
- r88 += rdelta87;
- g88 += gdelta87;
- b88 += bdelta87;
- }
- }
-
- #if 0
- void fill_gradient( const CHSV& c1, const CHSV& c2)
- {
- fill_gradient( FastLED[0].leds(), FastLED[0].size(), c1, c2);
- }
-
- void fill_gradient( const CHSV& c1, const CHSV& c2, const CHSV& c3)
- {
- fill_gradient( FastLED[0].leds(), FastLED[0].size(), c1, c2, c3);
- }
-
- void fill_gradient( const CHSV& c1, const CHSV& c2, const CHSV& c3, const CHSV& c4)
- {
- fill_gradient( FastLED[0].leds(), FastLED[0].size(), c1, c2, c3, c4);
- }
-
- void fill_gradient_RGB( const CRGB& c1, const CRGB& c2)
- {
- fill_gradient_RGB( FastLED[0].leds(), FastLED[0].size(), c1, c2);
- }
-
- void fill_gradient_RGB( const CRGB& c1, const CRGB& c2, const CRGB& c3)
- {
- fill_gradient_RGB( FastLED[0].leds(), FastLED[0].size(), c1, c2, c3);
- }
-
- void fill_gradient_RGB( const CRGB& c1, const CRGB& c2, const CRGB& c3, const CRGB& c4)
- {
- fill_gradient_RGB( FastLED[0].leds(), FastLED[0].size(), c1, c2, c3, c4);
- }
- #endif
-
-
-
-
- void fill_gradient_RGB( CRGB* leds, uint16_t numLeds, const CRGB& c1, const CRGB& c2)
- {
- uint16_t last = numLeds - 1;
- fill_gradient_RGB( leds, 0, c1, last, c2);
- }
-
-
- void fill_gradient_RGB( CRGB* leds, uint16_t numLeds, const CRGB& c1, const CRGB& c2, const CRGB& c3)
- {
- uint16_t half = (numLeds / 2);
- uint16_t last = numLeds - 1;
- fill_gradient_RGB( leds, 0, c1, half, c2);
- fill_gradient_RGB( leds, half, c2, last, c3);
- }
-
- void fill_gradient_RGB( CRGB* leds, uint16_t numLeds, const CRGB& c1, const CRGB& c2, const CRGB& c3, const CRGB& c4)
- {
- uint16_t onethird = (numLeds / 3);
- uint16_t twothirds = ((numLeds * 2) / 3);
- uint16_t last = numLeds - 1;
- fill_gradient_RGB( leds, 0, c1, onethird, c2);
- fill_gradient_RGB( leds, onethird, c2, twothirds, c3);
- fill_gradient_RGB( leds, twothirds, c3, last, c4);
- }
-
-
-
-
- void nscale8_video( CRGB* leds, uint16_t num_leds, uint8_t scale)
- {
- for( uint16_t i = 0; i < num_leds; i++) {
- leds[i].nscale8_video( scale);
- }
- }
-
- void fade_video(CRGB* leds, uint16_t num_leds, uint8_t fadeBy)
- {
- nscale8_video( leds, num_leds, 255 - fadeBy);
- }
-
- void fadeLightBy(CRGB* leds, uint16_t num_leds, uint8_t fadeBy)
- {
- nscale8_video( leds, num_leds, 255 - fadeBy);
- }
-
-
- void fadeToBlackBy( CRGB* leds, uint16_t num_leds, uint8_t fadeBy)
- {
- nscale8( leds, num_leds, 255 - fadeBy);
- }
-
- void fade_raw( CRGB* leds, uint16_t num_leds, uint8_t fadeBy)
- {
- nscale8( leds, num_leds, 255 - fadeBy);
- }
-
- void nscale8_raw( CRGB* leds, uint16_t num_leds, uint8_t scale)
- {
- nscale8( leds, num_leds, scale);
- }
-
- void nscale8( CRGB* leds, uint16_t num_leds, uint8_t scale)
- {
- for( uint16_t i = 0; i < num_leds; i++) {
- leds[i].nscale8( scale);
- }
- }
-
- void fadeUsingColor( CRGB* leds, uint16_t numLeds, const CRGB& colormask)
- {
- uint8_t fr, fg, fb;
- fr = colormask.r;
- fg = colormask.g;
- fb = colormask.b;
-
- for( uint16_t i = 0; i < numLeds; i++) {
- leds[i].r = scale8_LEAVING_R1_DIRTY( leds[i].r, fr);
- leds[i].g = scale8_LEAVING_R1_DIRTY( leds[i].g, fg);
- leds[i].b = scale8 ( leds[i].b, fb);
- }
- }
-
-
- CRGB& nblend( CRGB& existing, const CRGB& overlay, fract8 amountOfOverlay )
- {
- if( amountOfOverlay == 0) {
- return existing;
- }
-
- if( amountOfOverlay == 255) {
- existing = overlay;
- return existing;
- }
-
- #if 0
- // Old blend method which unfortunately had some rounding errors
- fract8 amountOfKeep = 255 - amountOfOverlay;
-
- existing.red = scale8_LEAVING_R1_DIRTY( existing.red, amountOfKeep)
- + scale8_LEAVING_R1_DIRTY( overlay.red, amountOfOverlay);
- existing.green = scale8_LEAVING_R1_DIRTY( existing.green, amountOfKeep)
- + scale8_LEAVING_R1_DIRTY( overlay.green, amountOfOverlay);
- existing.blue = scale8_LEAVING_R1_DIRTY( existing.blue, amountOfKeep)
- + scale8_LEAVING_R1_DIRTY( overlay.blue, amountOfOverlay);
-
- cleanup_R1();
- #else
- // Corrected blend method, with no loss-of-precision rounding errors
- existing.red = blend8( existing.red, overlay.red, amountOfOverlay);
- existing.green = blend8( existing.green, overlay.green, amountOfOverlay);
- existing.blue = blend8( existing.blue, overlay.blue, amountOfOverlay);
- #endif
-
- return existing;
- }
-
-
-
- void nblend( CRGB* existing, CRGB* overlay, uint16_t count, fract8 amountOfOverlay)
- {
- for( uint16_t i = count; i; i--) {
- nblend( *existing, *overlay, amountOfOverlay);
- existing++;
- overlay++;
- }
- }
-
- CRGB blend( const CRGB& p1, const CRGB& p2, fract8 amountOfP2 )
- {
- CRGB nu(p1);
- nblend( nu, p2, amountOfP2);
- return nu;
- }
-
- CRGB* blend( const CRGB* src1, const CRGB* src2, CRGB* dest, uint16_t count, fract8 amountOfsrc2 )
- {
- for( uint16_t i = 0; i < count; i++) {
- dest[i] = blend(src1[i], src2[i], amountOfsrc2);
- }
- return dest;
- }
-
-
-
- CHSV& nblend( CHSV& existing, const CHSV& overlay, fract8 amountOfOverlay, TGradientDirectionCode directionCode)
- {
- if( amountOfOverlay == 0) {
- return existing;
- }
-
- if( amountOfOverlay == 255) {
- existing = overlay;
- return existing;
- }
-
- fract8 amountOfKeep = 255 - amountOfOverlay;
-
- uint8_t huedelta8 = overlay.hue - existing.hue;
-
- if( directionCode == SHORTEST_HUES ) {
- directionCode = FORWARD_HUES;
- if( huedelta8 > 127) {
- directionCode = BACKWARD_HUES;
- }
- }
-
- if( directionCode == LONGEST_HUES ) {
- directionCode = FORWARD_HUES;
- if( huedelta8 < 128) {
- directionCode = BACKWARD_HUES;
- }
- }
-
- if( directionCode == FORWARD_HUES) {
- existing.hue = existing.hue + scale8( huedelta8, amountOfOverlay);
- }
- else /* directionCode == BACKWARD_HUES */
- {
- huedelta8 = -huedelta8;
- existing.hue = existing.hue - scale8( huedelta8, amountOfOverlay);
- }
-
- existing.sat = scale8_LEAVING_R1_DIRTY( existing.sat, amountOfKeep)
- + scale8_LEAVING_R1_DIRTY( overlay.sat, amountOfOverlay);
- existing.val = scale8_LEAVING_R1_DIRTY( existing.val, amountOfKeep)
- + scale8_LEAVING_R1_DIRTY( overlay.val, amountOfOverlay);
-
- cleanup_R1();
-
- return existing;
- }
-
-
-
- void nblend( CHSV* existing, CHSV* overlay, uint16_t count, fract8 amountOfOverlay, TGradientDirectionCode directionCode )
- {
- if(existing == overlay) return;
- for( uint16_t i = count; i; i--) {
- nblend( *existing, *overlay, amountOfOverlay, directionCode);
- existing++;
- overlay++;
- }
- }
-
- CHSV blend( const CHSV& p1, const CHSV& p2, fract8 amountOfP2, TGradientDirectionCode directionCode )
- {
- CHSV nu(p1);
- nblend( nu, p2, amountOfP2, directionCode);
- return nu;
- }
-
- CHSV* blend( const CHSV* src1, const CHSV* src2, CHSV* dest, uint16_t count, fract8 amountOfsrc2, TGradientDirectionCode directionCode )
- {
- for( uint16_t i = 0; i < count; i++) {
- dest[i] = blend(src1[i], src2[i], amountOfsrc2, directionCode);
- }
- return dest;
- }
-
-
-
- // Forward declaration of the function "XY" which must be provided by
- // the application for use in two-dimensional filter functions.
- uint16_t XY( uint8_t, uint8_t);// __attribute__ ((weak));
-
-
- // blur1d: one-dimensional blur filter. Spreads light to 2 line neighbors.
- // blur2d: two-dimensional blur filter. Spreads light to 8 XY neighbors.
- //
- // 0 = no spread at all
- // 64 = moderate spreading
- // 172 = maximum smooth, even spreading
- //
- // 173..255 = wider spreading, but increasing flicker
- //
- // Total light is NOT entirely conserved, so many repeated
- // calls to 'blur' will also result in the light fading,
- // eventually all the way to black; this is by design so that
- // it can be used to (slowly) clear the LEDs to black.
- void blur1d( CRGB* leds, uint16_t numLeds, fract8 blur_amount)
- {
- uint8_t keep = 255 - blur_amount;
- uint8_t seep = blur_amount >> 1;
- CRGB carryover = CRGB::Black;
- for( uint16_t i = 0; i < numLeds; i++) {
- CRGB cur = leds[i];
- CRGB part = cur;
- part.nscale8( seep);
- cur.nscale8( keep);
- cur += carryover;
- if( i) leds[i-1] += part;
- leds[i] = cur;
- carryover = part;
- }
- }
-
- void blur2d( CRGB* leds, uint8_t width, uint8_t height, fract8 blur_amount)
- {
- blurRows(leds, width, height, blur_amount);
- blurColumns(leds, width, height, blur_amount);
- }
-
- // blurRows: perform a blur1d on every row of a rectangular matrix
- void blurRows( CRGB* leds, uint8_t width, uint8_t height, fract8 blur_amount)
- {
- for( uint8_t row = 0; row < height; row++) {
- CRGB* rowbase = leds + (row * width);
- blur1d( rowbase, width, blur_amount);
- }
- }
-
- // blurColumns: perform a blur1d on each column of a rectangular matrix
- void blurColumns(CRGB* leds, uint8_t width, uint8_t height, fract8 blur_amount)
- {
- // blur columns
- uint8_t keep = 255 - blur_amount;
- uint8_t seep = blur_amount >> 1;
- for( uint8_t col = 0; col < width; col++) {
- CRGB carryover = CRGB::Black;
- for( uint8_t i = 0; i < height; i++) {
- CRGB cur = leds[XY(col,i)];
- CRGB part = cur;
- part.nscale8( seep);
- cur.nscale8( keep);
- cur += carryover;
- if( i) leds[XY(col,i-1)] += part;
- leds[XY(col,i)] = cur;
- carryover = part;
- }
- }
- }
-
-
-
- // CRGB HeatColor( uint8_t temperature)
- //
- // Approximates a 'black body radiation' spectrum for
- // a given 'heat' level. This is useful for animations of 'fire'.
- // Heat is specified as an arbitrary scale from 0 (cool) to 255 (hot).
- // This is NOT a chromatically correct 'black body radiation'
- // spectrum, but it's surprisingly close, and it's fast and small.
- //
- // On AVR/Arduino, this typically takes around 70 bytes of program memory,
- // versus 768 bytes for a full 256-entry RGB lookup table.
-
- CRGB HeatColor( uint8_t temperature)
- {
- CRGB heatcolor;
-
- // Scale 'heat' down from 0-255 to 0-191,
- // which can then be easily divided into three
- // equal 'thirds' of 64 units each.
- uint8_t t192 = scale8_video( temperature, 191);
-
- // calculate a value that ramps up from
- // zero to 255 in each 'third' of the scale.
- uint8_t heatramp = t192 & 0x3F; // 0..63
- heatramp <<= 2; // scale up to 0..252
-
- // now figure out which third of the spectrum we're in:
- if( t192 & 0x80) {
- // we're in the hottest third
- heatcolor.r = 255; // full red
- heatcolor.g = 255; // full green
- heatcolor.b = heatramp; // ramp up blue
-
- } else if( t192 & 0x40 ) {
- // we're in the middle third
- heatcolor.r = 255; // full red
- heatcolor.g = heatramp; // ramp up green
- heatcolor.b = 0; // no blue
-
- } else {
- // we're in the coolest third
- heatcolor.r = heatramp; // ramp up red
- heatcolor.g = 0; // no green
- heatcolor.b = 0; // no blue
- }
-
- return heatcolor;
- }
-
-
- // lsrX4: helper function to divide a number by 16, aka four LSR's.
- // On avr-gcc, "u8 >> 4" generates a loop, which is big, and slow.
- // merely forcing it to be four /=2's causes avr-gcc to emit
- // a SWAP instruction followed by an AND 0x0F, which is faster, and smaller.
- inline uint8_t lsrX4( uint8_t dividend) __attribute__((always_inline));
- inline uint8_t lsrX4( uint8_t dividend)
- {
- #if defined(__AVR__)
- dividend /= 2;
- dividend /= 2;
- dividend /= 2;
- dividend /= 2;
- #else
- dividend >>= 4;
- #endif
- return dividend;
- }
-
-
- CRGB ColorFromPalette( const CRGBPalette16& pal, uint8_t index, uint8_t brightness, TBlendType blendType)
- {
- // hi4 = index >> 4;
- uint8_t hi4 = lsrX4(index);
- uint8_t lo4 = index & 0x0F;
-
- // const CRGB* entry = &(pal[0]) + hi4;
- // since hi4 is always 0..15, hi4 * sizeof(CRGB) can be a single-byte value,
- // instead of the two byte 'int' that avr-gcc defaults to.
- // So, we multiply hi4 X sizeof(CRGB), giving hi4XsizeofCRGB;
- uint8_t hi4XsizeofCRGB = hi4 * sizeof(CRGB);
- // We then add that to a base array pointer.
- const CRGB* entry = (CRGB*)( (uint8_t*)(&(pal[0])) + hi4XsizeofCRGB);
-
- uint8_t blend = lo4 && (blendType != NOBLEND);
-
- uint8_t red1 = entry->red;
- uint8_t green1 = entry->green;
- uint8_t blue1 = entry->blue;
-
-
- if( blend ) {
-
- if( hi4 == 15 ) {
- entry = &(pal[0]);
- } else {
- entry++;
- }
-
- uint8_t f2 = lo4 << 4;
- uint8_t f1 = 255 - f2;
-
- // rgb1.nscale8(f1);
- uint8_t red2 = entry->red;
- red1 = scale8_LEAVING_R1_DIRTY( red1, f1);
- red2 = scale8_LEAVING_R1_DIRTY( red2, f2);
- red1 += red2;
-
- uint8_t green2 = entry->green;
- green1 = scale8_LEAVING_R1_DIRTY( green1, f1);
- green2 = scale8_LEAVING_R1_DIRTY( green2, f2);
- green1 += green2;
-
- uint8_t blue2 = entry->blue;
- blue1 = scale8_LEAVING_R1_DIRTY( blue1, f1);
- blue2 = scale8_LEAVING_R1_DIRTY( blue2, f2);
- blue1 += blue2;
-
- cleanup_R1();
- }
-
- if( brightness != 255) {
- if( brightness ) {
- brightness++; // adjust for rounding
- // Now, since brightness is nonzero, we don't need the full scale8_video logic;
- // we can just to scale8 and then add one (unless scale8 fixed) to all nonzero inputs.
- if( red1 ) {
- red1 = scale8_LEAVING_R1_DIRTY( red1, brightness);
- #if !(FASTLED_SCALE8_FIXED==1)
- red1++;
- #endif
- }
- if( green1 ) {
- green1 = scale8_LEAVING_R1_DIRTY( green1, brightness);
- #if !(FASTLED_SCALE8_FIXED==1)
- green1++;
- #endif
- }
- if( blue1 ) {
- blue1 = scale8_LEAVING_R1_DIRTY( blue1, brightness);
- #if !(FASTLED_SCALE8_FIXED==1)
- blue1++;
- #endif
- }
- cleanup_R1();
- } else {
- red1 = 0;
- green1 = 0;
- blue1 = 0;
- }
- }
-
- return CRGB( red1, green1, blue1);
- }
-
- CRGB ColorFromPalette( const TProgmemRGBPalette16& pal, uint8_t index, uint8_t brightness, TBlendType blendType)
- {
- // hi4 = index >> 4;
- uint8_t hi4 = lsrX4(index);
- uint8_t lo4 = index & 0x0F;
-
- CRGB entry = FL_PGM_READ_DWORD_NEAR( &(pal[0]) + hi4 );
-
-
- uint8_t red1 = entry.red;
- uint8_t green1 = entry.green;
- uint8_t blue1 = entry.blue;
-
- uint8_t blend = lo4 && (blendType != NOBLEND);
-
- if( blend ) {
-
- if( hi4 == 15 ) {
- entry = FL_PGM_READ_DWORD_NEAR( &(pal[0]) );
- } else {
- entry = FL_PGM_READ_DWORD_NEAR( &(pal[1]) + hi4 );
- }
-
- uint8_t f2 = lo4 << 4;
- uint8_t f1 = 255 - f2;
-
- uint8_t red2 = entry.red;
- red1 = scale8_LEAVING_R1_DIRTY( red1, f1);
- red2 = scale8_LEAVING_R1_DIRTY( red2, f2);
- red1 += red2;
-
- uint8_t green2 = entry.green;
- green1 = scale8_LEAVING_R1_DIRTY( green1, f1);
- green2 = scale8_LEAVING_R1_DIRTY( green2, f2);
- green1 += green2;
-
- uint8_t blue2 = entry.blue;
- blue1 = scale8_LEAVING_R1_DIRTY( blue1, f1);
- blue2 = scale8_LEAVING_R1_DIRTY( blue2, f2);
- blue1 += blue2;
-
- cleanup_R1();
- }
-
- if( brightness != 255) {
- if( brightness ) {
- brightness++; // adjust for rounding
- // Now, since brightness is nonzero, we don't need the full scale8_video logic;
- // we can just to scale8 and then add one (unless scale8 fixed) to all nonzero inputs.
- if( red1 ) {
- red1 = scale8_LEAVING_R1_DIRTY( red1, brightness);
- #if !(FASTLED_SCALE8_FIXED==1)
- red1++;
- #endif
- }
- if( green1 ) {
- green1 = scale8_LEAVING_R1_DIRTY( green1, brightness);
- #if !(FASTLED_SCALE8_FIXED==1)
- green1++;
- #endif
- }
- if( blue1 ) {
- blue1 = scale8_LEAVING_R1_DIRTY( blue1, brightness);
- #if !(FASTLED_SCALE8_FIXED==1)
- blue1++;
- #endif
- }
- cleanup_R1();
- } else {
- red1 = 0;
- green1 = 0;
- blue1 = 0;
- }
- }
-
- return CRGB( red1, green1, blue1);
- }
-
-
- CRGB ColorFromPalette( const CRGBPalette32& pal, uint8_t index, uint8_t brightness, TBlendType blendType)
- {
- uint8_t hi5 = index;
- #if defined(__AVR__)
- hi5 /= 2;
- hi5 /= 2;
- hi5 /= 2;
- #else
- hi5 >>= 3;
- #endif
- uint8_t lo3 = index & 0x07;
-
- // const CRGB* entry = &(pal[0]) + hi5;
- // since hi5 is always 0..31, hi4 * sizeof(CRGB) can be a single-byte value,
- // instead of the two byte 'int' that avr-gcc defaults to.
- // So, we multiply hi5 X sizeof(CRGB), giving hi5XsizeofCRGB;
- uint8_t hi5XsizeofCRGB = hi5 * sizeof(CRGB);
- // We then add that to a base array pointer.
- const CRGB* entry = (CRGB*)( (uint8_t*)(&(pal[0])) + hi5XsizeofCRGB);
-
- uint8_t red1 = entry->red;
- uint8_t green1 = entry->green;
- uint8_t blue1 = entry->blue;
-
- uint8_t blend = lo3 && (blendType != NOBLEND);
-
- if( blend ) {
-
- if( hi5 == 31 ) {
- entry = &(pal[0]);
- } else {
- entry++;
- }
-
- uint8_t f2 = lo3 << 5;
- uint8_t f1 = 255 - f2;
-
- uint8_t red2 = entry->red;
- red1 = scale8_LEAVING_R1_DIRTY( red1, f1);
- red2 = scale8_LEAVING_R1_DIRTY( red2, f2);
- red1 += red2;
-
- uint8_t green2 = entry->green;
- green1 = scale8_LEAVING_R1_DIRTY( green1, f1);
- green2 = scale8_LEAVING_R1_DIRTY( green2, f2);
- green1 += green2;
-
- uint8_t blue2 = entry->blue;
- blue1 = scale8_LEAVING_R1_DIRTY( blue1, f1);
- blue2 = scale8_LEAVING_R1_DIRTY( blue2, f2);
- blue1 += blue2;
-
- cleanup_R1();
-
- }
-
- if( brightness != 255) {
- if( brightness ) {
- brightness++; // adjust for rounding
- // Now, since brightness is nonzero, we don't need the full scale8_video logic;
- // we can just to scale8 and then add one (unless scale8 fixed) to all nonzero inputs.
- if( red1 ) {
- red1 = scale8_LEAVING_R1_DIRTY( red1, brightness);
- #if !(FASTLED_SCALE8_FIXED==1)
- red1++;
- #endif
- }
- if( green1 ) {
- green1 = scale8_LEAVING_R1_DIRTY( green1, brightness);
- #if !(FASTLED_SCALE8_FIXED==1)
- green1++;
- #endif
- }
- if( blue1 ) {
- blue1 = scale8_LEAVING_R1_DIRTY( blue1, brightness);
- #if !(FASTLED_SCALE8_FIXED==1)
- blue1++;
- #endif
- }
- cleanup_R1();
- } else {
- red1 = 0;
- green1 = 0;
- blue1 = 0;
- }
- }
-
- return CRGB( red1, green1, blue1);
- }
-
-
- CRGB ColorFromPalette( const TProgmemRGBPalette32& pal, uint8_t index, uint8_t brightness, TBlendType blendType)
- {
- uint8_t hi5 = index;
- #if defined(__AVR__)
- hi5 /= 2;
- hi5 /= 2;
- hi5 /= 2;
- #else
- hi5 >>= 3;
- #endif
- uint8_t lo3 = index & 0x07;
-
- CRGB entry = FL_PGM_READ_DWORD_NEAR( &(pal[0]) + hi5);
-
- uint8_t red1 = entry.red;
- uint8_t green1 = entry.green;
- uint8_t blue1 = entry.blue;
-
- uint8_t blend = lo3 && (blendType != NOBLEND);
-
- if( blend ) {
-
- if( hi5 == 31 ) {
- entry = FL_PGM_READ_DWORD_NEAR( &(pal[0]) );
- } else {
- entry = FL_PGM_READ_DWORD_NEAR( &(pal[1]) + hi5 );
- }
-
- uint8_t f2 = lo3 << 5;
- uint8_t f1 = 255 - f2;
-
- uint8_t red2 = entry.red;
- red1 = scale8_LEAVING_R1_DIRTY( red1, f1);
- red2 = scale8_LEAVING_R1_DIRTY( red2, f2);
- red1 += red2;
-
- uint8_t green2 = entry.green;
- green1 = scale8_LEAVING_R1_DIRTY( green1, f1);
- green2 = scale8_LEAVING_R1_DIRTY( green2, f2);
- green1 += green2;
-
- uint8_t blue2 = entry.blue;
- blue1 = scale8_LEAVING_R1_DIRTY( blue1, f1);
- blue2 = scale8_LEAVING_R1_DIRTY( blue2, f2);
- blue1 += blue2;
-
- cleanup_R1();
- }
-
- if( brightness != 255) {
- if( brightness ) {
- brightness++; // adjust for rounding
- // Now, since brightness is nonzero, we don't need the full scale8_video logic;
- // we can just to scale8 and then add one (unless scale8 fixed) to all nonzero inputs.
- if( red1 ) {
- red1 = scale8_LEAVING_R1_DIRTY( red1, brightness);
- #if !(FASTLED_SCALE8_FIXED==1)
- red1++;
- #endif
- }
- if( green1 ) {
- green1 = scale8_LEAVING_R1_DIRTY( green1, brightness);
- #if !(FASTLED_SCALE8_FIXED==1)
- green1++;
- #endif
- }
- if( blue1 ) {
- blue1 = scale8_LEAVING_R1_DIRTY( blue1, brightness);
- #if !(FASTLED_SCALE8_FIXED==1)
- blue1++;
- #endif
- }
- cleanup_R1();
- } else {
- red1 = 0;
- green1 = 0;
- blue1 = 0;
- }
- }
-
- return CRGB( red1, green1, blue1);
- }
-
-
-
- CRGB ColorFromPalette( const CRGBPalette256& pal, uint8_t index, uint8_t brightness, TBlendType)
- {
- const CRGB* entry = &(pal[0]) + index;
-
- uint8_t red = entry->red;
- uint8_t green = entry->green;
- uint8_t blue = entry->blue;
-
- if( brightness != 255) {
- brightness++; // adjust for rounding
- red = scale8_video_LEAVING_R1_DIRTY( red, brightness);
- green = scale8_video_LEAVING_R1_DIRTY( green, brightness);
- blue = scale8_video_LEAVING_R1_DIRTY( blue, brightness);
- cleanup_R1();
- }
-
- return CRGB( red, green, blue);
- }
-
-
- CHSV ColorFromPalette( const struct CHSVPalette16& pal, uint8_t index, uint8_t brightness, TBlendType blendType)
- {
- // hi4 = index >> 4;
- uint8_t hi4 = lsrX4(index);
- uint8_t lo4 = index & 0x0F;
-
- // CRGB rgb1 = pal[ hi4];
- const CHSV* entry = &(pal[0]) + hi4;
-
- uint8_t hue1 = entry->hue;
- uint8_t sat1 = entry->sat;
- uint8_t val1 = entry->val;
-
- uint8_t blend = lo4 && (blendType != NOBLEND);
-
- if( blend ) {
-
- if( hi4 == 15 ) {
- entry = &(pal[0]);
- } else {
- entry++;
- }
-
- uint8_t f2 = lo4 << 4;
- uint8_t f1 = 255 - f2;
-
- uint8_t hue2 = entry->hue;
- uint8_t sat2 = entry->sat;
- uint8_t val2 = entry->val;
-
- // Now some special casing for blending to or from
- // either black or white. Black and white don't have
- // proper 'hue' of their own, so when ramping from
- // something else to/from black/white, we set the 'hue'
- // of the black/white color to be the same as the hue
- // of the other color, so that you get the expected
- // brightness or saturation ramp, with hue staying
- // constant:
-
- // If we are starting from white (sat=0)
- // or black (val=0), adopt the target hue.
- if( sat1 == 0 || val1 == 0) {
- hue1 = hue2;
- }
-
- // If we are ending at white (sat=0)
- // or black (val=0), adopt the starting hue.
- if( sat2 == 0 || val2 == 0) {
- hue2 = hue1;
- }
-
-
- sat1 = scale8_LEAVING_R1_DIRTY( sat1, f1);
- val1 = scale8_LEAVING_R1_DIRTY( val1, f1);
-
- sat2 = scale8_LEAVING_R1_DIRTY( sat2, f2);
- val2 = scale8_LEAVING_R1_DIRTY( val2, f2);
-
- // cleanup_R1();
-
- // These sums can't overflow, so no qadd8 needed.
- sat1 += sat2;
- val1 += val2;
-
- uint8_t deltaHue = (uint8_t)(hue2 - hue1);
- if( deltaHue & 0x80 ) {
- // go backwards
- hue1 -= scale8( 256 - deltaHue, f2);
- } else {
- // go forwards
- hue1 += scale8( deltaHue, f2);
- }
-
- cleanup_R1();
- }
-
- if( brightness != 255) {
- val1 = scale8_video( val1, brightness);
- }
-
- return CHSV( hue1, sat1, val1);
- }
-
-
- CHSV ColorFromPalette( const struct CHSVPalette32& pal, uint8_t index, uint8_t brightness, TBlendType blendType)
- {
- uint8_t hi5 = index;
- #if defined(__AVR__)
- hi5 /= 2;
- hi5 /= 2;
- hi5 /= 2;
- #else
- hi5 >>= 3;
- #endif
- uint8_t lo3 = index & 0x07;
-
- uint8_t hi5XsizeofCHSV = hi5 * sizeof(CHSV);
- const CHSV* entry = (CHSV*)( (uint8_t*)(&(pal[0])) + hi5XsizeofCHSV);
-
- uint8_t hue1 = entry->hue;
- uint8_t sat1 = entry->sat;
- uint8_t val1 = entry->val;
-
- uint8_t blend = lo3 && (blendType != NOBLEND);
-
- if( blend ) {
-
- if( hi5 == 31 ) {
- entry = &(pal[0]);
- } else {
- entry++;
- }
-
- uint8_t f2 = lo3 << 5;
- uint8_t f1 = 255 - f2;
-
- uint8_t hue2 = entry->hue;
- uint8_t sat2 = entry->sat;
- uint8_t val2 = entry->val;
-
- // Now some special casing for blending to or from
- // either black or white. Black and white don't have
- // proper 'hue' of their own, so when ramping from
- // something else to/from black/white, we set the 'hue'
- // of the black/white color to be the same as the hue
- // of the other color, so that you get the expected
- // brightness or saturation ramp, with hue staying
- // constant:
-
- // If we are starting from white (sat=0)
- // or black (val=0), adopt the target hue.
- if( sat1 == 0 || val1 == 0) {
- hue1 = hue2;
- }
-
- // If we are ending at white (sat=0)
- // or black (val=0), adopt the starting hue.
- if( sat2 == 0 || val2 == 0) {
- hue2 = hue1;
- }
-
-
- sat1 = scale8_LEAVING_R1_DIRTY( sat1, f1);
- val1 = scale8_LEAVING_R1_DIRTY( val1, f1);
-
- sat2 = scale8_LEAVING_R1_DIRTY( sat2, f2);
- val2 = scale8_LEAVING_R1_DIRTY( val2, f2);
-
- // cleanup_R1();
-
- // These sums can't overflow, so no qadd8 needed.
- sat1 += sat2;
- val1 += val2;
-
- uint8_t deltaHue = (uint8_t)(hue2 - hue1);
- if( deltaHue & 0x80 ) {
- // go backwards
- hue1 -= scale8( 256 - deltaHue, f2);
- } else {
- // go forwards
- hue1 += scale8( deltaHue, f2);
- }
-
- cleanup_R1();
- }
-
- if( brightness != 255) {
- val1 = scale8_video( val1, brightness);
- }
-
- return CHSV( hue1, sat1, val1);
- }
-
- CHSV ColorFromPalette( const struct CHSVPalette256& pal, uint8_t index, uint8_t brightness, TBlendType)
- {
- CHSV hsv = *( &(pal[0]) + index );
-
- if( brightness != 255) {
- hsv.value = scale8_video( hsv.value, brightness);
- }
-
- return hsv;
- }
-
-
- void UpscalePalette(const struct CRGBPalette16& srcpal16, struct CRGBPalette256& destpal256)
- {
- for( int i = 0; i < 256; i++) {
- destpal256[(uint8_t)(i)] = ColorFromPalette( srcpal16, i);
- }
- }
-
- void UpscalePalette(const struct CHSVPalette16& srcpal16, struct CHSVPalette256& destpal256)
- {
- for( int i = 0; i < 256; i++) {
- destpal256[(uint8_t)(i)] = ColorFromPalette( srcpal16, i);
- }
- }
-
-
- void UpscalePalette(const struct CRGBPalette16& srcpal16, struct CRGBPalette32& destpal32)
- {
- for( uint8_t i = 0; i < 16; i++) {
- uint8_t j = i * 2;
- destpal32[j+0] = srcpal16[i];
- destpal32[j+1] = srcpal16[i];
- }
- }
-
- void UpscalePalette(const struct CHSVPalette16& srcpal16, struct CHSVPalette32& destpal32)
- {
- for( uint8_t i = 0; i < 16; i++) {
- uint8_t j = i * 2;
- destpal32[j+0] = srcpal16[i];
- destpal32[j+1] = srcpal16[i];
- }
- }
-
- void UpscalePalette(const struct CRGBPalette32& srcpal32, struct CRGBPalette256& destpal256)
- {
- for( int i = 0; i < 256; i++) {
- destpal256[(uint8_t)(i)] = ColorFromPalette( srcpal32, i);
- }
- }
-
- void UpscalePalette(const struct CHSVPalette32& srcpal32, struct CHSVPalette256& destpal256)
- {
- for( int i = 0; i < 256; i++) {
- destpal256[(uint8_t)(i)] = ColorFromPalette( srcpal32, i);
- }
- }
-
-
-
- #if 0
- // replaced by PartyColors_p
- void SetupPartyColors(CRGBPalette16& pal)
- {
- fill_gradient( pal, 0, CHSV( HUE_PURPLE,255,255), 7, CHSV(HUE_YELLOW - 18,255,255), FORWARD_HUES);
- fill_gradient( pal, 8, CHSV( HUE_ORANGE,255,255), 15, CHSV(HUE_BLUE + 18,255,255), BACKWARD_HUES);
- }
- #endif
-
-
- void nblendPaletteTowardPalette( CRGBPalette16& current, CRGBPalette16& target, uint8_t maxChanges)
- {
- uint8_t* p1;
- uint8_t* p2;
- uint8_t changes = 0;
-
- p1 = (uint8_t*)current.entries;
- p2 = (uint8_t*)target.entries;
-
- const uint8_t totalChannels = sizeof(CRGBPalette16);
- for( uint8_t i = 0; i < totalChannels; i++) {
- // if the values are equal, no changes are needed
- if( p1[i] == p2[i] ) { continue; }
-
- // if the current value is less than the target, increase it by one
- if( p1[i] < p2[i] ) { p1[i]++; changes++; }
-
- // if the current value is greater than the target,
- // increase it by one (or two if it's still greater).
- if( p1[i] > p2[i] ) {
- p1[i]--; changes++;
- if( p1[i] > p2[i] ) { p1[i]--; }
- }
-
- // if we've hit the maximum number of changes, exit
- if( changes >= maxChanges) { break; }
- }
- }
-
-
- uint8_t applyGamma_video( uint8_t brightness, float gamma)
- {
- float orig;
- float adj;
- orig = (float)(brightness) / (255.0);
- adj = pow( orig, gamma) * (255.0);
- uint8_t result = (uint8_t)(adj);
- if( (brightness > 0) && (result == 0)) {
- result = 1; // never gamma-adjust a positive number down to zero
- }
- return result;
- }
-
- CRGB applyGamma_video( const CRGB& orig, float gamma)
- {
- CRGB adj;
- adj.r = applyGamma_video( orig.r, gamma);
- adj.g = applyGamma_video( orig.g, gamma);
- adj.b = applyGamma_video( orig.b, gamma);
- return adj;
- }
-
- CRGB applyGamma_video( const CRGB& orig, float gammaR, float gammaG, float gammaB)
- {
- CRGB adj;
- adj.r = applyGamma_video( orig.r, gammaR);
- adj.g = applyGamma_video( orig.g, gammaG);
- adj.b = applyGamma_video( orig.b, gammaB);
- return adj;
- }
-
- CRGB& napplyGamma_video( CRGB& rgb, float gamma)
- {
- rgb = applyGamma_video( rgb, gamma);
- return rgb;
- }
-
- CRGB& napplyGamma_video( CRGB& rgb, float gammaR, float gammaG, float gammaB)
- {
- rgb = applyGamma_video( rgb, gammaR, gammaG, gammaB);
- return rgb;
- }
-
- void napplyGamma_video( CRGB* rgbarray, uint16_t count, float gamma)
- {
- for( uint16_t i = 0; i < count; i++) {
- rgbarray[i] = applyGamma_video( rgbarray[i], gamma);
- }
- }
-
- void napplyGamma_video( CRGB* rgbarray, uint16_t count, float gammaR, float gammaG, float gammaB)
- {
- for( uint16_t i = 0; i < count; i++) {
- rgbarray[i] = applyGamma_video( rgbarray[i], gammaR, gammaG, gammaB);
- }
- }
-
-
- FASTLED_NAMESPACE_END
|