From 1d3ded9080774b574dc2bd2aee9c9cb5be28b668 Mon Sep 17 00:00:00 2001 From: Mateusz Salamon Date: Sun, 28 Oct 2018 00:08:20 +0100 Subject: [PATCH] FX effects for whole strip. Based in WS2812BFX Arduino library by kitesurfer1404. Every effect concerns whole strip - no segments. I added also HSV color mode and two FX modes related to it. --- Inc/ws2812b.h | 5 +- Inc/ws2812b_fx.h | 196 ++++++ README.md | 5 +- Src/main.c | 95 +-- Src/ws2812b.c | 23 +- Src/ws2812b_fx.c | 1673 ++++++++++++++++++++++++++++++++++++++++++++++ 6 files changed, 1913 insertions(+), 84 deletions(-) create mode 100644 Inc/ws2812b_fx.h create mode 100644 Src/ws2812b_fx.c diff --git a/Inc/ws2812b.h b/Inc/ws2812b.h index 64c7d83..ff8eb2f 100644 --- a/Inc/ws2812b.h +++ b/Inc/ws2812b.h @@ -20,9 +20,12 @@ typedef struct ws2812b_color { } ws2812b_color; void WS2812B_Init(SPI_HandleTypeDef * spi_handler); -void WS2812B_SetDiodeColor(int16_t diode_id, ws2812b_color color); +void WS2812B_SetDiodeColor(int16_t diode_id, uint32_t color); +void WS2812B_SetDiodeColorStruct(int16_t diode_id, ws2812b_color color); void WS2812B_SetDiodeRGB(int16_t diode_id, uint8_t R, uint8_t G, uint8_t B); void WS2812B_SetDiodeHSV(int16_t diode_id, uint16_t Hue, uint8_t Saturation, uint8_t Brightness); +uint32_t WS2812B_GetColor(int16_t diode_id); +uint8_t* WS2812B_GetPixels(void); void WS2812B_Refresh(); // color correction diff --git a/Inc/ws2812b_fx.h b/Inc/ws2812b_fx.h new file mode 100644 index 0000000..9bbf1eb --- /dev/null +++ b/Inc/ws2812b_fx.h @@ -0,0 +1,196 @@ +/* + * ws2812b_fx.h + * + * Library based on Harm Aldick's Arduino library + * https://github.com/kitesurfer1404/WS2812FX + * + * The MIT License. + * Created on: 20.10.2018 + * Author: Mateusz Salamon + * www.msalamon.pl + * mateusz@msalamon.pl + */ + +#ifndef WS2812B_FX_H_ +#define WS2812B_FX_H_ + +#define DEFAULT_COLOR 0x00FF000000 +#define NUM_COLORS 3 + +#define SPEED_MIN 10 +#define DEFAULT_SPEED 150 +#define SPEED_MAX 65535 + +#define BRIGHTNESS_MIN 0 +#define DEFAULT_BRIGHTNESS 50 +#define BRIGHTNESS_MAX 255 + +#define MODE_COUNT 62 +#define DEFAULT_MODE 0 + +#define FADE_RATE 2 + +// some common colors +#define RED (uint32_t)0xFF0000 +#define GREEN (uint32_t)0x00FF00 +#define BLUE (uint32_t)0x0000FF +#define WHITE (uint32_t)0xFFFFFF +#define BLACK (uint32_t)0x000000 +#define YELLOW (uint32_t)0xFFFF00 +#define CYAN (uint32_t)0x00FFFF +#define MAGENTA (uint32_t)0xFF00FF +#define PURPLE (uint32_t)0x400080 +#define ORANGE (uint32_t)0xFF3000 +#define PINK (uint32_t)0xFF1493 + +typedef enum { +FX_MODE_STATIC, +FX_MODE_WHITE_TO_COLOR, +FX_MODE_BLACK_TO_COLOR, +FX_MODE_BLINK, +FX_MODE_BLINK_RAINBOW, +FX_MODE_STROBE, +FX_MODE_STROBE_RAINBOW, +FX_MODE_BREATH, +FX_MODE_COLOR_WIPE, +FX_MODE_COLOR_WIPE_INV, +FX_MODE_COLOR_WIPE_REV, +FX_MODE_COLOR_WIPE_REV_INV, +FX_MODE_COLOR_WIPE_RANDOM, +FX_MODE_COLOR_SWEEP_RANDOM, +FX_MODE_RANDOM_COLOR, +FX_MODE_SINGLE_DYNAMIC, +FX_MODE_MULTI_DYNAMIC, +FX_MODE_RAINBOW, +FX_MODE_RAINBOW_CYCLE, +FX_MODE_FADE, +FX_MODE_SCAN, +FX_MODE_DUAL_SCAN, +FX_MODE_THEATER_CHASE, +FX_MODE_THEATER_CHASE_RAINBOW, +FX_MODE_RUNNING_LIGHTS, +FX_MODE_TWINKLE, +FX_MODE_TWINKLE_RANDOM, +FX_MODE_TWINKLE_FADE, +FX_MODE_TWINKLE_FADE_RANDOM, +FX_MODE_SPARKLE, +FX_MODE_FLASH_SPARKLE, +FX_MODE_HYPER_SPARKLE, +FX_MODE_MULTI_STROBE, +FX_MODE_CHASE_WHITE, +FX_MODE_CHASE_COLOR, +FX_MODE_CHASE_RANDOM, +FX_MODE_CHASE_RAINBOW, +FX_MODE_CHASE_FLASH, +FX_MODE_CHASE_FLASH_RANDOM, +FX_MODE_CHASE_RAINBOW_WHITE, +FX_MODE_CHASE_BLACKOUT, +FX_MODE_CHASE_BLACKOUT_RAINBOW, +FX_MODE_RUNNING_COLOR, +FX_MODE_RUNNING_RED_BLUE, +FX_MODE_RUNNING_RANDOM, +FX_MODE_LARSON_SCANNER, +FX_MODE_COMET, +FX_MODE_FIREWORKS, +FX_MODE_FIREWORKS_RANDOM, +FX_MODE_MERRY_CHRISTMAS, +FX_MODE_FIRE_FLICKER, +FX_MODE_FIRE_FLICKER_SOFT, +FX_MODE_FIRE_FLICKER_INTENSE, +FX_MODE_CIRCUS_COMBUSTUS, +FX_MODE_HALLOWEEN, +FX_MODE_BICOLOR_CHASE, +FX_MODE_TRICOLOR_CHASE, +FX_MODE_ICU, +} fx_mode; + +void WS2812BFX_SysTickCallback(void); +void WS2812BFX_Callback(void); +void WS2812BFX_Start(void); +void WS2812BFX_Stop(void); + +void WS2812BFX_SetMode(fx_mode mode); +void WS2812BFX_NextMode(void); +void WS2812BFX_PrevMode(void); + +void WS2812BFX_SetSpeed(uint16_t Speed); +void WS2812BFX_IncreaseSpeed(uint16_t Speed); +void WS2812BFX_DecreaseSpeed(uint16_t Speed); + +void WS2812BFX_SetColorStruct(uint8_t id, ws2812b_color c); +void WS2812BFX_SetColorRGB(uint8_t id, uint8_t r, uint8_t g, uint8_t b); +void WS2812BFX_SetColorHSV(uint8_t id, uint16_t h, uint8_t s, uint8_t v); +void WS2812BFX_SetColor(uint8_t id, uint32_t c); +void WS2812BFX_SetColorAll(uint32_t c); +void WS2812BFX_SetColorAllRGB(uint8_t r, uint8_t g, uint8_t b); + +void WS2812BFX_RGBtoHSV(uint8_t r, uint8_t g, uint8_t b, uint16_t *h, uint8_t *s, uint8_t *v); +void WS2812BFX_HSVtoRGB(uint16_t h, uint8_t s, uint8_t v, uint8_t *r, uint8_t *g, uint8_t *b); + +/* + * + * MODES + * + * */ +void +strip_off(void), +mode_static(void), +mode_white_to_color(void), +mode_black_to_color(void), +mode_blink(void), +mode_blink_rainbow(void), +mode_strobe(void), +mode_strobe_rainbow(void), +mode_breath(void), +mode_color_wipe(void), +mode_color_wipe_inv(void), +mode_color_wipe_rev(void), +mode_color_wipe_rev_inv(void), +mode_color_wipe_random(void), +mode_color_sweep_random(void), +mode_random_color(void), +mode_single_dynamic(void), +mode_multi_dynamic(void), +mode_rainbow(void), +mode_rainbow_cycle(void), +mode_fade(void), +mode_scan(void), +mode_dual_scan(void), +mode_theater_chase(void), +mode_theater_chase_rainbow(void), +mode_running_lights(void), +mode_twinkle(void), +mode_twinkle_random(void), +mode_twinkle_fade(void), +mode_twinkle_fade_random(void), +mode_sparkle(void), +mode_flash_sparkle(void), +mode_hyper_sparkle(void), +mode_multi_strobe(void), +mode_chase_white(void), +mode_chase_color(void), +mode_chase_random(void), +mode_chase_rainbow(void), +mode_chase_flash(void), +mode_chase_flash_random(void), +mode_chase_rainbow_white(void), +mode_chase_blackout(void), +mode_chase_blackout_rainbow(void), +mode_running_color(void), +mode_running_red_blue(void), +mode_running_random(void), +mode_larson_scanner(void), +mode_comet(void), +mode_fireworks(void), +mode_fireworks_random(void), +mode_merry_christmas(void), +mode_fire_flicker(void), +mode_fire_flicker_soft(void), +mode_fire_flicker_intense(void), +mode_circus_combustus(void), +mode_halloween(void), +mode_bicolor_chase(void), +mode_tricolor_chase(void), +mode_icu(void) +; +#endif /* WS2812B_FX_H_ */ diff --git a/README.md b/README.md index f9ec0f5..eca0885 100644 --- a/README.md +++ b/README.md @@ -1,5 +1,8 @@ WS2812B diodes using STM32F103C8T6 MCU. +www.msalamon.pl + Diodes are driving by SPI interface with DMA. -Color could be changed by RGB color model. \ No newline at end of file +The FX library is based od WS2812FX Arduino librarby by kitesurfer1404 - https://github.com/kitesurfer1404/WS2812FX + diff --git a/Src/main.c b/Src/main.c index 3f333ae..d89b5d7 100644 --- a/Src/main.c +++ b/Src/main.c @@ -45,16 +45,14 @@ /* USER CODE BEGIN Includes */ #include "ws2812b.h" +#include "ws2812b_fx.h" /* USER CODE END Includes */ /* Private variables ---------------------------------------------------------*/ /* USER CODE BEGIN PV */ /* Private variables ---------------------------------------------------------*/ -ws2812b_color color; -uint8_t R[] = {255,0,0,255,0,128,0,0,128,0,64,0,0,64,0,32,0,0,32,0,16,0,0,16,0,8,0,0,8,0}; -uint8_t G[] = {0,255,0,255,255,0,128,0,255,255,0,64,0,64,64,0,32,0,32,32,0,16,0,16,16,0,8,0,8,8}; -uint8_t B[] = {0,0,255,0,255,0,0,128,0,255,0,0,64,0,64,0,0,32,0,32,0,0,16,0,18,0,0,8,0,8}; + /* USER CODE END PV */ /* Private function prototypes -----------------------------------------------*/ @@ -101,18 +99,14 @@ int main(void) MX_DMA_Init(); MX_SPI1_Init(); /* USER CODE BEGIN 2 */ - HAL_GPIO_WritePin(LED_GPIO_Port, LED_Pin, 1); WS2812B_Init(&hspi1); - for(uint8_t i = 0; i <= WS2812B_LEDS; i++) - { - color.red = 0; - color.green = 0; - color.blue = 0; - WS2812B_SetDiodeColor(i,color); - } + WS2812BFX_SetSpeed(5000); + WS2812BFX_SetColorRGB(0, 0,255,0); + WS2812BFX_SetColorRGB(1, 32,0,0); + WS2812BFX_SetColorRGB(2, 0,64,0); + WS2812BFX_SetMode(FX_MODE_WHITE_TO_COLOR); + WS2812BFX_Start(); - WS2812B_Refresh(); - uint16_t j; /* USER CODE END 2 */ /* Infinite loop */ @@ -122,72 +116,8 @@ int main(void) /* USER CODE END WHILE */ /* USER CODE BEGIN 3 */ - - // HSV Value manipulation demo - for(int loop = 0; loop < 10; loop++) - { - for(int v =0; v<256; v++) - { - for(uint8_t i = 0; i <= WS2812B_LEDS; i++) - { - j = (360/35) * i; - WS2812B_SetDiodeHSV(i, j, 255, v); - } - HAL_Delay(1); - WS2812B_Refresh(); - } - - for(int v =255; v>-1; v--) - { - for(uint8_t i = 0; i <= WS2812B_LEDS; i++) - { - j = 360/35 * i; - WS2812B_SetDiodeHSV(i, j, 255, v); - } - HAL_Delay(1); - WS2812B_Refresh(); - } - } - - // RGB manipulation demo - for(int v =0; v<30; v++) - { - for(uint8_t i = 0; i <= WS2812B_LEDS; i++) - { - WS2812B_SetDiodeRGB(i, R[v], G[v], B[v]); - } - HAL_Delay(500); - WS2812B_Refresh(); - } - - // HSV Saturation manipulation demo - for(int loop = 0; loop < 10; loop++) - { - for(int v =0; v<256; v++) - { - for(uint8_t i = 0; i <= WS2812B_LEDS; i++) - { - j = (360/35) * i; - WS2812B_SetDiodeHSV(i, j, v, 255); - } - HAL_Delay(1); - WS2812B_Refresh(); - } - - for(int v =255; v>-1; v--) - { - for(uint8_t i = 0; i <= WS2812B_LEDS; i++) - { - j = 360/35 * i; - WS2812B_SetDiodeHSV(i, j, v, 255); - } - HAL_Delay(1); - WS2812B_Refresh(); - } - } - - HAL_Delay(10); - } + WS2812BFX_Callback(); + } /* USER CODE END 3 */ @@ -244,7 +174,10 @@ void SystemClock_Config(void) } /* USER CODE BEGIN 4 */ - +void HAL_SYSTICK_Callback(void) +{ + WS2812BFX_SysTickCallback(); +} /* USER CODE END 4 */ /** diff --git a/Src/ws2812b.c b/Src/ws2812b.c index 020583e..11d810f 100644 --- a/Src/ws2812b.c +++ b/Src/ws2812b.c @@ -31,7 +31,15 @@ void WS2812B_Init(SPI_HandleTypeDef * spi_handler) hspi_ws2812b = spi_handler; } -void WS2812B_SetDiodeColor(int16_t diode_id, ws2812b_color color) +void WS2812B_SetDiodeColor(int16_t diode_id, uint32_t color) +{ + if(diode_id >= WS2812B_LEDS || diode_id < 0) return; + ws2812b_array[diode_id].red = ((color>>16)&0x000000FF); + ws2812b_array[diode_id].green = ((color>>8)&0x000000FF); + ws2812b_array[diode_id].blue = (color&0x000000FF); +} + +void WS2812B_SetDiodeColorStruct(int16_t diode_id, ws2812b_color color) { if(diode_id >= WS2812B_LEDS || diode_id < 0) return; ws2812b_array[diode_id] = color; @@ -45,6 +53,19 @@ void WS2812B_SetDiodeRGB(int16_t diode_id, uint8_t R, uint8_t G, uint8_t B) ws2812b_array[diode_id].blue = B; } +uint32_t WS2812B_GetColor(int16_t diode_id) +{ + uint32_t color = 0; + color |= ((ws2812b_array[diode_id].red&0xFF)<<16); + color |= ((ws2812b_array[diode_id].green&0xFF)<<8); + color |= (ws2812b_array[diode_id].blue&0xFF); + return color; +} + +uint8_t* WS2812B_GetPixels(void) +{ + return (uint8_t*)ws2812b_array; +} // // Set diode with HSV model // diff --git a/Src/ws2812b_fx.c b/Src/ws2812b_fx.c new file mode 100644 index 0000000..0071ea0 --- /dev/null +++ b/Src/ws2812b_fx.c @@ -0,0 +1,1673 @@ +/* + * ws2812b_fx.c + * + * Library based on Harm Aldick's Arduino library + * https://github.com/kitesurfer1404/WS2812FX + * + * The MIT License. + * Created on: 20.10.2018 + * Author: Mateusz Salamon + * www.msalamon.pl + * mateusz@msalamon.pl + */ +#include "stm32f1xx_hal.h" +#include + +#include "ws2812b.h" +#include "ws2812b_fx.h" + +#define MIN(a,b) (((a)<(b))?(a):(b)) +#define MAX(a,b) (((a)<(b))?(b):(a)) + +uint8_t mMode_index = DEFAULT_MODE; +uint16_t mSpeed = DEFAULT_SPEED; +uint8_t mBrightness = DEFAULT_BRIGHTNESS; +uint8_t mRunning; +uint8_t mTriggered; + +volatile uint32_t iModeDelay; +uint8_t mActualMode; +uint32_t mCounterModeCall; +uint32_t mCounterModeStep; +unsigned long mModeLastCallTime; + +uint32_t mColor[NUM_COLORS]; +uint32_t mModeColor[NUM_COLORS]; +ws2812b_color mColor_w[NUM_COLORS]; +ws2812b_color mModeColor_w[NUM_COLORS]; + +uint8_t mAuxParam; +uint16_t mAuxParam16b; +uint8_t mCycle; + +void (*mModeCallback)(void); + +void (*mMode[MODE_COUNT])(void) = +{ + mode_static, + mode_white_to_color, + mode_black_to_color, + mode_blink, + mode_blink_rainbow, + mode_strobe, + mode_strobe_rainbow, + mode_breath, + mode_color_wipe, + mode_color_wipe_inv, + mode_color_wipe_rev, + mode_color_wipe_rev_inv, + mode_color_wipe_random, + mode_color_sweep_random, + mode_random_color, + mode_single_dynamic, + mode_multi_dynamic, + mode_rainbow, + mode_rainbow_cycle, + mode_fade, + mode_scan, + mode_dual_scan, + mode_theater_chase, + mode_theater_chase_rainbow, + mode_running_lights, + mode_twinkle, + mode_twinkle_random, + mode_twinkle_fade, + mode_twinkle_fade_random, + mode_sparkle, + mode_flash_sparkle, + mode_hyper_sparkle, + mode_multi_strobe, + mode_chase_white, + mode_chase_color, + mode_chase_random, + mode_chase_rainbow, + mode_chase_flash, + mode_chase_flash_random, + mode_chase_rainbow_white, + mode_chase_blackout, + mode_chase_blackout_rainbow, + mode_running_color, + mode_running_red_blue, + mode_running_random, + mode_larson_scanner, + mode_comet, + mode_fireworks, + mode_fireworks_random, + mode_merry_christmas, + mode_fire_flicker, + mode_fire_flicker_soft, + mode_fire_flicker_intense, + mode_circus_combustus, + mode_halloween, + mode_bicolor_chase, + mode_tricolor_chase, + mode_icu +}; + +void WS2812BFX_SysTickCallback(void) +{ + if(iModeDelay > 0) iModeDelay--; +} + +void WS2812BFX_Callback(void) + { + if(mRunning || mTriggered) + { + if(!iModeDelay) + { + mModeCallback(); + WS2812B_Refresh(); + mCounterModeCall++; + } + } +} + +void WS2812BFX_SetMode(fx_mode mode) +{ + mCounterModeCall = 0; + mCounterModeStep = 0; + mModeLastCallTime = 0; + mActualMode = mode; + mCycle = 0; + mModeCallback = mMode[mActualMode]; + for(uint8_t i = 0; i < NUM_COLORS; i++) + { + mModeColor[i] = mColor[i]; + mModeColor_w[i].red = mColor_w[i].red; + mModeColor_w[i].green = mColor_w[i].green; + mModeColor_w[i].blue = mColor_w[i].blue; + } +} + +void WS2812BFX_NextMode(void) +{ + mCounterModeCall = 0; + mCounterModeStep = 0; + mModeLastCallTime = 0; + mActualMode++; + mCycle = 0; + if(mActualMode >= MODE_COUNT) mActualMode = 0; + mModeCallback = mMode[mActualMode]; + for(uint8_t i = 0; i < NUM_COLORS; i++) + { + mModeColor[i] = mColor[i]; + mModeColor_w[i].red = mColor_w[i].red; + mModeColor_w[i].green = mColor_w[i].green; + mModeColor_w[i].blue = mColor_w[i].blue; + } +} + +void WS2812BFX_PrevMode(void) +{ + mCounterModeCall = 0; + mCounterModeStep = 0; + mModeLastCallTime = 0; + mCycle = 0; + if(mActualMode == 0) mActualMode = MODE_COUNT; + else mActualMode--; + mModeCallback = mMode[mActualMode]; + for(uint8_t i = 0; i < NUM_COLORS; i++) + { + mModeColor[i] = mColor[i]; + mModeColor_w[i].red = mColor_w[i].red; + mModeColor_w[i].green = mColor_w[i].green; + mModeColor_w[i].blue = mColor_w[i].blue; + } +} + +void WS2812BFX_Start() +{ + mCounterModeCall = 0; + mCounterModeStep = 0; + mModeLastCallTime = 0; + iModeDelay = 0; + mRunning = 1; +} + +void WS2812BFX_Stop() +{ + mRunning = 0; +// strip_off(); +} + +void WS2812BFX_SetColorStruct(uint8_t id, ws2812b_color c) +{ + mColor[id] = ((c.red<<16)|(c.green<<8)|c.blue); + mColor_w[id].red = c.red; + mColor_w[id].green = c.green; + mColor_w[id].blue = c.blue; +} + +void WS2812BFX_SetColorRGB(uint8_t id, uint8_t r, uint8_t g, uint8_t b) +{ + mColor[id] = ((r<<16)|(g<<8)|b); + mColor_w[id].red = r; + mColor_w[id].green = g; + mColor_w[id].blue = b; +} + +// +// Set color with HSV model +// +// Hue 0-359 +// Saturation 0-255 +// Value 0-255 +// +void WS2812BFX_SetColorHSV(uint8_t id, uint16_t h, uint8_t s, uint8_t v) +{ + + WS2812BFX_HSVtoRGB(h, s, v, &mColor_w[id].red, &mColor_w[id].green, &mColor_w[id].blue); + mColor[id] = ((mColor_w[id].red<<16)|(mColor_w[id].green<<8)|mColor_w[id]. blue); +} + +void WS2812BFX_SetColor(uint8_t id, uint32_t c) +{ + mColor[id] = c; + mColor_w[id].red = ((c>>16)&0x000000FF); + mColor_w[id].green = ((c>>8)&0x000000FF); + mColor_w[id].blue = (c&0x000000FF); +} + +void WS2812BFX_SetColorAll(uint32_t c) +{ + mColor[0] = c; + mColor_w[0].red = ((c>>16)&0x000000FF); + mColor_w[0].green = ((c>>8)&0x000000FF); + mColor_w[0].blue = (c&0x000000FF); + for(uint16_t i = 0; i < WS2812B_LEDS; i++) + { + WS2812B_SetDiodeRGB(i, mColor_w[0].red, mColor_w[0].green, mColor_w[0].blue); + } +} + +void WS2812BFX_SetColorAllRGB(uint8_t r, uint8_t g, uint8_t b) +{ + mColor[0] = ((r<<16)|(g<<8)|b); + mColor_w[0].red = r; + mColor_w[0].green = g; + mColor_w[0].blue = b; + for(uint16_t i = 0; i < WS2812B_LEDS; i++) + { + WS2812B_SetDiodeRGB(i, r, g, b); + } +} + +void WS2812BFX_RGBtoHSV(uint8_t r, uint8_t g, uint8_t b, uint16_t *h, uint8_t *s, uint8_t *v) +{ + uint16_t min, max, delta; + int16_t h_tmp = *h; + + min = r < g ? r : g; + min = min < b ? min : b; + + max = r > g ? r : g; + max = max > b ? max : b; + + *v = max; // v + delta = max - min; + if (delta < 1) + { + *s = 0; + *h = 0; // undefined, maybe nan? + return; + } + if( max > 0 ) + { // NOTE: if Max is == 0, this divide would cause a crash + *s = (((delta * 100) / max) * 255) / 100; // s + } + else + { + // if max is 0, then r = g = b = 0 + // s = 0, h is undefined + *s = 0; + *h = 0; // its now undefined + return; + } + + if( r == max ) + { + h_tmp = (( g - b )*100) / delta; // between yellow & magenta + // *100 to avoid fracts + } + else + { + if( g == max ) + { + h_tmp = 720 + (( b - r )*100) / delta; // between cyan & yellow + } + else + { + h_tmp = 1440 + (( r - g )*100) / delta; // between magenta & cyan + } + } + h_tmp *= 60; // Degrees + h_tmp /= 100; // Back from fracts + + if( h_tmp < 0.0 ) + h_tmp += 360; + + *h = h_tmp; +} + +void WS2812BFX_HSVtoRGB(uint16_t h, uint8_t s, uint8_t v, uint8_t *r, uint8_t *g, uint8_t *b) +{ + uint16_t Sector, Fracts, p, q, t; + + if(s == 0) + { + *r = v; + *g = v; + *b = v; + } + else + { + if(h >= 360) h = 359; + + Sector = h / 60; // Sector 0 to 5 + Fracts = h % 60; + p = (v * (255 - s)) / 256; + q = (v * (255 - (s * Fracts)/360)) / 256; + t = (v * (255 - (s * (360 - Fracts))/360)) / 256; + + + switch(Sector) + { + case 0: + *r = v; + *g = (uint8_t)t; + *b = (uint8_t)p; + break; + case 1: + *r = (uint8_t)q; + *g = v; + *b = (uint8_t)p; + break; + case 2: + *r = (uint8_t)p; + *g = v; + *b = (uint8_t)t; + break; + case 3: + *r = (uint8_t)p; + *g = (uint8_t)q; + *b = v; + break; + case 4: + *r = (uint8_t)t; + *g = (uint8_t)p; + *b = v; + break; + default: // case 5: + *r = v; + *g = (uint8_t)p; + *b = (uint8_t)q; + break; + } + } +} + +uint8_t WS2812BFX_IsRunning() +{ + return mRunning; +} + +uint8_t WS2812BFX_GetMode(void) +{ + return mMode_index; +} + +uint8_t WS2812BFX_GetSpeed(void) +{ + return mSpeed; +} + +uint8_t WS2812BFX_GetBrightness(void) +{ + return mBrightness; +} + +uint16_t WS2812BFX_GetLength(void) +{ + return WS2812B_LEDS; +} + +uint8_t WS2812BFX_GetModeCount(void) +{ + return MODE_COUNT; +} + +ws2812b_color WS2812BFX_GetColorStruct(uint8_t id) +{ + return mColor_w[id]; +} + +uint32_t WS2812BFX_GetColor(uint8_t id) +{ + return mColor[id]; +} + +void WS2812BFX_SetSpeed(uint16_t Speed) +{ + if(Speed < SPEED_MIN) Speed = SPEED_MIN; + if(Speed > SPEED_MAX) Speed = SPEED_MAX; + + mSpeed = Speed; +} + +void WS2812BFX_IncreaseSpeed(uint16_t Speed) +{ + WS2812BFX_SetSpeed(mSpeed + Speed); +} + +void WS2812BFX_DecreaseSpeed(uint16_t Speed) +{ + WS2812BFX_SetSpeed(mSpeed - Speed); +} + +// +// +// COLOR EFECTS FUNCTIONS +// +// + +/* + * Turns everything off. + */ +void strip_off() +{ + for(uint16_t i = 0; i < WS2812B_LEDS; i++) + { + WS2812B_SetDiodeRGB(i, 0, 0, 0); + } + WS2812B_Refresh(); +} + +/* + * Put a value 0 to 255 in to get a color value. + * The colours are a transition r -> g -> b -> back to r + * Inspired by the Adafruit examples. + */ +uint32_t color_wheel(uint8_t pos) { + pos = 255 - pos; + if(pos < 85) { + return ((uint32_t)(255 - pos * 3) << 16) | ((uint32_t)(0) << 8) | (pos * 3); + } else if(pos < 170) { + pos -= 85; + return ((uint32_t)(0) << 16) | ((uint32_t)(pos * 3) << 8) | (255 - pos * 3); + } else { + pos -= 170; + return ((uint32_t)(pos * 3) << 16) | ((uint32_t)(255 - pos * 3) << 8) | (0); + } +} + +/* + * Returns a new, random wheel index with a minimum distance of 42 from pos. + */ +uint8_t get_random_wheel_index(uint8_t pos) { + uint8_t r = 0; + uint8_t x = 0; + uint8_t y = 0; + uint8_t d = 0; + + while(d < 42) { + r = rand()%256; + x = abs(pos - r); + y = 255 - x; + d = MIN(x, y); + } + + return r; +} + +/* + * fade out function + */ +void fade_out() { + static const uint8_t rateMapH[] = {0, 1, 1, 1, 2, 3, 4, 6}; + static const uint8_t rateMapL[] = {0, 2, 3, 8, 8, 8, 8, 8}; + + uint8_t rate = FADE_RATE; + uint8_t rateH = rateMapH[rate]; + uint8_t rateL = rateMapL[rate]; + + uint32_t color = mModeColor[1]; // target color + int r2 = (color >> 16) & 0xff; + int g2 = (color >> 8) & 0xff; + int b2 = color & 0xff; + + for(uint16_t i=0; i <= WS2812B_LEDS; i++) { + color = WS2812B_GetColor(i); + if(rate == 0) { // old fade-to-black algorithm + WS2812B_SetDiodeColor(i, (color >> 1) & 0x7F7F7F7F); + } else { // new fade-to-color algorithm + int r1 = (color >> 16) & 0xff; + int g1 = (color >> 8) & 0xff; + int b1 = color & 0xff; + + // calculate the color differences between the current and target colors + int rdelta = r2 - r1; + int gdelta = g2 - g1; + int bdelta = b2 - b1; + + // if the current and target colors are almost the same, jump right to the target color, + // otherwise calculate an intermediate color. (fixes rounding issues) + rdelta = abs(rdelta) < 3 ? rdelta : (rdelta >> rateH) + (rdelta >> rateL); + gdelta = abs(gdelta) < 3 ? gdelta : (gdelta >> rateH) + (gdelta >> rateL); + bdelta = abs(bdelta) < 3 ? bdelta : (bdelta >> rateH) + (bdelta >> rateL); + + WS2812B_SetDiodeRGB(i, r1 + rdelta, g1 + gdelta, b1 + bdelta); + } + } +} + +/* + * color blend function + */ +uint32_t color_blend(uint32_t color1, uint32_t color2, uint8_t blend) +{ + if(blend == 0) return color1; + if(blend == 255) return color2; + + int w1 = (color1 >> 24) & 0xff; + int r1 = (color1 >> 16) & 0xff; + int g1 = (color1 >> 8) & 0xff; + int b1 = color1 & 0xff; + + int w2 = (color2 >> 24) & 0xff; + int r2 = (color2 >> 16) & 0xff; + int g2 = (color2 >> 8) & 0xff; + int b2 = color2 & 0xff; + + uint32_t w3 = ((w2 * blend) + (w1 * (255 - blend))) / 256; + uint32_t r3 = ((r2 * blend) + (r1 * (255 - blend))) / 256; + uint32_t g3 = ((g2 * blend) + (g1 * (255 - blend))) / 256; + uint32_t b3 = ((b2 * blend) + (b1 * (255 - blend))) / 256; + + return ((w3 << 24) | (r3 << 16) | (g3 << 8) | (b3)); +} + +/* + * No blinking. Just plain old static light. + */ +void mode_static(void) { + for(uint16_t i=0; i < WS2812B_LEDS; i++) { + WS2812B_SetDiodeColorStruct(i, mModeColor_w[0]); + } + iModeDelay = 250; +} + +// +// from: 0 - black to color, 1 - white to color +// +void to_color(uint8_t from) +{ + // HSV Saturatioin modifing + uint16_t h; + uint8_t s, v, r, g, b; + + WS2812BFX_RGBtoHSV(mModeColor_w[0].red, mModeColor_w[0].green, mModeColor_w[0].blue, &h, &s, &v); + + if(from) + WS2812BFX_HSVtoRGB(h, s - mCounterModeStep, v, &r, &g, &b); + else + WS2812BFX_HSVtoRGB(h, s, v - mCounterModeStep, &r, &g, &b); + + for(uint16_t i = 0; i < WS2812B_LEDS; i++) + { + WS2812B_SetDiodeRGB(i, r, g, b); + } + + if(!mCycle) + { + if(from) + { + if(mCounterModeStep < s) + mCounterModeStep++; + else + mCycle = 1; + } + else + { + if(mCounterModeStep < v) + mCounterModeStep++; + else + mCycle = 1; + } + } + else + { + if(mCounterModeStep > 0) + mCounterModeStep--; + else + mCycle = 0; + } + + if(from) + { + iModeDelay = mSpeed / s / 2; + } + else + { + iModeDelay = mSpeed / v / 2; + } +} + + +void mode_white_to_color(void) +{ + to_color(1); +} + +void mode_black_to_color(void) +{ + to_color(0); +} + +// +// Blink helper function +// +void blink(uint32_t color1, uint32_t color2, uint8_t strobe) +{ + uint32_t color = ((mCounterModeCall & 1) == 0) ? color1 : color2; + WS2812BFX_SetColorAll(color); + if((mCounterModeCall & 1) == 0) + iModeDelay = strobe ? 20 : mSpeed / 2; + else + iModeDelay = strobe? mSpeed - 20 : (mSpeed / 2); +} + +/* + * Normal blinking. 50% on/off time. + */ +void mode_blink(void) { + blink(mModeColor[0], mModeColor[1], 0); +} + +void mode_blink_rainbow(void) +{ + blink(color_wheel(mCounterModeCall & 0xFF), mModeColor[1], 0); +} + +void mode_strobe(void) { + blink(mModeColor[0], mModeColor[1], 1); +} + +void mode_strobe_rainbow(void) +{ + blink(color_wheel(mCounterModeCall & 0xFF), mModeColor[1], 1); +} + +/* + * Breathing effect + */ +void mode_breath(void) +{ + uint32_t lum = mCounterModeStep; + if(lum > 255) lum = 511 - lum; + + uint16_t delay; + if(lum == 15) delay = 970; // 970 pause before each breath + else if(lum <= 25) delay = 38; // 19 + else if(lum <= 50) delay = 36; // 18 + else if(lum <= 75) delay = 28; // 14 + else if(lum <= 100) delay = 20; // 10 + else if(lum <= 125) delay = 14; // 7 + else if(lum <= 150) delay = 11; // 5 + else delay = 10; // 4 + + uint8_t r = mModeColor_w[0].red * lum / 256; + uint8_t g = mModeColor_w[0].green * lum / 256; + uint8_t b = mModeColor_w[0].blue * lum / 256; + + WS2812BFX_SetColorAllRGB(r, g, b); + mCounterModeStep += 2; + if(mCounterModeStep > (512-15)) mCounterModeStep = 15; + iModeDelay = delay; +} + +/* + * 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 + */ +void color_wipe(uint32_t color1, uint32_t color2, uint8_t rev) +{ + if(mCounterModeStep < WS2812B_LEDS) + { + uint32_t led_offset = mCounterModeStep; + if(rev) + { + WS2812B_SetDiodeColor(WS2812B_LEDS - led_offset, color1); + } + else + { + WS2812B_SetDiodeColor(0 + led_offset, color1); + } + } + else + { + uint32_t led_offset = mCounterModeStep - WS2812B_LEDS; + if(rev) + { + WS2812B_SetDiodeColor(WS2812B_LEDS - led_offset, color2); + } + else + { + WS2812B_SetDiodeColor(0 + led_offset, color2); + } + + } + mCounterModeStep = (mCounterModeStep + 1) % (WS2812B_LEDS * 2); + iModeDelay = mSpeed; +} + +/* + * Lights all LEDs one after another. + */ +void mode_color_wipe(void) +{ + color_wipe(mModeColor[0], mModeColor[1], 0); +} + +void mode_color_wipe_inv(void) +{ + color_wipe(mModeColor[1], mModeColor[0], 0); +} + +void mode_color_wipe_rev(void) +{ + color_wipe(mModeColor[0], mModeColor[1], 1); +} + +void mode_color_wipe_rev_inv(void) +{ + color_wipe(mModeColor[1], mModeColor[0], 1); +} + +void mode_color_wipe_random(void) + { + if(mCounterModeStep % WS2812B_LEDS == 0) + { + mAuxParam = get_random_wheel_index(mAuxParam); + } + uint32_t color = color_wheel(mAuxParam); + + color_wipe(color, color, 0); + iModeDelay = mSpeed; +} + +/* + * Random color introduced alternating from start and end of strip. + */ +void mode_color_sweep_random(void) +{ + if(mCounterModeStep % WS2812B_LEDS == 0) + { // aux_param will store our random color wheel index + mAuxParam = get_random_wheel_index(mAuxParam); + } + uint32_t color = color_wheel(mAuxParam); + color_wipe(color, color, 1); +} + +/* + * Lights all LEDs in one random color up. Then switches them + * to the next random color. + */ +void mode_random_color(void) +{ + mAuxParam = get_random_wheel_index(mAuxParam); // aux_param will store our random color wheel index + WS2812BFX_SetColorAll(color_wheel(mAuxParam)); + iModeDelay = mSpeed; +} + +/* + * Lights every LED in a random color. Changes one random LED after the other + * to another random color. + */ +void mode_single_dynamic(void) +{ + if(mCounterModeCall == 0) + { + for(uint16_t i = 0; i < WS2812B_LEDS; i++) + { + WS2812B_SetDiodeColor(i, color_wheel(rand()%256)); + } + } + + WS2812B_SetDiodeColor(0 + rand() % WS2812B_LEDS, color_wheel(rand()%256)); + iModeDelay = mSpeed; +} + + +/* + * Lights every LED in a random color. Changes all LED at the same time + * to new random colors. + */ +void mode_multi_dynamic(void) +{ + for(uint16_t i = 0; i < WS2812B_LEDS; i++) + { + WS2812B_SetDiodeColor(i, color_wheel(rand()%256)); + } + iModeDelay = mSpeed; +} + +/* + * Cycles all LEDs at once through a rainbow. + */ +void mode_rainbow(void) +{ + uint32_t color = color_wheel(mCounterModeStep); + for(uint16_t i = 0; i < WS2812B_LEDS; i++) + { + WS2812B_SetDiodeColor(i, color); + } + + mCounterModeStep = (mCounterModeStep + 1) & 0xFF; + iModeDelay = mSpeed; +} + + +/* + * Cycles a rainbow over the entire string of LEDs. + */ +void mode_rainbow_cycle(void) +{ + for(uint16_t i=0; i < WS2812B_LEDS; i++) + { + uint32_t color = color_wheel(((i * 256 / WS2812B_LEDS) + mCounterModeStep) & 0xFF); + WS2812B_SetDiodeColor(0 + i, color); + } + + mCounterModeStep = (mCounterModeStep + 1) & 0xFF; + iModeDelay = mSpeed; +} + + +/* + * Fades the LEDs between two colors + */ +void mode_fade(void) { + int lum = mCounterModeStep; + if(lum > 255) lum = 511 - lum; // lum = 0 -> 255 -> 0 + + uint32_t color = color_blend(mModeColor[0], mModeColor[1], lum); + + for(uint16_t i=0; i 511) mCounterModeStep = 0; + iModeDelay = mSpeed; +} + + +/* + * Runs a single pixel back and forth. + */ +void mode_scan(void) { + if(mCounterModeStep > (WS2812B_LEDS * 2) - 3) { + mCounterModeStep = 0; + } + + for(uint16_t i=0; i (WS2812B_LEDS * 2) - 3) { + mCounterModeStep = 0; + } + + for(uint16_t i=0; i > 16) & 0xFF); + uint8_t g = ((mModeColor[0] >> 8) & 0xFF); + uint8_t b = (mModeColor[0] & 0xFF); + + uint8_t sineIncr = MAX(1, (256 / WS2812B_LEDS)); + for(uint16_t i=0; i < WS2812B_LEDS; i++) { + int lum = (int)sine8(((i + mCounterModeStep) * sineIncr)); +// if(IS_REVERSE) { +// WS2812B_SetDiodeColor(0 + i, (r * lum) / 256, (g * lum) / 256, (b * lum) / 256, (w * lum) / 256); +// } else { + WS2812B_SetDiodeRGB(WS2812B_LEDS - i, (r * lum) / 256, (g * lum) / 256, (b * lum) / 256); +// } + } + mCounterModeStep = (mCounterModeStep + 1) % 256; + iModeDelay = mSpeed; +} + + +/* + * twinkle function + */ +void twinkle(uint32_t color1, uint32_t color2) +{ + if(mCounterModeStep == 0) + { + for(uint16_t i=0; i <= WS2812B_LEDS; i++) + { + WS2812B_SetDiodeColor(i, color2); + } + uint16_t min_leds = MAX(1, WS2812B_LEDS / 5); // make sure, at least one LED is on + uint16_t max_leds = MAX(1, WS2812B_LEDS / 2); // make sure, at least one LED is on + mCounterModeStep = rand() % (max_leds + 1 - min_leds) + min_leds; + } + + WS2812B_SetDiodeColor(0 + rand() % WS2812B_LEDS, color1); + + mCounterModeStep--; + iModeDelay = mSpeed; +} + +/* + * Blink several LEDs on, reset, repeat. + * Inspired by www.tweaking4all.com/hardware/arduino/arduino-led-strip-effects/ + */ +void mode_twinkle(void) { + return twinkle(mModeColor[0], mModeColor[1]); +} + +/* + * Blink several LEDs in random colors on, reset, repeat. + * Inspired by www.tweaking4all.com/hardware/arduino/arduino-led-strip-effects/ + */ +void mode_twinkle_random(void) { + return twinkle(color_wheel(rand() % 256), mModeColor[1]); +} + +/* + * twinkle_fade function + */ +void twinkle_fade(uint32_t color) +{ + fade_out(); + + if((rand() %3) == 0) + { + WS2812B_SetDiodeColor(0 + rand() % WS2812B_LEDS, color); + } + iModeDelay = mSpeed; +} + + +/* + * Blink several LEDs on, fading out. + */ +void mode_twinkle_fade(void) +{ + twinkle_fade(mModeColor[0]); +} + + +/* + * Blink several LEDs in random colors on, fading out. + */ +void mode_twinkle_fade_random(void) +{ + twinkle_fade(color_wheel(rand() % 256)); +} + +/* + * Blinks one LED at a time. + * Inspired by www.tweaking4all.com/hardware/arduino/arduino-led-strip-effects/ + */ +void mode_sparkle(void) +{ + WS2812B_SetDiodeColor(0 + mAuxParam16b, mModeColor[1]); + mAuxParam16b = rand() % WS2812B_LEDS; // aux_param3 stores the random led index + WS2812B_SetDiodeColor(0 + mAuxParam16b, mModeColor[0]); + iModeDelay = mSpeed; +} + + +/* + * Lights all LEDs in the color. Flashes single white pixels randomly. + * Inspired by www.tweaking4all.com/hardware/arduino/arduino-led-strip-effects/ + */ +void mode_flash_sparkle(void) { + if(mCounterModeCall == 0) { + for(uint16_t i=0; i <= WS2812B_LEDS; i++) + { + WS2812B_SetDiodeColor(i, mModeColor[0]); + } + } + + WS2812B_SetDiodeColor(0 + mAuxParam16b, mModeColor[0]); + + if(rand() % 5 == 0) + { + mAuxParam16b = rand() % WS2812B_LEDS; // aux_param3 stores the random led index + WS2812B_SetDiodeColor(0 + mAuxParam16b, WHITE); + iModeDelay = 20; + } + iModeDelay = mSpeed; +} + + +/* + * Like flash sparkle. With more flash. + * Inspired by www.tweaking4all.com/hardware/arduino/arduino-led-strip-effects/ + */ +void mode_hyper_sparkle(void) +{ + for(uint16_t i=0; i <= WS2812B_LEDS; i++) + { + WS2812B_SetDiodeColor(i, mModeColor[0]); + } + + if(rand() % 5 < 2) + { + for(uint16_t i=0; i < MAX(1, WS2812B_LEDS/3); i++) + { + WS2812B_SetDiodeColor(0 + rand() % WS2812B_LEDS, WHITE); + } + iModeDelay = 20; + } + iModeDelay = mSpeed; +} + +/* + * Strobe effect with different strobe count and pause, controlled by speed. + */ +void mode_multi_strobe(void) +{ + for(uint16_t i=0; i <= WS2812B_LEDS; i++) + { + WS2812B_SetDiodeColor(i, BLACK); + } + + uint16_t delay = 200 + ((9 - (mSpeed % 10)) * 100); + uint16_t count = 2 * ((mSpeed / 100) + 1); + if(mCounterModeStep < count) + { + if((mCounterModeStep & 1) == 0) + { + for(uint16_t i=0; i <= WS2812B_LEDS; i++) + { + WS2812B_SetDiodeColor(i, mModeColor[0]); + } + delay = 20; + } + else + { + delay = 50; + } + } + mCounterModeStep = (mCounterModeStep + 1) % (count + 1); + iModeDelay = delay; +} + +/* + * color chase function. + * color1 = background color + * color2 and color3 = colors of two adjacent leds + */ +void chase(uint32_t color1, uint32_t color2, uint32_t color3) +{ + uint16_t a = mCounterModeStep; + uint16_t b = (a + 1) % WS2812B_LEDS; + uint16_t c = (b + 1) % WS2812B_LEDS; +// if(IS_REVERSE) { +// WS2812B_SetDiodeColor(WS2812B_LEDS - a, color1); +// WS2812B_SetDiodeColor(WS2812B_LEDS - b, color2); +// WS2812B_SetDiodeColor(WS2812B_LEDS - c, color3); +// } else { + WS2812B_SetDiodeColor(0 + a, color1); + WS2812B_SetDiodeColor(0 + b, color2); + WS2812B_SetDiodeColor(0 + c, color3); +// } + + if(b == 0) mCycle = 1; + else mCycle = 0; + + mCounterModeStep = (mCounterModeStep + 1) % WS2812B_LEDS; + iModeDelay = mSpeed; +} + + +/* + * Bicolor chase mode + */ +void mode_bicolor_chase(void) +{ + return chase(mModeColor[0], mModeColor[1], mModeColor[2]); +} + + +/* + * White running on _color. + */ +void mode_chase_color(void) +{ + return chase(mModeColor[0], WHITE, WHITE); +} + + +/* + * Black running on _color. + */ +void mode_chase_blackout(void) +{ + return chase(mModeColor[0], BLACK, BLACK); +} + + +/* + * _color running on white. + */ +void mode_chase_white(void) +{ + return chase(WHITE, mModeColor[0], mModeColor[0]); +} + + +/* + * White running followed by random color. + */ +void mode_chase_random(void) +{ + if(mCounterModeStep == 0) + { + mAuxParam = get_random_wheel_index(mAuxParam); + } + return chase(color_wheel(mAuxParam), WHITE, WHITE); +} + + +/* + * Rainbow running on white. + */ +void mode_chase_rainbow_white(void) +{ + uint16_t n = mCounterModeStep; + uint16_t m = (mCounterModeStep + 1) % WS2812B_LEDS; + uint32_t color2 = color_wheel(((n * 256 / WS2812B_LEDS) + (mCounterModeCall & 0xFF)) & 0xFF); + uint32_t color3 = color_wheel(((m * 256 / WS2812B_LEDS) + (mCounterModeCall & 0xFF)) & 0xFF); + + return chase(WHITE, color2, color3); +} + + +/* + * White running on rainbow. + */ +void mode_chase_rainbow(void) +{ + uint8_t color_sep = 256 / WS2812B_LEDS; + uint8_t color_index = mCounterModeCall & 0xFF; + uint32_t color = color_wheel(((mCounterModeStep * color_sep) + color_index) & 0xFF); + + return chase(color, WHITE, WHITE); +} + + +/* + * Black running on rainbow. + */ +void mode_chase_blackout_rainbow(void) +{ + uint8_t color_sep = 256 / WS2812B_LEDS; + uint8_t color_index = mCounterModeCall & 0xFF; + uint32_t color = color_wheel(((mCounterModeStep * color_sep) + color_index) & 0xFF); + + return chase(color, 0, 0); +} + +/* + * White flashes running on _color. + */ +void mode_chase_flash(void) +{ + const static uint8_t flash_count = 4; + uint8_t flash_step = mCounterModeCall % ((flash_count * 2) + 1); + + for(uint16_t i=0; i <= WS2812B_LEDS; i++) + { + WS2812B_SetDiodeColor(i, mModeColor[0]); + } + + uint16_t delay = mSpeed; + if(flash_step < (flash_count * 2)) + { + if(flash_step % 2 == 0) + { + uint16_t n = mCounterModeStep; + uint16_t m = (mCounterModeStep + 1) % WS2812B_LEDS; +// if(IS_REVERSE) +// { +// WS2812B_SetDiodeColor(WS2812B_LEDS - n, WHITE); +// WS2812B_SetDiodeColor(WS2812B_LEDS - m, WHITE); +// } +// else +// { + WS2812B_SetDiodeColor(0 + n, WHITE); + WS2812B_SetDiodeColor(0 + m, WHITE); +// } + delay = 20; + } + else + { + delay = 30; + } + } + else + { + mCounterModeStep = (mCounterModeStep + 1) % WS2812B_LEDS; + } + iModeDelay = delay; +} + + +/* + * White flashes running, followed by random color. + */ +void mode_chase_flash_random(void) +{ + const static uint8_t flash_count = 4; + uint8_t flash_step = mCounterModeCall % ((flash_count * 2) + 1); + + for(uint16_t i=0; i < mCounterModeStep; i++) + { + WS2812B_SetDiodeColor(0 + i, color_wheel(mAuxParam)); + } + + uint16_t delay = mSpeed; + if(flash_step < (flash_count * 2)) + { + uint16_t n = mCounterModeStep; + uint16_t m = (mCounterModeStep + 1) % WS2812B_LEDS; + if(flash_step % 2 == 0) + { + WS2812B_SetDiodeColor(0 + n, WHITE); + WS2812B_SetDiodeColor(0 + m, WHITE); + delay = 20; + } + else + { + WS2812B_SetDiodeColor(0 + n, color_wheel(mAuxParam)); + WS2812B_SetDiodeColor(0 + m, BLACK); + delay = 30; + } + } + else + { + mCounterModeStep = (mCounterModeStep + 1) % WS2812B_LEDS; + + if(mCounterModeStep == 0) + { + mAuxParam = get_random_wheel_index(mAuxParam); + } + } + iModeDelay = delay; +} + + +/* + * Alternating pixels running function. + */ +void running(uint32_t color1, uint32_t color2) +{ + for(uint16_t i=0; i < WS2812B_LEDS; i++) + { + if((i + mCounterModeStep) % 4 < 2) + { +// if(IS_REVERSE) { +// WS2812B_SetDiodeColor(0 + i, color1); +// } else { + WS2812B_SetDiodeColor(WS2812B_LEDS - i, color1); +// } + } else { +// if(IS_REVERSE) { +// WS2812B_SetDiodeColor(0 + i, color2); +// } else { + WS2812B_SetDiodeColor(WS2812B_LEDS - i, color2); +// } + } + } + + mCounterModeStep = (mCounterModeStep + 1) & 0x3; + iModeDelay = mSpeed; +} + +/* + * Alternating color/white pixels running. + */ +void mode_running_color(void) +{ + return running(mModeColor[0], WHITE); +} + + +/* + * Alternating red/blue pixels running. + */ +void mode_running_red_blue(void) +{ + return running(RED, BLUE); +} + + +/* + * Alternating red/green pixels running. + */ +void mode_merry_christmas(void) +{ + return running(RED, GREEN); +} + +/* + * Alternating orange/purple pixels running. + */ +void mode_halloween(void) +{ + return running(PURPLE, ORANGE); +} + + +/* + * Random colored pixels running. + */ +void mode_running_random(void) { + for(uint16_t i=WS2812B_LEDS-1; i > 0; i--) { +// if(IS_REVERSE) { +// WS2812B_SetDiodeColor(WS2812B_LEDS - i, Adafruit_NeoPixel::getPixelColor(WS2812B_LEDS - i + 1)); +// } else { + WS2812B_SetDiodeColor(0 + i, WS2812B_GetColor(0 + i - 1)); +// } + } + + if(mCounterModeStep == 0) + { + mAuxParam = get_random_wheel_index(mAuxParam); +// if(IS_REVERSE) { +// WS2812B_SetDiodeColor(WS2812B_LEDS, color_wheel(mAuxParam)); +// } else { + WS2812B_SetDiodeColor(0, color_wheel(mAuxParam)); +// } + } + + mCounterModeStep = (mCounterModeStep == 0) ? 1 : 0; + iModeDelay = mSpeed; +} + + +/* + * K.I.T.T. + */ +void mode_larson_scanner(void) { + fade_out(); + + if(mCounterModeStep < WS2812B_LEDS) + { +// if(IS_REVERSE) { +// WS2812B_SetDiodeColor(WS2812B_LEDS - mCounterModeStep, mModeColor[0]); +// } else { + WS2812B_SetDiodeColor(0 + mCounterModeStep, mModeColor[0]); +// } + } + else + { +// if(IS_REVERSE) { +// WS2812B_SetDiodeColor(WS2812B_LEDS - ((WS2812B_LEDS * 2) - mCounterModeStep) + 2, mModeColor[0]); +// } else { + WS2812B_SetDiodeColor(0 + ((WS2812B_LEDS * 2) - mCounterModeStep) - 2, mModeColor[0]); +// } + } + + if(mCounterModeStep % WS2812B_LEDS == 0) mCycle = 1; + else mCycle = 1; + + mCounterModeStep = (mCounterModeStep + 1) % ((WS2812B_LEDS * 2) - 2); + iModeDelay = mSpeed; +} + + +/* + * Firing comets from one end. + */ +void mode_comet(void) { + fade_out(); + +// if(IS_REVERSE) { +// WS2812B_SetDiodeColor(WS2812B_LEDS - mCounterModeStep, mModeColor[0]); +// } else { + WS2812B_SetDiodeColor(0 + mCounterModeStep, mModeColor[0]); +// } + + mCounterModeStep = (mCounterModeStep + 1) % WS2812B_LEDS; + iModeDelay = mSpeed; +} + + +/* + * Fireworks function. + */ +void fireworks(uint32_t color) { + fade_out(); + +//// set brightness(i) = brightness(i-1)/4 + brightness(i) + brightness(i+1)/4 +/* +// the old way, so many calls to the pokey getPixelColor() function made this super slow + for(uint16_t i=0 + 1; i > 2) & 0x3F3F3F3F; + uint32_t thisLed = Adafruit_NeoPixel::getPixelColor(i); + uint32_t nextLed = (Adafruit_NeoPixel::getPixelColor(i+1) >> 2) & 0x3F3F3F3F; + WS2812B_SetDiodeColor(i, prevLed + thisLed + nextLed); + } +*/ + +// the new way, manipulate the Adafruit_NeoPixels pixels[] array directly, about 5x faster + uint8_t *pixels = WS2812B_GetPixels(); + uint8_t pixelsPerLed = 3; + uint16_t startPixel = 0 * pixelsPerLed + pixelsPerLed; + uint16_t stopPixel = WS2812B_LEDS * pixelsPerLed ; + for(uint16_t i=startPixel; i > 2) + + pixels[i] + + (pixels[i + pixelsPerLed] >> 2); + pixels[i] = tmpPixel > 255 ? 255 : tmpPixel; + } + + if(!mTriggered) + { + for(uint16_t i=0; i> 16) & 0xFF; + uint8_t g = (mModeColor[0] >> 8) & 0xFF; + uint8_t b = (mModeColor[0] & 0xFF); + uint8_t lum = MAX(r, MAX(g, b)) / rev_intensity; + for(uint16_t i=0; i <= WS2812B_LEDS; i++) + { + int flicker = rand()%lum; + WS2812B_SetDiodeRGB(i, MAX(r - flicker, 0), MAX(g - flicker, 0), MAX(b - flicker, 0)); + } + iModeDelay = mSpeed; +} + +/* + * Random flickering. + */ +void mode_fire_flicker(void) +{ + return fire_flicker(3); +} + +/* +* Random flickering, less intensity. +*/ +void mode_fire_flicker_soft(void) +{ + return fire_flicker(6); +} + +/* +* Random flickering, more intensity. +*/ +void mode_fire_flicker_intense(void) +{ + return fire_flicker(1.7); +} + + +/* + * Tricolor chase function + */ +void tricolor_chase(uint32_t color1, uint32_t color2, uint32_t color3) +{ + uint16_t index = mCounterModeStep % 6; + for(uint16_t i=0; i < WS2812B_LEDS; i++, index++) + { + if(index > 5) index = 0; + + uint32_t color = color3; + if(index < 2) color = color1; + else if(index < 4) color = color2; + +// if(IS_REVERSE) { +// WS2812B_SetDiodeColor(0 + i, color); +// } else { + WS2812B_SetDiodeColor(WS2812B_LEDS - i, color); +// } + } + + mCounterModeStep++; + iModeDelay = mSpeed; +} + + +/* + * Tricolor chase mode + */ +void mode_tricolor_chase(void) +{ + return tricolor_chase(mModeColor[0], mModeColor[1], mModeColor[2]); +} + + +/* + * Alternating white/red/black pixels running. + */ +void mode_circus_combustus(void) +{ + return tricolor_chase(RED, WHITE, BLACK); +} + +/* + * ICU mode + */ +void mode_icu(void) +{ + uint16_t dest = mCounterModeStep & 0xFFFF; + + WS2812B_SetDiodeColor(0 + dest, mModeColor[0]); + WS2812B_SetDiodeColor(0 + dest + WS2812B_LEDS/2, mModeColor[0]); + + if(mAuxParam16b == dest) + { // pause between eye movements + if(rand()%6 == 0) + { // blink once in a while + WS2812B_SetDiodeColor(0 + dest, BLACK); + WS2812B_SetDiodeColor(0 + dest + WS2812B_LEDS/2, BLACK); + iModeDelay = 200; + } + mAuxParam16b = rand() %(WS2812B_LEDS/2); + iModeDelay = 1000 + rand() %2000; + } + + WS2812B_SetDiodeColor(0 + dest, BLACK); + WS2812B_SetDiodeColor(0 + dest + WS2812B_LEDS/2, BLACK); + + if(mAuxParam16b > mCounterModeStep) + { + mCounterModeStep++; + dest++; + } else if (mAuxParam16b < mCounterModeStep) + { + mCounterModeStep--; + dest--; + } + + WS2812B_SetDiodeColor(0 + dest, mModeColor[0]); + WS2812B_SetDiodeColor(0 + dest + WS2812B_LEDS/2, mModeColor[0]); + + iModeDelay = mSpeed; +}