7778 lines
		
	
	
		
			276 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			7778 lines
		
	
	
		
			276 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|   WS2812FX.cpp contains all effect methods
 | |
|   Harm Aldick - 2016
 | |
|   www.aldick.org
 | |
|   LICENSE
 | |
|   The MIT License (MIT)
 | |
|   Copyright (c) 2016  Harm Aldick
 | |
|   Permission is hereby granted, free of charge, to any person obtaining a copy
 | |
|   of this software and associated documentation files (the "Software"), to deal
 | |
|   in the Software without restriction, including without limitation the rights
 | |
|   to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | |
|   copies of the Software, and to permit persons to whom the Software is
 | |
|   furnished to do so, subject to the following conditions:
 | |
|   The above copyright notice and this permission notice shall be included in
 | |
|   all copies or substantial portions of the Software.
 | |
|   THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | |
|   IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | |
|   FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | |
|   AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | |
|   LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | |
|   OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 | |
|   THE SOFTWARE.
 | |
| 
 | |
|   Modified heavily for WLED
 | |
| */
 | |
| 
 | |
| #include "FX.h"
 | |
| #include "wled.h"
 | |
| #include "fcn_declare.h"
 | |
| 
 | |
| #define IBN 5100
 | |
| #define PALETTE_SOLID_WRAP (paletteBlend == 1 || paletteBlend == 3)
 | |
| 
 | |
| /*
 | |
|  * No blinking. Just plain old static light.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_static(void) {
 | |
|   fill(SEGCOLOR(0));
 | |
|   return (SEGMENT.getOption(SEG_OPTION_TRANSITIONAL)) ? FRAMETIME : 350; //update faster if in transition
 | |
| }
 | |
| static const char *_data_FX_MODE_STATIC PROGMEM = "Solid";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Blink/strobe function
 | |
|  * Alternate between color1 and color2
 | |
|  * if(strobe == true) then create a strobe effect
 | |
|  */
 | |
| uint16_t WS2812FX::blink(uint32_t color1, uint32_t color2, bool strobe, bool do_palette) {
 | |
|   uint32_t cycleTime = (255 - SEGMENT.speed)*20;
 | |
|   uint32_t onTime = FRAMETIME;
 | |
|   if (!strobe) onTime += ((cycleTime * SEGMENT.intensity) >> 8);
 | |
|   cycleTime += FRAMETIME*2;
 | |
|   uint32_t it = now / cycleTime;
 | |
|   uint32_t rem = now % cycleTime;
 | |
|   
 | |
|   bool on = false;
 | |
|   if (it != SEGENV.step //new iteration, force on state for one frame, even if set time is too brief
 | |
|       || rem <= onTime) { 
 | |
|     on = true;
 | |
|   }
 | |
|   
 | |
|   SEGENV.step = it; //save previous iteration
 | |
| 
 | |
|   uint32_t color = on ? color1 : color2;
 | |
|   if (color == color1 && do_palette)
 | |
|   {
 | |
|     for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|       setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 0));
 | |
|     }
 | |
|   } else fill(color);
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Normal blinking. 50% on/off time.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_blink(void) {
 | |
|   return blink(SEGCOLOR(0), SEGCOLOR(1), false, true);
 | |
| }
 | |
| static const char *_data_FX_MODE_BLINK PROGMEM = "Blink@!,;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Classic Blink effect. Cycling through the rainbow.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_blink_rainbow(void) {
 | |
|   return blink(color_wheel(SEGENV.call & 0xFF), SEGCOLOR(1), false, false);
 | |
| }
 | |
| static const char *_data_FX_MODE_BLINK_RAINBOW PROGMEM = "Blink Rainbow@Frequency,Blink duration;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Classic Strobe effect.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_strobe(void) {
 | |
|   return blink(SEGCOLOR(0), SEGCOLOR(1), true, true);
 | |
| }
 | |
| static const char *_data_FX_MODE_STROBE PROGMEM = "Strobe@!,;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Classic Strobe effect. Cycling through the rainbow.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_strobe_rainbow(void) {
 | |
|   return blink(color_wheel(SEGENV.call & 0xFF), SEGCOLOR(1), true, false);
 | |
| }
 | |
| static const char *_data_FX_MODE_STROBE_RAINBOW PROGMEM = "Strobe Rainbow@!,;,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Color wipe function
 | |
|  * LEDs are turned on (color1) in sequence, then turned off (color2) in sequence.
 | |
|  * if (bool rev == true) then LEDs are turned off in reverse order
 | |
|  */
 | |
| uint16_t WS2812FX::color_wipe(bool rev, bool useRandomColors) {
 | |
|   uint32_t cycleTime = 750 + (255 - SEGMENT.speed)*150;
 | |
|   uint32_t perc = now % cycleTime;
 | |
|   uint16_t prog = (perc * 65535) / cycleTime;
 | |
|   bool back = (prog > 32767);
 | |
|   if (back) {
 | |
|     prog -= 32767;
 | |
|     if (SEGENV.step == 0) SEGENV.step = 1;
 | |
|   } else {
 | |
|     if (SEGENV.step == 2) SEGENV.step = 3; //trigger color change
 | |
|   }
 | |
| 
 | |
|   if (useRandomColors) {
 | |
|     if (SEGENV.call == 0) {
 | |
|       SEGENV.aux0 = random8();
 | |
|       SEGENV.step = 3;
 | |
|     }
 | |
|     if (SEGENV.step == 1) { //if flag set, change to new random color
 | |
|       SEGENV.aux1 = get_random_wheel_index(SEGENV.aux0);
 | |
|       SEGENV.step = 2;
 | |
|     }
 | |
|     if (SEGENV.step == 3) {
 | |
|       SEGENV.aux0 = get_random_wheel_index(SEGENV.aux1);
 | |
|       SEGENV.step = 0;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   uint16_t ledIndex = (prog * SEGLEN) >> 15;
 | |
|   uint16_t rem = 0;
 | |
|   rem = (prog * SEGLEN) * 2; //mod 0xFFFF
 | |
|   rem /= (SEGMENT.intensity +1);
 | |
|   if (rem > 255) rem = 255;
 | |
| 
 | |
|   uint32_t col1 = useRandomColors? color_wheel(SEGENV.aux1) : SEGCOLOR(1);
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++)
 | |
|   {
 | |
|     uint16_t index = (rev && back)? SEGLEN -1 -i : i;
 | |
|     uint32_t col0 = useRandomColors? color_wheel(SEGENV.aux0) : color_from_palette(index, true, PALETTE_SOLID_WRAP, 0);
 | |
|     
 | |
|     if (i < ledIndex) 
 | |
|     {
 | |
|       setPixelColor(index, back? col1 : col0);
 | |
|     } else
 | |
|     {
 | |
|       setPixelColor(index, back? col0 : col1);
 | |
|       if (i == ledIndex) setPixelColor(index, color_blend(back? col0 : col1, back? col1 : col0, rem));
 | |
|     }
 | |
|   } 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Lights all LEDs one after another.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_color_wipe(void) {
 | |
|   return color_wipe(false, false);
 | |
| }
 | |
| static const char *_data_FX_MODE_COLOR_WIPE PROGMEM = "Wipe@!,!;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Lights all LEDs one after another. Turns off opposite
 | |
|  */
 | |
| uint16_t WS2812FX::mode_color_sweep(void) {
 | |
|   return color_wipe(true, false);
 | |
| }
 | |
| static const char *_data_FX_MODE_COLOR_SWEEP PROGMEM = "Sweep@!,!;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Turns all LEDs after each other to a random color.
 | |
|  * Then starts over with another color.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_color_wipe_random(void) {
 | |
|   return color_wipe(false, true);
 | |
| }
 | |
| static const char *_data_FX_MODE_COLOR_WIPE_RANDOM PROGMEM = "Wipe Random@!,;1,2,3;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Random color introduced alternating from start and end of strip.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_color_sweep_random(void) {
 | |
|   return color_wipe(true, true);
 | |
| }
 | |
| static const char *_data_FX_MODE_COLOR_SWEEP_RANDOM PROGMEM = "Sweep Random";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Lights all LEDs in one random color up. Then switches them
 | |
|  * to the next random color.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_random_color(void) {
 | |
|   uint32_t cycleTime = 200 + (255 - SEGMENT.speed)*50;
 | |
|   uint32_t it = now / cycleTime;
 | |
|   uint32_t rem = now % cycleTime;
 | |
|   uint16_t fadedur = (cycleTime * SEGMENT.intensity) >> 8;
 | |
| 
 | |
|   uint32_t fade = 255;
 | |
|   if (fadedur) {
 | |
|     fade = (rem * 255) / fadedur;
 | |
|     if (fade > 255) fade = 255;
 | |
|   }
 | |
| 
 | |
|   if (SEGENV.call == 0) {
 | |
|     SEGENV.aux0 = random8();
 | |
|     SEGENV.step = 2;
 | |
|   }
 | |
|   if (it != SEGENV.step) //new color
 | |
|   {
 | |
|     SEGENV.aux1 = SEGENV.aux0;
 | |
|     SEGENV.aux0 = get_random_wheel_index(SEGENV.aux0); //aux0 will store our random color wheel index
 | |
|     SEGENV.step = it;
 | |
|   }
 | |
| 
 | |
|   fill(color_blend(color_wheel(SEGENV.aux1), color_wheel(SEGENV.aux0), fade));
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_RANDOM_COLOR PROGMEM = "Random Colors@!,Fade time;1,2,3;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Lights every LED in a random color. Changes all LED at the same time
 | |
|  * to new random colors.
 | |
|  */
 | |
| uint16_t WS2812FX::dynamic(boolean smooth=false) {
 | |
|   if (!SEGENV.allocateData(SEGLEN)) return mode_static(); //allocation failed
 | |
|   
 | |
|   if(SEGENV.call == 0) {
 | |
|     for (uint16_t i = 0; i < SEGLEN; i++) SEGENV.data[i] = random8();
 | |
|   }
 | |
| 
 | |
|   uint32_t cycleTime = 50 + (255 - SEGMENT.speed)*15;
 | |
|   uint32_t it = now / cycleTime;
 | |
|   if (it != SEGENV.step && SEGMENT.speed != 0) //new color
 | |
|   {
 | |
|     for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|       if (random8() <= SEGMENT.intensity) SEGENV.data[i] = random8();
 | |
|     }
 | |
|     SEGENV.step = it;
 | |
|   }
 | |
|   
 | |
|   if (smooth) {
 | |
|     for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|       blendPixelColor(i, color_wheel(SEGENV.data[i]),16);
 | |
|     }
 | |
|   } else {
 | |
|     for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|       setPixelColor(i, color_wheel(SEGENV.data[i]));
 | |
|     }
 | |
|   } 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Original effect "Dynamic"
 | |
|  */
 | |
| uint16_t WS2812FX::mode_dynamic(void) {
 | |
|   return dynamic(false);
 | |
| }
 | |
| static const char *_data_FX_MODE_DYNAMIC PROGMEM = "Dynamic@!,!;1,2,3;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * effect "Dynamic" with smoth color-fading
 | |
|  */
 | |
| uint16_t WS2812FX::mode_dynamic_smooth(void) {
 | |
|   return dynamic(true);
 | |
|  }
 | |
| static const char *_data_FX_MODE_DYNAMIC_SMOOTH PROGMEM = "Dynamic Smooth";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Does the "standby-breathing" of well known i-Devices.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_breath(void) {
 | |
|   uint16_t var = 0;
 | |
|   uint16_t counter = (now * ((SEGMENT.speed >> 3) +10));
 | |
|   counter = (counter >> 2) + (counter >> 4); //0-16384 + 0-2048
 | |
|   if (counter < 16384) {
 | |
|     if (counter > 8192) counter = 8192 - (counter - 8192);
 | |
|     var = sin16(counter) / 103; //close to parabolic in range 0-8192, max val. 23170
 | |
|   }
 | |
|   
 | |
|   uint8_t lum = 30 + var;
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     setPixelColor(i, color_blend(SEGCOLOR(1), color_from_palette(i, true, PALETTE_SOLID_WRAP, 0), lum));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_BREATH PROGMEM = "Breathe@!,;!,!;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Fades the LEDs between two colors
 | |
|  */
 | |
| uint16_t WS2812FX::mode_fade(void) {
 | |
|   uint16_t counter = (now * ((SEGMENT.speed >> 3) +10));
 | |
|   uint8_t lum = triwave16(counter) >> 8;
 | |
| 
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     setPixelColor(i, color_blend(SEGCOLOR(1), color_from_palette(i, true, PALETTE_SOLID_WRAP, 0), lum));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_FADE PROGMEM = "Fade@!,;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Scan mode parent function
 | |
|  */
 | |
| uint16_t WS2812FX::scan(bool dual)
 | |
| {
 | |
|   uint32_t cycleTime = 750 + (255 - SEGMENT.speed)*150;
 | |
|   uint32_t perc = now % cycleTime;
 | |
|   uint16_t prog = (perc * 65535) / cycleTime;
 | |
|   uint16_t size = 1 + ((SEGMENT.intensity * SEGLEN) >> 9);
 | |
|   uint16_t ledIndex = (prog * ((SEGLEN *2) - size *2)) >> 16;
 | |
| 
 | |
|   fill(SEGCOLOR(1));
 | |
| 
 | |
|   int led_offset = ledIndex - (SEGLEN - size);
 | |
|   led_offset = abs(led_offset);
 | |
| 
 | |
|   if (dual) {
 | |
|     for (uint16_t j = led_offset; j < led_offset + size; j++) {
 | |
|       uint16_t i2 = SEGLEN -1 -j;
 | |
|       setPixelColor(i2, color_from_palette(i2, true, PALETTE_SOLID_WRAP, (SEGCOLOR(2))? 2:0));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for (uint16_t j = led_offset; j < led_offset + size; j++) {
 | |
|     setPixelColor(j, color_from_palette(j, true, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Runs a single pixel back and forth.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_scan(void) {
 | |
|   return scan(false);
 | |
| }
 | |
| static const char *_data_FX_MODE_SCAN PROGMEM = "Scan@!,# of dots;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Runs two pixel back and forth in opposite directions.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_dual_scan(void) {
 | |
|   return scan(true);
 | |
| }
 | |
| static const char *_data_FX_MODE_DUAL_SCAN PROGMEM = "Scan Dual@!,# of dots;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Cycles all LEDs at once through a rainbow.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_rainbow(void) {
 | |
|   uint16_t counter = (now * ((SEGMENT.speed >> 2) +2)) & 0xFFFF;
 | |
|   counter = counter >> 8;
 | |
| 
 | |
|   if (SEGMENT.intensity < 128){
 | |
|     fill(color_blend(color_wheel(counter),WHITE,128-SEGMENT.intensity));
 | |
|   } else {
 | |
|     fill(color_wheel(counter));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_RAINBOW PROGMEM = "Colorloop@!,Saturation;1,2,3;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Cycles a rainbow over the entire string of LEDs.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_rainbow_cycle(void) {
 | |
|   uint16_t counter = (now * ((SEGMENT.speed >> 2) +2)) & 0xFFFF;
 | |
|   counter = counter >> 8;
 | |
|   
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     //intensity/29 = 0 (1/16) 1 (1/8) 2 (1/4) 3 (1/2) 4 (1) 5 (2) 6 (4) 7 (8) 8 (16)
 | |
|     uint8_t index = (i * (16 << (SEGMENT.intensity /29)) / SEGLEN) + counter;
 | |
|     setPixelColor(i, color_wheel(index));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_RAINBOW_CYCLE PROGMEM = "Rainbow@!,Size;1,2,3;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Theatre-style crawling lights.
 | |
|  * Inspired by the Adafruit examples.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_theater_chase(void) {
 | |
|   return running(SEGCOLOR(0), SEGCOLOR(1), true);
 | |
| }
 | |
| static const char *_data_FX_MODE_THEATER_CHASE PROGMEM = "Theater@!,Gap size;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Theatre-style crawling lights with rainbow effect.
 | |
|  * Inspired by the Adafruit examples.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_theater_chase_rainbow(void) {
 | |
|   return running(color_wheel(SEGENV.step), SEGCOLOR(1), true);
 | |
| }
 | |
| static const char *_data_FX_MODE_THEATER_CHASE_RAINBOW PROGMEM = "Theater Rainbow@!,Gap size;1,2,3;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Running lights effect with smooth sine transition base.
 | |
|  */
 | |
| uint16_t WS2812FX::running_base(bool saw, bool dual=false) {
 | |
|   uint8_t x_scale = SEGMENT.intensity >> 2;
 | |
|   uint32_t counter = (now * SEGMENT.speed) >> 9;
 | |
| 
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     uint16_t a = i*x_scale - counter;
 | |
|     if (saw) {
 | |
|       a &= 0xFF;
 | |
|       if (a < 16)
 | |
|       {
 | |
|         a = 192 + a*8;
 | |
|       } else {
 | |
|         a = map(a,16,255,64,192);
 | |
|       }
 | |
|       a = 255 - a;
 | |
|     }
 | |
|     uint8_t s = dual ? sin_gap(a) : sin8(a);
 | |
|     uint32_t ca = color_blend(SEGCOLOR(1), color_from_palette(i, true, PALETTE_SOLID_WRAP, 0), s);
 | |
|     if (dual) {
 | |
|       uint16_t b = (SEGLEN-1-i)*x_scale - counter;
 | |
|       uint8_t t = sin_gap(b);
 | |
|       uint32_t cb = color_blend(SEGCOLOR(1), color_from_palette(i, true, PALETTE_SOLID_WRAP, 2), t);
 | |
|       ca = color_blend(ca, cb, 127);
 | |
|     }
 | |
|     setPixelColor(i, ca);
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Running lights in opposite directions.
 | |
|  * Idea: Make the gap width controllable with a third slider in the future
 | |
|  */
 | |
| uint16_t WS2812FX::mode_running_dual(void) {
 | |
|   return running_base(false, true);
 | |
| }
 | |
| static const char *_data_FX_MODE_RUNNING_DUAL PROGMEM = "Running Dual";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Running lights effect with smooth sine transition.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_running_lights(void) {
 | |
|   return running_base(false);
 | |
| }
 | |
| static const char *_data_FX_MODE_RUNNING_LIGHTS PROGMEM = "Running@!,Wave width;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Running lights effect with sawtooth transition.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_saw(void) {
 | |
|   return running_base(true);
 | |
| }
 | |
| static const char *_data_FX_MODE_SAW PROGMEM = "Saw@!,Width;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Blink several LEDs in random colors on, reset, repeat.
 | |
|  * Inspired by www.tweaking4all.com/hardware/arduino/adruino-led-strip-effects/
 | |
|  */
 | |
| uint16_t WS2812FX::mode_twinkle(void) {
 | |
|   const uint16_t cols = isMatrix ? SEGMENT.virtualWidth() : SEGMENT.virtualLength();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
| 
 | |
|   fill(SEGCOLOR(1));
 | |
| 
 | |
|   uint32_t cycleTime = 20 + (255 - SEGMENT.speed)*5;
 | |
|   uint32_t it = now / cycleTime;
 | |
|   if (it != SEGENV.step)
 | |
|   {
 | |
|     uint16_t maxOn = map(SEGMENT.intensity, 0, 255, 1, cols*rows-1); // make sure at least one LED is on
 | |
|     if (SEGENV.aux0 >= maxOn)
 | |
|     {
 | |
|       SEGENV.aux0 = 0;
 | |
|       SEGENV.aux1 = random16(); //new seed for our PRNG
 | |
|     }
 | |
|     SEGENV.aux0++;
 | |
|     SEGENV.step = it;
 | |
|   }
 | |
|   
 | |
|   uint16_t PRNG16 = SEGENV.aux1;
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGENV.aux0; i++)
 | |
|   {
 | |
|     PRNG16 = (uint16_t)(PRNG16 * 2053) + 13849; // next 'random' number
 | |
|     uint32_t p = ((uint32_t)cols*rows * (uint32_t)PRNG16) >> 16;
 | |
|     uint16_t j = p % cols;
 | |
|     uint16_t k = p / cols;
 | |
|     uint32_t col = color_from_palette(map(p, 0, cols*rows, 0, 255), false, PALETTE_SOLID_WRAP, 0);
 | |
|     if (isMatrix) setPixelColorXY(j, k, col);
 | |
|     else          setPixelColor(j, col);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_TWINKLE PROGMEM = "Twinkle@!,;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Dissolve function
 | |
|  */
 | |
| uint16_t WS2812FX::dissolve(uint32_t color) {
 | |
|   bool wa = (SEGCOLOR(1) != 0 && _brightness < 255); //workaround, can't compare getPixel to color if not full brightness
 | |
|   
 | |
|   for (uint16_t j = 0; j <= SEGLEN / 15; j++)
 | |
|   {
 | |
|     if (random8() <= SEGMENT.intensity) {
 | |
|       for (uint8_t times = 0; times < 10; times++) //attempt to spawn a new pixel 5 times
 | |
|       {
 | |
|         uint16_t i = random16(SEGLEN);
 | |
|         if (SEGENV.aux0) { //dissolve to primary/palette
 | |
|           if (getPixelColor(i) == SEGCOLOR(1) || wa) {
 | |
|             if (color == SEGCOLOR(0))
 | |
|             {
 | |
|               setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 0));
 | |
|             } else { setPixelColor(i, color); }     
 | |
|             break; //only spawn 1 new pixel per frame per 50 LEDs
 | |
|           }
 | |
|         } else { //dissolve to secondary
 | |
|           if (getPixelColor(i) != SEGCOLOR(1)) { setPixelColor(i, SEGCOLOR(1)); break; }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (SEGENV.call > (255 - SEGMENT.speed) + 15U) 
 | |
|   {
 | |
|     SEGENV.aux0 = !SEGENV.aux0;
 | |
|     SEGENV.call = 0;
 | |
|   }
 | |
|   
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Blink several LEDs on and then off
 | |
|  */
 | |
| uint16_t WS2812FX::mode_dissolve(void) {
 | |
|   return dissolve(SEGCOLOR(0));
 | |
| }
 | |
| static const char *_data_FX_MODE_DISSOLVE PROGMEM = "Dissolve@Repeat speed,Dissolve speed;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Blink several LEDs on and then off in random colors
 | |
|  */
 | |
| uint16_t WS2812FX::mode_dissolve_random(void) {
 | |
|   return dissolve(color_wheel(random8()));
 | |
| }
 | |
| static const char *_data_FX_MODE_DISSOLVE_RANDOM PROGMEM = "Dissolve Rnd@Repeat speed,Dissolve speed;,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Blinks one LED at a time.
 | |
|  * Inspired by www.tweaking4all.com/hardware/arduino/adruino-led-strip-effects/
 | |
|  */
 | |
| uint16_t WS2812FX::mode_sparkle(void) {
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 1));
 | |
|   }
 | |
|   uint32_t cycleTime = 10 + (255 - SEGMENT.speed)*2;
 | |
|   uint32_t it = now / cycleTime;
 | |
|   if (it != SEGENV.step)
 | |
|   {
 | |
|     SEGENV.aux0 = random16(SEGLEN); // aux0 stores the random led index
 | |
|     SEGENV.aux1 = random16(0,SEGMENT.virtualHeight()-1);
 | |
|     SEGENV.step = it;
 | |
|   }
 | |
|   
 | |
|   if (isMatrix) setPixelColorXY(SEGENV.aux0, SEGENV.aux1, SEGCOLOR(0));
 | |
|   else          setPixelColor(SEGENV.aux0, SEGCOLOR(0));
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_SPARKLE PROGMEM = "Sparkle@!,;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Lights all LEDs in the color. Flashes single col 1 pixels randomly. (List name: Sparkle Dark)
 | |
|  * Inspired by www.tweaking4all.com/hardware/arduino/adruino-led-strip-effects/
 | |
|  */
 | |
| uint16_t WS2812FX::mode_flash_sparkle(void) {
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   if (now - SEGENV.aux0 > SEGENV.step) {
 | |
|     if(random8((255-SEGMENT.intensity) >> 4) == 0) {
 | |
|       if (isMatrix) setPixelColorXY(random16(SEGLEN), random16(0,SEGMENT.virtualHeight()-1), SEGCOLOR(1));
 | |
|       else          setPixelColor(random16(SEGLEN), SEGCOLOR(1)); //flash
 | |
|     }
 | |
|     SEGENV.step = now;
 | |
|     SEGENV.aux0 = 255-SEGMENT.speed;
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_FLASH_SPARKLE PROGMEM = "Sparkle Dark@!,!;Bg,Fx,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Like flash sparkle. With more flash.
 | |
|  * Inspired by www.tweaking4all.com/hardware/arduino/adruino-led-strip-effects/
 | |
|  */
 | |
| uint16_t WS2812FX::mode_hyper_sparkle(void) {
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   if (now - SEGENV.aux0 > SEGENV.step) {
 | |
|     if(random8((255-SEGMENT.intensity) >> 4) == 0) {
 | |
|       for(uint16_t i = 0; i < MAX(1, SEGLEN/3); i++) {
 | |
|         if (isMatrix) setPixelColorXY(random16(SEGLEN), random16(0,SEGMENT.virtualHeight()), SEGCOLOR(1));
 | |
|         else          setPixelColor(random16(SEGLEN), SEGCOLOR(1));
 | |
|       }
 | |
|     }
 | |
|     SEGENV.step = now;
 | |
|     SEGENV.aux0 = 255-SEGMENT.speed;
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_HYPER_SPARKLE PROGMEM = "Sparkle+@!,!;Bg,Fx,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Strobe effect with different strobe count and pause, controlled by speed.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_multi_strobe(void) {
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 1));
 | |
|   }
 | |
| 
 | |
|   SEGENV.aux0 = 50 + 20*(uint16_t)(255-SEGMENT.speed);
 | |
|   uint16_t count = 2 * ((SEGMENT.intensity / 10) + 1);
 | |
|   if(SEGENV.aux1 < count) {
 | |
|     if((SEGENV.aux1 & 1) == 0) {
 | |
|       fill(SEGCOLOR(0));
 | |
|       SEGENV.aux0 = 15;
 | |
|     } else {
 | |
|       SEGENV.aux0 = 50;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (now - SEGENV.aux0 > SEGENV.step) {
 | |
|     SEGENV.aux1++;
 | |
|     if (SEGENV.aux1 > count) SEGENV.aux1 = 0;
 | |
|     SEGENV.step = now;
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_MULTI_STROBE PROGMEM = "Strobe Mega@!,!;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Android loading circle
 | |
|  */
 | |
| uint16_t WS2812FX::mode_android(void) {
 | |
|   
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 1));
 | |
|   }
 | |
| 
 | |
|   if (SEGENV.aux1 > ((float)SEGMENT.intensity/255.0)*(float)SEGLEN)
 | |
|   {
 | |
|     SEGENV.aux0 = 1;
 | |
|   } else
 | |
|   {
 | |
|     if (SEGENV.aux1 < 2) SEGENV.aux0 = 0;
 | |
|   }
 | |
| 
 | |
|   uint16_t a = SEGENV.step;
 | |
|   
 | |
|   if (SEGENV.aux0 == 0)
 | |
|   {
 | |
|     if (SEGENV.call %3 == 1) {a++;}
 | |
|     else {SEGENV.aux1++;}
 | |
|   } else
 | |
|   {
 | |
|     a++;
 | |
|     if (SEGENV.call %3 != 1) SEGENV.aux1--;
 | |
|   }
 | |
|   
 | |
|   if (a >= SEGLEN) a = 0;
 | |
| 
 | |
|   if (a + SEGENV.aux1 < SEGLEN)
 | |
|   {
 | |
|     for(uint16_t i = a; i < a+SEGENV.aux1; i++) {
 | |
|       setPixelColor(i, SEGCOLOR(0));
 | |
|     }
 | |
|   } else
 | |
|   {
 | |
|     for(uint16_t i = a; i < SEGLEN; i++) {
 | |
|       setPixelColor(i, SEGCOLOR(0));
 | |
|     }
 | |
|     for(uint16_t i = 0; i < SEGENV.aux1 - (SEGLEN -a); i++) {
 | |
|       setPixelColor(i, SEGCOLOR(0));
 | |
|     }
 | |
|   }
 | |
|   SEGENV.step = a;
 | |
| 
 | |
|   return 3 + ((8 * (uint32_t)(255 - SEGMENT.speed)) / SEGLEN);
 | |
| }
 | |
| static const char *_data_FX_MODE_ANDROID PROGMEM = "Android@!,Width;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * color chase function.
 | |
|  * color1 = background color
 | |
|  * color2 and color3 = colors of two adjacent leds
 | |
|  */
 | |
| uint16_t WS2812FX::chase(uint32_t color1, uint32_t color2, uint32_t color3, bool do_palette) {
 | |
|   uint16_t counter = now * ((SEGMENT.speed >> 2) + 1);
 | |
|   uint16_t a = counter * SEGLEN  >> 16;
 | |
| 
 | |
|   bool chase_random = (SEGMENT.mode == FX_MODE_CHASE_RANDOM);
 | |
|   if (chase_random) {
 | |
|     if (a < SEGENV.step) //we hit the start again, choose new color for Chase random
 | |
|     {
 | |
|       SEGENV.aux1 = SEGENV.aux0; //store previous random color
 | |
|       SEGENV.aux0 = get_random_wheel_index(SEGENV.aux0);
 | |
|     }
 | |
|     color1 = color_wheel(SEGENV.aux0);
 | |
|   }
 | |
|   SEGENV.step = a;
 | |
| 
 | |
|   // Use intensity setting to vary chase up to 1/2 string length
 | |
|   uint8_t size = 1 + (SEGMENT.intensity * SEGLEN >> 10);
 | |
| 
 | |
|   uint16_t b = a + size; //"trail" of chase, filled with color1 
 | |
|   if (b > SEGLEN) b -= SEGLEN;
 | |
|   uint16_t c = b + size;
 | |
|   if (c > SEGLEN) c -= SEGLEN;
 | |
| 
 | |
|   //background
 | |
|   if (do_palette)
 | |
|   {
 | |
|     for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|       setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 1));
 | |
|     }
 | |
|   } else fill(color1);
 | |
| 
 | |
|   //if random, fill old background between a and end
 | |
|   if (chase_random)
 | |
|   {
 | |
|     color1 = color_wheel(SEGENV.aux1);
 | |
|     for (uint16_t i = a; i < SEGLEN; i++)
 | |
|       setPixelColor(i, color1);
 | |
|   }
 | |
| 
 | |
|   //fill between points a and b with color2
 | |
|   if (a < b)
 | |
|   {
 | |
|     for (uint16_t i = a; i < b; i++)
 | |
|       setPixelColor(i, color2);
 | |
|   } else {
 | |
|     for (uint16_t i = a; i < SEGLEN; i++) //fill until end
 | |
|       setPixelColor(i, color2);
 | |
|     for (uint16_t i = 0; i < b; i++) //fill from start until b
 | |
|       setPixelColor(i, color2);
 | |
|   }
 | |
| 
 | |
|   //fill between points b and c with color2
 | |
|   if (b < c)
 | |
|   {
 | |
|     for (uint16_t i = b; i < c; i++)
 | |
|       setPixelColor(i, color3);
 | |
|   } else {
 | |
|     for (uint16_t i = b; i < SEGLEN; i++) //fill until end
 | |
|       setPixelColor(i, color3);
 | |
|     for (uint16_t i = 0; i < c; i++) //fill from start until c
 | |
|       setPixelColor(i, color3);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Bicolor chase, more primary color.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_chase_color(void) {
 | |
|   return chase(SEGCOLOR(1), (SEGCOLOR(2)) ? SEGCOLOR(2) : SEGCOLOR(0), SEGCOLOR(0), true);
 | |
| }
 | |
| static const char *_data_FX_MODE_CHASE_COLOR PROGMEM = "Chase@!,Width;!,!,!;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Primary running followed by random color.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_chase_random(void) {
 | |
|   return chase(SEGCOLOR(1), (SEGCOLOR(2)) ? SEGCOLOR(2) : SEGCOLOR(0), SEGCOLOR(0), false);
 | |
| }
 | |
| static const char *_data_FX_MODE_CHASE_RANDOM PROGMEM = "Chase Random@!,Width;!,,!;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Primary, secondary running on rainbow.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_chase_rainbow(void) {
 | |
|   uint8_t color_sep = 256 / SEGLEN;
 | |
|   if (color_sep == 0) color_sep = 1;                                           // correction for segments longer than 256 LEDs
 | |
|   uint8_t color_index = SEGENV.call & 0xFF;
 | |
|   uint32_t color = color_wheel(((SEGENV.step * color_sep) + color_index) & 0xFF);
 | |
| 
 | |
|   return chase(color, SEGCOLOR(0), SEGCOLOR(1), false);
 | |
| }
 | |
| static const char *_data_FX_MODE_CHASE_RAINBOW PROGMEM = "Chase Rainbow@!,Width;!,!,;0";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Primary running on rainbow.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_chase_rainbow_white(void) {
 | |
|   uint16_t n = SEGENV.step;
 | |
|   uint16_t m = (SEGENV.step + 1) % SEGLEN;
 | |
|   uint32_t color2 = color_wheel(((n * 256 / SEGLEN) + (SEGENV.call & 0xFF)) & 0xFF);
 | |
|   uint32_t color3 = color_wheel(((m * 256 / SEGLEN) + (SEGENV.call & 0xFF)) & 0xFF);
 | |
| 
 | |
|   return chase(SEGCOLOR(0), color2, color3, false);
 | |
| }
 | |
| static const char *_data_FX_MODE_CHASE_RAINBOW_WHITE PROGMEM = "Rainbow Runner@!,Size;Bg,,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Red - Amber - Green - Blue lights running
 | |
|  */
 | |
| uint16_t WS2812FX::mode_colorful(void) {
 | |
|   uint8_t numColors = 4; //3, 4, or 5
 | |
|   uint32_t cols[9]{0x00FF0000,0x00EEBB00,0x0000EE00,0x000077CC};
 | |
|   if (SEGMENT.intensity > 160 || SEGMENT.palette) { //palette or color
 | |
|     if (!SEGMENT.palette) {
 | |
|       numColors = 3;
 | |
|       for (uint8_t i = 0; i < 3; i++) cols[i] = SEGCOLOR(i);
 | |
|     } else {
 | |
|       uint16_t fac = 80;
 | |
|       if (SEGMENT.palette == 52) {numColors = 5; fac = 61;} //C9 2 has 5 colors
 | |
|       for (uint8_t i = 0; i < numColors; i++) {
 | |
|         cols[i] = color_from_palette(i*fac, false, true, 255);
 | |
|       }
 | |
|     }
 | |
|   } else if (SEGMENT.intensity < 80) //pastel (easter) colors
 | |
|   {
 | |
|     cols[0] = 0x00FF8040;
 | |
|     cols[1] = 0x00E5D241;
 | |
|     cols[2] = 0x0077FF77;
 | |
|     cols[3] = 0x0077F0F0;
 | |
|   }
 | |
|   for (uint8_t i = numColors; i < numColors*2 -1; i++) cols[i] = cols[i-numColors];
 | |
|   
 | |
|   uint32_t cycleTime = 50 + (8 * (uint32_t)(255 - SEGMENT.speed));
 | |
|   uint32_t it = now / cycleTime;
 | |
|   if (it != SEGENV.step)
 | |
|   {
 | |
|     if (SEGMENT.speed > 0) SEGENV.aux0++;
 | |
|     if (SEGENV.aux0 >= numColors) SEGENV.aux0 = 0;
 | |
|     SEGENV.step = it;
 | |
|   }
 | |
|   
 | |
|   for (uint16_t i = 0; i < SEGLEN; i+= numColors)
 | |
|   {
 | |
|     for (uint16_t j = 0; j < numColors; j++) setPixelColor(i + j, cols[SEGENV.aux0 + j]);
 | |
|   }
 | |
|   
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_COLORFUL PROGMEM = "Colorful@!,Saturation;1,2,3;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Emulates a traffic light.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_traffic_light(void) {
 | |
|   for(uint16_t i=0; i < SEGLEN; i++)
 | |
|     setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 1));
 | |
|   uint32_t mdelay = 500;
 | |
|   for (int i = 0; i < SEGLEN-2 ; i+=3)
 | |
|   {
 | |
|     switch (SEGENV.aux0)
 | |
|     {
 | |
|       case 0: setPixelColor(i, 0x00FF0000); mdelay = 150 + (100 * (uint32_t)(255 - SEGMENT.speed));break;
 | |
|       case 1: setPixelColor(i, 0x00FF0000); mdelay = 150 + (20 * (uint32_t)(255 - SEGMENT.speed)); setPixelColor(i+1, 0x00EECC00); break;
 | |
|       case 2: setPixelColor(i+2, 0x0000FF00); mdelay = 150 + (100 * (uint32_t)(255 - SEGMENT.speed));break;
 | |
|       case 3: setPixelColor(i+1, 0x00EECC00); mdelay = 150 + (20 * (uint32_t)(255 - SEGMENT.speed));break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (now - SEGENV.step > mdelay)
 | |
|   {
 | |
|     SEGENV.aux0++;
 | |
|     if (SEGENV.aux0 == 1 && SEGMENT.intensity > 140) SEGENV.aux0 = 2; //skip Red + Amber, to get US-style sequence
 | |
|     if (SEGENV.aux0 > 3) SEGENV.aux0 = 0;
 | |
|     SEGENV.step = now;
 | |
|   }
 | |
|   
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_TRAFFIC_LIGHT PROGMEM = "Traffic Light@!,;,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Sec flashes running on prim.
 | |
|  */
 | |
| #define FLASH_COUNT 4
 | |
| uint16_t WS2812FX::mode_chase_flash(void) {
 | |
|   uint8_t flash_step = SEGENV.call % ((FLASH_COUNT * 2) + 1);
 | |
| 
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   uint16_t delay = 10 + ((30 * (uint16_t)(255 - SEGMENT.speed)) / SEGLEN);
 | |
|   if(flash_step < (FLASH_COUNT * 2)) {
 | |
|     if(flash_step % 2 == 0) {
 | |
|       uint16_t n = SEGENV.step;
 | |
|       uint16_t m = (SEGENV.step + 1) % SEGLEN;
 | |
|       setPixelColor( n, SEGCOLOR(1));
 | |
|       setPixelColor( m, SEGCOLOR(1));
 | |
|       delay = 20;
 | |
|     } else {
 | |
|       delay = 30;
 | |
|     }
 | |
|   } else {
 | |
|     SEGENV.step = (SEGENV.step + 1) % SEGLEN;
 | |
|   }
 | |
|   return delay;
 | |
| }
 | |
| static const char *_data_FX_MODE_CHASE_FLASH PROGMEM = "Chase Flash@!,;Bg,Fx,!;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Prim flashes running, followed by random color.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_chase_flash_random(void) {
 | |
|   uint8_t flash_step = SEGENV.call % ((FLASH_COUNT * 2) + 1);
 | |
| 
 | |
|   for(uint16_t i = 0; i < SEGENV.step; i++) {
 | |
|     setPixelColor(i, color_wheel(SEGENV.aux0));
 | |
|   }
 | |
| 
 | |
|   uint16_t delay = 1 + ((10 * (uint16_t)(255 - SEGMENT.speed)) / SEGLEN);
 | |
|   if(flash_step < (FLASH_COUNT * 2)) {
 | |
|     uint16_t n = SEGENV.step;
 | |
|     uint16_t m = (SEGENV.step + 1) % SEGLEN;
 | |
|     if(flash_step % 2 == 0) {
 | |
|       setPixelColor( n, SEGCOLOR(0));
 | |
|       setPixelColor( m, SEGCOLOR(0));
 | |
|       delay = 20;
 | |
|     } else {
 | |
|       setPixelColor( n, color_wheel(SEGENV.aux0));
 | |
|       setPixelColor( m, SEGCOLOR(1));
 | |
|       delay = 30;
 | |
|     }
 | |
|   } else {
 | |
|     SEGENV.step = (SEGENV.step + 1) % SEGLEN;
 | |
| 
 | |
|     if (SEGENV.step == 0) {
 | |
|       SEGENV.aux0 = get_random_wheel_index(SEGENV.aux0);
 | |
|     }
 | |
|   }
 | |
|   return delay;
 | |
| }
 | |
| static const char *_data_FX_MODE_CHASE_FLASH_RANDOM PROGMEM = "Chase Flash Rnd@!,;,Fx,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Alternating pixels running function.
 | |
|  */
 | |
| uint16_t WS2812FX::running(uint32_t color1, uint32_t color2, bool theatre) {
 | |
|   uint8_t width = (theatre ? 3 : 1) + (SEGMENT.intensity >> 4);  // window
 | |
|   uint32_t cycleTime = 50 + (255 - SEGMENT.speed);
 | |
|   uint32_t it = now / cycleTime;
 | |
|   bool usePalette = color1 == SEGCOLOR(0);
 | |
|   
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     uint32_t col = color2;
 | |
|     if (usePalette) color1 = color_from_palette(i, true, PALETTE_SOLID_WRAP, 0);
 | |
|     if (theatre) {
 | |
|       if ((i % width) == SEGENV.aux0) col = color1;
 | |
|     } else {
 | |
|       int8_t pos = (i % (width<<1));
 | |
|       if ((pos < SEGENV.aux0-width) || ((pos >= SEGENV.aux0) && (pos < SEGENV.aux0+width))) col = color1;
 | |
|     }
 | |
|     setPixelColor(i,col);
 | |
|   }
 | |
| 
 | |
|   if (it != SEGENV.step) {
 | |
|     SEGENV.aux0 = (SEGENV.aux0 +1) % (theatre ? width : (width<<1));
 | |
|     SEGENV.step = it;
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Alternating color/sec pixels running.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_running_color(void) {
 | |
|   return running(SEGCOLOR(0), SEGCOLOR(1));
 | |
| }
 | |
| static const char *_data_FX_MODE_RUNNING_COLOR PROGMEM = "Chase 2@!,Width;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Alternating red/white pixels running.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_candy_cane(void) {
 | |
|   return running(RED, WHITE);
 | |
| }
 | |
| static const char *_data_FX_MODE_CANDY_CANE PROGMEM = "Candy Cane@!,Width;;";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Alternating orange/purple pixels running.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_halloween(void) {
 | |
|   return running(PURPLE, ORANGE);
 | |
| }
 | |
| static const char *_data_FX_MODE_HALLOWEEN PROGMEM = "Halloween@!,Width;;";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Random colored pixels running. ("Stream")
 | |
|  */
 | |
| uint16_t WS2812FX::mode_running_random(void) {
 | |
|   uint32_t cycleTime = 25 + (3 * (uint32_t)(255 - SEGMENT.speed));
 | |
|   uint32_t it = now / cycleTime;
 | |
|   if (SEGENV.call == 0) SEGENV.aux0 = random16(); // random seed for PRNG on start
 | |
| 
 | |
|   uint8_t zoneSize = ((255-SEGMENT.intensity) >> 4) +1;
 | |
|   uint16_t PRNG16 = SEGENV.aux0;
 | |
| 
 | |
|   uint8_t z = it % zoneSize;
 | |
|   bool nzone = (!z && it != SEGENV.aux1);
 | |
|   for (uint16_t i=SEGLEN-1; i > 0; i--) {
 | |
|     if (nzone || z >= zoneSize) {
 | |
|       uint8_t lastrand = PRNG16 >> 8;
 | |
|       int16_t diff = 0;
 | |
|       while (abs(diff) < 42) { // make sure the difference between adjacent colors is big enough
 | |
|         PRNG16 = (uint16_t)(PRNG16 * 2053) + 13849; // next zone, next 'random' number
 | |
|         diff = (PRNG16 >> 8) - lastrand;
 | |
|       }
 | |
|       if (nzone) {
 | |
|         SEGENV.aux0 = PRNG16; // save next starting seed
 | |
|         nzone = false;
 | |
|       }
 | |
|       z = 0;
 | |
|     }
 | |
|     setPixelColor(i, color_wheel(PRNG16 >> 8));
 | |
|     z++;
 | |
|   }
 | |
| 
 | |
|   SEGENV.aux1 = it;
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_RUNNING_RANDOM PROGMEM = "Stream";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * K.I.T.T.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_larson_scanner(void){
 | |
|   return larson_scanner(false);
 | |
| }
 | |
| static const char *_data_FX_MODE_LARSON_SCANNER PROGMEM = "Scanner";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::larson_scanner(bool dual) {
 | |
|   uint16_t counter = now * ((SEGMENT.speed >> 2) +8);
 | |
|   uint16_t index = counter * SEGLEN  >> 16;
 | |
| 
 | |
|   fade_out(SEGMENT.intensity);
 | |
| 
 | |
|   if (SEGENV.step > index && SEGENV.step - index > SEGLEN/2) {
 | |
|     SEGENV.aux0 = !SEGENV.aux0;
 | |
|   }
 | |
|   
 | |
|   for (uint16_t i = SEGENV.step; i < index; i++) {
 | |
|     uint16_t j = (SEGENV.aux0)?i:SEGLEN-1-i;
 | |
|     setPixelColor( j, color_from_palette(j, true, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
|   if (dual) {
 | |
|     uint32_t c;
 | |
|     if (SEGCOLOR(2) != 0) {
 | |
|       c = SEGCOLOR(2);
 | |
|     } else {
 | |
|       c = color_from_palette(index, true, PALETTE_SOLID_WRAP, 0);
 | |
|     }
 | |
| 
 | |
|     for (uint16_t i = SEGENV.step; i < index; i++) {
 | |
|       uint16_t j = (SEGENV.aux0)?SEGLEN-1-i:i;
 | |
|       setPixelColor(j, c);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   SEGENV.step = index;
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Firing comets from one end. "Lighthouse"
 | |
|  */
 | |
| uint16_t WS2812FX::mode_comet(void) {
 | |
|   uint16_t counter = now * ((SEGMENT.speed >>2) +1);
 | |
|   uint16_t index = (counter * SEGLEN) >> 16;
 | |
|   if (SEGENV.call == 0) SEGENV.aux0 = index;
 | |
| 
 | |
|   fade_out(SEGMENT.intensity);
 | |
| 
 | |
|   setPixelColor( index, color_from_palette(index, true, PALETTE_SOLID_WRAP, 0));
 | |
|   if (index > SEGENV.aux0) {
 | |
|     for (uint16_t i = SEGENV.aux0; i < index ; i++) {
 | |
|        setPixelColor( i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 0));
 | |
|     }
 | |
|   } else if (index < SEGENV.aux0 && index < 10) {
 | |
|     for (uint16_t i = 0; i < index ; i++) {
 | |
|        setPixelColor( i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 0));
 | |
|     }      
 | |
|   }
 | |
|   SEGENV.aux0 = index++;
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_COMET PROGMEM = "Lighthouse@!,Fade rate;!,!,!;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Fireworks function.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_fireworks() {
 | |
|   const uint16_t width  = isMatrix ? SEGMENT.virtualWidth() : SEGMENT.virtualLength();
 | |
|   const uint16_t height = SEGMENT.virtualHeight();
 | |
| 
 | |
|   fade_out(0);
 | |
| 
 | |
|   if (SEGENV.call == 0) {
 | |
|     SEGENV.aux0 = UINT16_MAX;
 | |
|     SEGENV.aux1 = UINT16_MAX;
 | |
|   }
 | |
|   bool valid1 = (SEGENV.aux0 < width*height);
 | |
|   bool valid2 = (SEGENV.aux1 < width*height);
 | |
|   uint32_t sv1 = 0, sv2 = 0;
 | |
|   if (valid1) sv1 = isMatrix ? getPixelColorXY(SEGENV.aux0%width, SEGENV.aux0/width) : getPixelColor(SEGENV.aux0); // get spark color
 | |
|   if (valid2) sv2 = isMatrix ? getPixelColorXY(SEGENV.aux1%width, SEGENV.aux1/width) : getPixelColor(SEGENV.aux1);
 | |
|   if (!SEGENV.step) blur(16);
 | |
|   if (valid1) { if (isMatrix) setPixelColorXY(SEGENV.aux0%width, SEGENV.aux0/width, sv1); else setPixelColor(SEGENV.aux0, sv1); } // restore spark color after blur
 | |
|   if (valid2) { if (isMatrix) setPixelColorXY(SEGENV.aux1%width, SEGENV.aux1/width, sv2); else setPixelColor(SEGENV.aux1, sv2); } // restore old spark color after blur
 | |
| 
 | |
|   for (uint16_t i=0; i<MAX(1, width/20); i++) {
 | |
|     if (random8(129 - (SEGMENT.intensity >> 1)) == 0) {
 | |
|       uint16_t index = random16(width*height);
 | |
|       uint16_t j = index % width, k = index / width;
 | |
|       uint32_t col = color_from_palette(random8(), false, false, 0);
 | |
|       if (isMatrix) setPixelColorXY(j, k, col);
 | |
|       else          setPixelColor(index, col);
 | |
|       SEGENV.aux1 = SEGENV.aux0;  // old spark
 | |
|       SEGENV.aux0 = index;        // remember where spark occured
 | |
|     }
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_FIREWORKS PROGMEM = "Fireworks@,Frequency=192;!,!,;!=11";
 | |
| 
 | |
| 
 | |
| //Twinkling LEDs running. Inspired by https://github.com/kitesurfer1404/WS2812FX/blob/master/src/custom/Rain.h
 | |
| uint16_t WS2812FX::mode_rain()
 | |
| {
 | |
|   const uint16_t width  = SEGMENT.virtualWidth();
 | |
|   const uint16_t height = SEGMENT.virtualHeight();
 | |
|   SEGENV.step += FRAMETIME;
 | |
|   if (SEGENV.step > SPEED_FORMULA_L) {
 | |
|     SEGENV.step = 1;
 | |
|     if (isMatrix) {
 | |
|       move(6,1);  // move all pixels down
 | |
|       SEGENV.aux0 = (SEGENV.aux0 % width) + (SEGENV.aux0 / width + 1) * width;
 | |
|       SEGENV.aux1 = (SEGENV.aux1 % width) + (SEGENV.aux1 / width + 1) * width;
 | |
|     } else {
 | |
|       //shift all leds left
 | |
|       uint32_t ctemp = getPixelColor(0);
 | |
|       for(uint16_t i = 0; i < SEGLEN - 1; i++) {
 | |
|         setPixelColor(i, getPixelColor(i+1));
 | |
|       }
 | |
|       setPixelColor(SEGLEN -1, ctemp); // wrap around
 | |
|       SEGENV.aux0++;  // increase spark index
 | |
|       SEGENV.aux1++;
 | |
|     }
 | |
|     if (SEGENV.aux0 == 0) SEGENV.aux0 = UINT16_MAX; // reset previous spark positiom
 | |
|     if (SEGENV.aux1 == 0) SEGENV.aux0 = UINT16_MAX; // reset previous spark positiom
 | |
|     if (SEGENV.aux0 >= width*height) SEGENV.aux0 = 0;     // ignore
 | |
|     if (SEGENV.aux1 >= width*height) SEGENV.aux1 = 0;
 | |
|   }
 | |
|   return mode_fireworks();
 | |
| }
 | |
| static const char *_data_FX_MODE_RAIN PROGMEM = "Rain@!,Spawning rate=128;!,!,;";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Fire flicker function
 | |
|  */
 | |
| uint16_t WS2812FX::mode_fire_flicker(void) {
 | |
|   uint32_t cycleTime = 40 + (255 - SEGMENT.speed);
 | |
|   uint32_t it = now / cycleTime;
 | |
|   if (SEGENV.step == it) return FRAMETIME;
 | |
|   
 | |
|   byte w = (SEGCOLOR(0) >> 24);
 | |
|   byte r = (SEGCOLOR(0) >> 16);
 | |
|   byte g = (SEGCOLOR(0) >>  8);
 | |
|   byte b = (SEGCOLOR(0)      );
 | |
|   byte lum = (SEGMENT.palette == 0) ? MAX(w, MAX(r, MAX(g, b))) : 255;
 | |
|   lum /= (((256-SEGMENT.intensity)/16)+1);
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     byte flicker = random8(lum);
 | |
|     if (SEGMENT.palette == 0) {
 | |
|       setPixelColor(i, MAX(r - flicker, 0), MAX(g - flicker, 0), MAX(b - flicker, 0), MAX(w - flicker, 0));
 | |
|     } else {
 | |
|       setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 0, 255 - flicker));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   SEGENV.step = it;
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_FIRE_FLICKER PROGMEM = "Fire Flicker@!,!;!,,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Gradient run base function
 | |
|  */
 | |
| uint16_t WS2812FX::gradient_base(bool loading) {
 | |
|   uint16_t counter = now * ((SEGMENT.speed >> 2) + 1);
 | |
|   uint16_t pp = counter * SEGLEN >> 16;
 | |
|   if (SEGENV.call == 0) pp = 0;
 | |
|   float val; //0.0 = sec 1.0 = pri
 | |
|   float brd = loading ? SEGMENT.intensity : SEGMENT.intensity/2;
 | |
|   if (brd <1.0) brd = 1.0;
 | |
|   int p1 = pp-SEGLEN;
 | |
|   int p2 = pp+SEGLEN;
 | |
| 
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++)
 | |
|   {
 | |
|     if (loading)
 | |
|     {
 | |
|       val = abs(((i>pp) ? p2:pp) -i);
 | |
|     } else {
 | |
|       val = MIN(abs(pp-i),MIN(abs(p1-i),abs(p2-i)));
 | |
|     }
 | |
|     val = (brd > val) ? val/brd * 255 : 255;
 | |
|     setPixelColor(i, color_blend(SEGCOLOR(0), color_from_palette(i, true, PALETTE_SOLID_WRAP, 1), val));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Gradient run
 | |
|  */
 | |
| uint16_t WS2812FX::mode_gradient(void) {
 | |
|   return gradient_base(false);
 | |
| }
 | |
| static const char *_data_FX_MODE_GRADIENT PROGMEM = "Gradient@!,Spread=16;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Gradient run with hard transition
 | |
|  */
 | |
| uint16_t WS2812FX::mode_loading(void) {
 | |
|   return gradient_base(true);
 | |
| }
 | |
| static const char *_data_FX_MODE_LOADING PROGMEM = "Loading@!,Fade=16;!,!,;!";
 | |
| 
 | |
| 
 | |
| //American Police Light with all LEDs Red and Blue 
 | |
| uint16_t WS2812FX::police_base(uint32_t color1, uint32_t color2)
 | |
| {
 | |
|   uint16_t delay = 1 + (FRAMETIME<<3) / SEGLEN;  // longer segments should change faster
 | |
|   uint32_t it = now / map(SEGMENT.speed, 0, 255, delay<<4, delay);
 | |
|   uint16_t offset = it % SEGLEN;
 | |
|   
 | |
| 	uint16_t width = ((SEGLEN*(SEGMENT.intensity+1))>>9); //max width is half the strip
 | |
|   if (!width) width = 1;
 | |
|   for (uint16_t i = 0; i < width; i++) {
 | |
|     uint16_t indexR = (offset + i) % SEGLEN;
 | |
|     uint16_t indexB = (offset + i + (SEGLEN>>1)) % SEGLEN;
 | |
|     setPixelColor(indexR, color1);
 | |
|     setPixelColor(indexB, color2);
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| //Police Lights Red and Blue 
 | |
| uint16_t WS2812FX::mode_police()
 | |
| {
 | |
|   fill(SEGCOLOR(1));
 | |
|   return police_base(RED, BLUE);
 | |
| }
 | |
| static const char *_data_FX_MODE_POLICE PROGMEM = "Police@!,Width;,Bg,;0";
 | |
| 
 | |
| 
 | |
| //Police Lights with custom colors 
 | |
| uint16_t WS2812FX::mode_two_dots()
 | |
| {
 | |
|   fill(SEGCOLOR(2));
 | |
|   uint32_t color2 = (SEGCOLOR(1) == SEGCOLOR(2)) ? SEGCOLOR(0) : SEGCOLOR(1);
 | |
| 
 | |
|   return police_base(SEGCOLOR(0), color2);
 | |
| }
 | |
| static const char *_data_FX_MODE_TWO_DOTS PROGMEM = "Two Dots@!,Dot size;1,2,Bg;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Fairy, inspired by https://www.youtube.com/watch?v=zeOw5MZWq24
 | |
|  */
 | |
| //4 bytes
 | |
| typedef struct Flasher {
 | |
|   uint16_t stateStart;
 | |
|   uint8_t stateDur;
 | |
| 	bool stateOn;
 | |
| } flasher;
 | |
| 
 | |
| #define FLASHERS_PER_ZONE 6
 | |
| #define MAX_SHIMMER 92
 | |
| 
 | |
| uint16_t WS2812FX::mode_fairy() {
 | |
| 	//set every pixel to a 'random' color from palette (using seed so it doesn't change between frames)
 | |
| 	uint16_t PRNG16 = 5100 + _segment_index;
 | |
| 	for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
| 		PRNG16 = (uint16_t)(PRNG16 * 2053) + 1384; //next 'random' number
 | |
| 		setPixelColor(i, color_from_palette(PRNG16 >> 8, false, false, 0));
 | |
| 	}
 | |
| 
 | |
| 	//amount of flasher pixels depending on intensity (0: none, 255: every LED)
 | |
| 	if (SEGMENT.intensity == 0) return FRAMETIME;
 | |
| 	uint8_t flasherDistance = ((255 - SEGMENT.intensity) / 28) +1; //1-10
 | |
| 	uint16_t numFlashers = (SEGLEN / flasherDistance) +1;
 | |
| 	
 | |
| 	uint16_t dataSize = sizeof(flasher) * numFlashers;
 | |
|   if (!SEGENV.allocateData(dataSize)) return FRAMETIME; //allocation failed
 | |
| 	Flasher* flashers = reinterpret_cast<Flasher*>(SEGENV.data);
 | |
| 	uint16_t now16 = now & 0xFFFF;
 | |
| 
 | |
| 	//Up to 11 flashers in one brightness zone, afterwards a new zone for every 6 flashers
 | |
| 	uint16_t zones = numFlashers/FLASHERS_PER_ZONE;
 | |
| 	if (!zones) zones = 1;
 | |
| 	uint8_t flashersInZone = numFlashers/zones;
 | |
| 	uint8_t flasherBri[FLASHERS_PER_ZONE*2 -1];
 | |
| 
 | |
| 	for (uint16_t z = 0; z < zones; z++) {
 | |
| 		uint16_t flasherBriSum = 0;
 | |
| 		uint16_t firstFlasher = z*flashersInZone;
 | |
| 		if (z == zones-1) flashersInZone = numFlashers-(flashersInZone*(zones-1));
 | |
| 
 | |
| 		for (uint16_t f = firstFlasher; f < firstFlasher + flashersInZone; f++) {
 | |
| 			uint16_t stateTime = now16 - flashers[f].stateStart;
 | |
| 			//random on/off time reached, switch state
 | |
| 			if (stateTime > flashers[f].stateDur * 10) {
 | |
| 				flashers[f].stateOn = !flashers[f].stateOn;
 | |
| 				if (flashers[f].stateOn) {
 | |
| 					flashers[f].stateDur = 12 + random8(12 + ((255 - SEGMENT.speed) >> 2)); //*10, 250ms to 1250ms
 | |
| 				} else {
 | |
| 					flashers[f].stateDur = 20 + random8(6 + ((255 - SEGMENT.speed) >> 2)); //*10, 250ms to 1250ms
 | |
| 				}
 | |
| 				//flashers[f].stateDur = 51 + random8(2 + ((255 - SEGMENT.speed) >> 1));
 | |
| 				flashers[f].stateStart = now16;
 | |
| 				if (stateTime < 255) {
 | |
| 					flashers[f].stateStart -= 255 -stateTime; //start early to get correct bri
 | |
| 					flashers[f].stateDur += 26 - stateTime/10;
 | |
| 					stateTime = 255 - stateTime;
 | |
| 				} else {
 | |
| 					stateTime = 0;
 | |
| 				}
 | |
| 			}
 | |
| 			if (stateTime > 255) stateTime = 255; //for flasher brightness calculation, fades in first 255 ms of state
 | |
| 			//flasherBri[f - firstFlasher] = (flashers[f].stateOn) ? 255-gamma8((510 - stateTime) >> 1) : gamma8((510 - stateTime) >> 1);
 | |
| 			flasherBri[f - firstFlasher] = (flashers[f].stateOn) ? stateTime : 255 - (stateTime >> 0);
 | |
| 			flasherBriSum += flasherBri[f - firstFlasher];
 | |
| 		}
 | |
| 		//dim factor, to create "shimmer" as other pixels get less voltage if a lot of flashers are on
 | |
| 		uint8_t avgFlasherBri = flasherBriSum / flashersInZone;
 | |
| 		uint8_t globalPeakBri = 255 - ((avgFlasherBri * MAX_SHIMMER) >> 8); //183-255, suitable for 1/5th of LEDs flashers
 | |
| 
 | |
| 		for (uint16_t f = firstFlasher; f < firstFlasher + flashersInZone; f++) {
 | |
| 			uint8_t bri = (flasherBri[f - firstFlasher] * globalPeakBri) / 255;
 | |
| 			PRNG16 = (uint16_t)(PRNG16 * 2053) + 1384; //next 'random' number
 | |
| 			uint16_t flasherPos = f*flasherDistance;
 | |
| 			setPixelColor(flasherPos, color_blend(SEGCOLOR(1), color_from_palette(PRNG16 >> 8, false, false, 0), bri));
 | |
| 			for (uint16_t i = flasherPos+1; i < flasherPos+flasherDistance && i < SEGLEN; i++) {
 | |
| 				PRNG16 = (uint16_t)(PRNG16 * 2053) + 1384; //next 'random' number
 | |
| 				setPixelColor(i, color_from_palette(PRNG16 >> 8, false, false, 0, globalPeakBri));
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_FAIRY PROGMEM = "Fairy";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Fairytwinkle. Like Colortwinkle, but starting from all lit and not relying on getPixelColor
 | |
|  * Warning: Uses 4 bytes of segment data per pixel
 | |
|  */
 | |
| uint16_t WS2812FX::mode_fairytwinkle() {
 | |
| 	uint16_t dataSize = sizeof(flasher) * SEGLEN;
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
| 	Flasher* flashers = reinterpret_cast<Flasher*>(SEGENV.data);
 | |
| 	uint16_t now16 = now & 0xFFFF;
 | |
| 	uint16_t PRNG16 = 5100 + _segment_index;
 | |
| 
 | |
| 	uint16_t riseFallTime = 400 + (255-SEGMENT.speed)*3;
 | |
| 	uint16_t maxDur = riseFallTime/100 + ((255 - SEGMENT.intensity) >> 2) + 13 + ((255 - SEGMENT.intensity) >> 1);
 | |
| 
 | |
| 	for (uint16_t f = 0; f < SEGLEN; f++) {
 | |
| 		uint16_t stateTime = now16 - flashers[f].stateStart;
 | |
| 		//random on/off time reached, switch state
 | |
| 		if (stateTime > flashers[f].stateDur * 100) {
 | |
| 			flashers[f].stateOn = !flashers[f].stateOn;
 | |
| 			bool init = !flashers[f].stateDur;
 | |
| 			if (flashers[f].stateOn) {
 | |
| 				flashers[f].stateDur = riseFallTime/100 + ((255 - SEGMENT.intensity) >> 2) + random8(12 + ((255 - SEGMENT.intensity) >> 1)) +1;
 | |
| 			} else {
 | |
| 				flashers[f].stateDur = riseFallTime/100 + random8(3 + ((255 - SEGMENT.speed) >> 6)) +1;
 | |
| 			}
 | |
| 			flashers[f].stateStart = now16;
 | |
| 			stateTime = 0;
 | |
| 			if (init) {
 | |
| 				flashers[f].stateStart -= riseFallTime; //start lit
 | |
| 				flashers[f].stateDur = riseFallTime/100 + random8(12 + ((255 - SEGMENT.intensity) >> 1)) +5; //fire up a little quicker
 | |
| 				stateTime = riseFallTime;
 | |
| 			}
 | |
| 		}
 | |
| 		if (flashers[f].stateOn && flashers[f].stateDur > maxDur) flashers[f].stateDur = maxDur; //react more quickly on intensity change
 | |
| 		if (stateTime > riseFallTime) stateTime = riseFallTime; //for flasher brightness calculation, fades in first 255 ms of state
 | |
| 		uint8_t fadeprog = 255 - ((stateTime * 255) / riseFallTime);
 | |
| 		uint8_t flasherBri = (flashers[f].stateOn) ? 255-gamma8(fadeprog) : gamma8(fadeprog);
 | |
| 		uint16_t lastR = PRNG16;
 | |
| 		uint16_t diff = 0;
 | |
| 		while (diff < 0x4000) { //make sure colors of two adjacent LEDs differ enough
 | |
| 			PRNG16 = (uint16_t)(PRNG16 * 2053) + 1384; //next 'random' number
 | |
| 			diff = (PRNG16 > lastR) ? PRNG16 - lastR : lastR - PRNG16;
 | |
| 		}
 | |
| 		setPixelColor(f, color_blend(SEGCOLOR(1), color_from_palette(PRNG16 >> 8, false, false, 0), flasherBri));
 | |
| 	}
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_FAIRYTWINKLE PROGMEM = "Fairy Twinkle";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Tricolor chase function
 | |
|  */
 | |
| uint16_t WS2812FX::tricolor_chase(uint32_t color1, uint32_t color2) {
 | |
|   uint32_t cycleTime = 50 + ((255 - SEGMENT.speed)<<1);
 | |
|   uint32_t it = now / cycleTime;  // iterator
 | |
|   uint8_t width = (1 + (SEGMENT.intensity>>4)); // value of 1-16 for each colour
 | |
|   uint8_t index = it % (width*3);
 | |
|   
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++, index++) {
 | |
|     if (index > (width*3)-1) index = 0;
 | |
| 
 | |
|     uint32_t color = color1;
 | |
|     if (index > (width<<1)-1) color = color_from_palette(i, true, PALETTE_SOLID_WRAP, 1);
 | |
|     else if (index > width-1) color = color2;
 | |
| 
 | |
|     setPixelColor(SEGLEN - i -1, color);
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Tricolor chase mode
 | |
|  */
 | |
| uint16_t WS2812FX::mode_tricolor_chase(void) {
 | |
|   return tricolor_chase(SEGCOLOR(2), SEGCOLOR(0));
 | |
| }
 | |
| static const char *_data_FX_MODE_TRICOLOR_CHASE PROGMEM = "Chase 3@!,Size;1,2,3;0";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * ICU mode
 | |
|  */
 | |
| uint16_t WS2812FX::mode_icu(void) {
 | |
|   uint16_t dest = SEGENV.step & 0xFFFF;
 | |
|   uint8_t space = (SEGMENT.intensity >> 3) +2;
 | |
| 
 | |
|   fill(SEGCOLOR(1));
 | |
| 
 | |
|   byte pindex = map(dest, 0, SEGLEN-SEGLEN/space, 0, 255);
 | |
|   uint32_t col = color_from_palette(pindex, false, false, 0);
 | |
| 
 | |
|   setPixelColor(dest, col);
 | |
|   setPixelColor(dest + SEGLEN/space, col);
 | |
| 
 | |
|   if(SEGENV.aux0 == dest) { // pause between eye movements
 | |
|     if(random8(6) == 0) { // blink once in a while
 | |
|       setPixelColor(dest, SEGCOLOR(1));
 | |
|       setPixelColor(dest + SEGLEN/space, SEGCOLOR(1));
 | |
|       return 200;
 | |
|     }
 | |
|     SEGENV.aux0 = random16(SEGLEN-SEGLEN/space);
 | |
|     return 1000 + random16(2000);
 | |
|   }
 | |
| 
 | |
|   if(SEGENV.aux0 > SEGENV.step) {
 | |
|     SEGENV.step++;
 | |
|     dest++;
 | |
|   } else if (SEGENV.aux0 < SEGENV.step) {
 | |
|     SEGENV.step--;
 | |
|     dest--;
 | |
|   }
 | |
| 
 | |
|   setPixelColor(dest, col);
 | |
|   setPixelColor(dest + SEGLEN/space, col);
 | |
| 
 | |
|   return SPEED_FORMULA_L;
 | |
| }
 | |
| static const char *_data_FX_MODE_ICU PROGMEM = "ICU";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Custom mode by Aircoookie. Color Wipe, but with 3 colors
 | |
|  */
 | |
| uint16_t WS2812FX::mode_tricolor_wipe(void)
 | |
| {
 | |
|   uint32_t cycleTime = 1000 + (255 - SEGMENT.speed)*200;
 | |
|   uint32_t perc = now % cycleTime;
 | |
|   uint16_t prog = (perc * 65535) / cycleTime;
 | |
|   uint16_t ledIndex = (prog * SEGLEN * 3) >> 16;
 | |
|   uint16_t ledOffset = ledIndex;
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++)
 | |
|   {
 | |
|     setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 2));
 | |
|   }
 | |
|   
 | |
|   if(ledIndex < SEGLEN) { //wipe from 0 to 1
 | |
|     for (uint16_t i = 0; i < SEGLEN; i++)
 | |
|     {
 | |
|       setPixelColor(i, (i > ledOffset)? SEGCOLOR(0) : SEGCOLOR(1));
 | |
|     }
 | |
|   } else if (ledIndex < SEGLEN*2) { //wipe from 1 to 2
 | |
|     ledOffset = ledIndex - SEGLEN;
 | |
|     for (uint16_t i = ledOffset +1; i < SEGLEN; i++)
 | |
|     {
 | |
|       setPixelColor(i, SEGCOLOR(1));
 | |
|     }
 | |
|   } else //wipe from 2 to 0
 | |
|   {
 | |
|     ledOffset = ledIndex - SEGLEN*2;
 | |
|     for (uint16_t i = 0; i <= ledOffset; i++)
 | |
|     {
 | |
|       setPixelColor(i, SEGCOLOR(0));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_TRICOLOR_WIPE PROGMEM = "Tri Wipe@!,;1,2,3;0";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Fades between 3 colors
 | |
|  * Custom mode by Keith Lord: https://github.com/kitesurfer1404/WS2812FX/blob/master/src/custom/TriFade.h
 | |
|  * Modified by Aircoookie
 | |
|  */
 | |
| uint16_t WS2812FX::mode_tricolor_fade(void)
 | |
| {
 | |
|   uint16_t counter = now * ((SEGMENT.speed >> 3) +1);
 | |
|   uint32_t prog = (counter * 768) >> 16;
 | |
| 
 | |
|   uint32_t color1 = 0, color2 = 0;
 | |
|   byte stage = 0;
 | |
| 
 | |
|   if(prog < 256) {
 | |
|     color1 = SEGCOLOR(0);
 | |
|     color2 = SEGCOLOR(1);
 | |
|     stage = 0;
 | |
|   } else if(prog < 512) {
 | |
|     color1 = SEGCOLOR(1);
 | |
|     color2 = SEGCOLOR(2);
 | |
|     stage = 1;
 | |
|   } else {
 | |
|     color1 = SEGCOLOR(2);
 | |
|     color2 = SEGCOLOR(0);
 | |
|     stage = 2;
 | |
|   }
 | |
| 
 | |
|   byte stp = prog; // % 256
 | |
|   for(uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     uint32_t color;
 | |
|     if (stage == 2) {
 | |
|       color = color_blend(color_from_palette(i, true, PALETTE_SOLID_WRAP, 2), color2, stp);
 | |
|     } else if (stage == 1) {
 | |
|       color = color_blend(color1, color_from_palette(i, true, PALETTE_SOLID_WRAP, 2), stp);
 | |
|     } else {
 | |
|       color = color_blend(color1, color2, stp);
 | |
|     }
 | |
|     setPixelColor(i, color);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_TRICOLOR_FADE PROGMEM = "Tri Fade";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Creates random comets
 | |
|  * Custom mode by Keith Lord: https://github.com/kitesurfer1404/WS2812FX/blob/master/src/custom/MultiComet.h
 | |
|  */
 | |
| uint16_t WS2812FX::mode_multi_comet(void)
 | |
| {
 | |
|   uint32_t cycleTime = 10 + (uint32_t)(255 - SEGMENT.speed);
 | |
|   uint32_t it = now / cycleTime;
 | |
|   if (SEGENV.step == it) return FRAMETIME;
 | |
|   if (!SEGENV.allocateData(sizeof(uint16_t) * 8)) return mode_static(); //allocation failed
 | |
|   
 | |
|   fade_out(SEGMENT.intensity);
 | |
|   
 | |
|   uint16_t* comets = reinterpret_cast<uint16_t*>(SEGENV.data);
 | |
| 
 | |
|   for(uint8_t i=0; i < 8; i++) {
 | |
|     if(comets[i] < SEGLEN) {
 | |
|       uint16_t index = comets[i];
 | |
|       if (SEGCOLOR(2) != 0)
 | |
|       {
 | |
|         setPixelColor(index, i % 2 ? color_from_palette(index, true, PALETTE_SOLID_WRAP, 0) : SEGCOLOR(2));
 | |
|       } else
 | |
|       {
 | |
|         setPixelColor(index, color_from_palette(index, true, PALETTE_SOLID_WRAP, 0));
 | |
|       }
 | |
|       comets[i]++;
 | |
|     } else {
 | |
|       if(!random(SEGLEN)) {
 | |
|         comets[i] = 0;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   SEGENV.step = it;
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_MULTI_COMET PROGMEM = "Multi Comet";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Creates two Larson scanners moving in opposite directions
 | |
|  * Custom mode by Keith Lord: https://github.com/kitesurfer1404/WS2812FX/blob/master/src/custom/DualLarson.h
 | |
|  */
 | |
| uint16_t WS2812FX::mode_dual_larson_scanner(void){
 | |
|   return larson_scanner(true);
 | |
| }
 | |
| static const char *_data_FX_MODE_DUAL_LARSON_SCANNER PROGMEM = "Scanner Dual";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Running random pixels ("Stream 2")
 | |
|  * Custom mode by Keith Lord: https://github.com/kitesurfer1404/WS2812FX/blob/master/src/custom/RandomChase.h
 | |
|  */
 | |
| uint16_t WS2812FX::mode_random_chase(void)
 | |
| {
 | |
|   if (SEGENV.call == 0) {
 | |
|     SEGENV.step = RGBW32(random8(), random8(), random8(), 0);
 | |
|     SEGENV.aux0 = random16();
 | |
|   }
 | |
|   uint16_t prevSeed = random16_get_seed(); // save seed so we can restore it at the end of the function
 | |
|   uint32_t cycleTime = 25 + (3 * (uint32_t)(255 - SEGMENT.speed));
 | |
|   uint32_t it = now / cycleTime;
 | |
|   uint32_t color = SEGENV.step;
 | |
|   random16_set_seed(SEGENV.aux0);
 | |
| 
 | |
|   for(uint16_t i = SEGLEN -1; i > 0; i--) {
 | |
|     uint8_t r = random8(6) != 0 ? (color >> 16 & 0xFF) : random8();
 | |
|     uint8_t g = random8(6) != 0 ? (color >> 8  & 0xFF) : random8();
 | |
|     uint8_t b = random8(6) != 0 ? (color       & 0xFF) : random8();
 | |
|     color = RGBW32(r, g, b, 0);
 | |
|     setPixelColor(i, r, g, b);
 | |
|     if (i == SEGLEN -1 && SEGENV.aux1 != (it & 0xFFFF)) { //new first color in next frame
 | |
|       SEGENV.step = color;
 | |
|       SEGENV.aux0 = random16_get_seed();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   SEGENV.aux1 = it & 0xFFFF;
 | |
| 
 | |
|   random16_set_seed(prevSeed); // restore original seed so other effects can use "random" PRNG
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_RANDOM_CHASE PROGMEM = "Stream 2";
 | |
| 
 | |
| 
 | |
| //7 bytes
 | |
| typedef struct Oscillator {
 | |
|   int16_t pos;
 | |
|   int8_t  size;
 | |
|   int8_t  dir;
 | |
|   int8_t  speed;
 | |
| } oscillator;
 | |
| 
 | |
| /*
 | |
| /  Oscillating bars of color, updated with standard framerate
 | |
| */
 | |
| uint16_t WS2812FX::mode_oscillate(void)
 | |
| {
 | |
|   uint8_t numOscillators = 3;
 | |
|   uint16_t dataSize = sizeof(oscillator) * numOscillators;
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   
 | |
|   Oscillator* oscillators = reinterpret_cast<Oscillator*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0)
 | |
|   {
 | |
|     oscillators[0] = {(int16_t)(SEGLEN/4),   (int8_t)(SEGLEN/8),  1, 1};
 | |
|     oscillators[1] = {(int16_t)(SEGLEN/4*3), (int8_t)(SEGLEN/8),  1, 2};
 | |
|     oscillators[2] = {(int16_t)(SEGLEN/4*2), (int8_t)(SEGLEN/8), -1, 1};
 | |
|   }
 | |
| 
 | |
|   uint32_t cycleTime = 20 + (2 * (uint32_t)(255 - SEGMENT.speed));
 | |
|   uint32_t it = now / cycleTime;
 | |
| 
 | |
|   for(uint8_t i = 0; i < numOscillators; i++) {
 | |
|     // if the counter has increased, move the oscillator by the random step
 | |
|     if (it != SEGENV.step) oscillators[i].pos += oscillators[i].dir * oscillators[i].speed;
 | |
|     oscillators[i].size = SEGLEN/(3+SEGMENT.intensity/8);
 | |
|     if((oscillators[i].dir == -1) && (oscillators[i].pos <= 0)) {
 | |
|       oscillators[i].pos = 0;
 | |
|       oscillators[i].dir = 1;
 | |
|       // make bigger steps for faster speeds
 | |
|       oscillators[i].speed = SEGMENT.speed > 100 ? random8(2, 4):random8(1, 3);
 | |
|     }
 | |
|     if((oscillators[i].dir == 1) && (oscillators[i].pos >= (SEGLEN - 1))) {
 | |
|       oscillators[i].pos = SEGLEN - 1;
 | |
|       oscillators[i].dir = -1;
 | |
|       oscillators[i].speed = SEGMENT.speed > 100 ? random8(2, 4):random8(1, 3);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for(uint16_t i=0; i < SEGLEN; i++) {
 | |
|     uint32_t color = BLACK;
 | |
|     for(uint8_t j=0; j < numOscillators; j++) {
 | |
|       if(i >= oscillators[j].pos - oscillators[j].size && i <= oscillators[j].pos + oscillators[j].size) {
 | |
|         color = (color == BLACK) ? SEGCOLOR(j) : color_blend(color, SEGCOLOR(j), 128);
 | |
|       }
 | |
|     }
 | |
|     setPixelColor(i, color);
 | |
|   }
 | |
|  
 | |
|   SEGENV.step = it;
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_OSCILLATE PROGMEM = "Oscillate";
 | |
| 
 | |
| 
 | |
| //TODO
 | |
| uint16_t WS2812FX::mode_lightning(void)
 | |
| {
 | |
|   uint16_t ledstart = random16(SEGLEN);               // Determine starting location of flash
 | |
|   uint16_t ledlen = 1 + random16(SEGLEN -ledstart);   // Determine length of flash (not to go beyond NUM_LEDS-1)
 | |
|   uint8_t bri = 255/random8(1, 3);
 | |
| 
 | |
|   if (SEGENV.aux1 == 0) //init, leader flash
 | |
|   {
 | |
|     SEGENV.aux1 = random8(4, 4 + SEGMENT.intensity/20); //number of flashes
 | |
|     SEGENV.aux1 *= 2;
 | |
| 
 | |
|     bri = 52; //leader has lower brightness
 | |
|     SEGENV.aux0 = 200; //200ms delay after leader
 | |
|   }
 | |
| 
 | |
|   fill(SEGCOLOR(1));
 | |
| 
 | |
|   if (SEGENV.aux1 > 3 && !(SEGENV.aux1 & 0x01)) { //flash on even number >2
 | |
|     for (int i = ledstart; i < ledstart + ledlen; i++)
 | |
|     {
 | |
|       setPixelColor(i,color_from_palette(i, true, PALETTE_SOLID_WRAP, 0, bri));
 | |
|     }
 | |
|     SEGENV.aux1--;
 | |
| 
 | |
|     SEGENV.step = millis();
 | |
|     //return random8(4, 10); // each flash only lasts one frame/every 24ms... originally 4-10 milliseconds
 | |
|   } else {
 | |
|     if (millis() - SEGENV.step > SEGENV.aux0) {
 | |
|       SEGENV.aux1--;
 | |
|       if (SEGENV.aux1 < 2) SEGENV.aux1 = 0;
 | |
| 
 | |
|       SEGENV.aux0 = (50 + random8(100)); //delay between flashes
 | |
|       if (SEGENV.aux1 == 2) {
 | |
|         SEGENV.aux0 = (random8(255 - SEGMENT.speed) * 100); // delay between strikes
 | |
|       }
 | |
|       SEGENV.step = millis();
 | |
|     }
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_LIGHTNING PROGMEM = "Lightning";
 | |
| 
 | |
| 
 | |
| // Pride2015
 | |
| // Animated, ever-changing rainbows.
 | |
| // by Mark Kriegsman: https://gist.github.com/kriegsman/964de772d64c502760e5
 | |
| uint16_t WS2812FX::mode_pride_2015(void)
 | |
| {
 | |
|   uint16_t duration = 10 + SEGMENT.speed;
 | |
|   uint16_t sPseudotime = SEGENV.step;
 | |
|   uint16_t sHue16 = SEGENV.aux0;
 | |
| 
 | |
|   uint8_t sat8 = beatsin88( 87, 220, 250);
 | |
|   uint8_t brightdepth = beatsin88( 341, 96, 224);
 | |
|   uint16_t brightnessthetainc16 = beatsin88( 203, (25 * 256), (40 * 256));
 | |
|   uint8_t msmultiplier = beatsin88(147, 23, 60);
 | |
| 
 | |
|   uint16_t hue16 = sHue16;//gHue * 256;
 | |
|   uint16_t hueinc16 = beatsin88(113, 1, 3000);
 | |
| 
 | |
|   sPseudotime += duration * msmultiplier;
 | |
|   sHue16 += duration * beatsin88( 400, 5,9);
 | |
|   uint16_t brightnesstheta16 = sPseudotime;
 | |
|   CRGB fastled_col;
 | |
| 
 | |
|   for (uint16_t i = 0 ; i < SEGLEN; i++) {
 | |
|     hue16 += hueinc16;
 | |
|     uint8_t hue8 = hue16 >> 8;
 | |
| 
 | |
|     brightnesstheta16  += brightnessthetainc16;
 | |
|     uint16_t b16 = sin16( brightnesstheta16  ) + 32768;
 | |
| 
 | |
|     uint16_t bri16 = (uint32_t)((uint32_t)b16 * (uint32_t)b16) / 65536;
 | |
|     uint8_t bri8 = (uint32_t)(((uint32_t)bri16) * brightdepth) / 65536;
 | |
|     bri8 += (255 - brightdepth);
 | |
| 
 | |
|     CRGB newcolor = CHSV( hue8, sat8, bri8);
 | |
|     fastled_col = col_to_crgb(getPixelColor(i));
 | |
| 
 | |
|     nblend(fastled_col, newcolor, 64);
 | |
|     setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|   }
 | |
|   SEGENV.step = sPseudotime;
 | |
|   SEGENV.aux0 = sHue16;
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_PRIDE_2015 PROGMEM = "Pride 2015@!,;;";
 | |
| 
 | |
| 
 | |
| //eight colored dots, weaving in and out of sync with each other
 | |
| uint16_t WS2812FX::mode_juggle(void){
 | |
|   fade_out(SEGMENT.intensity);
 | |
|   CRGB fastled_col;
 | |
|   byte dothue = 0;
 | |
|   for ( byte i = 0; i < 8; i++) {
 | |
|     uint16_t index = 0 + beatsin88((128 + SEGMENT.speed)*(i + 7), 0, SEGLEN -1);
 | |
|     fastled_col = col_to_crgb(getPixelColor(index));
 | |
|     fastled_col |= (SEGMENT.palette==0)?CHSV(dothue, 220, 255):ColorFromPalette(currentPalette, dothue, 255);
 | |
|     setPixelColor(index, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|     dothue += 32;
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_JUGGLE PROGMEM = "Juggle@!=16,Trail=240;!,!,;!";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_palette()
 | |
| {
 | |
|   uint16_t counter = 0;
 | |
|   if (SEGMENT.speed != 0) 
 | |
|   {
 | |
|     counter = (now * ((SEGMENT.speed >> 3) +1)) & 0xFFFF;
 | |
|     counter = counter >> 8;
 | |
|   }
 | |
|   
 | |
|   bool noWrap = (paletteBlend == 2 || (paletteBlend == 0 && SEGMENT.speed == 0));
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++)
 | |
|   {
 | |
|     uint8_t colorIndex = (i * 255 / SEGLEN) - counter;
 | |
|     
 | |
|     if (noWrap) colorIndex = map(colorIndex, 0, 255, 0, 240); //cut off blend at palette "end"
 | |
|     
 | |
|     setPixelColor(i, color_from_palette(colorIndex, false, true, 255));
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_PALETTE PROGMEM = "Palette@!,;1,2,3;!";
 | |
| 
 | |
| 
 | |
| // WLED limitation: Analog Clock overlay will NOT work when Fire2012 is active
 | |
| // Fire2012 by Mark Kriegsman, July 2012
 | |
| // as part of "Five Elements" shown here: http://youtu.be/knWiGsmgycY
 | |
| //// 
 | |
| // This basic one-dimensional 'fire' simulation works roughly as follows:
 | |
| // There's a underlying array of 'heat' cells, that model the temperature
 | |
| // at each point along the line.  Every cycle through the simulation, 
 | |
| // four steps are performed:
 | |
| //  1) All cells cool down a little bit, losing heat to the air
 | |
| //  2) The heat from each cell drifts 'up' and diffuses a little
 | |
| //  3) Sometimes randomly new 'sparks' of heat are added at the bottom
 | |
| //  4) The heat from each cell is rendered as a color into the leds array
 | |
| //     The heat-to-color mapping uses a black-body radiation approximation.
 | |
| //
 | |
| // Temperature is in arbitrary units from 0 (cold black) to 255 (white hot).
 | |
| //
 | |
| // This simulation scales it self a bit depending on NUM_LEDS; it should look
 | |
| // "OK" on anywhere from 20 to 100 LEDs without too much tweaking. 
 | |
| //
 | |
| // I recommend running this simulation at anywhere from 30-100 frames per second,
 | |
| // meaning an interframe delay of about 10-35 milliseconds.
 | |
| //
 | |
| // Looks best on a high-density LED setup (60+ pixels/meter).
 | |
| //
 | |
| //
 | |
| // There are two main parameters you can play with to control the look and
 | |
| // feel of your fire: COOLING (used in step 1 above) (Speed = COOLING), and SPARKING (used
 | |
| // in step 3 above) (Effect Intensity = Sparking).
 | |
| uint16_t WS2812FX::mode_fire_2012()
 | |
| {
 | |
|   const uint16_t cols = isMatrix ? SEGMENT.virtualWidth() : 1;
 | |
|   const uint16_t rows = isMatrix ? SEGMENT.virtualHeight() : SEGMENT.virtualLength();
 | |
| 
 | |
|   uint32_t it = now >> 5; //div 32
 | |
|   uint16_t q  = cols>>2; // a quarter of flames
 | |
| 
 | |
|   if (!SEGENV.allocateData(cols*rows)) return mode_static(); //allocation failed
 | |
|   
 | |
|   byte* heat = SEGENV.data;
 | |
| 
 | |
|   if (it != SEGENV.step) {
 | |
|     SEGENV.step = it;
 | |
|     uint8_t ignition = max(3,rows/10);  // ignition area: 10% of segment length or minimum 3 pixels
 | |
| 
 | |
|     for (uint16_t f = 0; f < cols; f++) {
 | |
|       // Step 1.  Cool down every cell a little
 | |
|       for (uint16_t i = 0; i < rows; i++) {
 | |
|         uint8_t cool = (((20 + SEGMENT.speed/3) * 16) / rows);
 | |
|         // 2D enhancement: cool sides of the flame a bit more
 | |
|         if (cols>5) {
 | |
|           if (f < q)   cool = qadd8(cool, 2*(uint16_t)((cool *    (q-f))/cols)); // cool segment sides a bit more
 | |
|           if (f > 3*q) cool = qadd8(cool, 2*(uint16_t)((cool * (cols-f))/cols)); // cool segment sides a bit more
 | |
|         }
 | |
|         uint8_t temp = qsub8(heat[i+rows*f], random8(0, cool + 2));
 | |
|         heat[i+rows*f] = (temp==0 && i<ignition) ? random8(8,16) : temp; // prevent ignition area from becoming black
 | |
|       }
 | |
| 
 | |
|       // Step 2.  Heat from each cell drifts 'up' and diffuses a little
 | |
|       for (uint16_t k = rows -1; k > 1; k--) {
 | |
|         heat[k+rows*f] = (heat[k+rows*f - 1] + (heat[k+rows*f - 2]<<1) ) / 3;  // heat[k-2] multiplied by 2
 | |
|       }
 | |
| 
 | |
|       // Step 3.  Randomly ignite new 'sparks' of heat near the bottom
 | |
|       if (random8() <= SEGMENT.intensity) {
 | |
|         uint8_t y = random8(ignition);
 | |
|         heat[y+rows*f] = qadd8(heat[y+rows*f], random8(160,255));
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for (uint16_t f = 0; f < cols; f++) {
 | |
|     // Step 4.  Map from heat cells to LED colors
 | |
|     for (uint16_t j = 0; j < rows; j++) {
 | |
|       CRGB color = ColorFromPalette(currentPalette, /*MIN(*/heat[j+rows*f]/*,240)*/, 255, LINEARBLEND);
 | |
|       if (isMatrix) setPixelColorXY(f, rows -j -1, color);
 | |
|       else          setPixelColor(j, color);
 | |
|     }
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_FIRE_2012 PROGMEM = "Fire 2012@Cooling=120,Spark rate=64;1,2,3;!";
 | |
| 
 | |
| 
 | |
| // ColorWavesWithPalettes by Mark Kriegsman: https://gist.github.com/kriegsman/8281905786e8b2632aeb
 | |
| // This function draws color waves with an ever-changing,
 | |
| // widely-varying set of parameters, using a color palette.
 | |
| uint16_t WS2812FX::mode_colorwaves()
 | |
| {
 | |
|   uint16_t duration = 10 + SEGMENT.speed;
 | |
|   uint16_t sPseudotime = SEGENV.step;
 | |
|   uint16_t sHue16 = SEGENV.aux0;
 | |
| 
 | |
|   uint8_t brightdepth = beatsin88(341, 96, 224);
 | |
|   uint16_t brightnessthetainc16 = beatsin88( 203, (25 * 256), (40 * 256));
 | |
|   uint8_t msmultiplier = beatsin88(147, 23, 60);
 | |
| 
 | |
|   uint16_t hue16 = sHue16;//gHue * 256;
 | |
|   // uint16_t hueinc16 = beatsin88(113, 300, 1500);
 | |
|   uint16_t hueinc16 = beatsin88(113, 60, 300)*SEGMENT.intensity*10/255;  // Use the Intensity Slider for the hues
 | |
| 
 | |
|   sPseudotime += duration * msmultiplier;
 | |
|   sHue16 += duration * beatsin88(400, 5, 9);
 | |
|   uint16_t brightnesstheta16 = sPseudotime;
 | |
|   CRGB fastled_col;
 | |
| 
 | |
|   for ( uint16_t i = 0 ; i < SEGLEN; i++) {
 | |
|     hue16 += hueinc16;
 | |
|     uint8_t hue8 = hue16 >> 8;
 | |
|     uint16_t h16_128 = hue16 >> 7;
 | |
|     if ( h16_128 & 0x100) {
 | |
|       hue8 = 255 - (h16_128 >> 1);
 | |
|     } else {
 | |
|       hue8 = h16_128 >> 1;
 | |
|     }
 | |
| 
 | |
|     brightnesstheta16  += brightnessthetainc16;
 | |
|     uint16_t b16 = sin16( brightnesstheta16  ) + 32768;
 | |
| 
 | |
|     uint16_t bri16 = (uint32_t)((uint32_t)b16 * (uint32_t)b16) / 65536;
 | |
|     uint8_t bri8 = (uint32_t)(((uint32_t)bri16) * brightdepth) / 65536;
 | |
|     bri8 += (255 - brightdepth);
 | |
| 
 | |
|     CRGB newcolor = ColorFromPalette(currentPalette, hue8, bri8);
 | |
|     fastled_col = col_to_crgb(getPixelColor(i));
 | |
| 
 | |
|     nblend(fastled_col, newcolor, 128);
 | |
|     setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|   }
 | |
|   SEGENV.step = sPseudotime;
 | |
|   SEGENV.aux0 = sHue16;
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_COLORWAVES PROGMEM = "Colorwaves";
 | |
| 
 | |
| 
 | |
| // colored stripes pulsing at a defined Beats-Per-Minute (BPM)
 | |
| uint16_t WS2812FX::mode_bpm()
 | |
| {
 | |
|   CRGB fastled_col;
 | |
|   uint32_t stp = (now / 20) & 0xFF;
 | |
|   uint8_t beat = beatsin8(SEGMENT.speed, 64, 255);
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     fastled_col = ColorFromPalette(currentPalette, stp + (i * 2), beat - stp + (i * 10));
 | |
|     setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_BPM PROGMEM = "Bpm@!=64,;1,2,3;!";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_fillnoise8()
 | |
| {
 | |
|   if (SEGENV.call == 0) SEGENV.step = random16(12345);
 | |
|   CRGB fastled_col;
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     uint8_t index = inoise8(i * SEGLEN, SEGENV.step + i * SEGLEN);
 | |
|     fastled_col = ColorFromPalette(currentPalette, index, 255, LINEARBLEND);
 | |
|     setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|   }
 | |
|   SEGENV.step += beatsin8(SEGMENT.speed, 1, 6); //10,1,4
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_FILLNOISE8 PROGMEM = "Fill Noise";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_noise16_1()
 | |
| {
 | |
|   uint16_t scale = 320;                                      // the "zoom factor" for the noise
 | |
|   CRGB fastled_col;
 | |
|   SEGENV.step += (1 + SEGMENT.speed/16);
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
| 
 | |
|     uint16_t shift_x = beatsin8(11);                           // the x position of the noise field swings @ 17 bpm
 | |
|     uint16_t shift_y = SEGENV.step/42;             // the y position becomes slowly incremented
 | |
| 
 | |
| 
 | |
|     uint16_t real_x = (i + shift_x) * scale;                  // the x position of the noise field swings @ 17 bpm
 | |
|     uint16_t real_y = (i + shift_y) * scale;                  // the y position becomes slowly incremented
 | |
|     uint32_t real_z = SEGENV.step;                          // the z position becomes quickly incremented
 | |
| 
 | |
|     uint8_t noise = inoise16(real_x, real_y, real_z) >> 8;   // get the noise data and scale it down
 | |
| 
 | |
|     uint8_t index = sin8(noise * 3);                         // map LED color based on noise data
 | |
| 
 | |
|     fastled_col = ColorFromPalette(currentPalette, index, 255, LINEARBLEND);   // With that value, look up the 8 bit colour palette value and assign it to the current LED.
 | |
|     setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_NOISE16_1 PROGMEM = "Noise 1";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_noise16_2()
 | |
| {
 | |
|   uint16_t scale = 1000;                                       // the "zoom factor" for the noise
 | |
|   CRGB fastled_col;
 | |
|   SEGENV.step += (1 + (SEGMENT.speed >> 1));
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
| 
 | |
|     uint16_t shift_x = SEGENV.step >> 6;                         // x as a function of time
 | |
| 
 | |
|     uint32_t real_x = (i + shift_x) * scale;                  // calculate the coordinates within the noise field
 | |
| 
 | |
|     uint8_t noise = inoise16(real_x, 0, 4223) >> 8;    // get the noise data and scale it down
 | |
| 
 | |
|     uint8_t index = sin8(noise * 3);                          // map led color based on noise data
 | |
| 
 | |
|     fastled_col = ColorFromPalette(currentPalette, index, noise, LINEARBLEND);   // With that value, look up the 8 bit colour palette value and assign it to the current LED.
 | |
|     setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_NOISE16_2 PROGMEM = "Noise 2";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_noise16_3()
 | |
| {
 | |
|   uint16_t scale = 800;                                       // the "zoom factor" for the noise
 | |
|   CRGB fastled_col;
 | |
|   SEGENV.step += (1 + SEGMENT.speed);
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
| 
 | |
|     uint16_t shift_x = 4223;                                  // no movement along x and y
 | |
|     uint16_t shift_y = 1234;
 | |
| 
 | |
|     uint32_t real_x = (i + shift_x) * scale;                  // calculate the coordinates within the noise field
 | |
|     uint32_t real_y = (i + shift_y) * scale;                  // based on the precalculated positions
 | |
|     uint32_t real_z = SEGENV.step*8;  
 | |
| 
 | |
|     uint8_t noise = inoise16(real_x, real_y, real_z) >> 8;    // get the noise data and scale it down
 | |
| 
 | |
|     uint8_t index = sin8(noise * 3);                          // map led color based on noise data
 | |
| 
 | |
|     fastled_col = ColorFromPalette(currentPalette, index, noise, LINEARBLEND);   // With that value, look up the 8 bit colour palette value and assign it to the current LED.
 | |
|     setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_NOISE16_3 PROGMEM = "Noise 3";
 | |
| 
 | |
| 
 | |
| //https://github.com/aykevl/ledstrip-spark/blob/master/ledstrip.ino
 | |
| uint16_t WS2812FX::mode_noise16_4()
 | |
| {
 | |
|   CRGB fastled_col;
 | |
|   uint32_t stp = (now * SEGMENT.speed) >> 7;
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     int16_t index = inoise16(uint32_t(i) << 12, stp);
 | |
|     fastled_col = ColorFromPalette(currentPalette, index);
 | |
|     setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_NOISE16_4 PROGMEM = "Noise 4";
 | |
| 
 | |
| 
 | |
| //based on https://gist.github.com/kriegsman/5408ecd397744ba0393e
 | |
| uint16_t WS2812FX::mode_colortwinkle()
 | |
| {
 | |
|   const uint16_t cols = isMatrix ? SEGMENT.virtualWidth() : 1;
 | |
|   const uint16_t rows = isMatrix ? SEGMENT.virtualHeight() : SEGMENT.virtualLength();
 | |
| 
 | |
|   uint16_t dataSize = (cols*rows+7) >> 3; //1 bit per LED
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   
 | |
|   CRGB fastled_col, prev;
 | |
|   fract8 fadeUpAmount = _brightness>28 ? 8 + (SEGMENT.speed>>2) : 68-_brightness, fadeDownAmount = _brightness>28 ? 8 + (SEGMENT.speed>>3) : 68-_brightness;
 | |
| 
 | |
|   for (uint16_t i = 0; i < rows*cols; i++) {
 | |
|     uint16_t j = i % rows, k = i / rows;
 | |
|     fastled_col = col_to_crgb(isMatrix ? getPixelColorXY(j, k) : getPixelColor(i));
 | |
|     prev = fastled_col;
 | |
|     uint16_t index = i >> 3;
 | |
|     uint8_t  bitNum = i & 0x07;
 | |
|     bool fadeUp = bitRead(SEGENV.data[index], bitNum);
 | |
|     
 | |
|     if (fadeUp) {
 | |
|       CRGB incrementalColor = fastled_col;
 | |
|       incrementalColor.nscale8_video(fadeUpAmount);
 | |
|       fastled_col += incrementalColor;
 | |
| 
 | |
|       if (fastled_col.red == 255 || fastled_col.green == 255 || fastled_col.blue == 255) {
 | |
|         bitWrite(SEGENV.data[index], bitNum, false);
 | |
|       }
 | |
| 
 | |
|       if (isMatrix) setPixelColorXY(j, k, fastled_col);
 | |
|       else          setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
| 
 | |
|       uint32_t col = isMatrix ? getPixelColorXY(j, k) : getPixelColor(i);
 | |
|       if (col_to_crgb(col) == prev) {  //fix "stuck" pixels
 | |
|         fastled_col += fastled_col;
 | |
|         if (isMatrix) setPixelColorXY(j, k, fastled_col);
 | |
|         else          setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|       }
 | |
|     } else {
 | |
|       fastled_col.nscale8(255 - fadeDownAmount);
 | |
|       if (isMatrix) setPixelColorXY(j, k, fastled_col);
 | |
|       else          setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for (uint16_t j = 0; j <= rows*cols / 50; j++) {
 | |
|     if (random8() <= SEGMENT.intensity) {
 | |
|       for (uint8_t times = 0; times < 5; times++) { //attempt to spawn a new pixel 5 times
 | |
|         uint16_t i = random16(rows*cols);
 | |
|         uint16_t j = i % rows, k = i / rows;
 | |
|         uint32_t col = isMatrix ? getPixelColorXY(j, k) : getPixelColor(i);
 | |
|         if (col == 0) {
 | |
|           fastled_col = ColorFromPalette(currentPalette, random8(), 64, NOBLEND);
 | |
|           uint16_t index = i >> 3;
 | |
|           uint8_t  bitNum = i & 0x07;
 | |
|           bitWrite(SEGENV.data[index], bitNum, true);
 | |
|           if (isMatrix) setPixelColorXY(j, k, fastled_col);
 | |
|           else          setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|           break; //only spawn 1 new pixel per frame per 50 LEDs
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   return FRAMETIME_FIXED;
 | |
| }
 | |
| static const char *_data_FX_MODE_COLORTWINKLE PROGMEM = "Colortwinkles@Fade speed,Spawn speed;1,2,3;!";
 | |
| 
 | |
| 
 | |
| //Calm effect, like a lake at night
 | |
| uint16_t WS2812FX::mode_lake() {
 | |
|   uint8_t sp = SEGMENT.speed/10;
 | |
|   int wave1 = beatsin8(sp +2, -64,64);
 | |
|   int wave2 = beatsin8(sp +1, -64,64);
 | |
|   uint8_t wave3 = beatsin8(sp +2,   0,80);
 | |
|   CRGB fastled_col;
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++)
 | |
|   {
 | |
|     int index = cos8((i*15)+ wave1)/2 + cubicwave8((i*23)+ wave2)/2;           
 | |
|     uint8_t lum = (index > wave3) ? index - wave3 : 0;
 | |
|     fastled_col = ColorFromPalette(currentPalette, map(index,0,255,0,240), lum, LINEARBLEND);
 | |
|     setPixelColor(i, fastled_col.red, fastled_col.green, fastled_col.blue);
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_LAKE PROGMEM = "Lake@!,;1,2,3;!";
 | |
| 
 | |
| 
 | |
| // meteor effect
 | |
| // send a meteor from begining to to the end of the strip with a trail that randomly decays.
 | |
| // adapted from https://www.tweaking4all.com/hardware/arduino/adruino-led-strip-effects/#LEDStripEffectMeteorRain
 | |
| uint16_t WS2812FX::mode_meteor() {
 | |
|   if (!SEGENV.allocateData(SEGLEN)) return mode_static(); //allocation failed
 | |
| 
 | |
|   byte* trail = SEGENV.data;
 | |
|   
 | |
|   byte meteorSize= 1+ SEGLEN / 10;
 | |
|   uint16_t counter = now * ((SEGMENT.speed >> 2) +8);
 | |
|   uint16_t in = counter * SEGLEN >> 16;
 | |
| 
 | |
|   // fade all leds to colors[1] in LEDs one step
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     if (random8() <= 255 - SEGMENT.intensity)
 | |
|     {
 | |
|       byte meteorTrailDecay = 128 + random8(127);
 | |
|       trail[i] = scale8(trail[i], meteorTrailDecay);
 | |
|       setPixelColor(i, color_from_palette(trail[i], false, true, 255));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // draw meteor
 | |
|   for(int j = 0; j < meteorSize; j++) {
 | |
|     uint16_t index = in + j;
 | |
|     if(index >= SEGLEN) {
 | |
|       index = (in + j - SEGLEN);
 | |
|     }
 | |
| 
 | |
|     trail[index] = 240;
 | |
|     setPixelColor(index, color_from_palette(trail[index], false, true, 255));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_METEOR PROGMEM = "Meteor@!,Trail length;!,!,;!";
 | |
| 
 | |
| 
 | |
| // smooth meteor effect
 | |
| // send a meteor from begining to to the end of the strip with a trail that randomly decays.
 | |
| // adapted from https://www.tweaking4all.com/hardware/arduino/adruino-led-strip-effects/#LEDStripEffectMeteorRain
 | |
| uint16_t WS2812FX::mode_meteor_smooth() {
 | |
|   if (!SEGENV.allocateData(SEGLEN)) return mode_static(); //allocation failed
 | |
| 
 | |
|   byte* trail = SEGENV.data;
 | |
|   
 | |
|   byte meteorSize= 1+ SEGLEN / 10;
 | |
|   uint16_t in = map((SEGENV.step >> 6 & 0xFF), 0, 255, 0, SEGLEN -1);
 | |
| 
 | |
|   // fade all leds to colors[1] in LEDs one step
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     if (trail[i] != 0 && random8() <= 255 - SEGMENT.intensity)
 | |
|     {
 | |
|       int change = 3 - random8(12); //change each time between -8 and +3
 | |
|       trail[i] += change;
 | |
|       if (trail[i] > 245) trail[i] = 0;
 | |
|       if (trail[i] > 240) trail[i] = 240;
 | |
|       setPixelColor(i, color_from_palette(trail[i], false, true, 255));
 | |
|     }
 | |
|   }
 | |
|   
 | |
|   // draw meteor
 | |
|   for(int j = 0; j < meteorSize; j++) {  
 | |
|     uint16_t index = in + j;   
 | |
|     if(in + j >= SEGLEN) {
 | |
|       index = (in + j - SEGLEN);
 | |
|     }
 | |
|     setPixelColor(index, color_blend(getPixelColor(index), color_from_palette(240, false, true, 255), 48));
 | |
|     trail[index] = 240;
 | |
|   }
 | |
| 
 | |
|   SEGENV.step += SEGMENT.speed +1;
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_METEOR_SMOOTH PROGMEM = "Meteor Smooth@!,Trail length;!,!,;!";
 | |
| 
 | |
| 
 | |
| //Railway Crossing / Christmas Fairy lights
 | |
| uint16_t WS2812FX::mode_railway()
 | |
| {
 | |
|   uint16_t dur = 40 + (255 - SEGMENT.speed) * 10;
 | |
|   uint16_t rampdur = (dur * SEGMENT.intensity) >> 8;
 | |
|   if (SEGENV.step > dur)
 | |
|   {
 | |
|     //reverse direction
 | |
|     SEGENV.step = 0;
 | |
|     SEGENV.aux0 = !SEGENV.aux0;
 | |
|   }
 | |
|   uint8_t pos = 255;
 | |
|   if (rampdur != 0)
 | |
|   {
 | |
|     uint16_t p0 = (SEGENV.step * 255) / rampdur;
 | |
|     if (p0 < 255) pos = p0;
 | |
|   }
 | |
|   if (SEGENV.aux0) pos = 255 - pos;
 | |
|   for (uint16_t i = 0; i < SEGLEN; i += 2)
 | |
|   {
 | |
|     setPixelColor(i, color_from_palette(255 - pos, false, false, 255));
 | |
|     if (i < SEGLEN -1)
 | |
|     {
 | |
|       setPixelColor(i + 1, color_from_palette(pos, false, false, 255));
 | |
|     }
 | |
|   }
 | |
|   SEGENV.step += FRAMETIME;
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_RAILWAY PROGMEM = "Railway";
 | |
| 
 | |
| 
 | |
| //Water ripple
 | |
| //propagation velocity from speed
 | |
| //drop rate from intensity
 | |
| 
 | |
| //4 bytes
 | |
| typedef struct Ripple {
 | |
|   uint8_t state;
 | |
|   uint8_t color;
 | |
|   uint16_t pos;
 | |
| } ripple;
 | |
| 
 | |
| #ifdef ESP8266
 | |
|   #define MAX_RIPPLES   56
 | |
| #else
 | |
|   #define MAX_RIPPLES  100
 | |
| #endif
 | |
| uint16_t WS2812FX::ripple_base(bool rainbow)
 | |
| {
 | |
|   uint16_t maxRipples = min(1 + (SEGLEN >> 2), MAX_RIPPLES);  // 56 max for 16 segment ESP8266
 | |
|   uint16_t dataSize = sizeof(ripple) * maxRipples;
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|  
 | |
|   Ripple* ripples = reinterpret_cast<Ripple*>(SEGENV.data);
 | |
| 
 | |
|   // ranbow background or chosen background, all very dim.
 | |
|   if (rainbow) {
 | |
|     if (SEGENV.call ==0) {
 | |
|       SEGENV.aux0 = random8();
 | |
|       SEGENV.aux1 = random8();
 | |
|     }
 | |
|     if (SEGENV.aux0 == SEGENV.aux1) {
 | |
|       SEGENV.aux1 = random8();
 | |
|     }
 | |
|     else if (SEGENV.aux1 > SEGENV.aux0) {
 | |
|       SEGENV.aux0++;
 | |
|     } else {
 | |
|       SEGENV.aux0--;
 | |
|     }
 | |
|     fill(color_blend(color_wheel(SEGENV.aux0),BLACK,235));
 | |
|   } else {
 | |
|     fill(SEGCOLOR(1));
 | |
|   }
 | |
|   
 | |
|   //draw wave
 | |
|   for (uint16_t i = 0; i < maxRipples; i++)
 | |
|   {
 | |
|     uint16_t ripplestate = ripples[i].state;
 | |
|     if (ripplestate)
 | |
|     {
 | |
|       uint8_t rippledecay = (SEGMENT.speed >> 4) +1; //faster decay if faster propagation
 | |
|       uint16_t rippleorigin = ripples[i].pos;
 | |
|       uint32_t col = color_from_palette(ripples[i].color, false, false, 255);
 | |
|       uint16_t propagation = ((ripplestate/rippledecay -1) * SEGMENT.speed);
 | |
|       int16_t propI = propagation >> 8;
 | |
|       uint8_t propF = propagation & 0xFF;
 | |
|       int16_t left = rippleorigin - propI -1;
 | |
|       uint8_t amp = (ripplestate < 17) ? triwave8((ripplestate-1)*8) : map(ripplestate,17,255,255,2);
 | |
| 
 | |
|       for (int16_t v = left; v < left +4; v++)
 | |
|       {
 | |
|         uint8_t mag = scale8(cubicwave8((propF>>2)+(v-left)*64), amp);
 | |
|         if (v < SEGLEN && v >= 0)
 | |
|         {
 | |
|           setPixelColor(v, color_blend(getPixelColor(v), col, mag));
 | |
|         }
 | |
|         int16_t w = left + propI*2 + 3 -(v-left);
 | |
|         if (w < SEGLEN && w >= 0)
 | |
|         {
 | |
|           setPixelColor(w, color_blend(getPixelColor(w), col, mag));
 | |
|         }
 | |
|       }  
 | |
|       ripplestate += rippledecay;
 | |
|       ripples[i].state = (ripplestate > 254) ? 0 : ripplestate;
 | |
|     } else //randomly create new wave
 | |
|     {
 | |
|       if (random16(IBN + 10000) <= SEGMENT.intensity)
 | |
|       {
 | |
|         ripples[i].state = 1;
 | |
|         ripples[i].pos = random16(SEGLEN);
 | |
|         ripples[i].color = random8(); //color
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| #undef MAX_RIPPLES
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_ripple(void) {
 | |
|   return ripple_base(false);
 | |
| }
 | |
| static const char *_data_FX_MODE_RIPPLE PROGMEM = "Ripple";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_ripple_rainbow(void) {
 | |
|   return ripple_base(true);
 | |
| }
 | |
| static const char *_data_FX_MODE_RIPPLE_RAINBOW PROGMEM = "Ripple Rainbow";
 | |
| 
 | |
| 
 | |
| //  TwinkleFOX by Mark Kriegsman: https://gist.github.com/kriegsman/756ea6dcae8e30845b5a
 | |
| //
 | |
| //  TwinkleFOX: Twinkling 'holiday' lights that fade in and out.
 | |
| //  Colors are chosen from a palette. Read more about this effect using the link above!
 | |
| 
 | |
| // If COOL_LIKE_INCANDESCENT is set to 1, colors will
 | |
| // fade out slighted 'reddened', similar to how
 | |
| // incandescent bulbs change color as they get dim down.
 | |
| #define COOL_LIKE_INCANDESCENT 1
 | |
| 
 | |
| CRGB WS2812FX::twinklefox_one_twinkle(uint32_t ms, uint8_t salt, bool cat)
 | |
| {
 | |
|   // Overall twinkle speed (changed)
 | |
|   uint16_t ticks = ms / SEGENV.aux0;
 | |
|   uint8_t fastcycle8 = ticks;
 | |
|   uint16_t slowcycle16 = (ticks >> 8) + salt;
 | |
|   slowcycle16 += sin8(slowcycle16);
 | |
|   slowcycle16 = (slowcycle16 * 2053) + 1384;
 | |
|   uint8_t slowcycle8 = (slowcycle16 & 0xFF) + (slowcycle16 >> 8);
 | |
|   
 | |
|   // Overall twinkle density.
 | |
|   // 0 (NONE lit) to 8 (ALL lit at once).
 | |
|   // Default is 5.
 | |
|   uint8_t twinkleDensity = (SEGMENT.intensity >> 5) +1;
 | |
| 
 | |
|   uint8_t bright = 0;
 | |
|   if (((slowcycle8 & 0x0E)/2) < twinkleDensity) {
 | |
|     uint8_t ph = fastcycle8;
 | |
|     // This is like 'triwave8', which produces a
 | |
|     // symmetrical up-and-down triangle sawtooth waveform, except that this
 | |
|     // function produces a triangle wave with a faster attack and a slower decay
 | |
|     if (cat) //twinklecat, variant where the leds instantly turn on
 | |
|     {
 | |
|       bright = 255 - ph;
 | |
|     } else { //vanilla twinklefox
 | |
|       if (ph < 86) {
 | |
|       bright = ph * 3;
 | |
|       } else {
 | |
|         ph -= 86;
 | |
|         bright = 255 - (ph + (ph/2));
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   uint8_t hue = slowcycle8 - salt;
 | |
|   CRGB c;
 | |
|   if (bright > 0) {
 | |
|     c = ColorFromPalette(currentPalette, hue, bright, NOBLEND);
 | |
|     if(COOL_LIKE_INCANDESCENT == 1) {
 | |
|       // This code takes a pixel, and if its in the 'fading down'
 | |
|       // part of the cycle, it adjusts the color a little bit like the
 | |
|       // way that incandescent bulbs fade toward 'red' as they dim.
 | |
|       if (fastcycle8 >= 128) 
 | |
|       {
 | |
|         uint8_t cooling = (fastcycle8 - 128) >> 4;
 | |
|         c.g = qsub8(c.g, cooling);
 | |
|         c.b = qsub8(c.b, cooling * 2);
 | |
|       }
 | |
|     }
 | |
|   } else {
 | |
|     c = CRGB::Black;
 | |
|   }
 | |
|   return c;
 | |
| }
 | |
| 
 | |
| //  This function loops over each pixel, calculates the
 | |
| //  adjusted 'clock' that this pixel should use, and calls
 | |
| //  "CalculateOneTwinkle" on each pixel.  It then displays
 | |
| //  either the twinkle color of the background color,
 | |
| //  whichever is brighter.
 | |
| uint16_t WS2812FX::twinklefox_base(bool cat)
 | |
| {
 | |
|   // "PRNG16" is the pseudorandom number generator
 | |
|   // It MUST be reset to the same starting value each time
 | |
|   // this function is called, so that the sequence of 'random'
 | |
|   // numbers that it generates is (paradoxically) stable.
 | |
|   uint16_t PRNG16 = 11337;
 | |
| 
 | |
|   // Calculate speed
 | |
|   if (SEGMENT.speed > 100) SEGENV.aux0 = 3 + ((255 - SEGMENT.speed) >> 3);
 | |
|   else SEGENV.aux0 = 22 + ((100 - SEGMENT.speed) >> 1);
 | |
| 
 | |
|   // Set up the background color, "bg".
 | |
|   CRGB bg;
 | |
|   bg = col_to_crgb(SEGCOLOR(1));
 | |
|   uint8_t bglight = bg.getAverageLight();
 | |
|   if (bglight > 64) {
 | |
|     bg.nscale8_video(16); // very bright, so scale to 1/16th
 | |
|   } else if (bglight > 16) {
 | |
|     bg.nscale8_video(64); // not that bright, so scale to 1/4th
 | |
|   } else {
 | |
|     bg.nscale8_video(86); // dim, scale to 1/3rd.
 | |
|   }
 | |
| 
 | |
|   uint8_t backgroundBrightness = bg.getAverageLight();
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|   
 | |
|     PRNG16 = (uint16_t)(PRNG16 * 2053) + 1384; // next 'random' number
 | |
|     uint16_t myclockoffset16= PRNG16; // use that number as clock offset
 | |
|     PRNG16 = (uint16_t)(PRNG16 * 2053) + 1384; // next 'random' number
 | |
|     // use that number as clock speed adjustment factor (in 8ths, from 8/8ths to 23/8ths)
 | |
|     uint8_t myspeedmultiplierQ5_3 =  ((((PRNG16 & 0xFF)>>4) + (PRNG16 & 0x0F)) & 0x0F) + 0x08;
 | |
|     uint32_t myclock30 = (uint32_t)((now * myspeedmultiplierQ5_3) >> 3) + myclockoffset16;
 | |
|     uint8_t  myunique8 = PRNG16 >> 8; // get 'salt' value for this pixel
 | |
| 
 | |
|     // We now have the adjusted 'clock' for this pixel, now we call
 | |
|     // the function that computes what color the pixel should be based
 | |
|     // on the "brightness = f( time )" idea.
 | |
|     CRGB c = twinklefox_one_twinkle(myclock30, myunique8, cat);
 | |
| 
 | |
|     uint8_t cbright = c.getAverageLight();
 | |
|     int16_t deltabright = cbright - backgroundBrightness;
 | |
|     if (deltabright >= 32 || (!bg)) {
 | |
|       // If the new pixel is significantly brighter than the background color,
 | |
|       // use the new color.
 | |
|       setPixelColor(i, c.red, c.green, c.blue);
 | |
|     } else if (deltabright > 0) {
 | |
|       // If the new pixel is just slightly brighter than the background color,
 | |
|       // mix a blend of the new color and the background color
 | |
|       setPixelColor(i, color_blend(crgb_to_col(bg), crgb_to_col(c), deltabright * 8));
 | |
|     } else {
 | |
|       // if the new pixel is not at all brighter than the background color,
 | |
|       // just use the background color.
 | |
|       setPixelColor(i, bg.r, bg.g, bg.b);
 | |
|     }
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_twinklefox()
 | |
| {
 | |
|   return twinklefox_base(false);
 | |
| }
 | |
| static const char *_data_FX_MODE_TWINKLEFOX PROGMEM = "Twinklefox";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_twinklecat()
 | |
| {
 | |
|   return twinklefox_base(true);
 | |
| }
 | |
| static const char *_data_FX_MODE_TWINKLECAT PROGMEM = "Twinklecat";
 | |
| 
 | |
| 
 | |
| //inspired by https://www.tweaking4all.com/hardware/arduino/adruino-led-strip-effects/#LEDStripEffectBlinkingHalloweenEyes
 | |
| #define HALLOWEEN_EYE_SPACE (2*MAX(1,SEGLEN>>5))
 | |
| #define HALLOWEEN_EYE_WIDTH MAX(1,SEGLEN>>5)
 | |
| 
 | |
| uint16_t WS2812FX::mode_halloween_eyes()
 | |
| {  
 | |
|   uint16_t eyeLength = (2*HALLOWEEN_EYE_WIDTH) + HALLOWEEN_EYE_SPACE;
 | |
|   if (eyeLength > SEGLEN) return mode_static(); //bail if segment too short
 | |
| 
 | |
|   fill(SEGCOLOR(1)); //fill background
 | |
| 
 | |
|   uint8_t state = SEGENV.aux1 >> 8;
 | |
|   uint16_t stateTime = SEGENV.call;
 | |
|   if (stateTime == 0) stateTime = 2000;
 | |
| 
 | |
|   if (state == 0) { //spawn eyes
 | |
|     SEGENV.aux0 = random16(0, SEGLEN - eyeLength); //start pos
 | |
|     SEGENV.aux1 = random8(); //color
 | |
|     if (isMatrix) SEGMENT.offset = random16(SEGMENT.virtualHeight()-1); // a hack: reuse offset since it is not used in matrices
 | |
|     state = 1;
 | |
|   }
 | |
|   
 | |
|   if (state < 2) { //fade eyes
 | |
|     uint16_t startPos    = SEGENV.aux0;
 | |
|     uint16_t start2ndEye = startPos + HALLOWEEN_EYE_WIDTH + HALLOWEEN_EYE_SPACE;
 | |
|     
 | |
|     uint32_t fadestage = (now - SEGENV.step)*255 / stateTime;
 | |
|     if (fadestage > 255) fadestage = 255;
 | |
|     uint32_t c = color_blend(color_from_palette(SEGENV.aux1 & 0xFF, false, false, 0), SEGCOLOR(1), fadestage);
 | |
|     
 | |
|     for (uint16_t i = 0; i < HALLOWEEN_EYE_WIDTH; i++) {
 | |
|       if (isMatrix) {
 | |
|         setPixelColorXY(startPos    + i, SEGMENT.offset, c);
 | |
|         setPixelColorXY(start2ndEye + i, SEGMENT.offset, c);
 | |
|       } else {
 | |
|         setPixelColor(startPos    + i, c);
 | |
|         setPixelColor(start2ndEye + i, c);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (now - SEGENV.step > stateTime) {
 | |
|     state++;
 | |
|     if (state > 2) state = 0;
 | |
|     
 | |
|     if (state < 2) {
 | |
|       stateTime = 100 + (255 - SEGMENT.intensity)*10; //eye fade time
 | |
|     } else {
 | |
|       uint16_t eyeOffTimeBase = (255 - SEGMENT.speed)*10;
 | |
|       stateTime = eyeOffTimeBase + random16(eyeOffTimeBase);
 | |
|     }
 | |
|     SEGENV.step = now;
 | |
|     SEGENV.call = stateTime;
 | |
|   }
 | |
| 
 | |
|   SEGENV.aux1 = (SEGENV.aux1 & 0xFF) + (state << 8); //save state
 | |
|   
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_HALLOWEEN_EYES PROGMEM = "Halloween Eyes@Duration,Eye fade time;!,!,;!";
 | |
| 
 | |
| 
 | |
| //Speed slider sets amount of LEDs lit, intensity sets unlit
 | |
| uint16_t WS2812FX::mode_static_pattern()
 | |
| {
 | |
|   uint16_t lit = 1 + SEGMENT.speed;
 | |
|   uint16_t unlit = 1 + SEGMENT.intensity;
 | |
|   bool drawingLit = true;
 | |
|   uint16_t cnt = 0;
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     setPixelColor(i, (drawingLit) ? color_from_palette(i, true, PALETTE_SOLID_WRAP, 0) : SEGCOLOR(1));
 | |
|     cnt++;
 | |
|     if (cnt >= ((drawingLit) ? lit : unlit)) {
 | |
|       cnt = 0;
 | |
|       drawingLit = !drawingLit;
 | |
|     }
 | |
|   }
 | |
|   
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_STATIC_PATTERN PROGMEM = "Solid Pattern@Fg size,Bg size;Fg,Bg,;!=0";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_tri_static_pattern()
 | |
| {
 | |
|   uint8_t segSize = (SEGMENT.intensity >> 5) +1;
 | |
|   uint8_t currSeg = 0;
 | |
|   uint16_t currSegCount = 0;
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     if ( currSeg % 3 == 0 ) {
 | |
|       setPixelColor(i, SEGCOLOR(0));
 | |
|     } else if( currSeg % 3 == 1) {
 | |
|       setPixelColor(i, SEGCOLOR(1));
 | |
|     } else {
 | |
|       setPixelColor(i, (SEGCOLOR(2) > 0 ? SEGCOLOR(2) : WHITE));
 | |
|     }
 | |
|     currSegCount += 1;
 | |
|     if (currSegCount >= segSize) {
 | |
|       currSeg +=1;
 | |
|       currSegCount = 0;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_TRI_STATIC_PATTERN PROGMEM = "Solid Pattern Tri@,Size;1,2,3;!=0";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::spots_base(uint16_t threshold)
 | |
| {
 | |
|   fill(SEGCOLOR(1));
 | |
|   
 | |
|   uint16_t maxZones = SEGLEN >> 2;
 | |
|   uint16_t zones = 1 + ((SEGMENT.intensity * maxZones) >> 8);
 | |
|   uint16_t zoneLen = SEGLEN / zones;
 | |
|   uint16_t offset = (SEGLEN - zones * zoneLen) >> 1;
 | |
| 
 | |
|   for (uint16_t z = 0; z < zones; z++)
 | |
|   {
 | |
|     uint16_t pos = offset + z * zoneLen;
 | |
|     for (uint16_t i = 0; i < zoneLen; i++)
 | |
|     {
 | |
|       uint16_t wave = triwave16((i * 0xFFFF) / zoneLen);
 | |
|       if (wave > threshold) {
 | |
|         uint16_t index = 0 + pos + i;
 | |
|         uint8_t s = (wave - threshold)*255 / (0xFFFF - threshold);
 | |
|         setPixelColor(index, color_blend(color_from_palette(index, true, PALETTE_SOLID_WRAP, 0), SEGCOLOR(1), 255-s));
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| //Intensity slider sets number of "lights", speed sets LEDs per light
 | |
| uint16_t WS2812FX::mode_spots()
 | |
| {
 | |
|   return spots_base((255 - SEGMENT.speed) << 8);
 | |
| }
 | |
| static const char *_data_FX_MODE_SPOTS PROGMEM = "Spots@Spread,Width;!,!,;!";
 | |
| 
 | |
| 
 | |
| //Intensity slider sets number of "lights", LEDs per light fade in and out
 | |
| uint16_t WS2812FX::mode_spots_fade()
 | |
| {
 | |
|   uint16_t counter = now * ((SEGMENT.speed >> 2) +8);
 | |
|   uint16_t t = triwave16(counter);
 | |
|   uint16_t tr = (t >> 1) + (t >> 2);
 | |
|   return spots_base(tr);
 | |
| }
 | |
| static const char *_data_FX_MODE_SPOTS_FADE PROGMEM = "Spots Fade@Spread,Width;!,!,;!";
 | |
| 
 | |
| 
 | |
| //each needs 12 bytes
 | |
| typedef struct Ball {
 | |
|   unsigned long lastBounceTime;
 | |
|   float impactVelocity;
 | |
|   float height;
 | |
| } ball;
 | |
| 
 | |
| /*
 | |
| *  Bouncing Balls Effect
 | |
| */
 | |
| uint16_t WS2812FX::mode_bouncing_balls(void) {
 | |
|   //allocate segment data
 | |
|   uint16_t maxNumBalls = 16; 
 | |
|   uint16_t dataSize = sizeof(ball) * maxNumBalls;
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   
 | |
|   Ball* balls = reinterpret_cast<Ball*>(SEGENV.data);
 | |
|   
 | |
|   // number of balls based on intensity setting to max of 7 (cycles colors)
 | |
|   // non-chosen color is a random color
 | |
|   uint8_t numBalls = int(((SEGMENT.intensity * (maxNumBalls - 0.8f)) / 255) + 1);
 | |
|   
 | |
|   float gravity                           = -9.81; // standard value of gravity
 | |
|   float impactVelocityStart               = sqrt( -2 * gravity);
 | |
| 
 | |
|   unsigned long time = millis();
 | |
| 
 | |
|   if (SEGENV.call == 0) {
 | |
|     for (uint8_t i = 0; i < maxNumBalls; i++) balls[i].lastBounceTime = time;
 | |
|   }
 | |
|   
 | |
|   bool hasCol2 = SEGCOLOR(2);
 | |
|   fill(hasCol2 ? BLACK : SEGCOLOR(1));
 | |
|   
 | |
|   for (uint8_t i = 0; i < numBalls; i++) {
 | |
|     float timeSinceLastBounce = (time - balls[i].lastBounceTime)/((255-SEGMENT.speed)*8/256 +1);
 | |
|     balls[i].height = 0.5 * gravity * pow(timeSinceLastBounce/1000 , 2.0) + balls[i].impactVelocity * timeSinceLastBounce/1000;
 | |
| 
 | |
|     if (balls[i].height < 0) { //start bounce
 | |
|       balls[i].height = 0;
 | |
|       //damping for better effect using multiple balls
 | |
|       float dampening = 0.90 - float(i)/pow(numBalls,2);
 | |
|       balls[i].impactVelocity = dampening * balls[i].impactVelocity;
 | |
|       balls[i].lastBounceTime = time;
 | |
| 
 | |
|       if (balls[i].impactVelocity < 0.015) {
 | |
|         balls[i].impactVelocity = impactVelocityStart;
 | |
|       }
 | |
|     }
 | |
|     
 | |
|     uint32_t color = SEGCOLOR(0);
 | |
|     if (SEGMENT.palette) {
 | |
|       color = color_wheel(i*(256/MAX(numBalls, 8)));
 | |
|     } else if (hasCol2) {
 | |
|       color = SEGCOLOR(i % NUM_COLORS);
 | |
|     }
 | |
| 
 | |
|     uint16_t pos = round(balls[i].height * (SEGLEN - 1));
 | |
|     setPixelColor(pos, color);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_BOUNCINGBALLS PROGMEM = "Bouncing Balls@Gravity,# of balls;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
| * Sinelon stolen from FASTLED examples
 | |
| */
 | |
| uint16_t WS2812FX::sinelon_base(bool dual, bool rainbow=false) {
 | |
|   fade_out(SEGMENT.intensity);
 | |
|   uint16_t pos = beatsin16(SEGMENT.speed/10,0,SEGLEN-1);
 | |
|   if (SEGENV.call == 0) SEGENV.aux0 = pos;
 | |
|   uint32_t color1 = color_from_palette(pos, true, false, 0);
 | |
|   uint32_t color2 = SEGCOLOR(2);
 | |
|   if (rainbow) {
 | |
|     color1 = color_wheel((pos & 0x07) * 32);
 | |
|   }
 | |
|   setPixelColor(pos, color1);
 | |
|   if (dual) {
 | |
|     if (!color2) color2 = color_from_palette(pos, true, false, 0);
 | |
|     if (rainbow) color2 = color1; //rainbow
 | |
|     setPixelColor(SEGLEN-1-pos, color2);
 | |
|   }
 | |
|   if (SEGENV.aux0 != pos) { 
 | |
|     if (SEGENV.aux0 < pos) {
 | |
|       for (uint16_t i = SEGENV.aux0; i < pos ; i++) {
 | |
|         setPixelColor(i, color1);
 | |
|         if (dual) setPixelColor(SEGLEN-1-i, color2);
 | |
|       }
 | |
|     } else {
 | |
|       for (uint16_t i = SEGENV.aux0; i > pos ; i--) {
 | |
|         setPixelColor(i, color1);
 | |
|         if (dual) setPixelColor(SEGLEN-1-i, color2);
 | |
|       }
 | |
|     }
 | |
|     SEGENV.aux0 = pos;
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_sinelon(void) {
 | |
|   return sinelon_base(false);
 | |
| }
 | |
| static const char *_data_FX_MODE_SINELON PROGMEM = "Sinelon";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_sinelon_dual(void) {
 | |
|   return sinelon_base(true);
 | |
| }
 | |
| static const char *_data_FX_MODE_SINELON_DUAL PROGMEM = "Sinelon Dual";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_sinelon_rainbow(void) {
 | |
|   return sinelon_base(false, true);
 | |
| }
 | |
| static const char *_data_FX_MODE_SINELON_RAINBOW PROGMEM = "Sinelon Rainbow";
 | |
| 
 | |
| 
 | |
| //Rainbow with glitter, inspired by https://gist.github.com/kriegsman/062e10f7f07ba8518af6
 | |
| uint16_t WS2812FX::mode_glitter()
 | |
| {
 | |
|   mode_palette();
 | |
| 
 | |
|   if (isMatrix) {
 | |
|     uint16_t height = SEGMENT.virtualHeight();
 | |
|     uint16_t width  = SEGMENT.virtualWidth();
 | |
|     for (uint16_t i = 0; i<height; i++) {
 | |
|       if (SEGMENT.intensity > random8()) setPixelColorXY(random16(width-1), i, ULTRAWHITE);
 | |
|     }
 | |
|   } else
 | |
|     if (SEGMENT.intensity > random8()) setPixelColor(random16(SEGLEN), ULTRAWHITE);
 | |
|   
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_GLITTER PROGMEM = "Glitter@,!;!,!,!;!";
 | |
| 
 | |
| 
 | |
| //each needs 19 bytes
 | |
| //Spark type is used for popcorn, 1D fireworks, and drip
 | |
| typedef struct Spark {
 | |
|   float pos, posX;
 | |
|   float vel, velX;
 | |
|   uint16_t col;
 | |
|   uint8_t colIndex;
 | |
| } spark;
 | |
| 
 | |
| /*
 | |
| *  POPCORN
 | |
| *  modified from https://github.com/kitesurfer1404/WS2812FX/blob/master/src/custom/Popcorn.h
 | |
| */
 | |
| uint16_t WS2812FX::mode_popcorn(void) {
 | |
|   const uint16_t cols = isMatrix ? SEGMENT.virtualWidth() : 1;
 | |
|   const uint16_t rows = isMatrix ? SEGMENT.virtualHeight() : SEGMENT.virtualLength();
 | |
| 
 | |
|   //allocate segment data
 | |
|   uint16_t maxNumPopcorn = 21; // max 21 on 16 segment ESP8266
 | |
|   uint16_t dataSize = sizeof(spark) * maxNumPopcorn;
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   
 | |
|   Spark* popcorn = reinterpret_cast<Spark*>(SEGENV.data);
 | |
| 
 | |
|   float gravity = -0.0001 - (SEGMENT.speed/200000.0); // m/s/s
 | |
|   gravity *= rows; //SEGLEN
 | |
| 
 | |
|   bool hasCol2 = SEGCOLOR(2);
 | |
|   fill(hasCol2 ? BLACK : SEGCOLOR(1));
 | |
| 
 | |
|   uint8_t numPopcorn = SEGMENT.intensity*maxNumPopcorn/255;
 | |
|   if (numPopcorn == 0) numPopcorn = 1;
 | |
| 
 | |
|   for(uint8_t i = 0; i < numPopcorn; i++) {
 | |
|     if (popcorn[i].pos >= 0.0f) { // if kernel is active, update its position
 | |
|       popcorn[i].pos += popcorn[i].vel;
 | |
|       popcorn[i].vel += gravity;
 | |
|     } else { // if kernel is inactive, randomly pop it
 | |
|       if (random8() < 2) { // POP!!!
 | |
|         popcorn[i].pos = 0.01f;
 | |
|         popcorn[i].posX = random16(cols);
 | |
|         
 | |
|         uint16_t peakHeight = 128 + random8(128); //0-255
 | |
|         peakHeight = (peakHeight * (rows -1)) >> 8;
 | |
|         popcorn[i].vel = sqrt(-2.0 * gravity * peakHeight);
 | |
|         popcorn[i].velX = 0;
 | |
|         
 | |
|         if (SEGMENT.palette) {
 | |
|           popcorn[i].colIndex = random8();
 | |
|         } else {
 | |
|           byte col = random8(0, NUM_COLORS);
 | |
|           if (!hasCol2 || !SEGCOLOR(col)) col = 0;
 | |
|           popcorn[i].colIndex = col;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|     if (popcorn[i].pos >= 0.0f) { // draw now active popcorn (either active before or just popped)
 | |
|       uint32_t col = color_wheel(popcorn[i].colIndex);
 | |
|       if (!SEGMENT.palette && popcorn[i].colIndex < NUM_COLORS) col = SEGCOLOR(popcorn[i].colIndex);
 | |
|       
 | |
|       uint16_t ledIndex = popcorn[i].pos;
 | |
|       if (ledIndex < rows) {
 | |
|         if (isMatrix) setPixelColorXY(popcorn[i].posX, rows - 1 - ledIndex, col);
 | |
|         else          setPixelColor(ledIndex, col);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_POPCORN PROGMEM = "Popcorn";
 | |
| 
 | |
| 
 | |
| //values close to 100 produce 5Hz flicker, which looks very candle-y
 | |
| //Inspired by https://github.com/avanhanegem/ArduinoCandleEffectNeoPixel
 | |
| //and https://cpldcpu.wordpress.com/2016/01/05/reverse-engineering-a-real-candle/
 | |
| 
 | |
| uint16_t WS2812FX::candle(bool multi)
 | |
| {
 | |
|   if (multi)
 | |
|   {
 | |
|     //allocate segment data
 | |
|     uint16_t dataSize = (SEGLEN -1) *3; //max. 1365 pixels (ESP8266)
 | |
|     if (!SEGENV.allocateData(dataSize)) return candle(false); //allocation failed
 | |
|   }
 | |
| 
 | |
|   //max. flicker range controlled by intensity
 | |
|   uint8_t valrange = SEGMENT.intensity;
 | |
|   uint8_t rndval = valrange >> 1; //max 127
 | |
| 
 | |
|   //step (how much to move closer to target per frame) coarsely set by speed
 | |
|   uint8_t speedFactor = 4;
 | |
|   if (SEGMENT.speed > 252) { //epilepsy
 | |
|     speedFactor = 1;
 | |
|   } else if (SEGMENT.speed > 99) { //regular candle (mode called every ~25 ms, so 4 frames to have a new target every 100ms)
 | |
|     speedFactor = 2;
 | |
|   } else if (SEGMENT.speed > 49) { //slower fade
 | |
|     speedFactor = 3;
 | |
|   } //else 4 (slowest)
 | |
| 
 | |
|   uint16_t numCandles = (multi) ? SEGLEN : 1;
 | |
| 
 | |
|   for (uint16_t i = 0; i < numCandles; i++)
 | |
|   {
 | |
|     uint16_t d = 0; //data location
 | |
| 
 | |
|     uint8_t s = SEGENV.aux0, s_target = SEGENV.aux1, fadeStep = SEGENV.step;
 | |
|     if (i > 0) {
 | |
|       d = (i-1) *3;
 | |
|       s = SEGENV.data[d]; s_target = SEGENV.data[d+1]; fadeStep = SEGENV.data[d+2];
 | |
|     }
 | |
|     if (fadeStep == 0) { //init vals
 | |
|       s = 128; s_target = 130 + random8(4); fadeStep = 1;
 | |
|     }
 | |
| 
 | |
|     bool newTarget = false;
 | |
|     if (s_target > s) { //fade up
 | |
|       s = qadd8(s, fadeStep);
 | |
|       if (s >= s_target) newTarget = true;
 | |
|     } else {
 | |
|       s = qsub8(s, fadeStep);
 | |
|       if (s <= s_target) newTarget = true;
 | |
|     }
 | |
| 
 | |
|     if (newTarget) {
 | |
|       s_target = random8(rndval) + random8(rndval); //between 0 and rndval*2 -2 = 252
 | |
|       if (s_target < (rndval >> 1)) s_target = (rndval >> 1) + random8(rndval);
 | |
|       uint8_t offset = (255 - valrange);
 | |
|       s_target += offset;
 | |
| 
 | |
|       uint8_t dif = (s_target > s) ? s_target - s : s - s_target;
 | |
|     
 | |
|       fadeStep = dif >> speedFactor;
 | |
|       if (fadeStep == 0) fadeStep = 1;
 | |
|     }
 | |
| 
 | |
|      if (i > 0) {
 | |
|       setPixelColor(i, color_blend(SEGCOLOR(1), color_from_palette(i, true, PALETTE_SOLID_WRAP, 0), s));
 | |
| 
 | |
|       SEGENV.data[d] = s; SEGENV.data[d+1] = s_target; SEGENV.data[d+2] = fadeStep;
 | |
|     } else {
 | |
|       for (uint16_t j = 0; j < SEGLEN; j++) {
 | |
|         setPixelColor(j, color_blend(SEGCOLOR(1), color_from_palette(j, true, PALETTE_SOLID_WRAP, 0), s));
 | |
|       }
 | |
| 
 | |
|       SEGENV.aux0 = s; SEGENV.aux1 = s_target; SEGENV.step = fadeStep;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME_FIXED;
 | |
| }
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_candle()
 | |
| {
 | |
|   return candle(false);
 | |
| }
 | |
| static const char *_data_FX_MODE_CANDLE PROGMEM = "Candle@Flicker rate=96,Flicker intensity=224;!,!,;0";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_candle_multi()
 | |
| {
 | |
|   return candle(true);
 | |
| }
 | |
| static const char *_data_FX_MODE_CANDLE_MULTI PROGMEM = "Candle Multi@Flicker rate=96,Flicker intensity=224;!,!,;0";
 | |
| 
 | |
| 
 | |
| /*
 | |
| / Fireworks in starburst effect
 | |
| / based on the video: https://www.reddit.com/r/arduino/comments/c3sd46/i_made_this_fireworks_effect_for_my_led_strips/
 | |
| / Speed sets frequency of new starbursts, intensity is the intensity of the burst
 | |
| */
 | |
| #ifdef ESP8266
 | |
|   #define STARBURST_MAX_FRAG   8 //52 bytes / star
 | |
| #else
 | |
|   #define STARBURST_MAX_FRAG  10 //60 bytes / star
 | |
| #endif
 | |
| //each needs 20+STARBURST_MAX_FRAG*4 bytes
 | |
| typedef struct particle {
 | |
|   CRGB     color;
 | |
|   uint32_t birth  =0;
 | |
|   uint32_t last   =0;
 | |
|   float    vel    =0;
 | |
|   uint16_t pos    =-1;
 | |
|   float    fragment[STARBURST_MAX_FRAG];
 | |
| } star;
 | |
| 
 | |
| uint16_t WS2812FX::mode_starburst(void) {
 | |
|   uint16_t maxData = FAIR_DATA_PER_SEG; //ESP8266: 256 ESP32: 640
 | |
|   uint8_t segs = getActiveSegmentsNum();
 | |
|   if (segs <= (MAX_NUM_SEGMENTS /2)) maxData *= 2; //ESP8266: 512 if <= 8 segs ESP32: 1280 if <= 16 segs
 | |
|   if (segs <= (MAX_NUM_SEGMENTS /4)) maxData *= 2; //ESP8266: 1024 if <= 4 segs ESP32: 2560 if <= 8 segs
 | |
|   uint16_t maxStars = maxData / sizeof(star); //ESP8266: max. 4/9/19 stars/seg, ESP32: max. 10/21/42 stars/seg
 | |
| 
 | |
|   uint8_t numStars = 1 + (SEGLEN >> 3);
 | |
|   if (numStars > maxStars) numStars = maxStars;
 | |
|   uint16_t dataSize = sizeof(star) * numStars;
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   
 | |
|   uint32_t it = millis();
 | |
|   
 | |
|   star* stars = reinterpret_cast<star*>(SEGENV.data);
 | |
|   
 | |
|   float          maxSpeed                = 375.0f;  // Max velocity
 | |
|   float          particleIgnition        = 250.0f;  // How long to "flash"
 | |
|   float          particleFadeTime        = 1500.0f; // Fade out time
 | |
|      
 | |
|   for (int j = 0; j < numStars; j++)
 | |
|   {
 | |
|     // speed to adjust chance of a burst, max is nearly always.
 | |
|     if (random8((144-(SEGMENT.speed >> 1))) == 0 && stars[j].birth == 0)
 | |
|     {
 | |
|       // Pick a random color and location.  
 | |
|       uint16_t startPos = random16(SEGLEN-1);
 | |
|       float multiplier = (float)(random8())/255.0 * 1.0;
 | |
| 
 | |
|       stars[j].color = col_to_crgb(color_wheel(random8()));
 | |
|       stars[j].pos = startPos; 
 | |
|       stars[j].vel = maxSpeed * (float)(random8())/255.0 * multiplier;
 | |
|       stars[j].birth = it;
 | |
|       stars[j].last = it;
 | |
|       // more fragments means larger burst effect
 | |
|       int num = random8(3,6 + (SEGMENT.intensity >> 5));
 | |
| 
 | |
|       for (int i=0; i < STARBURST_MAX_FRAG; i++) {
 | |
|         if (i < num) stars[j].fragment[i] = startPos;
 | |
|         else stars[j].fragment[i] = -1;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   
 | |
|   fill(SEGCOLOR(1));
 | |
|   
 | |
|   for (int j=0; j<numStars; j++)
 | |
|   {
 | |
|     if (stars[j].birth != 0) {
 | |
|       float dt = (it-stars[j].last)/1000.0;
 | |
| 
 | |
|       for (int i=0; i < STARBURST_MAX_FRAG; i++) {
 | |
|         int var = i >> 1;
 | |
|         
 | |
|         if (stars[j].fragment[i] > 0) {
 | |
|           //all fragments travel right, will be mirrored on other side
 | |
|           stars[j].fragment[i] += stars[j].vel * dt * (float)var/3.0;
 | |
|         }
 | |
|       }
 | |
|       stars[j].last = it;
 | |
|       stars[j].vel -= 3*stars[j].vel*dt;
 | |
|     }
 | |
|   
 | |
|     CRGB c = stars[j].color;
 | |
| 
 | |
|     // If the star is brand new, it flashes white briefly.  
 | |
|     // Otherwise it just fades over time.
 | |
|     float fade = 0.0f;
 | |
|     float age = it-stars[j].birth;
 | |
| 
 | |
|     if (age < particleIgnition) {
 | |
|       c = col_to_crgb(color_blend(WHITE, crgb_to_col(c), 254.5f*((age / particleIgnition))));
 | |
|     } else {
 | |
|       // Figure out how much to fade and shrink the star based on 
 | |
|       // its age relative to its lifetime
 | |
|       if (age > particleIgnition + particleFadeTime) {
 | |
|         fade = 1.0f;                  // Black hole, all faded out
 | |
|         stars[j].birth = 0;
 | |
|         c = col_to_crgb(SEGCOLOR(1));
 | |
|       } else {
 | |
|         age -= particleIgnition;
 | |
|         fade = (age / particleFadeTime);  // Fading star
 | |
|         byte f = 254.5f*fade;
 | |
|         c = col_to_crgb(color_blend(crgb_to_col(c), SEGCOLOR(1), f));
 | |
|       }
 | |
|     }
 | |
|     
 | |
|     float particleSize = (1.0 - fade) * 2;
 | |
| 
 | |
|     for (uint8_t index=0; index < STARBURST_MAX_FRAG*2; index++) {
 | |
|       bool mirrored = index & 0x1;
 | |
|       uint8_t i = index >> 1;
 | |
|       if (stars[j].fragment[i] > 0) {
 | |
|         float loc = stars[j].fragment[i];
 | |
|         if (mirrored) loc -= (loc-stars[j].pos)*2;
 | |
|         int start = loc - particleSize;
 | |
|         int end = loc + particleSize;
 | |
|         if (start < 0) start = 0;
 | |
|         if (start == end) end++;
 | |
|         if (end > SEGLEN) end = SEGLEN;    
 | |
|         for (int p = start; p < end; p++) {
 | |
|           setPixelColor(p, c.r, c.g, c.b);
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| #undef STARBURST_MAX_FRAG
 | |
| static const char *_data_FX_MODE_STARBURST PROGMEM = "Fireworks Starburst";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Exploding fireworks effect
 | |
|  * adapted from: http://www.anirama.com/1000leds/1d-fireworks/
 | |
|  * adapted for 2D WLED by blazoncek (Blaz Kristan)
 | |
|  */
 | |
| uint16_t WS2812FX::mode_exploding_fireworks(void)
 | |
| {
 | |
|   const uint16_t cols = isMatrix ? SEGMENT.virtualWidth() : 1;
 | |
|   const uint16_t rows = isMatrix ? SEGMENT.virtualHeight() : SEGMENT.virtualLength();
 | |
| 
 | |
|   //allocate segment data
 | |
|   uint16_t maxData = FAIR_DATA_PER_SEG; //ESP8266: 256 ESP32: 640
 | |
|   uint8_t segs = getActiveSegmentsNum();
 | |
|   if (segs <= (MAX_NUM_SEGMENTS /2)) maxData *= 2; //ESP8266: 512 if <= 8 segs ESP32: 1280 if <= 16 segs
 | |
|   if (segs <= (MAX_NUM_SEGMENTS /4)) maxData *= 2; //ESP8266: 1024 if <= 4 segs ESP32: 2560 if <= 8 segs
 | |
|   int maxSparks = maxData / sizeof(spark); //ESP8266: max. 21/42/85 sparks/seg, ESP32: max. 53/106/213 sparks/seg
 | |
| 
 | |
|   uint16_t numSparks = min(2 + ((rows*cols) >> 1), maxSparks);
 | |
|   uint16_t dataSize = sizeof(spark) * numSparks;
 | |
|   if (!SEGENV.allocateData(dataSize + sizeof(float))) return mode_static(); //allocation failed
 | |
|   float *dying_gravity = reinterpret_cast<float*>(SEGENV.data + dataSize);
 | |
| 
 | |
|   if (dataSize != SEGENV.aux1) { //reset to flare if sparks were reallocated (it may be good idea to reset segment if bounds change)
 | |
|     *dying_gravity = 0.0f;
 | |
|     SEGENV.aux0 = 0;
 | |
|     SEGENV.aux1 = dataSize;
 | |
|   }
 | |
| 
 | |
|   //fill(BLACK);
 | |
|   fade_out(252);
 | |
|   
 | |
|   Spark* sparks = reinterpret_cast<Spark*>(SEGENV.data);
 | |
|   Spark* flare = sparks; //first spark is flare data
 | |
| 
 | |
|   float gravity = -0.0004 - (SEGMENT.speed/800000.0); // m/s/s
 | |
|   gravity *= rows;
 | |
|   
 | |
|   if (SEGENV.aux0 < 2) { //FLARE
 | |
|     if (SEGENV.aux0 == 0) { //init flare
 | |
|       flare->pos = 0;
 | |
|       flare->posX = isMatrix ? random16(2,cols-1) : (SEGMENT.intensity > random8()); // will enable random firing side on 1D
 | |
|       uint16_t peakHeight = 75 + random8(180); //0-255
 | |
|       peakHeight = (peakHeight * (rows -1)) >> 8;
 | |
|       flare->vel = sqrt(-2.0 * gravity * peakHeight);
 | |
|       flare->velX = isMatrix ? (random8(8)-4)/32.f : 0; // no X velocity on 1D
 | |
|       flare->col = 255; //brightness
 | |
|       SEGENV.aux0 = 1; 
 | |
|     }
 | |
|     
 | |
|     // launch 
 | |
|     if (flare->vel > 12 * gravity) {
 | |
|       // flare
 | |
|       if (isMatrix) setPixelColorXY(flare->posX, rows - uint16_t(flare->pos) - 1, flare->col, flare->col, flare->col);
 | |
|       else          setPixelColor(int(flare->posX) ? rows - int(flare->pos) - 1 : int(flare->pos), flare->col, flare->col, flare->col);
 | |
|       flare->pos  += flare->vel;
 | |
|       flare->posX += flare->velX;
 | |
|       flare->pos  = constrain(flare->pos, 0, rows-1);
 | |
|       flare->posX = constrain(flare->posX, 0, cols-isMatrix);
 | |
|       flare->vel  += gravity;
 | |
|       flare->col  -= 2;
 | |
|     } else {
 | |
|       SEGENV.aux0 = 2;  // ready to explode
 | |
|     }
 | |
|   } else if (SEGENV.aux0 < 4) {
 | |
|     /*
 | |
|      * Explode!
 | |
|      * 
 | |
|      * Explosion happens where the flare ended.
 | |
|      * Size is proportional to the height.
 | |
|      */
 | |
|     int nSparks = flare->pos + random8(4);
 | |
|     nSparks = constrain(nSparks, 1, numSparks);
 | |
|   
 | |
|     // initialize sparks
 | |
|     if (SEGENV.aux0 == 2) {
 | |
|       for (int i = 1; i < nSparks; i++) { 
 | |
|         sparks[i].pos  = flare->pos;
 | |
|         sparks[i].posX = flare->posX;
 | |
|         sparks[i].vel  = (float(random16(0, 20000)) / 10000.0f) - 0.9f; // from -0.9 to 1.1
 | |
|         sparks[i].vel *= rows<32 ? 0.5 : 1; // reduce velocity for smaller strips
 | |
|         sparks[i].velX = isMatrix ? (float(random16(0, 4000)) / 10000.0f) - 0.2f : 0; // from -0.2 to 0.2
 | |
|         sparks[i].col  = 345;//abs(sparks[i].vel * 750.0); // set colors before scaling velocity to keep them bright 
 | |
|         //sparks[i].col = constrain(sparks[i].col, 0, 345); 
 | |
|         sparks[i].colIndex = random8();
 | |
|         sparks[i].vel  *= flare->pos/rows; // proportional to height 
 | |
|         sparks[i].velX *= isMatrix ? flare->posX/cols : 0; // proportional to width
 | |
|         sparks[i].vel  *= -gravity *50;
 | |
|       } 
 | |
|       //sparks[1].col = 345; // this will be our known spark 
 | |
|       *dying_gravity = gravity/2; 
 | |
|       SEGENV.aux0 = 3;
 | |
|     }
 | |
|   
 | |
|     if (sparks[1].col > 4) {//&& sparks[1].pos > 0) { // as long as our known spark is lit, work with all the sparks
 | |
|       for (int i = 1; i < nSparks; i++) {
 | |
|         sparks[i].pos  += sparks[i].vel;
 | |
|         sparks[i].posX += sparks[i].velX;
 | |
|         sparks[i].vel  += *dying_gravity;
 | |
|         sparks[i].velX += isMatrix ? *dying_gravity : 0;
 | |
|         if (sparks[i].col > 3) sparks[i].col -= 4; 
 | |
| 
 | |
|         if (sparks[i].pos > 0 && sparks[i].pos < rows) {
 | |
|           if (isMatrix && !(sparks[i].posX >= 0 && sparks[i].posX < cols)) continue;
 | |
|           uint16_t prog = sparks[i].col;
 | |
|           uint32_t spColor = (SEGMENT.palette) ? color_wheel(sparks[i].colIndex) : SEGCOLOR(0);
 | |
|           CRGB c = CRGB::Black; //HeatColor(sparks[i].col);
 | |
|           if (prog > 300) { //fade from white to spark color
 | |
|             c = col_to_crgb(color_blend(spColor, WHITE, (prog - 300)*5));
 | |
|           } else if (prog > 45) { //fade from spark color to black
 | |
|             c = col_to_crgb(color_blend(BLACK, spColor, prog - 45));
 | |
|             uint8_t cooling = (300 - prog) >> 5;
 | |
|             c.g = qsub8(c.g, cooling);
 | |
|             c.b = qsub8(c.b, cooling * 2);
 | |
|           }
 | |
|           if (isMatrix) setPixelColorXY(sparks[i].posX, rows - int(sparks[i].pos) - 1, c.red, c.green, c.blue);
 | |
|           else          setPixelColor(int(sparks[i].posX) ? rows - int(sparks[i].pos) - 1 : int(sparks[i].pos), c.red, c.green, c.blue);
 | |
|         }
 | |
|       }
 | |
|       blur(16);
 | |
|       *dying_gravity *= .8; // as sparks burn out they fall slower
 | |
|     } else {
 | |
|       SEGENV.aux0 = 6 + random8(10); //wait for this many frames
 | |
|     }
 | |
|   } else {
 | |
|     SEGENV.aux0--;
 | |
|     if (SEGENV.aux0 < 4) {
 | |
|       SEGENV.aux0 = 0; //back to flare
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;  
 | |
| }
 | |
| #undef MAX_SPARKS
 | |
| static const char *_data_FX_MODE_EXPLODING_FIREWORKS PROGMEM = "Fireworks 1D@Gravity,Firing side;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Drip Effect
 | |
|  * ported of: https://www.youtube.com/watch?v=sru2fXh4r7k
 | |
|  */
 | |
| uint16_t WS2812FX::mode_drip(void)
 | |
| {
 | |
|   const uint16_t cols = isMatrix ? SEGMENT.virtualWidth() : 1;
 | |
|   const uint16_t rows = isMatrix ? SEGMENT.virtualHeight() : SEGMENT.virtualLength();
 | |
| 
 | |
|   //allocate segment data
 | |
|   uint8_t numDrops = 4; 
 | |
|   uint16_t dataSize = sizeof(spark) * numDrops;
 | |
|   if (!SEGENV.allocateData(dataSize * cols)) return mode_static(); //allocation failed
 | |
| 
 | |
|   fill(SEGCOLOR(1));
 | |
|   
 | |
|   Spark* drops = reinterpret_cast<Spark*>(SEGENV.data);
 | |
| 
 | |
|   numDrops = 1 + (SEGMENT.intensity >> 6); // 255>>6 = 3
 | |
| 
 | |
|   float gravity = -0.0005 - (SEGMENT.speed/50000.0);
 | |
|   gravity *= rows-1;
 | |
|   int sourcedrop = 12;
 | |
| 
 | |
|   for (uint16_t k=0; k < cols; k++) {
 | |
|     for (uint8_t j=0; j < numDrops; j++) {
 | |
|       uint16_t idx = k*numDrops + j;
 | |
| 
 | |
|       if (drops[idx].colIndex == 0) { //init
 | |
|         drops[idx].pos = rows-1;       // start at end
 | |
|         drops[idx].vel = 0;           // speed
 | |
|         drops[idx].col = sourcedrop;  // brightness
 | |
|         drops[idx].colIndex = 1;      // drop state (0 init, 1 forming, 2 falling, 5 bouncing) 
 | |
|       }
 | |
|       
 | |
|       uint32_t col = color_blend(BLACK, SEGCOLOR(0), sourcedrop);
 | |
|       if (isMatrix) setPixelColorXY(k, 0, col);
 | |
|       else          setPixelColor(rows-1, col);// water source
 | |
| 
 | |
|       if (drops[idx].colIndex == 1) {
 | |
|         if (drops[idx].col > 255) drops[idx].col = 255;
 | |
|         col = color_blend(BLACK,SEGCOLOR(0),drops[idx].col);
 | |
|         if (isMatrix) setPixelColorXY(k, rows - 1 - uint16_t(drops[idx].pos), col);
 | |
|         else          setPixelColor(uint16_t(drops[idx].pos), col);
 | |
|         
 | |
|         drops[idx].col += map(SEGMENT.speed, 0, 255, 1, 6); // swelling
 | |
|         
 | |
|         if (random8() < drops[idx].col/10) {   // random drop
 | |
|           drops[idx].colIndex = 2;             //fall
 | |
|           drops[idx].col = 255;
 | |
|         }
 | |
|       }  
 | |
|       if (drops[idx].colIndex > 1) {           // falling
 | |
|         if (drops[idx].pos > 0) {              // fall until end of segment
 | |
|           drops[idx].pos += drops[idx].vel;
 | |
|           if (drops[idx].pos < 0) drops[idx].pos = 0;
 | |
|           drops[idx].vel += gravity;           // gravity is negative
 | |
| 
 | |
|           for (uint16_t i = 1; i < 7 - drops[idx].colIndex; i++) { // some minor math so we don't expand bouncing droplets
 | |
|             uint16_t pos = constrain(uint16_t(drops[idx].pos) +i, 0, rows-1); //this is BAD, returns a pos >= SEGLEN occasionally
 | |
|             col = color_blend(BLACK, SEGCOLOR(0), drops[idx].col/i);
 | |
|             if (isMatrix) setPixelColorXY(k, rows - 1 - pos, col);
 | |
|             else          setPixelColor(pos, col); //spread pixel with fade while falling
 | |
|           }
 | |
| 
 | |
|           if (drops[idx].colIndex > 2) {         // during bounce, some water is on the floor
 | |
|             col = color_blend(SEGCOLOR(0), BLACK, drops[idx].col);
 | |
|             if (isMatrix) setPixelColorXY(k, rows - 1, col);
 | |
|             else          setPixelColor(0, col);
 | |
|           }
 | |
|         } else {                                 // we hit bottom
 | |
|           if (drops[idx].colIndex > 2) {         // already hit once, so back to forming
 | |
|             drops[idx].colIndex = 0;
 | |
|             drops[idx].col = sourcedrop;
 | |
|             
 | |
|           } else {
 | |
| 
 | |
|             if (drops[idx].colIndex == 2) {      // init bounce
 | |
|               drops[idx].vel = -drops[idx].vel/4;// reverse velocity with damping 
 | |
|               drops[idx].pos += drops[idx].vel;
 | |
|             } 
 | |
|             drops[idx].col = sourcedrop*2;
 | |
|             drops[idx].colIndex = 5;             // bouncing
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_DRIP PROGMEM = "Drip@Gravity,# of drips;!,!;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Tetris or Stacking (falling bricks) Effect
 | |
|  * by Blaz Kristan (https://github.com/blazoncek, https://blaz.at/home)
 | |
|  */
 | |
| //12 bytes
 | |
| typedef struct Tetris {
 | |
|   float    pos;
 | |
|   float    speed;
 | |
|   uint32_t col;
 | |
| } tetris;
 | |
| 
 | |
| uint16_t WS2812FX::mode_tetrix(void) {
 | |
| 
 | |
|   uint16_t dataSize = sizeof(tetris);
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   Tetris* drop = reinterpret_cast<Tetris*>(SEGENV.data);
 | |
| 
 | |
|   // initialize dropping on first call or segment full
 | |
|   if (SEGENV.call == 0 || SEGENV.aux1 >= SEGLEN) {
 | |
|     SEGENV.aux1 = 0;                            // reset brick stack size
 | |
|     SEGENV.step = 0;
 | |
|     fill(SEGCOLOR(1));
 | |
|     return 250;  // short wait
 | |
|   }
 | |
|   
 | |
|   if (SEGENV.step == 0) {             //init
 | |
|     drop->speed = 0.0238 * (SEGMENT.speed ? (SEGMENT.speed>>2)+1 : random8(6,64)); // set speed
 | |
|     drop->pos   = SEGLEN;             // start at end of segment (no need to subtract 1)
 | |
|     drop->col   = color_from_palette(random8(0,15)<<4,false,false,0);     // limit color choices so there is enough HUE gap
 | |
|     SEGENV.step = 1;                  // drop state (0 init, 1 forming, 2 falling)
 | |
|     SEGENV.aux0 = (SEGMENT.intensity ? (SEGMENT.intensity>>5)+1 : random8(1,5)) * (1+(SEGLEN>>6));  // size of brick
 | |
|   }
 | |
|   
 | |
|   if (SEGENV.step == 1) {             // forming
 | |
|     if (random8()>>6) {               // random drop
 | |
|       SEGENV.step = 2;                // fall
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (SEGENV.step > 1) {              // falling
 | |
|     if (drop->pos > SEGENV.aux1) {    // fall until top of stack
 | |
|       drop->pos -= drop->speed;       // may add gravity as: speed += gravity
 | |
|       if (int(drop->pos) < SEGENV.aux1) drop->pos = SEGENV.aux1;
 | |
|       for (uint16_t i=int(drop->pos); i<SEGLEN; i++) setPixelColor(i,i<int(drop->pos)+SEGENV.aux0 ? drop->col : SEGCOLOR(1));
 | |
|     } else {                          // we hit bottom
 | |
|       SEGENV.step = 0;                // go back to init
 | |
|       SEGENV.aux1 += SEGENV.aux0;     // increase the stack size
 | |
|       if (SEGENV.aux1 >= SEGLEN) return 1000;   // wait for a second
 | |
|     }
 | |
|   }
 | |
|   return FRAMETIME;  
 | |
| }
 | |
| static const char *_data_FX_MODE_TETRIX PROGMEM = "Tetrix@!=224,Width=0;!,!,;!=11";
 | |
| 
 | |
| 
 | |
| /*
 | |
| / Plasma Effect
 | |
| / adapted from https://github.com/atuline/FastLED-Demos/blob/master/plasma/plasma.ino
 | |
| */
 | |
| uint16_t WS2812FX::mode_plasma(void) {
 | |
|   // initialize phases on start
 | |
|   if (SEGENV.call == 0) {
 | |
|     SEGENV.aux0 = random8(0,2);  // add a bit of randomness
 | |
|   }
 | |
|   uint8_t thisPhase = beatsin8(6+SEGENV.aux0,-64,64);
 | |
|   uint8_t thatPhase = beatsin8(7+SEGENV.aux0,-64,64);
 | |
| 
 | |
|   for (int i = 0; i < SEGLEN; i++) {   // For each of the LED's in the strand, set color &  brightness based on a wave as follows:
 | |
|     uint8_t colorIndex = cubicwave8((i*(2+ 3*(SEGMENT.speed >> 5))+thisPhase) & 0xFF)/2   // factor=23 // Create a wave and add a phase change and add another wave with its own phase change.
 | |
|                              + cos8((i*(1+ 2*(SEGMENT.speed >> 5))+thatPhase) & 0xFF)/2;  // factor=15 // Hey, you can even change the frequencies if you wish.
 | |
|     uint8_t thisBright = qsub8(colorIndex, beatsin8(7,0, (128 - (SEGMENT.intensity>>1))));
 | |
|     CRGB color = ColorFromPalette(currentPalette, colorIndex, thisBright, LINEARBLEND);
 | |
|     setPixelColor(i, color.red, color.green, color.blue);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } 
 | |
| static const char *_data_FX_MODE_PLASMA PROGMEM = "Plasma@Phase,;1,2,3;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Percentage display
 | |
|  * Intesity values from 0-100 turn on the leds.
 | |
|  */
 | |
| uint16_t WS2812FX::mode_percent(void) {
 | |
| 
 | |
| 	uint8_t percent = MAX(0, MIN(200, SEGMENT.intensity));
 | |
| 	uint16_t active_leds = (percent < 100) ? SEGLEN * percent / 100.0
 | |
|                                          : SEGLEN * (200 - percent) / 100.0;
 | |
|   
 | |
|   uint8_t size = (1 + ((SEGMENT.speed * SEGLEN) >> 11));
 | |
|   if (SEGMENT.speed == 255) size = 255;
 | |
|     
 | |
|   if (percent < 100) {
 | |
|     for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
| 	  	if (i < SEGENV.step) {
 | |
|         setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 0));
 | |
| 	  	}
 | |
| 	  	else {
 | |
|         setPixelColor(i, SEGCOLOR(1));
 | |
| 	  	}
 | |
| 	  }
 | |
|   } else {
 | |
|     for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
| 	  	if (i < (SEGLEN - SEGENV.step)) {
 | |
|         setPixelColor(i, SEGCOLOR(1));
 | |
| 	  	}
 | |
| 	  	else {
 | |
|         setPixelColor(i, color_from_palette(i, true, PALETTE_SOLID_WRAP, 0));
 | |
| 	  	}
 | |
| 	  }
 | |
|   }
 | |
| 
 | |
|   if(active_leds > SEGENV.step) {  // smooth transition to the target value
 | |
|     SEGENV.step += size;
 | |
|     if (SEGENV.step > active_leds) SEGENV.step = active_leds;
 | |
|   } else if (active_leds < SEGENV.step) {
 | |
|     if (SEGENV.step > size) SEGENV.step -= size; else SEGENV.step = 0;
 | |
|     if (SEGENV.step < active_leds) SEGENV.step = active_leds;
 | |
|   }
 | |
| 
 | |
|  	return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_PERCENT PROGMEM = "Percent@,% of fill;!,!,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Modulates the brightness similar to a heartbeat
 | |
|  * tries to draw an ECG aproximation on a 2D matrix
 | |
|  */
 | |
| uint16_t WS2812FX::mode_heartbeat(void) {
 | |
|   uint8_t bpm = 40 + (SEGMENT.speed >> 3);
 | |
|   uint32_t msPerBeat = (60000L / bpm);
 | |
|   uint32_t secondBeat = (msPerBeat / 3);
 | |
|   uint32_t bri_lower = SEGENV.aux1;
 | |
|   unsigned long beatTimer = now - SEGENV.step;
 | |
| 
 | |
|   bri_lower = bri_lower * 2042 / (2048 + SEGMENT.intensity);
 | |
|   SEGENV.aux1 = bri_lower;
 | |
| 
 | |
|   if ((beatTimer > secondBeat) && !SEGENV.aux0) { // time for the second beat?
 | |
|     SEGENV.aux1 = isMatrix ? UINT16_MAX*3L/4 : UINT16_MAX; //3/4 bri
 | |
|     SEGENV.aux0 = 1;
 | |
|   }
 | |
|   if (beatTimer > msPerBeat) { // time to reset the beat timer?
 | |
|     SEGENV.aux1 = UINT16_MAX; //full bri
 | |
|     SEGENV.aux0 = 0;
 | |
|     SEGENV.step = now;
 | |
|   }
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     setPixelColor(i, color_blend(color_from_palette(i, true, PALETTE_SOLID_WRAP, 0), SEGCOLOR(1), 255 - (SEGENV.aux1 >> 8)));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_HEARTBEAT PROGMEM = "Heartbeat@!,!;!,!,;!";
 | |
| 
 | |
| 
 | |
| //  "Pacifica"
 | |
| //  Gentle, blue-green ocean waves.
 | |
| //  December 2019, Mark Kriegsman and Mary Corey March.
 | |
| //  For Dan.
 | |
| //
 | |
| //
 | |
| // In this animation, there are four "layers" of waves of light.  
 | |
| //
 | |
| // Each layer moves independently, and each is scaled separately.
 | |
| //
 | |
| // All four wave layers are added together on top of each other, and then 
 | |
| // another filter is applied that adds "whitecaps" of brightness where the 
 | |
| // waves line up with each other more.  Finally, another pass is taken
 | |
| // over the led array to 'deepen' (dim) the blues and greens.
 | |
| //
 | |
| // The speed and scale and motion each layer varies slowly within independent 
 | |
| // hand-chosen ranges, which is why the code has a lot of low-speed 'beatsin8' functions
 | |
| // with a lot of oddly specific numeric ranges.
 | |
| //
 | |
| // These three custom blue-green color palettes were inspired by the colors found in
 | |
| // the waters off the southern coast of California, https://goo.gl/maps/QQgd97jjHesHZVxQ7
 | |
| //
 | |
| // Modified for WLED, based on https://github.com/FastLED/FastLED/blob/master/examples/Pacifica/Pacifica.ino
 | |
| //
 | |
| uint16_t WS2812FX::mode_pacifica()
 | |
| {
 | |
|   uint32_t nowOld = now;
 | |
| 
 | |
|   CRGBPalette16 pacifica_palette_1 = 
 | |
|     { 0x000507, 0x000409, 0x00030B, 0x00030D, 0x000210, 0x000212, 0x000114, 0x000117, 
 | |
|       0x000019, 0x00001C, 0x000026, 0x000031, 0x00003B, 0x000046, 0x14554B, 0x28AA50 };
 | |
|   CRGBPalette16 pacifica_palette_2 = 
 | |
|     { 0x000507, 0x000409, 0x00030B, 0x00030D, 0x000210, 0x000212, 0x000114, 0x000117, 
 | |
|       0x000019, 0x00001C, 0x000026, 0x000031, 0x00003B, 0x000046, 0x0C5F52, 0x19BE5F };
 | |
|   CRGBPalette16 pacifica_palette_3 = 
 | |
|     { 0x000208, 0x00030E, 0x000514, 0x00061A, 0x000820, 0x000927, 0x000B2D, 0x000C33, 
 | |
|       0x000E39, 0x001040, 0x001450, 0x001860, 0x001C70, 0x002080, 0x1040BF, 0x2060FF };
 | |
| 
 | |
|   if (SEGMENT.palette) {
 | |
|     pacifica_palette_1 = currentPalette;
 | |
|     pacifica_palette_2 = currentPalette;
 | |
|     pacifica_palette_3 = currentPalette;
 | |
|   }
 | |
| 
 | |
|   // Increment the four "color index start" counters, one for each wave layer.
 | |
|   // Each is incremented at a different speed, and the speeds vary over time.
 | |
|   uint16_t sCIStart1 = SEGENV.aux0, sCIStart2 = SEGENV.aux1, sCIStart3 = SEGENV.step, sCIStart4 = SEGENV.step >> 16;
 | |
|   uint32_t deltams = (FRAMETIME >> 2) + ((FRAMETIME * SEGMENT.speed) >> 7);
 | |
|   uint64_t deltat = (now >> 2) + ((now * SEGMENT.speed) >> 7);
 | |
|   now = deltat;
 | |
| 
 | |
|   uint16_t speedfactor1 = beatsin16(3, 179, 269);
 | |
|   uint16_t speedfactor2 = beatsin16(4, 179, 269);
 | |
|   uint32_t deltams1 = (deltams * speedfactor1) / 256;
 | |
|   uint32_t deltams2 = (deltams * speedfactor2) / 256;
 | |
|   uint32_t deltams21 = (deltams1 + deltams2) / 2;
 | |
|   sCIStart1 += (deltams1 * beatsin88(1011,10,13));
 | |
|   sCIStart2 -= (deltams21 * beatsin88(777,8,11));
 | |
|   sCIStart3 -= (deltams1 * beatsin88(501,5,7));
 | |
|   sCIStart4 -= (deltams2 * beatsin88(257,4,6));
 | |
|   SEGENV.aux0 = sCIStart1; SEGENV.aux1 = sCIStart2;
 | |
|   SEGENV.step = sCIStart4; SEGENV.step = (SEGENV.step << 16) + sCIStart3;
 | |
| 
 | |
|   // Clear out the LED array to a dim background blue-green
 | |
|   //fill(132618);
 | |
| 
 | |
|   uint8_t basethreshold = beatsin8( 9, 55, 65);
 | |
|   uint8_t wave = beat8( 7 );
 | |
|   
 | |
|   for( uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     CRGB c = CRGB(2, 6, 10);
 | |
|     // Render each of four layers, with different scales and speeds, that vary over time
 | |
|     c += pacifica_one_layer(i, pacifica_palette_1, sCIStart1, beatsin16(3, 11 * 256, 14 * 256), beatsin8(10, 70, 130), 0-beat16(301));
 | |
|     c += pacifica_one_layer(i, pacifica_palette_2, sCIStart2, beatsin16(4,  6 * 256,  9 * 256), beatsin8(17, 40,  80),   beat16(401));
 | |
|     c += pacifica_one_layer(i, pacifica_palette_3, sCIStart3,                         6 * 256 , beatsin8(9, 10,38)   , 0-beat16(503));
 | |
|     c += pacifica_one_layer(i, pacifica_palette_3, sCIStart4,                         5 * 256 , beatsin8(8, 10,28)   ,   beat16(601));
 | |
|     
 | |
|     // Add extra 'white' to areas where the four layers of light have lined up brightly
 | |
|     uint8_t threshold = scale8( sin8( wave), 20) + basethreshold;
 | |
|     wave += 7;
 | |
|     uint8_t l = c.getAverageLight();
 | |
|     if (l > threshold) {
 | |
|       uint8_t overage = l - threshold;
 | |
|       uint8_t overage2 = qadd8(overage, overage);
 | |
|       c += CRGB(overage, overage2, qadd8(overage2, overage2));
 | |
|     }
 | |
| 
 | |
|     //deepen the blues and greens
 | |
|     c.blue  = scale8(c.blue,  145); 
 | |
|     c.green = scale8(c.green, 200); 
 | |
|     c |= CRGB( 2, 5, 7);
 | |
| 
 | |
|     setPixelColor(i, c.red, c.green, c.blue);
 | |
|   }
 | |
| 
 | |
|   now = nowOld;
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_PACIFICA PROGMEM = "Pacifica";
 | |
| 
 | |
| 
 | |
| // Add one layer of waves into the led array
 | |
| CRGB WS2812FX::pacifica_one_layer(uint16_t i, CRGBPalette16& p, uint16_t cistart, uint16_t wavescale, uint8_t bri, uint16_t ioff)
 | |
| {
 | |
|   uint16_t ci = cistart;
 | |
|   uint16_t waveangle = ioff;
 | |
|   uint16_t wavescale_half = (wavescale >> 1) + 20;
 | |
|   
 | |
|   waveangle += ((120 + SEGMENT.intensity) * i); //original 250 * i
 | |
|   uint16_t s16 = sin16(waveangle) + 32768;
 | |
|   uint16_t cs = scale16(s16, wavescale_half) + wavescale_half;
 | |
|   ci += (cs * i);
 | |
|   uint16_t sindex16 = sin16(ci) + 32768;
 | |
|   uint8_t sindex8 = scale16(sindex16, 240);
 | |
|   return ColorFromPalette(p, sindex8, bri, LINEARBLEND);
 | |
| }
 | |
| 
 | |
| 
 | |
| //Solid colour background with glitter
 | |
| uint16_t WS2812FX::mode_solid_glitter()
 | |
| {
 | |
|   fill(SEGCOLOR(0));
 | |
| 
 | |
|   if (isMatrix) {
 | |
|     uint16_t height = SEGMENT.virtualHeight();
 | |
|     uint16_t width = SEGMENT.virtualWidth();
 | |
|     for (uint16_t i = 0; i<height; i++) {
 | |
|       if (SEGMENT.intensity > random8()) setPixelColorXY(random16(width-1), i, ULTRAWHITE);
 | |
|     }
 | |
|   } else
 | |
|     if (SEGMENT.intensity > random8()) setPixelColor(random16(SEGLEN), ULTRAWHITE);
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_SOLID_GLITTER PROGMEM = "Solid Glitter@,!;!,,;0";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Mode simulates a gradual sunrise
 | |
|  */
 | |
| uint16_t WS2812FX::mode_sunrise() {
 | |
|   //speed 0 - static sun
 | |
|   //speed 1 - 60: sunrise time in minutes
 | |
|   //speed 60 - 120 : sunset time in minutes - 60;
 | |
|   //speed above: "breathing" rise and set
 | |
|   if (SEGENV.call == 0 || SEGMENT.speed != SEGENV.aux0) {
 | |
| 	  SEGENV.step = millis(); //save starting time, millis() because now can change from sync
 | |
|     SEGENV.aux0 = SEGMENT.speed;
 | |
|   }
 | |
|   
 | |
|   fill(0);
 | |
|   uint16_t stage = 0xFFFF;
 | |
|   
 | |
|   uint32_t s10SinceStart = (millis() - SEGENV.step) /100; //tenths of seconds
 | |
|   
 | |
|   if (SEGMENT.speed > 120) { //quick sunrise and sunset
 | |
| 	  uint16_t counter = (now >> 1) * (((SEGMENT.speed -120) >> 1) +1);
 | |
| 	  stage = triwave16(counter);
 | |
|   } else if (SEGMENT.speed) { //sunrise
 | |
| 	  uint8_t durMins = SEGMENT.speed;
 | |
| 	  if (durMins > 60) durMins -= 60;
 | |
| 	  uint32_t s10Target = durMins * 600;
 | |
| 	  if (s10SinceStart > s10Target) s10SinceStart = s10Target;
 | |
| 	  stage = map(s10SinceStart, 0, s10Target, 0, 0xFFFF);
 | |
| 	  if (SEGMENT.speed > 60) stage = 0xFFFF - stage; //sunset
 | |
|   }
 | |
|   
 | |
|   for (uint16_t i = 0; i <= SEGLEN/2; i++)
 | |
|   {
 | |
|     //default palette is Fire
 | |
|     uint32_t c = color_from_palette(0, false, true, 255); //background
 | |
| 
 | |
|     uint16_t wave = triwave16((i * stage) / SEGLEN);
 | |
| 
 | |
|     wave = (wave >> 8) + ((wave * SEGMENT.intensity) >> 15);
 | |
| 
 | |
|     if (wave > 240) { //clipped, full white sun
 | |
|       c = color_from_palette( 240, false, true, 255);
 | |
|     } else { //transition
 | |
|       c = color_from_palette(wave, false, true, 255);
 | |
|     }
 | |
|     setPixelColor(i, c);
 | |
|     setPixelColor(SEGLEN - i - 1, c);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_SUNRISE PROGMEM = "Sunrise@Time [min]=60,;;0";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Effects by Andrew Tuline
 | |
|  */
 | |
| uint16_t WS2812FX::phased_base(uint8_t moder) {                  // We're making sine waves here. By Andrew Tuline.
 | |
| 
 | |
|   uint8_t allfreq = 16;                                          // Base frequency.
 | |
|   float *phase = reinterpret_cast<float*>(&SEGENV.step);         // Phase change value gets calculated (float fits into unsigned long).
 | |
|   uint8_t cutOff = (255-SEGMENT.intensity);                      // You can change the number of pixels.  AKA INTENSITY (was 192).
 | |
|   uint8_t modVal = 5;//SEGMENT.fft1/8+1;                         // You can change the modulus. AKA FFT1 (was 5).
 | |
| 
 | |
|   uint8_t index = now/64;                                        // Set color rotation speed
 | |
|   *phase += SEGMENT.speed/32.0;                                  // You can change the speed of the wave. AKA SPEED (was .4)
 | |
| 
 | |
|   for (int i = 0; i < SEGLEN; i++) {
 | |
|     if (moder == 1) modVal = (inoise8(i*10 + i*10) /16);         // Let's randomize our mod length with some Perlin noise.
 | |
|     uint16_t val = (i+1) * allfreq;                              // This sets the frequency of the waves. The +1 makes sure that leds[0] is used.
 | |
|     if (modVal == 0) modVal = 1;
 | |
|     val += *phase * (i % modVal +1) /2;                          // This sets the varying phase change of the waves. By Andrew Tuline.
 | |
|     uint8_t b = cubicwave8(val);                                 // Now we make an 8 bit sinewave.
 | |
|     b = (b > cutOff) ? (b - cutOff) : 0;                         // A ternary operator to cutoff the light.
 | |
|     setPixelColor(i, color_blend(SEGCOLOR(1), color_from_palette(index, false, false, 0), b));
 | |
|     index += 256 / SEGLEN;
 | |
|     if (SEGLEN > 256) index ++;                                  // Correction for segments longer than 256 LEDs
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_phased(void) {
 | |
|   return phased_base(0);
 | |
| }
 | |
| static const char *_data_FX_MODE_PHASED PROGMEM = "Phased";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_phased_noise(void) {
 | |
|   return phased_base(1);
 | |
| }
 | |
| static const char *_data_FX_MODE_PHASEDNOISE PROGMEM = "Phased Noise";
 | |
| 
 | |
| 
 | |
| uint16_t WS2812FX::mode_twinkleup(void) {                 // A very short twinkle routine with fade-in and dual controls. By Andrew Tuline.
 | |
|   const uint16_t cols = isMatrix ? SEGMENT.virtualWidth() : 1;
 | |
|   const uint16_t rows = isMatrix ? SEGMENT.virtualHeight() : SEGMENT.virtualLength();
 | |
| 
 | |
|   random16_set_seed(535);                                 // The randomizer needs to be re-set each time through the loop in order for the same 'random' numbers to be the same each time through.
 | |
| 
 | |
|   for (int i = 0; i<rows*cols; i++) {
 | |
|     uint16_t j = i % rows, k = i / rows;
 | |
|     uint8_t ranstart = random8();                         // The starting value (aka brightness) for each pixel. Must be consistent each time through the loop for this to work.
 | |
|     uint8_t pixBri = sin8(ranstart + 16 * now/(256-SEGMENT.speed));
 | |
|     if (random8() > SEGMENT.intensity) pixBri = 0;
 | |
|     uint32_t col = color_blend(SEGCOLOR(1), color_from_palette(random8()+now/100, false, PALETTE_SOLID_WRAP, 0), pixBri);
 | |
|     if (isMatrix) setPixelColorXY(j, k, col);
 | |
|     else          setPixelColor(i, col);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_TWINKLEUP PROGMEM = "Twinkleup@!,Intensity;!,!,;!";
 | |
| 
 | |
| 
 | |
| // Peaceful noise that's slow and with gradually changing palettes. Does not support WLED palettes or default colours or controls.
 | |
| uint16_t WS2812FX::mode_noisepal(void) {                                    // Slow noise palette by Andrew Tuline.
 | |
|   uint16_t scale = 15 + (SEGMENT.intensity >> 2); //default was 30
 | |
|   //#define scale 30
 | |
| 
 | |
|   uint16_t dataSize = sizeof(CRGBPalette16) * 2; //allocate space for 2 Palettes (2 * 16 * 3 = 96 bytes)
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
| 
 | |
|   CRGBPalette16* palettes = reinterpret_cast<CRGBPalette16*>(SEGENV.data);
 | |
| 
 | |
|   uint16_t changePaletteMs = 4000 + SEGMENT.speed *10; //between 4 - 6.5sec
 | |
|   if (millis() - SEGENV.step > changePaletteMs)
 | |
|   {
 | |
|     SEGENV.step = millis();
 | |
| 
 | |
|     uint8_t baseI = random8();
 | |
|     palettes[1] = CRGBPalette16(CHSV(baseI+random8(64), 255, random8(128,255)), CHSV(baseI+128, 255, random8(128,255)), CHSV(baseI+random8(92), 192, random8(128,255)), CHSV(baseI+random8(92), 255, random8(128,255)));
 | |
|   }
 | |
| 
 | |
|   CRGB color;
 | |
| 
 | |
|   //EVERY_N_MILLIS(10) { //(don't have to time this, effect function is only called every 24ms)
 | |
|   nblendPaletteTowardPalette(palettes[0], palettes[1], 48);               // Blend towards the target palette over 48 iterations.
 | |
| 
 | |
|   if (SEGMENT.palette > 0) palettes[0] = currentPalette;
 | |
| 
 | |
|   for(int i = 0; i < SEGLEN; i++) {
 | |
|     uint8_t index = inoise8(i*scale, SEGENV.aux0+i*scale);                // Get a value from the noise function. I'm using both x and y axis.
 | |
|     color = ColorFromPalette(palettes[0], index, 255, LINEARBLEND);       // Use the my own palette.
 | |
|     setPixelColor(i, color.red, color.green, color.blue);
 | |
|   }
 | |
| 
 | |
|   SEGENV.aux0 += beatsin8(10,1,4);                                        // Moving along the distance. Vary it a bit with a sine wave.
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_NOISEPAL PROGMEM = "Noise Pal";
 | |
| 
 | |
| 
 | |
| // Sine waves that have controllable phase change speed, frequency and cutoff. By Andrew Tuline.
 | |
| // SEGMENT.speed ->Speed, SEGMENT.intensity -> Frequency (SEGMENT.fft1 -> Color change, SEGMENT.fft2 -> PWM cutoff)
 | |
| //
 | |
| uint16_t WS2812FX::mode_sinewave(void) {             // Adjustable sinewave. By Andrew Tuline
 | |
|   //#define qsuba(x, b)  ((x>b)?x-b:0)               // Analog Unsigned subtraction macro. if result <0, then => 0
 | |
| 
 | |
|   uint16_t colorIndex = now /32;//(256 - SEGMENT.fft1);  // Amount of colour change.
 | |
| 
 | |
|   SEGENV.step += SEGMENT.speed/16;                   // Speed of animation.
 | |
|   uint16_t freq = SEGMENT.intensity/4;//SEGMENT.fft2/8;                       // Frequency of the signal.
 | |
| 
 | |
|   for (int i=0; i<SEGLEN; i++) {                   // For each of the LED's in the strand, set a brightness based on a wave as follows:
 | |
|     int pixBri = cubicwave8((i*freq)+SEGENV.step);//qsuba(cubicwave8((i*freq)+SEGENV.step), (255-SEGMENT.intensity)); // qsub sets a minimum value called thiscutoff. If < thiscutoff, then bright = 0. Otherwise, bright = 128 (as defined in qsub)..
 | |
|     //setPixCol(i, i*colorIndex/255, pixBri);
 | |
|     setPixelColor(i, color_blend(SEGCOLOR(1), color_from_palette(i*colorIndex/255, false, PALETTE_SOLID_WRAP, 0), pixBri));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_SINEWAVE PROGMEM = "Sine";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Best of both worlds from Palette and Spot effects. By Aircoookie
 | |
|  */
 | |
| uint16_t WS2812FX::mode_flow(void)
 | |
| {
 | |
|   uint16_t counter = 0;
 | |
|   if (SEGMENT.speed != 0) 
 | |
|   {
 | |
|     counter = now * ((SEGMENT.speed >> 2) +1);
 | |
|     counter = counter >> 8;
 | |
|   }
 | |
|   
 | |
|   uint16_t maxZones = SEGLEN / 6; //only looks good if each zone has at least 6 LEDs
 | |
|   uint16_t zones = (SEGMENT.intensity * maxZones) >> 8;
 | |
|   if (zones & 0x01) zones++; //zones must be even
 | |
|   if (zones < 2) zones = 2;
 | |
|   uint16_t zoneLen = SEGLEN / zones;
 | |
|   uint16_t offset = (SEGLEN - zones * zoneLen) >> 1;
 | |
| 
 | |
|   fill(color_from_palette(-counter, false, true, 255));
 | |
| 
 | |
|   for (uint16_t z = 0; z < zones; z++)
 | |
|   {
 | |
|     uint16_t pos = offset + z * zoneLen;
 | |
|     for (uint16_t i = 0; i < zoneLen; i++)
 | |
|     {
 | |
|       uint8_t colorIndex = (i * 255 / zoneLen) - counter;
 | |
|       uint16_t led = (z & 0x01) ? i : (zoneLen -1) -i;
 | |
|       if (SEGMENT.getOption(SEG_OPTION_REVERSED)) led = (zoneLen -1) -led;
 | |
|       setPixelColor(pos + led, color_from_palette(colorIndex, false, true, 255));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_FLOW PROGMEM = "Flow";
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Dots waving around in a sine/pendulum motion.
 | |
|  * Little pixel birds flying in a circle. By Aircoookie
 | |
|  */
 | |
| uint16_t WS2812FX::mode_chunchun(void)
 | |
| {
 | |
|   fill(SEGCOLOR(1));
 | |
|   uint16_t counter = now*(6 + (SEGMENT.speed >> 4));
 | |
|   uint16_t numBirds = 2 + (SEGLEN >> 3);  // 2 + 1/8 of a segment
 | |
|   uint16_t span = (SEGMENT.intensity << 8) / numBirds;
 | |
| 
 | |
|   for (uint16_t i = 0; i < numBirds; i++)
 | |
|   {
 | |
|     counter -= span;
 | |
|     uint16_t megumin = sin16(counter) + 0x8000;
 | |
|     uint32_t bird = (megumin * SEGLEN) >> 16;
 | |
|     uint32_t c = color_from_palette((i * 255)/ numBirds, false, false, 0);  // no palette wrapping
 | |
|     setPixelColor(bird, c);
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_CHUNCHUN PROGMEM = "Chunchun@!,Gap size;!,!,;!";
 | |
| 
 | |
| 
 | |
| //13 bytes
 | |
| typedef struct Spotlight {
 | |
|   float speed;
 | |
|   uint8_t colorIdx;
 | |
|   int16_t position;
 | |
|   unsigned long lastUpdateTime;
 | |
|   uint8_t width;
 | |
|   uint8_t type;
 | |
| } spotlight;
 | |
| 
 | |
| #define SPOT_TYPE_SOLID       0
 | |
| #define SPOT_TYPE_GRADIENT    1
 | |
| #define SPOT_TYPE_2X_GRADIENT 2
 | |
| #define SPOT_TYPE_2X_DOT      3
 | |
| #define SPOT_TYPE_3X_DOT      4
 | |
| #define SPOT_TYPE_4X_DOT      5
 | |
| #define SPOT_TYPES_COUNT      6
 | |
| #ifdef ESP8266
 | |
|   #define SPOT_MAX_COUNT 17          //Number of simultaneous waves
 | |
| #else
 | |
|   #define SPOT_MAX_COUNT 49          //Number of simultaneous waves
 | |
| #endif
 | |
| 
 | |
| /*
 | |
|  * Spotlights moving back and forth that cast dancing shadows.
 | |
|  * Shine this through tree branches/leaves or other close-up objects that cast
 | |
|  * interesting shadows onto a ceiling or tarp.
 | |
|  *
 | |
|  * By Steve Pomeroy @xxv
 | |
|  */
 | |
| uint16_t WS2812FX::mode_dancing_shadows(void)
 | |
| {
 | |
|   uint8_t numSpotlights = map(SEGMENT.intensity, 0, 255, 2, SPOT_MAX_COUNT);  // 49 on 32 segment ESP32, 17 on 16 segment ESP8266
 | |
|   bool initialize = SEGENV.aux0 != numSpotlights;
 | |
|   SEGENV.aux0 = numSpotlights;
 | |
| 
 | |
|   uint16_t dataSize = sizeof(spotlight) * numSpotlights;
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   Spotlight* spotlights = reinterpret_cast<Spotlight*>(SEGENV.data);
 | |
| 
 | |
|   fill(BLACK);
 | |
| 
 | |
|   unsigned long time = millis();
 | |
|   bool respawn = false;
 | |
| 
 | |
|   for (uint8_t i = 0; i < numSpotlights; i++) {
 | |
|     if (!initialize) {
 | |
|       // advance the position of the spotlight
 | |
|       int16_t delta = (float)(time - spotlights[i].lastUpdateTime) *
 | |
|                   (spotlights[i].speed * ((1.0 + SEGMENT.speed)/100.0));
 | |
| 
 | |
|       if (abs(delta) >= 1) {
 | |
|         spotlights[i].position += delta;
 | |
|         spotlights[i].lastUpdateTime = time;
 | |
|       }
 | |
| 
 | |
|       respawn = (spotlights[i].speed > 0.0 && spotlights[i].position > (SEGLEN + 2))
 | |
|              || (spotlights[i].speed < 0.0 && spotlights[i].position < -(spotlights[i].width + 2));
 | |
|     }
 | |
| 
 | |
|     if (initialize || respawn) {
 | |
|       spotlights[i].colorIdx = random8();
 | |
|       spotlights[i].width = random8(1, 10);
 | |
| 
 | |
|       spotlights[i].speed = 1.0/random8(4, 50);
 | |
| 
 | |
|       if (initialize) {
 | |
|         spotlights[i].position = random16(SEGLEN);
 | |
|         spotlights[i].speed *= random8(2) ? 1.0 : -1.0;
 | |
|       } else {
 | |
|         if (random8(2)) {
 | |
|           spotlights[i].position = SEGLEN + spotlights[i].width;
 | |
|           spotlights[i].speed *= -1.0;
 | |
|         }else {
 | |
|           spotlights[i].position = -spotlights[i].width;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       spotlights[i].lastUpdateTime = time;
 | |
|       spotlights[i].type = random8(SPOT_TYPES_COUNT);
 | |
|     }
 | |
| 
 | |
|     uint32_t color = color_from_palette(spotlights[i].colorIdx, false, false, 0);
 | |
|     int start = spotlights[i].position;
 | |
| 
 | |
|     if (spotlights[i].width <= 1) {
 | |
|       if (start >= 0 && start < SEGLEN) {
 | |
|         blendPixelColor(start, color, 128);
 | |
|       }
 | |
|     } else {
 | |
|       switch (spotlights[i].type) {
 | |
|         case SPOT_TYPE_SOLID:
 | |
|           for (uint8_t j = 0; j < spotlights[i].width; j++) {
 | |
|             if ((start + j) >= 0 && (start + j) < SEGLEN) {
 | |
|               blendPixelColor(start + j, color, 128);
 | |
|             }
 | |
|           }
 | |
|         break;
 | |
| 
 | |
|         case SPOT_TYPE_GRADIENT:
 | |
|           for (uint8_t j = 0; j < spotlights[i].width; j++) {
 | |
|             if ((start + j) >= 0 && (start + j) < SEGLEN) {
 | |
|               blendPixelColor(start + j, color,
 | |
|                               cubicwave8(map(j, 0, spotlights[i].width - 1, 0, 255)));
 | |
|             }
 | |
|           }
 | |
|         break;
 | |
| 
 | |
|         case SPOT_TYPE_2X_GRADIENT:
 | |
|           for (uint8_t j = 0; j < spotlights[i].width; j++) {
 | |
|             if ((start + j) >= 0 && (start + j) < SEGLEN) {
 | |
|               blendPixelColor(start + j, color,
 | |
|                               cubicwave8(2 * map(j, 0, spotlights[i].width - 1, 0, 255)));
 | |
|             }
 | |
|           }
 | |
|         break;
 | |
| 
 | |
|         case SPOT_TYPE_2X_DOT:
 | |
|           for (uint8_t j = 0; j < spotlights[i].width; j += 2) {
 | |
|             if ((start + j) >= 0 && (start + j) < SEGLEN) {
 | |
|               blendPixelColor(start + j, color, 128);
 | |
|             }
 | |
|           }
 | |
|         break;
 | |
| 
 | |
|         case SPOT_TYPE_3X_DOT:
 | |
|           for (uint8_t j = 0; j < spotlights[i].width; j += 3) {
 | |
|             if ((start + j) >= 0 && (start + j) < SEGLEN) {
 | |
|               blendPixelColor(start + j, color, 128);
 | |
|             }
 | |
|           }
 | |
|         break;
 | |
| 
 | |
|         case SPOT_TYPE_4X_DOT:
 | |
|           for (uint8_t j = 0; j < spotlights[i].width; j += 4) {
 | |
|             if ((start + j) >= 0 && (start + j) < SEGLEN) {
 | |
|               blendPixelColor(start + j, color, 128);
 | |
|             }
 | |
|           }
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_DANCING_SHADOWS PROGMEM = "Dancing Shadows@!,# of shadows;!,,;!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|   Imitates a washing machine, rotating same waves forward, then pause, then backward.
 | |
|   By Stefan Seegel
 | |
| */
 | |
| uint16_t WS2812FX::mode_washing_machine(void) {
 | |
|   float speed = tristate_square8(now >> 7, 90, 15);
 | |
|   float quot  = 32.0f - ((float)SEGMENT.speed / 16.0f);
 | |
|   speed /= quot;
 | |
| 
 | |
|   SEGENV.step += (speed * 128.0f);
 | |
|   
 | |
|   for (int i=0; i<SEGLEN; i++) {
 | |
|     uint8_t col = sin8(((SEGMENT.intensity / 25 + 1) * 255 * i / SEGLEN) + (SEGENV.step >> 7));
 | |
|     setPixelColor(i, color_from_palette(col, false, PALETTE_SOLID_WRAP, 3));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_WASHING_MACHINE PROGMEM = "Washing Machine";
 | |
| 
 | |
| 
 | |
| /*
 | |
|   Blends random colors across palette
 | |
|   Modified, originally by Mark Kriegsman https://gist.github.com/kriegsman/1f7ccbbfa492a73c015e
 | |
| */
 | |
| uint16_t WS2812FX::mode_blends(void) {
 | |
|   uint16_t pixelLen = SEGLEN > UINT8_MAX ? UINT8_MAX : SEGLEN;
 | |
|   uint16_t dataSize = sizeof(uint32_t) * (pixelLen + 1);  // max segment length of 56 pixels on 16 segment ESP8266
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   uint32_t* pixels = reinterpret_cast<uint32_t*>(SEGENV.data);
 | |
|   uint8_t blendSpeed = map(SEGMENT.intensity, 0, UINT8_MAX, 10, 128);
 | |
|   uint8_t shift = (now * ((SEGMENT.speed >> 3) +1)) >> 8;
 | |
| 
 | |
|   for (int i = 0; i < pixelLen; i++) {
 | |
|     pixels[i] = color_blend(pixels[i], color_from_palette(shift + quadwave8((i + 1) * 16), false, PALETTE_SOLID_WRAP, 255), blendSpeed);
 | |
|     shift += 3;
 | |
|   }
 | |
| 
 | |
|   uint16_t offset = 0;
 | |
|   for (int i = 0; i < SEGLEN; i++) {
 | |
|     setPixelColor(i, pixels[offset++]);
 | |
|     if (offset > pixelLen) offset = 0;
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_BLENDS PROGMEM = "Blends@Shift speed,Blend speed;1,2,3,!";
 | |
| 
 | |
| 
 | |
| /*
 | |
|   TV Simulator
 | |
|   Modified and adapted to WLED by Def3nder, based on "Fake TV Light for Engineers" by Phillip Burgess https://learn.adafruit.com/fake-tv-light-for-engineers/arduino-sketch
 | |
| */
 | |
| //43 bytes
 | |
| typedef struct TvSim {
 | |
|   uint32_t totalTime = 0;
 | |
|   uint32_t fadeTime  = 0;
 | |
|   uint32_t startTime = 0;
 | |
|   uint32_t elapsed   = 0;
 | |
|   uint32_t pixelNum  = 0;
 | |
|   uint16_t sliderValues = 0;
 | |
|   uint32_t sceeneStart    = 0;
 | |
|   uint32_t sceeneDuration = 0;
 | |
|   uint16_t sceeneColorHue = 0;
 | |
|   uint8_t  sceeneColorSat = 0;
 | |
|   uint8_t  sceeneColorBri = 0;
 | |
|   uint8_t  actualColorR = 0;
 | |
|   uint8_t  actualColorG = 0;
 | |
|   uint8_t  actualColorB = 0;
 | |
|   uint16_t pr = 0; // Prev R, G, B
 | |
|   uint16_t pg = 0;
 | |
|   uint16_t pb = 0;
 | |
| } tvSim;
 | |
| 
 | |
| uint16_t WS2812FX::mode_tv_simulator(void) {
 | |
|   uint16_t nr, ng, nb, r, g, b, i, hue;
 | |
|   uint8_t  sat, bri, j;
 | |
| 
 | |
|   if (!SEGENV.allocateData(sizeof(tvSim))) return mode_static(); //allocation failed
 | |
|   TvSim* tvSimulator = reinterpret_cast<TvSim*>(SEGENV.data);
 | |
| 
 | |
|   uint8_t colorSpeed     = map(SEGMENT.speed,     0, UINT8_MAX,  1, 20);
 | |
|   uint8_t colorIntensity = map(SEGMENT.intensity, 0, UINT8_MAX, 10, 30);
 | |
| 
 | |
|   i = SEGMENT.speed << 8 | SEGMENT.intensity;
 | |
|   if (i != tvSimulator->sliderValues) {
 | |
|     tvSimulator->sliderValues = i;
 | |
|     SEGENV.aux1 = 0;
 | |
|   }
 | |
| 
 | |
|     // create a new sceene
 | |
|     if (((millis() - tvSimulator->sceeneStart) >= tvSimulator->sceeneDuration) || SEGENV.aux1 == 0) {
 | |
|       tvSimulator->sceeneStart    = millis();                                               // remember the start of the new sceene
 | |
|       tvSimulator->sceeneDuration = random16(60* 250* colorSpeed, 60* 750 * colorSpeed);    // duration of a "movie sceene" which has similar colors (5 to 15 minutes with max speed slider)
 | |
|       tvSimulator->sceeneColorHue = random16(   0, 768);                                    // random start color-tone for the sceene
 | |
|       tvSimulator->sceeneColorSat = random8 ( 100, 130 + colorIntensity);                   // random start color-saturation for the sceene
 | |
|       tvSimulator->sceeneColorBri = random8 ( 200, 240);                                    // random start color-brightness for the sceene
 | |
|       SEGENV.aux1 = 1;
 | |
|       SEGENV.aux0 = 0;
 | |
|     } 
 | |
|     
 | |
|     // slightly change the color-tone in this sceene
 | |
|     if ( SEGENV.aux0 == 0) {
 | |
|       // hue change in both directions
 | |
|       j = random8(4 * colorIntensity);
 | |
|       hue = (random8() < 128) ? ((j < tvSimulator->sceeneColorHue)       ? tvSimulator->sceeneColorHue - j : 767 - tvSimulator->sceeneColorHue - j) :  // negative
 | |
|                                 ((j + tvSimulator->sceeneColorHue) < 767 ? tvSimulator->sceeneColorHue + j : tvSimulator->sceeneColorHue + j - 767) ;  // positive
 | |
|       
 | |
|       // saturation
 | |
|       j = random8(2 * colorIntensity);
 | |
|       sat = (tvSimulator->sceeneColorSat - j) < 0 ? 0 : tvSimulator->sceeneColorSat - j;
 | |
|       
 | |
|       // brightness
 | |
|       j = random8(100);
 | |
|       bri = (tvSimulator->sceeneColorBri - j) < 0 ? 0 : tvSimulator->sceeneColorBri - j;
 | |
| 
 | |
|       // calculate R,G,B from HSV
 | |
|       // Source: https://blog.adafruit.com/2012/03/14/constant-brightness-hsb-to-rgb-algorithm/
 | |
|       { // just to create a local scope for  the variables
 | |
|         uint8_t temp[5], n = (hue >> 8) % 3;
 | |
|         uint8_t x = ((((hue & 255) * sat) >> 8) * bri) >> 8;
 | |
|         uint8_t s = (  (256 - sat) * bri) >> 8;
 | |
|         temp[0] = temp[3] =       s;
 | |
|         temp[1] = temp[4] =   x + s;
 | |
|         temp[2] =           bri - x;
 | |
|         tvSimulator->actualColorR = temp[n + 2];
 | |
|         tvSimulator->actualColorG = temp[n + 1];
 | |
|         tvSimulator->actualColorB = temp[n    ];
 | |
|       }
 | |
|     }
 | |
|     // Apply gamma correction, further expand to 16/16/16
 | |
|     nr = (uint8_t)gamma8(tvSimulator->actualColorR) * 257; // New R/G/B
 | |
|     ng = (uint8_t)gamma8(tvSimulator->actualColorG) * 257;
 | |
|     nb = (uint8_t)gamma8(tvSimulator->actualColorB) * 257;
 | |
| 
 | |
|   if (SEGENV.aux0 == 0) {  // initialize next iteration 
 | |
|     SEGENV.aux0 = 1;
 | |
| 
 | |
|     // randomize total duration and fade duration for the actual color
 | |
|     tvSimulator->totalTime = random16(250, 2500);                   // Semi-random pixel-to-pixel time
 | |
|     tvSimulator->fadeTime  = random16(0, tvSimulator->totalTime);   // Pixel-to-pixel transition time
 | |
|     if (random8(10) < 3) tvSimulator->fadeTime = 0;                 // Force scene cut 30% of time
 | |
| 
 | |
|     tvSimulator->startTime = millis();
 | |
|   } // end of initialization
 | |
| 
 | |
|   // how much time is elapsed ?
 | |
|   tvSimulator->elapsed = millis() - tvSimulator->startTime;
 | |
| 
 | |
|   // fade from prev volor to next color
 | |
|   if (tvSimulator->elapsed < tvSimulator->fadeTime) {
 | |
|     r = map(tvSimulator->elapsed, 0, tvSimulator->fadeTime, tvSimulator->pr, nr); 
 | |
|     g = map(tvSimulator->elapsed, 0, tvSimulator->fadeTime, tvSimulator->pg, ng);
 | |
|     b = map(tvSimulator->elapsed, 0, tvSimulator->fadeTime, tvSimulator->pb, nb);
 | |
|   } else { // Avoid divide-by-zero in map()
 | |
|     r = nr;
 | |
|     g = ng;
 | |
|     b = nb;
 | |
|   }
 | |
| 
 | |
|   // set strip color
 | |
|   for (i = 0; i < SEGLEN; i++) {
 | |
|     setPixelColor(i, r >> 8, g >> 8, b >> 8);  // Quantize to 8-bit
 | |
|   }
 | |
| 
 | |
|   // if total duration has passed, remember last color and restart the loop
 | |
|   if ( tvSimulator->elapsed >= tvSimulator->totalTime) {
 | |
|     tvSimulator->pr = nr; // Prev RGB = new RGB
 | |
|     tvSimulator->pg = ng;
 | |
|     tvSimulator->pb = nb;
 | |
|     SEGENV.aux0 = 0;
 | |
|   }
 | |
|   
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_TV_SIMULATOR PROGMEM = "TV Simulator";
 | |
| 
 | |
| 
 | |
| /*
 | |
|   Aurora effect
 | |
| */
 | |
| 
 | |
| //CONFIG
 | |
| #ifdef ESP8266
 | |
|   #define W_MAX_COUNT  9          //Number of simultaneous waves
 | |
| #else
 | |
|   #define W_MAX_COUNT 20          //Number of simultaneous waves
 | |
| #endif
 | |
| #define W_MAX_SPEED 6             //Higher number, higher speed
 | |
| #define W_WIDTH_FACTOR 6          //Higher number, smaller waves
 | |
| 
 | |
| //24 bytes
 | |
| class AuroraWave {
 | |
|   private:
 | |
|     uint16_t ttl;
 | |
|     CRGB basecolor;
 | |
|     float basealpha;
 | |
|     uint16_t age;
 | |
|     uint16_t width;
 | |
|     float center;
 | |
|     bool goingleft;
 | |
|     float speed_factor;
 | |
|     bool alive = true;
 | |
| 
 | |
|   public:
 | |
|     void init(uint32_t segment_length, CRGB color) {
 | |
|       ttl = random(500, 1501);
 | |
|       basecolor = color;
 | |
|       basealpha = random(60, 101) / (float)100;
 | |
|       age = 0;
 | |
|       width = random(segment_length / 20, segment_length / W_WIDTH_FACTOR); //half of width to make math easier
 | |
|       if (!width) width = 1;
 | |
|       center = random(101) / (float)100 * segment_length;
 | |
|       goingleft = random(0, 2) == 0;
 | |
|       speed_factor = (random(10, 31) / (float)100 * W_MAX_SPEED / 255);
 | |
|       alive = true;
 | |
|     }
 | |
| 
 | |
|     CRGB getColorForLED(int ledIndex) {      
 | |
|       if(ledIndex < center - width || ledIndex > center + width) return 0; //Position out of range of this wave
 | |
| 
 | |
|       CRGB rgb;
 | |
| 
 | |
|       //Offset of this led from center of wave
 | |
|       //The further away from the center, the dimmer the LED
 | |
|       float offset = ledIndex - center;
 | |
|       if (offset < 0) offset = -offset;
 | |
|       float offsetFactor = offset / width;
 | |
| 
 | |
|       //The age of the wave determines it brightness.
 | |
|       //At half its maximum age it will be the brightest.
 | |
|       float ageFactor = 0.1;        
 | |
|       if((float)age / ttl < 0.5) {
 | |
|         ageFactor = (float)age / (ttl / 2);
 | |
|       } else {
 | |
|         ageFactor = (float)(ttl - age) / ((float)ttl * 0.5);
 | |
|       }
 | |
| 
 | |
|       //Calculate color based on above factors and basealpha value
 | |
|       float factor = (1 - offsetFactor) * ageFactor * basealpha;
 | |
|       rgb.r = basecolor.r * factor;
 | |
|       rgb.g = basecolor.g * factor;
 | |
|       rgb.b = basecolor.b * factor;
 | |
|     
 | |
|       return rgb;
 | |
|     };
 | |
| 
 | |
|     //Change position and age of wave
 | |
|     //Determine if its sill "alive"
 | |
|     void update(uint32_t segment_length, uint32_t speed) {
 | |
|       if(goingleft) {
 | |
|         center -= speed_factor * speed;
 | |
|       } else {
 | |
|         center += speed_factor * speed;
 | |
|       }
 | |
| 
 | |
|       age++;
 | |
| 
 | |
|       if(age > ttl) {
 | |
|         alive = false;
 | |
|       } else {
 | |
|         if(goingleft) {
 | |
|           if(center + width < 0) {
 | |
|             alive = false;
 | |
|           }
 | |
|         } else {
 | |
|           if(center - width > segment_length) {
 | |
|             alive = false;
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     };
 | |
| 
 | |
|     bool stillAlive() {
 | |
|       return alive;
 | |
|     };
 | |
| };
 | |
| 
 | |
| uint16_t WS2812FX::mode_aurora(void) {
 | |
|   //aux1 = Wavecount
 | |
|   //aux2 = Intensity in last loop
 | |
| 
 | |
|   AuroraWave* waves;
 | |
| 
 | |
| //TODO: I am not sure this is a correct way of handling memory allocation since if it fails on 1st run
 | |
| // it will display static effect but on second run it may crash ESP since data will be nullptr
 | |
| 
 | |
|   if(SEGENV.aux0 != SEGMENT.intensity || SEGENV.call == 0) {
 | |
|     //Intensity slider changed or first call
 | |
|     SEGENV.aux1 = map(SEGMENT.intensity, 0, 255, 2, W_MAX_COUNT);
 | |
|     SEGENV.aux0 = SEGMENT.intensity;
 | |
| 
 | |
|     if(!SEGENV.allocateData(sizeof(AuroraWave) * SEGENV.aux1)) { // 26 on 32 segment ESP32, 9 on 16 segment ESP8266
 | |
|       return mode_static(); //allocation failed
 | |
|     }
 | |
| 
 | |
|     waves = reinterpret_cast<AuroraWave*>(SEGENV.data);
 | |
| 
 | |
|     for(int i = 0; i < SEGENV.aux1; i++) {
 | |
|       waves[i].init(SEGLEN, col_to_crgb(color_from_palette(random8(), false, false, random(0, 3))));
 | |
|     }
 | |
|   } else {
 | |
|     waves = reinterpret_cast<AuroraWave*>(SEGENV.data);
 | |
|   }
 | |
| 
 | |
|   for(int i = 0; i < SEGENV.aux1; i++) {
 | |
|     //Update values of wave
 | |
|     waves[i].update(SEGLEN, SEGMENT.speed);
 | |
| 
 | |
|     if(!(waves[i].stillAlive())) {
 | |
|       //If a wave dies, reinitialize it starts over.
 | |
|       waves[i].init(SEGLEN, col_to_crgb(color_from_palette(random8(), false, false, random(0, 3))));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   uint8_t backlight = 1; //dimmer backlight if less active colors
 | |
|   if (SEGCOLOR(0)) backlight++;
 | |
|   if (SEGCOLOR(1)) backlight++;
 | |
|   if (SEGCOLOR(2)) backlight++;
 | |
|   //Loop through LEDs to determine color
 | |
|   for(int i = 0; i < SEGLEN; i++) {    
 | |
|     CRGB mixedRgb = CRGB(backlight, backlight, backlight);
 | |
| 
 | |
|     //For each LED we must check each wave if it is "active" at this position.
 | |
|     //If there are multiple waves active on a LED we multiply their values.
 | |
|     for(int  j = 0; j < SEGENV.aux1; j++) {
 | |
|       CRGB rgb = waves[j].getColorForLED(i);
 | |
|       
 | |
|       if(rgb != CRGB(0)) {       
 | |
|         mixedRgb += rgb;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     setPixelColor(i, mixedRgb[0], mixedRgb[1], mixedRgb[2]);
 | |
|   }
 | |
|   
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_AURORA PROGMEM = "Aurora@!=24,!;1,2,3;!=50";
 | |
| 
 | |
| // WLED-SR effects
 | |
| 
 | |
| /////////////////////////
 | |
| //     Perlin Move     //
 | |
| /////////////////////////
 | |
| // 16 bit perlinmove. Use Perlin Noise instead of sinewaves for movement. By Andrew Tuline.
 | |
| // Controls are speed, # of pixels, faderate.
 | |
| uint16_t WS2812FX::mode_perlinmove(void) {
 | |
| 
 | |
|   fade_out(255-SEGMENT.custom1);
 | |
|   for (uint16_t i = 0; i < SEGMENT.intensity/16 + 1; i++) {
 | |
|     uint16_t locn = inoise16(millis()*128/(260-SEGMENT.speed)+i*15000, millis()*128/(260-SEGMENT.speed)); // Get a new pixel location from moving noise.
 | |
|     uint16_t pixloc = map(locn, 50*256, 192*256, 0, SEGLEN-1);                                            // Map that to the length of the strand, and ensure we don't go over.
 | |
|     setPixelColor(pixloc, color_from_palette(pixloc%255, false, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_perlinmove()
 | |
| static const char *_data_FX_MODE_PERLINMOVE PROGMEM = "Perlin Move@!,# of pixels,fade rate;,!;!";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     Waveins         //
 | |
| /////////////////////////
 | |
| // Uses beatsin8() + phase shifting. By: Andrew Tuline
 | |
| uint16_t WS2812FX::mode_wavesins(void) {
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     uint8_t bri = sin8(millis()/4 + i * SEGMENT.intensity);
 | |
|     setPixelColor(i, ColorFromPalette(currentPalette, beatsin8(SEGMENT.speed, SEGMENT.custom1, SEGMENT.custom1+SEGMENT.custom2, 0, i * SEGMENT.custom3), bri, LINEARBLEND));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_waveins()
 | |
| static const char *_data_FX_MODE_WAVESINS PROGMEM = "Wavesins@Speed,Brightness variation,Starting Color,Range of Colors,Color variation;;!";
 | |
| 
 | |
| 
 | |
| //////////////////////////////
 | |
| //     Flow Stripe          //
 | |
| //////////////////////////////
 | |
| // By: ldirko  https://editor.soulmatelights.com/gallery/392-flow-led-stripe , modifed by: Andrew Tuline
 | |
| uint16_t WS2812FX::mode_FlowStripe(void) {
 | |
| 
 | |
|   const uint16_t hl = SEGLEN * 10 / 13;
 | |
|   uint8_t hue = millis() / (SEGMENT.speed+1);
 | |
|   uint32_t t = millis() / (SEGMENT.intensity/8+1);
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     int c = (abs(i - hl) / hl) * 127;
 | |
|     c = sin8(c);
 | |
|     c = sin8(c / 2 + t);
 | |
|     byte b = sin8(c + t/8);
 | |
|     setPixelColor(i, CHSV(b + hue, 255, 255));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_FlowStripe()
 | |
| static const char *_data_FX_MODE_FLOWSTRIPE PROGMEM = "Flow Stripe@Hue speed,Effect speed;;";
 | |
| 
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| //***************************  2D routines  ***********************************
 | |
| 
 | |
| 
 | |
| // Black hole
 | |
| uint16_t WS2812FX::mode_2DBlackHole(void) {            // By: Stepko https://editor.soulmatelights.com/gallery/1012 , Modified by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   uint16_t x, y;
 | |
| 
 | |
|   // initialize on first call
 | |
|   if (SEGENV.call == 0) {
 | |
|     fill_solid(leds, CRGB::Black);
 | |
|   }
 | |
| 
 | |
|   fadeToBlackBy(leds, 16 + (SEGMENT.speed>>3)); // create fading trails
 | |
|   float t = (float)(millis())/128;              // timebase
 | |
|   // outer stars
 | |
|   for (byte i = 0; i < 8; i++) {
 | |
|     x = beatsin8(SEGMENT.custom1>>3,   0, cols - 1, 0, ((i % 2) ? 128 : 0) + t * i);
 | |
|     y = beatsin8(SEGMENT.intensity>>3, 0, rows - 1, 0, ((i % 2) ? 192 : 64) + t * i);
 | |
|     leds[XY(x,y)] += CHSV(i*32, 255, 255);
 | |
|   }
 | |
|   // inner stars
 | |
|   for (byte i = 0; i < 4; i++) {
 | |
|     x = beatsin8(SEGMENT.custom2>>3, cols/4, cols - 1 - cols/4, 0, ((i % 2) ? 128 : 0) + t * i);
 | |
|     y = beatsin8(SEGMENT.custom3>>3, rows/4, rows - 1 - rows/4, 0, ((i % 2) ? 192 : 64) + t * i);
 | |
|     leds[XY(x,y)] += CHSV(i*32, 255, 255);
 | |
|   }
 | |
|   // central white dot
 | |
|   leds[XY(cols/2,rows/2)] = CHSV(0,0,255);
 | |
|   // blur everything a bit
 | |
|   blur2d(leds, 16);
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2DBlackHole()
 | |
| static const char *_data_FX_MODE_2DBLACKHOLE PROGMEM = "2D Black Hole@Fade rate,Outer Y freq.,Outer X freq.,Inner X freq.,Inner Y freq.;;";
 | |
| 
 | |
| 
 | |
| ////////////////////////////
 | |
| //     2D Colored Bursts  //
 | |
| ////////////////////////////
 | |
| uint16_t WS2812FX::mode_2DColoredBursts() {              // By: ldirko   https://editor.soulmatelights.com/gallery/819-colored-bursts , modified by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) {
 | |
|     fill_solid(leds, CRGB::Black);
 | |
|     //for (uint16_t i = 0; i < w*h; i++) leds[i] = CRGB::Black;
 | |
|     SEGENV.aux0 = 0; // start with red hue
 | |
|   }
 | |
| 
 | |
|   bool dot = false;
 | |
|   bool grad = true;
 | |
| 
 | |
|   byte numLines = SEGMENT.intensity/16 + 1;
 | |
| 
 | |
|   SEGENV.aux0++;  // hue
 | |
|   fadeToBlackBy(leds, 40);
 | |
| 
 | |
|   for (byte i = 0; i < numLines; i++) {
 | |
|     byte x1 = beatsin8(2 + SEGMENT.speed/16, 0, (cols - 1));
 | |
|     byte x2 = beatsin8(1 + SEGMENT.speed/16, 0, (cols - 1));
 | |
|     byte y1 = beatsin8(5 + SEGMENT.speed/16, 0, (rows - 1), 0, i * 24);
 | |
|     byte y2 = beatsin8(3 + SEGMENT.speed/16, 0, (rows - 1), 0, i * 48 + 64);
 | |
|     CRGB color = ColorFromPalette(currentPalette, i * 255 / numLines + (SEGENV.aux0&0xFF), 255, LINEARBLEND);
 | |
| 
 | |
|     byte xsteps = abs8(x1 - y1) + 1;
 | |
|     byte ysteps = abs8(x2 - y2) + 1;
 | |
|     byte steps = xsteps >= ysteps ? xsteps : ysteps;
 | |
| 
 | |
|     for (byte i = 1; i <= steps; i++) {
 | |
|       byte dx = lerp8by8(x1, y1, i * 255 / steps);
 | |
|       byte dy = lerp8by8(x2, y2, i * 255 / steps);
 | |
|       int index = XY(dx, dy);
 | |
|       leds[index] += color;           // change to += for brightness look
 | |
|       if (grad) leds[index] %= (i * 255 / steps); //Draw gradient line
 | |
|     }
 | |
| 
 | |
|     if (dot) { //add white point at the ends of line
 | |
|       leds[XY(x1, x2)] += CRGB::White;
 | |
|       leds[XY(y1, y2)] += CRGB::White;
 | |
|     }
 | |
|   }
 | |
|   blur2d(leds, 4);
 | |
| 
 | |
|   setPixels(leds);       // Use this ONLY if we're going to display via leds[x] method.
 | |
|   return FRAMETIME;
 | |
| } // mode_2DColoredBursts()
 | |
| static const char *_data_FX_MODE_2DCOLOREDBURSTS PROGMEM = "2D Colored Bursts@Speed,# of lines;;!";
 | |
| 
 | |
| 
 | |
| /////////////////////
 | |
| //      2D DNA     //
 | |
| /////////////////////
 | |
| uint16_t WS2812FX::mode_2Ddna(void) {         // dna originally by by ldirko at https://pastebin.com/pCkkkzcs. Updated by Preyy. WLED conversion by Andrew Tuline.
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, 0);
 | |
| 
 | |
|   fadeToBlackBy(leds, 64);
 | |
| 
 | |
|   for(int i = 0; i < cols; i++) {
 | |
|     leds[XY(i, beatsin8(SEGMENT.speed/8, 0, rows-1, 0, i*4))] = ColorFromPalette(currentPalette, i*5+millis()/17, beatsin8(5, 55, 255, 0, i*10), LINEARBLEND);
 | |
|     leds[XY(i, beatsin8(SEGMENT.speed/8, 0, rows-1, 0, i*4+128))] = ColorFromPalette(currentPalette,i*5+128+millis()/17, beatsin8(5, 55, 255, 0, i*10+128), LINEARBLEND); // 180 degrees (128) out of phase
 | |
|   }
 | |
|   blur2d(leds, SEGMENT.intensity/8);
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2Ddna()
 | |
| static const char *_data_FX_MODE_2DDNA PROGMEM = "2D DNA@Scroll speed,Blur;;!";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     2D DNA Spiral   //
 | |
| /////////////////////////
 | |
| uint16_t WS2812FX::mode_2DDNASpiral() {               // By: ldirko  https://editor.soulmatelights.com/gallery/810 , modified by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) {
 | |
|     fill_solid(leds, CRGB::Black);
 | |
|     SEGENV.aux0 = 0; // hue
 | |
|   }
 | |
| 
 | |
|   uint8_t speeds = SEGMENT.speed/2;
 | |
|   uint8_t freq = SEGMENT.intensity/8;
 | |
| 
 | |
|   uint32_t ms = millis() / 20;
 | |
|   nscale8(leds, 120);
 | |
| 
 | |
|   for (uint16_t i = 0; i < rows; i++) {
 | |
|     uint16_t x  = beatsin8(speeds, 0, cols - 1, 0, i * freq) + beatsin8(speeds - 7, 0, cols - 1, 0, i * freq + 128);
 | |
|     uint16_t x1 = beatsin8(speeds, 0, cols - 1, 0, 128 + i * freq) + beatsin8(speeds - 7, 0, cols - 1, 0, 128 + 64 + i * freq);
 | |
|     SEGENV.aux0 = i * 128 / cols + ms; //ewowi20210629: not width - 1 to avoid crash if width = 1
 | |
|     if ((i + ms / 8) & 3) {
 | |
|       x = x / 2; x1 = x1 / 2;
 | |
|       byte steps = abs8(x - x1) + 1;
 | |
|       for (byte k = 1; k <= steps; k++) {
 | |
|         byte dx = lerp8by8(x, x1, k * 255 / steps);
 | |
|         uint16_t index = XY(dx, i);
 | |
|         leds[index] += ColorFromPalette(currentPalette, SEGENV.aux0, 255, LINEARBLEND);
 | |
|         leds[index] %= (k * 255 / steps); //for draw gradient line
 | |
|       }
 | |
|       leds[XY(x, i)]  += CRGB::DarkSlateGray;
 | |
|       leds[XY(x1, i)] += CRGB::White;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   setPixels(leds);       // Use this ONLY if we're going to display via leds[x] method.
 | |
|   return FRAMETIME;
 | |
| } // mode_2DDNASpiral()
 | |
| static const char *_data_FX_MODE_2DDNASPIRAL PROGMEM = "2D DNA Spiral@Scroll speed,Blur;;!";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     2D Drift        //
 | |
| /////////////////////////
 | |
| uint16_t WS2812FX::mode_2DDrift() {              // By: Stepko   https://editor.soulmatelights.com/gallery/884-drift , Modified by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   //if (cols<8 || rows<8) return mode_static(); // makes no sense to run on smaller than 8x8
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   fadeToBlackBy(leds, 128);
 | |
| 
 | |
|   const uint16_t maxDim = MAX(cols, rows)/2;
 | |
|   unsigned long t = millis() / (32 - (SEGMENT.speed>>3));
 | |
|   for (float i = 1; i < maxDim; i += 0.25) {
 | |
|     float angle = radians(t * (maxDim - i));
 | |
|     uint16_t myX = (cols>>1) + (uint16_t)(sin_t(angle) * i) + (cols%2);
 | |
|     uint16_t myY = (rows>>1) + (uint16_t)(cos_t(angle) * i) + (rows%2);
 | |
|     leds[XY(myX,myY)] = ColorFromPalette(currentPalette, (i * 20) + (t / 20), 255, LINEARBLEND);
 | |
|   }
 | |
|   blur2d(leds, SEGMENT.intensity>>3);
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2DDrift()
 | |
| static const char *_data_FX_MODE_2DDRIFT PROGMEM = "2D Drift@Rotation speed,Blur amount;;!";
 | |
| 
 | |
| 
 | |
| //////////////////////////
 | |
| //     2D Firenoise     //
 | |
| //////////////////////////
 | |
| uint16_t WS2812FX::mode_2Dfirenoise(void) {               // firenoise2d. By Andrew Tuline. Yet another short routine.
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   uint16_t xscale = SEGMENT.intensity*4;
 | |
|   uint32_t yscale = SEGMENT.speed*8;
 | |
|   uint8_t indexx = 0;
 | |
| 
 | |
|   currentPalette = CRGBPalette16( CRGB(0,0,0), CRGB(0,0,0), CRGB(0,0,0), CRGB(0,0,0),
 | |
|                                   CRGB::Red, CRGB::Red, CRGB::Red, CRGB::DarkOrange,
 | |
|                                   CRGB::DarkOrange,CRGB::DarkOrange, CRGB::Orange, CRGB::Orange,
 | |
|                                   CRGB::Yellow, CRGB::Orange, CRGB::Yellow, CRGB::Yellow);
 | |
| 
 | |
|   for (uint16_t j=0; j < cols; j++) {
 | |
|     for (uint16_t i=0; i < rows; i++) {
 | |
|       indexx = inoise8(j*yscale*rows/255, i*xscale+millis()/4);                                           // We're moving along our Perlin map.
 | |
|       leds[XY(j,i)] = ColorFromPalette(currentPalette, min(i*(indexx)>>4, 255), i*255/cols, LINEARBLEND); // With that value, look up the 8 bit colour palette value and assign it to the current LED.
 | |
|     } // for i
 | |
|   } // for j
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2Dfirenoise()
 | |
| static const char *_data_FX_MODE_2DFIRENOISE PROGMEM = "2D Firenoise@X scale,Y scale;;";
 | |
| 
 | |
| 
 | |
| //////////////////////////////
 | |
| //     2D Frizzles          //
 | |
| //////////////////////////////
 | |
| uint16_t WS2812FX::mode_2DFrizzles(void) {                 // By: Stepko https://editor.soulmatelights.com/gallery/640-color-frizzles , Modified by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   fadeToBlackBy(leds, 16);
 | |
|   for (byte i = 8; i > 0; i--) {
 | |
|     leds[XY(beatsin8(SEGMENT.speed/8 + i, 0, cols - 1), beatsin8(SEGMENT.intensity/8 - i, 0, rows - 1))] += ColorFromPalette(currentPalette, beatsin8(12, 0, 255), 255, LINEARBLEND);
 | |
|   }
 | |
|   blur2d(leds, 16);
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2DFrizzles()
 | |
| static const char *_data_FX_MODE_2DFRIZZLES PROGMEM = "2D Frizzles@X frequency,Y frequency;;!";
 | |
| 
 | |
| 
 | |
| ///////////////////////////////////////////
 | |
| //   2D Cellular Automata Game of life   //
 | |
| ///////////////////////////////////////////
 | |
| typedef struct ColorCount {
 | |
|   CRGB color;
 | |
|   int8_t  count;
 | |
| } colorCount;
 | |
| 
 | |
| uint16_t WS2812FX::mode_2Dgameoflife(void) { // Written by Ewoud Wijma, inspired by https://natureofcode.com/book/chapter-7-cellular-automata/ and https://github.com/DougHaber/nlife-color
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize*2 + sizeof(unsigned long))) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
|   CRGB *prevLeds = reinterpret_cast<CRGB*>(SEGENV.data + dataSize);
 | |
|   unsigned long *resetMillis = reinterpret_cast<unsigned long*>(SEGENV.data + 2*dataSize); // triggers reset
 | |
| 
 | |
|   CRGB backgroundColor = SEGCOLOR(1);
 | |
| 
 | |
|   if (SEGENV.call == 0 || now - *resetMillis > 5000) {
 | |
|     *resetMillis = now;
 | |
| 
 | |
|     random16_set_seed(now); //seed the random generator
 | |
| 
 | |
|     //give the leds random state and colors (based on intensity, colors from palette or all posible colors are chosen)
 | |
|     for (int x = 0; x < cols; x++) for (int y = 0; y < rows; y++) {
 | |
|       uint8_t state = random8()%2;
 | |
|       if (state == 0)
 | |
|         leds[XY(x,y)] = backgroundColor;
 | |
|       else
 | |
|         leds[XY(x,y)] = (CRGB)color_from_palette(random8(), false, PALETTE_SOLID_WRAP, 0);
 | |
|     }
 | |
| 
 | |
|     fill_solid(prevLeds, CRGB::Black);
 | |
| 
 | |
|     SEGENV.aux1 = 0;
 | |
|     SEGENV.aux0 = 0xFFFF;
 | |
|   }
 | |
| 
 | |
|   //copy previous leds (save previous generation)
 | |
|   for (int x = 0; x < cols; x++) for (int y = 0; y < rows; y++) prevLeds[XY(x,y)] = leds[XY(x,y)];
 | |
| 
 | |
|   //calculate new leds
 | |
|   for (int x = 0; x < cols; x++) for (int y = 0; y < rows; y++) {
 | |
|     colorCount colorsCount[9];//count the different colors in the 9*9 matrix
 | |
|     for (int i=0; i<9; i++) colorsCount[i] = {backgroundColor, 0}; //init colorsCount
 | |
| 
 | |
|     //iterate through neighbors and count them and their different colors
 | |
|     int neighbors = 0;
 | |
|     for (int i = -1; i <= 1; i++) for (int j = -1; j <= 1; j++) { //iterate through 9*9 matrix
 | |
|       // wrap around segment
 | |
|       int16_t xx = x+i, yy = y+j;
 | |
|       if (x+i < 0) xx = cols-1; else if (x+i >= cols)  xx = 0;
 | |
|       if (y+j < 0) yy = rows-1; else if (y+j >= rows) yy = 0;
 | |
|       uint16_t xy = XY(xx, yy); // previous cell xy to check
 | |
| 
 | |
|       // count different neighbours and colors, except the centre cell
 | |
|       if (xy != XY(x,y) && prevLeds[xy] != backgroundColor) {
 | |
|         neighbors++;
 | |
|         bool colorFound = false;
 | |
|         int k;
 | |
|         for (k=0; k<9 && colorsCount[i].count != 0; k++)
 | |
|           if (colorsCount[k].color == prevLeds[xy]) {
 | |
|             colorsCount[k].count++;
 | |
|             colorFound = true;
 | |
|           }
 | |
| 
 | |
|         if (!colorFound) colorsCount[k] = {prevLeds[xy], 1}; //add new color found in the array
 | |
|       }
 | |
|     } // i,j
 | |
| 
 | |
|     // Rules of Life
 | |
|     if      ((leds[XY(x,y)] != backgroundColor) && (neighbors <  2)) leds[XY(x,y)] = backgroundColor; // Loneliness
 | |
|     else if ((leds[XY(x,y)] != backgroundColor) && (neighbors >  3)) leds[XY(x,y)] = backgroundColor; // Overpopulation
 | |
|     else if ((leds[XY(x,y)] == backgroundColor) && (neighbors == 3)) {                                // Reproduction
 | |
|       //find dominantcolor and assign to cell
 | |
|       colorCount dominantColorCount = {backgroundColor, 0};
 | |
|       for (int i=0; i<9 && colorsCount[i].count != 0; i++)
 | |
|         if (colorsCount[i].count > dominantColorCount.count) dominantColorCount = colorsCount[i];
 | |
|       if (dominantColorCount.count > 0) leds[XY(x,y)] = dominantColorCount.color; //assign the dominant color
 | |
|     }
 | |
|     // else do nothing!
 | |
|   } //x,y
 | |
| 
 | |
|   // calculate CRC16 of leds[]
 | |
|   uint16_t crc = crc16((const unsigned char*)leds, dataSize-1);
 | |
| 
 | |
|   // check if we had same CRC and reset if needed
 | |
|   // same CRC would mean image did not change or was repeating itself
 | |
|   if (!(crc == SEGENV.aux0 || crc == SEGENV.aux1)) *resetMillis = now; //if no repetition avoid reset
 | |
|   // remeber last two
 | |
|   SEGENV.aux1 = SEGENV.aux0;
 | |
|   SEGENV.aux0 = crc;
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return (SEGMENT.getOption(SEG_OPTION_TRANSITIONAL)) ? FRAMETIME : FRAMETIME_FIXED * (128-(SEGMENT.speed>>1)); // update only when appropriate time passes (in 42 FPS slots)
 | |
| } // mode_2Dgameoflife()
 | |
| static const char *_data_FX_MODE_2DGAMEOFLIFE PROGMEM = "2D Game Of Life@!,;!,!;!";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     2D Hiphotic     //
 | |
| /////////////////////////
 | |
| uint16_t WS2812FX::mode_2DHiphotic() {                        //  By: ldirko  https://editor.soulmatelights.com/gallery/810 , Modified by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint32_t a = now / 8;
 | |
| 
 | |
|   for (uint16_t x = 0; x < cols; x++) {
 | |
|     for (uint16_t y = 0; y < rows; y++) {
 | |
|       setPixelColorXY(x, y, color_from_palette(sin8(cos8(x * SEGMENT.speed/16 + a / 3) + sin8(y * SEGMENT.intensity/16 + a / 4) + a), false, PALETTE_SOLID_WRAP, 0));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_2DHiphotic()
 | |
| static const char *_data_FX_MODE_2DHIPHOTIC PROGMEM = "2D Hiphotic@X scale,Y scale;;!";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     2D Julia        //
 | |
| /////////////////////////
 | |
| // Sliders are:
 | |
| // intensity = Maximum number of iterations per pixel.
 | |
| // Custom1 = Location of X centerpoint
 | |
| // Custom2 = Location of Y centerpoint
 | |
| // Custom3 = Size of the area (small value = smaller area)
 | |
| typedef struct Julia {
 | |
|   float xcen;
 | |
|   float ycen;
 | |
|   float xymag;
 | |
| } julia;
 | |
| 
 | |
| uint16_t WS2812FX::mode_2DJulia(void) {                           // An animated Julia set by Andrew Tuline.
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
| 
 | |
|   if (!SEGENV.allocateData(sizeof(julia))) return mode_static();
 | |
|   Julia* julias = reinterpret_cast<Julia*>(SEGENV.data);
 | |
| 
 | |
|   float reAl;
 | |
|   float imAg;
 | |
| 
 | |
|   if (SEGENV.call == 0) {           // Reset the center if we've just re-started this animation.
 | |
|     julias->xcen = 0.;
 | |
|     julias->ycen = 0.;
 | |
|     julias->xymag = 1.0;
 | |
| 
 | |
|     SEGMENT.custom1 = 128;              // Make sure the location widgets are centered to start.
 | |
|     SEGMENT.custom2 = 128;
 | |
|     SEGMENT.custom3 = 128;
 | |
|     SEGMENT.intensity = 24;
 | |
|   }
 | |
| 
 | |
|   julias->xcen = julias->xcen + (float)(SEGMENT.custom1 - 128)/100000.;
 | |
|   julias->ycen = julias->ycen + (float)(SEGMENT.custom2 - 128)/100000.;
 | |
|   julias->xymag = julias->xymag + (float)(SEGMENT.custom3-128)/100000.;
 | |
|   if (julias->xymag < 0.01) julias->xymag = 0.01;
 | |
|   if (julias->xymag > 1.0) julias->xymag = 1.0;
 | |
| 
 | |
|   float xmin = julias->xcen - julias->xymag;
 | |
|   float xmax = julias->xcen + julias->xymag;
 | |
|   float ymin = julias->ycen - julias->xymag;
 | |
|   float ymax = julias->ycen + julias->xymag;
 | |
| 
 | |
|   // Whole set should be within -1.2,1.2 to -.8 to 1.
 | |
|   xmin = constrain(xmin,-1.2,1.2);
 | |
|   xmax = constrain(xmax,-1.2,1.2);
 | |
|   ymin = constrain(ymin,-.8,1.0);
 | |
|   ymax = constrain(ymax,-.8,1.0);
 | |
| 
 | |
|   float dx;                       // Delta x is mapped to the matrix size.
 | |
|   float dy;                       // Delta y is mapped to the matrix size.
 | |
| 
 | |
|   int maxIterations = 15;         // How many iterations per pixel before we give up. Make it 8 bits to match our range of colours.
 | |
|   float maxCalc = 16.0;           // How big is each calculation allowed to be before we give up.
 | |
| 
 | |
|   maxIterations = SEGMENT.intensity/2;
 | |
| 
 | |
| 
 | |
|   // Resize section on the fly for some animaton.
 | |
|   reAl = -0.94299;                // PixelBlaze example
 | |
|   imAg = 0.3162;
 | |
| 
 | |
|   reAl += sin((float)millis()/305.)/20.;
 | |
|   imAg += sin((float)millis()/405.)/20.;
 | |
| 
 | |
|   dx = (xmax - xmin) / (cols);     // Scale the delta x and y values to our matrix size.
 | |
|   dy = (ymax - ymin) / (rows);
 | |
| 
 | |
|   // Start y
 | |
|   float y = ymin;
 | |
|   for (int j = 0; j < rows; j++) {
 | |
| 
 | |
|     // Start x
 | |
|     float x = xmin;
 | |
|     for (int i = 0; i < cols; i++) {
 | |
| 
 | |
|       // Now we test, as we iterate z = z^2 + c does z tend towards infinity?
 | |
|       float a = x;
 | |
|       float b = y;
 | |
|       int iter = 0;
 | |
| 
 | |
|       while (iter < maxIterations) {    // Here we determine whether or not we're out of bounds.
 | |
|         float aa = a * a;
 | |
|         float bb = b * b;
 | |
|         float len = aa + bb;
 | |
|         if (len > maxCalc) {            // |z| = sqrt(a^2+b^2) OR z^2 = a^2+b^2 to save on having to perform a square root.
 | |
|           break;  // Bail
 | |
|         }
 | |
| 
 | |
|        // This operation corresponds to z -> z^2+c where z=a+ib c=(x,y). Remember to use 'foil'.
 | |
|         b = 2*a*b + imAg;
 | |
|         a = aa - bb + reAl;
 | |
|         iter++;
 | |
|       } // while
 | |
| 
 | |
|       // We color each pixel based on how long it takes to get to infinity, or black if it never gets there.
 | |
|       if (iter == maxIterations) {
 | |
|         setPixelColorXY(i, j, 0);
 | |
|       } else {
 | |
|         setPixelColorXY(i, j, color_from_palette(iter*255/maxIterations, false, PALETTE_SOLID_WRAP, 0));
 | |
|       }
 | |
|       x += dx;
 | |
|     }
 | |
|     y += dy;
 | |
|   }
 | |
| //  blur2d( leds, 64);
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_2DJulia()
 | |
| static const char *_data_FX_MODE_2DJULIA PROGMEM = "2D Julia@,Max iterations per pixel,X center,Y center,Area size;;!";
 | |
| 
 | |
| 
 | |
| //////////////////////////////
 | |
| //     2D Lissajous         //
 | |
| //////////////////////////////
 | |
| uint16_t WS2812FX::mode_2DLissajous(void) {            // By: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
| 
 | |
|   //uint16_t dataSize = sizeof(CRGB) * rows * cols;
 | |
| 
 | |
|   //if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   //CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   //if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   //fadeToBlackBy(leds, SEGMENT.intensity);
 | |
|   fade_out(SEGMENT.intensity);
 | |
| 
 | |
|   for (int i=0; i < 256; i ++) {
 | |
|     uint8_t xlocn = sin8(now/2+i*SEGMENT.speed/64);
 | |
|     uint8_t ylocn = cos8(now/2+i*128/64);
 | |
| 
 | |
|     xlocn = map(xlocn,0,255,0,cols-1);
 | |
|     ylocn = map(ylocn,0,255,0,rows-1);
 | |
|     //leds[XY(xlocn,ylocn)] = ColorFromPalette(currentPalette, now/100+i, 255, LINEARBLEND);
 | |
|     //setPixelColorXY(xlocn, ylocn, crgb_to_col(ColorFromPalette(currentPalette, now/100+i, 255, LINEARBLEND)));
 | |
|     setPixelColorXY(xlocn, ylocn, color_from_palette(now/100+i, false, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   //setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2DLissajous()
 | |
| static const char *_data_FX_MODE_2DLISSAJOUS PROGMEM = "2D Lissajous@X frequency,Fadetime;!,!,!;!";
 | |
| 
 | |
| 
 | |
| ///////////////////////
 | |
| //    2D Matrix      //
 | |
| ///////////////////////
 | |
| uint16_t WS2812FX::mode_2Dmatrix(void) {                  // Matrix2D. By Jeremy Williams. Adapted by Andrew Tuline & improved by merkisoft and ewowi.
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   uint8_t fade = map(SEGMENT.custom1, 0, 255, 50, 250);    // equals trail size
 | |
|   uint8_t speed = (256-SEGMENT.speed) >> map(MIN(rows, 150), 0, 150, 0, 3);    // slower speeds for small displays
 | |
| 
 | |
|   CRGB spawnColor;
 | |
|   CRGB trailColor;
 | |
|   if (SEGMENT.custom2 > 128) {
 | |
|     spawnColor = SEGCOLOR(0);
 | |
|     trailColor = SEGCOLOR(1);
 | |
|   } else {
 | |
|     spawnColor = CRGB(175,255,175);
 | |
|     trailColor = CRGB(27,130,39);
 | |
|   }
 | |
| 
 | |
|   if (now - SEGENV.step >= speed) {
 | |
|     SEGENV.step = now;
 | |
|     for (int16_t row=rows-1; row>=0; row--) {
 | |
|       for (int16_t col=0; col<cols; col++) {
 | |
|         if (leds[XY(col, row)] == spawnColor) {
 | |
|           leds[XY(col, row)] = trailColor;         // create trail
 | |
|           if (row < rows-1) leds[XY(col, row+1)] = spawnColor;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // fade all leds
 | |
|     for (int x=0; x<cols; x++) for (int y=0; y<rows; y++) {
 | |
|       if (leds[XY(x,y)] != spawnColor) leds[XY(x,y)].nscale8(fade);         // only fade trail
 | |
|     }
 | |
| 
 | |
|     // check for empty screen to ensure code spawn
 | |
|     bool emptyScreen = true;
 | |
|     for (uint16_t x=0; x<cols; x++) for (uint16_t y=0; y<rows; y++) {
 | |
|       if (leds[XY(x,y)]) {
 | |
|         emptyScreen = false;
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // spawn new falling code
 | |
|     if (random8() < SEGMENT.intensity || emptyScreen) {
 | |
|       uint8_t spawnX = random8(cols);
 | |
|       leds[XY(spawnX, 0)] = spawnColor;
 | |
|     }
 | |
| 
 | |
|     setPixels(leds);
 | |
|   } // if millis
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_2Dmatrix()
 | |
| static const char *_data_FX_MODE_2DMATRIX PROGMEM = "2D Matrix@Falling speed,Spawning rate,Trail,Custom color;Spawn,Trail;";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     2D Metaballs    //
 | |
| /////////////////////////
 | |
| uint16_t WS2812FX::mode_2Dmetaballs(void) {   // Metaballs by Stefan Petrick. Cannot have one of the dimensions be 2 or less. Adapted by Andrew Tuline.
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
| 
 | |
|   float speed = 0.25f * (1+(SEGMENT.speed>>6));
 | |
| 
 | |
|   // get some 2 random moving points
 | |
|   uint8_t x2 = inoise8(now * speed, 25355, 685 ) / 16;
 | |
|   uint8_t y2 = inoise8(now * speed, 355, 11685 ) / 16;
 | |
| 
 | |
|   uint8_t x3 = inoise8(now * speed, 55355, 6685 ) / 16;
 | |
|   uint8_t y3 = inoise8(now * speed, 25355, 22685 ) / 16;
 | |
| 
 | |
|   // and one Lissajou function
 | |
|   uint8_t x1 = beatsin8(23 * speed, 0, 15);
 | |
|   uint8_t y1 = beatsin8(28 * speed, 0, 15);
 | |
| 
 | |
|   for (uint16_t y = 0; y < rows; y++) {
 | |
|     for (uint16_t x = 0; x < cols; x++) {
 | |
|       // calculate distances of the 3 points from actual pixel
 | |
|       // and add them together with weightening
 | |
|       uint16_t dx = abs(x - x1);
 | |
|       uint16_t dy = abs(y - y1);
 | |
|       uint16_t dist = 2 * sqrt16((dx * dx) + (dy * dy));
 | |
| 
 | |
|       dx = abs(x - x2);
 | |
|       dy = abs(y - y2);
 | |
|       dist += sqrt16((dx * dx) + (dy * dy));
 | |
| 
 | |
|       dx = abs(x - x3);
 | |
|       dy = abs(y - y3);
 | |
|       dist += sqrt16((dx * dx) + (dy * dy));
 | |
| 
 | |
|       // inverse result
 | |
|       byte color = 1000 / dist;
 | |
| 
 | |
|       // map color between thresholds
 | |
|       if (color > 0 and color < 60) {
 | |
|         setPixelColorXY(x, y, color_from_palette(map(color * 9, 9, 531, 0, 255), false, PALETTE_SOLID_WRAP, 0));
 | |
|       } else {
 | |
|         setPixelColorXY(x, y, color_from_palette(0, false, PALETTE_SOLID_WRAP, 0));
 | |
|       }
 | |
|       // show the 3 points, too
 | |
|       setPixelColorXY(x1, y1, CRGB::White);
 | |
|       setPixelColorXY(x2, y2, CRGB::White);
 | |
|       setPixelColorXY(x3, y3, CRGB::White);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2Dmetaballs()
 | |
| static const char *_data_FX_MODE_2DMETABALLS PROGMEM = "2D Metaballs@Speed;!,!,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //    2D Noise      //
 | |
| //////////////////////
 | |
| uint16_t WS2812FX::mode_2Dnoise(void) {                  // By Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
| 
 | |
|   const uint16_t scale  = SEGMENT.intensity+2;
 | |
| 
 | |
|   for (uint16_t y = 0; y < rows; y++) {
 | |
|     for (uint16_t x = 0; x < cols; x++) {
 | |
|       uint8_t pixelHue8 = inoise8(x * scale, y * scale, millis() / (16 - SEGMENT.speed/16));
 | |
|       setPixelColorXY(x, y, crgb_to_col(ColorFromPalette(currentPalette, pixelHue8)));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_2Dnoise()
 | |
| static const char *_data_FX_MODE_2DNOISE PROGMEM = "2D Noise@Speed,Scale;!,!,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////////////
 | |
| //     2D Plasma Ball       //
 | |
| //////////////////////////////
 | |
| uint16_t WS2812FX::mode_2DPlasmaball(void) {                   // By: Stepko https://editor.soulmatelights.com/gallery/659-plasm-ball , Modified by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   fadeToBlackBy(leds, 64);
 | |
|   float t = millis() / (33 - SEGMENT.speed/8);
 | |
|   for (uint16_t i = 0; i < cols; i++) {
 | |
|     uint16_t thisVal = inoise8(i * 30, t, t);
 | |
|     uint16_t thisMax = map(thisVal, 0, 255, 0, cols-1);
 | |
|     for (uint16_t j = 0; j < rows; j++) {
 | |
|       uint16_t thisVal_ = inoise8(t, j * 30, t);
 | |
|       uint16_t thisMax_ = map(thisVal_, 0, 255, 0, rows-1);
 | |
|       uint16_t x = (i + thisMax_ - cols / 2);
 | |
|       uint16_t y = (j + thisMax - cols / 2);
 | |
|       uint16_t cx = (i + thisMax_);
 | |
|       uint16_t cy = (j + thisMax);
 | |
| 
 | |
|       leds[XY(i, j)] += ((x - y > -2) && (x - y < 2)) ||
 | |
|                         ((cols - 1 - x - y) > -2 && (cols - 1 - x - y < 2)) ||
 | |
|                         (cols - cx == 0) ||
 | |
|                         (cols - 1 - cx == 0) ||
 | |
|                         ((rows - cy == 0) ||
 | |
|                         (rows - 1 - cy == 0)) ? ColorFromPalette(currentPalette, beat8(5), thisVal, LINEARBLEND) : CRGB::Black;
 | |
|     }
 | |
|   }
 | |
|   blur2d(leds, 4);
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2DPlasmaball()
 | |
| static const char *_data_FX_MODE_2DPLASMABALL PROGMEM = "2D Plasma Ball@Speed;!,!,!;!";
 | |
| 
 | |
| 
 | |
| ////////////////////////////////
 | |
| //  2D Polar Lights           //
 | |
| ////////////////////////////////
 | |
| //static float fmap(const float x, const float in_min, const float in_max, const float out_min, const float out_max) {
 | |
| //  return (out_max - out_min) * (x - in_min) / (in_max - in_min) + out_min;
 | |
| //}
 | |
| uint16_t WS2812FX::mode_2DPolarLights(void) {        // By: Kostyantyn Matviyevskyy  https://editor.soulmatelights.com/gallery/762-polar-lights , Modified by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   CRGBPalette16 auroraPalette  = {0x000000, 0x003300, 0x006600, 0x009900, 0x00cc00, 0x00ff00, 0x33ff00, 0x66ff00, 0x99ff00, 0xccff00, 0xffff00, 0xffcc00, 0xff9900, 0xff6600, 0xff3300, 0xff0000};
 | |
| 
 | |
|   if (SEGENV.call == 0) {
 | |
|     SEGENV.step = 0;
 | |
|     fill_solid(leds, CRGB::Black);
 | |
|   }
 | |
| 
 | |
|   float adjustHeight = (float)map(rows, 8, 32, 28, 12);
 | |
|   uint16_t adjScale = map(cols, 8, 64, 310, 63);
 | |
| /*
 | |
|   if (SEGENV.aux1 != SEGMENT.custom1/12) {   // Hacky palette rotation. We need that black.
 | |
|     SEGENV.aux1 = SEGMENT.custom1/12;
 | |
|     for (int i = 0; i < 16; i++) {
 | |
|       long ilk;
 | |
|       ilk = (long)currentPalette[i].r << 16;
 | |
|       ilk += (long)currentPalette[i].g << 8;
 | |
|       ilk += (long)currentPalette[i].b;
 | |
|       ilk = (ilk << SEGENV.aux1) | (ilk >> (24 - SEGENV.aux1));
 | |
|       currentPalette[i].r = ilk >> 16;
 | |
|       currentPalette[i].g = ilk >> 8;
 | |
|       currentPalette[i].b = ilk;
 | |
|     }
 | |
|   }
 | |
| */
 | |
|   uint16_t _scale = map(SEGMENT.intensity, 0, 255, 30, adjScale);
 | |
|   byte _speed = map(SEGMENT.speed, 0, 255, 128, 16);
 | |
| 
 | |
|   for (uint16_t x = 0; x < cols; x++) {
 | |
|     for (uint16_t y = 0; y < rows; y++) {
 | |
|       SEGENV.step++;
 | |
|       leds[XY(x, y)] = ColorFromPalette(auroraPalette,
 | |
|                          qsub8(
 | |
|                            inoise8((SEGENV.step%2) + x * _scale, y * 16 + SEGENV.step % 16, SEGENV.step / _speed),
 | |
|                            fabs((float)rows / 2 - (float)y) * adjustHeight));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2DPolarLights()
 | |
| static const char *_data_FX_MODE_2DPOLARLIGHTS PROGMEM = "2D Polar Lights@Speed,Scale;;";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     2D Pulser       //
 | |
| /////////////////////////
 | |
| uint16_t WS2812FX::mode_2DPulser(void) {                       // By: ldirko   https://editor.soulmatelights.com/gallery/878-pulse-test , modifed by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   //const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   fadeToBlackBy(leds, 8 - (SEGMENT.intensity>>5));
 | |
| 
 | |
|   uint16_t a = now / (18 - SEGMENT.speed / 16);
 | |
|   uint16_t x = (a / 14);
 | |
|   uint16_t y = map((sin8(a * 5) + sin8(a * 4) + sin8(a * 2)), 0, 765, rows-1, 0);
 | |
|   uint16_t index = XY(x, y); // XY() will wrap x or y
 | |
|   leds[index] = ColorFromPalette(currentPalette, map(y, 0, rows-1, 0, 255), 255, LINEARBLEND);
 | |
|   blur2d(leds, 1 + (SEGMENT.intensity>>4));
 | |
| 
 | |
|   setPixels(leds);       // Use this ONLY if we're going to display via leds[x] method.
 | |
|   return FRAMETIME;
 | |
| } // mode_2DPulser()
 | |
| static const char *_data_FX_MODE_2DPULSER PROGMEM = "2D Pulser@Speed,Blur;;!";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     2D Sindots      //
 | |
| /////////////////////////
 | |
| uint16_t WS2812FX::mode_2DSindots(void) {                             // By: ldirko   https://editor.soulmatelights.com/gallery/597-sin-dots , modified by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   fadeToBlackBy(leds, 15);
 | |
|   byte t1 = millis() / (257 - SEGMENT.speed); // 20;
 | |
|   byte t2 = sin8(t1) / 4 * 2;
 | |
|   for (uint16_t i = 0; i < 13; i++) {
 | |
|     byte x = sin8(t1 + i * SEGMENT.intensity/8)*(cols-1)/255;  //   max index now 255x15/255=15!
 | |
|     byte y = sin8(t2 + i * SEGMENT.intensity/8)*(rows-1)/255;  //  max index now 255x15/255=15!
 | |
|     leds[XY(x, y)] = ColorFromPalette(currentPalette, i * 255 / 13, 255, LINEARBLEND);
 | |
|   }
 | |
|   blur2d(leds, 16);
 | |
| 
 | |
|   setPixels(leds);       // Use this ONLY if we're going to display via leds[x] method.
 | |
|   return FRAMETIME;
 | |
| } // mode_2DSindots()
 | |
| static const char *_data_FX_MODE_2DSINDOTS PROGMEM = "2D Sindots@Speed,Dot distance;;!";
 | |
| 
 | |
| 
 | |
| //////////////////////////////
 | |
| //     2D Squared Swirl     //
 | |
| //////////////////////////////
 | |
| // custom3 affects the blur amount.
 | |
| uint16_t WS2812FX::mode_2Dsquaredswirl(void) {            // By: Mark Kriegsman. https://gist.github.com/kriegsman/368b316c55221134b160
 | |
|                                                           // Modifed by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   const uint8_t kBorderWidth = 2;
 | |
| 
 | |
|   fadeToBlackBy(leds, 24);
 | |
|   uint8_t blurAmount = SEGMENT.custom3>>4;
 | |
|   blur2d(leds, blurAmount);
 | |
| 
 | |
|   // Use two out-of-sync sine waves
 | |
|   uint8_t i = beatsin8(19, kBorderWidth, cols-kBorderWidth);
 | |
|   uint8_t j = beatsin8(22, kBorderWidth, cols-kBorderWidth);
 | |
|   uint8_t k = beatsin8(17, kBorderWidth, cols-kBorderWidth);
 | |
|   uint8_t m = beatsin8(18, kBorderWidth, rows-kBorderWidth);
 | |
|   uint8_t n = beatsin8(15, kBorderWidth, rows-kBorderWidth);
 | |
|   uint8_t p = beatsin8(20, kBorderWidth, rows-kBorderWidth);
 | |
| 
 | |
|   uint16_t ms = millis();
 | |
| 
 | |
|   leds[XY(i, m)] += ColorFromPalette(currentPalette, ms/29, 255, LINEARBLEND);
 | |
|   leds[XY(j, n)] += ColorFromPalette(currentPalette, ms/41, 255, LINEARBLEND);
 | |
|   leds[XY(k, p)] += ColorFromPalette(currentPalette, ms/73, 255, LINEARBLEND);
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2Dsquaredswirl()
 | |
| static const char *_data_FX_MODE_2DSQUAREDSWIRL PROGMEM = "2D Squared Swirl@,,,,Blur;,,;!";
 | |
| 
 | |
| 
 | |
| //////////////////////////////
 | |
| //     2D Sun Radiation     //
 | |
| //////////////////////////////
 | |
| uint16_t WS2812FX::mode_2DSunradiation(void) {                   // By: ldirko https://editor.soulmatelights.com/gallery/599-sun-radiation  , modified by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize + (sizeof(byte)*(cols+2)*(rows+2)))) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
|   byte *bump = reinterpret_cast<byte*>(SEGENV.data + dataSize);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   unsigned long t = millis() / 4;
 | |
|   int index = 0;
 | |
|   uint8_t someVal = SEGMENT.speed/4;             // Was 25.
 | |
|   for (uint16_t j = 0; j < (rows + 2); j++) {
 | |
|     for (uint16_t i = 0; i < (cols + 2); i++) {
 | |
|       byte col = (inoise8_raw(i * someVal, j * someVal, t)) / 2;
 | |
|       bump[index++] = col;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   int yindex = cols + 3;
 | |
|   int16_t vly = -(rows / 2 + 1);
 | |
|   for (uint16_t y = 0; y < rows; y++) {
 | |
|     ++vly;
 | |
|     int16_t vlx = -(cols / 2 + 1);
 | |
|     for (uint16_t x = 0; x < cols; x++) {
 | |
|       ++vlx;
 | |
|       int8_t nx = bump[x + yindex + 1] - bump[x + yindex - 1];
 | |
|       int8_t ny = bump[x + yindex + (cols + 2)] - bump[x + yindex - (cols + 2)];
 | |
|       byte difx = abs8(vlx * 7 - nx);
 | |
|       byte dify = abs8(vly * 7 - ny);
 | |
|       int temp = difx * difx + dify * dify;
 | |
|       int col = 255 - temp / 8; //8 its a size of effect
 | |
|       if (col < 0) col = 0;
 | |
|       leds[XY(x, y)] = HeatColor(col / (3.0f-(float)(SEGMENT.intensity)/128.f));
 | |
|     }
 | |
|     yindex += (cols + 2);
 | |
|   }
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2DSunradiation()
 | |
| static const char *_data_FX_MODE_2DSUNRADIATION PROGMEM = "2D Sun Radiation@Variance,Brightness;;";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     2D Tartan       //
 | |
| /////////////////////////
 | |
| uint16_t WS2812FX::mode_2Dtartan(void) {          // By: Elliott Kember  https://editor.soulmatelights.com/gallery/3-tartan , Modified by: Andrew Tuline
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   uint8_t hue;
 | |
|   int offsetX = beatsin16(3, -360, 360);
 | |
|   int offsetY = beatsin16(2, -360, 360);
 | |
| 
 | |
|   for (uint16_t x = 0; x < cols; x++) {
 | |
|     for (uint16_t y = 0; y < rows; y++) {
 | |
|       uint16_t index = XY(x, y);
 | |
|       hue = x * beatsin16(10, 1, 10) + offsetY;
 | |
|       leds[index] = ColorFromPalette(currentPalette, hue, sin8(x * SEGMENT.speed + offsetX) * sin8(x * SEGMENT.speed + offsetX) / 255, LINEARBLEND);
 | |
|       hue = y * 3 + offsetX;
 | |
|       leds[index] += ColorFromPalette(currentPalette, hue, sin8(y * SEGMENT.intensity + offsetY) * sin8(y * SEGMENT.intensity + offsetY) / 255, LINEARBLEND);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   setPixels(leds);       // Use this ONLY if we're going to display via leds[x] method.
 | |
|   return FRAMETIME;
 | |
| } // mode_2DTartan()
 | |
| static const char *_data_FX_MODE_2DTARTAN PROGMEM = "2D Tartan@X scale,Y scale;;!";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     2D spaceships   //
 | |
| /////////////////////////
 | |
| uint16_t WS2812FX::mode_2Dspaceships(void) {    //// Space ships by stepko (c)05.02.21 [https://editor.soulmatelights.com/gallery/639-space-ships], adapted by Blaz Kristan
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   uint32_t tb = now >> 12;  // every ~4s
 | |
|   if (tb > SEGENV.step) {
 | |
|     int8_t dir = ++SEGENV.aux0;
 | |
|     dir  += (int)random8(3)-1;
 | |
|     if      (dir > 7) SEGENV.aux0 = 0;
 | |
|     else if (dir < 0) SEGENV.aux0 = 7;
 | |
|     else              SEGENV.aux0 = dir;
 | |
|     SEGENV.step = tb + random8(4);
 | |
|   }
 | |
| 
 | |
|   fadeToBlackBy(leds, map(SEGMENT.speed, 0, 255, 248, 16));
 | |
|   move(SEGENV.aux0, 1, leds);
 | |
|   for (byte i = 0; i < 8; i++) {
 | |
|     byte x = beatsin8(12 + i, 2, cols - 3);
 | |
|     byte y = beatsin8(15 + i, 2, rows - 3);
 | |
|     CRGB color = ColorFromPalette(currentPalette, beatsin8(12 + i, 0, 255), 255);
 | |
|     leds[XY(x, y)] += color;
 | |
|     if (cols > 24 || rows > 24) {
 | |
|       leds[XY(x + 1, y)] += color;
 | |
|       leds[XY(x - 1, y)] += color;
 | |
|       leds[XY(x, y + 1)] += color;
 | |
|       leds[XY(x, y - 1)] += color;
 | |
|     }
 | |
|   }
 | |
|   blur2d(leds, SEGMENT.intensity>>3);
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_SPACESHIPS PROGMEM = "2D Spaceships@!,Blur;!,!,!;!";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     2D Crazy Bees   //
 | |
| /////////////////////////
 | |
| //// Crazy bees by stepko (c)12.02.21 [https://editor.soulmatelights.com/gallery/651-crazy-bees], adapted by Blaz Kristan
 | |
| #define MAX_BEES 5
 | |
| uint16_t WS2812FX::mode_2Dcrazybees(void) {
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   byte n = MIN(MAX_BEES, (rows * cols) / 256 + 1);
 | |
| 
 | |
|   typedef struct Bee {
 | |
|     uint8_t posX, posY, aimX, aimY, hue;
 | |
|     int8_t deltaX, deltaY, signX, signY, error;
 | |
|     void aimed(uint16_t w, uint16_t h) {
 | |
|       randomSeed(millis());
 | |
|       aimX = random8(0, w);
 | |
|       aimY = random8(0, h);
 | |
|       hue = random8();
 | |
|       deltaX = abs(aimX - posX);
 | |
|       deltaY = abs(aimY - posY);
 | |
|       signX = posX < aimX ? 1 : -1;
 | |
|       signY = posY < aimY ? 1 : -1;
 | |
|       error = deltaX - deltaY;
 | |
|     };
 | |
|   } bee_t;
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize + sizeof(bee_t)*MAX_BEES)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
|   bee_t *bee = reinterpret_cast<bee_t*>(SEGENV.data + dataSize);
 | |
| 
 | |
|   if (SEGENV.call == 0) {
 | |
|     fill_solid(leds, CRGB::Black);
 | |
|     for (byte i = 0; i < n; i++) {
 | |
|       bee[i].posX = random8(0, cols);
 | |
|       bee[i].posY = random8(0, rows);
 | |
|       bee[i].aimed(cols, rows);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (millis() > SEGENV.step) {
 | |
|     SEGENV.step = millis() + (FRAMETIME * 8 / ((SEGMENT.speed>>5)+1));
 | |
| 
 | |
|     fadeToBlackBy(leds, 32);
 | |
|   
 | |
|     for (byte i = 0; i < n; i++) {
 | |
|       leds[XY(bee[i].aimX + 1, bee[i].aimY)] += CHSV(bee[i].hue, 255, 255);
 | |
|       leds[XY(bee[i].aimX, bee[i].aimY + 1)] += CHSV(bee[i].hue, 255, 255);
 | |
|       leds[XY(bee[i].aimX - 1, bee[i].aimY)] += CHSV(bee[i].hue, 255, 255);
 | |
|       leds[XY(bee[i].aimX, bee[i].aimY - 1)] += CHSV(bee[i].hue, 255, 255);
 | |
|       if (bee[i].posX != bee[i].aimX || bee[i].posY != bee[i].aimY) {
 | |
|         leds[XY(bee[i].posX, bee[i].posY)] = CHSV(bee[i].hue, 60, 255);
 | |
|         int8_t error2 = bee[i].error * 2;
 | |
|         if (error2 > -bee[i].deltaY) {
 | |
|           bee[i].error -= bee[i].deltaY;
 | |
|           bee[i].posX += bee[i].signX;
 | |
|         }
 | |
|         if (error2 < bee[i].deltaX) {
 | |
|           bee[i].error += bee[i].deltaX;
 | |
|           bee[i].posY += bee[i].signY;
 | |
|         }
 | |
|       } else {
 | |
|         bee[i].aimed(cols, rows);
 | |
|       }
 | |
|     }
 | |
|     blur2d(leds, SEGMENT.intensity>>4);
 | |
| 
 | |
|     setPixels(leds);
 | |
|   }
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_CRAZYBEES PROGMEM = "2D Crazy Bees@!,Blur;;";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     2D Ghost Rider  //
 | |
| /////////////////////////
 | |
| //// Ghost Rider by stepko (c)2021 [https://editor.soulmatelights.com/gallery/716-ghost-rider], adapted by Blaz Kristan
 | |
| #define LIGHTERS_AM 64  // max lighters (adequate for 32x32 matrix)
 | |
| uint16_t WS2812FX::mode_2Dghostrider(void) {
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   typedef struct Lighter {
 | |
|     int16_t  gPosX;
 | |
|     int16_t  gPosY;
 | |
|     uint16_t gAngle;
 | |
|     int8_t   angleSpeed;
 | |
|     uint16_t lightersPosX[LIGHTERS_AM];
 | |
|     uint16_t lightersPosY[LIGHTERS_AM];
 | |
|     uint16_t Angle[LIGHTERS_AM];
 | |
|     uint16_t time[LIGHTERS_AM];
 | |
|     bool     reg[LIGHTERS_AM];
 | |
|     int8_t   Vspeed;
 | |
|   } lighter_t;
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize + sizeof(lighter_t))) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
|   lighter_t *lighter = reinterpret_cast<lighter_t*>(SEGENV.data + dataSize);
 | |
| 
 | |
|   const int maxLighters = min(cols + rows, LIGHTERS_AM);
 | |
| 
 | |
|   if (SEGENV.call == 0 || SEGENV.aux0 != cols || SEGENV.aux1 != rows) {
 | |
|     SEGENV.aux0 = cols;
 | |
|     SEGENV.aux1 = rows;
 | |
|     fill_solid(leds, CRGB::Black);
 | |
|     randomSeed(now);
 | |
|     lighter->angleSpeed = random8(0,20) - 10;
 | |
|     lighter->Vspeed = 5;
 | |
|     lighter->gPosX = (cols/2) * 10;
 | |
|     lighter->gPosY = (rows/2) * 10;
 | |
|     for (byte i = 0; i < maxLighters; i++) {
 | |
|       lighter->lightersPosX[i] = lighter->gPosX;
 | |
|       lighter->lightersPosY[i] = lighter->gPosY + i;
 | |
|       lighter->time[i] = i * 2;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (millis() > SEGENV.step) {
 | |
|     SEGENV.step = millis() + 1024 / (cols+rows);
 | |
| 
 | |
|     fadeToBlackBy(leds, (SEGMENT.speed>>2)+64);
 | |
| 
 | |
|     CRGB color = CRGB::White;
 | |
|     wu_pixel(leds, lighter->gPosX * 256 / 10, lighter->gPosY * 256 / 10, color);
 | |
| 
 | |
|     lighter->gPosX += lighter->Vspeed * sin_t(radians(lighter->gAngle));
 | |
|     lighter->gPosY += lighter->Vspeed * cos_t(radians(lighter->gAngle));
 | |
|     lighter->gAngle += lighter->angleSpeed;
 | |
|     if (lighter->gPosX < 0)               lighter->gPosX = (cols - 1) * 10;
 | |
|     if (lighter->gPosX > (cols - 1) * 10) lighter->gPosX = 0;
 | |
|     if (lighter->gPosY < 0)               lighter->gPosY = (rows - 1) * 10;
 | |
|     if (lighter->gPosY > (rows - 1) * 10) lighter->gPosY = 0;
 | |
|     for (byte i = 0; i < maxLighters; i++) {
 | |
|       lighter->time[i] += random8(5, 20);
 | |
|       if (lighter->time[i] >= 255 ||
 | |
|         (lighter->lightersPosX[i] <= 0) ||
 | |
|           (lighter->lightersPosX[i] >= (cols - 1) * 10) ||
 | |
|           (lighter->lightersPosY[i] <= 0) ||
 | |
|           (lighter->lightersPosY[i] >= (rows - 1) * 10)) {
 | |
|         lighter->reg[i] = true;
 | |
|       }
 | |
|       if (lighter->reg[i]) {
 | |
|         lighter->lightersPosY[i] = lighter->gPosY;
 | |
|         lighter->lightersPosX[i] = lighter->gPosX;
 | |
|         lighter->Angle[i] = lighter->gAngle + random(-10, 10);
 | |
|         lighter->time[i] = 0;
 | |
|         lighter->reg[i] = false;
 | |
|       } else {
 | |
|         lighter->lightersPosX[i] += -7 * sin_t(radians(lighter->Angle[i]));
 | |
|         lighter->lightersPosY[i] += -7 * cos_t(radians(lighter->Angle[i]));
 | |
|       }
 | |
|       wu_pixel(leds, lighter->lightersPosX[i] * 256 / 10, lighter->lightersPosY[i] * 256 / 10, ColorFromPalette(currentPalette, (256 - lighter->time[i])));
 | |
|     }
 | |
|     blur2d(leds, SEGMENT.intensity>>3);
 | |
|   }
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_GHOST_RIDER PROGMEM = "2D Ghost Rider@Fade rate,Blur;!,!,!;!";
 | |
| 
 | |
| 
 | |
| ////////////////////////////
 | |
| //     2D Floating Blobs  //
 | |
| ////////////////////////////
 | |
| //// Floating Blobs by stepko (c)2021 [https://editor.soulmatelights.com/gallery/573-blobs], adapted by Blaz Kristan
 | |
| #define MAX_BLOBS 8
 | |
| uint16_t WS2812FX::mode_2Dfloatingblobs(void) {
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   typedef struct Blob {
 | |
|     float x[MAX_BLOBS], y[MAX_BLOBS];
 | |
|     float sX[MAX_BLOBS], sY[MAX_BLOBS]; // speed
 | |
|     float r[MAX_BLOBS];
 | |
|     bool grow[MAX_BLOBS];
 | |
|     byte color[MAX_BLOBS];
 | |
|   } blob_t;
 | |
| 
 | |
|   uint8_t Amount = (SEGMENT.intensity>>5) + 1; // NOTE: be sure to update MAX_BLOBS if you change this
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize + sizeof(blob_t))) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
|   blob_t *blob = reinterpret_cast<blob_t*>(SEGENV.data + dataSize);
 | |
| 
 | |
|   if (SEGENV.call == 0 || SEGENV.aux0 != cols || SEGENV.aux1 != rows) {
 | |
|     SEGENV.aux0 = cols;
 | |
|     SEGENV.aux1 = rows;
 | |
|     fill_solid(leds, CRGB::Black);
 | |
|     for (byte i = 0; i < MAX_BLOBS; i++) {
 | |
|       blob->r[i]  = cols>15 ? random8(1, cols/8.f) : 1;
 | |
|       blob->sX[i] = (float) random8(3, cols) / (float)(256 - SEGMENT.speed); // speed x
 | |
|       blob->sY[i] = (float) random8(3, rows) / (float)(256 - SEGMENT.speed); // speed y
 | |
|       blob->x[i]  = random8(0, cols-1);
 | |
|       blob->y[i]  = random8(0, rows-1);
 | |
|       blob->color[i] = random8();
 | |
|       blob->grow[i]  = (blob->r[i] < 1.f);
 | |
|       if (blob->sX[i] == 0) blob->sX[i] = 1;
 | |
|       if (blob->sY[i] == 0) blob->sY[i] = 1;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   fadeToBlackBy(leds, 20);
 | |
| 
 | |
|   // Bounce balls around
 | |
|   for (byte i = 0; i < Amount; i++) {
 | |
|     if (SEGENV.step < millis()) blob->color[i] = add8(blob->color[i], 4); // slowly change color
 | |
|     // change radius if needed
 | |
|     if (blob->grow[i]) {
 | |
|       // enlarge radius until it is >= 4
 | |
|       blob->r[i] += (fabs(blob->sX[i]) > fabs(blob->sY[i]) ? fabs(blob->sX[i]) : fabs(blob->sY[i])) * 0.05f;
 | |
|       if (blob->r[i] >= MIN(cols/8.f,2.f)) {
 | |
|         blob->grow[i] = false;
 | |
|       }
 | |
|     } else {
 | |
|       // reduce radius until it is < 1
 | |
|       blob->r[i] -= (fabs(blob->sX[i]) > fabs(blob->sY[i]) ? fabs(blob->sX[i]) : fabs(blob->sY[i])) * 0.05f;
 | |
|       if (blob->r[i] < 1.f) {
 | |
|         blob->grow[i] = true; 
 | |
|       }
 | |
|     }
 | |
|     CRGB c = ColorFromPalette(currentPalette, blob->color[i]);
 | |
|     //if (!SEGMENT.palette) c = SEGCOLOR(0);
 | |
|     if (blob->r[i] > 1.f) fill_circle(leds, blob->y[i], blob->x[i], blob->r[i], c);
 | |
|     else                  leds[XY(blob->y[i], blob->x[i])] += c;
 | |
|     // move x
 | |
|     if (blob->x[i] + blob->r[i] >= cols - 1) blob->x[i] += (blob->sX[i] * ((cols - 1 - blob->x[i]) / blob->r[i] + 0.005f));
 | |
|     else if (blob->x[i] - blob->r[i] <= 0)   blob->x[i] += (blob->sX[i] * (blob->x[i] / blob->r[i] + 0.005f));
 | |
|     else                                     blob->x[i] += blob->sX[i];
 | |
|     // move y
 | |
|     if (blob->y[i] + blob->r[i] >= rows - 1) blob->y[i] += (blob->sY[i] * ((rows - 1 - blob->y[i]) / blob->r[i] + 0.005f));
 | |
|     else if (blob->y[i] - blob->r[i] <= 0)   blob->y[i] += (blob->sY[i] * (blob->y[i] / blob->r[i] + 0.005f));
 | |
|     else                                     blob->y[i] += blob->sY[i];
 | |
|     // bounce x
 | |
|     if (blob->x[i] < 0.01f) {
 | |
|       blob->sX[i] = (float) random8(3, cols) / (256 - SEGMENT.speed);
 | |
|       blob->x[i]  = 0.01f;
 | |
|     } else if (blob->x[i] > cols - 1.01f) {
 | |
|       blob->sX[i] = (float) random8(3, cols) / (256 - SEGMENT.speed);
 | |
|       blob->sX[i] = -blob->sX[i];
 | |
|       blob->x[i]  = cols - 1.01f;
 | |
|     }
 | |
|     // bounce y
 | |
|     if (blob->y[i] < 0.01f) {
 | |
|       blob->sY[i] = (float) random8(3, rows) / (256 - SEGMENT.speed);
 | |
|       blob->y[i]  = 0.01f;
 | |
|     } else if (blob->y[i] > rows - 1.01f) {
 | |
|       blob->sY[i] = (float) random8(3, rows) / (256 - SEGMENT.speed);
 | |
|       blob->sY[i] = -blob->sY[i];
 | |
|       blob->y[i]  = rows - 1.01f;
 | |
|     }
 | |
|   }
 | |
|   blur2d(leds, cols+rows);
 | |
| 
 | |
|   if (SEGENV.step < millis()) SEGENV.step = millis() + 2000; // change colors every 2 seconds
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| }
 | |
| #undef MAX_BLOBS
 | |
| static const char *_data_FX_MODE_BLOBS PROGMEM = "2D Blobs@!,# blobs;!,!,!;!";
 | |
| 
 | |
| 
 | |
| ////////////////////////////
 | |
| //     2D Scrolling text  //
 | |
| ////////////////////////////
 | |
| uint16_t WS2812FX::mode_2Dscrollingtext(void) {
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
| 
 | |
|   const int letterWidth = 6;
 | |
|   const int letterHeight = 8;
 | |
|   const int yoffset = map(SEGMENT.intensity, 0, 255, -rows/2, rows/2) + (rows-letterHeight)/2;
 | |
|   const char *text = PSTR("Use segment name"); // fallback if empty segment name
 | |
|   if (SEGMENT.name && strlen(SEGMENT.name)) text = SEGMENT.name;
 | |
| 
 | |
|   char lineBuffer[17], sec[3];
 | |
|   if (strstr_P(text, PSTR("#DATETIME"))) {
 | |
|     byte AmPmHour = hour(localTime);
 | |
|     boolean isitAM = true;
 | |
|     if (useAMPM) {
 | |
|       if (AmPmHour > 11) { AmPmHour -= 12; isitAM = false; }
 | |
|       if (AmPmHour == 0) { AmPmHour  = 12; }
 | |
|     }
 | |
|     if (useAMPM) sprintf_P(sec, PSTR(" %2s"), (isitAM ? "AM" : "PM"));
 | |
|     else         sprintf_P(sec, PSTR(":%02d"), second(localTime));
 | |
|     sprintf_P(lineBuffer,PSTR("%s %2d %2d:%02d%s"), monthShortStr(month(localTime)), day(localTime), AmPmHour, minute(localTime), sec);
 | |
|     text = lineBuffer;
 | |
|   }
 | |
|   const int numberOfLetters = strlen(text);
 | |
| 
 | |
|   if (SEGENV.step < millis()) {
 | |
|     if ((numberOfLetters * letterWidth) > cols) ++SEGENV.aux0 %= (numberOfLetters * letterWidth) + cols;      // offset
 | |
|     else                                          SEGENV.aux0  = (cols - (numberOfLetters * letterWidth))/2;
 | |
|     ++SEGENV.aux1 &= 0xFF; // color shift
 | |
|     SEGENV.step = millis() + map(SEGMENT.speed, 0, 255, 10*FRAMETIME_FIXED, 2*FRAMETIME_FIXED);
 | |
|   }
 | |
| 
 | |
|   fade_out(255 - (SEGMENT.custom1>>5)); // fade to background color
 | |
| 
 | |
|   for (uint16_t i = 0; i < numberOfLetters; i++) {
 | |
|     if (int(cols) - int(SEGENV.aux0) + letterWidth*(i+1) < 0) continue; // don't draw characters off-screen
 | |
|     if (text[i]<32 || text[i]>126) continue; // skip non-ANSII characters (may add UTF translation at some point)
 | |
|     drawCharacter(text[i], int(cols) - int(SEGENV.aux0) + letterWidth*i, yoffset, color_from_palette(SEGENV.aux1, false, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_SCROLL_TEXT PROGMEM = "2D Scrolling Text@!,Y Offset,Trail;!,!;!";
 | |
| 
 | |
| 
 | |
| ////////////////////////////
 | |
| //     2D Drift Rose      //
 | |
| ////////////////////////////
 | |
| //// Drift Rose by stepko (c)2021 [https://editor.soulmatelights.com/gallery/1369-drift-rose-pattern], adapted by Blaz Kristan
 | |
| uint16_t WS2812FX::mode_2Ddriftrose(void) {
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   const float CX = cols/2.f - .5f;
 | |
|   const float CY = rows/2.f - .5f;
 | |
|   const float L = min(cols, rows) / 2.f;
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) {
 | |
|     fill_solid(leds, CRGB::Black);
 | |
|   }
 | |
| 
 | |
|   fadeToBlackBy(leds, 32+(SEGMENT.speed>>3));
 | |
|   for (byte i = 1; i < 37; i++) {
 | |
|     uint32_t x = (CX + (sin_t(radians(i * 10)) * (beatsin8(i, 0, L*2)-L))) * 255.f;
 | |
|     uint32_t y = (CY + (cos_t(radians(i * 10)) * (beatsin8(i, 0, L*2)-L))) * 255.f;
 | |
|     wu_pixel(leds, x, y, CHSV(i * 10, 255, 255));
 | |
|   }
 | |
|   blur2d(leds, (SEGMENT.intensity>>4)+1);
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| }
 | |
| static const char *_data_FX_MODE_DRIFT_ROSE PROGMEM = "2D Drift Rose@Fade,Blur;;";
 | |
| 
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| /********************     audio enhanced routines     ************************/
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| 
 | |
| /////////////////////////////////
 | |
| //     * Ripple Peak           //
 | |
| /////////////////////////////////
 | |
| uint16_t WS2812FX::mode_ripplepeak(void) {                // * Ripple peak. By Andrew Tuline.
 | |
|                                                           // This currently has no controls.
 | |
|   #define maxsteps 16                                     // Case statement wouldn't allow a variable.
 | |
| 
 | |
|   uint16_t maxRipples = 16;
 | |
|   uint16_t dataSize = sizeof(Ripple) * maxRipples;
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   Ripple* ripples = reinterpret_cast<Ripple*>(SEGENV.data);
 | |
| 
 | |
|   uint8_t *binNum = (uint8_t*)&SEGENV.aux1, *maxVol = (uint8_t*)(&SEGENV.aux1+1); // just in case assignment
 | |
|   uint8_t samplePeak = 0; // actually a bool
 | |
|   double FFT_MajorPeak = 0.0;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     FFT_MajorPeak = *(double*)um_data->u_data[8];
 | |
|     binNum        =  (uint8_t*)um_data->u_data[5];
 | |
|     maxVol        =  (uint8_t*)um_data->u_data[0];
 | |
|     samplePeak    = *(uint8_t*)um_data->u_data[4];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     samplePeak = random8() > 250;
 | |
|     FFT_MajorPeak = inoise8(beatsin8(90, 0, 200)*15 + (ms>>10), ms>>3);
 | |
|   }
 | |
| 
 | |
|   if (SEGENV.call == 0) SEGENV.aux0 = 255;
 | |
| 
 | |
|   *binNum = SEGMENT.custom2;                               // Select a bin.
 | |
|   *maxVol = SEGMENT.custom3/2;                             // Our volume comparator.
 | |
| 
 | |
|   fade_out(240);                                          // Lower frame rate means less effective fading than FastLED
 | |
|   fade_out(240);
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGMENT.intensity/16; i++) {   // Limit the number of ripples.
 | |
|     if (samplePeak) ripples[i].state = 255;
 | |
| 
 | |
|     switch (ripples[i].state) {
 | |
|       case 254:     // Inactive mode
 | |
|         break;
 | |
| 
 | |
|       case 255:                                           // Initialize ripple variables.
 | |
|         ripples[i].pos = random16(SEGLEN);
 | |
|         #ifdef ESP32
 | |
|           ripples[i].color = (int)(log10f(FFT_MajorPeak)*128);
 | |
|         #else
 | |
|           ripples[i].color = random8();
 | |
|         #endif
 | |
|         ripples[i].state = 0;
 | |
|         break;
 | |
| 
 | |
|       case 0:
 | |
|         setPixelColor(ripples[i].pos, color_blend(SEGCOLOR(1), color_from_palette(ripples[i].color, false, PALETTE_SOLID_WRAP, 0), SEGENV.aux0));
 | |
|         ripples[i].state++;
 | |
|         break;
 | |
| 
 | |
|       case maxsteps:                                      // At the end of the ripples. 254 is an inactive mode.
 | |
|         ripples[i].state = 254;
 | |
|         break;
 | |
| 
 | |
|       default:                                            // Middle of the ripples.
 | |
|         setPixelColor((ripples[i].pos + ripples[i].state + SEGLEN) % SEGLEN, color_blend(SEGCOLOR(1), color_from_palette(ripples[i].color, false, PALETTE_SOLID_WRAP, 0), SEGENV.aux0/ripples[i].state*2));
 | |
|         setPixelColor((ripples[i].pos - ripples[i].state + SEGLEN) % SEGLEN, color_blend(SEGCOLOR(1), color_from_palette(ripples[i].color, false, PALETTE_SOLID_WRAP, 0), SEGENV.aux0/ripples[i].state*2));
 | |
|         ripples[i].state++;                               // Next step.
 | |
|         break;
 | |
|     } // switch step
 | |
|   } // for i
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_ripplepeak()
 | |
| static const char *_data_FX_MODE_RIPPLEPEAK PROGMEM = " ♪ Ripple Peak@Fade rate,Max # of ripples,,Select bin,Volume (minimum);!,!;!";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //    * 2D Swirl       //
 | |
| /////////////////////////
 | |
| // By: Mark Kriegsman https://gist.github.com/kriegsman/5adca44e14ad025e6d3b , modified by Andrew Tuline
 | |
| uint16_t WS2812FX::mode_2DSwirl(void) {
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) fill_solid(leds, CRGB::Black);
 | |
| 
 | |
|   const uint8_t borderWidth = 2;
 | |
| 
 | |
|   blur2d(leds, SEGMENT.custom1);
 | |
| 
 | |
|   uint8_t  i = beatsin8( 27*SEGMENT.speed/255, borderWidth, cols - borderWidth);
 | |
|   uint8_t  j = beatsin8( 41*SEGMENT.speed/255, borderWidth, rows - borderWidth);
 | |
|   uint8_t ni = (cols - 1) - i;
 | |
|   uint8_t nj = (cols - 1) - j;
 | |
|   uint16_t ms = millis();
 | |
| 
 | |
|   uint8_t soundAgc = 0;
 | |
|   int16_t rawSampleAgc = 0, sample;
 | |
|   float sampleAvg = 0.0f;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     soundAgc  = *(uint8_t*)um_data->u_data[9];
 | |
|     rawSampleAgc = *(int16_t*)um_data->u_data[3];
 | |
|     sample       = *(int16_t*)um_data->u_data[2];
 | |
|     sampleAvg    = *(float*)um_data->u_data[8];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     sample = inoise8(beatsin8(120, 10, 30)*10 + (ms>>14), ms>>3);
 | |
|     sample = map(sample, 50, 190, 0, 224);
 | |
|     sampleAvg = inoise8(beatsin8(90, 0, 200)*15 + (ms>>10), ms>>3);
 | |
|     //sampleAvg = mapf(sampleAvg, 0, 255, 0, 255); // help me out here
 | |
|   }
 | |
| 
 | |
|   int tmpSound = (soundAgc) ? rawSampleAgc : sample;
 | |
| 
 | |
|   leds[XY( i, j)]  += ColorFromPalette(currentPalette, (ms / 11 + sampleAvg*4), tmpSound * SEGMENT.intensity / 64, LINEARBLEND); //CHSV( ms / 11, 200, 255);
 | |
|   leds[XY( j, i)]  += ColorFromPalette(currentPalette, (ms / 13 + sampleAvg*4), tmpSound * SEGMENT.intensity / 64, LINEARBLEND); //CHSV( ms / 13, 200, 255);
 | |
|   leds[XY(ni, nj)] += ColorFromPalette(currentPalette, (ms / 17 + sampleAvg*4), tmpSound * SEGMENT.intensity / 64, LINEARBLEND); //CHSV( ms / 17, 200, 255);
 | |
|   leds[XY(nj, ni)] += ColorFromPalette(currentPalette, (ms / 29 + sampleAvg*4), tmpSound * SEGMENT.intensity / 64, LINEARBLEND); //CHSV( ms / 29, 200, 255);
 | |
|   leds[XY( i, nj)] += ColorFromPalette(currentPalette, (ms / 37 + sampleAvg*4), tmpSound * SEGMENT.intensity / 64, LINEARBLEND); //CHSV( ms / 37, 200, 255);
 | |
|   leds[XY(ni, j)]  += ColorFromPalette(currentPalette, (ms / 41 + sampleAvg*4), tmpSound * SEGMENT.intensity / 64, LINEARBLEND); //CHSV( ms / 41, 200, 255);
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2DSwirl()
 | |
| static const char *_data_FX_MODE_2DSWIRL PROGMEM = " ♪ 2D Swirl@!,Sensitivity=64,Blur;,Bg Swirl;!";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //    * 2D Waverly     //
 | |
| /////////////////////////
 | |
| // By: Stepko, https://editor.soulmatelights.com/gallery/652-wave , modified by Andrew Tuline
 | |
| uint16_t WS2812FX::mode_2DWaverly(void) {
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   if (SEGENV.call == 0) {
 | |
|     fill_solid(leds, CRGB::Black);
 | |
|   }
 | |
| 
 | |
|   um_data_t *um_data;
 | |
|   uint8_t soundAgc = 0;
 | |
|   float sampleAgc = 0.0f, sampleAvg = 0.0f;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     soundAgc  = *(uint8_t*)um_data->u_data[9];
 | |
|     sampleAgc = *(float*)um_data->u_data[10];
 | |
|     sampleAvg = *(float*)um_data->u_data[8];
 | |
|   }
 | |
| 
 | |
|   fadeToBlackBy(leds, SEGMENT.speed);
 | |
| 
 | |
|   long t = millis() / 2;
 | |
|   for (uint16_t i = 0; i < cols; i++) {
 | |
|     uint16_t thisVal = (1 + SEGMENT.intensity/64) * inoise8(i * 45 , t , t)/2;
 | |
|     // use audio if available
 | |
|     if (um_data) {
 | |
|       thisVal /= 32; // reduce intensity of inoise8()
 | |
|       thisVal *= (soundAgc) ? sampleAgc : sampleAvg;
 | |
|     }
 | |
|     uint16_t thisMax = map(thisVal, 0, 512, 0, rows);
 | |
| 
 | |
|     for (uint16_t j = 0; j < thisMax; j++) {
 | |
|       leds[XY(i, j)] += ColorFromPalette(currentPalette, map(j, 0, thisMax, 250, 0), 255, LINEARBLEND);
 | |
|       leds[XY((cols - 1) - i, (rows - 1) - j)] += ColorFromPalette(currentPalette, map(j, 0, thisMax, 250, 0), 255, LINEARBLEND);
 | |
|     }
 | |
|   }
 | |
|   blur2d(leds, 16);
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2DWaverly()
 | |
| static const char *_data_FX_MODE_2DWAVERLY PROGMEM = " ♪ 2D Waverly@Amplification,Sensitivity=64;;!";
 | |
| 
 | |
| 
 | |
| // float version of map()
 | |
| static float mapf(float x, float in_min, float in_max, float out_min, float out_max){
 | |
|   return (x - in_min) * (out_max - out_min) / (in_max - in_min) + out_min;
 | |
| }
 | |
| 
 | |
| // Gravity struct requited for GRAV* effects
 | |
| typedef struct Gravity {
 | |
|   int    topLED;
 | |
|   int    gravityCounter;
 | |
| } gravity;
 | |
| 
 | |
| ///////////////////////
 | |
| //   * GRAVCENTER    //
 | |
| ///////////////////////
 | |
| uint16_t WS2812FX::mode_gravcenter(void) {                // Gravcenter. By Andrew Tuline.
 | |
| 
 | |
|   const uint16_t dataSize = sizeof(gravity);
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   Gravity* gravcen = reinterpret_cast<Gravity*>(SEGENV.data);
 | |
| 
 | |
|   um_data_t *um_data;
 | |
|   float tmpSound = (float)inoise8(23333); // I have no idea what that does
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     //soundAgc  = *(uint8_t*)um_data->u_data[9];
 | |
|     //sampleAgc = *(float*)um_data->u_data[10];
 | |
|     //sampleAvg = *(float*)um_data->u_data[8];
 | |
|     tmpSound = *(uint8_t*)um_data->u_data[9] ? *(float*)um_data->u_data[10] : *(float*)um_data->u_data[8];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     tmpSound = inoise8(beatsin8(120, 10, 30)*10 + (ms>>14), ms>>3);
 | |
|     //tmpSound = map(sample, 50, 190, 0, 255);
 | |
|   }
 | |
| 
 | |
|   fade_out(240);
 | |
| 
 | |
|   float segmentSampleAvg = tmpSound * (float)SEGMENT.intensity / 255.0f;
 | |
|   segmentSampleAvg *= 0.125; // divide by 8, to compensate for later "sensitivty" upscaling
 | |
| 
 | |
|   float mySampleAvg = mapf(segmentSampleAvg*2.0, 0, 32, 0, (float)SEGLEN/2.0); // map to pixels available in current segment 
 | |
|   uint16_t tempsamp = constrain(mySampleAvg, 0, SEGLEN/2);     // Keep the sample from overflowing.
 | |
|   uint8_t gravity = 8 - SEGMENT.speed/32;
 | |
| 
 | |
|   for (int i=0; i<tempsamp; i++) {
 | |
|     uint8_t index = inoise8(i*segmentSampleAvg+millis(), 5000+i*segmentSampleAvg);
 | |
|     setPixelColor(i+SEGLEN/2, color_blend(SEGCOLOR(1), color_from_palette(index, false, PALETTE_SOLID_WRAP, 0), segmentSampleAvg*8));
 | |
|     setPixelColor(SEGLEN/2-i-1, color_blend(SEGCOLOR(1), color_from_palette(index, false, PALETTE_SOLID_WRAP, 0), segmentSampleAvg*8));
 | |
|   }
 | |
| 
 | |
|   if (tempsamp >= gravcen->topLED)
 | |
|     gravcen->topLED = tempsamp-1;
 | |
|   else if (gravcen->gravityCounter % gravity == 0)
 | |
|     gravcen->topLED--;
 | |
| 
 | |
|   if (gravcen->topLED >= 0) {
 | |
|     setPixelColor(gravcen->topLED+SEGLEN/2, color_from_palette(millis(), false, PALETTE_SOLID_WRAP, 0));
 | |
|     setPixelColor(SEGLEN/2-1-gravcen->topLED, color_from_palette(millis(), false, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
|   gravcen->gravityCounter = (gravcen->gravityCounter + 1) % gravity;
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_gravcenter()
 | |
| static const char *_data_FX_MODE_GRAVCENTER PROGMEM = " ♪ Gravcenter@Rate of fall,Sensitivity=128;,!;!";
 | |
| 
 | |
| 
 | |
| ///////////////////////
 | |
| //   * GRAVCENTRIC   //
 | |
| ///////////////////////
 | |
| uint16_t WS2812FX::mode_gravcentric(void) {                     // Gravcentric. By Andrew Tuline.
 | |
| 
 | |
|   uint16_t dataSize = sizeof(gravity);
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static();     //allocation failed
 | |
|   Gravity* gravcen = reinterpret_cast<Gravity*>(SEGENV.data);
 | |
| 
 | |
|   um_data_t *um_data;
 | |
|   float tmpSound = (float)inoise8(23333); // I have no idea what that does
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     //soundAgc  = *(uint8_t*)um_data->u_data[9];
 | |
|     //sampleAgc = *(float*)um_data->u_data[10];
 | |
|     //sampleAvg = *(float*)um_data->u_data[8];
 | |
|     tmpSound = *(uint8_t*)um_data->u_data[9] ? *(float*)um_data->u_data[10] : *(float*)um_data->u_data[8];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     tmpSound = inoise8(beatsin8(120, 10, 30)*10 + (ms>>14), ms>>3);
 | |
|     //tmpSound = map(sample, 50, 190, 0, 255);
 | |
|   }
 | |
| 
 | |
|   fade_out(240);
 | |
|   fade_out(240); // twice? really?
 | |
| 
 | |
|   float segmentSampleAvg = tmpSound * (float)SEGMENT.intensity / 255.0;
 | |
|   segmentSampleAvg *= 0.125f; // divide by 8, to compensate for later "sensitivty" upscaling
 | |
| 
 | |
|   float mySampleAvg = mapf(segmentSampleAvg*2.0, 0.0f, 32.0f, 0.0f, (float)SEGLEN/2.0); // map to pixels availeable in current segment 
 | |
|   int tempsamp = constrain(mySampleAvg, 0, SEGLEN/2);     // Keep the sample from overflowing.
 | |
|   uint8_t gravity = 8 - SEGMENT.speed/32;
 | |
| 
 | |
|   for (int i=0; i<tempsamp; i++) {
 | |
|     uint8_t index = segmentSampleAvg*24+millis()/200;
 | |
|     setPixelColor(i+SEGLEN/2, color_from_palette(index, false, PALETTE_SOLID_WRAP, 0));
 | |
|     setPixelColor(SEGLEN/2-1-i, color_from_palette(index, false, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   if (tempsamp >= gravcen->topLED)
 | |
|     gravcen->topLED = tempsamp-1;
 | |
|   else if (gravcen->gravityCounter % gravity == 0)
 | |
|     gravcen->topLED--;
 | |
| 
 | |
|   if (gravcen->topLED >= 0) {
 | |
|     setPixelColor(gravcen->topLED+SEGLEN/2, CRGB::Gray);
 | |
|     setPixelColor(SEGLEN/2-1-gravcen->topLED, CRGB::Gray);
 | |
|   }
 | |
|   gravcen->gravityCounter = (gravcen->gravityCounter + 1) % gravity;
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_gravcentric()
 | |
| static const char *_data_FX_MODE_GRAVCENTRIC PROGMEM = " ♪ Gravcentric@Rate of fall,Sensitivity=128;!;!";
 | |
| 
 | |
| 
 | |
| ///////////////////////
 | |
| //   * GRAVIMETER    //
 | |
| ///////////////////////
 | |
| #ifndef SR_DEBUG_AGC
 | |
| uint16_t WS2812FX::mode_gravimeter(void) {                // Gravmeter. By Andrew Tuline.
 | |
| 
 | |
|   uint16_t dataSize = sizeof(gravity);
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   Gravity* gravcen = reinterpret_cast<Gravity*>(SEGENV.data);
 | |
| 
 | |
|   um_data_t *um_data;
 | |
|   float tmpSound = (float)inoise8(23333); // I have no idea what that does
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     //soundAgc  = *(uint8_t*)um_data->u_data[9];
 | |
|     //sampleAgc = *(float*)um_data->u_data[10];
 | |
|     //sampleAvg = *(float*)um_data->u_data[8];
 | |
|     tmpSound = *(uint8_t*)um_data->u_data[9] ? *(float*)um_data->u_data[10] : *(float*)um_data->u_data[8];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     tmpSound = inoise8(beatsin8(120, 10, 30)*10 + (ms>>14), ms>>3);
 | |
|     //tmpSound = map(sample, 50, 190, 0, 255);
 | |
|   }
 | |
| 
 | |
|   fade_out(240);
 | |
| 
 | |
|   float segmentSampleAvg = tmpSound * (float)SEGMENT.intensity / 255.0;
 | |
|   segmentSampleAvg *= 0.25; // divide by 4, to compensate for later "sensitivty" upscaling
 | |
| 
 | |
|   float mySampleAvg = mapf(segmentSampleAvg*2.0, 0, 64, 0, (SEGLEN-1)); // map to pixels availeable in current segment 
 | |
|   int tempsamp = constrain(mySampleAvg,0,SEGLEN-1);       // Keep the sample from overflowing.
 | |
|   uint8_t gravity = 8 - SEGMENT.speed/32;
 | |
| 
 | |
|   for (int i=0; i<tempsamp; i++) {
 | |
|     uint8_t index = inoise8(i*segmentSampleAvg+millis(), 5000+i*segmentSampleAvg);
 | |
|     setPixelColor(i, color_blend(SEGCOLOR(1), color_from_palette(index, false, PALETTE_SOLID_WRAP, 0), segmentSampleAvg*8));
 | |
|   }
 | |
| 
 | |
|   if (tempsamp >= gravcen->topLED)
 | |
|     gravcen->topLED = tempsamp;
 | |
|   else if (gravcen->gravityCounter % gravity == 0)
 | |
|     gravcen->topLED--;
 | |
| 
 | |
|   if (gravcen->topLED > 0) {
 | |
|     setPixelColor(gravcen->topLED, color_from_palette(millis(), false, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
|   gravcen->gravityCounter = (gravcen->gravityCounter + 1) % gravity;
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_gravimeter()
 | |
| static const char *_data_FX_MODE_GRAVIMETER PROGMEM = " ♪ Gravimeter@Rate of fall,Sensitivity=128;,!;!";
 | |
| #else
 | |
| // This an abuse of the gravimeter effect for AGC debugging
 | |
| // instead of sound volume, it uses the AGC gain multiplier as input
 | |
| uint16_t WS2812FX::mode_gravimeter(void) {                // Gravmeter. By Andrew Tuline.
 | |
| 
 | |
|   uint16_t dataSize = sizeof(gravity);
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   Gravity* gravcen = reinterpret_cast<Gravity*>(SEGENV.data);
 | |
| 
 | |
|   uint8_t *inputLevel = (uint8_t*)(&SEGENV.aux1+1);
 | |
|   um_data_t *um_data;
 | |
|   uint16_t sample = 0;
 | |
|   uint8_t soundAgc = 0;
 | |
|   float sampleAgc = 0.0f, sampleAgv = 0.0f, multAgc = 0.0f, sampleReal = 0.0f;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     sample     = *(uint16_t*)um_data->u_data[2];
 | |
|     soundAgc   = *(uint8_t*)um_data->u_data[9];
 | |
|     sampleAgc  = *(float*)um_data->u_data[10];
 | |
|     sampleAvg  = *(float*)um_data->u_data[8];
 | |
|     multAgc    = *(float*)um_data->u_data[11];
 | |
|     sampleReal = *(float*)um_data->u_data[12];
 | |
|     sampleGain = *(float*)um_data->u_data[13];
 | |
|     inputLevel =  (uint8_t*)um_data->u_data[17];
 | |
|   }
 | |
| 
 | |
|   fade_out(240);
 | |
| 
 | |
|   //TODO: implement inputLevel as a global or slider
 | |
|   *inputLevel = SEGMENT.custom1;
 | |
| 
 | |
|   float tmpSound = multAgc;                                                         // AGC gain
 | |
|   if (soundAgc == 0) {
 | |
|     if ((sampleAvg> 1.0f) && (sampleReal > 0.05f))
 | |
|       tmpSound = (float)sample / sampleReal;                                        // current non-AGC gain
 | |
|     else 
 | |
|       tmpSound = ((float)sampleGain/40.0f * (float)*inputLevel/128.0f) + 1.0f/16.0f;     // non-AGC gain from presets
 | |
|   }
 | |
| 
 | |
|   if (tmpSound > 2) tmpSound = ((tmpSound -2.0f) / 2) +2;  //compress ranges > 2
 | |
|   if (tmpSound > 1) tmpSound = ((tmpSound -1.0f) / 2) +1;  //compress ranges > 1
 | |
| 
 | |
|   float segmentSampleAvg = 64.0f * tmpSound * (float)SEGMENT.intensity / 128.0f;
 | |
|   float mySampleAvg = mapf(segmentSampleAvg, 0.0f, 128.0f, 0, (SEGLEN-1)); // map to pixels availeable in current segment 
 | |
|   int tempsamp = constrain(mySampleAvg, 0, SEGLEN-1);                  // Keep the sample from overflowing.
 | |
| 
 | |
|   //tempsamp = SEGLEN - tempsamp;                                      // uncomment to invert direction
 | |
|   segmentSampleAvg = fmax(64.0f - fmin(segmentSampleAvg, 63),8);         // inverted brightness
 | |
| 
 | |
|   uint8_t gravity = 8 - SEGMENT.speed/32;
 | |
| 
 | |
|   if (sampleAvg > 1)                                                 // disable bar "body" if below squelch
 | |
|   {
 | |
|     for (int i=0; i<tempsamp; i++) {
 | |
|       uint8_t index = inoise8(i*segmentSampleAvg+millis(), 5000+i*segmentSampleAvg);
 | |
|       setPixelColor(i, color_blend(SEGCOLOR(1), color_from_palette(index, false, PALETTE_SOLID_WRAP, 0), segmentSampleAvg*4.0));
 | |
|     }
 | |
|   }
 | |
|   if (tempsamp >= gravcen->topLED)
 | |
|     gravcen->topLED = tempsamp;
 | |
|   else if (gravcen->gravityCounter % gravity == 0)
 | |
|     gravcen->topLED--;
 | |
| 
 | |
|   if (gravcen->topLED > 0) {
 | |
|     setPixelColor(gravcen->topLED, color_from_palette(millis(), false, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
|   gravcen->gravityCounter = (gravcen->gravityCounter + 1) % gravity;
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_gravimeter()
 | |
| static const char *_data_FX_MODE_GRAVIMETER PROGMEM = " ♪ Gravimeter@Rate of fall,Sensitivity=128,Input level=128;,!;!";
 | |
| #endif
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //   * JUGGLES      //
 | |
| //////////////////////
 | |
| uint16_t WS2812FX::mode_juggles(void) {                   // Juggles. By Andrew Tuline.
 | |
|   um_data_t *um_data;
 | |
|   float sampleAgc = 0.0f;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     sampleAgc = *(float*)um_data->u_data[10];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     sampleAgc = inoise8(beatsin8(120, 10, 30)*10 + (ms>>14), ms>>3);
 | |
|     //sampleAgc = map(sample, 50, 190, 0, 255);
 | |
|   }
 | |
| 
 | |
|   fade_out(224);
 | |
|   uint16_t my_sampleAgc = fmax(fmin(sampleAgc, 255.0), 0);
 | |
| 
 | |
|   for (uint8_t i=0; i<SEGMENT.intensity/32+1; i++) {
 | |
|     setPixelColor(beatsin16(SEGMENT.speed/4+i*2,0,SEGLEN-1), color_blend(SEGCOLOR(1), color_from_palette(millis()/4+i*2, false, PALETTE_SOLID_WRAP, 0), my_sampleAgc));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_juggles()
 | |
| static const char *_data_FX_MODE_JUGGLES PROGMEM = " ♪ Juggles@!,# of balls;,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //   * MATRIPIX     //
 | |
| //////////////////////
 | |
| uint16_t WS2812FX::mode_matripix(void) {                  // Matripix. By Andrew Tuline.
 | |
|   um_data_t *um_data;
 | |
|   uint8_t soundAgc = 0;
 | |
|   int16_t rawSampleAgc = 0, sample;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     soundAgc     = *(uint8_t*)um_data->u_data[9];
 | |
|     rawSampleAgc = *(int16_t*)um_data->u_data[3];
 | |
|     sample       = *(int16_t*)um_data->u_data[2];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     sample = inoise8(beatsin8(120, 10, 30)*10 + (ms>>14), ms>>3);
 | |
|     sample = map(sample, 50, 190, 0, 255);
 | |
|   }
 | |
| 
 | |
|   if (SEGENV.call == 0) fill(BLACK);
 | |
| 
 | |
|   uint8_t secondHand = micros()/(256-SEGMENT.speed)/500 % 16;
 | |
|   if(SEGENV.aux0 != secondHand) {
 | |
|     SEGENV.aux0 = secondHand;
 | |
| 
 | |
|     uint8_t tmpSound = (soundAgc) ? rawSampleAgc : sample;
 | |
|     int pixBri = tmpSound * SEGMENT.intensity / 64;
 | |
|     for (uint16_t i=0; i<SEGLEN-1; i++) setPixelColor(i, getPixelColor(i+1)); // shift left
 | |
|     setPixelColor(SEGLEN-1, color_blend(SEGCOLOR(1), color_from_palette(millis(), false, PALETTE_SOLID_WRAP, 0), pixBri));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_matripix()
 | |
| static const char *_data_FX_MODE_MATRIPIX PROGMEM = " ♪ Matripix@!,Brightness=64;,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //   * MIDNOISE     //
 | |
| //////////////////////
 | |
| uint16_t WS2812FX::mode_midnoise(void) {                  // Midnoise. By Andrew Tuline.
 | |
| // Changing xdist to SEGENV.aux0 and ydist to SEGENV.aux1.
 | |
| 
 | |
|   um_data_t *um_data;
 | |
|   uint8_t soundAgc = 0;
 | |
|   float sampleAgc = 0.0f, sampleAvg = 0.0f;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     soundAgc  = *(uint8_t*)um_data->u_data[9];
 | |
|     sampleAgc = *(float*)um_data->u_data[10];
 | |
|     sampleAvg = *(float*)um_data->u_data[8];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     sampleAvg = inoise8(beatsin8(90, 0, 200)*15 + (ms>>10), ms>>3);
 | |
|     //sampleAvg = mapf(sampleAvg, 0, 255, 0, 255); // help me out here
 | |
|   }
 | |
| 
 | |
|   fade_out(SEGMENT.speed);
 | |
|   fade_out(SEGMENT.speed);
 | |
| 
 | |
|   float tmpSound = (soundAgc) ? sampleAgc : sampleAvg;
 | |
|   float tmpSound2 = tmpSound * (float)SEGMENT.intensity / 256.0;  // Too sensitive.
 | |
|   tmpSound2 *= (float)SEGMENT.intensity / 128.0;              // Reduce sensitity/length.
 | |
| 
 | |
|   int maxLen = mapf(tmpSound2, 0, 127, 0, SEGLEN/2);
 | |
|   if (maxLen >SEGLEN/2) maxLen = SEGLEN/2;
 | |
| 
 | |
|   for (int i=(SEGLEN/2-maxLen); i<(SEGLEN/2+maxLen); i++) {
 | |
|     uint8_t index = inoise8(i*tmpSound+SEGENV.aux0, SEGENV.aux1+i*tmpSound);  // Get a value from the noise function. I'm using both x and y axis.
 | |
|     setPixelColor(i, color_from_palette(index, false, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   SEGENV.aux0=SEGENV.aux0+beatsin8(5,0,10);
 | |
|   SEGENV.aux1=SEGENV.aux1+beatsin8(4,0,10);
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_midnoise()
 | |
| static const char *_data_FX_MODE_MIDNOISE PROGMEM = " ♪ Midnoise@Fade rate,Maximum length=128;,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //   * NOISEFIRE    //
 | |
| //////////////////////
 | |
| // I am the god of hellfire. . . Volume (only) reactive fire routine. Oh, look how short this is.
 | |
| uint16_t WS2812FX::mode_noisefire(void) {                 // Noisefire. By Andrew Tuline.
 | |
|   currentPalette = CRGBPalette16(CHSV(0,255,2), CHSV(0,255,4), CHSV(0,255,8), CHSV(0, 255, 8),  // Fire palette definition. Lower value = darker.
 | |
|                                  CHSV(0, 255, 16), CRGB::Red, CRGB::Red, CRGB::Red,
 | |
|                                  CRGB::DarkOrange,CRGB::DarkOrange, CRGB::Orange, CRGB::Orange,
 | |
|                                  CRGB::Yellow, CRGB::Orange, CRGB::Yellow, CRGB::Yellow);
 | |
| 
 | |
|   um_data_t *um_data;
 | |
|   uint8_t soundAgc = 0;
 | |
|   float sampleAgc = 0.0f, sampleAvg = 0.0f;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     soundAgc  = *(uint8_t*)um_data->u_data[9];
 | |
|     sampleAgc = *(float*)um_data->u_data[10];
 | |
|     sampleAvg = *(float*)um_data->u_data[8];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     sampleAvg = inoise8(beatsin8(90, 0, 200)*15 + (ms>>10), ms>>3);
 | |
|     //sampleAvg = mapf(sampleAvg, 0, 255, 0, 255); // help me out here
 | |
|   }
 | |
| 
 | |
|   for (uint16_t i = 0; i < SEGLEN; i++) {
 | |
|     uint16_t index = inoise8(i*SEGMENT.speed/64,millis()*SEGMENT.speed/64*SEGLEN/255);  // X location is constant, but we move along the Y at the rate of millis(). By Andrew Tuline.
 | |
|     index = (255 - i*256/SEGLEN) * index/(256-SEGMENT.intensity);                       // Now we need to scale index so that it gets blacker as we get close to one of the ends.
 | |
|                                                                                         // This is a simple y=mx+b equation that's been scaled. index/128 is another scaling.
 | |
|     uint8_t tmpSound = (soundAgc) ? sampleAgc : sampleAvg;
 | |
| 
 | |
|     CRGB color = ColorFromPalette(currentPalette, index, tmpSound*2, LINEARBLEND);     // Use the my own palette.
 | |
|     setPixelColor(i, color);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_noisefire()
 | |
| static const char *_data_FX_MODE_NOISEFIRE PROGMEM = " ♪ Noisefire@!,!;;";
 | |
| 
 | |
| 
 | |
| ///////////////////////
 | |
| //   * Noisemeter    //
 | |
| ///////////////////////
 | |
| uint16_t WS2812FX::mode_noisemeter(void) {                // Noisemeter. By Andrew Tuline.
 | |
| 
 | |
|   um_data_t *um_data;
 | |
|   uint8_t soundAgc = 0;
 | |
|   int16_t rawSampleAgc = 0, sample;
 | |
|   float sampleAgc = 0.0f, sampleAvg;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     soundAgc     = *(uint8_t*)um_data->u_data[9];
 | |
|     sampleAgc    = *(float*)um_data->u_data[10];
 | |
|     sampleAvg    = *(float*)um_data->u_data[8];
 | |
|     rawSampleAgc = *(int16_t*)um_data->u_data[3];
 | |
|     sample       = *(int16_t*)um_data->u_data[2];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     sample = inoise8(beatsin8(120, 10, 30)*10 + (ms>>14), ms>>3);
 | |
|     sample = map(sample, 50, 190, 0, 255);
 | |
|     sampleAvg = inoise8(beatsin8(90, 0, 200)*15 + (ms>>10), ms>>3);
 | |
|     //sampleAvg = mapf(sampleAvg, 0, 255, 0, 255); // help me out here
 | |
|   }
 | |
| 
 | |
|   uint8_t fadeRate = map(SEGMENT.speed,0,255,224,255);
 | |
|   fade_out(fadeRate);
 | |
| 
 | |
|   float tmpSound = (soundAgc) ? rawSampleAgc : sample;
 | |
|   float tmpSound2 = tmpSound * 2.0 * (float)SEGMENT.intensity / 255.0;
 | |
|   int maxLen = mapf(tmpSound2, 0, 255, 0, SEGLEN); // map to pixels availeable in current segment              // Still a bit too sensitive.
 | |
|   if (maxLen >SEGLEN) maxLen = SEGLEN;
 | |
| 
 | |
|   tmpSound = soundAgc ? sampleAgc : sampleAvg;                      // now use smoothed value (sampleAvg or sampleAgc)
 | |
|   for (int i=0; i<maxLen; i++) {                                    // The louder the sound, the wider the soundbar. By Andrew Tuline.
 | |
|     uint8_t index = inoise8(i*tmpSound+SEGENV.aux0, SEGENV.aux1+i*tmpSound);  // Get a value from the noise function. I'm using both x and y axis.
 | |
|     setPixelColor(i, color_from_palette(index, false, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   SEGENV.aux0+=beatsin8(5,0,10);
 | |
|   SEGENV.aux1+=beatsin8(4,0,10);
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_noisemeter()
 | |
| static const char *_data_FX_MODE_NOISEMETER PROGMEM = " ♪ Noisemeter@Fade rate,Width=128;!,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //     * PIXELS     //
 | |
| //////////////////////
 | |
| uint16_t WS2812FX::mode_pixels(void) {                    // Pixels. By Andrew Tuline.
 | |
| 
 | |
|   um_data_t *um_data;
 | |
|   uint16_t *myVals = nullptr;
 | |
|   float sampleAgc = 0.0f;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     sampleAgc = *(float*)um_data->u_data[10];
 | |
|     myVals    = (uint16_t*)um_data->u_data[14];
 | |
|   }
 | |
|   if (!myVals) return mode_static();
 | |
| 
 | |
|   fade_out(SEGMENT.speed);
 | |
| 
 | |
|   for (uint16_t i=0; i <SEGMENT.intensity/16; i++) {
 | |
|     uint16_t segLoc = random16(SEGLEN);                    // 16 bit for larger strands of LED's.
 | |
|     setPixelColor(segLoc, color_blend(SEGCOLOR(1), color_from_palette(myVals[i%32]+i*4, false, PALETTE_SOLID_WRAP, 0), sampleAgc));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_pixels()
 | |
| static const char *_data_FX_MODE_PIXELS PROGMEM = " ♪ Pixels@Fade rate,# of pixels;,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //   * PIXELWAVE    //
 | |
| //////////////////////
 | |
| uint16_t WS2812FX::mode_pixelwave(void) {                 // Pixelwave. By Andrew Tuline.
 | |
|   if (SEGENV.call == 0) fill(BLACK);
 | |
| 
 | |
|   um_data_t *um_data;
 | |
|   uint8_t soundAgc = 0;
 | |
|   int16_t rawSampleAgc = 0, sample;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     soundAgc     = *(uint8_t*)um_data->u_data[9];
 | |
|     rawSampleAgc = *(int16_t*)um_data->u_data[3];
 | |
|     sample       = *(int16_t*)um_data->u_data[2];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     sample = inoise8(beatsin8(120, 10, 30)*10 + (ms>>14), ms>>3);
 | |
|     sample = map(sample, 50, 190, 0, 255);
 | |
|   }
 | |
| 
 | |
|   uint8_t secondHand = micros()/(256-SEGMENT.speed)/500+1 % 16;
 | |
|   if(SEGENV.aux0 != secondHand) {
 | |
|     SEGENV.aux0 = secondHand;
 | |
| 
 | |
|     uint8_t tmpSound = (soundAgc) ? rawSampleAgc : sample;
 | |
|     int pixBri = tmpSound * SEGMENT.intensity / 64;
 | |
| 
 | |
|     setPixelColor(SEGLEN/2, color_blend(SEGCOLOR(1), color_from_palette(millis(), false, PALETTE_SOLID_WRAP, 0), pixBri));
 | |
| 
 | |
|     for (uint16_t i=SEGLEN-1; i>SEGLEN/2; i--) setPixelColor(i, getPixelColor(i-1)); // Move to the right.
 | |
|     for (uint16_t i=0; i<SEGLEN/2; i++)        setPixelColor(i, getPixelColor(i+1)); // Move to the left.
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_pixelwave()
 | |
| static const char *_data_FX_MODE_PIXELWAVE PROGMEM = " ♪ Pixelwave@!,Sensitivity=64;!,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //   * PLASMOID     //
 | |
| //////////////////////
 | |
| typedef struct Plasphase {
 | |
|   int16_t    thisphase;
 | |
|   int16_t    thatphase;
 | |
| } plasphase;
 | |
| 
 | |
| uint16_t WS2812FX::mode_plasmoid(void) {                  // Plasmoid. By Andrew Tuline.
 | |
|   uint16_t dataSize = sizeof(plasphase);
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   Plasphase* plasmoip = reinterpret_cast<Plasphase*>(SEGENV.data);
 | |
| 
 | |
|   um_data_t *um_data;
 | |
|   uint8_t soundAgc = 0;
 | |
|   float sampleAgc = 0.0f, sampleAvg;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     soundAgc  = *(uint8_t*)um_data->u_data[9];
 | |
|     sampleAgc = *(float*)um_data->u_data[10];
 | |
|     sampleAvg = *(float*)um_data->u_data[8];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     sampleAvg = inoise8(beatsin8(90, 0, 200)*15 + (ms>>10), ms>>3);
 | |
|     //sampleAvg = mapf(sampleAvg, 0, 255, 0, 255); // help me out here
 | |
|   }
 | |
| 
 | |
|   fade_out(64);
 | |
| 
 | |
|   plasmoip->thisphase += beatsin8(6,-4,4);                          // You can change direction and speed individually.
 | |
|   plasmoip->thatphase += beatsin8(7,-4,4);                          // Two phase values to make a complex pattern. By Andrew Tuline.
 | |
| 
 | |
|   for (uint16_t i=0; i<SEGLEN; i++) {                          // For each of the LED's in the strand, set a brightness based on a wave as follows.
 | |
|     // updated, similar to "plasma" effect - softhack007
 | |
|     uint8_t thisbright = cubicwave8(((i*(1 + (3*SEGMENT.speed/32)))+plasmoip->thisphase) & 0xFF)/2;
 | |
|     thisbright += cos8(((i*(97 +(5*SEGMENT.speed/32)))+plasmoip->thatphase) & 0xFF)/2; // Let's munge the brightness a bit and animate it all with the phases.
 | |
|     
 | |
|     uint8_t colorIndex=thisbright;
 | |
|     int tmpSound = (soundAgc) ? sampleAgc : sampleAvg;
 | |
|     if (tmpSound * SEGMENT.intensity / 64 < thisbright) {thisbright = 0;}
 | |
| 
 | |
|     setPixelColor(i, color_add(getPixelColor(i), color_blend(SEGCOLOR(1), color_from_palette(colorIndex, false, PALETTE_SOLID_WRAP, 0), thisbright)));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_plasmoid()
 | |
| static const char *_data_FX_MODE_PLASMOID PROGMEM = " ♪ Plasmoid@Phase=128,# of pixels=128;,!;!";
 | |
| 
 | |
| 
 | |
| ///////////////////////
 | |
| //   * PUDDLEPEAK    //
 | |
| ///////////////////////
 | |
| // Andrew's crappy peak detector. If I were 40+ years younger, I'd learn signal processing.
 | |
| uint16_t WS2812FX::mode_puddlepeak(void) {                // Puddlepeak. By Andrew Tuline.
 | |
| 
 | |
|   uint16_t size = 0;
 | |
|   uint8_t fadeVal = map(SEGMENT.speed,0,255, 224, 255);
 | |
|   uint16_t pos = random(SEGLEN);                          // Set a random starting position.
 | |
| 
 | |
|   uint8_t *binNum = (uint8_t*)&SEGENV.aux1, *maxVol = (uint8_t*)(&SEGENV.aux0); // just in case assignment
 | |
|   uint8_t samplePeak = 0;
 | |
|   float sampleAgc = 0.0f;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     sampleAgc  = *(float*)um_data->u_data[10];
 | |
|     binNum     = (uint8_t*)um_data->u_data[5];
 | |
|     maxVol     = (uint8_t*)um_data->u_data[0];
 | |
|     samplePeak = *(uint8_t*)um_data->u_data[4];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     samplePeak = random8() > 250;
 | |
|     sampleAgc = inoise8(beatsin8(90, 0, 200)*15 + (ms>>10), ms>>3);
 | |
|     //sampleAgc = mapf(sampleAvg, 0, 255, 0, 255); // help me out here
 | |
|   }
 | |
| 
 | |
|   *binNum = SEGMENT.custom2;                               // Select a bin.
 | |
|   *maxVol = SEGMENT.custom3/4;                             // Our volume comparator.
 | |
| 
 | |
|   fade_out(fadeVal);
 | |
| 
 | |
|   if (samplePeak == 1) {
 | |
|     size = sampleAgc * SEGMENT.intensity /256 /4 + 1;     // Determine size of the flash based on the volume.
 | |
|     if (pos+size>= SEGLEN) size = SEGLEN - pos;
 | |
|   }
 | |
| 
 | |
|   for (uint16_t i=0; i<size; i++) {                        // Flash the LED's.
 | |
|     setPixelColor(pos+i, color_from_palette(millis(), false, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_puddlepeak()
 | |
| static const char *_data_FX_MODE_PUDDLEPEAK PROGMEM = " ♪ Puddlepeak@Fade rate,Puddle size,,Select bin,Volume (minimum);!,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //   * PUDDLES      //
 | |
| //////////////////////
 | |
| uint16_t WS2812FX::mode_puddles(void) {                   // Puddles. By Andrew Tuline.
 | |
|   uint16_t size = 0;
 | |
|   uint8_t fadeVal = map(SEGMENT.speed, 0, 255, 224, 255);
 | |
|   uint16_t pos = random16(SEGLEN);                        // Set a random starting position.
 | |
| 
 | |
|   fade_out(fadeVal);
 | |
| 
 | |
|   uint8_t soundAgc = 0;
 | |
|   int16_t rawSampleAgc = 0, sample;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     soundAgc     = *(uint8_t*)um_data->u_data[9];
 | |
|     rawSampleAgc = *(int16_t*)um_data->u_data[3];
 | |
|     sample       = *(int16_t*)um_data->u_data[2];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     uint32_t ms = millis();
 | |
|     sample = inoise8(beatsin8(120, 10, 30)*10 + (ms>>14), ms>>3);
 | |
|     sample = map(sample, 50, 190, 0, 255);
 | |
|   }
 | |
| 
 | |
|   uint16_t tmpSound = (soundAgc) ? rawSampleAgc : sample;
 | |
|   
 | |
|   if (tmpSound > 1) {
 | |
|     size = tmpSound * SEGMENT.intensity /256 /8 + 1;        // Determine size of the flash based on the volume.
 | |
|     if (pos+size >= SEGLEN) size = SEGLEN - pos;
 | |
|   }
 | |
| 
 | |
|   for(uint16_t i=0; i<size; i++) {                          // Flash the LED's.
 | |
|     setPixelColor(pos+i, color_from_palette(millis(), false, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_puddles()
 | |
| static const char *_data_FX_MODE_PUDDLES PROGMEM = " ♪ Puddles@Fade rate,Puddle size;!,!;!";
 | |
| 
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| /********************       audio only routines       ************************/
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| 
 | |
| ///////////////////////////////
 | |
| //     BEGIN FFT ROUTINES    //
 | |
| ///////////////////////////////
 | |
| 
 | |
| ////////////////////
 | |
| //    ** Binmap   //
 | |
| ////////////////////
 | |
| // Binmap. Scale raw fftBin[] values to SEGLEN. Shows just how noisy those bins are.
 | |
| uint16_t WS2812FX::mode_binmap(void) {
 | |
|   #define FIRSTBIN 3                            // The first 3 bins are garbage.
 | |
|   #define LASTBIN 255                           // Don't use the highest bins, as they're (almost) a mirror of the first 256.
 | |
| 
 | |
|   float maxVal = 512;                           // Kind of a guess as to the maximum output value per combined logarithmic bins.
 | |
| 
 | |
|   uint8_t *maxVol = (uint8_t*)&SEGENV.aux1; // just in case assignment
 | |
|   uint8_t soundAgc = 0;
 | |
|   float sampleAvg = 0.0f;
 | |
|   float *fftBin = nullptr;
 | |
|   float multAgc, sampleGain;
 | |
|   uint8_t soundSquelch;
 | |
|   uint8_t *inputLevel = (uint8_t*)(&SEGENV.aux1+1);
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     maxVol        =  (uint8_t*)um_data->u_data[0];
 | |
|     sampleAvg     = *(float*)um_data->u_data[8];
 | |
|     soundAgc      = *(uint8_t*)um_data->u_data[9];
 | |
|     multAgc       = *(float*)um_data->u_data[11];
 | |
|     sampleGain    = *(float*)um_data->u_data[13];
 | |
|     soundSquelch  = *(uint8_t*)um_data->u_data[15];
 | |
|     fftBin        =  (float*)um_data->u_data[16];
 | |
|     inputLevel    =  (uint8_t*)um_data->u_data[17];
 | |
|   }
 | |
|   if (!fftBin) return mode_static();
 | |
| 
 | |
|   //TODO: implement inputLevel as a global or slider
 | |
|   *inputLevel = SEGMENT.custom1;
 | |
|   float binScale = (((float)sampleGain / 40.0f) + 1.0f/16.f) * ((float)*inputLevel/128.0f);  // non-AGC gain multiplier
 | |
|   if (soundAgc) binScale = multAgc;                                                    // AGC gain
 | |
|   if (sampleAvg < 1) binScale = 0.001f;                                                 // silentium!
 | |
| 
 | |
| #ifdef SR_DEBUG
 | |
|   //The next lines are good for debugging, however too much flickering for non-developers ;-)
 | |
|   float my_magnitude = FFT_Magnitude / 16.0f;   // scale magnitude to be aligned with scaling of FFT bins
 | |
|   my_magnitude *= binScale;                     // apply gain
 | |
|   *maxVol = fmax(64, my_magnitude);             // set maxVal = max FFT result
 | |
| #endif
 | |
| 
 | |
|   for (int i=0; i<SEGLEN; i++) {
 | |
| 
 | |
|     uint16_t startBin = FIRSTBIN+i*(LASTBIN-FIRSTBIN)/SEGLEN;        // This is the START bin for this particular pixel.
 | |
|     uint16_t   endBin = FIRSTBIN+(i+1)*(LASTBIN-FIRSTBIN)/SEGLEN;    // This is the END bin for this particular pixel.
 | |
|     if (endBin > startBin) endBin --;                     // avoid overlapping
 | |
| 
 | |
|     double sumBin = 0;
 | |
| 
 | |
|     for (int j=startBin; j<=endBin; j++) {
 | |
|       sumBin += (fftBin[j] < soundSquelch*1.75f) ? 0 : fftBin[j];  // We need some sound temporary squelch for fftBin, because we didn't do it for the raw bins in audio_reactive.h
 | |
|     }
 | |
| 
 | |
|     sumBin = sumBin/(endBin-startBin+1);                  // Normalize it.
 | |
|     sumBin = sumBin * (i+5) / (endBin-startBin+5);        // Disgusting frequency adjustment calculation. Lows were too bright. Am open to quick 'n dirty alternatives.
 | |
| 
 | |
|     sumBin = sumBin * 8;                                  // Need to use the 'log' version for this. Why " * 8" ??
 | |
|     sumBin *= binScale;                                   // apply gain
 | |
| 
 | |
|     if (sumBin > maxVal) sumBin = maxVal;                 // Make sure our bin isn't higher than the max . . which we capped earlier.
 | |
| 
 | |
|     uint8_t bright = constrain(mapf(sumBin, 0, maxVal, 0, 255),0,255);  // Map the brightness in relation to maxVal and crunch to 8 bits.
 | |
| 
 | |
|     setPixelColor(i, color_blend(SEGCOLOR(1), color_from_palette(i*8+millis()/50, false, PALETTE_SOLID_WRAP, 0), bright));  // 'i' is just an index in the palette. The FFT value, bright, is the intensity.
 | |
|   } // for i
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_binmap()
 | |
| static const char *_data_FX_MODE_BINMAP PROGMEM = " ♫ Binmap@,,Input level=128;!,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //    ** Blurz      //
 | |
| //////////////////////
 | |
| uint16_t WS2812FX::mode_blurz(void) {                    // Blurz. By Andrew Tuline.
 | |
|   uint8_t *fftResult = nullptr;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     fftResult     =  (uint8_t*)um_data->u_data[1];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|   }
 | |
|   if (!fftResult) return mode_static();
 | |
| 
 | |
|   if (SEGENV.call == 0) {
 | |
|     fill(BLACK);
 | |
|     SEGENV.aux0 = 0;
 | |
|   }
 | |
| 
 | |
|   fade_out(SEGMENT.speed);
 | |
| 
 | |
|   uint16_t segLoc = random16(SEGLEN);
 | |
|   setPixelColor(segLoc, color_blend(SEGCOLOR(1), color_from_palette(fftResult[SEGENV.aux0], false, PALETTE_SOLID_WRAP, 0), 2*fftResult[SEGENV.aux0]));
 | |
|   ++(SEGENV.aux0) %= 16; // make sure it doesn't cross 16
 | |
| 
 | |
|   blur(SEGMENT.intensity);
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_blurz()
 | |
| static const char *_data_FX_MODE_BLURZ PROGMEM = " ♫ Blurz@Fade rate,Blur amount;!,Color mix;!";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //   ** DJLight        //
 | |
| /////////////////////////
 | |
| uint16_t WS2812FX::mode_DJLight(void) {                   // Written by ??? Adapted by Will Tatam.
 | |
|   const int NUM_LEDS = SEGLEN;                            // aka SEGLEN
 | |
|   const int mid = NUM_LEDS / 2;
 | |
| 
 | |
|   const uint16_t dataSize = SEGLEN*sizeof(CRGB);
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   uint8_t *fftResult = nullptr;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     fftResult     =  (uint8_t*)um_data->u_data[1];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|   }
 | |
|   if (!fftResult) return mode_static();
 | |
| 
 | |
|   uint8_t secondHand = micros()/(256-SEGMENT.speed)/500+1 % 64;
 | |
|   if (SEGENV.aux0 != secondHand) {                        // Triggered millis timing.
 | |
|     SEGENV.aux0 = secondHand;
 | |
| 
 | |
|     leds[mid] = CRGB(fftResult[15]/2, fftResult[5]/2, fftResult[0]/2); // 16-> 15 as 16 is out of bounds
 | |
|     leds[mid].fadeToBlackBy(map(fftResult[1*4], 0, 255, 255, 10));     // TODO - Update
 | |
| 
 | |
|     for (int i = NUM_LEDS - 1; i > mid; i--) leds[i] = leds[i - 1];    //move to the left
 | |
|     for (int i = 0; i < mid; i++)            leds[i] = leds[i + 1];    // move to the right
 | |
| 
 | |
|     for (uint16_t i=0; i<SEGLEN; i++) setPixelColor(i, leds[i]);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_DJLight()
 | |
| static const char *_data_FX_MODE_DJLIGHT PROGMEM = " ♫ DJ Light@Speed;;";
 | |
| 
 | |
| 
 | |
| ////////////////////
 | |
| //   ** Freqmap   //
 | |
| ////////////////////
 | |
| uint16_t WS2812FX::mode_freqmap(void) {                   // Map FFT_MajorPeak to SEGLEN. Would be better if a higher framerate.
 | |
|   // Start frequency = 60 Hz and log10(60) = 1.78
 | |
|   // End frequency = 5120 Hz and lo10(5120) = 3.71
 | |
| 
 | |
|   double FFT_MajorPeak = 0.0;
 | |
|   double FFT_Magnitude = 0.0;
 | |
|   uint8_t soundAgc = 0;
 | |
|   float sampleAvg = 0.0f;
 | |
|   float multAgc = 0.0f;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     FFT_MajorPeak = *(double*)um_data->u_data[6];
 | |
|     FFT_Magnitude = *(double*)um_data->u_data[7];
 | |
|     sampleAvg     = *(float*)um_data->u_data[8];
 | |
|     soundAgc      = *(uint8_t*)um_data->u_data[9];
 | |
|     multAgc       = *(float*)um_data->u_data[11];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|   }
 | |
| 
 | |
|   float my_magnitude = FFT_Magnitude / 4.0;
 | |
|   if (soundAgc) my_magnitude *= multAgc;
 | |
|   if (sampleAvg < 1 ) my_magnitude = 0.001;              // noise gate closed - mute
 | |
| 
 | |
|   fade_out(SEGMENT.speed);
 | |
| 
 | |
|   uint16_t locn = (log10f((float)FFT_MajorPeak) - 1.78f) * (float)SEGLEN/(3.71f-1.78f);  // log10 frequency range is from 1.78 to 3.71. Let's scale to SEGLEN.
 | |
| 
 | |
|   if (locn >=SEGLEN) locn = SEGLEN-1;
 | |
|   uint16_t pixCol = (log10f(FFT_MajorPeak) - 1.78f) * 255.0f/(3.71f-1.78f);   // Scale log10 of frequency values to the 255 colour index.
 | |
|   uint16_t bright = (int)my_magnitude;
 | |
| 
 | |
|   setPixelColor(locn, color_blend(SEGCOLOR(1), color_from_palette(SEGMENT.intensity+pixCol, false, PALETTE_SOLID_WRAP, 0), bright));
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_freqmap()
 | |
| static const char *_data_FX_MODE_FREQMAP PROGMEM = " ♫ Freqmap@Fade rate,Starting color;,!;!";
 | |
| 
 | |
| 
 | |
| ///////////////////////
 | |
| //   ** Freqmatrix   //
 | |
| ///////////////////////
 | |
| uint16_t WS2812FX::mode_freqmatrix(void) {                // Freqmatrix. By Andreas Pleschung.
 | |
|   double FFT_MajorPeak = 0.0;
 | |
|   float sampleAgc = 0.0f;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     FFT_MajorPeak = *(double*)um_data->u_data[6];
 | |
|     sampleAgc     = *(float*)um_data->u_data[10];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|   }
 | |
| 
 | |
|   uint8_t secondHand = micros()/(256-SEGMENT.speed)/500 % 16;
 | |
|   if(SEGENV.aux0 != secondHand) {
 | |
|     SEGENV.aux0 = secondHand;
 | |
| 
 | |
|     uint8_t sensitivity = map(SEGMENT.custom3, 0, 255, 1, 10);
 | |
|     int pixVal = (sampleAgc * SEGMENT.intensity * sensitivity) / 256.0f;
 | |
|     if (pixVal > 255) pixVal = 255;
 | |
| 
 | |
|     float intensity = map(pixVal, 0, 255, 0, 100) / 100.0f;  // make a brightness from the last avg
 | |
| 
 | |
|     CRGB color = 0;
 | |
| 
 | |
|     if (FFT_MajorPeak > 5120) FFT_MajorPeak = 0;
 | |
|     // MajorPeak holds the freq. value which is most abundant in the last sample.
 | |
|     // With our sampling rate of 10240Hz we have a usable freq range from roughtly 80Hz to 10240/2 Hz
 | |
|     // we will treat everything with less than 65Hz as 0
 | |
| 
 | |
|     if (FFT_MajorPeak < 80) {
 | |
|       color = CRGB::Black;
 | |
|     } else {
 | |
|       int upperLimit = 20 * SEGMENT.custom2;
 | |
|       int lowerLimit = 2 * SEGMENT.custom1;
 | |
|       int i =  lowerLimit!=upperLimit ? map(FFT_MajorPeak, lowerLimit, upperLimit, 0, 255) : FFT_MajorPeak;
 | |
|       uint16_t b = 255 * intensity;
 | |
|       if (b > 255) b = 255;
 | |
|       color = CHSV(i, 240, (uint8_t)b); // implicit conversion to RGB supplied by FastLED
 | |
|     }
 | |
| 
 | |
|     // shift the pixels one pixel up
 | |
|     for (uint16_t i = SEGLEN-1; i > 0; i--) setPixelColor(i, getPixelColor(i-1));
 | |
|     setPixelColor(0, color);
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_freqmatrix()
 | |
| static const char *_data_FX_MODE_FREQMATRIX PROGMEM = " ♫ Freqmatrix@Time delay,Sound effect,Low bin,High bin,Sensivity;;";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //   ** Freqpixels  //
 | |
| //////////////////////
 | |
| // Start frequency = 60 Hz and log10(60) = 1.78
 | |
| // End frequency = 5120 Hz and lo10(5120) = 3.71
 | |
| //  SEGMENT.speed select faderate
 | |
| //  SEGMENT.intensity select colour index
 | |
| uint16_t WS2812FX::mode_freqpixels(void) {                // Freqpixel. By Andrew Tuline.
 | |
|   double FFT_MajorPeak = 0.0;
 | |
|   double FFT_Magnitude = 0.0;
 | |
|   uint8_t soundAgc = 0;
 | |
|   float sampleAvg = 0.0f;
 | |
|   float multAgc = 0.0f;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     FFT_MajorPeak = *(double*)um_data->u_data[6];
 | |
|     FFT_Magnitude = *(double*)um_data->u_data[7];
 | |
|     sampleAvg     = *(float*)um_data->u_data[8];
 | |
|     soundAgc      = *(uint8_t*)um_data->u_data[9];
 | |
|     multAgc       = *(float*)um_data->u_data[11];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|   }
 | |
| 
 | |
|   float my_magnitude = FFT_Magnitude / 16.0;
 | |
|   if (soundAgc) my_magnitude *= multAgc;
 | |
|   if (sampleAvg < 1 ) my_magnitude = 0.001;              // noise gate closed - mute
 | |
| 
 | |
|   uint16_t fadeRate = 2*SEGMENT.speed - SEGMENT.speed*SEGMENT.speed/255;    // Get to 255 as quick as you can.
 | |
|   fade_out(fadeRate);
 | |
| 
 | |
|   for (int i=0; i < SEGMENT.intensity/32+1; i++) {
 | |
|     uint16_t locn = random16(0,SEGLEN);
 | |
|     uint8_t pixCol = (log10f(FFT_MajorPeak) - 1.78) * 255.0/(3.71-1.78);  // Scale log10 of frequency values to the 255 colour index.
 | |
|     setPixelColor(locn, color_blend(SEGCOLOR(1), color_from_palette(SEGMENT.intensity+pixCol, false, PALETTE_SOLID_WRAP, 0), (int)my_magnitude));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_freqpixels()
 | |
| static const char *_data_FX_MODE_FREQPIXELS PROGMEM = " ♫ Freqpixels@Fade rate,Starting colour and # of pixels;;";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //   ** Freqwave    //
 | |
| //////////////////////
 | |
| // Assign a color to the central (starting pixels) based on the predominant frequencies and the volume. The color is being determined by mapping the MajorPeak from the FFT
 | |
| // and then mapping this to the HSV color circle. Currently we are sampling at 10240 Hz, so the highest frequency we can look at is 5120Hz.
 | |
| //
 | |
| // SEGMENT.custom1: the lower cut off point for the FFT. (many, most time the lowest values have very little information since they are FFT conversion artifacts. Suggested value is close to but above 0
 | |
| // SEGMENT.custom2: The high cut off point. This depends on your sound profile. Most music looks good when this slider is between 50% and 100%.
 | |
| // SEGMENT.custom3: "preamp" for the audio signal for audio10.
 | |
| //
 | |
| // I suggest that for this effect you turn the brightness to 95%-100% but again it depends on your soundprofile you find yourself in.
 | |
| // Instead of using colorpalettes, This effect works on the HSV color circle with red being the lowest frequency
 | |
| //
 | |
| // As a compromise between speed and accuracy we are currently sampling with 10240Hz, from which we can then determine with a 512bin FFT our max frequency is 5120Hz.
 | |
| // Depending on the music stream you have you might find it useful to change the frequency mapping.
 | |
| uint16_t WS2812FX::mode_freqwave(void) {                  // Freqwave. By Andreas Pleschung.
 | |
|   double FFT_MajorPeak = 0.0;
 | |
|   uint8_t soundAgc = 0;
 | |
|   float sampleAgc = 0.0f, sampleAvg = 0.0f;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     FFT_MajorPeak = *(double*)um_data->u_data[6];
 | |
|     sampleAvg     = *(float*)um_data->u_data[8];
 | |
|     soundAgc      = *(uint8_t*)um_data->u_data[9];
 | |
|     sampleAgc     = *(float*)um_data->u_data[10];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|   }
 | |
| 
 | |
|   uint8_t secondHand = micros()/(256-SEGMENT.speed)/500 % 16;
 | |
|   if(SEGENV.aux0 != secondHand) {
 | |
|     SEGENV.aux0 = secondHand;
 | |
| 
 | |
|     //uint8_t fade = SEGMENT.custom3;
 | |
|     //uint8_t fadeval;
 | |
| 
 | |
|     float tmpSound = (soundAgc) ? sampleAgc : sampleAvg;
 | |
| 
 | |
|     float sensitivity = mapf(SEGMENT.custom3, 1, 255, 1, 10);
 | |
|     float pixVal = tmpSound * (float)SEGMENT.intensity / 256.0f * sensitivity;
 | |
|     if (pixVal > 255) pixVal = 255;
 | |
| 
 | |
|     float intensity = mapf(pixVal, 0, 255, 0, 100) / 100.0f;  // make a brightness from the last avg
 | |
| 
 | |
|     CRGB color = 0;
 | |
| 
 | |
|     if (FFT_MajorPeak > 5120) FFT_MajorPeak = 0.0f;
 | |
|     // MajorPeak holds the freq. value which is most abundant in the last sample.
 | |
|     // With our sampling rate of 10240Hz we have a usable freq range from roughtly 80Hz to 10240/2 Hz
 | |
|     // we will treat everything with less than 65Hz as 0
 | |
| 
 | |
|     if (FFT_MajorPeak < 80) {
 | |
|       color = CRGB::Black;
 | |
|     } else {
 | |
|       int upperLimit = 20 * SEGMENT.custom2;
 | |
|       int lowerLimit = 2 * SEGMENT.custom1;
 | |
|       int i =  lowerLimit!=upperLimit ? map(FFT_MajorPeak, lowerLimit, upperLimit, 0, 255) : FFT_MajorPeak;
 | |
|       uint16_t b = 255.0 * intensity;
 | |
|       if (b > 255) b=255;
 | |
|       color = CHSV(i, 240, (uint8_t)b); // implicit conversion to RGB supplied by FastLED
 | |
|     }
 | |
| 
 | |
|     setPixelColor(SEGLEN/2, color);
 | |
| 
 | |
|     // shift the pixels one pixel outwards
 | |
|     for (uint16_t i = SEGLEN-1; i > SEGLEN/2; i--) setPixelColor(i, getPixelColor(i-1));  // Move to the right.
 | |
|     for (uint16_t i = 0; i < SEGLEN/2; i++)        setPixelColor(i, getPixelColor(i+1));  // Move to the left.
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_freqwave()
 | |
| static const char *_data_FX_MODE_FREQWAVE PROGMEM = " ♫ Freqwave@Time delay,Sound effect,Low bin,High bin,Pre-amp;;";
 | |
| 
 | |
| 
 | |
| ///////////////////////
 | |
| //    ** Gravfreq    //
 | |
| ///////////////////////
 | |
| uint16_t WS2812FX::mode_gravfreq(void) {                  // Gravfreq. By Andrew Tuline.
 | |
| 
 | |
|   uint16_t dataSize = sizeof(gravity);
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   Gravity* gravcen = reinterpret_cast<Gravity*>(SEGENV.data);
 | |
| 
 | |
|   um_data_t *um_data;
 | |
|   uint8_t soundAgc = 0;
 | |
|   float sampleAgc = 0.0f, sampleAvg = 0.0f;
 | |
|   double FFT_MajorPeak = 0.0;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     FFT_MajorPeak = *(double*)um_data->u_data[6];
 | |
|     soundAgc      = *(uint8_t*)um_data->u_data[9];
 | |
|     sampleAgc     = *(float*)um_data->u_data[10];
 | |
|     sampleAvg     = *(float*)um_data->u_data[8];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|     sampleAvg = inoise8(12345); // I have no idea what that does
 | |
|   }
 | |
| 
 | |
|   fade_out(240);
 | |
| 
 | |
|   float tmpSound = (soundAgc) ? sampleAgc : sampleAvg;
 | |
|   float segmentSampleAvg = tmpSound * (float)SEGMENT.intensity / 255.0;
 | |
|   segmentSampleAvg *= 0.125; // divide by 8,  to compensate for later "sensitivty" upscaling
 | |
| 
 | |
|   float mySampleAvg = mapf(segmentSampleAvg*2.0, 0,32, 0, (float)SEGLEN/2.0); // map to pixels availeable in current segment 
 | |
|   int tempsamp = constrain(mySampleAvg,0,SEGLEN/2);     // Keep the sample from overflowing.
 | |
|   uint8_t gravity = 8 - SEGMENT.speed/32;
 | |
| 
 | |
|   for (int i=0; i<tempsamp; i++) {
 | |
| 
 | |
|     uint8_t index = (log10((int)FFT_MajorPeak) - (3.71-1.78)) * 255; //int? shouldn't it be floor() or similar
 | |
| 
 | |
|     setPixelColor(i+SEGLEN/2, color_from_palette(index, false, PALETTE_SOLID_WRAP, 0));
 | |
|     setPixelColor(SEGLEN/2-i-1, color_from_palette(index, false, PALETTE_SOLID_WRAP, 0));
 | |
|   }
 | |
| 
 | |
|   if (tempsamp >= gravcen->topLED)
 | |
|     gravcen->topLED = tempsamp-1;
 | |
|   else if (gravcen->gravityCounter % gravity == 0)
 | |
|     gravcen->topLED--;
 | |
| 
 | |
|   if (gravcen->topLED >= 0) {
 | |
|     setPixelColor(gravcen->topLED+SEGLEN/2, CRGB::Gray);
 | |
|     setPixelColor(SEGLEN/2-1-gravcen->topLED, CRGB::Gray);
 | |
|   }
 | |
|   gravcen->gravityCounter = (gravcen->gravityCounter + 1) % gravity;
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_gravfreq()
 | |
| static const char *_data_FX_MODE_GRAVFREQ PROGMEM = " ♫ Gravfreq@Rate of fall,Sensivity=128;,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //   ** Noisemove   //
 | |
| //////////////////////
 | |
| uint16_t WS2812FX::mode_noisemove(void) {                 // Noisemove.    By: Andrew Tuline
 | |
|   uint8_t *fftResult = nullptr;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     fftResult     =  (uint8_t*)um_data->u_data[1];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|   }
 | |
|   if (!fftResult) return mode_static();
 | |
| 
 | |
|   fade_out(224);                                          // Just in case something doesn't get faded.
 | |
| 
 | |
|   uint8_t numBins = map(SEGMENT.intensity,0,255,0,16);    // Map slider to fftResult bins.
 | |
|   for (int i=0; i<numBins; i++) {                         // How many active bins are we using.
 | |
|     uint16_t locn = inoise16(millis()*SEGMENT.speed+i*50000, millis()*SEGMENT.speed);   // Get a new pixel location from moving noise.
 | |
|     locn = map(locn, 7500, 58000, 0, SEGLEN-1);           // Map that to the length of the strand, and ensure we don't go over.
 | |
|     setPixelColor(locn, color_blend(SEGCOLOR(1), color_from_palette(i*64, false, PALETTE_SOLID_WRAP, 0), fftResult[i % 16]*4));
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_noisemove()
 | |
| static const char *_data_FX_MODE_NOISEMOVE PROGMEM = " ♫ Noisemove@Speed of perlin movement,Fade rate;,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////
 | |
| //   ** Rocktaves   //
 | |
| //////////////////////
 | |
| uint16_t WS2812FX::mode_rocktaves(void) {                 // Rocktaves. Same note from each octave is same colour.    By: Andrew Tuline
 | |
|   double FFT_MajorPeak = 0.0;
 | |
|   double FFT_Magnitude = 0.0;
 | |
|   uint8_t soundAgc = 0;
 | |
|   float sampleAvg = 0.0f;
 | |
|   float multAgc = 0.0f;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     FFT_MajorPeak = *(double*)um_data->u_data[6];
 | |
|     FFT_Magnitude = *(double*)um_data->u_data[7];
 | |
|     sampleAvg     = *(float*)um_data->u_data[8];
 | |
|     soundAgc      = *(uint8_t*)um_data->u_data[9];
 | |
|     multAgc       = *(float*)um_data->u_data[11];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|   }
 | |
| 
 | |
|   fade_out(128);                          // Just in case something doesn't get faded.
 | |
| 
 | |
|   float frTemp = FFT_MajorPeak;
 | |
|   uint8_t octCount = 0;                                   // Octave counter.
 | |
|   uint8_t volTemp = 0;
 | |
| 
 | |
|   float my_magnitude = FFT_Magnitude / 16.0;             // scale magnitude to be aligned with scaling of FFT bins
 | |
|   if (soundAgc) my_magnitude *= multAgc;                 // apply gain
 | |
|   if (sampleAvg < 1 ) my_magnitude = 0.001;              // mute
 | |
| 
 | |
|   if (my_magnitude > 32) volTemp = 255;                 // We need to squelch out the background noise.
 | |
| 
 | |
|   while ( frTemp > 249 ) {
 | |
|     octCount++;                                           // This should go up to 5.
 | |
|     frTemp = frTemp/2;
 | |
|   }
 | |
| 
 | |
|   frTemp -=132;                                           // This should give us a base musical note of C3
 | |
|   frTemp = fabs(frTemp * 2.1);                            // Fudge factors to compress octave range starting at 0 and going to 255;
 | |
| 
 | |
| //  leds[beatsin8(8+octCount*4,0,SEGLEN-1,0,octCount*8)] += CHSV((uint8_t)frTemp,255,volTemp);                 // Back and forth with different frequencies and phase shift depending on current octave.
 | |
|   uint16_t i = map(beatsin8(8+octCount*4, 0, 255, 0, octCount*8), 0, 255, 0, SEGLEN-1);
 | |
|   setPixelColor(i, color_add(getPixelColor(i),color_blend(SEGCOLOR(1), color_from_palette((uint8_t)frTemp, false, PALETTE_SOLID_WRAP, 0), volTemp)));
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_rocktaves()
 | |
| static const char *_data_FX_MODE_ROCKTAVES PROGMEM = " ♫ Rocktaves@;,!;!";
 | |
| 
 | |
| 
 | |
| ///////////////////////
 | |
| //   ** Waterfall    //
 | |
| ///////////////////////
 | |
| // Combines peak detection with FFT_MajorPeak and FFT_Magnitude.
 | |
| uint16_t WS2812FX::mode_waterfall(void) {                   // Waterfall. By: Andrew Tuline
 | |
|   if (SEGENV.call == 0) fill(BLACK);
 | |
| 
 | |
|   uint8_t *binNum = (uint8_t*)&SEGENV.aux1, *maxVol = (uint8_t*)(&SEGENV.aux1+1); // just in case assignment
 | |
|   uint8_t samplePeak = 0;
 | |
|   double FFT_MajorPeak = 0.0;
 | |
|   double FFT_Magnitude = 0.0;
 | |
|   uint8_t soundAgc = 0;
 | |
|   float sampleAvg = 0.0f;
 | |
|   float multAgc = 0.0f;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     maxVol        =  (uint8_t*)um_data->u_data[0];
 | |
|     binNum        =  (uint8_t*)um_data->u_data[5];
 | |
|     FFT_MajorPeak = *(double*)um_data->u_data[6];
 | |
|     FFT_Magnitude = *(double*)um_data->u_data[7];
 | |
|     sampleAvg     = *(float*)um_data->u_data[8];
 | |
|     soundAgc      = *(uint8_t*)um_data->u_data[9];
 | |
|     multAgc       = *(float*)um_data->u_data[11];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|   }
 | |
| 
 | |
|   *binNum = SEGMENT.custom2;                               // Select a bin.
 | |
|   *maxVol = SEGMENT.custom3/2;                             // Our volume comparator.
 | |
| 
 | |
|   uint8_t secondHand = micros() / (256-SEGMENT.speed)/500 + 1 % 16;
 | |
|   if (SEGENV.aux0 != secondHand) {                        // Triggered millis timing.
 | |
|     SEGENV.aux0 = secondHand;
 | |
| 
 | |
|     float my_magnitude = FFT_Magnitude / 8.0f;
 | |
|     if (soundAgc) my_magnitude *= multAgc;
 | |
|     if (sampleAvg < 1 ) my_magnitude = 0.001f;             // noise gate closed - mute
 | |
| 
 | |
|     uint8_t pixCol = (log10f((float)FFT_MajorPeak) - 2.26f) * 177;  // log10 frequency range is from 2.26 to 3.7. Let's scale accordingly.
 | |
| 
 | |
|     if (samplePeak) {
 | |
|       setPixelColor(SEGLEN-1, CHSV(92,92,92));
 | |
|     } else {
 | |
|       setPixelColor(SEGLEN-1, color_blend(SEGCOLOR(1), color_from_palette(pixCol+SEGMENT.intensity, false, PALETTE_SOLID_WRAP, 0), (int)my_magnitude));
 | |
|     }
 | |
|     for (uint16_t i=0; i<SEGLEN-1; i++) setPixelColor(i, getPixelColor(i+1)); // shift left
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_waterfall()
 | |
| static const char *_data_FX_MODE_WATERFALL PROGMEM = " ♫ Waterfall@!,Adjust color,,Select bin, Volume (minimum);!,!;!";
 | |
| 
 | |
| 
 | |
| //////////////////////////////////////////////
 | |
| //       START of 2D REACTIVE ROUTINES      //
 | |
| //////////////////////////////////////////////
 | |
| 
 | |
| /////////////////////////
 | |
| //     ** 2D GEQ       //
 | |
| /////////////////////////
 | |
| uint16_t WS2812FX::GEQ_base(bool centered_horizontal, bool centered_vertical, bool color_vertical) {                     // By Will Tatam. Refactor by Ewoud Wijma.
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();    // using width*height prevents reallocation if mirroring is enabled
 | |
| 
 | |
|   if (!SEGENV.allocateData(dataSize + 64*sizeof(uint16_t))) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
|   uint16_t *previousBarHeight = reinterpret_cast<uint16_t*>(SEGENV.data + dataSize);
 | |
| 
 | |
|   uint8_t *fftResult = nullptr;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     fftResult     =  (uint8_t*)um_data->u_data[1];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|   }
 | |
|   if (!fftResult) return mode_static();
 | |
| 
 | |
|   fadeToBlackBy(leds, SEGMENT.speed);
 | |
| 
 | |
|   bool rippleTime;
 | |
|   if (millis() - SEGENV.step >= 255 - SEGMENT.intensity) {
 | |
|     SEGENV.step = millis();
 | |
|     rippleTime = true;
 | |
|   } else
 | |
|     rippleTime = false;
 | |
| 
 | |
|   //static int previousBarHeight[64]; //array of previous bar heights per frequency band
 | |
|   if (SEGENV.call == 0) for (uint16_t i=0; i<64; i++) previousBarHeight[i] = 0;
 | |
| 
 | |
|   int xCount = cols;
 | |
|   if (centered_vertical) xCount /= 2;
 | |
| 
 | |
|   for (int x=0; x < xCount; x++) {
 | |
|     int band = map(x, 0, xCount-1, 0, 15);
 | |
|     int barHeight = map(fftResult[band], 0, 255, 0, rows);
 | |
|     if ((barHeight % 2 == 1) && centered_horizontal) barHeight++;                 //get an even barHeight if centered_horizontal
 | |
|     int yStartBar = centered_horizontal ? (rows - barHeight) / 2 : 0;             //lift up the bar if centered_horizontal
 | |
|     int yStartPeak = centered_horizontal ? (rows - previousBarHeight[x]) / 2 : 0; //lift up the peaks if centered_horizontal
 | |
| 
 | |
|     for (int y=0; y < rows; y++)
 | |
|     {
 | |
|       CRGB heightColor = CRGB::Black;
 | |
|       uint16_t colorIndex;
 | |
|       if (color_vertical) {
 | |
|         if (centered_horizontal)
 | |
|           colorIndex = map(abs(y - (rows - 1)/2.0), 0, rows/2 - 1, 0, 255);
 | |
|         else
 | |
|           colorIndex = map(y, 0, rows - 1, 0, 255);
 | |
|       } else
 | |
|         colorIndex = band * 17;
 | |
|       heightColor = color_from_palette(colorIndex, false, PALETTE_SOLID_WRAP, 0);
 | |
| 
 | |
|       CRGB ledColor = CRGB::Black; //if not part of bars or peak, make black (not fade to black)
 | |
| 
 | |
|       //bar
 | |
|       if (y >= yStartBar && y < yStartBar + barHeight)
 | |
|         ledColor = heightColor;
 | |
| 
 | |
|       //low and high peak (must exist && on peak position && only below if centered_horizontal effect)
 | |
|       if ((previousBarHeight[x] > 0) && (SEGMENT.intensity < 255) && (y==yStartPeak || y==yStartPeak + previousBarHeight[x]-1) && (centered_horizontal || y!=yStartPeak))
 | |
|         ledColor = SEGCOLOR(2)==CRGB::Black ? heightColor : CRGB(SEGCOLOR(2)); //low peak
 | |
| 
 | |
|       if (centered_vertical) {
 | |
|         leds[XY(cols / 2 + x, rows - 1 - y)] = ledColor;
 | |
|         leds[XY(cols / 2 - 1 - x, rows - 1 - y)] = ledColor;
 | |
|       } else
 | |
|         leds[XY(x, rows - 1 - y)] = ledColor;
 | |
|     }
 | |
| 
 | |
|     if (rippleTime) previousBarHeight[x] -= centered_horizontal ? 2 : 1; //delay/ripple effect
 | |
|     if (barHeight > previousBarHeight[x]) previousBarHeight[x] = barHeight; //drive the peak up
 | |
|   }
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } //GEQ_base
 | |
| 
 | |
| uint16_t WS2812FX::mode_2DGEQ(void) {                     // By Will Tatam. Code reduction by Ewoud Wijma.
 | |
|   return GEQ_base(false, false, false);
 | |
| } // mode_2DGEQ()
 | |
| static const char *_data_FX_MODE_2DGEQ PROGMEM = " ♫ 2D GEQ@Bar speed,Ripple decay;,,Peak Color;!";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //   ** 2D CenterBars  //
 | |
| /////////////////////////
 | |
| uint16_t WS2812FX::mode_2DCenterBars(void) {              // Written by Scott Marley Adapted by  Spiro-C..
 | |
|   return GEQ_base(SEGMENT.custom1 > 128, SEGMENT.custom2 > 128, SEGMENT.custom3 > 128);
 | |
| } // mode_2DCenterBars()
 | |
| static const char *_data_FX_MODE_2DCENTERBARS PROGMEM = " ♫ 2D CenterBars@Bar speed=250,Ripple decay=250,Center ↔ ☑=192,Center ↕ ☑=192, Color ↕ ☑=192;,,Peak Color;!=11";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //  ** 2D Funky plank  //
 | |
| /////////////////////////
 | |
| uint16_t WS2812FX::mode_2DFunkyPlank(void) {              // Written by ??? Adapted by Will Tatam.
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
|   const uint16_t dataSize = sizeof(CRGB) * SEGMENT.width() * SEGMENT.height();  // using width*height prevents reallocation if mirroring is enabled
 | |
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed
 | |
|   CRGB *leds = reinterpret_cast<CRGB*>(SEGENV.data);
 | |
| 
 | |
|   int NUMB_BANDS = map(SEGMENT.custom1, 0, 255, 1, 16);
 | |
|   int barWidth = (cols / NUMB_BANDS);
 | |
|   int bandInc = 1;
 | |
|   if (barWidth == 0) {
 | |
|     // Matrix narrower than fft bands
 | |
|     barWidth = 1;
 | |
|     bandInc = (NUMB_BANDS / cols);
 | |
|   }
 | |
| 
 | |
|   uint8_t *fftResult = nullptr;
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     fftResult     =  (uint8_t*)um_data->u_data[1];
 | |
|   } else {
 | |
|     // add support for no audio data
 | |
|   }
 | |
|   if (!fftResult) return mode_static();
 | |
| 
 | |
|   uint8_t secondHand = micros()/(256-SEGMENT.speed)/500+1 % 64;
 | |
|   if (SEGENV.aux0 != secondHand) {                        // Triggered millis timing.
 | |
|     SEGENV.aux0 = secondHand;
 | |
| 
 | |
|     // display values of
 | |
|     int b = 0;
 | |
|     for (int band = 0; band < NUMB_BANDS; band += bandInc, b++) {
 | |
|       int hue = fftResult[band];
 | |
|       int v = map(fftResult[band], 0, 255, 10, 255);
 | |
|       for (int w = 0; w < barWidth; w++) {
 | |
|          int xpos = (barWidth * b) + w;
 | |
|          leds[XY(xpos, 0)] = CHSV(hue, 255, v);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Update the display:
 | |
|     for (int i = (rows - 1); i > 0; i--) {
 | |
|       for (int j = (cols - 1); j >= 0; j--) {
 | |
|         int src = XY(j, (i - 1));
 | |
|         int dst = XY(j, i);
 | |
|         leds[dst] = leds[src];
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   setPixels(leds);
 | |
|   return FRAMETIME;
 | |
| } // mode_2DFunkyPlank
 | |
| static const char *_data_FX_MODE_2DFUNKYPLANK PROGMEM = " ♫ 2D Funky Plank@Scroll speed,,# of bands;;";
 | |
| 
 | |
| 
 | |
| /////////////////////////
 | |
| //     2D Akemi        //
 | |
| /////////////////////////
 | |
| static uint8_t akemi[] PROGMEM = {
 | |
|   0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,0,0,0,0,0,2,2,2,2,2,2,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,0,0,0,2,2,3,3,3,3,3,3,2,2,0,0,0,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,0,0,2,3,3,0,0,0,0,0,0,3,3,2,0,0,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,0,2,3,0,0,0,6,5,5,4,0,0,0,3,2,0,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,2,3,0,0,6,6,5,5,5,5,4,4,0,0,3,2,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,2,3,0,6,5,5,5,5,5,5,5,5,4,0,3,2,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,2,3,0,6,5,5,5,5,5,5,5,5,5,5,4,0,3,2,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,3,2,0,6,5,5,5,5,5,5,5,5,5,5,4,0,2,3,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,3,2,3,6,5,5,7,7,5,5,5,5,7,7,5,5,4,3,2,3,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,2,3,1,3,6,5,1,7,7,7,5,5,1,7,7,7,5,4,3,1,3,2,0,0,0,0,0,
 | |
|   0,0,0,0,0,8,3,1,3,6,5,1,7,7,7,5,5,1,7,7,7,5,4,3,1,3,8,9,0,0,0,0,
 | |
|   0,0,0,0,0,8,3,1,3,6,5,5,1,1,5,5,5,5,1,1,5,5,4,3,1,3,8,0,0,0,0,0,
 | |
|   0,0,0,0,0,2,3,1,3,6,5,5,5,5,5,5,5,5,5,5,5,5,4,3,1,3,2,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,3,2,3,6,5,5,5,5,5,5,5,5,5,5,5,5,4,3,2,3,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,0,6,5,5,5,5,5,7,7,5,5,5,5,5,4,0,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,0,6,5,5,5,5,5,5,5,5,5,5,5,5,4,0,0,0,0,0,0,0,0,0,
 | |
|   1,0,0,0,0,0,0,0,0,6,5,5,5,5,5,5,5,5,5,5,5,5,4,0,0,0,0,0,0,0,0,2,
 | |
|   0,2,2,2,0,0,0,0,0,6,5,5,5,5,5,5,5,5,5,5,5,5,4,0,0,0,0,0,2,2,2,0,
 | |
|   0,0,0,3,2,0,0,0,6,5,4,4,4,4,4,4,4,4,4,4,4,4,4,4,0,0,0,2,2,0,0,0,
 | |
|   0,0,0,3,2,0,0,0,6,5,5,5,5,5,5,5,5,5,5,5,5,5,5,4,0,0,0,2,3,0,0,0,
 | |
|   0,0,0,0,3,2,0,0,0,0,3,3,0,3,3,0,0,3,3,0,3,3,0,0,0,0,2,2,0,0,0,0,
 | |
|   0,0,0,0,3,2,0,0,0,0,3,2,0,3,2,0,0,3,2,0,3,2,0,0,0,0,2,3,0,0,0,0,
 | |
|   0,0,0,0,0,3,2,0,0,3,2,0,0,3,2,0,0,3,2,0,0,3,2,0,0,2,3,0,0,0,0,0,
 | |
|   0,0,0,0,0,3,2,2,2,2,0,0,0,3,2,0,0,3,2,0,0,0,3,2,2,2,3,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,3,3,3,0,0,0,0,3,2,0,0,3,2,0,0,0,0,3,3,3,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,0,0,0,0,0,3,2,0,0,3,2,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,0,0,0,0,0,3,2,0,0,3,2,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,0,0,0,0,0,3,2,0,0,3,2,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,0,0,0,0,0,3,2,0,0,3,2,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,0,0,0,0,0,3,2,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | |
|   0,0,0,0,0,0,0,0,0,0,0,0,0,3,2,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
 | |
| };
 | |
| 
 | |
| uint16_t WS2812FX::mode_2DAkemi(void) {
 | |
|   if (!isMatrix) return mode_static(); // not a 2D set-up
 | |
| 
 | |
|   const uint16_t cols = SEGMENT.virtualWidth();
 | |
|   const uint16_t rows = SEGMENT.virtualHeight();
 | |
| 
 | |
|   uint16_t counter = (now * ((SEGMENT.speed >> 2) +2)) & 0xFFFF;
 | |
|   counter = counter >> 8;
 | |
| 
 | |
|   const float lightFactor  = 0.15f;
 | |
|   const float normalFactor = 0.4f;
 | |
|   float base = 0.0f;
 | |
|   uint8_t *fftResult = nullptr;
 | |
| 
 | |
|   um_data_t *um_data;
 | |
|   if (usermods.getUMData(&um_data, USERMOD_ID_AUDIOREACTIVE)) {
 | |
|     fftResult = (uint8_t*)um_data->u_data[1];
 | |
|     base = fftResult[0]/255.0f;
 | |
|   }
 | |
| 
 | |
|   //draw and color Akemi
 | |
|   for (uint16_t y=0; y < rows; y++) for (uint16_t x=0; x < cols; x++) {
 | |
|     CRGB color;
 | |
|     CRGB soundColor = ORANGE;
 | |
|     CRGB faceColor  = color_wheel(counter);
 | |
|     CRGB armsAndLegsColor = SEGCOLOR(1) > 0 ? SEGCOLOR(1) : 0xFFE0A0; //default warmish white 0xABA8FF; //0xFF52e5;//
 | |
|     uint8_t ak = pgm_read_byte_near(akemi + ((y * 32)/rows) * 32 + (x * 32)/cols); // akemi[(y * 32)/rows][(x * 32)/cols]
 | |
|     switch (ak) {
 | |
|       case 0: color = BLACK; break;
 | |
|       case 3: armsAndLegsColor.r *= lightFactor;  armsAndLegsColor.g *= lightFactor;  armsAndLegsColor.b *= lightFactor;  color = armsAndLegsColor; break; //light arms and legs 0x9B9B9B
 | |
|       case 2: armsAndLegsColor.r *= normalFactor; armsAndLegsColor.g *= normalFactor; armsAndLegsColor.b *= normalFactor; color = armsAndLegsColor; break; //normal arms and legs 0x888888
 | |
|       case 1: color = armsAndLegsColor; break; //dark arms and legs 0x686868
 | |
|       case 6: faceColor.r *= lightFactor;  faceColor.g *= lightFactor;  faceColor.b *= lightFactor;  color=faceColor; break; //light face 0x31AAFF
 | |
|       case 5: faceColor.r *= normalFactor; faceColor.g *= normalFactor; faceColor.b *= normalFactor; color=faceColor; break; //normal face 0x0094FF
 | |
|       case 4: color = faceColor; break; //dark face 0x007DC6
 | |
|       case 7: color = SEGCOLOR(2) > 0 ? SEGCOLOR(2) : 0xFFFFFF; break; //eyes and mouth default white
 | |
|       case 8: if (base > 0.4) {soundColor.r *= base; soundColor.g *= base; soundColor.b *= base; color=soundColor;} else color = armsAndLegsColor; break;
 | |
|       default: color = BLACK;
 | |
|     }
 | |
| 
 | |
|     if (SEGMENT.intensity > 128 && fftResult && fftResult[0] > 128) { //dance if base is high
 | |
|       setPixelColorXY(x, 0, BLACK);
 | |
|       setPixelColorXY(x, y+1, color);
 | |
|     } else
 | |
|       setPixelColorXY(x, y, color);
 | |
|   }
 | |
| 
 | |
|   //add geq left and right
 | |
|   if (um_data && fftResult) {
 | |
|     for (uint16_t x=0; x < cols/8; x++) {
 | |
|       uint16_t band = x * cols/8;
 | |
|       uint16_t barHeight = map(fftResult[band], 0, 255, 0, 17*rows/32);
 | |
|       CRGB color = color_from_palette((band * 35), false, PALETTE_SOLID_WRAP, 0);
 | |
| 
 | |
|       for (uint16_t y=0; y < barHeight; y++) {
 | |
|         setPixelColorXY(x, rows/2-y, color);
 | |
|         setPixelColorXY(cols-1-x, rows/2-y, color);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FRAMETIME;
 | |
| } // mode_2DAkemi
 | |
| static const char *_data_FX_MODE_2DAKEMI PROGMEM = "2D Akemi@Color speed,Dance;Head palette,Arms & Legs,Eyes & Mouth;Face palette";
 | |
| 
 | |
| 
 | |
| //////////////////////////////////////////////////////////////////////////////////////////
 | |
| // mode data
 | |
| static const char *_data_RESERVED PROGMEM = "Reserved";
 | |
| const char *WS2812FX::_modeData[MODE_COUNT] = {
 | |
|   _data_FX_MODE_STATIC,
 | |
|   _data_FX_MODE_BLINK,
 | |
|   _data_FX_MODE_BREATH,
 | |
|   _data_FX_MODE_COLOR_WIPE,
 | |
|   _data_FX_MODE_COLOR_WIPE_RANDOM,
 | |
|   _data_FX_MODE_RANDOM_COLOR,
 | |
|   _data_FX_MODE_COLOR_SWEEP,
 | |
|   _data_FX_MODE_DYNAMIC,
 | |
|   _data_FX_MODE_RAINBOW,
 | |
|   _data_FX_MODE_RAINBOW_CYCLE,
 | |
|   _data_FX_MODE_SCAN,
 | |
|   _data_FX_MODE_DUAL_SCAN,
 | |
|   _data_FX_MODE_FADE,
 | |
|   _data_FX_MODE_THEATER_CHASE,
 | |
|   _data_FX_MODE_THEATER_CHASE_RAINBOW,
 | |
|   _data_FX_MODE_RUNNING_LIGHTS,
 | |
|   _data_FX_MODE_SAW,
 | |
|   _data_FX_MODE_TWINKLE,
 | |
|   _data_FX_MODE_DISSOLVE,
 | |
|   _data_FX_MODE_DISSOLVE_RANDOM,
 | |
|   _data_FX_MODE_SPARKLE,
 | |
|   _data_FX_MODE_FLASH_SPARKLE,
 | |
|   _data_FX_MODE_HYPER_SPARKLE,
 | |
|   _data_FX_MODE_STROBE,
 | |
|   _data_FX_MODE_STROBE_RAINBOW,
 | |
|   _data_FX_MODE_MULTI_STROBE,
 | |
|   _data_FX_MODE_BLINK_RAINBOW,
 | |
|   _data_FX_MODE_ANDROID,
 | |
|   _data_FX_MODE_CHASE_COLOR,
 | |
|   _data_FX_MODE_CHASE_RANDOM,
 | |
|   _data_FX_MODE_CHASE_RAINBOW,
 | |
|   _data_FX_MODE_CHASE_FLASH,
 | |
|   _data_FX_MODE_CHASE_FLASH_RANDOM,
 | |
|   _data_FX_MODE_CHASE_RAINBOW_WHITE,
 | |
|   _data_FX_MODE_COLORFUL,
 | |
|   _data_FX_MODE_TRAFFIC_LIGHT,
 | |
|   _data_FX_MODE_COLOR_SWEEP_RANDOM,
 | |
|   _data_FX_MODE_RUNNING_COLOR,
 | |
|   _data_FX_MODE_AURORA,
 | |
|   _data_FX_MODE_RUNNING_RANDOM,
 | |
|   _data_FX_MODE_LARSON_SCANNER,
 | |
|   _data_FX_MODE_COMET,
 | |
|   _data_FX_MODE_FIREWORKS,
 | |
|   _data_FX_MODE_RAIN,
 | |
|   _data_FX_MODE_TETRIX,
 | |
|   _data_FX_MODE_FIRE_FLICKER,
 | |
|   _data_FX_MODE_GRADIENT,
 | |
|   _data_FX_MODE_LOADING,
 | |
|   _data_FX_MODE_POLICE,
 | |
|   _data_FX_MODE_FAIRY,
 | |
|   _data_FX_MODE_TWO_DOTS,
 | |
|   _data_FX_MODE_FAIRYTWINKLE,
 | |
|   _data_FX_MODE_RUNNING_DUAL,
 | |
|   _data_FX_MODE_HALLOWEEN,
 | |
|   _data_FX_MODE_TRICOLOR_CHASE,
 | |
|   _data_FX_MODE_TRICOLOR_WIPE,
 | |
|   _data_FX_MODE_TRICOLOR_FADE,
 | |
|   _data_FX_MODE_LIGHTNING,
 | |
|   _data_FX_MODE_ICU,
 | |
|   _data_FX_MODE_MULTI_COMET,
 | |
|   _data_FX_MODE_DUAL_LARSON_SCANNER,
 | |
|   _data_FX_MODE_RANDOM_CHASE,
 | |
|   _data_FX_MODE_OSCILLATE,
 | |
|   _data_FX_MODE_PRIDE_2015,
 | |
|   _data_FX_MODE_JUGGLE,
 | |
|   _data_FX_MODE_PALETTE,
 | |
|   _data_FX_MODE_FIRE_2012,
 | |
|   _data_FX_MODE_COLORWAVES,
 | |
|   _data_FX_MODE_BPM,
 | |
|   _data_FX_MODE_FILLNOISE8,
 | |
|   _data_FX_MODE_NOISE16_1,
 | |
|   _data_FX_MODE_NOISE16_2,
 | |
|   _data_FX_MODE_NOISE16_3,
 | |
|   _data_FX_MODE_NOISE16_4,
 | |
|   _data_FX_MODE_COLORTWINKLE,
 | |
|   _data_FX_MODE_LAKE,
 | |
|   _data_FX_MODE_METEOR,
 | |
|   _data_FX_MODE_METEOR_SMOOTH,
 | |
|   _data_FX_MODE_RAILWAY,
 | |
|   _data_FX_MODE_RIPPLE,
 | |
|   _data_FX_MODE_TWINKLEFOX,
 | |
|   _data_FX_MODE_TWINKLECAT,
 | |
|   _data_FX_MODE_HALLOWEEN_EYES,
 | |
|   _data_FX_MODE_STATIC_PATTERN,
 | |
|   _data_FX_MODE_TRI_STATIC_PATTERN,
 | |
|   _data_FX_MODE_SPOTS,
 | |
|   _data_FX_MODE_SPOTS_FADE,
 | |
|   _data_FX_MODE_GLITTER,
 | |
|   _data_FX_MODE_CANDLE,
 | |
|   _data_FX_MODE_STARBURST,
 | |
|   _data_FX_MODE_EXPLODING_FIREWORKS,
 | |
|   _data_FX_MODE_BOUNCINGBALLS,
 | |
|   _data_FX_MODE_SINELON,
 | |
|   _data_FX_MODE_SINELON_DUAL,
 | |
|   _data_FX_MODE_SINELON_RAINBOW,
 | |
|   _data_FX_MODE_POPCORN,
 | |
|   _data_FX_MODE_DRIP,
 | |
|   _data_FX_MODE_PLASMA,
 | |
|   _data_FX_MODE_PERCENT,
 | |
|   _data_FX_MODE_RIPPLE_RAINBOW,
 | |
|   _data_FX_MODE_HEARTBEAT,
 | |
|   _data_FX_MODE_PACIFICA,
 | |
|   _data_FX_MODE_CANDLE_MULTI,
 | |
|   _data_FX_MODE_SOLID_GLITTER,
 | |
|   _data_FX_MODE_SUNRISE,
 | |
|   _data_FX_MODE_PHASED,
 | |
|   _data_FX_MODE_TWINKLEUP,
 | |
|   _data_FX_MODE_NOISEPAL,
 | |
|   _data_FX_MODE_SINEWAVE,
 | |
|   _data_FX_MODE_PHASEDNOISE,
 | |
|   _data_FX_MODE_FLOW,
 | |
|   _data_FX_MODE_CHUNCHUN,
 | |
|   _data_FX_MODE_DANCING_SHADOWS,
 | |
|   _data_FX_MODE_WASHING_MACHINE,
 | |
|   _data_FX_MODE_CANDY_CANE,
 | |
|   _data_FX_MODE_BLENDS,
 | |
|   _data_FX_MODE_TV_SIMULATOR,
 | |
|   _data_FX_MODE_DYNAMIC_SMOOTH,
 | |
|   // new effects
 | |
|   _data_FX_MODE_SPACESHIPS,
 | |
|   _data_FX_MODE_CRAZYBEES,
 | |
|   _data_FX_MODE_GHOST_RIDER,
 | |
|   _data_FX_MODE_BLOBS,
 | |
|   _data_FX_MODE_SCROLL_TEXT,
 | |
|   _data_FX_MODE_DRIFT_ROSE,
 | |
|   _data_RESERVED,
 | |
|   _data_RESERVED,
 | |
|   _data_RESERVED,
 | |
|   _data_RESERVED,
 | |
|   // WLED-SR effects
 | |
|   _data_FX_MODE_PIXELS,
 | |
|   _data_FX_MODE_PIXELWAVE,
 | |
|   _data_FX_MODE_JUGGLES,
 | |
|   _data_FX_MODE_MATRIPIX,
 | |
|   _data_FX_MODE_GRAVIMETER,
 | |
|   _data_FX_MODE_PLASMOID,
 | |
|   _data_FX_MODE_PUDDLES,
 | |
|   _data_FX_MODE_MIDNOISE,
 | |
|   _data_FX_MODE_NOISEMETER,
 | |
|   _data_FX_MODE_FREQWAVE,
 | |
|   _data_FX_MODE_FREQMATRIX,
 | |
|   _data_FX_MODE_2DGEQ,
 | |
|   _data_FX_MODE_WATERFALL,
 | |
|   _data_FX_MODE_FREQPIXELS,
 | |
|   _data_FX_MODE_BINMAP,
 | |
|   _data_FX_MODE_NOISEFIRE,
 | |
|   _data_FX_MODE_PUDDLEPEAK,
 | |
|   _data_FX_MODE_NOISEMOVE,
 | |
|   _data_FX_MODE_2DNOISE,
 | |
|   _data_FX_MODE_PERLINMOVE,
 | |
|   _data_FX_MODE_RIPPLEPEAK,
 | |
|   _data_FX_MODE_2DFIRENOISE,
 | |
|   _data_FX_MODE_2DSQUAREDSWIRL,
 | |
|   _data_RESERVED,               // was 2D Fire2012
 | |
|   _data_FX_MODE_2DDNA,
 | |
|   _data_FX_MODE_2DMATRIX,
 | |
|   _data_FX_MODE_2DMETABALLS,
 | |
|   _data_FX_MODE_FREQMAP,
 | |
|   _data_FX_MODE_GRAVCENTER,
 | |
|   _data_FX_MODE_GRAVCENTRIC,
 | |
|   _data_FX_MODE_GRAVFREQ,
 | |
|   _data_FX_MODE_DJLIGHT,
 | |
|   _data_FX_MODE_2DFUNKYPLANK,
 | |
|   _data_FX_MODE_2DCENTERBARS,
 | |
|   _data_FX_MODE_2DPULSER,
 | |
|   _data_FX_MODE_BLURZ,
 | |
|   _data_FX_MODE_2DDRIFT,
 | |
|   _data_FX_MODE_2DWAVERLY,
 | |
|   _data_FX_MODE_2DSUNRADIATION,
 | |
|   _data_FX_MODE_2DCOLOREDBURSTS,
 | |
|   _data_FX_MODE_2DJULIA,
 | |
|   _data_RESERVED,               //reserved FX_MODE_2DPOOLNOISE
 | |
|   _data_RESERVED,               //reserved FX_MODE_2DTWISTER
 | |
|   _data_RESERVED,               //reserved FX_MODE_2DCAELEMENTATY
 | |
|   _data_FX_MODE_2DGAMEOFLIFE,
 | |
|   _data_FX_MODE_2DTARTAN,
 | |
|   _data_FX_MODE_2DPOLARLIGHTS,
 | |
|   _data_FX_MODE_2DSWIRL,
 | |
|   _data_FX_MODE_2DLISSAJOUS,
 | |
|   _data_FX_MODE_2DFRIZZLES,
 | |
|   _data_FX_MODE_2DPLASMABALL,
 | |
|   _data_FX_MODE_FLOWSTRIPE,
 | |
|   _data_FX_MODE_2DHIPHOTIC,
 | |
|   _data_FX_MODE_2DSINDOTS,
 | |
|   _data_FX_MODE_2DDNASPIRAL,
 | |
|   _data_FX_MODE_2DBLACKHOLE,
 | |
|   _data_FX_MODE_WAVESINS,
 | |
|   _data_FX_MODE_ROCKTAVES,
 | |
|   _data_FX_MODE_2DAKEMI
 | |
|   //_data_FX_MODE_CUSTOMEFFECT    //WLEDSR Custom Effects
 | |
| };
 | 
