From 0862859f93e38b83a9f0d8ba52b67e2eb0598cae Mon Sep 17 00:00:00 2001 From: Andy Hofmann Date: Fri, 9 Jul 2021 20:25:35 +0200 Subject: [PATCH] New usermod: Support for RGB Rotary Encoder Board (#2068) * Added RGB Rotary Encoder usermod v1 * RGB rotary encoder UM: Readme; Added example video * RGB rotary encoder UM: Readme; Added example video * RGB rotary encoder UM: Fixed getJsonValue usage * RGB rotary encoder UM: Removed spaces in JSON keys * RGB rotary encoder UM: Cleanup readFromConfig * RGB rotary encoder UM: Cleaned up type usages * RGB rotary encoder UM: Fixed crash on re-enable --- usermods/rgb-rotary-encoder/readme.md | 86 +++++ .../rgb-rotary-encoder/rgb-rotary-encoder.h | 344 ++++++++++++++++++ wled00/usermods_list.cpp | 8 + 3 files changed, 438 insertions(+) create mode 100644 usermods/rgb-rotary-encoder/readme.md create mode 100644 usermods/rgb-rotary-encoder/rgb-rotary-encoder.h diff --git a/usermods/rgb-rotary-encoder/readme.md b/usermods/rgb-rotary-encoder/readme.md new file mode 100644 index 0000000000..2bf0ecb72a --- /dev/null +++ b/usermods/rgb-rotary-encoder/readme.md @@ -0,0 +1,86 @@ +# RGB Encoder Board + +This usermod-v2 adds support for the awesome RGB Rotary Encoder Board by Adam Zeloof / "Isotope Engineering" to control the overall brightness of your WLED instance: https://github.com/isotope-engineering/RGB-Encoder-Board. A great DIY rotary encoder with 20 tiny SK6805 / "NeoPixel Nano" LEDs. + +https://user-images.githubusercontent.com/3090131/124680599-0180ab80-dec7-11eb-9065-a6d08ebe0287.mp4 + +## Credits +The actual / original code that does the different LED modes is from Adam Zeloof. So I don't take credit for these. But I ported it to WLED, which involved replacing the LED library he used (because, guess what, WLED already has one; so no need to add another one, but use whatever WLED uses), plus the rotary encoder library, because that one was not compatible with ESP, only Arduino. +So it was quite more work than I hoped, but I got there eventually :) + +## Requirements +* "ESP Rotary" by Lennart Hennigs, v1.5.0 or higher: https://github.com/LennartHennigs/ESPRotary + +## Usermod installation +Simply copy the below block (build task) to your `platformio_override.ini` and compile WLED using this new build task. Or use an existing one and add the buildflag `-D RGB_ROTARY_ENCODER`. + +ESP32: +``` +[env:custom_esp32dev_usermod_rgb_encoder_board] +extends = env:esp32dev +build_flags = ${common.build_flags_esp32} -D WLED_RELEASE_NAME=ESP32 -D RGB_ROTARY_ENCODER +lib_deps = ${esp32.lib_deps} + lennarthennigs/ESP Rotary@^1.5.0 +``` + +ESP8266 / D1 Mini: +``` +[env:custom_d1_mini_usermod_rgb_encoder_board] +extends = env:d1_mini +build_flags = ${common.build_flags_esp8266} -D RGB_ROTARY_ENCODER +lib_deps = ${esp8266.lib_deps} + lennarthennigs/ESP Rotary@^1.5.0 +``` + +## How to connect the board to your ESP +We gonna need (minimum) three or (maximum) four GPIOs for the board: +* "ea": Basically tells if the encoder goes into one or the other direction +* "eb": Same thing, but the other direction +* "di": LED data in. To actually control the LEDs +* *(optional)* "sw": The integrated switch in the rotary encoder. Can be omitted for the bare functionality of just controlling the brightness + +We also gonna need some power, so: + +* "vdd": Needs to be connected to **+5V**. +* "gnd": Well, it's GND. + +You can freely pick the GPIOs, it doesn't matter. Those will be configured in the "Usermods" section in the WLED web panel: + +## Configuration +Navigate to the "Config" and then to the "Usermods" section. If you compiled WLED with `-D RGB_ROTARY_ENCODER`, you will see the config for it there. The settings there are the GPIOs we mentioned before (*Note: The switch pin is not there, as this can just be configured the "normal" button on the "LED Preferences" page*), plus a few more: +* LED pin: + * Possible values: Any valid and available GPIO + * Default: 3 + * What it does: Pin to control the LED ring +* ea pin: + * Possible values: Any valid and available GPIO + * Default: 15 + * What it does: First of the two rotary encoder pins +* eb pin: + * Possible values: Any valid and available GPIO + * Default: 32 + * What it does: Second of the two rotary encoder pins +* LED Mode: + * Possible values: 1-3 + * Default: 3 + * What it does: The usermod provides three different modes of how the LEDs can look like. Here's an example: https://github.com/isotope-engineering/RGB-Encoder-Board/blob/master/images/rgb-encoder-animations.gif + * Up left is "1" + * Up right is not supported / doesn't make sense for brightness control + * Bottom left is "2" + * Bottom right is "3" +* LED Brightness: + * Possible values: 1-255 + * Default: 64 + * What it does: Brightness of the LED ring +* Steps per click: + * Possible values: Any positive number + * Default: 4 + * What it does: With each "click", a rotary encoder actually increments it's "steps". Most rotary encoder do four "steps" per "click". I know this sounds super weird, so just leave this the default value, unless your rotary encoder behaves weirdly, like with one click, it makes two LEDs light up, or you sometimes need two click for one LED. Then you should play around with this value or write a small sketch using the same "ESP Rotary" library and read out the steps it does. +* Increment per click: + * Possible values: Any positive number + * Default: 5 + * What it does: Most rotary encoder have 20 "clicks", so basically 20 positions. This value should be set to 100 / `number of clicks` + +## Change log +2021-07 +* First implementation. diff --git a/usermods/rgb-rotary-encoder/rgb-rotary-encoder.h b/usermods/rgb-rotary-encoder/rgb-rotary-encoder.h new file mode 100644 index 0000000000..ab680a8f3d --- /dev/null +++ b/usermods/rgb-rotary-encoder/rgb-rotary-encoder.h @@ -0,0 +1,344 @@ +#pragma once + +#include "ESPRotary.h" +#include +#include "wled.h" + +class RgbRotaryEncoderUsermod : public Usermod +{ + private: + bool enabled = false; + bool initDone = false; + bool isDirty = false; + BusDigital *ledBus; + /* + * Green - eb - Q4 - 32 + * Red - ea - Q1 - 15 + * Black - sw - Q2 - 12 + */ + ESPRotary *rotaryEncoder; + int8_t ledIo = 3; // GPIO to control the LEDs + int8_t eaIo = 15; // "ea" from RGB Encoder Board + int8_t ebIo = 32; // "eb" from RGB Encoder Board + byte stepsPerClick = 4; // How many "steps" your rotary encoder does per click. This varies per rotary encoder + /* This could vary per rotary encoder: Usually rotary encoders have 20 "clicks". + If yours has less/more, adjust this to: 100% = 20 LEDs * incrementPerClick */ + byte incrementPerClick = 5; + byte ledMode = 3; + byte ledBrightness = 64; + + // This is all needed to calculate the brightness, rotary position, etc. + const byte minPos = 5; // minPos is not zero, because if we want to turn the LEDs off, we use the built-in button ;) + const byte maxPos = 100; // maxPos=100, like 100% + const byte numLeds = 20; + byte lastKnownPos = 0; + + byte currentColors[3]; + byte lastKnownBri = 0; + + + void initRotaryEncoder() + { + if (!pinManager.allocatePin(eaIo, false)) { + eaIo = -1; + } + if (!pinManager.allocatePin(ebIo, false)) { + ebIo = -1; + } + if (eaIo == -1 || ebIo == -1) { + cleanup(); + return; + } + + // I don't know why, but setting the upper bound here does not work. It results into 1717922932 O_o + rotaryEncoder = new ESPRotary(eaIo, ebIo, stepsPerClick, incrementPerClick, maxPos, currentPos, incrementPerClick); + rotaryEncoder->setUpperBound(maxPos); // I have to again set it here and then it works / is actually 100... + + rotaryEncoder->setChangedHandler(RgbRotaryEncoderUsermod::cbRotate); + } + + void initLedBus() + { + byte _pins[5] = {(byte)ledIo, 255, 255, 255, 255}; + BusConfig busCfg = BusConfig(TYPE_WS2812_RGB, _pins, 0, numLeds, COL_ORDER_GRB, false, 0); + + ledBus = new BusDigital(busCfg, WLED_MAX_BUSSES - 1); + if (!ledBus->isOk()) { + cleanup(); + return; + } + + ledBus->setBrightness(ledBrightness); + } + + void updateLeds() + { + switch (ledMode) { + case 2: + { + currentColors[0] = 255; currentColors[1] = 0; currentColors[2] = 0; + for (int i = 0; i < currentPos / incrementPerClick - 1; i++) { + ledBus->setPixelColor(i, 0); + } + ledBus->setPixelColor(currentPos / incrementPerClick - 1, colorFromRgbw(currentColors)); + for (int i = currentPos / incrementPerClick; i < numLeds; i++) { + ledBus->setPixelColor(i, 0); + } + } + break; + + default: + case 1: + case 3: + // WLED orange (of course), which we will use in mode 1 + currentColors[0] = 255; currentColors[1] = 160; currentColors[2] = 0; + for (int i = 0; i < currentPos / incrementPerClick; i++) { + if (ledMode == 3) { + hsv2rgb((i) / float(numLeds), 1, .25); + } + ledBus->setPixelColor(i, colorFromRgbw(currentColors)); + } + for (int i = currentPos / incrementPerClick; i < numLeds; i++) { + ledBus->setPixelColor(i, 0); + } + break; + } + + isDirty = true; + } + + void cleanup() + { + // Only deallocate pins if we allocated them ;) + if (eaIo != -1) { + pinManager.deallocatePin(eaIo); + } + if (ebIo != -1) { + pinManager.deallocatePin(ebIo); + } + + delete rotaryEncoder; + delete ledBus; + + enabled = false; + } + + int getPositionForBrightness() + { + return int(((float)bri / (float)255) * 100); + } + + float fract(float x) { return x - int(x); } + + float mix(float a, float b, float t) { return a + (b - a) * t; } + + void hsv2rgb(float h, float s, float v) { + currentColors[0] = int((v * mix(1.0, constrain(abs(fract(h + 1.0) * 6.0 - 3.0) - 1.0, 0.0, 1.0), s)) * 255); + currentColors[1] = int((v * mix(1.0, constrain(abs(fract(h + 0.6666666) * 6.0 - 3.0) - 1.0, 0.0, 1.0), s)) * 255); + currentColors[2] = int((v * mix(1.0, constrain(abs(fract(h + 0.3333333) * 6.0 - 3.0) - 1.0, 0.0, 1.0), s)) * 255); + } + + public: + static byte currentPos; + + // strings to reduce flash memory usage (used more than twice) + static const char _name[]; + static const char _enabled[]; + static const char _ledIo[]; + static const char _eaIo[]; + static const char _ebIo[]; + static const char _ledMode[]; + static const char _ledBrightness[]; + static const char _stepsPerClick[]; + static const char _incrementPerClick[]; + + + static void cbRotate(ESPRotary& r) { + currentPos = r.getPosition(); + } + + /** + * Enable/Disable the usermod + */ + // inline void enable(bool enable) { enabled = enable; } + /** + * Get usermod enabled/disabled state + */ + // inline bool isEnabled() { return enabled; } + + /* + * setup() is called once at boot. WiFi is not yet connected at this point. + * You can use it to initialize variables, sensors or similar. + */ + void setup() + { + if (enabled) { + currentPos = getPositionForBrightness(); + lastKnownBri = bri; + + initRotaryEncoder(); + initLedBus(); + + // No updating of LEDs here, as that's sometimes not working; loop() will take care of that + + initDone = true; + } + } + + /* + * loop() is called continuously. Here you can check for events, read sensors, etc. + * + * Tips: + * 1. You can use "if (WLED_CONNECTED)" to check for a successful network connection. + * Additionally, "if (WLED_MQTT_CONNECTED)" is available to check for a connection to an MQTT broker. + * + * 2. Try to avoid using the delay() function. NEVER use delays longer than 10 milliseconds. + * Instead, use a timer check as shown here. + */ + void loop() + { + if (!enabled || strip.isUpdating()) return; + + rotaryEncoder->loop(); + + // If the rotary was changed + if(lastKnownPos != currentPos) { + lastKnownPos = currentPos; + + bri = min(int(round((2.55 * currentPos))), 255); + lastKnownBri = bri; + + updateLeds(); + colorUpdated(NOTIFIER_CALL_MODE_DIRECT_CHANGE); + } + + // If the brightness is changed not with the rotary, update the rotary + if (bri != lastKnownBri) { + currentPos = lastKnownPos = getPositionForBrightness(); + lastKnownBri = bri; + rotaryEncoder->resetPosition(currentPos); + updateLeds(); + } + + // Update LEDs here in loop to also validate that we can update/show + if (isDirty && ledBus->canShow()) { + isDirty = false; + ledBus->show(); + } + } + + void addToConfig(JsonObject &root) + { + JsonObject top = root.createNestedObject(FPSTR(_name)); // usermodname + + top[FPSTR(_enabled)] = enabled; + top[FPSTR(_ledIo)] = ledIo; + top[FPSTR(_eaIo)] = eaIo; + top[FPSTR(_ebIo)] = ebIo; + top[FPSTR(_ledMode)] = ledMode; + top[FPSTR(_ledBrightness)] = ledBrightness; + top[FPSTR(_stepsPerClick)] = stepsPerClick; + top[FPSTR(_incrementPerClick)] = incrementPerClick; + } + + /** + * readFromConfig() is called before setup() to populate properties from values stored in cfg.json + * + * The function should return true if configuration was successfully loaded or false if there was no configuration. + */ + bool readFromConfig(JsonObject &root) + { + JsonObject top = root[FPSTR(_name)]; + if (top.isNull()) { + DEBUG_PRINTF("[%s] No config found. (Using defaults.)\n", _name); + return false; + } + + bool oldEnabled = enabled; + int8_t oldLedIo = ledIo; + int8_t oldEaIo = eaIo; + int8_t oldEbIo = ebIo; + byte oldLedMode = ledMode; + byte oldStepsPerClick = stepsPerClick; + byte oldIncrementPerClick = incrementPerClick; + byte oldLedBrightness = ledBrightness; + + getJsonValue(top[FPSTR(_enabled)], enabled); + getJsonValue(top[FPSTR(_ledIo)], ledIo); + getJsonValue(top[FPSTR(_eaIo)], eaIo); + getJsonValue(top[FPSTR(_ebIo)], ebIo); + getJsonValue(top[FPSTR(_stepsPerClick)], stepsPerClick); + getJsonValue(top[FPSTR(_incrementPerClick)], incrementPerClick); + ledMode = top[FPSTR(_ledMode)] > 0 && top[FPSTR(_ledMode)] < 4 ? top[FPSTR(_ledMode)] : ledMode; + ledBrightness = top[FPSTR(_ledBrightness)] > 0 && top[FPSTR(_ledBrightness)] <= 255 ? top[FPSTR(_ledBrightness)] : ledBrightness; + + if (!initDone) { + // First run: reading from cfg.json + // Nothing to do here, will be all done in setup() + } + // Mod was disabled, so run setup() + else if (enabled && enabled != oldEnabled) { + DEBUG_PRINTF("[%s] Usermod has been re-enabled\n", _name); + setup(); + } + // Config has been changed, so adopt to changes + else { + if (!enabled) { + DEBUG_PRINTF("[%s] Usermod has been disabled\n", _name); + cleanup(); + } + else { + DEBUG_PRINTF("[%s] Usermod is enabled\n", _name); + if (ledIo != oldLedIo) { + delete ledBus; + initLedBus(); + } + + if (ledBrightness != oldLedBrightness) { + ledBus->setBrightness(ledBrightness); + isDirty = true; + } + + if (ledMode != oldLedMode) { + updateLeds(); + } + + if (eaIo != oldEaIo || ebIo != oldEbIo || stepsPerClick != oldStepsPerClick || incrementPerClick != oldIncrementPerClick) { + pinManager.deallocatePin(oldEaIo); + pinManager.deallocatePin(oldEbIo); + + delete rotaryEncoder; + initRotaryEncoder(); + } + } + + DEBUG_PRINTF("[%s] Config (re)loaded\n", _name); + } + + return true; + } + + /* + * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). + * This could be used in the future for the system to determine whether your usermod is installed. + */ + uint16_t getId() + { + return 0x4711; + } + + //More methods can be added in the future, this example will then be extended. + //Your usermod will remain compatible as it does not need to implement all methods from the Usermod base class! +}; + +byte RgbRotaryEncoderUsermod::currentPos = 5; +// strings to reduce flash memory usage (used more than twice) +const char RgbRotaryEncoderUsermod::_name[] PROGMEM = "RGB-Rotary-Encoder"; +const char RgbRotaryEncoderUsermod::_enabled[] PROGMEM = "Enabled"; +const char RgbRotaryEncoderUsermod::_ledIo[] PROGMEM = "LED-pin"; +const char RgbRotaryEncoderUsermod::_eaIo[] PROGMEM = "ea-pin"; +const char RgbRotaryEncoderUsermod::_ebIo[] PROGMEM = "eb-pin"; +const char RgbRotaryEncoderUsermod::_ledMode[] PROGMEM = "LED-Mode"; +const char RgbRotaryEncoderUsermod::_ledBrightness[] PROGMEM = "LED-Brightness"; +const char RgbRotaryEncoderUsermod::_stepsPerClick[] PROGMEM = "Steps-per-Click"; +const char RgbRotaryEncoderUsermod::_incrementPerClick[] PROGMEM = "Increment-per-Click"; \ No newline at end of file diff --git a/wled00/usermods_list.cpp b/wled00/usermods_list.cpp index 5669f2f30d..fb68d62e90 100644 --- a/wled00/usermods_list.cpp +++ b/wled00/usermods_list.cpp @@ -78,6 +78,10 @@ #include "../usermods/usermod_rotary_brightness_color/usermod_rotary_brightness_color.h" #endif +#ifdef RGB_ROTARY_ENCODER +#include "../usermods/rgb-rotary-encoder/rgb-rotary-encoder.h" +#endif + void registerUsermods() { /* @@ -151,4 +155,8 @@ void registerUsermods() #ifdef USERMOD_ROTARY_ENCODER_BRIGHTNESS_COLOR usermods.add(new RotaryEncoderBrightnessColor()); #endif + + #ifdef RGB_ROTARY_ENCODER + usermods.add(new RgbRotaryEncoderUsermod()); + #endif }