 46a3e3d353
			
		
	
	46a3e3d353
	
	
	
		
			
			* cleanup * added collision handling back in update from another commit that got lost * reformat * added latest version of functions this somehow also got lost from an earlier commit * Update platformio.ini * updated particle box and firework effects particle box now is more random in random mode (still a work in progress) firework is now more configurable by sliders * added preliminary functions and FX * added particle attractor, added two new FX but still buggy particle attractor animation does not work, somthing wrong with pointer allocation, it worked with static variables * bugfixes, attracot now works still unknown, why more than 256 particles are needed in memory allocation to not make it crash, but it works for now * added angle emitter, added fireworks effect using it * Added saturation to be set for each particle individually at the expense of more ram usage, animations now have more options for color control (already used in fireworks now) * bugfixes in impact animation * added option to use fast color add, may improve performance a little also fixed a bug in fire animation * collision detection is now a function plus some improvements & fixes * improved collision efficiency improved efficiency for stackup (pushback), added code to correctly determine direction if particles meed (probably overkill but now its there) * added fix for piling oscillations untested, need to verify it works * Improved collision handling (faster, less oscillations), changed variables to 32bit for faster calculation 32bit variables are faster on ESP32, so use them whenever a variable is used a lot, it saves one instruction per access. * removed option for fastcolor add it made very little difference in performance, but for ESP8266 it may matter so it is set permanently there. graphically the difference is also very small (sometimes a particle gets brighter or less saturated) * Fixed some bugs in particle system, runs much smoother now also tweaked some of the FX * Bugfix in particle push, now piling is working again particle pile-up did not work correctly, now fixed * changed particle pile demo into waterfall plus some tweaks * lots of bugfixes * updated particle attractor animation parameters and added more user options to perlin noise * removed TODOs * added particle WrapUpdate function, added spray FX, some bugfixes * fixed touch buttons for ESP32 S2 and S3 touch is implemented differently on S2 and S3, these changes make touch buttons work on S2 and S3 * some tuning for touch buttons on S2/S3 now better fits the default threshold value of 32 * updated rotating particle spray with more user options * chaned rotating spray default parameters * add todo * Revert "some tuning for touch buttons on S2/S3" This reverts commitd21ad8e7d1. * Revert "fixed touch buttons for ESP32 S2 and S3" This reverts commit0904155186. * removed comments, added comments * cleanup -removed wrap_update function, now integrated into move_update -added more 'out of bounds' checking in fire functions, may speed it up a little * cleanup session -removed particle box 'rocking boat' (buggy) and replaced with random sloshing. -removed comments -changed some variables into 32bit for speed boost on ESP32 -added link to KB at the top of FX.cpp * cleanup removed / added comments * added particle GEQ effect (untested) * Particle GEQ fixes, it now actually works * GEQ FX parameter tuning * added rotating GEQ, work in progress -animation works but sliders are too sensitive. need to adjust the ranges * FX update - changed firework exhaust to low saturation - updated rotating particle spray animation * Cleanup & Bugfixes plus major improvements for ESP8266 -added particle reductions for ESP8266 for all FX -Removed collisions from Particle Perlin Noise FX, slows things down and does not contribute to a better effect experience -lots of optimizations for ESP8266, all FX now work (at least on 160MHz but still slow) -Some bugfixes -removed unused variables to make compiler happy * Particle FX Rename, default parameter tuning, bugfix -Now shorter names, 'PS' in front to filter the list -Tuned default parameters to make them look better by default -Bugfix in particle system (removed duplicate application of velocity) -reduced PS fire RAM usage (less particles, less base flames, no noticeable difference) -some variable renaming * slight speed improvements in fire, like 1-2FPS * Big update: lots of little fixes and big speed boost on fire animation -fixed fire burning more on the left side -fixed crash in particle attractor -added many improvements for ESP8266 -improved particle rendering efficiency -efficiency improvements in general -changed the way fire is rendered, now more than 2x faster -re-tuned fire to new rendering, also seperately tuned it for ESP8266 -changed all random8() to random16() as it runs faster on ESPs -some reformating -some renaming of effect stuff -fine tuning on falling particle effect -improvements to collision handling (faster and better) -added a (temporary) function for speed tests, will be removed again * bugfix * updated PS Fireworks with many changes and fine-tuning of parameters -removed spiral explosions -added more versatility to circular explosions -removed user selectable amount of rockets -tuned explosion size of circular explosions to match random explosions (more or less, may need improvement) -changed order of sliders in volcano animation * put particle system in a class. work in progress. another huge update to the particle system. went through the whole code, rewrote many of the functions. many improvements over the previous code. fixed many bugs (and even an ancient one in rendering function). spent many hours optimizing the code for speed and usability. still a work in progress, debugging is ongoing. more updates to come. * many (many!) bugfixes, added fire FX back in (and improved it a lot) added local renderbuffer for huge speed boost -lots of bugfixes in Particle system -added local rendering buffer (renders to buffer in heap) -added fast and accurate color-add function -bugfixes in render function -added improved 'sticky' particles in collision (unfinished business) -added ballpit animation back -lots of tweaks to fire FX and fire rendering functions, looks even better now (still unfinished) -added palette render option to fire still many debug print outputs around, needs cleanup at one point * More Bugfixes, more converted FX * more bugfixes, some animation tuning, added volcano back in - fixed animation transitions with proper pointer setting in each FX call - added color by age setting to particle system - maybe fixed collision having a tendency to go to the left - fixed bugs in fireworks FX - added fire spread width as a slider - changed max number of particles and sprays based on some ram calculations - some other random fixes * updated fire, added some functions to PS, ported attractor FX - added turbulance to fire (after hours of fine-tuning) it now looks even more awesome - added attractor animation back in and improved it with more functionality - fixed the attractor function in the PS - renamed FX: 'candy' is now called 'vortex' - added new force function to apply a force immediately (and handle the timing in the FX) - added update function to PS for size update to handle dynamic segment size change - made waterfall width dynamic on segment width - removed some debug stuff - added #defines for maximum number of particles/sprays - updated fire parameter to make it look better on ESP8266 - some little bugfixes * another huge update, many improvements, fine-tune collision, fire and some other FX -removed classic fire render as palette now looks much better -tweaked fire parameters to more awesome -added (optional) turbulence to fire using perlin-noise -ported spray FX to use PS-class -now definitely fixed asymmetrical collision issue: do not use bitshifts on negative numbers! -changed piling behaviour of particles, full rework. looks way more natural now and works much better -changed bouncing behavour: they now bounce at full diameter, making them stay fully in frame when laying on the floor -replaced all relevant bitshifts with divisions for higher accuracy throughout the PS -added new modes to particle box FX -changed a lot of FX parameters (finetuning) -changed all config strings to proper settings (matrix only) -fixed newly introduced bugs -added speedup/slowdown to vortex FX (aka candy, aka rotating sprays) -some renaming -fixed bugs… lots of bugs -merged rendering functions, removed obsolete stuff * Fixed Speed limit limiting speed was incorrect, leading to overflows. fixed this. also fixed bugs in GEQ, removed some debug stuff, added FPS limit to fire (just en experiment) * work in progress, added motion blur and line attracto (non working) * work in progress, added test function to increase particle size, also added wobbling test * added more tests, non compiling at the moment * in the middle of fixing FX to use new PS functions * debugging going on * bugfix in wrap function and firwork FX * added local render blurring, boosting FPS, work in progress - changed fast-add function to use pointers - added fast-scaling function - added simple (but fast) 2D blurring function test shows that local blurring of full frame is almost double the speed (40FPS now is 80FPS). lots of comments still there * Bugfixes, improvements and added wall roughness setting - fixed bug in PS fuzzy noise which made it asymmetric for some reason, seems to work better now - added particle size option to attractor but had to remove speed setting (now fixed emit speed) - some parameter tuning of FX - improvements to code size in render function - added smear option to blurring (not tested much, may be buggy without smear) - speed improvement to caldForce_dv, added zero checking. * removed zero inits - removed zero initialisations in FX, segment.data is set to zero by alloc function * added individual size particle rendering plus some fixes - advanced particles can now be rendered to individual sizes. It is computationally intensive but it works well for up to 15 very large particles and more smaller ones - added collision handling for individual sizes (walls and collisions) - fixed bugs in particlebox - fixed fire not transitioning properly (flickering) when frame skip is active - removed 'wraparound' function as it can easily be done by casting to an unsigned and then modulo - fixed ballpit particles wandering left and right very fast if wrapX is set * Fixed some nasty memory bugs, fixed some FX parameters - fixed memory alignment bug in PS pointer assignment by making sure only multiples of 4 are allowed for particles and sources - added saturation back to particle struct, as it was aligned to 10 bytes anyway. - fixed a bug where a null pointer could be accessed - fixed rendering out of frame particles if buffer allocation failed - improvements on ESP8266 * Replaced Ghost Rider FX with PS version - new FX is kept close to original animation but added more user settings * added walls to ghostride, fixed some bugs * added 'perpetual' flag to particles * Fixed another memory / pointer bug, but there is still one left... -also some minor fixes * Found and fixed crashes, it was wrongly assigned pointers * Added advanced particle size control and new Blob FX - advanced size control allows for growing, shrinking, wobbling - render function updated to support asymmetric rendering - various code improvements and bugfixes - some FX parameter tuning - bugfix: removed sli() sei() calls in render function that caused random flickering on S3/C3 (may add that back in but only for ESP8266 to reduce fragmentation) - removed some debug / test stuff * code cleanup, removed some unused stuff * bugfix, sprayEmit() would not return and get stuck - forgot default return value... * remove esp_random() as it is not supported on ESP8266 * small bugfix for ESP32, cleanup / reformatting * added WLED_DISABLE_PARTICLESYSTEM option, fixed volcano movement use '-D WLED_DISABLE_PARTICLESYSTEM' to disable compiling * a line got lost in the last commit * - added WLED_DISABLE_PARTICLESYSTEM option & cleanup & bugfixes - cleanup / reformatting - fixed volcano movement - small bugfix for ESP32 (random() does not work, using random16() ) * Added PS source option to emit more than one particle, addes AR to Blobs and Spray * added center GEQ back in, changed AR behaviour of spray a little center GEQ is a mix between the PS equalizer and vortex, tuned for AR. some more tuning may be needed, it can probably be extended and improved a little. * Bugfix and minor improvements - fixed bug in center GEQ - added '2D washing machine' mode for particle box - improved color-change rate in ghostrider - added AR to attractor (experimental, may remove again) * hacked in a 1D port of the 2D PS, untested code * fixed first bugs in 1D system, added test FX * fixed bug in wrapping (also was wrong for 2D system) * fixed collisions, added bouncing ball replacement (work in progress) * replaced more FX, some tweaks to 1D PS - changed virtual particle size from 64 to 32, making them able to move faster - added 'rolling balls' option to bouncing balls (i.e. no gravity plus some tweaks) - added dancing shadows with PS (work in progress) - temporarily removed native FX: bouncing balls, rolling balls, popcorn, dancing shadows -and of course some bugfixes * updated dancing shadows to more closely mimic original * added drip FX, added #ifdefs, removed some common functions #ifdefs to individually disable 1D and 2D system. * fixed compile issue, removed replaced FX (again) * added Fireworks1D replacement (unfinished) also fixed a bug in `applyForce()` using uninitialized variable. * added sparkler and hourglass (both work in progress) tried fixing 1D collisions * many improvements in 1D collisions (fine tuned), hour glass FX is almost done spent A LOT of time in fine-tuning collisions for best stacking without collipsing, ringing, oscillations or particle flips. still not perfect but probably as good as it can be with the limited time-resolution * many bugfixes in PS and FX * minor FX adjustments * Improved 1D collisions, added 2 new 1D FX (work in progress) * bugfixes * added 'color by position' as a PS setting -made settings struct for 1D and 2D version to be different -added setting 'color by position' as that is used by multiple FX now * added large size rendering to 1D PS, work in progress * bugfix (forgot to free memory) also: made random size a permanent option in 1D fireworks as it looks kind of cool * Added PS based Chase, fixed some bugs * minor update * two new FX: Starburst and 1D GEQ * Added more advanced (and proper) size handling in collisions, work in progress - copied and adapted some stuff from 2D system (out of bounds size rendering, dynamic collision detection distance, dynamic wall bounce distance) * fixed some parameters in 1D FX * added #ifndef to disable FX replaced by PS - define DISABLE_1D_PS_REPLACEMENTS or DISABLE_2D_PS_REPLACEMENTS * bugfixes * default parameter change * explicit cast to fix compile error * Bugfixes * added color-waves to PS chase, some parameter tuning plus bugfixes * FX update and bugfixes - fixed wall collisions on larger particle sizes - update to bouncing balls: - added size setting - added proper updating on parameter change - bugfixes added support for 'colorwaves' and 'pride' to PS Chase to save on code size. currently broken... * minor update to rolling balls, fixed stuff in Chase, pride option still not working * update on 'pride' now working again but needs finetuning * SEGMENT -> SEGENV on data, aux0, aux1, step also removed debug outputs * added pride and colorwaves to 1D replacement list * removed unused variables * added variable to set number of particles if not all are required (saves ram) also fixed a nasty render bug * added preliminary 1D fire function, unfinished * added fractal FX test, fixed bugs in emit function * tweaked 1D fire a little, still work in progress * minor tuning on 1D fire to avoid 'oscillation' at start * improved 1D particle fire, much more natural now. * updated 2D fire, fixed init bug - source init was wrong making fire init weirdly - changed parameters on 2D fire making it look better and improving flame height for larger setups * fixed 'nervours' fire * slight improvements and some cleanup * removed douplicate code for fire emit (tradeoff for some speed), removed unfinished fractal FX - with the duplicate code removed, the fire particle emits are a bit slower but it saves on code. it is not really noticeable, its slower by about 1FPS * Re-license from MIT to EUPL * Started cleanup, speed improvement to rendering, renamed class - removed 'smar' parameter in blur functions as smear is always used - improved particle rendering (passing by reference, passing wrap parameters for faster access) - renamed class to ParticleSystem2D - removed some whitespaces - some reformating, removed some comments - minor tweaks - removed non-working line-attractor function * Optimization and bugfixes - added out of bounds checking function - fixed rendering at boundaries - various small changes and cleanup - bugfixes with comparing ints of different signeness - fixed bounce radius * Cleanup, bugfixes, speed improvements, refactoring - lots of code refactoring / reformating - fixed bug in fire particle update and improved speed a bit - refactoring of pixel rendering - removed 2D buffers, replaced with 1D buffer for faster access - bugfix in blur2D - many small improvements to 2D system * Refactoring and cleanup of 1D PS, removed debug printouts - code reformating for consistency and readability - replaced "'?" operator with min / max (produces the same code, more readable) - minor changes for speed (random16 used, can later be replaced with hardware random function) - renamed PS Equalizer to PS 2D GEQ * Cleanup and some refactoring of particle FX - moved 2D / length check to PS initi - code reformating - removed some comments * missed a spot * minor tweaks to fast_color_add() * Work in progress: update to fireworks, needs more testing & finetuning also bugs fixed and some minor cleanup * Work in Progress: added particle memory manager for transitions - uses only one, persistant buffer to render all present particle systems - Buffer for particles is shared for one segment (may allow interacitng systems in the future) - updated some of the FX to handle the new transfer, 1D FX still not done - updated parameters for particle impact FX * work in progress: lots of changes & fixes, transitions now work correctly in 2D - still fragile code with lots of cleanup to do * added transitions for big-size rendering - transitions now work with FX that use global large size rendering. this is done by handling the buffer correctly. to avoid creating a second buffer, it is transferred back and forth to the segment. this is a bit slow but a compromise solution. - multiple segment handling is still untested * speed improvement to fast_color_add, added preliminary buffer transfer function * merge fixes * bugfix in setCurrentPalette, corrected order of 2D memory allocation, increased max particles/sources - fixed timing for vortex * fixed particle buffer handover, added 2D blur option, increased particle limits, various fixes - fixed bug in memory watchdog, now works and also works when freezing segments - fixed PS Box to work again with particle handover - added smear blurring to waterfall - added smear blurring to vortex, fixed color distribution, removed random color distribution option - replaced all random() calls with hw_random() - transitions now work but only if FX transitions are enabled and not when chaning segment size * implemented correct 1D<->2D buffer handover+bugfixes work in progress * added smear to fuzzy noise, some cleanup * added blurring option to PS 2D fireworks, work in progress in finding that nasty bug that crashes all * BUGFIX: finally found and squased the memory bug, no more random crashes also: removed debug outputs and minor cleanup * minor cleanup and fixed a bug 1D system had a lot of crashes, some out of bounds memory issue. fixed it but I have no idea which change did that, maybe the order in 1D initialization * BUGFIX: out of bounds checking was wrong, leading to crashes * better handling of used particles in transitions for FX that do not use all particles, transitions were constantly shifting the pointer, resulting in newly generated particles and weird particle flickering, this is not fixed by keeping the pointer constant once the number of used particles is reached. * improved particle transition for low-count FX, updated 1D FX now most 1D FX work again, still needs some fine-tuning * Rework on Fireworks 1D, some minor fixes - fireworks 1D now matches as closely as possible to the old FX still not identical but as good as it gets. * bugfix in PS Chase, increased brightness of 1D fireworks * replaced sin16/cos16 with new versions * bugfixes * fixed hourglass init, rearranged PS FX init * speed optimization - moved out of bounds calculation to proper spot - removed repeated allocation attempts if initial buffer alloc fails * Added Sonic Stream FX, some cleanup, bugfix in emitter * changed TTL to brightness calculation, added collision binning - brightness is now doubled as some FX were really dim due to low TTL - added binning in x-direction giving a huge speed advantage on larger matrix sizes * collision binning bugfix, made particles less sticky, update to waterfall - updated waterfall intensity and blurring * WIP: fixed 1D system for over 1000 pixels, added collision binning for 1D - making x coordinate 32bit allows for larger strips but uses a lot of ram due to struct memory alignment (12bytes instead of 8 bytes), this needs some more work to properly fix. - adding collision binning significantly speeds things up, about a factor of 2 on most FX using collision - there are still some bugs in FX or 1D memory handling (or both) on large setups * moved particle flags to seperate array to save on RAM, refactoring - moving the flags optimizes ram alignment, saving memory for each particle - changed lots of parameters to `const` - moved fire intensity to a variable instead of passing it to every render() call - changed passing pointers to passing reference where possible - saves a total of 340 bytes of flash * removed todos after some checks, minor improvements * inverted y axis in 2D render, 1D collision improvements, cleanup and fixes - inverting the y-axis in the buffer instead of in buffer transfer fixes the FX flipping when transitiononing 1D<->2D - improved particle binning for collisions - added hard-pushing also when not using gravity so piles close to an edge dont collapse - some improvments to "balance" FX - renaming and cleanup * FX fixes and minor tweaks * increased min rockets * fixed #ifdefs * another fix * revert unnecessary changes to base code * merge fix * minor tweak * tweaked sparkler FX, some cleanup * Fix in volcano FX, changed blurring of volcano and waterfall * added ifdefs * minor tweaks, increased 1D minsurfacehardness * improved sparkler FX, made overlay possible (1D not yet), cleanup * cleanup, now using new hsv2rgb/rgb2hsv, add overlay rendering to 1D - new hsv2rgb is a tiny bit faster - removed redundant code (using transferBuffer instead) - tweked parameters in vortex, removed unneeded slider defaults * fix for non or partially overlapping segments - can not use overlay rendering if no segment clears the buffer, it will all add up to white eventually - now additive transfer i.e. overlay mode is only activated if the segment is fully overlapping an underlying segment * change got lost... * increased sparkler intensity, some cleanup * replaced #ifdefs, removed 1D replacements that have 2D version, removed notes * cleanup, improvements to PS bouncing ball, replaced multicomet - bouncing balls is now named pinball and has settings/parameters updated to be a replacement for multicomet * cleanup, improvements, bugfixes - large size rendering now works without framebuffer - background adding is now done on buffer (if avilable) instead of segment - fixed overflow bug in 2D large particle rendering (it worked only for powers of 2 size) * potential bugfix, compiler warning fix * removed colorwaves and pride option from chase (not worthy replacements) * updated #defines, removed PS from 1M and 2M builds * Adding Particle System and PS FX Adding all 187 commits from particle system dev branch * reverted some accidental changes * reverted some accidental changes * merge fixes * changed replacement: multicomet instead of comet (lighthouse) * changed replacement: multicomet instead of comet (lighthouse) * disable 2D PS for ESP8266, some cleanup, improved pinball FX parameters, bugfixes * Improved collision binning for large particles, improved pinball FX * improved speed handling in pinball FX - rolling * disable 2D PS for ESP8266, some cleanup, improved pinball FX parameters, bugfixes * improved hourglass: millis instead of frame timing, better stacking. and cleanup. * revert whitespaces * simplified 1D collisions, improved binning for larger particles - ran a lot of experiments with collisions in 1D, the new much simpler approach seems to be a good compromise with regards to stacking and normal collisions. * Improved collision binning for large particles, improved pinball FX * improved speed handling in pinball FX - rolling * improved hourglass: millis instead of frame timing, better stacking. and cleanup. * revert whitespaces * prohibit use of 1D and 2D system simultaneously on ESP8266 * prohibit use of 1D and 2D system simultaneously on ESP8266 * update to handle blending styles * fixed bugs, improved new transition handling * updated 1D system to work with new transitions, replaced NULL with nullptr * merge fixes * added single pixel particle rendering for 2D system, adjusted FX to work with it * improved collisions in 1D and 2D, some bugfixes in radius calculation, minor tweaks - collisions are now also velocity based in 1D, there was a bug that prevented that from working well (wrong collision distance calculation) - improvement and bugfix in 2D collision distance calculation - added distance based pushing in 2D (instead of only using the dotproduct) the combination of improved distance calculation and proper pushing make collisions a lot better in all tested FX * minor fix * fixed overlay detection, checking for partial overlay if a PS FX is partially overlapping, it will render in overlay mode * better blur range in PS Firworks * minor code consolidation * updated 1D collisions (yet again), improved 2D collision speed - slight improvement to 2D collision code efficiency - added faster "division" to C3/ESP8266 by using a right shift trick (biasing towards 0 also for negative numbers by applying proper rounding) * minor tweak in PS balance
		
			
				
	
	
		
			1042 lines
		
	
	
		
			52 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1042 lines
		
	
	
		
			52 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #pragma once
 | |
