1557 lines
		
	
	
		
			54 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1557 lines
		
	
	
		
			54 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|   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
 | |
| */
 | |
| #include "wled.h"
 | |
| #include "FX.h"
 | |
| #include "palettes.h"
 | |
| 
 | |
| /*
 | |
|   Custom per-LED mapping has moved!
 | |
| 
 | |
|   Create a file "ledmap.json" using the edit page.
 | |
| 
 | |
|   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]}
 | |
| 
 | |
|   another example. Switches direction every 5 LEDs.
 | |
|   {"map":[
 | |
|   0, 1, 2, 3, 4, 9, 8, 7, 6, 5, 10, 11, 12, 13, 14,
 | |
|   19, 18, 17, 16, 15, 20, 21, 22, 23, 24, 29, 28, 27, 26, 25]}
 | |
| */
 | |
| 
 | |
| //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
 | |
|   #define PIXEL_COUNTS DEFAULT_LED_COUNT
 | |
| #endif
 | |
| 
 | |
| #ifndef DATA_PINS
 | |
|   #define DATA_PINS LEDPIN
 | |
| #endif
 | |
| 
 | |
| #ifndef DEFAULT_LED_TYPE
 | |
|   #define DEFAULT_LED_TYPE TYPE_WS2812_RGB
 | |
| #endif
 | |
| 
 | |
| #ifndef DEFAULT_LED_COLOR_ORDER
 | |
|   #define DEFAULT_LED_COLOR_ORDER COL_ORDER_GRB  //default to GRB
 | |
| #endif
 | |
| 
 | |
| 
 | |
| #if MAX_NUM_SEGMENTS < WLED_MAX_BUSSES
 | |
|   #error "Max segments must be at least max number of busses!"
 | |
| #endif
 | |
| 
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // Segment class implementation
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| Segment::Segment(const Segment &orig) {
 | |
|   DEBUG_PRINTLN(F("-- Segment duplicated --"));
 | |
|   memcpy(this, &orig, sizeof(Segment));
 | |
|   name = nullptr;
 | |
|   data = nullptr;
 | |
|   _dataLen = 0;
 | |
|   //_t = nullptr;
 | |
|   if (orig.name) { name = new char[strlen(orig.name)+1]; if (name) strcpy(name, orig.name); }
 | |
|   if (orig.data) { allocateData(orig._dataLen); memcpy(data, orig.data, orig._dataLen); }
 | |
|   //if (orig._t) { _t = new Transition(orig._t->_dur, orig._t->_briT, orig._t->_cctT, orig._t->_colorT); }
 | |
|   DEBUG_PRINTF("  Original data: %p (%d)\n", orig.data, (int)orig._dataLen);
 | |
|   DEBUG_PRINTF("  Constructed data: %p (%d)\n", data, (int)_dataLen);
 | |
| }
 | |
| 
 | |
| Segment::Segment(Segment &&orig) noexcept {
 | |
|   DEBUG_PRINTLN(F("-- Move constructor --"));
 | |
|   memcpy(this, &orig, sizeof(Segment));
 | |
|   orig.name = nullptr;
 | |
|   orig.data = nullptr;
 | |
|   orig._dataLen = 0;
 | |
|   //orig._t   = nullptr;
 | |
| }
 | |
| 
 | |
| Segment& Segment::operator= (const Segment &orig) {
 | |
|   DEBUG_PRINTLN(F("-- Segment copied --"));
 | |
|   if (this != &orig) {
 | |
|     if (name) {
 | |
|       DEBUG_PRINTF("  Copy Deleting %s (%p)\n", name, name);
 | |
|       delete[] name;
 | |
|     }
 | |
|     //if (_t) delete _t;
 | |
|     deallocateData();
 | |
|     memcpy(this, &orig, sizeof(Segment));
 | |
|     name = nullptr;
 | |
|     data = nullptr;
 | |
|     _dataLen = 0;
 | |
|     //_t = nullptr;
 | |
|     if (orig.name) { name = new char[strlen(orig.name)+1]; if (name) strcpy(name, orig.name); }
 | |
|     if (orig.data) { allocateData(orig._dataLen); memcpy(data, orig.data, orig._dataLen); }
 | |
|     //if (orig._t) { _t = new Transition(orig._t->_dur, orig._t->_briT, orig._t->_cctT, orig._t->_colorT); }
 | |
|     DEBUG_PRINTF("  Original data: %p (%d)\n", orig.data, (int)orig._dataLen);
 | |
|     DEBUG_PRINTF("  Copied data: %p (%d)\n", data, (int)_dataLen);
 | |
|   }
 | |
|   return *this;
 | |
| }
 | |
| 
 | |
| Segment& Segment::operator= (Segment &&orig) noexcept {
 | |
|   DEBUG_PRINTLN(F("-- Moving segment --"));
 | |
|   if (this != &orig) {
 | |
|     if (name) {
 | |
|       DEBUG_PRINTF("  Move Deleting %s (%p)\n", name, name);
 | |
|       delete[] name; // free old name
 | |
|     }
 | |
|     //if (_t) delete _t;
 | |
|     deallocateData(); // free old runtime data
 | |
|     memcpy(this, &orig, sizeof(Segment));
 | |
|     orig.name = nullptr;
 | |
|     orig.data = nullptr;
 | |
|     orig._dataLen = 0;
 | |
|     //orig._t   = nullptr;
 | |
|   }
 | |
|   return *this;
 | |
| }
 | |
| 
 | |
