2019-02-09 16:37:20 +01:00
|
|
|
|
/*
|
|
|
|
|
WS2812FX_fcn.cpp contains all utility functions
|
|
|
|
|
Harm Aldick - 2016
|
|
|
|
|
www.aldick.org
|
|
|
|
|
LICENSE
|
|
|
|
|
The MIT License (MIT)
|
|
|
|
|
Copyright (c) 2016 Harm Aldick
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
|
THE SOFTWARE.
|
|
|
|
|
|
|
|
|
|
Modified heavily for WLED
|
|
|
|
|
*/
|
2021-02-24 20:23:32 +01:00
|
|
|
|
#include "wled.h"
|
2019-10-07 23:38:21 +02:00
|
|
|
|
#include "FX.h"
|
2019-02-09 16:37:20 +01:00
|
|
|
|
#include "palettes.h"
|
|
|
|
|
|
2021-02-13 01:02:14 +01:00
|
|
|
|
/*
|
|
|
|
|
Custom per-LED mapping has moved!
|
2020-03-21 00:57:54 +01:00
|
|
|
|
|
2021-02-13 01:02:14 +01:00
|
|
|
|
Create a file "ledmap.json" using the edit page.
|
2020-03-21 00:57:54 +01:00
|
|
|
|
|
2021-02-13 01:02:14 +01:00
|
|
|
|
this is just an example (30 LEDs). It will first set all even, then all uneven LEDs.
|
|
|
|
|
{"map":[
|
|
|
|
|
0, 2, 4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24, 26, 28,
|
|
|
|
|
1, 3, 5, 7, 9, 11, 13, 15, 17, 19, 21, 23, 25, 27, 29]}
|
2020-03-21 00:57:54 +01:00
|
|
|
|
|
2021-02-13 01:02:14 +01:00
|
|
|
|
another example. Switches direction every 5 LEDs.
|
|
|
|
|
{"map":[
|
|
|
|
|
0, 1, 2, 3, 4, 9, 8, 7, 6, 5, 10, 11, 12, 13, 14,
|
2021-02-24 20:23:32 +01:00
|
|
|
|
19, 18, 17, 16, 15, 20, 21, 22, 23, 24, 29, 28, 27, 26, 25]}
|
2021-02-13 01:02:14 +01:00
|
|
|
|
*/
|
2020-12-13 19:02:12 +01:00
|
|
|
|
|
2021-04-15 10:55:22 +02:00
|
|
|
|
//factory defaults LED setup
|
|
|
|
|
//#define PIXEL_COUNTS 30, 30, 30, 30
|
|
|
|
|
//#define DATA_PINS 16, 1, 3, 4
|
|
|
|
|
//#define DEFAULT_LED_TYPE TYPE_WS2812_RGB
|
|
|
|
|
|
|
|
|
|
#ifndef PIXEL_COUNTS
|
2021-12-02 13:09:53 +01:00
|
|
|
|
#define PIXEL_COUNTS DEFAULT_LED_COUNT
|
2021-04-15 10:55:22 +02:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifndef DATA_PINS
|
|
|
|
|
#define DATA_PINS LEDPIN
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifndef DEFAULT_LED_TYPE
|
|
|
|
|
#define DEFAULT_LED_TYPE TYPE_WS2812_RGB
|
|
|
|
|
#endif
|
|
|
|
|
|
2022-03-16 00:17:49 +01:00
|
|
|
|
#ifndef DEFAULT_LED_COLOR_ORDER
|
|
|
|
|
#define DEFAULT_LED_COLOR_ORDER COL_ORDER_GRB //default to GRB
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
2021-06-24 02:29:14 +02:00
|
|
|
|
#if MAX_NUM_SEGMENTS < WLED_MAX_BUSSES
|
|
|
|
|
#error "Max segments must be at least max number of busses!"
|
|
|
|
|
#endif
|
|
|
|
|
|
2022-07-06 13:13:54 +02:00
|
|
|
|
|
2022-07-10 22:23:25 +02:00
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
2022-07-17 15:58:41 +02:00
|
|
|
|
// Segment class implementation
|
2022-07-10 22:23:25 +02:00
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
2022-07-30 14:20:36 +02:00
|
|
|
|
uint16_t Segment::_usedSegmentData = 0U; // amount of RAM all segments use for their data[]
|
2022-12-16 22:31:07 +01:00
|
|
|
|
uint16_t Segment::maxWidth = DEFAULT_LED_COUNT;
|
|
|
|
|
uint16_t Segment::maxHeight = 1;
|
2022-07-10 22:23:25 +02:00
|
|
|
|
|
2022-07-30 14:50:11 +02:00
|
|
|
|
// copy constructor
|
2022-07-17 15:58:41 +02:00
|
|
|
|
Segment::Segment(const Segment &orig) {
|
2022-11-09 20:09:01 +01:00
|
|
|
|
//DEBUG_PRINTLN(F("-- Copy segment constructor --"));
|
2023-03-05 22:56:14 +01:00
|
|
|
|
memcpy((void*)this, (void*)&orig, sizeof(Segment));
|
2023-07-12 20:52:34 +02:00
|
|
|
|
transitional = false; // copied segment cannot be in transition
|
2022-07-17 15:58:41 +02:00
|
|
|
|
name = nullptr;
|
|
|
|
|
data = nullptr;
|
|
|
|
|
_dataLen = 0;
|
2022-07-30 14:50:11 +02:00
|
|
|
|
_t = nullptr;
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (orig.name) { name = new char[strlen(orig.name)+1]; if (name) strcpy(name, orig.name); }
|
2022-07-29 20:24:29 +02:00
|
|
|
|
if (orig.data) { if (allocateData(orig._dataLen)) memcpy(data, orig.data, orig._dataLen); }
|
2023-07-12 20:52:34 +02:00
|
|
|
|
//if (orig._t) { _t = new Transition(orig._t->_dur, orig._t->_briT, orig._t->_cctT, orig._t->_colorT); }
|
2022-07-17 15:58:41 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-30 14:50:11 +02:00
|
|
|
|
// move constructor
|
2022-07-17 15:58:41 +02:00
|
|
|
|
Segment::Segment(Segment &&orig) noexcept {
|
2022-11-09 20:09:01 +01:00
|
|
|
|
//DEBUG_PRINTLN(F("-- Move segment constructor --"));
|
2023-03-05 22:56:14 +01:00
|
|
|
|
memcpy((void*)this, (void*)&orig, sizeof(Segment));
|
2023-07-12 20:52:34 +02:00
|
|
|
|
orig.transitional = false; // old segment cannot be in transition any more
|
2022-07-17 15:58:41 +02:00
|
|
|
|
orig.name = nullptr;
|
|
|
|
|
orig.data = nullptr;
|
2022-07-19 16:16:43 +02:00
|
|
|
|
orig._dataLen = 0;
|
2022-07-30 14:50:11 +02:00
|
|
|
|
orig._t = nullptr;
|
2022-07-17 15:58:41 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-30 14:50:11 +02:00
|
|
|
|
// copy assignment
|
2022-07-17 15:58:41 +02:00
|
|
|
|
Segment& Segment::operator= (const Segment &orig) {
|
2022-11-09 20:09:01 +01:00
|
|
|
|
//DEBUG_PRINTLN(F("-- Copying segment --"));
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (this != &orig) {
|
2022-08-06 12:39:12 +02:00
|
|
|
|
// clean destination
|
2023-07-12 20:52:34 +02:00
|
|
|
|
transitional = false; // copied segment cannot be in transition
|
2022-08-06 12:39:12 +02:00
|
|
|
|
if (name) delete[] name;
|
|
|
|
|
if (_t) delete _t;
|
2022-07-17 15:58:41 +02:00
|
|
|
|
deallocateData();
|
2022-08-06 12:39:12 +02:00
|
|
|
|
// copy source
|
2023-03-05 22:56:14 +01:00
|
|
|
|
memcpy((void*)this, (void*)&orig, sizeof(Segment));
|
2023-07-12 20:52:34 +02:00
|
|
|
|
transitional = false;
|
2022-08-06 12:39:12 +02:00
|
|
|
|
// erase pointers to allocated data
|
2022-07-17 15:58:41 +02:00
|
|
|
|
name = nullptr;
|
|
|
|
|
data = nullptr;
|
|
|
|
|
_dataLen = 0;
|
2022-07-30 14:50:11 +02:00
|
|
|
|
_t = nullptr;
|
2022-08-06 12:39:12 +02:00
|
|
|
|
// copy source data
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (orig.name) { name = new char[strlen(orig.name)+1]; if (name) strcpy(name, orig.name); }
|
2022-07-29 20:24:29 +02:00
|
|
|
|
if (orig.data) { if (allocateData(orig._dataLen)) memcpy(data, orig.data, orig._dataLen); }
|
2023-07-12 20:52:34 +02:00
|
|
|
|
//if (orig._t) { _t = new Transition(orig._t->_dur, orig._t->_briT, orig._t->_cctT, orig._t->_colorT); }
|
2022-07-17 15:58:41 +02:00
|
|
|
|
}
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-30 14:50:11 +02:00
|
|
|
|
// move assignment
|
2022-07-17 15:58:41 +02:00
|
|
|
|
Segment& Segment::operator= (Segment &&orig) noexcept {
|
2022-11-09 20:09:01 +01:00
|
|
|
|
//DEBUG_PRINTLN(F("-- Moving segment --"));
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (this != &orig) {
|
2023-07-12 20:52:34 +02:00
|
|
|
|
transitional = false; // just temporary
|
|
|
|
|
if (name) { delete[] name; name = nullptr; } // free old name
|
2022-07-17 15:58:41 +02:00
|
|
|
|
deallocateData(); // free old runtime data
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (_t) { delete _t; _t = nullptr; }
|
2023-03-05 22:56:14 +01:00
|
|
|
|
memcpy((void*)this, (void*)&orig, sizeof(Segment));
|
2023-07-12 20:52:34 +02:00
|
|
|
|
orig.transitional = false; // old segment cannot be in transition
|
2022-07-17 15:58:41 +02:00
|
|
|
|
orig.name = nullptr;
|
|
|
|
|
orig.data = nullptr;
|
2022-07-19 16:16:43 +02:00
|
|
|
|
orig._dataLen = 0;
|
2022-07-30 14:50:11 +02:00
|
|
|
|
orig._t = nullptr;
|
2022-07-17 15:58:41 +02:00
|
|
|
|
}
|
|
|
|
|
return *this;
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-30 14:20:36 +02:00
|
|
|
|
bool Segment::allocateData(size_t len) {
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (data && _dataLen == len) return true; //already allocated
|
|
|
|
|
deallocateData();
|
2022-07-30 14:20:36 +02:00
|
|
|
|
if (Segment::getUsedSegmentData() + len > MAX_SEGMENT_DATA) return false; //not enough memory
|
2023-05-28 22:50:19 +02:00
|
|
|
|
// do not use SPI RAM on ESP32 since it is slow
|
2023-07-12 20:52:34 +02:00
|
|
|
|
data = (byte*) malloc(len);
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (!data) return false; //allocation failed
|
2022-07-30 14:20:36 +02:00
|
|
|
|
Segment::addUsedSegmentData(len);
|
2022-07-17 15:58:41 +02:00
|
|
|
|
_dataLen = len;
|
|
|
|
|
memset(data, 0, len);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Segment::deallocateData() {
|
|
|
|
|
if (!data) return;
|
|
|
|
|
free(data);
|
|
|
|
|
data = nullptr;
|
2022-07-30 14:20:36 +02:00
|
|
|
|
Segment::addUsedSegmentData(-_dataLen);
|
2022-07-17 15:58:41 +02:00
|
|
|
|
_dataLen = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-06 09:10:39 +01:00
|
|
|
|
/**
|
2022-07-17 15:58:41 +02:00
|
|
|
|
* If reset of this segment was requested, clears runtime
|
|
|
|
|
* settings of this segment.
|
|
|
|
|
* Must not be called while an effect mode function is running
|
2023-01-06 09:10:39 +01:00
|
|
|
|
* because it could access the data buffer and this method
|
2022-07-17 15:58:41 +02:00
|
|
|
|
* may free that data buffer.
|
|
|
|
|
*/
|
|
|
|
|
void Segment::resetIfRequired() {
|
2023-07-13 03:09:42 +02:00
|
|
|
|
if (!reset) return;
|
|
|
|
|
|
|
|
|
|
deallocateData();
|
|
|
|
|
next_time = 0; step = 0; call = 0; aux0 = 0; aux1 = 0;
|
|
|
|
|
reset = false;
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-29 12:15:56 +02:00
|
|
|
|
CRGBPalette16 &Segment::loadPalette(CRGBPalette16 &targetPalette, uint8_t pal) {
|
2022-07-28 23:19:58 +02:00
|
|
|
|
static unsigned long _lastPaletteChange = 0; // perhaps it should be per segment
|
2022-09-04 20:17:05 +02:00
|
|
|
|
static CRGBPalette16 randomPalette = CRGBPalette16(DEFAULT_COLOR);
|
2022-12-21 21:00:28 +01:00
|
|
|
|
static CRGBPalette16 prevRandomPalette = CRGBPalette16(CRGB(BLACK));
|
2022-07-28 23:19:58 +02:00
|
|
|
|
byte tcp[72];
|
2022-07-29 12:15:56 +02:00
|
|
|
|
if (pal < 245 && pal > GRADIENT_PALETTE_COUNT+13) pal = 0;
|
2022-07-29 16:26:15 +02:00
|
|
|
|
if (pal > 245 && (strip.customPalettes.size() == 0 || 255U-pal > strip.customPalettes.size()-1)) pal = 0;
|
2022-07-28 23:19:58 +02:00
|
|
|
|
//default palette. Differs depending on effect
|
|
|
|
|
if (pal == 0) switch (mode) {
|
|
|
|
|
case FX_MODE_FIRE_2012 : pal = 35; break; // heat palette
|
|
|
|
|
case FX_MODE_COLORWAVES : pal = 26; break; // landscape 33
|
|
|
|
|
case FX_MODE_FILLNOISE8 : pal = 9; break; // ocean colors
|
|
|
|
|
case FX_MODE_NOISE16_1 : pal = 20; break; // Drywet
|
|
|
|
|
case FX_MODE_NOISE16_2 : pal = 43; break; // Blue cyan yellow
|
|
|
|
|
case FX_MODE_NOISE16_3 : pal = 35; break; // heat palette
|
|
|
|
|
case FX_MODE_NOISE16_4 : pal = 26; break; // landscape 33
|
|
|
|
|
case FX_MODE_GLITTER : pal = 11; break; // rainbow colors
|
|
|
|
|
case FX_MODE_SUNRISE : pal = 35; break; // heat palette
|
2023-02-21 17:07:32 +01:00
|
|
|
|
case FX_MODE_RAILWAY : pal = 3; break; // prim + sec
|
2023-04-27 01:22:33 +02:00
|
|
|
|
case FX_MODE_2DSOAP : pal = 11; break; // rainbow colors
|
2022-07-28 23:19:58 +02:00
|
|
|
|
}
|
|
|
|
|
switch (pal) {
|
|
|
|
|
case 0: //default palette. Exceptions for specific effects above
|
|
|
|
|
targetPalette = PartyColors_p; break;
|
2022-12-21 21:00:28 +01:00
|
|
|
|
case 1: {//periodically replace palette with a random one. Transition palette change in 500ms
|
|
|
|
|
uint32_t timeSinceLastChange = millis() - _lastPaletteChange;
|
2023-03-25 21:28:30 +01:00
|
|
|
|
if (timeSinceLastChange > randomPaletteChangeTime * 1000U) {
|
2022-12-21 21:00:28 +01:00
|
|
|
|
prevRandomPalette = randomPalette;
|
2022-09-04 20:17:05 +02:00
|
|
|
|
randomPalette = CRGBPalette16(
|
|
|
|
|
CHSV(random8(), random8(160, 255), random8(128, 255)),
|
|
|
|
|
CHSV(random8(), random8(160, 255), random8(128, 255)),
|
|
|
|
|
CHSV(random8(), random8(160, 255), random8(128, 255)),
|
|
|
|
|
CHSV(random8(), random8(160, 255), random8(128, 255)));
|
2022-07-28 23:19:58 +02:00
|
|
|
|
_lastPaletteChange = millis();
|
2022-12-21 21:00:28 +01:00
|
|
|
|
timeSinceLastChange = 0;
|
2022-09-04 20:17:05 +02:00
|
|
|
|
}
|
2023-01-11 23:08:08 +01:00
|
|
|
|
if (timeSinceLastChange <= 250) {
|
2022-12-21 21:00:28 +01:00
|
|
|
|
targetPalette = prevRandomPalette;
|
|
|
|
|
// there needs to be 255 palette blends (48) for full blend but that is too resource intensive
|
2022-12-23 16:37:13 +01:00
|
|
|
|
// so 128 is a compromise (we need to perform full blend of the two palettes as each segment can have random
|
|
|
|
|
// palette selected but only 2 static palettes are used)
|
2023-01-11 23:08:08 +01:00
|
|
|
|
size_t noOfBlends = ((128U * timeSinceLastChange) / 250U);
|
2022-12-21 21:00:28 +01:00
|
|
|
|
for (size_t i=0; i<noOfBlends; i++) nblendPaletteTowardPalette(targetPalette, randomPalette, 48);
|
|
|
|
|
} else {
|
|
|
|
|
targetPalette = randomPalette;
|
|
|
|
|
}
|
|
|
|
|
break;}
|
2022-07-28 23:19:58 +02:00
|
|
|
|
case 2: {//primary color only
|
2022-09-04 20:17:05 +02:00
|
|
|
|
CRGB prim = gamma32(colors[0]);
|
2022-07-28 23:19:58 +02:00
|
|
|
|
targetPalette = CRGBPalette16(prim); break;}
|
|
|
|
|
case 3: {//primary + secondary
|
2022-09-04 20:17:05 +02:00
|
|
|
|
CRGB prim = gamma32(colors[0]);
|
|
|
|
|
CRGB sec = gamma32(colors[1]);
|
2022-07-28 23:19:58 +02:00
|
|
|
|
targetPalette = CRGBPalette16(prim,prim,sec,sec); break;}
|
|
|
|
|
case 4: {//primary + secondary + tertiary
|
2022-09-04 20:17:05 +02:00
|
|
|
|
CRGB prim = gamma32(colors[0]);
|
|
|
|
|
CRGB sec = gamma32(colors[1]);
|
|
|
|
|
CRGB ter = gamma32(colors[2]);
|
2022-07-28 23:19:58 +02:00
|
|
|
|
targetPalette = CRGBPalette16(ter,sec,prim); break;}
|
|
|
|
|
case 5: {//primary + secondary (+tert if not off), more distinct
|
2022-09-04 20:17:05 +02:00
|
|
|
|
CRGB prim = gamma32(colors[0]);
|
|
|
|
|
CRGB sec = gamma32(colors[1]);
|
2022-07-28 23:19:58 +02:00
|
|
|
|
if (colors[2]) {
|
2022-09-04 20:17:05 +02:00
|
|
|
|
CRGB ter = gamma32(colors[2]);
|
2022-07-28 23:19:58 +02:00
|
|
|
|
targetPalette = CRGBPalette16(prim,prim,prim,prim,prim,sec,sec,sec,sec,sec,ter,ter,ter,ter,ter,prim);
|
|
|
|
|
} else {
|
|
|
|
|
targetPalette = CRGBPalette16(prim,prim,prim,prim,prim,prim,prim,prim,sec,sec,sec,sec,sec,sec,sec,sec);
|
|
|
|
|
}
|
|
|
|
|
break;}
|
|
|
|
|
case 6: //Party colors
|
|
|
|
|
targetPalette = PartyColors_p; break;
|
|
|
|
|
case 7: //Cloud colors
|
|
|
|
|
targetPalette = CloudColors_p; break;
|
|
|
|
|
case 8: //Lava colors
|
|
|
|
|
targetPalette = LavaColors_p; break;
|
|
|
|
|
case 9: //Ocean colors
|
|
|
|
|
targetPalette = OceanColors_p; break;
|
|
|
|
|
case 10: //Forest colors
|
|
|
|
|
targetPalette = ForestColors_p; break;
|
|
|
|
|
case 11: //Rainbow colors
|
|
|
|
|
targetPalette = RainbowColors_p; break;
|
|
|
|
|
case 12: //Rainbow stripe colors
|
|
|
|
|
targetPalette = RainbowStripeColors_p; break;
|
|
|
|
|
default: //progmem palettes
|
|
|
|
|
if (pal>245) {
|
|
|
|
|
targetPalette = strip.customPalettes[255-pal]; // we checked bounds above
|
|
|
|
|
} else {
|
|
|
|
|
memcpy_P(tcp, (byte*)pgm_read_dword(&(gGradientPalettes[pal-13])), 72);
|
|
|
|
|
targetPalette.loadDynamicGradientPalette(tcp);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return targetPalette;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-24 23:04:51 +02:00
|
|
|
|
void Segment::startTransition(uint16_t dur) {
|
2023-06-23 23:49:54 +02:00
|
|
|
|
if (!dur) {
|
|
|
|
|
transitional = false;
|
|
|
|
|
if (_t) {
|
|
|
|
|
delete _t;
|
|
|
|
|
_t = nullptr;
|
|
|
|
|
}
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (transitional && _t) return; // already in transition no need to store anything
|
2022-08-24 23:04:51 +02:00
|
|
|
|
|
|
|
|
|
// starting a transition has to occur before change so we get current values 1st
|
2023-06-23 23:49:54 +02:00
|
|
|
|
_t = new Transition(dur); // no previous transition running
|
2022-08-24 23:04:51 +02:00
|
|
|
|
if (!_t) return; // failed to allocate data
|
2023-06-23 23:49:54 +02:00
|
|
|
|
|
|
|
|
|
CRGBPalette16 _palT = CRGBPalette16(DEFAULT_COLOR); loadPalette(_palT, palette);
|
|
|
|
|
_t->_briT = on ? opacity : 0;
|
|
|
|
|
_t->_cctT = cct;
|
2022-08-24 23:04:51 +02:00
|
|
|
|
_t->_palT = _palT;
|
2023-06-23 23:49:54 +02:00
|
|
|
|
_t->_modeP = mode;
|
|
|
|
|
for (size_t i=0; i<NUM_COLORS; i++) _t->_colorT[i] = colors[i];
|
2022-08-24 23:04:51 +02:00
|
|
|
|
transitional = true; // setOption(SEG_OPTION_TRANSITIONAL, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// transition progression between 0-65535
|
|
|
|
|
uint16_t Segment::progress() {
|
|
|
|
|
if (!transitional || !_t) return 0xFFFFU;
|
2023-07-12 20:52:34 +02:00
|
|
|
|
unsigned long timeNow = millis();
|
2022-08-24 23:04:51 +02:00
|
|
|
|
if (timeNow - _t->_start > _t->_dur || _t->_dur == 0) return 0xFFFFU;
|
|
|
|
|
return (timeNow - _t->_start) * 0xFFFFU / _t->_dur;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t Segment::currentBri(uint8_t briNew, bool useCct) {
|
2023-06-23 23:49:54 +02:00
|
|
|
|
uint32_t prog = progress();
|
|
|
|
|
if (transitional && _t && prog < 0xFFFFU) {
|
|
|
|
|
if (useCct) return ((briNew * prog) + _t->_cctT * (0xFFFFU - prog)) >> 16;
|
|
|
|
|
else return ((briNew * prog) + _t->_briT * (0xFFFFU - prog)) >> 16;
|
2022-08-24 23:04:51 +02:00
|
|
|
|
} else {
|
|
|
|
|
return briNew;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t Segment::currentMode(uint8_t newMode) {
|
2022-11-13 12:13:49 +01:00
|
|
|
|
return (progress()>32767U) ? newMode : _t->_modeP; // change effect in the middle of transition
|
2022-08-24 23:04:51 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint32_t Segment::currentColor(uint8_t slot, uint32_t colorNew) {
|
|
|
|
|
return transitional && _t ? color_blend(_t->_colorT[slot], colorNew, progress(), true) : colorNew;
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-29 12:15:56 +02:00
|
|
|
|
CRGBPalette16 &Segment::currentPalette(CRGBPalette16 &targetPalette, uint8_t pal) {
|
|
|
|
|
loadPalette(targetPalette, pal);
|
2022-07-30 14:50:11 +02:00
|
|
|
|
if (transitional && _t && progress() < 0xFFFFU) {
|
2022-07-28 23:19:58 +02:00
|
|
|
|
// blend palettes
|
2022-09-06 21:47:50 +02:00
|
|
|
|
// 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
|
2023-07-12 20:52:34 +02:00
|
|
|
|
unsigned long timeMS = millis() - _t->_start;
|
2022-09-06 21:47:50 +02:00
|
|
|
|
uint16_t noOfBlends = (255U * timeMS / _t->_dur) - _t->_prevPaletteBlends;
|
|
|
|
|
for (int i=0; i<noOfBlends; i++, _t->_prevPaletteBlends++) nblendPaletteTowardPalette(_t->_palT, targetPalette, 48);
|
2022-07-30 14:50:11 +02:00
|
|
|
|
targetPalette = _t->_palT; // copy transitioning/temporary palette
|
2022-07-28 23:19:58 +02:00
|
|
|
|
}
|
2022-07-29 12:15:56 +02:00
|
|
|
|
return targetPalette;
|
2022-07-28 23:19:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-17 15:58:41 +02:00
|
|
|
|
void Segment::handleTransition() {
|
2022-07-30 14:50:11 +02:00
|
|
|
|
if (!transitional) return;
|
2023-06-04 18:36:46 +02:00
|
|
|
|
uint16_t _progress = progress();
|
2023-06-23 23:49:54 +02:00
|
|
|
|
if (_progress == 0xFFFFU) transitional = false; // finish transitioning segment
|
2023-06-04 18:36:46 +02:00
|
|
|
|
if (_t) { // thanks to @nXm AKA https://github.com/NMeirer
|
|
|
|
|
if (_progress >= 32767U && _t->_modeP != mode) markForReset();
|
|
|
|
|
if (_progress == 0xFFFFU) {
|
2022-07-30 14:50:11 +02:00
|
|
|
|
delete _t;
|
|
|
|
|
_t = nullptr;
|
|
|
|
|
}
|
2022-07-17 15:58:41 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2022-07-10 22:23:25 +02:00
|
|
|
|
|
2023-07-13 03:09:42 +02:00
|
|
|
|
// segId is given when called from network callback, changes are queued if that segment is currently in its effect function
|
|
|
|
|
void Segment::setUp(uint16_t i1, uint16_t i2, uint8_t grp, uint8_t spc, uint16_t ofs, uint16_t i1Y, uint16_t i2Y, uint8_t segId) {
|
2023-07-12 20:52:34 +02:00
|
|
|
|
// return if neither bounds nor grouping have changed
|
2022-12-22 18:13:32 +01:00
|
|
|
|
bool boundsUnchanged = (start == i1 && stop == i2);
|
2022-12-23 16:37:13 +01:00
|
|
|
|
#ifndef WLED_DISABLE_2D
|
|
|
|
|
if (Segment::maxHeight>1) boundsUnchanged &= (startY == i1Y && stopY == i2Y); // 2D
|
|
|
|
|
#endif
|
2022-12-22 18:13:32 +01:00
|
|
|
|
if (boundsUnchanged
|
|
|
|
|
&& (!grp || (grouping == grp && spacing == spc))
|
|
|
|
|
&& (ofs == UINT16_MAX || ofs == offset)) return;
|
|
|
|
|
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (stop) fill(BLACK); // turn old segment range off (clears pixels if changing spacing)
|
|
|
|
|
if (grp) { // prevent assignment of 0
|
2022-12-22 18:13:32 +01:00
|
|
|
|
grouping = grp;
|
|
|
|
|
spacing = spc;
|
2023-07-12 20:52:34 +02:00
|
|
|
|
} else {
|
|
|
|
|
grouping = 1;
|
|
|
|
|
spacing = 0;
|
2022-12-22 18:13:32 +01:00
|
|
|
|
}
|
|
|
|
|
if (ofs < UINT16_MAX) offset = ofs;
|
2023-07-12 20:52:34 +02:00
|
|
|
|
|
2022-12-22 18:13:32 +01:00
|
|
|
|
markForReset();
|
2023-07-13 13:08:36 +02:00
|
|
|
|
if (boundsUnchanged) return;
|
2023-07-13 03:09:42 +02:00
|
|
|
|
|
|
|
|
|
// apply change immediately
|
|
|
|
|
if (i2 <= i1) { //disable segment
|
|
|
|
|
stop = 0;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (i1 < Segment::maxWidth || (i1 >= Segment::maxWidth*Segment::maxHeight && i1 < strip.getLengthTotal())) start = i1; // Segment::maxWidth equals strip.getLengthTotal() for 1D
|
|
|
|
|
stop = i2 > Segment::maxWidth*Segment::maxHeight ? MIN(i2,strip.getLengthTotal()) : (i2 > Segment::maxWidth ? Segment::maxWidth : MAX(1,i2));
|
|
|
|
|
startY = 0;
|
|
|
|
|
stopY = 1;
|
|
|
|
|
#ifndef WLED_DISABLE_2D
|
|
|
|
|
if (Segment::maxHeight>1) { // 2D
|
|
|
|
|
if (i1Y < Segment::maxHeight) startY = i1Y;
|
|
|
|
|
stopY = i2Y > Segment::maxHeight ? Segment::maxHeight : MAX(1,i2Y);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
// safety check
|
|
|
|
|
if (start >= stop || startY >= stopY) {
|
|
|
|
|
stop = 0;
|
|
|
|
|
return;
|
2023-07-12 20:52:34 +02:00
|
|
|
|
}
|
2023-07-13 03:09:42 +02:00
|
|
|
|
refreshLightCapabilities();
|
2022-12-22 18:13:32 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2022-07-10 22:23:25 +02:00
|
|
|
|
bool Segment::setColor(uint8_t slot, uint32_t c) { //returns true if changed
|
|
|
|
|
if (slot >= NUM_COLORS || c == colors[slot]) return false;
|
2023-03-02 18:21:55 +01:00
|
|
|
|
if (!_isRGB && !_hasW) {
|
|
|
|
|
if (slot == 0 && c == BLACK) return false; // on/off segment cannot have primary color black
|
|
|
|
|
if (slot == 1 && c != BLACK) return false; // on/off segment cannot have secondary color non black
|
|
|
|
|
}
|
2022-07-29 12:15:56 +02:00
|
|
|
|
if (fadeTransition) startTransition(strip.getTransition()); // start transition prior to change
|
2022-07-10 22:23:25 +02:00
|
|
|
|
colors[slot] = c;
|
2022-11-13 12:13:49 +01:00
|
|
|
|
stateChanged = true; // send UDP/WS broadcast
|
2022-07-10 22:23:25 +02:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Segment::setCCT(uint16_t k) {
|
2022-07-06 13:13:54 +02:00
|
|
|
|
if (k > 255) { //kelvin value, convert to 0-255
|
|
|
|
|
if (k < 1900) k = 1900;
|
|
|
|
|
if (k > 10091) k = 10091;
|
|
|
|
|
k = (k - 1900) >> 5;
|
|
|
|
|
}
|
|
|
|
|
if (cct == k) return;
|
2022-07-29 12:15:56 +02:00
|
|
|
|
if (fadeTransition) startTransition(strip.getTransition()); // start transition prior to change
|
2022-07-06 13:13:54 +02:00
|
|
|
|
cct = k;
|
2022-11-13 12:13:49 +01:00
|
|
|
|
stateChanged = true; // send UDP/WS broadcast
|
2022-07-06 13:13:54 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-10 22:23:25 +02:00
|
|
|
|
void Segment::setOpacity(uint8_t o) {
|
2022-07-06 13:13:54 +02:00
|
|
|
|
if (opacity == o) return;
|
2022-07-29 12:15:56 +02:00
|
|
|
|
if (fadeTransition) startTransition(strip.getTransition()); // start transition prior to change
|
2022-07-06 13:13:54 +02:00
|
|
|
|
opacity = o;
|
2022-11-13 12:13:49 +01:00
|
|
|
|
stateChanged = true; // send UDP/WS broadcast
|
2022-07-06 13:13:54 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-10 22:23:25 +02:00
|
|
|
|
void Segment::setOption(uint8_t n, bool val) {
|
2022-08-06 12:39:12 +02:00
|
|
|
|
bool prevOn = on;
|
2022-07-29 12:15:56 +02:00
|
|
|
|
if (fadeTransition && n == SEG_OPTION_ON && val != prevOn) startTransition(strip.getTransition()); // start transition prior to change
|
2022-07-06 13:13:54 +02:00
|
|
|
|
if (val) options |= 0x01 << n;
|
|
|
|
|
else options &= ~(0x01 << n);
|
2022-11-13 12:13:49 +01:00
|
|
|
|
if (!(n == SEG_OPTION_SELECTED || n == SEG_OPTION_RESET || n == SEG_OPTION_TRANSITIONAL)) stateChanged = true; // send UDP/WS broadcast
|
2022-07-06 13:13:54 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-09-29 12:49:12 +02:00
|
|
|
|
void Segment::setMode(uint8_t fx, bool loadDefaults) {
|
|
|
|
|
// if we have a valid mode & is not reserved
|
|
|
|
|
if (fx < strip.getModeCount() && strncmp_P("RSVD", strip.getModeData(fx), 4)) {
|
|
|
|
|
if (fx != mode) {
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (fadeTransition) startTransition(strip.getTransition()); // set effect transitions
|
2022-09-29 12:49:12 +02:00
|
|
|
|
mode = fx;
|
|
|
|
|
|
|
|
|
|
// load default values from effect string
|
|
|
|
|
if (loadDefaults) {
|
|
|
|
|
int16_t sOpt;
|
2023-01-15 15:21:39 +01:00
|
|
|
|
sOpt = extractModeDefaults(fx, "sx"); speed = (sOpt >= 0) ? sOpt : DEFAULT_SPEED;
|
|
|
|
|
sOpt = extractModeDefaults(fx, "ix"); intensity = (sOpt >= 0) ? sOpt : DEFAULT_INTENSITY;
|
|
|
|
|
sOpt = extractModeDefaults(fx, "c1"); custom1 = (sOpt >= 0) ? sOpt : DEFAULT_C1;
|
|
|
|
|
sOpt = extractModeDefaults(fx, "c2"); custom2 = (sOpt >= 0) ? sOpt : DEFAULT_C2;
|
|
|
|
|
sOpt = extractModeDefaults(fx, "c3"); custom3 = (sOpt >= 0) ? sOpt : DEFAULT_C3;
|
|
|
|
|
sOpt = extractModeDefaults(fx, "o1"); check1 = (sOpt >= 0) ? (bool)sOpt : false;
|
|
|
|
|
sOpt = extractModeDefaults(fx, "o2"); check2 = (sOpt >= 0) ? (bool)sOpt : false;
|
|
|
|
|
sOpt = extractModeDefaults(fx, "o3"); check3 = (sOpt >= 0) ? (bool)sOpt : false;
|
2022-11-24 04:15:24 +01:00
|
|
|
|
sOpt = extractModeDefaults(fx, "m12"); if (sOpt >= 0) map1D2D = constrain(sOpt, 0, 7);
|
2023-04-04 17:16:50 +02:00
|
|
|
|
sOpt = extractModeDefaults(fx, "si"); if (sOpt >= 0) soundSim = constrain(sOpt, 0, 1);
|
2022-09-29 12:49:12 +02:00
|
|
|
|
sOpt = extractModeDefaults(fx, "rev"); if (sOpt >= 0) reverse = (bool)sOpt;
|
|
|
|
|
sOpt = extractModeDefaults(fx, "mi"); if (sOpt >= 0) mirror = (bool)sOpt; // NOTE: setting this option is a risky business
|
|
|
|
|
sOpt = extractModeDefaults(fx, "rY"); if (sOpt >= 0) reverse_y = (bool)sOpt;
|
|
|
|
|
sOpt = extractModeDefaults(fx, "mY"); if (sOpt >= 0) mirror_y = (bool)sOpt; // NOTE: setting this option is a risky business
|
2023-01-15 15:21:39 +01:00
|
|
|
|
sOpt = extractModeDefaults(fx, "pal"); if (sOpt >= 0) setPalette(sOpt); //else setPalette(0);
|
2022-09-29 12:49:12 +02:00
|
|
|
|
}
|
2022-11-13 12:13:49 +01:00
|
|
|
|
stateChanged = true; // send UDP/WS broadcast
|
2022-09-29 12:49:12 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Segment::setPalette(uint8_t pal) {
|
2022-11-25 17:33:29 +01:00
|
|
|
|
if (pal < 245 && pal > GRADIENT_PALETTE_COUNT+13) pal = 0; // built in palettes
|
|
|
|
|
if (pal > 245 && (strip.customPalettes.size() == 0 || 255U-pal > strip.customPalettes.size()-1)) pal = 0; // custom palettes
|
|
|
|
|
if (pal != palette) {
|
|
|
|
|
if (strip.paletteFade) startTransition(strip.getTransition());
|
|
|
|
|
palette = pal;
|
|
|
|
|
stateChanged = true; // send UDP/WS broadcast
|
2022-09-29 12:49:12 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-06 13:13:54 +02:00
|
|
|
|
// 2D matrix
|
2022-08-17 20:45:30 +02:00
|
|
|
|
uint16_t Segment::virtualWidth() const {
|
2022-07-06 13:13:54 +02:00
|
|
|
|
uint16_t groupLen = groupLength();
|
2022-08-06 12:39:12 +02:00
|
|
|
|
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
|
2022-07-06 13:13:54 +02:00
|
|
|
|
return vWidth;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-17 20:45:30 +02:00
|
|
|
|
uint16_t Segment::virtualHeight() const {
|
2022-07-06 13:13:54 +02:00
|
|
|
|
uint16_t groupLen = groupLength();
|
2022-08-06 12:39:12 +02:00
|
|
|
|
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
|
2022-07-06 13:13:54 +02:00
|
|
|
|
return vHeight;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-25 21:57:43 +02:00
|
|
|
|
uint16_t Segment::nrOfVStrips() const {
|
|
|
|
|
uint16_t vLen = 1;
|
|
|
|
|
#ifndef WLED_DISABLE_2D
|
|
|
|
|
if (is2D()) {
|
2022-08-29 21:51:46 +02:00
|
|
|
|
switch (map1D2D) {
|
|
|
|
|
case M12_pBar:
|
|
|
|
|
vLen = virtualWidth();
|
|
|
|
|
break;
|
|
|
|
|
}
|
2022-08-25 21:57:43 +02:00
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
return vLen;
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-06 13:13:54 +02:00
|
|
|
|
// 1D strip
|
2022-08-17 20:45:30 +02:00
|
|
|
|
uint16_t Segment::virtualLength() const {
|
2022-07-14 13:22:34 +02:00
|
|
|
|
#ifndef WLED_DISABLE_2D
|
2022-07-30 14:20:36 +02:00
|
|
|
|
if (is2D()) {
|
2022-07-14 13:22:34 +02:00
|
|
|
|
uint16_t vW = virtualWidth();
|
|
|
|
|
uint16_t vH = virtualHeight();
|
2022-08-25 21:57:43 +02:00
|
|
|
|
uint16_t vLen = vW * vH; // use all pixels from segment
|
2022-07-20 21:22:23 +02:00
|
|
|
|
switch (map1D2D) {
|
2022-08-25 21:57:43 +02:00
|
|
|
|
case M12_pBar:
|
|
|
|
|
vLen = vH;
|
2022-07-14 13:22:34 +02:00
|
|
|
|
break;
|
2022-08-25 21:57:43 +02:00
|
|
|
|
case M12_pCorner:
|
|
|
|
|
case M12_pArc:
|
2022-07-25 21:31:50 +02:00
|
|
|
|
vLen = max(vW,vH); // get the longest dimension
|
2022-07-14 13:22:34 +02:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return vLen;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2023-07-12 20:52:34 +02:00
|
|
|
|
uint16_t groupLen = groupLength(); // is always >= 1
|
2022-07-06 13:13:54 +02:00
|
|
|
|
uint16_t vLength = (length() + groupLen - 1) / groupLen;
|
2022-08-06 12:39:12 +02:00
|
|
|
|
if (mirror) vLength = (vLength + 1) /2; // divide by 2 if mirror, leave at least a single LED
|
2022-07-06 13:13:54 +02:00
|
|
|
|
return vLength;
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-10 22:23:25 +02:00
|
|
|
|
void IRAM_ATTR Segment::setPixelColor(int i, uint32_t col)
|
|
|
|
|
{
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (!isActive()) return; // not active
|
2023-03-11 15:03:28 +01:00
|
|
|
|
#ifndef WLED_DISABLE_2D
|
2022-08-27 18:25:54 +02:00
|
|
|
|
int vStrip = i>>16; // hack to allow running on virtual strips (2D segment columns/rows)
|
2023-03-11 15:03:28 +01:00
|
|
|
|
#endif
|
2022-08-25 21:57:43 +02:00
|
|
|
|
i &= 0xFFFF;
|
|
|
|
|
|
2022-08-23 15:57:05 +02:00
|
|
|
|
if (i >= virtualLength() || i<0) return; // if pixel would fall out of segment just exit
|
|
|
|
|
|
2022-07-14 13:22:34 +02:00
|
|
|
|
#ifndef WLED_DISABLE_2D
|
2022-12-16 22:31:07 +01:00
|
|
|
|
if (is2D()) {
|
2022-07-14 13:22:34 +02:00
|
|
|
|
uint16_t vH = virtualHeight(); // segment height in logical pixels
|
|
|
|
|
uint16_t vW = virtualWidth();
|
2022-07-20 21:22:23 +02:00
|
|
|
|
switch (map1D2D) {
|
2022-07-11 11:20:30 +02:00
|
|
|
|
case M12_Pixels:
|
2022-07-14 13:22:34 +02:00
|
|
|
|
// use all available pixels as a long strip
|
|
|
|
|
setPixelColorXY(i % vW, i / vW, col);
|
2022-07-11 11:20:30 +02:00
|
|
|
|
break;
|
2022-08-25 21:57:43 +02:00
|
|
|
|
case M12_pBar:
|
|
|
|
|
// expand 1D effect vertically or have it play on virtual strips
|
|
|
|
|
if (vStrip>0) setPixelColorXY(vStrip - 1, vH - i - 1, col);
|
|
|
|
|
else for (int x = 0; x < vW; x++) setPixelColorXY(x, vH - i - 1, col);
|
2022-07-11 11:20:30 +02:00
|
|
|
|
break;
|
2022-08-25 21:57:43 +02:00
|
|
|
|
case M12_pArc:
|
2022-07-14 13:22:34 +02:00
|
|
|
|
// expand in circular fashion from center
|
2022-08-22 14:35:34 +02:00
|
|
|
|
if (i==0)
|
|
|
|
|
setPixelColorXY(0, 0, col);
|
|
|
|
|
else {
|
2022-08-25 21:57:43 +02:00
|
|
|
|
float step = HALF_PI / (2.85f*i);
|
2022-08-22 16:48:19 +02:00
|
|
|
|
for (float rad = 0.0f; rad <= HALF_PI+step/2; rad += step) {
|
2022-08-22 14:35:34 +02:00
|
|
|
|
// may want to try float version as well (with or without antialiasing)
|
|
|
|
|
int x = roundf(sin_t(rad) * i);
|
|
|
|
|
int y = roundf(cos_t(rad) * i);
|
|
|
|
|
setPixelColorXY(x, y, col);
|
|
|
|
|
}
|
2023-01-12 19:13:07 +01:00
|
|
|
|
// Bresenham’s Algorithm (may not fill every pixel)
|
2023-01-06 09:10:39 +01:00
|
|
|
|
//int d = 3 - (2*i);
|
|
|
|
|
//int y = i, x = 0;
|
|
|
|
|
//while (y >= x) {
|
|
|
|
|
// setPixelColorXY(x, y, col);
|
|
|
|
|
// setPixelColorXY(y, x, col);
|
|
|
|
|
// x++;
|
|
|
|
|
// if (d > 0) {
|
|
|
|
|
// y--;
|
|
|
|
|
// d += 4 * (x - y) + 10;
|
|
|
|
|
// } else {
|
|
|
|
|
// d += 4 * x + 6;
|
|
|
|
|
// }
|
|
|
|
|
//}
|
2022-07-11 11:20:30 +02:00
|
|
|
|
}
|
|
|
|
|
break;
|
2022-08-25 21:57:43 +02:00
|
|
|
|
case M12_pCorner:
|
2022-07-20 21:22:23 +02:00
|
|
|
|
for (int x = 0; x <= i; x++) setPixelColorXY(x, i, col);
|
|
|
|
|
for (int y = 0; y < i; y++) setPixelColorXY(i, y, col);
|
2022-07-11 11:20:30 +02:00
|
|
|
|
break;
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
|
|
|
|
return;
|
2022-12-16 22:31:07 +01:00
|
|
|
|
} else if (Segment::maxHeight!=1 && (width()==1 || height()==1)) {
|
2023-02-01 19:30:56 +01:00
|
|
|
|
if (start < Segment::maxWidth*Segment::maxHeight) {
|
|
|
|
|
// we have a vertical or horizontal 1D segment (WARNING: virtual...() may be transposed)
|
|
|
|
|
int x = 0, y = 0;
|
|
|
|
|
if (virtualHeight()>1) y = i;
|
|
|
|
|
if (virtualWidth() >1) x = i;
|
|
|
|
|
setPixelColorXY(x, y, col);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
2022-07-14 13:22:34 +02:00
|
|
|
|
#endif
|
2022-07-10 22:23:25 +02:00
|
|
|
|
|
|
|
|
|
uint16_t len = length();
|
2022-08-06 12:39:12 +02:00
|
|
|
|
uint8_t _bri_t = currentBri(on ? opacity : 0);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
if (_bri_t < 255) {
|
|
|
|
|
byte r = scale8(R(col), _bri_t);
|
|
|
|
|
byte g = scale8(G(col), _bri_t);
|
|
|
|
|
byte b = scale8(B(col), _bri_t);
|
|
|
|
|
byte w = scale8(W(col), _bri_t);
|
|
|
|
|
col = RGBW32(r, g, b, w);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// expand pixel (taking into account start, grouping, spacing [and offset])
|
|
|
|
|
i = i * groupLength();
|
2022-08-06 12:39:12 +02:00
|
|
|
|
if (reverse) { // is segment reversed?
|
|
|
|
|
if (mirror) { // is segment mirrored?
|
2022-07-10 22:23:25 +02:00
|
|
|
|
i = (len - 1) / 2 - i; //only need to index half the pixels
|
|
|
|
|
} else {
|
|
|
|
|
i = (len - 1) - i;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
i += start; // starting pixel in a group
|
|
|
|
|
|
|
|
|
|
// set all the pixels in the group
|
|
|
|
|
for (int j = 0; j < grouping; j++) {
|
2022-08-06 12:39:12 +02:00
|
|
|
|
uint16_t indexSet = i + ((reverse) ? -j : j);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
if (indexSet >= start && indexSet < stop) {
|
2022-08-06 12:39:12 +02:00
|
|
|
|
if (mirror) { //set the corresponding mirrored pixel
|
2023-01-06 09:10:39 +01:00
|
|
|
|
uint16_t indexMir = stop - indexSet + start - 1;
|
2022-07-10 22:23:25 +02:00
|
|
|
|
indexMir += offset; // offset/phase
|
|
|
|
|
if (indexMir >= stop) indexMir -= len; // wrap
|
|
|
|
|
strip.setPixelColor(indexMir, col);
|
|
|
|
|
}
|
|
|
|
|
indexSet += offset; // offset/phase
|
|
|
|
|
if (indexSet >= stop) indexSet -= len; // wrap
|
|
|
|
|
strip.setPixelColor(indexSet, col);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// anti-aliased normalized version of setPixelColor()
|
|
|
|
|
void Segment::setPixelColor(float i, uint32_t col, bool aa)
|
|
|
|
|
{
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (!isActive()) return; // not active
|
2022-08-27 18:25:54 +02:00
|
|
|
|
int vStrip = int(i/10.0f); // hack to allow running on virtual strips (2D segment columns/rows)
|
|
|
|
|
i -= int(i);
|
|
|
|
|
|
2022-07-10 22:23:25 +02:00
|
|
|
|
if (i<0.0f || i>1.0f) return; // not normalized
|
|
|
|
|
|
|
|
|
|
float fC = i * (virtualLength()-1);
|
|
|
|
|
if (aa) {
|
|
|
|
|
uint16_t iL = roundf(fC-0.49f);
|
|
|
|
|
uint16_t iR = roundf(fC+0.49f);
|
2022-08-30 17:18:56 +02:00
|
|
|
|
float dL = (fC - iL)*(fC - iL);
|
|
|
|
|
float dR = (iR - fC)*(iR - fC);
|
2022-08-27 18:25:54 +02:00
|
|
|
|
uint32_t cIL = getPixelColor(iL | (vStrip<<16));
|
|
|
|
|
uint32_t cIR = getPixelColor(iR | (vStrip<<16));
|
2022-07-10 22:23:25 +02:00
|
|
|
|
if (iR!=iL) {
|
|
|
|
|
// blend L pixel
|
|
|
|
|
cIL = color_blend(col, cIL, uint8_t(dL*255.0f));
|
2022-08-27 18:25:54 +02:00
|
|
|
|
setPixelColor(iL | (vStrip<<16), cIL);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
// blend R pixel
|
|
|
|
|
cIR = color_blend(col, cIR, uint8_t(dR*255.0f));
|
2022-08-27 18:25:54 +02:00
|
|
|
|
setPixelColor(iR | (vStrip<<16), cIR);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
} else {
|
|
|
|
|
// exact match (x & y land on a pixel)
|
2022-08-27 18:25:54 +02:00
|
|
|
|
setPixelColor(iL | (vStrip<<16), col);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
|
|
|
|
} else {
|
2022-08-27 18:25:54 +02:00
|
|
|
|
setPixelColor(uint16_t(roundf(fC)) | (vStrip<<16), col);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-25 21:57:43 +02:00
|
|
|
|
uint32_t Segment::getPixelColor(int i)
|
2022-07-10 22:23:25 +02:00
|
|
|
|
{
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (!isActive()) return 0; // not active
|
2023-03-11 15:03:28 +01:00
|
|
|
|
#ifndef WLED_DISABLE_2D
|
2022-08-27 18:25:54 +02:00
|
|
|
|
int vStrip = i>>16;
|
2023-03-11 15:03:28 +01:00
|
|
|
|
#endif
|
2022-08-25 21:57:43 +02:00
|
|
|
|
i &= 0xFFFF;
|
|
|
|
|
|
2022-07-14 13:22:34 +02:00
|
|
|
|
#ifndef WLED_DISABLE_2D
|
2022-12-16 22:31:07 +01:00
|
|
|
|
if (is2D()) {
|
2022-07-25 21:31:50 +02:00
|
|
|
|
uint16_t vH = virtualHeight(); // segment height in logical pixels
|
2022-07-14 13:22:34 +02:00
|
|
|
|
uint16_t vW = virtualWidth();
|
2022-07-20 21:22:23 +02:00
|
|
|
|
switch (map1D2D) {
|
2022-07-13 15:40:40 +02:00
|
|
|
|
case M12_Pixels:
|
2022-07-14 13:22:34 +02:00
|
|
|
|
return getPixelColorXY(i % vW, i / vW);
|
2022-07-13 15:40:40 +02:00
|
|
|
|
break;
|
2022-08-25 21:57:43 +02:00
|
|
|
|
case M12_pBar:
|
|
|
|
|
if (vStrip>0) return getPixelColorXY(vStrip - 1, vH - i -1);
|
|
|
|
|
else return getPixelColorXY(0, vH - i -1);
|
2022-07-25 21:31:50 +02:00
|
|
|
|
break;
|
2022-08-25 21:57:43 +02:00
|
|
|
|
case M12_pArc:
|
|
|
|
|
case M12_pCorner:
|
2022-07-25 21:31:50 +02:00
|
|
|
|
// use longest dimension
|
|
|
|
|
return vW>vH ? getPixelColorXY(i, 0) : getPixelColorXY(0, i);
|
2022-07-13 15:40:40 +02:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2022-07-14 13:22:34 +02:00
|
|
|
|
#endif
|
2022-07-13 15:40:40 +02:00
|
|
|
|
|
2022-08-06 12:39:12 +02:00
|
|
|
|
if (reverse) i = virtualLength() - i - 1;
|
2022-07-10 22:23:25 +02:00
|
|
|
|
i *= groupLength();
|
|
|
|
|
i += start;
|
|
|
|
|
/* offset/phase */
|
|
|
|
|
i += offset;
|
2023-07-03 15:43:47 +02:00
|
|
|
|
if ((i >= stop) && (stop>0)) i -= length(); // avoids negative pixel index (stop = 0 is a possible value)
|
2022-07-10 22:23:25 +02:00
|
|
|
|
return strip.getPixelColor(i);
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-17 20:45:30 +02:00
|
|
|
|
uint8_t Segment::differs(Segment& b) const {
|
2022-07-06 13:13:54 +02:00
|
|
|
|
uint8_t d = 0;
|
|
|
|
|
if (start != b.start) d |= SEG_DIFFERS_BOUNDS;
|
|
|
|
|
if (stop != b.stop) d |= SEG_DIFFERS_BOUNDS;
|
|
|
|
|
if (offset != b.offset) d |= SEG_DIFFERS_GSO;
|
|
|
|
|
if (grouping != b.grouping) d |= SEG_DIFFERS_GSO;
|
|
|
|
|
if (spacing != b.spacing) d |= SEG_DIFFERS_GSO;
|
|
|
|
|
if (opacity != b.opacity) d |= SEG_DIFFERS_BRI;
|
|
|
|
|
if (mode != b.mode) d |= SEG_DIFFERS_FX;
|
|
|
|
|
if (speed != b.speed) d |= SEG_DIFFERS_FX;
|
|
|
|
|
if (intensity != b.intensity) d |= SEG_DIFFERS_FX;
|
|
|
|
|
if (palette != b.palette) d |= SEG_DIFFERS_FX;
|
|
|
|
|
if (custom1 != b.custom1) d |= SEG_DIFFERS_FX;
|
|
|
|
|
if (custom2 != b.custom2) d |= SEG_DIFFERS_FX;
|
|
|
|
|
if (custom3 != b.custom3) d |= SEG_DIFFERS_FX;
|
|
|
|
|
if (startY != b.startY) d |= SEG_DIFFERS_BOUNDS;
|
|
|
|
|
if (stopY != b.stopY) d |= SEG_DIFFERS_BOUNDS;
|
|
|
|
|
|
2023-05-15 17:06:29 +02:00
|
|
|
|
//bit pattern: (msb first) set:2, sound:1, mapping:3, transposed, mirrorY, reverseY, [transitional, reset,] paused, mirrored, on, reverse, [selected]
|
2022-11-13 12:13:49 +01:00
|
|
|
|
if ((options & 0b1111111110011110U) != (b.options & 0b1111111110011110U)) d |= SEG_DIFFERS_OPT;
|
|
|
|
|
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;
|
2022-07-06 13:13:54 +02:00
|
|
|
|
|
|
|
|
|
return d;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Segment::refreshLightCapabilities() {
|
2023-02-14 01:33:06 +01:00
|
|
|
|
uint8_t capabilities = 0;
|
2023-02-15 20:36:54 +01:00
|
|
|
|
uint16_t segStartIdx = 0xFFFFU;
|
|
|
|
|
uint16_t segStopIdx = 0;
|
|
|
|
|
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (!isActive()) {
|
|
|
|
|
_capabilities = 0;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2023-02-15 20:36:54 +01:00
|
|
|
|
if (start < Segment::maxWidth * Segment::maxHeight) {
|
|
|
|
|
// we are withing 2D matrix (includes 1D segments)
|
|
|
|
|
for (int y = startY; y < stopY; y++) for (int x = start; x < stop; x++) {
|
|
|
|
|
uint16_t index = x + Segment::maxWidth * y;
|
|
|
|
|
if (index < strip.customMappingSize) index = strip.customMappingTable[index]; // convert logical address to physical
|
|
|
|
|
if (index < 0xFFFFU) {
|
|
|
|
|
if (segStartIdx > index) segStartIdx = index;
|
|
|
|
|
if (segStopIdx < index) segStopIdx = index;
|
|
|
|
|
}
|
2023-02-28 19:08:41 +01:00
|
|
|
|
if (segStartIdx == segStopIdx) segStopIdx++; // we only have 1 pixel segment
|
2023-02-15 20:36:54 +01:00
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// we are on the strip located after the matrix
|
|
|
|
|
segStartIdx = start;
|
|
|
|
|
segStopIdx = stop;
|
|
|
|
|
}
|
2022-07-06 13:13:54 +02:00
|
|
|
|
|
|
|
|
|
for (uint8_t b = 0; b < busses.getNumBusses(); b++) {
|
|
|
|
|
Bus *bus = busses.getBus(b);
|
|
|
|
|
if (bus == nullptr || bus->getLength()==0) break;
|
|
|
|
|
if (!bus->isOk()) continue;
|
2023-02-15 20:36:54 +01:00
|
|
|
|
if (bus->getStart() >= segStopIdx) continue;
|
|
|
|
|
if (bus->getStart() + bus->getLength() <= segStartIdx) continue;
|
2022-07-06 13:13:54 +02:00
|
|
|
|
|
2023-02-14 17:11:58 +01:00
|
|
|
|
//uint8_t type = bus->getType();
|
2023-02-14 01:33:06 +01:00
|
|
|
|
if (bus->hasRGB() || (cctFromRgb && bus->hasCCT())) capabilities |= SEG_CAPABILITY_RGB;
|
|
|
|
|
if (!cctFromRgb && bus->hasCCT()) capabilities |= SEG_CAPABILITY_CCT;
|
|
|
|
|
if (correctWB && (bus->hasRGB() || bus->hasCCT())) capabilities |= SEG_CAPABILITY_CCT; //white balance correction (CCT slider)
|
|
|
|
|
if (bus->hasWhite()) {
|
|
|
|
|
uint8_t aWM = Bus::getGlobalAWMode() == AW_GLOBAL_DISABLED ? bus->getAutoWhiteMode() : Bus::getGlobalAWMode();
|
|
|
|
|
bool whiteSlider = (aWM == RGBW_MODE_DUAL || aWM == RGBW_MODE_MANUAL_ONLY); // white slider allowed
|
|
|
|
|
// if auto white calculation from RGB is active (Accurate/Brighter), force RGB controls even if there are no RGB busses
|
|
|
|
|
if (!whiteSlider) capabilities |= SEG_CAPABILITY_RGB;
|
|
|
|
|
// if auto white calculation from RGB is disabled/optional (None/Dual), allow white channel adjustments
|
|
|
|
|
if ( whiteSlider) capabilities |= SEG_CAPABILITY_W;
|
2022-07-06 13:13:54 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
_capabilities = capabilities;
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-10 22:23:25 +02:00
|
|
|
|
/*
|
|
|
|
|
* Fills segment with color
|
|
|
|
|
*/
|
|
|
|
|
void Segment::fill(uint32_t c) {
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (!isActive()) return; // not active
|
2022-07-30 14:20:36 +02:00
|
|
|
|
const uint16_t cols = is2D() ? virtualWidth() : virtualLength();
|
2022-07-10 22:23:25 +02:00
|
|
|
|
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++) {
|
2022-07-30 14:20:36 +02:00
|
|
|
|
if (is2D()) setPixelColorXY(x, y, c);
|
|
|
|
|
else setPixelColor(x, c);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Blends the specified color with the existing pixel color.
|
2022-08-02 18:27:32 +02:00
|
|
|
|
void Segment::blendPixelColor(int n, uint32_t color, uint8_t blend) {
|
2022-07-10 22:23:25 +02:00
|
|
|
|
setPixelColor(n, color_blend(getPixelColor(n), color, blend));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Adds the specified color with the existing pixel color perserving color balance.
|
2023-04-27 17:31:55 +02:00
|
|
|
|
void Segment::addPixelColor(int n, uint32_t color, bool fast) {
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (!isActive()) return; // not active
|
2023-04-27 17:31:55 +02:00
|
|
|
|
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);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-02 18:27:32 +02:00
|
|
|
|
void Segment::fadePixelColor(uint16_t n, uint8_t fade) {
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (!isActive()) return; // not active
|
2022-08-02 18:27:32 +02:00
|
|
|
|
CRGB pix = CRGB(getPixelColor(n)).nscale8_video(fade);
|
|
|
|
|
setPixelColor(n, pix);
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-10 22:23:25 +02:00
|
|
|
|
/*
|
|
|
|
|
* fade out function, higher rate = quicker fade
|
|
|
|
|
*/
|
|
|
|
|
void Segment::fade_out(uint8_t rate) {
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (!isActive()) return; // not active
|
2022-07-30 14:20:36 +02:00
|
|
|
|
const uint16_t cols = is2D() ? virtualWidth() : virtualLength();
|
2022-07-10 22:23:25 +02:00
|
|
|
|
const uint16_t rows = virtualHeight(); // will be 1 for 1D
|
|
|
|
|
|
|
|
|
|
rate = (255-rate) >> 1;
|
|
|
|
|
float mappedRate = float(rate) +1.1;
|
|
|
|
|
|
|
|
|
|
uint32_t color = colors[1]; // SEGCOLOR(1); // target color
|
|
|
|
|
int w2 = W(color);
|
|
|
|
|
int r2 = R(color);
|
|
|
|
|
int g2 = G(color);
|
|
|
|
|
int b2 = B(color);
|
|
|
|
|
|
|
|
|
|
for (uint16_t y = 0; y < rows; y++) for (uint16_t x = 0; x < cols; x++) {
|
2022-07-30 14:20:36 +02:00
|
|
|
|
color = is2D() ? getPixelColorXY(x, y) : getPixelColor(x);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
int w1 = W(color);
|
|
|
|
|
int r1 = R(color);
|
|
|
|
|
int g1 = G(color);
|
|
|
|
|
int b1 = B(color);
|
|
|
|
|
|
|
|
|
|
int wdelta = (w2 - w1) / mappedRate;
|
|
|
|
|
int rdelta = (r2 - r1) / mappedRate;
|
|
|
|
|
int gdelta = (g2 - g1) / mappedRate;
|
|
|
|
|
int bdelta = (b2 - b1) / mappedRate;
|
|
|
|
|
|
|
|
|
|
// if fade isn't complete, make sure delta is at least 1 (fixes rounding issues)
|
|
|
|
|
wdelta += (w2 == w1) ? 0 : (w2 > w1) ? 1 : -1;
|
|
|
|
|
rdelta += (r2 == r1) ? 0 : (r2 > r1) ? 1 : -1;
|
|
|
|
|
gdelta += (g2 == g1) ? 0 : (g2 > g1) ? 1 : -1;
|
|
|
|
|
bdelta += (b2 == b1) ? 0 : (b2 > b1) ? 1 : -1;
|
|
|
|
|
|
2022-07-30 14:20:36 +02:00
|
|
|
|
if (is2D()) setPixelColorXY(x, y, r1 + rdelta, g1 + gdelta, b1 + bdelta, w1 + wdelta);
|
|
|
|
|
else setPixelColor(x, r1 + rdelta, g1 + gdelta, b1 + bdelta, w1 + wdelta);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// fades all pixels to black using nscale8()
|
|
|
|
|
void Segment::fadeToBlackBy(uint8_t fadeBy) {
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (!isActive() || fadeBy == 0) return; // optimization - no scaling to apply
|
2022-07-30 14:20:36 +02:00
|
|
|
|
const uint16_t cols = is2D() ? virtualWidth() : virtualLength();
|
2022-07-10 22:23:25 +02:00
|
|
|
|
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++) {
|
2022-07-30 14:20:36 +02:00
|
|
|
|
if (is2D()) setPixelColorXY(x, y, CRGB(getPixelColorXY(x,y)).nscale8(255-fadeBy));
|
|
|
|
|
else setPixelColor(x, CRGB(getPixelColor(x)).nscale8(255-fadeBy));
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* blurs segment content, source: FastLED colorutils.cpp
|
|
|
|
|
*/
|
|
|
|
|
void Segment::blur(uint8_t blur_amount)
|
|
|
|
|
{
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (!isActive() || blur_amount == 0) return; // optimization: 0 means "don't blur"
|
2022-07-30 14:20:36 +02:00
|
|
|
|
#ifndef WLED_DISABLE_2D
|
|
|
|
|
if (is2D()) {
|
2022-07-10 22:23:25 +02:00
|
|
|
|
// compatibility with 2D
|
2023-07-06 19:51:37 +02:00
|
|
|
|
const uint_fast16_t cols = virtualWidth();
|
|
|
|
|
const uint_fast16_t rows = virtualHeight();
|
|
|
|
|
for (uint_fast16_t 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
|
2022-07-10 22:23:25 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
2022-07-30 14:20:36 +02:00
|
|
|
|
#endif
|
2022-07-10 22:23:25 +02:00
|
|
|
|
uint8_t keep = 255 - blur_amount;
|
|
|
|
|
uint8_t seep = blur_amount >> 1;
|
|
|
|
|
CRGB carryover = CRGB::Black;
|
2023-07-06 19:51:37 +02:00
|
|
|
|
uint_fast16_t vlength = virtualLength();
|
|
|
|
|
for(uint_fast16_t i = 0; i < vlength; i++)
|
2022-07-10 22:23:25 +02:00
|
|
|
|
{
|
|
|
|
|
CRGB cur = CRGB(getPixelColor(i));
|
|
|
|
|
CRGB part = cur;
|
2023-07-06 19:51:37 +02:00
|
|
|
|
uint32_t before = uint32_t(cur); // remember color before blur
|
2022-07-10 22:23:25 +02:00
|
|
|
|
part.nscale8(seep);
|
|
|
|
|
cur.nscale8(keep);
|
|
|
|
|
cur += carryover;
|
|
|
|
|
if(i > 0) {
|
|
|
|
|
uint32_t c = getPixelColor(i-1);
|
|
|
|
|
uint8_t r = R(c);
|
|
|
|
|
uint8_t g = G(c);
|
|
|
|
|
uint8_t b = B(c);
|
2023-07-06 19:51:37 +02:00
|
|
|
|
setPixelColor((uint16_t)(i-1), qadd8(r, part.red), qadd8(g, part.green), qadd8(b, part.blue));
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
2023-07-06 19:51:37 +02:00
|
|
|
|
if (before != uint32_t(cur)) // optimization: only set pixel if color has changed
|
|
|
|
|
setPixelColor((uint16_t)i,cur.red, cur.green, cur.blue);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
carryover = part;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Put a value 0 to 255 in to get a color value.
|
|
|
|
|
* The colours are a transition r -> g -> b -> back to r
|
|
|
|
|
* Inspired by the Adafruit examples.
|
|
|
|
|
*/
|
2023-01-18 22:56:49 +01:00
|
|
|
|
uint32_t Segment::color_wheel(uint8_t pos) {
|
2022-07-10 22:23:25 +02:00
|
|
|
|
if (palette) return color_from_palette(pos, false, true, 0);
|
|
|
|
|
pos = 255 - pos;
|
|
|
|
|
if(pos < 85) {
|
|
|
|
|
return ((uint32_t)(255 - pos * 3) << 16) | ((uint32_t)(0) << 8) | (pos * 3);
|
|
|
|
|
} else if(pos < 170) {
|
|
|
|
|
pos -= 85;
|
|
|
|
|
return ((uint32_t)(0) << 16) | ((uint32_t)(pos * 3) << 8) | (255 - pos * 3);
|
|
|
|
|
} else {
|
|
|
|
|
pos -= 170;
|
|
|
|
|
return ((uint32_t)(pos * 3) << 16) | ((uint32_t)(255 - pos * 3) << 8) | (0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Returns a new, random wheel index with a minimum distance of 42 from pos.
|
|
|
|
|
*/
|
|
|
|
|
uint8_t 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.
|
|
|
|
|
* @param i Palette Index (if mapping is true, the full palette will be _virtualSegmentLength long, if false, 255). Will wrap around automatically.
|
|
|
|
|
* @param mapping if true, LED position in segment is considered for color
|
2023-01-11 11:21:45 +01:00
|
|
|
|
* @param wrap FastLED palettes will usually wrap back to the start smoothly. Set false to get a hard edge
|
2022-07-10 22:23:25 +02:00
|
|
|
|
* @param mcol If the default palette 0 is selected, return the standard color 0, 1 or 2 instead. If >2, Party palette is used instead
|
|
|
|
|
* @param pbri Value to scale the brightness of the returned color by. Default is 255. (no scaling)
|
|
|
|
|
* @returns Single color from palette
|
|
|
|
|
*/
|
2022-08-03 21:36:47 +02:00
|
|
|
|
uint32_t Segment::color_from_palette(uint16_t i, bool mapping, bool wrap, uint8_t mcol, uint8_t pbri)
|
2022-07-10 22:23:25 +02:00
|
|
|
|
{
|
2022-07-28 23:19:58 +02:00
|
|
|
|
// default palette or no RGB support on segment
|
2022-11-09 20:09:01 +01:00
|
|
|
|
if ((palette == 0 && mcol < NUM_COLORS) || !_isRGB) {
|
|
|
|
|
uint32_t color = currentColor(mcol, colors[mcol]);
|
2022-09-04 20:17:05 +02:00
|
|
|
|
color = gamma32(color);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
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));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t paletteIndex = i;
|
|
|
|
|
if (mapping && virtualLength() > 1) paletteIndex = (i*255)/(virtualLength() -1);
|
|
|
|
|
if (!wrap) paletteIndex = scale8(paletteIndex, 240); //cut off blend at palette "end"
|
|
|
|
|
CRGB fastled_col;
|
2022-07-29 16:26:15 +02:00
|
|
|
|
CRGBPalette16 curPal;
|
2022-07-30 14:50:11 +02:00
|
|
|
|
if (transitional && _t) curPal = _t->_palT;
|
|
|
|
|
else loadPalette(curPal, palette);
|
2022-07-28 23:19:58 +02:00
|
|
|
|
fastled_col = ColorFromPalette(curPal, paletteIndex, pbri, (strip.paletteBlend == 3)? NOBLEND:LINEARBLEND); // NOTE: paletteBlend should be global
|
2022-07-10 22:23:25 +02:00
|
|
|
|
|
|
|
|
|
return RGBW32(fastled_col.r, fastled_col.g, fastled_col.b, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// WS2812FX class implementation
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
2022-07-06 13:13:54 +02:00
|
|
|
|
|
2021-01-18 20:51:32 +01:00
|
|
|
|
//do not call this method from system context (network callback)
|
2021-02-24 20:23:32 +01:00
|
|
|
|
void WS2812FX::finalizeInit(void)
|
2019-02-09 16:37:20 +01:00
|
|
|
|
{
|
2022-02-09 08:43:35 +01:00
|
|
|
|
//reset segment runtimes
|
2022-07-17 15:58:41 +02:00
|
|
|
|
for (segment &seg : _segments) {
|
|
|
|
|
seg.markForReset();
|
|
|
|
|
seg.resetIfRequired();
|
2022-02-09 08:43:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-10 20:53:11 +02:00
|
|
|
|
// for the lack of better place enumerate ledmaps here
|
|
|
|
|
// if we do it in json.cpp (serializeInfo()) we are getting flashes on LEDs
|
|
|
|
|
// unfortunately this means we do not get updates after uploads
|
|
|
|
|
enumerateLedmaps();
|
|
|
|
|
|
2022-02-04 13:28:00 +01:00
|
|
|
|
_hasWhiteChannel = _isOffRefreshRequired = false;
|
2019-11-29 18:53:01 +01:00
|
|
|
|
|
2021-03-12 23:56:29 +01:00
|
|
|
|
//if busses failed to load, add default (fresh install, FS issue, ...)
|
2021-01-30 20:51:36 +01:00
|
|
|
|
if (busses.getNumBusses() == 0) {
|
2022-03-19 14:21:14 +01:00
|
|
|
|
DEBUG_PRINTLN(F("No busses, init default"));
|
2021-04-15 22:19:58 +02:00
|
|
|
|
const uint8_t defDataPins[] = {DATA_PINS};
|
|
|
|
|
const uint16_t defCounts[] = {PIXEL_COUNTS};
|
2022-04-12 16:08:17 +02:00
|
|
|
|
const uint8_t defNumBusses = ((sizeof defDataPins) / (sizeof defDataPins[0]));
|
|
|
|
|
const uint8_t defNumCounts = ((sizeof defCounts) / (sizeof defCounts[0]));
|
2021-04-15 10:55:22 +02:00
|
|
|
|
uint16_t prevLen = 0;
|
2022-12-31 17:06:18 +01:00
|
|
|
|
for (uint8_t i = 0; i < defNumBusses && i < WLED_MAX_BUSSES+WLED_MIN_VIRTUAL_BUSSES; i++) {
|
2021-04-15 10:55:22 +02:00
|
|
|
|
uint8_t defPin[] = {defDataPins[i]};
|
|
|
|
|
uint16_t start = prevLen;
|
2021-10-11 02:19:33 +02:00
|
|
|
|
uint16_t count = defCounts[(i < defNumCounts) ? i : defNumCounts -1];
|
2021-04-15 10:55:22 +02:00
|
|
|
|
prevLen += count;
|
2022-03-16 08:52:48 +01:00
|
|
|
|
BusConfig defCfg = BusConfig(DEFAULT_LED_TYPE, defPin, start, count, DEFAULT_LED_COLOR_ORDER, false, 0, RGBW_MODE_MANUAL_ONLY);
|
2022-12-31 17:06:18 +01:00
|
|
|
|
if (busses.add(defCfg) == -1) break;
|
2021-04-15 10:55:22 +02:00
|
|
|
|
}
|
2021-01-30 20:51:36 +01:00
|
|
|
|
}
|
2021-01-18 20:51:32 +01:00
|
|
|
|
|
2021-02-24 20:23:32 +01:00
|
|
|
|
_length = 0;
|
|
|
|
|
for (uint8_t i=0; i<busses.getNumBusses(); i++) {
|
|
|
|
|
Bus *bus = busses.getBus(i);
|
|
|
|
|
if (bus == nullptr) continue;
|
2021-10-11 02:19:33 +02:00
|
|
|
|
if (bus->getStart() + bus->getLength() > MAX_LEDS) break;
|
2021-05-22 00:13:49 +02:00
|
|
|
|
//RGBW mode is enabled if at least one of the strips is RGBW
|
2023-02-14 01:33:06 +01:00
|
|
|
|
_hasWhiteChannel |= bus->hasWhite();
|
2021-05-22 00:13:49 +02:00
|
|
|
|
//refresh is required to remain off if at least one of the strips requires the refresh.
|
2022-02-04 13:28:00 +01:00
|
|
|
|
_isOffRefreshRequired |= bus->isOffRefreshRequired();
|
2021-10-11 02:19:33 +02:00
|
|
|
|
uint16_t busEnd = bus->getStart() + bus->getLength();
|
|
|
|
|
if (busEnd > _length) _length = busEnd;
|
2021-06-24 02:29:14 +02:00
|
|
|
|
#ifdef ESP8266
|
2021-10-11 02:19:33 +02:00
|
|
|
|
if ((!IS_DIGITAL(bus->getType()) || IS_2PIN(bus->getType()))) continue;
|
2021-01-19 17:22:37 +01:00
|
|
|
|
uint8_t pins[5];
|
2021-10-11 02:19:33 +02:00
|
|
|
|
if (!bus->getPins(pins)) continue;
|
|
|
|
|
BusDigital* bd = static_cast<BusDigital*>(bus);
|
2021-01-19 17:22:37 +01:00
|
|
|
|
if (pins[0] == 3) bd->reinit();
|
2021-06-24 02:29:14 +02:00
|
|
|
|
#endif
|
2021-02-24 20:23:32 +01:00
|
|
|
|
}
|
2021-09-12 01:00:54 +02:00
|
|
|
|
|
2023-02-09 20:15:55 +01:00
|
|
|
|
if (isMatrix) setUpMatrix();
|
|
|
|
|
else {
|
2022-12-22 18:13:32 +01:00
|
|
|
|
Segment::maxWidth = _length;
|
|
|
|
|
Segment::maxHeight = 1;
|
2023-02-09 20:15:55 +01:00
|
|
|
|
}
|
2022-12-22 18:13:32 +01:00
|
|
|
|
|
2022-08-05 23:03:38 +02:00
|
|
|
|
//segments are created in makeAutoSegments();
|
2023-02-12 13:18:30 +01:00
|
|
|
|
DEBUG_PRINTLN(F("Loading custom palettes"));
|
2022-11-16 20:55:21 +01:00
|
|
|
|
loadCustomPalettes(); // (re)load all custom palettes
|
2023-02-12 13:18:30 +01:00
|
|
|
|
DEBUG_PRINTLN(F("Loading custom ledmaps"));
|
2022-11-16 20:55:21 +01:00
|
|
|
|
deserializeMap(); // (re)load default ledmap
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void WS2812FX::service() {
|
2023-07-12 20:52:34 +02:00
|
|
|
|
unsigned long nowUp = millis(); // Be aware, millis() rolls over every 49 days
|
2019-11-26 20:41:15 +01:00
|
|
|
|
now = nowUp + timebase;
|
|
|
|
|
if (nowUp - _lastShow < MIN_SHOW_DELAY) return;
|
2019-02-11 23:49:04 +01:00
|
|
|
|
bool doShow = false;
|
2020-05-10 23:58:50 +02:00
|
|
|
|
|
2022-07-19 16:16:43 +02:00
|
|
|
|
_isServicing = true;
|
2022-07-17 15:58:41 +02:00
|
|
|
|
_segment_index = 0;
|
|
|
|
|
for (segment &seg : _segments) {
|
2023-06-04 18:36:46 +02:00
|
|
|
|
// process transition (mode changes in the middle of transition)
|
|
|
|
|
seg.handleTransition();
|
2022-07-19 16:16:43 +02:00
|
|
|
|
// reset the segment runtime data if needed
|
2022-07-17 15:58:41 +02:00
|
|
|
|
seg.resetIfRequired();
|
2020-12-10 02:55:14 +01:00
|
|
|
|
|
2022-03-10 20:40:48 +01:00
|
|
|
|
// last condition ensures all solid segments are updated at the same time
|
2023-07-13 13:08:36 +02:00
|
|
|
|
if (seg.isActive() && (nowUp > seg.next_time || _triggered || (doShow && seg.mode == FX_MODE_STATIC)))
|
2019-02-09 16:37:20 +01:00
|
|
|
|
{
|
2023-07-06 21:16:29 +02:00
|
|
|
|
doShow = true;
|
2021-01-09 00:35:48 +01:00
|
|
|
|
uint16_t delay = FRAMETIME;
|
|
|
|
|
|
2022-08-06 12:39:12 +02:00
|
|
|
|
if (!seg.freeze) { //only run effect function if not frozen
|
2022-07-10 22:23:25 +02:00
|
|
|
|
_virtualSegmentLength = seg.virtualLength();
|
2022-07-30 14:20:36 +02:00
|
|
|
|
_colors_t[0] = seg.currentColor(0, seg.colors[0]);
|
|
|
|
|
_colors_t[1] = seg.currentColor(1, seg.colors[1]);
|
|
|
|
|
_colors_t[2] = seg.currentColor(2, seg.colors[2]);
|
2022-07-29 12:15:56 +02:00
|
|
|
|
seg.currentPalette(_currentPalette, seg.palette);
|
2022-07-28 23:19:58 +02:00
|
|
|
|
|
2022-07-30 14:20:36 +02:00
|
|
|
|
if (!cctFromRgb || correctWB) busses.setSegmentCCT(seg.currentBri(seg.cct, true), correctWB);
|
2022-07-30 23:58:29 +02:00
|
|
|
|
for (uint8_t c = 0; c < NUM_COLORS; c++) _colors_t[c] = gamma32(_colors_t[c]);
|
2022-03-08 02:16:33 +01:00
|
|
|
|
|
2022-07-29 12:15:56 +02:00
|
|
|
|
// effect blending (execute previous effect)
|
|
|
|
|
// actual code may be a bit more involved as effects have runtime data including allocated memory
|
2022-08-06 12:39:12 +02:00
|
|
|
|
//if (seg.transitional && seg._modeP) (*_mode[seg._modeP])(progress());
|
2022-08-03 22:09:27 +02:00
|
|
|
|
delay = (*_mode[seg.currentMode(seg.mode)])();
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (seg.mode != FX_MODE_HALLOWEEN_EYES) seg.call++;
|
2023-01-06 18:11:52 +01:00
|
|
|
|
if (seg.transitional && delay > FRAMETIME) delay = FRAMETIME; // force faster updates during transition
|
2019-06-20 14:40:12 +02:00
|
|
|
|
}
|
2021-01-09 00:35:48 +01:00
|
|
|
|
|
2022-07-17 15:58:41 +02:00
|
|
|
|
seg.next_time = nowUp + delay;
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
2023-07-13 13:08:36 +02:00
|
|
|
|
if (_segment_index == _queuedChangesSegId) setUpSegmentFromQueuedChanges();
|
2022-07-17 15:58:41 +02:00
|
|
|
|
_segment_index++;
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
2022-07-06 13:13:54 +02:00
|
|
|
|
_virtualSegmentLength = 0;
|
2021-11-24 11:02:25 +01:00
|
|
|
|
busses.setSegmentCCT(-1);
|
2023-07-12 20:52:34 +02:00
|
|
|
|
_isServicing = false;
|
|
|
|
|
_triggered = false;
|
|
|
|
|
|
|
|
|
|
#ifdef WLED_DEBUG
|
|
|
|
|
if (millis() - nowUp > _frametime) DEBUG_PRINTLN(F("Slow effects."));
|
|
|
|
|
#endif
|
|
|
|
|
if (doShow) {
|
2019-03-05 10:59:15 +01:00
|
|
|
|
yield();
|
2019-02-11 23:49:04 +01:00
|
|
|
|
show();
|
|
|
|
|
}
|
2023-07-12 20:52:34 +02:00
|
|
|
|
#ifdef WLED_DEBUG
|
|
|
|
|
if (millis() - nowUp > _frametime) DEBUG_PRINTLN(F("Slow strip."));
|
|
|
|
|
#endif
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-03 21:36:47 +02:00
|
|
|
|
void IRAM_ATTR WS2812FX::setPixelColor(int i, uint32_t col)
|
2019-02-09 16:37:20 +01:00
|
|
|
|
{
|
2022-08-03 21:36:47 +02:00
|
|
|
|
if (i < customMappingSize) i = customMappingTable[i];
|
2023-03-11 15:03:28 +01:00
|
|
|
|
if (i >= _length) return;
|
2023-06-30 21:12:59 +02:00
|
|
|
|
busses.setPixelColor(i, col);
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-10 22:23:25 +02:00
|
|
|
|
uint32_t WS2812FX::getPixelColor(uint16_t i)
|
|
|
|
|
{
|
|
|
|
|
if (i < customMappingSize) i = customMappingTable[i];
|
2023-03-11 15:03:28 +01:00
|
|
|
|
if (i >= _length) return 0;
|
2022-07-10 22:23:25 +02:00
|
|
|
|
return busses.getPixelColor(i);
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-09 16:37:20 +01:00
|
|
|
|
|
|
|
|
|
//DISCLAIMER
|
|
|
|
|
//The following function attemps to calculate the current LED power usage,
|
|
|
|
|
//and will limit the brightness to stay below a set amperage threshold.
|
|
|
|
|
//It is NOT a measurement and NOT guaranteed to stay within the ablMilliampsMax margin.
|
|
|
|
|
//Stay safe with high amperage and have a reasonable safety margin!
|
|
|
|
|
//I am NOT to be held liable for burned down garages!
|
|
|
|
|
|
2023-01-06 09:10:39 +01:00
|
|
|
|
//fine tune power estimation constants for your setup
|
2019-02-09 16:37:20 +01:00
|
|
|
|
#define MA_FOR_ESP 100 //how much mA does the ESP use (Wemos D1 about 80mA, ESP32 about 120mA)
|
|
|
|
|
//you can set it to 0 if the ESP is powered by USB and the LEDs by external
|
|
|
|
|
|
2023-06-26 22:12:32 +02:00
|
|
|
|
uint8_t WS2812FX::estimateCurrentAndLimitBri() {
|
2019-02-09 16:37:20 +01:00
|
|
|
|
//power limit calculation
|
|
|
|
|
//each LED can draw up 195075 "power units" (approx. 53mA)
|
|
|
|
|
//one PU is the power it takes to have 1 channel 1 step brighter per brightness step
|
|
|
|
|
//so A=2,R=255,G=0,B=0 would use 510 PU per LED (1mA is about 3700 PU)
|
2020-01-13 18:09:20 +01:00
|
|
|
|
bool useWackyWS2815PowerModel = false;
|
|
|
|
|
byte actualMilliampsPerLed = milliampsPerLed;
|
|
|
|
|
|
2021-10-11 02:19:53 +02:00
|
|
|
|
if (ablMilliampsMax < 150 || actualMilliampsPerLed == 0) { //0 mA per LED and too low numbers turn off calculation
|
|
|
|
|
currentMilliamps = 0;
|
2023-06-26 22:12:32 +02:00
|
|
|
|
return _brightness;
|
2021-10-11 02:19:53 +02:00
|
|
|
|
}
|
2019-02-09 16:37:20 +01:00
|
|
|
|
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (milliampsPerLed == 255) {
|
|
|
|
|
useWackyWS2815PowerModel = true;
|
|
|
|
|
actualMilliampsPerLed = 12; // from testing an actual strip
|
2021-10-11 02:19:53 +02:00
|
|
|
|
}
|
2019-02-09 16:37:20 +01:00
|
|
|
|
|
2023-07-12 20:52:34 +02:00
|
|
|
|
size_t powerBudget = (ablMilliampsMax - MA_FOR_ESP); //100mA for ESP power
|
2021-10-11 02:19:53 +02:00
|
|
|
|
|
2023-07-12 20:52:34 +02:00
|
|
|
|
size_t pLen = 0; //getLengthPhysical();
|
|
|
|
|
size_t powerSum = 0;
|
2023-04-14 13:09:25 +02:00
|
|
|
|
for (uint_fast8_t bNum = 0; bNum < busses.getNumBusses(); bNum++) {
|
|
|
|
|
Bus *bus = busses.getBus(bNum);
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (!IS_DIGITAL(bus->getType())) continue; //exclude non-digital network busses
|
2021-10-11 02:19:53 +02:00
|
|
|
|
uint16_t len = bus->getLength();
|
2023-07-12 20:52:34 +02:00
|
|
|
|
pLen += len;
|
2021-10-11 02:19:53 +02:00
|
|
|
|
uint32_t busPowerSum = 0;
|
2023-04-14 13:09:25 +02:00
|
|
|
|
for (uint_fast16_t i = 0; i < len; i++) { //sum up the usage of each LED
|
2023-07-09 12:32:28 +02:00
|
|
|
|
uint32_t c = bus->getPixelColor(i); // always returns original or restored color without brightness scaling
|
2021-10-26 20:35:45 +02:00
|
|
|
|
byte r = R(c), g = G(c), b = B(c), w = W(c);
|
2023-07-09 12:32:28 +02:00
|
|
|
|
|
2021-10-11 02:19:53 +02:00
|
|
|
|
if(useWackyWS2815PowerModel) { //ignore white component on WS2815 power calculation
|
|
|
|
|
busPowerSum += (MAX(MAX(r,g),b)) * 3;
|
|
|
|
|
} else {
|
|
|
|
|
busPowerSum += (r + g + b + w);
|
2020-01-13 18:09:20 +01:00
|
|
|
|
}
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
2023-02-14 01:33:06 +01:00
|
|
|
|
if (bus->hasWhite()) { //RGBW led total output with white LEDs enabled is still 50mA, so each channel uses less
|
2021-10-11 02:19:53 +02:00
|
|
|
|
busPowerSum *= 3;
|
2023-07-12 20:52:34 +02:00
|
|
|
|
busPowerSum >>= 2; //same as /= 4
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
2021-10-11 02:19:53 +02:00
|
|
|
|
powerSum += busPowerSum;
|
|
|
|
|
}
|
2019-02-09 16:37:20 +01:00
|
|
|
|
|
2023-07-12 20:52:34 +02:00
|
|
|
|
if (powerBudget > pLen) { //each LED uses about 1mA in standby, exclude that from power budget
|
|
|
|
|
powerBudget -= pLen;
|
|
|
|
|
} else {
|
|
|
|
|
powerBudget = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// powerSum has all the values of channels summed (max would be pLen*765 as white is excluded) so convert to milliAmps
|
|
|
|
|
powerSum = (powerSum * actualMilliampsPerLed) / 765;
|
2023-07-09 12:32:28 +02:00
|
|
|
|
|
2023-07-12 20:52:34 +02:00
|
|
|
|
uint8_t newBri = _brightness;
|
|
|
|
|
if (powerSum * _brightness / 255 > powerBudget) { //scale brightness down to stay in current limit
|
2023-07-17 16:15:17 +02:00
|
|
|
|
float scale = (float)(powerBudget * 255) / (float)(powerSum * _brightness);
|
2021-10-11 02:19:53 +02:00
|
|
|
|
uint16_t scaleI = scale * 255;
|
|
|
|
|
uint8_t scaleB = (scaleI > 255) ? 255 : scaleI;
|
2023-07-17 16:15:17 +02:00
|
|
|
|
newBri = scale8(_brightness, scaleB) + 1;
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
2023-07-12 20:52:34 +02:00
|
|
|
|
currentMilliamps = (powerSum * newBri) / 255;
|
2021-10-11 02:19:53 +02:00
|
|
|
|
currentMilliamps += MA_FOR_ESP; //add power of ESP back to estimate
|
2023-07-12 20:52:34 +02:00
|
|
|
|
currentMilliamps += pLen; //add standby power (1mA/LED) back to estimate
|
2023-06-26 22:12:32 +02:00
|
|
|
|
return newBri;
|
2021-10-11 02:19:53 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void WS2812FX::show(void) {
|
2023-07-12 20:52:34 +02:00
|
|
|
|
// avoid race condition, caputre _callback value
|
|
|
|
|
show_callback callback = _callback;
|
|
|
|
|
if (callback) callback();
|
|
|
|
|
|
2023-07-05 23:57:46 +02:00
|
|
|
|
#ifdef WLED_DEBUG
|
2023-07-03 21:13:01 +02:00
|
|
|
|
static unsigned long sumMicros = 0, sumCurrent = 0;
|
|
|
|
|
static size_t calls = 0;
|
|
|
|
|
unsigned long microsStart = micros();
|
2023-07-05 23:57:46 +02:00
|
|
|
|
#endif
|
2021-10-11 02:19:53 +02:00
|
|
|
|
|
2023-07-17 17:06:04 +02:00
|
|
|
|
busses.setBrightness(estimateCurrentAndLimitBri());
|
2023-07-05 23:57:46 +02:00
|
|
|
|
#ifdef WLED_DEBUG
|
2023-07-03 21:13:01 +02:00
|
|
|
|
sumCurrent += micros() - microsStart;
|
2023-07-05 23:57:46 +02:00
|
|
|
|
#endif
|
2023-01-06 09:10:39 +01:00
|
|
|
|
|
2020-12-10 05:29:53 +01:00
|
|
|
|
// some buses send asynchronously and this method will return before
|
|
|
|
|
// all of the data has been sent.
|
|
|
|
|
// See https://github.com/Makuna/NeoPixelBus/wiki/ESP32-NeoMethods#neoesp32rmt-methods
|
2021-01-21 01:21:16 +01:00
|
|
|
|
busses.show();
|
2023-07-03 21:13:01 +02:00
|
|
|
|
|
2023-07-05 23:57:46 +02:00
|
|
|
|
#ifdef WLED_DEBUG
|
2023-07-03 21:13:01 +02:00
|
|
|
|
sumMicros += micros() - microsStart;
|
|
|
|
|
if (++calls == 100) {
|
2023-07-12 20:52:34 +02:00
|
|
|
|
DEBUG_PRINTF("%d show calls: %lu[us] avg: %lu[us] (current: %lu[us] avg: %lu[us])\n", calls, sumMicros, sumMicros/calls, sumCurrent, sumCurrent/calls);
|
2023-07-03 21:13:01 +02:00
|
|
|
|
sumMicros = sumCurrent = 0;
|
|
|
|
|
calls = 0;
|
|
|
|
|
}
|
2023-07-05 23:57:46 +02:00
|
|
|
|
#endif
|
2023-07-12 20:52:34 +02:00
|
|
|
|
|
|
|
|
|
unsigned long now = millis();
|
|
|
|
|
size_t diff = now - _lastShow;
|
|
|
|
|
size_t fpsCurr = 200;
|
|
|
|
|
if (diff > 0) fpsCurr = 1000 / diff;
|
2023-07-13 12:49:19 +02:00
|
|
|
|
_cumulativeFps = (3 * _cumulativeFps + fpsCurr +2) >> 2; // "+2" for proper rounding (2/4 = 0.5)
|
2023-07-12 20:52:34 +02:00
|
|
|
|
_lastShow = now;
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-12-10 05:29:53 +01:00
|
|
|
|
/**
|
|
|
|
|
* Returns a true value if any of the strips are still being updated.
|
|
|
|
|
* On some hardware (ESP32), strip updates are done asynchronously.
|
|
|
|
|
*/
|
|
|
|
|
bool WS2812FX::isUpdating() {
|
2021-01-21 01:21:16 +01:00
|
|
|
|
return !busses.canAllShow();
|
2020-12-10 05:29:53 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-02-05 01:33:26 +01:00
|
|
|
|
/**
|
|
|
|
|
* Returns the refresh rate of the LED strip. Useful for finding out whether a given setup is fast enough.
|
|
|
|
|
* Only updates on show() or is set to 0 fps if last show is more than 2 secs ago, so accurary varies
|
|
|
|
|
*/
|
|
|
|
|
uint16_t WS2812FX::getFps() {
|
|
|
|
|
if (millis() - _lastShow > 2000) return 0;
|
|
|
|
|
return _cumulativeFps +1;
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-25 01:30:27 +01:00
|
|
|
|
void WS2812FX::setTargetFps(uint8_t fps) {
|
2022-08-24 23:04:51 +02:00
|
|
|
|
if (fps > 0 && fps <= 120) _targetFps = fps;
|
|
|
|
|
_frametime = 1000 / _targetFps;
|
2021-12-25 01:30:27 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-06-20 14:40:12 +02:00
|
|
|
|
void WS2812FX::setMode(uint8_t segid, uint8_t m) {
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (segid >= _segments.size()) return;
|
2023-01-06 09:10:39 +01:00
|
|
|
|
|
2022-06-26 23:01:22 +02:00
|
|
|
|
if (m >= getModeCount()) m = getModeCount() - 1;
|
2019-06-20 14:40:12 +02:00
|
|
|
|
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (_segments[segid].mode != m) {
|
2022-08-05 23:03:38 +02:00
|
|
|
|
_segments[segid].startTransition(_transitionDur); // set effect transitions
|
|
|
|
|
//_segments[segid].markForReset();
|
2019-06-20 14:40:12 +02:00
|
|
|
|
_segments[segid].mode = m;
|
|
|
|
|
}
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
2022-02-20 22:24:11 +01:00
|
|
|
|
//applies to all active and selected segments
|
2019-06-20 14:40:12 +02:00
|
|
|
|
void WS2812FX::setColor(uint8_t slot, uint32_t c) {
|
|
|
|
|
if (slot >= NUM_COLORS) return;
|
2020-03-14 11:28:42 +01:00
|
|
|
|
|
2022-07-17 15:58:41 +02:00
|
|
|
|
for (segment &seg : _segments) {
|
2022-09-16 14:07:04 +02:00
|
|
|
|
if (seg.isActive() && seg.isSelected()) {
|
2022-07-17 15:58:41 +02:00
|
|
|
|
seg.setColor(slot, c);
|
2019-12-02 12:41:35 +01:00
|
|
|
|
}
|
2020-03-14 11:28:42 +01:00
|
|
|
|
}
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
2022-02-23 19:20:07 +01:00
|
|
|
|
void WS2812FX::setCCT(uint16_t k) {
|
2022-07-17 15:58:41 +02:00
|
|
|
|
for (segment &seg : _segments) {
|
|
|
|
|
if (seg.isActive() && seg.isSelected()) {
|
|
|
|
|
seg.setCCT(k);
|
2022-02-23 19:20:07 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-10 20:40:48 +01:00
|
|
|
|
void WS2812FX::setBrightness(uint8_t b, bool direct) {
|
2020-12-14 23:32:57 +01:00
|
|
|
|
if (gammaCorrectBri) b = gamma8(b);
|
2019-02-09 16:37:20 +01:00
|
|
|
|
if (_brightness == b) return;
|
2020-12-14 23:32:57 +01:00
|
|
|
|
_brightness = b;
|
|
|
|
|
if (_brightness == 0) { //unfreeze all segments on power off
|
2022-07-17 15:58:41 +02:00
|
|
|
|
for (segment &seg : _segments) {
|
2022-08-19 21:14:49 +02:00
|
|
|
|
seg.freeze = false;
|
2020-08-29 22:26:39 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2022-03-10 20:40:48 +01:00
|
|
|
|
if (direct) {
|
|
|
|
|
// would be dangerous if applied immediately (could exceed ABL), but will not output until the next show()
|
|
|
|
|
busses.setBrightness(b);
|
|
|
|
|
} else {
|
2022-08-24 23:04:51 +02:00
|
|
|
|
unsigned long t = millis();
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (_segments[0].next_time > t + 22 && t - _lastShow > MIN_SHOW_DELAY) show(); //apply brightness change immediately if no refresh soon
|
2022-03-10 20:40:48 +01:00
|
|
|
|
}
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
2023-05-31 20:12:17 +02:00
|
|
|
|
uint8_t WS2812FX::getActiveSegsLightCapabilities(bool selectedOnly) {
|
|
|
|
|
uint8_t totalLC = 0;
|
|
|
|
|
for (segment &seg : _segments) {
|
|
|
|
|
if (seg.isActive() && (!selectedOnly || seg.isSelected())) totalLC |= seg.getLightCapabilities();
|
|
|
|
|
}
|
|
|
|
|
return totalLC;
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-23 19:20:07 +01:00
|
|
|
|
uint8_t WS2812FX::getFirstSelectedSegId(void)
|
|
|
|
|
{
|
2022-07-17 15:58:41 +02:00
|
|
|
|
size_t i = 0;
|
|
|
|
|
for (segment &seg : _segments) {
|
2022-09-16 14:07:04 +02:00
|
|
|
|
if (seg.isActive() && seg.isSelected()) return i;
|
2022-07-17 15:58:41 +02:00
|
|
|
|
i++;
|
2019-06-20 14:40:12 +02:00
|
|
|
|
}
|
2022-02-23 19:20:07 +01:00
|
|
|
|
// if none selected, use the main segment
|
|
|
|
|
return getMainSegmentId();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void WS2812FX::setMainSegmentId(uint8_t n) {
|
2022-02-20 22:24:11 +01:00
|
|
|
|
_mainSegment = 0;
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (n < _segments.size()) {
|
|
|
|
|
_mainSegment = n;
|
|
|
|
|
}
|
2022-02-20 22:24:11 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2019-06-20 14:40:12 +02:00
|
|
|
|
|
2022-02-20 22:24:11 +01:00
|
|
|
|
uint8_t WS2812FX::getLastActiveSegmentId(void) {
|
2022-08-05 23:03:38 +02:00
|
|
|
|
for (size_t i = _segments.size() -1; i > 0; i--) {
|
|
|
|
|
if (_segments[i].isActive()) return i;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
2019-06-20 14:40:12 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-19 16:16:43 +02:00
|
|
|
|
uint8_t WS2812FX::getActiveSegmentsNum(void) {
|
|
|
|
|
uint8_t c = 0;
|
2022-07-19 22:14:46 +02:00
|
|
|
|
for (size_t i = 0; i < _segments.size(); i++) {
|
2022-07-19 16:16:43 +02:00
|
|
|
|
if (_segments[i].isActive()) c++;
|
|
|
|
|
}
|
|
|
|
|
return c;
|
|
|
|
|
}
|
2021-09-20 21:22:50 +02:00
|
|
|
|
|
2023-03-11 15:03:28 +01:00
|
|
|
|
uint16_t WS2812FX::getLengthTotal(void) {
|
|
|
|
|
uint16_t len = Segment::maxWidth * Segment::maxHeight; // will be _length for 1D (see finalizeInit()) but should cover whole matrix for 2D
|
|
|
|
|
if (isMatrix && _length > len) len = _length; // for 2D with trailing strip
|
|
|
|
|
return len;
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-11 02:19:53 +02:00
|
|
|
|
uint16_t WS2812FX::getLengthPhysical(void) {
|
|
|
|
|
uint16_t len = 0;
|
2022-08-05 23:03:38 +02:00
|
|
|
|
for (size_t b = 0; b < busses.getNumBusses(); b++) {
|
2021-10-11 02:19:53 +02:00
|
|
|
|
Bus *bus = busses.getBus(b);
|
|
|
|
|
if (bus->getType() >= TYPE_NET_DDP_RGB) continue; //exclude non-physical network busses
|
|
|
|
|
len += bus->getLength();
|
|
|
|
|
}
|
|
|
|
|
return len;
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-20 22:24:11 +01:00
|
|
|
|
//used for JSON API info.leds.rgbw. Little practical use, deprecate with info.leds.rgbw.
|
|
|
|
|
//returns if there is an RGBW bus (supports RGB and White, not only white)
|
|
|
|
|
//not influenced by auto-white mode, also true if white slider does not affect output white channel
|
|
|
|
|
bool WS2812FX::hasRGBWBus(void) {
|
2022-08-24 23:04:51 +02:00
|
|
|
|
for (size_t b = 0; b < busses.getNumBusses(); b++) {
|
2022-02-20 22:24:11 +01:00
|
|
|
|
Bus *bus = busses.getBus(b);
|
|
|
|
|
if (bus == nullptr || bus->getLength()==0) break;
|
2023-02-14 01:33:06 +01:00
|
|
|
|
if (bus->hasRGB() && bus->hasWhite()) return true;
|
2022-02-20 22:24:11 +01:00
|
|
|
|
}
|
2022-08-24 23:04:51 +02:00
|
|
|
|
return false;
|
2022-02-20 22:24:11 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-11-28 04:01:58 +01:00
|
|
|
|
bool WS2812FX::hasCCTBus(void) {
|
2022-08-24 23:04:51 +02:00
|
|
|
|
if (cctFromRgb && !correctWB) return false;
|
|
|
|
|
for (size_t b = 0; b < busses.getNumBusses(); b++) {
|
2021-11-28 04:01:58 +01:00
|
|
|
|
Bus *bus = busses.getBus(b);
|
|
|
|
|
if (bus == nullptr || bus->getLength()==0) break;
|
|
|
|
|
switch (bus->getType()) {
|
|
|
|
|
case TYPE_ANALOG_5CH:
|
|
|
|
|
case TYPE_ANALOG_2CH:
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-08-24 23:04:51 +02:00
|
|
|
|
return false;
|
2021-11-28 04:01:58 +01:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-03 21:36:47 +02:00
|
|
|
|
void WS2812FX::purgeSegments(bool force) {
|
2022-07-17 15:58:41 +02:00
|
|
|
|
// remove all inactive segments (from the back)
|
|
|
|
|
int deleted = 0;
|
2022-08-03 21:36:47 +02:00
|
|
|
|
if (_segments.size() <= 1) return;
|
2022-08-05 23:03:38 +02:00
|
|
|
|
for (size_t i = _segments.size()-1; i > 0; i--)
|
2022-08-03 21:36:47 +02:00
|
|
|
|
if (_segments[i].stop == 0 || force) {
|
2022-07-19 16:16:43 +02:00
|
|
|
|
deleted++;
|
|
|
|
|
_segments.erase(_segments.begin() + i);
|
|
|
|
|
}
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (deleted) {
|
|
|
|
|
_segments.shrink_to_fit();
|
|
|
|
|
if (_mainSegment >= _segments.size()) setMainSegmentId(0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-19 16:16:43 +02:00
|
|
|
|
Segment& WS2812FX::getSegment(uint8_t id) {
|
|
|
|
|
return _segments[id >= _segments.size() ? getMainSegmentId() : id]; // vectors
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-13 13:08:36 +02:00
|
|
|
|
// sets new segment bounds, queues if that segment is currently running
|
|
|
|
|
void WS2812FX::setSegment(uint8_t segId, uint16_t i1, uint16_t i2, uint8_t grouping, uint8_t spacing, uint16_t offset, uint16_t startY, uint16_t stopY) {
|
|
|
|
|
if (segId >= getSegmentsNum()) {
|
|
|
|
|
if (i2 <= i1) return; // do not append empty/inactive segments
|
|
|
|
|
appendSegment(Segment(0, strip.getLengthTotal()));
|
|
|
|
|
segId = getSegmentsNum()-1; // segments are added at the end of list
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (_queuedChangesSegId == segId) _queuedChangesSegId = 255; // cancel queued change if already queued for this segment
|
|
|
|
|
|
|
|
|
|
if (segId < getMaxSegments() && segId == getCurrSegmentId() && isServicing()) { // queue change to prevent concurrent access
|
|
|
|
|
// queuing a change for a second segment will lead to the loss of the first change if not yet applied
|
|
|
|
|
// however this is not a problem as the queued change is applied immediately after the effect function in that segment returns
|
|
|
|
|
_qStart = i1; _qStop = i2; _qStartY = startY; _qStopY = stopY;
|
|
|
|
|
_qGrouping = grouping; _qSpacing = spacing; _qOffset = offset;
|
|
|
|
|
_queuedChangesSegId = segId;
|
|
|
|
|
return; // queued changes are applied immediately after effect function returns
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_segments[segId].setUp(i1, i2, grouping, spacing, offset, startY, stopY);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void WS2812FX::setUpSegmentFromQueuedChanges() {
|
|
|
|
|
if (_queuedChangesSegId >= getSegmentsNum()) return;
|
|
|
|
|
getSegment(_queuedChangesSegId).setUp(_qStart, _qStop, _qGrouping, _qSpacing, _qOffset, _qStartY, _qStopY);
|
|
|
|
|
_queuedChangesSegId = 255;
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-20 11:29:03 +01:00
|
|
|
|
void WS2812FX::restartRuntime() {
|
2022-07-17 15:58:41 +02:00
|
|
|
|
for (segment &seg : _segments) seg.markForReset();
|
2021-12-20 11:29:03 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-09 16:37:20 +01:00
|
|
|
|
void WS2812FX::resetSegments() {
|
2022-07-17 15:58:41 +02:00
|
|
|
|
_segments.clear(); // destructs all Segment as part of clearing
|
2022-07-30 10:49:54 +02:00
|
|
|
|
#ifndef WLED_DISABLE_2D
|
2022-12-16 22:31:07 +01:00
|
|
|
|
segment seg = isMatrix ? Segment(0, Segment::maxWidth, 0, Segment::maxHeight) : Segment(0, _length);
|
2022-07-30 10:49:54 +02:00
|
|
|
|
#else
|
|
|
|
|
segment seg = Segment(0, _length);
|
|
|
|
|
#endif
|
2022-07-17 15:58:41 +02:00
|
|
|
|
_segments.push_back(seg);
|
|
|
|
|
_mainSegment = 0;
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
2022-02-10 16:09:16 +01:00
|
|
|
|
void WS2812FX::makeAutoSegments(bool forceReset) {
|
2023-02-25 09:41:15 +01:00
|
|
|
|
if (autoSegments) { //make one segment per bus
|
|
|
|
|
uint16_t segStarts[MAX_NUM_SEGMENTS] = {0};
|
|
|
|
|
uint16_t segStops [MAX_NUM_SEGMENTS] = {0};
|
|
|
|
|
size_t s = 0;
|
|
|
|
|
|
2022-07-10 22:23:25 +02:00
|
|
|
|
#ifndef WLED_DISABLE_2D
|
2023-02-25 09:41:15 +01:00
|
|
|
|
// 2D segment is the 1st one using entire matrix
|
|
|
|
|
if (isMatrix) {
|
|
|
|
|
segStarts[0] = 0;
|
|
|
|
|
segStops[0] = Segment::maxWidth*Segment::maxHeight;
|
|
|
|
|
s++;
|
2023-02-01 19:30:56 +01:00
|
|
|
|
}
|
2022-07-10 22:23:25 +02:00
|
|
|
|
#endif
|
2023-02-25 09:41:15 +01:00
|
|
|
|
|
|
|
|
|
for (size_t i = s; i < busses.getNumBusses(); i++) {
|
2021-10-11 02:19:33 +02:00
|
|
|
|
Bus* b = busses.getBus(i);
|
|
|
|
|
|
|
|
|
|
segStarts[s] = b->getStart();
|
2022-07-17 15:58:41 +02:00
|
|
|
|
segStops[s] = segStarts[s] + b->getLength();
|
2021-10-11 02:19:33 +02:00
|
|
|
|
|
2023-02-25 09:41:15 +01:00
|
|
|
|
#ifndef WLED_DISABLE_2D
|
|
|
|
|
if (isMatrix && segStops[s] < Segment::maxWidth*Segment::maxHeight) continue; // ignore buses comprising matrix
|
|
|
|
|
if (isMatrix && segStarts[s] < Segment::maxWidth*Segment::maxHeight) segStarts[s] = Segment::maxWidth*Segment::maxHeight;
|
|
|
|
|
#endif
|
|
|
|
|
|
2021-10-11 02:19:33 +02:00
|
|
|
|
//check for overlap with previous segments
|
2022-08-05 23:03:38 +02:00
|
|
|
|
for (size_t j = 0; j < s; j++) {
|
2021-10-11 02:19:33 +02:00
|
|
|
|
if (segStops[j] > segStarts[s] && segStarts[j] < segStops[s]) {
|
|
|
|
|
//segments overlap, merge
|
|
|
|
|
segStarts[j] = min(segStarts[s],segStarts[j]);
|
|
|
|
|
segStops [j] = max(segStops [s],segStops [j]); segStops[s] = 0;
|
|
|
|
|
s--;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
s++;
|
|
|
|
|
}
|
2023-02-25 09:41:15 +01:00
|
|
|
|
|
2022-07-17 15:58:41 +02:00
|
|
|
|
_segments.clear();
|
2023-02-01 19:30:56 +01:00
|
|
|
|
_segments.reserve(s); // prevent reallocations
|
2023-02-25 09:41:15 +01:00
|
|
|
|
// there is always at least one segment (but we need to differentiate between 1D and 2D)
|
|
|
|
|
#ifndef WLED_DISABLE_2D
|
|
|
|
|
if (isMatrix)
|
|
|
|
|
_segments.push_back(Segment(0, Segment::maxWidth, 0, Segment::maxHeight));
|
|
|
|
|
else
|
|
|
|
|
#endif
|
|
|
|
|
_segments.push_back(Segment(segStarts[0], segStops[0]));
|
|
|
|
|
for (size_t i = 1; i < s; i++) {
|
|
|
|
|
_segments.push_back(Segment(segStarts[i], segStops[i]));
|
2021-09-12 01:00:54 +02:00
|
|
|
|
}
|
2023-02-25 09:41:15 +01:00
|
|
|
|
|
2021-09-12 01:00:54 +02:00
|
|
|
|
} else {
|
2023-02-25 09:41:15 +01:00
|
|
|
|
|
2022-07-19 22:14:46 +02:00
|
|
|
|
if (forceReset || getSegmentsNum() == 0) resetSegments();
|
2022-02-10 16:09:16 +01:00
|
|
|
|
//expand the main seg to the entire length, but only if there are no other segments, or reset is forced
|
2022-07-17 15:58:41 +02:00
|
|
|
|
else if (getActiveSegmentsNum() == 1) {
|
2022-07-19 22:14:46 +02:00
|
|
|
|
size_t i = getLastActiveSegmentId();
|
2023-02-25 09:41:15 +01:00
|
|
|
|
#ifndef WLED_DISABLE_2D
|
|
|
|
|
_segments[i].start = 0;
|
|
|
|
|
_segments[i].stop = Segment::maxWidth;
|
|
|
|
|
_segments[i].startY = 0;
|
|
|
|
|
_segments[i].stopY = Segment::maxHeight;
|
|
|
|
|
_segments[i].grouping = 1;
|
|
|
|
|
_segments[i].spacing = 0;
|
|
|
|
|
#else
|
2022-07-19 22:14:46 +02:00
|
|
|
|
_segments[i].start = 0;
|
|
|
|
|
_segments[i].stop = _length;
|
2023-02-25 09:41:15 +01:00
|
|
|
|
#endif
|
2021-10-11 02:19:33 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-02-25 09:41:15 +01:00
|
|
|
|
_mainSegment = 0;
|
2021-10-11 02:19:33 +02:00
|
|
|
|
|
|
|
|
|
fixInvalidSegments();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void WS2812FX::fixInvalidSegments() {
|
|
|
|
|
//make sure no segment is longer than total (sanity check)
|
2022-08-05 23:03:38 +02:00
|
|
|
|
for (size_t i = getSegmentsNum()-1; i > 0; i--) {
|
2023-02-14 17:11:58 +01:00
|
|
|
|
if (isMatrix) {
|
|
|
|
|
#ifndef WLED_DISABLE_2D
|
2023-02-15 20:36:54 +01:00
|
|
|
|
if (_segments[i].start >= Segment::maxWidth * Segment::maxHeight) {
|
|
|
|
|
// 1D segment at the end of matrix
|
|
|
|
|
if (_segments[i].start >= _length || _segments[i].startY > 0 || _segments[i].stopY > 1) { _segments.erase(_segments.begin()+i); continue; }
|
|
|
|
|
if (_segments[i].stop > _length) _segments[i].stop = _length;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (_segments[i].start >= Segment::maxWidth || _segments[i].startY >= Segment::maxHeight) { _segments.erase(_segments.begin()+i); continue; }
|
|
|
|
|
if (_segments[i].stop > Segment::maxWidth) _segments[i].stop = Segment::maxWidth;
|
|
|
|
|
if (_segments[i].stopY > Segment::maxHeight) _segments[i].stopY = Segment::maxHeight;
|
2023-02-14 17:11:58 +01:00
|
|
|
|
#endif
|
|
|
|
|
} else {
|
|
|
|
|
if (_segments[i].start >= _length) { _segments.erase(_segments.begin()+i); continue; }
|
|
|
|
|
if (_segments[i].stop > _length) _segments[i].stop = _length;
|
|
|
|
|
}
|
2021-10-11 02:19:33 +02:00
|
|
|
|
}
|
2023-02-15 20:36:54 +01:00
|
|
|
|
// this is always called as the last step after finalizeInit(), update covered bus types
|
|
|
|
|
for (segment &seg : _segments)
|
|
|
|
|
seg.refreshLightCapabilities();
|
2021-10-11 02:19:33 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//true if all segments align with a bus, or if a segment covers the total length
|
2023-02-11 18:41:30 +01:00
|
|
|
|
//irrelevant in 2D set-up
|
2021-10-11 02:19:33 +02:00
|
|
|
|
bool WS2812FX::checkSegmentAlignment() {
|
2022-07-17 15:58:41 +02:00
|
|
|
|
bool aligned = false;
|
|
|
|
|
for (segment &seg : _segments) {
|
2021-10-11 02:19:33 +02:00
|
|
|
|
for (uint8_t b = 0; b<busses.getNumBusses(); b++) {
|
|
|
|
|
Bus *bus = busses.getBus(b);
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (seg.start == bus->getStart() && seg.stop == bus->getStart() + bus->getLength()) aligned = true;
|
2021-10-11 02:19:33 +02:00
|
|
|
|
}
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (seg.start == 0 && seg.stop == _length) aligned = true;
|
2021-10-11 02:19:33 +02:00
|
|
|
|
if (!aligned) return false;
|
2021-02-27 12:06:14 +01:00
|
|
|
|
}
|
2021-10-11 02:19:33 +02:00
|
|
|
|
return true;
|
2021-02-27 12:06:14 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-08-29 22:26:39 +02:00
|
|
|
|
//After this function is called, setPixelColor() will use that segment (offsets, grouping, ... will apply)
|
2022-01-31 20:35:11 +01:00
|
|
|
|
//Note: If called in an interrupt (e.g. JSON API), original segment must be restored,
|
2021-12-03 20:36:37 +01:00
|
|
|
|
//otherwise it can lead to a crash on ESP32 because _segment_index is modified while in use by the main thread
|
2023-02-01 19:30:56 +01:00
|
|
|
|
uint8_t WS2812FX::setPixelSegment(uint8_t n) {
|
2022-01-31 20:35:11 +01:00
|
|
|
|
uint8_t prevSegId = _segment_index;
|
2022-07-17 15:58:41 +02:00
|
|
|
|
if (n < _segments.size()) {
|
2020-08-29 22:26:39 +02:00
|
|
|
|
_segment_index = n;
|
2022-07-12 18:10:07 +02:00
|
|
|
|
_virtualSegmentLength = _segments[_segment_index].virtualLength();
|
2020-08-29 22:26:39 +02:00
|
|
|
|
}
|
2022-01-31 20:35:11 +01:00
|
|
|
|
return prevSegId;
|
2020-08-29 22:26:39 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-02-01 19:30:56 +01:00
|
|
|
|
void WS2812FX::setRange(uint16_t i, uint16_t i2, uint32_t col) {
|
2019-02-09 16:37:20 +01:00
|
|
|
|
if (i2 >= i)
|
|
|
|
|
{
|
2020-01-02 22:10:59 +01:00
|
|
|
|
for (uint16_t x = i; x <= i2; x++) setPixelColor(x, col);
|
2019-02-09 16:37:20 +01:00
|
|
|
|
} else
|
|
|
|
|
{
|
2020-01-02 22:10:59 +01:00
|
|
|
|
for (uint16_t x = i2; x <= i; x++) setPixelColor(x, col);
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-02-01 19:30:56 +01:00
|
|
|
|
void WS2812FX::setTransitionMode(bool t) {
|
2023-06-23 23:49:54 +02:00
|
|
|
|
for (segment &seg : _segments) seg.startTransition(t ? _transitionDur : 0);
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-03 21:36:47 +02:00
|
|
|
|
#ifdef WLED_DEBUG
|
2023-02-01 19:30:56 +01:00
|
|
|
|
void WS2812FX::printSize() {
|
2022-08-03 21:36:47 +02:00
|
|
|
|
size_t size = 0;
|
2022-08-29 21:51:46 +02:00
|
|
|
|
for (const Segment &seg : _segments) size += seg.getSize();
|
2022-08-03 22:09:27 +02:00
|
|
|
|
DEBUG_PRINTF("Segments: %d -> %uB\n", _segments.size(), size);
|
2022-08-03 21:36:47 +02:00
|
|
|
|
DEBUG_PRINTF("Modes: %d*%d=%uB\n", sizeof(mode_ptr), _mode.size(), (_mode.capacity()*sizeof(mode_ptr)));
|
|
|
|
|
DEBUG_PRINTF("Data: %d*%d=%uB\n", sizeof(const char *), _modeData.size(), (_modeData.capacity()*sizeof(const char *)));
|
|
|
|
|
DEBUG_PRINTF("Map: %d*%d=%uB\n", sizeof(uint16_t), (int)customMappingSize, customMappingSize*sizeof(uint16_t));
|
2023-03-11 15:03:28 +01:00
|
|
|
|
size = getLengthTotal();
|
2023-06-26 18:16:38 +02:00
|
|
|
|
if (useGlobalLedBuffer) DEBUG_PRINTF("Buffer: %d*%u=%uB\n", sizeof(CRGB), size, size*sizeof(CRGB));
|
2022-08-03 21:36:47 +02:00
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2023-02-01 19:30:56 +01:00
|
|
|
|
void WS2812FX::loadCustomPalettes() {
|
2022-07-10 22:23:25 +02:00
|
|
|
|
byte tcp[72]; //support gradient palettes with up to 18 entries
|
2022-07-28 23:19:58 +02:00
|
|
|
|
CRGBPalette16 targetPalette;
|
2022-11-16 20:55:21 +01:00
|
|
|
|
customPalettes.clear(); // start fresh
|
2022-07-28 23:19:58 +02:00
|
|
|
|
for (int index = 0; index<10; index++) {
|
2022-07-10 22:23:25 +02:00
|
|
|
|
char fileName[32];
|
2022-10-04 22:10:20 +02:00
|
|
|
|
sprintf_P(fileName, PSTR("/palette%d.json"), index);
|
2022-07-10 22:23:25 +02:00
|
|
|
|
|
2022-07-28 23:19:58 +02:00
|
|
|
|
StaticJsonDocument<1536> pDoc; // barely enough to fit 72 numbers
|
2022-07-10 22:23:25 +02:00
|
|
|
|
if (WLED_FS.exists(fileName)) {
|
|
|
|
|
DEBUG_PRINT(F("Reading palette from "));
|
|
|
|
|
DEBUG_PRINTLN(fileName);
|
|
|
|
|
|
|
|
|
|
if (readObjectFromFile(fileName, nullptr, &pDoc)) {
|
|
|
|
|
JsonArray pal = pDoc[F("palette")];
|
2022-12-21 21:00:28 +01:00
|
|
|
|
if (!pal.isNull() && pal.size()>4) { // not an empty palette (at least 2 entries)
|
2022-12-18 21:02:19 +01:00
|
|
|
|
if (pal[0].is<int>() && pal[1].is<const char *>()) {
|
|
|
|
|
// we have an array of index & hex strings
|
|
|
|
|
size_t palSize = MIN(pal.size(), 36);
|
|
|
|
|
palSize -= palSize % 2; // make sure size is multiple of 2
|
|
|
|
|
for (size_t i=0, j=0; i<palSize && pal[i].as<int>()<256; i+=2, j+=4) {
|
|
|
|
|
uint8_t rgbw[] = {0,0,0,0};
|
|
|
|
|
tcp[ j ] = (uint8_t) pal[ i ].as<int>(); // index
|
|
|
|
|
colorFromHexString(rgbw, pal[i+1].as<const char *>()); // will catch non-string entires
|
2022-12-21 21:00:28 +01:00
|
|
|
|
for (size_t c=0; c<3; c++) tcp[j+1+c] = 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]));
|
2022-12-18 21:02:19 +01:00
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
size_t palSize = MIN(pal.size(), 72);
|
|
|
|
|
palSize -= palSize % 4; // make sure size is multiple of 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+1] = (uint8_t) pal[i+1].as<int>(); // R
|
|
|
|
|
tcp[i+2] = (uint8_t) pal[i+2].as<int>(); // G
|
|
|
|
|
tcp[i+3] = (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]));
|
|
|
|
|
}
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
2022-07-28 23:19:58 +02:00
|
|
|
|
customPalettes.push_back(targetPalette.loadDynamicGradientPalette(tcp));
|
2022-07-10 22:23:25 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2022-07-28 23:19:58 +02:00
|
|
|
|
} else {
|
|
|
|
|
break;
|
2019-02-11 23:49:04 +01:00
|
|
|
|
}
|
2019-02-09 16:37:20 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-11-03 14:52:22 +01:00
|
|
|
|
//load custom mapping table from JSON file (called from finalizeInit() or deserializeState())
|
2023-02-11 18:41:30 +01:00
|
|
|
|
bool WS2812FX::deserializeMap(uint8_t n) {
|
2023-02-10 19:49:43 +01:00
|
|
|
|
// 2D support creates its own ledmap (on the fly) if a ledmap.json exists it will overwrite built one.
|
2022-05-08 10:50:48 +02:00
|
|
|
|
|
2021-09-12 01:15:51 +02:00
|
|
|
|
char fileName[32];
|
|
|
|
|
strcpy_P(fileName, PSTR("/ledmap"));
|
|
|
|
|
if (n) sprintf(fileName +7, "%d", n);
|
|
|
|
|
strcat(fileName, ".json");
|
2021-04-04 13:54:34 +02:00
|
|
|
|
bool isFile = WLED_FS.exists(fileName);
|
|
|
|
|
|
|
|
|
|
if (!isFile) {
|
|
|
|
|
// erase custom mapping if selecting nonexistent ledmap.json (n==0)
|
2023-02-12 13:18:30 +01:00
|
|
|
|
if (!isMatrix && !n && customMappingTable != nullptr) {
|
2021-04-04 13:54:34 +02:00
|
|
|
|
customMappingSize = 0;
|
|
|
|
|
delete[] customMappingTable;
|
|
|
|
|
customMappingTable = nullptr;
|
|
|
|
|
}
|
2023-02-11 18:41:30 +01:00
|
|
|
|
return false;
|
2021-04-04 13:54:34 +02:00
|
|
|
|
}
|
2021-02-13 01:02:14 +01:00
|
|
|
|
|
2023-02-11 18:41:30 +01:00
|
|
|
|
if (!requestJSONBufferLock(7)) return false;
|
2021-02-13 01:02:14 +01:00
|
|
|
|
|
2021-11-03 14:52:22 +01:00
|
|
|
|
if (!readObjectFromFile(fileName, nullptr, &doc)) {
|
2021-11-12 23:33:10 +01:00
|
|
|
|
releaseJSONBufferLock();
|
2023-02-11 18:41:30 +01:00
|
|
|
|
return false; //if file does not exist just exit
|
2021-11-03 14:52:22 +01:00
|
|
|
|
}
|
2021-02-13 01:02:14 +01:00
|
|
|
|
|
2023-02-11 18:41:30 +01:00
|
|
|
|
DEBUG_PRINT(F("Reading LED map from "));
|
|
|
|
|
DEBUG_PRINTLN(fileName);
|
|
|
|
|
|
2021-04-04 13:54:34 +02:00
|
|
|
|
// erase old custom ledmap
|
2021-02-13 01:02:14 +01:00
|
|
|
|
if (customMappingTable != nullptr) {
|
2021-04-04 13:54:34 +02:00
|
|
|
|
customMappingSize = 0;
|
2021-02-13 01:02:14 +01:00
|
|
|
|
delete[] customMappingTable;
|
|
|
|
|
customMappingTable = nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
JsonArray map = doc[F("map")];
|
|
|
|
|
if (!map.isNull() && map.size()) { // not an empty map
|
|
|
|
|
customMappingSize = map.size();
|
|
|
|
|
customMappingTable = new uint16_t[customMappingSize];
|
|
|
|
|
for (uint16_t i=0; i<customMappingSize; i++) {
|
2023-01-30 17:11:14 +01:00
|
|
|
|
customMappingTable[i] = (uint16_t) (map[i]<0 ? 0xFFFFU : map[i]);
|
2021-02-13 01:02:14 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2021-11-03 14:52:22 +01:00
|
|
|
|
|
2021-11-12 23:33:10 +01:00
|
|
|
|
releaseJSONBufferLock();
|
2023-02-11 18:41:30 +01:00
|
|
|
|
return true;
|
2021-02-13 01:02:14 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-31 11:11:05 +01:00
|
|
|
|
|
2021-11-24 11:02:25 +01:00
|
|
|
|
WS2812FX* WS2812FX::instance = nullptr;
|
2021-11-30 22:52:17 +01:00
|
|
|
|
|
2022-11-11 03:10:41 +01:00
|
|
|
|
const char JSON_mode_names[] PROGMEM = R"=====(["FX names moved"])=====";
|
2021-12-28 18:09:52 +01:00
|
|
|
|
const char JSON_palette_names[] PROGMEM = R"=====([
|
|
|
|
|
"Default","* Random Cycle","* Color 1","* Colors 1&2","* Color Gradient","* Colors Only","Party","Cloud","Lava","Ocean",
|
|
|
|
|
"Forest","Rainbow","Rainbow Bands","Sunset","Rivendell","Breeze","Red & Blue","Yellowout","Analogous","Splash",
|
2022-08-31 21:21:53 +02:00
|
|
|
|
"Pastel","Sunset 2","Beach","Vintage","Departure","Landscape","Beech","Sherbet","Hult","Hult 64",
|
2021-12-28 18:09:52 +01:00
|
|
|
|
"Drywet","Jul","Grintage","Rewhi","Tertiary","Fire","Icefire","Cyane","Light Pink","Autumn",
|
|
|
|
|
"Magenta","Magred","Yelmag","Yelblu","Orange & Teal","Tiamat","April Night","Orangery","C9","Sakura",
|
|
|
|
|
"Aurora","Atlantica","C9 2","C9 New","Temperature","Aurora 2","Retro Clown","Candy","Toxy Reaf","Fairy Reaf",
|
|
|
|
|
"Semi Blue","Pink Candy","Red Reaf","Aqua Flash","Yelblu Hot","Lite Light","Red Flash","Blink Red","Red Shift","Red Tide",
|
|
|
|
|
"Candy2"
|
|
|
|
|
])=====";
|