| /*
 | |
|   WS2812FX.h - Library for WS2812 LED effects.
 | |
|   Harm Aldick - 2016
 | |
|   www.aldick.org
 | |
| 
 | |
|   Copyright (c) 2016  Harm Aldick
 | |
|   Licensed under the EUPL v. 1.2 or later
 | |
|   Adapted from code originally licensed under the MIT license
 | |
| 
 | |
|   Modified for WLED
 | |
| 
 | |
|   Segment class/struct (c) 2022 Blaz Kristan (@blazoncek)
 | |
| */
 | |
| 
 | |
| #ifndef WS2812FX_h
 | |
| #define WS2812FX_h
 | |
| 
 | |
| #include <vector>
 | |
| #include "wled.h"
 | |
| 
 | |
| #include "const.h"
 | |
| #include "bus_manager.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 DEFAULT_C1         (uint8_t)128
 | |
| #define DEFAULT_C2         (uint8_t)128
 | |
| #define DEFAULT_C3         (uint8_t)16
 | |
| 
 | |
| #ifndef MIN
 | |
| #define MIN(a,b) ((a)<(b)?(a):(b))
 | |
| #endif
 | |
| #ifndef MAX
 | |
| #define MAX(a,b) ((a)>(b)?(a):(b))
 | |
| #endif
 | |
