From fe2295d310fa11572dde87a9bfcdfcd05b9ccdba Mon Sep 17 00:00:00 2001 From: gene Date: Tue, 2 Jul 2024 00:44:11 +0200 Subject: [PATCH] v1.2.29 - Overall `color-light` firmware improvements - + added "White stripes" and "Kaleidoscope" styles fx - + added "Strobe effect" speed presets selection - Added `Config::onWiFiConfigured` callback --- examples/color-light/color-fx.h | 161 +++++++++++++++++++++ examples/color-light/color-light.cpp | 186 +++++++++++++------------ src/Config.cpp | 5 +- src/Config.h | 16 ++- src/defs.h | 7 +- src/io/IOManager.h | 2 +- src/net/WiFiManager.h | 4 + src/service/api/devices/ColorLight.cpp | 4 +- src/service/api/devices/ColorLight.h | 24 ++-- src/service/api/devices/Dimmer.cpp | 2 +- src/service/api/devices/Dimmer.h | 2 +- src/service/api/devices/Switch.cpp | 2 + src/service/api/devices/Switch.h | 1 + src/version.h | 2 +- 14 files changed, 304 insertions(+), 114 deletions(-) create mode 100644 examples/color-light/color-fx.h diff --git a/examples/color-light/color-fx.h b/examples/color-light/color-fx.h new file mode 100644 index 0000000..7206081 --- /dev/null +++ b/examples/color-light/color-fx.h @@ -0,0 +1,161 @@ +/* + * HomeGenie-Mini (c) 2018-2024 G-Labs + * + * + * This file is part of HomeGenie-Mini (HGM). + * + * HomeGenie-Mini is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * HomeGenie-Mini is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with HomeGenie-Mini. If not, see . + * + * + * Authors: + * - Generoso Martello + * + * + */ + +#include + +using namespace Service; +using namespace Service::API::devices; + +typedef LightColor* AnimatedColor; +AnimatedColor* animatedColors = nullptr; + +float currentSaturation; +float hueOffset = 0; +float hueRange = 1; +float cursorDirection = 1; + +void fx_reset(Adafruit_NeoPixel* pixels, LightColor& color) { +// hueOffset = 0; + hueRange = 1; +} + +void fx_solid(Adafruit_NeoPixel* pixels, LightColor& color) { + float r = color.getRed(); + float g = color.getGreen(); + float b = color.getBlue(); + for (int i = 0; i < pixels->numPixels(); i++) { + pixels->setPixelColor(i, r, g, b); +// animatedColors[i]->setColor(color.getHue(), color.getSaturation(), color.getValue(), 0); + } +} + +void fx_rainbow(Adafruit_NeoPixel* pixels, LightColor& color) { + if (color.getSaturation() > 1) { + return; + // TODO: ... + currentSaturation = 1; + } else { + currentSaturation = color.getSaturation(); + } + + if (pixels != nullptr) { + float h = color.getHue() + hueOffset; + if (h > 1) h = ((int)round(h * 10000) % 10000) / 10000.0f; + float hueStep = 1.0f / (float) pixels->numPixels(); + hueStep /= hueRange; + float v = color.getValue(); + for (int i = 0; i < pixels->numPixels(); i++) { + h += hueStep; + auto rgb = Utility::hsv2rgb(h, currentSaturation, v); + pixels->setPixelColor(i, rgb.r, rgb.g, rgb.b); + animatedColors[i]->setColor(h, currentSaturation, v, 0); + } + } + + // animate + hueOffset += (0.128f / pixels->numPixels()); + if (hueOffset > 1) hueOffset = 0; + hueRange += (1.5f / pixels->numPixels()) * cursorDirection; + if (hueRange > 5) { + cursorDirection *= -1; + hueRange = 5; + } else if (hueRange < 1) { + cursorDirection *= -1; + hueRange = 1; + } +} + + +unsigned long stripe_refresh_ts = 0; +int stripe_delay = 200; // ms +unsigned long stripe_transition = 200; // ms +int stripe_step = 3; +int stripe_length = 9; +int shift = 0; + +void fx_white_stripes(Adafruit_NeoPixel* pixels, LightColor& color) { + + // animate + if (millis() - stripe_refresh_ts > stripe_delay) { + + shift = shift % stripe_length; + + if (pixels != nullptr) { + for (int i = 0; i < pixels->numPixels(); i++) { + if ((i + shift) % stripe_length < stripe_step) { + // draw stripe + animatedColors[i]->setColor(0, 0, color.getValue(), stripe_transition); + } else { + // draw solid color + animatedColors[i]->setColor(color.getHue(), color.getSaturation(), + color.getValue(), stripe_transition); + } + } + } + + + shift++; + if (shift >= stripe_length) shift = 0; + stripe_refresh_ts = millis(); + } + + // render + if (pixels != nullptr) { + for (int i = 0; i < pixels->numPixels(); i++) { + pixels->setPixelColor(i, animatedColors[i]->getRed(), animatedColors[i]->getGreen(), + animatedColors[i]->getBlue()); + } + } + +} + + +unsigned long kaleidoscope_refresh_ts = 0; +int kaleidoscope_delay = 500; + +void fx_kaleidoscope(Adafruit_NeoPixel* pixels, LightColor& color) { + + // animate + if (millis() - kaleidoscope_refresh_ts > kaleidoscope_delay) { + for (int i = 0; i < pixels->numPixels(); i++) { + float rnd1 = random(1000); + float rnd2 = random(1000); + animatedColors[i]->setColor(rnd1 / 1000, rnd2 / 1000, color.getValue(), 500); + } + + kaleidoscope_refresh_ts = millis(); + } + + // render + if (pixels != nullptr) { + for (int i = 0; i < pixels->numPixels(); i++) { + animatedColors[i]->setValue(color.getValue()); + pixels->setPixelColor(i, animatedColors[i]->getRed(), animatedColors[i]->getGreen(), + animatedColors[i]->getBlue()); + } + } + +} diff --git a/examples/color-light/color-light.cpp b/examples/color-light/color-light.cpp index 72f7429..758b492 100644 --- a/examples/color-light/color-light.cpp +++ b/examples/color-light/color-light.cpp @@ -24,17 +24,17 @@ */ #include -#include #include #include "configuration.h" - +#include "color-fx.h" using namespace Service; using namespace Service::API::devices; HomeGenie* homeGenie; +bool isConfigured = false; // Optional RGB Status LED Adafruit_NeoPixel* statusLED = nullptr; @@ -43,9 +43,6 @@ Adafruit_NeoPixel* statusLED = nullptr; int count = 0; int pin = -1; Adafruit_NeoPixel* pixels = nullptr; -bool changed = false; -unsigned long lastRefreshTs = 0; - // LED Blink callback when statusLED is configured void statusLedCallback(bool isLedOn) { if (isLedOn) { @@ -56,71 +53,17 @@ void statusLedCallback(bool isLedOn) { statusLED->show(); } -bool isConfigured = false; -float currentSaturation; -float hueOffset = 0; -float hueRange = 1; -float cursorDirection = 1; -LightColor currentColor; +unsigned int refreshMs = 10; +unsigned long lastRefreshTs = 0; +LightColor currentColor; ColorLight* mainModule; -ColorLight* rainbowModule; - -void fx_reset(LightColor& color) { -// hueOffset = 0; - hueRange = 1; - if (color.getSaturation() > 1) { - // TODO: ... - currentSaturation = 1; - } else { - currentSaturation = color.getSaturation(); - } -} -void fx_main(LightColor& color) { - float r = color.getRed(); - float g = color.getGreen(); - float b = color.getBlue(); - if (statusLED != nullptr) { - statusLED->setPixelColor(0, r, g, b); - } - if (pixels != nullptr) { - for (int i = 0; i < count; i++) { - pixels->setPixelColor(i, r, g, b); - } - } -} - -void fx_rainbow(LightColor& color) { - if (statusLED != nullptr) { - statusLED->setPixelColor(0, color.getRed(), color.getGreen(), color.getBlue()); - } - - if (pixels != nullptr) { - float h = color.getHue() + hueOffset; - if (h > 1) h = ((int)round(h * 10000) % 10000) / 10000.0f; - float hueStep = 1.0f / (float) count; - hueStep /= hueRange; - float v = color.getValue(); - for (int i = 0; i < count; i++) { - h += hueStep; - auto rgb = Utility::hsv2rgb(h, currentSaturation, v); - pixels->setPixelColor(i, rgb.r, rgb.g, rgb.b); - } - } +ModuleParameter* fxStyle; +ModuleParameter* fxStrobe; - // animate - hueOffset += (0.128f / count); - if (hueOffset > 1) hueOffset = 0; - hueRange += (1.5f / count) * cursorDirection; - if (hueRange > 5) { - cursorDirection *= -1; - hueRange = 5; - } else if (hueRange < 1) { - cursorDirection *= -1; - hueRange = 1; - } -} +bool playFxStrobe; +String currentStyle = "solid"; void refresh() { if (statusLED != nullptr) { @@ -133,6 +76,19 @@ void refresh() { void setup() { homeGenie = HomeGenie::getInstance(); + auto miniModule = homeGenie->getDefaultModule(); + miniModule->name = "Smart light"; + miniModule->properties.add( + new ModuleParameter("Widget.OptionField.FX.Rainbow", + "select:FX.Style:light_style:solid|rainbow|white_stripes|kaleidoscope")); + miniModule->properties.add( + new ModuleParameter("Widget.OptionField.FX.Strobe", + "select:FX.Strobe:strobe_effect:off|slow|medium|fast")); + + fxStyle = new ModuleParameter("FX.Style", "solid"); + miniModule->properties.add(fxStyle); + fxStrobe = new ModuleParameter("FX.Strobe", "false"); + miniModule->properties.add(fxStrobe); // Get status LED config int statusLedPin = Config::getSetting("stld-pin", "-1").toInt(); @@ -151,6 +107,12 @@ void setup() { Config::statusLedCallback(&statusLedCallback); } + Config::setOnWiFiConfiguredCallback([]{ + Serial.println("Restarting now."); + delay(2000); + ESP.restart(); + }); + } else { // Get LED strip config @@ -166,34 +128,32 @@ void setup() { } } - // Setup Status LED as master channel + // Setup main LEDs control module mainModule = new ColorLight(IO::IOEventDomains::HomeAutomation_HomeGenie, "C1", "Main"); + mainModule->module->properties.add( + new ModuleParameter(IOEventPaths::Status_Level, "0")); + mainModule->module->properties.add( + new ModuleParameter(IOEventPaths::Status_ColorHsb, "0,0,1")); + mainModule->module->properties.add( + new ModuleParameter("Widget.Preference.AudioLight", "true")); mainModule->onSetColor([](LightColor color) { currentColor = color; - fx_reset(color); - fx_main(color); + fx_reset(pixels, color); + fx_solid(pixels, color); }); homeGenie->addAPIHandler(mainModule); - // Setup example input "processor" module - // changing color of this module will affect - // all LED pixels with a color cycle effect - rainbowModule = new ColorLight(IO::IOEventDomains::HomeAutomation_HomeGenie, "F1", "Rainbow"); - auto soundLightFeature = new ModuleParameter("Widget.Preference.AudioLight", "true"); - rainbowModule->module->properties.add(soundLightFeature); - rainbowModule->onSetColor([](LightColor color) { - currentColor = color; - fx_reset(color); - fx_rainbow(color); - }); - homeGenie->addAPIHandler(rainbowModule); + // Allocate "animated" colors + animatedColors = new AnimatedColor[count]; + for (int i = 0; i < count; i++) { + animatedColors[i] = new LightColor(); + animatedColors[i]->setColor(currentColor.getHue(), currentColor.getSaturation(), currentColor.getValue(), 0); + } } homeGenie->begin(); - // TODO: implement color/status recall on start - // TODO: implement color/status recall on start // TODO: implement color/status recall on start if (statusLED != nullptr) { @@ -212,19 +172,67 @@ void loop() if (isConfigured) { // refresh background/strobe layer - if (currentColor.isAnimating()) { + if (currentColor.isAnimating() && currentStyle == "solid") { refresh(); } // 40 fps animation FX layer - if (millis() - lastRefreshTs > 25) { + if (millis() - lastRefreshTs > refreshMs) { + + // Check if current rendering style changed and update fx switches + if (currentStyle != fxStyle->value) { + currentStyle = fxStyle->value; + if (currentStyle == "solid") { + fx_solid(pixels, currentColor); + refresh(); + } + } + + if (playFxStrobe && fxStrobe->value == "off") { + playFxStrobe = false; + fx_solid(pixels, currentColor); + refresh(); + } else if (!playFxStrobe && (fxStrobe->value == "slow" || fxStrobe->value == "medium" || fxStrobe->value == "fast")) { + playFxStrobe = true; + } - // FX - rainbow animation - if (rainbowModule->isOn()) { - fx_rainbow(currentColor); + if (currentStyle != "solid") { + // overlay selected effect + if (currentStyle == "rainbow") { + fx_rainbow(pixels, currentColor); + } else if (currentStyle == "kaleidoscope") { + fx_kaleidoscope(pixels, currentColor); + } else if (currentStyle == "white_stripes") { + fx_white_stripes(pixels, currentColor); + } refresh(); } - // TODO: add more FX modules + + if (playFxStrobe) { + float speed = 2; + if (fxStrobe->value == "medium") { + speed = 4; + } + if (fxStrobe->value == "slow") { + speed = 6; + } + // TODO: rewrite without using delays + delay(refreshMs * speed); + // invert solid color + auto c = LightColor(); + c.setColor(0, 0, 1, 0); + fx_solid(pixels, c); + refresh(); + delay(refreshMs); + // restore solid color + fx_solid(pixels, currentColor); + refresh(); + delay(refreshMs); + } + + if (statusLED != nullptr) { + statusLED->setPixelColor(0, currentColor.getRed(), currentColor.getGreen(), currentColor.getBlue()); + } lastRefreshTs = millis(); } diff --git a/src/Config.cpp b/src/Config.cpp index fb5811d..11bf40e 100644 --- a/src/Config.cpp +++ b/src/Config.cpp @@ -36,8 +36,10 @@ SystemConfig Config::system; short Config::ServiceButtonPin = -1; short Config::StatusLedPin = -1; - bool Config::isStatusLedOn = false; +std::function Config::ledCallback = nullptr; +std::function Config::wifiConfiguredCallback = nullptr; + void Config::statusLedOn() { isStatusLedOn = true; if (Config::StatusLedPin >= 0) digitalWrite(Config::StatusLedPin, HIGH); @@ -48,7 +50,6 @@ void Config::statusLedOff() { if (Config::StatusLedPin >= 0) digitalWrite(Config::StatusLedPin, LOW); if (ledCallback != nullptr) ledCallback(false); } -std::function Config::ledCallback = nullptr; void Config::statusLedCallback(std::function callback) { ledCallback = std::move(callback); } diff --git a/src/Config.h b/src/Config.h index df2d3dd..4f0ace2 100644 --- a/src/Config.h +++ b/src/Config.h @@ -103,8 +103,9 @@ class SystemConfig { class Config { public: static short ServiceButtonPin; - static short StatusLedPin; const static uint16_t ConfigureButtonPushInterval = 3000; + static short StatusLedPin; + static bool isStatusLedOn; static ZoneConfig zone; static SystemConfig system; #ifdef ESP32 @@ -129,6 +130,14 @@ class Config { static bool isWiFiConfigured() { return !system.systemMode.equals("config") && !system.ssid.isEmpty(); } + static void onWiFiConfigured() { + if (wifiConfiguredCallback != nullptr) { + wifiConfiguredCallback(); + } + } + static void setOnWiFiConfiguredCallback(std::function callback) { + wifiConfiguredCallback = std::move(callback); + } static bool saveSetting(const char* key, String& value) { String k = String("$") + String(key); @@ -168,8 +177,6 @@ class Config { return value; } - static std::function ledCallback; - static bool isStatusLedOn; static void statusLedOn(); static void statusLedOff(); @@ -217,6 +224,9 @@ class Config { } static void handleConfigCommand(String &message); +private: + static std::function ledCallback; + static std::function wifiConfiguredCallback; }; #endif //HOMEGENIE_MINI_CONFIG_H diff --git a/src/defs.h b/src/defs.h index f7cfa8c..af50dbc 100644 --- a/src/defs.h +++ b/src/defs.h @@ -47,9 +47,10 @@ // Automatically enable FreeRTOS task on devices with additional RAM // (is disabled otherwise saving about ~12K or RAM) -#ifdef BOARD_HAS_PSRAM -#define CONFIG_AUTOMATION_SPAWN_FREERTOS_TASK -#endif +//#ifdef BOARD_HAS_PSRAM +// TODO: experimental flag (rc issues) +//#define CONFIG_AUTOMATION_SPAWN_FREERTOS_TASK +//#endif // disabling SSE and MQTT saves only ~2K of RAM //#define DISABLE_SSE diff --git a/src/io/IOManager.h b/src/io/IOManager.h index 63922ed..682eb7a 100644 --- a/src/io/IOManager.h +++ b/src/io/IOManager.h @@ -51,7 +51,7 @@ namespace IO { bool addEventSender(IIOEventSender *); // IIOEventReceiver interface - void onIOEvent(IIOEventSender *, const char*, const char*, const char *, void *, IOEventDataType); + void onIOEvent(IIOEventSender *, const char*, const char*, const char *, void *, IOEventDataType) override; void setEventReceiver(IIOEventReceiver *); diff --git a/src/net/WiFiManager.h b/src/net/WiFiManager.h index a65695c..ae3d1fd 100644 --- a/src/net/WiFiManager.h +++ b/src/net/WiFiManager.h @@ -71,10 +71,14 @@ namespace Net { #ifdef CONFIGURE_WITH_WPS static void setWiFiConfigured() { + bool wasConfigured = Config::isWiFiConfigured(); Preferences preferences; preferences.begin(CONFIG_SYSTEM_NAME, false); preferences.putString(CONFIG_KEY_wifi_ssid, WiFi.SSID()); preferences.end(); + if (!wasConfigured) { + Config::onWiFiConfigured(); + } } #ifdef ESP32 static esp_wps_config_t wps_config; diff --git a/src/service/api/devices/ColorLight.cpp b/src/service/api/devices/ColorLight.cpp index 9b70bbd..5a07219 100644 --- a/src/service/api/devices/ColorLight.cpp +++ b/src/service/api/devices/ColorLight.cpp @@ -32,7 +32,7 @@ namespace Service { namespace API { namespace devices { // add properties auto propStatusColorHsb = new ModuleParameter(IOEventPaths::Status_ColorHsb); - propStatusColorHsb->setValue("1,1,0,.5"); + propStatusColorHsb->setValue("0,0,0,.5"); module->properties.add(propStatusColorHsb); onSetLevel([this](float l) { @@ -90,7 +90,7 @@ namespace Service { namespace API { namespace devices { if (o[2] > 0) { Switch::status = SWITCH_STATUS_ON; - Switch::onLevel = o[2]; + Dimmer::onLevel = Switch::onLevel = o[2]; } else { Switch::status = SWITCH_STATUS_OFF; } diff --git a/src/service/api/devices/ColorLight.h b/src/service/api/devices/ColorLight.h index 93cd935..c64475d 100644 --- a/src/service/api/devices/ColorLight.h +++ b/src/service/api/devices/ColorLight.h @@ -35,15 +35,10 @@ namespace Service { namespace API { namespace devices { class LightColor { public: - unsigned long duration = 0; void setColor(float hue, float saturation, float value, unsigned long transitionMs) { - oh = h; - os = s; - ov = v; - - //oh = getHue(); - //os = getSaturation(); - //ov = getValue(); + oh = getHue(); + os = getSaturation(); + ov = getValue(); // constraints if (hue > 1) hue = 1; @@ -55,6 +50,11 @@ namespace Service { namespace API { namespace devices { v = value; duration = transitionMs; + if (duration == 0) { + oh = h; + os = s; + ov = v; + } startTime = millis(); } bool isAnimating() const { @@ -73,6 +73,9 @@ namespace Service { namespace API { namespace devices { float getValue() const { return ov + ((v - ov) * getProgress()); } + void setValue(float val) { + v = val; + } float getRed() const { auto orgb = Utility::hsv2rgb(hueFix(oh), os, ov); auto crgb = Utility::hsv2rgb(hueFix(h), s, v); @@ -92,11 +95,10 @@ namespace Service { namespace API { namespace devices { return b; } private: - float h; - float s; - float v; + float h, s, v; float oh, os, ov; unsigned long startTime = -1; + unsigned long duration = 0; static float hueFix(float h) { return 1.325f - h; } diff --git a/src/service/api/devices/Dimmer.cpp b/src/service/api/devices/Dimmer.cpp index e628158..d2a7ef9 100644 --- a/src/service/api/devices/Dimmer.cpp +++ b/src/service/api/devices/Dimmer.cpp @@ -31,7 +31,7 @@ namespace Service { namespace API { namespace devices { setLoopInterval(10); // fixed transition frequency module->type = "Dimmer"; onSetStatus([this](SwitchStatus status) { - level.setLevel(status == SWITCH_STATUS_ON ? 1 : 0, defaultTransitionMs); + level.setLevel(status == SWITCH_STATUS_ON ? onLevel : 0, defaultTransitionMs); }); } diff --git a/src/service/api/devices/Dimmer.h b/src/service/api/devices/Dimmer.h index ec57e59..c6cc8a6 100644 --- a/src/service/api/devices/Dimmer.h +++ b/src/service/api/devices/Dimmer.h @@ -72,9 +72,9 @@ namespace Service { namespace API { namespace devices { setLevelCallback = std::move(callback); } private: - DimmerLevel level; std::function setLevelCallback = nullptr; protected: + DimmerLevel level; const unsigned long defaultTransitionMs = 500; }; diff --git a/src/service/api/devices/Switch.cpp b/src/service/api/devices/Switch.cpp index 14b6030..2959c5a 100644 --- a/src/service/api/devices/Switch.cpp +++ b/src/service/api/devices/Switch.cpp @@ -43,6 +43,8 @@ namespace Service { namespace API { namespace devices { void Switch::init() { } + void Switch::loop() { + } bool Switch::handleRequest(APIRequest *command, ResponseCallback* responseCallback) { auto m = getModule(command->Domain.c_str(), command->Address.c_str()); diff --git a/src/service/api/devices/Switch.h b/src/service/api/devices/Switch.h index b45f33f..52a9234 100644 --- a/src/service/api/devices/Switch.h +++ b/src/service/api/devices/Switch.h @@ -40,6 +40,7 @@ namespace Service { namespace API { namespace devices { public: Switch(const char* domain, const char* address, const char* name); void init() override; + void loop() override; bool canHandleDomain(String* domain) override; bool handleRequest(APIRequest*, ResponseCallback*) override; bool handleEvent(IIOEventSender*, diff --git a/src/version.h b/src/version.h index d233166..57ea5d2 100644 --- a/src/version.h +++ b/src/version.h @@ -29,7 +29,7 @@ #define VERSION_MAJOR 1 #define VERSION_MINOR 2 -#define VERSION_PATCH 28 +#define VERSION_PATCH 29 #define STRING_VALUE(...) STRING_VALUE__(__VA_ARGS__) #define STRING_VALUE__(...) #__VA_ARGS__