2023-11-07 20:26:15 +00:00
|
|
|
#include "led_handler.hpp"
|
|
|
|
|
|
|
|
TaskHandle_t ledTaskHandle = NULL;
|
2023-11-08 11:18:59 +00:00
|
|
|
QueueHandle_t ledTaskQueue = NULL;
|
|
|
|
Adafruit_NeoPixel pixels(NEOPIXEL_COUNT, NEOPIXEL_PIN, NEO_GRB + NEO_KHZ800);
|
2023-11-10 18:52:06 +00:00
|
|
|
uint ledTaskParams;
|
2023-11-07 20:26:15 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void ledTask(void *parameter) {
|
|
|
|
while (1) {
|
|
|
|
if (ledTaskQueue != NULL) {
|
|
|
|
if (xQueueReceive(ledTaskQueue, &ledTaskParams, portMAX_DELAY) ==
|
|
|
|
pdPASS) {
|
2024-03-11 20:21:15 +00:00
|
|
|
|
|
|
|
if (preferences.getBool("disableLeds", false)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
uint32_t oldLights[NEOPIXEL_COUNT];
|
|
|
|
|
|
|
|
// get current state
|
|
|
|
for (int i = 0; i < NEOPIXEL_COUNT; i++) {
|
|
|
|
oldLights[i] = pixels.getPixelColor(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (ledTaskParams) {
|
2023-11-30 21:56:50 +00:00
|
|
|
case LED_POWER_TEST:
|
|
|
|
ledRainbow(20);
|
|
|
|
pixels.clear();
|
|
|
|
break;
|
|
|
|
case LED_EFFECT_WIFI_CONNECT_ERROR:
|
|
|
|
blinkDelayTwoColor(100, 3, pixels.Color(8, 161, 236),
|
|
|
|
pixels.Color(255, 0, 0));
|
|
|
|
break;
|
|
|
|
case LED_FLASH_ERROR:
|
|
|
|
blinkDelayColor(250, 3, 255, 0, 0);
|
|
|
|
break;
|
|
|
|
case LED_EFFECT_HEARTBEAT:
|
|
|
|
blinkDelayColor(150, 2, 0, 0, 255);
|
|
|
|
break;
|
2024-01-31 22:45:26 +00:00
|
|
|
case LED_DATA_BLOCK_ERROR:
|
|
|
|
blinkDelayColor(150, 2, 128, 0, 128);
|
|
|
|
break;
|
|
|
|
case LED_DATA_PRICE_ERROR:
|
|
|
|
blinkDelayColor(150, 2, 177, 90, 31);
|
|
|
|
break;
|
2023-11-30 21:56:50 +00:00
|
|
|
case LED_EFFECT_WIFI_CONNECT_SUCCESS:
|
|
|
|
case LED_FLASH_SUCCESS:
|
|
|
|
blinkDelayColor(150, 3, 0, 255, 0);
|
|
|
|
break;
|
|
|
|
case LED_PROGRESS_100:
|
|
|
|
pixels.setPixelColor(0, pixels.Color(0, 255, 0));
|
|
|
|
case LED_PROGRESS_75:
|
|
|
|
pixels.setPixelColor(1, pixels.Color(0, 255, 0));
|
|
|
|
case LED_PROGRESS_50:
|
|
|
|
pixels.setPixelColor(2, pixels.Color(0, 255, 0));
|
|
|
|
case LED_PROGRESS_25:
|
|
|
|
pixels.setPixelColor(3, pixels.Color(0, 255, 0));
|
2023-11-30 21:38:01 +00:00
|
|
|
pixels.show();
|
2023-11-30 21:56:50 +00:00
|
|
|
break;
|
|
|
|
case LED_FLASH_UPDATE:
|
|
|
|
break;
|
|
|
|
case LED_FLASH_BLOCK_NOTIFY:
|
|
|
|
blinkDelayTwoColor(250, 3, pixels.Color(224, 67, 0),
|
|
|
|
pixels.Color(8, 2, 0));
|
|
|
|
break;
|
|
|
|
case LED_EFFECT_WIFI_WAIT_FOR_CONFIG:
|
|
|
|
blinkDelayTwoColor(100, 1, pixels.Color(8, 161, 236),
|
|
|
|
pixels.Color(156, 225, 240));
|
|
|
|
break;
|
|
|
|
case LED_EFFECT_WIFI_ERASE_SETTINGS:
|
|
|
|
blinkDelay(100, 3);
|
|
|
|
break;
|
|
|
|
case LED_EFFECT_WIFI_CONNECTING:
|
|
|
|
for (int i = NEOPIXEL_COUNT; i >= 0; i--) {
|
|
|
|
for (int j = NEOPIXEL_COUNT; j >= 0; j--) {
|
|
|
|
if (j == i) {
|
|
|
|
pixels.setPixelColor(i, pixels.Color(16, 197, 236));
|
|
|
|
} else {
|
|
|
|
pixels.setPixelColor(j, pixels.Color(0, 0, 0));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pixels.show();
|
|
|
|
vTaskDelay(pdMS_TO_TICKS(100));
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
2023-11-30 21:56:50 +00:00
|
|
|
break;
|
|
|
|
case LED_EFFECT_PAUSE_TIMER:
|
|
|
|
for (int i = NEOPIXEL_COUNT; i >= 0; i--) {
|
|
|
|
for (int j = NEOPIXEL_COUNT; j >= 0; j--) {
|
|
|
|
uint32_t c = pixels.Color(0, 0, 0);
|
|
|
|
if (i == j) c = pixels.Color(0, 255, 0);
|
|
|
|
pixels.setPixelColor(j, c);
|
|
|
|
}
|
2023-11-30 21:38:01 +00:00
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
pixels.show();
|
|
|
|
|
|
|
|
delay(100);
|
|
|
|
}
|
|
|
|
pixels.setPixelColor(0, pixels.Color(255, 0, 0));
|
2023-11-30 21:38:01 +00:00
|
|
|
pixels.show();
|
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
delay(900);
|
2023-11-30 21:38:01 +00:00
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
pixels.clear();
|
|
|
|
pixels.show();
|
|
|
|
break;
|
|
|
|
case LED_EFFECT_START_TIMER:
|
|
|
|
pixels.clear();
|
|
|
|
pixels.setPixelColor((NEOPIXEL_COUNT - 1), pixels.Color(255, 0, 0));
|
|
|
|
pixels.show();
|
2023-11-30 21:38:01 +00:00
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
delay(900);
|
2023-11-30 21:38:01 +00:00
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
for (int i = NEOPIXEL_COUNT; i--; i > 0) {
|
|
|
|
for (int j = NEOPIXEL_COUNT; j--; j > 0) {
|
|
|
|
uint32_t c = pixels.Color(0, 0, 0);
|
|
|
|
if (i == j) c = pixels.Color(0, 255, 0);
|
2023-11-30 21:38:01 +00:00
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
pixels.setPixelColor(j, c);
|
|
|
|
}
|
2023-11-30 21:38:01 +00:00
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
pixels.show();
|
2023-11-30 21:38:01 +00:00
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
delay(100);
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:56:50 +00:00
|
|
|
pixels.clear();
|
2023-11-30 21:38:01 +00:00
|
|
|
pixels.show();
|
2023-11-30 21:56:50 +00:00
|
|
|
break;
|
2023-11-08 11:18:59 +00:00
|
|
|
}
|
2023-11-30 21:38:01 +00:00
|
|
|
|
|
|
|
// revert to previous state unless power test
|
|
|
|
|
|
|
|
for (int i = 0; i < NEOPIXEL_COUNT; i++) {
|
|
|
|
pixels.setPixelColor(i, oldLights[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
pixels.show();
|
|
|
|
}
|
2023-11-07 20:26:15 +00:00
|
|
|
}
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
2023-11-07 20:26:15 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void setupLeds() {
|
|
|
|
pixels.begin();
|
|
|
|
pixels.setBrightness(preferences.getUInt("ledBrightness", 128));
|
|
|
|
pixels.clear();
|
|
|
|
pixels.show();
|
|
|
|
setupLedTask();
|
|
|
|
if (preferences.getBool("ledTestOnPower", true)) {
|
|
|
|
while (!ledTaskQueue) {
|
|
|
|
delay(1);
|
|
|
|
// wait until queue is available
|
2023-11-10 12:02:05 +00:00
|
|
|
}
|
2023-11-30 21:38:01 +00:00
|
|
|
queueLedEffect(LED_POWER_TEST);
|
|
|
|
}
|
2023-11-08 11:18:59 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void setupLedTask() {
|
|
|
|
ledTaskQueue = xQueueCreate(5, sizeof(uint));
|
2023-11-08 11:18:59 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
xTaskCreate(ledTask, "LedTask", 2048, NULL, tskIDLE_PRIORITY, &ledTaskHandle);
|
2023-11-08 11:18:59 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void blinkDelay(int d, int times) {
|
|
|
|
for (int j = 0; j < times; j++) {
|
|
|
|
pixels.setPixelColor(0, pixels.Color(255, 0, 0));
|
|
|
|
pixels.setPixelColor(1, pixels.Color(0, 255, 0));
|
|
|
|
pixels.setPixelColor(2, pixels.Color(255, 0, 0));
|
|
|
|
pixels.setPixelColor(3, pixels.Color(0, 255, 0));
|
|
|
|
pixels.show();
|
|
|
|
vTaskDelay(pdMS_TO_TICKS(d));
|
2023-11-08 11:18:59 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
pixels.setPixelColor(0, pixels.Color(255, 255, 0));
|
|
|
|
pixels.setPixelColor(1, pixels.Color(0, 255, 255));
|
|
|
|
pixels.setPixelColor(2, pixels.Color(255, 255, 0));
|
|
|
|
pixels.setPixelColor(3, pixels.Color(0, 255, 255));
|
2023-11-08 11:18:59 +00:00
|
|
|
pixels.show();
|
2023-11-30 21:38:01 +00:00
|
|
|
vTaskDelay(pdMS_TO_TICKS(d));
|
|
|
|
}
|
|
|
|
pixels.clear();
|
|
|
|
pixels.show();
|
2023-11-07 20:26:15 +00:00
|
|
|
}
|
2023-11-08 11:18:59 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void blinkDelayColor(int d, int times, uint r, uint g, uint b) {
|
|
|
|
for (int j = 0; j < times; j++) {
|
|
|
|
for (int i = 0; i < NEOPIXEL_COUNT; i++) {
|
|
|
|
pixels.setPixelColor(i, pixels.Color(r, g, b));
|
|
|
|
}
|
2023-11-08 11:18:59 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
pixels.show();
|
|
|
|
vTaskDelay(pdMS_TO_TICKS(d));
|
2023-11-08 11:18:59 +00:00
|
|
|
|
|
|
|
pixels.clear();
|
|
|
|
pixels.show();
|
2023-11-30 21:38:01 +00:00
|
|
|
vTaskDelay(pdMS_TO_TICKS(d));
|
|
|
|
}
|
|
|
|
pixels.clear();
|
|
|
|
pixels.show();
|
2023-11-08 11:18:59 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void blinkDelayTwoColor(int d, int times, uint32_t c1, uint32_t c2) {
|
|
|
|
for (int j = 0; j < times; j++) {
|
|
|
|
for (int i = 0; i < NEOPIXEL_COUNT; i++) {
|
|
|
|
pixels.setPixelColor(i, c1);
|
2023-11-08 11:18:59 +00:00
|
|
|
}
|
|
|
|
pixels.show();
|
2023-11-30 21:38:01 +00:00
|
|
|
vTaskDelay(pdMS_TO_TICKS(d));
|
2023-11-08 11:18:59 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
for (int i = 0; i < NEOPIXEL_COUNT; i++) {
|
|
|
|
pixels.setPixelColor(i, c2);
|
|
|
|
}
|
2023-11-08 11:18:59 +00:00
|
|
|
pixels.show();
|
2023-11-30 21:38:01 +00:00
|
|
|
vTaskDelay(pdMS_TO_TICKS(d));
|
|
|
|
}
|
|
|
|
pixels.clear();
|
|
|
|
pixels.show();
|
2023-11-08 11:18:59 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void clearLeds() {
|
|
|
|
preferences.putBool("ledStatus", false);
|
|
|
|
pixels.clear();
|
|
|
|
pixels.show();
|
2023-11-08 14:27:22 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void setLights(int r, int g, int b) { setLights(pixels.Color(r, g, b)); }
|
2023-11-12 23:33:48 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void setLights(uint32_t color) {
|
|
|
|
bool ledStatus = true;
|
2023-11-18 13:03:47 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
for (int i = 0; i < NEOPIXEL_COUNT; i++) {
|
|
|
|
pixels.setPixelColor(i, color);
|
|
|
|
}
|
|
|
|
pixels.show();
|
2023-11-08 14:27:22 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
if (color == pixels.Color(0, 0, 0)) {
|
|
|
|
ledStatus = false;
|
|
|
|
} else {
|
|
|
|
saveLedState();
|
|
|
|
}
|
|
|
|
preferences.putBool("ledStatus", ledStatus);
|
2023-11-18 13:03:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void saveLedState() {
|
2023-11-30 21:38:01 +00:00
|
|
|
for (int i = 0; i < pixels.numPixels(); i++) {
|
|
|
|
int pixelColor = pixels.getPixelColor(i);
|
|
|
|
char key[12];
|
|
|
|
snprintf(key, 12, "%s%d", "ledColor_", i);
|
|
|
|
preferences.putUInt(key, pixelColor);
|
|
|
|
}
|
|
|
|
|
|
|
|
xTaskNotifyGive(eventSourceTaskHandle);
|
2023-11-18 13:03:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void restoreLedState() {
|
2023-11-30 21:38:01 +00:00
|
|
|
for (int i = 0; i < pixels.numPixels(); i++) {
|
|
|
|
char key[12];
|
|
|
|
snprintf(key, 12, "%s%d", "ledColor_", i);
|
|
|
|
uint pixelColor = preferences.getUInt(key, pixels.Color(0, 0, 0));
|
|
|
|
pixels.setPixelColor(i, pixelColor);
|
|
|
|
}
|
|
|
|
|
|
|
|
pixels.show();
|
2023-11-08 11:18:59 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
QueueHandle_t getLedTaskQueue() { return ledTaskQueue; }
|
2023-11-08 11:18:59 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
bool queueLedEffect(uint effect) {
|
|
|
|
if (ledTaskQueue == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
2023-11-08 11:18:59 +00:00
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
uint flashType = effect;
|
|
|
|
xQueueSend(ledTaskQueue, &flashType, portMAX_DELAY);
|
2023-11-12 23:33:48 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void ledRainbow(int wait) {
|
|
|
|
// Hue of first pixel runs 5 complete loops through the color wheel.
|
|
|
|
// Color wheel has a range of 65536 but it's OK if we roll over, so
|
|
|
|
// just count from 0 to 5*65536. Adding 256 to firstPixelHue each time
|
|
|
|
// means we'll make 5*65536/256 = 1280 passes through this loop:
|
|
|
|
for (long firstPixelHue = 0; firstPixelHue < 5 * 65536;
|
|
|
|
firstPixelHue += 256) {
|
|
|
|
// strip.rainbow() can take a single argument (first pixel hue) or
|
|
|
|
// optionally a few extras: number of rainbow repetitions (default 1),
|
|
|
|
// saturation and value (brightness) (both 0-255, similar to the
|
|
|
|
// ColorHSV() function, default 255), and a true/false flag for whether
|
|
|
|
// to apply gamma correction to provide 'truer' colors (default true).
|
|
|
|
pixels.rainbow(firstPixelHue);
|
|
|
|
// Above line is equivalent to:
|
|
|
|
// strip.rainbow(firstPixelHue, 1, 255, 255, true);
|
2023-11-30 21:56:50 +00:00
|
|
|
pixels.show(); // Update strip with new contents
|
2023-11-30 21:38:01 +00:00
|
|
|
delayMicroseconds(wait);
|
|
|
|
// vTaskDelay(pdMS_TO_TICKS(wait)); // Pause for a moment
|
|
|
|
}
|
2023-11-12 23:33:48 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void ledTheaterChase(uint32_t color, int wait) {
|
2023-11-30 21:56:50 +00:00
|
|
|
for (int a = 0; a < 10; a++) { // Repeat 10 times...
|
|
|
|
for (int b = 0; b < 3; b++) { // 'b' counts from 0 to 2...
|
|
|
|
pixels.clear(); // Set all pixels in RAM to 0 (off)
|
2023-11-30 21:38:01 +00:00
|
|
|
// 'c' counts up from 'b' to end of strip in steps of 3...
|
|
|
|
for (int c = b; c < pixels.numPixels(); c += 3) {
|
2023-11-30 21:56:50 +00:00
|
|
|
pixels.setPixelColor(c, color); // Set pixel 'c' to value 'color'
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
2023-11-30 21:56:50 +00:00
|
|
|
pixels.show(); // Update strip with new contents
|
|
|
|
vTaskDelay(pdMS_TO_TICKS(wait)); // Pause for a moment
|
2023-11-12 23:33:48 +00:00
|
|
|
}
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
2023-11-12 23:33:48 +00:00
|
|
|
}
|
|
|
|
|
2023-11-30 21:38:01 +00:00
|
|
|
void ledTheaterChaseRainbow(int wait) {
|
2023-11-30 21:56:50 +00:00
|
|
|
int firstPixelHue = 0; // First pixel starts at red (hue 0)
|
|
|
|
for (int a = 0; a < 30; a++) { // Repeat 30 times...
|
|
|
|
for (int b = 0; b < 3; b++) { // 'b' counts from 0 to 2...
|
|
|
|
pixels.clear(); // Set all pixels in RAM to 0 (off)
|
2023-11-30 21:38:01 +00:00
|
|
|
// 'c' counts up from 'b' to end of strip in increments of 3...
|
|
|
|
for (int c = b; c < pixels.numPixels(); c += 3) {
|
|
|
|
// hue of pixel 'c' is offset by an amount to make one full
|
|
|
|
// revolution of the color wheel (range 65536) along the length
|
|
|
|
// of the strip (strip.numPixels() steps):
|
|
|
|
int hue = firstPixelHue + c * 65536L / pixels.numPixels();
|
2023-11-30 21:56:50 +00:00
|
|
|
uint32_t color = pixels.gamma32(pixels.ColorHSV(hue)); // hue -> RGB
|
|
|
|
pixels.setPixelColor(c, color); // Set pixel 'c' to value 'color'
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
2023-11-30 21:56:50 +00:00
|
|
|
pixels.show(); // Update strip with new contents
|
|
|
|
vTaskDelay(pdMS_TO_TICKS(wait)); // Pause for a moment
|
|
|
|
firstPixelHue += 65536 / 90; // One cycle of color wheel over 90 frames
|
2023-11-12 23:33:48 +00:00
|
|
|
}
|
2023-11-30 21:38:01 +00:00
|
|
|
}
|
2023-11-12 23:33:48 +00:00
|
|
|
}
|
|
|
|
|
2024-04-27 14:48:06 +00:00
|
|
|
Adafruit_NeoPixel getPixels() { return pixels; }
|
|
|
|
|
|
|
|
#ifdef HAS_FRONTLIGHT
|
|
|
|
int flDelayTime = 10;
|
|
|
|
|
|
|
|
void frontlightFadeInAll() {
|
|
|
|
for (int dutyCycle = 0; dutyCycle <= preferences.getUInt("flMaxBrightness"); dutyCycle += 5) {
|
|
|
|
for (int ledPin = 0; ledPin < NUM_SCREENS; ledPin++) {
|
|
|
|
flArray.setPWM(ledPin, 0, dutyCycle);
|
|
|
|
}
|
|
|
|
delay(flDelayTime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void frontlightFadeOutAll() {
|
|
|
|
for (int dutyCycle = preferences.getUInt("flMaxBrightness"); dutyCycle >= 0; dutyCycle -= 5) {
|
|
|
|
for (int ledPin = 0; ledPin < NUM_SCREENS; ledPin++) {
|
|
|
|
flArray.setPWM(ledPin, 0, dutyCycle);
|
|
|
|
}
|
|
|
|
delay(flDelayTime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void frontlightFadeIn(uint num) {
|
|
|
|
for (int dutyCycle = 0; dutyCycle <= preferences.getUInt("flMaxBrightness"); dutyCycle += 5) {
|
|
|
|
flArray.setPWM(num, 0, dutyCycle);
|
|
|
|
delay(flDelayTime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void frontlightFadeOut(uint num) {
|
|
|
|
for (int dutyCycle = preferences.getUInt("flMaxBrightness"); dutyCycle >= 0; dutyCycle -= 5) {
|
|
|
|
flArray.setPWM(num, 0, dutyCycle);
|
|
|
|
delay(flDelayTime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|