| 
 | |
| //color mangling macros
 | |
| #ifndef RGBW32
 | |
| #define RGBW32(r,g,b,w) (uint32_t((byte(w) << 24) | (byte(r) << 16) | (byte(g) << 8) | (byte(b))))
 | |
| #endif
 | |
| 
 | |
| extern bool realtimeRespectLedMaps; // used in getMappedPixelIndex()
 | |
| extern byte realtimeMode;           // used in getMappedPixelIndex()
 | |
| 
 | |
| /* Not used in all effects yet */
 | |
| #define WLED_FPS         42
 | |
| #define FRAMETIME_FIXED  (1000/WLED_FPS)
 | |
| #define FRAMETIME        strip.getFrameTime()
 | |
| #if defined(ARDUINO_ARCH_ESP32) && !defined(CONFIG_IDF_TARGET_ESP32C3) && !defined(CONFIG_IDF_TARGET_ESP32S2)
 | |
|   #define MIN_FRAME_DELAY  2                                              // minimum wait between repaints, to keep other functions like WiFi alive 
 | |
| #elif defined(CONFIG_IDF_TARGET_ESP32S2) || defined(CONFIG_IDF_TARGET_ESP32C3)
 | |
|   #define MIN_FRAME_DELAY  3                                              // S2/C3 are slower than normal esp32, and only have one core
 | |
| #else
 | |
|   #define MIN_FRAME_DELAY  8                                              // 8266 legacy MIN_SHOW_DELAY
 | |
| #endif
 | |
| #define FPS_UNLIMITED    0
 | |
| 
 | |
| // FPS calculation (can be defined as compile flag for debugging)
 | |
| #ifndef FPS_CALC_AVG
 | |
| #define FPS_CALC_AVG 7 // average FPS calculation over this many frames (moving average)
 | |
| #endif
 | |
| #ifndef FPS_MULTIPLIER
 | |
| #define FPS_MULTIPLIER 1 // dev option: multiplier to get sub-frame FPS without floats
 | |
| #endif
 | |
| #define FPS_CALC_SHIFT 7 // bit shift for fixed point math
 | |
| 
 | |
| /* each segment uses 82 bytes of SRAM memory, so if you're application fails because of
 | |
|   insufficient memory, decreasing MAX_NUM_SEGMENTS may help */
 | |
| #ifdef ESP8266
 | |
|   #define MAX_NUM_SEGMENTS  16
 | |
|   /* How much data bytes all segments combined may allocate */
 | |
|   #define MAX_SEGMENT_DATA  5120
 | |
| #elif defined(CONFIG_IDF_TARGET_ESP32S2)
 | |
|   #define MAX_NUM_SEGMENTS  20
 | |
|   #define MAX_SEGMENT_DATA  (MAX_NUM_SEGMENTS*512)  // 10k by default (S2 is short on free RAM)
 | |
| #else
 | |
|   #define MAX_NUM_SEGMENTS  32  // warning: going beyond 32 may consume too much RAM for stable operation
 | |
|   #define MAX_SEGMENT_DATA  (MAX_NUM_SEGMENTS*1280) // 40k by default
 | |
| #endif
 | |
| 
 | |
| /* How much data bytes each segment should max allocate to leave enough space for other segments,
 | |
|   assuming each segment uses the same amount of data. 256 for ESP8266, 640 for ESP32. */
 | |
| #define FAIR_DATA_PER_SEG (MAX_SEGMENT_DATA / strip.getMaxSegments())
 | |
| 
 | |
| #define NUM_COLORS       3 /* number of colors per segment */
 | |
| #define SEGMENT          strip._segments[strip.getCurrSegmentId()]
 | |
| #define SEGENV           strip._segments[strip.getCurrSegmentId()]
 | |
| #define SEGCOLOR(x)      Segment::getCurrentColor(x)
 | |
| #define SEGPALETTE       Segment::getCurrentPalette()
 | |
| #define SEGLEN           Segment::vLength()
 | |
| #define SEG_W            Segment::vWidth()
 | |
| #define SEG_H            Segment::vHeight()
 | |
| #define SPEED_FORMULA_L  (5U + (50U*(255U - SEGMENT.speed))/SEGLEN)
 | |
| 
 | |
| // some common colors
 | |
| #define RED        (uint32_t)0xFF0000
 | |
| #define GREEN      (uint32_t)0x00FF00
 | |
| #define BLUE       (uint32_t)0x0000FF
 | |
| #define WHITE      (uint32_t)0xFFFFFF
 | |
| #define BLACK      (uint32_t)0x000000
 | |
| #define YELLOW     (uint32_t)0xFFFF00
 | |
| #define CYAN       (uint32_t)0x00FFFF
 | |
| #define MAGENTA    (uint32_t)0xFF00FF
 | |
| #define PURPLE     (uint32_t)0x400080
 | |
| #define ORANGE     (uint32_t)0xFF3000
 | |
| #define PINK       (uint32_t)0xFF1493
 | |
| #define GREY       (uint32_t)0x808080
 | |
| #define GRAY       GREY
 | |
| #define DARKGREY   (uint32_t)0x333333
 | |
| #define DARKGRAY   DARKGREY
 | |
| #define ULTRAWHITE (uint32_t)0xFFFFFFFF
 | |
| #define DARKSLATEGRAY (uint32_t)0x2F4F4F
 | |
| #define DARKSLATEGREY DARKSLATEGRAY
 | |
| 
 | |
| // segment options
 | |
| #define NO_OPTIONS   (uint16_t)0x0000
 | |
| #define TRANSPOSED   (uint16_t)0x0100 // rotated 90deg & reversed
 | |
| #define MIRROR_Y_2D  (uint16_t)0x0080
 | |
| #define REVERSE_Y_2D (uint16_t)0x0040
 | |
| #define RESET_REQ    (uint16_t)0x0020
 | |
| #define FROZEN       (uint16_t)0x0010
 | |
| #define MIRROR       (uint16_t)0x0008
 | |
| #define SEGMENT_ON   (uint16_t)0x0004
 | |
| #define REVERSE      (uint16_t)0x0002
 | |
| #define SELECTED     (uint16_t)0x0001
 | |
| 
 | |
| #define FX_MODE_STATIC                   0
 | |
| #define FX_MODE_BLINK                    1
 | |
| #define FX_MODE_BREATH                   2
 | |
| #define FX_MODE_COLOR_WIPE               3
 | |
| #define FX_MODE_COLOR_WIPE_RANDOM        4
 | |
| #define FX_MODE_RANDOM_COLOR             5
 | |
| #define FX_MODE_COLOR_SWEEP              6
 | |
| #define FX_MODE_DYNAMIC                  7
 | |
| #define FX_MODE_RAINBOW                  8
 | |
| #define FX_MODE_RAINBOW_CYCLE            9
 | |
| #define FX_MODE_SCAN                    10
 | |
| #define FX_MODE_DUAL_SCAN               11
 | |
| #define FX_MODE_FADE                    12
 | |
| #define FX_MODE_THEATER_CHASE           13
 | |
| #define FX_MODE_THEATER_CHASE_RAINBOW   14
 | |
| #define FX_MODE_RUNNING_LIGHTS          15
 | |
| #define FX_MODE_SAW                     16
 | |
| #define FX_MODE_TWINKLE                 17
 | |
| #define FX_MODE_DISSOLVE                18
 | |
| #define FX_MODE_DISSOLVE_RANDOM         19  // candidate for removal (use Dissolve with with check 3)
 | |
| #define FX_MODE_SPARKLE                 20
 | |
| #define FX_MODE_FLASH_SPARKLE           21
 | |
| #define FX_MODE_HYPER_SPARKLE           22
 | |
| #define FX_MODE_STROBE                  23
 | |
| #define FX_MODE_STROBE_RAINBOW          24
 | |
| #define FX_MODE_MULTI_STROBE            25
 | |
| #define FX_MODE_BLINK_RAINBOW           26
 | |
| #define FX_MODE_ANDROID                 27
 | |
| #define FX_MODE_CHASE_COLOR             28
 | |
| #define FX_MODE_CHASE_RANDOM            29
 | |
| #define FX_MODE_CHASE_RAINBOW           30
 | |
| #define FX_MODE_CHASE_FLASH             31
 | |
| #define FX_MODE_CHASE_FLASH_RANDOM      32
 | |
| #define FX_MODE_CHASE_RAINBOW_WHITE     33
 | |
| #define FX_MODE_COLORFUL                34
 | |
| #define FX_MODE_TRAFFIC_LIGHT           35
 | |
| #define FX_MODE_COLOR_SWEEP_RANDOM      36
 | |
| #define FX_MODE_RUNNING_COLOR           37
 | |
| #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_TETRIX                  44  //was Merry Christmas prior to 0.12.0 (use "Chase 2" with Red/Green)
 | |
| #define FX_MODE_FIRE_FLICKER            45
 | |
| #define FX_MODE_GRADIENT                46
 | |
| #define FX_MODE_LOADING                 47
 | |
| #define FX_MODE_ROLLINGBALLS            48  //was Police before 0.14
 | |
| #define FX_MODE_FAIRY                   49  //was Police All prior to 0.13.0-b6 (use "Two Dots" with Red/Blue and full intensity)
 | |
| #define FX_MODE_TWO_DOTS                50
 | |
| #define FX_MODE_FAIRYTWINKLE            51  //was Two Areas prior to 0.13.0-b6 (use "Two Dots" with full intensity)
 | |
| #define FX_MODE_RUNNING_DUAL            52
 | |
| #define FX_MODE_IMAGE                   53
 | |
| #define FX_MODE_TRICOLOR_CHASE          54
 | |
| #define FX_MODE_TRICOLOR_WIPE           55
 | |
| #define FX_MODE_TRICOLOR_FADE           56
 | |
| #define FX_MODE_LIGHTNING               57
 | |
| #define FX_MODE_ICU                     58
 | |
| #define FX_MODE_MULTI_COMET             59
 | |
| #define FX_MODE_DUAL_LARSON_SCANNER     60  // candidate for removal (use Scanner with with check 1)
 | |
| #define FX_MODE_RANDOM_CHASE            61
 | |
| #define FX_MODE_OSCILLATE               62
 | |
