sync and merge master
This commit is contained in:
		
							
								
								
									
										1358
									
								
								wled00/FX.cpp
									
									
									
									
									
								
							
							
						
						
									
										1358
									
								
								wled00/FX.cpp
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										358
									
								
								wled00/FX.h
									
									
									
									
									
								
							
							
						
						
									
										358
									
								
								wled00/FX.h
									
									
									
									
									
								
							| @@ -24,21 +24,32 @@ | ||||
|   Modified for WLED | ||||
| */ | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| #ifndef WS2812FX_h | ||||
| #define WS2812FX_h | ||||
|  | ||||
| #include "NpbWrapper.h" | ||||
| #include "const.h" | ||||
|  | ||||
| #define FASTLED_INTERNAL //remove annoying pragma messages | ||||
| #define USE_GET_MILLISECOND_TIMER | ||||
| #include "FastLED.h" | ||||
|  | ||||
| #define DEFAULT_BRIGHTNESS (uint8_t)127 | ||||
| #define DEFAULT_MODE       (uint8_t)0 | ||||
| #define DEFAULT_SPEED      (uint8_t)128 | ||||
| #define DEFAULT_INTENSITY  (uint8_t)128 | ||||
| #define DEFAULT_COLOR      (uint32_t)0xFFAA00 | ||||
|  | ||||
| #define min(a,b) ((a)<(b)?(a):(b)) | ||||
| #define max(a,b) ((a)>(b)?(a):(b)) | ||||
| #ifndef MIN | ||||
| #define MIN(a,b) ((a)<(b)?(a):(b)) | ||||
| #endif | ||||
| #ifndef MAX | ||||
| #define MAX(a,b) ((a)>(b)?(a):(b)) | ||||
| #endif | ||||
|  | ||||
| /* Disable effects with high flash memory usage (currently TV simulator) - saves 18.5kB */ | ||||
| //#define WLED_DISABLE_FX_HIGH_FLASH_USE | ||||
|  | ||||
| /* Not used in all effects yet */ | ||||
| #define WLED_FPS         42 | ||||
| @@ -46,18 +57,24 @@ | ||||
|  | ||||
| /* each segment uses 52 bytes of SRAM memory, so if you're application fails because of | ||||
|   insufficient memory, decreasing MAX_NUM_SEGMENTS may help */ | ||||
| #define MAX_NUM_SEGMENTS 10 | ||||
|  | ||||
| /* How much data bytes all segments combined may allocate */ | ||||
| #ifdef ESP8266 | ||||
| #define MAX_SEGMENT_DATA 2048 | ||||
|   #define MAX_NUM_SEGMENTS    12 | ||||
|   /* How many color transitions can run at once */ | ||||
|   #define MAX_NUM_TRANSITIONS  8 | ||||
|   /* How much data bytes all segments combined may allocate */ | ||||
|   #define MAX_SEGMENT_DATA  2048 | ||||
| #else | ||||
| #define MAX_SEGMENT_DATA 8192 | ||||
|   #define MAX_NUM_SEGMENTS    16 | ||||
|   #define MAX_NUM_TRANSITIONS 16 | ||||
|   #define MAX_SEGMENT_DATA  8192 | ||||
| #endif | ||||
|  | ||||
| #define LED_SKIP_AMOUNT  1 | ||||
| #define MIN_SHOW_DELAY  15 | ||||
|  | ||||
| #define NUM_COLORS       3 /* number of colors per segment */ | ||||
| #define SEGMENT          _segments[_segment_index] | ||||
| #define SEGCOLOR(x)      gamma32(_segments[_segment_index].colors[x]) | ||||
| #define SEGCOLOR(x)      _colors_t[x] | ||||
| #define SEGENV           _segment_runtimes[_segment_index] | ||||
| #define SEGLEN           _virtualSegmentLength | ||||
| #define SEGACT           SEGMENT.stop | ||||
| @@ -80,18 +97,24 @@ | ||||
|  | ||||
| // options | ||||
| // bit    7: segment is in transition mode | ||||
| // bits 2-6: TBD | ||||
| // bits 4-6: TBD | ||||
| // bit    3: mirror effect within segment | ||||
| // bit    2: segment is on | ||||
| // bit    1: reverse segment | ||||
| // bit    0: segment is selected | ||||
| #define NO_OPTIONS   (uint8_t)0x00 | ||||
| #define TRANSITIONAL (uint8_t)0x80 | ||||
| #define MIRROR       (uint8_t)0x08 | ||||
| #define SEGMENT_ON   (uint8_t)0x04 | ||||
| #define REVERSE      (uint8_t)0x02 | ||||
| #define SELECTED     (uint8_t)0x01 | ||||
| #define IS_TRANSITIONAL ((SEGMENT.options & TRANSITIONAL) == TRANSITIONAL) | ||||
| #define IS_REVERSE      ((SEGMENT.options & REVERSE )     == REVERSE     ) | ||||
| #define IS_SELECTED     ((SEGMENT.options & SELECTED)     == SELECTED    ) | ||||
| #define IS_MIRROR       ((SEGMENT.options & MIRROR      ) == MIRROR      ) | ||||
| #define IS_SEGMENT_ON   ((SEGMENT.options & SEGMENT_ON  ) == SEGMENT_ON  ) | ||||
| #define IS_REVERSE      ((SEGMENT.options & REVERSE     ) == REVERSE     ) | ||||
| #define IS_SELECTED     ((SEGMENT.options & SELECTED    ) == SELECTED    ) | ||||
|  | ||||
| #define MODE_COUNT  102 | ||||
| #define MODE_COUNT  119 | ||||
|  | ||||
| #define FX_MODE_STATIC                   0 | ||||
| #define FX_MODE_BLINK                    1 | ||||
| @@ -131,13 +154,13 @@ | ||||
| #define FX_MODE_TRAFFIC_LIGHT           35 | ||||
| #define FX_MODE_COLOR_SWEEP_RANDOM      36 | ||||
| #define FX_MODE_RUNNING_COLOR           37 | ||||
| #define FX_MODE_RUNNING_RED_BLUE        38 | ||||
| #define FX_MODE_AURORA                  38 | ||||
| #define FX_MODE_RUNNING_RANDOM          39 | ||||
| #define FX_MODE_LARSON_SCANNER          40 | ||||
| #define FX_MODE_COMET                   41 | ||||
| #define FX_MODE_FIREWORKS               42 | ||||
| #define FX_MODE_RAIN                    43 | ||||
| #define FX_MODE_MERRY_CHRISTMAS         44 | ||||
| #define FX_MODE_TETRIX                  44 | ||||
| #define FX_MODE_FIRE_FLICKER            45 | ||||
| #define FX_MODE_GRADIENT                46 | ||||
| #define FX_MODE_LOADING                 47 | ||||
| @@ -194,13 +217,33 @@ | ||||
| #define FX_MODE_PERCENT                 98 | ||||
| #define FX_MODE_RIPPLE_RAINBOW          99 | ||||
| #define FX_MODE_HEARTBEAT              100 | ||||
| #define FX_MODE_RUNNING_DUAL           101 | ||||
| #define FX_MODE_PACIFICA               101 | ||||
| #define FX_MODE_CANDLE_MULTI           102 | ||||
| #define FX_MODE_SOLID_GLITTER          103 | ||||
| #define FX_MODE_SUNRISE                104 | ||||
| #define FX_MODE_PHASED                 105 | ||||
| #define FX_MODE_TWINKLEUP              106 | ||||
| #define FX_MODE_NOISEPAL               107 | ||||
| #define FX_MODE_SINEWAVE               108 | ||||
| #define FX_MODE_PHASEDNOISE            109 | ||||
| #define FX_MODE_FLOW                   110 | ||||
| #define FX_MODE_CHUNCHUN               111 | ||||
| #define FX_MODE_DANCING_SHADOWS        112 | ||||
| #define FX_MODE_WASHING_MACHINE        113 | ||||
| #define FX_MODE_CANDY_CANE             114 | ||||
| #define FX_MODE_BLENDS                 115 | ||||
| #define FX_MODE_TV_SIMULATOR           116 | ||||
| #define FX_MODE_DYNAMIC_SMOOTH         117 | ||||
| #define FX_MODE_RUNNING_DUAL           118 | ||||
|  | ||||
|  | ||||
| class WS2812FX { | ||||
|   typedef uint16_t (WS2812FX::*mode_ptr)(void); | ||||
|  | ||||
|   // pre show callback | ||||
|   typedef void (*show_callback) (void); | ||||
|  | ||||
|   static WS2812FX* instance; | ||||
|    | ||||
|   // segment parameters | ||||
|   public: | ||||
| @@ -211,18 +254,44 @@ class WS2812FX { | ||||
|       uint8_t intensity; | ||||
|       uint8_t palette; | ||||
|       uint8_t mode; | ||||
|       uint8_t options; //bit pattern: msb first: transitional tbd tbd tbd tbd paused reverse selected | ||||
|       uint8_t options; //bit pattern: msb first: transitional needspixelstate tbd tbd (paused) on reverse selected | ||||
|       uint8_t grouping, spacing; | ||||
|       uint8_t opacity; | ||||
|       uint32_t colors[NUM_COLORS]; | ||||
|       void setOption(uint8_t n, bool val) | ||||
|       bool setColor(uint8_t slot, uint32_t c, uint8_t segn) { //returns true if changed | ||||
|         if (slot >= NUM_COLORS || segn >= MAX_NUM_SEGMENTS) return false; | ||||
|         if (c == colors[slot]) return false; | ||||
|         ColorTransition::startTransition(opacity, colors[slot], instance->_transitionDur, segn, slot); | ||||
|         colors[slot] = c; return true; | ||||
|       } | ||||
|       void setOpacity(uint8_t o, uint8_t segn) { | ||||
|         if (segn >= MAX_NUM_SEGMENTS) return; | ||||
|         if (opacity == o) return; | ||||
|         ColorTransition::startTransition(opacity, colors[0], instance->_transitionDur, segn, 0); | ||||
|         opacity = o; | ||||
|       } | ||||
|       /*uint8_t actualOpacity() { //respects On/Off state | ||||
|         if (!getOption(SEG_OPTION_ON)) return 0; | ||||
|         return opacity; | ||||
|       }*/ | ||||
|       void setOption(uint8_t n, bool val, uint8_t segn = 255) | ||||
|       { | ||||
|         //bool prevOn = false; | ||||
|         //if (n == SEG_OPTION_ON) prevOn = getOption(SEG_OPTION_ON); | ||||
|         if (val) { | ||||
|           options |= 0x01 << n; | ||||
|         } else | ||||
|         { | ||||
|           options &= ~(0x01 << n); | ||||
|         } | ||||
|         //transitions on segment on/off don't work correctly at this point | ||||
|         /*if (n == SEG_OPTION_ON && segn < MAX_NUM_SEGMENTS && getOption(SEG_OPTION_ON) != prevOn) { | ||||
|           if (getOption(SEG_OPTION_ON)) { | ||||
|             ColorTransition::startTransition(0, colors[0], instance->_transitionDur, segn, 0); | ||||
|           } else { | ||||
|             ColorTransition::startTransition(opacity, colors[0], instance->_transitionDur, segn, 0); | ||||
|           } | ||||
|         }*/ | ||||
|       } | ||||
|       bool getOption(uint8_t n) | ||||
|       { | ||||
| @@ -247,7 +316,10 @@ class WS2812FX { | ||||
|       uint16_t virtualLength() | ||||
|       { | ||||
|         uint16_t groupLen = groupLength(); | ||||
|         return (length() + groupLen -1) / groupLen; | ||||
|         uint16_t vLength = (length() + groupLen - 1) / groupLen; | ||||
|         if (options & MIRROR) | ||||
|           vLength = (vLength + 1) /2;  // divide by 2 if mirror, leave at least a single LED | ||||
|         return vLength; | ||||
|       } | ||||
|     } segment; | ||||
|  | ||||
| @@ -262,10 +334,10 @@ class WS2812FX { | ||||
|       bool allocateData(uint16_t len){ | ||||
|         if (data && _dataLen == len) return true; //already allocated | ||||
|         deallocateData(); | ||||
|         if (WS2812FX::_usedSegmentData + len > MAX_SEGMENT_DATA) return false; //not enough memory | ||||
|         if (WS2812FX::instance->_usedSegmentData + len > MAX_SEGMENT_DATA) return false; //not enough memory | ||||
|         data = new (std::nothrow) byte[len]; | ||||
|         if (!data) return false; //allocation failed | ||||
|         WS2812FX::_usedSegmentData += len; | ||||
|         WS2812FX::instance->_usedSegmentData += len; | ||||
|         _dataLen = len; | ||||
|         memset(data, 0, len); | ||||
|         return true; | ||||
| @@ -273,15 +345,116 @@ class WS2812FX { | ||||
|       void deallocateData(){ | ||||
|         delete[] data; | ||||
|         data = nullptr; | ||||
|         WS2812FX::_usedSegmentData -= _dataLen; | ||||
|         WS2812FX::instance->_usedSegmentData -= _dataLen; | ||||
|         _dataLen = 0; | ||||
|       } | ||||
|       void reset(){next_time = 0; step = 0; call = 0; aux0 = 0; aux1 = 0; deallocateData();} | ||||
|  | ||||
|       /**  | ||||
|        * If reset of this segment was request, 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 resetIfRequired() { | ||||
|         if (_requiresReset) { | ||||
|           next_time = 0; step = 0; call = 0; aux0 = 0; aux1 = 0;  | ||||
|           deallocateData(); | ||||
|           _requiresReset = false; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       /**  | ||||
|        * Flags that before the next effect is calculated, | ||||
|        * the internal segment state should be reset.  | ||||
|        * Call resetIfRequired before calling the next effect function. | ||||
|        */ | ||||
|       void reset() { _requiresReset = true; } | ||||
|       private: | ||||
|         uint16_t _dataLen = 0; | ||||
|         bool _requiresReset = false; | ||||
|     } segment_runtime; | ||||
|  | ||||
|     typedef struct ColorTransition { // 12 bytes | ||||
|       uint32_t colorOld = 0; | ||||
|       uint32_t transitionStart; | ||||
|       uint16_t transitionDur; | ||||
|       uint8_t segment = 0xFF; //lower 6 bits: the segment this transition is for (255 indicates transition not in use/available) upper 2 bits: color channel | ||||
|       uint8_t briOld = 0; | ||||
|       static void startTransition(uint8_t oldBri, uint32_t oldCol, uint16_t dur, uint8_t segn, uint8_t slot) { | ||||
|         if (segn >= MAX_NUM_SEGMENTS || slot >= NUM_COLORS || dur == 0) return; | ||||
|         if (instance->_brightness == 0) return; //do not need transitions if master bri is off | ||||
|         uint8_t tIndex = 0xFF; //none found | ||||
|         uint16_t tProgression = 0; | ||||
|         uint8_t s = segn + (slot << 6); //merge slot and segment into one byte | ||||
|  | ||||
|         for (uint8_t i = 0; i < MAX_NUM_TRANSITIONS; i++) { | ||||
|           uint8_t tSeg = instance->transitions[i].segment; | ||||
|           //see if this segment + color already has a running transition | ||||
|           if (tSeg == s) { | ||||
|             tIndex = i; break; | ||||
|           } | ||||
|           if (tSeg == 0xFF) { //free transition | ||||
|             tIndex = i; tProgression = 0xFFFF; | ||||
|           } | ||||
|         } | ||||
|  | ||||
|         if (tIndex == 0xFF) { //no slot found yet | ||||
|           for (uint8_t i = 0; i < MAX_NUM_TRANSITIONS; i++) { | ||||
|             //find most progressed transition to overwrite | ||||
|             uint16_t prog = instance->transitions[i].progress(); | ||||
|             if (prog > tProgression) { | ||||
|               tIndex = i; tProgression = prog; | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|  | ||||
|         ColorTransition& t = instance->transitions[tIndex]; | ||||
|         if (t.segment == s) //this is an active transition on the same segment+color | ||||
|         { | ||||
|           t.briOld = t.currentBri(); | ||||
|           t.colorOld = t.currentColor(oldCol); | ||||
|         } else { | ||||
|           t.briOld = oldBri; | ||||
|           t.colorOld = oldCol; | ||||
|           uint8_t prevSeg = t.segment & 0x3F; | ||||
|           if (prevSeg < MAX_NUM_SEGMENTS) instance->_segments[prevSeg].setOption(SEG_OPTION_TRANSITIONAL, false); | ||||
|         } | ||||
|         t.transitionDur = dur; | ||||
|         t.transitionStart = millis(); | ||||
|         t.segment = s; | ||||
|         instance->_segments[segn].setOption(SEG_OPTION_TRANSITIONAL, true); | ||||
|         //refresh immediately, required for Solid mode | ||||
|         if (instance->_segment_runtimes[segn].next_time > t.transitionStart + 22) instance->_segment_runtimes[segn].next_time = t.transitionStart; | ||||
|       } | ||||
|       uint16_t progress(bool allowEnd = false) { //transition progression between 0-65535 | ||||
|         uint32_t timeNow = millis(); | ||||
|         if (timeNow - transitionStart > transitionDur) { | ||||
|           if (allowEnd) { | ||||
|             uint8_t segn = segment & 0x3F; | ||||
|             if (segn < MAX_NUM_SEGMENTS) instance->_segments[segn].setOption(SEG_OPTION_TRANSITIONAL, false); | ||||
|             segment = 0xFF; | ||||
|           } | ||||
|           return 0xFFFF; | ||||
|         } | ||||
|         uint32_t elapsed = timeNow - transitionStart; | ||||
|         uint32_t prog = elapsed * 0xFFFF / transitionDur; | ||||
|         return (prog > 0xFFFF) ? 0xFFFF : prog; | ||||
|       } | ||||
|       uint32_t currentColor(uint32_t colorNew) { | ||||
|         return instance->color_blend(colorOld, colorNew, progress(true), true); | ||||
|       } | ||||
|       uint8_t currentBri() { | ||||
|         uint8_t segn = segment & 0x3F; | ||||
|         if (segn >= MAX_NUM_SEGMENTS) return 0; | ||||
|         uint8_t briNew = instance->_segments[segn].opacity; | ||||
|         uint32_t prog = progress() + 1; | ||||
|         return ((briNew * prog) + (briOld * (0x10000 - prog))) >> 16; | ||||
|       } | ||||
|     } color_transition; | ||||
|  | ||||
|     WS2812FX() { | ||||
|       WS2812FX::instance = this; | ||||
|       //assign each member of the _mode[] array to its respective function reference  | ||||
|       _mode[FX_MODE_STATIC]                  = &WS2812FX::mode_static; | ||||
|       _mode[FX_MODE_BLINK]                   = &WS2812FX::mode_blink; | ||||
| @@ -319,13 +492,13 @@ class WS2812FX { | ||||
|       _mode[FX_MODE_TRAFFIC_LIGHT]           = &WS2812FX::mode_traffic_light; | ||||
|       _mode[FX_MODE_COLOR_SWEEP_RANDOM]      = &WS2812FX::mode_color_sweep_random; | ||||
|       _mode[FX_MODE_RUNNING_COLOR]           = &WS2812FX::mode_running_color; | ||||
|       _mode[FX_MODE_RUNNING_RED_BLUE]        = &WS2812FX::mode_running_red_blue; | ||||
|       _mode[FX_MODE_AURORA]                  = &WS2812FX::mode_aurora; | ||||
|       _mode[FX_MODE_RUNNING_RANDOM]          = &WS2812FX::mode_running_random; | ||||
|       _mode[FX_MODE_LARSON_SCANNER]          = &WS2812FX::mode_larson_scanner; | ||||
|       _mode[FX_MODE_COMET]                   = &WS2812FX::mode_comet; | ||||
|       _mode[FX_MODE_FIREWORKS]               = &WS2812FX::mode_fireworks; | ||||
|       _mode[FX_MODE_RAIN]                    = &WS2812FX::mode_rain; | ||||
|       _mode[FX_MODE_MERRY_CHRISTMAS]         = &WS2812FX::mode_merry_christmas; | ||||
|       _mode[FX_MODE_TETRIX]                  = &WS2812FX::mode_tetrix; | ||||
|       _mode[FX_MODE_FIRE_FLICKER]            = &WS2812FX::mode_fire_flicker; | ||||
|       _mode[FX_MODE_GRADIENT]                = &WS2812FX::mode_gradient; | ||||
|       _mode[FX_MODE_LOADING]                 = &WS2812FX::mode_loading; | ||||
| @@ -384,6 +557,23 @@ class WS2812FX { | ||||
|       _mode[FX_MODE_PERCENT]                 = &WS2812FX::mode_percent; | ||||
|       _mode[FX_MODE_RIPPLE_RAINBOW]          = &WS2812FX::mode_ripple_rainbow; | ||||
|       _mode[FX_MODE_HEARTBEAT]               = &WS2812FX::mode_heartbeat; | ||||
|       _mode[FX_MODE_PACIFICA]                = &WS2812FX::mode_pacifica; | ||||
|       _mode[FX_MODE_CANDLE_MULTI]            = &WS2812FX::mode_candle_multi; | ||||
|       _mode[FX_MODE_SOLID_GLITTER]           = &WS2812FX::mode_solid_glitter; | ||||
|       _mode[FX_MODE_SUNRISE]                 = &WS2812FX::mode_sunrise; | ||||
|       _mode[FX_MODE_PHASED]                  = &WS2812FX::mode_phased; | ||||
|       _mode[FX_MODE_TWINKLEUP]               = &WS2812FX::mode_twinkleup; | ||||
|       _mode[FX_MODE_NOISEPAL]                = &WS2812FX::mode_noisepal; | ||||
|       _mode[FX_MODE_SINEWAVE]                = &WS2812FX::mode_sinewave; | ||||
|       _mode[FX_MODE_PHASEDNOISE]             = &WS2812FX::mode_phased_noise; | ||||
|       _mode[FX_MODE_FLOW]                    = &WS2812FX::mode_flow; | ||||
|       _mode[FX_MODE_CHUNCHUN]                = &WS2812FX::mode_chunchun; | ||||
|       _mode[FX_MODE_DANCING_SHADOWS]         = &WS2812FX::mode_dancing_shadows; | ||||
|       _mode[FX_MODE_WASHING_MACHINE]         = &WS2812FX::mode_washing_machine; | ||||
|       _mode[FX_MODE_CANDY_CANE]              = &WS2812FX::mode_candy_cane; | ||||
|       _mode[FX_MODE_BLENDS]                  = &WS2812FX::mode_blends; | ||||
|       _mode[FX_MODE_TV_SIMULATOR]            = &WS2812FX::mode_tv_simulator; | ||||
|       _mode[FX_MODE_DYNAMIC_SMOOTH]          = &WS2812FX::mode_dynamic_smooth; | ||||
|       _mode[FX_MODE_RUNNING_DUAL]            = &WS2812FX::mode_running_dual; | ||||
|  | ||||
|       _brightness = DEFAULT_BRIGHTNESS; | ||||
| @@ -392,46 +582,51 @@ class WS2812FX { | ||||
|       ablMilliampsMax = 850; | ||||
|       currentMilliamps = 0; | ||||
|       timebase = 0; | ||||
|       bus = new NeoPixelWrapper(); | ||||
|       resetSegments(); | ||||
|     } | ||||
|  | ||||
|     void | ||||
|       init(bool supportWhite, uint16_t countPixels, bool skipFirst), | ||||
|       finalizeInit(uint16_t countPixels, bool skipFirst), | ||||
|       service(void), | ||||
|       blur(uint8_t), | ||||
|       fill(uint32_t), | ||||
|       fade_out(uint8_t r), | ||||
|       setMode(uint8_t segid, uint8_t m), | ||||
|       setColor(uint8_t slot, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0), | ||||
|       setColor(uint8_t slot, uint32_t c), | ||||
|       setBrightness(uint8_t b), | ||||
|       driverModeCronixie(bool b), | ||||
|       setCronixieDigits(byte* d), | ||||
|       setCronixieBacklight(bool b), | ||||
|       setRange(uint16_t i, uint16_t i2, uint32_t col), | ||||
|       setShowCallback(show_callback cb), | ||||
|       setTransition(uint16_t t), | ||||
|       setTransitionMode(bool t), | ||||
|       calcGammaTable(float), | ||||
|       trigger(void), | ||||
|       setSegment(uint8_t n, uint16_t start, uint16_t stop, uint8_t grouping = 0, uint8_t spacing = 0), | ||||
|       resetSegments(), | ||||
|       setPixelColor(uint16_t n, uint32_t c), | ||||
|       setPixelColor(uint16_t n, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0), | ||||
|       show(void); | ||||
|       show(void), | ||||
|       setColorOrder(uint8_t co), | ||||
|       setPixelSegment(uint8_t n); | ||||
|  | ||||
|     bool | ||||
|       reverseMode = false, | ||||
|       isRgbw = false, | ||||
|       gammaCorrectBri = false, | ||||
|       gammaCorrectCol = true, | ||||
|       applyToAllSelected = true, | ||||
|       segmentsAreIdentical(Segment* a, Segment* b), | ||||
|       setEffectConfig(uint8_t m, uint8_t s, uint8_t i, uint8_t p); | ||||
|       setEffectConfig(uint8_t m, uint8_t s, uint8_t i, uint8_t p), | ||||
|       // return true if the strip is being sent pixel updates | ||||
|       isUpdating(void); | ||||
|  | ||||
|     uint8_t | ||||
|       mainSegment = 0, | ||||
|       rgbwMode = RGBW_MODE_DUAL, | ||||
|       paletteFade = 0, | ||||
|       paletteBlend = 0, | ||||
|       colorOrder = 0, | ||||
|       milliampsPerLed = 55, | ||||
| //      getStripType(uint8_t strip=0), | ||||
| //      setStripType(uint8_t type, uint8_t strip=0), | ||||
|       getBrightness(void), | ||||
|       getMode(void), | ||||
|       getSpeed(void), | ||||
| @@ -440,19 +635,33 @@ class WS2812FX { | ||||
|       getMaxSegments(void), | ||||
|       //getFirstSelectedSegment(void), | ||||
|       getMainSegmentId(void), | ||||
|       getColorOrder(void), | ||||
|       gamma8(uint8_t), | ||||
|       gamma8_cal(uint8_t, float), | ||||
|       get_random_wheel_index(uint8_t); | ||||
|  | ||||
|     int8_t | ||||
| //      setStripPin(uint8_t strip, int8_t pin), | ||||
| //      getStripPin(uint8_t strip=0), | ||||
| //      setStripPinClk(uint8_t strip, int8_t pin), | ||||
| //      getStripPinClk(uint8_t strip=0), | ||||
|       tristate_square8(uint8_t x, uint8_t pulsewidth, uint8_t attdec); | ||||
|  | ||||
|     uint16_t | ||||
|       ablMilliampsMax, | ||||
|       currentMilliamps, | ||||
|       triwave16(uint16_t); | ||||
| //      setStripLen(uint8_t strip, uint16_t len), | ||||
| //      getStripLen(uint8_t strip=0), | ||||
|       triwave16(uint16_t), | ||||
|       getFps(); | ||||
|  | ||||
|     uint32_t | ||||
|       now, | ||||
|       timebase, | ||||
|       color_wheel(uint8_t), | ||||
|       color_from_palette(uint16_t, bool, bool, uint8_t, uint8_t pbri = 255), | ||||
|       color_blend(uint32_t,uint32_t,uint8_t), | ||||
|       color_from_palette(uint16_t, bool mapping, bool wrap, uint8_t mcol, uint8_t pbri = 255), | ||||
|       color_blend(uint32_t,uint32_t,uint16_t,bool b16=false), | ||||
|       currentColor(uint32_t colorNew, uint8_t tNr), | ||||
|       gamma32(uint32_t), | ||||
|       getLastShow(void), | ||||
|       getPixelColor(uint16_t), | ||||
| @@ -507,13 +716,13 @@ class WS2812FX { | ||||
|       mode_colorful(void), | ||||
|       mode_traffic_light(void), | ||||
|       mode_running_color(void), | ||||
|       mode_running_red_blue(void), | ||||
|       mode_aurora(void), | ||||
|       mode_running_random(void), | ||||
|       mode_larson_scanner(void), | ||||
|       mode_comet(void), | ||||
|       mode_fireworks(void), | ||||
|       mode_rain(void), | ||||
|       mode_merry_christmas(void), | ||||
|       mode_tetrix(void), | ||||
|       mode_halloween(void), | ||||
|       mode_fire_flicker(void), | ||||
|       mode_gradient(void), | ||||
| @@ -554,7 +763,7 @@ class WS2812FX { | ||||
|       mode_twinklecat(void), | ||||
|       mode_halloween_eyes(void), | ||||
|       mode_static_pattern(void), | ||||
| 	    mode_tri_static_pattern(void), | ||||
|       mode_tri_static_pattern(void), | ||||
|       mode_spots(void), | ||||
|       mode_spots_fade(void), | ||||
|       mode_glitter(void), | ||||
| @@ -571,35 +780,46 @@ class WS2812FX { | ||||
|       mode_percent(void), | ||||
|       mode_ripple_rainbow(void), | ||||
|       mode_heartbeat(void), | ||||
|       mode_pacifica(void), | ||||
|       mode_candle_multi(void), | ||||
|       mode_solid_glitter(void), | ||||
|       mode_sunrise(void), | ||||
|       mode_phased(void), | ||||
|       mode_twinkleup(void), | ||||
|       mode_noisepal(void), | ||||
|       mode_sinewave(void), | ||||
|       mode_phased_noise(void), | ||||
|       mode_flow(void), | ||||
|       mode_chunchun(void), | ||||
|       mode_dancing_shadows(void), | ||||
|       mode_washing_machine(void), | ||||
|       mode_candy_cane(void), | ||||
|       mode_blends(void), | ||||
|       mode_tv_simulator(void), | ||||
|       mode_dynamic_smooth(void), | ||||
|       mode_running_dual(void); | ||||
|        | ||||
|  | ||||
|   private: | ||||
|     NeoPixelWrapper *bus; | ||||
|  | ||||
|     uint32_t crgb_to_col(CRGB fastled); | ||||
|     CRGB col_to_crgb(uint32_t); | ||||
|     CRGBPalette16 currentPalette; | ||||
|     CRGBPalette16 targetPalette; | ||||
|  | ||||
|     uint32_t now; | ||||
|     uint16_t _length, _lengthRaw, _virtualSegmentLength; | ||||
|     uint16_t _rand16seed; | ||||
|     uint8_t _brightness; | ||||
|     static uint16_t _usedSegmentData; | ||||
|     uint16_t _usedSegmentData = 0; | ||||
|     uint16_t _transitionDur = 750; | ||||
|  | ||||
|     uint16_t _cumulativeFps = 2; | ||||
|  | ||||
|     void load_gradient_palette(uint8_t); | ||||
|     void handle_palette(void); | ||||
|     void fill(uint32_t); | ||||
|  | ||||
|     bool | ||||
|       _rgbwMode, | ||||
|       _cronixieMode, | ||||
|       _cronixieBacklightEnabled, | ||||
|       _skipFirstMode, | ||||
|       _triggered; | ||||
|  | ||||
|     byte _cronixieDigits[6]; | ||||
|  | ||||
|     mode_ptr _mode[MODE_COUNT]; // SRAM footprint: 4 bytes per element | ||||
|  | ||||
|     show_callback _callback = nullptr; | ||||
| @@ -607,7 +827,9 @@ class WS2812FX { | ||||
|     // mode helper functions | ||||
|     uint16_t | ||||
|       blink(uint32_t, uint32_t, bool strobe, bool), | ||||
|       candle(bool), | ||||
|       color_wipe(bool, bool), | ||||
|       dynamic(bool), | ||||
|       scan(bool), | ||||
|       theater_chase(uint32_t, uint32_t, bool), | ||||
|       running_base(bool,bool), | ||||
| @@ -621,12 +843,25 @@ class WS2812FX { | ||||
|       running(uint32_t, uint32_t), | ||||
|       tricolor_chase(uint32_t, uint32_t), | ||||
|       twinklefox_base(bool), | ||||
|       spots_base(uint16_t); | ||||
|       spots_base(uint16_t), | ||||
|       phased_base(uint8_t); | ||||
|  | ||||
|     CRGB twinklefox_one_twinkle(uint32_t ms, uint8_t salt, bool cat); | ||||
|     CRGB pacifica_one_layer(uint16_t i, CRGBPalette16& p, uint16_t cistart, uint16_t wavescale, uint8_t bri, uint16_t ioff); | ||||
|  | ||||
|     void | ||||
|       blendPixelColor(uint16_t n, uint32_t color, uint8_t blend), | ||||
|       startTransition(uint8_t oldBri, uint32_t oldCol, uint16_t dur, uint8_t segn, uint8_t slot), | ||||
|       deserializeMap(void); | ||||
|  | ||||
|     uint16_t* customMappingTable = nullptr; | ||||
|     uint16_t  customMappingSize  = 0; | ||||
|      | ||||
|     uint32_t _lastPaletteChange = 0; | ||||
|     uint32_t _lastShow = 0; | ||||
|  | ||||
|     uint32_t _colors_t[3]; | ||||
|     uint8_t _bri_t; | ||||
|      | ||||
|     uint8_t _segment_index = 0; | ||||
|     uint8_t _segment_index_palette_last = 99; | ||||
| @@ -637,33 +872,38 @@ class WS2812FX { | ||||
|     segment_runtime _segment_runtimes[MAX_NUM_SEGMENTS]; // SRAM footprint: 28 bytes per element | ||||
|     friend class Segment_runtime; | ||||
|  | ||||
|     uint16_t realPixelIndex(uint16_t i); | ||||
| }; | ||||
|     ColorTransition transitions[MAX_NUM_TRANSITIONS]; //12 bytes per element | ||||
|     friend class ColorTransition; | ||||
|  | ||||
|     uint16_t | ||||
|       realPixelIndex(uint16_t i), | ||||
|       transitionProgress(uint8_t tNr); | ||||
| }; | ||||
|  | ||||
| //10 names per line | ||||
| const char JSON_mode_names[] PROGMEM = R"=====([ | ||||
| "Solid","Blink","Breathe","Wipe","Wipe Random","Random Colors","Sweep","Dynamic","Colorloop","Rainbow", | ||||
| "Scan","Scan Dual","Fade","Theater","Theater Rainbow","Running","Saw","Twinkle","Dissolve","Dissolve Rnd", | ||||
| "Sparkle","Sparkle Dark","Sparkle+","Strobe","Strobe Rainbow","Strobe Mega","Blink Rainbow","Android","Chase","Chase Random", | ||||
| "Chase Rainbow","Chase Flash","Chase Flash Rnd","Rainbow Runner","Colorful","Traffic Light","Sweep Random","Running 2","Red & Blue","Stream", | ||||
| "Scanner","Lighthouse","Fireworks","Rain","Merry Christmas","Fire Flicker","Gradient","Loading","Police","Police All", | ||||
| "Chase Rainbow","Chase Flash","Chase Flash Rnd","Rainbow Runner","Colorful","Traffic Light","Sweep Random","Running 2","Aurora","Stream", | ||||
| "Scanner","Lighthouse","Fireworks","Rain","Tetrix","Fire Flicker","Gradient","Loading","Police","Police All", | ||||
| "Two Dots","Two Areas","Circus","Halloween","Tri Chase","Tri Wipe","Tri Fade","Lightning","ICU","Multi Comet", | ||||
| "Scanner Dual","Stream 2","Oscillate","Pride 2015","Juggle","Palette","Fire 2012","Colorwaves","Bpm","Fill Noise", | ||||
| "Noise 1","Noise 2","Noise 3","Noise 4","Colortwinkles","Lake","Meteor","Meteor Smooth","Railway","Ripple", | ||||
| "Twinklefox","Twinklecat","Halloween Eyes","Solid Pattern","Solid Pattern Tri","Spots","Spots Fade","Glitter","Candle","Fireworks Starburst", | ||||
| "Fireworks 1D","Bouncing Balls","Sinelon","Sinelon Dual","Sinelon Rainbow","Popcorn","Drip","Plasma","Percent","Ripple Rainbow", | ||||
| "Heartbeat","Running Dual" | ||||
| "Heartbeat","Pacifica","Candle Multi", "Solid Glitter","Sunrise","Phased","Twinkleup","Noise Pal", "Sine","Phased Noise", | ||||
| "Flow","Chunchun","Dancing Shadows","Washing Machine","Candy Cane","Blends","TV Simulator","Dynamic Smooth","Running Dual" | ||||
| ])====="; | ||||
|  | ||||
|  | ||||
| const char JSON_palette_names[] PROGMEM = R"=====([ | ||||
| "Default","Random Cycle","Primary Color","Based on Primary","Set Colors","Based on Set","Party","Cloud","Lava","Ocean", | ||||
| "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" | ||||
| "Aurora","Atlantica","C9 2","C9 New","Temperature","Aurora 2" | ||||
| ])====="; | ||||
|  | ||||
| #endif | ||||
|   | ||||
| @@ -27,30 +27,59 @@ | ||||
| #include "FX.h" | ||||
| #include "palettes.h" | ||||
|  | ||||
| #define LED_SKIP_AMOUNT  1 | ||||
| #define MIN_SHOW_DELAY  15 | ||||
| /* | ||||
|   Custom per-LED mapping has moved! | ||||
|  | ||||
| void WS2812FX::init(bool supportWhite, uint16_t countPixels, bool skipFirst) | ||||
|   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] | ||||
| */ | ||||
|  | ||||
| //do not call this method from system context (network callback) | ||||
| void WS2812FX::finalizeInit(uint16_t countPixels, bool skipFirst) | ||||
| { | ||||
|   if (supportWhite == _rgbwMode && countPixels == _length) return; | ||||
|   RESET_RUNTIME; | ||||
|   _rgbwMode = supportWhite; | ||||
|   _skipFirstMode = skipFirst; | ||||
|   _length = countPixels; | ||||
|   _skipFirstMode = skipFirst; | ||||
|  | ||||
|   uint8_t ty = 1; | ||||
|   if (supportWhite) ty = 2; | ||||
|   _lengthRaw = _length; | ||||
|   if (_skipFirstMode) { | ||||
|     _lengthRaw += LED_SKIP_AMOUNT; | ||||
|   } | ||||
|  | ||||
|   bus->Begin((NeoPixelType)ty, _lengthRaw); | ||||
|   //if busses failed to load, add default (FS issue...) | ||||
|   if (busses.getNumBusses() == 0) { | ||||
|     uint8_t defPin[] = {LEDPIN}; | ||||
|     BusConfig defCfg = BusConfig(TYPE_WS2812_RGB, defPin, 0, _lengthRaw, COL_ORDER_GRB); | ||||
|     busses.add(defCfg); | ||||
|   } | ||||
|    | ||||
|   deserializeMap(); | ||||
|  | ||||
|   //make segment 0 cover the entire strip | ||||
|   _segments[0].start = 0; | ||||
|   _segments[0].stop = _length; | ||||
|  | ||||
|   setBrightness(_brightness); | ||||
|  | ||||
|   #ifdef ESP8266 | ||||
|   for (uint8_t i = 0; i < busses.getNumBusses(); i++) { | ||||
|     Bus* b = busses.getBus(i); | ||||
|     if ((!IS_DIGITAL(b->getType()) || IS_2PIN(b->getType()))) continue; | ||||
|     uint8_t pins[5]; | ||||
|     b->getPins(pins); | ||||
|     BusDigital* bd = static_cast<BusDigital*>(b); | ||||
|     if (pins[0] == 3) bd->reinit(); | ||||
|   } | ||||
|   #endif | ||||
| } | ||||
|  | ||||
| void WS2812FX::service() { | ||||
| @@ -58,20 +87,40 @@ void WS2812FX::service() { | ||||
|   now = nowUp + timebase; | ||||
|   if (nowUp - _lastShow < MIN_SHOW_DELAY) return; | ||||
|   bool doShow = false; | ||||
|  | ||||
|   for(uint8_t i=0; i < MAX_NUM_SEGMENTS; i++) | ||||
|   { | ||||
|     _segment_index = i; | ||||
|     if (SEGMENT.isActive()) | ||||
|  | ||||
|     // reset the segment runtime data if needed, called before isActive to ensure deleted | ||||
|     // segment's buffers are cleared | ||||
|     SEGENV.resetIfRequired(); | ||||
|  | ||||
|     if (!SEGMENT.isActive()) continue; | ||||
|  | ||||
|     if(nowUp > SEGENV.next_time || _triggered || (doShow && SEGMENT.mode == 0)) //last is temporary | ||||
|     { | ||||
|       if(nowUp > SEGENV.next_time || _triggered || (doShow && SEGMENT.mode == 0)) //last is temporary | ||||
|       { | ||||
|       if (SEGMENT.grouping == 0) SEGMENT.grouping = 1; //sanity check | ||||
|       doShow = true; | ||||
|       uint16_t delay = FRAMETIME; | ||||
|  | ||||
|       if (!SEGMENT.getOption(SEG_OPTION_FREEZE)) { //only run effect function if not frozen | ||||
|         _virtualSegmentLength = SEGMENT.virtualLength(); | ||||
|         doShow = true; | ||||
|         _bri_t = SEGMENT.opacity; _colors_t[0] = SEGMENT.colors[0]; _colors_t[1] = SEGMENT.colors[1]; _colors_t[2] = SEGMENT.colors[2]; | ||||
|         if (!IS_SEGMENT_ON) _bri_t = 0; | ||||
|         for (uint8_t t = 0; t < MAX_NUM_TRANSITIONS; t++) { | ||||
|           if ((transitions[t].segment & 0x3F) != i) continue; | ||||
|           uint8_t slot = transitions[t].segment >> 6; | ||||
|           if (slot == 0) _bri_t = transitions[t].currentBri(); | ||||
|           _colors_t[slot] = transitions[t].currentColor(SEGMENT.colors[slot]); | ||||
|         } | ||||
|         for (uint8_t c = 0; c < 3; c++) _colors_t[c] = gamma32(_colors_t[c]); | ||||
|         handle_palette(); | ||||
|         uint16_t delay = (this->*_mode[SEGMENT.mode])(); | ||||
|         SEGENV.next_time = nowUp + delay; | ||||
|         delay = (this->*_mode[SEGMENT.mode])(); //effect function | ||||
|         if (SEGMENT.mode != FX_MODE_HALLOWEEN_EYES) SEGENV.call++; | ||||
|       } | ||||
|  | ||||
|       SEGENV.next_time = nowUp + delay; | ||||
|     } | ||||
|   } | ||||
|   _virtualSegmentLength = 0; | ||||
| @@ -90,112 +139,79 @@ void WS2812FX::setPixelColor(uint16_t n, uint32_t c) { | ||||
|   setPixelColor(n, r, g, b, w); | ||||
| } | ||||
|  | ||||
| //used to map from segment index to physical pixel, taking into account grouping, offsets, reverse and mirroring | ||||
| uint16_t WS2812FX::realPixelIndex(uint16_t i) { | ||||
|   int16_t iGroup = i * SEGMENT.groupLength(); | ||||
|  | ||||
|   /* reverse just an individual segment */ | ||||
|   int16_t realIndex = iGroup; | ||||
|   if (IS_REVERSE) realIndex = SEGMENT.length() -iGroup -1; | ||||
|   if (IS_REVERSE) { | ||||
|     if (IS_MIRROR) { | ||||
|       realIndex = (SEGMENT.length() -1) / 2 - iGroup;  //only need to index half the pixels | ||||
|     } else { | ||||
|       realIndex = SEGMENT.length() - iGroup - 1; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   realIndex += SEGMENT.start; | ||||
|   /* Reverse the whole string */ | ||||
|   if (reverseMode) realIndex = _length - 1 - realIndex; | ||||
|  | ||||
|   return realIndex; | ||||
| } | ||||
|  | ||||
| void WS2812FX::setPixelColor(uint16_t i, byte r, byte g, byte b, byte w) | ||||
| { | ||||
|   RgbwColor col; | ||||
|   switch (colorOrder) | ||||
|   { | ||||
|     case  0: col.G = g; col.R = r; col.B = b; break; //0 = GRB, default | ||||
|     case  1: col.G = r; col.R = g; col.B = b; break; //1 = RGB, common for WS2811 | ||||
|     case  2: col.G = b; col.R = r; col.B = g; break; //2 = BRG | ||||
|     case  3: col.G = r; col.R = b; col.B = g; break; //3 = RBG | ||||
|     case  4: col.G = b; col.R = g; col.B = r; break; //4 = BGR | ||||
|     default: col.G = g; col.R = b; col.B = r; break; //5 = GBR | ||||
|   //auto calculate white channel value if enabled | ||||
|   if (isRgbw) { | ||||
|     if (rgbwMode == RGBW_MODE_AUTO_BRIGHTER || (w == 0 && (rgbwMode == RGBW_MODE_DUAL || rgbwMode == RGBW_MODE_LEGACY))) | ||||
|     { | ||||
|       //white value is set to lowest RGB channel | ||||
|       //thank you to @Def3nder! | ||||
|       w = r < g ? (r < b ? r : b) : (g < b ? g : b); | ||||
|     } else if (rgbwMode == RGBW_MODE_AUTO_ACCURATE && w == 0) | ||||
|     { | ||||
|       w = r < g ? (r < b ? r : b) : (g < b ? g : b); | ||||
|       r -= w; g -= w; b -= w; | ||||
|     } | ||||
|   } | ||||
|   col.W = w; | ||||
|    | ||||
|   if (!_cronixieMode) | ||||
|   { | ||||
|     uint16_t skip = _skipFirstMode ? LED_SKIP_AMOUNT : 0; | ||||
|     if (SEGLEN) {//from segment | ||||
|       /* Set all the pixels in the group, ensuring _skipFirstMode is honored */ | ||||
|       bool reversed = reverseMode ^ IS_REVERSE; | ||||
|       uint16_t realIndex = realPixelIndex(i); | ||||
|   uint16_t skip = _skipFirstMode ? LED_SKIP_AMOUNT : 0; | ||||
|   if (SEGLEN) {//from segment | ||||
|  | ||||
|       for (uint16_t j = 0; j < SEGMENT.grouping; j++) { | ||||
|         int16_t indexSet = realIndex + (reversed ? -j : j); | ||||
|         int16_t indexSetRev = indexSet; | ||||
|         if (reverseMode) indexSetRev = _length - 1 - indexSet; | ||||
|         if (indexSetRev >= SEGMENT.start && indexSetRev < SEGMENT.stop) bus->SetPixelColor(indexSet + skip, col); | ||||
|       } | ||||
|     } else { //live data, etc. | ||||
|       if (reverseMode) i = _length - 1 - i; | ||||
|       bus->SetPixelColor(i + skip, col); | ||||
|     //color_blend(getpixel, col, _bri_t); (pseudocode for future blending of segments) | ||||
|     if (_bri_t < 255) {   | ||||
|       r = scale8(r, _bri_t); | ||||
|       g = scale8(g, _bri_t); | ||||
|       b = scale8(b, _bri_t); | ||||
|       w = scale8(w, _bri_t); | ||||
|     } | ||||
|     if (skip && i == 0) { | ||||
|       for (uint16_t j = 0; j < skip; j++) { | ||||
|         bus->SetPixelColor(j, RgbwColor(0, 0, 0, 0)); | ||||
|     uint32_t col = ((w << 24) | (r << 16) | (g << 8) | (b)); | ||||
|  | ||||
|     /* Set all the pixels in the group, ensuring _skipFirstMode is honored */ | ||||
|     bool reversed = IS_REVERSE; | ||||
|     uint16_t realIndex = realPixelIndex(i); | ||||
|  | ||||
|     for (uint16_t j = 0; j < SEGMENT.grouping; j++) { | ||||
|       int indexSet = realIndex + (reversed ? -j : j); | ||||
|       if (indexSet < customMappingSize) indexSet = customMappingTable[indexSet]; | ||||
|       if (indexSet >= SEGMENT.start && indexSet < SEGMENT.stop) { | ||||
|         busses.setPixelColor(indexSet + skip, col); | ||||
|         if (IS_MIRROR) { //set the corresponding mirrored pixel | ||||
|           uint16_t indexMir = SEGMENT.stop - indexSet + SEGMENT.start - 1; | ||||
|           if (indexMir < customMappingSize) indexMir = customMappingTable[indexMir]; | ||||
|           busses.setPixelColor(indexMir + skip, col); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     return; | ||||
|   } else { //live data, etc. | ||||
|     if (i < customMappingSize) i = customMappingTable[i]; | ||||
|      | ||||
|     uint32_t col = ((w << 24) | (r << 16) | (g << 8) | (b)); | ||||
|     busses.setPixelColor(i + skip, col); | ||||
|   } | ||||
|  | ||||
|   //CRONIXIE | ||||
|   if(i>6)return; | ||||
|   byte o = 10*i; | ||||
|   if (_cronixieBacklightEnabled && _cronixieDigits[i] <11) | ||||
|   { | ||||
|     byte r2 = _segments[0].colors[1] >>16; | ||||
|     byte g2 = _segments[0].colors[1] >> 8; | ||||
|     byte b2 = _segments[0].colors[1]; | ||||
|     byte w2 = _segments[0].colors[1] >>24; | ||||
|     for (int j=o; j< o+19; j++) | ||||
|     { | ||||
|       bus->SetPixelColor(j, RgbwColor(r2,g2,b2,w2)); | ||||
|   if (skip && i == 0) { | ||||
|     for (uint16_t j = 0; j < skip; j++) { | ||||
|       busses.setPixelColor(j, BLACK); | ||||
|     } | ||||
|   } else | ||||
|   { | ||||
|     for (int j=o; j< o+19; j++) | ||||
|     { | ||||
|       bus->SetPixelColor(j, RgbwColor(0,0,0,0)); | ||||
|     } | ||||
|   } | ||||
|   if (_skipFirstMode) o += LED_SKIP_AMOUNT; | ||||
|   switch(_cronixieDigits[i]) | ||||
|   { | ||||
|     case 0: bus->SetPixelColor(o+5, col); break; | ||||
|     case 1: bus->SetPixelColor(o+0, col); break; | ||||
|     case 2: bus->SetPixelColor(o+6, col); break; | ||||
|     case 3: bus->SetPixelColor(o+1, col); break; | ||||
|     case 4: bus->SetPixelColor(o+7, col); break; | ||||
|     case 5: bus->SetPixelColor(o+2, col); break; | ||||
|     case 6: bus->SetPixelColor(o+8, col); break; | ||||
|     case 7: bus->SetPixelColor(o+3, col); break; | ||||
|     case 8: bus->SetPixelColor(o+9, col); break; | ||||
|     case 9: bus->SetPixelColor(o+4, col); break; | ||||
|   } | ||||
| } | ||||
|  | ||||
| void WS2812FX::driverModeCronixie(bool b) | ||||
| { | ||||
|   _cronixieMode = b; | ||||
|   _segments[0].stop = (b) ? 6 : _length; | ||||
| } | ||||
|  | ||||
| void WS2812FX::setCronixieBacklight(bool b) | ||||
| { | ||||
|   _cronixieBacklightEnabled = b; | ||||
| } | ||||
|  | ||||
| void WS2812FX::setCronixieDigits(byte d[]) | ||||
| { | ||||
|   for (int i = 0; i<6; i++) | ||||
|   { | ||||
|     _cronixieDigits[i] = d[i]; | ||||
|   } | ||||
| } | ||||
|  | ||||
| @@ -212,8 +228,11 @@ void WS2812FX::setCronixieDigits(byte d[]) | ||||
|                               //you can set it to 0 if the ESP is powered by USB and the LEDs by external | ||||
|  | ||||
| void WS2812FX::show(void) { | ||||
|   if (_callback) _callback(); | ||||
|    | ||||
|  | ||||
|   // avoid race condition, caputre _callback value | ||||
|   show_callback callback = _callback; | ||||
|   if (callback) callback(); | ||||
|  | ||||
|   //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 | ||||
| @@ -242,21 +261,22 @@ void WS2812FX::show(void) { | ||||
|  | ||||
|     for (uint16_t i = 0; i < _length; i++) //sum up the usage of each LED | ||||
|     { | ||||
|       RgbwColor c = bus->GetPixelColorRgbw(i); | ||||
|       uint32_t c = busses.getPixelColor(i); | ||||
|       byte r = c >> 16, g = c >> 8, b = c, w = c >> 24; | ||||
|  | ||||
|       if(useWackyWS2815PowerModel) | ||||
|       { | ||||
|         // ignore white component on WS2815 power calculation | ||||
|         powerSum += (max(max(c.R,c.G),c.B)) * 3; | ||||
|         powerSum += (MAX(MAX(r,g),b)) * 3; | ||||
|       } | ||||
|       else  | ||||
|       { | ||||
|         powerSum += (c.R + c.G + c.B + c.W); | ||||
|         powerSum += (r + g + b + w); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     if (_rgbwMode) //RGBW led total output with white LEDs enabled is still 50mA, so each channel uses less | ||||
|     if (isRgbw) //RGBW led total output with white LEDs enabled is still 50mA, so each channel uses less | ||||
|     { | ||||
|       powerSum *= 3; | ||||
|       powerSum = powerSum >> 2; //same as /= 4 | ||||
| @@ -271,24 +291,52 @@ void WS2812FX::show(void) { | ||||
|       uint16_t scaleI = scale * 255; | ||||
|       uint8_t scaleB = (scaleI > 255) ? 255 : scaleI; | ||||
|       uint8_t newBri = scale8(_brightness, scaleB); | ||||
|       bus->SetBrightness(newBri); | ||||
|       busses.setBrightness(newBri); | ||||
|       currentMilliamps = (powerSum0 * newBri) / puPerMilliamp; | ||||
|     } else | ||||
|     { | ||||
|       currentMilliamps = powerSum / puPerMilliamp; | ||||
|       bus->SetBrightness(_brightness); | ||||
|       busses.setBrightness(_brightness); | ||||
|     } | ||||
|     currentMilliamps += MA_FOR_ESP; //add power of ESP back to estimate | ||||
|     currentMilliamps += _length; //add standby power back to estimate | ||||
|   } else { | ||||
|     currentMilliamps = 0; | ||||
|     bus->SetBrightness(_brightness); | ||||
|     busses.setBrightness(_brightness); | ||||
|   } | ||||
|    | ||||
|   bus->Show(); | ||||
|   _lastShow = millis(); | ||||
|   // 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; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Forces the next frame to be computed on all active segments. | ||||
|  */ | ||||
| void WS2812FX::trigger() { | ||||
|   _triggered = true; | ||||
| } | ||||
| @@ -312,17 +360,18 @@ uint8_t WS2812FX::getModeCount() | ||||
|  | ||||
| uint8_t WS2812FX::getPaletteCount() | ||||
| { | ||||
|   return 13 + gGradientPaletteCount; | ||||
|   return 13 + GRADIENT_PALETTE_COUNT; | ||||
| } | ||||
|  | ||||
| //TODO transitions | ||||
| //TODO effect transitions | ||||
|  | ||||
|  | ||||
| bool WS2812FX::setEffectConfig(uint8_t m, uint8_t s, uint8_t in, uint8_t p) { | ||||
|   uint8_t mainSeg = getMainSegmentId(); | ||||
|   Segment& seg = _segments[getMainSegmentId()]; | ||||
|   uint8_t modePrev = seg.mode, speedPrev = seg.speed, intensityPrev = seg.intensity, palettePrev = seg.palette; | ||||
|  | ||||
|   bool applied = false; | ||||
|    | ||||
|   if (applyToAllSelected) { | ||||
|     for (uint8_t i = 0; i < MAX_NUM_SEGMENTS; i++) | ||||
|     { | ||||
| @@ -332,9 +381,12 @@ bool WS2812FX::setEffectConfig(uint8_t m, uint8_t s, uint8_t in, uint8_t p) { | ||||
|         _segments[i].intensity = in; | ||||
|         _segments[i].palette = p; | ||||
|         setMode(i, m); | ||||
|         applied = true; | ||||
|       } | ||||
|     } | ||||
|   } else { | ||||
|   }  | ||||
|    | ||||
|   if (!applyToAllSelected || !applied) { | ||||
|     seg.speed = s; | ||||
|     seg.intensity = in; | ||||
|     seg.palette = p; | ||||
| @@ -351,21 +403,37 @@ void WS2812FX::setColor(uint8_t slot, uint8_t r, uint8_t g, uint8_t b, uint8_t w | ||||
|  | ||||
| void WS2812FX::setColor(uint8_t slot, uint32_t c) { | ||||
|   if (slot >= NUM_COLORS) return; | ||||
|  | ||||
|   bool applied = false; | ||||
|    | ||||
|   if (applyToAllSelected) { | ||||
|     for (uint8_t i = 0; i < MAX_NUM_SEGMENTS; i++) | ||||
|     { | ||||
|       if (_segments[i].isSelected()) _segments[i].colors[slot] = c; | ||||
|       if (_segments[i].isSelected()) { | ||||
|         _segments[i].setColor(slot, c, i); | ||||
|         applied = true; | ||||
|       } | ||||
|     } | ||||
|   } else { | ||||
|     _segments[getMainSegmentId()].colors[slot] = c; | ||||
|   } | ||||
|  | ||||
|   if (!applyToAllSelected || !applied) { | ||||
|     uint8_t mainseg = getMainSegmentId(); | ||||
|     _segments[mainseg].setColor(slot, c, mainseg); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void WS2812FX::setBrightness(uint8_t b) { | ||||
|   if (gammaCorrectBri) b = gamma8(b); | ||||
|   if (_brightness == b) return; | ||||
|   _brightness = (gammaCorrectBri) ? gamma8(b) : b; | ||||
|   _brightness = b; | ||||
|   _segment_index = 0; | ||||
|   if (SEGENV.next_time > millis() + 22) show();//apply brightness change immediately if no refresh soon | ||||
|   if (_brightness == 0) { //unfreeze all segments on power off | ||||
|     for (uint8_t i = 0; i < MAX_NUM_SEGMENTS; i++) | ||||
|     { | ||||
|       _segments[i].setOption(SEG_OPTION_FREEZE, false); | ||||
|     } | ||||
|   } | ||||
|   if (SEGENV.next_time > millis() + 22 && millis() - _lastShow > MIN_SHOW_DELAY) show();//apply brightness change immediately if no refresh soon | ||||
| } | ||||
|  | ||||
| uint8_t WS2812FX::getMode(void) { | ||||
| @@ -399,7 +467,12 @@ uint8_t WS2812FX::getMaxSegments(void) { | ||||
|  | ||||
| uint8_t WS2812FX::getMainSegmentId(void) { | ||||
|   if (mainSegment >= MAX_NUM_SEGMENTS) return 0; | ||||
|   return mainSegment; | ||||
|   if (_segments[mainSegment].isActive()) return mainSegment; | ||||
|   for (uint8_t i = 0; i < MAX_NUM_SEGMENTS; i++) //get first active | ||||
|   { | ||||
|     if (_segments[i].isActive()) return i; | ||||
|   } | ||||
|   return 0; | ||||
| } | ||||
|  | ||||
| uint32_t WS2812FX::getColor(void) { | ||||
| @@ -408,41 +481,15 @@ uint32_t WS2812FX::getColor(void) { | ||||
|  | ||||
| uint32_t WS2812FX::getPixelColor(uint16_t i) | ||||
| { | ||||
|   i = realPixelIndex(i) + (_skipFirstMode ? LED_SKIP_AMOUNT : 0); | ||||
|   i = realPixelIndex(i); | ||||
|    | ||||
|   if (i < customMappingSize) i = customMappingTable[i]; | ||||
|  | ||||
|   if (_skipFirstMode) i += LED_SKIP_AMOUNT; | ||||
|    | ||||
|   if (_cronixieMode) | ||||
|   { | ||||
|     if(i>6)return 0; | ||||
|     byte o = 10*i; | ||||
|     switch(_cronixieDigits[i]) | ||||
|     { | ||||
|       case 0: i=o+5; break; | ||||
|       case 1: i=o+0; break; | ||||
|       case 2: i=o+6; break; | ||||
|       case 3: i=o+1; break; | ||||
|       case 4: i=o+7; break; | ||||
|       case 5: i=o+2; break; | ||||
|       case 6: i=o+8; break; | ||||
|       case 7: i=o+3; break; | ||||
|       case 8: i=o+9; break; | ||||
|       case 9: i=o+4; break; | ||||
|       default: return 0; | ||||
|     } | ||||
|   } | ||||
|   if (i >= _lengthRaw) return 0; | ||||
|    | ||||
|   RgbwColor col = bus->GetPixelColorRgbw(i); | ||||
|   switch (colorOrder) | ||||
|   { | ||||
|     //                    W               G              R               B | ||||
|     case  0: return ((col.W << 24) | (col.G << 8) | (col.R << 16) | (col.B)); //0 = GRB, default | ||||
|     case  1: return ((col.W << 24) | (col.R << 8) | (col.G << 16) | (col.B)); //1 = RGB, common for WS2811 | ||||
|     case  2: return ((col.W << 24) | (col.B << 8) | (col.R << 16) | (col.G)); //2 = BRG | ||||
|     case  3: return ((col.W << 24) | (col.R << 8) | (col.B << 16) | (col.G)); //3 = RBG | ||||
|     case  4: return ((col.W << 24) | (col.B << 8) | (col.G << 16) | (col.R)); //4 = BGR | ||||
|     case  5: return ((col.W << 24) | (col.G << 8) | (col.B << 16) | (col.R)); //5 = GBR | ||||
|   } | ||||
|   return 0; | ||||
|   return busses.getPixelColor(i); | ||||
| } | ||||
|  | ||||
| WS2812FX::Segment& WS2812FX::getSegment(uint8_t id) { | ||||
| @@ -462,6 +509,15 @@ uint32_t WS2812FX::getLastShow(void) { | ||||
|   return _lastShow; | ||||
| } | ||||
|  | ||||
| //TODO these need to be on a per-strip basis | ||||
| uint8_t WS2812FX::getColorOrder(void) { | ||||
|   return COL_ORDER_GRB; | ||||
| } | ||||
|  | ||||
| void WS2812FX::setColorOrder(uint8_t co) { | ||||
|   //bus->SetColorOrder(co); | ||||
| } | ||||
|  | ||||
| void WS2812FX::setSegment(uint8_t n, uint16_t i1, uint16_t i2, uint8_t grouping, uint8_t spacing) { | ||||
|   if (n >= MAX_NUM_SEGMENTS) return; | ||||
|   Segment& seg = _segments[n]; | ||||
| @@ -505,18 +561,38 @@ void WS2812FX::resetSegments() { | ||||
|   _segments[0].colors[0] = DEFAULT_COLOR; | ||||
|   _segments[0].start = 0; | ||||
|   _segments[0].speed = DEFAULT_SPEED; | ||||
|   _segments[0].intensity = DEFAULT_INTENSITY; | ||||
|   _segments[0].stop = _length; | ||||
|   _segments[0].grouping = 1; | ||||
|   _segments[0].setOption(0, 1); //select | ||||
|   _segments[0].setOption(SEG_OPTION_SELECTED, 1); | ||||
|   _segments[0].setOption(SEG_OPTION_ON, 1); | ||||
|   _segments[0].opacity = 255; | ||||
|  | ||||
|   for (uint16_t i = 1; i < MAX_NUM_SEGMENTS; i++) | ||||
|   { | ||||
|     _segments[i].colors[0] = color_wheel(i*51); | ||||
|     _segments[i].grouping = 1; | ||||
|     _segments[i].setOption(SEG_OPTION_ON, 1); | ||||
|     _segments[i].opacity = 255; | ||||
|     _segments[i].speed = DEFAULT_SPEED; | ||||
|     _segments[i].intensity = DEFAULT_INTENSITY; | ||||
|     _segment_runtimes[i].reset(); | ||||
|   } | ||||
|   _segment_runtimes[0].reset(); | ||||
| } | ||||
|  | ||||
| //After this function is called, setPixelColor() will use that segment (offsets, grouping, ... will apply) | ||||
| void WS2812FX::setPixelSegment(uint8_t n) | ||||
| { | ||||
|   if (n < MAX_NUM_SEGMENTS) { | ||||
|     _segment_index = n; | ||||
|     _virtualSegmentLength = SEGMENT.length(); | ||||
|   } else { | ||||
|     _segment_index = 0; | ||||
|     _virtualSegmentLength = 0; | ||||
|   } | ||||
| } | ||||
|  | ||||
| void WS2812FX::setRange(uint16_t i, uint16_t i2, uint32_t col) | ||||
| { | ||||
|   if (i2 >= i) | ||||
| @@ -533,36 +609,46 @@ void WS2812FX::setShowCallback(show_callback cb) | ||||
|   _callback = cb; | ||||
| } | ||||
|  | ||||
| void WS2812FX::setTransition(uint16_t t) | ||||
| { | ||||
|   _transitionDur = t; | ||||
| } | ||||
|  | ||||
| void WS2812FX::setTransitionMode(bool t) | ||||
| { | ||||
|   _segment_index = getMainSegmentId(); | ||||
|   SEGMENT.setOption(7,t); | ||||
|   if (!t) return; | ||||
|   unsigned long waitMax = millis() + 20; //refresh after 20 ms if transition enabled | ||||
|   if (SEGMENT.mode == FX_MODE_STATIC && SEGENV.next_time > waitMax) SEGENV.next_time = waitMax; | ||||
|   for (uint16_t i = 0; i < MAX_NUM_SEGMENTS; i++) | ||||
|   { | ||||
|     _segment_index = i; | ||||
|     SEGMENT.setOption(SEG_OPTION_TRANSITIONAL, t); | ||||
|  | ||||
|     if (t && SEGMENT.mode == FX_MODE_STATIC && SEGENV.next_time > waitMax) SEGENV.next_time = waitMax; | ||||
|   } | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * color blend function | ||||
|  */ | ||||
| uint32_t WS2812FX::color_blend(uint32_t color1, uint32_t color2, uint8_t blend) { | ||||
| uint32_t WS2812FX::color_blend(uint32_t color1, uint32_t color2, uint16_t blend, bool b16) { | ||||
|   if(blend == 0)   return color1; | ||||
|   if(blend == 255) return color2; | ||||
|   uint16_t blendmax = b16 ? 0xFFFF : 0xFF; | ||||
|   if(blend == blendmax) return color2; | ||||
|   uint8_t shift = b16 ? 16 : 8; | ||||
|  | ||||
|   uint32_t w1 = (color1 >> 24) & 0xff; | ||||
|   uint32_t r1 = (color1 >> 16) & 0xff; | ||||
|   uint32_t g1 = (color1 >>  8) & 0xff; | ||||
|   uint32_t b1 =  color1        & 0xff; | ||||
|   uint32_t w1 = (color1 >> 24) & 0xFF; | ||||
|   uint32_t r1 = (color1 >> 16) & 0xFF; | ||||
|   uint32_t g1 = (color1 >>  8) & 0xFF; | ||||
|   uint32_t b1 =  color1        & 0xFF; | ||||
|  | ||||
|   uint32_t w2 = (color2 >> 24) & 0xff; | ||||
|   uint32_t r2 = (color2 >> 16) & 0xff; | ||||
|   uint32_t g2 = (color2 >>  8) & 0xff; | ||||
|   uint32_t b2 =  color2        & 0xff; | ||||
|   uint32_t w2 = (color2 >> 24) & 0xFF; | ||||
|   uint32_t r2 = (color2 >> 16) & 0xFF; | ||||
|   uint32_t g2 = (color2 >>  8) & 0xFF; | ||||
|   uint32_t b2 =  color2        & 0xFF; | ||||
|  | ||||
|   uint32_t w3 = ((w2 * blend) + (w1 * (255 - blend))) >> 8; | ||||
|   uint32_t r3 = ((r2 * blend) + (r1 * (255 - blend))) >> 8; | ||||
|   uint32_t g3 = ((g2 * blend) + (g1 * (255 - blend))) >> 8; | ||||
|   uint32_t b3 = ((b2 * blend) + (b1 * (255 - blend))) >> 8; | ||||
|   uint32_t w3 = ((w2 * blend) + (w1 * (blendmax - blend))) >> shift; | ||||
|   uint32_t r3 = ((r2 * blend) + (r1 * (blendmax - blend))) >> shift; | ||||
|   uint32_t g3 = ((g2 * blend) + (g1 * (blendmax - blend))) >> shift; | ||||
|   uint32_t b3 = ((b2 * blend) + (b1 * (blendmax - blend))) >> shift; | ||||
|  | ||||
|   return ((w3 << 24) | (r3 << 16) | (g3 << 8) | (b3)); | ||||
| } | ||||
| @@ -576,6 +662,14 @@ void WS2812FX::fill(uint32_t c) { | ||||
|   } | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Blends the specified color with the existing pixel color. | ||||
|  */ | ||||
| void WS2812FX::blendPixelColor(uint16_t n, uint32_t color, uint8_t blend) | ||||
| { | ||||
|   setPixelColor(n, color_blend(getPixelColor(n), color, blend)); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * fade out function, higher rate = quicker fade | ||||
|  */ | ||||
| @@ -644,6 +738,32 @@ uint16_t WS2812FX::triwave16(uint16_t in) | ||||
|   return 0xFFFF - (in - 0x8000)*2; | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Generates a tristate square wave w/ attac & decay  | ||||
|  * @param x input value 0-255 | ||||
|  * @param pulsewidth 0-127  | ||||
|  * @param attdec attac & decay, max. pulsewidth / 2 | ||||
|  * @returns signed waveform value | ||||
|  */ | ||||
| int8_t WS2812FX::tristate_square8(uint8_t x, uint8_t pulsewidth, uint8_t attdec) { | ||||
|   int8_t a = 127; | ||||
|   if (x > 127) { | ||||
|     a = -127; | ||||
|     x -= 127; | ||||
|   } | ||||
|  | ||||
|   if (x < attdec) { //inc to max | ||||
|     return (int16_t) x * a / attdec; | ||||
|   } | ||||
|   else if (x < pulsewidth - attdec) { //max | ||||
|     return a; | ||||
|   }   | ||||
|   else if (x < pulsewidth) { //dec to 0 | ||||
|     return (int16_t) (pulsewidth - x) * a / attdec; | ||||
|   } | ||||
|   return 0; | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Put a value 0 to 255 in to get a color value. | ||||
|  * The colours are a transition r -> g -> b -> back to r | ||||
| @@ -673,7 +793,7 @@ uint8_t WS2812FX::get_random_wheel_index(uint8_t pos) { | ||||
|     r = random8(); | ||||
|     x = abs(pos - r); | ||||
|     y = 255 - x; | ||||
|     d = min(x, y); | ||||
|     d = MIN(x, y); | ||||
|   } | ||||
|   return r; | ||||
| } | ||||
| @@ -695,6 +815,15 @@ CRGB WS2812FX::col_to_crgb(uint32_t color) | ||||
| } | ||||
|  | ||||
|  | ||||
| void WS2812FX::load_gradient_palette(uint8_t index) | ||||
| { | ||||
|   byte i = constrain(index, 0, GRADIENT_PALETTE_COUNT -1); | ||||
|   byte tcp[72]; //support gradient palettes with up to 18 entries | ||||
|   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 | ||||
|  */ | ||||
| @@ -704,26 +833,28 @@ void WS2812FX::handle_palette(void) | ||||
|   _segment_index_palette_last = _segment_index; | ||||
|  | ||||
|   byte paletteIndex = SEGMENT.palette; | ||||
|   if (SEGMENT.mode == FX_MODE_GLITTER && paletteIndex == 0) paletteIndex = 11; | ||||
|   if (paletteIndex == 0) //default palette. Differs depending on effect | ||||
|   { | ||||
|     switch (SEGMENT.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 (SEGMENT.mode >= FX_MODE_METEOR && paletteIndex == 0) paletteIndex = 4; | ||||
|    | ||||
|   switch (paletteIndex) | ||||
|   { | ||||
|     case 0: {//default palette. Differs depending on effect | ||||
|       switch (SEGMENT.mode) | ||||
|       { | ||||
|         case FX_MODE_FIRE_2012  : targetPalette = gGradientPalettes[22]; break;//heat palette | ||||
|         case FX_MODE_COLORWAVES : targetPalette = gGradientPalettes[13]; break;//landscape 33 | ||||
|         case FX_MODE_FILLNOISE8 : targetPalette = OceanColors_p;         break; | ||||
|         case FX_MODE_NOISE16_1  : targetPalette = gGradientPalettes[17]; break;//Drywet | ||||
|         case FX_MODE_NOISE16_2  : targetPalette = gGradientPalettes[30]; break;//Blue cyan yellow | ||||
|         case FX_MODE_NOISE16_3  : targetPalette = gGradientPalettes[22]; break;//heat palette | ||||
|         case FX_MODE_NOISE16_4  : targetPalette = gGradientPalettes[13]; break;//landscape 33 | ||||
|         //case FX_MODE_GLITTER    : targetPalette = RainbowColors_p;       break; | ||||
|          | ||||
|         default: targetPalette = PartyColors_p; break;//palette, bpm | ||||
|       } | ||||
|       break;} | ||||
|     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) | ||||
|       { | ||||
| @@ -741,25 +872,25 @@ void WS2812FX::handle_palette(void) | ||||
|     case 2: {//primary color only | ||||
|       CRGB prim = col_to_crgb(SEGCOLOR(0)); | ||||
|       targetPalette = CRGBPalette16(prim); break;} | ||||
|     case 3: {//based on primary | ||||
|       //considering performance implications | ||||
|       CRGB prim = col_to_crgb(SEGCOLOR(0)); | ||||
|       CHSV prim_hsv = rgb2hsv_approximate(prim); | ||||
|       targetPalette = CRGBPalette16( | ||||
|                       CHSV(prim_hsv.h, prim_hsv.s, prim_hsv.v), //color itself | ||||
|                       CHSV(prim_hsv.h, max(prim_hsv.s - 50,0), prim_hsv.v), //less saturated | ||||
|                       CHSV(prim_hsv.h, prim_hsv.s, max(prim_hsv.v - 50,0)), //darker | ||||
|                       CHSV(prim_hsv.h, prim_hsv.s, prim_hsv.v)); //color itself | ||||
|       break;} | ||||
|     case 4: {//primary + secondary | ||||
|     case 3: {//primary + secondary | ||||
|       CRGB prim = col_to_crgb(SEGCOLOR(0)); | ||||
|       CRGB sec  = col_to_crgb(SEGCOLOR(1)); | ||||
|       targetPalette = CRGBPalette16(sec,prim); break;} | ||||
|     case 5: {//based on primary + secondary | ||||
|       targetPalette = CRGBPalette16(prim,prim,sec,sec); break;} | ||||
|     case 4: {//primary + secondary + tertiary | ||||
|       CRGB prim = col_to_crgb(SEGCOLOR(0)); | ||||
|       CRGB sec  = col_to_crgb(SEGCOLOR(1)); | ||||
|       CRGB ter  = col_to_crgb(SEGCOLOR(2)); | ||||
|       targetPalette = CRGBPalette16(ter,sec,prim); break;} | ||||
|     case 5: {//primary + secondary (+tert if not off), more distinct | ||||
|       CRGB prim = col_to_crgb(SEGCOLOR(0)); | ||||
|       CRGB sec  = col_to_crgb(SEGCOLOR(1)); | ||||
|       if (SEGCOLOR(2)) { | ||||
|         CRGB ter = col_to_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 | ||||
| @@ -775,10 +906,10 @@ void WS2812FX::handle_palette(void) | ||||
|     case 12: //Rainbow stripe colors | ||||
|       targetPalette = RainbowStripeColors_p; break; | ||||
|     default: //progmem palettes | ||||
|       targetPalette = gGradientPalettes[constrain(SEGMENT.palette -13, 0, gGradientPaletteCount -1)]; | ||||
|       load_gradient_palette(paletteIndex -13); | ||||
|   } | ||||
|    | ||||
|   if (singleSegmentMode && paletteFade) //only blend if just one segment uses FastLED mode | ||||
|   if (singleSegmentMode && paletteFade && SEGENV.call > 0) //only blend if just one segment uses FastLED mode | ||||
|   { | ||||
|     nblendPaletteTowardPalette(currentPalette, targetPalette, 48); | ||||
|   } else | ||||
| @@ -787,17 +918,39 @@ void WS2812FX::handle_palette(void) | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| /* | ||||
|  * Gets a single color from the currently selected palette. | ||||
|  * @param i Palette Index (if mapping is true, the full palette will be SEGLEN 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 WS2812FX::color_from_palette(uint16_t i, bool mapping, bool wrap, uint8_t mcol, uint8_t pbri) | ||||
| { | ||||
|   if (SEGMENT.palette == 0 && mcol < 3) return SEGCOLOR(mcol); //WS2812FX default | ||||
|   if (SEGMENT.palette == 0 && mcol < 3) { | ||||
|     uint32_t color = SEGCOLOR(mcol); | ||||
|     if (pbri != 255) { | ||||
|       CRGB crgb_color = col_to_crgb(color); | ||||
|       crgb_color.nscale8_video(pbri); | ||||
|       return crgb_to_col(crgb_color); | ||||
|     } else { | ||||
|       return color; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   uint8_t paletteIndex = i; | ||||
|   if (mapping) paletteIndex = (i*255)/(SEGLEN -1); | ||||
|   if (!wrap) paletteIndex = scale8(paletteIndex, 240); //cut off blend at palette "end" | ||||
|   CRGB fastled_col; | ||||
|   fastled_col = ColorFromPalette( currentPalette, paletteIndex, pbri, (paletteBlend == 3)? NOBLEND:LINEARBLEND); | ||||
|   return  fastled_col.r*65536 +  fastled_col.g*256 +  fastled_col.b; | ||||
|  | ||||
|   return crgb_to_col(fastled_col); | ||||
| } | ||||
|  | ||||
| //@returns `true` if color, mode, speed, intensity and palette match | ||||
| bool WS2812FX::segmentsAreIdentical(Segment* a, Segment* b) | ||||
| { | ||||
|   //if (a->start != b->start) return false; | ||||
| @@ -810,12 +963,38 @@ bool WS2812FX::segmentsAreIdentical(Segment* a, Segment* b) | ||||
|   if (a->speed != b->speed) return false; | ||||
|   if (a->intensity != b->intensity) return false; | ||||
|   if (a->palette != b->palette) return false; | ||||
|   //if (a->getOption(1) != b->getOption(1)) return false; //reverse | ||||
|   //if (a->getOption(SEG_OPTION_REVERSED) != b->getOption(SEG_OPTION_REVERSED)) return false; | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| //gamma 2.4 lookup table used for color correction | ||||
| const byte gammaT[] = { | ||||
|  | ||||
| //load custom mapping table from JSON file | ||||
| void WS2812FX::deserializeMap(void) { | ||||
|   if (!WLED_FS.exists("/ledmap.json")) return; | ||||
|   DynamicJsonDocument doc(JSON_BUFFER_SIZE);  // full sized buffer for larger maps | ||||
|  | ||||
|   DEBUG_PRINTLN(F("Reading LED map from /ledmap.json...")); | ||||
|  | ||||
|   if (!readObjectFromFile("/ledmap.json", nullptr, &doc)) return; //if file does not exist just exit | ||||
|  | ||||
|   if (customMappingTable != nullptr) { | ||||
|     delete[] customMappingTable; | ||||
|     customMappingTable = nullptr; | ||||
|     customMappingSize = 0; | ||||
|   } | ||||
|  | ||||
|   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]; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| //gamma 2.8 lookup table used for color correction | ||||
| 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, | ||||
| @@ -833,6 +1012,17 @@ const byte gammaT[] = { | ||||
|   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)(pow((float)b / 255.0, gamma) * 255 + 0.5); | ||||
| } | ||||
|  | ||||
| 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]; | ||||
| @@ -852,4 +1042,4 @@ uint32_t WS2812FX::gamma32(uint32_t color) | ||||
|   return ((w << 24) | (r << 16) | (g << 8) | (b)); | ||||
| } | ||||
|  | ||||
| uint16_t WS2812FX::_usedSegmentData = 0; | ||||
| WS2812FX* WS2812FX::instance = nullptr; | ||||
							
								
								
									
										34
									
								
								wled00/NodeStruct.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								wled00/NodeStruct.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| #ifndef WLED_NODESTRUCT_H | ||||
| #define WLED_NODESTRUCT_H | ||||
|  | ||||
| /*********************************************************************************************\ | ||||
| * NodeStruct from the ESP Easy project (https://github.com/letscontrolit/ESPEasy) | ||||
| \*********************************************************************************************/ | ||||
|  | ||||
| #include <map> | ||||
| #include <IPAddress.h> | ||||
|  | ||||
| #define NODE_TYPE_ID_UNDEFINED        0 | ||||
| #define NODE_TYPE_ID_ESP8266         82 | ||||
| #define NODE_TYPE_ID_ESP32           32 | ||||
|  | ||||
| /*********************************************************************************************\ | ||||
| * NodeStruct | ||||
| \*********************************************************************************************/ | ||||
| struct NodeStruct | ||||
| { | ||||
|   String    nodeName; | ||||
|   IPAddress ip; | ||||
|   uint8_t   unit; | ||||
|   uint8_t   age; | ||||
|   uint8_t   nodeType; | ||||
|   uint32_t  build; | ||||
|  | ||||
|   NodeStruct() : age(0), nodeType(0), build(0) | ||||
|   { | ||||
|     for (uint8_t i = 0; i < 4; ++i) { ip[i] = 0; } | ||||
|   } | ||||
| }; | ||||
| typedef std::map<uint8_t, NodeStruct> NodesMap; | ||||
|  | ||||
| #endif // WLED_NODESTRUCT_H | ||||
| @@ -1,315 +0,0 @@ | ||||
| //this code is a modified version of https://github.com/Makuna/NeoPixelBus/issues/103 | ||||
| #ifndef NpbWrapper_h | ||||
| #define NpbWrapper_h | ||||
|  | ||||
| //PIN CONFIGURATION | ||||
| #define LEDPIN 2      //strip pin. Any for ESP32, gpio2 or 3 is recommended for ESP8266 (gpio2/3 are labeled D4/RX on NodeMCU and Wemos) | ||||
| //#define USE_APA102  // Uncomment for using APA102 LEDs. | ||||
| //#define USE_WS2801  // Uncomment for using WS2801 LEDs (make sure you have NeoPixelBus v2.5.6 or newer) | ||||
| //#define USE_LPD8806 // Uncomment for using LPD8806 | ||||
| //#define WLED_USE_ANALOG_LEDS //Uncomment for using "dumb" PWM controlled LEDs (see pins below, default R: gpio5, G: 12, B: 15, W: 13) | ||||
| //#define WLED_USE_H801 //H801 controller. Please uncomment #define WLED_USE_ANALOG_LEDS as well | ||||
| //#define WLED_USE_5CH_LEDS  //5 Channel H801 for cold and warm white | ||||
|  | ||||
| #define BTNPIN  0  //button pin. Needs to have pullup (gpio0 recommended) | ||||
| #define IR_PIN  4  //infrared pin (-1 to disable)  MagicHome: 4, H801 Wifi: 0 | ||||
| #define RLYPIN 12  //pin for relay, will be set HIGH if LEDs are on (-1 to disable). Also usable for standby leds, triggers,... | ||||
| #define AUXPIN -1  //debug auxiliary output pin (-1 to disable) | ||||
|  | ||||
| #define RLYMDE 1  //mode for relay, 0: LOW if LEDs are on 1: HIGH if LEDs are on | ||||
|  | ||||
| //END CONFIGURATION | ||||
|  | ||||
| #if defined(USE_APA102) || defined(USE_WS2801) || defined(USE_LPD8806) | ||||
|  #define CLKPIN 0 | ||||
|  #define DATAPIN 2 | ||||
|  #if BTNPIN == CLKPIN || BTNPIN == DATAPIN | ||||
|   #undef BTNPIN   // Deactivate button pin if it conflicts with one of the APA102 pins. | ||||
|  #endif | ||||
| #endif | ||||
|  | ||||
| #ifdef WLED_USE_ANALOG_LEDS | ||||
|   //PWM pins - PINs 15,13,12,14 (W2 = 04)are used with H801 Wifi LED Controller | ||||
|   #ifdef WLED_USE_H801 | ||||
|     #define RPIN 15   //R pin for analog LED strip    | ||||
|     #define GPIN 13   //G pin for analog LED strip | ||||
|     #define BPIN 12   //B pin for analog LED strip | ||||
|     #define WPIN 14   //W pin for analog LED strip  | ||||
|     #define W2PIN 04  //W2 pin for analog LED strip | ||||
|     #undef BTNPIN | ||||
|     #undef IR_PIN | ||||
|     #define IR_PIN  0 //infrared pin (-1 to disable)  MagicHome: 4, H801 Wifi: 0 | ||||
|   #else | ||||
|   //PWM pins - PINs 5,12,13,15 are used with Magic Home LED Controller | ||||
|     #define RPIN 5   //R pin for analog LED strip    | ||||
|     #define GPIN 12   //G pin for analog LED strip | ||||
|     #define BPIN 15   //B pin for analog LED strip | ||||
|     #define WPIN 13   //W pin for analog LED strip  | ||||
|   #endif | ||||
|   #undef RLYPIN | ||||
|   #define RLYPIN -1 //disable as pin 12 is used by analog LEDs | ||||
| #endif | ||||
|  | ||||
| //automatically uses the right driver method for each platform | ||||
| #ifdef ARDUINO_ARCH_ESP32 | ||||
|  #ifdef USE_APA102 | ||||
|   #define PIXELMETHOD DotStarMethod | ||||
|  #elif defined(USE_WS2801) | ||||
|   #define PIXELMETHOD NeoWs2801Method | ||||
|  #elif defined(USE_LPD8806) | ||||
|   #define PIXELMETHOD Lpd8806Method | ||||
|  #else | ||||
|   #define PIXELMETHOD NeoEsp32Rmt0Ws2812xMethod | ||||
|  #endif | ||||
| #else //esp8266 | ||||
|  //autoselect the right method depending on strip pin | ||||
|  #ifdef USE_APA102 | ||||
|   #define PIXELMETHOD DotStarMethod | ||||
|  #elif defined(USE_WS2801) | ||||
|   #define PIXELMETHOD NeoWs2801Method | ||||
|  #elif defined(USE_LPD8806) | ||||
|   #define PIXELMETHOD Lpd8806Method | ||||
|  #elif LEDPIN == 2 | ||||
|   #define PIXELMETHOD NeoEsp8266Uart1Ws2813Method //if you get an error here, try to change to NeoEsp8266UartWs2813Method or update Neopixelbus | ||||
|  #elif LEDPIN == 3 | ||||
|   #define PIXELMETHOD NeoEsp8266Dma800KbpsMethod | ||||
|  #else | ||||
|   #define PIXELMETHOD NeoEsp8266BitBang800KbpsMethod | ||||
|   #pragma message "Software BitBang will be used because of your selected LED pin. This may cause flicker. Use GPIO 2 or 3 for best results." | ||||
|  #endif | ||||
| #endif | ||||
|  | ||||
|  | ||||
| //you can now change the color order in the web settings | ||||
| #ifdef USE_APA102 | ||||
|  #define PIXELFEATURE3 DotStarBgrFeature | ||||
|  #define PIXELFEATURE4 DotStarLbgrFeature | ||||
| #elif defined(USE_LPD8806) | ||||
|  #define PIXELFEATURE3 Lpd8806GrbFeature  | ||||
|  #define PIXELFEATURE4 Lpd8806GrbFeature  | ||||
| #else | ||||
|  #define PIXELFEATURE3 NeoGrbFeature | ||||
|  #define PIXELFEATURE4 NeoGrbwFeature | ||||
| #endif | ||||
|  | ||||
|  | ||||
| #include <NeoPixelBrightnessBus.h> | ||||
|  | ||||
| enum NeoPixelType | ||||
| { | ||||
|   NeoPixelType_None = 0, | ||||
|   NeoPixelType_Grb  = 1, | ||||
|   NeoPixelType_Grbw = 2, | ||||
|   NeoPixelType_End  = 3 | ||||
| }; | ||||
|  | ||||
| class NeoPixelWrapper | ||||
| { | ||||
| public: | ||||
|   NeoPixelWrapper() : | ||||
|     // initialize each member to null | ||||
|     _pGrb(NULL), | ||||
|     _pGrbw(NULL), | ||||
|     _type(NeoPixelType_None) | ||||
|   { | ||||
|  | ||||
|   } | ||||
|  | ||||
|   ~NeoPixelWrapper() | ||||
|   { | ||||
|     cleanup(); | ||||
|   } | ||||
|  | ||||
|   void Begin(NeoPixelType type, uint16_t countPixels) | ||||
|   { | ||||
|     cleanup(); | ||||
|     _type = type; | ||||
|  | ||||
|     switch (_type) | ||||
|     { | ||||
|       case NeoPixelType_Grb: | ||||
|       #if defined(USE_APA102) || defined(USE_WS2801) || defined(USE_LPD8806) | ||||
|         _pGrb = new NeoPixelBrightnessBus<PIXELFEATURE3,PIXELMETHOD>(countPixels, CLKPIN, DATAPIN); | ||||
|       #else | ||||
|         _pGrb = new NeoPixelBrightnessBus<PIXELFEATURE3,PIXELMETHOD>(countPixels, LEDPIN); | ||||
|       #endif | ||||
|         _pGrb->Begin(); | ||||
|       break; | ||||
|  | ||||
|       case NeoPixelType_Grbw: | ||||
|       #if defined(USE_APA102) || defined(USE_WS2801) || defined(USE_LPD8806) | ||||
|         _pGrbw = new NeoPixelBrightnessBus<PIXELFEATURE4,PIXELMETHOD>(countPixels, CLKPIN, DATAPIN); | ||||
|       #else | ||||
|         _pGrbw = new NeoPixelBrightnessBus<PIXELFEATURE4,PIXELMETHOD>(countPixels, LEDPIN); | ||||
|       #endif | ||||
|         _pGrbw->Begin(); | ||||
|       break; | ||||
|     } | ||||
|  | ||||
|     #ifdef WLED_USE_ANALOG_LEDS  | ||||
|       #ifdef ARDUINO_ARCH_ESP32 | ||||
|         ledcSetup(0, 5000, 8); | ||||
|         ledcAttachPin(RPIN, 0); | ||||
|         ledcSetup(1, 5000, 8); | ||||
|         ledcAttachPin(GPIN, 1); | ||||
|         ledcSetup(2, 5000, 8);         | ||||
|         ledcAttachPin(BPIN, 2); | ||||
|         if(_type == NeoPixelType_Grbw)  | ||||
|         { | ||||
|           ledcSetup(3, 5000, 8);         | ||||
|           ledcAttachPin(WPIN, 3); | ||||
|           #ifdef WLED_USE_5CH_LEDS | ||||
|             ledcSetup(4, 5000, 8);         | ||||
|             ledcAttachPin(W2PIN, 4); | ||||
|           #endif | ||||
|         } | ||||
|       #else  // ESP8266 | ||||
|         //init PWM pins - PINs 5,12,13,15 are used with Magic Home LED Controller | ||||
|         pinMode(RPIN, OUTPUT); | ||||
|         pinMode(GPIN, OUTPUT); | ||||
|         pinMode(BPIN, OUTPUT);  | ||||
|         if(_type == NeoPixelType_Grbw)  | ||||
|         { | ||||
|           pinMode(WPIN, OUTPUT);  | ||||
|           #ifdef WLED_USE_5CH_LEDS | ||||
|             pinMode(W2PIN, OUTPUT); | ||||
|           #endif | ||||
|         } | ||||
|         analogWriteRange(255);  //same range as one RGB channel | ||||
|         analogWriteFreq(880);   //PWM frequency proven as good for LEDs | ||||
|       #endif  | ||||
|     #endif | ||||
|   } | ||||
|  | ||||
| #ifdef WLED_USE_ANALOG_LEDS       | ||||
|     void SetRgbwPwm(uint8_t r, uint8_t g, uint8_t b, uint8_t w, uint8_t w2=0) | ||||
|     { | ||||
|       #ifdef ARDUINO_ARCH_ESP32 | ||||
|         ledcWrite(0, r);  //RPIN | ||||
|         ledcWrite(1, g);  //GPIN | ||||
|         ledcWrite(2, b);  //BPIN | ||||
|         switch (_type) { | ||||
|           case NeoPixelType_Grb:                                                  break; | ||||
|           #ifdef WLED_USE_5CH_LEDS | ||||
|             case NeoPixelType_Grbw: ledcWrite(3, w); ledcWrite(4, w2);            break; | ||||
|           #else | ||||
|             case NeoPixelType_Grbw: ledcWrite(3, w);                              break; | ||||
|           #endif | ||||
|         }         | ||||
|       #else  | ||||
|         analogWrite(RPIN, r); | ||||
|         analogWrite(GPIN, g); | ||||
|         analogWrite(BPIN, b); | ||||
|         switch (_type) { | ||||
|           case NeoPixelType_Grb:                                                  break; | ||||
|           #ifdef WLED_USE_5CH_LEDS | ||||
|             case NeoPixelType_Grbw: analogWrite(WPIN, w); analogWrite(W2PIN, w2); break; | ||||
|           #else | ||||
|             case NeoPixelType_Grbw: analogWrite(WPIN, w);                         break; | ||||
|           #endif | ||||
|         } | ||||
|       #endif  | ||||
|     } | ||||
| #endif | ||||
|  | ||||
|   void Show() | ||||
|   { | ||||
|     byte b; | ||||
|     switch (_type) | ||||
|     { | ||||
|       case NeoPixelType_Grb:  _pGrb->Show();  break; | ||||
|       case NeoPixelType_Grbw: _pGrbw->Show(); break; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void SetPixelColor(uint16_t indexPixel, RgbwColor color) | ||||
|   { | ||||
|     switch (_type) { | ||||
|       case NeoPixelType_Grb: { | ||||
|         _pGrb->SetPixelColor(indexPixel, RgbColor(color.R,color.G,color.B)); | ||||
|         #ifdef WLED_USE_ANALOG_LEDS | ||||
|           if (indexPixel != 0) return; //set analog LEDs from first pixel | ||||
|           byte b = _pGrb->GetBrightness(); | ||||
|           SetRgbwPwm(color.R * b / 255, color.G * b / 255, color.B * b / 255, 0); | ||||
|         #endif | ||||
|       } | ||||
|       break; | ||||
|       case NeoPixelType_Grbw: { | ||||
|         #ifdef USE_LPD8806 | ||||
|         _pGrbw->SetPixelColor(indexPixel, RgbColor(color.R,color.G,color.B)); | ||||
|         #else | ||||
|         _pGrbw->SetPixelColor(indexPixel, color); | ||||
|         #endif | ||||
|         #ifdef WLED_USE_ANALOG_LEDS       | ||||
|           if (indexPixel != 0) return; //set analog LEDs from first pixel | ||||
|           byte b = _pGrbw->GetBrightness(); | ||||
|           // check color values for Warm / Cold white mix (for RGBW)  // EsplanexaDevice.cpp | ||||
|           #ifdef WLED_USE_5CH_LEDS | ||||
|             if        (color.R == 255 & color.G == 255 && color.B == 255 && color.W == 255) {   | ||||
|               SetRgbwPwm(0, 0, 0,                  0, color.W * b / 255); | ||||
|             } else if (color.R == 127 & color.G == 127 && color.B == 127 && color.W == 255) {   | ||||
|               SetRgbwPwm(0, 0, 0, color.W * b / 512, color.W * b / 255); | ||||
|             } else if (color.R ==   0 & color.G ==   0 && color.B ==   0 && color.W == 255) {   | ||||
|               SetRgbwPwm(0, 0, 0, color.W * b / 255,                  0); | ||||
|             } else if (color.R == 130 & color.G ==  90 && color.B ==   0 && color.W == 255) {   | ||||
|               SetRgbwPwm(0, 0, 0, color.W * b / 255, color.W * b / 512); | ||||
|             } else if (color.R == 255 & color.G == 153 && color.B ==   0 && color.W == 255) {   | ||||
|               SetRgbwPwm(0, 0, 0, color.W * b / 255,                  0); | ||||
|             } else {  // not only white colors | ||||
|               SetRgbwPwm(color.R * b / 255, color.G * b / 255, color.B * b / 255, color.W * b / 255); | ||||
|             } | ||||
|           #else | ||||
|             SetRgbwPwm(color.R * b / 255, color.G * b / 255, color.B * b / 255, color.W * b / 255); | ||||
|           #endif          | ||||
|         #endif | ||||
|       } | ||||
|       break; | ||||
|     } | ||||
|      | ||||
|   } | ||||
|  | ||||
|   void SetBrightness(byte b) | ||||
|   { | ||||
|     switch (_type) { | ||||
|       case NeoPixelType_Grb: _pGrb->SetBrightness(b);   break; | ||||
|       case NeoPixelType_Grbw:_pGrbw->SetBrightness(b);  break; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // NOTE:  Due to feature differences, some support RGBW but the method name | ||||
|   // here needs to be unique, thus GetPixeColorRgbw | ||||
|   RgbwColor GetPixelColorRgbw(uint16_t indexPixel) const | ||||
|   { | ||||
|     switch (_type) { | ||||
|       case NeoPixelType_Grb:  return _pGrb->GetPixelColor(indexPixel);  break; | ||||
|       case NeoPixelType_Grbw: return _pGrbw->GetPixelColor(indexPixel); break; | ||||
|     } | ||||
|     return 0; | ||||
|   } | ||||
|  | ||||
|   uint8_t* GetPixels(void) | ||||
|   { | ||||
|     switch (_type) { | ||||
|       case NeoPixelType_Grb:  return _pGrb->Pixels();  break; | ||||
|       case NeoPixelType_Grbw: return _pGrbw->Pixels(); break; | ||||
|     } | ||||
|     return 0; | ||||
|   } | ||||
|  | ||||
|  | ||||
| private: | ||||
|   NeoPixelType _type; | ||||
|  | ||||
|   // have a member for every possible type | ||||
|   NeoPixelBrightnessBus<PIXELFEATURE3,PIXELMETHOD>*  _pGrb; | ||||
|   NeoPixelBrightnessBus<PIXELFEATURE4,PIXELMETHOD>* _pGrbw; | ||||
|  | ||||
|   void cleanup() | ||||
|   { | ||||
|     switch (_type) { | ||||
|       case NeoPixelType_Grb:  delete _pGrb ; _pGrb  = NULL; break; | ||||
|       case NeoPixelType_Grbw: delete _pGrbw; _pGrbw = NULL; break; | ||||
|     } | ||||
|   } | ||||
| }; | ||||
| #endif | ||||
| @@ -17,8 +17,10 @@ | ||||
| #define __ESP32_ESP32__ | ||||
| #define ESP_PLATFORM | ||||
| #define HAVE_CONFIG_H | ||||
| #define GCC_NOT_5_2_0 0 | ||||
| #define WITH_POSIX | ||||
| #define F_CPU 240000000L | ||||
| #define ARDUINO 10809 | ||||
| #define ARDUINO 108011 | ||||
| #define ARDUINO_ESP32_DEV | ||||
| #define ARDUINO_ARCH_ESP32 | ||||
| #define ESP32 | ||||
| @@ -92,18 +94,11 @@ typedef long pthread_cond_t; | ||||
| #include "arduino.h" | ||||
| #include <pins_arduino.h>  | ||||
|  | ||||
| //#include "..\generic\Common.h" | ||||
| //#include "..\generic\pins_arduino.h" | ||||
|  | ||||
| //#undef F | ||||
| //#define F(string_literal) ((const PROGMEM char *)(string_literal)) | ||||
| //#undef PSTR | ||||
| //#define PSTR(string_literal) ((const PROGMEM char *)(string_literal)) | ||||
| //current vc++ does not understand this syntax so use older arduino example for intellisense | ||||
| //todo:move to the new clang/gcc project types. | ||||
| #define interrupts() sei() | ||||
| #define noInterrupts() cli() | ||||
|  | ||||
| #define ESP_LOGI(tag, ...) | ||||
|  | ||||
| #include "wled00.ino" | ||||
| #include "wled01_eeprom.ino" | ||||
| #include "wled02_xml.ino" | ||||
|   | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,10 +1,13 @@ | ||||
| #include "wled.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * Alexa Voice On/Off/Brightness Control. Emulates a Philips Hue bridge to Alexa. | ||||
|  * Alexa Voice On/Off/Brightness/Color Control. Emulates a Philips Hue bridge to Alexa. | ||||
|  *  | ||||
|  * This was put together from these two excellent projects: | ||||
|  * https://github.com/kakopappa/arduino-esp8266-alexa-wemo-switch
 | ||||
|  * https://github.com/probonopd/ESP8266HueEmulator
 | ||||
|  */ | ||||
| #include "src/dependencies/espalexa/EspalexaDevice.h" | ||||
| 
 | ||||
| #ifndef WLED_DISABLE_ALEXA | ||||
| void onAlexaChange(EspalexaDevice* dev); | ||||
| @@ -41,9 +44,9 @@ void onAlexaChange(EspalexaDevice* dev) | ||||
|       if (bri == 0) | ||||
|       { | ||||
|         bri = briLast; | ||||
|         colorUpdated(10); | ||||
|         colorUpdated(NOTIFIER_CALL_MODE_ALEXA); | ||||
|       } | ||||
|     } else applyMacro(macroAlexaOn); | ||||
|     } else applyPreset(macroAlexaOn); | ||||
|   } else if (m == EspalexaDeviceProperty::off) | ||||
|   { | ||||
|     if (!macroAlexaOff) | ||||
| @@ -52,22 +55,39 @@ void onAlexaChange(EspalexaDevice* dev) | ||||
|       { | ||||
|         briLast = bri; | ||||
|         bri = 0; | ||||
|         colorUpdated(10); | ||||
|         colorUpdated(NOTIFIER_CALL_MODE_ALEXA); | ||||
|       } | ||||
|     } else applyMacro(macroAlexaOff); | ||||
|     } else applyPreset(macroAlexaOff); | ||||
|   } else if (m == EspalexaDeviceProperty::bri) | ||||
|   { | ||||
|     bri = espalexaDevice->getValue(); | ||||
|     colorUpdated(10); | ||||
|     colorUpdated(NOTIFIER_CALL_MODE_ALEXA); | ||||
|   } else //color
 | ||||
|   { | ||||
|     uint32_t color = espalexaDevice->getRGB(); | ||||
|     col[3] = ((color >> 24) & 0xFF);  // white color from Alexa is "pure white only" 
 | ||||
|     col[0] = ((color >> 16) & 0xFF); | ||||
|     col[1] = ((color >>  8) & 0xFF); | ||||
|     col[2] = (color & 0xFF); | ||||
|     if (useRGBW && col[3] == 0) colorRGBtoRGBW(col);  // do not touch white value if EspalexaDevice.cpp did set the white channel
 | ||||
|     colorUpdated(10); | ||||
|     if (espalexaDevice->getColorMode() == EspalexaColorMode::ct) //shade of white
 | ||||
|     { | ||||
|       uint16_t ct = espalexaDevice->getCt(); | ||||
|       if (strip.isRgbw) | ||||
|       { | ||||
|         switch (ct) { //these values empirically look good on RGBW
 | ||||
|           case 199: col[0]=255; col[1]=255; col[2]=255; col[3]=255; break; | ||||
|           case 234: col[0]=127; col[1]=127; col[2]=127; col[3]=255; break; | ||||
|           case 284: col[0]=  0; col[1]=  0; col[2]=  0; col[3]=255; break; | ||||
|           case 350: col[0]=130; col[1]= 90; col[2]=  0; col[3]=255; break; | ||||
|           case 383: col[0]=255; col[1]=153; col[2]=  0; col[3]=255; break; | ||||
|         } | ||||
|       } else { | ||||
|         colorCTtoRGB(ct, col); | ||||
|       } | ||||
|     } else { | ||||
|       uint32_t color = espalexaDevice->getRGB(); | ||||
|      | ||||
|       col[0] = ((color >> 16) & 0xFF); | ||||
|       col[1] = ((color >>  8) & 0xFF); | ||||
|       col[2] = ( color        & 0xFF); | ||||
|       col[3] = 0; | ||||
|     } | ||||
|     colorUpdated(NOTIFIER_CALL_MODE_ALEXA); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| @@ -1,3 +1,6 @@ | ||||
| #include "wled.h" | ||||
| #include "src/dependencies/blynk/Blynk/BlynkHandlers.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * Remote light control with the free Blynk app | ||||
|  */ | ||||
| @@ -5,12 +8,12 @@ | ||||
| uint16_t blHue = 0; | ||||
| byte blSat = 255; | ||||
| 
 | ||||
| void initBlynk(const char* auth) | ||||
| void initBlynk(const char *auth, const char *host, uint16_t port) | ||||
| { | ||||
|   #ifndef WLED_DISABLE_BLYNK | ||||
|   if (!WLED_CONNECTED) return; | ||||
|   blynkEnabled = (auth[0] != 0); | ||||
|   if (blynkEnabled) Blynk.config(auth); | ||||
|   if (blynkEnabled) Blynk.config(auth, host, port); | ||||
|   #endif | ||||
| } | ||||
| 
 | ||||
| @@ -41,45 +44,45 @@ void updateBlynk() | ||||
| BLYNK_WRITE(V0) | ||||
| { | ||||
|   bri = param.asInt();//bri
 | ||||
|   colorUpdated(9); | ||||
|   colorUpdated(NOTIFIER_CALL_MODE_BLYNK); | ||||
| } | ||||
| 
 | ||||
| BLYNK_WRITE(V1) | ||||
| { | ||||
|   blHue = param.asInt();//hue
 | ||||
|   colorHStoRGB(blHue*10,blSat,(false)? colSec:col); | ||||
|   colorUpdated(9); | ||||
|   colorUpdated(NOTIFIER_CALL_MODE_BLYNK); | ||||
| } | ||||
| 
 | ||||
| BLYNK_WRITE(V2) | ||||
| { | ||||
|   blSat = param.asInt();//sat
 | ||||
|   colorHStoRGB(blHue*10,blSat,(false)? colSec:col); | ||||
|   colorUpdated(9); | ||||
|   colorUpdated(NOTIFIER_CALL_MODE_BLYNK); | ||||
| } | ||||
| 
 | ||||
| BLYNK_WRITE(V3) | ||||
| { | ||||
|   bool on = (param.asInt()>0); | ||||
|   if (!on != !bri) {toggleOnOff(); colorUpdated(9);} | ||||
|   if (!on != !bri) {toggleOnOff(); colorUpdated(NOTIFIER_CALL_MODE_BLYNK);} | ||||
| } | ||||
| 
 | ||||
| BLYNK_WRITE(V4) | ||||
| { | ||||
|   effectCurrent = param.asInt()-1;//fx
 | ||||
|   colorUpdated(9); | ||||
|   colorUpdated(NOTIFIER_CALL_MODE_BLYNK); | ||||
| } | ||||
| 
 | ||||
| BLYNK_WRITE(V5) | ||||
| { | ||||
|   effectSpeed = param.asInt();//sx
 | ||||
|   colorUpdated(9); | ||||
|   colorUpdated(NOTIFIER_CALL_MODE_BLYNK); | ||||
| } | ||||
| 
 | ||||
| BLYNK_WRITE(V6) | ||||
| { | ||||
|   effectIntensity = param.asInt();//ix
 | ||||
|   colorUpdated(9); | ||||
|   colorUpdated(NOTIFIER_CALL_MODE_BLYNK); | ||||
| } | ||||
| 
 | ||||
| BLYNK_WRITE(V7) | ||||
							
								
								
									
										406
									
								
								wled00/bus_manager.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										406
									
								
								wled00/bus_manager.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,406 @@ | ||||
| #ifndef BusManager_h | ||||
| #define BusManager_h | ||||
|  | ||||
| /* | ||||
|  * Class for addressing various light types | ||||
|  */ | ||||
|  | ||||
| #include "const.h" | ||||
| #include "pin_manager.h" | ||||
| #include "bus_wrapper.h" | ||||
| #include <Arduino.h> | ||||
|  | ||||
| //temporary struct for passing bus configuration to bus | ||||
| struct BusConfig { | ||||
|   uint8_t type = TYPE_WS2812_RGB; | ||||
|   uint16_t count = 1; | ||||
|   uint16_t start = 0; | ||||
|   uint8_t colorOrder = COL_ORDER_GRB; | ||||
|   bool reversed = false; | ||||
|   uint8_t pins[5] = {LEDPIN, 255, 255, 255, 255}; | ||||
|   BusConfig(uint8_t busType, uint8_t* ppins, uint16_t pstart, uint16_t len = 1, uint8_t pcolorOrder = COL_ORDER_GRB, bool rev = false) { | ||||
|     type = busType; count = len; start = pstart; colorOrder = pcolorOrder; reversed = rev; | ||||
|     uint8_t nPins = 1; | ||||
|     if (type > 47) nPins = 2; | ||||
|     else if (type > 41 && type < 46) nPins = NUM_PWM_PINS(type); | ||||
|     for (uint8_t i = 0; i < nPins; i++) pins[i] = ppins[i]; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| //parent class of BusDigital and BusPwm | ||||
| class Bus { | ||||
|   public: | ||||
|   Bus(uint8_t type, uint16_t start) { | ||||
|     _type = type; | ||||
|     _start = start; | ||||
|   }; | ||||
|    | ||||
|   virtual void show() {} | ||||
|   virtual bool canShow() { return true; } | ||||
|  | ||||
|   virtual void setPixelColor(uint16_t pix, uint32_t c) {}; | ||||
|  | ||||
|   virtual void setBrightness(uint8_t b) {}; | ||||
|  | ||||
|   virtual uint32_t getPixelColor(uint16_t pix) { return 0; }; | ||||
|  | ||||
|   virtual void cleanup() {}; | ||||
|  | ||||
|   virtual ~Bus() { //throw the bus under the bus | ||||
|   } | ||||
|  | ||||
|   virtual uint8_t getPins(uint8_t* pinArray) { return 0; } | ||||
|  | ||||
|   uint16_t getStart() { | ||||
|     return _start; | ||||
|   } | ||||
|  | ||||
|   void setStart(uint16_t start) { | ||||
|     _start = start; | ||||
|   } | ||||
|  | ||||
|   virtual uint16_t getLength() { | ||||
|     return 1; | ||||
|   } | ||||
|  | ||||
|   virtual void setColorOrder() {} | ||||
|  | ||||
|   virtual uint8_t getColorOrder() { | ||||
|     return COL_ORDER_RGB; | ||||
|   } | ||||
|  | ||||
|   uint8_t getType() { | ||||
|     return _type; | ||||
|   } | ||||
|  | ||||
|   bool isOk() { | ||||
|     return _valid; | ||||
|   } | ||||
|  | ||||
|   bool reversed = false; | ||||
|  | ||||
|   protected: | ||||
|   uint8_t _type = TYPE_NONE; | ||||
|   uint8_t _bri = 255; | ||||
|   uint16_t _start = 0; | ||||
|   bool _valid = false; | ||||
| }; | ||||
|  | ||||
|  | ||||
| class BusDigital : public Bus { | ||||
|   public: | ||||
|   BusDigital(BusConfig &bc, uint8_t nr) : Bus(bc.type, bc.start) { | ||||
|     if (!IS_DIGITAL(bc.type) || !bc.count) return; | ||||
|     _pins[0] = bc.pins[0]; | ||||
|     if (!pinManager.allocatePin(_pins[0])) return; | ||||
|     if (IS_2PIN(bc.type)) { | ||||
|       _pins[1] = bc.pins[1]; | ||||
|       if (!pinManager.allocatePin(_pins[1])) { | ||||
|         cleanup(); return; | ||||
|       } | ||||
|     } | ||||
|     _len = bc.count; | ||||
|     reversed = bc.reversed; | ||||
|     _iType = PolyBus::getI(bc.type, _pins, nr); | ||||
|     if (_iType == I_NONE) return; | ||||
|     _busPtr = PolyBus::create(_iType, _pins, _len); | ||||
|     _valid = (_busPtr != nullptr); | ||||
|     _colorOrder = bc.colorOrder; | ||||
|     //Serial.printf("Successfully inited strip %u (len %u) with type %u and pins %u,%u (itype %u)\n",nr, len, type, pins[0],pins[1],_iType); | ||||
|   }; | ||||
|  | ||||
|   void show() { | ||||
|     PolyBus::show(_busPtr, _iType); | ||||
|   } | ||||
|  | ||||
|   bool canShow() { | ||||
|     return PolyBus::canShow(_busPtr, _iType); | ||||
|   } | ||||
|  | ||||
|   void setBrightness(uint8_t b) { | ||||
|     //Fix for turning off onboard LED breaking bus | ||||
|     #ifdef LED_BUILTIN | ||||
|     if (_bri == 0 && b > 0) { | ||||
|       if (_pins[0] == LED_BUILTIN || _pins[1] == LED_BUILTIN) PolyBus::begin(_busPtr, _iType, _pins);  | ||||
|     } | ||||
|     #endif | ||||
|     _bri = b; | ||||
|     PolyBus::setBrightness(_busPtr, _iType, b); | ||||
|   } | ||||
|  | ||||
|   void setPixelColor(uint16_t pix, uint32_t c) { | ||||
|     if (reversed) pix = _len - pix -1; | ||||
|     PolyBus::setPixelColor(_busPtr, _iType, pix, c, _colorOrder); | ||||
|   } | ||||
|  | ||||
|   uint32_t getPixelColor(uint16_t pix) { | ||||
|     if (reversed) pix = _len - pix -1; | ||||
|     return PolyBus::getPixelColor(_busPtr, _iType, pix, _colorOrder); | ||||
|   } | ||||
|  | ||||
|   uint8_t getColorOrder() { | ||||
|     return _colorOrder; | ||||
|   } | ||||
|  | ||||
|   uint16_t getLength() { | ||||
|     return _len; | ||||
|   } | ||||
|  | ||||
|   uint8_t getPins(uint8_t* pinArray) { | ||||
|     uint8_t numPins = IS_2PIN(_type) ? 2 : 1; | ||||
|     for (uint8_t i = 0; i < numPins; i++) pinArray[i] = _pins[i]; | ||||
|     return numPins; | ||||
|   } | ||||
|  | ||||
|   void setColorOrder(uint8_t colorOrder) { | ||||
|     if (colorOrder > 5) return; | ||||
|     _colorOrder = colorOrder; | ||||
|   } | ||||
|  | ||||
|   void reinit() { | ||||
|     PolyBus::begin(_busPtr, _iType, _pins); | ||||
|   } | ||||
|  | ||||
|   void cleanup() { | ||||
|     //Serial.println("Digital Cleanup"); | ||||
|     PolyBus::cleanup(_busPtr, _iType); | ||||
|     _iType = I_NONE; | ||||
|     _valid = false; | ||||
|     _busPtr = nullptr; | ||||
|     pinManager.deallocatePin(_pins[0]); | ||||
|     pinManager.deallocatePin(_pins[1]); | ||||
|   } | ||||
|  | ||||
|   ~BusDigital() { | ||||
|     cleanup(); | ||||
|   } | ||||
|  | ||||
|   private:  | ||||
|   uint8_t _colorOrder = COL_ORDER_GRB; | ||||
|   uint8_t _pins[2] = {255, 255}; | ||||
|   uint8_t _iType = I_NONE; | ||||
|   uint16_t _len = 0; | ||||
|   void * _busPtr = nullptr; | ||||
| }; | ||||
|  | ||||
|  | ||||
| class BusPwm : public Bus { | ||||
|   public: | ||||
|   BusPwm(BusConfig &bc) : Bus(bc.type, bc.start) { | ||||
|     if (!IS_PWM(bc.type)) return; | ||||
|     uint8_t numPins = NUM_PWM_PINS(bc.type); | ||||
|  | ||||
|     #ifdef ESP8266 | ||||
|     analogWriteRange(255);  //same range as one RGB channel | ||||
|     analogWriteFreq(WLED_PWM_FREQ); | ||||
|     #else | ||||
|     _ledcStart = pinManager.allocateLedc(numPins); | ||||
|     if (_ledcStart == 255) { //no more free LEDC channels | ||||
|       deallocatePins(); return; | ||||
|     } | ||||
|     #endif | ||||
|  | ||||
|     for (uint8_t i = 0; i < numPins; i++) { | ||||
|       _pins[i] = bc.pins[i]; | ||||
|       if (!pinManager.allocatePin(_pins[i])) { | ||||
|         deallocatePins(); return; | ||||
|       } | ||||
|       #ifdef ESP8266 | ||||
|       pinMode(_pins[i], OUTPUT); | ||||
|       #else | ||||
|       ledcSetup(_ledcStart + i, WLED_PWM_FREQ, 8); | ||||
|       ledcAttachPin(_pins[i], _ledcStart + i); | ||||
|       #endif | ||||
|     } | ||||
|     reversed = bc.reversed; | ||||
|     _valid = true; | ||||
|   }; | ||||
|  | ||||
|   void setPixelColor(uint16_t pix, uint32_t c) { | ||||
|     if (pix != 0 || !_valid) return; //only react to first pixel | ||||
|     uint8_t r = c >> 16; | ||||
|     uint8_t g = c >>  8; | ||||
|     uint8_t b = c      ; | ||||
|     uint8_t w = c >> 24; | ||||
|  | ||||
|     switch (_type) { | ||||
|       case TYPE_ANALOG_1CH: //one channel (white), use highest RGBW value | ||||
|         _data[0] = max(r, max(g, max(b, w))); break; | ||||
|        | ||||
|       case TYPE_ANALOG_2CH: //warm white + cold white, we'll need some nice handling here, for now just R+G channels | ||||
|       case TYPE_ANALOG_3CH: //standard dumb RGB | ||||
|       case TYPE_ANALOG_4CH: //RGBW | ||||
|       case TYPE_ANALOG_5CH: //we'll want the white handling from 2CH here + RGB | ||||
|         _data[0] = r; _data[1] = g; _data[2] = b; _data[3] = w; _data[4] = 0; break; | ||||
|  | ||||
|       default: return; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //does no index check | ||||
|   uint32_t getPixelColor(uint16_t pix) { | ||||
|     return ((_data[3] << 24) | (_data[0] << 16) | (_data[1] << 8) | (_data[2])); | ||||
|   } | ||||
|  | ||||
|   void show() { | ||||
|     uint8_t numPins = NUM_PWM_PINS(_type); | ||||
|     for (uint8_t i = 0; i < numPins; i++) { | ||||
|       uint8_t scaled = (_data[i] * _bri) / 255; | ||||
|       if (reversed) scaled = 255 - scaled; | ||||
|       #ifdef ESP8266 | ||||
|       analogWrite(_pins[i], scaled); | ||||
|       #else | ||||
|       ledcWrite(_ledcStart + i, scaled); | ||||
|       #endif | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void setBrightness(uint8_t b) { | ||||
|     _bri = b; | ||||
|   } | ||||
|  | ||||
|   uint8_t getPins(uint8_t* pinArray) { | ||||
|     uint8_t numPins = NUM_PWM_PINS(_type); | ||||
|     for (uint8_t i = 0; i < numPins; i++) pinArray[i] = _pins[i]; | ||||
|     return numPins; | ||||
|   } | ||||
|  | ||||
|   void cleanup() { | ||||
|     deallocatePins(); | ||||
|   } | ||||
|  | ||||
|   ~BusPwm() { | ||||
|     cleanup(); | ||||
|   } | ||||
|  | ||||
|   private:  | ||||
|   uint8_t _pins[5] = {255, 255, 255, 255, 255}; | ||||
|   uint8_t _data[5] = {255, 255, 255, 255, 255}; | ||||
|   #ifdef ARDUINO_ARCH_ESP32 | ||||
|   uint8_t _ledcStart = 255; | ||||
|   #endif | ||||
|  | ||||
|   void deallocatePins() { | ||||
|     uint8_t numPins = NUM_PWM_PINS(_type); | ||||
|     for (uint8_t i = 0; i < numPins; i++) { | ||||
|       if (!pinManager.isPinOk(_pins[i])) continue; | ||||
|       #ifdef ESP8266 | ||||
|       digitalWrite(_pins[i], LOW); //turn off PWM interrupt | ||||
|       #else | ||||
|       if (_ledcStart < 16) ledcDetachPin(_pins[i]); | ||||
|       #endif | ||||
|       pinManager.deallocatePin(_pins[i]); | ||||
|     } | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|     pinManager.deallocateLedc(_ledcStart, numPins); | ||||
|     #endif | ||||
|   } | ||||
| }; | ||||
|  | ||||
| class BusManager { | ||||
|   public: | ||||
|   BusManager() { | ||||
|  | ||||
|   }; | ||||
|  | ||||
|   //utility to get the approx. memory usage of a given BusConfig | ||||
|   uint32_t memUsage(BusConfig &bc) { | ||||
|     uint8_t type = bc.type; | ||||
|     uint16_t len = bc.count; | ||||
|     if (type < 32) { | ||||
|       #ifdef ESP8266 | ||||
|         if (bc.pins[0] == 3) { //8266 DMA uses 5x the mem | ||||
|           if (type > 29) return len*20; //RGBW | ||||
|           return len*15; | ||||
|         } | ||||
|         if (type > 29) return len*4; //RGBW | ||||
|         return len*3; | ||||
|       #else //ESP32 RMT uses double buffer? | ||||
|         if (type > 29) return len*8; //RGBW | ||||
|         return len*6; | ||||
|       #endif | ||||
|     } | ||||
|  | ||||
|     if (type > 31 && type < 48) return 5; | ||||
|     if (type == 44 || type == 45) return len*4; //RGBW | ||||
|     return len*3; | ||||
|   } | ||||
|    | ||||
|   int add(BusConfig &bc) { | ||||
|     if (numBusses >= WLED_MAX_BUSSES) return -1; | ||||
|     if (IS_DIGITAL(bc.type)) { | ||||
|       busses[numBusses] = new BusDigital(bc, numBusses); | ||||
|     } else { | ||||
|       busses[numBusses] = new BusPwm(bc); | ||||
|     } | ||||
|     numBusses++; | ||||
|     return numBusses -1; | ||||
|   } | ||||
|  | ||||
|   //do not call this method from system context (network callback) | ||||
|   void removeAll() { | ||||
|     //Serial.println("Removing all."); | ||||
|     //prevents crashes due to deleting busses while in use.  | ||||
|     while (!canAllShow()) yield(); | ||||
|     for (uint8_t i = 0; i < numBusses; i++) delete busses[i]; | ||||
|     numBusses = 0; | ||||
|   } | ||||
|  | ||||
|   void show() { | ||||
|     for (uint8_t i = 0; i < numBusses; i++) { | ||||
|       busses[i]->show(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void setPixelColor(uint16_t pix, uint32_t c) { | ||||
|     for (uint8_t i = 0; i < numBusses; i++) { | ||||
|       Bus* b = busses[i]; | ||||
|       uint16_t bstart = b->getStart(); | ||||
|       if (pix < bstart || pix >= bstart + b->getLength()) continue; | ||||
|       busses[i]->setPixelColor(pix - bstart, c); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void setBrightness(uint8_t b) { | ||||
|     for (uint8_t i = 0; i < numBusses; i++) { | ||||
|       busses[i]->setBrightness(b); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   uint32_t getPixelColor(uint16_t pix) { | ||||
|     for (uint8_t i = 0; i < numBusses; i++) { | ||||
|       Bus* b = busses[i]; | ||||
|       uint16_t bstart = b->getStart(); | ||||
|       if (pix < bstart || pix >= bstart + b->getLength()) continue; | ||||
|       return b->getPixelColor(pix - bstart); | ||||
|     } | ||||
|     return 0; | ||||
|   } | ||||
|  | ||||
|   bool canAllShow() { | ||||
|     for (uint8_t i = 0; i < numBusses; i++) { | ||||
|       if (!busses[i]->canShow()) return false; | ||||
|     } | ||||
|     return true; | ||||
|   } | ||||
|  | ||||
|   Bus* getBus(uint8_t busNr) { | ||||
|     if (busNr >= numBusses) return nullptr; | ||||
|     return busses[busNr]; | ||||
|   } | ||||
|  | ||||
|   uint8_t getNumBusses() { | ||||
|     return numBusses; | ||||
|   } | ||||
|  | ||||
|   static bool isRgbw(uint8_t type) { | ||||
|     if (type == TYPE_SK6812_RGBW || type == TYPE_TM1814) return true; | ||||
|     if (type > TYPE_ONOFF && type <= TYPE_ANALOG_5CH && type != TYPE_ANALOG_3CH) return true; | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   private: | ||||
|   uint8_t numBusses = 0; | ||||
|   Bus* busses[WLED_MAX_BUSSES]; | ||||
| }; | ||||
| #endif | ||||
							
								
								
									
										882
									
								
								wled00/bus_wrapper.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										882
									
								
								wled00/bus_wrapper.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,882 @@ | ||||
| #ifndef BusWrapper_h | ||||
| #define BusWrapper_h | ||||
|  | ||||
| #include "NeoPixelBrightnessBus.h" | ||||
|  | ||||
| //Hardware SPI Pins | ||||
| #define P_8266_HS_MOSI 13 | ||||
| #define P_8266_HS_CLK  14 | ||||
| #define P_32_HS_MOSI   13 | ||||
| #define P_32_HS_CLK    14 | ||||
| #define P_32_VS_MOSI   23 | ||||
| #define P_32_VS_CLK    18 | ||||
|  | ||||
| //The dirty list of possible bus types. Quite a lot... | ||||
| #define I_NONE 0 | ||||
| //ESP8266 RGB | ||||
| #define I_8266_U0_NEO_3 1 | ||||
| #define I_8266_U1_NEO_3 2 | ||||
| #define I_8266_DM_NEO_3 3 | ||||
| #define I_8266_BB_NEO_3 4 | ||||
| //RGBW | ||||
| #define I_8266_U0_NEO_4 5 | ||||
| #define I_8266_U1_NEO_4 6 | ||||
| #define I_8266_DM_NEO_4 7 | ||||
| #define I_8266_BB_NEO_4 8 | ||||
| //400Kbps | ||||
| #define I_8266_U0_400_3 9 | ||||
| #define I_8266_U1_400_3 10 | ||||
| #define I_8266_DM_400_3 11 | ||||
| #define I_8266_BB_400_3 12 | ||||
| //TM1418 (RGBW) | ||||
| #define I_8266_U0_TM1_4 13 | ||||
| #define I_8266_U1_TM1_4 14 | ||||
| #define I_8266_DM_TM1_4 15 | ||||
| #define I_8266_BB_TM1_4 16 | ||||
|  | ||||
| /*** ESP32 Neopixel methods ***/ | ||||
| //RGB | ||||
| #define I_32_R0_NEO_3 17 | ||||
| #define I_32_R1_NEO_3 18 | ||||
| #define I_32_R2_NEO_3 19 | ||||
| #define I_32_R3_NEO_3 20 | ||||
| #define I_32_R4_NEO_3 21 | ||||
| #define I_32_R5_NEO_3 22 | ||||
| #define I_32_R6_NEO_3 23 | ||||
| #define I_32_R7_NEO_3 24 | ||||
| #define I_32_I0_NEO_3 25 | ||||
| #define I_32_I1_NEO_3 26 | ||||
| //RGBW | ||||
| #define I_32_R0_NEO_4 27 | ||||
| #define I_32_R1_NEO_4 28 | ||||
| #define I_32_R2_NEO_4 29 | ||||
| #define I_32_R3_NEO_4 30 | ||||
| #define I_32_R4_NEO_4 31 | ||||
| #define I_32_R5_NEO_4 32 | ||||
| #define I_32_R6_NEO_4 33 | ||||
| #define I_32_R7_NEO_4 34 | ||||
| #define I_32_I0_NEO_4 35 | ||||
| #define I_32_I1_NEO_4 36 | ||||
| //400Kbps | ||||
| #define I_32_R0_400_3 37 | ||||
| #define I_32_R1_400_3 38 | ||||
| #define I_32_R2_400_3 39 | ||||
| #define I_32_R3_400_3 40 | ||||
| #define I_32_R4_400_3 41 | ||||
| #define I_32_R5_400_3 42 | ||||
| #define I_32_R6_400_3 43 | ||||
| #define I_32_R7_400_3 44 | ||||
| #define I_32_I0_400_3 45 | ||||
| #define I_32_I1_400_3 46 | ||||
| //TM1418 (RGBW) | ||||
| #define I_32_R0_TM1_4 47 | ||||
| #define I_32_R1_TM1_4 48 | ||||
| #define I_32_R2_TM1_4 49 | ||||
| #define I_32_R3_TM1_4 50 | ||||
| #define I_32_R4_TM1_4 51 | ||||
| #define I_32_R5_TM1_4 52 | ||||
| #define I_32_R6_TM1_4 53 | ||||
| #define I_32_R7_TM1_4 54 | ||||
| #define I_32_I0_TM1_4 55 | ||||
| #define I_32_I1_TM1_4 56 | ||||
| //Bit Bang theoratically possible, but very undesirable and not needed (no pin restrictions on RMT and I2S) | ||||
|  | ||||
| //APA102 | ||||
| #define I_HS_DOT_3 57 //hardware SPI | ||||
| #define I_SS_DOT_3 58 //soft SPI | ||||
|  | ||||
| //LPD8806 | ||||
| #define I_HS_LPD_3 59 | ||||
| #define I_SS_LPD_3 60 | ||||
|  | ||||
| //WS2801 | ||||
| #define I_HS_WS1_3 61 | ||||
| #define I_SS_WS1_3 62 | ||||
|  | ||||
| //P9813 | ||||
| #define I_HS_P98_3 63 | ||||
| #define I_SS_P98_3 64 | ||||
|  | ||||
|  | ||||
| /*** ESP8266 Neopixel methods ***/ | ||||
| #ifdef ESP8266 | ||||
| //RGB | ||||
| #define B_8266_U0_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp8266Uart0Ws2813Method> //3 chan, esp8266, gpio1 | ||||
| #define B_8266_U1_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp8266Uart1Ws2813Method> //3 chan, esp8266, gpio2 | ||||
| #define B_8266_DM_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp8266Dma800KbpsMethod>  //3 chan, esp8266, gpio3 | ||||
| #define B_8266_BB_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp8266BitBang800KbpsMethod> //3 chan, esp8266, bb (any pin but 16) | ||||
| //RGBW | ||||
| #define B_8266_U0_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp8266Uart0Ws2813Method>   //4 chan, esp8266, gpio1 | ||||
| #define B_8266_U1_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp8266Uart1Ws2813Method>   //4 chan, esp8266, gpio2 | ||||
| #define B_8266_DM_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp8266Dma800KbpsMethod>    //4 chan, esp8266, gpio3 | ||||
| #define B_8266_BB_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp8266BitBang800KbpsMethod> //4 chan, esp8266, bb (any pin) | ||||
| //400Kbps | ||||
| #define B_8266_U0_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp8266Uart0400KbpsMethod>   //3 chan, esp8266, gpio1 | ||||
| #define B_8266_U1_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp8266Uart1400KbpsMethod>   //3 chan, esp8266, gpio2 | ||||
| #define B_8266_DM_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp8266Dma400KbpsMethod>     //3 chan, esp8266, gpio3 | ||||
| #define B_8266_BB_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp8266BitBang400KbpsMethod> //3 chan, esp8266, bb (any pin) | ||||
| //TM1418 (RGBW) | ||||
| #define B_8266_U0_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp8266Uart0Tm1814Method> | ||||
| #define B_8266_U1_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp8266Uart1Tm1814Method> | ||||
| #define B_8266_DM_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp8266DmaTm1814Method> | ||||
| #define B_8266_BB_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp8266BitBangTm1814Method> | ||||
| #endif | ||||
|  | ||||
| /*** ESP32 Neopixel methods ***/ | ||||
| #ifdef ARDUINO_ARCH_ESP32 | ||||
| //RGB | ||||
| #define B_32_R0_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt0Ws2812xMethod> | ||||
| #define B_32_R1_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt1Ws2812xMethod> | ||||
| #define B_32_R2_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt2Ws2812xMethod> | ||||
| #define B_32_R3_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt3Ws2812xMethod> | ||||
| #define B_32_R4_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt4Ws2812xMethod> | ||||
| #define B_32_R5_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt5Ws2812xMethod> | ||||
| #define B_32_R6_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt6Ws2812xMethod> | ||||
| #define B_32_R7_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt7Ws2812xMethod> | ||||
| #define B_32_I0_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32I2s0800KbpsMethod> | ||||
| #define B_32_I1_NEO_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32I2s1800KbpsMethod> | ||||
| //RGBW | ||||
| #define B_32_R0_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp32Rmt0Ws2812xMethod> | ||||
| #define B_32_R1_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp32Rmt1Ws2812xMethod> | ||||
| #define B_32_R2_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp32Rmt2Ws2812xMethod> | ||||
| #define B_32_R3_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp32Rmt3Ws2812xMethod> | ||||
| #define B_32_R4_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp32Rmt4Ws2812xMethod> | ||||
| #define B_32_R5_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp32Rmt5Ws2812xMethod> | ||||
| #define B_32_R6_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp32Rmt6Ws2812xMethod> | ||||
| #define B_32_R7_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp32Rmt7Ws2812xMethod> | ||||
| #define B_32_I0_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp32I2s0800KbpsMethod> | ||||
| #define B_32_I1_NEO_4 NeoPixelBrightnessBus<NeoGrbwFeature, NeoEsp32I2s1800KbpsMethod> | ||||
| //400Kbps | ||||
| #define B_32_R0_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt0400KbpsMethod> | ||||
| #define B_32_R1_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt1400KbpsMethod> | ||||
| #define B_32_R2_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt2400KbpsMethod> | ||||
| #define B_32_R3_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt3400KbpsMethod> | ||||
| #define B_32_R4_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt4400KbpsMethod> | ||||
| #define B_32_R5_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt5400KbpsMethod> | ||||
| #define B_32_R6_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt6400KbpsMethod> | ||||
| #define B_32_R7_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32Rmt7400KbpsMethod> | ||||
| #define B_32_I0_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32I2s0400KbpsMethod> | ||||
| #define B_32_I1_400_3 NeoPixelBrightnessBus<NeoGrbFeature, NeoEsp32I2s1400KbpsMethod> | ||||
| //TM1418 (RGBW) | ||||
| #define B_32_R0_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp32Rmt0Tm1814Method> | ||||
| #define B_32_R1_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp32Rmt1Tm1814Method> | ||||
| #define B_32_R2_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp32Rmt2Tm1814Method> | ||||
| #define B_32_R3_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp32Rmt3Tm1814Method> | ||||
| #define B_32_R4_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp32Rmt4Tm1814Method> | ||||
| #define B_32_R5_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp32Rmt5Tm1814Method> | ||||
| #define B_32_R6_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp32Rmt6Tm1814Method> | ||||
| #define B_32_R7_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp32Rmt7Tm1814Method> | ||||
| #define B_32_I0_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp32I2s0Tm1814Method> | ||||
| #define B_32_I1_TM1_4 NeoPixelBrightnessBus<NeoWrgbTm1814Feature, NeoEsp32I2s1Tm1814Method> | ||||
| //Bit Bang theoratically possible, but very undesirable and not needed (no pin restrictions on RMT and I2S) | ||||
|  | ||||
| #endif | ||||
|  | ||||
| //APA102 | ||||
| #define B_HS_DOT_3 NeoPixelBrightnessBus<DotStarBgrFeature, DotStarSpiMethod> //hardware SPI | ||||
| #define B_SS_DOT_3 NeoPixelBrightnessBus<DotStarBgrFeature, DotStarMethod> //soft SPI | ||||
|  | ||||
| //LPD8806 | ||||
| #define B_HS_LPD_3 NeoPixelBrightnessBus<Lpd8806GrbFeature, Lpd8806SpiMethod> | ||||
| #define B_SS_LPD_3 NeoPixelBrightnessBus<Lpd8806GrbFeature, Lpd8806Method> | ||||
|  | ||||
| //WS2801 | ||||
| #define B_HS_WS1_3 NeoPixelBrightnessBus<NeoRbgFeature, NeoWs2801SpiMethod> | ||||
| #define B_SS_WS1_3 NeoPixelBrightnessBus<NeoRbgFeature, NeoWs2801Method> | ||||
|  | ||||
| //P9813 | ||||
| #define B_HS_P98_3 NeoPixelBrightnessBus<P9813BgrFeature, P9813SpiMethod> | ||||
| #define B_SS_P98_3 NeoPixelBrightnessBus<P9813BgrFeature, P9813Method> | ||||
|  | ||||
| //handles pointer type conversion for all possible bus types | ||||
| class PolyBus { | ||||
|   public: | ||||
|   static void begin(void* busPtr, uint8_t busType, uint8_t* pins) { | ||||
|     switch (busType) { | ||||
|       case I_NONE: break; | ||||
|     #ifdef ESP8266 | ||||
|       case I_8266_U0_NEO_3: (static_cast<B_8266_U0_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_8266_U1_NEO_3: (static_cast<B_8266_U1_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_8266_DM_NEO_3: (static_cast<B_8266_DM_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_8266_BB_NEO_3: (static_cast<B_8266_BB_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_8266_U0_NEO_4: (static_cast<B_8266_U0_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_8266_U1_NEO_4: (static_cast<B_8266_U1_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_8266_DM_NEO_4: (static_cast<B_8266_DM_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_8266_BB_NEO_4: (static_cast<B_8266_BB_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_8266_U0_400_3: (static_cast<B_8266_U0_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_8266_U1_400_3: (static_cast<B_8266_U1_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_8266_DM_400_3: (static_cast<B_8266_DM_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_8266_BB_400_3: (static_cast<B_8266_BB_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_8266_U0_TM1_4: (static_cast<B_8266_U0_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_8266_U1_TM1_4: (static_cast<B_8266_U1_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_8266_DM_TM1_4: (static_cast<B_8266_DM_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_8266_BB_TM1_4: (static_cast<B_8266_BB_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_HS_DOT_3: (static_cast<B_HS_DOT_3*>(busPtr))->Begin(); break; | ||||
|       case I_HS_LPD_3: (static_cast<B_HS_LPD_3*>(busPtr))->Begin(); break; | ||||
|       case I_HS_WS1_3: (static_cast<B_HS_WS1_3*>(busPtr))->Begin(); break; | ||||
|       case I_HS_P98_3: (static_cast<B_HS_P98_3*>(busPtr))->Begin(); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_R0_NEO_3: (static_cast<B_32_R0_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R1_NEO_3: (static_cast<B_32_R1_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R2_NEO_3: (static_cast<B_32_R2_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R3_NEO_3: (static_cast<B_32_R3_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R4_NEO_3: (static_cast<B_32_R4_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R5_NEO_3: (static_cast<B_32_R5_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R6_NEO_3: (static_cast<B_32_R6_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R7_NEO_3: (static_cast<B_32_R7_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_I0_NEO_3: (static_cast<B_32_I0_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_I1_NEO_3: (static_cast<B_32_I1_NEO_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R0_NEO_4: (static_cast<B_32_R0_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R1_NEO_4: (static_cast<B_32_R1_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R2_NEO_4: (static_cast<B_32_R2_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R3_NEO_4: (static_cast<B_32_R3_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R4_NEO_4: (static_cast<B_32_R4_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R5_NEO_4: (static_cast<B_32_R5_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R6_NEO_4: (static_cast<B_32_R6_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R7_NEO_4: (static_cast<B_32_R7_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_I0_NEO_4: (static_cast<B_32_I0_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_I1_NEO_4: (static_cast<B_32_I1_NEO_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R0_400_3: (static_cast<B_32_R0_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R1_400_3: (static_cast<B_32_R1_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R2_400_3: (static_cast<B_32_R2_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R3_400_3: (static_cast<B_32_R3_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R4_400_3: (static_cast<B_32_R4_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R5_400_3: (static_cast<B_32_R5_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R6_400_3: (static_cast<B_32_R6_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R7_400_3: (static_cast<B_32_R7_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_I0_400_3: (static_cast<B_32_I0_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_I1_400_3: (static_cast<B_32_I1_400_3*>(busPtr))->Begin(); break; | ||||
|       case I_32_R0_TM1_4: (static_cast<B_32_R0_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R1_TM1_4: (static_cast<B_32_R1_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R2_TM1_4: (static_cast<B_32_R2_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R3_TM1_4: (static_cast<B_32_R3_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R4_TM1_4: (static_cast<B_32_R4_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R5_TM1_4: (static_cast<B_32_R5_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R6_TM1_4: (static_cast<B_32_R6_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_R7_TM1_4: (static_cast<B_32_R7_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_I0_TM1_4: (static_cast<B_32_I0_TM1_4*>(busPtr))->Begin(); break; | ||||
|       case I_32_I1_TM1_4: (static_cast<B_32_I1_TM1_4*>(busPtr))->Begin(); break; | ||||
|       // ESP32 can (and should, to avoid inadvertantly driving the chip select signal) specify the pins used for SPI, but only in begin() | ||||
|       case I_HS_DOT_3: (static_cast<B_HS_DOT_3*>(busPtr))->Begin(pins[1], -1, pins[0], -1); break; | ||||
|       case I_HS_LPD_3: (static_cast<B_HS_LPD_3*>(busPtr))->Begin(pins[1], -1, pins[0], -1); break; | ||||
|       case I_HS_WS1_3: (static_cast<B_HS_WS1_3*>(busPtr))->Begin(pins[1], -1, pins[0], -1); break; | ||||
|       case I_HS_P98_3: (static_cast<B_HS_P98_3*>(busPtr))->Begin(pins[1], -1, pins[0], -1); break; | ||||
|     #endif | ||||
|       case I_SS_DOT_3: (static_cast<B_SS_DOT_3*>(busPtr))->Begin(); break; | ||||
|       case I_SS_LPD_3: (static_cast<B_SS_LPD_3*>(busPtr))->Begin(); break; | ||||
|       case I_SS_WS1_3: (static_cast<B_SS_WS1_3*>(busPtr))->Begin(); break; | ||||
|       case I_SS_P98_3: (static_cast<B_SS_P98_3*>(busPtr))->Begin(); break; | ||||
|     } | ||||
|   }; | ||||
|   static void* create(uint8_t busType, uint8_t* pins, uint16_t len) { | ||||
|     void* busPtr = nullptr; | ||||
|     switch (busType) { | ||||
|       case I_NONE: break; | ||||
|     #ifdef ESP8266 | ||||
|       case I_8266_U0_NEO_3: busPtr = new B_8266_U0_NEO_3(len, pins[0]); break; | ||||
|       case I_8266_U1_NEO_3: busPtr = new B_8266_U1_NEO_3(len, pins[0]); break; | ||||
|       case I_8266_DM_NEO_3: busPtr = new B_8266_DM_NEO_3(len, pins[0]); break; | ||||
|       case I_8266_BB_NEO_3: busPtr = new B_8266_BB_NEO_3(len, pins[0]); break; | ||||
|       case I_8266_U0_NEO_4: busPtr = new B_8266_U0_NEO_4(len, pins[0]); break; | ||||
|       case I_8266_U1_NEO_4: busPtr = new B_8266_U1_NEO_4(len, pins[0]); break; | ||||
|       case I_8266_DM_NEO_4: busPtr = new B_8266_DM_NEO_4(len, pins[0]); break; | ||||
|       case I_8266_BB_NEO_4: busPtr = new B_8266_BB_NEO_4(len, pins[0]); break; | ||||
|       case I_8266_U0_400_3: busPtr = new B_8266_U0_400_3(len, pins[0]); break; | ||||
|       case I_8266_U1_400_3: busPtr = new B_8266_U1_400_3(len, pins[0]); break; | ||||
|       case I_8266_DM_400_3: busPtr = new B_8266_DM_400_3(len, pins[0]); break; | ||||
|       case I_8266_BB_400_3: busPtr = new B_8266_BB_400_3(len, pins[0]); break; | ||||
|       case I_8266_U0_TM1_4: busPtr = new B_8266_U0_TM1_4(len, pins[0]); break; | ||||
|       case I_8266_U1_TM1_4: busPtr = new B_8266_U1_TM1_4(len, pins[0]); break; | ||||
|       case I_8266_DM_TM1_4: busPtr = new B_8266_DM_TM1_4(len, pins[0]); break; | ||||
|       case I_8266_BB_TM1_4: busPtr = new B_8266_BB_TM1_4(len, pins[0]); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_R0_NEO_3: busPtr = new B_32_R0_NEO_3(len, pins[0]); break; | ||||
|       case I_32_R1_NEO_3: busPtr = new B_32_R1_NEO_3(len, pins[0]); break; | ||||
|       case I_32_R2_NEO_3: busPtr = new B_32_R2_NEO_3(len, pins[0]); break; | ||||
|       case I_32_R3_NEO_3: busPtr = new B_32_R3_NEO_3(len, pins[0]); break; | ||||
|       case I_32_R4_NEO_3: busPtr = new B_32_R4_NEO_3(len, pins[0]); break; | ||||
|       case I_32_R5_NEO_3: busPtr = new B_32_R5_NEO_3(len, pins[0]); break; | ||||
|       case I_32_R6_NEO_3: busPtr = new B_32_R6_NEO_3(len, pins[0]); break; | ||||
|       case I_32_R7_NEO_3: busPtr = new B_32_R7_NEO_3(len, pins[0]); break; | ||||
|       case I_32_I0_NEO_3: busPtr = new B_32_I0_NEO_3(len, pins[0]); break; | ||||
|       case I_32_I1_NEO_3: busPtr = new B_32_I1_NEO_3(len, pins[0]); break; | ||||
|       case I_32_R0_NEO_4: busPtr = new B_32_R0_NEO_4(len, pins[0]); break; | ||||
|       case I_32_R1_NEO_4: busPtr = new B_32_R1_NEO_4(len, pins[0]); break; | ||||
|       case I_32_R2_NEO_4: busPtr = new B_32_R2_NEO_4(len, pins[0]); break; | ||||
|       case I_32_R3_NEO_4: busPtr = new B_32_R3_NEO_4(len, pins[0]); break; | ||||
|       case I_32_R4_NEO_4: busPtr = new B_32_R4_NEO_4(len, pins[0]); break; | ||||
|       case I_32_R5_NEO_4: busPtr = new B_32_R5_NEO_4(len, pins[0]); break; | ||||
|       case I_32_R6_NEO_4: busPtr = new B_32_R6_NEO_4(len, pins[0]); break; | ||||
|       case I_32_R7_NEO_4: busPtr = new B_32_R7_NEO_4(len, pins[0]); break; | ||||
|       case I_32_I0_NEO_4: busPtr = new B_32_I0_NEO_4(len, pins[0]); break; | ||||
|       case I_32_I1_NEO_4: busPtr = new B_32_I1_NEO_4(len, pins[0]); break; | ||||
|       case I_32_R0_400_3: busPtr = new B_32_R0_400_3(len, pins[0]); break; | ||||
|       case I_32_R1_400_3: busPtr = new B_32_R1_400_3(len, pins[0]); break; | ||||
|       case I_32_R2_400_3: busPtr = new B_32_R2_400_3(len, pins[0]); break; | ||||
|       case I_32_R3_400_3: busPtr = new B_32_R3_400_3(len, pins[0]); break; | ||||
|       case I_32_R4_400_3: busPtr = new B_32_R4_400_3(len, pins[0]); break; | ||||
|       case I_32_R5_400_3: busPtr = new B_32_R5_400_3(len, pins[0]); break; | ||||
|       case I_32_R6_400_3: busPtr = new B_32_R6_400_3(len, pins[0]); break; | ||||
|       case I_32_R7_400_3: busPtr = new B_32_R7_400_3(len, pins[0]); break; | ||||
|       case I_32_I0_400_3: busPtr = new B_32_I0_400_3(len, pins[0]); break; | ||||
|       case I_32_I1_400_3: busPtr = new B_32_I1_400_3(len, pins[0]); break; | ||||
|       case I_32_R0_TM1_4: busPtr = new B_32_R0_TM1_4(len, pins[0]); break; | ||||
|       case I_32_R1_TM1_4: busPtr = new B_32_R1_TM1_4(len, pins[0]); break; | ||||
|       case I_32_R2_TM1_4: busPtr = new B_32_R2_TM1_4(len, pins[0]); break; | ||||
|       case I_32_R3_TM1_4: busPtr = new B_32_R3_TM1_4(len, pins[0]); break; | ||||
|       case I_32_R4_TM1_4: busPtr = new B_32_R4_TM1_4(len, pins[0]); break; | ||||
|       case I_32_R5_TM1_4: busPtr = new B_32_R5_TM1_4(len, pins[0]); break; | ||||
|       case I_32_R6_TM1_4: busPtr = new B_32_R6_TM1_4(len, pins[0]); break; | ||||
|       case I_32_R7_TM1_4: busPtr = new B_32_R7_TM1_4(len, pins[0]); break; | ||||
|       case I_32_I0_TM1_4: busPtr = new B_32_I0_TM1_4(len, pins[0]); break; | ||||
|       case I_32_I1_TM1_4: busPtr = new B_32_I1_TM1_4(len, pins[0]); break; | ||||
|     #endif | ||||
|       // for 2-wire: pins[1] is clk, pins[0] is dat.  begin expects (len, clk, dat) | ||||
|       case I_HS_DOT_3: busPtr = new B_HS_DOT_3(len, pins[1], pins[0]); break; | ||||
|       case I_SS_DOT_3: busPtr = new B_SS_DOT_3(len, pins[1], pins[0]); break; | ||||
|       case I_HS_LPD_3: busPtr = new B_HS_LPD_3(len, pins[1], pins[0]); break; | ||||
|       case I_SS_LPD_3: busPtr = new B_SS_LPD_3(len, pins[1], pins[0]); break; | ||||
|       case I_HS_WS1_3: busPtr = new B_HS_WS1_3(len, pins[1], pins[0]); break; | ||||
|       case I_SS_WS1_3: busPtr = new B_SS_WS1_3(len, pins[1], pins[0]); break; | ||||
|       case I_HS_P98_3: busPtr = new B_HS_P98_3(len, pins[1], pins[0]); break; | ||||
|       case I_SS_P98_3: busPtr = new B_SS_P98_3(len, pins[1], pins[0]); break; | ||||
|     } | ||||
|     begin(busPtr, busType, pins); | ||||
|     return busPtr; | ||||
|   }; | ||||
|   static void show(void* busPtr, uint8_t busType) { | ||||
|     switch (busType) { | ||||
|       case I_NONE: break; | ||||
|     #ifdef ESP8266 | ||||
|       case I_8266_U0_NEO_3: (static_cast<B_8266_U0_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_8266_U1_NEO_3: (static_cast<B_8266_U1_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_8266_DM_NEO_3: (static_cast<B_8266_DM_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_8266_BB_NEO_3: (static_cast<B_8266_BB_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_8266_U0_NEO_4: (static_cast<B_8266_U0_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_8266_U1_NEO_4: (static_cast<B_8266_U1_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_8266_DM_NEO_4: (static_cast<B_8266_DM_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_8266_BB_NEO_4: (static_cast<B_8266_BB_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_8266_U0_400_3: (static_cast<B_8266_U0_400_3*>(busPtr))->Show(); break; | ||||
|       case I_8266_U1_400_3: (static_cast<B_8266_U1_400_3*>(busPtr))->Show(); break; | ||||
|       case I_8266_DM_400_3: (static_cast<B_8266_DM_400_3*>(busPtr))->Show(); break; | ||||
|       case I_8266_BB_400_3: (static_cast<B_8266_BB_400_3*>(busPtr))->Show(); break; | ||||
|       case I_8266_U0_TM1_4: (static_cast<B_8266_U0_TM1_4*>(busPtr))->Show(); break; | ||||
|       case I_8266_U1_TM1_4: (static_cast<B_8266_U1_TM1_4*>(busPtr))->Show(); break; | ||||
|       case I_8266_DM_TM1_4: (static_cast<B_8266_DM_TM1_4*>(busPtr))->Show(); break; | ||||
|       case I_8266_BB_TM1_4: (static_cast<B_8266_BB_TM1_4*>(busPtr))->Show(); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_R0_NEO_3: (static_cast<B_32_R0_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R1_NEO_3: (static_cast<B_32_R1_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R2_NEO_3: (static_cast<B_32_R2_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R3_NEO_3: (static_cast<B_32_R3_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R4_NEO_3: (static_cast<B_32_R4_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R5_NEO_3: (static_cast<B_32_R5_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R6_NEO_3: (static_cast<B_32_R6_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R7_NEO_3: (static_cast<B_32_R7_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_32_I0_NEO_3: (static_cast<B_32_I0_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_32_I1_NEO_3: (static_cast<B_32_I1_NEO_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R0_NEO_4: (static_cast<B_32_R0_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R1_NEO_4: (static_cast<B_32_R1_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R2_NEO_4: (static_cast<B_32_R2_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R3_NEO_4: (static_cast<B_32_R3_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R4_NEO_4: (static_cast<B_32_R4_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R5_NEO_4: (static_cast<B_32_R5_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R6_NEO_4: (static_cast<B_32_R6_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R7_NEO_4: (static_cast<B_32_R7_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_32_I0_NEO_4: (static_cast<B_32_I0_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_32_I1_NEO_4: (static_cast<B_32_I1_NEO_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R0_400_3: (static_cast<B_32_R0_400_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R1_400_3: (static_cast<B_32_R1_400_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R2_400_3: (static_cast<B_32_R2_400_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R3_400_3: (static_cast<B_32_R3_400_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R4_400_3: (static_cast<B_32_R4_400_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R5_400_3: (static_cast<B_32_R5_400_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R6_400_3: (static_cast<B_32_R6_400_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R7_400_3: (static_cast<B_32_R7_400_3*>(busPtr))->Show(); break; | ||||
|       case I_32_I0_400_3: (static_cast<B_32_I0_400_3*>(busPtr))->Show(); break; | ||||
|       case I_32_I1_400_3: (static_cast<B_32_I1_400_3*>(busPtr))->Show(); break; | ||||
|       case I_32_R0_TM1_4: (static_cast<B_32_R0_TM1_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R1_TM1_4: (static_cast<B_32_R1_TM1_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R2_TM1_4: (static_cast<B_32_R2_TM1_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R3_TM1_4: (static_cast<B_32_R3_TM1_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R4_TM1_4: (static_cast<B_32_R4_TM1_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R5_TM1_4: (static_cast<B_32_R5_TM1_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R6_TM1_4: (static_cast<B_32_R6_TM1_4*>(busPtr))->Show(); break; | ||||
|       case I_32_R7_TM1_4: (static_cast<B_32_R7_TM1_4*>(busPtr))->Show(); break; | ||||
|       case I_32_I0_TM1_4: (static_cast<B_32_I0_TM1_4*>(busPtr))->Show(); break; | ||||
|       case I_32_I1_TM1_4: (static_cast<B_32_I1_TM1_4*>(busPtr))->Show(); break; | ||||
|     #endif | ||||
|       case I_HS_DOT_3: (static_cast<B_HS_DOT_3*>(busPtr))->Show(); break; | ||||
|       case I_SS_DOT_3: (static_cast<B_SS_DOT_3*>(busPtr))->Show(); break; | ||||
|       case I_HS_LPD_3: (static_cast<B_HS_LPD_3*>(busPtr))->Show(); break; | ||||
|       case I_SS_LPD_3: (static_cast<B_SS_LPD_3*>(busPtr))->Show(); break; | ||||
|       case I_HS_WS1_3: (static_cast<B_HS_WS1_3*>(busPtr))->Show(); break; | ||||
|       case I_SS_WS1_3: (static_cast<B_SS_WS1_3*>(busPtr))->Show(); break; | ||||
|       case I_HS_P98_3: (static_cast<B_HS_P98_3*>(busPtr))->Show(); break; | ||||
|       case I_SS_P98_3: (static_cast<B_SS_P98_3*>(busPtr))->Show(); break; | ||||
|     } | ||||
|   }; | ||||
|   static bool canShow(void* busPtr, uint8_t busType) { | ||||
|     switch (busType) { | ||||
|       case I_NONE: return true; | ||||
|     #ifdef ESP8266 | ||||
|       case I_8266_U0_NEO_3: return (static_cast<B_8266_U0_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_U1_NEO_3: return (static_cast<B_8266_U1_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_DM_NEO_3: return (static_cast<B_8266_DM_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_BB_NEO_3: return (static_cast<B_8266_BB_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_U0_NEO_4: return (static_cast<B_8266_U0_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_U1_NEO_4: return (static_cast<B_8266_U1_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_DM_NEO_4: return (static_cast<B_8266_DM_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_BB_NEO_4: return (static_cast<B_8266_BB_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_U0_400_3: return (static_cast<B_8266_U0_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_U1_400_3: return (static_cast<B_8266_U1_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_DM_400_3: return (static_cast<B_8266_DM_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_BB_400_3: return (static_cast<B_8266_BB_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_U0_TM1_4: return (static_cast<B_8266_U0_TM1_4*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_U1_TM1_4: return (static_cast<B_8266_U1_TM1_4*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_DM_TM1_4: return (static_cast<B_8266_DM_TM1_4*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_BB_TM1_4: return (static_cast<B_8266_BB_TM1_4*>(busPtr))->CanShow(); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_R0_NEO_3: return (static_cast<B_32_R0_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R1_NEO_3: return (static_cast<B_32_R1_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R2_NEO_3: return (static_cast<B_32_R2_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R3_NEO_3: return (static_cast<B_32_R3_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R4_NEO_3: return (static_cast<B_32_R4_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R5_NEO_3: return (static_cast<B_32_R5_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R6_NEO_3: return (static_cast<B_32_R6_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R7_NEO_3: return (static_cast<B_32_R7_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_I0_NEO_3: return (static_cast<B_32_I0_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_I1_NEO_3: return (static_cast<B_32_I1_NEO_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R0_NEO_4: return (static_cast<B_32_R0_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R1_NEO_4: return (static_cast<B_32_R1_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R2_NEO_4: return (static_cast<B_32_R2_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R3_NEO_4: return (static_cast<B_32_R3_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R4_NEO_4: return (static_cast<B_32_R4_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R5_NEO_4: return (static_cast<B_32_R5_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R6_NEO_4: return (static_cast<B_32_R6_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R7_NEO_4: return (static_cast<B_32_R7_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_I0_NEO_4: return (static_cast<B_32_I0_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_I1_NEO_4: return (static_cast<B_32_I1_NEO_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R0_400_3: return (static_cast<B_32_R0_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R1_400_3: return (static_cast<B_32_R1_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R2_400_3: return (static_cast<B_32_R2_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R3_400_3: return (static_cast<B_32_R3_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R4_400_3: return (static_cast<B_32_R4_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R5_400_3: return (static_cast<B_32_R5_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R6_400_3: return (static_cast<B_32_R6_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R7_400_3: return (static_cast<B_32_R7_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_I0_400_3: return (static_cast<B_32_I0_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_I1_400_3: return (static_cast<B_32_I1_400_3*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R0_TM1_4: return (static_cast<B_32_R0_TM1_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R1_TM1_4: return (static_cast<B_32_R1_TM1_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R2_TM1_4: return (static_cast<B_32_R2_TM1_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R3_TM1_4: return (static_cast<B_32_R3_TM1_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R4_TM1_4: return (static_cast<B_32_R4_TM1_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R5_TM1_4: return (static_cast<B_32_R5_TM1_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R6_TM1_4: return (static_cast<B_32_R6_TM1_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_R7_TM1_4: return (static_cast<B_32_R7_TM1_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_I0_TM1_4: return (static_cast<B_32_I0_TM1_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_I1_TM1_4: return (static_cast<B_32_I1_TM1_4*>(busPtr))->CanShow(); break; | ||||
|     #endif | ||||
|       case I_HS_DOT_3: return (static_cast<B_HS_DOT_3*>(busPtr))->CanShow(); break; | ||||
|       case I_SS_DOT_3: return (static_cast<B_SS_DOT_3*>(busPtr))->CanShow(); break; | ||||
|       case I_HS_LPD_3: return (static_cast<B_HS_LPD_3*>(busPtr))->CanShow(); break; | ||||
|       case I_SS_LPD_3: return (static_cast<B_SS_LPD_3*>(busPtr))->CanShow(); break; | ||||
|       case I_HS_WS1_3: return (static_cast<B_HS_WS1_3*>(busPtr))->CanShow(); break; | ||||
|       case I_SS_WS1_3: return (static_cast<B_SS_WS1_3*>(busPtr))->CanShow(); break; | ||||
|       case I_HS_P98_3: return (static_cast<B_HS_P98_3*>(busPtr))->CanShow(); break; | ||||
|       case I_SS_P98_3: return (static_cast<B_SS_P98_3*>(busPtr))->CanShow(); break; | ||||
|     } | ||||
|     return true; | ||||
|   }; | ||||
|   static void setPixelColor(void* busPtr, uint8_t busType, uint16_t pix, uint32_t c, uint8_t co) { | ||||
|     uint8_t r = c >> 16; | ||||
|     uint8_t g = c >> 8; | ||||
|     uint8_t b = c >> 0; | ||||
|     uint8_t w = c >> 24; | ||||
|     RgbwColor col; | ||||
|  | ||||
|     //TODO make color order override possible on a per-strip basis | ||||
|     #ifdef COLOR_ORDER_OVERRIDE | ||||
|     if (pix >= COO_MIN && pix < COO_MAX) co = COO_ORDER; | ||||
|     #endif | ||||
|  | ||||
|     //reorder channels to selected order | ||||
|     switch (co) | ||||
|     { | ||||
|       case  0: col.G = g; col.R = r; col.B = b; break; //0 = GRB, default | ||||
|       case  1: col.G = r; col.R = g; col.B = b; break; //1 = RGB, common for WS2811 | ||||
|       case  2: col.G = b; col.R = r; col.B = g; break; //2 = BRG | ||||
|       case  3: col.G = r; col.R = b; col.B = g; break; //3 = RBG | ||||
|       case  4: col.G = b; col.R = g; col.B = r; break; //4 = BGR | ||||
|       default: col.G = g; col.R = b; col.B = r; break; //5 = GBR | ||||
|     } | ||||
|     col.W = w; | ||||
|  | ||||
|     switch (busType) { | ||||
|       case I_NONE: break; | ||||
|     #ifdef ESP8266 | ||||
|       case I_8266_U0_NEO_3: (static_cast<B_8266_U0_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_8266_U1_NEO_3: (static_cast<B_8266_U1_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_8266_DM_NEO_3: (static_cast<B_8266_DM_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_8266_BB_NEO_3: (static_cast<B_8266_BB_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_8266_U0_NEO_4: (static_cast<B_8266_U0_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_8266_U1_NEO_4: (static_cast<B_8266_U1_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_8266_DM_NEO_4: (static_cast<B_8266_DM_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_8266_BB_NEO_4: (static_cast<B_8266_BB_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_8266_U0_400_3: (static_cast<B_8266_U0_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_8266_U1_400_3: (static_cast<B_8266_U1_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_8266_DM_400_3: (static_cast<B_8266_DM_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_8266_BB_400_3: (static_cast<B_8266_BB_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_8266_U0_TM1_4: (static_cast<B_8266_U0_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_8266_U1_TM1_4: (static_cast<B_8266_U1_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_8266_DM_TM1_4: (static_cast<B_8266_DM_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_8266_BB_TM1_4: (static_cast<B_8266_BB_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_R0_NEO_3: (static_cast<B_32_R0_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R1_NEO_3: (static_cast<B_32_R1_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R2_NEO_3: (static_cast<B_32_R2_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R3_NEO_3: (static_cast<B_32_R3_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R4_NEO_3: (static_cast<B_32_R4_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R5_NEO_3: (static_cast<B_32_R5_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R6_NEO_3: (static_cast<B_32_R6_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R7_NEO_3: (static_cast<B_32_R7_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_I0_NEO_3: (static_cast<B_32_I0_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_I1_NEO_3: (static_cast<B_32_I1_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R0_NEO_4: (static_cast<B_32_R0_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R1_NEO_4: (static_cast<B_32_R1_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R2_NEO_4: (static_cast<B_32_R2_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R3_NEO_4: (static_cast<B_32_R3_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R4_NEO_4: (static_cast<B_32_R4_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R5_NEO_4: (static_cast<B_32_R5_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R6_NEO_4: (static_cast<B_32_R6_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R7_NEO_4: (static_cast<B_32_R7_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_I0_NEO_4: (static_cast<B_32_I0_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_I1_NEO_4: (static_cast<B_32_I1_NEO_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R0_400_3: (static_cast<B_32_R0_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R1_400_3: (static_cast<B_32_R1_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R2_400_3: (static_cast<B_32_R2_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R3_400_3: (static_cast<B_32_R3_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R4_400_3: (static_cast<B_32_R4_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R5_400_3: (static_cast<B_32_R5_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R6_400_3: (static_cast<B_32_R6_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R7_400_3: (static_cast<B_32_R7_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_I0_400_3: (static_cast<B_32_I0_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_I1_400_3: (static_cast<B_32_I1_400_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_32_R0_TM1_4: (static_cast<B_32_R0_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R1_TM1_4: (static_cast<B_32_R1_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R2_TM1_4: (static_cast<B_32_R2_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R3_TM1_4: (static_cast<B_32_R3_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R4_TM1_4: (static_cast<B_32_R4_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R5_TM1_4: (static_cast<B_32_R5_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R6_TM1_4: (static_cast<B_32_R6_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_R7_TM1_4: (static_cast<B_32_R7_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_I0_TM1_4: (static_cast<B_32_I0_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|       case I_32_I1_TM1_4: (static_cast<B_32_I1_TM1_4*>(busPtr))->SetPixelColor(pix, col); break; | ||||
|     #endif | ||||
|       case I_HS_DOT_3: (static_cast<B_HS_DOT_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_SS_DOT_3: (static_cast<B_SS_DOT_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_HS_LPD_3: (static_cast<B_HS_LPD_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_SS_LPD_3: (static_cast<B_SS_LPD_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_HS_WS1_3: (static_cast<B_HS_WS1_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_SS_WS1_3: (static_cast<B_SS_WS1_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_HS_P98_3: (static_cast<B_HS_P98_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|       case I_SS_P98_3: (static_cast<B_SS_P98_3*>(busPtr))->SetPixelColor(pix, RgbColor(col.R,col.G,col.B)); break; | ||||
|     } | ||||
|   }; | ||||
|   static void setBrightness(void* busPtr, uint8_t busType, uint8_t b) { | ||||
|     switch (busType) { | ||||
|       case I_NONE: break; | ||||
|     #ifdef ESP8266 | ||||
|       case I_8266_U0_NEO_3: (static_cast<B_8266_U0_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_U1_NEO_3: (static_cast<B_8266_U1_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_DM_NEO_3: (static_cast<B_8266_DM_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_BB_NEO_3: (static_cast<B_8266_BB_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_U0_NEO_4: (static_cast<B_8266_U0_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_U1_NEO_4: (static_cast<B_8266_U1_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_DM_NEO_4: (static_cast<B_8266_DM_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_BB_NEO_4: (static_cast<B_8266_BB_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_U0_400_3: (static_cast<B_8266_U0_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_U1_400_3: (static_cast<B_8266_U1_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_DM_400_3: (static_cast<B_8266_DM_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_BB_400_3: (static_cast<B_8266_BB_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_U0_TM1_4: (static_cast<B_8266_U0_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_U1_TM1_4: (static_cast<B_8266_U1_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_DM_TM1_4: (static_cast<B_8266_DM_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_8266_BB_TM1_4: (static_cast<B_8266_BB_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_R0_NEO_3: (static_cast<B_32_R0_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R1_NEO_3: (static_cast<B_32_R1_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R2_NEO_3: (static_cast<B_32_R2_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R3_NEO_3: (static_cast<B_32_R3_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R4_NEO_3: (static_cast<B_32_R4_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R5_NEO_3: (static_cast<B_32_R5_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R6_NEO_3: (static_cast<B_32_R6_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R7_NEO_3: (static_cast<B_32_R7_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_I0_NEO_3: (static_cast<B_32_I0_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_I1_NEO_3: (static_cast<B_32_I1_NEO_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R0_NEO_4: (static_cast<B_32_R0_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R1_NEO_4: (static_cast<B_32_R1_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R2_NEO_4: (static_cast<B_32_R2_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R3_NEO_4: (static_cast<B_32_R3_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R4_NEO_4: (static_cast<B_32_R4_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R5_NEO_4: (static_cast<B_32_R5_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R6_NEO_4: (static_cast<B_32_R6_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R7_NEO_4: (static_cast<B_32_R7_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_I0_NEO_4: (static_cast<B_32_I0_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_I1_NEO_4: (static_cast<B_32_I1_NEO_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R0_400_3: (static_cast<B_32_R0_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R1_400_3: (static_cast<B_32_R1_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R2_400_3: (static_cast<B_32_R2_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R3_400_3: (static_cast<B_32_R3_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R4_400_3: (static_cast<B_32_R4_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R5_400_3: (static_cast<B_32_R5_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R6_400_3: (static_cast<B_32_R6_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R7_400_3: (static_cast<B_32_R7_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_I0_400_3: (static_cast<B_32_I0_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_I1_400_3: (static_cast<B_32_I1_400_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R0_TM1_4: (static_cast<B_32_R0_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R1_TM1_4: (static_cast<B_32_R1_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R2_TM1_4: (static_cast<B_32_R2_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R3_TM1_4: (static_cast<B_32_R3_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R4_TM1_4: (static_cast<B_32_R4_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R5_TM1_4: (static_cast<B_32_R5_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R6_TM1_4: (static_cast<B_32_R6_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_R7_TM1_4: (static_cast<B_32_R7_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_I0_TM1_4: (static_cast<B_32_I0_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_32_I1_TM1_4: (static_cast<B_32_I1_TM1_4*>(busPtr))->SetBrightness(b); break; | ||||
|     #endif | ||||
|       case I_HS_DOT_3: (static_cast<B_HS_DOT_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_SS_DOT_3: (static_cast<B_SS_DOT_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_HS_LPD_3: (static_cast<B_HS_LPD_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_SS_LPD_3: (static_cast<B_SS_LPD_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_HS_WS1_3: (static_cast<B_HS_WS1_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_SS_WS1_3: (static_cast<B_SS_WS1_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_HS_P98_3: (static_cast<B_HS_P98_3*>(busPtr))->SetBrightness(b); break; | ||||
|       case I_SS_P98_3: (static_cast<B_SS_P98_3*>(busPtr))->SetBrightness(b); break; | ||||
|     } | ||||
|   }; | ||||
|   static uint32_t getPixelColor(void* busPtr, uint8_t busType, uint16_t pix, uint8_t co) { | ||||
|     RgbwColor col(0,0,0,0);  | ||||
|     switch (busType) { | ||||
|       case I_NONE: break; | ||||
|     #ifdef ESP8266 | ||||
|       case I_8266_U0_NEO_3: col = (static_cast<B_8266_U0_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_U1_NEO_3: col = (static_cast<B_8266_U1_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_DM_NEO_3: col = (static_cast<B_8266_DM_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_BB_NEO_3: col = (static_cast<B_8266_BB_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_U0_NEO_4: col = (static_cast<B_8266_U0_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_U1_NEO_4: col = (static_cast<B_8266_U1_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_DM_NEO_4: col = (static_cast<B_8266_DM_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_BB_NEO_4: col = (static_cast<B_8266_BB_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_U0_400_3: col = (static_cast<B_8266_U0_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_U1_400_3: col = (static_cast<B_8266_U1_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_DM_400_3: col = (static_cast<B_8266_DM_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_BB_400_3: col = (static_cast<B_8266_BB_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_U0_TM1_4: col = (static_cast<B_8266_U0_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_U1_TM1_4: col = (static_cast<B_8266_U1_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_DM_TM1_4: col = (static_cast<B_8266_DM_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_BB_TM1_4: col = (static_cast<B_8266_BB_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_R0_NEO_3: col = (static_cast<B_32_R0_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R1_NEO_3: col = (static_cast<B_32_R1_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R2_NEO_3: col = (static_cast<B_32_R2_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R3_NEO_3: col = (static_cast<B_32_R3_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R4_NEO_3: col = (static_cast<B_32_R4_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R5_NEO_3: col = (static_cast<B_32_R5_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R6_NEO_3: col = (static_cast<B_32_R6_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R7_NEO_3: col = (static_cast<B_32_R7_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_I0_NEO_3: col = (static_cast<B_32_I0_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_I1_NEO_3: col = (static_cast<B_32_I1_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R0_NEO_4: col = (static_cast<B_32_R0_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R1_NEO_4: col = (static_cast<B_32_R1_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R2_NEO_4: col = (static_cast<B_32_R2_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R3_NEO_4: col = (static_cast<B_32_R3_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R4_NEO_4: col = (static_cast<B_32_R4_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R5_NEO_4: col = (static_cast<B_32_R5_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R6_NEO_4: col = (static_cast<B_32_R6_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R7_NEO_4: col = (static_cast<B_32_R7_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_I0_NEO_4: col = (static_cast<B_32_I0_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_I1_NEO_4: col = (static_cast<B_32_I1_NEO_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R0_400_3: col = (static_cast<B_32_R0_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R1_400_3: col = (static_cast<B_32_R1_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R2_400_3: col = (static_cast<B_32_R2_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R3_400_3: col = (static_cast<B_32_R3_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R4_400_3: col = (static_cast<B_32_R4_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R5_400_3: col = (static_cast<B_32_R5_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R6_400_3: col = (static_cast<B_32_R6_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R7_400_3: col = (static_cast<B_32_R7_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_I0_400_3: col = (static_cast<B_32_I0_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_I1_400_3: col = (static_cast<B_32_I1_400_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R0_TM1_4: col = (static_cast<B_32_R0_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R1_TM1_4: col = (static_cast<B_32_R1_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R2_TM1_4: col = (static_cast<B_32_R2_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R3_TM1_4: col = (static_cast<B_32_R3_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R4_TM1_4: col = (static_cast<B_32_R4_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R5_TM1_4: col = (static_cast<B_32_R5_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R6_TM1_4: col = (static_cast<B_32_R6_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_R7_TM1_4: col = (static_cast<B_32_R7_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_I0_TM1_4: col = (static_cast<B_32_I0_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_I1_TM1_4: col = (static_cast<B_32_I1_TM1_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|     #endif | ||||
|       case I_HS_DOT_3: col = (static_cast<B_HS_DOT_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_SS_DOT_3: col = (static_cast<B_SS_DOT_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_HS_LPD_3: col = (static_cast<B_HS_LPD_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_SS_LPD_3: col = (static_cast<B_SS_LPD_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_HS_WS1_3: col = (static_cast<B_HS_WS1_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_SS_WS1_3: col = (static_cast<B_SS_WS1_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_HS_P98_3: col = (static_cast<B_HS_P98_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_SS_P98_3: col = (static_cast<B_SS_P98_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|     } | ||||
|      | ||||
|     #ifdef COLOR_ORDER_OVERRIDE | ||||
|     if (pix >= COO_MIN && pix < COO_MAX) co = COO_ORDER; | ||||
|     #endif | ||||
|  | ||||
|     switch (co) | ||||
|     { | ||||
|       //                    W               G              R               B | ||||
|       case  0: return ((col.W << 24) | (col.G << 8) | (col.R << 16) | (col.B)); //0 = GRB, default | ||||
|       case  1: return ((col.W << 24) | (col.R << 8) | (col.G << 16) | (col.B)); //1 = RGB, common for WS2811 | ||||
|       case  2: return ((col.W << 24) | (col.B << 8) | (col.R << 16) | (col.G)); //2 = BRG | ||||
|       case  3: return ((col.W << 24) | (col.B << 8) | (col.G << 16) | (col.R)); //3 = RBG | ||||
|       case  4: return ((col.W << 24) | (col.R << 8) | (col.B << 16) | (col.G)); //4 = BGR | ||||
|       case  5: return ((col.W << 24) | (col.G << 8) | (col.B << 16) | (col.R)); //5 = GBR | ||||
|     } | ||||
|     return 0; | ||||
|   } | ||||
|  | ||||
|   static void cleanup(void* busPtr, uint8_t busType) { | ||||
|     if (busPtr == nullptr) return; | ||||
|     switch (busType) { | ||||
|       case I_NONE: break; | ||||
|     #ifdef ESP8266 | ||||
|       case I_8266_U0_NEO_3: delete (static_cast<B_8266_U0_NEO_3*>(busPtr)); break; | ||||
|       case I_8266_U1_NEO_3: delete (static_cast<B_8266_U1_NEO_3*>(busPtr)); break; | ||||
|       case I_8266_DM_NEO_3: delete (static_cast<B_8266_DM_NEO_3*>(busPtr)); break; | ||||
|       case I_8266_BB_NEO_3: delete (static_cast<B_8266_BB_NEO_3*>(busPtr)); break; | ||||
|       case I_8266_U0_NEO_4: delete (static_cast<B_8266_U0_NEO_4*>(busPtr)); break; | ||||
|       case I_8266_U1_NEO_4: delete (static_cast<B_8266_U1_NEO_4*>(busPtr)); break; | ||||
|       case I_8266_DM_NEO_4: delete (static_cast<B_8266_DM_NEO_4*>(busPtr)); break; | ||||
|       case I_8266_BB_NEO_4: delete (static_cast<B_8266_BB_NEO_4*>(busPtr)); break; | ||||
|       case I_8266_U0_400_3: delete (static_cast<B_8266_U0_400_3*>(busPtr)); break; | ||||
|       case I_8266_U1_400_3: delete (static_cast<B_8266_U1_400_3*>(busPtr)); break; | ||||
|       case I_8266_DM_400_3: delete (static_cast<B_8266_DM_400_3*>(busPtr)); break; | ||||
|       case I_8266_BB_400_3: delete (static_cast<B_8266_BB_400_3*>(busPtr)); break; | ||||
|       case I_8266_U0_TM1_4: delete (static_cast<B_8266_U0_TM1_4*>(busPtr)); break; | ||||
|       case I_8266_U1_TM1_4: delete (static_cast<B_8266_U1_TM1_4*>(busPtr)); break; | ||||
|       case I_8266_DM_TM1_4: delete (static_cast<B_8266_DM_TM1_4*>(busPtr)); break; | ||||
|       case I_8266_BB_TM1_4: delete (static_cast<B_8266_BB_TM1_4*>(busPtr)); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_R0_NEO_3: delete (static_cast<B_32_R0_NEO_3*>(busPtr)); break; | ||||
|       case I_32_R1_NEO_3: delete (static_cast<B_32_R1_NEO_3*>(busPtr)); break; | ||||
|       case I_32_R2_NEO_3: delete (static_cast<B_32_R2_NEO_3*>(busPtr)); break; | ||||
|       case I_32_R3_NEO_3: delete (static_cast<B_32_R3_NEO_3*>(busPtr)); break; | ||||
|       case I_32_R4_NEO_3: delete (static_cast<B_32_R4_NEO_3*>(busPtr)); break; | ||||
|       case I_32_R5_NEO_3: delete (static_cast<B_32_R5_NEO_3*>(busPtr)); break; | ||||
|       case I_32_R6_NEO_3: delete (static_cast<B_32_R6_NEO_3*>(busPtr)); break; | ||||
|       case I_32_R7_NEO_3: delete (static_cast<B_32_R7_NEO_3*>(busPtr)); break; | ||||
|       case I_32_I0_NEO_3: delete (static_cast<B_32_I0_NEO_3*>(busPtr)); break; | ||||
|       case I_32_I1_NEO_3: delete (static_cast<B_32_I1_NEO_3*>(busPtr)); break; | ||||
|       case I_32_R0_NEO_4: delete (static_cast<B_32_R0_NEO_4*>(busPtr)); break; | ||||
|       case I_32_R1_NEO_4: delete (static_cast<B_32_R1_NEO_4*>(busPtr)); break; | ||||
|       case I_32_R2_NEO_4: delete (static_cast<B_32_R2_NEO_4*>(busPtr)); break; | ||||
|       case I_32_R3_NEO_4: delete (static_cast<B_32_R3_NEO_4*>(busPtr)); break; | ||||
|       case I_32_R4_NEO_4: delete (static_cast<B_32_R4_NEO_4*>(busPtr)); break; | ||||
|       case I_32_R5_NEO_4: delete (static_cast<B_32_R5_NEO_4*>(busPtr)); break; | ||||
|       case I_32_R6_NEO_4: delete (static_cast<B_32_R6_NEO_4*>(busPtr)); break; | ||||
|       case I_32_R7_NEO_4: delete (static_cast<B_32_R7_NEO_4*>(busPtr)); break; | ||||
|       case I_32_I0_NEO_4: delete (static_cast<B_32_I0_NEO_4*>(busPtr)); break; | ||||
|       case I_32_I1_NEO_4: delete (static_cast<B_32_I1_NEO_4*>(busPtr)); break; | ||||
|       case I_32_R0_400_3: delete (static_cast<B_32_R0_400_3*>(busPtr)); break; | ||||
|       case I_32_R1_400_3: delete (static_cast<B_32_R1_400_3*>(busPtr)); break; | ||||
|       case I_32_R2_400_3: delete (static_cast<B_32_R2_400_3*>(busPtr)); break; | ||||
|       case I_32_R3_400_3: delete (static_cast<B_32_R3_400_3*>(busPtr)); break; | ||||
|       case I_32_R4_400_3: delete (static_cast<B_32_R4_400_3*>(busPtr)); break; | ||||
|       case I_32_R5_400_3: delete (static_cast<B_32_R5_400_3*>(busPtr)); break; | ||||
|       case I_32_R6_400_3: delete (static_cast<B_32_R6_400_3*>(busPtr)); break; | ||||
|       case I_32_R7_400_3: delete (static_cast<B_32_R7_400_3*>(busPtr)); break; | ||||
|       case I_32_I0_400_3: delete (static_cast<B_32_I0_400_3*>(busPtr)); break; | ||||
|       case I_32_I1_400_3: delete (static_cast<B_32_I1_400_3*>(busPtr)); break; | ||||
|       case I_32_R0_TM1_4: delete (static_cast<B_32_R0_TM1_4*>(busPtr)); break; | ||||
|       case I_32_R1_TM1_4: delete (static_cast<B_32_R1_TM1_4*>(busPtr)); break; | ||||
|       case I_32_R2_TM1_4: delete (static_cast<B_32_R2_TM1_4*>(busPtr)); break; | ||||
|       case I_32_R3_TM1_4: delete (static_cast<B_32_R3_TM1_4*>(busPtr)); break; | ||||
|       case I_32_R4_TM1_4: delete (static_cast<B_32_R4_TM1_4*>(busPtr)); break; | ||||
|       case I_32_R5_TM1_4: delete (static_cast<B_32_R5_TM1_4*>(busPtr)); break; | ||||
|       case I_32_R6_TM1_4: delete (static_cast<B_32_R6_TM1_4*>(busPtr)); break; | ||||
|       case I_32_R7_TM1_4: delete (static_cast<B_32_R7_TM1_4*>(busPtr)); break; | ||||
|       case I_32_I0_TM1_4: delete (static_cast<B_32_I0_TM1_4*>(busPtr)); break; | ||||
|       case I_32_I1_TM1_4: delete (static_cast<B_32_I1_TM1_4*>(busPtr)); break; | ||||
|     #endif | ||||
|       case I_HS_DOT_3: delete (static_cast<B_HS_DOT_3*>(busPtr)); break; | ||||
|       case I_SS_DOT_3: delete (static_cast<B_SS_DOT_3*>(busPtr)); break; | ||||
|       case I_HS_LPD_3: delete (static_cast<B_HS_LPD_3*>(busPtr)); break; | ||||
|       case I_SS_LPD_3: delete (static_cast<B_SS_LPD_3*>(busPtr)); break; | ||||
|       case I_HS_WS1_3: delete (static_cast<B_HS_WS1_3*>(busPtr)); break; | ||||
|       case I_SS_WS1_3: delete (static_cast<B_SS_WS1_3*>(busPtr)); break; | ||||
|       case I_HS_P98_3: delete (static_cast<B_HS_P98_3*>(busPtr)); break; | ||||
|       case I_SS_P98_3: delete (static_cast<B_SS_P98_3*>(busPtr)); break; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //gives back the internal type index (I_XX_XXX_X above) for the input  | ||||
|   static uint8_t getI(uint8_t busType, uint8_t* pins, uint8_t num = 0) { | ||||
|     if (!IS_DIGITAL(busType)) return I_NONE; | ||||
|     if (IS_2PIN(busType)) { //SPI LED chips | ||||
|       bool isHSPI = false; | ||||
|       #ifdef ESP8266 | ||||
|       if (pins[0] == P_8266_HS_MOSI && pins[1] == P_8266_HS_CLK) isHSPI = true; | ||||
|       #else | ||||
|         if(!num) isHSPI = true; // temporary hack to limit use of hardware SPI to a single SPI peripheral: only allow ESP32 hardware serial on segment 0 | ||||
|       #endif | ||||
|       uint8_t t = I_NONE; | ||||
|       switch (busType) { | ||||
|         case TYPE_APA102:  t = I_SS_DOT_3; break; | ||||
|         case TYPE_LPD8806: t = I_SS_LPD_3; break; | ||||
|         case TYPE_WS2801:  t = I_SS_WS1_3; break; | ||||
|         case TYPE_P9813:   t = I_SS_P98_3; break; | ||||
|         default: t=I_NONE; | ||||
|       } | ||||
|       if (t > I_NONE && isHSPI) t--; //hardware SPI has one smaller ID than software | ||||
|       return t; | ||||
|     } else { | ||||
|       #ifdef ESP8266 | ||||
|       uint8_t offset = pins[0] -1; //for driver: 0 = uart0, 1 = uart1, 2 = dma, 3 = bitbang | ||||
|       if (offset > 3) offset = 3; | ||||
|       switch (busType) { | ||||
|         case TYPE_WS2812_RGB: | ||||
|         case TYPE_WS2812_WWA: | ||||
|           return I_8266_U0_NEO_3 + offset; | ||||
|         case TYPE_SK6812_RGBW: | ||||
|           return I_8266_U0_NEO_4 + offset; | ||||
|         case TYPE_WS2811_400KHZ: | ||||
|           return I_8266_U0_400_3 + offset; | ||||
|       } | ||||
|       #else //ESP32 | ||||
|       uint8_t offset = num; //RMT bus # == bus index in BusManager | ||||
|       if (offset > 9) return I_NONE; | ||||
|       switch (busType) { | ||||
|         case TYPE_WS2812_RGB: | ||||
|         case TYPE_WS2812_WWA: | ||||
|           return I_32_R0_NEO_3 + offset; | ||||
|         case TYPE_SK6812_RGBW: | ||||
|           return I_32_R0_NEO_4 + offset; | ||||
|         case TYPE_WS2811_400KHZ: | ||||
|           return I_32_R0_400_3 + offset; | ||||
|       } | ||||
|       #endif | ||||
|     } | ||||
|     return I_NONE; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| #endif | ||||
| @@ -1,3 +1,5 @@ | ||||
| #include "wled.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * Physical IO | ||||
|  */ | ||||
| @@ -7,19 +9,27 @@ void shortPressAction() | ||||
|   if (!macroButton) | ||||
|   { | ||||
|     toggleOnOff(); | ||||
|     colorUpdated(2); | ||||
|     colorUpdated(NOTIFIER_CALL_MODE_BUTTON); | ||||
|   } else { | ||||
|     applyMacro(macroButton); | ||||
|     applyPreset(macroButton); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| bool isButtonPressed() | ||||
| { | ||||
|   if (btnPin>=0 && digitalRead(btnPin) == LOW) return true; | ||||
|   #ifdef TOUCHPIN | ||||
|     if (touchRead(TOUCHPIN) <= TOUCH_THRESHOLD) return true; | ||||
|   #endif | ||||
|   return false; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void handleButton() | ||||
| { | ||||
| #ifdef BTNPIN | ||||
|   if (!buttonEnabled) return; | ||||
|    | ||||
|   if (digitalRead(BTNPIN) == LOW) //pressed
 | ||||
|   if (btnPin<0 || !buttonEnabled) return; | ||||
| 
 | ||||
|   if (isButtonPressed()) //pressed
 | ||||
|   { | ||||
|     if (!buttonPressedBefore) buttonPressedTime = millis(); | ||||
|     buttonPressedBefore = true; | ||||
| @@ -28,14 +38,14 @@ void handleButton() | ||||
|     { | ||||
|       if (!buttonLongPressed)  | ||||
|       { | ||||
|         if (macroLongPress) {applyMacro(macroLongPress);} | ||||
|         if (macroLongPress) {applyPreset(macroLongPress);} | ||||
|         else _setRandomColor(false,true); | ||||
| 
 | ||||
|         buttonLongPressed = true; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   else if (digitalRead(BTNPIN) == HIGH && buttonPressedBefore) //released
 | ||||
|   else if (!isButtonPressed() && buttonPressedBefore) //released
 | ||||
|   { | ||||
|     long dur = millis() - buttonPressedTime; | ||||
|     if (dur < 50) {buttonPressedBefore = false; return;} //too short "press", debounce
 | ||||
| @@ -44,12 +54,12 @@ void handleButton() | ||||
| 
 | ||||
|     if (dur > 6000) //long press
 | ||||
|     { | ||||
|       initAP(true); | ||||
|       WLED::instance().initAP(true); | ||||
|     } | ||||
|     else if (!buttonLongPressed) { //short press
 | ||||
|       if (macroDoublePress) | ||||
|       { | ||||
|         if (doublePress) applyMacro(macroDoublePress); | ||||
|         if (doublePress) applyPreset(macroDoublePress); | ||||
|         else buttonWaitTime = millis(); | ||||
|       } else shortPressAction(); | ||||
|     } | ||||
| @@ -62,7 +72,6 @@ void handleButton() | ||||
|     buttonWaitTime = 0; | ||||
|     shortPressAction(); | ||||
|   } | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| void handleIO() | ||||
| @@ -74,46 +83,26 @@ void handleIO() | ||||
|   { | ||||
|     lastOnTime = millis(); | ||||
|     if (offMode) | ||||
|     {  | ||||
|       #if RLYPIN >= 0 | ||||
|        digitalWrite(RLYPIN, RLYMDE); | ||||
|       #endif | ||||
|     { | ||||
|       if (rlyPin>=0) { | ||||
|         pinMode(rlyPin, OUTPUT); | ||||
|         digitalWrite(rlyPin, rlyMde); | ||||
|       } | ||||
|       offMode = false; | ||||
|     } | ||||
|   } else if (millis() - lastOnTime > 600) | ||||
|   { | ||||
|     #if RLYPIN >= 0 | ||||
|      if (!offMode) digitalWrite(RLYPIN, !RLYMDE); | ||||
|     #endif | ||||
|     if (!offMode) { | ||||
|       #ifdef ESP8266 | ||||
|       //turn off built-in LED if strip is turned off
 | ||||
|       pinMode(LED_BUILTIN, OUTPUT); | ||||
|       digitalWrite(LED_BUILTIN, HIGH); | ||||
|       #endif | ||||
|       if (rlyPin>=0) { | ||||
|         pinMode(rlyPin, OUTPUT); | ||||
|         digitalWrite(rlyPin, !rlyMde); | ||||
|       } | ||||
|     } | ||||
|     offMode = true; | ||||
|   } | ||||
| 
 | ||||
|   #if AUXPIN >= 0 | ||||
|   //output
 | ||||
|   if (auxActive || auxActiveBefore) | ||||
|   { | ||||
|     if (!auxActiveBefore) | ||||
|     { | ||||
|       auxActiveBefore = true; | ||||
|       switch (auxTriggeredState) | ||||
|       { | ||||
|         case 0: pinMode(AUXPIN, INPUT); break; | ||||
|         case 1: pinMode(AUXPIN, OUTPUT); digitalWrite(AUXPIN, HIGH); break; | ||||
|         case 2: pinMode(AUXPIN, OUTPUT); digitalWrite(AUXPIN, LOW); break; | ||||
|       } | ||||
|       auxStartTime = millis(); | ||||
|     } | ||||
|     if ((millis() - auxStartTime > auxTime*1000 && auxTime != 255) || !auxActive) | ||||
|     { | ||||
|       auxActive = false; | ||||
|       auxActiveBefore = false; | ||||
|       switch (auxDefaultState) | ||||
|       { | ||||
|         case 0: pinMode(AUXPIN, INPUT); break; | ||||
|         case 1: pinMode(AUXPIN, OUTPUT); digitalWrite(AUXPIN, HIGH); break; | ||||
|         case 2: pinMode(AUXPIN, OUTPUT); digitalWrite(AUXPIN, LOW); break; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   #endif | ||||
| } | ||||
							
								
								
									
										746
									
								
								wled00/cfg.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										746
									
								
								wled00/cfg.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,746 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * Serializes and parses the cfg.json and wsec.json settings files, stored in internal FS. | ||||
|  * The structure of the JSON is not to be considered an official API and may change without notice. | ||||
|  */ | ||||
|  | ||||
| //simple macro for ArduinoJSON's or syntax | ||||
| #define CJSON(a,b) a = b | a | ||||
|  | ||||
| void getStringFromJson(char* dest, const char* src, size_t len) { | ||||
|   if (src != nullptr) strlcpy(dest, src, len); | ||||
| } | ||||
|  | ||||
| void deserializeConfig() { | ||||
|   bool fromeep = false; | ||||
|   bool success = deserializeConfigSec(); | ||||
|   if (!success) { //if file does not exist, try reading from EEPROM | ||||
|     deEEPSettings(); | ||||
|     fromeep = true; | ||||
|   } | ||||
|  | ||||
|   DynamicJsonDocument doc(JSON_BUFFER_SIZE); | ||||
|  | ||||
|   DEBUG_PRINTLN(F("Reading settings from /cfg.json...")); | ||||
|  | ||||
|   success = readObjectFromFile("/cfg.json", nullptr, &doc); | ||||
|   if (!success) { //if file does not exist, try reading from EEPROM | ||||
|     if (!fromeep) deEEPSettings(); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   //int rev_major = doc["rev"][0]; // 1 | ||||
|   //int rev_minor = doc["rev"][1]; // 0 | ||||
|  | ||||
|   //long vid = doc[F("vid")]; // 2010020 | ||||
|  | ||||
|   JsonObject id = doc["id"]; | ||||
|   getStringFromJson(cmDNS, id[F("mdns")], 33); | ||||
|   getStringFromJson(serverDescription, id[F("name")], 33); | ||||
|   getStringFromJson(alexaInvocationName, id[F("inv")], 33); | ||||
|  | ||||
|   JsonObject nw_ins_0 = doc["nw"][F("ins")][0]; | ||||
|   getStringFromJson(clientSSID, nw_ins_0[F("ssid")], 33); | ||||
|   //int nw_ins_0_pskl = nw_ins_0[F("pskl")]; | ||||
|   //The WiFi PSK is normally not contained in the regular file for security reasons. | ||||
|   //If it is present however, we will use it | ||||
|   getStringFromJson(clientPass, nw_ins_0["psk"], 65); | ||||
|  | ||||
|   JsonArray nw_ins_0_ip = nw_ins_0["ip"]; | ||||
|   JsonArray nw_ins_0_gw = nw_ins_0["gw"]; | ||||
|   JsonArray nw_ins_0_sn = nw_ins_0["sn"]; | ||||
|  | ||||
|   for (byte i = 0; i < 4; i++) { | ||||
|     CJSON(staticIP[i], nw_ins_0_ip[i]); | ||||
|     CJSON(staticGateway[i], nw_ins_0_gw[i]); | ||||
|     CJSON(staticSubnet[i], nw_ins_0_sn[i]); | ||||
|   } | ||||
|  | ||||
|   JsonObject ap = doc["ap"]; | ||||
|   getStringFromJson(apSSID, ap[F("ssid")], 33); | ||||
|   getStringFromJson(apPass, ap["psk"] , 65); //normally not present due to security | ||||
|   //int ap_pskl = ap[F("pskl")]; | ||||
|  | ||||
|   CJSON(apChannel, ap[F("chan")]); | ||||
|   if (apChannel > 13 || apChannel < 1) apChannel = 1; | ||||
|  | ||||
|   CJSON(apHide, ap[F("hide")]); | ||||
|   if (apHide > 1) apHide = 1; | ||||
|  | ||||
|   CJSON(apBehavior, ap[F("behav")]); | ||||
|    | ||||
|   #ifdef WLED_USE_ETHERNET | ||||
|   JsonObject ethernet = doc[F("eth")]; | ||||
|   CJSON(ethernetType, ethernet["type"]); | ||||
|   #endif | ||||
|  | ||||
|   /* | ||||
|   JsonArray ap_ip = ap["ip"]; | ||||
|   for (byte i = 0; i < 4; i++) { | ||||
|     apIP[i] = ap_ip; | ||||
|   }*/ | ||||
|  | ||||
|   noWifiSleep = doc[F("wifi")][F("sleep")] | !noWifiSleep; // inverted | ||||
|   noWifiSleep = !noWifiSleep; | ||||
|   //int wifi_phy = doc[F("wifi")][F("phy")]; //force phy mode n? | ||||
|  | ||||
|   JsonObject hw = doc[F("hw")]; | ||||
|  | ||||
|   // initialize LED pins and lengths prior to other HW | ||||
|   JsonObject hw_led = hw[F("led")]; | ||||
|  | ||||
|   CJSON(ledCount, hw_led[F("total")]); | ||||
|   if (ledCount > MAX_LEDS) ledCount = MAX_LEDS; | ||||
|  | ||||
|   CJSON(strip.ablMilliampsMax, hw_led[F("maxpwr")]); | ||||
|   CJSON(strip.milliampsPerLed, hw_led[F("ledma")]); | ||||
|   CJSON(strip.rgbwMode, hw_led[F("rgbwm")]); | ||||
|  | ||||
|   JsonArray ins = hw_led["ins"]; | ||||
|   uint8_t s = 0; //bus iterator | ||||
|   strip.isRgbw = false; | ||||
|   busses.removeAll(); | ||||
|   uint32_t mem = 0; | ||||
|   for (JsonObject elm : ins) { | ||||
|     if (s >= WLED_MAX_BUSSES) break; | ||||
|     uint8_t pins[5] = {255, 255, 255, 255, 255}; | ||||
|     JsonArray pinArr = elm[F("pin")]; | ||||
|     if (pinArr.size() == 0) continue; | ||||
|     pins[0] = pinArr[0]; | ||||
|     uint8_t i = 0; | ||||
|     for (int p : pinArr) { | ||||
|       pins[i] = p; | ||||
|       i++; | ||||
|       if (i>4) break; | ||||
|     } | ||||
|  | ||||
|     uint16_t length = elm[F("len")]; | ||||
|     if (length==0) continue; | ||||
|     uint8_t colorOrder = (int)elm[F("order")]; | ||||
|     //only use skip from the first strip (this shouldn't have been in ins obj. but remains here for compatibility) | ||||
|     if (s==0) skipFirstLed = elm[F("skip")]; | ||||
|     uint16_t start = elm[F("start")] | 0; | ||||
|     if (start >= ledCount) continue; | ||||
|     //limit length of strip if it would exceed total configured LEDs | ||||
|     if (start + length > ledCount) length = ledCount - start; | ||||
|     uint8_t ledType = elm["type"] | TYPE_WS2812_RGB; | ||||
|     bool reversed = elm["rev"]; | ||||
|     //RGBW mode is enabled if at least one of the strips is RGBW | ||||
|     strip.isRgbw = (strip.isRgbw || BusManager::isRgbw(ledType)); | ||||
|     s++; | ||||
|     BusConfig bc = BusConfig(ledType, pins, start, length, colorOrder, reversed); | ||||
|     mem += busses.memUsage(bc); | ||||
|     if (mem <= MAX_LED_MEMORY) busses.add(bc); | ||||
|   } | ||||
|   strip.finalizeInit(ledCount, skipFirstLed); | ||||
|   if (hw_led["rev"]) busses.getBus(0)->reversed = true; //set 0.11 global reversed setting for first bus | ||||
|  | ||||
|   JsonObject hw_btn_ins_0 = hw[F("btn")][F("ins")][0]; | ||||
|   CJSON(buttonEnabled, hw_btn_ins_0["type"]); | ||||
|   int hw_btn_pin = hw_btn_ins_0[F("pin")][0]; | ||||
|   if (pinManager.allocatePin(hw_btn_pin,false)) { | ||||
|     btnPin = hw_btn_pin; | ||||
|     pinMode(btnPin, INPUT_PULLUP); | ||||
|   } else { | ||||
|     btnPin = -1; | ||||
|   } | ||||
|  | ||||
|   JsonArray hw_btn_ins_0_macros = hw_btn_ins_0[F("macros")]; | ||||
|   CJSON(macroButton, hw_btn_ins_0_macros[0]); | ||||
|   CJSON(macroLongPress,hw_btn_ins_0_macros[1]); | ||||
|   CJSON(macroDoublePress, hw_btn_ins_0_macros[2]); | ||||
|  | ||||
|   //int hw_btn_ins_0_type = hw_btn_ins_0["type"]; // 0 | ||||
|  | ||||
|   #ifndef WLED_DISABLE_INFRARED | ||||
|   int hw_ir_pin = hw["ir"]["pin"] | -1; // 4 | ||||
|   if (pinManager.allocatePin(hw_ir_pin,false)) { | ||||
|     irPin = hw_ir_pin; | ||||
|   } else { | ||||
|     irPin = -1; | ||||
|   } | ||||
|   #endif | ||||
|   CJSON(irEnabled, hw["ir"]["type"]); | ||||
|  | ||||
|   JsonObject relay = hw[F("relay")]; | ||||
|  | ||||
|   int hw_relay_pin = relay["pin"]; | ||||
|   if (pinManager.allocatePin(hw_relay_pin,true)) { | ||||
|     rlyPin = hw_relay_pin; | ||||
|     pinMode(rlyPin, OUTPUT); | ||||
|   } else { | ||||
|     rlyPin = -1; | ||||
|   } | ||||
|   if (relay.containsKey("rev")) { | ||||
|     rlyMde = !relay["rev"]; | ||||
|   } | ||||
|  | ||||
|   //int hw_status_pin = hw[F("status")][F("pin")]; // -1 | ||||
|  | ||||
|   JsonObject light = doc[F("light")]; | ||||
|   CJSON(briMultiplier, light[F("scale-bri")]); | ||||
|   CJSON(strip.paletteBlend, light[F("pal-mode")]); | ||||
|  | ||||
|   float light_gc_bri = light[F("gc")]["bri"]; | ||||
|   float light_gc_col = light[F("gc")]["col"]; // 2.8 | ||||
|   if (light_gc_bri > 1.5) strip.gammaCorrectBri = true; | ||||
|   else if (light_gc_bri > 0.5) strip.gammaCorrectBri = false; | ||||
|   if (light_gc_col > 1.5) strip.gammaCorrectCol = true; | ||||
|   else if (light_gc_col > 0.5) strip.gammaCorrectCol = false; | ||||
|  | ||||
|   JsonObject light_tr = light[F("tr")]; | ||||
|   CJSON(fadeTransition, light_tr[F("mode")]); | ||||
|   int tdd = light_tr[F("dur")] | -1; | ||||
|   if (tdd >= 0) transitionDelayDefault = tdd * 100; | ||||
|   CJSON(strip.paletteFade, light_tr[F("pal")]); | ||||
|  | ||||
|   JsonObject light_nl = light["nl"]; | ||||
|   CJSON(nightlightMode, light_nl[F("mode")]); | ||||
|   CJSON(nightlightDelayMinsDefault, light_nl[F("dur")]); | ||||
|   nightlightDelayMins = nightlightDelayMinsDefault; | ||||
|  | ||||
|   CJSON(nightlightTargetBri, light_nl[F("tbri")]); | ||||
|   CJSON(macroNl, light_nl[F("macro")]); | ||||
|  | ||||
|   JsonObject def = doc[F("def")]; | ||||
|   CJSON(bootPreset, def[F("ps")]); | ||||
|   CJSON(turnOnAtBoot, def["on"]); // true | ||||
|   CJSON(briS, def["bri"]); // 128 | ||||
|  | ||||
|   JsonObject def_cy = def[F("cy")]; | ||||
|   CJSON(presetCyclingEnabled, def_cy["on"]); | ||||
|  | ||||
|   CJSON(presetCycleMin, def_cy[F("range")][0]); | ||||
|   CJSON(presetCycleMax, def_cy[F("range")][1]); | ||||
|  | ||||
|   tdd = def_cy[F("dur")] | -1; | ||||
|   if (tdd > 0) presetCycleTime = tdd; | ||||
|  | ||||
|   JsonObject interfaces = doc["if"]; | ||||
|  | ||||
|   JsonObject if_sync = interfaces[F("sync")]; | ||||
|   CJSON(udpPort, if_sync[F("port0")]); // 21324 | ||||
|   CJSON(udpPort2, if_sync[F("port1")]); // 65506 | ||||
|  | ||||
|   JsonObject if_sync_recv = if_sync["recv"]; | ||||
|   CJSON(receiveNotificationBrightness, if_sync_recv["bri"]); | ||||
|   CJSON(receiveNotificationColor, if_sync_recv["col"]); | ||||
|   CJSON(receiveNotificationEffects, if_sync_recv[F("fx")]); | ||||
|   receiveNotifications = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects); | ||||
|  | ||||
|   JsonObject if_sync_send = if_sync["send"]; | ||||
|   CJSON(notifyDirectDefault, if_sync_send[F("dir")]); | ||||
|   notifyDirect = notifyDirectDefault; | ||||
|   CJSON(notifyButton, if_sync_send[F("btn")]); | ||||
|   CJSON(notifyAlexa, if_sync_send[F("va")]); | ||||
|   CJSON(notifyHue, if_sync_send[F("hue")]); | ||||
|   CJSON(notifyMacro, if_sync_send[F("macro")]); | ||||
|   CJSON(notifyTwice, if_sync_send[F("twice")]); | ||||
|  | ||||
|   JsonObject if_nodes = interfaces["nodes"]; | ||||
|   CJSON(nodeListEnabled, if_nodes[F("list")]); | ||||
|   CJSON(nodeBroadcastEnabled, if_nodes[F("bcast")]); | ||||
|  | ||||
|   JsonObject if_live = interfaces["live"]; | ||||
|   CJSON(receiveDirect, if_live["en"]); | ||||
|   CJSON(e131Port, if_live["port"]); // 5568 | ||||
|   CJSON(e131Multicast, if_live[F("mc")]); | ||||
|  | ||||
|   JsonObject if_live_dmx = if_live[F("dmx")]; | ||||
|   CJSON(e131Universe, if_live_dmx[F("uni")]); | ||||
|   CJSON(e131SkipOutOfSequence, if_live_dmx[F("seqskip")]); | ||||
|   CJSON(DMXAddress, if_live_dmx[F("addr")]); | ||||
|   CJSON(DMXMode, if_live_dmx[F("mode")]); | ||||
|  | ||||
|   tdd = if_live[F("timeout")] | -1; | ||||
|   if (tdd >= 0) realtimeTimeoutMs = tdd * 100; | ||||
|   CJSON(arlsForceMaxBri, if_live[F("maxbri")]); | ||||
|   CJSON(arlsDisableGammaCorrection, if_live[F("no-gc")]); // false | ||||
|   CJSON(arlsOffset, if_live[F("offset")]); // 0 | ||||
|  | ||||
|   CJSON(alexaEnabled, interfaces[F("va")][F("alexa")]); // false | ||||
|  | ||||
|   CJSON(macroAlexaOn, interfaces[F("va")][F("macros")][0]); | ||||
|   CJSON(macroAlexaOff, interfaces[F("va")][F("macros")][1]); | ||||
|  | ||||
|   const char* apikey = interfaces["blynk"][F("token")] | "Hidden"; | ||||
|   tdd = strnlen(apikey, 36); | ||||
|   if (tdd > 20 || tdd == 0) | ||||
|     getStringFromJson(blynkApiKey, apikey, 36); //normally not present due to security | ||||
|  | ||||
|   JsonObject if_blynk = interfaces["blynk"]; | ||||
|   getStringFromJson(blynkHost, if_blynk[F("host")], 33); | ||||
|   CJSON(blynkPort, if_blynk["port"]); | ||||
|  | ||||
|   JsonObject if_mqtt = interfaces["mqtt"]; | ||||
|   CJSON(mqttEnabled, if_mqtt["en"]); | ||||
|   getStringFromJson(mqttServer, if_mqtt[F("broker")], 33); | ||||
|   CJSON(mqttPort, if_mqtt["port"]); // 1883 | ||||
|   getStringFromJson(mqttUser, if_mqtt[F("user")], 41); | ||||
|   getStringFromJson(mqttPass, if_mqtt["psk"], 41); //normally not present due to security | ||||
|   getStringFromJson(mqttClientID, if_mqtt[F("cid")], 41); | ||||
|  | ||||
|   getStringFromJson(mqttDeviceTopic, if_mqtt[F("topics")][F("device")], 33); // "wled/test" | ||||
|   getStringFromJson(mqttGroupTopic, if_mqtt[F("topics")][F("group")], 33); // "" | ||||
|  | ||||
|   JsonObject if_hue = interfaces[F("hue")]; | ||||
|   CJSON(huePollingEnabled, if_hue["en"]); | ||||
|   CJSON(huePollLightId, if_hue["id"]); | ||||
|   tdd = if_hue[F("iv")] | -1; | ||||
|   if (tdd >= 2) huePollIntervalMs = tdd * 100; | ||||
|  | ||||
|   JsonObject if_hue_recv = if_hue["recv"]; | ||||
|   CJSON(hueApplyOnOff, if_hue_recv["on"]); | ||||
|   CJSON(hueApplyBri, if_hue_recv["bri"]); | ||||
|   CJSON(hueApplyColor, if_hue_recv["col"]); | ||||
|  | ||||
|   JsonArray if_hue_ip = if_hue["ip"]; | ||||
|  | ||||
|   for (byte i = 0; i < 4; i++) | ||||
|     CJSON(hueIP[i], if_hue_ip[i]); | ||||
|  | ||||
|   JsonObject if_ntp = interfaces[F("ntp")]; | ||||
|   CJSON(ntpEnabled, if_ntp["en"]); | ||||
|   getStringFromJson(ntpServerName, if_ntp[F("host")], 33); // "1.wled.pool.ntp.org" | ||||
|   CJSON(currentTimezone, if_ntp[F("tz")]); | ||||
|   CJSON(utcOffsetSecs, if_ntp[F("offset")]); | ||||
|   CJSON(useAMPM, if_ntp[F("ampm")]); | ||||
|  | ||||
|   JsonObject ol = doc[F("ol")]; | ||||
|   CJSON(overlayDefault ,ol[F("clock")]); // 0 | ||||
|   CJSON(countdownMode, ol[F("cntdwn")]); | ||||
|   overlayCurrent = overlayDefault; | ||||
|  | ||||
|   CJSON(overlayMin, ol[F("min")]); | ||||
|   CJSON(overlayMax, ol[F("max")]); | ||||
|   CJSON(analogClock12pixel, ol[F("o12pix")]); | ||||
|   CJSON(analogClock5MinuteMarks, ol[F("o5m")]); | ||||
|   CJSON(analogClockSecondsTrail, ol[F("osec")]); | ||||
|  | ||||
|   //timed macro rules | ||||
|   JsonObject tm = doc[F("timers")]; | ||||
|   JsonObject cntdwn = tm[F("cntdwn")]; | ||||
|   JsonArray cntdwn_goal = cntdwn[F("goal")]; | ||||
|   CJSON(countdownYear,  cntdwn_goal[0]); | ||||
|   CJSON(countdownMonth, cntdwn_goal[1]); | ||||
|   CJSON(countdownDay,   cntdwn_goal[2]); | ||||
|   CJSON(countdownHour,  cntdwn_goal[3]); | ||||
|   CJSON(countdownMin,   cntdwn_goal[4]); | ||||
|   CJSON(countdownSec,   cntdwn_goal[5]); | ||||
|   CJSON(macroCountdown, cntdwn[F("macro")]); | ||||
|   setCountdown(); | ||||
|  | ||||
|   JsonArray timers = tm[F("ins")]; | ||||
|   uint8_t it = 0; | ||||
|   for (JsonObject timer : timers) { | ||||
|     if (it > 7) break; | ||||
|     CJSON(timerHours[it], timer[F("hour")]); | ||||
|     CJSON(timerMinutes[it], timer[F("min")]); | ||||
|     CJSON(timerMacro[it], timer[F("macro")]); | ||||
|  | ||||
|     byte dowPrev =  timerWeekday[it]; | ||||
|     //note: act is currently only 0 or 1. | ||||
|     //the reason we are not using bool is that the on-disk type in 0.11.0 was already int | ||||
|     int actPrev = timerWeekday[it] & 0x01; | ||||
|     CJSON(timerWeekday[it], timer[F("dow")]); | ||||
|     if (timerWeekday[it] != dowPrev) { //present in JSON | ||||
|       timerWeekday[it] <<= 1; //add active bit | ||||
|       int act = timer["en"] | actPrev; | ||||
|       if (act) timerWeekday[it]++; | ||||
|     } | ||||
|  | ||||
|     it++; | ||||
|   } | ||||
|  | ||||
|   JsonObject ota = doc["ota"]; | ||||
|   const char* pwd = ota["psk"]; //normally not present due to security | ||||
|  | ||||
|   bool pwdCorrect = !otaLock; //always allow access if ota not locked | ||||
|   if (pwd != nullptr && strncmp(otaPass, pwd, 33) == 0) pwdCorrect = true; | ||||
|  | ||||
|   if (pwdCorrect) { //only accept these values from cfg.json if ota is unlocked (else from wsec.json) | ||||
|     CJSON(otaLock, ota[F("lock")]); | ||||
|     CJSON(wifiLock, ota[F("lock-wifi")]); | ||||
|     CJSON(aOtaEnabled, ota[F("aota")]); | ||||
|     getStringFromJson(otaPass, pwd, 33); //normally not present due to security | ||||
|   } | ||||
|  | ||||
|   #ifdef WLED_ENABLE_DMX | ||||
|   JsonObject dmx = doc["dmx"]; | ||||
|   CJSON(DMXChannels, dmx[F("chan")]); | ||||
|   CJSON(DMXGap,dmx[F("gap")]); | ||||
|   CJSON(DMXStart, dmx[F("start")]); | ||||
|   CJSON(DMXStartLED,dmx[F("start-led")]); | ||||
|  | ||||
|   JsonArray dmx_fixmap = dmx[F("fixmap")]; | ||||
|   it = 0; | ||||
|   for (int i : dmx_fixmap) { | ||||
|     if (it > 14) break; | ||||
|     CJSON(DMXFixtureMap[i],dmx_fixmap[i]); | ||||
|     it++; | ||||
|   } | ||||
|   #endif | ||||
|  | ||||
|   JsonObject usermods_settings = doc["um"]; | ||||
|   usermods.readFromConfig(usermods_settings); | ||||
| } | ||||
|  | ||||
| void serializeConfig() { | ||||
|   serializeConfigSec(); | ||||
|  | ||||
|   DEBUG_PRINTLN(F("Writing settings to /cfg.json...")); | ||||
|  | ||||
|   DynamicJsonDocument doc(JSON_BUFFER_SIZE); | ||||
|  | ||||
|   JsonArray rev = doc.createNestedArray("rev"); | ||||
|   rev.add(1); //major settings revision | ||||
|   rev.add(0); //minor settings revision | ||||
|  | ||||
|   doc[F("vid")] = VERSION; | ||||
|  | ||||
|   JsonObject id = doc.createNestedObject("id"); | ||||
|   id[F("mdns")] = cmDNS; | ||||
|   id[F("name")] = serverDescription; | ||||
|   id[F("inv")] = alexaInvocationName; | ||||
|  | ||||
|   JsonObject nw = doc.createNestedObject("nw"); | ||||
|  | ||||
|   JsonArray nw_ins = nw.createNestedArray("ins"); | ||||
|  | ||||
|   JsonObject nw_ins_0 = nw_ins.createNestedObject(); | ||||
|   nw_ins_0[F("ssid")] = clientSSID; | ||||
|   nw_ins_0[F("pskl")] = strlen(clientPass); | ||||
|  | ||||
|   JsonArray nw_ins_0_ip = nw_ins_0.createNestedArray("ip"); | ||||
|   JsonArray nw_ins_0_gw = nw_ins_0.createNestedArray("gw"); | ||||
|   JsonArray nw_ins_0_sn = nw_ins_0.createNestedArray("sn"); | ||||
|  | ||||
|   for (byte i = 0; i < 4; i++) { | ||||
|     nw_ins_0_ip.add(staticIP[i]); | ||||
|     nw_ins_0_gw.add(staticGateway[i]); | ||||
|     nw_ins_0_sn.add(staticSubnet[i]); | ||||
|   } | ||||
|  | ||||
|   JsonObject ap = doc.createNestedObject("ap"); | ||||
|   ap[F("ssid")] = apSSID; | ||||
|   ap[F("pskl")] = strlen(apPass); | ||||
|   ap[F("chan")] = apChannel; | ||||
|   ap[F("hide")] = apHide; | ||||
|   ap[F("behav")] = apBehavior; | ||||
|  | ||||
|   JsonArray ap_ip = ap.createNestedArray("ip"); | ||||
|   ap_ip.add(4); | ||||
|   ap_ip.add(3); | ||||
|   ap_ip.add(2); | ||||
|   ap_ip.add(1); | ||||
|  | ||||
|   JsonObject wifi = doc.createNestedObject("wifi"); | ||||
|   wifi[F("sleep")] = !noWifiSleep; | ||||
|   wifi[F("phy")] = 1; | ||||
|  | ||||
|   #ifdef WLED_USE_ETHERNET | ||||
|   JsonObject ethernet = doc.createNestedObject("eth"); | ||||
|   ethernet["type"] = ethernetType; | ||||
|   #endif | ||||
|  | ||||
|   JsonObject hw = doc.createNestedObject("hw"); | ||||
|  | ||||
|   JsonObject hw_led = hw.createNestedObject("led"); | ||||
|   hw_led[F("total")] = ledCount; | ||||
|   hw_led[F("maxpwr")] = strip.ablMilliampsMax; | ||||
|   hw_led[F("ledma")] = strip.milliampsPerLed; | ||||
|   hw_led[F("rgbwm")] = strip.rgbwMode; | ||||
|  | ||||
|   JsonArray hw_led_ins = hw_led.createNestedArray("ins"); | ||||
|  | ||||
|   for (uint8_t s = 0; s < busses.getNumBusses(); s++) { | ||||
|     Bus *bus = busses.getBus(s); | ||||
|     if (!bus || bus->getLength()==0) break; | ||||
|     JsonObject ins = hw_led_ins.createNestedObject(); | ||||
|     ins["en"] = true; | ||||
|     ins[F("start")] = bus->getStart(); | ||||
|     ins[F("len")] = bus->getLength(); | ||||
|     JsonArray ins_pin = ins.createNestedArray("pin"); | ||||
|     uint8_t pins[5]; | ||||
|     uint8_t nPins = bus->getPins(pins); | ||||
|     for (uint8_t i = 0; i < nPins; i++) ins_pin.add(pins[i]); | ||||
|     ins[F("order")] = bus->getColorOrder(); | ||||
|     ins["rev"] = bus->reversed; | ||||
|     ins[F("skip")] = (skipFirstLed && s == 0) ? 1 : 0; | ||||
|     ins["type"] = bus->getType(); | ||||
|   } | ||||
|  | ||||
|   JsonObject hw_btn = hw.createNestedObject("btn"); | ||||
|  | ||||
|   JsonArray hw_btn_ins = hw_btn.createNestedArray("ins"); | ||||
|  | ||||
|   // button BTNPIN | ||||
|   JsonObject hw_btn_ins_0 = hw_btn_ins.createNestedObject(); | ||||
|   hw_btn_ins_0["type"] = (buttonEnabled) ? BTN_TYPE_PUSH : BTN_TYPE_NONE; | ||||
|  | ||||
|   JsonArray hw_btn_ins_0_pin = hw_btn_ins_0.createNestedArray("pin"); | ||||
|   hw_btn_ins_0_pin.add(btnPin); | ||||
|  | ||||
|   JsonArray hw_btn_ins_0_macros = hw_btn_ins_0.createNestedArray("macros"); | ||||
|   hw_btn_ins_0_macros.add(macroButton); | ||||
|   hw_btn_ins_0_macros.add(macroLongPress); | ||||
|   hw_btn_ins_0_macros.add(macroDoublePress); | ||||
|  | ||||
|   #ifndef WLED_DISABLE_INFRARED | ||||
|   JsonObject hw_ir = hw.createNestedObject("ir"); | ||||
|   hw_ir["pin"] = irPin; | ||||
|   hw_ir[F("type")] = irEnabled;              // the byte 'irEnabled' does contain the IR-Remote Type ( 0=disabled ) | ||||
|   #endif | ||||
|  | ||||
|   JsonObject hw_relay = hw.createNestedObject(F("relay")); | ||||
|   hw_relay["pin"] = rlyPin; | ||||
|   hw_relay["rev"] = !rlyMde; | ||||
|  | ||||
|   //JsonObject hw_status = hw.createNestedObject("status"); | ||||
|   //hw_status["pin"] = -1; | ||||
|  | ||||
|   JsonObject light = doc.createNestedObject(F("light")); | ||||
|   light[F("scale-bri")] = briMultiplier; | ||||
|   light[F("pal-mode")] = strip.paletteBlend; | ||||
|  | ||||
|   JsonObject light_gc = light.createNestedObject("gc"); | ||||
|   light_gc["bri"] = (strip.gammaCorrectBri) ? 2.8 : 1.0; | ||||
|   light_gc["col"] = (strip.gammaCorrectCol) ? 2.8 : 1.0; | ||||
|  | ||||
|   JsonObject light_tr = light.createNestedObject("tr"); | ||||
|   light_tr[F("mode")] = fadeTransition; | ||||
|   light_tr[F("dur")] = transitionDelayDefault / 100; | ||||
|   light_tr[F("pal")] = strip.paletteFade; | ||||
|  | ||||
|   JsonObject light_nl = light.createNestedObject("nl"); | ||||
|   light_nl[F("mode")] = nightlightMode; | ||||
|   light_nl[F("dur")] = nightlightDelayMinsDefault; | ||||
|   light_nl[F("tbri")] = nightlightTargetBri; | ||||
|   light_nl[F("macro")] = macroNl; | ||||
|  | ||||
|   JsonObject def = doc.createNestedObject("def"); | ||||
|   def[F("ps")] = bootPreset; | ||||
|   def["on"] = turnOnAtBoot; | ||||
|   def["bri"] = briS; | ||||
|  | ||||
|   //to be removed once preset cycles are presets | ||||
|   if (saveCurrPresetCycConf) { | ||||
|     JsonObject def_cy = def.createNestedObject("cy"); | ||||
|     def_cy["on"] = presetCyclingEnabled; | ||||
|  | ||||
|     JsonArray def_cy_range = def_cy.createNestedArray(F("range")); | ||||
|     def_cy_range.add(presetCycleMin); | ||||
|     def_cy_range.add(presetCycleMax); | ||||
|     def_cy[F("dur")] = presetCycleTime; | ||||
|   } | ||||
|  | ||||
|   JsonObject interfaces = doc.createNestedObject("if"); | ||||
|  | ||||
|   JsonObject if_sync = interfaces.createNestedObject("sync"); | ||||
|   if_sync[F("port0")] = udpPort; | ||||
|   if_sync[F("port1")] = udpPort2; | ||||
|  | ||||
|   JsonObject if_sync_recv = if_sync.createNestedObject("recv"); | ||||
|   if_sync_recv["bri"] = receiveNotificationBrightness; | ||||
|   if_sync_recv["col"] = receiveNotificationColor; | ||||
|   if_sync_recv[F("fx")] = receiveNotificationEffects; | ||||
|  | ||||
|   JsonObject if_sync_send = if_sync.createNestedObject("send"); | ||||
|   if_sync_send[F("dir")] = notifyDirect; | ||||
|   if_sync_send[F("btn")] = notifyButton; | ||||
|   if_sync_send[F("va")] = notifyAlexa; | ||||
|   if_sync_send[F("hue")] = notifyHue; | ||||
|   if_sync_send[F("macro")] = notifyMacro; | ||||
|   if_sync_send[F("twice")] = notifyTwice; | ||||
|  | ||||
|   JsonObject if_nodes = interfaces.createNestedObject("nodes"); | ||||
|   if_nodes[F("list")] = nodeListEnabled; | ||||
|   if_nodes[F("bcast")] = nodeBroadcastEnabled; | ||||
|  | ||||
|   JsonObject if_live = interfaces.createNestedObject("live"); | ||||
|   if_live["en"] = receiveDirect; | ||||
|   if_live["port"] = e131Port; | ||||
|   if_live[F("mc")] = e131Multicast; | ||||
|  | ||||
|   JsonObject if_live_dmx = if_live.createNestedObject("dmx"); | ||||
|   if_live_dmx[F("uni")] = e131Universe; | ||||
|   if_live_dmx[F("seqskip")] = e131SkipOutOfSequence; | ||||
|   if_live_dmx[F("addr")] = DMXAddress; | ||||
|   if_live_dmx[F("mode")] = DMXMode; | ||||
|   if_live[F("timeout")] = realtimeTimeoutMs / 100; | ||||
|   if_live[F("maxbri")] = arlsForceMaxBri; | ||||
|   if_live[F("no-gc")] = arlsDisableGammaCorrection; | ||||
|   if_live[F("offset")] = arlsOffset; | ||||
|  | ||||
|   JsonObject if_va = interfaces.createNestedObject("va"); | ||||
|   if_va[F("alexa")] = alexaEnabled; | ||||
|  | ||||
|   JsonArray if_va_macros = if_va.createNestedArray("macros"); | ||||
|   if_va_macros.add(macroAlexaOn); | ||||
|   if_va_macros.add(macroAlexaOff); | ||||
|   JsonObject if_blynk = interfaces.createNestedObject("blynk"); | ||||
|   if_blynk[F("token")] = strlen(blynkApiKey) ? "Hidden":""; | ||||
|   if_blynk[F("host")] = blynkHost; | ||||
|   if_blynk["port"] = blynkPort; | ||||
|  | ||||
|   JsonObject if_mqtt = interfaces.createNestedObject("mqtt"); | ||||
|   if_mqtt["en"] = mqttEnabled; | ||||
|   if_mqtt[F("broker")] = mqttServer; | ||||
|   if_mqtt["port"] = mqttPort; | ||||
|   if_mqtt[F("user")] = mqttUser; | ||||
|   if_mqtt[F("pskl")] = strlen(mqttPass); | ||||
|   if_mqtt[F("cid")] = mqttClientID; | ||||
|  | ||||
|   JsonObject if_mqtt_topics = if_mqtt.createNestedObject(F("topics")); | ||||
|   if_mqtt_topics[F("device")] = mqttDeviceTopic; | ||||
|   if_mqtt_topics[F("group")] = mqttGroupTopic; | ||||
|  | ||||
|   JsonObject if_hue = interfaces.createNestedObject("hue"); | ||||
|   if_hue["en"] = huePollingEnabled; | ||||
|   if_hue["id"] = huePollLightId; | ||||
|   if_hue[F("iv")] = huePollIntervalMs / 100; | ||||
|  | ||||
|   JsonObject if_hue_recv = if_hue.createNestedObject("recv"); | ||||
|   if_hue_recv["on"] = hueApplyOnOff; | ||||
|   if_hue_recv["bri"] = hueApplyBri; | ||||
|   if_hue_recv["col"] = hueApplyColor; | ||||
|  | ||||
|   JsonArray if_hue_ip = if_hue.createNestedArray("ip"); | ||||
|   for (byte i = 0; i < 4; i++) { | ||||
|     if_hue_ip.add(hueIP[i]); | ||||
|   } | ||||
|  | ||||
|   JsonObject if_ntp = interfaces.createNestedObject("ntp"); | ||||
|   if_ntp["en"] = ntpEnabled; | ||||
|   if_ntp[F("host")] = ntpServerName; | ||||
|   if_ntp[F("tz")] = currentTimezone; | ||||
|   if_ntp[F("offset")] = utcOffsetSecs; | ||||
|   if_ntp[F("ampm")] = useAMPM; | ||||
|  | ||||
|   JsonObject ol = doc.createNestedObject("ol"); | ||||
|   ol[F("clock")] = overlayDefault; | ||||
|   ol[F("cntdwn")] = countdownMode; | ||||
|  | ||||
|   ol[F("min")] = overlayMin; | ||||
|   ol[F("max")] = overlayMax; | ||||
|   ol[F("o12pix")] = analogClock12pixel; | ||||
|   ol[F("o5m")] = analogClock5MinuteMarks; | ||||
|   ol[F("osec")] = analogClockSecondsTrail; | ||||
|  | ||||
|   JsonObject timers = doc.createNestedObject(F("timers")); | ||||
|  | ||||
|   JsonObject cntdwn = timers.createNestedObject(F("cntdwn")); | ||||
|   JsonArray goal = cntdwn.createNestedArray(F("goal")); | ||||
|   goal.add(countdownYear); goal.add(countdownMonth); goal.add(countdownDay); | ||||
|   goal.add(countdownHour); goal.add(countdownMin); goal.add(countdownSec); | ||||
|   cntdwn[F("macro")] = macroCountdown; | ||||
|  | ||||
|   JsonArray timers_ins = timers.createNestedArray("ins"); | ||||
|  | ||||
|   for (byte i = 0; i < 8; i++) { | ||||
|     if (timerMacro[i] == 0 && timerHours[i] == 0 && timerMinutes[i] == 0) continue; | ||||
|     JsonObject timers_ins0 = timers_ins.createNestedObject(); | ||||
|     timers_ins0["en"] = (timerWeekday[i] & 0x01); | ||||
|     timers_ins0[F("hour")] = timerHours[i]; | ||||
|     timers_ins0[F("min")] = timerMinutes[i]; | ||||
|     timers_ins0[F("macro")] = timerMacro[i]; | ||||
|     timers_ins0[F("dow")] = timerWeekday[i] >> 1; | ||||
|   } | ||||
|  | ||||
|   JsonObject ota = doc.createNestedObject("ota"); | ||||
|   ota[F("lock")] = otaLock; | ||||
|   ota[F("lock-wifi")] = wifiLock; | ||||
|   ota[F("pskl")] = strlen(otaPass); | ||||
|   ota[F("aota")] = aOtaEnabled; | ||||
|  | ||||
|   #ifdef WLED_ENABLE_DMX | ||||
|   JsonObject dmx = doc.createNestedObject("dmx"); | ||||
|   dmx[F("chan")] = DMXChannels; | ||||
|   dmx[F("gap")] = DMXGap; | ||||
|   dmx[F("start")] = DMXStart; | ||||
|   dmx[F("start-led")] = DMXStartLED; | ||||
|  | ||||
|   JsonArray dmx_fixmap = dmx.createNestedArray(F("fixmap")); | ||||
|   for (byte i = 0; i < 15; i++) | ||||
|     dmx_fixmap.add(DMXFixtureMap[i]); | ||||
|   #endif | ||||
|   //} | ||||
|  | ||||
|   JsonObject usermods_settings = doc.createNestedObject("um"); | ||||
|   usermods.addToConfig(usermods_settings); | ||||
|  | ||||
|   File f = WLED_FS.open("/cfg.json", "w"); | ||||
|   if (f) serializeJson(doc, f); | ||||
|   f.close(); | ||||
| } | ||||
|  | ||||
| //settings in /wsec.json, not accessible via webserver, for passwords and tokens | ||||
| bool deserializeConfigSec() { | ||||
|   DEBUG_PRINTLN(F("Reading settings from /wsec.json...")); | ||||
|  | ||||
|   DynamicJsonDocument doc(JSON_BUFFER_SIZE); | ||||
|  | ||||
|   bool success = readObjectFromFile("/wsec.json", nullptr, &doc); | ||||
|   if (!success) return false; | ||||
|  | ||||
|   JsonObject nw_ins_0 = doc["nw"][F("ins")][0]; | ||||
|   getStringFromJson(clientPass, nw_ins_0["psk"], 65); | ||||
|  | ||||
|   JsonObject ap = doc["ap"]; | ||||
|   getStringFromJson(apPass, ap["psk"] , 65); | ||||
|  | ||||
|   JsonObject interfaces = doc["if"]; | ||||
|  | ||||
|   const char* apikey = interfaces["blynk"][F("token")] | "Hidden"; | ||||
|   int tdd = strnlen(apikey, 36); | ||||
|   if (tdd > 20 || tdd == 0) | ||||
|     getStringFromJson(blynkApiKey, apikey, 36); | ||||
|  | ||||
|   JsonObject if_mqtt = interfaces["mqtt"]; | ||||
|   getStringFromJson(mqttPass, if_mqtt["psk"], 41); | ||||
|  | ||||
|   getStringFromJson(hueApiKey, interfaces[F("hue")][F("key")], 47); | ||||
|  | ||||
|   JsonObject ota = doc["ota"]; | ||||
|   getStringFromJson(otaPass, ota[F("pwd")], 33); | ||||
|   CJSON(otaLock, ota[F("lock")]); | ||||
|   CJSON(wifiLock, ota[F("lock-wifi")]); | ||||
|   CJSON(aOtaEnabled, ota[F("aota")]); | ||||
|  | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| void serializeConfigSec() { | ||||
|   DEBUG_PRINTLN(F("Writing settings to /wsec.json...")); | ||||
|  | ||||
|   DynamicJsonDocument doc(JSON_BUFFER_SIZE); | ||||
|  | ||||
|   JsonObject nw = doc.createNestedObject("nw"); | ||||
|  | ||||
|   JsonArray nw_ins = nw.createNestedArray("ins"); | ||||
|  | ||||
|   JsonObject nw_ins_0 = nw_ins.createNestedObject(); | ||||
|   nw_ins_0["psk"] = clientPass; | ||||
|  | ||||
|   JsonObject ap = doc.createNestedObject("ap"); | ||||
|   ap["psk"] = apPass; | ||||
|  | ||||
|   JsonObject interfaces = doc.createNestedObject("if"); | ||||
|   JsonObject if_blynk = interfaces.createNestedObject("blynk"); | ||||
|   if_blynk[F("token")] = blynkApiKey; | ||||
|   JsonObject if_mqtt = interfaces.createNestedObject("mqtt"); | ||||
|   if_mqtt["psk"] = mqttPass; | ||||
|   JsonObject if_hue = interfaces.createNestedObject("hue"); | ||||
|   if_hue[F("key")] = hueApiKey; | ||||
|  | ||||
|   JsonObject ota = doc.createNestedObject("ota"); | ||||
|   ota[F("pwd")] = otaPass; | ||||
|   ota[F("lock")] = otaLock; | ||||
|   ota[F("lock-wifi")] = wifiLock; | ||||
|   ota[F("aota")] = aOtaEnabled; | ||||
|  | ||||
|   File f = WLED_FS.open("/wsec.json", "w"); | ||||
|   if (f) serializeJson(doc, f); | ||||
|   f.close(); | ||||
| } | ||||
| @@ -1,3 +1,5 @@ | ||||
| #include "wled.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * Color conversion methods | ||||
|  */ | ||||
| @@ -18,7 +20,7 @@ void colorFromUint32(uint32_t in, bool secondary) | ||||
| } | ||||
| 
 | ||||
| //load a color without affecting the white channel
 | ||||
| void colorFromUint24(uint32_t in, bool secondary = false) | ||||
| void colorFromUint24(uint32_t in, bool secondary) | ||||
| { | ||||
|   if (secondary) { | ||||
|     colSec[0] = in >> 16 & 0xFF; | ||||
| @@ -31,8 +33,13 @@ void colorFromUint24(uint32_t in, bool secondary = false) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| //store color components in uint32_t
 | ||||
| uint32_t colorFromRgbw(byte* rgbw) { | ||||
|   return (rgbw[0] << 16) + (rgbw[1] << 8) + rgbw[2] + (rgbw[3] << 24); | ||||
| } | ||||
| 
 | ||||
| //relatively change white brightness, minumum A=5
 | ||||
| void relativeChangeWhite(int8_t amount, byte lowerBoundary =0) | ||||
| void relativeChangeWhite(int8_t amount, byte lowerBoundary) | ||||
| { | ||||
|   int16_t new_val = (int16_t) col[3] + amount; | ||||
|   if (new_val > 0xFF) new_val = 0xFF; | ||||
| @@ -57,11 +64,34 @@ void colorHStoRGB(uint16_t hue, byte sat, byte* rgb) //hue, sat to rgb | ||||
|     case 4: rgb[0]=t,rgb[1]=p,rgb[2]=255;break; | ||||
|     case 5: rgb[0]=255,rgb[1]=p,rgb[2]=q; | ||||
|   } | ||||
|   if (useRGBW) colorRGBtoRGBW(col); | ||||
|   if (strip.isRgbw && strip.rgbwMode == RGBW_MODE_LEGACY) colorRGBtoRGBW(col); | ||||
| } | ||||
| 
 | ||||
| #ifndef WLED_DISABLE_HUESYNC | ||||
| void colorCTtoRGB(uint16_t mired, byte* rgb) //white spectrum to rgb
 | ||||
| void colorKtoRGB(uint16_t kelvin, byte* rgb) //white spectrum to rgb, calc
 | ||||
| { | ||||
|   float r = 0, g = 0, b = 0; | ||||
|   float temp = kelvin / 100; | ||||
|   if (temp <= 66) { | ||||
|     r = 255; | ||||
|     g = round(99.4708025861 * log(temp) - 161.1195681661); | ||||
|     if (temp <= 19) { | ||||
|       b = 0; | ||||
|     } else { | ||||
|       b = round(138.5177312231 * log((temp - 10)) - 305.0447927307); | ||||
|     } | ||||
|   } else { | ||||
|     r = round(329.698727446 * pow((temp - 60), -0.1332047592)); | ||||
|     g = round(288.1221695283 * pow((temp - 60), -0.0755148492)); | ||||
|     b = 255; | ||||
|   }  | ||||
|   g += 15; //mod by Aircoookie, a bit less accurate but visibly less pinkish
 | ||||
|   rgb[0] = (uint8_t) constrain(r, 0, 255); | ||||
|   rgb[1] = (uint8_t) constrain(g, 0, 255); | ||||
|   rgb[2] = (uint8_t) constrain(b, 0, 255); | ||||
|   rgb[3] = 0; | ||||
| } | ||||
| 
 | ||||
| void colorCTtoRGB(uint16_t mired, byte* rgb) //white spectrum to rgb, bins
 | ||||
| { | ||||
|   //this is only an approximation using WS2812B with gamma correction enabled
 | ||||
|   if (mired > 475) { | ||||
| @@ -81,9 +111,10 @@ void colorCTtoRGB(uint16_t mired, byte* rgb) //white spectrum to rgb | ||||
|   } else { | ||||
|     rgb[0]=237;rgb[1]=255;rgb[2]=239;//150
 | ||||
|   } | ||||
|   if (useRGBW) colorRGBtoRGBW(col); | ||||
|   if (strip.isRgbw && strip.rgbwMode == RGBW_MODE_LEGACY) colorRGBtoRGBW(col); | ||||
| } | ||||
| 
 | ||||
| #ifndef WLED_DISABLE_HUESYNC | ||||
| void colorXYtoRGB(float x, float y, byte* rgb) //coordinates to rgb (https://www.developers.meethue.com/documentation/color-conversions-rgb-xy)
 | ||||
| { | ||||
|   float z = 1.0f - x - y; | ||||
| @@ -138,7 +169,7 @@ void colorXYtoRGB(float x, float y, byte* rgb) //coordinates to rgb (https://www | ||||
|   rgb[0] = 255.0*r; | ||||
|   rgb[1] = 255.0*g; | ||||
|   rgb[2] = 255.0*b; | ||||
|   if (useRGBW) colorRGBtoRGBW(col); | ||||
|   if (strip.isRgbw && strip.rgbwMode == RGBW_MODE_LEGACY) colorRGBtoRGBW(col); | ||||
| } | ||||
| 
 | ||||
| void colorRGBtoXY(byte* rgb, float* xy) //rgb to coordinates (https://www.developers.meethue.com/documentation/color-conversions-rgb-xy)
 | ||||
| @@ -149,8 +180,9 @@ void colorRGBtoXY(byte* rgb, float* xy) //rgb to coordinates (https://www.develo | ||||
|   xy[0] = X / (X + Y + Z); | ||||
|   xy[1] = Y / (X + Y + Z); | ||||
| } | ||||
| #endif | ||||
| #endif // WLED_DISABLE_HUESYNC
 | ||||
| 
 | ||||
| //RRGGBB / WWRRGGBB order for hex
 | ||||
| void colorFromDecOrHexString(byte* rgb, char* in) | ||||
| { | ||||
|   if (in[0] == 0) return; | ||||
| @@ -171,6 +203,27 @@ void colorFromDecOrHexString(byte* rgb, char* in) | ||||
|   rgb[2] =  c        & 0xFF; | ||||
| } | ||||
| 
 | ||||
| //contrary to the colorFromDecOrHexString() function, this uses the more standard RRGGBB / RRGGBBWW order
 | ||||
| bool colorFromHexString(byte* rgb, const char* in) { | ||||
|   if (in == nullptr) return false; | ||||
|   size_t inputSize = strnlen(in, 9); | ||||
|   if (inputSize != 6 && inputSize != 8) return false; | ||||
| 
 | ||||
|   uint32_t c = strtoul(in, NULL, 16); | ||||
| 
 | ||||
|   if (inputSize == 6) { | ||||
|     rgb[0] = (c >> 16) & 0xFF; | ||||
|     rgb[1] = (c >>  8) & 0xFF; | ||||
|     rgb[2] =  c        & 0xFF; | ||||
|   } else { | ||||
|     rgb[0] = (c >> 24) & 0xFF; | ||||
|     rgb[1] = (c >> 16) & 0xFF; | ||||
|     rgb[2] = (c >>  8) & 0xFF; | ||||
|     rgb[3] =  c        & 0xFF; | ||||
|   } | ||||
|   return true; | ||||
| } | ||||
| 
 | ||||
| float minf (float v, float w) | ||||
| { | ||||
|   if (w > v) return v; | ||||
| @@ -183,7 +236,7 @@ float maxf (float v, float w) | ||||
|   return v; | ||||
| } | ||||
| 
 | ||||
| void colorRGBtoRGBW(byte* rgb) //rgb to rgbw (http://codewelt.com/rgbw)
 | ||||
| void colorRGBtoRGBW(byte* rgb) //rgb to rgbw (http://codewelt.com/rgbw). (RGBW_MODE_LEGACY)
 | ||||
| { | ||||
|   float low = minf(rgb[0],minf(rgb[1],rgb[2])); | ||||
|   float high = maxf(rgb[0],maxf(rgb[1],rgb[2])); | ||||
							
								
								
									
										254
									
								
								wled00/const.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										254
									
								
								wled00/const.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,254 @@ | ||||
| #ifndef WLED_CONST_H | ||||
| #define WLED_CONST_H | ||||
|  | ||||
| /* | ||||
|  * Readability defines and their associated numerical values + compile-time constants | ||||
|  */ | ||||
|  | ||||
| //Defaults | ||||
| #define DEFAULT_CLIENT_SSID "Your_Network" | ||||
| #define DEFAULT_AP_PASS     "wled1234" | ||||
| #define DEFAULT_OTA_PASS    "wledota" | ||||
|  | ||||
| //increase if you need more | ||||
| #define WLED_MAX_USERMODS 4 | ||||
|  | ||||
| #ifdef ESP8266 | ||||
| #define WLED_MAX_BUSSES 3 | ||||
| #else | ||||
| #define WLED_MAX_BUSSES 10 | ||||
| #endif | ||||
|  | ||||
| //Usermod IDs | ||||
| #define USERMOD_ID_RESERVED       0            //Unused. Might indicate no usermod present | ||||
| #define USERMOD_ID_UNSPECIFIED    1            //Default value for a general user mod that does not specify a custom ID | ||||
| #define USERMOD_ID_EXAMPLE        2            //Usermod "usermod_v2_example.h" | ||||
| #define USERMOD_ID_TEMPERATURE    3            //Usermod "usermod_temperature.h" | ||||
| #define USERMOD_ID_FIXNETSERVICES 4            //Usermod "usermod_Fix_unreachable_netservices.h" | ||||
| #define USERMOD_ID_PIRSWITCH      5            //Usermod "usermod_PIR_sensor_switch.h" | ||||
| #define USERMOD_ID_IMU            6            //Usermod "usermod_mpu6050_imu.h" | ||||
| #define USERMOD_ID_FOUR_LINE_DISP 7            //Usermod "usermod_v2_four_line_display.h  | ||||
| #define USERMOD_ID_ROTARY_ENC_UI  8            //Usermod "usermod_v2_rotary_encoder_ui.h" | ||||
| #define USERMOD_ID_AUTO_SAVE      9            //Usermod "usermod_v2_auto_save.h" | ||||
| #define USERMOD_ID_DHT           10            //Usermod "usermod_dht.h" | ||||
| #define USERMOD_ID_MODE_SORT     11            //Usermod "usermod_v2_mode_sort.h" | ||||
|  | ||||
| //Access point behavior | ||||
| #define AP_BEHAVIOR_BOOT_NO_CONN  0            //Open AP when no connection after boot | ||||
| #define AP_BEHAVIOR_NO_CONN       1            //Open when no connection (either after boot or if connection is lost) | ||||
| #define AP_BEHAVIOR_ALWAYS        2            //Always open | ||||
| #define AP_BEHAVIOR_BUTTON_ONLY   3            //Only when button pressed for 6 sec | ||||
|  | ||||
| //Notifier callMode  | ||||
| #define NOTIFIER_CALL_MODE_INIT           0    //no updates on init, can be used to disable updates | ||||
| #define NOTIFIER_CALL_MODE_DIRECT_CHANGE  1 | ||||
| #define NOTIFIER_CALL_MODE_BUTTON         2 | ||||
| #define NOTIFIER_CALL_MODE_NOTIFICATION   3 | ||||
| #define NOTIFIER_CALL_MODE_NIGHTLIGHT     4 | ||||
| #define NOTIFIER_CALL_MODE_NO_NOTIFY      5 | ||||
| #define NOTIFIER_CALL_MODE_FX_CHANGED     6    //no longer used | ||||
| #define NOTIFIER_CALL_MODE_HUE            7 | ||||
| #define NOTIFIER_CALL_MODE_PRESET_CYCLE   8 | ||||
| #define NOTIFIER_CALL_MODE_BLYNK          9 | ||||
| #define NOTIFIER_CALL_MODE_ALEXA         10 | ||||
|  | ||||
| //RGB to RGBW conversion mode | ||||
| #define RGBW_MODE_MANUAL_ONLY     0            //No automatic white channel calculation. Manual white channel slider | ||||
| #define RGBW_MODE_AUTO_BRIGHTER   1            //New algorithm. Adds as much white as the darkest RGBW channel | ||||
| #define RGBW_MODE_AUTO_ACCURATE   2            //New algorithm. Adds as much white as the darkest RGBW channel and subtracts this amount from each RGB channel | ||||
| #define RGBW_MODE_DUAL            3            //Manual slider + auto calculation. Automatically calculates only if manual slider is set to off (0)   | ||||
| #define RGBW_MODE_LEGACY          4            //Old floating algorithm. Too slow for realtime and palette support | ||||
|  | ||||
| //realtime modes | ||||
| #define REALTIME_MODE_INACTIVE    0 | ||||
| #define REALTIME_MODE_GENERIC     1 | ||||
| #define REALTIME_MODE_UDP         2 | ||||
| #define REALTIME_MODE_HYPERION    3 | ||||
| #define REALTIME_MODE_E131        4 | ||||
| #define REALTIME_MODE_ADALIGHT    5 | ||||
| #define REALTIME_MODE_ARTNET      6 | ||||
| #define REALTIME_MODE_TPM2NET     7 | ||||
| #define REALTIME_MODE_DDP         8 | ||||
|  | ||||
| //realtime override modes | ||||
| #define REALTIME_OVERRIDE_NONE    0 | ||||
| #define REALTIME_OVERRIDE_ONCE    1 | ||||
| #define REALTIME_OVERRIDE_ALWAYS  2 | ||||
|  | ||||
| //E1.31 DMX modes | ||||
| #define DMX_MODE_DISABLED         0            //not used | ||||
| #define DMX_MODE_SINGLE_RGB       1            //all LEDs same RGB color (3 channels) | ||||
| #define DMX_MODE_SINGLE_DRGB      2            //all LEDs same RGB color and master dimmer (4 channels) | ||||
| #define DMX_MODE_EFFECT           3            //trigger standalone effects of WLED (11 channels) | ||||
| #define DMX_MODE_MULTIPLE_RGB     4            //every LED is addressed with its own RGB (ledCount * 3 channels) | ||||
| #define DMX_MODE_MULTIPLE_DRGB    5            //every LED is addressed with its own RGB and share a master dimmer (ledCount * 3 + 1 channels) | ||||
| #define DMX_MODE_MULTIPLE_RGBW    6            //every LED is addressed with its own RGBW (ledCount * 4 channels) | ||||
|  | ||||
| //Light capability byte (unused) 0bRRCCTTTT | ||||
| //bits 0/1/2/3: specifies a type of LED driver. A single "driver" may have different chip models but must have the same protocol/behavior | ||||
| //bits 4/5: specifies the class of LED driver - 0b00 (dec. 0-15)  unconfigured/reserved | ||||
| //                                            - 0b01 (dec. 16-31) digital (data pin only) | ||||
| //                                            - 0b10 (dec. 32-47) analog (PWM) | ||||
| //                                            - 0b11 (dec. 48-63) digital (data + clock / SPI) | ||||
| //bits 6/7 are reserved and set to 0b00 | ||||
|  | ||||
| #define TYPE_NONE                 0            //light is not configured | ||||
| #define TYPE_RESERVED             1            //unused. Might indicate a "virtual" light | ||||
| //Digital types (data pin only) (16-31) | ||||
| #define TYPE_WS2812_1CH          20            //white-only chips | ||||
| #define TYPE_WS2812_WWA          21            //amber + warm + cold white | ||||
| #define TYPE_WS2812_RGB          22 | ||||
| #define TYPE_GS8608              23            //same driver as WS2812, but will require signal 2x per second (else displays test pattern) | ||||
| #define TYPE_WS2811_400KHZ       24            //half-speed WS2812 protocol, used by very old WS2811 units | ||||
| #define TYPE_SK6812_RGBW         30 | ||||
| #define TYPE_TM1814              31 | ||||
| //"Analog" types (PWM) (32-47) | ||||
| #define TYPE_ONOFF               40            //binary output (relays etc.) | ||||
| #define TYPE_ANALOG_1CH          41            //single channel PWM. Uses value of brightest RGBW channel | ||||
| #define TYPE_ANALOG_2CH          42            //analog WW + CW | ||||
| #define TYPE_ANALOG_3CH          43            //analog RGB | ||||
| #define TYPE_ANALOG_4CH          44            //analog RGBW | ||||
| #define TYPE_ANALOG_5CH          45            //analog RGB + WW + CW | ||||
| //Digital types (data + clock / SPI) (48-63) | ||||
| #define TYPE_WS2801              50 | ||||
| #define TYPE_APA102              51 | ||||
| #define TYPE_LPD8806             52 | ||||
| #define TYPE_P9813               53 | ||||
|  | ||||
| #define IS_DIGITAL(t) (t & 0x10) //digital are 16-31 and 48-63 | ||||
| #define IS_PWM(t)     (t > 40 && t < 46) | ||||
| #define NUM_PWM_PINS(t) (t - 40) //for analog PWM 41-45 only | ||||
| #define IS_2PIN(t)      (t > 47) | ||||
|  | ||||
| //Color orders | ||||
| #define COL_ORDER_GRB             0           //GRB(w),defaut | ||||
| #define COL_ORDER_RGB             1           //common for WS2811 | ||||
| #define COL_ORDER_BRG             2 | ||||
| #define COL_ORDER_RBG             3 | ||||
| #define COL_ORDER_BGR             4 | ||||
| #define COL_ORDER_GBR             5 | ||||
|  | ||||
|  | ||||
| //Button type | ||||
| #define BTN_TYPE_NONE             0 | ||||
| #define BTN_TYPE_RESERVED         1 | ||||
| #define BTN_TYPE_PUSH             2 | ||||
| #define BTN_TYPE_PUSH_ACT_HIGH    3 //not implemented | ||||
| #define BTN_TYPE_SWITCH           4 //not implemented | ||||
| #define BTN_TYPE_SWITCH_ACT_HIGH  5 //not implemented | ||||
|  | ||||
| //Ethernet board types | ||||
| #define WLED_NUM_ETH_TYPES        5 | ||||
|  | ||||
| #define WLED_ETH_NONE             0 | ||||
| #define WLED_ETH_WT32_ETH01       1 | ||||
| #define WLED_ETH_ESP32_POE        2 | ||||
| #define WLED_ETH_WESP32           3 | ||||
| #define WLED_ETH_QUINLED          4 | ||||
|  | ||||
| //Hue error codes | ||||
| #define HUE_ERROR_INACTIVE        0 | ||||
| #define HUE_ERROR_UNAUTHORIZED    1 | ||||
| #define HUE_ERROR_LIGHTID         3 | ||||
| #define HUE_ERROR_PUSHLINK      101 | ||||
| #define HUE_ERROR_JSON_PARSING  250 | ||||
| #define HUE_ERROR_TIMEOUT       251 | ||||
| #define HUE_ERROR_ACTIVE        255 | ||||
|  | ||||
| //Segment option byte bits | ||||
| #define SEG_OPTION_SELECTED       0 | ||||
| #define SEG_OPTION_REVERSED       1 | ||||
| #define SEG_OPTION_ON             2 | ||||
| #define SEG_OPTION_MIRROR         3            //Indicates that the effect will be mirrored within the segment | ||||
| #define SEG_OPTION_NONUNITY       4            //Indicates that the effect does not use FRAMETIME or needs getPixelColor | ||||
| #define SEG_OPTION_FREEZE         5            //Segment contents will not be refreshed | ||||
| #define SEG_OPTION_TRANSITIONAL   7 | ||||
|  | ||||
| // WLED Error modes | ||||
| #define ERR_NONE         0  // All good :) | ||||
| #define ERR_EEP_COMMIT   2  // Could not commit to EEPROM (wrong flash layout?) | ||||
| #define ERR_JSON         9  // JSON parsing failed (input too large?) | ||||
| #define ERR_FS_BEGIN    10  // Could not init filesystem (no partition?) | ||||
| #define ERR_FS_QUOTA    11  // The FS is full or the maximum file size is reached | ||||
| #define ERR_FS_PLOAD    12  // It was attempted to load a preset that does not exist | ||||
| #define ERR_FS_GENERAL  19  // A general unspecified filesystem error occured | ||||
| #define ERR_OVERTEMP    30  // An attached temperature sensor has measured above threshold temperature (not implemented) | ||||
| #define ERR_OVERCURRENT 31  // An attached current sensor has measured a current above the threshold (not implemented) | ||||
| #define ERR_UNDERVOLT   32  // An attached voltmeter has measured a voltage below the threshold (not implemented) | ||||
|  | ||||
| //Timer mode types | ||||
| #define NL_MODE_SET               0            //After nightlight time elapsed, set to target brightness | ||||
| #define NL_MODE_FADE              1            //Fade to target brightness gradually | ||||
| #define NL_MODE_COLORFADE         2            //Fade to target brightness and secondary color gradually | ||||
| #define NL_MODE_SUN               3            //Sunrise/sunset. Target brightness is set immediately, then Sunrise effect is started. Max 60 min. | ||||
|  | ||||
|  | ||||
| #define NTP_PACKET_SIZE 48 | ||||
|  | ||||
| // maximum number of LEDs - more than 1500 LEDs (or 500 DMA "LEDPIN 3" driven ones) will cause a low memory condition on ESP8266 | ||||
| #ifndef MAX_LEDS | ||||
| #ifdef ESP8266 | ||||
| #define MAX_LEDS 8192 //rely on memory limit to limit this to 1600 LEDs | ||||
| #else | ||||
| #define MAX_LEDS 8192 | ||||
| #endif | ||||
| #endif | ||||
|  | ||||
| #ifndef MAX_LED_MEMORY | ||||
| #ifdef ESP8266 | ||||
| #define MAX_LED_MEMORY 5000 | ||||
| #else | ||||
| #define MAX_LED_MEMORY 64000 | ||||
| #endif | ||||
| #endif | ||||
|  | ||||
| #ifndef MAX_LEDS_PER_BUS | ||||
| #define MAX_LEDS_PER_BUS 4096 | ||||
| #endif | ||||
|  | ||||
| // string temp buffer (now stored in stack locally) | ||||
| #define OMAX 2048 | ||||
|  | ||||
| #define E131_MAX_UNIVERSE_COUNT 9 | ||||
|  | ||||
| #define ABL_MILLIAMPS_DEFAULT 850  // auto lower brightness to stay close to milliampere limit | ||||
|  | ||||
| // PWM settings | ||||
| #ifndef WLED_PWM_FREQ | ||||
| #ifdef ESP8266 | ||||
|   #define WLED_PWM_FREQ    880 //PWM frequency proven as good for LEDs | ||||
| #else | ||||
|   #define WLED_PWM_FREQ  19531 | ||||
| #endif | ||||
| #endif | ||||
|  | ||||
| #define TOUCH_THRESHOLD 32 // limit to recognize a touch, higher value means more sensitive | ||||
|  | ||||
| // Size of buffer for API JSON object (increase for more segments) | ||||
| #ifdef ESP8266 | ||||
|   #define JSON_BUFFER_SIZE 9216 | ||||
| #else | ||||
|   #define JSON_BUFFER_SIZE 16384 | ||||
| #endif | ||||
|  | ||||
| // Maximum size of node map (list of other WLED instances) | ||||
| #ifdef ESP8266 | ||||
|   #define WLED_MAX_NODES 15 | ||||
| #else | ||||
|   #define WLED_MAX_NODES 150 | ||||
| #endif | ||||
|  | ||||
| //this is merely a default now and can be changed at runtime | ||||
| #ifndef LEDPIN | ||||
| #define LEDPIN 2 | ||||
| #endif | ||||
|  | ||||
| #ifdef WLED_ENABLE_DMX | ||||
| #if (LEDPIN == 2) | ||||
|   #undef LEDPIN | ||||
|   #define LEDPIN 3 | ||||
|   #warning "Pin conflict compiling with DMX and LEDs on pin 2. The default LED pin has been changed to pin 3." | ||||
| #endif | ||||
| #endif | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										47
									
								
								wled00/data/404.htm
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								wled00/data/404.htm
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <head> | ||||
|   <meta charset="utf-8"> | ||||
| 	<meta content='width=device-width' name='viewport'> | ||||
|   <meta name="theme-color" content="#222222"> | ||||
| 	<title>Not found</title> | ||||
| 	<style> | ||||
| 		body { | ||||
| 			font-family: Verdana, Helvetica, sans-serif; | ||||
| 			text-align: center; | ||||
| 			background-color: #222; | ||||
| 			margin: 0; | ||||
| 			color: #fff; | ||||
| 		} | ||||
|  | ||||
|     img { | ||||
|       width: 400px; | ||||
|       max-width: 50%; | ||||
|       image-rendering: pixelated; | ||||
|       image-rendering: crisp-edges; | ||||
|       margin: 25px 0 -10px 0; | ||||
|     } | ||||
|      | ||||
|     button { | ||||
|       outline: none; | ||||
|       cursor: pointer; | ||||
|       padding: 8px; | ||||
|       margin: 10px; | ||||
|       width: 230px; | ||||
|       text-transform: uppercase; | ||||
|       font-family: helvetica; | ||||
|       font-size: 19px; | ||||
|       background-color: #333; | ||||
|       color: white; | ||||
|       border: 0px solid white; | ||||
|       border-radius: 25px; | ||||
|     } | ||||
| 	</style> | ||||
| </head> | ||||
| <body> | ||||
| <img alt="" src=" data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACAAAAAgCAYAAABzenr0AAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsEAAA7BAbiRa+0AAAFMSURBVFhH7ZTfbYNADMaPKDNUrJA85CEjdIjOgNQV+sASlZgmI/AIK6AuQfngnDrmjtpHItQ/P+l0juHsz2cH9+fJ/G7nreldfnDnp+ln/ZIlxbIfQmIwJOekCrEJ8FUvASEWEXoBiuSERcTO75uhuwFWff86bi57n3ZC+rW3YLqB5rn11ldCEPNr2LwFJgHHy8G1bTsu3oKYX4N5BrQ8ZAYewSoBGDjr0ElWCUC/rT2X7MqynL7tG4Dc45BwEYM9H5w7DqHMdfNCURR9nue3Iobk55MtOYeLoOQ8vmoG6o+0FaLrOm9FwC3wayLgx5I2WHpGIGYorulfgPYQ3AZLz4hQ9TMBVVVleJGrRUWz2YgQOg8bPjzzrit7vwcRQb5NTiARRPPzMYItoCpoWZITMkao+mRkddpqQ6z6FN+DfwFJrOm55GfewC/CuU/E4tQYg7BPYQAAAABJRU5ErkJggg=="> | ||||
| <h1>404 Not Found</h1> | ||||
| <b>Akemi does not know where you are headed...</b><br><br> | ||||
| <button onclick="window.location.href='/sliders'">Back to controls</button> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										28
									
								
								wled00/data/dmxmap.htm
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								wled00/data/dmxmap.htm
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| <!DOCTYPE html> | ||||
| <html><head><meta content='width=device-width' name='viewport'> | ||||
| <title>DMX Map</title> | ||||
| <script>function B(){window.history.back()};function RS(){window.location = "/settings";}function RP(){top.location.href="/";}function FM() { | ||||
|   var dmxlabels = ["SET 0","RED","GREEN","BLUE","WHITE","SHUTTER","SET 255", "DISABLED"]; | ||||
|   var dmxchans = []; | ||||
|   for (i=0;i<512;i++) { | ||||
|     dmxchans.push(7); // set all to DISABLED | ||||
|   } | ||||
|   for (i=0;i<LC;i++) { | ||||
|     FS = CS + (CG * i); | ||||
|     for (j=0;j<CN;j++) { | ||||
|       DA=FS+j; | ||||
|       dmxchans[DA-1] = CH[j]; | ||||
|     } | ||||
|   } | ||||
|   DMXMap = ""; | ||||
|   for (i=0;i<512;i++) { | ||||
|     isstart = ""; | ||||
|     if ((i+1) % 10 == 0) { | ||||
|       isstart="S" | ||||
|     } | ||||
|     DMXMap += "<div class=\"anytype " + isstart + " type" + dmxchans[i] + "\">" + String(i+1) + "<br />" + dmxlabels[dmxchans[i]] + "</div>"; | ||||
|   } | ||||
|   document.getElementById("map").innerHTML = DMXMap; | ||||
| }</script> | ||||
| <style>.anytype{border: 1px solid white; margin: 1px; float: left; width: 100px; height: 100px;}.S { margin: 0px; border: 2px solid white;} .type7{color: #888; border: 1px dotted grey;}.type6{color: #FFF;}.type4{color: #FFF; font-weight: bold; }.type3{color: #00F; font-weight: bold; }.type2{color: #0F0; font-weight: bold; }.type1{color: #F00; font-weight: bold; } .bt{background:#333;color:#fff;font-family:Verdana,sans-serif;border:.3ch solid #333;display:inline-block;font-size:20px;margin:8px;margin-top:12px}body{font-family:Verdana,sans-serif;text-align:center;background:#222;color:#fff;line-height:200%%;margin:0}</style></head> | ||||
| <body onload="FM();"><div id="map">...</div></body></html> | ||||
							
								
								
									
										1064
									
								
								wled00/data/index.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1064
									
								
								wled00/data/index.css
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										1778
									
								
								wled00/data/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1778
									
								
								wled00/data/index.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										7
									
								
								wled00/data/iro.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								wled00/data/iro.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										62
									
								
								wled00/data/liveviewws.htm
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								wled00/data/liveviewws.htm
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <head> | ||||
|   <meta name="viewport" content="width=device-width, initial-scale=1, minimum-scale=1"> | ||||
|   <meta charset="utf-8"> | ||||
|   <meta name="theme-color" content="#222222"> | ||||
|   <title>WLED Live Preview</title> | ||||
|   <style> | ||||
|   body { | ||||
|     margin: 0; | ||||
|   } | ||||
|   #canv { | ||||
|     background: black; | ||||
|     filter: brightness(175%); | ||||
|     width: 100%; | ||||
|     height: 100%; | ||||
|     position: absolute; | ||||
|   } | ||||
|   </style> | ||||
| </head> | ||||
| <body> | ||||
|   <div id="canv" /> | ||||
|   <script> | ||||
|     console.info("Live-Preview websocket opening"); | ||||
|     var socket = new WebSocket("ws://"+document.location.host+"/ws"); | ||||
|  | ||||
|     socket.onopen = function () { | ||||
|       console.info("Live-Preview websocket is opened"); | ||||
|       socket.send("{'lv':true}"); | ||||
|     } | ||||
|  | ||||
|     socket.onclose = function () { console.info("Live-Preview websocket is closing"); } | ||||
|  | ||||
|     socket.onerror = function (event) { console.error("Live-Preview websocket error:", event); } | ||||
|  | ||||
|     function updatePreview(leds) { | ||||
|       var str = "linear-gradient(90deg,"; | ||||
|       var len = leds.length;         | ||||
|       for (i = 0; i < len; i++) { | ||||
|         var leddata = leds[i]; | ||||
|         if (leddata.length > 6) leddata = leddata.substring(2); | ||||
|         str += "#" + leddata; | ||||
|         if (i < len -1) str += "," | ||||
|       } | ||||
|       str += ")"; | ||||
|       document.getElementById("canv").style.background = str; | ||||
|     } | ||||
|  | ||||
|     socket.onmessage = function (event) { | ||||
|       try { | ||||
|         var json = JSON.parse(event.data); | ||||
|         if (json && json.leds) { | ||||
|           requestAnimationFrame(function () {updatePreview(json.leds);}); | ||||
|         }  | ||||
|       } | ||||
|       catch (err) { | ||||
|         console.error("Live-Preview websocket error:",err); | ||||
|       }       | ||||
|     } | ||||
|   </script> | ||||
| </body> | ||||
| </html> | ||||
| @@ -1,56 +1,36 @@ | ||||
| <!DOCTYPE html> | ||||
| <html><head> | ||||
| <html> | ||||
|  | ||||
| <head> | ||||
| 	<meta content='width=device-width' name='viewport'> | ||||
| 	<title>WLED Message</title> | ||||
| 	<script> | ||||
| 	function B() { | ||||
| 		window.history.back(); | ||||
| 	} | ||||
| 	function RS() { | ||||
| 		window.location = "/settings"; | ||||
| 	} | ||||
| 	function RP() { | ||||
| 		top.location.href="/"; | ||||
| 	} | ||||
| 	</script> | ||||
| 	<script>function B() { window.history.back() }; function RS() { window.location = "/settings"; } function RP() { top.location.href = "/"; }</script> | ||||
| 	<style> | ||||
| 		:root { | ||||
| 			--aCol: #D9B310; | ||||
| 			--bCol: #0B3C5D; | ||||
| 			--cCol: #1D2731; | ||||
| 			--dCol: #328CC1; | ||||
| 			--sCol: #000; | ||||
| 			--tCol: #328CC1; | ||||
| 			--cFn: Verdana; | ||||
| 		} | ||||
| 		.bt { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--tCol); | ||||
| 			border: 0.3ch solid var(--bCol); | ||||
| 			background: #333; | ||||
| 			color: #fff; | ||||
| 			font-family: Verdana, sans-serif; | ||||
| 			border: .3ch solid #333; | ||||
| 			display: inline-block; | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 			font-family: var(--cFn), sans-serif; | ||||
| 			font-size: 20px; | ||||
| 			margin: 8px; | ||||
| 			margin-top: 12px; | ||||
| 			margin-top: 12px | ||||
| 		} | ||||
|     input[type=file] { | ||||
|       font-size: 16px; | ||||
|     } | ||||
|  | ||||
| 		body { | ||||
| 			font-family: var(--cFn), sans-serif; | ||||
| 			font-family: Verdana, sans-serif; | ||||
| 			text-align: center; | ||||
| 			background: var(--cCol); | ||||
| 			color: var(--tCol); | ||||
| 			line-height: 200%; | ||||
| 			margin: 0; | ||||
| 			background-attachment: fixed; | ||||
| 			background: #222; | ||||
| 			color: #fff; | ||||
| 			line-height: 200%%; | ||||
| 			margin: 0 | ||||
| 		} | ||||
| 	</style> | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
| <h2>Sample message.</h2> | ||||
| Sample detail. | ||||
| 	<h2>Sample Message.</h2> | ||||
| 	Sample Detail. | ||||
| </body> | ||||
| </html> | ||||
| 	 | ||||
|  | ||||
| </html> | ||||
							
								
								
									
										8
									
								
								wled00/data/rangetouch.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								wled00/data/rangetouch.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| // ========================================================================== | ||||
| // rangetouch.js v2.0.1 | ||||
| // Making <input type="range"> work on touch devices | ||||
| // https://github.com/sampotts/rangetouch | ||||
| // License: The MIT License (MIT) | ||||
| // ========================================================================== | ||||
| !function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t():"function"==typeof define&&define.amd?define("RangeTouch",t):(e=e||self).RangeTouch=t()}(this,(function(){"use strict";function e(e,t){for(var n=0;n<t.length;n++){var r=t[n];r.enumerable=r.enumerable||!1,r.configurable=!0,"value"in r&&(r.writable=!0),Object.defineProperty(e,r.key,r)}}function t(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function n(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,r)}return n}function r(e){for(var r=1;r<arguments.length;r++){var i=null!=arguments[r]?arguments[r]:{};r%2?n(Object(i),!0).forEach((function(n){t(e,n,i[n])})):Object.getOwnPropertyDescriptors?Object.defineProperties(e,Object.getOwnPropertyDescriptors(i)):n(Object(i)).forEach((function(t){Object.defineProperty(e,t,Object.getOwnPropertyDescriptor(i,t))}))}return e}var i={addCSS:!0,thumbWidth:15,watch:!0};function u(e,t){return function(){return Array.from(document.querySelectorAll(t)).includes(this)}.call(e,t)}var o=function(e){return null!=e?e.constructor:null},c=function(e,t){return!!(e&&t&&e instanceof t)},l=function(e){return null==e},a=function(e){return o(e)===Object},s=function(e){return o(e)===String},f=function(e){return Array.isArray(e)},h=function(e){return c(e,NodeList)},d=s,y=f,b=h,m=function(e){return c(e,Element)},g=function(e){return c(e,Event)},p=function(e){return l(e)||(s(e)||f(e)||h(e))&&!e.length||a(e)&&!Object.keys(e).length};function v(e,t){if(1>t){var n=function(e){var t="".concat(e).match(/(?:\.(\d+))?(?:[eE]([+-]?\d+))?$/);return t?Math.max(0,(t[1]?t[1].length:0)-(t[2]?+t[2]:0)):0}(t);return parseFloat(e.toFixed(n))}return Math.round(e/t)*t}return function(){function t(e,n){(function(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")})(this,t),m(e)?this.element=e:d(e)&&(this.element=document.querySelector(e)),m(this.element)&&p(this.element.rangeTouch)&&(this.config=r({},i,{},n),this.init())}return n=t,c=[{key:"setup",value:function(e){var n=1<arguments.length&&void 0!==arguments[1]?arguments[1]:{},o=null;if(p(e)||d(e)?o=Array.from(document.querySelectorAll(d(e)?e:'input[type="range"]')):m(e)?o=[e]:b(e)?o=Array.from(e):y(e)&&(o=e.filter(m)),p(o))return null;var c=r({},i,{},n);if(d(e)&&c.watch){var l=new MutationObserver((function(n){Array.from(n).forEach((function(n){Array.from(n.addedNodes).forEach((function(n){m(n)&&u(n,e)&&new t(n,c)}))}))}));l.observe(document.body,{childList:!0,subtree:!0})}return o.map((function(e){return new t(e,n)}))}},{key:"enabled",get:function(){return"ontouchstart"in document.documentElement}}],(o=[{key:"init",value:function(){t.enabled&&(this.config.addCSS&&(this.element.style.userSelect="none",this.element.style.webKitUserSelect="none",this.element.style.touchAction="manipulation"),this.listeners(!0),this.element.rangeTouch=this)}},{key:"destroy",value:function(){t.enabled&&(this.config.addCSS&&(this.element.style.userSelect="",this.element.style.webKitUserSelect="",this.element.style.touchAction=""),this.listeners(!1),this.element.rangeTouch=null)}},{key:"listeners",value:function(e){var t=this,n=e?"addEventListener":"removeEventListener";["touchstart","touchmove","touchend"].forEach((function(e){t.element[n](e,(function(e){return t.set(e)}),!1)}))}},{key:"get",value:function(e){if(!t.enabled||!g(e))return null;var n,r=e.target,i=e.changedTouches[0],u=parseFloat(r.getAttribute("min"))||0,o=parseFloat(r.getAttribute("max"))||100,c=parseFloat(r.getAttribute("step"))||1,l=r.getBoundingClientRect(),a=100/l.width*(this.config.thumbWidth/2)/100;return 0>(n=100/l.width*(i.clientX-l.left))?n=0:100<n&&(n=100),50>n?n-=(100-2*n)*a:50<n&&(n+=2*(n-50)*a),u+v(n/100*(o-u),c)}},{key:"set",value:function(e){t.enabled&&g(e)&&!e.target.disabled&&(e.preventDefault(),e.target.value=this.get(e),function(e,t){if(e&&t){var n=new Event(t,{bubbles:!0});e.dispatchEvent(n)}}(e.target,"touchend"===e.type?"change":"input"))}}])&&e(n.prototype,o),c&&e(n,c),t;var n,o,c}()})); | ||||
| //# sourceMappingURL=rangetouch.js.map | ||||
| @@ -1,36 +1,25 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <head><meta http-equiv="Content-Type" content="text/html; charset=windows-1252"> | ||||
| <html lang="en"> | ||||
| <head><meta charset="UTF-8"> | ||||
| 	<title>WLED Settings</title> | ||||
| 	<style> | ||||
| 		:root { | ||||
| 			--aCol: #D9B310; | ||||
| 			--bCol: #0B3C5D; | ||||
| 			--cCol: #1D2731; | ||||
| 			--dCol: #328CC1; | ||||
| 			--sCol: #000; | ||||
| 			--tCol: #328CC1; | ||||
| 			--cFn: Verdana; | ||||
| 		} | ||||
| 		body { | ||||
| 			text-align: center; | ||||
| 			background: var(--cCol); | ||||
| 			height: 100%; | ||||
| 			background: #222; | ||||
| 			height: 100px; | ||||
| 			margin: 0; | ||||
| 			background-attachment: fixed; | ||||
| 		} | ||||
| 		html { | ||||
| 			--h:11.55vh; | ||||
| 			--h: 11.55vh; | ||||
| 		} | ||||
| 		button { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--tCol); | ||||
| 			font-family: var(--cFn), Helvetica, sans-serif; | ||||
| 			border: 0.3ch solid var(--bCol); | ||||
| 			background: #333; | ||||
| 			color: #fff; | ||||
| 			font-family: Verdana, Helvetica, sans-serif; | ||||
| 			border: 0.3ch solid #333; | ||||
| 			display: inline-block; | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 			font-size: 8vmin; | ||||
| 			height:var(--h); | ||||
| 			height: var(--h); | ||||
| 			width: 95%; | ||||
| 			margin-top: 2.4vh; | ||||
| 		} | ||||
|   | ||||
							
								
								
									
										59
									
								
								wled00/data/settings_dmx.htm
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								wled00/data/settings_dmx.htm
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"><head><meta name="viewport" content="width=500"><meta charset="utf-8"><title>DMX Settings</title> | ||||
| <script> | ||||
| function GCH(num) { | ||||
|   d=document; | ||||
|   d.getElementById('dmxchannels').innerHTML += ""; | ||||
|   for (i=0;i<num;i++) { | ||||
|     d.getElementById('dmxchannels').innerHTML += "<span id=CH" + (i+1) + "s >Channel " + (i+1) + ": <select name=CH" + (i+1) + " id=\"CH" + (i+1) + "\"><option value=0>Set to 0</option><option value=1>Red</option><option value=2>Green</option><option value=3>Blue</option><option value=4>White</option><option value=5>Shutter (Brightness)</option><option value=6>Set to 255</option></select></span><br />\n"; | ||||
|   } | ||||
| } | ||||
| function mMap(){ | ||||
|   d=document; | ||||
|   numCh=document.Sf.CN.value; | ||||
|   numGap=document.Sf.CG.value; | ||||
|   if (parseInt(numCh)>parseInt(numGap)) { | ||||
|     d.getElementById("gapwarning").style.display="block"; | ||||
|   } else { | ||||
|     d.getElementById("gapwarning").style.display="none"; | ||||
|   } | ||||
|   for (i=0;i<15;i++) { | ||||
|     if (i>=numCh) { | ||||
|       d.getElementById("CH"+(i+1) + "s").style.opacity = "0.5"; | ||||
|       d.getElementById("CH"+(i+1)).disabled = true; | ||||
|        | ||||
|     } else { | ||||
|       d.getElementById("CH"+(i+1) + "s").style.opacity = "1"; | ||||
|       d.getElementById("CH"+(i+1)).disabled = false; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| function S(){GCH(15);GetV();mMap();}function H(){window.open("https://github.com/Aircoookie/WLED/wiki/DMX");}function B(){window.history.back();} | ||||
| function GetV(){} | ||||
| </script> | ||||
| <style> | ||||
| 	@import url("style.css"); | ||||
| </style> | ||||
| </head> | ||||
| <body onload="S()"> | ||||
| <form id="form_s" name="Sf" method="post"> | ||||
| <div class="helpB"><button type="button" onclick="H()">?</button></div> | ||||
| <button type="button" onclick="B()">Back</button><button type="submit">Save</button><hr> | ||||
| <h2>Imma firin ma lazer (if it has DMX support)</h2><!-- TODO: Change to something less-meme-related //--> | ||||
|  | ||||
| Proxy Universe <input name=PU type=number min=0 max=63999 required> from E1.31 to DMX (0=disabled)<br> | ||||
| <i>This will disable the LED data output to DMX configurable below</i><br><br> | ||||
| <i>Number of fixtures is taken from LED config page</i><br> | ||||
|  | ||||
| Channels per fixture (15 max): <input type="number" min="1" max="15" name="CN" maxlength="2" onchange="mMap();"><br /> | ||||
| Start channel: <input type="number" min="1" max="512" name="CS" maxlength="2"><br /> | ||||
| Spacing between start channels: <input type="number" min="1" max="512" name="CG" maxlength="2" onchange="mMap();"> [ <a href="javascript:alert('if set to 10, first fixture will start at 10,\nsecond will start at 20 etc.\nRegardless of the channel count.\nMakes memorizing channel numbers easier.');">info</a> ]<br> | ||||
| <div id="gapwarning" style="color: orange; display: none;">WARNING: Channel gap is lower than channels per fixture.<br />This will cause overlap.</div> | ||||
| <button type="button" onclick="location.href='/dmxmap';">DMX Map</button><br> | ||||
| DMX fixtures start LED: <input type="number" min="0" max="1500" name="SL"> | ||||
| <h3>Channel functions</h3> | ||||
| <div id="dmxchannels"></div> | ||||
| <hr><button type="button" onclick="B()">Back</button><button type="submit">Save</button> | ||||
| </form> | ||||
| </body> | ||||
| </html> | ||||
										
											Binary file not shown.
										
									
								
							| @@ -1,7 +1,8 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <html lang="en"> | ||||
| <head> | ||||
| 	<meta name="viewport" content="width=500"> | ||||
| 	<meta charset="utf-8"> | ||||
| 	<title>Misc Settings</title> | ||||
| 	<script> | ||||
| 		function H() | ||||
| @@ -22,51 +23,7 @@ | ||||
| 		} | ||||
| 	</script> | ||||
| 	<style> | ||||
| 		:root { | ||||
| 			--aCol: #D9B310; | ||||
| 			--bCol: #0B3C5D; | ||||
| 			--cCol: #1D2731; | ||||
| 			--dCol: #328CC1; | ||||
| 			--sCol: #000; | ||||
| 		} | ||||
| 		body { | ||||
| 			font-family: Verdana, Helvetica, sans-serif; | ||||
| 			text-align: center; | ||||
| 			background: var(--cCol); | ||||
| 			color: var(--dCol); | ||||
| 			line-height: 2em; | ||||
| 			font-size: 16px; | ||||
| 			margin: 0; | ||||
| 			background-attachment: fixed; | ||||
| 		} | ||||
| 		hr { | ||||
| 			border-color: var(--dCol); | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 		} | ||||
| 		button { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--dCol); | ||||
| 			border: 0.3ch solid var(--bCol); | ||||
| 			display: inline-block; | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 			font-size: 20px; | ||||
| 			margin: 8px; | ||||
| 			margin-top: 12px; | ||||
| 		} | ||||
| 		.helpB { | ||||
| 			text-align: left; | ||||
| 			position: absolute; | ||||
| 			width:60px; | ||||
| 		} | ||||
| 		input { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--dCol); | ||||
| 			border: 0.5ch solid var(--bCol); | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 		} | ||||
| 		input[type=number] { | ||||
| 			width: 3em; | ||||
| 		} | ||||
| 		@import url("style.css"); | ||||
| 	</style> | ||||
| </head> | ||||
| <body onload="GetV()"> | ||||
| @@ -88,12 +45,12 @@ | ||||
| 		<button type="button" onclick="U()">Manual OTA Update</button><br> | ||||
| 		Enable ArduinoOTA: <input type="checkbox" name="AO"><br> | ||||
| 		<h3>About</h3> | ||||
| 		<a href="https://github.com/Aircoookie/WLED" target="_blank">WLED</a> version 0.9.0<br><br> | ||||
| 		<a href="https://github.com/Aircoookie/WLED/wiki/Contributors-&-About" target="_blank">Contributors, dependencies and special thanks</a><br> | ||||
| 		<a href="https://github.com/Aircoookie/WLED/" target="_blank">WLED</a> version ##VERSION##<!-- Autoreplaced from package.json --><br><br> | ||||
| 		<a href="https://github.com/Aircoookie/WLED/wiki/Contributors-and-credits" target="_blank">Contributors, dependencies and special thanks</a><br> | ||||
| 		A huge thank you to everyone who helped me create WLED!<br><br> | ||||
| 		(c) 2016-2019 Christian Schwinne <br> | ||||
| 		<i>Licensed under the MIT license</i><br><br> | ||||
| 		Server message: <span class="msg"> Response error! </span><hr> | ||||
| 		(c) 2016-2021 Christian Schwinne <br> | ||||
| 		<i>Licensed under the <a href="https://github.com/Aircoookie/WLED/blob/master/LICENSE" target="_blank">MIT license</a></i><br><br> | ||||
| 		Server message: <span class="sip"> Response error! </span><hr> | ||||
| 		<button type="button" onclick="B()">Back</button><button type="submit">Save & Reboot</button> | ||||
| 	</form> | ||||
| </body> | ||||
|   | ||||
| @@ -1,140 +1,116 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <head> | ||||
| 	<meta name="viewport" content="width=500"> | ||||
| 	<title>Sync Settings</title> | ||||
| 	<script> | ||||
| 		function H() | ||||
| 		{ | ||||
| 			window.open("https://github.com/Aircoookie/WLED/wiki/Settings#sync-settings"); | ||||
| 		} | ||||
| 		function B() | ||||
| 		{ | ||||
| 			window.open("/settings","_self"); | ||||
| 		} | ||||
| 		function GetV() | ||||
| 		{ | ||||
| 			//values injected by server while sending HTML | ||||
| 		} | ||||
| 	</script> | ||||
| 	<style> | ||||
| 		:root { | ||||
| 			--aCol: #abc; | ||||
| 			--bCol: #fff; | ||||
| 			--cCol: #ddd; | ||||
| 			--dCol: #000; | ||||
| 			--sCol: #0004; | ||||
| 		} | ||||
| 		body { | ||||
| 			font-family: Verdana, Helvetica, sans-serif; | ||||
| 			text-align: center; | ||||
| 			background: var(--cCol); | ||||
| 			color: var(--dCol); | ||||
| 			line-height: 200%; | ||||
| 			margin: 0; | ||||
| 			background-attachment: fixed; | ||||
| 		} | ||||
| 		hr { | ||||
| 			border-color: var(--dCol); | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 		} | ||||
| 		button { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--dCol); | ||||
| 			border: 0.3ch solid var(--bCol); | ||||
| 			display: inline-block; | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 			font-size: 20px; | ||||
| 			margin: 8px; | ||||
| 			margin-top: 12px; | ||||
| 		} | ||||
| 		.helpB { | ||||
| 			text-align: left; | ||||
| 			position: absolute; | ||||
| 			width:60px; | ||||
| 		} | ||||
| 		input { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--dCol); | ||||
| 			border: 0.5ch solid var(--bCol); | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 		} | ||||
| 		input[type=number] { | ||||
| 			width: 3em; | ||||
| 		} | ||||
| 	</style> | ||||
| </head> | ||||
| <body onload="GetV()"> | ||||
| 	<form id="form_s" name="Sf" method="post"> | ||||
| 		<div class="helpB"><button type="button" onclick="H()">?</button></div> | ||||
| 		<button type="button" onclick="B()">Back</button><button type="submit">Save</button><hr> | ||||
| 		<h2>Sync setup</h2> | ||||
| 		<h3>Button setup</h3> | ||||
| 		On/Off button enabled: <input type="checkbox" name="BT"><br> | ||||
| 		Infrared receiver enabled: <input type="checkbox" name="IR"><br> | ||||
| 		<a href="https://github.com/Aircoookie/WLED/wiki/Infrared-Control" target="_blank">IR info</a> | ||||
| 		<h3>WLED Sync UDP Broadcast</h3> | ||||
| 		UDP Port: <input name="UP" type="number" min="1" max="65535" required><br> | ||||
| 		Receive <input type="checkbox" name="RB">Brightness, <input type="checkbox" name="RC">Color, and <input type="checkbox" name="RX">Effects<br> | ||||
| 		Send notifications on direct change: <input type="checkbox" name="SD"><br> | ||||
| 		Send notifications on button press: <input type="checkbox" name="SB"><br> | ||||
| 		Send Alexa notifications: <input type="checkbox" name="SA"><br> | ||||
| 		Send Philips Hue change notifications: <input type="checkbox" name="SH"><br> | ||||
|     Send Macro notifications: <input type="checkbox" name="SM"><br> | ||||
| 		Send notifications twice: <input type="checkbox" name="S2"> | ||||
| 		<h3>Realtime</h3> | ||||
| 		Receive UDP realtime: <input type="checkbox" name="RD"><br><br> | ||||
| 		E1.31 (sACN)<br> | ||||
| 		Multicast mode: <input type="checkbox" name="EM"><br> | ||||
| 		E1.31 start universe: <input name="EU" type="number" min="1" max="63999" required><br> | ||||
| 		<i>Reboot required.</i> Check out <a href="https://github.com/ahodges9/LedFx" target="_blank">LedFx</a>!<br><br> | ||||
| 		DMX start address: <input name="DA" type="number" min="1" max="510" value="1" required><br> | ||||
| 		DMX mode: <input name="DM" type="radio" value="0"> disabled<br> | ||||
| 		<input name="DM" type="radio" value="1"> Single RGB (3 Channels for all LEDs: Red Green Blue)<br> | ||||
| 		<input name="DM" type="radio" value="2"> Single DRGB (4 Channels for all LEDs: Dimmer Red Green Blue)<br> | ||||
| 		<input name="DM" type="radio" value="3"> Effect (11 Channels parametrizing Effects: Dimmer Effect Speed Intensity Palette PriRed PriGreen PriBlue SecRed SecGreen SecBlue)<br> | ||||
| 		<input name="DM" type="radio" value="4"> Multiple RGB (3 Channels for each LED: Red Green Blue)<br> | ||||
| 		<input name="DM" type="radio" value="5"> Multiple DRGB (1+3 Channels for each LED: Dimmer Red1 Green1 Blue1 Red2 Green2 Blue2...)<br> | ||||
| 		<i>Reboot required.</i> Check out <a href="https://github.com/ahodges9/LedFx" target="_blank">LedFx</a>!<br><br> | ||||
| 		Timeout: <input name="ET" type="number" min="100" max="65000" required> ms<br> | ||||
| 		Force max brightness: <input type="checkbox" name="FB"><br> | ||||
| 		Disable realtime gamma correction: <input type="checkbox" name="RG"><br> | ||||
| 		Realtime LED offset: <input name="WO" type="number" min="-255" max="255" required> | ||||
| 		<h3>Alexa Voice Assistant</h3> | ||||
| 		Emulate Alexa device: <input type="checkbox" name="AL"><br> | ||||
| 		Alexa invocation name: <input name="AI" maxlength="32"> | ||||
| 		<h3>Blynk</h3> | ||||
| 		<b>Blynk, MQTT and Hue sync all connect to external hosts!<br> | ||||
| 		This may impact the responsiveness of the ESP8266.</b><br> | ||||
| 		For best results, only use one of these services at a time.<br> | ||||
| 		(alternatively, connect a second ESP to them and use the UDP sync)<br><br> | ||||
| 		Device Auth token: <input name="BK" maxlength="33"><br> | ||||
| 		<i>Clear the token field to disable. </i><a href="https://github.com/Aircoookie/WLED/wiki/Blynk" target="_blank">Setup info</a> | ||||
| 		<h3>MQTT</h3> | ||||
|     Enable MQTT: <input type="checkbox" name="MQ"><br> | ||||
| 		Broker: <input name="MS" maxlength="32"><br> | ||||
|     Port: <input name="MQPORT" type="number" min="1" max="65535"><br> | ||||
|     <b>The MQTT credentials are sent over an unsecured connection.<br> | ||||
|     Never use the MQTT password for another service!</b><br> | ||||
| 		Username: <input name="MQTTUSER" maxlength="32"><br> | ||||
| 		Password: <input type="password" input name="MQTTPASS" maxlength="32"><br> | ||||
| 		Client ID: <input name="MQTTCID" maxlength="32"><br> | ||||
| 		Device Topic: <input name="MD" maxlength="32"><br> | ||||
| 		Group Topic: <input name="MG" maxlength="32"><br> | ||||
| 		<a href="https://github.com/Aircoookie/WLED/wiki/MQTT" target="_blank">MQTT info</a> | ||||
| 		<h3>Philips Hue</h3> | ||||
| 		<i>You can find the bridge IP and the light number in the 'About' section of the hue app.</i><br> | ||||
| 		Poll Hue light <input name="HL" type="number" min="1" max="99"> every <input name="HI" type="number" min="100" max="65000"> ms: <input type="checkbox" name="HP"><br> | ||||
| 		Then, receive <input type="checkbox" name="HO"> On/Off, <input type="checkbox" name="HB"> Brightness, and <input type="checkbox" name="HC"> Color<br> | ||||
| 		Hue Bridge IP:<br> | ||||
| 		<input name="H0" type="number" min="0" max="255"> . | ||||
| 		<input name="H1" type="number" min="0" max="255"> . | ||||
| 		<input name="H2" type="number" min="0" max="255"> . | ||||
| 		<input name="H3" type="number" min="0" max="255"><br> | ||||
| 		<b>Press the pushlink button on the bridge, after that save this page!</b><br> | ||||
| 		(when first connecting)<br> | ||||
| 		Hue status: <span class="hms"> Internal ESP Error! </span><hr> | ||||
| 		<button type="button" onclick="B()">Back</button><button type="submit">Save</button> | ||||
| 	</form> | ||||
| <html lang="en"><head><meta name="viewport" content="width=500"><meta charset="utf-8"><title>Sync Settings</title> | ||||
| <script>var d=document; | ||||
| function H(){window.open("https://github.com/Aircoookie/WLED/wiki/Settings#sync-settings");}function B(){window.open("/settings","_self");} | ||||
| function adj(){if (d.Sf.DI.value == 6454) {if (d.Sf.DA.value == 1) d.Sf.DA.value = 0; if (d.Sf.EU.value == 1) d.Sf.EU.value = 0;} | ||||
|                else if (d.Sf.DI.value == 5568) {if (d.Sf.DA.value == 0) d.Sf.DA.value = 1; if (d.Sf.EU.value == 0) d.Sf.EU.value = 1;} } | ||||
| function SP(){var p = d.Sf.DI.value; d.getElementById("xp").style.display = (p > 0)?"none":"block"; if (p > 0) d.Sf.EP.value = p;} | ||||
| function SetVal(){switch(parseInt(d.Sf.EP.value)){case 5568: d.Sf.DI.value = 5568; break; case 6454: d.Sf.DI.value = 6454; break; case 4048: d.Sf.DI.value = 4048; break; }; SP();} | ||||
| function S(){GetV();SetVal();} | ||||
| function GetV(){var d=document;} | ||||
| </script> | ||||
| <style>@import url("style.css");</style></head> | ||||
| <body onload="S()"> | ||||
| <form id="form_s" name="Sf" method="post"> | ||||
| <div class="helpB"><button type="button" onclick="H()">?</button></div> | ||||
| <button type="button" onclick="B()">Back</button><button type="submit">Save</button><hr> | ||||
| <h2>Sync setup</h2> | ||||
| <h3>Button setup</h3> | ||||
| On/Off button enabled: <input type="checkbox" name="BT"><br> | ||||
| Infrared remote: | ||||
| <select name=IR> | ||||
| <option value=0>Disabled</option> | ||||
| <option value=1>24-key RGB</option> | ||||
| <option value=2>24-key with CT</option> | ||||
| <option value=3>40-key blue</option> | ||||
| <option value=4>44-key RGB</option> | ||||
| <option value=5>21-key RGB</option> | ||||
| <option value=6>6-key black</option> | ||||
| <option value=7>9-key red</option> | ||||
| </select><br> | ||||
| <a href="https://github.com/Aircoookie/WLED/wiki/Infrared-Control" target="_blank">IR info</a> | ||||
| <h3>WLED Broadcast</h3> | ||||
| UDP Port: <input name="UP" type="number" min="1" max="65535" class="d5" required><br> | ||||
| 2nd Port: <input name="U2" type="number" min="1" max="65535" class="d5" required><br> | ||||
| Receive <input type="checkbox" name="RB">Brightness, <input type="checkbox" name="RC">Color, and <input type="checkbox" name="RX">Effects<br> | ||||
| Send notifications on direct change: <input type="checkbox" name="SD"><br> | ||||
| Send notifications on button press or IR: <input type="checkbox" name="SB"><br> | ||||
| Send Alexa notifications: <input type="checkbox" name="SA"><br> | ||||
| Send Philips Hue change notifications: <input type="checkbox" name="SH"><br> | ||||
| Send Macro notifications: <input type="checkbox" name="SM"><br> | ||||
| Send notifications twice: <input type="checkbox" name="S2"><br> | ||||
| <i>Reboot required to apply changes. </i> | ||||
| <h3>Instance List</h3> | ||||
| Enable instance list: <input type="checkbox" name="NL"><br> | ||||
| Make this instance discoverable: <input type="checkbox" name="NB"><br> | ||||
| <h3>Realtime</h3> | ||||
| Receive UDP realtime: <input type="checkbox" name="RD"><br><br> | ||||
| <i>Network DMX input</i><br> | ||||
| Type: | ||||
| <select name=DI onchange="SP(); adj();"> | ||||
| <option value=5568>E1.31 (sACN)</option> | ||||
| <option value=6454>Art-Net</option> | ||||
| <option value=4048>DDP</option> | ||||
| <option value=0 selected>Custom port</option> | ||||
| </select><br> | ||||
| <div id=xp>Port: <input name="EP" type="number" min="1" max="65535" value="5568" class="d5" required><br></div> | ||||
| Multicast: <input type="checkbox" name="EM"><br> | ||||
| Start universe: <input name="EU" type="number" min="0" max="63999" required><br> | ||||
| <i>Reboot required.</i> Check out <a href="https://github.com/LedFx/LedFx" target="_blank">LedFx</a>!<br> | ||||
| Skip out-of-sequence packets: <input type="checkbox" name="ES"><br> | ||||
| DMX start address: <input name="DA" type="number" min="0" max="510" required><br> | ||||
| DMX mode: | ||||
| <select name=DM> | ||||
| <option value=0>Disabled</option> | ||||
| <option value=1>Single RGB</option> | ||||
| <option value=2>Single DRGB</option> | ||||
| <option value=3>Effect</option> | ||||
| <option value=4>Multi RGB</option> | ||||
| <option value=5>Dimmer + Multi RGB</option> | ||||
| <option value=6>Multi RGBW</option> | ||||
| </select><br> | ||||
| <a href="https://github.com/Aircoookie/WLED/wiki/E1.31-DMX" target="_blank">E1.31 info</a><br> | ||||
| Timeout: <input name="ET" type="number" min="1" max="65000" required> ms<br> | ||||
| Force max brightness: <input type="checkbox" name="FB"><br> | ||||
| Disable realtime gamma correction: <input type="checkbox" name="RG"><br> | ||||
| Realtime LED offset: <input name="WO" type="number" min="-255" max="255" required> | ||||
| <h3>Alexa Voice Assistant</h3> | ||||
| Emulate Alexa device: <input type="checkbox" name="AL"><br> | ||||
| Alexa invocation name: <input name="AI" maxlength="32"> | ||||
| <h3>Blynk</h3> | ||||
| <b>Blynk, MQTT and Hue sync all connect to external hosts!<br> | ||||
| This may impact the responsiveness of the ESP8266.</b><br> | ||||
| For best results, only use one of these services at a time.<br> | ||||
| (alternatively, connect a second ESP to them and use the UDP sync)<br><br> | ||||
| Host: <input name="BH" maxlength="32"> | ||||
| Port: <input name="BP" type="number" min="1" max="65535" value="80" class="d5"><br> | ||||
| Device Auth token: <input name="BK" maxlength="33"><br> | ||||
| <i>Clear the token field to disable. </i><a href="https://github.com/Aircoookie/WLED/wiki/Blynk" target="_blank">Setup info</a> | ||||
| <h3>MQTT</h3> | ||||
| Enable MQTT: <input type="checkbox" name="MQ"><br> | ||||
| Broker: <input name="MS" maxlength="32"> | ||||
| Port: <input name="MQPORT" type="number" min="1" max="65535" class="d5"><br> | ||||
| <b>The MQTT credentials are sent over an unsecured connection.<br> | ||||
| Never use the MQTT password for another service!</b><br> | ||||
| Username: <input name="MQUSER" maxlength="40"><br> | ||||
| Password: <input type="password" name="MQPASS" maxlength="40"><br> | ||||
| Client ID: <input name="MQCID" maxlength="40"><br> | ||||
| Device Topic: <input name="MD" maxlength="32"><br> | ||||
| Group Topic: <input name="MG" maxlength="32"><br> | ||||
| <i>Reboot required to apply changes. </i><a href="https://github.com/Aircoookie/WLED/wiki/MQTT" target="_blank">MQTT info</a> | ||||
| <h3>Philips Hue</h3> | ||||
| <i>You can find the bridge IP and the light number in the 'About' section of the hue app.</i><br> | ||||
| Poll Hue light <input name="HL" type="number" min="1" max="99" > every <input name="HI" type="number" min="100" max="65000"> ms: <input type="checkbox" name="HP"><br> | ||||
| Then, receive <input type="checkbox" name="HO"> On/Off, <input type="checkbox" name="HB"> Brightness, and <input type="checkbox" name="HC"> Color<br> | ||||
| Hue Bridge IP:<br> | ||||
| <input name="H0" type="number" min="0" max="255" > . | ||||
| <input name="H1" type="number" min="0" max="255" > . | ||||
| <input name="H2" type="number" min="0" max="255" > . | ||||
| <input name="H3" type="number" min="0" max="255" ><br> | ||||
| <b>Press the pushlink button on the bridge, after that save this page!</b><br> | ||||
| (when first connecting)<br> | ||||
| Hue status: <span class="sip"> Disabled in this build </span><hr> | ||||
| <button type="button" onclick="B()">Back</button><button type="submit">Save</button> | ||||
| </form> | ||||
| </body> | ||||
| </html> | ||||
| @@ -1,7 +1,8 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <html lang="en"> | ||||
| <head> | ||||
| 	<meta name="viewport" content="width=500"> | ||||
| 	<meta charset="utf-8"> | ||||
| 	<title>Time Settings</title> | ||||
| 	<script> | ||||
|     var d=document; | ||||
| @@ -42,10 +43,10 @@ | ||||
| 		} | ||||
|     function BTa() | ||||
|     { | ||||
|       var ih="<tr><th>Active</th><th>Hour</th><th>Minute</th><th>Macro</th><th>M</th><th>T</th><th>W</th><th>T</th><th>F</th><th>S</th><th>S</th></tr>"; | ||||
|       var ih="<tr><th>Active</th><th>Hour</th><th>Minute</th><th>Preset</th><th>M</th><th>T</th><th>W</th><th>T</th><th>F</th><th>S</th><th>S</th></tr>"; | ||||
|       for (i=0;i<8;i++) | ||||
|       { | ||||
|         ih+="<tr><td><input name=\"W"+i+"\" id=\"W"+i+"\" type=\"number\" style=\"display:none\"><input id=\"W"+i+"0\" type=\"checkbox\"></td><td><input name=\"H"+i+"\" type=\"number\" min=\"0\" max=\"24\"></td><td><input name=\"N"+i+"\" type=\"number\" min=\"0\" max=\"59\"></td><td><input name=\"T"+i+"\" type=\"number\" min=\"0\" max=\"16\"></td>"; | ||||
|         ih+="<tr><td><input name=\"W"+i+"\" id=\"W"+i+"\" type=\"number\" style=\"display:none\"><input id=\"W"+i+"0\" type=\"checkbox\"></td><td><input name=\"H"+i+"\" type=\"number\" min=\"0\" max=\"24\"></td><td><input name=\"N"+i+"\" type=\"number\" min=\"0\" max=\"59\"></td><td><input name=\"T"+i+"\" type=\"number\" min=\"0\" max=\"250\"></td>"; | ||||
|         for (j=1;j<8;j++) ih+="<td><input id=\"W"+i+j+"\" type=\"checkbox\"></td>"; | ||||
|       } | ||||
|       gId("TMT").innerHTML=ih; | ||||
| @@ -76,64 +77,7 @@ | ||||
| 		} | ||||
| 	</script> | ||||
| 	<style> | ||||
| 		:root { | ||||
| 			--aCol: #D9B310; | ||||
| 			--bCol: #0B3C5D; | ||||
| 			--cCol: #1D2731; | ||||
| 			--dCol: #328CC1; | ||||
| 			--sCol: #000; | ||||
| 			--tCol: #328CC1; | ||||
| 			--cFn: Verdana; | ||||
| 		} | ||||
| 		body { | ||||
| 			font-family: var(--cFn), sans-serif; | ||||
| 			text-align: center; | ||||
| 			background: var(--cCol); | ||||
| 			color: var(--dCol); | ||||
| 			line-height: 200%; | ||||
| 			margin: 0; | ||||
| 			background-attachment: fixed; | ||||
| 		} | ||||
| 		hr { | ||||
| 			border-color: var(--dCol); | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 		} | ||||
| 		button { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--dCol); | ||||
| 			border: 0.3ch solid var(--bCol); | ||||
| 			font-family: var(--cFn), sans-serif; | ||||
| 			display: inline-block; | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 			font-size: 20px; | ||||
| 			margin: 8px; | ||||
| 			margin-top: 12px; | ||||
| 		} | ||||
| 		.helpB { | ||||
| 			text-align: left; | ||||
| 			position: absolute; | ||||
| 			width:60px; | ||||
| 		} | ||||
| 		input { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--dCol); | ||||
| 			font-family: var(--cFn), sans-serif; | ||||
| 			border: 0.5ch solid var(--bCol); | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 		} | ||||
| 		select { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--dCol); | ||||
| 			border: 0.5ch solid var(--bCol); | ||||
| 			font-family: var(--cFn), sans-serif; | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 		} | ||||
|     td { | ||||
|       padding:2px; | ||||
|     } | ||||
| 		input[type=number] { | ||||
| 			width: 4em; | ||||
| 		} | ||||
| 		@import url("style.css"); | ||||
| 	</style> | ||||
| </head> | ||||
| <body onload="S()"> | ||||
| @@ -159,7 +103,12 @@ | ||||
| 			<option value="10">JST(KST)</option> | ||||
| 			<option value="11">AEST/AEDT</option> | ||||
| 			<option value="12">NZST/NZDT</option> | ||||
| 			<option value="13">North Korea</option> | ||||
|       <option value="13">North Korea</option> | ||||
|       <option value="14">IST (India)</option> | ||||
|       <option value="15">CA-Saskatchewan</option> | ||||
|       <option value="16">ACST</option> | ||||
|       <option value="17">ACST/ACDT</option> | ||||
|       <option value="18">HST (Hawaii)</option> | ||||
| 		</select><br> | ||||
| 		UTC offset: <input name="UO" type="number" min="-65500" max="65500" required> seconds (max. 18 hours)<br> | ||||
| 		Current local time is <span class="times">unknown</span>. | ||||
| @@ -186,31 +135,17 @@ | ||||
| 		Countdown Goal:<br> | ||||
| 		Year: 20 <input name="CY" type="number" min="0" max="99" required> Month: <input name="CI" type="number" min="1" max="12" required> Day: <input name="CD" type="number" min="1" max="31" required><br> | ||||
| 		Hour: <input name="CH" type="number" min="0" max="23" required> Minute: <input name="CM" type="number" min="0" max="59" required> Second: <input name="CS" type="number" min="0" max="59" required><br> | ||||
| 		<h3>Advanced Macros</h3> | ||||
| 		Define API macros here:<br> | ||||
| 		1: <input name="M1" maxlength="64"><br> | ||||
| 		2: <input name="M2" maxlength="64"><br> | ||||
| 		3: <input name="M3" maxlength="64"><br> | ||||
| 		4: <input name="M4" maxlength="64"><br> | ||||
| 		5: <input name="M5" maxlength="64"><br> | ||||
| 		6: <input name="M6" maxlength="64"><br> | ||||
| 		7: <input name="M7" maxlength="64"><br> | ||||
| 		8: <input name="M8" maxlength="64"><br> | ||||
| 		9: <input name="M9" maxlength="64"><br> | ||||
| 		10: <input name="M10" maxlength="64"><br> | ||||
| 		11: <input name="M11" maxlength="64"><br> | ||||
| 		12: <input name="M12" maxlength="64"><br> | ||||
| 		13: <input name="M13" maxlength="64"><br> | ||||
| 		14: <input name="M14" maxlength="64"><br> | ||||
| 		15: <input name="M15" maxlength="64"><br> | ||||
| 		16: <input name="M16" maxlength="64"><br><br> | ||||
| 		<i>Use 0 for the default action instead of a macro</i><br> | ||||
| 		Boot Macro: <input name="MB" type="number" min="0" max="16" required><br> | ||||
| 		Alexa On/Off Macros: <input name="A0" type="number" min="0" max="16" required> <input name="A1" type="number" min="0" max="16" required><br> | ||||
| 		Button Macro: <input name="MP" type="number" min="0" max="16" required> Long Press: <input name="ML" type="number" min="0" max="16" required><br> | ||||
| 		Countdown-Over Macro: <input name="MC" type="number" min="0" max="16" required><br> | ||||
| 		Timed-Light-Over Macro: <input name="MN" type="number" min="0" max="16" required><br> | ||||
| 		Time-Controlled Macros:<br> | ||||
| 		<h3>Macro presets</h3> | ||||
|     <b>Macros have moved!</b><br> | ||||
|     <i>Presets now also can be used as macros to save both JSON and HTTP API commands.<br> | ||||
|     Just enter the preset id below!</i> | ||||
| 		<i>Use 0 for the default action instead of a preset</i><br> | ||||
| 		Alexa On/Off Preset: <input name="A0" type="number" min="0" max="250" required> <input name="A1" type="number" min="0" max="250" required><br> | ||||
| 		Button short press Preset: <input name="MP" type="number" min="0" max="250" required><br> | ||||
|     Long Press: <input name="ML" type="number" min="0" max="250" required> Double press: <input name="MD" type="number" min="0" max="250" required><br>	 | ||||
| 		Countdown-Over Preset: <input name="MC" type="number" min="0" max="250" required><br> | ||||
| 		Timed-Light-Over Presets: <input name="MN" type="number" min="0" max="250" required><br> | ||||
| 		<h3>Time-controlled presets</h3> | ||||
|     <div style="display: inline-block"> | ||||
|     <table id="TMT"> | ||||
|     </table></div><hr> | ||||
|   | ||||
| @@ -1,16 +1,154 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <head> | ||||
| <head lang="en"> | ||||
|   <meta charset="utf-8"> | ||||
| 	<meta name="viewport" content="width=500"> | ||||
| 	<title>UI Settings</title> | ||||
| 	<script> | ||||
| 		var d = document; | ||||
| 		var initial_ds, initial_st; | ||||
| 		var sett = null; | ||||
| 		var l = { | ||||
| 			"comp":{ | ||||
| 			"labels":"Show button labels", | ||||
| 			"colors":{ | ||||
| 				"LABEL":"Color selection methods", | ||||
| 				"picker": "Color Wheel", | ||||
| 				"rgb": "RGB sliders", | ||||
| 				"quick": "Quick color selectors", | ||||
| 				"hex": "HEX color input" | ||||
| 			}, | ||||
|       "pcmbot": "Show bottom tab bar in PC mode", | ||||
|       "pid": "Show preset IDs" | ||||
| 			}, | ||||
| 			"theme":{ | ||||
| 				"alpha": { | ||||
| 					"bg":"Background opacity", | ||||
| 					"tab":"Button opacity" | ||||
| 				}, | ||||
| 				"bg":{ | ||||
| 					"url":"BG image URL", | ||||
| 					"random":"Random BG image" | ||||
| 				}, | ||||
| 				"color":{ | ||||
| 					"bg":"BG HEX color" | ||||
| 				} | ||||
| 			} | ||||
| 			 | ||||
| 		}; | ||||
| 		function gId(s) | ||||
| 		{ | ||||
| 			return document.getElementById(s); | ||||
| 			return d.getElementById(s); | ||||
| 		} | ||||
| 		function isObject(item) { | ||||
| 			return (item && typeof item === 'object' && !Array.isArray(item)); | ||||
| 		} | ||||
| 		function set(path, obj, val) { | ||||
| 			var tar = obj; | ||||
| 			var pList = path.split('_'); | ||||
| 			var len = pList.length; | ||||
| 			for(var i = 0; i < len-1; i++) { | ||||
| 				var elem = pList[i]; | ||||
| 				if( !tar[elem] ) tar[elem] = {} | ||||
| 				tar = tar[elem]; | ||||
| 			} | ||||
|  | ||||
| 			tar[pList[len-1]] = val; | ||||
| 		} | ||||
|  | ||||
| 		function addRec(s, path = "", label = null) | ||||
| 		{ | ||||
| 			var str = ""; | ||||
| 			for (i in s) | ||||
| 			{ | ||||
| 				var fk = path + (path?'_':'') + i; | ||||
| 				if (isObject(s[i])) { | ||||
| 					if (label && label[i] && label[i]["LABEL"]) str += `<h3>${label[i]["LABEL"]}</h3>`; | ||||
| 					str += addRec(s[i], fk, label? label[i] : null); | ||||
| 				} else { | ||||
| 					var lb = fk; | ||||
| 					if (label && label[i]) lb = label[i]; | ||||
| 					else if (s[i+'LABEL']) lb = s[i+'LABEL']; | ||||
| 					if (i.indexOf('LABEL') > 0) continue; | ||||
| 					var t = typeof s[i]; | ||||
| 					if (gId(fk)) { //already exists | ||||
| 						if(t === 'boolean') | ||||
| 						{ | ||||
| 							gId(fk).checked = s[i]; | ||||
| 						} else { | ||||
| 							gId(fk).value = s[i]; | ||||
| 						} | ||||
| 						if (gId(fk).previousElementSibling.matches('.l')) { | ||||
| 							gId(fk).previousElementSibling.innerHTML = lb; | ||||
| 						} | ||||
| 					} else { | ||||
| 						if(t === 'boolean') | ||||
| 						{ | ||||
| 							str += `${lb}: <input class="agi cb" type="checkbox" id=${fk} ${s[i]?"checked":""}><br>`; | ||||
| 						} else if (t === 'number') | ||||
| 						{ | ||||
| 							str += `${lb}: <input class="agi" type="number" id=${fk} value=${s[i]}><br>`; | ||||
| 						} else if (t === 'string') | ||||
| 						{ | ||||
| 							str += `${lb}:<br><input class="agi" id=${fk} value=${s[i]}><br>`; | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 			return str; | ||||
| 		} | ||||
|  | ||||
| 		function genForm(s) { | ||||
| 			var str = ""; | ||||
| 			str = addRec(s,"",l); | ||||
| 			 | ||||
| 			gId('gen').innerHTML = str; | ||||
| 		} | ||||
| 		function GetLS() | ||||
| 		{ | ||||
| 			sett = localStorage.getItem('wledUiCfg'); | ||||
| 			if (!sett) gId('lserr').style.display = "inline"; | ||||
| 			try { | ||||
| 				sett = JSON.parse(sett); | ||||
| 			} catch (e) { | ||||
| 				sett = {}; | ||||
| 				gId('lserr').style.display = "inline"; | ||||
| 				gId('lserr').innerHTML = "⚠ Settings JSON parsing failed. (" + e + ")"; | ||||
| 			} | ||||
| 			genForm(sett); | ||||
| 			gId('dm').checked = (gId('theme_base').value === 'light'); | ||||
| 		} | ||||
| 	 | ||||
| 		function SetLS() | ||||
| 		{ | ||||
| 			var l = d.querySelectorAll('.agi'); | ||||
| 			for (var i = 0; i < l.length; i++) { | ||||
| 				var e = l[i]; | ||||
| 				var val = e.classList.contains('cb') ? e.checked : e.value; | ||||
| 				set(e.id, sett, val); | ||||
| 				console.log(`${e.id} set to ${val}`); | ||||
| 			} | ||||
| 			try { | ||||
| 				localStorage.setItem('wledUiCfg', JSON.stringify(sett)); | ||||
| 				gId('lssuc').style.display = "inline"; | ||||
| 			} catch (e) { | ||||
| 				gId('lssuc').style.display = "none"; | ||||
| 				gId('lserr').style.display = "inline"; | ||||
| 				gId('lserr').innerHTML = "⚠ Settings JSON saving failed. (" + e + ")"; | ||||
| 			} | ||||
| 		} | ||||
| 		 | ||||
| 		function Save() { | ||||
| 			SetLS(); | ||||
| 			if (d.Sf.DS.value != initial_ds || d.Sf.ST.checked != initial_st) d.Sf.submit(); | ||||
| 		} | ||||
| 		 | ||||
| 		function S() | ||||
| 		{ | ||||
| 			GetV();Ct(); | ||||
| 			GetV();  | ||||
| 			initial_ds = d.Sf.DS.value; | ||||
| 			initial_st = d.Sf.ST.checked; | ||||
| 			GetLS(); | ||||
| 		} | ||||
| 		function H() | ||||
| 		{ | ||||
| @@ -20,87 +158,68 @@ | ||||
| 		{ | ||||
| 			window.open("/settings","_self"); | ||||
| 		} | ||||
| 		function Ct() | ||||
| 		function UI() | ||||
| 		{ | ||||
| 			if (gId("co").selected) | ||||
| 			{ | ||||
| 				gId("cth").style.display="block"; | ||||
| 			} else | ||||
| 			{ | ||||
| 				gId("cth").style.display="none"; | ||||
| 			gId('idonthateyou').style.display = (gId('dm').checked) ? 'inline':'none'; | ||||
| 			var f = gId('theme_base'); | ||||
| 			if (f) f.value = (gId('dm').checked) ? 'light':'dark'; | ||||
| 		} | ||||
|  | ||||
| 		// random BG image | ||||
| 		function setRandomBg() { | ||||
| 			if (gId("theme_bg_random").checked) { | ||||
| 				gId("theme_bg_url").value = "https://picsum.photos/1920/1080"; | ||||
| 			} else { | ||||
| 				gId("theme_bg_url").value = ""; | ||||
| 			} | ||||
| 			 | ||||
| 		} | ||||
| 		function checkRandomBg() { | ||||
| 			if (gId("theme_bg_url").value === "https://picsum.photos/1920/1080") { | ||||
| 				gId("theme_bg_random").checked = true; | ||||
| 			} else { | ||||
| 				gId("theme_bg_random").checked = false; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		function GetV() | ||||
| 		{ | ||||
| 			var d = document; | ||||
| 		} | ||||
| 	</script> | ||||
| 	<style> | ||||
| 		:root { | ||||
| 			--aCol: #666; | ||||
| 			--bCol: #333; | ||||
| 			--cCol: #222; | ||||
| 			--dCol: #666; | ||||
|       --tCol: #fff; | ||||
| 			--sCol: #0000; | ||||
| 			--cFn: Verdana; | ||||
| 		} | ||||
| 		body { | ||||
| 			font-family: var(--cFn), Helvetica, sans-serif; | ||||
| 			text-align: center; | ||||
| 			background: var(--cCol); | ||||
| 			color: var(--tCol); | ||||
| 			line-height: 200%; | ||||
| 			margin: 0; | ||||
| 			background-attachment: fixed; | ||||
| 		} | ||||
| 		hr { | ||||
| 			border-color: var(--dCol); | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 		} | ||||
| 		button { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--tCol); | ||||
| 			border: 0.3ch solid var(--bCol); | ||||
| 			display: inline-block; | ||||
| 			font-family: var(--cFn), Helvetica, sans-serif; | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 			font-size: 20px; | ||||
| 			margin: 8px; | ||||
| 			margin-top: 12px; | ||||
| 		} | ||||
| 		.helpB { | ||||
| 			text-align: left; | ||||
| 			position: absolute; | ||||
| 			width:60px; | ||||
| 		} | ||||
| 		input { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--dCol); | ||||
| 			border: 0.5ch solid var(--bCol); | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 			font-family: var(--cFn), Helvetica, sans-serif; | ||||
| 		} | ||||
| 		select { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(-tCol); | ||||
| 			border: 0.5ch solid var(--bCol); | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 			font-family: var(--cFn), Helvetica, sans-serif; | ||||
| 		} | ||||
| 		input[type=number] { | ||||
| 			width: 3em; | ||||
| 		} | ||||
| 		@import url("style.css"); | ||||
| 	</style> | ||||
| </head> | ||||
| <body onload="S()"> | ||||
| 	<form id="form_s" name="Sf" method="post"> | ||||
| 		<div style="position:sticky;top:0;background-color:#222;"> | ||||
| 		<div class="helpB"><button type="button" onclick="H()">?</button></div> | ||||
| 		<button type="button" onclick="B()">Back</button><button type="submit">Save</button><hr> | ||||
| 		<button type="button" onclick="B()">Back</button><button type="button" onclick="Save()">Save</button><br> | ||||
| 		<span id="lssuc" style="color:green; display:none">✔ Local UI settings saved!</span> | ||||
| 		<span id="lserr" style="color:red; display:none">⚠ Could not access local storage. Make sure it is enabled in your browser.</span><hr> | ||||
| 		</div> | ||||
| 		<h2>Web Setup</h2> | ||||
| 		Server description: <input name="DS" maxlength="32"><br> | ||||
|     Sync button toggles both send and receive: <input type="checkbox" name="ST"><br><br> | ||||
| 		<hr><button type="button" onclick="B()">Back</button><button type="submit">Save</button> | ||||
|     Sync button toggles both send and receive: <input type="checkbox" name="ST"><br> | ||||
| 		<i>The following UI customization settings are unique both to the WLED device and this browser.<br> | ||||
| 		You will need to set them again if using a different browser, device or WLED IP address.<br> | ||||
| 		Refresh the main UI to apply changes.</i><br> | ||||
| 		 | ||||
| 		<div id="gen">Loading settings...</div> | ||||
| 		 | ||||
| 		<h3>UI Appearance</h3> | ||||
| 		<span class="l"></span>: <input type="checkbox" id="comp_labels" class="agi cb"><br> | ||||
|     <span class="l"></span>: <input type="checkbox" id="comp_pcmbot" class="agi cb"><br> | ||||
|     <span class="l"></span>: <input type="checkbox" id="comp_pid" class="agi cb"><br> | ||||
| 		I hate dark mode: <input type="checkbox" id="dm" onchange="UI()"><br> | ||||
| 		<span id="idonthateyou" style="display:none"><i>Why would you? </i>🥺<br></span> | ||||
| 		<span class="l"></span>: <input type="number" min=0.0 max=1.0 step=0.01 id="theme_alpha_tab" class="agi"><br> | ||||
| 		<span class="l"></span>: <input type="number" min=0.0 max=1.0 step=0.01 id="theme_alpha_bg" class="agi"><br> | ||||
| 		<span class="l"></span>: <input id="theme_color_bg" maxlength="9" class="agi"><br> | ||||
| 		<span class="l">BG image URL</span>: <input id="theme_bg_url" class="agi" oninput="checkRandomBg()"><br> | ||||
| 		<span class="l">Random BG image</span>: <input type="checkbox" id="theme_bg_random" class="agi cb" onchange="setRandomBg()"><br> | ||||
| 		<input id="theme_base" class="agi" style="display:none"> | ||||
| 		<hr><button type="button" onclick="B()">Back</button><button type="button" onclick="Save()">Save</button> | ||||
| 	</form> | ||||
| </body> | ||||
| </html> | ||||
| @@ -1,6 +1,7 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <html lang="en"> | ||||
| <head> | ||||
|   <meta charset="utf-8"> | ||||
| 	<meta name="viewport" content="width=500"> | ||||
| 	<title>WiFi Settings</title> | ||||
| 	<script> | ||||
| @@ -18,57 +19,13 @@ | ||||
| 		} | ||||
| 	</script> | ||||
| 	<style> | ||||
| 		:root { | ||||
| 			--aCol: #D9B310; | ||||
| 			--bCol: #0B3C5D; | ||||
| 			--cCol: #1D2731; | ||||
| 			--dCol: #328CC1; | ||||
| 			--sCol: #000; | ||||
| 			--cFn: Verdana; | ||||
| 		} | ||||
| 		body { | ||||
| 			font-family: Verdana, Helvetica, sans-serif; | ||||
| 			text-align: center; | ||||
| 			background: var(--cCol); | ||||
| 			color: var(--dCol); | ||||
| 			line-height: 200%; | ||||
| 			margin: 0; | ||||
| 			background-attachment: fixed; | ||||
| 		} | ||||
| 		hr { | ||||
| 			border-color: var(--dCol); | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 		} | ||||
| 		button { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--dCol); | ||||
| 			border: 0.3ch solid var(--bCol); | ||||
| 			display: inline-block; | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 			font-size: 20px; | ||||
| 			margin: 8px; | ||||
| 			margin-top: 12px; | ||||
| 		} | ||||
| 		.helpB { | ||||
| 			text-align: left; | ||||
| 			position: absolute; | ||||
| 			width:60px; | ||||
| 		} | ||||
| 		input { | ||||
| 			background: var(--bCol); | ||||
| 			color: var(--dCol); | ||||
| 			border: 0.5ch solid var(--bCol); | ||||
| 			filter: drop-shadow( -5px -5px 5px var(--sCol) ); | ||||
| 		} | ||||
| 		input[type=number] { | ||||
| 			width: 3em; | ||||
| 		} | ||||
| 		@import url("style.css"); | ||||
| 	</style> | ||||
| </head> | ||||
| <body onload="GetV()"> | ||||
| 	<form id="form_s" name="Sf" method="post"> | ||||
| 		<div class="helpB"><button type="button" onclick="H()">?</button></div> | ||||
| 		<button type="button" onclick="B()">Back</button><button type="submit">Save & Reboot</button><hr> | ||||
| 		<button type="button" onclick="B()">Back</button><button type="submit">Save & Connect</button><hr> | ||||
| 		<h2>WiFi setup</h2> | ||||
| 		<h3>Connect to existing network</h3> | ||||
| 		Network name (SSID, empty to not connect): <br><input name="CS" maxlength="32"><br> | ||||
| @@ -94,7 +51,7 @@ | ||||
| 		<h3>Configure Access Point</h3> | ||||
| 		AP SSID (leave empty for no AP):<br> <input name="AS" maxlength="32"><br> | ||||
| 		Hide AP name: <input type="checkbox" name="AH"><br> | ||||
| 		AP password (leave empty for open):<br> <input type="password" name="AP" maxlength="63"><br> | ||||
| 		AP password (leave empty for open):<br> <input type="password" name="AP" maxlength="63" pattern="(.{8,63})|()" title="Empty or min. 8 characters"><br> | ||||
| 		Access Point WiFi channel: <input name="AC" type="number" min="1" max="13" required><br> | ||||
|     AP opens: | ||||
|     <select name="AB"> | ||||
| @@ -102,8 +59,22 @@ | ||||
|     <option value="1">Disconnected</option> | ||||
|     <option value="2">Always</option> | ||||
|     <option value="3">Never (not recommended)</option></select><br> | ||||
| 		AP IP: <span class="sip"> Not active </span><hr> | ||||
| 		<button type="button" onclick="B()">Back</button><button type="submit">Save & Reboot</button> | ||||
| 		AP IP: <span class="sip"> Not active </span><br> | ||||
| 		<h3>Experimental</h3> | ||||
| 		Disable WiFi sleep: <input type="checkbox" name="WS"><br> | ||||
| 		<i>Can help with connectivity issues.<br> | ||||
|     Do not enable if WiFi is working correctly, increases power consumption.</i> | ||||
|     <div id="ethd"> | ||||
| 		<h3>Ethernet Type</h3> | ||||
| 		<select name="ETH"> | ||||
| 		<option value="0">None</option> | ||||
|     <option value="2">ESP32-POE</option> | ||||
|     <option value="4">QuinLED-ESP32</option> | ||||
|     <option value="3">WESP32</option> | ||||
| 		<option value="1">WT32-ETH01</option> | ||||
|     </select><br><br></div> | ||||
| 		<hr> | ||||
| 		<button type="button" onclick="B()">Back</button><button type="submit">Save & Connect</button> | ||||
| 	</form> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										48
									
								
								wled00/data/style.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								wled00/data/style.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| body { | ||||
|   font-family: Verdana, sans-serif; | ||||
|   text-align: center; | ||||
|   background: #222; | ||||
|   color: #fff; | ||||
|   line-height: 200%; | ||||
|   margin: 0; | ||||
| } | ||||
| hr { | ||||
|   border-color: #666; | ||||
| } | ||||
| button { | ||||
|   background: #333; | ||||
|   color: #fff; | ||||
|   font-family: Verdana, sans-serif; | ||||
|   border: 0.3ch solid #333; | ||||
|   display: inline-block; | ||||
|   font-size: 20px; | ||||
|   margin: 8px; | ||||
|   margin-top: 12px; | ||||
|   cursor: pointer; | ||||
| } | ||||
| .helpB { | ||||
|   text-align: left; | ||||
|   position: absolute; | ||||
|   width: 60px; | ||||
| } | ||||
| input { | ||||
|   background: #333; | ||||
|   color: #fff; | ||||
|   font-family: Verdana, sans-serif; | ||||
|   border: 0.5ch solid #333; | ||||
| } | ||||
| input[type="number"] { | ||||
|   width: 4em; | ||||
| } | ||||
| select { | ||||
|   background: #333; | ||||
|   color: #fff; | ||||
|   font-family: Verdana, sans-serif; | ||||
|   border: 0.5ch solid #333; | ||||
| } | ||||
| td { | ||||
|   padding: 2px; | ||||
| } | ||||
| .d5 { | ||||
|   width: 4.5em !important; | ||||
| } | ||||
| @@ -1,4 +1,52 @@ | ||||
| <!DOCTYPE html> | ||||
| <html><head><meta content='width=device-width' name='viewport'><title>WLED Message</title><script>function B(){window.history.back()}</script> | ||||
| <style>:root{--aCol:#D9B310;--bCol:#0B3C5D;--cCol:#1D2731;--dCol:#328CC1;--sCol:#000;--tCol:#328CC1;--cFn:Verdana;}.bt{background:var(--bCol);color:var(--tCol);font-family:var(--cFn),sans-serif;border:.3ch solid var(--bCol);display:inline-block;filter:drop-shadow(-5px -5px 5px var(--sCol));font-size:20px;margin:8px;margin-top:12px}input[type=file]{font-size:16px}body{font-family:var(--cFn),sans-serif;text-align:center;background:var(--cCol);color:var(--tCol);line-height:200%}</style></head> | ||||
| <body><h2>WLED Software Update</h2>Installed version: 0.8.5-dev<br>Download the latest binary: <a href="https://github.com/Aircoookie/WLED/releases"><img src="https://img.shields.io/github/release/Aircoookie/WLED.svg?style=flat-square"></a><br><form method='POST' action='/update' enctype='multipart/form-data'><input type='file' class="bt" name='update' required><br><input type='submit' class="bt" value='Update!'></form><button type="button" class="bt" onclick="B()">Back</button></body></html> | ||||
| <html> | ||||
|  | ||||
| <head> | ||||
|     <meta content='width=device-width' name='viewport'> | ||||
|     <title>WLED Update</title> | ||||
|     <script>function B() { window.history.back() } | ||||
|     function U() {document.getElementById("uf").style.display="none";document.getElementById("msg").style.display="block";} | ||||
|     </script> | ||||
|     <style> | ||||
|       .bt { | ||||
|         background: #333; | ||||
|         color: #fff; | ||||
|         font-family: Verdana, sans-serif; | ||||
|         border: .3ch solid #333; | ||||
|         display: inline-block; | ||||
|         font-size: 20px; | ||||
|         margin: 8px; | ||||
|         margin-top: 12px | ||||
|       } | ||||
|  | ||||
|       input[type=file] { | ||||
|         font-size: 16px | ||||
|       } | ||||
|  | ||||
|       body { | ||||
|         font-family: Verdana, sans-serif; | ||||
|         text-align: center; | ||||
|         background: #222; | ||||
|         color: #fff; | ||||
|         line-height: 200% | ||||
|       } | ||||
|        | ||||
|       #msg { | ||||
|         display: none; | ||||
|       } | ||||
|     </style> | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
|     <h2>WLED Software Update</h2> | ||||
|     <form method='POST' action='/update' id='uf' enctype='multipart/form-data' onsubmit="U()"> | ||||
|       Installed version: ##VERSION##<br> | ||||
|       Download the latest binary: <a href="https://github.com/Aircoookie/WLED/releases" target="_blank"> | ||||
|       <img src="https://img.shields.io/github/release/Aircoookie/WLED.svg?style=flat-square"></a><br> | ||||
|       <input type='file' class="bt" name='update' accept=".bin" required><br> | ||||
|       <input type='submit' class="bt" value='Update!' ><br> | ||||
|       <button type="button" class="bt" onclick="B()">Back</button></form> | ||||
|     <div id="msg"><b>Updating...</b><br>Please do not close or refresh the page :)</div> | ||||
| </body> | ||||
|  | ||||
| </html> | ||||
							
								
								
									
										6
									
								
								wled00/data/usermod.htm
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								wled00/data/usermod.htm
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
|  | ||||
| <body>No usermod custom web page set.</body> | ||||
|  | ||||
| </html> | ||||
| @@ -3,13 +3,13 @@ | ||||
| <head> | ||||
|   <meta charset="utf-8"> | ||||
| 	<meta content='width=device-width' name='viewport'> | ||||
|   <meta name="theme-color" content="#333333"> | ||||
| 	<title>WLED Setup</title> | ||||
|   <meta name="theme-color" content="#222222"> | ||||
| 	<title>Welcome!</title> | ||||
| 	<style> | ||||
| 		body { | ||||
| 			font-family: Verdana, Helvetica, sans-serif; | ||||
| 			text-align: center; | ||||
| 			background-color: #333; | ||||
| 			background-color: #222; | ||||
| 			margin: 0; | ||||
| 			color: #fff; | ||||
| 		} | ||||
| @@ -23,32 +23,42 @@ | ||||
|       text-transform: uppercase; | ||||
|       font-family: helvetica; | ||||
|       font-size: 19px; | ||||
|       background-color: #222; | ||||
|       background-color: #333; | ||||
|       color: white; | ||||
|       border: 0px solid white; | ||||
|       border-radius: 5px; | ||||
|       border-radius: 25px; | ||||
|       filter: drop-shadow(0px 0px 1px #000); | ||||
|     } | ||||
|  | ||||
| 		svg { | ||||
| 			fill: #fff; | ||||
| 		} | ||||
|     img { | ||||
|       width: 999px; | ||||
|       max-width: 85%; | ||||
|       image-rendering: pixelated; | ||||
|       image-rendering: crisp-edges; | ||||
|       margin: 25px 0 -10px 0; | ||||
|       animation: fi 1s; | ||||
|     } | ||||
|      | ||||
|     @keyframes fi { | ||||
|     from { opacity: 0; } | ||||
|     to   { opacity: 1; } | ||||
|     } | ||||
|      | ||||
|     .main { | ||||
|       animation: fi 1.5s .7s both; | ||||
|     } | ||||
| 	</style> | ||||
| </head> | ||||
| <body> | ||||
| <svg style="position: absolute; width: 0; height: 0; overflow: hidden;" version="1.1" xmlns="http://www.w3.org/2000/svg"> | ||||
| <defs> | ||||
| 	<symbol id="lnr-smile" viewBox="0 0 1024 1024"><path d="M486.4 1024c-129.922 0-252.067-50.594-343.936-142.464s-142.464-214.014-142.464-343.936c0-129.923 50.595-252.067 142.464-343.936s214.013-142.464 343.936-142.464c129.922 0 252.067 50.595 343.936 142.464s142.464 214.014 142.464 343.936-50.594 252.067-142.464 343.936c-91.869 91.87-214.014 142.464-343.936 142.464zM486.4 102.4c-239.97 0-435.2 195.23-435.2 435.2s195.23 435.2 435.2 435.2 435.2-195.23 435.2-435.2-195.23-435.2-435.2-435.2z"></path><path d="M332.8 409.6c-42.347 0-76.8-34.453-76.8-76.8s34.453-76.8 76.8-76.8 76.8 34.453 76.8 76.8-34.453 76.8-76.8 76.8zM332.8 307.2c-14.115 0-25.6 11.485-25.6 25.6s11.485 25.6 25.6 25.6 25.6-11.485 25.6-25.6-11.485-25.6-25.6-25.6z"></path><path d="M640 409.6c-42.349 0-76.8-34.453-76.8-76.8s34.451-76.8 76.8-76.8 76.8 34.453 76.8 76.8-34.451 76.8-76.8 76.8zM640 307.2c-14.115 0-25.6 11.485-25.6 25.6s11.485 25.6 25.6 25.6 25.6-11.485 25.6-25.6-11.485-25.6-25.6-25.6z"></path><path d="M486.4 870.4c-183.506 0-332.8-149.294-332.8-332.8 0-14.139 11.462-25.6 25.6-25.6s25.6 11.461 25.6 25.6c0 155.275 126.325 281.6 281.6 281.6s281.6-126.325 281.6-281.6c0-14.139 11.461-25.6 25.6-25.6s25.6 11.461 25.6 25.6c0 183.506-149.294 332.8-332.8 332.8z"></path></symbol> | ||||
| </defs></svg> | ||||
| <br><br> | ||||
| <svg><use xlink:href="#lnr-smile"></use></svg> | ||||
| <img alt="" src=" data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAHEAAAAjCAYAAACjOOUsAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsAAAA7AAWrWiQkAAATfSURBVGhD7ZlPaBVHHMdnS2Ox5iKYGIyYJwQlUTHG6kUwtSAIBsEiBopgKR5i68mbIPgEwZun0uYgouIlobQIKQheIsSLaBSaRFoEc3gtMa9HLaiH7X4n81tn583u/HkTaZ/7gR+zMzvv93bfZ3/777GSkpL/AJFo3Tg/H7OuPtGxY7CTsZnjkd/3lRTi/qMKgfGXt8TAO6KfT2TG1f7u6ROlyBXgI9HaQRW4+JR3ox92Z1p1XO3PLCUVORHHvFMSDDeJBuJvH4mlZdS+C3c7B+Ln3fvivw59w0MMOzP/FYvj75K4PMhDDDszdbM7rk1V4hcPD/MQw858MbEnPvJgf3x29hQPMdwUbqc2qRIhCBVGrQk+Lzm9mq6NkIe2t20N7xOf7NjK2w13rlltM+Sh7VvLu+/YOMib6NyMVR7IQ9u7qY33ibb2bbxd/9mvVnkgD217ZTXvE72fbuHtle1X3VxIuH1QOZ2mmG5yaH4yr0giBKryVCDTJBICG+SpJDJNIiFQlacCmSaREKjKU4FMX5F+p1NIk8OExTwbgeD1b7+zotOrlUBQm2FFp1cbgeDtyzlWdHq1EQie/fMH8z29Br0myiwdr7PkWpSGC2se/8IqtWnRa+y7Ur+Q3FF97/X7ZFjV/4B1Dz1Pw5WrlZ/Y7T330gjFikkE9Xo9jSJ0Vah+Ru7nVWNRFWq3Iacai6qQ9kfOl1eNuirUfZ7wrcbgEqkCOyc6MmEDVdyrXUd5ENR3rUiqwI6LnTx8oQp8M783E66cWjiWCarMZlnRSix5PwSTSBXY0WFXdf9HfK+FKlSBoX6rjEQ8XMutD3nn+1Yg5L755Mrzk0qkh2xMwLOYr0iXa6CJZu9KQ+N7LSSoAuma6EKRn1TiwaUn0bO3r/gy7v42/3nf+cEzxFGq5jD1TYTYJhAqj4pt3iI/mdMpTbQRqD4HhqhA9S40r28L3ZWmz4kUjlAFNvucSNBzomtF5vlJJVarVb53mMgHLMBRRBEaNWez3xFqW5vN0+znyQ/5Ahlho6OjcVdXl+jxiVmhHn8Gq+jendq+cpPRvUO1fuUmo3mHavvKTUb3DtX2lZtM3jtUWRqQ3TRMBrLMjMgVkAiB2waX/1nwgURygTv885BICOzrH+BjPpBICBzo38nHfJBFksDFxUU2NjaWjhMNAwSJ1Ems/n2RDQ0NsQNzn4sVjF1vvyGWGOvp6cmsk+frJLpWISFXo1cVElI1+lQhIVejTxUSajVCYp5AoB0k6AhIRQqJo7OnGRe87gIfxl9N1Y8nlpcTKpUK+/rlSdFLDghpft5fUTho0GJD5WW0RetUWi1PgwMNmbtTFdhXaRBowdj2H3kuVGQeyIkg5GVQtE6mFfPoPMgUSqSjgo4Gdqk/whfxpGeSaqJIxjFGsbCwkF2fBHJh3ZH5RpE2RxtBedJtkmi1PLSMdXwgh0KJAIkBEiLQV5MODw9ndiBvp/I2VIwZd5T4EPJQLvRNGCVSYgr0xSotIyMjYkkPfV7eUFNOHa2cx+X3BkaJAIkoxJAWCBwfHxc9PdhBbFzSRgixzHfahVbNA2x/b8JKoi0QaKpE7KC8cVjGmOha06p5fAgq0aYSS8ITROLk5GRUq9UyAt/XUVgSGNyl0kNrMyBHchCUeSwJejotKSkp+VBh7F9fIy7OdlOyMwAAAABJRU5ErkJggg=="> | ||||
| <div class="main"> | ||||
| <h1>Welcome to WLED!</h1> | ||||
| <h3>Thank you for installing my application!</h3> | ||||
| If you encounter a bug or have a question/feature suggestion, feel free to open a GitHub issue!<br><br> | ||||
| <b>Next steps:</b><br><br> | ||||
| Connect the module to your local WiFi here!<br> | ||||
| <button onclick="window.location.href='/settings/wifi'">WiFi settings</button><br> | ||||
| <i>Just trying this out in AP mode?</i><br> | ||||
| <button onclick="window.location.href='/sliders'">To the controls!</button> | ||||
|  | ||||
| <button onclick="window.location.href='/sliders'">To the controls!</button><br> | ||||
| </div> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										68
									
								
								wled00/dmx.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								wled00/dmx.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,68 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * Support for DMX via MAX485. | ||||
|  * Change the output pin in src/dependencies/ESPDMX.cpp if needed. | ||||
|  * Library from: | ||||
|  * https://github.com/Rickgg/ESP-Dmx | ||||
|  */ | ||||
|  | ||||
| #ifdef WLED_ENABLE_DMX | ||||
|  | ||||
| void handleDMX() | ||||
| { | ||||
|   // don't act, when in DMX Proxy mode | ||||
|   if (e131ProxyUniverse != 0) return; | ||||
|  | ||||
|   // TODO: calculate brightness manually if no shutter channel is set | ||||
|  | ||||
|   uint8_t brightness = strip.getBrightness(); | ||||
|  | ||||
|   for (int i = DMXStartLED; i < ledCount; i++) {        // uses the amount of LEDs as fixture count | ||||
|  | ||||
|     uint32_t in = strip.getPixelColor(i);     // get the colors for the individual fixtures as suggested by Aircoookie in issue #462 | ||||
|     byte w = in >> 24 & 0xFF; | ||||
|     byte r = in >> 16 & 0xFF; | ||||
|     byte g = in >> 8 & 0xFF; | ||||
|     byte b = in & 0xFF; | ||||
|  | ||||
|     int DMXFixtureStart = DMXStart + (DMXGap * (i - DMXStartLED)); | ||||
|     for (int j = 0; j < DMXChannels; j++) { | ||||
|       int DMXAddr = DMXFixtureStart + j; | ||||
|       switch (DMXFixtureMap[j]) { | ||||
|         case 0:        // Set this channel to 0. Good way to tell strobe- and fade-functions to fuck right off. | ||||
|           dmx.write(DMXAddr, 0); | ||||
|           break; | ||||
|         case 1:        // Red | ||||
|           dmx.write(DMXAddr, r); | ||||
|           break; | ||||
|         case 2:        // Green | ||||
|           dmx.write(DMXAddr, g); | ||||
|           break; | ||||
|         case 3:        // Blue | ||||
|           dmx.write(DMXAddr, b); | ||||
|           break; | ||||
|         case 4:        // White | ||||
|           dmx.write(DMXAddr, w); | ||||
|           break; | ||||
|         case 5:        // Shutter channel. Controls the brightness. | ||||
|           dmx.write(DMXAddr, brightness); | ||||
|           break; | ||||
|         case 6:        // Sets this channel to 255. Like 0, but more wholesome. | ||||
|           dmx.write(DMXAddr, 255); | ||||
|           break; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   dmx.update();        // update the DMX bus | ||||
| } | ||||
|  | ||||
| void initDMX() { | ||||
|   dmx.init(512);        // initialize with bus length | ||||
| } | ||||
|  | ||||
| #else | ||||
| void handleDMX() {} | ||||
| void initDMX() {} | ||||
| #endif | ||||
							
								
								
									
										209
									
								
								wled00/e131.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										209
									
								
								wled00/e131.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,209 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| #define MAX_3_CH_LEDS_PER_UNIVERSE 170 | ||||
| #define MAX_4_CH_LEDS_PER_UNIVERSE 128 | ||||
| #define MAX_CHANNELS_PER_UNIVERSE 512 | ||||
|  | ||||
| /* | ||||
|  * E1.31 handler | ||||
|  */ | ||||
|  | ||||
| //DDP protocol support, called by handleE131Packet | ||||
| //handles RGB data only | ||||
| void handleDDPPacket(e131_packet_t* p) { | ||||
|   int lastPushSeq = e131LastSequenceNumber[0]; | ||||
|    | ||||
|   //reject late packets belonging to previous frame (assuming 4 packets max. before push) | ||||
|   if (e131SkipOutOfSequence && lastPushSeq) { | ||||
|     int sn = p->sequenceNum & 0xF; | ||||
|     if (sn) { | ||||
|       if (lastPushSeq > 5) { | ||||
|         if (sn > (lastPushSeq -5) && sn < lastPushSeq) return; | ||||
|       } else { | ||||
|         if (sn > (10 + lastPushSeq) || sn < lastPushSeq) return; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   uint32_t start = htonl(p->channelOffset) /3; | ||||
|   start += DMXAddress /3; | ||||
|   uint16_t stop = start + htons(p->dataLen) /3; | ||||
|   uint8_t* data = p->data; | ||||
|   uint16_t c = 0; | ||||
|   if (p->flags & DDP_TIMECODE_FLAG) c = 4; //packet has timecode flag, we do not support it, but data starts 4 bytes later | ||||
|  | ||||
|   realtimeLock(realtimeTimeoutMs, REALTIME_MODE_DDP); | ||||
|    | ||||
|   for (uint16_t i = start; i < stop; i++) { | ||||
|     setRealtimePixel(i, data[c], data[c+1], data[c+2], 0); | ||||
|     c+=3; | ||||
|   } | ||||
|  | ||||
|   bool push = p->flags & DDP_PUSH_FLAG; | ||||
|   if (push) { | ||||
|     e131NewData = true; | ||||
|     byte sn = p->sequenceNum & 0xF; | ||||
|     if (sn) e131LastSequenceNumber[0] = sn; | ||||
|   } | ||||
| } | ||||
|  | ||||
| //E1.31 and Art-Net protocol support | ||||
| void handleE131Packet(e131_packet_t* p, IPAddress clientIP, byte protocol){ | ||||
|  | ||||
|   uint16_t uni = 0, dmxChannels = 0; | ||||
|   uint8_t* e131_data = nullptr; | ||||
|   uint8_t seq = 0, mde = REALTIME_MODE_E131; | ||||
|  | ||||
|   if (protocol == P_ARTNET) | ||||
|   { | ||||
|     uni = p->art_universe; | ||||
|     dmxChannels = htons(p->art_length); | ||||
|     e131_data = p->art_data; | ||||
|     seq = p->art_sequence_number; | ||||
|     mde = REALTIME_MODE_ARTNET; | ||||
|   } else if (protocol == P_E131) { | ||||
|     uni = htons(p->universe); | ||||
|     dmxChannels = htons(p->property_value_count) -1; | ||||
|     e131_data = p->property_values; | ||||
|     seq = p->sequence_number; | ||||
|   } else { //DDP | ||||
|     realtimeIP = clientIP; | ||||
|     handleDDPPacket(p); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   #ifdef WLED_ENABLE_DMX | ||||
|   // does not act on out-of-order packets yet | ||||
|   if (e131ProxyUniverse > 0 && uni == e131ProxyUniverse) { | ||||
|     for (uint16_t i = 1; i <= dmxChannels; i++) | ||||
|       dmx.write(i, e131_data[i]); | ||||
|     dmx.update(); | ||||
|   } | ||||
|   #endif | ||||
|  | ||||
|   // only listen for universes we're handling & allocated memory | ||||
|   if (uni >= (e131Universe + E131_MAX_UNIVERSE_COUNT)) return; | ||||
|  | ||||
|   uint8_t previousUniverses = uni - e131Universe; | ||||
|  | ||||
|   if (e131SkipOutOfSequence) | ||||
|     if (seq < e131LastSequenceNumber[uni-e131Universe] && seq > 20 && e131LastSequenceNumber[uni-e131Universe] < 250){ | ||||
|       DEBUG_PRINT("skipping E1.31 frame (last seq="); | ||||
|       DEBUG_PRINT(e131LastSequenceNumber[uni-e131Universe]); | ||||
|       DEBUG_PRINT(", current seq="); | ||||
|       DEBUG_PRINT(seq); | ||||
|       DEBUG_PRINT(", universe="); | ||||
|       DEBUG_PRINT(uni); | ||||
|       DEBUG_PRINTLN(")"); | ||||
|       return; | ||||
|     } | ||||
|   e131LastSequenceNumber[uni-e131Universe] = seq; | ||||
|  | ||||
|   // update status info | ||||
|   realtimeIP = clientIP; | ||||
|   byte wChannel = 0; | ||||
|  | ||||
|   switch (DMXMode) { | ||||
|     case DMX_MODE_DISABLED: | ||||
|       return;  // nothing to do | ||||
|       break; | ||||
|  | ||||
|     case DMX_MODE_SINGLE_RGB: | ||||
|       if (uni != e131Universe) return; | ||||
|       if (dmxChannels-DMXAddress+1 < 3) return; | ||||
|       realtimeLock(realtimeTimeoutMs, mde); | ||||
|       if (realtimeOverride) return; | ||||
|       wChannel = (dmxChannels-DMXAddress+1 > 3) ? e131_data[DMXAddress+3] : 0; | ||||
|       for (uint16_t i = 0; i < ledCount; i++) | ||||
|         setRealtimePixel(i, e131_data[DMXAddress+0], e131_data[DMXAddress+1], e131_data[DMXAddress+2], wChannel); | ||||
|       break; | ||||
|  | ||||
|     case DMX_MODE_SINGLE_DRGB: | ||||
|       if (uni != e131Universe) return; | ||||
|       if (dmxChannels-DMXAddress+1 < 4) return; | ||||
|       realtimeLock(realtimeTimeoutMs, mde); | ||||
|       if (realtimeOverride) return; | ||||
|       wChannel = (dmxChannels-DMXAddress+1 > 4) ? e131_data[DMXAddress+4] : 0; | ||||
|       if (DMXOldDimmer != e131_data[DMXAddress+0]) { | ||||
|         DMXOldDimmer = e131_data[DMXAddress+0]; | ||||
|         bri = e131_data[DMXAddress+0]; | ||||
|         strip.setBrightness(bri); | ||||
|       } | ||||
|       for (uint16_t i = 0; i < ledCount; i++) | ||||
|         setRealtimePixel(i, e131_data[DMXAddress+1], e131_data[DMXAddress+2], e131_data[DMXAddress+3], wChannel); | ||||
|       break; | ||||
|  | ||||
|     case DMX_MODE_EFFECT: | ||||
|       if (uni != e131Universe) return; | ||||
|       if (dmxChannels-DMXAddress+1 < 11) return; | ||||
|       if (DMXOldDimmer != e131_data[DMXAddress+0]) { | ||||
|         DMXOldDimmer = e131_data[DMXAddress+0]; | ||||
|         bri = e131_data[DMXAddress+0]; | ||||
|       } | ||||
|       if (e131_data[DMXAddress+1] < MODE_COUNT) | ||||
|         effectCurrent = e131_data[DMXAddress+ 1]; | ||||
|       effectSpeed     = e131_data[DMXAddress+ 2];  // flickers | ||||
|       effectIntensity = e131_data[DMXAddress+ 3]; | ||||
|       effectPalette   = e131_data[DMXAddress+ 4]; | ||||
|       col[0]          = e131_data[DMXAddress+ 5]; | ||||
|       col[1]          = e131_data[DMXAddress+ 6]; | ||||
|       col[2]          = e131_data[DMXAddress+ 7]; | ||||
|       colSec[0]       = e131_data[DMXAddress+ 8]; | ||||
|       colSec[1]       = e131_data[DMXAddress+ 9]; | ||||
|       colSec[2]       = e131_data[DMXAddress+10]; | ||||
|       if (dmxChannels-DMXAddress+1 > 11) | ||||
|       { | ||||
|         col[3]        = e131_data[DMXAddress+11]; //white | ||||
|         colSec[3]     = e131_data[DMXAddress+12]; | ||||
|       } | ||||
|       transitionDelayTemp = 0;                        // act fast | ||||
|       colorUpdated(NOTIFIER_CALL_MODE_NOTIFICATION);  // don't send UDP | ||||
|       return;                                         // don't activate realtime live mode | ||||
|       break; | ||||
|  | ||||
|     case DMX_MODE_MULTIPLE_DRGB: | ||||
|     case DMX_MODE_MULTIPLE_RGB: | ||||
|     case DMX_MODE_MULTIPLE_RGBW: | ||||
|       { | ||||
|         realtimeLock(realtimeTimeoutMs, mde); | ||||
|         bool is4Chan = (DMXMode == DMX_MODE_MULTIPLE_RGBW); | ||||
|         const uint16_t dmxChannelsPerLed = is4Chan ? 4 : 3; | ||||
|         const uint16_t ledsPerUniverse = is4Chan ? MAX_4_CH_LEDS_PER_UNIVERSE : MAX_3_CH_LEDS_PER_UNIVERSE; | ||||
|         if (realtimeOverride) return; | ||||
|         uint16_t previousLeds, dmxOffset; | ||||
|         if (previousUniverses == 0) { | ||||
|           if (dmxChannels-DMXAddress < 1) return; | ||||
|           dmxOffset = DMXAddress; | ||||
|           previousLeds = 0; | ||||
|           // First DMX address is dimmer in DMX_MODE_MULTIPLE_DRGB mode. | ||||
|           if (DMXMode == DMX_MODE_MULTIPLE_DRGB) { | ||||
|             strip.setBrightness(e131_data[dmxOffset++]); | ||||
|           } | ||||
|         } else { | ||||
|           // All subsequent universes start at the first channel. | ||||
|           dmxOffset = (protocol == P_ARTNET) ? 0 : 1; | ||||
|           uint16_t ledsInFirstUniverse = (MAX_CHANNELS_PER_UNIVERSE - DMXAddress) / dmxChannelsPerLed; | ||||
|           previousLeds = ledsInFirstUniverse + (previousUniverses - 1) * ledsPerUniverse; | ||||
|         } | ||||
|         uint16_t ledsTotal = previousLeds + (dmxChannels - dmxOffset +1) / dmxChannelsPerLed; | ||||
|         if (!is4Chan) { | ||||
|           for (uint16_t i = previousLeds; i < ledsTotal; i++) { | ||||
|             setRealtimePixel(i, e131_data[dmxOffset], e131_data[dmxOffset+1], e131_data[dmxOffset+2], 0); | ||||
|             dmxOffset+=3; | ||||
|           } | ||||
|         } else { | ||||
|           for (uint16_t i = previousLeds; i < ledsTotal; i++) { | ||||
|             setRealtimePixel(i, e131_data[dmxOffset], e131_data[dmxOffset+1], e131_data[dmxOffset+2], e131_data[dmxOffset+3]); | ||||
|             dmxOffset+=4; | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|     default: | ||||
|       DEBUG_PRINTLN(F("unknown E1.31 DMX mode")); | ||||
|       return;  // nothing to do | ||||
|       break; | ||||
|   } | ||||
|  | ||||
|   e131NewData = true; | ||||
| } | ||||
							
								
								
									
										255
									
								
								wled00/fcn_declare.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										255
									
								
								wled00/fcn_declare.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,255 @@ | ||||
| #ifndef WLED_FCN_DECLARE_H | ||||
| #define WLED_FCN_DECLARE_H | ||||
| #include <Arduino.h> | ||||
| #include "src/dependencies/espalexa/EspalexaDevice.h" | ||||
| #include "src/dependencies/e131/ESPAsyncE131.h" | ||||
|  | ||||
| /* | ||||
|  * All globally accessible functions are declared here | ||||
|  */ | ||||
|  | ||||
| //alexa.cpp | ||||
| void onAlexaChange(EspalexaDevice* dev); | ||||
| void alexaInit(); | ||||
| void handleAlexa(); | ||||
| void onAlexaChange(EspalexaDevice* dev); | ||||
|  | ||||
| //blynk.cpp | ||||
| void initBlynk(const char* auth, const char* host, uint16_t port); | ||||
| void handleBlynk(); | ||||
| void updateBlynk(); | ||||
|  | ||||
| //button.cpp | ||||
| void shortPressAction(); | ||||
| bool isButtonPressed(); | ||||
| void handleButton(); | ||||
| void handleIO(); | ||||
|  | ||||
| //cfg.cpp | ||||
| void deserializeConfig(); | ||||
| bool deserializeConfigSec(); | ||||
| void serializeConfig(); | ||||
| void serializeConfigSec(); | ||||
|  | ||||
| //colors.cpp | ||||
| void colorFromUint32(uint32_t in, bool secondary = false); | ||||
| void colorFromUint24(uint32_t in, bool secondary = false); | ||||
| uint32_t colorFromRgbw(byte* rgbw); | ||||
| void relativeChangeWhite(int8_t amount, byte lowerBoundary = 0); | ||||
| void colorHStoRGB(uint16_t hue, byte sat, byte* rgb); //hue, sat to rgb | ||||
| void colorKtoRGB(uint16_t kelvin, byte* rgb); | ||||
| void colorCTtoRGB(uint16_t mired, byte* rgb); //white spectrum to rgb | ||||
|  | ||||
| void colorXYtoRGB(float x, float y, byte* rgb); // only defined if huesync disabled TODO | ||||
| void colorRGBtoXY(byte* rgb, float* xy); // only defined if huesync disabled TODO | ||||
|  | ||||
| void colorFromDecOrHexString(byte* rgb, char* in); | ||||
| bool colorFromHexString(byte* rgb, const char* in); | ||||
| void colorRGBtoRGBW(byte* rgb); //rgb to rgbw (http://codewelt.com/rgbw). (RGBW_MODE_LEGACY) | ||||
|  | ||||
| //dmx.cpp | ||||
| void initDMX(); | ||||
| void handleDMX(); | ||||
|  | ||||
| //e131.cpp | ||||
| void handleE131Packet(e131_packet_t* p, IPAddress clientIP, byte protocol); | ||||
|  | ||||
| //file.cpp | ||||
| bool handleFileRead(AsyncWebServerRequest*, String path); | ||||
| bool writeObjectToFileUsingId(const char* file, uint16_t id, JsonDocument* content); | ||||
| bool writeObjectToFile(const char* file, const char* key, JsonDocument* content); | ||||
| bool readObjectFromFileUsingId(const char* file, uint16_t id, JsonDocument* dest); | ||||
| bool readObjectFromFile(const char* file, const char* key, JsonDocument* dest); | ||||
| void updateFSInfo(); | ||||
| void closeFile(); | ||||
|  | ||||
| //hue.cpp | ||||
| void handleHue(); | ||||
| void reconnectHue(); | ||||
| void onHueError(void* arg, AsyncClient* client, int8_t error); | ||||
| void onHueConnect(void* arg, AsyncClient* client); | ||||
| void sendHuePoll(); | ||||
| void onHueData(void* arg, AsyncClient* client, void *data, size_t len); | ||||
|  | ||||
| //ir.cpp | ||||
| bool decodeIRCustom(uint32_t code); | ||||
| void applyRepeatActions(); | ||||
| void relativeChange(byte* property, int8_t amount, byte lowerBoundary = 0, byte higherBoundary = 0xFF); | ||||
| void changeEffectSpeed(int8_t amount); | ||||
| void changeEffectIntensity(int8_t amount); | ||||
| void decodeIR(uint32_t code); | ||||
| void decodeIR24(uint32_t code); | ||||
| void decodeIR24OLD(uint32_t code); | ||||
| void decodeIR24CT(uint32_t code); | ||||
| void decodeIR40(uint32_t code); | ||||
| void decodeIR44(uint32_t code); | ||||
| void decodeIR21(uint32_t code); | ||||
| void decodeIR6(uint32_t code); | ||||
| void decodeIR9(uint32_t code); | ||||
|  | ||||
| void initIR(); | ||||
| void handleIR(); | ||||
|  | ||||
| //json.cpp | ||||
| #include "ESPAsyncWebServer.h" | ||||
| #include "src/dependencies/json/ArduinoJson-v6.h" | ||||
| #include "src/dependencies/json/AsyncJson-v6.h" | ||||
| #include "FX.h" | ||||
|  | ||||
| void deserializeSegment(JsonObject elem, byte it); | ||||
| bool deserializeState(JsonObject root); | ||||
| void serializeSegment(JsonObject& root, WS2812FX::Segment& seg, byte id, bool forPreset = false, bool segmentBounds = true); | ||||
| void serializeState(JsonObject root, bool forPreset = false, bool includeBri = true, bool segmentBounds = true); | ||||
| void serializeInfo(JsonObject root); | ||||
| void serveJson(AsyncWebServerRequest* request); | ||||
| bool serveLiveLeds(AsyncWebServerRequest* request, uint32_t wsClient = 0); | ||||
|  | ||||
| //led.cpp | ||||
| void setValuesFromMainSeg(); | ||||
| void resetTimebase(); | ||||
| void toggleOnOff(); | ||||
| void setAllLeds(); | ||||
| void setLedsStandard(); | ||||
| bool colorChanged(); | ||||
| void colorUpdated(int callMode); | ||||
| void updateInterfaces(uint8_t callMode); | ||||
| void handleTransitions(); | ||||
| void handleNightlight(); | ||||
| byte scaledBri(byte in); | ||||
|  | ||||
| //lx_parser.cpp | ||||
| bool parseLx(int lxValue, byte* rgbw); | ||||
| void parseLxJson(int lxValue, byte segId, bool secondary); | ||||
|  | ||||
| //mqtt.cpp | ||||
| bool initMqtt(); | ||||
| void publishMqtt(); | ||||
|  | ||||
| //ntp.cpp | ||||
| void handleNetworkTime(); | ||||
| void sendNTPPacket(); | ||||
| bool checkNTPResponse();     | ||||
| void updateLocalTime(); | ||||
| void getTimeString(char* out); | ||||
| bool checkCountdown(); | ||||
| void setCountdown(); | ||||
| byte weekdayMondayFirst(); | ||||
| void checkTimers(); | ||||
|  | ||||
| //overlay.cpp | ||||
| void initCronixie(); | ||||
| void handleOverlays(); | ||||
| void handleOverlayDraw(); | ||||
| void _overlayAnalogCountdown(); | ||||
| void _overlayAnalogClock(); | ||||
|  | ||||
| byte getSameCodeLength(char code, int index, char const cronixieDisplay[]); | ||||
| void setCronixie(); | ||||
| void _overlayCronixie();     | ||||
| void _drawOverlayCronixie(); | ||||
|  | ||||
| //playlist.cpp | ||||
| void unloadPlaylist(); | ||||
| void loadPlaylist(JsonObject playlistObject); | ||||
| void handlePlaylist(); | ||||
|  | ||||
| //presets.cpp | ||||
| bool applyPreset(byte index); | ||||
| void savePreset(byte index, bool persist = true, const char* pname = nullptr, JsonObject saveobj = JsonObject()); | ||||
| void deletePreset(byte index); | ||||
|  | ||||
| //set.cpp | ||||
| void _setRandomColor(bool _sec,bool fromButton=false); | ||||
| bool isAsterisksOnly(const char* str, byte maxLen); | ||||
| void handleSettingsSet(AsyncWebServerRequest *request, byte subPage); | ||||
| bool handleSet(AsyncWebServerRequest *request, const String& req, bool apply=true); | ||||
| int getNumVal(const String* req, uint16_t pos); | ||||
| bool updateVal(const String* req, const char* key, byte* val, byte minv=0, byte maxv=255); | ||||
|  | ||||
| //udp.cpp | ||||
| void notify(byte callMode, bool followUp=false); | ||||
| void realtimeLock(uint32_t timeoutMs, byte md = REALTIME_MODE_GENERIC); | ||||
| void handleNotifications(); | ||||
| void setRealtimePixel(uint16_t i, byte r, byte g, byte b, byte w); | ||||
| void refreshNodeList(); | ||||
| void sendSysInfoUDP(); | ||||
|  | ||||
| //um_manager.cpp | ||||
| class Usermod { | ||||
|   public: | ||||
|     virtual void loop() {} | ||||
|     virtual void setup() {} | ||||
|     virtual void connected() {} | ||||
|     virtual void addToJsonState(JsonObject& obj) {} | ||||
|     virtual void addToJsonInfo(JsonObject& obj) {} | ||||
|     virtual void readFromJsonState(JsonObject& obj) {} | ||||
|     virtual void addToConfig(JsonObject& obj) {} | ||||
|     virtual void readFromConfig(JsonObject& obj) {} | ||||
|     virtual uint16_t getId() {return USERMOD_ID_UNSPECIFIED;} | ||||
| }; | ||||
|  | ||||
| class UsermodManager { | ||||
|   private: | ||||
|     Usermod* ums[WLED_MAX_USERMODS]; | ||||
|     byte numMods = 0; | ||||
|  | ||||
|   public: | ||||
|     void loop(); | ||||
|  | ||||
|     void setup(); | ||||
|     void connected(); | ||||
|  | ||||
|     void addToJsonState(JsonObject& obj); | ||||
|     void addToJsonInfo(JsonObject& obj); | ||||
|     void readFromJsonState(JsonObject& obj); | ||||
|  | ||||
|     void addToConfig(JsonObject& obj); | ||||
|     void readFromConfig(JsonObject& obj); | ||||
|  | ||||
|     bool add(Usermod* um); | ||||
|     Usermod* lookup(uint16_t mod_id); | ||||
|     byte getModCount(); | ||||
| }; | ||||
|  | ||||
| //usermods_list.cpp | ||||
| void registerUsermods(); | ||||
|  | ||||
| //usermod.cpp | ||||
| void userSetup(); | ||||
| void userConnected(); | ||||
| void userLoop(); | ||||
|  | ||||
| //wled_eeprom.cpp | ||||
| void applyMacro(byte index); | ||||
| void deEEP(); | ||||
| void deEEPSettings(); | ||||
| void clearEEPROM(); | ||||
|  | ||||
| //wled_serial.cpp | ||||
| void handleSerial(); | ||||
|  | ||||
| //wled_server.cpp | ||||
| bool isIp(String str); | ||||
| bool captivePortal(AsyncWebServerRequest *request); | ||||
| void initServer(); | ||||
| void serveIndexOrWelcome(AsyncWebServerRequest *request); | ||||
| void serveIndex(AsyncWebServerRequest* request); | ||||
| String msgProcessor(const String& var); | ||||
| void serveMessage(AsyncWebServerRequest* request, uint16_t code, const String& headl, const String& subl="", byte optionT=255); | ||||
| String settingsProcessor(const String& var); | ||||
| String dmxProcessor(const String& var); | ||||
| void serveSettings(AsyncWebServerRequest* request, bool post = false); | ||||
|  | ||||
| //ws.cpp | ||||
| void handleWs(); | ||||
| void wsEvent(AsyncWebSocket * server, AsyncWebSocketClient * client, AwsEventType type, void * arg, uint8_t *data, size_t len); | ||||
| void sendDataWs(AsyncWebSocketClient * client = nullptr); | ||||
|  | ||||
| //xml.cpp | ||||
| void XML_response(AsyncWebServerRequest *request, char* dest = nullptr); | ||||
| void URL_response(AsyncWebServerRequest *request); | ||||
| void sappend(char stype, const char* key, int val); | ||||
| void sappends(char stype, const char* key, char* val); | ||||
| void getSettingsJS(byte subPage, char* dest); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										414
									
								
								wled00/file.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										414
									
								
								wled00/file.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,414 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * Utility for SPIFFS filesystem | ||||
|  */ | ||||
|  | ||||
| #ifdef ARDUINO_ARCH_ESP32 //FS info bare IDF function until FS wrapper is available for ESP32 | ||||
| #if WLED_FS != LITTLEFS | ||||
|   #include "esp_spiffs.h" | ||||
| #endif | ||||
| #endif | ||||
|  | ||||
| #define FS_BUFSIZE 256 | ||||
|  | ||||
| /* | ||||
|  * Structural requirements for files managed by writeObjectToFile() and readObjectFromFile() utilities: | ||||
|  * 1. File must be a string representation of a valid JSON object | ||||
|  * 2. File must have '{' as first character | ||||
|  * 3. There must not be any additional characters between a root-level key and its value object (e.g. space, tab, newline) | ||||
|  * 4. There must not be any characters between an root object-separating ',' and the next object key string | ||||
|  * 5. There may be any number of spaces, tabs, and/or newlines before such object-separating ',' | ||||
|  * 6. There must not be more than 5 consecutive spaces at any point except for those permitted in condition 5 | ||||
|  * 7. If it is desired to delete the first usable object (e.g. preset file), a dummy object '"0":{}' is inserted at the beginning. | ||||
|  *    It shall be disregarded by receiving software. | ||||
|  *    The reason for it is that deleting the first preset would require special code to handle commas between it and the 2nd preset | ||||
|  */ | ||||
|  | ||||
| // There are no consecutive spaces longer than this in the file, so if more space is required, findSpace() can return false immediately | ||||
| // Actual space may be lower | ||||
| uint16_t knownLargestSpace = UINT16_MAX; | ||||
|  | ||||
| File f; | ||||
|  | ||||
| //wrapper to find out how long closing takes | ||||
| void closeFile() { | ||||
|   #ifdef WLED_DEBUG_FS | ||||
|     DEBUGFS_PRINT(F("Close -> ")); | ||||
|     uint32_t s = millis(); | ||||
|   #endif | ||||
|   f.close(); | ||||
|   DEBUGFS_PRINTF("took %d ms\n", millis() - s); | ||||
|   doCloseFile = false; | ||||
| } | ||||
|  | ||||
| //find() that reads and buffers data from file stream in 256-byte blocks. | ||||
| //Significantly faster, f.find(key) can take SECONDS for multi-kB files | ||||
| bool bufferedFind(const char *target, bool fromStart = true) { | ||||
|   #ifdef WLED_DEBUG_FS | ||||
|     DEBUGFS_PRINT("Find "); | ||||
|     DEBUGFS_PRINTLN(target); | ||||
|     uint32_t s = millis(); | ||||
|   #endif | ||||
|  | ||||
|   if (!f || !f.size()) return false; | ||||
|   size_t targetLen = strlen(target); | ||||
|  | ||||
|   size_t index = 0; | ||||
|   uint16_t bufsize = 0, count = 0; | ||||
|   byte buf[FS_BUFSIZE]; | ||||
|   if (fromStart) f.seek(0); | ||||
|  | ||||
|   while (f.position() < f.size() -1) { | ||||
|     bufsize = f.read(buf, FS_BUFSIZE); | ||||
|     count = 0; | ||||
|     while (count < bufsize) { | ||||
|       if(buf[count] != target[index]) | ||||
|       index = 0; // reset index if any char does not match | ||||
|  | ||||
|       if(buf[count] == target[index]) { | ||||
|         if(++index >= targetLen) { // return true if all chars in the target match | ||||
|           f.seek((f.position() - bufsize) + count +1); | ||||
|           DEBUGFS_PRINTF("Found at pos %d, took %d ms", f.position(), millis() - s); | ||||
|           return true; | ||||
|         } | ||||
|       } | ||||
|       count++; | ||||
|     } | ||||
|   } | ||||
|   DEBUGFS_PRINTF("No match, took %d ms\n", millis() - s); | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| //find empty spots in file stream in 256-byte blocks. | ||||
| bool bufferedFindSpace(uint16_t targetLen, bool fromStart = true) { | ||||
|  | ||||
|   #ifdef WLED_DEBUG_FS | ||||
|     DEBUGFS_PRINTF("Find %d spaces\n", targetLen); | ||||
|     uint32_t s = millis(); | ||||
|   #endif | ||||
|  | ||||
|   if (knownLargestSpace < targetLen) { | ||||
|     DEBUGFS_PRINT(F("No match, KLS ")); | ||||
|     DEBUGFS_PRINTLN(knownLargestSpace); | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   if (!f || !f.size()) return false; | ||||
|  | ||||
|   uint16_t index = 0; | ||||
|   uint16_t bufsize = 0, count = 0; | ||||
|   byte buf[FS_BUFSIZE]; | ||||
|   if (fromStart) f.seek(0); | ||||
|  | ||||
|   while (f.position() < f.size() -1) { | ||||
|     bufsize = f.read(buf, FS_BUFSIZE); | ||||
|     count = 0; | ||||
|      | ||||
|     while (count < bufsize) { | ||||
|       if(buf[count] == ' ') { | ||||
|         if(++index >= targetLen) { // return true if space long enough | ||||
|           if (fromStart) { | ||||
|             f.seek((f.position() - bufsize) + count +1 - targetLen); | ||||
|             knownLargestSpace = UINT16_MAX; //there may be larger spaces after, so we don't know | ||||
|           } | ||||
|           DEBUGFS_PRINTF("Found at pos %d, took %d ms", f.position(), millis() - s); | ||||
|           return true; | ||||
|         } | ||||
|       } else { | ||||
|         if (!fromStart) return false; | ||||
|         if (index) { | ||||
|           if (knownLargestSpace < index || knownLargestSpace == UINT16_MAX) knownLargestSpace = index; | ||||
|           index = 0; // reset index if not space | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       count++; | ||||
|     } | ||||
|   } | ||||
|   DEBUGFS_PRINTF("No match, took %d ms\n", millis() - s); | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| //find the closing bracket corresponding to the opening bracket at the file pos when calling this function | ||||
| bool bufferedFindObjectEnd() { | ||||
|   #ifdef WLED_DEBUG_FS | ||||
|     DEBUGFS_PRINTLN(F("Find obj end")); | ||||
|     uint32_t s = millis(); | ||||
|   #endif | ||||
|  | ||||
|   if (!f || !f.size()) return false; | ||||
|  | ||||
|   uint16_t objDepth = 0; //num of '{' minus num of '}'. return once 0 | ||||
|   uint16_t bufsize = 0, count = 0; | ||||
|   //size_t start = f.position(); | ||||
|   byte buf[FS_BUFSIZE]; | ||||
|  | ||||
|   while (f.position() < f.size() -1) { | ||||
|     bufsize = f.read(buf, FS_BUFSIZE); | ||||
|     count = 0; | ||||
|      | ||||
|     while (count < bufsize) { | ||||
|       if (buf[count] == '{') objDepth++; | ||||
|       if (buf[count] == '}') objDepth--; | ||||
|       if (objDepth == 0) { | ||||
|         f.seek((f.position() - bufsize) + count +1); | ||||
|         DEBUGFS_PRINTF("} at pos %d, took %d ms", f.position(), millis() - s); | ||||
|         return true; | ||||
|       } | ||||
|       count++; | ||||
|     } | ||||
|   } | ||||
|   DEBUGFS_PRINTF("No match, took %d ms\n", millis() - s); | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| //fills n bytes from current file pos with ' ' characters | ||||
| void writeSpace(uint16_t l) | ||||
| { | ||||
|   byte buf[FS_BUFSIZE]; | ||||
|   memset(buf, ' ', FS_BUFSIZE); | ||||
|  | ||||
|   while (l > 0) { | ||||
|     uint16_t block = (l>FS_BUFSIZE) ? FS_BUFSIZE : l; | ||||
|     f.write(buf, block); | ||||
|     l -= block; | ||||
|   } | ||||
|  | ||||
|   if (knownLargestSpace < l) knownLargestSpace = l; | ||||
| } | ||||
|  | ||||
| bool appendObjectToFile(const char* key, JsonDocument* content, uint32_t s, uint32_t contentLen = 0) | ||||
| { | ||||
|   #ifdef WLED_DEBUG_FS | ||||
|     DEBUGFS_PRINTLN(F("Append")); | ||||
|     uint32_t s1 = millis(); | ||||
|   #endif | ||||
|   uint32_t pos = 0; | ||||
|   if (!f) return false; | ||||
|  | ||||
|   if (f.size() < 3) { | ||||
|     char init[10]; | ||||
|     strcpy_P(init, PSTR("{\"0\":{}}")); | ||||
|     f.print(init); | ||||
|   } | ||||
|  | ||||
|   if (content->isNull()) { | ||||
|     doCloseFile = true; | ||||
|     return true; //nothing  to append | ||||
|   } | ||||
|    | ||||
|   //if there is enough empty space in file, insert there instead of appending | ||||
|   if (!contentLen) contentLen = measureJson(*content); | ||||
|   DEBUGFS_PRINTF("CLen %d\n", contentLen); | ||||
|   if (bufferedFindSpace(contentLen + strlen(key) + 1)) { | ||||
|     if (f.position() > 2) f.write(','); //add comma if not first object | ||||
|     f.print(key); | ||||
|     serializeJson(*content, f); | ||||
|     DEBUGFS_PRINTF("Inserted, took %d ms (total %d)", millis() - s1, millis() - s); | ||||
|     doCloseFile = true; | ||||
|     return true; | ||||
|   } | ||||
|  | ||||
|   //not enough space, append at end | ||||
|  | ||||
|   //permitted space for presets exceeded | ||||
|   updateFSInfo(); | ||||
|    | ||||
|   if (f.size() + 9000 > (fsBytesTotal - fsBytesUsed)) { //make sure there is enough space to at least copy the file once | ||||
|     errorFlag = ERR_FS_QUOTA; | ||||
|     doCloseFile = true; | ||||
|     return false; | ||||
|   } | ||||
|    | ||||
|   //check if last character in file is '}' (typical) | ||||
|   uint32_t eof = f.size() -1; | ||||
|   f.seek(eof, SeekSet); | ||||
|   if (f.read() == '}') pos = eof; | ||||
|    | ||||
|   if (pos == 0) //not found | ||||
|   { | ||||
|     DEBUGFS_PRINTLN("not }"); | ||||
|     f.seek(0); | ||||
|     while (bufferedFind("}",false)) //find last closing bracket in JSON if not last char | ||||
|     { | ||||
|       pos = f.position(); | ||||
|     } | ||||
|     if (pos > 0) pos--; | ||||
|   } | ||||
|   DEBUGFS_PRINT("pos "); DEBUGFS_PRINTLN(pos); | ||||
|   if (pos > 2) | ||||
|   { | ||||
|     f.seek(pos, SeekSet); | ||||
|     f.write(','); | ||||
|   } else { //file content is not valid JSON object | ||||
|     f.seek(0, SeekSet); | ||||
|     f.print('{'); //start JSON | ||||
|   } | ||||
|  | ||||
|   f.print(key); | ||||
|  | ||||
|   //Append object | ||||
|   serializeJson(*content, f); | ||||
|   f.write('}'); | ||||
|  | ||||
|   doCloseFile = true; | ||||
|   DEBUGFS_PRINTF("Appended, took %d ms (total %d)", millis() - s1, millis() - s); | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| bool writeObjectToFileUsingId(const char* file, uint16_t id, JsonDocument* content) | ||||
| { | ||||
|   char objKey[10]; | ||||
|   sprintf(objKey, "\"%d\":", id); | ||||
|   return writeObjectToFile(file, objKey, content); | ||||
| } | ||||
|  | ||||
| bool writeObjectToFile(const char* file, const char* key, JsonDocument* content) | ||||
| { | ||||
|   uint32_t s = 0; //timing | ||||
|   #ifdef WLED_DEBUG_FS | ||||
|     DEBUGFS_PRINTF("Write to %s with key %s >>>\n", file, (key==nullptr)?"nullptr":key); | ||||
|     serializeJson(*content, Serial); DEBUGFS_PRINTLN(); | ||||
|     s = millis(); | ||||
|   #endif | ||||
|  | ||||
|   uint32_t pos = 0; | ||||
|   f = WLED_FS.open(file, "r+"); | ||||
|   if (!f && !WLED_FS.exists(file)) f = WLED_FS.open(file, "w+"); | ||||
|   if (!f) { | ||||
|     DEBUGFS_PRINTLN(F("Failed to open!")); | ||||
|     return false; | ||||
|   } | ||||
|    | ||||
|   if (!bufferedFind(key)) //key does not exist in file | ||||
|   { | ||||
|     return appendObjectToFile(key, content, s); | ||||
|   }  | ||||
|    | ||||
|   //an object with this key already exists, replace or delete it | ||||
|   pos = f.position(); | ||||
|   //measure out end of old object | ||||
|   bufferedFindObjectEnd(); | ||||
|   uint32_t pos2 = f.position(); | ||||
|  | ||||
|   uint32_t oldLen = pos2 - pos; | ||||
|   DEBUGFS_PRINTF("Old obj len %d\n", oldLen); | ||||
|  | ||||
|   //Three cases: | ||||
|   //1. The new content is null, overwrite old obj with spaces | ||||
|   //2. The new content is smaller than the old, overwrite and fill diff with spaces | ||||
|   //3. The new content is larger than the old, but smaller than old + trailing spaces, overwrite with new | ||||
|   //4. The new content is larger than old + trailing spaces, delete old and append | ||||
|    | ||||
|   uint32_t contentLen = 0; | ||||
|   if (!content->isNull()) contentLen = measureJson(*content); | ||||
|  | ||||
|   if (contentLen && contentLen <= oldLen) { //replace and fill diff with spaces | ||||
|     DEBUGFS_PRINTLN(F("replace")); | ||||
|     f.seek(pos); | ||||
|     serializeJson(*content, f); | ||||
|     writeSpace(pos2 - f.position()); | ||||
|   } else if (contentLen && bufferedFindSpace(contentLen - oldLen, false)) { //enough leading spaces to replace | ||||
|     DEBUGFS_PRINTLN(F("replace (trailing)")); | ||||
|     f.seek(pos); | ||||
|     serializeJson(*content, f); | ||||
|   } else { | ||||
|     DEBUGFS_PRINTLN(F("delete")); | ||||
|     pos -= strlen(key); | ||||
|     if (pos > 3) pos--; //also delete leading comma if not first object | ||||
|     f.seek(pos); | ||||
|     writeSpace(pos2 - pos); | ||||
|     if (contentLen) return appendObjectToFile(key, content, s, contentLen); | ||||
|   } | ||||
|  | ||||
|   doCloseFile = true; | ||||
|   DEBUGFS_PRINTF("Replaced/deleted, took %d ms\n", millis() - s); | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| bool readObjectFromFileUsingId(const char* file, uint16_t id, JsonDocument* dest) | ||||
| { | ||||
|   char objKey[10]; | ||||
|   sprintf(objKey, "\"%d\":", id); | ||||
|   return readObjectFromFile(file, objKey, dest); | ||||
| } | ||||
|  | ||||
| //if the key is a nullptr, deserialize entire object | ||||
| bool readObjectFromFile(const char* file, const char* key, JsonDocument* dest) | ||||
| { | ||||
|   if (doCloseFile) closeFile(); | ||||
|   #ifdef WLED_DEBUG_FS | ||||
|     DEBUGFS_PRINTF("Read from %s with key %s >>>\n", file, (key==nullptr)?"nullptr":key); | ||||
|     uint32_t s = millis(); | ||||
|   #endif | ||||
|   f = WLED_FS.open(file, "r"); | ||||
|   if (!f) return false; | ||||
|  | ||||
|   if (key != nullptr && !bufferedFind(key)) //key does not exist in file | ||||
|   { | ||||
|     f.close(); | ||||
|     dest->clear(); | ||||
|     DEBUGFS_PRINTLN(F("Obj not found.")); | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   deserializeJson(*dest, f); | ||||
|  | ||||
|   f.close(); | ||||
|   DEBUGFS_PRINTF("Read, took %d ms\n", millis() - s); | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| void updateFSInfo() { | ||||
|   #ifdef ARDUINO_ARCH_ESP32 | ||||
|     #if WLED_FS == LITTLEFS | ||||
|     fsBytesTotal = LITTLEFS.totalBytes(); | ||||
|     fsBytesUsed = LITTLEFS.usedBytes(); | ||||
|     #else | ||||
|     esp_spiffs_info(nullptr, &fsBytesTotal, &fsBytesUsed); | ||||
|     #endif | ||||
|   #else | ||||
|     FSInfo fsi; | ||||
|     WLED_FS.info(fsi); | ||||
|     fsBytesUsed  = fsi.usedBytes; | ||||
|     fsBytesTotal = fsi.totalBytes; | ||||
|   #endif | ||||
| } | ||||
|  | ||||
|  | ||||
| //Un-comment any file types you need | ||||
| String getContentType(AsyncWebServerRequest* request, String filename){ | ||||
|   if(request->hasArg("download")) return "application/octet-stream"; | ||||
|   else if(filename.endsWith(".htm")) return "text/html"; | ||||
|   else if(filename.endsWith(".html")) return "text/html"; | ||||
| //  else if(filename.endsWith(".css")) return "text/css"; | ||||
| //  else if(filename.endsWith(".js")) return "application/javascript"; | ||||
|   else if(filename.endsWith(".json")) return "application/json"; | ||||
|   else if(filename.endsWith(".png")) return "image/png"; | ||||
| //  else if(filename.endsWith(".gif")) return "image/gif"; | ||||
|   else if(filename.endsWith(".jpg")) return "image/jpeg"; | ||||
|   else if(filename.endsWith(".ico")) return "image/x-icon"; | ||||
| //  else if(filename.endsWith(".xml")) return "text/xml"; | ||||
| //  else if(filename.endsWith(".pdf")) return "application/x-pdf"; | ||||
| //  else if(filename.endsWith(".zip")) return "application/x-zip"; | ||||
| //  else if(filename.endsWith(".gz")) return "application/x-gzip"; | ||||
|   return "text/plain"; | ||||
| } | ||||
|  | ||||
| bool handleFileRead(AsyncWebServerRequest* request, String path){ | ||||
|   DEBUG_PRINTLN("FileRead: " + path); | ||||
|   if(path.endsWith("/")) path += "index.htm"; | ||||
|   if(path.indexOf("sec") > -1) return false; | ||||
|   String contentType = getContentType(request, path); | ||||
|   /*String pathWithGz = path + ".gz"; | ||||
|   if(WLED_FS.exists(pathWithGz)){ | ||||
|     request->send(WLED_FS, pathWithGz, contentType); | ||||
|     return true; | ||||
|   }*/ | ||||
|   if(WLED_FS.exists(path)) { | ||||
|     request->send(WLED_FS, path, contentType); | ||||
|     return true; | ||||
|   } | ||||
|   return false; | ||||
| } | ||||
| @@ -1,103 +1,118 @@ | ||||
| /* | ||||
|  * Various pages | ||||
|  */ | ||||
|  * More web UI HTML source arrays. | ||||
|  * This file is auto generated, please don't make any changes manually. | ||||
|  * Instead, see https://github.com/Aircoookie/WLED/wiki/Add-own-functionality#web-ui | ||||
|  * to find out how to easily modify the web UI source! | ||||
|  */  | ||||
|  | ||||
| //USER HTML HERE (/u subpage) | ||||
| const char PAGE_usermod[] PROGMEM = R"=====(<!DOCTYPE html> | ||||
| <html><body>No usermod custom web page set.</body></html>)====="; | ||||
| // Autogenerated from wled00/data/usermod.htm, do not edit!! | ||||
| const char PAGE_usermod[] PROGMEM = R"=====(<!DOCTYPE html><html><body>No usermod custom web page set.</body></html>)====="; | ||||
|  | ||||
|  | ||||
| //server message | ||||
| const char PAGE_msg[] PROGMEM = R"=====(<!DOCTYPE html> | ||||
| <html><head><meta content='width=device-width' name='viewport'> | ||||
| <title>WLED Message</title> | ||||
| <script>function B(){window.history.back()};function RS(){window.location = "/settings";}function RP(){top.location.href="/";}</script> | ||||
| <style>.bt{background:#333;color:#fff;font-family:Verdana,sans-serif;border:.3ch solid #333;display:inline-block;font-size:20px;margin:8px;margin-top:12px}body{font-family:Verdana,sans-serif;text-align:center;background:#222;color:#fff;line-height:200%%;margin:0}</style></head> | ||||
| <body><h2>%MSG%</body></html>)====="; | ||||
| // Autogenerated from wled00/data/msg.htm, do not edit!! | ||||
| const char PAGE_msg[] PROGMEM = R"=====(<!DOCTYPE html><html><head><meta content="width=device-width" name="viewport"> | ||||
| <title>WLED Message</title><script> | ||||
| function B(){window.history.back()}function RS(){window.location="/settings"}function RP(){top.location.href="/"} | ||||
| </script><style> | ||||
| .bt{background:#333;color:#fff;font-family:Verdana,sans-serif;border:.3ch solid #333;display:inline-block;font-size:20px;margin:8px;margin-top:12px}body{font-family:Verdana,sans-serif;text-align:center;background:#222;color:#fff;line-height:200%%;margin:0} | ||||
| </style></head><body><h2>%MSG%</body></html>)====="; | ||||
|  | ||||
|  | ||||
| //firmware update page | ||||
| const char PAGE_update[] PROGMEM = R"=====(<!DOCTYPE html> | ||||
| <html><head><meta content='width=device-width' name='viewport'><title>WLED Update</title><script>function B(){window.history.back()}</script> | ||||
| <style>.bt{background:#333;color:#fff;font-family:Verdana,sans-serif;border:.3ch solid #333;display:inline-block;font-size:20px;margin:8px;margin-top:12px}input[type=file]{font-size:16px}body{font-family:Verdana,sans-serif;text-align:center;background:#222;color:#fff;line-height:200%}</style></head> | ||||
| <body><h2>WLED Software Update</h2>Installed version: 0.9.1<br>Download the latest binary: <a href="https://github.com/Aircoookie/WLED/releases"><img src="https://img.shields.io/github/release/Aircoookie/WLED.svg?style=flat-square"></a><br><form method='POST' action='/update' enctype='multipart/form-data'><input type='file' class="bt" name='update' required><br><input type='submit' class="bt" value='Update!'></form><button type="button" class="bt" onclick="B()">Back</button></body></html>)====="; | ||||
| #ifdef WLED_ENABLE_DMX | ||||
|  | ||||
| // Autogenerated from wled00/data/dmxmap.htm, do not edit!! | ||||
| const char PAGE_dmxmap[] PROGMEM = R"=====(<!DOCTYPE html><html><head><meta content="width=device-width" name="viewport"> | ||||
| <title>DMX Map</title><script> | ||||
| function B(){window.history.back()}function RS(){window.location="/settings"}function RP(){top.location.href="/"}function FM() {%DMXVARS% | ||||
| var t=["SET 0","RED","GREEN","BLUE","WHITE","SHUTTER","SET 255","DISABLED"],n=[];for(i=0;i<512;i++)n.push(7);for(i=0;i<LC;i++)for(FS=CS+CG*i,j=0;j<CN;j++)DA=FS+j,n[DA-1]=CH[j];for(DMXMap="",i=0;i<512;i++)isstart="",(i+1)%10==0&&(isstart="S"),DMXMap+='<div class="anytype '+isstart+" type"+n[i]+'">'+String(i+1)+"<br />"+t[n[i]]+"</div>";document.getElementById("map").innerHTML=DMXMap} | ||||
| </script><style> | ||||
| .anytype{border:1px solid #fff;margin:1px;float:left;width:100px;height:100px}.S{margin:0;border:2px solid #fff}.type7{color:#888;border:1px dotted grey}.type6{color:#fff}.type4{color:#fff;font-weight:700}.type3{color:#00f;font-weight:700}.type2{color:#0f0;font-weight:700}.type1{color:red;font-weight:700}.bt{background:#333;color:#fff;font-family:Verdana,sans-serif;border:.3ch solid #333;display:inline-block;font-size:20px;margin:8px;margin-top:12px}body{font-family:Verdana,sans-serif;text-align:center;background:#222;color:#fff;line-height:200%%;margin:0} | ||||
| </style></head><body onload="FM()"><div id="map">...</div></body></html>)====="; | ||||
|  | ||||
|  | ||||
| //new user welcome page | ||||
| const char PAGE_welcome[] PROGMEM = R"=====(<!DOCTYPE html><html><head><meta charset=utf-8><meta content='width=device-width' name=viewport><meta name=theme-color content=#333333><title>WLED Setup</title> <style>body{font-family:Verdana,Helvetica,sans-serif;text-align:center;background-color:#333;margin:0;color:#fff}button{outline:0;cursor:pointer}.btn{padding:8px;margin:10px;width:230px;text-transform:uppercase;font-family:helvetica;font-size:19px;background-color:#222;color:white;border:0 solid white;border-radius:5px}svg{fill:#fff}</style></head> | ||||
| <body> <svg style=position:absolute;width:0;height:0;overflow:hidden version=1.1 xmlns=http://www.w3.org/2000/svg> <defs> <symbol id=lnr-smile viewBox="0 0 1024 1024"><path d="M486.4 1024c-129.922 0-252.067-50.594-343.936-142.464s-142.464-214.014-142.464-343.936c0-129.923 50.595-252.067 142.464-343.936s214.013-142.464 343.936-142.464c129.922 0 252.067 50.595 343.936 142.464s142.464 214.014 142.464 343.936-50.594 252.067-142.464 343.936c-91.869 91.87-214.014 142.464-343.936 142.464zM486.4 102.4c-239.97 0-435.2 195.23-435.2 435.2s195.23 435.2 435.2 435.2 435.2-195.23 435.2-435.2-195.23-435.2-435.2-435.2z"></path><path d="M332.8 409.6c-42.347 0-76.8-34.453-76.8-76.8s34.453-76.8 76.8-76.8 76.8 34.453 76.8 76.8-34.453 76.8-76.8 76.8zM332.8 307.2c-14.115 0-25.6 11.485-25.6 25.6s11.485 25.6 25.6 25.6 25.6-11.485 25.6-25.6-11.485-25.6-25.6-25.6z"></path><path d="M640 409.6c-42.349 0-76.8-34.453-76.8-76.8s34.451-76.8 76.8-76.8 76.8 34.453 76.8 76.8-34.451 76.8-76.8 76.8zM640 307.2c-14.115 0-25.6 11.485-25.6 25.6s11.485 25.6 25.6 25.6 25.6-11.485 25.6-25.6-11.485-25.6-25.6-25.6z"></path><path d="M486.4 870.4c-183.506 0-332.8-149.294-332.8-332.8 0-14.139 11.462-25.6 25.6-25.6s25.6 11.461 25.6 25.6c0 155.275 126.325 281.6 281.6 281.6s281.6-126.325 281.6-281.6c0-14.139 11.461-25.6 25.6-25.6s25.6 11.461 25.6 25.6c0 183.506-149.294 332.8-332.8 332.8z"></path></symbol> </defs></svg> <br><br> | ||||
| <svg><use xlink:href=#lnr-smile></use></svg><h1>Welcome to WLED!</h1><h3>Thank you for installing my application!</h3> If you encounter a bug or have a question/feature suggestion, feel free to open a GitHub issue!<br><br> <b>Next steps:</b><br><br> Connect the module to your local WiFi here!<br> <button class=btn onclick="window.location.href='/settings/wifi'">WiFi settings</button><br> <i>Just trying this out in AP mode?</i><br> <button class=btn onclick="window.location.href='/sliders'">To the controls!</button></body></html>)====="; | ||||
| #else | ||||
| const char PAGE_dmxmap[] PROGMEM = R"=====()====="; | ||||
| #endif | ||||
|  | ||||
| // Autogenerated from wled00/data/update.htm, do not edit!! | ||||
| const char PAGE_update[] PROGMEM = R"=====(<!DOCTYPE html><html><head><meta content="width=device-width" name="viewport"> | ||||
| <title>WLED Update</title><script> | ||||
| function B(){window.history.back()}function U(){document.getElementById("uf").style.display="none",document.getElementById("msg").style.display="block"} | ||||
| </script><style> | ||||
| .bt{background:#333;color:#fff;font-family:Verdana,sans-serif;border:.3ch solid #333;display:inline-block;font-size:20px;margin:8px;margin-top:12px}input[type=file]{font-size:16px}body{font-family:Verdana,sans-serif;text-align:center;background:#222;color:#fff;line-height:200%}#msg{display:none} | ||||
| </style></head><body><h2>WLED Software Update</h2><form method="POST"  | ||||
| action="/update" id="uf" enctype="multipart/form-data" onsubmit="U()"> | ||||
| Installed version: 0.12.0<br>Download the latest binary: <a  | ||||
| href="https://github.com/Aircoookie/WLED/releases" target="_blank"><img  | ||||
| src="https://img.shields.io/github/release/Aircoookie/WLED.svg?style=flat-square"> | ||||
| </a><br><input type="file" class="bt" name="update" accept=".bin" required><br> | ||||
| <input type="submit" class="bt" value="Update!"><br><button type="button"  | ||||
| class="bt" onclick="B()">Back</button></form><div id="msg"><b>Updating...</b> | ||||
| <br>Please do not close or refresh the page :)</div></body></html>)====="; | ||||
|  | ||||
|  | ||||
| //liveview | ||||
| const char PAGE_liveview[] PROGMEM = R"=====(<!DOCTYPE html> | ||||
| <html><head> | ||||
| <meta name=viewport content="width=device-width, initial-scale=1, minimum-scale=1"> | ||||
| <meta charset=utf-8> | ||||
| <meta name=theme-color content=#222222> | ||||
| <title>WLED Live Preview</title> | ||||
| <style> | ||||
| body {margin: 0;} | ||||
| #canv {background: black;filter: brightness(175%);width: 100%;height: 100%;position: absolute;} | ||||
| </style></head> | ||||
| <body> | ||||
| <div id="canv" /> | ||||
| <script> | ||||
| update(); | ||||
| var tmout = null; | ||||
| function update() | ||||
| { | ||||
| if (document.hidden) { | ||||
| clearTimeout(tmout); | ||||
| tmout = setTimeout(update, 250); | ||||
| return; | ||||
| } | ||||
| fetch('/json/live') | ||||
| .then(res => { | ||||
| if (!res.ok) { | ||||
| clearTimeout(tmout); | ||||
| tmout = setTimeout(update, 2500); | ||||
| } | ||||
| return res.json(); | ||||
| }) | ||||
| .then(json => { | ||||
| var str = "linear-gradient(90deg,"; | ||||
| var len = json.leds.length; | ||||
| for (i = 0; i < len; i++) { | ||||
| var leddata = json.leds[i]; | ||||
| if (leddata.length > 6) leddata = leddata.substring(2); | ||||
| str += "#" + leddata; | ||||
| if (i < len -1) str += "," | ||||
| } | ||||
| str += ")"; | ||||
| document.getElementById("canv").style.background = str; | ||||
| clearTimeout(tmout); | ||||
| tmout = setTimeout(update, 40); | ||||
| }) | ||||
| .catch(function (error) { | ||||
| clearTimeout(tmout); | ||||
| tmout = setTimeout(update, 2500); | ||||
| }) | ||||
| } | ||||
| </script> | ||||
| // Autogenerated from wled00/data/welcome.htm, do not edit!! | ||||
| const char PAGE_welcome[] PROGMEM = R"=====(<!DOCTYPE html><html><head><meta charset="utf-8"><meta  | ||||
| content="width=device-width" name="viewport"><meta name="theme-color"  | ||||
| content="#222222"><title>Welcome!</title><style> | ||||
| body{font-family:Verdana,Helvetica,sans-serif;text-align:center;background-color:#222;margin:0;color:#fff}button{outline:0;cursor:pointer;padding:8px;margin:10px;width:230px;text-transform:uppercase;font-family:helvetica;font-size:19px;background-color:#333;color:#fff;border:0 solid #fff;border-radius:25px;filter:drop-shadow(0 0 1px #000)}img{width:999px;max-width:85%;image-rendering:pixelated;image-rendering:crisp-edges;margin:25px 0 -10px 0;animation:fi 1s}@keyframes fi{from{opacity:0}to{opacity:1}}.main{animation:fi 1.5s .7s both} | ||||
| </style></head><body><img alt=""  | ||||
| src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAHEAAAAjCAYAAACjOOUsAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsAAAA7AAWrWiQkAAATfSURBVGhD7ZlPaBVHHMdnS2Ox5iKYGIyYJwQlUTHG6kUwtSAIBsEiBopgKR5i68mbIPgEwZun0uYgouIlobQIKQheIsSLaBSaRFoEc3gtMa9HLaiH7X4n81tn583u/HkTaZ/7gR+zMzvv93bfZ3/777GSkpL/AJFo3Tg/H7OuPtGxY7CTsZnjkd/3lRTi/qMKgfGXt8TAO6KfT2TG1f7u6ROlyBXgI9HaQRW4+JR3ox92Z1p1XO3PLCUVORHHvFMSDDeJBuJvH4mlZdS+C3c7B+Ln3fvivw59w0MMOzP/FYvj75K4PMhDDDszdbM7rk1V4hcPD/MQw858MbEnPvJgf3x29hQPMdwUbqc2qRIhCBVGrQk+Lzm9mq6NkIe2t20N7xOf7NjK2w13rlltM+Sh7VvLu+/YOMib6NyMVR7IQ9u7qY33ibb2bbxd/9mvVnkgD217ZTXvE72fbuHtle1X3VxIuH1QOZ2mmG5yaH4yr0giBKryVCDTJBICG+SpJDJNIiFQlacCmSaREKjKU4FMX5F+p1NIk8OExTwbgeD1b7+zotOrlUBQm2FFp1cbgeDtyzlWdHq1EQie/fMH8z29Br0myiwdr7PkWpSGC2se/8IqtWnRa+y7Ur+Q3FF97/X7ZFjV/4B1Dz1Pw5WrlZ/Y7T330gjFikkE9Xo9jSJ0Vah+Ru7nVWNRFWq3Iacai6qQ9kfOl1eNuirUfZ7wrcbgEqkCOyc6MmEDVdyrXUd5ENR3rUiqwI6LnTx8oQp8M783E66cWjiWCarMZlnRSix5PwSTSBXY0WFXdf9HfK+FKlSBoX6rjEQ8XMutD3nn+1Yg5L755Mrzk0qkh2xMwLOYr0iXa6CJZu9KQ+N7LSSoAuma6EKRn1TiwaUn0bO3r/gy7v42/3nf+cEzxFGq5jD1TYTYJhAqj4pt3iI/mdMpTbQRqD4HhqhA9S40r28L3ZWmz4kUjlAFNvucSNBzomtF5vlJJVarVb53mMgHLMBRRBEaNWez3xFqW5vN0+znyQ/5Ahlho6OjcVdXl+jxiVmhHn8Gq+jendq+cpPRvUO1fuUmo3mHavvKTUb3DtX2lZtM3jtUWRqQ3TRMBrLMjMgVkAiB2waX/1nwgURygTv885BICOzrH+BjPpBICBzo38nHfJBFksDFxUU2NjaWjhMNAwSJ1Ems/n2RDQ0NsQNzn4sVjF1vvyGWGOvp6cmsk+frJLpWISFXo1cVElI1+lQhIVejTxUSajVCYp5AoB0k6AhIRQqJo7OnGRe87gIfxl9N1Y8nlpcTKpUK+/rlSdFLDghpft5fUTho0GJD5WW0RetUWi1PgwMNmbtTFdhXaRBowdj2H3kuVGQeyIkg5GVQtE6mFfPoPMgUSqSjgo4Gdqk/whfxpGeSaqJIxjFGsbCwkF2fBHJh3ZH5RpE2RxtBedJtkmi1PLSMdXwgh0KJAIkBEiLQV5MODw9ndiBvp/I2VIwZd5T4EPJQLvRNGCVSYgr0xSotIyMjYkkPfV7eUFNOHa2cx+X3BkaJAIkoxJAWCBwfHxc9PdhBbFzSRgixzHfahVbNA2x/b8JKoi0QaKpE7KC8cVjGmOha06p5fAgq0aYSS8ITROLk5GRUq9UyAt/XUVgSGNyl0kNrMyBHchCUeSwJejotKSkp+VBh7F9fIy7OdlOyMwAAAABJRU5ErkJggg=="> | ||||
| <div class="main"><h1>Welcome to WLED!</h1><h3> | ||||
| Thank you for installing my application!</h3><b>Next steps:</b><br><br> | ||||
| Connect the module to your local WiFi here!<br><button  | ||||
| onclick='window.location.href="/settings/wifi"'>WiFi settings</button><br><i> | ||||
| Just trying this out in AP mode?</i><br><button  | ||||
| onclick='window.location.href="/sliders"'>To the controls!</button><br></div> | ||||
| </body></html>)====="; | ||||
|  | ||||
|  | ||||
| /* | ||||
|  * favicon | ||||
|  */ | ||||
| // Autogenerated from wled00/data/liveview.htm, do not edit!! | ||||
| const char PAGE_liveview[] PROGMEM = R"=====(<!DOCTYPE html><html><head><meta name="viewport"  | ||||
| content="width=device-width,initial-scale=1,minimum-scale=1"><meta  | ||||
| charset="utf-8"><meta name="theme-color" content="#222222"><title> | ||||
| WLED Live Preview</title><style> | ||||
| body{margin:0}#canv{background:#000;filter:brightness(175%);width:100%;height:100%;position:absolute} | ||||
| </style></head><body><div id="canv"><script> | ||||
| update();var tmout=null;function update(){if(document.hidden)return clearTimeout(tmout),void(tmout=setTimeout(update,250));fetch("/json/live").then(t=>(t.ok||(clearTimeout(tmout),tmout=setTimeout(update,2500)),t.json())).then(t=>{var e="linear-gradient(90deg,",u=t.leds.length;for(i=0;i<u;i++){var o=t.leds[i];o.length>6&&(o=o.substring(2)),e+="#"+o,i<u-1&&(e+=",")}e+=")",document.getElementById("canv").style.background=e,clearTimeout(tmout),tmout=setTimeout(update,40)}).catch((function(t){clearTimeout(tmout),tmout=setTimeout(update,2500)}))} | ||||
| </script></body></html>)====="; | ||||
|  | ||||
|  | ||||
| // Autogenerated from wled00/data/liveviewws.htm, do not edit!! | ||||
| const char PAGE_liveviewws[] PROGMEM = R"=====(<!DOCTYPE html><html><head><meta name="viewport"  | ||||
| content="width=device-width,initial-scale=1,minimum-scale=1"><meta  | ||||
| charset="utf-8"><meta name="theme-color" content="#222222"><title> | ||||
| WLED Live Preview</title><style> | ||||
| body{margin:0}#canv{background:#000;filter:brightness(175%);width:100%;height:100%;position:absolute} | ||||
| </style></head><body><div id="canv"><script> | ||||
| console.info("Live-Preview websocket opening");var socket=new WebSocket("ws://"+document.location.host+"/ws");function updatePreview(e){var o="linear-gradient(90deg,",n=e.length;for(i=0;i<n;i++){var t=e[i];t.length>6&&(t=t.substring(2)),o+="#"+t,i<n-1&&(o+=",")}o+=")",document.getElementById("canv").style.background=o}socket.onopen=function(){console.info("Live-Preview websocket is opened"),socket.send("{'lv':true}")},socket.onclose=function(){console.info("Live-Preview websocket is closing")},socket.onerror=function(e){console.error("Live-Preview websocket error:",e)},socket.onmessage=function(e){try{var o=JSON.parse(e.data);o&&o.leds&&requestAnimationFrame((function(){updatePreview(o.leds)}))}catch(e){console.error("Live-Preview websocket error:",e)}} | ||||
| </script></body></html>)====="; | ||||
|  | ||||
|  | ||||
| // Autogenerated from wled00/data/404.htm, do not edit!! | ||||
| const char PAGE_404[] PROGMEM = R"=====(<!DOCTYPE html><html><head><meta charset="utf-8"><meta  | ||||
| content="width=device-width" name="viewport"><meta name="theme-color"  | ||||
| content="#222222"><title>Not found</title><style> | ||||
| body{font-family:Verdana,Helvetica,sans-serif;text-align:center;background-color:#222;margin:0;color:#fff}img{width:400px;max-width:50%;image-rendering:pixelated;image-rendering:crisp-edges;margin:25px 0 -10px 0}button{outline:0;cursor:pointer;padding:8px;margin:10px;width:230px;text-transform:uppercase;font-family:helvetica;font-size:19px;background-color:#333;color:#fff;border:0 solid #fff;border-radius:25px} | ||||
| </style></head><body><img alt=""  | ||||
| src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACAAAAAgCAYAAABzenr0AAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsEAAA7BAbiRa+0AAAFMSURBVFhH7ZTfbYNADMaPKDNUrJA85CEjdIjOgNQV+sASlZgmI/AIK6AuQfngnDrmjtpHItQ/P+l0juHsz2cH9+fJ/G7nreldfnDnp+ln/ZIlxbIfQmIwJOekCrEJ8FUvASEWEXoBiuSERcTO75uhuwFWff86bi57n3ZC+rW3YLqB5rn11ldCEPNr2LwFJgHHy8G1bTsu3oKYX4N5BrQ8ZAYewSoBGDjr0ElWCUC/rT2X7MqynL7tG4Dc45BwEYM9H5w7DqHMdfNCURR9nue3Iobk55MtOYeLoOQ8vmoG6o+0FaLrOm9FwC3wayLgx5I2WHpGIGYorulfgPYQ3AZLz4hQ9TMBVVVleJGrRUWz2YgQOg8bPjzzrit7vwcRQb5NTiARRPPzMYItoCpoWZITMkao+mRkddpqQ6z6FN+DfwFJrOm55GfewC/CuU/E4tQYg7BPYQAAAABJRU5ErkJggg=="> | ||||
| <h1>404 Not Found</h1><b>Akemi does not know where you are headed...</b><br><br> | ||||
| <button onclick='window.location.href="/sliders"'>Back to controls</button> | ||||
| </body></html>)====="; | ||||
|  | ||||
|  | ||||
| // Autogenerated from wled00/data/favicon.ico, do not edit!! | ||||
| const uint16_t favicon_length = 954; | ||||
| const uint8_t favicon[] PROGMEM = { | ||||
|   0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x10, 0x10, 0x00, 0x00, 0x01, 0x00, | ||||
|   0x18, 0x00, 0x86, 0x00, 0x00, 0x00, 0x16, 0x00, 0x00, 0x00, 0x89, 0x50, | ||||
|   0x4E, 0x47, 0x0D, 0x0A, 0x1A, 0x0A, 0x00, 0x00, 0x00, 0x0D, 0x49, 0x48, | ||||
|   0x44, 0x52, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x10, 0x08, 0x06, | ||||
|   0x00, 0x00, 0x00, 0x1F, 0xF3, 0xFF, 0x61, 0x00, 0x00, 0x00, 0x4D, 0x49, | ||||
|   0x44, 0x41, 0x54, 0x38, 0x8D, 0x63, 0xFC, 0xFF, 0xFF, 0x3F, 0x03, 0xB1, | ||||
|   0x80, 0xD1, 0x9E, 0x01, 0x43, 0x31, 0x13, 0xD1, 0xBA, 0x71, 0x00, 0x8A, | ||||
|   0x0D, 0x60, 0x21, 0xA4, 0x00, 0xD9, 0xD9, 0xFF, 0x0F, 0x32, 0x30, 0x52, | ||||
|   0xDD, 0x05, 0xB4, 0xF1, 0x02, 0xB6, 0xD0, 0xA6, 0x99, 0x0B, 0x68, 0x1F, | ||||
|   0x0B, 0xD8, 0x42, 0x9E, 0xAA, 0x2E, 0xA0, 0xD8, 0x00, 0x46, 0x06, 0x3B, | ||||
|   0xCC, 0xCC, 0x40, 0xC8, 0xD9, 0x54, 0x75, 0x01, 0xE5, 0x5E, 0x20, 0x25, | ||||
|   0x3B, 0x63, 0x03, 0x00, 0x3E, 0xB7, 0x11, 0x5A, 0x8D, 0x1C, 0x07, 0xB4, | ||||
|   0x00, 0x00, 0x00, 0x00, 0x49, 0x45, 0x4E, 0x44, 0xAE, 0x42, 0x60, 0x82 | ||||
|   0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x10, 0x10, 0x00, 0x00, 0x01, 0x00, 0x18, 0x00, 0x86, 0x00, | ||||
|   0x00, 0x00, 0x16, 0x00, 0x00, 0x00, 0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a, 0x00, 0x00, | ||||
|   0x00, 0x0d, 0x49, 0x48, 0x44, 0x52, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x10, 0x08, 0x06, | ||||
|   0x00, 0x00, 0x00, 0x1f, 0xf3, 0xff, 0x61, 0x00, 0x00, 0x00, 0x4d, 0x49, 0x44, 0x41, 0x54, 0x38, | ||||
|   0x8d, 0x63, 0xfc, 0xff, 0xff, 0x3f, 0x03, 0xb1, 0x80, 0xd1, 0x9e, 0x01, 0x43, 0x31, 0x13, 0xd1, | ||||
|   0xba, 0x71, 0x00, 0x8a, 0x0d, 0x60, 0x21, 0xa4, 0x00, 0xd9, 0xd9, 0xff, 0x0f, 0x32, 0x30, 0x52, | ||||
|   0xdd, 0x05, 0xb4, 0xf1, 0x02, 0xb6, 0xd0, 0xa6, 0x99, 0x0b, 0x68, 0x1f, 0x0b, 0xd8, 0x42, 0x9e, | ||||
|   0xaa, 0x2e, 0xa0, 0xd8, 0x00, 0x46, 0x06, 0x3b, 0xcc, 0xcc, 0x40, 0xc8, 0xd9, 0x54, 0x75, 0x01, | ||||
|   0xe5, 0x5e, 0x20, 0x25, 0x3b, 0x63, 0x03, 0x00, 0x3e, 0xb7, 0x11, 0x5a, 0x8d, 0x1c, 0x07, 0xb4, | ||||
|   0x00, 0x00, 0x00, 0x00, 0x49, 0x45, 0x4e, 0x44, 0xae, 0x42, 0x60, 0x82 | ||||
| }; | ||||
|  | ||||
|   | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										3588
									
								
								wled00/html_ui.h
									
									
									
									
									
								
							
							
						
						
									
										3588
									
								
								wled00/html_ui.h
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,15 +1,19 @@ | ||||
| #include "wled.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * Sync to Philips hue lights | ||||
|  */ | ||||
| 
 | ||||
| #ifndef WLED_DISABLE_HUESYNC | ||||
| 
 | ||||
| void handleHue() | ||||
| { | ||||
|   if (hueReceived) | ||||
|   { | ||||
|     colorUpdated(7); hueReceived = false; | ||||
|     colorUpdated(NOTIFIER_CALL_MODE_HUE); hueReceived = false; | ||||
|     if (hueStoreAllowed && hueNewKey) | ||||
|     { | ||||
|       saveSettingsToEEPROM(); //save api key
 | ||||
|       serializeConfigSec(); //save api key
 | ||||
|       hueStoreAllowed = false; | ||||
|       hueNewKey = false; | ||||
|     } | ||||
| @@ -23,14 +27,14 @@ void handleHue() | ||||
|     reconnectHue(); | ||||
|   } else { | ||||
|     hueClient->close(); | ||||
|     if (hueError[0] == 'A') strcpy(hueError,"Inactive"); | ||||
|     if (hueError == HUE_ERROR_ACTIVE) hueError = HUE_ERROR_INACTIVE; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| void reconnectHue() | ||||
| { | ||||
|   if (!WLED_CONNECTED || !huePollingEnabled) return; | ||||
|   DEBUG_PRINTLN("Hue reconnect"); | ||||
|   DEBUG_PRINTLN(F("Hue reconnect")); | ||||
|   if (hueClient == nullptr) { | ||||
|     hueClient = new AsyncClient(); | ||||
|     hueClient->onConnect(&onHueConnect, hueClient); | ||||
| @@ -43,13 +47,13 @@ void reconnectHue() | ||||
| 
 | ||||
| void onHueError(void* arg, AsyncClient* client, int8_t error) | ||||
| { | ||||
|   DEBUG_PRINTLN("Hue err"); | ||||
|   strcpy(hueError,"Request timeout"); | ||||
|   DEBUG_PRINTLN(F("Hue err")); | ||||
|   hueError = HUE_ERROR_TIMEOUT; | ||||
| } | ||||
| 
 | ||||
| void onHueConnect(void* arg, AsyncClient* client) | ||||
| { | ||||
|   DEBUG_PRINTLN("Hue connect"); | ||||
|   DEBUG_PRINTLN(F("Hue connect")); | ||||
|   sendHuePoll(); | ||||
| } | ||||
| 
 | ||||
| @@ -59,15 +63,16 @@ void sendHuePoll() | ||||
|   String req = ""; | ||||
|   if (hueAuthRequired) | ||||
|   { | ||||
|     req += "POST /api HTTP/1.1\r\nHost: "; | ||||
|     req += F("POST /api HTTP/1.1\r\nHost: "); | ||||
|     req += hueIP.toString(); | ||||
|     req += "\r\nContent-Length: 25\r\n\r\n{\"devicetype\":\"wled#esp\"}"; | ||||
|     req += F("\r\nContent-Length: 25\r\n\r\n{\"devicetype\":\"wled#esp\"}"); | ||||
|   } else | ||||
|   { | ||||
|     req += "GET /api/"; | ||||
|     req += F("GET /api/"); | ||||
|     req += hueApiKey; | ||||
|     req += "/lights/" + String(huePollLightId); | ||||
|     req += " HTTP/1.1\r\nHost: "; | ||||
|     req += F("/lights/"); | ||||
|     req += String(huePollLightId); | ||||
|     req += F(" HTTP/1.1\r\nHost: "); | ||||
|     req += hueIP.toString(); | ||||
|     req += "\r\n\r\n"; | ||||
|   } | ||||
| @@ -93,28 +98,25 @@ void onHueData(void* arg, AsyncClient* client, void *data, size_t len) | ||||
|     auto error = deserializeJson(root, str); | ||||
|     if (error) | ||||
|     { | ||||
|       strcpy(hueError,"JSON parsing error"); return; | ||||
|       hueError = HUE_ERROR_JSON_PARSING; return; | ||||
|     } | ||||
|      | ||||
|     int hueErrorCode = root[0]["error"]["type"]; | ||||
|     int hueErrorCode = root[0][F("error")][F("type")]; | ||||
|     if (hueErrorCode)//hue bridge returned error
 | ||||
|     { | ||||
|       hueError = hueErrorCode; | ||||
|       switch (hueErrorCode) | ||||
|       { | ||||
|         case 1: strcpy(hueError,"Unauthorized"); hueAuthRequired = true; break; | ||||
|         case 3: strcpy(hueError,"Invalid light ID"); huePollingEnabled = false; break; | ||||
|         case 101: strcpy(hueError,"Link button not pressed"); hueAuthRequired = true; break; | ||||
|         default: | ||||
|           char coerr[18]; | ||||
|           sprintf(coerr,"Bridge Error %i",hueErrorCode); | ||||
|           strcpy(hueError,coerr); | ||||
|         case 1:   hueAuthRequired = true;    break; //Unauthorized user
 | ||||
|         case 3:   huePollingEnabled = false; break; //Invalid light ID
 | ||||
|         case 101: hueAuthRequired = true;    break; //link button not presset
 | ||||
|       } | ||||
|       return; | ||||
|     } | ||||
|      | ||||
|     if (hueAuthRequired) | ||||
|     { | ||||
|       const char* apikey = root[0]["success"]["username"]; | ||||
|       const char* apikey = root[0][F("success")][F("username")]; | ||||
|       if (apikey != nullptr && strlen(apikey) < sizeof(hueApiKey)) | ||||
|       { | ||||
|         strcpy(hueApiKey, apikey); | ||||
| @@ -133,7 +135,7 @@ void onHueData(void* arg, AsyncClient* client, void *data, size_t len) | ||||
|   auto error = deserializeJson(root, str); | ||||
|   if (error) | ||||
|   { | ||||
|     strcpy(hueError,"JSON parsing error"); return; | ||||
|     hueError = HUE_ERROR_JSON_PARSING; return; | ||||
|   } | ||||
| 
 | ||||
|   float hueX=0, hueY=0; | ||||
| @@ -145,10 +147,10 @@ void onHueData(void* arg, AsyncClient* client, void *data, size_t len) | ||||
|     { | ||||
|       hueBri = root["bri"]; | ||||
|       hueBri++; | ||||
|       const char* cm =root["colormode"]; | ||||
|       const char* cm =root[F("colormode")]; | ||||
|       if (cm != nullptr) //Color device
 | ||||
|       { | ||||
|         if (strstr(cm,"ct") != nullptr) //ct mode
 | ||||
|         if (strstr(cm,("ct")) != nullptr) //ct mode
 | ||||
|         { | ||||
|           hueCt = root["ct"]; | ||||
|           hueColormode = 3; | ||||
| @@ -159,8 +161,8 @@ void onHueData(void* arg, AsyncClient* client, void *data, size_t len) | ||||
|           hueColormode = 1; | ||||
|         } else //hs mode
 | ||||
|         { | ||||
|           hueHue = root["hue"]; | ||||
|           hueSat = root["sat"]; | ||||
|           hueHue = root[F("hue")]; | ||||
|           hueSat = root[F("sat")]; | ||||
|           hueColormode = 2; | ||||
|         } | ||||
|       } | ||||
| @@ -173,7 +175,7 @@ void onHueData(void* arg, AsyncClient* client, void *data, size_t len) | ||||
|     hueBri = 0; | ||||
|   } | ||||
| 
 | ||||
|   strcpy(hueError,"Active"); | ||||
|   hueError = HUE_ERROR_ACTIVE; | ||||
|    | ||||
|   //apply vals
 | ||||
|   if (hueBri != hueBriLast) | ||||
| @@ -1,3 +1,5 @@ | ||||
| #include "wled.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * Infrared sensor support for generic 24/40/44 key RGB remotes | ||||
|  */ | ||||
| @@ -13,10 +15,60 @@ decode_results results; | ||||
| 
 | ||||
| unsigned long irCheckedTime = 0; | ||||
| uint32_t lastValidCode = 0; | ||||
| byte lastRepeatableAction = ACTION_NONE; | ||||
| uint8_t lastRepeatableValue = 0; | ||||
| uint16_t irTimesRepeated = 0; | ||||
| uint8_t lastIR6ColourIdx = 0; | ||||
| 
 | ||||
| 
 | ||||
| // brightnessSteps: a static array of brightness levels following a geometric
 | ||||
| // progression.  Can be generated from the following Python, adjusting the
 | ||||
| // arbitrary 4.5 value to taste:
 | ||||
| //
 | ||||
| // def values(level):
 | ||||
| //     while level >= 5:
 | ||||
| //         yield int(level)
 | ||||
| //         level -= level / 4.5
 | ||||
| // result = [v for v in reversed(list(values(255)))]
 | ||||
| // print("%d values: %s" % (len(result), result))
 | ||||
| //
 | ||||
| // It would be hard to maintain repeatable steps if calculating this on the fly.
 | ||||
| const byte brightnessSteps[] = { | ||||
|   5, 7, 9, 12, 16, 20, 26, 34, 43, 56, 72, 93, 119, 154, 198, 255 | ||||
| }; | ||||
| const size_t numBrightnessSteps = sizeof(brightnessSteps) / sizeof(uint8_t); | ||||
| 
 | ||||
| // increment `bri` to the next `brightnessSteps` value
 | ||||
| void incBrightness() | ||||
| { | ||||
|   // dumb incremental search is efficient enough for so few items
 | ||||
|   for (int index = 0; index < numBrightnessSteps; ++index) | ||||
|   { | ||||
|     if (brightnessSteps[index] > bri) | ||||
|     { | ||||
|       bri = brightnessSteps[index]; | ||||
|       lastRepeatableAction = ACTION_BRIGHT_UP; | ||||
|       break; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // decrement `bri` to the next `brightnessSteps` value
 | ||||
| void decBrightness() | ||||
| { | ||||
|   // dumb incremental search is efficient enough for so few items
 | ||||
|   for (int index = numBrightnessSteps - 1; index >= 0; --index) | ||||
|   { | ||||
|     if (brightnessSteps[index] < bri) | ||||
|     { | ||||
|       bri = brightnessSteps[index]; | ||||
|       lastRepeatableAction = ACTION_BRIGHT_DOWN; | ||||
|       break; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| //Add what your custom IR codes should trigger here. Guide: https://github.com/Aircoookie/WLED/wiki/Infrared-Control
 | ||||
| //IR codes themselves can be defined directly after "case" or in "ir_codes.h"
 | ||||
| bool decodeIRCustom(uint32_t code) | ||||
| @@ -25,56 +77,83 @@ bool decodeIRCustom(uint32_t code) | ||||
|   { | ||||
|     //just examples, feel free to modify or remove
 | ||||
|     case IRCUSTOM_ONOFF : toggleOnOff(); break; | ||||
|     case IRCUSTOM_MACRO1 : applyMacro(1); break; | ||||
|     case IRCUSTOM_MACRO1 : applyPreset(1); break; | ||||
| 
 | ||||
|     default: return false; | ||||
|   } | ||||
|   if (code != IRCUSTOM_MACRO1) colorUpdated(2); //don't update color again if we apply macro, it already does it
 | ||||
|   if (code != IRCUSTOM_MACRO1) colorUpdated(NOTIFIER_CALL_MODE_BUTTON); //don't update color again if we apply macro, it already does it
 | ||||
|   return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| //relatively change brightness, minumum A=5
 | ||||
| void relativeChange(byte* property, int8_t amount, byte lowerBoundary =0) | ||||
| 
 | ||||
| void relativeChange(byte* property, int8_t amount, byte lowerBoundary, byte higherBoundary) | ||||
| { | ||||
|   int16_t new_val = (int16_t) *property + amount; | ||||
|   if (new_val > 0xFF) new_val = 0xFF; | ||||
|   if (new_val > higherBoundary) new_val = higherBoundary; | ||||
|   else if (new_val < lowerBoundary) new_val = lowerBoundary; | ||||
|   *property = new_val; | ||||
|   *property = (byte)constrain(new_val,0.1,255.1); | ||||
| } | ||||
| 
 | ||||
| void changeEffectSpeed(int8_t amount) | ||||
| { | ||||
|   if (effectCurrent != 0) { | ||||
|     int16_t new_val = (int16_t) effectSpeed + amount; | ||||
|     effectSpeed = (byte)constrain(new_val,0.1,255.1); | ||||
|   } else {                              // if Effect == "solid Color", change the hue of the primary color
 | ||||
|     CRGB fastled_col; | ||||
|     fastled_col.red =   col[0]; | ||||
|     fastled_col.green = col[1]; | ||||
|     fastled_col.blue =  col[2]; | ||||
|     CHSV prim_hsv = rgb2hsv_approximate(fastled_col); | ||||
|     int16_t new_val = (int16_t) prim_hsv.h + amount; | ||||
|     if (new_val > 255) new_val -= 255;  // roll-over if  bigger than 255
 | ||||
|     if (new_val < 0) new_val += 255;    // roll-over if smaller than 0
 | ||||
|     prim_hsv.h = (byte)new_val; | ||||
|     hsv2rgb_rainbow(prim_hsv, fastled_col); | ||||
|     col[0] = fastled_col.red;  | ||||
|     col[1] = fastled_col.green;  | ||||
|     col[2] = fastled_col.blue; | ||||
|   } | ||||
| 
 | ||||
|   if(amount > 0) lastRepeatableAction = ACTION_SPEED_UP; | ||||
|   if(amount < 0) lastRepeatableAction = ACTION_SPEED_DOWN; | ||||
|   lastRepeatableValue = amount; | ||||
| } | ||||
| 
 | ||||
| void changeEffectIntensity(int8_t amount) | ||||
| { | ||||
|   if (effectCurrent != 0) { | ||||
|     int16_t new_val = (int16_t) effectIntensity + amount; | ||||
|     effectIntensity = (byte)constrain(new_val,0.1,255.1); | ||||
|   } else {                                            // if Effect == "solid Color", change the saturation of the primary color
 | ||||
|     CRGB fastled_col; | ||||
|     fastled_col.red =   col[0]; | ||||
|     fastled_col.green = col[1]; | ||||
|     fastled_col.blue =  col[2]; | ||||
|     CHSV prim_hsv = rgb2hsv_approximate(fastled_col); | ||||
|     int16_t new_val = (int16_t) prim_hsv.s + amount; | ||||
|     prim_hsv.s = (byte)constrain(new_val,0.1,255.1);  // constrain to 0-255
 | ||||
|     hsv2rgb_rainbow(prim_hsv, fastled_col); | ||||
|     col[0] = fastled_col.red;  | ||||
|     col[1] = fastled_col.green;  | ||||
|     col[2] = fastled_col.blue; | ||||
|   } | ||||
| 
 | ||||
|   if(amount > 0) lastRepeatableAction = ACTION_INTENSITY_UP; | ||||
|   if(amount < 0) lastRepeatableAction = ACTION_INTENSITY_DOWN; | ||||
|   lastRepeatableValue = amount; | ||||
| } | ||||
| 
 | ||||
| void decodeIR(uint32_t code) | ||||
| { | ||||
|   if (code == 0xFFFFFFFF) //repeated code, continue brightness up/down
 | ||||
|   { | ||||
|     irTimesRepeated++; | ||||
|     if (lastValidCode == IR24_BRIGHTER || lastValidCode == IR40_BPLUS ) | ||||
|     {  | ||||
|       relativeChange(&bri, 10); colorUpdated(2); | ||||
|     } | ||||
|     else if (lastValidCode == IR24_DARKER || lastValidCode == IR40_BMINUS ) | ||||
|     { | ||||
|       relativeChange(&bri, -10, 5); colorUpdated(2); | ||||
|     } | ||||
|     if (lastValidCode == IR40_WPLUS) | ||||
|     {  | ||||
|       relativeChangeWhite(10); colorUpdated(2); | ||||
|     } | ||||
|     else if (lastValidCode == IR40_WMINUS) | ||||
|     { | ||||
|       relativeChangeWhite(-10, 5); colorUpdated(2); | ||||
|     } | ||||
|     else if ((lastValidCode == IR24_ON || lastValidCode == IR40_ON) && irTimesRepeated > 7 ) | ||||
|     { | ||||
|       nightlightActive = true; | ||||
|       nightlightStartTime = millis(); | ||||
|       colorUpdated(2); | ||||
|     } | ||||
|     applyRepeatActions(); | ||||
|     return; | ||||
|   } | ||||
|   lastValidCode = 0; irTimesRepeated = 0; | ||||
| 
 | ||||
|   if (decodeIRCustom(code)) return; | ||||
|   if      (code > 0xFFFFFF) return; //invalid code
 | ||||
|   else if (code > 0xF70000 && code < 0xF80000) decodeIR24(code); //is in 24-key remote range
 | ||||
| @@ -85,21 +164,69 @@ void decodeIR(uint32_t code) | ||||
|       case 3: decodeIR40(code);    break;  // blue  40-key remote with 25%, 50%, 75% and 100% keys
 | ||||
|       case 4: decodeIR44(code);    break;  // white 44-key remote with color-up/down keys and DIY1 to 6 keys 
 | ||||
|       case 5: decodeIR21(code);    break;  // white 21-key remote  
 | ||||
|       case 6: decodeIR6(code);    break;   // black 6-key learning remote defaults: "CH" controls brightness,
 | ||||
|       case 6: decodeIR6(code);     break;  // black 6-key learning remote defaults: "CH" controls brightness,
 | ||||
|                                            // "VOL +" controls effect, "VOL -" controls colour/palette, "MUTE" 
 | ||||
|                                            // sets bright plain white
 | ||||
|       case 7: decodeIR9(code);    break; | ||||
|       default: return; | ||||
|     } | ||||
|   } | ||||
|   if (nightlightActive && bri == 0) nightlightActive = false; | ||||
|   colorUpdated(NOTIFIER_CALL_MODE_BUTTON); //for notifier, IR is considered a button input
 | ||||
|   //code <= 0xF70000 also invalid
 | ||||
| } | ||||
| 
 | ||||
| void applyRepeatActions(){ | ||||
|    | ||||
|     if (lastRepeatableAction == ACTION_BRIGHT_UP) | ||||
|     {  | ||||
|       incBrightness(); colorUpdated(NOTIFIER_CALL_MODE_BUTTON); | ||||
|     } | ||||
|     else if (lastRepeatableAction == ACTION_BRIGHT_DOWN ) | ||||
|     { | ||||
|       decBrightness(); colorUpdated(NOTIFIER_CALL_MODE_BUTTON); | ||||
|     } | ||||
| 
 | ||||
|     if (lastRepeatableAction == ACTION_SPEED_UP) | ||||
|     {  | ||||
|       changeEffectSpeed(lastRepeatableValue); colorUpdated(NOTIFIER_CALL_MODE_BUTTON); | ||||
|     } | ||||
|     else if (lastRepeatableAction == ACTION_SPEED_DOWN ) | ||||
|     { | ||||
|       changeEffectSpeed(lastRepeatableValue); colorUpdated(NOTIFIER_CALL_MODE_BUTTON); | ||||
|     } | ||||
| 
 | ||||
|     if (lastRepeatableAction == ACTION_INTENSITY_UP) | ||||
|     {  | ||||
|       changeEffectIntensity(lastRepeatableValue); colorUpdated(NOTIFIER_CALL_MODE_BUTTON); | ||||
|     } | ||||
|     else if (lastRepeatableAction == ACTION_INTENSITY_DOWN ) | ||||
|     { | ||||
|       changeEffectIntensity(lastRepeatableValue); colorUpdated(NOTIFIER_CALL_MODE_BUTTON); | ||||
|     } | ||||
| 
 | ||||
|     if (lastValidCode == IR40_WPLUS) | ||||
|     {  | ||||
|       relativeChangeWhite(10); colorUpdated(NOTIFIER_CALL_MODE_BUTTON); | ||||
|     } | ||||
|     else if (lastValidCode == IR40_WMINUS) | ||||
|     { | ||||
|       relativeChangeWhite(-10, 5); colorUpdated(NOTIFIER_CALL_MODE_BUTTON); | ||||
|     } | ||||
|     else if ((lastValidCode == IR24_ON || lastValidCode == IR40_ON) && irTimesRepeated > 7 ) | ||||
|     { | ||||
|       nightlightActive = true; | ||||
|       nightlightStartTime = millis(); | ||||
|       colorUpdated(NOTIFIER_CALL_MODE_BUTTON); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void decodeIR24(uint32_t code) | ||||
| { | ||||
|   switch (code) { | ||||
|     case IR24_BRIGHTER  : relativeChange(&bri, 10);         break; | ||||
|     case IR24_DARKER    : relativeChange(&bri, -10, 5);     break; | ||||
|     case IR24_BRIGHTER  : incBrightness();                  break; | ||||
|     case IR24_DARKER    : decBrightness();                  break; | ||||
|     case IR24_OFF       : briLast = bri; bri = 0;           break; | ||||
|     case IR24_ON        : bri = briLast;                    break; | ||||
|     case IR24_RED       : colorFromUint32(COLOR_RED);       break; | ||||
| @@ -125,14 +252,13 @@ void decodeIR24(uint32_t code) | ||||
|     default: return; | ||||
|   } | ||||
|   lastValidCode = code; | ||||
|   colorUpdated(2); //for notifier, IR is considered a button input
 | ||||
| } | ||||
| 
 | ||||
| void decodeIR24OLD(uint32_t code) | ||||
| { | ||||
|   switch (code) { | ||||
|     case IR24_OLD_BRIGHTER  : relativeChange(&bri, 10);            break; | ||||
|     case IR24_OLD_DARKER    : relativeChange(&bri, -10, 5);        break; | ||||
|     case IR24_OLD_BRIGHTER  : incBrightness();                     break; | ||||
|     case IR24_OLD_DARKER    : decBrightness();                     break; | ||||
|     case IR24_OLD_OFF       : briLast = bri; bri = 0;              break; | ||||
|     case IR24_OLD_ON        : bri = briLast;                       break; | ||||
|     case IR24_OLD_RED       : colorFromUint32(COLOR_RED);          break; | ||||
| @@ -158,15 +284,14 @@ void decodeIR24OLD(uint32_t code) | ||||
|     default: return; | ||||
|   } | ||||
|   lastValidCode = code; | ||||
|   colorUpdated(2); //for notifier, IR is considered a button input
 | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void decodeIR24CT(uint32_t code) | ||||
| { | ||||
|   switch (code) { | ||||
|     case IR24_CT_BRIGHTER   : relativeChange(&bri, 10);            break; | ||||
|     case IR24_CT_DARKER     : relativeChange(&bri, -10, 5);        break; | ||||
|     case IR24_CT_BRIGHTER   : incBrightness();                     break; | ||||
|     case IR24_CT_DARKER     : decBrightness();                     break; | ||||
|     case IR24_CT_OFF        : briLast = bri; bri = 0;              break; | ||||
|     case IR24_CT_ON         : bri = briLast;                       break; | ||||
|     case IR24_CT_RED        : colorFromUint32(COLOR_RED);          break; | ||||
| @@ -194,15 +319,14 @@ void decodeIR24CT(uint32_t code) | ||||
|     default: return;  | ||||
|   } | ||||
|   lastValidCode = code; | ||||
|   colorUpdated(2); //for notifier, IR is considered a button input
 | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void decodeIR40(uint32_t code) | ||||
| { | ||||
|   switch (code) { | ||||
|     case IR40_BPLUS        : relativeChange(&bri, 10);                                   break; | ||||
|     case IR40_BMINUS       : relativeChange(&bri, -10, 5);                               break; | ||||
|     case IR40_BPLUS        : incBrightness();                                            break; | ||||
|     case IR40_BMINUS       : decBrightness();                                            break; | ||||
|     case IR40_OFF          : briLast = bri; bri = 0;                                     break; | ||||
|     case IR40_ON           : bri = briLast;                                              break; | ||||
|     case IR40_RED          : colorFromUint24(COLOR_RED);                                 break; | ||||
| @@ -221,19 +345,19 @@ void decodeIR40(uint32_t code) | ||||
|     case IR40_MAGENTA      : colorFromUint24(COLOR_MAGENTA);                             break; | ||||
|     case IR40_PINK         : colorFromUint24(COLOR_PINK);                                break; | ||||
|     case IR40_WARMWHITE2   : { | ||||
|       if (useRGBW) {        colorFromUint32(COLOR2_WARMWHITE2);   effectCurrent = 0; }     | ||||
|       if (strip.isRgbw) {        colorFromUint32(COLOR2_WARMWHITE2);   effectCurrent = 0; }     | ||||
|       else                  colorFromUint24(COLOR_WARMWHITE2);                       }   break; | ||||
|     case IR40_WARMWHITE    : { | ||||
|       if (useRGBW) {        colorFromUint32(COLOR2_WARMWHITE);    effectCurrent = 0; }     | ||||
|       if (strip.isRgbw) {        colorFromUint32(COLOR2_WARMWHITE);    effectCurrent = 0; }     | ||||
|       else                  colorFromUint24(COLOR_WARMWHITE);                        }   break; | ||||
|     case IR40_WHITE        : { | ||||
|       if (useRGBW) {        colorFromUint32(COLOR2_NEUTRALWHITE); effectCurrent = 0; }     | ||||
|       if (strip.isRgbw) {        colorFromUint32(COLOR2_NEUTRALWHITE); effectCurrent = 0; }     | ||||
|       else                  colorFromUint24(COLOR_NEUTRALWHITE);                     }   break; | ||||
|     case IR40_COLDWHITE    : { | ||||
|       if (useRGBW) {        colorFromUint32(COLOR2_COLDWHITE);    effectCurrent = 0; }    | ||||
|       if (strip.isRgbw) {        colorFromUint32(COLOR2_COLDWHITE);    effectCurrent = 0; }    | ||||
|       else                  colorFromUint24(COLOR_COLDWHITE);                        }   break; | ||||
|     case IR40_COLDWHITE2    : { | ||||
|       if (useRGBW) {        colorFromUint32(COLOR2_COLDWHITE2);   effectCurrent = 0; }    | ||||
|       if (strip.isRgbw) {        colorFromUint32(COLOR2_COLDWHITE2);   effectCurrent = 0; }    | ||||
|       else                  colorFromUint24(COLOR_COLDWHITE2);                       }   break; | ||||
|     case IR40_WPLUS        : relativeChangeWhite(10);                                    break; | ||||
|     case IR40_WMINUS       : relativeChangeWhite(-10, 5);                                break; | ||||
| @@ -243,24 +367,23 @@ void decodeIR40(uint32_t code) | ||||
|     case IR40_W50          : bri = 127;                                                  break; | ||||
|     case IR40_W75          : bri = 191;                                                  break; | ||||
|     case IR40_W100         : bri = 255;                                                  break; | ||||
|     case IR40_QUICK        : relativeChange(&effectSpeed, 10);                           break; | ||||
|     case IR40_SLOW         : relativeChange(&effectSpeed, -10, 5);                       break; | ||||
|     case IR40_JUMP7        : relativeChange(&effectIntensity, 10);                       break; | ||||
|     case IR40_AUTO         : relativeChange(&effectIntensity, -10, 5);                   break; | ||||
|     case IR40_QUICK        : changeEffectSpeed( 16);                                     break; | ||||
|     case IR40_SLOW         : changeEffectSpeed(-16);                                     break; | ||||
|     case IR40_JUMP7        : changeEffectIntensity( 16);                                 break; | ||||
|     case IR40_AUTO         : changeEffectIntensity(-16);                                 break; | ||||
|     case IR40_JUMP3        : if (!applyPreset(1)) { effectCurrent = FX_MODE_STATIC;        effectPalette = 0; } break; | ||||
|     case IR40_FADE3        : if (!applyPreset(2)) { effectCurrent = FX_MODE_BREATH;        effectPalette = 0; } break; | ||||
|     case IR40_FADE7        : if (!applyPreset(3)) { effectCurrent = FX_MODE_FIRE_FLICKER;  effectPalette = 0; } break; | ||||
|     case IR40_FLASH        : if (!applyPreset(4)) { effectCurrent = FX_MODE_RAINBOW;       effectPalette = 0; } break; | ||||
|   } | ||||
|   lastValidCode = code; | ||||
|   colorUpdated(2); //for notifier, IR is considered a button input 
 | ||||
| } | ||||
| 
 | ||||
| void decodeIR44(uint32_t code) | ||||
| { | ||||
|   switch (code) { | ||||
|     case IR44_BPLUS       : relativeChange(&bri, 10);                                   break; | ||||
|     case IR44_BMINUS      : relativeChange(&bri, -10, 5);                               break; | ||||
|     case IR44_BPLUS       : incBrightness();                                            break; | ||||
|     case IR44_BMINUS      : decBrightness();                                            break; | ||||
|     case IR44_OFF         : briLast = bri; bri = 0;                                     break; | ||||
|     case IR44_ON          : bri = briLast;                                              break; | ||||
|     case IR44_RED         : colorFromUint24(COLOR_RED);                                 break; | ||||
| @@ -279,30 +402,30 @@ void decodeIR44(uint32_t code) | ||||
|     case IR44_MAGENTA     : colorFromUint24(COLOR_MAGENTA);                             break; | ||||
|     case IR44_PINK        : colorFromUint24(COLOR_PINK);                                break; | ||||
|     case IR44_WHITE       : { | ||||
|       if (useRGBW) { | ||||
|       if (strip.isRgbw) { | ||||
|         if (col[3] > 0) col[3] = 0;  | ||||
|         else {              colorFromUint32(COLOR2_NEUTRALWHITE); effectCurrent = 0; } | ||||
|       } else                colorFromUint24(COLOR_NEUTRALWHITE);                     }  break; | ||||
|     case IR44_WARMWHITE2  : { | ||||
|       if (useRGBW) {        colorFromUint32(COLOR2_WARMWHITE2);   effectCurrent = 0; }     | ||||
|       if (strip.isRgbw) {        colorFromUint32(COLOR2_WARMWHITE2);   effectCurrent = 0; }     | ||||
|       else                  colorFromUint24(COLOR_WARMWHITE2);                       }  break; | ||||
|     case IR44_WARMWHITE   : { | ||||
|       if (useRGBW) {        colorFromUint32(COLOR2_WARMWHITE);    effectCurrent = 0; }     | ||||
|       if (strip.isRgbw) {        colorFromUint32(COLOR2_WARMWHITE);    effectCurrent = 0; }     | ||||
|       else                  colorFromUint24(COLOR_WARMWHITE);                        }  break; | ||||
|     case IR44_COLDWHITE   : { | ||||
|       if (useRGBW) {        colorFromUint32(COLOR2_COLDWHITE);    effectCurrent = 0; }    | ||||
|       if (strip.isRgbw) {        colorFromUint32(COLOR2_COLDWHITE);    effectCurrent = 0; }    | ||||
|       else                  colorFromUint24(COLOR_COLDWHITE);                        }  break; | ||||
|     case IR44_COLDWHITE2  : { | ||||
|       if (useRGBW) {        colorFromUint32(COLOR2_COLDWHITE2);   effectCurrent = 0; }     | ||||
|       if (strip.isRgbw) {        colorFromUint32(COLOR2_COLDWHITE2);   effectCurrent = 0; }     | ||||
|       else                  colorFromUint24(COLOR_COLDWHITE2);                       }  break; | ||||
|     case IR44_REDPLUS     : relativeChange(&effectCurrent, 1);                          break; | ||||
|     case IR44_REDPLUS     : relativeChange(&effectCurrent,  1, 0, MODE_COUNT);          break; | ||||
|     case IR44_REDMINUS    : relativeChange(&effectCurrent, -1, 0);                      break; | ||||
|     case IR44_GREENPLUS   : relativeChange(&effectPalette, 1);                          break; | ||||
|     case IR44_GREENPLUS   : relativeChange(&effectPalette,  1, 0, strip.getPaletteCount() -1);     break; | ||||
|     case IR44_GREENMINUS  : relativeChange(&effectPalette, -1, 0);                      break; | ||||
|     case IR44_BLUEPLUS    : relativeChange(&effectIntensity, 10);                       break; | ||||
|     case IR44_BLUEMINUS   : relativeChange(&effectIntensity, -10, 5);                   break; | ||||
|     case IR44_QUICK       : relativeChange(&effectSpeed, 10);                           break; | ||||
|     case IR44_SLOW        : relativeChange(&effectSpeed, -10, 5);                       break; | ||||
|     case IR44_BLUEPLUS    : changeEffectIntensity( 16);                                 break; | ||||
|     case IR44_BLUEMINUS   : changeEffectIntensity(-16);                                 break; | ||||
|     case IR44_QUICK       : changeEffectSpeed( 16);                                     break; | ||||
|     case IR44_SLOW        : changeEffectSpeed(-16);                                     break; | ||||
|     case IR44_DIY1        : if (!applyPreset(1)) { effectCurrent = FX_MODE_STATIC;        effectPalette = 0; } break; | ||||
|     case IR44_DIY2        : if (!applyPreset(2)) { effectCurrent = FX_MODE_BREATH;        effectPalette = 0; } break; | ||||
|     case IR44_DIY3        : if (!applyPreset(3)) { effectCurrent = FX_MODE_FIRE_FLICKER;  effectPalette = 0; } break; | ||||
| @@ -317,14 +440,13 @@ void decodeIR44(uint32_t code) | ||||
|     case IR44_FADE7       : bri = 255;                                                  break; | ||||
|   } | ||||
|   lastValidCode = code; | ||||
|   colorUpdated(2); //for notifier, IR is considered a button input 
 | ||||
| } | ||||
| 
 | ||||
| void decodeIR21(uint32_t code) | ||||
| { | ||||
|     switch (code) { | ||||
|     case IR21_BRIGHTER:  relativeChange(&bri, 10);         break; | ||||
|     case IR21_DARKER:    relativeChange(&bri, -10, 5);     break; | ||||
|     case IR21_BRIGHTER:  incBrightness();                  break; | ||||
|     case IR21_DARKER:    decBrightness();                  break; | ||||
|     case IR21_OFF:       briLast = bri; bri = 0;           break; | ||||
|     case IR21_ON:        bri = briLast;                    break; | ||||
|     case IR21_RED:       colorFromUint32(COLOR_RED);       break; | ||||
| @@ -347,57 +469,64 @@ void decodeIR21(uint32_t code) | ||||
|     default: return; | ||||
|     } | ||||
|     lastValidCode = code; | ||||
|     colorUpdated(2); //for notifier, IR is considered a button input
 | ||||
| } | ||||
| 
 | ||||
| void decodeIR6(uint32_t code) | ||||
| { | ||||
|    | ||||
|     switch (code) { | ||||
|       case IR6_POWER: toggleOnOff(); break; | ||||
|       case IR6_CHANNEL_UP: relativeChange(&bri, 10);         break; | ||||
|       case IR6_CHANNEL_DOWN: relativeChange(&bri, -10, 5);     break; | ||||
|       case IR6_VOLUME_UP: /* next effect */ relativeChange(&effectCurrent, 1); break; | ||||
|       case IR6_VOLUME_DOWN:  | ||||
|       /* next palette */  | ||||
|        | ||||
|         relativeChange(&effectPalette, 1);  | ||||
| 
 | ||||
|         switch(lastIR6ColourIdx) | ||||
|         { | ||||
|           case 0: colorFromUint32(COLOR_RED); break; | ||||
|           case 1: colorFromUint32(COLOR_REDDISH); break; | ||||
|           case 2:colorFromUint32(COLOR_ORANGE);    break; | ||||
|           case 3:colorFromUint32(COLOR_YELLOWISH); break; | ||||
|           case 4:colorFromUint32(COLOR_GREEN);     break; | ||||
|           case 5:colorFromUint32(COLOR_GREENISH);  break; | ||||
|           case 6:colorFromUint32(COLOR_TURQUOISE); break; | ||||
|           case 7: colorFromUint32(COLOR_CYAN);      break; | ||||
|           case 8:colorFromUint32(COLOR_BLUE);      break; | ||||
|           case 9:colorFromUint32(COLOR_DEEPBLUE);  break; | ||||
|           case 10:colorFromUint32(COLOR_PURPLE);    break; | ||||
|           case 11:colorFromUint32(COLOR_PINK);      break; | ||||
|           case 12:colorFromUint32(COLOR_WHITE);      break; | ||||
|             default:break; | ||||
| 
 | ||||
|         } | ||||
|          | ||||
|         lastIR6ColourIdx++; | ||||
|         if(lastIR6ColourIdx > 12) lastIR6ColourIdx = 0; | ||||
|        | ||||
|       break; | ||||
|       case IR6_MUTE: effectCurrent = 0;  effectPalette = 0; colorFromUint32(COLOR_WHITE); bri=255; break; | ||||
|     } | ||||
|     lastValidCode = code; | ||||
|     colorUpdated(2); //for notifier, IR is considered a button input
 | ||||
|   switch (code) { | ||||
|     case IR6_POWER: toggleOnOff();                                          break; | ||||
|     case IR6_CHANNEL_UP: incBrightness();                                   break; | ||||
|     case IR6_CHANNEL_DOWN: decBrightness();                                 break; | ||||
|     case IR6_VOLUME_UP:   relativeChange(&effectCurrent, 1, 0, MODE_COUNT); break;  // next effect
 | ||||
|     case IR6_VOLUME_DOWN:                                                           // next palette
 | ||||
|       relativeChange(&effectPalette, 1, 0, strip.getPaletteCount() -1);  | ||||
|       switch(lastIR6ColourIdx) { | ||||
|         case 0: colorFromUint32(COLOR_RED);       break; | ||||
|         case 1: colorFromUint32(COLOR_REDDISH);   break; | ||||
|         case 2: colorFromUint32(COLOR_ORANGE);    break; | ||||
|         case 3: colorFromUint32(COLOR_YELLOWISH); break; | ||||
|         case 4: colorFromUint32(COLOR_GREEN);     break; | ||||
|         case 5: colorFromUint32(COLOR_GREENISH);  break; | ||||
|         case 6: colorFromUint32(COLOR_TURQUOISE); break; | ||||
|         case 7: colorFromUint32(COLOR_CYAN);      break; | ||||
|         case 8: colorFromUint32(COLOR_BLUE);      break; | ||||
|         case 9: colorFromUint32(COLOR_DEEPBLUE);  break; | ||||
|         case 10:colorFromUint32(COLOR_PURPLE);    break; | ||||
|         case 11:colorFromUint32(COLOR_PINK);      break; | ||||
|         case 12:colorFromUint32(COLOR_WHITE);     break; | ||||
|         default:                                  break; | ||||
|       } | ||||
|       lastIR6ColourIdx++; | ||||
|       if(lastIR6ColourIdx > 12) lastIR6ColourIdx = 0;                      break; | ||||
|     case IR6_MUTE: effectCurrent = 0; effectPalette = 0; colorFromUint32(COLOR_WHITE); bri=255; break; | ||||
|   } | ||||
|   lastValidCode = code; | ||||
| } | ||||
| 
 | ||||
| void decodeIR9(uint32_t code) | ||||
| { | ||||
|   switch (code) { | ||||
|     case IR9_POWER      : toggleOnOff();  break; | ||||
|     case IR9_A          : if (!applyPreset(1)) effectCurrent = FX_MODE_COLORTWINKLE;  break; | ||||
|     case IR9_B          : if (!applyPreset(2)) effectCurrent = FX_MODE_RAINBOW_CYCLE; break; | ||||
|     case IR9_C          : if (!applyPreset(3)) effectCurrent = FX_MODE_BREATH;        break; | ||||
|     case IR9_UP         : incBrightness();                                            break; | ||||
|     case IR9_DOWN       : decBrightness();                                            break; | ||||
|     //case IR9_UP         : changeEffectIntensity(16);         break;
 | ||||
|     //case IR9_DOWN       : changeEffectIntensity(-16);     break;
 | ||||
|     case IR9_LEFT       : changeEffectSpeed(-16);                                     break; | ||||
|     case IR9_RIGHT      : changeEffectSpeed(16);                                      break; | ||||
|     case IR9_SELECT     : relativeChange(&effectCurrent, 1, 0, MODE_COUNT);           break; | ||||
|     default: return; | ||||
|   } | ||||
|   lastValidCode = code; | ||||
| } | ||||
| 
 | ||||
| void initIR() | ||||
| { | ||||
|   if (irEnabled > 0) | ||||
|   { | ||||
|     irrecv = new IRrecv(IR_PIN); | ||||
|     irrecv = new IRrecv(irPin); | ||||
|     irrecv->enableIRIn(); | ||||
|   } | ||||
| } | ||||
| @@ -13,6 +13,15 @@ | ||||
| #define IR6_VOLUME_DOWN  0xFF2FD0 | ||||
| #define IR6_MUTE         0xFFAF50 | ||||
|  | ||||
| #define IR9_POWER       0xFF629D | ||||
| #define IR9_A           0xFF22DD | ||||
| #define IR9_B           0xFF02FD | ||||
| #define IR9_C           0xFFC23D | ||||
| #define IR9_LEFT        0xFF30CF | ||||
| #define IR9_RIGHT       0xFF7A85 | ||||
| #define IR9_UP          0xFF9867 | ||||
| #define IR9_DOWN        0xFF38C7 | ||||
| #define IR9_SELECT      0xFF18E7 | ||||
|  | ||||
| //Infrared codes for 24-key remote from http://woodsgood.ca/projects/2015/02/13/rgb-led-strip-controllers-ir-codes/ | ||||
| #define IR24_BRIGHTER  0xF700FF | ||||
| @@ -229,3 +238,12 @@ | ||||
| #define COLOR2_NEUTRALWHITE  0xFF000000 | ||||
| #define COLOR2_COLDWHITE     0xFF7F7F7F | ||||
| #define COLOR2_COLDWHITE2    0xFFFFFFFF | ||||
|  | ||||
| #define ACTION_NONE             0 | ||||
| #define ACTION_BRIGHT_UP        1 | ||||
| #define ACTION_BRIGHT_DOWN      2 | ||||
| #define ACTION_SPEED_UP         3 | ||||
| #define ACTION_SPEED_DOWN       4 | ||||
| #define ACTION_INTENSITY_UP     5 | ||||
| #define ACTION_INTENSITY_DOWN   6 | ||||
| #define ACTION_POWER            7 | ||||
							
								
								
									
										800
									
								
								wled00/json.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										800
									
								
								wled00/json.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,800 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| #include "palettes.h" | ||||
|  | ||||
| /* | ||||
|  * JSON API (De)serialization | ||||
|  */ | ||||
|  | ||||
| void deserializeSegment(JsonObject elem, byte it) | ||||
| { | ||||
|   byte id = elem["id"] | it; | ||||
|   if (id < strip.getMaxSegments()) | ||||
|   { | ||||
|     WS2812FX::Segment& seg = strip.getSegment(id); | ||||
|     uint16_t start = elem[F("start")] | seg.start; | ||||
|     int stop = elem["stop"] | -1; | ||||
|  | ||||
|     if (stop < 0) { | ||||
|       uint16_t len = elem[F("len")]; | ||||
|       stop = (len > 0) ? start + len : seg.stop; | ||||
|     } | ||||
|     uint16_t grp = elem[F("grp")] | seg.grouping; | ||||
|     uint16_t spc = elem[F("spc")] | seg.spacing; | ||||
|     strip.setSegment(id, start, stop, grp, spc); | ||||
|  | ||||
|     int segbri = elem["bri"] | -1; | ||||
|     if (segbri == 0) { | ||||
|       seg.setOption(SEG_OPTION_ON, 0, id); | ||||
|     } else if (segbri > 0) { | ||||
|       seg.setOpacity(segbri, id); | ||||
|       seg.setOption(SEG_OPTION_ON, 1, id); | ||||
|     } | ||||
|    | ||||
|     seg.setOption(SEG_OPTION_ON, elem["on"] | seg.getOption(SEG_OPTION_ON), id); | ||||
|      | ||||
|     JsonArray colarr = elem["col"]; | ||||
|     if (!colarr.isNull()) | ||||
|     { | ||||
|       for (uint8_t i = 0; i < 3; i++) | ||||
|       { | ||||
|         int rgbw[] = {0,0,0,0}; | ||||
|         bool colValid = false; | ||||
|         JsonArray colX = colarr[i]; | ||||
|         if (colX.isNull()) { | ||||
|           byte brgbw[] = {0,0,0,0}; | ||||
|           const char* hexCol = colarr[i]; | ||||
|           if (hexCol == nullptr) { //Kelvin color temperature (or invalid), e.g 2400 | ||||
|             int kelvin = colarr[i] | -1; | ||||
|             if (kelvin <  0) continue; | ||||
|             if (kelvin == 0) seg.setColor(i, 0, id); | ||||
|             if (kelvin >  0) colorKtoRGB(kelvin, brgbw); | ||||
|             colValid = true; | ||||
|           } else { //HEX string, e.g. "FFAA00" | ||||
|             colValid = colorFromHexString(brgbw, hexCol); | ||||
|           } | ||||
|           for (uint8_t c = 0; c < 4; c++) rgbw[c] = brgbw[c]; | ||||
|         } else { //Array of ints (RGB or RGBW color), e.g. [255,160,0] | ||||
|           byte sz = colX.size(); | ||||
|           if (sz == 0) continue; //do nothing on empty array | ||||
|  | ||||
|           byte cp = copyArray(colX, rgbw, 4);       | ||||
|           if (cp == 1 && rgbw[0] == 0)  | ||||
|             seg.setColor(i, 0, id); | ||||
|           colValid = true; | ||||
|         } | ||||
|  | ||||
|         if (!colValid) continue; | ||||
|         if (id == strip.getMainSegmentId() && i < 2) //temporary, to make transition work on main segment | ||||
|         { | ||||
|           if (i == 0) {col[0] = rgbw[0]; col[1] = rgbw[1]; col[2] = rgbw[2]; col[3] = rgbw[3];} | ||||
|           if (i == 1) {colSec[0] = rgbw[0]; colSec[1] = rgbw[1]; colSec[2] = rgbw[2]; colSec[3] = rgbw[3];} | ||||
|         } else { //normal case, apply directly to segment | ||||
|           seg.setColor(i, ((rgbw[3] << 24) | ((rgbw[0]&0xFF) << 16) | ((rgbw[1]&0xFF) << 8) | ((rgbw[2]&0xFF))), id); | ||||
|           if (seg.mode == FX_MODE_STATIC) strip.trigger(); //instant refresh | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     // lx parser | ||||
|     #ifdef WLED_ENABLE_LOXONE | ||||
|     int lx = elem[F("lx")] | -1; | ||||
|     if (lx > 0) { | ||||
|       parseLxJson(lx, id, false); | ||||
|     } | ||||
|     int ly = elem[F("ly")] | -1; | ||||
|     if (ly > 0) { | ||||
|       parseLxJson(ly, id, true); | ||||
|     } | ||||
|     #endif | ||||
|  | ||||
|     //if (pal != seg.palette && pal < strip.getPaletteCount()) strip.setPalette(pal); | ||||
|     seg.setOption(SEG_OPTION_SELECTED, elem[F("sel")] | seg.getOption(SEG_OPTION_SELECTED)); | ||||
|     seg.setOption(SEG_OPTION_REVERSED, elem["rev"] | seg.getOption(SEG_OPTION_REVERSED)); | ||||
|     seg.setOption(SEG_OPTION_MIRROR  , elem[F("mi")]  | seg.getOption(SEG_OPTION_MIRROR  )); | ||||
|  | ||||
|     //temporary, strip object gets updated via colorUpdated() | ||||
|     if (id == strip.getMainSegmentId()) { | ||||
|       effectCurrent = elem[F("fx")] | effectCurrent; | ||||
|       effectSpeed = elem[F("sx")] | effectSpeed; | ||||
|       effectIntensity = elem[F("ix")] | effectIntensity; | ||||
|       effectPalette = elem[F("pal")] | effectPalette; | ||||
|     } else { //permanent | ||||
|       byte fx = elem[F("fx")] | seg.mode; | ||||
|       if (fx != seg.mode && fx < strip.getModeCount()) strip.setMode(id, fx); | ||||
|       seg.speed = elem[F("sx")] | seg.speed; | ||||
|       seg.intensity = elem[F("ix")] | seg.intensity; | ||||
|       seg.palette = elem[F("pal")] | seg.palette; | ||||
|     } | ||||
|  | ||||
|     JsonArray iarr = elem[F("i")]; //set individual LEDs | ||||
|     if (!iarr.isNull()) { | ||||
|       strip.setPixelSegment(id); | ||||
|  | ||||
|       //freeze and init to black | ||||
|       if (!seg.getOption(SEG_OPTION_FREEZE)) { | ||||
|         seg.setOption(SEG_OPTION_FREEZE, true); | ||||
|         strip.fill(0); | ||||
|       } | ||||
|  | ||||
|       uint16_t start = 0, stop = 0; | ||||
|       byte set = 0; //0 nothing set, 1 start set, 2 range set | ||||
|  | ||||
|       for (uint16_t i = 0; i < iarr.size(); i++) { | ||||
|         if(iarr[i].is<JsonInteger>()) { | ||||
|           if (!set) { | ||||
|             start = iarr[i]; | ||||
|             set = 1; | ||||
|           } else { | ||||
|             stop = iarr[i]; | ||||
|             set = 2; | ||||
|           } | ||||
|         } else { | ||||
|           JsonArray icol = iarr[i]; | ||||
|           if (icol.isNull()) break; | ||||
|  | ||||
|           byte sz = icol.size(); | ||||
|           if (sz == 0 && sz > 4) break; | ||||
|  | ||||
|           int rgbw[] = {0,0,0,0}; | ||||
|           copyArray(icol, rgbw); | ||||
|  | ||||
|           if (set < 2) stop = start + 1; | ||||
|           for (uint16_t i = start; i < stop; i++) { | ||||
|             strip.setPixelColor(i, rgbw[0], rgbw[1], rgbw[2], rgbw[3]); | ||||
|           } | ||||
|           if (!set) start++; | ||||
|           set = 0; | ||||
|         } | ||||
|       } | ||||
|       strip.setPixelSegment(255); | ||||
|       strip.trigger(); | ||||
|     } else { //return to regular effect | ||||
|       seg.setOption(SEG_OPTION_FREEZE, false); | ||||
|     } | ||||
|  | ||||
|   } | ||||
| } | ||||
|  | ||||
| bool deserializeState(JsonObject root) | ||||
| { | ||||
|   strip.applyToAllSelected = false; | ||||
|   bool stateResponse = root[F("v")] | false; | ||||
|  | ||||
|   bri = root["bri"] | bri; | ||||
|  | ||||
|   bool on = root["on"] | (bri > 0); | ||||
|   if (!on != !bri) toggleOnOff(); | ||||
|  | ||||
|   int tr = root[F("transition")] | -1; | ||||
|   if (tr >= 0) | ||||
|   { | ||||
|     transitionDelay = tr; | ||||
|     transitionDelay *= 100; | ||||
|     transitionDelayTemp = transitionDelay; | ||||
|   } | ||||
|  | ||||
|   tr = root[F("tt")] | -1; | ||||
|   if (tr >= 0) | ||||
|   { | ||||
|     transitionDelayTemp = tr; | ||||
|     transitionDelayTemp *= 100; | ||||
|     jsonTransitionOnce = true; | ||||
|   } | ||||
|   strip.setTransition(transitionDelayTemp); | ||||
|    | ||||
|   int cy = root[F("pl")] | -2; | ||||
|   if (cy > -2) presetCyclingEnabled = (cy >= 0); | ||||
|   JsonObject ccnf = root["ccnf"]; | ||||
|   presetCycleMin = ccnf[F("min")] | presetCycleMin; | ||||
|   presetCycleMax = ccnf[F("max")] | presetCycleMax; | ||||
|   tr = ccnf[F("time")] | -1; | ||||
|   if (tr >= 2) presetCycleTime = tr; | ||||
|  | ||||
|   JsonObject nl = root["nl"]; | ||||
|   nightlightActive    = nl["on"]      | nightlightActive; | ||||
|   nightlightDelayMins = nl[F("dur")]  | nightlightDelayMins; | ||||
|   nightlightMode      = nl[F("fade")] | nightlightMode; //deprecated, remove for v0.13.0 | ||||
|   nightlightMode      = nl[F("mode")] | nightlightMode; | ||||
|   nightlightTargetBri = nl[F("tbri")] | nightlightTargetBri; | ||||
|  | ||||
|   JsonObject udpn = root["udpn"]; | ||||
|   notifyDirect         = udpn["send"] | notifyDirect; | ||||
|   receiveNotifications = udpn["recv"] | receiveNotifications; | ||||
|   bool noNotification  = udpn[F("nn")]; //send no notification just for this request | ||||
|  | ||||
|   unsigned long timein = root[F("time")] | UINT32_MAX; | ||||
|   if (timein != UINT32_MAX) { | ||||
|     if (millis() - ntpLastSyncTime > 50000000L) setTime(timein); | ||||
|     if (presetsModifiedTime == 0) presetsModifiedTime = timein; | ||||
|   } | ||||
|  | ||||
|   doReboot = root[F("rb")] | doReboot; | ||||
|  | ||||
|   realtimeOverride = root[F("lor")] | realtimeOverride; | ||||
|   if (realtimeOverride > 2) realtimeOverride = REALTIME_OVERRIDE_ALWAYS; | ||||
|  | ||||
|   if (root.containsKey("live")) { | ||||
|     bool lv = root["live"]; | ||||
|     if (lv) realtimeLock(65000); //enter realtime without timeout | ||||
|     else    realtimeTimeout = 0; //cancel realtime mode immediately | ||||
|   } | ||||
|  | ||||
|   byte prevMain = strip.getMainSegmentId(); | ||||
|   strip.mainSegment = root[F("mainseg")] | prevMain; | ||||
|   if (strip.getMainSegmentId() != prevMain) setValuesFromMainSeg(); | ||||
|  | ||||
|   int it = 0; | ||||
|   JsonVariant segVar = root["seg"]; | ||||
|   if (segVar.is<JsonObject>()) | ||||
|   { | ||||
|     int id = segVar["id"] | -1; | ||||
|      | ||||
|     if (id < 0) { //set all selected segments | ||||
|       bool didSet = false; | ||||
|       byte lowestActive = 99; | ||||
|       for (byte s = 0; s < strip.getMaxSegments(); s++) | ||||
|       { | ||||
|         WS2812FX::Segment sg = strip.getSegment(s); | ||||
|         if (sg.isActive()) | ||||
|         { | ||||
|           if (lowestActive == 99) lowestActive = s; | ||||
|           if (sg.isSelected()) { | ||||
|             deserializeSegment(segVar, s); | ||||
|             didSet = true; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|       if (!didSet && lowestActive < strip.getMaxSegments()) deserializeSegment(segVar, lowestActive); | ||||
|     } else { //set only the segment with the specified ID | ||||
|       deserializeSegment(segVar, it); | ||||
|     } | ||||
|   } else { | ||||
|     JsonArray segs = segVar.as<JsonArray>(); | ||||
|     for (JsonObject elem : segs) | ||||
|     { | ||||
|       deserializeSegment(elem, it); | ||||
|       it++; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   usermods.readFromJsonState(root); | ||||
|  | ||||
|   int ps = root[F("psave")] | -1; | ||||
|   if (ps > 0) { | ||||
|     savePreset(ps, true, nullptr, root); | ||||
|   } else { | ||||
|     ps = root[F("pdel")] | -1; //deletion | ||||
|     if (ps > 0) { | ||||
|       deletePreset(ps); | ||||
|     } | ||||
|     ps = root["ps"] | -1; //load preset (clears state request!) | ||||
|     if (ps >= 0) {applyPreset(ps); return stateResponse;} | ||||
|  | ||||
|     //HTTP API commands | ||||
|     const char* httpwin = root["win"]; | ||||
|     if (httpwin) { | ||||
|       String apireq = "win&"; | ||||
|       apireq += httpwin; | ||||
|       handleSet(nullptr, apireq, false); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   JsonObject playlist = root[F("playlist")]; | ||||
|   if (!playlist.isNull()) { | ||||
|     loadPlaylist(playlist); | ||||
|     noNotification = true; //do not notify both for this request and the first playlist entry | ||||
|   } | ||||
|  | ||||
|   colorUpdated(noNotification ? NOTIFIER_CALL_MODE_NO_NOTIFY : NOTIFIER_CALL_MODE_DIRECT_CHANGE); | ||||
|  | ||||
|   return stateResponse; | ||||
| } | ||||
|  | ||||
| void serializeSegment(JsonObject& root, WS2812FX::Segment& seg, byte id, bool forPreset, bool segmentBounds) | ||||
| { | ||||
| 	root["id"] = id; | ||||
|   if (segmentBounds) { | ||||
|     root[F("start")] = seg.start; | ||||
|     root["stop"] = seg.stop; | ||||
|   } | ||||
| 	if (!forPreset)  root[F("len")] = seg.stop - seg.start; | ||||
|   root[F("grp")] = seg.grouping; | ||||
|   root[F("spc")] = seg.spacing; | ||||
|   root["on"] = seg.getOption(SEG_OPTION_ON); | ||||
|   byte segbri = seg.opacity; | ||||
|   root["bri"] = (segbri) ? segbri : 255; | ||||
|  | ||||
| 	JsonArray colarr = root.createNestedArray("col"); | ||||
|  | ||||
| 	for (uint8_t i = 0; i < 3; i++) | ||||
| 	{ | ||||
| 		JsonArray colX = colarr.createNestedArray(); | ||||
|     if (id == strip.getMainSegmentId() && i < 2) //temporary, to make transition work on main segment | ||||
|     { | ||||
|       if (i == 0) { | ||||
|         colX.add(col[0]); colX.add(col[1]); colX.add(col[2]); if (strip.isRgbw) colX.add(col[3]); | ||||
|       } else { | ||||
|          colX.add(colSec[0]); colX.add(colSec[1]); colX.add(colSec[2]); if (strip.isRgbw) colX.add(colSec[3]); | ||||
|       } | ||||
|     } else { | ||||
|   		colX.add((seg.colors[i] >> 16) & 0xFF); | ||||
|   		colX.add((seg.colors[i] >> 8) & 0xFF); | ||||
|   		colX.add((seg.colors[i]) & 0xFF); | ||||
|   		if (strip.isRgbw) | ||||
|   			colX.add((seg.colors[i] >> 24) & 0xFF); | ||||
|     } | ||||
| 	} | ||||
|  | ||||
| 	root[F("fx")]  = seg.mode; | ||||
| 	root[F("sx")]  = seg.speed; | ||||
| 	root[F("ix")]  = seg.intensity; | ||||
| 	root[F("pal")] = seg.palette; | ||||
| 	root[F("sel")] = seg.isSelected(); | ||||
| 	root["rev"] = seg.getOption(SEG_OPTION_REVERSED); | ||||
|   root[F("mi")]  = seg.getOption(SEG_OPTION_MIRROR); | ||||
| } | ||||
|  | ||||
| void serializeState(JsonObject root, bool forPreset, bool includeBri, bool segmentBounds) | ||||
| { | ||||
|   if (includeBri) { | ||||
|     root["on"] = (bri > 0); | ||||
|     root["bri"] = briLast; | ||||
|     root[F("transition")] = transitionDelay/100; //in 100ms | ||||
|   } | ||||
|  | ||||
|   if (!forPreset) { | ||||
|     if (errorFlag) root[F("error")] = errorFlag; | ||||
|  | ||||
|     root[F("ps")] = currentPreset; | ||||
|     root[F("pl")] = (presetCyclingEnabled) ? 0: -1; | ||||
|  | ||||
|     usermods.addToJsonState(root); | ||||
|  | ||||
|     //temporary for preset cycle | ||||
|     JsonObject ccnf = root.createNestedObject("ccnf"); | ||||
|     ccnf[F("min")] = presetCycleMin; | ||||
|     ccnf[F("max")] = presetCycleMax; | ||||
|     ccnf[F("time")] = presetCycleTime; | ||||
|  | ||||
|     JsonObject nl = root.createNestedObject("nl"); | ||||
|     nl["on"] = nightlightActive; | ||||
|     nl[F("dur")] = nightlightDelayMins; | ||||
|     nl[F("fade")] = (nightlightMode > NL_MODE_SET); //deprecated | ||||
|     nl[F("mode")] = nightlightMode; | ||||
|     nl[F("tbri")] = nightlightTargetBri; | ||||
|     if (nightlightActive) { | ||||
|       nl[F("rem")] = (nightlightDelayMs - (millis() - nightlightStartTime)) / 1000; // seconds remaining | ||||
|     } else { | ||||
|       nl[F("rem")] = -1; | ||||
|     } | ||||
|  | ||||
|     JsonObject udpn = root.createNestedObject("udpn"); | ||||
|     udpn["send"] = notifyDirect; | ||||
|     udpn["recv"] = receiveNotifications; | ||||
|  | ||||
|     root[F("lor")] = realtimeOverride; | ||||
|   } | ||||
|  | ||||
|   root[F("mainseg")] = strip.getMainSegmentId(); | ||||
|  | ||||
|   JsonArray seg = root.createNestedArray("seg"); | ||||
|   for (byte s = 0; s < strip.getMaxSegments(); s++) | ||||
|   { | ||||
|     WS2812FX::Segment sg = strip.getSegment(s); | ||||
|     if (sg.isActive()) | ||||
|     { | ||||
|       JsonObject seg0 = seg.createNestedObject(); | ||||
|       serializeSegment(seg0, sg, s, forPreset, segmentBounds); | ||||
|     } else if (forPreset && segmentBounds) { //disable segments not part of preset | ||||
|       JsonObject seg0 = seg.createNestedObject(); | ||||
|       seg0["stop"] = 0; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| //by https://github.com/tzapu/WiFiManager/blob/master/WiFiManager.cpp | ||||
| int getSignalQuality(int rssi) | ||||
| { | ||||
|     int quality = 0; | ||||
|  | ||||
|     if (rssi <= -100) | ||||
|     { | ||||
|         quality = 0; | ||||
|     } | ||||
|     else if (rssi >= -50) | ||||
|     { | ||||
|         quality = 100; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         quality = 2 * (rssi + 100); | ||||
|     } | ||||
|     return quality; | ||||
| } | ||||
|  | ||||
| void serializeInfo(JsonObject root) | ||||
| { | ||||
|   root[F("ver")] = versionString; | ||||
|   root[F("vid")] = VERSION; | ||||
|   //root[F("cn")] = WLED_CODENAME; | ||||
|  | ||||
|   JsonObject leds = root.createNestedObject("leds"); | ||||
|   leds[F("count")] = ledCount; | ||||
|   leds[F("rgbw")] = strip.isRgbw; | ||||
|   leds[F("wv")] = strip.isRgbw && (strip.rgbwMode == RGBW_MODE_MANUAL_ONLY || strip.rgbwMode == RGBW_MODE_DUAL); //should a white channel slider be displayed? | ||||
|   JsonArray leds_pin = leds.createNestedArray("pin"); | ||||
|   leds_pin.add(LEDPIN); | ||||
|  | ||||
|   leds[F("pwr")] = strip.currentMilliamps; | ||||
|   leds[F("fps")] = strip.getFps(); | ||||
|   leds[F("maxpwr")] = (strip.currentMilliamps)? strip.ablMilliampsMax : 0; | ||||
|   leds[F("maxseg")] = strip.getMaxSegments(); | ||||
|   leds[F("seglock")] = false; //will be used in the future to prevent modifications to segment config | ||||
|  | ||||
|   root[F("str")] = syncToggleReceive; | ||||
|  | ||||
|   root[F("name")] = serverDescription; | ||||
|   root[F("udpport")] = udpPort; | ||||
|   root["live"] = (bool)realtimeMode; | ||||
|  | ||||
|   switch (realtimeMode) { | ||||
|     case REALTIME_MODE_INACTIVE: root["lm"] = ""; break; | ||||
|     case REALTIME_MODE_GENERIC:  root["lm"] = ""; break; | ||||
|     case REALTIME_MODE_UDP:      root["lm"] = F("UDP"); break; | ||||
|     case REALTIME_MODE_HYPERION: root["lm"] = F("Hyperion"); break; | ||||
|     case REALTIME_MODE_E131:     root["lm"] = F("E1.31"); break; | ||||
|     case REALTIME_MODE_ADALIGHT: root["lm"] = F("USB Adalight/TPM2"); break; | ||||
|     case REALTIME_MODE_ARTNET:   root["lm"] = F("Art-Net"); break; | ||||
|     case REALTIME_MODE_TPM2NET:  root["lm"] = F("tpm2.net"); break; | ||||
|     case REALTIME_MODE_DDP:      root["lm"] = F("DDP"); break; | ||||
|   } | ||||
|  | ||||
|   if (realtimeIP[0] == 0) | ||||
|   { | ||||
|     root[F("lip")] = ""; | ||||
|   } else { | ||||
|     root[F("lip")] = realtimeIP.toString(); | ||||
|   } | ||||
|  | ||||
|   #ifdef WLED_ENABLE_WEBSOCKETS | ||||
|   root[F("ws")] = ws.count(); | ||||
|   #else | ||||
|   root[F("ws")] = -1; | ||||
|   #endif | ||||
|  | ||||
|   root[F("fxcount")] = strip.getModeCount(); | ||||
|   root[F("palcount")] = strip.getPaletteCount(); | ||||
|  | ||||
|   JsonObject wifi_info = root.createNestedObject("wifi"); | ||||
|   wifi_info[F("bssid")] = WiFi.BSSIDstr(); | ||||
|   int qrssi = WiFi.RSSI(); | ||||
|   wifi_info[F("rssi")] = qrssi; | ||||
|   wifi_info[F("signal")] = getSignalQuality(qrssi); | ||||
|   wifi_info[F("channel")] = WiFi.channel(); | ||||
|  | ||||
|   JsonObject fs_info = root.createNestedObject("fs"); | ||||
|   fs_info["u"] = fsBytesUsed / 1000; | ||||
|   fs_info["t"] = fsBytesTotal / 1000; | ||||
|   fs_info[F("pmt")] = presetsModifiedTime; | ||||
|  | ||||
|   root[F("ndc")] = nodeListEnabled ? (int)Nodes.size() : -1; | ||||
|    | ||||
|   #ifdef ARDUINO_ARCH_ESP32 | ||||
|   #ifdef WLED_DEBUG | ||||
|     wifi_info[F("txPower")] = (int) WiFi.getTxPower(); | ||||
|     wifi_info[F("sleep")] = (bool) WiFi.getSleep(); | ||||
|   #endif | ||||
|   root[F("arch")] = "esp32"; | ||||
|   root[F("core")] = ESP.getSdkVersion(); | ||||
|   //root[F("maxalloc")] = ESP.getMaxAllocHeap(); | ||||
|   #ifdef WLED_DEBUG | ||||
|     root[F("resetReason0")] = (int)rtc_get_reset_reason(0); | ||||
|     root[F("resetReason1")] = (int)rtc_get_reset_reason(1); | ||||
|   #endif | ||||
|   root[F("lwip")] = 0; | ||||
|   #else | ||||
|   root[F("arch")] = "esp8266"; | ||||
|   root[F("core")] = ESP.getCoreVersion(); | ||||
|   //root[F("maxalloc")] = ESP.getMaxFreeBlockSize(); | ||||
|   #ifdef WLED_DEBUG | ||||
|     root[F("resetReason")] = (int)ESP.getResetInfoPtr()->reason; | ||||
|   #endif | ||||
|   root[F("lwip")] = LWIP_VERSION_MAJOR; | ||||
|   #endif | ||||
|  | ||||
|   root[F("freeheap")] = ESP.getFreeHeap(); | ||||
|   root[F("uptime")] = millis()/1000 + rolloverMillis*4294967; | ||||
|  | ||||
|  | ||||
|   usermods.addToJsonInfo(root); | ||||
|  | ||||
|   byte os = 0; | ||||
|   #ifdef WLED_DEBUG | ||||
|   os  = 0x80; | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_ALEXA | ||||
|   os += 0x40; | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_BLYNK | ||||
|   os += 0x20; | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_CRONIXIE | ||||
|   os += 0x10; | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_FILESYSTEM | ||||
|   os += 0x08; | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_HUESYNC | ||||
|   os += 0x04; | ||||
|   #endif | ||||
|   #ifdef WLED_ENABLE_ADALIGHT | ||||
|   os += 0x02; | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_OTA | ||||
|   os += 0x01; | ||||
|   #endif | ||||
|   root[F("opt")] = os; | ||||
|  | ||||
|   root[F("brand")] = "WLED"; | ||||
|   root[F("product")] = F("FOSS"); | ||||
|   root["mac"] = escapedMac; | ||||
| } | ||||
|  | ||||
| void setPaletteColors(JsonArray json, CRGBPalette16 palette) | ||||
| { | ||||
|     for (int i = 0; i < 16; i++) { | ||||
|       JsonArray colors =  json.createNestedArray(); | ||||
|       CRGB color = palette[i]; | ||||
|       colors.add((((float)i / (float)16) * 255)); | ||||
|       colors.add(color.red); | ||||
|       colors.add(color.green); | ||||
|       colors.add(color.blue); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void setPaletteColors(JsonArray json, byte* tcp) | ||||
| { | ||||
|     TRGBGradientPaletteEntryUnion* ent = (TRGBGradientPaletteEntryUnion*)(tcp); | ||||
|     TRGBGradientPaletteEntryUnion u; | ||||
|  | ||||
|     // Count entries | ||||
|     uint16_t count = 0; | ||||
|     do { | ||||
|         u = *(ent + count); | ||||
|         count++; | ||||
|     } while ( u.index != 255); | ||||
|  | ||||
|     u = *ent; | ||||
|     int indexstart = 0; | ||||
|     while( indexstart < 255) { | ||||
|       indexstart = u.index; | ||||
|  | ||||
|       JsonArray colors =  json.createNestedArray(); | ||||
|       colors.add(u.index); | ||||
|       colors.add(u.r); | ||||
|       colors.add(u.g); | ||||
|       colors.add(u.b); | ||||
|  | ||||
|       ent++; | ||||
|       u = *ent; | ||||
|     } | ||||
| } | ||||
|  | ||||
| void serializePalettes(JsonObject root, AsyncWebServerRequest* request) | ||||
| { | ||||
|   #ifdef ESP8266 | ||||
|   int itemPerPage = 5; | ||||
|   #else | ||||
|   int itemPerPage = 8; | ||||
|   #endif | ||||
|  | ||||
|   int page = 0; | ||||
|   if (request->hasParam("page")) { | ||||
|     page = request->getParam("page")->value().toInt(); | ||||
|   } | ||||
|  | ||||
|   int palettesCount = strip.getPaletteCount(); | ||||
|  | ||||
|   int maxPage = (palettesCount -1) / itemPerPage; | ||||
|   if (page > maxPage) page = maxPage; | ||||
|  | ||||
|   int start = itemPerPage * page; | ||||
|   int end = start + itemPerPage; | ||||
|   if (end >= palettesCount) end = palettesCount; | ||||
|  | ||||
|   root[F("m")] = maxPage; | ||||
|   JsonObject palettes  = root.createNestedObject("p"); | ||||
|  | ||||
|   for (int i = start; i < end; i++) { | ||||
|     JsonArray curPalette = palettes.createNestedArray(String(i)); | ||||
|     CRGB prim; | ||||
|     CRGB sec; | ||||
|     CRGB ter; | ||||
|     switch (i) { | ||||
|       case 0: //default palette | ||||
|         setPaletteColors(curPalette, PartyColors_p);  | ||||
|         break; | ||||
|       case 1: //random | ||||
|           curPalette.add("r"); | ||||
|           curPalette.add("r"); | ||||
|           curPalette.add("r"); | ||||
|           curPalette.add("r"); | ||||
|         break; | ||||
|       case 2: //primary color only | ||||
|         curPalette.add(F("c1")); | ||||
|         break; | ||||
|       case 3: //primary + secondary | ||||
|         curPalette.add(F("c1")); | ||||
|         curPalette.add(F("c1")); | ||||
|         curPalette.add(F("c2")); | ||||
|         curPalette.add(F("c2")); | ||||
|         break; | ||||
|       case 4: //primary + secondary + tertiary | ||||
|         curPalette.add(F("c3")); | ||||
|         curPalette.add(F("c2")); | ||||
|         curPalette.add(F("c1")); | ||||
|         break; | ||||
|       case 5: {//primary + secondary (+tert if not off), more distinct | ||||
|        | ||||
|         curPalette.add(F("c1")); | ||||
|         curPalette.add(F("c1")); | ||||
|         curPalette.add(F("c1")); | ||||
|         curPalette.add(F("c1")); | ||||
|         curPalette.add(F("c1")); | ||||
|         curPalette.add(F("c2")); | ||||
|         curPalette.add(F("c2")); | ||||
|         curPalette.add(F("c2")); | ||||
|         curPalette.add(F("c2")); | ||||
|         curPalette.add(F("c2")); | ||||
|         curPalette.add(F("c3")); | ||||
|         curPalette.add(F("c3")); | ||||
|         curPalette.add(F("c3")); | ||||
|         curPalette.add(F("c3")); | ||||
|         curPalette.add(F("c3")); | ||||
|         curPalette.add(F("c1")); | ||||
|         break;} | ||||
|       case 6: //Party colors | ||||
|         setPaletteColors(curPalette, PartyColors_p); | ||||
|         break; | ||||
|       case 7: //Cloud colors | ||||
|         setPaletteColors(curPalette, CloudColors_p); | ||||
|         break; | ||||
|       case 8: //Lava colors | ||||
|         setPaletteColors(curPalette, LavaColors_p); | ||||
|         break; | ||||
|       case 9: //Ocean colors | ||||
|         setPaletteColors(curPalette, OceanColors_p); | ||||
|         break; | ||||
|       case 10: //Forest colors | ||||
|         setPaletteColors(curPalette, ForestColors_p); | ||||
|         break; | ||||
|       case 11: //Rainbow colors | ||||
|         setPaletteColors(curPalette, RainbowColors_p); | ||||
|         break; | ||||
|       case 12: //Rainbow stripe colors | ||||
|         setPaletteColors(curPalette, RainbowStripeColors_p); | ||||
|         break; | ||||
|  | ||||
|       default: | ||||
|         if (i < 13) { | ||||
|           break; | ||||
|         } | ||||
|         byte tcp[72]; | ||||
|         memcpy_P(tcp, (byte*)pgm_read_dword(&(gGradientPalettes[i - 13])), 72); | ||||
|         setPaletteColors(curPalette, tcp); | ||||
|         break; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| void serializeNodes(JsonObject root) | ||||
| { | ||||
|   JsonArray nodes = root.createNestedArray("nodes"); | ||||
|  | ||||
|   for (NodesMap::iterator it = Nodes.begin(); it != Nodes.end(); ++it) | ||||
|   { | ||||
|     if (it->second.ip[0] != 0) | ||||
|     { | ||||
|       JsonObject node = nodes.createNestedObject(); | ||||
|       node[F("name")] = it->second.nodeName; | ||||
|       node["type"]    = it->second.nodeType; | ||||
|       node["ip"]      = it->second.ip.toString(); | ||||
|       node[F("age")]  = it->second.age; | ||||
|       node[F("vid")]  = it->second.build; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| void serveJson(AsyncWebServerRequest* request) | ||||
| { | ||||
|   byte subJson = 0; | ||||
|   const String& url = request->url(); | ||||
|   if      (url.indexOf("state") > 0) subJson = 1; | ||||
|   else if (url.indexOf("info")  > 0) subJson = 2; | ||||
|   else if (url.indexOf("si") > 0) subJson = 3; | ||||
|   else if (url.indexOf("nodes") > 0) subJson = 4; | ||||
|   else if (url.indexOf("palx") > 0) subJson = 5; | ||||
|   else if (url.indexOf("live")  > 0) { | ||||
|     serveLiveLeds(request); | ||||
|     return; | ||||
|   } | ||||
|   else if (url.indexOf(F("eff"))   > 0) { | ||||
|     request->send_P(200, "application/json", JSON_mode_names); | ||||
|     return; | ||||
|   } | ||||
|   else if (url.indexOf(F("pal"))   > 0) { | ||||
|     request->send_P(200, "application/json", JSON_palette_names); | ||||
|     return; | ||||
|   } | ||||
|   else if (url.length() > 6) { //not just /json | ||||
|     request->send(  501, "application/json", F("{\"error\":\"Not implemented\"}")); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   AsyncJsonResponse* response = new AsyncJsonResponse(JSON_BUFFER_SIZE); | ||||
|   JsonObject doc = response->getRoot(); | ||||
|  | ||||
|   switch (subJson) | ||||
|   { | ||||
|     case 1: //state | ||||
|       serializeState(doc); break; | ||||
|     case 2: //info | ||||
|       serializeInfo(doc); break; | ||||
|     case 4: //node list | ||||
|       serializeNodes(doc); break; | ||||
|     case 5: //palettes | ||||
|       serializePalettes(doc, request); break; | ||||
|     default: //all | ||||
|       JsonObject state = doc.createNestedObject("state"); | ||||
|       serializeState(state); | ||||
|       JsonObject info = doc.createNestedObject("info"); | ||||
|       serializeInfo(info); | ||||
|       if (subJson != 3) | ||||
|       { | ||||
|         doc[F("effects")]  = serialized((const __FlashStringHelper*)JSON_mode_names); | ||||
|         doc[F("palettes")] = serialized((const __FlashStringHelper*)JSON_palette_names); | ||||
|       } | ||||
|   } | ||||
|  | ||||
|   response->setLength(); | ||||
|   request->send(response); | ||||
| } | ||||
|  | ||||
| #define MAX_LIVE_LEDS 180 | ||||
|  | ||||
| bool serveLiveLeds(AsyncWebServerRequest* request, uint32_t wsClient) | ||||
| { | ||||
|   AsyncWebSocketClient * wsc = nullptr; | ||||
|   if (!request) { //not HTTP, use Websockets | ||||
|     #ifdef WLED_ENABLE_WEBSOCKETS | ||||
|     wsc = ws.client(wsClient); | ||||
|     if (!wsc || wsc->queueLength() > 0) return false; //only send if queue free | ||||
|     #endif | ||||
|   } | ||||
|  | ||||
|   uint16_t used = ledCount; | ||||
|   uint16_t n = (used -1) /MAX_LIVE_LEDS +1; //only serve every n'th LED if count over MAX_LIVE_LEDS | ||||
|   char buffer[2000]; | ||||
|   strcpy_P(buffer, PSTR("{\"leds\":[")); | ||||
|   obuf = buffer; | ||||
|   olen = 9; | ||||
|  | ||||
|   for (uint16_t i= 0; i < used; i += n) | ||||
|   { | ||||
|     olen += sprintf(obuf + olen, "\"%06X\",", strip.getPixelColor(i) & 0xFFFFFF); | ||||
|   } | ||||
|   olen -= 1; | ||||
|   oappend((const char*)F("],\"n\":")); | ||||
|   oappendi(n); | ||||
|   oappend("}"); | ||||
|   if (request) { | ||||
|     request->send(200, "application/json", buffer); | ||||
|   } | ||||
|   #ifdef WLED_ENABLE_WEBSOCKETS | ||||
|   else { | ||||
|     wsc->text(obuf, olen); | ||||
|   } | ||||
|   #endif | ||||
|   return true; | ||||
| } | ||||
							
								
								
									
										319
									
								
								wled00/led.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										319
									
								
								wled00/led.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,319 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * LED methods | ||||
|  */ | ||||
| void setValuesFromMainSeg() | ||||
| { | ||||
|   WS2812FX::Segment& seg = strip.getSegment(strip.getMainSegmentId()); | ||||
|   colorFromUint32(seg.colors[0]); | ||||
|   colorFromUint32(seg.colors[1], true); | ||||
|   effectCurrent = seg.mode; | ||||
|   effectSpeed = seg.speed; | ||||
|   effectIntensity = seg.intensity; | ||||
|   effectPalette = seg.palette; | ||||
| } | ||||
|  | ||||
|  | ||||
| void resetTimebase() | ||||
| { | ||||
|   strip.timebase = 0 - millis(); | ||||
| } | ||||
|  | ||||
|  | ||||
| void toggleOnOff() | ||||
| { | ||||
|   if (bri == 0) | ||||
|   { | ||||
|     bri = briLast; | ||||
|   } else | ||||
|   { | ||||
|     briLast = bri; | ||||
|     bri = 0; | ||||
|     unloadPlaylist(); | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| //scales the brightness with the briMultiplier factor | ||||
| byte scaledBri(byte in) | ||||
| { | ||||
|   uint32_t d = in*briMultiplier; | ||||
|   uint32_t val = d/100; | ||||
|   if (val > 255) val = 255; | ||||
|   return (byte)val; | ||||
| } | ||||
|  | ||||
|  | ||||
| void setAllLeds() { | ||||
|   if (!realtimeMode || !arlsForceMaxBri) | ||||
|   { | ||||
|     strip.setBrightness(scaledBri(briT)); | ||||
|   } | ||||
|   if (strip.isRgbw && strip.rgbwMode == RGBW_MODE_LEGACY) | ||||
|   { | ||||
|     colorRGBtoRGBW(col); | ||||
|     colorRGBtoRGBW(colSec); | ||||
|   } | ||||
|   strip.setColor(0, col[0], col[1], col[2], col[3]); | ||||
|   strip.setColor(1, colSec[0], colSec[1], colSec[2], colSec[3]); | ||||
|   if (strip.isRgbw && strip.rgbwMode == RGBW_MODE_LEGACY) | ||||
|   { | ||||
|     col[3] = 0; colSec[3] = 0; | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| void setLedsStandard() | ||||
| { | ||||
|   briOld = bri; | ||||
|   briT = bri; | ||||
|   setAllLeds(); | ||||
| } | ||||
|  | ||||
|  | ||||
| bool colorChanged() | ||||
| { | ||||
|   for (byte i=0; i<4; i++) | ||||
|   { | ||||
|     if (col[i] != colIT[i]) return true; | ||||
|     if (colSec[i] != colSecIT[i]) return true; | ||||
|   } | ||||
|   if (bri != briIT) return true; | ||||
|   return false; | ||||
| } | ||||
|  | ||||
|  | ||||
| void colorUpdated(int callMode) | ||||
| { | ||||
|   //call for notifier -> 0: init 1: direct change 2: button 3: notification 4: nightlight 5: other (No notification) | ||||
|   //                     6: fx changed 7: hue 8: preset cycle 9: blynk 10: alexa | ||||
|   if (callMode != NOTIFIER_CALL_MODE_INIT &&  | ||||
|       callMode != NOTIFIER_CALL_MODE_DIRECT_CHANGE &&  | ||||
|       callMode != NOTIFIER_CALL_MODE_NO_NOTIFY) strip.applyToAllSelected = true; //if not from JSON api, which directly sets segments | ||||
|  | ||||
|   bool someSel = false; | ||||
|  | ||||
|   if (callMode == NOTIFIER_CALL_MODE_NOTIFICATION) { | ||||
|     someSel = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects); | ||||
|   } | ||||
|    | ||||
|   //Notifier: apply received FX to selected segments only if actually receiving FX | ||||
|   if (someSel) strip.applyToAllSelected = receiveNotificationEffects; | ||||
|  | ||||
|   bool fxChanged = strip.setEffectConfig(effectCurrent, effectSpeed, effectIntensity, effectPalette) || effectChanged; | ||||
|   bool colChanged = colorChanged(); | ||||
|  | ||||
|   //Notifier: apply received color to selected segments only if actually receiving color | ||||
|   if (someSel) strip.applyToAllSelected = receiveNotificationColor; | ||||
|  | ||||
|   if (fxChanged || colChanged) | ||||
|   { | ||||
|     effectChanged = false; | ||||
|     if (realtimeTimeout == UINT32_MAX) realtimeTimeout = 0; | ||||
|     if (isPreset) {isPreset = false;} | ||||
|         else {currentPreset = -1;} | ||||
|          | ||||
|     notify(callMode); | ||||
|      | ||||
|     //set flag to update blynk and mqtt | ||||
|     interfaceUpdateCallMode = callMode; | ||||
|   } else { | ||||
|     if (nightlightActive && !nightlightActiveOld &&  | ||||
|         callMode != NOTIFIER_CALL_MODE_NOTIFICATION &&  | ||||
|         callMode != NOTIFIER_CALL_MODE_NO_NOTIFY) | ||||
|     { | ||||
|       notify(NOTIFIER_CALL_MODE_NIGHTLIGHT);  | ||||
|       interfaceUpdateCallMode = NOTIFIER_CALL_MODE_NIGHTLIGHT; | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   if (!colChanged) return; //following code is for e.g. initiating transitions | ||||
|    | ||||
|   if (callMode != NOTIFIER_CALL_MODE_NO_NOTIFY && nightlightActive && (nightlightMode == NL_MODE_FADE || nightlightMode == NL_MODE_COLORFADE)) | ||||
|   { | ||||
|     briNlT = bri; | ||||
|     nightlightDelayMs -= (millis() - nightlightStartTime); | ||||
|     nightlightStartTime = millis(); | ||||
|   } | ||||
|   for (byte i=0; i<4; i++) | ||||
|   { | ||||
|     colIT[i] = col[i]; | ||||
|     colSecIT[i] = colSec[i]; | ||||
|   } | ||||
|   if (briT == 0) | ||||
|   { | ||||
|     //setLedsStandard(true); //do not color transition if starting from off! | ||||
|     if (callMode != NOTIFIER_CALL_MODE_NOTIFICATION) resetTimebase(); //effect start from beginning | ||||
|   } | ||||
|  | ||||
|   briIT = bri; | ||||
|   if (bri > 0) briLast = bri; | ||||
|  | ||||
|   //deactivate nightlight if target brightness is reached | ||||
|   if (bri == nightlightTargetBri && callMode != NOTIFIER_CALL_MODE_NO_NOTIFY && nightlightMode != NL_MODE_SUN) nightlightActive = false; | ||||
|    | ||||
|   if (fadeTransition) | ||||
|   { | ||||
|     //set correct delay if not using notification delay | ||||
|     if (callMode != NOTIFIER_CALL_MODE_NOTIFICATION && !jsonTransitionOnce) transitionDelayTemp = transitionDelay; | ||||
|     jsonTransitionOnce = false; | ||||
|     strip.setTransition(transitionDelayTemp); | ||||
|     if (transitionDelayTemp == 0) {setLedsStandard(); strip.trigger(); return;} | ||||
|      | ||||
|     if (transitionActive) | ||||
|     { | ||||
|       briOld = briT; | ||||
|       tperLast = 0; | ||||
|     } | ||||
|     strip.setTransitionMode(true); | ||||
|     transitionActive = true; | ||||
|     transitionStartTime = millis(); | ||||
|   } else | ||||
|   { | ||||
|     strip.setTransition(0); | ||||
|     setLedsStandard(); | ||||
|     strip.trigger(); | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| void updateInterfaces(uint8_t callMode) | ||||
| { | ||||
|   sendDataWs(); | ||||
|   #ifndef WLED_DISABLE_ALEXA | ||||
|   if (espalexaDevice != nullptr && callMode != NOTIFIER_CALL_MODE_ALEXA) { | ||||
|     espalexaDevice->setValue(bri); | ||||
|     espalexaDevice->setColor(col[0], col[1], col[2]); | ||||
|   } | ||||
|   #endif | ||||
|   if (callMode != NOTIFIER_CALL_MODE_BLYNK &&  | ||||
|       callMode != NOTIFIER_CALL_MODE_NO_NOTIFY) updateBlynk(); | ||||
|   doPublishMqtt = true; | ||||
|   lastInterfaceUpdate = millis(); | ||||
| } | ||||
|  | ||||
|  | ||||
| void handleTransitions() | ||||
| { | ||||
|   //handle still pending interface update | ||||
|   if (interfaceUpdateCallMode && millis() - lastInterfaceUpdate > 2000) | ||||
|   { | ||||
|     updateInterfaces(interfaceUpdateCallMode); | ||||
|     interfaceUpdateCallMode = 0; //disable | ||||
|   } | ||||
|   if (doPublishMqtt) publishMqtt(); | ||||
|    | ||||
|   if (transitionActive && transitionDelayTemp > 0) | ||||
|   { | ||||
|     float tper = (millis() - transitionStartTime)/(float)transitionDelayTemp; | ||||
|     if (tper >= 1.0) | ||||
|     { | ||||
|       strip.setTransitionMode(false); | ||||
|       transitionActive = false; | ||||
|       tperLast = 0; | ||||
|       setLedsStandard(); | ||||
|       return; | ||||
|     } | ||||
|     if (tper - tperLast < 0.004) return; | ||||
|     tperLast = tper; | ||||
|     briT    = briOld   +((bri    - briOld   )*tper); | ||||
|      | ||||
|     setAllLeds(); | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| void handleNightlight() | ||||
| { | ||||
|   if (nightlightActive) | ||||
|   { | ||||
|     if (!nightlightActiveOld) //init | ||||
|     { | ||||
|       nightlightStartTime = millis(); | ||||
|       nightlightDelayMs = (int)(nightlightDelayMins*60000); | ||||
|       nightlightActiveOld = true; | ||||
|       briNlT = bri; | ||||
|       for (byte i=0; i<4; i++) colNlT[i] = col[i]; // remember starting color | ||||
|       if (nightlightMode == NL_MODE_SUN) | ||||
|       { | ||||
|         //save current | ||||
|         colNlT[0] = effectCurrent; | ||||
|         colNlT[1] = effectSpeed; | ||||
|         colNlT[2] = effectPalette; | ||||
|  | ||||
|         strip.setMode(strip.getMainSegmentId(), FX_MODE_STATIC); //make sure seg runtime is reset if left in sunrise mode | ||||
|         effectCurrent = FX_MODE_SUNRISE; | ||||
|         effectSpeed = nightlightDelayMins; | ||||
|         effectPalette = 0; | ||||
|         if (effectSpeed > 60) effectSpeed = 60; //currently limited to 60 minutes | ||||
|         if (bri) effectSpeed += 60; //sunset if currently on | ||||
|         briNlT = !bri; //true == sunrise, false == sunset | ||||
|         if (!bri) bri = briLast; | ||||
|         colorUpdated(NOTIFIER_CALL_MODE_NO_NOTIFY); | ||||
|       } | ||||
|     } | ||||
|     float nper = (millis() - nightlightStartTime)/((float)nightlightDelayMs); | ||||
|     if (nightlightMode == NL_MODE_FADE || nightlightMode == NL_MODE_COLORFADE) | ||||
|     { | ||||
|       bri = briNlT + ((nightlightTargetBri - briNlT)*nper); | ||||
|       if (nightlightMode == NL_MODE_COLORFADE)                                         // color fading only is enabled with "NF=2" | ||||
|       { | ||||
|         for (byte i=0; i<4; i++) col[i] = colNlT[i]+ ((colSec[i] - colNlT[i])*nper);   // fading from actual color to secondary color | ||||
|       } | ||||
|       colorUpdated(NOTIFIER_CALL_MODE_NO_NOTIFY); | ||||
|     } | ||||
|     if (nper >= 1) //nightlight duration over | ||||
|     { | ||||
|       nightlightActive = false; | ||||
|       if (nightlightMode == NL_MODE_SET) | ||||
|       { | ||||
|         bri = nightlightTargetBri; | ||||
|         colorUpdated(NOTIFIER_CALL_MODE_NO_NOTIFY); | ||||
|       } | ||||
|       if (bri == 0) briLast = briNlT; | ||||
|       if (nightlightMode == NL_MODE_SUN) | ||||
|       { | ||||
|         if (!briNlT) { //turn off if sunset | ||||
|           effectCurrent = colNlT[0]; | ||||
|           effectSpeed = colNlT[1]; | ||||
|           effectPalette = colNlT[2]; | ||||
|           toggleOnOff(); | ||||
|           setLedsStandard(); | ||||
|         } | ||||
|       } | ||||
|       updateBlynk(); | ||||
|       if (macroNl > 0) | ||||
|         applyPreset(macroNl); | ||||
|       nightlightActiveOld = false; | ||||
|     } | ||||
|   } else if (nightlightActiveOld) //early de-init | ||||
|   { | ||||
|     if (nightlightMode == NL_MODE_SUN) { //restore previous effect | ||||
|       effectCurrent = colNlT[0]; | ||||
|       effectSpeed = colNlT[1]; | ||||
|       effectPalette = colNlT[2]; | ||||
|       colorUpdated(NOTIFIER_CALL_MODE_NO_NOTIFY); | ||||
|     } | ||||
|     nightlightActiveOld = false; | ||||
|   } | ||||
|  | ||||
|   //also handle preset cycle here | ||||
|   if (presetCyclingEnabled && (millis() - presetCycledTime > (100*presetCycleTime))) | ||||
|   { | ||||
|     presetCycledTime = millis(); | ||||
|     if (bri == 0 || nightlightActive) return; | ||||
|  | ||||
|     if (presetCycCurr < presetCycleMin || presetCycCurr > presetCycleMax) presetCycCurr = presetCycleMin; | ||||
|     applyPreset(presetCycCurr); //this handles colorUpdated() for us | ||||
|     presetCycCurr++; | ||||
|     if (presetCycCurr > 250) presetCycCurr = 1; | ||||
|     interfaceUpdateCallMode = 0; //disable updates to MQTT and Blynk | ||||
|   } | ||||
| } | ||||
|  | ||||
| //utility for FastLED to use our custom timer | ||||
| uint32_t get_millisecond_timer() | ||||
| { | ||||
|   return strip.now; | ||||
| } | ||||
							
								
								
									
										76
									
								
								wled00/lx_parser.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								wled00/lx_parser.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,76 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * Parser for Loxone formats | ||||
|  */ | ||||
| bool parseLx(int lxValue, byte rgbw[4]) | ||||
| { | ||||
|   #ifdef WLED_ENABLE_LOXONE | ||||
|   DEBUG_PRINT(F("LX: Lox = ")); | ||||
|   DEBUG_PRINTLN(lxValue); | ||||
|  | ||||
|   bool ok = false; | ||||
|   float lxRed = 0, lxGreen = 0, lxBlue = 0; | ||||
|  | ||||
|   if (lxValue < 200000000) {  | ||||
|     // Loxone RGB | ||||
|     ok = true; | ||||
|     lxRed = round((lxValue % 1000) * 2.55); | ||||
|     lxGreen = round(((lxValue / 1000) % 1000) * 2.55); | ||||
|     lxBlue = round(((lxValue / 1000000) % 1000) * 2.55); | ||||
|   } else if ((lxValue >= 200000000) && (lxValue <= 201006500)) {  | ||||
|     // Loxone Lumitech | ||||
|     ok = true; | ||||
|     float tmpBri = floor((lxValue - 200000000) / 10000); ; | ||||
|     uint16_t ct = (lxValue - 200000000) - (((uint8_t)tmpBri) * 10000); | ||||
|  | ||||
|     tmpBri *= 2.55; | ||||
|     tmpBri = constrain(tmpBri, 0, 255); | ||||
|  | ||||
|     colorKtoRGB(ct, rgbw); | ||||
|     lxRed = rgbw[0]; lxGreen = rgbw[1]; lxBlue = rgbw[2]; | ||||
|  | ||||
|     lxRed *= (tmpBri/255); | ||||
|     lxGreen *= (tmpBri/255); | ||||
|     lxBlue *= (tmpBri/255); | ||||
|   } | ||||
|  | ||||
|   if (ok) { | ||||
|     rgbw[0] = (uint8_t) constrain(lxRed, 0, 255); | ||||
|     rgbw[1] = (uint8_t) constrain(lxGreen, 0, 255); | ||||
|     rgbw[2] = (uint8_t) constrain(lxBlue, 0, 255); | ||||
|     rgbw[3] = 0; | ||||
|     return true; | ||||
|   } | ||||
|   #endif | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| void parseLxJson(int lxValue, byte segId, bool secondary) | ||||
| { | ||||
|   if (secondary) { | ||||
|     DEBUG_PRINT(F("LY: Lox secondary = ")); | ||||
|   } else { | ||||
|     DEBUG_PRINT(F("LX: Lox primary = ")); | ||||
|   } | ||||
|   DEBUG_PRINTLN(lxValue); | ||||
|   byte rgbw[] = {0,0,0,0}; | ||||
|   if (parseLx(lxValue, rgbw)) { | ||||
|     if (bri == 0) { | ||||
|       DEBUG_PRINTLN(F("LX: turn on")); | ||||
|       toggleOnOff(); | ||||
|     } | ||||
|     bri = 255; | ||||
|     nightlightActive = false; //always disable nightlight when toggling | ||||
|     if (segId == strip.getMainSegmentId()) { | ||||
|       DEBUG_PRINTLN(F("LX: main segment")); | ||||
|       if (secondary) for (byte i = 0; i < 4; i++) colSec[i] = rgbw[i]; | ||||
|       else           for (byte i = 0; i < 4; i++) col[i]    = rgbw[i]; | ||||
|     } else { | ||||
|       DEBUG_PRINT(F("LX: segment ")); | ||||
|       DEBUG_PRINTLN(segId); | ||||
|       strip.getSegment(segId).setColor(secondary, ((rgbw[3] << 24) | ((rgbw[0]&0xFF) << 16) | ((rgbw[1]&0xFF) << 8) | ((rgbw[2]&0xFF))), segId); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| @@ -1,8 +1,11 @@ | ||||
| #include "wled.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * MQTT communication protocol for home automation | ||||
|  */ | ||||
| 
 | ||||
| #ifdef WLED_ENABLE_MQTT | ||||
| #define MQTT_KEEP_ALIVE_TIME 60    // contact the MQTT broker every 60 seconds
 | ||||
| 
 | ||||
| void parseMQTTBriPayload(char* payload) | ||||
| { | ||||
| @@ -12,7 +15,7 @@ void parseMQTTBriPayload(char* payload) | ||||
|     uint8_t in = strtoul(payload, NULL, 10); | ||||
|     if (in == 0 && bri > 0) briLast = bri; | ||||
|     bri = in; | ||||
|     colorUpdated(1); | ||||
|     colorUpdated(NOTIFIER_CALL_MODE_DIRECT_CHANGE); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| @@ -45,41 +48,69 @@ void onMqttConnect(bool sessionPresent) | ||||
|   } | ||||
| 
 | ||||
|   doPublishMqtt = true; | ||||
|   DEBUG_PRINTLN("MQTT ready"); | ||||
|   DEBUG_PRINTLN(F("MQTT ready")); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void onMqttMessage(char* topic, char* payload, AsyncMqttClientMessageProperties properties, size_t len, size_t index, size_t total) { | ||||
| 
 | ||||
|   DEBUG_PRINT("MQTT msg: "); | ||||
|   DEBUG_PRINT(F("MQTT msg: ")); | ||||
|   DEBUG_PRINTLN(topic); | ||||
| 
 | ||||
|   // paranoia check to avoid npe if no payload
 | ||||
|   if (payload==nullptr) { | ||||
|     DEBUG_PRINTLN(F("no payload -> leave")); | ||||
|     return; | ||||
|   } | ||||
|   DEBUG_PRINTLN(payload); | ||||
| 
 | ||||
|   //no need to check the topic because we only get topics we are subscribed to
 | ||||
|   size_t topicPrefixLen = strlen(mqttDeviceTopic); | ||||
|   if (strncmp(topic, mqttDeviceTopic, topicPrefixLen) == 0) { | ||||
|       topic += topicPrefixLen; | ||||
|   } else { | ||||
|       topicPrefixLen = strlen(mqttGroupTopic); | ||||
|       if (strncmp(topic, mqttGroupTopic, topicPrefixLen) == 0) { | ||||
|           topic += topicPrefixLen; | ||||
|       } else { | ||||
|           // Topic not used here. Probably a usermod subscribed to this topic.
 | ||||
|           return; | ||||
|       } | ||||
|   } | ||||
| 
 | ||||
|   if (strstr(topic, "/col")) | ||||
|   //Prefix is stripped from the topic at this point
 | ||||
| 
 | ||||
|   if (strcmp(topic, "/col") == 0) | ||||
|   { | ||||
|     colorFromDecOrHexString(col, (char*)payload); | ||||
|     colorUpdated(1); | ||||
|   } else if (strstr(topic, "/api")) | ||||
|     colorUpdated(NOTIFIER_CALL_MODE_DIRECT_CHANGE); | ||||
|   } else if (strcmp(topic, "/api") == 0) | ||||
|   { | ||||
|     String apireq = "win&"; | ||||
|     apireq += (char*)payload; | ||||
|     handleSet(nullptr, apireq); | ||||
|   } else parseMQTTBriPayload(payload); | ||||
|     if (payload[0] == '{') { //JSON API
 | ||||
|       DynamicJsonDocument doc(JSON_BUFFER_SIZE); | ||||
|       deserializeJson(doc, payload); | ||||
|       deserializeState(doc.as<JsonObject>()); | ||||
|     } else { //HTTP API
 | ||||
|       String apireq = "win&"; | ||||
|       apireq += (char*)payload; | ||||
|       handleSet(nullptr, apireq); | ||||
|     } | ||||
|   } else if (strcmp(topic, "") == 0) | ||||
|   { | ||||
|     parseMQTTBriPayload(payload); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void publishMqtt() | ||||
| { | ||||
|   doPublishMqtt = false; | ||||
|   if (mqtt == nullptr || !mqtt->connected()) return; | ||||
|   DEBUG_PRINTLN("Publish MQTT"); | ||||
|   if (!WLED_MQTT_CONNECTED) return; | ||||
|   DEBUG_PRINTLN(F("Publish MQTT")); | ||||
| 
 | ||||
|   char s[10]; | ||||
|   char subuf[38]; | ||||
| 
 | ||||
|   sprintf(s, "%ld", bri); | ||||
|   sprintf(s, "%u", bri); | ||||
|   strcpy(subuf, mqttDeviceTopic); | ||||
|   strcat(subuf, "/g"); | ||||
|   mqtt->publish(subuf, 0, true, s); | ||||
| @@ -115,7 +146,7 @@ bool initMqtt() | ||||
|   } | ||||
|   if (mqtt->connected()) return true; | ||||
| 
 | ||||
|   DEBUG_PRINTLN("Reconnecting MQTT"); | ||||
|   DEBUG_PRINTLN(F("Reconnecting MQTT")); | ||||
|   IPAddress mqttIP; | ||||
|   if (mqttIP.fromString(mqttServer)) //see if server is IP or domain
 | ||||
|   { | ||||
| @@ -129,6 +160,7 @@ bool initMqtt() | ||||
|   strcpy(mqttStatusTopic, mqttDeviceTopic); | ||||
|   strcat(mqttStatusTopic, "/status"); | ||||
|   mqtt->setWill(mqttStatusTopic, 0, true, "offline"); | ||||
|   mqtt->setKeepAlive(MQTT_KEEP_ALIVE_TIME); | ||||
|   mqtt->connect(); | ||||
|   return true; | ||||
| } | ||||
							
								
								
									
										25
									
								
								wled00/my_config_sample.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								wled00/my_config_sample.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| #pragma once | ||||
|  | ||||
| /* | ||||
|  * Welcome!  | ||||
|  * You can use the file "my_config.h" to make changes to the way WLED is compiled! | ||||
|  * It is possible to enable and disable certain features as well as set defaults for some runtime changeable settings. | ||||
|  *  | ||||
|  * How to use: | ||||
|  * PlatformIO: Just compile the unmodified code once! The file "my_config.h" will be generated automatically and now you can make your changes. | ||||
|  *  | ||||
|  * ArduinoIDE: Make a copy of this file and name it "my_config.h". Go to wled.h and uncomment "#define WLED_USE_MY_CONFIG" in the top of the file. | ||||
|  *  | ||||
|  * DO NOT make changes to the "my_config_sample.h" file directly! Your changes will not be applied. | ||||
|  */ | ||||
|  | ||||
| // force the compiler to show a warning to confirm that this file is included | ||||
| #warning **** my_config.h: Settings from this file are honored **** | ||||
|  | ||||
| /* Uncomment to use your WIFI settings as defaults | ||||
|   //WARNING: this will hardcode these as the default even after a factory reset | ||||
| #define CLIENT_SSID "Your_SSID" | ||||
| #define CLIENT_PASS "Your_Password" | ||||
| */ | ||||
|  | ||||
| //#define MAX_LEDS 1500       //Maximum total LEDs. More than 1500 might create a low memory situation on ESP8266. | ||||
							
								
								
									
										279
									
								
								wled00/ntp.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										279
									
								
								wled00/ntp.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,279 @@ | ||||
| #include "src/dependencies/timezone/Timezone.h" | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * Acquires time from NTP server | ||||
|  */ | ||||
| Timezone* tz; | ||||
|  | ||||
| #define TZ_UTC                  0 | ||||
| #define TZ_UK                   1 | ||||
| #define TZ_EUROPE_CENTRAL       2 | ||||
| #define TZ_EUROPE_EASTERN       3 | ||||
| #define TZ_US_EASTERN           4 | ||||
| #define TZ_US_CENTRAL           5 | ||||
| #define TZ_US_MOUNTAIN          6 | ||||
| #define TZ_US_ARIZONA           7 | ||||
| #define TZ_US_PACIFIC           8 | ||||
| #define TZ_CHINA                9 | ||||
| #define TZ_JAPAN               10 | ||||
| #define TZ_AUSTRALIA_EASTERN   11 | ||||
| #define TZ_NEW_ZEALAND         12 | ||||
| #define TZ_NORTH_KOREA         13 | ||||
| #define TZ_INDIA               14 | ||||
| #define TZ_SASKACHEWAN         15 | ||||
| #define TZ_AUSTRALIA_NORTHERN  16 | ||||
| #define TZ_AUSTRALIA_SOUTHERN  17 | ||||
| #define TZ_HAWAII              18 | ||||
| #define TZ_INIT               255 | ||||
|  | ||||
| byte tzCurrent = TZ_INIT; //uninitialized | ||||
|  | ||||
| void updateTimezone() { | ||||
|   delete tz; | ||||
|   TimeChangeRule tcrDaylight = {Last, Sun, Mar, 1, 0}; //UTC | ||||
|   TimeChangeRule tcrStandard = tcrDaylight;            //UTC | ||||
|  | ||||
|   switch (currentTimezone) { | ||||
|     case TZ_UK : { | ||||
|       tcrDaylight = {Last, Sun, Mar, 1, 60};      //British Summer Time | ||||
|       tcrStandard = {Last, Sun, Oct, 2, 0};       //Standard Time | ||||
|       break; | ||||
|     } | ||||
|     case TZ_EUROPE_CENTRAL : { | ||||
|       tcrDaylight = {Last, Sun, Mar, 2, 120};     //Central European Summer Time | ||||
|       tcrStandard = {Last, Sun, Oct, 3, 60};      //Central European Standard Time | ||||
|       break; | ||||
|     } | ||||
|     case TZ_EUROPE_EASTERN : { | ||||
|       tcrDaylight = {Last, Sun, Mar, 3, 180};     //East European Summer Time | ||||
|       tcrStandard = {Last, Sun, Oct, 4, 120};     //East European Standard Time | ||||
|       break; | ||||
|     } | ||||
|     case TZ_US_EASTERN : { | ||||
|       tcrDaylight = {Second, Sun, Mar, 2, -240};  //EDT = UTC - 4 hours | ||||
|       tcrStandard = {First,  Sun, Nov, 2, -300};  //EST = UTC - 5 hours | ||||
|       break; | ||||
|     } | ||||
|     case TZ_US_CENTRAL : { | ||||
|       tcrDaylight = {Second, Sun, Mar, 2, -300};  //CDT = UTC - 5 hours | ||||
|       tcrStandard = {First,  Sun, Nov, 2, -360};  //CST = UTC - 6 hours | ||||
|       break; | ||||
|     } | ||||
|     case TZ_US_MOUNTAIN : { | ||||
|       tcrDaylight = {Second, Sun, Mar, 2, -360};  //MDT = UTC - 6 hours | ||||
|       tcrStandard = {First,  Sun, Nov, 2, -420};  //MST = UTC - 7 hours | ||||
|       break; | ||||
|     } | ||||
|     case TZ_US_ARIZONA : { | ||||
|       tcrDaylight = {First,  Sun, Nov, 2, -420};  //MST = UTC - 7 hours | ||||
|       tcrStandard = {First,  Sun, Nov, 2, -420};  //MST = UTC - 7 hours | ||||
|       break; | ||||
|     } | ||||
|     case TZ_US_PACIFIC : { | ||||
|       tcrDaylight = {Second, Sun, Mar, 2, -420};  //PDT = UTC - 7 hours | ||||
|       tcrStandard = {First,  Sun, Nov, 2, -480};  //PST = UTC - 8 hours | ||||
|       break; | ||||
|     } | ||||
|     case TZ_CHINA : { | ||||
|       tcrDaylight = {Last, Sun, Mar, 1, 480};     //CST = UTC + 8 hours | ||||
|       tcrStandard = tcrDaylight; | ||||
|       break; | ||||
|     } | ||||
|     case TZ_JAPAN : { | ||||
|       tcrDaylight = {Last, Sun, Mar, 1, 540};     //JST = UTC + 9 hours | ||||
|       tcrStandard = tcrDaylight; | ||||
|       break; | ||||
|     } | ||||
|     case TZ_AUSTRALIA_EASTERN : { | ||||
|       tcrDaylight = {Second, Sun, Oct, 2, 660};   //AEDT = UTC + 11 hours | ||||
|       tcrStandard = {First,  Sun, Apr, 3, 600};   //AEST = UTC + 10 hours | ||||
|       break; | ||||
|     } | ||||
|     case TZ_NEW_ZEALAND : { | ||||
|       tcrDaylight = {Second, Sun, Sep, 2, 780};   //NZDT = UTC + 13 hours | ||||
|       tcrStandard = {First,  Sun, Apr, 3, 720};   //NZST = UTC + 12 hours | ||||
|       break; | ||||
|     } | ||||
|     case TZ_NORTH_KOREA : { | ||||
|       tcrDaylight = {Last, Sun, Mar, 1, 510};     //Pyongyang Time = UTC + 8.5 hours | ||||
|       tcrStandard = tcrDaylight; | ||||
|       break; | ||||
|     } | ||||
|     case TZ_INDIA : { | ||||
|       tcrDaylight = {Last, Sun, Mar, 1, 330};     //India Standard Time = UTC + 5.5 hours | ||||
|       tcrStandard = tcrDaylight; | ||||
|       break; | ||||
|     } | ||||
|     case TZ_SASKACHEWAN : { | ||||
|       tcrDaylight = {First,  Sun, Nov, 2, -360};  //CST = UTC - 6 hours | ||||
|       tcrStandard = tcrDaylight; | ||||
|       break; | ||||
|     } | ||||
|     case TZ_AUSTRALIA_NORTHERN : { | ||||
|       tcrStandard = {First, Sun, Apr, 3, 570};   //ACST = UTC + 9.5 hours | ||||
|       tcrStandard = tcrDaylight; | ||||
|       break; | ||||
|     } | ||||
|     case TZ_AUSTRALIA_SOUTHERN : { | ||||
|       tcrDaylight = {First, Sun, Oct, 2, 630};   //ACDT = UTC + 10.5 hours | ||||
|       tcrStandard = {First, Sun, Apr, 3, 570};   //ACST = UTC + 9.5 hours | ||||
|       break; | ||||
|     } | ||||
|     case TZ_HAWAII : { | ||||
|       tcrDaylight = {Last, Sun, Mar, 1, -600};   //HST =  UTC - 10 hours | ||||
|       tcrStandard = tcrDaylight; | ||||
|       break; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   tzCurrent = currentTimezone; | ||||
|  | ||||
|   tz = new Timezone(tcrDaylight, tcrStandard); | ||||
| } | ||||
|  | ||||
| void handleNetworkTime() | ||||
| { | ||||
|   if (ntpEnabled && ntpConnected && millis() - ntpLastSyncTime > 50000000L && WLED_CONNECTED) | ||||
|   { | ||||
|     if (millis() - ntpPacketSentTime > 10000) | ||||
|     { | ||||
|       sendNTPPacket(); | ||||
|       ntpPacketSentTime = millis(); | ||||
|     } | ||||
|     if (checkNTPResponse()) | ||||
|     { | ||||
|       ntpLastSyncTime = millis(); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| void sendNTPPacket() | ||||
| { | ||||
|   if (!ntpServerIP.fromString(ntpServerName)) //see if server is IP or domain | ||||
|   { | ||||
|     #ifdef ESP8266 | ||||
|     WiFi.hostByName(ntpServerName, ntpServerIP, 750); | ||||
|     #else | ||||
|     WiFi.hostByName(ntpServerName, ntpServerIP); | ||||
|     #endif | ||||
|   } | ||||
|  | ||||
|   DEBUG_PRINTLN(F("send NTP")); | ||||
|   byte pbuf[NTP_PACKET_SIZE]; | ||||
|   memset(pbuf, 0, NTP_PACKET_SIZE); | ||||
|  | ||||
|   pbuf[0] = 0b11100011;   // LI, Version, Mode | ||||
|   pbuf[1] = 0;     // Stratum, or type of clock | ||||
|   pbuf[2] = 6;     // Polling Interval | ||||
|   pbuf[3] = 0xEC;  // Peer Clock Precision | ||||
|   // 8 bytes of zero for Root Delay & Root Dispersion | ||||
|   pbuf[12]  = 49; | ||||
|   pbuf[13]  = 0x4E; | ||||
|   pbuf[14]  = 49; | ||||
|   pbuf[15]  = 52; | ||||
|  | ||||
|   ntpUdp.beginPacket(ntpServerIP, 123); //NTP requests are to port 123 | ||||
|   ntpUdp.write(pbuf, NTP_PACKET_SIZE); | ||||
|   ntpUdp.endPacket(); | ||||
| } | ||||
|  | ||||
| bool checkNTPResponse() | ||||
| { | ||||
|   int cb = ntpUdp.parsePacket(); | ||||
|   if (cb) { | ||||
|     DEBUG_PRINT(F("NTP recv, l=")); | ||||
|     DEBUG_PRINTLN(cb); | ||||
|     byte pbuf[NTP_PACKET_SIZE]; | ||||
|     ntpUdp.read(pbuf, NTP_PACKET_SIZE); // read the packet into the buffer | ||||
|  | ||||
|     unsigned long highWord = word(pbuf[40], pbuf[41]); | ||||
|     unsigned long lowWord = word(pbuf[42], pbuf[43]); | ||||
|     if (highWord == 0 && lowWord == 0) return false; | ||||
|      | ||||
|     unsigned long secsSince1900 = highWord << 16 | lowWord; | ||||
|   | ||||
|     DEBUG_PRINT(F("Unix time = ")); | ||||
|     unsigned long epoch = secsSince1900 - 2208988799UL; //subtract 70 years -1sec (on avg. more precision) | ||||
|     setTime(epoch); | ||||
|     DEBUG_PRINTLN(epoch); | ||||
|     if (countdownTime - now() > 0) countdownOverTriggered = false; | ||||
|     return true; | ||||
|   } | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| void updateLocalTime() | ||||
| { | ||||
|   if (currentTimezone != tzCurrent) updateTimezone(); | ||||
|   unsigned long tmc = now()+ utcOffsetSecs; | ||||
|   localTime = tz->toLocal(tmc); | ||||
| } | ||||
|  | ||||
| void getTimeString(char* out) | ||||
| { | ||||
|   updateLocalTime(); | ||||
|   byte hr = hour(localTime); | ||||
|   if (useAMPM) | ||||
|   { | ||||
|     if (hr > 11) hr -= 12; | ||||
|     if (hr == 0) hr  = 12; | ||||
|   } | ||||
|   sprintf(out,"%i-%i-%i, %i:%s%i:%s%i",year(localTime), month(localTime), day(localTime),  | ||||
|                                        hr,(minute(localTime)<10)?"0":"",minute(localTime), | ||||
|                                        (second(localTime)<10)?"0":"",second(localTime)); | ||||
|   if (useAMPM) | ||||
|   { | ||||
|     strcat(out,(hour(localTime) > 11)? " PM":" AM"); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void setCountdown() | ||||
| { | ||||
|   if (currentTimezone != tzCurrent) updateTimezone(); | ||||
|   countdownTime = tz->toUTC(getUnixTime(countdownHour, countdownMin, countdownSec, countdownDay, countdownMonth, countdownYear)); | ||||
|   if (countdownTime - now() > 0) countdownOverTriggered = false; | ||||
| } | ||||
|  | ||||
| //returns true if countdown just over | ||||
| bool checkCountdown() | ||||
| { | ||||
|   unsigned long n = now(); | ||||
|   if (countdownMode) localTime = countdownTime - n + utcOffsetSecs; | ||||
|   if (n > countdownTime) { | ||||
|     if (countdownMode) localTime = n - countdownTime + utcOffsetSecs; | ||||
|     if (!countdownOverTriggered) | ||||
|     { | ||||
|       if (macroCountdown != 0) applyPreset(macroCountdown); | ||||
|       countdownOverTriggered = true; | ||||
|       return true; | ||||
|     } | ||||
|   } | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| byte weekdayMondayFirst() | ||||
| { | ||||
|   byte wd = weekday(localTime) -1; | ||||
|   if (wd == 0) wd = 7; | ||||
|   return wd; | ||||
| } | ||||
|  | ||||
| void checkTimers() | ||||
| { | ||||
|   if (lastTimerMinute != minute(localTime)) //only check once a new minute begins | ||||
|   { | ||||
|     lastTimerMinute = minute(localTime); | ||||
|     for (uint8_t i = 0; i < 8; i++) | ||||
|     { | ||||
|       if (timerMacro[i] != 0 | ||||
|           && (timerHours[i] == hour(localTime) || timerHours[i] == 24) //if hour is set to 24, activate every hour  | ||||
|           && timerMinutes[i] == minute(localTime) | ||||
|           && (timerWeekday[i] & 0x01) //timer is enabled | ||||
|           && timerWeekday[i] >> weekdayMondayFirst() & 0x01) //timer should activate at current day of week | ||||
|       { | ||||
|         applyPreset(timerMacro[i]); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,6 +1,142 @@ | ||||
| #include "wled.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * Used to draw clock overlays over the strip | ||||
|  */ | ||||
|   | ||||
| void initCronixie() | ||||
| { | ||||
|   if (overlayCurrent == 3 && !cronixieInit) | ||||
|   { | ||||
|     setCronixie(); | ||||
|     strip.getSegment(0).grouping = 10; //10 LEDs per digit
 | ||||
|     cronixieInit = true; | ||||
|   } else if (cronixieInit && overlayCurrent != 3) | ||||
|   { | ||||
|     strip.getSegment(0).grouping = 1; | ||||
|     cronixieInit = false;  | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void handleOverlays() | ||||
| { | ||||
|   if (millis() - overlayRefreshedTime > overlayRefreshMs) | ||||
|   { | ||||
|     initCronixie(); | ||||
|     updateLocalTime(); | ||||
|     checkTimers(); | ||||
|     checkCountdown(); | ||||
|     if (overlayCurrent == 3) _overlayCronixie();//Diamex cronixie clock kit
 | ||||
|     overlayRefreshedTime = millis(); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void _overlayAnalogClock() | ||||
| { | ||||
|   int overlaySize = overlayMax - overlayMin +1; | ||||
|   if (countdownMode) | ||||
|   { | ||||
|     _overlayAnalogCountdown(); return; | ||||
|   } | ||||
|   double hourP = ((double)(hour(localTime)%12))/12; | ||||
|   double minuteP = ((double)minute(localTime))/60; | ||||
|   hourP = hourP + minuteP/12; | ||||
|   double secondP = ((double)second(localTime))/60; | ||||
|   int hourPixel = floor(analogClock12pixel + overlaySize*hourP); | ||||
|   if (hourPixel > overlayMax) hourPixel = overlayMin -1 + hourPixel - overlayMax; | ||||
|   int minutePixel = floor(analogClock12pixel + overlaySize*minuteP); | ||||
|   if (minutePixel > overlayMax) minutePixel = overlayMin -1 + minutePixel - overlayMax;  | ||||
|   int secondPixel = floor(analogClock12pixel + overlaySize*secondP); | ||||
|   if (secondPixel > overlayMax) secondPixel = overlayMin -1 + secondPixel - overlayMax; | ||||
|   if (analogClockSecondsTrail) | ||||
|   { | ||||
|     if (secondPixel < analogClock12pixel) | ||||
|     { | ||||
|       strip.setRange(analogClock12pixel, overlayMax, 0xFF0000); | ||||
|       strip.setRange(overlayMin, secondPixel, 0xFF0000); | ||||
|     } else | ||||
|     { | ||||
|       strip.setRange(analogClock12pixel, secondPixel, 0xFF0000); | ||||
|     } | ||||
|   } | ||||
|   if (analogClock5MinuteMarks) | ||||
|   { | ||||
|     int pix; | ||||
|     for (int i = 0; i <= 12; i++) | ||||
|     { | ||||
|       pix = analogClock12pixel + round((overlaySize / 12.0) *i); | ||||
|       if (pix > overlayMax) pix -= overlaySize; | ||||
|       strip.setPixelColor(pix, 0x00FFAA); | ||||
|     } | ||||
|   } | ||||
|   if (!analogClockSecondsTrail) strip.setPixelColor(secondPixel, 0xFF0000); | ||||
|   strip.setPixelColor(minutePixel, 0x00FF00); | ||||
|   strip.setPixelColor(hourPixel, 0x0000FF); | ||||
|   overlayRefreshMs = 998; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void _overlayAnalogCountdown() | ||||
| { | ||||
|   if (now() < countdownTime) | ||||
|   { | ||||
|     long diff = countdownTime - now(); | ||||
|     double pval = 60; | ||||
|     if (diff > 31557600L) //display in years if more than 365 days
 | ||||
|     { | ||||
|       pval = 315576000L; //10 years
 | ||||
|     } else if (diff > 2592000L) //display in months if more than a month
 | ||||
|     { | ||||
|       pval = 31557600L; //1 year
 | ||||
|     } else if (diff > 604800) //display in weeks if more than a week
 | ||||
|     { | ||||
|       pval = 2592000L; //1 month
 | ||||
|     } else if (diff > 86400) //display in days if more than 24 hours
 | ||||
|     { | ||||
|       pval = 604800; //1 week
 | ||||
|     } else if (diff > 3600) //display in hours if more than 60 minutes
 | ||||
|     { | ||||
|       pval = 86400; //1 day
 | ||||
|     } else if (diff > 60) //display in minutes if more than 60 seconds
 | ||||
|     { | ||||
|       pval = 3600; //1 hour
 | ||||
|     } | ||||
|     int overlaySize = overlayMax - overlayMin +1; | ||||
|     double perc = (pval-(double)diff)/pval; | ||||
|     if (perc > 1.0) perc = 1.0; | ||||
|     byte pixelCnt = perc*overlaySize; | ||||
|     if (analogClock12pixel + pixelCnt > overlayMax) | ||||
|     { | ||||
|       strip.setRange(analogClock12pixel, overlayMax, ((uint32_t)colSec[3] << 24)| ((uint32_t)colSec[0] << 16) | ((uint32_t)colSec[1] << 8) | colSec[2]); | ||||
|       strip.setRange(overlayMin, overlayMin +pixelCnt -(1+ overlayMax -analogClock12pixel), ((uint32_t)colSec[3] << 24)| ((uint32_t)colSec[0] << 16) | ((uint32_t)colSec[1] << 8) | colSec[2]); | ||||
|     } else | ||||
|     { | ||||
|       strip.setRange(analogClock12pixel, analogClock12pixel + pixelCnt, ((uint32_t)colSec[3] << 24)| ((uint32_t)colSec[0] << 16) | ((uint32_t)colSec[1] << 8) | colSec[2]); | ||||
|     } | ||||
|   } | ||||
|   overlayRefreshMs = 998; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void handleOverlayDraw() { | ||||
|   if (!overlayCurrent) return; | ||||
|   switch (overlayCurrent) | ||||
|   { | ||||
|     case 1: _overlayAnalogClock(); break; | ||||
|     case 3: _drawOverlayCronixie(); break; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /*
 | ||||
|  * Support for the Cronixie clock | ||||
|  */ | ||||
|   | ||||
| #ifndef WLED_DISABLE_CRONIXIE | ||||
| byte _digitOut[6] = {10,10,10,10,10,10}; | ||||
|   | ||||
| byte getSameCodeLength(char code, int index, char const cronixieDisplay[]) | ||||
| { | ||||
|   byte counter = 0; | ||||
| @@ -19,7 +155,6 @@ byte getSameCodeLength(char code, int index, char const cronixieDisplay[]) | ||||
| 
 | ||||
| void setCronixie() | ||||
| { | ||||
|   #ifndef WLED_DISABLE_CRONIXIE | ||||
|   /*
 | ||||
|    * digit purpose index | ||||
|    * 0-9 | 0-9 (incl. random) | ||||
| @@ -97,8 +232,8 @@ void setCronixie() | ||||
|       case '-': dP[i] = 11; break;  | ||||
|       case 'r': dP[i] = random(1,7); break; //random btw. 1-6
 | ||||
|       case 'R': dP[i] = random(0,10); break; //random btw. 0-9
 | ||||
|       case 't': break; //Test upw.
 | ||||
|       case 'T': break; //Test dnw.
 | ||||
|       //case 't': break; //Test upw.
 | ||||
|       //case 'T': break; //Test dnw.
 | ||||
|       case 'b': dP[i] = 14 + getSameCodeLength('b',i,cronixieDisplay); i = i+dP[i]-14; break;  | ||||
|       case 'B': dP[i] = 14 + getSameCodeLength('B',i,cronixieDisplay); i = i+dP[i]-14; break; | ||||
|       case 'h': dP[i] = 70 + getSameCodeLength('h',i,cronixieDisplay); i = i+dP[i]-70; break; | ||||
| @@ -113,8 +248,8 @@ void setCronixie() | ||||
|       case 'y': dP[i] = 86 + getSameCodeLength('y',i,cronixieDisplay); i = i+dP[i]-86; break;  | ||||
|       case 'I': dP[i] = 39 + getSameCodeLength('I',i,cronixieDisplay); i = i+dP[i]-39; break;  //Month. Don't ask me why month and minute both start with M.
 | ||||
|       case 'i': dP[i] = 89 + getSameCodeLength('i',i,cronixieDisplay); i = i+dP[i]-89; break;  | ||||
|       case 'W': break; | ||||
|       case 'w': break; | ||||
|       //case 'W': break;
 | ||||
|       //case 'w': break;
 | ||||
|       case 'D': dP[i] = 43 + getSameCodeLength('D',i,cronixieDisplay); i = i+dP[i]-43; break; | ||||
|       case 'd': dP[i] = 93 + getSameCodeLength('d',i,cronixieDisplay); i = i+dP[i]-93; break; | ||||
|       case '0': dP[i] = 0; break; | ||||
| @@ -127,8 +262,8 @@ void setCronixie() | ||||
|       case '7': dP[i] = 7; break; | ||||
|       case '8': dP[i] = 8; break; | ||||
|       case '9': dP[i] = 9; break; | ||||
|       case 'V': break; //user var0
 | ||||
|       case 'v': break; //user var1
 | ||||
|       //case 'V': break; //user var0
 | ||||
|       //case 'v': break; //user var1
 | ||||
|     } | ||||
|   } | ||||
|   DEBUG_PRINT("result "); | ||||
| @@ -140,19 +275,17 @@ void setCronixie() | ||||
|   DEBUG_PRINTLN((int)dP[5]); | ||||
| 
 | ||||
|   _overlayCronixie(); //refresh
 | ||||
|   #endif | ||||
| } | ||||
| 
 | ||||
| void _overlayCronixie() | ||||
| { | ||||
|   #ifndef WLED_DISABLE_CRONIXIE | ||||
|   byte h = hour(local); | ||||
|   byte h = hour(localTime); | ||||
|   byte h0 = h; | ||||
|   byte m = minute(local); | ||||
|   byte s = second(local); | ||||
|   byte d = day(local); | ||||
|   byte mi = month(local); | ||||
|   int y = year(local); | ||||
|   byte m = minute(localTime); | ||||
|   byte s = second(localTime); | ||||
|   byte d = day(localTime); | ||||
|   byte mi = month(localTime); | ||||
|   int y = year(localTime); | ||||
|   //this has to be changed in time for 22nd century
 | ||||
|   y -= 2000; if (y<0) y += 30; //makes countdown work
 | ||||
| 
 | ||||
| @@ -161,7 +294,6 @@ void _overlayCronixie() | ||||
|     if (h>12) h-=12; | ||||
|     else if (h==0) h+=12; | ||||
|   } | ||||
|   byte _digitOut[]{10,10,10,10,10,10}; | ||||
|   for (int i = 0; i < 6; i++) | ||||
|   { | ||||
|     if (dP[i] < 12) _digitOut[i] = dP[i]; | ||||
| @@ -178,14 +310,14 @@ void _overlayCronixie() | ||||
|           case 24: _digitOut[i] = m/10; break; //M
 | ||||
|           case 30: _digitOut[i] = s/10; break; //S
 | ||||
|            | ||||
|           case 43: _digitOut[i] = weekday(local); _digitOut[i]--; if (_digitOut[i]<1) _digitOut[i]= 7; break; //D
 | ||||
|           case 43: _digitOut[i] = weekday(localTime); _digitOut[i]--; if (_digitOut[i]<1) _digitOut[i]= 7; break; //D
 | ||||
|           case 44: _digitOut[i] = d/10; _digitOut[i+1] = d- _digitOut[i]*10; i++; break; //DD
 | ||||
|           case 40: _digitOut[i] = mi/10; _digitOut[i+1] = mi- _digitOut[i]*10; i++; break; //II
 | ||||
|           case 37: _digitOut[i] = y/10; _digitOut[i+1] = y- _digitOut[i]*10; i++; break; //YY
 | ||||
|           case 39: _digitOut[i] = 2; _digitOut[i+1] = 0; _digitOut[i+2] = y/10; _digitOut[i+3] = y- _digitOut[i+2]*10; i+=3; break; //YYYY
 | ||||
|            | ||||
|           case 16: _digitOut[i+2] = ((h0/3)&1)?1:0; i++; //BBB (BBBB NI)
 | ||||
|           case 15: _digitOut[i+1] = (h0>17 || (h0>5 && h0<12))?1:0; i++; //BB
 | ||||
|           //case 16: _digitOut[i+2] = ((h0/3)&1)?1:0; i++; //BBB (BBBB NI)
 | ||||
|           //case 15: _digitOut[i+1] = (h0>17 || (h0>5 && h0<12))?1:0; i++; //BB
 | ||||
|           case 14: _digitOut[i] = (h0>11)?1:0; break; //B
 | ||||
|         } | ||||
|       } else | ||||
| @@ -195,11 +327,11 @@ void _overlayCronixie() | ||||
|           case 71: _digitOut[i] = h/10; _digitOut[i+1] = h- _digitOut[i]*10; if(_digitOut[i] == 0) _digitOut[i]=10; i++; break; //hh
 | ||||
|           case 75: _digitOut[i] = m/10; _digitOut[i+1] = m- _digitOut[i]*10; if(_digitOut[i] == 0) _digitOut[i]=10; i++; break; //mm
 | ||||
|           case 81: _digitOut[i] = s/10; _digitOut[i+1] = s- _digitOut[i]*10; if(_digitOut[i] == 0) _digitOut[i]=10; i++; break; //ss
 | ||||
|           case 66: _digitOut[i+2] = ((h0/3)&1)?1:10; i++; //bbb (bbbb NI)
 | ||||
|           case 65: _digitOut[i+1] = (h0>17 || (h0>5 && h0<12))?1:10; i++; //bb
 | ||||
|           //case 66: _digitOut[i+2] = ((h0/3)&1)?1:10; i++; //bbb (bbbb NI)
 | ||||
|           //case 65: _digitOut[i+1] = (h0>17 || (h0>5 && h0<12))?1:10; i++; //bb
 | ||||
|           case 64: _digitOut[i] = (h0>11)?1:10; break; //b
 | ||||
| 
 | ||||
|           case 93: _digitOut[i] = weekday(local); _digitOut[i]--; if (_digitOut[i]<1) _digitOut[i]= 7; break; //d
 | ||||
|           case 93: _digitOut[i] = weekday(localTime); _digitOut[i]--; if (_digitOut[i]<1) _digitOut[i]= 7; break; //d
 | ||||
|           case 94: _digitOut[i] = d/10; _digitOut[i+1] = d- _digitOut[i]*10; if(_digitOut[i] == 0) _digitOut[i]=10; i++; break; //dd
 | ||||
|           case 90: _digitOut[i] = mi/10; _digitOut[i+1] = mi- _digitOut[i]*10; if(_digitOut[i] == 0) _digitOut[i]=10; i++; break; //ii
 | ||||
|           case 87: _digitOut[i] = y/10; _digitOut[i+1] = y- _digitOut[i]*10; i++; break; //yy
 | ||||
| @@ -208,7 +340,37 @@ void _overlayCronixie() | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   strip.setCronixieDigits(_digitOut); | ||||
|   //strip.trigger(); //this has a drawback, no effects slower than RefreshMs. advantage: Quick update, not dependant on effect time
 | ||||
|   #endif | ||||
| } | ||||
| 
 | ||||
| void _drawOverlayCronixie() | ||||
| { | ||||
|   byte offsets[] = {5, 0, 6, 1, 7, 2, 8, 3, 9, 4}; | ||||
|    | ||||
|   for (uint16_t i = 0; i < 6; i++) | ||||
|   { | ||||
|     byte o = 10*i; | ||||
|     byte excl = 10; | ||||
|     if(_digitOut[i] < 10) excl = offsets[_digitOut[i]]; | ||||
|     excl += o; | ||||
|      | ||||
|     if (cronixieBacklight && _digitOut[i] <11) | ||||
|     { | ||||
|       uint32_t col = strip.gamma32(strip.getSegment(0).colors[1]); | ||||
|       for (uint16_t j=o; j< o+10; j++) { | ||||
|         if (j != excl) strip.setPixelColor(j, col); | ||||
|       } | ||||
|     } else | ||||
|     { | ||||
|       for (uint16_t j=o; j< o+10; j++) { | ||||
|         if (j != excl) strip.setPixelColor(j, 0); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| #else // WLED_DISABLE_CRONIXIE
 | ||||
| byte getSameCodeLength(char code, int index, char const cronixieDisplay[]) {} | ||||
| void setCronixie() {} | ||||
| void _overlayCronixie() {} | ||||
| void _drawOverlayCronixie() {} | ||||
| #endif | ||||
| @@ -13,7 +13,9 @@ | ||||
| #ifndef PalettesWLED_h | ||||
| #define PalettesWLED_h | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( ib_jul01_gp ) { | ||||
| #define GRADIENT_PALETTE_COUNT 43 | ||||
|  | ||||
| const byte ib_jul01_gp[] PROGMEM = { | ||||
|     0, 194,  1,  1, | ||||
|    94,   1, 29, 18, | ||||
|   132,  57,131, 28, | ||||
| @@ -24,7 +26,7 @@ DEFINE_GRADIENT_PALETTE( ib_jul01_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 20 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( es_vintage_57_gp ) { | ||||
| const byte es_vintage_57_gp[] PROGMEM = { | ||||
|     0,   2,  1,  1, | ||||
|    53,  18,  1,  0, | ||||
|   104,  69, 29,  1, | ||||
| @@ -37,7 +39,7 @@ DEFINE_GRADIENT_PALETTE( es_vintage_57_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 32 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( es_vintage_01_gp ) { | ||||
| const byte es_vintage_01_gp[] PROGMEM = { | ||||
|     0,   4,  1,  1, | ||||
|    51,  16,  0,  1, | ||||
|    76,  97,104,  3, | ||||
| @@ -53,7 +55,7 @@ DEFINE_GRADIENT_PALETTE( es_vintage_01_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 20 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( es_rivendell_15_gp ) { | ||||
| const byte es_rivendell_15_gp[] PROGMEM = { | ||||
|     0,   1, 14,  5, | ||||
|   101,  16, 36, 14, | ||||
|   165,  56, 68, 30, | ||||
| @@ -67,7 +69,7 @@ DEFINE_GRADIENT_PALETTE( es_rivendell_15_gp ) { | ||||
| // Size: 36 bytes of program space. | ||||
| // Edited to be brighter | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( rgi_15_gp ) { | ||||
| const byte rgi_15_gp[] PROGMEM = { | ||||
|     0,   4,  1, 70, | ||||
|    31,  55,  1, 30, | ||||
|    63, 255,  4,  7, | ||||
| @@ -84,7 +86,7 @@ DEFINE_GRADIENT_PALETTE( rgi_15_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 8 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( retro2_16_gp ) { | ||||
| const byte retro2_16_gp[] PROGMEM = { | ||||
|     0, 188,135,  1, | ||||
|   255,  46,  7,  1}; | ||||
|  | ||||
| @@ -94,7 +96,7 @@ DEFINE_GRADIENT_PALETTE( retro2_16_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 20 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Analogous_1_gp ) { | ||||
| const byte Analogous_1_gp[] PROGMEM = { | ||||
|     0,   3,  0,255, | ||||
|    63,  23,  0,255, | ||||
|   127,  67,  0,255, | ||||
| @@ -107,7 +109,7 @@ DEFINE_GRADIENT_PALETTE( Analogous_1_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 20 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( es_pinksplash_08_gp ) { | ||||
| const byte es_pinksplash_08_gp[] PROGMEM = { | ||||
|     0, 126, 11,255, | ||||
|   127, 197,  1, 22, | ||||
|   175, 210,157,172, | ||||
| @@ -120,7 +122,7 @@ DEFINE_GRADIENT_PALETTE( es_pinksplash_08_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 16 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( es_ocean_breeze_036_gp ) { | ||||
| const byte es_ocean_breeze_036_gp[] PROGMEM = { | ||||
|     0,   1,  6,  7, | ||||
|    89,   1, 99,111, | ||||
|   153, 144,209,255, | ||||
| @@ -132,7 +134,7 @@ DEFINE_GRADIENT_PALETTE( es_ocean_breeze_036_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 88 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( departure_gp ) { | ||||
| const byte departure_gp[] PROGMEM = { | ||||
|     0,   8,  3,  0, | ||||
|    42,  23,  7,  0, | ||||
|    63,  75, 38,  6, | ||||
| @@ -152,7 +154,7 @@ DEFINE_GRADIENT_PALETTE( departure_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 36 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( es_landscape_64_gp ) { | ||||
| const byte es_landscape_64_gp[] PROGMEM = { | ||||
|     0,   0,  0,  0, | ||||
|    37,   2, 25,  1, | ||||
|    76,  15,115,  5, | ||||
| @@ -169,7 +171,7 @@ DEFINE_GRADIENT_PALETTE( es_landscape_64_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 24 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( es_landscape_33_gp ) { | ||||
| const byte es_landscape_33_gp[] PROGMEM = { | ||||
|     0,   1,  5,  0, | ||||
|    19,  32, 23,  1, | ||||
|    38, 161, 55,  1, | ||||
| @@ -183,7 +185,7 @@ DEFINE_GRADIENT_PALETTE( es_landscape_33_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 28 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( rainbowsherbet_gp ) { | ||||
| const byte rainbowsherbet_gp[] PROGMEM = { | ||||
|     0, 255, 33,  4, | ||||
|    43, 255, 68, 25, | ||||
|    86, 255,  7, 25, | ||||
| @@ -198,7 +200,7 @@ DEFINE_GRADIENT_PALETTE( rainbowsherbet_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 24 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( gr65_hult_gp ) { | ||||
| const byte gr65_hult_gp[] PROGMEM = { | ||||
|     0, 247,176,247, | ||||
|    48, 255,136,255, | ||||
|    89, 220, 29,226, | ||||
| @@ -212,7 +214,7 @@ DEFINE_GRADIENT_PALETTE( gr65_hult_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 32 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( gr64_hult_gp ) { | ||||
| const byte gr64_hult_gp[] PROGMEM = { | ||||
|     0,   1,124,109, | ||||
|    66,   1, 93, 79, | ||||
|   104,  52, 65,  1, | ||||
| @@ -228,7 +230,7 @@ DEFINE_GRADIENT_PALETTE( gr64_hult_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 28 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( GMT_drywet_gp ) { | ||||
| const byte GMT_drywet_gp[] PROGMEM = { | ||||
|     0,  47, 30,  2, | ||||
|    42, 213,147, 24, | ||||
|    84, 103,219, 52, | ||||
| @@ -243,7 +245,7 @@ DEFINE_GRADIENT_PALETTE( GMT_drywet_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 24 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( ib15_gp ) { | ||||
| const byte ib15_gp[] PROGMEM = { | ||||
|     0, 113, 91,147, | ||||
|    72, 157, 88, 78, | ||||
|    89, 208, 85, 33, | ||||
| @@ -257,7 +259,7 @@ DEFINE_GRADIENT_PALETTE( ib15_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 20 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Tertiary_01_gp ) { | ||||
| const byte Tertiary_01_gp[] PROGMEM = { | ||||
|     0,   0,  1,255, | ||||
|    63,   3, 68, 45, | ||||
|   127,  23,255,  0, | ||||
| @@ -270,7 +272,7 @@ DEFINE_GRADIENT_PALETTE( Tertiary_01_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 52 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( lava_gp ) { | ||||
| const byte lava_gp[] PROGMEM = { | ||||
|     0,   0,  0,  0, | ||||
|    46,  18,  0,  0, | ||||
|    96, 113,  0,  0, | ||||
| @@ -291,7 +293,7 @@ DEFINE_GRADIENT_PALETTE( lava_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 28 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( fierce_ice_gp ) { | ||||
| const byte fierce_ice_gp[] PROGMEM = { | ||||
|     0,   0,  0,  0, | ||||
|    59,   0,  9, 45, | ||||
|   119,   0, 38,255, | ||||
| @@ -306,7 +308,7 @@ DEFINE_GRADIENT_PALETTE( fierce_ice_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 44 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Colorfull_gp ) { | ||||
| const byte Colorfull_gp[] PROGMEM = { | ||||
|     0,  10, 85,  5, | ||||
|    25,  29,109, 18, | ||||
|    60,  59,138, 42, | ||||
| @@ -325,7 +327,7 @@ DEFINE_GRADIENT_PALETTE( Colorfull_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 44 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Pink_Purple_gp ) { | ||||
| const byte Pink_Purple_gp[] PROGMEM = { | ||||
|     0,  19,  2, 39, | ||||
|    25,  26,  4, 45, | ||||
|    51,  33,  6, 52, | ||||
| @@ -344,7 +346,7 @@ DEFINE_GRADIENT_PALETTE( Pink_Purple_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 28 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Sunset_Real_gp ) { | ||||
| const byte Sunset_Real_gp[] PROGMEM = { | ||||
|     0, 120,  0,  0, | ||||
|    22, 179, 22,  0, | ||||
|    51, 255,104,  0, | ||||
| @@ -359,7 +361,7 @@ DEFINE_GRADIENT_PALETTE( Sunset_Real_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 44 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Sunset_Yellow_gp ) { | ||||
| const byte Sunset_Yellow_gp[] PROGMEM = { | ||||
|     0,  10, 62,123, | ||||
|    36,  56,130,103, | ||||
|    87, 153,225, 85, | ||||
| @@ -378,7 +380,7 @@ DEFINE_GRADIENT_PALETTE( Sunset_Yellow_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 60 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Beech_gp ) { | ||||
| const byte Beech_gp[] PROGMEM = { | ||||
|     0, 255,252,214, | ||||
|    12, 255,252,214, | ||||
|    22, 255,252,214, | ||||
| @@ -401,7 +403,7 @@ DEFINE_GRADIENT_PALETTE( Beech_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 32 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Another_Sunset_gp ) { | ||||
| const byte Another_Sunset_gp[] PROGMEM = { | ||||
|     0, 110, 49, 11, | ||||
|    29,  55, 34, 10, | ||||
|    68,  22, 22,  9, | ||||
| @@ -420,7 +422,7 @@ DEFINE_GRADIENT_PALETTE( Another_Sunset_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 52 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( es_autumn_19_gp ) { | ||||
| const byte es_autumn_19_gp[] PROGMEM = { | ||||
|     0,  26,  1,  1, | ||||
|    51,  67,  4,  1, | ||||
|    84, 118, 14,  1, | ||||
| @@ -441,7 +443,7 @@ DEFINE_GRADIENT_PALETTE( es_autumn_19_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 28 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( BlacK_Blue_Magenta_White_gp ) { | ||||
| const byte BlacK_Blue_Magenta_White_gp[] PROGMEM = { | ||||
|     0,   0,  0,  0, | ||||
|    42,   0,  0, 45, | ||||
|    84,   0,  0,255, | ||||
| @@ -456,7 +458,7 @@ DEFINE_GRADIENT_PALETTE( BlacK_Blue_Magenta_White_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 20 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( BlacK_Magenta_Red_gp ) { | ||||
| const byte BlacK_Magenta_Red_gp[] PROGMEM = { | ||||
|     0,   0,  0,  0, | ||||
|    63,  42,  0, 45, | ||||
|   127, 255,  0,255, | ||||
| @@ -469,7 +471,7 @@ DEFINE_GRADIENT_PALETTE( BlacK_Magenta_Red_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 28 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( BlacK_Red_Magenta_Yellow_gp ) { | ||||
| const byte BlacK_Red_Magenta_Yellow_gp[] PROGMEM = { | ||||
|     0,   0,  0,  0, | ||||
|    42,  42,  0,  0, | ||||
|    84, 255,  0,  0, | ||||
| @@ -484,7 +486,7 @@ DEFINE_GRADIENT_PALETTE( BlacK_Red_Magenta_Yellow_gp ) { | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 20 bytes of program space. | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Blue_Cyan_Yellow_gp ) { | ||||
| const byte Blue_Cyan_Yellow_gp[] PROGMEM = { | ||||
|     0,   0,  0,255, | ||||
|    63,   0, 55,255, | ||||
|   127,   0,255,255, | ||||
| @@ -494,7 +496,7 @@ DEFINE_GRADIENT_PALETTE( Blue_Cyan_Yellow_gp ) { | ||||
|  | ||||
| //Custom palette by Aircoookie | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Orange_Teal_gp ) { | ||||
| const byte Orange_Teal_gp[] PROGMEM = { | ||||
|     0,   0,150, 92, | ||||
|    55,   0,150, 92, | ||||
|   200, 255, 72,  0, | ||||
| @@ -502,7 +504,7 @@ DEFINE_GRADIENT_PALETTE( Orange_Teal_gp ) { | ||||
|  | ||||
| //Custom palette by Aircoookie | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Tiamat_gp ) { | ||||
| const byte Tiamat_gp[] PROGMEM = { | ||||
|     0,   1,  2, 14, //gc | ||||
|    33,   2,  5, 35, //gc from 47, 61,126 | ||||
|   100,  13,135, 92, //gc from 88,242,247 | ||||
| @@ -517,7 +519,7 @@ DEFINE_GRADIENT_PALETTE( Tiamat_gp ) { | ||||
|    | ||||
| //Custom palette by Aircoookie | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( April_Night_gp ) { | ||||
| const byte April_Night_gp[] PROGMEM = { | ||||
|     0,   1,  5, 45, //deep blue | ||||
|    10,   1,  5, 45, | ||||
|    25,   5,169,175, //light blue | ||||
| @@ -536,7 +538,7 @@ DEFINE_GRADIENT_PALETTE( April_Night_gp ) { | ||||
|   244,   1,  5, 45, | ||||
|   255,   1,  5, 45}; | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Orangery_gp ) { | ||||
| const byte Orangery_gp[] PROGMEM = { | ||||
|     0, 255, 95, 23, | ||||
|    30, 255, 82,  0, | ||||
|    60, 223, 13,  8, | ||||
| @@ -548,7 +550,7 @@ DEFINE_GRADIENT_PALETTE( Orangery_gp ) { | ||||
|   255, 213, 37,  4}; | ||||
|  | ||||
| //inspired by Mark Kriegsman https://gist.github.com/kriegsman/756ea6dcae8e30845b5a | ||||
| DEFINE_GRADIENT_PALETTE( C9_gp ) { | ||||
| const byte C9_gp[] PROGMEM = { | ||||
|     0, 184,  4,  0, //red | ||||
|    60, 184,  4,  0, | ||||
|    65, 144, 44,  2, //amber | ||||
| @@ -558,33 +560,90 @@ DEFINE_GRADIENT_PALETTE( C9_gp ) { | ||||
|   195,   7,  7, 88, //blue | ||||
|   255,   7,  7, 88}; | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Sakura_gp ) { | ||||
| const byte Sakura_gp[] PROGMEM = { | ||||
|     0, 196, 19, 10, | ||||
|    65, 255, 69, 45, | ||||
|   130, 223, 45, 72, | ||||
|   195, 255, 82,103, | ||||
|   255, 223, 13, 17}; | ||||
|  | ||||
| DEFINE_GRADIENT_PALETTE( Aurora_gp ) { | ||||
| const byte Aurora_gp[] PROGMEM = { | ||||
|     0,   1,  5, 45, //deep blue | ||||
|    64,   0,200, 23, | ||||
|   128,   0,255,  0, //green | ||||
|   170,   0,243, 45, | ||||
|   200,   0,135,  7, | ||||
|   255,   1,  5, 45};//deep blue | ||||
|  | ||||
| const byte Atlantica_gp[] PROGMEM = { | ||||
|     0,   0, 28,112, //#001C70 | ||||
|    50,  32, 96,255, //#2060FF | ||||
|   100,   0,243, 45, | ||||
|   150,  12, 95, 82, //#0C5F52 | ||||
|   200,  25,190, 95, //#19BE5F | ||||
|   255,  40,170, 80};//#28AA50 | ||||
|  | ||||
|   const byte C9_2_gp[] PROGMEM = { | ||||
|     0,   6, 126,   2, //green | ||||
|    45,   6, 126,   2, | ||||
|    45,   4,  30, 114, //blue | ||||
|    90,   4,  30, 114, | ||||
|    90, 255,   5,   0, //red | ||||
|   135, 255,   5,   0, | ||||
|   135, 196,  57,   2, //amber | ||||
|   180, 196,  57,   2, | ||||
|   180, 137,  85,   2, //yellow | ||||
|   255, 137,  85,   2}; | ||||
|  | ||||
|   //C9, but brighter and with a less purple blue | ||||
|   const byte C9_new_gp[] PROGMEM = { | ||||
|     0, 255,   5,   0, //red | ||||
|    60, 255,   5,   0, | ||||
|    60, 196,  57,   2, //amber (start 61?) | ||||
|   120, 196,  57,   2,  | ||||
|   120,   6, 126,   2, //green (start 126?) | ||||
|   180,   6, 126,   2, | ||||
|   180,   4,  30, 114, //blue (start 191?) | ||||
|   255,   4,  30, 114}; | ||||
|    | ||||
| // Gradient palette "temperature_gp", originally from | ||||
| // http://soliton.vm.bytemark.co.uk/pub/cpt-city/arendal/tn/temperature.png.index.html | ||||
| // converted for FastLED with gammas (2.6, 2.2, 2.5) | ||||
| // Size: 144 bytes of program space.   | ||||
|  | ||||
| const byte temperature_gp[] PROGMEM = { | ||||
|     0,   1, 27,105, | ||||
|    14,   1, 40,127, | ||||
|    28,   1, 70,168, | ||||
|    42,   1, 92,197, | ||||
|    56,   1,119,221, | ||||
|    70,   3,130,151, | ||||
|    84,  23,156,149, | ||||
|    99,  67,182,112, | ||||
|   113, 121,201, 52, | ||||
|   127, 142,203, 11, | ||||
|   141, 224,223,  1, | ||||
|   155, 252,187,  2, | ||||
|   170, 247,147,  1, | ||||
|   184, 237, 87,  1, | ||||
|   198, 229, 43,  1, | ||||
|   226, 171,  2,  2, | ||||
|   240,  80,  3,  3, | ||||
|   255,  80,  3,  3}; | ||||
|  | ||||
|   const byte Aurora2[] PROGMEM = { | ||||
|       0,  17, 177,  13,    //Greenish | ||||
|      64, 121, 242,   5,    //Greenish | ||||
|     128,  25, 173, 121,    //Turquoise | ||||
|     192, 250,  77, 127,    //Pink | ||||
|     255, 171, 101, 221     //Purple | ||||
|   }; | ||||
|  | ||||
| // Single array of defined cpt-city color palettes. | ||||
| // This will let us programmatically choose one based on | ||||
| // a number, rather than having to activate each explicitly | ||||
| // by name every time. | ||||
| // Since it is const, this array could also be moved | ||||
| // into PROGMEM to save SRAM, but for simplicity of illustration | ||||
| // we'll keep it in a regular SRAM array. | ||||
| // | ||||
| // This list of color palettes acts as a "playlist"; you can | ||||
| // add or delete, or re-arrange as you wish. | ||||
| const TProgmemRGBGradientPalettePtr gGradientPalettes[] = { | ||||
| const byte* const gGradientPalettes[] PROGMEM = { | ||||
|   Sunset_Real_gp,               //13-00 Sunset | ||||
|   es_rivendell_15_gp,           //14-01 Rivendell | ||||
|   es_ocean_breeze_036_gp,       //15-02 Breeze | ||||
| @@ -623,11 +682,11 @@ const TProgmemRGBGradientPalettePtr gGradientPalettes[] = { | ||||
|   C9_gp,                        //48-35 C9 | ||||
|   Sakura_gp,                    //49-36 Sakura | ||||
|   Aurora_gp,                    //50-37 Aurora | ||||
|   Atlantica_gp,                 //51-38 Atlantica | ||||
|   C9_2_gp,                      //52-39 C9 2 | ||||
|   C9_new_gp,                    //53-40 C9 New | ||||
|   temperature_gp,               //54-41 Temperature | ||||
|   Aurora2                       //55-42 Aurora 2 | ||||
| }; | ||||
|  | ||||
|  | ||||
| // Count of how many cpt-city gradients are defined: | ||||
| const uint8_t gGradientPaletteCount = | ||||
|   sizeof( gGradientPalettes) / sizeof( TProgmemRGBGradientPalettePtr ); | ||||
|  | ||||
| #endif | ||||
|   | ||||
							
								
								
									
										91
									
								
								wled00/pin_manager.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								wled00/pin_manager.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | ||||
| #include "pin_manager.h" | ||||
| #include "wled.h" | ||||
|  | ||||
| void PinManagerClass::deallocatePin(byte gpio) | ||||
| { | ||||
|   if (!isPinOk(gpio, false)) return; | ||||
|    | ||||
|   byte by = gpio >> 3; | ||||
|   byte bi = gpio - 8*by; | ||||
|   bitWrite(pinAlloc[by], bi, false); | ||||
| } | ||||
|  | ||||
| bool PinManagerClass::allocatePin(byte gpio, bool output) | ||||
| { | ||||
|   if (!isPinOk(gpio, output)) return false; | ||||
|   if (isPinAllocated(gpio)) { | ||||
|     DEBUG_PRINT(F("Attempted duplicate allocation of pin ")); | ||||
|     DEBUG_PRINTLN(gpio); | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   byte by = gpio >> 3; | ||||
|   byte bi = gpio - 8*by; | ||||
|   bitWrite(pinAlloc[by], bi, true); | ||||
|    | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| bool PinManagerClass::isPinAllocated(byte gpio) | ||||
| { | ||||
|   if (!isPinOk(gpio, false)) return true; | ||||
|    | ||||
|   byte by = gpio >> 3; | ||||
|   byte bi = gpio - 8*by; | ||||
|   return bitRead(pinAlloc[by], bi); | ||||
| } | ||||
|  | ||||
| bool PinManagerClass::isPinOk(byte gpio, bool output) | ||||
| { | ||||
|   if (gpio <  6) return  true; | ||||
|   if (gpio < 12) return false; //SPI flash pins | ||||
|    | ||||
|   #ifdef ESP8266 | ||||
|   if (gpio < 17) return true; | ||||
|   #else //ESP32 | ||||
|   if (gpio < 34) return true; | ||||
|   if (gpio < 40 && !output) return true; //34-39 input only | ||||
|   #endif | ||||
|    | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| #ifdef ARDUINO_ARCH_ESP32 | ||||
| byte PinManagerClass::allocateLedc(byte channels) | ||||
| { | ||||
|   if (channels > 16 || channels == 0) return 255; | ||||
|   byte ca = 0; | ||||
|   for (byte i = 0; i < 16; i++) { | ||||
|     byte by = i >> 3; | ||||
|     byte bi = i - 8*by; | ||||
|     if (bitRead(ledcAlloc[by], bi)) { //found occupied pin | ||||
|       ca = 0; | ||||
|     } else { | ||||
|       ca++; | ||||
|     } | ||||
|     if (ca >= channels) { //enough free channels | ||||
|       byte in = (i + 1) - ca; | ||||
|       for (byte j = 0; j < ca; j++) { | ||||
|         byte b = in + j; | ||||
|         byte by = b >> 3; | ||||
|         byte bi = b - 8*by; | ||||
|         bitWrite(ledcAlloc[by], bi, true); | ||||
|       } | ||||
|       return in; | ||||
|     } | ||||
|   } | ||||
|   return 255; //not enough consecutive free LEDC channels | ||||
| } | ||||
|  | ||||
| void PinManagerClass::deallocateLedc(byte pos, byte channels) | ||||
| { | ||||
|   for (byte j = pos; j < pos + channels; j++) { | ||||
|     if (j > 16) return; | ||||
|     byte by = j >> 3; | ||||
|     byte bi = j - 8*by; | ||||
|     bitWrite(ledcAlloc[by], bi, false); | ||||
|   } | ||||
| } | ||||
| #endif | ||||
|  | ||||
| PinManagerClass pinManager = PinManagerClass(); | ||||
							
								
								
									
										29
									
								
								wled00/pin_manager.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								wled00/pin_manager.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| #ifndef WLED_PIN_MANAGER_H | ||||
| #define WLED_PIN_MANAGER_H | ||||
| /* | ||||
|  * Registers pins so there is no attempt for two interfaces to use the same pin | ||||
|  */ | ||||
| #include <Arduino.h> | ||||
|  | ||||
| class PinManagerClass { | ||||
|   private: | ||||
|   #ifdef ESP8266 | ||||
|   uint8_t pinAlloc[3] = {0x00, 0x00, 0x00}; //24bit, 1 bit per pin, we use first 17bits | ||||
|   #else | ||||
|   uint8_t pinAlloc[5] = {0x00, 0x00, 0x00, 0x00, 0x00}; //40bit, 1 bit per pin, we use all bits | ||||
|   uint8_t ledcAlloc[2] = {0x00, 0x00}; //16 LEDC channels | ||||
|   #endif | ||||
|  | ||||
|   public: | ||||
|   void deallocatePin(byte gpio); | ||||
|   bool allocatePin(byte gpio, bool output = true); | ||||
|   bool isPinAllocated(byte gpio); | ||||
|   bool isPinOk(byte gpio, bool output = true); | ||||
|   #ifdef ARDUINO_ARCH_ESP32 | ||||
|   byte allocateLedc(byte channels); | ||||
|   void deallocateLedc(byte pos, byte channels); | ||||
|   #endif | ||||
| }; | ||||
|  | ||||
| extern PinManagerClass pinManager; | ||||
| #endif | ||||
							
								
								
									
										131
									
								
								wled00/playlist.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										131
									
								
								wled00/playlist.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,131 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * Handles playlists, timed sequences of presets | ||||
|  */ | ||||
|  | ||||
| typedef struct PlaylistEntry { | ||||
|   uint8_t preset; | ||||
|   uint16_t dur; | ||||
|   uint16_t tr; | ||||
| } ple; | ||||
|  | ||||
| int8_t   playlistRepeat = 1; | ||||
| byte     playlistEndPreset = 0; | ||||
| byte    *playlistEntries = nullptr; | ||||
| byte     playlistLen; | ||||
| int8_t   playlistIndex = -1; | ||||
| uint16_t playlistEntryDur = 0; | ||||
|  | ||||
|  | ||||
| void shufflePlaylist() { | ||||
|   int currentIndex = playlistLen, randomIndex; | ||||
|  | ||||
|   PlaylistEntry temporaryValue, *entries = reinterpret_cast<PlaylistEntry*>(playlistEntries); | ||||
|  | ||||
|   // While there remain elements to shuffle... | ||||
|   while (currentIndex--) { | ||||
|     // Pick a random element... | ||||
|     randomIndex = random(0, currentIndex); | ||||
|     // And swap it with the current element. | ||||
|     temporaryValue = entries[currentIndex]; | ||||
|     entries[currentIndex] = entries[randomIndex]; | ||||
|     entries[randomIndex] = temporaryValue; | ||||
|   } | ||||
| } | ||||
|  | ||||
| void unloadPlaylist() { | ||||
|   if (playlistEntries != nullptr) { | ||||
|     delete[] playlistEntries; | ||||
|     playlistEntries = nullptr; | ||||
|   } | ||||
|   currentPlaylist = playlistIndex = -1; | ||||
|   playlistLen = playlistEntryDur = 0; | ||||
| } | ||||
|  | ||||
| void loadPlaylist(JsonObject playlistObj) { | ||||
|   unloadPlaylist(); | ||||
|    | ||||
|   JsonArray presets = playlistObj["ps"]; | ||||
|   playlistLen = presets.size(); | ||||
|   if (playlistLen == 0) return; | ||||
|   if (playlistLen > 100) playlistLen = 100; | ||||
|   uint16_t dataSize = sizeof(ple) * playlistLen; | ||||
|   playlistEntries = new byte[dataSize]; | ||||
|   PlaylistEntry* entries = reinterpret_cast<PlaylistEntry*>(playlistEntries); | ||||
|  | ||||
|   byte it = 0; | ||||
|   for (int ps : presets) { | ||||
|     if (it >= playlistLen) break; | ||||
|     entries[it].preset = ps; | ||||
|     it++; | ||||
|   } | ||||
|  | ||||
|   it = 0; | ||||
|   JsonArray durations = playlistObj["dur"]; | ||||
|   if (durations.isNull()) { | ||||
|     entries[0].dur = playlistObj["dur"] | 100; | ||||
|     it = 1; | ||||
|   } else { | ||||
|     for (int dur : durations) { | ||||
|       if (it >= playlistLen) break; | ||||
|       entries[it].dur = dur; | ||||
|       it++; | ||||
|     } | ||||
|   } | ||||
|   for (int i = it; i < playlistLen; i++) entries[i].dur = entries[it -1].dur; | ||||
|  | ||||
|   it = 0; | ||||
|   JsonArray tr = playlistObj["transition"]; | ||||
|   if (tr.isNull()) { | ||||
|     entries[0].tr = playlistObj["transition"] | (transitionDelay / 100); | ||||
|     it = 1; | ||||
|   } else { | ||||
|     for (int transition : tr) { | ||||
|       if (it >= playlistLen) break; | ||||
|       entries[it].tr = transition; | ||||
|       it++; | ||||
|     } | ||||
|   } | ||||
|   for (int i = it; i < playlistLen; i++) entries[i].tr = entries[it -1].tr; | ||||
|  | ||||
|   playlistRepeat = playlistObj[F("repeat")] | 0; | ||||
|   playlistEndPreset = playlistObj[F("end")] | 0; | ||||
|  | ||||
|   currentPlaylist = 0; //TODO here we need the preset ID where the playlist is saved | ||||
| } | ||||
|  | ||||
|  | ||||
| void handlePlaylist() { | ||||
|   if (currentPlaylist < 0 || playlistEntries == nullptr || presetCyclingEnabled) return; | ||||
|    | ||||
|   if (millis() - presetCycledTime > (100*playlistEntryDur)) { | ||||
|     presetCycledTime = millis(); | ||||
|     if (bri == 0 || nightlightActive) return; | ||||
|  | ||||
|     ++playlistIndex %= playlistLen; // -1 at 1st run (limit to playlistLen) | ||||
|  | ||||
|     if (!playlistRepeat && !playlistIndex) { //stop if repeat == 0 and restart of playlist | ||||
|       unloadPlaylist(); | ||||
|       if (playlistEndPreset) applyPreset(playlistEndPreset); | ||||
|       return; | ||||
|     } | ||||
|     // playlist roll-over | ||||
|     if (!playlistIndex) { | ||||
|       if (playlistRepeat > 0) {// playlistRepeat < 0 => endless loop with shuffling presets | ||||
|         playlistRepeat--; // decrease repeat count on each index reset | ||||
|       } else { | ||||
|         shufflePlaylist();  // shuffle playlist and start over | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     PlaylistEntry* entries = reinterpret_cast<PlaylistEntry*>(playlistEntries); | ||||
|  | ||||
|     jsonTransitionOnce = true; | ||||
|     transitionDelayTemp = entries[playlistIndex].tr * 100; | ||||
|  | ||||
|     applyPreset(entries[playlistIndex].preset); | ||||
|     playlistEntryDur = entries[playlistIndex].dur; | ||||
|     if (playlistEntryDur == 0) playlistEntryDur = 10; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										81
									
								
								wled00/presets.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								wled00/presets.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,81 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * Methods to handle saving and loading presets to/from the filesystem | ||||
|  */ | ||||
|  | ||||
| bool applyPreset(byte index) | ||||
| { | ||||
|   if (index == 0) return false; | ||||
|   if (fileDoc) { | ||||
|     errorFlag = readObjectFromFileUsingId("/presets.json", index, fileDoc) ? ERR_NONE : ERR_FS_PLOAD; | ||||
|     JsonObject fdo = fileDoc->as<JsonObject>(); | ||||
|     if (fdo["ps"] == index) fdo.remove("ps"); //remove load request for same presets to prevent recursive crash | ||||
|     #ifdef WLED_DEBUG_FS | ||||
|       serializeJson(*fileDoc, Serial); | ||||
|     #endif | ||||
|     deserializeState(fdo); | ||||
|   } else { | ||||
|     DEBUGFS_PRINTLN(F("Make read buf")); | ||||
|     DynamicJsonDocument fDoc(JSON_BUFFER_SIZE); | ||||
|     errorFlag = readObjectFromFileUsingId("/presets.json", index, &fDoc) ? ERR_NONE : ERR_FS_PLOAD; | ||||
|     JsonObject fdo = fDoc.as<JsonObject>(); | ||||
|     if (fdo["ps"] == index) fdo.remove("ps"); | ||||
|     #ifdef WLED_DEBUG_FS | ||||
|       serializeJson(fDoc, Serial); | ||||
|     #endif | ||||
|     deserializeState(fdo); | ||||
|   } | ||||
|  | ||||
|   if (!errorFlag) { | ||||
|     currentPreset = index; | ||||
|     isPreset = true; | ||||
|     return true; | ||||
|   } | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| void savePreset(byte index, bool persist, const char* pname, JsonObject saveobj) | ||||
| { | ||||
|   if (index == 0 || index > 250) return; | ||||
|   bool docAlloc = (fileDoc != nullptr); | ||||
|   JsonObject sObj = saveobj; | ||||
|  | ||||
|   if (!docAlloc) { | ||||
|     DEBUGFS_PRINTLN(F("Allocating saving buffer")); | ||||
|     DynamicJsonDocument lDoc(JSON_BUFFER_SIZE); | ||||
|     sObj = lDoc.to<JsonObject>(); | ||||
|     if (pname) sObj["n"] = pname; | ||||
|     DEBUGFS_PRINTLN(F("Save current state")); | ||||
|     serializeState(sObj, true); | ||||
|     currentPreset = index; | ||||
|  | ||||
|     writeObjectToFileUsingId("/presets.json", index, &lDoc); | ||||
|   } else { //from JSON API | ||||
|     DEBUGFS_PRINTLN(F("Reuse recv buffer")); | ||||
|     sObj.remove(F("psave")); | ||||
|     sObj.remove(F("v")); | ||||
|  | ||||
|     if (!sObj["o"]) { | ||||
|       DEBUGFS_PRINTLN(F("Save current state")); | ||||
|       serializeState(sObj, true, sObj["ib"], sObj["sb"]); | ||||
|       currentPreset = index; | ||||
|     } | ||||
|     sObj.remove("o"); | ||||
|     sObj.remove("ib"); | ||||
|     sObj.remove("sb"); | ||||
|     sObj.remove(F("error")); | ||||
|     sObj.remove(F("time")); | ||||
|  | ||||
|     writeObjectToFileUsingId("/presets.json", index, fileDoc); | ||||
|   } | ||||
|   presetsModifiedTime = now(); //unix time | ||||
|   updateFSInfo(); | ||||
| } | ||||
|  | ||||
| void deletePreset(byte index) { | ||||
|   StaticJsonDocument<24> empty; | ||||
|   writeObjectToFileUsingId("/presets.json", index, &empty); | ||||
|   presetsModifiedTime = now(); //unix time | ||||
|   updateFSInfo(); | ||||
| } | ||||
							
								
								
									
										846
									
								
								wled00/set.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										846
									
								
								wled00/set.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,846 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * Receives client input | ||||
|  */ | ||||
|  | ||||
| void _setRandomColor(bool _sec,bool fromButton) | ||||
| { | ||||
|   lastRandomIndex = strip.get_random_wheel_index(lastRandomIndex); | ||||
|   if (_sec){ | ||||
|     colorHStoRGB(lastRandomIndex*256,255,colSec); | ||||
|   } else { | ||||
|     colorHStoRGB(lastRandomIndex*256,255,col); | ||||
|   } | ||||
|   if (fromButton) colorUpdated(2); | ||||
| } | ||||
|  | ||||
|  | ||||
| bool isAsterisksOnly(const char* str, byte maxLen) | ||||
| { | ||||
|   for (byte i = 0; i < maxLen; i++) { | ||||
|     if (str[i] == 0) break; | ||||
|     if (str[i] != '*') return false; | ||||
|   } | ||||
|   //at this point the password contains asterisks only | ||||
|   return (str[0] != 0); //false on empty string | ||||
| } | ||||
|  | ||||
|  | ||||
| //called upon POST settings form submit | ||||
| void handleSettingsSet(AsyncWebServerRequest *request, byte subPage) | ||||
| { | ||||
|  | ||||
|   //0: menu 1: wifi 2: leds 3: ui 4: sync 5: time 6: sec 7: DMX | ||||
|   if (subPage <1 || subPage >7) return; | ||||
|  | ||||
|   //WIFI SETTINGS | ||||
|   if (subPage == 1) | ||||
|   { | ||||
|     strlcpy(clientSSID,request->arg(F("CS")).c_str(), 33); | ||||
|  | ||||
|     if (!isAsterisksOnly(request->arg(F("CP")).c_str(), 65)) strlcpy(clientPass, request->arg(F("CP")).c_str(), 65); | ||||
|  | ||||
|     strlcpy(cmDNS, request->arg(F("CM")).c_str(), 33); | ||||
|  | ||||
|     apBehavior = request->arg(F("AB")).toInt(); | ||||
|     strlcpy(apSSID, request->arg(F("AS")).c_str(), 33); | ||||
|     apHide = request->hasArg(F("AH")); | ||||
|     int passlen = request->arg(F("AP")).length(); | ||||
|     if (passlen == 0 || (passlen > 7 && !isAsterisksOnly(request->arg(F("AP")).c_str(), 65))) strlcpy(apPass, request->arg(F("AP")).c_str(), 65); | ||||
|     int t = request->arg(F("AC")).toInt(); if (t > 0 && t < 14) apChannel = t; | ||||
|  | ||||
|     noWifiSleep = request->hasArg(F("WS")); | ||||
|  | ||||
|     #ifdef WLED_USE_ETHERNET | ||||
|     ethernetType = request->arg(F("ETH")).toInt(); | ||||
|     #endif | ||||
|  | ||||
|     char k[3]; k[2] = 0; | ||||
|     for (int i = 0; i<4; i++) | ||||
|     { | ||||
|       k[1] = i+48;//ascii 0,1,2,3 | ||||
|  | ||||
|       k[0] = 'I'; //static IP | ||||
|       staticIP[i] = request->arg(k).toInt(); | ||||
|  | ||||
|       k[0] = 'G'; //gateway | ||||
|       staticGateway[i] = request->arg(k).toInt(); | ||||
|  | ||||
|       k[0] = 'S'; //subnet | ||||
|       staticSubnet[i] = request->arg(k).toInt(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //LED SETTINGS | ||||
|   if (subPage == 2) | ||||
|   { | ||||
|     int t = 0; | ||||
|  | ||||
|     if (rlyPin>=0 && pinManager.isPinAllocated(rlyPin)) pinManager.deallocatePin(rlyPin); | ||||
|     #ifndef WLED_DISABLE_INFRARED | ||||
|     if (irPin>=0 && pinManager.isPinAllocated(irPin)) pinManager.deallocatePin(irPin); | ||||
|     #endif | ||||
|     if (btnPin>=0 && pinManager.isPinAllocated(btnPin)) pinManager.deallocatePin(btnPin); | ||||
|     //TODO remove all busses, but not in this system call | ||||
|     //busses->removeAll(); | ||||
|  | ||||
|     uint8_t colorOrder, type; | ||||
|     uint16_t length, start; | ||||
|     uint8_t pins[5] = {255, 255, 255, 255, 255}; | ||||
|  | ||||
|     for (uint8_t s = 0; s < WLED_MAX_BUSSES; s++) { | ||||
|       char lp[4] = "L0"; lp[2] = 48+s; lp[3] = 0; //ascii 0-9 //strip data pin | ||||
|       char lc[4] = "LC"; lc[2] = 48+s; lc[3] = 0; //strip length | ||||
|       char co[4] = "CO"; co[2] = 48+s; co[3] = 0; //strip color order | ||||
|       char lt[4] = "LT"; lt[2] = 48+s; lt[3] = 0; //strip type | ||||
|       char ls[4] = "LS"; ls[2] = 48+s; ls[3] = 0; //strip start LED | ||||
|       char cv[4] = "CV"; cv[2] = 48+s; cv[3] = 0; //strip reverse | ||||
|       if (!request->hasArg(lp)) { | ||||
|         DEBUG_PRINTLN("No data."); break; | ||||
|       } | ||||
|       for (uint8_t i = 0; i < 5; i++) { | ||||
|         lp[1] = 48+i; | ||||
|         if (!request->hasArg(lp)) break; | ||||
|         pins[i] = (request->arg(lp).length() > 0) ? request->arg(lp).toInt() : 255; | ||||
|       } | ||||
|       type = request->arg(lt).toInt(); | ||||
|       //if (isRgbw(type)) strip.isRgbw = true; //30fps | ||||
|       //strip.isRgbw = true; | ||||
|        | ||||
|       if (request->hasArg(lc) && request->arg(lc).toInt() > 0) { | ||||
|         length = request->arg(lc).toInt(); | ||||
|       } else { | ||||
|         break;  // no parameter | ||||
|       } | ||||
|  | ||||
|       colorOrder = request->arg(co).toInt(); | ||||
|       start = (request->hasArg(ls)) ? request->arg(ls).toInt() : 0; | ||||
|  | ||||
|       if (busConfigs[s] != nullptr) delete busConfigs[s]; | ||||
|       busConfigs[s] = new BusConfig(type, pins, start, length, colorOrder, request->hasArg(cv)); | ||||
|       //if (BusManager::isRgbw(type)) strip.isRgbw = true; //20fps | ||||
|       //strip.isRgbw = true; | ||||
|       doInitBusses = true; | ||||
|     } | ||||
|  | ||||
|     ledCount = request->arg(F("LC")).toInt(); | ||||
|     if (t > 0 && t <= MAX_LEDS) ledCount = t; | ||||
|  | ||||
|     // upate other pins | ||||
|     #ifndef WLED_DISABLE_INFRARED | ||||
|     int hw_ir_pin = request->arg(F("IR")).toInt(); | ||||
|     if (pinManager.isPinOk(hw_ir_pin) && pinManager.allocatePin(hw_ir_pin,false)) { | ||||
|       irPin = hw_ir_pin; | ||||
|     } else { | ||||
|       irPin = -1; | ||||
|     } | ||||
|     #endif | ||||
|  | ||||
|     int hw_rly_pin = request->arg(F("RL")).toInt(); | ||||
|     if (pinManager.allocatePin(hw_rly_pin,true)) { | ||||
|       rlyPin = hw_rly_pin; | ||||
|     } else { | ||||
|       rlyPin = -1; | ||||
|     } | ||||
|     rlyMde = (bool)request->hasArg(F("RM")); | ||||
|  | ||||
|     int hw_btn_pin = request->arg(F("BT")).toInt(); | ||||
|     if (pinManager.allocatePin(hw_btn_pin,false)) { | ||||
|       btnPin = hw_btn_pin; | ||||
|       pinMode(btnPin, INPUT_PULLUP); | ||||
|     } else { | ||||
|       btnPin = -1; | ||||
|     } | ||||
|  | ||||
|     strip.ablMilliampsMax = request->arg(F("MA")).toInt(); | ||||
|     strip.milliampsPerLed = request->arg(F("LA")).toInt(); | ||||
|      | ||||
|     strip.rgbwMode = request->arg(F("AW")).toInt(); | ||||
|  | ||||
|     briS = request->arg(F("CA")).toInt(); | ||||
|  | ||||
|     saveCurrPresetCycConf = request->hasArg(F("PC")); | ||||
|     turnOnAtBoot = request->hasArg(F("BO")); | ||||
|     t = request->arg(F("BP")).toInt(); | ||||
|     if (t <= 250) bootPreset = t; | ||||
|     strip.gammaCorrectBri = request->hasArg(F("GB")); | ||||
|     strip.gammaCorrectCol = request->hasArg(F("GC")); | ||||
|  | ||||
|     fadeTransition = request->hasArg(F("TF")); | ||||
|     t = request->arg(F("TD")).toInt(); | ||||
|     if (t > 0) transitionDelay = t; | ||||
|     transitionDelayDefault = t; | ||||
|     strip.paletteFade = request->hasArg(F("PF")); | ||||
|  | ||||
|     nightlightTargetBri = request->arg(F("TB")).toInt(); | ||||
|     t = request->arg(F("TL")).toInt(); | ||||
|     if (t > 0) nightlightDelayMinsDefault = t; | ||||
|     nightlightDelayMins = nightlightDelayMinsDefault; | ||||
|     nightlightMode = request->arg(F("TW")).toInt(); | ||||
|  | ||||
|     t = request->arg(F("PB")).toInt(); | ||||
|     if (t >= 0 && t < 4) strip.paletteBlend = t; | ||||
|     skipFirstLed = request->hasArg(F("SL")); | ||||
|     t = request->arg(F("BF")).toInt(); | ||||
|     if (t > 0) briMultiplier = t; | ||||
|   } | ||||
|  | ||||
|   //UI | ||||
|   if (subPage == 3) | ||||
|   { | ||||
|     strlcpy(serverDescription, request->arg(F("DS")).c_str(), 33); | ||||
|     syncToggleReceive = request->hasArg(F("ST")); | ||||
|   } | ||||
|  | ||||
|   //SYNC | ||||
|   if (subPage == 4) | ||||
|   { | ||||
|     buttonEnabled = request->hasArg(F("BT")); | ||||
|     irEnabled = request->arg(F("IR")).toInt(); | ||||
|     int t = request->arg(F("UP")).toInt(); | ||||
|     if (t > 0) udpPort = t; | ||||
|     t = request->arg(F("U2")).toInt(); | ||||
|     if (t > 0) udpPort2 = t; | ||||
|     receiveNotificationBrightness = request->hasArg(F("RB")); | ||||
|     receiveNotificationColor = request->hasArg(F("RC")); | ||||
|     receiveNotificationEffects = request->hasArg(F("RX")); | ||||
|     receiveNotifications = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects); | ||||
|     notifyDirectDefault = request->hasArg(F("SD")); | ||||
|     notifyDirect = notifyDirectDefault; | ||||
|     notifyButton = request->hasArg(F("SB")); | ||||
|     notifyAlexa = request->hasArg(F("SA")); | ||||
|     notifyHue = request->hasArg(F("SH")); | ||||
|     notifyMacro = request->hasArg(F("SM")); | ||||
|     notifyTwice = request->hasArg(F("S2")); | ||||
|  | ||||
|     nodeListEnabled = request->hasArg(F("NL")); | ||||
|     if (!nodeListEnabled) Nodes.clear(); | ||||
|     nodeBroadcastEnabled = request->hasArg(F("NB")); | ||||
|  | ||||
|     receiveDirect = request->hasArg(F("RD")); | ||||
|     e131SkipOutOfSequence = request->hasArg(F("ES")); | ||||
|     e131Multicast = request->hasArg(F("EM")); | ||||
|     t = request->arg(F("EP")).toInt(); | ||||
|     if (t > 0) e131Port = t; | ||||
|     t = request->arg(F("EU")).toInt(); | ||||
|     if (t >= 0  && t <= 63999) e131Universe = t; | ||||
|     t = request->arg(F("DA")).toInt(); | ||||
|     if (t >= 0  && t <= 510) DMXAddress = t; | ||||
|     t = request->arg(F("DM")).toInt(); | ||||
|     if (t >= DMX_MODE_DISABLED && t <= DMX_MODE_MULTIPLE_RGBW) DMXMode = t; | ||||
|     t = request->arg(F("ET")).toInt(); | ||||
|     if (t > 99  && t <= 65000) realtimeTimeoutMs = t; | ||||
|     arlsForceMaxBri = request->hasArg(F("FB")); | ||||
|     arlsDisableGammaCorrection = request->hasArg(F("RG")); | ||||
|     t = request->arg(F("WO")).toInt(); | ||||
|     if (t >= -255  && t <= 255) arlsOffset = t; | ||||
|  | ||||
|     alexaEnabled = request->hasArg(F("AL")); | ||||
|     strlcpy(alexaInvocationName, request->arg(F("AI")).c_str(), 33); | ||||
|  | ||||
|     strlcpy(blynkHost, request->arg("BH").c_str(), 33); | ||||
|     t = request->arg(F("BP")).toInt(); | ||||
|     if (t > 0) blynkPort = t; | ||||
|  | ||||
|     if (request->hasArg("BK") && !request->arg("BK").equals(F("Hidden"))) { | ||||
|       strlcpy(blynkApiKey, request->arg("BK").c_str(), 36); initBlynk(blynkApiKey, blynkHost, blynkPort); | ||||
|     } | ||||
|  | ||||
|     #ifdef WLED_ENABLE_MQTT | ||||
|     mqttEnabled = request->hasArg(F("MQ")); | ||||
|     strlcpy(mqttServer, request->arg(F("MS")).c_str(), 33); | ||||
|     t = request->arg(F("MQPORT")).toInt(); | ||||
|     if (t > 0) mqttPort = t; | ||||
|     strlcpy(mqttUser, request->arg(F("MQUSER")).c_str(), 41); | ||||
|     if (!isAsterisksOnly(request->arg(F("MQPASS")).c_str(), 41)) strlcpy(mqttPass, request->arg(F("MQPASS")).c_str(), 41); | ||||
|     strlcpy(mqttClientID, request->arg(F("MQCID")).c_str(), 41); | ||||
|     strlcpy(mqttDeviceTopic, request->arg(F("MD")).c_str(), 33); | ||||
|     strlcpy(mqttGroupTopic, request->arg(F("MG")).c_str(), 33); | ||||
|     #endif | ||||
|  | ||||
|     #ifndef WLED_DISABLE_HUESYNC | ||||
|     for (int i=0;i<4;i++){ | ||||
|       String a = "H"+String(i); | ||||
|       hueIP[i] = request->arg(a).toInt(); | ||||
|     } | ||||
|  | ||||
|     t = request->arg(F("HL")).toInt(); | ||||
|     if (t > 0) huePollLightId = t; | ||||
|  | ||||
|     t = request->arg(F("HI")).toInt(); | ||||
|     if (t > 50) huePollIntervalMs = t; | ||||
|  | ||||
|     hueApplyOnOff = request->hasArg(F("HO")); | ||||
|     hueApplyBri = request->hasArg(F("HB")); | ||||
|     hueApplyColor = request->hasArg(F("HC")); | ||||
|     huePollingEnabled = request->hasArg(F("HP")); | ||||
|     hueStoreAllowed = true; | ||||
|     reconnectHue(); | ||||
|     #endif | ||||
|   } | ||||
|  | ||||
|   //TIME | ||||
|   if (subPage == 5) | ||||
|   { | ||||
|     ntpEnabled = request->hasArg(F("NT")); | ||||
|     strlcpy(ntpServerName, request->arg(F("NS")).c_str(), 33); | ||||
|     useAMPM = !request->hasArg(F("CF")); | ||||
|     currentTimezone = request->arg(F("TZ")).toInt(); | ||||
|     utcOffsetSecs = request->arg(F("UO")).toInt(); | ||||
|  | ||||
|     //start ntp if not already connected | ||||
|     if (ntpEnabled && WLED_CONNECTED && !ntpConnected) ntpConnected = ntpUdp.begin(ntpLocalPort); | ||||
|  | ||||
|     if (request->hasArg(F("OL"))) { | ||||
|       overlayDefault = request->arg(F("OL")).toInt(); | ||||
|       overlayCurrent = overlayDefault; | ||||
|     } | ||||
|  | ||||
|     overlayMin = request->arg(F("O1")).toInt(); | ||||
|     overlayMax = request->arg(F("O2")).toInt(); | ||||
|     analogClock12pixel = request->arg(F("OM")).toInt(); | ||||
|     analogClock5MinuteMarks = request->hasArg(F("O5")); | ||||
|     analogClockSecondsTrail = request->hasArg(F("OS")); | ||||
|  | ||||
|     strcpy(cronixieDisplay,request->arg(F("CX")).c_str()); | ||||
|     cronixieBacklight = request->hasArg(F("CB")); | ||||
|     countdownMode = request->hasArg(F("CE")); | ||||
|     countdownYear = request->arg(F("CY")).toInt(); | ||||
|     countdownMonth = request->arg(F("CI")).toInt(); | ||||
|     countdownDay = request->arg(F("CD")).toInt(); | ||||
|     countdownHour = request->arg(F("CH")).toInt(); | ||||
|     countdownMin = request->arg(F("CM")).toInt(); | ||||
|     countdownSec = request->arg(F("CS")).toInt(); | ||||
|     setCountdown(); | ||||
|  | ||||
|     macroAlexaOn = request->arg(F("A0")).toInt(); | ||||
|     macroAlexaOff = request->arg(F("A1")).toInt(); | ||||
|     macroButton = request->arg(F("MP")).toInt(); | ||||
|     macroLongPress = request->arg(F("ML")).toInt(); | ||||
|     macroCountdown = request->arg(F("MC")).toInt(); | ||||
|     macroNl = request->arg(F("MN")).toInt(); | ||||
|     macroDoublePress = request->arg(F("MD")).toInt(); | ||||
|  | ||||
|     char k[3]; k[2] = 0; | ||||
|     for (int i = 0; i<8; i++) | ||||
|     { | ||||
|       k[1] = i+48;//ascii 0,1,2,3 | ||||
|  | ||||
|       k[0] = 'H'; //timer hours | ||||
|       timerHours[i] = request->arg(k).toInt(); | ||||
|  | ||||
|       k[0] = 'N'; //minutes | ||||
|       timerMinutes[i] = request->arg(k).toInt(); | ||||
|  | ||||
|       k[0] = 'T'; //macros | ||||
|       timerMacro[i] = request->arg(k).toInt(); | ||||
|  | ||||
|       k[0] = 'W'; //weekdays | ||||
|       timerWeekday[i] = request->arg(k).toInt(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //SECURITY | ||||
|   if (subPage == 6) | ||||
|   { | ||||
|     if (request->hasArg(F("RS"))) //complete factory reset | ||||
|     { | ||||
|       WLED_FS.format(); | ||||
|       clearEEPROM(); | ||||
|       serveMessage(request, 200, F("All Settings erased."), F("Connect to WLED-AP to setup again"),255); | ||||
|       doReboot = true; | ||||
|     } | ||||
|  | ||||
|     bool pwdCorrect = !otaLock; //always allow access if ota not locked | ||||
|     if (request->hasArg(F("OP"))) | ||||
|     { | ||||
|       if (otaLock && strcmp(otaPass,request->arg(F("OP")).c_str()) == 0) | ||||
|       { | ||||
|         pwdCorrect = true; | ||||
|       } | ||||
|       if (!otaLock && request->arg(F("OP")).length() > 0) | ||||
|       { | ||||
|         strlcpy(otaPass,request->arg(F("OP")).c_str(), 33); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if (pwdCorrect) //allow changes if correct pwd or no ota active | ||||
|     { | ||||
|       otaLock = request->hasArg(F("NO")); | ||||
|       wifiLock = request->hasArg(F("OW")); | ||||
|       aOtaEnabled = request->hasArg(F("AO")); | ||||
|     } | ||||
|   } | ||||
|   #ifdef WLED_ENABLE_DMX // include only if DMX is enabled | ||||
|   if (subPage == 7) | ||||
|   { | ||||
|     int t = request->arg(F("PU")).toInt(); | ||||
|     if (t >= 0  && t <= 63999) e131ProxyUniverse = t; | ||||
|  | ||||
|     t = request->arg(F("CN")).toInt(); | ||||
|     if (t>0 && t<16) { | ||||
|       DMXChannels = t; | ||||
|     } | ||||
|     t = request->arg(F("CS")).toInt(); | ||||
|     if (t>0 && t<513) { | ||||
|       DMXStart = t; | ||||
|     } | ||||
|     t = request->arg(F("CG")).toInt(); | ||||
|     if (t>0 && t<513) { | ||||
|       DMXGap = t; | ||||
|     } | ||||
|     t = request->arg(F("SL")).toInt(); | ||||
|     if (t>=0 && t < MAX_LEDS) { | ||||
|       DMXStartLED = t; | ||||
|     } | ||||
|     for (int i=0; i<15; i++) { | ||||
|       String argname = "CH" + String((i+1)); | ||||
|       t = request->arg(argname).toInt(); | ||||
|       DMXFixtureMap[i] = t; | ||||
|     } | ||||
|   } | ||||
|   #endif | ||||
|  | ||||
|   if (subPage != 2 && (subPage != 6 || !doReboot)) serializeConfig(); //do not save if factory reset or LED settings (which are saved after LED re-init) | ||||
|   if (subPage == 4) alexaInit(); | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| //helper to get int value at a position in string | ||||
| int getNumVal(const String* req, uint16_t pos) | ||||
| { | ||||
|   return req->substring(pos+3).toInt(); | ||||
| } | ||||
|  | ||||
|  | ||||
| //helper to get int value at a position in string | ||||
| bool updateVal(const String* req, const char* key, byte* val, byte minv, byte maxv) | ||||
| { | ||||
|   int pos = req->indexOf(key); | ||||
|   if (pos < 1) return false; | ||||
|  | ||||
|   if (req->charAt(pos+3) == '~') { | ||||
|     int out = getNumVal(req, pos+1); | ||||
|     if (out == 0) | ||||
|     { | ||||
|       if (req->charAt(pos+4) == '-') | ||||
|       { | ||||
|         *val = (*val <= minv)? maxv : *val -1; | ||||
|       } else { | ||||
|         *val = (*val >= maxv)? minv : *val +1; | ||||
|       } | ||||
|     } else { | ||||
|       out += *val; | ||||
|       if (out > maxv) out = maxv; | ||||
|       if (out < minv) out = minv; | ||||
|       *val = out; | ||||
|     } | ||||
|   } else | ||||
|   { | ||||
|     *val = getNumVal(req, pos); | ||||
|   } | ||||
|   return true; | ||||
| } | ||||
|  | ||||
|  | ||||
| //HTTP API request parser | ||||
| bool handleSet(AsyncWebServerRequest *request, const String& req, bool apply) | ||||
| { | ||||
|   if (!(req.indexOf("win") >= 0)) return false; | ||||
|  | ||||
|   int pos = 0; | ||||
|   DEBUG_PRINT(F("API req: ")); | ||||
|   DEBUG_PRINTLN(req); | ||||
|  | ||||
|   strip.applyToAllSelected = false; | ||||
|   //snapshot to check if request changed values later, temporary. | ||||
|   byte prevCol[4] = {col[0], col[1], col[2], col[3]}; | ||||
|   byte prevColSec[4] = {colSec[0], colSec[1], colSec[2], colSec[3]}; | ||||
|   byte prevEffect = effectCurrent; | ||||
|   byte prevSpeed = effectSpeed; | ||||
|   byte prevIntensity = effectIntensity; | ||||
|   byte prevPalette = effectPalette; | ||||
|  | ||||
|   //segment select (sets main segment) | ||||
|   byte prevMain = strip.getMainSegmentId(); | ||||
|   pos = req.indexOf(F("SM=")); | ||||
|   if (pos > 0) { | ||||
|     strip.mainSegment = getNumVal(&req, pos); | ||||
|   } | ||||
|   byte selectedSeg = strip.getMainSegmentId(); | ||||
|   if (selectedSeg != prevMain) setValuesFromMainSeg(); | ||||
|  | ||||
|   pos = req.indexOf(F("SS=")); | ||||
|   if (pos > 0) { | ||||
|     byte t = getNumVal(&req, pos); | ||||
|     if (t < strip.getMaxSegments()) selectedSeg = t; | ||||
|   } | ||||
|  | ||||
|   WS2812FX::Segment& mainseg = strip.getSegment(selectedSeg); | ||||
|   pos = req.indexOf(F("SV=")); //segment selected | ||||
|   if (pos > 0) { | ||||
|     byte t = getNumVal(&req, pos); | ||||
|     if (t == 2) { | ||||
|       for (uint8_t i = 0; i < strip.getMaxSegments(); i++) | ||||
|       { | ||||
|         strip.getSegment(i).setOption(SEG_OPTION_SELECTED, 0); | ||||
|       } | ||||
|     } | ||||
|     mainseg.setOption(SEG_OPTION_SELECTED, t); | ||||
|   } | ||||
|  | ||||
|   uint16_t startI = mainseg.start; | ||||
|   uint16_t stopI = mainseg.stop; | ||||
|   uint8_t grpI = mainseg.grouping; | ||||
|   uint16_t spcI = mainseg.spacing; | ||||
|   pos = req.indexOf(F("&S=")); //segment start | ||||
|   if (pos > 0) { | ||||
|     startI = getNumVal(&req, pos); | ||||
|   } | ||||
|   pos = req.indexOf(F("S2=")); //segment stop | ||||
|   if (pos > 0) { | ||||
|     stopI = getNumVal(&req, pos); | ||||
|   } | ||||
|   pos = req.indexOf(F("GP=")); //segment grouping | ||||
|   if (pos > 0) { | ||||
|     grpI = getNumVal(&req, pos); | ||||
|     if (grpI == 0) grpI = 1; | ||||
|   } | ||||
|   pos = req.indexOf(F("SP=")); //segment spacing | ||||
|   if (pos > 0) { | ||||
|     spcI = getNumVal(&req, pos); | ||||
|   } | ||||
|   strip.setSegment(selectedSeg, startI, stopI, grpI, spcI); | ||||
|  | ||||
|    //set presets | ||||
|   pos = req.indexOf(F("P1=")); //sets first preset for cycle | ||||
|   if (pos > 0) presetCycleMin = getNumVal(&req, pos); | ||||
|  | ||||
|   pos = req.indexOf(F("P2=")); //sets last preset for cycle | ||||
|   if (pos > 0) presetCycleMax = getNumVal(&req, pos); | ||||
|  | ||||
|   //preset cycle | ||||
|   pos = req.indexOf(F("CY=")); | ||||
|   if (pos > 0) | ||||
|   { | ||||
|     char cmd = req.charAt(pos+3); | ||||
|     if (cmd == '2') presetCyclingEnabled = !presetCyclingEnabled; | ||||
|     else presetCyclingEnabled = (cmd != '0'); | ||||
|     presetCycCurr = presetCycleMin; | ||||
|   } | ||||
|  | ||||
|   pos = req.indexOf(F("PT=")); //sets cycle time in ms | ||||
|   if (pos > 0) { | ||||
|     int v = getNumVal(&req, pos); | ||||
|     if (v > 100) presetCycleTime = v/100; | ||||
|   } | ||||
|  | ||||
|   pos = req.indexOf(F("PS=")); //saves current in preset | ||||
|   if (pos > 0) savePreset(getNumVal(&req, pos)); | ||||
|  | ||||
|   //apply preset | ||||
|   if (updateVal(&req, "PL=", &presetCycCurr, presetCycleMin, presetCycleMax)) { | ||||
|     applyPreset(presetCycCurr); | ||||
|   } | ||||
|  | ||||
|   //set brightness | ||||
|   updateVal(&req, "&A=", &bri); | ||||
|  | ||||
|   //set colors | ||||
|   updateVal(&req, "&R=", &col[0]); | ||||
|   updateVal(&req, "&G=", &col[1]); | ||||
|   updateVal(&req, "&B=", &col[2]); | ||||
|   updateVal(&req, "&W=", &col[3]); | ||||
|   updateVal(&req, "R2=", &colSec[0]); | ||||
|   updateVal(&req, "G2=", &colSec[1]); | ||||
|   updateVal(&req, "B2=", &colSec[2]); | ||||
|   updateVal(&req, "W2=", &colSec[3]); | ||||
|  | ||||
|   #ifdef WLED_ENABLE_LOXONE | ||||
|   //lox parser | ||||
|   pos = req.indexOf(F("LX=")); // Lox primary color | ||||
|   if (pos > 0) { | ||||
|     int lxValue = getNumVal(&req, pos); | ||||
|     if (parseLx(lxValue, col)) { | ||||
|       bri = 255; | ||||
|       nightlightActive = false; //always disable nightlight when toggling | ||||
|     } | ||||
|   } | ||||
|   pos = req.indexOf(F("LY=")); // Lox secondary color | ||||
|   if (pos > 0) { | ||||
|     int lxValue = getNumVal(&req, pos); | ||||
|     if(parseLx(lxValue, colSec)) { | ||||
|       bri = 255; | ||||
|       nightlightActive = false; //always disable nightlight when toggling | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   #endif | ||||
|  | ||||
|   //set hue | ||||
|   pos = req.indexOf(F("HU=")); | ||||
|   if (pos > 0) { | ||||
|     uint16_t temphue = getNumVal(&req, pos); | ||||
|     byte tempsat = 255; | ||||
|     pos = req.indexOf(F("SA=")); | ||||
|     if (pos > 0) { | ||||
|       tempsat = getNumVal(&req, pos); | ||||
|     } | ||||
|     colorHStoRGB(temphue,tempsat,(req.indexOf(F("H2"))>0)? colSec:col); | ||||
|   } | ||||
|  | ||||
|   //set white spectrum (kelvin) | ||||
|   pos = req.indexOf(F("&K=")); | ||||
|   if (pos > 0) { | ||||
|     colorKtoRGB(getNumVal(&req, pos),(req.indexOf(F("K2"))>0)? colSec:col); | ||||
|   } | ||||
|  | ||||
|   //set color from HEX or 32bit DEC | ||||
|   pos = req.indexOf(F("CL=")); | ||||
|   if (pos > 0) { | ||||
|     colorFromDecOrHexString(col, (char*)req.substring(pos + 3).c_str()); | ||||
|   } | ||||
|   pos = req.indexOf(F("C2=")); | ||||
|   if (pos > 0) { | ||||
|     colorFromDecOrHexString(colSec, (char*)req.substring(pos + 3).c_str()); | ||||
|   } | ||||
|   pos = req.indexOf(F("C3=")); | ||||
|   if (pos > 0) { | ||||
|     byte t[4]; | ||||
|     colorFromDecOrHexString(t, (char*)req.substring(pos + 3).c_str()); | ||||
|     if (selectedSeg != strip.getMainSegmentId()) { | ||||
|       strip.applyToAllSelected = true; | ||||
|       strip.setColor(2, t[0], t[1], t[2], t[3]); | ||||
|     } else { | ||||
|       strip.getSegment(selectedSeg).setColor(2,((t[0] << 16) + (t[1] << 8) + t[2] + (t[3] << 24)), selectedSeg); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //set to random hue SR=0->1st SR=1->2nd | ||||
|   pos = req.indexOf(F("SR")); | ||||
|   if (pos > 0) { | ||||
|     _setRandomColor(getNumVal(&req, pos)); | ||||
|   } | ||||
|  | ||||
|   //swap 2nd & 1st | ||||
|   pos = req.indexOf(F("SC")); | ||||
|   if (pos > 0) { | ||||
|     byte temp; | ||||
|     for (uint8_t i=0; i<4; i++) | ||||
|     { | ||||
|       temp = col[i]; | ||||
|       col[i] = colSec[i]; | ||||
|       colSec[i] = temp; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //set effect parameters | ||||
|   if (updateVal(&req, "FX=", &effectCurrent, 0, strip.getModeCount()-1)) presetCyclingEnabled = false; | ||||
|   updateVal(&req, "SX=", &effectSpeed); | ||||
|   updateVal(&req, "IX=", &effectIntensity); | ||||
|   updateVal(&req, "FP=", &effectPalette, 0, strip.getPaletteCount()-1); | ||||
|  | ||||
|   //set advanced overlay | ||||
|   pos = req.indexOf(F("OL=")); | ||||
|   if (pos > 0) { | ||||
|     overlayCurrent = getNumVal(&req, pos); | ||||
|   } | ||||
|  | ||||
|   //apply macro (deprecated, added for compatibility with pre-0.11 automations) | ||||
|   pos = req.indexOf(F("&M=")); | ||||
|   if (pos > 0) { | ||||
|     applyPreset(getNumVal(&req, pos) + 16); | ||||
|   } | ||||
|  | ||||
|   //toggle send UDP direct notifications | ||||
|   pos = req.indexOf(F("SN=")); | ||||
|   if (pos > 0) notifyDirect = (req.charAt(pos+3) != '0'); | ||||
|  | ||||
|   //toggle receive UDP direct notifications | ||||
|   pos = req.indexOf(F("RN=")); | ||||
|   if (pos > 0) receiveNotifications = (req.charAt(pos+3) != '0'); | ||||
|  | ||||
|   //receive live data via UDP/Hyperion | ||||
|   pos = req.indexOf(F("RD=")); | ||||
|   if (pos > 0) receiveDirect = (req.charAt(pos+3) != '0'); | ||||
|  | ||||
|   //main toggle on/off (parse before nightlight, #1214) | ||||
|   pos = req.indexOf(F("&T=")); | ||||
|   if (pos > 0) { | ||||
|     nightlightActive = false; //always disable nightlight when toggling | ||||
|     switch (getNumVal(&req, pos)) | ||||
|     { | ||||
|       case 0: if (bri != 0){briLast = bri; bri = 0;} break; //off, only if it was previously on | ||||
|       case 1: if (bri == 0) bri = briLast; break; //on, only if it was previously off | ||||
|       default: toggleOnOff(); //toggle | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //toggle nightlight mode | ||||
|   bool aNlDef = false; | ||||
|   if (req.indexOf(F("&ND")) > 0) aNlDef = true; | ||||
|   pos = req.indexOf(F("NL=")); | ||||
|   if (pos > 0) | ||||
|   { | ||||
|     if (req.charAt(pos+3) == '0') | ||||
|     { | ||||
|       nightlightActive = false; | ||||
|     } else { | ||||
|       nightlightActive = true; | ||||
|       if (!aNlDef) nightlightDelayMins = getNumVal(&req, pos); | ||||
|       nightlightStartTime = millis(); | ||||
|     } | ||||
|   } else if (aNlDef) | ||||
|   { | ||||
|     nightlightActive = true; | ||||
|     nightlightStartTime = millis(); | ||||
|   } | ||||
|  | ||||
|   //set nightlight target brightness | ||||
|   pos = req.indexOf(F("NT=")); | ||||
|   if (pos > 0) { | ||||
|     nightlightTargetBri = getNumVal(&req, pos); | ||||
|     nightlightActiveOld = false; //re-init | ||||
|   } | ||||
|  | ||||
|   //toggle nightlight fade | ||||
|   pos = req.indexOf(F("NF=")); | ||||
|   if (pos > 0) | ||||
|   { | ||||
|     nightlightMode = getNumVal(&req, pos); | ||||
|  | ||||
|     nightlightActiveOld = false; //re-init | ||||
|   } | ||||
|   if (nightlightMode > NL_MODE_SUN) nightlightMode = NL_MODE_SUN; | ||||
|  | ||||
|   pos = req.indexOf(F("TT=")); | ||||
|   if (pos > 0) transitionDelay = getNumVal(&req, pos); | ||||
|  | ||||
|   //Segment reverse | ||||
|   pos = req.indexOf(F("RV=")); | ||||
|   if (pos > 0) strip.getSegment(selectedSeg).setOption(SEG_OPTION_REVERSED, req.charAt(pos+3) != '0'); | ||||
|  | ||||
|   //Segment reverse | ||||
|   pos = req.indexOf(F("MI=")); | ||||
|   if (pos > 0) strip.getSegment(selectedSeg).setOption(SEG_OPTION_MIRROR, req.charAt(pos+3) != '0'); | ||||
|  | ||||
|   //Segment brightness/opacity | ||||
|   pos = req.indexOf(F("SB=")); | ||||
|   if (pos > 0) { | ||||
|     byte segbri = getNumVal(&req, pos); | ||||
|     strip.getSegment(selectedSeg).setOption(SEG_OPTION_ON, segbri, selectedSeg); | ||||
|     if (segbri) { | ||||
|       strip.getSegment(selectedSeg).setOpacity(segbri, selectedSeg); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //set time (unix timestamp) | ||||
|   pos = req.indexOf(F("ST=")); | ||||
|   if (pos > 0) { | ||||
|     setTime(getNumVal(&req, pos)); | ||||
|   } | ||||
|  | ||||
|   //set countdown goal (unix timestamp) | ||||
|   pos = req.indexOf(F("CT=")); | ||||
|   if (pos > 0) { | ||||
|     countdownTime = getNumVal(&req, pos); | ||||
|     if (countdownTime - now() > 0) countdownOverTriggered = false; | ||||
|   } | ||||
|  | ||||
|   pos = req.indexOf(F("LO=")); | ||||
|   if (pos > 0) { | ||||
|     realtimeOverride = getNumVal(&req, pos); | ||||
|     if (realtimeOverride > 2) realtimeOverride = REALTIME_OVERRIDE_ALWAYS; | ||||
|   } | ||||
|  | ||||
|   pos = req.indexOf(F("RB")); | ||||
|   if (pos > 0) doReboot = true; | ||||
|  | ||||
|   //cronixie | ||||
|   #ifndef WLED_DISABLE_CRONIXIE | ||||
|   //mode, 1 countdown | ||||
|   pos = req.indexOf(F("NM=")); | ||||
|   if (pos > 0) countdownMode = (req.charAt(pos+3) != '0'); | ||||
|    | ||||
|   pos = req.indexOf(F("NX=")); //sets digits to code | ||||
|   if (pos > 0) { | ||||
|     strlcpy(cronixieDisplay, req.substring(pos + 3, pos + 9).c_str(), 6); | ||||
|     setCronixie(); | ||||
|   } | ||||
|  | ||||
|   pos = req.indexOf(F("NB=")); | ||||
|   if (pos > 0) //sets backlight | ||||
|   { | ||||
|     cronixieBacklight = (req.charAt(pos+3) != '0'); | ||||
|     overlayRefreshedTime = 0; | ||||
|   } | ||||
|   #endif | ||||
|  | ||||
|   pos = req.indexOf(F("U0=")); //user var 0 | ||||
|   if (pos > 0) { | ||||
|     userVar0 = getNumVal(&req, pos); | ||||
|   } | ||||
|  | ||||
|   pos = req.indexOf(F("U1=")); //user var 1 | ||||
|   if (pos > 0) { | ||||
|     userVar1 = getNumVal(&req, pos); | ||||
|   } | ||||
|   //you can add more if you need | ||||
|  | ||||
|   //apply to all selected manually to prevent #1618. Temporary | ||||
|   bool col0Changed = false, col1Changed = false; | ||||
|   for (uint8_t i = 0; i < 4; i++) { | ||||
|     if (col[i] != prevCol[i]) col0Changed = true; | ||||
|     if (colSec[i] != prevColSec[i]) col1Changed = true; | ||||
|   } | ||||
|   for (uint8_t i = 0; i < strip.getMaxSegments(); i++) | ||||
|   { | ||||
|     WS2812FX::Segment& seg = strip.getSegment(i); | ||||
|     if (!seg.isSelected()) continue; | ||||
|     if (effectCurrent != prevEffect) { | ||||
|       seg.mode = effectCurrent; | ||||
|       effectChanged = true; | ||||
|     } | ||||
|     if (effectSpeed != prevSpeed) { | ||||
|       seg.speed = effectSpeed; | ||||
|       effectChanged = true; | ||||
|     } | ||||
|     if (effectIntensity != prevIntensity) { | ||||
|       seg.intensity = effectIntensity; | ||||
|       effectChanged = true; | ||||
|     } | ||||
|     if (effectPalette != prevPalette) { | ||||
|       seg.palette = effectPalette; | ||||
|       effectChanged = true; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (col0Changed) { | ||||
|     if (selectedSeg == strip.getMainSegmentId()) { | ||||
|       strip.applyToAllSelected = true; | ||||
|       strip.setColor(0, colorFromRgbw(col)); | ||||
|     } | ||||
|   } | ||||
|   if (col1Changed) { | ||||
|     if (selectedSeg == strip.getMainSegmentId()) { | ||||
|       strip.applyToAllSelected = true; | ||||
|       strip.setColor(1, colorFromRgbw(colSec)); | ||||
|     } | ||||
|   } | ||||
|   //end of temporary fix code | ||||
|  | ||||
|   if (!apply) return true; //when called by JSON API, do not call colorUpdated() here | ||||
|    | ||||
|   //internal call, does not send XML response | ||||
|   pos = req.indexOf(F("IN")); | ||||
|   if (pos < 1) XML_response(request); | ||||
|  | ||||
|   strip.applyToAllSelected = false; | ||||
|  | ||||
|   pos = req.indexOf(F("&NN")); //do not send UDP notifications this time | ||||
|   colorUpdated((pos > 0) ? NOTIFIER_CALL_MODE_NO_NOTIFY : NOTIFIER_CALL_MODE_DIRECT_CHANGE); | ||||
|  | ||||
|   return true; | ||||
| } | ||||
| @@ -37,10 +37,10 @@ AsyncMqttClient::AsyncMqttClient() | ||||
|   _client.onPoll([](void* obj, AsyncClient* c) { (static_cast<AsyncMqttClient*>(obj))->_onPoll(c); }, this); | ||||
|  | ||||
| #ifdef ESP32 | ||||
|   sprintf(_generatedClientId, "esp32%06x", ESP.getEfuseMac()); | ||||
|   sprintf(_generatedClientId, "esp32%06x", (uint32_t)ESP.getEfuseMac()); | ||||
|   _xSemaphore = xSemaphoreCreateMutex(); | ||||
| #elif defined(ESP8266) | ||||
|   sprintf(_generatedClientId, "esp8266%06x", ESP.getChipId()); | ||||
|   sprintf(_generatedClientId, "esp8266%06x", (uint32_t)ESP.getChipId()); | ||||
| #endif | ||||
|   _clientId = _generatedClientId; | ||||
|  | ||||
|   | ||||
| @@ -93,7 +93,9 @@ struct BlynkHeader | ||||
| } | ||||
| BLYNK_ATTR_PACKED; | ||||
|  | ||||
| #if !defined(htons) && (defined(ARDUINO) || defined(ESP8266) || defined(PARTICLE) || defined(__MBED__)) | ||||
| #if defined(ESP32) | ||||
|     #include <lwip/ip_addr.h> | ||||
| #elif !defined(htons) && (defined(ARDUINO) || defined(ESP8266) || defined(PARTICLE) || defined(__MBED__)) | ||||
|     #if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ | ||||
|         #define htons(x) ( ((x)<<8) | (((x)>>8)&0xFF) ) | ||||
|         #define htonl(x) ( ((x)<<24 & 0xFF000000UL) | \ | ||||
|   | ||||
							
								
								
									
										5
									
								
								wled00/src/dependencies/blynk/BlynkSimpleEsp.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								wled00/src/dependencies/blynk/BlynkSimpleEsp.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| #include "BlynkSimpleEsp.h" | ||||
|  | ||||
| WiFiClient _blynkWifiClient; | ||||
| BlynkArduinoClient _blynkTransport(_blynkWifiClient); | ||||
| BlynkWifi Blynk(_blynkTransport); | ||||
| @@ -45,8 +45,7 @@ public: | ||||
|         } | ||||
|         BLYNK_LOG1(BLYNK_F("Connected to WiFi")); | ||||
|  | ||||
|         IPAddress myip = WiFi.localIP(); | ||||
|         BLYNK_LOG_IP("IP: ", myip); | ||||
|         BLYNK_LOG_IP("IP: ", WiFi.localIP()); | ||||
|     } | ||||
|  | ||||
|     void config(const char* auth, | ||||
| @@ -89,8 +88,6 @@ public: | ||||
|  | ||||
| }; | ||||
|  | ||||
| static WiFiClient _blynkWifiClient; | ||||
| static BlynkArduinoClient _blynkTransport(_blynkWifiClient); | ||||
| BlynkWifi Blynk(_blynkTransport); | ||||
| extern BlynkWifi Blynk; | ||||
|  | ||||
| #endif | ||||
|   | ||||
							
								
								
									
										105
									
								
								wled00/src/dependencies/dmx/ESPDMX.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								wled00/src/dependencies/dmx/ESPDMX.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | ||||
| // - - - - - | ||||
| // ESPDMX - A Arduino library for sending and receiving DMX using the builtin serial hardware port. | ||||
| // ESPDMX.cpp: Library implementation file | ||||
| // | ||||
| // Copyright (C) 2015  Rick <ricardogg95@gmail.com> | ||||
| // This work is licensed under a GNU style license. | ||||
| // | ||||
| // Last change: Marcel Seerig <https://github.com/mseerig> | ||||
| // | ||||
| // Documentation and samples are available at https://github.com/Rickgg/ESP-Dmx | ||||
| // - - - - - | ||||
|  | ||||
| /* ----- LIBRARIES ----- */ | ||||
| #include <Arduino.h> | ||||
|  | ||||
| #include "ESPDMX.h" | ||||
|  | ||||
|  | ||||
|  | ||||
| #define dmxMaxChannel  512 | ||||
| #define defaultMax 32 | ||||
|  | ||||
| #define DMXSPEED       250000 | ||||
| #define DMXFORMAT      SERIAL_8N2 | ||||
| #define BREAKSPEED     83333 | ||||
| #define BREAKFORMAT    SERIAL_8N1 | ||||
|  | ||||
| bool dmxStarted = false; | ||||
| int sendPin = 2;		//dafault on ESP8266 | ||||
|  | ||||
| //DMX value array and size. Entry 0 will hold startbyte | ||||
| uint8_t dmxData[dmxMaxChannel] = {}; | ||||
| int chanSize; | ||||
|  | ||||
|  | ||||
| void DMXESPSerial::init() { | ||||
|   chanSize = defaultMax; | ||||
|  | ||||
|   Serial1.begin(DMXSPEED); | ||||
|   pinMode(sendPin, OUTPUT); | ||||
|   dmxStarted = true; | ||||
| } | ||||
|  | ||||
| // Set up the DMX-Protocol | ||||
| void DMXESPSerial::init(int chanQuant) { | ||||
|  | ||||
|   if (chanQuant > dmxMaxChannel || chanQuant <= 0) { | ||||
|     chanQuant = defaultMax; | ||||
|   } | ||||
|  | ||||
|   chanSize = chanQuant; | ||||
|  | ||||
|   Serial1.begin(DMXSPEED); | ||||
|   pinMode(sendPin, OUTPUT); | ||||
|   dmxStarted = true; | ||||
| } | ||||
|  | ||||
| // Function to read DMX data | ||||
| uint8_t DMXESPSerial::read(int Channel) { | ||||
|   if (dmxStarted == false) init(); | ||||
|  | ||||
|   if (Channel < 1) Channel = 1; | ||||
|   if (Channel > dmxMaxChannel) Channel = dmxMaxChannel; | ||||
|   return(dmxData[Channel]); | ||||
| } | ||||
|  | ||||
| // Function to send DMX data | ||||
| void DMXESPSerial::write(int Channel, uint8_t value) { | ||||
|   if (dmxStarted == false) init(); | ||||
|  | ||||
|   if (Channel < 1) Channel = 1; | ||||
|   if (Channel > chanSize) Channel = chanSize; | ||||
|   if (value < 0) value = 0; | ||||
|   if (value > 255) value = 255; | ||||
|  | ||||
|   dmxData[Channel] = value; | ||||
| } | ||||
|  | ||||
| void DMXESPSerial::end() { | ||||
|   chanSize = 0; | ||||
|   Serial1.end(); | ||||
|   dmxStarted = false; | ||||
| } | ||||
|  | ||||
| void DMXESPSerial::update() { | ||||
|   if (dmxStarted == false) init(); | ||||
|  | ||||
|   //Send break | ||||
|   digitalWrite(sendPin, HIGH); | ||||
|   Serial1.begin(BREAKSPEED, BREAKFORMAT); | ||||
|   Serial1.write(0); | ||||
|   Serial1.flush(); | ||||
|   delay(1); | ||||
|   Serial1.end(); | ||||
|  | ||||
|   //send data | ||||
|   Serial1.begin(DMXSPEED, DMXFORMAT); | ||||
|   digitalWrite(sendPin, LOW); | ||||
|   Serial1.write(dmxData, chanSize); | ||||
|   Serial1.flush(); | ||||
|   delay(1); | ||||
|   Serial1.end(); | ||||
| } | ||||
|  | ||||
| // Function to update the DMX bus | ||||
							
								
								
									
										31
									
								
								wled00/src/dependencies/dmx/ESPDMX.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								wled00/src/dependencies/dmx/ESPDMX.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| // - - - - - | ||||
| // ESPDMX - A Arduino library for sending and receiving DMX using the builtin serial hardware port. | ||||
| // ESPDMX.cpp: Library implementation file | ||||
| // | ||||
| // Copyright (C) 2015  Rick <ricardogg95@gmail.com> | ||||
| // This work is licensed under a GNU style license. | ||||
| // | ||||
| // Last change: Marcel Seerig <https://github.com/mseerig> | ||||
| // | ||||
| // Documentation and samples are available at https://github.com/Rickgg/ESP-Dmx | ||||
| // - - - - - | ||||
|  | ||||
| #include <inttypes.h> | ||||
|  | ||||
|  | ||||
| #ifndef ESPDMX_h | ||||
| #define ESPDMX_h | ||||
|  | ||||
| // ---- Methods ---- | ||||
|  | ||||
| class DMXESPSerial { | ||||
| public: | ||||
|   void init(); | ||||
|   void init(int MaxChan); | ||||
|   uint8_t read(int Channel); | ||||
|   void write(int channel, uint8_t value); | ||||
|   void update(); | ||||
|   void end(); | ||||
| }; | ||||
|  | ||||
| #endif | ||||
| @@ -18,11 +18,15 @@ | ||||
| */ | ||||
|  | ||||
| #include "ESPAsyncE131.h" | ||||
| #include "../network/Network.h" | ||||
| #include <string.h> | ||||
|  | ||||
| // E1.17 ACN Packet Identifier | ||||
| const byte ESPAsyncE131::ACN_ID[12] = { 0x41, 0x53, 0x43, 0x2d, 0x45, 0x31, 0x2e, 0x31, 0x37, 0x00, 0x00, 0x00 }; | ||||
|  | ||||
| // Art-Net Packet Identifier | ||||
| const byte ESPAsyncE131::ART_ID[8]  = { 0x41, 0x72, 0x74, 0x2d, 0x4e, 0x65, 0x74, 0x00 }; | ||||
|  | ||||
| // Constructor | ||||
| ESPAsyncE131::ESPAsyncE131(e131_packet_callback_function callback) { | ||||
|   _callback = callback; | ||||
| @@ -34,15 +38,16 @@ ESPAsyncE131::ESPAsyncE131(e131_packet_callback_function callback) { | ||||
| // | ||||
| ///////////////////////////////////////////////////////// | ||||
|  | ||||
| bool ESPAsyncE131::begin(e131_listen_t type, uint16_t universe, uint8_t n) { | ||||
|     bool success = false; | ||||
| bool ESPAsyncE131::begin(bool multicast, uint16_t port, uint16_t universe, uint8_t n) { | ||||
|   bool success = false; | ||||
|  | ||||
|     if (type == E131_UNICAST) | ||||
|         success = initUnicast(); | ||||
|     if (type == E131_MULTICAST) | ||||
|         success = initMulticast(universe, n); | ||||
|   if (multicast) { | ||||
| 		success = initMulticast(port, universe, n); | ||||
| 	} else { | ||||
|     success = initUnicast(port); | ||||
| 	} | ||||
|  | ||||
|     return success; | ||||
|   return success; | ||||
| } | ||||
|  | ||||
| ///////////////////////////////////////////////////////// | ||||
| @@ -51,41 +56,39 @@ bool ESPAsyncE131::begin(e131_listen_t type, uint16_t universe, uint8_t n) { | ||||
| // | ||||
| ///////////////////////////////////////////////////////// | ||||
|  | ||||
| bool ESPAsyncE131::initUnicast() { | ||||
|     bool success = false; | ||||
| bool ESPAsyncE131::initUnicast(uint16_t port) { | ||||
|   bool success = false; | ||||
|  | ||||
|     if (udp.listen(E131_DEFAULT_PORT)) { | ||||
|         udp.onPacket(std::bind(&ESPAsyncE131::parsePacket, this, | ||||
|                 std::placeholders::_1)); | ||||
|         success = true; | ||||
|     } | ||||
|     return success; | ||||
|   if (udp.listen(port)) { | ||||
|     udp.onPacket(std::bind(&ESPAsyncE131::parsePacket, this, std::placeholders::_1)); | ||||
|     success = true; | ||||
|   } | ||||
|   return success; | ||||
| } | ||||
|  | ||||
| bool ESPAsyncE131::initMulticast(uint16_t universe, uint8_t n) { | ||||
|     bool success = false; | ||||
| bool ESPAsyncE131::initMulticast(uint16_t port, uint16_t universe, uint8_t n) { | ||||
|   bool success = false; | ||||
|  | ||||
|     IPAddress address = IPAddress(239, 255, ((universe >> 8) & 0xff), | ||||
|         ((universe >> 0) & 0xff)); | ||||
|   IPAddress address = IPAddress(239, 255, ((universe >> 8) & 0xff), | ||||
|     ((universe >> 0) & 0xff)); | ||||
|  | ||||
|     if (udp.listenMulticast(address, E131_DEFAULT_PORT)) { | ||||
|         ip4_addr_t ifaddr; | ||||
|         ip4_addr_t multicast_addr; | ||||
|   if (udp.listenMulticast(address, port)) { | ||||
|     ip4_addr_t ifaddr; | ||||
|     ip4_addr_t multicast_addr; | ||||
|  | ||||
|         ifaddr.addr = static_cast<uint32_t>(WiFi.localIP()); | ||||
|         for (uint8_t i = 1; i < n; i++) { | ||||
|             multicast_addr.addr = static_cast<uint32_t>(IPAddress(239, 255, | ||||
|                     (((universe + i) >> 8) & 0xff), (((universe + i) >> 0) | ||||
|                     & 0xff))); | ||||
|             igmp_joingroup(&ifaddr, &multicast_addr); | ||||
|         } | ||||
|  | ||||
|         udp.onPacket(std::bind(&ESPAsyncE131::parsePacket, this, | ||||
|                 std::placeholders::_1)); | ||||
|  | ||||
|         success = true; | ||||
|     ifaddr.addr = static_cast<uint32_t>(Network.localIP()); | ||||
|     for (uint8_t i = 1; i < n; i++) { | ||||
|         multicast_addr.addr = static_cast<uint32_t>(IPAddress(239, 255, | ||||
|           (((universe + i) >> 8) & 0xff), (((universe + i) >> 0) | ||||
|           & 0xff))); | ||||
|       igmp_joingroup(&ifaddr, &multicast_addr); | ||||
|     } | ||||
|     return success; | ||||
|  | ||||
|     udp.onPacket(std::bind(&ESPAsyncE131::parsePacket, this, std::placeholders::_1)); | ||||
|  | ||||
|     success = true; | ||||
|   } | ||||
|   return success; | ||||
| } | ||||
|  | ||||
| ///////////////////////////////////////////////////////// | ||||
| @@ -95,22 +98,37 @@ bool ESPAsyncE131::initMulticast(uint16_t universe, uint8_t n) { | ||||
| ///////////////////////////////////////////////////////// | ||||
|  | ||||
| void ESPAsyncE131::parsePacket(AsyncUDPPacket _packet) { | ||||
|     e131_error_t error = ERROR_NONE; | ||||
|   bool error = false; | ||||
|   uint8_t protocol = P_E131; | ||||
|  | ||||
|     sbuff = reinterpret_cast<e131_packet_t *>(_packet.data()); | ||||
|     if (memcmp(sbuff->acn_id, ESPAsyncE131::ACN_ID, sizeof(sbuff->acn_id))) | ||||
|         error = ERROR_ACN_ID; | ||||
|     if (htonl(sbuff->root_vector) != ESPAsyncE131::VECTOR_ROOT) | ||||
|         error = ERROR_VECTOR_ROOT; | ||||
|     if (htonl(sbuff->frame_vector) != ESPAsyncE131::VECTOR_FRAME) | ||||
|         error = ERROR_VECTOR_FRAME; | ||||
|     if (sbuff->dmp_vector != ESPAsyncE131::VECTOR_DMP) | ||||
|         error = ERROR_VECTOR_DMP; | ||||
|     if (sbuff->property_values[0] != 0) | ||||
|         error = ERROR_IGNORE; | ||||
|  | ||||
|     if (!error) { | ||||
|       _callback(sbuff, _packet.remoteIP()); | ||||
|     } | ||||
| } | ||||
|   sbuff = reinterpret_cast<e131_packet_t *>(_packet.data()); | ||||
| 	 | ||||
| 	//E1.31 packet identifier ("ACS-E1.17") | ||||
|   if (memcmp(sbuff->acn_id, ESPAsyncE131::ACN_ID, sizeof(sbuff->acn_id))) | ||||
|     protocol = P_ARTNET; | ||||
| 	 | ||||
| 	if (protocol == P_ARTNET) { | ||||
| 		if (memcmp(sbuff->art_id, ESPAsyncE131::ART_ID, sizeof(sbuff->art_id))) | ||||
| 			error = true; //not "Art-Net" | ||||
| 		if (sbuff->art_opcode != ARTNET_OPCODE_OPDMX) | ||||
| 			error = true; //not a DMX packet | ||||
| 	} else { //E1.31 error handling | ||||
| 		if (htonl(sbuff->root_vector) != ESPAsyncE131::VECTOR_ROOT) | ||||
| 			error = true; | ||||
| 		if (htonl(sbuff->frame_vector) != ESPAsyncE131::VECTOR_FRAME) | ||||
| 			error = true; | ||||
| 		if (sbuff->dmp_vector != ESPAsyncE131::VECTOR_DMP) | ||||
| 			error = true; | ||||
| 		if (sbuff->property_values[0] != 0) | ||||
| 			error = true; | ||||
| 	}  | ||||
|    | ||||
|   if (error && _packet.localPort() == DDP_DEFAULT_PORT) { //DDP packet | ||||
|     error = false; | ||||
|     protocol = P_DDP; | ||||
|   } | ||||
|  | ||||
|   if (!error) { | ||||
|     _callback(sbuff, _packet.remoteIP(), protocol); | ||||
|   } | ||||
| } | ||||
| @@ -5,6 +5,9 @@ | ||||
| * Copyright (c) 2019 Shelby Merrick | ||||
| * http://www.forkineye.com | ||||
| * | ||||
| *  Project: ESPAsyncDDP - Asynchronous DDP library for Arduino ESP8266 and ESP32 | ||||
| * Copyright (c) 2019 Daniel Kulp | ||||
| * | ||||
| *  This program is provided free for you to use in any way that you wish, | ||||
| *  subject to the laws and regulations where you are using it.  Due diligence | ||||
| *  is strongly suggested before using this code.  Please give credit where due. | ||||
| @@ -14,9 +17,12 @@ | ||||
| *  Author shall not be liable in any event for incidental or consequential | ||||
| *  damages in connection with, or arising out of, the furnishing, performance | ||||
| *  or use of these programs. | ||||
| * | ||||
| */ | ||||
|  | ||||
| /* | ||||
|  * Inspired by https://github.com/hideakitai/ArtNet for ArtNet support | ||||
|  */ | ||||
|  | ||||
| #ifndef ESPASYNCE131_H_ | ||||
| #define ESPASYNCE131_H_ | ||||
|  | ||||
| @@ -40,7 +46,18 @@ typedef struct ip_addr ip4_addr_t; | ||||
| #endif | ||||
|  | ||||
| // Defaults | ||||
| #define E131_DEFAULT_PORT 5568 | ||||
| #define E131_DEFAULT_PORT   5568 | ||||
| #define ARTNET_DEFAULT_PORT 6454 | ||||
| #define DDP_DEFAULT_PORT    4048 | ||||
|  | ||||
| #define DDP_PUSH_FLAG 0x01 | ||||
| #define DDP_TIMECODE_FLAG 0x10 | ||||
|  | ||||
| #define ARTNET_OPCODE_OPDMX 0x5000 | ||||
|  | ||||
| #define P_E131   0 | ||||
| #define P_ARTNET 1 | ||||
| #define P_DDP    2 | ||||
|  | ||||
| // E1.31 Packet Offsets | ||||
| #define E131_ROOT_PREAMBLE_SIZE 0 | ||||
| @@ -69,62 +86,79 @@ typedef struct ip_addr ip4_addr_t; | ||||
|  | ||||
| // E1.31 Packet Structure | ||||
| typedef union { | ||||
|     struct { | ||||
|         // Root Layer | ||||
|         uint16_t preamble_size; | ||||
|         uint16_t postamble_size; | ||||
|         uint8_t  acn_id[12]; | ||||
|         uint16_t root_flength; | ||||
|         uint32_t root_vector; | ||||
|         uint8_t  cid[16]; | ||||
|     struct { //E1.31 packet | ||||
|       // Root Layer | ||||
|       uint16_t preamble_size; | ||||
|       uint16_t postamble_size; | ||||
|       uint8_t  acn_id[12]; | ||||
|       uint16_t root_flength; | ||||
|       uint32_t root_vector; | ||||
|       uint8_t  cid[16]; | ||||
|  | ||||
|         // Frame Layer | ||||
|         uint16_t frame_flength; | ||||
|         uint32_t frame_vector; | ||||
|         uint8_t  source_name[64]; | ||||
|         uint8_t  priority; | ||||
|         uint16_t reserved; | ||||
|         uint8_t  sequence_number; | ||||
|         uint8_t  options; | ||||
|         uint16_t universe; | ||||
|       // Frame Layer | ||||
|       uint16_t frame_flength; | ||||
|       uint32_t frame_vector; | ||||
|       uint8_t  source_name[64]; | ||||
|       uint8_t  priority; | ||||
|       uint16_t reserved; | ||||
|       uint8_t  sequence_number; | ||||
|       uint8_t  options; | ||||
|       uint16_t universe; | ||||
|  | ||||
|         // DMP Layer | ||||
|         uint16_t dmp_flength; | ||||
|         uint8_t  dmp_vector; | ||||
|         uint8_t  type; | ||||
|         uint16_t first_address; | ||||
|         uint16_t address_increment; | ||||
|         uint16_t property_value_count; | ||||
|         uint8_t  property_values[513]; | ||||
|       // DMP Layer | ||||
|       uint16_t dmp_flength; | ||||
|       uint8_t  dmp_vector; | ||||
|       uint8_t  type; | ||||
|       uint16_t first_address; | ||||
|       uint16_t address_increment; | ||||
|       uint16_t property_value_count; | ||||
|       uint8_t  property_values[513]; | ||||
|     } __attribute__((packed)); | ||||
| 	 | ||||
| 	struct { //Art-Net packet | ||||
|     uint8_t  art_id[8]; | ||||
|     uint16_t art_opcode; | ||||
|     uint16_t art_protocol_ver; | ||||
|     uint8_t  art_sequence_number; | ||||
|     uint8_t  art_physical; | ||||
|     uint16_t art_universe; | ||||
|     uint16_t art_length; | ||||
|  | ||||
|     uint8_t raw[638]; | ||||
|     uint8_t  art_data[512]; | ||||
|   } __attribute__((packed)); | ||||
|  | ||||
|   struct { //DDP Header | ||||
|     uint8_t flags; | ||||
|     uint8_t sequenceNum; | ||||
|     uint8_t dataType; | ||||
|     uint8_t destination; | ||||
|     uint32_t channelOffset; | ||||
|     uint16_t dataLen; | ||||
|     uint8_t data[1]; | ||||
|   } __attribute__((packed)); | ||||
|  | ||||
|   /*struct { //DDP Time code Header (unsupported) | ||||
|     uint8_t flags; | ||||
|     uint8_t sequenceNum; | ||||
|     uint8_t dataType; | ||||
|     uint8_t destination; | ||||
|     uint32_t channelOffset; | ||||
|     uint16_t dataLen; | ||||
|     uint32_t timeCode; | ||||
|     uint8_t data[1]; | ||||
|   } __attribute__((packed));*/ | ||||
|  | ||||
|   uint8_t raw[1458]; | ||||
| } e131_packet_t; | ||||
|  | ||||
| // Error Types | ||||
| typedef enum { | ||||
|     ERROR_NONE, | ||||
|     ERROR_IGNORE, | ||||
|     ERROR_ACN_ID, | ||||
|     ERROR_PACKET_SIZE, | ||||
|     ERROR_VECTOR_ROOT, | ||||
|     ERROR_VECTOR_FRAME, | ||||
|     ERROR_VECTOR_DMP | ||||
| } e131_error_t; | ||||
|  | ||||
| // E1.31 Listener Types | ||||
| typedef enum { | ||||
|     E131_UNICAST, | ||||
|     E131_MULTICAST | ||||
| } e131_listen_t; | ||||
|  | ||||
| // new packet callback | ||||
| typedef void (*e131_packet_callback_function) (e131_packet_t* p, IPAddress clientIP); | ||||
| typedef void (*e131_packet_callback_function) (e131_packet_t* p, IPAddress clientIP, byte protocol); | ||||
|  | ||||
| class ESPAsyncE131 { | ||||
|  private: | ||||
|     // Constants for packet validation | ||||
|     static const uint8_t ACN_ID[]; | ||||
| 	  static const uint8_t ART_ID[]; | ||||
|     static const uint32_t VECTOR_ROOT = 4; | ||||
|     static const uint32_t VECTOR_FRAME = 2; | ||||
|     static const uint8_t VECTOR_DMP = 2; | ||||
| @@ -133,8 +167,8 @@ class ESPAsyncE131 { | ||||
|     AsyncUDP        udp;        // AsyncUDP | ||||
|  | ||||
|     // Internal Initializers | ||||
|     bool initUnicast(); | ||||
|     bool initMulticast(uint16_t universe, uint8_t n = 1); | ||||
|     bool initUnicast(uint16_t port); | ||||
|     bool initMulticast(uint16_t port, uint16_t universe, uint8_t n = 1); | ||||
|  | ||||
|     // Packet parser callback | ||||
|     void parsePacket(AsyncUDPPacket _packet); | ||||
| @@ -145,7 +179,7 @@ class ESPAsyncE131 { | ||||
|     ESPAsyncE131(e131_packet_callback_function callback); | ||||
|  | ||||
|     // Generic UDP listener, no physical or IP configuration | ||||
|     bool begin(e131_listen_t type, uint16_t universe = 1, uint8_t n = 1); | ||||
|     bool begin(bool multicast, uint16_t port = E131_DEFAULT_PORT, uint16_t universe = 1, uint8_t n = 1); | ||||
| }; | ||||
|  | ||||
| #endif  // ESPASYNCE131_H_ | ||||
| #endif  // ESPASYNCE131_H_ | ||||
| @@ -10,7 +10,7 @@ | ||||
|  */ | ||||
| /* | ||||
|  * @title Espalexa library | ||||
|  * @version 2.4.4 | ||||
|  * @version 2.7.0 | ||||
|  * @author Christian Schwinne | ||||
|  * @license MIT | ||||
|  * @contributors d-999 | ||||
| @@ -25,7 +25,7 @@ | ||||
| //#define ESPALEXA_NO_SUBPAGE | ||||
|  | ||||
| #ifndef ESPALEXA_MAXDEVICES | ||||
|  #define ESPALEXA_MAXDEVICES 10 //this limit only has memory reasons, set it higher should you need to | ||||
|  #define ESPALEXA_MAXDEVICES 10 //this limit only has memory reasons, set it higher should you need to, max 128 | ||||
| #endif | ||||
|  | ||||
| //#define ESPALEXA_DEBUG | ||||
| @@ -47,9 +47,10 @@ | ||||
|  #endif | ||||
| #endif | ||||
| #include <WiFiUdp.h> | ||||
| #include "../network/Network.h" | ||||
|  | ||||
| #ifdef ESPALEXA_DEBUG | ||||
|  #pragma message "Espalexa 2.4.4 debug mode" | ||||
|  #pragma message "Espalexa 2.7.0 debug mode" | ||||
|  #define EA_DEBUG(x)  Serial.print (x) | ||||
|  #define EA_DEBUGLN(x) Serial.println (x) | ||||
| #else | ||||
| @@ -59,6 +60,7 @@ | ||||
|  | ||||
| #include "EspalexaDevice.h" | ||||
|  | ||||
| #define DEVICE_UNIQUE_ID_LENGTH 12 | ||||
|  | ||||
| class Espalexa { | ||||
| private: | ||||
| @@ -74,42 +76,37 @@ private: | ||||
|   #endif | ||||
|   uint8_t currentDeviceCount = 0; | ||||
|   bool discoverable = true; | ||||
|   bool udpConnected = false; | ||||
|  | ||||
|   EspalexaDevice* devices[ESPALEXA_MAXDEVICES] = {}; | ||||
|   //Keep in mind that Device IDs go from 1 to DEVICES, cpp arrays from 0 to DEVICES-1!! | ||||
|    | ||||
|   WiFiUDP espalexaUdp; | ||||
|   IPAddress ipMulti; | ||||
|   bool udpConnected = false; | ||||
|   char packetBuffer[255]; //buffer to hold incoming udp packet | ||||
|   uint32_t mac24; //bottom 24 bits of mac | ||||
|   String escapedMac=""; //lowercase mac address | ||||
|    | ||||
|   //private member functions | ||||
|   String boolString(bool st) | ||||
|   { | ||||
|     return(st)?"true":"false"; | ||||
|   } | ||||
|    | ||||
|   String modeString(EspalexaColorMode m) | ||||
|   const char* modeString(EspalexaColorMode m) | ||||
|   { | ||||
|     if (m == EspalexaColorMode::xy) return "xy"; | ||||
|     if (m == EspalexaColorMode::hs) return "hs"; | ||||
|     return "ct"; | ||||
|   } | ||||
|    | ||||
|   String typeString(EspalexaDeviceType t) | ||||
|   const char* typeString(EspalexaDeviceType t) | ||||
|   { | ||||
|     switch (t) | ||||
|     { | ||||
|       case EspalexaDeviceType::dimmable:      return "Dimmable light"; | ||||
|       case EspalexaDeviceType::whitespectrum: return "Color temperature light"; | ||||
|       case EspalexaDeviceType::color:         return "Color light"; | ||||
|       case EspalexaDeviceType::extendedcolor: return "Extended color light"; | ||||
|       case EspalexaDeviceType::dimmable:      return PSTR("Dimmable light"); | ||||
|       case EspalexaDeviceType::whitespectrum: return PSTR("Color temperature light"); | ||||
|       case EspalexaDeviceType::color:         return PSTR("Color light"); | ||||
|       case EspalexaDeviceType::extendedcolor: return PSTR("Extended color light"); | ||||
|       default: return ""; | ||||
|     } | ||||
|     return "Light"; | ||||
|   } | ||||
|    | ||||
|   String modelidString(EspalexaDeviceType t) | ||||
|   const char* modelidString(EspalexaDeviceType t) | ||||
|   { | ||||
|     switch (t) | ||||
|     { | ||||
| @@ -117,56 +114,63 @@ private: | ||||
|       case EspalexaDeviceType::whitespectrum: return "LWT010"; | ||||
|       case EspalexaDeviceType::color:         return "LST001"; | ||||
|       case EspalexaDeviceType::extendedcolor: return "LCT015"; | ||||
|       default: return ""; | ||||
|     } | ||||
|     return "Plug"; | ||||
|   } | ||||
|    | ||||
|   //Workaround functions courtesy of Sonoff-Tasmota | ||||
|   uint32_t encodeLightId(uint8_t idx) | ||||
|   void encodeLightId(uint8_t idx, char* out) | ||||
|   { | ||||
|     uint8_t mac[6]; | ||||
|     WiFi.macAddress(mac); | ||||
|     uint32_t id = (mac[3] << 20) | (mac[4] << 12) | (mac[5] << 4) | (idx & 0xF); | ||||
|     return id; | ||||
|  | ||||
|     sprintf_P(out, PSTR("%02X:%02X:%02X:%02X:%02X:%02X:00:11-%02X"), mac[0],mac[1],mac[2],mac[3],mac[4],mac[5], idx); | ||||
|   } | ||||
|  | ||||
|   uint32_t decodeLightId(uint32_t id) { | ||||
|     return id & 0xF; | ||||
|   } | ||||
|    | ||||
|   //device JSON string: color+temperature device emulates LCT015, dimmable device LWB010, (TODO: on/off Plug 01, color temperature device LWT010, color device LST001) | ||||
|   String deviceJsonString(uint8_t deviceId) | ||||
|   // construct 'globally unique' Json dict key fitting into signed int | ||||
|   inline int encodeLightKey(uint8_t idx) | ||||
|   { | ||||
|     deviceId--; | ||||
|     if (deviceId >= currentDeviceCount) return "{}"; //error | ||||
|     EspalexaDevice* dev = devices[deviceId]; | ||||
|     //return idx +1; | ||||
|     static_assert(ESPALEXA_MAXDEVICES <= 128, ""); | ||||
|     return (mac24<<7) | idx; | ||||
|   } | ||||
|  | ||||
|     String json = "{\"state\":{\"on\":"; | ||||
|     json += boolString(dev->getValue()); | ||||
|     if (dev->getType() != EspalexaDeviceType::onoff) //bri support | ||||
|     { | ||||
|       json += ",\"bri\":" + String(dev->getLastValue()-1); | ||||
|       if (static_cast<uint8_t>(dev->getType()) > 2) //color support | ||||
|       { | ||||
|         json += ",\"hue\":" + String(dev->getHue()) + ",\"sat\":" + String(dev->getSat()); | ||||
|         json += ",\"effect\":\"none\",\"xy\":[" + String(dev->getX()) + "," + String(dev->getY()) + "]"; | ||||
|       } | ||||
|       if (static_cast<uint8_t>(dev->getType()) > 1 && dev->getType() != EspalexaDeviceType::color) //white spectrum support | ||||
|       { | ||||
|         json += ",\"ct\":" + String(dev->getCt()); | ||||
|       } | ||||
|     } | ||||
|     json += ",\"alert\":\"none"; | ||||
|     if (static_cast<uint8_t>(dev->getType()) > 1) json += "\",\"colormode\":\"" + modeString(dev->getColorMode()); | ||||
|     json += "\",\"mode\":\"homeautomation\",\"reachable\":true},"; | ||||
|     json += "\"type\":\"" + typeString(dev->getType()); | ||||
|     json += "\",\"name\":\"" + dev->getName(); | ||||
|     json += "\",\"modelid\":\"" + modelidString(dev->getType()); | ||||
|     json += "\",\"manufacturername\":\"Philips\",\"productname\":\"E" + String(static_cast<uint8_t>(dev->getType())); | ||||
|     json += "\",\"uniqueid\":\"" + String(encodeLightId(deviceId+1)); | ||||
|     json += "\",\"swversion\":\"espalexa-2.4.4\"}"; | ||||
|   // get device index from Json key | ||||
|   uint8_t decodeLightKey(int key) | ||||
|   { | ||||
|     //return key -1; | ||||
|     return (((uint32_t)key>>7) == mac24) ? (key & 127U) : 255U; | ||||
|   } | ||||
|  | ||||
|   //device JSON string: color+temperature device emulates LCT015, dimmable device LWB010, (TODO: on/off Plug 01, color temperature device LWT010, color device LST001) | ||||
|   void deviceJsonString(EspalexaDevice* dev, char* buf) | ||||
|   { | ||||
|     char buf_lightid[27]; | ||||
|     encodeLightId(dev->getId() + 1, buf_lightid); | ||||
|      | ||||
|     return json; | ||||
|     char buf_col[80] = ""; | ||||
|     //color support | ||||
|     if (static_cast<uint8_t>(dev->getType()) > 2) | ||||
|       //TODO: %f is not working for some reason on ESP8266 in v0.11.0 (was fine in 0.10.2). Need to investigate | ||||
|       //sprintf_P(buf_col,PSTR(",\"hue\":%u,\"sat\":%u,\"effect\":\"none\",\"xy\":[%f,%f]") | ||||
|       //  ,dev->getHue(), dev->getSat(), dev->getX(), dev->getY()); | ||||
|       sprintf_P(buf_col,PSTR(",\"hue\":%u,\"sat\":%u,\"effect\":\"none\",\"xy\":[%s,%s]"),dev->getHue(), dev->getSat(), | ||||
|         ((String)dev->getX()).c_str(), ((String)dev->getY()).c_str()); | ||||
|        | ||||
|     char buf_ct[16] = ""; | ||||
|     //white spectrum support | ||||
|     if (static_cast<uint8_t>(dev->getType()) > 1 && dev->getType() != EspalexaDeviceType::color) | ||||
|       sprintf(buf_ct, ",\"ct\":%u", dev->getCt()); | ||||
|      | ||||
|     char buf_cm[20] = ""; | ||||
|     if (static_cast<uint8_t>(dev->getType()) > 1) | ||||
|       sprintf(buf_cm,PSTR("\",\"colormode\":\"%s"), modeString(dev->getColorMode())); | ||||
|      | ||||
|     sprintf_P(buf, PSTR("{\"state\":{\"on\":%s,\"bri\":%u%s%s,\"alert\":\"none%s\",\"mode\":\"homeautomation\",\"reachable\":true}," | ||||
|                    "\"type\":\"%s\",\"name\":\"%s\",\"modelid\":\"%s\",\"manufacturername\":\"Philips\",\"productname\":\"E%u" | ||||
|                    "\",\"uniqueid\":\"%s\",\"swversion\":\"espalexa-2.7.0\"}") | ||||
|                     | ||||
|     , (dev->getValue())?"true":"false", dev->getLastValue()-1, buf_col, buf_ct, buf_cm, typeString(dev->getType()), | ||||
|     dev->getName().c_str(), modelidString(dev->getType()), static_cast<uint8_t>(dev->getType()), buf_lightid); | ||||
|   } | ||||
|    | ||||
|   //Espalexa status page /espalexa | ||||
| @@ -181,14 +185,14 @@ private: | ||||
|       res += "Value of device " + String(i+1) + " (" + dev->getName() + "): " + String(dev->getValue()) + " (" + typeString(dev->getType()); | ||||
|       if (static_cast<uint8_t>(dev->getType()) > 1) //color support | ||||
|       { | ||||
|         res += ", colormode=" + modeString(dev->getColorMode()) + ", r=" + String(dev->getR()) + ", g=" + String(dev->getG()) + ", b=" + String(dev->getB()); | ||||
|         res += ", colormode=" + String(modeString(dev->getColorMode())) + ", r=" + String(dev->getR()) + ", g=" + String(dev->getG()) + ", b=" + String(dev->getB()); | ||||
|         res +=", ct=" + String(dev->getCt()) + ", hue=" + String(dev->getHue()) + ", sat=" + String(dev->getSat()) + ", x=" + String(dev->getX()) + ", y=" + String(dev->getY()); | ||||
|       } | ||||
|       res += ")\r\n"; | ||||
|     } | ||||
|     res += "\r\nFree Heap: " + (String)ESP.getFreeHeap(); | ||||
|     res += "\r\nUptime: " + (String)millis(); | ||||
|     res += "\r\n\r\nEspalexa library v2.4.4 by Christian Schwinne 2020"; | ||||
|     res += "\r\n\r\nEspalexa library v2.7.0 by Christian Schwinne 2021"; | ||||
|     server->send(200, "text/plain", res); | ||||
|   } | ||||
|   #endif | ||||
| @@ -206,40 +210,41 @@ private: | ||||
|     EA_DEBUGLN("Body: " + body); | ||||
|     if(!handleAlexaApiCall(server)) | ||||
|     #endif | ||||
|       server->send(404, "text/plain", "Not Found (espalexa-internal)"); | ||||
|       server->send(404, "text/plain", "Not Found (espalexa)"); | ||||
|   } | ||||
|  | ||||
|   //send description.xml device property page | ||||
|   void serveDescription() | ||||
|   { | ||||
|     EA_DEBUGLN("# Responding to description.xml ... #\n"); | ||||
|     IPAddress localIP = WiFi.localIP(); | ||||
|     IPAddress localIP = Network.localIP(); | ||||
|     char s[16]; | ||||
|     sprintf(s, "%d.%d.%d.%d", localIP[0], localIP[1], localIP[2], localIP[3]); | ||||
|  | ||||
|     String setup_xml = "<?xml version=\"1.0\" ?>" | ||||
|     char buf[1024]; | ||||
|      | ||||
|     sprintf_P(buf,PSTR("<?xml version=\"1.0\" ?>" | ||||
|         "<root xmlns=\"urn:schemas-upnp-org:device-1-0\">" | ||||
|         "<specVersion><major>1</major><minor>0</minor></specVersion>" | ||||
|         "<URLBase>http://"+ String(s) +":80/</URLBase>" | ||||
|         "<URLBase>http://%s:80/</URLBase>" | ||||
|         "<device>" | ||||
|           "<deviceType>urn:schemas-upnp-org:device:Basic:1</deviceType>" | ||||
|           "<friendlyName>Espalexa ("+ String(s) +")</friendlyName>" | ||||
|           "<friendlyName>Espalexa (%s:80)</friendlyName>" | ||||
|           "<manufacturer>Royal Philips Electronics</manufacturer>" | ||||
|           "<manufacturerURL>http://www.philips.com</manufacturerURL>" | ||||
|           "<modelDescription>Philips hue Personal Wireless Lighting</modelDescription>" | ||||
|           "<modelName>Philips hue bridge 2012</modelName>" | ||||
|           "<modelNumber>929000226503</modelNumber>" | ||||
|           "<modelURL>http://www.meethue.com</modelURL>" | ||||
|           "<serialNumber>"+ escapedMac +"</serialNumber>" | ||||
|           "<UDN>uuid:2f402f80-da50-11e1-9b23-"+ escapedMac +"</UDN>" | ||||
|           "<serialNumber>%s</serialNumber>" | ||||
|           "<UDN>uuid:2f402f80-da50-11e1-9b23-%s</UDN>" | ||||
|           "<presentationURL>index.html</presentationURL>" | ||||
|         "</device>" | ||||
|         "</root>"; | ||||
|         "</root>"),s,s,escapedMac.c_str(),escapedMac.c_str()); | ||||
|            | ||||
|     server->send(200, "text/xml", setup_xml.c_str()); | ||||
|     server->send(200, "text/xml", buf); | ||||
|      | ||||
|     EA_DEBUG("Sending :"); | ||||
|     EA_DEBUGLN(setup_xml); | ||||
|     EA_DEBUGLN("Send setup.xml"); | ||||
|     EA_DEBUGLN(buf); | ||||
|   } | ||||
|    | ||||
|   //init the server | ||||
| @@ -286,26 +291,27 @@ private: | ||||
|   //respond to UDP SSDP M-SEARCH | ||||
|   void respondToSearch() | ||||
|   { | ||||
|     IPAddress localIP = WiFi.localIP(); | ||||
|     IPAddress localIP = Network.localIP(); | ||||
|     char s[16]; | ||||
|     sprintf(s, "%d.%d.%d.%d", localIP[0], localIP[1], localIP[2], localIP[3]); | ||||
|  | ||||
|     String response =  | ||||
|       "HTTP/1.1 200 OK\r\n" | ||||
|     char buf[1024]; | ||||
|  | ||||
|     sprintf_P(buf,PSTR("HTTP/1.1 200 OK\r\n" | ||||
|       "EXT:\r\n" | ||||
|       "CACHE-CONTROL: max-age=100\r\n" // SSDP_INTERVAL | ||||
|       "LOCATION: http://"+ String(s) +":80/description.xml\r\n" | ||||
|       "LOCATION: http://%s:80/description.xml\r\n" | ||||
|       "SERVER: FreeRTOS/6.0.5, UPnP/1.0, IpBridge/1.17.0\r\n" // _modelName, _modelNumber | ||||
|       "hue-bridgeid: "+ escapedMac +"\r\n" | ||||
|       "hue-bridgeid: %s\r\n" | ||||
|       "ST: urn:schemas-upnp-org:device:basic:1\r\n"  // _deviceType | ||||
|       "USN: uuid:2f402f80-da50-11e1-9b23-"+ escapedMac +"::ssdp:all\r\n" // _uuid::_deviceType | ||||
|       "\r\n"; | ||||
|       "USN: uuid:2f402f80-da50-11e1-9b23-%s::upnp:rootdevice\r\n" // _uuid::_deviceType | ||||
|       "\r\n"),s,escapedMac.c_str(),escapedMac.c_str()); | ||||
|  | ||||
|     espalexaUdp.beginPacket(espalexaUdp.remoteIP(), espalexaUdp.remotePort()); | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|     espalexaUdp.write((uint8_t*)response.c_str(), response.length()); | ||||
|     espalexaUdp.write((uint8_t*)buf, strlen(buf)); | ||||
|     #else | ||||
|     espalexaUdp.write(response.c_str()); | ||||
|     espalexaUdp.write(buf); | ||||
|     #endif | ||||
|     espalexaUdp.endPacket();                     | ||||
|   } | ||||
| @@ -329,6 +335,9 @@ public: | ||||
|     escapedMac.replace(":", ""); | ||||
|     escapedMac.toLowerCase(); | ||||
|  | ||||
|     String macSubStr = escapedMac.substring(6, 12); | ||||
|     mac24 = strtol(macSubStr.c_str(), 0, 16); | ||||
|  | ||||
|     #ifdef ESPALEXA_ASYNC | ||||
|     serverAsync = externalServer; | ||||
|     #else | ||||
| @@ -337,7 +346,7 @@ public: | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|     udpConnected = espalexaUdp.beginMulticast(IPAddress(239, 255, 255, 250), 1900); | ||||
|     #else | ||||
|     udpConnected = espalexaUdp.beginMulticast(WiFi.localIP(), IPAddress(239, 255, 255, 250), 1900); | ||||
|     udpConnected = espalexaUdp.beginMulticast(Network.localIP(), IPAddress(239, 255, 255, 250), 1900); | ||||
|     #endif | ||||
|  | ||||
|     if (udpConnected){ | ||||
| @@ -359,68 +368,80 @@ public: | ||||
|      | ||||
|     if (!udpConnected) return;    | ||||
|     int packetSize = espalexaUdp.parsePacket();     | ||||
|     if (!packetSize) return; //no new udp packet | ||||
|     if (packetSize < 1) return; //no new udp packet | ||||
|      | ||||
|     EA_DEBUGLN("Got UDP!"); | ||||
|     int len = espalexaUdp.read(packetBuffer, 254); | ||||
|     if (len > 0) { | ||||
|       packetBuffer[len] = 0; | ||||
|     } | ||||
|  | ||||
|     unsigned char packetBuffer[packetSize+1]; //buffer to hold incoming udp packet | ||||
|     espalexaUdp.read(packetBuffer, packetSize); | ||||
|     packetBuffer[packetSize] = 0; | ||||
|    | ||||
|     espalexaUdp.flush(); | ||||
|     if (!discoverable) return; //do not reply to M-SEARCH if not discoverable | ||||
|      | ||||
|     String request = packetBuffer; | ||||
|     if(request.indexOf("M-SEARCH") >= 0) { | ||||
|       EA_DEBUGLN(request); | ||||
|       if(request.indexOf("upnp:rootdevice") > 0 || request.indexOf("asic:1") > 0 || request.indexOf("ssdp:all") > 0) { | ||||
|         EA_DEBUGLN("Responding search req..."); | ||||
|         respondToSearch(); | ||||
|       } | ||||
|    | ||||
|     const char* request = (const char *) packetBuffer; | ||||
|     if (strstr(request, "M-SEARCH") == nullptr) return; | ||||
|  | ||||
|     EA_DEBUGLN(request); | ||||
|     if (strstr(request, "ssdp:disc")  != nullptr &&  //short for "ssdp:discover" | ||||
|         (strstr(request, "upnp:rootd") != nullptr || //short for "upnp:rootdevice" | ||||
|          strstr(request, "ssdp:all")   != nullptr || | ||||
|          strstr(request, "asic:1")     != nullptr )) //short for "device:basic:1" | ||||
|     { | ||||
|       EA_DEBUGLN("Responding search req..."); | ||||
|       respondToSearch(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   bool addDevice(EspalexaDevice* d) | ||||
|   // returns device index or 0 on failure | ||||
|   uint8_t addDevice(EspalexaDevice* d) | ||||
|   { | ||||
|     EA_DEBUG("Adding device "); | ||||
|     EA_DEBUGLN((currentDeviceCount+1)); | ||||
|     if (currentDeviceCount >= ESPALEXA_MAXDEVICES) return false; | ||||
|     if (d == nullptr) return false; | ||||
|     if (currentDeviceCount >= ESPALEXA_MAXDEVICES) return 0; | ||||
|     if (d == nullptr) return 0; | ||||
|     d->setId(currentDeviceCount); | ||||
|     devices[currentDeviceCount] = d; | ||||
|     currentDeviceCount++; | ||||
|     return true; | ||||
|     return ++currentDeviceCount; | ||||
|   } | ||||
|    | ||||
|   //brightness-only callback | ||||
|   bool addDevice(String deviceName, BrightnessCallbackFunction callback, uint8_t initialValue = 0) | ||||
|   uint8_t addDevice(String deviceName, BrightnessCallbackFunction callback, uint8_t initialValue = 0) | ||||
|   { | ||||
|     EA_DEBUG("Constructing device "); | ||||
|     EA_DEBUGLN((currentDeviceCount+1)); | ||||
|     if (currentDeviceCount >= ESPALEXA_MAXDEVICES) return false; | ||||
|     if (currentDeviceCount >= ESPALEXA_MAXDEVICES) return 0; | ||||
|     EspalexaDevice* d = new EspalexaDevice(deviceName, callback, initialValue); | ||||
|     return addDevice(d); | ||||
|   } | ||||
|    | ||||
|   //brightness-only callback | ||||
|   bool addDevice(String deviceName, ColorCallbackFunction callback, uint8_t initialValue = 0) | ||||
|   uint8_t addDevice(String deviceName, ColorCallbackFunction callback, uint8_t initialValue = 0) | ||||
|   { | ||||
|     EA_DEBUG("Constructing device "); | ||||
|     EA_DEBUGLN((currentDeviceCount+1)); | ||||
|     if (currentDeviceCount >= ESPALEXA_MAXDEVICES) return false; | ||||
|     if (currentDeviceCount >= ESPALEXA_MAXDEVICES) return 0; | ||||
|     EspalexaDevice* d = new EspalexaDevice(deviceName, callback, initialValue); | ||||
|     return addDevice(d); | ||||
|   } | ||||
|  | ||||
|  | ||||
|   bool addDevice(String deviceName, DeviceCallbackFunction callback, EspalexaDeviceType t = EspalexaDeviceType::dimmable, uint8_t initialValue = 0) | ||||
|   uint8_t addDevice(String deviceName, DeviceCallbackFunction callback, EspalexaDeviceType t = EspalexaDeviceType::dimmable, uint8_t initialValue = 0) | ||||
|   { | ||||
|     EA_DEBUG("Constructing device "); | ||||
|     EA_DEBUGLN((currentDeviceCount+1)); | ||||
|     if (currentDeviceCount >= ESPALEXA_MAXDEVICES) return false; | ||||
|     if (currentDeviceCount >= ESPALEXA_MAXDEVICES) return 0; | ||||
|     EspalexaDevice* d = new EspalexaDevice(deviceName, callback, t, initialValue); | ||||
|     return addDevice(d); | ||||
|   } | ||||
|    | ||||
|  | ||||
|   void renameDevice(uint8_t id, const String& deviceName) | ||||
|   { | ||||
|     unsigned int index = id - 1; | ||||
|     if (index < currentDeviceCount) | ||||
|       devices[index]->setName(deviceName); | ||||
|   } | ||||
|  | ||||
|   //basic implementation of Philips hue api functions needed for basic Alexa control | ||||
|   #ifdef ESPALEXA_ASYNC | ||||
|   bool handleAlexaApiCall(AsyncWebServerRequest* request) | ||||
| @@ -439,6 +460,8 @@ public: | ||||
|   bool handleAlexaApiCall(String req, String body) | ||||
|   {   | ||||
|   #endif | ||||
|     EA_DEBUG("URL: "); | ||||
|     EA_DEBUGLN(req); | ||||
|     EA_DEBUGLN("AlexaApiCall"); | ||||
|     if (req.indexOf("api") <0) return false; //return if not an API call | ||||
|     EA_DEBUGLN("ok"); | ||||
| @@ -447,35 +470,36 @@ public: | ||||
|     { | ||||
|       EA_DEBUGLN("devType"); | ||||
|       body = ""; | ||||
|       server->send(200, "application/json", "[{\"success\":{\"username\":\"2WLEDHardQrI3WHYTHoMcXHgEspsM8ZZRpSKtBQr\"}}]"); | ||||
|       server->send(200, "application/json", F("[{\"success\":{\"username\":\"2BLEDHardQrI3WHYTHoMcXHgEspsM8ZZRpSKtBGr\"}}]")); | ||||
|       return true; | ||||
|     } | ||||
|  | ||||
|     if (req.indexOf("state") > 0) //client wants to control light | ||||
|     if ((req.indexOf("state") > 0) && (body.length() > 0)) //client wants to control light | ||||
|     { | ||||
|       server->send(200, "application/json", "[{\"success\":{\"/lights/1/state/\": true}}]"); | ||||
|  | ||||
|       uint32_t devId = req.substring(req.indexOf("lights")+7).toInt(); | ||||
|       EA_DEBUG("ls"); EA_DEBUGLN(devId); | ||||
|       devId = decodeLightId(devId); | ||||
|       EA_DEBUGLN(devId); | ||||
|       devId--; //zero-based for devices array | ||||
|       if (devId >= currentDeviceCount) return true; //return if invalid ID | ||||
|       unsigned idx = decodeLightKey(devId); | ||||
|       EA_DEBUGLN(idx); | ||||
|       char buf[50]; | ||||
|       sprintf_P(buf,PSTR("[{\"success\":{\"/lights/%u/state/\": true}}]"),devId); | ||||
|       server->send(200, "application/json", buf); | ||||
|       if (idx >= currentDeviceCount) return true; //return if invalid ID | ||||
|       EspalexaDevice* dev = devices[idx]; | ||||
|        | ||||
|       devices[devId]->setPropertyChanged(EspalexaDeviceProperty::none); | ||||
|       dev->setPropertyChanged(EspalexaDeviceProperty::none); | ||||
|        | ||||
|       if (body.indexOf("false")>0) //OFF command | ||||
|       { | ||||
|         devices[devId]->setValue(0); | ||||
|         devices[devId]->setPropertyChanged(EspalexaDeviceProperty::off); | ||||
|         devices[devId]->doCallback(); | ||||
|         dev->setValue(0); | ||||
|         dev->setPropertyChanged(EspalexaDeviceProperty::off); | ||||
|         dev->doCallback(); | ||||
|         return true; | ||||
|       } | ||||
|        | ||||
|       if (body.indexOf("true") >0) //ON command | ||||
|       { | ||||
|         devices[devId]->setValue(devices[devId]->getLastValue()); | ||||
|         devices[devId]->setPropertyChanged(EspalexaDeviceProperty::on); | ||||
|         dev->setValue(dev->getLastValue()); | ||||
|         dev->setPropertyChanged(EspalexaDeviceProperty::on); | ||||
|       } | ||||
|        | ||||
|       if (body.indexOf("bri")  >0) //BRIGHTNESS command | ||||
| @@ -483,35 +507,35 @@ public: | ||||
|         uint8_t briL = body.substring(body.indexOf("bri") +5).toInt(); | ||||
|         if (briL == 255) | ||||
|         { | ||||
|          devices[devId]->setValue(255); | ||||
|          dev->setValue(255); | ||||
|         } else { | ||||
|          devices[devId]->setValue(briL+1);  | ||||
|          dev->setValue(briL+1);  | ||||
|         } | ||||
|         devices[devId]->setPropertyChanged(EspalexaDeviceProperty::bri); | ||||
|         dev->setPropertyChanged(EspalexaDeviceProperty::bri); | ||||
|       } | ||||
|        | ||||
|       if (body.indexOf("xy")   >0) //COLOR command (XY mode) | ||||
|       { | ||||
|         devices[devId]->setColorXY(body.substring(body.indexOf("[") +1).toFloat(), body.substring(body.indexOf(",0") +1).toFloat()); | ||||
|         devices[devId]->setPropertyChanged(EspalexaDeviceProperty::xy); | ||||
|         dev->setColorXY(body.substring(body.indexOf("[") +1).toFloat(), body.substring(body.indexOf(",0") +1).toFloat()); | ||||
|         dev->setPropertyChanged(EspalexaDeviceProperty::xy); | ||||
|       } | ||||
|        | ||||
|       if (body.indexOf("hue")  >0) //COLOR command (HS mode) | ||||
|       { | ||||
|         devices[devId]->setColor(body.substring(body.indexOf("hue") +5).toInt(), body.substring(body.indexOf("sat") +5).toInt()); | ||||
|         devices[devId]->setPropertyChanged(EspalexaDeviceProperty::hs); | ||||
|         dev->setColor(body.substring(body.indexOf("hue") +5).toInt(), body.substring(body.indexOf("sat") +5).toInt()); | ||||
|         dev->setPropertyChanged(EspalexaDeviceProperty::hs); | ||||
|       } | ||||
|        | ||||
|       if (body.indexOf("ct")   >0) //COLOR TEMP command (white spectrum) | ||||
|       { | ||||
|         devices[devId]->setColor(body.substring(body.indexOf("ct") +4).toInt()); | ||||
|         devices[devId]->setPropertyChanged(EspalexaDeviceProperty::ct); | ||||
|         dev->setColor(body.substring(body.indexOf("ct") +4).toInt()); | ||||
|         dev->setPropertyChanged(EspalexaDeviceProperty::ct); | ||||
|       } | ||||
|        | ||||
|       devices[devId]->doCallback(); | ||||
|       dev->doCallback(); | ||||
|        | ||||
|       #ifdef ESPALEXA_DEBUG | ||||
|       if (devices[devId]->getLastChangedProperty() == EspalexaDeviceProperty::none) | ||||
|       if (dev->getLastChangedProperty() == EspalexaDeviceProperty::none) | ||||
|         EA_DEBUGLN("STATE REQ WITHOUT BODY (likely Content-Type issue #6)"); | ||||
|       #endif | ||||
|       return true; | ||||
| @@ -529,22 +553,31 @@ public: | ||||
|         String jsonTemp = "{"; | ||||
|         for (int i = 0; i<currentDeviceCount; i++) | ||||
|         { | ||||
|           jsonTemp += "\"" + String(encodeLightId(i+1)) + "\":"; | ||||
|           jsonTemp += deviceJsonString(i+1); | ||||
|           if (i < currentDeviceCount-1) jsonTemp += ","; | ||||
|           jsonTemp += '"'; | ||||
|           jsonTemp += encodeLightKey(i); | ||||
|           jsonTemp += '"'; | ||||
|           jsonTemp += ':'; | ||||
|  | ||||
|           char buf[512]; | ||||
|           deviceJsonString(devices[i], buf); | ||||
|           jsonTemp += buf; | ||||
|           if (i < currentDeviceCount-1) jsonTemp += ','; | ||||
|         } | ||||
|         jsonTemp += "}"; | ||||
|         jsonTemp += '}'; | ||||
|         server->send(200, "application/json", jsonTemp); | ||||
|       } else //client wants one light (devId) | ||||
|       { | ||||
|         devId = decodeLightId(devId); | ||||
|         EA_DEBUGLN(devId); | ||||
|         if (devId > currentDeviceCount) | ||||
|         { | ||||
|         unsigned int idx = decodeLightKey(devId); | ||||
|  | ||||
|         if (idx >= currentDeviceCount) idx = 0; //send first device if invalid | ||||
|         if (currentDeviceCount == 0) { | ||||
|           server->send(200, "application/json", "{}"); | ||||
|         } else { | ||||
|           server->send(200, "application/json", deviceJsonString(devId)); | ||||
|           return true; | ||||
|         } | ||||
|         char buf[512]; | ||||
|         deviceJsonString(devices[idx], buf); | ||||
|         server->send(200, "application/json", buf); | ||||
|       } | ||||
|        | ||||
|       return true; | ||||
| @@ -581,7 +614,7 @@ public: | ||||
|     return perc / 255; | ||||
|   } | ||||
|    | ||||
|   ~Espalexa(){delete devices;} //note: Espalexa is NOT meant to be destructed | ||||
|   ~Espalexa(){} //note: Espalexa is NOT meant to be destructed | ||||
| }; | ||||
|  | ||||
| #endif | ||||
|   | ||||
| @@ -28,6 +28,7 @@ EspalexaDevice::EspalexaDevice(String deviceName, DeviceCallbackFunction gnCallb | ||||
|   _callbackDev = gnCallback; | ||||
|   _type = t; | ||||
|   if (t == EspalexaDeviceType::onoff) _type = EspalexaDeviceType::dimmable; //on/off is broken, so make dimmable device instead | ||||
|   if (t == EspalexaDeviceType::whitespectrum) _mode = EspalexaColorMode::ct; | ||||
|   _val = initialValue; | ||||
|   _val_last = _val; | ||||
| } | ||||
| @@ -64,6 +65,11 @@ uint8_t EspalexaDevice::getValue() | ||||
|   return _val; | ||||
| } | ||||
|  | ||||
| bool EspalexaDevice::getState() | ||||
| { | ||||
|   return _val; | ||||
| } | ||||
|  | ||||
| uint8_t EspalexaDevice::getPercent() | ||||
| { | ||||
|   uint16_t perc = _val * 100; | ||||
| @@ -110,8 +116,7 @@ uint32_t EspalexaDevice::getKelvin() | ||||
| uint32_t EspalexaDevice::getRGB() | ||||
| { | ||||
|   if (_rgb != 0) return _rgb; //color has not changed | ||||
|   byte rgb[4]{0, 0, 0, 0};  | ||||
|   float r, g, b, w; | ||||
|   byte rgb[4]{0, 0, 0, 0}; | ||||
|    | ||||
|   if (_mode == EspalexaColorMode::none) return 0; | ||||
|  | ||||
| @@ -122,37 +127,28 @@ uint32_t EspalexaDevice::getRGB() | ||||
|     float temp = 10000/ _ct; //kelvins = 1,000,000/mired (and that /100) | ||||
|     float r, g, b; | ||||
|  | ||||
| // Cold white to warm white receiving from Alexa: _ct = 199, 234, 284, 350, 383 (from cold white to warm white) | ||||
|     switch (_ct) { | ||||
|       case 199: rgb[0]=255,rgb[1]=255,rgb[2]=255;rgb[3]=255;break; | ||||
|       case 234: rgb[0]=127,rgb[1]=127,rgb[2]=127;rgb[3]=255;break; | ||||
|       case 284: rgb[0]=0,rgb[1]=0,rgb[2]=0;rgb[3]=255;break; | ||||
|       case 350: rgb[0]=130,rgb[1]=90,rgb[2]=0;rgb[3]=255;break; | ||||
|       case 383: rgb[0]=255,rgb[1]=153,rgb[2]=0;rgb[3]=255;break; | ||||
|       default: { | ||||
|         if( temp <= 66 ){  | ||||
|           r = 255;  | ||||
|           g = temp; | ||||
|           g = 99.470802 * log(g) - 161.119568; | ||||
|           if( temp <= 19){ | ||||
|               b = 0; | ||||
|           } else { | ||||
|               b = temp-10; | ||||
|               b = 138.517731 * log(b) - 305.044793; | ||||
|           } | ||||
|         } else { | ||||
|           r = temp - 60; | ||||
|           r = 329.698727 * pow(r, -0.13320476); | ||||
|           g = temp - 60; | ||||
|           g = 288.12217 * pow(g, -0.07551485 ); | ||||
|           b = 255; | ||||
|         } | ||||
|      | ||||
|         rgb[0] = (byte)constrain(r,0.1,255.1); | ||||
|         rgb[1] = (byte)constrain(g,0.1,255.1); | ||||
|         rgb[2] = (byte)constrain(b,0.1,255.1); | ||||
|     if (temp <= 66) {  | ||||
|       r = 255;  | ||||
|       g = temp; | ||||
|       g = 99.470802 * log(g) - 161.119568; | ||||
|       if (temp <= 19) { | ||||
|           b = 0; | ||||
|       } else { | ||||
|           b = temp-10; | ||||
|           b = 138.517731 * log(b) - 305.044793; | ||||
|       } | ||||
|     } else { | ||||
|       r = temp - 60; | ||||
|       r = 329.698727 * pow(r, -0.13320476); | ||||
|       g = temp - 60; | ||||
|       g = 288.12217 * pow(g, -0.07551485 ); | ||||
|       b = 255; | ||||
|     } | ||||
|      | ||||
|     rgb[0] = (byte)constrain(r,0.1,255.1); | ||||
|     rgb[1] = (byte)constrain(g,0.1,255.1); | ||||
|     rgb[2] = (byte)constrain(b,0.1,255.1); | ||||
|      | ||||
|   } else if (_mode == EspalexaColorMode::hs) | ||||
|   { | ||||
|     float h = ((float)_hue)/65535.0; | ||||
| @@ -226,7 +222,7 @@ uint32_t EspalexaDevice::getRGB() | ||||
|     rgb[1] = 255.0*g; | ||||
|     rgb[2] = 255.0*b; | ||||
|   } | ||||
|   _rgb = ((rgb[3] << 24) | (rgb[0] << 16) | (rgb[1] << 8) | (rgb[2]));  //white value is only >0 if Alexa did provide a CT value, RGB colors will not be touched. | ||||
|   _rgb = ((rgb[0] << 16) | (rgb[1] << 8) | (rgb[2])); | ||||
|   return _rgb; | ||||
| } | ||||
|  | ||||
| @@ -286,6 +282,16 @@ void EspalexaDevice::setValue(uint8_t val) | ||||
|   _val = val; | ||||
| } | ||||
|  | ||||
| void EspalexaDevice::setState(bool onoff) | ||||
| { | ||||
|   if (onoff)  | ||||
|   { | ||||
|     setValue(_val_last); | ||||
|   } else { | ||||
|     setValue(0); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void EspalexaDevice::setPercent(uint8_t perc) | ||||
| { | ||||
|   uint16_t val = perc * 255; | ||||
|   | ||||
| @@ -2,12 +2,13 @@ | ||||
| #define EspalexaDevice_h | ||||
|  | ||||
| #include "Arduino.h" | ||||
| #include <functional> | ||||
|  | ||||
| typedef class EspalexaDevice; | ||||
| class EspalexaDevice; | ||||
|  | ||||
| typedef void (*BrightnessCallbackFunction) (uint8_t b); | ||||
| typedef void (*DeviceCallbackFunction) (EspalexaDevice* d); | ||||
| typedef void (*ColorCallbackFunction) (uint8_t br, uint32_t col); | ||||
| typedef std::function<void(uint8_t b)> BrightnessCallbackFunction; | ||||
| typedef std::function<void(EspalexaDevice* d)> DeviceCallbackFunction; | ||||
| typedef std::function<void(uint8_t br, uint32_t col)> ColorCallbackFunction; | ||||
|  | ||||
| enum class EspalexaColorMode : uint8_t { none = 0, ct = 1, hs = 2, xy = 3 }; | ||||
| enum class EspalexaDeviceType : uint8_t { onoff = 0, dimmable = 1, whitespectrum = 2, color = 3, extendedcolor = 4 }; | ||||
| @@ -39,6 +40,8 @@ public: | ||||
|   uint8_t getId(); | ||||
|   EspalexaDeviceProperty getLastChangedProperty(); | ||||
|   uint8_t getValue(); | ||||
|   uint8_t getLastValue(); //last value that was not off (1-255) | ||||
|   bool    getState(); | ||||
|   uint8_t getPercent(); | ||||
|   uint8_t getDegrees(); | ||||
|   uint16_t getHue(); | ||||
| @@ -58,6 +61,7 @@ public: | ||||
|   void setId(uint8_t id); | ||||
|   void setPropertyChanged(EspalexaDeviceProperty p); | ||||
|   void setValue(uint8_t bri); | ||||
|   void setState(bool onoff); | ||||
|   void setPercent(uint8_t perc); | ||||
|   void setName(String name); | ||||
|   void setColor(uint16_t ct); | ||||
| @@ -66,8 +70,6 @@ public: | ||||
|   void setColor(uint8_t r, uint8_t g, uint8_t b); | ||||
|    | ||||
|   void doCallback(); | ||||
|    | ||||
|   uint8_t getLastValue(); //last value that was not off (1-255) | ||||
| }; | ||||
|  | ||||
| #endif | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -15,7 +15,11 @@ | ||||
| #include "ArduinoJson-v6.h" | ||||
| #include <Print.h> | ||||
|  | ||||
| #define DYNAMIC_JSON_DOCUMENT_SIZE 8192 | ||||
| #ifdef ESP8266 | ||||
|   #define DYNAMIC_JSON_DOCUMENT_SIZE 8192 | ||||
| #else | ||||
|   #define DYNAMIC_JSON_DOCUMENT_SIZE 16384 | ||||
| #endif | ||||
|  | ||||
| constexpr const char* JSON_MIMETYPE = "application/json"; | ||||
|  | ||||
| @@ -140,7 +144,7 @@ public: | ||||
|   virtual void handleBody(AsyncWebServerRequest *request, uint8_t *data, size_t len, size_t index, size_t total) override final { | ||||
|     if (_onRequest) { | ||||
|       _contentLength = total; | ||||
|       if (total > 0 && request->_tempObject == NULL && total < _maxContentLength) { | ||||
|       if (total > 0 && request->_tempObject == NULL && (int)total < _maxContentLength) { | ||||
|         request->_tempObject = malloc(total); | ||||
|       } | ||||
|       if (request->_tempObject != NULL) { | ||||
|   | ||||
							
								
								
									
										59
									
								
								wled00/src/dependencies/network/Network.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								wled00/src/dependencies/network/Network.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | ||||
| #include "Network.h" | ||||
|  | ||||
| IPAddress NetworkClass::localIP() | ||||
| { | ||||
| #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_USE_ETHERNET) | ||||
|   if (ETH.localIP()[0] != 0) { | ||||
|     return ETH.localIP(); | ||||
|   } | ||||
| #endif | ||||
|   if (WiFi.localIP()[0] != 0) { | ||||
|     return WiFi.localIP(); | ||||
|   } | ||||
|   return INADDR_NONE; | ||||
| } | ||||
|  | ||||
| IPAddress NetworkClass::subnetMask() | ||||
| { | ||||
| #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_USE_ETHERNET) | ||||
|   if (ETH.localIP()[0] != 0) { | ||||
|     return ETH.subnetMask(); | ||||
|   } | ||||
| #endif | ||||
|   if (WiFi.localIP()[0] != 0) { | ||||
|     return WiFi.subnetMask(); | ||||
|   } | ||||
|   return IPAddress(255, 255, 255, 0); | ||||
| } | ||||
|  | ||||
| IPAddress NetworkClass::gatewayIP() | ||||
| { | ||||
| #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_USE_ETHERNET) | ||||
|   if (ETH.localIP()[0] != 0) { | ||||
|       return ETH.gatewayIP(); | ||||
|   } | ||||
| #endif | ||||
|   if (WiFi.localIP()[0] != 0) { | ||||
|       return WiFi.gatewayIP(); | ||||
|   } | ||||
|   return INADDR_NONE; | ||||
| } | ||||
|  | ||||
| bool NetworkClass::isConnected() | ||||
| { | ||||
| #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_USE_ETHERNET) | ||||
|   return (WiFi.localIP()[0] != 0 && WiFi.status() == WL_CONNECTED) || ETH.localIP()[0] != 0; | ||||
| #else | ||||
|   return (WiFi.localIP()[0] != 0 && WiFi.status() == WL_CONNECTED); | ||||
| #endif | ||||
| } | ||||
|  | ||||
| bool NetworkClass::isEthernet() | ||||
| { | ||||
| #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_USE_ETHERNET) | ||||
|   return (ETH.localIP()[0] != 0); | ||||
| #endif | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| NetworkClass Network; | ||||
							
								
								
									
										23
									
								
								wled00/src/dependencies/network/Network.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								wled00/src/dependencies/network/Network.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| #ifdef ESP8266 | ||||
|   #include <ESP8266WiFi.h> | ||||
| #else // ESP32 | ||||
|   #include <WiFi.h> | ||||
|   #include <ETH.h> | ||||
| #endif | ||||
|  | ||||
| #ifndef Network_h | ||||
| #define Network_h | ||||
|  | ||||
| class NetworkClass | ||||
| { | ||||
| public: | ||||
|   IPAddress localIP(); | ||||
|   IPAddress subnetMask(); | ||||
|   IPAddress gatewayIP(); | ||||
|   bool isConnected(); | ||||
|   bool isEthernet(); | ||||
| }; | ||||
|  | ||||
| extern NetworkClass Network; | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										1506
									
								
								wled00/tv_colors.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1506
									
								
								wled00/tv_colors.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										461
									
								
								wled00/udp.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										461
									
								
								wled00/udp.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,461 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * UDP sync notifier / Realtime / Hyperion / TPM2.NET | ||||
|  */ | ||||
|  | ||||
| #define WLEDPACKETSIZE 29 | ||||
| #define UDP_IN_MAXSIZE 1472 | ||||
|  | ||||
| void notify(byte callMode, bool followUp) | ||||
| { | ||||
|   if (!udpConnected) return; | ||||
|   switch (callMode) | ||||
|   { | ||||
|     case NOTIFIER_CALL_MODE_INIT:          return; | ||||
|     case NOTIFIER_CALL_MODE_DIRECT_CHANGE: if (!notifyDirect) return; break; | ||||
|     case NOTIFIER_CALL_MODE_BUTTON:        if (!notifyButton) return; break; | ||||
|     case NOTIFIER_CALL_MODE_NIGHTLIGHT:    if (!notifyDirect) return; break; | ||||
|     case NOTIFIER_CALL_MODE_HUE:           if (!notifyHue)    return; break; | ||||
|     case NOTIFIER_CALL_MODE_PRESET_CYCLE:  if (!notifyDirect) return; break; | ||||
|     case NOTIFIER_CALL_MODE_BLYNK:         if (!notifyDirect) return; break; | ||||
|     case NOTIFIER_CALL_MODE_ALEXA:         if (!notifyAlexa)  return; break; | ||||
|     default: return; | ||||
|   } | ||||
|   byte udpOut[WLEDPACKETSIZE]; | ||||
|   udpOut[0] = 0; //0: wled notifier protocol 1: WARLS protocol | ||||
|   udpOut[1] = callMode; | ||||
|   udpOut[2] = bri; | ||||
|   udpOut[3] = col[0]; | ||||
|   udpOut[4] = col[1]; | ||||
|   udpOut[5] = col[2]; | ||||
|   udpOut[6] = nightlightActive; | ||||
|   udpOut[7] = nightlightDelayMins; | ||||
|   udpOut[8] = effectCurrent; | ||||
|   udpOut[9] = effectSpeed; | ||||
|   udpOut[10] = col[3]; | ||||
|   //compatibilityVersionByte:  | ||||
|   //0: old 1: supports white 2: supports secondary color | ||||
|   //3: supports FX intensity, 24 byte packet 4: supports transitionDelay 5: sup palette | ||||
|   //6: supports timebase syncing, 29 byte packet 7: supports tertiary color  | ||||
|   udpOut[11] = 7;  | ||||
|   udpOut[12] = colSec[0]; | ||||
|   udpOut[13] = colSec[1]; | ||||
|   udpOut[14] = colSec[2]; | ||||
|   udpOut[15] = colSec[3]; | ||||
|   udpOut[16] = effectIntensity; | ||||
|   udpOut[17] = (transitionDelay >> 0) & 0xFF; | ||||
|   udpOut[18] = (transitionDelay >> 8) & 0xFF; | ||||
|   udpOut[19] = effectPalette; | ||||
|   uint32_t colTer = strip.getSegment(strip.getMainSegmentId()).colors[2]; | ||||
|   udpOut[20] = (colTer >> 16) & 0xFF; | ||||
|   udpOut[21] = (colTer >>  8) & 0xFF; | ||||
|   udpOut[22] = (colTer >>  0) & 0xFF; | ||||
|   udpOut[23] = (colTer >> 24) & 0xFF; | ||||
|    | ||||
|   udpOut[24] = followUp; | ||||
|   uint32_t t = millis() + strip.timebase; | ||||
|   udpOut[25] = (t >> 24) & 0xFF; | ||||
|   udpOut[26] = (t >> 16) & 0xFF; | ||||
|   udpOut[27] = (t >>  8) & 0xFF; | ||||
|   udpOut[28] = (t >>  0) & 0xFF; | ||||
|    | ||||
|   IPAddress broadcastIp; | ||||
|   broadcastIp = ~uint32_t(Network.subnetMask()) | uint32_t(Network.gatewayIP()); | ||||
|  | ||||
|   notifierUdp.beginPacket(broadcastIp, udpPort); | ||||
|   notifierUdp.write(udpOut, WLEDPACKETSIZE); | ||||
|   notifierUdp.endPacket(); | ||||
|   notificationSentCallMode = callMode; | ||||
|   notificationSentTime = millis(); | ||||
|   notificationTwoRequired = (followUp)? false:notifyTwice; | ||||
| } | ||||
|  | ||||
|  | ||||
| void realtimeLock(uint32_t timeoutMs, byte md) | ||||
| { | ||||
|   if (!realtimeMode && !realtimeOverride){ | ||||
|     for (uint16_t i = 0; i < ledCount; i++) | ||||
|     { | ||||
|       strip.setPixelColor(i,0,0,0,0); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   realtimeTimeout = millis() + timeoutMs; | ||||
|   if (timeoutMs == 255001 || timeoutMs == 65000) realtimeTimeout = UINT32_MAX; | ||||
|   realtimeMode = md; | ||||
|  | ||||
|   if (arlsForceMaxBri && !realtimeOverride) strip.setBrightness(scaledBri(255)); | ||||
|   if (md == REALTIME_MODE_GENERIC) strip.show(); | ||||
| } | ||||
|  | ||||
|  | ||||
| #define TMP2NET_OUT_PORT 65442 | ||||
|  | ||||
| void sendTPM2Ack() { | ||||
|   notifierUdp.beginPacket(notifierUdp.remoteIP(), TMP2NET_OUT_PORT); | ||||
|   uint8_t response_ack = 0xac; | ||||
|   notifierUdp.write(&response_ack, 1); | ||||
|   notifierUdp.endPacket(); | ||||
| } | ||||
|  | ||||
|  | ||||
| void handleNotifications() | ||||
| { | ||||
|   //send second notification if enabled | ||||
|   if(udpConnected && notificationTwoRequired && millis()-notificationSentTime > 250){ | ||||
|     notify(notificationSentCallMode,true); | ||||
|   } | ||||
|    | ||||
|   if (e131NewData && millis() - strip.getLastShow() > 15) | ||||
|   { | ||||
|     e131NewData = false; | ||||
|     strip.show(); | ||||
|   } | ||||
|  | ||||
|   //unlock strip when realtime UDP times out | ||||
|   if (realtimeMode && millis() > realtimeTimeout) | ||||
|   { | ||||
|     if (realtimeOverride == REALTIME_OVERRIDE_ONCE) realtimeOverride = REALTIME_OVERRIDE_NONE; | ||||
|     strip.setBrightness(scaledBri(bri)); | ||||
|     realtimeMode = REALTIME_MODE_INACTIVE; | ||||
|     realtimeIP[0] = 0; | ||||
|   } | ||||
|  | ||||
|   //receive UDP notifications | ||||
|   if (!udpConnected) return; | ||||
|      | ||||
|   bool isSupp = false; | ||||
|   uint16_t packetSize = notifierUdp.parsePacket(); | ||||
|   if (!packetSize && udp2Connected) { | ||||
|     packetSize = notifier2Udp.parsePacket(); | ||||
|     isSupp = true; | ||||
|   } | ||||
|  | ||||
|   //hyperion / raw RGB | ||||
|   if (!packetSize && udpRgbConnected) { | ||||
|     packetSize = rgbUdp.parsePacket(); | ||||
|     if (packetSize) { | ||||
|       if (!receiveDirect) return; | ||||
|       if (packetSize > UDP_IN_MAXSIZE || packetSize < 3) return; | ||||
|       realtimeIP = rgbUdp.remoteIP(); | ||||
|       DEBUG_PRINTLN(rgbUdp.remoteIP()); | ||||
|       uint8_t lbuf[packetSize]; | ||||
|       rgbUdp.read(lbuf, packetSize); | ||||
|       realtimeLock(realtimeTimeoutMs, REALTIME_MODE_HYPERION); | ||||
|       if (realtimeOverride) return; | ||||
|       uint16_t id = 0; | ||||
|       for (uint16_t i = 0; i < packetSize -2; i += 3) | ||||
|       { | ||||
|         setRealtimePixel(id, lbuf[i], lbuf[i+1], lbuf[i+2], 0); | ||||
|          | ||||
|         id++; if (id >= ledCount) break; | ||||
|       } | ||||
|       strip.show(); | ||||
|       return; | ||||
|     }  | ||||
|   } | ||||
|  | ||||
|   if (!(receiveNotifications || receiveDirect)) return; | ||||
|    | ||||
|   //notifier and UDP realtime | ||||
|   if (!packetSize || packetSize > UDP_IN_MAXSIZE) return; | ||||
|   if (!isSupp && notifierUdp.remoteIP() == Network.localIP()) return; //don't process broadcasts we send ourselves | ||||
|  | ||||
|   uint8_t udpIn[packetSize +1]; | ||||
|   uint16_t len; | ||||
|   if (isSupp) len = notifier2Udp.read(udpIn, packetSize); | ||||
|   else        len =  notifierUdp.read(udpIn, packetSize); | ||||
|  | ||||
|   // WLED nodes info notifications | ||||
|   if (isSupp && udpIn[0] == 255 && udpIn[1] == 1 && len >= 40) { | ||||
|     if (!nodeListEnabled || notifier2Udp.remoteIP() == Network.localIP()) return; | ||||
|  | ||||
|     uint8_t unit = udpIn[39]; | ||||
|     NodesMap::iterator it = Nodes.find(unit); | ||||
|     if (it == Nodes.end() && Nodes.size() < WLED_MAX_NODES) { // Create a new element when not present | ||||
|       Nodes[unit].age = 0; | ||||
|       it = Nodes.find(unit); | ||||
|     } | ||||
|  | ||||
|     if (it != Nodes.end()) { | ||||
|       for (byte x = 0; x < 4; x++) { | ||||
|         it->second.ip[x] = udpIn[x + 2]; | ||||
|       } | ||||
|       it->second.age = 0; // reset 'age counter' | ||||
|       char tmpNodeName[33] = { 0 }; | ||||
|       memcpy(&tmpNodeName[0], reinterpret_cast<byte *>(&udpIn[6]), 32); | ||||
|       tmpNodeName[32]     = 0; | ||||
|       it->second.nodeName = tmpNodeName; | ||||
|       it->second.nodeName.trim(); | ||||
|       it->second.nodeType = udpIn[38]; | ||||
|       uint32_t build = 0; | ||||
|       if (len >= 44) | ||||
|         for (byte i=0; i<sizeof(uint32_t); i++) | ||||
|           build |= udpIn[40+i]<<(8*i); | ||||
|       it->second.build = build; | ||||
|     } | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   //wled notifier, ignore if realtime packets active | ||||
|   if (udpIn[0] == 0 && !realtimeMode && receiveNotifications) | ||||
|   { | ||||
|     //ignore notification if received within a second after sending a notification ourselves | ||||
|     if (millis() - notificationSentTime < 1000) return; | ||||
|     if (udpIn[1] > 199) return; //do not receive custom versions | ||||
|      | ||||
|     bool someSel = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects); | ||||
|     //apply colors from notification | ||||
|     if (receiveNotificationColor || !someSel) | ||||
|     { | ||||
|       col[0] = udpIn[3]; | ||||
|       col[1] = udpIn[4]; | ||||
|       col[2] = udpIn[5]; | ||||
|       if (udpIn[11] > 0) //sending module's white val is intended | ||||
|       { | ||||
|         col[3] = udpIn[10]; | ||||
|         if (udpIn[11] > 1) | ||||
|         { | ||||
|           colSec[0] = udpIn[12]; | ||||
|           colSec[1] = udpIn[13]; | ||||
|           colSec[2] = udpIn[14]; | ||||
|           colSec[3] = udpIn[15]; | ||||
|         } | ||||
|         if (udpIn[11] > 5) | ||||
|         { | ||||
|           uint32_t t = (udpIn[25] << 24) | (udpIn[26] << 16) | (udpIn[27] << 8) | (udpIn[28]); | ||||
|           t += 2; | ||||
|           t -= millis(); | ||||
|           strip.timebase = t; | ||||
|         } | ||||
|         if (udpIn[11] > 6) | ||||
|         { | ||||
|           strip.setColor(2, udpIn[20], udpIn[21], udpIn[22], udpIn[23]); //tertiary color | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     //apply effects from notification | ||||
|     if (udpIn[11] < 200 && (receiveNotificationEffects || !someSel)) | ||||
|     { | ||||
|       if (udpIn[8] < strip.getModeCount()) effectCurrent = udpIn[8]; | ||||
|       effectSpeed   = udpIn[9]; | ||||
|       if (udpIn[11] > 2) effectIntensity = udpIn[16]; | ||||
|       if (udpIn[11] > 4 && udpIn[19] < strip.getPaletteCount()) effectPalette = udpIn[19]; | ||||
|     } | ||||
|      | ||||
|     if (udpIn[11] > 3) | ||||
|     { | ||||
|       transitionDelayTemp = ((udpIn[17] << 0) & 0xFF) + ((udpIn[18] << 8) & 0xFF00); | ||||
|     } | ||||
|  | ||||
|     nightlightActive = udpIn[6]; | ||||
|     if (nightlightActive) nightlightDelayMins = udpIn[7]; | ||||
|      | ||||
|     if (receiveNotificationBrightness || !someSel) bri = udpIn[2]; | ||||
|     colorUpdated(NOTIFIER_CALL_MODE_NOTIFICATION); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   if (!receiveDirect) return; | ||||
|    | ||||
|   //TPM2.NET | ||||
|   if (udpIn[0] == 0x9c) | ||||
|   { | ||||
|     //WARNING: this code assumes that the final TMP2.NET payload is evenly distributed if using multiple packets (ie. frame size is constant) | ||||
|     //if the number of LEDs in your installation doesn't allow that, please include padding bytes at the end of the last packet | ||||
|     byte tpmType = udpIn[1]; | ||||
|     if (tpmType == 0xaa) { //TPM2.NET polling, expect answer | ||||
|       sendTPM2Ack(); return; | ||||
|     } | ||||
|     if (tpmType != 0xda) return; //return if notTPM2.NET data | ||||
|  | ||||
|     realtimeIP = (isSupp) ? notifier2Udp.remoteIP() : notifierUdp.remoteIP(); | ||||
|     realtimeLock(realtimeTimeoutMs, REALTIME_MODE_TPM2NET); | ||||
|     if (realtimeOverride) return; | ||||
|  | ||||
|     tpmPacketCount++; //increment the packet count | ||||
|     if (tpmPacketCount == 1) tpmPayloadFrameSize = (udpIn[2] << 8) + udpIn[3]; //save frame size for the whole payload if this is the first packet | ||||
|     byte packetNum = udpIn[4]; //starts with 1! | ||||
|     byte numPackets = udpIn[5]; | ||||
|  | ||||
|     uint16_t id = (tpmPayloadFrameSize/3)*(packetNum-1); //start LED | ||||
|     for (uint16_t i = 6; i < tpmPayloadFrameSize + 4; i += 3) | ||||
|     { | ||||
|       if (id < ledCount) | ||||
|       { | ||||
|         setRealtimePixel(id, udpIn[i], udpIn[i+1], udpIn[i+2], 0); | ||||
|         id++; | ||||
|       } | ||||
|       else break; | ||||
|     } | ||||
|     if (tpmPacketCount == numPackets) //reset packet count and show if all packets were received | ||||
|     { | ||||
|       tpmPacketCount = 0; | ||||
|       strip.show(); | ||||
|     } | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   //UDP realtime: 1 warls 2 drgb 3 drgbw | ||||
|   if (udpIn[0] > 0 && udpIn[0] < 5) | ||||
|   { | ||||
|     realtimeIP = (isSupp) ? notifier2Udp.remoteIP() : notifierUdp.remoteIP(); | ||||
|     DEBUG_PRINTLN(realtimeIP); | ||||
|     if (packetSize < 2) return; | ||||
|  | ||||
|     if (udpIn[1] == 0) | ||||
|     { | ||||
|       realtimeTimeout = 0; | ||||
|       return; | ||||
|     } else { | ||||
|       realtimeLock(udpIn[1]*1000 +1, REALTIME_MODE_UDP); | ||||
|     } | ||||
|     if (realtimeOverride) return; | ||||
|  | ||||
|     if (udpIn[0] == 1) //warls | ||||
|     { | ||||
|       for (uint16_t i = 2; i < packetSize -3; i += 4) | ||||
|       { | ||||
|         setRealtimePixel(udpIn[i], udpIn[i+1], udpIn[i+2], udpIn[i+3], 0); | ||||
|       } | ||||
|     } else if (udpIn[0] == 2) //drgb | ||||
|     { | ||||
|       uint16_t id = 0; | ||||
|       for (uint16_t i = 2; i < packetSize -2; i += 3) | ||||
|       { | ||||
|         setRealtimePixel(id, udpIn[i], udpIn[i+1], udpIn[i+2], 0); | ||||
|  | ||||
|         id++; if (id >= ledCount) break; | ||||
|       } | ||||
|     } else if (udpIn[0] == 3) //drgbw | ||||
|     { | ||||
|       uint16_t id = 0; | ||||
|       for (uint16_t i = 2; i < packetSize -3; i += 4) | ||||
|       { | ||||
|         setRealtimePixel(id, udpIn[i], udpIn[i+1], udpIn[i+2], udpIn[i+3]); | ||||
|          | ||||
|         id++; if (id >= ledCount) break; | ||||
|       } | ||||
|     } else if (udpIn[0] == 4) //dnrgb | ||||
|     { | ||||
|       uint16_t id = ((udpIn[3] << 0) & 0xFF) + ((udpIn[2] << 8) & 0xFF00); | ||||
|       for (uint16_t i = 4; i < packetSize -2; i += 3) | ||||
|       { | ||||
|           if (id >= ledCount) break; | ||||
|         setRealtimePixel(id, udpIn[i], udpIn[i+1], udpIn[i+2], 0); | ||||
|         id++; | ||||
|       } | ||||
|     } else if (udpIn[0] == 5) //dnrgbw | ||||
|     { | ||||
|       uint16_t id = ((udpIn[3] << 0) & 0xFF) + ((udpIn[2] << 8) & 0xFF00); | ||||
|       for (uint16_t i = 4; i < packetSize -2; i += 4) | ||||
|       { | ||||
|           if (id >= ledCount) break; | ||||
|         setRealtimePixel(id, udpIn[i], udpIn[i+1], udpIn[i+2], udpIn[i+3]); | ||||
|         id++; | ||||
|       } | ||||
|     } | ||||
|     strip.show(); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   // API over UDP | ||||
|   udpIn[packetSize] = '\0'; | ||||
|  | ||||
|   if (udpIn[0] >= 'A' && udpIn[0] <= 'Z') { //HTTP API | ||||
|     String apireq = "win&"; | ||||
|     apireq += (char*)udpIn; | ||||
|     handleSet(nullptr, apireq); | ||||
|   } else if (udpIn[0] == '{') { //JSON API | ||||
|     DynamicJsonDocument jsonBuffer(2048); | ||||
|     DeserializationError error = deserializeJson(jsonBuffer, udpIn); | ||||
|     JsonObject root = jsonBuffer.as<JsonObject>(); | ||||
|     if (!error && !root.isNull()) deserializeState(root); | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| void setRealtimePixel(uint16_t i, byte r, byte g, byte b, byte w) | ||||
| { | ||||
|   uint16_t pix = i + arlsOffset; | ||||
|   if (pix < ledCount) | ||||
|   { | ||||
|     if (!arlsDisableGammaCorrection && strip.gammaCorrectCol) | ||||
|     { | ||||
|       strip.setPixelColor(pix, strip.gamma8(r), strip.gamma8(g), strip.gamma8(b), strip.gamma8(w)); | ||||
|     } else { | ||||
|       strip.setPixelColor(pix, r, g, b, w); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| /*********************************************************************************************\ | ||||
|    Refresh aging for remote units, drop if too old... | ||||
| \*********************************************************************************************/ | ||||
| void refreshNodeList() | ||||
| { | ||||
|   for (NodesMap::iterator it = Nodes.begin(); it != Nodes.end();) { | ||||
|     bool mustRemove = true; | ||||
|  | ||||
|     if (it->second.ip[0] != 0) { | ||||
|       if (it->second.age < 10) { | ||||
|         it->second.age++; | ||||
|         mustRemove = false; | ||||
|         ++it; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if (mustRemove) { | ||||
|       it = Nodes.erase(it); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| /*********************************************************************************************\ | ||||
|    Broadcast system info to other nodes. (to update node lists) | ||||
| \*********************************************************************************************/ | ||||
| void sendSysInfoUDP() | ||||
| { | ||||
|   if (!udp2Connected) return; | ||||
|  | ||||
|   IPAddress ip = Network.localIP(); | ||||
|  | ||||
|   // TODO: make a nice struct of it and clean up | ||||
|   //  0: 1 byte 'binary token 255' | ||||
|   //  1: 1 byte id '1' | ||||
|   //  2: 4 byte ip | ||||
|   //  6: 32 char name | ||||
|   // 38: 1 byte node type id | ||||
|   // 39: 1 byte node id | ||||
|   // 40: 4 byte version ID | ||||
|   // 44 bytes total | ||||
|  | ||||
|   // send my info to the world... | ||||
|   uint8_t data[44] = {0}; | ||||
|   data[0] = 255; | ||||
|   data[1] = 1; | ||||
|    | ||||
|   for (byte x = 0; x < 4; x++) { | ||||
|     data[x + 2] = ip[x]; | ||||
|   } | ||||
|   memcpy((byte *)data + 6, serverDescription, 32); | ||||
|   #ifdef ESP8266 | ||||
|   data[38] = NODE_TYPE_ID_ESP8266; | ||||
|   #elif defined(ARDUINO_ARCH_ESP32) | ||||
|   data[38] = NODE_TYPE_ID_ESP32; | ||||
|   #else | ||||
|   data[38] = NODE_TYPE_ID_UNDEFINED; | ||||
|   #endif | ||||
|   data[39] = ip[3]; // unit ID == last IP number | ||||
|  | ||||
|   uint32_t build = VERSION; | ||||
|   for (byte i=0; i<sizeof(uint32_t); i++) | ||||
|     data[40+i] = (build>>(8*i)) & 0xFF; | ||||
|  | ||||
|   IPAddress broadcastIP(255, 255, 255, 255); | ||||
|   notifier2Udp.beginPacket(broadcastIP, udpPort2); | ||||
|   notifier2Udp.write(data, sizeof(data)); | ||||
|   notifier2Udp.endPacket(); | ||||
| } | ||||
							
								
								
									
										38
									
								
								wled00/um_manager.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								wled00/um_manager.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| #include "wled.h" | ||||
| /* | ||||
|  * Registration and management utility for v2 usermods | ||||
|  */ | ||||
|  | ||||
| //Usermod Manager internals | ||||
| void UsermodManager::loop()      { for (byte i = 0; i < numMods; i++) ums[i]->loop();  } | ||||
|  | ||||
| void UsermodManager::setup()     { for (byte i = 0; i < numMods; i++) ums[i]->setup(); } | ||||
| void UsermodManager::connected() { for (byte i = 0; i < numMods; i++) ums[i]->connected(); } | ||||
|  | ||||
| void UsermodManager::addToJsonState(JsonObject& obj)    { for (byte i = 0; i < numMods; i++) ums[i]->addToJsonState(obj); } | ||||
| void UsermodManager::addToJsonInfo(JsonObject& obj)     { for (byte i = 0; i < numMods; i++) ums[i]->addToJsonInfo(obj); } | ||||
| void UsermodManager::readFromJsonState(JsonObject& obj) { for (byte i = 0; i < numMods; i++) ums[i]->readFromJsonState(obj); } | ||||
| void UsermodManager::addToConfig(JsonObject& obj)       { for (byte i = 0; i < numMods; i++) ums[i]->addToConfig(obj); } | ||||
| void UsermodManager::readFromConfig(JsonObject& obj)    { for (byte i = 0; i < numMods; i++) ums[i]->readFromConfig(obj); } | ||||
|  | ||||
| /* | ||||
|  * Enables usermods to lookup another Usermod. | ||||
|  */ | ||||
| Usermod* UsermodManager::lookup(uint16_t mod_id) { | ||||
|   for (byte i = 0; i < numMods; i++) { | ||||
|     if (ums[i]->getId() == mod_id) { | ||||
|       return ums[i]; | ||||
|     } | ||||
|   } | ||||
|   return nullptr; | ||||
| } | ||||
|  | ||||
| bool UsermodManager::add(Usermod* um) | ||||
| { | ||||
|   if (numMods >= WLED_MAX_USERMODS || um == nullptr) return false; | ||||
|   ums[numMods] = um; | ||||
|   numMods++; | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| byte UsermodManager::getModCount() {return numMods;} | ||||
| @@ -1,8 +1,11 @@ | ||||
| #include "wled.h" | ||||
| /*
 | ||||
|  * This file allows you to add own functionality to WLED more easily | ||||
|  * This v1 usermod file allows you to add own functionality to WLED more easily | ||||
|  * See: https://github.com/Aircoookie/WLED/wiki/Add-own-functionality
 | ||||
|  * EEPROM bytes 2750+ are reserved for your custom use case. (if you extend #define EEPSIZE in wled01_eeprom.h) | ||||
|  * bytes 2400+ are currently ununsed, but might be used for future wled features | ||||
|  * EEPROM bytes 2750+ are reserved for your custom use case. (if you extend #define EEPSIZE in const.h) | ||||
|  * If you just need 8 bytes, use 2551-2559 (you do not need to increase EEPSIZE) | ||||
|  *  | ||||
|  * Consider the v2 usermod API if you need a more advanced feature set! | ||||
|  */ | ||||
| 
 | ||||
| //Use userVar0 and userVar1 (API calls &U0=,&U1=, uint16_t)
 | ||||
							
								
								
									
										18
									
								
								wled00/usermod_v2_empty.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								wled00/usermod_v2_empty.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| //This is an empty v2 usermod template. Please see the file usermod_v2_example.h in the EXAMPLE_v2 usermod folder for documentation on the functions you can use! | ||||
|  | ||||
| class UsermodRenameMe : public Usermod { | ||||
|   private: | ||||
|      | ||||
|   public: | ||||
|     void setup() { | ||||
|        | ||||
|     } | ||||
|  | ||||
|     void loop() { | ||||
|        | ||||
|     } | ||||
| }; | ||||
							
								
								
									
										90
									
								
								wled00/usermods_list.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								wled00/usermods_list.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,90 @@ | ||||
| #include "wled.h" | ||||
| /* | ||||
|  * Register your v2 usermods here! | ||||
|  *   (for v1 usermods using just usermod.cpp, you can ignore this file) | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * Add/uncomment your usermod filename here (and once more below) | ||||
|  * || || || | ||||
|  * \/ \/ \/ | ||||
|  */ | ||||
| //#include "usermod_v2_example.h" | ||||
| #ifdef USERMOD_DALLASTEMPERATURE | ||||
| #include "../usermods/Temperature/usermod_temperature.h" | ||||
| #endif | ||||
|  | ||||
| //#include "usermod_v2_empty.h" | ||||
|  | ||||
| #ifdef USERMOD_BUZZER | ||||
| #include "../usermods/buzzer/usermod_v2_buzzer.h" | ||||
| #endif | ||||
| #ifdef USERMOD_SENSORSTOMQTT | ||||
| #include "usermod_v2_SensorsToMqtt.h" | ||||
| #endif | ||||
|  | ||||
| #ifdef USERMOD_MODE_SORT | ||||
| #include "../usermods/usermod_v2_mode_sort/usermod_v2_mode_sort.h" | ||||
| #endif | ||||
|  | ||||
| // BME280 v2 usermod. Define "USERMOD_BME280" in my_config.h | ||||
| #ifdef USERMOD_BME280 | ||||
| #include "../usermods/BME280_v2/usermod_bme280.h" | ||||
| #endif | ||||
| #ifdef USERMOD_FOUR_LINE_DISLAY | ||||
| #include "../usermods/usermod_v2_four_line_display/usermod_v2_four_line_display.h" | ||||
| #endif | ||||
| #ifdef USERMOD_ROTARY_ENCODER_UI | ||||
| #include "../usermods/usermod_v2_rotary_encoder_ui/usermod_v2_rotary_encoder_ui.h" | ||||
| #endif | ||||
| #ifdef USERMOD_AUTO_SAVE | ||||
| #include "../usermods/usermod_v2_auto_save/usermod_v2_auto_save.h" | ||||
| #endif | ||||
|  | ||||
| #ifdef USERMOD_DHT | ||||
| #include "../usermods/DHT/usermod_dht.h" | ||||
| #endif | ||||
|  | ||||
| void registerUsermods() | ||||
| { | ||||
| /* | ||||
|    * Add your usermod class name here | ||||
|    * || || || | ||||
|    * \/ \/ \/ | ||||
|    */ | ||||
|   //usermods.add(new MyExampleUsermod()); | ||||
|    | ||||
|   #ifdef USERMOD_DALLASTEMPERATURE | ||||
|   usermods.add(new UsermodTemperature()); | ||||
|   #endif | ||||
|    | ||||
|   //usermods.add(new UsermodRenameMe()); | ||||
|    | ||||
|   #ifdef USERMOD_BUZZER | ||||
|   usermods.add(new BuzzerUsermod()); | ||||
|   #endif | ||||
|    | ||||
|   #ifdef USERMOD_BME280 | ||||
|   usermods.add(new UsermodBME280()); | ||||
|   #endif | ||||
| #ifdef USERMOD_SENSORSTOMQTT | ||||
|   usermods.add(new UserMod_SensorsToMQTT()); | ||||
| #endif | ||||
|  | ||||
| #ifdef USERMOD_MODE_SORT | ||||
|   usermods.add(new ModeSortUsermod()); | ||||
| #endif | ||||
| #ifdef USERMOD_FOUR_LINE_DISLAY | ||||
|   usermods.add(new FourLineDisplayUsermod()); | ||||
| #endif | ||||
| #ifdef USERMOD_ROTARY_ENCODER_UI | ||||
|   usermods.add(new RotaryEncoderUIUsermod()); | ||||
| #endif | ||||
| #ifdef USERMOD_AUTO_SAVE | ||||
|   usermods.add(new AutoSaveUsermod()); | ||||
| #endif | ||||
|  | ||||
| #ifdef USERMOD_DHT | ||||
| usermods.add(new UsermodDHT()); | ||||
| #endif | ||||
| } | ||||
							
								
								
									
										697
									
								
								wled00/wled.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										697
									
								
								wled00/wled.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,697 @@ | ||||
| #define WLED_DEFINE_GLOBAL_VARS //only in one source file, wled.cpp! | ||||
| #include "wled.h" | ||||
| #include <Arduino.h> | ||||
|  | ||||
| #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_DISABLE_BROWNOUT_DET) | ||||
| #include "soc/soc.h" | ||||
| #include "soc/rtc_cntl_reg.h" | ||||
| #endif | ||||
|  | ||||
| /* | ||||
|  * Main WLED class implementation. Mostly initialization and connection logic | ||||
|  */ | ||||
|  | ||||
| WLED::WLED() | ||||
| { | ||||
| } | ||||
|  | ||||
| #ifdef WLED_USE_ETHERNET | ||||
| // settings for various ethernet boards | ||||
| typedef struct EthernetSettings { | ||||
|   uint8_t        eth_address; | ||||
|   int            eth_power; | ||||
|   int            eth_mdc; | ||||
|   int            eth_mdio; | ||||
|   eth_phy_type_t eth_type; | ||||
|   eth_clock_mode_t eth_clk_mode; | ||||
| } ethernet_settings; | ||||
|  | ||||
| ethernet_settings ethernetBoards[] = { | ||||
|   // None | ||||
|   { | ||||
|   }, | ||||
|    | ||||
|   // WT32-EHT01 | ||||
|   // Please note, from my testing only these pins work for LED outputs: | ||||
|   //   IO2, IO4, IO12, IO14, IO15 | ||||
|   // These pins do not appear to work from my testing: | ||||
|   //   IO35, IO36, IO39 | ||||
|   { | ||||
|     1,                 // eth_address,  | ||||
|     16,                // eth_power,  | ||||
|     23,                // eth_mdc,  | ||||
|     18,                // eth_mdio,  | ||||
|     ETH_PHY_LAN8720,   // eth_type, | ||||
|     ETH_CLOCK_GPIO0_IN // eth_clk_mode | ||||
|   }, | ||||
|  | ||||
|   // ESP32-POE | ||||
|   { | ||||
|      0,                  // eth_address,  | ||||
|     12,                  // eth_power,  | ||||
|     23,                  // eth_mdc,  | ||||
|     18,                  // eth_mdio,  | ||||
|     ETH_PHY_LAN8720,     // eth_type, | ||||
|     ETH_CLOCK_GPIO17_OUT // eth_clk_mode | ||||
|   }, | ||||
|  | ||||
|    // WESP32 | ||||
|   { | ||||
|     0,			              // eth_address, | ||||
|     -1,			              // eth_power, | ||||
|     16,			              // eth_mdc, | ||||
|     17,			              // eth_mdio, | ||||
|     ETH_PHY_LAN8720,      // eth_type, | ||||
|     ETH_CLOCK_GPIO0_IN	  // eth_clk_mode | ||||
|   }, | ||||
|  | ||||
|   // QuinLed-ESP32-Ethernet | ||||
|   { | ||||
|     0,			              // eth_address, | ||||
|     5,			              // eth_power, | ||||
|     23,			              // eth_mdc, | ||||
|     18,			              // eth_mdio, | ||||
|     ETH_PHY_LAN8720,      // eth_type, | ||||
|     ETH_CLOCK_GPIO17_OUT	// eth_clk_mode | ||||
|   } | ||||
| }; | ||||
|  | ||||
| #endif | ||||
|  | ||||
| // turns all LEDs off and restarts ESP | ||||
| void WLED::reset() | ||||
| { | ||||
|   briT = 0; | ||||
|   #ifdef WLED_ENABLE_WEBSOCKETS | ||||
|   ws.closeAll(1012); | ||||
|   #endif | ||||
|   long dly = millis(); | ||||
|   while (millis() - dly < 450) { | ||||
|     yield();        // enough time to send response to client | ||||
|   } | ||||
|   setAllLeds(); | ||||
|   DEBUG_PRINTLN("MODULE RESET"); | ||||
|   ESP.restart(); | ||||
| } | ||||
|  | ||||
| bool oappendi(int i) | ||||
| { | ||||
|   char s[11]; | ||||
|   sprintf(s, "%d", i); | ||||
|   return oappend(s); | ||||
| } | ||||
|  | ||||
| bool oappend(const char* txt) | ||||
| { | ||||
|   uint16_t len = strlen(txt); | ||||
|   if (olen + len >= OMAX) | ||||
|     return false;        // buffer full | ||||
|   strcpy(obuf + olen, txt); | ||||
|   olen += len; | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| void prepareHostname(char* hostname) | ||||
| { | ||||
|   const char *pC = serverDescription; | ||||
|   uint8_t pos = 5; | ||||
|  | ||||
|   while (*pC && pos < 24) { // while !null and not over length | ||||
|     if (isalnum(*pC)) {     // if the current char is alpha-numeric append it to the hostname | ||||
|       hostname[pos] = *pC; | ||||
|       pos++; | ||||
|     } else if (*pC == ' ' || *pC == '_' || *pC == '-' || *pC == '+' || *pC == '!' || *pC == '?' || *pC == '*') { | ||||
|       hostname[pos] = '-'; | ||||
|       pos++; | ||||
|     } | ||||
|       // else do nothing - no leading hyphens and do not include hyphens for all other characters. | ||||
|       pC++; | ||||
|     } | ||||
|     // if the hostname is left blank, use the mac address/default mdns name | ||||
|     if (pos < 6) { | ||||
|       sprintf(hostname + 5, "%*s", 6, escapedMac.c_str() + 6); | ||||
|     } else { //last character must not be hyphen | ||||
|       while (pos > 0 && hostname[pos -1] == '-') { | ||||
|         hostname[pos -1] = 0; | ||||
|         pos--; | ||||
|       } | ||||
|     } | ||||
| } | ||||
|  | ||||
| //handle Ethernet connection event | ||||
| void WiFiEvent(WiFiEvent_t event) | ||||
| { | ||||
|   #ifdef WLED_USE_ETHERNET | ||||
|   char hostname[25] = "wled-"; | ||||
|   #endif | ||||
|    | ||||
|   switch (event) { | ||||
| #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_USE_ETHERNET) | ||||
|     case SYSTEM_EVENT_ETH_START: | ||||
|       DEBUG_PRINT("ETH Started"); | ||||
|       break; | ||||
|     case SYSTEM_EVENT_ETH_CONNECTED: | ||||
|       DEBUG_PRINT("ETH Connected"); | ||||
|       if (!apActive) { | ||||
|         WiFi.disconnect(true); | ||||
|       } | ||||
|       if (staticIP != (uint32_t)0x00000000 && staticGateway != (uint32_t)0x00000000) { | ||||
|         ETH.config(staticIP, staticGateway, staticSubnet, IPAddress(8, 8, 8, 8)); | ||||
|       } else { | ||||
|         ETH.config(INADDR_NONE, INADDR_NONE, INADDR_NONE); | ||||
|       } | ||||
|       // convert the "serverDescription" into a valid DNS hostname (alphanumeric) | ||||
|       prepareHostname(hostname); | ||||
|       ETH.setHostname(hostname); | ||||
|       showWelcomePage = false; | ||||
|       break; | ||||
|     case SYSTEM_EVENT_ETH_DISCONNECTED: | ||||
|       DEBUG_PRINT("ETH Disconnected"); | ||||
|       forceReconnect = true; | ||||
|       break; | ||||
| #endif | ||||
|     default: | ||||
|       break; | ||||
|   } | ||||
| } | ||||
|  | ||||
| void WLED::loop() | ||||
| { | ||||
|   handleIR();        // 2nd call to function needed for ESP32 to return valid results -- should be good for ESP8266, too | ||||
|   handleConnection(); | ||||
|   handleSerial(); | ||||
|   handleNotifications(); | ||||
|   handleTransitions(); | ||||
| #ifdef WLED_ENABLE_DMX | ||||
|   handleDMX(); | ||||
| #endif | ||||
|   userLoop(); | ||||
|   usermods.loop(); | ||||
|  | ||||
|   yield(); | ||||
|   handleIO(); | ||||
|   handleIR(); | ||||
|   handleNetworkTime(); | ||||
|   handleAlexa(); | ||||
|  | ||||
|   handleOverlays(); | ||||
|   yield(); | ||||
|  | ||||
|   if (doReboot) | ||||
|     reset(); | ||||
|   if (doCloseFile) { | ||||
|     closeFile(); | ||||
|     yield(); | ||||
|   } | ||||
|  | ||||
|   if (!realtimeMode || realtimeOverride)  // block stuff if WARLS/Adalight is enabled | ||||
|   { | ||||
|     if (apActive) | ||||
|       dnsServer.processNextRequest(); | ||||
| #ifndef WLED_DISABLE_OTA | ||||
|     if (WLED_CONNECTED && aOtaEnabled) | ||||
|       ArduinoOTA.handle(); | ||||
| #endif | ||||
|     handleNightlight(); | ||||
|     handlePlaylist(); | ||||
|     yield(); | ||||
|  | ||||
|     handleHue(); | ||||
|     handleBlynk(); | ||||
|  | ||||
|     yield(); | ||||
|  | ||||
|     if (!offMode) | ||||
|       strip.service(); | ||||
| #ifdef ESP8266 | ||||
|     else if (!noWifiSleep) | ||||
|       delay(1); //required to make sure ESP enters modem sleep (see #1184) | ||||
| #endif | ||||
|   } | ||||
|   yield(); | ||||
| #ifdef ESP8266 | ||||
|   MDNS.update(); | ||||
| #endif | ||||
|   if (millis() - lastMqttReconnectAttempt > 30000) { | ||||
|     if (lastMqttReconnectAttempt > millis()) rolloverMillis++; //millis() rolls over every 50 days | ||||
|     initMqtt(); | ||||
|     refreshNodeList(); | ||||
|     if (nodeBroadcastEnabled) sendSysInfoUDP(); | ||||
|     yield(); | ||||
|   } | ||||
|  | ||||
|   //LED settings have been saved, re-init busses | ||||
|   //This code block causes severe FPS drop on ESP32 with the original "if (busConfigs[0] != nullptr)" conditional. Investigate!  | ||||
|   if (doInitBusses) { | ||||
|     doInitBusses = false; | ||||
|     busses.removeAll(); | ||||
|     uint32_t mem = 0; | ||||
|     strip.isRgbw = false; | ||||
|     for (uint8_t i = 0; i < WLED_MAX_BUSSES; i++) { | ||||
|       if (busConfigs[i] == nullptr) break; | ||||
|       mem += busses.memUsage(*busConfigs[i]); | ||||
|       if (mem <= MAX_LED_MEMORY) busses.add(*busConfigs[i]); | ||||
|       //if (BusManager::isRgbw(busConfigs[i]->type)) strip.isRgbw = true; | ||||
|       strip.isRgbw = (strip.isRgbw || BusManager::isRgbw(busConfigs[i]->type)); | ||||
|       delete busConfigs[i]; busConfigs[i] = nullptr; | ||||
|     } | ||||
|     strip.finalizeInit(ledCount, skipFirstLed); | ||||
|     yield(); | ||||
|     serializeConfig(); | ||||
|   } | ||||
|    | ||||
|   yield(); | ||||
|   handleWs(); | ||||
|   handleStatusLED(); | ||||
|  | ||||
| // DEBUG serial logging | ||||
| #ifdef WLED_DEBUG | ||||
|   if (millis() - debugTime > 9999) { | ||||
|     DEBUG_PRINTLN("---DEBUG INFO---"); | ||||
|     DEBUG_PRINT("Runtime: ");       DEBUG_PRINTLN(millis()); | ||||
|     DEBUG_PRINT("Unix time: ");     DEBUG_PRINTLN(now()); | ||||
|     DEBUG_PRINT("Free heap: ");     DEBUG_PRINTLN(ESP.getFreeHeap()); | ||||
|     DEBUG_PRINT("Wifi state: ");    DEBUG_PRINTLN(WiFi.status()); | ||||
|  | ||||
|     if (WiFi.status() != lastWifiState) { | ||||
|       wifiStateChangedTime = millis(); | ||||
|     } | ||||
|     lastWifiState = WiFi.status(); | ||||
|     DEBUG_PRINT("State time: ");    DEBUG_PRINTLN(wifiStateChangedTime); | ||||
|     DEBUG_PRINT("NTP last sync: "); DEBUG_PRINTLN(ntpLastSyncTime); | ||||
|     DEBUG_PRINT("Client IP: ");     DEBUG_PRINTLN(Network.localIP()); | ||||
|     DEBUG_PRINT("Loops/sec: ");     DEBUG_PRINTLN(loops / 10); | ||||
|     loops = 0; | ||||
|     debugTime = millis(); | ||||
|   } | ||||
|   loops++; | ||||
| #endif        // WLED_DEBUG | ||||
| } | ||||
|  | ||||
| void WLED::setup() | ||||
| { | ||||
|   #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_DISABLE_BROWNOUT_DET) | ||||
|   WRITE_PERI_REG(RTC_CNTL_BROWN_OUT_REG, 0); //disable brownout detection | ||||
|   #endif | ||||
|  | ||||
|   Serial.begin(115200); | ||||
|   Serial.setTimeout(50); | ||||
|   DEBUG_PRINTLN(); | ||||
|   DEBUG_PRINT("---WLED "); | ||||
|   DEBUG_PRINT(versionString); | ||||
|   DEBUG_PRINT(" "); | ||||
|   DEBUG_PRINT(VERSION); | ||||
|   DEBUG_PRINTLN(" INIT---"); | ||||
| #ifdef ARDUINO_ARCH_ESP32 | ||||
|   DEBUG_PRINT("esp32 "); | ||||
|   DEBUG_PRINTLN(ESP.getSdkVersion()); | ||||
| #else | ||||
|   DEBUG_PRINT("esp8266 "); | ||||
|   DEBUG_PRINTLN(ESP.getCoreVersion()); | ||||
| #endif | ||||
|   DEBUG_PRINT("heap "); | ||||
|   DEBUG_PRINTLN(ESP.getFreeHeap()); | ||||
|   registerUsermods(); | ||||
|  | ||||
|   //DEBUG_PRINT(F("LEDs inited. heap usage ~")); | ||||
|   //DEBUG_PRINTLN(heapPreAlloc - ESP.getFreeHeap()); | ||||
|  | ||||
| #ifdef WLED_USE_DMX //reserve GPIO2 as hardcoded DMX pin | ||||
|   pinManager.allocatePin(2); | ||||
| #endif | ||||
|  | ||||
|   bool fsinit = false; | ||||
|   DEBUGFS_PRINTLN(F("Mount FS")); | ||||
| #ifdef ARDUINO_ARCH_ESP32 | ||||
|   fsinit = WLED_FS.begin(true); | ||||
| #else | ||||
|   fsinit = WLED_FS.begin(); | ||||
| #endif | ||||
|   if (!fsinit) { | ||||
|     DEBUGFS_PRINTLN(F("FS failed!")); | ||||
|     errorFlag = ERR_FS_BEGIN; | ||||
|   } else deEEP(); | ||||
|   updateFSInfo(); | ||||
|   deserializeConfig(); | ||||
|  | ||||
| #if STATUSLED | ||||
|   bool lStatusLed = false; | ||||
|   for (uint8_t i=0; i<strip.numStrips; i++) { | ||||
|     if (strip.getStripPin(i)==STATUSLED) { | ||||
|       lStatusLed = true; | ||||
|       break; | ||||
|     } | ||||
|   } | ||||
|   if (!lStatusLed) | ||||
|     pinMode(STATUSLED, OUTPUT); | ||||
| #endif | ||||
|  | ||||
|   beginStrip(); | ||||
|   userSetup(); | ||||
|   usermods.setup(); | ||||
|   if (strcmp(clientSSID, DEFAULT_CLIENT_SSID) == 0) | ||||
|     showWelcomePage = true; | ||||
|   WiFi.persistent(false); | ||||
|   #ifdef WLED_USE_ETHERNET | ||||
|   WiFi.onEvent(WiFiEvent); | ||||
|   #endif | ||||
|  | ||||
|   Serial.println(F("Ada")); | ||||
|  | ||||
|   // generate module IDs | ||||
|   escapedMac = WiFi.macAddress(); | ||||
|   escapedMac.replace(":", ""); | ||||
|   escapedMac.toLowerCase(); | ||||
|   if (strcmp(cmDNS, "x") == 0)        // fill in unique mdns default | ||||
|   { | ||||
|     strcpy_P(cmDNS, PSTR("wled-")); | ||||
|     sprintf(cmDNS + 5, "%*s", 6, escapedMac.c_str() + 6); | ||||
|   } | ||||
|   if (mqttDeviceTopic[0] == 0) { | ||||
|     strcpy_P(mqttDeviceTopic, PSTR("wled/")); | ||||
|     sprintf(mqttDeviceTopic + 5, "%*s", 6, escapedMac.c_str() + 6); | ||||
|   } | ||||
|   if (mqttClientID[0] == 0) { | ||||
|     strcpy_P(mqttClientID, PSTR("WLED-")); | ||||
|     sprintf(mqttClientID + 5, "%*s", 6, escapedMac.c_str() + 6); | ||||
|   } | ||||
|  | ||||
|   strip.service(); | ||||
|  | ||||
| #ifndef WLED_DISABLE_OTA | ||||
|   if (aOtaEnabled) { | ||||
|     ArduinoOTA.onStart([]() { | ||||
| #ifdef ESP8266 | ||||
|       wifi_set_sleep_type(NONE_SLEEP_T); | ||||
| #endif | ||||
|       DEBUG_PRINTLN(F("Start ArduinoOTA")); | ||||
|     }); | ||||
|     if (strlen(cmDNS) > 0) | ||||
|       ArduinoOTA.setHostname(cmDNS); | ||||
|   } | ||||
| #endif | ||||
| #ifdef WLED_ENABLE_DMX | ||||
|   initDMX(); | ||||
| #endif | ||||
|   // HTTP server page init | ||||
|   initServer(); | ||||
| } | ||||
|  | ||||
| void WLED::beginStrip() | ||||
| { | ||||
|   // Initialize NeoPixel Strip and button | ||||
|  | ||||
|   if (ledCount > MAX_LEDS || ledCount == 0) | ||||
|     ledCount = 30; | ||||
|  | ||||
|   strip.finalizeInit(ledCount, skipFirstLed); | ||||
|   strip.setBrightness(0); | ||||
|   strip.setShowCallback(handleOverlayDraw); | ||||
|  | ||||
|   if (bootPreset > 0) applyPreset(bootPreset); | ||||
|   if (turnOnAtBoot) { | ||||
|     if (briS > 0) bri = briS; | ||||
|     else if (bri == 0) bri = 128; | ||||
|   } else { | ||||
|     briLast = briS; bri = 0; | ||||
|   } | ||||
|   colorUpdated(NOTIFIER_CALL_MODE_INIT); | ||||
|  | ||||
|   // init relay pin | ||||
|   if (rlyPin>=0) | ||||
|     digitalWrite(rlyPin, (rlyMde ? bri : !bri)); | ||||
|  | ||||
|   // disable button if it is "pressed" unintentionally | ||||
|   if (btnPin>=0 && isButtonPressed()) | ||||
|     buttonEnabled = false; | ||||
| } | ||||
|  | ||||
| void WLED::initAP(bool resetAP) | ||||
| { | ||||
|   if (apBehavior == AP_BEHAVIOR_BUTTON_ONLY && !resetAP) | ||||
|     return; | ||||
|  | ||||
|   if (!apSSID[0] || resetAP) | ||||
|     strcpy_P(apSSID, PSTR("WLED-AP")); | ||||
|   if (resetAP) | ||||
|     strcpy_P(apPass, PSTR(DEFAULT_AP_PASS)); | ||||
|   DEBUG_PRINT(F("Opening access point ")); | ||||
|   DEBUG_PRINTLN(apSSID); | ||||
|   WiFi.softAPConfig(IPAddress(4, 3, 2, 1), IPAddress(4, 3, 2, 1), IPAddress(255, 255, 255, 0)); | ||||
|   WiFi.softAP(apSSID, apPass, apChannel, apHide); | ||||
|  | ||||
|   if (!apActive) // start captive portal if AP active | ||||
|   { | ||||
|     DEBUG_PRINTLN(F("Init AP interfaces")); | ||||
|     server.begin(); | ||||
|     if (udpPort > 0 && udpPort != ntpLocalPort) { | ||||
|       udpConnected = notifierUdp.begin(udpPort); | ||||
|     } | ||||
|     if (udpRgbPort > 0 && udpRgbPort != ntpLocalPort && udpRgbPort != udpPort) { | ||||
|       udpRgbConnected = rgbUdp.begin(udpRgbPort); | ||||
|     } | ||||
|     if (udpPort2 > 0 && udpPort2 != ntpLocalPort && udpPort2 != udpPort && udpPort2 != udpRgbPort) { | ||||
|       udp2Connected = notifier2Udp.begin(udpPort2); | ||||
|     } | ||||
|     e131.begin(false, e131Port, e131Universe, E131_MAX_UNIVERSE_COUNT); | ||||
|  | ||||
|     dnsServer.setErrorReplyCode(DNSReplyCode::NoError); | ||||
|     dnsServer.start(53, "*", WiFi.softAPIP()); | ||||
|   } | ||||
|   apActive = true; | ||||
| } | ||||
|  | ||||
| void WLED::initConnection() | ||||
| { | ||||
|   #ifdef WLED_ENABLE_WEBSOCKETS | ||||
|   ws.onEvent(wsEvent); | ||||
|   #endif | ||||
|  | ||||
| #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_USE_ETHERNET) | ||||
|   // Only initialize ethernet board if not NONE | ||||
|   if (ethernetType != WLED_ETH_NONE && ethernetType < WLED_NUM_ETH_TYPES) { | ||||
|     ethernet_settings es = ethernetBoards[ethernetType]; | ||||
|     ETH.begin( | ||||
|       (uint8_t) es.eth_address,  | ||||
|       (int)     es.eth_power,  | ||||
|       (int)     es.eth_mdc,  | ||||
|       (int)     es.eth_mdio,  | ||||
|       (eth_phy_type_t)   es.eth_type, | ||||
|       (eth_clock_mode_t) es.eth_clk_mode | ||||
|     ); | ||||
|   } | ||||
| #endif | ||||
|  | ||||
|   WiFi.disconnect(true);        // close old connections | ||||
| #ifdef ESP8266 | ||||
|   WiFi.setPhyMode(WIFI_PHY_MODE_11N); | ||||
| #endif | ||||
|  | ||||
|   if (staticIP[0] != 0 && staticGateway[0] != 0) { | ||||
|     WiFi.config(staticIP, staticGateway, staticSubnet, IPAddress(8, 8, 8, 8)); | ||||
|   } else { | ||||
|     WiFi.config(0U, 0U, 0U); | ||||
|   } | ||||
|  | ||||
|   lastReconnectAttempt = millis(); | ||||
|  | ||||
|   if (!WLED_WIFI_CONFIGURED) { | ||||
|     DEBUG_PRINT(F("No connection configured. ")); | ||||
|     if (!apActive) | ||||
|       initAP();        // instantly go to ap mode | ||||
|     return; | ||||
|   } else if (!apActive) { | ||||
|     if (apBehavior == AP_BEHAVIOR_ALWAYS) { | ||||
|       initAP(); | ||||
|     } else { | ||||
|       DEBUG_PRINTLN(F("Access point disabled.")); | ||||
|       WiFi.softAPdisconnect(true); | ||||
|       WiFi.mode(WIFI_STA); | ||||
|     } | ||||
|   } | ||||
|   showWelcomePage = false; | ||||
|  | ||||
|   DEBUG_PRINT(F("Connecting to ")); | ||||
|   DEBUG_PRINT(clientSSID); | ||||
|   DEBUG_PRINTLN("..."); | ||||
|  | ||||
|   // convert the "serverDescription" into a valid DNS hostname (alphanumeric) | ||||
|   char hostname[25] = "wled-"; | ||||
|   prepareHostname(hostname); | ||||
|  | ||||
| #ifdef ESP8266 | ||||
|   WiFi.hostname(hostname); | ||||
| #endif | ||||
|  | ||||
|   WiFi.begin(clientSSID, clientPass); | ||||
|  | ||||
| #ifdef ARDUINO_ARCH_ESP32 | ||||
|   WiFi.setSleep(!noWifiSleep); | ||||
|   WiFi.setHostname(hostname); | ||||
| #else | ||||
|   wifi_set_sleep_type((noWifiSleep) ? NONE_SLEEP_T : MODEM_SLEEP_T); | ||||
| #endif | ||||
| } | ||||
|  | ||||
| void WLED::initInterfaces() | ||||
| { | ||||
|   DEBUG_PRINTLN(F("Init STA interfaces")); | ||||
|  | ||||
|   if (hueIP[0] == 0) { | ||||
|     hueIP[0] = Network.localIP()[0]; | ||||
|     hueIP[1] = Network.localIP()[1]; | ||||
|     hueIP[2] = Network.localIP()[2]; | ||||
|   } | ||||
|  | ||||
|   // init Alexa hue emulation | ||||
|   if (alexaEnabled) | ||||
|     alexaInit(); | ||||
|  | ||||
| #ifndef WLED_DISABLE_OTA | ||||
|   if (aOtaEnabled) | ||||
|     ArduinoOTA.begin(); | ||||
| #endif | ||||
|  | ||||
|   strip.service(); | ||||
|   // Set up mDNS responder: | ||||
|   if (strlen(cmDNS) > 0) { | ||||
|   #ifndef WLED_DISABLE_OTA | ||||
|     if (!aOtaEnabled) //ArduinoOTA begins mDNS for us if enabled | ||||
|       MDNS.begin(cmDNS); | ||||
|   #else | ||||
|     MDNS.begin(cmDNS); | ||||
|   #endif | ||||
|  | ||||
|     DEBUG_PRINTLN(F("mDNS started")); | ||||
|     MDNS.addService("http", "tcp", 80); | ||||
|     MDNS.addService("wled", "tcp", 80); | ||||
|     MDNS.addServiceTxt("wled", "tcp", "mac", escapedMac.c_str()); | ||||
|   } | ||||
|   server.begin(); | ||||
|  | ||||
|   if (udpPort > 0 && udpPort != ntpLocalPort) { | ||||
|     udpConnected = notifierUdp.begin(udpPort); | ||||
|     if (udpConnected && udpRgbPort != udpPort) | ||||
|       udpRgbConnected = rgbUdp.begin(udpRgbPort); | ||||
|     if (udpConnected && udpPort2 != udpPort && udpPort2 != udpRgbPort) | ||||
|       udp2Connected = notifier2Udp.begin(udpPort2); | ||||
|   } | ||||
|   if (ntpEnabled) | ||||
|     ntpConnected = ntpUdp.begin(ntpLocalPort); | ||||
|  | ||||
|   initBlynk(blynkApiKey, blynkHost, blynkPort); | ||||
|   e131.begin(e131Multicast, e131Port, e131Universe, E131_MAX_UNIVERSE_COUNT); | ||||
|   reconnectHue(); | ||||
|   initMqtt(); | ||||
|   interfacesInited = true; | ||||
|   wasConnected = true; | ||||
| } | ||||
|  | ||||
| byte stacO = 0; | ||||
| uint32_t lastHeap; | ||||
| unsigned long heapTime = 0; | ||||
|  | ||||
| void WLED::handleConnection() | ||||
| { | ||||
|   if (millis() < 2000 && (!WLED_WIFI_CONFIGURED || apBehavior == AP_BEHAVIOR_ALWAYS)) | ||||
|     return; | ||||
|   if (lastReconnectAttempt == 0) | ||||
|     initConnection(); | ||||
|  | ||||
|   // reconnect WiFi to clear stale allocations if heap gets too low | ||||
|   if (millis() - heapTime > 5000) { | ||||
|     uint32_t heap = ESP.getFreeHeap(); | ||||
|     if (heap < 9000 && lastHeap < 9000) { | ||||
|       DEBUG_PRINT(F("Heap too low! ")); | ||||
|       DEBUG_PRINTLN(heap); | ||||
|       forceReconnect = true; | ||||
|     } | ||||
|     lastHeap = heap; | ||||
|     heapTime = millis(); | ||||
|   } | ||||
|  | ||||
|   byte stac = 0; | ||||
|   if (apActive) { | ||||
| #ifdef ESP8266 | ||||
|     stac = wifi_softap_get_station_num(); | ||||
| #else | ||||
|     wifi_sta_list_t stationList; | ||||
|     esp_wifi_ap_get_sta_list(&stationList); | ||||
|     stac = stationList.num; | ||||
| #endif | ||||
|     if (stac != stacO) { | ||||
|       stacO = stac; | ||||
|       DEBUG_PRINT(F("Connected AP clients: ")); | ||||
|       DEBUG_PRINTLN(stac); | ||||
|       if (!WLED_CONNECTED && WLED_WIFI_CONFIGURED) {        // trying to connect, but not connected | ||||
|         if (stac) | ||||
|           WiFi.disconnect();        // disable search so that AP can work | ||||
|         else | ||||
|           initConnection();        // restart search | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   if (forceReconnect) { | ||||
|     DEBUG_PRINTLN(F("Forcing reconnect.")); | ||||
|     initConnection(); | ||||
|     interfacesInited = false; | ||||
|     forceReconnect = false; | ||||
|     wasConnected = false; | ||||
|     return; | ||||
|   } | ||||
|   if (!Network.isConnected()) { | ||||
|     if (interfacesInited) { | ||||
|       DEBUG_PRINTLN(F("Disconnected!")); | ||||
|       interfacesInited = false; | ||||
|       initConnection(); | ||||
|     } | ||||
|     if (millis() - lastReconnectAttempt > ((stac) ? 300000 : 20000) && WLED_WIFI_CONFIGURED) | ||||
|       initConnection(); | ||||
|     if (!apActive && millis() - lastReconnectAttempt > 12000 && (!wasConnected || apBehavior == AP_BEHAVIOR_NO_CONN)) | ||||
|       initAP(); | ||||
|   } else if (!interfacesInited) {        // newly connected | ||||
|     DEBUG_PRINTLN(""); | ||||
|     DEBUG_PRINT(F("Connected! IP address: ")); | ||||
|     DEBUG_PRINTLN(Network.localIP()); | ||||
|     initInterfaces(); | ||||
|     userConnected(); | ||||
|     usermods.connected(); | ||||
|  | ||||
|     // shut down AP | ||||
|     if (apBehavior != AP_BEHAVIOR_ALWAYS && apActive) { | ||||
|       dnsServer.stop(); | ||||
|       WiFi.softAPdisconnect(true); | ||||
|       apActive = false; | ||||
|       DEBUG_PRINTLN(F("Access point disabled.")); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| void WLED::handleStatusLED() | ||||
| { | ||||
|   #if STATUSLED | ||||
|   for (uint8_t s=0; s<strip.numStrips; s++) { | ||||
|     if (strip.getStripPin(s)==STATUSLED) { | ||||
|       return; // pin used for strip | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   ledStatusType = WLED_CONNECTED ? 0 : 2; | ||||
|   if (mqttEnabled && ledStatusType != 2) // Wi-Fi takes presendence over MQTT | ||||
|     ledStatusType = WLED_MQTT_CONNECTED ? 0 : 4; | ||||
|   if (ledStatusType) { | ||||
|     if (millis() - ledStatusLastMillis >= (1000/ledStatusType)) { | ||||
|       ledStatusLastMillis = millis(); | ||||
|       ledStatusState = ledStatusState ? 0 : 1; | ||||
|       digitalWrite(STATUSLED, ledStatusState); | ||||
|     } | ||||
|   } else { | ||||
|     #ifdef STATUSLEDINVERTED | ||||
|       digitalWrite(STATUSLED, HIGH); | ||||
|     #else | ||||
|       digitalWrite(STATUSLED, LOW); | ||||
|     #endif | ||||
|  | ||||
|   } | ||||
|   #endif | ||||
| } | ||||
							
								
								
									
										627
									
								
								wled00/wled.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										627
									
								
								wled00/wled.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,627 @@ | ||||
| #ifndef WLED_H | ||||
| #define WLED_H | ||||
| /* | ||||
|    Main sketch, global variable declarations | ||||
|    @title WLED project sketch | ||||
|    @version 0.12.0 | ||||
|    @author Christian Schwinne | ||||
|  */ | ||||
|  | ||||
| // version code in format yymmddb (b = daily build) | ||||
| #define VERSION 2104030 | ||||
|  | ||||
| //uncomment this if you have a "my_config.h" file you'd like to use | ||||
| //#define WLED_USE_MY_CONFIG | ||||
|  | ||||
| // ESP8266-01 (blue) got too little storage space to work with WLED. 0.10.2 is the last release supporting this unit. | ||||
|  | ||||
| // ESP8266-01 (black) has 1MB flash and can thus fit the whole program, although OTA update is not possible. Use 1M(128K SPIFFS). | ||||
| // Uncomment some of the following lines to disable features: | ||||
| // Alternatively, with platformio pass your chosen flags to your custom build target in platformio_override.ini | ||||
|  | ||||
| // You are required to disable over-the-air updates: | ||||
| //#define WLED_DISABLE_OTA         // saves 14kb | ||||
|  | ||||
| // You can choose some of these features to disable: | ||||
| //#define WLED_DISABLE_ALEXA       // saves 11kb | ||||
| //#define WLED_DISABLE_BLYNK       // saves 6kb | ||||
| //#define WLED_DISABLE_CRONIXIE    // saves 3kb | ||||
| //WLED_DISABLE_FX_HIGH_FLASH_USE (need to enable in PIO config or FX.h, saves 18kb) | ||||
| //#define WLED_DISABLE_HUESYNC     // saves 4kb | ||||
| //#define WLED_DISABLE_INFRARED    // there is no pin left for this on ESP8266-01, saves 12kb | ||||
| #ifndef WLED_DISABLE_MQTT | ||||
|   #define WLED_ENABLE_MQTT         // saves 12kb | ||||
| #endif | ||||
| #define WLED_ENABLE_ADALIGHT       // saves 500b only | ||||
| //#define WLED_ENABLE_DMX          // uses 3.5kb (use LEDPIN other than 2) | ||||
| #define WLED_ENABLE_LOXONE         // uses 1.2kb | ||||
| #ifndef WLED_DISABLE_WEBSOCKETS | ||||
|   #define WLED_ENABLE_WEBSOCKETS | ||||
| #endif | ||||
|  | ||||
| #define WLED_ENABLE_FS_EDITOR      // enable /edit page for editing FS content. Will also be disabled with OTA lock | ||||
|  | ||||
| // to toggle usb serial debug (un)comment the following line | ||||
| //#define WLED_DEBUG | ||||
|  | ||||
| // filesystem specific debugging | ||||
| //#define WLED_DEBUG_FS | ||||
|  | ||||
| //optionally disable brownout detector on ESP32. | ||||
| //This is generally a terrible idea, but improves boot success on boards with a 3.3v regulator + cap setup that can't provide 400mA peaks | ||||
| //#define WLED_DISABLE_BROWNOUT_DET | ||||
|  | ||||
| // Library inclusions.  | ||||
| #include <Arduino.h> | ||||
| #ifdef ESP8266 | ||||
|   #include <ESP8266WiFi.h> | ||||
|   #include <ESP8266mDNS.h> | ||||
|   #include <ESPAsyncTCP.h> | ||||
|   #include <LittleFS.h> | ||||
|   extern "C" | ||||
|   { | ||||
|   #include <user_interface.h> | ||||
|   } | ||||
| #else // ESP32 | ||||
|   #include <WiFi.h> | ||||
|   #include <ETH.h> | ||||
|   #include "esp_wifi.h" | ||||
|   #include <ESPmDNS.h> | ||||
|   #include <AsyncTCP.h> | ||||
|   //#include "SPIFFS.h" | ||||
|   #ifndef CONFIG_LITTLEFS_FOR_IDF_3_2 | ||||
|     #define CONFIG_LITTLEFS_FOR_IDF_3_2 | ||||
|   #endif | ||||
|   #include <LITTLEFS.h> | ||||
| #endif | ||||
|  | ||||
| #include "src/dependencies/network/Network.h" | ||||
|  | ||||
| #ifdef WLED_USE_MY_CONFIG | ||||
|   #include "my_config.h" | ||||
| #endif | ||||
|  | ||||
| #include <ESPAsyncWebServer.h> | ||||
| #include <EEPROM.h> | ||||
| #include <WiFiUdp.h> | ||||
| #include <DNSServer.h> | ||||
| #ifndef WLED_DISABLE_OTA | ||||
|   #include <ArduinoOTA.h> | ||||
| #endif | ||||
| #include <SPIFFSEditor.h> | ||||
| #include "src/dependencies/time/TimeLib.h" | ||||
| #include "src/dependencies/timezone/Timezone.h" | ||||
|  | ||||
| #ifndef WLED_DISABLE_ALEXA | ||||
|   #define ESPALEXA_ASYNC | ||||
|   #define ESPALEXA_NO_SUBPAGE | ||||
|   #define ESPALEXA_MAXDEVICES 1 | ||||
|   // #define ESPALEXA_DEBUG | ||||
|   #include "src/dependencies/espalexa/Espalexa.h" | ||||
| #endif | ||||
| #ifndef WLED_DISABLE_BLYNK | ||||
|   #include "src/dependencies/blynk/BlynkSimpleEsp.h" | ||||
| #endif | ||||
|  | ||||
| #ifdef WLED_ENABLE_DMX | ||||
|   #include "src/dependencies/dmx/ESPDMX.h" | ||||
| #endif | ||||
|  | ||||
| #include "src/dependencies/e131/ESPAsyncE131.h" | ||||
| #include "src/dependencies/async-mqtt-client/AsyncMqttClient.h" | ||||
|  | ||||
| #define ARDUINOJSON_DECODE_UNICODE 0 | ||||
| #include "src/dependencies/json/AsyncJson-v6.h" | ||||
| #include "src/dependencies/json/ArduinoJson-v6.h" | ||||
|  | ||||
| #include "fcn_declare.h" | ||||
| #include "html_ui.h" | ||||
| #include "html_settings.h" | ||||
| #include "html_other.h" | ||||
| #include "FX.h" | ||||
| #include "ir_codes.h" | ||||
| #include "const.h" | ||||
| #include "NodeStruct.h" | ||||
| #include "pin_manager.h" | ||||
| #include "bus_manager.h" | ||||
|  | ||||
| #ifndef CLIENT_SSID | ||||
|   #define CLIENT_SSID DEFAULT_CLIENT_SSID | ||||
| #endif | ||||
|  | ||||
| #ifndef CLIENT_PASS | ||||
|   #define CLIENT_PASS "" | ||||
| #endif | ||||
|  | ||||
| #ifndef SPIFFS_EDITOR_AIRCOOOKIE | ||||
|   #error You are not using the Aircoookie fork of the ESPAsyncWebserver library.\ | ||||
|   Using upstream puts your WiFi password at risk of being served by the filesystem.\ | ||||
|   Comment out this error message to build regardless. | ||||
| #endif | ||||
|  | ||||
| #ifndef WLED_DISABLE_INFRARED | ||||
|   #include <IRremoteESP8266.h> | ||||
|   #include <IRrecv.h> | ||||
|   #include <IRutils.h> | ||||
| #endif | ||||
|  | ||||
| //Filesystem to use for preset and config files. SPIFFS or LittleFS on ESP8266, SPIFFS only on ESP32 (now using LITTLEFS port by lorol) | ||||
| #ifdef ESP8266 | ||||
|   #define WLED_FS LittleFS | ||||
| #else | ||||
|   #define WLED_FS LITTLEFS | ||||
| #endif | ||||
|  | ||||
| // GLOBAL VARIABLES | ||||
| // both declared and defined in header (solution from http://www.keil.com/support/docs/1868.htm) | ||||
| // | ||||
| //e.g. byte test = 2 becomes WLED_GLOBAL byte test _INIT(2); | ||||
| //     int arr[]{0,1,2} becomes WLED_GLOBAL int arr[] _INIT_N(({0,1,2})); | ||||
|  | ||||
| #ifndef WLED_DEFINE_GLOBAL_VARS | ||||
| # define WLED_GLOBAL extern | ||||
| # define _INIT(x) | ||||
| # define _INIT_N(x) | ||||
| #else | ||||
| # define WLED_GLOBAL | ||||
| # define _INIT(x) = x | ||||
|  | ||||
| //needed to ignore commas in array definitions | ||||
| #define UNPACK( ... ) __VA_ARGS__ | ||||
| # define _INIT_N(x) UNPACK x | ||||
| #endif | ||||
|  | ||||
| // Global Variable definitions | ||||
| WLED_GLOBAL char versionString[] _INIT("0.12.0"); | ||||
| #define WLED_CODENAME "Hikari" | ||||
|  | ||||
| // AP and OTA default passwords (for maximum security change them!) | ||||
| WLED_GLOBAL char apPass[65]  _INIT(DEFAULT_AP_PASS); | ||||
| WLED_GLOBAL char otaPass[33] _INIT(DEFAULT_OTA_PASS); | ||||
|  | ||||
| // Hardware CONFIG (only changeble HERE, not at runtime) | ||||
| // LED strip pin, button pin and IR pin changeable in NpbWrapper.h! | ||||
| #ifndef BTNPIN | ||||
| WLED_GLOBAL int8_t btnPin _INIT(-1); | ||||
| #else | ||||
| WLED_GLOBAL int8_t btnPin _INIT(BTNPIN); | ||||
| #endif | ||||
| #ifndef RLYPIN | ||||
| WLED_GLOBAL int8_t rlyPin _INIT(-1); | ||||
| #else | ||||
| WLED_GLOBAL int8_t rlyPin _INIT(RLYPIN); | ||||
| #endif | ||||
| //Relay mode (1 = active high, 0 = active low, flipped in cfg.json) | ||||
| #ifndef RLYMDE | ||||
| WLED_GLOBAL bool rlyMde _INIT(true); | ||||
| #else | ||||
| WLED_GLOBAL bool rlyMde _INIT(RLYMDE); | ||||
| #endif | ||||
| #ifndef IRPIN | ||||
| WLED_GLOBAL int8_t irPin _INIT(-1); | ||||
| #else | ||||
| WLED_GLOBAL int8_t irPin _INIT(IRPIN); | ||||
| #endif | ||||
|  | ||||
| //WLED_GLOBAL byte presetToApply _INIT(0);  | ||||
|  | ||||
| WLED_GLOBAL char ntpServerName[33] _INIT("0.wled.pool.ntp.org");   // NTP server to use | ||||
|  | ||||
| // WiFi CONFIG (all these can be changed via web UI, no need to set them here) | ||||
| WLED_GLOBAL char clientSSID[33] _INIT(CLIENT_SSID); | ||||
| WLED_GLOBAL char clientPass[65] _INIT(CLIENT_PASS); | ||||
| WLED_GLOBAL char cmDNS[33] _INIT("x");                             // mDNS address (placeholder, is replaced by wledXXXXXX.local) | ||||
| WLED_GLOBAL char apSSID[33] _INIT("");                             // AP off by default (unless setup) | ||||
| WLED_GLOBAL byte apChannel _INIT(1);                               // 2.4GHz WiFi AP channel (1-13) | ||||
| WLED_GLOBAL byte apHide    _INIT(0);                               // hidden AP SSID | ||||
| WLED_GLOBAL byte apBehavior _INIT(AP_BEHAVIOR_BOOT_NO_CONN);       // access point opens when no connection after boot by default | ||||
| WLED_GLOBAL IPAddress staticIP      _INIT_N(((  0,   0,  0,  0))); // static IP of ESP | ||||
| WLED_GLOBAL IPAddress staticGateway _INIT_N(((  0,   0,  0,  0))); // gateway (router) IP | ||||
| WLED_GLOBAL IPAddress staticSubnet  _INIT_N(((255, 255, 255, 0))); // most common subnet in home networks | ||||
| WLED_GLOBAL bool noWifiSleep _INIT(false);                         // disabling modem sleep modes will increase heat output and power usage, but may help with connection issues | ||||
| #ifdef WLED_USE_ETHERNET | ||||
|   #ifdef WLED_ETH_DEFAULT                                          // default ethernet board type if specified | ||||
|     WLED_GLOBAL int ethernetType _INIT(WLED_ETH_DEFAULT);          // ethernet board type | ||||
|   #else | ||||
|     WLED_GLOBAL int ethernetType _INIT(WLED_ETH_NONE);             // use none for ethernet board type if default not defined | ||||
|   #endif                | ||||
| #endif | ||||
|  | ||||
| // LED CONFIG | ||||
| WLED_GLOBAL uint16_t ledCount _INIT(30);          // overcurrent prevented by ABL | ||||
| WLED_GLOBAL bool turnOnAtBoot _INIT(true);        // turn on LEDs at power-up | ||||
| WLED_GLOBAL byte bootPreset   _INIT(0);           // save preset to load after power-up | ||||
|  | ||||
| WLED_GLOBAL byte col[]    _INIT_N(({ 255, 160, 0, 0 }));  // current RGB(W) primary color. col[] should be updated if you want to change the color. | ||||
| WLED_GLOBAL byte colSec[] _INIT_N(({ 0, 0, 0, 0 }));      // current RGB(W) secondary color | ||||
| WLED_GLOBAL byte briS     _INIT(128);                     // default brightness | ||||
|  | ||||
| WLED_GLOBAL byte nightlightTargetBri _INIT(0);      // brightness after nightlight is over | ||||
| WLED_GLOBAL byte nightlightDelayMins _INIT(60); | ||||
| WLED_GLOBAL byte nightlightMode      _INIT(NL_MODE_FADE); // See const.h for available modes. Was nightlightFade | ||||
| WLED_GLOBAL bool fadeTransition      _INIT(true);   // enable crossfading color transition | ||||
| WLED_GLOBAL uint16_t transitionDelay _INIT(750);    // default crossfade duration in ms | ||||
|  | ||||
| WLED_GLOBAL bool skipFirstLed  _INIT(false);        // ignore first LED in strip (useful if you need the LED as signal repeater) | ||||
| WLED_GLOBAL byte briMultiplier _INIT(100);          // % of brightness to set (to limit power, if you set it to 50 and set bri to 255, actual brightness will be 127) | ||||
|  | ||||
| // User Interface CONFIG | ||||
| WLED_GLOBAL char serverDescription[33] _INIT("WLED");  // Name of module | ||||
| WLED_GLOBAL bool syncToggleReceive     _INIT(false);   // UIs which only have a single button for sync should toggle send+receive if this is true, only send otherwise | ||||
|  | ||||
| // Sync CONFIG | ||||
| WLED_GLOBAL NodesMap Nodes; | ||||
| WLED_GLOBAL bool nodeListEnabled _INIT(true); | ||||
| WLED_GLOBAL bool nodeBroadcastEnabled _INIT(true); | ||||
|  | ||||
| WLED_GLOBAL bool buttonEnabled  _INIT(true); | ||||
| WLED_GLOBAL byte irEnabled      _INIT(0);     // Infrared receiver | ||||
|  | ||||
| WLED_GLOBAL uint16_t udpPort    _INIT(21324); // WLED notifier default port | ||||
| WLED_GLOBAL uint16_t udpPort2   _INIT(65506); // WLED notifier supplemental port | ||||
| WLED_GLOBAL uint16_t udpRgbPort _INIT(19446); // Hyperion port | ||||
|  | ||||
| WLED_GLOBAL bool receiveNotificationBrightness _INIT(true);       // apply brightness from incoming notifications | ||||
| WLED_GLOBAL bool receiveNotificationColor      _INIT(true);       // apply color | ||||
| WLED_GLOBAL bool receiveNotificationEffects    _INIT(true);       // apply effects setup | ||||
| WLED_GLOBAL bool notifyDirect _INIT(false);                       // send notification if change via UI or HTTP API | ||||
| WLED_GLOBAL bool notifyButton _INIT(false);                       // send if updated by button or infrared remote | ||||
| WLED_GLOBAL bool notifyAlexa  _INIT(false);                       // send notification if updated via Alexa | ||||
| WLED_GLOBAL bool notifyMacro  _INIT(false);                       // send notification for macro | ||||
| WLED_GLOBAL bool notifyHue    _INIT(true);                        // send notification if Hue light changes | ||||
| WLED_GLOBAL bool notifyTwice  _INIT(false);                       // notifications use UDP: enable if devices don't sync reliably | ||||
|  | ||||
| WLED_GLOBAL bool alexaEnabled _INIT(false);                       // enable device discovery by Amazon Echo | ||||
| WLED_GLOBAL char alexaInvocationName[33] _INIT("Light");          // speech control name of device. Choose something voice-to-text can understand | ||||
|  | ||||
| WLED_GLOBAL char blynkApiKey[36] _INIT("");                       // Auth token for Blynk server. If empty, no connection will be made | ||||
| WLED_GLOBAL char blynkHost[33] _INIT("blynk-cloud.com");          // Default Blynk host | ||||
| WLED_GLOBAL uint16_t blynkPort _INIT(80);                         // Default Blynk port | ||||
|  | ||||
| WLED_GLOBAL uint16_t realtimeTimeoutMs _INIT(2500);               // ms timeout of realtime mode before returning to normal mode | ||||
| WLED_GLOBAL int arlsOffset _INIT(0);                              // realtime LED offset | ||||
| WLED_GLOBAL bool receiveDirect _INIT(true);                       // receive UDP realtime | ||||
| WLED_GLOBAL bool arlsDisableGammaCorrection _INIT(true);          // activate if gamma correction is handled by the source | ||||
| WLED_GLOBAL bool arlsForceMaxBri _INIT(false);                    // enable to force max brightness if source has very dark colors that would be black | ||||
|  | ||||
| #ifdef WLED_ENABLE_DMX | ||||
| WLED_GLOBAL DMXESPSerial dmx; | ||||
| WLED_GLOBAL uint16_t e131ProxyUniverse _INIT(0);                  // output this E1.31 (sACN) / ArtNet universe via MAX485 (0 = disabled) | ||||
| #endif | ||||
| WLED_GLOBAL uint16_t e131Universe _INIT(1);                       // settings for E1.31 (sACN) protocol (only DMX_MODE_MULTIPLE_* can span over consequtive universes) | ||||
| WLED_GLOBAL uint16_t e131Port _INIT(5568);                        // DMX in port. E1.31 default is 5568, Art-Net is 6454 | ||||
| WLED_GLOBAL byte DMXMode _INIT(DMX_MODE_MULTIPLE_RGB);            // DMX mode (s.a.) | ||||
| WLED_GLOBAL uint16_t DMXAddress _INIT(1);                         // DMX start address of fixture, a.k.a. first Channel [for E1.31 (sACN) protocol] | ||||
| WLED_GLOBAL byte DMXOldDimmer _INIT(0);                           // only update brightness on change | ||||
| WLED_GLOBAL byte e131LastSequenceNumber[E131_MAX_UNIVERSE_COUNT]; // to detect packet loss | ||||
| WLED_GLOBAL bool e131Multicast _INIT(false);                      // multicast or unicast | ||||
| WLED_GLOBAL bool e131SkipOutOfSequence _INIT(false);              // freeze instead of flickering | ||||
|  | ||||
| WLED_GLOBAL bool mqttEnabled _INIT(false); | ||||
| WLED_GLOBAL char mqttDeviceTopic[33] _INIT("");            // main MQTT topic (individual per device, default is wled/mac) | ||||
| WLED_GLOBAL char mqttGroupTopic[33] _INIT("wled/all");     // second MQTT topic (for example to group devices) | ||||
| WLED_GLOBAL char mqttServer[33] _INIT("");                 // both domains and IPs should work (no SSL) | ||||
| WLED_GLOBAL char mqttUser[41] _INIT("");                   // optional: username for MQTT auth | ||||
| WLED_GLOBAL char mqttPass[41] _INIT("");                   // optional: password for MQTT auth | ||||
| WLED_GLOBAL char mqttClientID[41] _INIT("");               // override the client ID | ||||
| WLED_GLOBAL uint16_t mqttPort _INIT(1883); | ||||
|  | ||||
| WLED_GLOBAL bool huePollingEnabled _INIT(false);           // poll hue bridge for light state | ||||
| WLED_GLOBAL uint16_t huePollIntervalMs _INIT(2500);        // low values (< 1sec) may cause lag but offer quicker response | ||||
| WLED_GLOBAL char hueApiKey[47] _INIT("api");               // key token will be obtained from bridge | ||||
| WLED_GLOBAL byte huePollLightId _INIT(1);                  // ID of hue lamp to sync to. Find the ID in the hue app ("about" section) | ||||
| WLED_GLOBAL IPAddress hueIP _INIT_N(((0, 0, 0, 0))); // IP address of the bridge | ||||
| WLED_GLOBAL bool hueApplyOnOff _INIT(true); | ||||
| WLED_GLOBAL bool hueApplyBri _INIT(true); | ||||
| WLED_GLOBAL bool hueApplyColor _INIT(true); | ||||
|  | ||||
| // Time CONFIG | ||||
| WLED_GLOBAL bool ntpEnabled _INIT(false);         // get internet time. Only required if you use clock overlays or time-activated macros | ||||
| WLED_GLOBAL bool useAMPM _INIT(false);            // 12h/24h clock format | ||||
| WLED_GLOBAL byte currentTimezone _INIT(0);        // Timezone ID. Refer to timezones array in wled10_ntp.ino | ||||
| WLED_GLOBAL int utcOffsetSecs _INIT(0);           // Seconds to offset from UTC before timzone calculation | ||||
|  | ||||
| WLED_GLOBAL byte overlayDefault _INIT(0);                               // 0: no overlay 1: analog clock 2: single-digit clocl 3: cronixie | ||||
| WLED_GLOBAL byte overlayMin _INIT(0), overlayMax _INIT(ledCount - 1);   // boundaries of overlay mode | ||||
|  | ||||
| WLED_GLOBAL byte analogClock12pixel _INIT(0);               // The pixel in your strip where "midnight" would be | ||||
| WLED_GLOBAL bool analogClockSecondsTrail _INIT(false);      // Display seconds as trail of LEDs instead of a single pixel | ||||
| WLED_GLOBAL bool analogClock5MinuteMarks _INIT(false);      // Light pixels at every 5-minute position | ||||
|  | ||||
| WLED_GLOBAL char cronixieDisplay[7] _INIT("HHMMSS");        // Cronixie Display mask. See wled13_cronixie.ino | ||||
| WLED_GLOBAL bool cronixieBacklight _INIT(true);             // Allow digits to be back-illuminated | ||||
|  | ||||
| WLED_GLOBAL bool countdownMode _INIT(false);                         // Clock will count down towards date | ||||
| WLED_GLOBAL byte countdownYear _INIT(20), countdownMonth _INIT(1);   // Countdown target date, year is last two digits | ||||
| WLED_GLOBAL byte countdownDay  _INIT(1) , countdownHour  _INIT(0); | ||||
| WLED_GLOBAL byte countdownMin  _INIT(0) , countdownSec   _INIT(0); | ||||
|  | ||||
| WLED_GLOBAL byte macroNl   _INIT(0);        // after nightlight delay over | ||||
| WLED_GLOBAL byte macroCountdown _INIT(0); | ||||
| WLED_GLOBAL byte macroAlexaOn _INIT(0), macroAlexaOff _INIT(0); | ||||
| WLED_GLOBAL byte macroButton _INIT(0), macroLongPress _INIT(0), macroDoublePress _INIT(0); | ||||
|  | ||||
| // Security CONFIG | ||||
| WLED_GLOBAL bool otaLock     _INIT(false);  // prevents OTA firmware updates without password. ALWAYS enable if system exposed to any public networks | ||||
| WLED_GLOBAL bool wifiLock    _INIT(false);  // prevents access to WiFi settings when OTA lock is enabled | ||||
| WLED_GLOBAL bool aOtaEnabled _INIT(true);   // ArduinoOTA allows easy updates directly from the IDE. Careful, it does not auto-disable when OTA lock is on | ||||
|  | ||||
| WLED_GLOBAL uint16_t userVar0 _INIT(0), userVar1 _INIT(0); //available for use in usermod | ||||
|  | ||||
| #ifdef WLED_ENABLE_DMX | ||||
|   // dmx CONFIG | ||||
|   WLED_GLOBAL byte DMXChannels _INIT(7);        // number of channels per fixture | ||||
|   WLED_GLOBAL byte DMXFixtureMap[15] _INIT_N(({ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 })); | ||||
|   // assigns the different channels to different functions. See wled21_dmx.ino for more information. | ||||
|   WLED_GLOBAL uint16_t DMXGap _INIT(10);          // gap between the fixtures. makes addressing easier because you don't have to memorize odd numbers when climbing up onto a rig. | ||||
|   WLED_GLOBAL uint16_t DMXStart _INIT(10);        // start address of the first fixture | ||||
|   WLED_GLOBAL uint16_t DMXStartLED _INIT(0);      // LED from which DMX fixtures start | ||||
| #endif | ||||
|  | ||||
| // internal global variable declarations | ||||
| // wifi | ||||
| WLED_GLOBAL bool apActive _INIT(false); | ||||
| WLED_GLOBAL bool forceReconnect _INIT(false); | ||||
| WLED_GLOBAL uint32_t lastReconnectAttempt _INIT(0); | ||||
| WLED_GLOBAL bool interfacesInited _INIT(false); | ||||
| WLED_GLOBAL bool wasConnected _INIT(false); | ||||
|  | ||||
| // color | ||||
| WLED_GLOBAL byte colIT[]     _INIT_N(({ 0, 0, 0, 0 }));         // color that was last sent to LEDs | ||||
| WLED_GLOBAL byte colSecIT[]  _INIT_N(({ 0, 0, 0, 0 })); | ||||
|  | ||||
| WLED_GLOBAL byte lastRandomIndex _INIT(0);        // used to save last random color so the new one is not the same | ||||
|  | ||||
| // transitions | ||||
| WLED_GLOBAL bool transitionActive _INIT(false); | ||||
| WLED_GLOBAL uint16_t transitionDelayDefault _INIT(transitionDelay); | ||||
| WLED_GLOBAL uint16_t transitionDelayTemp _INIT(transitionDelay); | ||||
| WLED_GLOBAL unsigned long transitionStartTime; | ||||
| WLED_GLOBAL float tperLast _INIT(0);        // crossfade transition progress, 0.0f - 1.0f | ||||
| WLED_GLOBAL bool jsonTransitionOnce _INIT(false); | ||||
|  | ||||
| // nightlight | ||||
| WLED_GLOBAL bool nightlightActive _INIT(false); | ||||
| WLED_GLOBAL bool nightlightActiveOld _INIT(false); | ||||
| WLED_GLOBAL uint32_t nightlightDelayMs _INIT(10); | ||||
| WLED_GLOBAL byte nightlightDelayMinsDefault _INIT(nightlightDelayMins); | ||||
| WLED_GLOBAL unsigned long nightlightStartTime; | ||||
| WLED_GLOBAL byte briNlT _INIT(0);                     // current nightlight brightness | ||||
| WLED_GLOBAL byte colNlT[] _INIT_N(({ 0, 0, 0, 0 }));        // current nightlight color | ||||
|  | ||||
| // brightness | ||||
| WLED_GLOBAL unsigned long lastOnTime _INIT(0); | ||||
| WLED_GLOBAL bool offMode _INIT(!turnOnAtBoot); | ||||
| WLED_GLOBAL byte bri _INIT(briS); | ||||
| WLED_GLOBAL byte briOld _INIT(0); | ||||
| WLED_GLOBAL byte briT _INIT(0); | ||||
| WLED_GLOBAL byte briIT _INIT(0); | ||||
| WLED_GLOBAL byte briLast _INIT(128);          // brightness before turned off. Used for toggle function | ||||
| WLED_GLOBAL byte whiteLast _INIT(128);        // white channel before turned off. Used for toggle function | ||||
|  | ||||
| // button | ||||
| WLED_GLOBAL bool buttonPressedBefore _INIT(false); | ||||
| WLED_GLOBAL bool buttonLongPressed _INIT(false); | ||||
| WLED_GLOBAL unsigned long buttonPressedTime _INIT(0); | ||||
| WLED_GLOBAL unsigned long buttonWaitTime _INIT(0); | ||||
|  | ||||
| // notifications | ||||
| WLED_GLOBAL bool notifyDirectDefault _INIT(notifyDirect); | ||||
| WLED_GLOBAL bool receiveNotifications _INIT(true); | ||||
| WLED_GLOBAL unsigned long notificationSentTime _INIT(0); | ||||
| WLED_GLOBAL byte notificationSentCallMode _INIT(NOTIFIER_CALL_MODE_INIT); | ||||
| WLED_GLOBAL bool notificationTwoRequired _INIT(false); | ||||
|  | ||||
| // effects | ||||
| WLED_GLOBAL byte effectCurrent _INIT(0); | ||||
| WLED_GLOBAL byte effectSpeed _INIT(128); | ||||
| WLED_GLOBAL byte effectIntensity _INIT(128); | ||||
| WLED_GLOBAL byte effectPalette _INIT(0); | ||||
| WLED_GLOBAL bool effectChanged _INIT(false); | ||||
|  | ||||
| // network | ||||
| WLED_GLOBAL bool udpConnected _INIT(false), udp2Connected _INIT(false), udpRgbConnected _INIT(false); | ||||
|  | ||||
| // ui style | ||||
| WLED_GLOBAL bool showWelcomePage _INIT(false); | ||||
|  | ||||
| // hue | ||||
| WLED_GLOBAL byte hueError _INIT(HUE_ERROR_INACTIVE); | ||||
| // WLED_GLOBAL uint16_t hueFailCount _INIT(0); | ||||
| WLED_GLOBAL float hueXLast _INIT(0), hueYLast _INIT(0); | ||||
| WLED_GLOBAL uint16_t hueHueLast _INIT(0), hueCtLast _INIT(0); | ||||
| WLED_GLOBAL byte hueSatLast _INIT(0), hueBriLast _INIT(0); | ||||
| WLED_GLOBAL unsigned long hueLastRequestSent _INIT(0); | ||||
| WLED_GLOBAL bool hueAuthRequired _INIT(false); | ||||
| WLED_GLOBAL bool hueReceived _INIT(false); | ||||
| WLED_GLOBAL bool hueStoreAllowed _INIT(false), hueNewKey _INIT(false); | ||||
|  | ||||
| // overlays | ||||
| WLED_GLOBAL byte overlayCurrent _INIT(overlayDefault); | ||||
| WLED_GLOBAL byte overlaySpeed _INIT(200); | ||||
| WLED_GLOBAL unsigned long overlayRefreshMs _INIT(200); | ||||
| WLED_GLOBAL unsigned long overlayRefreshedTime; | ||||
|  | ||||
| // cronixie | ||||
| WLED_GLOBAL byte dP[] _INIT_N(({ 0, 0, 0, 0, 0, 0 })); | ||||
| WLED_GLOBAL bool cronixieInit _INIT(false); | ||||
|  | ||||
| // countdown | ||||
| WLED_GLOBAL unsigned long countdownTime _INIT(1514764800L); | ||||
| WLED_GLOBAL bool countdownOverTriggered _INIT(true); | ||||
|  | ||||
| // timer | ||||
| WLED_GLOBAL byte lastTimerMinute _INIT(0); | ||||
| WLED_GLOBAL byte timerHours[] _INIT_N(({ 0, 0, 0, 0, 0, 0, 0, 0 })); | ||||
| WLED_GLOBAL byte timerMinutes[] _INIT_N(({ 0, 0, 0, 0, 0, 0, 0, 0 })); | ||||
| WLED_GLOBAL byte timerMacro[] _INIT_N(({ 0, 0, 0, 0, 0, 0, 0, 0 })); | ||||
| WLED_GLOBAL byte timerWeekday[] _INIT_N(({ 255, 255, 255, 255, 255, 255, 255, 255 }));        // weekdays to activate on | ||||
| // bit pattern of arr elem: 0b11111111: sun,sat,fri,thu,wed,tue,mon,validity | ||||
|  | ||||
| // blynk | ||||
| WLED_GLOBAL bool blynkEnabled _INIT(false); | ||||
|  | ||||
| // preset cycling | ||||
| WLED_GLOBAL bool presetCyclingEnabled _INIT(false); | ||||
| WLED_GLOBAL byte presetCycleMin _INIT(1), presetCycleMax _INIT(5); | ||||
| WLED_GLOBAL uint16_t presetCycleTime _INIT(12); | ||||
| WLED_GLOBAL unsigned long presetCycledTime _INIT(0); | ||||
| WLED_GLOBAL byte presetCycCurr _INIT(presetCycleMin); | ||||
| WLED_GLOBAL bool saveCurrPresetCycConf _INIT(false); | ||||
|  | ||||
| WLED_GLOBAL int16_t currentPlaylist _INIT(0); | ||||
|  | ||||
| // realtime | ||||
| WLED_GLOBAL byte realtimeMode _INIT(REALTIME_MODE_INACTIVE); | ||||
| WLED_GLOBAL byte realtimeOverride _INIT(REALTIME_OVERRIDE_NONE); | ||||
| WLED_GLOBAL IPAddress realtimeIP _INIT_N(((0, 0, 0, 0)));; | ||||
| WLED_GLOBAL unsigned long realtimeTimeout _INIT(0); | ||||
| WLED_GLOBAL uint8_t tpmPacketCount _INIT(0); | ||||
| WLED_GLOBAL uint16_t tpmPayloadFrameSize _INIT(0); | ||||
|  | ||||
| // mqtt | ||||
| WLED_GLOBAL unsigned long lastMqttReconnectAttempt _INIT(0); | ||||
| WLED_GLOBAL unsigned long lastInterfaceUpdate _INIT(0); | ||||
| WLED_GLOBAL byte interfaceUpdateCallMode _INIT(NOTIFIER_CALL_MODE_INIT); | ||||
| WLED_GLOBAL char mqttStatusTopic[40] _INIT("");        // this must be global because of async handlers | ||||
|  | ||||
| // alexa udp | ||||
| WLED_GLOBAL String escapedMac; | ||||
| #ifndef WLED_DISABLE_ALEXA | ||||
|   WLED_GLOBAL Espalexa espalexa; | ||||
|   WLED_GLOBAL EspalexaDevice* espalexaDevice; | ||||
| #endif | ||||
|  | ||||
| // dns server | ||||
| WLED_GLOBAL DNSServer dnsServer; | ||||
|  | ||||
| // network time | ||||
| WLED_GLOBAL bool ntpConnected _INIT(false); | ||||
| WLED_GLOBAL time_t localTime _INIT(0); | ||||
| WLED_GLOBAL unsigned long ntpLastSyncTime _INIT(999000000L); | ||||
| WLED_GLOBAL unsigned long ntpPacketSentTime _INIT(999000000L); | ||||
| WLED_GLOBAL IPAddress ntpServerIP; | ||||
| WLED_GLOBAL uint16_t ntpLocalPort _INIT(2390); | ||||
| WLED_GLOBAL uint16_t rolloverMillis _INIT(0); | ||||
|  | ||||
| // Temp buffer | ||||
| WLED_GLOBAL char* obuf; | ||||
| WLED_GLOBAL uint16_t olen _INIT(0); | ||||
|  | ||||
| // General filesystem | ||||
| WLED_GLOBAL size_t fsBytesUsed _INIT(0); | ||||
| WLED_GLOBAL size_t fsBytesTotal _INIT(0); | ||||
| WLED_GLOBAL unsigned long presetsModifiedTime _INIT(0L); | ||||
| WLED_GLOBAL JsonDocument* fileDoc; | ||||
| WLED_GLOBAL bool doCloseFile _INIT(false); | ||||
|  | ||||
| // presets | ||||
| WLED_GLOBAL int16_t currentPreset _INIT(-1); | ||||
| WLED_GLOBAL bool isPreset _INIT(false); | ||||
|  | ||||
| WLED_GLOBAL byte errorFlag _INIT(0); | ||||
|  | ||||
| WLED_GLOBAL String messageHead, messageSub; | ||||
| WLED_GLOBAL byte optionType; | ||||
|  | ||||
| WLED_GLOBAL bool doReboot _INIT(false);        // flag to initiate reboot from async handlers | ||||
| WLED_GLOBAL bool doPublishMqtt _INIT(false); | ||||
|  | ||||
| // server library objects | ||||
| WLED_GLOBAL AsyncWebServer server _INIT_N(((80))); | ||||
| #ifdef WLED_ENABLE_WEBSOCKETS | ||||
| WLED_GLOBAL AsyncWebSocket ws _INIT_N((("/ws"))); | ||||
| #endif | ||||
| WLED_GLOBAL AsyncClient* hueClient _INIT(NULL); | ||||
| WLED_GLOBAL AsyncMqttClient* mqtt _INIT(NULL); | ||||
|  | ||||
| // udp interface objects | ||||
| WLED_GLOBAL WiFiUDP notifierUdp, rgbUdp, notifier2Udp; | ||||
| WLED_GLOBAL WiFiUDP ntpUdp; | ||||
| WLED_GLOBAL ESPAsyncE131 e131 _INIT_N(((handleE131Packet))); | ||||
| WLED_GLOBAL bool e131NewData _INIT(false); | ||||
|  | ||||
| // led fx library object | ||||
| WLED_GLOBAL BusManager busses _INIT(BusManager()); | ||||
| WLED_GLOBAL WS2812FX strip _INIT(WS2812FX()); | ||||
| WLED_GLOBAL BusConfig* busConfigs[WLED_MAX_BUSSES] _INIT({nullptr}); //temporary, to remember values from network callback until after  | ||||
| WLED_GLOBAL bool doInitBusses _INIT(false); | ||||
|  | ||||
| // Usermod manager | ||||
| WLED_GLOBAL UsermodManager usermods _INIT(UsermodManager()); | ||||
|  | ||||
| // Status LED | ||||
| #if STATUSLED | ||||
|   WLED_GLOBAL unsigned long ledStatusLastMillis _INIT(0); | ||||
|   WLED_GLOBAL unsigned short ledStatusType _INIT(0); // current status type - corresponds to number of blinks per second | ||||
|   WLED_GLOBAL bool ledStatusState _INIT(0); // the current LED state | ||||
| #endif | ||||
|  | ||||
| // enable additional debug output | ||||
| #ifdef WLED_DEBUG | ||||
|   #ifndef ESP8266 | ||||
|   #include <rom/rtc.h> | ||||
|   #endif | ||||
|   #define DEBUG_PRINT(x) Serial.print(x) | ||||
|   #define DEBUG_PRINTLN(x) Serial.println(x) | ||||
|   #define DEBUG_PRINTF(x...) Serial.printf(x) | ||||
| #else | ||||
|   #define DEBUG_PRINT(x) | ||||
|   #define DEBUG_PRINTLN(x) | ||||
|   #define DEBUG_PRINTF(x) | ||||
| #endif | ||||
|  | ||||
| #ifdef WLED_DEBUG_FS | ||||
|   #define DEBUGFS_PRINT(x) Serial.print(x) | ||||
|   #define DEBUGFS_PRINTLN(x) Serial.println(x) | ||||
|   #define DEBUGFS_PRINTF(x...) Serial.printf(x) | ||||
| #else | ||||
|   #define DEBUGFS_PRINT(x) | ||||
|   #define DEBUGFS_PRINTLN(x) | ||||
|   #define DEBUGFS_PRINTF(x...) | ||||
| #endif | ||||
|  | ||||
| // debug macro variable definitions | ||||
| #ifdef WLED_DEBUG | ||||
|   WLED_GLOBAL unsigned long debugTime _INIT(0); | ||||
|   WLED_GLOBAL int lastWifiState _INIT(3); | ||||
|   WLED_GLOBAL unsigned long wifiStateChangedTime _INIT(0); | ||||
|   WLED_GLOBAL int loops _INIT(0); | ||||
| #endif | ||||
|  | ||||
| #ifdef ARDUINO_ARCH_ESP32 | ||||
|   #define WLED_CONNECTED (WiFi.status() == WL_CONNECTED || ETH.localIP()[0] != 0) | ||||
| #else | ||||
|   #define WLED_CONNECTED (WiFi.status() == WL_CONNECTED) | ||||
| #endif | ||||
| #define WLED_WIFI_CONFIGURED (strlen(clientSSID) >= 1 && strcmp(clientSSID, DEFAULT_CLIENT_SSID) != 0) | ||||
| #define WLED_MQTT_CONNECTED (mqtt != nullptr && mqtt->connected()) | ||||
|  | ||||
| // append new c string to temp buffer efficiently | ||||
| bool oappend(const char* txt); | ||||
| // append new number to temp buffer efficiently | ||||
| bool oappendi(int i); | ||||
|  | ||||
| class WLED { | ||||
| public: | ||||
|   WLED(); | ||||
|   static WLED& instance() | ||||
|   { | ||||
|     static WLED instance; | ||||
|     return instance; | ||||
|   } | ||||
|  | ||||
|   // boot starts here | ||||
|   void setup(); | ||||
|  | ||||
|   void loop(); | ||||
|   void reset(); | ||||
|  | ||||
|   void beginStrip(); | ||||
|   void handleConnection(); | ||||
|   void initAP(bool resetAP = false); | ||||
|   void initConnection(); | ||||
|   void initInterfaces(); | ||||
|   void handleStatusLED(); | ||||
| }; | ||||
| #endif        // WLED_H | ||||
| @@ -1,582 +1,21 @@ | ||||
| /* | ||||
|  * Main sketch, global variable declarations | ||||
|  */ | ||||
| /* | ||||
|  * @title WLED project sketch | ||||
|  * @version 0.9.1 | ||||
|  * @author Christian Schwinne | ||||
|  * WLED Arduino IDE compatibility file. | ||||
|  *  | ||||
|  * Where has everything gone? | ||||
|  *  | ||||
|  * In April 2020, the project's structure underwent a major change.  | ||||
|  * Global variables are now found in file "wled.h" | ||||
|  * Global function declarations are found in "fcn_declare.h" | ||||
|  *  | ||||
|  * Usermod compatibility: Existing wled06_usermod.ino mods should continue to work. Delete usermod.cpp. | ||||
|  * New usermods should use usermod.cpp instead. | ||||
|  */ | ||||
| #include "wled.h" | ||||
|  | ||||
|  | ||||
| //ESP8266-01 (blue) got too little storage space to work with all features of WLED. To use it, you must use ESP8266 Arduino Core v2.4.2 and the setting 512K(No SPIFFS). | ||||
|  | ||||
| //ESP8266-01 (black) has 1MB flash and can thus fit the whole program. Use 1M(64K SPIFFS). | ||||
| //Uncomment some of the following lines to disable features to compile for ESP8266-01 (max flash size 434kB): | ||||
|  | ||||
| //You are required to disable over-the-air updates: | ||||
| //#define WLED_DISABLE_OTA         //saves 14kb | ||||
|  | ||||
| //You need to choose some of these features to disable: | ||||
| //#define WLED_DISABLE_ALEXA       //saves 11kb | ||||
| //#define WLED_DISABLE_BLYNK       //saves 6kb | ||||
| //#define WLED_DISABLE_CRONIXIE    //saves 3kb | ||||
| //#define WLED_DISABLE_HUESYNC     //saves 4kb | ||||
| //#define WLED_DISABLE_INFRARED    //there is no pin left for this on ESP8266-01, saves 25kb (!) | ||||
| #define WLED_ENABLE_MQTT           //saves 12kb | ||||
| #define WLED_ENABLE_ADALIGHT       //saves 500b only | ||||
|  | ||||
| #define WLED_DISABLE_FILESYSTEM    //SPIFFS is not used by any WLED feature yet | ||||
| //#define WLED_ENABLE_FS_SERVING   //Enable sending html file from SPIFFS before serving progmem version | ||||
| //#define WLED_ENABLE_FS_EDITOR    //enable /edit page for editing SPIFFS content. Will also be disabled with OTA lock | ||||
|  | ||||
| //to toggle usb serial debug (un)comment the following line | ||||
| //#define WLED_DEBUG | ||||
|  | ||||
| //library inclusions | ||||
| #include <Arduino.h> | ||||
| #ifdef ESP8266 | ||||
|  #include <ESP8266WiFi.h> | ||||
|  #include <ESP8266mDNS.h> | ||||
|  #include <ESPAsyncTCP.h> | ||||
| #else | ||||
|  #include <WiFi.h> | ||||
|  #include "esp_wifi.h" | ||||
|  #include <ESPmDNS.h> | ||||
|  #include <AsyncTCP.h> | ||||
|  #include "SPIFFS.h" | ||||
| #endif | ||||
|  | ||||
| #include <ESPAsyncWebServer.h> | ||||
| #include <EEPROM.h> | ||||
| #include <WiFiUdp.h> | ||||
| #include <DNSServer.h> | ||||
| #ifndef WLED_DISABLE_OTA | ||||
|  #include <ArduinoOTA.h> | ||||
| #endif | ||||
| #include <SPIFFSEditor.h> | ||||
| #include "src/dependencies/time/TimeLib.h" | ||||
| #include "src/dependencies/timezone/Timezone.h" | ||||
| #ifndef WLED_DISABLE_ALEXA | ||||
|  #define ESPALEXA_ASYNC | ||||
|  #define ESPALEXA_NO_SUBPAGE | ||||
|  #define ESPALEXA_MAXDEVICES 1 | ||||
|  //#define ESPALEXA_DEBUG | ||||
|  #include "src/dependencies/espalexa/Espalexa.h" | ||||
| #endif | ||||
| #ifndef WLED_DISABLE_BLYNK | ||||
|  #include "src/dependencies/blynk/BlynkSimpleEsp.h" | ||||
| #endif | ||||
| #include "src/dependencies/e131/ESPAsyncE131.h" | ||||
| #include "src/dependencies/async-mqtt-client/AsyncMqttClient.h" | ||||
| #include "src/dependencies/json/AsyncJson-v6.h" | ||||
| #include "src/dependencies/json/ArduinoJson-v6.h" | ||||
| #include "html_ui.h" | ||||
| #include "html_settings.h" | ||||
| #include "html_other.h" | ||||
| #include "FX.h" | ||||
| #include "ir_codes.h" | ||||
|  | ||||
|  | ||||
| #if IR_PIN < 0 | ||||
|  #ifndef WLED_DISABLE_INFRARED | ||||
|   #define WLED_DISABLE_INFRARED | ||||
|  #endif | ||||
| #endif | ||||
|  | ||||
|  #ifndef WLED_DISABLE_INFRARED | ||||
|   #include <IRremoteESP8266.h> | ||||
|   #include <IRrecv.h> | ||||
|   #include <IRutils.h> | ||||
|  #endif | ||||
|  | ||||
| //version code in format yymmddb (b = daily build) | ||||
| #define VERSION 2002181 | ||||
|  | ||||
| char versionString[] = "0.9.1"; | ||||
|  | ||||
|  | ||||
| //AP and OTA default passwords (for maximum change them!) | ||||
| char apPass[65] = "wled1234"; | ||||
| char otaPass[33] = "wledota"; | ||||
|  | ||||
|  | ||||
| //Hardware CONFIG (only changeble HERE, not at runtime) | ||||
| //LED strip pin, button pin and IR pin changeable in NpbWrapper.h! | ||||
|  | ||||
| byte auxDefaultState   = 0;                   //0: input 1: high 2: low | ||||
| byte auxTriggeredState = 0;                   //0: input 1: high 2: low | ||||
| char ntpServerName[33] = "0.wled.pool.ntp.org";//NTP server to use | ||||
|  | ||||
|  | ||||
| //WiFi CONFIG (all these can be changed via web UI, no need to set them here) | ||||
| char clientSSID[33] = "Your_Network"; | ||||
| char clientPass[65] = ""; | ||||
| char cmDNS[33] = "x";                         //mDNS address (placeholder, will be replaced by wledXXXXXXXXXXXX.local) | ||||
| char apSSID[33] = "";                         //AP off by default (unless setup) | ||||
| byte apChannel = 1;                           //2.4GHz WiFi AP channel (1-13) | ||||
| byte apHide = 0;                              //hidden AP SSID | ||||
| byte apBehavior = 0;                          //0: Open AP when no connection after boot 1: Open when no connection 2: Always open 3: Only when button pressed for 6 sec | ||||
| IPAddress staticIP(0, 0, 0, 0);               //static IP of ESP | ||||
| IPAddress staticGateway(0, 0, 0, 0);          //gateway (router) IP | ||||
| IPAddress staticSubnet(255, 255, 255, 0);     //most common subnet in home networks | ||||
|  | ||||
| //LED CONFIG | ||||
| uint16_t ledCount = 30;                       //overcurrent prevented by ABL | ||||
| bool useRGBW = false;                         //SK6812 strips can contain an extra White channel | ||||
| bool autoRGBtoRGBW = false;                   //if RGBW enabled, calculate White channel from RGB | ||||
| #define ABL_MILLIAMPS_DEFAULT 850;            //auto lower brightness to stay close to milliampere limit | ||||
| bool turnOnAtBoot  = true;                    //turn on LEDs at power-up | ||||
| byte bootPreset = 0;                          //save preset to load after power-up | ||||
|  | ||||
| byte col[]{255, 160, 0, 0};                   //default RGB(W) color | ||||
| byte colSec[]{0, 0, 0, 0};                    //default RGB(W) secondary color | ||||
| byte briS = 128;                              //default brightness | ||||
|  | ||||
| byte nightlightTargetBri = 0;                 //brightness after nightlight is over | ||||
| byte nightlightDelayMins = 60; | ||||
| bool nightlightFade = true;                   //if enabled, light will gradually dim towards the target bri. Otherwise, it will instantly set after delay over | ||||
| bool nightlightColorFade = false;             //if enabled, light will gradually fade color from primary to secondary color. | ||||
| bool fadeTransition = true;                   //enable crossfading color transition | ||||
| bool enableSecTransition = true;              //also enable transition for secondary color | ||||
| uint16_t transitionDelay = 750;               //default crossfade duration in ms | ||||
|  | ||||
| bool skipFirstLed = false;                    //ignore first LED in strip (useful if you need the LED as signal repeater) | ||||
| byte briMultiplier =  100;                    //% of brightness to set (to limit power, if you set it to 50 and set bri to 255, actual brightness will be 127) | ||||
|  | ||||
|  | ||||
| //User Interface CONFIG | ||||
| char serverDescription[33] = "WLED";          //Name of module | ||||
| bool syncToggleReceive = false;               //UIs which only have a single button for sync should toggle send+receive if this is true, only send otherwise | ||||
|  | ||||
|  | ||||
| //Sync CONFIG | ||||
| bool buttonEnabled =  true; | ||||
| byte irEnabled     =  0;                      //Infrared receiver | ||||
|  | ||||
| uint16_t udpPort    = 21324;                  //WLED notifier default port | ||||
| uint16_t udpRgbPort = 19446;                  //Hyperion port | ||||
|  | ||||
| bool receiveNotificationBrightness = true;    //apply brightness from incoming notifications | ||||
| bool receiveNotificationColor      = true;    //apply color | ||||
| bool receiveNotificationEffects    = true;    //apply effects setup | ||||
| bool notifyDirect = false;                    //send notification if change via UI or HTTP API | ||||
| bool notifyButton = false;                    //send if updated by button or infrared remote | ||||
| bool notifyAlexa  = false;                    //send notification if updated via Alexa | ||||
| bool notifyMacro  = false;                    //send notification for macro | ||||
| bool notifyHue    =  true;                    //send notification if Hue light changes | ||||
| bool notifyTwice  = false;                    //notifications use UDP: enable if devices don't sync reliably | ||||
|  | ||||
| bool alexaEnabled = true;                     //enable device discovery by Amazon Echo | ||||
| char alexaInvocationName[33] = "Light";       //speech control name of device. Choose something voice-to-text can understand | ||||
|  | ||||
| char blynkApiKey[36] = "";                    //Auth token for Blynk server. If empty, no connection will be made | ||||
|  | ||||
| uint16_t realtimeTimeoutMs = 2500;            //ms timeout of realtime mode before returning to normal mode | ||||
| int  arlsOffset = 0;                          //realtime LED offset | ||||
| bool receiveDirect    =  true;                //receive UDP realtime | ||||
| bool arlsDisableGammaCorrection = true;       //activate if gamma correction is handled by the source | ||||
| bool arlsForceMaxBri = false;                 //enable to force max brightness if source has very dark colors that would be black | ||||
|  | ||||
| uint16_t e131Universe = 1;                    //settings for E1.31 (sACN) protocol (only DMX_MODE_MULTIPLE_* can span over consequtive universes) | ||||
| #define  DMX_MODE_DISABLED      0             //not used | ||||
| #define  DMX_MODE_SINGLE_RGB    1             //all LEDs same RGB color (3 channels) | ||||
| #define  DMX_MODE_SINGLE_DRGB   2             //all LEDs same RGB color and master dimmer (4 channels) | ||||
| #define  DMX_MODE_EFFECT        3             //trigger standalone effects of WLED (11 channels) | ||||
| #define  DMX_MODE_MULTIPLE_RGB  4             //every LED is addressed with its own RGB (ledCount * 3 channels) | ||||
| #define  DMX_MODE_MULTIPLE_DRGB 5             //every LED is addressed with its own RGB and share a master dimmer (ledCount * 3 + 1 channels) | ||||
| uint8_t  DMXMode;                             //DMX mode (s.a.) | ||||
| uint16_t DMXAddress;                          //DMX start address of fixture, a.k.a. first Channel [for E1.31 (sACN) protocol] | ||||
| uint8_t  DMXOldDimmer = 0;                    //only update brightness on change | ||||
| uint8_t  e131LastSequenceNumber = 0;          //to detect packet loss | ||||
| bool     e131Multicast = false;               //multicast or unicast | ||||
|  | ||||
| bool mqttEnabled = false; | ||||
| char mqttDeviceTopic[33] = "";                //main MQTT topic (individual per device, default is wled/mac) | ||||
| char mqttGroupTopic[33] = "wled/all";         //second MQTT topic (for example to group devices) | ||||
| char mqttServer[33] = "";                     //both domains and IPs should work (no SSL) | ||||
| char mqttUser[41] = "";                       //optional: username for MQTT auth | ||||
| char mqttPass[41] = "";                       //optional: password for MQTT auth | ||||
| char mqttClientID[41] = "";                   //override the client ID | ||||
| uint16_t mqttPort = 1883; | ||||
|  | ||||
| bool huePollingEnabled = false;               //poll hue bridge for light state | ||||
| uint16_t huePollIntervalMs = 2500;            //low values (< 1sec) may cause lag but offer quicker response | ||||
| char hueApiKey[47] = "api";                   //key token will be obtained from bridge | ||||
| byte huePollLightId = 1;                      //ID of hue lamp to sync to. Find the ID in the hue app ("about" section) | ||||
| IPAddress hueIP = (0,0,0,0);                  //IP address of the bridge | ||||
| bool hueApplyOnOff = true; | ||||
| bool hueApplyBri   = true; | ||||
| bool hueApplyColor = true; | ||||
|  | ||||
|  | ||||
| //Time CONFIG | ||||
| bool ntpEnabled = false;                      //get internet time. Only required if you use clock overlays or time-activated macros | ||||
| bool useAMPM = false;                         //12h/24h clock format | ||||
| byte currentTimezone = 0;                     //Timezone ID. Refer to timezones array in wled10_ntp.ino | ||||
| int  utcOffsetSecs   = 0;                     //Seconds to offset from UTC before timzone calculation | ||||
|  | ||||
| byte overlayDefault = 0;                      //0: no overlay 1: analog clock 2: single-digit clocl 3: cronixie | ||||
| byte overlayMin = 0, overlayMax = ledCount-1; //boundaries of overlay mode | ||||
|  | ||||
| byte analogClock12pixel = 0;                  //The pixel in your strip where "midnight" would be | ||||
| bool analogClockSecondsTrail = false;         //Display seconds as trail of LEDs instead of a single pixel | ||||
| bool analogClock5MinuteMarks = false;         //Light pixels at every 5-minute position | ||||
|  | ||||
| char cronixieDisplay[7] = "HHMMSS";           //Cronixie Display mask. See wled13_cronixie.ino | ||||
| bool cronixieBacklight = true;                //Allow digits to be back-illuminated | ||||
|  | ||||
| bool countdownMode = false;                   //Clock will count down towards date | ||||
| byte countdownYear = 20, countdownMonth = 1;  //Countdown target date, year is last two digits | ||||
| byte countdownDay  =  1, countdownHour  = 0; | ||||
| byte countdownMin  =  0, countdownSec   = 0; | ||||
|  | ||||
|  | ||||
| byte macroBoot = 0;                           //macro loaded after startup | ||||
| byte macroNl = 0;                             //after nightlight delay over | ||||
| byte macroCountdown = 0; | ||||
| byte macroAlexaOn = 0, macroAlexaOff = 0; | ||||
| byte macroButton = 0, macroLongPress = 0, macroDoublePress = 0; | ||||
|  | ||||
|  | ||||
| //Security CONFIG | ||||
| bool otaLock = false;                         //prevents OTA firmware updates without password. ALWAYS enable if system exposed to any public networks | ||||
| bool wifiLock = false;                        //prevents access to WiFi settings when OTA lock is enabled | ||||
| bool aOtaEnabled = true;                      //ArduinoOTA allows easy updates directly from the IDE. Careful, it does not auto-disable when OTA lock is on | ||||
|  | ||||
|  | ||||
| uint16_t userVar0 = 0, userVar1 = 0; | ||||
|  | ||||
|  | ||||
| //internal global variable declarations | ||||
| //wifi | ||||
| bool apActive = false; | ||||
| bool forceReconnect = false; | ||||
| uint32_t lastReconnectAttempt = 0; | ||||
| bool interfacesInited = false; | ||||
| bool wasConnected = false; | ||||
|  | ||||
| //color | ||||
| byte colOld[]{0, 0, 0, 0};                    //color before transition | ||||
| byte colT[]{0, 0, 0, 0};                      //current color | ||||
| byte colIT[]{0, 0, 0, 0};                     //color that was last sent to LEDs | ||||
| byte colSecT[]{0, 0, 0, 0}; | ||||
| byte colSecOld[]{0, 0, 0, 0}; | ||||
| byte colSecIT[]{0, 0, 0, 0}; | ||||
|  | ||||
| byte lastRandomIndex = 0;                     //used to save last random color so the new one is not the same | ||||
|  | ||||
| //transitions | ||||
| bool transitionActive = false; | ||||
| uint16_t transitionDelayDefault = transitionDelay; | ||||
| uint16_t transitionDelayTemp = transitionDelay; | ||||
| unsigned long transitionStartTime; | ||||
| float tperLast = 0;                           //crossfade transition progress, 0.0f - 1.0f | ||||
| bool jsonTransitionOnce = false; | ||||
|  | ||||
| //nightlight | ||||
| bool nightlightActive = false; | ||||
| bool nightlightActiveOld = false; | ||||
| uint32_t nightlightDelayMs = 10; | ||||
| uint8_t nightlightDelayMinsDefault = nightlightDelayMins; | ||||
| unsigned long nightlightStartTime; | ||||
| byte briNlT = 0;                              //current nightlight brightness | ||||
| byte colNlT[]{0, 0, 0, 0};                    //current nightlight color | ||||
|  | ||||
| //brightness | ||||
| unsigned long lastOnTime = 0; | ||||
| bool offMode = !turnOnAtBoot; | ||||
| byte bri = briS; | ||||
| byte briOld = 0; | ||||
| byte briT = 0; | ||||
| byte briIT = 0; | ||||
| byte briLast = 128;                           //brightness before turned off. Used for toggle function | ||||
| byte whiteLast = 128;                         //white channel before turned off. Used for toggle function | ||||
|  | ||||
| //button | ||||
| bool buttonPressedBefore = false; | ||||
| bool buttonLongPressed = false; | ||||
| unsigned long buttonPressedTime = 0; | ||||
| unsigned long buttonWaitTime = 0; | ||||
|  | ||||
| //notifications | ||||
| bool notifyDirectDefault = notifyDirect; | ||||
| bool receiveNotifications = true; | ||||
| unsigned long notificationSentTime = 0; | ||||
| byte notificationSentCallMode = 0; | ||||
| bool notificationTwoRequired = false; | ||||
|  | ||||
| //effects | ||||
| byte effectCurrent = 0; | ||||
| byte effectSpeed = 128; | ||||
| byte effectIntensity = 128; | ||||
| byte effectPalette = 0; | ||||
|  | ||||
| //network | ||||
| bool udpConnected = false, udpRgbConnected = false; | ||||
|  | ||||
| //ui style | ||||
| bool showWelcomePage = false; | ||||
|  | ||||
| //hue | ||||
| char hueError[25] = "Inactive"; | ||||
| //uint16_t hueFailCount = 0; | ||||
| float hueXLast=0, hueYLast=0; | ||||
| uint16_t hueHueLast=0, hueCtLast=0; | ||||
| byte hueSatLast=0, hueBriLast=0; | ||||
| unsigned long hueLastRequestSent = 0; | ||||
| bool hueAuthRequired = false; | ||||
| bool hueReceived = false; | ||||
| bool hueStoreAllowed = false, hueNewKey = false; | ||||
|  | ||||
| //overlays | ||||
| byte overlayCurrent = overlayDefault; | ||||
| byte overlaySpeed = 200; | ||||
| unsigned long overlayRefreshMs = 200; | ||||
| unsigned long overlayRefreshedTime; | ||||
|  | ||||
| //cronixie | ||||
| byte dP[]{0,0,0,0,0,0}; | ||||
| bool cronixieInit = false; | ||||
|  | ||||
| //countdown | ||||
| unsigned long countdownTime = 1514764800L; | ||||
| bool countdownOverTriggered = true; | ||||
|  | ||||
| //timer | ||||
| byte lastTimerMinute = 0; | ||||
| byte timerHours[]   = {0,0,0,0,0,0,0,0}; | ||||
| byte timerMinutes[] = {0,0,0,0,0,0,0,0}; | ||||
| byte timerMacro[]   = {0,0,0,0,0,0,0,0}; | ||||
| byte timerWeekday[] = {255,255,255,255,255,255,255,255}; //weekdays to activate on | ||||
| //bit pattern of arr elem: 0b11111111: sun,sat,fri,thu,wed,tue,mon,validity | ||||
|  | ||||
| //blynk | ||||
| bool blynkEnabled = false; | ||||
|  | ||||
| //preset cycling | ||||
| bool presetCyclingEnabled = false; | ||||
| byte presetCycleMin = 1, presetCycleMax = 5; | ||||
| uint16_t presetCycleTime = 1250; | ||||
| unsigned long presetCycledTime = 0; byte presetCycCurr = presetCycleMin; | ||||
| bool presetApplyBri = false, presetApplyCol = true, presetApplyFx = true; | ||||
| bool saveCurrPresetCycConf = false; | ||||
|  | ||||
| //realtime | ||||
| #define REALTIME_MODE_INACTIVE 0 | ||||
| #define REALTIME_MODE_GENERIC  1 | ||||
| #define REALTIME_MODE_UDP      2 | ||||
| #define REALTIME_MODE_HYPERION 3 | ||||
| #define REALTIME_MODE_E131     4 | ||||
| #define REALTIME_MODE_ADALIGHT 5 | ||||
| byte realtimeMode = 0; | ||||
| IPAddress realtimeIP = (0,0,0,0); | ||||
| unsigned long realtimeTimeout = 0; | ||||
|  | ||||
|  | ||||
| //mqtt | ||||
| long lastMqttReconnectAttempt = 0; | ||||
| long lastInterfaceUpdate = 0; | ||||
| byte interfaceUpdateCallMode = 0; | ||||
| char mqttStatusTopic[40] = ""; //this must be global because of async handlers | ||||
|  | ||||
| #if AUXPIN >= 0 | ||||
| //auxiliary debug pin | ||||
| byte auxTime = 0; | ||||
| unsigned long auxStartTime = 0; | ||||
| bool auxActive = false, auxActiveBefore = false; | ||||
| #endif | ||||
|  | ||||
| //alexa udp | ||||
| String escapedMac; | ||||
| #ifndef WLED_DISABLE_ALEXA | ||||
| Espalexa espalexa; | ||||
| EspalexaDevice* espalexaDevice; | ||||
| #endif | ||||
|  | ||||
| //dns server | ||||
| DNSServer dnsServer; | ||||
|  | ||||
| //network time | ||||
| bool ntpConnected = false; | ||||
| time_t local = 0; | ||||
| unsigned long ntpLastSyncTime = 999000000L; | ||||
| unsigned long ntpPacketSentTime = 999000000L; | ||||
| IPAddress ntpServerIP; | ||||
| uint16_t ntpLocalPort = 2390; | ||||
| #define NTP_PACKET_SIZE 48 | ||||
|  | ||||
| #define MAX_LEDS 1500 | ||||
| #define MAX_LEDS_DMA 500 | ||||
|  | ||||
| //string temp buffer (now stored in stack locally) | ||||
| #define OMAX 2048 | ||||
| char* obuf; | ||||
| uint16_t olen = 0; | ||||
|  | ||||
| uint16_t savedPresets = 0; | ||||
| int8_t currentPreset = -1; | ||||
| bool isPreset = false; | ||||
|  | ||||
| byte errorFlag = 0; | ||||
|  | ||||
| String messageHead, messageSub; | ||||
| byte optionType; | ||||
|  | ||||
| bool doReboot = false; //flag to initiate reboot from async handlers | ||||
| bool doPublishMqtt = false; | ||||
|  | ||||
| //server library objects | ||||
| AsyncWebServer server(80); | ||||
| AsyncClient* hueClient = NULL; | ||||
| AsyncMqttClient* mqtt = NULL; | ||||
|  | ||||
| //function prototypes | ||||
| void colorFromUint32(uint32_t,bool=false); | ||||
| void serveMessage(AsyncWebServerRequest*,uint16_t,String,String,byte); | ||||
| void handleE131Packet(e131_packet_t*, IPAddress); | ||||
| void arlsLock(uint32_t,byte); | ||||
| void handleOverlayDraw(); | ||||
|  | ||||
| #define E131_MAX_UNIVERSE_COUNT 9 | ||||
|  | ||||
| //udp interface objects | ||||
| WiFiUDP notifierUdp, rgbUdp; | ||||
| WiFiUDP ntpUdp; | ||||
| ESPAsyncE131 e131(handleE131Packet); | ||||
| bool e131NewData = false; | ||||
|  | ||||
| //led fx library object | ||||
| WS2812FX strip = WS2812FX(); | ||||
|  | ||||
| #define WLED_CONNECTED (WiFi.status() == WL_CONNECTED) | ||||
| #define WLED_WIFI_CONFIGURED (strlen(clientSSID) >= 1 && strcmp(clientSSID,"Your_Network") != 0) | ||||
|  | ||||
| //debug macros | ||||
| #ifdef WLED_DEBUG | ||||
|  #define DEBUG_PRINT(x)  Serial.print (x) | ||||
|  #define DEBUG_PRINTLN(x) Serial.println (x) | ||||
|  #define DEBUG_PRINTF(x) Serial.printf (x) | ||||
|  unsigned long debugTime = 0; | ||||
|  int lastWifiState = 3; | ||||
|  unsigned long wifiStateChangedTime = 0; | ||||
|  int loops = 0; | ||||
| #else | ||||
|  #define DEBUG_PRINT(x) | ||||
|  #define DEBUG_PRINTLN(x) | ||||
|  #define DEBUG_PRINTF(x) | ||||
| #endif | ||||
|  | ||||
| //filesystem | ||||
| #ifndef WLED_DISABLE_FILESYSTEM | ||||
|  #include <FS.h> | ||||
|  #ifdef ARDUINO_ARCH_ESP32 | ||||
|   #include "SPIFFS.h" | ||||
|  #endif | ||||
|  #include "SPIFFSEditor.h" | ||||
| #endif | ||||
|  | ||||
| //turns all LEDs off and restarts ESP | ||||
| void reset() | ||||
| { | ||||
|   briT = 0; | ||||
|   long dly = millis(); | ||||
|   while(millis() - dly < 250) | ||||
|   { | ||||
|     yield(); //enough time to send response to client | ||||
|   } | ||||
|   setAllLeds(); | ||||
|   DEBUG_PRINTLN("MODULE RESET"); | ||||
|   ESP.restart(); | ||||
| } | ||||
|  | ||||
|  | ||||
| //append new c string to temp buffer efficiently | ||||
| bool oappend(const char* txt) | ||||
| { | ||||
|   uint16_t len = strlen(txt); | ||||
|   if (olen + len >= OMAX) return false; //buffer full | ||||
|   strcpy(obuf + olen, txt); | ||||
|   olen += len; | ||||
|   return true; | ||||
| } | ||||
|  | ||||
|  | ||||
| //append new number to temp buffer efficiently | ||||
| bool oappendi(int i) | ||||
| { | ||||
|   char s[11]; | ||||
|   sprintf(s,"%ld", i); | ||||
|   return oappend(s); | ||||
| } | ||||
|  | ||||
|  | ||||
| //boot starts here | ||||
| void setup() { | ||||
|   wledInit(); | ||||
|   WLED::instance().setup(); | ||||
| } | ||||
|  | ||||
|  | ||||
| //main program loop | ||||
| void loop() { | ||||
|   handleIR();          //2nd call to function needed for ESP32 to return valid results -- should be good for ESP8266, too | ||||
|   handleConnection(); | ||||
|   handleSerial(); | ||||
|   handleNotifications(); | ||||
|   handleTransitions(); | ||||
|   userLoop(); | ||||
|  | ||||
|   yield(); | ||||
|   handleIO(); | ||||
|   handleIR(); | ||||
|   handleNetworkTime(); | ||||
|   handleAlexa(); | ||||
|  | ||||
|   handleOverlays(); | ||||
|   yield(); | ||||
|   if (doReboot) reset(); | ||||
|  | ||||
|   if (!realtimeMode) //block stuff if WARLS/Adalight is enabled | ||||
|   { | ||||
|     if (apActive) dnsServer.processNextRequest(); | ||||
|     #ifndef WLED_DISABLE_OTA | ||||
|     if (WLED_CONNECTED && aOtaEnabled) ArduinoOTA.handle(); | ||||
|     #endif | ||||
|     handleNightlight(); | ||||
|     yield(); | ||||
|  | ||||
|     handleHue(); | ||||
|     handleBlynk(); | ||||
|  | ||||
|     yield(); | ||||
|     if (!offMode) strip.service(); | ||||
|   } | ||||
|   yield(); | ||||
|   #ifdef ESP8266 | ||||
|   MDNS.update(); | ||||
|   #endif | ||||
|   if (millis() - lastMqttReconnectAttempt > 30000) initMqtt(); | ||||
|  | ||||
|   //DEBUG serial logging | ||||
|   #ifdef WLED_DEBUG | ||||
|    if (millis() - debugTime > 9999) | ||||
|    { | ||||
|      DEBUG_PRINTLN("---DEBUG INFO---"); | ||||
|      DEBUG_PRINT("Runtime: "); DEBUG_PRINTLN(millis()); | ||||
|      DEBUG_PRINT("Unix time: "); DEBUG_PRINTLN(now()); | ||||
|      DEBUG_PRINT("Free heap: "); DEBUG_PRINTLN(ESP.getFreeHeap()); | ||||
|      DEBUG_PRINT("Wifi state: "); DEBUG_PRINTLN(WiFi.status()); | ||||
|      if (WiFi.status() != lastWifiState) | ||||
|      { | ||||
|        wifiStateChangedTime = millis(); | ||||
|      } | ||||
|      lastWifiState = WiFi.status(); | ||||
|      DEBUG_PRINT("State time: "); DEBUG_PRINTLN(wifiStateChangedTime); | ||||
|      DEBUG_PRINT("NTP last sync: "); DEBUG_PRINTLN(ntpLastSyncTime); | ||||
|      DEBUG_PRINT("Client IP: "); DEBUG_PRINTLN(WiFi.localIP()); | ||||
|      DEBUG_PRINT("Loops/sec: "); DEBUG_PRINTLN(loops/10); | ||||
|      loops = 0; | ||||
|      debugTime = millis(); | ||||
|    } | ||||
|    loops++; | ||||
|   #endif | ||||
|   WLED::instance().loop(); | ||||
| } | ||||
|   | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -13,29 +13,15 @@ | ||||
|       <UniqueIdentifier>{67DA6AB6-F800-4c08-8B7A-83BB121AAD01}</UniqueIdentifier> | ||||
|       <Extensions>rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms</Extensions> | ||||
|     </Filter> | ||||
|     <Filter Include="Source Files\Dependencies"> | ||||
|       <UniqueIdentifier>{72fe60da-ba26-45b4-82c1-bdff809975da}</UniqueIdentifier> | ||||
|     </Filter> | ||||
|     <Filter Include="Header Files\Dependencies"> | ||||
|       <UniqueIdentifier>{8880888d-efea-4189-a25a-834b7b3bb756}</UniqueIdentifier> | ||||
|     </Filter> | ||||
|   </ItemGroup> | ||||
|   <ItemGroup> | ||||
|     <None Include="wled00.ino" /> | ||||
|     <None Include="wled01_eeprom.ino" /> | ||||
|     <None Include="wled02_xml.ino" /> | ||||
|     <None Include="wled03_set.ino" /> | ||||
|     <None Include="wled04_file.ino" /> | ||||
|     <None Include="wled05_init.ino" /> | ||||
|     <None Include="wled06_usermod.ino" /> | ||||
|     <None Include="wled07_notify.ino" /> | ||||
|     <None Include="wled08_led.ino" /> | ||||
|     <None Include="wled09_button.ino" /> | ||||
|     <None Include="wled10_ntp.ino" /> | ||||
|     <None Include="wled11_ol.ino" /> | ||||
|     <None Include="wled12_alexa.ino" /> | ||||
|     <None Include="wled13_cronixie.ino" /> | ||||
|     <None Include="wled14_colors.ino" /> | ||||
|     <None Include="wled15_hue.ino" /> | ||||
|     <None Include="wled16_blynk.ino" /> | ||||
|     <None Include="wled17_mqtt.ino" /> | ||||
|     <None Include="wled18_server.ino" /> | ||||
|     <None Include="wled19_json.ino" /> | ||||
|     <None Include="wled20_ir.ino" /> | ||||
|   </ItemGroup> | ||||
|   <ItemGroup> | ||||
|     <ClInclude Include="__vm\.wled00.vsarduino.h"> | ||||
| @@ -53,219 +39,348 @@ | ||||
|     <ClInclude Include="html_settings.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="ir_codes.h"> | ||||
|     <ClInclude Include="FX.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\json\ArduinoJson-v6.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\json\AsyncJson-v6.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkApi.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkApiArduino.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkArduinoClient.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkConfig.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkDateTime.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkDebug.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkDetectDevice.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkEveryN.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkFifo.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkHandlers.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkParam.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkProtocol.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkProtocolDefs.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\BlynkSimpleEsp.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkTemplates.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkTimer.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkUtility.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkWidgetBase.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkWiFiCommon.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Callbacks.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\ConnAckPacket.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\DisconnectReasons.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\espalexa\Espalexa.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\espalexa\EspalexaDevice.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\e131\ESPAsyncE131.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Flags.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Helpers.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\MessageProperties.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="NpbWrapper.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\Packet.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\ParsingInformation.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PingRespPacket.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubAckPacket.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubCompPacket.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PublishPacket.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubRecPacket.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubRelPacket.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Storage.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\SubAckPacket.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\time\Time.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\time\TimeLib.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\timezone\Timezone.h"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\UnsubAckPacket.hpp"> | ||||
|       <Filter>Header Files\Dependencies</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="alexa.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="blynk.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="button.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="colors.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="const.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="cronixie.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="dmx.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="file.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="html_ui.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="hue.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="ir.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="json.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="led.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="mqtt.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="notify.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="ntp.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="overlay.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="set.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="usermod.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="wled.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="wled_eeprom.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="wled_server.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="xml.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="ir_codes.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="palettes.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Callbacks.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\DisconnectReasons.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Flags.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Helpers.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\MessageProperties.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\ParsingInformation.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Storage.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\ConnAckPacket.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\Packet.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PingRespPacket.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubAckPacket.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubCompPacket.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PublishPacket.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubRecPacket.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubRelPacket.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\SubAckPacket.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\UnsubAckPacket.hpp"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\BlynkSimpleEsp.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkApi.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkApiArduino.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkArduinoClient.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkConfig.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkDateTime.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkDebug.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkDetectDevice.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkEveryN.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkFifo.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkHandlers.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkParam.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkProtocol.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkProtocolDefs.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkTemplates.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkTimer.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkUtility.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkWidgetBase.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\blynk\Blynk\BlynkWiFiCommon.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\espalexa\Espalexa.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\espalexa\EspalexaDevice.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\json\ArduinoJson-v6.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\json\AsyncJson-v6.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\time\Time.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\time\TimeLib.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\timezone\Timezone.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="FX.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|     <ClInclude Include="src\dependencies\e131\ESPAsyncE131.h"> | ||||
|       <Filter>Header Files</Filter> | ||||
|     </ClInclude> | ||||
|   </ItemGroup> | ||||
|   <ItemGroup> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\ConnAckPacket.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PingRespPacket.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubAckPacket.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubCompPacket.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PublishPacket.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubRecPacket.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubRelPacket.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\SubAckPacket.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\UnsubAckPacket.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\blynk\Blynk\BlynkDebug.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\blynk\Blynk\BlynkHandlers.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\blynk\Blynk\BlynkTimer.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\blynk\Blynk\utility.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\espalexa\EspalexaDevice.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\time\DateStrings.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\time\Time.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\timezone\Timezone.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="FX.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="FX_fcn.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\blynk\Blynk\utility.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\blynk\Blynk\BlynkDebug.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\blynk\Blynk\BlynkHandlers.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\blynk\Blynk\BlynkTimer.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\ConnAckPacket.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\time\DateStrings.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\espalexa\EspalexaDevice.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\e131\ESPAsyncE131.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PingRespPacket.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubAckPacket.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubCompPacket.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PublishPacket.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubRecPacket.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\PubRelPacket.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\SubAckPacket.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\time\Time.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\timezone\Timezone.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="src\dependencies\async-mqtt-client\AsyncMqttClient\Packets\UnsubAckPacket.cpp"> | ||||
|       <Filter>Source Files\Dependencies</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="alexa.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="blynk.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="button.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="colors.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="cronixie.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="file.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="hue.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="ir.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="json.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="led.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="mqtt.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="notify.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="ntp.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="overlay.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="set.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="usermod.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="wled.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="wled_eeprom.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="wled_server.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|     <ClCompile Include="xml.cpp"> | ||||
|       <Filter>Source Files</Filter> | ||||
|     </ClCompile> | ||||
|   </ItemGroup> | ||||
|   | ||||
| @@ -1,674 +0,0 @@ | ||||
| /* | ||||
|  * Methods to handle saving and loading to non-volatile memory | ||||
|  * EEPROM Map: https://github.com/Aircoookie/WLED/wiki/EEPROM-Map | ||||
|  */ | ||||
|  | ||||
| #define EEPSIZE 2560 | ||||
|  | ||||
| //eeprom Version code, enables default settings instead of 0 init on update | ||||
| #define EEPVER 15 | ||||
| //0 -> old version, default | ||||
| //1 -> 0.4p 1711272 and up | ||||
| //2 -> 0.4p 1711302 and up | ||||
| //3 -> 0.4  1712121 and up | ||||
| //4 -> 0.5.0 and up | ||||
| //5 -> 0.5.1 and up | ||||
| //6 -> 0.6.0 and up | ||||
| //7 -> 0.7.1 and up | ||||
| //8 -> 0.8.0-a and up | ||||
| //9 -> 0.8.0 | ||||
| //10-> 0.8.2 | ||||
| //11-> 0.8.5-dev #mqttauth @TimothyBrown | ||||
| //12-> 0.8.7-dev | ||||
| //13-> 0.9.0-dev | ||||
| //14-> 0.9.0-b1 | ||||
| //15-> 0.9.0-b3 | ||||
|  | ||||
| void commit() | ||||
| { | ||||
|   if (!EEPROM.commit()) errorFlag = 2; | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Erase all configuration data | ||||
|  */ | ||||
| void clearEEPROM() | ||||
| { | ||||
|   for (int i = 0; i < EEPSIZE; i++) | ||||
|   { | ||||
|     EEPROM.write(i, 0); | ||||
|   } | ||||
|   commit(); | ||||
| } | ||||
|  | ||||
|  | ||||
| void writeStringToEEPROM(uint16_t pos, char* str, uint16_t len) | ||||
| { | ||||
|   for (int i = 0; i < len; ++i) | ||||
|   { | ||||
|     EEPROM.write(pos + i, str[i]); | ||||
|     if (str[i] == 0) return; | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| void readStringFromEEPROM(uint16_t pos, char* str, uint16_t len) | ||||
| { | ||||
|   for (int i = 0; i < len; ++i) | ||||
|   { | ||||
|     str[i] = EEPROM.read(pos + i); | ||||
|     if (str[i] == 0) return; | ||||
|   } | ||||
|   str[len] = 0; //make sure every string is properly terminated. str must be at least len +1 big. | ||||
| } | ||||
|  | ||||
|  | ||||
| /* | ||||
|  * Write configuration to flash | ||||
|  */ | ||||
| void saveSettingsToEEPROM() | ||||
| { | ||||
|   if (EEPROM.read(233) != 233) //set no first boot flag | ||||
|   { | ||||
|     clearEEPROM(); | ||||
|     EEPROM.write(233, 233); | ||||
|   } | ||||
|  | ||||
|   writeStringToEEPROM(  0, clientSSID, 32); | ||||
|   writeStringToEEPROM( 32, clientPass, 64); | ||||
|   writeStringToEEPROM( 96,      cmDNS, 32); | ||||
|   writeStringToEEPROM(128,     apSSID, 32); | ||||
|   writeStringToEEPROM(160,     apPass, 64); | ||||
|  | ||||
|   EEPROM.write(224, nightlightDelayMinsDefault); | ||||
|   EEPROM.write(225, nightlightFade); | ||||
|   EEPROM.write(226, notifyDirectDefault); | ||||
|   EEPROM.write(227, apChannel); | ||||
|   EEPROM.write(228, apHide); | ||||
|   EEPROM.write(229, ledCount & 0xFF); | ||||
|   EEPROM.write(230, notifyButton); | ||||
|   EEPROM.write(231, notifyTwice); | ||||
|   EEPROM.write(232, buttonEnabled); | ||||
|   //233 reserved for first boot flag | ||||
|  | ||||
|   for (int i = 0; i<4; i++) //ip addresses | ||||
|   { | ||||
|     EEPROM.write(234+i, staticIP[i]); | ||||
|     EEPROM.write(238+i, staticGateway[i]); | ||||
|     EEPROM.write(242+i, staticSubnet[i]); | ||||
|   } | ||||
|  | ||||
|   EEPROM.write(249, briS); | ||||
|  | ||||
|   EEPROM.write(250, receiveNotificationBrightness); | ||||
|   EEPROM.write(251, fadeTransition); | ||||
|   EEPROM.write(252, strip.reverseMode); | ||||
|   EEPROM.write(253, transitionDelayDefault & 0xFF); | ||||
|   EEPROM.write(254, (transitionDelayDefault >> 8) & 0xFF); | ||||
|   EEPROM.write(255, briMultiplier); | ||||
|  | ||||
|   //255,250,231,230,226 notifier bytes | ||||
|   writeStringToEEPROM(256, otaPass, 32); | ||||
|  | ||||
|   EEPROM.write(288, nightlightTargetBri); | ||||
|   EEPROM.write(289, otaLock); | ||||
|   EEPROM.write(290, udpPort & 0xFF); | ||||
|   EEPROM.write(291, (udpPort >> 8) & 0xFF); | ||||
|   writeStringToEEPROM(292, serverDescription, 32); | ||||
|  | ||||
|   EEPROM.write(327, ntpEnabled); | ||||
|   EEPROM.write(328, currentTimezone); | ||||
|   EEPROM.write(329, useAMPM); | ||||
|   EEPROM.write(330, strip.gammaCorrectBri); | ||||
|   EEPROM.write(331, strip.gammaCorrectCol); | ||||
|   EEPROM.write(332, overlayDefault); | ||||
|  | ||||
|   EEPROM.write(333, alexaEnabled); | ||||
|   writeStringToEEPROM(334, alexaInvocationName, 32); | ||||
|   EEPROM.write(366, notifyAlexa); | ||||
|  | ||||
|   EEPROM.write(367, (arlsOffset>=0)); | ||||
|   EEPROM.write(368, abs(arlsOffset)); | ||||
|   EEPROM.write(369, turnOnAtBoot); | ||||
|  | ||||
|   EEPROM.write(372, useRGBW); | ||||
|   EEPROM.write(374, strip.paletteFade); | ||||
|   EEPROM.write(375, strip.milliampsPerLed); //was apWaitTimeSecs up to 0.8.5 | ||||
|   EEPROM.write(376, apBehavior); | ||||
|  | ||||
|   EEPROM.write(377, EEPVER); //eeprom was updated to latest | ||||
|  | ||||
|   EEPROM.write(382, strip.paletteBlend); | ||||
|   EEPROM.write(383, strip.colorOrder); | ||||
|  | ||||
|   EEPROM.write(385, irEnabled); | ||||
|  | ||||
|   EEPROM.write(387, strip.ablMilliampsMax & 0xFF); | ||||
|   EEPROM.write(388, (strip.ablMilliampsMax >> 8) & 0xFF); | ||||
|   EEPROM.write(389, bootPreset); | ||||
|   EEPROM.write(390, aOtaEnabled); | ||||
|   EEPROM.write(391, receiveNotificationColor); | ||||
|   EEPROM.write(392, receiveNotificationEffects); | ||||
|   EEPROM.write(393, wifiLock); | ||||
|  | ||||
|   EEPROM.write(394, abs(utcOffsetSecs) & 0xFF); | ||||
|   EEPROM.write(395, (abs(utcOffsetSecs) >> 8) & 0xFF); | ||||
|   EEPROM.write(396, (utcOffsetSecs<0)); //is negative | ||||
|   EEPROM.write(397, syncToggleReceive); | ||||
|   EEPROM.write(398, (ledCount >> 8) & 0xFF); | ||||
|   EEPROM.write(399, !enableSecTransition); | ||||
|  | ||||
|   //favorite setting (preset) memory (25 slots/ each 20byte) | ||||
|   //400 - 940 reserved | ||||
|   writeStringToEEPROM(990, ntpServerName, 32); | ||||
|  | ||||
|   EEPROM.write(2048, huePollingEnabled); | ||||
|   //EEPROM.write(2049, hueUpdatingEnabled); | ||||
|   for (int i = 2050; i < 2054; ++i) | ||||
|   { | ||||
|     EEPROM.write(i, hueIP[i-2050]); | ||||
|   } | ||||
|   writeStringToEEPROM(2054, hueApiKey, 46); | ||||
|   EEPROM.write(2100, huePollIntervalMs & 0xFF); | ||||
|   EEPROM.write(2101, (huePollIntervalMs >> 8) & 0xFF); | ||||
|   EEPROM.write(2102, notifyHue); | ||||
|   EEPROM.write(2103, hueApplyOnOff); | ||||
|   EEPROM.write(2104, hueApplyBri); | ||||
|   EEPROM.write(2105, hueApplyColor); | ||||
|   EEPROM.write(2106, huePollLightId); | ||||
|  | ||||
|   EEPROM.write(2150, overlayMin); | ||||
|   EEPROM.write(2151, overlayMax); | ||||
|   EEPROM.write(2152, analogClock12pixel); | ||||
|   EEPROM.write(2153, analogClock5MinuteMarks); | ||||
|   EEPROM.write(2154, analogClockSecondsTrail); | ||||
|  | ||||
|   EEPROM.write(2155, countdownMode); | ||||
|   EEPROM.write(2156, countdownYear); | ||||
|   EEPROM.write(2157, countdownMonth); | ||||
|   EEPROM.write(2158, countdownDay); | ||||
|   EEPROM.write(2159, countdownHour); | ||||
|   EEPROM.write(2160, countdownMin); | ||||
|   EEPROM.write(2161, countdownSec); | ||||
|   setCountdown(); | ||||
|  | ||||
|   writeStringToEEPROM(2165, cronixieDisplay, 6); | ||||
|   EEPROM.write(2171, cronixieBacklight); | ||||
|   setCronixie(); | ||||
|  | ||||
|   EEPROM.write(2175, macroBoot); | ||||
|   EEPROM.write(2176, macroAlexaOn); | ||||
|   EEPROM.write(2177, macroAlexaOff); | ||||
|   EEPROM.write(2178, macroButton); | ||||
|   EEPROM.write(2179, macroLongPress); | ||||
|   EEPROM.write(2180, macroCountdown); | ||||
|   EEPROM.write(2181, macroNl); | ||||
|   EEPROM.write(2182, macroDoublePress); | ||||
|  | ||||
|   EEPROM.write(2190, e131Universe & 0xFF); | ||||
|   EEPROM.write(2191, (e131Universe >> 8) & 0xFF); | ||||
|   EEPROM.write(2192, e131Multicast); | ||||
|   EEPROM.write(2193, realtimeTimeoutMs & 0xFF); | ||||
|   EEPROM.write(2194, (realtimeTimeoutMs >> 8) & 0xFF); | ||||
|   EEPROM.write(2195, arlsForceMaxBri); | ||||
|   EEPROM.write(2196, arlsDisableGammaCorrection); | ||||
|   EEPROM.write(2197, DMXAddress & 0xFF); | ||||
|   EEPROM.write(2198, (DMXAddress >> 8) & 0xFF); | ||||
|   EEPROM.write(2199, DMXMode); | ||||
|  | ||||
|   EEPROM.write(2200, !receiveDirect); | ||||
|   EEPROM.write(2201, notifyMacro); //was enableRealtime | ||||
|   EEPROM.write(2203, autoRGBtoRGBW); | ||||
|   EEPROM.write(2204, skipFirstLed); | ||||
|  | ||||
|   if (saveCurrPresetCycConf) | ||||
|   { | ||||
|     EEPROM.write(2205, presetCyclingEnabled); | ||||
|     EEPROM.write(2206, presetCycleTime & 0xFF); | ||||
|     EEPROM.write(2207, (presetCycleTime >> 8) & 0xFF); | ||||
|     EEPROM.write(2208, presetCycleMin); | ||||
|     EEPROM.write(2209, presetCycleMax); | ||||
|     EEPROM.write(2210, presetApplyBri); | ||||
|     EEPROM.write(2211, presetApplyCol); | ||||
|     EEPROM.write(2212, presetApplyFx); | ||||
|     saveCurrPresetCycConf = false; | ||||
|   } | ||||
|  | ||||
|   writeStringToEEPROM(2220, blynkApiKey, 35); | ||||
|  | ||||
|   for (int i = 0; i < 8; ++i) | ||||
|   { | ||||
|     EEPROM.write(2260 + i, timerHours[i]  ); | ||||
|     EEPROM.write(2270 + i, timerMinutes[i]); | ||||
|     EEPROM.write(2280 + i, timerWeekday[i]); | ||||
|     EEPROM.write(2290 + i, timerMacro[i]  ); | ||||
|   } | ||||
|  | ||||
|   EEPROM.write(2299, mqttEnabled); | ||||
|   writeStringToEEPROM(2300, mqttServer, 32); | ||||
|   writeStringToEEPROM(2333, mqttDeviceTopic, 32); | ||||
|   writeStringToEEPROM(2366, mqttGroupTopic, 32); | ||||
|   writeStringToEEPROM(2399, mqttUser, 40); | ||||
|   writeStringToEEPROM(2440, mqttPass, 40); | ||||
|   writeStringToEEPROM(2481, mqttClientID, 40); | ||||
|   EEPROM.write(2522, mqttPort & 0xFF); | ||||
|   EEPROM.write(2523, (mqttPort >> 8) & 0xFF); | ||||
|  | ||||
|   commit(); | ||||
| } | ||||
|  | ||||
|  | ||||
| /* | ||||
|  * Read all configuration from flash | ||||
|  */ | ||||
| void loadSettingsFromEEPROM(bool first) | ||||
| { | ||||
|   if (EEPROM.read(233) != 233) //first boot/reset to default | ||||
|   { | ||||
|     DEBUG_PRINT("Settings invalid, restoring defaults..."); | ||||
|     saveSettingsToEEPROM(); | ||||
|     DEBUG_PRINTLN("done"); | ||||
|     return; | ||||
|   } | ||||
|   int lastEEPROMversion = EEPROM.read(377); //last EEPROM version before update | ||||
|  | ||||
|  | ||||
|   readStringFromEEPROM(  0, clientSSID, 32); | ||||
|   readStringFromEEPROM( 32, clientPass, 64); | ||||
|   readStringFromEEPROM( 96,      cmDNS, 32); | ||||
|   readStringFromEEPROM(128,     apSSID, 32); | ||||
|   readStringFromEEPROM(160,     apPass, 64); | ||||
|  | ||||
|   nightlightDelayMinsDefault = EEPROM.read(224); | ||||
|   nightlightDelayMins = nightlightDelayMinsDefault; | ||||
|   nightlightFade = EEPROM.read(225); | ||||
|   notifyDirectDefault = EEPROM.read(226); | ||||
|   notifyDirect = notifyDirectDefault; | ||||
|  | ||||
|   apChannel = EEPROM.read(227); | ||||
|   if (apChannel > 13 || apChannel < 1) apChannel = 1; | ||||
|   apHide = EEPROM.read(228); | ||||
|   if (apHide > 1) apHide = 1; | ||||
|   ledCount = EEPROM.read(229) + ((EEPROM.read(398) << 8) & 0xFF00); if (ledCount > MAX_LEDS || ledCount == 0) ledCount = 30; | ||||
|  | ||||
|   notifyButton = EEPROM.read(230); | ||||
|   notifyTwice = EEPROM.read(231); | ||||
|   buttonEnabled = EEPROM.read(232); | ||||
|  | ||||
|   staticIP[0] = EEPROM.read(234); | ||||
|   staticIP[1] = EEPROM.read(235); | ||||
|   staticIP[2] = EEPROM.read(236); | ||||
|   staticIP[3] = EEPROM.read(237); | ||||
|   staticGateway[0] = EEPROM.read(238); | ||||
|   staticGateway[1] = EEPROM.read(239); | ||||
|   staticGateway[2] = EEPROM.read(240); | ||||
|   staticGateway[3] = EEPROM.read(241); | ||||
|   staticSubnet[0] = EEPROM.read(242); | ||||
|   staticSubnet[1] = EEPROM.read(243); | ||||
|   staticSubnet[2] = EEPROM.read(244); | ||||
|   staticSubnet[3] = EEPROM.read(245); | ||||
|  | ||||
|   briS = EEPROM.read(249); bri = briS; | ||||
|   if (!EEPROM.read(369) && first) | ||||
|   { | ||||
|     bri = 0; briLast = briS; | ||||
|   } | ||||
|   receiveNotificationBrightness = EEPROM.read(250); | ||||
|   fadeTransition = EEPROM.read(251); | ||||
|   strip.reverseMode = EEPROM.read(252); | ||||
|   transitionDelayDefault = EEPROM.read(253) + ((EEPROM.read(254) << 8) & 0xFF00); | ||||
|   transitionDelay = transitionDelayDefault; | ||||
|   briMultiplier = EEPROM.read(255); | ||||
|  | ||||
|   readStringFromEEPROM(256, otaPass, 32); | ||||
|  | ||||
|   nightlightTargetBri = EEPROM.read(288); | ||||
|   otaLock = EEPROM.read(289); | ||||
|   udpPort = EEPROM.read(290) + ((EEPROM.read(291) << 8) & 0xFF00); | ||||
|  | ||||
|   readStringFromEEPROM(292, serverDescription, 32); | ||||
|  | ||||
|   ntpEnabled = EEPROM.read(327); | ||||
|   currentTimezone = EEPROM.read(328); | ||||
|   useAMPM = EEPROM.read(329); | ||||
|   strip.gammaCorrectBri = EEPROM.read(330); | ||||
|   strip.gammaCorrectCol = EEPROM.read(331); | ||||
|   overlayDefault = EEPROM.read(332); | ||||
|   if (lastEEPROMversion < 8 && overlayDefault > 0) overlayDefault--; //overlay mode 1 (solid) was removed | ||||
|  | ||||
|   alexaEnabled = EEPROM.read(333); | ||||
|  | ||||
|   readStringFromEEPROM(334, alexaInvocationName, 32); | ||||
|  | ||||
|   notifyAlexa = EEPROM.read(366); | ||||
|   arlsOffset = EEPROM.read(368); | ||||
|   if (!EEPROM.read(367)) arlsOffset = -arlsOffset; | ||||
|   turnOnAtBoot = EEPROM.read(369); | ||||
|   useRGBW = EEPROM.read(372); | ||||
|   //374 - strip.paletteFade | ||||
|    | ||||
|   apBehavior = EEPROM.read(376); | ||||
|      | ||||
|   //377 = lastEEPROMversion | ||||
|   if (lastEEPROMversion > 3) { | ||||
|     aOtaEnabled = EEPROM.read(390); | ||||
|     receiveNotificationColor = EEPROM.read(391); | ||||
|     receiveNotificationEffects = EEPROM.read(392); | ||||
|   } | ||||
|   receiveNotifications = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects); | ||||
|    | ||||
|   if (lastEEPROMversion > 4) { | ||||
|     huePollingEnabled = EEPROM.read(2048); | ||||
|     //hueUpdatingEnabled = EEPROM.read(2049); | ||||
|     for (int i = 2050; i < 2054; ++i) | ||||
|     { | ||||
|       hueIP[i-2050] = EEPROM.read(i); | ||||
|     } | ||||
|  | ||||
|     readStringFromEEPROM(2054, hueApiKey, 46); | ||||
|  | ||||
|     huePollIntervalMs = EEPROM.read(2100) + ((EEPROM.read(2101) << 8) & 0xFF00); | ||||
|     notifyHue = EEPROM.read(2102); | ||||
|     hueApplyOnOff = EEPROM.read(2103); | ||||
|     hueApplyBri = EEPROM.read(2104); | ||||
|     hueApplyColor = EEPROM.read(2105); | ||||
|     huePollLightId = EEPROM.read(2106); | ||||
|   } | ||||
|   if (lastEEPROMversion > 5) { | ||||
|     overlayMin = EEPROM.read(2150); | ||||
|     overlayMax = EEPROM.read(2151); | ||||
|     analogClock12pixel = EEPROM.read(2152); | ||||
|     analogClock5MinuteMarks = EEPROM.read(2153); | ||||
|     analogClockSecondsTrail = EEPROM.read(2154); | ||||
|     countdownMode = EEPROM.read(2155); | ||||
|     countdownYear = EEPROM.read(2156); | ||||
|     countdownMonth = EEPROM.read(2157); | ||||
|     countdownDay = EEPROM.read(2158); | ||||
|     countdownHour = EEPROM.read(2159); | ||||
|     countdownMin = EEPROM.read(2160); | ||||
|     countdownSec = EEPROM.read(2161); | ||||
|     setCountdown(); | ||||
|  | ||||
|     readStringFromEEPROM(2165, cronixieDisplay, 6); | ||||
|     cronixieBacklight = EEPROM.read(2171); | ||||
|  | ||||
|     macroBoot = EEPROM.read(2175); | ||||
|     macroAlexaOn = EEPROM.read(2176); | ||||
|     macroAlexaOff = EEPROM.read(2177); | ||||
|     macroButton = EEPROM.read(2178); | ||||
|     macroLongPress = EEPROM.read(2179); | ||||
|     macroCountdown = EEPROM.read(2180); | ||||
|     macroNl = EEPROM.read(2181); | ||||
|     macroDoublePress = EEPROM.read(2182); | ||||
|     if (macroDoublePress > 16) macroDoublePress = 0; | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 6) | ||||
|   { | ||||
|     e131Universe = EEPROM.read(2190) + ((EEPROM.read(2191) << 8) & 0xFF00); | ||||
|     e131Multicast = EEPROM.read(2192); | ||||
|     realtimeTimeoutMs = EEPROM.read(2193) + ((EEPROM.read(2194) << 8) & 0xFF00); | ||||
|     arlsForceMaxBri = EEPROM.read(2195); | ||||
|     arlsDisableGammaCorrection = EEPROM.read(2196); | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 7) | ||||
|   { | ||||
|     strip.paletteFade  = EEPROM.read(374); | ||||
|     strip.paletteBlend = EEPROM.read(382); | ||||
|  | ||||
|     for (int i = 0; i < 8; ++i) | ||||
|     { | ||||
|       timerHours[i]   = EEPROM.read(2260 + i); | ||||
|       timerMinutes[i] = EEPROM.read(2270 + i); | ||||
|       timerWeekday[i] = EEPROM.read(2280 + i); | ||||
|       timerMacro[i]   = EEPROM.read(2290 + i); | ||||
|       if (timerWeekday[i] == 0) timerWeekday[i] = 255; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 8) | ||||
|   { | ||||
|     readStringFromEEPROM(2300, mqttServer, 32); | ||||
|     readStringFromEEPROM(2333, mqttDeviceTopic, 32); | ||||
|     readStringFromEEPROM(2366, mqttGroupTopic, 32); | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 9) | ||||
|   { | ||||
|     strip.colorOrder = EEPROM.read(383); | ||||
|     irEnabled = EEPROM.read(385); | ||||
|     strip.ablMilliampsMax = EEPROM.read(387) + ((EEPROM.read(388) << 8) & 0xFF00); | ||||
|   } else if (lastEEPROMversion > 1) //ABL is off by default when updating from version older than 0.8.2 | ||||
|   { | ||||
|     strip.ablMilliampsMax = 65000; | ||||
|   } else { | ||||
|     strip.ablMilliampsMax = ABL_MILLIAMPS_DEFAULT; | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 10) | ||||
|   { | ||||
|     readStringFromEEPROM(2399, mqttUser, 40); | ||||
|     readStringFromEEPROM(2440, mqttPass, 40); | ||||
|     readStringFromEEPROM(2481, mqttClientID, 40); | ||||
|     mqttPort = EEPROM.read(2522) + ((EEPROM.read(2523) << 8) & 0xFF00); | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 11) | ||||
|   { | ||||
|     strip.milliampsPerLed = EEPROM.read(375); | ||||
|   } else if (strip.ablMilliampsMax == 65000) //65000 indicates disabled ABL in <0.8.7 | ||||
|   { | ||||
|     strip.ablMilliampsMax = ABL_MILLIAMPS_DEFAULT; | ||||
|     strip.milliampsPerLed = 0; //disable ABL | ||||
|   } | ||||
|   if (lastEEPROMversion > 12) | ||||
|   { | ||||
|     readStringFromEEPROM(990, ntpServerName, 32); | ||||
|   } | ||||
|   if (lastEEPROMversion > 13) | ||||
|   { | ||||
|     mqttEnabled = EEPROM.read(2299); | ||||
|     syncToggleReceive = EEPROM.read(397); | ||||
|   } else { | ||||
|     mqttEnabled = true; | ||||
|     syncToggleReceive = false; | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 14) | ||||
|   { | ||||
|     DMXAddress = EEPROM.read(2197) + ((EEPROM.read(2198) << 8) & 0xFF00); | ||||
|     DMXMode = EEPROM.read(2199); | ||||
|   } else { | ||||
|     DMXAddress = 1; | ||||
|     DMXMode = DMX_MODE_MULTIPLE_RGB; | ||||
|   } | ||||
|  | ||||
|  | ||||
|   receiveDirect = !EEPROM.read(2200); | ||||
|   notifyMacro = EEPROM.read(2201); | ||||
|  | ||||
|   autoRGBtoRGBW = EEPROM.read(2203); | ||||
|   skipFirstLed = EEPROM.read(2204); | ||||
|  | ||||
|   if (EEPROM.read(2210) || EEPROM.read(2211) || EEPROM.read(2212)) | ||||
|   { | ||||
|     presetCyclingEnabled = EEPROM.read(2205); | ||||
|     presetCycleTime = EEPROM.read(2206) + ((EEPROM.read(2207) << 8) & 0xFF00); | ||||
|     presetCycleMin = EEPROM.read(2208); | ||||
|     presetCycleMax = EEPROM.read(2209); | ||||
|     presetApplyBri = EEPROM.read(2210); | ||||
|     presetApplyCol = EEPROM.read(2211); | ||||
|     presetApplyFx = EEPROM.read(2212); | ||||
|   } | ||||
|  | ||||
|   bootPreset = EEPROM.read(389); | ||||
|   wifiLock = EEPROM.read(393); | ||||
|   utcOffsetSecs = EEPROM.read(394) + ((EEPROM.read(395) << 8) & 0xFF00); | ||||
|   if (EEPROM.read(396)) utcOffsetSecs = -utcOffsetSecs; //negative | ||||
|   enableSecTransition = !EEPROM.read(399); | ||||
|  | ||||
|   //favorite setting (preset) memory (25 slots/ each 20byte) | ||||
|   //400 - 899 reserved | ||||
|  | ||||
|   //custom macro memory (16 slots/ each 64byte) | ||||
|   //1024-2047 reserved | ||||
|  | ||||
|   readStringFromEEPROM(2220, blynkApiKey, 35); | ||||
|   if (strlen(blynkApiKey) < 25) blynkApiKey[0] = 0; | ||||
|  | ||||
|   //user MOD memory | ||||
|   //2944 - 3071 reserved | ||||
|  | ||||
|   overlayCurrent = overlayDefault; | ||||
|  | ||||
|   savedToPresets(); | ||||
| } | ||||
|  | ||||
|  | ||||
| //PRESET PROTOCOL 20 bytes | ||||
| //0: preset purpose byte 0:invalid 1:valid preset 2:segment preset 2.0 | ||||
| //1:a 2:r 3:g 4:b 5:w 6:er 7:eg 8:eb 9:ew 10:fx 11:sx | custom chase 12:numP 13:numS 14:(0:fs 1:both 2:fe) 15:step 16:ix 17: fp 18-19:Zeros | ||||
| //determines which presets already contain save data | ||||
| void savedToPresets() | ||||
| { | ||||
|   for (byte index = 1; index < 16; index++) | ||||
|   { | ||||
|     uint16_t i = 380 + index*20; | ||||
|  | ||||
|     if (EEPROM.read(i) == 1) { | ||||
|       savedPresets |= 0x01 << (index-1); | ||||
|     } else | ||||
|     { | ||||
|       savedPresets &= ~(0x01 << (index-1)); | ||||
|     } | ||||
|   } | ||||
|   if (EEPROM.read(700) == 2) { | ||||
|     savedPresets |= 0x01 << 15; | ||||
|   } else | ||||
|   { | ||||
|     savedPresets &= ~(0x01 << 15); | ||||
|   } | ||||
| } | ||||
|  | ||||
| bool applyPreset(byte index, bool loadBri = true, bool loadCol = true, bool loadFX = true) | ||||
| { | ||||
|   if (index == 255 || index == 0) | ||||
|   { | ||||
|     loadSettingsFromEEPROM(false);//load boot defaults | ||||
|     return true; | ||||
|   } | ||||
|   if (index > 16 || index < 1) return false; | ||||
|   uint16_t i = 380 + index*20; | ||||
|   if (index < 16) { | ||||
|     if (EEPROM.read(i) != 1) return false; | ||||
|     strip.applyToAllSelected = true; | ||||
|     if (loadBri) bri = EEPROM.read(i+1); | ||||
|     if (loadCol) | ||||
|     { | ||||
|       for (byte j=0; j<4; j++) | ||||
|       { | ||||
|         col[j] = EEPROM.read(i+j+2); | ||||
|         colSec[j] = EEPROM.read(i+j+6); | ||||
|       } | ||||
|       strip.setColor(2, EEPROM.read(i+12), EEPROM.read(i+13), EEPROM.read(i+14), EEPROM.read(i+15)); //tertiary color | ||||
|     } | ||||
|     if (loadFX) | ||||
|     { | ||||
|       effectCurrent = EEPROM.read(i+10); | ||||
|       effectSpeed = EEPROM.read(i+11); | ||||
|       effectIntensity = EEPROM.read(i+16); | ||||
|       effectPalette = EEPROM.read(i+17); | ||||
|     } | ||||
|   } else { | ||||
|     if (EEPROM.read(i) != 2) return false; | ||||
|     strip.applyToAllSelected = false; | ||||
|     if (loadBri) bri = EEPROM.read(i+1); | ||||
|     WS2812FX::Segment* seg = strip.getSegments(); | ||||
|     memcpy(seg, EEPROM.getDataPtr() +i+2, 240); | ||||
|     setValuesFromMainSeg(); | ||||
|   } | ||||
|   currentPreset = index; | ||||
|   isPreset = true; | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| void savePreset(byte index) | ||||
| { | ||||
|   if (index > 16) return; | ||||
|   if (index < 1) {saveSettingsToEEPROM();return;} | ||||
|   uint16_t i = 380 + index*20;//min400 | ||||
|    | ||||
|   if (index < 16) { | ||||
|     EEPROM.write(i, 1); | ||||
|     EEPROM.write(i+1, bri); | ||||
|     for (uint16_t j=0; j<4; j++) | ||||
|     { | ||||
|       EEPROM.write(i+j+2, col[j]); | ||||
|       EEPROM.write(i+j+6, colSec[j]); | ||||
|     } | ||||
|     EEPROM.write(i+10, effectCurrent); | ||||
|     EEPROM.write(i+11, effectSpeed); | ||||
|  | ||||
|     uint32_t colTer = strip.getSegment(strip.getMainSegmentId()).colors[2]; | ||||
|     EEPROM.write(i+12, (colTer >> 16) & 0xFF); | ||||
|     EEPROM.write(i+13, (colTer >>  8) & 0xFF); | ||||
|     EEPROM.write(i+14, (colTer >>  0) & 0xFF); | ||||
|     EEPROM.write(i+15, (colTer >> 24) & 0xFF); | ||||
|    | ||||
|     EEPROM.write(i+16, effectIntensity); | ||||
|     EEPROM.write(i+17, effectPalette); | ||||
|   } else { //segment 16 can save segments | ||||
|     EEPROM.write(i, 2); | ||||
|     EEPROM.write(i+1, bri); | ||||
|     WS2812FX::Segment* seg = strip.getSegments(); | ||||
|     memcpy(EEPROM.getDataPtr() +i+2, seg, 240); | ||||
|   } | ||||
|    | ||||
|   commit(); | ||||
|   savedToPresets(); | ||||
|   currentPreset = index; | ||||
|   isPreset = true; | ||||
| } | ||||
|  | ||||
|  | ||||
| void loadMacro(byte index, char* m) | ||||
| { | ||||
|   index-=1; | ||||
|   if (index > 15) return; | ||||
|   readStringFromEEPROM(1024+64*index, m, 64); | ||||
| } | ||||
|  | ||||
|  | ||||
| void applyMacro(byte index) | ||||
| { | ||||
|   index-=1; | ||||
|   if (index > 15) return; | ||||
|   String mc="win&"; | ||||
|   char m[65]; | ||||
|   loadMacro(index+1, m); | ||||
|   mc += m; | ||||
|   mc += "&IN"; //internal, no XML response | ||||
|   if (!notifyMacro) mc += "&NN"; | ||||
|   String forbidden = "&M="; //dont apply if called by the macro itself to prevent loop | ||||
|   /* | ||||
|    * NOTE: loop is still possible if you call a different macro from a macro, which then calls the first macro again. | ||||
|    * To prevent that, but also disable calling macros within macros, comment the next line out. | ||||
|    */ | ||||
|   forbidden = forbidden + index; | ||||
|   if (mc.indexOf(forbidden) >= 0) return; | ||||
|   handleSet(nullptr, mc); | ||||
| } | ||||
|  | ||||
|  | ||||
| void saveMacro(byte index, String mc, bool sing=true) //only commit on single save, not in settings | ||||
| { | ||||
|   index-=1; | ||||
|   if (index > 15) return; | ||||
|   int s = 1024+index*64; | ||||
|   for (int i = s; i < s+64; i++) | ||||
|   { | ||||
|     EEPROM.write(i, mc.charAt(i-s)); | ||||
|   } | ||||
|   if (sing) commit(); | ||||
| } | ||||
| @@ -1,393 +0,0 @@ | ||||
| /* | ||||
|  * Sending XML status files to client | ||||
|  */ | ||||
|  | ||||
| //build XML response to HTTP /win API request | ||||
| char* XML_response(AsyncWebServerRequest *request, char* dest = nullptr) | ||||
| { | ||||
|   char sbuf[(dest == nullptr)?1024:1]; //allocate local buffer if none passed | ||||
|   obuf = (dest == nullptr)? sbuf:dest; | ||||
|  | ||||
|   olen = 0; | ||||
|   oappend("<?xml version=\"1.0\" ?><vs><ac>"); | ||||
|   oappendi((nightlightActive && nightlightFade) ? briT : bri); | ||||
|   oappend("</ac>"); | ||||
|  | ||||
|   for (int i = 0; i < 3; i++) | ||||
|   { | ||||
|    oappend("<cl>"); | ||||
|    oappendi(col[i]); | ||||
|    oappend("</cl>"); | ||||
|   } | ||||
|   for (int i = 0; i < 3; i++) | ||||
|   { | ||||
|    oappend("<cs>"); | ||||
|    oappendi(colSec[i]); | ||||
|    oappend("</cs>"); | ||||
|   } | ||||
|   oappend("<ns>"); | ||||
|   oappendi(notifyDirect); | ||||
|   oappend("</ns><nr>"); | ||||
|   oappendi(receiveNotifications); | ||||
|   oappend("</nr><nl>"); | ||||
|   oappendi(nightlightActive); | ||||
|   oappend("</nl><nf>"); | ||||
|   oappendi(nightlightFade); | ||||
|   oappend("</nf><nd>"); | ||||
|   oappendi(nightlightDelayMins); | ||||
|   oappend("</nd><nt>"); | ||||
|   oappendi(nightlightTargetBri); | ||||
|   oappend("</nt><fx>"); | ||||
|   oappendi(effectCurrent); | ||||
|   oappend("</fx><sx>"); | ||||
|   oappendi(effectSpeed); | ||||
|   oappend("</sx><ix>"); | ||||
|   oappendi(effectIntensity); | ||||
|   oappend("</ix><fp>"); | ||||
|   oappendi(effectPalette); | ||||
|   oappend("</fp><wv>"); | ||||
|   if (useRGBW && !autoRGBtoRGBW) { | ||||
|    oappendi(col[3]); | ||||
|   } else { | ||||
|    oappend("-1"); | ||||
|   } | ||||
|   oappend("</wv><ws>"); | ||||
|   oappendi(colSec[3]); | ||||
|   oappend("</ws><ps>"); | ||||
|   oappendi((currentPreset < 1) ? 0:currentPreset); | ||||
|   oappend("</ps><cy>"); | ||||
|   oappendi(presetCyclingEnabled); | ||||
|   oappend("</cy><ds>"); | ||||
|   if (realtimeMode) | ||||
|   { | ||||
|     String mesg = "Live "; | ||||
|     if (realtimeMode == REALTIME_MODE_E131) | ||||
|     { | ||||
|       mesg += "E1.31 mode "; | ||||
|       mesg += DMXMode; | ||||
|       mesg += " at DMX Address "; | ||||
|       mesg += DMXAddress; | ||||
|       mesg += " from "; | ||||
|       mesg += realtimeIP[0]; | ||||
|       for (int i = 1; i < 4; i++) | ||||
|       { | ||||
|         mesg += "."; | ||||
|         mesg += realtimeIP[i]; | ||||
|       } | ||||
|       mesg += " seq="; | ||||
|       mesg += e131LastSequenceNumber; | ||||
|     } else if (realtimeMode == REALTIME_MODE_UDP || realtimeMode == REALTIME_MODE_HYPERION) { | ||||
|       mesg += "UDP from "; | ||||
|       mesg += realtimeIP[0]; | ||||
|       for (int i = 1; i < 4; i++) | ||||
|       { | ||||
|         mesg += "."; | ||||
|         mesg += realtimeIP[i]; | ||||
|       } | ||||
|     } else if (realtimeMode == REALTIME_MODE_ADALIGHT) { | ||||
|       mesg += "USB Adalight"; | ||||
|     } else { //generic | ||||
|       mesg += "data"; | ||||
|     } | ||||
|     oappend((char*)mesg.c_str()); | ||||
|   } else { | ||||
|     oappend(serverDescription); | ||||
|   } | ||||
|   oappend("</ds><ss>"); | ||||
|   oappendi(strip.getMainSegmentId()); | ||||
|   oappend("</ss></vs>"); | ||||
|   if (request != nullptr) request->send(200, "text/xml", obuf); | ||||
| } | ||||
|  | ||||
| //append a numeric setting to string buffer | ||||
| void sappend(char stype, const char* key, int val) | ||||
| { | ||||
|   char ds[] = "d.Sf."; | ||||
|  | ||||
|   switch(stype) | ||||
|   { | ||||
|     case 'c': //checkbox | ||||
|       oappend(ds); | ||||
|       oappend(key); | ||||
|       oappend(".checked="); | ||||
|       oappendi(val); | ||||
|       oappend(";"); | ||||
|       break; | ||||
|     case 'v': //numeric | ||||
|       oappend(ds); | ||||
|       oappend(key); | ||||
|       oappend(".value="); | ||||
|       oappendi(val); | ||||
|       oappend(";"); | ||||
|       break; | ||||
|     case 'i': //selectedIndex | ||||
|       oappend(ds); | ||||
|       oappend(key); | ||||
|       oappend(".selectedIndex="); | ||||
|       oappendi(val); | ||||
|       oappend(";"); | ||||
|       break; | ||||
|   } | ||||
| } | ||||
|  | ||||
| //append a string setting to buffer | ||||
| void sappends(char stype, const char* key, char* val) | ||||
| { | ||||
|   switch(stype) | ||||
|   { | ||||
|     case 's': //string (we can interpret val as char*) | ||||
|       oappend("d.Sf."); | ||||
|       oappend(key); | ||||
|       oappend(".value=\""); | ||||
|       oappend(val); | ||||
|       oappend("\";"); | ||||
|       break; | ||||
|     case 'm': //message | ||||
|       oappend("d.getElementsByClassName"); | ||||
|       oappend(key); | ||||
|       oappend(".innerHTML=\""); | ||||
|       oappend(val); | ||||
|       oappend("\";"); | ||||
|       break; | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| //get values for settings form in javascript | ||||
| void getSettingsJS(byte subPage, char* dest) | ||||
| { | ||||
|   //0: menu 1: wifi 2: leds 3: ui 4: sync 5: time 6: sec | ||||
|   DEBUG_PRINT("settings resp"); | ||||
|   DEBUG_PRINTLN(subPage); | ||||
|   obuf = dest; | ||||
|   olen = 0; | ||||
|  | ||||
|   if (subPage <1 || subPage >6) return; | ||||
|  | ||||
|   if (subPage == 1) { | ||||
|     sappends('s',"CS",clientSSID); | ||||
|  | ||||
|     byte l = strlen(clientPass); | ||||
|     char fpass[l+1]; //fill password field with *** | ||||
|     fpass[l] = 0; | ||||
|     memset(fpass,'*',l); | ||||
|     sappends('s',"CP",fpass); | ||||
|  | ||||
|     char k[3]; k[2] = 0; //IP addresses | ||||
|     for (int i = 0; i<4; i++) | ||||
|     { | ||||
|       k[1] = 48+i; //ascii 0,1,2,3 | ||||
|       k[0] = 'I'; sappend('v',k,staticIP[i]); | ||||
|       k[0] = 'G'; sappend('v',k,staticGateway[i]); | ||||
|       k[0] = 'S'; sappend('v',k,staticSubnet[i]); | ||||
|     } | ||||
|  | ||||
|     sappends('s',"CM",cmDNS); | ||||
|     sappend('i',"AB",apBehavior); | ||||
|     sappends('s',"AS",apSSID); | ||||
|     sappend('c',"AH",apHide); | ||||
|  | ||||
|     l = strlen(apPass); | ||||
|     char fapass[l+1]; //fill password field with *** | ||||
|     fapass[l] = 0; | ||||
|     memset(fapass,'*',l); | ||||
|     sappends('s',"AP",fapass); | ||||
|  | ||||
|     sappend('v',"AC",apChannel); | ||||
|  | ||||
|     if (WiFi.localIP()[0] != 0) //is connected | ||||
|     { | ||||
|       char s[16]; | ||||
|       IPAddress localIP = WiFi.localIP(); | ||||
|       sprintf(s, "%d.%d.%d.%d", localIP[0], localIP[1], localIP[2], localIP[3]); | ||||
|       sappends('m',"(\"sip\")[0]",s); | ||||
|     } else | ||||
|     { | ||||
|       sappends('m',"(\"sip\")[0]","Not connected"); | ||||
|     } | ||||
|  | ||||
|     if (WiFi.softAPIP()[0] != 0) //is active | ||||
|     { | ||||
|       char s[16]; | ||||
|       IPAddress apIP = WiFi.softAPIP(); | ||||
|       sprintf(s, "%d.%d.%d.%d", apIP[0], apIP[1], apIP[2], apIP[3]); | ||||
|       sappends('m',"(\"sip\")[1]",s); | ||||
|     } else | ||||
|     { | ||||
|       sappends('m',"(\"sip\")[1]","Not active"); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (subPage == 2) { | ||||
|     #ifdef ESP8266 | ||||
|     #if LEDPIN == 3 | ||||
|     oappend("d.Sf.LC.max=500;"); | ||||
|     #endif | ||||
|     #endif | ||||
|     sappend('v',"LC",ledCount); | ||||
|     sappend('v',"MA",strip.ablMilliampsMax); | ||||
|     sappend('v',"LA",strip.milliampsPerLed); | ||||
|     if (strip.currentMilliamps) | ||||
|     { | ||||
|       sappends('m',"(\"pow\")[0]",""); | ||||
|       olen -= 2; //delete "; | ||||
|       oappendi(strip.currentMilliamps); | ||||
|       oappend("mA\";"); | ||||
|     } | ||||
|  | ||||
|     sappend('v',"CA",briS); | ||||
|     sappend('c',"EW",useRGBW); | ||||
|     sappend('i',"CO",strip.colorOrder); | ||||
|     sappend('c',"AW",autoRGBtoRGBW); | ||||
|  | ||||
|     sappend('c',"BO",turnOnAtBoot); | ||||
|     sappend('v',"BP",bootPreset); | ||||
|  | ||||
|     sappend('c',"GB",strip.gammaCorrectBri); | ||||
|     sappend('c',"GC",strip.gammaCorrectCol); | ||||
|     sappend('c',"TF",fadeTransition); | ||||
|     sappend('v',"TD",transitionDelayDefault); | ||||
|     sappend('c',"PF",strip.paletteFade); | ||||
|     sappend('v',"BF",briMultiplier); | ||||
|     sappend('v',"TB",nightlightTargetBri); | ||||
|     sappend('v',"TL",nightlightDelayMinsDefault); | ||||
|     sappend('c',"TW",nightlightFade); | ||||
|     sappend('i',"PB",strip.paletteBlend); | ||||
|     sappend('c',"RV",strip.reverseMode); | ||||
|     sappend('c',"SL",skipFirstLed); | ||||
|   } | ||||
|  | ||||
|   if (subPage == 3) | ||||
|   { | ||||
|     sappends('s',"DS",serverDescription); | ||||
|     sappend('c',"ST",syncToggleReceive); | ||||
|   } | ||||
|  | ||||
|   if (subPage == 4) | ||||
|   { | ||||
|     sappend('c',"BT",buttonEnabled); | ||||
|     sappend('v',"IR",irEnabled); | ||||
|     sappend('v',"UP",udpPort); | ||||
|     sappend('c',"RB",receiveNotificationBrightness); | ||||
|     sappend('c',"RC",receiveNotificationColor); | ||||
|     sappend('c',"RX",receiveNotificationEffects); | ||||
|     sappend('c',"SD",notifyDirectDefault); | ||||
|     sappend('c',"SB",notifyButton); | ||||
|     sappend('c',"SH",notifyHue); | ||||
|     sappend('c',"SM",notifyMacro); | ||||
|     sappend('c',"S2",notifyTwice); | ||||
|     sappend('c',"RD",receiveDirect); | ||||
|     sappend('c',"EM",e131Multicast); | ||||
|     sappend('v',"EU",e131Universe); | ||||
|     sappend('v',"DA",DMXAddress); | ||||
|     sappend('v',"DM",DMXMode); | ||||
|     sappend('v',"ET",realtimeTimeoutMs); | ||||
|     sappend('c',"FB",arlsForceMaxBri); | ||||
|     sappend('c',"RG",arlsDisableGammaCorrection); | ||||
|     sappend('v',"WO",arlsOffset); | ||||
|     sappend('c',"AL",alexaEnabled); | ||||
|     sappends('s',"AI",alexaInvocationName); | ||||
|     sappend('c',"SA",notifyAlexa); | ||||
|     sappends('s',"BK",(char*)((blynkEnabled)?"Hidden":"")); | ||||
|  | ||||
|     #ifdef WLED_ENABLE_MQTT | ||||
|     sappend('c',"MQ",mqttEnabled); | ||||
|     sappends('s',"MS",mqttServer); | ||||
|     sappend('v',"MQPORT",mqttPort); | ||||
|     sappends('s',"MQUSER",mqttUser); | ||||
|     sappends('s',"MQPASS",mqttPass); | ||||
|     byte l = strlen(mqttPass); | ||||
|     char fpass[l+1]; //fill password field with *** | ||||
|     fpass[l] = 0; | ||||
|     memset(fpass,'*',l); | ||||
|     sappends('s',"MQPASS",fpass); | ||||
|     sappends('s',"MQCID",mqttClientID); | ||||
|     sappends('s',"MD",mqttDeviceTopic); | ||||
|     sappends('s',"MG",mqttGroupTopic); | ||||
|     #endif | ||||
|  | ||||
|     #ifdef WLED_DISABLE_HUESYNC | ||||
|     sappends('m',"(\"hms\")[0]","Unsupported in build"); | ||||
|     #else | ||||
|     sappend('v',"H0",hueIP[0]); | ||||
|     sappend('v',"H1",hueIP[1]); | ||||
|     sappend('v',"H2",hueIP[2]); | ||||
|     sappend('v',"H3",hueIP[3]); | ||||
|     sappend('v',"HL",huePollLightId); | ||||
|     sappend('v',"HI",huePollIntervalMs); | ||||
|     sappend('c',"HP",huePollingEnabled); | ||||
|     sappend('c',"HO",hueApplyOnOff); | ||||
|     sappend('c',"HB",hueApplyBri); | ||||
|     sappend('c',"HC",hueApplyColor); | ||||
|     sappends('m',"(\"hms\")[0]",hueError); | ||||
|     #endif | ||||
|   } | ||||
|  | ||||
|   if (subPage == 5) | ||||
|   { | ||||
|     sappend('c',"NT",ntpEnabled); | ||||
|     sappends('s',"NS",ntpServerName); | ||||
|     sappend('c',"CF",!useAMPM); | ||||
|     sappend('i',"TZ",currentTimezone); | ||||
|     sappend('v',"UO",utcOffsetSecs); | ||||
|     char tm[32]; | ||||
|     getTimeString(tm); | ||||
|     sappends('m',"(\"times\")[0]",tm); | ||||
|     sappend('i',"OL",overlayCurrent); | ||||
|     sappend('v',"O1",overlayMin); | ||||
|     sappend('v',"O2",overlayMax); | ||||
|     sappend('v',"OM",analogClock12pixel); | ||||
|     sappend('c',"OS",analogClockSecondsTrail); | ||||
|     sappend('c',"O5",analogClock5MinuteMarks); | ||||
|     sappends('s',"CX",cronixieDisplay); | ||||
|     sappend('c',"CB",cronixieBacklight); | ||||
|     sappend('c',"CE",countdownMode); | ||||
|     sappend('v',"CY",countdownYear); | ||||
|     sappend('v',"CI",countdownMonth); | ||||
|     sappend('v',"CD",countdownDay); | ||||
|     sappend('v',"CH",countdownHour); | ||||
|     sappend('v',"CM",countdownMin); | ||||
|     sappend('v',"CS",countdownSec); | ||||
|     char k[4]; k[0]= 'M'; | ||||
|     for (int i=1;i<17;i++) | ||||
|     { | ||||
|       char m[65]; | ||||
|       loadMacro(i, m); | ||||
|       sprintf(k+1,"%i",i); | ||||
|       sappends('s',k,m); | ||||
|     } | ||||
|  | ||||
|     sappend('v',"MB",macroBoot); | ||||
|     sappend('v',"A0",macroAlexaOn); | ||||
|     sappend('v',"A1",macroAlexaOff); | ||||
|     sappend('v',"MP",macroButton); | ||||
|     sappend('v',"ML",macroLongPress); | ||||
|     sappend('v',"MC",macroCountdown); | ||||
|     sappend('v',"MN",macroNl); | ||||
|     sappend('v',"MD",macroDoublePress); | ||||
|  | ||||
|     k[2] = 0; //Time macros | ||||
|     for (int i = 0; i<8; i++) | ||||
|     { | ||||
|       k[1] = 48+i; //ascii 0,1,2,3 | ||||
|       k[0] = 'H'; sappend('v',k,timerHours[i]); | ||||
|       k[0] = 'N'; sappend('v',k,timerMinutes[i]); | ||||
|       k[0] = 'T'; sappend('v',k,timerMacro[i]); | ||||
|       k[0] = 'W'; sappend('v',k,timerWeekday[i]); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (subPage == 6) | ||||
|   { | ||||
|     sappend('c',"NO",otaLock); | ||||
|     sappend('c',"OW",wifiLock); | ||||
|     sappend('c',"AO",aOtaEnabled); | ||||
|     sappends('m',"(\"msg\")[0]","WLED "); | ||||
|     olen -= 2; //delete "; | ||||
|     oappend(versionString); | ||||
|     oappend(" (build "); | ||||
|     oappendi(VERSION); | ||||
|     oappend(") OK\";"); | ||||
|   } | ||||
|   oappend("}</script>"); | ||||
| } | ||||
| @@ -1,655 +0,0 @@ | ||||
| /* | ||||
|  * Receives client input | ||||
|  */ | ||||
|  | ||||
| void _setRandomColor(bool _sec,bool fromButton=false) | ||||
| { | ||||
|   lastRandomIndex = strip.get_random_wheel_index(lastRandomIndex); | ||||
|   if (_sec){ | ||||
|     colorHStoRGB(lastRandomIndex*256,255,colSec); | ||||
|   } else { | ||||
|     colorHStoRGB(lastRandomIndex*256,255,col); | ||||
|   } | ||||
|   if (fromButton) colorUpdated(2); | ||||
| } | ||||
|  | ||||
|  | ||||
| bool isAsterisksOnly(const char* str, byte maxLen) | ||||
| { | ||||
|   for (byte i = 0; i < maxLen; i++) { | ||||
|     if (str[i] == 0) break; | ||||
|     if (str[i] != '*') return false; | ||||
|   } | ||||
|   return true; | ||||
| } | ||||
|  | ||||
|  | ||||
| //called upon POST settings form submit | ||||
| void handleSettingsSet(AsyncWebServerRequest *request, byte subPage) | ||||
| { | ||||
|   //0: menu 1: wifi 2: leds 3: ui 4: sync 5: time 6: sec | ||||
|   if (subPage <1 || subPage >6) return; | ||||
|  | ||||
|   //WIFI SETTINGS | ||||
|   if (subPage == 1) | ||||
|   { | ||||
|     strlcpy(clientSSID,request->arg("CS").c_str(), 33); | ||||
|  | ||||
|     if (!isAsterisksOnly(request->arg("CP").c_str(), 65)) strlcpy(clientPass, request->arg("CP").c_str(), 65); | ||||
|  | ||||
|     strlcpy(cmDNS, request->arg("CM").c_str(), 33); | ||||
|  | ||||
|     apBehavior = request->arg("AB").toInt(); | ||||
|     strlcpy(apSSID, request->arg("AS").c_str(), 33); | ||||
|     apHide = request->hasArg("AH"); | ||||
|     int passlen = request->arg("AP").length(); | ||||
|     if (passlen == 0 || (passlen > 7 && !isAsterisksOnly(request->arg("AP").c_str(), 65))) strlcpy(apPass, request->arg("AP").c_str(), 65); | ||||
|     int t = request->arg("AC").toInt(); if (t > 0 && t < 14) apChannel = t; | ||||
|  | ||||
|     char k[3]; k[2] = 0; | ||||
|     for (int i = 0; i<4; i++) | ||||
|     { | ||||
|       k[1] = i+48;//ascii 0,1,2,3 | ||||
|  | ||||
|       k[0] = 'I'; //static IP | ||||
|       staticIP[i] = request->arg(k).toInt(); | ||||
|  | ||||
|       k[0] = 'G'; //gateway | ||||
|       staticGateway[i] = request->arg(k).toInt(); | ||||
|  | ||||
|       k[0] = 'S'; //subnet | ||||
|       staticSubnet[i] = request->arg(k).toInt(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //LED SETTINGS | ||||
|   if (subPage == 2) | ||||
|   { | ||||
|     int t = request->arg("LC").toInt(); | ||||
|     if (t > 0 && t <= MAX_LEDS) ledCount = t; | ||||
|     #ifdef ESP8266 | ||||
|     #if LEDPIN == 3 | ||||
|     if (ledCount > MAX_LEDS_DMA) ledCount = MAX_LEDS_DMA; //DMA method uses too much ram | ||||
|     #endif | ||||
|     #endif | ||||
|     strip.ablMilliampsMax = request->arg("MA").toInt(); | ||||
|     strip.milliampsPerLed = request->arg("LA").toInt(); | ||||
|      | ||||
|     useRGBW = request->hasArg("EW"); | ||||
|     strip.colorOrder = request->arg("CO").toInt(); | ||||
|     autoRGBtoRGBW = request->hasArg("AW"); | ||||
|  | ||||
|     briS = request->arg("CA").toInt(); | ||||
|  | ||||
|     saveCurrPresetCycConf = request->hasArg("PC"); | ||||
|     turnOnAtBoot = request->hasArg("BO"); | ||||
|     t = request->arg("BP").toInt(); | ||||
|     if (t <= 25) bootPreset = t; | ||||
|     strip.gammaCorrectBri = request->hasArg("GB"); | ||||
|     strip.gammaCorrectCol = request->hasArg("GC"); | ||||
|  | ||||
|     fadeTransition = request->hasArg("TF"); | ||||
|     t = request->arg("TD").toInt(); | ||||
|     if (t > 0) transitionDelay = t; | ||||
|     transitionDelayDefault = t; | ||||
|     strip.paletteFade = request->hasArg("PF"); | ||||
|  | ||||
|     nightlightTargetBri = request->arg("TB").toInt(); | ||||
|     t = request->arg("TL").toInt(); | ||||
|     if (t > 0) nightlightDelayMinsDefault = t; | ||||
|     nightlightDelayMins = nightlightDelayMinsDefault; | ||||
|     nightlightFade = request->hasArg("TW"); | ||||
|  | ||||
|     t = request->arg("PB").toInt(); | ||||
|     if (t >= 0 && t < 4) strip.paletteBlend = t; | ||||
|     strip.reverseMode = request->hasArg("RV"); | ||||
|     skipFirstLed = request->hasArg("SL"); | ||||
|     t = request->arg("BF").toInt(); | ||||
|     if (t > 0) briMultiplier = t; | ||||
|   } | ||||
|  | ||||
|   //UI | ||||
|   if (subPage == 3) | ||||
|   { | ||||
|     strlcpy(serverDescription, request->arg("DS").c_str(), 33); | ||||
|     syncToggleReceive = request->hasArg("ST"); | ||||
|   } | ||||
|  | ||||
|   //SYNC | ||||
|   if (subPage == 4) | ||||
|   { | ||||
|     buttonEnabled = request->hasArg("BT"); | ||||
|     irEnabled = request->arg("IR").toInt(); | ||||
|     int t = request->arg("UP").toInt(); | ||||
|     if (t > 0) udpPort = t; | ||||
|     receiveNotificationBrightness = request->hasArg("RB"); | ||||
|     receiveNotificationColor = request->hasArg("RC"); | ||||
|     receiveNotificationEffects = request->hasArg("RX"); | ||||
|     receiveNotifications = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects); | ||||
|     notifyDirectDefault = request->hasArg("SD"); | ||||
|     notifyDirect = notifyDirectDefault; | ||||
|     notifyButton = request->hasArg("SB"); | ||||
|     notifyAlexa = request->hasArg("SA"); | ||||
|     notifyHue = request->hasArg("SH"); | ||||
|     notifyMacro = request->hasArg("SM"); | ||||
|     notifyTwice = request->hasArg("S2"); | ||||
|  | ||||
|     receiveDirect = request->hasArg("RD"); | ||||
|     e131Multicast = request->hasArg("EM"); | ||||
|     t = request->arg("EU").toInt(); | ||||
|     if (t > 0  && t <= 63999) e131Universe = t; | ||||
|     t = request->arg("DA").toInt(); | ||||
|     if (t > 0  && t <= 510) DMXAddress = t; | ||||
|     t = request->arg("DM").toInt(); | ||||
|     if (t >= DMX_MODE_DISABLED && t <= DMX_MODE_MULTIPLE_DRGB) DMXMode = t; | ||||
|     t = request->arg("ET").toInt(); | ||||
|     if (t > 99  && t <= 65000) realtimeTimeoutMs = t; | ||||
|     arlsForceMaxBri = request->hasArg("FB"); | ||||
|     arlsDisableGammaCorrection = request->hasArg("RG"); | ||||
|     t = request->arg("WO").toInt(); | ||||
|     if (t >= -255  && t <= 255) arlsOffset = t; | ||||
|  | ||||
|     alexaEnabled = request->hasArg("AL"); | ||||
|     strlcpy(alexaInvocationName, request->arg("AI").c_str(), 33); | ||||
|  | ||||
|     if (request->hasArg("BK") && !request->arg("BK").equals("Hidden")) { | ||||
|       strlcpy(blynkApiKey, request->arg("BK").c_str(), 36); initBlynk(blynkApiKey); | ||||
|     } | ||||
|  | ||||
|     #ifdef WLED_ENABLE_MQTT | ||||
|     mqttEnabled = request->hasArg("MQ"); | ||||
|     strlcpy(mqttServer, request->arg("MS").c_str(), 33); | ||||
|     t = request->arg("MQPORT").toInt(); | ||||
|     if (t > 0) mqttPort = t; | ||||
|     strlcpy(mqttUser, request->arg("MQUSER").c_str(), 41); | ||||
|     if (!isAsterisksOnly(request->arg("MQPASS").c_str(), 41)) strlcpy(mqttPass, request->arg("MQPASS").c_str(), 41); | ||||
|     strlcpy(mqttClientID, request->arg("MQCID").c_str(), 41); | ||||
|     strlcpy(mqttDeviceTopic, request->arg("MD").c_str(), 33); | ||||
|     strlcpy(mqttGroupTopic, request->arg("MG").c_str(), 33); | ||||
|     #endif | ||||
|  | ||||
|     #ifndef WLED_DISABLE_HUESYNC | ||||
|     for (int i=0;i<4;i++){ | ||||
|       String a = "H"+String(i); | ||||
|       hueIP[i] = request->arg(a).toInt(); | ||||
|     } | ||||
|  | ||||
|     t = request->arg("HL").toInt(); | ||||
|     if (t > 0) huePollLightId = t; | ||||
|  | ||||
|     t = request->arg("HI").toInt(); | ||||
|     if (t > 50) huePollIntervalMs = t; | ||||
|  | ||||
|     hueApplyOnOff = request->hasArg("HO"); | ||||
|     hueApplyBri = request->hasArg("HB"); | ||||
|     hueApplyColor = request->hasArg("HC"); | ||||
|     huePollingEnabled = request->hasArg("HP"); | ||||
|     hueStoreAllowed = true; | ||||
|     reconnectHue(); | ||||
|     #endif | ||||
|   } | ||||
|  | ||||
|   //TIME | ||||
|   if (subPage == 5) | ||||
|   { | ||||
|     ntpEnabled = request->hasArg("NT"); | ||||
|     strlcpy(ntpServerName, request->arg("NS").c_str(), 33); | ||||
|     useAMPM = !request->hasArg("CF"); | ||||
|     currentTimezone = request->arg("TZ").toInt(); | ||||
|     utcOffsetSecs = request->arg("UO").toInt(); | ||||
|  | ||||
|     //start ntp if not already connected | ||||
|     if (ntpEnabled && WLED_CONNECTED && !ntpConnected) ntpConnected = ntpUdp.begin(ntpLocalPort); | ||||
|  | ||||
|     if (request->hasArg("OL")){ | ||||
|       overlayDefault = request->arg("OL").toInt(); | ||||
|       overlayCurrent = overlayDefault; | ||||
|     } | ||||
|  | ||||
|     overlayMin = request->arg("O1").toInt(); | ||||
|     overlayMax = request->arg("O2").toInt(); | ||||
|     analogClock12pixel = request->arg("OM").toInt(); | ||||
|     analogClock5MinuteMarks = request->hasArg("O5"); | ||||
|     analogClockSecondsTrail = request->hasArg("OS"); | ||||
|  | ||||
|     strcpy(cronixieDisplay,request->arg("CX").c_str()); | ||||
|     bool cbOld = cronixieBacklight; | ||||
|     cronixieBacklight = request->hasArg("CB"); | ||||
|     if (cbOld != cronixieBacklight && overlayCurrent == 3) | ||||
|     { | ||||
|       strip.setCronixieBacklight(cronixieBacklight); overlayRefreshedTime = 0; | ||||
|     } | ||||
|     countdownMode = request->hasArg("CE"); | ||||
|     countdownYear = request->arg("CY").toInt(); | ||||
|     countdownMonth = request->arg("CI").toInt(); | ||||
|     countdownDay = request->arg("CD").toInt(); | ||||
|     countdownHour = request->arg("CH").toInt(); | ||||
|     countdownMin = request->arg("CM").toInt(); | ||||
|     countdownSec = request->arg("CS").toInt(); | ||||
|  | ||||
|     for (int i=1;i<17;i++) | ||||
|     { | ||||
|       String a = "M"+String(i); | ||||
|       if (request->hasArg(a.c_str())) saveMacro(i,request->arg(a),false); | ||||
|     } | ||||
|  | ||||
|     macroBoot = request->arg("MB").toInt(); | ||||
|     macroAlexaOn = request->arg("A0").toInt(); | ||||
|     macroAlexaOff = request->arg("A1").toInt(); | ||||
|     macroButton = request->arg("MP").toInt(); | ||||
|     macroLongPress = request->arg("ML").toInt(); | ||||
|     macroCountdown = request->arg("MC").toInt(); | ||||
|     macroNl = request->arg("MN").toInt(); | ||||
|     macroDoublePress = request->arg("MD").toInt(); | ||||
|  | ||||
|     char k[3]; k[2] = 0; | ||||
|     for (int i = 0; i<8; i++) | ||||
|     { | ||||
|       k[1] = i+48;//ascii 0,1,2,3 | ||||
|  | ||||
|       k[0] = 'H'; //timer hours | ||||
|       timerHours[i] = request->arg(k).toInt(); | ||||
|  | ||||
|       k[0] = 'N'; //minutes | ||||
|       timerMinutes[i] = request->arg(k).toInt(); | ||||
|  | ||||
|       k[0] = 'T'; //macros | ||||
|       timerMacro[i] = request->arg(k).toInt(); | ||||
|  | ||||
|       k[0] = 'W'; //weekdays | ||||
|       timerWeekday[i] = request->arg(k).toInt(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //SECURITY | ||||
|   if (subPage == 6) | ||||
|   { | ||||
|     if (request->hasArg("RS")) //complete factory reset | ||||
|     { | ||||
|       clearEEPROM(); | ||||
|       serveMessage(request, 200, "All Settings erased.", "Connect to WLED-AP to setup again",255); | ||||
|       doReboot = true; | ||||
|     } | ||||
|  | ||||
|     bool pwdCorrect = !otaLock; //always allow access if ota not locked | ||||
|     if (request->hasArg("OP")) | ||||
|     { | ||||
|       if (otaLock && strcmp(otaPass,request->arg("OP").c_str()) == 0) | ||||
|       { | ||||
|         pwdCorrect = true; | ||||
|       } | ||||
|       if (!otaLock && request->arg("OP").length() > 0) | ||||
|       { | ||||
|         strlcpy(otaPass,request->arg("OP").c_str(), 33); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if (pwdCorrect) //allow changes if correct pwd or no ota active | ||||
|     { | ||||
|       otaLock = request->hasArg("NO"); | ||||
|       wifiLock = request->hasArg("OW"); | ||||
|       aOtaEnabled = request->hasArg("AO"); | ||||
|     } | ||||
|   } | ||||
|   if (subPage != 6 || !doReboot) saveSettingsToEEPROM(); //do not save if factory reset | ||||
|   if (subPage == 2) { | ||||
|     strip.init(useRGBW,ledCount,skipFirstLed); | ||||
|   } | ||||
|   if (subPage == 4) alexaInit(); | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| //helper to get int value at a position in string | ||||
| int getNumVal(const String* req, uint16_t pos) | ||||
| { | ||||
|   return req->substring(pos+3).toInt(); | ||||
| } | ||||
|  | ||||
|  | ||||
| //helper to get int value at a position in string | ||||
| bool updateVal(const String* req, const char* key, byte* val, byte minv=0, byte maxv=255) | ||||
| { | ||||
|   int pos = req->indexOf(key); | ||||
|   if (pos < 1) return false; | ||||
|  | ||||
|   if (req->charAt(pos+3) == '~') { | ||||
|     int out = getNumVal(req, pos+1); | ||||
|     if (out == 0) | ||||
|     { | ||||
|       if (req->charAt(pos+4) == '-') | ||||
|       { | ||||
|         *val = (*val <= minv)? maxv : *val -1; | ||||
|       } else { | ||||
|         *val = (*val >= maxv)? minv : *val +1; | ||||
|       } | ||||
|     } else { | ||||
|       out += *val; | ||||
|       if (out > maxv) out = maxv; | ||||
|       if (out < minv) out = minv; | ||||
|       *val = out; | ||||
|     } | ||||
|   } else | ||||
|   { | ||||
|     *val = getNumVal(req, pos); | ||||
|   } | ||||
|   return true; | ||||
| } | ||||
|  | ||||
|  | ||||
| //HTTP API request parser | ||||
| bool handleSet(AsyncWebServerRequest *request, const String& req) | ||||
| { | ||||
|   if (!(req.indexOf("win") >= 0)) return false; | ||||
|  | ||||
|   int pos = 0; | ||||
|   DEBUG_PRINT("API req: "); | ||||
|   DEBUG_PRINTLN(req); | ||||
|  | ||||
|   //save macro, requires &MS=<slot>(<macro>) format | ||||
|   pos = req.indexOf("&MS="); | ||||
|   if (pos > 0) { | ||||
|     int i = req.substring(pos + 4).toInt(); | ||||
|     pos = req.indexOf('(') +1; | ||||
|     if (pos > 0) { | ||||
|       int en = req.indexOf(')'); | ||||
|       String mc = req.substring(pos); | ||||
|       if (en > 0) mc = req.substring(pos, en); | ||||
|       saveMacro(i, mc); | ||||
|     } | ||||
|  | ||||
|     pos = req.indexOf("IN"); | ||||
|     if (pos < 1) XML_response(request); | ||||
|     return true; | ||||
|     //if you save a macro in one request, other commands in that request are ignored due to unwanted behavior otherwise | ||||
|   } | ||||
|  | ||||
|   strip.applyToAllSelected = true; | ||||
|  | ||||
|   //segment select (sets main segment) | ||||
|   byte prevMain = strip.getMainSegmentId(); | ||||
|   pos = req.indexOf("SM="); | ||||
|   if (pos > 0) { | ||||
|     strip.mainSegment = getNumVal(&req, pos); | ||||
|   } | ||||
|   byte main = strip.getMainSegmentId(); | ||||
|   if (main != prevMain) setValuesFromMainSeg(); | ||||
|  | ||||
|   pos = req.indexOf("SS="); | ||||
|   if (pos > 0) { | ||||
|     byte t = getNumVal(&req, pos); | ||||
|     if (t < strip.getMaxSegments()) main = t; | ||||
|   } | ||||
|  | ||||
|   WS2812FX::Segment& mainseg = strip.getSegment(main); | ||||
|   pos = req.indexOf("SV="); //segment selected | ||||
|   if (pos > 0) mainseg.setOption(0, (req.charAt(pos+3) != '0')); | ||||
|  | ||||
|   uint16_t startI = mainseg.start; | ||||
|   uint16_t stopI = mainseg.stop; | ||||
|   uint8_t grpI = mainseg.grouping; | ||||
|   uint16_t spcI = mainseg.spacing; | ||||
|   pos = req.indexOf("&S="); //segment start | ||||
|   if (pos > 0) { | ||||
|     startI = getNumVal(&req, pos); | ||||
|   } | ||||
|   pos = req.indexOf("S2="); //segment stop | ||||
|   if (pos > 0) { | ||||
|     stopI = getNumVal(&req, pos); | ||||
|   } | ||||
|   pos = req.indexOf("GP="); //segment grouping | ||||
|   if (pos > 0) { | ||||
|     grpI = getNumVal(&req, pos); | ||||
|     if (grpI == 0) grpI = 1; | ||||
|   } | ||||
|   pos = req.indexOf("SP="); //segment spacing | ||||
|   if (pos > 0) { | ||||
|     spcI = getNumVal(&req, pos); | ||||
|   } | ||||
|   strip.setSegment(main, startI, stopI, grpI, spcI); | ||||
|  | ||||
|   main = strip.getMainSegmentId(); | ||||
|  | ||||
|    //set presets | ||||
|   pos = req.indexOf("P1="); //sets first preset for cycle | ||||
|   if (pos > 0) presetCycleMin = getNumVal(&req, pos); | ||||
|  | ||||
|   pos = req.indexOf("P2="); //sets last preset for cycle | ||||
|   if (pos > 0) presetCycleMax = getNumVal(&req, pos); | ||||
|  | ||||
|   //preset cycle | ||||
|   pos = req.indexOf("CY="); | ||||
|   if (pos > 0) | ||||
|   { | ||||
|     presetCyclingEnabled = (req.charAt(pos+3) != '0'); | ||||
|     presetCycCurr = presetCycleMin; | ||||
|   } | ||||
|  | ||||
|   pos = req.indexOf("PT="); //sets cycle time in ms | ||||
|   if (pos > 0) { | ||||
|     int v = getNumVal(&req, pos); | ||||
|     if (v > 49) presetCycleTime = v; | ||||
|   } | ||||
|  | ||||
|   pos = req.indexOf("PA="); //apply brightness from preset | ||||
|   if (pos > 0) presetApplyBri = (req.charAt(pos+3) != '0'); | ||||
|  | ||||
|   pos = req.indexOf("PC="); //apply color from preset | ||||
|   if (pos > 0) presetApplyCol = (req.charAt(pos+3) != '0'); | ||||
|  | ||||
|   pos = req.indexOf("PX="); //apply effects from preset | ||||
|   if (pos > 0) presetApplyFx = (req.charAt(pos+3) != '0'); | ||||
|  | ||||
|   pos = req.indexOf("PS="); //saves current in preset | ||||
|   if (pos > 0) savePreset(getNumVal(&req, pos)); | ||||
|  | ||||
|   //apply preset | ||||
|   if (updateVal(&req, "PL=", &presetCycCurr, presetCycleMin, presetCycleMax)) { | ||||
|     applyPreset(presetCycCurr, presetApplyBri, presetApplyCol, presetApplyFx); | ||||
|   } | ||||
|  | ||||
|   //set brightness | ||||
|   updateVal(&req, "&A=", &bri); | ||||
|  | ||||
|   //set colors | ||||
|   updateVal(&req, "&R=", &col[0]); | ||||
|   updateVal(&req, "&G=", &col[1]); | ||||
|   updateVal(&req, "&B=", &col[2]); | ||||
|   updateVal(&req, "&W=", &col[3]); | ||||
|   updateVal(&req, "R2=", &colSec[0]); | ||||
|   updateVal(&req, "G2=", &colSec[1]); | ||||
|   updateVal(&req, "B2=", &colSec[2]); | ||||
|   updateVal(&req, "W2=", &colSec[3]); | ||||
|  | ||||
|   //set hue | ||||
|   pos = req.indexOf("HU="); | ||||
|   if (pos > 0) { | ||||
|     uint16_t temphue = getNumVal(&req, pos); | ||||
|     byte tempsat = 255; | ||||
|     pos = req.indexOf("SA="); | ||||
|     if (pos > 0) { | ||||
|       tempsat = getNumVal(&req, pos); | ||||
|     } | ||||
|     colorHStoRGB(temphue,tempsat,(req.indexOf("H2")>0)? colSec:col); | ||||
|   } | ||||
|  | ||||
|   //set color from HEX or 32bit DEC | ||||
|   pos = req.indexOf("CL="); | ||||
|   if (pos > 0) { | ||||
|     colorFromDecOrHexString(col, (char*)req.substring(pos + 3).c_str()); | ||||
|   } | ||||
|   pos = req.indexOf("C2="); | ||||
|   if (pos > 0) { | ||||
|     colorFromDecOrHexString(colSec, (char*)req.substring(pos + 3).c_str()); | ||||
|   } | ||||
|  | ||||
|   //set to random hue SR=0->1st SR=1->2nd | ||||
|   pos = req.indexOf("SR"); | ||||
|   if (pos > 0) { | ||||
|     _setRandomColor(getNumVal(&req, pos)); | ||||
|   } | ||||
|  | ||||
|   //swap 2nd & 1st | ||||
|   pos = req.indexOf("SC"); | ||||
|   if (pos > 0) { | ||||
|     byte temp; | ||||
|     for (uint8_t i=0; i<4; i++) | ||||
|     { | ||||
|       temp = col[i]; | ||||
|       col[i] = colSec[i]; | ||||
|       colSec[i] = temp; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //set effect parameters | ||||
|   if (updateVal(&req, "FX=", &effectCurrent, 0, strip.getModeCount()-1)) presetCyclingEnabled = false; | ||||
|   updateVal(&req, "SX=", &effectSpeed); | ||||
|   updateVal(&req, "IX=", &effectIntensity); | ||||
|   updateVal(&req, "FP=", &effectPalette, 0, strip.getPaletteCount()-1); | ||||
|  | ||||
|   //set advanced overlay | ||||
|   pos = req.indexOf("OL="); | ||||
|   if (pos > 0) { | ||||
|     overlayCurrent = getNumVal(&req, pos); | ||||
|   } | ||||
|  | ||||
|   //apply macro | ||||
|   pos = req.indexOf("&M="); | ||||
|   if (pos > 0) { | ||||
|     applyMacro(getNumVal(&req, pos)); | ||||
|   } | ||||
|  | ||||
|   //toggle send UDP direct notifications | ||||
|   pos = req.indexOf("SN="); | ||||
|   if (pos > 0) notifyDirect = (req.charAt(pos+3) != '0'); | ||||
|  | ||||
|   //toggle receive UDP direct notifications | ||||
|   pos = req.indexOf("RN="); | ||||
|   if (pos > 0) receiveNotifications = (req.charAt(pos+3) != '0'); | ||||
|  | ||||
|   //receive live data via UDP/Hyperion | ||||
|   pos = req.indexOf("RD="); | ||||
|   if (pos > 0) receiveDirect = (req.charAt(pos+3) != '0'); | ||||
|  | ||||
|   //toggle nightlight mode | ||||
|   bool aNlDef = false; | ||||
|   if (req.indexOf("&ND") > 0) aNlDef = true; | ||||
|   pos = req.indexOf("NL="); | ||||
|   if (pos > 0) | ||||
|   { | ||||
|     if (req.charAt(pos+3) == '0') | ||||
|     { | ||||
|       nightlightActive = false; | ||||
|       bri = briT; | ||||
|     } else { | ||||
|       nightlightActive = true; | ||||
|       if (!aNlDef) nightlightDelayMins = getNumVal(&req, pos); | ||||
|       nightlightStartTime = millis(); | ||||
|     } | ||||
|   } else if (aNlDef) | ||||
|   { | ||||
|     nightlightActive = true; | ||||
|     nightlightStartTime = millis(); | ||||
|   } | ||||
|  | ||||
|   //set nightlight target brightness | ||||
|   pos = req.indexOf("NT="); | ||||
|   if (pos > 0) { | ||||
|     nightlightTargetBri = getNumVal(&req, pos); | ||||
|     nightlightActiveOld = false; //re-init | ||||
|   } | ||||
|  | ||||
|   //toggle nightlight fade | ||||
|   pos = req.indexOf("NF="); | ||||
|   if (pos > 0) | ||||
|   { | ||||
|     nightlightFade = (req.charAt(pos+3) != '0'); | ||||
|     nightlightColorFade = (req.charAt(pos+3) == '2');  //NighLightColorFade can only be enabled via API or Macro with "NF=2" | ||||
|     nightlightActiveOld = false; //re-init | ||||
|   } | ||||
|  | ||||
|   #if AUXPIN >= 0 | ||||
|   //toggle general purpose output | ||||
|   pos = req.indexOf("AX="); | ||||
|   if (pos > 0) { | ||||
|     auxTime = getNumVal(&req, pos); | ||||
|     auxActive = true; | ||||
|     if (auxTime == 0) auxActive = false; | ||||
|   } | ||||
|   #endif | ||||
|  | ||||
|   pos = req.indexOf("TT="); | ||||
|   if (pos > 0) transitionDelay = getNumVal(&req, pos); | ||||
|  | ||||
|   //main toggle on/off | ||||
|   pos = req.indexOf("&T="); | ||||
|   if (pos > 0) { | ||||
|     nightlightActive = false; //always disable nightlight when toggling | ||||
|     switch (getNumVal(&req, pos)) | ||||
|     { | ||||
|       case 0: if (bri != 0){briLast = bri; bri = 0;} break; //off | ||||
|       case 1: bri = briLast; break; //on | ||||
|       default: toggleOnOff(); //toggle | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //Segment reverse | ||||
|   pos = req.indexOf("RV="); | ||||
|   if (pos > 0) strip.getSegment(main).setOption(1, req.charAt(pos+3) != '0'); | ||||
|  | ||||
|   //deactivate nightlight if target brightness is reached | ||||
|   if (bri == nightlightTargetBri) nightlightActive = false; | ||||
|   //set time (unix timestamp) | ||||
|   pos = req.indexOf("ST="); | ||||
|   if (pos > 0) { | ||||
|     setTime(getNumVal(&req, pos)); | ||||
|   } | ||||
|  | ||||
|   //set countdown goal (unix timestamp) | ||||
|   pos = req.indexOf("CT="); | ||||
|   if (pos > 0) { | ||||
|     countdownTime = getNumVal(&req, pos); | ||||
|     if (countdownTime - now() > 0) countdownOverTriggered = false; | ||||
|   } | ||||
|  | ||||
|   //cronixie | ||||
|   #ifndef WLED_DISABLE_CRONIXIE | ||||
|   //mode, 1 countdown | ||||
|   pos = req.indexOf("NM="); | ||||
|   if (pos > 0) countdownMode = (req.charAt(pos+3) != '0'); | ||||
|    | ||||
|   pos = req.indexOf("NX="); //sets digits to code | ||||
|   if (pos > 0) { | ||||
|     strlcpy(cronixieDisplay, req.substring(pos + 3, pos + 9).c_str(), 6); | ||||
|     setCronixie(); | ||||
|   } | ||||
|  | ||||
|   pos = req.indexOf("NB="); | ||||
|   if (pos > 0) //sets backlight | ||||
|   { | ||||
|     presetApplyFx = (req.charAt(pos+3) != '0'); | ||||
|     if (overlayCurrent == 3) strip.setCronixieBacklight(cronixieBacklight); | ||||
|     overlayRefreshedTime = 0; | ||||
|   } | ||||
|   #endif | ||||
|  | ||||
|   pos = req.indexOf("U0="); //user var 0 | ||||
|   if (pos > 0) { | ||||
|     userVar0 = getNumVal(&req, pos); | ||||
|   } | ||||
|  | ||||
|   pos = req.indexOf("U1="); //user var 1 | ||||
|   if (pos > 0) { | ||||
|     userVar1 = getNumVal(&req, pos); | ||||
|   } | ||||
|   //you can add more if you need | ||||
|  | ||||
|   //internal call, does not send XML response | ||||
|   pos = req.indexOf("IN"); | ||||
|   if (pos < 1) XML_response(request); | ||||
|  | ||||
|   pos = req.indexOf("&NN"); //do not send UDP notifications this time | ||||
|   colorUpdated((pos > 0) ? 5:1); | ||||
|  | ||||
|   return true; | ||||
| } | ||||
| @@ -1,332 +0,0 @@ | ||||
| /* | ||||
|  * Setup code | ||||
|  */ | ||||
|  | ||||
| void wledInit() | ||||
| { | ||||
|   EEPROM.begin(EEPSIZE); | ||||
|   ledCount = EEPROM.read(229) + ((EEPROM.read(398) << 8) & 0xFF00); | ||||
|   if (ledCount > MAX_LEDS || ledCount == 0) ledCount = 30; | ||||
|  | ||||
|   #ifdef ESP8266 | ||||
|   #if LEDPIN == 3 | ||||
|   if (ledCount > MAX_LEDS_DMA) ledCount = MAX_LEDS_DMA; //DMA method uses too much ram | ||||
|   #endif | ||||
|   #endif | ||||
|   Serial.begin(115200); | ||||
|   Serial.setTimeout(50); | ||||
|   DEBUG_PRINTLN(); | ||||
|   DEBUG_PRINT("---WLED "); DEBUG_PRINT(versionString); DEBUG_PRINT(" "); DEBUG_PRINT(VERSION); DEBUG_PRINTLN(" INIT---"); | ||||
|   #ifdef ARDUINO_ARCH_ESP32 | ||||
|   DEBUG_PRINT("esp32 ");   DEBUG_PRINTLN(ESP.getSdkVersion()); | ||||
|   #else | ||||
|   DEBUG_PRINT("esp8266 "); DEBUG_PRINTLN(ESP.getCoreVersion()); | ||||
|   #endif | ||||
|   int heapPreAlloc = ESP.getFreeHeap(); | ||||
|   DEBUG_PRINT("heap "); | ||||
|   DEBUG_PRINTLN(ESP.getFreeHeap()); | ||||
|  | ||||
|   strip.init(EEPROM.read(372),ledCount,EEPROM.read(2204)); //init LEDs quickly | ||||
|   strip.setBrightness(0); | ||||
|  | ||||
|   DEBUG_PRINT("LEDs inited. heap usage ~"); | ||||
|   DEBUG_PRINTLN(heapPreAlloc - ESP.getFreeHeap()); | ||||
|  | ||||
|   #ifndef WLED_DISABLE_FILESYSTEM | ||||
|    #ifdef ARDUINO_ARCH_ESP32 | ||||
|     SPIFFS.begin(true); | ||||
|    #endif | ||||
|     SPIFFS.begin(); | ||||
|   #endif | ||||
|  | ||||
|   DEBUG_PRINTLN("Load EEPROM"); | ||||
|   loadSettingsFromEEPROM(true); | ||||
|   beginStrip(); | ||||
|   userSetup(); | ||||
|   if (strcmp(clientSSID,"Your_Network") == 0) showWelcomePage = true; | ||||
|   WiFi.persistent(false); | ||||
|  | ||||
|   if (macroBoot>0) applyMacro(macroBoot); | ||||
|   Serial.println("Ada"); | ||||
|  | ||||
|   //generate module IDs | ||||
|   escapedMac = WiFi.macAddress(); | ||||
|   escapedMac.replace(":", ""); | ||||
|   escapedMac.toLowerCase(); | ||||
|   if (strcmp(cmDNS,"x") == 0) //fill in unique mdns default | ||||
|   { | ||||
|     strcpy(cmDNS, "wled-"); | ||||
|     sprintf(cmDNS+5, "%*s", 6, escapedMac.c_str()+6); | ||||
|   } | ||||
|   if (mqttDeviceTopic[0] == 0) | ||||
|   { | ||||
|     strcpy(mqttDeviceTopic, "wled/"); | ||||
|     sprintf(mqttDeviceTopic+5, "%*s", 6, escapedMac.c_str()+6); | ||||
|   } | ||||
|   if (mqttClientID[0] == 0) | ||||
|   { | ||||
|     strcpy(mqttClientID, "WLED-"); | ||||
|     sprintf(mqttClientID+5, "%*s", 6, escapedMac.c_str()+6); | ||||
|   } | ||||
|  | ||||
|   strip.service(); | ||||
|  | ||||
|   #ifndef WLED_DISABLE_OTA | ||||
|     if (aOtaEnabled) | ||||
|     { | ||||
|       ArduinoOTA.onStart([]() { | ||||
|         #ifdef ESP8266 | ||||
|         wifi_set_sleep_type(NONE_SLEEP_T); | ||||
|         #endif | ||||
|         DEBUG_PRINTLN("Start ArduinoOTA"); | ||||
|       }); | ||||
|       if (strlen(cmDNS) > 0) ArduinoOTA.setHostname(cmDNS); | ||||
|     } | ||||
|   #endif | ||||
|    | ||||
|   //HTTP server page init | ||||
|   initServer(); | ||||
| } | ||||
|  | ||||
|  | ||||
| void beginStrip() | ||||
| { | ||||
|   // Initialize NeoPixel Strip and button | ||||
|   strip.setShowCallback(handleOverlayDraw); | ||||
|  | ||||
| #ifdef BTNPIN | ||||
|   pinMode(BTNPIN, INPUT_PULLUP); | ||||
| #endif | ||||
|  | ||||
|   if (bootPreset>0) applyPreset(bootPreset, turnOnAtBoot, true, true); | ||||
|   colorUpdated(0); | ||||
|  | ||||
|   //init relay pin | ||||
|   #if RLYPIN >= 0 | ||||
|     pinMode(RLYPIN, OUTPUT); | ||||
|     #if RLYMDE | ||||
|       digitalWrite(RLYPIN, bri); | ||||
|     #else | ||||
|       digitalWrite(RLYPIN, !bri); | ||||
|     #endif | ||||
|   #endif | ||||
|  | ||||
|   //disable button if it is "pressed" unintentionally | ||||
| #ifdef BTNPIN | ||||
|   if(digitalRead(BTNPIN) == LOW) buttonEnabled = false; | ||||
| #else | ||||
|   buttonEnabled = false; | ||||
| #endif | ||||
| } | ||||
|  | ||||
|  | ||||
| void initAP(bool resetAP=false){ | ||||
|   if (apBehavior == 3 && !resetAP) return; | ||||
|  | ||||
|   if (!apSSID[0] || resetAP) strcpy(apSSID, "WLED-AP"); | ||||
|   if (resetAP) strcpy(apPass,"wled1234"); | ||||
|   DEBUG_PRINT("Opening access point "); | ||||
|   DEBUG_PRINTLN(apSSID); | ||||
|   WiFi.softAPConfig(IPAddress(4, 3, 2, 1), IPAddress(4, 3, 2, 1), IPAddress(255,255,255,0)); | ||||
|   WiFi.softAP(apSSID, apPass, apChannel, apHide); | ||||
|    | ||||
|   if (!apActive) //start captive portal if AP active | ||||
|   { | ||||
|     DEBUG_PRINTLN("Init AP interfaces"); | ||||
|     server.begin(); | ||||
|     if (udpPort > 0 && udpPort != ntpLocalPort) | ||||
|     { | ||||
|       udpConnected = notifierUdp.begin(udpPort); | ||||
|     } | ||||
|     if (udpRgbPort > 0 && udpRgbPort != ntpLocalPort && udpRgbPort != udpPort) | ||||
|     { | ||||
|       udpRgbConnected = rgbUdp.begin(udpRgbPort); | ||||
|     } | ||||
|  | ||||
|     dnsServer.setErrorReplyCode(DNSReplyCode::NoError); | ||||
|     dnsServer.start(53, "*", WiFi.softAPIP()); | ||||
|   } | ||||
|   apActive = true; | ||||
| } | ||||
|  | ||||
| void initConnection() | ||||
| { | ||||
|   WiFi.disconnect(); //close old connections | ||||
|   #ifdef ESP8266 | ||||
|   WiFi.setPhyMode(WIFI_PHY_MODE_11N); | ||||
|   #endif | ||||
|  | ||||
|   if (staticIP[0] != 0 && staticGateway[0] != 0) | ||||
|   { | ||||
|     WiFi.config(staticIP, staticGateway, staticSubnet, IPAddress(8,8,8,8)); | ||||
|   } else | ||||
|   { | ||||
|     WiFi.config(0U, 0U, 0U); | ||||
|   } | ||||
|  | ||||
|   lastReconnectAttempt = millis(); | ||||
|  | ||||
|   if (!WLED_WIFI_CONFIGURED) | ||||
|   { | ||||
|     DEBUG_PRINT("No connection configured. "); | ||||
|     if (!apActive) initAP(); //instantly go to ap mode | ||||
|     return; | ||||
|   } else if (!apActive) { | ||||
|     if (apBehavior == 2) | ||||
|     { | ||||
|       initAP(); | ||||
|     } else | ||||
|     { | ||||
|       DEBUG_PRINTLN("Access point disabled."); | ||||
|       WiFi.softAPdisconnect(true); | ||||
|     } | ||||
|   } | ||||
|   showWelcomePage = false; | ||||
|    | ||||
|   DEBUG_PRINT("Connecting to "); | ||||
|   DEBUG_PRINT(clientSSID); | ||||
|   DEBUG_PRINTLN("..."); | ||||
|  | ||||
|   #ifdef ESP8266 | ||||
|    WiFi.hostname(serverDescription); | ||||
|   #endif | ||||
|    WiFi.begin(clientSSID, clientPass); | ||||
|   #ifdef ARDUINO_ARCH_ESP32 | ||||
|    WiFi.setHostname(serverDescription); | ||||
|   #endif | ||||
| } | ||||
|  | ||||
| void initInterfaces() { | ||||
|   DEBUG_PRINTLN("Init STA interfaces"); | ||||
|    | ||||
|   if (hueIP[0] == 0) | ||||
|   { | ||||
|     hueIP[0] = WiFi.localIP()[0]; | ||||
|     hueIP[1] = WiFi.localIP()[1]; | ||||
|     hueIP[2] = WiFi.localIP()[2]; | ||||
|   } | ||||
|  | ||||
|   //init Alexa hue emulation | ||||
|   if (alexaEnabled) alexaInit(); | ||||
|  | ||||
|   #ifndef WLED_DISABLE_OTA | ||||
|    if (aOtaEnabled) ArduinoOTA.begin(); | ||||
|   #endif | ||||
|  | ||||
|   strip.service(); | ||||
|   // Set up mDNS responder: | ||||
|   if (strlen(cmDNS) > 0) | ||||
|   { | ||||
|     if (!aOtaEnabled) MDNS.begin(cmDNS); | ||||
|  | ||||
|     DEBUG_PRINTLN("mDNS started"); | ||||
|     MDNS.addService("http", "tcp", 80); | ||||
|     MDNS.addService("wled", "tcp", 80); | ||||
|     MDNS.addServiceTxt("wled", "tcp", "mac", escapedMac.c_str()); | ||||
|   } | ||||
|   server.begin(); | ||||
|  | ||||
|   if (udpPort > 0 && udpPort != ntpLocalPort) | ||||
|   { | ||||
|     udpConnected = notifierUdp.begin(udpPort); | ||||
|     if (udpConnected && udpRgbPort != udpPort) udpRgbConnected = rgbUdp.begin(udpRgbPort); | ||||
|   } | ||||
|   if (ntpEnabled) ntpConnected = ntpUdp.begin(ntpLocalPort); | ||||
|  | ||||
|   initBlynk(blynkApiKey); | ||||
|   e131.begin((e131Multicast) ? E131_MULTICAST : E131_UNICAST , e131Universe, E131_MAX_UNIVERSE_COUNT); | ||||
|   reconnectHue(); | ||||
|   initMqtt(); | ||||
|   interfacesInited = true; | ||||
|   wasConnected = true; | ||||
| } | ||||
|  | ||||
| byte stacO = 0; | ||||
| uint32_t lastHeap; | ||||
| unsigned long heapTime = 0; | ||||
|  | ||||
| void handleConnection() { | ||||
|   if (millis() < 2000 && (!WLED_WIFI_CONFIGURED || apBehavior == 2)) return; | ||||
|   if (lastReconnectAttempt == 0) initConnection(); | ||||
|  | ||||
|   //reconnect WiFi to clear stale allocations if heap gets too low | ||||
|   if (millis() - heapTime > 5000) | ||||
|   { | ||||
|     uint32_t heap = ESP.getFreeHeap(); | ||||
|     if (heap < 9000 && lastHeap < 9000) { | ||||
|       DEBUG_PRINT("Heap too low! "); | ||||
|       DEBUG_PRINTLN(heap); | ||||
|       forceReconnect = true; | ||||
|     } | ||||
|     lastHeap = heap; | ||||
|     heapTime = millis(); | ||||
|   } | ||||
|    | ||||
|   byte stac = 0; | ||||
|   if (apActive) { | ||||
|     #ifdef ESP8266 | ||||
|     stac = wifi_softap_get_station_num(); | ||||
|     #else | ||||
|     wifi_sta_list_t stationList; | ||||
|     esp_wifi_ap_get_sta_list(&stationList); | ||||
|     stac = stationList.num; | ||||
|     #endif | ||||
|     if (stac != stacO) | ||||
|     { | ||||
|       stacO = stac; | ||||
|       DEBUG_PRINT("Connected AP clients: "); | ||||
|       DEBUG_PRINTLN(stac); | ||||
|       if (!WLED_CONNECTED && WLED_WIFI_CONFIGURED) { //trying to connect, but not connected | ||||
|         if (stac) WiFi.disconnect(); //disable search so that AP can work | ||||
|         else initConnection(); //restart search  | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   if (forceReconnect) { | ||||
|     DEBUG_PRINTLN("Forcing reconnect."); | ||||
|     initConnection(); | ||||
|     interfacesInited = false; | ||||
|     forceReconnect = false; | ||||
|     wasConnected = false; | ||||
|     return; | ||||
|   } | ||||
|   if (!WLED_CONNECTED) { | ||||
|     if (interfacesInited) { | ||||
|       DEBUG_PRINTLN("Disconnected!"); | ||||
|       interfacesInited = false; | ||||
|       initConnection(); | ||||
|     } | ||||
|     if (millis() - lastReconnectAttempt > ((stac) ? 300000 : 20000) && WLED_WIFI_CONFIGURED) initConnection(); | ||||
|     if (!apActive && millis() - lastReconnectAttempt > 12000 && (!wasConnected || apBehavior == 1)) initAP();  | ||||
|   } else if (!interfacesInited) { //newly connected | ||||
|     DEBUG_PRINTLN(""); | ||||
|     DEBUG_PRINT("Connected! IP address: "); | ||||
|     DEBUG_PRINTLN(WiFi.localIP()); | ||||
|     initInterfaces(); | ||||
|     userConnected(); | ||||
|  | ||||
|     //shut down AP | ||||
|     if (apBehavior != 2 && apActive) | ||||
|     { | ||||
|       dnsServer.stop(); | ||||
|       WiFi.softAPdisconnect(true); | ||||
|       apActive = false; | ||||
|       DEBUG_PRINTLN("Access point disabled."); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| //by https://github.com/tzapu/WiFiManager/blob/master/WiFiManager.cpp | ||||
| int getSignalQuality(int rssi) | ||||
| { | ||||
|   int quality = 0; | ||||
|  | ||||
|   if (rssi <= -100) { | ||||
|     quality = 0; | ||||
|   } else if (rssi >= -50) { | ||||
|     quality = 100; | ||||
|   } else { | ||||
|     quality = 2 * (rssi + 100); | ||||
|   } | ||||
|   return quality; | ||||
| } | ||||
| @@ -1,399 +0,0 @@ | ||||
| /* | ||||
|  * UDP notifier | ||||
|  */ | ||||
|  | ||||
| #define WLEDPACKETSIZE 29 | ||||
| #define UDP_IN_MAXSIZE 1472 | ||||
|  | ||||
|  | ||||
| void notify(byte callMode, bool followUp=false) | ||||
| { | ||||
|   if (!udpConnected) return; | ||||
|   switch (callMode) | ||||
|   { | ||||
|     case 0: return; | ||||
|     case 1: if (!notifyDirect) return; break; | ||||
|     case 2: if (!notifyButton) return; break; | ||||
|     case 4: if (!notifyDirect) return; break; | ||||
|     case 6: if (!notifyDirect) return; break; //fx change | ||||
|     case 7: if (!notifyHue)    return; break; | ||||
|     case 8: if (!notifyDirect) return; break; | ||||
|     case 9: if (!notifyDirect) return; break; | ||||
|     case 10: if (!notifyAlexa) return; break; | ||||
|     default: return; | ||||
|   } | ||||
|   byte udpOut[WLEDPACKETSIZE]; | ||||
|   udpOut[0] = 0; //0: wled notifier protocol 1: WARLS protocol | ||||
|   udpOut[1] = callMode; | ||||
|   udpOut[2] = bri; | ||||
|   udpOut[3] = col[0]; | ||||
|   udpOut[4] = col[1]; | ||||
|   udpOut[5] = col[2]; | ||||
|   udpOut[6] = nightlightActive; | ||||
|   udpOut[7] = nightlightDelayMins; | ||||
|   udpOut[8] = effectCurrent; | ||||
|   udpOut[9] = effectSpeed; | ||||
|   udpOut[10] = col[3]; | ||||
|   //compatibilityVersionByte:  | ||||
|   //0: old 1: supports white 2: supports secondary color | ||||
|   //3: supports FX intensity, 24 byte packet 4: supports transitionDelay 5: sup palette | ||||
|   //6: supports timebase syncing, 29 byte packet 7: supports tertiary color  | ||||
|   udpOut[11] = 7;  | ||||
|   udpOut[12] = colSec[0]; | ||||
|   udpOut[13] = colSec[1]; | ||||
|   udpOut[14] = colSec[2]; | ||||
|   udpOut[15] = colSec[3]; | ||||
|   udpOut[16] = effectIntensity; | ||||
|   udpOut[17] = (transitionDelay >> 0) & 0xFF; | ||||
|   udpOut[18] = (transitionDelay >> 8) & 0xFF; | ||||
|   udpOut[19] = effectPalette; | ||||
|   uint32_t colTer = strip.getSegment(strip.getMainSegmentId()).colors[2]; | ||||
|   udpOut[20] = (colTer >> 16) & 0xFF; | ||||
|   udpOut[21] = (colTer >>  8) & 0xFF; | ||||
|   udpOut[22] = (colTer >>  0) & 0xFF; | ||||
|   udpOut[23] = (colTer >> 24) & 0xFF; | ||||
|    | ||||
|   udpOut[24] = followUp; | ||||
|   uint32_t t = millis() + strip.timebase; | ||||
|   udpOut[25] = (t >> 24) & 0xFF; | ||||
|   udpOut[26] = (t >> 16) & 0xFF; | ||||
|   udpOut[27] = (t >>  8) & 0xFF; | ||||
|   udpOut[28] = (t >>  0) & 0xFF; | ||||
|    | ||||
|   IPAddress broadcastIp; | ||||
|   broadcastIp = ~uint32_t(WiFi.subnetMask()) | uint32_t(WiFi.gatewayIP()); | ||||
|  | ||||
|   notifierUdp.beginPacket(broadcastIp, udpPort); | ||||
|   notifierUdp.write(udpOut, WLEDPACKETSIZE); | ||||
|   notifierUdp.endPacket(); | ||||
|   notificationSentCallMode = callMode; | ||||
|   notificationSentTime = millis(); | ||||
|   notificationTwoRequired = (followUp)? false:notifyTwice; | ||||
| } | ||||
|  | ||||
|  | ||||
| void arlsLock(uint32_t timeoutMs, byte md = REALTIME_MODE_GENERIC) | ||||
| { | ||||
|   if (!realtimeMode){ | ||||
|     for (uint16_t i = 0; i < ledCount; i++) | ||||
|     { | ||||
|       strip.setPixelColor(i,0,0,0,0); | ||||
|     } | ||||
|     realtimeMode = md; | ||||
|   } | ||||
|   realtimeTimeout = millis() + timeoutMs; | ||||
|   if (timeoutMs == 255001 || timeoutMs == 65000) realtimeTimeout = UINT32_MAX; | ||||
|   if (arlsForceMaxBri) strip.setBrightness(255); | ||||
| } | ||||
|  | ||||
|  | ||||
| void handleE131Packet(e131_packet_t* p, IPAddress clientIP){ | ||||
|   //E1.31 protocol support | ||||
|    | ||||
|   // skip out-of-sequence packets | ||||
|   if (p->sequence_number < e131LastSequenceNumber && p->sequence_number - e131LastSequenceNumber > -20){ | ||||
|     DEBUG_PRINT("skipping E1.31 frame (last seq="); | ||||
|     DEBUG_PRINT(e131LastSequenceNumber); | ||||
|     DEBUG_PRINT(", current seq="); | ||||
|     DEBUG_PRINT(p->sequence_number); | ||||
|     DEBUG_PRINTLN(")"); | ||||
|     e131LastSequenceNumber = p->sequence_number; | ||||
|     return; | ||||
|   } | ||||
|   e131LastSequenceNumber = p->sequence_number; | ||||
|  | ||||
|   // update status info | ||||
|   realtimeIP = clientIP; | ||||
|    | ||||
|   uint16_t uni = htons(p->universe); | ||||
|   uint8_t previousUniverses = uni - e131Universe; | ||||
|   uint16_t possibleLEDsInCurrentUniverse; | ||||
|   uint16_t dmxChannels = htons(p->property_value_count) -1; | ||||
|  | ||||
|   switch (DMXMode) { | ||||
|     case DMX_MODE_DISABLED: | ||||
|       return;  // nothing to do | ||||
|       break; | ||||
|  | ||||
|     case DMX_MODE_SINGLE_RGB: | ||||
|       if (uni != e131Universe) return; | ||||
|       if (dmxChannels-DMXAddress+1 < 3) return; | ||||
|       for (uint16_t i = 0; i < ledCount; i++) | ||||
|         setRealtimePixel(i, p->property_values[DMXAddress+0], p->property_values[DMXAddress+1], p->property_values[DMXAddress+2], 0); | ||||
|       break; | ||||
|  | ||||
|     case DMX_MODE_SINGLE_DRGB: | ||||
|       if (uni != e131Universe) return; | ||||
|       if (dmxChannels-DMXAddress+1 < 4) return; | ||||
|       if (DMXOldDimmer != p->property_values[DMXAddress+0]) { | ||||
|         DMXOldDimmer = p->property_values[DMXAddress+0]; | ||||
|         bri = p->property_values[DMXAddress+0]; | ||||
|         strip.setBrightness(bri); | ||||
|       } | ||||
|       for (uint16_t i = 0; i < ledCount; i++) | ||||
|         setRealtimePixel(i, p->property_values[DMXAddress+1], p->property_values[DMXAddress+2], p->property_values[DMXAddress+3], 0); | ||||
|       break; | ||||
|  | ||||
|     case DMX_MODE_EFFECT: | ||||
|       if (uni != e131Universe) return; | ||||
|       if (dmxChannels-DMXAddress+1 < 11) return; | ||||
|       if (DMXOldDimmer != p->property_values[DMXAddress+0]) { | ||||
|         DMXOldDimmer = p->property_values[DMXAddress+0]; | ||||
|         bri = p->property_values[DMXAddress+0]; | ||||
|       } | ||||
|       if (p->property_values[DMXAddress+1] < MODE_COUNT) | ||||
|         effectCurrent = p->property_values[DMXAddress+ 1]; | ||||
|       effectSpeed     = p->property_values[DMXAddress+ 2];  // flickers | ||||
|       effectIntensity = p->property_values[DMXAddress+ 3]; | ||||
|       effectPalette   = p->property_values[DMXAddress+ 4]; | ||||
|       col[0]          = p->property_values[DMXAddress+ 5]; | ||||
|       col[1]          = p->property_values[DMXAddress+ 6]; | ||||
|       col[2]          = p->property_values[DMXAddress+ 7]; | ||||
|       colSec[0]       = p->property_values[DMXAddress+ 8]; | ||||
|       colSec[1]       = p->property_values[DMXAddress+ 9]; | ||||
|       colSec[2]       = p->property_values[DMXAddress+10]; | ||||
|       if (dmxChannels-DMXAddress+1 > 11) | ||||
|       { | ||||
|         col[3]          = p->property_values[DMXAddress+11]; //white | ||||
|         colSec[3]       = p->property_values[DMXAddress+12]; | ||||
|       } | ||||
|       transitionDelayTemp = 0; // act fast | ||||
|       colorUpdated(3);         // don't send UDP | ||||
|       return;                  // don't activate realtime live mode | ||||
|       break; | ||||
|  | ||||
|     case DMX_MODE_MULTIPLE_RGB: | ||||
|       if (previousUniverses == 0) { | ||||
|         // first universe of this fixture | ||||
|         possibleLEDsInCurrentUniverse = (dmxChannels - DMXAddress + 1) / 3; | ||||
|         for (uint16_t i = 0; i < ledCount; i++) { | ||||
|           if (i >= possibleLEDsInCurrentUniverse) break;  // more LEDs will follow in next universe(s) | ||||
|           setRealtimePixel(i, p->property_values[DMXAddress+i*3+0], p->property_values[DMXAddress+i*3+1], p->property_values[DMXAddress+i*3+2], 0); | ||||
|         } | ||||
|       } else if (previousUniverses > 0 && uni < (e131Universe + E131_MAX_UNIVERSE_COUNT)) { | ||||
|         // additional universe(s) of this fixture | ||||
|         uint16_t numberOfLEDsInPreviousUniverses = ((512 - DMXAddress + 1) / 3);                            // first universe | ||||
|         if (previousUniverses > 1) numberOfLEDsInPreviousUniverses += (512 / 3) * (previousUniverses - 1);  // extended universe(s) before current | ||||
|         possibleLEDsInCurrentUniverse = dmxChannels / 3; | ||||
|         for (uint16_t i = numberOfLEDsInPreviousUniverses; i < ledCount; i++) { | ||||
|           uint8_t j = i - numberOfLEDsInPreviousUniverses; | ||||
|           if (j >= possibleLEDsInCurrentUniverse) break;   // more LEDs will follow in next universe(s) | ||||
|           setRealtimePixel(i, p->property_values[j*3+1], p->property_values[j*3+2], p->property_values[j*3+3], 0); | ||||
|         } | ||||
|       } | ||||
|       break; | ||||
|  | ||||
|     case DMX_MODE_MULTIPLE_DRGB: | ||||
|       if (previousUniverses == 0) { | ||||
|         // first universe of this fixture | ||||
|         if (DMXOldDimmer != p->property_values[DMXAddress+0]) { | ||||
|           DMXOldDimmer = p->property_values[DMXAddress+0]; | ||||
|           bri = p->property_values[DMXAddress+0]; | ||||
|           strip.setBrightness(bri); | ||||
|         } | ||||
|         possibleLEDsInCurrentUniverse = (dmxChannels - DMXAddress) / 3; | ||||
|         for (uint16_t i = 0; i < ledCount; i++) { | ||||
|           if (i >= possibleLEDsInCurrentUniverse) break;  // more LEDs will follow in next universe(s) | ||||
|           setRealtimePixel(i, p->property_values[DMXAddress+i*3+1], p->property_values[DMXAddress+i*3+2], p->property_values[DMXAddress+i*3+3], 0); | ||||
|         } | ||||
|       } else if (previousUniverses > 0 && uni < (e131Universe + E131_MAX_UNIVERSE_COUNT)) { | ||||
|         // additional universe(s) of this fixture | ||||
|         uint16_t numberOfLEDsInPreviousUniverses = ((512 - DMXAddress + 1) / 3);                            // first universe | ||||
|         if (previousUniverses > 1) numberOfLEDsInPreviousUniverses += (512 / 3) * (previousUniverses - 1);  // extended universe(s) before current | ||||
|         possibleLEDsInCurrentUniverse = dmxChannels / 3; | ||||
|         for (uint16_t i = numberOfLEDsInPreviousUniverses; i < ledCount; i++) { | ||||
|           uint8_t j = i - numberOfLEDsInPreviousUniverses; | ||||
|           if (j >= possibleLEDsInCurrentUniverse) break;   // more LEDs will follow in next universe(s) | ||||
|           setRealtimePixel(i, p->property_values[j*3+1], p->property_values[j*3+2], p->property_values[j*3+3], 0); | ||||
|         } | ||||
|       } | ||||
|       break; | ||||
|  | ||||
|     default: | ||||
|       DEBUG_PRINTLN("unknown E1.31 DMX mode"); | ||||
|       return;  // nothing to do | ||||
|       break; | ||||
|   } | ||||
|  | ||||
|   arlsLock(realtimeTimeoutMs, REALTIME_MODE_E131); | ||||
|   e131NewData = true; | ||||
| } | ||||
|  | ||||
|  | ||||
| void handleNotifications() | ||||
| { | ||||
|   //send second notification if enabled | ||||
|   if(udpConnected && notificationTwoRequired && millis()-notificationSentTime > 250){ | ||||
|     notify(notificationSentCallMode,true); | ||||
|   } | ||||
|  | ||||
|   if (e131NewData && millis() - strip.getLastShow() > 15) | ||||
|   { | ||||
|     e131NewData = false; | ||||
|     strip.show(); | ||||
|   } | ||||
|  | ||||
|   //unlock strip when realtime UDP times out | ||||
|   if (realtimeMode && millis() > realtimeTimeout) | ||||
|   { | ||||
|     strip.setBrightness(bri); | ||||
|     realtimeMode = REALTIME_MODE_INACTIVE; | ||||
|   } | ||||
|  | ||||
|   //receive UDP notifications | ||||
|   if (!udpConnected || !(receiveNotifications || receiveDirect)) return; | ||||
|      | ||||
|   uint16_t packetSize = notifierUdp.parsePacket(); | ||||
|  | ||||
|   //hyperion / raw RGB | ||||
|   if (!packetSize && udpRgbConnected) { | ||||
|     packetSize = rgbUdp.parsePacket(); | ||||
|     if (!receiveDirect) return; | ||||
|     if (packetSize > UDP_IN_MAXSIZE || packetSize < 3) return; | ||||
|     realtimeIP = rgbUdp.remoteIP(); | ||||
|     DEBUG_PRINTLN(rgbUdp.remoteIP()); | ||||
|     uint8_t lbuf[packetSize]; | ||||
|     rgbUdp.read(lbuf, packetSize); | ||||
|     arlsLock(realtimeTimeoutMs, REALTIME_MODE_HYPERION); | ||||
|     uint16_t id = 0; | ||||
|     for (uint16_t i = 0; i < packetSize -2; i += 3) | ||||
|     { | ||||
|       setRealtimePixel(id, lbuf[i], lbuf[i+1], lbuf[i+2], 0); | ||||
|        | ||||
|       id++; if (id >= ledCount) break; | ||||
|     } | ||||
|     strip.show(); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   //notifier and UDP realtime | ||||
|   if (packetSize > UDP_IN_MAXSIZE) return; | ||||
|   if(packetSize && notifierUdp.remoteIP() != WiFi.localIP()) //don't process broadcasts we send ourselves | ||||
|   { | ||||
|     uint8_t udpIn[packetSize]; | ||||
|     notifierUdp.read(udpIn, packetSize); | ||||
|  | ||||
|     //wled notifier, block if realtime packets active | ||||
|     if (udpIn[0] == 0 && !realtimeMode && receiveNotifications) | ||||
|     { | ||||
|       //ignore notification if received within a second after sending a notification ourselves | ||||
|       if (millis() - notificationSentTime < 1000) return; | ||||
|       if (udpIn[1] > 199) return; //do not receive custom versions | ||||
|        | ||||
|       bool someSel = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects); | ||||
|       //apply colors from notification | ||||
|       if (receiveNotificationColor || !someSel) | ||||
|       { | ||||
|         col[0] = udpIn[3]; | ||||
|         col[1] = udpIn[4]; | ||||
|         col[2] = udpIn[5]; | ||||
|         if (udpIn[11] > 0) //sending module's white val is intended | ||||
|         { | ||||
|           col[3] = udpIn[10]; | ||||
|           if (udpIn[11] > 1) | ||||
|           { | ||||
|             colSec[0] = udpIn[12]; | ||||
|             colSec[1] = udpIn[13]; | ||||
|             colSec[2] = udpIn[14]; | ||||
|             colSec[3] = udpIn[15]; | ||||
|           } | ||||
|           if (udpIn[11] > 5) | ||||
|           { | ||||
|             uint32_t t = (udpIn[25] << 24) | (udpIn[26] << 16) | (udpIn[27] << 8) | (udpIn[28]); | ||||
|             t += 2; | ||||
|             t -= millis(); | ||||
|             strip.timebase = t; | ||||
|           } | ||||
|           if (udpIn[11] > 6) | ||||
|           { | ||||
|             strip.setColor(2, udpIn[20], udpIn[21], udpIn[22], udpIn[23]); //tertiary color | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       //apply effects from notification | ||||
|       if (udpIn[11] < 200 && (receiveNotificationEffects || !someSel)) | ||||
|       { | ||||
|         if (udpIn[8] < strip.getModeCount()) effectCurrent = udpIn[8]; | ||||
|         effectSpeed   = udpIn[9]; | ||||
|         if (udpIn[11] > 2) effectIntensity = udpIn[16]; | ||||
|         if (udpIn[11] > 4 && udpIn[19] < strip.getPaletteCount()) effectPalette = udpIn[19]; | ||||
|       } | ||||
|        | ||||
|       if (udpIn[11] > 3) | ||||
|       { | ||||
|         transitionDelayTemp = ((udpIn[17] << 0) & 0xFF) + ((udpIn[18] << 8) & 0xFF00); | ||||
|       } | ||||
|  | ||||
|       nightlightActive = udpIn[6]; | ||||
|       if (nightlightActive) nightlightDelayMins = udpIn[7]; | ||||
|        | ||||
|       if (receiveNotificationBrightness || !someSel) bri = udpIn[2]; | ||||
|       colorUpdated(3); | ||||
|        | ||||
|     }  else if (udpIn[0] > 0 && udpIn[0] < 5 && receiveDirect) //1 warls //2 drgb //3 drgbw | ||||
|     { | ||||
|       realtimeIP = notifierUdp.remoteIP(); | ||||
|       DEBUG_PRINTLN(notifierUdp.remoteIP()); | ||||
|       if (packetSize > 1) { | ||||
|         if (udpIn[1] == 0) | ||||
|         { | ||||
|           realtimeTimeout = 0; | ||||
|           return; | ||||
|         } else { | ||||
|           arlsLock(udpIn[1]*1000 +1, REALTIME_MODE_UDP); | ||||
|         } | ||||
|         if (udpIn[0] == 1) //warls | ||||
|         { | ||||
|           for (uint16_t i = 2; i < packetSize -3; i += 4) | ||||
|           { | ||||
|             setRealtimePixel(udpIn[i], udpIn[i+1], udpIn[i+2], udpIn[i+3], 0); | ||||
|           } | ||||
|         } else if (udpIn[0] == 2) //drgb | ||||
|         { | ||||
|           uint16_t id = 0; | ||||
|           for (uint16_t i = 2; i < packetSize -2; i += 3) | ||||
|           { | ||||
|             setRealtimePixel(id, udpIn[i], udpIn[i+1], udpIn[i+2], 0); | ||||
|  | ||||
|             id++; if (id >= ledCount) break; | ||||
|           } | ||||
|         } else if (udpIn[0] == 3) //drgbw | ||||
|         { | ||||
|           uint16_t id = 0; | ||||
|           for (uint16_t i = 2; i < packetSize -3; i += 4) | ||||
|           { | ||||
|             setRealtimePixel(id, udpIn[i], udpIn[i+1], udpIn[i+2], udpIn[i+3]); | ||||
|              | ||||
|             id++; if (id >= ledCount) break; | ||||
|           } | ||||
|         } else if (udpIn[0] == 4) //dnrgb | ||||
|         { | ||||
|           uint16_t id = ((udpIn[3] << 0) & 0xFF) + ((udpIn[2] << 8) & 0xFF00); | ||||
|           for (uint16_t i = 4; i < packetSize -2; i += 3) | ||||
|           { | ||||
|              if (id >= ledCount) break; | ||||
|             setRealtimePixel(id, udpIn[i], udpIn[i+1], udpIn[i+2], 0); | ||||
|             id++; | ||||
|           } | ||||
|         } | ||||
|         strip.show(); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| void setRealtimePixel(uint16_t i, byte r, byte g, byte b, byte w) | ||||
| { | ||||
|   uint16_t pix = i + arlsOffset; | ||||
|   if (pix < ledCount) | ||||
|   { | ||||
|     if (!arlsDisableGammaCorrection && strip.gammaCorrectCol) | ||||
|     { | ||||
|       strip.setPixelColor(pix, strip.gamma8(r), strip.gamma8(g), strip.gamma8(b), strip.gamma8(w)); | ||||
|     } else { | ||||
|       strip.setPixelColor(pix, r, g, b, w); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,263 +0,0 @@ | ||||
| /* | ||||
|  * LED methods | ||||
|  */ | ||||
| void setValuesFromMainSeg() | ||||
| { | ||||
|   WS2812FX::Segment& seg = strip.getSegment(strip.getMainSegmentId()); | ||||
|   colorFromUint32(seg.colors[0]); | ||||
|   colorFromUint32(seg.colors[1], true); | ||||
|   effectCurrent = seg.mode; | ||||
|   effectSpeed = seg.speed; | ||||
|   effectIntensity = seg.intensity; | ||||
|   effectPalette = seg.palette; | ||||
| } | ||||
|  | ||||
|  | ||||
| void resetTimebase() | ||||
| { | ||||
|   strip.timebase = 0 - millis(); | ||||
| } | ||||
|  | ||||
|  | ||||
| void toggleOnOff() | ||||
| { | ||||
|   if (bri == 0) | ||||
|   { | ||||
|     bri = briLast; | ||||
|   } else | ||||
|   { | ||||
|     briLast = bri; | ||||
|     bri = 0; | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| void setAllLeds() { | ||||
|   if (!realtimeMode || !arlsForceMaxBri) | ||||
|   { | ||||
|     double d = briT*briMultiplier; | ||||
|     int val = d/100; | ||||
|     if (val > 255) val = 255; | ||||
|     strip.setBrightness(val); | ||||
|   } | ||||
|   if (!enableSecTransition) | ||||
|   { | ||||
|     for (byte i = 0; i<4; i++) | ||||
|     { | ||||
|       colSecT[i] = colSec[i]; | ||||
|     } | ||||
|   } | ||||
|   if (useRGBW && autoRGBtoRGBW) | ||||
|   { | ||||
|     colorRGBtoRGBW(colT); | ||||
|     colorRGBtoRGBW(colSecT); | ||||
|   } | ||||
|   strip.setColor(0, colT[0], colT[1], colT[2], colT[3]); | ||||
|   strip.setColor(1, colSecT[0], colSecT[1], colSecT[2], colSecT[3]); | ||||
| } | ||||
|  | ||||
|  | ||||
| void setLedsStandard(bool justColors = false) | ||||
| { | ||||
|   for (byte i=0; i<4; i++) | ||||
|   { | ||||
|     colOld[i] = col[i]; | ||||
|     colT[i] = col[i]; | ||||
|     colSecOld[i] = colSec[i]; | ||||
|     colSecT[i] = colSec[i]; | ||||
|   } | ||||
|   if (justColors) return; | ||||
|   briOld = bri; | ||||
|   briT = bri; | ||||
|   setAllLeds(); | ||||
| } | ||||
|  | ||||
|  | ||||
| bool colorChanged() | ||||
| { | ||||
|   for (byte i=0; i<4; i++) | ||||
|   { | ||||
|     if (col[i] != colIT[i]) return true; | ||||
|     if (colSec[i] != colSecIT[i]) return true; | ||||
|   } | ||||
|   if (bri != briIT) return true; | ||||
|   return false; | ||||
| } | ||||
|  | ||||
|  | ||||
| void colorUpdated(int callMode) | ||||
| { | ||||
|   //call for notifier -> 0: init 1: direct change 2: button 3: notification 4: nightlight 5: other (No notification) | ||||
|   //                     6: fx changed 7: hue 8: preset cycle 9: blynk 10: alexa | ||||
|   if (callMode != 0 && callMode != 1 && callMode != 5) strip.applyToAllSelected = true; //if not from JSON api, which directly sets segments | ||||
|    | ||||
|   bool fxChanged = strip.setEffectConfig(effectCurrent, effectSpeed, effectIntensity, effectPalette); | ||||
|   if (!colorChanged()) | ||||
|   { | ||||
|     if (nightlightActive && !nightlightActiveOld && callMode != 3 && callMode != 5) | ||||
|     { | ||||
|       notify(4); interfaceUpdateCallMode = 4; return; | ||||
|     } | ||||
|     else if (fxChanged) { | ||||
|       notify(6); | ||||
|       if (callMode != 8) interfaceUpdateCallMode = 6; | ||||
|       if (realtimeTimeout == UINT32_MAX) realtimeTimeout = 0; | ||||
|       if (isPreset) {isPreset = false;} | ||||
|           else {currentPreset = -1;} | ||||
|     } | ||||
|     return; //no change | ||||
|   } | ||||
|   if (realtimeTimeout == UINT32_MAX) realtimeTimeout = 0; | ||||
|   if (isPreset) {isPreset = false;} | ||||
|       else {currentPreset = -1;} | ||||
|   if (callMode != 5 && nightlightActive && nightlightFade) | ||||
|   { | ||||
|     briNlT = bri; | ||||
|     nightlightDelayMs -= (millis() - nightlightStartTime); | ||||
|     nightlightStartTime = millis(); | ||||
|   } | ||||
|   for (byte i=0; i<4; i++) | ||||
|   { | ||||
|     colIT[i] = col[i]; | ||||
|     colSecIT[i] = colSec[i]; | ||||
|   } | ||||
|   if (briT == 0) | ||||
|   { | ||||
|     setLedsStandard(true); //do not color transition if starting from off | ||||
|     if (callMode != 3) resetTimebase(); //effect start from beginning | ||||
|   } | ||||
|  | ||||
|   briIT = bri; | ||||
|   if (bri > 0) briLast = bri; | ||||
|    | ||||
|   notify(callMode); | ||||
|    | ||||
|   if (fadeTransition) | ||||
|   { | ||||
|     //set correct delay if not using notification delay | ||||
|     if (callMode != 3 && !jsonTransitionOnce) transitionDelayTemp = transitionDelay; | ||||
|     jsonTransitionOnce = false; | ||||
|     if (transitionDelayTemp == 0) {setLedsStandard(); strip.trigger(); return;} | ||||
|      | ||||
|     if (transitionActive) | ||||
|     { | ||||
|       for (byte i=0; i<4; i++) | ||||
|       { | ||||
|         colOld[i] = colT[i]; | ||||
|         colSecOld[i] = colSecT[i]; | ||||
|       } | ||||
|       briOld = briT; | ||||
|       tperLast = 0; | ||||
|     } | ||||
|     strip.setTransitionMode(true); | ||||
|     transitionActive = true; | ||||
|     transitionStartTime = millis(); | ||||
|   } else | ||||
|   { | ||||
|     setLedsStandard(); | ||||
|     strip.trigger(); | ||||
|   } | ||||
|  | ||||
|   if (callMode == 8) return; | ||||
|   //set flag to update blynk and mqtt | ||||
|   interfaceUpdateCallMode = callMode; | ||||
| } | ||||
|  | ||||
|  | ||||
| void updateInterfaces(uint8_t callMode) | ||||
| { | ||||
|   #ifndef WLED_DISABLE_ALEXA | ||||
|   if (espalexaDevice != nullptr && callMode != 10) { | ||||
|     espalexaDevice->setValue(bri); | ||||
|     espalexaDevice->setColor(col[0], col[1], col[2]); | ||||
|   } | ||||
|   #endif | ||||
|   if (callMode != 9 && callMode != 5) updateBlynk(); | ||||
|   doPublishMqtt = true; | ||||
|   lastInterfaceUpdate = millis(); | ||||
| } | ||||
|  | ||||
|  | ||||
| void handleTransitions() | ||||
| { | ||||
|   //handle still pending interface update | ||||
|   if (interfaceUpdateCallMode && millis() - lastInterfaceUpdate > 2000) | ||||
|   { | ||||
|     updateInterfaces(interfaceUpdateCallMode); | ||||
|     interfaceUpdateCallMode = 0; //disable | ||||
|   } | ||||
|   if (doPublishMqtt) publishMqtt(); | ||||
|    | ||||
|   if (transitionActive && transitionDelayTemp > 0) | ||||
|   { | ||||
|     float tper = (millis() - transitionStartTime)/(float)transitionDelayTemp; | ||||
|     if (tper >= 1.0) | ||||
|     { | ||||
|       strip.setTransitionMode(false); | ||||
|       transitionActive = false; | ||||
|       tperLast = 0; | ||||
|       setLedsStandard(); | ||||
|       return; | ||||
|     } | ||||
|     if (tper - tperLast < 0.004) return; | ||||
|     tperLast = tper; | ||||
|     for (byte i=0; i<4; i++) | ||||
|     { | ||||
|       colT[i] = colOld[i]+((col[i] - colOld[i])*tper); | ||||
|       colSecT[i] = colSecOld[i]+((colSec[i] - colSecOld[i])*tper); | ||||
|     } | ||||
|     briT    = briOld   +((bri    - briOld   )*tper); | ||||
|      | ||||
|     setAllLeds(); | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| void handleNightlight() | ||||
| { | ||||
|   if (nightlightActive) | ||||
|   { | ||||
|     if (!nightlightActiveOld) //init | ||||
|     { | ||||
|       nightlightStartTime = millis(); | ||||
|       nightlightDelayMs = (int)(nightlightDelayMins*60000); | ||||
|       nightlightActiveOld = true; | ||||
|       briNlT = bri; | ||||
|       for (byte i=0; i<4; i++) colNlT[i] = col[i];                                    // remember starting color | ||||
|     } | ||||
|     float nper = (millis() - nightlightStartTime)/((float)nightlightDelayMs); | ||||
|     if (nightlightFade) | ||||
|     { | ||||
|       bri = briNlT + ((nightlightTargetBri - briNlT)*nper); | ||||
|       if (nightlightColorFade)                                                        // color fading only is enabled with "NF=2" | ||||
|       { | ||||
|         for (byte i=0; i<4; i++) col[i] = colNlT[i]+ ((colSec[i] - colNlT[i])*nper);  // fading from actual color to secondary color | ||||
|       } | ||||
|       colorUpdated(5); | ||||
|     } | ||||
|     if (nper >= 1) | ||||
|     { | ||||
|       nightlightActive = false; | ||||
|       if (!nightlightFade) | ||||
|       { | ||||
|         bri = nightlightTargetBri; | ||||
|         colorUpdated(5); | ||||
|       } | ||||
|       updateBlynk(); | ||||
|       if (bri == 0) briLast = briNlT; | ||||
|     } | ||||
|   } else if (nightlightActiveOld) //early de-init | ||||
|   { | ||||
|     nightlightActiveOld = false; | ||||
|   } | ||||
|  | ||||
|   //also handle preset cycle here | ||||
|   if (presetCyclingEnabled && (millis() - presetCycledTime > presetCycleTime)) | ||||
|   { | ||||
|     applyPreset(presetCycCurr,presetApplyBri,presetApplyCol,presetApplyFx); | ||||
|     presetCycCurr++; if (presetCycCurr > presetCycleMax) presetCycCurr = presetCycleMin; | ||||
|     if (presetCycCurr > 25) presetCycCurr = 1; | ||||
|     colorUpdated(8); | ||||
|     presetCycledTime = millis(); | ||||
|   } | ||||
| } | ||||
| @@ -1,204 +0,0 @@ | ||||
| /* | ||||
|  * Acquires time from NTP server | ||||
|  */ | ||||
|  | ||||
| TimeChangeRule UTCr = {Last, Sun, Mar, 1, 0};     // UTC | ||||
| Timezone tzUTC(UTCr, UTCr); | ||||
|  | ||||
| TimeChangeRule BST = {Last, Sun, Mar, 1, 60};        // British Summer Time | ||||
| TimeChangeRule GMT = {Last, Sun, Oct, 2, 0};         // Standard Time | ||||
| Timezone tzUK(BST, GMT); | ||||
|  | ||||
| TimeChangeRule CEST = {Last, Sun, Mar, 2, 120};     //Central European Summer Time | ||||
| TimeChangeRule CET = {Last, Sun, Oct, 3, 60};       //Central European Standard Time | ||||
| Timezone tzEUCentral(CEST, CET); | ||||
|  | ||||
| TimeChangeRule EEST = {Last, Sun, Mar, 3, 180};     //Central European Summer Time | ||||
| TimeChangeRule EET = {Last, Sun, Oct, 4, 120};       //Central European Standard Time | ||||
| Timezone tzEUEastern(EEST, EET); | ||||
|  | ||||
| TimeChangeRule EDT = {Second, Sun, Mar, 2, -240 };    //Daylight time = UTC - 4 hours | ||||
| TimeChangeRule EST = {First, Sun, Nov, 2, -300 };     //Standard time = UTC - 5 hours | ||||
| Timezone tzUSEastern(EDT, EST); | ||||
|  | ||||
| TimeChangeRule CDT = {Second, Sun, Mar, 2, -300 };    //Daylight time = UTC - 5 hours | ||||
| TimeChangeRule CST = {First, Sun, Nov, 2, -360 };     //Standard time = UTC - 6 hours | ||||
| Timezone tzUSCentral(CDT, CST); | ||||
|  | ||||
| Timezone tzCASaskatchewan(CST, CST); //Central without DST | ||||
|  | ||||
| TimeChangeRule MDT = {Second, Sun, Mar, 2, -360 };    //Daylight time = UTC - 6 hours | ||||
| TimeChangeRule MST = {First, Sun, Nov, 2, -420 };     //Standard time = UTC - 7 hours | ||||
| Timezone tzUSMountain(MDT, MST); | ||||
|  | ||||
| Timezone tzUSArizona(MST, MST); //Mountain without DST | ||||
|  | ||||
| TimeChangeRule PDT = {Second, Sun, Mar, 2, -420 };    //Daylight time = UTC - 7 hours | ||||
| TimeChangeRule PST = {First, Sun, Nov, 2, -480 };     //Standard time = UTC - 8 hours | ||||
| Timezone tzUSPacific(PDT, PST); | ||||
|  | ||||
| TimeChangeRule ChST = {Last, Sun, Mar, 1, 480};     // China Standard Time = UTC + 8 hours | ||||
| Timezone tzChina(ChST, ChST); | ||||
|  | ||||
| TimeChangeRule JST = {Last, Sun, Mar, 1, 540};     // Japan Standard Time = UTC + 9 hours | ||||
| Timezone tzJapan(JST, JST); | ||||
|  | ||||
| TimeChangeRule AEDT = {Second, Sun, Oct, 2, 660 };    //Daylight time = UTC + 11 hours | ||||
| TimeChangeRule AEST = {First, Sun, Apr, 3, 600 };     //Standard time = UTC + 10 hours | ||||
| Timezone tzAUEastern(AEDT, AEST); | ||||
|  | ||||
| TimeChangeRule NZDT = {Second, Sun, Sep, 2, 780 };    //Daylight time = UTC + 13 hours | ||||
| TimeChangeRule NZST = {First, Sun, Apr, 3, 720 };     //Standard time = UTC + 12 hours | ||||
| Timezone tzNZ(NZDT, NZST); | ||||
|  | ||||
| TimeChangeRule NKST = {Last, Sun, Mar, 1, 510};     //Pyongyang Time = UTC + 8.5 hours | ||||
| Timezone tzNK(NKST, NKST); | ||||
|  | ||||
| TimeChangeRule IST = {Last, Sun, Mar, 1, 330};     // India Standard Time = UTC + 5.5 hours | ||||
| Timezone tzIndia(IST, IST); | ||||
|  | ||||
| Timezone* timezones[] = {&tzUTC, &tzUK, &tzEUCentral, &tzEUEastern, &tzUSEastern, &tzUSCentral, &tzUSMountain, &tzUSArizona, &tzUSPacific, &tzChina, &tzJapan, &tzAUEastern, &tzNZ, &tzNK, &tzIndia, &tzCASaskatchewan};   | ||||
|  | ||||
| void handleNetworkTime() | ||||
| { | ||||
|   if (ntpEnabled && ntpConnected && millis() - ntpLastSyncTime > 50000000L && WLED_CONNECTED) | ||||
|   { | ||||
|     if (millis() - ntpPacketSentTime > 10000) | ||||
|     { | ||||
|       sendNTPPacket(); | ||||
|       ntpPacketSentTime = millis(); | ||||
|     } | ||||
|     if (checkNTPResponse()) | ||||
|     { | ||||
|       ntpLastSyncTime = millis(); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| void sendNTPPacket() | ||||
| { | ||||
|   if (!ntpServerIP.fromString(ntpServerName)) //see if server is IP or domain | ||||
|   { | ||||
|     #ifdef ESP8266 | ||||
|     WiFi.hostByName(ntpServerName, ntpServerIP, 750); | ||||
|     #else | ||||
|     WiFi.hostByName(ntpServerName, ntpServerIP); | ||||
|     #endif | ||||
|   } | ||||
|  | ||||
|   DEBUG_PRINTLN("send NTP"); | ||||
|   byte pbuf[NTP_PACKET_SIZE]; | ||||
|   memset(pbuf, 0, NTP_PACKET_SIZE); | ||||
|  | ||||
|   pbuf[0] = 0b11100011;   // LI, Version, Mode | ||||
|   pbuf[1] = 0;     // Stratum, or type of clock | ||||
|   pbuf[2] = 6;     // Polling Interval | ||||
|   pbuf[3] = 0xEC;  // Peer Clock Precision | ||||
|   // 8 bytes of zero for Root Delay & Root Dispersion | ||||
|   pbuf[12]  = 49; | ||||
|   pbuf[13]  = 0x4E; | ||||
|   pbuf[14]  = 49; | ||||
|   pbuf[15]  = 52; | ||||
|  | ||||
|   ntpUdp.beginPacket(ntpServerIP, 123); //NTP requests are to port 123 | ||||
|   ntpUdp.write(pbuf, NTP_PACKET_SIZE); | ||||
|   ntpUdp.endPacket(); | ||||
| } | ||||
|  | ||||
| bool checkNTPResponse() | ||||
| { | ||||
|   int cb = ntpUdp.parsePacket(); | ||||
|   if (cb) { | ||||
|     DEBUG_PRINT("NTP recv, l="); | ||||
|     DEBUG_PRINTLN(cb); | ||||
|     byte pbuf[NTP_PACKET_SIZE]; | ||||
|     ntpUdp.read(pbuf, NTP_PACKET_SIZE); // read the packet into the buffer | ||||
|  | ||||
|     unsigned long highWord = word(pbuf[40], pbuf[41]); | ||||
|     unsigned long lowWord = word(pbuf[42], pbuf[43]); | ||||
|     if (highWord == 0 && lowWord == 0) return false; | ||||
|      | ||||
|     unsigned long secsSince1900 = highWord << 16 | lowWord; | ||||
|   | ||||
|     DEBUG_PRINT("Unix time = "); | ||||
|     unsigned long epoch = secsSince1900 - 2208988799UL; //subtract 70 years -1sec (on avg. more precision) | ||||
|     setTime(epoch); | ||||
|     DEBUG_PRINTLN(epoch); | ||||
|     if (countdownTime - now() > 0) countdownOverTriggered = false; | ||||
|     return true; | ||||
|   } | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| void updateLocalTime() | ||||
| { | ||||
|   unsigned long tmc = now()+ utcOffsetSecs; | ||||
|   local = timezones[currentTimezone]->toLocal(tmc); | ||||
| } | ||||
|  | ||||
| void getTimeString(char* out) | ||||
| { | ||||
|   updateLocalTime(); | ||||
|   byte hr = hour(local); | ||||
|   if (useAMPM) | ||||
|   { | ||||
|     if (hr > 11) hr -= 12; | ||||
|     if (hr == 0) hr  = 12; | ||||
|   } | ||||
|   sprintf(out,"%i-%i-%i, %i:%s%i:%s%i",year(local), month(local), day(local),  | ||||
|                                        hr,(minute(local)<10)?"0":"",minute(local), | ||||
|                                        (second(local)<10)?"0":"",second(local)); | ||||
|   if (useAMPM) | ||||
|   { | ||||
|     strcat(out,(hour(local) > 11)? " PM":" AM"); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void setCountdown() | ||||
| { | ||||
|   countdownTime = timezones[currentTimezone]->toUTC(getUnixTime(countdownHour, countdownMin, countdownSec, countdownDay, countdownMonth, countdownYear)); | ||||
|   if (countdownTime - now() > 0) countdownOverTriggered = false; | ||||
| } | ||||
|  | ||||
| //returns true if countdown just over | ||||
| bool checkCountdown() | ||||
| { | ||||
|   unsigned long n = now(); | ||||
|   local = countdownTime - n; | ||||
|   if (n > countdownTime) { | ||||
|     local = n - countdownTime; | ||||
|     if (!countdownOverTriggered) | ||||
|     { | ||||
|       if (macroCountdown != 0) applyMacro(macroCountdown); | ||||
|       countdownOverTriggered = true; | ||||
|       return true; | ||||
|     } | ||||
|   } | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| byte weekdayMondayFirst() | ||||
| { | ||||
|   byte wd = weekday(local) -1; | ||||
|   if (wd == 0) wd = 7; | ||||
|   return wd; | ||||
| } | ||||
|  | ||||
| void checkTimers() | ||||
| { | ||||
|   if (lastTimerMinute != minute(local)) //only check once a new minute begins | ||||
|   { | ||||
|     lastTimerMinute = minute(local); | ||||
|     for (uint8_t i = 0; i < 8; i++) | ||||
|     { | ||||
|       if (timerMacro[i] != 0 | ||||
|           && (timerHours[i] == hour(local) || timerHours[i] == 24) //if hour is set to 24, activate every hour  | ||||
|           && timerMinutes[i] == minute(local) | ||||
|           && (timerWeekday[i] & 0x01) //timer is enabled | ||||
|           && timerWeekday[i] >> weekdayMondayFirst() & 0x01) //timer should activate at current day of week | ||||
|       { | ||||
|         applyMacro(timerMacro[i]); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,125 +0,0 @@ | ||||
| /* | ||||
|  * Used to draw clock overlays over the strip | ||||
|  */ | ||||
|   | ||||
| void initCronixie() | ||||
| { | ||||
|   if (overlayCurrent == 3 && !cronixieInit) | ||||
|   { | ||||
|     strip.driverModeCronixie(true); | ||||
|     strip.setCronixieBacklight(cronixieBacklight); | ||||
|     setCronixie(); | ||||
|     cronixieInit = true; | ||||
|   } else if (cronixieInit && overlayCurrent != 3) | ||||
|   { | ||||
|     strip.driverModeCronixie(false); | ||||
|     cronixieInit = false;  | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| void handleOverlays() | ||||
| { | ||||
|   if (millis() - overlayRefreshedTime > overlayRefreshMs) | ||||
|   { | ||||
|     initCronixie(); | ||||
|     updateLocalTime(); | ||||
|     checkTimers(); | ||||
|     checkCountdown(); | ||||
|     if (overlayCurrent == 3) _overlayCronixie();//Diamex cronixie clock kit | ||||
|     overlayRefreshedTime = millis(); | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| void _overlayAnalogClock() | ||||
| { | ||||
|   int overlaySize = overlayMax - overlayMin +1; | ||||
|   if (countdownMode) | ||||
|   { | ||||
|     _overlayAnalogCountdown(); return; | ||||
|   } | ||||
|   double hourP = ((double)(hour(local)%12))/12; | ||||
|   double minuteP = ((double)minute(local))/60; | ||||
|   hourP = hourP + minuteP/12; | ||||
|   double secondP = ((double)second(local))/60; | ||||
|   int hourPixel = floor(analogClock12pixel + overlaySize*hourP); | ||||
|   if (hourPixel > overlayMax) hourPixel = overlayMin -1 + hourPixel - overlayMax; | ||||
|   int minutePixel = floor(analogClock12pixel + overlaySize*minuteP); | ||||
|   if (minutePixel > overlayMax) minutePixel = overlayMin -1 + minutePixel - overlayMax;  | ||||
|   int secondPixel = floor(analogClock12pixel + overlaySize*secondP); | ||||
|   if (secondPixel > overlayMax) secondPixel = overlayMin -1 + secondPixel - overlayMax; | ||||
|   if (analogClockSecondsTrail) | ||||
|   { | ||||
|     if (secondPixel < analogClock12pixel) | ||||
|     { | ||||
|       strip.setRange(analogClock12pixel, overlayMax, 0xFF0000); | ||||
|       strip.setRange(overlayMin, secondPixel, 0xFF0000); | ||||
|     } else | ||||
|     { | ||||
|       strip.setRange(analogClock12pixel, secondPixel, 0xFF0000); | ||||
|     } | ||||
|   } | ||||
|   if (analogClock5MinuteMarks) | ||||
|   { | ||||
|     int pix; | ||||
|     for (int i = 0; i <= 12; i++) | ||||
|     { | ||||
|       pix = analogClock12pixel + round((overlaySize / 12.0) *i); | ||||
|       if (pix > overlayMax) pix -= overlaySize; | ||||
|       strip.setPixelColor(pix, 0x00FFAA); | ||||
|     } | ||||
|   } | ||||
|   if (!analogClockSecondsTrail) strip.setPixelColor(secondPixel, 0xFF0000); | ||||
|   strip.setPixelColor(minutePixel, 0x00FF00); | ||||
|   strip.setPixelColor(hourPixel, 0x0000FF); | ||||
|   overlayRefreshMs = 998; | ||||
| } | ||||
|  | ||||
|  | ||||
| void _overlayAnalogCountdown() | ||||
| { | ||||
|   if (now() < countdownTime) | ||||
|   { | ||||
|     long diff = countdownTime - now(); | ||||
|     double pval = 60; | ||||
|     if (diff > 31557600L) //display in years if more than 365 days | ||||
|     { | ||||
|       pval = 315576000L; //10 years | ||||
|     } else if (diff > 2592000L) //display in months if more than a month | ||||
|     { | ||||
|       pval = 31557600L; //1 year | ||||
|     } else if (diff > 604800) //display in weeks if more than a week | ||||
|     { | ||||
|       pval = 2592000L; //1 month | ||||
|     } else if (diff > 86400) //display in days if more than 24 hours | ||||
|     { | ||||
|       pval = 604800; //1 week | ||||
|     } else if (diff > 3600) //display in hours if more than 60 minutes | ||||
|     { | ||||
|       pval = 86400; //1 day | ||||
|     } else if (diff > 60) //display in minutes if more than 60 seconds | ||||
|     { | ||||
|       pval = 3600; //1 hour | ||||
|     } | ||||
|     int overlaySize = overlayMax - overlayMin +1; | ||||
|     double perc = (pval-(double)diff)/pval; | ||||
|     if (perc > 1.0) perc = 1.0; | ||||
|     byte pixelCnt = perc*overlaySize; | ||||
|     if (analogClock12pixel + pixelCnt > overlayMax) | ||||
|     { | ||||
|       strip.setRange(analogClock12pixel, overlayMax, ((uint32_t)colSec[3] << 24)| ((uint32_t)colSec[0] << 16) | ((uint32_t)colSec[1] << 8) | colSec[2]); | ||||
|       strip.setRange(overlayMin, overlayMin +pixelCnt -(1+ overlayMax -analogClock12pixel), ((uint32_t)colSec[3] << 24)| ((uint32_t)colSec[0] << 16) | ((uint32_t)colSec[1] << 8) | colSec[2]); | ||||
|     } else | ||||
|     { | ||||
|       strip.setRange(analogClock12pixel, analogClock12pixel + pixelCnt, ((uint32_t)colSec[3] << 24)| ((uint32_t)colSec[0] << 16) | ((uint32_t)colSec[1] << 8) | colSec[2]); | ||||
|     } | ||||
|   } | ||||
|   overlayRefreshMs = 998; | ||||
| } | ||||
|  | ||||
|  | ||||
| void handleOverlayDraw() { | ||||
|   if (overlayCurrent != 1) return; //only analog clock | ||||
|   _overlayAnalogClock(); | ||||
| } | ||||
| @@ -1,381 +0,0 @@ | ||||
| /* | ||||
|  * JSON API (De)serialization | ||||
|  */ | ||||
|  | ||||
| void deserializeSegment(JsonObject elem, byte it) | ||||
| { | ||||
|   byte id = elem["id"] | it; | ||||
|   if (id < strip.getMaxSegments()) | ||||
|   { | ||||
|     WS2812FX::Segment& seg = strip.getSegment(id); | ||||
|     uint16_t start = elem["start"] | seg.start; | ||||
|     int stop = elem["stop"] | -1; | ||||
|  | ||||
|     if (stop < 0) { | ||||
|       uint16_t len = elem["len"]; | ||||
|       stop = (len > 0) ? start + len : seg.stop; | ||||
|     } | ||||
|     uint16_t grp = elem["grp"] | seg.grouping; | ||||
|     uint16_t spc = elem["spc"] | seg.spacing; | ||||
|     strip.setSegment(id, start, stop, grp, spc); | ||||
|      | ||||
|     JsonArray colarr = elem["col"]; | ||||
|     if (!colarr.isNull()) | ||||
|     { | ||||
|       for (uint8_t i = 0; i < 3; i++) | ||||
|       { | ||||
|         JsonArray colX = colarr[i]; | ||||
|         if (colX.isNull()) break; | ||||
|         byte sz = colX.size(); | ||||
|         if (sz > 0 && sz < 5) | ||||
|         { | ||||
|           int rgbw[] = {0,0,0,0}; | ||||
|           byte cp = copyArray(colX, rgbw); | ||||
|           seg.colors[i] = ((rgbw[3] << 24) | ((rgbw[0]&0xFF) << 16) | ((rgbw[1]&0xFF) << 8) | ((rgbw[2]&0xFF))); | ||||
|           if (cp == 1 && rgbw[0] == 0) seg.colors[i] = 0; | ||||
|           if (id == strip.getMainSegmentId()) //temporary | ||||
|           {  | ||||
|             if (i == 0) {col[0] = rgbw[0]; col[1] = rgbw[1]; col[2] = rgbw[2]; col[3] = rgbw[3];} | ||||
|             if (i == 1) {colSec[0] = rgbw[0]; colSec[1] = rgbw[1]; colSec[2] = rgbw[2]; colSec[3] = rgbw[3];} | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|      | ||||
|     //if (pal != seg.palette && pal < strip.getPaletteCount()) strip.setPalette(pal); | ||||
|     seg.setOption(0, elem["sel"] | seg.getOption(0)); //selected | ||||
|     seg.setOption(1, elem["rev"] | seg.getOption(1)); //reverse | ||||
|     //int cln = seg_0["cln"]; | ||||
|     //temporary, strip object gets updated via colorUpdated() | ||||
|     if (id == strip.getMainSegmentId()) { | ||||
|       effectCurrent = elem["fx"] | effectCurrent; | ||||
|       effectSpeed = elem["sx"] | effectSpeed; | ||||
|       effectIntensity = elem["ix"] | effectIntensity; | ||||
|       effectPalette = elem["pal"] | effectPalette; | ||||
|     } else { //permanent | ||||
|       byte fx = elem["fx"] | seg.mode; | ||||
|       if (fx != seg.mode && fx < strip.getModeCount()) strip.setMode(id, fx); | ||||
|       seg.speed = elem["sx"] | seg.speed; | ||||
|       seg.intensity = elem["ix"] | seg.intensity; | ||||
|       seg.palette = elem["pal"] | seg.palette; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| bool deserializeState(JsonObject root) | ||||
| { | ||||
|   strip.applyToAllSelected = false; | ||||
|   bool stateResponse = root["v"] | false; | ||||
|  | ||||
|   int ps = root["ps"] | -1; | ||||
|   if (ps >= 0) applyPreset(ps); | ||||
|    | ||||
|   bri = root["bri"] | bri; | ||||
|    | ||||
|   bool on = root["on"] | (bri > 0); | ||||
|   if (!on != !bri) toggleOnOff(); | ||||
|  | ||||
|   int tr = root["transition"] | -1; | ||||
|   if (tr >= 0) | ||||
|   { | ||||
|     transitionDelay = tr; | ||||
|     transitionDelay *= 100; | ||||
|   } | ||||
|  | ||||
|   tr = root["tt"] | -1; | ||||
|   if (tr >= 0) | ||||
|   { | ||||
|     transitionDelayTemp = tr; | ||||
|     transitionDelayTemp *= 100; | ||||
|     jsonTransitionOnce = true; | ||||
|   } | ||||
|    | ||||
|   int cy = root["pl"] | -2; | ||||
|   if (cy > -2) presetCyclingEnabled = (cy >= 0); | ||||
|   JsonObject ccnf = root["ccnf"]; | ||||
|   presetCycleMin = ccnf["min"] | presetCycleMin; | ||||
|   presetCycleMax = ccnf["max"] | presetCycleMax; | ||||
|   tr = ccnf["time"] | -1; | ||||
|   if (tr >= 2) | ||||
|   { | ||||
|     presetCycleTime = tr; | ||||
|     presetCycleTime *= 100; | ||||
|   } | ||||
|  | ||||
|   JsonObject nl = root["nl"]; | ||||
|   nightlightActive    = nl["on"]   | nightlightActive; | ||||
|   nightlightDelayMins = nl["dur"]  | nightlightDelayMins; | ||||
|   nightlightFade      = nl["fade"] | nightlightFade; | ||||
|   nightlightTargetBri = nl["tbri"] | nightlightTargetBri; | ||||
|  | ||||
|   JsonObject udpn = root["udpn"]; | ||||
|   notifyDirect         = udpn["send"] | notifyDirect; | ||||
|   receiveNotifications = udpn["recv"] | receiveNotifications; | ||||
|   bool noNotification  = udpn["nn"]; //send no notification just for this request | ||||
|  | ||||
|   int timein = root["time"] | -1; | ||||
|   if (timein != -1) setTime(timein); | ||||
|  | ||||
|   byte prevMain = strip.getMainSegmentId(); | ||||
|   strip.mainSegment = root["mainseg"] | prevMain; | ||||
|   if (strip.getMainSegmentId() != prevMain) setValuesFromMainSeg(); | ||||
|  | ||||
|   int it = 0; | ||||
|   JsonVariant segVar = root["seg"]; | ||||
|   if (segVar.is<JsonObject>()) | ||||
|   { | ||||
|     int id = segVar["id"] | -1; | ||||
|      | ||||
|     if (id < 0) { //set all selected segments | ||||
|       bool didSet = false; | ||||
|       byte lowestActive = 99; | ||||
|       for (byte s = 0; s < strip.getMaxSegments(); s++) | ||||
|       { | ||||
|         WS2812FX::Segment sg = strip.getSegment(s); | ||||
|         if (sg.isActive()) | ||||
|         { | ||||
|           if (lowestActive == 99) lowestActive = s; | ||||
|           if (sg.isSelected()) { | ||||
|             deserializeSegment(segVar, s); | ||||
|             didSet = true; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|       if (!didSet && lowestActive < strip.getMaxSegments()) deserializeSegment(segVar, lowestActive); | ||||
|     } else { //set only the segment with the specified ID | ||||
|       deserializeSegment(segVar, it); | ||||
|     } | ||||
|   } else { | ||||
|     JsonArray segs = segVar.as<JsonArray>(); | ||||
|     for (JsonObject elem : segs) | ||||
|     { | ||||
|       deserializeSegment(elem, it); | ||||
|       it++; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   colorUpdated(noNotification ? 5:1); | ||||
|  | ||||
|   ps = root["psave"] | -1; | ||||
|   if (ps >= 0) savePreset(ps); | ||||
|  | ||||
|   return stateResponse; | ||||
| } | ||||
|  | ||||
| void serializeSegment(JsonObject& root, WS2812FX::Segment& seg, byte id) | ||||
| { | ||||
| 	root["id"] = id; | ||||
| 	root["start"] = seg.start; | ||||
| 	root["stop"] = seg.stop; | ||||
| 	root["len"] = seg.stop - seg.start; | ||||
|   root["grp"] = seg.grouping; | ||||
|   root["spc"] = seg.spacing; | ||||
|  | ||||
| 	JsonArray colarr = root.createNestedArray("col"); | ||||
|  | ||||
| 	for (uint8_t i = 0; i < 3; i++) | ||||
| 	{ | ||||
| 		JsonArray colX = colarr.createNestedArray(); | ||||
| 		colX.add((seg.colors[i] >> 16) & 0xFF); | ||||
| 		colX.add((seg.colors[i] >> 8) & 0xFF); | ||||
| 		colX.add((seg.colors[i]) & 0xFF); | ||||
| 		if (useRGBW) | ||||
| 			colX.add((seg.colors[i] >> 24) & 0xFF); | ||||
| 	} | ||||
|  | ||||
| 	root["fx"] = seg.mode; | ||||
| 	root["sx"] = seg.speed; | ||||
| 	root["ix"] = seg.intensity; | ||||
| 	root["pal"] = seg.palette; | ||||
| 	root["sel"] = seg.isSelected(); | ||||
| 	root["rev"] = seg.getOption(1); | ||||
| } | ||||
|  | ||||
|  | ||||
| void serializeState(JsonObject root) | ||||
| { | ||||
|   if (errorFlag) root["error"] = errorFlag; | ||||
|    | ||||
|   root["on"] = (bri > 0); | ||||
|   root["bri"] = briLast; | ||||
|   root["transition"] = transitionDelay/100; //in 100ms | ||||
|  | ||||
|   root["ps"] = currentPreset; | ||||
|   root["pss"] = savedPresets; | ||||
|   root["pl"] = (presetCyclingEnabled) ? 0: -1; | ||||
|  | ||||
|   //temporary for preser cycle | ||||
|   JsonObject ccnf = root.createNestedObject("ccnf"); | ||||
|   ccnf["min"] = presetCycleMin; | ||||
|   ccnf["max"] = presetCycleMax; | ||||
|   ccnf["time"] = presetCycleTime/100; | ||||
|    | ||||
|   JsonObject nl = root.createNestedObject("nl"); | ||||
|   nl["on"] = nightlightActive; | ||||
|   nl["dur"] = nightlightDelayMins; | ||||
|   nl["fade"] = nightlightFade; | ||||
|   nl["tbri"] = nightlightTargetBri; | ||||
|    | ||||
|   JsonObject udpn = root.createNestedObject("udpn"); | ||||
|   udpn["send"] = notifyDirect; | ||||
|   udpn["recv"] = receiveNotifications; | ||||
|  | ||||
|   root["mainseg"] = strip.getMainSegmentId(); | ||||
|    | ||||
|   JsonArray seg = root.createNestedArray("seg"); | ||||
|   for (byte s = 0; s < strip.getMaxSegments(); s++) | ||||
|   { | ||||
|     WS2812FX::Segment sg = strip.getSegment(s); | ||||
|     if (sg.isActive()) | ||||
|     { | ||||
|       JsonObject seg0 = seg.createNestedObject(); | ||||
|       serializeSegment(seg0, sg, s); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| void serializeInfo(JsonObject root) | ||||
| { | ||||
|   root["ver"] = versionString; | ||||
|   root["vid"] = VERSION; | ||||
|    | ||||
|   JsonObject leds = root.createNestedObject("leds"); | ||||
|   leds["count"] = ledCount; | ||||
|   leds["rgbw"] = useRGBW; | ||||
|   leds["wv"] = useRGBW && !autoRGBtoRGBW; //should a white channel slider be displayed? | ||||
|   JsonArray leds_pin = leds.createNestedArray("pin"); | ||||
|   leds_pin.add(LEDPIN); | ||||
|    | ||||
|   leds["pwr"] = strip.currentMilliamps; | ||||
|   leds["maxpwr"] = (strip.currentMilliamps)? strip.ablMilliampsMax : 0; | ||||
|   leds["maxseg"] = strip.getMaxSegments(); | ||||
|   leds["seglock"] = false; //will be used in the future to prevent modifications to segment config | ||||
|  | ||||
|   root["str"] = syncToggleReceive; | ||||
|    | ||||
|   root["name"] = serverDescription; | ||||
|   root["udpport"] = udpPort; | ||||
|   root["live"] = (bool)realtimeMode; | ||||
|   root["fxcount"] = strip.getModeCount(); | ||||
|   root["palcount"] = strip.getPaletteCount(); | ||||
|  | ||||
|   JsonObject wifi_info = root.createNestedObject("wifi"); | ||||
|   wifi_info["bssid"] = WiFi.BSSIDstr(); | ||||
|   int qrssi = WiFi.RSSI(); | ||||
|   wifi_info["rssi"] = qrssi; | ||||
|   wifi_info["signal"] = getSignalQuality(qrssi); | ||||
|   wifi_info["channel"] = WiFi.channel(); | ||||
|    | ||||
|   #ifdef ARDUINO_ARCH_ESP32 | ||||
|   root["arch"] = "esp32"; | ||||
|   root["core"] = ESP.getSdkVersion(); | ||||
|   //root["maxalloc"] = ESP.getMaxAllocHeap(); | ||||
|   root["lwip"] = 0; | ||||
|   #else | ||||
|   root["arch"] = "esp8266"; | ||||
|   root["core"] = ESP.getCoreVersion(); | ||||
|   //root["maxalloc"] = ESP.getMaxFreeBlockSize(); | ||||
|   root["lwip"] = LWIP_VERSION_MAJOR; | ||||
|   #endif | ||||
|    | ||||
|   root["freeheap"] = ESP.getFreeHeap(); | ||||
|   root["uptime"] = millis()/1000; | ||||
|    | ||||
|   byte os = 0; | ||||
|   #ifdef WLED_DEBUG | ||||
|   os  = 0x80; | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_ALEXA | ||||
|   os += 0x40; | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_BLYNK | ||||
|   os += 0x20; | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_CRONIXIE | ||||
|   os += 0x10; | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_FILESYSTEM | ||||
|   os += 0x08; | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_HUESYNC | ||||
|   os += 0x04; | ||||
|   #endif | ||||
|   #ifdef WLED_ENABLE_ADALIGHT | ||||
|   os += 0x02; | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_OTA  | ||||
|   os += 0x01; | ||||
|   #endif | ||||
|   root["opt"] = os; | ||||
|    | ||||
|   root["brand"] = "WLED"; | ||||
|   root["product"] = "DIY light"; | ||||
|   root["btype"] = "src"; | ||||
|   root["mac"] = escapedMac; | ||||
| } | ||||
|  | ||||
| void serveJson(AsyncWebServerRequest* request) | ||||
| { | ||||
|   byte subJson = 0; | ||||
|   const String& url = request->url(); | ||||
|   if      (url.indexOf("state") > 0) subJson = 1; | ||||
|   else if (url.indexOf("info")  > 0) subJson = 2; | ||||
|   else if (url.indexOf("live")  > 0) { | ||||
|     serveLiveLeds(request); | ||||
|     return; | ||||
|   } | ||||
|   else if (url.indexOf("eff")   > 0) { | ||||
|     request->send_P(200, "application/json", JSON_mode_names); | ||||
|     return; | ||||
|   } | ||||
|   else if (url.indexOf("pal")   > 0) { | ||||
|     request->send_P(200, "application/json", JSON_palette_names); | ||||
|     return; | ||||
|   } | ||||
|   else if (url.length() > 6) { //not just /json | ||||
|     request->send(  501, "application/json", "{\"error\":\"Not implemented\"}"); | ||||
|     return; | ||||
|   } | ||||
|    | ||||
|   AsyncJsonResponse* response = new AsyncJsonResponse(); | ||||
|   JsonObject doc = response->getRoot(); | ||||
|  | ||||
|   switch (subJson) | ||||
|   { | ||||
|     case 1: //state | ||||
|       serializeState(doc); break; | ||||
|     case 2: //info | ||||
|       serializeInfo(doc); break; | ||||
|     default: //all | ||||
|       JsonObject state = doc.createNestedObject("state"); | ||||
|       serializeState(state); | ||||
|       JsonObject info  = doc.createNestedObject("info"); | ||||
|       serializeInfo(info); | ||||
|       doc["effects"]  = serialized((const __FlashStringHelper*)JSON_mode_names); | ||||
|       doc["palettes"] = serialized((const __FlashStringHelper*)JSON_palette_names); | ||||
|   } | ||||
|    | ||||
|   response->setLength(); | ||||
|   request->send(response); | ||||
| } | ||||
|  | ||||
| #define MAX_LIVE_LEDS 180 | ||||
|  | ||||
| void serveLiveLeds(AsyncWebServerRequest* request) | ||||
| { | ||||
|   byte used = ledCount; | ||||
|   byte n = (used -1) /MAX_LIVE_LEDS +1; //only serve every n'th LED if count over MAX_LIVE_LEDS | ||||
|   char buffer[2000] = "{\"leds\":["; | ||||
|   olen = 9; | ||||
|   obuf = buffer; | ||||
|  | ||||
|   for (uint16_t i= 0; i < used; i += n) | ||||
|   { | ||||
|     olen += sprintf(buffer + olen, "\"%06X\",", strip.getPixelColor(i)); | ||||
|   } | ||||
|   olen -= 1; | ||||
|   oappend("],\"n\":"); | ||||
|   oappendi(n); | ||||
|   oappend("}"); | ||||
|   request->send(200, "application/json", buffer); | ||||
| } | ||||
							
								
								
									
										477
									
								
								wled00/wled_eeprom.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										477
									
								
								wled00/wled_eeprom.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,477 @@ | ||||
| #include <EEPROM.h> | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * DEPRECATED, do not use for new settings | ||||
|  * Only used to restore config from pre-0.11 installations using the deEEP() methods | ||||
|  *  | ||||
|  * Methods to handle saving and loading to non-volatile memory | ||||
|  * EEPROM Map: https://github.com/Aircoookie/WLED/wiki/EEPROM-Map | ||||
|  */ | ||||
|  | ||||
| //eeprom Version code, enables default settings instead of 0 init on update | ||||
| #define EEPVER 22 | ||||
| #define EEPSIZE 2560  //Maximum is 4096 | ||||
| //0 -> old version, default | ||||
| //1 -> 0.4p 1711272 and up | ||||
| //2 -> 0.4p 1711302 and up | ||||
| //3 -> 0.4  1712121 and up | ||||
| //4 -> 0.5.0 and up | ||||
| //5 -> 0.5.1 and up | ||||
| //6 -> 0.6.0 and up | ||||
| //7 -> 0.7.1 and up | ||||
| //8 -> 0.8.0-a and up | ||||
| //9 -> 0.8.0 | ||||
| //10-> 0.8.2 | ||||
| //11-> 0.8.5-dev #mqttauth @TimothyBrown | ||||
| //12-> 0.8.7-dev | ||||
| //13-> 0.9.0-dev | ||||
| //14-> 0.9.0-b1 | ||||
| //15-> 0.9.0-b3 | ||||
| //16-> 0.9.1 | ||||
| //17-> 0.9.1-dmx | ||||
| //18-> 0.9.1-e131 | ||||
| //19-> 0.9.1n | ||||
| //20-> 0.9.1p | ||||
| //21-> 0.10.1p | ||||
| //22-> 2009260 | ||||
|  | ||||
| /* | ||||
|  * Erase all (pre 0.11) configuration data on factory reset | ||||
|  */ | ||||
| void clearEEPROM() | ||||
| { | ||||
|   EEPROM.begin(EEPSIZE); | ||||
|   for (int i = 0; i < EEPSIZE; i++) | ||||
|   { | ||||
|     EEPROM.write(i, 0); | ||||
|   } | ||||
|   EEPROM.end(); | ||||
| } | ||||
|  | ||||
|  | ||||
| void readStringFromEEPROM(uint16_t pos, char* str, uint16_t len) | ||||
| { | ||||
|   for (int i = 0; i < len; ++i) | ||||
|   { | ||||
|     str[i] = EEPROM.read(pos + i); | ||||
|     if (str[i] == 0) return; | ||||
|   } | ||||
|   str[len] = 0; //make sure every string is properly terminated. str must be at least len +1 big. | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Read all configuration from flash | ||||
|  */ | ||||
| void loadSettingsFromEEPROM() | ||||
| { | ||||
|   if (EEPROM.read(233) != 233) //first boot/reset to default | ||||
|   { | ||||
|     DEBUG_PRINTLN(F("EEPROM settings invalid, using defaults...")); | ||||
|     return; | ||||
|   } | ||||
|   int lastEEPROMversion = EEPROM.read(377); //last EEPROM version before update | ||||
|  | ||||
|  | ||||
|   readStringFromEEPROM(  0, clientSSID, 32); | ||||
|   readStringFromEEPROM( 32, clientPass, 64); | ||||
|   readStringFromEEPROM( 96,      cmDNS, 32); | ||||
|   readStringFromEEPROM(128,     apSSID, 32); | ||||
|   readStringFromEEPROM(160,     apPass, 64); | ||||
|  | ||||
|   nightlightDelayMinsDefault = EEPROM.read(224); | ||||
|   nightlightDelayMins = nightlightDelayMinsDefault; | ||||
|   nightlightMode = EEPROM.read(225); | ||||
|   notifyDirectDefault = EEPROM.read(226); | ||||
|   notifyDirect = notifyDirectDefault; | ||||
|  | ||||
|   apChannel = EEPROM.read(227); | ||||
|   if (apChannel > 13 || apChannel < 1) apChannel = 1; | ||||
|   apHide = EEPROM.read(228); | ||||
|   if (apHide > 1) apHide = 1; | ||||
|   ledCount = EEPROM.read(229) + ((EEPROM.read(398) << 8) & 0xFF00); if (ledCount > MAX_LEDS || ledCount == 0) ledCount = 30; | ||||
|  | ||||
|   notifyButton = EEPROM.read(230); | ||||
|   notifyTwice = EEPROM.read(231); | ||||
|   buttonEnabled = EEPROM.read(232); | ||||
|  | ||||
|   staticIP[0] = EEPROM.read(234); | ||||
|   staticIP[1] = EEPROM.read(235); | ||||
|   staticIP[2] = EEPROM.read(236); | ||||
|   staticIP[3] = EEPROM.read(237); | ||||
|   staticGateway[0] = EEPROM.read(238); | ||||
|   staticGateway[1] = EEPROM.read(239); | ||||
|   staticGateway[2] = EEPROM.read(240); | ||||
|   staticGateway[3] = EEPROM.read(241); | ||||
|   staticSubnet[0] = EEPROM.read(242); | ||||
|   staticSubnet[1] = EEPROM.read(243); | ||||
|   staticSubnet[2] = EEPROM.read(244); | ||||
|   staticSubnet[3] = EEPROM.read(245); | ||||
|  | ||||
|   briS = EEPROM.read(249); bri = briS; | ||||
|   if (!EEPROM.read(369)) | ||||
|   { | ||||
|     bri = 0; briLast = briS; | ||||
|   } | ||||
|   receiveNotificationBrightness = EEPROM.read(250); | ||||
|   fadeTransition = EEPROM.read(251); | ||||
|   transitionDelayDefault = EEPROM.read(253) + ((EEPROM.read(254) << 8) & 0xFF00); | ||||
|   transitionDelay = transitionDelayDefault; | ||||
|   briMultiplier = EEPROM.read(255); | ||||
|  | ||||
|   readStringFromEEPROM(256, otaPass, 32); | ||||
|  | ||||
|   nightlightTargetBri = EEPROM.read(288); | ||||
|   otaLock = EEPROM.read(289); | ||||
|   udpPort = EEPROM.read(290) + ((EEPROM.read(291) << 8) & 0xFF00); | ||||
|  | ||||
|   readStringFromEEPROM(292, serverDescription, 32); | ||||
|  | ||||
|   ntpEnabled = EEPROM.read(327); | ||||
|   currentTimezone = EEPROM.read(328); | ||||
|   useAMPM = EEPROM.read(329); | ||||
|   strip.gammaCorrectBri = EEPROM.read(330); | ||||
|   strip.gammaCorrectCol = EEPROM.read(331); | ||||
|   overlayDefault = EEPROM.read(332); | ||||
|   if (lastEEPROMversion < 8 && overlayDefault > 0) overlayDefault--; //overlay mode 1 (solid) was removed | ||||
|  | ||||
|   alexaEnabled = EEPROM.read(333); | ||||
|  | ||||
|   readStringFromEEPROM(334, alexaInvocationName, 32); | ||||
|  | ||||
|   notifyAlexa = EEPROM.read(366); | ||||
|   arlsOffset = EEPROM.read(368); | ||||
|   if (!EEPROM.read(367)) arlsOffset = -arlsOffset; | ||||
|   turnOnAtBoot = EEPROM.read(369); | ||||
|   strip.isRgbw = EEPROM.read(372); | ||||
|   //374 - strip.paletteFade | ||||
|    | ||||
|   apBehavior = EEPROM.read(376); | ||||
|      | ||||
|   //377 = lastEEPROMversion | ||||
|   if (lastEEPROMversion > 3) { | ||||
|     aOtaEnabled = EEPROM.read(390); | ||||
|     receiveNotificationColor = EEPROM.read(391); | ||||
|     receiveNotificationEffects = EEPROM.read(392); | ||||
|   } | ||||
|   receiveNotifications = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects); | ||||
|    | ||||
|   if (lastEEPROMversion > 4) { | ||||
|     huePollingEnabled = EEPROM.read(2048); | ||||
|     //hueUpdatingEnabled = EEPROM.read(2049); | ||||
|     for (int i = 2050; i < 2054; ++i) | ||||
|     { | ||||
|       hueIP[i-2050] = EEPROM.read(i); | ||||
|     } | ||||
|  | ||||
|     readStringFromEEPROM(2054, hueApiKey, 46); | ||||
|  | ||||
|     huePollIntervalMs = EEPROM.read(2100) + ((EEPROM.read(2101) << 8) & 0xFF00); | ||||
|     notifyHue = EEPROM.read(2102); | ||||
|     hueApplyOnOff = EEPROM.read(2103); | ||||
|     hueApplyBri = EEPROM.read(2104); | ||||
|     hueApplyColor = EEPROM.read(2105); | ||||
|     huePollLightId = EEPROM.read(2106); | ||||
|   } | ||||
|   if (lastEEPROMversion > 5) { | ||||
|     overlayMin = EEPROM.read(2150); | ||||
|     overlayMax = EEPROM.read(2151); | ||||
|     analogClock12pixel = EEPROM.read(2152); | ||||
|     analogClock5MinuteMarks = EEPROM.read(2153); | ||||
|     analogClockSecondsTrail = EEPROM.read(2154); | ||||
|     countdownMode = EEPROM.read(2155); | ||||
|     countdownYear = EEPROM.read(2156); | ||||
|     countdownMonth = EEPROM.read(2157); | ||||
|     countdownDay = EEPROM.read(2158); | ||||
|     countdownHour = EEPROM.read(2159); | ||||
|     countdownMin = EEPROM.read(2160); | ||||
|     countdownSec = EEPROM.read(2161); | ||||
|     setCountdown(); | ||||
|  | ||||
|     readStringFromEEPROM(2165, cronixieDisplay, 6); | ||||
|     cronixieBacklight = EEPROM.read(2171); | ||||
|  | ||||
|     //macroBoot = EEPROM.read(2175); | ||||
|     macroAlexaOn = EEPROM.read(2176); | ||||
|     macroAlexaOff = EEPROM.read(2177); | ||||
|     macroButton = EEPROM.read(2178); | ||||
|     macroLongPress = EEPROM.read(2179); | ||||
|     macroCountdown = EEPROM.read(2180); | ||||
|     macroNl = EEPROM.read(2181); | ||||
|     macroDoublePress = EEPROM.read(2182); | ||||
|     if (macroDoublePress > 16) macroDoublePress = 0; | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 6) | ||||
|   { | ||||
|     e131Universe = EEPROM.read(2190) + ((EEPROM.read(2191) << 8) & 0xFF00); | ||||
|     e131Multicast = EEPROM.read(2192); | ||||
|     realtimeTimeoutMs = EEPROM.read(2193) + ((EEPROM.read(2194) << 8) & 0xFF00); | ||||
|     arlsForceMaxBri = EEPROM.read(2195); | ||||
|     arlsDisableGammaCorrection = EEPROM.read(2196); | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 7) | ||||
|   { | ||||
|     strip.paletteFade  = EEPROM.read(374); | ||||
|     strip.paletteBlend = EEPROM.read(382); | ||||
|  | ||||
|     for (int i = 0; i < 8; ++i) | ||||
|     { | ||||
|       timerHours[i]   = EEPROM.read(2260 + i); | ||||
|       timerMinutes[i] = EEPROM.read(2270 + i); | ||||
|       timerWeekday[i] = EEPROM.read(2280 + i); | ||||
|       timerMacro[i]   = EEPROM.read(2290 + i); | ||||
|       if (timerMacro[i] > 0) timerMacro[i] += 16; //add 16 to work with macro --> preset mapping | ||||
|       if (timerWeekday[i] == 0) timerWeekday[i] = 255; | ||||
|       if (timerMacro[i] == 0) timerWeekday[i] = timerWeekday[i] & 0b11111110;  | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 8) | ||||
|   { | ||||
|     readStringFromEEPROM(2300, mqttServer, 32); | ||||
|     readStringFromEEPROM(2333, mqttDeviceTopic, 32); | ||||
|     readStringFromEEPROM(2366, mqttGroupTopic, 32); | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 9) | ||||
|   { | ||||
|     strip.setColorOrder(EEPROM.read(383)); | ||||
|     irEnabled = EEPROM.read(385); | ||||
|     strip.ablMilliampsMax = EEPROM.read(387) + ((EEPROM.read(388) << 8) & 0xFF00); | ||||
|   } else if (lastEEPROMversion > 1) //ABL is off by default when updating from version older than 0.8.2 | ||||
|   { | ||||
|     strip.ablMilliampsMax = 65000; | ||||
|   } else { | ||||
|     strip.ablMilliampsMax = ABL_MILLIAMPS_DEFAULT; | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 10) | ||||
|   { | ||||
|     readStringFromEEPROM(2399, mqttUser, 40); | ||||
|     readStringFromEEPROM(2440, mqttPass, 40); | ||||
|     readStringFromEEPROM(2481, mqttClientID, 40); | ||||
|     mqttPort = EEPROM.read(2522) + ((EEPROM.read(2523) << 8) & 0xFF00); | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 11) | ||||
|   { | ||||
|     strip.milliampsPerLed = EEPROM.read(375); | ||||
|   } else if (strip.ablMilliampsMax == 65000) //65000 indicates disabled ABL in <0.8.7 | ||||
|   { | ||||
|     strip.ablMilliampsMax = ABL_MILLIAMPS_DEFAULT; | ||||
|     strip.milliampsPerLed = 0; //disable ABL | ||||
|   } | ||||
|   if (lastEEPROMversion > 12) | ||||
|   { | ||||
|     readStringFromEEPROM(990, ntpServerName, 32); | ||||
|   } | ||||
|   if (lastEEPROMversion > 13) | ||||
|   { | ||||
|     mqttEnabled = EEPROM.read(2299); | ||||
|     syncToggleReceive = EEPROM.read(397); | ||||
|   } else { | ||||
|     mqttEnabled = true; | ||||
|     syncToggleReceive = false; | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 14) | ||||
|   { | ||||
|     DMXAddress = EEPROM.read(2197) + ((EEPROM.read(2198) << 8) & 0xFF00); | ||||
|     DMXMode = EEPROM.read(2199); | ||||
|   } else { | ||||
|     DMXAddress = 1; | ||||
|     DMXMode = DMX_MODE_MULTIPLE_RGB; | ||||
|   } | ||||
|  | ||||
|   //if (lastEEPROMversion > 15) | ||||
|   //{ | ||||
|     noWifiSleep = EEPROM.read(370); | ||||
|   //} | ||||
|  | ||||
|   if (lastEEPROMversion > 17) | ||||
|   { | ||||
|     e131SkipOutOfSequence = EEPROM.read(2189); | ||||
|   } else { | ||||
|     e131SkipOutOfSequence = true; | ||||
|   } | ||||
|  | ||||
|   if (lastEEPROMversion > 18) | ||||
|   { | ||||
|     e131Port = EEPROM.read(2187) + ((EEPROM.read(2188) << 8) & 0xFF00); | ||||
|   } | ||||
|  | ||||
|   #ifdef WLED_ENABLE_DMX | ||||
|   if (lastEEPROMversion > 19) | ||||
|   { | ||||
|     e131ProxyUniverse = EEPROM.read(2185) + ((EEPROM.read(2186) << 8) & 0xFF00); | ||||
|   } | ||||
|   #endif | ||||
|  | ||||
|   if (lastEEPROMversion > 21) { | ||||
|     udpPort2 = EEPROM.read(378) + ((EEPROM.read(379) << 8) & 0xFF00); | ||||
|   }  | ||||
|    | ||||
|   receiveDirect = !EEPROM.read(2200); | ||||
|   notifyMacro = EEPROM.read(2201); | ||||
|  | ||||
|   strip.rgbwMode = EEPROM.read(2203); | ||||
|   skipFirstLed = EEPROM.read(2204); | ||||
|  | ||||
|   if (EEPROM.read(2210) || EEPROM.read(2211) || EEPROM.read(2212)) | ||||
|   { | ||||
|     presetCyclingEnabled = EEPROM.read(2205); | ||||
|     presetCycleTime = EEPROM.read(2206) + ((EEPROM.read(2207) << 8) & 0xFF00); | ||||
|     if (lastEEPROMversion < 21) presetCycleTime /= 100; //was stored in ms, now is in tenths of a second | ||||
|     presetCycleMin = EEPROM.read(2208); | ||||
|     presetCycleMax = EEPROM.read(2209); | ||||
|     //was presetApplyBri = EEPROM.read(2210); | ||||
|     //was presetApplyCol = EEPROM.read(2211); | ||||
|     //was presetApplyFx = EEPROM.read(2212); | ||||
|   } | ||||
|  | ||||
|   bootPreset = EEPROM.read(389); | ||||
|   wifiLock = EEPROM.read(393); | ||||
|   utcOffsetSecs = EEPROM.read(394) + ((EEPROM.read(395) << 8) & 0xFF00); | ||||
|   if (EEPROM.read(396)) utcOffsetSecs = -utcOffsetSecs; //negative | ||||
|   //!EEPROM.read(399); was enableSecTransition | ||||
|  | ||||
|   //favorite setting (preset) memory (25 slots/ each 20byte) | ||||
|   //400 - 899 reserved | ||||
|  | ||||
|   //custom macro memory (16 slots/ each 64byte) | ||||
|   //1024-2047 reserved | ||||
|  | ||||
|   readStringFromEEPROM(2220, blynkApiKey, 35); | ||||
|   if (strlen(blynkApiKey) < 25) blynkApiKey[0] = 0; | ||||
|  | ||||
|   #ifdef WLED_ENABLE_DMX | ||||
|   // DMX (2530 - 2549)2535 | ||||
|   DMXChannels = EEPROM.read(2530); | ||||
|   DMXGap = EEPROM.read(2531) + ((EEPROM.read(2532) << 8) & 0xFF00); | ||||
|   DMXStart = EEPROM.read(2533) + ((EEPROM.read(2534) << 8) & 0xFF00); | ||||
|    | ||||
|   for (int i=0;i<15;i++) { | ||||
|     DMXFixtureMap[i] = EEPROM.read(2535+i); | ||||
|   } //last used: 2549 | ||||
|   DMXStartLED = EEPROM.read(2550); | ||||
|   #endif | ||||
|  | ||||
|   //Usermod memory | ||||
|   //2551 - 2559 reserved for Usermods, usable by default | ||||
|   //2560 - 2943 usable, NOT reserved (need to increase EEPSIZE accordingly, new WLED core features may override this section) | ||||
|   //2944 - 3071 reserved for Usermods (need to increase EEPSIZE to 3072 in const.h) | ||||
|  | ||||
|   overlayCurrent = overlayDefault; | ||||
| } | ||||
|  | ||||
|  | ||||
| //provided for increased compatibility with usermods written for v0.10 | ||||
| void applyMacro(byte index) { | ||||
|   applyPreset(index+16); | ||||
| } | ||||
|  | ||||
|  | ||||
| // De-EEPROM routine, upgrade from previous versions to v0.11 | ||||
| void deEEP() { | ||||
|   if (WLED_FS.exists("/presets.json")) return; | ||||
|    | ||||
|   DEBUG_PRINTLN(F("Preset file not found, attempting to load from EEPROM")); | ||||
|   DEBUGFS_PRINTLN(F("Allocating saving buffer for dEEP")); | ||||
|   DynamicJsonDocument dDoc(JSON_BUFFER_SIZE *2); | ||||
|   JsonObject sObj = dDoc.to<JsonObject>(); | ||||
|   sObj.createNestedObject("0"); | ||||
|  | ||||
|   EEPROM.begin(EEPSIZE); | ||||
|   if (EEPROM.read(233) == 233) { //valid EEPROM save | ||||
|     for (uint16_t index = 1; index <= 16; index++) { //copy presets to presets.json | ||||
|       uint16_t i = 380 + index*20; | ||||
|       byte ver = EEPROM.read(i); | ||||
|  | ||||
|       if ((index < 16 && ver != 1) || (index == 16 && (ver < 2 || ver > 3))) continue; | ||||
|  | ||||
|       char nbuf[16]; | ||||
|       sprintf(nbuf, "%d", index); | ||||
|  | ||||
|       JsonObject pObj = sObj.createNestedObject(nbuf); | ||||
|  | ||||
|       sprintf_P(nbuf, (char*)F("Preset %d"), index); | ||||
|       pObj["n"] = nbuf; | ||||
|  | ||||
|       pObj["bri"] = EEPROM.read(i+1); | ||||
|  | ||||
|       if (index < 16) { | ||||
|         JsonObject segObj = pObj.createNestedObject("seg"); | ||||
|  | ||||
|         JsonArray colarr = segObj.createNestedArray("col"); | ||||
|  | ||||
|         byte numChannels = (strip.isRgbw)? 4:3; | ||||
|  | ||||
|         for (uint8_t k = 0; k < 3; k++) //k=0 primary (i+2) k=1 secondary (i+6) k=2 tertiary color (i+12) | ||||
|         { | ||||
|           JsonArray colX = colarr.createNestedArray(); | ||||
|           uint16_t memloc = i + 6*k; | ||||
|           if (k == 0) memloc += 2; | ||||
|  | ||||
|           for (byte j = 0; j < numChannels; j++) colX.add(EEPROM.read(memloc + j)); | ||||
|         } | ||||
|          | ||||
|         segObj[F("fx")]  = EEPROM.read(i+10); | ||||
|         segObj[F("sx")]  = EEPROM.read(i+11); | ||||
|         segObj[F("ix")]  = EEPROM.read(i+16); | ||||
|         segObj[F("pal")] = EEPROM.read(i+17); | ||||
|       } else { | ||||
|         WS2812FX::Segment* seg = strip.getSegments(); | ||||
|         memcpy(seg, EEPROM.getDataPtr() +i+2, 240); | ||||
|         if (ver == 2) { //versions before 2004230 did not have opacity | ||||
|           for (byte j = 0; j < strip.getMaxSegments(); j++) | ||||
|           { | ||||
|             strip.getSegment(j).opacity = 255; | ||||
|             strip.getSegment(j).setOption(SEG_OPTION_ON, 1); | ||||
|           } | ||||
|         } | ||||
|         setValuesFromMainSeg(); | ||||
|         serializeState(pObj, true, false, true); | ||||
|  | ||||
|         strip.resetSegments(); | ||||
|         setValuesFromMainSeg(); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|      | ||||
|      | ||||
|     for (uint16_t index = 1; index <= 16; index++) { //copy macros to presets.json | ||||
|       char m[65]; | ||||
|       readStringFromEEPROM(1024+64*(index-1), m, 64); | ||||
|       if (m[0]) { //macro exists | ||||
|         char nbuf[16]; | ||||
|         sprintf(nbuf, "%d", index + 16); | ||||
|         JsonObject pObj = sObj.createNestedObject(nbuf); | ||||
|         sprintf_P(nbuf, "Z Macro %d", index); | ||||
|         pObj["n"] = nbuf; | ||||
|         pObj["win"] = m; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   EEPROM.end(); | ||||
|  | ||||
|   File f = WLED_FS.open("/presets.json", "w"); | ||||
|   if (!f) { | ||||
|     errorFlag = ERR_FS_GENERAL; | ||||
|     return; | ||||
|   } | ||||
|   serializeJson(dDoc, f); | ||||
|   f.close(); | ||||
|   DEBUG_PRINTLN(F("deEEP complete!")); | ||||
| } | ||||
|  | ||||
| void deEEPSettings() { | ||||
|   DEBUG_PRINTLN(F("Restore settings from EEPROM")); | ||||
|   EEPROM.begin(EEPSIZE); | ||||
|   loadSettingsFromEEPROM(); | ||||
|   EEPROM.end(); | ||||
|  | ||||
|   serializeConfig(); | ||||
| } | ||||
| @@ -1,6 +1,9 @@ | ||||
| #include "wled.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * Utility for SPIFFS filesystem & Serial console | ||||
|  * Adalight and TPM2 handler | ||||
|  */ | ||||
| 
 | ||||
| enum class AdaState { | ||||
|   Header_A, | ||||
|   Header_d, | ||||
| @@ -10,7 +13,10 @@ enum class AdaState { | ||||
|   Header_CountCheck, | ||||
|   Data_Red, | ||||
|   Data_Green, | ||||
|   Data_Blue | ||||
|   Data_Blue, | ||||
|   TPM2_Header_Type, | ||||
|   TPM2_Header_CountHi, | ||||
|   TPM2_Header_CountLo | ||||
| }; | ||||
| 
 | ||||
| void handleSerial() | ||||
| @@ -30,6 +36,9 @@ void handleSerial() | ||||
|     switch (state) { | ||||
|       case AdaState::Header_A: | ||||
|         if (next == 'A') state = AdaState::Header_d; | ||||
|         else if (next == 0xC9) { //TPM2 start byte
 | ||||
|           state = AdaState::TPM2_Header_Type; | ||||
|         } | ||||
|         break; | ||||
|       case AdaState::Header_d: | ||||
|         if (next == 'd') state = AdaState::Header_a; | ||||
| @@ -54,6 +63,20 @@ void handleSerial() | ||||
|         if (check == next) state = AdaState::Data_Red; | ||||
|         else               state = AdaState::Header_A; | ||||
|         break; | ||||
|       case AdaState::TPM2_Header_Type: | ||||
|         state = AdaState::Header_A; //(unsupported) TPM2 command or invalid type
 | ||||
|         if (next == 0xDA) state = AdaState::TPM2_Header_CountHi; //TPM2 data
 | ||||
|         else if (next == 0xAA) Serial.write(0xAC); //TPM2 ping
 | ||||
|         break; | ||||
|       case AdaState::TPM2_Header_CountHi: | ||||
|         pixel = 0; | ||||
|         count = (next * 0x100) /3; | ||||
|         state = AdaState::TPM2_Header_CountLo; | ||||
|         break; | ||||
|       case AdaState::TPM2_Header_CountLo: | ||||
|         count += next /3; | ||||
|         state = AdaState::Data_Red; | ||||
|         break; | ||||
|       case AdaState::Data_Red: | ||||
|         red   = next; | ||||
|         state = AdaState::Data_Green; | ||||
| @@ -64,13 +87,13 @@ void handleSerial() | ||||
|         break; | ||||
|       case AdaState::Data_Blue: | ||||
|         byte blue  = next; | ||||
|         setRealtimePixel(pixel++, red, green, blue, 0); | ||||
|         if (!realtimeOverride) setRealtimePixel(pixel++, red, green, blue, 0); | ||||
|         if (--count > 0) state = AdaState::Data_Red; | ||||
|         else { | ||||
|           if (!realtimeMode && bri == 0) strip.setBrightness(briLast); | ||||
|           arlsLock(realtimeTimeoutMs, REALTIME_MODE_ADALIGHT); | ||||
|           realtimeLock(realtimeTimeoutMs, REALTIME_MODE_ADALIGHT); | ||||
| 
 | ||||
|           strip.show(); | ||||
|           if (!realtimeOverride) strip.show(); | ||||
|           state = AdaState::Header_A; | ||||
|         } | ||||
|         break; | ||||
| @@ -78,44 +101,3 @@ void handleSerial() | ||||
|   } | ||||
|   #endif | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| #if !defined WLED_DISABLE_FILESYSTEM && defined WLED_ENABLE_FS_SERVING | ||||
| //Un-comment any file types you need
 | ||||
| String getContentType(AsyncWebServerRequest* request, String filename){ | ||||
|   if(request->hasArg("download")) return "application/octet-stream"; | ||||
|   else if(filename.endsWith(".htm")) return "text/html"; | ||||
|   else if(filename.endsWith(".html")) return "text/html"; | ||||
| //  else if(filename.endsWith(".css")) return "text/css";
 | ||||
| //  else if(filename.endsWith(".js")) return "application/javascript";
 | ||||
|   else if(filename.endsWith(".json")) return "application/json"; | ||||
|   else if(filename.endsWith(".png")) return "image/png"; | ||||
| //  else if(filename.endsWith(".gif")) return "image/gif";
 | ||||
|   else if(filename.endsWith(".jpg")) return "image/jpeg"; | ||||
|   else if(filename.endsWith(".ico")) return "image/x-icon"; | ||||
| //  else if(filename.endsWith(".xml")) return "text/xml";
 | ||||
| //  else if(filename.endsWith(".pdf")) return "application/x-pdf";
 | ||||
| //  else if(filename.endsWith(".zip")) return "application/x-zip";
 | ||||
| //  else if(filename.endsWith(".gz")) return "application/x-gzip";
 | ||||
|   return "text/plain"; | ||||
| } | ||||
| 
 | ||||
| bool handleFileRead(AsyncWebServerRequest* request, String path){ | ||||
|   DEBUG_PRINTLN("FileRead: " + path); | ||||
|   if(path.endsWith("/")) path += "index.htm"; | ||||
|   String contentType = getContentType(request, path); | ||||
|   String pathWithGz = path + ".gz"; | ||||
|   if(SPIFFS.exists(pathWithGz)){ | ||||
|     request->send(SPIFFS, pathWithGz, contentType); | ||||
|     return true; | ||||
|   } | ||||
|   if(SPIFFS.exists(path)) { | ||||
|     request->send(SPIFFS, path, contentType); | ||||
|     return true; | ||||
|   } | ||||
|   return false; | ||||
| } | ||||
| 
 | ||||
| #else | ||||
| bool handleFileRead(AsyncWebServerRequest*, String path){return false;} | ||||
| #endif | ||||
| @@ -1,5 +1,7 @@ | ||||
| #include "wled.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * Server page definitions | ||||
|  * Integrated HTTP web server page declarations | ||||
|  */ | ||||
| 
 | ||||
| //Is this an IP?
 | ||||
| @@ -23,7 +25,7 @@ bool captivePortal(AsyncWebServerRequest *request) | ||||
|   if (!isIp(hostH) && hostH.indexOf("wled.me") < 0 && hostH.indexOf(cmDNS) < 0) { | ||||
|     DEBUG_PRINTLN("Captive portal"); | ||||
|     AsyncWebServerResponse *response = request->beginResponse(302); | ||||
|     response->addHeader("Location", "http://4.3.2.1"); | ||||
|     response->addHeader(F("Location"), F("http://4.3.2.1")); | ||||
|     request->send(response); | ||||
|     return true; | ||||
|   } | ||||
| @@ -33,13 +35,19 @@ bool captivePortal(AsyncWebServerRequest *request) | ||||
| void initServer() | ||||
| { | ||||
|   //CORS compatiblity
 | ||||
|   DefaultHeaders::Instance().addHeader("Access-Control-Allow-Origin", "*"); | ||||
|   DefaultHeaders::Instance().addHeader("Access-Control-Allow-Methods", "*"); | ||||
|   DefaultHeaders::Instance().addHeader("Access-Control-Allow-Headers", "*"); | ||||
|   DefaultHeaders::Instance().addHeader(F("Access-Control-Allow-Origin"), "*"); | ||||
|   DefaultHeaders::Instance().addHeader(F("Access-Control-Allow-Methods"), "*"); | ||||
|   DefaultHeaders::Instance().addHeader(F("Access-Control-Allow-Headers"), "*"); | ||||
| 
 | ||||
|   server.on("/liveview", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|     request->send_P(200, "text/html", PAGE_liveview); | ||||
|   }); | ||||
|  #ifdef WLED_ENABLE_WEBSOCKETS | ||||
|     server.on("/liveview", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|       request->send_P(200, "text/html", PAGE_liveviewws); | ||||
|     }); | ||||
|  #else | ||||
|     server.on("/liveview", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|       request->send_P(200, "text/html", PAGE_liveview); | ||||
|     }); | ||||
|   #endif | ||||
|    | ||||
|   //settings page
 | ||||
|   server.on("/settings", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
| @@ -62,40 +70,12 @@ void initServer() | ||||
|   }); | ||||
|    | ||||
|   server.on("/reset", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|     serveMessage(request, 200,"Rebooting now...","Please wait ~10 seconds...",129); | ||||
|     serveMessage(request, 200,F("Rebooting now..."),F("Please wait ~10 seconds..."),129); | ||||
|     doReboot = true; | ||||
|   }); | ||||
|    | ||||
|   server.on("/settings/wifi", HTTP_POST, [](AsyncWebServerRequest *request){ | ||||
|     if (!(wifiLock && otaLock)) handleSettingsSet(request, 1); | ||||
|     serveMessage(request, 200,"WiFi settings saved.","Please connect to the new IP (if changed)",129); | ||||
|     forceReconnect = true; | ||||
|   }); | ||||
| 
 | ||||
|   server.on("/settings/leds", HTTP_POST, [](AsyncWebServerRequest *request){ | ||||
|     handleSettingsSet(request, 2); | ||||
|     serveMessage(request, 200,"LED settings saved.","Redirecting...",1); | ||||
|   }); | ||||
| 
 | ||||
|   server.on("/settings/ui", HTTP_POST, [](AsyncWebServerRequest *request){ | ||||
|     handleSettingsSet(request, 3); | ||||
|     serveMessage(request, 200,"UI settings saved.","Redirecting...",1); | ||||
|   }); | ||||
| 
 | ||||
|   server.on("/settings/sync", HTTP_POST, [](AsyncWebServerRequest *request){ | ||||
|     handleSettingsSet(request, 4); | ||||
|     serveMessage(request, 200,"Sync settings saved.","Redirecting...",1); | ||||
|   }); | ||||
| 
 | ||||
|   server.on("/settings/time", HTTP_POST, [](AsyncWebServerRequest *request){ | ||||
|     handleSettingsSet(request, 5); | ||||
|     serveMessage(request, 200,"Time settings saved.","Redirecting...",1); | ||||
|   }); | ||||
| 
 | ||||
|   server.on("/settings/sec", HTTP_POST, [](AsyncWebServerRequest *request){ | ||||
|     handleSettingsSet(request, 6); | ||||
|     if (!doReboot) serveMessage(request, 200,"Security settings saved.","Rebooting, please wait ~10 seconds...",129); | ||||
|     doReboot = true; | ||||
|   server.on("/settings", HTTP_POST, [](AsyncWebServerRequest *request){ | ||||
|     serveSettings(request, true); | ||||
|   }); | ||||
| 
 | ||||
|   server.on("/json", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
| @@ -105,22 +85,23 @@ void initServer() | ||||
|   AsyncCallbackJsonWebHandler* handler = new AsyncCallbackJsonWebHandler("/json", [](AsyncWebServerRequest *request) { | ||||
|     bool verboseResponse = false; | ||||
|     { //scope JsonDocument so it releases its buffer
 | ||||
|       DynamicJsonDocument jsonBuffer(8192); | ||||
|       DynamicJsonDocument jsonBuffer(JSON_BUFFER_SIZE); | ||||
|       DeserializationError error = deserializeJson(jsonBuffer, (uint8_t*)(request->_tempObject)); | ||||
|       JsonObject root = jsonBuffer.as<JsonObject>(); | ||||
|       if (error || root.isNull()) { | ||||
|         request->send(400, "application/json", "{\"error\":10}"); return; | ||||
|         request->send(400, "application/json", F("{\"error\":9}")); return; | ||||
|       } | ||||
|       fileDoc = &jsonBuffer; | ||||
|       verboseResponse = deserializeState(root); | ||||
|       fileDoc = nullptr; | ||||
|     } | ||||
|     if (verboseResponse) { //if JSON contains "v"
 | ||||
|       serveJson(request); return;  | ||||
|     }  | ||||
|     request->send(200, "application/json", "{\"success\":true}"); | ||||
|     request->send(200, "application/json", F("{\"success\":true}")); | ||||
|   }); | ||||
|   server.addHandler(handler); | ||||
| 
 | ||||
|   //*******DEPRECATED*******
 | ||||
|   server.on("/version", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|     request->send(200, "text/plain", (String)VERSION); | ||||
|     }); | ||||
| @@ -132,27 +113,30 @@ void initServer() | ||||
|   server.on("/freeheap", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|     request->send(200, "text/plain", (String)ESP.getFreeHeap()); | ||||
|     }); | ||||
|   //*******END*******/
 | ||||
|    | ||||
|   server.on("/u", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|     request->send_P(200, "text/html", PAGE_usermod); | ||||
|     }); | ||||
|      | ||||
|   server.on("/url", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|     URL_response(request); | ||||
|     }); | ||||
|      | ||||
|   server.on("/teapot", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|     serveMessage(request, 418, "418. I'm a teapot.", "(Tangible Embedded Advanced Project Of Twinkling)", 254); | ||||
|     serveMessage(request, 418, F("418. I'm a teapot."), F("(Tangible Embedded Advanced Project Of Twinkling)"), 254); | ||||
|     }); | ||||
|      | ||||
|   //if OTA is allowed
 | ||||
|   if (!otaLock){ | ||||
|     #if !defined WLED_DISABLE_FILESYSTEM && defined WLED_ENABLE_FS_EDITOR | ||||
|     #ifdef WLED_ENABLE_FS_EDITOR | ||||
|      #ifdef ARDUINO_ARCH_ESP32 | ||||
|       server.addHandler(new SPIFFSEditor(SPIFFS));//http_username,http_password));
 | ||||
|       server.addHandler(new SPIFFSEditor(WLED_FS));//http_username,http_password));
 | ||||
|      #else | ||||
|       server.addHandler(new SPIFFSEditor());//http_username,http_password));
 | ||||
|       server.addHandler(new SPIFFSEditor("","",WLED_FS));//http_username,http_password));
 | ||||
|      #endif | ||||
|     #else | ||||
|     server.on("/edit", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|       serveMessage(request, 501, "Not implemented", "The SPIFFS editor is disabled in this build.", 254); | ||||
|       serveMessage(request, 501, "Not implemented", F("The FS editor is disabled in this build."), 254); | ||||
|     }); | ||||
|     #endif | ||||
|     //init ota page
 | ||||
| @@ -164,13 +148,13 @@ void initServer() | ||||
|     server.on("/update", HTTP_POST, [](AsyncWebServerRequest *request){ | ||||
|       if (Update.hasError()) | ||||
|       { | ||||
|         serveMessage(request, 500, "Failed updating firmware!", "Please check your file and retry!", 254); return; | ||||
|         serveMessage(request, 500, F("Failed updating firmware!"), F("Please check your file and retry!"), 254); return; | ||||
|       } | ||||
|       serveMessage(request, 200, "Successfully updated firmware!", "Please wait while the module reboots...", 131);  | ||||
|       serveMessage(request, 200, F("Successfully updated firmware!"), F("Please wait while the module reboots..."), 131);  | ||||
|       doReboot = true; | ||||
|     },[](AsyncWebServerRequest *request, String filename, size_t index, uint8_t *data, size_t len, bool final){ | ||||
|       if(!index){ | ||||
|         DEBUG_PRINTLN("OTA Update Start"); | ||||
|         DEBUG_PRINTLN(F("OTA Update Start")); | ||||
|         #ifdef ESP8266 | ||||
|         Update.runAsync(true); | ||||
|         #endif | ||||
| @@ -179,32 +163,46 @@ void initServer() | ||||
|       if(!Update.hasError()) Update.write(data, len); | ||||
|       if(final){ | ||||
|         if(Update.end(true)){ | ||||
|           DEBUG_PRINTLN("Update Success"); | ||||
|           DEBUG_PRINTLN(F("Update Success")); | ||||
|         } else { | ||||
|           DEBUG_PRINTLN("Update Failed"); | ||||
|           DEBUG_PRINTLN(F("Update Failed")); | ||||
|         } | ||||
|       } | ||||
|     }); | ||||
|      | ||||
|     #else | ||||
|     server.on("/update", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|       serveMessage(request, 501, "Not implemented", "OTA updates are disabled in this build.", 254); | ||||
|       serveMessage(request, 501, "Not implemented", F("OTA updates are disabled in this build."), 254); | ||||
|     }); | ||||
|     #endif | ||||
|   } else | ||||
|   { | ||||
|     server.on("/edit", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|       serveMessage(request, 500, "Access Denied", "Please unlock OTA in security settings!", 254); | ||||
|       serveMessage(request, 500, "Access Denied", F("Please unlock OTA in security settings!"), 254); | ||||
|     }); | ||||
|     server.on("/update", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|       serveMessage(request, 500, "Access Denied", "Please unlock OTA in security settings!", 254); | ||||
|       serveMessage(request, 500, "Access Denied", F("Please unlock OTA in security settings!"), 254); | ||||
|     }); | ||||
|   } | ||||
| 
 | ||||
| 
 | ||||
|     #ifdef WLED_ENABLE_DMX | ||||
|     server.on("/dmxmap", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|       request->send_P(200, "text/html", PAGE_dmxmap     , dmxProcessor); | ||||
|     }); | ||||
|     #else | ||||
|     server.on("/dmxmap", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|       serveMessage(request, 501, "Not implemented", F("DMX support is not enabled in this build."), 254); | ||||
|     }); | ||||
|     #endif | ||||
|   server.on("/", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|     if (captivePortal(request)) return; | ||||
|     serveIndexOrWelcome(request); | ||||
|   }); | ||||
| 
 | ||||
|   #ifdef WLED_ENABLE_WEBSOCKETS | ||||
|   server.addHandler(&ws); | ||||
|   #endif | ||||
|    | ||||
|   //called when the url is not defined here, ajax-in; get-settings
 | ||||
|   server.onNotFound([](AsyncWebServerRequest *request){ | ||||
| @@ -222,10 +220,8 @@ void initServer() | ||||
|     #ifndef WLED_DISABLE_ALEXA | ||||
|     if(espalexa.handleAlexaApiCall(request)) return; | ||||
|     #endif | ||||
|     #ifdef WLED_ENABLE_FS_SERVING | ||||
|     if(handleFileRead(request, request->url())) return; | ||||
|     #endif | ||||
|     request->send(404, "text/plain", "Not Found"); | ||||
|     request->send_P(404, "text/html", PAGE_404); | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| @@ -239,16 +235,32 @@ void serveIndexOrWelcome(AsyncWebServerRequest *request) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| bool handleIfNoneMatchCacheHeader(AsyncWebServerRequest* request) | ||||
| { | ||||
|   AsyncWebHeader* header = request->getHeader("If-None-Match"); | ||||
|   if (header && header->value() == String(VERSION)) { | ||||
|     request->send(304); | ||||
|     return true; | ||||
|   } | ||||
|   return false; | ||||
| } | ||||
| 
 | ||||
| void setStaticContentCacheHeaders(AsyncWebServerResponse *response) | ||||
| { | ||||
|   response->addHeader(F("Cache-Control"),"no-cache"); | ||||
|   response->addHeader(F("ETag"), String(VERSION)); | ||||
| } | ||||
| 
 | ||||
| void serveIndex(AsyncWebServerRequest* request) | ||||
| { | ||||
|   #ifdef WLED_ENABLE_FS_SERVING | ||||
|   if (handleFileRead(request, "/index.htm")) return; | ||||
|   #endif | ||||
| 
 | ||||
|   if (handleIfNoneMatchCacheHeader(request)) return; | ||||
| 
 | ||||
|   AsyncWebServerResponse *response = request->beginResponse_P(200, "text/html", PAGE_index, PAGE_index_L); | ||||
| 
 | ||||
|   response->addHeader("Content-Encoding","gzip"); | ||||
|   response->addHeader(F("Content-Encoding"),"gzip"); | ||||
|   setStaticContentCacheHeaders(response); | ||||
|    | ||||
|   request->send(response); | ||||
| } | ||||
| @@ -258,25 +270,29 @@ String msgProcessor(const String& var) | ||||
| { | ||||
|   if (var == "MSG") { | ||||
|     String messageBody = messageHead; | ||||
|     messageBody += "</h2>"; | ||||
|     messageBody += F("</h2>"); | ||||
|     messageBody += messageSub; | ||||
|     uint32_t optt = optionType; | ||||
| 
 | ||||
|     if (optt < 60) //redirect to settings after optionType seconds
 | ||||
|     { | ||||
|       messageBody += "<script>setTimeout(RS," + String(optt*1000) + ")</script>"; | ||||
|       messageBody += F("<script>setTimeout(RS,"); | ||||
|       messageBody +=String(optt*1000); | ||||
|       messageBody += F(")</script>"); | ||||
|     } else if (optt < 120) //redirect back after optionType-60 seconds, unused
 | ||||
|     { | ||||
|       //messageBody += "<script>setTimeout(B," + String((optt-60)*1000) + ")</script>";
 | ||||
|     } else if (optt < 180) //reload parent after optionType-120 seconds
 | ||||
|     { | ||||
|       messageBody += "<script>setTimeout(RP," + String((optt-120)*1000) + ")</script>"; | ||||
|       messageBody += F("<script>setTimeout(RP,"); | ||||
|       messageBody += String((optt-120)*1000); | ||||
|       messageBody += F(")</script>"); | ||||
|     } else if (optt == 253) | ||||
|     { | ||||
|       messageBody += "<br><br><form action=/settings><button class=\"bt\" type=submit>Back</button></form>"; //button to settings
 | ||||
|       messageBody += F("<br><br><form action=/settings><button class=\"bt\" type=submit>Back</button></form>"); //button to settings
 | ||||
|     } else if (optt == 254) | ||||
|     { | ||||
|       messageBody += "<br><br><button type=\"button\" class=\"bt\" onclick=\"B()\">Back</button>"; | ||||
|       messageBody += F("<br><br><button type=\"button\" class=\"bt\" onclick=\"B()\">Back</button>"); | ||||
|     } | ||||
|     return messageBody; | ||||
|   } | ||||
| @@ -284,7 +300,7 @@ String msgProcessor(const String& var) | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void serveMessage(AsyncWebServerRequest* request, uint16_t code, String headl, String subl="", byte optionT=255) | ||||
| void serveMessage(AsyncWebServerRequest* request, uint16_t code, const String& headl, const String& subl, byte optionT) | ||||
| { | ||||
|   messageHead = headl; | ||||
|   messageSub = subl; | ||||
| @@ -301,12 +317,40 @@ String settingsProcessor(const String& var) | ||||
|     getSettingsJS(optionType, buf); | ||||
|     return String(buf); | ||||
|   } | ||||
|    | ||||
|   #ifdef WLED_ENABLE_DMX | ||||
| 
 | ||||
|   if (var == "DMXMENU") { | ||||
|     return String(F("<form action=/settings/dmx><button type=submit>DMX Output</button></form>")); | ||||
|   } | ||||
|    | ||||
|   #endif | ||||
|   if (var == "SCSS") return String(FPSTR(PAGE_settingsCss)); | ||||
|   return String(); | ||||
| } | ||||
| 
 | ||||
| String dmxProcessor(const String& var) | ||||
| { | ||||
|   String mapJS; | ||||
|   #ifdef WLED_ENABLE_DMX | ||||
|     if (var == "DMXVARS") { | ||||
|       mapJS += "\nCN=" + String(DMXChannels) + ";\n"; | ||||
|       mapJS += "CS=" + String(DMXStart) + ";\n"; | ||||
|       mapJS += "CG=" + String(DMXGap) + ";\n"; | ||||
|       mapJS += "LC=" + String(ledCount) + ";\n"; | ||||
|       mapJS += "var CH=["; | ||||
|       for (int i=0;i<15;i++) { | ||||
|         mapJS += String(DMXFixtureMap[i]) + ","; | ||||
|       } | ||||
|       mapJS += "0];"; | ||||
|     } | ||||
|   #endif | ||||
|    | ||||
|   return mapJS; | ||||
| } | ||||
| 
 | ||||
| void serveSettings(AsyncWebServerRequest* request) | ||||
| 
 | ||||
| void serveSettings(AsyncWebServerRequest* request, bool post) | ||||
| { | ||||
|   byte subPage = 0; | ||||
|   const String& url = request->url(); | ||||
| @@ -318,11 +362,39 @@ void serveSettings(AsyncWebServerRequest* request) | ||||
|     else if (url.indexOf("sync") > 0) subPage = 4; | ||||
|     else if (url.indexOf("time") > 0) subPage = 5; | ||||
|     else if (url.indexOf("sec")  > 0) subPage = 6; | ||||
|     #ifdef WLED_ENABLE_DMX // include only if DMX is enabled
 | ||||
|     else if (url.indexOf("dmx")  > 0) subPage = 7; | ||||
|     #endif | ||||
|   } else subPage = 255; //welcome page
 | ||||
| 
 | ||||
|   if (subPage == 1 && wifiLock && otaLock) | ||||
|   { | ||||
|     serveMessage(request, 500, "Access Denied", "Please unlock OTA in security settings!", 254); return; | ||||
|     serveMessage(request, 500, "Access Denied", F("Please unlock OTA in security settings!"), 254); return; | ||||
|   } | ||||
| 
 | ||||
|   if (post) { //settings/set POST request, saving
 | ||||
|     if (subPage != 1 || !(wifiLock && otaLock)) handleSettingsSet(request, subPage); | ||||
| 
 | ||||
|     char s[32]; | ||||
|     char s2[45] = ""; | ||||
| 
 | ||||
|     switch (subPage) { | ||||
|       case 1: strcpy_P(s, PSTR("WiFi")); strcpy_P(s2, PSTR("Please connect to the new IP (if changed)")); forceReconnect = true; break; | ||||
|       case 2: strcpy_P(s, PSTR("LED")); break; | ||||
|       case 3: strcpy_P(s, PSTR("UI")); break; | ||||
|       case 4: strcpy_P(s, PSTR("Sync")); break; | ||||
|       case 5: strcpy_P(s, PSTR("Time")); break; | ||||
|       case 6: strcpy_P(s, PSTR("Security")); strcpy_P(s2, PSTR("Rebooting, please wait ~10 seconds...")); break; | ||||
|       case 7: strcpy_P(s, PSTR("DMX")); break; | ||||
|     } | ||||
| 
 | ||||
|     strcat_P(s, PSTR(" settings saved.")); | ||||
|     if (!s2[0]) strcpy_P(s2, PSTR("Redirecting...")); | ||||
| 
 | ||||
|     if (!doReboot) serveMessage(request, 200, s, s2, (subPage == 1 || subPage == 6) ? 129 : 1); | ||||
|     if (subPage == 6) doReboot = true; | ||||
| 
 | ||||
|     return; | ||||
|   } | ||||
|    | ||||
|   #ifdef WLED_DISABLE_MOBILE_UI //disable welcome page if not enough storage
 | ||||
| @@ -339,7 +411,8 @@ void serveSettings(AsyncWebServerRequest* request) | ||||
|     case 4:   request->send_P(200, "text/html", PAGE_settings_sync, settingsProcessor); break; | ||||
|     case 5:   request->send_P(200, "text/html", PAGE_settings_time, settingsProcessor); break; | ||||
|     case 6:   request->send_P(200, "text/html", PAGE_settings_sec , settingsProcessor); break; | ||||
|     case 7:   request->send_P(200, "text/html", PAGE_settings_dmx , settingsProcessor); break; | ||||
|     case 255: request->send_P(200, "text/html", PAGE_welcome); break; | ||||
|     default:  request->send_P(200, "text/html", PAGE_settings);  | ||||
|     default:  request->send_P(200, "text/html", PAGE_settings     , settingsProcessor);  | ||||
|   } | ||||
| } | ||||
							
								
								
									
										114
									
								
								wled00/ws.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								wled00/ws.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,114 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * WebSockets server for bidirectional communication | ||||
|  */ | ||||
| #ifdef WLED_ENABLE_WEBSOCKETS | ||||
|  | ||||
| uint16_t wsLiveClientId = 0; | ||||
| unsigned long wsLastLiveTime = 0; | ||||
| //uint8_t* wsFrameBuffer = nullptr; | ||||
|  | ||||
| #define WS_LIVE_INTERVAL 40 | ||||
|  | ||||
| void wsEvent(AsyncWebSocket * server, AsyncWebSocketClient * client, AwsEventType type, void * arg, uint8_t *data, size_t len) | ||||
| { | ||||
|   if(type == WS_EVT_CONNECT){ | ||||
|     //client connected | ||||
|     sendDataWs(client); | ||||
|     //client->ping(); | ||||
|   } else if(type == WS_EVT_DISCONNECT){ | ||||
|     //client disconnected | ||||
|     if (client->id() == wsLiveClientId) wsLiveClientId = 0; | ||||
|   } else if(type == WS_EVT_DATA){ | ||||
|     //data packet | ||||
|     AwsFrameInfo * info = (AwsFrameInfo*)arg; | ||||
|     if(info->final && info->index == 0 && info->len == len){ | ||||
|       //the whole message is in a single frame and we got all of it's data (max. 1450byte) | ||||
|       if(info->opcode == WS_TEXT) | ||||
|       { | ||||
|         bool verboseResponse = false; | ||||
|         { //scope JsonDocument so it releases its buffer | ||||
|           DynamicJsonDocument jsonBuffer(JSON_BUFFER_SIZE); | ||||
|           DeserializationError error = deserializeJson(jsonBuffer, data, len); | ||||
|           JsonObject root = jsonBuffer.as<JsonObject>(); | ||||
|           if (error || root.isNull()) return; | ||||
|  | ||||
|           if (root.containsKey("lv")) | ||||
|           { | ||||
|             wsLiveClientId = root["lv"] ? client->id() : 0; | ||||
|           } | ||||
|  | ||||
|           verboseResponse = deserializeState(root); | ||||
|         } | ||||
|         if (verboseResponse || millis() - lastInterfaceUpdate < 1900) sendDataWs(client); //update if it takes longer than 100ms until next "broadcast" | ||||
|       } | ||||
|     } else { | ||||
|       //message is comprised of multiple frames or the frame is split into multiple packets | ||||
|       //if(info->index == 0){ | ||||
|         //if (!wsFrameBuffer && len < 4096) wsFrameBuffer = new uint8_t[4096]; | ||||
|       //} | ||||
|  | ||||
|       //if (wsFrameBuffer && len < 4096 && info->index + info->) | ||||
|       //{ | ||||
|  | ||||
|       //} | ||||
|  | ||||
|       if((info->index + len) == info->len){ | ||||
|         if(info->final){ | ||||
|           if(info->message_opcode == WS_TEXT) { | ||||
|             client->text(F("{\"error\":9}")); //we do not handle split packets right now | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } else if(type == WS_EVT_ERROR){ | ||||
|     //error was received from the other end | ||||
|  | ||||
|   } else if(type == WS_EVT_PONG){ | ||||
|     //pong message was received (in response to a ping request maybe) | ||||
|  | ||||
|   } | ||||
| } | ||||
|  | ||||
| void sendDataWs(AsyncWebSocketClient * client) | ||||
| { | ||||
|   if (!ws.count()) return; | ||||
|   AsyncWebSocketMessageBuffer * buffer; | ||||
|  | ||||
|   { //scope JsonDocument so it releases its buffer | ||||
|     DynamicJsonDocument doc(JSON_BUFFER_SIZE); | ||||
|     JsonObject state = doc.createNestedObject("state"); | ||||
|     serializeState(state); | ||||
|     JsonObject info  = doc.createNestedObject("info"); | ||||
|     serializeInfo(info); | ||||
|     size_t len = measureJson(doc); | ||||
|     buffer = ws.makeBuffer(len); | ||||
|     if (!buffer) return; //out of memory | ||||
|  | ||||
|     serializeJson(doc, (char *)buffer->get(), len +1); | ||||
|   }  | ||||
|   if (client) { | ||||
|     client->text(buffer); | ||||
|   } else { | ||||
|     ws.textAll(buffer); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void handleWs() | ||||
| { | ||||
|   if (millis() - wsLastLiveTime > WS_LIVE_INTERVAL) | ||||
|   { | ||||
|     ws.cleanupClients(); | ||||
|     bool success = true; | ||||
|     if (wsLiveClientId) | ||||
|       success = serveLiveLeds(nullptr, wsLiveClientId); | ||||
|     wsLastLiveTime = millis(); | ||||
|     if (!success) wsLastLiveTime -= 20; //try again in 20ms if failed due to non-empty WS queue | ||||
|   } | ||||
| } | ||||
|  | ||||
| #else | ||||
| void handleWs() {} | ||||
| void sendDataWs(AsyncWebSocketClient * client) {} | ||||
| #endif | ||||
							
								
								
									
										516
									
								
								wled00/xml.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										516
									
								
								wled00/xml.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,516 @@ | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * Sending XML status files to client | ||||
|  */ | ||||
|  | ||||
| //macro to convert F to const | ||||
| #define SET_F(x)  (const char*)F(x) | ||||
|  | ||||
| //build XML response to HTTP /win API request | ||||
| void XML_response(AsyncWebServerRequest *request, char* dest) | ||||
| { | ||||
|   char sbuf[(dest == nullptr)?1024:1]; //allocate local buffer if none passed | ||||
|   obuf = (dest == nullptr)? sbuf:dest; | ||||
|  | ||||
|   olen = 0; | ||||
|   oappend(SET_F("<?xml version=\"1.0\" ?><vs><ac>")); | ||||
|   oappendi((nightlightActive && nightlightMode > NL_MODE_SET) ? briT : bri); | ||||
|   oappend(SET_F("</ac>")); | ||||
|  | ||||
|   for (int i = 0; i < 3; i++) | ||||
|   { | ||||
|    oappend("<cl>"); | ||||
|    oappendi(col[i]); | ||||
|    oappend("</cl>"); | ||||
|   } | ||||
|   for (int i = 0; i < 3; i++) | ||||
|   { | ||||
|    oappend("<cs>"); | ||||
|    oappendi(colSec[i]); | ||||
|    oappend("</cs>"); | ||||
|   } | ||||
|   oappend(SET_F("<ns>")); | ||||
|   oappendi(notifyDirect); | ||||
|   oappend(SET_F("</ns><nr>")); | ||||
|   oappendi(receiveNotifications); | ||||
|   oappend(SET_F("</nr><nl>")); | ||||
|   oappendi(nightlightActive); | ||||
|   oappend(SET_F("</nl><nf>")); | ||||
|   oappendi(nightlightMode > NL_MODE_SET); | ||||
|   oappend(SET_F("</nf><nd>")); | ||||
|   oappendi(nightlightDelayMins); | ||||
|   oappend(SET_F("</nd><nt>")); | ||||
|   oappendi(nightlightTargetBri); | ||||
|   oappend(SET_F("</nt><fx>")); | ||||
|   oappendi(effectCurrent); | ||||
|   oappend(SET_F("</fx><sx>")); | ||||
|   oappendi(effectSpeed); | ||||
|   oappend(SET_F("</sx><ix>")); | ||||
|   oappendi(effectIntensity); | ||||
|   oappend(SET_F("</ix><fp>")); | ||||
|   oappendi(effectPalette); | ||||
|   oappend(SET_F("</fp><wv>")); | ||||
|   if (strip.rgbwMode) { | ||||
|    oappendi(col[3]); | ||||
|   } else { | ||||
|    oappend("-1"); | ||||
|   } | ||||
|   oappend(SET_F("</wv><ws>")); | ||||
|   oappendi(colSec[3]); | ||||
|   oappend(SET_F("</ws><ps>")); | ||||
|   oappendi((currentPreset < 1) ? 0:currentPreset); | ||||
|   oappend(SET_F("</ps><cy>")); | ||||
|   oappendi(presetCyclingEnabled); | ||||
|   oappend(SET_F("</cy><ds>")); | ||||
|   oappend(serverDescription); | ||||
|   if (realtimeMode) | ||||
|   { | ||||
|     oappend(SET_F(" (live)")); | ||||
|   } | ||||
|   oappend(SET_F("</ds><ss>")); | ||||
|   oappendi(strip.getMainSegmentId()); | ||||
|   oappend(SET_F("</ss></vs>")); | ||||
|   if (request != nullptr) request->send(200, "text/xml", obuf); | ||||
| } | ||||
|  | ||||
| void URL_response(AsyncWebServerRequest *request) | ||||
| { | ||||
|   char sbuf[256]; | ||||
|   char s2buf[100]; | ||||
|   obuf = s2buf; | ||||
|   olen = 0; | ||||
|  | ||||
|   char s[16]; | ||||
|   oappend(SET_F("http://")); | ||||
|   IPAddress localIP = Network.localIP(); | ||||
|   sprintf(s, "%d.%d.%d.%d", localIP[0], localIP[1], localIP[2], localIP[3]); | ||||
|   oappend(s); | ||||
|  | ||||
|   oappend(SET_F("/win&A=")); | ||||
|   oappendi(bri); | ||||
|   oappend(SET_F("&CL=h")); | ||||
|   for (int i = 0; i < 3; i++) | ||||
|   { | ||||
|    sprintf(s,"%02X", col[i]); | ||||
|    oappend(s);  | ||||
|   } | ||||
|   oappend(SET_F("&C2=h")); | ||||
|   for (int i = 0; i < 3; i++) | ||||
|   { | ||||
|    sprintf(s,"%02X", colSec[i]); | ||||
|    oappend(s); | ||||
|   } | ||||
|   oappend(SET_F("&FX=")); | ||||
|   oappendi(effectCurrent); | ||||
|   oappend(SET_F("&SX=")); | ||||
|   oappendi(effectSpeed); | ||||
|   oappend(SET_F("&IX=")); | ||||
|   oappendi(effectIntensity); | ||||
|   oappend(SET_F("&FP=")); | ||||
|   oappendi(effectPalette); | ||||
|  | ||||
|   obuf = sbuf; | ||||
|   olen = 0; | ||||
|  | ||||
|   oappend(SET_F("<html><body><a href=\"")); | ||||
|   oappend(s2buf); | ||||
|   oappend(SET_F("\" target=\"_blank\">")); | ||||
|   oappend(s2buf);   | ||||
|   oappend(SET_F("</a></body></html>")); | ||||
|  | ||||
|   if (request != nullptr) request->send(200, "text/html", obuf); | ||||
| } | ||||
|  | ||||
| //append a numeric setting to string buffer | ||||
| void sappend(char stype, const char* key, int val) | ||||
| { | ||||
|   char ds[] = "d.Sf."; | ||||
|  | ||||
|   switch(stype) | ||||
|   { | ||||
|     case 'c': //checkbox | ||||
|       oappend(ds); | ||||
|       oappend(key); | ||||
|       oappend(".checked="); | ||||
|       oappendi(val); | ||||
|       oappend(";"); | ||||
|       break; | ||||
|     case 'v': //numeric | ||||
|       oappend(ds); | ||||
|       oappend(key); | ||||
|       oappend(".value="); | ||||
|       oappendi(val); | ||||
|       oappend(";"); | ||||
|       break; | ||||
|     case 'i': //selectedIndex | ||||
|       oappend(ds); | ||||
|       oappend(key); | ||||
|       oappend(SET_F(".selectedIndex=")); | ||||
|       oappendi(val); | ||||
|       oappend(";"); | ||||
|       break; | ||||
|   } | ||||
| } | ||||
|  | ||||
| //append a string setting to buffer | ||||
| void sappends(char stype, const char* key, char* val) | ||||
| { | ||||
|   switch(stype) | ||||
|   { | ||||
|     case 's': //string (we can interpret val as char*) | ||||
|       oappend("d.Sf."); | ||||
|       oappend(key); | ||||
|       oappend(".value=\""); | ||||
|       oappend(val); | ||||
|       oappend("\";"); | ||||
|       break; | ||||
|     case 'm': //message | ||||
|       oappend(SET_F("d.getElementsByClassName")); | ||||
|       oappend(key); | ||||
|       oappend(SET_F(".innerHTML=\"")); | ||||
|       oappend(val); | ||||
|       oappend("\";"); | ||||
|       break; | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| //get values for settings form in javascript | ||||
| void getSettingsJS(byte subPage, char* dest) | ||||
| { | ||||
|   //0: menu 1: wifi 2: leds 3: ui 4: sync 5: time 6: sec | ||||
|   DEBUG_PRINT(F("settings resp")); | ||||
|   DEBUG_PRINTLN(subPage); | ||||
|   obuf = dest; | ||||
|   olen = 0; | ||||
|  | ||||
|   if (subPage <1 || subPage >7) return; | ||||
|  | ||||
|   if (subPage == 1) { | ||||
|     sappends('s',SET_F("CS"),clientSSID); | ||||
|  | ||||
|     byte l = strlen(clientPass); | ||||
|     char fpass[l+1]; //fill password field with *** | ||||
|     fpass[l] = 0; | ||||
|     memset(fpass,'*',l); | ||||
|     sappends('s',SET_F("CP"),fpass); | ||||
|  | ||||
|     char k[3]; k[2] = 0; //IP addresses | ||||
|     for (int i = 0; i<4; i++) | ||||
|     { | ||||
|       k[1] = 48+i; //ascii 0,1,2,3 | ||||
|       k[0] = 'I'; sappend('v',k,staticIP[i]); | ||||
|       k[0] = 'G'; sappend('v',k,staticGateway[i]); | ||||
|       k[0] = 'S'; sappend('v',k,staticSubnet[i]); | ||||
|     } | ||||
|  | ||||
|     sappends('s',SET_F("CM"),cmDNS); | ||||
|     sappend('i',SET_F("AB"),apBehavior); | ||||
|     sappends('s',SET_F("AS"),apSSID); | ||||
|     sappend('c',SET_F("AH"),apHide); | ||||
|  | ||||
|     l = strlen(apPass); | ||||
|     char fapass[l+1]; //fill password field with *** | ||||
|     fapass[l] = 0; | ||||
|     memset(fapass,'*',l); | ||||
|     sappends('s',SET_F("AP"),fapass); | ||||
|  | ||||
|     sappend('v',SET_F("AC"),apChannel); | ||||
|     sappend('c',SET_F("WS"),noWifiSleep); | ||||
|  | ||||
|     #ifdef WLED_USE_ETHERNET | ||||
|     sappend('v',SET_F("ETH"),ethernetType); | ||||
|     #else | ||||
|     //hide ethernet setting if not compiled in | ||||
|     oappend(SET_F("document.getElementById('ethd').style.display='none';")); | ||||
|     #endif | ||||
|  | ||||
|     if (Network.isConnected()) //is connected | ||||
|     { | ||||
|       char s[32]; | ||||
|       IPAddress localIP = Network.localIP(); | ||||
|       sprintf(s, "%d.%d.%d.%d", localIP[0], localIP[1], localIP[2], localIP[3]); | ||||
|  | ||||
|       #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_USE_ETHERNET) | ||||
|       if (Network.isEthernet()) strcat_P(s ,SET_F(" (Ethernet)")); | ||||
|       #endif | ||||
|       sappends('m',SET_F("(\"sip\")[0]"),s); | ||||
|     } else | ||||
|     { | ||||
|       sappends('m',SET_F("(\"sip\")[0]"),(char*)F("Not connected")); | ||||
|     } | ||||
|  | ||||
|     if (WiFi.softAPIP()[0] != 0) //is active | ||||
|     { | ||||
|       char s[16]; | ||||
|       IPAddress apIP = WiFi.softAPIP(); | ||||
|       sprintf(s, "%d.%d.%d.%d", apIP[0], apIP[1], apIP[2], apIP[3]); | ||||
|       sappends('m',SET_F("(\"sip\")[1]"),s); | ||||
|     } else | ||||
|     { | ||||
|       sappends('m',SET_F("(\"sip\")[1]"),(char*)F("Not active")); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (subPage == 2) { | ||||
|     char nS[8]; | ||||
|  | ||||
|     // add usermod pins as d.um_p array (TODO: usermod config shouldn't use state. instead we should load "um" object from cfg.json) | ||||
|     /*DynamicJsonDocument doc(JSON_BUFFER_SIZE); | ||||
|     JsonObject mods = doc.createNestedObject(F("mods")); | ||||
|     usermods.addToJsonState(mods); | ||||
|     if (!mods.isNull()) { | ||||
|       uint8_t i=0; | ||||
|       oappend(SET_F("d.um_p=[")); | ||||
|       for (JsonPair kv : mods) { | ||||
|         if (strncmp_P(kv.key().c_str(),PSTR("pin_"),4) == 0) { | ||||
|           if (i++) oappend(SET_F(",")); | ||||
|           oappend(itoa((int)kv.value(),nS,10)); | ||||
|         } | ||||
|       } | ||||
|       oappend(SET_F("];")); | ||||
|     }*/ | ||||
|  | ||||
|     oappend(SET_F("bLimits(")); | ||||
|     oappend(itoa(WLED_MAX_BUSSES,nS,10)); | ||||
|     oappend(","); | ||||
|     oappend(itoa(MAX_LEDS_PER_BUS,nS,10)); | ||||
|     oappend(","); | ||||
|     oappend(itoa(MAX_LED_MEMORY,nS,10)); | ||||
|     oappend(SET_F(");")); | ||||
|  | ||||
|     oappend(SET_F("d.Sf.LC.max=")); //TODO Formula for max LEDs on ESP8266 depending on types. 500 DMA or 1500 UART (about 4kB mem usage) | ||||
|     oappendi(MAX_LEDS); | ||||
|     oappend(";"); | ||||
|  | ||||
|     sappend('v',SET_F("LC"),ledCount); | ||||
|  | ||||
|     for (uint8_t s=0; s < busses.getNumBusses(); s++){ | ||||
|       Bus* bus = busses.getBus(s); | ||||
|       char lp[4] = "L0"; lp[2] = 48+s; lp[3] = 0; //ascii 0-9 //strip data pin | ||||
|       char lc[4] = "LC"; lc[2] = 48+s; lc[3] = 0; //strip length | ||||
|       char co[4] = "CO"; co[2] = 48+s; co[3] = 0; //strip color order | ||||
|       char lt[4] = "LT"; lt[2] = 48+s; lt[3] = 0; //strip type | ||||
|       char ls[4] = "LS"; ls[2] = 48+s; ls[3] = 0; //strip start LED | ||||
|       char cv[4] = "CV"; cv[2] = 48+s; cv[3] = 0; //strip reverse | ||||
|       oappend(SET_F("addLEDs(1);")); | ||||
|       uint8_t pins[5]; | ||||
|       uint8_t nPins = bus->getPins(pins); | ||||
|       for (uint8_t i = 0; i < nPins; i++) { | ||||
|         lp[1] = 48+i; | ||||
|         if (pinManager.isPinOk(pins[i])) sappend('v', lp, pins[i]); | ||||
|       } | ||||
|       sappend('v', lc, bus->getLength()); | ||||
|       sappend('v',lt,bus->getType()); | ||||
|       sappend('v',co,bus->getColorOrder()); | ||||
|       sappend('v',ls,bus->getStart()); | ||||
|       sappend('c',cv,bus->reversed); | ||||
|     } | ||||
|     sappend('v',SET_F("MA"),strip.ablMilliampsMax); | ||||
|     sappend('v',SET_F("LA"),strip.milliampsPerLed); | ||||
|     if (strip.currentMilliamps) | ||||
|     { | ||||
|       sappends('m',SET_F("(\"pow\")[0]"),(char*)""); | ||||
|       olen -= 2; //delete "; | ||||
|       oappendi(strip.currentMilliamps); | ||||
|       oappend(SET_F("mA\";")); | ||||
|     } | ||||
|  | ||||
|     sappend('v',SET_F("CA"),briS); | ||||
|  | ||||
|     sappend('v',SET_F("AW"),strip.rgbwMode); | ||||
|  | ||||
|     sappend('c',SET_F("BO"),turnOnAtBoot); | ||||
|     sappend('v',SET_F("BP"),bootPreset); | ||||
|  | ||||
|     sappend('c',SET_F("GB"),strip.gammaCorrectBri); | ||||
|     sappend('c',SET_F("GC"),strip.gammaCorrectCol); | ||||
|     sappend('c',SET_F("TF"),fadeTransition); | ||||
|     sappend('v',SET_F("TD"),transitionDelayDefault); | ||||
|     sappend('c',SET_F("PF"),strip.paletteFade); | ||||
|     sappend('v',SET_F("BF"),briMultiplier); | ||||
|     sappend('v',SET_F("TB"),nightlightTargetBri); | ||||
|     sappend('v',SET_F("TL"),nightlightDelayMinsDefault); | ||||
|     sappend('v',SET_F("TW"),nightlightMode); | ||||
|     sappend('i',SET_F("PB"),strip.paletteBlend); | ||||
|     sappend('c',SET_F("SL"),skipFirstLed); | ||||
|     sappend('v',SET_F("RL"),rlyPin); | ||||
|     sappend('c',SET_F("RM"),rlyMde); | ||||
|     sappend('v',SET_F("BT"),btnPin); | ||||
|     sappend('v',SET_F("IR"),irPin); | ||||
|   } | ||||
|  | ||||
|   if (subPage == 3) | ||||
|   { | ||||
|     sappends('s',SET_F("DS"),serverDescription); | ||||
|     sappend('c',SET_F("ST"),syncToggleReceive); | ||||
|   } | ||||
|  | ||||
|   if (subPage == 4) | ||||
|   { | ||||
|     sappend('c',SET_F("BT"),buttonEnabled); | ||||
|     sappend('v',SET_F("IR"),irEnabled); | ||||
|     sappend('v',SET_F("UP"),udpPort); | ||||
|     sappend('v',SET_F("U2"),udpPort2); | ||||
|     sappend('c',SET_F("RB"),receiveNotificationBrightness); | ||||
|     sappend('c',SET_F("RC"),receiveNotificationColor); | ||||
|     sappend('c',SET_F("RX"),receiveNotificationEffects); | ||||
|     sappend('c',SET_F("SD"),notifyDirectDefault); | ||||
|     sappend('c',SET_F("SB"),notifyButton); | ||||
|     sappend('c',SET_F("SH"),notifyHue); | ||||
|     sappend('c',SET_F("SM"),notifyMacro); | ||||
|     sappend('c',SET_F("S2"),notifyTwice); | ||||
|  | ||||
|     sappend('c',SET_F("NL"),nodeListEnabled); | ||||
|     sappend('c',SET_F("NB"),nodeBroadcastEnabled); | ||||
|  | ||||
|     sappend('c',SET_F("RD"),receiveDirect); | ||||
|     sappend('v',SET_F("EP"),e131Port); | ||||
|     sappend('c',SET_F("ES"),e131SkipOutOfSequence); | ||||
|     sappend('c',SET_F("EM"),e131Multicast); | ||||
|     sappend('v',SET_F("EU"),e131Universe); | ||||
|     sappend('v',SET_F("DA"),DMXAddress); | ||||
|     sappend('v',SET_F("DM"),DMXMode); | ||||
|     sappend('v',SET_F("ET"),realtimeTimeoutMs); | ||||
|     sappend('c',SET_F("FB"),arlsForceMaxBri); | ||||
|     sappend('c',SET_F("RG"),arlsDisableGammaCorrection); | ||||
|     sappend('v',SET_F("WO"),arlsOffset); | ||||
|     sappend('c',SET_F("AL"),alexaEnabled); | ||||
|     sappends('s',SET_F("AI"),alexaInvocationName); | ||||
|     sappend('c',SET_F("SA"),notifyAlexa); | ||||
|     sappends('s',SET_F("BK"),(char*)((blynkEnabled)?SET_F("Hidden"):"")); | ||||
|     sappends('s',SET_F("BH"),blynkHost); | ||||
|     sappend('v',SET_F("BP"),blynkPort); | ||||
|  | ||||
|     #ifdef WLED_ENABLE_MQTT | ||||
|     sappend('c',SET_F("MQ"),mqttEnabled); | ||||
|     sappends('s',SET_F("MS"),mqttServer); | ||||
|     sappend('v',SET_F("MQPORT"),mqttPort); | ||||
|     sappends('s',SET_F("MQUSER"),mqttUser); | ||||
|     byte l = strlen(mqttPass); | ||||
|     char fpass[l+1]; //fill password field with *** | ||||
|     fpass[l] = 0; | ||||
|     memset(fpass,'*',l); | ||||
|     sappends('s',SET_F("MQPASS"),fpass); | ||||
|     sappends('s',SET_F("MQCID"),mqttClientID); | ||||
|     sappends('s',SET_F("MD"),mqttDeviceTopic); | ||||
|     sappends('s',SET_F("MG"),mqttGroupTopic); | ||||
|     #endif | ||||
|  | ||||
|     #ifndef WLED_DISABLE_HUESYNC | ||||
|     sappend('v',SET_F("H0"),hueIP[0]); | ||||
|     sappend('v',SET_F("H1"),hueIP[1]); | ||||
|     sappend('v',SET_F("H2"),hueIP[2]); | ||||
|     sappend('v',SET_F("H3"),hueIP[3]); | ||||
|     sappend('v',SET_F("HL"),huePollLightId); | ||||
|     sappend('v',SET_F("HI"),huePollIntervalMs); | ||||
|     sappend('c',SET_F("HP"),huePollingEnabled); | ||||
|     sappend('c',SET_F("HO"),hueApplyOnOff); | ||||
|     sappend('c',SET_F("HB"),hueApplyBri); | ||||
|     sappend('c',SET_F("HC"),hueApplyColor); | ||||
|     char hueErrorString[25]; | ||||
|     switch (hueError) | ||||
|     { | ||||
|       case HUE_ERROR_INACTIVE     : strcpy(hueErrorString,(char*)F("Inactive"));                break; | ||||
|       case HUE_ERROR_ACTIVE       : strcpy(hueErrorString,(char*)F("Active"));                  break; | ||||
|       case HUE_ERROR_UNAUTHORIZED : strcpy(hueErrorString,(char*)F("Unauthorized"));            break; | ||||
|       case HUE_ERROR_LIGHTID      : strcpy(hueErrorString,(char*)F("Invalid light ID"));        break; | ||||
|       case HUE_ERROR_PUSHLINK     : strcpy(hueErrorString,(char*)F("Link button not pressed")); break; | ||||
|       case HUE_ERROR_JSON_PARSING : strcpy(hueErrorString,(char*)F("JSON parsing error"));      break; | ||||
|       case HUE_ERROR_TIMEOUT      : strcpy(hueErrorString,(char*)F("Timeout"));                 break; | ||||
|       default: sprintf(hueErrorString,(char*)F("Bridge Error %i"),hueError); | ||||
|     } | ||||
|      | ||||
|     sappends('m',SET_F("(\"sip\")[0]"),hueErrorString); | ||||
|     #endif | ||||
|   } | ||||
|  | ||||
|   if (subPage == 5) | ||||
|   { | ||||
|     sappend('c',SET_F("NT"),ntpEnabled); | ||||
|     sappends('s',SET_F("NS"),ntpServerName); | ||||
|     sappend('c',SET_F("CF"),!useAMPM); | ||||
|     sappend('i',SET_F("TZ"),currentTimezone); | ||||
|     sappend('v',SET_F("UO"),utcOffsetSecs); | ||||
|     char tm[32]; | ||||
|     getTimeString(tm); | ||||
|     sappends('m',SET_F("(\"times\")[0]"),tm); | ||||
|     sappend('i',SET_F("OL"),overlayCurrent); | ||||
|     sappend('v',SET_F("O1"),overlayMin); | ||||
|     sappend('v',SET_F("O2"),overlayMax); | ||||
|     sappend('v',SET_F("OM"),analogClock12pixel); | ||||
|     sappend('c',SET_F("OS"),analogClockSecondsTrail); | ||||
|     sappend('c',SET_F("O5"),analogClock5MinuteMarks); | ||||
|     sappends('s',SET_F("CX"),cronixieDisplay); | ||||
|     sappend('c',SET_F("CB"),cronixieBacklight); | ||||
|     sappend('c',SET_F("CE"),countdownMode); | ||||
|     sappend('v',SET_F("CY"),countdownYear); | ||||
|     sappend('v',SET_F("CI"),countdownMonth); | ||||
|     sappend('v',SET_F("CD"),countdownDay); | ||||
|     sappend('v',SET_F("CH"),countdownHour); | ||||
|     sappend('v',SET_F("CM"),countdownMin); | ||||
|     sappend('v',SET_F("CS"),countdownSec); | ||||
|  | ||||
|     sappend('v',SET_F("A0"),macroAlexaOn); | ||||
|     sappend('v',SET_F("A1"),macroAlexaOff); | ||||
|     sappend('v',SET_F("MP"),macroButton); | ||||
|     sappend('v',SET_F("ML"),macroLongPress); | ||||
|     sappend('v',SET_F("MC"),macroCountdown); | ||||
|     sappend('v',SET_F("MN"),macroNl); | ||||
|     sappend('v',SET_F("MD"),macroDoublePress); | ||||
|  | ||||
|     char k[4]; | ||||
|     k[2] = 0; //Time macros | ||||
|     for (int i = 0; i<8; i++) | ||||
|     { | ||||
|       k[1] = 48+i; //ascii 0,1,2,3 | ||||
|       k[0] = 'H'; sappend('v',k,timerHours[i]); | ||||
|       k[0] = 'N'; sappend('v',k,timerMinutes[i]); | ||||
|       k[0] = 'T'; sappend('v',k,timerMacro[i]); | ||||
|       k[0] = 'W'; sappend('v',k,timerWeekday[i]); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (subPage == 6) | ||||
|   { | ||||
|     sappend('c',SET_F("NO"),otaLock); | ||||
|     sappend('c',SET_F("OW"),wifiLock); | ||||
|     sappend('c',SET_F("AO"),aOtaEnabled); | ||||
|     sappends('m',SET_F("(\"sip\")[0]"),(char*)F("WLED ")); | ||||
|     olen -= 2; //delete "; | ||||
|     oappend(versionString); | ||||
|     oappend(SET_F(" (build ")); | ||||
|     oappendi(VERSION); | ||||
|     oappend(SET_F(")\";")); | ||||
|   } | ||||
|    | ||||
|   #ifdef WLED_ENABLE_DMX // include only if DMX is enabled | ||||
|   if (subPage == 7) | ||||
|   { | ||||
|     sappend('v',SET_F("PU"),e131ProxyUniverse); | ||||
|      | ||||
|     sappend('v',SET_F("CN"),DMXChannels); | ||||
|     sappend('v',SET_F("CG"),DMXGap); | ||||
|     sappend('v',SET_F("CS"),DMXStart); | ||||
|     sappend('v',SET_F("SL"),DMXStartLED); | ||||
|      | ||||
|     sappend('i',SET_F("CH1"),DMXFixtureMap[0]); | ||||
|     sappend('i',SET_F("CH2"),DMXFixtureMap[1]); | ||||
|     sappend('i',SET_F("CH3"),DMXFixtureMap[2]); | ||||
|     sappend('i',SET_F("CH4"),DMXFixtureMap[3]); | ||||
|     sappend('i',SET_F("CH5"),DMXFixtureMap[4]); | ||||
|     sappend('i',SET_F("CH6"),DMXFixtureMap[5]); | ||||
|     sappend('i',SET_F("CH7"),DMXFixtureMap[6]); | ||||
|     sappend('i',SET_F("CH8"),DMXFixtureMap[7]); | ||||
|     sappend('i',SET_F("CH9"),DMXFixtureMap[8]); | ||||
|     sappend('i',SET_F("CH10"),DMXFixtureMap[9]); | ||||
|     sappend('i',SET_F("CH11"),DMXFixtureMap[10]); | ||||
|     sappend('i',SET_F("CH12"),DMXFixtureMap[11]); | ||||
|     sappend('i',SET_F("CH13"),DMXFixtureMap[12]); | ||||
|     sappend('i',SET_F("CH14"),DMXFixtureMap[13]); | ||||
|     sappend('i',SET_F("CH15"),DMXFixtureMap[14]); | ||||
|     } | ||||
|   #endif | ||||
|   oappend(SET_F("}</script>")); | ||||
| } | ||||
		Reference in New Issue
	
	Block a user
	 fishbone-git
					fishbone-git