| bool Segment::allocateData(uint16_t len) {
 | |
|   if (data && _dataLen == len) return true; //already allocated
 | |
|   deallocateData();
 | |
|   // TODO: move out to WS2812FX class: for (seg : _segments) sum += seg.dataSize();
 | |
|   if (strip.getUsedSegmentData() + len > MAX_SEGMENT_DATA) return false; //not enough memory
 | |
|   // if possible use SPI RAM on ESP32
 | |
|   #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_USE_PSRAM)
 | |
|   if (psramFound())
 | |
|     data = (byte*) ps_malloc(len);
 | |
|   else
 | |
|   #endif
 | |
|     data = (byte*) malloc(len);
 | |
|   if (!data) return false; //allocation failed
 | |
|   strip.addUsedSegmentData(len); // TODO: move out to WS2812FX class: for (seg : _segments) sum += seg.dataSize();
 | |
|   _dataLen = len;
 | |
|   memset(data, 0, len);
 | |
|   DEBUG_PRINTF("-- Allocated data %p (%d)\n", data, (int)len);
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void Segment::deallocateData() {
 | |
|   // NOTE: deallocating data sometimes produces corrupt heap.
 | |
|   if (!data) return;
 | |
|   DEBUG_PRINTF("-- Deallocating data: %p (%d)\n", data, (int)_dataLen);
 | |
|   free(data);
 | |
|   DEBUG_PRINTLN(F("-- Data freed."));
 | |
|   data = nullptr;
 | |
|   strip.addUsedSegmentData(-(int16_t)_dataLen); // TODO: move out to WS2812FX class: for (seg : _segments) sum -= seg.dataSize();
 | |
|   _dataLen = 0;
 | |
|   DEBUG_PRINTLN(F("-- Dealocated data."));
 | |
| }
 | |
| 
 | |
| /** 
 | |
|   * If reset of this segment was requested, clears runtime
 | |
|   * settings of this segment.
 | |
|   * Must not be called while an effect mode function is running
 | |
|   * because it could access the data buffer and this method 
 | |
|   * may free that data buffer.
 | |
|   */
 | |
| void Segment::resetIfRequired() {
 | |
|   if (reset) { // (getOption(SEG_OPTION_RESET))
 | |
|     next_time = 0; step = 0; call = 0; aux0 = 0; aux1 = 0; 
 | |
|     reset = false; // setOption(SEG_OPTION_RESET, false);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void Segment::startTransition(uint16_t dur) {
 | |
|   // starting a transition has to occur before change so we get current values 1st
 | |
|   /*uint8_t*/ _briT = currentBri(getOption(SEG_OPTION_ON) ? opacity : 0); // comment out uint8_t if not using Transition struct
 | |
|   /*uint8_t*/ _cctT = currentBri(cct, true); // comment out uint8_t if not using Transition struct
 | |
|   //uint32_t _colorT[NUM_COLORS]; // comment out if not using Transition struct
 | |
|   for (size_t i=0; i<NUM_COLORS; i++) _colorT[i] = currentColor(i, colors[i]);
 | |
| 
 | |
|   // comment out if not using Transition struct
 | |
|   //if (!_t) _t = new Transition(dur); // no previous transition running
 | |
|   //if (!_t) return; // failed to allocat data
 | |
|   //_t->_briT = _briT;
 | |
|   //_t->_cctT = _cctT;
 | |
|   //for (size_t i=0; i<NUM_COLORS; i++) _t->_colorT[i] = _colorT[i];
 | |
| 
 | |
|   setOption(SEG_OPTION_TRANSITIONAL, true);
 | |
| }
 | |
| 
 | |
| uint16_t Segment::progress() { //transition progression between 0-65535
 | |
|   //if (!_t) return 0xFFFFU;
 | |
|   uint32_t timeNow = millis();
 | |
|   if (timeNow - /*_t->*/_start > /*_t->*/_dur) return 0xFFFFU;
 | |
|   return (timeNow - /*_t->*/_start) * 0xFFFFU / /*_t->*/_dur;
 | |
| }
 | |
| 
 | |
| uint8_t Segment::currentBri(uint8_t briNew, bool useCct) {
 | |
|   //if (!_t) return (useCct) ? cct : opacity;
 | |
|   if (getOption(SEG_OPTION_TRANSITIONAL)) {
 | |
|     uint32_t prog = progress() + 1;
 | |
|     if (useCct) return ((briNew * prog) + /*_t->*/_cctT * (0x10000 - prog)) >> 16;
 | |
|     else        return ((briNew * prog) + /*_t->*/_briT * (0x10000 - prog)) >> 16;
 | |
|   } else {
 | |
|     return (useCct) ? cct : (getOption(SEG_OPTION_ON) ? opacity : 0);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void Segment::handleTransition() {
 | |
|   if (!getOption(SEG_OPTION_TRANSITIONAL)) return;
 | |
|   unsigned long maxWait = millis() + 20;
 | |
|   if (mode == FX_MODE_STATIC && next_time > maxWait) next_time = maxWait;
 | |
|   if (progress() == 0xFFFFU) {
 | |
|     //if (_t) { delete _t; _t = nullptr; }
 | |
|     setOption(SEG_OPTION_TRANSITIONAL, false); // finish transitioning segment
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool Segment::setColor(uint8_t slot, uint32_t c) { //returns true if changed
 | |
|   if (slot >= NUM_COLORS || c == colors[slot]) return false;
 | |
|   startTransition(strip.getTransition()); // start transition prior to change
 | |
|   colors[slot] = c;
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void Segment::setCCT(uint16_t k) {
 | |
|   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;
 | |
|   startTransition(strip.getTransition()); // start transition prior to change
 | |
|   cct = k;
 | |
| }
 | |
| 
 | |
| void Segment::setOpacity(uint8_t o) {
 | |
|   if (opacity == o) return;
 | |
|   startTransition(strip.getTransition()); // start transition prior to change
 | |
|   opacity = o;
 | |
| }
 | |
| 
 | |
| void Segment::setOption(uint8_t n, bool val) {
 | |
|   bool prevOn = getOption(SEG_OPTION_ON);
 | |
|   if (n == SEG_OPTION_ON && val != prevOn) startTransition(strip.getTransition()); // start transition prior to change
 | |
|   if (val) options |=   0x01 << n;
 | |
|   else     options &= ~(0x01 << n);
 | |
| }
 | |
| 
 | |
| // 2D matrix
 | |
| uint16_t Segment::virtualWidth() {
 | |
|   uint16_t groupLen = groupLength();
 | |
|   uint16_t vWidth = ((getOption(SEG_OPTION_TRANSPOSED) ? height() : width()) + groupLen - 1) / groupLen;
 | |
|   if (getOption(SEG_OPTION_MIRROR)) vWidth = (vWidth + 1) /2;  // divide by 2 if mirror, leave at least a single LED
 | |
|   return vWidth;
 | |
| }
 | |
| 
 | |
| uint16_t Segment::virtualHeight() {
 | |
|   uint16_t groupLen = groupLength();
 | |
|   uint16_t vHeight = ((getOption(SEG_OPTION_TRANSPOSED) ? width() : height()) + groupLen - 1) / groupLen;
 | |
|   if (getOption(SEG_OPTION_MIRROR_Y)) vHeight = (vHeight + 1) /2;  // divide by 2 if mirror, leave at least a single LED
 | |
|   return vHeight;
 | |
| }
 | |
| 
 | |
| // 1D strip
 | |
| uint16_t Segment::virtualLength() {
 | |
|   uint16_t groupLen = groupLength();
 | |
|   uint16_t vLength = (length() + groupLen - 1) / groupLen;
 | |
|   if (getOption(SEG_OPTION_MIRROR)) vLength = (vLength + 1) /2;  // divide by 2 if mirror, leave at least a single LED
 | |
|   return vLength;
 | |
| }
 | |
| 
 | |
| void IRAM_ATTR Segment::setPixelColor(int i, uint32_t col)
 | |
| {
 | |
|   if (strip.isMatrix) {
 | |
|     // map linear pixel into 2D segment area (even for 1D segments, expanding vertically)
 | |
|     uint16_t h = virtualHeight();  // segment height in logical pixels
 | |
|     for (int y = 0; y < h; y++) { // expand 1D effect vertically
 | |
|       setPixelColorXY(i, y * groupLength(), col);
 | |
|     }
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   uint16_t len = length();
 | |
|   uint8_t _bri_t = strip._bri_t;
 | |
|   //uint8_t _bri_t = currentBri(getOption(SEG_OPTION_ON) ? opacity : 0);
 | |
|   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();
 | |
|   if (getOption(SEG_OPTION_REVERSED)) { // is segment reversed?
 | |
|     if (getOption(SEG_OPTION_MIRROR)) { // is segment mirrored?
 | |
|       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++) {
 | |
|     uint16_t indexSet = i + ((getOption(SEG_OPTION_REVERSED)) ? -j : j);
 | |
|     if (indexSet >= start && indexSet < stop) {
 | |
|       if (getOption(SEG_OPTION_MIRROR)) { //set the corresponding mirrored pixel
 | |
|         uint16_t indexMir = stop - indexSet + start - 1;          
 | |
|         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)
 | |
| {
 | |
|   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);
 | |
|     float    dL = fC - iL;
 | |
|     float    dR = iR - fC;
 | |
|     uint32_t cIL = getPixelColor(iL);
 | |
|     uint32_t cIR = getPixelColor(iR);
 | |
|     if (iR!=iL) {
 | |
|       // blend L pixel
 | |
|       cIL = color_blend(col, cIL, uint8_t(dL*255.0f));
 | |
|       setPixelColor(iL, cIL);
 | |
|       // blend R pixel
 | |
|       cIR = color_blend(col, cIR, uint8_t(dR*255.0f));
 | |
|       setPixelColor(iR, cIR);
 | |
|     } else {
 | |
|       // exact match (x & y land on a pixel)
 | |
|       setPixelColor(iL, col);
 | |
|     }
 | |
|   } else {
 | |
|     setPixelColor(uint16_t(roundf(fC)), col);
 | |
|   }
 | |
| }
 | |
| 
 | |
| uint32_t Segment::getPixelColor(uint16_t i)
 | |
| {
 | |
|   if (getOption(SEG_OPTION_REVERSED)) i = virtualLength() - i - 1;
 | |
|   i *= groupLength();
 | |
|   i += start;
 | |
|   /* offset/phase */
 | |
|   i += offset;
 | |
|   if (i >= stop) i -= length();
 | |
|   return strip.getPixelColor(i);
 | |
| }
 | |
| 
 | |
| uint8_t Segment::differs(Segment& b) {
 | |
|   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;
 | |
| 
 | |
|   //bit pattern: msb first: [transposed mirrorY reverseY] transitional (tbd) paused needspixelstate mirrored on reverse selected
 | |
|   if ((options & 0b11100101110) != (b.options & 0b11100101110)) d |= SEG_DIFFERS_OPT;
 | |
|   if ((options & 0x01) != (b.options & 0x01))                   d |= SEG_DIFFERS_SEL;
 | |
|   
 | |
|   for (uint8_t i = 0; i < NUM_COLORS; i++) if (colors[i] != b.colors[i]) d |= SEG_DIFFERS_COL;
 | |
| 
 | |
|   return d;
 | |
| }
 | |
| 
 | |
| void Segment::refreshLightCapabilities() {
 | |
|   uint8_t capabilities = 0x01;
 | |
| 
 | |
|   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;
 | |
|     if (bus->getStart() >= stop) continue;
 | |
|     if (bus->getStart() + bus->getLength() <= start) continue;
 | |
| 
 | |
|     uint8_t type = bus->getType();
 | |
|     if (type == TYPE_ANALOG_1CH || (!cctFromRgb && type == TYPE_ANALOG_2CH)) capabilities &= 0xFE; // does not support RGB
 | |
|     if (bus->isRgbw()) capabilities |= 0x02; // segment supports white channel
 | |
|     if (!cctFromRgb) {
 | |
|       switch (type) {
 | |
|         case TYPE_ANALOG_5CH:
 | |
|         case TYPE_ANALOG_2CH:
 | |
|           capabilities |= 0x04; //segment supports white CCT 
 | |
|       }
 | |
|     }
 | |
|     if (correctWB && type != TYPE_ANALOG_1CH) capabilities |= 0x04; //white balance correction (uses CCT slider)
 | |
|     uint8_t aWM = Bus::getAutoWhiteMode()<255 ? Bus::getAutoWhiteMode() : bus->getAWMode();
 | |
|     bool whiteSlider = (aWM == RGBW_MODE_DUAL || aWM == RGBW_MODE_MANUAL_ONLY); // white slider allowed
 | |
|     if (bus->isRgbw() && (whiteSlider || !(capabilities & 0x01))) capabilities |= 0x08; // allow white channel adjustments (AWM allows or is not RGB)
 | |
|   }
 | |
|   _capabilities = capabilities;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Fills segment with color
 | |
|  */
 | |
| void Segment::fill(uint32_t c) {
 | |
|   const uint16_t cols = strip.isMatrix ? virtualWidth() : virtualLength();
 | |
|   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++) {
 | |
|     if (strip.isMatrix) setPixelColorXY(x, y, c);
 | |
|     else                setPixelColor(x, c);
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Blends the specified color with the existing pixel color.
 | |
| void Segment::blendPixelColor(uint16_t n, uint32_t color, uint8_t blend) {
 | |
|   setPixelColor(n, color_blend(getPixelColor(n), color, blend));
 | |
| }
 | |
| 
 | |
| // Adds the specified color with the existing pixel color perserving color balance.
 | |
| void Segment::addPixelColor(uint16_t n, uint32_t color) {
 | |
|   setPixelColor(n, color_add(getPixelColor(n), color));
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * fade out function, higher rate = quicker fade
 | |
|  */
 | |
| void Segment::fade_out(uint8_t rate) {
 | |
|   const uint16_t cols = strip.isMatrix ? virtualWidth() : virtualLength();
 | |
|   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++) {
 | |
|     color = strip.isMatrix ? getPixelColorXY(x, y) : getPixelColor(x);
 | |
|     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;
 | |
| 
 | |
|     if (strip.isMatrix) setPixelColorXY(x, y, r1 + rdelta, g1 + gdelta, b1 + bdelta, w1 + wdelta);
 | |
|     else                setPixelColor(x, r1 + rdelta, g1 + gdelta, b1 + bdelta, w1 + wdelta);
 | |
|   }
 | |
| }
 | |
| 
 | |
| // fades all pixels to black using nscale8()
 | |
| void Segment::fadeToBlackBy(uint8_t fadeBy) {
 | |
|   const uint16_t cols = strip.isMatrix ? virtualWidth() : virtualLength();
 | |
|   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++) {
 | |
|     if (strip.isMatrix) setPixelColorXY(x, y, CRGB(getPixelColorXY(x,y)).nscale8(255-fadeBy));
 | |
|     else                setPixelColor(x, CRGB(getPixelColor(x)).nscale8(255-fadeBy));
 | |
|   }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * blurs segment content, source: FastLED colorutils.cpp
 | |
|  */
 | |
| void Segment::blur(uint8_t blur_amount)
 | |
| {
 | |
|   if (strip.isMatrix) {
 | |
|     // compatibility with 2D
 | |
|     const uint16_t cols = virtualWidth();
 | |
|     const uint16_t rows = virtualHeight();
 | |
|     for (uint16_t i = 0; i < rows; i++) blurRow(i, blur_amount); // blur all rows
 | |
|     for (uint16_t k = 0; k < cols; k++) blurCol(k, blur_amount); // blur all columns
 | |
|     return;
 | |
|   }
 | |
|   uint8_t keep = 255 - blur_amount;
 | |
|   uint8_t seep = blur_amount >> 1;
 | |
|   CRGB carryover = CRGB::Black;
 | |
|   for(uint16_t i = 0; i < virtualLength(); i++)
 | |
|   {
 | |
|     CRGB cur = CRGB(getPixelColor(i));
 | |
|     CRGB part = cur;
 | |
|     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);
 | |
|       setPixelColor(i-1, qadd8(r, part.red), qadd8(g, part.green), qadd8(b, part.blue));
 | |
|     }
 | |
|     setPixelColor(i,cur.red, cur.green, cur.blue);
 | |
|     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.
 | |
|  */
 | |
| uint32_t Segment::color_wheel(uint8_t pos) { // TODO
 | |
|   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
 | |
|  * @param wrap FastLED palettes will usally wrap back to the start smoothly. Set false to get a hard edge
 | |
|  * @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
 | |
|  */
 | |
| uint32_t IRAM_ATTR Segment::color_from_palette(uint16_t i, bool mapping, bool wrap, uint8_t mcol, uint8_t pbri)
 | |
| {
 | |
|   if ((palette == 0 && mcol < 3) || strip._no_rgb) {
 | |
|     uint32_t color = colors[mcol]; // SEGCOLOR(mcol);
 | |
|     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;
 | |
|   fastled_col = ColorFromPalette(strip.currentPalette, paletteIndex, pbri, (strip.paletteBlend == 3)? NOBLEND:LINEARBLEND);
 | |
| 
 | |
|   return RGBW32(fastled_col.r, fastled_col.g, fastled_col.b, 0);
 | |
| }
 | |
| 
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // WS2812FX class implementation
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| //do not call this method from system context (network callback)
 | |
| void WS2812FX::finalizeInit(void)
 | |
| {
 | |
|   //reset segment runtimes
 | |
|   for (segment &seg : _segments) {
 | |
|     seg.markForReset();
 | |
|     seg.resetIfRequired();
 | |
|   }
 | |
| //  for (uint8_t i = 0; i < getMaxSegments(); i++) {
 | |
| //    _segments[i].markForReset();
 | |
| //    _segments[i].resetIfRequired();
 | |
| //  }
 | |
| 
 | |
|   _hasWhiteChannel = _isOffRefreshRequired = false;
 | |
| 
 | |
|   //if busses failed to load, add default (fresh install, FS issue, ...)
 | |
|   if (busses.getNumBusses() == 0) {
 | |
|     DEBUG_PRINTLN(F("No busses, init default"));
 | |
|     const uint8_t defDataPins[] = {DATA_PINS};
 | |
|     const uint16_t defCounts[] = {PIXEL_COUNTS};
 | |
|     const uint8_t defNumBusses = ((sizeof defDataPins) / (sizeof defDataPins[0]));
 | |
|     const uint8_t defNumCounts = ((sizeof defCounts)   / (sizeof defCounts[0]));
 | |
|     uint16_t prevLen = 0;
 | |
|     for (uint8_t i = 0; i < defNumBusses && i < WLED_MAX_BUSSES; i++) {
 | |
|       uint8_t defPin[] = {defDataPins[i]};
 | |
|       uint16_t start = prevLen;
 | |
|       uint16_t count = defCounts[(i < defNumCounts) ? i : defNumCounts -1];
 | |
|       prevLen += count;
 | |
|       BusConfig defCfg = BusConfig(DEFAULT_LED_TYPE, defPin, start, count, DEFAULT_LED_COLOR_ORDER, false, 0, RGBW_MODE_MANUAL_ONLY);
 | |
|       busses.add(defCfg);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   _length = 0;
 | |
|   for (uint8_t i=0; i<busses.getNumBusses(); i++) {
 | |
|     Bus *bus = busses.getBus(i);
 | |
|     if (bus == nullptr) continue;
 | |
|     if (bus->getStart() + bus->getLength() > MAX_LEDS) break;
 | |
|     //RGBW mode is enabled if at least one of the strips is RGBW
 | |
|     _hasWhiteChannel |= bus->isRgbw();
 | |
|     //refresh is required to remain off if at least one of the strips requires the refresh.
 | |
|     _isOffRefreshRequired |= bus->isOffRefreshRequired();
 | |
|     uint16_t busEnd = bus->getStart() + bus->getLength();
 | |
|     if (busEnd > _length) _length = busEnd;
 | |
|     #ifdef ESP8266
 | |
|     if ((!IS_DIGITAL(bus->getType()) || IS_2PIN(bus->getType()))) continue;
 | |
|     uint8_t pins[5];
 | |
|     if (!bus->getPins(pins)) continue;
 | |
|     BusDigital* bd = static_cast<BusDigital*>(bus);
 | |
|     if (pins[0] == 3) bd->reinit();
 | |
|     #endif
 | |
|   }
 | |
| 
 | |
|   //segments are created in makeAutoSegments();
 | |
| 
 | |
|   setBrightness(_brightness);
 | |
| }
 | |
| 
 | |
| void WS2812FX::service() {
 | |
|   uint32_t nowUp = millis(); // Be aware, millis() rolls over every 49 days
 | |
|   now = nowUp + timebase;
 | |
|   if (nowUp - _lastShow < MIN_SHOW_DELAY) return;
 | |
|   bool doShow = false;
 | |
| 
 | |
|   _isServicing = true;
 | |
|   _segment_index = 0;
 | |
|   for (segment &seg : _segments) {
 | |
| //  for (int i = 0; i < getMaxSegments(); i++) {
 | |
| //    Segment &seg = getSegment(i);
 | |
|     // reset the segment runtime data if needed
 | |
|     seg.resetIfRequired();
 | |
| 
 | |
|     if (!seg.isActive()) continue;
 | |
| 
 | |
|     // last condition ensures all solid segments are updated at the same time
 | |
|     if(nowUp > seg.next_time || _triggered || (doShow && seg.mode == FX_MODE_STATIC))
 | |
|     {
 | |
|       if (seg.grouping == 0) seg.grouping = 1; //sanity check
 | |
|       doShow = true;
 | |
|       uint16_t delay = FRAMETIME;
 | |
| 
 | |
|       if (!seg.getOption(SEG_OPTION_FREEZE)) { //only run effect function if not frozen
 | |
|         _virtualSegmentLength = seg.virtualLength();
 | |
|         _bri_t = seg.currentBri(seg.getOption(SEG_OPTION_ON) ? seg.opacity : 0);
 | |
|         uint8_t _cct_t = seg.currentBri(seg.cct, true);
 | |
|         _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]);
 | |
|         seg.handleTransition();
 | |
| 
 | |
|         if (!cctFromRgb || correctWB) busses.setSegmentCCT(_cct_t, correctWB);
 | |
|         for (uint8_t c = 0; c < NUM_COLORS; c++) {
 | |
|           _colors_t[c] = gamma32(_colors_t[c]);
 | |
|         }
 | |
|         handle_palette();
 | |
| 
 | |
|         delay = (*_mode[seg.mode])();
 | |
|         if (seg.mode != FX_MODE_HALLOWEEN_EYES) seg.call++;
 | |
|       }
 | |
| 
 | |
|       seg.next_time = nowUp + delay;
 | |
|     }
 | |
|     _segment_index++;
 | |
|   }
 | |
|   _virtualSegmentLength = 0;
 | |
|   busses.setSegmentCCT(-1);
 | |
|   if(doShow) {
 | |
|     yield();
 | |
|     show();
 | |
|   }
 | |
|   _triggered = false;
 | |
|   _isServicing = false;
 | |
| }
 | |
| 
 | |
| void WS2812FX::setPixelColor(int i, uint32_t col)
 | |
| {
 | |
|   if (i >= _length) return;
 | |
| 
 | |
|   // if realtime mode is active and applying to main segment
 | |
|   if (realtimeMode && useMainSegmentOnly) {
 | |
|     Segment &seg = _segments[_mainSegment];
 | |
|     uint16_t len = seg.length();  // length of segment in number of pixels
 | |
| 
 | |
|     if (seg.opacity < 255) {
 | |
|       byte r = scale8(R(col), seg.opacity);
 | |
|       byte g = scale8(G(col), seg.opacity);
 | |
|       byte b = scale8(B(col), seg.opacity);
 | |
|       byte w = scale8(W(col), seg.opacity);
 | |
|       col = RGBW32(r, g, b, w);
 | |
|     }
 | |
| 
 | |
|     // get physical pixel address (taking into account start, grouping, spacing [and offset])
 | |
|     i = i * seg.groupLength();
 | |
|     if (seg.getOption(SEG_OPTION_REVERSED)) { // is segment reversed?
 | |
|       if (seg.getOption(SEG_OPTION_MIRROR)) { // is segment mirrored?
 | |
|         i = (len - 1) / 2 - i;  //only need to index half the pixels
 | |
|       } else {
 | |
|         i = (len - 1) - i;
 | |
|       }
 | |
|     }
 | |
|     i += seg.start; // starting pixel in a group
 | |
| 
 | |
|     // set all the pixels in the group
 | |
|     for (uint16_t j = 0; j < seg.grouping; j++) {
 | |
|       uint16_t indexSet = i + ((seg.getOption(SEG_OPTION_REVERSED)) ? -j : j);
 | |
|       if (indexSet >= seg.start && indexSet < seg.stop) {
 | |
| 
 | |
|         if (seg.getOption(SEG_OPTION_MIRROR)) { //set the corresponding mirrored pixel
 | |
|           uint16_t indexMir = seg.stop - indexSet + seg.start - 1;          
 | |
|           indexMir += seg.offset; // offset/phase
 | |
|           if (indexMir >= seg.stop) indexMir -= len; // wrap
 | |
|           if (indexMir < customMappingSize) indexMir = customMappingTable[indexMir];
 | |
| 
 | |
|           busses.setPixelColor(indexMir, col);
 | |
|         }
 | |
|         indexSet += seg.offset; // offset/phase
 | |
|         if (indexSet >= seg.stop) indexSet -= len; // wrap
 | |
|         if (indexSet < customMappingSize) indexSet = customMappingTable[indexSet];
 | |
| 
 | |
|         busses.setPixelColor(indexSet, col);
 | |
|       }
 | |
|     }
 | |
|   } else {
 | |
|     if (i < customMappingSize) i = customMappingTable[i];
 | |
|     busses.setPixelColor(i, col);
 | |
|   }
 | |
| }
 | |
| 
 | |
| uint32_t WS2812FX::getPixelColor(uint16_t i)
 | |
| {
 | |
|   if (i >= _length) return 0;
 | |
|   //#ifndef WLED_DISABLE_2D
 | |
|   //if (isMatrix) return getPixelColorXY(i%matrixWidth, i/matrixWidth); // compatibility w/ non-effect fn
 | |
|   //#endif
 | |
|   if (i < customMappingSize) i = customMappingTable[i];
 | |
|   return busses.getPixelColor(i);
 | |
| }
 | |
| 
 | |
| 
 | |
| //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!
 | |
| 
 | |
| //fine tune power estimation constants for your setup                  
 | |
| #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
 | |
| 
 | |
| void WS2812FX::estimateCurrentAndLimitBri() {
 | |
|   //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)
 | |
|   bool useWackyWS2815PowerModel = false;
 | |
|   byte actualMilliampsPerLed = milliampsPerLed;
 | |
| 
 | |
|   if(milliampsPerLed == 255) {
 | |
|     useWackyWS2815PowerModel = true;
 | |
|     actualMilliampsPerLed = 12; // from testing an actual strip
 | |
|   }
 | |
| 
 | |
|   if (ablMilliampsMax < 150 || actualMilliampsPerLed == 0) { //0 mA per LED and too low numbers turn off calculation
 | |
|     currentMilliamps = 0;
 | |
|     busses.setBrightness(_brightness);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   uint16_t pLen = getLengthPhysical();
 | |
|   uint32_t puPerMilliamp = 195075 / actualMilliampsPerLed;
 | |
|   uint32_t powerBudget = (ablMilliampsMax - MA_FOR_ESP) * puPerMilliamp; //100mA for ESP power
 | |
|   if (powerBudget > puPerMilliamp * pLen) { //each LED uses about 1mA in standby, exclude that from power budget
 | |
|     powerBudget -= puPerMilliamp * pLen;
 | |
|   } else {
 | |
|     powerBudget = 0;
 | |
|   }
 | |
| 
 | |
|   uint32_t powerSum = 0;
 | |
| 
 | |
|   for (uint8_t b = 0; b < busses.getNumBusses(); b++) {
 | |
|     Bus *bus = busses.getBus(b);
 | |
|     if (bus->getType() >= TYPE_NET_DDP_RGB) continue; //exclude non-physical network busses
 | |
|     uint16_t len = bus->getLength();
 | |
|     uint32_t busPowerSum = 0;
 | |
|     for (uint16_t i = 0; i < len; i++) { //sum up the usage of each LED
 | |
|       uint32_t c = bus->getPixelColor(i);
 | |
|       byte r = R(c), g = G(c), b = B(c), w = W(c);
 | |
| 
 | |
|       if(useWackyWS2815PowerModel) { //ignore white component on WS2815 power calculation
 | |
|         busPowerSum += (MAX(MAX(r,g),b)) * 3;
 | |
|       } else {
 | |
|         busPowerSum += (r + g + b + w);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (bus->isRgbw()) { //RGBW led total output with white LEDs enabled is still 50mA, so each channel uses less
 | |
|       busPowerSum *= 3;
 | |
|       busPowerSum = busPowerSum >> 2; //same as /= 4
 | |
|     }
 | |
|     powerSum += busPowerSum;
 | |
|   }
 | |
| 
 | |
|   uint32_t powerSum0 = powerSum;
 | |
|   powerSum *= _brightness;
 | |
|   
 | |
|   if (powerSum > powerBudget) //scale brightness down to stay in current limit
 | |
|   {
 | |
|     float scale = (float)powerBudget / (float)powerSum;
 | |
|     uint16_t scaleI = scale * 255;
 | |
|     uint8_t scaleB = (scaleI > 255) ? 255 : scaleI;
 | |
|     uint8_t newBri = scale8(_brightness, scaleB);
 | |
|     busses.setBrightness(newBri); //to keep brightness uniform, sets virtual busses too
 | |
|     currentMilliamps = (powerSum0 * newBri) / puPerMilliamp;
 | |
|   } else {
 | |
|     currentMilliamps = powerSum / puPerMilliamp;
 | |
|     busses.setBrightness(_brightness);
 | |
|   }
 | |
|   currentMilliamps += MA_FOR_ESP; //add power of ESP back to estimate
 | |
|   currentMilliamps += pLen; //add standby power back to estimate
 | |
| }
 | |
| 
 | |
| void WS2812FX::show(void) {
 | |
| 
 | |
|   // avoid race condition, caputre _callback value
 | |
|   show_callback callback = _callback;
 | |
|   if (callback) callback();
 | |
| 
 | |
|   estimateCurrentAndLimitBri();
 | |
|   
 | |
|   // 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
 | |
|   busses.show();
 | |
|   unsigned long now = millis();
 | |
|   unsigned long diff = now - _lastShow;
 | |
|   uint16_t fpsCurr = 200;
 | |
|   if (diff > 0) fpsCurr = 1000 / diff;
 | |
|   _cumulativeFps = (3 * _cumulativeFps + fpsCurr) >> 2;
 | |
|   _lastShow = now;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * 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() {
 | |
|   return !busses.canAllShow();
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * 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;
 | |
| }
 | |
| 
 | |
| void WS2812FX::setTargetFps(uint8_t fps) {
 | |
| 	if (fps > 0 && fps <= 120) _targetFps = fps;
 | |
| 	_frametime = 1000 / _targetFps;
 | |
| }
 | |
| 
 | |
| void WS2812FX::setMode(uint8_t segid, uint8_t m) {
 | |
|   if (segid >= _segments.size()) return;
 | |
| //  if (segid >= getMaxSegments()) return;
 | |
|    
 | |
|   if (m >= getModeCount()) m = getModeCount() - 1;
 | |
| 
 | |
|   if (_segments[segid].mode != m) {
 | |
|     //_segments[segid].startTransition(strip.getTransition()); // set effect transitions
 | |
|     _segments[segid].markForReset();
 | |
|     _segments[segid].mode = m;
 | |
|   }
 | |
| }
 | |
| 
 | |
| //applies to all active and selected segments
 | |
| void WS2812FX::setColor(uint8_t slot, uint32_t c) {
 | |
|   if (slot >= NUM_COLORS) return;
 | |
| 
 | |
|   for (segment &seg : _segments) {
 | |
| //  for (int i = 0; i < getMaxSegments(); i++) {
 | |
| //    Segment &seg = getSegment(i);
 | |
|     if (seg.isSelected()) {
 | |
|       seg.setColor(slot, c);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void WS2812FX::setCCT(uint16_t k) {
 | |
|   for (segment &seg : _segments) {
 | |
| //  for (int i = 0; i < getMaxSegments(); i++) {
 | |
| //    Segment &seg = getSegment(i);
 | |
|     if (seg.isActive() && seg.isSelected()) {
 | |
|       seg.setCCT(k);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void WS2812FX::setBrightness(uint8_t b, bool direct) {
 | |
|   if (gammaCorrectBri) b = gamma8(b);
 | |
|   if (_brightness == b) return;
 | |
|   _brightness = b;
 | |
|   if (_brightness == 0) { //unfreeze all segments on power off
 | |
|     for (segment &seg : _segments) {
 | |
| //    for (int i = 0; i < getMaxSegments(); i++) {
 | |
| //      Segment &seg = getSegment(i);
 | |
|       seg.setOption(SEG_OPTION_FREEZE, false);
 | |
|     }
 | |
|   }
 | |
|   if (direct) {
 | |
|     // would be dangerous if applied immediately (could exceed ABL), but will not output until the next show()
 | |
|     busses.setBrightness(b);
 | |
|   } else {
 | |
| 	  unsigned long t = millis();
 | |
|     if (_segments[0].next_time > t + 22 && t - _lastShow > MIN_SHOW_DELAY) show(); //apply brightness change immediately if no refresh soon
 | |
|   }
 | |
| }
 | |
| 
 | |
| uint8_t WS2812FX::getFirstSelectedSegId(void)
 | |
| {
 | |
|   size_t i = 0;
 | |
|   for (segment &seg : _segments) {
 | |
| //  for (int i = 0; i < getMaxSegments(); i++) {
 | |
| //    Segment &seg = getSegment(i);
 | |
|     if (seg.isSelected()) return i;
 | |
|     i++;
 | |
|   }
 | |
|   // if none selected, use the main segment
 | |
|   return getMainSegmentId();
 | |
| }
 | |
| 
 | |
| void WS2812FX::setMainSegmentId(uint8_t n) {
 | |
| //  if (n >= getMaxSegments()) return;
 | |
| //  //use supplied n if active, or first active
 | |
| //  if (_segments[n].isActive()) {
 | |
| //    _mainSegment = n; return;
 | |
| //  }
 | |
| //  for (uint8_t i = 0; i < getMaxSegments(); i++) {
 | |
| //    if (_segments[i].isActive()) {
 | |
| //      _mainSegment = i; return;
 | |
| //    }
 | |
| //  }
 | |
|   _mainSegment = 0;
 | |
|   if (n < _segments.size()) {
 | |
|     _mainSegment = n;
 | |
|   }
 | |
|   return;
 | |
| }
 | |
| 
 | |
| uint8_t WS2812FX::getLastActiveSegmentId(void) {
 | |
| //  for (uint8_t i = getMaxSegments() -1; i > 0; i--) {
 | |
| //    if (_segments[i].isActive()) return i;
 | |
| //  }
 | |
| //  return 0;
 | |
|   return _segments.size()-1;
 | |
| }
 | |
| 
 | |
| uint8_t WS2812FX::getActiveSegmentsNum(void) {
 | |
|   uint8_t c = 0;
 | |
| //  for (uint8_t i = 0; i < getMaxSegments(); i++) {
 | |
|   for (size_t i = 0; i < _segments.size(); i++) {
 | |
|     if (_segments[i].isActive()) c++;
 | |
|   }
 | |
|   return c;
 | |
| }
 | |
| 
 | |
| uint16_t WS2812FX::getLengthPhysical(void) {
 | |
|   uint16_t len = 0;
 | |
|   for (uint8_t b = 0; b < busses.getNumBusses(); b++) {
 | |
|     Bus *bus = busses.getBus(b);
 | |
|     if (bus->getType() >= TYPE_NET_DDP_RGB) continue; //exclude non-physical network busses
 | |
|     len += bus->getLength();
 | |
|   }
 | |
|   return len;
 | |
| }
 | |
| 
 | |
| //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) {
 | |
| 	for (uint8_t b = 0; b < busses.getNumBusses(); b++) {
 | |
|     Bus *bus = busses.getBus(b);
 | |
|     if (bus == nullptr || bus->getLength()==0) break;
 | |
|     switch (bus->getType()) {
 | |
|       case TYPE_SK6812_RGBW:
 | |
|       case TYPE_TM1814:
 | |
|       case TYPE_ANALOG_4CH:
 | |
|         return true;
 | |
|     }
 | |
|   }
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool WS2812FX::hasCCTBus(void) {
 | |
| 	if (cctFromRgb && !correctWB) return false;
 | |
| 	for (uint8_t b = 0; b < busses.getNumBusses(); b++) {
 | |
|     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;
 | |
|     }
 | |
|   }
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| void WS2812FX::purgeSegments(void) {
 | |
|   // remove all inactive segments (from the back)
 | |
|   int deleted = 0;
 | |
|   if (_segments.size() <= 1 || _isServicing) return;
 | |
|   for (int i = _segments.size()-1; i > 0; i--)
 | |
|     if (_segments[i].stop == 0) {
 | |
|       DEBUG_PRINT(F("-- Removing segment: ")); DEBUG_PRINTLN(i);
 | |
|       deleted++;
 | |
|       _segments.erase(_segments.begin() + i);
 | |
|     }
 | |
|   if (deleted) {
 | |
|     _segments.shrink_to_fit();
 | |
|     if (_mainSegment >= _segments.size()) setMainSegmentId(0);
 | |
|   }
 | |
| }
 | |
| 
 | |
| Segment& WS2812FX::getSegment(uint8_t id) {
 | |
| //  return _segments[id >= getMaxSegments() ? getMainSegmentId() : id];
 | |
|   return _segments[id >= _segments.size() ? getMainSegmentId() : id]; // vectors
 | |
| }
 | |
| 
 | |
| void WS2812FX::setSegment(uint8_t n, uint16_t i1, uint16_t i2, uint8_t grouping, uint8_t spacing, uint16_t offset, uint16_t startY, uint16_t stopY) {
 | |
|   if (n >= _segments.size()) return;
 | |
| //  if (n >= getMaxSegments()) return;
 | |
|   Segment& seg = _segments[n];
 | |
| 
 | |
|   //return if neither bounds nor grouping have changed
 | |
|   bool boundsUnchanged = (seg.start == i1 && seg.stop == i2);
 | |
|   if (isMatrix) {
 | |
|     boundsUnchanged &= (seg.startY == startY && seg.stopY == stopY);
 | |
|   }
 | |
|   if (boundsUnchanged
 | |
| 			&& (!grouping || (seg.grouping == grouping && seg.spacing == spacing))
 | |
| 			&& (offset == UINT16_MAX || offset == seg.offset)) return;
 | |
| 
 | |
|   //if (seg.stop) setRange(seg.start, seg.stop -1, BLACK); //turn old segment range off
 | |
|   if (seg.stop) seg.fill(BLACK); //turn old segment range off
 | |
|   if (i2 <= i1) //disable segment
 | |
|   {
 | |
|     // disabled segments should get removed using purgeSegments()
 | |
|     DEBUG_PRINT(F("-- Segment ")); DEBUG_PRINT(n); DEBUG_PRINTLN(F(" marked inactive."));
 | |
|     seg.stop = 0;
 | |
|     //if (seg.name) {
 | |
|     //  delete[] seg.name;
 | |
|     //  seg.name = nullptr;
 | |
|     //}
 | |
|     // if main segment is deleted, set first active as main segment
 | |
|     if (n == _mainSegment) setMainSegmentId(0);
 | |
|     seg.markForReset();
 | |
|     return;
 | |
|   }
 | |
|   if (isMatrix) {
 | |
|     #ifndef WLED_DISABLE_2D
 | |
|     if (i1 < matrixWidth) seg.start = i1;
 | |
|     seg.stop = i2 > matrixWidth ? matrixWidth : i2;
 | |
|     if (startY < matrixHeight) seg.startY = startY;
 | |
|     seg.stopY = stopY > matrixHeight ? matrixHeight : MAX(1,stopY);
 | |
|     #endif
 | |
|   } else {
 | |
|     if (i1 < _length) seg.start = i1;
 | |
|     seg.stop = i2 > _length ? _length : i2;
 | |
|     seg.startY = 0;
 | |
|     seg.stopY  = 1;
 | |
|   }
 | |
|   if (grouping) {
 | |
|     seg.grouping = grouping;
 | |
|     seg.spacing = spacing;
 | |
|   }
 | |
| 	if (offset < UINT16_MAX) seg.offset = offset;
 | |
|   seg.markForReset();
 | |
|   if (!boundsUnchanged) seg.refreshLightCapabilities();
 | |
| }
 | |
| 
 | |
| void WS2812FX::restartRuntime() {
 | |
|   for (segment &seg : _segments) seg.markForReset();
 | |
| //  for (uint8_t i = 0; i < getMaxSegments(); i++) {
 | |
| //    Segment &seg = getSegment(i);
 | |
| //    seg.markForReset();
 | |
| //  }
 | |
| }
 | |
| 
 | |
| void WS2812FX::resetSegments() {
 | |
|   _segments.clear(); // destructs all Segment as part of clearing
 | |
|   segment seg = isMatrix ? Segment(0, matrixWidth, 0, matrixHeight) : Segment(0, _length);
 | |
|   _segments.push_back(seg);
 | |
|   _mainSegment = 0;
 | |
| /*
 | |
|   for (uint8_t i = 0; i < getMaxSegments(); i++) if (_segments[i].name) delete[] _segments[i].name;
 | |
|   _mainSegment = 0;
 | |
|   memset(_segments, 0, sizeof(_segments));
 | |
|   //memset(_segmentruntimes, 0, sizeof(_segmentruntimes));
 | |
|   _segment_index = 0;
 | |
|   _segments[0].mode = DEFAULT_MODE;
 | |
|   _segments[0].colors[0] = DEFAULT_COLOR;
 | |
|   _segments[0].start = 0;
 | |
|   _segments[0].startY = 0;
 | |
|   _segments[0].speed = DEFAULT_SPEED;
 | |
|   _segments[0].intensity = DEFAULT_INTENSITY;
 | |
|   _segments[0].stop = isMatrix ? matrixWidth : _length;
 | |
|   _segments[0].stopY = isMatrix ? matrixHeight : 1;
 | |
|   _segments[0].grouping = 1;
 | |
|   _segments[0].setOption(SEG_OPTION_SELECTED, 1);
 | |
|   _segments[0].setOption(SEG_OPTION_ON, 1);
 | |
|   _segments[0].opacity = 255;
 | |
|   _segments[0].cct = 127;
 | |
|   _segments[0].custom1 = DEFAULT_C1;
 | |
|   _segments[0].custom2 = DEFAULT_C2;
 | |
|   _segments[0].custom3 = DEFAULT_C3;
 | |
| 
 | |
|   for (uint16_t i = 1; i < getMaxSegments(); i++)
 | |
|   {
 | |
|     _segments[i].colors[0] = _segments[i].color_wheel(i*51);
 | |
|     _segments[i].grouping = 1;
 | |
|     _segments[i].setOption(SEG_OPTION_ON, 1);
 | |
|     _segments[i].opacity = 255;
 | |
|     _segments[i].cct = 127;
 | |
|     _segments[i].speed = DEFAULT_SPEED;
 | |
|     _segments[i].intensity = DEFAULT_INTENSITY;
 | |
|     _segments[i].custom1 = DEFAULT_C1;
 | |
|     _segments[i].custom2 = DEFAULT_C2;
 | |
|     _segments[i].custom3 = DEFAULT_C3;
 | |
|     _segments[i].markForReset();
 | |
|   }
 | |
|   _segments[0].markForReset();
 | |
| */
 | |
| }
 | |
| 
 | |
| void WS2812FX::makeAutoSegments(bool forceReset) {
 | |
|   if (isMatrix) {
 | |
|     #ifndef WLED_DISABLE_2D
 | |
|     // only create 1 2D segment
 | |
|     if (forceReset || getSegmentsNum() == 0) resetSegments(); // initialises 1 segment
 | |
|     else if (getActiveSegmentsNum() == 1) {
 | |
|       size_t i = getLastActiveSegmentId();
 | |
|       _segments[i].start  = 0;
 | |
|       _segments[i].stop   = matrixWidth;
 | |
|       _segments[i].startY = 0;
 | |
|       _segments[i].stopY  = matrixHeight;
 | |
|       _segments[i].grouping = 1;
 | |
|       _segments[i].spacing  = 0;
 | |
|       _mainSegment = i;
 | |
|     }
 | |
|     #endif
 | |
|   } else if (autoSegments) { //make one segment per bus
 | |
|     uint16_t segStarts[MAX_NUM_SEGMENTS] = {0};
 | |
|     uint16_t segStops [MAX_NUM_SEGMENTS] = {0};
 | |
|     uint8_t s = 0;
 | |
|     for (uint8_t i = 0; i < busses.getNumBusses(); i++) {
 | |
|       Bus* b = busses.getBus(i);
 | |
| 
 | |
|       segStarts[s] = b->getStart();
 | |
|       segStops[s]  = segStarts[s] + b->getLength();
 | |
| 
 | |
|       //check for overlap with previous segments
 | |
|       for (uint8_t j = 0; j < s; j++) {
 | |
|         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++;
 | |
|     }
 | |
|     _segments.clear();
 | |
|     for (uint8_t i = 0; i < s; i++) {
 | |
|       Segment seg = Segment(segStarts[i], segStops[i]);
 | |
|       seg.setOption(SEG_OPTION_SELECTED, true);
 | |
|       _segments.push_back(seg);
 | |
|     }
 | |
| //    for (uint8_t i = 0; i < getMaxSegments(); i++) {
 | |
| //      _segments[i].setOption(SEG_OPTION_SELECTED, true);
 | |
| //      setSegment(i, segStarts[i], segStops[i]);
 | |
| //    }
 | |
|     _mainSegment = 0;
 | |
|   } else {
 | |
|     if (forceReset || getSegmentsNum() == 0) resetSegments();
 | |
|     //expand the main seg to the entire length, but only if there are no other segments, or reset is forced
 | |
|     else if (getActiveSegmentsNum() == 1) {
 | |
|       size_t i = getLastActiveSegmentId();
 | |
|       _segments[i].start = 0;
 | |
|       _segments[i].stop  = _length;
 | |
|       _mainSegment = 0;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   fixInvalidSegments();
 | |
| }
 | |
| 
 | |
| void WS2812FX::fixInvalidSegments() {
 | |
|   //make sure no segment is longer than total (sanity check)
 | |
|   for (int i = getSegmentsNum()-1; i > 0; i--) {
 | |
|     if (_segments[i].start >= _length) { _segments.erase(_segments.begin()+i); continue; }
 | |
|     if (_segments[i].stop  >  _length) _segments[i].stop = _length;
 | |
|     // this is always called as the last step after finalizeInit(), update covered bus types
 | |
|     _segments[i].refreshLightCapabilities();
 | |
|   }
 | |
| //  for (uint8_t i = 0; i < getMaxSegments(); i++) {
 | |
| //    if (_segments[i].start >= _length) { _segments[i].start = _segments[i].stop = 0; _segments[i].markForReset(); }
 | |
| //    if (_segments[i].stop  >  _length) { _segments[i].stop = _length; _segments[i].markForReset(); }
 | |
| //    // this is always called as the last step after finalizeInit(), update covered bus types
 | |
| //    if (_segments[i].isActive()) _segments[i].refreshLightCapabilities();
 | |
| //  }
 | |
| }
 | |
| 
 | |
| //true if all segments align with a bus, or if a segment covers the total length
 | |
| bool WS2812FX::checkSegmentAlignment() {
 | |
|   bool aligned = false;
 | |
|   for (segment &seg : _segments) {
 | |
| //  for (uint8_t i = 0; i < getMaxSegments(); i++) {
 | |
| //    Segment &seg = getSegment(i);
 | |
|     for (uint8_t b = 0; b<busses.getNumBusses(); b++) {
 | |
|       Bus *bus = busses.getBus(b);
 | |
|       if (seg.start == bus->getStart() && seg.stop == bus->getStart() + bus->getLength()) aligned = true;
 | |
|     }
 | |
|     if (seg.start == 0 && seg.stop == _length) aligned = true;
 | |
|     if (!aligned) return false;
 | |
|   }
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| //After this function is called, setPixelColor() will use that segment (offsets, grouping, ... will apply)
 | |
| //Note: If called in an interrupt (e.g. JSON API), original segment must be restored,
 | |
| //otherwise it can lead to a crash on ESP32 because _segment_index is modified while in use by the main thread
 | |
| uint8_t WS2812FX::setPixelSegment(uint8_t n)
 | |
| {
 | |
|   uint8_t prevSegId = _segment_index;
 | |
|   if (n < _segments.size()) {
 | |
| //  if (n < getMaxSegments()) {
 | |
|     _segment_index = n;
 | |
|     _virtualSegmentLength = _segments[_segment_index].virtualLength();
 | |
|   }
 | |
|   return prevSegId;
 | |
| }
 | |
| 
 | |
| void WS2812FX::setRange(uint16_t i, uint16_t i2, uint32_t col)
 | |
| {
 | |
|   if (i2 >= i)
 | |
|   {
 | |
|     for (uint16_t x = i; x <= i2; x++) setPixelColor(x, col);
 | |
|   } else
 | |
|   {
 | |
|     for (uint16_t x = i2; x <= i; x++) setPixelColor(x, col);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void WS2812FX::setTransitionMode(bool t)
 | |
| {
 | |
|   for (segment &seg : _segments) seg.startTransition(t ? getTransition() : 0);
 | |
| //  for (uint8_t i = 0; i < getMaxSegments(); i++) {
 | |
| //    Segment &seg = getSegment(i);
 | |
| //    seg.startTransition(t ? getTransition() : 0);
 | |
| //  }
 | |
| }
 | |
| 
 | |
| void WS2812FX::load_gradient_palette(uint8_t index)
 | |
| {
 | |
|   byte tcp[72]; //support gradient palettes with up to 18 entries
 | |
|   if (index>127) {
 | |
|     char fileName[32];
 | |
|     strcpy_P(fileName, PSTR("/palette"));
 | |
|     if (index>128) sprintf(fileName +7, "%d", index-127);
 | |
|     strcat(fileName, ".json");
 | |
| 
 | |
|     if (WLED_FS.exists(fileName)) {
 | |
|       StaticJsonDocument<1536> pDoc; // barely enough to fit 72 numbers
 | |
|       DEBUG_PRINT(F("Reading palette from "));
 | |
|       DEBUG_PRINTLN(fileName);
 | |
| 
 | |
|       if (readObjectFromFile(fileName, nullptr, &pDoc)) {
 | |
|         JsonArray pal = pDoc[F("palette")];
 | |
|         if (!pal.isNull() && pal.size()>7) { // not an empty palette (at least 2 entries)
 | |
|           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
 | |
|           }
 | |
|           targetPalette.loadDynamicGradientPalette(tcp);
 | |
|         }
 | |
|       }
 | |
|       releaseJSONBufferLock();
 | |
|     }
 | |
|   } else {
 | |
|     byte i = constrain(index, 0, GRADIENT_PALETTE_COUNT -1);
 | |
|     memcpy_P(tcp, (byte*)pgm_read_dword(&(gGradientPalettes[i])), 72);
 | |
|     targetPalette.loadDynamicGradientPalette(tcp);
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * FastLED palette modes helper function. Limitation: Due to memory reasons, multiple active segments with FastLED will disable the Palette transitions
 | |
|  */
 | |
| void WS2812FX::handle_palette(void)
 | |
| {
 | |
|   bool singleSegmentMode = (_segment_index == _segment_index_palette_last);
 | |
|   _segment_index_palette_last = _segment_index;
 | |
| 
 | |
|   byte paletteIndex = _segments[_segment_index].palette;
 | |
|   if (paletteIndex == 0) //default palette. Differs depending on effect
 | |
|   {
 | |
|     // TODO: get default palette ID from _modeData[]
 | |
|     switch (_segments[_segment_index].mode)
 | |
|     {
 | |
|       case FX_MODE_FIRE_2012  : paletteIndex = 35; break; //heat palette
 | |
|       case FX_MODE_COLORWAVES : paletteIndex = 26; break; //landscape 33
 | |
|       case FX_MODE_FILLNOISE8 : paletteIndex =  9; break; //ocean colors
 | |
|       case FX_MODE_NOISE16_1  : paletteIndex = 20; break; //Drywet
 | |
|       case FX_MODE_NOISE16_2  : paletteIndex = 43; break; //Blue cyan yellow
 | |
|       case FX_MODE_NOISE16_3  : paletteIndex = 35; break; //heat palette
 | |
|       case FX_MODE_NOISE16_4  : paletteIndex = 26; break; //landscape 33
 | |
|       case FX_MODE_GLITTER    : paletteIndex = 11; break; //rainbow colors
 | |
|       case FX_MODE_SUNRISE    : paletteIndex = 35; break; //heat palette
 | |
|       case FX_MODE_FLOW       : paletteIndex =  6; break; //party
 | |
|     }
 | |
|   }
 | |
|   if (_segments[_segment_index].mode >= FX_MODE_METEOR && paletteIndex == 0) paletteIndex = 4;
 | |
|   
 | |
|   switch (paletteIndex)
 | |
|   {
 | |
|     case 0: //default palette. Exceptions for specific effects above
 | |
|       targetPalette = PartyColors_p; break;
 | |
|     case 1: {//periodically replace palette with a random one. Doesn't work with multiple FastLED segments
 | |
|       if (!singleSegmentMode)
 | |
|       {
 | |
|         targetPalette = PartyColors_p; break; //fallback
 | |
|       }
 | |
|       if (millis() - _lastPaletteChange > 1000 + ((uint32_t)(255-_segments[_segment_index].intensity))*100)
 | |
|       {
 | |
|         targetPalette = CRGBPalette16(
 | |
|                         CHSV(random8(), 255, random8(128, 255)),
 | |
|                         CHSV(random8(), 255, random8(128, 255)),
 | |
|                         CHSV(random8(), 192, random8(128, 255)),
 | |
|                         CHSV(random8(), 255, random8(128, 255)));
 | |
|         _lastPaletteChange = millis();
 | |
|       } break;}
 | |
|     case 2: {//primary color only
 | |
|       CRGB prim = CRGB(SEGCOLOR(0));
 | |
|       targetPalette = CRGBPalette16(prim); break;}
 | |
|     case 3: {//primary + secondary
 | |
|       CRGB prim = CRGB(SEGCOLOR(0));
 | |
|       CRGB sec  = CRGB(SEGCOLOR(1));
 | |
|       targetPalette = CRGBPalette16(prim,prim,sec,sec); break;}
 | |
|     case 4: {//primary + secondary + tertiary
 | |
|       CRGB prim = CRGB(SEGCOLOR(0));
 | |
|       CRGB sec  = CRGB(SEGCOLOR(1));
 | |
|       CRGB ter  = CRGB(SEGCOLOR(2));
 | |
|       targetPalette = CRGBPalette16(ter,sec,prim); break;}
 | |
|     case 5: {//primary + secondary (+tert if not off), more distinct
 | |
|       CRGB prim = CRGB(SEGCOLOR(0));
 | |
|       CRGB sec  = CRGB(SEGCOLOR(1));
 | |
|       if (SEGCOLOR(2)) {
 | |
|         CRGB ter = CRGB(SEGCOLOR(2));
 | |
|         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
 | |
|       load_gradient_palette(paletteIndex -13);
 | |
|   }
 | |
|   
 | |
|   if (singleSegmentMode && paletteFade && _segments[_segment_index].call > 0) //only blend if just one segment uses FastLED mode
 | |
|   {
 | |
|     nblendPaletteTowardPalette(currentPalette, targetPalette, 48);
 | |
|   } else
 | |
|   {
 | |
|     currentPalette = targetPalette;
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| //load custom mapping table from JSON file (called from finalizeInit() or deserializeState())
 | |
| void WS2812FX::deserializeMap(uint8_t n) {
 | |
|   if (isMatrix) return; // 2D support creates its own ledmap
 | |
| 
 | |
|   char fileName[32];
 | |
|   strcpy_P(fileName, PSTR("/ledmap"));
 | |
|   if (n) sprintf(fileName +7, "%d", n);
 | |
|   strcat(fileName, ".json");
 | |
|   bool isFile = WLED_FS.exists(fileName);
 | |
| 
 | |
|   if (!isFile) {
 | |
|     // erase custom mapping if selecting nonexistent ledmap.json (n==0)
 | |
|     if (!n && customMappingTable != nullptr) {
 | |
|       customMappingSize = 0;
 | |
|       delete[] customMappingTable;
 | |
|       customMappingTable = nullptr;
 | |
|     }
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!requestJSONBufferLock(7)) return;
 | |
| 
 | |
|   DEBUG_PRINT(F("Reading LED map from "));
 | |
|   DEBUG_PRINTLN(fileName);
 | |
| 
 | |
|   if (!readObjectFromFile(fileName, nullptr, &doc)) {
 | |
|     releaseJSONBufferLock();
 | |
|     return; //if file does not exist just exit
 | |
|   }
 | |
| 
 | |
|   // erase old custom ledmap
 | |
|   if (customMappingTable != nullptr) {
 | |
|     customMappingSize = 0;
 | |
|     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++) {
 | |
|       customMappingTable[i] = (uint16_t) map[i];
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   releaseJSONBufferLock();
 | |
| }
 | |
| 
 | |
| //gamma 2.8 lookup table used for color correction
 | |
| static byte gammaT[] = {
 | |
|     0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,
 | |
|     0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  1,  1,  1,  1,
 | |
|     1,  1,  1,  1,  1,  1,  1,  1,  1,  2,  2,  2,  2,  2,  2,  2,
 | |
|     2,  3,  3,  3,  3,  3,  3,  3,  4,  4,  4,  4,  4,  5,  5,  5,
 | |
|     5,  6,  6,  6,  6,  7,  7,  7,  7,  8,  8,  8,  9,  9,  9, 10,
 | |
|    10, 10, 11, 11, 11, 12, 12, 13, 13, 13, 14, 14, 15, 15, 16, 16,
 | |
|    17, 17, 18, 18, 19, 19, 20, 20, 21, 21, 22, 22, 23, 24, 24, 25,
 | |
|    25, 26, 27, 27, 28, 29, 29, 30, 31, 32, 32, 33, 34, 35, 35, 36,
 | |
|    37, 38, 39, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 50,
 | |
|    51, 52, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 66, 67, 68,
 | |
|    69, 70, 72, 73, 74, 75, 77, 78, 79, 81, 82, 83, 85, 86, 87, 89,
 | |
|    90, 92, 93, 95, 96, 98, 99,101,102,104,105,107,109,110,112,114,
 | |
|   115,117,119,120,122,124,126,127,129,131,133,135,137,138,140,142,
 | |
|   144,146,148,150,152,154,156,158,160,162,164,167,169,171,173,175,
 | |
|   177,180,182,184,186,189,191,193,196,198,200,203,205,208,210,213,
 | |
|   215,218,220,223,225,228,231,233,236,239,241,244,247,249,252,255 };
 | |
| 
 | |
| uint8_t WS2812FX::gamma8_cal(uint8_t b, float gamma) {
 | |
|   return (int)(powf((float)b / 255.0f, gamma) * 255.0f + 0.5f);
 | |
| }
 | |
| 
 | |
| void WS2812FX::calcGammaTable(float gamma)
 | |
| {
 | |
|   for (uint16_t i = 0; i < 256; i++) {
 | |
|     gammaT[i] = gamma8_cal(i, gamma);
 | |
|   }
 | |
| }
 | |
| 
 | |
| uint8_t WS2812FX::gamma8(uint8_t b)
 | |
| {
 | |
|   return gammaT[b];
 | |
| }
 | |
| 
 | |
| uint32_t WS2812FX::gamma32(uint32_t color)
 | |
| {
 | |
|   if (!gammaCorrectCol) return color;
 | |
|   uint8_t w = W(color);
 | |
|   uint8_t r = R(color);
 | |
|   uint8_t g = G(color);
 | |
|   uint8_t b = B(color);
 | |
|   w = gammaT[w];
 | |
|   r = gammaT[r];
 | |
|   g = gammaT[g];
 | |
|   b = gammaT[b];
 | |
|   return RGBW32(r, g, b, w);
 | |
| }
 | |
| 
 | |
| WS2812FX* WS2812FX::instance = nullptr;
 | |
| 
 | |
| //Bus static member definition, would belong in bus_manager.cpp
 | |
| int16_t Bus::_cct = -1;
 | |
| uint8_t Bus::_cctBlend = 0;
 | |
| uint8_t Bus::_gAWM = 255;
 | |
| 
 | |
| const char JSON_mode_names[] PROGMEM = R"=====(["Mode names have moved"])=====";
 | |
| 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",
 | |
| "Pastel","Sunset 2","Beech","Vintage","Departure","Landscape","Beach","Sherbet","Hult","Hult 64",
 | |
| "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"
 | |
| ])=====";
 | 