| #define FX_MODE_PRIDE_2015              63
 | |
| #define FX_MODE_JUGGLE                  64
 | |
| #define FX_MODE_PALETTE                 65
 | |
| #define FX_MODE_FIRE_2012               66
 | |
| #define FX_MODE_COLORWAVES              67
 | |
| #define FX_MODE_BPM                     68
 | |
| #define FX_MODE_FILLNOISE8              69
 | |
| #define FX_MODE_NOISE16_1               70
 | |
| #define FX_MODE_NOISE16_2               71
 | |
| #define FX_MODE_NOISE16_3               72
 | |
| #define FX_MODE_NOISE16_4               73
 | |
| #define FX_MODE_COLORTWINKLE            74
 | |
| #define FX_MODE_LAKE                    75
 | |
| #define FX_MODE_METEOR                  76
 | |
| //#define FX_MODE_METEOR_SMOOTH           77 // merged with meteor
 | |
| #define FX_MODE_RAILWAY                 78
 | |
| #define FX_MODE_RIPPLE                  79
 | |
| #define FX_MODE_TWINKLEFOX              80
 | |
| #define FX_MODE_TWINKLECAT              81
 | |
| #define FX_MODE_HALLOWEEN_EYES          82
 | |
| #define FX_MODE_STATIC_PATTERN          83
 | |
| #define FX_MODE_TRI_STATIC_PATTERN      84
 | |
| #define FX_MODE_SPOTS                   85
 | |
| #define FX_MODE_SPOTS_FADE              86
 | |
| #define FX_MODE_GLITTER                 87
 | |
| #define FX_MODE_CANDLE                  88
 | |
| #define FX_MODE_STARBURST               89
 | |
| #define FX_MODE_EXPLODING_FIREWORKS     90
 | |
| #define FX_MODE_BOUNCINGBALLS           91
 | |
| #define FX_MODE_SINELON                 92
 | |
| #define FX_MODE_SINELON_DUAL            93
 | |
| #define FX_MODE_SINELON_RAINBOW         94
 | |
| #define FX_MODE_POPCORN                 95
 | |
| #define FX_MODE_DRIP                    96
 | |
| #define FX_MODE_PLASMA                  97
 | |
| #define FX_MODE_PERCENT                 98
 | |
| #define FX_MODE_RIPPLE_RAINBOW          99
 | |
| #define FX_MODE_HEARTBEAT              100
 | |
| #define FX_MODE_PACIFICA               101
 | |
| #define FX_MODE_CANDLE_MULTI           102
 | |
| #define FX_MODE_SOLID_GLITTER          103  // candidate for removal (use glitter)
 | |
| #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_2DPLASMAROTOZOOM       114 // was Candy Cane prior to 0.14 (use Chase 2)
 | |
| #define FX_MODE_BLENDS                 115
 | |
| #define FX_MODE_TV_SIMULATOR           116
 | |
| #define FX_MODE_DYNAMIC_SMOOTH         117 // candidate for removal (check3 in dynamic)
 | |
| 
 | |
| // new 0.14 2D effects
 | |
| #define FX_MODE_2DSPACESHIPS           118 //gap fill
 | |
| #define FX_MODE_2DCRAZYBEES            119 //gap fill
 | |
| #define FX_MODE_2DGHOSTRIDER           120 //gap fill
 | |
| #define FX_MODE_2DBLOBS                121 //gap fill
 | |
| #define FX_MODE_2DSCROLLTEXT           122 //gap fill
 | |
| #define FX_MODE_2DDRIFTROSE            123 //gap fill
 | |
| #define FX_MODE_2DDISTORTIONWAVES      124 //gap fill
 | |
| #define FX_MODE_2DSOAP                 125 //gap fill
 | |
| #define FX_MODE_2DOCTOPUS              126 //gap fill
 | |
| #define FX_MODE_2DWAVINGCELL           127 //gap fill
 | |
| 
 | |
| // WLED-SR effects (SR compatible IDs !!!)
 | |
| #define FX_MODE_PIXELS                 128
 | |
| #define FX_MODE_PIXELWAVE              129
 | |
| #define FX_MODE_JUGGLES                130
 | |
| #define FX_MODE_MATRIPIX               131
 | |
| #define FX_MODE_GRAVIMETER             132
 | |
| #define FX_MODE_PLASMOID               133
 | |
| #define FX_MODE_PUDDLES                134
 | |
| #define FX_MODE_MIDNOISE               135
 | |
| #define FX_MODE_NOISEMETER             136
 | |
| #define FX_MODE_FREQWAVE               137
 | |
| #define FX_MODE_FREQMATRIX             138
 | |
| #define FX_MODE_2DGEQ                  139
 | |
| #define FX_MODE_WATERFALL              140
 | |
| #define FX_MODE_FREQPIXELS             141
 | |
| #define FX_MODE_BINMAP                 142
 | |
| #define FX_MODE_NOISEFIRE              143
 | |
| #define FX_MODE_PUDDLEPEAK             144
 | |
| #define FX_MODE_NOISEMOVE              145
 | |
| #define FX_MODE_2DNOISE                146
 | |
| #define FX_MODE_PERLINMOVE             147
 | |
| #define FX_MODE_RIPPLEPEAK             148
 | |
| #define FX_MODE_2DFIRENOISE            149
 | |
| #define FX_MODE_2DSQUAREDSWIRL         150
 | |
| // #define FX_MODE_2DFIRE2012             151
 | |
| #define FX_MODE_2DDNA                  152
 | |
| #define FX_MODE_2DMATRIX               153
 | |
| #define FX_MODE_2DMETABALLS            154
 | |
| #define FX_MODE_FREQMAP                155
 | |
| #define FX_MODE_GRAVCENTER             156
 | |
| #define FX_MODE_GRAVCENTRIC            157
 | |
| #define FX_MODE_GRAVFREQ               158
 | |
| #define FX_MODE_DJLIGHT                159
 | |
| #define FX_MODE_2DFUNKYPLANK           160
 | |
| //#define FX_MODE_2DCENTERBARS           161
 | |
| #define FX_MODE_2DPULSER               162
 | |
| #define FX_MODE_BLURZ                  163
 | |
| #define FX_MODE_2DDRIFT                164
 | |
| #define FX_MODE_2DWAVERLY              165
 | |
| #define FX_MODE_2DSUNRADIATION         166
 | |
| #define FX_MODE_2DCOLOREDBURSTS        167
 | |
| #define FX_MODE_2DJULIA                168
 | |
| // #define FX_MODE_2DPOOLNOISE            169 //have been removed in WLED SR in the past because of low mem but should be added back
 | |
| // #define FX_MODE_2DTWISTER              170 //have been removed in WLED SR in the past because of low mem but should be added back
 | |
| // #define FX_MODE_2DCAELEMENTATY         171 //have been removed in WLED SR in the past because of low mem but should be added back
 | |
| #define FX_MODE_2DGAMEOFLIFE           172
 | |
| #define FX_MODE_2DTARTAN               173
 | |
| #define FX_MODE_2DPOLARLIGHTS          174
 | |
| #define FX_MODE_2DSWIRL                175
 | |
| #define FX_MODE_2DLISSAJOUS            176
 | |
| #define FX_MODE_2DFRIZZLES             177
 | |
| #define FX_MODE_2DPLASMABALL           178
 | |
| #define FX_MODE_FLOWSTRIPE             179
 | |
| #define FX_MODE_2DHIPHOTIC             180
 | |
| #define FX_MODE_2DSINDOTS              181
 | |
| #define FX_MODE_2DDNASPIRAL            182
 | |
| #define FX_MODE_2DBLACKHOLE            183
 | |
| #define FX_MODE_WAVESINS               184
 | |
| #define FX_MODE_ROCKTAVES              185
 | |
| #define FX_MODE_2DAKEMI                186
 | |
| #define FX_MODE_PARTICLEVOLCANO        187
 | |
| #define FX_MODE_PARTICLEFIRE           188
 | |
| #define FX_MODE_PARTICLEFIREWORKS      189
 | |
| #define FX_MODE_PARTICLEVORTEX         190
 | |
| #define FX_MODE_PARTICLEPERLIN         191
 | |
| #define FX_MODE_PARTICLEPIT            192
 | |
| #define FX_MODE_PARTICLEBOX            193
 | |
| #define FX_MODE_PARTICLEATTRACTOR      194
 | |
| #define FX_MODE_PARTICLEIMPACT         195
 | |
| #define FX_MODE_PARTICLEWATERFALL      196
 | |
| #define FX_MODE_PARTICLESPRAY          197
 | |
| #define FX_MODE_PARTICLESGEQ           198
 | |
| #define FX_MODE_PARTICLECENTERGEQ      199
 | |
| #define FX_MODE_PARTICLEGHOSTRIDER     200
 | |
| #define FX_MODE_PARTICLEBLOBS          201
 | |
| #define FX_MODE_PSDRIP                 202
 | |
| #define FX_MODE_PSPINBALL              203
 | |
| #define FX_MODE_PSDANCINGSHADOWS       204
 | |
| #define FX_MODE_PSFIREWORKS1D          205
 | |
| #define FX_MODE_PSSPARKLER             206
 | |
| #define FX_MODE_PSHOURGLASS            207
 | |
| #define FX_MODE_PS1DSPRAY              208
 | |
| #define FX_MODE_PSBALANCE              209
 | |
| #define FX_MODE_PSCHASE                210
 | |
| #define FX_MODE_PSSTARBURST            211
 | |
| #define FX_MODE_PS1DGEQ                212
 | |
| #define FX_MODE_PSFIRE1D               213
 | |
| #define FX_MODE_PS1DSONICSTREAM        214
 | |
| #define MODE_COUNT                     215
 | |
| 
 | |
| 
 | |
| #define BLEND_STYLE_FADE            0x00  // universal
 | |
| #define BLEND_STYLE_FAIRY_DUST      0x01  // universal
 | |
| #define BLEND_STYLE_SWIPE_RIGHT     0x02  // 1D or 2D
 | |
| #define BLEND_STYLE_SWIPE_LEFT      0x03  // 1D or 2D
 | |
| #define BLEND_STYLE_PINCH_OUT       0x04  // 1D or 2D
 | |
| #define BLEND_STYLE_INSIDE_OUT      0x05  // 1D or 2D
 | |
| #define BLEND_STYLE_SWIPE_UP        0x06  // 2D
 | |
| #define BLEND_STYLE_SWIPE_DOWN      0x07  // 2D
 | |
| #define BLEND_STYLE_OPEN_H          0x08  // 2D
 | |
| #define BLEND_STYLE_OPEN_V          0x09  // 2D
 | |
| // as there are many push variants to optimise if statements they are groupped together
 | |
| #define BLEND_STYLE_PUSH_RIGHT      0x10  // 1D or 2D (& 0b00010000)
 | |
| #define BLEND_STYLE_PUSH_LEFT       0x11  // 1D or 2D (& 0b00010000)
 | |
| #define BLEND_STYLE_PUSH_UP         0x12  // 2D (& 0b00010000)
 | |
| #define BLEND_STYLE_PUSH_DOWN       0x13  // 2D (& 0b00010000)
 | |
| #define BLEND_STYLE_PUSH_TL         0x14  // 2D (& 0b00010000)
 | |
| #define BLEND_STYLE_PUSH_TR         0x15  // 2D (& 0b00010000)
 | |
