Compare commits
	
		
			62 Commits
		
	
	
		
			copilot/fi
			...
			v0.15.1.be
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | f593d404cb | ||
|   | b75a2de485 | ||
|   | 642a9e3652 | ||
|   | 85d3f6f11c | ||
|   | f490908278 | ||
|   | 1fc3cc83bd | ||
|   | e96fd8ae58 | ||
|   | c46e328b59 | ||
|   | fa2f831044 | ||
|   | 1bf13ea525 | ||
|   | edc6022441 | ||
|   | 2ac4d03160 | ||
|   | cc4a4c4ae1 | ||
|   | 4e11ecda4b | ||
|   | 473700e4c0 | ||
|   | 0d5a0fb830 | ||
|   | 012143bd7b | ||
|   | 700a7076fd | ||
|   | 5fc2175dd4 | ||
|   | c9b95e22d3 | ||
|   | a265318037 | ||
|   | 866a4c8ab6 | ||
|   | 9dc1022010 | ||
|   | faadb67eb0 | ||
|   | a111a2e7a1 | ||
|   | 32864d8986 | ||
|   | d7bebc2659 | ||
|   | af410ae2d0 | ||
|   | 1891cc816f | ||
|   | 9a4073e606 | ||
|   | b78229d1e2 | ||
|   | 71b242874f | ||
|   | 9328e6faca | ||
|   | bbacc2daae | ||
|   | 3e22f9cabb | ||
|   | 685ad83d4b | ||
|   | 62ddb18a1a | ||
|   | 6a12378475 | ||
|   | d26b3108da | ||
|   | c89e4576b4 | ||
|   | bc79f44a26 | ||
|   | 7ece14ff3f | ||
|   | 0b3643132b | ||
|   | a5693fbf8d | ||
|   | 5c5b70f52b | ||
|   | ae97e388a6 | ||
|   | 37cddcaacc | ||
|   | a1b332fc78 | ||
|   | 86d7c24513 | ||
|   | b28add3b8b | ||
|   | 5fd3a513a4 | ||
|   | b98a8a10b0 | ||
|   | 2bee2793ef | ||
|   | 2f6fa66f4d | ||
|   | 5d38acd787 | ||
|   | e607fcb5c5 | ||
|   | 8a18555ae4 | ||
|   | beb709dc8f | ||
|   | 7a58c69a80 | ||
|   | 1082c85789 | ||
|   | 568d2edd96 | ||
|   | d2d56ebbd2 | 
							
								
								
									
										18
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,5 +1,23 @@ | ||||
| ## WLED changelog | ||||
|  | ||||
| #### Build 2412100 | ||||
| -   WLED 0.15.0 release | ||||
| -   Usermod BME280: Fix "Unit of Measurement" for temperature | ||||
| -   WiFi reconnect bugfix (@blazoncek) | ||||
|  | ||||
| #### Build 2411250 | ||||
| -   WLED 0.15.0-rc1 release | ||||
| -   Add support for esp32S3_wroom2 (#4243 by @softhack007) | ||||
| -   Fix mixed LED SK6812 and ws2812b booloop (#4301 by @willmmiles) | ||||
| -   Improved FPS calculation (by DedeHai) | ||||
| -   Fix crashes when using HTTP API within MQTT (#4269 by @willmmiles) | ||||
| -   Fix array overflow in exploding_fireworks (#4120 by @willmmiles) | ||||
| -   Fix MQTT topic buffer length (#4293 by @WouterGritter) | ||||
| -   Fix SparkFunDMX fix for possible array bounds violation in DMX.write (by @softhack007) | ||||
| -   Allow TV Simulator on single LED segments (by @softhack007) | ||||
| -   Fix WLED_RELEASE_NAME (by @netmindz) | ||||
|  | ||||
|  | ||||
| #### Build 2410270 | ||||
| -   WLED 0.15.0-b7 release | ||||
| -   Re-license the WLED project from MIT to EUPL (#4194 by @Aircoookie) | ||||
|   | ||||
							
								
								
									
										9
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										9
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -1,18 +1,21 @@ | ||||
| { | ||||
|   "name": "wled", | ||||
|   "version": "0.15.0-b7", | ||||
|   "version": "0.15.0", | ||||
|   "lockfileVersion": 3, | ||||
|   "requires": true, | ||||
|   "packages": { | ||||
|     "": { | ||||
|       "name": "wled", | ||||
|       "version": "0.15.0-b7", | ||||
|       "version": "0.15.0", | ||||
|       "license": "ISC", | ||||
|       "dependencies": { | ||||
|         "clean-css": "^5.3.3", | ||||
|         "html-minifier-terser": "^7.2.0", | ||||
|         "inliner": "^1.13.1", | ||||
|         "nodemon": "^3.0.2" | ||||
|         "nodemon": "^3.1.7" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=20.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@jridgewell/gen-mapping": { | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|   "name": "wled", | ||||
|   "version": "0.15.0-b7", | ||||
|   "version": "0.15.1.beta1", | ||||
|   "description": "Tools for WLED project", | ||||
|   "main": "tools/cdata.js", | ||||
|   "directories": { | ||||
|   | ||||
| @@ -1,3 +1,21 @@ | ||||
| Import('env') | ||||
| Import("env") | ||||
| import shutil | ||||
|  | ||||
| env.Execute("npm run build") | ||||
| node_ex = shutil.which("node") | ||||
| # Check if Node.js is installed and present in PATH if it failed, abort the build | ||||
| if node_ex is None: | ||||
|     print('\x1b[0;31;43m' + 'Node.js is not installed or missing from PATH html css js will not be processed check https://kno.wled.ge/advanced/compiling-wled/' + '\x1b[0m') | ||||
|     exitCode = env.Execute("null") | ||||
|     exit(exitCode) | ||||
| else: | ||||
|     # Install the necessary node packages for the pre-build asset bundling script | ||||
|     print('\x1b[6;33;42m' + 'Installing node packages' + '\x1b[0m') | ||||
|     env.Execute("npm install") | ||||
|  | ||||
|     # Call the bundling script | ||||
|     exitCode = env.Execute("npm run build") | ||||
|  | ||||
|     # If it failed, abort the build | ||||
|     if (exitCode): | ||||
|       print('\x1b[0;31;43m' + 'npm run build fails check https://kno.wled.ge/advanced/compiling-wled/' + '\x1b[0m') | ||||
|       exit(exitCode) | ||||
| @@ -176,6 +176,7 @@ lib_deps = | ||||
| extra_scripts = ${scripts_defaults.extra_scripts} | ||||
|  | ||||
| [esp8266] | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = | ||||
|   -DESP8266 | ||||
|   -DFP_IN_IROM | ||||
| @@ -242,6 +243,7 @@ lib_deps_compat = | ||||
| #platform = https://github.com/tasmota/platform-espressif32/releases/download/v2.0.2.3/platform-espressif32-2.0.2.3.zip | ||||
| platform = espressif32@3.5.0 | ||||
| platform_packages = framework-arduinoespressif32 @ https://github.com/Aircoookie/arduino-esp32.git#1.0.6.4 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = -g | ||||
|   -DARDUINO_ARCH_ESP32 | ||||
|   #-DCONFIG_LITTLEFS_FOR_IDF_3_2 | ||||
| @@ -263,6 +265,7 @@ lib_deps = | ||||
| AR_build_flags = -D USERMOD_AUDIOREACTIVE  | ||||
|   -D sqrt_internal=sqrtf ;; -fsingle-precision-constant ;; forces ArduinoFFT to use float math (2x faster) | ||||
| AR_lib_deps = kosme/arduinoFFT @ 2.0.1 | ||||
| board_build.partitions = ${esp32.default_partitions}   ;; default partioning for 4MB Flash - can be overridden in build envs | ||||
|  | ||||
| [esp32_idf_V4] | ||||
| ;; experimental build environment for ESP32 using ESP-IDF 4.4.x / arduino-esp32 v2.0.5 | ||||
| @@ -272,6 +275,7 @@ AR_lib_deps = kosme/arduinoFFT @ 2.0.1 | ||||
| ;; You need to completely erase your device (esptool erase_flash) first, then install the "V4" build from VSCode+platformio. | ||||
| platform = espressif32@ ~6.3.2 | ||||
| platform_packages = platformio/framework-arduinoespressif32 @ 3.20009.0    ;; select arduino-esp32 v2.0.9 (arduino-esp32 2.0.10 thru 2.0.14 are buggy so avoid them) | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = -g | ||||
|   -Wshadow=compatible-local ;; emit warning in case a local variable "shadows" another local one | ||||
|   -DARDUINO_ARCH_ESP32 -DESP32 | ||||
| @@ -280,11 +284,13 @@ build_flags = -g | ||||
| lib_deps = | ||||
|   https://github.com/pbolduc/AsyncTCP.git @ 1.2.0 | ||||
|   ${env.lib_deps} | ||||
| board_build.partitions = ${esp32.default_partitions}   ;; default partioning for 4MB Flash - can be overridden in build envs | ||||
|  | ||||
| [esp32s2] | ||||
| ;; generic definitions for all ESP32-S2 boards | ||||
| platform = espressif32@ ~6.3.2 | ||||
| platform_packages = platformio/framework-arduinoespressif32 @ 3.20009.0    ;; select arduino-esp32 v2.0.9 (arduino-esp32 2.0.10 thru 2.0.14 are buggy so avoid them) | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = -g | ||||
|   -DARDUINO_ARCH_ESP32 | ||||
|   -DARDUINO_ARCH_ESP32S2 | ||||
| @@ -298,11 +304,13 @@ build_flags = -g | ||||
| lib_deps = | ||||
|   https://github.com/pbolduc/AsyncTCP.git @ 1.2.0 | ||||
|   ${env.lib_deps} | ||||
| board_build.partitions = ${esp32.default_partitions}   ;; default partioning for 4MB Flash - can be overridden in build envs | ||||
|  | ||||
| [esp32c3] | ||||
| ;; generic definitions for all ESP32-C3 boards | ||||
| platform = espressif32@ ~6.3.2 | ||||
| platform_packages = platformio/framework-arduinoespressif32 @ 3.20009.0    ;; select arduino-esp32 v2.0.9 (arduino-esp32 2.0.10 thru 2.0.14 are buggy so avoid them) | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = -g | ||||
|   -DARDUINO_ARCH_ESP32 | ||||
|   -DARDUINO_ARCH_ESP32C3 | ||||
| @@ -315,11 +323,13 @@ build_flags = -g | ||||
| lib_deps = | ||||
|   https://github.com/pbolduc/AsyncTCP.git @ 1.2.0 | ||||
|   ${env.lib_deps} | ||||
| board_build.partitions = ${esp32.default_partitions}   ;; default partioning for 4MB Flash - can be overridden in build envs | ||||
|  | ||||
| [esp32s3] | ||||
| ;; generic definitions for all ESP32-S3 boards | ||||
| platform = espressif32@ ~6.3.2 | ||||
| platform_packages = platformio/framework-arduinoespressif32 @ 3.20009.0    ;; select arduino-esp32 v2.0.9 (arduino-esp32 2.0.10 thru 2.0.14 are buggy so avoid them) | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = -g | ||||
|   -DESP32 | ||||
|   -DARDUINO_ARCH_ESP32 | ||||
| @@ -333,6 +343,7 @@ build_flags = -g | ||||
| lib_deps = | ||||
|   https://github.com/pbolduc/AsyncTCP.git @ 1.2.0 | ||||
|   ${env.lib_deps} | ||||
| board_build.partitions = ${esp32.large_partitions}   ;; default partioning for 8MB flash - can be overridden in build envs | ||||
|  | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
|   | ||||
| @@ -5,7 +5,7 @@ | ||||
| # Please visit documentation: https://docs.platformio.org/page/projectconf.html | ||||
|  | ||||
| [platformio] | ||||
| default_envs = WLED_tasmota_1M  # define as many as you need | ||||
| default_envs = WLED_generic8266_1M, esp32dev_V4_dio80  # put the name(s) of your own build environment here. You can define as many as you need | ||||
|  | ||||
| #---------- | ||||
| # SAMPLE | ||||
| @@ -28,8 +28,8 @@ lib_deps = ${esp8266.lib_deps} | ||||
| ;  robtillaart/SHT85@~0.3.3 | ||||
| ;  ;gmag11/QuickESPNow @ ~0.7.0 # will also load QuickDebug | ||||
| ;  https://github.com/blazoncek/QuickESPNow.git#optional-debug  ;; exludes debug library | ||||
| ;  ${esp32.AR_lib_deps} ;; used for USERMOD_AUDIOREACTIVE | ||||
| ;  bitbank2/PNGdec@^1.0.1 ;; used for POV display uncomment following | ||||
| ;  ${esp32.AR_lib_deps} ;; needed for USERMOD_AUDIOREACTIVE | ||||
|  | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags} ${esp8266.build_flags} | ||||
| @@ -141,7 +141,8 @@ build_flags = ${common.build_flags} ${esp8266.build_flags} | ||||
| ;   -D PIR_SENSOR_MAX_SENSORS=2 # max allowable sensors (uses OR logic for triggering) | ||||
| ; | ||||
| ; Use Audioreactive usermod and configure I2S microphone | ||||
| ;   -D USERMOD_AUDIOREACTIVE | ||||
| ;   ${esp32.AR_build_flags} ;; default flags required to properly configure ArduinoFFT | ||||
| ;   ;; don't forget to add ArduinoFFT to your libs_deps: ${esp32.AR_lib_deps} | ||||
| ;   -D AUDIOPIN=-1 | ||||
| ;   -D DMTYPE=1     # 0-analog/disabled, 1-I2S generic, 2-ES7243, 3-SPH0645, 4-I2S+mclk, 5-I2S PDM | ||||
| ;   -D I2S_SDPIN=36 | ||||
| @@ -157,17 +158,22 @@ build_flags = ${common.build_flags} ${esp8266.build_flags} | ||||
| ;   -D USERMOD_POV_DISPLAY | ||||
| ; Use built-in or custom LED as a status indicator (assumes LED is connected to GPIO16) | ||||
| ;   -D STATUSLED=16 | ||||
| ;    | ||||
| ; | ||||
| ; set the name of the module - make sure there is a quote-backslash-quote before the name and a backslash-quote-quote after the name | ||||
| ;   -D SERVERNAME="\"WLED\"" | ||||
| ;    | ||||
| ; | ||||
| ; set the number of LEDs | ||||
| ;   -D DEFAULT_LED_COUNT=30 | ||||
| ;   -D PIXEL_COUNTS=30 | ||||
| ; or this for multiple outputs | ||||
| ;   -D PIXEL_COUNTS=30,30 | ||||
| ; | ||||
| ; set the default LED type | ||||
| ;   -D DEFAULT_LED_TYPE=22    # see const.h (TYPE_xxxx) | ||||
| ;   -D LED_TYPES=22    # see const.h (TYPE_xxxx) | ||||
| ; or this for multiple outputs | ||||
| ;   -D LED_TYPES=TYPE_SK6812_RGBW,TYPE_WS2812_RGB | ||||
| ; | ||||
| ; set default color order of your led strip | ||||
| ;   -D DEFAULT_LED_COLOR_ORDER=COL_ORDER_GRB | ||||
| ; | ||||
| ; set milliampere limit when using ESP power pin (or inadequate PSU) to power LEDs | ||||
| ;   -D ABL_MILLIAMPS_DEFAULT=850 | ||||
| @@ -176,9 +182,6 @@ build_flags = ${common.build_flags} ${esp8266.build_flags} | ||||
| ; enable IR by setting remote type | ||||
| ;   -D IRTYPE=0   # 0 Remote disabled | 1 24-key RGB | 2 24-key with CT | 3 40-key blue | 4 40-key RGB | 5 21-key RGB | 6 6-key black | 7 9-key red | 8 JSON remote | ||||
| ;    | ||||
| ; set default color order of your led strip | ||||
| ;   -D DEFAULT_LED_COLOR_ORDER=COL_ORDER_GRB | ||||
| ; | ||||
| ; use PSRAM on classic ESP32 rev.1 (rev.3 or above has no issues) | ||||
| ;   -DBOARD_HAS_PSRAM -mfix-esp32-psram-cache-issue   # needed only for classic ESP32 rev.1 | ||||
| ; | ||||
| @@ -236,14 +239,13 @@ build_flags = ${common.build_flags} ${esp8266.build_flags} -D DATA_PINS=1 -D WLE | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:esp32dev_qio80] | ||||
| extends = env:esp32dev  # we want to extend the existing esp32dev environment (and define only updated options) | ||||
| board = esp32dev | ||||
| platform = ${esp32.platform} | ||||
| platform_packages = ${esp32.platform_packages} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags} ${esp32.build_flags} #-D WLED_DISABLE_BROWNOUT_DET | ||||
|   ${esp32.AR_build_flags} ;; optional - includes USERMOD_AUDIOREACTIVE | ||||
| lib_deps = ${esp32.lib_deps} | ||||
|   ${esp32.AR_lib_deps} ;; needed for USERMOD_AUDIOREACTIVE | ||||
| monitor_filters = esp32_exception_decoder | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
| board_build.f_flash = 80000000L | ||||
| board_build.flash_mode = qio | ||||
|  | ||||
| @@ -251,26 +253,25 @@ board_build.flash_mode = qio | ||||
| ;; experimental ESP32 env using ESP-IDF V4.4.x | ||||
| ;; Warning: this build environment is not stable!! | ||||
| ;; please erase your device before installing. | ||||
| extends = esp32_idf_V4  # based on newer "esp-idf V4" platform environment | ||||
| board = esp32dev | ||||
| platform = ${esp32_idf_V4.platform} | ||||
| platform_packages = ${esp32_idf_V4.platform_packages} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags}  ${esp32_idf_V4.build_flags} #-D WLED_DISABLE_BROWNOUT_DET | ||||
|   ${esp32.AR_build_flags} ;; includes USERMOD_AUDIOREACTIVE | ||||
| lib_deps = ${esp32_idf_V4.lib_deps} | ||||
|   ${esp32.AR_lib_deps} ;; needed for USERMOD_AUDIOREACTIVE | ||||
| monitor_filters = esp32_exception_decoder | ||||
| board_build.partitions = ${esp32_idf_V4.default_partitions} | ||||
| board_build.partitions = ${esp32.default_partitions}  ;; if you get errors about "out of program space", change this to ${esp32.extended_partitions} or even ${esp32.big_partitions} | ||||
| board_build.f_flash = 80000000L | ||||
| board_build.flash_mode = dio | ||||
|  | ||||
| [env:esp32s2_saola] | ||||
| extends = esp32s2 | ||||
| board = esp32-s2-saola-1 | ||||
| platform = ${esp32s2.platform} | ||||
| platform_packages = ${esp32s2.platform_packages} | ||||
| framework = arduino | ||||
| board_build.partitions = tools/WLED_ESP32_4MB_1MB_FS.csv | ||||
| board_build.flash_mode = qio | ||||
| upload_speed = 460800 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags} ${esp32s2.build_flags} | ||||
|   ;-DLOLIN_WIFI_FIX ;; try this in case Wifi does not work | ||||
|   -DARDUINO_USB_CDC_ON_BOOT=1 | ||||
| @@ -307,7 +308,7 @@ platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_4m1m} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags} ${esp8266.build_flags} -D WLED_USE_SHOJO_PCB | ||||
| build_flags = ${common.build_flags} ${esp8266.build_flags} -D WLED_USE_SHOJO_PCB ;; NB: WLED_USE_SHOJO_PCB is not used anywhere in the source code. Not sure why its needed. | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:d1_mini_debug] | ||||
| @@ -362,35 +363,48 @@ board_upload.flash_size = 2MB | ||||
| board_upload.maximum_size = 2097152 | ||||
|  | ||||
| [env:wemos_shield_esp32] | ||||
| extends = esp32              ;; use default esp32 platform | ||||
| board = esp32dev | ||||
| platform = ${esp32.platform} | ||||
| platform_packages = ${esp32.platform_packages} | ||||
| upload_speed = 460800 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags} ${esp32.build_flags} | ||||
|   -D WLED_RELEASE_NAME=\"ESP32_wemos_shield\" | ||||
|   -D DATA_PINS=16 | ||||
|   -D RLYPIN=19 | ||||
|   -D BTNPIN=17 | ||||
|   -D IRPIN=18 | ||||
|   -D UWLED_USE_MY_CONFIG | ||||
|   -UWLED_USE_MY_CONFIG | ||||
|   -D USERMOD_DALLASTEMPERATURE | ||||
|   -D USERMOD_FOUR_LINE_DISPLAY | ||||
|   -D TEMPERATURE_PIN=23 | ||||
|   -D USERMOD_AUDIOREACTIVE | ||||
|   ${esp32.AR_build_flags} ;; includes USERMOD_AUDIOREACTIVE | ||||
| lib_deps = ${esp32.lib_deps} | ||||
|   OneWire@~2.3.5 | ||||
|   olikraus/U8g2 @ ^2.28.8 | ||||
|   https://github.com/blazoncek/arduinoFFT.git | ||||
|   OneWire@~2.3.5          ;; needed for USERMOD_DALLASTEMPERATURE | ||||
|   olikraus/U8g2 @ ^2.28.8 ;; needed for USERMOD_FOUR_LINE_DISPLAY | ||||
|   ${esp32.AR_lib_deps}    ;; needed for USERMOD_AUDIOREACTIVE | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
|  | ||||
| [env:m5atom] | ||||
| board = esp32dev | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags} ${esp32.build_flags} -D DATA_PINS=27 -D BTNPIN=39 | ||||
| [env:esp32_pico-D4] | ||||
| extends = esp32              ;; use default esp32 platform | ||||
| board = pico32               ;; pico32-D4 is different from the standard esp32dev | ||||
|                              ;; hardware details from https://github.com/srg74/WLED-ESP32-pico | ||||
| build_flags = ${common.build_flags} ${esp32.build_flags} | ||||
|   -D WLED_RELEASE_NAME=\"pico32-D4\" -D SERVERNAME='"WLED-pico32"' | ||||
|   -D WLED_DISABLE_ADALIGHT   ;; no serial-to-USB chip on this board - better to disable serial protocols | ||||
|   -D DATA_PINS=2,18          ;; LED pins | ||||
|   -D RLYPIN=19 -D BTNPIN=0 -D IRPIN=-1 ;; no default pin for IR | ||||
|   ${esp32.AR_build_flags}    ;; include USERMOD_AUDIOREACTIVE | ||||
|   -D UM_AUDIOREACTIVE_ENABLE ;; enable AR by default | ||||
|   ;; Audioreactive settings for on-board microphone (ICS-43432) | ||||
|   -D SR_DMTYPE=1 -D I2S_SDPIN=25 -D I2S_WSPIN=15 -D I2S_CKPIN=14 | ||||
|   -D SR_SQUELCH=5 -D SR_GAIN=30 | ||||
| lib_deps = ${esp32.lib_deps} | ||||
| platform = ${esp32.platform} | ||||
| platform_packages = ${esp32.platform_packages} | ||||
|   ${esp32.AR_lib_deps}       ;; needed for USERMOD_AUDIOREACTIVE | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
| board_build.f_flash = 80000000L | ||||
|  | ||||
| [env:m5atom] | ||||
| extends = env:esp32dev  # we want to extend the existing esp32dev environment (and define only updated options) | ||||
| build_flags = ${common.build_flags} ${esp32.build_flags} -D DATA_PINS=27 -D BTNPIN=39 | ||||
|  | ||||
| [env:sp501e] | ||||
| board = esp_wroom_02 | ||||
| @@ -413,7 +427,7 @@ platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags} ${esp8266.build_flags} -D BTNPIN=-1 -D RLYPIN=-1 -D DATA_PINS=4,12,14,13,5 | ||||
|                                             -D DEFAULT_LED_TYPE=TYPE_ANALOG_5CH -D WLED_DISABLE_INFRARED -D WLED_MAX_CCT_BLEND=0 | ||||
|                                             -D LED_TYPES=TYPE_ANALOG_5CH -D WLED_DISABLE_INFRARED -D WLED_MAX_CCT_BLEND=0 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:Athom_15w_RGBCW]        ;15w bulb | ||||
| @@ -423,7 +437,7 @@ platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags} ${esp8266.build_flags} -D BTNPIN=-1 -D RLYPIN=-1 -D DATA_PINS=4,12,14,5,13 | ||||
|                                             -D DEFAULT_LED_TYPE=TYPE_ANALOG_5CH -D WLED_DISABLE_INFRARED -D WLED_MAX_CCT_BLEND=0 -D WLED_USE_IC_CCT | ||||
|                                             -D LED_TYPES=TYPE_ANALOG_5CH -D WLED_DISABLE_INFRARED -D WLED_MAX_CCT_BLEND=0 -D WLED_USE_IC_CCT | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:Athom_3Pin_Controller]        ;small controller with only data | ||||
| @@ -489,9 +503,8 @@ lib_deps = ${esp8266.lib_deps} | ||||
| # EleksTube-IPS | ||||
| # ------------------------------------------------------------------------------ | ||||
| [env:elekstube_ips] | ||||
| extends = esp32              ;; use default esp32 platform | ||||
| board = esp32dev | ||||
| platform = ${esp32.platform} | ||||
| platform_packages = ${esp32.platform_packages} | ||||
| upload_speed = 921600 | ||||
| build_flags = ${common.build_flags} ${esp32.build_flags} -D WLED_DISABLE_BROWNOUT_DET -D WLED_DISABLE_INFRARED | ||||
|   -D USERMOD_RTC | ||||
| @@ -499,7 +512,7 @@ build_flags = ${common.build_flags} ${esp32.build_flags} -D WLED_DISABLE_BROWNOU | ||||
|   -D DATA_PINS=12 | ||||
|   -D RLYPIN=27 | ||||
|   -D BTNPIN=34 | ||||
|   -D DEFAULT_LED_COUNT=6 | ||||
|   -D PIXEL_COUNTS=6 | ||||
|   # Display config | ||||
|   -D ST7789_DRIVER | ||||
|   -D TFT_WIDTH=135 | ||||
| @@ -515,5 +528,4 @@ build_flags = ${common.build_flags} ${esp32.build_flags} -D WLED_DISABLE_BROWNOU | ||||
| monitor_filters = esp32_exception_decoder | ||||
| lib_deps = | ||||
|   ${esp32.lib_deps} | ||||
|   TFT_eSPI @ ^2.3.70 | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
|   TFT_eSPI @ 2.5.33  ;; this is the last version that compiles with the WLED default framework - newer versions require platform = espressif32 @ ^6.3.2 | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								tools/AutoCubeMap.xlsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								tools/AutoCubeMap.xlsx
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -444,6 +444,7 @@ public: | ||||
|     configComplete &= getJsonValue(top[F("PublishAlways")], PublishAlways, false); | ||||
|     configComplete &= getJsonValue(top[F("UseCelsius")], UseCelsius, true); | ||||
|     configComplete &= getJsonValue(top[F("HomeAssistantDiscovery")], HomeAssistantDiscovery, false); | ||||
|     tempScale = UseCelsius ? "°C" : "°F"; | ||||
|  | ||||
|     DEBUG_PRINT(FPSTR(_name)); | ||||
|     if (!initDone) { | ||||
|   | ||||
| @@ -75,7 +75,7 @@ static uint8_t  soundAgc = 0;                   // Automagic gain control: 0 - n | ||||
| //static float    volumeSmth = 0.0f;              // either sampleAvg or sampleAgc depending on soundAgc; smoothed sample | ||||
| static float FFT_MajorPeak = 1.0f;              // FFT: strongest (peak) frequency | ||||
| static float FFT_Magnitude = 0.0f;              // FFT: volume (magnitude) of peak frequency | ||||
| static bool samplePeak = false;      // Boolean flag for peak - used in effects. Responding routine may reset this flag. Auto-reset after strip.getMinShowDelay() | ||||
| static bool samplePeak = false;      // Boolean flag for peak - used in effects. Responding routine may reset this flag. Auto-reset after strip.getFrameTime() | ||||
| static bool udpSamplePeak = false;   // Boolean flag for peak. Set at the same time as samplePeak, but reset by transmitAudioData | ||||
| static unsigned long timeOfPeak = 0; // time of last sample peak detection. | ||||
| static uint8_t fftResult[NUM_GEQ_CHANNELS]= {0};// Our calculated freq. channel result table to be used by effects | ||||
| @@ -536,8 +536,8 @@ static void detectSamplePeak(void) { | ||||
| #endif | ||||
|  | ||||
| static void autoResetPeak(void) { | ||||
|   uint16_t MinShowDelay = MAX(50, strip.getMinShowDelay());  // Fixes private class variable compiler error. Unsure if this is the correct way of fixing the root problem. -THATDONFC | ||||
|   if (millis() - timeOfPeak > MinShowDelay) {          // Auto-reset of samplePeak after a complete frame has passed. | ||||
|   uint16_t peakDelay = max(uint16_t(50), strip.getFrameTime()); | ||||
|   if (millis() - timeOfPeak > peakDelay) {          // Auto-reset of samplePeak after at least one complete frame has passed. | ||||
|     samplePeak = false; | ||||
|     if (audioSyncEnabled == 0) udpSamplePeak = false;  // this is normally reset by transmitAudioData | ||||
|   } | ||||
|   | ||||
| @@ -9,7 +9,7 @@ The actual / original code that controls the LED modes is from Adam Zeloof. I ta | ||||
| It was quite a bit more work than I hoped, but I got there eventually :) | ||||
|  | ||||
| ## Requirements | ||||
| * "ESP Rotary" by Lennart Hennigs, v1.5.0 or higher: https://github.com/LennartHennigs/ESPRotary | ||||
| * "ESP Rotary" by Lennart Hennigs, v2.1.1 or higher: https://github.com/LennartHennigs/ESPRotary | ||||
|  | ||||
| ## Usermod installation | ||||
| Simply copy the below block (build task) to your `platformio_override.ini` and compile WLED using this new build task. Or use an existing one and add the buildflag `-D RGB_ROTARY_ENCODER`. | ||||
| @@ -20,7 +20,7 @@ ESP32: | ||||
| extends = env:esp32dev | ||||
| build_flags = ${common.build_flags_esp32} -D WLED_RELEASE_NAME=ESP32 -D RGB_ROTARY_ENCODER | ||||
| lib_deps = ${esp32.lib_deps} | ||||
|     lennarthennigs/ESP Rotary@^1.5.0 | ||||
|     lennarthennigs/ESP Rotary@^2.1.1 | ||||
| ``` | ||||
|  | ||||
| ESP8266 / D1 Mini: | ||||
| @@ -29,7 +29,7 @@ ESP8266 / D1 Mini: | ||||
| extends = env:d1_mini | ||||
| build_flags = ${common.build_flags_esp8266} -D RGB_ROTARY_ENCODER | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|     lennarthennigs/ESP Rotary@^1.5.0 | ||||
|     lennarthennigs/ESP Rotary@^2.1.1 | ||||
| ``` | ||||
|  | ||||
| ## How to connect the board to your ESP | ||||
|   | ||||
| @@ -600,11 +600,12 @@ static const char _data_FX_MODE_TWINKLE[] PROGMEM = "Twinkle@!,!;!,!;!;;m12=0"; | ||||
|  * Dissolve function | ||||
|  */ | ||||
| uint16_t dissolve(uint32_t color) { | ||||
|   unsigned dataSize = (SEGLEN+7) >> 3; //1 bit per LED | ||||
|   unsigned dataSize = sizeof(uint32_t) * SEGLEN; | ||||
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed | ||||
|   uint32_t* pixels = reinterpret_cast<uint32_t*>(SEGENV.data); | ||||
|  | ||||
|   if (SEGENV.call == 0) { | ||||
|     memset(SEGMENT.data, 0xFF, dataSize); // start by fading pixels up | ||||
|     for (unsigned i = 0; i < SEGLEN; i++) pixels[i] = SEGCOLOR(1); | ||||
|     SEGENV.aux0 = 1; | ||||
|   } | ||||
|  | ||||
| @@ -612,33 +613,26 @@ uint16_t dissolve(uint32_t color) { | ||||
|     if (random8() <= SEGMENT.intensity) { | ||||
|       for (size_t times = 0; times < 10; times++) { //attempt to spawn a new pixel 10 times | ||||
|         unsigned i = random16(SEGLEN); | ||||
|         unsigned index = i >> 3; | ||||
|         unsigned bitNum = i & 0x07; | ||||
|         bool fadeUp = bitRead(SEGENV.data[index], bitNum); | ||||
|         if (SEGENV.aux0) { //dissolve to primary/palette | ||||
|           if (fadeUp) { | ||||
|             if (color == SEGCOLOR(0)) { | ||||
|               SEGMENT.setPixelColor(i, SEGMENT.color_from_palette(i, true, PALETTE_SOLID_WRAP, 0)); | ||||
|             } else { | ||||
|               SEGMENT.setPixelColor(i, color); | ||||
|             } | ||||
|             bitWrite(SEGENV.data[index], bitNum, false); | ||||
|           if (pixels[i] == SEGCOLOR(1)) { | ||||
|             pixels[i] = color == SEGCOLOR(0) ? SEGMENT.color_from_palette(i, true, PALETTE_SOLID_WRAP, 0) : color; | ||||
|             break; //only spawn 1 new pixel per frame per 50 LEDs | ||||
|           } | ||||
|         } else { //dissolve to secondary | ||||
|           if (!fadeUp) { | ||||
|             SEGMENT.setPixelColor(i, SEGCOLOR(1)); break; | ||||
|             bitWrite(SEGENV.data[index], bitNum, true); | ||||
|           if (pixels[i] != SEGCOLOR(1)) { | ||||
|             pixels[i] = SEGCOLOR(1); | ||||
|             break; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   // fix for #4401 | ||||
|   for (unsigned i = 0; i < SEGLEN; i++) SEGMENT.setPixelColor(i, pixels[i]); | ||||
|  | ||||
|   if (SEGENV.step > (255 - SEGMENT.speed) + 15U) { | ||||
|     SEGENV.aux0 = !SEGENV.aux0; | ||||
|     SEGENV.step = 0; | ||||
|     memset(SEGMENT.data, (SEGENV.aux0 ? 0xFF : 0), dataSize); // switch fading | ||||
|   } else { | ||||
|     SEGENV.step++; | ||||
|   } | ||||
| @@ -1097,7 +1091,7 @@ uint16_t mode_running_random(void) { | ||||
|  | ||||
|   unsigned z = it % zoneSize; | ||||
|   bool nzone = (!z && it != SEGENV.aux1); | ||||
|   for (unsigned i=SEGLEN-1; i > 0; i--) { | ||||
|   for (int i=SEGLEN-1; i >= 0; i--) { | ||||
|     if (nzone || z >= zoneSize) { | ||||
|       unsigned lastrand = PRNG16 >> 8; | ||||
|       int16_t diff = 0; | ||||
| @@ -1441,7 +1435,7 @@ uint16_t mode_fairy() { | ||||
|     if (z == zones-1) flashersInZone = numFlashers-(flashersInZone*(zones-1)); | ||||
|  | ||||
|     for (unsigned f = firstFlasher; f < firstFlasher + flashersInZone; f++) { | ||||
|       unsigned stateTime = now16 - flashers[f].stateStart; | ||||
|       unsigned stateTime = uint16_t(now16 - flashers[f].stateStart); | ||||
|       //random on/off time reached, switch state | ||||
|       if (stateTime > flashers[f].stateDur * 10) { | ||||
|         flashers[f].stateOn = !flashers[f].stateOn; | ||||
| @@ -1500,7 +1494,7 @@ uint16_t mode_fairytwinkle() { | ||||
|   unsigned maxDur = riseFallTime/100 + ((255 - SEGMENT.intensity) >> 2) + 13 + ((255 - SEGMENT.intensity) >> 1); | ||||
|  | ||||
|   for (int f = 0; f < SEGLEN; f++) { | ||||
|     unsigned stateTime = now16 - flashers[f].stateStart; | ||||
|     uint16_t stateTime = now16 - flashers[f].stateStart; | ||||
|     //random on/off time reached, switch state | ||||
|     if (stateTime > flashers[f].stateDur * 100) { | ||||
|       flashers[f].stateOn = !flashers[f].stateOn; | ||||
| @@ -1745,7 +1739,7 @@ uint16_t mode_random_chase(void) { | ||||
|   uint32_t color = SEGENV.step; | ||||
|   random16_set_seed(SEGENV.aux0); | ||||
|  | ||||
|   for (unsigned i = SEGLEN -1; i > 0; i--) { | ||||
|   for (int i = SEGLEN -1; i >= 0; i--) { | ||||
|     uint8_t r = random8(6) != 0 ? (color >> 16 & 0xFF) : random8(); | ||||
|     uint8_t g = random8(6) != 0 ? (color >> 8  & 0xFF) : random8(); | ||||
|     uint8_t b = random8(6) != 0 ? (color       & 0xFF) : random8(); | ||||
| @@ -1798,7 +1792,7 @@ uint16_t mode_oscillate(void) { | ||||
|     // if the counter has increased, move the oscillator by the random step | ||||
|     if (it != SEGENV.step) oscillators[i].pos += oscillators[i].dir * oscillators[i].speed; | ||||
|     oscillators[i].size = SEGLEN/(3+SEGMENT.intensity/8); | ||||
|     if((oscillators[i].dir == -1) && (oscillators[i].pos <= 0)) { | ||||
|     if((oscillators[i].dir == -1) && (oscillators[i].pos > SEGLEN << 1)) { // use integer overflow | ||||
|       oscillators[i].pos = 0; | ||||
|       oscillators[i].dir = 1; | ||||
|       // make bigger steps for faster speeds | ||||
| @@ -1814,8 +1808,8 @@ uint16_t mode_oscillate(void) { | ||||
|   for (unsigned i = 0; i < SEGLEN; i++) { | ||||
|     uint32_t color = BLACK; | ||||
|     for (unsigned j = 0; j < numOscillators; j++) { | ||||
|       if(i >= (unsigned)oscillators[j].pos - oscillators[j].size && i <= oscillators[j].pos + oscillators[j].size) { | ||||
|         color = (color == BLACK) ? SEGCOLOR(j) : color_blend(color, SEGCOLOR(j), 128); | ||||
|       if((int)i >= (int)oscillators[j].pos - oscillators[j].size && i <= oscillators[j].pos + oscillators[j].size) { | ||||
|         color = (color == BLACK) ? SEGCOLOR(j) : color_blend(color, SEGCOLOR(j), uint8_t(128)); | ||||
|       } | ||||
|     } | ||||
|     SEGMENT.setPixelColor(i, color); | ||||
| @@ -2003,7 +1997,7 @@ uint16_t mode_palette() { | ||||
|       const mathType sourceX = xtSinTheta + ytCosTheta + centerX; | ||||
|       // The computation was scaled just right so that the result should always be in range [0, maxXOut], but enforce this anyway | ||||
|       // to account for imprecision. Then scale it so that the range is [0, 255], which we can use with the palette. | ||||
|       int colorIndex = (std::min(std::max(sourceX, mathType(0)), maxXOut * sInt16Scale) * 255) / (sInt16Scale * maxXOut); | ||||
|       int colorIndex = (std::min(std::max(sourceX, mathType(0)), maxXOut * sInt16Scale) * wideMathType(255)) / (sInt16Scale * maxXOut); | ||||
|       // inputSize determines by how much we want to scale the palette: | ||||
|       // values < 128 display a fraction of a palette, | ||||
|       // values > 128 display multiple palettes. | ||||
| @@ -2565,11 +2559,11 @@ static CRGB twinklefox_one_twinkle(uint32_t ms, uint8_t salt, bool cat) | ||||
| { | ||||
|   // Overall twinkle speed (changed) | ||||
|   unsigned ticks = ms / SEGENV.aux0; | ||||
|   unsigned fastcycle8 = ticks; | ||||
|   unsigned slowcycle16 = (ticks >> 8) + salt; | ||||
|   unsigned fastcycle8 = uint8_t(ticks); | ||||
|   uint16_t slowcycle16 = (ticks >> 8) + salt; | ||||
|   slowcycle16 += sin8_t(slowcycle16); | ||||
|   slowcycle16 = (slowcycle16 * 2053) + 1384; | ||||
|   unsigned slowcycle8 = (slowcycle16 & 0xFF) + (slowcycle16 >> 8); | ||||
|   uint8_t slowcycle8 = (slowcycle16 & 0xFF) + (slowcycle16 >> 8); | ||||
|  | ||||
|   // Overall twinkle density. | ||||
|   // 0 (NONE lit) to 8 (ALL lit at once). | ||||
| @@ -5170,7 +5164,7 @@ uint16_t mode_2Dgameoflife(void) { // Written by Ewoud Wijma, inspired by https: | ||||
|         neighbors++; | ||||
|         bool colorFound = false; | ||||
|         int k; | ||||
|         for (k=0; k<9 && colorsCount[i].count != 0; k++) | ||||
|         for (k=0; k<9 && colorsCount[k].count != 0; k++) | ||||
|           if (colorsCount[k].color == prevLeds[xy]) { | ||||
|             colorsCount[k].count++; | ||||
|             colorFound = true; | ||||
| @@ -6639,14 +6633,16 @@ static const char _data_FX_MODE_JUGGLES[] PROGMEM = "Juggles@!,# of balls;!,!;!; | ||||
| //   * MATRIPIX     // | ||||
| ////////////////////// | ||||
| uint16_t mode_matripix(void) {                  // Matripix. By Andrew Tuline. | ||||
|   if (SEGLEN == 1) return mode_static(); | ||||
|   // even with 1D effect we have to take logic for 2D segments for allocation as fill_solid() fills whole segment | ||||
|   // effect can work on single pixels, we just lose the shifting effect | ||||
|   unsigned dataSize = sizeof(uint32_t) * SEGLEN; | ||||
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed | ||||
|   uint32_t* pixels = reinterpret_cast<uint32_t*>(SEGENV.data); | ||||
|  | ||||
|   um_data_t *um_data = getAudioData(); | ||||
|   int volumeRaw    = *(int16_t*)um_data->u_data[1]; | ||||
|  | ||||
|   if (SEGENV.call == 0) { | ||||
|     SEGMENT.fill(BLACK); | ||||
|     for (unsigned i = 0; i < SEGLEN; i++) pixels[i] = BLACK;   // may not be needed as resetIfRequired() clears buffer | ||||
|   } | ||||
|  | ||||
|   uint8_t secondHand = micros()/(256-SEGMENT.speed)/500 % 16; | ||||
| @@ -6654,8 +6650,14 @@ uint16_t mode_matripix(void) {                  // Matripix. By Andrew Tuline. | ||||
|     SEGENV.aux0 = secondHand; | ||||
|  | ||||
|     int pixBri = volumeRaw * SEGMENT.intensity / 64; | ||||
|     for (int i = 0; i < SEGLEN-1; i++) SEGMENT.setPixelColor(i, SEGMENT.getPixelColor(i+1)); // shift left | ||||
|     SEGMENT.setPixelColor(SEGLEN-1, color_blend(SEGCOLOR(1), SEGMENT.color_from_palette(strip.now, false, PALETTE_SOLID_WRAP, 0), pixBri)); | ||||
|     unsigned k = SEGLEN-1; | ||||
|     // loop will not execute if SEGLEN equals 1 | ||||
|     for (unsigned i = 0; i < k; i++) { | ||||
|       pixels[i] = pixels[i+1]; // shift left | ||||
|       SEGMENT.setPixelColor(i, pixels[i]); | ||||
|     } | ||||
|     pixels[k] = color_blend(SEGCOLOR(1), SEGMENT.color_from_palette(strip.now, false, PALETTE_SOLID_WRAP, 0), pixBri); | ||||
|     SEGMENT.setPixelColor(k, pixels[k]); | ||||
|   } | ||||
|  | ||||
|   return FRAMETIME; | ||||
| @@ -7283,8 +7285,11 @@ static const char _data_FX_MODE_ROCKTAVES[] PROGMEM = "Rocktaves@;!,!;!;01f;m12= | ||||
| // Combines peak detection with FFT_MajorPeak and FFT_Magnitude. | ||||
| uint16_t mode_waterfall(void) {                   // Waterfall. By: Andrew Tuline | ||||
|   // effect can work on single pixels, we just lose the shifting effect | ||||
|    | ||||
|   um_data_t *um_data = getAudioData(); | ||||
|   unsigned dataSize = sizeof(uint32_t) * SEGLEN; | ||||
|   if (!SEGENV.allocateData(dataSize)) return mode_static(); //allocation failed | ||||
|   uint32_t* pixels = reinterpret_cast<uint32_t*>(SEGENV.data); | ||||
|  | ||||
|   um_data_t *um_data    = getAudioData(); | ||||
|   uint8_t samplePeak    = *(uint8_t*)um_data->u_data[3]; | ||||
|   float   FFT_MajorPeak = *(float*)  um_data->u_data[4]; | ||||
|   uint8_t *maxVol       =  (uint8_t*)um_data->u_data[6]; | ||||
| @@ -7294,7 +7299,7 @@ uint16_t mode_waterfall(void) {                   // Waterfall. By: Andrew Tulin | ||||
|   if (FFT_MajorPeak < 1) FFT_MajorPeak = 1;                                         // log10(0) is "forbidden" (throws exception) | ||||
|  | ||||
|   if (SEGENV.call == 0) { | ||||
|     SEGMENT.fill(BLACK); | ||||
|     for (unsigned i = 0; i < SEGLEN; i++) pixels[i] = BLACK;   // may not be needed as resetIfRequired() clears buffer | ||||
|     SEGENV.aux0 = 255; | ||||
|     SEGMENT.custom1 = *binNum; | ||||
|     SEGMENT.custom2 = *maxVol * 2; | ||||
| @@ -7311,13 +7316,18 @@ uint16_t mode_waterfall(void) {                   // Waterfall. By: Andrew Tulin | ||||
|     uint8_t pixCol = (log10f(FFT_MajorPeak) - 2.26f) * 150;           // 22Khz sampling - log10 frequency range is from 2.26 (182hz) to 3.967 (9260hz). Let's scale accordingly. | ||||
|     if (FFT_MajorPeak < 182.0f) pixCol = 0;                           // handle underflow | ||||
|  | ||||
|     unsigned k = SEGLEN-1; | ||||
|     if (samplePeak) { | ||||
|       SEGMENT.setPixelColor(SEGLEN-1, CHSV(92,92,92)); | ||||
|       pixels[k] = (uint32_t)CRGB(CHSV(92,92,92)); | ||||
|     } else { | ||||
|       SEGMENT.setPixelColor(SEGLEN-1, color_blend(SEGCOLOR(1), SEGMENT.color_from_palette(pixCol+SEGMENT.intensity, false, PALETTE_SOLID_WRAP, 0), (int)my_magnitude)); | ||||
|       pixels[k] = color_blend(SEGCOLOR(1), SEGMENT.color_from_palette(pixCol+SEGMENT.intensity, false, PALETTE_SOLID_WRAP, 0), (uint8_t)my_magnitude); | ||||
|     } | ||||
|     SEGMENT.setPixelColor(k, pixels[k]); | ||||
|     // loop will not execute if SEGLEN equals 1 | ||||
|     for (int i = 0; i < SEGLEN-1; i++) SEGMENT.setPixelColor(i, SEGMENT.getPixelColor(i+1)); // shift left | ||||
|     for (unsigned i = 0; i < k; i++) { | ||||
|       pixels[i] = pixels[i+1]; // shift left | ||||
|       SEGMENT.setPixelColor(i, pixels[i]); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return FRAMETIME; | ||||
|   | ||||
							
								
								
									
										14
									
								
								wled00/FX.h
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								wled00/FX.h
									
									
									
									
									
								
							| @@ -46,6 +46,14 @@ | ||||
| #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 | ||||
| @@ -77,8 +85,6 @@ | ||||
|   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 MIN_SHOW_DELAY   (_frametime < 16 ? 8 : 15) | ||||
|  | ||||
| #define NUM_COLORS       3 /* number of colors per segment */ | ||||
| #define SEGMENT          strip._segments[strip.getCurrSegmentId()] | ||||
| #define SEGENV           strip._segments[strip.getCurrSegmentId()] | ||||
| @@ -748,6 +754,7 @@ class WS2812FX {  // 96 bytes | ||||
|       customMappingTable(nullptr), | ||||
|       customMappingSize(0), | ||||
|       _lastShow(0), | ||||
|       _lastServiceShow(0), | ||||
|       _segment_index(0), | ||||
|       _mainSegment(0) | ||||
|     { | ||||
| @@ -846,7 +853,7 @@ class WS2812FX {  // 96 bytes | ||||
|       getMappedPixelIndex(uint16_t index) const; | ||||
|  | ||||
|     inline uint16_t getFrameTime() const    { return _frametime; }        // returns amount of time a frame should take (in ms) | ||||
|     inline uint16_t getMinShowDelay() const { return MIN_SHOW_DELAY; }    // returns minimum amount of time strip.service() can be delayed (constant) | ||||
|     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) | ||||
|  | ||||
| @@ -958,6 +965,7 @@ class WS2812FX {  // 96 bytes | ||||
|     uint16_t  customMappingSize; | ||||
|  | ||||
|     unsigned long _lastShow; | ||||
|     unsigned long _lastServiceShow; | ||||
|  | ||||
|     uint8_t _segment_index; | ||||
|     uint8_t _mainSegment; | ||||
|   | ||||
| @@ -1304,7 +1304,14 @@ void WS2812FX::finalizeInit() { | ||||
| void WS2812FX::service() { | ||||
|   unsigned long nowUp = millis(); // Be aware, millis() rolls over every 49 days | ||||
|   now = nowUp + timebase; | ||||
|   if (nowUp - _lastShow < MIN_SHOW_DELAY || _suspend) return; | ||||
|   if (_suspend) return; | ||||
|   unsigned long elapsed = nowUp - _lastServiceShow; | ||||
|  | ||||
|   if (elapsed <= MIN_FRAME_DELAY) return;                                        // keep wifi alive - no matter if triggered or unlimited | ||||
|   if ( !_triggered && (_targetFps != FPS_UNLIMITED)) {                           // unlimited mode = no frametime | ||||
|     if (elapsed < _frametime) return;                                            // too early for service | ||||
|   } | ||||
|  | ||||
|   bool doShow = false; | ||||
|  | ||||
|   _isServicing = true; | ||||
| @@ -1321,7 +1328,7 @@ void WS2812FX::service() { | ||||
|     if (!seg.isActive()) continue; | ||||
|  | ||||
|     // last condition ensures all solid segments are updated at the same time | ||||
|     if (nowUp > seg.next_time || _triggered || (doShow && seg.mode == FX_MODE_STATIC)) | ||||
|     if (nowUp >= seg.next_time || _triggered || (doShow && seg.mode == FX_MODE_STATIC)) | ||||
|     { | ||||
|       doShow = true; | ||||
|       unsigned frameDelay = FRAMETIME; | ||||
| @@ -1371,15 +1378,16 @@ void WS2812FX::service() { | ||||
|   _triggered = false; | ||||
|  | ||||
|   #ifdef WLED_DEBUG | ||||
|   if (millis() - nowUp > _frametime) DEBUG_PRINTF_P(PSTR("Slow effects %u/%d.\n"), (unsigned)(millis()-nowUp), (int)_frametime); | ||||
|   if ((_targetFps != FPS_UNLIMITED) && (millis() - nowUp > _frametime)) DEBUG_PRINTF_P(PSTR("Slow effects %u/%d.\n"), (unsigned)(millis()-nowUp), (int)_frametime); | ||||
|   #endif | ||||
|   if (doShow) { | ||||
|     yield(); | ||||
|     Segment::handleRandomPalette(); // slowly transition random palette; move it into for loop when each segment has individual random palette | ||||
|     show(); | ||||
|     _lastServiceShow = nowUp; // update timestamp, for precise FPS control | ||||
|   } | ||||
|   #ifdef WLED_DEBUG | ||||
|   if (millis() - nowUp > _frametime) DEBUG_PRINTF_P(PSTR("Slow strip %u/%d.\n"), (unsigned)(millis()-nowUp), (int)_frametime); | ||||
|   if ((_targetFps != FPS_UNLIMITED) && (millis() - nowUp > _frametime)) DEBUG_PRINTF_P(PSTR("Slow strip %u/%d.\n"), (unsigned)(millis()-nowUp), (int)_frametime); | ||||
|   #endif | ||||
| } | ||||
|  | ||||
| @@ -1399,13 +1407,13 @@ void WS2812FX::show() { | ||||
|   // avoid race condition, capture _callback value | ||||
|   show_callback callback = _callback; | ||||
|   if (callback) callback(); | ||||
|   unsigned long showNow = 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 | ||||
|   BusManager::show(); | ||||
|  | ||||
|   unsigned long showNow = millis(); | ||||
|   size_t diff = showNow - _lastShow; | ||||
|  | ||||
|   if (diff > 0) { // skip calculation if no time has passed | ||||
| @@ -1433,8 +1441,9 @@ uint16_t WS2812FX::getFps() const { | ||||
| } | ||||
|  | ||||
| void WS2812FX::setTargetFps(uint8_t fps) { | ||||
|   if (fps > 0 && fps <= 120) _targetFps = fps; | ||||
|   _frametime = 1000 / _targetFps; | ||||
|   if (fps <= 250) _targetFps = fps; | ||||
|   if (_targetFps > 0) _frametime = 1000 / _targetFps; | ||||
|   else _frametime = MIN_FRAME_DELAY;     // unlimited mode | ||||
| } | ||||
|  | ||||
| void WS2812FX::setMode(uint8_t segid, uint8_t m) { | ||||
| @@ -1482,7 +1491,7 @@ void WS2812FX::setBrightness(uint8_t b, bool direct) { | ||||
|   BusManager::setBrightness(b); | ||||
|   if (!direct) { | ||||
|     unsigned long t = millis(); | ||||
|     if (_segments[0].next_time > t + 22 && t - _lastShow > MIN_SHOW_DELAY) trigger(); //apply brightness change immediately if no refresh soon | ||||
|     if (_segments[0].next_time > t + 22 && t - _lastShow > MIN_FRAME_DELAY) trigger(); //apply brightness change immediately if no refresh soon | ||||
|   } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -155,16 +155,6 @@ BusDigital::BusDigital(BusConfig &bc, uint8_t nr, const ColorOrderMap &com) | ||||
|   DEBUG_PRINTF_P(PSTR("%successfully inited strip %u (len %u) with type %u and pins %u,%u (itype %u). mA=%d/%d\n"), _valid?"S":"Uns", nr, bc.count, bc.type, _pins[0], is2Pin(bc.type)?_pins[1]:255, _iType, _milliAmpsPerLed, _milliAmpsMax); | ||||
| } | ||||
|  | ||||
| //fine tune power estimation constants for your setup | ||||
| //you can set it to 0 if the ESP is powered by USB and the LEDs by external | ||||
| #ifndef MA_FOR_ESP | ||||
|   #ifdef ESP8266 | ||||
|     #define MA_FOR_ESP         80 //how much mA does the ESP use (Wemos D1 about 80mA) | ||||
|   #else | ||||
|     #define MA_FOR_ESP        120 //how much mA does the ESP use (ESP32 about 120mA) | ||||
|   #endif | ||||
| #endif | ||||
|  | ||||
| //DISCLAIMER | ||||
| //The following function attemps to calculate the current LED power usage, | ||||
| //and will limit the brightness to stay below a set amperage threshold. | ||||
| @@ -567,19 +557,15 @@ void BusPwm::show() { | ||||
|   const unsigned maxBri = (1<<_depth);      // possible values: 16384 (14), 8192 (13), 4096 (12), 2048 (11), 1024 (10), 512 (9) and 256 (8)  | ||||
|   [[maybe_unused]] const unsigned bitShift = dithering * 4;  // if dithering, _depth is 12 bit but LEDC channel is set to 8 bit (using 4 fractional bits) | ||||
|  | ||||
|   // use CIE brightness formula (cubic) to fit (or approximate linearity of) human eye perceived brightness | ||||
|   // the formula is based on 12 bit resolution as there is no need for greater precision | ||||
|   // use CIE brightness formula (linear + cubic) to approximate human eye perceived brightness | ||||
|   // see: https://en.wikipedia.org/wiki/Lightness | ||||
|   unsigned pwmBri = (unsigned)_bri * 100;  // enlarge to use integer math for linear response | ||||
|   if (pwmBri < 2040) { | ||||
|     // linear response for values [0-20] | ||||
|     pwmBri = ((pwmBri << 12) + 115043) / 230087; //adding '0.5' before division for correct rounding | ||||
|   } else { | ||||
|     // cubic response for values [21-255] | ||||
|     pwmBri += 4080; | ||||
|     float temp = (float)pwmBri / 29580.0f; | ||||
|     temp = temp * temp * temp * (float)maxBri;  | ||||
|     pwmBri = (unsigned)temp;  // pwmBri is in range [0-maxBri]  | ||||
|   unsigned pwmBri = _bri; | ||||
|   if (pwmBri < 21) {                                   // linear response for values [0-20] | ||||
|     pwmBri = (pwmBri * maxBri + 2300 / 2) / 2300 ;     // adding '0.5' before division for correct rounding, 2300 gives a good match to CIE curve | ||||
|   } else {                                             // cubic response for values [21-255] | ||||
|     float temp = float(pwmBri + 41) / float(255 + 41); // 41 is to match offset & slope to linear part | ||||
|     temp = temp * temp * temp * (float)maxBri; | ||||
|     pwmBri = (unsigned)temp;                           // pwmBri is in range [0-maxBri] C | ||||
|   } | ||||
|  | ||||
|   [[maybe_unused]] unsigned hPoint = 0;  // phase shift (0 - maxBri) | ||||
| @@ -943,7 +929,6 @@ void BusManager::show() { | ||||
|     busses[i]->show(); | ||||
|     _milliAmpsUsed += busses[i]->getUsedCurrent(); | ||||
|   } | ||||
|   if (_milliAmpsUsed) _milliAmpsUsed += MA_FOR_ESP; | ||||
| } | ||||
|  | ||||
| void BusManager::setStatusPixel(uint32_t c) { | ||||
|   | ||||
| @@ -363,6 +363,16 @@ struct BusConfig { | ||||
| }; | ||||
|  | ||||
|  | ||||
| //fine tune power estimation constants for your setup | ||||
| //you can set it to 0 if the ESP is powered by USB and the LEDs by external | ||||
| #ifndef MA_FOR_ESP | ||||
|   #ifdef ESP8266 | ||||
|     #define MA_FOR_ESP         80 //how much mA does the ESP use (Wemos D1 about 80mA) | ||||
|   #else | ||||
|     #define MA_FOR_ESP        120 //how much mA does the ESP use (ESP32 about 120mA) | ||||
|   #endif | ||||
| #endif | ||||
|  | ||||
| class BusManager { | ||||
|   public: | ||||
|     BusManager() {}; | ||||
| @@ -370,7 +380,7 @@ class BusManager { | ||||
|     //utility to get the approx. memory usage of a given BusConfig | ||||
|     static uint32_t memUsage(BusConfig &bc); | ||||
|     static uint32_t memUsage(unsigned channels, unsigned count, unsigned buses = 1); | ||||
|     static uint16_t currentMilliamps() { return _milliAmpsUsed; } | ||||
|     static uint16_t currentMilliamps() { return _milliAmpsUsed + MA_FOR_ESP; } | ||||
|     static uint16_t ablMilliampsMax()  { return _milliAmpsMax; } | ||||
|  | ||||
|     static int add(BusConfig &bc); | ||||
|   | ||||
| @@ -375,6 +375,7 @@ void handleIO() | ||||
|       if (rlyPin>=0) { | ||||
|         pinMode(rlyPin, rlyOpenDrain ? OUTPUT_OPEN_DRAIN : OUTPUT); | ||||
|         digitalWrite(rlyPin, rlyMde); | ||||
|         delay(50); // wait for relay to switch and power to stabilize | ||||
|       } | ||||
|       offMode = false; | ||||
|     } | ||||
|   | ||||
| @@ -16,7 +16,7 @@ function isI(n)     { return n === +n && n === (n|0); } // isInteger | ||||
| function toggle(el) { gId(el).classList.toggle("hide"); gId('No'+el).classList.toggle("hide"); } | ||||
| function tooltip(cont=null) { | ||||
| 	d.querySelectorAll((cont?cont+" ":"")+"[title]").forEach((element)=>{ | ||||
| 		element.addEventListener("mouseover", ()=>{ | ||||
| 		element.addEventListener("pointerover", ()=>{ | ||||
| 			// save title | ||||
| 			element.setAttribute("data-title", element.getAttribute("title")); | ||||
| 			const tooltip = d.createElement("span"); | ||||
| @@ -41,7 +41,7 @@ function tooltip(cont=null) { | ||||
| 			tooltip.classList.add("visible"); | ||||
| 		}); | ||||
|  | ||||
| 		element.addEventListener("mouseout", ()=>{ | ||||
| 		element.addEventListener("pointerout", ()=>{ | ||||
| 			d.querySelectorAll('.tooltip').forEach((tooltip)=>{ | ||||
| 				tooltip.classList.remove("visible"); | ||||
| 				d.body.removeChild(tooltip); | ||||
|   | ||||
| @@ -3114,10 +3114,9 @@ function mergeDeep(target, ...sources) | ||||
| 	return mergeDeep(target, ...sources); | ||||
| } | ||||
|  | ||||
| function tooltip(cont=null) | ||||
| { | ||||
| function tooltip(cont=null) { | ||||
| 	d.querySelectorAll((cont?cont+" ":"")+"[title]").forEach((element)=>{ | ||||
| 		element.addEventListener("mouseover", ()=>{ | ||||
| 		element.addEventListener("pointerover", ()=>{ | ||||
| 			// save title | ||||
| 			element.setAttribute("data-title", element.getAttribute("title")); | ||||
| 			const tooltip = d.createElement("span"); | ||||
| @@ -3142,7 +3141,7 @@ function tooltip(cont=null) | ||||
| 			tooltip.classList.add("visible"); | ||||
| 		}); | ||||
|  | ||||
| 		element.addEventListener("mouseout", ()=>{ | ||||
| 		element.addEventListener("pointerout", ()=>{ | ||||
| 			d.querySelectorAll('.tooltip').forEach((tooltip)=>{ | ||||
| 				tooltip.classList.remove("visible"); | ||||
| 				d.body.removeChild(tooltip); | ||||
|   | ||||
| @@ -379,6 +379,11 @@ | ||||
| 			gId('psu').innerHTML = s; | ||||
| 			gId('psu2').innerHTML = s2; | ||||
| 			gId("json").style.display = d.Sf.IT.value==8 ? "" : "none"; | ||||
|  | ||||
| 			// show/hide FPS warning messages | ||||
| 			gId('fpsNone').style.display = (d.Sf.FR.value == 0) ? 'block':'none'; | ||||
| 			gId('fpsWarn').style.display = (d.Sf.FR.value == 0) || (d.Sf.FR.value >= 80) ? 'block':'none'; | ||||
| 			gId('fpsHigh').style.display = (d.Sf.FR.value >= 80) ? 'block':'none'; | ||||
| 		} | ||||
| 		function lastEnd(i) { | ||||
| 			if (i-- < 1) return 0; | ||||
| @@ -869,7 +874,10 @@ Swap: <select id="xw${s}" name="XW${s}"> | ||||
| 			<option value="2">Linear (never wrap)</option> | ||||
| 			<option value="3">None (not recommended)</option> | ||||
| 		</select><br> | ||||
| 		Target refresh rate: <input type="number" class="s" min="1" max="120" name="FR" required> FPS | ||||
| 		Target refresh rate: <input type="number" class="s" min="0" max="250" name="FR" oninput="UI()" required> FPS | ||||
| 		<div id="fpsNone" class="warn" style="display: none;">⚠ Unlimited FPS Mode  is experimental ⚠<br></div> | ||||
| 		<div id="fpsHigh" class="warn" style="display: none;">⚠ High FPS Mode is experimental.<br></div> | ||||
| 		<div id="fpsWarn" class="warn" style="display: none;">Please <a class="lnk" href="sec#backup">backup</a> WLED configuration and presets first!<br></div> | ||||
| 		<hr class="sml"> | ||||
| 		<div id="cfg">Config template: <input type="file" name="data2" accept=".json"><button type="button" class="sml" onclick="loadCfg(d.Sf.data2)">Apply</button><br></div> | ||||
| 		<hr> | ||||
|   | ||||
| @@ -57,7 +57,7 @@ | ||||
| 		<h3>Software Update</h3> | ||||
| 		<button type="button" onclick="U()">Manual OTA Update</button><br> | ||||
| 		Enable ArduinoOTA: <input type="checkbox" name="AO"> | ||||
| 		<hr> | ||||
| 		<hr id="backup"> | ||||
| 		<h3>Backup & Restore</h3> | ||||
| 		<div class="warn">⚠ Restoring presets/configuration will OVERWRITE your current presets/configuration.<br> | ||||
| 		Incorrect upload or configuration may require a factory reset or re-flashing of your ESP.<br> | ||||
|   | ||||
| @@ -230,7 +230,8 @@ void deletePreset(byte index); | ||||
| bool getPresetName(byte index, String& name); | ||||
|  | ||||
| //remote.cpp | ||||
| void handleRemote(uint8_t *data, size_t len); | ||||
| void handleWiZdata(uint8_t *incomingData, size_t len); | ||||
| void handleRemote(); | ||||
|  | ||||
| //set.cpp | ||||
| bool isAsterisksOnly(const char* str, byte maxLen); | ||||
| @@ -372,7 +373,7 @@ void userLoop(); | ||||
| //util.cpp | ||||
| int getNumVal(const String* req, uint16_t pos); | ||||
| void parseNumber(const char* str, byte* val, byte minv=0, byte maxv=255); | ||||
| bool getVal(JsonVariant elem, byte* val, byte minv=0, byte maxv=255); | ||||
| bool getVal(JsonVariant elem, byte* val, byte minv=0, byte maxv=255); // getVal supports inc/decrementing and random ("X~Y(r|~[w][-][Z])" form) | ||||
| bool getBoolVal(JsonVariant elem, bool dflt); | ||||
| bool updateVal(const char* req, const char* key, byte* val, byte minv=0, byte maxv=255); | ||||
| size_t printSetFormCheckbox(Print& settingsScript, const char* key, int val); | ||||
|   | ||||
| @@ -223,30 +223,17 @@ bool deserializeSegment(JsonObject elem, byte it, byte presetId) | ||||
|   #endif | ||||
|  | ||||
|   byte fx = seg.mode; | ||||
|   byte last = strip.getModeCount(); | ||||
|   // partial fix for #3605 | ||||
|   if (!elem["fx"].isNull() && elem["fx"].is<const char*>()) { | ||||
|     const char *tmp = elem["fx"].as<const char *>(); | ||||
|     if (strlen(tmp) > 3 && (strchr(tmp,'r') || strchr(tmp,'~') != strrchr(tmp,'~'))) last = 0; // we have "X~Y(r|[w]~[-])" form | ||||
|   } | ||||
|   // end fix | ||||
|   if (getVal(elem["fx"], &fx, 0, last)) { //load effect ('r' random, '~' inc/dec, 0-255 exact value, 5~10r pick random between 5 & 10) | ||||
|   if (getVal(elem["fx"], &fx, 0, strip.getModeCount())) { | ||||
|     if (!presetId && currentPlaylist>=0) unloadPlaylist(); | ||||
|     if (fx != seg.mode) seg.setMode(fx, elem[F("fxdef")]); | ||||
|   } | ||||
|  | ||||
|   //getVal also supports inc/decrementing and random | ||||
|   getVal(elem["sx"], &seg.speed); | ||||
|   getVal(elem["ix"], &seg.intensity); | ||||
|  | ||||
|   uint8_t pal = seg.palette; | ||||
|   last = strip.getPaletteCount(); | ||||
|   if (!elem["pal"].isNull() && elem["pal"].is<const char*>()) { | ||||
|     const char *tmp = elem["pal"].as<const char *>(); | ||||
|     if (strlen(tmp) > 3 && (strchr(tmp,'r') || strchr(tmp,'~') != strrchr(tmp,'~'))) last = 0; // we have "X~Y(r|[w]~[-])" form | ||||
|   } | ||||
|   if (seg.getLightCapabilities() & 1) {  // ignore palette for White and On/Off segments | ||||
|     if (getVal(elem["pal"], &pal, 0, last)) seg.setPalette(pal); | ||||
|     if (getVal(elem["pal"], &pal, 0, strip.getPaletteCount())) seg.setPalette(pal); | ||||
|   } | ||||
|  | ||||
|   getVal(elem["c1"], &seg.custom1); | ||||
| @@ -467,7 +454,7 @@ bool deserializeState(JsonObject root, byte callMode, byte presetId) | ||||
|     DEBUG_PRINTF_P(PSTR("Preset direct: %d\n"), currentPreset); | ||||
|   } else if (!root["ps"].isNull()) { | ||||
|     // we have "ps" call (i.e. from button or external API call) or "pd" that includes "ps" (i.e. from UI call) | ||||
|     if (root["win"].isNull() && getVal(root["ps"], &presetCycCurr, 0, 0) && presetCycCurr > 0 && presetCycCurr < 251 && presetCycCurr != currentPreset) { | ||||
|     if (root["win"].isNull() && getVal(root["ps"], &presetCycCurr, 1, 250) && presetCycCurr > 0 && presetCycCurr < 251 && presetCycCurr != currentPreset) { | ||||
|       DEBUG_PRINTF_P(PSTR("Preset select: %d\n"), presetCycCurr); | ||||
|       // b) preset ID only or preset that does not change state (use embedded cycling limits if they exist in getVal()) | ||||
|       applyPreset(presetCycCurr, callMode); // async load from file system (only preset ID was specified) | ||||
|   | ||||
| @@ -73,8 +73,8 @@ byte scaledBri(byte in) | ||||
|  | ||||
| //applies global brightness | ||||
| void applyBri() { | ||||
|   if (!realtimeMode || !arlsForceMaxBri) | ||||
|   { | ||||
|   if (!(realtimeMode && arlsForceMaxBri)) { | ||||
|     //DEBUG_PRINTF_P(PSTR("Applying strip brightness: %d (%d,%d)\n"), (int)briT, (int)bri, (int)briOld); | ||||
|     strip.setBrightness(scaledBri(briT)); | ||||
|   } | ||||
| } | ||||
| @@ -85,6 +85,7 @@ void applyFinalBri() { | ||||
|   briOld = bri; | ||||
|   briT = bri; | ||||
|   applyBri(); | ||||
|   strip.trigger(); | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -146,7 +147,6 @@ void stateUpdated(byte callMode) { | ||||
|     transitionStartTime = millis(); | ||||
|   } else { | ||||
|     applyFinalBri(); | ||||
|     strip.trigger(); | ||||
|   } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -207,6 +207,7 @@ void WiFiEvent(WiFiEvent_t event) | ||||
|       break; | ||||
| #endif | ||||
|     default: | ||||
|       DEBUG_PRINTF_P(PSTR("Network event: %d\n"), (int)event); | ||||
|       break; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -214,8 +214,20 @@ bool PinManager::isPinOk(byte gpio, bool output) | ||||
|     // JTAG: GPIO39-42 are usually used for inline debugging | ||||
|     // GPIO46 is input only and pulled down | ||||
|   #else | ||||
|     if (gpio > 5 && gpio < 12) return false;      //SPI flash pins | ||||
|     if (strncmp_P(PSTR("ESP32-PICO"), ESP.getChipModel(), 10) == 0 && (gpio == 16 || gpio == 17)) return false; // PICO-D4: gpio16+17 are in use for onboard SPI FLASH | ||||
|  | ||||
|     if ((strncmp_P(PSTR("ESP32-U4WDH"), ESP.getChipModel(), 11) == 0) ||    // this is the correct identifier, but.... | ||||
|         (strncmp_P(PSTR("ESP32-PICO-D2"), ESP.getChipModel(), 13) == 0)) {  // https://github.com/espressif/arduino-esp32/issues/10683 | ||||
|       // this chip has 4 MB of internal Flash and different packaging, so available pins are different! | ||||
|       if (((gpio > 5) && (gpio < 9)) || (gpio == 11)) | ||||
|         return false; | ||||
|     } else { | ||||
|       // for classic ESP32 (non-mini) modules, these are the SPI flash pins | ||||
|       if (gpio > 5 && gpio < 12) return false;      //SPI flash pins | ||||
|     } | ||||
|  | ||||
|     if (((strncmp_P(PSTR("ESP32-PICO"), ESP.getChipModel(), 10) == 0) || | ||||
|          (strncmp_P(PSTR("ESP32-U4WDH"), ESP.getChipModel(), 11) == 0)) | ||||
|         && (gpio == 16 || gpio == 17)) return false; // PICO-D4/U4WDH: gpio16+17 are in use for onboard SPI FLASH | ||||
|     if (gpio == 16 || gpio == 17) return !psramFound(); //PSRAM pins on ESP32 (these are IO) | ||||
|   #endif | ||||
|     if (output) return digitalPinCanOutput(gpio); | ||||
|   | ||||
| @@ -164,6 +164,11 @@ void handlePresets() | ||||
|  | ||||
|   DEBUG_PRINTF_P(PSTR("Applying preset: %u\n"), (unsigned)tmpPreset); | ||||
|  | ||||
|   #if defined(ARDUINO_ARCH_ESP32S3) || defined(ARDUINO_ARCH_ESP32S2) || defined(ARDUINO_ARCH_ESP32C3) | ||||
|   unsigned long start = millis(); | ||||
|   while (strip.isUpdating() && millis() - start < FRAMETIME_FIXED) yield(); // wait for strip to finish updating, accessing FS during sendout causes glitches | ||||
|   #endif | ||||
|  | ||||
|   #ifdef ARDUINO_ARCH_ESP32 | ||||
|   if (tmpPreset==255 && tmpRAMbuffer!=nullptr) { | ||||
|     deserializeJson(*pDoc,tmpRAMbuffer); | ||||
|   | ||||
| @@ -1,6 +1,8 @@ | ||||
| #include "wled.h" | ||||
| #ifndef WLED_DISABLE_ESPNOW | ||||
|  | ||||
| #define ESPNOW_BUSWAIT_TIMEOUT 24 // one frame timeout to wait for bus to finish updating | ||||
|  | ||||
| #define NIGHT_MODE_DEACTIVATED     -1 | ||||
| #define NIGHT_MODE_BRIGHTNESS      5 | ||||
|  | ||||
| @@ -38,6 +40,7 @@ typedef struct WizMoteMessageStructure { | ||||
|  | ||||
| static uint32_t last_seq = UINT32_MAX; | ||||
| static int brightnessBeforeNightMode = NIGHT_MODE_DEACTIVATED; | ||||
| static int16_t ESPNowButton = -1; // set in callback if new button value is received | ||||
|  | ||||
| // Pulled from the IR Remote logic but reduced to 10 steps with a constant of 3 | ||||
| static const byte brightnessSteps[] = { | ||||
| @@ -121,6 +124,9 @@ static bool remoteJson(int button) | ||||
|  | ||||
|   sprintf_P(objKey, PSTR("\"%d\":"), button); | ||||
|  | ||||
|   unsigned long start = millis(); | ||||
|   while (strip.isUpdating() && millis()-start < ESPNOW_BUSWAIT_TIMEOUT) yield(); // wait for strip to finish updating, accessing FS during sendout causes glitches | ||||
|  | ||||
|   // attempt to read command from remote.json | ||||
|   readObjectFromFile(PSTR("/remote.json"), objKey, pDoc); | ||||
|   JsonObject fdo = pDoc->as<JsonObject>(); | ||||
| @@ -176,7 +182,7 @@ static bool remoteJson(int button) | ||||
| } | ||||
|  | ||||
| // Callback function that will be executed when data is received | ||||
| void handleRemote(uint8_t *incomingData, size_t len) { | ||||
| void handleWiZdata(uint8_t *incomingData, size_t len) { | ||||
|   message_structure_t *incoming = reinterpret_cast<message_structure_t *>(incomingData); | ||||
|  | ||||
|   if (strcmp(last_signal_src, linked_remote) != 0) { | ||||
| @@ -202,8 +208,15 @@ void handleRemote(uint8_t *incomingData, size_t len) { | ||||
|   DEBUG_PRINT(F("] button: ")); | ||||
|   DEBUG_PRINTLN(incoming->button); | ||||
|  | ||||
|   if (!remoteJson(incoming->button)) | ||||
|     switch (incoming->button) { | ||||
|   ESPNowButton = incoming->button; // save state, do not process in callback (can cause glitches) | ||||
|   last_seq = cur_seq; | ||||
| } | ||||
|  | ||||
| // process ESPNow button data (acesses FS, should not be called while update to avoid glitches) | ||||
| void handleRemote() { | ||||
|   if(ESPNowButton >= 0) { | ||||
|   if (!remoteJson(ESPNowButton)) | ||||
|     switch (ESPNowButton) { | ||||
|       case WIZMOTE_BUTTON_ON             : setOn();                                         break; | ||||
|       case WIZMOTE_BUTTON_OFF            : setOff();                                        break; | ||||
|       case WIZMOTE_BUTTON_ONE            : presetWithFallback(1, FX_MODE_STATIC,        0); break; | ||||
| @@ -219,9 +232,10 @@ void handleRemote(uint8_t *incomingData, size_t len) { | ||||
|       case WIZ_SMART_BUTTON_BRIGHT_DOWN  : brightnessDown();                                break; | ||||
|       default: break; | ||||
|     } | ||||
|   last_seq = cur_seq; | ||||
|   } | ||||
|   ESPNowButton = -1; | ||||
| } | ||||
|  | ||||
| #else | ||||
| void handleRemote(uint8_t *incomingData, size_t len) {} | ||||
| void handleRemote() {} | ||||
| #endif | ||||
|   | ||||
| @@ -979,7 +979,7 @@ void espNowReceiveCB(uint8_t* address, uint8_t* data, uint8_t len, signed int rs | ||||
|  | ||||
|   // handle WiZ Mote data | ||||
|   if (data[0] == 0x91 || data[0] == 0x81 || data[0] == 0x80) { | ||||
|     handleRemote(data, len); | ||||
|     handleWiZdata(data, len); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -52,7 +52,7 @@ void parseNumber(const char* str, byte* val, byte minv, byte maxv) | ||||
|   *val = atoi(str); | ||||
| } | ||||
|  | ||||
|  | ||||
| //getVal supports inc/decrementing and random ("X~Y(r|~[w][-][Z])" form) | ||||
| bool getVal(JsonVariant elem, byte* val, byte vmin, byte vmax) { | ||||
|   if (elem.is<int>()) { | ||||
| 		if (elem < 0) return false; //ignore e.g. {"ps":-1} | ||||
| @@ -60,8 +60,12 @@ bool getVal(JsonVariant elem, byte* val, byte vmin, byte vmax) { | ||||
|     return true; | ||||
|   } else if (elem.is<const char*>()) { | ||||
|     const char* str = elem; | ||||
|     size_t len = strnlen(str, 12); | ||||
|     if (len == 0 || len > 10) return false; | ||||
|     size_t len = strnlen(str, 14); | ||||
|     if (len == 0 || len > 12) return false; | ||||
|     // fix for #3605 & #4346 | ||||
|     // ignore vmin and vmax and use as specified in API | ||||
|     if (len > 3 && (strchr(str,'r') || strchr(str,'~') != strrchr(str,'~'))) vmax = vmin = 0; // we have "X~Y(r|~[w][-][Z])" form | ||||
|     // end fix | ||||
|     parseNumber(str, val, vmin, vmax); | ||||
|     return true; | ||||
|   } | ||||
|   | ||||
| @@ -84,6 +84,9 @@ void WLED::loop() | ||||
|   #ifndef WLED_DISABLE_INFRARED | ||||
|   handleIR(); | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_ESPNOW | ||||
|   handleRemote(); | ||||
|   #endif | ||||
|   #ifndef WLED_DISABLE_ALEXA | ||||
|   handleAlexa(); | ||||
|   #endif | ||||
| @@ -478,10 +481,7 @@ void WLED::setup() | ||||
|   if (strcmp(multiWiFi[0].clientSSID, DEFAULT_CLIENT_SSID) == 0) | ||||
|     showWelcomePage = true; | ||||
|   WiFi.persistent(false); | ||||
|   #ifdef WLED_USE_ETHERNET | ||||
|   WiFi.onEvent(WiFiEvent); | ||||
|   #endif | ||||
|  | ||||
|   WiFi.mode(WIFI_STA); // enable scanning | ||||
|   findWiFi(true);      // start scanning for available WiFi-s | ||||
|  | ||||
| @@ -781,7 +781,7 @@ int8_t WLED::findWiFi(bool doScan) { | ||||
|  | ||||
| void WLED::initConnection() | ||||
| { | ||||
|   DEBUG_PRINTLN(F("initConnection() called.")); | ||||
|   DEBUG_PRINTF_P(PSTR("initConnection() called @ %lus.\n"), millis()/1000); | ||||
|  | ||||
|   #ifdef WLED_ENABLE_WEBSOCKETS | ||||
|   ws.onEvent(wsEvent); | ||||
| @@ -796,6 +796,7 @@ void WLED::initConnection() | ||||
| #endif | ||||
|  | ||||
|   WiFi.disconnect(true); // close old connections | ||||
|   delay(5);              // wait for hardware to be ready | ||||
| #ifdef ESP8266 | ||||
|   WiFi.setPhyMode(force802_3g ? WIFI_PHY_MODE_11G : WIFI_PHY_MODE_11N); | ||||
| #endif | ||||
| @@ -825,9 +826,7 @@ void WLED::initConnection() | ||||
|   if (WLED_WIFI_CONFIGURED) { | ||||
|     showWelcomePage = false; | ||||
|      | ||||
|     DEBUG_PRINT(F("Connecting to ")); | ||||
|     DEBUG_PRINT(multiWiFi[selectedWiFi].clientSSID); | ||||
|     DEBUG_PRINTLN(F("...")); | ||||
|     DEBUG_PRINTF_P(PSTR("Connecting to %s...\n"), multiWiFi[selectedWiFi].clientSSID); | ||||
|  | ||||
|     // convert the "serverDescription" into a valid DNS hostname (alphanumeric) | ||||
|     char hostname[25]; | ||||
| @@ -926,7 +925,8 @@ void WLED::handleConnection() | ||||
| { | ||||
|   static bool scanDone = true; | ||||
|   static byte stacO = 0; | ||||
|   unsigned long now = millis(); | ||||
|   const unsigned long now = millis(); | ||||
|   const unsigned long nowS = now/1000; | ||||
|   const bool wifiConfigured = WLED_WIFI_CONFIGURED; | ||||
|  | ||||
|   // ignore connection handling if WiFi is configured and scan still running | ||||
| @@ -935,7 +935,7 @@ void WLED::handleConnection() | ||||
|     return; | ||||
|  | ||||
|   if (lastReconnectAttempt == 0 || forceReconnect) { | ||||
|     DEBUG_PRINTLN(F("Initial connect or forced reconnect.")); | ||||
|     DEBUG_PRINTF_P(PSTR("Initial connect or forced reconnect (@ %lus).\n"), nowS); | ||||
|     selectedWiFi = findWiFi(); // find strongest WiFi | ||||
|     initConnection(); | ||||
|     interfacesInited = false; | ||||
| @@ -955,8 +955,7 @@ void WLED::handleConnection() | ||||
| #endif | ||||
|     if (stac != stacO) { | ||||
|       stacO = stac; | ||||
|       DEBUG_PRINT(F("Connected AP clients: ")); | ||||
|       DEBUG_PRINTLN(stac); | ||||
|       DEBUG_PRINTF_P(PSTR("Connected AP clients: %d\n"), (int)stac); | ||||
|       if (!WLED_CONNECTED && wifiConfigured) {        // trying to connect, but not connected | ||||
|         if (stac) | ||||
|           WiFi.disconnect();        // disable search so that AP can work | ||||
| @@ -979,6 +978,7 @@ void WLED::handleConnection() | ||||
|       initConnection(); | ||||
|       interfacesInited = false; | ||||
|       scanDone = true; | ||||
|       return; | ||||
|     } | ||||
|     //send improv failed 6 seconds after second init attempt (24 sec. after provisioning) | ||||
|     if (improvActive > 2 && now - lastReconnectAttempt > 6000) { | ||||
| @@ -987,13 +987,13 @@ void WLED::handleConnection() | ||||
|     } | ||||
|     if (now - lastReconnectAttempt > ((stac) ? 300000 : 18000) && wifiConfigured) { | ||||
|       if (improvActive == 2) improvActive = 3; | ||||
|       DEBUG_PRINTLN(F("Last reconnect too old.")); | ||||
|       DEBUG_PRINTF_P(PSTR("Last reconnect (%lus) too old (@ %lus).\n"), lastReconnectAttempt/1000, nowS); | ||||
|       if (++selectedWiFi >= multiWiFi.size()) selectedWiFi = 0; // we couldn't connect, try with another network from the list | ||||
|       initConnection(); | ||||
|     } | ||||
|     if (!apActive && now - lastReconnectAttempt > 12000 && (!wasConnected || apBehavior == AP_BEHAVIOR_NO_CONN)) { | ||||
|       if (!(apBehavior == AP_BEHAVIOR_TEMPORARY && now > WLED_AP_TIMEOUT)) { | ||||
|         DEBUG_PRINTLN(F("Not connected AP.")); | ||||
|         DEBUG_PRINTF_P(PSTR("Not connected AP (@ %lus).\n"), nowS); | ||||
|         initAP();  // start AP only within first 5min | ||||
|       } | ||||
|     } | ||||
| @@ -1003,7 +1003,7 @@ void WLED::handleConnection() | ||||
|         dnsServer.stop(); | ||||
|         WiFi.softAPdisconnect(true); | ||||
|         apActive = false; | ||||
|         DEBUG_PRINTLN(F("Temporary AP disabled.")); | ||||
|         DEBUG_PRINTF_P(PSTR("Temporary AP disabled (@ %lus).\n"), nowS); | ||||
|       } | ||||
|     } | ||||
|   } else if (!interfacesInited) { //newly connected | ||||
|   | ||||
| @@ -3,12 +3,11 @@ | ||||
| /* | ||||
|    Main sketch, global variable declarations | ||||
|    @title WLED project sketch | ||||
|    @version 0.15.0-b7 | ||||
|    @author Christian Schwinne | ||||
|  */ | ||||
|  | ||||
| // version code in format yymmddb (b = daily build) | ||||
| #define VERSION 2410270 | ||||
| #define VERSION 2501160 | ||||
|  | ||||
| //uncomment this if you have a "my_config.h" file you'd like to use | ||||
| //#define WLED_USE_MY_CONFIG | ||||
|   | ||||
		Reference in New Issue
	
	Block a user