Merge branch '0_15' into random-bg-img
This commit is contained in:
commit
4204393337
6
.vscode/tasks.json
vendored
6
.vscode/tasks.json
vendored
@ -9,8 +9,8 @@
|
|||||||
],
|
],
|
||||||
"dependsOrder": "sequence",
|
"dependsOrder": "sequence",
|
||||||
"problemMatcher": [
|
"problemMatcher": [
|
||||||
"$platformio",
|
"$platformio"
|
||||||
],
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": "PlatformIO",
|
"type": "PlatformIO",
|
||||||
@ -18,7 +18,7 @@
|
|||||||
"task": "Build",
|
"task": "Build",
|
||||||
"group": {
|
"group": {
|
||||||
"kind": "build",
|
"kind": "build",
|
||||||
"isDefault": true,
|
"isDefault": true
|
||||||
},
|
},
|
||||||
"problemMatcher": [
|
"problemMatcher": [
|
||||||
"$platformio"
|
"$platformio"
|
||||||
|
2
package-lock.json
generated
2
package-lock.json
generated
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "wled",
|
"name": "wled",
|
||||||
"version": "0.14.0",
|
"version": "0.15.0-a0",
|
||||||
"lockfileVersion": 1,
|
"lockfileVersion": 1,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "wled",
|
"name": "wled",
|
||||||
"version": "0.14.0",
|
"version": "0.15.0-a0",
|
||||||
"description": "Tools for WLED project",
|
"description": "Tools for WLED project",
|
||||||
"main": "tools/cdata.js",
|
"main": "tools/cdata.js",
|
||||||
"directories": {
|
"directories": {
|
||||||
|
@ -181,6 +181,9 @@ lib_deps =
|
|||||||
IRremoteESP8266 @ 2.8.2
|
IRremoteESP8266 @ 2.8.2
|
||||||
makuna/NeoPixelBus @ 2.7.5
|
makuna/NeoPixelBus @ 2.7.5
|
||||||
https://github.com/Aircoookie/ESPAsyncWebServer.git @ ~2.0.7
|
https://github.com/Aircoookie/ESPAsyncWebServer.git @ ~2.0.7
|
||||||
|
# ESP-NOW library (includes mandatory QuickDebug library)
|
||||||
|
; gmag11/QuickESPNow @ 0.6.2
|
||||||
|
https://github.com/blazoncek/QuickESPNow.git#optional-debug
|
||||||
#For use of the TTGO T-Display ESP32 Module with integrated TFT display uncomment the following line
|
#For use of the TTGO T-Display ESP32 Module with integrated TFT display uncomment the following line
|
||||||
#TFT_eSPI
|
#TFT_eSPI
|
||||||
#For compatible OLED display uncomment following
|
#For compatible OLED display uncomment following
|
||||||
@ -217,7 +220,8 @@ build_flags =
|
|||||||
; restrict to minimal mime-types
|
; restrict to minimal mime-types
|
||||||
-DMIMETYPE_MINIMAL
|
-DMIMETYPE_MINIMAL
|
||||||
; other special-purpose framework flags (see https://docs.platformio.org/en/latest/platforms/espressif8266.html)
|
; other special-purpose framework flags (see https://docs.platformio.org/en/latest/platforms/espressif8266.html)
|
||||||
; -D PIO_FRAMEWORK_ARDUINO_MMU_CACHE16_IRAM48 ;; in case of linker errors like "section `.text1' will not fit in region `iram1_0_seg'"
|
; decrease code cache size and increase IRAM to fit all pixel functions
|
||||||
|
-D PIO_FRAMEWORK_ARDUINO_MMU_CACHE16_IRAM48 ;; in case of linker errors like "section `.text1' will not fit in region `iram1_0_seg'"
|
||||||
; -D PIO_FRAMEWORK_ARDUINO_MMU_CACHE16_IRAM48_SECHEAP_SHARED ;; (experimental) adds some extra heap, but may cause slowdown
|
; -D PIO_FRAMEWORK_ARDUINO_MMU_CACHE16_IRAM48_SECHEAP_SHARED ;; (experimental) adds some extra heap, but may cause slowdown
|
||||||
|
|
||||||
lib_deps =
|
lib_deps =
|
||||||
|
@ -52,9 +52,15 @@ class PWMFanUsermod : public Usermod {
|
|||||||
uint8_t tachoUpdateSec = 30;
|
uint8_t tachoUpdateSec = 30;
|
||||||
float targetTemperature = 35.0;
|
float targetTemperature = 35.0;
|
||||||
uint8_t minPWMValuePct = 0;
|
uint8_t minPWMValuePct = 0;
|
||||||
|
uint8_t maxPWMValuePct = 100;
|
||||||
uint8_t numberOfInterrupsInOneSingleRotation = 2; // Number of interrupts ESP32 sees on tacho signal on a single fan rotation. All the fans I've seen trigger two interrups.
|
uint8_t numberOfInterrupsInOneSingleRotation = 2; // Number of interrupts ESP32 sees on tacho signal on a single fan rotation. All the fans I've seen trigger two interrups.
|
||||||
uint8_t pwmValuePct = 0;
|
uint8_t pwmValuePct = 0;
|
||||||
|
|
||||||
|
// constant values
|
||||||
|
static const uint8_t _pwmMaxValue = 255;
|
||||||
|
static const uint8_t _pwmMaxStepCount = 7;
|
||||||
|
float _pwmTempStepSize = 0.5f;
|
||||||
|
|
||||||
// strings to reduce flash memory usage (used more than twice)
|
// strings to reduce flash memory usage (used more than twice)
|
||||||
static const char _name[];
|
static const char _name[];
|
||||||
static const char _enabled[];
|
static const char _enabled[];
|
||||||
@ -63,6 +69,7 @@ class PWMFanUsermod : public Usermod {
|
|||||||
static const char _temperature[];
|
static const char _temperature[];
|
||||||
static const char _tachoUpdateSec[];
|
static const char _tachoUpdateSec[];
|
||||||
static const char _minPWMValuePct[];
|
static const char _minPWMValuePct[];
|
||||||
|
static const char _maxPWMValuePct[];
|
||||||
static const char _IRQperRotation[];
|
static const char _IRQperRotation[];
|
||||||
static const char _speed[];
|
static const char _speed[];
|
||||||
static const char _lock[];
|
static const char _lock[];
|
||||||
@ -156,31 +163,25 @@ class PWMFanUsermod : public Usermod {
|
|||||||
|
|
||||||
void setFanPWMbasedOnTemperature(void) {
|
void setFanPWMbasedOnTemperature(void) {
|
||||||
float temp = getActualTemperature();
|
float temp = getActualTemperature();
|
||||||
float difftemp = temp - targetTemperature;
|
// dividing minPercent and maxPercent into equal pwmvalue sizes
|
||||||
// Default to run fan at full speed.
|
int pwmStepSize = ((maxPWMValuePct - minPWMValuePct) * _pwmMaxValue) / (_pwmMaxStepCount*100);
|
||||||
int newPWMvalue = 255;
|
int pwmStep = calculatePwmStep(temp - targetTemperature);
|
||||||
int pwmStep = ((100 - minPWMValuePct) * newPWMvalue) / (7*100);
|
// minimum based on full speed - not entered MaxPercent
|
||||||
int pwmMinimumValue = (minPWMValuePct * newPWMvalue) / 100;
|
int pwmMinimumValue = (minPWMValuePct * _pwmMaxValue) / 100;
|
||||||
|
updateFanSpeed(pwmMinimumValue + pwmStep*pwmStepSize);
|
||||||
|
}
|
||||||
|
|
||||||
if ((temp == NAN) || (temp <= -100.0)) {
|
uint8_t calculatePwmStep(float diffTemp){
|
||||||
|
if ((diffTemp == NAN) || (diffTemp <= -100.0)) {
|
||||||
DEBUG_PRINTLN(F("WARNING: no temperature value available. Cannot do temperature control. Will set PWM fan to 255."));
|
DEBUG_PRINTLN(F("WARNING: no temperature value available. Cannot do temperature control. Will set PWM fan to 255."));
|
||||||
} else if (difftemp <= 0.0) {
|
return _pwmMaxStepCount;
|
||||||
// Temperature is below target temperature. Run fan at minimum speed.
|
|
||||||
newPWMvalue = pwmMinimumValue;
|
|
||||||
} else if (difftemp <= 0.5) {
|
|
||||||
newPWMvalue = pwmMinimumValue + pwmStep;
|
|
||||||
} else if (difftemp <= 1.0) {
|
|
||||||
newPWMvalue = pwmMinimumValue + 2*pwmStep;
|
|
||||||
} else if (difftemp <= 1.5) {
|
|
||||||
newPWMvalue = pwmMinimumValue + 3*pwmStep;
|
|
||||||
} else if (difftemp <= 2.0) {
|
|
||||||
newPWMvalue = pwmMinimumValue + 4*pwmStep;
|
|
||||||
} else if (difftemp <= 2.5) {
|
|
||||||
newPWMvalue = pwmMinimumValue + 5*pwmStep;
|
|
||||||
} else if (difftemp <= 3.0) {
|
|
||||||
newPWMvalue = pwmMinimumValue + 6*pwmStep;
|
|
||||||
}
|
}
|
||||||
updateFanSpeed(newPWMvalue);
|
if(diffTemp <=0){
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
int calculatedStep = (diffTemp / _pwmTempStepSize)+1;
|
||||||
|
// anything greater than max stepcount gets max
|
||||||
|
return (uint8_t)min((int)_pwmMaxStepCount,calculatedStep);
|
||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
@ -312,6 +313,7 @@ class PWMFanUsermod : public Usermod {
|
|||||||
top[FPSTR(_tachoUpdateSec)] = tachoUpdateSec;
|
top[FPSTR(_tachoUpdateSec)] = tachoUpdateSec;
|
||||||
top[FPSTR(_temperature)] = targetTemperature;
|
top[FPSTR(_temperature)] = targetTemperature;
|
||||||
top[FPSTR(_minPWMValuePct)] = minPWMValuePct;
|
top[FPSTR(_minPWMValuePct)] = minPWMValuePct;
|
||||||
|
top[FPSTR(_maxPWMValuePct)] = maxPWMValuePct;
|
||||||
top[FPSTR(_IRQperRotation)] = numberOfInterrupsInOneSingleRotation;
|
top[FPSTR(_IRQperRotation)] = numberOfInterrupsInOneSingleRotation;
|
||||||
DEBUG_PRINTLN(F("Autosave config saved."));
|
DEBUG_PRINTLN(F("Autosave config saved."));
|
||||||
}
|
}
|
||||||
@ -345,6 +347,8 @@ class PWMFanUsermod : public Usermod {
|
|||||||
targetTemperature = top[FPSTR(_temperature)] | targetTemperature;
|
targetTemperature = top[FPSTR(_temperature)] | targetTemperature;
|
||||||
minPWMValuePct = top[FPSTR(_minPWMValuePct)] | minPWMValuePct;
|
minPWMValuePct = top[FPSTR(_minPWMValuePct)] | minPWMValuePct;
|
||||||
minPWMValuePct = (uint8_t) min(100,max(0,(int)minPWMValuePct)); // bounds checking
|
minPWMValuePct = (uint8_t) min(100,max(0,(int)minPWMValuePct)); // bounds checking
|
||||||
|
maxPWMValuePct = top[FPSTR(_maxPWMValuePct)] | maxPWMValuePct;
|
||||||
|
maxPWMValuePct = (uint8_t) min(100,max((int)minPWMValuePct,(int)maxPWMValuePct)); // bounds checking
|
||||||
numberOfInterrupsInOneSingleRotation = top[FPSTR(_IRQperRotation)] | numberOfInterrupsInOneSingleRotation;
|
numberOfInterrupsInOneSingleRotation = top[FPSTR(_IRQperRotation)] | numberOfInterrupsInOneSingleRotation;
|
||||||
numberOfInterrupsInOneSingleRotation = (uint8_t) max(1,(int)numberOfInterrupsInOneSingleRotation); // bounds checking
|
numberOfInterrupsInOneSingleRotation = (uint8_t) max(1,(int)numberOfInterrupsInOneSingleRotation); // bounds checking
|
||||||
|
|
||||||
@ -389,6 +393,7 @@ const char PWMFanUsermod::_pwmPin[] PROGMEM = "PWM-pin";
|
|||||||
const char PWMFanUsermod::_temperature[] PROGMEM = "target-temp-C";
|
const char PWMFanUsermod::_temperature[] PROGMEM = "target-temp-C";
|
||||||
const char PWMFanUsermod::_tachoUpdateSec[] PROGMEM = "tacho-update-s";
|
const char PWMFanUsermod::_tachoUpdateSec[] PROGMEM = "tacho-update-s";
|
||||||
const char PWMFanUsermod::_minPWMValuePct[] PROGMEM = "min-PWM-percent";
|
const char PWMFanUsermod::_minPWMValuePct[] PROGMEM = "min-PWM-percent";
|
||||||
|
const char PWMFanUsermod::_maxPWMValuePct[] PROGMEM = "max-PWM-percent";
|
||||||
const char PWMFanUsermod::_IRQperRotation[] PROGMEM = "IRQs-per-rotation";
|
const char PWMFanUsermod::_IRQperRotation[] PROGMEM = "IRQs-per-rotation";
|
||||||
const char PWMFanUsermod::_speed[] PROGMEM = "speed";
|
const char PWMFanUsermod::_speed[] PROGMEM = "speed";
|
||||||
const char PWMFanUsermod::_lock[] PROGMEM = "lock";
|
const char PWMFanUsermod::_lock[] PROGMEM = "lock";
|
||||||
|
@ -51,6 +51,8 @@
|
|||||||
#define PLOT_PRINTF(x...)
|
#define PLOT_PRINTF(x...)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#define MAX_PALETTES 3
|
||||||
|
|
||||||
// use audio source class (ESP32 specific)
|
// use audio source class (ESP32 specific)
|
||||||
#include "audio_source.h"
|
#include "audio_source.h"
|
||||||
constexpr i2s_port_t I2S_PORT = I2S_NUM_0; // I2S port to use (do not change !)
|
constexpr i2s_port_t I2S_PORT = I2S_NUM_0; // I2S port to use (do not change !)
|
||||||
@ -614,6 +616,8 @@ class AudioReactive : public Usermod {
|
|||||||
// set your config variables to their boot default value (this can also be done in readFromConfig() or a constructor if you prefer)
|
// set your config variables to their boot default value (this can also be done in readFromConfig() or a constructor if you prefer)
|
||||||
bool enabled = false;
|
bool enabled = false;
|
||||||
bool initDone = false;
|
bool initDone = false;
|
||||||
|
bool addPalettes = false;
|
||||||
|
int8_t palettes = 0;
|
||||||
|
|
||||||
// variables for UDP sound sync
|
// variables for UDP sound sync
|
||||||
WiFiUDP fftUdp; // UDP object for sound sync (from WiFi UDP, not Async UDP!)
|
WiFiUDP fftUdp; // UDP object for sound sync (from WiFi UDP, not Async UDP!)
|
||||||
@ -652,10 +656,15 @@ class AudioReactive : public Usermod {
|
|||||||
static const char _inputLvl[];
|
static const char _inputLvl[];
|
||||||
static const char _analogmic[];
|
static const char _analogmic[];
|
||||||
static const char _digitalmic[];
|
static const char _digitalmic[];
|
||||||
|
static const char _addPalettes[];
|
||||||
static const char UDP_SYNC_HEADER[];
|
static const char UDP_SYNC_HEADER[];
|
||||||
static const char UDP_SYNC_HEADER_v1[];
|
static const char UDP_SYNC_HEADER_v1[];
|
||||||
|
|
||||||
// private methods
|
// private methods
|
||||||
|
void removeAudioPalettes(void);
|
||||||
|
void createAudioPalettes(void);
|
||||||
|
CRGB getCRGBForBand(int x, int pal);
|
||||||
|
void fillAudioPalette(int pal);
|
||||||
|
|
||||||
////////////////////
|
////////////////////
|
||||||
// Debug support //
|
// Debug support //
|
||||||
@ -1199,6 +1208,7 @@ class AudioReactive : public Usermod {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (enabled) connectUDPSoundSync();
|
if (enabled) connectUDPSoundSync();
|
||||||
|
if (enabled && addPalettes) createAudioPalettes();
|
||||||
initDone = true;
|
initDone = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1361,6 +1371,7 @@ class AudioReactive : public Usermod {
|
|||||||
lastTime = millis();
|
lastTime = millis();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (int i=0; i<MAX_PALETTES; i++) fillAudioPalette(i);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -1613,13 +1624,28 @@ class AudioReactive : public Usermod {
|
|||||||
if (usermod[FPSTR(_enabled)].is<bool>()) {
|
if (usermod[FPSTR(_enabled)].is<bool>()) {
|
||||||
enabled = usermod[FPSTR(_enabled)].as<bool>();
|
enabled = usermod[FPSTR(_enabled)].as<bool>();
|
||||||
if (prevEnabled != enabled) onUpdateBegin(!enabled);
|
if (prevEnabled != enabled) onUpdateBegin(!enabled);
|
||||||
|
if (addPalettes) {
|
||||||
|
// add/remove custom/audioreactive palettes
|
||||||
|
if (prevEnabled && !enabled) removeAudioPalettes();
|
||||||
|
if (!prevEnabled && enabled) createAudioPalettes();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (usermod[FPSTR(_inputLvl)].is<int>()) {
|
if (usermod[FPSTR(_inputLvl)].is<int>()) {
|
||||||
inputLevel = min(255,max(0,usermod[FPSTR(_inputLvl)].as<int>()));
|
inputLevel = min(255,max(0,usermod[FPSTR(_inputLvl)].as<int>()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (root.containsKey(F("rmcpal")) && root[F("rmcpal")].as<bool>()) {
|
||||||
|
// handle removal of custom palettes from JSON call so we don't break things
|
||||||
|
removeAudioPalettes();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void onStateChange(uint8_t callMode) {
|
||||||
|
if (initDone && enabled && addPalettes && palettes==0 && strip.customPalettes.size()<10) {
|
||||||
|
// if palettes were removed during JSON call re-add them
|
||||||
|
createAudioPalettes();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* addToConfig() can be used to add custom persistent settings to the cfg.json file in the "um" (usermod) object.
|
* addToConfig() can be used to add custom persistent settings to the cfg.json file in the "um" (usermod) object.
|
||||||
@ -1660,6 +1686,7 @@ class AudioReactive : public Usermod {
|
|||||||
{
|
{
|
||||||
JsonObject top = root.createNestedObject(FPSTR(_name));
|
JsonObject top = root.createNestedObject(FPSTR(_name));
|
||||||
top[FPSTR(_enabled)] = enabled;
|
top[FPSTR(_enabled)] = enabled;
|
||||||
|
top[FPSTR(_addPalettes)] = addPalettes;
|
||||||
|
|
||||||
#if !defined(CONFIG_IDF_TARGET_ESP32S2) && !defined(CONFIG_IDF_TARGET_ESP32C3) && !defined(CONFIG_IDF_TARGET_ESP32S3)
|
#if !defined(CONFIG_IDF_TARGET_ESP32S2) && !defined(CONFIG_IDF_TARGET_ESP32C3) && !defined(CONFIG_IDF_TARGET_ESP32S3)
|
||||||
JsonObject amic = top.createNestedObject(FPSTR(_analogmic));
|
JsonObject amic = top.createNestedObject(FPSTR(_analogmic));
|
||||||
@ -1712,8 +1739,11 @@ class AudioReactive : public Usermod {
|
|||||||
{
|
{
|
||||||
JsonObject top = root[FPSTR(_name)];
|
JsonObject top = root[FPSTR(_name)];
|
||||||
bool configComplete = !top.isNull();
|
bool configComplete = !top.isNull();
|
||||||
|
bool oldEnabled = enabled;
|
||||||
|
bool oldAddPalettes = addPalettes;
|
||||||
|
|
||||||
configComplete &= getJsonValue(top[FPSTR(_enabled)], enabled);
|
configComplete &= getJsonValue(top[FPSTR(_enabled)], enabled);
|
||||||
|
configComplete &= getJsonValue(top[FPSTR(_addPalettes)], addPalettes);
|
||||||
|
|
||||||
#if !defined(CONFIG_IDF_TARGET_ESP32S2) && !defined(CONFIG_IDF_TARGET_ESP32C3) && !defined(CONFIG_IDF_TARGET_ESP32S3)
|
#if !defined(CONFIG_IDF_TARGET_ESP32S2) && !defined(CONFIG_IDF_TARGET_ESP32C3) && !defined(CONFIG_IDF_TARGET_ESP32S3)
|
||||||
configComplete &= getJsonValue(top[FPSTR(_analogmic)]["pin"], audioPin);
|
configComplete &= getJsonValue(top[FPSTR(_analogmic)]["pin"], audioPin);
|
||||||
@ -1747,6 +1777,11 @@ class AudioReactive : public Usermod {
|
|||||||
configComplete &= getJsonValue(top["sync"][F("port")], audioSyncPort);
|
configComplete &= getJsonValue(top["sync"][F("port")], audioSyncPort);
|
||||||
configComplete &= getJsonValue(top["sync"][F("mode")], audioSyncEnabled);
|
configComplete &= getJsonValue(top["sync"][F("mode")], audioSyncEnabled);
|
||||||
|
|
||||||
|
if (initDone) {
|
||||||
|
// add/remove custom/audioreactive palettes
|
||||||
|
if ((oldAddPalettes && !addPalettes) || (oldAddPalettes && !enabled)) removeAudioPalettes();
|
||||||
|
if ((addPalettes && !oldAddPalettes && enabled) || (addPalettes && !oldEnabled && enabled)) createAudioPalettes();
|
||||||
|
} // else setup() will create palettes
|
||||||
return configComplete;
|
return configComplete;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1822,6 +1857,88 @@ class AudioReactive : public Usermod {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
void AudioReactive::removeAudioPalettes(void) {
|
||||||
|
DEBUG_PRINTLN(F("Removing audio palettes."));
|
||||||
|
while (palettes) {
|
||||||
|
strip.customPalettes.pop_back();
|
||||||
|
DEBUG_PRINTLN(palettes);
|
||||||
|
palettes--;
|
||||||
|
}
|
||||||
|
DEBUG_PRINT(F("Total # of palettes: ")); DEBUG_PRINTLN(strip.customPalettes.size());
|
||||||
|
}
|
||||||
|
|
||||||
|
void AudioReactive::createAudioPalettes(void) {
|
||||||
|
DEBUG_PRINT(F("Total # of palettes: ")); DEBUG_PRINTLN(strip.customPalettes.size());
|
||||||
|
DEBUG_PRINTLN(F("Adding audio palettes."));
|
||||||
|
for (int i=0; i<MAX_PALETTES; i++)
|
||||||
|
if (strip.customPalettes.size() < 10) {
|
||||||
|
strip.customPalettes.push_back(CRGBPalette16(CRGB(BLACK)));
|
||||||
|
palettes++;
|
||||||
|
DEBUG_PRINTLN(palettes);
|
||||||
|
} else break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// credit @netmindz ar palette, adapted for usermod @blazoncek
|
||||||
|
CRGB AudioReactive::getCRGBForBand(int x, int pal) {
|
||||||
|
CRGB value;
|
||||||
|
CHSV hsv;
|
||||||
|
int b;
|
||||||
|
switch (pal) {
|
||||||
|
case 2:
|
||||||
|
b = map(x, 0, 255, 0, NUM_GEQ_CHANNELS/2); // convert palette position to lower half of freq band
|
||||||
|
hsv = CHSV(fftResult[b], 255, x);
|
||||||
|
hsv2rgb_rainbow(hsv, value); // convert to R,G,B
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
b = map(x, 1, 255, 0, 10); // convert palette position to lower half of freq band
|
||||||
|
hsv = CHSV(fftResult[b], 255, map(fftResult[b], 0, 255, 30, 255)); // pick hue
|
||||||
|
hsv2rgb_rainbow(hsv, value); // convert to R,G,B
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (x == 1) {
|
||||||
|
value = CRGB(fftResult[10]/2, fftResult[4]/2, fftResult[0]/2);
|
||||||
|
} else if(x == 255) {
|
||||||
|
value = CRGB(fftResult[10]/2, fftResult[0]/2, fftResult[4]/2);
|
||||||
|
} else {
|
||||||
|
value = CRGB(fftResult[0]/2, fftResult[4]/2, fftResult[10]/2);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
void AudioReactive::fillAudioPalette(int pal) {
|
||||||
|
if (pal>=palettes) return; // palette does not exist
|
||||||
|
|
||||||
|
uint8_t tcp[16]; // Needs to be 4 times however many colors are being used.
|
||||||
|
// 3 colors = 12, 4 colors = 16, etc.
|
||||||
|
|
||||||
|
tcp[0] = 0; // anchor of first color - must be zero
|
||||||
|
tcp[1] = 0;
|
||||||
|
tcp[2] = 0;
|
||||||
|
tcp[3] = 0;
|
||||||
|
|
||||||
|
CRGB rgb = getCRGBForBand(1, pal);
|
||||||
|
tcp[4] = 1; // anchor of first color
|
||||||
|
tcp[5] = rgb.r;
|
||||||
|
tcp[6] = rgb.g;
|
||||||
|
tcp[7] = rgb.b;
|
||||||
|
|
||||||
|
rgb = getCRGBForBand(128, pal);
|
||||||
|
tcp[8] = 128;
|
||||||
|
tcp[9] = rgb.r;
|
||||||
|
tcp[10] = rgb.g;
|
||||||
|
tcp[11] = rgb.b;
|
||||||
|
|
||||||
|
rgb = getCRGBForBand(255, pal);
|
||||||
|
tcp[12] = 255; // anchor of last color - must be 255
|
||||||
|
tcp[13] = rgb.r;
|
||||||
|
tcp[14] = rgb.g;
|
||||||
|
tcp[15] = rgb.b;
|
||||||
|
|
||||||
|
strip.customPalettes[strip.customPalettes.size()-1-palettes+pal].loadDynamicGradientPalette(tcp);
|
||||||
|
}
|
||||||
|
|
||||||
// strings to reduce flash memory usage (used more than twice)
|
// strings to reduce flash memory usage (used more than twice)
|
||||||
const char AudioReactive::_name[] PROGMEM = "AudioReactive";
|
const char AudioReactive::_name[] PROGMEM = "AudioReactive";
|
||||||
const char AudioReactive::_enabled[] PROGMEM = "enabled";
|
const char AudioReactive::_enabled[] PROGMEM = "enabled";
|
||||||
@ -1830,5 +1947,6 @@ const char AudioReactive::_inputLvl[] PROGMEM = "inputLevel";
|
|||||||
const char AudioReactive::_analogmic[] PROGMEM = "analogmic";
|
const char AudioReactive::_analogmic[] PROGMEM = "analogmic";
|
||||||
#endif
|
#endif
|
||||||
const char AudioReactive::_digitalmic[] PROGMEM = "digitalmic";
|
const char AudioReactive::_digitalmic[] PROGMEM = "digitalmic";
|
||||||
|
const char AudioReactive::_addPalettes[] PROGMEM = "add-palettes";
|
||||||
const char AudioReactive::UDP_SYNC_HEADER[] PROGMEM = "00002"; // new sync header version, as format no longer compatible with previous structure
|
const char AudioReactive::UDP_SYNC_HEADER[] PROGMEM = "00002"; // new sync header version, as format no longer compatible with previous structure
|
||||||
const char AudioReactive::UDP_SYNC_HEADER_v1[] PROGMEM = "00001"; // old sync header version - need to add backwards-compatibility feature
|
const char AudioReactive::UDP_SYNC_HEADER_v1[] PROGMEM = "00001"; // old sync header version - need to add backwards-compatibility feature
|
||||||
|
@ -24,6 +24,9 @@ Enables the inverted mode in which the background should be enabled and the digi
|
|||||||
### Colon-blinking
|
### Colon-blinking
|
||||||
Enables the blinking colon(s) if they are defined
|
Enables the blinking colon(s) if they are defined
|
||||||
|
|
||||||
|
### Leading-Zero
|
||||||
|
Shows the leading zero of the hour if it exists (i.e. shows `07` instead of `7`)
|
||||||
|
|
||||||
### enable-auto-brightness
|
### enable-auto-brightness
|
||||||
Enables the auto brightness feature. Can be used only when the usermod SN_Photoresistor is installed.
|
Enables the auto brightness feature. Can be used only when the usermod SN_Photoresistor is installed.
|
||||||
|
|
||||||
|
@ -17,6 +17,7 @@ private:
|
|||||||
bool umSSDRDisplayTime = false;
|
bool umSSDRDisplayTime = false;
|
||||||
bool umSSDRInverted = false;
|
bool umSSDRInverted = false;
|
||||||
bool umSSDRColonblink = true;
|
bool umSSDRColonblink = true;
|
||||||
|
bool umSSDRLeadingZero = false;
|
||||||
bool umSSDREnableLDR = false;
|
bool umSSDREnableLDR = false;
|
||||||
String umSSDRHours = "";
|
String umSSDRHours = "";
|
||||||
String umSSDRMinutes = "";
|
String umSSDRMinutes = "";
|
||||||
@ -79,6 +80,7 @@ private:
|
|||||||
static const char _str_timeEnabled[];
|
static const char _str_timeEnabled[];
|
||||||
static const char _str_inverted[];
|
static const char _str_inverted[];
|
||||||
static const char _str_colonblink[];
|
static const char _str_colonblink[];
|
||||||
|
static const char _str_leadingZero[];
|
||||||
static const char _str_displayMask[];
|
static const char _str_displayMask[];
|
||||||
static const char _str_hours[];
|
static const char _str_hours[];
|
||||||
static const char _str_minutes[];
|
static const char _str_minutes[];
|
||||||
@ -105,15 +107,15 @@ private:
|
|||||||
switch (umSSDRDisplayMask[index]) {
|
switch (umSSDRDisplayMask[index]) {
|
||||||
case 'h':
|
case 'h':
|
||||||
timeVar = hourFormat12(localTime);
|
timeVar = hourFormat12(localTime);
|
||||||
_showElements(&umSSDRHours, timeVar, 0, 1);
|
_showElements(&umSSDRHours, timeVar, 0, !umSSDRLeadingZero);
|
||||||
break;
|
break;
|
||||||
case 'H':
|
case 'H':
|
||||||
timeVar = hour(localTime);
|
timeVar = hour(localTime);
|
||||||
_showElements(&umSSDRHours, timeVar, 0, 1);
|
_showElements(&umSSDRHours, timeVar, 0, !umSSDRLeadingZero);
|
||||||
break;
|
break;
|
||||||
case 'k':
|
case 'k':
|
||||||
timeVar = hour(localTime) + 1;
|
timeVar = hour(localTime) + 1;
|
||||||
_showElements(&umSSDRHours, timeVar, 0, 0);
|
_showElements(&umSSDRHours, timeVar, 0, !umSSDRLeadingZero);
|
||||||
break;
|
break;
|
||||||
case 'm':
|
case 'm':
|
||||||
timeVar = minute(localTime);
|
timeVar = minute(localTime);
|
||||||
@ -309,6 +311,9 @@ private:
|
|||||||
if (_cmpIntSetting_P(topic, payload, _str_colonblink, &umSSDRColonblink)) {
|
if (_cmpIntSetting_P(topic, payload, _str_colonblink, &umSSDRColonblink)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
if (_cmpIntSetting_P(topic, payload, _str_leadingZero, &umSSDRLeadingZero)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
if (strcmp_P(topic, _str_displayMask) == 0) {
|
if (strcmp_P(topic, _str_displayMask) == 0) {
|
||||||
umSSDRDisplayMask = String(payload);
|
umSSDRDisplayMask = String(payload);
|
||||||
_publishMQTTstr_P(_str_displayMask, umSSDRDisplayMask);
|
_publishMQTTstr_P(_str_displayMask, umSSDRDisplayMask);
|
||||||
@ -323,6 +328,7 @@ private:
|
|||||||
_publishMQTTint_P(_str_ldrEnabled, umSSDREnableLDR);
|
_publishMQTTint_P(_str_ldrEnabled, umSSDREnableLDR);
|
||||||
_publishMQTTint_P(_str_inverted, umSSDRInverted);
|
_publishMQTTint_P(_str_inverted, umSSDRInverted);
|
||||||
_publishMQTTint_P(_str_colonblink, umSSDRColonblink);
|
_publishMQTTint_P(_str_colonblink, umSSDRColonblink);
|
||||||
|
_publishMQTTint_P(_str_leadingZero, umSSDRLeadingZero);
|
||||||
|
|
||||||
_publishMQTTstr_P(_str_hours, umSSDRHours);
|
_publishMQTTstr_P(_str_hours, umSSDRHours);
|
||||||
_publishMQTTstr_P(_str_minutes, umSSDRMinutes);
|
_publishMQTTstr_P(_str_minutes, umSSDRMinutes);
|
||||||
@ -347,6 +353,7 @@ private:
|
|||||||
ssdrObj[FPSTR(_str_ldrEnabled)] = umSSDREnableLDR;
|
ssdrObj[FPSTR(_str_ldrEnabled)] = umSSDREnableLDR;
|
||||||
ssdrObj[FPSTR(_str_inverted)] = umSSDRInverted;
|
ssdrObj[FPSTR(_str_inverted)] = umSSDRInverted;
|
||||||
ssdrObj[FPSTR(_str_colonblink)] = umSSDRColonblink;
|
ssdrObj[FPSTR(_str_colonblink)] = umSSDRColonblink;
|
||||||
|
ssdrObj[FPSTR(_str_leadingZero)] = umSSDRLeadingZero;
|
||||||
ssdrObj[FPSTR(_str_displayMask)] = umSSDRDisplayMask;
|
ssdrObj[FPSTR(_str_displayMask)] = umSSDRDisplayMask;
|
||||||
ssdrObj[FPSTR(_str_hours)] = umSSDRHours;
|
ssdrObj[FPSTR(_str_hours)] = umSSDRHours;
|
||||||
ssdrObj[FPSTR(_str_minutes)] = umSSDRMinutes;
|
ssdrObj[FPSTR(_str_minutes)] = umSSDRMinutes;
|
||||||
@ -425,6 +432,8 @@ public:
|
|||||||
invert.add(umSSDRInverted);
|
invert.add(umSSDRInverted);
|
||||||
JsonArray blink = user.createNestedArray("Blinking colon");
|
JsonArray blink = user.createNestedArray("Blinking colon");
|
||||||
blink.add(umSSDRColonblink);
|
blink.add(umSSDRColonblink);
|
||||||
|
JsonArray zero = user.createNestedArray("Show the hour leading zero");
|
||||||
|
zero.add(umSSDRLeadingZero);
|
||||||
JsonArray ldrEnable = user.createNestedArray("Auto Brightness enabled");
|
JsonArray ldrEnable = user.createNestedArray("Auto Brightness enabled");
|
||||||
ldrEnable.add(umSSDREnableLDR);
|
ldrEnable.add(umSSDREnableLDR);
|
||||||
|
|
||||||
@ -454,6 +463,7 @@ public:
|
|||||||
umSSDREnableLDR = ssdrObj[FPSTR(_str_ldrEnabled)] | umSSDREnableLDR;
|
umSSDREnableLDR = ssdrObj[FPSTR(_str_ldrEnabled)] | umSSDREnableLDR;
|
||||||
umSSDRInverted = ssdrObj[FPSTR(_str_inverted)] | umSSDRInverted;
|
umSSDRInverted = ssdrObj[FPSTR(_str_inverted)] | umSSDRInverted;
|
||||||
umSSDRColonblink = ssdrObj[FPSTR(_str_colonblink)] | umSSDRColonblink;
|
umSSDRColonblink = ssdrObj[FPSTR(_str_colonblink)] | umSSDRColonblink;
|
||||||
|
umSSDRLeadingZero = ssdrObj[FPSTR(_str_leadingZero)] | umSSDRLeadingZero;
|
||||||
umSSDRDisplayMask = ssdrObj[FPSTR(_str_displayMask)] | umSSDRDisplayMask;
|
umSSDRDisplayMask = ssdrObj[FPSTR(_str_displayMask)] | umSSDRDisplayMask;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -516,6 +526,7 @@ public:
|
|||||||
umSSDREnableLDR = (top[FPSTR(_str_ldrEnabled)] | umSSDREnableLDR);
|
umSSDREnableLDR = (top[FPSTR(_str_ldrEnabled)] | umSSDREnableLDR);
|
||||||
umSSDRInverted = (top[FPSTR(_str_inverted)] | umSSDRInverted);
|
umSSDRInverted = (top[FPSTR(_str_inverted)] | umSSDRInverted);
|
||||||
umSSDRColonblink = (top[FPSTR(_str_colonblink)] | umSSDRColonblink);
|
umSSDRColonblink = (top[FPSTR(_str_colonblink)] | umSSDRColonblink);
|
||||||
|
umSSDRLeadingZero = (top[FPSTR(_str_leadingZero)] | umSSDRLeadingZero);
|
||||||
|
|
||||||
umSSDRDisplayMask = top[FPSTR(_str_displayMask)] | umSSDRDisplayMask;
|
umSSDRDisplayMask = top[FPSTR(_str_displayMask)] | umSSDRDisplayMask;
|
||||||
umSSDRHours = top[FPSTR(_str_hours)] | umSSDRHours;
|
umSSDRHours = top[FPSTR(_str_hours)] | umSSDRHours;
|
||||||
@ -546,6 +557,7 @@ const char UsermodSSDR::_str_name[] PROGMEM = "UsermodSSDR";
|
|||||||
const char UsermodSSDR::_str_timeEnabled[] PROGMEM = "enabled";
|
const char UsermodSSDR::_str_timeEnabled[] PROGMEM = "enabled";
|
||||||
const char UsermodSSDR::_str_inverted[] PROGMEM = "inverted";
|
const char UsermodSSDR::_str_inverted[] PROGMEM = "inverted";
|
||||||
const char UsermodSSDR::_str_colonblink[] PROGMEM = "Colon-blinking";
|
const char UsermodSSDR::_str_colonblink[] PROGMEM = "Colon-blinking";
|
||||||
|
const char UsermodSSDR::_str_leadingZero[] PROGMEM = "Leading-Zero";
|
||||||
const char UsermodSSDR::_str_displayMask[] PROGMEM = "Display-Mask";
|
const char UsermodSSDR::_str_displayMask[] PROGMEM = "Display-Mask";
|
||||||
const char UsermodSSDR::_str_hours[] PROGMEM = "LED-Numbers-Hours";
|
const char UsermodSSDR::_str_hours[] PROGMEM = "LED-Numbers-Hours";
|
||||||
const char UsermodSSDR::_str_minutes[] PROGMEM = "LED-Numbers-Minutes";
|
const char UsermodSSDR::_str_minutes[] PROGMEM = "LED-Numbers-Minutes";
|
||||||
|
@ -177,11 +177,11 @@ uint16_t color_wipe(bool rev, bool useRandomColors) {
|
|||||||
SEGENV.step = 3;
|
SEGENV.step = 3;
|
||||||
}
|
}
|
||||||
if (SEGENV.step == 1) { //if flag set, change to new random color
|
if (SEGENV.step == 1) { //if flag set, change to new random color
|
||||||
SEGENV.aux1 = SEGMENT.get_random_wheel_index(SEGENV.aux0);
|
SEGENV.aux1 = get_random_wheel_index(SEGENV.aux0);
|
||||||
SEGENV.step = 2;
|
SEGENV.step = 2;
|
||||||
}
|
}
|
||||||
if (SEGENV.step == 3) {
|
if (SEGENV.step == 3) {
|
||||||
SEGENV.aux0 = SEGMENT.get_random_wheel_index(SEGENV.aux1);
|
SEGENV.aux0 = get_random_wheel_index(SEGENV.aux1);
|
||||||
SEGENV.step = 0;
|
SEGENV.step = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -271,7 +271,7 @@ uint16_t mode_random_color(void) {
|
|||||||
if (it != SEGENV.step) //new color
|
if (it != SEGENV.step) //new color
|
||||||
{
|
{
|
||||||
SEGENV.aux1 = SEGENV.aux0;
|
SEGENV.aux1 = SEGENV.aux0;
|
||||||
SEGENV.aux0 = SEGMENT.get_random_wheel_index(SEGENV.aux0); //aux0 will store our random color wheel index
|
SEGENV.aux0 = get_random_wheel_index(SEGENV.aux0); //aux0 will store our random color wheel index
|
||||||
SEGENV.step = it;
|
SEGENV.step = it;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -587,7 +587,7 @@ uint16_t mode_twinkle(void) {
|
|||||||
|
|
||||||
uint16_t PRNG16 = SEGENV.aux1;
|
uint16_t PRNG16 = SEGENV.aux1;
|
||||||
|
|
||||||
for (uint16_t i = 0; i < SEGENV.aux0; i++)
|
for (unsigned i = 0; i < SEGENV.aux0; i++)
|
||||||
{
|
{
|
||||||
PRNG16 = (uint16_t)(PRNG16 * 2053) + 13849; // next 'random' number
|
PRNG16 = (uint16_t)(PRNG16 * 2053) + 13849; // next 'random' number
|
||||||
uint32_t p = (uint32_t)SEGLEN * (uint32_t)PRNG16;
|
uint32_t p = (uint32_t)SEGLEN * (uint32_t)PRNG16;
|
||||||
@ -681,7 +681,7 @@ static const char _data_FX_MODE_SPARKLE[] PROGMEM = "Sparkle@!,,,,,,Overlay;!,!;
|
|||||||
* Inspired by www.tweaking4all.com/hardware/arduino/adruino-led-strip-effects/
|
* Inspired by www.tweaking4all.com/hardware/arduino/adruino-led-strip-effects/
|
||||||
*/
|
*/
|
||||||
uint16_t mode_flash_sparkle(void) {
|
uint16_t mode_flash_sparkle(void) {
|
||||||
if (!SEGMENT.check2) for(uint16_t i = 0; i < SEGLEN; i++) {
|
if (!SEGMENT.check2) for (int i = 0; i < SEGLEN; i++) {
|
||||||
SEGMENT.setPixelColor(i, SEGMENT.color_from_palette(i, true, PALETTE_SOLID_WRAP, 0));
|
SEGMENT.setPixelColor(i, SEGMENT.color_from_palette(i, true, PALETTE_SOLID_WRAP, 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -816,7 +816,7 @@ uint16_t chase(uint32_t color1, uint32_t color2, uint32_t color3, bool do_palett
|
|||||||
if (a < SEGENV.step) //we hit the start again, choose new color for Chase random
|
if (a < SEGENV.step) //we hit the start again, choose new color for Chase random
|
||||||
{
|
{
|
||||||
SEGENV.aux1 = SEGENV.aux0; //store previous random color
|
SEGENV.aux1 = SEGENV.aux0; //store previous random color
|
||||||
SEGENV.aux0 = SEGMENT.get_random_wheel_index(SEGENV.aux0);
|
SEGENV.aux0 = get_random_wheel_index(SEGENV.aux0);
|
||||||
}
|
}
|
||||||
color1 = SEGMENT.color_wheel(SEGENV.aux0);
|
color1 = SEGMENT.color_wheel(SEGENV.aux0);
|
||||||
}
|
}
|
||||||
@ -1056,7 +1056,7 @@ uint16_t mode_chase_flash_random(void) {
|
|||||||
SEGENV.aux1 = (SEGENV.aux1 + 1) % SEGLEN;
|
SEGENV.aux1 = (SEGENV.aux1 + 1) % SEGLEN;
|
||||||
|
|
||||||
if (SEGENV.aux1 == 0) {
|
if (SEGENV.aux1 == 0) {
|
||||||
SEGENV.aux0 = SEGMENT.get_random_wheel_index(SEGENV.aux0);
|
SEGENV.aux0 = get_random_wheel_index(SEGENV.aux0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return delay;
|
return delay;
|
||||||
@ -2182,7 +2182,7 @@ uint16_t mode_colortwinkle() {
|
|||||||
CRGB fastled_col, prev;
|
CRGB fastled_col, prev;
|
||||||
fract8 fadeUpAmount = strip.getBrightness()>28 ? 8 + (SEGMENT.speed>>2) : 68-strip.getBrightness();
|
fract8 fadeUpAmount = strip.getBrightness()>28 ? 8 + (SEGMENT.speed>>2) : 68-strip.getBrightness();
|
||||||
fract8 fadeDownAmount = strip.getBrightness()>28 ? 8 + (SEGMENT.speed>>3) : 68-strip.getBrightness();
|
fract8 fadeDownAmount = strip.getBrightness()>28 ? 8 + (SEGMENT.speed>>3) : 68-strip.getBrightness();
|
||||||
for (uint16_t i = 0; i < SEGLEN; i++) {
|
for (int i = 0; i < SEGLEN; i++) {
|
||||||
fastled_col = SEGMENT.getPixelColor(i);
|
fastled_col = SEGMENT.getPixelColor(i);
|
||||||
prev = fastled_col;
|
prev = fastled_col;
|
||||||
uint16_t index = i >> 3;
|
uint16_t index = i >> 3;
|
||||||
@ -2209,9 +2209,9 @@ uint16_t mode_colortwinkle() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (uint16_t j = 0; j <= SEGLEN / 50; j++) {
|
for (unsigned j = 0; j <= SEGLEN / 50; j++) {
|
||||||
if (random8() <= SEGMENT.intensity) {
|
if (random8() <= SEGMENT.intensity) {
|
||||||
for (uint8_t times = 0; times < 5; times++) { //attempt to spawn a new pixel 5 times
|
for (unsigned times = 0; times < 5; times++) { //attempt to spawn a new pixel 5 times
|
||||||
int i = random16(SEGLEN);
|
int i = random16(SEGLEN);
|
||||||
if (SEGMENT.getPixelColor(i) == 0) {
|
if (SEGMENT.getPixelColor(i) == 0) {
|
||||||
fastled_col = ColorFromPalette(SEGPALETTE, random8(), 64, NOBLEND);
|
fastled_col = ColorFromPalette(SEGPALETTE, random8(), 64, NOBLEND);
|
||||||
@ -4661,7 +4661,7 @@ static const char _data_FX_MODE_FLOWSTRIPE[] PROGMEM = "Flow Stripe@Hue speed,Ef
|
|||||||
|
|
||||||
// Black hole
|
// Black hole
|
||||||
uint16_t mode_2DBlackHole(void) { // By: Stepko https://editor.soulmatelights.com/gallery/1012 , Modified by: Andrew Tuline
|
uint16_t mode_2DBlackHole(void) { // By: Stepko https://editor.soulmatelights.com/gallery/1012 , Modified by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -4695,7 +4695,7 @@ static const char _data_FX_MODE_2DBLACKHOLE[] PROGMEM = "Black Hole@Fade rate,Ou
|
|||||||
// 2D Colored Bursts //
|
// 2D Colored Bursts //
|
||||||
////////////////////////////
|
////////////////////////////
|
||||||
uint16_t mode_2DColoredBursts() { // By: ldirko https://editor.soulmatelights.com/gallery/819-colored-bursts , modified by: Andrew Tuline
|
uint16_t mode_2DColoredBursts() { // By: ldirko https://editor.soulmatelights.com/gallery/819-colored-bursts , modified by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -4747,7 +4747,7 @@ static const char _data_FX_MODE_2DCOLOREDBURSTS[] PROGMEM = "Colored Bursts@Spee
|
|||||||
// 2D DNA //
|
// 2D DNA //
|
||||||
/////////////////////
|
/////////////////////
|
||||||
uint16_t mode_2Ddna(void) { // dna originally by by ldirko at https://pastebin.com/pCkkkzcs. Updated by Preyy. WLED conversion by Andrew Tuline.
|
uint16_t mode_2Ddna(void) { // dna originally by by ldirko at https://pastebin.com/pCkkkzcs. Updated by Preyy. WLED conversion by Andrew Tuline.
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -4768,7 +4768,7 @@ static const char _data_FX_MODE_2DDNA[] PROGMEM = "DNA@Scroll speed,Blur;;!;2";
|
|||||||
// 2D DNA Spiral //
|
// 2D DNA Spiral //
|
||||||
/////////////////////////
|
/////////////////////////
|
||||||
uint16_t mode_2DDNASpiral() { // By: ldirko https://editor.soulmatelights.com/gallery/810 , modified by: Andrew Tuline
|
uint16_t mode_2DDNASpiral() { // By: ldirko https://editor.soulmatelights.com/gallery/810 , modified by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -4813,7 +4813,7 @@ static const char _data_FX_MODE_2DDNASPIRAL[] PROGMEM = "DNA Spiral@Scroll speed
|
|||||||
// 2D Drift //
|
// 2D Drift //
|
||||||
/////////////////////////
|
/////////////////////////
|
||||||
uint16_t mode_2DDrift() { // By: Stepko https://editor.soulmatelights.com/gallery/884-drift , Modified by: Andrew Tuline
|
uint16_t mode_2DDrift() { // By: Stepko https://editor.soulmatelights.com/gallery/884-drift , Modified by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -4839,7 +4839,7 @@ static const char _data_FX_MODE_2DDRIFT[] PROGMEM = "Drift@Rotation speed,Blur a
|
|||||||
// 2D Firenoise //
|
// 2D Firenoise //
|
||||||
//////////////////////////
|
//////////////////////////
|
||||||
uint16_t mode_2Dfirenoise(void) { // firenoise2d. By Andrew Tuline. Yet another short routine.
|
uint16_t mode_2Dfirenoise(void) { // firenoise2d. By Andrew Tuline. Yet another short routine.
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -4873,7 +4873,7 @@ static const char _data_FX_MODE_2DFIRENOISE[] PROGMEM = "Firenoise@X scale,Y sca
|
|||||||
// 2D Frizzles //
|
// 2D Frizzles //
|
||||||
//////////////////////////////
|
//////////////////////////////
|
||||||
uint16_t mode_2DFrizzles(void) { // By: Stepko https://editor.soulmatelights.com/gallery/640-color-frizzles , Modified by: Andrew Tuline
|
uint16_t mode_2DFrizzles(void) { // By: Stepko https://editor.soulmatelights.com/gallery/640-color-frizzles , Modified by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -4900,7 +4900,7 @@ typedef struct ColorCount {
|
|||||||
} colorCount;
|
} colorCount;
|
||||||
|
|
||||||
uint16_t mode_2Dgameoflife(void) { // Written by Ewoud Wijma, inspired by https://natureofcode.com/book/chapter-7-cellular-automata/ and https://github.com/DougHaber/nlife-color
|
uint16_t mode_2Dgameoflife(void) { // Written by Ewoud Wijma, inspired by https://natureofcode.com/book/chapter-7-cellular-automata/ and https://github.com/DougHaber/nlife-color
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5006,7 +5006,7 @@ static const char _data_FX_MODE_2DGAMEOFLIFE[] PROGMEM = "Game Of Life@!;!,!;!;2
|
|||||||
// 2D Hiphotic //
|
// 2D Hiphotic //
|
||||||
/////////////////////////
|
/////////////////////////
|
||||||
uint16_t mode_2DHiphotic() { // By: ldirko https://editor.soulmatelights.com/gallery/810 , Modified by: Andrew Tuline
|
uint16_t mode_2DHiphotic() { // By: ldirko https://editor.soulmatelights.com/gallery/810 , Modified by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5038,7 +5038,7 @@ typedef struct Julia {
|
|||||||
} julia;
|
} julia;
|
||||||
|
|
||||||
uint16_t mode_2DJulia(void) { // An animated Julia set by Andrew Tuline.
|
uint16_t mode_2DJulia(void) { // An animated Julia set by Andrew Tuline.
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5144,7 +5144,7 @@ static const char _data_FX_MODE_2DJULIA[] PROGMEM = "Julia@,Max iterations per p
|
|||||||
// 2D Lissajous //
|
// 2D Lissajous //
|
||||||
//////////////////////////////
|
//////////////////////////////
|
||||||
uint16_t mode_2DLissajous(void) { // By: Andrew Tuline
|
uint16_t mode_2DLissajous(void) { // By: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5172,7 +5172,7 @@ static const char _data_FX_MODE_2DLISSAJOUS[] PROGMEM = "Lissajous@X frequency,F
|
|||||||
// 2D Matrix //
|
// 2D Matrix //
|
||||||
///////////////////////
|
///////////////////////
|
||||||
uint16_t mode_2Dmatrix(void) { // Matrix2D. By Jeremy Williams. Adapted by Andrew Tuline & improved by merkisoft and ewowi, and softhack007.
|
uint16_t mode_2Dmatrix(void) { // Matrix2D. By Jeremy Williams. Adapted by Andrew Tuline & improved by merkisoft and ewowi, and softhack007.
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5247,7 +5247,7 @@ static const char _data_FX_MODE_2DMATRIX[] PROGMEM = "Matrix@!,Spawning rate,Tra
|
|||||||
// 2D Metaballs //
|
// 2D Metaballs //
|
||||||
/////////////////////////
|
/////////////////////////
|
||||||
uint16_t mode_2Dmetaballs(void) { // Metaballs by Stefan Petrick. Cannot have one of the dimensions be 2 or less. Adapted by Andrew Tuline.
|
uint16_t mode_2Dmetaballs(void) { // Metaballs by Stefan Petrick. Cannot have one of the dimensions be 2 or less. Adapted by Andrew Tuline.
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5306,7 +5306,7 @@ static const char _data_FX_MODE_2DMETABALLS[] PROGMEM = "Metaballs@!;;!;2";
|
|||||||
// 2D Noise //
|
// 2D Noise //
|
||||||
//////////////////////
|
//////////////////////
|
||||||
uint16_t mode_2Dnoise(void) { // By Andrew Tuline
|
uint16_t mode_2Dnoise(void) { // By Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5329,7 +5329,7 @@ static const char _data_FX_MODE_2DNOISE[] PROGMEM = "Noise2D@!,Scale;;!;2";
|
|||||||
// 2D Plasma Ball //
|
// 2D Plasma Ball //
|
||||||
//////////////////////////////
|
//////////////////////////////
|
||||||
uint16_t mode_2DPlasmaball(void) { // By: Stepko https://editor.soulmatelights.com/gallery/659-plasm-ball , Modified by: Andrew Tuline
|
uint16_t mode_2DPlasmaball(void) { // By: Stepko https://editor.soulmatelights.com/gallery/659-plasm-ball , Modified by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5369,7 +5369,7 @@ static const char _data_FX_MODE_2DPLASMABALL[] PROGMEM = "Plasma Ball@Speed,,Fad
|
|||||||
// return (out_max - out_min) * (x - in_min) / (in_max - in_min) + out_min;
|
// return (out_max - out_min) * (x - in_min) / (in_max - in_min) + out_min;
|
||||||
//}
|
//}
|
||||||
uint16_t mode_2DPolarLights(void) { // By: Kostyantyn Matviyevskyy https://editor.soulmatelights.com/gallery/762-polar-lights , Modified by: Andrew Tuline
|
uint16_t mode_2DPolarLights(void) { // By: Kostyantyn Matviyevskyy https://editor.soulmatelights.com/gallery/762-polar-lights , Modified by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5420,7 +5420,7 @@ static const char _data_FX_MODE_2DPOLARLIGHTS[] PROGMEM = "Polar Lights@!,Scale;
|
|||||||
// 2D Pulser //
|
// 2D Pulser //
|
||||||
/////////////////////////
|
/////////////////////////
|
||||||
uint16_t mode_2DPulser(void) { // By: ldirko https://editor.soulmatelights.com/gallery/878-pulse-test , modifed by: Andrew Tuline
|
uint16_t mode_2DPulser(void) { // By: ldirko https://editor.soulmatelights.com/gallery/878-pulse-test , modifed by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5442,7 +5442,7 @@ static const char _data_FX_MODE_2DPULSER[] PROGMEM = "Pulser@!,Blur;;!;2";
|
|||||||
// 2D Sindots //
|
// 2D Sindots //
|
||||||
/////////////////////////
|
/////////////////////////
|
||||||
uint16_t mode_2DSindots(void) { // By: ldirko https://editor.soulmatelights.com/gallery/597-sin-dots , modified by: Andrew Tuline
|
uint16_t mode_2DSindots(void) { // By: ldirko https://editor.soulmatelights.com/gallery/597-sin-dots , modified by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5473,7 +5473,7 @@ static const char _data_FX_MODE_2DSINDOTS[] PROGMEM = "Sindots@!,Dot distance,Fa
|
|||||||
// custom3 affects the blur amount.
|
// custom3 affects the blur amount.
|
||||||
uint16_t mode_2Dsquaredswirl(void) { // By: Mark Kriegsman. https://gist.github.com/kriegsman/368b316c55221134b160
|
uint16_t mode_2Dsquaredswirl(void) { // By: Mark Kriegsman. https://gist.github.com/kriegsman/368b316c55221134b160
|
||||||
// Modifed by: Andrew Tuline
|
// Modifed by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5507,7 +5507,7 @@ static const char _data_FX_MODE_2DSQUAREDSWIRL[] PROGMEM = "Squared Swirl@,,,,Bl
|
|||||||
// 2D Sun Radiation //
|
// 2D Sun Radiation //
|
||||||
//////////////////////////////
|
//////////////////////////////
|
||||||
uint16_t mode_2DSunradiation(void) { // By: ldirko https://editor.soulmatelights.com/gallery/599-sun-radiation , modified by: Andrew Tuline
|
uint16_t mode_2DSunradiation(void) { // By: ldirko https://editor.soulmatelights.com/gallery/599-sun-radiation , modified by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5557,7 +5557,7 @@ static const char _data_FX_MODE_2DSUNRADIATION[] PROGMEM = "Sun Radiation@Varian
|
|||||||
// 2D Tartan //
|
// 2D Tartan //
|
||||||
/////////////////////////
|
/////////////////////////
|
||||||
uint16_t mode_2Dtartan(void) { // By: Elliott Kember https://editor.soulmatelights.com/gallery/3-tartan , Modified by: Andrew Tuline
|
uint16_t mode_2Dtartan(void) { // By: Elliott Kember https://editor.soulmatelights.com/gallery/3-tartan , Modified by: Andrew Tuline
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5596,7 +5596,7 @@ static const char _data_FX_MODE_2DTARTAN[] PROGMEM = "Tartan@X scale,Y scale,,,S
|
|||||||
// 2D spaceships //
|
// 2D spaceships //
|
||||||
/////////////////////////
|
/////////////////////////
|
||||||
uint16_t mode_2Dspaceships(void) { //// Space ships by stepko (c)05.02.21 [https://editor.soulmatelights.com/gallery/639-space-ships], adapted by Blaz Kristan (AKA blazoncek)
|
uint16_t mode_2Dspaceships(void) { //// Space ships by stepko (c)05.02.21 [https://editor.soulmatelights.com/gallery/639-space-ships], adapted by Blaz Kristan (AKA blazoncek)
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5639,7 +5639,7 @@ static const char _data_FX_MODE_2DSPACESHIPS[] PROGMEM = "Spaceships@!,Blur;;!;2
|
|||||||
//// Crazy bees by stepko (c)12.02.21 [https://editor.soulmatelights.com/gallery/651-crazy-bees], adapted by Blaz Kristan (AKA blazoncek)
|
//// Crazy bees by stepko (c)12.02.21 [https://editor.soulmatelights.com/gallery/651-crazy-bees], adapted by Blaz Kristan (AKA blazoncek)
|
||||||
#define MAX_BEES 5
|
#define MAX_BEES 5
|
||||||
uint16_t mode_2Dcrazybees(void) {
|
uint16_t mode_2Dcrazybees(void) {
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5711,7 +5711,7 @@ static const char _data_FX_MODE_2DCRAZYBEES[] PROGMEM = "Crazy Bees@!,Blur;;;2";
|
|||||||
//// Ghost Rider by stepko (c)2021 [https://editor.soulmatelights.com/gallery/716-ghost-rider], adapted by Blaz Kristan (AKA blazoncek)
|
//// Ghost Rider by stepko (c)2021 [https://editor.soulmatelights.com/gallery/716-ghost-rider], adapted by Blaz Kristan (AKA blazoncek)
|
||||||
#define LIGHTERS_AM 64 // max lighters (adequate for 32x32 matrix)
|
#define LIGHTERS_AM 64 // max lighters (adequate for 32x32 matrix)
|
||||||
uint16_t mode_2Dghostrider(void) {
|
uint16_t mode_2Dghostrider(void) {
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5801,7 +5801,7 @@ static const char _data_FX_MODE_2DGHOSTRIDER[] PROGMEM = "Ghost Rider@Fade rate,
|
|||||||
//// Floating Blobs by stepko (c)2021 [https://editor.soulmatelights.com/gallery/573-blobs], adapted by Blaz Kristan (AKA blazoncek)
|
//// Floating Blobs by stepko (c)2021 [https://editor.soulmatelights.com/gallery/573-blobs], adapted by Blaz Kristan (AKA blazoncek)
|
||||||
#define MAX_BLOBS 8
|
#define MAX_BLOBS 8
|
||||||
uint16_t mode_2Dfloatingblobs(void) {
|
uint16_t mode_2Dfloatingblobs(void) {
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5899,7 +5899,7 @@ static const char _data_FX_MODE_2DBLOBS[] PROGMEM = "Blobs@!,# blobs,Blur,Trail;
|
|||||||
// 2D Scrolling text //
|
// 2D Scrolling text //
|
||||||
////////////////////////////
|
////////////////////////////
|
||||||
uint16_t mode_2Dscrollingtext(void) {
|
uint16_t mode_2Dscrollingtext(void) {
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -5946,6 +5946,8 @@ uint16_t mode_2Dscrollingtext(void) {
|
|||||||
else if (!strncmp_P(text,PSTR("#MMDD"),5)) sprintf_P(text, zero?PSTR("%02d/%02d") :PSTR("%d/%d"), month(localTime), day(localTime));
|
else if (!strncmp_P(text,PSTR("#MMDD"),5)) sprintf_P(text, zero?PSTR("%02d/%02d") :PSTR("%d/%d"), month(localTime), day(localTime));
|
||||||
else if (!strncmp_P(text,PSTR("#TIME"),5)) sprintf_P(text, zero?PSTR("%02d:%02d%s") :PSTR("%2d:%02d%s"), AmPmHour, minute(localTime), sec);
|
else if (!strncmp_P(text,PSTR("#TIME"),5)) sprintf_P(text, zero?PSTR("%02d:%02d%s") :PSTR("%2d:%02d%s"), AmPmHour, minute(localTime), sec);
|
||||||
else if (!strncmp_P(text,PSTR("#HHMM"),5)) sprintf_P(text, zero?PSTR("%02d:%02d") :PSTR("%d:%02d"), AmPmHour, minute(localTime));
|
else if (!strncmp_P(text,PSTR("#HHMM"),5)) sprintf_P(text, zero?PSTR("%02d:%02d") :PSTR("%d:%02d"), AmPmHour, minute(localTime));
|
||||||
|
else if (!strncmp_P(text,PSTR("#HH"),3)) sprintf_P(text, zero?PSTR("%02d") :PSTR("%d"), AmPmHour);
|
||||||
|
else if (!strncmp_P(text,PSTR("#MM"),3)) sprintf_P(text, zero?PSTR("%02d") :PSTR("%d"), minute(localTime));
|
||||||
}
|
}
|
||||||
|
|
||||||
const int numberOfLetters = strlen(text);
|
const int numberOfLetters = strlen(text);
|
||||||
@ -5999,7 +6001,7 @@ static const char _data_FX_MODE_2DSCROLLTEXT[] PROGMEM = "Scrolling Text@!,Y Off
|
|||||||
////////////////////////////
|
////////////////////////////
|
||||||
//// Drift Rose by stepko (c)2021 [https://editor.soulmatelights.com/gallery/1369-drift-rose-pattern], adapted by Blaz Kristan (AKA blazoncek)
|
//// Drift Rose by stepko (c)2021 [https://editor.soulmatelights.com/gallery/1369-drift-rose-pattern], adapted by Blaz Kristan (AKA blazoncek)
|
||||||
uint16_t mode_2Ddriftrose(void) {
|
uint16_t mode_2Ddriftrose(void) {
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -6153,7 +6155,7 @@ static const char _data_FX_MODE_RIPPLEPEAK[] PROGMEM = "Ripple Peak@Fade rate,Ma
|
|||||||
/////////////////////////
|
/////////////////////////
|
||||||
// By: Mark Kriegsman https://gist.github.com/kriegsman/5adca44e14ad025e6d3b , modified by Andrew Tuline
|
// By: Mark Kriegsman https://gist.github.com/kriegsman/5adca44e14ad025e6d3b , modified by Andrew Tuline
|
||||||
uint16_t mode_2DSwirl(void) {
|
uint16_t mode_2DSwirl(void) {
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -6197,7 +6199,7 @@ static const char _data_FX_MODE_2DSWIRL[] PROGMEM = "Swirl@!,Sensitivity,Blur;,B
|
|||||||
/////////////////////////
|
/////////////////////////
|
||||||
// By: Stepko, https://editor.soulmatelights.com/gallery/652-wave , modified by Andrew Tuline
|
// By: Stepko, https://editor.soulmatelights.com/gallery/652-wave , modified by Andrew Tuline
|
||||||
uint16_t mode_2DWaverly(void) {
|
uint16_t mode_2DWaverly(void) {
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -7184,7 +7186,7 @@ static const char _data_FX_MODE_WATERFALL[] PROGMEM = "Waterfall@!,Adjust color,
|
|||||||
// ** 2D GEQ //
|
// ** 2D GEQ //
|
||||||
/////////////////////////
|
/////////////////////////
|
||||||
uint16_t mode_2DGEQ(void) { // By Will Tatam. Code reduction by Ewoud Wijma.
|
uint16_t mode_2DGEQ(void) { // By Will Tatam. Code reduction by Ewoud Wijma.
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const int NUM_BANDS = map(SEGMENT.custom1, 0, 255, 1, 16);
|
const int NUM_BANDS = map(SEGMENT.custom1, 0, 255, 1, 16);
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
@ -7242,7 +7244,7 @@ static const char _data_FX_MODE_2DGEQ[] PROGMEM = "GEQ@Fade speed,Ripple decay,#
|
|||||||
// ** 2D Funky plank //
|
// ** 2D Funky plank //
|
||||||
/////////////////////////
|
/////////////////////////
|
||||||
uint16_t mode_2DFunkyPlank(void) { // Written by ??? Adapted by Will Tatam.
|
uint16_t mode_2DFunkyPlank(void) { // Written by ??? Adapted by Will Tatam.
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -7334,7 +7336,7 @@ static uint8_t akemi[] PROGMEM = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
uint16_t mode_2DAkemi(void) {
|
uint16_t mode_2DAkemi(void) {
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -7402,7 +7404,7 @@ static const char _data_FX_MODE_2DAKEMI[] PROGMEM = "Akemi@Color speed,Dance;Hea
|
|||||||
// https://editor.soulmatelights.com/gallery/1089-distorsion-waves
|
// https://editor.soulmatelights.com/gallery/1089-distorsion-waves
|
||||||
// adapted for WLED by @blazoncek
|
// adapted for WLED by @blazoncek
|
||||||
uint16_t mode_2Ddistortionwaves() {
|
uint16_t mode_2Ddistortionwaves() {
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -7457,7 +7459,7 @@ static const char _data_FX_MODE_2DDISTORTIONWAVES[] PROGMEM = "Distortion Waves@
|
|||||||
//Idea from https://www.youtube.com/watch?v=DiHBgITrZck&ab_channel=StefanPetrick
|
//Idea from https://www.youtube.com/watch?v=DiHBgITrZck&ab_channel=StefanPetrick
|
||||||
// adapted for WLED by @blazoncek
|
// adapted for WLED by @blazoncek
|
||||||
uint16_t mode_2Dsoap() {
|
uint16_t mode_2Dsoap() {
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -7569,7 +7571,7 @@ static const char _data_FX_MODE_2DSOAP[] PROGMEM = "Soap@!,Smoothness;;!;2";
|
|||||||
//Stepko and Sutaburosu
|
//Stepko and Sutaburosu
|
||||||
// adapted for WLED by @blazoncek
|
// adapted for WLED by @blazoncek
|
||||||
uint16_t mode_2Doctopus() {
|
uint16_t mode_2Doctopus() {
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
@ -7625,7 +7627,7 @@ static const char _data_FX_MODE_2DOCTOPUS[] PROGMEM = "Octopus@!,,Offset X,Offse
|
|||||||
//@Stepko (https://editor.soulmatelights.com/gallery/1704-wavingcells)
|
//@Stepko (https://editor.soulmatelights.com/gallery/1704-wavingcells)
|
||||||
// adapted for WLED by @blazoncek
|
// adapted for WLED by @blazoncek
|
||||||
uint16_t mode_2Dwavingcell() {
|
uint16_t mode_2Dwavingcell() {
|
||||||
if (!strip.isMatrix) return mode_static(); // not a 2D set-up
|
if (!strip.isMatrix || !SEGMENT.is2D()) return mode_static(); // not a 2D set-up
|
||||||
|
|
||||||
const uint16_t cols = SEGMENT.virtualWidth();
|
const uint16_t cols = SEGMENT.virtualWidth();
|
||||||
const uint16_t rows = SEGMENT.virtualHeight();
|
const uint16_t rows = SEGMENT.virtualHeight();
|
||||||
|
172
wled00/FX.h
172
wled00/FX.h
@ -109,20 +109,15 @@
|
|||||||
#define PINK (uint32_t)0xFF1493
|
#define PINK (uint32_t)0xFF1493
|
||||||
#define ULTRAWHITE (uint32_t)0xFFFFFFFF
|
#define ULTRAWHITE (uint32_t)0xFFFFFFFF
|
||||||
#define DARKSLATEGRAY (uint32_t)0x2F4F4F
|
#define DARKSLATEGRAY (uint32_t)0x2F4F4F
|
||||||
#define DARKSLATEGREY (uint32_t)0x2F4F4F
|
#define DARKSLATEGREY DARKSLATEGRAY
|
||||||
|
|
||||||
// options
|
// segment options
|
||||||
// bit 7: segment is in transition mode
|
|
||||||
// bits 4-6: TBD
|
|
||||||
// bit 3: mirror effect within segment
|
|
||||||
// bit 2: segment is on
|
|
||||||
// bit 1: reverse segment
|
|
||||||
// bit 0: segment is selected
|
|
||||||
#define NO_OPTIONS (uint16_t)0x0000
|
#define NO_OPTIONS (uint16_t)0x0000
|
||||||
#define TRANSPOSED (uint16_t)0x0400 // rotated 90deg & reversed
|
#define TRANSPOSED (uint16_t)0x0100 // rotated 90deg & reversed
|
||||||
#define REVERSE_Y_2D (uint16_t)0x0200
|
#define MIRROR_Y_2D (uint16_t)0x0080
|
||||||
#define MIRROR_Y_2D (uint16_t)0x0100
|
#define REVERSE_Y_2D (uint16_t)0x0040
|
||||||
#define TRANSITIONAL (uint16_t)0x0080
|
#define RESET_REQ (uint16_t)0x0020
|
||||||
|
#define FROZEN (uint16_t)0x0010
|
||||||
#define MIRROR (uint16_t)0x0008
|
#define MIRROR (uint16_t)0x0008
|
||||||
#define SEGMENT_ON (uint16_t)0x0004
|
#define SEGMENT_ON (uint16_t)0x0004
|
||||||
#define REVERSE (uint16_t)0x0002
|
#define REVERSE (uint16_t)0x0002
|
||||||
@ -348,12 +343,11 @@ typedef struct Segment {
|
|||||||
bool mirror : 1; // 3 : mirrored
|
bool mirror : 1; // 3 : mirrored
|
||||||
bool freeze : 1; // 4 : paused/frozen
|
bool freeze : 1; // 4 : paused/frozen
|
||||||
bool reset : 1; // 5 : indicates that Segment runtime requires reset
|
bool reset : 1; // 5 : indicates that Segment runtime requires reset
|
||||||
bool transitional: 1; // 6 : transitional (there is transition occuring)
|
bool reverse_y : 1; // 6 : reversed Y (2D)
|
||||||
bool reverse_y : 1; // 7 : reversed Y (2D)
|
bool mirror_y : 1; // 7 : mirrored Y (2D)
|
||||||
bool mirror_y : 1; // 8 : mirrored Y (2D)
|
bool transpose : 1; // 8 : transposed (2D, swapped X & Y)
|
||||||
bool transpose : 1; // 9 : transposed (2D, swapped X & Y)
|
uint8_t map1D2D : 3; // 9-11 : mapping for 1D effect on 2D (0-use as strip, 1-expand vertically, 2-circular/arc, 3-rectangular/corner, ...)
|
||||||
uint8_t map1D2D : 3; // 10-12 : mapping for 1D effect on 2D (0-use as strip, 1-expand vertically, 2-circular/arc, 3-rectangular/corner, ...)
|
uint8_t soundSim : 2; // 12-13 : 0-3 sound simulation types ("soft" & "hard" or "on"/"off")
|
||||||
uint8_t soundSim : 1; // 13 : 0-1 sound simulation types ("soft" & "hard" or "on"/"off")
|
|
||||||
uint8_t set : 2; // 14-15 : 0-3 UI segment sets/groups
|
uint8_t set : 2; // 14-15 : 0-3 UI segment sets/groups
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
@ -484,7 +478,6 @@ typedef struct Segment {
|
|||||||
_dataLen(0),
|
_dataLen(0),
|
||||||
_t(nullptr)
|
_t(nullptr)
|
||||||
{
|
{
|
||||||
//refreshLightCapabilities();
|
|
||||||
#ifdef WLED_DEBUG
|
#ifdef WLED_DEBUG
|
||||||
//Serial.printf("-- Creating segment: %p\n", this);
|
//Serial.printf("-- Creating segment: %p\n", this);
|
||||||
#endif
|
#endif
|
||||||
@ -519,6 +512,7 @@ typedef struct Segment {
|
|||||||
|
|
||||||
inline bool getOption(uint8_t n) const { return ((options >> n) & 0x01); }
|
inline bool getOption(uint8_t n) const { return ((options >> n) & 0x01); }
|
||||||
inline bool isSelected(void) const { return selected; }
|
inline bool isSelected(void) const { return selected; }
|
||||||
|
inline bool isInTransition(void) const { return _t != nullptr; }
|
||||||
inline bool isActive(void) const { return stop > start; }
|
inline bool isActive(void) const { return stop > start; }
|
||||||
inline bool is2D(void) const { return (width()>1 && height()>1); }
|
inline bool is2D(void) const { return (width()>1 && height()>1); }
|
||||||
inline bool hasRGB(void) const { return _isRGB; }
|
inline bool hasRGB(void) const { return _isRGB; }
|
||||||
@ -569,33 +563,33 @@ typedef struct Segment {
|
|||||||
void restoreSegenv(tmpsegd_t &tmpSegD);
|
void restoreSegenv(tmpsegd_t &tmpSegD);
|
||||||
#endif
|
#endif
|
||||||
uint16_t progress(void); //transition progression between 0-65535
|
uint16_t progress(void); //transition progression between 0-65535
|
||||||
uint8_t currentBri(uint8_t briNew, bool useCct = false);
|
uint8_t currentBri(bool useCct = false);
|
||||||
uint8_t currentMode(uint8_t modeNew);
|
uint8_t currentMode(void);
|
||||||
uint32_t currentColor(uint8_t slot, uint32_t colorNew);
|
uint32_t currentColor(uint8_t slot);
|
||||||
CRGBPalette16 &loadPalette(CRGBPalette16 &tgt, uint8_t pal);
|
CRGBPalette16 &loadPalette(CRGBPalette16 &tgt, uint8_t pal);
|
||||||
CRGBPalette16 ¤tPalette(CRGBPalette16 &tgt, uint8_t paletteID);
|
CRGBPalette16 ¤tPalette(CRGBPalette16 &tgt, uint8_t paletteID);
|
||||||
|
|
||||||
// 1D strip
|
// 1D strip
|
||||||
uint16_t virtualLength(void) const;
|
uint16_t virtualLength(void) const;
|
||||||
void setPixelColor(int n, uint32_t c); // set relative pixel within segment with color
|
void setPixelColor(int n, uint32_t c); // set relative pixel within segment with color
|
||||||
void setPixelColor(int n, byte r, byte g, byte b, byte w = 0) { setPixelColor(n, RGBW32(r,g,b,w)); } // automatically inline
|
inline void setPixelColor(unsigned n, uint32_t c) { setPixelColor(int(n), c); }
|
||||||
void setPixelColor(int n, CRGB c) { setPixelColor(n, RGBW32(c.r,c.g,c.b,0)); } // automatically inline
|
inline void setPixelColor(int n, byte r, byte g, byte b, byte w = 0) { setPixelColor(n, RGBW32(r,g,b,w)); }
|
||||||
|
inline void setPixelColor(int n, CRGB c) { setPixelColor(n, RGBW32(c.r,c.g,c.b,0)); }
|
||||||
void setPixelColor(float i, uint32_t c, bool aa = true);
|
void setPixelColor(float i, uint32_t c, bool aa = true);
|
||||||
void setPixelColor(float i, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0, bool aa = true) { setPixelColor(i, RGBW32(r,g,b,w), aa); }
|
inline void setPixelColor(float i, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0, bool aa = true) { setPixelColor(i, RGBW32(r,g,b,w), aa); }
|
||||||
void setPixelColor(float i, CRGB c, bool aa = true) { setPixelColor(i, RGBW32(c.r,c.g,c.b,0), aa); }
|
inline void setPixelColor(float i, CRGB c, bool aa = true) { setPixelColor(i, RGBW32(c.r,c.g,c.b,0), aa); }
|
||||||
uint32_t getPixelColor(int i);
|
uint32_t getPixelColor(int i);
|
||||||
// 1D support functions (some implement 2D as well)
|
// 1D support functions (some implement 2D as well)
|
||||||
void blur(uint8_t);
|
void blur(uint8_t);
|
||||||
void fill(uint32_t c);
|
void fill(uint32_t c);
|
||||||
void fade_out(uint8_t r);
|
void fade_out(uint8_t r);
|
||||||
void fadeToBlackBy(uint8_t fadeBy);
|
void fadeToBlackBy(uint8_t fadeBy);
|
||||||
void blendPixelColor(int n, uint32_t color, uint8_t blend);
|
inline void blendPixelColor(int n, uint32_t color, uint8_t blend) { setPixelColor(n, color_blend(getPixelColor(n), color, blend)); }
|
||||||
void blendPixelColor(int n, CRGB c, uint8_t blend) { blendPixelColor(n, RGBW32(c.r,c.g,c.b,0), blend); }
|
inline void blendPixelColor(int n, CRGB c, uint8_t blend) { blendPixelColor(n, RGBW32(c.r,c.g,c.b,0), blend); }
|
||||||
void addPixelColor(int n, uint32_t color, bool fast = false);
|
inline void addPixelColor(int n, uint32_t color, bool fast = false) { setPixelColor(n, color_add(getPixelColor(n), color, fast)); }
|
||||||
void addPixelColor(int n, byte r, byte g, byte b, byte w = 0, bool fast = false) { addPixelColor(n, RGBW32(r,g,b,w), fast); } // automatically inline
|
inline void addPixelColor(int n, byte r, byte g, byte b, byte w = 0, bool fast = false) { addPixelColor(n, RGBW32(r,g,b,w), fast); }
|
||||||
void addPixelColor(int n, CRGB c, bool fast = false) { addPixelColor(n, RGBW32(c.r,c.g,c.b,0), fast); } // automatically inline
|
inline void addPixelColor(int n, CRGB c, bool fast = false) { addPixelColor(n, RGBW32(c.r,c.g,c.b,0), fast); }
|
||||||
void fadePixelColor(uint16_t n, uint8_t fade);
|
inline void fadePixelColor(uint16_t n, uint8_t fade) { setPixelColor(n, color_fade(getPixelColor(n), fade, true)); }
|
||||||
uint8_t get_random_wheel_index(uint8_t pos);
|
|
||||||
uint32_t color_from_palette(uint16_t, bool mapping, bool wrap, uint8_t mcol, uint8_t pbri = 255);
|
uint32_t color_from_palette(uint16_t, bool mapping, bool wrap, uint8_t mcol, uint8_t pbri = 255);
|
||||||
uint32_t color_wheel(uint8_t pos);
|
uint32_t color_wheel(uint8_t pos);
|
||||||
|
|
||||||
@ -606,19 +600,20 @@ typedef struct Segment {
|
|||||||
#ifndef WLED_DISABLE_2D
|
#ifndef WLED_DISABLE_2D
|
||||||
uint16_t XY(uint16_t x, uint16_t y); // support function to get relative index within segment
|
uint16_t XY(uint16_t x, uint16_t y); // support function to get relative index within segment
|
||||||
void setPixelColorXY(int x, int y, uint32_t c); // set relative pixel within segment with color
|
void setPixelColorXY(int x, int y, uint32_t c); // set relative pixel within segment with color
|
||||||
void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) { setPixelColorXY(x, y, RGBW32(r,g,b,w)); } // automatically inline
|
inline void setPixelColorXY(unsigned x, unsigned y, uint32_t c) { setPixelColorXY(int(x), int(y), c); }
|
||||||
void setPixelColorXY(int x, int y, CRGB c) { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0)); } // automatically inline
|
inline void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) { setPixelColorXY(x, y, RGBW32(r,g,b,w)); }
|
||||||
|
inline void setPixelColorXY(int x, int y, CRGB c) { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0)); }
|
||||||
void setPixelColorXY(float x, float y, uint32_t c, bool aa = true);
|
void setPixelColorXY(float x, float y, uint32_t c, bool aa = true);
|
||||||
void setPixelColorXY(float x, float y, byte r, byte g, byte b, byte w = 0, bool aa = true) { setPixelColorXY(x, y, RGBW32(r,g,b,w), aa); }
|
inline void setPixelColorXY(float x, float y, byte r, byte g, byte b, byte w = 0, bool aa = true) { setPixelColorXY(x, y, RGBW32(r,g,b,w), aa); }
|
||||||
void setPixelColorXY(float x, float y, CRGB c, bool aa = true) { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), aa); }
|
inline void setPixelColorXY(float x, float y, CRGB c, bool aa = true) { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), aa); }
|
||||||
uint32_t getPixelColorXY(uint16_t x, uint16_t y);
|
uint32_t getPixelColorXY(uint16_t x, uint16_t y);
|
||||||
// 2D support functions
|
// 2D support functions
|
||||||
void blendPixelColorXY(uint16_t x, uint16_t y, uint32_t color, uint8_t blend);
|
inline void blendPixelColorXY(uint16_t x, uint16_t y, uint32_t color, uint8_t blend) { setPixelColorXY(x, y, color_blend(getPixelColorXY(x,y), color, blend)); }
|
||||||
void blendPixelColorXY(uint16_t x, uint16_t y, CRGB c, uint8_t blend) { blendPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), blend); }
|
inline void blendPixelColorXY(uint16_t x, uint16_t y, CRGB c, uint8_t blend) { blendPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), blend); }
|
||||||
void addPixelColorXY(int x, int y, uint32_t color, bool fast = false);
|
inline void addPixelColorXY(int x, int y, uint32_t color, bool fast = false) { setPixelColorXY(x, y, color_add(getPixelColorXY(x,y), color, fast)); }
|
||||||
void addPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0, bool fast = false) { addPixelColorXY(x, y, RGBW32(r,g,b,w), fast); } // automatically inline
|
inline void addPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0, bool fast = false) { addPixelColorXY(x, y, RGBW32(r,g,b,w), fast); }
|
||||||
void addPixelColorXY(int x, int y, CRGB c, bool fast = false) { addPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), fast); }
|
inline void addPixelColorXY(int x, int y, CRGB c, bool fast = false) { addPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), fast); }
|
||||||
void fadePixelColorXY(uint16_t x, uint16_t y, uint8_t fade);
|
inline void fadePixelColorXY(uint16_t x, uint16_t y, uint8_t fade) { setPixelColorXY(x, y, color_fade(getPixelColorXY(x,y), fade, true)); }
|
||||||
void box_blur(uint16_t i, bool vertical, fract8 blur_amount); // 1D box blur (with weight)
|
void box_blur(uint16_t i, bool vertical, fract8 blur_amount); // 1D box blur (with weight)
|
||||||
void blurRow(uint16_t row, fract8 blur_amount);
|
void blurRow(uint16_t row, fract8 blur_amount);
|
||||||
void blurCol(uint16_t col, fract8 blur_amount);
|
void blurCol(uint16_t col, fract8 blur_amount);
|
||||||
@ -628,43 +623,43 @@ typedef struct Segment {
|
|||||||
void draw_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c);
|
void draw_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c);
|
||||||
void fill_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c);
|
void fill_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c);
|
||||||
void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, uint32_t c);
|
void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, uint32_t c);
|
||||||
void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, CRGB c) { drawLine(x0, y0, x1, y1, RGBW32(c.r,c.g,c.b,0)); } // automatic inline
|
inline void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, CRGB c) { drawLine(x0, y0, x1, y1, RGBW32(c.r,c.g,c.b,0)); } // automatic inline
|
||||||
void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, uint32_t color, uint32_t col2 = 0, int8_t rotate = 0);
|
void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, uint32_t color, uint32_t col2 = 0, int8_t rotate = 0);
|
||||||
void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB c) { drawCharacter(chr, x, y, w, h, RGBW32(c.r,c.g,c.b,0)); } // automatic inline
|
inline void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB c) { drawCharacter(chr, x, y, w, h, RGBW32(c.r,c.g,c.b,0)); } // automatic inline
|
||||||
void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB c, CRGB c2, int8_t rotate = 0) { drawCharacter(chr, x, y, w, h, RGBW32(c.r,c.g,c.b,0), RGBW32(c2.r,c2.g,c2.b,0), rotate); } // automatic inline
|
inline void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB c, CRGB c2, int8_t rotate = 0) { drawCharacter(chr, x, y, w, h, RGBW32(c.r,c.g,c.b,0), RGBW32(c2.r,c2.g,c2.b,0), rotate); } // automatic inline
|
||||||
void wu_pixel(uint32_t x, uint32_t y, CRGB c);
|
void wu_pixel(uint32_t x, uint32_t y, CRGB c);
|
||||||
void blur1d(fract8 blur_amount); // blur all rows in 1 dimension
|
void blur1d(fract8 blur_amount); // blur all rows in 1 dimension
|
||||||
void blur2d(fract8 blur_amount) { blur(blur_amount); }
|
inline void blur2d(fract8 blur_amount) { blur(blur_amount); }
|
||||||
void fill_solid(CRGB c) { fill(RGBW32(c.r,c.g,c.b,0)); }
|
inline void fill_solid(CRGB c) { fill(RGBW32(c.r,c.g,c.b,0)); }
|
||||||
void nscale8(uint8_t scale);
|
void nscale8(uint8_t scale);
|
||||||
#else
|
#else
|
||||||
uint16_t XY(uint16_t x, uint16_t y) { return x; }
|
inline uint16_t XY(uint16_t x, uint16_t y) { return x; }
|
||||||
void setPixelColorXY(int x, int y, uint32_t c) { setPixelColor(x, c); }
|
inline void setPixelColorXY(int x, int y, uint32_t c) { setPixelColor(x, c); }
|
||||||
void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) { setPixelColor(x, RGBW32(r,g,b,w)); }
|
inline void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) { setPixelColor(x, RGBW32(r,g,b,w)); }
|
||||||
void setPixelColorXY(int x, int y, CRGB c) { setPixelColor(x, RGBW32(c.r,c.g,c.b,0)); }
|
inline void setPixelColorXY(int x, int y, CRGB c) { setPixelColor(x, RGBW32(c.r,c.g,c.b,0)); }
|
||||||
void setPixelColorXY(float x, float y, uint32_t c, bool aa = true) { setPixelColor(x, c, aa); }
|
inline void setPixelColorXY(float x, float y, uint32_t c, bool aa = true) { setPixelColor(x, c, aa); }
|
||||||
void setPixelColorXY(float x, float y, byte r, byte g, byte b, byte w = 0, bool aa = true) { setPixelColor(x, RGBW32(r,g,b,w), aa); }
|
inline void setPixelColorXY(float x, float y, byte r, byte g, byte b, byte w = 0, bool aa = true) { setPixelColor(x, RGBW32(r,g,b,w), aa); }
|
||||||
void setPixelColorXY(float x, float y, CRGB c, bool aa = true) { setPixelColor(x, RGBW32(c.r,c.g,c.b,0), aa); }
|
inline void setPixelColorXY(float x, float y, CRGB c, bool aa = true) { setPixelColor(x, RGBW32(c.r,c.g,c.b,0), aa); }
|
||||||
uint32_t getPixelColorXY(uint16_t x, uint16_t y) { return getPixelColor(x); }
|
inline uint32_t getPixelColorXY(uint16_t x, uint16_t y) { return getPixelColor(x); }
|
||||||
void blendPixelColorXY(uint16_t x, uint16_t y, uint32_t c, uint8_t blend) { blendPixelColor(x, c, blend); }
|
inline void blendPixelColorXY(uint16_t x, uint16_t y, uint32_t c, uint8_t blend) { blendPixelColor(x, c, blend); }
|
||||||
void blendPixelColorXY(uint16_t x, uint16_t y, CRGB c, uint8_t blend) { blendPixelColor(x, RGBW32(c.r,c.g,c.b,0), blend); }
|
inline void blendPixelColorXY(uint16_t x, uint16_t y, CRGB c, uint8_t blend) { blendPixelColor(x, RGBW32(c.r,c.g,c.b,0), blend); }
|
||||||
void addPixelColorXY(int x, int y, uint32_t color, bool fast = false) { addPixelColor(x, color, fast); }
|
inline void addPixelColorXY(int x, int y, uint32_t color, bool fast = false) { addPixelColor(x, color, fast); }
|
||||||
void addPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0, bool fast = false) { addPixelColor(x, RGBW32(r,g,b,w), fast); }
|
inline void addPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0, bool fast = false) { addPixelColor(x, RGBW32(r,g,b,w), fast); }
|
||||||
void addPixelColorXY(int x, int y, CRGB c, bool fast = false) { addPixelColor(x, RGBW32(c.r,c.g,c.b,0), fast); }
|
inline void addPixelColorXY(int x, int y, CRGB c, bool fast = false) { addPixelColor(x, RGBW32(c.r,c.g,c.b,0), fast); }
|
||||||
void fadePixelColorXY(uint16_t x, uint16_t y, uint8_t fade) { fadePixelColor(x, fade); }
|
inline void fadePixelColorXY(uint16_t x, uint16_t y, uint8_t fade) { fadePixelColor(x, fade); }
|
||||||
void box_blur(uint16_t i, bool vertical, fract8 blur_amount) {}
|
inline void box_blur(uint16_t i, bool vertical, fract8 blur_amount) {}
|
||||||
void blurRow(uint16_t row, fract8 blur_amount) {}
|
inline void blurRow(uint16_t row, fract8 blur_amount) {}
|
||||||
void blurCol(uint16_t col, fract8 blur_amount) {}
|
inline void blurCol(uint16_t col, fract8 blur_amount) {}
|
||||||
void moveX(int8_t delta, bool wrap = false) {}
|
inline void moveX(int8_t delta, bool wrap = false) {}
|
||||||
void moveY(int8_t delta, bool wrap = false) {}
|
inline void moveY(int8_t delta, bool wrap = false) {}
|
||||||
void move(uint8_t dir, uint8_t delta, bool wrap = false) {}
|
inline void move(uint8_t dir, uint8_t delta, bool wrap = false) {}
|
||||||
void fill_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c) {}
|
inline void fill_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c) {}
|
||||||
void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, uint32_t c) {}
|
inline void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, uint32_t c) {}
|
||||||
void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, CRGB c) {}
|
inline void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, CRGB c) {}
|
||||||
void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, uint32_t color, uint32_t = 0, int8_t = 0) {}
|
inline void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, uint32_t color, uint32_t = 0, int8_t = 0) {}
|
||||||
void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB color) {}
|
inline void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB color) {}
|
||||||
void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB c, CRGB c2, int8_t rotate = 0) {}
|
inline void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB c, CRGB c2, int8_t rotate = 0) {}
|
||||||
void wu_pixel(uint32_t x, uint32_t y, CRGB c) {}
|
inline void wu_pixel(uint32_t x, uint32_t y, CRGB c) {}
|
||||||
#endif
|
#endif
|
||||||
} segment;
|
} segment;
|
||||||
//static int segSize = sizeof(Segment);
|
//static int segSize = sizeof(Segment);
|
||||||
@ -759,23 +754,22 @@ class WS2812FX { // 96 bytes
|
|||||||
setCCT(uint16_t k),
|
setCCT(uint16_t k),
|
||||||
setBrightness(uint8_t b, bool direct = false),
|
setBrightness(uint8_t b, bool direct = false),
|
||||||
setRange(uint16_t i, uint16_t i2, uint32_t col),
|
setRange(uint16_t i, uint16_t i2, uint32_t col),
|
||||||
setTransitionMode(bool t),
|
|
||||||
purgeSegments(bool force = false),
|
purgeSegments(bool force = false),
|
||||||
setSegment(uint8_t n, uint16_t start, uint16_t stop, uint8_t grouping = 1, uint8_t spacing = 0, uint16_t offset = UINT16_MAX, uint16_t startY=0, uint16_t stopY=1),
|
setSegment(uint8_t n, uint16_t start, uint16_t stop, uint8_t grouping = 1, uint8_t spacing = 0, uint16_t offset = UINT16_MAX, uint16_t startY=0, uint16_t stopY=1),
|
||||||
setMainSegmentId(uint8_t n),
|
setMainSegmentId(uint8_t n),
|
||||||
restartRuntime(),
|
|
||||||
resetSegments(),
|
resetSegments(),
|
||||||
makeAutoSegments(bool forceReset = false),
|
makeAutoSegments(bool forceReset = false),
|
||||||
fixInvalidSegments(),
|
fixInvalidSegments(),
|
||||||
setPixelColor(int n, uint32_t c),
|
setPixelColor(int n, uint32_t c),
|
||||||
show(void),
|
show(void),
|
||||||
setTargetFps(uint8_t fps);
|
setTargetFps(uint8_t fps),
|
||||||
|
addEffect(uint8_t id, mode_ptr mode_fn, const char *mode_name), // add effect to the list; defined in FX.cpp
|
||||||
void setColor(uint8_t slot, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0) { setColor(slot, RGBW32(r,g,b,w)); }
|
setupEffectData(void); // add default effects to the list; defined in FX.cpp
|
||||||
void fill(uint32_t c) { for (int i = 0; i < getLengthTotal(); i++) setPixelColor(i, c); } // fill whole strip with color (inline)
|
|
||||||
void addEffect(uint8_t id, mode_ptr mode_fn, const char *mode_name); // add effect to the list; defined in FX.cpp
|
|
||||||
void setupEffectData(void); // add default effects to the list; defined in FX.cpp
|
|
||||||
|
|
||||||
|
inline void restartRuntime() { for (Segment &seg : _segments) seg.markForReset(); }
|
||||||
|
inline void setTransitionMode(bool t) { for (Segment &seg : _segments) seg.startTransition(t ? _transitionDur : 0); }
|
||||||
|
inline void setColor(uint8_t slot, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0) { setColor(slot, RGBW32(r,g,b,w)); }
|
||||||
|
inline void fill(uint32_t c) { for (int i = 0; i < getLengthTotal(); i++) setPixelColor(i, c); } // fill whole strip with color (inline)
|
||||||
// outsmart the compiler :) by correctly overloading
|
// outsmart the compiler :) by correctly overloading
|
||||||
inline void setPixelColor(int n, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0) { setPixelColor(n, RGBW32(r,g,b,w)); }
|
inline void setPixelColor(int n, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0) { setPixelColor(n, RGBW32(r,g,b,w)); }
|
||||||
inline void setPixelColor(int n, CRGB c) { setPixelColor(n, c.red, c.green, c.blue); }
|
inline void setPixelColor(int n, CRGB c) { setPixelColor(n, c.red, c.green, c.blue); }
|
||||||
@ -881,16 +875,14 @@ class WS2812FX { // 96 bytes
|
|||||||
std::vector<Panel> panel;
|
std::vector<Panel> panel;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
void
|
void setUpMatrix();
|
||||||
setUpMatrix(),
|
|
||||||
setPixelColorXY(int x, int y, uint32_t c);
|
|
||||||
|
|
||||||
// outsmart the compiler :) by correctly overloading
|
// outsmart the compiler :) by correctly overloading
|
||||||
inline void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) { setPixelColorXY(x, y, RGBW32(r,g,b,w)); } // automatically inline
|
inline void setPixelColorXY(int x, int y, uint32_t c) { setPixelColor(y * Segment::maxWidth + x, c); }
|
||||||
inline void setPixelColorXY(int x, int y, CRGB c) { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0)); }
|
inline void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) { setPixelColorXY(x, y, RGBW32(r,g,b,w)); }
|
||||||
|
inline void setPixelColorXY(int x, int y, CRGB c) { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0)); }
|
||||||
|
|
||||||
uint32_t
|
inline uint32_t getPixelColorXY(uint16_t x, uint16_t y) { return getPixelColor(isMatrix ? y * Segment::maxWidth + x : x);}
|
||||||
getPixelColorXY(uint16_t, uint16_t);
|
|
||||||
|
|
||||||
// end 2D support
|
// end 2D support
|
||||||
|
|
||||||
|
@ -134,7 +134,7 @@ void WS2812FX::setUpMatrix() {
|
|||||||
|
|
||||||
#ifdef WLED_DEBUG
|
#ifdef WLED_DEBUG
|
||||||
DEBUG_PRINT(F("Matrix ledmap:"));
|
DEBUG_PRINT(F("Matrix ledmap:"));
|
||||||
for (uint16_t i=0; i<customMappingSize; i++) {
|
for (unsigned i=0; i<customMappingSize; i++) {
|
||||||
if (!(i%Segment::maxWidth)) DEBUG_PRINTLN();
|
if (!(i%Segment::maxWidth)) DEBUG_PRINTLN();
|
||||||
DEBUG_PRINTF("%4d,", customMappingTable[i]);
|
DEBUG_PRINTF("%4d,", customMappingTable[i]);
|
||||||
}
|
}
|
||||||
@ -155,31 +155,6 @@ void WS2812FX::setUpMatrix() {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
// absolute matrix version of setPixelColor()
|
|
||||||
void /*IRAM_ATTR*/ WS2812FX::setPixelColorXY(int x, int y, uint32_t col)
|
|
||||||
{
|
|
||||||
#ifndef WLED_DISABLE_2D
|
|
||||||
if (!isMatrix) return; // not a matrix set-up
|
|
||||||
uint16_t index = y * Segment::maxWidth + x;
|
|
||||||
#else
|
|
||||||
uint16_t index = x;
|
|
||||||
#endif
|
|
||||||
if (index < customMappingSize) index = customMappingTable[index];
|
|
||||||
if (index >= _length) return;
|
|
||||||
busses.setPixelColor(index, col);
|
|
||||||
}
|
|
||||||
|
|
||||||
// returns RGBW values of pixel
|
|
||||||
uint32_t WS2812FX::getPixelColorXY(uint16_t x, uint16_t y) {
|
|
||||||
#ifndef WLED_DISABLE_2D
|
|
||||||
uint16_t index = (y * Segment::maxWidth + x);
|
|
||||||
#else
|
|
||||||
uint16_t index = x;
|
|
||||||
#endif
|
|
||||||
if (index < customMappingSize) index = customMappingTable[index];
|
|
||||||
if (index >= _length) return 0;
|
|
||||||
return busses.getPixelColor(index);
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////
|
||||||
// Segment:: routines
|
// Segment:: routines
|
||||||
@ -188,18 +163,19 @@ uint32_t WS2812FX::getPixelColorXY(uint16_t x, uint16_t y) {
|
|||||||
#ifndef WLED_DISABLE_2D
|
#ifndef WLED_DISABLE_2D
|
||||||
|
|
||||||
// XY(x,y) - gets pixel index within current segment (often used to reference leds[] array element)
|
// XY(x,y) - gets pixel index within current segment (often used to reference leds[] array element)
|
||||||
uint16_t /*IRAM_ATTR*/ Segment::XY(uint16_t x, uint16_t y) {
|
uint16_t IRAM_ATTR Segment::XY(uint16_t x, uint16_t y)
|
||||||
|
{
|
||||||
uint16_t width = virtualWidth(); // segment width in logical pixels (can be 0 if segment is inactive)
|
uint16_t width = virtualWidth(); // segment width in logical pixels (can be 0 if segment is inactive)
|
||||||
uint16_t height = virtualHeight(); // segment height in logical pixels (is always >= 1)
|
uint16_t height = virtualHeight(); // segment height in logical pixels (is always >= 1)
|
||||||
return isActive() ? (x%width) + (y%height) * width : 0;
|
return isActive() ? (x%width) + (y%height) * width : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void /*IRAM_ATTR*/ Segment::setPixelColorXY(int x, int y, uint32_t col)
|
void IRAM_ATTR Segment::setPixelColorXY(int x, int y, uint32_t col)
|
||||||
{
|
{
|
||||||
if (!isActive()) return; // not active
|
if (!isActive()) return; // not active
|
||||||
if (x >= virtualWidth() || y >= virtualHeight() || x<0 || y<0) return; // if pixel would fall out of virtual segment just exit
|
if (x >= virtualWidth() || y >= virtualHeight() || x<0 || y<0) return; // if pixel would fall out of virtual segment just exit
|
||||||
|
|
||||||
uint8_t _bri_t = currentBri(on ? opacity : 0);
|
uint8_t _bri_t = currentBri();
|
||||||
if (_bri_t < 255) {
|
if (_bri_t < 255) {
|
||||||
byte r = scale8(R(col), _bri_t);
|
byte r = scale8(R(col), _bri_t);
|
||||||
byte g = scale8(G(col), _bri_t);
|
byte g = scale8(G(col), _bri_t);
|
||||||
@ -289,7 +265,7 @@ void Segment::setPixelColorXY(float x, float y, uint32_t col, bool aa)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// returns RGBW values of pixel
|
// returns RGBW values of pixel
|
||||||
uint32_t Segment::getPixelColorXY(uint16_t x, uint16_t y) {
|
uint32_t IRAM_ATTR Segment::getPixelColorXY(uint16_t x, uint16_t y) {
|
||||||
if (!isActive()) return 0; // not active
|
if (!isActive()) return 0; // not active
|
||||||
if (x >= virtualWidth() || y >= virtualHeight() || x<0 || y<0) return 0; // if pixel would fall out of virtual segment just exit
|
if (x >= virtualWidth() || y >= virtualHeight() || x<0 || y<0) return 0; // if pixel would fall out of virtual segment just exit
|
||||||
if (reverse ) x = virtualWidth() - x - 1;
|
if (reverse ) x = virtualWidth() - x - 1;
|
||||||
@ -301,41 +277,9 @@ uint32_t Segment::getPixelColorXY(uint16_t x, uint16_t y) {
|
|||||||
return strip.getPixelColorXY(start + x, startY + y);
|
return strip.getPixelColorXY(start + x, startY + y);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Blends the specified color with the existing pixel color.
|
|
||||||
void Segment::blendPixelColorXY(uint16_t x, uint16_t y, uint32_t color, uint8_t blend) {
|
|
||||||
setPixelColorXY(x, y, color_blend(getPixelColorXY(x,y), color, blend));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Adds the specified color with the existing pixel color perserving color balance.
|
|
||||||
void Segment::addPixelColorXY(int x, int y, uint32_t color, bool fast) {
|
|
||||||
if (!isActive()) return; // not active
|
|
||||||
if (x >= virtualWidth() || y >= virtualHeight() || x<0 || y<0) return; // if pixel would fall out of virtual segment just exit
|
|
||||||
uint32_t col = getPixelColorXY(x,y);
|
|
||||||
uint8_t r = R(col);
|
|
||||||
uint8_t g = G(col);
|
|
||||||
uint8_t b = B(col);
|
|
||||||
uint8_t w = W(col);
|
|
||||||
if (fast) {
|
|
||||||
r = qadd8(r, R(color));
|
|
||||||
g = qadd8(g, G(color));
|
|
||||||
b = qadd8(b, B(color));
|
|
||||||
w = qadd8(w, W(color));
|
|
||||||
col = RGBW32(r,g,b,w);
|
|
||||||
} else {
|
|
||||||
col = color_add(col, color);
|
|
||||||
}
|
|
||||||
setPixelColorXY(x, y, col);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Segment::fadePixelColorXY(uint16_t x, uint16_t y, uint8_t fade) {
|
|
||||||
if (!isActive()) return; // not active
|
|
||||||
CRGB pix = CRGB(getPixelColorXY(x,y)).nscale8_video(fade);
|
|
||||||
setPixelColorXY(x, y, pix);
|
|
||||||
}
|
|
||||||
|
|
||||||
// blurRow: perform a blur on a row of a rectangular matrix
|
// blurRow: perform a blur on a row of a rectangular matrix
|
||||||
void Segment::blurRow(uint16_t row, fract8 blur_amount) {
|
void Segment::blurRow(uint16_t row, fract8 blur_amount) {
|
||||||
if (!isActive()) return; // not active
|
if (!isActive() || blur_amount == 0) return; // not active
|
||||||
const uint_fast16_t cols = virtualWidth();
|
const uint_fast16_t cols = virtualWidth();
|
||||||
const uint_fast16_t rows = virtualHeight();
|
const uint_fast16_t rows = virtualHeight();
|
||||||
|
|
||||||
@ -344,7 +288,7 @@ void Segment::blurRow(uint16_t row, fract8 blur_amount) {
|
|||||||
uint8_t keep = 255 - blur_amount;
|
uint8_t keep = 255 - blur_amount;
|
||||||
uint8_t seep = blur_amount >> 1;
|
uint8_t seep = blur_amount >> 1;
|
||||||
CRGB carryover = CRGB::Black;
|
CRGB carryover = CRGB::Black;
|
||||||
for (uint_fast16_t x = 0; x < cols; x++) {
|
for (unsigned x = 0; x < cols; x++) {
|
||||||
CRGB cur = getPixelColorXY(x, row);
|
CRGB cur = getPixelColorXY(x, row);
|
||||||
CRGB before = cur; // remember color before blur
|
CRGB before = cur; // remember color before blur
|
||||||
CRGB part = cur;
|
CRGB part = cur;
|
||||||
@ -363,7 +307,7 @@ void Segment::blurRow(uint16_t row, fract8 blur_amount) {
|
|||||||
|
|
||||||
// blurCol: perform a blur on a column of a rectangular matrix
|
// blurCol: perform a blur on a column of a rectangular matrix
|
||||||
void Segment::blurCol(uint16_t col, fract8 blur_amount) {
|
void Segment::blurCol(uint16_t col, fract8 blur_amount) {
|
||||||
if (!isActive()) return; // not active
|
if (!isActive() || blur_amount == 0) return; // not active
|
||||||
const uint_fast16_t cols = virtualWidth();
|
const uint_fast16_t cols = virtualWidth();
|
||||||
const uint_fast16_t rows = virtualHeight();
|
const uint_fast16_t rows = virtualHeight();
|
||||||
|
|
||||||
@ -372,7 +316,7 @@ void Segment::blurCol(uint16_t col, fract8 blur_amount) {
|
|||||||
uint8_t keep = 255 - blur_amount;
|
uint8_t keep = 255 - blur_amount;
|
||||||
uint8_t seep = blur_amount >> 1;
|
uint8_t seep = blur_amount >> 1;
|
||||||
CRGB carryover = CRGB::Black;
|
CRGB carryover = CRGB::Black;
|
||||||
for (uint_fast16_t y = 0; y < rows; y++) {
|
for (unsigned y = 0; y < rows; y++) {
|
||||||
CRGB cur = getPixelColorXY(col, y);
|
CRGB cur = getPixelColorXY(col, y);
|
||||||
CRGB part = cur;
|
CRGB part = cur;
|
||||||
CRGB before = cur; // remember color before blur
|
CRGB before = cur; // remember color before blur
|
||||||
@ -391,7 +335,7 @@ void Segment::blurCol(uint16_t col, fract8 blur_amount) {
|
|||||||
|
|
||||||
// 1D Box blur (with added weight - blur_amount: [0=no blur, 255=max blur])
|
// 1D Box blur (with added weight - blur_amount: [0=no blur, 255=max blur])
|
||||||
void Segment::box_blur(uint16_t i, bool vertical, fract8 blur_amount) {
|
void Segment::box_blur(uint16_t i, bool vertical, fract8 blur_amount) {
|
||||||
if (!isActive()) return; // not active
|
if (!isActive() || blur_amount == 0) return; // not active
|
||||||
const uint16_t cols = virtualWidth();
|
const uint16_t cols = virtualWidth();
|
||||||
const uint16_t rows = virtualHeight();
|
const uint16_t rows = virtualHeight();
|
||||||
const uint16_t dim1 = vertical ? rows : cols;
|
const uint16_t dim1 = vertical ? rows : cols;
|
||||||
@ -401,7 +345,7 @@ void Segment::box_blur(uint16_t i, bool vertical, fract8 blur_amount) {
|
|||||||
const float keep = 3.f - 2.f*seep;
|
const float keep = 3.f - 2.f*seep;
|
||||||
// 1D box blur
|
// 1D box blur
|
||||||
CRGB tmp[dim1];
|
CRGB tmp[dim1];
|
||||||
for (uint16_t j = 0; j < dim1; j++) {
|
for (int j = 0; j < dim1; j++) {
|
||||||
uint16_t x = vertical ? i : j;
|
uint16_t x = vertical ? i : j;
|
||||||
uint16_t y = vertical ? j : i;
|
uint16_t y = vertical ? j : i;
|
||||||
int16_t xp = vertical ? x : x-1; // "signed" to prevent underflow
|
int16_t xp = vertical ? x : x-1; // "signed" to prevent underflow
|
||||||
@ -417,7 +361,7 @@ void Segment::box_blur(uint16_t i, bool vertical, fract8 blur_amount) {
|
|||||||
b = (curr.b*keep + (prev.b + next.b)*seep) / 3;
|
b = (curr.b*keep + (prev.b + next.b)*seep) / 3;
|
||||||
tmp[j] = CRGB(r,g,b);
|
tmp[j] = CRGB(r,g,b);
|
||||||
}
|
}
|
||||||
for (uint16_t j = 0; j < dim1; j++) {
|
for (int j = 0; j < dim1; j++) {
|
||||||
uint16_t x = vertical ? i : j;
|
uint16_t x = vertical ? i : j;
|
||||||
uint16_t y = vertical ? j : i;
|
uint16_t y = vertical ? j : i;
|
||||||
setPixelColorXY(x, y, tmp[j]);
|
setPixelColorXY(x, y, tmp[j]);
|
||||||
@ -440,7 +384,7 @@ void Segment::box_blur(uint16_t i, bool vertical, fract8 blur_amount) {
|
|||||||
|
|
||||||
void Segment::blur1d(fract8 blur_amount) {
|
void Segment::blur1d(fract8 blur_amount) {
|
||||||
const uint16_t rows = virtualHeight();
|
const uint16_t rows = virtualHeight();
|
||||||
for (uint16_t y = 0; y < rows; y++) blurRow(y, blur_amount);
|
for (unsigned y = 0; y < rows; y++) blurRow(y, blur_amount);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Segment::moveX(int8_t delta, bool wrap) {
|
void Segment::moveX(int8_t delta, bool wrap) {
|
||||||
@ -498,7 +442,7 @@ void Segment::move(uint8_t dir, uint8_t delta, bool wrap) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void Segment::draw_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB col) {
|
void Segment::draw_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB col) {
|
||||||
if (!isActive()) return; // not active
|
if (!isActive() || radius == 0) return; // not active
|
||||||
// Bresenham’s Algorithm
|
// Bresenham’s Algorithm
|
||||||
int d = 3 - (2*radius);
|
int d = 3 - (2*radius);
|
||||||
int y = radius, x = 0;
|
int y = radius, x = 0;
|
||||||
@ -523,7 +467,7 @@ void Segment::draw_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB col) {
|
|||||||
|
|
||||||
// by stepko, taken from https://editor.soulmatelights.com/gallery/573-blobs
|
// by stepko, taken from https://editor.soulmatelights.com/gallery/573-blobs
|
||||||
void Segment::fill_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB col) {
|
void Segment::fill_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB col) {
|
||||||
if (!isActive()) return; // not active
|
if (!isActive() || radius == 0) return; // not active
|
||||||
const uint16_t cols = virtualWidth();
|
const uint16_t cols = virtualWidth();
|
||||||
const uint16_t rows = virtualHeight();
|
const uint16_t rows = virtualHeight();
|
||||||
for (int16_t y = -radius; y <= radius; y++) {
|
for (int16_t y = -radius; y <= radius; y++) {
|
||||||
@ -540,7 +484,7 @@ void Segment::nscale8(uint8_t scale) {
|
|||||||
if (!isActive()) return; // not active
|
if (!isActive()) return; // not active
|
||||||
const uint16_t cols = virtualWidth();
|
const uint16_t cols = virtualWidth();
|
||||||
const uint16_t rows = virtualHeight();
|
const uint16_t rows = virtualHeight();
|
||||||
for(uint16_t y = 0; y < rows; y++) for (uint16_t x = 0; x < cols; x++) {
|
for (int y = 0; y < rows; y++) for (int x = 0; x < cols; x++) {
|
||||||
setPixelColorXY(x, y, CRGB(getPixelColorXY(x, y)).nscale8(scale));
|
setPixelColorXY(x, y, CRGB(getPixelColorXY(x, y)).nscale8(scale));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -89,25 +89,19 @@ bool Segment::_modeBlend = false;
|
|||||||
Segment::Segment(const Segment &orig) {
|
Segment::Segment(const Segment &orig) {
|
||||||
//DEBUG_PRINTF("-- Copy segment constructor: %p -> %p\n", &orig, this);
|
//DEBUG_PRINTF("-- Copy segment constructor: %p -> %p\n", &orig, this);
|
||||||
memcpy((void*)this, (void*)&orig, sizeof(Segment));
|
memcpy((void*)this, (void*)&orig, sizeof(Segment));
|
||||||
transitional = false; // copied segment cannot be in transition
|
_t = nullptr; // copied segment cannot be in transition
|
||||||
name = nullptr;
|
if (orig.name) { name = new char[strlen(orig.name)+1]; if (name) strcpy(name, orig.name); } else { name = nullptr; }
|
||||||
data = nullptr;
|
if (orig.data) { if (allocateData(orig._dataLen)) memcpy(data, orig.data, orig._dataLen); } else { data = nullptr; _dataLen = 0; }
|
||||||
_dataLen = 0;
|
|
||||||
_t = nullptr;
|
|
||||||
if (orig.name) { name = new char[strlen(orig.name)+1]; if (name) strcpy(name, orig.name); }
|
|
||||||
if (orig.data) { if (allocateData(orig._dataLen)) memcpy(data, orig.data, orig._dataLen); }
|
|
||||||
//if (orig._t) { _t = new Transition(orig._t->_dur); }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// move constructor
|
// move constructor
|
||||||
Segment::Segment(Segment &&orig) noexcept {
|
Segment::Segment(Segment &&orig) noexcept {
|
||||||
//DEBUG_PRINTF("-- Move segment constructor: %p -> %p\n", &orig, this);
|
//DEBUG_PRINTF("-- Move segment constructor: %p -> %p\n", &orig, this);
|
||||||
memcpy((void*)this, (void*)&orig, sizeof(Segment));
|
memcpy((void*)this, (void*)&orig, sizeof(Segment));
|
||||||
orig.transitional = false; // old segment cannot be in transition any more
|
|
||||||
orig.name = nullptr;
|
orig.name = nullptr;
|
||||||
orig.data = nullptr;
|
orig.data = nullptr;
|
||||||
orig._dataLen = 0;
|
orig._dataLen = 0;
|
||||||
orig._t = nullptr;
|
orig._t = nullptr; // old segment cannot be in transition any more
|
||||||
}
|
}
|
||||||
|
|
||||||
// copy assignment
|
// copy assignment
|
||||||
@ -115,27 +109,23 @@ Segment& Segment::operator= (const Segment &orig) {
|
|||||||
//DEBUG_PRINTF("-- Copying segment: %p -> %p\n", &orig, this);
|
//DEBUG_PRINTF("-- Copying segment: %p -> %p\n", &orig, this);
|
||||||
if (this != &orig) {
|
if (this != &orig) {
|
||||||
// clean destination
|
// clean destination
|
||||||
transitional = false; // copied segment cannot be in transition
|
if (name) { delete[] name; name = nullptr; }
|
||||||
if (name) delete[] name;
|
if (orig.name) { name = new char[strlen(orig.name)+1]; if (name) strcpy(name, orig.name); }
|
||||||
if (_t) {
|
if (_t) {
|
||||||
#ifndef WLED_DISABLE_MODE_BLEND
|
#ifndef WLED_DISABLE_MODE_BLEND
|
||||||
if (_t->_segT._dataT) free(_t->_segT._dataT);
|
if (_t->_segT._dataT) free(_t->_segT._dataT);
|
||||||
#endif
|
#endif
|
||||||
delete _t;
|
delete _t;
|
||||||
|
_t = nullptr; // copied segment cannot be in transition
|
||||||
}
|
}
|
||||||
deallocateData();
|
deallocateData();
|
||||||
// copy source
|
// copy source
|
||||||
memcpy((void*)this, (void*)&orig, sizeof(Segment));
|
memcpy((void*)this, (void*)&orig, sizeof(Segment));
|
||||||
transitional = false;
|
|
||||||
// erase pointers to allocated data
|
// erase pointers to allocated data
|
||||||
name = nullptr;
|
|
||||||
data = nullptr;
|
data = nullptr;
|
||||||
_dataLen = 0;
|
_dataLen = 0;
|
||||||
_t = nullptr;
|
|
||||||
// copy source data
|
// copy source data
|
||||||
if (orig.name) { name = new char[strlen(orig.name)+1]; if (name) strcpy(name, orig.name); }
|
|
||||||
if (orig.data) { if (allocateData(orig._dataLen)) memcpy(data, orig.data, orig._dataLen); }
|
if (orig.data) { if (allocateData(orig._dataLen)) memcpy(data, orig.data, orig._dataLen); }
|
||||||
//if (orig._t) { _t = new Transition(orig._t->_dur, orig._t->_briT, orig._t->_cctT, orig._t->_colorT); }
|
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
@ -144,9 +134,7 @@ Segment& Segment::operator= (const Segment &orig) {
|
|||||||
Segment& Segment::operator= (Segment &&orig) noexcept {
|
Segment& Segment::operator= (Segment &&orig) noexcept {
|
||||||
//DEBUG_PRINTF("-- Moving segment: %p -> %p\n", &orig, this);
|
//DEBUG_PRINTF("-- Moving segment: %p -> %p\n", &orig, this);
|
||||||
if (this != &orig) {
|
if (this != &orig) {
|
||||||
transitional = false; // just temporary
|
|
||||||
if (name) { delete[] name; name = nullptr; } // free old name
|
if (name) { delete[] name; name = nullptr; } // free old name
|
||||||
deallocateData(); // free old runtime data
|
|
||||||
if (_t) {
|
if (_t) {
|
||||||
#ifndef WLED_DISABLE_MODE_BLEND
|
#ifndef WLED_DISABLE_MODE_BLEND
|
||||||
if (_t->_segT._dataT) free(_t->_segT._dataT);
|
if (_t->_segT._dataT) free(_t->_segT._dataT);
|
||||||
@ -154,17 +142,17 @@ Segment& Segment::operator= (Segment &&orig) noexcept {
|
|||||||
delete _t;
|
delete _t;
|
||||||
_t = nullptr;
|
_t = nullptr;
|
||||||
}
|
}
|
||||||
|
deallocateData(); // free old runtime data
|
||||||
memcpy((void*)this, (void*)&orig, sizeof(Segment));
|
memcpy((void*)this, (void*)&orig, sizeof(Segment));
|
||||||
orig.transitional = false; // old segment cannot be in transition
|
|
||||||
orig.name = nullptr;
|
orig.name = nullptr;
|
||||||
orig.data = nullptr;
|
orig.data = nullptr;
|
||||||
orig._dataLen = 0;
|
orig._dataLen = 0;
|
||||||
orig._t = nullptr;
|
orig._t = nullptr; // old segment cannot be in transition
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Segment::allocateData(size_t len) {
|
bool IRAM_ATTR Segment::allocateData(size_t len) {
|
||||||
if (data && _dataLen == len) return true; //already allocated
|
if (data && _dataLen == len) return true; //already allocated
|
||||||
//DEBUG_PRINTF("-- Allocating data (%d): %p\n", len, this);
|
//DEBUG_PRINTF("-- Allocating data (%d): %p\n", len, this);
|
||||||
deallocateData();
|
deallocateData();
|
||||||
@ -185,7 +173,7 @@ bool Segment::allocateData(size_t len) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Segment::deallocateData() {
|
void IRAM_ATTR Segment::deallocateData() {
|
||||||
if (!data) { _dataLen = 0; return; }
|
if (!data) { _dataLen = 0; return; }
|
||||||
//DEBUG_PRINTF("--- Released data (%p): %d/%d -> %p\n", this, _dataLen, Segment::getUsedSegmentData(), data);
|
//DEBUG_PRINTF("--- Released data (%p): %d/%d -> %p\n", this, _dataLen, Segment::getUsedSegmentData(), data);
|
||||||
if ((Segment::getUsedSegmentData() > 0) && (_dataLen > 0)) { // check that we don't have a dangling / inconsistent data pointer
|
if ((Segment::getUsedSegmentData() > 0) && (_dataLen > 0)) { // check that we don't have a dangling / inconsistent data pointer
|
||||||
@ -217,7 +205,7 @@ void Segment::resetIfRequired() {
|
|||||||
reset = false;
|
reset = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
CRGBPalette16 &Segment::loadPalette(CRGBPalette16 &targetPalette, uint8_t pal) {
|
CRGBPalette16 IRAM_ATTR &Segment::loadPalette(CRGBPalette16 &targetPalette, uint8_t pal) {
|
||||||
if (pal < 245 && pal > GRADIENT_PALETTE_COUNT+13) pal = 0;
|
if (pal < 245 && pal > GRADIENT_PALETTE_COUNT+13) pal = 0;
|
||||||
if (pal > 245 && (strip.customPalettes.size() == 0 || 255U-pal > strip.customPalettes.size()-1)) pal = 0;
|
if (pal > 245 && (strip.customPalettes.size() == 0 || 255U-pal > strip.customPalettes.size()-1)) pal = 0;
|
||||||
//default palette. Differs depending on effect
|
//default palette. Differs depending on effect
|
||||||
@ -237,7 +225,7 @@ CRGBPalette16 &Segment::loadPalette(CRGBPalette16 &targetPalette, uint8_t pal) {
|
|||||||
switch (pal) {
|
switch (pal) {
|
||||||
case 0: //default palette. Exceptions for specific effects above
|
case 0: //default palette. Exceptions for specific effects above
|
||||||
targetPalette = PartyColors_p; break;
|
targetPalette = PartyColors_p; break;
|
||||||
case 1: {//periodically replace palette with a random one. Transition palette change in 500ms
|
case 1: {//periodically replace palette with a random one
|
||||||
unsigned long timeSinceLastChange = millis() - _lastPaletteChange;
|
unsigned long timeSinceLastChange = millis() - _lastPaletteChange;
|
||||||
if (timeSinceLastChange > randomPaletteChangeTime * 1000U) {
|
if (timeSinceLastChange > randomPaletteChangeTime * 1000U) {
|
||||||
_randomPalette = _newRandomPalette;
|
_randomPalette = _newRandomPalette;
|
||||||
@ -301,47 +289,45 @@ CRGBPalette16 &Segment::loadPalette(CRGBPalette16 &targetPalette, uint8_t pal) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void Segment::startTransition(uint16_t dur) {
|
void Segment::startTransition(uint16_t dur) {
|
||||||
if (!dur) {
|
if (dur == 0) {
|
||||||
if (_t) _t->_dur = dur; // this will stop transition in next handleTransisiton()
|
if (isInTransition()) _t->_dur = dur; // this will stop transition in next handleTransisiton()
|
||||||
else transitional = false;
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (transitional && _t) return; // already in transition no need to store anything
|
if (isInTransition()) return; // already in transition no need to store anything
|
||||||
|
|
||||||
// starting a transition has to occur before change so we get current values 1st
|
// starting a transition has to occur before change so we get current values 1st
|
||||||
_t = new Transition(dur); // no previous transition running
|
_t = new Transition(dur); // no previous transition running
|
||||||
if (!_t) return; // failed to allocate data
|
if (!_t) return; // failed to allocate data
|
||||||
|
|
||||||
//DEBUG_PRINTF("-- Started transition: %p\n", this);
|
//DEBUG_PRINTF("-- Started transition: %p\n", this);
|
||||||
CRGBPalette16 _palT = CRGBPalette16(DEFAULT_COLOR); loadPalette(_palT, palette);
|
loadPalette(_t->_palT, palette);
|
||||||
_t->_palT = _palT;
|
|
||||||
_t->_briT = on ? opacity : 0;
|
_t->_briT = on ? opacity : 0;
|
||||||
_t->_cctT = cct;
|
_t->_cctT = cct;
|
||||||
#ifndef WLED_DISABLE_MODE_BLEND
|
#ifndef WLED_DISABLE_MODE_BLEND
|
||||||
swapSegenv(_t->_segT);
|
if (modeBlending) {
|
||||||
_t->_modeT = mode;
|
swapSegenv(_t->_segT);
|
||||||
_t->_segT._optionsT |= 0b0000000001000000; // mark old segment transitional
|
_t->_modeT = mode;
|
||||||
_t->_segT._dataLenT = 0;
|
_t->_segT._dataLenT = 0;
|
||||||
_t->_segT._dataT = nullptr;
|
_t->_segT._dataT = nullptr;
|
||||||
if (_dataLen > 0 && data) {
|
if (_dataLen > 0 && data) {
|
||||||
_t->_segT._dataT = (byte *)malloc(_dataLen);
|
_t->_segT._dataT = (byte *)malloc(_dataLen);
|
||||||
if (_t->_segT._dataT) {
|
if (_t->_segT._dataT) {
|
||||||
//DEBUG_PRINTF("-- Allocated duplicate data (%d): %p\n", _dataLen, _t->_segT._dataT);
|
//DEBUG_PRINTF("-- Allocated duplicate data (%d): %p\n", _dataLen, _t->_segT._dataT);
|
||||||
memcpy(_t->_segT._dataT, data, _dataLen);
|
memcpy(_t->_segT._dataT, data, _dataLen);
|
||||||
_t->_segT._dataLenT = _dataLen;
|
_t->_segT._dataLenT = _dataLen;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
for (size_t i=0; i<NUM_COLORS; i++) _t->_segT._colorT[i] = colors[i];
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
for (size_t i=0; i<NUM_COLORS; i++) _t->_colorT[i] = colors[i];
|
for (size_t i=0; i<NUM_COLORS; i++) _t->_colorT[i] = colors[i];
|
||||||
#endif
|
#endif
|
||||||
transitional = true; // setOption(SEG_OPTION_TRANSITIONAL, true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Segment::stopTransition() {
|
void Segment::stopTransition() {
|
||||||
if (!transitional) return;
|
|
||||||
transitional = false; // finish transitioning segment
|
|
||||||
//DEBUG_PRINTF("-- Stopping transition: %p\n", this);
|
//DEBUG_PRINTF("-- Stopping transition: %p\n", this);
|
||||||
if (_t) {
|
if (isInTransition()) {
|
||||||
#ifndef WLED_DISABLE_MODE_BLEND
|
#ifndef WLED_DISABLE_MODE_BLEND
|
||||||
if (_t->_segT._dataT && _t->_segT._dataLenT > 0) {
|
if (_t->_segT._dataT && _t->_segT._dataLenT > 0) {
|
||||||
//DEBUG_PRINTF("-- Released duplicate data (%d): %p\n", _t->_segT._dataLenT, _t->_segT._dataT);
|
//DEBUG_PRINTF("-- Released duplicate data (%d): %p\n", _t->_segT._dataLenT, _t->_segT._dataT);
|
||||||
@ -356,14 +342,13 @@ void Segment::stopTransition() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void Segment::handleTransition() {
|
void Segment::handleTransition() {
|
||||||
if (!transitional) return;
|
|
||||||
uint16_t _progress = progress();
|
uint16_t _progress = progress();
|
||||||
if (_progress == 0xFFFFU) stopTransition();
|
if (_progress == 0xFFFFU) stopTransition();
|
||||||
}
|
}
|
||||||
|
|
||||||
// transition progression between 0-65535
|
// transition progression between 0-65535
|
||||||
uint16_t Segment::progress() {
|
uint16_t IRAM_ATTR Segment::progress() {
|
||||||
if (transitional && _t) {
|
if (isInTransition()) {
|
||||||
unsigned long timeNow = millis();
|
unsigned long timeNow = millis();
|
||||||
if (_t->_dur > 0 && timeNow - _t->_start < _t->_dur) return (timeNow - _t->_start) * 0xFFFFU / _t->_dur;
|
if (_t->_dur > 0 && timeNow - _t->_start < _t->_dur) return (timeNow - _t->_start) * 0xFFFFU / _t->_dur;
|
||||||
}
|
}
|
||||||
@ -420,8 +405,8 @@ void Segment::restoreSegenv(tmpsegd_t &tmpSeg) {
|
|||||||
_t->_segT._stepT = step;
|
_t->_segT._stepT = step;
|
||||||
_t->_segT._callT = call;
|
_t->_segT._callT = call;
|
||||||
//if (_t->_segT._dataT != data) DEBUG_PRINTF("--- data re-allocated: (%p) %p -> %p\n", this, _t->_segT._dataT, data);
|
//if (_t->_segT._dataT != data) DEBUG_PRINTF("--- data re-allocated: (%p) %p -> %p\n", this, _t->_segT._dataT, data);
|
||||||
_t->_segT._dataT = data; // sometimes memory gets re-allocated (!! INVESTIGATE WHY !!)
|
_t->_segT._dataT = data;
|
||||||
_t->_segT._dataLenT = _dataLen; // sometimes memory gets re-allocated (!! INVESTIGATE WHY !!)
|
_t->_segT._dataLenT = _dataLen;
|
||||||
}
|
}
|
||||||
options = tmpSeg._optionsT;
|
options = tmpSeg._optionsT;
|
||||||
for (size_t i=0; i<NUM_COLORS; i++) colors[i] = tmpSeg._colorT[i];
|
for (size_t i=0; i<NUM_COLORS; i++) colors[i] = tmpSeg._colorT[i];
|
||||||
@ -443,39 +428,40 @@ void Segment::restoreSegenv(tmpsegd_t &tmpSeg) {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
uint8_t Segment::currentBri(uint8_t briNew, bool useCct) {
|
uint8_t IRAM_ATTR Segment::currentBri(bool useCct) {
|
||||||
uint32_t prog = progress();
|
uint32_t prog = progress();
|
||||||
if (prog < 0xFFFFU) {
|
if (prog < 0xFFFFU) {
|
||||||
if (useCct) return ((briNew * prog) + _t->_cctT * (0xFFFFU - prog)) >> 16;
|
uint32_t curBri = (useCct ? cct : (on ? opacity : 0)) * prog;
|
||||||
else return ((briNew * prog) + _t->_briT * (0xFFFFU - prog)) >> 16;
|
curBri += (useCct ? _t->_cctT : (on ? _t->_briT : 0)) * (0xFFFFU - prog);
|
||||||
|
return curBri / 0xFFFFU;
|
||||||
}
|
}
|
||||||
return briNew;
|
return (useCct ? cct : (on ? opacity : 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t Segment::currentMode(uint8_t newMode) {
|
uint8_t IRAM_ATTR Segment::currentMode() {
|
||||||
#ifndef WLED_DISABLE_MODE_BLEND
|
#ifndef WLED_DISABLE_MODE_BLEND
|
||||||
uint16_t prog = progress(); // implicit check for transitional & _t in progress()
|
uint16_t prog = progress();
|
||||||
if (prog < 0xFFFFU) return _t->_modeT;
|
if (modeBlending && prog < 0xFFFFU) return _t->_modeT;
|
||||||
#endif
|
#endif
|
||||||
return newMode;
|
return mode;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t Segment::currentColor(uint8_t slot, uint32_t colorNew) {
|
uint32_t IRAM_ATTR Segment::currentColor(uint8_t slot) {
|
||||||
#ifndef WLED_DISABLE_MODE_BLEND
|
#ifndef WLED_DISABLE_MODE_BLEND
|
||||||
return transitional && _t ? color_blend(_t->_segT._colorT[slot], colorNew, progress(), true) : colorNew;
|
return isInTransition() ? color_blend(_t->_segT._colorT[slot], colors[slot], progress(), true) : colors[slot];
|
||||||
#else
|
#else
|
||||||
return transitional && _t ? color_blend(_t->_colorT[slot], colorNew, progress(), true) : colorNew;
|
return isInTransition() ? color_blend(_t->_colorT[slot], colors[slot], progress(), true) : colors[slot];
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
CRGBPalette16 &Segment::currentPalette(CRGBPalette16 &targetPalette, uint8_t pal) {
|
CRGBPalette16 IRAM_ATTR &Segment::currentPalette(CRGBPalette16 &targetPalette, uint8_t pal) {
|
||||||
loadPalette(targetPalette, pal);
|
loadPalette(targetPalette, pal);
|
||||||
if (progress() < 0xFFFFU) {
|
uint16_t prog = progress();
|
||||||
|
if (strip.paletteFade && prog < 0xFFFFU) {
|
||||||
// blend palettes
|
// blend palettes
|
||||||
// there are about 255 blend passes of 48 "blends" to completely blend two palettes (in _dur time)
|
// there are about 255 blend passes of 48 "blends" to completely blend two palettes (in _dur time)
|
||||||
// minimum blend time is 100ms maximum is 65535ms
|
// minimum blend time is 100ms maximum is 65535ms
|
||||||
unsigned long timeMS = millis() - _t->_start;
|
uint16_t noOfBlends = ((255U * prog) / 0xFFFFU) - _t->_prevPaletteBlends;
|
||||||
uint16_t noOfBlends = (255U * timeMS / _t->_dur) - _t->_prevPaletteBlends;
|
|
||||||
for (int i=0; i<noOfBlends; i++, _t->_prevPaletteBlends++) nblendPaletteTowardPalette(_t->_palT, targetPalette, 48);
|
for (int i=0; i<noOfBlends; i++, _t->_prevPaletteBlends++) nblendPaletteTowardPalette(_t->_palT, targetPalette, 48);
|
||||||
targetPalette = _t->_palT; // copy transitioning/temporary palette
|
targetPalette = _t->_palT; // copy transitioning/temporary palette
|
||||||
}
|
}
|
||||||
@ -500,6 +486,8 @@ void Segment::setUp(uint16_t i1, uint16_t i2, uint8_t grp, uint8_t spc, uint16_t
|
|||||||
&& (!grp || (grouping == grp && spacing == spc))
|
&& (!grp || (grouping == grp && spacing == spc))
|
||||||
&& (ofs == UINT16_MAX || ofs == offset)) return;
|
&& (ofs == UINT16_MAX || ofs == offset)) return;
|
||||||
|
|
||||||
|
stateChanged = true; // send UDP/WS broadcast
|
||||||
|
|
||||||
if (stop) fill(BLACK); // turn old segment range off (clears pixels if changing spacing)
|
if (stop) fill(BLACK); // turn old segment range off (clears pixels if changing spacing)
|
||||||
if (grp) { // prevent assignment of 0
|
if (grp) { // prevent assignment of 0
|
||||||
grouping = grp;
|
grouping = grp;
|
||||||
@ -510,6 +498,10 @@ void Segment::setUp(uint16_t i1, uint16_t i2, uint8_t grp, uint8_t spc, uint16_t
|
|||||||
}
|
}
|
||||||
if (ofs < UINT16_MAX) offset = ofs;
|
if (ofs < UINT16_MAX) offset = ofs;
|
||||||
|
|
||||||
|
DEBUG_PRINT(F("setUp segment: ")); DEBUG_PRINT(i1);
|
||||||
|
DEBUG_PRINT(','); DEBUG_PRINT(i2);
|
||||||
|
DEBUG_PRINT(F(" -> ")); DEBUG_PRINT(i1Y);
|
||||||
|
DEBUG_PRINT(','); DEBUG_PRINTLN(i2Y);
|
||||||
markForReset();
|
markForReset();
|
||||||
if (boundsUnchanged) return;
|
if (boundsUnchanged) return;
|
||||||
|
|
||||||
@ -564,7 +556,6 @@ void Segment::setCCT(uint16_t k) {
|
|||||||
void Segment::setOpacity(uint8_t o) {
|
void Segment::setOpacity(uint8_t o) {
|
||||||
if (opacity == o) return;
|
if (opacity == o) return;
|
||||||
if (fadeTransition) startTransition(strip.getTransition()); // start transition prior to change
|
if (fadeTransition) startTransition(strip.getTransition()); // start transition prior to change
|
||||||
DEBUG_PRINT(F("-- Setting opacity: ")); DEBUG_PRINTLN(o);
|
|
||||||
opacity = o;
|
opacity = o;
|
||||||
stateChanged = true; // send UDP/WS broadcast
|
stateChanged = true; // send UDP/WS broadcast
|
||||||
}
|
}
|
||||||
@ -574,14 +565,16 @@ void Segment::setOption(uint8_t n, bool val) {
|
|||||||
if (fadeTransition && n == SEG_OPTION_ON && val != prevOn) startTransition(strip.getTransition()); // start transition prior to change
|
if (fadeTransition && n == SEG_OPTION_ON && val != prevOn) startTransition(strip.getTransition()); // start transition prior to change
|
||||||
if (val) options |= 0x01 << n;
|
if (val) options |= 0x01 << n;
|
||||||
else options &= ~(0x01 << n);
|
else options &= ~(0x01 << n);
|
||||||
if (!(n == SEG_OPTION_SELECTED || n == SEG_OPTION_RESET || n == SEG_OPTION_TRANSITIONAL)) stateChanged = true; // send UDP/WS broadcast
|
if (!(n == SEG_OPTION_SELECTED || n == SEG_OPTION_RESET)) stateChanged = true; // send UDP/WS broadcast
|
||||||
}
|
}
|
||||||
|
|
||||||
void Segment::setMode(uint8_t fx, bool loadDefaults) {
|
void Segment::setMode(uint8_t fx, bool loadDefaults) {
|
||||||
// if we have a valid mode & is not reserved
|
// if we have a valid mode & is not reserved
|
||||||
if (fx < strip.getModeCount() && strncmp_P("RSVD", strip.getModeData(fx), 4)) {
|
if (fx < strip.getModeCount() && strncmp_P("RSVD", strip.getModeData(fx), 4)) {
|
||||||
if (fx != mode) {
|
if (fx != mode) {
|
||||||
if (fadeTransition) startTransition(strip.getTransition()); // set effect transitions
|
#ifndef WLED_DISABLE_MODE_BLEND
|
||||||
|
if (modeBlending) startTransition(strip.getTransition()); // set effect transitions
|
||||||
|
#endif
|
||||||
mode = fx;
|
mode = fx;
|
||||||
// load default values from effect string
|
// load default values from effect string
|
||||||
if (loadDefaults) {
|
if (loadDefaults) {
|
||||||
@ -619,21 +612,21 @@ void Segment::setPalette(uint8_t pal) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 2D matrix
|
// 2D matrix
|
||||||
uint16_t Segment::virtualWidth() const {
|
uint16_t IRAM_ATTR Segment::virtualWidth() const {
|
||||||
uint16_t groupLen = groupLength();
|
uint16_t groupLen = groupLength();
|
||||||
uint16_t vWidth = ((transpose ? height() : width()) + groupLen - 1) / groupLen;
|
uint16_t vWidth = ((transpose ? height() : width()) + groupLen - 1) / groupLen;
|
||||||
if (mirror) vWidth = (vWidth + 1) /2; // divide by 2 if mirror, leave at least a single LED
|
if (mirror) vWidth = (vWidth + 1) /2; // divide by 2 if mirror, leave at least a single LED
|
||||||
return vWidth;
|
return vWidth;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint16_t Segment::virtualHeight() const {
|
uint16_t IRAM_ATTR Segment::virtualHeight() const {
|
||||||
uint16_t groupLen = groupLength();
|
uint16_t groupLen = groupLength();
|
||||||
uint16_t vHeight = ((transpose ? width() : height()) + groupLen - 1) / groupLen;
|
uint16_t vHeight = ((transpose ? width() : height()) + groupLen - 1) / groupLen;
|
||||||
if (mirror_y) vHeight = (vHeight + 1) /2; // divide by 2 if mirror, leave at least a single LED
|
if (mirror_y) vHeight = (vHeight + 1) /2; // divide by 2 if mirror, leave at least a single LED
|
||||||
return vHeight;
|
return vHeight;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint16_t Segment::nrOfVStrips() const {
|
uint16_t IRAM_ATTR Segment::nrOfVStrips() const {
|
||||||
uint16_t vLen = 1;
|
uint16_t vLen = 1;
|
||||||
#ifndef WLED_DISABLE_2D
|
#ifndef WLED_DISABLE_2D
|
||||||
if (is2D()) {
|
if (is2D()) {
|
||||||
@ -648,7 +641,7 @@ uint16_t Segment::nrOfVStrips() const {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 1D strip
|
// 1D strip
|
||||||
uint16_t Segment::virtualLength() const {
|
uint16_t IRAM_ATTR Segment::virtualLength() const {
|
||||||
#ifndef WLED_DISABLE_2D
|
#ifndef WLED_DISABLE_2D
|
||||||
if (is2D()) {
|
if (is2D()) {
|
||||||
uint16_t vW = virtualWidth();
|
uint16_t vW = virtualWidth();
|
||||||
@ -743,7 +736,7 @@ void IRAM_ATTR Segment::setPixelColor(int i, uint32_t col)
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
uint16_t len = length();
|
uint16_t len = length();
|
||||||
uint8_t _bri_t = currentBri(on ? opacity : 0);
|
uint8_t _bri_t = currentBri();
|
||||||
if (_bri_t < 255) {
|
if (_bri_t < 255) {
|
||||||
byte r = scale8(R(col), _bri_t);
|
byte r = scale8(R(col), _bri_t);
|
||||||
byte g = scale8(G(col), _bri_t);
|
byte g = scale8(G(col), _bri_t);
|
||||||
@ -820,7 +813,7 @@ void Segment::setPixelColor(float i, uint32_t col, bool aa)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t Segment::getPixelColor(int i)
|
uint32_t IRAM_ATTR Segment::getPixelColor(int i)
|
||||||
{
|
{
|
||||||
if (!isActive()) return 0; // not active
|
if (!isActive()) return 0; // not active
|
||||||
#ifndef WLED_DISABLE_2D
|
#ifndef WLED_DISABLE_2D
|
||||||
@ -877,10 +870,11 @@ uint8_t Segment::differs(Segment& b) const {
|
|||||||
if (startY != b.startY) d |= SEG_DIFFERS_BOUNDS;
|
if (startY != b.startY) d |= SEG_DIFFERS_BOUNDS;
|
||||||
if (stopY != b.stopY) d |= SEG_DIFFERS_BOUNDS;
|
if (stopY != b.stopY) d |= SEG_DIFFERS_BOUNDS;
|
||||||
|
|
||||||
//bit pattern: (msb first) set:2, sound:1, mapping:3, transposed, mirrorY, reverseY, [transitional, reset,] paused, mirrored, on, reverse, [selected]
|
//bit pattern: (msb first)
|
||||||
if ((options & 0b1111111110011110U) != (b.options & 0b1111111110011110U)) d |= SEG_DIFFERS_OPT;
|
// set:2, sound:2, mapping:3, transposed, mirrorY, reverseY, [reset,] paused, mirrored, on, reverse, [selected]
|
||||||
|
if ((options & 0b1111111111011110U) != (b.options & 0b1111111111011110U)) d |= SEG_DIFFERS_OPT;
|
||||||
if ((options & 0x0001U) != (b.options & 0x0001U)) d |= SEG_DIFFERS_SEL;
|
if ((options & 0x0001U) != (b.options & 0x0001U)) d |= SEG_DIFFERS_SEL;
|
||||||
for (uint8_t i = 0; i < NUM_COLORS; i++) if (colors[i] != b.colors[i]) d |= SEG_DIFFERS_COL;
|
for (unsigned i = 0; i < NUM_COLORS; i++) if (colors[i] != b.colors[i]) d |= SEG_DIFFERS_COL;
|
||||||
|
|
||||||
return d;
|
return d;
|
||||||
}
|
}
|
||||||
@ -912,7 +906,7 @@ void Segment::refreshLightCapabilities() {
|
|||||||
segStopIdx = stop;
|
segStopIdx = stop;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (uint8_t b = 0; b < busses.getNumBusses(); b++) {
|
for (unsigned b = 0; b < busses.getNumBusses(); b++) {
|
||||||
Bus *bus = busses.getBus(b);
|
Bus *bus = busses.getBus(b);
|
||||||
if (bus == nullptr || bus->getLength()==0) break;
|
if (bus == nullptr || bus->getLength()==0) break;
|
||||||
if (!bus->isOk()) continue;
|
if (!bus->isOk()) continue;
|
||||||
@ -942,43 +936,12 @@ void Segment::fill(uint32_t c) {
|
|||||||
if (!isActive()) return; // not active
|
if (!isActive()) return; // not active
|
||||||
const uint16_t cols = is2D() ? virtualWidth() : virtualLength();
|
const uint16_t cols = is2D() ? virtualWidth() : virtualLength();
|
||||||
const uint16_t rows = virtualHeight(); // will be 1 for 1D
|
const uint16_t rows = virtualHeight(); // will be 1 for 1D
|
||||||
for(uint16_t y = 0; y < rows; y++) for (uint16_t x = 0; x < cols; x++) {
|
for (int y = 0; y < rows; y++) for (int x = 0; x < cols; x++) {
|
||||||
if (is2D()) setPixelColorXY(x, y, c);
|
if (is2D()) setPixelColorXY(x, y, c);
|
||||||
else setPixelColor(x, c);
|
else setPixelColor(x, c);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Blends the specified color with the existing pixel color.
|
|
||||||
void Segment::blendPixelColor(int n, uint32_t color, uint8_t blend) {
|
|
||||||
setPixelColor(n, color_blend(getPixelColor(n), color, blend));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Adds the specified color with the existing pixel color perserving color balance.
|
|
||||||
void Segment::addPixelColor(int n, uint32_t color, bool fast) {
|
|
||||||
if (!isActive()) return; // not active
|
|
||||||
uint32_t col = getPixelColor(n);
|
|
||||||
uint8_t r = R(col);
|
|
||||||
uint8_t g = G(col);
|
|
||||||
uint8_t b = B(col);
|
|
||||||
uint8_t w = W(col);
|
|
||||||
if (fast) {
|
|
||||||
r = qadd8(r, R(color));
|
|
||||||
g = qadd8(g, G(color));
|
|
||||||
b = qadd8(b, B(color));
|
|
||||||
w = qadd8(w, W(color));
|
|
||||||
col = RGBW32(r,g,b,w);
|
|
||||||
} else {
|
|
||||||
col = color_add(col, color);
|
|
||||||
}
|
|
||||||
setPixelColor(n, col);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Segment::fadePixelColor(uint16_t n, uint8_t fade) {
|
|
||||||
if (!isActive()) return; // not active
|
|
||||||
CRGB pix = CRGB(getPixelColor(n)).nscale8_video(fade);
|
|
||||||
setPixelColor(n, pix);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* fade out function, higher rate = quicker fade
|
* fade out function, higher rate = quicker fade
|
||||||
*/
|
*/
|
||||||
@ -996,7 +959,7 @@ void Segment::fade_out(uint8_t rate) {
|
|||||||
int g2 = G(color);
|
int g2 = G(color);
|
||||||
int b2 = B(color);
|
int b2 = B(color);
|
||||||
|
|
||||||
for (uint16_t y = 0; y < rows; y++) for (uint16_t x = 0; x < cols; x++) {
|
for (int y = 0; y < rows; y++) for (int x = 0; x < cols; x++) {
|
||||||
color = is2D() ? getPixelColorXY(x, y) : getPixelColor(x);
|
color = is2D() ? getPixelColorXY(x, y) : getPixelColor(x);
|
||||||
int w1 = W(color);
|
int w1 = W(color);
|
||||||
int r1 = R(color);
|
int r1 = R(color);
|
||||||
@ -1025,49 +988,40 @@ void Segment::fadeToBlackBy(uint8_t fadeBy) {
|
|||||||
const uint16_t cols = is2D() ? virtualWidth() : virtualLength();
|
const uint16_t cols = is2D() ? virtualWidth() : virtualLength();
|
||||||
const uint16_t rows = virtualHeight(); // will be 1 for 1D
|
const uint16_t rows = virtualHeight(); // will be 1 for 1D
|
||||||
|
|
||||||
for (uint16_t y = 0; y < rows; y++) for (uint16_t x = 0; x < cols; x++) {
|
for (int y = 0; y < rows; y++) for (int x = 0; x < cols; x++) {
|
||||||
if (is2D()) setPixelColorXY(x, y, CRGB(getPixelColorXY(x,y)).nscale8(255-fadeBy));
|
if (is2D()) setPixelColorXY(x, y, color_fade(getPixelColorXY(x,y), 255-fadeBy));
|
||||||
else setPixelColor(x, CRGB(getPixelColor(x)).nscale8(255-fadeBy));
|
else setPixelColor(x, color_fade(getPixelColor(x), 255-fadeBy));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* blurs segment content, source: FastLED colorutils.cpp
|
* blurs segment content, source: FastLED colorutils.cpp
|
||||||
*/
|
*/
|
||||||
void Segment::blur(uint8_t blur_amount)
|
void Segment::blur(uint8_t blur_amount) {
|
||||||
{
|
|
||||||
if (!isActive() || blur_amount == 0) return; // optimization: 0 means "don't blur"
|
if (!isActive() || blur_amount == 0) return; // optimization: 0 means "don't blur"
|
||||||
#ifndef WLED_DISABLE_2D
|
#ifndef WLED_DISABLE_2D
|
||||||
if (is2D()) {
|
if (is2D()) {
|
||||||
// compatibility with 2D
|
// compatibility with 2D
|
||||||
const uint_fast16_t cols = virtualWidth();
|
const unsigned cols = virtualWidth();
|
||||||
const uint_fast16_t rows = virtualHeight();
|
const unsigned rows = virtualHeight();
|
||||||
for (uint_fast16_t i = 0; i < rows; i++) blurRow(i, blur_amount); // blur all rows
|
for (unsigned i = 0; i < rows; i++) blurRow(i, blur_amount); // blur all rows
|
||||||
for (uint_fast16_t k = 0; k < cols; k++) blurCol(k, blur_amount); // blur all columns
|
for (unsigned k = 0; k < cols; k++) blurCol(k, blur_amount); // blur all columns
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
uint8_t keep = 255 - blur_amount;
|
uint8_t keep = 255 - blur_amount;
|
||||||
uint8_t seep = blur_amount >> 1;
|
uint8_t seep = blur_amount >> 1;
|
||||||
CRGB carryover = CRGB::Black;
|
uint32_t carryover = BLACK;
|
||||||
uint_fast16_t vlength = virtualLength();
|
unsigned vlength = virtualLength();
|
||||||
for(uint_fast16_t i = 0; i < vlength; i++)
|
for (unsigned i = 0; i < vlength; i++) {
|
||||||
{
|
uint32_t cur = getPixelColor(i);
|
||||||
CRGB cur = CRGB(getPixelColor(i));
|
uint32_t part = color_fade(cur, seep);
|
||||||
CRGB part = cur;
|
cur = color_add(color_fade(cur, keep), carryover, true);
|
||||||
CRGB before = cur; // remember color before blur
|
if (i > 0) {
|
||||||
part.nscale8(seep);
|
|
||||||
cur.nscale8(keep);
|
|
||||||
cur += carryover;
|
|
||||||
if(i > 0) {
|
|
||||||
uint32_t c = getPixelColor(i-1);
|
uint32_t c = getPixelColor(i-1);
|
||||||
uint8_t r = R(c);
|
setPixelColor(i-1, color_add(c, part, true));
|
||||||
uint8_t g = G(c);
|
|
||||||
uint8_t b = B(c);
|
|
||||||
setPixelColor((uint16_t)(i-1), qadd8(r, part.red), qadd8(g, part.green), qadd8(b, part.blue));
|
|
||||||
}
|
}
|
||||||
if (before != cur) // optimization: only set pixel if color has changed
|
setPixelColor(i, cur);
|
||||||
setPixelColor((uint16_t)i,cur.red, cur.green, cur.blue);
|
|
||||||
carryover = part;
|
carryover = part;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1080,7 +1034,7 @@ void Segment::blur(uint8_t blur_amount)
|
|||||||
uint32_t Segment::color_wheel(uint8_t pos) {
|
uint32_t Segment::color_wheel(uint8_t pos) {
|
||||||
if (palette) return color_from_palette(pos, false, true, 0);
|
if (palette) return color_from_palette(pos, false, true, 0);
|
||||||
pos = 255 - pos;
|
pos = 255 - pos;
|
||||||
if(pos < 85) {
|
if (pos < 85) {
|
||||||
return ((uint32_t)(255 - pos * 3) << 16) | ((uint32_t)(0) << 8) | (pos * 3);
|
return ((uint32_t)(255 - pos * 3) << 16) | ((uint32_t)(0) << 8) | (pos * 3);
|
||||||
} else if(pos < 170) {
|
} else if(pos < 170) {
|
||||||
pos -= 85;
|
pos -= 85;
|
||||||
@ -1091,21 +1045,6 @@ uint32_t Segment::color_wheel(uint8_t pos) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Returns a new, random wheel index with a minimum distance of 42 from pos.
|
|
||||||
*/
|
|
||||||
uint8_t Segment::get_random_wheel_index(uint8_t pos) {
|
|
||||||
uint8_t r = 0, x = 0, y = 0, d = 0;
|
|
||||||
|
|
||||||
while(d < 42) {
|
|
||||||
r = random8();
|
|
||||||
x = abs(pos - r);
|
|
||||||
y = 255 - x;
|
|
||||||
d = MIN(x, y);
|
|
||||||
}
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Gets a single color from the currently selected palette.
|
* Gets a single color from the currently selected palette.
|
||||||
* @param i Palette Index (if mapping is true, the full palette will be _virtualSegmentLength long, if false, 255). Will wrap around automatically.
|
* @param i Palette Index (if mapping is true, the full palette will be _virtualSegmentLength long, if false, 255). Will wrap around automatically.
|
||||||
@ -1115,24 +1054,21 @@ uint8_t Segment::get_random_wheel_index(uint8_t pos) {
|
|||||||
* @param pbri Value to scale the brightness of the returned color by. Default is 255. (no scaling)
|
* @param pbri Value to scale the brightness of the returned color by. Default is 255. (no scaling)
|
||||||
* @returns Single color from palette
|
* @returns Single color from palette
|
||||||
*/
|
*/
|
||||||
uint32_t Segment::color_from_palette(uint16_t i, bool mapping, bool wrap, uint8_t mcol, uint8_t pbri)
|
uint32_t Segment::color_from_palette(uint16_t i, bool mapping, bool wrap, uint8_t mcol, uint8_t pbri) {
|
||||||
{
|
|
||||||
// default palette or no RGB support on segment
|
// default palette or no RGB support on segment
|
||||||
if ((palette == 0 && mcol < NUM_COLORS) || !_isRGB) {
|
if ((palette == 0 && mcol < NUM_COLORS) || !_isRGB) {
|
||||||
uint32_t color = currentColor(mcol, colors[mcol]);
|
uint32_t color = currentColor(mcol);
|
||||||
color = gamma32(color);
|
color = gamma32(color);
|
||||||
if (pbri == 255) return color;
|
if (pbri == 255) return color;
|
||||||
return RGBW32(scale8_video(R(color),pbri), scale8_video(G(color),pbri), scale8_video(B(color),pbri), scale8_video(W(color),pbri));
|
return color_fade(color, pbri, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t paletteIndex = i;
|
uint8_t paletteIndex = i;
|
||||||
if (mapping && virtualLength() > 1) paletteIndex = (i*255)/(virtualLength() -1);
|
if (mapping && virtualLength() > 1) paletteIndex = (i*255)/(virtualLength() -1);
|
||||||
if (!wrap) paletteIndex = scale8(paletteIndex, 240); //cut off blend at palette "end"
|
if (!wrap && strip.paletteBlend != 3) paletteIndex = scale8(paletteIndex, 240); //cut off blend at palette "end"
|
||||||
CRGB fastled_col;
|
|
||||||
CRGBPalette16 curPal;
|
CRGBPalette16 curPal;
|
||||||
if (transitional && _t) curPal = _t->_palT;
|
curPal = currentPalette(curPal, palette);
|
||||||
else loadPalette(curPal, palette);
|
CRGB fastled_col = ColorFromPalette(curPal, paletteIndex, pbri, (strip.paletteBlend == 3)? NOBLEND:LINEARBLEND); // NOTE: paletteBlend should be global
|
||||||
fastled_col = ColorFromPalette(curPal, paletteIndex, pbri, (strip.paletteBlend == 3)? NOBLEND:LINEARBLEND); // NOTE: paletteBlend should be global
|
|
||||||
|
|
||||||
return RGBW32(fastled_col.r, fastled_col.g, fastled_col.b, 0);
|
return RGBW32(fastled_col.r, fastled_col.g, fastled_col.b, 0);
|
||||||
}
|
}
|
||||||
@ -1143,8 +1079,7 @@ uint32_t Segment::color_from_palette(uint16_t i, bool mapping, bool wrap, uint8_
|
|||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
//do not call this method from system context (network callback)
|
//do not call this method from system context (network callback)
|
||||||
void WS2812FX::finalizeInit(void)
|
void WS2812FX::finalizeInit(void) {
|
||||||
{
|
|
||||||
//reset segment runtimes
|
//reset segment runtimes
|
||||||
for (segment &seg : _segments) {
|
for (segment &seg : _segments) {
|
||||||
seg.markForReset();
|
seg.markForReset();
|
||||||
@ -1166,7 +1101,7 @@ void WS2812FX::finalizeInit(void)
|
|||||||
const uint8_t defNumBusses = ((sizeof defDataPins) / (sizeof defDataPins[0]));
|
const uint8_t defNumBusses = ((sizeof defDataPins) / (sizeof defDataPins[0]));
|
||||||
const uint8_t defNumCounts = ((sizeof defCounts) / (sizeof defCounts[0]));
|
const uint8_t defNumCounts = ((sizeof defCounts) / (sizeof defCounts[0]));
|
||||||
uint16_t prevLen = 0;
|
uint16_t prevLen = 0;
|
||||||
for (uint8_t i = 0; i < defNumBusses && i < WLED_MAX_BUSSES+WLED_MIN_VIRTUAL_BUSSES; i++) {
|
for (int i = 0; i < defNumBusses && i < WLED_MAX_BUSSES+WLED_MIN_VIRTUAL_BUSSES; i++) {
|
||||||
uint8_t defPin[] = {defDataPins[i]};
|
uint8_t defPin[] = {defDataPins[i]};
|
||||||
uint16_t start = prevLen;
|
uint16_t start = prevLen;
|
||||||
uint16_t count = defCounts[(i < defNumCounts) ? i : defNumCounts -1];
|
uint16_t count = defCounts[(i < defNumCounts) ? i : defNumCounts -1];
|
||||||
@ -1177,7 +1112,7 @@ void WS2812FX::finalizeInit(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
_length = 0;
|
_length = 0;
|
||||||
for (uint8_t i=0; i<busses.getNumBusses(); i++) {
|
for (int i=0; i<busses.getNumBusses(); i++) {
|
||||||
Bus *bus = busses.getBus(i);
|
Bus *bus = busses.getBus(i);
|
||||||
if (bus == nullptr) continue;
|
if (bus == nullptr) continue;
|
||||||
if (bus->getStart() + bus->getLength() > MAX_LEDS) break;
|
if (bus->getStart() + bus->getLength() > MAX_LEDS) break;
|
||||||
@ -1234,24 +1169,24 @@ void WS2812FX::service() {
|
|||||||
|
|
||||||
if (!seg.freeze) { //only run effect function if not frozen
|
if (!seg.freeze) { //only run effect function if not frozen
|
||||||
_virtualSegmentLength = seg.virtualLength();
|
_virtualSegmentLength = seg.virtualLength();
|
||||||
_colors_t[0] = seg.currentColor(0, seg.colors[0]);
|
_colors_t[0] = seg.currentColor(0);
|
||||||
_colors_t[1] = seg.currentColor(1, seg.colors[1]);
|
_colors_t[1] = seg.currentColor(1);
|
||||||
_colors_t[2] = seg.currentColor(2, seg.colors[2]);
|
_colors_t[2] = seg.currentColor(2);
|
||||||
seg.currentPalette(_currentPalette, seg.palette);
|
seg.currentPalette(_currentPalette, seg.palette); // we need to pass reference
|
||||||
|
|
||||||
if (!cctFromRgb || correctWB) busses.setSegmentCCT(seg.currentBri(seg.cct, true), correctWB);
|
if (!cctFromRgb || correctWB) busses.setSegmentCCT(seg.currentBri(true), correctWB);
|
||||||
for (uint8_t c = 0; c < NUM_COLORS; c++) _colors_t[c] = gamma32(_colors_t[c]);
|
for (int c = 0; c < NUM_COLORS; c++) _colors_t[c] = gamma32(_colors_t[c]);
|
||||||
|
|
||||||
// Effect blending
|
// Effect blending
|
||||||
// When two effects are being blended, each may have different segment data, this
|
// When two effects are being blended, each may have different segment data, this
|
||||||
// data needs to be saved first and then restored before running previous/transitional mode.
|
// data needs to be saved first and then restored before running previous mode.
|
||||||
// The blending will largely depend on the effect behaviour since actual output (LEDs) may be
|
// The blending will largely depend on the effect behaviour since actual output (LEDs) may be
|
||||||
// overwritten by later effect. To enable seamless blending for every effect, additional LED buffer
|
// overwritten by later effect. To enable seamless blending for every effect, additional LED buffer
|
||||||
// would need to be allocated for each effect and then blended together for each pixel.
|
// would need to be allocated for each effect and then blended together for each pixel.
|
||||||
[[maybe_unused]] uint8_t tmpMode = seg.currentMode(seg.mode); // this will return old mode while in transition
|
[[maybe_unused]] uint8_t tmpMode = seg.currentMode(); // this will return old mode while in transition
|
||||||
delay = (*_mode[seg.mode])(); // run new/current mode
|
delay = (*_mode[seg.mode])(); // run new/current mode
|
||||||
#ifndef WLED_DISABLE_MODE_BLEND
|
#ifndef WLED_DISABLE_MODE_BLEND
|
||||||
if (seg.mode != tmpMode) {
|
if (modeBlending && seg.mode != tmpMode) {
|
||||||
Segment::tmpsegd_t _tmpSegData;
|
Segment::tmpsegd_t _tmpSegData;
|
||||||
Segment::modeBlend(true); // set semaphore
|
Segment::modeBlend(true); // set semaphore
|
||||||
seg.swapSegenv(_tmpSegData); // temporarily store new mode state (and swap it with transitional state)
|
seg.swapSegenv(_tmpSegData); // temporarily store new mode state (and swap it with transitional state)
|
||||||
@ -1262,7 +1197,7 @@ void WS2812FX::service() {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
if (seg.mode != FX_MODE_HALLOWEEN_EYES) seg.call++;
|
if (seg.mode != FX_MODE_HALLOWEEN_EYES) seg.call++;
|
||||||
if (seg.transitional && delay > FRAMETIME) delay = FRAMETIME; // force faster updates during transition
|
if (seg.isInTransition() && delay > FRAMETIME) delay = FRAMETIME; // force faster updates during transition
|
||||||
}
|
}
|
||||||
|
|
||||||
seg.next_time = nowUp + delay;
|
seg.next_time = nowUp + delay;
|
||||||
@ -1287,15 +1222,13 @@ void WS2812FX::service() {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void IRAM_ATTR WS2812FX::setPixelColor(int i, uint32_t col)
|
void IRAM_ATTR WS2812FX::setPixelColor(int i, uint32_t col) {
|
||||||
{
|
|
||||||
if (i < customMappingSize) i = customMappingTable[i];
|
if (i < customMappingSize) i = customMappingTable[i];
|
||||||
if (i >= _length) return;
|
if (i >= _length) return;
|
||||||
busses.setPixelColor(i, col);
|
busses.setPixelColor(i, col);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t WS2812FX::getPixelColor(uint16_t i)
|
uint32_t IRAM_ATTR WS2812FX::getPixelColor(uint16_t i) {
|
||||||
{
|
|
||||||
if (i < customMappingSize) i = customMappingTable[i];
|
if (i < customMappingSize) i = customMappingTable[i];
|
||||||
if (i >= _length) return 0;
|
if (i >= _length) return 0;
|
||||||
return busses.getPixelColor(i);
|
return busses.getPixelColor(i);
|
||||||
@ -1335,13 +1268,13 @@ uint8_t WS2812FX::estimateCurrentAndLimitBri() {
|
|||||||
|
|
||||||
size_t pLen = 0; //getLengthPhysical();
|
size_t pLen = 0; //getLengthPhysical();
|
||||||
size_t powerSum = 0;
|
size_t powerSum = 0;
|
||||||
for (uint_fast8_t bNum = 0; bNum < busses.getNumBusses(); bNum++) {
|
for (unsigned bNum = 0; bNum < busses.getNumBusses(); bNum++) {
|
||||||
Bus *bus = busses.getBus(bNum);
|
Bus *bus = busses.getBus(bNum);
|
||||||
if (!IS_DIGITAL(bus->getType())) continue; //exclude non-digital network busses
|
if (!IS_DIGITAL(bus->getType())) continue; //exclude non-digital network busses
|
||||||
uint16_t len = bus->getLength();
|
uint16_t len = bus->getLength();
|
||||||
pLen += len;
|
pLen += len;
|
||||||
uint32_t busPowerSum = 0;
|
uint32_t busPowerSum = 0;
|
||||||
for (uint_fast16_t i = 0; i < len; i++) { //sum up the usage of each LED
|
for (unsigned i = 0; i < len; i++) { //sum up the usage of each LED
|
||||||
uint32_t c = bus->getPixelColor(i); // always returns original or restored color without brightness scaling
|
uint32_t c = bus->getPixelColor(i); // always returns original or restored color without brightness scaling
|
||||||
byte r = R(c), g = G(c), b = B(c), w = W(c);
|
byte r = R(c), g = G(c), b = B(c), w = W(c);
|
||||||
|
|
||||||
@ -1399,12 +1332,12 @@ void WS2812FX::show(void) {
|
|||||||
// or async show has a separate buffer (ESP32 RMT and I2S are ok)
|
// or async show has a separate buffer (ESP32 RMT and I2S are ok)
|
||||||
if (newBri < _brightness) busses.setBrightness(_brightness);
|
if (newBri < _brightness) busses.setBrightness(_brightness);
|
||||||
|
|
||||||
unsigned long now = millis();
|
unsigned long showNow = millis();
|
||||||
size_t diff = now - _lastShow;
|
size_t diff = showNow - _lastShow;
|
||||||
size_t fpsCurr = 200;
|
size_t fpsCurr = 200;
|
||||||
if (diff > 0) fpsCurr = 1000 / diff;
|
if (diff > 0) fpsCurr = 1000 / diff;
|
||||||
_cumulativeFps = (3 * _cumulativeFps + fpsCurr +2) >> 2; // "+2" for proper rounding (2/4 = 0.5)
|
_cumulativeFps = (3 * _cumulativeFps + fpsCurr +2) >> 2; // "+2" for proper rounding (2/4 = 0.5)
|
||||||
_lastShow = now;
|
_lastShow = showNow;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1486,8 +1419,7 @@ uint8_t WS2812FX::getActiveSegsLightCapabilities(bool selectedOnly) {
|
|||||||
return totalLC;
|
return totalLC;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t WS2812FX::getFirstSelectedSegId(void)
|
uint8_t WS2812FX::getFirstSelectedSegId(void) {
|
||||||
{
|
|
||||||
size_t i = 0;
|
size_t i = 0;
|
||||||
for (segment &seg : _segments) {
|
for (segment &seg : _segments) {
|
||||||
if (seg.isActive() && seg.isSelected()) return i;
|
if (seg.isActive() && seg.isSelected()) return i;
|
||||||
@ -1597,10 +1529,12 @@ void WS2812FX::setSegment(uint8_t segId, uint16_t i1, uint16_t i2, uint8_t group
|
|||||||
_qStart = i1; _qStop = i2; _qStartY = startY; _qStopY = stopY;
|
_qStart = i1; _qStop = i2; _qStartY = startY; _qStopY = stopY;
|
||||||
_qGrouping = grouping; _qSpacing = spacing; _qOffset = offset;
|
_qGrouping = grouping; _qSpacing = spacing; _qOffset = offset;
|
||||||
_queuedChangesSegId = segId;
|
_queuedChangesSegId = segId;
|
||||||
|
DEBUG_PRINT(F("Segment queued: ")); DEBUG_PRINTLN(segId);
|
||||||
return; // queued changes are applied immediately after effect function returns
|
return; // queued changes are applied immediately after effect function returns
|
||||||
}
|
}
|
||||||
|
|
||||||
_segments[segId].setUp(i1, i2, grouping, spacing, offset, startY, stopY);
|
_segments[segId].setUp(i1, i2, grouping, spacing, offset, startY, stopY);
|
||||||
|
if (segId > 0 && segId == getSegmentsNum()-1 && i2 <= i1) _segments.pop_back(); // if last segment was deleted remove it from vector
|
||||||
}
|
}
|
||||||
|
|
||||||
void WS2812FX::setUpSegmentFromQueuedChanges() {
|
void WS2812FX::setUpSegmentFromQueuedChanges() {
|
||||||
@ -1609,10 +1543,6 @@ void WS2812FX::setUpSegmentFromQueuedChanges() {
|
|||||||
_queuedChangesSegId = 255;
|
_queuedChangesSegId = 255;
|
||||||
}
|
}
|
||||||
|
|
||||||
void WS2812FX::restartRuntime() {
|
|
||||||
for (segment &seg : _segments) seg.markForReset();
|
|
||||||
}
|
|
||||||
|
|
||||||
void WS2812FX::resetSegments() {
|
void WS2812FX::resetSegments() {
|
||||||
_segments.clear(); // destructs all Segment as part of clearing
|
_segments.clear(); // destructs all Segment as part of clearing
|
||||||
#ifndef WLED_DISABLE_2D
|
#ifndef WLED_DISABLE_2D
|
||||||
@ -1729,7 +1659,7 @@ void WS2812FX::fixInvalidSegments() {
|
|||||||
bool WS2812FX::checkSegmentAlignment() {
|
bool WS2812FX::checkSegmentAlignment() {
|
||||||
bool aligned = false;
|
bool aligned = false;
|
||||||
for (segment &seg : _segments) {
|
for (segment &seg : _segments) {
|
||||||
for (uint8_t b = 0; b<busses.getNumBusses(); b++) {
|
for (unsigned b = 0; b<busses.getNumBusses(); b++) {
|
||||||
Bus *bus = busses.getBus(b);
|
Bus *bus = busses.getBus(b);
|
||||||
if (seg.start == bus->getStart() && seg.stop == bus->getStart() + bus->getLength()) aligned = true;
|
if (seg.start == bus->getStart() && seg.stop == bus->getStart() + bus->getLength()) aligned = true;
|
||||||
}
|
}
|
||||||
@ -1752,17 +1682,8 @@ uint8_t WS2812FX::setPixelSegment(uint8_t n) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void WS2812FX::setRange(uint16_t i, uint16_t i2, uint32_t col) {
|
void WS2812FX::setRange(uint16_t i, uint16_t i2, uint32_t col) {
|
||||||
if (i2 >= i)
|
if (i2 < i) std::swap(i,i2);
|
||||||
{
|
for (unsigned x = i; x <= i2; x++) setPixelColor(x, col);
|
||||||
for (uint16_t x = i; x <= i2; x++) setPixelColor(x, col);
|
|
||||||
} else
|
|
||||||
{
|
|
||||||
for (uint16_t x = i2; x <= i; x++) setPixelColor(x, col);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void WS2812FX::setTransitionMode(bool t) {
|
|
||||||
for (segment &seg : _segments) seg.startTransition(t ? _transitionDur : 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef WLED_DEBUG
|
#ifdef WLED_DEBUG
|
||||||
@ -1793,7 +1714,7 @@ void WS2812FX::loadCustomPalettes() {
|
|||||||
|
|
||||||
if (readObjectFromFile(fileName, nullptr, &pDoc)) {
|
if (readObjectFromFile(fileName, nullptr, &pDoc)) {
|
||||||
JsonArray pal = pDoc[F("palette")];
|
JsonArray pal = pDoc[F("palette")];
|
||||||
if (!pal.isNull() && pal.size()>4) { // not an empty palette (at least 2 entries)
|
if (!pal.isNull() && pal.size()>3) { // not an empty palette (at least 2 entries)
|
||||||
if (pal[0].is<int>() && pal[1].is<const char *>()) {
|
if (pal[0].is<int>() && pal[1].is<const char *>()) {
|
||||||
// we have an array of index & hex strings
|
// we have an array of index & hex strings
|
||||||
size_t palSize = MIN(pal.size(), 36);
|
size_t palSize = MIN(pal.size(), 36);
|
||||||
@ -1802,7 +1723,7 @@ void WS2812FX::loadCustomPalettes() {
|
|||||||
uint8_t rgbw[] = {0,0,0,0};
|
uint8_t rgbw[] = {0,0,0,0};
|
||||||
tcp[ j ] = (uint8_t) pal[ i ].as<int>(); // index
|
tcp[ j ] = (uint8_t) pal[ i ].as<int>(); // index
|
||||||
colorFromHexString(rgbw, pal[i+1].as<const char *>()); // will catch non-string entires
|
colorFromHexString(rgbw, pal[i+1].as<const char *>()); // will catch non-string entires
|
||||||
for (size_t c=0; c<3; c++) tcp[j+1+c] = rgbw[c]; // only use RGB component
|
for (size_t c=0; c<3; c++) tcp[j+1+c] = gamma8(rgbw[c]); // only use RGB component
|
||||||
DEBUG_PRINTF("%d(%d) : %d %d %d\n", i, int(tcp[j]), int(tcp[j+1]), int(tcp[j+2]), int(tcp[j+3]));
|
DEBUG_PRINTF("%d(%d) : %d %d %d\n", i, int(tcp[j]), int(tcp[j+1]), int(tcp[j+2]), int(tcp[j+3]));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -1810,13 +1731,15 @@ void WS2812FX::loadCustomPalettes() {
|
|||||||
palSize -= palSize % 4; // make sure size is multiple of 4
|
palSize -= palSize % 4; // make sure size is multiple of 4
|
||||||
for (size_t i=0; i<palSize && pal[i].as<int>()<256; i+=4) {
|
for (size_t i=0; i<palSize && pal[i].as<int>()<256; i+=4) {
|
||||||
tcp[ i ] = (uint8_t) pal[ i ].as<int>(); // index
|
tcp[ i ] = (uint8_t) pal[ i ].as<int>(); // index
|
||||||
tcp[i+1] = (uint8_t) pal[i+1].as<int>(); // R
|
tcp[i+1] = gamma8((uint8_t) pal[i+1].as<int>()); // R
|
||||||
tcp[i+2] = (uint8_t) pal[i+2].as<int>(); // G
|
tcp[i+2] = gamma8((uint8_t) pal[i+2].as<int>()); // G
|
||||||
tcp[i+3] = (uint8_t) pal[i+3].as<int>(); // B
|
tcp[i+3] = gamma8((uint8_t) pal[i+3].as<int>()); // B
|
||||||
DEBUG_PRINTF("%d(%d) : %d %d %d\n", i, int(tcp[i]), int(tcp[i+1]), int(tcp[i+2]), int(tcp[i+3]));
|
DEBUG_PRINTF("%d(%d) : %d %d %d\n", i, int(tcp[i]), int(tcp[i+1]), int(tcp[i+2]), int(tcp[i+3]));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
customPalettes.push_back(targetPalette.loadDynamicGradientPalette(tcp));
|
customPalettes.push_back(targetPalette.loadDynamicGradientPalette(tcp));
|
||||||
|
} else {
|
||||||
|
DEBUG_PRINTLN(F("Wrong palette format."));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -1832,7 +1755,7 @@ bool WS2812FX::deserializeMap(uint8_t n) {
|
|||||||
char fileName[32];
|
char fileName[32];
|
||||||
strcpy_P(fileName, PSTR("/ledmap"));
|
strcpy_P(fileName, PSTR("/ledmap"));
|
||||||
if (n) sprintf(fileName +7, "%d", n);
|
if (n) sprintf(fileName +7, "%d", n);
|
||||||
strcat(fileName, ".json");
|
strcat_P(fileName, PSTR(".json"));
|
||||||
bool isFile = WLED_FS.exists(fileName);
|
bool isFile = WLED_FS.exists(fileName);
|
||||||
|
|
||||||
if (!isFile) {
|
if (!isFile) {
|
||||||
@ -1866,7 +1789,7 @@ bool WS2812FX::deserializeMap(uint8_t n) {
|
|||||||
if (!map.isNull() && map.size()) { // not an empty map
|
if (!map.isNull() && map.size()) { // not an empty map
|
||||||
customMappingSize = map.size();
|
customMappingSize = map.size();
|
||||||
customMappingTable = new uint16_t[customMappingSize];
|
customMappingTable = new uint16_t[customMappingSize];
|
||||||
for (uint16_t i=0; i<customMappingSize; i++) {
|
for (unsigned i=0; i<customMappingSize; i++) {
|
||||||
customMappingTable[i] = (uint16_t) (map[i]<0 ? 0xFFFFU : map[i]);
|
customMappingTable[i] = (uint16_t) (map[i]<0 ? 0xFFFFU : map[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -66,7 +66,7 @@ uint8_t IRAM_ATTR ColorOrderMap::getPixelColorOrder(uint16_t pix, uint8_t defaul
|
|||||||
if (_count == 0) return defaultColorOrder;
|
if (_count == 0) return defaultColorOrder;
|
||||||
// upper nibble containd W swap information
|
// upper nibble containd W swap information
|
||||||
uint8_t swapW = defaultColorOrder >> 4;
|
uint8_t swapW = defaultColorOrder >> 4;
|
||||||
for (uint8_t i = 0; i < _count; i++) {
|
for (unsigned i = 0; i < _count; i++) {
|
||||||
if (pix >= _mappings[i].start && pix < (_mappings[i].start + _mappings[i].len)) {
|
if (pix >= _mappings[i].start && pix < (_mappings[i].start + _mappings[i].len)) {
|
||||||
return _mappings[i].colorOrder | (swapW << 4);
|
return _mappings[i].colorOrder | (swapW << 4);
|
||||||
}
|
}
|
||||||
@ -180,7 +180,7 @@ void BusDigital::setBrightness(uint8_t b) {
|
|||||||
// (which we can't rely on)
|
// (which we can't rely on)
|
||||||
uint16_t hwLen = _len;
|
uint16_t hwLen = _len;
|
||||||
if (_type == TYPE_WS2812_1CH_X3) hwLen = NUM_ICS_WS2812_1CH_3X(_len); // only needs a third of "RGB" LEDs for NeoPixelBus
|
if (_type == TYPE_WS2812_1CH_X3) hwLen = NUM_ICS_WS2812_1CH_3X(_len); // only needs a third of "RGB" LEDs for NeoPixelBus
|
||||||
for (uint_fast16_t i = 0; i < hwLen; i++) {
|
for (unsigned i = 0; i < hwLen; i++) {
|
||||||
// use 0 as color order, actual order does not matter here as we just update the channel values as-is
|
// use 0 as color order, actual order does not matter here as we just update the channel values as-is
|
||||||
uint32_t c = restoreColorLossy(PolyBus::getPixelColor(_busPtr, _iType, i, 0),prevBri);
|
uint32_t c = restoreColorLossy(PolyBus::getPixelColor(_busPtr, _iType, i, 0),prevBri);
|
||||||
PolyBus::setPixelColor(_busPtr, _iType, i, c, 0);
|
PolyBus::setPixelColor(_busPtr, _iType, i, c, 0);
|
||||||
@ -261,7 +261,7 @@ uint32_t BusDigital::getPixelColor(uint16_t pix) {
|
|||||||
|
|
||||||
uint8_t BusDigital::getPins(uint8_t* pinArray) {
|
uint8_t BusDigital::getPins(uint8_t* pinArray) {
|
||||||
uint8_t numPins = IS_2PIN(_type) ? 2 : 1;
|
uint8_t numPins = IS_2PIN(_type) ? 2 : 1;
|
||||||
for (uint8_t i = 0; i < numPins; i++) pinArray[i] = _pins[i];
|
for (unsigned i = 0; i < numPins; i++) pinArray[i] = _pins[i];
|
||||||
return numPins;
|
return numPins;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -305,7 +305,7 @@ BusPwm::BusPwm(BusConfig &bc)
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
for (uint8_t i = 0; i < numPins; i++) {
|
for (unsigned i = 0; i < numPins; i++) {
|
||||||
uint8_t currentPin = bc.pins[i];
|
uint8_t currentPin = bc.pins[i];
|
||||||
if (!pinManager.allocatePin(currentPin, true, PinOwner::BusPwm)) {
|
if (!pinManager.allocatePin(currentPin, true, PinOwner::BusPwm)) {
|
||||||
deallocatePins(); return;
|
deallocatePins(); return;
|
||||||
@ -384,7 +384,7 @@ uint32_t BusPwm::getPixelColor(uint16_t pix) {
|
|||||||
void BusPwm::show() {
|
void BusPwm::show() {
|
||||||
if (!_valid) return;
|
if (!_valid) return;
|
||||||
uint8_t numPins = NUM_PWM_PINS(_type);
|
uint8_t numPins = NUM_PWM_PINS(_type);
|
||||||
for (uint8_t i = 0; i < numPins; i++) {
|
for (unsigned i = 0; i < numPins; i++) {
|
||||||
uint8_t scaled = (_data[i] * _bri) / 255;
|
uint8_t scaled = (_data[i] * _bri) / 255;
|
||||||
if (_reversed) scaled = 255 - scaled;
|
if (_reversed) scaled = 255 - scaled;
|
||||||
#ifdef ESP8266
|
#ifdef ESP8266
|
||||||
@ -398,7 +398,7 @@ void BusPwm::show() {
|
|||||||
uint8_t BusPwm::getPins(uint8_t* pinArray) {
|
uint8_t BusPwm::getPins(uint8_t* pinArray) {
|
||||||
if (!_valid) return 0;
|
if (!_valid) return 0;
|
||||||
uint8_t numPins = NUM_PWM_PINS(_type);
|
uint8_t numPins = NUM_PWM_PINS(_type);
|
||||||
for (uint8_t i = 0; i < numPins; i++) {
|
for (unsigned i = 0; i < numPins; i++) {
|
||||||
pinArray[i] = _pins[i];
|
pinArray[i] = _pins[i];
|
||||||
}
|
}
|
||||||
return numPins;
|
return numPins;
|
||||||
@ -406,7 +406,7 @@ uint8_t BusPwm::getPins(uint8_t* pinArray) {
|
|||||||
|
|
||||||
void BusPwm::deallocatePins() {
|
void BusPwm::deallocatePins() {
|
||||||
uint8_t numPins = NUM_PWM_PINS(_type);
|
uint8_t numPins = NUM_PWM_PINS(_type);
|
||||||
for (uint8_t i = 0; i < numPins; i++) {
|
for (unsigned i = 0; i < numPins; i++) {
|
||||||
pinManager.deallocatePin(_pins[i], PinOwner::BusPwm);
|
pinManager.deallocatePin(_pins[i], PinOwner::BusPwm);
|
||||||
if (!pinManager.isPinOk(_pins[i])) continue;
|
if (!pinManager.isPinOk(_pins[i])) continue;
|
||||||
#ifdef ESP8266
|
#ifdef ESP8266
|
||||||
@ -512,7 +512,7 @@ void BusNetwork::show() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
uint8_t BusNetwork::getPins(uint8_t* pinArray) {
|
uint8_t BusNetwork::getPins(uint8_t* pinArray) {
|
||||||
for (uint8_t i = 0; i < 4; i++) {
|
for (unsigned i = 0; i < 4; i++) {
|
||||||
pinArray[i] = _client[i];
|
pinArray[i] = _client[i];
|
||||||
}
|
}
|
||||||
return 4;
|
return 4;
|
||||||
@ -566,24 +566,24 @@ void BusManager::removeAll() {
|
|||||||
DEBUG_PRINTLN(F("Removing all."));
|
DEBUG_PRINTLN(F("Removing all."));
|
||||||
//prevents crashes due to deleting busses while in use.
|
//prevents crashes due to deleting busses while in use.
|
||||||
while (!canAllShow()) yield();
|
while (!canAllShow()) yield();
|
||||||
for (uint8_t i = 0; i < numBusses; i++) delete busses[i];
|
for (unsigned i = 0; i < numBusses; i++) delete busses[i];
|
||||||
numBusses = 0;
|
numBusses = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void BusManager::show() {
|
void BusManager::show() {
|
||||||
for (uint8_t i = 0; i < numBusses; i++) {
|
for (unsigned i = 0; i < numBusses; i++) {
|
||||||
busses[i]->show();
|
busses[i]->show();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void BusManager::setStatusPixel(uint32_t c) {
|
void BusManager::setStatusPixel(uint32_t c) {
|
||||||
for (uint8_t i = 0; i < numBusses; i++) {
|
for (unsigned i = 0; i < numBusses; i++) {
|
||||||
busses[i]->setStatusPixel(c);
|
busses[i]->setStatusPixel(c);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void IRAM_ATTR BusManager::setPixelColor(uint16_t pix, uint32_t c) {
|
void IRAM_ATTR BusManager::setPixelColor(uint16_t pix, uint32_t c) {
|
||||||
for (uint8_t i = 0; i < numBusses; i++) {
|
for (unsigned i = 0; i < numBusses; i++) {
|
||||||
Bus* b = busses[i];
|
Bus* b = busses[i];
|
||||||
uint16_t bstart = b->getStart();
|
uint16_t bstart = b->getStart();
|
||||||
if (pix < bstart || pix >= bstart + b->getLength()) continue;
|
if (pix < bstart || pix >= bstart + b->getLength()) continue;
|
||||||
@ -592,7 +592,7 @@ void IRAM_ATTR BusManager::setPixelColor(uint16_t pix, uint32_t c) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void BusManager::setBrightness(uint8_t b) {
|
void BusManager::setBrightness(uint8_t b) {
|
||||||
for (uint8_t i = 0; i < numBusses; i++) {
|
for (unsigned i = 0; i < numBusses; i++) {
|
||||||
busses[i]->setBrightness(b);
|
busses[i]->setBrightness(b);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -607,7 +607,7 @@ void BusManager::setSegmentCCT(int16_t cct, bool allowWBCorrection) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
uint32_t BusManager::getPixelColor(uint16_t pix) {
|
uint32_t BusManager::getPixelColor(uint16_t pix) {
|
||||||
for (uint8_t i = 0; i < numBusses; i++) {
|
for (unsigned i = 0; i < numBusses; i++) {
|
||||||
Bus* b = busses[i];
|
Bus* b = busses[i];
|
||||||
uint16_t bstart = b->getStart();
|
uint16_t bstart = b->getStart();
|
||||||
if (pix < bstart || pix >= bstart + b->getLength()) continue;
|
if (pix < bstart || pix >= bstart + b->getLength()) continue;
|
||||||
@ -617,7 +617,7 @@ uint32_t BusManager::getPixelColor(uint16_t pix) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool BusManager::canAllShow() {
|
bool BusManager::canAllShow() {
|
||||||
for (uint8_t i = 0; i < numBusses; i++) {
|
for (unsigned i = 0; i < numBusses; i++) {
|
||||||
if (!busses[i]->canShow()) return false;
|
if (!busses[i]->canShow()) return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
@ -631,7 +631,7 @@ Bus* BusManager::getBus(uint8_t busNr) {
|
|||||||
//semi-duplicate of strip.getLengthTotal() (though that just returns strip._length, calculated in finalizeInit())
|
//semi-duplicate of strip.getLengthTotal() (though that just returns strip._length, calculated in finalizeInit())
|
||||||
uint16_t BusManager::getTotalLength() {
|
uint16_t BusManager::getTotalLength() {
|
||||||
uint16_t len = 0;
|
uint16_t len = 0;
|
||||||
for (uint8_t i=0; i<numBusses; i++) len += busses[i]->getLength();
|
for (unsigned i=0; i<numBusses; i++) len += busses[i]->getLength();
|
||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -35,7 +35,14 @@ bool deserializeConfig(JsonObject doc, bool fromFS) {
|
|||||||
CJSON(simplifiedUI, id[F("sui")]);
|
CJSON(simplifiedUI, id[F("sui")]);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
JsonObject nw_ins_0 = doc["nw"]["ins"][0];
|
JsonObject nw = doc["nw"];
|
||||||
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
|
CJSON(enableESPNow, nw[F("espnow")]);
|
||||||
|
getStringFromJson(linked_remote, nw[F("linked_remote")], 13);
|
||||||
|
linked_remote[12] = '\0';
|
||||||
|
#endif
|
||||||
|
|
||||||
|
JsonObject nw_ins_0 = nw["ins"][0];
|
||||||
getStringFromJson(clientSSID, nw_ins_0[F("ssid")], 33);
|
getStringFromJson(clientSSID, nw_ins_0[F("ssid")], 33);
|
||||||
//int nw_ins_0_pskl = nw_ins_0[F("pskl")];
|
//int nw_ins_0_pskl = nw_ins_0[F("pskl")];
|
||||||
//The WiFi PSK is normally not contained in the regular file for security reasons.
|
//The WiFi PSK is normally not contained in the regular file for security reasons.
|
||||||
@ -216,7 +223,9 @@ bool deserializeConfig(JsonObject doc, bool fromFS) {
|
|||||||
if (((buttonType[s] == BTN_TYPE_ANALOG) || (buttonType[s] == BTN_TYPE_ANALOG_INVERTED)) && (digitalPinToAnalogChannel(btnPin[s]) < 0))
|
if (((buttonType[s] == BTN_TYPE_ANALOG) || (buttonType[s] == BTN_TYPE_ANALOG_INVERTED)) && (digitalPinToAnalogChannel(btnPin[s]) < 0))
|
||||||
{
|
{
|
||||||
// not an ADC analog pin
|
// not an ADC analog pin
|
||||||
DEBUG_PRINTF("PIN ALLOC error: GPIO%d for analog button #%d is not an analog pin!\n", btnPin[s], s);
|
DEBUG_PRINT(F("PIN ALLOC error: GPIO")); DEBUG_PRINT(btnPin[s]);
|
||||||
|
DEBUG_PRINT(F("for analog button #")); DEBUG_PRINT(s);
|
||||||
|
DEBUG_PRINTLN(F(" is not an analog pin!"));
|
||||||
btnPin[s] = -1;
|
btnPin[s] = -1;
|
||||||
pinManager.deallocatePin(pin,PinOwner::Button);
|
pinManager.deallocatePin(pin,PinOwner::Button);
|
||||||
}
|
}
|
||||||
@ -357,6 +366,7 @@ bool deserializeConfig(JsonObject doc, bool fromFS) {
|
|||||||
|
|
||||||
JsonObject light_tr = light["tr"];
|
JsonObject light_tr = light["tr"];
|
||||||
CJSON(fadeTransition, light_tr["mode"]);
|
CJSON(fadeTransition, light_tr["mode"]);
|
||||||
|
CJSON(modeBlending, light_tr["fx"]);
|
||||||
int tdd = light_tr["dur"] | -1;
|
int tdd = light_tr["dur"] | -1;
|
||||||
if (tdd >= 0) transitionDelay = transitionDelayDefault = tdd * 100;
|
if (tdd >= 0) transitionDelay = transitionDelayDefault = tdd * 100;
|
||||||
CJSON(strip.paletteFade, light_tr["pal"]);
|
CJSON(strip.paletteFade, light_tr["pal"]);
|
||||||
@ -382,6 +392,10 @@ bool deserializeConfig(JsonObject doc, bool fromFS) {
|
|||||||
CJSON(udpPort, if_sync[F("port0")]); // 21324
|
CJSON(udpPort, if_sync[F("port0")]); // 21324
|
||||||
CJSON(udpPort2, if_sync[F("port1")]); // 65506
|
CJSON(udpPort2, if_sync[F("port1")]); // 65506
|
||||||
|
|
||||||
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
|
CJSON(useESPNowSync, if_sync[F("espnow")]);
|
||||||
|
#endif
|
||||||
|
|
||||||
JsonObject if_sync_recv = if_sync["recv"];
|
JsonObject if_sync_recv = if_sync["recv"];
|
||||||
CJSON(receiveNotificationBrightness, if_sync_recv["bri"]);
|
CJSON(receiveNotificationBrightness, if_sync_recv["bri"]);
|
||||||
CJSON(receiveNotificationColor, if_sync_recv["col"]);
|
CJSON(receiveNotificationColor, if_sync_recv["col"]);
|
||||||
@ -389,17 +403,15 @@ bool deserializeConfig(JsonObject doc, bool fromFS) {
|
|||||||
CJSON(receiveGroups, if_sync_recv["grp"]);
|
CJSON(receiveGroups, if_sync_recv["grp"]);
|
||||||
CJSON(receiveSegmentOptions, if_sync_recv["seg"]);
|
CJSON(receiveSegmentOptions, if_sync_recv["seg"]);
|
||||||
CJSON(receiveSegmentBounds, if_sync_recv["sb"]);
|
CJSON(receiveSegmentBounds, if_sync_recv["sb"]);
|
||||||
//! following line might be a problem if called after boot
|
|
||||||
receiveNotifications = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects || receiveSegmentOptions);
|
|
||||||
|
|
||||||
JsonObject if_sync_send = if_sync["send"];
|
JsonObject if_sync_send = if_sync["send"];
|
||||||
prev = notifyDirectDefault;
|
CJSON(sendNotifications, if_sync_send["en"]);
|
||||||
CJSON(notifyDirectDefault, if_sync_send[F("dir")]);
|
sendNotificationsRT = sendNotifications;
|
||||||
if (notifyDirectDefault != prev) notifyDirect = notifyDirectDefault;
|
CJSON(notifyDirect, if_sync_send[F("dir")]);
|
||||||
CJSON(notifyButton, if_sync_send["btn"]);
|
CJSON(notifyButton, if_sync_send["btn"]);
|
||||||
CJSON(notifyAlexa, if_sync_send["va"]);
|
CJSON(notifyAlexa, if_sync_send["va"]);
|
||||||
CJSON(notifyHue, if_sync_send["hue"]);
|
CJSON(notifyHue, if_sync_send["hue"]);
|
||||||
CJSON(notifyMacro, if_sync_send["macro"]);
|
// CJSON(notifyMacro, if_sync_send["macro"]);
|
||||||
CJSON(syncGroups, if_sync_send["grp"]);
|
CJSON(syncGroups, if_sync_send["grp"]);
|
||||||
if (if_sync_send[F("twice")]) udpNumRetries = 1; // import setting from 0.13 and earlier
|
if (if_sync_send[F("twice")]) udpNumRetries = 1; // import setting from 0.13 and earlier
|
||||||
CJSON(udpNumRetries, if_sync_send["ret"]);
|
CJSON(udpNumRetries, if_sync_send["ret"]);
|
||||||
@ -409,7 +421,7 @@ bool deserializeConfig(JsonObject doc, bool fromFS) {
|
|||||||
CJSON(nodeBroadcastEnabled, if_nodes[F("bcast")]);
|
CJSON(nodeBroadcastEnabled, if_nodes[F("bcast")]);
|
||||||
|
|
||||||
JsonObject if_live = interfaces["live"];
|
JsonObject if_live = interfaces["live"];
|
||||||
CJSON(receiveDirect, if_live["en"]);
|
CJSON(receiveDirect, if_live["en"]); // UDP/Hyperion realtime
|
||||||
CJSON(useMainSegmentOnly, if_live[F("mso")]);
|
CJSON(useMainSegmentOnly, if_live[F("mso")]);
|
||||||
CJSON(e131Port, if_live["port"]); // 5568
|
CJSON(e131Port, if_live["port"]); // 5568
|
||||||
if (e131Port == DDP_DEFAULT_PORT) e131Port = E131_DEFAULT_PORT; // prevent double DDP port allocation
|
if (e131Port == DDP_DEFAULT_PORT) e131Port = E131_DEFAULT_PORT; // prevent double DDP port allocation
|
||||||
@ -453,13 +465,6 @@ bool deserializeConfig(JsonObject doc, bool fromFS) {
|
|||||||
CJSON(retainMqttMsg, if_mqtt[F("rtn")]);
|
CJSON(retainMqttMsg, if_mqtt[F("rtn")]);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef WLED_DISABLE_ESPNOW
|
|
||||||
JsonObject remote = doc["remote"];
|
|
||||||
CJSON(enable_espnow_remote, remote[F("remote_enabled")]);
|
|
||||||
getStringFromJson(linked_remote, remote[F("linked_remote")], 13);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
#ifndef WLED_DISABLE_HUESYNC
|
#ifndef WLED_DISABLE_HUESYNC
|
||||||
JsonObject if_hue = interfaces["hue"];
|
JsonObject if_hue = interfaces["hue"];
|
||||||
CJSON(huePollingEnabled, if_hue["en"]);
|
CJSON(huePollingEnabled, if_hue["en"]);
|
||||||
@ -647,6 +652,10 @@ void serializeConfig() {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
JsonObject nw = doc.createNestedObject("nw");
|
JsonObject nw = doc.createNestedObject("nw");
|
||||||
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
|
nw[F("espnow")] = enableESPNow;
|
||||||
|
nw[F("linked_remote")] = linked_remote;
|
||||||
|
#endif
|
||||||
|
|
||||||
JsonArray nw_ins = nw.createNestedArray("ins");
|
JsonArray nw_ins = nw.createNestedArray("ins");
|
||||||
|
|
||||||
@ -827,6 +836,7 @@ void serializeConfig() {
|
|||||||
|
|
||||||
JsonObject light_tr = light.createNestedObject("tr");
|
JsonObject light_tr = light.createNestedObject("tr");
|
||||||
light_tr["mode"] = fadeTransition;
|
light_tr["mode"] = fadeTransition;
|
||||||
|
light_tr["fx"] = modeBlending;
|
||||||
light_tr["dur"] = transitionDelayDefault / 100;
|
light_tr["dur"] = transitionDelayDefault / 100;
|
||||||
light_tr["pal"] = strip.paletteFade;
|
light_tr["pal"] = strip.paletteFade;
|
||||||
light_tr[F("rpc")] = randomPaletteChangeTime;
|
light_tr[F("rpc")] = randomPaletteChangeTime;
|
||||||
@ -848,6 +858,10 @@ void serializeConfig() {
|
|||||||
if_sync[F("port0")] = udpPort;
|
if_sync[F("port0")] = udpPort;
|
||||||
if_sync[F("port1")] = udpPort2;
|
if_sync[F("port1")] = udpPort2;
|
||||||
|
|
||||||
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
|
if_sync[F("espnow")] = useESPNowSync;
|
||||||
|
#endif
|
||||||
|
|
||||||
JsonObject if_sync_recv = if_sync.createNestedObject("recv");
|
JsonObject if_sync_recv = if_sync.createNestedObject("recv");
|
||||||
if_sync_recv["bri"] = receiveNotificationBrightness;
|
if_sync_recv["bri"] = receiveNotificationBrightness;
|
||||||
if_sync_recv["col"] = receiveNotificationColor;
|
if_sync_recv["col"] = receiveNotificationColor;
|
||||||
@ -857,11 +871,12 @@ void serializeConfig() {
|
|||||||
if_sync_recv["sb"] = receiveSegmentBounds;
|
if_sync_recv["sb"] = receiveSegmentBounds;
|
||||||
|
|
||||||
JsonObject if_sync_send = if_sync.createNestedObject("send");
|
JsonObject if_sync_send = if_sync.createNestedObject("send");
|
||||||
|
if_sync_send["en"] = sendNotifications;
|
||||||
if_sync_send[F("dir")] = notifyDirect;
|
if_sync_send[F("dir")] = notifyDirect;
|
||||||
if_sync_send["btn"] = notifyButton;
|
if_sync_send["btn"] = notifyButton;
|
||||||
if_sync_send["va"] = notifyAlexa;
|
if_sync_send["va"] = notifyAlexa;
|
||||||
if_sync_send["hue"] = notifyHue;
|
if_sync_send["hue"] = notifyHue;
|
||||||
if_sync_send["macro"] = notifyMacro;
|
// if_sync_send["macro"] = notifyMacro;
|
||||||
if_sync_send["grp"] = syncGroups;
|
if_sync_send["grp"] = syncGroups;
|
||||||
if_sync_send["ret"] = udpNumRetries;
|
if_sync_send["ret"] = udpNumRetries;
|
||||||
|
|
||||||
@ -870,7 +885,7 @@ void serializeConfig() {
|
|||||||
if_nodes[F("bcast")] = nodeBroadcastEnabled;
|
if_nodes[F("bcast")] = nodeBroadcastEnabled;
|
||||||
|
|
||||||
JsonObject if_live = interfaces.createNestedObject("live");
|
JsonObject if_live = interfaces.createNestedObject("live");
|
||||||
if_live["en"] = receiveDirect;
|
if_live["en"] = receiveDirect; // UDP/Hyperion realtime
|
||||||
if_live[F("mso")] = useMainSegmentOnly;
|
if_live[F("mso")] = useMainSegmentOnly;
|
||||||
if_live["port"] = e131Port;
|
if_live["port"] = e131Port;
|
||||||
if_live[F("mc")] = e131Multicast;
|
if_live[F("mc")] = e131Multicast;
|
||||||
@ -888,6 +903,7 @@ void serializeConfig() {
|
|||||||
if_live[F("no-gc")] = arlsDisableGammaCorrection;
|
if_live[F("no-gc")] = arlsDisableGammaCorrection;
|
||||||
if_live[F("offset")] = arlsOffset;
|
if_live[F("offset")] = arlsOffset;
|
||||||
|
|
||||||
|
#ifndef WLED_DISABLE_ALEXA
|
||||||
JsonObject if_va = interfaces.createNestedObject("va");
|
JsonObject if_va = interfaces.createNestedObject("va");
|
||||||
if_va[F("alexa")] = alexaEnabled;
|
if_va[F("alexa")] = alexaEnabled;
|
||||||
|
|
||||||
@ -896,6 +912,7 @@ void serializeConfig() {
|
|||||||
if_va_macros.add(macroAlexaOff);
|
if_va_macros.add(macroAlexaOff);
|
||||||
|
|
||||||
if_va["p"] = alexaNumPresets;
|
if_va["p"] = alexaNumPresets;
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef WLED_ENABLE_MQTT
|
#ifdef WLED_ENABLE_MQTT
|
||||||
JsonObject if_mqtt = interfaces.createNestedObject("mqtt");
|
JsonObject if_mqtt = interfaces.createNestedObject("mqtt");
|
||||||
@ -912,13 +929,6 @@ void serializeConfig() {
|
|||||||
if_mqtt_topics[F("group")] = mqttGroupTopic;
|
if_mqtt_topics[F("group")] = mqttGroupTopic;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef WLED_DISABLE_ESPNOW
|
|
||||||
JsonObject remote = doc.createNestedObject(F("remote"));
|
|
||||||
remote[F("remote_enabled")] = enable_espnow_remote;
|
|
||||||
remote[F("linked_remote")] = linked_remote;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
#ifndef WLED_DISABLE_HUESYNC
|
#ifndef WLED_DISABLE_HUESYNC
|
||||||
JsonObject if_hue = interfaces.createNestedObject("hue");
|
JsonObject if_hue = interfaces.createNestedObject("hue");
|
||||||
if_hue["en"] = huePollingEnabled;
|
if_hue["en"] = huePollingEnabled;
|
||||||
@ -1033,7 +1043,7 @@ bool deserializeConfigSec() {
|
|||||||
JsonObject ap = doc["ap"];
|
JsonObject ap = doc["ap"];
|
||||||
getStringFromJson(apPass, ap["psk"] , 65);
|
getStringFromJson(apPass, ap["psk"] , 65);
|
||||||
|
|
||||||
JsonObject interfaces = doc["if"];
|
[[maybe_unused]] JsonObject interfaces = doc["if"];
|
||||||
|
|
||||||
#ifdef WLED_ENABLE_MQTT
|
#ifdef WLED_ENABLE_MQTT
|
||||||
JsonObject if_mqtt = interfaces["mqtt"];
|
JsonObject if_mqtt = interfaces["mqtt"];
|
||||||
@ -1072,7 +1082,7 @@ void serializeConfigSec() {
|
|||||||
JsonObject ap = doc.createNestedObject("ap");
|
JsonObject ap = doc.createNestedObject("ap");
|
||||||
ap["psk"] = apPass;
|
ap["psk"] = apPass;
|
||||||
|
|
||||||
JsonObject interfaces = doc.createNestedObject("if");
|
[[maybe_unused]] JsonObject interfaces = doc.createNestedObject("if");
|
||||||
#ifdef WLED_ENABLE_MQTT
|
#ifdef WLED_ENABLE_MQTT
|
||||||
JsonObject if_mqtt = interfaces.createNestedObject("mqtt");
|
JsonObject if_mqtt = interfaces.createNestedObject("mqtt");
|
||||||
if_mqtt["psk"] = mqttPass;
|
if_mqtt["psk"] = mqttPass;
|
||||||
|
@ -35,23 +35,59 @@ uint32_t color_blend(uint32_t color1, uint32_t color2, uint16_t blend, bool b16)
|
|||||||
* color add function that preserves ratio
|
* color add function that preserves ratio
|
||||||
* idea: https://github.com/Aircoookie/WLED/pull/2465 by https://github.com/Proto-molecule
|
* idea: https://github.com/Aircoookie/WLED/pull/2465 by https://github.com/Proto-molecule
|
||||||
*/
|
*/
|
||||||
uint32_t color_add(uint32_t c1, uint32_t c2)
|
uint32_t color_add(uint32_t c1, uint32_t c2, bool fast)
|
||||||
{
|
{
|
||||||
uint32_t r = R(c1) + R(c2);
|
if (fast) {
|
||||||
uint32_t g = G(c1) + G(c2);
|
uint8_t r = R(c1);
|
||||||
uint32_t b = B(c1) + B(c2);
|
uint8_t g = G(c1);
|
||||||
uint32_t w = W(c1) + W(c2);
|
uint8_t b = B(c1);
|
||||||
uint16_t max = r;
|
uint8_t w = W(c1);
|
||||||
if (g > max) max = g;
|
r = qadd8(r, R(c2));
|
||||||
if (b > max) max = b;
|
g = qadd8(g, G(c2));
|
||||||
if (w > max) max = w;
|
b = qadd8(b, B(c2));
|
||||||
if (max < 256) return RGBW32(r, g, b, w);
|
w = qadd8(w, W(c2));
|
||||||
else return RGBW32(r * 255 / max, g * 255 / max, b * 255 / max, w * 255 / max);
|
return RGBW32(r,g,b,w);
|
||||||
|
} else {
|
||||||
|
uint32_t r = R(c1) + R(c2);
|
||||||
|
uint32_t g = G(c1) + G(c2);
|
||||||
|
uint32_t b = B(c1) + B(c2);
|
||||||
|
uint32_t w = W(c1) + W(c2);
|
||||||
|
uint16_t max = r;
|
||||||
|
if (g > max) max = g;
|
||||||
|
if (b > max) max = b;
|
||||||
|
if (w > max) max = w;
|
||||||
|
if (max < 256) return RGBW32(r, g, b, w);
|
||||||
|
else return RGBW32(r * 255 / max, g * 255 / max, b * 255 / max, w * 255 / max);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* fades color toward black
|
||||||
|
* if using "video" method the resulting color will never become black unless it is already black
|
||||||
|
*/
|
||||||
|
uint32_t color_fade(uint32_t c1, uint8_t amount, bool video)
|
||||||
|
{
|
||||||
|
uint8_t r = R(c1);
|
||||||
|
uint8_t g = G(c1);
|
||||||
|
uint8_t b = B(c1);
|
||||||
|
uint8_t w = W(c1);
|
||||||
|
if (video) {
|
||||||
|
r = scale8_video(r, amount);
|
||||||
|
g = scale8_video(g, amount);
|
||||||
|
b = scale8_video(b, amount);
|
||||||
|
w = scale8_video(w, amount);
|
||||||
|
} else {
|
||||||
|
r = scale8(r, amount);
|
||||||
|
g = scale8(g, amount);
|
||||||
|
b = scale8(b, amount);
|
||||||
|
w = scale8(w, amount);
|
||||||
|
}
|
||||||
|
return RGBW32(r, g, b, w);
|
||||||
}
|
}
|
||||||
|
|
||||||
void setRandomColor(byte* rgb)
|
void setRandomColor(byte* rgb)
|
||||||
{
|
{
|
||||||
lastRandomIndex = strip.getMainSegment().get_random_wheel_index(lastRandomIndex);
|
lastRandomIndex = get_random_wheel_index(lastRandomIndex);
|
||||||
colorHStoRGB(lastRandomIndex*256,255,rgb);
|
colorHStoRGB(lastRandomIndex*256,255,rgb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -166,7 +166,7 @@
|
|||||||
#define CALL_MODE_NO_NOTIFY 5
|
#define CALL_MODE_NO_NOTIFY 5
|
||||||
#define CALL_MODE_FX_CHANGED 6 //no longer used
|
#define CALL_MODE_FX_CHANGED 6 //no longer used
|
||||||
#define CALL_MODE_HUE 7
|
#define CALL_MODE_HUE 7
|
||||||
#define CALL_MODE_PRESET_CYCLE 8
|
#define CALL_MODE_PRESET_CYCLE 8 //no longer used
|
||||||
#define CALL_MODE_BLYNK 9 //no longer used
|
#define CALL_MODE_BLYNK 9 //no longer used
|
||||||
#define CALL_MODE_ALEXA 10
|
#define CALL_MODE_ALEXA 10
|
||||||
#define CALL_MODE_WS_SEND 11 //special call mode, not for notifier, updates websocket only
|
#define CALL_MODE_WS_SEND 11 //special call mode, not for notifier, updates websocket only
|
||||||
@ -270,6 +270,10 @@
|
|||||||
#define COL_ORDER_GBR 5
|
#define COL_ORDER_GBR 5
|
||||||
#define COL_ORDER_MAX 5
|
#define COL_ORDER_MAX 5
|
||||||
|
|
||||||
|
//ESP-NOW
|
||||||
|
#define ESP_NOW_STATE_UNINIT 0
|
||||||
|
#define ESP_NOW_STATE_ON 1
|
||||||
|
#define ESP_NOW_STATE_ERROR 2
|
||||||
|
|
||||||
//Button type
|
//Button type
|
||||||
#define BTN_TYPE_NONE 0
|
#define BTN_TYPE_NONE 0
|
||||||
@ -313,10 +317,9 @@
|
|||||||
#define SEG_OPTION_MIRROR 3 //Indicates that the effect will be mirrored within the segment
|
#define SEG_OPTION_MIRROR 3 //Indicates that the effect will be mirrored within the segment
|
||||||
#define SEG_OPTION_FREEZE 4 //Segment contents will not be refreshed
|
#define SEG_OPTION_FREEZE 4 //Segment contents will not be refreshed
|
||||||
#define SEG_OPTION_RESET 5 //Segment runtime requires reset
|
#define SEG_OPTION_RESET 5 //Segment runtime requires reset
|
||||||
#define SEG_OPTION_TRANSITIONAL 6
|
#define SEG_OPTION_REVERSED_Y 6
|
||||||
#define SEG_OPTION_REVERSED_Y 7
|
#define SEG_OPTION_MIRROR_Y 7
|
||||||
#define SEG_OPTION_MIRROR_Y 8
|
#define SEG_OPTION_TRANSPOSED 8
|
||||||
#define SEG_OPTION_TRANSPOSED 9
|
|
||||||
|
|
||||||
//Segment differs return byte
|
//Segment differs return byte
|
||||||
#define SEG_DIFFERS_BRI 0x01 // opacity
|
#define SEG_DIFFERS_BRI 0x01 // opacity
|
||||||
@ -345,6 +348,7 @@
|
|||||||
#define ERR_FS_QUOTA 11 // The FS is full or the maximum file size is reached
|
#define ERR_FS_QUOTA 11 // The FS is full or the maximum file size is reached
|
||||||
#define ERR_FS_PLOAD 12 // It was attempted to load a preset that does not exist
|
#define ERR_FS_PLOAD 12 // It was attempted to load a preset that does not exist
|
||||||
#define ERR_FS_IRLOAD 13 // It was attempted to load an IR JSON cmd, but the "ir.json" file does not exist
|
#define ERR_FS_IRLOAD 13 // It was attempted to load an IR JSON cmd, but the "ir.json" file does not exist
|
||||||
|
#define ERR_FS_RMLOAD 14 // It was attempted to load an remote JSON cmd, but the "remote.json" file does not exist
|
||||||
#define ERR_FS_GENERAL 19 // A general unspecified filesystem error occured
|
#define ERR_FS_GENERAL 19 // A general unspecified filesystem error occured
|
||||||
#define ERR_OVERTEMP 30 // An attached temperature sensor has measured above threshold temperature (not implemented)
|
#define ERR_OVERTEMP 30 // An attached temperature sensor has measured above threshold temperature (not implemented)
|
||||||
#define ERR_OVERCURRENT 31 // An attached current sensor has measured a current above the threshold (not implemented)
|
#define ERR_OVERCURRENT 31 // An attached current sensor has measured a current above the threshold (not implemented)
|
||||||
|
@ -20,8 +20,8 @@
|
|||||||
--c-g: #2c1;
|
--c-g: #2c1;
|
||||||
--c-l: #48a;
|
--c-l: #48a;
|
||||||
--c-y: #a90;
|
--c-y: #a90;
|
||||||
--t-b: 0.5;
|
--t-b: .5;
|
||||||
--c-o: rgba(34, 34, 34, 0.9);
|
--c-o: rgba(34, 34, 34, .9);
|
||||||
--c-tb : rgba(34, 34, 34, var(--t-b));
|
--c-tb : rgba(34, 34, 34, var(--t-b));
|
||||||
--c-tba: rgba(102, 102, 102, var(--t-b));
|
--c-tba: rgba(102, 102, 102, var(--t-b));
|
||||||
--c-tbh: rgba(51, 51, 51, var(--t-b));
|
--c-tbh: rgba(51, 51, 51, var(--t-b));
|
||||||
@ -33,7 +33,7 @@
|
|||||||
--bbp: 9px 0 7px 0;
|
--bbp: 9px 0 7px 0;
|
||||||
--bhd: none;
|
--bhd: none;
|
||||||
--sgp: "block";
|
--sgp: "block";
|
||||||
--bmt: 0px;
|
--bmt: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
html {
|
html {
|
||||||
@ -88,7 +88,7 @@ a, a:visited {
|
|||||||
}
|
}
|
||||||
|
|
||||||
button {
|
button {
|
||||||
outline: none;
|
outline: 0;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -238,7 +238,7 @@ button {
|
|||||||
.flr {
|
.flr {
|
||||||
color: var(--c-f);
|
color: var(--c-f);
|
||||||
transform: rotate(0deg);
|
transform: rotate(0deg);
|
||||||
transition: transform 0.3s;
|
transition: transform .3s;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 0;
|
top: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
@ -258,13 +258,13 @@ button {
|
|||||||
#liveview {
|
#liveview {
|
||||||
height: 4px;
|
height: 4px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
border: 0px;
|
border: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
#liveview2D {
|
#liveview2D {
|
||||||
height: 90%;
|
height: 90%;
|
||||||
width: 90%;
|
width: 90%;
|
||||||
border: 0px;
|
border: 0;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 50%;
|
top: 50%;
|
||||||
left: 50%;
|
left: 50%;
|
||||||
@ -287,8 +287,8 @@ button {
|
|||||||
.tab button {
|
.tab button {
|
||||||
background-color: transparent;
|
background-color: transparent;
|
||||||
float: left;
|
float: left;
|
||||||
border: none;
|
border: 0;
|
||||||
transition: color 0.3s, background-color 0.3s;
|
transition: color .3s, background-color .3s;
|
||||||
font-size: 17px;
|
font-size: 17px;
|
||||||
color: var(--c-c);
|
color: var(--c-c);
|
||||||
min-width: 44px;
|
min-width: 44px;
|
||||||
@ -301,7 +301,7 @@ button {
|
|||||||
|
|
||||||
.bot button {
|
.bot button {
|
||||||
padding: var(--bbp);
|
padding: var(--bbp);
|
||||||
width:25%;
|
width: 25%;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -336,8 +336,9 @@ button {
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
width: calc(100%/var(--n));
|
width: calc(100%/var(--n));
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
border: 0px;
|
border: 0;
|
||||||
overflow: auto;
|
overflow-y: auto;
|
||||||
|
overflow-x: hidden;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
overscroll-behavior: none;
|
overscroll-behavior: none;
|
||||||
padding: 0 4px;
|
padding: 0 4px;
|
||||||
@ -388,8 +389,8 @@ button {
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
z-index: 11;
|
z-index: 11;
|
||||||
opacity: 0.95;
|
opacity: .95;
|
||||||
transition: 0.7s;
|
transition: .7s;
|
||||||
pointer-events: none;
|
pointer-events: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -399,7 +400,7 @@ button {
|
|||||||
position: sticky !important;
|
position: sticky !important;
|
||||||
top: 0;
|
top: 0;
|
||||||
z-index: 2;
|
z-index: 2;
|
||||||
margin: 0 auto auto;
|
margin: 0 auto auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.staybot {
|
.staybot {
|
||||||
@ -456,65 +457,50 @@ button {
|
|||||||
padding: 4px 2px;
|
padding: 4px 2px;
|
||||||
position: relative;
|
position: relative;
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
transition: opacity 0.5s linear, height 0.5s, transform 0.5s;
|
transition: opacity .5s linear, height .5s, transform .5s;
|
||||||
}
|
}
|
||||||
|
|
||||||
.filter {
|
.filter {
|
||||||
z-index: 1;
|
z-index: 1;
|
||||||
overflow: hidden;
|
/*overflow: visible;*/
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Tooltip text */
|
/* New tooltip */
|
||||||
.slider .tooltiptext, .option .tooltiptext {
|
.tooltip {
|
||||||
|
position: absolute;
|
||||||
|
opacity: 0;
|
||||||
visibility: hidden;
|
visibility: hidden;
|
||||||
|
transition: opacity .4s ease, visibility .4s ease;
|
||||||
background-color: var(--c-5);
|
background-color: var(--c-5);
|
||||||
/*border: 2px solid var(--c-2);*/
|
|
||||||
box-shadow: 4px 4px 10px 4px var(--c-1);
|
box-shadow: 4px 4px 10px 4px var(--c-1);
|
||||||
color: var(--c-f);
|
color: var(--c-f);
|
||||||
text-align: center;
|
text-align: center;
|
||||||
padding: 4px 8px;
|
padding: 8px 16px;
|
||||||
border-radius: 6px;
|
border-radius: 6px;
|
||||||
|
|
||||||
/* Position the tooltip text */
|
|
||||||
width: 160px;
|
|
||||||
position: absolute;
|
|
||||||
z-index: 1;
|
z-index: 1;
|
||||||
bottom: 80%;
|
|
||||||
left: 50%;
|
|
||||||
margin-left: -92px;
|
|
||||||
|
|
||||||
/* Ensure tooltip goes away when mouse leaves control */
|
|
||||||
pointer-events: none;
|
|
||||||
|
|
||||||
/* Fade in tooltip */
|
|
||||||
opacity: 0;
|
|
||||||
transition: opacity 0.75s;
|
|
||||||
}
|
}
|
||||||
.option .tooltiptext {
|
|
||||||
bottom: 120%;
|
.tooltip::after {
|
||||||
}
|
|
||||||
/* Tooltip arrow */
|
|
||||||
.slider .tooltiptext::after, .option .tooltiptext::after {
|
|
||||||
content: "";
|
content: "";
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 100%;
|
border: 8px;
|
||||||
left: 50%;
|
|
||||||
margin-left: -5px;
|
|
||||||
border-width: 5px;
|
|
||||||
border-style: solid;
|
border-style: solid;
|
||||||
border-color: var(--c-5) transparent transparent transparent;
|
border-color: var(--c-5) transparent transparent transparent;
|
||||||
}
|
top: 100%;
|
||||||
/* Show the tooltip text when you mouse over the tooltip container */
|
left: calc(50% - 8px);
|
||||||
.slider:hover .tooltiptext, .option .check:hover .tooltiptext {
|
z-index: 0;
|
||||||
visibility: visible;
|
}
|
||||||
|
|
||||||
|
.tooltip.visible {
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
|
visibility: visible;
|
||||||
}
|
}
|
||||||
|
|
||||||
.fade {
|
.fade {
|
||||||
visibility: hidden; /* hide it */
|
visibility: hidden; /* hide it */
|
||||||
opacity: 0; /* make it transparent */
|
opacity: 0; /* make it transparent */
|
||||||
transform: scaleY(0); /* shrink content */
|
transform: scaleY(0); /* shrink content */
|
||||||
height: 0px; /* force other elements to move */
|
height: 0; /* force other elements to move */
|
||||||
padding: 0; /* remove empty space */
|
padding: 0; /* remove empty space */
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -542,24 +528,24 @@ button {
|
|||||||
|
|
||||||
#toast.show {
|
#toast.show {
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
animation: fadein 0.5s, fadein 0.5s 2.5s reverse;
|
animation: fadein .5s, fadein .5s 2.5s reverse;
|
||||||
}
|
}
|
||||||
|
|
||||||
#toast.error {
|
#toast.error {
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
background-color: #b21;
|
background-color: #b21;
|
||||||
animation: fadein 0.5s;
|
animation: fadein .5s;
|
||||||
}
|
}
|
||||||
|
|
||||||
.modal {
|
.modal {
|
||||||
position:fixed;
|
position: fixed;
|
||||||
left: 0px;
|
left: 0;
|
||||||
bottom: 0px;
|
bottom: 0;
|
||||||
right: 0px;
|
right: 0;
|
||||||
top: calc(var(--th) - 1px);
|
top: calc(var(--th) - 1px);
|
||||||
background-color: var(--c-o);
|
background-color: var(--c-o);
|
||||||
transform: translateY(100%);
|
transform: translateY(100%);
|
||||||
transition: transform 0.4s;
|
transition: transform .4s;
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
font-size: 20px;
|
font-size: 20px;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
@ -641,7 +627,7 @@ button {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#heart {
|
#heart {
|
||||||
transition: color 0.9s;
|
transition: color .9s;
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
color: #f00;
|
color: #f00;
|
||||||
}
|
}
|
||||||
@ -720,7 +706,7 @@ input[type=range] {
|
|||||||
}
|
}
|
||||||
|
|
||||||
input[type=range]:focus {
|
input[type=range]:focus {
|
||||||
outline: none;
|
outline: 0;
|
||||||
}
|
}
|
||||||
input[type=range]::-webkit-slider-runnable-track {
|
input[type=range]::-webkit-slider-runnable-track {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
@ -743,7 +729,7 @@ input[type=range]::-moz-range-track {
|
|||||||
background-color: rgba(0, 0, 0, 0);
|
background-color: rgba(0, 0, 0, 0);
|
||||||
}
|
}
|
||||||
input[type=range]::-moz-range-thumb {
|
input[type=range]::-moz-range-thumb {
|
||||||
border: 0px solid rgba(0, 0, 0, 0);
|
border: 0 solid rgba(0, 0, 0, 0);
|
||||||
height: 16px;
|
height: 16px;
|
||||||
width: 16px;
|
width: 16px;
|
||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
@ -770,7 +756,7 @@ input[type=range]::-moz-range-thumb {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#briwrap {
|
#briwrap {
|
||||||
min-width: 267px;
|
min-width: 300px;
|
||||||
float: right;
|
float: right;
|
||||||
margin-top: var(--bmt);
|
margin-top: var(--bmt);
|
||||||
}
|
}
|
||||||
@ -789,11 +775,11 @@ input[type=range]::-moz-range-thumb {
|
|||||||
color: var(--c-d);
|
color: var(--c-d);
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
border-radius: 25px;
|
border-radius: 25px;
|
||||||
transition-duration: 0.3s;
|
transition-duration: .3s;
|
||||||
-webkit-backface-visibility: hidden;
|
-webkit-backface-visibility: hidden;
|
||||||
-webkit-transform:translate3d(0,0,0);
|
-webkit-transform: translate3d(0,0,0);
|
||||||
backface-visibility: hidden;
|
backface-visibility: hidden;
|
||||||
transform:translate3d(0,0,0);
|
transform: translate3d(0,0,0);
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
border: 1px solid var(--c-3);
|
border: 1px solid var(--c-3);
|
||||||
@ -893,13 +879,13 @@ select {
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
border: 0 solid var(--c-2);
|
border: 0 solid var(--c-2);
|
||||||
border-radius: 20px;
|
border-radius: 20px;
|
||||||
transition-duration: 0.5s;
|
transition-duration: .5s;
|
||||||
-webkit-backface-visibility: hidden;
|
-webkit-backface-visibility: hidden;
|
||||||
-webkit-transform:translate3d(0,0,0);
|
-webkit-transform: translate3d(0,0,0);
|
||||||
-webkit-appearance: none;
|
-webkit-appearance: none;
|
||||||
-moz-appearance: none;
|
-moz-appearance: none;
|
||||||
backface-visibility: hidden;
|
backface-visibility: hidden;
|
||||||
transform:translate3d(0,0,0);
|
transform: translate3d(0,0,0);
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
}
|
}
|
||||||
#tt {
|
#tt {
|
||||||
@ -917,15 +903,15 @@ div.sel-p {
|
|||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
div.sel-p:after {
|
div.sel-p:after {
|
||||||
content: "";
|
content: "";
|
||||||
position: absolute;
|
position: absolute;
|
||||||
right: 10px;
|
right: 10px;
|
||||||
top: 22px;
|
top: 22px;
|
||||||
width: 0;
|
width: 0;
|
||||||
height: 0;
|
height: 0;
|
||||||
border-left: 8px solid transparent;
|
border-left: 8px solid transparent;
|
||||||
border-right: 8px solid transparent;
|
border-right: 8px solid transparent;
|
||||||
border-top: 8px solid var(--c-f);
|
border-top: 8px solid var(--c-f);
|
||||||
}
|
}
|
||||||
select.sel-ple {
|
select.sel-ple {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
@ -942,13 +928,13 @@ input[type=number],
|
|||||||
input[type=text] {
|
input[type=text] {
|
||||||
background: var(--c-3);
|
background: var(--c-3);
|
||||||
color: var(--c-f);
|
color: var(--c-f);
|
||||||
border: 0px solid var(--c-2);
|
border: 0 solid var(--c-2);
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
/*margin: 6px 6px 6px 0;*/
|
/*margin: 6px 6px 6px 0;*/
|
||||||
font-size: 19px;
|
font-size: 19px;
|
||||||
transition: background-color 0.2s;
|
transition: background-color .2s;
|
||||||
outline: none;
|
outline: 0;
|
||||||
-webkit-appearance: textfield;
|
-webkit-appearance: textfield;
|
||||||
-moz-appearance: textfield;
|
-moz-appearance: textfield;
|
||||||
appearance: textfield;
|
appearance: textfield;
|
||||||
@ -988,7 +974,7 @@ textarea {
|
|||||||
height: 90px;
|
height: 90px;
|
||||||
border-radius: 5px;
|
border-radius: 5px;
|
||||||
border: 2px solid var(--c-5);
|
border: 2px solid var(--c-5);
|
||||||
outline: none;
|
outline: 0;
|
||||||
resize: none;
|
resize: none;
|
||||||
font-size: 19px;
|
font-size: 19px;
|
||||||
padding: 5px;
|
padding: 5px;
|
||||||
@ -1041,7 +1027,7 @@ textarea {
|
|||||||
top: 1px;
|
top: 1px;
|
||||||
}
|
}
|
||||||
.plname {
|
.plname {
|
||||||
top:0;
|
top: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* preset id number */
|
/* preset id number */
|
||||||
@ -1121,8 +1107,8 @@ textarea {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.revchkl {
|
.revchkl {
|
||||||
padding: 4px 0px 0px 35px;
|
padding: 4px 0 0 35px;
|
||||||
margin-bottom: 0px;
|
margin-bottom: 0;
|
||||||
margin-top: 8px;
|
margin-top: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1218,9 +1204,9 @@ TD .checkmark, TD .radiomark {
|
|||||||
.seg, .pres {
|
.seg, .pres {
|
||||||
background-color: var(--c-2);
|
background-color: var(--c-2);
|
||||||
/*color: var(--c-f);*/ /* seems to affect only the Add segment button, which should be same color as reset segments */
|
/*color: var(--c-f);*/ /* seems to affect only the Add segment button, which should be same color as reset segments */
|
||||||
border: 0px solid var(--c-f);
|
border: 0 solid var(--c-f);
|
||||||
text-align: left;
|
text-align: left;
|
||||||
transition: background-color 0.5s;
|
transition: background-color .5s;
|
||||||
border-radius: 21px;
|
border-radius: 21px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1237,8 +1223,8 @@ TD .checkmark, TD .radiomark {
|
|||||||
/* checkmark labels */
|
/* checkmark labels */
|
||||||
.filter .fchkl, .option .ochkl {
|
.filter .fchkl, .option .ochkl {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
min-width: 0.7em;
|
min-width: .7em;
|
||||||
padding: 1px 4px 4px 32px;
|
padding: 1px 4px 1px 32px;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
line-height: 24px;
|
line-height: 24px;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
@ -1263,15 +1249,15 @@ TD .checkmark, TD .radiomark {
|
|||||||
/* list wrapper */
|
/* list wrapper */
|
||||||
.list {
|
.list {
|
||||||
position: relative;
|
position: relative;
|
||||||
transition: background-color 0.5s;
|
transition: background-color .5s;
|
||||||
margin: auto auto 10px;
|
margin: auto auto 10px;
|
||||||
line-height: 24px;
|
line-height: 24px;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* list item */
|
/* list item */
|
||||||
.lstI {
|
.lstI {
|
||||||
align-items: center;
|
align-items: center;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
background-color: var(--c-2);
|
background-color: var(--c-2);
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
position: -webkit-sticky;
|
position: -webkit-sticky;
|
||||||
@ -1293,7 +1279,7 @@ TD .checkmark, TD .radiomark {
|
|||||||
|
|
||||||
#segcont .seg:hover:not([class*="expanded"]),
|
#segcont .seg:hover:not([class*="expanded"]),
|
||||||
.lstI:hover:not([class*="expanded"]) {
|
.lstI:hover:not([class*="expanded"]) {
|
||||||
background: var(--c-5);
|
background: var(--c-5);
|
||||||
}
|
}
|
||||||
|
|
||||||
.selected .checkmark,
|
.selected .checkmark,
|
||||||
@ -1313,7 +1299,7 @@ TD .checkmark, TD .radiomark {
|
|||||||
.lstI.sticky,
|
.lstI.sticky,
|
||||||
.lstI.selected {
|
.lstI.selected {
|
||||||
z-index: 1;
|
z-index: 1;
|
||||||
box-shadow: 0px 0px 10px 4px var(--c-1);
|
box-shadow: 0 0 10px 4px var(--c-1);
|
||||||
}
|
}
|
||||||
|
|
||||||
#pcont .selected:not([class*="expanded"]) {
|
#pcont .selected:not([class*="expanded"]) {
|
||||||
@ -1370,8 +1356,8 @@ TD .checkmark, TD .radiomark {
|
|||||||
display: block;
|
display: block;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
padding: 8px 40px 8px 44px;
|
padding: 8px 40px 8px 44px;
|
||||||
margin: 5px auto 0;
|
margin: 5px auto 0;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
border-radius: 21px;
|
border-radius: 21px;
|
||||||
background: var(--c-2);
|
background: var(--c-2);
|
||||||
@ -1457,7 +1443,7 @@ TD .checkmark, TD .radiomark {
|
|||||||
}
|
}
|
||||||
::-webkit-scrollbar-thumb {
|
::-webkit-scrollbar-thumb {
|
||||||
background: var(--c-sb);
|
background: var(--c-sb);
|
||||||
opacity: 0.2;
|
opacity: .2;
|
||||||
border-radius: 5px;
|
border-radius: 5px;
|
||||||
}
|
}
|
||||||
::-webkit-scrollbar-thumb:hover {
|
::-webkit-scrollbar-thumb:hover {
|
||||||
@ -1483,7 +1469,7 @@ TD .checkmark, TD .radiomark {
|
|||||||
|
|
||||||
@media all and (max-width: 335px) {
|
@media all and (max-width: 335px) {
|
||||||
.sliderbubble {
|
.sliderbubble {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -48,7 +48,7 @@
|
|||||||
});
|
});
|
||||||
setTimeout(()=>{h.appendChild(l)},100);
|
setTimeout(()=>{h.appendChild(l)},100);
|
||||||
});
|
});
|
||||||
setTimeout(()=>{h.appendChild(l)},100);
|
setTimeout(()=>{h.appendChild(l)},200);
|
||||||
</script>
|
</script>
|
||||||
<link rel="stylesheet" href="index.css">
|
<link rel="stylesheet" href="index.css">
|
||||||
</head>
|
</head>
|
||||||
@ -72,8 +72,8 @@
|
|||||||
</div>
|
</div>
|
||||||
<div id="briwrap">
|
<div id="briwrap">
|
||||||
<p class="hd">Brightness</p>
|
<p class="hd">Brightness</p>
|
||||||
<div class="slider" style="padding-right:32px;">
|
<div class="slider">
|
||||||
<i class="icons slider-icon" onclick="tglTheme()" style="transform: translate(-32px,5px);"></i>
|
<i class="icons slider-icon" onclick="tglTheme()" style="transform: translateY(2px);"></i>
|
||||||
<div class="sliderwrap il">
|
<div class="sliderwrap il">
|
||||||
<input id="sliderBri" onchange="setBri()" oninput="updateTrail(this)" max="255" min="1" type="range" value="128" />
|
<input id="sliderBri" onchange="setBri()" oninput="updateTrail(this)" max="255" min="1" type="range" value="128" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
@ -88,98 +88,91 @@
|
|||||||
<div class ="container">
|
<div class ="container">
|
||||||
<div id="Colors" class="tabcontent">
|
<div id="Colors" class="tabcontent">
|
||||||
<div id="picker" class="noslide"></div>
|
<div id="picker" class="noslide"></div>
|
||||||
<div id="hwrap" class="slider" style="margin-top: 20px;">
|
<div id="hwrap" class="slider">
|
||||||
<div class="sliderwrap il">
|
<div tooltip="Hue" class="sliderwrap il">
|
||||||
<input id="sliderH" class="noslide" oninput="fromH()" onchange="setColor(0)" max="359" min="0" type="range" value="0" step="any">
|
<input id="sliderH" class="noslide" oninput="fromH()" onchange="setColor(0)" max="359" min="0" type="range" value="0" step="any">
|
||||||
<div class="sliderdisplay" style="background: linear-gradient(90deg, #f00 2%, #ff0 19%, #0f0 35%, #0ff 52%, #00f 68%, #f0f 85%, #f00)"></div>
|
<div class="sliderdisplay" style="background: linear-gradient(90deg, #f00 2%, #ff0 19%, #0f0 35%, #0ff 52%, #00f 68%, #f0f 85%, #f00)"></div>
|
||||||
</div>
|
</div>
|
||||||
<span class="tooltiptext">Hue</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="swrap" class="slider">
|
<div id="swrap" class="slider">
|
||||||
<div class="sliderwrap il">
|
<div tooltip="Saturation" class="sliderwrap il">
|
||||||
<input id="sliderS" class="noslide" oninput="fromS()" onchange="setColor(0)" max="100" min="0" type="range" value="100" step="any">
|
<input id="sliderS" class="noslide" oninput="fromS()" onchange="setColor(0)" max="100" min="0" type="range" value="100" step="any">
|
||||||
<div class="sliderdisplay" style="background: linear-gradient(90deg, #aaa 0%, #f00)"></div>
|
<div class="sliderdisplay" style="background: linear-gradient(90deg, #aaa 0%, #f00)"></div>
|
||||||
</div>
|
</div>
|
||||||
<span class="tooltiptext">Saturation</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="vwrap" class="slider">
|
<div id="vwrap" class="slider">
|
||||||
<div class="sliderwrap il">
|
<div tooltip="Value/Brightness" class="sliderwrap il">
|
||||||
<input id="sliderV" class="noslide" oninput="fromV()" onchange="setColor(0)" max="100" min="0" type="range" value="100" step="any" />
|
<input id="sliderV" class="noslide" oninput="fromV()" onchange="setColor(0)" max="100" min="0" type="range" value="100" step="any" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
</div>
|
</div>
|
||||||
<span class="tooltiptext">Value/Brightness</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="kwrap" class="slider">
|
<div id="kwrap" class="slider">
|
||||||
<div class="sliderwrap il">
|
<div tooltip="Kelvin/Temperature" class="sliderwrap il">
|
||||||
<input id="sliderK" class="noslide" oninput="fromK()" onchange="setColor(0)" max="10091" min="1900" type="range" value="6550" />
|
<input id="sliderK" class="noslide" oninput="fromK()" onchange="setColor(0)" max="10091" min="1900" type="range" value="6550" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
</div>
|
</div>
|
||||||
<span class="tooltiptext">Kelvin/Temperature</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="rgbwrap">
|
<div id="rgbwrap">
|
||||||
<!--p class="labels hd">RGB color</p-->
|
|
||||||
<div id="rwrap" class="slider">
|
<div id="rwrap" class="slider">
|
||||||
<div class="sliderwrap il">
|
<div tooltip="Red channel" class="sliderwrap il">
|
||||||
<input id="sliderR" class="noslide" oninput="fromRgb()" onchange="setColor(0)" max="255" min="0" type="range" value="128" />
|
<input id="sliderR" class="noslide" oninput="fromRgb()" onchange="setColor(0)" max="255" min="0" type="range" value="128" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
</div>
|
</div>
|
||||||
<span class="tooltiptext">Red channel</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="gwrap" class="slider">
|
<div id="gwrap" class="slider">
|
||||||
<div class="sliderwrap il">
|
<div tooltip="Green channel" class="sliderwrap il">
|
||||||
<input id="sliderG" class="noslide" oninput="fromRgb()" onchange="setColor(0)" max="255" min="0" type="range" value="128" />
|
<input id="sliderG" class="noslide" oninput="fromRgb()" onchange="setColor(0)" max="255" min="0" type="range" value="128" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
</div>
|
</div>
|
||||||
<span class="tooltiptext">Green channel</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="bwrap" class="slider">
|
<div id="bwrap" class="slider">
|
||||||
<div class="sliderwrap il">
|
<div tooltip="Blue channel" class="sliderwrap il">
|
||||||
<input id="sliderB" class="noslide" oninput="fromRgb()" onchange="setColor(0)" max="255" min="0" type="range" value="128" />
|
<input id="sliderB" class="noslide" oninput="fromRgb()" onchange="setColor(0)" max="255" min="0" type="range" value="128" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
</div>
|
</div>
|
||||||
<span class="tooltiptext">Blue channel</span>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div id="wwrap" class="slider">
|
<div id="wwrap" class="slider">
|
||||||
<!--p class="labels hd">White channel</p-->
|
<div id="whibri" tooltip="White channel" class="sliderwrap il">
|
||||||
<div id="whibri" class="sliderwrap il">
|
|
||||||
<input id="sliderW" class="noslide" oninput="fromW()" onchange="setColor(0)" max="255" min="0" type="range" value="128" />
|
<input id="sliderW" class="noslide" oninput="fromW()" onchange="setColor(0)" max="255" min="0" type="range" value="128" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
</div>
|
</div>
|
||||||
<span class="tooltiptext">White channel</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="wbal" class="slider">
|
<div id="wbal" class="slider">
|
||||||
<!--p class="labels hd">White balance</p-->
|
<div tooltip="White balance" class="sliderwrap il">
|
||||||
<div class="sliderwrap il">
|
|
||||||
<input id="sliderA" class="noslide" onchange="setBalance(this.value)" max="255" min="0" type="range" value="128" />
|
<input id="sliderA" class="noslide" onchange="setBalance(this.value)" max="255" min="0" type="range" value="128" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
</div>
|
</div>
|
||||||
<span class="tooltiptext">White balance</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="qcs-w">
|
<div id="qcs-w">
|
||||||
<div class="qcs" onclick="pC('#ff0000');" title="Red" style="background-color:#ff0000;"></div>
|
<div class="qcs" onclick="pC('#ff0000');" tooltip="Red" style="background-color:#ff0000;"></div>
|
||||||
<div class="qcs" onclick="pC('#ffa000');" title="Orange" style="background-color:#ffa000;"></div>
|
<div class="qcs" onclick="pC('#ffa000');" tooltip="Orange" style="background-color:#ffa000;"></div>
|
||||||
<div class="qcs" onclick="pC('#ffc800');" title="Yellow" style="background-color:#ffc800;"></div>
|
<div class="qcs" onclick="pC('#ffc800');" tooltip="Yellow" style="background-color:#ffc800;"></div>
|
||||||
<div class="qcs" onclick="pC('#ffe0a0');" title="Warm White" style="background-color:#ffe0a0;"></div>
|
<div class="qcs" onclick="pC('#ffe0a0');" tooltip="Warm White" style="background-color:#ffe0a0;"></div>
|
||||||
<div class="qcs" onclick="pC('#ffffff');" title="White" style="background-color:#ffffff;"></div>
|
<div class="qcs" onclick="pC('#ffffff');" tooltip="White" style="background-color:#ffffff;"></div>
|
||||||
<div class="qcs qcsb" onclick="pC('#000000');" title="Black" style="background-color:#000000;"></div><br>
|
<div class="qcs qcsb" onclick="pC('#000000');" tooltip="Black" style="background-color:#000000;"></div><br>
|
||||||
<div class="qcs" onclick="pC('#ff00ff');" title="Pink" style="background-color:#ff00ff;"></div>
|
<div class="qcs" onclick="pC('#ff00ff');" tooltip="Pink" style="background-color:#ff00ff;"></div>
|
||||||
<div class="qcs" onclick="pC('#0000ff');" title="Blue" style="background-color:#0000ff;"></div>
|
<div class="qcs" onclick="pC('#0000ff');" tooltip="Blue" style="background-color:#0000ff;"></div>
|
||||||
<div class="qcs" onclick="pC('#00ffc8');" title="Cyan" style="background-color:#00ffc8;"></div>
|
<div class="qcs" onclick="pC('#00ffc8');" tooltip="Cyan" style="background-color:#00ffc8;"></div>
|
||||||
<div class="qcs" onclick="pC('#08ff00');" title="Green" style="background-color:#08ff00;"></div>
|
<div class="qcs" onclick="pC('#08ff00');" tooltip="Green" style="background-color:#08ff00;"></div>
|
||||||
<div class="qcs" onclick="pC('rnd');" title="Random" style="background:linear-gradient(to right, red, orange, yellow, green, blue, purple);transform: translateY(-11px);">R</div>
|
<div class="qcs" onclick="pC('rnd');" tooltip="Random" style="background:linear-gradient(to right, red, orange, yellow, green, blue, purple);transform: translateY(-11px);">R</div>
|
||||||
</div>
|
</div>
|
||||||
<div id="csl">
|
<div id="csl">
|
||||||
<button id="csl0" class="btn xxs cl" onclick="selectSlot(0);" data-r="0" data-g="0" data-b="0" data-w="0">1</button>
|
<button id="csl0" tooltip="Select slot" class="btn xxs cl" onclick="selectSlot(0);" data-r="0" data-g="0" data-b="0" data-w="0">1</button>
|
||||||
<button id="csl1" class="btn xxs cl" onclick="selectSlot(1);" data-r="0" data-g="0" data-b="0" data-w="0">2</button>
|
<button id="csl1" tooltip="Select slot" class="btn xxs cl" onclick="selectSlot(1);" data-r="0" data-g="0" data-b="0" data-w="0">2</button>
|
||||||
<button id="csl2" class="btn xxs cl" onclick="selectSlot(2);" data-r="0" data-g="0" data-b="0" data-w="0">3</button>
|
<button id="csl2" tooltip="Select slot" class="btn xxs cl" onclick="selectSlot(2);" data-r="0" data-g="0" data-b="0" data-w="0">3</button>
|
||||||
</div>
|
</div>
|
||||||
<p class="labels h" id="cslLabel"></p>
|
<p class="labels h" id="cslLabel"></p>
|
||||||
<div id="hexw">
|
<div id="hexw">
|
||||||
<i class="icons sel-icon" onclick="tglRgb()"></i>
|
<i class="icons sel-icon" onclick="tglRgb()"></i>
|
||||||
<input id="hexc" type="text" class="noslide" onkeydown="hexEnter()" autocomplete="off" maxlength="8" />
|
<input id="hexc" tooltip="Hex RGB" type="text" class="noslide" onkeydown="hexEnter()" autocomplete="off" maxlength="8" />
|
||||||
<button id="hexcnf" class="btn btn-xs" onclick="fromHex();"><i class="icons btn-icon"></i></button>
|
<button id="hexcnf" class="btn btn-xs" onclick="fromHex();"><i class="icons btn-icon"></i></button>
|
||||||
</div>
|
</div>
|
||||||
|
<div style="padding: 8px 0;">
|
||||||
|
<button class="btn btn-xs" tooltip="Pixel Magic Tool" type="button" id="pxmb" onclick="window.location.href=getURL('/pxmagic.htm')"><i class="icons btn-icon"></i></button>
|
||||||
|
<button class="btn btn-xs" tooltip="Add custom palette" type="button" onclick="window.location.href=getURL('/cpal.htm')"><i class="icons btn-icon"></i></button>
|
||||||
|
<button class="btn btn-xs" tooltip="Remove custom palette" type="button" id="rmPal" onclick="palettesData=null;localStorage.removeItem('wledPalx');requestJson({rmcpal:true});setTimeout(loadPalettes,250,loadPalettesData);"><i class="icons btn-icon"></i></button>
|
||||||
|
</div>
|
||||||
<p class="labels" id="pall"><i class="icons sel-icon" onclick="tglHex()"></i> Color palette</p>
|
<p class="labels" id="pall"><i class="icons sel-icon" onclick="tglHex()"></i> Color palette</p>
|
||||||
<div id="palw" class="il">
|
<div id="palw" class="il">
|
||||||
<div class="staytop fnd">
|
<div class="staytop fnd">
|
||||||
@ -198,10 +191,6 @@
|
|||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div style="padding-bottom: 10px;">
|
|
||||||
<button class="btn btn-xs" type="button" onclick="window.location.href=getURL('/cpal.htm')"><i class="icons btn-icon"></i></button>
|
|
||||||
<button class="btn btn-xs" type="button" onclick="palettesData=null;localStorage.removeItem('wledPalx');requestJson({rmcpal:true});setTimeout(loadPalettes,250,loadPalettesData);"><i class="icons btn-icon"></i></button>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -227,86 +216,81 @@
|
|||||||
</div>
|
</div>
|
||||||
<div id="sliders">
|
<div id="sliders">
|
||||||
<div id="filters" class="filter">
|
<div id="filters" class="filter">
|
||||||
<label id="filterPal" class="check fchkl">🎨
|
<label id="filterPal" tooltip="Uses palette" class="check fchkl">🎨
|
||||||
<input type="checkbox" data-flt="🎨" onchange="filterFx(this)">
|
<input type="checkbox" data-flt="🎨" onchange="filterFx(this)">
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>
|
</label>
|
||||||
<label id="filter0D" class="check fchkl hide">•
|
<label id="filter0D" tooltip="Single pixel" class="check fchkl hide">•
|
||||||
<input type="checkbox" data-flt="•" onchange="filterFx(this)">
|
<input type="checkbox" data-flt="•" onchange="filterFx(this)">
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>
|
</label>
|
||||||
<label id="filter1D" class="check fchkl">⋮
|
<label id="filter1D" tooltip="1D" class="check fchkl">⋮
|
||||||
<input type="checkbox" data-flt="⋮" onchange="filterFx(this)">
|
<input type="checkbox" data-flt="⋮" onchange="filterFx(this)">
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>
|
</label>
|
||||||
<label id="filter2D" class="check fchkl">▦
|
<label id="filter2D" tooltip="2D" class="check fchkl">▦
|
||||||
<input type="checkbox" data-flt="▦" onchange="filterFx(this)">
|
<input type="checkbox" data-flt="▦" onchange="filterFx(this)">
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>
|
</label>
|
||||||
<label id="filterVol" class="check fchkl">♪
|
<label id="filterVol" tooltip="Volume" class="check fchkl">♪
|
||||||
<input type="checkbox" data-flt="♪" onchange="filterFx(this)">
|
<input type="checkbox" data-flt="♪" onchange="filterFx(this)">
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>
|
</label>
|
||||||
<label id="filterFreq" class="check fchkl">♫
|
<label id="filterFreq" tooltip="Frequency" class="check fchkl">♫
|
||||||
<input type="checkbox" data-flt="♫" onchange="filterFx(this)">
|
<input type="checkbox" data-flt="♫" onchange="filterFx(this)">
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
<div id="slider0" class="slider">
|
<div id="slider0" class="slider">
|
||||||
<i class="icons slider-icon" onclick="tglFreeze()"></i>
|
<i class="icons slider-icon" onclick="tglFreeze()"></i>
|
||||||
<div class="sliderwrap il">
|
<div tooltip="Effect speed" class="sliderwrap il">
|
||||||
<input id="sliderSpeed" class="noslide" onchange="setSpeed()" oninput="updateTrail(this)" max="255" min="0" type="range" value="128" />
|
<input id="sliderSpeed" class="noslide" onchange="setSpeed()" oninput="updateTrail(this)" max="255" min="0" type="range" value="128" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
</div>
|
</div>
|
||||||
<output class="sliderbubble"></output>
|
<output class="sliderbubble"></output>
|
||||||
<span id="sliderLabel0" class="tooltiptext">Effect speed</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="slider1" class="slider">
|
<div id="slider1" class="slider">
|
||||||
<i class="icons slider-icon" onclick="tglLabels()"></i>
|
<i class="icons slider-icon" onclick="tglLabels()"></i>
|
||||||
<div class="sliderwrap il">
|
<div tooltip="Effect intensity" class="sliderwrap il">
|
||||||
<input id="sliderIntensity" class="noslide" onchange="setIntensity()" oninput="updateTrail(this)" max="255" min="0" type="range" value="128" />
|
<input id="sliderIntensity" class="noslide" onchange="setIntensity()" oninput="updateTrail(this)" max="255" min="0" type="range" value="128" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
</div>
|
</div>
|
||||||
<output class="sliderbubble"></output>
|
<output class="sliderbubble"></output>
|
||||||
<span class="tooltiptext" id="sliderLabel1">Effect intensity</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="slider2" class="slider hide">
|
<div id="slider2" class="slider hide">
|
||||||
<i class="icons slider-icon"></i>
|
<i class="icons slider-icon"></i>
|
||||||
<div class="sliderwrap il">
|
<div tooltip="Custom 1" class="sliderwrap il">
|
||||||
<input id="sliderC1" class="noslide" onchange="setCustom(1)" oninput="updateTrail(this)" max="255" min="0" type="range" value="0" />
|
<input id="sliderC1" class="noslide" onchange="setCustom(1)" oninput="updateTrail(this)" max="255" min="0" type="range" value="0" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
</div>
|
</div>
|
||||||
<output class="sliderbubble"></output>
|
<output class="sliderbubble"></output>
|
||||||
<span class="tooltiptext" id="sliderLabel2">Custom 1</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="slider3" class="slider hide">
|
<div id="slider3" class="slider hide">
|
||||||
<i class="icons slider-icon"></i>
|
<i class="icons slider-icon"></i>
|
||||||
<div class="sliderwrap il">
|
<div tooltip="Custom 2" class="sliderwrap il">
|
||||||
<input id="sliderC2" class="noslide" onchange="setCustom(2)" oninput="updateTrail(this)" max="255" min="0" type="range" value="0" />
|
<input id="sliderC2" class="noslide" onchange="setCustom(2)" oninput="updateTrail(this)" max="255" min="0" type="range" value="0" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
</div>
|
</div>
|
||||||
<output class="sliderbubble"></output>
|
<output class="sliderbubble"></output>
|
||||||
<span class="tooltiptext" id="sliderLabel3">Custom 2</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="slider4" class="slider hide">
|
<div id="slider4" class="slider hide">
|
||||||
<i class="icons slider-icon"></i>
|
<i class="icons slider-icon"></i>
|
||||||
<div class="sliderwrap il">
|
<div tooltip="Custom 3" class="sliderwrap il">
|
||||||
<input id="sliderC3" class="noslide" onchange="setCustom(3)" oninput="updateTrail(this)" max="31" min="0" type="range" value="0" />
|
<input id="sliderC3" class="noslide" onchange="setCustom(3)" oninput="updateTrail(this)" max="31" min="0" type="range" value="0" />
|
||||||
<div class="sliderdisplay"></div>
|
<div class="sliderdisplay"></div>
|
||||||
</div>
|
</div>
|
||||||
<output class="sliderbubble"></output>
|
<output class="sliderbubble"></output>
|
||||||
<span class="tooltiptext" id="sliderLabel4">Custom 3</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div id="fxopt" class="option fade">
|
<div id="fxopt" class="option fade">
|
||||||
<label id="opt0" class="check ochkl hide"><i class="icons"></i><span class="tooltiptext" id="optLabel0">Check 1</span>
|
<label id="opt0" tooltip="Check 1" class="check ochkl hide"><i class="icons"></i>
|
||||||
<input id="checkO1" type="checkbox" onchange="setOption(1, this.checked)">
|
<input id="checkO1" type="checkbox" onchange="setOption(1, this.checked)">
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>
|
</label>
|
||||||
<label id="opt1" class="check ochkl hide"><i class="icons"></i><span class="tooltiptext" id="optLabel1">Check 2</span>
|
<label id="opt1" tooltip="Check 2" class="check ochkl hide"><i class="icons"></i>
|
||||||
<input id="checkO2" type="checkbox" onchange="setOption(2, this.checked)">
|
<input id="checkO2" type="checkbox" onchange="setOption(2, this.checked)">
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>
|
</label>
|
||||||
<label id="opt2" class="check ochkl hide"><i class="icons"></i><span class="tooltiptext" id="optLabel2">Check 3</span>
|
<label id="opt2" tooltip="Check 3" class="check ochkl hide"><i class="icons"></i>
|
||||||
<input id="checkO3" type="checkbox" onchange="setOption(3, this.checked)">
|
<input id="checkO3" type="checkbox" onchange="setOption(3, this.checked)">
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>
|
</label>
|
||||||
@ -392,6 +376,7 @@
|
|||||||
<button class="btn" onclick="setLor(2)">Override until reboot</button><br>
|
<button class="btn" onclick="setLor(2)">Override until reboot</button><br>
|
||||||
<span class="h">For best performance, it is recommended to turn off the streaming source when not in use.</span>
|
<span class="h">For best performance, it is recommended to turn off the streaming source when not in use.</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<i id="roverstar" class="icons huge" onclick="setLor(0)"></i><br>
|
<i id="roverstar" class="icons huge" onclick="setLor(0)"></i><br>
|
||||||
<script src="index.js"></script>
|
<script src="index.js"></script>
|
||||||
</body>
|
</body>
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
//page js
|
//page js
|
||||||
var loc = false, locip, locproto = "http:";
|
var loc = false, locip, locproto = "http:";
|
||||||
var isOn = false, nlA = false, isLv = false, isInfo = false, isNodes = false, syncSend = false, syncTglRecv = true;
|
var isOn = false, nlA = false, isLv = false, isInfo = false, isNodes = false, syncSend = false/*, syncTglRecv = true*/;
|
||||||
var hasWhite = false, hasRGB = false, hasCCT = false;
|
var hasWhite = false, hasRGB = false, hasCCT = false;
|
||||||
var nlDur = 60, nlTar = 0;
|
var nlDur = 60, nlTar = 0;
|
||||||
var nlMode = false;
|
var nlMode = false;
|
||||||
@ -26,17 +26,16 @@ var ws, cpick, ranges, wsRpt=0;
|
|||||||
var cfg = {
|
var cfg = {
|
||||||
theme:{base:"dark", bg:{url:"", random: false, randomGrayscale: false, randomBlur: false}, alpha:{bg:0.6,tab:0.8}, color:{bg:""}},
|
theme:{base:"dark", bg:{url:"", random: false, randomGrayscale: false, randomBlur: false}, alpha:{bg:0.6,tab:0.8}, color:{bg:""}},
|
||||||
comp :{colors:{picker: true, rgb: false, quick: true, hex: false},
|
comp :{colors:{picker: true, rgb: false, quick: true, hex: false},
|
||||||
labels:true, pcmbot:false, pid:true, seglen:false, segpwr:false, segexp:false,
|
labels:true, pcmbot:false, pid:true, seglen:false, segpwr:false, segexp:false,
|
||||||
css:true, hdays:false, fxdef:true}
|
css:true, hdays:false, fxdef:true, on:0, off:0}
|
||||||
};
|
};
|
||||||
var hol = [
|
var hol = [
|
||||||
[0,11,24,4,"https://aircoookie.github.io/xmas.png"], // christmas
|
[0,11,24,4,"https://aircoookie.github.io/xmas.png"], // christmas
|
||||||
[0,2,17,1,"https://images.alphacoders.com/491/491123.jpg"], // st. Patrick's day
|
[0,2,17,1,"https://images.alphacoders.com/491/491123.jpg"], // st. Patrick's day
|
||||||
[2025,3,20,2,"https://aircoookie.github.io/easter.png"],
|
[2025,3,20,2,"https://aircoookie.github.io/easter.png"],
|
||||||
[2023,3,9,2,"https://aircoookie.github.io/easter.png"],
|
|
||||||
[2024,2,31,2,"https://aircoookie.github.io/easter.png"],
|
[2024,2,31,2,"https://aircoookie.github.io/easter.png"],
|
||||||
[0,6,4,1,"https://initiate.alphacoders.com/download/wallpaper/516792/images/jpg/510921363292536"], // 4th of July
|
[0,6,4,1,"https://images.alphacoders.com/516/516792.jpg"], // 4th of July
|
||||||
[0,0,1,1,"https://initiate.alphacoders.com/download/wallpaper/1198800/images/jpg/2522807481585600"] // new year
|
[0,0,1,1,"https://images.alphacoders.com/119/1198800.jpg"] // new year
|
||||||
];
|
];
|
||||||
|
|
||||||
function handleVisibilityChange() {if (!d.hidden && new Date () - lastUpdate > 3000) requestJson();}
|
function handleVisibilityChange() {if (!d.hidden && new Date () - lastUpdate > 3000) requestJson();}
|
||||||
@ -74,6 +73,7 @@ function setCSL(cs)
|
|||||||
function applyCfg()
|
function applyCfg()
|
||||||
{
|
{
|
||||||
cTheme(cfg.theme.base === "light");
|
cTheme(cfg.theme.base === "light");
|
||||||
|
gId("Colors").style.paddingTop = cfg.comp.colors.picker ? "0" : "28px";
|
||||||
var bg = cfg.theme.color.bg;
|
var bg = cfg.theme.color.bg;
|
||||||
if (bg) sCol('--c-1', bg);
|
if (bg) sCol('--c-1', bg);
|
||||||
var l = cfg.comp.labels;
|
var l = cfg.comp.labels;
|
||||||
@ -109,6 +109,7 @@ function tglLabels()
|
|||||||
function tglRgb()
|
function tglRgb()
|
||||||
{
|
{
|
||||||
cfg.comp.colors.rgb = !cfg.comp.colors.rgb;
|
cfg.comp.colors.rgb = !cfg.comp.colors.rgb;
|
||||||
|
cfg.comp.colors.picker = !cfg.comp.colors.picker;
|
||||||
applyCfg();
|
applyCfg();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -255,7 +256,6 @@ function onLoad()
|
|||||||
});
|
});
|
||||||
} else
|
} else
|
||||||
loadBg(cfg.theme.bg.url);
|
loadBg(cfg.theme.bg.url);
|
||||||
if (cfg.comp.css) loadSkinCSS('skinCss');
|
|
||||||
|
|
||||||
selectSlot(0);
|
selectSlot(0);
|
||||||
updateTablinks(0);
|
updateTablinks(0);
|
||||||
@ -265,14 +265,17 @@ function onLoad()
|
|||||||
loadPalettes(()=>{
|
loadPalettes(()=>{
|
||||||
// fill effect extra data array
|
// fill effect extra data array
|
||||||
loadFXData(()=>{
|
loadFXData(()=>{
|
||||||
// load and populate effects
|
setTimeout(()=>{ // ESP8266 can't handle quick requests
|
||||||
loadFX(()=>{
|
// load and populate effects
|
||||||
setTimeout(()=>{ // ESP8266 can't handle quick requests
|
loadFX(()=>{
|
||||||
loadPalettesData(()=>{
|
setTimeout(()=>{ // ESP8266 can't handle quick requests
|
||||||
requestJson();// will load presets and create WS
|
loadPalettesData(()=>{
|
||||||
});
|
requestJson();// will load presets and create WS
|
||||||
},100);
|
if (cfg.comp.css) setTimeout(()=>{loadSkinCSS('skinCss')},100);
|
||||||
});
|
});
|
||||||
|
},100);
|
||||||
|
});
|
||||||
|
},100);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
resetUtil();
|
resetUtil();
|
||||||
@ -426,18 +429,30 @@ function presetError(empty)
|
|||||||
if (hasBackup) {
|
if (hasBackup) {
|
||||||
cn += `<br><br>`;
|
cn += `<br><br>`;
|
||||||
if (empty)
|
if (empty)
|
||||||
cn += `However, there is backup preset data of a previous installation available.<br>
|
cn += `However, there is backup preset data of a previous installation available.<br>(Saving a preset will hide this and overwrite the backup)`;
|
||||||
(Saving a preset will hide this and overwrite the backup)`;
|
|
||||||
else
|
else
|
||||||
cn += `Here is a backup of the last known good state:`;
|
cn += `Here is a backup of the last known good state:`;
|
||||||
cn += `<textarea id="bck"></textarea><br>
|
cn += `<textarea id="bck"></textarea><br><button class="btn" onclick="cpBck()">Copy to clipboard</button>`;
|
||||||
<button class="btn" onclick="cpBck()">Copy to clipboard</button>`;
|
cn += `<br><button type="button" class="btn" onclick="restore(gId('bck').value)">Restore</button>`;
|
||||||
}
|
}
|
||||||
cn += `</div>`;
|
cn += `</div>`;
|
||||||
gId('pcont').innerHTML = cn;
|
gId('pcont').innerHTML = cn;
|
||||||
if (hasBackup) gId('bck').value = bckstr;
|
if (hasBackup) gId('bck').value = bckstr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function restore(txt) {
|
||||||
|
var req = new XMLHttpRequest();
|
||||||
|
req.addEventListener('load', function(){showToast(this.responseText,this.status >= 400)});
|
||||||
|
req.addEventListener('error', function(e){showToast(e.stack,true);});
|
||||||
|
req.open("POST", getURL("/upload"));
|
||||||
|
var formData = new FormData();
|
||||||
|
var b = new Blob([txt], {type: "application/json"});
|
||||||
|
formData.append("data", b, '/presets.json');
|
||||||
|
req.send(formData);
|
||||||
|
setTimeout(loadPresets, 2000);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
function loadPresets(callback = null)
|
function loadPresets(callback = null)
|
||||||
{
|
{
|
||||||
// 1st boot (because there is a callback)
|
// 1st boot (because there is a callback)
|
||||||
@ -613,7 +628,7 @@ function parseInfo(i) {
|
|||||||
if (loc) name = "(L) " + name;
|
if (loc) name = "(L) " + name;
|
||||||
d.title = name;
|
d.title = name;
|
||||||
ledCount = i.leds.count;
|
ledCount = i.leds.count;
|
||||||
syncTglRecv = i.str;
|
//syncTglRecv = i.str;
|
||||||
maxSeg = i.leds.maxseg;
|
maxSeg = i.leds.maxseg;
|
||||||
pmt = i.fs.pmt;
|
pmt = i.fs.pmt;
|
||||||
gId('buttonNodes').style.display = lastinfo.ndc > 0 ? null:"none";
|
gId('buttonNodes').style.display = lastinfo.ndc > 0 ? null:"none";
|
||||||
@ -672,8 +687,6 @@ function populateInfo(i)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
var vcn = "Kuuhaku";
|
var vcn = "Kuuhaku";
|
||||||
if (i.ver.startsWith("0.14.")) vcn = "Hoshi";
|
|
||||||
// if (i.ver.includes("-bl")) vcn = "Supāku";
|
|
||||||
if (i.cn) vcn = i.cn;
|
if (i.cn) vcn = i.cn;
|
||||||
|
|
||||||
cn += `v${i.ver} "${vcn}"<br><br><table>
|
cn += `v${i.ver} "${vcn}"<br><br><table>
|
||||||
@ -908,7 +921,7 @@ function populatePalettes()
|
|||||||
for (let pa of lJson) {
|
for (let pa of lJson) {
|
||||||
html += generateListItemHtml(
|
html += generateListItemHtml(
|
||||||
'palette',
|
'palette',
|
||||||
pa[0],
|
pa[0],
|
||||||
pa[1],
|
pa[1],
|
||||||
'setPalette',
|
'setPalette',
|
||||||
`<div class="lstIprev" style="${genPalPrevCss(pa[0])}"></div>`
|
`<div class="lstIprev" style="${genPalPrevCss(pa[0])}"></div>`
|
||||||
@ -916,8 +929,9 @@ function populatePalettes()
|
|||||||
}
|
}
|
||||||
gId('pallist').innerHTML=html;
|
gId('pallist').innerHTML=html;
|
||||||
// append custom palettes (when loading for the 1st time)
|
// append custom palettes (when loading for the 1st time)
|
||||||
if (!isEmpty(lastinfo) && lastinfo.cpalcount) {
|
let li = lastinfo;
|
||||||
for (let j = 0; j<lastinfo.cpalcount; j++) {
|
if (!isEmpty(li) && li.cpalcount) {
|
||||||
|
for (let j = 0; j<li.cpalcount; j++) {
|
||||||
let div = d.createElement("div");
|
let div = d.createElement("div");
|
||||||
gId('pallist').appendChild(div);
|
gId('pallist').appendChild(div);
|
||||||
div.outerHTML = generateListItemHtml(
|
div.outerHTML = generateListItemHtml(
|
||||||
@ -929,6 +943,8 @@ function populatePalettes()
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (li.cpalcount>0) gId("rmPal").classList.remove("hide");
|
||||||
|
else gId("rmPal").classList.add("hide");
|
||||||
}
|
}
|
||||||
|
|
||||||
function redrawPalPrev()
|
function redrawPalPrev()
|
||||||
@ -1074,7 +1090,7 @@ function updateTrail(e)
|
|||||||
{
|
{
|
||||||
if (e==null) return;
|
if (e==null) return;
|
||||||
let sd = e.parentNode.getElementsByClassName('sliderdisplay')[0];
|
let sd = e.parentNode.getElementsByClassName('sliderdisplay')[0];
|
||||||
if (sd && getComputedStyle(sd).getPropertyValue("--bg") !== "none") {
|
if (sd && getComputedStyle(sd).getPropertyValue("--bg").trim() !== "none") { // trim() for Safari
|
||||||
var max = e.hasAttribute('max') ? e.attributes.max.value : 255;
|
var max = e.hasAttribute('max') ? e.attributes.max.value : 255;
|
||||||
var perc = Math.round(e.value * 100 / max);
|
var perc = Math.round(e.value * 100 / max);
|
||||||
if (perc < 50) perc += 2;
|
if (perc < 50) perc += 2;
|
||||||
@ -1194,6 +1210,7 @@ function updateUI()
|
|||||||
gId('buttonPower').className = (isOn) ? 'active':'';
|
gId('buttonPower').className = (isOn) ? 'active':'';
|
||||||
gId('buttonNl').className = (nlA) ? 'active':'';
|
gId('buttonNl').className = (nlA) ? 'active':'';
|
||||||
gId('buttonSync').className = (syncSend) ? 'active':'';
|
gId('buttonSync').className = (syncSend) ? 'active':'';
|
||||||
|
gId('pxmb').style.display = (isM) ? "inline-block" : "none";
|
||||||
|
|
||||||
updateSelectedFx();
|
updateSelectedFx();
|
||||||
updateSelectedPalette(selectedPal); // must be after updateSelectedFx() to un-hide color slots for * palettes
|
updateSelectedPalette(selectedPal); // must be after updateSelectedFx() to un-hide color slots for * palettes
|
||||||
@ -1209,7 +1226,7 @@ function updateUI()
|
|||||||
if (hasRGB) {
|
if (hasRGB) {
|
||||||
updateTrail(gId('sliderR'));
|
updateTrail(gId('sliderR'));
|
||||||
updateTrail(gId('sliderG'));
|
updateTrail(gId('sliderG'));
|
||||||
updateTrail(gId('sliderB'));
|
updateTrail(gId('sliderB'));
|
||||||
}
|
}
|
||||||
if (hasWhite) updateTrail(gId('sliderW'));
|
if (hasWhite) updateTrail(gId('sliderW'));
|
||||||
|
|
||||||
@ -1485,41 +1502,41 @@ function setEffectParameters(idx)
|
|||||||
var paOnOff = (effectPars.length<3 || effectPars[2]=='')?[]:effectPars[2].split(",");
|
var paOnOff = (effectPars.length<3 || effectPars[2]=='')?[]:effectPars[2].split(",");
|
||||||
|
|
||||||
// set html slider items on/off
|
// set html slider items on/off
|
||||||
let nSliders = 5;
|
let sliders = d.querySelectorAll("#sliders .sliderwrap");
|
||||||
for (let i=0; i<nSliders; i++) {
|
sliders.forEach((slider, i)=>{
|
||||||
var slider = gId("slider" + i);
|
let text = slider.getAttribute("tooltip");
|
||||||
var label = gId("sliderLabel" + i);
|
if ((!controlDefined && i<((idx<128)?2:nSliders)) || (slOnOff.length>i && slOnOff[i]!="")) {
|
||||||
// if (not controlDefined and for AC speed or intensity and for SR all sliders) or slider has a value
|
if (slOnOff.length>i && slOnOff[i]!="!") text = slOnOff[i];
|
||||||
if ((!controlDefined && i < ((idx<128)?2:nSliders)) || (slOnOff.length>i && slOnOff[i] != "")) {
|
slider.setAttribute("tooltip", text);
|
||||||
if (slOnOff.length>i && slOnOff[i]!="!") label.innerHTML = slOnOff[i];
|
slider.parentElement.classList.remove('hide');
|
||||||
else if (i==0) label.innerHTML = "Effect speed";
|
} else
|
||||||
else if (i==1) label.innerHTML = "Effect intensity";
|
slider.parentElement.classList.add('hide');
|
||||||
else label.innerHTML = "Custom" + (i-1);
|
});
|
||||||
slider.classList.remove('hide');
|
|
||||||
} else {
|
if (slOnOff.length > 5) { // up to 3 checkboxes
|
||||||
slider.classList.add('hide');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (slOnOff.length>5) { // up to 3 checkboxes
|
|
||||||
gId('fxopt').classList.remove('fade');
|
gId('fxopt').classList.remove('fade');
|
||||||
for (let i = 0; i<3; i++) {
|
let checks = d.querySelectorAll("#sliders .ochkl");
|
||||||
|
checks.forEach((check, i)=>{
|
||||||
|
let text = check.getAttribute("tooltip");
|
||||||
if (5+i<slOnOff.length && slOnOff[5+i]!=='') {
|
if (5+i<slOnOff.length && slOnOff[5+i]!=='') {
|
||||||
gId('opt'+i).classList.remove('hide');
|
if (slOnOff.length>5+i && slOnOff[5+i]!="!") text = slOnOff[5+i];
|
||||||
gId('optLabel'+i).innerHTML = slOnOff[5+i]=="!" ? 'Option' : slOnOff[5+i].substr(0,16);
|
check.setAttribute("tooltip", text);
|
||||||
|
check.classList.remove('hide');
|
||||||
} else
|
} else
|
||||||
gId('opt'+i).classList.add('hide');
|
check.classList.add('hide');
|
||||||
}
|
});
|
||||||
} else {
|
} else gId('fxopt').classList.add('fade');
|
||||||
gId('fxopt').classList.add('fade');
|
|
||||||
}
|
|
||||||
|
|
||||||
// set the bottom position of selected effect (sticky) as the top of sliders div
|
// set the bottom position of selected effect (sticky) as the top of sliders div
|
||||||
setInterval(()=>{
|
function setSelectedEffectPosition() {
|
||||||
let top = parseInt(getComputedStyle(gId("sliders")).height);
|
let top = parseInt(getComputedStyle(gId("sliders")).height);
|
||||||
top += 5;
|
top += 5;
|
||||||
let sel = d.querySelector('#fxlist .selected');
|
let sel = d.querySelector('#fxlist .selected');
|
||||||
if (sel) sel.style.bottom = top + "px"; // we will need to remove this when unselected (in setFX())
|
if (sel) sel.style.bottom = top + "px"; // we will need to remove this when unselected (in setFX())
|
||||||
},750);
|
}
|
||||||
|
|
||||||
|
setSelectedEffectPosition();
|
||||||
|
setInterval(setSelectedEffectPosition,750);
|
||||||
// set html color items on/off
|
// set html color items on/off
|
||||||
var cslLabel = '';
|
var cslLabel = '';
|
||||||
var sep = '';
|
var sep = '';
|
||||||
@ -1556,10 +1573,14 @@ function setEffectParameters(idx)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
gId("cslLabel").innerHTML = cslLabel;
|
gId("cslLabel").innerHTML = cslLabel;
|
||||||
|
if (cslLabel!=="") gId("cslLabel").classList.remove("hide");
|
||||||
|
else gId("cslLabel").classList.add("hide");
|
||||||
|
|
||||||
// set palette on/off
|
// set palette on/off
|
||||||
var palw = gId("palw"); // wrapper
|
var palw = gId("palw"); // wrapper
|
||||||
var pall = gId("pall"); // label
|
var pall = gId("pall"); // label
|
||||||
|
var icon = '<i class="icons sel-icon" onclick="tglHex()"></i> ';
|
||||||
|
var text = 'Color palette';
|
||||||
// if not controlDefined or palette has a value
|
// if not controlDefined or palette has a value
|
||||||
if (hasRGB && ((!controlDefined) || (paOnOff.length>0 && paOnOff[0]!="" && isNaN(paOnOff[0])))) {
|
if (hasRGB && ((!controlDefined) || (paOnOff.length>0 && paOnOff[0]!="" && isNaN(paOnOff[0])))) {
|
||||||
palw.style.display = "inline-block";
|
palw.style.display = "inline-block";
|
||||||
@ -1569,13 +1590,13 @@ function setEffectParameters(idx)
|
|||||||
var v = Math.max(0,Math.min(255,parseInt(paOnOff[0].substr(dPos+1))));
|
var v = Math.max(0,Math.min(255,parseInt(paOnOff[0].substr(dPos+1))));
|
||||||
paOnOff[0] = paOnOff[0].substring(0,dPos);
|
paOnOff[0] = paOnOff[0].substring(0,dPos);
|
||||||
}
|
}
|
||||||
if (paOnOff.length>0 && paOnOff[0] != "!") pall.innerHTML = paOnOff[0];
|
if (paOnOff.length>0 && paOnOff[0] != "!") text = paOnOff[0];
|
||||||
else pall.innerHTML = '<i class="icons sel-icon" onclick="tglHex()"></i> Color palette';
|
|
||||||
} else {
|
} else {
|
||||||
// disable palette list
|
// disable palette list
|
||||||
pall.innerHTML = '<i class="icons sel-icon" onclick="tglHex()"></i> Color palette not used';
|
text += ' not used';
|
||||||
palw.style.display = "none";
|
palw.style.display = "none";
|
||||||
}
|
}
|
||||||
|
pall.innerHTML = icon + text;
|
||||||
// not all color selectors shown, hide palettes created from color selectors
|
// not all color selectors shown, hide palettes created from color selectors
|
||||||
// NOTE: this will disallow user to select "* Color ..." palettes which may be undesirable in some cases or for some users
|
// NOTE: this will disallow user to select "* Color ..." palettes which may be undesirable in some cases or for some users
|
||||||
//for (let e of (gId('pallist').querySelectorAll('.lstI')||[])) {
|
//for (let e of (gId('pallist').querySelectorAll('.lstI')||[])) {
|
||||||
@ -1666,6 +1687,8 @@ function togglePower()
|
|||||||
obj.seg = [];
|
obj.seg = [];
|
||||||
obj.seg[0] = {"id": lastinfo.liveseg, "frz": false};
|
obj.seg[0] = {"id": lastinfo.liveseg, "frz": false};
|
||||||
}
|
}
|
||||||
|
if (cfg.comp.on >0 && isOn) obj = {"ps": cfg.comp.on }; // don't use setPreset()
|
||||||
|
if (cfg.comp.off>0 && !isOn) obj = {"ps": cfg.comp.off}; // don't use setPreset()
|
||||||
requestJson(obj);
|
requestJson(obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1688,7 +1711,7 @@ function toggleSync()
|
|||||||
if (syncSend) showToast('Other lights in the network will now sync to this one.');
|
if (syncSend) showToast('Other lights in the network will now sync to this one.');
|
||||||
else showToast('This light and other lights in the network will no longer sync.');
|
else showToast('This light and other lights in the network will no longer sync.');
|
||||||
var obj = {"udpn": {"send": syncSend}};
|
var obj = {"udpn": {"send": syncSend}};
|
||||||
if (syncTglRecv) obj.udpn.recv = syncSend;
|
//if (syncTglRecv) obj.udpn.recv = syncSend;
|
||||||
requestJson(obj);
|
requestJson(obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1700,7 +1723,7 @@ function toggleLiveview()
|
|||||||
let wsOn = ws && ws.readyState === WebSocket.OPEN;
|
let wsOn = ws && ws.readyState === WebSocket.OPEN;
|
||||||
|
|
||||||
var lvID = "liveview";
|
var lvID = "liveview";
|
||||||
if (isM && wsOn) {
|
if (isM && wsOn) {
|
||||||
lvID += "2D";
|
lvID += "2D";
|
||||||
if (isLv) gId('klv2D').innerHTML = `<iframe id="${lvID}" src="about:blank"></iframe>`;
|
if (isLv) gId('klv2D').innerHTML = `<iframe id="${lvID}" src="about:blank"></iframe>`;
|
||||||
gId('mlv2D').style.transform = (isLv) ? "translateY(0px)":"translateY(100%)";
|
gId('mlv2D').style.transform = (isLv) ? "translateY(0px)":"translateY(100%)";
|
||||||
@ -1887,7 +1910,7 @@ function makeP(i,pl)
|
|||||||
end: 0
|
end: 0
|
||||||
};
|
};
|
||||||
var rep = plJson[i].repeat ? plJson[i].repeat : 0;
|
var rep = plJson[i].repeat ? plJson[i].repeat : 0;
|
||||||
content =
|
content =
|
||||||
`<div id="ple${i}" style="margin-top:10px;"></div><label class="check revchkl">Shuffle
|
`<div id="ple${i}" style="margin-top:10px;"></div><label class="check revchkl">Shuffle
|
||||||
<input type="checkbox" id="pl${i}rtgl" onchange="plR(${i})" ${plJson[i].r||rep<0?"checked":""}>
|
<input type="checkbox" id="pl${i}rtgl" onchange="plR(${i})" ${plJson[i].r||rep<0?"checked":""}>
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
@ -1909,23 +1932,17 @@ ${makePlSel(plJson[i].end?plJson[i].end:0, true)}
|
|||||||
} else {
|
} else {
|
||||||
content =
|
content =
|
||||||
`<label class="check revchkl">
|
`<label class="check revchkl">
|
||||||
<span class="lstIname">
|
<span class="lstIname">Include brightness</span>
|
||||||
Include brightness
|
|
||||||
</span>
|
|
||||||
<input type="checkbox" id="p${i}ibtgl" checked>
|
<input type="checkbox" id="p${i}ibtgl" checked>
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>
|
</label>
|
||||||
<label class="check revchkl">
|
<label class="check revchkl">
|
||||||
<span class="lstIname">
|
<span class="lstIname">Save segment bounds</span>
|
||||||
Save segment bounds
|
|
||||||
</span>
|
|
||||||
<input type="checkbox" id="p${i}sbtgl" checked>
|
<input type="checkbox" id="p${i}sbtgl" checked>
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>
|
</label>
|
||||||
<label class="check revchkl">
|
<label class="check revchkl">
|
||||||
<span class="lstIname">
|
<span class="lstIname">Checked segments only</span>
|
||||||
Checked segments only
|
|
||||||
</span>
|
|
||||||
<input type="checkbox" id="p${i}sbchk">
|
<input type="checkbox" id="p${i}sbchk">
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>`;
|
</label>`;
|
||||||
@ -1941,9 +1958,7 @@ ${makePlSel(plJson[i].end?plJson[i].end:0, true)}
|
|||||||
<div class="h">(leave empty for no Quick load button)</div>
|
<div class="h">(leave empty for no Quick load button)</div>
|
||||||
<div ${pl&&i==0?"style='display:none'":""}>
|
<div ${pl&&i==0?"style='display:none'":""}>
|
||||||
<label class="check revchkl">
|
<label class="check revchkl">
|
||||||
<span class="lstIname">
|
<span class="lstIname">${pl?"Show playlist editor":(i>0)?"Overwrite with state":"Use current state"}</span>
|
||||||
${pl?"Show playlist editor":(i>0)?"Overwrite with state":"Use current state"}
|
|
||||||
</span>
|
|
||||||
<input type="checkbox" id="p${i}cstgl" onchange="tglCs(${i})" ${(i==0||pl)?"checked":""}>
|
<input type="checkbox" id="p${i}cstgl" onchange="tglCs(${i})" ${(i==0||pl)?"checked":""}>
|
||||||
<span class="checkmark"></span>
|
<span class="checkmark"></span>
|
||||||
</label>
|
</label>
|
||||||
@ -2915,6 +2930,41 @@ function mergeDeep(target, ...sources)
|
|||||||
return mergeDeep(target, ...sources);
|
return mergeDeep(target, ...sources);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function tooltip()
|
||||||
|
{
|
||||||
|
const elements = d.querySelectorAll("[tooltip]");
|
||||||
|
|
||||||
|
elements.forEach((element)=>{
|
||||||
|
element.addEventListener("mouseover", ()=>{
|
||||||
|
const tooltip = d.createElement("span");
|
||||||
|
|
||||||
|
tooltip.className = "tooltip";
|
||||||
|
tooltip.textContent = element.getAttribute("tooltip");
|
||||||
|
|
||||||
|
let { top, left, width } = element.getBoundingClientRect();
|
||||||
|
|
||||||
|
d.body.appendChild(tooltip);
|
||||||
|
|
||||||
|
const { offsetHeight, offsetWidth } = tooltip;
|
||||||
|
|
||||||
|
const offset = element.classList.contains("sliderwrap") ? 6 : 12;
|
||||||
|
top -= offsetHeight + offset;
|
||||||
|
left += (width - offsetWidth) / 2;
|
||||||
|
|
||||||
|
tooltip.style.top = top + "px";
|
||||||
|
tooltip.style.left = left + "px";
|
||||||
|
|
||||||
|
tooltip.classList.add("visible");
|
||||||
|
});
|
||||||
|
|
||||||
|
element.addEventListener("mouseout", ()=>{
|
||||||
|
const tooltip = d.querySelector('.tooltip');
|
||||||
|
tooltip.classList.remove("visible");
|
||||||
|
d.body.removeChild(tooltip);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
size();
|
size();
|
||||||
_C.style.setProperty('--n', N);
|
_C.style.setProperty('--n', N);
|
||||||
|
|
||||||
@ -2926,3 +2976,5 @@ _C.addEventListener('touchstart', lock, false);
|
|||||||
_C.addEventListener('mouseout', move, false);
|
_C.addEventListener('mouseout', move, false);
|
||||||
_C.addEventListener('mouseup', move, false);
|
_C.addEventListener('mouseup', move, false);
|
||||||
_C.addEventListener('touchend', move, false);
|
_C.addEventListener('touchend', move, false);
|
||||||
|
|
||||||
|
d.addEventListener('DOMContentLoaded', tooltip);
|
@ -7,13 +7,10 @@
|
|||||||
|
|
||||||
<title>Pixel Magic Tool</title>
|
<title>Pixel Magic Tool</title>
|
||||||
|
|
||||||
<!-- <link
|
|
||||||
rel="shortcut icon"
|
|
||||||
href="
|
|
||||||
" /> -->
|
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
:root {
|
:root {
|
||||||
|
--s-thumb: #0006;
|
||||||
|
--s-background: #0003;
|
||||||
--overlay: rgba(0, 0, 0, 0.5);
|
--overlay: rgba(0, 0, 0, 0.5);
|
||||||
--background: #111;
|
--background: #111;
|
||||||
--text: #bbb;
|
--text: #bbb;
|
||||||
@ -34,6 +31,24 @@
|
|||||||
--warning-light: #f48c06;
|
--warning-light: #f48c06;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
::-webkit-scrollbar {
|
||||||
|
width: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
::-webkit-scrollbar-track {
|
||||||
|
background: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
::-webkit-scrollbar-thumb {
|
||||||
|
background: var(--s-thumb);
|
||||||
|
opacity: 0.2;
|
||||||
|
border-radius: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
::-webkit-scrollbar-thumb:hover {
|
||||||
|
background: var(--s-background);
|
||||||
|
}
|
||||||
|
|
||||||
::selection {
|
::selection {
|
||||||
background: var(--blue-light);
|
background: var(--blue-light);
|
||||||
}
|
}
|
||||||
@ -65,7 +80,7 @@
|
|||||||
display: block;
|
display: block;
|
||||||
font-weight: 400;
|
font-weight: 400;
|
||||||
margin: 2px 0 5px;
|
margin: 2px 0 5px;
|
||||||
color: var(--text);
|
color: var(--gray-light);
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -83,10 +98,19 @@
|
|||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
}
|
}
|
||||||
|
|
||||||
:is(a:hover, a:focus, a:active) {
|
a:is(:hover, :focus, :active) {
|
||||||
color: var(--blue-medium);
|
color: var(--blue-medium);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#wledEdit {
|
||||||
|
padding: 4px 8px;
|
||||||
|
background: var(--blue-light);
|
||||||
|
margin-left: 6px;
|
||||||
|
display: inline-block;
|
||||||
|
border-radius: 4px;
|
||||||
|
color: var(--gray-light);
|
||||||
|
}
|
||||||
|
|
||||||
.m-zero {
|
.m-zero {
|
||||||
margin: 0 !important;
|
margin: 0 !important;
|
||||||
}
|
}
|
||||||
@ -108,8 +132,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.content {
|
.content {
|
||||||
width: calc(100% - 40px);
|
width: min(768px, calc(100% - 40px));
|
||||||
max-width: 768px;
|
|
||||||
margin: 20px;
|
margin: 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -117,26 +140,43 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: nowrap;
|
flex-wrap: nowrap;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
margin: 20px 0 0;
|
margin-top: 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.column {
|
.column {
|
||||||
flex-basis: calc(50% - 10px);
|
flex-basis: calc(50% - 10px);
|
||||||
position: relative;
|
position: relative;
|
||||||
padding: 0 5px;
|
padding-inline: 5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.column-full {
|
.column-full {
|
||||||
flex-basis: 100%;
|
flex-basis: 100%;
|
||||||
position: relative;
|
position: relative;
|
||||||
padding: 0 5px;
|
padding-inline: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
padding-bottom: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header .brand {
|
||||||
|
width: 100%;
|
||||||
|
max-width: 200px;
|
||||||
|
height: 100%;
|
||||||
|
display: block;
|
||||||
|
outline: none;
|
||||||
|
border: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
label {
|
label {
|
||||||
display: block;
|
display: flex;
|
||||||
margin-bottom: 5px;
|
margin-bottom: 5px;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
|
align-items: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
input[type="text"],
|
input[type="text"],
|
||||||
@ -157,7 +197,7 @@
|
|||||||
width: 32px;
|
width: 32px;
|
||||||
height: 32px;
|
height: 32px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
padding: 0px 1px;
|
padding-inline: 1px;
|
||||||
outline: none;
|
outline: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -172,18 +212,19 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.input-group .input-description {
|
.input-group .input-description {
|
||||||
width: 38px;
|
width: 100%;
|
||||||
|
max-width: 38px;
|
||||||
height: 38px;
|
height: 38px;
|
||||||
padding: 10px 0;
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
color: var(--gray-dark);
|
color: var(--gray-dark);
|
||||||
background: var(--gray-light);
|
background: var(--gray-light);
|
||||||
border-radius: 0px 5px 5px 0;
|
border-radius: 0px 8px 8px 0;
|
||||||
border: 1px solid var(--gray-light);
|
border: 1px solid var(--gray-light);
|
||||||
border-left: 0;
|
border-left: 0;
|
||||||
text-align: center;
|
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
line-height: 16px;
|
line-height: 16px;
|
||||||
font-weight: 600;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.input-group .square {
|
.input-group .square {
|
||||||
@ -191,10 +232,19 @@
|
|||||||
margin-left: 10px;
|
margin-left: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.input-group .square input {
|
||||||
|
text-align: center;
|
||||||
|
background: none;
|
||||||
|
padding: 0;
|
||||||
|
border: 0;
|
||||||
|
color: var(--gray-dark);
|
||||||
|
}
|
||||||
|
|
||||||
textarea {
|
textarea {
|
||||||
resize: vertical;
|
resize: none;
|
||||||
min-height: 200px;
|
min-height: 200px;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
|
overflow-x: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
.custom-select {
|
.custom-select {
|
||||||
@ -231,7 +281,7 @@
|
|||||||
text-align: center;
|
text-align: center;
|
||||||
padding: 40px 10px;
|
padding: 40px 10px;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
margin: 20px 0 0;
|
margin-top: 20px;
|
||||||
transition: all 0.5s ease-in-out;
|
transition: all 0.5s ease-in-out;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -253,14 +303,15 @@
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
outline: none;
|
outline: none;
|
||||||
margin: 16px 0;
|
margin-block: 15px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.range-slider::-webkit-slider-thumb {
|
.range-slider::-webkit-slider-thumb,
|
||||||
|
.range-slider::-moz-range-thumb {
|
||||||
appearance: none;
|
appearance: none;
|
||||||
height: 16px;
|
height: 16px;
|
||||||
width: 16px;
|
width: 16px;
|
||||||
background-color: var(--gray-dark);
|
background-color: var(--blue-light);
|
||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
border: 0;
|
border: 0;
|
||||||
@ -326,7 +377,7 @@
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
width: auto;
|
width: auto;
|
||||||
padding: 6px 12px;
|
padding: 6px 12px;
|
||||||
margin: 10px 0 0;
|
margin-top: 10px;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
transform: translateY(30px);
|
transform: translateY(30px);
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
@ -334,7 +385,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.toast .toast-body {
|
.toast .toast-body {
|
||||||
padding: 8px 0;
|
padding-block: 8px;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
letter-spacing: 0.5px;
|
letter-spacing: 0.5px;
|
||||||
@ -360,7 +411,7 @@
|
|||||||
height: 3px;
|
height: 3px;
|
||||||
transform: scaleX(0);
|
transform: scaleX(0);
|
||||||
transform-origin: left;
|
transform-origin: left;
|
||||||
border-radius: inherit;
|
border-radius: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.toast.success .toast-progress {
|
.toast.success .toast-progress {
|
||||||
@ -387,22 +438,6 @@
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
.header {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
align-items: center;
|
|
||||||
padding: 0 0 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.header .brand {
|
|
||||||
width: 100%;
|
|
||||||
max-width: 200px;
|
|
||||||
height: 100%;
|
|
||||||
display: block;
|
|
||||||
outline: none;
|
|
||||||
border: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.carousel {
|
.carousel {
|
||||||
display: flex;
|
display: flex;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
@ -410,18 +445,6 @@
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
.carousel img {
|
|
||||||
display: block;
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
|
||||||
margin-right: 20px;
|
|
||||||
border: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.carousel img:last-child {
|
|
||||||
margin-right: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.button {
|
.button {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
border: 0;
|
border: 0;
|
||||||
@ -429,7 +452,7 @@
|
|||||||
border-radius: 50px;
|
border-radius: 50px;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
margin: 0 0 10px;
|
margin-bottom: 10px;
|
||||||
background: var(--gray-medium);
|
background: var(--gray-medium);
|
||||||
border: 1px solid var(--gray-dark);
|
border: 1px solid var(--gray-dark);
|
||||||
transition: all 0.5s ease-in-out;
|
transition: all 0.5s ease-in-out;
|
||||||
@ -477,7 +500,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
#recreatedImage {
|
#recreatedImage {
|
||||||
margin: 20px 0;
|
margin-block: 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.invalid {
|
.invalid {
|
||||||
@ -487,16 +510,12 @@
|
|||||||
.error-message {
|
.error-message {
|
||||||
display: block;
|
display: block;
|
||||||
color: var(--error-dark);
|
color: var(--error-dark);
|
||||||
padding: 4px 0;
|
padding-block: 4px;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (max-width: 767px) {
|
@media (max-width: 767px) {
|
||||||
.header {
|
|
||||||
padding-bottom: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.row {
|
.row {
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
@ -506,7 +525,7 @@
|
|||||||
.column,
|
.column,
|
||||||
.column-full {
|
.column-full {
|
||||||
flex-basis: 100%;
|
flex-basis: 100%;
|
||||||
margin: 20px 0 0;
|
margin-top: 20px;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -542,68 +561,8 @@
|
|||||||
<div class="content">
|
<div class="content">
|
||||||
<form id="formGenerate" novalidate>
|
<form id="formGenerate" novalidate>
|
||||||
<div class="header">
|
<div class="header">
|
||||||
<svg
|
<img alt="Pixel Magic Tool" width="200" height="130" src="" />
|
||||||
class="brand"
|
<!--img width="200" height="130" src="" alt="Pixel Magic Tool" class="brand" /-->
|
||||||
version="1.1"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
xmlns:xlink="http://www.w3.org/1999/xlink"
|
|
||||||
x="0px"
|
|
||||||
y="0px"
|
|
||||||
width="200px"
|
|
||||||
height="130px"
|
|
||||||
viewBox="0 0 200 130"
|
|
||||||
enable-background="new 0 0 200 130"
|
|
||||||
xml:space="preserve">
|
|
||||||
<image
|
|
||||||
width="200"
|
|
||||||
height="130"
|
|
||||||
x="0"
|
|
||||||
y="0"
|
|
||||||
href="
|
|
||||||
AAB6JgAAgIQAAPoAAACA6AAAdTAAAOpgAAA6mAAAF3CculE8AAAAAmJLR0QA/4ePzL8AAAAHdElN
|
|
||||||
RQfnBg4TByTFKGw2AAAIWElEQVR42u2d+5GjOBCHf3t1AVBKYAiBDBZHMg7hHMEwETiEsyMxkwEh
|
|
||||||
cAl0OQPfH+IhgQQIBDRefVu1ZWMhHj20+iUBBN4LSulFL3pRuveZvAd/7X0CAZ0gEGb8WrIzvbpb
|
|
||||||
xKL+AuEJYcffWxyk/yRVnES+9w0AAMrwBfB4vpcJ5AQgwRUAcEGx98W8A4sEInKA6i8Fj7/2o+Ms
|
|
||||||
EEoQAXiKYlLr1+DPD0JXUVCKBwDgW2RrXnhzHGa4PyFXpABynKrvRfWp2PtS3oPFg7p4Ivd1MpQC
|
|
||||||
SKovH5QCKMRz+5uyJw52BaVIAXwiBlDiDiDXxw06y9/ErdmSAYC0YSpy/OA30nZDq7IMCm41O4xi
|
|
||||||
nAF84Gw6k0NAWRW1av9lnRYPetGLerq5u4/ek6XdJhGyJhLXOZP92MQPGUbe9GCjSTwJhM74RK3/
|
|
||||||
E3oAuLeKq8cNOYDPSl3IJ+oXdL/mhjv+QFPBQSAiQwbQQ1pZ4qT9GCujQoQUwM9ATyXK+snQtueK
|
|
||||||
X/Pfn/nMMFBZLlSjViluzWfJTZQj+4y244G7QArl/5YSOYDKaUQBYO6F136NeX9pr+W4oWu72Y83
|
|
||||||
tR0LnAUiLsatN9wadVZ01Jlb/x79miPCVGVVXgJwE2Xl/wzzSSkUhdQ1yDvtAAAiA0P8CuSOH8xX
|
|
||||||
Vipxo5xKpJrSaRC/lIjUuWkt+bL0e1Y+Z16v3RNeBTJg6AYmwkplKT5+NNhwwNOhaxMN83deigJd
|
|
||||||
+x6wEoh5rBCq/yPpezptzCtR2nW8pToz6IyqQFcm5NSZoTwh1kDeYAicIpuKsHvaVaAlrr5a2+ln
|
|
||||||
AUB/hr5o3t87UOA5tSk90CrQKz2BJUb9OKrKsmXQToO3LLHuZw9mx1r4vblAzW7qIP0fTxHZi0NY
|
|
||||||
RjlT/6NTn6CymLHPoF7O9sblfqnWVwkMqSDtqZLKaqC18Yi1WnZQdXPZRSAy0DJrzxPQucV3kTl0
|
|
||||||
4KKsmiM2atR5b3eCymIGLz9kFMOgLm2todz7t/K5nHXYsuqj2duSFfLAwQQyB5Et7qHcLu51GIFU
|
|
||||||
BXpuRJQCKH0GPChGjNozWaH/wwikKtCT5UdAp4DHgvSRvuf9fVvU0lkJvyzq38xxBCIpRXXxlE4Q
|
|
||||||
yAE5gEA0JWEjIcAW5PnoBYVG6iHXVktDHEAgmpJokZZPrbhk6dAJOS6IoIeBzr1n6TTimA6pJVnC
|
|
||||||
JIP8BS7wk5BrOIJAjEjLp6+4ZFU+OffncNwSoCcA4OnfUTyaQGLKoFUPvxtbCkTWIpZOrQvoSkKm
|
|
||||||
ivJe6EXO3yqa79IyevT6kxSAcf+2bXtEs1q6VAVP3tlSIG61iE1rTUnY6Mzfkt/I2N/4/uNqadp0
|
|
||||||
pTlMEci1uRlFXZWlZK6jtU7tz0RJIim1TL8tlVCNk6RluLUWbaZbZAg4ozwhIqs/UTahNM3MT9tL
|
|
||||||
YA4h/M6MIBBmBIEwg4VjqKWdZiV9ZpfAadSTPq1z2CdPQZ00C944Ez88IcwIAmFGEAgzgkCYsfGg
|
|
||||||
PmEhgEhpM5IW0mILO1Els7okE3ZVEmetsbC1lTVue6i1wmPZaptlZaowtBVJ5EMTuCdwnm3fqYmz
|
|
||||||
JoTFwuz1jqHC0Bp9YxbsCWMIM4JAmBEEwgy3MSTlsYTR/ijTU2O/PW89qJvjVCxXPxwktWzvrs2q
|
|
||||||
23YTqi03Fog5OLde0c7ss5y7tlx3bVbt25RqyzCGMCMIhBnmadEfs/szhgOY9HYIpkyLdsEYDmDS
|
|
||||||
m0refCr7P1rn3m+waO2UQb3Kk3WycvVWDytEKwvFesn8TTjicFbSNvd+rEjbDWPG8D1jWYnBbov2
|
|
||||||
PqlpvKdArnufwHyClcWMrRNU2UiD3RJNXNhaZfkdstuh+dPiA8vJBFwotTnzuanJoceQ1jexpoZX
|
|
||||||
mOO04HwnzHcPYwgzgkCYcWiVNYHIrMw4KTKdDQTix/tWUmMub6ey+dxsX93i/HYEj5gsoOTITp0P
|
|
||||||
9lRZBguIW6pqe8KgzowgEGa8u5VlQQnh5F4trvbtCzNXm+AmEPXtIS5LluXGrb+t1SFfo/tOPZ5e
|
|
||||||
bH1WWtzm3ABmAhFFG5+yVuOa9stNt3XB9O6x47Vn6TmlFsYQZgSBMCMIhBl7jiEJmcIahZiRwTAX
|
|
||||||
YExCeceC13fhmq/OQjsm7SmQaK0ht+FmtKC2Cc/MvDpmVpZf5LpXXXiHZ8IYwowgEGYcWmVxmBbt
|
|
||||||
m0MLZJuy022xCcQlKzeyn1ASW9YpcaOGqleT1COTrs7hOsIYwowgEGYEgTAjCIQZ6qA+WndqZEK9
|
|
||||||
quUYei997guXhfGDWhtTzLw6B5jaLtOYZ7PZ53wZ11xc4cVfQwSVxYwgEGYc21O3kQxGdJO9T2+I
|
|
||||||
9xTIavkO5a0QKvf5L5ihBFcuCarFeJlObQoSqZZjqf2SGNNOS6zBTiLr0AJZC/NMJ4vQc/zMm71u
|
|
||||||
LnMKAlmK5zUbg0Cm02boZVZe6v1ydn8XRH2PKAhkMnWGnjK5OM/SmmBRmPL77yKQfMM9V02LHTp0
|
|
||||||
sg91wKafalKNYj3gQmd8Nl8UI7nf17s8ITxIrLVYsfLLoJEcnhBnzE+IySiWLYzBzHpxq15fIZbF
|
|
||||||
jKCyfJE3n7qLwap+/6hBEFSWMxaVFSPGUxSaijrhKQq5YKCuonCRC8raDYTAZOgl/3W2ZvSS9e4U
|
|
||||||
UUr/Vq0eAKVta0rpH3X/fl9BZXmAYpzb2knxRN4u6FG7kdVv+VipdxCID+KBscHRjQxWFjOCQJgR
|
|
||||||
BOKDEt/IAaT06q0q+S3nq9dD/hhBIB4QpciagMgHpZS2A7nIcK8+RpRSOpbRD4O6b9Tl0U/a2+LU
|
|
||||||
tbsutqK7IJAVGciYFHzXtDsc9KKXeeFAelSOXu9XylpHcbivMIYEAkP8DyhqAZcAHYDzAAAAJXRF
|
|
||||||
WHRkYXRlOmNyZWF0ZQAyMDIzLTA2LTE0VDE5OjA3OjM2KzAwOjAw7raFWwAAACV0RVh0ZGF0ZTpt
|
|
||||||
b2RpZnkAMjAyMy0wNi0xNFQxOTowNzozNiswMDowMJ/rPecAAAAodEVYdGRhdGU6dGltZXN0YW1w
|
|
||||||
ADIwMjMtMDYtMTRUMTk6MDc6MzYrMDA6MDDI/hw4AAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFn
|
|
||||||
ZVJlYWR5ccllPAAAAABJRU5ErkJggg==" />
|
|
||||||
</svg>
|
|
||||||
</div>
|
</div>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="column" validate>
|
<div class="column" validate>
|
||||||
@ -626,11 +585,11 @@
|
|||||||
<label for="pattern">Pattern</label>
|
<label for="pattern">Pattern</label>
|
||||||
<select name="pattern" id="pattern" required>
|
<select name="pattern" id="pattern" required>
|
||||||
<option value="">Select a choice</option>
|
<option value="">Select a choice</option>
|
||||||
<option value="1" title="['ffffff']" selected>
|
<option value="1" title="['ffffff']">Individual</option>
|
||||||
Individual
|
|
||||||
</option>
|
|
||||||
<option value="2" title="[0, 'ffffff']">Index</option>
|
<option value="2" title="[0, 'ffffff']">Index</option>
|
||||||
<option value="3" title="[0, 5, 'ffffff']">Range</option>
|
<option value="3" title="[0, 5, 'ffffff']" selected>
|
||||||
|
Range
|
||||||
|
</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -686,11 +645,13 @@
|
|||||||
max="255"
|
max="255"
|
||||||
value="128"
|
value="128"
|
||||||
class="range-slider" />
|
class="range-slider" />
|
||||||
<input
|
<div class="input-description square">
|
||||||
type="text"
|
<input
|
||||||
id="brightnessValue"
|
type="text"
|
||||||
class="input-description square"
|
name="brightnessValue"
|
||||||
value="128" />
|
id="brightnessValue"
|
||||||
|
value="128" />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -698,7 +659,11 @@
|
|||||||
<div class="column" validate>
|
<div class="column" validate>
|
||||||
<label for="animation">Animation</label>
|
<label for="animation">Animation</label>
|
||||||
<label class="switch">
|
<label class="switch">
|
||||||
<input type="checkbox" name="animation" id="animation" />
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
name="animation"
|
||||||
|
id="animation"
|
||||||
|
data-parent="animation" />
|
||||||
<span class="switch-slider"></span>
|
<span class="switch-slider"></span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
@ -708,19 +673,25 @@
|
|||||||
<input
|
<input
|
||||||
type="checkbox"
|
type="checkbox"
|
||||||
name="transparentImage"
|
name="transparentImage"
|
||||||
id="transparentImage" />
|
id="transparentImage"
|
||||||
|
data-parent="transparentImage" />
|
||||||
<span class="switch-slider"></span>
|
<span class="switch-slider"></span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
<div class="column" validate>
|
<div class="column" validate>
|
||||||
<label for="resizeImage">Resize Image</label>
|
<label for="resizeImage">Resize Image</label>
|
||||||
<label class="switch">
|
<label class="switch">
|
||||||
<input type="checkbox" name="resizeImage" id="resizeImage" />
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
name="resizeImage"
|
||||||
|
id="resizeImage"
|
||||||
|
data-parent="resizeImage"
|
||||||
|
checked />
|
||||||
<span class="switch-slider"></span>
|
<span class="switch-slider"></span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="row resizeImage" style="display: none">
|
<div class="row resizeImage">
|
||||||
<div class="column" validate>
|
<div class="column" validate>
|
||||||
<label for="width">Width</label>
|
<label for="width">Width</label>
|
||||||
<input type="number" name="width" id="width" value="16" />
|
<input type="number" name="width" id="width" value="16" />
|
||||||
@ -747,7 +718,9 @@
|
|||||||
min="0"
|
min="0"
|
||||||
step="0.1"
|
step="0.1"
|
||||||
inputmode="numeric" />
|
inputmode="numeric" />
|
||||||
<div class="input-description">sec</div>
|
<div class="input-description">
|
||||||
|
<span>sec</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="column" validate>
|
<div class="column" validate>
|
||||||
@ -762,7 +735,9 @@
|
|||||||
min="0"
|
min="0"
|
||||||
step="0.1"
|
step="0.1"
|
||||||
inputmode="numeric" />
|
inputmode="numeric" />
|
||||||
<div class="input-description">sec</div>
|
<div class="input-description">
|
||||||
|
<span>sec</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -773,30 +748,27 @@
|
|||||||
Color that will replace the
|
Color that will replace the
|
||||||
<strong>transparent pixels</strong> in the image
|
<strong>transparent pixels</strong> in the image
|
||||||
</small>
|
</small>
|
||||||
<input type="color" name="color" id="color" value="#eeeeee" />
|
<input type="color" name="color" id="color" value="#00BFFF" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="column-full" validate>
|
<div class="column-full" validate>
|
||||||
<div class="custom-select">
|
<div class="custom-select">
|
||||||
<label for="images">Images</label>
|
<label for="images">
|
||||||
<select name="images" id="images" required>
|
<span>Images upload to WLED</span>
|
||||||
<option value="">Select a choice</option>
|
<a id="wledEdit" href="http://[wled-ip]/edit" target="_blank">
|
||||||
<option value="upload">Upload</option>
|
upload
|
||||||
|
</a>
|
||||||
|
</label>
|
||||||
|
<select name="images" id="images">
|
||||||
|
<option value="">Select image</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<small>
|
|
||||||
Images uploaded to
|
|
||||||
<a id="wledEdit" href="http://[wled-ip]/edit" target="_blank">
|
|
||||||
<strong>WLED</strong>
|
|
||||||
</a>
|
|
||||||
or upload image
|
|
||||||
</small>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div id="dropzone" class="dropzone" validate style="display: none">
|
<div id="dropzone" class="dropzone" validate>
|
||||||
<p id="dropzoneLabel">
|
<p id="dropzoneLabel">
|
||||||
Drag and drop a file here or click to select a file
|
Drag and drop a file here or click to select a local file
|
||||||
</p>
|
</p>
|
||||||
<input
|
<input
|
||||||
type="file"
|
type="file"
|
||||||
@ -868,7 +840,7 @@
|
|||||||
const hostname = element("hostname");
|
const hostname = element("hostname");
|
||||||
hostname.value = host;
|
hostname.value = host;
|
||||||
|
|
||||||
hostname.addEventListener("change", async () => {
|
hostname.addEventListener("blur", async () => {
|
||||||
WLED_URL = `${protocol}//${hostname.value}`;
|
WLED_URL = `${protocol}//${hostname.value}`;
|
||||||
|
|
||||||
await segments();
|
await segments();
|
||||||
@ -893,7 +865,7 @@
|
|||||||
|
|
||||||
function hostnameLabel() {
|
function hostnameLabel() {
|
||||||
const link = element("wledEdit");
|
const link = element("wledEdit");
|
||||||
link.href = link.href.replace("[wled-ip]", hostname.value);
|
link.href = WLED_URL + "/edit";
|
||||||
}
|
}
|
||||||
|
|
||||||
async function playlist() {
|
async function playlist() {
|
||||||
@ -999,6 +971,7 @@
|
|||||||
|
|
||||||
if (success) {
|
if (success) {
|
||||||
toast(`Preset "${item.n}" save successfully`);
|
toast(`Preset "${item.n}" save successfully`);
|
||||||
|
window.parent.postMessage("loadPresets", WLED_URL);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
toast(`Error saving preset: ${error}`, "error");
|
toast(`Error saving preset: ${error}`, "error");
|
||||||
@ -1047,12 +1020,9 @@
|
|||||||
return mimeTypes.includes(mimetype);
|
return mimeTypes.includes(mimetype);
|
||||||
});
|
});
|
||||||
|
|
||||||
const options = [
|
const options = [{ text: "Select image", value: "" }];
|
||||||
{ text: "Select a choice", value: "" },
|
|
||||||
{ text: "Upload", value: "upload" },
|
|
||||||
];
|
|
||||||
|
|
||||||
if (images) {
|
if (images.length > 0) {
|
||||||
options.push(
|
options.push(
|
||||||
...images.map(({ name }) => ({
|
...images.map(({ name }) => ({
|
||||||
text: name,
|
text: name,
|
||||||
@ -1064,6 +1034,11 @@
|
|||||||
|
|
||||||
options.forEach(({ text, value }) => {
|
options.forEach(({ text, value }) => {
|
||||||
const option = new Option(text, value);
|
const option = new Option(text, value);
|
||||||
|
|
||||||
|
if (index === 0) {
|
||||||
|
option.selected = true;
|
||||||
|
}
|
||||||
|
|
||||||
select.appendChild(option);
|
select.appendChild(option);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -1076,7 +1051,6 @@
|
|||||||
|
|
||||||
async function segments() {
|
async function segments() {
|
||||||
const select = element("segments");
|
const select = element("segments");
|
||||||
const pattern = element("pattern");
|
|
||||||
const width = element("width");
|
const width = element("width");
|
||||||
const height = element("height");
|
const height = element("height");
|
||||||
|
|
||||||
@ -1114,7 +1088,6 @@
|
|||||||
option.selected = true;
|
option.selected = true;
|
||||||
width.value = w;
|
width.value = w;
|
||||||
height.value = h;
|
height.value = h;
|
||||||
pattern.value = w * h > 512 ? 3 : 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
select.add(option);
|
select.add(option);
|
||||||
@ -1278,12 +1251,12 @@
|
|||||||
const dropzone = element("dropzone");
|
const dropzone = element("dropzone");
|
||||||
const { value } = e.target.selectedOptions[0];
|
const { value } = e.target.selectedOptions[0];
|
||||||
|
|
||||||
if (value === "upload") {
|
if (!value) {
|
||||||
const dropzoneLabel = element("dropzoneLabel");
|
const dropzoneLabel = element("dropzoneLabel");
|
||||||
const source = element("source");
|
const source = element("source");
|
||||||
|
|
||||||
dropzoneLabel.textContent =
|
dropzoneLabel.textContent =
|
||||||
"Drag and drop a file here or click to select a file";
|
"Drag and drop a file here or click to select a local file";
|
||||||
source.value = "";
|
source.value = "";
|
||||||
|
|
||||||
dropzone.style.display = "block";
|
dropzone.style.display = "block";
|
||||||
@ -1293,62 +1266,49 @@
|
|||||||
});
|
});
|
||||||
|
|
||||||
element("transparentImage").addEventListener("change", (e) => {
|
element("transparentImage").addEventListener("change", (e) => {
|
||||||
const transparentImage = d.getElementsByClassName("transparentImage");
|
const transparentImage = d.getElementsByClassName("transparentImage")[0];
|
||||||
const { checked } = e.target;
|
const { checked } = e.target;
|
||||||
|
|
||||||
Array.from(transparentImage).forEach(function (element) {
|
if (checked) {
|
||||||
if (checked) {
|
transparentImage.style.display = "flex";
|
||||||
element.style.display = "flex";
|
} else {
|
||||||
} else {
|
transparentImage.style.display = "none";
|
||||||
element.style.display = "none";
|
}
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
element("resizeImage").addEventListener("change", (e) => {
|
element("resizeImage").addEventListener("change", (e) => {
|
||||||
const resizeImage = d.getElementsByClassName("resizeImage");
|
const resizeImage = d.getElementsByClassName("resizeImage")[0];
|
||||||
const pattern = element("pattern");
|
|
||||||
const { checked } = e.target;
|
const { checked } = e.target;
|
||||||
|
|
||||||
Array.from(resizeImage).forEach(function (element) {
|
if (checked) {
|
||||||
if (checked) {
|
resizeImage.style.display = "flex";
|
||||||
pattern.value = 3;
|
} else {
|
||||||
element.style.display = "flex";
|
resizeImage.style.display = "none";
|
||||||
} else {
|
}
|
||||||
pattern.value = 1;
|
|
||||||
element.style.display = "none";
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
element("animation").addEventListener("change", (e) => {
|
element("animation").addEventListener("change", (e) => {
|
||||||
const animation = d.getElementsByClassName("animation");
|
const animation = d.getElementsByClassName("animation")[0];
|
||||||
|
|
||||||
const pattern = element("pattern");
|
|
||||||
const source = element("source");
|
const source = element("source");
|
||||||
|
|
||||||
const { checked } = e.target;
|
const { checked } = e.target;
|
||||||
|
|
||||||
Array.from(animation).forEach(function (element) {
|
if (checked) {
|
||||||
if (checked) {
|
toast(
|
||||||
toast(
|
'If you want all frames in the image, set it to "0"',
|
||||||
'If you want all frames in the image, set it to "0"',
|
"warning",
|
||||||
"warning",
|
5000
|
||||||
5000
|
);
|
||||||
);
|
|
||||||
|
|
||||||
source.setAttribute("accept", "image/gif");
|
source.setAttribute("accept", "image/gif");
|
||||||
element.style.display = "flex";
|
animation.style.display = "flex";
|
||||||
pattern.value = 3;
|
} else {
|
||||||
} else {
|
source.setAttribute(
|
||||||
source.setAttribute(
|
"accept",
|
||||||
"accept",
|
"image/jpg,image/jpeg,image/png,image/gif"
|
||||||
"image/jpg,image/jpeg,image/png,image/gif"
|
);
|
||||||
);
|
animation.style.display = "none";
|
||||||
element.style.display = "none";
|
}
|
||||||
pattern.value = 1;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
element("btnGenerate").addEventListener("click", async (event) => {
|
element("btnGenerate").addEventListener("click", async (event) => {
|
||||||
@ -1402,8 +1362,7 @@
|
|||||||
const response = element("response");
|
const response = element("response");
|
||||||
const recreatedImage = element("recreatedImage");
|
const recreatedImage = element("recreatedImage");
|
||||||
|
|
||||||
const urlImage =
|
const urlImage = !images ? URL.createObjectURL(file) : images;
|
||||||
images === "upload" ? URL.createObjectURL(file) : images;
|
|
||||||
|
|
||||||
const image = await loadImage(urlImage);
|
const image = await loadImage(urlImage);
|
||||||
const { canvas, bri, id, i } = recreate(image);
|
const { canvas, bri, id, i } = recreate(image);
|
||||||
|
@ -773,6 +773,7 @@ Length: <input type="number" name="XC${i}" id="xc${i}" class="l" min="1" max="65
|
|||||||
Brightness factor: <input name="BF" type="number" class="m" min="1" max="255" required> %
|
Brightness factor: <input name="BF" type="number" class="m" min="1" max="255" required> %
|
||||||
<h3>Transitions</h3>
|
<h3>Transitions</h3>
|
||||||
Crossfade: <input type="checkbox" name="TF"><br>
|
Crossfade: <input type="checkbox" name="TF"><br>
|
||||||
|
Effect blending: <input type="checkbox" name="EB"><br>
|
||||||
Transition Time: <input name="TD" type="number" class="xl" min="0" max="65500"> ms<br>
|
Transition Time: <input name="TD" type="number" class="xl" min="0" max="65500"> ms<br>
|
||||||
Enable Palette transitions: <input type="checkbox" name="PF"><br>
|
Enable Palette transitions: <input type="checkbox" name="PF"><br>
|
||||||
<i>Random Cycle</i> Palette Time: <input name="TP" type="number" class="m" min="1" max="255"> s<br>
|
<i>Random Cycle</i> Palette Time: <input name="TP" type="number" class="m" min="1" max="255"> s<br>
|
||||||
|
@ -91,6 +91,12 @@
|
|||||||
<h3>WLED Broadcast</h3>
|
<h3>WLED Broadcast</h3>
|
||||||
UDP Port: <input name="UP" type="number" min="1" max="65535" class="d5" required><br>
|
UDP Port: <input name="UP" type="number" min="1" max="65535" class="d5" required><br>
|
||||||
2nd Port: <input name="U2" type="number" min="1" max="65535" class="d5" required><br>
|
2nd Port: <input name="U2" type="number" min="1" max="65535" class="d5" required><br>
|
||||||
|
<div id="NoESPNOW" class="hide">
|
||||||
|
<i class="warn">ESP-NOW support is disabled.<br></i>
|
||||||
|
</div>
|
||||||
|
<div id="ESPNOW">
|
||||||
|
Use ESP-NOW sync: <input type="checkbox" name="EN"><br><i>(in AP mode or no WiFi)</i><br>
|
||||||
|
</div>
|
||||||
<h3>Sync groups</h3>
|
<h3>Sync groups</h3>
|
||||||
<input name="GS" id="GS" type="number" style="display: none;"><!-- hidden inputs for bitwise group checkboxes -->
|
<input name="GS" id="GS" type="number" style="display: none;"><!-- hidden inputs for bitwise group checkboxes -->
|
||||||
<input name="GR" id="GR" type="number" style="display: none;">
|
<input name="GR" id="GR" type="number" style="display: none;">
|
||||||
@ -128,14 +134,17 @@ UDP Port: <input name="UP" type="number" min="1" max="65535" class="d5" required
|
|||||||
<td><input type="checkbox" id="R7" name="R7"></td>
|
<td><input type="checkbox" id="R7" name="R7"></td>
|
||||||
<td><input type="checkbox" id="R8" name="R8"></td>
|
<td><input type="checkbox" id="R8" name="R8"></td>
|
||||||
</tr>
|
</tr>
|
||||||
</table><br>
|
</table>
|
||||||
Receive: <nowrap><input type="checkbox" name="RB">Brightness,</nowrap> <nowrap><input type="checkbox" name="RC">Color,</nowrap> <nowrap>and <input type="checkbox" name="RX">Effects</nowrap><br>
|
<h3>Receive</h3>
|
||||||
<input type="checkbox" name="SO"> Segment options, <input type="checkbox" name="SG"> bounds<br>
|
<nowrap><input type="checkbox" name="RB">Brightness,</nowrap> <nowrap><input type="checkbox" name="RC">Color,</nowrap> <nowrap>and <input type="checkbox" name="RX">Effects</nowrap><br>
|
||||||
|
<input type="checkbox" name="SO"> Segment options, <input type="checkbox" name="SG"> bounds
|
||||||
|
<h3>Send</h3>
|
||||||
|
Enable Sync on start: <input type="checkbox" name="SS"><br>
|
||||||
Send notifications on direct change: <input type="checkbox" name="SD"><br>
|
Send notifications on direct change: <input type="checkbox" name="SD"><br>
|
||||||
Send notifications on button press or IR: <input type="checkbox" name="SB"><br>
|
Send notifications on button press or IR: <input type="checkbox" name="SB"><br>
|
||||||
Send Alexa notifications: <input type="checkbox" name="SA"><br>
|
Send Alexa notifications: <input type="checkbox" name="SA"><br>
|
||||||
Send Philips Hue change notifications: <input type="checkbox" name="SH"><br>
|
Send Philips Hue change notifications: <input type="checkbox" name="SH"><br>
|
||||||
Send Macro notifications: <input type="checkbox" name="SM"><br>
|
<!-- Send Macro notifications: <input type="checkbox" name="SM"><br> -->
|
||||||
UDP packet retransmissions: <input name="UR" type="number" min="0" max="30" class="d5" required><br><br>
|
UDP packet retransmissions: <input name="UR" type="number" min="0" max="30" class="d5" required><br><br>
|
||||||
<i>Reboot required to apply changes. </i>
|
<i>Reboot required to apply changes. </i>
|
||||||
<hr class="sml">
|
<hr class="sml">
|
||||||
|
@ -26,7 +26,9 @@
|
|||||||
"segexp" : "Always expand first segment",
|
"segexp" : "Always expand first segment",
|
||||||
"css": "Enable custom CSS",
|
"css": "Enable custom CSS",
|
||||||
"hdays": "Enable custom Holidays list",
|
"hdays": "Enable custom Holidays list",
|
||||||
"fxdef": "Use effect default parameters"
|
"fxdef": "Use effect default parameters",
|
||||||
|
"on": "Power button preset override for On",
|
||||||
|
"off": "Power button preset override for Off"
|
||||||
},
|
},
|
||||||
"theme":{
|
"theme":{
|
||||||
"alpha": {
|
"alpha": {
|
||||||
@ -163,7 +165,7 @@
|
|||||||
|
|
||||||
function Save() {
|
function Save() {
|
||||||
SetLS();
|
SetLS();
|
||||||
if (d.Sf.DS.value != initial_ds || d.Sf.ST.checked != initial_st || d.Sf.SU.checked != initial_su) d.Sf.submit();
|
if (d.Sf.DS.value != initial_ds || /*d.Sf.ST.checked != initial_st ||*/ d.Sf.SU.checked != initial_su) d.Sf.submit();
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://www.educative.io/edpresso/how-to-dynamically-load-a-js-file-in-javascript
|
// https://www.educative.io/edpresso/how-to-dynamically-load-a-js-file-in-javascript
|
||||||
@ -178,7 +180,7 @@
|
|||||||
//console.log("File loaded");
|
//console.log("File loaded");
|
||||||
GetV();
|
GetV();
|
||||||
initial_ds = d.Sf.DS.value;
|
initial_ds = d.Sf.DS.value;
|
||||||
initial_st = d.Sf.ST.checked;
|
//initial_st = d.Sf.ST.checked;
|
||||||
initial_su = d.Sf.SU.checked;
|
initial_su = d.Sf.SU.checked;
|
||||||
GetLS();
|
GetLS();
|
||||||
});
|
});
|
||||||
@ -261,7 +263,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<h2>Web Setup</h2>
|
<h2>Web Setup</h2>
|
||||||
Server description: <input type="text" name="DS" maxlength="32"><br>
|
Server description: <input type="text" name="DS" maxlength="32"><br>
|
||||||
Sync button toggles both send and receive: <input type="checkbox" name="ST"><br>
|
<!-- Sync button toggles both send and receive: <input type="checkbox" name="ST"><br> -->
|
||||||
<div id="NoSimple" class="hide">
|
<div id="NoSimple" class="hide">
|
||||||
<i class="warn">This firmware build does not include simplified UI support.<br></i>
|
<i class="warn">This firmware build does not include simplified UI support.<br></i>
|
||||||
</div>
|
</div>
|
||||||
|
@ -11,6 +11,7 @@
|
|||||||
|
|
||||||
function gId(e) { return d.getElementById(e); }
|
function gId(e) { return d.getElementById(e); }
|
||||||
function cE(e) { return d.createElement(e); }
|
function cE(e) { return d.createElement(e); }
|
||||||
|
function toggle(el){gId(el).classList.toggle("hide"); gId('No'+el).classList.toggle("hide");}
|
||||||
function H(){window.open("https://kno.wled.ge/features/settings/#wifi-settings");}
|
function H(){window.open("https://kno.wled.ge/features/settings/#wifi-settings");}
|
||||||
function B(){window.open(getURL("/settings"),"_self");}
|
function B(){window.open(getURL("/settings"),"_self");}
|
||||||
function N() {
|
function N() {
|
||||||
@ -193,14 +194,16 @@
|
|||||||
<i>Can help with connectivity issues.<br>
|
<i>Can help with connectivity issues.<br>
|
||||||
Do not enable if WiFi is working correctly, increases power consumption.</i>
|
Do not enable if WiFi is working correctly, increases power consumption.</i>
|
||||||
|
|
||||||
<div id="remd">
|
<h3>ESP-NOW Wireless</h3>
|
||||||
<h3>Wireless Remote</h3>
|
<div id="NoESPNOW" class="hide">
|
||||||
|
<i class="warn">This firmware build does not include ESP-NOW support.<br></i>
|
||||||
|
</div>
|
||||||
|
<div id="ESPNOW">
|
||||||
|
Enable ESP-NOW: <input type="checkbox" name="RE"><br>
|
||||||
<i>Listen for events over ESP-NOW<br>
|
<i>Listen for events over ESP-NOW<br>
|
||||||
Keep disabled if not using a remote, increases power consumption.<br></i>
|
Keep disabled if not using a remote or wireless sync, increases power consumption.<br></i>
|
||||||
|
Paired Remote MAC: <input type="text" name="RMAC" minlength="12" maxlength="12"><br>
|
||||||
Enable Remote: <input type="checkbox" name="RE"><br>
|
Last device seen: <span class="rlid" onclick="d.Sf.RMAC.value=this.textContent;" style="cursor:pointer;">None</span> <br>
|
||||||
Hardware MAC: <input type="text" name="RMAC"><br>
|
|
||||||
Last Seen: <span class="rlid">None</span> <br>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="ethd">
|
<div id="ethd">
|
||||||
|
@ -19,8 +19,8 @@
|
|||||||
--c-r: #e42;
|
--c-r: #e42;
|
||||||
--c-g: #4e2;
|
--c-g: #4e2;
|
||||||
--c-l: #48a;
|
--c-l: #48a;
|
||||||
--t-b: 0.5;
|
--t-b: .5;
|
||||||
--c-o: rgba(34, 34, 34, 0.9);
|
--c-o: rgba(34, 34, 34, .9);
|
||||||
--c-tb : rgba(34, 34, 34, var(--t-b));
|
--c-tb : rgba(34, 34, 34, var(--t-b));
|
||||||
--c-tba: rgba(102, 102, 102, var(--t-b));
|
--c-tba: rgba(102, 102, 102, var(--t-b));
|
||||||
--c-tbh: rgba(51, 51, 51, var(--t-b));
|
--c-tbh: rgba(51, 51, 51, var(--t-b));
|
||||||
@ -31,7 +31,7 @@
|
|||||||
--tbp: 14px 8px 10px;
|
--tbp: 14px 8px 10px;
|
||||||
--bbp: 9px 0 7px 0;
|
--bbp: 9px 0 7px 0;
|
||||||
--bhd: none;
|
--bhd: none;
|
||||||
--bmt: 0px;
|
--bmt: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
html {
|
html {
|
||||||
@ -86,11 +86,11 @@ a, a:visited {
|
|||||||
}
|
}
|
||||||
|
|
||||||
button {
|
button {
|
||||||
outline: none;
|
outline: 0;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
background-color: transparent;
|
background-color: transparent;
|
||||||
border: none;
|
border: 0;
|
||||||
transition: color 0.3s, background-color 0.3s;
|
transition: color .3s, background-color .3s;
|
||||||
font-size: 19px;
|
font-size: 19px;
|
||||||
color: var(--c-c);
|
color: var(--c-c);
|
||||||
min-width: 40px;
|
min-width: 40px;
|
||||||
@ -180,8 +180,8 @@ button:hover {
|
|||||||
.tab button {
|
.tab button {
|
||||||
background-color: transparent;
|
background-color: transparent;
|
||||||
float: left;
|
float: left;
|
||||||
border: none;
|
border: 0;
|
||||||
transition: color 0.3s, background-color 0.3s;
|
transition: color .3s, background-color .3s;
|
||||||
font-size: 17px;
|
font-size: 17px;
|
||||||
color: var(--c-c);
|
color: var(--c-c);
|
||||||
min-width: 44px;
|
min-width: 44px;
|
||||||
@ -218,7 +218,7 @@ button:hover {
|
|||||||
position: relative;
|
position: relative;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
border: 0px;
|
border: 0;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
overscroll-behavior: none;
|
overscroll-behavior: none;
|
||||||
@ -243,8 +243,8 @@ button:hover {
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
z-index: 11;
|
z-index: 11;
|
||||||
opacity: 0.95;
|
opacity: .95;
|
||||||
transition: 0.7s;
|
transition: .7s;
|
||||||
pointer-events: none;
|
pointer-events: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -267,24 +267,24 @@ button:hover {
|
|||||||
|
|
||||||
#toast.show {
|
#toast.show {
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
animation: fadein 0.5s, fadein 0.5s 2.5s reverse;
|
animation: fadein .5s, fadein .5s 2.5s reverse;
|
||||||
}
|
}
|
||||||
|
|
||||||
#toast.error {
|
#toast.error {
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
background-color: #b21;
|
background-color: #b21;
|
||||||
animation: fadein 0.5s;
|
animation: fadein .5s;
|
||||||
}
|
}
|
||||||
|
|
||||||
.modal {
|
.modal {
|
||||||
position:fixed;
|
position: fixed;
|
||||||
left: 0px;
|
left: 0;
|
||||||
bottom: 0px;
|
bottom: 0;
|
||||||
right: 0px;
|
right: 0;
|
||||||
top: calc(var(--th) - 1px);
|
top: calc(var(--th) - 1px);
|
||||||
background-color: var(--c-o);
|
background-color: var(--c-o);
|
||||||
transform: translateY(100%);
|
transform: translateY(100%);
|
||||||
transition: transform 0.4s;
|
transition: transform .4s;
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
font-size: 20px;
|
font-size: 20px;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
@ -356,7 +356,7 @@ button:hover {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#heart {
|
#heart {
|
||||||
transition: color 0.9s;
|
transition: color .9s;
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
color: #f00;
|
color: #f00;
|
||||||
}
|
}
|
||||||
@ -421,7 +421,7 @@ img {
|
|||||||
padding: 4px 4px 2px;
|
padding: 4px 4px 2px;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
right: 3px;
|
right: 3px;
|
||||||
transition: visibility 0.25s ease, opacity 0.25s ease;
|
transition: visibility .25s ease, opacity .25s ease;
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
visibility: hidden;
|
visibility: hidden;
|
||||||
}
|
}
|
||||||
@ -531,9 +531,9 @@ input[type=range]::-moz-range-thumb {
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
border: 1px solid var(--c-3);
|
border: 1px solid var(--c-3);
|
||||||
border-radius: 25px;
|
border-radius: 25px;
|
||||||
transition-duration: 0.3s;
|
transition-duration: .3s;
|
||||||
-webkit-backface-visibility: hidden;
|
-webkit-backface-visibility: hidden;
|
||||||
-webkit-transform:translate3d(0,0,0);
|
-webkit-transform: translate3d(0,0,0);
|
||||||
overflow: clip;
|
overflow: clip;
|
||||||
text-overflow: clip;
|
text-overflow: clip;
|
||||||
min-height: 40px;
|
min-height: 40px;
|
||||||
@ -586,13 +586,13 @@ option {
|
|||||||
input[type=number], input[type=text] {
|
input[type=number], input[type=text] {
|
||||||
background: var(--c-3);
|
background: var(--c-3);
|
||||||
color: var(--c-f);
|
color: var(--c-f);
|
||||||
border: 0px solid var(--c-f);
|
border: 0 solid var(--c-f);
|
||||||
border-radius: 5px;
|
border-radius: 5px;
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
margin: 6px 6px 6px 0;
|
margin: 6px 6px 6px 0;
|
||||||
font-size: 19px;
|
font-size: 19px;
|
||||||
transition: background-color 0.2s;
|
transition: background-color .2s;
|
||||||
outline: none;
|
outline: 0;
|
||||||
width: 50px;
|
width: 50px;
|
||||||
-webkit-appearance: textfield;
|
-webkit-appearance: textfield;
|
||||||
-moz-appearance: textfield;
|
-moz-appearance: textfield;
|
||||||
@ -614,9 +614,9 @@ input[type=number]::-webkit-outer-spin-button {
|
|||||||
|
|
||||||
.pid {
|
.pid {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 0px;
|
top: 0;
|
||||||
left: 0px;
|
left: 0;
|
||||||
padding: 12px 0px 0px 12px;
|
padding: 12px 0 0 12px;
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
width: 20px;
|
width: 20px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
@ -746,7 +746,7 @@ input[type=number]::-webkit-outer-spin-button {
|
|||||||
.list {
|
.list {
|
||||||
position: relative;
|
position: relative;
|
||||||
width: 280px;
|
width: 280px;
|
||||||
transition: background-color 0.5s;
|
transition: background-color .5s;
|
||||||
margin: auto auto 20px;
|
margin: auto auto 20px;
|
||||||
font-size: 19px;
|
font-size: 19px;
|
||||||
line-height: 24px;
|
line-height: 24px;
|
||||||
@ -774,7 +774,7 @@ input[type=number]::-webkit-outer-spin-button {
|
|||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
.lstI:last-child {
|
.lstI:last-child {
|
||||||
border: none;
|
border: 0;
|
||||||
border-radius: 0 0 20px 20px;
|
border-radius: 0 0 20px 20px;
|
||||||
padding-bottom: 10px;
|
padding-bottom: 10px;
|
||||||
}
|
}
|
||||||
@ -879,7 +879,7 @@ input[type=text].fnd:not(:placeholder-shown), input[type=text].fnd:hover {
|
|||||||
}
|
}
|
||||||
::-webkit-scrollbar-thumb {
|
::-webkit-scrollbar-thumb {
|
||||||
background: var(--c-sb);
|
background: var(--c-sb);
|
||||||
opacity: 0.2;
|
opacity: .2;
|
||||||
border-radius: 5px;
|
border-radius: 5px;
|
||||||
}
|
}
|
||||||
::-webkit-scrollbar-thumb:hover {
|
::-webkit-scrollbar-thumb:hover {
|
||||||
@ -927,7 +927,7 @@ input[type=text].fnd:not(:placeholder-shown), input[type=text].fnd:hover {
|
|||||||
display: none !important;
|
display: none !important;
|
||||||
}
|
}
|
||||||
#briwrap {
|
#briwrap {
|
||||||
margin-top: 0px !important;
|
margin-top: 0 !important;
|
||||||
float: none;
|
float: none;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -489,7 +489,7 @@ function parseInfo() {
|
|||||||
d.title = name;
|
d.title = name;
|
||||||
isRgbw = li.leds.wv;
|
isRgbw = li.leds.wv;
|
||||||
ledCount = li.leds.count;
|
ledCount = li.leds.count;
|
||||||
syncTglRecv = li.str;
|
// syncTglRecv = li.str;
|
||||||
maxSeg = li.leds.maxseg;
|
maxSeg = li.leds.maxseg;
|
||||||
pmt = li.fs.pmt;
|
pmt = li.fs.pmt;
|
||||||
cct = li.leds.cct;
|
cct = li.leds.cct;
|
||||||
|
@ -63,7 +63,8 @@ class NeoGammaWLEDMethod {
|
|||||||
#define gamma32(c) NeoGammaWLEDMethod::Correct32(c)
|
#define gamma32(c) NeoGammaWLEDMethod::Correct32(c)
|
||||||
#define gamma8(c) NeoGammaWLEDMethod::rawGamma8(c)
|
#define gamma8(c) NeoGammaWLEDMethod::rawGamma8(c)
|
||||||
uint32_t color_blend(uint32_t,uint32_t,uint16_t,bool b16=false);
|
uint32_t color_blend(uint32_t,uint32_t,uint16_t,bool b16=false);
|
||||||
uint32_t color_add(uint32_t,uint32_t);
|
uint32_t color_add(uint32_t,uint32_t, bool fast=false);
|
||||||
|
uint32_t color_fade(uint32_t c1, uint8_t amount, bool video=false);
|
||||||
inline uint32_t colorFromRgbw(byte* rgbw) { return uint32_t((byte(rgbw[3]) << 24) | (byte(rgbw[0]) << 16) | (byte(rgbw[1]) << 8) | (byte(rgbw[2]))); }
|
inline uint32_t colorFromRgbw(byte* rgbw) { return uint32_t((byte(rgbw[3]) << 24) | (byte(rgbw[0]) << 16) | (byte(rgbw[1]) << 8) | (byte(rgbw[2]))); }
|
||||||
void colorHStoRGB(uint16_t hue, byte sat, byte* rgb); //hue, sat to rgb
|
void colorHStoRGB(uint16_t hue, byte sat, byte* rgb); //hue, sat to rgb
|
||||||
void colorKtoRGB(uint16_t kelvin, byte* rgb);
|
void colorKtoRGB(uint16_t kelvin, byte* rgb);
|
||||||
@ -218,7 +219,7 @@ void deletePreset(byte index);
|
|||||||
bool getPresetName(byte index, String& name);
|
bool getPresetName(byte index, String& name);
|
||||||
|
|
||||||
//remote.cpp
|
//remote.cpp
|
||||||
void handleRemote();
|
void handleRemote(uint8_t *data, size_t len);
|
||||||
|
|
||||||
//set.cpp
|
//set.cpp
|
||||||
bool isAsterisksOnly(const char* str, byte maxLen);
|
bool isAsterisksOnly(const char* str, byte maxLen);
|
||||||
@ -234,6 +235,9 @@ void handleNotifications();
|
|||||||
void setRealtimePixel(uint16_t i, byte r, byte g, byte b, byte w);
|
void setRealtimePixel(uint16_t i, byte r, byte g, byte b, byte w);
|
||||||
void refreshNodeList();
|
void refreshNodeList();
|
||||||
void sendSysInfoUDP();
|
void sendSysInfoUDP();
|
||||||
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
|
void espNowReceiveCB(uint8_t* address, uint8_t* data, uint8_t len, signed int rssi, bool broadcast);
|
||||||
|
#endif
|
||||||
|
|
||||||
//network.cpp
|
//network.cpp
|
||||||
int getSignalQuality(int rssi);
|
int getSignalQuality(int rssi);
|
||||||
@ -353,6 +357,7 @@ void checkSettingsPIN(const char *pin);
|
|||||||
uint16_t crc16(const unsigned char* data_p, size_t length);
|
uint16_t crc16(const unsigned char* data_p, size_t length);
|
||||||
um_data_t* simulateSound(uint8_t simulationId);
|
um_data_t* simulateSound(uint8_t simulationId);
|
||||||
void enumerateLedmaps();
|
void enumerateLedmaps();
|
||||||
|
uint8_t get_random_wheel_index(uint8_t pos);
|
||||||
|
|
||||||
#ifdef WLED_ADD_EEPROM_SUPPORT
|
#ifdef WLED_ADD_EEPROM_SUPPORT
|
||||||
//wled_eeprom.cpp
|
//wled_eeprom.cpp
|
||||||
|
@ -9,7 +9,7 @@
|
|||||||
// Autogenerated from wled00/data/cpal/cpal.htm, do not edit!!
|
// Autogenerated from wled00/data/cpal/cpal.htm, do not edit!!
|
||||||
const uint16_t PAGE_cpal_L = 4721;
|
const uint16_t PAGE_cpal_L = 4721;
|
||||||
const uint8_t PAGE_cpal[] PROGMEM = {
|
const uint8_t PAGE_cpal[] PROGMEM = {
|
||||||
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x0a, 0xbd, 0x3b, 0x7f, 0x73, 0xdb, 0xb6,
|
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x13, 0xbd, 0x3b, 0x7f, 0x73, 0xdb, 0xb6,
|
||||||
0x92, 0xff, 0xe7, 0x53, 0x20, 0x4c, 0x5f, 0x42, 0xd6, 0x14, 0x45, 0xd2, 0xb6, 0x64, 0x4b, 0xa2,
|
0x92, 0xff, 0xe7, 0x53, 0x20, 0x4c, 0x5f, 0x42, 0xd6, 0x14, 0x45, 0xd2, 0xb6, 0x64, 0x4b, 0xa2,
|
||||||
0x3b, 0xa9, 0x93, 0x77, 0xce, 0x8d, 0xdd, 0x64, 0x5e, 0x7c, 0x6e, 0x7b, 0x3e, 0xbf, 0x31, 0x4d,
|
0x3b, 0xa9, 0x93, 0x77, 0xce, 0x8d, 0xdd, 0x64, 0x5e, 0x7c, 0x6e, 0x7b, 0x3e, 0xbf, 0x31, 0x4d,
|
||||||
0x42, 0x12, 0x1b, 0x8a, 0xe0, 0x03, 0x21, 0xd9, 0xae, 0xac, 0xef, 0x7e, 0xbb, 0x00, 0x48, 0x91,
|
0x42, 0x12, 0x1b, 0x8a, 0xe0, 0x03, 0x21, 0xd9, 0xae, 0xac, 0xef, 0x7e, 0xbb, 0x00, 0x48, 0x91,
|
||||||
|
1574
wled00/html_other.h
1574
wled00/html_other.h
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
2261
wled00/html_simple.h
2261
wled00/html_simple.h
File diff suppressed because it is too large
Load Diff
@ -210,7 +210,7 @@ void sendImprovInfoResponse() {
|
|||||||
//Use serverDescription if it has been changed from the default "WLED", else mDNS name
|
//Use serverDescription if it has been changed from the default "WLED", else mDNS name
|
||||||
bool useMdnsName = (strcmp(serverDescription, "WLED") == 0 && strlen(cmDNS) > 0);
|
bool useMdnsName = (strcmp(serverDescription, "WLED") == 0 && strlen(cmDNS) > 0);
|
||||||
char vString[20];
|
char vString[20];
|
||||||
sprintf_P(vString, PSTR("0.14.0/%i"), VERSION);
|
sprintf_P(vString, PSTR("0.15.0-a0/%i"), VERSION);
|
||||||
const char *str[4] = {"WLED", vString, bString, useMdnsName ? cmDNS : serverDescription};
|
const char *str[4] = {"WLED", vString, bString, useMdnsName ? cmDNS : serverDescription};
|
||||||
|
|
||||||
sendImprovRPCResult(ImprovRPCType::Request_Info, 4, str);
|
sendImprovRPCResult(ImprovRPCType::Request_Info, 4, str);
|
||||||
|
@ -355,9 +355,8 @@ bool deserializeState(JsonObject root, byte callMode, byte presetId)
|
|||||||
nightlightTargetBri = nl[F("tbri")] | nightlightTargetBri;
|
nightlightTargetBri = nl[F("tbri")] | nightlightTargetBri;
|
||||||
|
|
||||||
JsonObject udpn = root["udpn"];
|
JsonObject udpn = root["udpn"];
|
||||||
notifyDirect = udpn["send"] | notifyDirect;
|
sendNotificationsRT = udpn["send"] | sendNotificationsRT;
|
||||||
syncGroups = udpn["sgrp"] | syncGroups;
|
syncGroups = udpn["sgrp"] | syncGroups;
|
||||||
receiveNotifications = udpn["recv"] | receiveNotifications;
|
|
||||||
receiveGroups = udpn["rgrp"] | receiveGroups;
|
receiveGroups = udpn["rgrp"] | receiveGroups;
|
||||||
if ((bool)udpn[F("nn")]) callMode = CALL_MODE_NO_NOTIFY; //send no notification just for this request
|
if ((bool)udpn[F("nn")]) callMode = CALL_MODE_NO_NOTIFY; //send no notification just for this request
|
||||||
|
|
||||||
@ -573,8 +572,8 @@ void serializeState(JsonObject root, bool forPreset, bool includeBri, bool segme
|
|||||||
}
|
}
|
||||||
|
|
||||||
JsonObject udpn = root.createNestedObject("udpn");
|
JsonObject udpn = root.createNestedObject("udpn");
|
||||||
udpn["send"] = notifyDirect;
|
udpn["send"] = sendNotificationsRT;
|
||||||
udpn["recv"] = receiveNotifications;
|
udpn["recv"] = receiveGroups != 0;
|
||||||
udpn["sgrp"] = syncGroups;
|
udpn["sgrp"] = syncGroups;
|
||||||
udpn["rgrp"] = receiveGroups;
|
udpn["rgrp"] = receiveGroups;
|
||||||
|
|
||||||
@ -609,7 +608,7 @@ void serializeInfo(JsonObject root)
|
|||||||
{
|
{
|
||||||
root[F("ver")] = versionString;
|
root[F("ver")] = versionString;
|
||||||
root[F("vid")] = VERSION;
|
root[F("vid")] = VERSION;
|
||||||
//root[F("cn")] = WLED_CODENAME;
|
root[F("cn")] = F(WLED_CODENAME);
|
||||||
|
|
||||||
JsonObject leds = root.createNestedObject("leds");
|
JsonObject leds = root.createNestedObject("leds");
|
||||||
leds[F("count")] = strip.getLengthTotal();
|
leds[F("count")] = strip.getLengthTotal();
|
||||||
@ -654,7 +653,7 @@ void serializeInfo(JsonObject root)
|
|||||||
spi.add(spi_miso);
|
spi.add(spi_miso);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
root[F("str")] = syncToggleReceive;
|
root[F("str")] = false; //syncToggleReceive;
|
||||||
|
|
||||||
root[F("name")] = serverDescription;
|
root[F("name")] = serverDescription;
|
||||||
root[F("udpport")] = udpPort;
|
root[F("udpport")] = udpPort;
|
||||||
|
@ -165,6 +165,8 @@ void updateInterfaces(uint8_t callMode)
|
|||||||
|
|
||||||
sendDataWs();
|
sendDataWs();
|
||||||
lastInterfaceUpdate = millis();
|
lastInterfaceUpdate = millis();
|
||||||
|
interfaceUpdateCallMode = 0; //disable further updates
|
||||||
|
|
||||||
if (callMode == CALL_MODE_WS_SEND) return;
|
if (callMode == CALL_MODE_WS_SEND) return;
|
||||||
|
|
||||||
#ifndef WLED_DISABLE_ALEXA
|
#ifndef WLED_DISABLE_ALEXA
|
||||||
@ -174,7 +176,6 @@ void updateInterfaces(uint8_t callMode)
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
doPublishMqtt = true;
|
doPublishMqtt = true;
|
||||||
interfaceUpdateCallMode = 0; //disable
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -226,7 +227,7 @@ void handleNightlight()
|
|||||||
if (!nightlightActiveOld) //init
|
if (!nightlightActiveOld) //init
|
||||||
{
|
{
|
||||||
nightlightStartTime = millis();
|
nightlightStartTime = millis();
|
||||||
nightlightDelayMs = (int)(nightlightDelayMins*60000);
|
nightlightDelayMs = (unsigned)(nightlightDelayMins*60000);
|
||||||
nightlightActiveOld = true;
|
nightlightActiveOld = true;
|
||||||
briNlT = bri;
|
briNlT = bri;
|
||||||
for (byte i=0; i<4; i++) colNlT[i] = col[i]; // remember starting color
|
for (byte i=0; i<4; i++) colNlT[i] = col[i]; // remember starting color
|
||||||
|
@ -1,8 +1,5 @@
|
|||||||
#include "wled.h"
|
#include "wled.h"
|
||||||
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
#define ESP_NOW_STATE_UNINIT 0
|
|
||||||
#define ESP_NOW_STATE_ON 1
|
|
||||||
#define ESP_NOW_STATE_ERROR 2
|
|
||||||
|
|
||||||
#define NIGHT_MODE_DEACTIVATED -1
|
#define NIGHT_MODE_DEACTIVATED -1
|
||||||
#define NIGHT_MODE_BRIGHTNESS 5
|
#define NIGHT_MODE_BRIGHTNESS 5
|
||||||
@ -17,59 +14,54 @@
|
|||||||
#define WIZMOTE_BUTTON_BRIGHT_UP 9
|
#define WIZMOTE_BUTTON_BRIGHT_UP 9
|
||||||
#define WIZMOTE_BUTTON_BRIGHT_DOWN 8
|
#define WIZMOTE_BUTTON_BRIGHT_DOWN 8
|
||||||
|
|
||||||
#ifdef WLED_DISABLE_ESPNOW
|
|
||||||
void handleRemote(){}
|
|
||||||
#else
|
|
||||||
|
|
||||||
// This is kind of an esoteric strucure because it's pulled from the "Wizmote"
|
// This is kind of an esoteric strucure because it's pulled from the "Wizmote"
|
||||||
// product spec. That remote is used as the baseline for behavior and availability
|
// product spec. That remote is used as the baseline for behavior and availability
|
||||||
// since it's broadly commercially available and works out of the box as a drop-in
|
// since it's broadly commercially available and works out of the box as a drop-in
|
||||||
typedef struct message_structure {
|
typedef struct WizMoteMessageStructure {
|
||||||
uint8_t program; // 0x91 for ON button, 0x81 for all others
|
uint8_t program; // 0x91 for ON button, 0x81 for all others
|
||||||
uint8_t seq[4]; // Incremetal sequence number 32 bit unsigned integer LSB first
|
uint8_t seq[4]; // Incremetal sequence number 32 bit unsigned integer LSB first
|
||||||
uint8_t byte5 = 32; // Unknown
|
uint8_t byte5; // Unknown (seen 0x20)
|
||||||
uint8_t button; // Identifies which button is being pressed
|
uint8_t button; // Identifies which button is being pressed
|
||||||
uint8_t byte8 = 1; // Unknown, but always 0x01
|
uint8_t byte8; // Unknown, but always 0x01
|
||||||
uint8_t byte9 = 100; // Unnkown, but always 0x64
|
uint8_t byte9; // Unnkown, but always 0x64
|
||||||
|
|
||||||
uint8_t byte10; // Unknown, maybe checksum
|
uint8_t byte10; // Unknown, maybe checksum
|
||||||
uint8_t byte11; // Unknown, maybe checksum
|
uint8_t byte11; // Unknown, maybe checksum
|
||||||
uint8_t byte12; // Unknown, maybe checksum
|
uint8_t byte12; // Unknown, maybe checksum
|
||||||
uint8_t byte13; // Unknown, maybe checksum
|
uint8_t byte13; // Unknown, maybe checksum
|
||||||
} message_structure;
|
} message_structure_t;
|
||||||
|
|
||||||
static int esp_now_state = ESP_NOW_STATE_UNINIT;
|
|
||||||
static uint32_t last_seq = UINT32_MAX;
|
static uint32_t last_seq = UINT32_MAX;
|
||||||
static int brightnessBeforeNightMode = NIGHT_MODE_DEACTIVATED;
|
static int brightnessBeforeNightMode = NIGHT_MODE_DEACTIVATED;
|
||||||
static message_structure incoming;
|
|
||||||
|
|
||||||
// Pulled from the IR Remote logic but reduced to 10 steps with a constant of 3
|
// Pulled from the IR Remote logic but reduced to 10 steps with a constant of 3
|
||||||
static const byte brightnessSteps[] = {
|
static const byte brightnessSteps[] = {
|
||||||
6, 9, 14, 22, 33, 50, 75, 113, 170, 255
|
6, 9, 14, 22, 33, 50, 75, 113, 170, 255
|
||||||
};
|
};
|
||||||
static const size_t numBrightnessSteps = sizeof(brightnessSteps) / sizeof(uint8_t);
|
static const size_t numBrightnessSteps = sizeof(brightnessSteps) / sizeof(byte);
|
||||||
|
|
||||||
static bool nightModeActive() {
|
inline bool nightModeActive() {
|
||||||
return brightnessBeforeNightMode != NIGHT_MODE_DEACTIVATED;
|
return brightnessBeforeNightMode != NIGHT_MODE_DEACTIVATED;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void activateNightMode() {
|
static void activateNightMode() {
|
||||||
|
if (nightModeActive()) return;
|
||||||
brightnessBeforeNightMode = bri;
|
brightnessBeforeNightMode = bri;
|
||||||
bri = NIGHT_MODE_BRIGHTNESS;
|
bri = NIGHT_MODE_BRIGHTNESS;
|
||||||
|
stateUpdated(CALL_MODE_BUTTON);
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool resetNightMode() {
|
static bool resetNightMode() {
|
||||||
if (!nightModeActive()) {
|
if (!nightModeActive()) return false;
|
||||||
return false;
|
|
||||||
}
|
|
||||||
bri = brightnessBeforeNightMode;
|
bri = brightnessBeforeNightMode;
|
||||||
brightnessBeforeNightMode = NIGHT_MODE_DEACTIVATED;
|
brightnessBeforeNightMode = NIGHT_MODE_DEACTIVATED;
|
||||||
|
stateUpdated(CALL_MODE_BUTTON);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// increment `bri` to the next `brightnessSteps` value
|
// increment `bri` to the next `brightnessSteps` value
|
||||||
static void brightnessUp() {
|
static void brightnessUp() {
|
||||||
if (nightModeActive()) { return; }
|
if (nightModeActive()) return;
|
||||||
// dumb incremental search is efficient enough for so few items
|
// dumb incremental search is efficient enough for so few items
|
||||||
for (uint8_t index = 0; index < numBrightnessSteps; ++index) {
|
for (uint8_t index = 0; index < numBrightnessSteps; ++index) {
|
||||||
if (brightnessSteps[index] > bri) {
|
if (brightnessSteps[index] > bri) {
|
||||||
@ -77,11 +69,12 @@ static void brightnessUp() {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
stateUpdated(CALL_MODE_BUTTON);
|
||||||
}
|
}
|
||||||
|
|
||||||
// decrement `bri` to the next `brightnessSteps` value
|
// decrement `bri` to the next `brightnessSteps` value
|
||||||
static void brightnessDown() {
|
static void brightnessDown() {
|
||||||
if (nightModeActive()) { return; }
|
if (nightModeActive()) return;
|
||||||
// dumb incremental search is efficient enough for so few items
|
// dumb incremental search is efficient enough for so few items
|
||||||
for (int index = numBrightnessSteps - 1; index >= 0; --index) {
|
for (int index = numBrightnessSteps - 1; index >= 0; --index) {
|
||||||
if (brightnessSteps[index] < bri) {
|
if (brightnessSteps[index] < bri) {
|
||||||
@ -89,39 +82,97 @@ static void brightnessDown() {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
stateUpdated(CALL_MODE_BUTTON);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void setOn() {
|
static void setOn() {
|
||||||
if (resetNightMode()) {
|
resetNightMode();
|
||||||
stateUpdated(CALL_MODE_BUTTON);
|
|
||||||
}
|
|
||||||
if (!bri) {
|
if (!bri) {
|
||||||
toggleOnOff();
|
toggleOnOff();
|
||||||
|
stateUpdated(CALL_MODE_BUTTON);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void setOff() {
|
static void setOff() {
|
||||||
if (resetNightMode()) {
|
resetNightMode();
|
||||||
|
if (bri) {
|
||||||
|
toggleOnOff();
|
||||||
stateUpdated(CALL_MODE_BUTTON);
|
stateUpdated(CALL_MODE_BUTTON);
|
||||||
}
|
}
|
||||||
if (bri) {
|
|
||||||
toggleOnOff();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void presetWithFallback(uint8_t presetID, uint8_t effectID, uint8_t paletteID) {
|
inline void presetWithFallback(uint8_t presetID, uint8_t effectID, uint8_t paletteID) {
|
||||||
|
resetNightMode();
|
||||||
applyPresetWithFallback(presetID, CALL_MODE_BUTTON_PRESET, effectID, paletteID);
|
applyPresetWithFallback(presetID, CALL_MODE_BUTTON_PRESET, effectID, paletteID);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Callback function that will be executed when data is received
|
|
||||||
#ifdef ESP8266
|
|
||||||
void OnDataRecv(uint8_t * mac, uint8_t *incomingData, uint8_t len) {
|
|
||||||
#else
|
|
||||||
void OnDataRecv(const uint8_t * mac, const uint8_t *incomingData, int len) {
|
|
||||||
#endif
|
|
||||||
|
|
||||||
sprintf (last_signal_src, "%02x%02x%02x%02x%02x%02x",
|
// this function follows the same principle as decodeIRJson()
|
||||||
mac [0], mac [1], mac [2], mac [3], mac [4], mac [5]);
|
static bool remoteJson(int button)
|
||||||
|
{
|
||||||
|
char objKey[10];
|
||||||
|
bool parsed = false;
|
||||||
|
|
||||||
|
if (!requestJSONBufferLock(22)) return false;
|
||||||
|
|
||||||
|
sprintf_P(objKey, PSTR("\"%d\":"), button);
|
||||||
|
|
||||||
|
// attempt to read command from remote.json
|
||||||
|
readObjectFromFile("/remote.json", objKey, &doc);
|
||||||
|
JsonObject fdo = doc.as<JsonObject>();
|
||||||
|
if (fdo.isNull()) {
|
||||||
|
// the received button does not exist
|
||||||
|
if (!WLED_FS.exists("/remote.json")) errorFlag = ERR_FS_RMLOAD; //warn if file itself doesn't exist
|
||||||
|
releaseJSONBufferLock();
|
||||||
|
return parsed;
|
||||||
|
}
|
||||||
|
|
||||||
|
String cmdStr = fdo["cmd"].as<String>();
|
||||||
|
JsonObject jsonCmdObj = fdo["cmd"]; //object
|
||||||
|
|
||||||
|
if (jsonCmdObj.isNull()) // we could also use: fdo["cmd"].is<String>()
|
||||||
|
{
|
||||||
|
if (cmdStr.startsWith("!")) {
|
||||||
|
// call limited set of C functions
|
||||||
|
if (cmdStr.startsWith(F("!incBri"))) {
|
||||||
|
brightnessUp();
|
||||||
|
parsed = true;
|
||||||
|
} else if (cmdStr.startsWith(F("!decBri"))) {
|
||||||
|
brightnessDown();
|
||||||
|
parsed = true;
|
||||||
|
} else if (cmdStr.startsWith(F("!presetF"))) { //!presetFallback
|
||||||
|
uint8_t p1 = fdo["PL"] | 1;
|
||||||
|
uint8_t p2 = fdo["FX"] | random8(strip.getModeCount() -1);
|
||||||
|
uint8_t p3 = fdo["FP"] | 0;
|
||||||
|
presetWithFallback(p1, p2, p3);
|
||||||
|
parsed = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// HTTP API command
|
||||||
|
String apireq = "win"; apireq += '&'; // reduce flash string usage
|
||||||
|
//if (cmdStr.indexOf("~") || fdo["rpt"]) lastValidCode = code; // repeatable action
|
||||||
|
if (!cmdStr.startsWith(apireq)) cmdStr = apireq + cmdStr; // if no "win&" prefix
|
||||||
|
if (!irApplyToAllSelected && cmdStr.indexOf(F("SS="))<0) {
|
||||||
|
char tmp[10];
|
||||||
|
sprintf_P(tmp, PSTR("&SS=%d"), strip.getMainSegmentId());
|
||||||
|
cmdStr += tmp;
|
||||||
|
}
|
||||||
|
fdo.clear(); // clear JSON buffer (it is no longer needed)
|
||||||
|
handleSet(nullptr, cmdStr, false); // no stateUpdated() call here
|
||||||
|
stateUpdated(CALL_MODE_BUTTON);
|
||||||
|
parsed = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// command is JSON object (TODO: currently will not handle irApplyToAllSelected correctly)
|
||||||
|
deserializeState(jsonCmdObj, CALL_MODE_BUTTON);
|
||||||
|
parsed = true;
|
||||||
|
}
|
||||||
|
releaseJSONBufferLock();
|
||||||
|
return parsed;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Callback function that will be executed when data is received
|
||||||
|
void handleRemote(uint8_t *incomingData, size_t len) {
|
||||||
|
message_structure_t *incoming = reinterpret_cast<message_structure_t *>(incomingData);
|
||||||
|
|
||||||
if (strcmp(last_signal_src, linked_remote) != 0) {
|
if (strcmp(last_signal_src, linked_remote) != 0) {
|
||||||
DEBUG_PRINT(F("ESP Now Message Received from Unlinked Sender: "));
|
DEBUG_PRINT(F("ESP Now Message Received from Unlinked Sender: "));
|
||||||
@ -129,72 +180,40 @@ void OnDataRecv(const uint8_t * mac, const uint8_t *incomingData, int len) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (len != sizeof(incoming)) {
|
if (len != sizeof(message_structure_t)) {
|
||||||
DEBUG_PRINT(F("Unknown incoming ESP Now message received of length "));
|
DEBUG_PRINT(F("Unknown incoming ESP Now message received of length "));
|
||||||
DEBUG_PRINTLN(len);
|
DEBUG_PRINTLN(len);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy(&(incoming.program), incomingData, sizeof(incoming));
|
uint32_t cur_seq = incoming->seq[0] | (incoming->seq[1] << 8) | (incoming->seq[2] << 16) | (incoming->seq[3] << 24);
|
||||||
uint32_t cur_seq = incoming.seq[0] | (incoming.seq[1] << 8) | (incoming.seq[2] << 16) | (incoming.seq[3] << 24);
|
|
||||||
|
|
||||||
if (cur_seq == last_seq) {
|
if (cur_seq == last_seq) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
DEBUG_PRINT(F("Incoming ESP Now Packet ["));
|
||||||
DEBUG_PRINT(F("Incoming ESP Now Packet["));
|
|
||||||
DEBUG_PRINT(cur_seq);
|
DEBUG_PRINT(cur_seq);
|
||||||
DEBUG_PRINT(F("] from sender["));
|
DEBUG_PRINT(F("] from sender ["));
|
||||||
DEBUG_PRINT(last_signal_src);
|
DEBUG_PRINT(last_signal_src);
|
||||||
DEBUG_PRINT(F("] button: "));
|
DEBUG_PRINT(F("] button: "));
|
||||||
DEBUG_PRINTLN(incoming.button);
|
DEBUG_PRINTLN(incoming->button);
|
||||||
switch (incoming.button) {
|
|
||||||
case WIZMOTE_BUTTON_ON : setOn(); stateUpdated(CALL_MODE_BUTTON); break;
|
|
||||||
case WIZMOTE_BUTTON_OFF : setOff(); stateUpdated(CALL_MODE_BUTTON); break;
|
|
||||||
case WIZMOTE_BUTTON_ONE : presetWithFallback(1, FX_MODE_STATIC, 0); resetNightMode(); break;
|
|
||||||
case WIZMOTE_BUTTON_TWO : presetWithFallback(2, FX_MODE_BREATH, 0); resetNightMode(); break;
|
|
||||||
case WIZMOTE_BUTTON_THREE : presetWithFallback(3, FX_MODE_FIRE_FLICKER, 0); resetNightMode(); break;
|
|
||||||
case WIZMOTE_BUTTON_FOUR : presetWithFallback(4, FX_MODE_RAINBOW, 0); resetNightMode(); break;
|
|
||||||
case WIZMOTE_BUTTON_NIGHT : activateNightMode(); stateUpdated(CALL_MODE_BUTTON); break;
|
|
||||||
case WIZMOTE_BUTTON_BRIGHT_UP : brightnessUp(); stateUpdated(CALL_MODE_BUTTON); break;
|
|
||||||
case WIZMOTE_BUTTON_BRIGHT_DOWN : brightnessDown(); stateUpdated(CALL_MODE_BUTTON); break;
|
|
||||||
default: break;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
if (!remoteJson(incoming->button))
|
||||||
|
switch (incoming->button) {
|
||||||
|
case WIZMOTE_BUTTON_ON : setOn(); break;
|
||||||
|
case WIZMOTE_BUTTON_OFF : setOff(); break;
|
||||||
|
case WIZMOTE_BUTTON_ONE : presetWithFallback(1, FX_MODE_STATIC, 0); break;
|
||||||
|
case WIZMOTE_BUTTON_TWO : presetWithFallback(2, FX_MODE_BREATH, 0); break;
|
||||||
|
case WIZMOTE_BUTTON_THREE : presetWithFallback(3, FX_MODE_FIRE_FLICKER, 0); break;
|
||||||
|
case WIZMOTE_BUTTON_FOUR : presetWithFallback(4, FX_MODE_RAINBOW, 0); break;
|
||||||
|
case WIZMOTE_BUTTON_NIGHT : activateNightMode(); break;
|
||||||
|
case WIZMOTE_BUTTON_BRIGHT_UP : brightnessUp(); break;
|
||||||
|
case WIZMOTE_BUTTON_BRIGHT_DOWN : brightnessDown(); break;
|
||||||
|
default: break;
|
||||||
|
}
|
||||||
last_seq = cur_seq;
|
last_seq = cur_seq;
|
||||||
}
|
}
|
||||||
|
|
||||||
void handleRemote() {
|
#else
|
||||||
if (enable_espnow_remote) {
|
void handleRemote(uint8_t *incomingData, size_t len) {}
|
||||||
if ((esp_now_state == ESP_NOW_STATE_UNINIT) && (interfacesInited || apActive)) { // ESPNOW requires Wifi to be initialized (either STA, or AP Mode)
|
#endif
|
||||||
DEBUG_PRINTLN(F("Initializing ESP_NOW listener"));
|
|
||||||
// Init ESP-NOW
|
|
||||||
if (esp_now_init() != 0) {
|
|
||||||
DEBUG_PRINTLN(F("Error initializing ESP-NOW"));
|
|
||||||
esp_now_state = ESP_NOW_STATE_ERROR;
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef ESP8266
|
|
||||||
esp_now_set_self_role(ESP_NOW_ROLE_SLAVE);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
esp_now_register_recv_cb(OnDataRecv);
|
|
||||||
esp_now_state = ESP_NOW_STATE_ON;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (esp_now_state == ESP_NOW_STATE_ON) {
|
|
||||||
DEBUG_PRINTLN(F("Disabling ESP-NOW Remote Listener"));
|
|
||||||
if (esp_now_deinit() != 0) {
|
|
||||||
DEBUG_PRINTLN(F("Error de-initializing ESP-NOW"));
|
|
||||||
}
|
|
||||||
esp_now_state = ESP_NOW_STATE_UNINIT;
|
|
||||||
} else if (esp_now_state == ESP_NOW_STATE_ERROR) {
|
|
||||||
//Clear any error states (allows retrying by cycling)
|
|
||||||
esp_now_state = ESP_NOW_STATE_UNINIT;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
@ -19,27 +19,41 @@ void handleSettingsSet(AsyncWebServerRequest *request, byte subPage)
|
|||||||
//WIFI SETTINGS
|
//WIFI SETTINGS
|
||||||
if (subPage == SUBPAGE_WIFI)
|
if (subPage == SUBPAGE_WIFI)
|
||||||
{
|
{
|
||||||
strlcpy(clientSSID,request->arg(F("CS")).c_str(), 33);
|
char oldSSID[sizeof(clientSSID)];
|
||||||
|
|
||||||
if (!isAsterisksOnly(request->arg(F("CP")).c_str(), 65)) strlcpy(clientPass, request->arg(F("CP")).c_str(), 65);
|
strcpy(oldSSID, clientSSID);
|
||||||
|
strlcpy(clientSSID,request->arg(F("CS")).c_str(), 33);
|
||||||
|
if (!strcmp(oldSSID, clientSSID)) forceReconnect = true;
|
||||||
|
|
||||||
|
if (!isAsterisksOnly(request->arg(F("CP")).c_str(), 65)) {
|
||||||
|
strlcpy(clientPass, request->arg(F("CP")).c_str(), 65);
|
||||||
|
forceReconnect = true;
|
||||||
|
}
|
||||||
|
|
||||||
strlcpy(cmDNS, request->arg(F("CM")).c_str(), 33);
|
strlcpy(cmDNS, request->arg(F("CM")).c_str(), 33);
|
||||||
|
|
||||||
apBehavior = request->arg(F("AB")).toInt();
|
apBehavior = request->arg(F("AB")).toInt();
|
||||||
|
strcpy(oldSSID, apSSID);
|
||||||
strlcpy(apSSID, request->arg(F("AS")).c_str(), 33);
|
strlcpy(apSSID, request->arg(F("AS")).c_str(), 33);
|
||||||
|
if (!strcmp(oldSSID, apSSID) && apActive) forceReconnect = true;
|
||||||
apHide = request->hasArg(F("AH"));
|
apHide = request->hasArg(F("AH"));
|
||||||
int passlen = request->arg(F("AP")).length();
|
int passlen = request->arg(F("AP")).length();
|
||||||
if (passlen == 0 || (passlen > 7 && !isAsterisksOnly(request->arg(F("AP")).c_str(), 65))) strlcpy(apPass, request->arg(F("AP")).c_str(), 65);
|
if (passlen == 0 || (passlen > 7 && !isAsterisksOnly(request->arg(F("AP")).c_str(), 65))) {
|
||||||
int t = request->arg(F("AC")).toInt(); if (t > 0 && t < 14) apChannel = t;
|
strlcpy(apPass, request->arg(F("AP")).c_str(), 65);
|
||||||
|
forceReconnect = true;
|
||||||
|
}
|
||||||
|
int t = request->arg(F("AC")).toInt();
|
||||||
|
if (t != apChannel) forceReconnect = true;
|
||||||
|
if (t > 0 && t < 14) apChannel = t;
|
||||||
|
|
||||||
noWifiSleep = request->hasArg(F("WS"));
|
noWifiSleep = request->hasArg(F("WS"));
|
||||||
|
|
||||||
#ifndef WLED_DISABLE_ESPNOW
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
enable_espnow_remote = request->hasArg(F("RE"));
|
bool oldESPNow = enableESPNow;
|
||||||
strlcpy(linked_remote,request->arg(F("RMAC")).c_str(), 13);
|
enableESPNow = request->hasArg(F("RE"));
|
||||||
|
if (oldESPNow != enableESPNow) forceReconnect = true;
|
||||||
//Normalize MAC format to lowercase
|
strlcpy(linked_remote, request->arg(F("RMAC")).c_str(), 13);
|
||||||
strlcpy(linked_remote,strlwr(linked_remote), 13);
|
strlwr(linked_remote); //Normalize MAC format to lowercase
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef WLED_USE_ETHERNET
|
#ifdef WLED_USE_ETHERNET
|
||||||
@ -247,6 +261,7 @@ void handleSettingsSet(AsyncWebServerRequest *request, byte subPage)
|
|||||||
}
|
}
|
||||||
|
|
||||||
fadeTransition = request->hasArg(F("TF"));
|
fadeTransition = request->hasArg(F("TF"));
|
||||||
|
modeBlending = request->hasArg(F("EB"));
|
||||||
t = request->arg(F("TD")).toInt();
|
t = request->arg(F("TD")).toInt();
|
||||||
if (t >= 0) transitionDelayDefault = t;
|
if (t >= 0) transitionDelayDefault = t;
|
||||||
strip.paletteFade = request->hasArg(F("PF"));
|
strip.paletteFade = request->hasArg(F("PF"));
|
||||||
@ -271,7 +286,7 @@ void handleSettingsSet(AsyncWebServerRequest *request, byte subPage)
|
|||||||
if (subPage == SUBPAGE_UI)
|
if (subPage == SUBPAGE_UI)
|
||||||
{
|
{
|
||||||
strlcpy(serverDescription, request->arg(F("DS")).c_str(), 33);
|
strlcpy(serverDescription, request->arg(F("DS")).c_str(), 33);
|
||||||
syncToggleReceive = request->hasArg(F("ST"));
|
//syncToggleReceive = request->hasArg(F("ST"));
|
||||||
#ifdef WLED_ENABLE_SIMPLE_UI
|
#ifdef WLED_ENABLE_SIMPLE_UI
|
||||||
if (simplifiedUI ^ request->hasArg(F("SU"))) {
|
if (simplifiedUI ^ request->hasArg(F("SU"))) {
|
||||||
// UI selection changed, invalidate browser cache
|
// UI selection changed, invalidate browser cache
|
||||||
@ -293,6 +308,10 @@ void handleSettingsSet(AsyncWebServerRequest *request, byte subPage)
|
|||||||
t = request->arg(F("U2")).toInt();
|
t = request->arg(F("U2")).toInt();
|
||||||
if (t > 0) udpPort2 = t;
|
if (t > 0) udpPort2 = t;
|
||||||
|
|
||||||
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
|
useESPNowSync = request->hasArg(F("EN"));
|
||||||
|
#endif
|
||||||
|
|
||||||
syncGroups = request->arg(F("GS")).toInt();
|
syncGroups = request->arg(F("GS")).toInt();
|
||||||
receiveGroups = request->arg(F("GR")).toInt();
|
receiveGroups = request->arg(F("GR")).toInt();
|
||||||
|
|
||||||
@ -301,9 +320,8 @@ void handleSettingsSet(AsyncWebServerRequest *request, byte subPage)
|
|||||||
receiveNotificationEffects = request->hasArg(F("RX"));
|
receiveNotificationEffects = request->hasArg(F("RX"));
|
||||||
receiveSegmentOptions = request->hasArg(F("SO"));
|
receiveSegmentOptions = request->hasArg(F("SO"));
|
||||||
receiveSegmentBounds = request->hasArg(F("SG"));
|
receiveSegmentBounds = request->hasArg(F("SG"));
|
||||||
receiveNotifications = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects || receiveSegmentOptions);
|
sendNotifications = request->hasArg(F("SS"));
|
||||||
notifyDirectDefault = request->hasArg(F("SD"));
|
notifyDirect = request->hasArg(F("SD"));
|
||||||
notifyDirect = notifyDirectDefault;
|
|
||||||
notifyButton = request->hasArg(F("SB"));
|
notifyButton = request->hasArg(F("SB"));
|
||||||
notifyAlexa = request->hasArg(F("SA"));
|
notifyAlexa = request->hasArg(F("SA"));
|
||||||
notifyHue = request->hasArg(F("SH"));
|
notifyHue = request->hasArg(F("SH"));
|
||||||
@ -317,7 +335,7 @@ void handleSettingsSet(AsyncWebServerRequest *request, byte subPage)
|
|||||||
if (!nodeListEnabled) Nodes.clear();
|
if (!nodeListEnabled) Nodes.clear();
|
||||||
nodeBroadcastEnabled = request->hasArg(F("NB"));
|
nodeBroadcastEnabled = request->hasArg(F("NB"));
|
||||||
|
|
||||||
receiveDirect = request->hasArg(F("RD"));
|
receiveDirect = request->hasArg(F("RD")); // UDP realtime
|
||||||
useMainSegmentOnly = request->hasArg(F("MO"));
|
useMainSegmentOnly = request->hasArg(F("MO"));
|
||||||
e131SkipOutOfSequence = request->hasArg(F("ES"));
|
e131SkipOutOfSequence = request->hasArg(F("ES"));
|
||||||
e131Multicast = request->hasArg(F("EM"));
|
e131Multicast = request->hasArg(F("EM"));
|
||||||
@ -1003,7 +1021,7 @@ bool handleSet(AsyncWebServerRequest *request, const String& req, bool apply)
|
|||||||
|
|
||||||
//toggle receive UDP direct notifications
|
//toggle receive UDP direct notifications
|
||||||
pos = req.indexOf(F("RN="));
|
pos = req.indexOf(F("RN="));
|
||||||
if (pos > 0) receiveNotifications = (req.charAt(pos+3) != '0');
|
if (pos > 0) receiveGroups = (req.charAt(pos+3) != '0') ? receiveGroups | 1 : receiveGroups & 0xFE;
|
||||||
|
|
||||||
//receive live data via UDP/Hyperion
|
//receive live data via UDP/Hyperion
|
||||||
pos = req.indexOf(F("RD="));
|
pos = req.indexOf(F("RD="));
|
||||||
|
480
wled00/udp.cpp
480
wled00/udp.cpp
@ -5,15 +5,26 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#define UDP_SEG_SIZE 36
|
#define UDP_SEG_SIZE 36
|
||||||
#define SEG_OFFSET (41+(MAX_NUM_SEGMENTS*UDP_SEG_SIZE))
|
#define SEG_OFFSET (41)
|
||||||
#define WLEDPACKETSIZE (41+(MAX_NUM_SEGMENTS*UDP_SEG_SIZE)+0)
|
#define WLEDPACKETSIZE (41+(MAX_NUM_SEGMENTS*UDP_SEG_SIZE)+0)
|
||||||
#define UDP_IN_MAXSIZE 1472
|
#define UDP_IN_MAXSIZE 1472
|
||||||
#define PRESUMED_NETWORK_DELAY 3 //how many ms could it take on avg to reach the receiver? This will be added to transmitted times
|
#define PRESUMED_NETWORK_DELAY 3 //how many ms could it take on avg to reach the receiver? This will be added to transmitted times
|
||||||
|
|
||||||
|
typedef struct PartialEspNowPacket {
|
||||||
|
uint8_t magic;
|
||||||
|
uint8_t packet;
|
||||||
|
uint8_t segs;
|
||||||
|
uint8_t data[247];
|
||||||
|
} partial_packet_t;
|
||||||
|
|
||||||
void notify(byte callMode, bool followUp)
|
void notify(byte callMode, bool followUp)
|
||||||
{
|
{
|
||||||
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
|
if (!udpConnected && !useESPNowSync) return;
|
||||||
|
#else
|
||||||
if (!udpConnected) return;
|
if (!udpConnected) return;
|
||||||
if (!syncGroups) return;
|
#endif
|
||||||
|
if (!syncGroups || !sendNotificationsRT) return;
|
||||||
switch (callMode)
|
switch (callMode)
|
||||||
{
|
{
|
||||||
case CALL_MODE_INIT: return;
|
case CALL_MODE_INIT: return;
|
||||||
@ -26,7 +37,7 @@ void notify(byte callMode, bool followUp)
|
|||||||
case CALL_MODE_ALEXA: if (!notifyAlexa) return; break;
|
case CALL_MODE_ALEXA: if (!notifyAlexa) return; break;
|
||||||
default: return;
|
default: return;
|
||||||
}
|
}
|
||||||
byte udpOut[WLEDPACKETSIZE];
|
byte udpOut[WLEDPACKETSIZE]; //TODO: optimize size to use only active segments
|
||||||
Segment& mainseg = strip.getMainSegment();
|
Segment& mainseg = strip.getMainSegment();
|
||||||
udpOut[0] = 0; //0: wled notifier protocol 1: WARLS protocol
|
udpOut[0] = 0; //0: wled notifier protocol 1: WARLS protocol
|
||||||
udpOut[1] = callMode;
|
udpOut[1] = callMode;
|
||||||
@ -138,17 +149,232 @@ void notify(byte callMode, bool followUp)
|
|||||||
//uint16_t offs = SEG_OFFSET;
|
//uint16_t offs = SEG_OFFSET;
|
||||||
//next value to be added has index: udpOut[offs + 0]
|
//next value to be added has index: udpOut[offs + 0]
|
||||||
|
|
||||||
IPAddress broadcastIp;
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
broadcastIp = ~uint32_t(Network.subnetMask()) | uint32_t(Network.gatewayIP());
|
if (enableESPNow && useESPNowSync && statusESPNow == ESP_NOW_STATE_ON) {
|
||||||
|
partial_packet_t buffer = {'W', 0, (uint8_t)s, {0}};
|
||||||
notifierUdp.beginPacket(broadcastIp, udpPort);
|
// send global data
|
||||||
notifierUdp.write(udpOut, WLEDPACKETSIZE);
|
DEBUG_PRINTLN(F("ESP-NOW sending first packet."));
|
||||||
notifierUdp.endPacket();
|
memcpy(buffer.data, udpOut, 41);
|
||||||
|
auto err = quickEspNow.send(ESPNOW_BROADCAST_ADDRESS, reinterpret_cast<const uint8_t*>(&buffer), 41+3);
|
||||||
|
if (!err) {
|
||||||
|
// send segment data
|
||||||
|
buffer.packet++;
|
||||||
|
size_t packetSize = 0;
|
||||||
|
int32_t err = 0;
|
||||||
|
for (size_t i = 0; i < s; i++) {
|
||||||
|
memcpy(buffer.data + packetSize, &udpOut[41+i*UDP_SEG_SIZE], UDP_SEG_SIZE);
|
||||||
|
packetSize += UDP_SEG_SIZE;
|
||||||
|
if (packetSize + UDP_SEG_SIZE < sizeof(buffer.data)/sizeof(uint8_t)) continue;
|
||||||
|
DEBUG_PRINTF("ESP-NOW sending packet: %d (%d)\n", (int)buffer.packet, packetSize+3);
|
||||||
|
err = quickEspNow.send(ESPNOW_BROADCAST_ADDRESS, reinterpret_cast<const uint8_t*>(&buffer), packetSize+3);
|
||||||
|
buffer.packet++;
|
||||||
|
packetSize = 0;
|
||||||
|
if (err) break;
|
||||||
|
}
|
||||||
|
if (!err && packetSize > 0) {
|
||||||
|
DEBUG_PRINTF("ESP-NOW sending last packet: %d (%d)\n", (int)buffer.packet, packetSize+3);
|
||||||
|
err = quickEspNow.send(ESPNOW_BROADCAST_ADDRESS, reinterpret_cast<const uint8_t*>(&buffer), packetSize+3);
|
||||||
|
}
|
||||||
|
if (err) {
|
||||||
|
DEBUG_PRINTLN(F("ESP-NOW sending packet failed."));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (udpConnected)
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
DEBUG_PRINTLN(F("UDP sending packet."));
|
||||||
|
IPAddress broadcastIp = ~uint32_t(Network.subnetMask()) | uint32_t(Network.gatewayIP());
|
||||||
|
notifierUdp.beginPacket(broadcastIp, udpPort);
|
||||||
|
notifierUdp.write(udpOut, WLEDPACKETSIZE); // TODO: add actual used buffer size
|
||||||
|
notifierUdp.endPacket();
|
||||||
|
}
|
||||||
notificationSentCallMode = callMode;
|
notificationSentCallMode = callMode;
|
||||||
notificationSentTime = millis();
|
notificationSentTime = millis();
|
||||||
notificationCount = followUp ? notificationCount + 1 : 0;
|
notificationCount = followUp ? notificationCount + 1 : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void parseNotifyPacket(uint8_t *udpIn) {
|
||||||
|
//ignore notification if received within a second after sending a notification ourselves
|
||||||
|
if (millis() - notificationSentTime < 1000) return;
|
||||||
|
if (udpIn[1] > 199) return; //do not receive custom versions
|
||||||
|
|
||||||
|
//compatibilityVersionByte:
|
||||||
|
byte version = udpIn[11];
|
||||||
|
DEBUG_PRINT(F("UDP packet version: ")); DEBUG_PRINTLN(version);
|
||||||
|
|
||||||
|
// if we are not part of any sync group ignore message
|
||||||
|
if (version < 9) {
|
||||||
|
// legacy senders are treated as if sending in sync group 1 only
|
||||||
|
if (!(receiveGroups & 0x01)) return;
|
||||||
|
} else if (!(receiveGroups & udpIn[36])) return;
|
||||||
|
|
||||||
|
bool someSel = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects);
|
||||||
|
|
||||||
|
//apply colors from notification to main segment, only if not syncing full segments
|
||||||
|
if ((receiveNotificationColor || !someSel) && (version < 11 || !receiveSegmentOptions)) {
|
||||||
|
// primary color, only apply white if intented (version > 0)
|
||||||
|
strip.setColor(0, RGBW32(udpIn[3], udpIn[4], udpIn[5], (version > 0) ? udpIn[10] : 0));
|
||||||
|
if (version > 1) {
|
||||||
|
strip.setColor(1, RGBW32(udpIn[12], udpIn[13], udpIn[14], udpIn[15])); // secondary color
|
||||||
|
}
|
||||||
|
if (version > 6) {
|
||||||
|
strip.setColor(2, RGBW32(udpIn[20], udpIn[21], udpIn[22], udpIn[23])); // tertiary color
|
||||||
|
if (version > 9 && udpIn[37] < 255) { // valid CCT/Kelvin value
|
||||||
|
uint16_t cct = udpIn[38];
|
||||||
|
if (udpIn[37] > 0) { //Kelvin
|
||||||
|
cct |= (udpIn[37] << 8);
|
||||||
|
}
|
||||||
|
strip.setCCT(cct);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool timebaseUpdated = false;
|
||||||
|
//apply effects from notification
|
||||||
|
bool applyEffects = (receiveNotificationEffects || !someSel);
|
||||||
|
if (applyEffects && currentPlaylist >= 0) unloadPlaylist();
|
||||||
|
if (version > 10 && (receiveSegmentOptions || receiveSegmentBounds)) {
|
||||||
|
uint8_t numSrcSegs = udpIn[39];
|
||||||
|
DEBUG_PRINT(F("UDP segments: ")); DEBUG_PRINTLN(numSrcSegs);
|
||||||
|
// are we syncing bounds and slave has more active segments than master?
|
||||||
|
if (receiveSegmentBounds && numSrcSegs < strip.getActiveSegmentsNum()) {
|
||||||
|
DEBUG_PRINTLN(F("Removing excessive segments."));
|
||||||
|
for (size_t i=strip.getSegmentsNum(); i>numSrcSegs; i--) {
|
||||||
|
if (strip.getSegment(i).isActive()) {
|
||||||
|
strip.setSegment(i-1,0,0); // delete segment
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
size_t inactiveSegs = 0;
|
||||||
|
for (size_t i = 0; i < numSrcSegs && i < strip.getMaxSegments(); i++) {
|
||||||
|
uint16_t ofs = 41 + i*udpIn[40]; //start of segment offset byte
|
||||||
|
uint8_t id = udpIn[0 +ofs];
|
||||||
|
DEBUG_PRINT(F("UDP segment received: ")); DEBUG_PRINTLN(id);
|
||||||
|
if (id > strip.getSegmentsNum()) break;
|
||||||
|
else if (id == strip.getSegmentsNum()) {
|
||||||
|
if (receiveSegmentBounds && id < strip.getMaxSegments()) strip.appendSegment();
|
||||||
|
else break;
|
||||||
|
}
|
||||||
|
DEBUG_PRINT(F("UDP segment check: ")); DEBUG_PRINTLN(id);
|
||||||
|
Segment& selseg = strip.getSegment(id);
|
||||||
|
// if we are not syncing bounds skip unselected segments
|
||||||
|
if (selseg.isActive() && !(selseg.isSelected() || receiveSegmentBounds)) continue;
|
||||||
|
// ignore segment if it is inactive and we are not syncing bounds
|
||||||
|
if (!receiveSegmentBounds) {
|
||||||
|
if (!selseg.isActive()) {
|
||||||
|
inactiveSegs++;
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
id += inactiveSegs; // adjust id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
DEBUG_PRINT(F("UDP segment processing: ")); DEBUG_PRINTLN(id);
|
||||||
|
|
||||||
|
uint16_t startY = 0, start = (udpIn[1+ofs] << 8 | udpIn[2+ofs]);
|
||||||
|
uint16_t stopY = 1, stop = (udpIn[3+ofs] << 8 | udpIn[4+ofs]);
|
||||||
|
uint16_t offset = (udpIn[7+ofs] << 8 | udpIn[8+ofs]);
|
||||||
|
if (!receiveSegmentOptions) {
|
||||||
|
//selseg.setUp(start, stop, selseg.grouping, selseg.spacing, offset, startY, stopY);
|
||||||
|
// we have to use strip.setSegment() instead of selseg.setUp() to prevent crash if segment would change during drawing
|
||||||
|
strip.setSegment(id, start, stop, selseg.grouping, selseg.spacing, offset, startY, stopY);
|
||||||
|
continue; // we do receive bounds, but not options
|
||||||
|
}
|
||||||
|
selseg.options = (selseg.options & 0x0071U) | (udpIn[9 +ofs] & 0x0E); // ignore selected, freeze, reset & transitional
|
||||||
|
selseg.setOpacity(udpIn[10+ofs]);
|
||||||
|
if (applyEffects) {
|
||||||
|
selseg.setMode(udpIn[11+ofs]);
|
||||||
|
selseg.speed = udpIn[12+ofs];
|
||||||
|
selseg.intensity = udpIn[13+ofs];
|
||||||
|
selseg.palette = udpIn[14+ofs];
|
||||||
|
}
|
||||||
|
if (receiveNotificationColor || !someSel) {
|
||||||
|
selseg.setColor(0, RGBW32(udpIn[15+ofs],udpIn[16+ofs],udpIn[17+ofs],udpIn[18+ofs]));
|
||||||
|
selseg.setColor(1, RGBW32(udpIn[19+ofs],udpIn[20+ofs],udpIn[21+ofs],udpIn[22+ofs]));
|
||||||
|
selseg.setColor(2, RGBW32(udpIn[23+ofs],udpIn[24+ofs],udpIn[25+ofs],udpIn[26+ofs]));
|
||||||
|
selseg.setCCT(udpIn[27+ofs]);
|
||||||
|
}
|
||||||
|
if (version > 11) {
|
||||||
|
// when applying synced options ignore selected as it may be used as indicator of which segments to sync
|
||||||
|
// freeze, reset & transitional should never be synced
|
||||||
|
selseg.options = (selseg.options & 0x0071U) | (udpIn[28+ofs]<<8) | (udpIn[9 +ofs] & 0x8E); // ignore selected, freeze, reset & transitional
|
||||||
|
if (applyEffects) {
|
||||||
|
selseg.custom1 = udpIn[29+ofs];
|
||||||
|
selseg.custom2 = udpIn[30+ofs];
|
||||||
|
selseg.custom3 = udpIn[31+ofs] & 0x1F;
|
||||||
|
selseg.check1 = (udpIn[31+ofs]>>5) & 0x1;
|
||||||
|
selseg.check1 = (udpIn[31+ofs]>>6) & 0x1;
|
||||||
|
selseg.check1 = (udpIn[31+ofs]>>7) & 0x1;
|
||||||
|
}
|
||||||
|
startY = (udpIn[32+ofs] << 8 | udpIn[33+ofs]);
|
||||||
|
stopY = (udpIn[34+ofs] << 8 | udpIn[35+ofs]);
|
||||||
|
}
|
||||||
|
if (receiveSegmentBounds) {
|
||||||
|
// we have to use strip.setSegment() instead of selseg.setUp() to prevent crash if segment would change during drawing
|
||||||
|
strip.setSegment(id, start, stop, udpIn[5+ofs], udpIn[6+ofs], offset, startY, stopY);
|
||||||
|
} else {
|
||||||
|
// we have to use strip.setSegment() instead of selseg.setUp() to prevent crash if segment would change during drawing
|
||||||
|
strip.setSegment(id, selseg.start, selseg.stop, udpIn[5+ofs], udpIn[6+ofs], selseg.offset, selseg.startY, selseg.stopY);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
stateChanged = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// simple effect sync, applies to all selected segments
|
||||||
|
if (applyEffects && (version < 11 || !receiveSegmentOptions)) {
|
||||||
|
for (size_t i = 0; i < strip.getSegmentsNum(); i++) {
|
||||||
|
Segment& seg = strip.getSegment(i);
|
||||||
|
if (!seg.isActive() || !seg.isSelected()) continue;
|
||||||
|
seg.setMode(udpIn[8]);
|
||||||
|
seg.speed = udpIn[9];
|
||||||
|
if (version > 2) seg.intensity = udpIn[16];
|
||||||
|
if (version > 4) seg.setPalette(udpIn[19]);
|
||||||
|
}
|
||||||
|
stateChanged = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (applyEffects && version > 5) {
|
||||||
|
uint32_t t = (udpIn[25] << 24) | (udpIn[26] << 16) | (udpIn[27] << 8) | (udpIn[28]);
|
||||||
|
t += PRESUMED_NETWORK_DELAY; //adjust trivially for network delay
|
||||||
|
t -= millis();
|
||||||
|
strip.timebase = t;
|
||||||
|
timebaseUpdated = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
//adjust system time, but only if sender is more accurate than self
|
||||||
|
if (version > 7) {
|
||||||
|
Toki::Time tm;
|
||||||
|
tm.sec = (udpIn[30] << 24) | (udpIn[31] << 16) | (udpIn[32] << 8) | (udpIn[33]);
|
||||||
|
tm.ms = (udpIn[34] << 8) | (udpIn[35]);
|
||||||
|
if (udpIn[29] > toki.getTimeSource()) { //if sender's time source is more accurate
|
||||||
|
toki.adjust(tm, PRESUMED_NETWORK_DELAY); //adjust trivially for network delay
|
||||||
|
uint8_t ts = TOKI_TS_UDP;
|
||||||
|
if (udpIn[29] > 99) ts = TOKI_TS_UDP_NTP;
|
||||||
|
else if (udpIn[29] >= TOKI_TS_SEC) ts = TOKI_TS_UDP_SEC;
|
||||||
|
toki.setTime(tm, ts);
|
||||||
|
} else if (timebaseUpdated && toki.getTimeSource() > 99) { //if we both have good times, get a more accurate timebase
|
||||||
|
Toki::Time myTime = toki.getTime();
|
||||||
|
uint32_t diff = toki.msDifference(tm, myTime);
|
||||||
|
strip.timebase -= PRESUMED_NETWORK_DELAY; //no need to presume, use difference between NTP times at send and receive points
|
||||||
|
if (toki.isLater(tm, myTime)) {
|
||||||
|
strip.timebase += diff;
|
||||||
|
} else {
|
||||||
|
strip.timebase -= diff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (version > 3) {
|
||||||
|
transitionDelayTemp = ((udpIn[17] << 0) & 0xFF) + ((udpIn[18] << 8) & 0xFF00);
|
||||||
|
}
|
||||||
|
|
||||||
|
nightlightActive = udpIn[6];
|
||||||
|
if (nightlightActive) nightlightDelayMins = udpIn[7];
|
||||||
|
|
||||||
|
if (receiveNotificationBrightness || !someSel) bri = udpIn[2];
|
||||||
|
stateUpdated(CALL_MODE_NOTIFICATION);
|
||||||
|
}
|
||||||
|
|
||||||
void realtimeLock(uint32_t timeoutMs, byte md)
|
void realtimeLock(uint32_t timeoutMs, byte md)
|
||||||
{
|
{
|
||||||
if (!realtimeMode && !realtimeOverride) {
|
if (!realtimeMode && !realtimeOverride) {
|
||||||
@ -262,8 +488,6 @@ void handleNotifications()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!(receiveNotifications || receiveDirect)) return;
|
|
||||||
|
|
||||||
localIP = Network.localIP();
|
localIP = Network.localIP();
|
||||||
//notifier and UDP realtime
|
//notifier and UDP realtime
|
||||||
if (!packetSize || packetSize > UDP_IN_MAXSIZE) return;
|
if (!packetSize || packetSize > UDP_IN_MAXSIZE) return;
|
||||||
@ -306,160 +530,9 @@ void handleNotifications()
|
|||||||
}
|
}
|
||||||
|
|
||||||
//wled notifier, ignore if realtime packets active
|
//wled notifier, ignore if realtime packets active
|
||||||
if (udpIn[0] == 0 && !realtimeMode && receiveNotifications)
|
if (udpIn[0] == 0 && !realtimeMode && receiveGroups)
|
||||||
{
|
{
|
||||||
//ignore notification if received within a second after sending a notification ourselves
|
parseNotifyPacket(udpIn);
|
||||||
if (millis() - notificationSentTime < 1000) return;
|
|
||||||
if (udpIn[1] > 199) return; //do not receive custom versions
|
|
||||||
|
|
||||||
//compatibilityVersionByte:
|
|
||||||
byte version = udpIn[11];
|
|
||||||
|
|
||||||
// if we are not part of any sync group ignore message
|
|
||||||
if (version < 9 || version > 199) {
|
|
||||||
// legacy senders are treated as if sending in sync group 1 only
|
|
||||||
if (!(receiveGroups & 0x01)) return;
|
|
||||||
} else if (!(receiveGroups & udpIn[36])) return;
|
|
||||||
|
|
||||||
bool someSel = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects);
|
|
||||||
|
|
||||||
//apply colors from notification to main segment, only if not syncing full segments
|
|
||||||
if ((receiveNotificationColor || !someSel) && (version < 11 || !receiveSegmentOptions)) {
|
|
||||||
// primary color, only apply white if intented (version > 0)
|
|
||||||
strip.setColor(0, RGBW32(udpIn[3], udpIn[4], udpIn[5], (version > 0) ? udpIn[10] : 0));
|
|
||||||
if (version > 1) {
|
|
||||||
strip.setColor(1, RGBW32(udpIn[12], udpIn[13], udpIn[14], udpIn[15])); // secondary color
|
|
||||||
}
|
|
||||||
if (version > 6) {
|
|
||||||
strip.setColor(2, RGBW32(udpIn[20], udpIn[21], udpIn[22], udpIn[23])); // tertiary color
|
|
||||||
if (version > 9 && version < 200 && udpIn[37] < 255) { // valid CCT/Kelvin value
|
|
||||||
uint16_t cct = udpIn[38];
|
|
||||||
if (udpIn[37] > 0) { //Kelvin
|
|
||||||
cct |= (udpIn[37] << 8);
|
|
||||||
}
|
|
||||||
strip.setCCT(cct);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool timebaseUpdated = false;
|
|
||||||
//apply effects from notification
|
|
||||||
bool applyEffects = (receiveNotificationEffects || !someSel);
|
|
||||||
if (version < 200)
|
|
||||||
{
|
|
||||||
if (applyEffects && currentPlaylist >= 0) unloadPlaylist();
|
|
||||||
if (version > 10 && (receiveSegmentOptions || receiveSegmentBounds)) {
|
|
||||||
uint8_t numSrcSegs = udpIn[39];
|
|
||||||
for (size_t i = 0; i < numSrcSegs; i++) {
|
|
||||||
uint16_t ofs = 41 + i*udpIn[40]; //start of segment offset byte
|
|
||||||
uint8_t id = udpIn[0 +ofs];
|
|
||||||
if (id > strip.getSegmentsNum()) break;
|
|
||||||
|
|
||||||
Segment& selseg = strip.getSegment(id);
|
|
||||||
if (!selseg.isActive() || !selseg.isSelected()) continue; //do not apply to non selected segments
|
|
||||||
|
|
||||||
uint16_t startY = 0, start = (udpIn[1+ofs] << 8 | udpIn[2+ofs]);
|
|
||||||
uint16_t stopY = 1, stop = (udpIn[3+ofs] << 8 | udpIn[4+ofs]);
|
|
||||||
uint16_t offset = (udpIn[7+ofs] << 8 | udpIn[8+ofs]);
|
|
||||||
if (!receiveSegmentOptions) {
|
|
||||||
selseg.setUp(start, stop, selseg.grouping, selseg.spacing, offset, startY, stopY);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
//for (size_t j = 1; j<4; j++) selseg.setOption(j, (udpIn[9 +ofs] >> j) & 0x01); //only take into account mirrored, on, reversed; ignore selected
|
|
||||||
selseg.options = (selseg.options & 0x0071U) | (udpIn[9 +ofs] & 0x0E); // ignore selected, freeze, reset & transitional
|
|
||||||
selseg.setOpacity(udpIn[10+ofs]);
|
|
||||||
if (applyEffects) {
|
|
||||||
strip.setMode(id, udpIn[11+ofs]);
|
|
||||||
selseg.speed = udpIn[12+ofs];
|
|
||||||
selseg.intensity = udpIn[13+ofs];
|
|
||||||
selseg.palette = udpIn[14+ofs];
|
|
||||||
}
|
|
||||||
if (receiveNotificationColor || !someSel) {
|
|
||||||
selseg.setColor(0, RGBW32(udpIn[15+ofs],udpIn[16+ofs],udpIn[17+ofs],udpIn[18+ofs]));
|
|
||||||
selseg.setColor(1, RGBW32(udpIn[19+ofs],udpIn[20+ofs],udpIn[21+ofs],udpIn[22+ofs]));
|
|
||||||
selseg.setColor(2, RGBW32(udpIn[23+ofs],udpIn[24+ofs],udpIn[25+ofs],udpIn[26+ofs]));
|
|
||||||
selseg.setCCT(udpIn[27+ofs]);
|
|
||||||
}
|
|
||||||
if (version > 11) {
|
|
||||||
// when applying synced options ignore selected as it may be used as indicator of which segments to sync
|
|
||||||
// freeze, reset & transitional should never be synced
|
|
||||||
selseg.options = (selseg.options & 0x0071U) | (udpIn[28+ofs]<<8) | (udpIn[9 +ofs] & 0x8E); // ignore selected, freeze, reset & transitional
|
|
||||||
if (applyEffects) {
|
|
||||||
selseg.custom1 = udpIn[29+ofs];
|
|
||||||
selseg.custom2 = udpIn[30+ofs];
|
|
||||||
selseg.custom3 = udpIn[31+ofs] & 0x1F;
|
|
||||||
selseg.check1 = (udpIn[31+ofs]>>5) & 0x1;
|
|
||||||
selseg.check1 = (udpIn[31+ofs]>>6) & 0x1;
|
|
||||||
selseg.check1 = (udpIn[31+ofs]>>7) & 0x1;
|
|
||||||
}
|
|
||||||
startY = (udpIn[32+ofs] << 8 | udpIn[33+ofs]);
|
|
||||||
stopY = (udpIn[34+ofs] << 8 | udpIn[35+ofs]);
|
|
||||||
}
|
|
||||||
if (receiveSegmentBounds) {
|
|
||||||
selseg.setUp(start, stop, udpIn[5+ofs], udpIn[6+ofs], offset, startY, stopY);
|
|
||||||
} else {
|
|
||||||
selseg.setUp(selseg.start, selseg.stop, udpIn[5+ofs], udpIn[6+ofs], selseg.offset, selseg.startY, selseg.stopY);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
stateChanged = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// simple effect sync, applies to all selected segments
|
|
||||||
if (applyEffects && (version < 11 || !receiveSegmentOptions)) {
|
|
||||||
for (size_t i = 0; i < strip.getSegmentsNum(); i++) {
|
|
||||||
Segment& seg = strip.getSegment(i);
|
|
||||||
if (!seg.isActive() || !seg.isSelected()) continue;
|
|
||||||
seg.setMode(udpIn[8]);
|
|
||||||
seg.speed = udpIn[9];
|
|
||||||
if (version > 2) seg.intensity = udpIn[16];
|
|
||||||
if (version > 4) seg.setPalette(udpIn[19]);
|
|
||||||
}
|
|
||||||
stateChanged = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (applyEffects && version > 5) {
|
|
||||||
uint32_t t = (udpIn[25] << 24) | (udpIn[26] << 16) | (udpIn[27] << 8) | (udpIn[28]);
|
|
||||||
t += PRESUMED_NETWORK_DELAY; //adjust trivially for network delay
|
|
||||||
t -= millis();
|
|
||||||
strip.timebase = t;
|
|
||||||
timebaseUpdated = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//adjust system time, but only if sender is more accurate than self
|
|
||||||
if (version > 7 && version < 200)
|
|
||||||
{
|
|
||||||
Toki::Time tm;
|
|
||||||
tm.sec = (udpIn[30] << 24) | (udpIn[31] << 16) | (udpIn[32] << 8) | (udpIn[33]);
|
|
||||||
tm.ms = (udpIn[34] << 8) | (udpIn[35]);
|
|
||||||
if (udpIn[29] > toki.getTimeSource()) { //if sender's time source is more accurate
|
|
||||||
toki.adjust(tm, PRESUMED_NETWORK_DELAY); //adjust trivially for network delay
|
|
||||||
uint8_t ts = TOKI_TS_UDP;
|
|
||||||
if (udpIn[29] > 99) ts = TOKI_TS_UDP_NTP;
|
|
||||||
else if (udpIn[29] >= TOKI_TS_SEC) ts = TOKI_TS_UDP_SEC;
|
|
||||||
toki.setTime(tm, ts);
|
|
||||||
} else if (timebaseUpdated && toki.getTimeSource() > 99) { //if we both have good times, get a more accurate timebase
|
|
||||||
Toki::Time myTime = toki.getTime();
|
|
||||||
uint32_t diff = toki.msDifference(tm, myTime);
|
|
||||||
strip.timebase -= PRESUMED_NETWORK_DELAY; //no need to presume, use difference between NTP times at send and receive points
|
|
||||||
if (toki.isLater(tm, myTime)) {
|
|
||||||
strip.timebase += diff;
|
|
||||||
} else {
|
|
||||||
strip.timebase -= diff;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (version > 3)
|
|
||||||
{
|
|
||||||
transitionDelayTemp = ((udpIn[17] << 0) & 0xFF) + ((udpIn[18] << 8) & 0xFF00);
|
|
||||||
}
|
|
||||||
|
|
||||||
nightlightActive = udpIn[6];
|
|
||||||
if (nightlightActive) nightlightDelayMins = udpIn[7];
|
|
||||||
|
|
||||||
if (receiveNotificationBrightness || !someSel) bri = udpIn[2];
|
|
||||||
stateUpdated(CALL_MODE_NOTIFICATION);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -739,7 +812,7 @@ uint8_t realtimeBroadcast(uint8_t type, IPAddress client, uint16_t length, uint8
|
|||||||
if (sequenceNumber > 15) sequenceNumber = 0;
|
if (sequenceNumber > 15) sequenceNumber = 0;
|
||||||
|
|
||||||
if (!ddpUdp.beginPacket(client, DDP_DEFAULT_PORT)) { // port defined in ESPAsyncE131.h
|
if (!ddpUdp.beginPacket(client, DDP_DEFAULT_PORT)) { // port defined in ESPAsyncE131.h
|
||||||
DEBUG_PRINTLN(F("WiFiUDP.beginPacket returned an error"));
|
//DEBUG_PRINTLN(F("WiFiUDP.beginPacket returned an error"));
|
||||||
return 1; // problem
|
return 1; // problem
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -780,7 +853,7 @@ uint8_t realtimeBroadcast(uint8_t type, IPAddress client, uint16_t length, uint8
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!ddpUdp.endPacket()) {
|
if (!ddpUdp.endPacket()) {
|
||||||
DEBUG_PRINTLN(F("WiFiUDP.endPacket returned an error"));
|
//DEBUG_PRINTLN(F("WiFiUDP.endPacket returned an error"));
|
||||||
return 1; // problem
|
return 1; // problem
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -849,3 +922,78 @@ uint8_t realtimeBroadcast(uint8_t type, IPAddress client, uint16_t length, uint8
|
|||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
|
// ESP-NOW message receive callback function
|
||||||
|
void espNowReceiveCB(uint8_t* address, uint8_t* data, uint8_t len, signed int rssi, bool broadcast) {
|
||||||
|
sprintf_P(last_signal_src, PSTR("%02x%02x%02x%02x%02x%02x"), address[0], address[1], address[2], address[3], address[4], address[5]);
|
||||||
|
|
||||||
|
#ifdef WLED_DEBUG
|
||||||
|
DEBUG_PRINT(F("ESP-NOW: ")); DEBUG_PRINT(last_signal_src); DEBUG_PRINT(F(" -> ")); DEBUG_PRINTLN(len);
|
||||||
|
for (int i=0; i<len; i++) DEBUG_PRINTF("%02x ", data[i]);
|
||||||
|
DEBUG_PRINTLN();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// handle WiZ Mote data
|
||||||
|
if (data[0] == 0x91 || data[0] == 0x81 || data[0] == 0x80) {
|
||||||
|
handleRemote(data, len);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strlen(linked_remote) == 12 && strcmp(last_signal_src, linked_remote) != 0) {
|
||||||
|
DEBUG_PRINTLN(F("ESP-NOW unpaired remote sender."));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
partial_packet_t *buffer = reinterpret_cast<partial_packet_t *>(data);
|
||||||
|
if (len < 3 || !broadcast || buffer->magic != 'W' || !useESPNowSync || WLED_CONNECTED) {
|
||||||
|
DEBUG_PRINTLN(F("ESP-NOW unexpected packet, not syncing or connected to WiFi."));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint8_t *udpIn = nullptr;
|
||||||
|
static uint8_t packetsReceived = 0; // bitfield (max 5 packets ATM)
|
||||||
|
static uint8_t segsReceived = 0;
|
||||||
|
static unsigned long lastProcessed = 0;
|
||||||
|
|
||||||
|
if (buffer->packet == 0) {
|
||||||
|
if (udpIn == nullptr) udpIn = (uint8_t *)malloc(WLEDPACKETSIZE); // we cannot use stack as we are in callback
|
||||||
|
DEBUG_PRINTLN(F("ESP-NOW inited UDP buffer."));
|
||||||
|
memcpy(udpIn, buffer->data, len-3); // global data (41 bytes)
|
||||||
|
packetsReceived |= 0x01 << buffer->packet;
|
||||||
|
segsReceived = 0;
|
||||||
|
return;
|
||||||
|
} else if (((len-3)/UDP_SEG_SIZE)*UDP_SEG_SIZE != (len-3)) {
|
||||||
|
DEBUG_PRINTF("ESP-NOW incorrect packet size: %d (%d) [%d]\n", (int)buffer->packet, (int)len-3, (int)UDP_SEG_SIZE);
|
||||||
|
if (udpIn) free(udpIn);
|
||||||
|
udpIn = nullptr;
|
||||||
|
packetsReceived = 0;
|
||||||
|
segsReceived = 0;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!udpIn) return;
|
||||||
|
|
||||||
|
// TODO add verification if segsReceived > MAX_NUM_SEGMENTS or WLEDPACKETSIZE
|
||||||
|
|
||||||
|
memcpy(udpIn+41+segsReceived, buffer->data, len-3);
|
||||||
|
packetsReceived |= 0x01 << buffer->packet;
|
||||||
|
segsReceived += (len-3)/UDP_SEG_SIZE;
|
||||||
|
|
||||||
|
DEBUG_PRINTF("ESP-NOW packet received: %d (%d) [%d]\n", (int)buffer->packet, (int)len-3, (int)segsReceived);
|
||||||
|
|
||||||
|
if (segsReceived == buffer->segs) {
|
||||||
|
// last packet received
|
||||||
|
if (millis() - lastProcessed > 250) {
|
||||||
|
DEBUG_PRINTLN(F("ESP-NOW processing complete message."));
|
||||||
|
parseNotifyPacket(udpIn);
|
||||||
|
lastProcessed = millis();
|
||||||
|
} else {
|
||||||
|
DEBUG_PRINTLN(F("ESP-NOW ignoring complete message."));
|
||||||
|
}
|
||||||
|
free(udpIn);
|
||||||
|
udpIn = nullptr;
|
||||||
|
packetsReceived = 0;
|
||||||
|
segsReceived = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
@ -416,9 +416,9 @@ uint16_t crc16(const unsigned char* data_p, size_t length) {
|
|||||||
// (only 2 used as stored in 1 bit in segment options, consider switching to a single global simulation type)
|
// (only 2 used as stored in 1 bit in segment options, consider switching to a single global simulation type)
|
||||||
typedef enum UM_SoundSimulations {
|
typedef enum UM_SoundSimulations {
|
||||||
UMS_BeatSin = 0,
|
UMS_BeatSin = 0,
|
||||||
UMS_WeWillRockYou
|
UMS_WeWillRockYou,
|
||||||
//UMS_10_13,
|
UMS_10_13,
|
||||||
//UMS_14_3
|
UMS_14_3
|
||||||
} um_soundSimulations_t;
|
} um_soundSimulations_t;
|
||||||
|
|
||||||
um_data_t* simulateSound(uint8_t simulationId)
|
um_data_t* simulateSound(uint8_t simulationId)
|
||||||
@ -503,7 +503,7 @@ um_data_t* simulateSound(uint8_t simulationId)
|
|||||||
fftResult[i] = 0;
|
fftResult[i] = 0;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
/*case UMS_10_3:
|
case UMS_10_13:
|
||||||
for (int i = 0; i<16; i++)
|
for (int i = 0; i<16; i++)
|
||||||
fftResult[i] = inoise8(beatsin8(90 / (i+1), 0, 200)*15 + (ms>>10), ms>>3);
|
fftResult[i] = inoise8(beatsin8(90 / (i+1), 0, 200)*15 + (ms>>10), ms>>3);
|
||||||
volumeSmth = fftResult[8];
|
volumeSmth = fftResult[8];
|
||||||
@ -512,7 +512,7 @@ um_data_t* simulateSound(uint8_t simulationId)
|
|||||||
for (int i = 0; i<16; i++)
|
for (int i = 0; i<16; i++)
|
||||||
fftResult[i] = inoise8(beatsin8(120 / (i+1), 10, 30)*10 + (ms>>14), ms>>3);
|
fftResult[i] = inoise8(beatsin8(120 / (i+1), 10, 30)*10 + (ms>>14), ms>>3);
|
||||||
volumeSmth = fftResult[8];
|
volumeSmth = fftResult[8];
|
||||||
break;*/
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
samplePeak = random8() > 250;
|
samplePeak = random8() > 250;
|
||||||
@ -573,3 +573,17 @@ void enumerateLedmaps() {
|
|||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Returns a new, random color wheel index with a minimum distance of 42 from pos.
|
||||||
|
*/
|
||||||
|
uint8_t get_random_wheel_index(uint8_t pos) {
|
||||||
|
uint8_t r = 0, x = 0, y = 0, d = 0;
|
||||||
|
while (d < 42) {
|
||||||
|
r = random8();
|
||||||
|
x = abs(pos - r);
|
||||||
|
y = 255 - x;
|
||||||
|
d = MIN(x, y);
|
||||||
|
}
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
@ -54,9 +54,6 @@ void WLED::loop()
|
|||||||
handleIR(); // 2nd call to function needed for ESP32 to return valid results -- should be good for ESP8266, too
|
handleIR(); // 2nd call to function needed for ESP32 to return valid results -- should be good for ESP8266, too
|
||||||
#endif
|
#endif
|
||||||
handleConnection();
|
handleConnection();
|
||||||
#ifndef WLED_DISABLE_ESPNOW
|
|
||||||
handleRemote();
|
|
||||||
#endif
|
|
||||||
handleSerial();
|
handleSerial();
|
||||||
handleImprovWifiScan();
|
handleImprovWifiScan();
|
||||||
handleNotifications();
|
handleNotifications();
|
||||||
@ -676,6 +673,14 @@ void WLED::initConnection()
|
|||||||
ws.onEvent(wsEvent);
|
ws.onEvent(wsEvent);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
|
if (statusESPNow == ESP_NOW_STATE_ON) {
|
||||||
|
DEBUG_PRINTLN(F("ESP-NOW stopping."));
|
||||||
|
quickEspNow.stop();
|
||||||
|
statusESPNow = ESP_NOW_STATE_UNINIT;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
WiFi.disconnect(true); // close old connections
|
WiFi.disconnect(true); // close old connections
|
||||||
#ifdef ESP8266
|
#ifdef ESP8266
|
||||||
WiFi.setPhyMode(WIFI_PHY_MODE_11N);
|
WiFi.setPhyMode(WIFI_PHY_MODE_11N);
|
||||||
@ -692,7 +697,6 @@ void WLED::initConnection()
|
|||||||
if (!WLED_WIFI_CONFIGURED) {
|
if (!WLED_WIFI_CONFIGURED) {
|
||||||
DEBUG_PRINTLN(F("No connection configured."));
|
DEBUG_PRINTLN(F("No connection configured."));
|
||||||
if (!apActive) initAP(); // instantly go to ap mode
|
if (!apActive) initAP(); // instantly go to ap mode
|
||||||
return;
|
|
||||||
} else if (!apActive) {
|
} else if (!apActive) {
|
||||||
if (apBehavior == AP_BEHAVIOR_ALWAYS) {
|
if (apBehavior == AP_BEHAVIOR_ALWAYS) {
|
||||||
DEBUG_PRINTLN(F("Access point ALWAYS enabled."));
|
DEBUG_PRINTLN(F("Access point ALWAYS enabled."));
|
||||||
@ -705,27 +709,43 @@ void WLED::initConnection()
|
|||||||
}
|
}
|
||||||
showWelcomePage = false;
|
showWelcomePage = false;
|
||||||
|
|
||||||
DEBUG_PRINT(F("Connecting to "));
|
if (WLED_WIFI_CONFIGURED) {
|
||||||
DEBUG_PRINT(clientSSID);
|
DEBUG_PRINT(F("Connecting to "));
|
||||||
DEBUG_PRINTLN("...");
|
DEBUG_PRINT(clientSSID);
|
||||||
|
DEBUG_PRINTLN("...");
|
||||||
|
|
||||||
// convert the "serverDescription" into a valid DNS hostname (alphanumeric)
|
// convert the "serverDescription" into a valid DNS hostname (alphanumeric)
|
||||||
char hostname[25];
|
char hostname[25];
|
||||||
prepareHostname(hostname);
|
prepareHostname(hostname);
|
||||||
|
WiFi.begin(clientSSID, clientPass);
|
||||||
#ifdef ESP8266
|
|
||||||
WiFi.hostname(hostname);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
WiFi.begin(clientSSID, clientPass);
|
|
||||||
#ifdef ARDUINO_ARCH_ESP32
|
#ifdef ARDUINO_ARCH_ESP32
|
||||||
#if defined(LOLIN_WIFI_FIX) && (defined(ARDUINO_ARCH_ESP32C3) || defined(ARDUINO_ARCH_ESP32S2) || defined(ARDUINO_ARCH_ESP32S3))
|
#if defined(LOLIN_WIFI_FIX) && (defined(ARDUINO_ARCH_ESP32C3) || defined(ARDUINO_ARCH_ESP32S2) || defined(ARDUINO_ARCH_ESP32S3))
|
||||||
WiFi.setTxPower(WIFI_POWER_8_5dBm);
|
WiFi.setTxPower(WIFI_POWER_8_5dBm);
|
||||||
#endif
|
#endif
|
||||||
WiFi.setSleep(!noWifiSleep);
|
WiFi.setSleep(!noWifiSleep);
|
||||||
WiFi.setHostname(hostname);
|
WiFi.setHostname(hostname);
|
||||||
#else
|
#else
|
||||||
wifi_set_sleep_type((noWifiSleep) ? NONE_SLEEP_T : MODEM_SLEEP_T);
|
wifi_set_sleep_type((noWifiSleep) ? NONE_SLEEP_T : MODEM_SLEEP_T);
|
||||||
|
WiFi.hostname(hostname);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
|
if (enableESPNow) {
|
||||||
|
quickEspNow.onDataRcvd(espNowReceiveCB);
|
||||||
|
bool espNowOK;
|
||||||
|
if (apActive) {
|
||||||
|
DEBUG_PRINTLN(F("ESP-NOW initing in AP mode."));
|
||||||
|
#ifdef ESP32
|
||||||
|
quickEspNow.setWiFiBandwidth(WIFI_IF_AP, WIFI_BW_HT20); // Only needed for ESP32 in case you need coexistence with ESP8266 in the same network
|
||||||
|
#endif //ESP32
|
||||||
|
espNowOK = quickEspNow.begin(apChannel, WIFI_IF_AP); // Same channel must be used for both AP and ESP-NOW
|
||||||
|
} else {
|
||||||
|
DEBUG_PRINTLN(F("ESP-NOW initing in STA mode."));
|
||||||
|
espNowOK = quickEspNow.begin(); // Use no parameters to start ESP-NOW on same channel as WiFi, in STA mode
|
||||||
|
}
|
||||||
|
statusESPNow = espNowOK ? ESP_NOW_STATE_ON : ESP_NOW_STATE_ERROR;
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -850,8 +870,8 @@ void WLED::handleConnection()
|
|||||||
if (!Network.isConnected()) {
|
if (!Network.isConnected()) {
|
||||||
if (interfacesInited) {
|
if (interfacesInited) {
|
||||||
DEBUG_PRINTLN(F("Disconnected!"));
|
DEBUG_PRINTLN(F("Disconnected!"));
|
||||||
interfacesInited = false;
|
|
||||||
initConnection();
|
initConnection();
|
||||||
|
interfacesInited = false;
|
||||||
}
|
}
|
||||||
//send improv failed 6 seconds after second init attempt (24 sec. after provisioning)
|
//send improv failed 6 seconds after second init attempt (24 sec. after provisioning)
|
||||||
if (improvActive > 2 && now - lastReconnectAttempt > 6000) {
|
if (improvActive > 2 && now - lastReconnectAttempt > 6000) {
|
||||||
|
@ -3,12 +3,12 @@
|
|||||||
/*
|
/*
|
||||||
Main sketch, global variable declarations
|
Main sketch, global variable declarations
|
||||||
@title WLED project sketch
|
@title WLED project sketch
|
||||||
@version 0.14.0-b4
|
@version 0.15.0-a0
|
||||||
@author Christian Schwinne
|
@author Christian Schwinne
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// version code in format yymmddb (b = daily build)
|
// version code in format yymmddb (b = daily build)
|
||||||
#define VERSION 2310130
|
#define VERSION 2310180
|
||||||
|
|
||||||
//uncomment this if you have a "my_config.h" file you'd like to use
|
//uncomment this if you have a "my_config.h" file you'd like to use
|
||||||
//#define WLED_USE_MY_CONFIG
|
//#define WLED_USE_MY_CONFIG
|
||||||
@ -77,6 +77,9 @@
|
|||||||
}
|
}
|
||||||
#ifndef WLED_DISABLE_ESPNOW
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
#include <espnow.h>
|
#include <espnow.h>
|
||||||
|
#define WIFI_MODE_STA WIFI_STA
|
||||||
|
#define WIFI_MODE_AP WIFI_AP
|
||||||
|
#include <QuickEspNow.h>
|
||||||
#endif
|
#endif
|
||||||
#else // ESP32
|
#else // ESP32
|
||||||
#include <HardwareSerial.h> // ensure we have the correct "Serial" on new MCUs (depends on ARDUINO_USB_MODE and ARDUINO_USB_CDC_ON_BOOT)
|
#include <HardwareSerial.h> // ensure we have the correct "Serial" on new MCUs (depends on ARDUINO_USB_MODE and ARDUINO_USB_CDC_ON_BOOT)
|
||||||
@ -97,6 +100,7 @@
|
|||||||
|
|
||||||
#ifndef WLED_DISABLE_ESPNOW
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
#include <esp_now.h>
|
#include <esp_now.h>
|
||||||
|
#include <QuickEspNow.h>
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
#include <Wire.h>
|
#include <Wire.h>
|
||||||
@ -256,7 +260,7 @@ using PSRAMDynamicJsonDocument = BasicJsonDocument<PSRAM_Allocator>;
|
|||||||
|
|
||||||
// Global Variable definitions
|
// Global Variable definitions
|
||||||
WLED_GLOBAL char versionString[] _INIT(TOSTRING(WLED_VERSION));
|
WLED_GLOBAL char versionString[] _INIT(TOSTRING(WLED_VERSION));
|
||||||
#define WLED_CODENAME "Hoshi"
|
#define WLED_CODENAME "Kōsen"
|
||||||
|
|
||||||
// AP and OTA default passwords (for maximum security change them!)
|
// AP and OTA default passwords (for maximum security change them!)
|
||||||
WLED_GLOBAL char apPass[65] _INIT(WLED_AP_PASS);
|
WLED_GLOBAL char apPass[65] _INIT(WLED_AP_PASS);
|
||||||
@ -351,6 +355,7 @@ WLED_GLOBAL byte nightlightTargetBri _INIT(0); // brightness after nightlig
|
|||||||
WLED_GLOBAL byte nightlightDelayMins _INIT(60);
|
WLED_GLOBAL byte nightlightDelayMins _INIT(60);
|
||||||
WLED_GLOBAL byte nightlightMode _INIT(NL_MODE_FADE); // See const.h for available modes. Was nightlightFade
|
WLED_GLOBAL byte nightlightMode _INIT(NL_MODE_FADE); // See const.h for available modes. Was nightlightFade
|
||||||
WLED_GLOBAL bool fadeTransition _INIT(true); // enable crossfading color transition
|
WLED_GLOBAL bool fadeTransition _INIT(true); // enable crossfading color transition
|
||||||
|
WLED_GLOBAL bool modeBlending _INIT(true); // enable effect blending
|
||||||
WLED_GLOBAL uint16_t transitionDelay _INIT(750); // default crossfade duration in ms
|
WLED_GLOBAL uint16_t transitionDelay _INIT(750); // default crossfade duration in ms
|
||||||
|
|
||||||
WLED_GLOBAL byte briMultiplier _INIT(100); // % of brightness to set (to limit power, if you set it to 50 and set bri to 255, actual brightness will be 127)
|
WLED_GLOBAL byte briMultiplier _INIT(100); // % of brightness to set (to limit power, if you set it to 50 and set bri to 255, actual brightness will be 127)
|
||||||
@ -361,7 +366,7 @@ WLED_GLOBAL char serverDescription[33] _INIT("WLED"); // Name of module - use d
|
|||||||
#else
|
#else
|
||||||
WLED_GLOBAL char serverDescription[33] _INIT(SERVERNAME); // use predefined name
|
WLED_GLOBAL char serverDescription[33] _INIT(SERVERNAME); // use predefined name
|
||||||
#endif
|
#endif
|
||||||
WLED_GLOBAL bool syncToggleReceive _INIT(false); // UIs which only have a single button for sync should toggle send+receive if this is true, only send otherwise
|
//WLED_GLOBAL bool syncToggleReceive _INIT(false); // UIs which only have a single button for sync should toggle send+receive if this is true, only send otherwise
|
||||||
WLED_GLOBAL bool simplifiedUI _INIT(false); // enable simplified UI
|
WLED_GLOBAL bool simplifiedUI _INIT(false); // enable simplified UI
|
||||||
WLED_GLOBAL byte cacheInvalidate _INIT(0); // used to invalidate browser cache when switching from regular to simplified UI
|
WLED_GLOBAL byte cacheInvalidate _INIT(0); // used to invalidate browser cache when switching from regular to simplified UI
|
||||||
|
|
||||||
@ -402,7 +407,7 @@ WLED_GLOBAL byte alexaNumPresets _INIT(0); // number of p
|
|||||||
|
|
||||||
WLED_GLOBAL uint16_t realtimeTimeoutMs _INIT(2500); // ms timeout of realtime mode before returning to normal mode
|
WLED_GLOBAL uint16_t realtimeTimeoutMs _INIT(2500); // ms timeout of realtime mode before returning to normal mode
|
||||||
WLED_GLOBAL int arlsOffset _INIT(0); // realtime LED offset
|
WLED_GLOBAL int arlsOffset _INIT(0); // realtime LED offset
|
||||||
WLED_GLOBAL bool receiveDirect _INIT(true); // receive UDP realtime
|
WLED_GLOBAL bool receiveDirect _INIT(true); // receive UDP/Hyperion realtime
|
||||||
WLED_GLOBAL bool arlsDisableGammaCorrection _INIT(true); // activate if gamma correction is handled by the source
|
WLED_GLOBAL bool arlsDisableGammaCorrection _INIT(true); // activate if gamma correction is handled by the source
|
||||||
WLED_GLOBAL bool arlsForceMaxBri _INIT(false); // enable to force max brightness if source has very dark colors that would be black
|
WLED_GLOBAL bool arlsForceMaxBri _INIT(false); // enable to force max brightness if source has very dark colors that would be black
|
||||||
|
|
||||||
@ -465,9 +470,11 @@ WLED_GLOBAL bool hueApplyColor _INIT(true);
|
|||||||
WLED_GLOBAL uint16_t serialBaud _INIT(1152); // serial baud rate, multiply by 100
|
WLED_GLOBAL uint16_t serialBaud _INIT(1152); // serial baud rate, multiply by 100
|
||||||
|
|
||||||
#ifndef WLED_DISABLE_ESPNOW
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
WLED_GLOBAL bool enable_espnow_remote _INIT(false);
|
WLED_GLOBAL bool enableESPNow _INIT(false); // global on/off for ESP-NOW
|
||||||
WLED_GLOBAL char linked_remote[13] _INIT("");
|
WLED_GLOBAL byte statusESPNow _INIT(ESP_NOW_STATE_UNINIT); // state of ESP-NOW stack (0 uninitialised, 1 initialised, 2 error)
|
||||||
WLED_GLOBAL char last_signal_src[13] _INIT("");
|
WLED_GLOBAL bool useESPNowSync _INIT(false); // use ESP-NOW wireless technology for sync
|
||||||
|
WLED_GLOBAL char linked_remote[13] _INIT(""); // MAC of ESP-NOW remote (Wiz Mote)
|
||||||
|
WLED_GLOBAL char last_signal_src[13] _INIT(""); // last seen ESP-NOW sender
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Time CONFIG
|
// Time CONFIG
|
||||||
@ -563,8 +570,8 @@ WLED_GLOBAL bool disablePullUp _INIT(false);
|
|||||||
WLED_GLOBAL byte touchThreshold _INIT(TOUCH_THRESHOLD);
|
WLED_GLOBAL byte touchThreshold _INIT(TOUCH_THRESHOLD);
|
||||||
|
|
||||||
// notifications
|
// notifications
|
||||||
WLED_GLOBAL bool notifyDirectDefault _INIT(notifyDirect);
|
WLED_GLOBAL bool sendNotifications _INIT(false); // master notification switch
|
||||||
WLED_GLOBAL bool receiveNotifications _INIT(true);
|
WLED_GLOBAL bool sendNotificationsRT _INIT(false); // master notification switch (runtime)
|
||||||
WLED_GLOBAL unsigned long notificationSentTime _INIT(0);
|
WLED_GLOBAL unsigned long notificationSentTime _INIT(0);
|
||||||
WLED_GLOBAL byte notificationSentCallMode _INIT(CALL_MODE_INIT);
|
WLED_GLOBAL byte notificationSentCallMode _INIT(CALL_MODE_INIT);
|
||||||
WLED_GLOBAL uint8_t notificationCount _INIT(0);
|
WLED_GLOBAL uint8_t notificationCount _INIT(0);
|
||||||
|
@ -83,8 +83,8 @@ void loadSettingsFromEEPROM()
|
|||||||
nightlightDelayMinsDefault = EEPROM.read(224);
|
nightlightDelayMinsDefault = EEPROM.read(224);
|
||||||
nightlightDelayMins = nightlightDelayMinsDefault;
|
nightlightDelayMins = nightlightDelayMinsDefault;
|
||||||
nightlightMode = EEPROM.read(225);
|
nightlightMode = EEPROM.read(225);
|
||||||
notifyDirectDefault = EEPROM.read(226);
|
notifyDirect = EEPROM.read(226);
|
||||||
notifyDirect = notifyDirectDefault;
|
sendNotificationsRT = notifyDirect;
|
||||||
|
|
||||||
apChannel = EEPROM.read(227);
|
apChannel = EEPROM.read(227);
|
||||||
if (apChannel > 13 || apChannel < 1) apChannel = 1;
|
if (apChannel > 13 || apChannel < 1) apChannel = 1;
|
||||||
@ -163,7 +163,6 @@ void loadSettingsFromEEPROM()
|
|||||||
receiveNotificationColor = EEPROM.read(391);
|
receiveNotificationColor = EEPROM.read(391);
|
||||||
receiveNotificationEffects = EEPROM.read(392);
|
receiveNotificationEffects = EEPROM.read(392);
|
||||||
}
|
}
|
||||||
receiveNotifications = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects);
|
|
||||||
|
|
||||||
if (lastEEPROMversion > 4) {
|
if (lastEEPROMversion > 4) {
|
||||||
#ifndef WLED_DISABLE_HUESYNC
|
#ifndef WLED_DISABLE_HUESYNC
|
||||||
@ -278,10 +277,10 @@ void loadSettingsFromEEPROM()
|
|||||||
if (lastEEPROMversion > 13)
|
if (lastEEPROMversion > 13)
|
||||||
{
|
{
|
||||||
mqttEnabled = EEPROM.read(2299);
|
mqttEnabled = EEPROM.read(2299);
|
||||||
syncToggleReceive = EEPROM.read(397);
|
//syncToggleReceive = EEPROM.read(397);
|
||||||
} else {
|
} else {
|
||||||
mqttEnabled = true;
|
mqttEnabled = true;
|
||||||
syncToggleReceive = false;
|
//syncToggleReceive = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (lastEEPROMversion > 14)
|
if (lastEEPROMversion > 14)
|
||||||
|
@ -40,7 +40,7 @@ bool isIp(String str) {
|
|||||||
|
|
||||||
void handleUpload(AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final) {
|
void handleUpload(AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final) {
|
||||||
if (!correctPIN) {
|
if (!correctPIN) {
|
||||||
if (final) request->send(500, "text/plain", FPSTR(s_unlock_cfg));
|
if (final) request->send(401, "text/plain", FPSTR(s_unlock_cfg));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!index) {
|
if (!index) {
|
||||||
@ -86,7 +86,7 @@ void createEditHandler(bool enable) {
|
|||||||
#endif
|
#endif
|
||||||
} else {
|
} else {
|
||||||
editHandler = &server.on("/edit", HTTP_ANY, [](AsyncWebServerRequest *request){
|
editHandler = &server.on("/edit", HTTP_ANY, [](AsyncWebServerRequest *request){
|
||||||
serveMessage(request, 500, "Access Denied", FPSTR(s_unlock_cfg), 254);
|
serveMessage(request, 401, "Access Denied", FPSTR(s_unlock_cfg), 254);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -201,7 +201,7 @@ void initServer()
|
|||||||
verboseResponse = deserializeState(root);
|
verboseResponse = deserializeState(root);
|
||||||
} else {
|
} else {
|
||||||
if (!correctPIN && strlen(settingsPIN)>0) {
|
if (!correctPIN && strlen(settingsPIN)>0) {
|
||||||
request->send(403, "application/json", F("{\"error\":1}")); // ERR_DENIED
|
request->send(401, "application/json", F("{\"error\":1}")); // ERR_DENIED
|
||||||
releaseJSONBufferLock();
|
releaseJSONBufferLock();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -211,6 +211,8 @@ void initServer()
|
|||||||
|
|
||||||
if (verboseResponse) {
|
if (verboseResponse) {
|
||||||
if (!isConfig) {
|
if (!isConfig) {
|
||||||
|
lastInterfaceUpdate = millis(); // prevent WS update until cooldown
|
||||||
|
interfaceUpdateCallMode = CALL_MODE_WS_SEND; // schedule WS update
|
||||||
serveJson(request); return; //if JSON contains "v"
|
serveJson(request); return; //if JSON contains "v"
|
||||||
} else {
|
} else {
|
||||||
doSerializeConfig = true; //serializeConfig(); //Save new settings to FS
|
doSerializeConfig = true; //serializeConfig(); //Save new settings to FS
|
||||||
@ -282,7 +284,7 @@ void initServer()
|
|||||||
//init ota page
|
//init ota page
|
||||||
server.on("/update", HTTP_GET, [](AsyncWebServerRequest *request){
|
server.on("/update", HTTP_GET, [](AsyncWebServerRequest *request){
|
||||||
if (otaLock) {
|
if (otaLock) {
|
||||||
serveMessage(request, 500, "Access Denied", FPSTR(s_unlock_ota), 254);
|
serveMessage(request, 401, "Access Denied", FPSTR(s_unlock_ota), 254);
|
||||||
} else
|
} else
|
||||||
serveSettings(request); // checks for "upd" in URL and handles PIN
|
serveSettings(request); // checks for "upd" in URL and handles PIN
|
||||||
});
|
});
|
||||||
@ -292,7 +294,11 @@ void initServer()
|
|||||||
serveSettings(request, true); // handle PIN page POST request
|
serveSettings(request, true); // handle PIN page POST request
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (Update.hasError() || otaLock) {
|
if (otaLock) {
|
||||||
|
serveMessage(request, 401, "Access Denied", FPSTR(s_unlock_ota), 254);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (Update.hasError()) {
|
||||||
serveMessage(request, 500, F("Update failed!"), F("Please check your file and retry!"), 254);
|
serveMessage(request, 500, F("Update failed!"), F("Please check your file and retry!"), 254);
|
||||||
} else {
|
} else {
|
||||||
serveMessage(request, 200, F("Update successful!"), F("Rebooting..."), 131);
|
serveMessage(request, 200, F("Update successful!"), F("Rebooting..."), 131);
|
||||||
@ -533,7 +539,7 @@ void serveSettingsJS(AsyncWebServerRequest* request)
|
|||||||
}
|
}
|
||||||
if (subPage > 0 && !correctPIN && strlen(settingsPIN)>0) {
|
if (subPage > 0 && !correctPIN && strlen(settingsPIN)>0) {
|
||||||
strcpy_P(buf, PSTR("alert('PIN incorrect.');"));
|
strcpy_P(buf, PSTR("alert('PIN incorrect.');"));
|
||||||
request->send(403, "application/javascript", buf);
|
request->send(401, "application/javascript", buf);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
strcat_P(buf,PSTR("function GetV(){var d=document;"));
|
strcat_P(buf,PSTR("function GetV(){var d=document;"));
|
||||||
@ -575,7 +581,7 @@ void serveSettings(AsyncWebServerRequest* request, bool post)
|
|||||||
// if OTA locked or too frequent PIN entry requests fail hard
|
// if OTA locked or too frequent PIN entry requests fail hard
|
||||||
if ((subPage == SUBPAGE_WIFI && wifiLock && otaLock) || (post && !correctPIN && millis()-lastEditTime < PIN_RETRY_COOLDOWN))
|
if ((subPage == SUBPAGE_WIFI && wifiLock && otaLock) || (post && !correctPIN && millis()-lastEditTime < PIN_RETRY_COOLDOWN))
|
||||||
{
|
{
|
||||||
serveMessage(request, 500, "Access Denied", FPSTR(s_unlock_ota), 254); return;
|
serveMessage(request, 401, "Access Denied", FPSTR(s_unlock_ota), 254); return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (post) { //settings/set POST request, saving
|
if (post) { //settings/set POST request, saving
|
||||||
@ -585,7 +591,7 @@ void serveSettings(AsyncWebServerRequest* request, bool post)
|
|||||||
char s2[45] = "";
|
char s2[45] = "";
|
||||||
|
|
||||||
switch (subPage) {
|
switch (subPage) {
|
||||||
case SUBPAGE_WIFI : strcpy_P(s, PSTR("WiFi")); strcpy_P(s2, PSTR("Please connect to the new IP (if changed)")); forceReconnect = true; break;
|
case SUBPAGE_WIFI : strcpy_P(s, PSTR("WiFi")); strcpy_P(s2, PSTR("Please connect to the new IP (if changed)")); break;
|
||||||
case SUBPAGE_LEDS : strcpy_P(s, PSTR("LED")); break;
|
case SUBPAGE_LEDS : strcpy_P(s, PSTR("LED")); break;
|
||||||
case SUBPAGE_UI : strcpy_P(s, PSTR("UI")); break;
|
case SUBPAGE_UI : strcpy_P(s, PSTR("UI")); break;
|
||||||
case SUBPAGE_SYNC : strcpy_P(s, PSTR("Sync")); break;
|
case SUBPAGE_SYNC : strcpy_P(s, PSTR("Sync")); break;
|
||||||
@ -605,7 +611,7 @@ void serveSettings(AsyncWebServerRequest* request, bool post)
|
|||||||
if (!s2[0]) strcpy_P(s2, s_redirecting);
|
if (!s2[0]) strcpy_P(s2, s_redirecting);
|
||||||
|
|
||||||
bool redirectAfter9s = (subPage == SUBPAGE_WIFI || ((subPage == SUBPAGE_SEC || subPage == SUBPAGE_UM) && doReboot));
|
bool redirectAfter9s = (subPage == SUBPAGE_WIFI || ((subPage == SUBPAGE_SEC || subPage == SUBPAGE_UM) && doReboot));
|
||||||
serveMessage(request, 200, s, s2, redirectAfter9s ? 129 : (correctPIN ? 1 : 3));
|
serveMessage(request, (correctPIN ? 200 : 401), s, s2, redirectAfter9s ? 129 : (correctPIN ? 1 : 3));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -633,7 +639,7 @@ void serveSettings(AsyncWebServerRequest* request, bool post)
|
|||||||
serveMessage(request, 200, strlen(settingsPIN) > 0 ? PSTR("Settings locked") : PSTR("No PIN set"), FPSTR(s_redirecting), 1);
|
serveMessage(request, 200, strlen(settingsPIN) > 0 ? PSTR("Settings locked") : PSTR("No PIN set"), FPSTR(s_redirecting), 1);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
case SUBPAGE_PINREQ : response = request->beginResponse_P(200, "text/html", PAGE_settings_pin, PAGE_settings_pin_length); break;
|
case SUBPAGE_PINREQ : response = request->beginResponse_P(401, "text/html", PAGE_settings_pin, PAGE_settings_pin_length); break;
|
||||||
case SUBPAGE_CSS : response = request->beginResponse_P(200, "text/css", PAGE_settingsCss, PAGE_settingsCss_length); break;
|
case SUBPAGE_CSS : response = request->beginResponse_P(200, "text/css", PAGE_settingsCss, PAGE_settingsCss_length); break;
|
||||||
case SUBPAGE_JS : serveSettingsJS(request); return;
|
case SUBPAGE_JS : serveSettingsJS(request); return;
|
||||||
case SUBPAGE_WELCOME : response = request->beginResponse_P(200, "text/html", PAGE_welcome, PAGE_welcome_length); break;
|
case SUBPAGE_WELCOME : response = request->beginResponse_P(200, "text/html", PAGE_welcome, PAGE_welcome_length); break;
|
||||||
|
@ -31,7 +31,7 @@ void XML_response(AsyncWebServerRequest *request, char* dest)
|
|||||||
oappend(SET_F("<ns>"));
|
oappend(SET_F("<ns>"));
|
||||||
oappendi(notifyDirect);
|
oappendi(notifyDirect);
|
||||||
oappend(SET_F("</ns><nr>"));
|
oappend(SET_F("</ns><nr>"));
|
||||||
oappendi(receiveNotifications);
|
oappendi(receiveGroups!=0);
|
||||||
oappend(SET_F("</nr><nl>"));
|
oappend(SET_F("</nr><nl>"));
|
||||||
oappendi(nightlightActive);
|
oappendi(nightlightActive);
|
||||||
oappend(SET_F("</nl><nf>"));
|
oappend(SET_F("</nl><nf>"));
|
||||||
@ -280,11 +280,11 @@ void getSettingsJS(byte subPage, char* dest)
|
|||||||
sappend('c',SET_F("WS"),noWifiSleep);
|
sappend('c',SET_F("WS"),noWifiSleep);
|
||||||
|
|
||||||
#ifndef WLED_DISABLE_ESPNOW
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
sappend('c',SET_F("RE"),enable_espnow_remote);
|
sappend('c',SET_F("RE"),enableESPNow);
|
||||||
sappends('s',SET_F("RMAC"),linked_remote);
|
sappends('s',SET_F("RMAC"),linked_remote);
|
||||||
#else
|
#else
|
||||||
//hide remote settings if not compiled
|
//hide remote settings if not compiled
|
||||||
oappend(SET_F("document.getElementById('remd').style.display='none';"));
|
oappend(SET_F("toggle('ESPNOW');")); // hide ESP-NOW setting
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef WLED_USE_ETHERNET
|
#ifdef WLED_USE_ETHERNET
|
||||||
@ -321,14 +321,11 @@ void getSettingsJS(byte subPage, char* dest)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#ifndef WLED_DISABLE_ESPNOW
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
if (last_signal_src[0] != 0) //Have seen an ESP-NOW Remote
|
if (strlen(last_signal_src) > 0) { //Have seen an ESP-NOW Remote
|
||||||
{
|
|
||||||
sappends('m',SET_F("(\"rlid\")[0]"),last_signal_src);
|
sappends('m',SET_F("(\"rlid\")[0]"),last_signal_src);
|
||||||
} else if (!enable_espnow_remote)
|
} else if (!enableESPNow) {
|
||||||
{
|
sappends('m',SET_F("(\"rlid\")[0]"),(char*)F("(Enable ESP-NOW to listen)"));
|
||||||
sappends('m',SET_F("(\"rlid\")[0]"),(char*)F("(Enable remote to listen)"));
|
} else {
|
||||||
} else
|
|
||||||
{
|
|
||||||
sappends('m',SET_F("(\"rlid\")[0]"),(char*)F("None"));
|
sappends('m',SET_F("(\"rlid\")[0]"),(char*)F("None"));
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@ -441,6 +438,7 @@ void getSettingsJS(byte subPage, char* dest)
|
|||||||
sappend('c',SET_F("GC"),gammaCorrectCol);
|
sappend('c',SET_F("GC"),gammaCorrectCol);
|
||||||
dtostrf(gammaCorrectVal,3,1,nS); sappends('s',SET_F("GV"),nS);
|
dtostrf(gammaCorrectVal,3,1,nS); sappends('s',SET_F("GV"),nS);
|
||||||
sappend('c',SET_F("TF"),fadeTransition);
|
sappend('c',SET_F("TF"),fadeTransition);
|
||||||
|
sappend('c',SET_F("EB"),modeBlending);
|
||||||
sappend('v',SET_F("TD"),transitionDelayDefault);
|
sappend('v',SET_F("TD"),transitionDelayDefault);
|
||||||
sappend('c',SET_F("PF"),strip.paletteFade);
|
sappend('c',SET_F("PF"),strip.paletteFade);
|
||||||
sappend('v',SET_F("TP"),randomPaletteChangeTime);
|
sappend('v',SET_F("TP"),randomPaletteChangeTime);
|
||||||
@ -468,7 +466,7 @@ void getSettingsJS(byte subPage, char* dest)
|
|||||||
if (subPage == SUBPAGE_UI)
|
if (subPage == SUBPAGE_UI)
|
||||||
{
|
{
|
||||||
sappends('s',SET_F("DS"),serverDescription);
|
sappends('s',SET_F("DS"),serverDescription);
|
||||||
sappend('c',SET_F("ST"),syncToggleReceive);
|
//sappend('c',SET_F("ST"),syncToggleReceive);
|
||||||
#ifdef WLED_ENABLE_SIMPLE_UI
|
#ifdef WLED_ENABLE_SIMPLE_UI
|
||||||
sappend('c',SET_F("SU"),simplifiedUI);
|
sappend('c',SET_F("SU"),simplifiedUI);
|
||||||
#else
|
#else
|
||||||
@ -478,9 +476,15 @@ void getSettingsJS(byte subPage, char* dest)
|
|||||||
|
|
||||||
if (subPage == SUBPAGE_SYNC)
|
if (subPage == SUBPAGE_SYNC)
|
||||||
{
|
{
|
||||||
char nS[32];
|
[[maybe_unused]] char nS[32];
|
||||||
sappend('v',SET_F("UP"),udpPort);
|
sappend('v',SET_F("UP"),udpPort);
|
||||||
sappend('v',SET_F("U2"),udpPort2);
|
sappend('v',SET_F("U2"),udpPort2);
|
||||||
|
#ifndef WLED_DISABLE_ESPNOW
|
||||||
|
if (enableESPNow) sappend('c',SET_F("EN"),useESPNowSync);
|
||||||
|
else oappend(SET_F("toggle('ESPNOW');")); // hide ESP-NOW setting
|
||||||
|
#else
|
||||||
|
oappend(SET_F("toggle('ESPNOW');")); // hide ESP-NOW setting
|
||||||
|
#endif
|
||||||
sappend('v',SET_F("GS"),syncGroups);
|
sappend('v',SET_F("GS"),syncGroups);
|
||||||
sappend('v',SET_F("GR"),receiveGroups);
|
sappend('v',SET_F("GR"),receiveGroups);
|
||||||
|
|
||||||
@ -489,10 +493,11 @@ void getSettingsJS(byte subPage, char* dest)
|
|||||||
sappend('c',SET_F("RX"),receiveNotificationEffects);
|
sappend('c',SET_F("RX"),receiveNotificationEffects);
|
||||||
sappend('c',SET_F("SO"),receiveSegmentOptions);
|
sappend('c',SET_F("SO"),receiveSegmentOptions);
|
||||||
sappend('c',SET_F("SG"),receiveSegmentBounds);
|
sappend('c',SET_F("SG"),receiveSegmentBounds);
|
||||||
sappend('c',SET_F("SD"),notifyDirectDefault);
|
sappend('c',SET_F("SS"),sendNotifications);
|
||||||
|
sappend('c',SET_F("SD"),notifyDirect);
|
||||||
sappend('c',SET_F("SB"),notifyButton);
|
sappend('c',SET_F("SB"),notifyButton);
|
||||||
sappend('c',SET_F("SH"),notifyHue);
|
sappend('c',SET_F("SH"),notifyHue);
|
||||||
sappend('c',SET_F("SM"),notifyMacro);
|
// sappend('c',SET_F("SM"),notifyMacro);
|
||||||
sappend('v',SET_F("UR"),udpNumRetries);
|
sappend('v',SET_F("UR"),udpNumRetries);
|
||||||
|
|
||||||
sappend('c',SET_F("NL"),nodeListEnabled);
|
sappend('c',SET_F("NL"),nodeListEnabled);
|
||||||
|
Loading…
Reference in New Issue
Block a user