| #define BLEND_STYLE_PUSH_BR         0x16  // 2D (& 0b00010000)
 | |
| #define BLEND_STYLE_PUSH_BL         0x17  // 2D (& 0b00010000)
 | |
| #define BLEND_STYLE_PUSH_MASK       0x10
 | |
| #define BLEND_STYLE_COUNT           18
 | |
| 
 | |
| 
 | |
| typedef enum mapping1D2D {
 | |
|   M12_Pixels = 0,
 | |
|   M12_pBar = 1,
 | |
|   M12_pArc = 2,
 | |
|   M12_pCorner = 3,
 | |
|   M12_sPinwheel = 4
 | |
| } mapping1D2D_t;
 | |
| 
 | |
| // segment, 68 bytes
 | |
| typedef struct Segment {
 | |
|   public:
 | |
|     uint16_t start; // start index / start X coordinate 2D (left)
 | |
|     uint16_t stop;  // stop index / stop X coordinate 2D (right); segment is invalid if stop == 0
 | |
|     uint16_t offset;
 | |
|     uint8_t  speed;
 | |
|     uint8_t  intensity;
 | |
|     uint8_t  palette;
 | |
|     uint8_t  mode;
 | |
|     union {
 | |
|       uint16_t options; //bit pattern: msb first: [transposed mirrorY reverseY] transitional (tbd) paused needspixelstate mirrored on reverse selected
 | |
|       struct {
 | |
|         bool    selected    : 1;  //     0 : selected
 | |
|         bool    reverse     : 1;  //     1 : reversed
 | |
|         bool    on          : 1;  //     2 : is On
 | |
|         bool    mirror      : 1;  //     3 : mirrored
 | |
|         bool    freeze      : 1;  //     4 : paused/frozen
 | |
|         bool    reset       : 1;  //     5 : indicates that Segment runtime requires reset
 | |
|         bool    reverse_y   : 1;  //     6 : reversed Y (2D)
 | |
|         bool    mirror_y    : 1;  //     7 : mirrored Y (2D)
 | |
|         bool    transpose   : 1;  //     8 : transposed (2D, swapped X & Y)
 | |
|         uint8_t map1D2D     : 3;  //  9-11 : mapping for 1D effect on 2D (0-use as strip, 1-expand vertically, 2-circular/arc, 3-rectangular/corner, ...)
 | |
|         uint8_t soundSim    : 2;  // 12-13 : 0-3 sound simulation types ("soft" & "hard" or "on"/"off")
 | |
|         uint8_t set         : 2;  // 14-15 : 0-3 UI segment sets/groups
 | |
|       };
 | |
|     };
 | |
|     uint8_t  grouping, spacing;
 | |
|     uint8_t  opacity;
 | |
|     uint32_t colors[NUM_COLORS];
 | |
|     uint8_t  cct;                 //0==1900K, 255==10091K
 | |
|     uint8_t  custom1, custom2;    // custom FX parameters/sliders
 | |
|     struct {
 | |
|       uint8_t custom3 : 5;        // reduced range slider (0-31)
 | |
|       bool    check1  : 1;        // checkmark 1
 | |
|       bool    check2  : 1;        // checkmark 2
 | |
|       bool    check3  : 1;        // checkmark 3
 | |
|     };
 | |
|     uint8_t startY;  // start Y coodrinate 2D (top); there should be no more than 255 rows
 | |
|     uint8_t stopY;   // stop Y coordinate 2D (bottom); there should be no more than 255 rows
 | |
|     // note: two bytes of padding are added here
 | |
|     char    *name;
 | |
| 
 | |
|     // runtime data
 | |
|     unsigned long next_time;  // millis() of next update
 | |
|     uint32_t step;  // custom "step" var
 | |
|     uint32_t call;  // call counter
 | |
|     uint16_t aux0;  // custom var
 | |
|     uint16_t aux1;  // custom var
 | |
|     byte     *data; // effect data pointer
 | |
|     static uint16_t maxWidth, maxHeight;  // these define matrix width & height (max. segment dimensions)
 | |
| 
 | |
|     typedef struct TemporarySegmentData {
 | |
|       uint16_t _optionsT;
 | |
|       uint32_t _colorT[NUM_COLORS];
 | |
|       uint8_t  _speedT;
 | |
|       uint8_t  _intensityT;
 | |
|       uint8_t  _custom1T, _custom2T;   // custom FX parameters/sliders
 | |
|       struct {
 | |
|         uint8_t _custom3T : 5;        // reduced range slider (0-31)
 | |
|         bool    _check1T  : 1;        // checkmark 1
 | |
|         bool    _check2T  : 1;        // checkmark 2
 | |
|         bool    _check3T  : 1;        // checkmark 3
 | |
|       };
 | |
|       uint16_t _aux0T;
 | |
|       uint16_t _aux1T;
 | |
|       uint32_t _stepT;
 | |
|       uint32_t _callT;
 | |
|       uint8_t *_dataT;
 | |
|       unsigned _dataLenT;
 | |
|       TemporarySegmentData()
 | |
|         : _dataT(nullptr) // just in case...
 | |
|         , _dataLenT(0)
 | |
|       {}
 | |
|     } tmpsegd_t;
 | |
| 
 | |
|   private:
 | |
|     union {
 | |
|       uint8_t  _capabilities;
 | |
|       struct {
 | |
|         bool    _isRGB    : 1;
 | |
|         bool    _hasW     : 1;
 | |
|         bool    _isCCT    : 1;
 | |
|         bool    _manualW  : 1;
 | |
|         uint8_t _reserved : 4;
 | |
|       };
 | |
|     };
 | |
|     uint8_t         _default_palette;  // palette number that gets assigned to pal0
 | |
|     unsigned        _dataLen;
 | |
|     static unsigned _usedSegmentData;
 | |
|     static uint8_t  _segBri;                  // brightness of segment for current effect
 | |
|     static unsigned _vLength;                 // 1D dimension used for current effect
 | |
|     static unsigned _vWidth, _vHeight;        // 2D dimensions used for current effect
 | |
|     static uint32_t _currentColors[NUM_COLORS]; // colors used for current effect
 | |
|     static bool     _colorScaled;             // color has been scaled prior to setPixelColor() call
 | |
|     static CRGBPalette16 _currentPalette;     // palette used for current effect (includes transition, used in color_from_palette())
 | |
|     static CRGBPalette16 _randomPalette;      // actual random palette
 | |
|     static CRGBPalette16 _newRandomPalette;   // target random palette
 | |
|     static uint16_t _lastPaletteChange;       // last random palette change time in millis()/1000
 | |
|     static uint16_t _lastPaletteBlend;        // blend palette according to set Transition Delay in millis()%0xFFFF
 | |
|     static uint16_t _transitionprogress;      // current transition progress 0 - 0xFFFF
 | |
|     #ifndef WLED_DISABLE_MODE_BLEND
 | |
|     static bool          _modeBlend;          // mode/effect blending semaphore
 | |
|     // clipping
 | |
|     static uint16_t _clipStart, _clipStop;
 | |
|     static uint8_t  _clipStartY, _clipStopY;
 | |
|     #endif
 | |
| 
 | |
|     // transition data, valid only if transitional==true, holds values during transition (72 bytes)
 | |
|     struct Transition {
 | |
|       #ifndef WLED_DISABLE_MODE_BLEND
 | |
|       tmpsegd_t     _segT;        // previous segment environment
 | |
|       uint8_t       _modeT;       // previous mode/effect
 | |
|       #else
 | |
|       uint32_t      _colorT[NUM_COLORS];
 | |
|       #endif
 | |
|       uint8_t       _palTid;      // previous palette
 | |
|       uint8_t       _briT;        // temporary brightness
 | |
|       uint8_t       _cctT;        // temporary CCT
 | |
|       CRGBPalette16 _palT;        // temporary palette
 | |
|       uint8_t       _prevPaletteBlends; // number of previous palette blends (there are max 255 blends possible)
 | |
|       unsigned long _start;       // must accommodate millis()
 | |
|       uint16_t      _dur;
 | |
|       // -> here is one byte of padding
 | |
|       Transition(uint16_t dur=750)
 | |
|         : _palT(CRGBPalette16(CRGB::Black))
 | |
|         , _prevPaletteBlends(0)
 | |
|         , _start(millis())
 | |
|         , _dur(dur)
 | |
|       {}
 | |
|     } *_t;
 | |
| 
 | |
|     [[gnu::hot]] void _setPixelColorXY_raw(const int& x, const int& y, uint32_t& col) const; // set pixel without mapping (internal use only)
 | |
| 
 | |
|   public:
 | |
| 
 | |
|     Segment(uint16_t sStart=0, uint16_t sStop=30) :
 | |
|       start(sStart),
 | |
|       stop(sStop),
 | |
|       offset(0),
 | |
|       speed(DEFAULT_SPEED),
 | |
|       intensity(DEFAULT_INTENSITY),
 | |
|       palette(0),
 | |
|       mode(DEFAULT_MODE),
 | |
|       options(SELECTED | SEGMENT_ON),
 | |
|       grouping(1),
 | |
|       spacing(0),
 | |
|       opacity(255),
 | |
|       colors{DEFAULT_COLOR,BLACK,BLACK},
 | |
|       cct(127),
 | |
|       custom1(DEFAULT_C1),
 | |
|       custom2(DEFAULT_C2),
 | |
|       custom3(DEFAULT_C3),
 | |
|       check1(false),
 | |
|       check2(false),
 | |
|       check3(false),
 | |
|       startY(0),
 | |
|       stopY(1),
 | |
|       name(nullptr),
 | |
|       next_time(0),
 | |
|       step(0),
 | |
|       call(0),
 | |
|       aux0(0),
 | |
|       aux1(0),
 | |
|       data(nullptr),
 | |
|       _capabilities(0),
 | |
|       _default_palette(0),
 | |
|       _dataLen(0),
 | |
|       _t(nullptr)
 | |
|     {
 | |
|       #ifdef WLED_DEBUG
 | |
|       //Serial.printf("-- Creating segment: %p\n", this);
 | |
|       #endif
 | |
|     }
 | |
| 
 | |
|     Segment(uint16_t sStartX, uint16_t sStopX, uint16_t sStartY, uint16_t sStopY) : Segment(sStartX, sStopX) {
 | |
|       startY = sStartY;
 | |
|       stopY  = sStopY;
 | |
|     }
 | |
| 
 | |
|     Segment(const Segment &orig); // copy constructor
 | |
|     Segment(Segment &&orig) noexcept; // move constructor
 | |
| 
 | |
|     ~Segment() {
 | |
|       #ifdef WLED_DEBUG
 | |
|       //Serial.printf("-- Destroying segment: %p", this);
 | |
|       //if (name) Serial.printf(" %s (%p)", name, name);
 | |
|       //if (data) Serial.printf(" %d->(%p)", (int)_dataLen, data);
 | |
|       //Serial.println();
 | |
|       #endif
 | |
|       if (name) { free(name); name = nullptr; }
 | |
|       stopTransition();
 | |
|       deallocateData();
 | |
|     }
 | |
| 
 | |
|     Segment& operator= (const Segment &orig); // copy assignment
 | |
|     Segment& operator= (Segment &&orig) noexcept; // move assignment
 | |
| 
 | |
| #ifdef WLED_DEBUG
 | |
|     size_t getSize() const { return sizeof(Segment) + (data?_dataLen:0) + (name?strlen(name):0) + (_t?sizeof(Transition):0); }
 | |
| #endif
 | |
| 
 | |
|     inline bool     getOption(uint8_t n) const { return ((options >> n) & 0x01); }
 | |
|     inline bool     isSelected()         const { return selected; }
 | |
|     inline bool     isInTransition()     const { return _t != nullptr; }
 | |
|     inline bool     isActive()           const { return stop > start; }
 | |
|     inline bool     hasRGB()             const { return _isRGB; }
 | |
|     inline bool     hasWhite()           const { return _hasW; }
 | |
|     inline bool     isCCT()              const { return _isCCT; }
 | |
|     inline uint16_t width()              const { return isActive() ? (stop - start) : 0; }  // segment width in physical pixels (length if 1D)
 | |
|     inline uint16_t height()             const { return stopY - startY; }                   // segment height (if 2D) in physical pixels (it *is* always >=1)
 | |
|     inline uint16_t length()             const { return width() * height(); }               // segment length (count) in physical pixels
 | |
|     inline uint16_t groupLength()        const { return grouping + spacing; }
 | |
|     inline uint8_t  getLightCapabilities() const { return _capabilities; }
 | |
|     inline void     deactivate()               { setGeometry(0,0); }
 | |
|     inline Segment &clearName()                { if (name) free(name); name = nullptr; return *this; }
 | |
|     inline Segment &setName(const String &name) { return setName(name.c_str()); }
 | |
| 
 | |
|     inline static unsigned getUsedSegmentData()            { return Segment::_usedSegmentData; }
 | |
|     inline static void     addUsedSegmentData(int len)     { Segment::_usedSegmentData += len; }
 | |
|     #ifndef WLED_DISABLE_MODE_BLEND
 | |
|     inline static void     modeBlend(bool blend)           { _modeBlend = blend; }
 | |
|     inline static bool     getmodeBlend(void)              { return _modeBlend; }
 | |
|     #endif
 | |
|     inline static unsigned vLength()                       { return Segment::_vLength; }
 | |
|     inline static unsigned vWidth()                        { return Segment::_vWidth; }
 | |
|     inline static unsigned vHeight()                       { return Segment::_vHeight; }
 | |
|     inline static uint32_t getCurrentColor(unsigned i)     { return Segment::_currentColors[i]; } // { return i < 3 ? Segment::_currentColors[i] : 0; }
 | |
|     inline static const CRGBPalette16 &getCurrentPalette() { return Segment::_currentPalette; }
 | |
|     inline static uint8_t getCurrentBrightness()           { return Segment::_segBri; }
 | |
|     static void handleRandomPalette();
 | |
| 
 | |
|     void    beginDraw();            // set up parameters for current effect
 | |
|     void    setGeometry(uint16_t i1, uint16_t i2, uint8_t grp=1, uint8_t spc=0, uint16_t ofs=UINT16_MAX, uint16_t i1Y=0, uint16_t i2Y=1, uint8_t m12=0);
 | |
|     Segment &setColor(uint8_t slot, uint32_t c);
 | |
|     Segment &setCCT(uint16_t k);
 | |
|     Segment &setOpacity(uint8_t o);
 | |
|     Segment &setOption(uint8_t n, bool val);
 | |
|     Segment &setMode(uint8_t fx, bool loadDefaults = false);
 | |
|     Segment &setPalette(uint8_t pal);
 | |
|     Segment &setName(const char* name);
 | |
|     uint8_t differs(const Segment& b) const;
 | |
|     void    refreshLightCapabilities();
 | |
| 
 | |
|     // runtime data functions
 | |
|     inline uint16_t dataSize() const { return _dataLen; }
 | |
|     bool allocateData(size_t len);  // allocates effect data buffer in heap and clears it
 | |
|     void deallocateData();          // deallocates (frees) effect data buffer from heap
 | |
|     void resetIfRequired();         // sets all SEGENV variables to 0 and clears data buffer
 | |
|     /**
 | |
|       * Flags that before the next effect is calculated,
 | |
|       * the internal segment state should be reset.
 | |
|       * Call resetIfRequired before calling the next effect function.
 | |
|       * Safe to call from interrupts and network requests.
 | |
|       */
 | |
|     inline Segment &markForReset() { reset = true; return *this; }  // setOption(SEG_OPTION_RESET, true)
 | |
| 
 | |
|     // transition functions
 | |
|     void     startTransition(uint16_t dur);     // transition has to start before actual segment values change
 | |
|     void     stopTransition();                  // ends transition mode by destroying transition structure (does nothing if not in transition)
 | |
|     inline void handleTransition() { updateTransitionProgress(); if (progress() == 0xFFFFU) stopTransition(); }
 | |
|     #ifndef WLED_DISABLE_MODE_BLEND
 | |
|     void     swapSegenv(tmpsegd_t &tmpSegD);    // copies segment data into specifed buffer, if buffer is not a transition buffer, segment data is overwritten from transition buffer
 | |
|     void     restoreSegenv(const tmpsegd_t &tmpSegD); // restores segment data from buffer, if buffer is not transition buffer, changed values are copied to transition buffer
 | |
|     #endif
 | |
|     [[gnu::hot]] void updateTransitionProgress();            // set current progression of transition
 | |
|     inline uint16_t progress() const { return Segment::_transitionprogress; }  // transition progression between 0-65535
 | |
|     [[gnu::hot]] uint8_t  currentBri(bool useCct = false) const; // current segment brightness/CCT (blended while in transition)
 | |
|     uint8_t  currentMode() const;                            // currently active effect/mode (while in transition)
 | |
|     [[gnu::hot]] uint32_t currentColor(uint8_t slot) const;  // currently active segment color (blended while in transition)
 | |
|     CRGBPalette16 &loadPalette(CRGBPalette16 &tgt, uint8_t pal);
 | |
|     void     loadOldPalette(); // loads old FX palette into _currentPalette
 | |
| 
 | |
|     // 1D strip
 | |
|     [[gnu::hot]] uint16_t virtualLength() const;
 | |
|     [[gnu::hot]] void setPixelColor(int i, uint32_t c) const; // set relative pixel within segment with color
 | |
|     inline void setPixelColor(unsigned n, uint32_t c) const                    { setPixelColor(int(n), c); }
 | |
|     inline void setPixelColor(int n, byte r, byte g, byte b, byte w = 0) const { setPixelColor(n, RGBW32(r,g,b,w)); }
 | |
|     inline void setPixelColor(int n, CRGB c) const                             { setPixelColor(n, RGBW32(c.r,c.g,c.b,0)); }
 | |
|     #ifdef WLED_USE_AA_PIXELS
 | |
|     void setPixelColor(float i, uint32_t c, bool aa = true) const;
 | |
|     inline void setPixelColor(float i, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0, bool aa = true) const { setPixelColor(i, RGBW32(r,g,b,w), aa); }
 | |
|     inline void setPixelColor(float i, CRGB c, bool aa = true) const                                         { setPixelColor(i, RGBW32(c.r,c.g,c.b,0), aa); }
 | |
|     #endif
 | |
|     #ifndef WLED_DISABLE_MODE_BLEND
 | |
|     static inline void setClippingRect(int startX, int stopX, int startY = 0, int stopY = 1) { _clipStart = startX; _clipStop = stopX; _clipStartY = startY; _clipStopY = stopY; };
 | |
|     #endif
 | |
|     bool isPixelClipped(int i) const;
 | |
|     [[gnu::hot]] uint32_t getPixelColor(int i) const;
 | |
|     // 1D support functions (some implement 2D as well)
 | |
|     void blur(uint8_t, bool smear = false);
 | |
|     void fill(uint32_t c);
 | |
|     void fade_out(uint8_t r);
 | |
|     void fadeToBlackBy(uint8_t fadeBy);
 | |
|     inline void blendPixelColor(int n, uint32_t color, uint8_t blend)    { setPixelColor(n, color_blend(getPixelColor(n), color, blend)); }
 | |
|     inline void blendPixelColor(int n, CRGB c, uint8_t blend)            { blendPixelColor(n, RGBW32(c.r,c.g,c.b,0), blend); }
 | |
|     inline void addPixelColor(int n, uint32_t color, bool preserveCR = true)                     { setPixelColor(n, color_add(getPixelColor(n), color, preserveCR)); }
 | |
|     inline void addPixelColor(int n, byte r, byte g, byte b, byte w = 0, bool preserveCR = true) { addPixelColor(n, RGBW32(r,g,b,w), preserveCR); }
 | |
|     inline void addPixelColor(int n, CRGB c, bool preserveCR = true)                             { addPixelColor(n, RGBW32(c.r,c.g,c.b,0), preserveCR); }
 | |
|     inline void fadePixelColor(uint16_t n, uint8_t fade)                 { setPixelColor(n, color_fade(getPixelColor(n), fade, true)); }
 | |
|     [[gnu::hot]] uint32_t color_from_palette(uint16_t, bool mapping, bool wrap, uint8_t mcol, uint8_t pbri = 255) const;
 | |
|     [[gnu::hot]] uint32_t color_wheel(uint8_t pos) const;
 | |
| 
 | |
|     // 2D Blur: shortcuts for bluring columns or rows only (50% faster than full 2D blur)
 | |
|     inline void blurCols(fract8 blur_amount, bool smear = false) { // blur all columns
 | |
|       blur2D(0, blur_amount, smear);
 | |
|     }
 | |
|     inline void blurRows(fract8 blur_amount, bool smear = false) { // blur all rows
 | |
|       blur2D(blur_amount, 0, smear);
 | |
|     }
 | |
| 
 | |
|     // 2D matrix
 | |
|     [[gnu::hot]] unsigned virtualWidth()  const; // segment width in virtual pixels (accounts for groupping and spacing)
 | |
|     [[gnu::hot]] unsigned virtualHeight() const; // segment height in virtual pixels (accounts for groupping and spacing)
 | |
|     inline unsigned nrOfVStrips() const {        // returns number of virtual vertical strips in 2D matrix (used to expand 1D effects into 2D)
 | |
|     #ifndef WLED_DISABLE_2D
 | |
|       return (is2D() &&  map1D2D == M12_pBar) ? virtualWidth() : 1;
 | |
|     #else
 | |
|       return 1;
 | |
|     #endif
 | |
|     }
 | |
|   #ifndef WLED_DISABLE_2D
 | |
|     inline bool is2D() const                                                            { return (width()>1 && height()>1); }
 | |
|     [[gnu::hot]] void setPixelColorXY(int x, int y, uint32_t c) const; // set relative pixel within segment with color
 | |
|     inline void setPixelColorXY(unsigned x, unsigned y, uint32_t c) const               { setPixelColorXY(int(x), int(y), c); }
 | |
|     inline void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) const { setPixelColorXY(x, y, RGBW32(r,g,b,w)); }
 | |
|     inline void setPixelColorXY(int x, int y, CRGB c) const                             { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0)); }
 | |
|     inline void setPixelColorXY(unsigned x, unsigned y, CRGB c) const                   { setPixelColorXY(int(x), int(y), RGBW32(c.r,c.g,c.b,0)); }
 | |
|     #ifdef WLED_USE_AA_PIXELS
 | |
|     void setPixelColorXY(float x, float y, uint32_t c, bool aa = true) const;
 | |
|     inline void setPixelColorXY(float x, float y, byte r, byte g, byte b, byte w = 0, bool aa = true) const { setPixelColorXY(x, y, RGBW32(r,g,b,w), aa); }
 | |
|     inline void setPixelColorXY(float x, float y, CRGB c, bool aa = true) const                             { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), aa); }
 | |
|     #endif
 | |
|     [[gnu::hot]] bool isPixelXYClipped(int x, int y) const;
 | |
|     [[gnu::hot]] uint32_t getPixelColorXY(int x, int y) const;
 | |
|     // 2D support functions
 | |
|     inline void blendPixelColorXY(uint16_t x, uint16_t y, uint32_t color, uint8_t blend) { setPixelColorXY(x, y, color_blend(getPixelColorXY(x,y), color, blend)); }
 | |
|     inline void blendPixelColorXY(uint16_t x, uint16_t y, CRGB c, uint8_t blend)         { blendPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), blend); }
 | |
|     inline void addPixelColorXY(int x, int y, uint32_t color, bool preserveCR = true)                     { setPixelColorXY(x, y, color_add(getPixelColorXY(x,y), color, preserveCR)); }
 | |
|     inline void addPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0, bool preserveCR = true) { addPixelColorXY(x, y, RGBW32(r,g,b,w), preserveCR); }
 | |
|     inline void addPixelColorXY(int x, int y, CRGB c, bool preserveCR = true)                             { addPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), preserveCR); }
 | |
|     inline void fadePixelColorXY(uint16_t x, uint16_t y, uint8_t fade)                   { setPixelColorXY(x, y, color_fade(getPixelColorXY(x,y), fade, true)); }
 | |
|     //void box_blur(unsigned r = 1U, bool smear = false); // 2D box blur
 | |
|     void blur2D(uint8_t blur_x, uint8_t blur_y, bool smear = false);
 | |
|     void moveX(int delta, bool wrap = false);
 | |
|     void moveY(int delta, bool wrap = false);
 | |
|     void move(unsigned dir, unsigned delta, bool wrap = false);
 | |
|     void drawCircle(uint16_t cx, uint16_t cy, uint8_t radius, uint32_t c, bool soft = false);
 | |
|     inline void drawCircle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c, bool soft = false) { drawCircle(cx, cy, radius, RGBW32(c.r,c.g,c.b,0), soft); }
 | |
|     void fillCircle(uint16_t cx, uint16_t cy, uint8_t radius, uint32_t c, bool soft = false);
 | |
|     inline void fillCircle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c, bool soft = false) { fillCircle(cx, cy, radius, RGBW32(c.r,c.g,c.b,0), soft); }
 | |
|     void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, uint32_t c, bool soft = false);
 | |
|     inline void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, CRGB c, bool soft = false) { drawLine(x0, y0, x1, y1, RGBW32(c.r,c.g,c.b,0), soft); } // automatic inline
 | |
|     void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, uint32_t color, uint32_t col2 = 0, int8_t rotate = 0, bool usePalGrad = false);
 | |
|     inline void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB c) { drawCharacter(chr, x, y, w, h, RGBW32(c.r,c.g,c.b,0)); } // automatic inline
 | |
|     inline void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB c, CRGB c2, int8_t rotate = 0, bool usePalGrad = false) { drawCharacter(chr, x, y, w, h, RGBW32(c.r,c.g,c.b,0), RGBW32(c2.r,c2.g,c2.b,0), rotate, usePalGrad); } // automatic inline
 | |
|     void wu_pixel(uint32_t x, uint32_t y, CRGB c);
 | |
|     inline void fill_solid(CRGB c) { fill(RGBW32(c.r,c.g,c.b,0)); }
 | |
|   #else
 | |
|     inline bool is2D() const                                                      { return false; }
 | |
|     inline void setPixelColorXY(int x, int y, uint32_t c)                         { setPixelColor(x, c); }
 | |
|     inline void setPixelColorXY(unsigned x, unsigned y, uint32_t c)               { setPixelColor(int(x), c); }
 | |
|     inline void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) { setPixelColor(x, RGBW32(r,g,b,w)); }
 | |
|     inline void setPixelColorXY(int x, int y, CRGB c)                             { setPixelColor(x, RGBW32(c.r,c.g,c.b,0)); }
 | |
|     inline void setPixelColorXY(unsigned x, unsigned y, CRGB c)                   { setPixelColor(int(x), RGBW32(c.r,c.g,c.b,0)); }
 | |
|     #ifdef WLED_USE_AA_PIXELS
 | |
|     inline void setPixelColorXY(float x, float y, uint32_t c, bool aa = true)     { setPixelColor(x, c, aa); }
 | |
|     inline void setPixelColorXY(float x, float y, byte r, byte g, byte b, byte w = 0, bool aa = true) { setPixelColor(x, RGBW32(r,g,b,w), aa); }
 | |
|     inline void setPixelColorXY(float x, float y, CRGB c, bool aa = true)         { setPixelColor(x, RGBW32(c.r,c.g,c.b,0), aa); }
 | |
|     #endif
 | |
|     inline bool isPixelXYClipped(int x, int y)                                    { return isPixelClipped(x); }
 | |
|     inline uint32_t getPixelColorXY(int x, int y)                                 { return getPixelColor(x); }
 | |
|     inline void blendPixelColorXY(uint16_t x, uint16_t y, uint32_t c, uint8_t blend) { blendPixelColor(x, c, blend); }
 | |
|     inline void blendPixelColorXY(uint16_t x, uint16_t y, CRGB c, uint8_t blend)  { blendPixelColor(x, RGBW32(c.r,c.g,c.b,0), blend); }
 | |
|     inline void addPixelColorXY(int x, int y, uint32_t color, bool saturate = false) { addPixelColor(x, color, saturate); }
 | |
|     inline void addPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0, bool saturate = false) { addPixelColor(x, RGBW32(r,g,b,w), saturate); }
 | |
|     inline void addPixelColorXY(int x, int y, CRGB c, bool saturate = false)         { addPixelColor(x, RGBW32(c.r,c.g,c.b,0), saturate); }
 | |
|     inline void fadePixelColorXY(uint16_t x, uint16_t y, uint8_t fade)            { fadePixelColor(x, fade); }
 | |
|     //inline void box_blur(unsigned i, bool vertical, fract8 blur_amount) {}
 | |
|     inline void blur2D(uint8_t blur_x, uint8_t blur_y, bool smear = false) {}
 | |
|     inline void blurRow(int row, fract8 blur_amount, bool smear = false) {}
 | |
|     inline void blurCol(int col, fract8 blur_amount, bool smear = false) {}
 | |
|     inline void moveX(int delta, bool wrap = false) {}
 | |
|     inline void moveY(int delta, bool wrap = false) {}
 | |
|     inline void move(uint8_t dir, uint8_t delta, bool wrap = false) {}
 | |
|     inline void drawCircle(uint16_t cx, uint16_t cy, uint8_t radius, uint32_t c, bool soft = false) {}
 | |
|     inline void drawCircle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c, bool soft = false) {}
 | |
|     inline void fillCircle(uint16_t cx, uint16_t cy, uint8_t radius, uint32_t c, bool soft = false) {}
 | |
|     inline void fillCircle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c, bool soft = false) {}
 | |
|     inline void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, uint32_t c, bool soft = false) {}
 | |
|     inline void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, CRGB c, bool soft = false) {}
 | |
|     inline void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, uint32_t color, uint32_t = 0, int8_t = 0, bool = false) {}
 | |
|     inline void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB color) {}
 | |
|     inline void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB c, CRGB c2, int8_t rotate = 0, bool usePalGrad = false) {}
 | |
|     inline void wu_pixel(uint32_t x, uint32_t y, CRGB c) {}
 | |
|   #endif
 | |
| } segment;
 | |
| //static int segSize = sizeof(Segment);
 | |
| 
 | |
| // main "strip" class
 | |
| class WS2812FX {  // 96 bytes
 | |
|   typedef uint16_t (*mode_ptr)(); // pointer to mode function
 | |
|   typedef void (*show_callback)(); // pre show callback
 | |
|   typedef struct ModeData {
 | |
|     uint8_t     _id;   // mode (effect) id
 | |
|     mode_ptr    _fcn;  // mode (effect) function
 | |
|     const char *_data; // mode (effect) name and its UI control data
 | |
|     ModeData(uint8_t id, uint16_t (*fcn)(void), const char *data) : _id(id), _fcn(fcn), _data(data) {}
 | |
|   } mode_data_t;
 | |
| 
 | |
|   static WS2812FX* instance;
 | |
| 
 | |
|   public:
 | |
| 
 | |
|     WS2812FX() :
 | |
|       paletteBlend(0),
 | |
|       now(millis()),
 | |
|       timebase(0),
 | |
|       isMatrix(false),
 | |
| #ifndef WLED_DISABLE_2D
 | |
|       panels(1),
 | |
| #endif
 | |
| #ifdef WLED_AUTOSEGMENTS
 | |
|       autoSegments(true),
 | |
| #else
 | |
|       autoSegments(false),
 | |
| #endif
 | |
|       correctWB(false),
 | |
|       cctFromRgb(false),
 | |
|       // true private variables
 | |
|       _suspend(false),
 | |
|       _length(DEFAULT_LED_COUNT),
 | |
|       _brightness(DEFAULT_BRIGHTNESS),
 | |
|       _transitionDur(750),
 | |
|       _targetFps(WLED_FPS),
 | |
|       _frametime(FRAMETIME_FIXED),
 | |
|       _cumulativeFps(50 << FPS_CALC_SHIFT),
 | |
|       _isServicing(false),
 | |
|       _isOffRefreshRequired(false),
 | |
|       _hasWhiteChannel(false),
 | |
|       _triggered(false),
 | |
|       _modeCount(MODE_COUNT),
 | |
|       _callback(nullptr),
 | |
|       customMappingTable(nullptr),
 | |
|       customMappingSize(0),
 | |
|       _lastShow(0),
 | |
|       _lastServiceShow(0),
 | |
|       _segment_index(0),
 | |
|       _mainSegment(0)
 | |
|     {
 | |
|       WS2812FX::instance = this;
 | |
|       _mode.reserve(_modeCount);     // allocate memory to prevent initial fragmentation (does not increase size())
 | |
|       _modeData.reserve(_modeCount); // allocate memory to prevent initial fragmentation (does not increase size())
 | |
|       if (_mode.capacity() <= 1 || _modeData.capacity() <= 1) _modeCount = 1; // memory allocation failed only show Solid
 | |
|       else setupEffectData();
 | |
|     }
 | |
| 
 | |
|     ~WS2812FX() {
 | |
|       if (customMappingTable) free(customMappingTable);
 | |
|       _mode.clear();
 | |
|       _modeData.clear();
 | |
|       _segments.clear();
 | |
| #ifndef WLED_DISABLE_2D
 | |
|       panel.clear();
 | |
| #endif
 | |
|       customPalettes.clear();
 | |
|     }
 | |
| 
 | |
|     static WS2812FX* getInstance() { return instance; }
 | |
| 
 | |
|     void
 | |
| #ifdef WLED_DEBUG
 | |
|       printSize(),                                // prints memory usage for strip components
 | |
| #endif
 | |
|       finalizeInit(),                             // initialises strip components
 | |
|       service(),                                  // executes effect functions when due and calls strip.show()
 | |
|       setCCT(uint16_t k),                         // sets global CCT (either in relative 0-255 value or in K)
 | |
|       setBrightness(uint8_t b, bool direct = false),    // sets strip brightness
 | |
|       setRange(uint16_t i, uint16_t i2, uint32_t col),  // used for clock overlay
 | |
|       purgeSegments(),                            // removes inactive segments from RAM (may incure penalty and memory fragmentation but reduces vector footprint)
 | |
|       setMainSegmentId(unsigned n = 0),
 | |
|       resetSegments(),                            // marks all segments for reset
 | |
|       makeAutoSegments(bool forceReset = false),  // will create segments based on configured outputs
 | |
|       fixInvalidSegments(),                       // fixes incorrect segment configuration
 | |
|       setPixelColor(unsigned i, uint32_t c) const,      // paints absolute strip pixel with index n and color c
 | |
|       show(),                                     // initiates LED output
 | |
|       setTargetFps(unsigned fps),
 | |
|       setupEffectData();                          // add default effects to the list; defined in FX.cpp
 | |
| 
 | |
|     inline void resetTimebase()           { timebase = 0UL - millis(); }
 | |
|     inline void restartRuntime()          { for (Segment &seg : _segments) { seg.markForReset().resetIfRequired(); } }
 | |
|     inline void setTransitionMode(bool t) { for (Segment &seg : _segments) seg.startTransition(t ? _transitionDur : 0); }
 | |
|     inline void setPixelColor(unsigned n, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0) const { setPixelColor(n, RGBW32(r,g,b,w)); }
 | |
|     inline void setPixelColor(unsigned n, CRGB c) const                                         { setPixelColor(n, c.red, c.green, c.blue); }
 | |
|     inline void fill(uint32_t c) const    { for (unsigned i = 0; i < getLengthTotal(); i++) setPixelColor(i, c); } // fill whole strip with color (inline)
 | |
|     inline void trigger()                                     { _triggered = true; }  // Forces the next frame to be computed on all active segments.
 | |
|     inline void setShowCallback(show_callback cb)             { _callback = cb; }
 | |
|     inline void setTransition(uint16_t t)                     { _transitionDur = t; } // sets transition time (in ms)
 | |
|     inline void appendSegment(const Segment &seg = Segment()) { if (_segments.size() < getMaxSegments()) _segments.push_back(seg); }
 | |
|     inline void suspend()                                     { _suspend = true; }    // will suspend (and canacel) strip.service() execution
 | |
|     inline void resume()                                      { _suspend = false; }   // will resume strip.service() execution
 | |
| 
 | |
|     bool
 | |
|       checkSegmentAlignment() const,
 | |
|       hasRGBWBus() const,
 | |
|       hasCCTBus() const,
 | |
|       deserializeMap(unsigned n = 0);
 | |
| 
 | |
|     inline bool isUpdating() const           { return !BusManager::canAllShow(); } // return true if the strip is being sent pixel updates
 | |
|     inline bool isServicing() const          { return _isServicing; }           // returns true if strip.service() is executing
 | |
|     inline bool hasWhiteChannel() const      { return _hasWhiteChannel; }       // returns true if strip contains separate white chanel
 | |
|     inline bool isOffRefreshRequired() const { return _isOffRefreshRequired; }  // returns true if strip requires regular updates (i.e. TM1814 chipset)
 | |
|     inline bool isSuspended() const          { return _suspend; }               // returns true if strip.service() execution is suspended
 | |
|     inline bool needsUpdate() const          { return _triggered; }             // returns true if strip received a trigger() request
 | |
| 
 | |
|     uint8_t
 | |
|       paletteBlend,
 | |
|       getActiveSegmentsNum() const,
 | |
|       getFirstSelectedSegId() const,
 | |
|       getLastActiveSegmentId() const,
 | |
|       getActiveSegsLightCapabilities(bool selectedOnly = false) const,
 | |
|       addEffect(uint8_t id, mode_ptr mode_fn, const char *mode_name);         // add effect to the list; defined in FX.cpp;
 | |
| 
 | |
|     inline uint8_t getBrightness() const    { return _brightness; }       // returns current strip brightness
 | |
|     inline static constexpr unsigned getMaxSegments() { return MAX_NUM_SEGMENTS; }  // returns maximum number of supported segments (fixed value)
 | |
|     inline uint8_t getSegmentsNum() const   { return _segments.size(); }  // returns currently present segments
 | |
|     inline uint8_t getCurrSegmentId() const { return _segment_index; }    // returns current segment index (only valid while strip.isServicing())
 | |
|     inline uint8_t getMainSegmentId() const { return _mainSegment; }      // returns main segment index
 | |
|     inline uint8_t getPaletteCount() const  { return 13 + GRADIENT_PALETTE_COUNT + customPalettes.size(); }
 | |
|     inline uint8_t getTargetFps() const     { return _targetFps; }        // returns rough FPS value for las 2s interval
 | |
|     inline uint8_t getModeCount() const     { return _modeCount; }        // returns number of registered modes/effects
 | |
| 
 | |
|     uint16_t
 | |
|       getLengthPhysical() const,
 | |
|       getLengthTotal() const; // will include virtual/nonexistent pixels in matrix
 | |
| 
 | |
|     inline uint16_t getFps() const          { return (millis() - _lastShow > 2000) ? 0 : (FPS_MULTIPLIER * _cumulativeFps) >> FPS_CALC_SHIFT; } // Returns the refresh rate of the LED strip (_cumulativeFps is stored in fixed point)
 | |
|     inline uint16_t getFrameTime() const    { return _frametime; }        // returns amount of time a frame should take (in ms)
 | |
|     inline uint16_t getMinShowDelay() const { return MIN_FRAME_DELAY; }   // returns minimum amount of time strip.service() can be delayed (constant)
 | |
|     inline uint16_t getLength() const       { return _length; }           // returns actual amount of LEDs on a strip (2D matrix may have less LEDs than W*H)
 | |
|     inline uint16_t getTransition() const   { return _transitionDur; }    // returns currently set transition time (in ms)
 | |
|     inline uint16_t getMappedPixelIndex(uint16_t index) const {           // convert logical address to physical
 | |
|       if (index < customMappingSize && (realtimeMode == REALTIME_MODE_INACTIVE || realtimeRespectLedMaps)) index = customMappingTable[index];
 | |
|       return index;
 | |
|     };
 | |
| 
 | |
|     unsigned long now, timebase;
 | |
|     uint32_t getPixelColor(unsigned i) const;
 | |
| 
 | |
|     inline uint32_t getLastShow() const   { return _lastShow; }           // returns millis() timestamp of last strip.show() call
 | |
| 
 | |
|     const char *getModeData(unsigned id = 0) const { return (id && id < _modeCount) ? _modeData[id] : PSTR("Solid"); }
 | |
|     inline const char **getModeDataSrc()  { return &(_modeData[0]); } // vectors use arrays for underlying data
 | |
| 
 | |
|     Segment&        getSegment(unsigned id);
 | |
|     inline Segment& getFirstSelectedSeg() { return _segments[getFirstSelectedSegId()]; }  // returns reference to first segment that is "selected"
 | |
|     inline Segment& getMainSegment()      { return _segments[getMainSegmentId()]; }       // returns reference to main segment
 | |
|     inline Segment* getSegments()         { return &(_segments[0]); }                     // returns pointer to segment vector structure (warning: use carefully)
 | |
| 
 | |
|   // 2D support (panels)
 | |
|     bool
 | |
|       isMatrix;
 | |
| 
 | |
| #ifndef WLED_DISABLE_2D
 | |
|     #define WLED_MAX_PANELS 18
 | |
|     uint8_t
 | |
|       panels;
 | |
| 
 | |
|     typedef struct panel_t {
 | |
|       uint16_t xOffset; // x offset relative to the top left of matrix in LEDs
 | |
|       uint16_t yOffset; // y offset relative to the top left of matrix in LEDs
 | |
|       uint8_t  width;   // width of the panel
 | |
|       uint8_t  height;  // height of the panel
 | |
|       union {
 | |
|         uint8_t options;
 | |
|         struct {
 | |
|           bool bottomStart : 1; // starts at bottom?
 | |
|           bool rightStart  : 1; // starts on right?
 | |
|           bool vertical    : 1; // is vertical?
 | |
|           bool serpentine  : 1; // is serpentine?
 | |
|         };
 | |
|       };
 | |
|       panel_t()
 | |
|         : xOffset(0)
 | |
|         , yOffset(0)
 | |
|         , width(8)
 | |
|         , height(8)
 | |
|         , options(0)
 | |
|       {}
 | |
|     } Panel;
 | |
|     std::vector<Panel> panel;
 | |
| #endif
 | |
| 
 | |
|     void setUpMatrix();     // sets up automatic matrix ledmap from panel configuration
 | |
| 
 | |
|     // outsmart the compiler :) by correctly overloading
 | |
|     inline void setPixelColorXY(int x, int y, uint32_t c) const { setPixelColor((unsigned)(y * Segment::maxWidth + x), c); }
 | |
|     inline void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) const { setPixelColorXY(x, y, RGBW32(r,g,b,w)); }
 | |
|     inline void setPixelColorXY(int x, int y, CRGB c) const     { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0)); }
 | |
| 
 | |
|     inline uint32_t getPixelColorXY(int x, int y) const         { return getPixelColor(isMatrix ? y * Segment::maxWidth + x : x); }
 | |
| 
 | |
|   // end 2D support
 | |
| 
 | |
|     void loadCustomPalettes(); // loads custom palettes from JSON
 | |
|     std::vector<CRGBPalette16> customPalettes; // TODO: move custom palettes out of WS2812FX class
 | |
| 
 | |
|     struct {
 | |
|       bool autoSegments : 1;
 | |
|       bool correctWB    : 1;
 | |
|       bool cctFromRgb   : 1;
 | |
|     };
 | |
| 
 | |
|     std::vector<segment> _segments;
 | |
|     friend struct Segment;
 | |
| 
 | |
|   private:
 | |
|     volatile bool _suspend;
 | |
| 
 | |
|     uint16_t _length;
 | |
|     uint8_t  _brightness;
 | |
|     uint16_t _transitionDur;
 | |
| 
 | |
|     uint8_t  _targetFps;
 | |
|     uint16_t _frametime;
 | |
|     uint16_t _cumulativeFps;
 | |
| 
 | |
|     // will require only 1 byte
 | |
|     struct {
 | |
|       bool _isServicing          : 1;
 | |
|       bool _isOffRefreshRequired : 1; //periodic refresh is required for the strip to remain off.
 | |
|       bool _hasWhiteChannel      : 1;
 | |
|       bool _triggered            : 1;
 | |
|     };
 | |
| 
 | |
|     uint8_t                  _modeCount;
 | |
|     std::vector<mode_ptr>    _mode;     // SRAM footprint: 4 bytes per element
 | |
|     std::vector<const char*> _modeData; // mode (effect) name and its slider control data array
 | |
| 
 | |
|     show_callback _callback;
 | |
| 
 | |
|     uint16_t* customMappingTable;
 | |
|     uint16_t  customMappingSize;
 | |
| 
 | |
|     unsigned long _lastShow;
 | |
|     unsigned long _lastServiceShow;
 | |
| 
 | |
|     uint8_t _segment_index;
 | |
|     uint8_t _mainSegment;
 | |
| };
 | |
| 
 | |
| extern const char JSON_mode_names[];
 | |
| extern const char JSON_palette_names[];
 | |
| 
 | |
| #endif |