Compare commits
	
		
			479 Commits
		
	
	
		
			v0.14.4
			...
			v0.15.0-b1
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 5d3f22e06a | ||
|   | c77f6c5f7b | ||
|   | 6d03c3a54c | ||
|   | 92dc63e5c9 | ||
|   | 505d319e01 | ||
|   | 62f845a94e | ||
|   | b849ea8eaa | ||
|   | f1635fa302 | ||
|   | 6110b72b87 | ||
|   | 6de617ecd5 | ||
|   | 7c8df97968 | ||
|   | 52a1b0453c | ||
|   | 7b366d49d2 | ||
|   | 88b30e7e28 | ||
|   | 05c0febd04 | ||
|   | a4384bd340 | ||
|   | a2368a75f7 | ||
|   | 68e5e74882 | ||
|   | 51b9e81c3d | ||
|   | 7723011594 | ||
|   | 215ba35fe1 | ||
|   | 7e17011ebc | ||
|   | cb5eb9edc7 | ||
|   | da5c12f466 | ||
|   | f2ad10bbce | ||
|   | 99a7bece2e | ||
|   | b305fd8865 | ||
|   | c0beb621e2 | ||
|   | da6d64e581 | ||
|   | a6d58ee360 | ||
|   | 2a480ab7db | ||
|   | d937d473f9 | ||
|   | 66f14c6343 | ||
|   | f25fadafd8 | ||
|   | 200d11ca99 | ||
|   | f8c48ef60a | ||
|   | ec4afb2cbc | ||
|   | a2a632c415 | ||
|   | 962e64106c | ||
|   | c40b6088ee | ||
|   | 0d1a254ca8 | ||
|   | fdbcb61a12 | ||
|   | 6435cb1466 | ||
|   | 4739cfab9a | ||
|   | 00a73e9cc8 | ||
|   | 0ff5dec004 | ||
|   | 41129cf379 | ||
|   | a28d2c869f | ||
|   | 6f1fff44fa | ||
|   | 108978d1a5 | ||
|   | 66f4671ec0 | ||
|   | 21173dc907 | ||
|   | 989bdfb0d5 | ||
|   | 5761dce957 | ||
|   | 95e2e574b8 | ||
|   | c789d80ce5 | ||
|   | 2d30535b69 | ||
|   | aa24e5d284 | ||
|   | e7dc2048ad | ||
|   | af6094bb06 | ||
|   | 0ab139cb24 | ||
|   | 500a7301f2 | ||
|   | c92cef1780 | ||
|   | 49ceac1ac7 | ||
|   | 276a93605d | ||
|   | 0150c3fe23 | ||
|   | 34674501c3 | ||
|   | f6206d4c30 | ||
|   | 6dcd9596a2 | ||
|   | e165838e54 | ||
|   | eae1707e97 | ||
|   | ed2950f73b | ||
|   | 7eae8f68d8 | ||
|   | 0ab2d18b52 | ||
|   | 8ab621bc91 | ||
|   | 41e51bbeb5 | ||
|   | 66d9e8c038 | ||
|   | f5e47b2b74 | ||
|   | 678a823e8b | ||
|   | 246746a82e | ||
|   | 00038453e1 | ||
|   | 9ce485eade | ||
|   | 8a6ff5a42b | ||
|   | 35716a7e4c | ||
|   | e0f89beebb | ||
|   | b8bf2a707c | ||
|   | 7d616be8f4 | ||
|   | 0a7e9f9f8f | ||
|   | 226e188903 | ||
|   | 590d454119 | ||
|   | 06d5bd799f | ||
|   | 7fe6541b7c | ||
|   | 0f30fa5fb9 | ||
|   | 2c362315ad | ||
|   | 7dd79edc52 | ||
|   | b3c21feba3 | ||
|   | fe54fadbf8 | ||
|   | caa0722d49 | ||
|   | fee32622f1 | ||
|   | d1910e4274 | ||
|   | 45eb11d7f4 | ||
|   | 874179fa75 | ||
|   | 8b16cb4bae | ||
|   | 777ed55c21 | ||
|   | fcafd8b05e | ||
|   | 5f1cadfea4 | ||
|   | ca05aa84ff | ||
|   | d10d7dc298 | ||
|   | 063af1dbc7 | ||
|   | ef6fe43251 | ||
|   | 2659055c31 | ||
|   | f5ed757780 | ||
|   | e114b842ba | ||
|   | 12e2116acb | ||
|   | bccc97d25f | ||
|   | a9bcc75733 | ||
|   | 7b87475af8 | ||
|   | 2ac218886a | ||
|   | 97503897f0 | ||
|   | 7e06e32cb6 | ||
|   | 15b813c6af | ||
|   | a71c9107bd | ||
|   | 4c2110189b | ||
|   | df750c2a71 | ||
|   | 3eb412b750 | ||
|   | 5c7b7e4182 | ||
|   | 8817d41275 | ||
|   | 04ef3f9496 | ||
|   | fbe26e13ae | ||
|   | 4408dffa87 | ||
|   | 546192865c | ||
|   | 21a9372320 | ||
|   | f4a2eec984 | ||
|   | 94b9455c9c | ||
|   | c41c71c8c1 | ||
|   | dde647c570 | ||
|   | 445a89ba6e | ||
|   | 07495f6621 | ||
|   | 5952edc550 | ||
|   | b66a038a2f | ||
|   | df1c8a64e5 | ||
|   | 3e2aebcd10 | ||
|   | 0d279cb4dd | ||
|   | 3942a76d15 | ||
|   | 1bebf3d3d6 | ||
|   | d5f54c240a | ||
|   | 8868ecd11a | ||
|   | 8a31c57bfa | ||
|   | 7e2cce4a8d | ||
|   | bfb217c203 | ||
|   | 593e55af95 | ||
|   | 29af62f956 | ||
|   | 2b022e1871 | ||
|   | 783a4d3996 | ||
|   | ddfe925f8c | ||
|   | 1735bcab21 | ||
|   | ff6cacf766 | ||
|   | 2a72eae77f | ||
|   | 39fb7fdfe3 | ||
|   | e0facf35c8 | ||
|   | 73d48c8ac4 | ||
|   | 5080059143 | ||
|   | 90d5b77d74 | ||
|   | e0f987328b | ||
|   | 04c3a925cc | ||
|   | 219b29991b | ||
|   | 5120045459 | ||
|   | 3bca4a73f4 | ||
|   | b4857ab036 | ||
|   | f3ae041691 | ||
|   | a84627a465 | ||
|   | 78aeca6399 | ||
|   | 5dd8f0a0b7 | ||
|   | 9378fc7276 | ||
|   | 3a0d8069b6 | ||
|   | bab42014a4 | ||
|   | 3be368c333 | ||
|   | 58eb151ba1 | ||
|   | 21d21ea7f0 | ||
|   | 45dc5e236d | ||
|   | 0a66fe196a | ||
|   | 3f21b4aa7d | ||
|   | b97186c5bf | ||
|   | 874b24fb32 | ||
|   | 97889b5e55 | ||
|   | 4b134194cf | ||
|   | 3e069376a7 | ||
|   | 5bdb2c9e86 | ||
|   | 71b0c5f805 | ||
|   | a1f62e7a1f | ||
|   | 186c40801a | ||
|   | bf867e84d6 | ||
|   | db3770c3e5 | ||
|   | 46337ca554 | ||
|   | 4f0f288000 | ||
|   | 8060bae1b8 | ||
|   | ea32c97b82 | ||
|   | 99efbd30f1 | ||
|   | 447324b59f | ||
|   | e580e815f0 | ||
|   | 5fc00840f2 | ||
|   | ff59dcd66e | ||
|   | 0a1d82de2a | ||
|   | 26837645ac | ||
|   | c318699d6a | ||
|   | e5caab635a | ||
|   | d5f96295ff | ||
|   | 4a6c78aaf3 | ||
|   | c4d214f5c0 | ||
|   | 1ed7e6cb1c | ||
|   | 6b7f80f24a | ||
|   | 220217561a | ||
|   | 9e54de8a8a | ||
|   | fd2809e367 | ||
|   | 18874c2069 | ||
|   | 115eb0ddef | ||
|   | 9940258725 | ||
|   | 74e14a4fc3 | ||
|   | 90ea386c62 | ||
|   | a5b132dfd7 | ||
|   | 157debf8e4 | ||
|   | 74a5528f17 | ||
|   | 21de9e938a | ||
|   | 9c4d3cd6e2 | ||
|   | 0616d184e0 | ||
|   | 6f8b3a6242 | ||
|   | 9a006dc84a | ||
|   | d2dbaf52a1 | ||
|   | 0a9145cd3c | ||
|   | e778b02c0c | ||
|   | 43f5e4360d | ||
|   | 21f8d7a967 | ||
|   | 801d92bbba | ||
|   | 77116172e4 | ||
|   | 9d3c0f4ff0 | ||
|   | 6382d2b730 | ||
|   | 6ad4493a91 | ||
|   | f070dc5527 | ||
|   | fc6e7c81d3 | ||
|   | 59a725c52c | ||
|   | fd6ce57003 | ||
|   | bd20c83919 | ||
|   | 48f8a45031 | ||
|   | 680cec5abb | ||
|   | ecdc3be52e | ||
|   | 301bdf2186 | ||
|   | 1c1b67e000 | ||
|   | bc85520f5e | ||
|   | d3be7a3edf | ||
|   | 6cd0da821a | ||
|   | 1333c41811 | ||
|   | 55eceb4080 | ||
|   | 6332ee6edb | ||
|   | 901d56f898 | ||
|   | b3a768a04b | ||
|   | f414af4d82 | ||
|   | 7971f3cbd8 | ||
|   | 08d9f7d967 | ||
|   | 776502e6db | ||
|   | 5ed02d2f0d | ||
|   | cf3f6ede72 | ||
|   | b76a90304d | ||
|   | 7af5b24b03 | ||
|   | 1f81fb9284 | ||
|   | 4e105492ca | ||
|   | abf7dd5fa8 | ||
|   | 6c877b6d44 | ||
|   | 86d2998dc0 | ||
|   | 201daf8ff3 | ||
|   | 1140f5f6cb | ||
|   | 0a91d60677 | ||
|   | 5cd8f5681e | ||
|   | b2babd9502 | ||
|   | 42dba3170d | ||
|   | 9e57ef074d | ||
|   | 9e9b1c3c98 | ||
|   | 13d5deddec | ||
|   | 8522760c70 | ||
|   | 5bc2282ac5 | ||
|   | 2944b2a8f6 | ||
|   | 264480b9bc | ||
|   | 7d52cc46dc | ||
|   | 37c9fd278c | ||
|   | 3d6fe0a565 | ||
|   | 076449a33d | ||
|   | 1b5cc2abf1 | ||
|   | b176225910 | ||
|   | e99dfbae0d | ||
|   | b024d93fa0 | ||
|   | 37b807210e | ||
|   | 15e26ba929 | ||
|   | db1795981f | ||
|   | 2a40baf509 | ||
|   | 2dbf72e452 | ||
|   | 67ae716c60 | ||
|   | 8dfb130d19 | ||
|   | 2a42fea32a | ||
|   | f69edefddf | ||
|   | 6f3b5fc559 | ||
|   | 7275a34e1f | ||
|   | 687e2ecaaa | ||
|   | 64312aedae | ||
|   | 391876e557 | ||
|   | 830f806dee | ||
|   | 21c2316f6b | ||
|   | 930a1ae450 | ||
|   | aec331b5a6 | ||
|   | 1db6c281b1 | ||
|   | 5b70ca81a7 | ||
|   | d00df785a5 | ||
|   | d0c6e7a285 | ||
|   | 058995a0b6 | ||
|   | fd3be887b4 | ||
|   | 23d44cb28b | ||
|   | de4b352495 | ||
|   | d635117194 | ||
|   | c0412d368e | ||
|   | 7ec30f0796 | ||
|   | 9815a55221 | ||
|   | 15d23870c1 | ||
|   | 2abbf7e762 | ||
|   | 1578aa7cb6 | ||
|   | f2230751e7 | ||
|   | 8fc0dda26d | ||
|   | 0ab71badfa | ||
|   | 0f24d924f9 | ||
|   | 518a0de95f | ||
|   | fa5648c0c3 | ||
|   | bd620a764d | ||
|   | cff42b5d27 | ||
|   | 25eef2a40e | ||
|   | db276d5f1f | ||
|   | ce0d906c35 | ||
|   | 520826abc0 | ||
|   | 5e1d6014d5 | ||
|   | 32af17317c | ||
|   | 3e7cc8a0a0 | ||
|   | f717ce52fa | ||
|   | d880f3c628 | ||
|   | 0a97717b35 | ||
|   | 029403ea10 | ||
|   | e4ec65622f | ||
|   | 3ee3b97255 | ||
|   | a3dd6ce891 | ||
|   | 0c97d7701e | ||
|   | b852b6f55c | ||
|   | d3b7b857a4 | ||
|   | 63ad3f9290 | ||
|   | 40cc30e054 | ||
|   | 23374e98ae | ||
|   | 40f544fffc | ||
|   | 4fc69c9a2c | ||
|   | 2e5a485804 | ||
|   | 2d10b5be90 | ||
|   | 8c69b85280 | ||
|   | 9d6d1e4ff8 | ||
|   | e71c72b4a2 | ||
|   | bf4b29b3e3 | ||
|   | 8e0809ceba | ||
|   | 230af48899 | ||
|   | 9dc459192e | ||
|   | 8de2301c6b | ||
|   | 25c5e823d9 | ||
|   | 528b8b1271 | ||
|   | b88344a6c0 | ||
|   | 9e135cc81d | ||
|   | 72ef288357 | ||
|   | 426ac29ca5 | ||
|   | ceb330e3e4 | ||
|   | 7ea8f741f2 | ||
|   | 443c9fba03 | ||
|   | 0227f79bd7 | ||
|   | 2812eaa13d | ||
|   | cb293fcdb3 | ||
|   | 706428fe98 | ||
|   | 698f5f951e | ||
|   | 155c12f657 | ||
|   | a36c731d15 | ||
|   | 6a793536dd | ||
|   | 27532a4237 | ||
|   | 467f69f50e | ||
|   | b5a8f3156c | ||
|   | de245c08ce | ||
|   | d9a96fbb29 | ||
|   | 41bc799c3f | ||
|   | 8f7f9ec367 | ||
|   | 21de8f2284 | ||
|   | 15797a89e7 | ||
|   | 71cec39bc4 | ||
|   | 882542318c | ||
|   | 1049d65621 | ||
|   | 552ae836e9 | ||
|   | 1852c0a88c | ||
|   | d0d399bdfc | ||
|   | 3975fe0f02 | ||
|   | 0739cfc240 | ||
|   | f31bf4f7c9 | ||
|   | f413671915 | ||
|   | ae92a5c25e | ||
|   | fabd3b8421 | ||
|   | 94d754e63a | ||
|   | f869542a47 | ||
|   | 2b616b688d | ||
|   | 0925ea6f9c | ||
|   | 87b5565679 | ||
|   | adcecf969a | ||
|   | f3831665d4 | ||
|   | e1075a3bbf | ||
|   | 3b6f499fc5 | ||
|   | 2c1fbe103b | ||
|   | e42ca2dbf2 | ||
|   | 1495b756d3 | ||
|   | 7f78d00b97 | ||
|   | bfd58b3cdf | ||
|   | 31a4e38bc0 | ||
|   | 9c6dda9bd2 | ||
|   | 0232117de5 | ||
|   | 367a4e17a2 | ||
|   | eba0d7c1e0 | ||
|   | 341a4a0d60 | ||
|   | d116424241 | ||
|   | db881ee011 | ||
|   | 2512bebc62 | ||
|   | 309dbd7585 | ||
|   | a1f28caa8e | ||
|   | 4204393337 | ||
|   | 789e34702b | ||
|   | 47bc1bf88d | ||
|   | 26302d478e | ||
|   | b5af6b0bf9 | ||
|   | fb3816e2e5 | ||
|   | c4f4b247bc | ||
|   | bdc78a826f | ||
|   | 6570062e61 | ||
|   | 820facb833 | ||
|   | a933fcf7e2 | ||
|   | 2d3039c6a2 | ||
|   | dffd1b1d69 | ||
|   | 14fc64f2f1 | ||
|   | 04791d06d0 | ||
|   | a485451328 | ||
|   | df155bed89 | ||
|   | 3ee3630d22 | ||
|   | 6b2911b8c8 | ||
|   | d474200d7f | ||
|   | cc6f6277f6 | ||
|   | 0521e988bc | ||
|   | 832ff39eb6 | ||
|   | 134798e82f | ||
|   | afa7be6fdc | ||
|   | a49b49aaa9 | ||
|   | f329a5950e | ||
|   | 1be6223a31 | ||
|   | 76178b423e | ||
|   | 195e537499 | ||
|   | 4c1861dd27 | ||
|   | 1b2a50f9a3 | ||
|   | 347ec91bfc | ||
|   | 3b74cd5676 | ||
|   | 6f96dc8f23 | ||
|   | 76e0e935f0 | ||
|   | 3255382132 | ||
|   | 25831bfb60 | ||
|   | d011ca0626 | ||
|   | 2e28df3e8a | ||
|   | dcace43ce2 | ||
|   | befce5b887 | ||
|   | 9929c96650 | ||
|   | b4aec91d67 | ||
|   | d2a149ee23 | ||
|   | 8a72f94b3d | ||
|   | ae1d0a18f3 | ||
|   | cc68e6b6e6 | ||
|   | 6be5360bdd | ||
|   | 2b9007958b | ||
|   | 2f9c126d34 | ||
|   | ed69692f08 | ||
|   | 7119999df8 | ||
|   | c16462a0ce | 
							
								
								
									
										79
									
								
								.github/workflows/wled-ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										79
									
								
								.github/workflows/wled-ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,4 @@ | ||||
| name: PlatformIO CI | ||||
| name: WLED CI | ||||
|  | ||||
| on: [push, pull_request] | ||||
|  | ||||
| @@ -8,17 +8,11 @@ jobs: | ||||
|     name: Gather Environments | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v3 | ||||
|     - name: Cache pip | ||||
|       uses: actions/cache@v3 | ||||
|     - uses: actions/checkout@v4 | ||||
|     - uses: actions/setup-python@v5 | ||||
|       with: | ||||
|         path: ~/.cache/pip | ||||
|         key: ${{ runner.os }}-pip-${{ hashFiles('**/requirements.txt') }} | ||||
|         restore-keys: | | ||||
|           ${{ runner.os }}-pip- | ||||
|     - uses: actions/setup-python@v4 | ||||
|       with: | ||||
|         python-version: '3.9' | ||||
|         python-version: '3.12' | ||||
|         cache: 'pip' | ||||
|     - name: Install PlatformIO | ||||
|       run: pip install -r requirements.txt | ||||
|     - name: Get default environments | ||||
| @@ -38,54 +32,63 @@ jobs: | ||||
|       matrix: | ||||
|         environment: ${{ fromJSON(needs.get_default_envs.outputs.environments) }} | ||||
|     steps: | ||||
|     - uses: actions/checkout@v3 | ||||
|     - name: Cache pip | ||||
|       uses: actions/cache@v3 | ||||
|     - uses: actions/checkout@v4 | ||||
|     - name: Set up Node.js | ||||
|       uses: actions/setup-node@v4 | ||||
|       with: | ||||
|         path: ~/.cache/pip | ||||
|         key: ${{ runner.os }}-pip-${{ hashFiles('**/requirements.txt') }} | ||||
|         restore-keys: | | ||||
|           ${{ runner.os }}-pip- | ||||
|         cache: 'npm' | ||||
|     - run: npm install | ||||
|     - name: Cache PlatformIO | ||||
|       uses: actions/cache@v3 | ||||
|       uses: actions/cache@v4 | ||||
|       with: | ||||
|         path: ~/.platformio | ||||
|         key: ${{ runner.os }}-${{ hashFiles('**/lockfiles') }} | ||||
|         path: | | ||||
|               ~/.platformio/.cache | ||||
|               ~/.buildcache | ||||
|               build_output | ||||
|         key: pio-${{ runner.os }}-${{ matrix.environment }}-${{ hashFiles('platformio.ini', 'pio-scripts/output_bins.py') }}-${{ hashFiles('wled00/**', 'usermods/**') }} | ||||
|         restore-keys: pio-${{ runner.os }}-${{ matrix.environment }}-${{ hashFiles('platformio.ini', 'pio-scripts/output_bins.py') }}- | ||||
|     - name: Set up Python | ||||
|       uses: actions/setup-python@v4 | ||||
|       uses: actions/setup-python@v5 | ||||
|       with: | ||||
|           python-version: '3.9' | ||||
|           python-version: '3.12' | ||||
|           cache: 'pip' | ||||
|     - name: Install PlatformIO | ||||
|       run: pip install -r requirements.txt | ||||
|     - name: Build firmware | ||||
|       env: | ||||
|         WLED_RELEASE: True | ||||
|       run: pio run -e ${{ matrix.environment }} | ||||
|     - uses: actions/upload-artifact@v2 | ||||
|     - uses: actions/upload-artifact@v4 | ||||
|       with: | ||||
|         name: firmware-${{ matrix.environment }} | ||||
|         path: | | ||||
|           build_output/firmware/*.bin | ||||
|           build_output/firmware/*.gz | ||||
|     - uses: actions/upload-artifact@v2 | ||||
|       if: startsWith(github.ref, 'refs/tags/') | ||||
|       with: | ||||
|         name: firmware-release | ||||
|         path: build_output/release/*.bin | ||||
|           build_output/release/*.bin | ||||
|           build_output/release/*_ESP02.bin.gz | ||||
|   release: | ||||
|     name: Create Release | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: [get_default_envs, build] | ||||
|     needs: build | ||||
|     if: startsWith(github.ref, 'refs/tags/') | ||||
|     steps: | ||||
|     - uses: actions/download-artifact@v2 | ||||
|     - uses: actions/download-artifact@v4 | ||||
|       with: | ||||
|         name: firmware-release | ||||
|         merge-multiple: true | ||||
|     - name: Create draft release | ||||
|       uses: softprops/action-gh-release@v1 | ||||
|       with: | ||||
|         draft: True | ||||
|         files: | | ||||
|           *.bin | ||||
|       env: | ||||
|         GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|           *.bin.gz | ||||
|  | ||||
|  | ||||
|   testCdata: | ||||
|     name: Test cdata.js | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - name: Use Node.js | ||||
|         uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version: '20.x' | ||||
|           cache: 'npm' | ||||
|       - run: npm ci | ||||
|       - run: npm test | ||||
|   | ||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -2,7 +2,6 @@ | ||||
| .clang-format | ||||
| .direnv | ||||
| .DS_Store | ||||
| .gitignore | ||||
| .idea | ||||
| .pio | ||||
| .pioenvs | ||||
| @@ -22,3 +21,4 @@ wled-update.sh | ||||
| /wled00/my_config.h | ||||
| /wled00/Release | ||||
| /wled00/wled00.ino.cpp | ||||
| /wled00/html_*.h | ||||
							
								
								
									
										6
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							| @@ -9,8 +9,8 @@ | ||||
|       ], | ||||
|       "dependsOrder": "sequence", | ||||
|       "problemMatcher": [ | ||||
|         "$platformio", | ||||
|       ], | ||||
|         "$platformio" | ||||
|       ] | ||||
|     }, | ||||
|     { | ||||
|       "type": "PlatformIO", | ||||
| @@ -18,7 +18,7 @@ | ||||
|       "task": "Build", | ||||
|       "group": { | ||||
|         "kind": "build", | ||||
|         "isDefault": true, | ||||
|         "isDefault": true | ||||
|       }, | ||||
|       "problemMatcher": [ | ||||
|         "$platformio" | ||||
|   | ||||
							
								
								
									
										134
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										134
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,5 +1,139 @@ | ||||
| ## WLED changelog | ||||
|  | ||||
| #### Build 2403190 | ||||
| -   limit max PWM frequency (fix incorrect PWM resolution) | ||||
| -   Segment UI bugfix | ||||
| -   Updated AsyncWebServer (by @wlillmmiles) | ||||
| -   Simpler boot preset (fix for #3806) | ||||
| -   Effect: Fix for 2D Drift animation (#3816 by @BaptisteHudyma) | ||||
| -   Effect: Add twin option to 2D Drift | ||||
| -   MQTT cleanup | ||||
| -   DDP: Support sources that don't push (#3833 by @willmmiles) | ||||
| -   Usermod: Tetris AI usermod (#3711 by @muebau) | ||||
|  | ||||
| #### Build 2403171 | ||||
| -   merge 0.14.2 changes into 0.15 | ||||
|  | ||||
| #### Build 2403070 | ||||
| -   Add additional segment options when controlling over e1.31 (#3616 by @demophoon) | ||||
| -   LockedJsonResponse: Release early if possible (#3760 by @willmmiles) | ||||
| -   Update setup-node and cache usermods in wled-ci.yml (#3737 by @WoodyLetsCode) | ||||
| -   Fix preset sorting (#3790 by @WoodyLetsCode) | ||||
| -   compile time button configuration #3792 | ||||
| -   remove IR config if not compiled | ||||
| -   additional string optimisations | ||||
| -   Better low brightness level PWM handling (fixes #2767, #2868) | ||||
|  | ||||
| #### Build 2402290 | ||||
| -   Multiple analog button fix for #3549 | ||||
| -   Preset caching on chips with PSRAM (credit @akaricchi) | ||||
| -   Fixing stairway usermod and adding buildflags (by @lost-hope) | ||||
| -   ESP-NOW packet modification | ||||
| -   JSON buffer lock error messages / Reduce wait time for lock to 100ms | ||||
| -   Reduce string RAM usage for ESP8266 | ||||
| -   Fixing a potential array bounds violation in ESPDMX | ||||
| -   Move timezone table to PROGMEM (#3766 by @willmmiles) | ||||
| -   Reposition upload warning message. (fixes #3778) | ||||
| -   ABL display fix & optimisation | ||||
| -   Add virtual Art-Net RGBW option (#3783 by @shammy642) | ||||
|  | ||||
| #### Build 2402090 | ||||
| -   Added new Ethernet controller RGB2Go Tetra (duplicate of ESP3DEUXQuattro) | ||||
| -   Usermod: httpPullLightControl (#3560 by @roelbroersma) | ||||
| -   DMX: S2 & C3 support via modified ESPDMX | ||||
| -   Bugfix: prevent cleaning of JSON buffer after a failed lock attempt (BufferGuard) | ||||
| -   Product/Brand override (API & AP SSID) (#3750 by @moustachauve) | ||||
|  | ||||
| #### Build 2402060 | ||||
| -   WLED version 0.15.0-b1 | ||||
| -   Harmonic Random Cycle palette (#3729 by @dedehai) | ||||
| -   Multi PIR sensor usermod (added support for attaching multiple PIR sensors) | ||||
| -   Removed obsolete (and nonfunctional) usermods | ||||
|  | ||||
| #### Build 2309120 till build 2402010 | ||||
| -   WLED version 0.15.0-a0 | ||||
| -   Multi-WiFi support. Add up to 3 (or more via cusom compile) WiFis to connect to | ||||
| -   Temporary AP. Use your WLED in public with temporary AP. | ||||
| -   Github CI build system enhancements (#3718 by @WoodyLetsCode) | ||||
| -   Accessibility: Node list ( #3715 by @WoodyLetsCode) | ||||
| -   Analog clock overlay enhancement (#3489 by @WoodyLetsCode) | ||||
| -   ESP32-POE-WROVER from Olimex ethernet support (#3625 by @m-wachter) | ||||
| -   APA106 support (#3580 by @itstefanjanos) | ||||
| -   BREAKING: Effect: updated Palette effect to support 2D (#3683 by @TripleWhy) | ||||
| -   "SuperSync" from WLED MM (by @MoonModules) | ||||
| -   Effect: DNA Spiral Effect Speed Fix (#3723 by @Derek4aty1) | ||||
| -   Fix for #3693 | ||||
| -   Orange flash fix (#3196) for transitions | ||||
| -   Add own background image upload (#3596 by @WoodyLetsCode) | ||||
| -   WLED time overrides (`WLED_NTP_ENABLED`, `WLED_TIMEZONE`, `WLED_UTC_OFFSET`, `WLED_LAT` and `WLED_LON`) | ||||
| -   Better sorting and naming of static palettes (by @WoodyLetsCode) | ||||
| -   ANIMartRIX usermod and effects (#3673 by @netmindz) | ||||
| -   Use canvas instead of CSS gradient for liveview (#3621 by @zanhecht) | ||||
| -   Fix for #3672 | ||||
| -   ColoOrderMap W channel swap (color order overrides now have W swap) | ||||
| -   En-/disable LED maps when receiving realtime data (#3554 by @ezcGman) | ||||
| -   Added PWM frequency selection to UI (Settings) | ||||
| -   Automatically build UI before compiling (#3598, #3666 by @WoodyLetsCode) | ||||
| -   Internal: Added *suspend* API to `strip` (`WS2812FX class`) | ||||
| -   Possible fix for #3589 & partial fix for #3605 | ||||
| -   MPU6050 upgrade (#3654 by @willmmiles) | ||||
| -   UI internals (#3656 by @WoodyLetsCode) | ||||
| -   ColorPicker fix (#3658 by @WoodyLetsCode) | ||||
| -   Global JSON buffer guarding (#3648 by @willmmiles, resolves #3641, #3312, #3367, #3637, #3646, #3447) | ||||
| -   Effect: Fireworks 1D (fix for matrix trailing strip) | ||||
| -   BREAKING: Reduced number of segments (12) on ESP8266 due to less available RAM | ||||
| -   Increased available effect data buffer (increases more if board has PSRAM) | ||||
| -   Custom palette editor mobile UI enhancement (by @imeszaros) | ||||
| -   Per port Auto Brightness Limiter (ABL) | ||||
| -   Use PSRAM for JSON buffer (double size, larger ledmaps, up to 2k) | ||||
| -   Reduced heap fragmentation by allocating ledmap array only once and not deallocating effect buffer | ||||
| -   HTTP retries on failed UI load | ||||
| -   UI Search: scroll to top (#3587 by @WoodyLetsCode) | ||||
| -   Return to inline iro.js and rangetouch.js (#3597 by @WoodyLetsCode) | ||||
| -   Better caching (#3591 by @WoodyLetsCode) | ||||
| -   Do not send 404 for missing `skin.css` (#3590 by @WoodyLetsCode) | ||||
| -   Simplified UI rework (#3511 by @WoodyLetsCode) | ||||
| -   Domoticz device ID for PIR and Temperature usermods | ||||
| -   Bugfix for UCS8904 `hasWhite()` | ||||
| -   Better search in UI (#3540 by @WoodyLetsCode) | ||||
| -   Seeding FastLED PRNG (#3552 by @TripleWhy) | ||||
| -   WIZ Smart Button support (#3547 by @micw) | ||||
| -   New button type (button switch, fix for #3537) | ||||
| -   Pixel Magic Tool update (#3483 by @ajotanc) | ||||
| -   Effect: 2D Matrix fix for gaps | ||||
| -   Bugfix #3526, #3533, #3561 | ||||
| -   Spookier Halloween Eyes (#3501) | ||||
| -   Compile time options for Multi Relay usermod (#3498) | ||||
| -   Effect: Fix for Dissolve (#3502) | ||||
| -   Better reverse proxy support (nested paths) | ||||
| -   Implement global JSON API boolean toggle (i.e. instead of "var":true or "var":false -> "var":"t"). | ||||
| -   Sort presets by ID | ||||
| -   Fix for #3641, #3312, #3367, #3637, #3646, #3447, #3632, #3496, #2922, #3593, #3514, #3522, #3578 (partial), #3606 (@WoodyLetsCode) | ||||
| -   Improved random bg image and added random bg image options (@WoodyLetsCode, #3481) | ||||
| -   Audio palettes (Audioreactive usermod, credit @netmindz) | ||||
| -   Better UI tooltips (@ajotnac, #3464) | ||||
| -   Better effect filters (filter dropdown) | ||||
| -   UDP sync fix (for #3487) | ||||
| -   Power button override (solves #3431) | ||||
| -   Additional HTTP request throttling (ESP8266) | ||||
| -   Additional UI/UX improvements | ||||
| -   Segment class optimisations (internal) | ||||
| -   ESP-NOW sync | ||||
| -   ESP-NOW Wiz remote JSON overrides (similar to IR JSON) & bugfixes | ||||
| -   Gamma correction for custom palettes (#3399). | ||||
| -   Restore presets from browser local storage | ||||
| -   Optional effect blending | ||||
| -   Restructured UDP Sync (internal) | ||||
|     -   Remove sync receive | ||||
|     -   Sync clarification | ||||
| -   Disallow 2D effects on non-2D segments | ||||
| -   Return of 2 audio simulations | ||||
| -   Bugfix in sync #3344 (internal) | ||||
|     -   remove excessive segments | ||||
|     -   ignore inactive segments if not syncing bounds | ||||
|     -   send UDP/WS on segment change | ||||
|     -   pop_back() when removing last segment | ||||
|  | ||||
| #### Build 2403170 | ||||
| -   WLED 0.14.2 release | ||||
|  | ||||
|   | ||||
| @@ -89,4 +89,4 @@ Good: | ||||
|  | ||||
| There is no hard character limit for a comment within a line, | ||||
| though as a rule of thumb consider wrapping after 120 characters. | ||||
| Inline comments are OK if they describe that line only and are not exceedingly wide. | ||||
| Inline comments are OK if they describe that line only and are not exceedingly wide. | ||||
							
								
								
									
										368
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										368
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -1,26 +1,89 @@ | ||||
| { | ||||
|   "name": "wled", | ||||
|   "version": "0.14.2", | ||||
|   "version": "0.15.0-b1", | ||||
|   "lockfileVersion": 3, | ||||
|   "requires": true, | ||||
|   "packages": { | ||||
|     "": { | ||||
|       "name": "wled", | ||||
|       "version": "0.14.2", | ||||
|       "version": "0.15.0-b1", | ||||
|       "license": "ISC", | ||||
|       "dependencies": { | ||||
|         "clean-css": "^4.2.3", | ||||
|         "html-minifier-terser": "^5.1.1", | ||||
|         "clean-css": "^5.3.3", | ||||
|         "html-minifier-terser": "^7.2.0", | ||||
|         "inliner": "^1.13.1", | ||||
|         "nodemon": "^2.0.20", | ||||
|         "nodemon": "^3.0.2", | ||||
|         "zlib": "^1.0.5" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@jridgewell/gen-mapping": { | ||||
|       "version": "0.3.5", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.5.tgz", | ||||
|       "integrity": "sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==", | ||||
|       "dependencies": { | ||||
|         "@jridgewell/set-array": "^1.2.1", | ||||
|         "@jridgewell/sourcemap-codec": "^1.4.10", | ||||
|         "@jridgewell/trace-mapping": "^0.3.24" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=6.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@jridgewell/resolve-uri": { | ||||
|       "version": "3.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", | ||||
|       "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", | ||||
|       "engines": { | ||||
|         "node": ">=6.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@jridgewell/set-array": { | ||||
|       "version": "1.2.1", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.2.1.tgz", | ||||
|       "integrity": "sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==", | ||||
|       "engines": { | ||||
|         "node": ">=6.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@jridgewell/source-map": { | ||||
|       "version": "0.3.6", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/source-map/-/source-map-0.3.6.tgz", | ||||
|       "integrity": "sha512-1ZJTZebgqllO79ue2bm3rIGud/bOe0pP5BjSRCRxxYkEZS8STV7zN84UBbiYu7jy+eCKSnVIUgoWWE/tt+shMQ==", | ||||
|       "dependencies": { | ||||
|         "@jridgewell/gen-mapping": "^0.3.5", | ||||
|         "@jridgewell/trace-mapping": "^0.3.25" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@jridgewell/sourcemap-codec": { | ||||
|       "version": "1.4.15", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz", | ||||
|       "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==" | ||||
|     }, | ||||
|     "node_modules/@jridgewell/trace-mapping": { | ||||
|       "version": "0.3.25", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", | ||||
|       "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", | ||||
|       "dependencies": { | ||||
|         "@jridgewell/resolve-uri": "^3.1.0", | ||||
|         "@jridgewell/sourcemap-codec": "^1.4.14" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/abbrev": { | ||||
|       "version": "1.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", | ||||
|       "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==" | ||||
|     }, | ||||
|     "node_modules/acorn": { | ||||
|       "version": "8.11.3", | ||||
|       "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.11.3.tgz", | ||||
|       "integrity": "sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg==", | ||||
|       "bin": { | ||||
|         "acorn": "bin/acorn" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=0.4.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/ajv": { | ||||
|       "version": "6.12.6", | ||||
|       "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", | ||||
| @@ -179,9 +242,18 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/buffer-from": { | ||||
|       "version": "1.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", | ||||
|       "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==" | ||||
|       "version": "1.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", | ||||
|       "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==" | ||||
|     }, | ||||
|     "node_modules/camel-case": { | ||||
|       "version": "4.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/camel-case/-/camel-case-4.1.2.tgz", | ||||
|       "integrity": "sha512-gxGWBrTT1JuMx6R+o5PTXMmUnhnVzLQ9SNutD4YqKtI6ap897t3tKECYla6gCWEkplXnlNybEkZg9GEGxKFCgw==", | ||||
|       "dependencies": { | ||||
|         "pascal-case": "^3.1.2", | ||||
|         "tslib": "^2.0.3" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/camelcase": { | ||||
|       "version": "1.2.1", | ||||
| @@ -284,14 +356,14 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/clean-css": { | ||||
|       "version": "4.2.3", | ||||
|       "resolved": "https://registry.npmjs.org/clean-css/-/clean-css-4.2.3.tgz", | ||||
|       "integrity": "sha512-VcMWDN54ZN/DS+g58HYL5/n4Zrqe8vHJpGA8KdgUXFU4fuP/aHNw8eld9SyEIyabIMJX/0RaY/fplOo5hYLSFA==", | ||||
|       "version": "5.3.3", | ||||
|       "resolved": "https://registry.npmjs.org/clean-css/-/clean-css-5.3.3.tgz", | ||||
|       "integrity": "sha512-D5J+kHaVb/wKSFcyyV75uCn8fiY4sV38XJoe4CUyGQ+mOU/fMVYUdH1hJC+CJQ5uY3EnW27SbJYS4X8BiLrAFg==", | ||||
|       "dependencies": { | ||||
|         "source-map": "~0.6.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">= 4.0" | ||||
|         "node": ">= 10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/clean-css/node_modules/source-map": { | ||||
| @@ -343,9 +415,12 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/commander": { | ||||
|       "version": "2.20.3", | ||||
|       "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", | ||||
|       "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==" | ||||
|       "version": "10.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/commander/-/commander-10.0.1.tgz", | ||||
|       "integrity": "sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug==", | ||||
|       "engines": { | ||||
|         "node": ">=14" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/concat-map": { | ||||
|       "version": "0.0.1", | ||||
| @@ -489,29 +564,12 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/dot-case": { | ||||
|       "version": "3.0.3", | ||||
|       "resolved": "https://registry.npmjs.org/dot-case/-/dot-case-3.0.3.tgz", | ||||
|       "integrity": "sha512-7hwEmg6RiSQfm/GwPL4AAWXKy3YNNZA3oFv2Pdiey0mwkRCPZ9x6SZbkLcn8Ma5PYeVokzoD4Twv2n7LKp5WeA==", | ||||
|       "version": "3.0.4", | ||||
|       "resolved": "https://registry.npmjs.org/dot-case/-/dot-case-3.0.4.tgz", | ||||
|       "integrity": "sha512-Kv5nKlh6yRrdrGvxeJ2e5y2eRUpkUosIW4A2AS38zwSz27zu7ufDwQPi5Jhs3XAlGNetl3bmnGhQsMtkKJnj3w==", | ||||
|       "dependencies": { | ||||
|         "no-case": "^3.0.3", | ||||
|         "tslib": "^1.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/dot-case/node_modules/lower-case": { | ||||
|       "version": "2.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-2.0.1.tgz", | ||||
|       "integrity": "sha512-LiWgfDLLb1dwbFQZsSglpRj+1ctGnayXz3Uv0/WO8n558JycT5fg6zkNcnW0G68Nn0aEldTFeEfmjCfmqry/rQ==", | ||||
|       "dependencies": { | ||||
|         "tslib": "^1.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/dot-case/node_modules/no-case": { | ||||
|       "version": "3.0.3", | ||||
|       "resolved": "https://registry.npmjs.org/no-case/-/no-case-3.0.3.tgz", | ||||
|       "integrity": "sha512-ehY/mVQCf9BL0gKfsJBvFJen+1V//U+0HQMPrWct40ixE4jnv0bfvxDbWtAHL9EcaPEOJHVVYKoQn1TlZUB8Tw==", | ||||
|       "dependencies": { | ||||
|         "lower-case": "^2.0.1", | ||||
|         "tslib": "^1.10.0" | ||||
|         "no-case": "^3.0.4", | ||||
|         "tslib": "^2.0.3" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/duplexify": { | ||||
| @@ -764,58 +822,35 @@ | ||||
|         "node": ">=4" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/he": { | ||||
|       "version": "1.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", | ||||
|       "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", | ||||
|       "bin": { | ||||
|         "he": "bin/he" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/html-minifier-terser": { | ||||
|       "version": "5.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/html-minifier-terser/-/html-minifier-terser-5.1.1.tgz", | ||||
|       "integrity": "sha512-ZPr5MNObqnV/T9akshPKbVgyOqLmy+Bxo7juKCfTfnjNniTAMdy4hz21YQqoofMBJD2kdREaqPPdThoR78Tgxg==", | ||||
|       "version": "7.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/html-minifier-terser/-/html-minifier-terser-7.2.0.tgz", | ||||
|       "integrity": "sha512-tXgn3QfqPIpGl9o+K5tpcj3/MN4SfLtsx2GWwBC3SSd0tXQGyF3gsSqad8loJgKZGM3ZxbYDd5yhiBIdWpmvLA==", | ||||
|       "dependencies": { | ||||
|         "camel-case": "^4.1.1", | ||||
|         "clean-css": "^4.2.3", | ||||
|         "commander": "^4.1.1", | ||||
|         "he": "^1.2.0", | ||||
|         "param-case": "^3.0.3", | ||||
|         "camel-case": "^4.1.2", | ||||
|         "clean-css": "~5.3.2", | ||||
|         "commander": "^10.0.0", | ||||
|         "entities": "^4.4.0", | ||||
|         "param-case": "^3.0.4", | ||||
|         "relateurl": "^0.2.7", | ||||
|         "terser": "^4.6.3" | ||||
|         "terser": "^5.15.1" | ||||
|       }, | ||||
|       "bin": { | ||||
|         "html-minifier-terser": "cli.js" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=6" | ||||
|         "node": "^14.13.1 || >=16.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/html-minifier-terser/node_modules/camel-case": { | ||||
|       "version": "4.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/camel-case/-/camel-case-4.1.1.tgz", | ||||
|       "integrity": "sha512-7fa2WcG4fYFkclIvEmxBbTvmibwF2/agfEBc6q3lOpVu0A13ltLsA+Hr/8Hp6kp5f+G7hKi6t8lys6XxP+1K6Q==", | ||||
|       "dependencies": { | ||||
|         "pascal-case": "^3.1.1", | ||||
|         "tslib": "^1.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/html-minifier-terser/node_modules/commander": { | ||||
|       "version": "4.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz", | ||||
|       "integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==", | ||||
|     "node_modules/html-minifier-terser/node_modules/entities": { | ||||
|       "version": "4.5.0", | ||||
|       "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", | ||||
|       "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==", | ||||
|       "engines": { | ||||
|         "node": ">= 6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/html-minifier-terser/node_modules/param-case": { | ||||
|       "version": "3.0.3", | ||||
|       "resolved": "https://registry.npmjs.org/param-case/-/param-case-3.0.3.tgz", | ||||
|       "integrity": "sha512-VWBVyimc1+QrzappRs7waeN2YmoZFCGXWASRYX1/rGHtXqEcrGEIDm+jqIwFa2fRXNgQEwrxaYuIrX0WcAguTA==", | ||||
|       "dependencies": { | ||||
|         "dot-case": "^3.0.3", | ||||
|         "tslib": "^1.10.0" | ||||
|         "node": ">=0.12" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/fb55/entities?sponsor=1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/htmlparser2": { | ||||
| @@ -1226,6 +1261,14 @@ | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/lower-case": { | ||||
|       "version": "2.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-2.0.2.tgz", | ||||
|       "integrity": "sha512-7fm3l3NAF9WfN6W3JOmf5drwpVqX78JtoGJ3A6W0a6ZnldM41w2fV5D490psKFTpMds8TJse/eHLFFsNHHjHgg==", | ||||
|       "dependencies": { | ||||
|         "tslib": "^2.0.3" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/lowercase-keys": { | ||||
|       "version": "1.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", | ||||
| @@ -1234,6 +1277,17 @@ | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/lru-cache": { | ||||
|       "version": "6.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", | ||||
|       "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", | ||||
|       "dependencies": { | ||||
|         "yallist": "^4.0.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=10" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/mime": { | ||||
|       "version": "1.6.0", | ||||
|       "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", | ||||
| @@ -1304,18 +1358,27 @@ | ||||
|         "inherits": "~2.0.1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/no-case": { | ||||
|       "version": "3.0.4", | ||||
|       "resolved": "https://registry.npmjs.org/no-case/-/no-case-3.0.4.tgz", | ||||
|       "integrity": "sha512-fgAN3jGAh+RoxUGZHTSOLJIqUc2wmoBwGR4tbpNAKmmovFoWq0OdRkb0VkldReO2a2iBT/OEulG9XSUc10r3zg==", | ||||
|       "dependencies": { | ||||
|         "lower-case": "^2.0.2", | ||||
|         "tslib": "^2.0.3" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/nodemon": { | ||||
|       "version": "2.0.20", | ||||
|       "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.20.tgz", | ||||
|       "integrity": "sha512-Km2mWHKKY5GzRg6i1j5OxOHQtuvVsgskLfigG25yTtbyfRGn/GNvIbRyOf1PSCKJ2aT/58TiuUsuOU5UToVViw==", | ||||
|       "version": "3.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-3.1.0.tgz", | ||||
|       "integrity": "sha512-xqlktYlDMCepBJd43ZQhjWwMw2obW/JRvkrLxq5RCNcuDDX1DbcPT+qT1IlIIdf+DhnWs90JpTMe+Y5KxOchvA==", | ||||
|       "dependencies": { | ||||
|         "chokidar": "^3.5.2", | ||||
|         "debug": "^3.2.7", | ||||
|         "debug": "^4", | ||||
|         "ignore-by-default": "^1.0.1", | ||||
|         "minimatch": "^3.1.2", | ||||
|         "pstree.remy": "^1.1.8", | ||||
|         "semver": "^5.7.1", | ||||
|         "simple-update-notifier": "^1.0.7", | ||||
|         "semver": "^7.5.3", | ||||
|         "simple-update-notifier": "^2.0.0", | ||||
|         "supports-color": "^5.5.0", | ||||
|         "touch": "^3.1.0", | ||||
|         "undefsafe": "^2.0.5" | ||||
| @@ -1324,7 +1387,7 @@ | ||||
|         "nodemon": "bin/nodemon.js" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8.10.0" | ||||
|         "node": ">=10" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "type": "opencollective", | ||||
| @@ -1332,17 +1395,39 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/nodemon/node_modules/debug": { | ||||
|       "version": "3.2.7", | ||||
|       "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", | ||||
|       "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", | ||||
|       "version": "4.3.4", | ||||
|       "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", | ||||
|       "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", | ||||
|       "dependencies": { | ||||
|         "ms": "^2.1.1" | ||||
|         "ms": "2.1.2" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=6.0" | ||||
|       }, | ||||
|       "peerDependenciesMeta": { | ||||
|         "supports-color": { | ||||
|           "optional": true | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/nodemon/node_modules/ms": { | ||||
|       "version": "2.1.3", | ||||
|       "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", | ||||
|       "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" | ||||
|       "version": "2.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", | ||||
|       "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" | ||||
|     }, | ||||
|     "node_modules/nodemon/node_modules/semver": { | ||||
|       "version": "7.6.0", | ||||
|       "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", | ||||
|       "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", | ||||
|       "dependencies": { | ||||
|         "lru-cache": "^6.0.0" | ||||
|       }, | ||||
|       "bin": { | ||||
|         "semver": "bin/semver.js" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=10" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/nodemon/node_modules/supports-color": { | ||||
|       "version": "5.5.0", | ||||
| @@ -1446,30 +1531,22 @@ | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/param-case": { | ||||
|       "version": "3.0.4", | ||||
|       "resolved": "https://registry.npmjs.org/param-case/-/param-case-3.0.4.tgz", | ||||
|       "integrity": "sha512-RXlj7zCYokReqWpOPH9oYivUzLYZ5vAPIfEmCTNViosC78F8F0H9y7T7gG2M39ymgutxF5gcFEsyZQSph9Bp3A==", | ||||
|       "dependencies": { | ||||
|         "dot-case": "^3.0.4", | ||||
|         "tslib": "^2.0.3" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/pascal-case": { | ||||
|       "version": "3.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/pascal-case/-/pascal-case-3.1.1.tgz", | ||||
|       "integrity": "sha512-XIeHKqIrsquVTQL2crjq3NfJUxmdLasn3TYOU0VBM+UX2a6ztAWBlJQBePLGY7VHW8+2dRadeIPK5+KImwTxQA==", | ||||
|       "version": "3.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/pascal-case/-/pascal-case-3.1.2.tgz", | ||||
|       "integrity": "sha512-uWlGT3YSnK9x3BQJaOdcZwrnV6hPpd8jFH1/ucpiLRPh/2zCVJKS19E4GvYHvaCcACn3foXZ0cLB9Wrx1KGe5g==", | ||||
|       "dependencies": { | ||||
|         "no-case": "^3.0.3", | ||||
|         "tslib": "^1.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/pascal-case/node_modules/lower-case": { | ||||
|       "version": "2.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-2.0.1.tgz", | ||||
|       "integrity": "sha512-LiWgfDLLb1dwbFQZsSglpRj+1ctGnayXz3Uv0/WO8n558JycT5fg6zkNcnW0G68Nn0aEldTFeEfmjCfmqry/rQ==", | ||||
|       "dependencies": { | ||||
|         "tslib": "^1.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/pascal-case/node_modules/no-case": { | ||||
|       "version": "3.0.3", | ||||
|       "resolved": "https://registry.npmjs.org/no-case/-/no-case-3.0.3.tgz", | ||||
|       "integrity": "sha512-ehY/mVQCf9BL0gKfsJBvFJen+1V//U+0HQMPrWct40ixE4jnv0bfvxDbWtAHL9EcaPEOJHVVYKoQn1TlZUB8Tw==", | ||||
|       "dependencies": { | ||||
|         "lower-case": "^2.0.1", | ||||
|         "tslib": "^1.10.0" | ||||
|         "no-case": "^3.0.4", | ||||
|         "tslib": "^2.0.3" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/performance-now": { | ||||
| @@ -1775,22 +1852,28 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/simple-update-notifier": { | ||||
|       "version": "1.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/simple-update-notifier/-/simple-update-notifier-1.1.0.tgz", | ||||
|       "integrity": "sha512-VpsrsJSUcJEseSbMHkrsrAVSdvVS5I96Qo1QAQ4FxQ9wXFcB+pjj7FB7/us9+GcgfW4ziHtYMc1J0PLczb55mg==", | ||||
|       "version": "2.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/simple-update-notifier/-/simple-update-notifier-2.0.0.tgz", | ||||
|       "integrity": "sha512-a2B9Y0KlNXl9u/vsW6sTIu9vGEpfKu2wRV6l1H3XEas/0gUIzGzBoP/IouTcUQbm9JWZLH3COxyn03TYlFax6w==", | ||||
|       "dependencies": { | ||||
|         "semver": "~7.0.0" | ||||
|         "semver": "^7.5.3" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8.10.0" | ||||
|         "node": ">=10" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/simple-update-notifier/node_modules/semver": { | ||||
|       "version": "7.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/semver/-/semver-7.0.0.tgz", | ||||
|       "integrity": "sha512-+GB6zVA9LWh6zovYQLALHwv5rb2PHGlJi3lfiqIHxR0uuwCgefcOJc59v9fv1w8GbStwxuuqqAjI9NMAOOgq1A==", | ||||
|       "version": "7.6.0", | ||||
|       "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", | ||||
|       "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", | ||||
|       "dependencies": { | ||||
|         "lru-cache": "^6.0.0" | ||||
|       }, | ||||
|       "bin": { | ||||
|         "semver": "bin/semver.js" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=10" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/slide": { | ||||
| @@ -1810,9 +1893,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/source-map-support": { | ||||
|       "version": "0.5.19", | ||||
|       "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.19.tgz", | ||||
|       "integrity": "sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw==", | ||||
|       "version": "0.5.21", | ||||
|       "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz", | ||||
|       "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", | ||||
|       "dependencies": { | ||||
|         "buffer-from": "^1.0.0", | ||||
|         "source-map": "^0.6.0" | ||||
| @@ -1925,28 +2008,26 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/terser": { | ||||
|       "version": "4.8.1", | ||||
|       "resolved": "https://registry.npmjs.org/terser/-/terser-4.8.1.tgz", | ||||
|       "integrity": "sha512-4GnLC0x667eJG0ewJTa6z/yXrbLGv80D9Ru6HIpCQmO+Q4PfEtBFi0ObSckqwL6VyQv/7ENJieXHo2ANmdQwgw==", | ||||
|       "version": "5.29.2", | ||||
|       "resolved": "https://registry.npmjs.org/terser/-/terser-5.29.2.tgz", | ||||
|       "integrity": "sha512-ZiGkhUBIM+7LwkNjXYJq8svgkd+QK3UUr0wJqY4MieaezBSAIPgbSPZyIx0idM6XWK5CMzSWa8MJIzmRcB8Caw==", | ||||
|       "dependencies": { | ||||
|         "@jridgewell/source-map": "^0.3.3", | ||||
|         "acorn": "^8.8.2", | ||||
|         "commander": "^2.20.0", | ||||
|         "source-map": "~0.6.1", | ||||
|         "source-map-support": "~0.5.12" | ||||
|         "source-map-support": "~0.5.20" | ||||
|       }, | ||||
|       "bin": { | ||||
|         "terser": "bin/terser" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=6.0.0" | ||||
|         "node": ">=10" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/terser/node_modules/source-map": { | ||||
|       "version": "0.6.1", | ||||
|       "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", | ||||
|       "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", | ||||
|       "engines": { | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     "node_modules/terser/node_modules/commander": { | ||||
|       "version": "2.20.3", | ||||
|       "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", | ||||
|       "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==" | ||||
|     }, | ||||
|     "node_modules/then-fs": { | ||||
|       "version": "2.0.0", | ||||
| @@ -1999,9 +2080,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/tslib": { | ||||
|       "version": "1.14.1", | ||||
|       "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", | ||||
|       "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==" | ||||
|       "version": "2.6.2", | ||||
|       "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz", | ||||
|       "integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==" | ||||
|     }, | ||||
|     "node_modules/tunnel-agent": { | ||||
|       "version": "0.6.0", | ||||
| @@ -2150,6 +2231,11 @@ | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/yallist": { | ||||
|       "version": "4.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", | ||||
|       "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==" | ||||
|     }, | ||||
|     "node_modules/yargs": { | ||||
|       "version": "3.10.0", | ||||
|       "resolved": "https://registry.npmjs.org/yargs/-/yargs-3.10.0.tgz", | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|   "name": "wled", | ||||
|   "version": "0.14.2", | ||||
|   "version": "0.15.0-b1", | ||||
|   "description": "Tools for WLED project", | ||||
|   "main": "tools/cdata.js", | ||||
|   "directories": { | ||||
| @@ -9,6 +9,7 @@ | ||||
|   }, | ||||
|   "scripts": { | ||||
|     "build": "node tools/cdata.js", | ||||
|     "test": "node --test", | ||||
|     "dev": "nodemon -e js,html,htm,css,png,jpg,gif,ico,js -w tools/ -w wled00/data/ -x node tools/cdata.js" | ||||
|   }, | ||||
|   "repository": { | ||||
| @@ -22,10 +23,10 @@ | ||||
|   }, | ||||
|   "homepage": "https://github.com/Aircoookie/WLED#readme", | ||||
|   "dependencies": { | ||||
|     "clean-css": "^4.2.3", | ||||
|     "html-minifier-terser": "^5.1.1", | ||||
|     "clean-css": "^5.3.3", | ||||
|     "html-minifier-terser": "^7.2.0", | ||||
|     "inliner": "^1.13.1", | ||||
|     "nodemon": "^2.0.20", | ||||
|     "nodemon": "^3.0.2", | ||||
|     "zlib": "^1.0.5" | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										3
									
								
								pio-scripts/build_ui.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								pio-scripts/build_ui.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| Import('env') | ||||
|  | ||||
| env.Execute("npm run build") | ||||
| @@ -22,6 +22,16 @@ def _create_dirs(dirs=["firmware", "map"]): | ||||
|         if not os.path.isdir("{}{}".format(OUTPUT_DIR, d)): | ||||
|             os.mkdir("{}{}".format(OUTPUT_DIR, d)) | ||||
|  | ||||
| def create_release(source): | ||||
|     release_name = _get_cpp_define_value(env, "WLED_RELEASE_NAME") | ||||
|     if release_name: | ||||
|         _create_dirs(["release"]) | ||||
|         version = _get_cpp_define_value(env, "WLED_VERSION") | ||||
|         # get file extension of source file (.bin or .bin.gz) | ||||
|         ext = source.split(".", 1)[1] | ||||
|         release_file = "{}release{}WLED_{}_{}.{}".format(OUTPUT_DIR, os.path.sep, version, release_name, ext) | ||||
|         shutil.copy(source, release_file) | ||||
|  | ||||
| def bin_rename_copy(source, target, env): | ||||
|     _create_dirs() | ||||
|     variant = env["PIOENV"] | ||||
| @@ -30,14 +40,6 @@ def bin_rename_copy(source, target, env): | ||||
|     map_file = "{}map{}{}.map".format(OUTPUT_DIR, os.path.sep, variant) | ||||
|     bin_file = "{}firmware{}{}.bin".format(OUTPUT_DIR, os.path.sep, variant) | ||||
|  | ||||
|     release_name = _get_cpp_define_value(env, "WLED_RELEASE_NAME") | ||||
|  | ||||
|     if release_name: | ||||
|         _create_dirs(["release"]) | ||||
|         version = _get_cpp_define_value(env, "WLED_VERSION") | ||||
|         release_file = "{}release{}WLED_{}_{}.bin".format(OUTPUT_DIR, os.path.sep, version, release_name) | ||||
|         shutil.copy(str(target[0]), release_file) | ||||
|  | ||||
|     # check if new target files exist and remove if necessary | ||||
|     for f in [map_file, bin_file]: | ||||
|         if os.path.isfile(f): | ||||
| @@ -46,6 +48,8 @@ def bin_rename_copy(source, target, env): | ||||
|     # copy firmware.bin to firmware/<variant>.bin | ||||
|     shutil.copy(str(target[0]), bin_file) | ||||
|  | ||||
|     create_release(bin_file) | ||||
|  | ||||
|     # copy firmware.map to map/<variant>.map | ||||
|     if os.path.isfile("firmware.map"): | ||||
|         shutil.move("firmware.map", map_file) | ||||
| @@ -66,4 +70,6 @@ def bin_gzip(source, target, env): | ||||
|         with gzip.open(gzip_file, "wb", compresslevel = 9) as f: | ||||
|             shutil.copyfileobj(fp, f) | ||||
|  | ||||
|     create_release(gzip_file) | ||||
|  | ||||
| env.AddPostAction("$BUILD_DIR/${PROGNAME}.bin", [bin_rename_copy, bin_gzip]) | ||||
|   | ||||
							
								
								
									
										446
									
								
								platformio.ini
									
									
									
									
									
								
							
							
						
						
									
										446
									
								
								platformio.ini
									
									
									
									
									
								
							| @@ -9,40 +9,8 @@ | ||||
| # (use `platformio_override.ini` when building for your own board; see `platformio_override.ini.sample` for an example) | ||||
| # ------------------------------------------------------------------------------ | ||||
|  | ||||
| # CI binaries | ||||
| ; default_envs = nodemcuv2, esp8266_2m, esp01_1m_full, esp32dev, esp32_eth # ESP32 variant builds are temporarily excluded from CI due to toolchain issues on the GitHub Actions Linux environment | ||||
| default_envs = nodemcuv2, esp8266_2m, esp01_1m_full, nodemcuv2_160, esp8266_2m_160, esp01_1m_full_160, esp32dev, esp32_eth, esp32dev_audioreactive, lolin_s2_mini, esp32c3dev, esp32s3dev_8MB, esp32s3dev_8MB_PSRAM_opi | ||||
|  | ||||
| # Release binaries | ||||
| ; default_envs = nodemcuv2, esp8266_2m, esp01_1m_full, esp32dev, esp32_eth, lolin_s2_mini, esp32c3dev, esp32s3dev_8MB | ||||
|  | ||||
| # Build everything | ||||
| ; default_envs = esp32dev, esp8285_4CH_MagicHome, codm-controller-0_6-rev2, codm-controller-0_6, esp32s2_saola, d1_mini_5CH_Shojo_PCB, d1_mini, sp501e, nodemcuv2, esp32_eth, anavi_miracle_controller, esp07, esp01_1m_full, m5atom, h803wf, d1_mini_ota, heltec_wifi_kit_8, esp8285_H801, d1_mini_debug, wemos_shield_esp32, elekstube_ips | ||||
|  | ||||
| # Single binaries (uncomment your board) | ||||
| ; default_envs = elekstube_ips | ||||
| ; default_envs = nodemcuv2 | ||||
| ; default_envs = esp8266_2m | ||||
| ; default_envs = esp01_1m_full | ||||
| ; default_envs = esp07 | ||||
| ; default_envs = d1_mini | ||||
| ; default_envs = heltec_wifi_kit_8 | ||||
| ; default_envs = h803wf | ||||
| ; default_envs = d1_mini_debug | ||||
| ; default_envs = d1_mini_ota | ||||
| ; default_envs = esp32dev | ||||
| ; default_envs = esp8285_4CH_MagicHome | ||||
| ; default_envs = esp8285_H801 | ||||
| ; default_envs = d1_mini_5CH_Shojo_PCB | ||||
| ; default_envs = wemos_shield_esp32 | ||||
| ; default_envs = m5atom | ||||
| ; default_envs = esp32_eth | ||||
| ; default_envs = esp32dev_qio80 | ||||
| ; default_envs = esp32_eth_ota1mapp | ||||
| ; default_envs = esp32s2_saola | ||||
| ; default_envs = esp32c3dev | ||||
| ; default_envs = lolin_s2_mini | ||||
| ; default_envs = esp32s3dev_16MB_PSRAM_opi | ||||
| # CI/release binaries | ||||
| default_envs = nodemcuv2, esp8266_2m, esp01_1m_full, nodemcuv2_160, esp8266_2m_160, esp01_1m_full_160, esp32dev, esp32_eth, esp32dev_audioreactive, lolin_s2_mini, esp32c3dev, esp32s3dev_8MB, esp32s3dev_8MB_PSRAM_opi, esp32_wrover | ||||
|  | ||||
| src_dir  = ./wled00 | ||||
| data_dir = ./wled00/data | ||||
| @@ -61,9 +29,9 @@ extra_configs = | ||||
| arduino_core_2_6_3 = espressif8266@2.3.3 | ||||
| arduino_core_2_7_4 = espressif8266@2.6.2 | ||||
| arduino_core_3_0_0 = espressif8266@3.0.0 | ||||
| arduino_core_3_2_0 = espressif8266@3.2.0 | ||||
| arduino_core_4_1_0 = espressif8266@4.1.0 | ||||
| arduino_core_3_1_2 = espressif8266@4.2.0 | ||||
| arduino_core_3_0_2 = espressif8266@3.2.0 | ||||
| arduino_core_3_1_0 = espressif8266@4.1.0 | ||||
| arduino_core_3_1_2 = espressif8266@4.2.1 | ||||
|  | ||||
| # Development platforms | ||||
| arduino_core_develop = https://github.com/platformio/platform-espressif8266#develop | ||||
| @@ -133,11 +101,7 @@ build_flags = | ||||
|   -D DECODE_SONY=true | ||||
|   -D DECODE_SAMSUNG=true | ||||
|   -D DECODE_LG=true | ||||
|   ;-Dregister= # remove warnings in C++17 due to use of deprecated register keyword by the FastLED library ;; warning: this breaks framework code on ESP32-C3 and ESP32-S2 | ||||
|   -DWLED_USE_MY_CONFIG | ||||
|   ; -D USERMOD_SENSORSTOMQTT | ||||
|   #For ADS1115 sensor uncomment following | ||||
|   ; -D USERMOD_ADS1115 | ||||
|  | ||||
| build_unflags = | ||||
|  | ||||
| @@ -156,6 +120,7 @@ extra_scripts = | ||||
|   post:pio-scripts/output_bins.py | ||||
|   post:pio-scripts/strip-floats.py | ||||
|   pre:pio-scripts/user_config_copy.py | ||||
|   pre:pio-scripts/build_ui.py | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # COMMON SETTINGS: | ||||
| @@ -164,10 +129,8 @@ extra_scripts = | ||||
| framework = arduino | ||||
| board_build.flash_mode = dout | ||||
| monitor_speed = 115200 | ||||
| # slow upload speed (comment this out with a ';' when building for development use) | ||||
| # slow upload speed but most compatible (use platformio_override.ini to use faster speed) | ||||
| upload_speed = 115200 | ||||
| # fast upload speed (remove ';' when building for development use) | ||||
| ; upload_speed = 921600 | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # LIBRARIES: required dependencies | ||||
| @@ -182,20 +145,34 @@ lib_deps = | ||||
|     IRremoteESP8266 @ 2.8.2 | ||||
|     makuna/NeoPixelBus @ 2.7.5 | ||||
|     https://github.com/Aircoookie/ESPAsyncWebServer.git @ ^2.1.0 | ||||
|   # for I2C interface | ||||
|     ;Wire | ||||
|   # ESP-NOW library | ||||
|     ;gmag11/QuickESPNow @ ~0.7.0 | ||||
|     https://github.com/blazoncek/QuickESPNow.git#optional-debug | ||||
|   #For use of the TTGO T-Display ESP32 Module with integrated TFT display uncomment the following line | ||||
|     #TFT_eSPI | ||||
|   #For compatible OLED display uncomment following | ||||
|     #U8g2 #@ ~2.33.15 | ||||
|     #olikraus/U8g2 #@ ~2.33.15 | ||||
|   #For Dallas sensor uncomment following | ||||
|     #OneWire @ ~2.3.7 | ||||
|     #paulstoffregen/OneWire @ ~2.3.8 | ||||
|   #For BME280 sensor uncomment following | ||||
|     #BME280 @ ~3.0.0 | ||||
|     ; adafruit/Adafruit BMP280 Library @ 2.1.0 | ||||
|     ; adafruit/Adafruit CCS811 Library @ 1.0.4 | ||||
|     ; adafruit/Adafruit Si7021 Library @ 1.4.0 | ||||
|     ;adafruit/Adafruit BMP280 Library @ 2.1.0 | ||||
|     ;adafruit/Adafruit CCS811 Library @ 1.0.4 | ||||
|     ;adafruit/Adafruit Si7021 Library @ 1.4.0 | ||||
|   #For ADS1115 sensor uncomment following | ||||
|     ; adafruit/Adafruit BusIO @ 1.13.2 | ||||
|     ; adafruit/Adafruit ADS1X15 @ 2.4.0 | ||||
|     ;adafruit/Adafruit BusIO @ 1.13.2 | ||||
|     ;adafruit/Adafruit ADS1X15 @ 2.4.0 | ||||
|   #For MPU6050 IMU uncomment follwoing | ||||
|     ;electroniccats/MPU6050 @1.0.1 | ||||
|   # For -D USERMOD_ANIMARTRIX | ||||
|   # CC BY-NC 3.0 licensed effects by Stefan Petrick, include this usermod only if you accept the terms! | ||||
|     ;https://github.com/netmindz/animartrix.git#18bf17389e57c69f11bc8d04ebe1d215422c7fb7 | ||||
|   # SHT85 | ||||
|     ;robtillaart/SHT85@~0.3.3 | ||||
|   # Audioreactive usermod | ||||
|     ;kosme/arduinoFFT @ 2.0.0 | ||||
|  | ||||
| extra_scripts = ${scripts_defaults.extra_scripts} | ||||
|  | ||||
| @@ -218,6 +195,7 @@ build_flags = | ||||
|   ; restrict to minimal mime-types | ||||
|   -DMIMETYPE_MINIMAL | ||||
|   ; other special-purpose framework flags (see https://docs.platformio.org/en/latest/platforms/espressif8266.html) | ||||
|   ; decrease code cache size and increase IRAM to fit all pixel functions | ||||
|   -D PIO_FRAMEWORK_ARDUINO_MMU_CACHE16_IRAM48 ;; in case of linker errors like "section `.text1' will not fit in region `iram1_0_seg'" | ||||
|   ; -D PIO_FRAMEWORK_ARDUINO_MMU_CACHE16_IRAM48_SECHEAP_SHARED ;; (experimental) adds some extra heap, but may cause slowdown | ||||
|  | ||||
| @@ -244,8 +222,8 @@ lib_deps = | ||||
|   https://github.com/pbolduc/AsyncTCP.git @ 1.2.0 | ||||
|   ${env.lib_deps} | ||||
| # additional build flags for audioreactive | ||||
| AR_build_flags = -D USERMOD_AUDIOREACTIVE -D UM_AUDIOREACTIVE_USE_NEW_FFT | ||||
| AR_lib_deps = https://github.com/kosme/arduinoFFT#419d7b0 | ||||
| AR_build_flags = -D USERMOD_AUDIOREACTIVE | ||||
| AR_lib_deps = kosme/arduinoFFT @ 2.0.0 | ||||
|  | ||||
| [esp32_idf_V4] | ||||
| ;; experimental build environment for ESP32 using ESP-IDF 4.4.x / arduino-esp32 v2.0.5 | ||||
| @@ -270,6 +248,7 @@ lib_deps = | ||||
| ;; generic definitions for all ESP32-S2 boards | ||||
| platform = espressif32@5.3.0 | ||||
| platform_packages = | ||||
| default_partitions = tools/WLED_ESP32_4MB_1MB_FS.csv | ||||
| build_flags = -g | ||||
|   -DARDUINO_ARCH_ESP32 | ||||
|   -DARDUINO_ARCH_ESP32S2 | ||||
| @@ -280,7 +259,6 @@ build_flags = -g | ||||
|   -DARDUINO_USB_MODE=0 ;; this flag is mandatory for ESP32-S2 ! | ||||
|   ;; please make sure that the following flags are properly set (to 0 or 1) by your board.json, or included in your custom platformio_override.ini entry: | ||||
|   ;; ARDUINO_USB_CDC_ON_BOOT | ||||
|  | ||||
| lib_deps = | ||||
|   https://github.com/pbolduc/AsyncTCP.git @ 1.2.0 | ||||
|   ${env.lib_deps} | ||||
| @@ -298,7 +276,6 @@ build_flags = -g | ||||
|   -DARDUINO_USB_MODE=1 ;; this flag is mandatory for ESP32-C3 | ||||
|   ;; please make sure that the following flags are properly set (to 0 or 1) by your board.json, or included in your custom platformio_override.ini entry: | ||||
|   ;; ARDUINO_USB_CDC_ON_BOOT | ||||
|  | ||||
| lib_deps = | ||||
|   https://github.com/pbolduc/AsyncTCP.git @ 1.2.0 | ||||
|   ${env.lib_deps} | ||||
| @@ -372,44 +349,6 @@ board_build.f_cpu = 160000000L | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP01_160 -D WLED_DISABLE_OTA | ||||
|   ; -D WLED_USE_UNREAL_MATH ;; may cause wrong sunset/sunrise times, but saves 7064 bytes FLASH and 975 bytes RAM | ||||
|  | ||||
| [env:esp07] | ||||
| board = esp07 | ||||
| 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} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:d1_mini] | ||||
| board = d1_mini | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| upload_speed = 921600 | ||||
| board_build.ldscript = ${common.ldscript_4m1m} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
| monitor_filters = esp8266_exception_decoder | ||||
|  | ||||
| [env:heltec_wifi_kit_8] | ||||
| board = d1_mini | ||||
| 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} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:h803wf] | ||||
| board = d1_mini | ||||
| 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} -D LEDPIN=1 -D WLED_DISABLE_INFRARED | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:esp32dev] | ||||
| board = esp32dev | ||||
| platform = ${esp32.platform} | ||||
| @@ -434,33 +373,6 @@ board_build.partitions = ${esp32.default_partitions} | ||||
| ; board_build.f_flash = 80000000L | ||||
| ; board_build.flash_mode = dio | ||||
|  | ||||
| [env:esp32dev_qio80] | ||||
| board = esp32dev | ||||
| platform = ${esp32.platform} | ||||
| platform_packages = ${esp32.platform_packages} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} -D WLED_RELEASE_NAME=ESP32_qio80 #-D WLED_DISABLE_BROWNOUT_DET | ||||
| lib_deps = ${esp32.lib_deps} | ||||
| monitor_filters = esp32_exception_decoder | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
| board_build.f_flash = 80000000L | ||||
| board_build.flash_mode = qio | ||||
|  | ||||
| [env:esp32dev_V4_dio80] | ||||
| ;; experimental ESP32 env using ESP-IDF V4.4.x | ||||
| ;; Warning: this build environment is not stable!! | ||||
| ;; please erase your device before installing. | ||||
| 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_RELEASE_NAME=ESP32_V4_qio80 #-D WLED_DISABLE_BROWNOUT_DET | ||||
| lib_deps = ${esp32_idf_V4.lib_deps} | ||||
| monitor_filters = esp32_exception_decoder | ||||
| board_build.partitions = ${esp32_idf_V4.default_partitions} | ||||
| board_build.f_flash = 80000000L | ||||
| board_build.flash_mode = dio | ||||
|  | ||||
| [env:esp32_eth] | ||||
| board = esp32-poe | ||||
| platform = ${esp32.platform} | ||||
| @@ -472,19 +384,18 @@ build_flags = ${common.build_flags_esp32} -D WLED_RELEASE_NAME=ESP32_Ethernet -D | ||||
| lib_deps = ${esp32.lib_deps} | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
|  | ||||
| [env:esp32s2_saola] | ||||
| board = esp32-s2-saola-1 | ||||
| platform = https://github.com/tasmota/platform-espressif32/releases/download/v2.0.2.2/platform-tasmota-espressif32-2.0.2.zip | ||||
| platform_packages = | ||||
| framework = arduino | ||||
| board_build.partitions = tools/WLED_ESP32_4MB_1MB_FS.csv | ||||
| [env:esp32_wrover] | ||||
| platform = ${esp32.platform} | ||||
| board = ttgo-t7-v14-mini32 | ||||
| board_build.f_flash = 80000000L | ||||
| board_build.flash_mode = qio | ||||
| upload_speed = 460800 | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags} ${esp32s2.build_flags} #-D WLED_RELEASE_NAME=S2_saola | ||||
|   ;-DLOLIN_WIFI_FIX ;; try this in case Wifi does not work | ||||
|   -DARDUINO_USB_CDC_ON_BOOT=1 | ||||
| lib_deps = ${esp32s2.lib_deps} | ||||
| build_flags = ${common.build_flags_esp32} -D WLED_RELEASE_NAME=ESP32_WROVER | ||||
|   -DBOARD_HAS_PSRAM -mfix-esp32-psram-cache-issue | ||||
|   -D WLED_USE_PSRAM | ||||
|   -D LEDPIN=25 | ||||
| lib_deps = ${esp32.lib_deps} | ||||
|  | ||||
| [env:esp32c3dev] | ||||
| extends = esp32c3 | ||||
| @@ -529,7 +440,7 @@ platform = ${esp32s3.platform} | ||||
| platform_packages = ${esp32s3.platform_packages} | ||||
| upload_speed = 921600 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags} ${esp32s3.build_flags} | ||||
| build_flags = ${common.build_flags} ${esp32s3.build_flags} -D WLED_RELEASE_NAME=ESP32-S3_8MB_PSRAM_opi | ||||
|   -D CONFIG_LITTLEFS_FOR_IDF_3_2 -D WLED_WATCHDOG_TIMEOUT=0 | ||||
|   ;-D ARDUINO_USB_CDC_ON_BOOT=0  ;; -D ARDUINO_USB_MODE=1 ;; for boards with serial-to-USB chip | ||||
|   -D ARDUINO_USB_CDC_ON_BOOT=1 -D ARDUINO_USB_MODE=1      ;; for boards with USB-OTG connector only (USBCDC or "TinyUSB") | ||||
| @@ -541,100 +452,24 @@ board_build.f_flash = 80000000L | ||||
| board_build.flash_mode = qio | ||||
| monitor_filters = esp32_exception_decoder | ||||
|  | ||||
| [env:esp32s3dev_16MB_PSRAM_opi] | ||||
| extends = env:esp32s3dev_8MB_PSRAM_opi | ||||
| board_build.partitions = tools/WLED_ESP32_16MB.csv | ||||
| board_upload.flash_size = 16MB | ||||
|  | ||||
| [env:esp32s3dev_8MB_PSRAM_qspi] | ||||
| ;; ESP32-TinyS3 development board, with 8MB FLASH and PSRAM (memory_type: qio_qspi) | ||||
| extends = env:esp32s3dev_8MB_PSRAM_opi | ||||
| ;board = um_tinys3 ;    -> needs workaround from https://github.com/Aircoookie/WLED/pull/2905#issuecomment-1328049860 | ||||
| board = esp32-s3-devkitc-1 ;; generic dev board; the next line adds PSRAM support | ||||
| board_build.arduino.memory_type = qio_qspi ;; use with PSRAM: 2MB or  4MB | ||||
|  | ||||
| [env:esp8285_4CH_MagicHome] | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_1m128k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_DISABLE_OTA | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:esp8285_H801] | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_1m128k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_DISABLE_OTA | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:d1_mini_5CH_Shojo_PCB] | ||||
| board = d1_mini | ||||
| 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} -D WLED_USE_SHOJO_PCB | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # DEVELOPMENT BOARDS | ||||
| # ------------------------------------------------------------------------------ | ||||
|  | ||||
| [env:d1_mini_debug] | ||||
| board = d1_mini | ||||
| build_type = debug | ||||
| 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} ${common.debug_flags} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:d1_mini_ota] | ||||
| board = d1_mini | ||||
| upload_protocol = espota | ||||
| # exchange for your WLED IP | ||||
| upload_port = "10.10.1.27" | ||||
| 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} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:anavi_miracle_controller] | ||||
| board = d1_mini | ||||
| 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} -D LEDPIN=12 -D IRPIN=-1 -D RLYPIN=2 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:lolin_s2_mini] | ||||
| platform = ${esp32s2.platform} | ||||
| platform_packages = ${esp32s2.platform_packages} | ||||
| board = lolin_s2_mini | ||||
| board_build.partitions = tools/WLED_ESP32_4MB_1MB_FS.csv | ||||
| build_unflags = ${common.build_unflags} #-DARDUINO_USB_CDC_ON_BOOT=1 | ||||
| ;board_build.flash_mode = qio | ||||
| ;board_build.f_flash = 80000000L | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags} ${esp32s2.build_flags} -D WLED_RELEASE_NAME=ESP32-S2 | ||||
|   -DBOARD_HAS_PSRAM | ||||
|   -DARDUINO_USB_CDC_ON_BOOT=1 # try disabling and enabling unflag above in case of board-specific issues, will disable Serial | ||||
|   -DARDUINO_USB_CDC_ON_BOOT=1 | ||||
|   -DARDUINO_USB_MSC_ON_BOOT=0 | ||||
|   -DARDUINO_USB_DFU_ON_BOOT=0 | ||||
|   -DLOLIN_WIFI_FIX ; seems to work much better with this | ||||
|   -D WLED_USE_PSRAM | ||||
|   ; -D WLED_USE_UNREAL_MATH ;; may cause wrong sunset/sunrise times, but saves 6792 bytes FLASH | ||||
|   -D WLED_WATCHDOG_TIMEOUT=0 | ||||
|   -D CONFIG_ASYNC_TCP_USE_WDT=0 | ||||
|   -D LEDPIN=16 | ||||
|   -D BTNPIN=18 | ||||
|   -D RLYPIN=9 | ||||
|   -D IRPIN=7 | ||||
|   -D HW_PIN_SCL=35 | ||||
|   -D HW_PIN_SDA=33 | ||||
|   -D HW_PIN_CLOCKSPI=7 | ||||
| @@ -642,190 +477,3 @@ build_flags = ${common.build_flags} ${esp32s2.build_flags} -D WLED_RELEASE_NAME= | ||||
|   -D HW_PIN_MISOSPI=9 | ||||
| ;  -D STATUSLED=15 | ||||
| lib_deps = ${esp32s2.lib_deps} | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # custom board configurations | ||||
| # ------------------------------------------------------------------------------ | ||||
|  | ||||
| [env:esp32c3dev_2MB] | ||||
| ;; for ESP32-C3 boards with 2MB flash (instead of 4MB). | ||||
| ;; this board need a specific partition file. OTA not possible. | ||||
| extends = esp32c3 | ||||
| platform = ${esp32c3.platform} | ||||
| platform_packages = ${esp32c3.platform_packages} | ||||
| board = esp32-c3-devkitm-1 | ||||
| build_flags = ${common.build_flags} ${esp32c3.build_flags} #-D WLED_RELEASE_NAME=ESP32-C3 | ||||
|   -D WLED_WATCHDOG_TIMEOUT=0 | ||||
|   -D WLED_DISABLE_OTA | ||||
|   ; -DARDUINO_USB_CDC_ON_BOOT=1 ;; for virtual CDC USB | ||||
|   -DARDUINO_USB_CDC_ON_BOOT=0   ;; for serial-to-USB chip | ||||
| build_unflags = ${common.build_unflags} | ||||
| upload_speed = 115200 | ||||
| lib_deps = ${esp32c3.lib_deps} | ||||
| board_build.partitions = tools/WLED_ESP32_2MB_noOTA.csv | ||||
| board_build.flash_mode = dio | ||||
| board_upload.flash_size = 2MB | ||||
| board_upload.maximum_size = 2097152 | ||||
|  | ||||
| [env:wemos_shield_esp32] | ||||
| board = esp32dev | ||||
| platform = ${esp32.platform} | ||||
| platform_packages = ${esp32.platform_packages} | ||||
| upload_speed = 460800 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} | ||||
|   -D LEDPIN=16 | ||||
|   -D RLYPIN=19 | ||||
|   -D BTNPIN=17 | ||||
|   -D IRPIN=18 | ||||
|   -U WLED_USE_MY_CONFIG | ||||
|   -D USERMOD_DALLASTEMPERATURE | ||||
|   -D USERMOD_FOUR_LINE_DISPLAY | ||||
|   -D TEMPERATURE_PIN=23 | ||||
|   -D USE_ALT_DISPlAY ; new versions of USERMOD_FOUR_LINE_DISPLAY and USERMOD_ROTARY_ENCODER_UI | ||||
|   -D USERMOD_AUDIOREACTIVE | ||||
| lib_deps = ${esp32.lib_deps} | ||||
|   OneWire@~2.3.5 | ||||
|   olikraus/U8g2 @ ^2.28.8 | ||||
|   https://github.com/blazoncek/arduinoFFT.git | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
|  | ||||
| [env:m5atom] | ||||
| board = esp32dev | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} -D LEDPIN=27 -D BTNPIN=39 | ||||
| lib_deps = ${esp32.lib_deps} | ||||
| platform = ${esp32.platform} | ||||
| platform_packages = ${esp32.platform_packages} | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
|  | ||||
| [env:sp501e] | ||||
| board = esp_wroom_02 | ||||
| platform = ${common.platform_wled_default} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_flags = ${common.build_flags_esp8266} -D LEDPIN=3 -D BTNPIN=1 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:sp511e] | ||||
| board = esp_wroom_02 | ||||
| platform = ${common.platform_wled_default} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_flags = ${common.build_flags_esp8266} -D LEDPIN=3 -D BTNPIN=2 -D IRPIN=5 -D WLED_MAX_BUTTONS=3 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:Athom_RGBCW]        ;7w and 5w(GU10) bulbs | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP8266 -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 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
|  | ||||
| [env:Athom_15w_RGBCW]        ;15w bulb | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP8266 -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 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
|  | ||||
| [env:Athom_3Pin_Controller]        ;small controller with only data | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP8266 -D BTNPIN=0 -D RLYPIN=-1 -D LEDPIN=1 -D WLED_DISABLE_INFRARED | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
|  | ||||
| [env:Athom_4Pin_Controller]       ; With clock and data interface | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP8266 -D BTNPIN=0 -D RLYPIN=12 -D LEDPIN=1 -D WLED_DISABLE_INFRARED | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
|  | ||||
| [env:Athom_5Pin_Controller]      ;Analog light strip controller | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP8266 -D BTNPIN=0 -D RLYPIN=-1 DATA_PINS=4,12,14,13 -D WLED_DISABLE_INFRARED | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
|  | ||||
| [env:MY9291] | ||||
| board = esp01_1m | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_1m128k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP01 -D WLED_DISABLE_OTA -D USERMOD_MY9291 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # codm pixel controller board configurations | ||||
| # codm-controller-0_6 can also be used for the TYWE3S controller | ||||
| # ------------------------------------------------------------------------------ | ||||
|  | ||||
| [env:codm-controller-0_6] | ||||
| board = esp_wroom_02 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:codm-controller-0_6-rev2] | ||||
| board = esp_wroom_02 | ||||
| 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} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # EleksTube-IPS | ||||
| # ------------------------------------------------------------------------------ | ||||
| [env:elekstube_ips] | ||||
| board = esp32dev | ||||
| platform = ${esp32.platform} | ||||
| platform_packages = ${esp32.platform_packages} | ||||
| upload_speed = 921600 | ||||
| build_flags = ${common.build_flags_esp32} -D WLED_DISABLE_BROWNOUT_DET -D WLED_DISABLE_INFRARED | ||||
|   -D USERMOD_RTC | ||||
|   -D USERMOD_ELEKSTUBE_IPS | ||||
|   -D LEDPIN=12 | ||||
|   -D RLYPIN=27 | ||||
|   -D BTNPIN=34 | ||||
|   -D DEFAULT_LED_COUNT=6 | ||||
|   # Display config | ||||
|   -D ST7789_DRIVER | ||||
|   -D TFT_WIDTH=135 | ||||
|   -D TFT_HEIGHT=240 | ||||
|   -D CGRAM_OFFSET | ||||
|   -D TFT_SDA_READ | ||||
|   -D TFT_MOSI=23 | ||||
|   -D TFT_SCLK=18 | ||||
|   -D TFT_DC=25 | ||||
|   -D TFT_RST=26 | ||||
|   -D SPI_FREQUENCY=40000000 | ||||
|   -D USER_SETUP_LOADED | ||||
| monitor_filters = esp32_exception_decoder | ||||
| lib_deps = | ||||
|   ${esp32.lib_deps} | ||||
|   TFT_eSPI @ ^2.3.70 | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
|   | ||||
| @@ -1,65 +0,0 @@ | ||||
| # Example PlatformIO Project Configuration Override | ||||
| # ------------------------------------------------------------------------------ | ||||
| # Copy to platformio_override.ini to activate overrides | ||||
| # ------------------------------------------------------------------------------ | ||||
| # Please visit documentation: https://docs.platformio.org/page/projectconf.html | ||||
|  | ||||
| [platformio] | ||||
| default_envs = WLED_tasmota_1M | ||||
|  | ||||
| [env:WLED_tasmota_1M] | ||||
| board = esp01_1m | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_1m128k} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} | ||||
| ; ********************************************************************* | ||||
| ; *** Use custom settings from file my_config.h | ||||
|    -DWLED_USE_MY_CONFIG | ||||
| ; ********************************************************************* | ||||
| ; | ||||
| ; | ||||
| ; *** To use the below defines/overrides, copy and paste each onto it's own line just below build_flags in the section above. | ||||
| ;  | ||||
| ; disable specific features | ||||
| ;  -D WLED_DISABLE_OTA | ||||
| ;  -D WLED_DISABLE_ALEXA | ||||
| ;  -D WLED_DISABLE_HUESYNC | ||||
| ;  -D WLED_DISABLE_INFRARED | ||||
| ;  -D WLED_DISABLE_WEBSOCKETS | ||||
| ; PIN defines - uncomment and change, if needed: | ||||
| ;   -D LEDPIN=2 | ||||
| ;   -D BTNPIN=0 | ||||
| ;   -D TOUCHPIN=T0 | ||||
| ;   -D IRPIN=4 | ||||
| ;   -D RLYPIN=12 | ||||
| ;   -D RLYMDE=1 | ||||
| ; digital LED strip types - uncomment only one ! - this will disable WS281x / SK681x support | ||||
| ;   -D USE_APA102 | ||||
| ;   -D USE_WS2801 | ||||
| ;   -D USE_LPD8806 | ||||
| ; PIN defines for 2 wire LEDs | ||||
|    -D CLKPIN=0 | ||||
|    -D DATAPIN=2 | ||||
| ; to drive analog LED strips (aka 5050) hardware configuration is no longer necessary | ||||
| ; configure the settings in the UI as follows (hard): | ||||
| ;   for the Magic Home LED Controller use PWM pins 5,12,13,15 | ||||
| ;   for the H801 controller use PINs 15,13,12,14 (W2 = 04) | ||||
| ;   for the BW-LT11 controller use PINs 12,4,14,5 | ||||
| ;    | ||||
| ; 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 | ||||
| ;    | ||||
| ; set milliampere limit when using ESP pin to power leds | ||||
| ;   -D ABL_MILLIAMPS_DEFAULT=850 | ||||
| ; | ||||
| ; 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 | ||||
							
								
								
									
										495
									
								
								platformio_override.sample.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										495
									
								
								platformio_override.sample.ini
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,495 @@ | ||||
| # Example PlatformIO Project Configuration Override | ||||
| # ------------------------------------------------------------------------------ | ||||
| # Copy to platformio_override.ini to activate overrides | ||||
| # ------------------------------------------------------------------------------ | ||||
| # Please visit documentation: https://docs.platformio.org/page/projectconf.html | ||||
|  | ||||
| [platformio] | ||||
| default_envs = WLED_tasmota_1M  # define as many as you need | ||||
|  | ||||
| #---------- | ||||
| # SAMPLE | ||||
| #---------- | ||||
| [env:WLED_tasmota_1M] | ||||
| extends = env:esp01_1m_full  # when you want to extend the existing environment (define only updated options) | ||||
| ; board = esp01_1m  # uncomment when ou need different board | ||||
| ; platform = ${common.platform_wled_default}  # uncomment and change when you want particular platform | ||||
| ; platform_packages = ${common.platform_packages} | ||||
| ; board_build.ldscript = ${common.ldscript_1m128k} | ||||
| ; upload_speed = 921600 # fast upload speed (remove ';' if your board supports fast upload speed) | ||||
| # Sample libraries used for various usermods. Uncomment when using particular usermod. | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
| ;  olikraus/U8g2 # @~2.33.15 | ||||
| ;  paulstoffregen/OneWire@~2.3.8 | ||||
| ;  adafruit/Adafruit Unified Sensor@^1.1.4 | ||||
| ;  adafruit/DHT sensor library@^1.4.1 | ||||
| ;  adafruit/Adafruit BME280 Library@^2.2.2 | ||||
| ;  Wire | ||||
| ;  robtillaart/SHT85@~0.3.3 | ||||
| ;  gmag11/QuickESPNow ;@ 0.6.2 | ||||
| ;  https://github.com/blazoncek/QuickESPNow.git#optional-debug  ;; exludes debug library | ||||
| ;  https://github.com/kosme/arduinoFFT#develop @ 1.9.2+sha.419d7b0 ;; used for USERMOD_AUDIOREACTIVE - using "known working" hash | ||||
| ; build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} | ||||
| ; | ||||
| ; *** To use the below defines/overrides, copy and paste each onto it's own line just below build_flags in the section above. | ||||
| ;  | ||||
| ; disable specific features | ||||
| ;   -D WLED_DISABLE_OTA | ||||
| ;   -D WLED_DISABLE_ALEXA | ||||
| ;   -D WLED_DISABLE_HUESYNC | ||||
| ;   -D WLED_DISABLE_LOXONE | ||||
| ;   -D WLED_DISABLE_INFRARED | ||||
| ;   -D WLED_DISABLE_WEBSOCKETS | ||||
| ;   -D WLED_DISABLE_MQTT | ||||
| ;   -D WLED_DISABLE_ADALIGHT | ||||
| ;   -D WLED_DISABLE_2D | ||||
| ;   -D WLED_DISABLE_PXMAGIC | ||||
| ;   -D WLED_DISABLE_ESPNOW | ||||
| ;   -D WLED_DISABLE_BROWNOUT_DET | ||||
| ; | ||||
| ; PIN defines - uncomment and change, if needed: | ||||
| ;   -D LEDPIN=2 | ||||
| ; or use this for multiple outputs | ||||
| ;   -D DATA_PINS=1,3 | ||||
| ;   -D BTNPIN=0 | ||||
| ;   -D IRPIN=4 | ||||
| ;   -D RLYPIN=12 | ||||
| ;   -D RLYMDE=1 | ||||
| ;   -D LED_BUILTIN=2 # GPIO of built-in LED | ||||
| ; | ||||
| ; Limit max buses | ||||
| ;   -D WLED_MAX_BUSSES=2 | ||||
| ; | ||||
| ; Configure default WiFi | ||||
| ;   -D CLIENT_SSID='"MyNetwork"' | ||||
| ;   -D CLIENT_PASS='"Netw0rkPassw0rd"' | ||||
| ; | ||||
| ; Configure and use Ethernet | ||||
| ;   -D WLED_USE_ETHERNET | ||||
| ;   -D WLED_ETH_DEFAULT=5 | ||||
| ; do not use pins 5, (16,) 17, 18, 19, 21, 22, 23, 25, 26, 27 for anything but ethernet | ||||
| ;   -D PHY_ADDR=0 -D ETH_PHY_POWER=5 -D ETH_PHY_MDC=23 -D ETH_PHY_MDIO=18 | ||||
| ;   -D ETH_CLK_MODE=ETH_CLOCK_GPIO17_OUT | ||||
| ; | ||||
| ; NTP time configuration | ||||
| ;   -D WLED_NTP_ENABLED=true | ||||
| ;   -D WLED_TIMEZONE=2 | ||||
| ;   -D WLED_LAT=48.86 | ||||
| ;   -D WLED_LON=2.33 | ||||
| ; | ||||
| ; Use Watchdog timer with 10s guard | ||||
| ;   -D WLED_WATCHDOG_TIMEOUT=10 | ||||
| ; | ||||
| ; Create debug build (with remote debug) | ||||
| ;   -D WLED_DEBUG | ||||
| ;   -D WLED_DEBUG_HOST='"192.168.0.100"' | ||||
| ;   -D WLED_DEBUG_PORT=7868 | ||||
| ; | ||||
| ; Use Autosave usermod and set it to do save after 90s | ||||
| ;   -D USERMOD_AUTO_SAVE | ||||
| ;   -D AUTOSAVE_AFTER_SEC=90 | ||||
| ; | ||||
| ; Use 4 Line Display usermod with SPI display | ||||
| ;   -D USERMOD_FOUR_LINE_DISPLAY | ||||
| ;   -D USE_ALT_DISPlAY # mandatory | ||||
| ;   -DFLD_SPI_DEFAULT | ||||
| ;   -D FLD_TYPE=SSD1306_SPI64 | ||||
| ;   -D FLD_PIN_CLOCKSPI=14 | ||||
| ;   -D FLD_PIN_DATASPI=13 | ||||
| ;   -D FLD_PIN_DC=26 | ||||
| ;   -D FLD_PIN_CS=15 | ||||
| ;   -D FLD_PIN_RESET=27 | ||||
| ; | ||||
| ; Use Rotary encoder usermod (in conjunction with 4LD) | ||||
| ;   -D USERMOD_ROTARY_ENCODER_UI | ||||
| ;   -D ENCODER_DT_PIN=5 | ||||
| ;   -D ENCODER_CLK_PIN=18 | ||||
| ;   -D ENCODER_SW_PIN=19 | ||||
| ; | ||||
| ; Use Dallas DS18B20 temperature sensor usermod and configure it to use GPIO13 | ||||
| ;   -D USERMOD_DALLASTEMPERATURE | ||||
| ;   -D TEMPERATURE_PIN=13 | ||||
| ; | ||||
| ; Use Multi Relay usermod and configure it to use 6 relays and appropriate GPIO | ||||
| ;   -D USERMOD_MULTI_RELAY | ||||
| ;   -D MULTI_RELAY_MAX_RELAYS=6 | ||||
| ;   -D MULTI_RELAY_PINS=12,23,22,21,24,25 | ||||
| ; | ||||
| ; Use PIR sensor usermod and configure it to use GPIO4 and timer of 60s | ||||
| ;   -D USERMOD_PIRSWITCH | ||||
| ;   -D PIR_SENSOR_PIN=4 | ||||
| ;   -D PIR_SENSOR_OFF_SEC=60 | ||||
| ; | ||||
| ; Use Audioreactive usermod and configure I2S microphone | ||||
| ;   -D USERMOD_AUDIOREACTIVE | ||||
| ;   -D UM_AUDIOREACTIVE_USE_NEW_FFT | ||||
| ;   -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 | ||||
| ;   -D I2S_WSPIN=23 | ||||
| ;   -D I2S_CKPIN=19 | ||||
| ; | ||||
| ; Use PWM fan usermod | ||||
| ;   -D USERMOD_PWM_FAN | ||||
| ;   -D TACHO_PIN=33 | ||||
| ;   -D PWM_PIN=32 | ||||
| ; | ||||
| ; 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 | ||||
| ; or this for multiple outputs | ||||
| ;   -D PIXEL_COUNTS=30,30 | ||||
| ;    | ||||
| ; set milliampere limit when using ESP pin to power leds | ||||
| ;   -D ABL_MILLIAMPS_DEFAULT=850 | ||||
| ; | ||||
| ; 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 if a device (ESP) has one | ||||
| ;   -DBOARD_HAS_PSRAM | ||||
| ;   -D WLED_USE_PSRAM | ||||
| ; | ||||
| ; configure I2C and SPI interface (for various hardware) | ||||
| ;   -D I2CSDAPIN=33 # initialise interface | ||||
| ;   -D I2CSCLPIN=35 # initialise interface | ||||
| ;   -D HW_PIN_SCL=35 | ||||
| ;   -D HW_PIN_SDA=33 | ||||
| ;   -D HW_PIN_CLOCKSPI=7 | ||||
| ;   -D HW_PIN_DATASPI=11 | ||||
| ;   -D HW_PIN_MISOSPI=9 | ||||
|  | ||||
|  | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # PRE-CONFIGURED DEVELOPMENT BOARDS AND CONTROLLERS | ||||
| # ------------------------------------------------------------------------------ | ||||
|  | ||||
| [env:esp07] | ||||
| board = esp07 | ||||
| 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} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:d1_mini] | ||||
| board = d1_mini | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| upload_speed = 921600 | ||||
| board_build.ldscript = ${common.ldscript_4m1m} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
| monitor_filters = esp8266_exception_decoder | ||||
|  | ||||
| [env:heltec_wifi_kit_8] | ||||
| board = d1_mini | ||||
| 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} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:h803wf] | ||||
| board = d1_mini | ||||
| 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} -D LEDPIN=1 -D WLED_DISABLE_INFRARED | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:esp32dev_qio80] | ||||
| board = esp32dev | ||||
| platform = ${esp32.platform} | ||||
| platform_packages = ${esp32.platform_packages} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} -D WLED_RELEASE_NAME=ESP32_qio80 #-D WLED_DISABLE_BROWNOUT_DET | ||||
| lib_deps = ${esp32.lib_deps} | ||||
| monitor_filters = esp32_exception_decoder | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
| board_build.f_flash = 80000000L | ||||
| board_build.flash_mode = qio | ||||
|  | ||||
| [env:esp32dev_V4_dio80] | ||||
| ;; experimental ESP32 env using ESP-IDF V4.4.x | ||||
| ;; Warning: this build environment is not stable!! | ||||
| ;; please erase your device before installing. | ||||
| 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_RELEASE_NAME=ESP32_V4_qio80 #-D WLED_DISABLE_BROWNOUT_DET | ||||
| lib_deps = ${esp32_idf_V4.lib_deps} | ||||
| monitor_filters = esp32_exception_decoder | ||||
| board_build.partitions = ${esp32_idf_V4.default_partitions} | ||||
| board_build.f_flash = 80000000L | ||||
| board_build.flash_mode = dio | ||||
|  | ||||
| [env:esp32s2_saola] | ||||
| board = esp32-s2-saola-1 | ||||
| platform = https://github.com/tasmota/platform-espressif32/releases/download/v2.0.2.2/platform-tasmota-espressif32-2.0.2.zip | ||||
| 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} #-D WLED_RELEASE_NAME=S2_saola | ||||
|   ;-DLOLIN_WIFI_FIX ;; try this in case Wifi does not work | ||||
|   -DARDUINO_USB_CDC_ON_BOOT=1 | ||||
| lib_deps = ${esp32s2.lib_deps} | ||||
|  | ||||
| [env:esp32s3dev_8MB_PSRAM_qspi] | ||||
| ;; ESP32-TinyS3 development board, with 8MB FLASH and PSRAM (memory_type: qio_qspi) | ||||
| extends = env:esp32s3dev_8MB_PSRAM_opi | ||||
| ;board = um_tinys3 ;    -> needs workaround from https://github.com/Aircoookie/WLED/pull/2905#issuecomment-1328049860 | ||||
| board = esp32-s3-devkitc-1 ;; generic dev board; the next line adds PSRAM support | ||||
| board_build.arduino.memory_type = qio_qspi ;; use with PSRAM: 2MB or  4MB | ||||
|  | ||||
| [env:esp8285_4CH_MagicHome] | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_1m128k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_DISABLE_OTA | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:esp8285_H801] | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_1m128k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_DISABLE_OTA | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:d1_mini_5CH_Shojo_PCB] | ||||
| board = d1_mini | ||||
| 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} -D WLED_USE_SHOJO_PCB | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:d1_mini_debug] | ||||
| board = d1_mini | ||||
| build_type = debug | ||||
| 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} ${common.debug_flags} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:d1_mini_ota] | ||||
| board = d1_mini | ||||
| upload_protocol = espota | ||||
| # exchange for your WLED IP | ||||
| upload_port = "10.10.1.27" | ||||
| 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} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:anavi_miracle_controller] | ||||
| board = d1_mini | ||||
| 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} -D LEDPIN=12 -D IRPIN=-1 -D RLYPIN=2 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:esp32c3dev_2MB] | ||||
| ;; for ESP32-C3 boards with 2MB flash (instead of 4MB). | ||||
| ;; this board need a specific partition file. OTA not possible. | ||||
| extends = esp32c3 | ||||
| platform = ${esp32c3.platform} | ||||
| platform_packages = ${esp32c3.platform_packages} | ||||
| board = esp32-c3-devkitm-1 | ||||
| build_flags = ${common.build_flags} ${esp32c3.build_flags} #-D WLED_RELEASE_NAME=ESP32-C3 | ||||
|   -D WLED_WATCHDOG_TIMEOUT=0 | ||||
|   -D WLED_DISABLE_OTA | ||||
|   ; -DARDUINO_USB_CDC_ON_BOOT=1 ;; for virtual CDC USB | ||||
|   -DARDUINO_USB_CDC_ON_BOOT=0   ;; for serial-to-USB chip | ||||
| build_unflags = ${common.build_unflags} | ||||
| upload_speed = 115200 | ||||
| lib_deps = ${esp32c3.lib_deps} | ||||
| board_build.partitions = tools/WLED_ESP32_2MB_noOTA.csv | ||||
| board_build.flash_mode = dio | ||||
|  | ||||
| [env:wemos_shield_esp32] | ||||
| board = esp32dev | ||||
| platform = ${esp32.platform} | ||||
| platform_packages = ${esp32.platform_packages} | ||||
| upload_speed = 460800 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} | ||||
|   -D LEDPIN=16 | ||||
|   -D RLYPIN=19 | ||||
|   -D BTNPIN=17 | ||||
|   -D IRPIN=18 | ||||
|   -D UWLED_USE_MY_CONFIG | ||||
|   -D USERMOD_DALLASTEMPERATURE | ||||
|   -D USERMOD_FOUR_LINE_DISPLAY | ||||
|   -D TEMPERATURE_PIN=23 | ||||
|   -D USE_ALT_DISPlAY ; new versions of USERMOD_FOUR_LINE_DISPLAY and USERMOD_ROTARY_ENCODER_UI | ||||
|   -D USERMOD_AUDIOREACTIVE | ||||
| lib_deps = ${esp32.lib_deps} | ||||
|   OneWire@~2.3.5 | ||||
|   olikraus/U8g2 @ ^2.28.8 | ||||
|   https://github.com/blazoncek/arduinoFFT.git | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
|  | ||||
| [env:m5atom] | ||||
| board = esp32dev | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} -D LEDPIN=27 -D BTNPIN=39 | ||||
| lib_deps = ${esp32.lib_deps} | ||||
| platform = ${esp32.platform} | ||||
| platform_packages = ${esp32.platform_packages} | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
|  | ||||
| [env:sp501e] | ||||
| board = esp_wroom_02 | ||||
| platform = ${common.platform_wled_default} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_flags = ${common.build_flags_esp8266} -D LEDPIN=3 -D BTNPIN=1 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:sp511e] | ||||
| board = esp_wroom_02 | ||||
| platform = ${common.platform_wled_default} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_flags = ${common.build_flags_esp8266} -D LEDPIN=3 -D BTNPIN=2 -D IRPIN=5 -D WLED_MAX_BUTTONS=3 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:Athom_RGBCW]        ;7w and 5w(GU10) bulbs | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP8266 -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 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:Athom_15w_RGBCW]        ;15w bulb | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP8266 -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 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:Athom_3Pin_Controller]        ;small controller with only data | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP8266 -D BTNPIN=0 -D RLYPIN=-1 -D LEDPIN=1 -D WLED_DISABLE_INFRARED | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:Athom_4Pin_Controller]       ; With clock and data interface | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP8266 -D BTNPIN=0 -D RLYPIN=12 -D LEDPIN=1 -D WLED_DISABLE_INFRARED | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:Athom_5Pin_Controller]      ;Analog light strip controller | ||||
| board = esp8285 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP8266 -D BTNPIN=0 -D RLYPIN=-1 DATA_PINS=4,12,14,13 -D WLED_DISABLE_INFRARED | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:MY9291] | ||||
| board = esp01_1m | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_1m128k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} -D WLED_RELEASE_NAME=ESP01 -D WLED_DISABLE_OTA -D USERMOD_MY9291 | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # codm pixel controller board configurations | ||||
| # codm-controller-0_6 can also be used for the TYWE3S controller | ||||
| # ------------------------------------------------------------------------------ | ||||
|  | ||||
| [env:codm-controller-0_6] | ||||
| board = esp_wroom_02 | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| board_build.ldscript = ${common.ldscript_2m512k} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| [env:codm-controller-0_6-rev2] | ||||
| board = esp_wroom_02 | ||||
| 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} | ||||
| lib_deps = ${esp8266.lib_deps} | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # EleksTube-IPS | ||||
| # ------------------------------------------------------------------------------ | ||||
| [env:elekstube_ips] | ||||
| board = esp32dev | ||||
| platform = ${esp32.platform} | ||||
| platform_packages = ${esp32.platform_packages} | ||||
| upload_speed = 921600 | ||||
| build_flags = ${common.build_flags_esp32} -D WLED_DISABLE_BROWNOUT_DET -D WLED_DISABLE_INFRARED | ||||
|   -D USERMOD_RTC | ||||
|   -D USERMOD_ELEKSTUBE_IPS | ||||
|   -D LEDPIN=12 | ||||
|   -D RLYPIN=27 | ||||
|   -D BTNPIN=34 | ||||
|   -D DEFAULT_LED_COUNT=6 | ||||
|   # Display config | ||||
|   -D ST7789_DRIVER | ||||
|   -D TFT_WIDTH=135 | ||||
|   -D TFT_HEIGHT=240 | ||||
|   -D CGRAM_OFFSET | ||||
|   -D TFT_SDA_READ | ||||
|   -D TFT_MOSI=23 | ||||
|   -D TFT_SCLK=18 | ||||
|   -D TFT_DC=25 | ||||
|   -D TFT_RST=26 | ||||
|   -D SPI_FREQUENCY=40000000 | ||||
|   -D USER_SETUP_LOADED | ||||
| monitor_filters = esp32_exception_decoder | ||||
| lib_deps = | ||||
|   ${esp32.lib_deps} | ||||
|   TFT_eSPI @ ^2.3.70 | ||||
| board_build.partitions = ${esp32.default_partitions} | ||||
| @@ -1,6 +1,6 @@ | ||||
| # | ||||
| # This file is autogenerated by pip-compile with python 3.8 | ||||
| # To update, run: | ||||
| # This file is autogenerated by pip-compile with Python 3.12 | ||||
| # by the following command: | ||||
| # | ||||
| #    pip-compile | ||||
| # | ||||
| @@ -21,7 +21,9 @@ click==8.1.3 | ||||
|     #   platformio | ||||
|     #   uvicorn | ||||
| colorama==0.4.6 | ||||
|     # via platformio | ||||
|     # via | ||||
|     #   click | ||||
|     #   platformio | ||||
| h11==0.14.0 | ||||
|     # via | ||||
|     #   uvicorn | ||||
|   | ||||
							
								
								
									
										205
									
								
								tools/cdata-test.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										205
									
								
								tools/cdata-test.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,205 @@ | ||||
| 'use strict'; | ||||
|  | ||||
| const assert = require('node:assert'); | ||||
| const { describe, it, before, after } = require('node:test'); | ||||
| const fs = require('fs'); | ||||
| const path = require('path'); | ||||
| const child_process = require('child_process'); | ||||
| const util = require('util'); | ||||
| const execPromise = util.promisify(child_process.exec); | ||||
|  | ||||
| process.env.NODE_ENV = 'test'; // Set the environment to testing | ||||
| const cdata = require('./cdata.js'); | ||||
|  | ||||
| describe('Function', () => { | ||||
|   const testFolderPath = path.join(__dirname, 'testFolder'); | ||||
|   const oldFilePath = path.join(testFolderPath, 'oldFile.txt'); | ||||
|   const newFilePath = path.join(testFolderPath, 'newFile.txt'); | ||||
|  | ||||
|   // Create a temporary file before the test | ||||
|   before(() => { | ||||
|     // Create test folder | ||||
|     if (!fs.existsSync(testFolderPath)) { | ||||
|       fs.mkdirSync(testFolderPath); | ||||
|     } | ||||
|  | ||||
|     // Create an old file | ||||
|     fs.writeFileSync(oldFilePath, 'This is an old file.'); | ||||
|     // Modify the 'mtime' to simulate an old file | ||||
|     const oldTime = new Date(); | ||||
|     oldTime.setFullYear(oldTime.getFullYear() - 1); | ||||
|     fs.utimesSync(oldFilePath, oldTime, oldTime); | ||||
|  | ||||
|     // Create a new file | ||||
|     fs.writeFileSync(newFilePath, 'This is a new file.'); | ||||
|   }); | ||||
|  | ||||
|   // delete the temporary files after the test | ||||
|   after(() => { | ||||
|     fs.rmSync(testFolderPath, { recursive: true }); | ||||
|   }); | ||||
|  | ||||
|   describe('isFileNewerThan', async () => { | ||||
|     it('should return true if the file is newer than the provided time', async () => { | ||||
|       const pastTime = Date.now() - 10000; // 10 seconds ago | ||||
|       assert.strictEqual(cdata.isFileNewerThan(newFilePath, pastTime), true); | ||||
|     }); | ||||
|  | ||||
|     it('should return false if the file is older than the provided time', async () => { | ||||
|       assert.strictEqual(cdata.isFileNewerThan(oldFilePath, Date.now()), false); | ||||
|     }); | ||||
|  | ||||
|     it('should throw an exception if the file does not exist', async () => { | ||||
|       assert.throws(() => { | ||||
|         cdata.isFileNewerThan('nonexistent.txt', Date.now()); | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   describe('isAnyFileInFolderNewerThan', async () => { | ||||
|     it('should return true if a file in the folder is newer than the given time', async () => { | ||||
|       const time = fs.statSync(path.join(testFolderPath, 'oldFile.txt')).mtime; | ||||
|       assert.strictEqual(cdata.isAnyFileInFolderNewerThan(testFolderPath, time), true); | ||||
|     }); | ||||
|  | ||||
|     it('should return false if no files in the folder are newer than the given time', async () => { | ||||
|       assert.strictEqual(cdata.isAnyFileInFolderNewerThan(testFolderPath, new Date()), false); | ||||
|     }); | ||||
|  | ||||
|     it('should throw an exception if the folder does not exist', async () => { | ||||
|       assert.throws(() => { | ||||
|         cdata.isAnyFileInFolderNewerThan('nonexistent', new Date()); | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
|  | ||||
| describe('Script', () => { | ||||
|   const folderPath = 'wled00'; | ||||
|   const dataPath = path.join(folderPath, 'data'); | ||||
|  | ||||
|   before(() => { | ||||
|     process.env.NODE_ENV = 'production'; | ||||
|     // Backup files | ||||
|     fs.cpSync("wled00/data", "wled00Backup", { recursive: true }); | ||||
|     fs.cpSync("tools/cdata.js", "cdata.bak.js"); | ||||
|   }); | ||||
|   after(() => { | ||||
|     // Restore backup | ||||
|     fs.rmSync("wled00/data", { recursive: true }); | ||||
|     fs.renameSync("wled00Backup", "wled00/data"); | ||||
|     fs.rmSync("tools/cdata.js"); | ||||
|     fs.renameSync("cdata.bak.js", "tools/cdata.js"); | ||||
|   }); | ||||
|  | ||||
|   // delete all html_*.h files | ||||
|   async function deleteBuiltFiles() { | ||||
|     const files = await fs.promises.readdir(folderPath); | ||||
|     await Promise.all(files.map(file => { | ||||
|       if (file.startsWith('html_') && path.extname(file) === '.h') { | ||||
|         return fs.promises.unlink(path.join(folderPath, file)); | ||||
|       } | ||||
|     })); | ||||
|   } | ||||
|  | ||||
|   // check if html_*.h files were created | ||||
|   async function checkIfBuiltFilesExist() { | ||||
|     const files = await fs.promises.readdir(folderPath); | ||||
|     const htmlFiles = files.filter(file => file.startsWith('html_') && path.extname(file) === '.h'); | ||||
|     assert(htmlFiles.length > 0, 'html_*.h files were not created'); | ||||
|   } | ||||
|  | ||||
|   async function runAndCheckIfBuiltFilesExist() { | ||||
|     await execPromise('node tools/cdata.js'); | ||||
|     await checkIfBuiltFilesExist(); | ||||
|   } | ||||
|  | ||||
|   async function checkIfFileWasNewlyCreated(file) { | ||||
|     const modifiedTime = fs.statSync(file).mtimeMs; | ||||
|     assert(Date.now() - modifiedTime < 500, file + ' was not modified'); | ||||
|   } | ||||
|  | ||||
|   async function testFileModification(sourceFilePath, resultFile) { | ||||
|     // run cdata.js to ensure html_*.h files are created | ||||
|     await execPromise('node tools/cdata.js'); | ||||
|  | ||||
|     // modify file | ||||
|     fs.appendFileSync(sourceFilePath, ' '); | ||||
|     // delay for 1 second to ensure the modified time is different | ||||
|     await new Promise(resolve => setTimeout(resolve, 1000)); | ||||
|  | ||||
|     // run script cdata.js again and wait for it to finish | ||||
|     await execPromise('node tools/cdata.js'); | ||||
|  | ||||
|     checkIfFileWasNewlyCreated(path.join(folderPath, resultFile)); | ||||
|   } | ||||
|  | ||||
|   describe('should build if', () => { | ||||
|     it('html_*.h files are missing', async () => { | ||||
|       await deleteBuiltFiles(); | ||||
|       await runAndCheckIfBuiltFilesExist(); | ||||
|     }); | ||||
|  | ||||
|     it('only one html_*.h file is missing', async () => { | ||||
|       // run script cdata.js and wait for it to finish | ||||
|       await execPromise('node tools/cdata.js'); | ||||
|  | ||||
|       // delete a random html_*.h file | ||||
|       let files = await fs.promises.readdir(folderPath); | ||||
|       let htmlFiles = files.filter(file => file.startsWith('html_') && path.extname(file) === '.h'); | ||||
|       const randomFile = htmlFiles[Math.floor(Math.random() * htmlFiles.length)]; | ||||
|       await fs.promises.unlink(path.join(folderPath, randomFile)); | ||||
|  | ||||
|       await runAndCheckIfBuiltFilesExist(); | ||||
|     }); | ||||
|  | ||||
|     it('script was executed with -f or --force', async () => { | ||||
|       await execPromise('node tools/cdata.js'); | ||||
|       await new Promise(resolve => setTimeout(resolve, 1000)); | ||||
|       await execPromise('node tools/cdata.js --force'); | ||||
|       await checkIfFileWasNewlyCreated(path.join(folderPath, 'html_ui.h')); | ||||
|       await new Promise(resolve => setTimeout(resolve, 1000)); | ||||
|       await execPromise('node tools/cdata.js -f'); | ||||
|       await checkIfFileWasNewlyCreated(path.join(folderPath, 'html_ui.h')); | ||||
|     }); | ||||
|  | ||||
|     it('a file changes', async () => { | ||||
|       await testFileModification(path.join(dataPath, 'index.htm'), 'html_ui.h'); | ||||
|     }); | ||||
|  | ||||
|     it('a inlined file changes', async () => { | ||||
|       await testFileModification(path.join(dataPath, 'index.js'), 'html_ui.h'); | ||||
|     }); | ||||
|  | ||||
|     it('a settings file changes', async () => { | ||||
|       await testFileModification(path.join(dataPath, 'settings_leds.htm'), 'html_ui.h'); | ||||
|     }); | ||||
|  | ||||
|     it('the favicon changes', async () => { | ||||
|       await testFileModification(path.join(dataPath, 'favicon.ico'), 'html_ui.h'); | ||||
|     }); | ||||
|  | ||||
|     it('cdata.js changes', async () => { | ||||
|       await testFileModification('tools/cdata.js', 'html_ui.h'); | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   describe('should not build if', () => { | ||||
|     it('the files are already built', async () => { | ||||
|       await deleteBuiltFiles(); | ||||
|  | ||||
|       // run script cdata.js and wait for it to finish | ||||
|       let startTime = Date.now(); | ||||
|       await execPromise('node tools/cdata.js'); | ||||
|       const firstRunTime = Date.now() - startTime; | ||||
|  | ||||
|       // run script cdata.js and wait for it to finish | ||||
|       startTime = Date.now(); | ||||
|       await execPromise('node tools/cdata.js'); | ||||
|       const secondRunTime = Date.now() - startTime; | ||||
|  | ||||
|       // check if second run was faster than the first (must be at least 2x faster) | ||||
|       assert(secondRunTime < firstRunTime / 2, 'html_*.h files were rebuilt'); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
							
								
								
									
										341
									
								
								tools/cdata.js
									
									
									
									
									
								
							
							
						
						
									
										341
									
								
								tools/cdata.js
									
									
									
									
									
								
							| @@ -16,25 +16,57 @@ | ||||
|  */ | ||||
|  | ||||
| const fs = require("fs"); | ||||
| const path = require("path"); | ||||
| const inliner = require("inliner"); | ||||
| const zlib = require("zlib"); | ||||
| const CleanCSS = require("clean-css"); | ||||
| const MinifyHTML = require("html-minifier-terser").minify; | ||||
| const minifyHtml = require("html-minifier-terser").minify; | ||||
| const packageJson = require("../package.json"); | ||||
|  | ||||
| /** | ||||
|  * | ||||
| // Export functions for testing | ||||
| module.exports = { isFileNewerThan, isAnyFileInFolderNewerThan }; | ||||
|  | ||||
| const output = ["wled00/html_ui.h", "wled00/html_pixart.h", "wled00/html_cpal.h", "wled00/html_pxmagic.h", "wled00/html_settings.h", "wled00/html_other.h"] | ||||
|  | ||||
| // \x1b[34m is blue, \x1b[36m is cyan, \x1b[0m is reset | ||||
| const wledBanner = ` | ||||
| \t\x1b[34m  ##  ##      ##        ######    ###### | ||||
| \t\x1b[34m##      ##    ##      ##        ##  ##  ## | ||||
| \t\x1b[34m##  ##  ##  ##        ######        ##  ## | ||||
| \t\x1b[34m##  ##  ##  ##        ##            ##  ## | ||||
| \t\x1b[34m  ##  ##      ######    ######    ###### | ||||
| \t\t\x1b[36m build script for web UI | ||||
| \x1b[0m`; | ||||
|  | ||||
| const singleHeader = `/* | ||||
|  * Binary array for the Web UI. | ||||
|  * gzip is used for smaller size and improved speeds. | ||||
|  *  | ||||
|  * Please see https://kno.wled.ge/advanced/custom-features/#changing-web-ui | ||||
|  * to find out how to easily modify the web UI source! | ||||
|  */ | ||||
| function hexdump(buffer,isHex=false) { | ||||
|   | ||||
| `; | ||||
|  | ||||
| const multiHeader = `/* | ||||
|  * More web UI HTML source arrays. | ||||
|  * This file is auto generated, please don't make any changes manually. | ||||
|  * | ||||
|  * Instead, see https://kno.wled.ge/advanced/custom-features/#changing-web-ui | ||||
|  * to find out how to easily modify the web UI source! | ||||
|  */ | ||||
| `; | ||||
|  | ||||
| function hexdump(buffer, isHex = false) { | ||||
|   let lines = []; | ||||
|  | ||||
|   for (let i = 0; i < buffer.length; i +=(isHex?32:16)) { | ||||
|   for (let i = 0; i < buffer.length; i += (isHex ? 32 : 16)) { | ||||
|     var block; | ||||
|     let hexArray = []; | ||||
|     if (isHex) { | ||||
|       block = buffer.slice(i, i + 32) | ||||
|       for (let j = 0; j < block.length; j +=2 ) { | ||||
|         hexArray.push("0x" + block.slice(j,j+2)) | ||||
|       for (let j = 0; j < block.length; j += 2) { | ||||
|         hexArray.push("0x" + block.slice(j, j + 2)) | ||||
|       } | ||||
|     } else { | ||||
|       block = buffer.slice(i, i + 16); // cut buffer into blocks of 16 | ||||
| @@ -51,204 +83,163 @@ function hexdump(buffer,isHex=false) { | ||||
|   return lines.join(",\n"); | ||||
| } | ||||
|  | ||||
| function strReplace(str, search, replacement) { | ||||
|   return str.split(search).join(replacement); | ||||
| } | ||||
|  | ||||
| function adoptVersionAndRepo(html) { | ||||
|   let repoUrl = packageJson.repository ? packageJson.repository.url : undefined; | ||||
|   if (repoUrl) { | ||||
|     repoUrl = repoUrl.replace(/^git\+/, ""); | ||||
|     repoUrl = repoUrl.replace(/\.git$/, ""); | ||||
|     // Replace we | ||||
|     html = strReplace(html, "https://github.com/atuline/WLED", repoUrl); | ||||
|     html = strReplace(html, "https://github.com/Aircoookie/WLED", repoUrl); | ||||
|     html = html.replaceAll("https://github.com/atuline/WLED", repoUrl); | ||||
|     html = html.replaceAll("https://github.com/Aircoookie/WLED", repoUrl); | ||||
|   } | ||||
|   let version = packageJson.version; | ||||
|   if (version) { | ||||
|     html = strReplace(html, "##VERSION##", version); | ||||
|     html = html.replaceAll("##VERSION##", version); | ||||
|   } | ||||
|   return html; | ||||
| } | ||||
|  | ||||
| function filter(str, type) { | ||||
|   str = adoptVersionAndRepo(str); | ||||
|   if (type === undefined) { | ||||
| async function minify(str, type = "plain") { | ||||
|   const options = { | ||||
|     collapseWhitespace: true, | ||||
|     collapseBooleanAttributes: true, | ||||
|     collapseInlineTagWhitespace: true, | ||||
|     minifyCSS: true, | ||||
|     minifyJS: true, | ||||
|     removeAttributeQuotes: true, | ||||
|     removeComments: true, | ||||
|     sortAttributes: true, | ||||
|     sortClassName: true, | ||||
|   }; | ||||
|  | ||||
|   if (type == "plain") { | ||||
|     return str; | ||||
|   } else if (type == "css-minify") { | ||||
|     return new CleanCSS({}).minify(str).styles; | ||||
|   } else if (type == "js-minify") { | ||||
|     return MinifyHTML('<script>' + str + '</script>', { | ||||
|       collapseWhitespace: true, | ||||
|       minifyJS: true,  | ||||
|       continueOnParseError: false, | ||||
|       removeComments: true, | ||||
|     }).replace(/<[\/]*script>/g,''); | ||||
|     return await minifyHtml('<script>' + str + '</script>', options).replace(/<[\/]*script>/g, ''); | ||||
|   } else if (type == "html-minify") { | ||||
|     return MinifyHTML(str, { | ||||
|       collapseWhitespace: true, | ||||
|       maxLineLength: 80, | ||||
|       minifyCSS: true, | ||||
|       minifyJS: true,  | ||||
|       continueOnParseError: false, | ||||
|       removeComments: true, | ||||
|     }); | ||||
|   } else if (type == "html-minify-ui") { | ||||
|     return MinifyHTML(str, { | ||||
|       collapseWhitespace: true, | ||||
|       conservativeCollapse: true, | ||||
|       maxLineLength: 80, | ||||
|       minifyCSS: true, | ||||
|       minifyJS: true,  | ||||
|       continueOnParseError: false, | ||||
|       removeComments: true, | ||||
|     }); | ||||
|   } else { | ||||
|     console.warn("Unknown filter: " + type); | ||||
|     return str; | ||||
|     return await minifyHtml(str, options); | ||||
|   } | ||||
|  | ||||
|   throw new Error("Unknown filter: " + type); | ||||
| } | ||||
|  | ||||
| function writeHtmlGzipped(sourceFile, resultFile, page) { | ||||
| async function writeHtmlGzipped(sourceFile, resultFile, page) { | ||||
|   console.info("Reading " + sourceFile); | ||||
|   new inliner(sourceFile, function (error, html) { | ||||
|     console.info("Inlined " + html.length + " characters"); | ||||
|     html = filter(html, "html-minify-ui"); | ||||
|     console.info("Minified to " + html.length + " characters"); | ||||
|  | ||||
|     if (error) { | ||||
|       console.warn(error); | ||||
|       throw error; | ||||
|     } | ||||
|   new inliner(sourceFile, async function (error, html) { | ||||
|     if (error) throw error; | ||||
|  | ||||
|     html = adoptVersionAndRepo(html); | ||||
|     zlib.gzip(html, { level: zlib.constants.Z_BEST_COMPRESSION }, function (error, result) { | ||||
|       if (error) { | ||||
|         console.warn(error); | ||||
|         throw error; | ||||
|       } | ||||
|  | ||||
|       console.info("Compressed " + result.length + " bytes"); | ||||
|       const array = hexdump(result); | ||||
|       const src = `/* | ||||
|  * Binary array for the Web UI. | ||||
|  * gzip is used for smaller size and improved speeds. | ||||
|  *  | ||||
|  * Please see https://kno.wled.ge/advanced/custom-features/#changing-web-ui | ||||
|  * to find out how to easily modify the web UI source! | ||||
|  */ | ||||
|   | ||||
| // Autogenerated from ${sourceFile}, do not edit!! | ||||
| const uint16_t PAGE_${page}_L = ${result.length}; | ||||
| const uint8_t PAGE_${page}[] PROGMEM = { | ||||
| ${array} | ||||
| }; | ||||
| `; | ||||
|       console.info("Writing " + resultFile); | ||||
|       fs.writeFileSync(resultFile, src); | ||||
|     }); | ||||
|     const originalLength = html.length; | ||||
|     html = await minify(html, "html-minify"); | ||||
|     const result = zlib.gzipSync(html, { level: zlib.constants.Z_BEST_COMPRESSION }); | ||||
|     console.info("Minified and compressed " + sourceFile + " from " + originalLength + " to " + result.length + " bytes"); | ||||
|     const array = hexdump(result); | ||||
|     let src = singleHeader; | ||||
|     src += `const uint16_t PAGE_${page}_L = ${result.length};\n`; | ||||
|     src += `const uint8_t PAGE_${page}[] PROGMEM = {\n${array}\n};\n\n`; | ||||
|     console.info("Writing " + resultFile); | ||||
|     fs.writeFileSync(resultFile, src); | ||||
|   }); | ||||
| } | ||||
|  | ||||
| function specToChunk(srcDir, s) { | ||||
|   if (s.method == "plaintext") { | ||||
|     const buf = fs.readFileSync(srcDir + "/" + s.file); | ||||
|     const str = buf.toString("utf-8"); | ||||
|     const chunk = ` | ||||
| // Autogenerated from ${srcDir}/${s.file}, do not edit!! | ||||
| const char ${s.name}[] PROGMEM = R"${s.prepend || ""}${filter(str, s.filter)}${ | ||||
|       s.append || "" | ||||
|     }"; | ||||
| async function specToChunk(srcDir, s) { | ||||
|   const buf = fs.readFileSync(srcDir + "/" + s.file); | ||||
|   let chunk = `\n// Autogenerated from ${srcDir}/${s.file}, do not edit!!\n` | ||||
|  | ||||
| `; | ||||
|     return s.mangle ? s.mangle(chunk) : chunk; | ||||
|   } else if (s.method == "gzip") { | ||||
|     const buf = fs.readFileSync(srcDir + "/" + s.file); | ||||
|     var str = buf.toString('utf-8'); | ||||
|     if (s.mangle) str = s.mangle(str); | ||||
|     const zip = zlib.gzipSync(filter(str, s.filter), { level: zlib.constants.Z_BEST_COMPRESSION }); | ||||
|     const result = hexdump(zip.toString('hex'), true); | ||||
|     const chunk = ` | ||||
| // Autogenerated from ${srcDir}/${s.file}, do not edit!! | ||||
| const uint16_t ${s.name}_length = ${zip.length}; | ||||
| const uint8_t ${s.name}[] PROGMEM = { | ||||
| ${result} | ||||
| }; | ||||
|  | ||||
| `; | ||||
|     return chunk; | ||||
|   } else if (s.method == "binary") { | ||||
|     const buf = fs.readFileSync(srcDir + "/" + s.file); | ||||
|     const result = hexdump(buf); | ||||
|     const chunk = ` | ||||
| // Autogenerated from ${srcDir}/${s.file}, do not edit!! | ||||
| const uint16_t ${s.name}_length = ${result.length}; | ||||
| const uint8_t ${s.name}[] PROGMEM = { | ||||
| ${result} | ||||
| }; | ||||
|  | ||||
| `; | ||||
|     return chunk; | ||||
|   } else { | ||||
|     console.warn("Unknown method: " + s.method); | ||||
|     return undefined; | ||||
|   } | ||||
| } | ||||
|  | ||||
| function writeChunks(srcDir, specs, resultFile) { | ||||
|   let src = `/* | ||||
|  * More web UI HTML source arrays. | ||||
|  * This file is auto generated, please don't make any changes manually. | ||||
|  * Instead, see https://kno.wled.ge/advanced/custom-features/#changing-web-ui | ||||
|  * to find out how to easily modify the web UI source! | ||||
|  */  | ||||
| `; | ||||
|   specs.forEach((s) => { | ||||
|     try { | ||||
|       console.info("Reading " + srcDir + "/" + s.file + " as " + s.name); | ||||
|       src += specToChunk(srcDir, s); | ||||
|     } catch (e) { | ||||
|       console.warn( | ||||
|         "Failed " + s.name + " from " + srcDir + "/" + s.file, | ||||
|         e.message.length > 60 ? e.message.substring(0, 60) : e.message | ||||
|       ); | ||||
|   if (s.method == "plaintext" || s.method == "gzip") { | ||||
|     let str = buf.toString("utf-8"); | ||||
|     str = adoptVersionAndRepo(str); | ||||
|     const originalLength = str.length; | ||||
|     if (s.method == "gzip") { | ||||
|       if (s.mangle) str = s.mangle(str); | ||||
|       const zip = zlib.gzipSync(await minify(str, s.filter), { level: zlib.constants.Z_BEST_COMPRESSION }); | ||||
|       console.info("Minified and compressed " + s.file + " from " + originalLength + " to " + zip.length + " bytes"); | ||||
|       const result = hexdump(zip); | ||||
|       chunk += `const uint16_t ${s.name}_length = ${zip.length};\n`; | ||||
|       chunk += `const uint8_t ${s.name}[] PROGMEM = {\n${result}\n};\n\n`; | ||||
|       return chunk; | ||||
|     } else { | ||||
|       const minified = await minify(str, s.filter); | ||||
|       console.info("Minified " + s.file + " from " + originalLength + " to " + minified.length + " bytes"); | ||||
|       chunk += `const char ${s.name}[] PROGMEM = R"${s.prepend || ""}${minified}${s.append || ""}";\n\n`; | ||||
|       return s.mangle ? s.mangle(chunk) : chunk; | ||||
|     } | ||||
|   }); | ||||
|   } else if (s.method == "binary") { | ||||
|     const result = hexdump(buf); | ||||
|     chunk += `const uint16_t ${s.name}_length = ${buf.length};\n`; | ||||
|     chunk += `const uint8_t ${s.name}[] PROGMEM = {\n${result}\n};\n\n`; | ||||
|     return chunk; | ||||
|   } | ||||
|  | ||||
|   throw new Error("Unknown method: " + s.method); | ||||
| } | ||||
|  | ||||
| async function writeChunks(srcDir, specs, resultFile) { | ||||
|   let src = multiHeader; | ||||
|   for (const s of specs) { | ||||
|     console.info("Reading " + srcDir + "/" + s.file + " as " + s.name); | ||||
|     src += await specToChunk(srcDir, s); | ||||
|   } | ||||
|   console.info("Writing " + src.length + " characters into " + resultFile); | ||||
|   fs.writeFileSync(resultFile, src); | ||||
| } | ||||
|  | ||||
| // Check if a file is newer than a given time | ||||
| function isFileNewerThan(filePath, time) { | ||||
|   const stats = fs.statSync(filePath); | ||||
|   return stats.mtimeMs > time; | ||||
| } | ||||
|  | ||||
| // Check if any file in a folder (or its subfolders) is newer than a given time | ||||
| function isAnyFileInFolderNewerThan(folderPath, time) { | ||||
|   const files = fs.readdirSync(folderPath, { withFileTypes: true }); | ||||
|   for (const file of files) { | ||||
|     const filePath = path.join(folderPath, file.name); | ||||
|     if (isFileNewerThan(filePath, time)) { | ||||
|       return true; | ||||
|     } | ||||
|     if (file.isDirectory() && isAnyFileInFolderNewerThan(filePath, time)) { | ||||
|       return true; | ||||
|     } | ||||
|   } | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| // Check if the web UI is already built | ||||
| function isAlreadyBuilt(folderPath) { | ||||
|   let lastBuildTime = Infinity; | ||||
|  | ||||
|   for (const file of output) { | ||||
|     try { | ||||
|       lastBuildTime = Math.min(lastBuildTime, fs.statSync(file).mtimeMs); | ||||
|     } catch (e) { | ||||
|       if (e.code !== 'ENOENT') throw e; | ||||
|       console.info("File " + file + " does not exist. Rebuilding..."); | ||||
|       return false; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return !isAnyFileInFolderNewerThan(folderPath, lastBuildTime) && !isFileNewerThan("tools/cdata.js", lastBuildTime); | ||||
| } | ||||
|  | ||||
| // Don't run this script if we're in a test environment | ||||
| if (process.env.NODE_ENV === 'test') { | ||||
|   return; | ||||
| } | ||||
|  | ||||
| console.info(wledBanner); | ||||
|  | ||||
| if (isAlreadyBuilt("wled00/data") && process.argv[2] !== '--force' && process.argv[2] !== '-f') { | ||||
|   console.info("Web UI is already built"); | ||||
|   return; | ||||
| } | ||||
|  | ||||
| writeHtmlGzipped("wled00/data/index.htm", "wled00/html_ui.h", 'index'); | ||||
| writeHtmlGzipped("wled00/data/simple.htm", "wled00/html_simple.h", 'simple'); | ||||
| writeHtmlGzipped("wled00/data/pixart/pixart.htm", "wled00/html_pixart.h", 'pixart'); | ||||
| writeHtmlGzipped("wled00/data/cpal/cpal.htm", "wled00/html_cpal.h", 'cpal'); | ||||
| writeHtmlGzipped("wled00/data/pxmagic/pxmagic.htm", "wled00/html_pxmagic.h", 'pxmagic'); | ||||
| /* | ||||
| writeChunks( | ||||
|   "wled00/data", | ||||
|   [ | ||||
|     { | ||||
|       file: "simple.css", | ||||
|       name: "PAGE_simpleCss", | ||||
|       method: "gzip", | ||||
|       filter: "css-minify", | ||||
|     }, | ||||
|     { | ||||
|       file: "simple.js", | ||||
|       name: "PAGE_simpleJs", | ||||
|       method: "gzip", | ||||
|       filter: "js-minify", | ||||
|     }, | ||||
|     { | ||||
|       file: "simple.htm", | ||||
|       name: "PAGE_simple", | ||||
|       method: "gzip", | ||||
|       filter: "html-minify-ui", | ||||
|     } | ||||
|   ], | ||||
|   "wled00/html_simplex.h" | ||||
| ); | ||||
| */ | ||||
|  | ||||
| writeChunks( | ||||
|   "wled00/data", | ||||
|   [ | ||||
| @@ -259,7 +250,7 @@ writeChunks( | ||||
|       filter: "css-minify", | ||||
|       mangle: (str) => | ||||
|         str | ||||
|           .replace("%%","%") | ||||
|           .replace("%%", "%") | ||||
|     }, | ||||
|     { | ||||
|       file: "settings.htm", | ||||
| @@ -406,16 +397,6 @@ const char PAGE_dmxmap[] PROGMEM = R"=====()====="; | ||||
|       file: "favicon.ico", | ||||
|       name: "favicon", | ||||
|       method: "binary", | ||||
|     }, | ||||
|     { | ||||
|       file: "iro.js", | ||||
|       name: "iroJs", | ||||
|       method: "gzip" | ||||
|     }, | ||||
|     { | ||||
|       file: "rangetouch.js", | ||||
|       name: "rangetouchJs", | ||||
|       method: "gzip" | ||||
|     } | ||||
|   ], | ||||
|   "wled00/html_other.h" | ||||
|   | ||||
| @@ -86,7 +86,7 @@ private: | ||||
|      | ||||
|     StaticJsonDocument<600> doc; | ||||
|      | ||||
|     doc[F("name")] = String(serverDescription) + F(" ") + name; | ||||
|     doc[F("name")] = String(serverDescription) + " " + name; | ||||
|     doc[F("state_topic")] = topic; | ||||
|     doc[F("unique_id")] = String(mqttClientID) + name; | ||||
|     if (unitOfMeasurement != "") | ||||
| @@ -98,8 +98,8 @@ private: | ||||
|     JsonObject device = doc.createNestedObject(F("device")); // attach the sensor to the same device | ||||
|     device[F("name")] = serverDescription; | ||||
|     device[F("identifiers")] = "wled-sensor-" + String(mqttClientID); | ||||
|     device[F("manufacturer")] = F("WLED"); | ||||
|     device[F("model")] = F("FOSS"); | ||||
|     device[F("manufacturer")] = F(WLED_BRAND); | ||||
|     device[F("model")] = F(WLED_PRODUCT_NAME); | ||||
|     device[F("sw_version")] = versionString; | ||||
|  | ||||
|     String temp; | ||||
|   | ||||
| @@ -160,8 +160,8 @@ private: | ||||
|     JsonObject device = doc.createNestedObject(F("device")); // attach the sensor to the same device | ||||
|     device[F("name")] = serverDescription; | ||||
|     device[F("identifiers")] = "wled-sensor-" + String(mqttClientID); | ||||
|     device[F("manufacturer")] = F("WLED"); | ||||
|     device[F("model")] = F("FOSS"); | ||||
|     device[F("manufacturer")] = F(WLED_BRAND); | ||||
|     device[F("model")] = F(WLED_PRODUCT_NAME); | ||||
|     device[F("sw_version")] = versionString; | ||||
|  | ||||
|     String temp; | ||||
|   | ||||
| @@ -114,7 +114,7 @@ class UsermodCronixie : public Usermod { | ||||
|       //W Week of Month | WW Week of Year | ||||
|       //D Day of Week | DD Day Of Month | DDD Day Of Year | ||||
|  | ||||
|       DEBUG_PRINT("cset "); | ||||
|       DEBUG_PRINT(F("cset ")); | ||||
|       DEBUG_PRINTLN(cronixieDisplay); | ||||
|        | ||||
|       for (int i = 0; i < 6; i++) | ||||
| @@ -160,7 +160,7 @@ class UsermodCronixie : public Usermod { | ||||
|           //case 'v': break; //user var1 | ||||
|         } | ||||
|       } | ||||
|       DEBUG_PRINT("result "); | ||||
|       DEBUG_PRINT(F("result ")); | ||||
|       for (int i = 0; i < 5; i++) | ||||
|       { | ||||
|         DEBUG_PRINT((int)dP[i]); | ||||
|   | ||||
| @@ -87,7 +87,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      * readFromConfig() is called prior to setup() | ||||
|      * You can use it to initialize variables, sensors or similar. | ||||
|      */ | ||||
|     void setup() { | ||||
|     void setup() override { | ||||
|       // do your set-up here | ||||
|       //Serial.println("Hello from my usermod!"); | ||||
|       initDone = true; | ||||
| @@ -98,7 +98,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      * connected() is called every time the WiFi is (re)connected | ||||
|      * Use it to initialize network interfaces | ||||
|      */ | ||||
|     void connected() { | ||||
|     void connected() override { | ||||
|       //Serial.println("Connected to WiFi!"); | ||||
|     } | ||||
|  | ||||
| @@ -113,7 +113,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      * 2. Try to avoid using the delay() function. NEVER use delays longer than 10 milliseconds. | ||||
|      *    Instead, use a timer check as shown here. | ||||
|      */ | ||||
|     void loop() { | ||||
|     void loop() override { | ||||
|       // if usermod is disabled or called during strip updating just exit | ||||
|       // NOTE: on very long strips strip.isUpdating() may always return true so update accordingly | ||||
|       if (!enabled || strip.isUpdating()) return; | ||||
| @@ -131,7 +131,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      * Creating an "u" object allows you to add custom key/value pairs to the Info section of the WLED web UI. | ||||
|      * Below it is shown how this could be used for e.g. a light sensor | ||||
|      */ | ||||
|     void addToJsonInfo(JsonObject& root) | ||||
|     void addToJsonInfo(JsonObject& root) override | ||||
|     { | ||||
|       // if "u" object does not exist yet wee need to create it | ||||
|       JsonObject user = root["u"]; | ||||
| @@ -156,7 +156,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      * addToJsonState() can be used to add custom entries to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void addToJsonState(JsonObject& root) | ||||
|     void addToJsonState(JsonObject& root) override | ||||
|     { | ||||
|       if (!initDone || !enabled) return;  // prevent crash on boot applyPreset() | ||||
|  | ||||
| @@ -171,7 +171,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void readFromJsonState(JsonObject& root) | ||||
|     void readFromJsonState(JsonObject& root) override | ||||
|     { | ||||
|       if (!initDone) return;  // prevent crash on boot applyPreset() | ||||
|  | ||||
| @@ -220,7 +220,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      *  | ||||
|      * I highly recommend checking out the basics of ArduinoJson serialization and deserialization in order to use custom settings! | ||||
|      */ | ||||
|     void addToConfig(JsonObject& root) | ||||
|     void addToConfig(JsonObject& root) override | ||||
|     { | ||||
|       JsonObject top = root.createNestedObject(FPSTR(_name)); | ||||
|       top[FPSTR(_enabled)] = enabled; | ||||
| @@ -253,7 +253,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      *  | ||||
|      * This function is guaranteed to be called on boot, but could also be called every time settings are updated | ||||
|      */ | ||||
|     bool readFromConfig(JsonObject& root) | ||||
|     bool readFromConfig(JsonObject& root) override | ||||
|     { | ||||
|       // default settings values could be set here (or below using the 3-argument getJsonValue()) instead of in the class definition or constructor | ||||
|       // setting them inside readFromConfig() is slightly more robust, handling the rare but plausible use case of single value being missing after boot (e.g. if the cfg.json was manually edited and a value was removed) | ||||
| @@ -285,7 +285,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      * it may add additional metadata for certain entry fields (adding drop down is possible) | ||||
|      * be careful not to add too much as oappend() buffer is limited to 3k | ||||
|      */ | ||||
|     void appendConfigData() | ||||
|     void appendConfigData() override | ||||
|     { | ||||
|       oappend(SET_F("addInfo('")); oappend(String(FPSTR(_name)).c_str()); oappend(SET_F(":great")); oappend(SET_F("',1,'<i>(this is a great config value)</i>');")); | ||||
|       oappend(SET_F("addInfo('")); oappend(String(FPSTR(_name)).c_str()); oappend(SET_F(":testString")); oappend(SET_F("',1,'enter any string you want');")); | ||||
| @@ -300,7 +300,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      * Use this to blank out some LEDs or set them to a different color regardless of the set effect mode. | ||||
|      * Commonly used for custom clocks (Cronixie, 7 segment) | ||||
|      */ | ||||
|     void handleOverlayDraw() | ||||
|     void handleOverlayDraw() override | ||||
|     { | ||||
|       //strip.setPixelColor(0, RGBW32(0,0,0,0)) // set the first pixel to black | ||||
|     } | ||||
| @@ -311,7 +311,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      * will prevent button working in a default way. | ||||
|      * Replicating button.cpp | ||||
|      */ | ||||
|     bool handleButton(uint8_t b) { | ||||
|     bool handleButton(uint8_t b) override { | ||||
|       yield(); | ||||
|       // ignore certain button types as they may have other consequences | ||||
|       if (!enabled | ||||
| @@ -334,7 +334,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      * handling of MQTT message | ||||
|      * topic only contains stripped topic (part after /wled/MAC) | ||||
|      */ | ||||
|     bool onMqttMessage(char* topic, char* payload) { | ||||
|     bool onMqttMessage(char* topic, char* payload) override { | ||||
|       // check if we received a command | ||||
|       //if (strlen(topic) == 8 && strncmp_P(topic, PSTR("/command"), 8) == 0) { | ||||
|       //  String action = payload; | ||||
| @@ -355,7 +355,7 @@ class MyExampleUsermod : public Usermod { | ||||
|     /** | ||||
|      * onMqttConnect() is called when MQTT connection is established | ||||
|      */ | ||||
|     void onMqttConnect(bool sessionPresent) { | ||||
|     void onMqttConnect(bool sessionPresent) override { | ||||
|       // do any MQTT related initialisation here | ||||
|       //publishMqtt("I am alive!"); | ||||
|     } | ||||
| @@ -366,7 +366,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      * onStateChanged() is used to detect WLED state change | ||||
|      * @mode parameter is CALL_MODE_... parameter used for notifications | ||||
|      */ | ||||
|     void onStateChange(uint8_t mode) { | ||||
|     void onStateChange(uint8_t mode) override { | ||||
|       // do something if WLED state changed (color, brightness, effect, preset, etc) | ||||
|     } | ||||
|  | ||||
| @@ -375,7 +375,7 @@ class MyExampleUsermod : public Usermod { | ||||
|      * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
|      * This could be used in the future for the system to determine whether your usermod is installed. | ||||
|      */ | ||||
|     uint16_t getId() | ||||
|     uint16_t getId() override | ||||
|     { | ||||
|       return USERMOD_ID_EXAMPLE; | ||||
|     } | ||||
|   | ||||
| @@ -75,6 +75,9 @@ Usermod can be configured via the Usermods settings page. | ||||
| * `mqtt-only` - send only MQTT messages, do not interact with WLED | ||||
| * `off-only` - only trigger presets or turn WLED on/off if WLED is not already on (displaying effect) | ||||
| * `notifications` - enable or disable sending notifications to other WLED instances using Sync button | ||||
| * `HA-discovery` - enable automatic discovery in Home Assistant | ||||
| * `override` - override PIR input when WLED state is changed using UI | ||||
| * `domoticz-idx` - Domoticz virtual switch ID (used with MQTT `domoticz/in`) | ||||
|  | ||||
|  | ||||
| Have fun - @gegu & @blazoncek | ||||
| @@ -91,3 +94,10 @@ Have fun - @gegu & @blazoncek | ||||
| * Added compile time option for off timer. | ||||
| * Added Home Assistant autodiscovery MQTT broadcast. | ||||
| * Updated info on compiling. | ||||
|  | ||||
| 2023-?? | ||||
| * Override option | ||||
| * Domoticz virtual switch ID (used with MQTT `domoticz/in`) | ||||
|  | ||||
| 2024-02 | ||||
| * Added compile time option to expand number of PIR sensors (they are logically ORed) `-D PIR_SENSOR_MAX_SENSORS=3` | ||||
| @@ -15,6 +15,9 @@ | ||||
|   #define PIR_SENSOR_OFF_SEC 600 | ||||
| #endif | ||||
|  | ||||
| #ifndef PIR_SENSOR_MAX_SENSORS | ||||
|   #define PIR_SENSOR_MAX_SENSORS 1 | ||||
| #endif | ||||
|  | ||||
| /* | ||||
|  * This usermod handles PIR sensor states. | ||||
| @@ -50,14 +53,13 @@ private: | ||||
|  | ||||
|   volatile unsigned long offTimerStart = 0;     // off timer start time | ||||
|   volatile bool PIRtriggered           = false; // did PIR trigger? | ||||
|   byte NotifyUpdateMode  = CALL_MODE_NO_NOTIFY; // notification mode for stateUpdated(): CALL_MODE_NO_NOTIFY or CALL_MODE_DIRECT_CHANGE | ||||
|   byte sensorPinState    = LOW;                 // current PIR sensor pin state | ||||
|   bool initDone          = false;               // status of initialization | ||||
|   unsigned long lastLoop = 0; | ||||
|   bool          initDone               = false; // status of initialization | ||||
|   unsigned long lastLoop               = 0; | ||||
|   bool sensorPinState[PIR_SENSOR_MAX_SENSORS] = {LOW}; // current PIR sensor pin state | ||||
|  | ||||
|   // configurable parameters | ||||
|   bool enabled              = true;           // PIR sensor enabled | ||||
|   int8_t PIRsensorPin       = PIR_SENSOR_PIN; // PIR sensor pin | ||||
|   int8_t PIRsensorPin[PIR_SENSOR_MAX_SENSORS] = {PIR_SENSOR_PIN}; // PIR sensor pin | ||||
|   uint32_t m_switchOffDelay = PIR_SENSOR_OFF_SEC*1000;  // delay before switch off after the sensor state goes LOW (10min) | ||||
|   uint8_t m_onPreset        = 0;              // on preset | ||||
|   uint8_t m_offPreset       = 0;              // off preset | ||||
| @@ -70,6 +72,7 @@ private: | ||||
|  | ||||
|   // Home Assistant | ||||
|   bool HomeAssistantDiscovery = false;        // is HA discovery turned on | ||||
|   int16_t idx = -1; // Domoticz virtual switch idx | ||||
|  | ||||
|   // strings to reduce flash memory usage (used more than twice) | ||||
|   static const char _name[]; | ||||
| @@ -81,8 +84,8 @@ private: | ||||
|   static const char _mqttOnly[]; | ||||
|   static const char _offOnly[]; | ||||
|   static const char _haDiscovery[]; | ||||
|   static const char _notify[]; | ||||
|   static const char _override[]; | ||||
|   static const char _domoticzIDX[]; | ||||
|  | ||||
|   /** | ||||
|    * check if it is daytime | ||||
| @@ -94,7 +97,7 @@ private: | ||||
|    * switch strip on/off | ||||
|    */ | ||||
|   void switchStrip(bool switchOn); | ||||
|   void publishMqtt(const char* state); | ||||
|   void publishMqtt(bool switchOn); | ||||
|  | ||||
|   // Create an MQTT Binary Sensor for Home Assistant Discovery purposes, this includes a pointer to the topic that is published to in the Loop. | ||||
|   void publishHomeAssistantAutodiscovery(); | ||||
| @@ -117,7 +120,7 @@ public: | ||||
|    * setup() is called once at boot. WiFi is not yet connected at this point. | ||||
|    * You can use it to initialize variables, sensors or similar. | ||||
|    */ | ||||
|   void setup(); | ||||
|   void setup() override; | ||||
|  | ||||
|   /** | ||||
|    * connected() is called every time the WiFi is (re)connected | ||||
| @@ -128,24 +131,24 @@ public: | ||||
|   /** | ||||
|    * onMqttConnect() is called when MQTT connection is established | ||||
|    */ | ||||
|   void onMqttConnect(bool sessionPresent); | ||||
|   void onMqttConnect(bool sessionPresent) override; | ||||
|  | ||||
|   /** | ||||
|    * loop() is called continuously. Here you can check for events, read sensors, etc. | ||||
|    */ | ||||
|   void loop(); | ||||
|   void loop() override; | ||||
|  | ||||
|   /** | ||||
|    * addToJsonInfo() can be used to add custom entries to the /json/info part of the JSON API. | ||||
|    *  | ||||
|    * Add PIR sensor state and switch off timer duration to jsoninfo | ||||
|    */ | ||||
|   void addToJsonInfo(JsonObject &root); | ||||
|   void addToJsonInfo(JsonObject &root) override; | ||||
|  | ||||
|   /** | ||||
|    * onStateChanged() is used to detect WLED state change | ||||
|    */ | ||||
|   void onStateChange(uint8_t mode); | ||||
|   void onStateChange(uint8_t mode) override; | ||||
|  | ||||
|   /** | ||||
|    * addToJsonState() can be used to add custom entries to the /json/state part of the JSON API (state object). | ||||
| @@ -157,17 +160,17 @@ public: | ||||
|    * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|    * Values in the state object may be modified by connected clients | ||||
|    */ | ||||
|   void readFromJsonState(JsonObject &root); | ||||
|   void readFromJsonState(JsonObject &root) override; | ||||
|  | ||||
|   /** | ||||
|    * provide the changeable values | ||||
|    */ | ||||
|   void addToConfig(JsonObject &root); | ||||
|   void addToConfig(JsonObject &root) override; | ||||
|  | ||||
|   /** | ||||
|    * provide UI information and allow extending UI options | ||||
|    */ | ||||
|   void appendConfigData(); | ||||
|   void appendConfigData() override; | ||||
|  | ||||
|   /** | ||||
|    * restore the changeable values | ||||
| @@ -175,13 +178,13 @@ public: | ||||
|    * | ||||
|    * The function should return true if configuration was successfully loaded or false if there was no configuration. | ||||
|    */ | ||||
|   bool readFromConfig(JsonObject &root); | ||||
|   bool readFromConfig(JsonObject &root) override; | ||||
|  | ||||
|   /** | ||||
|    * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
|    * This could be used in the future for the system to determine whether your usermod is installed. | ||||
|    */ | ||||
|   uint16_t getId() { return USERMOD_ID_PIRSWITCH; } | ||||
|   uint16_t getId() override { return USERMOD_ID_PIRSWITCH; } | ||||
| }; | ||||
|  | ||||
| // strings to reduce flash memory usage (used more than twice) | ||||
| @@ -194,8 +197,8 @@ const char PIRsensorSwitch::_nightTime[]      PROGMEM = "nighttime-only"; | ||||
| const char PIRsensorSwitch::_mqttOnly[]       PROGMEM = "mqtt-only"; | ||||
| const char PIRsensorSwitch::_offOnly[]        PROGMEM = "off-only"; | ||||
| const char PIRsensorSwitch::_haDiscovery[]    PROGMEM = "HA-discovery"; | ||||
| const char PIRsensorSwitch::_notify[]         PROGMEM = "notifications"; | ||||
| const char PIRsensorSwitch::_override[]       PROGMEM = "override"; | ||||
| const char PIRsensorSwitch::_domoticzIDX[]    PROGMEM = "domoticz-idx"; | ||||
|  | ||||
| bool PIRsensorSwitch::isDayTime() { | ||||
|   updateLocalTime(); | ||||
| @@ -235,24 +238,24 @@ void PIRsensorSwitch::switchStrip(bool switchOn) | ||||
|         prevPlaylist = 0; | ||||
|         prevPreset   = 255; | ||||
|       } | ||||
|       applyPreset(m_onPreset, NotifyUpdateMode); | ||||
|       applyPreset(m_onPreset, CALL_MODE_BUTTON_PRESET); | ||||
|       return; | ||||
|     } | ||||
|     // preset not assigned | ||||
|     if (bri == 0) { | ||||
|       bri = briLast; | ||||
|       stateUpdated(NotifyUpdateMode); | ||||
|       stateUpdated(CALL_MODE_BUTTON); | ||||
|     } | ||||
|   } else { | ||||
|     if (m_offPreset) { | ||||
|       applyPreset(m_offPreset, NotifyUpdateMode); | ||||
|       applyPreset(m_offPreset, CALL_MODE_BUTTON_PRESET); | ||||
|       return; | ||||
|     } else if (prevPlaylist) { | ||||
|       if (currentPreset==m_onPreset || currentPlaylist==m_onPreset) applyPreset(prevPlaylist, NotifyUpdateMode); | ||||
|       if (currentPreset==m_onPreset || currentPlaylist==m_onPreset) applyPreset(prevPlaylist, CALL_MODE_BUTTON_PRESET); | ||||
|       prevPlaylist = 0; | ||||
|       return; | ||||
|     } else if (prevPreset) { | ||||
|       if (prevPreset<255) { if (currentPreset==m_onPreset || currentPlaylist==m_onPreset) applyPreset(prevPreset, NotifyUpdateMode); } | ||||
|       if (prevPreset<255) { if (currentPreset==m_onPreset || currentPlaylist==m_onPreset) applyPreset(prevPreset, CALL_MODE_BUTTON_PRESET); } | ||||
|       else                { if (currentPreset==m_onPreset || currentPlaylist==m_onPreset) applyTemporaryPreset(); } | ||||
|       prevPreset = 0; | ||||
|       return; | ||||
| @@ -261,19 +264,29 @@ void PIRsensorSwitch::switchStrip(bool switchOn) | ||||
|     if (bri != 0) { | ||||
|       briLast = bri; | ||||
|       bri = 0; | ||||
|       stateUpdated(NotifyUpdateMode); | ||||
|       stateUpdated(CALL_MODE_BUTTON); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| void PIRsensorSwitch::publishMqtt(const char* state) | ||||
| void PIRsensorSwitch::publishMqtt(bool switchOn) | ||||
| { | ||||
| #ifndef WLED_DISABLE_MQTT | ||||
|   //Check if MQTT Connected, otherwise it will crash the 8266 | ||||
|   if (WLED_MQTT_CONNECTED) { | ||||
|     char buf[64]; | ||||
|     char buf[128]; | ||||
|     sprintf_P(buf, PSTR("%s/motion"), mqttDeviceTopic);   //max length: 33 + 7 = 40 | ||||
|     mqtt->publish(buf, 0, false, state); | ||||
|     mqtt->publish(buf, 0, false, switchOn?"on":"off"); | ||||
|     // Domoticz formatted message | ||||
|     if (idx > 0) { | ||||
|       StaticJsonDocument <128> msg; | ||||
|       msg[F("idx")]       = idx; | ||||
|       msg[F("RSSI")]      = WiFi.RSSI(); | ||||
|       msg[F("command")]   = F("switchlight"); | ||||
|       msg[F("switchcmd")] = switchOn ? F("On") : F("Off"); | ||||
|       serializeJson(msg, buf, 128); | ||||
|       mqtt->publish("domoticz/in", 0, false, buf); | ||||
|     } | ||||
|   } | ||||
| #endif | ||||
| } | ||||
| @@ -299,8 +312,8 @@ void PIRsensorSwitch::publishHomeAssistantAutodiscovery() | ||||
|     JsonObject device = doc.createNestedObject(F("device")); // attach the sensor to the same device | ||||
|     device[F("name")] = serverDescription; | ||||
|     device[F("ids")]  = String(F("wled-sensor-")) + mqttClientID; | ||||
|     device[F("mf")]   = "WLED"; | ||||
|     device[F("mdl")]  = F("FOSS"); | ||||
|     device[F("mf")]   = F(WLED_BRAND); | ||||
|     device[F("mdl")]  = F(WLED_PRODUCT_NAME); | ||||
|     device[F("sw")]   = versionString; | ||||
|      | ||||
|     sprintf_P(buf, PSTR("homeassistant/binary_sensor/%s/config"), uid); | ||||
| @@ -315,34 +328,36 @@ void PIRsensorSwitch::publishHomeAssistantAutodiscovery() | ||||
|  | ||||
| bool PIRsensorSwitch::updatePIRsensorState() | ||||
| { | ||||
|   bool pinState = digitalRead(PIRsensorPin); | ||||
|   if (pinState != sensorPinState) { | ||||
|     sensorPinState = pinState; // change previous state | ||||
|   bool stateChanged = false; | ||||
|   bool allOff = true; | ||||
|   for (int i = 0; i < PIR_SENSOR_MAX_SENSORS; i++) { | ||||
|     if (PIRsensorPin[i] < 0) continue; | ||||
|  | ||||
|     if (sensorPinState == HIGH) { | ||||
|       offTimerStart = 0; | ||||
|       if (!m_mqttOnly && (!m_nightTimeOnly || (m_nightTimeOnly && !isDayTime()))) switchStrip(true); | ||||
|       else if (NotifyUpdateMode != CALL_MODE_NO_NOTIFY) updateInterfaces(CALL_MODE_WS_SEND); | ||||
|       publishMqtt("on"); | ||||
|     } else { | ||||
|       // start switch off timer | ||||
|       offTimerStart = millis(); | ||||
|       if (NotifyUpdateMode != CALL_MODE_NO_NOTIFY) updateInterfaces(CALL_MODE_WS_SEND); | ||||
|     bool pinState = digitalRead(PIRsensorPin[i]); | ||||
|     if (pinState != sensorPinState[i]) { | ||||
|       sensorPinState[i] = pinState; // change previous state | ||||
|       stateChanged = true; | ||||
|  | ||||
|       if (sensorPinState[i] == HIGH) { | ||||
|         offTimerStart = 0; | ||||
|         allOff = false; | ||||
|         if (!m_mqttOnly && (!m_nightTimeOnly || (m_nightTimeOnly && !isDayTime()))) switchStrip(true); | ||||
|       } | ||||
|     } | ||||
|     return true; | ||||
|   } | ||||
|   return false; | ||||
|   if (stateChanged) { | ||||
|     publishMqtt(!allOff); | ||||
|     // start switch off timer | ||||
|     if (allOff) offTimerStart = millis(); | ||||
|   } | ||||
|   return stateChanged; | ||||
| } | ||||
|  | ||||
| bool PIRsensorSwitch::handleOffTimer() | ||||
| { | ||||
|   if (offTimerStart > 0 && millis() - offTimerStart > m_switchOffDelay) { | ||||
|     offTimerStart = 0; | ||||
|     if (enabled == true) { | ||||
|       if (!m_mqttOnly && (!m_nightTimeOnly || (m_nightTimeOnly && !isDayTime()) || PIRtriggered)) switchStrip(false); | ||||
|       else if (NotifyUpdateMode != CALL_MODE_NO_NOTIFY) updateInterfaces(CALL_MODE_WS_SEND); | ||||
|       publishMqtt("off"); | ||||
|     } | ||||
|     if (!m_mqttOnly && (!m_nightTimeOnly || (m_nightTimeOnly && !isDayTime()) || PIRtriggered)) switchStrip(false); | ||||
|     return true; | ||||
|   } | ||||
|   return false; | ||||
| @@ -352,18 +367,21 @@ bool PIRsensorSwitch::handleOffTimer() | ||||
|  | ||||
| void PIRsensorSwitch::setup() | ||||
| { | ||||
|   if (enabled) { | ||||
|   for (int i = 0; i < PIR_SENSOR_MAX_SENSORS; i++) { | ||||
|     sensorPinState[i] = LOW; | ||||
|     if (PIRsensorPin[i] < 0) continue; | ||||
|     // pin retrieved from cfg.json (readFromConfig()) prior to running setup() | ||||
|     if (PIRsensorPin >= 0 && pinManager.allocatePin(PIRsensorPin, false, PinOwner::UM_PIR)) { | ||||
|       // PIR Sensor mode INPUT_PULLUP | ||||
|       pinMode(PIRsensorPin, INPUT_PULLUP); | ||||
|       sensorPinState = digitalRead(PIRsensorPin); | ||||
|     if (pinManager.allocatePin(PIRsensorPin[i], false, PinOwner::UM_PIR)) { | ||||
|       // PIR Sensor mode INPUT_PULLDOWN | ||||
|       #ifdef ESP8266 | ||||
|       pinMode(PIRsensorPin[i], PIRsensorPin[i]==16 ? INPUT_PULLDOWN_16 : INPUT_PULLUP); // ESP8266 has INPUT_PULLDOWN on GPIO16 only | ||||
|       #else | ||||
|       pinMode(PIRsensorPin[i], INPUT_PULLDOWN); | ||||
|       #endif | ||||
|       sensorPinState[i] = digitalRead(PIRsensorPin[i]); | ||||
|     } else { | ||||
|       if (PIRsensorPin >= 0) { | ||||
|         DEBUG_PRINTLN(F("PIRSensorSwitch pin allocation failed.")); | ||||
|       } | ||||
|       PIRsensorPin = -1;  // allocation failed | ||||
|       enabled = false; | ||||
|       DEBUG_PRINT(F("PIRSensorSwitch pin ")); DEBUG_PRINTLN(i); DEBUG_PRINTLN(F(" allocation failed.")); | ||||
|       PIRsensorPin[i] = -1;  // allocation failed | ||||
|     } | ||||
|   } | ||||
|   initDone = true; | ||||
| @@ -378,8 +396,8 @@ void PIRsensorSwitch::onMqttConnect(bool sessionPresent) | ||||
|  | ||||
| void PIRsensorSwitch::loop() | ||||
| { | ||||
|   // only check sensors 4x/s | ||||
|   if (!enabled || millis() - lastLoop < 250 || strip.isUpdating()) return; | ||||
|   // only check sensors 5x/s | ||||
|   if (!enabled || millis() - lastLoop < 200) return; | ||||
|   lastLoop = millis(); | ||||
|  | ||||
|   if (!updatePIRsensorState()) { | ||||
| @@ -392,37 +410,35 @@ void PIRsensorSwitch::addToJsonInfo(JsonObject &root) | ||||
|   JsonObject user = root["u"]; | ||||
|   if (user.isNull()) user = root.createNestedObject("u"); | ||||
|  | ||||
|   bool state = LOW; | ||||
|   for (int i = 0; i < PIR_SENSOR_MAX_SENSORS; i++) | ||||
|     if (PIRsensorPin[i] >= 0) state |= sensorPinState[i]; | ||||
|  | ||||
|   JsonArray infoArr = user.createNestedArray(FPSTR(_name)); | ||||
|  | ||||
|   String uiDomString; | ||||
|   if (enabled) { | ||||
|     if (offTimerStart > 0) | ||||
|     { | ||||
|     if (offTimerStart > 0) { | ||||
|       uiDomString = ""; | ||||
|       unsigned int offSeconds = (m_switchOffDelay - (millis() - offTimerStart)) / 1000; | ||||
|       if (offSeconds >= 3600) | ||||
|       { | ||||
|       if (offSeconds >= 3600) { | ||||
|         uiDomString += (offSeconds / 3600); | ||||
|         uiDomString += F("h "); | ||||
|         offSeconds %= 3600; | ||||
|       } | ||||
|       if (offSeconds >= 60) | ||||
|       { | ||||
|       if (offSeconds >= 60) { | ||||
|         uiDomString += (offSeconds / 60); | ||||
|         offSeconds %= 60; | ||||
|       } | ||||
|       else if (uiDomString.length() > 0) | ||||
|       { | ||||
|       } else if (uiDomString.length() > 0) { | ||||
|         uiDomString += 0; | ||||
|       } | ||||
|       if (uiDomString.length() > 0) | ||||
|       { | ||||
|       if (uiDomString.length() > 0) { | ||||
|         uiDomString += F("min "); | ||||
|       } | ||||
|       uiDomString += (offSeconds); | ||||
|       infoArr.add(uiDomString + F("s")); | ||||
|     } else { | ||||
|       infoArr.add(sensorPinState ? F("sensor on") : F("inactive")); | ||||
|       infoArr.add(state ? F("sensor on") : F("inactive")); | ||||
|     } | ||||
|   } else { | ||||
|     infoArr.add(F("disabled")); | ||||
| @@ -442,9 +458,11 @@ void PIRsensorSwitch::addToJsonInfo(JsonObject &root) | ||||
|   uiDomString += F("</button>"); | ||||
|   infoArr.add(uiDomString); | ||||
|  | ||||
|   JsonObject sensor = root[F("sensor")]; | ||||
|   if (sensor.isNull()) sensor = root.createNestedObject(F("sensor")); | ||||
|   sensor[F("motion")] = sensorPinState || offTimerStart>0 ? true : false; | ||||
|   if (enabled) { | ||||
|     JsonObject sensor = root[F("sensor")]; | ||||
|     if (sensor.isNull()) sensor = root.createNestedObject(F("sensor")); | ||||
|     sensor[F("motion")] = state || offTimerStart>0 ? true : false; | ||||
|   } | ||||
| } | ||||
|  | ||||
| void PIRsensorSwitch::onStateChange(uint8_t mode) { | ||||
| @@ -474,7 +492,8 @@ void PIRsensorSwitch::addToConfig(JsonObject &root) | ||||
|   JsonObject top = root.createNestedObject(FPSTR(_name)); | ||||
|   top[FPSTR(_enabled)]        = enabled; | ||||
|   top[FPSTR(_switchOffDelay)] = m_switchOffDelay / 1000; | ||||
|   top["pin"]                  = PIRsensorPin; | ||||
|   JsonArray pinArray          = top.createNestedArray("pin"); | ||||
|   for (int i = 0; i < PIR_SENSOR_MAX_SENSORS; i++) pinArray.add(PIRsensorPin[i]); | ||||
|   top[FPSTR(_onPreset)]       = m_onPreset; | ||||
|   top[FPSTR(_offPreset)]      = m_offPreset; | ||||
|   top[FPSTR(_nightTime)]      = m_nightTimeOnly; | ||||
| @@ -482,21 +501,28 @@ void PIRsensorSwitch::addToConfig(JsonObject &root) | ||||
|   top[FPSTR(_offOnly)]        = m_offOnly; | ||||
|   top[FPSTR(_override)]       = m_override; | ||||
|   top[FPSTR(_haDiscovery)]    = HomeAssistantDiscovery; | ||||
|   top[FPSTR(_notify)]         = (NotifyUpdateMode != CALL_MODE_NO_NOTIFY); | ||||
|   top[FPSTR(_domoticzIDX)]    = idx; | ||||
|   DEBUG_PRINTLN(F("PIR config saved.")); | ||||
| } | ||||
|  | ||||
| void PIRsensorSwitch::appendConfigData() | ||||
| { | ||||
|   oappend(SET_F("addInfo('PIRsensorSwitch:HA-discovery',1,'HA=Home Assistant');"));     // 0 is field type, 1 is actual field | ||||
|   oappend(SET_F("addInfo('PIRsensorSwitch:notifications',1,'Periodic WS updates');"));  // 0 is field type, 1 is actual field | ||||
|   oappend(SET_F("addInfo('PIRsensorSwitch:override',1,'Cancel timer on change');"));    // 0 is field type, 1 is actual field | ||||
|   for (int i = 0; i < PIR_SENSOR_MAX_SENSORS; i++) { | ||||
|     char str[128]; | ||||
|     sprintf_P(str, PSTR("addInfo('PIRsensorSwitch:pin[]',%d,'','#%d');"), i, i); | ||||
|     oappend(str); | ||||
|   } | ||||
| } | ||||
|  | ||||
| bool PIRsensorSwitch::readFromConfig(JsonObject &root) | ||||
| { | ||||
|   bool oldEnabled = enabled; | ||||
|   int8_t oldPin = PIRsensorPin; | ||||
|   int8_t oldPin[PIR_SENSOR_MAX_SENSORS]; | ||||
|   for (int i = 0; i < PIR_SENSOR_MAX_SENSORS; i++) { | ||||
|     oldPin[i] = PIRsensorPin[i]; | ||||
|     PIRsensorPin[i] = -1; | ||||
|   } | ||||
|  | ||||
|   DEBUG_PRINT(FPSTR(_name)); | ||||
|   JsonObject top = root[FPSTR(_name)]; | ||||
| @@ -505,7 +531,13 @@ bool PIRsensorSwitch::readFromConfig(JsonObject &root) | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   PIRsensorPin = top["pin"] | PIRsensorPin; | ||||
|   JsonArray pins = top["pin"]; | ||||
|   if (!pins.isNull()) { | ||||
|     for (size_t i = 0; i < PIR_SENSOR_MAX_SENSORS; i++) | ||||
|       if (i < pins.size()) PIRsensorPin[i] = pins[i] | PIRsensorPin[i]; | ||||
|   } else { | ||||
|     PIRsensorPin[0] = top["pin"] | oldPin[0]; | ||||
|   } | ||||
|  | ||||
|   enabled = top[FPSTR(_enabled)] | enabled; | ||||
|  | ||||
| @@ -521,33 +553,17 @@ bool PIRsensorSwitch::readFromConfig(JsonObject &root) | ||||
|   m_offOnly       = top[FPSTR(_offOnly)] | m_offOnly; | ||||
|   m_override      = top[FPSTR(_override)] | m_override; | ||||
|   HomeAssistantDiscovery = top[FPSTR(_haDiscovery)] | HomeAssistantDiscovery; | ||||
|  | ||||
|   NotifyUpdateMode = top[FPSTR(_notify)] ? CALL_MODE_DIRECT_CHANGE : CALL_MODE_NO_NOTIFY; | ||||
|   idx             = top[FPSTR(_domoticzIDX)] | idx; | ||||
|  | ||||
|   if (!initDone) { | ||||
|     // reading config prior to setup() | ||||
|     DEBUG_PRINTLN(F(" config loaded.")); | ||||
|   } else { | ||||
|     if (oldPin != PIRsensorPin || oldEnabled != enabled) { | ||||
|       // check if pin is OK | ||||
|       if (oldPin != PIRsensorPin && oldPin >= 0) { | ||||
|         // if we are changing pin in settings page | ||||
|         // deallocate old pin | ||||
|         pinManager.deallocatePin(oldPin, PinOwner::UM_PIR); | ||||
|         if (pinManager.allocatePin(PIRsensorPin, false, PinOwner::UM_PIR)) { | ||||
|           pinMode(PIRsensorPin, INPUT_PULLUP); | ||||
|         } else { | ||||
|           // allocation failed | ||||
|           PIRsensorPin = -1; | ||||
|           enabled = false; | ||||
|         } | ||||
|       } | ||||
|       if (enabled) { | ||||
|         sensorPinState = digitalRead(PIRsensorPin); | ||||
|       } | ||||
|     } | ||||
|     for (int i = 0; i < PIR_SENSOR_MAX_SENSORS; i++) | ||||
|       if (oldPin[i] >= 0) pinManager.deallocatePin(oldPin[i], PinOwner::UM_PIR); | ||||
|     setup(); | ||||
|     DEBUG_PRINTLN(F(" config (re)loaded.")); | ||||
|   } | ||||
|   // use "return !top["newestParameter"].isNull();" when updating Usermod with new features | ||||
|   return !top[FPSTR(_override)].isNull(); | ||||
|   return !(pins.isNull() || pins.size() != PIR_SENSOR_MAX_SENSORS); | ||||
| } | ||||
|   | ||||
| @@ -188,7 +188,7 @@ class PWMFanUsermod : public Usermod { | ||||
|  | ||||
|     // gets called once at boot. Do all initialization that doesn't depend on | ||||
|     // network here | ||||
|     void setup() { | ||||
|     void setup() override { | ||||
|       #ifdef USERMOD_DALLASTEMPERATURE    | ||||
|       // This Usermod requires Temperature usermod | ||||
|       tempUM = (UsermodTemperature*) usermods.lookup(USERMOD_ID_TEMPERATURE); | ||||
| @@ -203,12 +203,12 @@ class PWMFanUsermod : public Usermod { | ||||
|  | ||||
|     // gets called every time WiFi is (re-)connected. Initialize own network | ||||
|     // interfaces here | ||||
|     void connected() {} | ||||
|     void connected() override {} | ||||
|  | ||||
|     /* | ||||
|      * Da loop. | ||||
|      */ | ||||
|     void loop() { | ||||
|     void loop() override { | ||||
|       if (!enabled || strip.isUpdating()) return; | ||||
|  | ||||
|       unsigned long now = millis(); | ||||
| @@ -223,7 +223,7 @@ class PWMFanUsermod : public Usermod { | ||||
|      * Creating an "u" object allows you to add custom key/value pairs to the Info section of the WLED web UI. | ||||
|      * Below it is shown how this could be used for e.g. a light sensor | ||||
|      */ | ||||
|     void addToJsonInfo(JsonObject& root) { | ||||
|     void addToJsonInfo(JsonObject& root) override { | ||||
|       JsonObject user = root["u"]; | ||||
|       if (user.isNull()) user = root.createNestedObject("u"); | ||||
|  | ||||
| @@ -272,7 +272,7 @@ class PWMFanUsermod : public Usermod { | ||||
|      * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void readFromJsonState(JsonObject& root) { | ||||
|     void readFromJsonState(JsonObject& root) override { | ||||
|       if (!initDone) return;  // prevent crash on boot applyPreset() | ||||
|       JsonObject usermod = root[FPSTR(_name)]; | ||||
|       if (!usermod.isNull()) { | ||||
| @@ -305,7 +305,7 @@ class PWMFanUsermod : public Usermod { | ||||
|      *  | ||||
|      * I highly recommend checking out the basics of ArduinoJson serialization and deserialization in order to use custom settings! | ||||
|      */ | ||||
|     void addToConfig(JsonObject& root) { | ||||
|     void addToConfig(JsonObject& root) override { | ||||
|       JsonObject top = root.createNestedObject(FPSTR(_name)); // usermodname | ||||
|       top[FPSTR(_enabled)]        = enabled; | ||||
|       top[FPSTR(_pwmPin)]         = pwmPin; | ||||
| @@ -328,7 +328,7 @@ class PWMFanUsermod : public Usermod { | ||||
|      *  | ||||
|      * The function should return true if configuration was successfully loaded or false if there was no configuration. | ||||
|      */ | ||||
|     bool readFromConfig(JsonObject& root) { | ||||
|     bool readFromConfig(JsonObject& root) override { | ||||
|       int8_t newTachoPin = tachoPin; | ||||
|       int8_t newPwmPin   = pwmPin; | ||||
|  | ||||
| @@ -380,7 +380,7 @@ class PWMFanUsermod : public Usermod { | ||||
|      * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
|      * This could be used in the future for the system to determine whether your usermod is installed. | ||||
|      */ | ||||
|     uint16_t getId() { | ||||
|     uint16_t getId() override { | ||||
|         return USERMOD_ID_PWM_FAN; | ||||
|     } | ||||
| }; | ||||
|   | ||||
| @@ -43,12 +43,12 @@ void handleRelay() | ||||
|       digitalWrite(PIN_UP_RELAY, LOW); | ||||
|       upActiveBefore = true; | ||||
|       upStartTime = millis(); | ||||
|       DEBUG_PRINTLN("UPA"); | ||||
|       DEBUG_PRINTLN(F("UPA")); | ||||
|     } | ||||
|     if (millis()- upStartTime > PIN_ON_TIME) | ||||
|     { | ||||
|       upActive = false; | ||||
|       DEBUG_PRINTLN("UPN"); | ||||
|       DEBUG_PRINTLN(F("UPN")); | ||||
|     } | ||||
|   } else if (upActiveBefore) | ||||
|   { | ||||
|   | ||||
| @@ -119,7 +119,7 @@ public: | ||||
|       } | ||||
|       else | ||||
|       { | ||||
|         DEBUG_PRINTLN("Missing MQTT connection. Not publishing data"); | ||||
|         DEBUG_PRINTLN(F("Missing MQTT connection. Not publishing data")); | ||||
|       } | ||||
|     } | ||||
| #endif | ||||
|   | ||||
| @@ -132,7 +132,7 @@ class St7789DisplayUsermod : public Usermod { | ||||
|      * setup() is called once at boot. WiFi is not yet connected at this point. | ||||
|      * You can use it to initialize variables, sensors or similar. | ||||
|      */ | ||||
|     void setup() | ||||
|     void setup() override | ||||
|     { | ||||
|         PinManagerPinType spiPins[] = { { spi_mosi, true }, { spi_miso, false}, { spi_sclk, true } }; | ||||
|         if (!pinManager.allocateMultiplePins(spiPins, 3, PinOwner::HW_SPI)) { enabled = false; return; } | ||||
| @@ -162,7 +162,7 @@ class St7789DisplayUsermod : public Usermod { | ||||
|      * connected() is called every time the WiFi is (re)connected | ||||
|      * Use it to initialize network interfaces | ||||
|      */ | ||||
|     void connected() { | ||||
|     void connected() override { | ||||
|       //Serial.println("Connected to WiFi!"); | ||||
|     } | ||||
|  | ||||
| @@ -176,7 +176,7 @@ class St7789DisplayUsermod : public Usermod { | ||||
|      * 2. Try to avoid using the delay() function. NEVER use delays longer than 10 milliseconds. | ||||
|      *    Instead, use a timer check as shown here. | ||||
|      */ | ||||
|     void loop() { | ||||
|     void loop() override { | ||||
|         char buff[LINE_BUFFER_SIZE]; | ||||
|  | ||||
|         // Check if we time interval for redrawing passes. | ||||
| @@ -307,7 +307,7 @@ class St7789DisplayUsermod : public Usermod { | ||||
|         // Print estimated milliamp usage (must specify the LED type in LED prefs for this to be a reasonable estimate). | ||||
|         tft.print("Current: "); | ||||
|         tft.setTextColor(TFT_ORANGE); | ||||
|         tft.print(strip.currentMilliamps); | ||||
|         tft.print(BusManager::currentMilliamps()); | ||||
|         tft.print("mA"); | ||||
|     } | ||||
|  | ||||
| @@ -316,7 +316,7 @@ class St7789DisplayUsermod : public Usermod { | ||||
|      * Creating an "u" object allows you to add custom key/value pairs to the Info section of the WLED web UI. | ||||
|      * Below it is shown how this could be used for e.g. a light sensor | ||||
|      */ | ||||
|     void addToJsonInfo(JsonObject& root) | ||||
|     void addToJsonInfo(JsonObject& root) override | ||||
|     { | ||||
|       JsonObject user = root["u"]; | ||||
|       if (user.isNull()) user = root.createNestedObject("u"); | ||||
| @@ -330,7 +330,7 @@ class St7789DisplayUsermod : public Usermod { | ||||
|      * addToJsonState() can be used to add custom entries to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void addToJsonState(JsonObject& root) | ||||
|     void addToJsonState(JsonObject& root) override | ||||
|     { | ||||
|       //root["user0"] = userVar0; | ||||
|     } | ||||
| @@ -340,7 +340,7 @@ class St7789DisplayUsermod : public Usermod { | ||||
|      * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void readFromJsonState(JsonObject& root) | ||||
|     void readFromJsonState(JsonObject& root) override | ||||
|     { | ||||
|       //userVar0 = root["user0"] | userVar0; //if "user0" key exists in JSON, update, else keep old value | ||||
|       //if (root["bri"] == 255) Serial.println(F("Don't burn down your garage!")); | ||||
| @@ -361,7 +361,7 @@ class St7789DisplayUsermod : public Usermod { | ||||
|      * | ||||
|      * I highly recommend checking out the basics of ArduinoJson serialization and deserialization in order to use custom settings! | ||||
|      */ | ||||
|     void addToConfig(JsonObject& root) | ||||
|     void addToConfig(JsonObject& root) override | ||||
|     { | ||||
|       JsonObject top = root.createNestedObject("ST7789"); | ||||
|       JsonArray pins = top.createNestedArray("pin"); | ||||
| @@ -373,7 +373,7 @@ class St7789DisplayUsermod : public Usermod { | ||||
|     } | ||||
|  | ||||
|  | ||||
|     void appendConfigData() { | ||||
|     void appendConfigData() override { | ||||
|       oappend(SET_F("addInfo('ST7789:pin[]',0,'','SPI CS');")); | ||||
|       oappend(SET_F("addInfo('ST7789:pin[]',1,'','SPI DC');")); | ||||
|       oappend(SET_F("addInfo('ST7789:pin[]',2,'','SPI RST');")); | ||||
| @@ -388,7 +388,7 @@ class St7789DisplayUsermod : public Usermod { | ||||
|      * but also that if you want to write persistent values to a dynamic buffer, you'd need to allocate it here instead of in setup. | ||||
|      * If you don't know what that is, don't fret. It most likely doesn't affect your use case :) | ||||
|      */ | ||||
|     bool readFromConfig(JsonObject& root) | ||||
|     bool readFromConfig(JsonObject& root) override | ||||
|     { | ||||
|       //JsonObject top = root["top"]; | ||||
|       //userVar0 = top["great"] | 42; //The value right of the pipe "|" is the default value in case your setting was not present in cfg.json (e.g. first boot) | ||||
| @@ -400,7 +400,7 @@ class St7789DisplayUsermod : public Usermod { | ||||
|      * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
|      * This could be used in the future for the system to determine whether your usermod is installed. | ||||
|      */ | ||||
|     uint16_t getId() | ||||
|     uint16_t getId() override | ||||
|     { | ||||
|       return USERMOD_ID_ST7789_DISPLAY; | ||||
|     } | ||||
|   | ||||
| @@ -93,8 +93,8 @@ class Si7021_MQTT_HA : public Usermod | ||||
|  | ||||
|         JsonObject device = doc.createNestedObject("device"); // attach the sensor to the same device | ||||
|         device["name"] = String(serverDescription); | ||||
|         device["model"] = "WLED"; | ||||
|         device["manufacturer"] = "Aircoookie"; | ||||
|         device["model"] = F(WLED_PRODUCT_NAME); | ||||
|         device["manufacturer"] = F(WLED_BRAND); | ||||
|         device["identifiers"] = String("wled-") + String(serverDescription); | ||||
|         device["sw_version"] = VERSION; | ||||
|  | ||||
|   | ||||
| @@ -48,6 +48,7 @@ class UsermodTemperature : public Usermod { | ||||
|     bool enabled = true; | ||||
|  | ||||
|     bool HApublished = false; | ||||
|     int16_t idx = -1;   // Domoticz virtual sensor idx | ||||
|  | ||||
|     // strings to reduce flash memory usage (used more than twice) | ||||
|     static const char _name[]; | ||||
| @@ -55,7 +56,11 @@ class UsermodTemperature : public Usermod { | ||||
|     static const char _readInterval[]; | ||||
|     static const char _parasite[]; | ||||
|     static const char _parasitePin[]; | ||||
|  | ||||
|     static const char _domoticzIDX[]; | ||||
|     static const char _sensor[]; | ||||
|     static const char _temperature[]; | ||||
|     static const char _Temperature[]; | ||||
|      | ||||
|     //Dallas sensor quick (& dirty) reading. Credit to - Author: Peter Scargill, August 17th, 2013 | ||||
|     float readDallas(); | ||||
|     void requestTemperatures(); | ||||
| @@ -74,26 +79,26 @@ class UsermodTemperature : public Usermod { | ||||
|     inline float getTemperatureF() { return temperature * 1.8f + 32.0f; } | ||||
|     float getTemperature(); | ||||
|     const char *getTemperatureUnit(); | ||||
|     uint16_t getId() { return USERMOD_ID_TEMPERATURE; } | ||||
|     uint16_t getId() override { return USERMOD_ID_TEMPERATURE; } | ||||
|  | ||||
|     void setup(); | ||||
|     void loop(); | ||||
|     //void connected(); | ||||
|     void setup() override; | ||||
|     void loop() override; | ||||
|     //void connected() override; | ||||
| #ifndef WLED_DISABLE_MQTT | ||||
|     void onMqttConnect(bool sessionPresent); | ||||
|     void onMqttConnect(bool sessionPresent) override; | ||||
| #endif | ||||
|     //void onUpdateBegin(bool init); | ||||
|     //void onUpdateBegin(bool init) override; | ||||
|  | ||||
|     //bool handleButton(uint8_t b); | ||||
|     //void handleOverlayDraw(); | ||||
|     //bool handleButton(uint8_t b) override; | ||||
|     //void handleOverlayDraw() override; | ||||
|  | ||||
|     void addToJsonInfo(JsonObject& root); | ||||
|     //void addToJsonState(JsonObject &root); | ||||
|     //void readFromJsonState(JsonObject &root); | ||||
|     void addToConfig(JsonObject &root); | ||||
|     bool readFromConfig(JsonObject &root); | ||||
|     void addToJsonInfo(JsonObject& root) override; | ||||
|     //void addToJsonState(JsonObject &root) override; | ||||
|     //void readFromJsonState(JsonObject &root) override; | ||||
|     void addToConfig(JsonObject &root) override; | ||||
|     bool readFromConfig(JsonObject &root) override; | ||||
|  | ||||
|     void appendConfigData(); | ||||
|     void appendConfigData() override; | ||||
| }; | ||||
|  | ||||
| //Dallas sensor quick (& dirty) reading. Credit to - Author: Peter Scargill, August 17th, 2013 | ||||
| @@ -108,9 +113,9 @@ float UsermodTemperature::readDallas() { | ||||
|     #ifdef WLED_DEBUG | ||||
|     if (OneWire::crc8(data,8) != data[8]) { | ||||
|       DEBUG_PRINTLN(F("CRC error reading temperature.")); | ||||
|       for (byte i=0; i < 9; i++) DEBUG_PRINTF("0x%02X ", data[i]); | ||||
|       for (byte i=0; i < 9; i++) DEBUG_PRINTF_P(PSTR("0x%02X "), data[i]); | ||||
|       DEBUG_PRINT(F(" => ")); | ||||
|       DEBUG_PRINTF("0x%02X\n", OneWire::crc8(data,8)); | ||||
|       DEBUG_PRINTF_P(PSTR("0x%02X\n"), OneWire::crc8(data,8)); | ||||
|     } | ||||
|     #endif | ||||
|     switch(sensorFound) { | ||||
| @@ -147,7 +152,7 @@ void UsermodTemperature::readTemperature() { | ||||
|   temperature = readDallas(); | ||||
|   lastMeasurement = millis(); | ||||
|   waitingForConversion = false; | ||||
|   //DEBUG_PRINTF("Read temperature %2.1f.\n", temperature); // does not work properly on 8266 | ||||
|   //DEBUG_PRINTF_P(PSTR("Read temperature %2.1f.\n"), temperature); // does not work properly on 8266 | ||||
|   DEBUG_PRINT(F("Read temperature ")); | ||||
|   DEBUG_PRINTLN(temperature); | ||||
| } | ||||
| @@ -169,7 +174,7 @@ bool UsermodTemperature::findSensor() { | ||||
|         case 0x42:  // DS28EA00 | ||||
|           DEBUG_PRINTLN(F("Sensor found.")); | ||||
|           sensorFound = deviceAddress[0]; | ||||
|           DEBUG_PRINTF("0x%02X\n", sensorFound); | ||||
|           DEBUG_PRINTF_P(PSTR("0x%02X\n"), sensorFound); | ||||
|           return true; | ||||
|       } | ||||
|     } | ||||
| @@ -189,9 +194,9 @@ void UsermodTemperature::publishHomeAssistantAutodiscovery() { | ||||
|   sprintf_P(buf, PSTR("%s Temperature"), serverDescription); | ||||
|   json[F("name")] = buf; | ||||
|   strcpy(buf, mqttDeviceTopic); | ||||
|   strcat_P(buf, PSTR("/temperature")); | ||||
|   strcat_P(buf, _Temperature); | ||||
|   json[F("state_topic")] = buf; | ||||
|   json[F("device_class")] = F("temperature"); | ||||
|   json[F("device_class")] = FPSTR(_temperature); | ||||
|   json[F("unique_id")] = escapedMac.c_str(); | ||||
|   json[F("unit_of_measurement")] = F("°C"); | ||||
|   payload_size = serializeJson(json, json_str); | ||||
| @@ -264,16 +269,25 @@ void UsermodTemperature::loop() { | ||||
|  | ||||
| #ifndef WLED_DISABLE_MQTT | ||||
|     if (WLED_MQTT_CONNECTED) { | ||||
|       char subuf[64]; | ||||
|       char subuf[128]; | ||||
|       strcpy(subuf, mqttDeviceTopic); | ||||
|       if (temperature > -100.0f) { | ||||
|         // dont publish super low temperature as the graph will get messed up | ||||
|         // the DallasTemperature library returns -127C or -196.6F when problem | ||||
|         // reading the sensor | ||||
|         strcat_P(subuf, PSTR("/temperature")); | ||||
|         strcat_P(subuf, _Temperature); | ||||
|         mqtt->publish(subuf, 0, false, String(getTemperatureC()).c_str()); | ||||
|         strcat_P(subuf, PSTR("_f")); | ||||
|         mqtt->publish(subuf, 0, false, String(getTemperatureF()).c_str()); | ||||
|         if (idx > 0) { | ||||
|           StaticJsonDocument <128> msg; | ||||
|           msg[F("idx")]    = idx; | ||||
|           msg[F("RSSI")]   = WiFi.RSSI(); | ||||
|           msg[F("nvalue")] = 0; | ||||
|           msg[F("svalue")] = String(getTemperatureC()); | ||||
|           serializeJson(msg, subuf, 127); | ||||
|           mqtt->publish("domoticz/in", 0, false, subuf); | ||||
|         } | ||||
|       } else { | ||||
|         // publish something else to indicate status? | ||||
|       } | ||||
| @@ -324,9 +338,9 @@ void UsermodTemperature::addToJsonInfo(JsonObject& root) { | ||||
|   temp.add(getTemperature()); | ||||
|   temp.add(getTemperatureUnit()); | ||||
|  | ||||
|   JsonObject sensor = root[F("sensor")]; | ||||
|   if (sensor.isNull()) sensor = root.createNestedObject(F("sensor")); | ||||
|   temp = sensor.createNestedArray(F("temperature")); | ||||
|   JsonObject sensor = root[FPSTR(_sensor)]; | ||||
|   if (sensor.isNull()) sensor = root.createNestedObject(FPSTR(_sensor)); | ||||
|   temp = sensor.createNestedArray(FPSTR(_temperature)); | ||||
|   temp.add(getTemperature()); | ||||
|   temp.add(getTemperatureUnit()); | ||||
| } | ||||
| @@ -356,10 +370,11 @@ void UsermodTemperature::addToConfig(JsonObject &root) { | ||||
|   JsonObject top = root.createNestedObject(FPSTR(_name)); // usermodname | ||||
|   top[FPSTR(_enabled)] = enabled; | ||||
|   top["pin"]  = temperaturePin;     // usermodparam | ||||
|   top["degC"] = degC;  // usermodparam | ||||
|   top[F("degC")] = degC;  // usermodparam | ||||
|   top[FPSTR(_readInterval)] = readingInterval / 1000; | ||||
|   top[FPSTR(_parasite)] = parasite; | ||||
|   top[FPSTR(_parasitePin)] = parasitePin; | ||||
|   top[FPSTR(_domoticzIDX)] = idx; | ||||
|   DEBUG_PRINTLN(F("Temperature config saved.")); | ||||
| } | ||||
|  | ||||
| @@ -381,11 +396,12 @@ bool UsermodTemperature::readFromConfig(JsonObject &root) { | ||||
|  | ||||
|   enabled           = top[FPSTR(_enabled)] | enabled; | ||||
|   newTemperaturePin = top["pin"] | newTemperaturePin; | ||||
|   degC              = top["degC"] | degC; | ||||
|   degC              = top[F("degC")] | degC; | ||||
|   readingInterval   = top[FPSTR(_readInterval)] | readingInterval/1000; | ||||
|   readingInterval   = min(120,max(10,(int)readingInterval)) * 1000;  // convert to ms | ||||
|   parasite          = top[FPSTR(_parasite)] | parasite; | ||||
|   parasitePin       = top[FPSTR(_parasitePin)] | parasitePin; | ||||
|   idx               = top[FPSTR(_domoticzIDX)] | idx; | ||||
|  | ||||
|   if (!initDone) { | ||||
|     // first run: reading from cfg.json | ||||
| @@ -406,7 +422,7 @@ bool UsermodTemperature::readFromConfig(JsonObject &root) { | ||||
|     } | ||||
|   } | ||||
|   // use "return !top["newestParameter"].isNull();" when updating Usermod with new features | ||||
|   return !top[FPSTR(_parasitePin)].isNull(); | ||||
|   return !top[FPSTR(_domoticzIDX)].isNull(); | ||||
| } | ||||
|  | ||||
| void UsermodTemperature::appendConfigData() { | ||||
| @@ -430,3 +446,7 @@ const char UsermodTemperature::_enabled[]      PROGMEM = "enabled"; | ||||
| const char UsermodTemperature::_readInterval[] PROGMEM = "read-interval-s"; | ||||
| const char UsermodTemperature::_parasite[]     PROGMEM = "parasite-pwr"; | ||||
| const char UsermodTemperature::_parasitePin[]  PROGMEM = "parasite-pwr-pin"; | ||||
| const char UsermodTemperature::_domoticzIDX[]  PROGMEM = "domoticz-idx"; | ||||
| const char UsermodTemperature::_sensor[]       PROGMEM = "sensor"; | ||||
| const char UsermodTemperature::_temperature[]  PROGMEM = "temperature"; | ||||
| const char UsermodTemperature::_Temperature[]  PROGMEM = "/temperature"; | ||||
							
								
								
									
										117
									
								
								usermods/TetrisAI_v2/gridbw.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								usermods/TetrisAI_v2/gridbw.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,117 @@ | ||||
| /****************************************************************************** | ||||
|   * @file           : gridbw.h | ||||
|   * @brief          : contains the tetris grid as binary so black and white version | ||||
|   ****************************************************************************** | ||||
|   * @attention | ||||
|   * | ||||
|   * Copyright (c) muebau 2023 | ||||
|   * All rights reserved.</center></h2> | ||||
|   * | ||||
|   ****************************************************************************** | ||||
| */ | ||||
|  | ||||
| #ifndef __GRIDBW_H__ | ||||
| #define __GRIDBW_H__ | ||||
|  | ||||
| #include <iterator> | ||||
| #include <vector> | ||||
| #include "pieces.h" | ||||
|  | ||||
| using namespace std; | ||||
|  | ||||
| class GridBW | ||||
| { | ||||
| private: | ||||
| public: | ||||
|     uint8_t width; | ||||
|     uint8_t height; | ||||
|     std::vector<uint32_t> pixels; | ||||
|  | ||||
|     GridBW(uint8_t width, uint8_t height): | ||||
|         width(width), | ||||
|         height(height), | ||||
|         pixels(height) | ||||
|     { | ||||
|         if (width > 32) | ||||
|         { | ||||
|             throw std::invalid_argument("maximal width is 32"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void placePiece(Piece* piece, uint8_t x, uint8_t y) | ||||
|     { | ||||
|         for (uint8_t row = 4 - piece->getRotation().height; row < 4; row++) | ||||
|         { | ||||
|             pixels[y + (row - (4 - piece->getRotation().height))] |= piece->getGridRow(x, row, width); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void erasePiece(Piece* piece, uint8_t x, uint8_t y) | ||||
|     { | ||||
|         for (uint8_t row = 4 - piece->getRotation().height; row < 4; row++) | ||||
|         { | ||||
|             pixels[y + (row - (4 - piece->getRotation().height))] &= ~piece->getGridRow(x, row, width); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     bool noCollision(Piece* piece, uint8_t x, uint8_t y) | ||||
|     { | ||||
|         //if it touches a wall it is a collision | ||||
|         if (x > (this->width - piece->getRotation().width) || y > this->height - piece->getRotation().height) | ||||
|         { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         for (uint8_t row = 4 - piece->getRotation().height; row < 4; row++) | ||||
|         { | ||||
|             if (piece->getGridRow(x, row, width) & pixels[y + (row - (4 - piece->getRotation().height))]) | ||||
|             { | ||||
|                 return false; | ||||
|             } | ||||
|         } | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     void findLandingPosition(Piece* piece) | ||||
|     { | ||||
|         // move down until the piece bumps into some occupied pixels or the 'wall' | ||||
|         while (noCollision(piece, piece->x, piece->landingY)) | ||||
|         { | ||||
|             piece->landingY++; | ||||
|         } | ||||
|  | ||||
|         //at this point the positon is 'in the wall' or 'over some occupied pixel' | ||||
|         //so the previous position was the last correct one (clamped to 0 as minimum). | ||||
|         piece->landingY = piece->landingY > 0 ? piece->landingY - 1 : 0; | ||||
|     } | ||||
|  | ||||
|     void cleanupFullLines() | ||||
|     { | ||||
|         uint8_t offset = 0; | ||||
|  | ||||
|         //from "height - 1" to "0", so from bottom row to top | ||||
|         for (uint8_t row = height; row-- > 0; ) | ||||
|         { | ||||
|             //full line? | ||||
|             if (isLineFull(row)) | ||||
|             { | ||||
|                 offset++; | ||||
|                 pixels[row] = 0x0; | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             if (offset > 0) | ||||
|             { | ||||
|                 pixels[row + offset] = pixels[row]; | ||||
|                 pixels[row] = 0x0; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     bool isLineFull(uint8_t y) | ||||
|     { | ||||
|         return pixels[y] == (uint32_t)((1 << width) - 1); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| #endif /* __GRIDBW_H__ */ | ||||
							
								
								
									
										132
									
								
								usermods/TetrisAI_v2/gridcolor.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								usermods/TetrisAI_v2/gridcolor.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,132 @@ | ||||
| /****************************************************************************** | ||||
|   * @file           : gridcolor.h | ||||
|   * @brief          : contains the tetris grid as 8bit indexed color version | ||||
|   ****************************************************************************** | ||||
|   * @attention | ||||
|   * | ||||
|   * Copyright (c) muebau 2023 | ||||
|   * All rights reserved.</center></h2> | ||||
|   * | ||||
|   ****************************************************************************** | ||||
| */ | ||||
|  | ||||
| #ifndef __GRIDCOLOR_H__ | ||||
| #define __GRIDCOLOR_H__ | ||||
| #include <stdint.h> | ||||
| #include <stdbool.h> | ||||
| #include <vector> | ||||
| #include "gridbw.h" | ||||
| #include "gridcolor.h" | ||||
|  | ||||
| using namespace std; | ||||
|  | ||||
| class GridColor | ||||
| { | ||||
| private: | ||||
| public: | ||||
|     uint8_t width; | ||||
|     uint8_t height; | ||||
|     GridBW gridBW; | ||||
|     std::vector<uint8_t> pixels; | ||||
|  | ||||
|     GridColor(uint8_t width, uint8_t height): | ||||
|         width(width), | ||||
|         height(height), | ||||
|         gridBW(width, height), | ||||
|         pixels(width* height) | ||||
|     {} | ||||
|  | ||||
|     void clear() | ||||
|     { | ||||
|         for (uint8_t y = 0; y < height; y++) | ||||
|         { | ||||
|             gridBW.pixels[y] = 0x0; | ||||
|             for (int8_t x = 0; x < width; x++) | ||||
|             { | ||||
|                 *getPixel(x, y) = 0; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void placePiece(Piece* piece, uint8_t x, uint8_t y) | ||||
|     { | ||||
|         for (uint8_t pieceY = 0; pieceY < piece->getRotation().height; pieceY++) | ||||
|         { | ||||
|             for (uint8_t pieceX = 0; pieceX < piece->getRotation().width; pieceX++) | ||||
|             { | ||||
|                 if (piece->getPixel(pieceX, pieceY)) | ||||
|                 { | ||||
|                     *getPixel(x + pieceX, y + pieceY) = piece->pieceData->colorIndex; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void erasePiece(Piece* piece, uint8_t x, uint8_t y) | ||||
|     { | ||||
|         for (uint8_t pieceY = 0; pieceY < piece->getRotation().height; pieceY++) | ||||
|         { | ||||
|             for (uint8_t pieceX = 0; pieceX < piece->getRotation().width; pieceX++) | ||||
|             { | ||||
|                 if (piece->getPixel(pieceX, pieceY)) | ||||
|                 { | ||||
|                     *getPixel(x + pieceX, y + pieceY) = 0; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void cleanupFullLines() | ||||
|     { | ||||
|         uint8_t offset = 0; | ||||
|         //from "height - 1" to "0", so from bottom row to top | ||||
|         for (uint8_t y = height; y-- > 0; ) | ||||
|         { | ||||
|             if (gridBW.isLineFull(y)) | ||||
|             { | ||||
|                 offset++; | ||||
|                 for (uint8_t x = 0; x < width; x++) | ||||
|                 { | ||||
|                     pixels[y * width + x] = 0; | ||||
|                 } | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             if (offset > 0) | ||||
|             { | ||||
|                 if (gridBW.pixels[y]) | ||||
|                 { | ||||
|                     for (uint8_t x = 0; x < width; x++) | ||||
|                     { | ||||
|                         pixels[(y + offset) * width + x] = pixels[y * width + x]; | ||||
|                         pixels[y * width + x] = 0; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         gridBW.cleanupFullLines(); | ||||
|     } | ||||
|  | ||||
|     uint8_t* getPixel(uint8_t x, uint8_t y) | ||||
|     { | ||||
|         return &pixels[y * width + x]; | ||||
|     } | ||||
|  | ||||
|     void sync() | ||||
|     { | ||||
|         for (uint8_t y = 0; y < height; y++) | ||||
|         { | ||||
|             gridBW.pixels[y] = 0x0; | ||||
|             for (int8_t x = 0; x < width; x++) | ||||
|             { | ||||
|                 gridBW.pixels[y] <<= 1; | ||||
|                 if (*getPixel(x, y) != 0) | ||||
|                 { | ||||
|                     gridBW.pixels[y] |= 0x1; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
| #endif /* __GRIDCOLOR_H__ */ | ||||
							
								
								
									
										184
									
								
								usermods/TetrisAI_v2/pieces.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										184
									
								
								usermods/TetrisAI_v2/pieces.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,184 @@ | ||||
| /****************************************************************************** | ||||
|   * @file           : pieces.h | ||||
|   * @brief          : contains the tetris pieces with their colors indecies | ||||
|   ****************************************************************************** | ||||
|   * @attention | ||||
|   * | ||||
|   * Copyright (c) muebau 2022 | ||||
|   * All rights reserved.</center></h2> | ||||
|   * | ||||
|   ****************************************************************************** | ||||
| */ | ||||
|  | ||||
| #ifndef __PIECES_H__ | ||||
| #define __PIECES_H__ | ||||
|  | ||||
| #include <stdint.h> | ||||
| #include <stdbool.h> | ||||
|  | ||||
| #include <bitset> | ||||
| #include <cstddef> | ||||
| #include <cassert> | ||||
| #include <iostream> | ||||
|  | ||||
| #define numPieces 7 | ||||
|  | ||||
| struct PieceRotation | ||||
| { | ||||
|     uint8_t width; | ||||
|     uint8_t height; | ||||
|     uint16_t rows; | ||||
| }; | ||||
|  | ||||
| struct PieceData | ||||
| { | ||||
|     uint8_t rotCount; | ||||
|     uint8_t colorIndex; | ||||
|     PieceRotation rotations[4]; | ||||
| }; | ||||
|  | ||||
| PieceData piecesData[numPieces] = { | ||||
|     // I | ||||
|     { | ||||
|             2, | ||||
|             1, | ||||
|             { | ||||
|                 { 1, 4, 0b0001000100010001}, | ||||
|                 { 4, 1, 0b0000000000001111} | ||||
|             } | ||||
|     }, | ||||
|     // O | ||||
|     { | ||||
|             1, | ||||
|             2, | ||||
|             { | ||||
|                 { 2, 2, 0b0000000000110011} | ||||
|             } | ||||
|     }, | ||||
|     // Z | ||||
|     { | ||||
|             2, | ||||
|             3, | ||||
|             { | ||||
|                 { 3, 2, 0b0000000001100011}, | ||||
|                 { 2, 3, 0b0000000100110010} | ||||
|             } | ||||
|     }, | ||||
|     // S | ||||
|     { | ||||
|             2, | ||||
|             4, | ||||
|             { | ||||
|                 { 3, 2, 0b0000000000110110}, | ||||
|                 { 2, 3, 0b0000001000110001} | ||||
|             } | ||||
|     }, | ||||
|     // L | ||||
|     { | ||||
|             4, | ||||
|             5, | ||||
|             { | ||||
|                 { 2, 3, 0b0000001000100011}, | ||||
|                 { 3, 2, 0b0000000001110100}, | ||||
|                 { 2, 3, 0b0000001100010001}, | ||||
|                 { 3, 2, 0b0000000000010111} | ||||
|             } | ||||
|     }, | ||||
|     // J | ||||
|     { | ||||
|             4, | ||||
|             6, | ||||
|             { | ||||
|                 { 2, 3, 0b0000000100010011}, | ||||
|                 { 3, 2, 0b0000000001000111}, | ||||
|                 { 2, 3, 0b0000001100100010}, | ||||
|                 { 3, 2, 0b0000000001110001} | ||||
|             } | ||||
|     }, | ||||
|     // T | ||||
|     { | ||||
|             4, | ||||
|             7, | ||||
|             { | ||||
|                 { 3, 2, 0b0000000001110010}, | ||||
|                 { 2, 3, 0b0000000100110001}, | ||||
|                 { 3, 2, 0b0000000000100111}, | ||||
|                 { 2, 3, 0b0000001000110010} | ||||
|             } | ||||
|     }, | ||||
| }; | ||||
|  | ||||
| class Piece | ||||
| { | ||||
| private: | ||||
| public: | ||||
|     uint8_t x; | ||||
|     uint8_t y; | ||||
|     PieceData* pieceData; | ||||
|     uint8_t rotation; | ||||
|     uint8_t landingY; | ||||
|  | ||||
|     Piece(uint8_t pieceIndex = 0): | ||||
|         x(0), | ||||
|         y(0), | ||||
|         rotation(0), | ||||
|         landingY(0) | ||||
|     { | ||||
|         this->pieceData = &piecesData[pieceIndex]; | ||||
|     } | ||||
|  | ||||
|     void reset() | ||||
|     { | ||||
|         this->rotation = 0; | ||||
|         this->x = 0; | ||||
|         this->y = 0; | ||||
|         this->landingY = 0; | ||||
|     } | ||||
|  | ||||
|     uint32_t getGridRow(uint8_t x, uint8_t y, uint8_t width) | ||||
|     { | ||||
|         if (x < width) | ||||
|         { | ||||
|             //shift the row with the "top-left" position to the "x" position | ||||
|             auto shiftx = (width - 1) - x; | ||||
|             auto topleftx = (getRotation().width - 1); | ||||
|  | ||||
|             auto finalShift = shiftx - topleftx; | ||||
|             auto row = getRow(y); | ||||
|             auto finalResult = row << finalShift; | ||||
|  | ||||
|             return finalResult; | ||||
|         } | ||||
|         return 0xffffffff; | ||||
|     } | ||||
|  | ||||
|     uint8_t getRow(uint8_t y) | ||||
|     { | ||||
|         if (y < 4) | ||||
|         { | ||||
|             return (getRotation().rows >> (12 - (4 * y))) & 0xf; | ||||
|         } | ||||
|         return 0xf; | ||||
|     } | ||||
|  | ||||
|     bool getPixel(uint8_t x, uint8_t y) | ||||
|     { | ||||
|         if(x > getRotation().width - 1 || y > getRotation().height - 1 ) | ||||
|         { | ||||
|             return false; | ||||
|         } | ||||
|          | ||||
|         if (x < 4 && y < 4) | ||||
|         { | ||||
|             return (getRow((4 - getRotation().height) + y) >> (3 - ((4 - getRotation().width) + x))) & 0x1; | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     PieceRotation getRotation() | ||||
|     { | ||||
|         return this->pieceData->rotations[rotation]; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| #endif /* __PIECES_H__ */ | ||||
							
								
								
									
										64
									
								
								usermods/TetrisAI_v2/rating.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								usermods/TetrisAI_v2/rating.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | ||||
| /****************************************************************************** | ||||
|   * @file           : rating.h | ||||
|   * @brief          : contains the tetris rating of a grid | ||||
|   ****************************************************************************** | ||||
|   * @attention | ||||
|   * | ||||
|   * Copyright (c) muebau 2022 | ||||
|   * All rights reserved.</center></h2> | ||||
|   * | ||||
|   ****************************************************************************** | ||||
| */ | ||||
|  | ||||
| #ifndef __RATING_H__ | ||||
| #define __RATING_H__ | ||||
|  | ||||
| #include <stdint.h> | ||||
| #include <float.h> | ||||
| #include <stdbool.h> | ||||
| #include <math.h> | ||||
| #include <vector> | ||||
| #include "rating.h" | ||||
|  | ||||
| using namespace std; | ||||
|  | ||||
| class Rating | ||||
| { | ||||
| private: | ||||
| public: | ||||
|     uint8_t minHeight; | ||||
|     uint8_t maxHeight; | ||||
|     uint16_t holes; | ||||
|     uint8_t fullLines; | ||||
|     uint16_t bumpiness; | ||||
|     uint16_t aggregatedHeight; | ||||
|     double score; | ||||
|     uint8_t width; | ||||
|     std::vector<uint8_t> lineHights; | ||||
|  | ||||
|     Rating(uint8_t width): | ||||
|         width(width), | ||||
|         lineHights(width) | ||||
|     { | ||||
|         reset(); | ||||
|     } | ||||
|  | ||||
|     void reset() | ||||
|     { | ||||
|         this->minHeight = 0; | ||||
|         this->maxHeight = 0; | ||||
|  | ||||
|         for (uint8_t line = 0; line < this->width; line++) | ||||
|         { | ||||
|             this->lineHights[line] = 0; | ||||
|         } | ||||
|  | ||||
|         this->holes = 0; | ||||
|         this->fullLines = 0; | ||||
|         this->bumpiness = 0; | ||||
|         this->aggregatedHeight = 0; | ||||
|         this->score = -DBL_MAX; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| #endif /* __RATING_H__ */ | ||||
							
								
								
									
										33
									
								
								usermods/TetrisAI_v2/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								usermods/TetrisAI_v2/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| # Tetris AI effect usermod | ||||
|  | ||||
| This usermod brings you a effect brings a self playing Tetris game. The mod needs version 0.14 or above as it is based on matrix support. The effect was tested on an ESP32 with a WS2812B 16x16 matrix. | ||||
|  | ||||
| Version 1.0 | ||||
|  | ||||
| ## Installation  | ||||
|  | ||||
| Just activate the usermod with `-D USERMOD_TETRISAI` and the effect will become available under the name 'Tetris AI'. | ||||
|  | ||||
| ## Usage | ||||
|  | ||||
| It is best to set the background color to black, the border color to light grey and the game over color (foreground) to dark grey. | ||||
|  | ||||
| ### Sliders and boxes | ||||
|  | ||||
| #### Sliders | ||||
|  | ||||
| * speed: speed the game plays | ||||
| * look ahead: how many pieces is the AI allowed to know the next pieces (0 - 2) | ||||
| * intelligence: how good the AI will play | ||||
| * Rotate color: make the colors shift (rotate) every few cicles | ||||
| * Mistakes free: how many good moves between mistakes (if activated) | ||||
|  | ||||
| #### Checkboxes | ||||
|  | ||||
| * show next: if true a space of 5 pixels from the right is used to show the next pieces. The whole segment is used for the grid otherwise. | ||||
| * show border: if true an additional column of 1 pixel is used to draw a border between the grid and the next pieces | ||||
| * mistakes: if true the worst instead of the best move is choosen every few moves (read above) | ||||
|  | ||||
| ## Best results | ||||
|  | ||||
|  If the speed is set to be a little bit faster than a good human could play with maximal intelligence and very few mistakes it makes people furious/happy at a party. | ||||
							
								
								
									
										302
									
								
								usermods/TetrisAI_v2/tetrisai.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										302
									
								
								usermods/TetrisAI_v2/tetrisai.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,302 @@ | ||||
| /****************************************************************************** | ||||
|   * @file           : ai.h | ||||
|   * @brief          : contains the heuristic | ||||
|   ****************************************************************************** | ||||
|   * @attention | ||||
|   * | ||||
|   * Copyright (c) muebau 2023 | ||||
|   * All rights reserved.</center></h2> | ||||
|   * | ||||
|   ****************************************************************************** | ||||
| */ | ||||
|  | ||||
| #ifndef __AI_H__ | ||||
| #define __AI_H__ | ||||
|  | ||||
| #include "gridbw.h" | ||||
| #include "rating.h" | ||||
|  | ||||
| using namespace std; | ||||
|  | ||||
| class TetrisAI | ||||
| { | ||||
| private: | ||||
| public: | ||||
|     double aHeight; | ||||
|     double fullLines; | ||||
|     double holes; | ||||
|     double bumpiness; | ||||
|     bool findWorstMove = false; | ||||
|  | ||||
|     uint8_t countOnes(uint32_t vector) | ||||
|     { | ||||
|         uint8_t count = 0; | ||||
|         while (vector) | ||||
|         { | ||||
|             vector &= (vector - 1); | ||||
|             count++; | ||||
|         } | ||||
|         return count; | ||||
|     } | ||||
|  | ||||
|     void updateRating(GridBW grid, Rating* rating) | ||||
|     { | ||||
|         rating->minHeight = 0; | ||||
|         rating->maxHeight = 0; | ||||
|         rating->holes = 0; | ||||
|         rating->fullLines = 0; | ||||
|         rating->bumpiness = 0; | ||||
|         rating->aggregatedHeight = 0; | ||||
|         fill(rating->lineHights.begin(), rating->lineHights.end(), 0); | ||||
|  | ||||
|         uint32_t columnvector = 0x0; | ||||
|         uint32_t lastcolumnvector = 0x0; | ||||
|         for (uint8_t row = 0; row < grid.height; row++) | ||||
|         { | ||||
|             columnvector |= grid.pixels[row]; | ||||
|  | ||||
|             //first (highest) column makes it | ||||
|             if (rating->maxHeight == 0 && columnvector) | ||||
|             { | ||||
|                 rating->maxHeight = grid.height - row; | ||||
|             } | ||||
|  | ||||
|             //if column vector is full we found the minimal height (or it stays zero) | ||||
|             if (rating->minHeight == 0 && (columnvector == (uint32_t)((1 << grid.width) - 1))) | ||||
|             { | ||||
|                 rating->minHeight = grid.height - row; | ||||
|             } | ||||
|  | ||||
|             //line full if all ones in mask :-) | ||||
|             if (grid.isLineFull(row)) | ||||
|             { | ||||
|                 rating->fullLines++; | ||||
|             } | ||||
|  | ||||
|             //holes are basically a XOR with the "full" columns | ||||
|             rating->holes += countOnes(columnvector ^ grid.pixels[row]); | ||||
|  | ||||
|             //calculate the difference (XOR) between the current column vector and the last one | ||||
|             uint32_t columnDelta = columnvector ^ lastcolumnvector; | ||||
|  | ||||
|             //process every new column | ||||
|             uint8_t index = 0; | ||||
|             while (columnDelta) | ||||
|             { | ||||
|                 //if this is a new column | ||||
|                 if (columnDelta & 0x1) | ||||
|                 { | ||||
|                     //update hight of this column | ||||
|                     rating->lineHights[(grid.width - 1) - index] = grid.height - row; | ||||
|  | ||||
|                     // update aggregatedHeight | ||||
|                     rating->aggregatedHeight += grid.height - row; | ||||
|                 } | ||||
|                 index++; | ||||
|                 columnDelta >>= 1; | ||||
|             } | ||||
|             lastcolumnvector = columnvector; | ||||
|         } | ||||
|  | ||||
|         //compare every two columns to get the difference and add them up | ||||
|         for (uint8_t column = 1; column < grid.width; column++) | ||||
|         { | ||||
|             rating->bumpiness += abs(rating->lineHights[column - 1] - rating->lineHights[column]); | ||||
|         } | ||||
|  | ||||
|         rating->score = (aHeight * (rating->aggregatedHeight)) + (fullLines * (rating->fullLines)) + (holes * (rating->holes)) + (bumpiness * (rating->bumpiness)); | ||||
|     } | ||||
|  | ||||
|     TetrisAI(): TetrisAI(-0.510066, 0.760666, -0.35663, -0.184483) | ||||
|     {} | ||||
|  | ||||
|     TetrisAI(double aHeight, double fullLines, double holes, double bumpiness): | ||||
|         aHeight(aHeight), | ||||
|         fullLines(fullLines), | ||||
|         holes(holes), | ||||
|         bumpiness(bumpiness) | ||||
|     {} | ||||
|  | ||||
|     void findBestMove(GridBW grid, Piece *piece) | ||||
|     { | ||||
|         vector<Piece> pieces = {*piece}; | ||||
|         findBestMove(grid, &pieces); | ||||
|         *piece = pieces[0]; | ||||
|     } | ||||
|  | ||||
|     void findBestMove(GridBW grid, std::vector<Piece> *pieces) | ||||
|     { | ||||
|         findBestMove(grid, pieces->begin(), pieces->end()); | ||||
|     } | ||||
|  | ||||
|     void findBestMove(GridBW grid, std::vector<Piece>::iterator start, std::vector<Piece>::iterator end) | ||||
|     { | ||||
|         Rating bestRating(grid.width); | ||||
|         findBestMove(grid, start, end, &bestRating); | ||||
|     } | ||||
|  | ||||
|     void findBestMove(GridBW grid, std::vector<Piece>::iterator start, std::vector<Piece>::iterator end, Rating* bestRating) | ||||
|     { | ||||
|         grid.cleanupFullLines(); | ||||
|         Rating curRating(grid.width); | ||||
|         Rating deeperRating(grid.width); | ||||
|         Piece piece = *start; | ||||
|  | ||||
|         // for every rotation of the piece | ||||
|         for (piece.rotation = 0; piece.rotation < piece.pieceData->rotCount; piece.rotation++) | ||||
|         { | ||||
|             // put piece to top left corner | ||||
|             piece.x = 0; | ||||
|             piece.y = 0; | ||||
|  | ||||
|             //test for every column | ||||
|             for (piece.x = 0; piece.x <= grid.width - piece.getRotation().width; piece.x++) | ||||
|             { | ||||
|                 //todo optimise by the use of the previous grids height | ||||
|                 piece.landingY = 0; | ||||
|                 //will set landingY to final position | ||||
|                 grid.findLandingPosition(&piece); | ||||
|  | ||||
|                 // draw piece | ||||
|                 grid.placePiece(&piece, piece.x, piece.landingY); | ||||
|  | ||||
|                 if(start == end - 1) | ||||
|                 { | ||||
|                     //at the deepest level | ||||
|                     updateRating(grid, &curRating); | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     //go deeper to take another piece into account | ||||
|                     findBestMove(grid, start + 1, end, &deeperRating); | ||||
|                     curRating = deeperRating; | ||||
|                 } | ||||
|  | ||||
|                 // eraese piece | ||||
|                 grid.erasePiece(&piece, piece.x, piece.landingY); | ||||
|  | ||||
|                 if(findWorstMove) | ||||
|                 { | ||||
|                     //init rating for worst | ||||
|                     if(bestRating->score == -DBL_MAX) | ||||
|                     { | ||||
|                         bestRating->score = DBL_MAX; | ||||
|                     } | ||||
|  | ||||
|                     // update if we found a worse one | ||||
|                     if (bestRating->score > curRating.score) | ||||
|                     { | ||||
|                         *bestRating = curRating; | ||||
|                         (*start) = piece; | ||||
|                     } | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     // update if we found a better one | ||||
|                     if (bestRating->score < curRating.score) | ||||
|                     { | ||||
|                         *bestRating = curRating; | ||||
|                         (*start) = piece; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     bool findBestMoveNonBlocking(GridBW grid, std::vector<Piece>::iterator start, std::vector<Piece>::iterator end, Rating* bestRating) | ||||
|     { | ||||
|         //vector with pieces | ||||
|         //for every piece | ||||
|             //for every  | ||||
|         switch (expression) | ||||
|         { | ||||
|             case INIT: | ||||
|                 break; | ||||
|  | ||||
|             default: | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     bool findBestMoveNonBlocking(GridBW grid, std::vector<Piece>::iterator start, std::vector<Piece>::iterator end, Rating* bestRating) | ||||
|     { | ||||
|         //INIT | ||||
|         grid.cleanupFullLines(); | ||||
|         Rating curRating(grid.width); | ||||
|         Rating deeperRating(grid.width); | ||||
|         Piece piece = *start; | ||||
|  | ||||
|         // for every rotation of the piece | ||||
|         piece.rotation = 0; | ||||
|  | ||||
|         //HANDLE | ||||
|         while (piece.rotation < piece.pieceData->rotCount) | ||||
|         { | ||||
|             // put piece to top left corner | ||||
|             piece.x = 0; | ||||
|             piece.y = 0; | ||||
|  | ||||
|             //test for every column | ||||
|             piece.x = 0; | ||||
|             while (piece.x <= grid.width - piece.getRotation().width) | ||||
|             { | ||||
|  | ||||
|                 //todo optimise by the use of the previous grids height | ||||
|                 piece.landingY = 0; | ||||
|                 //will set landingY to final position | ||||
|                 grid.findLandingPosition(&piece); | ||||
|  | ||||
|                 // draw piece | ||||
|                 grid.placePiece(&piece, piece.x, piece.landingY); | ||||
|  | ||||
|                 if(start == end - 1) | ||||
|                 { | ||||
|                     //at the deepest level | ||||
|                     updateRating(grid, &curRating); | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     //go deeper to take another piece into account | ||||
|                     findBestMove(grid, start + 1, end, &deeperRating); | ||||
|                     curRating = deeperRating; | ||||
|                 } | ||||
|  | ||||
|                 // eraese piece | ||||
|                 grid.erasePiece(&piece, piece.x, piece.landingY); | ||||
|  | ||||
|                 if(findWorstMove) | ||||
|                 { | ||||
|                     //init rating for worst | ||||
|                     if(bestRating->score == -DBL_MAX) | ||||
|                     { | ||||
|                         bestRating->score = DBL_MAX; | ||||
|                     } | ||||
|  | ||||
|                     // update if we found a worse one | ||||
|                     if (bestRating->score > curRating.score) | ||||
|                     { | ||||
|                         *bestRating = curRating; | ||||
|                         (*start) = piece; | ||||
|                     } | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     // update if we found a better one | ||||
|                     if (bestRating->score < curRating.score) | ||||
|                     { | ||||
|                         *bestRating = curRating; | ||||
|                         (*start) = piece; | ||||
|                     } | ||||
|                 } | ||||
|                 piece.x++; | ||||
|             } | ||||
|             piece.rotation++; | ||||
|         } | ||||
|  | ||||
|         //EXIT | ||||
|  | ||||
|         return true; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| #endif /* __AI_H__ */ | ||||
							
								
								
									
										150
									
								
								usermods/TetrisAI_v2/tetrisaigame.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										150
									
								
								usermods/TetrisAI_v2/tetrisaigame.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,150 @@ | ||||
| /****************************************************************************** | ||||
|   * @file           : tetrisaigame.h | ||||
|   * @brief          : main tetris functions | ||||
|   ****************************************************************************** | ||||
|   * @attention | ||||
|   * | ||||
|   * Copyright (c) muebau 2022 | ||||
|   * All rights reserved.</center></h2> | ||||
|   * | ||||
|   ****************************************************************************** | ||||
| */ | ||||
|  | ||||
| #ifndef __TETRISAIGAME_H__ | ||||
| #define __TETRISAIGAME_H__ | ||||
|  | ||||
| #include <stdint.h> | ||||
| #include <stdbool.h> | ||||
| #include <vector> | ||||
| #include "pieces.h" | ||||
| #include "gridcolor.h" | ||||
| #include "tetrisbag.h" | ||||
| #include "tetrisai.h" | ||||
|  | ||||
| using namespace std; | ||||
|  | ||||
| class TetrisAIGame | ||||
| { | ||||
| private: | ||||
|     bool animateFallOfPiece(Piece* piece, bool skip) | ||||
|     { | ||||
|         if (skip || piece->y >= piece->landingY) | ||||
|         { | ||||
|             piece->y = piece->landingY; | ||||
|             grid.gridBW.placePiece(piece, piece->x, piece->landingY); | ||||
|             grid.placePiece(piece, piece->x, piece->y); | ||||
|             return false; | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             // eraese last drawing | ||||
|             grid.erasePiece(piece, piece->x, piece->y); | ||||
|  | ||||
|             //move piece down | ||||
|             piece->y++; | ||||
|  | ||||
|             // draw piece | ||||
|             grid.placePiece(piece, piece->x, piece->y); | ||||
|  | ||||
|             return true; | ||||
|         } | ||||
|     } | ||||
|  | ||||
| public: | ||||
|     uint8_t width; | ||||
|     uint8_t height; | ||||
|     uint8_t nLookAhead; | ||||
|     TetrisBag bag; | ||||
|     GridColor grid; | ||||
|     TetrisAI ai; | ||||
|     Piece curPiece; | ||||
|     PieceData* piecesData; | ||||
|     enum States { INIT, TEST_GAME_OVER, GET_NEXT_PIECE, FIND_BEST_MOVE, ANIMATE_MOVE, ANIMATE_GAME_OVER } state = INIT; | ||||
|  | ||||
|     TetrisAIGame(uint8_t width, uint8_t height, uint8_t nLookAhead, PieceData* piecesData, uint8_t nPieces): | ||||
|         width(width), | ||||
|         height(height), | ||||
|         nLookAhead(nLookAhead), | ||||
|         bag(nPieces, 1, nLookAhead), | ||||
|         grid(width, height + 4), | ||||
|         ai(), | ||||
|         piecesData(piecesData) | ||||
|     { | ||||
|     } | ||||
|  | ||||
|     void nextPiece() | ||||
|     { | ||||
|         grid.cleanupFullLines(); | ||||
|         bag.queuePiece(); | ||||
|     } | ||||
|  | ||||
|     void findBestMove() | ||||
|     { | ||||
|         ai.findBestMove(grid.gridBW, &bag.piecesQueue); | ||||
|     } | ||||
|  | ||||
|     bool animateFall(bool skip) | ||||
|     { | ||||
|         return animateFallOfPiece(&(bag.piecesQueue[0]), skip); | ||||
|     } | ||||
|  | ||||
|     bool isGameOver() | ||||
|     { | ||||
|         //if there is something in the 4 lines of the hidden area the game is over | ||||
|         return grid.gridBW.pixels[0] || grid.gridBW.pixels[1] || grid.gridBW.pixels[2] || grid.gridBW.pixels[3]; | ||||
|     } | ||||
|  | ||||
|     void poll() | ||||
|     { | ||||
|         switch (state) | ||||
|         { | ||||
|             case INIT: | ||||
|                 reset(); | ||||
|                 state = TEST_GAME_OVER; | ||||
|                 break; | ||||
|             case TEST_GAME_OVER: | ||||
|                 if (isGameOver()) | ||||
|                 { | ||||
|                     state = ANIMATE_GAME_OVER; | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     state = GET_NEXT_PIECE; | ||||
|                 } | ||||
|                 break; | ||||
|             case GET_NEXT_PIECE: | ||||
|                 nextPiece(); | ||||
|                 state = FIND_BEST_MOVE; | ||||
|                 break; | ||||
|             case FIND_BEST_MOVE: | ||||
|                 findBestMove(); | ||||
|                 state = ANIMATE_MOVE; | ||||
|                 break; | ||||
|             case ANIMATE_MOVE: | ||||
|                 if (!animateFall(false)) | ||||
|                 { | ||||
|                     state = TEST_GAME_OVER; | ||||
|                 } | ||||
|                 break; | ||||
|             case ANIMATE_GAME_OVER: | ||||
|                 static auto curPixel = grid.pixels.size(); | ||||
|                 grid.pixels[curPixel] = 254; | ||||
|  | ||||
|                 if (curPixel == 0) | ||||
|                 { | ||||
|                     state = INIT; | ||||
|                     curPixel = grid.pixels.size(); | ||||
|                 } | ||||
|                 curPixel--; | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void reset() | ||||
|     { | ||||
|         grid.clear(); | ||||
|         bag.init(); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| #endif /* __TETRISAIGAME_H__ */ | ||||
							
								
								
									
										100
									
								
								usermods/TetrisAI_v2/tetrisbag.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								usermods/TetrisAI_v2/tetrisbag.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | ||||
| /****************************************************************************** | ||||
|   * @file           : tetrisbag.h | ||||
|   * @brief          : the tetris implementation of a random piece generator | ||||
|   ****************************************************************************** | ||||
|   * @attention | ||||
|   * | ||||
|   * Copyright (c) muebau 2022 | ||||
|   * All rights reserved.</center></h2> | ||||
|   * | ||||
|   ****************************************************************************** | ||||
| */ | ||||
|  | ||||
| #ifndef __TETRISBAG_H__ | ||||
| #define __TETRISBAG_H__ | ||||
|  | ||||
| #include <stdint.h> | ||||
| #include <vector> | ||||
| #include <algorithm> | ||||
|  | ||||
| #include "tetrisbag.h" | ||||
|  | ||||
| class TetrisBag | ||||
| { | ||||
| private: | ||||
| public: | ||||
|     uint8_t nPieces; | ||||
|     uint8_t nBagLength; | ||||
|     uint8_t bagIdx; | ||||
|     std::vector<uint8_t> bag; | ||||
|     std::vector<Piece> piecesQueue; | ||||
|  | ||||
|     TetrisBag(uint8_t nPieces, uint8_t nBagLength, uint8_t queueLength): | ||||
|         nPieces(nPieces), | ||||
|         nBagLength(nBagLength), | ||||
|         bag(nPieces * nBagLength), | ||||
|         piecesQueue(queueLength) | ||||
|     { | ||||
|         init(); | ||||
|     } | ||||
|  | ||||
|     void init() | ||||
|     { | ||||
|         //will shuffle the bag at first use | ||||
|         bagIdx = nPieces - 1; | ||||
|  | ||||
|         for (uint8_t bagIndex = 0; bagIndex < nPieces * nBagLength; bagIndex++) | ||||
|         { | ||||
|             bag[bagIndex] = bagIndex % nPieces; | ||||
|         } | ||||
|  | ||||
|         //will init the queue | ||||
|         for (uint8_t index = 0; index < piecesQueue.size(); index++) | ||||
|         { | ||||
|             queuePiece(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void shuffleBag() | ||||
|     { | ||||
|         uint8_t temp; | ||||
|         uint8_t swapIdx; | ||||
|         for (int index = nPieces - 1; index > 0; index--) | ||||
|         { | ||||
|             //get candidate to swap | ||||
|             swapIdx = rand() % index; | ||||
|  | ||||
|             //swap it! | ||||
|             temp = bag[swapIdx]; | ||||
|             bag[swapIdx] = bag[index]; | ||||
|             bag[index] = temp; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     Piece getNextPiece() | ||||
|     { | ||||
|         bagIdx++; | ||||
|         if (bagIdx >= nPieces) | ||||
|         { | ||||
|             shuffleBag(); | ||||
|             bagIdx = 0; | ||||
|         } | ||||
|         return Piece(bag[bagIdx]); | ||||
|     } | ||||
|  | ||||
|     void queuePiece() | ||||
|     { | ||||
|         //move vector to left | ||||
|         std::rotate(piecesQueue.begin(), piecesQueue.begin() + 1, piecesQueue.end()); | ||||
|         piecesQueue[piecesQueue.size() - 1] = getNextPiece(); | ||||
|     } | ||||
|  | ||||
|     void queuePiece(uint8_t idx) | ||||
|     { | ||||
|         //move vector to left | ||||
|         std::rotate(piecesQueue.begin(), piecesQueue.begin() + 1, piecesQueue.end()); | ||||
|         piecesQueue[piecesQueue.size() - 1] = Piece(idx % nPieces); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| #endif /* __TETRISBAG_H__ */ | ||||
							
								
								
									
										222
									
								
								usermods/TetrisAI_v2/usermod_v2_tetrisai.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										222
									
								
								usermods/TetrisAI_v2/usermod_v2_tetrisai.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,222 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
| #include "FX.h" | ||||
| #include "fcn_declare.h" | ||||
|  | ||||
| #include "tetrisaigame.h" | ||||
| // By: muebau | ||||
|  | ||||
| typedef struct TetrisAI_data | ||||
| { | ||||
|   unsigned long lastTime = 0; | ||||
|   TetrisAIGame tetris; | ||||
|   uint8_t   intelligence; | ||||
|   uint8_t   rotate; | ||||
|   bool      showNext; | ||||
|   bool      showBorder; | ||||
|   uint8_t   colorOffset; | ||||
|   uint8_t   colorInc; | ||||
|   uint8_t   mistaceCountdown; | ||||
| } tetrisai_data; | ||||
|  | ||||
| void drawGrid(TetrisAIGame* tetris, TetrisAI_data* tetrisai_data) | ||||
| { | ||||
|   SEGMENT.fill(SEGCOLOR(1)); | ||||
|  | ||||
|   //GRID | ||||
|   for (auto index_y = 4; index_y < tetris->grid.height; index_y++) | ||||
|   { | ||||
|     for (auto index_x = 0; index_x < tetris->grid.width; index_x++) | ||||
|     { | ||||
|       CRGB color; | ||||
|       if (*tetris->grid.getPixel(index_x, index_y) == 0) | ||||
|       { | ||||
|         //BG color | ||||
|         color = SEGCOLOR(1); | ||||
|       } | ||||
|       //game over animation | ||||
|       else if(*tetris->grid.getPixel(index_x, index_y) == 254) | ||||
|       { | ||||
|         //use fg | ||||
|         color = SEGCOLOR(0); | ||||
|       } | ||||
|       else | ||||
|       { | ||||
|         //spread the color over the whole palette | ||||
|         uint8_t colorIndex = *tetris->grid.getPixel(index_x, index_y) * 32; | ||||
|         colorIndex += tetrisai_data->colorOffset; | ||||
|         color = ColorFromPalette(SEGPALETTE, colorIndex, 255, NOBLEND); | ||||
|       } | ||||
|  | ||||
|       SEGMENT.setPixelColorXY(index_x, index_y - 4, color); | ||||
|     } | ||||
|   } | ||||
|   tetrisai_data->colorOffset += tetrisai_data->colorInc; | ||||
|  | ||||
|   //NEXT PIECE AREA | ||||
|   if (tetrisai_data->showNext) | ||||
|   { | ||||
|     //BORDER | ||||
|     if (tetrisai_data->showBorder) | ||||
|     { | ||||
|       //draw a line 6 pixels from right with the border color | ||||
|       for (auto index_y = 0; index_y < SEGMENT.virtualHeight(); index_y++) | ||||
|       { | ||||
|         SEGMENT.setPixelColorXY(SEGMENT.virtualWidth() - 6, index_y, SEGCOLOR(2)); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     //NEXT PIECE | ||||
|     int piecesOffsetX = SEGMENT.virtualWidth() - 4; | ||||
|     int piecesOffsetY = 1; | ||||
|     for (uint8_t nextPieceIdx = 1; nextPieceIdx < tetris->nLookAhead; nextPieceIdx++) | ||||
|     { | ||||
|       uint8_t pieceNbrOffsetY = (nextPieceIdx - 1) * 5; | ||||
|  | ||||
|       Piece piece(tetris->bag.piecesQueue[nextPieceIdx]); | ||||
|  | ||||
|       for (uint8_t pieceY = 0; pieceY < piece.getRotation().height; pieceY++) | ||||
|       { | ||||
|         for (uint8_t pieceX = 0; pieceX < piece.getRotation().width; pieceX++) | ||||
|         { | ||||
|           if (piece.getPixel(pieceX, pieceY)) | ||||
|           { | ||||
|             uint8_t colIdx = ((piece.pieceData->colorIndex * 32) + tetrisai_data->colorOffset); | ||||
|             SEGMENT.setPixelColorXY(piecesOffsetX + pieceX, piecesOffsetY + pieceNbrOffsetY + pieceY, ColorFromPalette(SEGPALETTE, colIdx, 255, NOBLEND)); | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| //////////////////////////// | ||||
| //     2D Tetris AI       // | ||||
| //////////////////////////// | ||||
| uint16_t mode_2DTetrisAI() | ||||
| { | ||||
|   if (!strip.isMatrix || !SEGENV.allocateData(sizeof(tetrisai_data))) | ||||
|   { | ||||
|     // not a 2D set-up | ||||
|     SEGMENT.fill(SEGCOLOR(0)); | ||||
|     return 350; | ||||
|   } | ||||
|   TetrisAI_data* tetrisai_data = reinterpret_cast<TetrisAI_data*>(SEGENV.data); | ||||
|  | ||||
|   const uint16_t cols = SEGMENT.virtualWidth(); | ||||
|   const uint16_t rows = SEGMENT.virtualHeight(); | ||||
|  | ||||
|   //range 0 - 1024ms => 1024/255 ~ 4 | ||||
|   uint16_t msDelayMove = 1024 - (4 * SEGMENT.speed); | ||||
|   int16_t msDelayGameOver = msDelayMove / 4; | ||||
|  | ||||
|   //range 0 - 2 (not including current) | ||||
|   uint8_t nLookAhead = SEGMENT.intensity ? (SEGMENT.intensity >> 7) + 2 : 1; | ||||
|   //range 0 - 16 | ||||
|   tetrisai_data->colorInc = SEGMENT.custom2 >> 4; | ||||
|  | ||||
|   if (!tetrisai_data->tetris || (tetrisai_data->tetris.nLookAhead != nLookAhead | ||||
|     || tetrisai_data->showNext != SEGMENT.check1 | ||||
|     || tetrisai_data->showBorder != SEGMENT.check2 | ||||
|       ) | ||||
|     ) | ||||
|   { | ||||
|     tetrisai_data->showNext = SEGMENT.check1; | ||||
|     tetrisai_data->showBorder = SEGMENT.check2; | ||||
|  | ||||
|     //not more than 32 as this is the limit of this implementation | ||||
|     uint8_t gridWidth = cols < 32 ? cols : 32; | ||||
|     uint8_t gridHeight = rows; | ||||
|  | ||||
|     // do we need space for the 'next' section? | ||||
|     if (tetrisai_data->showNext) | ||||
|     { | ||||
|       // make space for the piece and one pixel of space | ||||
|       gridWidth = gridWidth - 5; | ||||
|  | ||||
|       // do we need space for a border? | ||||
|       if (tetrisai_data->showBorder) | ||||
|       { | ||||
|         gridWidth = gridWidth - 1; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     tetrisai_data->tetris = TetrisAIGame(gridWidth, gridHeight, nLookAhead, piecesData, numPieces); | ||||
|     SEGMENT.fill(SEGCOLOR(1)); | ||||
|   } | ||||
|  | ||||
|   if (tetrisai_data->intelligence != SEGMENT.custom1) | ||||
|   { | ||||
|     tetrisai_data->intelligence = SEGMENT.custom1; | ||||
|     double dui = 0.2 - (0.2 * (tetrisai_data->intelligence / 255.0)); | ||||
|  | ||||
|     tetrisai_data->tetris.ai.aHeight = -0.510066 + dui; | ||||
|     tetrisai_data->tetris.ai.fullLines = 0.760666 - dui; | ||||
|     tetrisai_data->tetris.ai.holes = -0.35663 + dui; | ||||
|     tetrisai_data->tetris.ai.bumpiness = -0.184483 + dui; | ||||
|   } | ||||
|  | ||||
|   if (tetrisai_data->tetris.state == TetrisAIGame::ANIMATE_MOVE) | ||||
|   { | ||||
|     if (millis() - tetrisai_data->lastTime > msDelayMove) | ||||
|     { | ||||
|       drawGrid(&tetrisai_data->tetris, tetrisai_data); | ||||
|       tetrisai_data->lastTime = millis(); | ||||
|       tetrisai_data->tetris.poll(); | ||||
|     } | ||||
|   } | ||||
|   else if (tetrisai_data->tetris.state == TetrisAIGame::ANIMATE_GAME_OVER) | ||||
|   { | ||||
|     if (millis() - tetrisai_data->lastTime > msDelayGameOver) | ||||
|     { | ||||
|       drawGrid(&tetrisai_data->tetris, tetrisai_data); | ||||
|       tetrisai_data->lastTime = millis(); | ||||
|       tetrisai_data->tetris.poll(); | ||||
|     } | ||||
|   } | ||||
|   else if (tetrisai_data->tetris.state == TetrisAIGame::FIND_BEST_MOVE) | ||||
|   { | ||||
|     if (SEGMENT.check3) | ||||
|     { | ||||
|       if(tetrisai_data->mistaceCountdown == 0) | ||||
|       { | ||||
|         tetrisai_data->tetris.ai.findWorstMove = true; | ||||
|         tetrisai_data->tetris.poll(); | ||||
|         tetrisai_data->tetris.ai.findWorstMove = false; | ||||
|         tetrisai_data->mistaceCountdown = SEGMENT.custom3; | ||||
|       } | ||||
|       tetrisai_data->mistaceCountdown--;       | ||||
|     } | ||||
|     tetrisai_data->tetris.poll(); | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     tetrisai_data->tetris.poll(); | ||||
|   } | ||||
|  | ||||
|   return FRAMETIME; | ||||
| } // mode_2DTetrisAI() | ||||
| static const char _data_FX_MODE_2DTETRISAI[] PROGMEM = "Tetris AI@!,Look ahead,Intelligence,Rotate color,Mistake free,Show next,Border,Mistakes;Game Over,!,Border;!;2;sx=127,ix=64,c1=255,c2=0,c3=31,o1=1,o2=1,o3=0,pal=11"; | ||||
|  | ||||
| class TetrisAIUsermod : public Usermod | ||||
| { | ||||
|  | ||||
| private: | ||||
|  | ||||
| public: | ||||
|   void setup() | ||||
|   { | ||||
|     strip.addEffect(255, &mode_2DTetrisAI, _data_FX_MODE_2DTETRISAI); | ||||
|   } | ||||
|  | ||||
|   void loop() | ||||
|   { | ||||
|  | ||||
|   } | ||||
|  | ||||
|   uint16_t getId() | ||||
|   { | ||||
|     return USERMOD_ID_TETRISAI; | ||||
|   } | ||||
| }; | ||||
| @@ -4,6 +4,10 @@ | ||||
| #include <driver/i2s.h> | ||||
| #include <driver/adc.h> | ||||
|  | ||||
| #ifdef WLED_ENABLE_DMX | ||||
|   #error This audio reactive usermod is not compatible with DMX Out. | ||||
| #endif | ||||
|  | ||||
| #ifndef ARDUINO_ARCH_ESP32 | ||||
|   #error This audio reactive usermod does not support the ESP8266. | ||||
| #endif | ||||
| @@ -51,6 +55,8 @@ | ||||
|   #define PLOT_PRINTF(x...) | ||||
| #endif | ||||
|  | ||||
| #define MAX_PALETTES 3 | ||||
|  | ||||
| // use audio source class (ESP32 specific) | ||||
| #include "audio_source.h" | ||||
| constexpr i2s_port_t I2S_PORT = I2S_NUM_0;       // I2S port to use (do not change !) | ||||
| @@ -177,31 +183,20 @@ constexpr uint16_t samplesFFT_2 = 256;          // meaningfull part of FFT resul | ||||
| // These are the input and output vectors.  Input vectors receive computed results from FFT. | ||||
| static float vReal[samplesFFT] = {0.0f};       // FFT sample inputs / freq output -  these are our raw result bins | ||||
| static float vImag[samplesFFT] = {0.0f};       // imaginary parts | ||||
| #ifdef UM_AUDIOREACTIVE_USE_NEW_FFT | ||||
| static float windowWeighingFactors[samplesFFT] = {0.0f}; | ||||
| #endif | ||||
|  | ||||
| // Create FFT object | ||||
| #ifdef UM_AUDIOREACTIVE_USE_NEW_FFT | ||||
|   // lib_deps += https://github.com/kosme/arduinoFFT#develop @ 1.9.2 | ||||
|   // these options actually cause slow-downs on all esp32 processors, don't use them. | ||||
|   // #define FFT_SPEED_OVER_PRECISION     // enables use of reciprocals (1/x etc) - not faster on ESP32 | ||||
|   // #define FFT_SQRT_APPROXIMATION       // enables "quake3" style inverse sqrt  - slower on ESP32 | ||||
|   // Below options are forcing ArduinoFFT to use sqrtf() instead of sqrt() | ||||
|   #define sqrt(x) sqrtf(x)             // little hack that reduces FFT time by 10-50% on ESP32 | ||||
|   #define sqrt_internal sqrtf          // see https://github.com/kosme/arduinoFFT/pull/83 | ||||
| #else | ||||
|   // around 40% slower on -S2 | ||||
|   // lib_deps += https://github.com/blazoncek/arduinoFFT.git | ||||
| #endif | ||||
| // lib_deps += https://github.com/kosme/arduinoFFT#develop @ 1.9.2 | ||||
| // these options actually cause slow-downs on all esp32 processors, don't use them. | ||||
| // #define FFT_SPEED_OVER_PRECISION     // enables use of reciprocals (1/x etc) - not faster on ESP32 | ||||
| // #define FFT_SQRT_APPROXIMATION       // enables "quake3" style inverse sqrt  - slower on ESP32 | ||||
| // Below options are forcing ArduinoFFT to use sqrtf() instead of sqrt() | ||||
| #define sqrt(x) sqrtf(x)             // little hack that reduces FFT time by 10-50% on ESP32 | ||||
| #define sqrt_internal sqrtf          // see https://github.com/kosme/arduinoFFT/pull/83 | ||||
|  | ||||
| #include <arduinoFFT.h> | ||||
|  | ||||
| #ifdef UM_AUDIOREACTIVE_USE_NEW_FFT | ||||
| static ArduinoFFT<float> FFT = ArduinoFFT<float>( vReal, vImag, samplesFFT, SAMPLE_RATE, windowWeighingFactors); | ||||
| #else | ||||
| static arduinoFFT FFT = arduinoFFT(vReal, vImag, samplesFFT, SAMPLE_RATE); | ||||
| #endif | ||||
|  | ||||
| // Helper functions | ||||
|  | ||||
| @@ -282,28 +277,13 @@ void FFTcode(void * parameter) | ||||
| #endif | ||||
|  | ||||
|       // run FFT (takes 3-5ms on ESP32, ~12ms on ESP32-S2) | ||||
| #ifdef UM_AUDIOREACTIVE_USE_NEW_FFT | ||||
|       FFT.dcRemoval();                                            // remove DC offset | ||||
|       FFT.windowing( FFTWindow::Flat_top, FFTDirection::Forward); // Weigh data using "Flat Top" function - better amplitude accuracy | ||||
|       //FFT.windowing(FFTWindow::Blackman_Harris, FFTDirection::Forward);  // Weigh data using "Blackman- Harris" window - sharp peaks due to excellent sideband rejection | ||||
|       FFT.compute( FFTDirection::Forward );                       // Compute FFT | ||||
|       FFT.complexToMagnitude();                                   // Compute magnitudes | ||||
| #else | ||||
|       FFT.DCRemoval(); // let FFT lib remove DC component, so we don't need to care about this in getSamples() | ||||
|  | ||||
|       //FFT.Windowing( FFT_WIN_TYP_HAMMING, FFT_FORWARD );        // Weigh data - standard Hamming window | ||||
|       //FFT.Windowing( FFT_WIN_TYP_BLACKMAN, FFT_FORWARD );       // Blackman window - better side freq rejection | ||||
|       //FFT.Windowing( FFT_WIN_TYP_BLACKMAN_HARRIS, FFT_FORWARD );// Blackman-Harris - excellent sideband rejection | ||||
|       FFT.Windowing( FFT_WIN_TYP_FLT_TOP, FFT_FORWARD );          // Flat Top Window - better amplitude accuracy | ||||
|       FFT.Compute( FFT_FORWARD );                             // Compute FFT | ||||
|       FFT.ComplexToMagnitude();                               // Compute magnitudes | ||||
| #endif | ||||
|  | ||||
| #ifdef UM_AUDIOREACTIVE_USE_NEW_FFT | ||||
|       FFT.majorPeak(FFT_MajorPeak, FFT_Magnitude);                // let the effects know which freq was most dominant | ||||
| #else | ||||
|       FFT.MajorPeak(&FFT_MajorPeak, &FFT_Magnitude);              // let the effects know which freq was most dominant | ||||
| #endif | ||||
|       FFT.majorPeak(&FFT_MajorPeak, &FFT_Magnitude);                // let the effects know which freq was most dominant | ||||
|       FFT_MajorPeak = constrain(FFT_MajorPeak, 1.0f, 11025.0f);   // restrict value to range expected by effects | ||||
|  | ||||
| #if defined(WLED_DEBUG) || defined(SR_DEBUG) | ||||
| @@ -623,6 +603,8 @@ class AudioReactive : public Usermod { | ||||
|     #endif | ||||
|  | ||||
|     bool     initDone = false; | ||||
|     bool     addPalettes = false; | ||||
|     int8_t   palettes = 0; | ||||
|  | ||||
|     // variables  for UDP sound sync | ||||
|     WiFiUDP fftUdp;               // UDP object for sound sync (from WiFi UDP, not Async UDP!)  | ||||
| @@ -658,13 +640,21 @@ class AudioReactive : public Usermod { | ||||
|     // strings to reduce flash memory usage (used more than twice) | ||||
|     static const char _name[]; | ||||
|     static const char _enabled[]; | ||||
|     static const char _config[]; | ||||
|     static const char _dynamics[]; | ||||
|     static const char _frequency[]; | ||||
|     static const char _inputLvl[]; | ||||
|     static const char _analogmic[]; | ||||
|     static const char _digitalmic[]; | ||||
|     static const char _addPalettes[]; | ||||
|     static const char UDP_SYNC_HEADER[]; | ||||
|     static const char UDP_SYNC_HEADER_v1[]; | ||||
|  | ||||
|     // private methods | ||||
|     void removeAudioPalettes(void); | ||||
|     void createAudioPalettes(void); | ||||
|     CRGB getCRGBForBand(int x, int pal); | ||||
|     void fillAudioPalettes(void); | ||||
|  | ||||
|     //////////////////// | ||||
|     // Debug support  // | ||||
| @@ -1094,7 +1084,7 @@ class AudioReactive : public Usermod { | ||||
|      * You can use it to initialize variables, sensors or similar. | ||||
|      * It is called *AFTER* readFromConfig() | ||||
|      */ | ||||
|     void setup() | ||||
|     void setup() override | ||||
|     { | ||||
|       disableSoundProcessing = true; // just to be sure | ||||
|       if (!initDone) { | ||||
| @@ -1208,6 +1198,7 @@ class AudioReactive : public Usermod { | ||||
|       } | ||||
|  | ||||
|       if (enabled) connectUDPSoundSync(); | ||||
|       if (enabled && addPalettes) createAudioPalettes(); | ||||
|       initDone = true; | ||||
|     } | ||||
|  | ||||
| @@ -1216,7 +1207,7 @@ class AudioReactive : public Usermod { | ||||
|      * connected() is called every time the WiFi is (re)connected | ||||
|      * Use it to initialize network interfaces | ||||
|      */ | ||||
|     void connected() | ||||
|     void connected() override | ||||
|     { | ||||
|       if (udpSyncConnected) {   // clean-up: if open, close old UDP sync connection | ||||
|         udpSyncConnected = false; | ||||
| @@ -1243,7 +1234,7 @@ class AudioReactive : public Usermod { | ||||
|      * 2. Try to avoid using the delay() function. NEVER use delays longer than 10 milliseconds. | ||||
|      *    Instead, use a timer check as shown here. | ||||
|      */ | ||||
|     void loop() | ||||
|     void loop() override | ||||
|     { | ||||
|       static unsigned long lastUMRun = millis(); | ||||
|  | ||||
| @@ -1265,16 +1256,16 @@ class AudioReactive : public Usermod { | ||||
|       { | ||||
|         #ifdef WLED_DEBUG | ||||
|         if ((disableSoundProcessing == false) && (audioSyncEnabled == 0)) {  // we just switched to "disabled" | ||||
|           DEBUG_PRINTLN("[AR userLoop]  realtime mode active - audio processing suspended."); | ||||
|           DEBUG_PRINTF( "               RealtimeMode = %d; RealtimeOverride = %d\n", int(realtimeMode), int(realtimeOverride)); | ||||
|           DEBUG_PRINTLN(F("[AR userLoop]  realtime mode active - audio processing suspended.")); | ||||
|           DEBUG_PRINTF_P(PSTR("               RealtimeMode = %d; RealtimeOverride = %d\n"), int(realtimeMode), int(realtimeOverride)); | ||||
|         } | ||||
|         #endif | ||||
|         disableSoundProcessing = true; | ||||
|       } else { | ||||
|         #ifdef WLED_DEBUG | ||||
|         if ((disableSoundProcessing == true) && (audioSyncEnabled == 0) && audioSource->isInitialized()) {    // we just switched to "enabled" | ||||
|           DEBUG_PRINTLN("[AR userLoop]  realtime mode ended - audio processing resumed."); | ||||
|           DEBUG_PRINTF( "               RealtimeMode = %d; RealtimeOverride = %d\n", int(realtimeMode), int(realtimeOverride)); | ||||
|           DEBUG_PRINTLN(F("[AR userLoop]  realtime mode ended - audio processing resumed.")); | ||||
|           DEBUG_PRINTF_P(PSTR("               RealtimeMode = %d; RealtimeOverride = %d\n"), int(realtimeMode), int(realtimeOverride)); | ||||
|         } | ||||
|         #endif | ||||
|         if ((disableSoundProcessing == true) && (audioSyncEnabled == 0)) lastUMRun = millis();  // just left "realtime mode" - update timekeeping | ||||
| @@ -1298,7 +1289,7 @@ class AudioReactive : public Usermod { | ||||
|           // complain when audio userloop has been delayed for long time. Currently we need userloop running between 500 and 1500 times per second.  | ||||
|           // softhack007 disabled temporarily - avoid serial console spam with MANY leds and low FPS | ||||
|           //if ((userloopDelay > 65) && !disableSoundProcessing && (audioSyncEnabled == 0)) { | ||||
|             //DEBUG_PRINTF("[AR userLoop] hiccup detected -> was inactive for last %d millis!\n", userloopDelay); | ||||
|           //  DEBUG_PRINTF_P(PSTR("[AR userLoop] hiccup detected -> was inactive for last %d millis!\n"), userloopDelay); | ||||
|           //} | ||||
|         #endif | ||||
|  | ||||
| @@ -1370,10 +1361,11 @@ class AudioReactive : public Usermod { | ||||
|         lastTime = millis(); | ||||
|       } | ||||
|  | ||||
|       fillAudioPalettes(); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     bool getUMData(um_data_t **data) | ||||
|     bool getUMData(um_data_t **data) override | ||||
|     { | ||||
|       if (!data || !enabled) return false; // no pointer provided by caller or not enabled -> exit | ||||
|       *data = um_data; | ||||
| @@ -1381,7 +1373,7 @@ class AudioReactive : public Usermod { | ||||
|     } | ||||
|  | ||||
|  | ||||
|     void onUpdateBegin(bool init) | ||||
|     void onUpdateBegin(bool init) override | ||||
|     { | ||||
| #ifdef WLED_DEBUG | ||||
|       fftTime = sampleTime = 0; | ||||
| @@ -1436,7 +1428,7 @@ class AudioReactive : public Usermod { | ||||
|      * handleButton() can be used to override default button behaviour. Returning true | ||||
|      * will prevent button working in a default way. | ||||
|      */ | ||||
|     bool handleButton(uint8_t b) { | ||||
|     bool handleButton(uint8_t b) override { | ||||
|       yield(); | ||||
|       // crude way of determining if audio input is analog | ||||
|       // better would be for AudioSource to implement getType() | ||||
| @@ -1459,7 +1451,7 @@ class AudioReactive : public Usermod { | ||||
|      * Creating an "u" object allows you to add custom key/value pairs to the Info section of the WLED web UI. | ||||
|      * Below it is shown how this could be used for e.g. a light sensor | ||||
|      */ | ||||
|     void addToJsonInfo(JsonObject& root) | ||||
|     void addToJsonInfo(JsonObject& root) override | ||||
|     { | ||||
|       char myStringBuffer[16]; // buffer for snprintf() | ||||
|       JsonObject user = root["u"]; | ||||
| @@ -1598,7 +1590,7 @@ class AudioReactive : public Usermod { | ||||
|      * addToJsonState() can be used to add custom entries to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void addToJsonState(JsonObject& root) | ||||
|     void addToJsonState(JsonObject& root) override | ||||
|     { | ||||
|       if (!initDone) return;  // prevent crash on boot applyPreset() | ||||
|       JsonObject usermod = root[FPSTR(_name)]; | ||||
| @@ -1613,7 +1605,7 @@ class AudioReactive : public Usermod { | ||||
|      * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void readFromJsonState(JsonObject& root) | ||||
|     void readFromJsonState(JsonObject& root) override | ||||
|     { | ||||
|       if (!initDone) return;  // prevent crash on boot applyPreset() | ||||
|       bool prevEnabled = enabled; | ||||
| @@ -1622,13 +1614,28 @@ class AudioReactive : public Usermod { | ||||
|         if (usermod[FPSTR(_enabled)].is<bool>()) { | ||||
|           enabled = usermod[FPSTR(_enabled)].as<bool>(); | ||||
|           if (prevEnabled != enabled) onUpdateBegin(!enabled); | ||||
|           if (addPalettes) { | ||||
|             // add/remove custom/audioreactive palettes | ||||
|             if (prevEnabled && !enabled) removeAudioPalettes(); | ||||
|             if (!prevEnabled && enabled) createAudioPalettes(); | ||||
|           } | ||||
|         } | ||||
|         if (usermod[FPSTR(_inputLvl)].is<int>()) { | ||||
|           inputLevel = min(255,max(0,usermod[FPSTR(_inputLvl)].as<int>())); | ||||
|         } | ||||
|       } | ||||
|       if (root.containsKey(F("rmcpal")) && root[F("rmcpal")].as<bool>()) { | ||||
|         // handle removal of custom palettes from JSON call so we don't break things | ||||
|         removeAudioPalettes(); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     void onStateChange(uint8_t callMode) override { | ||||
|       if (initDone && enabled && addPalettes && palettes==0 && strip.customPalettes.size()<10) { | ||||
|         // if palettes were removed during JSON call re-add them | ||||
|         createAudioPalettes(); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * addToConfig() can be used to add custom persistent settings to the cfg.json file in the "um" (usermod) object. | ||||
| @@ -1665,10 +1672,11 @@ class AudioReactive : public Usermod { | ||||
|      *  | ||||
|      * I highly recommend checking out the basics of ArduinoJson serialization and deserialization in order to use custom settings! | ||||
|      */ | ||||
|     void addToConfig(JsonObject& root) | ||||
|     void addToConfig(JsonObject& root) override | ||||
|     { | ||||
|       JsonObject top = root.createNestedObject(FPSTR(_name)); | ||||
|       top[FPSTR(_enabled)] = enabled; | ||||
|       top[FPSTR(_addPalettes)] = addPalettes; | ||||
|  | ||||
|     #if !defined(CONFIG_IDF_TARGET_ESP32S2) && !defined(CONFIG_IDF_TARGET_ESP32C3) && !defined(CONFIG_IDF_TARGET_ESP32S3) | ||||
|       JsonObject amic = top.createNestedObject(FPSTR(_analogmic)); | ||||
| @@ -1676,29 +1684,29 @@ class AudioReactive : public Usermod { | ||||
|     #endif | ||||
|  | ||||
|       JsonObject dmic = top.createNestedObject(FPSTR(_digitalmic)); | ||||
|       dmic[F("type")] = dmType; | ||||
|       dmic["type"] = dmType; | ||||
|       JsonArray pinArray = dmic.createNestedArray("pin"); | ||||
|       pinArray.add(i2ssdPin); | ||||
|       pinArray.add(i2swsPin); | ||||
|       pinArray.add(i2sckPin); | ||||
|       pinArray.add(mclkPin); | ||||
|  | ||||
|       JsonObject cfg = top.createNestedObject("config"); | ||||
|       JsonObject cfg = top.createNestedObject(FPSTR(_config)); | ||||
|       cfg[F("squelch")] = soundSquelch; | ||||
|       cfg[F("gain")] = sampleGain; | ||||
|       cfg[F("AGC")] = soundAgc; | ||||
|  | ||||
|       JsonObject dynLim = top.createNestedObject("dynamics"); | ||||
|       JsonObject dynLim = top.createNestedObject(FPSTR(_dynamics)); | ||||
|       dynLim[F("limiter")] = limiterOn; | ||||
|       dynLim[F("rise")] = attackTime; | ||||
|       dynLim[F("fall")] = decayTime; | ||||
|  | ||||
|       JsonObject freqScale = top.createNestedObject("frequency"); | ||||
|       JsonObject freqScale = top.createNestedObject(FPSTR(_frequency)); | ||||
|       freqScale[F("scale")] = FFTScalingMode; | ||||
|  | ||||
|       JsonObject sync = top.createNestedObject("sync"); | ||||
|       sync[F("port")] = audioSyncPort; | ||||
|       sync[F("mode")] = audioSyncEnabled; | ||||
|       sync["port"] = audioSyncPort; | ||||
|       sync["mode"] = audioSyncEnabled; | ||||
|     } | ||||
|  | ||||
|  | ||||
| @@ -1717,12 +1725,15 @@ class AudioReactive : public Usermod { | ||||
|      *  | ||||
|      * This function is guaranteed to be called on boot, but could also be called every time settings are updated | ||||
|      */ | ||||
|     bool readFromConfig(JsonObject& root) | ||||
|     bool readFromConfig(JsonObject& root) override | ||||
|     { | ||||
|       JsonObject top = root[FPSTR(_name)]; | ||||
|       bool configComplete = !top.isNull(); | ||||
|       bool oldEnabled = enabled; | ||||
|       bool oldAddPalettes = addPalettes; | ||||
|  | ||||
|       configComplete &= getJsonValue(top[FPSTR(_enabled)], enabled); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_addPalettes)], addPalettes); | ||||
|  | ||||
|     #if !defined(CONFIG_IDF_TARGET_ESP32S2) && !defined(CONFIG_IDF_TARGET_ESP32C3) && !defined(CONFIG_IDF_TARGET_ESP32S3) | ||||
|       configComplete &= getJsonValue(top[FPSTR(_analogmic)]["pin"], audioPin); | ||||
| @@ -1743,24 +1754,29 @@ class AudioReactive : public Usermod { | ||||
|       configComplete &= getJsonValue(top[FPSTR(_digitalmic)]["pin"][2], i2sckPin); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_digitalmic)]["pin"][3], mclkPin); | ||||
|  | ||||
|       configComplete &= getJsonValue(top["config"][F("squelch")], soundSquelch); | ||||
|       configComplete &= getJsonValue(top["config"][F("gain")],    sampleGain); | ||||
|       configComplete &= getJsonValue(top["config"][F("AGC")],     soundAgc); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_config)][F("squelch")], soundSquelch); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_config)][F("gain")],    sampleGain); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_config)][F("AGC")],     soundAgc); | ||||
|  | ||||
|       configComplete &= getJsonValue(top["dynamics"][F("limiter")], limiterOn); | ||||
|       configComplete &= getJsonValue(top["dynamics"][F("rise")],  attackTime); | ||||
|       configComplete &= getJsonValue(top["dynamics"][F("fall")],  decayTime); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_dynamics)][F("limiter")], limiterOn); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_dynamics)][F("rise")],  attackTime); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_dynamics)][F("fall")],  decayTime); | ||||
|  | ||||
|       configComplete &= getJsonValue(top["frequency"][F("scale")], FFTScalingMode); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_frequency)][F("scale")], FFTScalingMode); | ||||
|  | ||||
|       configComplete &= getJsonValue(top["sync"][F("port")], audioSyncPort); | ||||
|       configComplete &= getJsonValue(top["sync"][F("mode")], audioSyncEnabled); | ||||
|       configComplete &= getJsonValue(top["sync"]["port"], audioSyncPort); | ||||
|       configComplete &= getJsonValue(top["sync"]["mode"], audioSyncEnabled); | ||||
|  | ||||
|       if (initDone) { | ||||
|         // add/remove custom/audioreactive palettes | ||||
|         if ((oldAddPalettes && !addPalettes) || (oldAddPalettes && !enabled)) removeAudioPalettes(); | ||||
|         if ((addPalettes && !oldAddPalettes && enabled) || (addPalettes && !oldEnabled && enabled)) createAudioPalettes(); | ||||
|       } // else setup() will create palettes | ||||
|       return configComplete; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     void appendConfigData() | ||||
|     void appendConfigData() override | ||||
|     { | ||||
|       oappend(SET_F("dd=addDropdown('AudioReactive','digitalmic:type');")); | ||||
|     #if  !defined(CONFIG_IDF_TARGET_ESP32S2) && !defined(CONFIG_IDF_TARGET_ESP32C3) && !defined(CONFIG_IDF_TARGET_ESP32S3) | ||||
| @@ -1815,7 +1831,7 @@ class AudioReactive : public Usermod { | ||||
|      * Use this to blank out some LEDs or set them to a different color regardless of the set effect mode. | ||||
|      * Commonly used for custom clocks (Cronixie, 7 segment) | ||||
|      */ | ||||
|     //void handleOverlayDraw() | ||||
|     //void handleOverlayDraw() override | ||||
|     //{ | ||||
|       //strip.setPixelColor(0, RGBW32(0,0,0,0)) // set the first pixel to black | ||||
|     //} | ||||
| @@ -1825,19 +1841,109 @@ class AudioReactive : public Usermod { | ||||
|      * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
|      * This could be used in the future for the system to determine whether your usermod is installed. | ||||
|      */ | ||||
|     uint16_t getId() | ||||
|     uint16_t getId() override | ||||
|     { | ||||
|       return USERMOD_ID_AUDIOREACTIVE; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| void AudioReactive::removeAudioPalettes(void) { | ||||
|   DEBUG_PRINTLN(F("Removing audio palettes.")); | ||||
|   while (palettes>0) { | ||||
|     strip.customPalettes.pop_back(); | ||||
|     DEBUG_PRINTLN(palettes); | ||||
|     palettes--; | ||||
|   } | ||||
|   DEBUG_PRINT(F("Total # of palettes: ")); DEBUG_PRINTLN(strip.customPalettes.size()); | ||||
| } | ||||
|  | ||||
| void AudioReactive::createAudioPalettes(void) { | ||||
|   DEBUG_PRINT(F("Total # of palettes: ")); DEBUG_PRINTLN(strip.customPalettes.size()); | ||||
|   if (palettes) return; | ||||
|   DEBUG_PRINTLN(F("Adding audio palettes.")); | ||||
|   for (int i=0; i<MAX_PALETTES; i++) | ||||
|     if (strip.customPalettes.size() < 10) { | ||||
|       strip.customPalettes.push_back(CRGBPalette16(CRGB(BLACK))); | ||||
|       palettes++; | ||||
|       DEBUG_PRINTLN(palettes); | ||||
|     } else break; | ||||
| } | ||||
|  | ||||
| // credit @netmindz ar palette, adapted for usermod @blazoncek | ||||
| CRGB AudioReactive::getCRGBForBand(int x, int pal) { | ||||
|   CRGB value; | ||||
|   CHSV hsv; | ||||
|   int b; | ||||
|   switch (pal) { | ||||
|     case 2: | ||||
|       b = map(x, 0, 255, 0, NUM_GEQ_CHANNELS/2); // convert palette position to lower half of freq band | ||||
|       hsv = CHSV(fftResult[b], 255, x); | ||||
|       hsv2rgb_rainbow(hsv, value);  // convert to R,G,B | ||||
|       break; | ||||
|     case 1: | ||||
|       b = map(x, 1, 255, 0, 10); // convert palette position to lower half of freq band | ||||
|       hsv = CHSV(fftResult[b], 255, map(fftResult[b], 0, 255, 30, 255));  // pick hue | ||||
|       hsv2rgb_rainbow(hsv, value);  // convert to R,G,B | ||||
|       break; | ||||
|     default: | ||||
|       if (x == 1) { | ||||
|         value = CRGB(fftResult[10]/2, fftResult[4]/2, fftResult[0]/2); | ||||
|       } else if(x == 255) { | ||||
|         value = CRGB(fftResult[10]/2, fftResult[0]/2, fftResult[4]/2); | ||||
|       } else { | ||||
|         value = CRGB(fftResult[0]/2, fftResult[4]/2, fftResult[10]/2); | ||||
|       } | ||||
|       break; | ||||
|   } | ||||
|   return value; | ||||
| } | ||||
|  | ||||
| void AudioReactive::fillAudioPalettes() { | ||||
|   if (!palettes) return; | ||||
|   size_t lastCustPalette = strip.customPalettes.size(); | ||||
|   if (lastCustPalette >= palettes) lastCustPalette -= palettes; | ||||
|   for (size_t pal=0; pal<palettes; pal++) { | ||||
|     uint8_t tcp[16];  // Needs to be 4 times however many colors are being used. | ||||
|                       // 3 colors = 12, 4 colors = 16, etc. | ||||
|  | ||||
|     tcp[0] = 0;  // anchor of first color - must be zero | ||||
|     tcp[1] = 0; | ||||
|     tcp[2] = 0; | ||||
|     tcp[3] = 0; | ||||
|      | ||||
|     CRGB rgb = getCRGBForBand(1, pal); | ||||
|     tcp[4] = 1;  // anchor of first color | ||||
|     tcp[5] = rgb.r; | ||||
|     tcp[6] = rgb.g; | ||||
|     tcp[7] = rgb.b; | ||||
|      | ||||
|     rgb = getCRGBForBand(128, pal); | ||||
|     tcp[8] = 128; | ||||
|     tcp[9] = rgb.r; | ||||
|     tcp[10] = rgb.g; | ||||
|     tcp[11] = rgb.b; | ||||
|      | ||||
|     rgb = getCRGBForBand(255, pal); | ||||
|     tcp[12] = 255;  // anchor of last color - must be 255 | ||||
|     tcp[13] = rgb.r; | ||||
|     tcp[14] = rgb.g; | ||||
|     tcp[15] = rgb.b; | ||||
|  | ||||
|     strip.customPalettes[lastCustPalette+pal].loadDynamicGradientPalette(tcp); | ||||
|   } | ||||
| } | ||||
|  | ||||
| // strings to reduce flash memory usage (used more than twice) | ||||
| const char AudioReactive::_name[]       PROGMEM = "AudioReactive"; | ||||
| const char AudioReactive::_enabled[]    PROGMEM = "enabled"; | ||||
| const char AudioReactive::_config[]     PROGMEM = "config"; | ||||
| const char AudioReactive::_dynamics[]   PROGMEM = "dynamics"; | ||||
| const char AudioReactive::_frequency[]  PROGMEM = "frequency"; | ||||
| const char AudioReactive::_inputLvl[]   PROGMEM = "inputLevel"; | ||||
| #if defined(ARDUINO_ARCH_ESP32) && !defined(CONFIG_IDF_TARGET_ESP32S2) && !defined(CONFIG_IDF_TARGET_ESP32C3) && !defined(CONFIG_IDF_TARGET_ESP32S3) | ||||
| const char AudioReactive::_analogmic[]  PROGMEM = "analogmic"; | ||||
| #endif | ||||
| const char AudioReactive::_digitalmic[] PROGMEM = "digitalmic"; | ||||
| const char AudioReactive::_addPalettes[]       PROGMEM = "add-palettes"; | ||||
| const char AudioReactive::UDP_SYNC_HEADER[]    PROGMEM = "00002"; // new sync header version, as format no longer compatible with previous structure | ||||
| const char AudioReactive::UDP_SYNC_HEADER_v1[] PROGMEM = "00001"; // old sync header version - need to add backwards-compatibility feature | ||||
|   | ||||
| @@ -192,7 +192,7 @@ class I2SSource : public AudioSource { | ||||
|     } | ||||
|  | ||||
|     virtual void initialize(int8_t i2swsPin = I2S_PIN_NO_CHANGE, int8_t i2ssdPin = I2S_PIN_NO_CHANGE, int8_t i2sckPin = I2S_PIN_NO_CHANGE, int8_t mclkPin = I2S_PIN_NO_CHANGE) { | ||||
|       DEBUGSR_PRINTLN("I2SSource:: initialize()."); | ||||
|       DEBUGSR_PRINTLN(F("I2SSource:: initialize().")); | ||||
|       if (i2swsPin != I2S_PIN_NO_CHANGE && i2ssdPin != I2S_PIN_NO_CHANGE) { | ||||
|         if (!pinManager.allocatePin(i2swsPin, true, PinOwner::UM_Audioreactive) || | ||||
|             !pinManager.allocatePin(i2ssdPin, false, PinOwner::UM_Audioreactive)) { // #206 | ||||
| @@ -413,7 +413,7 @@ public: | ||||
|     }; | ||||
|  | ||||
|     void initialize(int8_t i2swsPin, int8_t i2ssdPin, int8_t i2sckPin, int8_t mclkPin) { | ||||
|       DEBUGSR_PRINTLN("ES7243:: initialize();"); | ||||
|       DEBUGSR_PRINTLN(F("ES7243:: initialize();")); | ||||
|       if ((i2sckPin < 0) || (mclkPin < 0)) { | ||||
|         DEBUGSR_PRINTF("\nAR: invalid I2S pin: SCK=%d, MCLK=%d\n", i2sckPin, mclkPin);  | ||||
|         return; | ||||
| @@ -529,7 +529,7 @@ class ES8388Source : public I2SSource { | ||||
|     }; | ||||
|  | ||||
|     void initialize(int8_t i2swsPin, int8_t i2ssdPin, int8_t i2sckPin, int8_t mclkPin) { | ||||
|       DEBUGSR_PRINTLN("ES8388Source:: initialize();"); | ||||
|       DEBUGSR_PRINTLN(F("ES8388Source:: initialize();")); | ||||
|       if ((i2sckPin < 0) || (mclkPin < 0)) { | ||||
|         DEBUGSR_PRINTF("\nAR: invalid I2S pin: SCK=%d, MCLK=%d\n", i2sckPin, mclkPin);  | ||||
|         return; | ||||
| @@ -587,7 +587,7 @@ class I2SAdcSource : public I2SSource { | ||||
|     AudioSourceType getType(void) {return(Type_I2SAdc);} | ||||
|  | ||||
|     void initialize(int8_t audioPin, int8_t = I2S_PIN_NO_CHANGE, int8_t = I2S_PIN_NO_CHANGE, int8_t = I2S_PIN_NO_CHANGE) { | ||||
|       DEBUGSR_PRINTLN("I2SAdcSource:: initialize()."); | ||||
|       DEBUGSR_PRINTLN(F("I2SAdcSource:: initialize().")); | ||||
|       _myADCchannel = 0x0F; | ||||
|       if(!pinManager.allocatePin(audioPin, false, PinOwner::UM_Audioreactive)) { | ||||
|          DEBUGSR_PRINTF("failed to allocate GPIO for audio analog input: %d\n", audioPin); | ||||
| @@ -759,7 +759,7 @@ class SPH0654 : public I2SSource { | ||||
|     {} | ||||
|  | ||||
|     void initialize(int8_t i2swsPin, int8_t i2ssdPin, int8_t i2sckPin, int8_t = I2S_PIN_NO_CHANGE) { | ||||
|       DEBUGSR_PRINTLN("SPH0654:: initialize();"); | ||||
|       DEBUGSR_PRINTLN(F("SPH0654:: initialize();")); | ||||
|       I2SSource::initialize(i2swsPin, i2ssdPin, i2sckPin); | ||||
| #if !defined(CONFIG_IDF_TARGET_ESP32S2) && !defined(CONFIG_IDF_TARGET_ESP32C3) && !defined(CONFIG_IDF_TARGET_ESP32S3) | ||||
| // these registers are only existing in "classic" ESP32 | ||||
|   | ||||
| @@ -174,9 +174,9 @@ class BobLightUsermod : public Usermod { | ||||
|  | ||||
|       #if WLED_DEBUG | ||||
|       DEBUG_PRINTLN(F("Fill light data: ")); | ||||
|       DEBUG_PRINTF(" lights %d\n", numLights); | ||||
|       DEBUG_PRINTF_P(PSTR(" lights %d\n"), numLights); | ||||
|       for (int i=0; i<numLights; i++) { | ||||
|         DEBUG_PRINTF(" light %s scan %2.1f %2.1f %2.1f %2.1f\n", lights[i].lightname, lights[i].vscan[0], lights[i].vscan[1], lights[i].hscan[0], lights[i].hscan[1]); | ||||
|         DEBUG_PRINTF_P(PSTR(" light %s scan %2.1f %2.1f %2.1f %2.1f\n"), lights[i].lightname, lights[i].vscan[0], lights[i].vscan[1], lights[i].hscan[0], lights[i].hscan[1]); | ||||
|       } | ||||
|       #endif | ||||
|     } | ||||
| @@ -187,11 +187,11 @@ class BobLightUsermod : public Usermod { | ||||
|  | ||||
|   public: | ||||
|  | ||||
|     void setup() { | ||||
|     void setup() override { | ||||
|       uint16_t totalLights = bottom + left + top + right; | ||||
|       if ( totalLights > strip.getLengthTotal() ) { | ||||
|         DEBUG_PRINTLN(F("BobLight: Too many lights.")); | ||||
|         DEBUG_PRINTF("%d+%d+%d+%d>%d\n", bottom, left, top, right, strip.getLengthTotal()); | ||||
|         DEBUG_PRINTF_P(PSTR("%d+%d+%d+%d>%d\n"), bottom, left, top, right, strip.getLengthTotal()); | ||||
|         totalLights = strip.getLengthTotal(); | ||||
|         top = bottom = (uint16_t) roundf((float)totalLights * 16.0f / 50.0f); | ||||
|         left = right = (uint16_t) roundf((float)totalLights *  9.0f / 50.0f); | ||||
| @@ -202,14 +202,14 @@ class BobLightUsermod : public Usermod { | ||||
|       initDone = true; | ||||
|     } | ||||
|  | ||||
|     void connected() { | ||||
|     void connected() override { | ||||
|       // we can only start server when WiFi is connected | ||||
|       if (!bob) bob = new WiFiServer(bobPort, 1); | ||||
|       bob->begin(); | ||||
|       bob->setNoDelay(true); | ||||
|     } | ||||
|  | ||||
|     void loop() { | ||||
|     void loop() override { | ||||
|       if (!enabled || strip.isUpdating()) return; | ||||
|       if (millis() - lastTime > 10) { | ||||
|         lastTime = millis(); | ||||
| @@ -225,7 +225,7 @@ class BobLightUsermod : public Usermod { | ||||
|      * topic only contains stripped topic (part after /wled/MAC) | ||||
|      * topic should look like: /swipe with amessage of [up|down] | ||||
|      */ | ||||
|     bool onMqttMessage(char* topic, char* payload) { | ||||
|     bool onMqttMessage(char* topic, char* payload) override { | ||||
|       //if (strlen(topic) == 6 && strncmp_P(topic, PSTR("/subtopic"), 6) == 0) { | ||||
|       //  String action = payload; | ||||
|       //  if (action == "on") { | ||||
| @@ -242,7 +242,7 @@ class BobLightUsermod : public Usermod { | ||||
|     /** | ||||
|      * subscribe to MQTT topic for controlling usermod | ||||
|      */ | ||||
|     void onMqttConnect(bool sessionPresent) { | ||||
|     void onMqttConnect(bool sessionPresent) override { | ||||
|       //char subuf[64]; | ||||
|       //if (mqttDeviceTopic[0] != 0) { | ||||
|       //  strcpy(subuf, mqttDeviceTopic); | ||||
| @@ -252,7 +252,7 @@ class BobLightUsermod : public Usermod { | ||||
|     } | ||||
| #endif | ||||
|  | ||||
|     void addToJsonInfo(JsonObject& root) | ||||
|     void addToJsonInfo(JsonObject& root) override | ||||
|     { | ||||
|       JsonObject user = root["u"]; | ||||
|       if (user.isNull()) user = root.createNestedObject("u"); | ||||
| @@ -273,7 +273,7 @@ class BobLightUsermod : public Usermod { | ||||
|      * addToJsonState() can be used to add custom entries to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void addToJsonState(JsonObject& root) | ||||
|     void addToJsonState(JsonObject& root) override | ||||
|     { | ||||
|     } | ||||
|  | ||||
| @@ -281,7 +281,7 @@ class BobLightUsermod : public Usermod { | ||||
|      * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void readFromJsonState(JsonObject& root) { | ||||
|     void readFromJsonState(JsonObject& root) override { | ||||
|       if (!initDone) return;  // prevent crash on boot applyPreset() | ||||
|       bool en = enabled; | ||||
|       JsonObject um = root[FPSTR(_name)]; | ||||
| @@ -304,7 +304,7 @@ class BobLightUsermod : public Usermod { | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     void appendConfigData() { | ||||
|     void appendConfigData() override { | ||||
|       //oappend(SET_F("dd=addDropdown('usermod','selectfield');")); | ||||
|       //oappend(SET_F("addOption(dd,'1st value',0);")); | ||||
|       //oappend(SET_F("addOption(dd,'2nd value',1);")); | ||||
| @@ -315,10 +315,10 @@ class BobLightUsermod : public Usermod { | ||||
|       oappend(SET_F("addInfo('BobLight:pct',1,'Depth of scan [%]');"));   // 0 is field type, 1 is actual field | ||||
|     } | ||||
|  | ||||
|     void addToConfig(JsonObject& root) { | ||||
|     void addToConfig(JsonObject& root) override { | ||||
|       JsonObject umData = root.createNestedObject(FPSTR(_name)); | ||||
|       umData[FPSTR(_enabled)] = enabled; | ||||
|       umData[F("port")]       = bobPort; | ||||
|       umData[  "port" ]       = bobPort; | ||||
|       umData[F("top")]        = top; | ||||
|       umData[F("bottom")]     = bottom; | ||||
|       umData[F("left")]       = left; | ||||
| @@ -326,7 +326,7 @@ class BobLightUsermod : public Usermod { | ||||
|       umData[F("pct")]        = pct; | ||||
|     } | ||||
|  | ||||
|     bool readFromConfig(JsonObject& root) { | ||||
|     bool readFromConfig(JsonObject& root) override { | ||||
|       JsonObject umData = root[FPSTR(_name)]; | ||||
|       bool configComplete = !umData.isNull(); | ||||
|  | ||||
| @@ -334,7 +334,7 @@ class BobLightUsermod : public Usermod { | ||||
|       configComplete &= getJsonValue(umData[FPSTR(_enabled)], en); | ||||
|       enable(en); | ||||
|  | ||||
|       configComplete &= getJsonValue(umData[F("port")],   bobPort); | ||||
|       configComplete &= getJsonValue(umData[  "port" ],   bobPort); | ||||
|       configComplete &= getJsonValue(umData[F("bottom")], bottom,    16); | ||||
|       configComplete &= getJsonValue(umData[F("top")],    top,       16); | ||||
|       configComplete &= getJsonValue(umData[F("left")],   left,       9); | ||||
| @@ -355,11 +355,11 @@ class BobLightUsermod : public Usermod { | ||||
|      * Use this to blank out some LEDs or set them to a different color regardless of the set effect mode. | ||||
|      * Commonly used for custom clocks (Cronixie, 7 segment) | ||||
|      */ | ||||
|     void handleOverlayDraw() { | ||||
|     void handleOverlayDraw() override { | ||||
|       //strip.setPixelColor(0, RGBW32(0,0,0,0)) // set the first pixel to black | ||||
|     } | ||||
|  | ||||
|     uint16_t getId() { return USERMOD_ID_BOBLIGHT; } | ||||
|     uint16_t getId() override { return USERMOD_ID_BOBLIGHT; } | ||||
|  | ||||
| }; | ||||
|  | ||||
| @@ -392,7 +392,7 @@ void BobLightUsermod::pollBob() { | ||||
|     //get data from the client | ||||
|     while (bobClient.available()) { | ||||
|       String input = bobClient.readStringUntil('\n'); | ||||
|       // DEBUG_PRINT("Client: "); DEBUG_PRINTLN(input); // may be to stressful on Serial | ||||
|       // DEBUG_PRINT(F("Client: ")); DEBUG_PRINTLN(input); // may be to stressful on Serial | ||||
|       if (input.startsWith(F("hello"))) { | ||||
|         DEBUG_PRINTLN(F("hello")); | ||||
|         bobClient.print(F("hello\n")); | ||||
| @@ -445,7 +445,7 @@ void BobLightUsermod::pollBob() { | ||||
|           //strip.setPixelColor(light_id, RGBW32(red, green, blue, 0)); | ||||
|           setRealtimePixel(light_id, red, green, blue, 0); | ||||
|         } // currently no support for interpolation or speed, we just ignore this | ||||
|       } else if (input.startsWith(F("sync"))) { | ||||
|       } else if (input.startsWith("sync")) { | ||||
|         BobSync(); | ||||
|       } else { | ||||
|         // Client sent gibberish | ||||
|   | ||||
| @@ -20,14 +20,11 @@ react to the globes orientation. See the blog post on building it <https://www.r | ||||
|  | ||||
| I2Cdev and MPU6050 must be installed. | ||||
|  | ||||
| To install them, add I2Cdevlib-MPU6050@fbde122cc5 to lib_deps in the platformio.ini file. | ||||
|  | ||||
| You also need to change lib_compat_mode from strict to soft in platformio.ini (This ignores that I2Cdevlib-MPU6050 doesn't list platform compatibility) | ||||
| To install them, add electroniccats/MPU6050@1.0.1 to lib_deps in the platformio.ini file. | ||||
|  | ||||
| For example: | ||||
|  | ||||
| ``` | ||||
| lib_compat_mode = soft | ||||
| lib_deps = | ||||
|     FastLED@3.3.2 | ||||
|     NeoPixelBus@2.5.7 | ||||
| @@ -36,7 +33,7 @@ lib_deps = | ||||
|     AsyncTCP@1.0.3 | ||||
|     Esp Async WebServer@1.2.0 | ||||
|     IRremoteESP8266@2.7.3 | ||||
|     jrowberg/I2Cdevlib-MPU6050@^1.0.0 | ||||
|     electroniccats/MPU6050@1.0.1 | ||||
| ``` | ||||
|  | ||||
| ## Wiring | ||||
|   | ||||
							
								
								
									
										219
									
								
								usermods/mpu6050_imu/usermod_gyro_surge.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										219
									
								
								usermods/mpu6050_imu/usermod_gyro_surge.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,219 @@ | ||||
| #pragma once | ||||
|  | ||||
| /* This usermod uses gyro data to provide a "surge" effect on movement | ||||
|  | ||||
| Requires lib_deps = bolderflight/Bolder Flight Systems Eigen@^3.0.0 | ||||
|  | ||||
| */ | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| // Eigen include block | ||||
| #ifdef A0 | ||||
| namespace { constexpr size_t A0_temp {A0}; } | ||||
| #undef A0 | ||||
| static constexpr size_t A0 {A0_temp}; | ||||
| #endif | ||||
|  | ||||
| #ifdef A1 | ||||
| namespace { constexpr size_t A1_temp {A1}; } | ||||
| #undef A1 | ||||
| static constexpr size_t A1 {A1_temp}; | ||||
| #endif | ||||
|  | ||||
| #ifdef B0 | ||||
| namespace { constexpr size_t B0_temp {B0}; } | ||||
| #undef B0 | ||||
| static constexpr size_t B0 {B0_temp}; | ||||
| #endif | ||||
|  | ||||
| #ifdef B1 | ||||
| namespace { constexpr size_t B1_temp {B1}; } | ||||
| #undef B1 | ||||
| static constexpr size_t B1 {B1_temp}; | ||||
| #endif | ||||
|  | ||||
| #ifdef D0 | ||||
| namespace { constexpr size_t D0_temp {D0}; } | ||||
| #undef D0 | ||||
| static constexpr size_t D0 {D0_temp}; | ||||
| #endif | ||||
|  | ||||
| #ifdef D1 | ||||
| namespace { constexpr size_t D1_temp {D1}; } | ||||
| #undef D1 | ||||
| static constexpr size_t D1 {D1_temp}; | ||||
| #endif | ||||
|  | ||||
| #ifdef D2 | ||||
| namespace { constexpr size_t D2_temp {D2}; } | ||||
| #undef D2 | ||||
| static constexpr size_t D2 {D2_temp}; | ||||
| #endif | ||||
|  | ||||
| #ifdef D3 | ||||
| namespace { constexpr size_t D3_temp {D3}; } | ||||
| #undef D3 | ||||
| static constexpr size_t D3 {D3_temp}; | ||||
| #endif | ||||
|  | ||||
| #include "eigen.h" | ||||
| #include <Eigen/Geometry> | ||||
|  | ||||
| constexpr auto ESTIMATED_G = 9.801;  // m/s^2 | ||||
| constexpr auto ESTIMATED_G_COUNTS = 8350.; | ||||
| constexpr auto ESTIMATED_ANGULAR_RATE = (M_PI * 2000) / (INT16_MAX * 180); // radians per second | ||||
|  | ||||
| // Horribly lame digital filter code | ||||
| // Currently implements a static IIR filter. | ||||
| template<typename T, unsigned C> | ||||
| class xir_filter { | ||||
|     typedef Eigen::Array<T, C, 1> array_t; | ||||
|     const array_t a_coeff, b_coeff; | ||||
|     const T gain; | ||||
|     array_t x, y; | ||||
|  | ||||
|     public: | ||||
|     xir_filter(T gain_, array_t a, array_t b) : a_coeff(std::move(a)), b_coeff(std::move(b)), gain(gain_), x(array_t::Zero()), y(array_t::Zero()) {}; | ||||
|  | ||||
|     T operator()(T input) { | ||||
|         x.head(C-1) = x.tail(C-1);  // shift by one | ||||
|         x(C-1) = input / gain; | ||||
|         y.head(C-1) = y.tail(C-1);  // shift by one | ||||
|         y(C-1) = (x * b_coeff).sum(); | ||||
|         y(C-1) -= (y.head(C-1) * a_coeff.head(C-1)).sum(); | ||||
|         return y(C-1); | ||||
|     } | ||||
|  | ||||
|     T last() { return y(C-1); }; | ||||
| }; | ||||
|  | ||||
|  | ||||
|  | ||||
| class GyroSurge : public Usermod { | ||||
|   private: | ||||
|     static const char _name[]; | ||||
|     bool enabled = true; | ||||
|  | ||||
|     // Params | ||||
|     uint8_t max = 0; | ||||
|     float sensitivity = 0; | ||||
|  | ||||
|     // State | ||||
|     uint32_t last_sample; | ||||
|     // 100hz input | ||||
|     // butterworth low pass filter at 20hz | ||||
|     xir_filter<float, 3> filter = { 1., { -0.36952738, 0.19581571, 1.}, {0.20657208, 0.41314417, 0.20657208} }; | ||||
|                                   // { 1., { 0., 0., 1.}, { 0., 0., 1. } }; // no filter | ||||
|  | ||||
|  | ||||
|   public: | ||||
|  | ||||
|     /* | ||||
|      * setup() is called once at boot. WiFi is not yet connected at this point. | ||||
|      */ | ||||
|     void setup() {}; | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * addToConfig() can be used to add custom persistent settings to the cfg.json file in the "um" (usermod) object. | ||||
|      * It will be called by WLED when settings are actually saved (for example, LED settings are saved) | ||||
|      * I highly recommend checking out the basics of ArduinoJson serialization and deserialization in order to use custom settings! | ||||
|      */ | ||||
|     void addToConfig(JsonObject& root) | ||||
|     { | ||||
|       JsonObject top = root.createNestedObject(FPSTR(_name)); | ||||
|  | ||||
|       //save these vars persistently whenever settings are saved | ||||
|       top["max"] = max; | ||||
|       top["sensitivity"] = sensitivity; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * readFromConfig() can be used to read back the custom settings you added with addToConfig(). | ||||
|      * This is called by WLED when settings are loaded (currently this only happens immediately after boot, or after saving on the Usermod Settings page) | ||||
|      *  | ||||
|      * readFromConfig() is called BEFORE setup(). This means you can use your persistent values in setup() (e.g. pin assignments, buffer sizes), | ||||
|      * but also that if you want to write persistent values to a dynamic buffer, you'd need to allocate it here instead of in setup. | ||||
|      * If you don't know what that is, don't fret. It most likely doesn't affect your use case :) | ||||
|      *  | ||||
|      * Return true in case the config values returned from Usermod Settings were complete, or false if you'd like WLED to save your defaults to disk (so any missing values are editable in Usermod Settings) | ||||
|      *  | ||||
|      * getJsonValue() returns false if the value is missing, or copies the value into the variable provided and returns true if the value is present | ||||
|      * The configComplete variable is true only if the "exampleUsermod" object and all values are present.  If any values are missing, WLED will know to call addToConfig() to save them | ||||
|      *  | ||||
|      * This function is guaranteed to be called on boot, but could also be called every time settings are updated | ||||
|      */ | ||||
|     bool readFromConfig(JsonObject& root) | ||||
|     { | ||||
|       // default settings values could be set here (or below using the 3-argument getJsonValue()) instead of in the class definition or constructor | ||||
|       // setting them inside readFromConfig() is slightly more robust, handling the rare but plausible use case of single value being missing after boot (e.g. if the cfg.json was manually edited and a value was removed) | ||||
|  | ||||
|       JsonObject top = root[FPSTR(_name)]; | ||||
|  | ||||
|       bool configComplete = !top.isNull(); | ||||
|  | ||||
|       configComplete &= getJsonValue(top["max"], max, 0); | ||||
|       configComplete &= getJsonValue(top["sensitivity"], sensitivity, 10); | ||||
|  | ||||
|       return configComplete; | ||||
|     } | ||||
|  | ||||
|     void loop() { | ||||
|       // get IMU data | ||||
|       um_data_t *um_data; | ||||
|       if (!usermods.getUMData(&um_data, USERMOD_ID_IMU)) { | ||||
|         // Apply max | ||||
|         strip.getSegment(0).fadeToBlackBy(max); | ||||
|         return; | ||||
|       } | ||||
|       uint32_t sample_count = *(uint32_t*)(um_data->u_data[8]); | ||||
|  | ||||
|       if (sample_count != last_sample) {         | ||||
|         last_sample = sample_count; | ||||
|         // Calculate based on new data | ||||
|         // We use the raw gyro data (angular rate)         | ||||
|         auto gyros = (int16_t*)um_data->u_data[4];  // 16384 == 2000 deg/s | ||||
|  | ||||
|         // Compute the overall rotation rate | ||||
|         // For my application (a plasma sword) we ignore X axis rotations (eg. around the long axis) | ||||
|         auto gyro_q = Eigen::AngleAxis<float> { | ||||
|                         //Eigen::AngleAxis<float>(ESTIMATED_ANGULAR_RATE * gyros[0], Eigen::Vector3f::UnitX()) * | ||||
|                         Eigen::AngleAxis<float>(ESTIMATED_ANGULAR_RATE * gyros[1], Eigen::Vector3f::UnitY()) * | ||||
|                         Eigen::AngleAxis<float>(ESTIMATED_ANGULAR_RATE * gyros[2], Eigen::Vector3f::UnitZ()) }; | ||||
|          | ||||
|         // Filter the results | ||||
|         filter(std::min(sensitivity * gyro_q.angle(), 1.0f));   // radians per second | ||||
| /* | ||||
|         Serial.printf("[%lu] Gy: %d, %d, %d -- ", millis(), (int)gyros[0], (int)gyros[1], (int)gyros[2]); | ||||
|         Serial.print(gyro_q.angle()); | ||||
|         Serial.print(", "); | ||||
|         Serial.print(sensitivity * gyro_q.angle()); | ||||
|         Serial.print(" --> "); | ||||
|         Serial.println(filter.last()); | ||||
| */ | ||||
|       } | ||||
|     }; // noop | ||||
|  | ||||
|     /* | ||||
|      * handleOverlayDraw() is called just before every show() (LED strip update frame) after effects have set the colors. | ||||
|      * Use this to blank out some LEDs or set them to a different color regardless of the set effect mode. | ||||
|      * Commonly used for custom clocks (Cronixie, 7 segment) | ||||
|      */ | ||||
|     void handleOverlayDraw() | ||||
|     { | ||||
|  | ||||
|       // TODO: some kind of timing analysis for filtering ... | ||||
|  | ||||
|       // Calculate brightness boost | ||||
|       auto r_float = std::max(std::min(filter.last(), 1.0f), 0.f); | ||||
|       auto result = (uint8_t) (r_float * max); | ||||
|       //Serial.printf("[%lu] %d -- ", millis(), result); | ||||
|       //Serial.println(r_float); | ||||
|       // TODO - multiple segment handling?? | ||||
|       strip.getSegment(0).fadeToBlackBy(max - result); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| const char GyroSurge::_name[] PROGMEM = "GyroSurge"; | ||||
| @@ -20,11 +20,11 @@ | ||||
|   XCL     not connected | ||||
|   AD0     not connected | ||||
|   INT     D8 (GPIO15)   Interrupt pin | ||||
|    | ||||
|  | ||||
|   Using usermod: | ||||
|   1. Copy the usermod into the sketch folder (same folder as wled00.ino) | ||||
|   2. Register the usermod by adding #include "usermod_filename.h" in the top and registerUsermod(new MyUsermodClass()) in the bottom of usermods_list.cpp | ||||
|   3. I2Cdev and MPU6050 must be installed as libraries, or else the .cpp/.h file  | ||||
|   3. I2Cdev and MPU6050 must be installed as libraries, or else the .cpp/.h file | ||||
|      for both classes must be in the include path of your project. To install the | ||||
|      libraries add I2Cdevlib-MPU6050@fbde122cc5 to lib_deps in the platformio.ini file. | ||||
|   4. You also need to change lib_compat_mode from strict to soft in platformio.ini (This ignores that I2Cdevlib-MPU6050 doesn't list platform compatibility) | ||||
| @@ -33,6 +33,9 @@ | ||||
|  | ||||
| #include "I2Cdev.h" | ||||
|  | ||||
| #undef DEBUG_PRINT | ||||
| #undef DEBUG_PRINTLN | ||||
| #undef DEBUG_PRINTF | ||||
| #include "MPU6050_6Axis_MotionApps20.h" | ||||
| //#include "MPU6050.h" // not necessary if using MotionApps include file | ||||
|  | ||||
| @@ -42,6 +45,23 @@ | ||||
|     #include "Wire.h" | ||||
| #endif | ||||
|  | ||||
| // Restore debug macros | ||||
| // MPU6050 unfortunately uses the same macro names as WLED :( | ||||
| #undef DEBUG_PRINT | ||||
| #undef DEBUG_PRINTLN | ||||
| #undef DEBUG_PRINTF | ||||
| #ifdef WLED_DEBUG | ||||
|   #define DEBUG_PRINT(x) DEBUGOUT.print(x) | ||||
|   #define DEBUG_PRINTLN(x) DEBUGOUT.println(x) | ||||
|   #define DEBUG_PRINTF(x...) DEBUGOUT.printf(x) | ||||
| #else | ||||
|   #define DEBUG_PRINT(x) | ||||
|   #define DEBUG_PRINTLN(x) | ||||
|   #define DEBUG_PRINTF(x...) | ||||
| #endif | ||||
|  | ||||
|  | ||||
|  | ||||
| // ================================================================ | ||||
| // ===               INTERRUPT DETECTION ROUTINE                === | ||||
| // ================================================================ | ||||
| @@ -52,21 +72,31 @@ void IRAM_ATTR dmpDataReady() { | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| class MPU6050Driver : public Usermod { | ||||
|   private: | ||||
|     MPU6050 mpu; | ||||
|     bool enabled = true; | ||||
|  | ||||
|     // configuration state | ||||
|     // default values are set in readFromConfig | ||||
|     // By making this a struct, we enable easy backup and comparison in the readFromConfig class | ||||
|     struct config_t { | ||||
|       bool enabled; | ||||
|       int8_t interruptPin;     | ||||
|       int16_t gyro_offset[3];  | ||||
|       int16_t accel_offset[3]; | ||||
|     }; | ||||
|     config_t config; | ||||
|  | ||||
|     // MPU control/status vars | ||||
|     bool irqBound = false; // set true if we have bound the IRQ pin | ||||
|     bool dmpReady = false;  // set true if DMP init was successful | ||||
|     uint8_t mpuIntStatus;   // holds actual interrupt status byte from MPU | ||||
|     uint8_t devStatus;      // return status after each device operation (0 = success, !0 = error) | ||||
|     uint16_t packetSize;    // expected DMP packet size (default is 42 bytes) | ||||
|     uint16_t fifoCount;     // count of all bytes currently in FIFO | ||||
|     uint8_t fifoBuffer[64]; // FIFO storage buffer | ||||
|  | ||||
|     //NOTE: some of these can be removed to save memory, processing time | ||||
|     //      if the measurement isn't needed | ||||
|     // TODO: some of these can be removed to save memory, processing time if the measurement isn't needed | ||||
|     Quaternion qat;         // [w, x, y, z]         quaternion container | ||||
|     float euler[3];         // [psi, theta, phi]    Euler angle container | ||||
|     float ypr[3];           // [yaw, pitch, roll]   yaw/pitch/roll container | ||||
| @@ -75,17 +105,67 @@ class MPU6050Driver : public Usermod { | ||||
|     VectorInt16 aaReal;     // [x, y, z]            gravity-free accel sensor measurements | ||||
|     VectorInt16 aaWorld;    // [x, y, z]            world-frame accel sensor measurements | ||||
|     VectorFloat gravity;    // [x, y, z]            gravity vector | ||||
|     uint32 sample_count; | ||||
|  | ||||
|     static const int INTERRUPT_PIN = 15; // use pin 15 on ESP8266 | ||||
|     // Usermod output | ||||
|     um_data_t um_data; | ||||
|  | ||||
|     // config element names as progmem strs | ||||
|     static const char _name[]; | ||||
|     static const char _enabled[]; | ||||
|     static const char _interrupt_pin[]; | ||||
|     static const char _x_acc_bias[]; | ||||
|     static const char _y_acc_bias[]; | ||||
|     static const char _z_acc_bias[]; | ||||
|     static const char _x_gyro_bias[]; | ||||
|     static const char _y_gyro_bias[]; | ||||
|     static const char _z_gyro_bias[]; | ||||
|  | ||||
|   public: | ||||
|     //Functions called by WLED | ||||
|  | ||||
|     inline bool initDone() { return um_data.u_size != 0; };   // recycle this instead of storing an extra variable | ||||
|  | ||||
|     //Functions called by WLED | ||||
|     /* | ||||
|      * setup() is called once at boot. WiFi is not yet connected at this point. | ||||
|      */ | ||||
|     void setup() { | ||||
|       if (i2c_scl<0 || i2c_sda<0) { enabled = false; return; } | ||||
|       dmpReady = false;   // Start clean | ||||
|  | ||||
|       // one time init | ||||
|       if (!initDone()) { | ||||
|         um_data.u_size = 9; | ||||
|         um_data.u_type = new um_types_t[um_data.u_size]; | ||||
|         um_data.u_data = new void*[um_data.u_size]; | ||||
|         um_data.u_data[0] = &qat; | ||||
|         um_data.u_type[0] = UMT_FLOAT_ARR; | ||||
|         um_data.u_data[1] = &euler; | ||||
|         um_data.u_type[1] = UMT_FLOAT_ARR; | ||||
|         um_data.u_data[2] = &ypr; | ||||
|         um_data.u_type[2] = UMT_FLOAT_ARR; | ||||
|         um_data.u_data[3] = &aa; | ||||
|         um_data.u_type[3] = UMT_INT16_ARR; | ||||
|         um_data.u_data[4] = &gy; | ||||
|         um_data.u_type[4] = UMT_INT16_ARR; | ||||
|         um_data.u_data[5] = &aaReal; | ||||
|         um_data.u_type[5] = UMT_INT16_ARR; | ||||
|         um_data.u_data[6] = &aaWorld; | ||||
|         um_data.u_type[6] = UMT_INT16_ARR; | ||||
|         um_data.u_data[7] = &gravity; | ||||
|         um_data.u_type[7] = UMT_FLOAT_ARR; | ||||
|         um_data.u_data[8] = &sample_count; | ||||
|         um_data.u_type[8] = UMT_UINT32; | ||||
|       } | ||||
|  | ||||
|       if (!config.enabled) return; | ||||
|       if (i2c_scl<0 || i2c_sda<0) { DEBUG_PRINTLN(F("MPU6050: I2C is no good."));  return; } | ||||
|       // Check the interrupt pin | ||||
|       if (config.interruptPin >= 0) { | ||||
|         irqBound = pinManager.allocatePin(config.interruptPin, false, PinOwner::UM_IMU); | ||||
|         if (!irqBound) { DEBUG_PRINTLN(F("MPU6050: IRQ pin already in use.")); return; } | ||||
|         pinMode(config.interruptPin, INPUT); | ||||
|       }; | ||||
|  | ||||
|       #if I2CDEV_IMPLEMENTATION == I2CDEV_ARDUINO_WIRE | ||||
|         Wire.setClock(400000U); // 400kHz I2C clock. Comment this line if having compilation difficulties | ||||
|       #elif I2CDEV_IMPLEMENTATION == I2CDEV_BUILTIN_FASTWIRE | ||||
| @@ -95,7 +175,6 @@ class MPU6050Driver : public Usermod { | ||||
|       // initialize device | ||||
|       DEBUG_PRINTLN(F("Initializing I2C devices...")); | ||||
|       mpu.initialize(); | ||||
|       pinMode(INTERRUPT_PIN, INPUT); | ||||
|  | ||||
|       // verify connection | ||||
|       DEBUG_PRINTLN(F("Testing device connections...")); | ||||
| @@ -105,11 +184,16 @@ class MPU6050Driver : public Usermod { | ||||
|       DEBUG_PRINTLN(F("Initializing DMP...")); | ||||
|       devStatus = mpu.dmpInitialize(); | ||||
|  | ||||
|       // supply your own gyro offsets here, scaled for min sensitivity | ||||
|       mpu.setXGyroOffset(220); | ||||
|       mpu.setYGyroOffset(76); | ||||
|       mpu.setZGyroOffset(-85); | ||||
|       mpu.setZAccelOffset(1788); // 1688 factory default for my test chip | ||||
|       // set offsets (from config) | ||||
|       mpu.setXGyroOffset(config.gyro_offset[0]); | ||||
|       mpu.setYGyroOffset(config.gyro_offset[1]); | ||||
|       mpu.setZGyroOffset(config.gyro_offset[2]); | ||||
|       mpu.setXAccelOffset(config.accel_offset[0]); | ||||
|       mpu.setYAccelOffset(config.accel_offset[1]); | ||||
|       mpu.setZAccelOffset(config.accel_offset[2]); | ||||
|  | ||||
|       // set sample rate | ||||
|       mpu.setRate(16);  // ~100Hz | ||||
|  | ||||
|       // make sure it worked (returns 0 if so) | ||||
|       if (devStatus == 0) { | ||||
| @@ -117,17 +201,19 @@ class MPU6050Driver : public Usermod { | ||||
|         DEBUG_PRINTLN(F("Enabling DMP...")); | ||||
|         mpu.setDMPEnabled(true); | ||||
|  | ||||
|         // enable Arduino interrupt detection | ||||
|         DEBUG_PRINTLN(F("Enabling interrupt detection (Arduino external interrupt 0)...")); | ||||
|         attachInterrupt(digitalPinToInterrupt(INTERRUPT_PIN), dmpDataReady, RISING); | ||||
|         mpuIntStatus = mpu.getIntStatus(); | ||||
|  | ||||
|         // set our DMP Ready flag so the main loop() function knows it's okay to use it | ||||
|         DEBUG_PRINTLN(F("DMP ready! Waiting for first interrupt...")); | ||||
|         dmpReady = true; | ||||
|         mpuInterrupt = true; | ||||
|         if (irqBound) { | ||||
|           // enable Arduino interrupt detection | ||||
|           DEBUG_PRINTLN(F("Enabling interrupt detection (Arduino external interrupt 0)..."));           | ||||
|           attachInterrupt(digitalPinToInterrupt(config.interruptPin), dmpDataReady, RISING); | ||||
|         } | ||||
|  | ||||
|         // get expected DMP packet size for later comparison | ||||
|         packetSize = mpu.dmpGetFIFOPacketSize(); | ||||
|  | ||||
|         // set our DMP Ready flag so the main loop() function knows it's okay to use it | ||||
|         DEBUG_PRINTLN(F("DMP ready!")); | ||||
|         dmpReady = true; | ||||
|       } else { | ||||
|         // ERROR! | ||||
|         // 1 = initial memory load failed | ||||
| @@ -137,6 +223,9 @@ class MPU6050Driver : public Usermod { | ||||
|         DEBUG_PRINT(devStatus); | ||||
|         DEBUG_PRINTLN(")"); | ||||
|       } | ||||
|  | ||||
|       fifoCount = 0; | ||||
|       sample_count = 0; | ||||
|     } | ||||
|  | ||||
|     /* | ||||
| @@ -144,7 +233,7 @@ class MPU6050Driver : public Usermod { | ||||
|      * Use it to initialize network interfaces | ||||
|      */ | ||||
|     void connected() { | ||||
|       //DEBUG_PRINTLN("Connected to WiFi!"); | ||||
|       //DEBUG_PRINTLN(F("Connected to WiFi!")); | ||||
|     } | ||||
|  | ||||
|  | ||||
| @@ -153,28 +242,31 @@ class MPU6050Driver : public Usermod { | ||||
|      */ | ||||
|     void loop() { | ||||
|       // if programming failed, don't try to do anything | ||||
|       if (!enabled || !dmpReady || strip.isUpdating()) return; | ||||
|       if (!config.enabled || !dmpReady || strip.isUpdating()) return; | ||||
|  | ||||
|       // wait for MPU interrupt or extra packet(s) available | ||||
|       // mpuInterrupt is fixed on if interrupt pin is disabled | ||||
|       if (!mpuInterrupt && fifoCount < packetSize) return; | ||||
|  | ||||
|       // reset interrupt flag and get INT_STATUS byte | ||||
|       mpuInterrupt = false; | ||||
|       mpuIntStatus = mpu.getIntStatus(); | ||||
|  | ||||
|       // get current FIFO count | ||||
|       auto mpuIntStatus = mpu.getIntStatus(); | ||||
|       // Update current FIFO count | ||||
|       fifoCount = mpu.getFIFOCount(); | ||||
|  | ||||
|       // check for overflow (this should never happen unless our code is too inefficient) | ||||
|       if ((mpuIntStatus & 0x10) || fifoCount == 1024) { | ||||
|         // reset so we can continue cleanly | ||||
|         mpu.resetFIFO(); | ||||
|         DEBUG_PRINTLN(F("FIFO overflow!")); | ||||
|         DEBUG_PRINTLN(F("MPU6050: FIFO overflow!")); | ||||
|  | ||||
|         // otherwise, check for DMP data ready interrupt (this should happen frequently) | ||||
|       } else if (mpuIntStatus & 0x02) { | ||||
|         // wait for correct available data length, should be a VERY short wait | ||||
|         while (fifoCount < packetSize) fifoCount = mpu.getFIFOCount(); | ||||
|         // otherwise, check for data ready | ||||
|       } else if (fifoCount >= packetSize) { | ||||
|         // clear local interrupt pending status, if not polling | ||||
|         mpuInterrupt = !irqBound; | ||||
|  | ||||
|         // DEBUG_PRINT(F("MPU6050: Processing packet: ")); | ||||
|         // DEBUG_PRINT(fifoCount); | ||||
|         // DEBUG_PRINTLN(F(" bytes in FIFO")); | ||||
|  | ||||
|         // read a packet from FIFO | ||||
|         mpu.getFIFOBytes(fifoBuffer, packetSize); | ||||
| @@ -183,7 +275,6 @@ class MPU6050Driver : public Usermod { | ||||
|         // (this lets us immediately read more without waiting for an interrupt) | ||||
|         fifoCount -= packetSize; | ||||
|  | ||||
|  | ||||
|         //NOTE: some of these can be removed to save memory, processing time | ||||
|         //      if the measurement isn't needed | ||||
|         mpu.dmpGetQuaternion(&qat, fifoBuffer); | ||||
| @@ -194,87 +285,141 @@ class MPU6050Driver : public Usermod { | ||||
|         mpu.dmpGetLinearAccel(&aaReal, &aa, &gravity); | ||||
|         mpu.dmpGetLinearAccelInWorld(&aaWorld, &aaReal, &qat); | ||||
|         mpu.dmpGetYawPitchRoll(ypr, &qat, &gravity); | ||||
|         ++sample_count; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|  | ||||
|  | ||||
|     void addToJsonInfo(JsonObject& root) | ||||
|     { | ||||
|       int reading = 20; | ||||
|       //this code adds "u":{"Light":[20," lux"]} to the info object | ||||
|       JsonObject user = root["u"]; | ||||
|       if (user.isNull()) user = root.createNestedObject("u"); | ||||
|  | ||||
|       JsonObject imu_meas = user.createNestedObject("IMU"); | ||||
|       JsonArray quat_json = imu_meas.createNestedArray("Quat"); | ||||
|       // Unfortunately the web UI doesn't know how to print sub-objects: you just see '[object Object]' | ||||
|       // For now, we just put everything in the root userdata object. | ||||
|       //auto imu_meas = user.createNestedObject("IMU"); | ||||
|       auto& imu_meas = user; | ||||
|  | ||||
|       // If an element is an array, the UI expects two elements in the form [value, unit] | ||||
|       // Since our /value/ is an array, wrap it, eg. [[a, b, c]] | ||||
|       JsonArray quat_json = imu_meas.createNestedArray("Quat").createNestedArray(); | ||||
|       quat_json.add(qat.w); | ||||
|       quat_json.add(qat.x); | ||||
|       quat_json.add(qat.y); | ||||
|       quat_json.add(qat.z); | ||||
|       JsonArray euler_json = imu_meas.createNestedArray("Euler"); | ||||
|       JsonArray euler_json = imu_meas.createNestedArray("Euler").createNestedArray(); | ||||
|       euler_json.add(euler[0]); | ||||
|       euler_json.add(euler[1]); | ||||
|       euler_json.add(euler[2]); | ||||
|       JsonArray accel_json = imu_meas.createNestedArray("Accel"); | ||||
|       JsonArray accel_json = imu_meas.createNestedArray("Accel").createNestedArray(); | ||||
|       accel_json.add(aa.x); | ||||
|       accel_json.add(aa.y); | ||||
|       accel_json.add(aa.z); | ||||
|       JsonArray gyro_json = imu_meas.createNestedArray("Gyro"); | ||||
|       JsonArray gyro_json = imu_meas.createNestedArray("Gyro").createNestedArray(); | ||||
|       gyro_json.add(gy.x); | ||||
|       gyro_json.add(gy.y); | ||||
|       gyro_json.add(gy.z); | ||||
|       JsonArray world_json = imu_meas.createNestedArray("WorldAccel"); | ||||
|       JsonArray world_json = imu_meas.createNestedArray("WorldAccel").createNestedArray(); | ||||
|       world_json.add(aaWorld.x); | ||||
|       world_json.add(aaWorld.y); | ||||
|       world_json.add(aaWorld.z); | ||||
|       JsonArray real_json = imu_meas.createNestedArray("RealAccel"); | ||||
|       JsonArray real_json = imu_meas.createNestedArray("RealAccel").createNestedArray(); | ||||
|       real_json.add(aaReal.x); | ||||
|       real_json.add(aaReal.y); | ||||
|       real_json.add(aaReal.z); | ||||
|       JsonArray grav_json = imu_meas.createNestedArray("Gravity"); | ||||
|       JsonArray grav_json = imu_meas.createNestedArray("Gravity").createNestedArray(); | ||||
|       grav_json.add(gravity.x); | ||||
|       grav_json.add(gravity.y); | ||||
|       grav_json.add(gravity.z); | ||||
|       JsonArray orient_json = imu_meas.createNestedArray("Orientation"); | ||||
|       JsonArray orient_json = imu_meas.createNestedArray("Orientation").createNestedArray(); | ||||
|       orient_json.add(ypr[0]); | ||||
|       orient_json.add(ypr[1]); | ||||
|       orient_json.add(ypr[2]); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * addToJsonState() can be used to add custom entries to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     //void addToJsonState(JsonObject& root) | ||||
|     //{ | ||||
|       //root["user0"] = userVar0; | ||||
|     //} | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     //void readFromJsonState(JsonObject& root) | ||||
|     //{ | ||||
|       //if (root["bri"] == 255) DEBUG_PRINTLN(F("Don't burn down your garage!")); | ||||
|     //} | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * addToConfig() can be used to add custom persistent settings to the cfg.json file in the "um" (usermod) object. | ||||
|      * It will be called by WLED when settings are actually saved (for example, LED settings are saved) | ||||
|      * I highly recommend checking out the basics of ArduinoJson serialization and deserialization in order to use custom settings! | ||||
|      */ | ||||
| //    void addToConfig(JsonObject& root) | ||||
| //    { | ||||
| //      JsonObject top = root.createNestedObject("MPU6050_IMU"); | ||||
| //      JsonArray pins = top.createNestedArray("pin"); | ||||
| //      pins.add(HW_PIN_SCL); | ||||
| //      pins.add(HW_PIN_SDA); | ||||
| //    } | ||||
|     void addToConfig(JsonObject& root) | ||||
|     { | ||||
|       JsonObject top = root.createNestedObject(FPSTR(_name)); | ||||
|  | ||||
|       //save these vars persistently whenever settings are saved | ||||
|       top[FPSTR(_enabled)] = config.enabled; | ||||
|       top[FPSTR(_interrupt_pin)] = config.interruptPin; | ||||
|       top[FPSTR(_x_acc_bias)] = config.accel_offset[0]; | ||||
|       top[FPSTR(_y_acc_bias)] = config.accel_offset[1]; | ||||
|       top[FPSTR(_z_acc_bias)] = config.accel_offset[2]; | ||||
|       top[FPSTR(_x_gyro_bias)] = config.gyro_offset[0]; | ||||
|       top[FPSTR(_y_gyro_bias)] = config.gyro_offset[1]; | ||||
|       top[FPSTR(_z_gyro_bias)] = config.gyro_offset[2]; | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * readFromConfig() can be used to read back the custom settings you added with addToConfig(). | ||||
|      * This is called by WLED when settings are loaded (currently this only happens immediately after boot, or after saving on the Usermod Settings page) | ||||
|      * | ||||
|      * readFromConfig() is called BEFORE setup(). This means you can use your persistent values in setup() (e.g. pin assignments, buffer sizes), | ||||
|      * but also that if you want to write persistent values to a dynamic buffer, you'd need to allocate it here instead of in setup. | ||||
|      * If you don't know what that is, don't fret. It most likely doesn't affect your use case :) | ||||
|      * | ||||
|      * Return true in case the config values returned from Usermod Settings were complete, or false if you'd like WLED to save your defaults to disk (so any missing values are editable in Usermod Settings) | ||||
|      * | ||||
|      * getJsonValue() returns false if the value is missing, or copies the value into the variable provided and returns true if the value is present | ||||
|      * The configComplete variable is true only if the "exampleUsermod" object and all values are present.  If any values are missing, WLED will know to call addToConfig() to save them | ||||
|      * | ||||
|      * This function is guaranteed to be called on boot, but could also be called every time settings are updated | ||||
|      */ | ||||
|     bool readFromConfig(JsonObject& root) | ||||
|     { | ||||
|       // default settings values could be set here (or below using the 3-argument getJsonValue()) instead of in the class definition or constructor | ||||
|       // setting them inside readFromConfig() is slightly more robust, handling the rare but plausible use case of single value being missing after boot (e.g. if the cfg.json was manually edited and a value was removed) | ||||
|       auto old_cfg = config; | ||||
|  | ||||
|       JsonObject top = root[FPSTR(_name)]; | ||||
|  | ||||
|       bool configComplete = top.isNull(); | ||||
|       // Ensure default configuration is loaded | ||||
|       configComplete &= getJsonValue(top[FPSTR(_enabled)], config.enabled, true); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_interrupt_pin)], config.interruptPin, -1); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_x_acc_bias)], config.accel_offset[0], 0); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_y_acc_bias)], config.accel_offset[1], 0); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_z_acc_bias)], config.accel_offset[2], 0); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_x_gyro_bias)], config.gyro_offset[0], 0); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_y_gyro_bias)], config.gyro_offset[1], 0); | ||||
|       configComplete &= getJsonValue(top[FPSTR(_z_gyro_bias)], config.gyro_offset[2], 0); | ||||
|  | ||||
|       DEBUG_PRINT(FPSTR(_name)); | ||||
|       if (top.isNull()) { | ||||
|         DEBUG_PRINTLN(F(": No config found. (Using defaults.)")); | ||||
|       } else if (!initDone()) { | ||||
|         DEBUG_PRINTLN(F(": config loaded.")); | ||||
|       } else if (memcmp(&config, &old_cfg, sizeof(config)) == 0) { | ||||
|         DEBUG_PRINTLN(F(": config unchanged.")); | ||||
|       } else { | ||||
|         DEBUG_PRINTLN(F(": config updated.")); | ||||
|         // Previously loaded and config changed | ||||
|         if (irqBound && ((old_cfg.interruptPin != config.interruptPin) || !config.enabled)) { | ||||
|           detachInterrupt(old_cfg.interruptPin); | ||||
|           pinManager.deallocatePin(old_cfg.interruptPin, PinOwner::UM_IMU);             | ||||
|           irqBound = false; | ||||
|         } | ||||
|  | ||||
|         // Just re-init | ||||
|         setup(); | ||||
|       } | ||||
|  | ||||
|       return configComplete; | ||||
|     } | ||||
|  | ||||
|     bool getUMData(um_data_t **data) | ||||
|     { | ||||
|       if (!data || !config.enabled || !dmpReady) return false; // no pointer provided by caller or not enabled -> exit | ||||
|       *data = &um_data; | ||||
|       return true; | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
| @@ -285,3 +430,14 @@ class MPU6050Driver : public Usermod { | ||||
|     } | ||||
|  | ||||
| }; | ||||
|  | ||||
|  | ||||
| const char MPU6050Driver::_name[] PROGMEM = "MPU6050_IMU"; | ||||
| const char MPU6050Driver::_enabled[] PROGMEM = "enabled"; | ||||
| const char MPU6050Driver::_interrupt_pin[] PROGMEM = "interrupt_pin"; | ||||
| const char MPU6050Driver::_x_acc_bias[] PROGMEM = "x_acc_bias"; | ||||
| const char MPU6050Driver::_y_acc_bias[] PROGMEM = "y_acc_bias"; | ||||
| const char MPU6050Driver::_z_acc_bias[] PROGMEM = "z_acc_bias"; | ||||
| const char MPU6050Driver::_x_gyro_bias[] PROGMEM = "x_gyro_bias"; | ||||
| const char MPU6050Driver::_y_gyro_bias[] PROGMEM = "y_gyro_bias"; | ||||
| const char MPU6050Driver::_z_gyro_bias[] PROGMEM = "z_gyro_bias"; | ||||
|   | ||||
| @@ -47,6 +47,24 @@ or | ||||
|  | ||||
| You can override the default maximum number of relays (which is 4) by defining MULTI_RELAY_MAX_RELAYS. | ||||
|  | ||||
| Some settings can be defined (defaults) at compile time by setting the following defines: | ||||
|  | ||||
| ```cpp | ||||
| // enable or disable HA discovery for externally controlled relays | ||||
| #define MULTI_RELAY_HA_DISCOVERY true | ||||
| ``` | ||||
|  | ||||
| The following definitions should be a list of values (maximum number of entries is MULTI_RELAY_MAX_RELAYS) that will be applied to the relays in order: | ||||
| (e.g. assuming MULTI_RELAY_MAX_RELAYS=2) | ||||
|  | ||||
| ```cpp | ||||
| #define MULTI_RELAY_PINS 12,18 | ||||
| #define MULTI_RELAY_DELAYS 0,0 | ||||
| #define MULTI_RELAY_EXTERNALS false,true | ||||
| #define MULTI_RELAY_INVERTS false,false | ||||
| ``` | ||||
| These can be set via your `platformio_override.ini` file or as `#define` in your `my_config.h` (remember to set `WLED_USE_MY_CONFIG` in your `platformio_override.ini`) | ||||
|  | ||||
| Example **usermods_list.cpp**: | ||||
|  | ||||
| ```cpp | ||||
| @@ -107,4 +125,7 @@ Have fun - @blazoncek | ||||
| * Added button support. | ||||
|  | ||||
| 2023-05 | ||||
| * Added support for PCF8574 I2C port expander (multiple) | ||||
| * Added support for PCF8574 I2C port expander (multiple) | ||||
|  | ||||
| 2023-11 | ||||
| * @chrisburrows Added support for compile time defaults for setting DELAY, EXTERNAL, INVERTS and HA discovery | ||||
| @@ -2,6 +2,8 @@ | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| #define COUNT_OF(x) ((sizeof(x)/sizeof(0[x])) / ((size_t)(!(sizeof(x) % sizeof(0[x]))))) | ||||
|  | ||||
| #ifndef MULTI_RELAY_MAX_RELAYS | ||||
|   #define MULTI_RELAY_MAX_RELAYS 4 | ||||
| #else | ||||
| @@ -19,6 +21,22 @@ | ||||
|   #define MULTI_RELAY_ENABLED true | ||||
| #endif | ||||
|  | ||||
| #ifndef MULTI_RELAY_HA_DISCOVERY | ||||
|   #define MULTI_RELAY_HA_DISCOVERY false | ||||
| #endif | ||||
|  | ||||
| #ifndef MULTI_RELAY_DELAYS | ||||
|   #define MULTI_RELAY_DELAYS 0 | ||||
| #endif | ||||
|  | ||||
| #ifndef MULTI_RELAY_EXTERNALS | ||||
|   #define MULTI_RELAY_EXTERNALS false | ||||
| #endif | ||||
|  | ||||
| #ifndef MULTI_RELAY_INVERTS | ||||
|   #define MULTI_RELAY_INVERTS false | ||||
| #endif | ||||
|  | ||||
| #define WLED_DEBOUNCE_THRESHOLD 50 //only consider button input of at least 50ms as valid (debouncing) | ||||
|  | ||||
| #define ON  true | ||||
| @@ -86,6 +104,9 @@ class MultiRelay : public Usermod { | ||||
|     static const char _HAautodiscovery[]; | ||||
|     static const char _pcf8574[]; | ||||
|     static const char _pcfAddress[]; | ||||
|     static const char _switch[]; | ||||
|     static const char _toggle[]; | ||||
|     static const char _Command[]; | ||||
|  | ||||
|     void handleOffTimer(); | ||||
|     void InitHtmlAPIHandle(); | ||||
| @@ -125,7 +146,7 @@ class MultiRelay : public Usermod { | ||||
|      * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
|      * This could be used in the future for the system to determine whether your usermod is installed. | ||||
|      */ | ||||
|     inline uint16_t getId() { return USERMOD_ID_MULTI_RELAY; } | ||||
|     inline uint16_t getId() override { return USERMOD_ID_MULTI_RELAY; } | ||||
|  | ||||
|     /** | ||||
|      * switch relay on/off | ||||
| @@ -143,22 +164,22 @@ class MultiRelay : public Usermod { | ||||
|      * setup() is called once at boot. WiFi is not yet connected at this point. | ||||
|      * You can use it to initialize variables, sensors or similar. | ||||
|      */ | ||||
|     void setup(); | ||||
|     void setup() override; | ||||
|  | ||||
|     /** | ||||
|      * connected() is called every time the WiFi is (re)connected | ||||
|      * Use it to initialize network interfaces | ||||
|      */ | ||||
|     inline void connected() { InitHtmlAPIHandle(); } | ||||
|     inline void connected() override { InitHtmlAPIHandle(); } | ||||
|  | ||||
|     /** | ||||
|      * loop() is called continuously. Here you can check for events, read sensors, etc. | ||||
|      */ | ||||
|     void loop(); | ||||
|     void loop() override; | ||||
|  | ||||
| #ifndef WLED_DISABLE_MQTT | ||||
|     bool onMqttMessage(char* topic, char* payload); | ||||
|     void onMqttConnect(bool sessionPresent); | ||||
|     bool onMqttMessage(char* topic, char* payload) override; | ||||
|     void onMqttConnect(bool sessionPresent) override; | ||||
| #endif | ||||
|  | ||||
|     /** | ||||
| @@ -166,31 +187,31 @@ class MultiRelay : public Usermod { | ||||
|      * will prevent button working in a default way. | ||||
|      * Replicating button.cpp | ||||
|      */ | ||||
|     bool handleButton(uint8_t b); | ||||
|     bool handleButton(uint8_t b) override; | ||||
|  | ||||
|     /** | ||||
|      * addToJsonInfo() can be used to add custom entries to the /json/info part of the JSON API. | ||||
|      */ | ||||
|     void addToJsonInfo(JsonObject &root); | ||||
|     void addToJsonInfo(JsonObject &root) override; | ||||
|  | ||||
|     /** | ||||
|      * addToJsonState() can be used to add custom entries to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void addToJsonState(JsonObject &root); | ||||
|     void addToJsonState(JsonObject &root) override; | ||||
|  | ||||
|     /** | ||||
|      * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void readFromJsonState(JsonObject &root); | ||||
|     void readFromJsonState(JsonObject &root) override; | ||||
|  | ||||
|     /** | ||||
|      * provide the changeable values | ||||
|      */ | ||||
|     void addToConfig(JsonObject &root); | ||||
|     void addToConfig(JsonObject &root) override; | ||||
|  | ||||
|     void appendConfigData(); | ||||
|     void appendConfigData() override; | ||||
|  | ||||
|     /** | ||||
|      * restore the changeable values | ||||
| @@ -198,7 +219,7 @@ class MultiRelay : public Usermod { | ||||
|      *  | ||||
|      * The function should return true if configuration was successfully loaded or false if there was no configuration. | ||||
|      */ | ||||
|     bool readFromConfig(JsonObject &root); | ||||
|     bool readFromConfig(JsonObject &root) override; | ||||
| }; | ||||
|  | ||||
|  | ||||
| @@ -243,8 +264,8 @@ void MultiRelay::handleOffTimer() { | ||||
| void MultiRelay::InitHtmlAPIHandle() {  // https://github.com/me-no-dev/ESPAsyncWebServer | ||||
|   DEBUG_PRINTLN(F("Relays: Initialize HTML API")); | ||||
|  | ||||
|   server.on("/relays", HTTP_GET, [this](AsyncWebServerRequest *request) { | ||||
|     DEBUG_PRINTLN("Relays: HTML API"); | ||||
|   server.on(SET_F("/relays"), HTTP_GET, [this](AsyncWebServerRequest *request) { | ||||
|     DEBUG_PRINTLN(F("Relays: HTML API")); | ||||
|     String janswer; | ||||
|     String error = ""; | ||||
|     //int params = request->params(); | ||||
| @@ -253,9 +274,9 @@ void MultiRelay::InitHtmlAPIHandle() {  // https://github.com/me-no-dev/ESPAsync | ||||
|  | ||||
|     if (getActiveRelayCount()) { | ||||
|       // Commands | ||||
|       if(request->hasParam("switch")) { | ||||
|       if (request->hasParam(FPSTR(_switch))) { | ||||
|         /**** Switch ****/ | ||||
|         AsyncWebParameter* p = request->getParam("switch"); | ||||
|         AsyncWebParameter* p = request->getParam(FPSTR(_switch)); | ||||
|         // Get Values | ||||
|         for (int i=0; i<MULTI_RELAY_MAX_RELAYS; i++) { | ||||
|           int value = getValue(p->value(), ',', i); | ||||
| @@ -266,9 +287,9 @@ void MultiRelay::InitHtmlAPIHandle() {  // https://github.com/me-no-dev/ESPAsync | ||||
|             if (_relay[i].external) switchRelay(i, (bool)value); | ||||
|           } | ||||
|         } | ||||
|       } else if(request->hasParam("toggle")) { | ||||
|       } else if (request->hasParam(FPSTR(_toggle))) { | ||||
|         /**** Toggle ****/ | ||||
|         AsyncWebParameter* p = request->getParam("toggle"); | ||||
|         AsyncWebParameter* p = request->getParam(FPSTR(_toggle)); | ||||
|         // Get Values | ||||
|         for (int i=0;i<MULTI_RELAY_MAX_RELAYS;i++) { | ||||
|           int value = getValue(p->value(), ',', i); | ||||
| @@ -296,7 +317,7 @@ void MultiRelay::InitHtmlAPIHandle() {  // https://github.com/me-no-dev/ESPAsync | ||||
|     janswer += error; | ||||
|     janswer += F("\","); | ||||
|     janswer += F("\"SW Version\":\""); | ||||
|     janswer += String(GEOGABVERSION); | ||||
|     janswer += String(F(GEOGABVERSION)); | ||||
|     janswer += F("\"}"); | ||||
|     request->send(200, "application/json", janswer); | ||||
|   }); | ||||
| @@ -343,18 +364,22 @@ MultiRelay::MultiRelay() | ||||
|   , initDone(false) | ||||
|   , usePcf8574(USE_PCF8574) | ||||
|   , addrPcf8574(PCF8574_ADDRESS) | ||||
|   , HAautodiscovery(false) | ||||
|   , HAautodiscovery(MULTI_RELAY_HA_DISCOVERY) | ||||
|   , periodicBroadcastSec(60) | ||||
|   , lastBroadcast(0) | ||||
| { | ||||
|   const int8_t defPins[] = {MULTI_RELAY_PINS}; | ||||
|   const int8_t relayDelays[] = {MULTI_RELAY_DELAYS}; | ||||
|   const bool relayExternals[] = {MULTI_RELAY_EXTERNALS}; | ||||
|   const bool relayInverts[] = {MULTI_RELAY_INVERTS}; | ||||
|  | ||||
|   for (size_t i=0; i<MULTI_RELAY_MAX_RELAYS; i++) { | ||||
|     _relay[i].pin      = i<sizeof(defPins) ? defPins[i] : -1; | ||||
|     _relay[i].delay    = 0; | ||||
|     _relay[i].invert   = false; | ||||
|     _relay[i].pin      = i < COUNT_OF(defPins) ? defPins[i] : -1; | ||||
|     _relay[i].delay    = i < COUNT_OF(relayDelays) ? relayDelays[i] : 0; | ||||
|     _relay[i].invert   = i < COUNT_OF(relayInverts) ? relayInverts[i] : false; | ||||
|     _relay[i].active   = false; | ||||
|     _relay[i].state    = false; | ||||
|     _relay[i].external = false; | ||||
|     _relay[i].external = i < COUNT_OF(relayExternals) ? relayExternals[i] : false; | ||||
|     _relay[i].button   = -1; | ||||
|   } | ||||
| } | ||||
| @@ -398,7 +423,7 @@ uint8_t MultiRelay::getActiveRelayCount() { | ||||
|  * topic should look like: /relay/X/command; where X is relay number, 0 based | ||||
|  */ | ||||
| bool MultiRelay::onMqttMessage(char* topic, char* payload) { | ||||
|   if (strlen(topic) > 8 && strncmp_P(topic, PSTR("/relay/"), 7) == 0 && strncmp_P(topic+8, PSTR("/command"), 8) == 0) { | ||||
|   if (strlen(topic) > 8 && strncmp_P(topic, PSTR("/relay/"), 7) == 0 && strncmp_P(topic+8, _Command, 8) == 0) { | ||||
|     uint8_t relay = strtoul(topic+7, NULL, 10); | ||||
|     if (relay<MULTI_RELAY_MAX_RELAYS) { | ||||
|       String action = payload; | ||||
| @@ -408,7 +433,7 @@ bool MultiRelay::onMqttMessage(char* topic, char* payload) { | ||||
|       } else if (action == "off") { | ||||
|         if (_relay[relay].external) switchRelay(relay, false); | ||||
|         return true; | ||||
|       } else if (action == "toggle") { | ||||
|       } else if (action == FPSTR(_toggle)) { | ||||
|         if (_relay[relay].external) toggleRelay(relay); | ||||
|         return true; | ||||
|       } | ||||
| @@ -448,7 +473,7 @@ void MultiRelay::publishHomeAssistantAutodiscovery() { | ||||
|  | ||||
|       sprintf_P(buf, PSTR("%s/relay/%d"), mqttDeviceTopic, i); //max length: 33 + 7 + 3 = 43 | ||||
|       json["~"] = buf; | ||||
|       strcat_P(buf, PSTR("/command")); | ||||
|       strcat_P(buf, _Command); | ||||
|       mqtt->subscribe(buf, 0); | ||||
|  | ||||
|       json[F("stat_t")]  = "~"; | ||||
| @@ -653,8 +678,8 @@ void MultiRelay::addToJsonInfo(JsonObject &root) { | ||||
|       uiDomString += F(",on:"); | ||||
|       uiDomString += _relay[i].state ? "false" : "true"; | ||||
|       uiDomString += F("}});\">"); | ||||
|       uiDomString += F("<i class=\"icons"); | ||||
|       uiDomString += _relay[i].state ? F(" on") : F(" off"); | ||||
|       uiDomString += F("<i class=\"icons "); | ||||
|       uiDomString += _relay[i].state ? "on" : "off"; | ||||
|       uiDomString += F("\"></i></button>"); | ||||
|       infoArr.add(uiDomString); | ||||
|     } | ||||
| @@ -677,11 +702,11 @@ void MultiRelay::addToJsonState(JsonObject &root) { | ||||
|     if (_relay[i].pin < 0) continue; | ||||
|     JsonObject relay = rel_arr.createNestedObject(); | ||||
|     relay[FPSTR(_relay_str)] = i; | ||||
|     relay[F("state")] = _relay[i].state; | ||||
|     relay["state"] = _relay[i].state; | ||||
|   } | ||||
|   #else | ||||
|   multiRelay[FPSTR(_relay_str)] = 0; | ||||
|   multiRelay[F("state")] = _relay[0].state; | ||||
|   multiRelay["state"] = _relay[0].state; | ||||
|   #endif | ||||
| } | ||||
|  | ||||
| @@ -781,13 +806,6 @@ bool MultiRelay::readFromConfig(JsonObject &root) { | ||||
|     _relay[i].external = top[parName][FPSTR(_external)]   | _relay[i].external; | ||||
|     _relay[i].delay    = top[parName][FPSTR(_delay_str)]  | _relay[i].delay; | ||||
|     _relay[i].button   = top[parName][FPSTR(_button)]     | _relay[i].button; | ||||
|     // begin backwards compatibility (beta) remove when 0.13 is released | ||||
|     parName += '-'; | ||||
|     _relay[i].pin      = top[parName+"pin"] | _relay[i].pin; | ||||
|     _relay[i].invert   = top[parName+FPSTR(_activeHigh)] | _relay[i].invert; | ||||
|     _relay[i].external = top[parName+FPSTR(_external)]   | _relay[i].external; | ||||
|     _relay[i].delay    = top[parName+FPSTR(_delay_str)]  | _relay[i].delay; | ||||
|     // end compatibility | ||||
|     _relay[i].delay    = min(600,max(0,abs((int)_relay[i].delay))); // bounds checking max 10min | ||||
|   } | ||||
|  | ||||
| @@ -821,3 +839,6 @@ const char MultiRelay::_broadcast[]       PROGMEM = "broadcast-sec"; | ||||
| const char MultiRelay::_HAautodiscovery[] PROGMEM = "HA-autodiscovery"; | ||||
| const char MultiRelay::_pcf8574[]         PROGMEM = "use-PCF8574"; | ||||
| const char MultiRelay::_pcfAddress[]      PROGMEM = "PCF8574-address"; | ||||
| const char MultiRelay::_switch[]          PROGMEM = "switch"; | ||||
| const char MultiRelay::_toggle[]          PROGMEM = "toggle"; | ||||
| const char MultiRelay::_Command[]         PROGMEM = "/command"; | ||||
|   | ||||
| @@ -84,11 +84,11 @@ class QuinLEDAnPentaUsermod : public Usermod | ||||
|     void getCurrentUsedLedPins() | ||||
|     { | ||||
|       for (int8_t lp = 0; lp <= 4; lp++) currentLedPins[lp] = 0; | ||||
|       byte numBusses = busses.getNumBusses(); | ||||
|       byte numBusses = BusManager::getNumBusses(); | ||||
|       byte numUsedPins = 0; | ||||
|  | ||||
|       for (int8_t b = 0; b < numBusses; b++) { | ||||
|         Bus* curBus = busses.getBus(b); | ||||
|         Bus* curBus = BusManager::getBus(b); | ||||
|         if (curBus != nullptr) { | ||||
|           uint8_t pins[5] = {0, 0, 0, 0, 0}; | ||||
|           currentBussesNumPins[b] = curBus->getPins(pins); | ||||
| @@ -104,11 +104,11 @@ class QuinLEDAnPentaUsermod : public Usermod | ||||
|  | ||||
|     void getCurrentLedcValues() | ||||
|     { | ||||
|       byte numBusses = busses.getNumBusses(); | ||||
|       byte numBusses = BusManager::getNumBusses(); | ||||
|       byte numLedc = 0; | ||||
|  | ||||
|       for (int8_t b = 0; b < numBusses; b++) { | ||||
|         Bus* curBus = busses.getBus(b); | ||||
|         Bus* curBus = BusManager::getBus(b); | ||||
|         if (curBus != nullptr) { | ||||
|           uint32_t curPixColor = curBus->getPixelColor(0); | ||||
|           uint8_t _data[5] = {255, 255, 255, 255, 255}; | ||||
|   | ||||
| @@ -85,8 +85,8 @@ private: | ||||
|  | ||||
|     JsonObject device = doc.createNestedObject("device"); // attach the sensor to the same device | ||||
|     device["identifiers"] = String("wled-sensor-") + mqttClientID; | ||||
|     device["manufacturer"] = "Aircoookie"; | ||||
|     device["model"] = "WLED"; | ||||
|     device["manufacturer"] = F(WLED_BRAND); | ||||
|     device["model"] = F(WLED_PRODUCT_NAME); | ||||
|     device["sw_version"] = VERSION; | ||||
|     device["name"] = mqttClientID; | ||||
|  | ||||
|   | ||||
							
								
								
									
										110
									
								
								usermods/usermod_v2_HttpPullLightControl/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										110
									
								
								usermods/usermod_v2_HttpPullLightControl/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,110 @@ | ||||
| # usermod_v2_HttpPullLightControl | ||||
|  | ||||
| The `usermod_v2_HttpPullLightControl` is a custom user module for WLED that enables remote control over the lighting state and color through HTTP requests. It periodically polls a specified URL to obtain a JSON response containing instructions for controlling individual lights. | ||||
|  | ||||
| ## Features | ||||
|  | ||||
| * Configure the URL endpoint (only support HTTP for now, no HTTPS) and polling interval via the WLED user interface. | ||||
| * All options from the JSON API are supported (since v0.0.3). See: https://kno.wled.ge/interfaces/json-api/  | ||||
| * The ability to control the brightness of all lights and the state (on/off) and color of individual lights remotely. | ||||
| * Start or stop an effect and when you run the same effect when its's already running, it won't restart. | ||||
| * The ability to control all these settings per segment. | ||||
| * Remotely turn on/off relays, change segments or presets. | ||||
| * Unique ID generation based on the device's MAC address and a configurable salt value, appended to the request URL for identification. | ||||
|  | ||||
| ## Configuration | ||||
| * Enable the `usermod_v2_HttpPullLightControl` via the WLED user interface. | ||||
| * Specify the URL endpoint and polling interval. | ||||
|  | ||||
| ## JSON Format and examples | ||||
| * The module sends a GET request to the configured URL, appending a unique identifier as a query parameter: `https://www.example.com/mycustompage.php?id=xxxxxxxx` where xxxxxxx is a 40 character long SHA1 hash of the MAC address combined with a given salt. | ||||
|  | ||||
| * Response Format (since v0.0.3) it is eactly the same as the WLED JSON API, see: https://kno.wled.ge/interfaces/json-api/ | ||||
| After getting the URL (it can be a static file like static.json or a mylogic.php which gives a dynamic response), the response is read and parsed to WLED. | ||||
|  | ||||
| * An example of a response to set the individual lights: 0 to RED, 12 to Green and 14 to BLUE. Remember that is will SET lights, you might want to set all the others to black. | ||||
| `{ | ||||
|   "seg": | ||||
|     { | ||||
|       "i": [ | ||||
|         0, "FF0000", | ||||
|         12, "00FF00", | ||||
|         14, "0000FF" | ||||
|       ] | ||||
|     } | ||||
| }` | ||||
|  | ||||
| * Another example setting the first 10 LEDs to RED, LED 40 to a PURPLE (using RGB values) and all LEDs in between OFF (black color) | ||||
| `{ | ||||
|   "seg": | ||||
|     { | ||||
|       "i": [ | ||||
|         0,10, "FF0000", | ||||
|         10,40, "00FF00", | ||||
|         40, [0,100,100] | ||||
|       ] | ||||
|     } | ||||
| }` | ||||
|  | ||||
| * Or first set all lights to black (off), then the LED5 to color RED: | ||||
| `{ | ||||
|   "seg": | ||||
|     { | ||||
|       "i": [ | ||||
|         0,40, "000000", | ||||
|         5, "FF0000" | ||||
|       ] | ||||
|     } | ||||
| }` | ||||
|  | ||||
| * Or use the following example to start an effect, but first we UNFREEZE (frz=false) the segment because it was frozen by individual light control in the previous examples (28=Chase effect, Speed=180m Intensity=128). The three color slots are the slots you see under the color wheel and used by the effect. RED, Black, White in this case. | ||||
| `{ | ||||
|   "seg": | ||||
|     { | ||||
| 	    "frz": false, | ||||
|       "fx": 28, | ||||
|       "sx": 200, | ||||
|       "ix": 128, | ||||
| 	    "col": [ | ||||
| 		    "FF0000", | ||||
| 			  "000000", | ||||
| 			  "FFFFFF" | ||||
| 	    ] | ||||
| 	  } | ||||
| }` | ||||
|  | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| 1. Add `usermod_v2_HttpPullLightControl` to your WLED project following the instructions provided in the WLED documentation. | ||||
| 2. Compile by setting the build_flag: -D USERMOD_HTTP_PULL_LIGHT_CONTROL and upload to your ESP32/ESP8266! | ||||
| 3. There are several compile options which you can put in your platformio.ini or platformio_override.ini: | ||||
| - -DUSERMOD_HTTP_PULL_LIGHT_CONTROL   ;To Enable the usermod | ||||
| - -DHTTP_PULL_LIGHT_CONTROL_URL="\"http://mydomain.com/json-response.php\""   ; The URL which will be requested all the time to set the lights/effects | ||||
| -  -DHTTP_PULL_LIGHT_CONTROL_SALT="\"my_very-S3cret_C0de\""  ; A secret SALT which will help by making the ID more safe | ||||
| -  -DHTTP_PULL_LIGHT_CONTROL_INTERVAL=30 ; The interval at which the URL is requested in seconds | ||||
| -  -DHTTP_PULL_LIGHT_CONTROL_HIDE_SALT ; Do you want to Hide the SALT in the User Interface? If yes, Set this flag. Note that the salt can now only be set via the above -DHTTP_PULL_LIGHT_CONTROL_SALT= setting | ||||
|  | ||||
| -  -DWLED_AP_SSID="\"Christmas Card\"" ; These flags are not just for my Usermod but you probably want to set them | ||||
| -  -DWLED_AP_PASS="\"christmas\"" | ||||
| -  -DWLED_OTA_PASS="\"otapw-secret\"" | ||||
| -  -DMDNS_NAME="\"christmascard\"" | ||||
| -  -DSERVERNAME="\"CHRISTMASCARD\"" | ||||
| -  -D ABL_MILLIAMPS_DEFAULT=450 | ||||
| -  -D DEFAULT_LED_COUNT=60 ; For a LED Ring of 60 LEDs | ||||
| -  -D BTNPIN=41  ; The M5Stack Atom S3 Lite has a button on GPIO41 | ||||
| -  -D LEDPIN=2 ; The M5Stack Atom S3 Lite has a Grove connector on the front, we use this GPIO2 | ||||
| -  -D STATUSLED=35 ; The M5Stack Atom S3 Lite has a Multi-Color LED on GPIO35, although I didnt managed to control it | ||||
| -  -D IRPIN=4  ; The M5Stack Atom S3 Lite has a IR LED on GPIO4 | ||||
|  | ||||
| -  -D DEBUG=1  ; Set these DEBUG flags ONLY if you want to debug and read out Serial (using Visual Studio Code - Serial Monitor) | ||||
| -  -DDEBUG_LEVEL=5 | ||||
| -  -DWLED_DEBUG | ||||
|  | ||||
| ## Use Case: Interactive Christmas Cards | ||||
|  | ||||
| Imagine distributing interactive Christmas cards embedded with a tiny ESP32 and a string of 20 LEDs to 20 friends. When a friend powers on their card, it connects to their Wi-Fi network and starts polling your server via the `usermod_v2_HttpPullLightControl`. (Tip: Let them scan a QR code to connect to the WLED WiFi, from there they configure their own WiFi). | ||||
|  | ||||
| Your server keeps track of how many cards are active at any given time. If all 20 cards are active, your server instructs each card to light up all of its LEDs. However, if only 4 cards are active, your server instructs each card to light up only 4 LEDs. This creates a real-time interactive experience, symbolizing the collective spirit of the holiday season. Each lit LED represents a friend who's thinking about the others, and the visual feedback creates a sense of connection among the group, despite the physical distance. | ||||
|  | ||||
| This setup demonstrates a unique way to blend traditional holiday sentiments with modern technology, offering an engaging and memorable experience. | ||||
| @@ -0,0 +1,319 @@ | ||||
| #include "usermod_v2_HttpPullLightControl.h" | ||||
|  | ||||
| // add more strings here to reduce flash memory usage | ||||
| const char HttpPullLightControl::_name[]    PROGMEM = "HttpPullLightControl"; | ||||
| const char HttpPullLightControl::_enabled[] PROGMEM = "Enable"; | ||||
|  | ||||
| void HttpPullLightControl::setup() { | ||||
|   //Serial.begin(115200); | ||||
|  | ||||
|   // Print version number | ||||
|   DEBUG_PRINT(F("HttpPullLightControl version: ")); | ||||
|   DEBUG_PRINTLN(HTTP_PULL_LIGHT_CONTROL_VERSION); | ||||
|  | ||||
|   // Start a nice chase so we know its booting and searching for its first http pull. | ||||
|   DEBUG_PRINTLN(F("Starting a nice chase so we now it is booting.")); | ||||
|   Segment& seg = strip.getMainSegment(); | ||||
|   seg.setMode(28); // Set to chase | ||||
|   seg.speed = 200; | ||||
|   seg.intensity = 255; | ||||
|   seg.setPalette(128); | ||||
|   seg.setColor(0, 5263440); | ||||
|   seg.setColor(1, 0); | ||||
|   seg.setColor(2, 4605510); | ||||
|  | ||||
|   // Go on with generating a unique ID and splitting the URL into parts | ||||
|   uniqueId = generateUniqueId();  // Cache the unique ID | ||||
|   DEBUG_PRINT(F("UniqueId calculated: ")); | ||||
|   DEBUG_PRINTLN(uniqueId); | ||||
|   parseUrl(); | ||||
|   DEBUG_PRINTLN(F("HttpPullLightControl successfully setup")); | ||||
| } | ||||
|  | ||||
| // This is the main loop function, from here we check the URL and handle the response. | ||||
| // Effects or individual lights are set as a result from this. | ||||
| void HttpPullLightControl::loop() { | ||||
|   if (!enabled || offMode) return; // Do nothing when not enabled or powered off | ||||
|   if (millis() - lastCheck >= checkInterval * 1000) { | ||||
|     DEBUG_PRINTLN(F("Calling checkUrl function")); | ||||
|     checkUrl(); | ||||
|     lastCheck = millis(); | ||||
|   } | ||||
|  | ||||
| } | ||||
|  | ||||
| // Generate a unique ID based on the MAC address and a SALT | ||||
| String HttpPullLightControl::generateUniqueId() { | ||||
|   uint8_t mac[6]; | ||||
|   WiFi.macAddress(mac); | ||||
|   char macStr[18]; | ||||
|   sprintf(macStr, "%02x:%02x:%02x:%02x:%02x:%02x", mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]); | ||||
|   // Set the MAC Address to a string and make it UPPERcase | ||||
|   String macString = String(macStr); | ||||
|   macString.toUpperCase(); | ||||
|   DEBUG_PRINT(F("WiFi MAC address is: ")); | ||||
|   DEBUG_PRINTLN(macString); | ||||
|   DEBUG_PRINT(F("Salt is: ")); | ||||
|   DEBUG_PRINTLN(salt); | ||||
|   String input = macString + salt; | ||||
|  | ||||
|   #ifdef ESP8266 | ||||
|     // For ESP8266 we use the Hash.h library which is built into the ESP8266 Core | ||||
|     return sha1(input); | ||||
|   #endif | ||||
|  | ||||
|   #ifdef ESP32 | ||||
|     // For ESP32 we use the mbedtls library which is built into the ESP32 core | ||||
|     int status = 0; | ||||
|     unsigned char shaResult[20]; // SHA1 produces a hash of 20 bytes  (which is 40 HEX characters) | ||||
|     mbedtls_sha1_context ctx; | ||||
|     mbedtls_sha1_init(&ctx); | ||||
|     status = mbedtls_sha1_starts_ret(&ctx); | ||||
|     if (status != 0) { | ||||
|       DEBUG_PRINTLN(F("Error starting SHA1 checksum calculation")); | ||||
|     } | ||||
|     status = mbedtls_sha1_update_ret(&ctx, reinterpret_cast<const unsigned char*>(input.c_str()), input.length()); | ||||
|     if (status != 0) { | ||||
|       DEBUG_PRINTLN(F("Error feeding update buffer into ongoing SHA1 checksum calculation")); | ||||
|     } | ||||
|     status = mbedtls_sha1_finish_ret(&ctx, shaResult); | ||||
|     if (status != 0) { | ||||
|       DEBUG_PRINTLN(F("Error finishing SHA1 checksum calculation")); | ||||
|     } | ||||
|     mbedtls_sha1_free(&ctx); | ||||
|  | ||||
|     // Convert the Hash to a hexadecimal string | ||||
|     char buf[41]; | ||||
|     for (int i = 0; i < 20; i++) { | ||||
|       sprintf(&buf[i*2], "%02x", shaResult[i]); | ||||
|     } | ||||
|     return String(buf); | ||||
|   #endif | ||||
| } | ||||
|  | ||||
| // This function is called when the user updates the Sald and so we need to re-calculate the unique ID | ||||
| void HttpPullLightControl::updateSalt(String newSalt) { | ||||
|   DEBUG_PRINTLN(F("Salt updated")); | ||||
|   this->salt = newSalt; | ||||
|   uniqueId = generateUniqueId(); | ||||
|   DEBUG_PRINT(F("New UniqueId is: ")); | ||||
|   DEBUG_PRINTLN(uniqueId); | ||||
| } | ||||
|  | ||||
| // The function is used to separate the URL in a host part and a path part | ||||
| void HttpPullLightControl::parseUrl() { | ||||
|   int firstSlash = url.indexOf('/', 7);  // Skip http(s):// | ||||
|   host = url.substring(7, firstSlash); | ||||
|   path = url.substring(firstSlash); | ||||
| } | ||||
|  | ||||
| // This function is called by WLED when the USERMOD config is read | ||||
| bool HttpPullLightControl::readFromConfig(JsonObject& root) { | ||||
|   // Attempt to retrieve the nested object for this usermod | ||||
|   JsonObject top = root[FPSTR(_name)]; | ||||
|   bool configComplete = !top.isNull();  // check if the object exists | ||||
|  | ||||
|   // Retrieve the values using the getJsonValue function for better error handling | ||||
|   configComplete &= getJsonValue(top[FPSTR(_enabled)], enabled, enabled);  // default value=enabled | ||||
|   configComplete &= getJsonValue(top["checkInterval"], checkInterval, checkInterval);  // default value=60 | ||||
|   #ifndef HTTP_PULL_LIGHT_CONTROL_HIDE_URL | ||||
|     configComplete &= getJsonValue(top["url"], url, url);  // default value="http://example.com" | ||||
|   #endif | ||||
|   #ifndef HTTP_PULL_LIGHT_CONTROL_HIDE_SALT | ||||
|     configComplete &= getJsonValue(top["salt"], salt, salt);  // default value=your_salt_here | ||||
|   #endif | ||||
|  | ||||
|   return configComplete; | ||||
| } | ||||
|  | ||||
| // This function is called by WLED when the USERMOD config is saved in the frontend | ||||
| void HttpPullLightControl::addToConfig(JsonObject& root) { | ||||
|   // Create a nested object for this usermod | ||||
|   JsonObject top = root.createNestedObject(FPSTR(_name)); | ||||
|  | ||||
|   // Write the configuration parameters to the nested object | ||||
|   top[FPSTR(_enabled)] = enabled; | ||||
|   if (enabled==false) | ||||
|     // To make it a bit more user-friendly, we unfreeze the main segment after disabling the module. Because individual light control (like for a christmas card) might have been done. | ||||
|     strip.getMainSegment().freeze=false; | ||||
|   top["checkInterval"] = checkInterval; | ||||
|   #ifndef HTTP_PULL_LIGHT_CONTROL_HIDE_URL | ||||
|     top["url"] = url; | ||||
|   #endif | ||||
|   #ifndef HTTP_PULL_LIGHT_CONTROL_HIDE_SALT | ||||
|     top["salt"] = salt; | ||||
|     updateSalt(salt);  // Update the UniqueID | ||||
|   #endif | ||||
|   parseUrl();  // Re-parse the URL, maybe path and host is changed | ||||
| } | ||||
|  | ||||
| // Do the http request here. Note that we can not do https requests with the AsyncTCP library | ||||
| // We do everything Asynchronous, so all callbacks are defined here | ||||
| void HttpPullLightControl::checkUrl() { | ||||
|   // Extra Inactivity check to see if AsyncCLient hangs | ||||
|   if (client != nullptr && ( millis() - lastActivityTime > inactivityTimeout ) ) { | ||||
|       DEBUG_PRINTLN(F("Inactivity detected, deleting client.")); | ||||
|       delete client; | ||||
|       client = nullptr; | ||||
|   } | ||||
|   if (client != nullptr && client->connected()) { | ||||
|       DEBUG_PRINTLN(F("We are still connected, do nothing")); | ||||
|       // Do nothing, Client is still connected | ||||
|       return; | ||||
|   } | ||||
|  | ||||
|   if (client != nullptr) { | ||||
|     // Delete previous client instance if exists, just to prevent any memory leaks | ||||
|     DEBUG_PRINTLN(F("Delete previous instances")); | ||||
|     delete client; | ||||
|     client = nullptr; | ||||
|   } | ||||
|  | ||||
|   DEBUG_PRINTLN(F("Creating new AsyncClient instance.")); | ||||
|   client = new AsyncClient(); | ||||
|   if(client) { | ||||
|     client->onData([](void *arg, AsyncClient *c, void *data, size_t len) { | ||||
|       DEBUG_PRINTLN(F("Data received.")); | ||||
|       // Cast arg back to the usermod class instance | ||||
|       HttpPullLightControl *instance = (HttpPullLightControl *)arg; | ||||
|       instance->lastActivityTime = millis(); // Update lastactivity time when data is received | ||||
|       // Convertert to Safe-String | ||||
|       char *strData = new char[len + 1]; | ||||
|       strncpy(strData, (char*)data, len); | ||||
|       strData[len] = '\0'; | ||||
|       String responseData = String(strData); | ||||
|       //String responseData = String((char *)data); | ||||
|       // Make sure its zero-terminated String | ||||
|       //responseData[len] = '\0'; | ||||
|       delete[] strData; // Do not forget to remove this one | ||||
|       instance->handleResponse(responseData); | ||||
|     }, this); | ||||
|     client->onDisconnect([](void *arg, AsyncClient *c) { | ||||
|       DEBUG_PRINTLN(F("Disconnected.")); | ||||
|       //Set the class-own client pointer to nullptr if its the current client | ||||
|       HttpPullLightControl *instance = static_cast<HttpPullLightControl*>(arg); | ||||
|       if (instance->client == c) { | ||||
|         delete instance->client; // Delete the client instance | ||||
|         instance->client = nullptr; | ||||
|       } | ||||
|     }, this); | ||||
|     client->onTimeout([](void *arg, AsyncClient *c, uint32_t time) { | ||||
|       DEBUG_PRINTLN(F("Timeout")); | ||||
|       //Set the class-own client pointer to nullptr if its the current client | ||||
|       HttpPullLightControl *instance = static_cast<HttpPullLightControl*>(arg); | ||||
|       if (instance->client == c) { | ||||
|         delete instance->client; // Delete the client instance | ||||
|         instance->client = nullptr; | ||||
|       } | ||||
|     }, this); | ||||
|     client->onError([](void *arg, AsyncClient *c, int8_t error) { | ||||
|       DEBUG_PRINTLN("Connection error occurred!"); | ||||
|       DEBUG_PRINT("Error code: "); | ||||
|       DEBUG_PRINTLN(error); | ||||
|       //Set the class-own client pointer to nullptr if its the current client | ||||
|       HttpPullLightControl *instance = static_cast<HttpPullLightControl*>(arg); | ||||
|       if (instance->client == c) { | ||||
|         delete instance->client; | ||||
|         instance->client = nullptr; | ||||
|       } | ||||
|       // Do not remove client here, it is maintained by AsyncClient | ||||
|     }, this); | ||||
|     client->onConnect([](void *arg, AsyncClient *c) { | ||||
|       // Cast arg back to the usermod class instance | ||||
|       HttpPullLightControl *instance = (HttpPullLightControl *)arg; | ||||
|       instance->onClientConnect(c);  // Call a method on the instance when the client connects | ||||
|     }, this); | ||||
|     client->setAckTimeout(ackTimeout); // Just some safety measures because we do not want any memory fillup | ||||
|     client->setRxTimeout(rxTimeout); | ||||
|     DEBUG_PRINT(F("Connecting to: ")); | ||||
|     DEBUG_PRINT(host); | ||||
|     DEBUG_PRINT(F(" via port ")); | ||||
|     DEBUG_PRINTLN((url.startsWith("https")) ? 443 : 80); | ||||
|     // Update lastActivityTime just before sending the request | ||||
|     lastActivityTime = millis(); | ||||
|     //Try to connect | ||||
|     if (!client->connect(host.c_str(), (url.startsWith("https")) ? 443 : 80)) { | ||||
|       DEBUG_PRINTLN(F("Failed to initiate connection.")); | ||||
|       // Connection failed, so cleanup | ||||
|       delete client; | ||||
|       client = nullptr; | ||||
|     } else { | ||||
|       // Connection successfull, wait for callbacks to go on. | ||||
|       DEBUG_PRINTLN(F("Connection initiated, awaiting response...")); | ||||
|     } | ||||
|   } else { | ||||
|     DEBUG_PRINTLN(F("Failed to create AsyncClient instance.")); | ||||
|   } | ||||
| } | ||||
|  | ||||
| // This function is called from the checkUrl function when the connection is establised | ||||
| // We request the data here | ||||
| void HttpPullLightControl::onClientConnect(AsyncClient *c) { | ||||
|   DEBUG_PRINT(F("Client connected: ")); | ||||
|   DEBUG_PRINTLN(c->connected() ? F("Yes") : F("No")); | ||||
|  | ||||
|   if (c->connected()) { | ||||
|     String request = "GET " + path + (path.indexOf('?') > 0 ? "&id=" : "?id=") + uniqueId + " HTTP/1.1\r\n" | ||||
|                     "Host: " + host + "\r\n" | ||||
|                     "Connection: close\r\n" | ||||
|                     "Accept: application/json\r\n" | ||||
|                     "Accept-Encoding: identity\r\n" // No compression | ||||
|                     "User-Agent: ESP32 HTTP Client\r\n\r\n"; // Optional: User-Agent and end with a double rnrn ! | ||||
|     DEBUG_PRINT(request.c_str()); | ||||
|     auto bytesSent  = c->write(request.c_str()); | ||||
|     if (bytesSent  == 0) { | ||||
|       // Connection could not be made | ||||
|       DEBUG_PRINT(F("Failed to send HTTP request.")); | ||||
|     } else { | ||||
|       DEBUG_PRINT(F("Request sent successfully, bytes sent: ")); | ||||
|       DEBUG_PRINTLN(bytesSent ); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| // This function is called when we receive data after connecting and doing our request | ||||
| // It parses the JSON data to WLED | ||||
| void HttpPullLightControl::handleResponse(String& responseStr) { | ||||
|   DEBUG_PRINTLN(F("Received response for handleResponse.")); | ||||
|  | ||||
|   // Get a Bufferlock, we can not use doc | ||||
|   if (!requestJSONBufferLock(myLockId)) { | ||||
|     DEBUG_PRINT(F("ERROR: Can not request JSON Buffer Lock, number: ")); | ||||
|       DEBUG_PRINTLN(myLockId); | ||||
|       releaseJSONBufferLock(); // Just release in any case, maybe there was already a buffer lock | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   // Search for two linebreaks between headers and content | ||||
|   int bodyPos = responseStr.indexOf("\r\n\r\n"); | ||||
|   if (bodyPos > 0) { | ||||
|     String jsonStr = responseStr.substring(bodyPos + 4); // +4 Skip the two CRLFs | ||||
|     jsonStr.trim(); | ||||
|  | ||||
|     DEBUG_PRINTLN("Response: "); | ||||
|     DEBUG_PRINTLN(jsonStr); | ||||
|  | ||||
|     // Check for valid JSON, otherwise we brick the program runtime | ||||
|     if (jsonStr[0] == '{' || jsonStr[0] == '[') { | ||||
|       // Attempt to deserialize the JSON response | ||||
|       DeserializationError error = deserializeJson(doc, jsonStr); | ||||
|       if (error == DeserializationError::Ok) { | ||||
|         // Get JSON object from th doc | ||||
|         JsonObject obj = doc.as<JsonObject>(); | ||||
|         // Parse the object throuhg deserializeState  (use CALL_MODE_NO_NOTIFY or OR CALL_MODE_DIRECT_CHANGE) | ||||
|         deserializeState(obj, CALL_MODE_NO_NOTIFY); | ||||
|       } else { | ||||
|         // If there is an error in deserialization, exit the function | ||||
|         DEBUG_PRINT(F("DeserializationError: ")); | ||||
|         DEBUG_PRINTLN(error.c_str()); | ||||
|       } | ||||
|     } else { | ||||
|       DEBUG_PRINTLN(F("Invalid JSON response")); | ||||
|     } | ||||
|   } else { | ||||
|     DEBUG_PRINTLN(F("No body found in the response")); | ||||
|   } | ||||
|   // Release the BufferLock again | ||||
|   releaseJSONBufferLock(); | ||||
| } | ||||
| @@ -0,0 +1,104 @@ | ||||
| #pragma once | ||||
| /* | ||||
|  * Usermod: HttpPullLightControl | ||||
|  * Versie: 0.0.4 | ||||
|  * Repository: https://github.com/roelbroersma/WLED-usermodv2_HttpPullLightControl | ||||
|  * Author: Roel Broersma | ||||
|  * Website: https://www.roelbroersma.nl | ||||
|  * Github author: github.com/roelbroersma | ||||
|  * Description: This usermod for WLED will request a given URL to know which effects | ||||
|  *              or individual lights it should turn on/off. So you can remote control a WLED | ||||
|  *              installation without having access to it (if no port forward, vpn or public IP is available). | ||||
|  * Use Case: Create a WLED 'Ring of Thought' christmas card. Sent a LED ring with 60 LEDs to 60 friends. | ||||
|  *           When they turn it on and put it at their WiFi, it will contact your server. Now you can reply with a given | ||||
|  *           number of lights that should turn on. Each light is a friend who did contact your server in the past 5 minutes. | ||||
|  *           So on each of your friends LED rings, the number of lights will be the number of friends who have it turned on. | ||||
|  * Features: It sends a unique ID (has of MAC and salt) to the URL, so you can define each client without a need to map their IP address. | ||||
|  * Tested: Tested on WLED v0.14 with ESP32-S3 (M5Stack Atom S3 Lite), but should also workd for other ESPs and ESP8266. | ||||
|  */ | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| // Use the following for SHA1 computation of our HASH, unfortunatelly PlatformIO doesnt recognize Hash.h while its already in the Core. | ||||
| // We use Hash.h for ESP8266 (in the core) and mbedtls/sha256.h for ESP32 (in the core). | ||||
| #ifdef ESP8266 | ||||
|   #include <Hash.h> | ||||
| #endif | ||||
| #ifdef ESP32 | ||||
|   #include "mbedtls/sha1.h" | ||||
| #endif | ||||
|  | ||||
| #define HTTP_PULL_LIGHT_CONTROL_VERSION "0.0.4" | ||||
|  | ||||
| class HttpPullLightControl : public Usermod { | ||||
| private: | ||||
|   static const char _name[]; | ||||
|   static const char _enabled[]; | ||||
|   static const char _salt[]; | ||||
|   static const char _url[]; | ||||
|  | ||||
|   bool enabled = true; | ||||
|  | ||||
|   #ifdef HTTP_PULL_LIGHT_CONTROL_INTERVAL | ||||
|     uint16_t checkInterval = HTTP_PULL_LIGHT_CONTROL_INTERVAL; | ||||
|   #else | ||||
|     uint16_t checkInterval = 60;  // Default interval of 1 minute | ||||
|   #endif | ||||
|  | ||||
|   #ifdef HTTP_PULL_LIGHT_CONTROL_URL | ||||
|     String url = HTTP_PULL_LIGHT_CONTROL_URL; | ||||
|   #else | ||||
|     String url = "http://example.org/example.php";  // Default-URL (http only!), can also be url with IP address in it. HttpS urls are not supported (yet) because of AsyncTCP library | ||||
|   #endif | ||||
|  | ||||
|   #ifdef HTTP_PULL_LIGHT_CONTROL_SALT | ||||
|     String salt = HTTP_PULL_LIGHT_CONTROL_SALT; | ||||
|   #else | ||||
|     String salt = "1just_a_very-secret_salt2";  // Salt for generating a unique ID when requesting the URL (in this way you can give different answers based on the WLED device who does the request) | ||||
|   #endif | ||||
|   // NOTE THAT THERE IS ALSO A #ifdef HTTP_PULL_LIGHT_CONTROL_HIDE_URL and a HTTP_PULL_LIGHT_CONTROL_HIDE_SALT IF YOU DO NOT WANT TO SHOW THE OPTIONS IN THE USERMOD SETTINGS | ||||
|  | ||||
|   // Define constants | ||||
|   static const uint8_t myLockId = USERMOD_ID_HTTP_PULL_LIGHT_CONTROL ; // Used for the requestJSONBufferLock(id) function | ||||
|   static const int16_t ackTimeout = 9000;  // ACK timeout in milliseconds when doing the URL request | ||||
|   static const uint16_t rxTimeout = 9000;  // RX timeout in milliseconds when doing the URL request | ||||
|   static const unsigned long FNV_offset_basis = 2166136261; | ||||
|   static const unsigned long FNV_prime = 16777619; | ||||
|   static const unsigned long inactivityTimeout = 30000; // When the AsyncClient is inactive (hanging) for this many milliseconds, we kill it | ||||
|  | ||||
|   unsigned long lastCheck = 0;    // Timestamp of last check | ||||
|   unsigned long lastActivityTime = 0; // Time of last activity of AsyncClient | ||||
|   String host;                    // Host extracted from the URL | ||||
|   String path;                    // Path extracted from the URL | ||||
|   String uniqueId;                // Cached unique ID | ||||
|   AsyncClient *client = nullptr;  // Used very often, beware of closing and freeing | ||||
|   String generateUniqueId(); | ||||
|  | ||||
|   void parseUrl(); | ||||
|   void updateSalt(String newSalt);  // Update the salt value and recalculate the unique ID | ||||
|   void checkUrl();                  // Check the specified URL for light control instructions | ||||
|   void handleResponse(String& response); | ||||
|   void onClientConnect(AsyncClient *c); | ||||
|  | ||||
| public: | ||||
|   void setup(); | ||||
|   void loop(); | ||||
|   bool readFromConfig(JsonObject& root); | ||||
|   void addToConfig(JsonObject& root); | ||||
|   uint16_t getId() { return USERMOD_ID_HTTP_PULL_LIGHT_CONTROL; } | ||||
|   inline void enable(bool enable) { enabled = enable; }   // Enable or Disable the usermod | ||||
|   inline bool isEnabled() { return enabled; }             // Get usermod enabled or disabled state | ||||
|   virtual ~HttpPullLightControl() {  | ||||
|    // Remove the cached client if needed | ||||
|     if (client) { | ||||
|       client->onDisconnect(nullptr); | ||||
|       client->onError(nullptr); | ||||
|       client->onTimeout(nullptr); | ||||
|       client->onData(nullptr); | ||||
|       client->onConnect(nullptr); | ||||
|       // Now it is safe to delete the client. | ||||
|       delete client; // This is safe even if client is nullptr. | ||||
|       client = nullptr; | ||||
|     } | ||||
|   } | ||||
| }; | ||||
							
								
								
									
										14
									
								
								usermods/usermod_v2_animartrix/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								usermods/usermod_v2_animartrix/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| # ANIMartRIX | ||||
|  | ||||
| Addes the effects from ANIMartRIX to WLED | ||||
|  | ||||
| CC BY-NC 3.0 licensed effects by Stefan Petrick, include this usermod only if you accept the terms! | ||||
|  | ||||
| ## Installation  | ||||
|  | ||||
| Please uncomment the two references to ANIMartRIX in your platform.ini  | ||||
|  | ||||
| lib_dep to a version of https://github.com/netmindz/animartrix.git | ||||
| and the build_flags  -D USERMOD_ANIMARTRIX | ||||
|  | ||||
|  | ||||
							
								
								
									
										456
									
								
								usermods/usermod_v2_animartrix/usermod_v2_animartrix.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										456
									
								
								usermods/usermod_v2_animartrix/usermod_v2_animartrix.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,456 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
| #include <ANIMartRIX.h> | ||||
|  | ||||
| #warning WLED usermod: CC BY-NC 3.0 licensed effects by Stefan Petrick, include this usermod only if you accept the terms! | ||||
| //======================================================================================================================== | ||||
|  | ||||
|  | ||||
| static const char _data_FX_mode_Module_Experiment10[] PROGMEM = "Z💡Module_Experiment10@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Module_Experiment9[] PROGMEM = "Z💡Module_Experiment9@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Module_Experiment8[] PROGMEM = "Z💡Module_Experiment8@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Module_Experiment7[] PROGMEM = "Z💡Module_Experiment7@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Module_Experiment6[] PROGMEM = "Z💡Module_Experiment6@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Module_Experiment5[] PROGMEM = "Z💡Module_Experiment5@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Module_Experiment4[] PROGMEM = "Z💡Module_Experiment4@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Zoom2[] PROGMEM = "Z💡Zoom2@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Module_Experiment3[] PROGMEM = "Z💡Module_Experiment3@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Module_Experiment2[] PROGMEM = "Z💡Module_Experiment2@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Module_Experiment1[] PROGMEM = "Z💡Module_Experiment1@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Parametric_Water[] PROGMEM = "Z💡Parametric_Water@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Water[] PROGMEM = "Z💡Water@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Complex_Kaleido_6[] PROGMEM = "Z💡Complex_Kaleido_6@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Complex_Kaleido_5[] PROGMEM = "Z💡Complex_Kaleido_5@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Complex_Kaleido_4[] PROGMEM = "Z💡Complex_Kaleido_4@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Complex_Kaleido_3[] PROGMEM = "Z💡Complex_Kaleido_3@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Complex_Kaleido_2[] PROGMEM = "Z💡Complex_Kaleido_2@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Complex_Kaleido[] PROGMEM = "Z💡Complex_Kaleido@Speed;;1;2"; | ||||
| static const char _data_FX_mode_SM10[] PROGMEM = "Z💡SM10@Speed;;1;2"; | ||||
| static const char _data_FX_mode_SM9[] PROGMEM = "Z💡SM9@Speed;;1;2"; | ||||
| static const char _data_FX_mode_SM8[] PROGMEM = "Z💡SM8@Speed;;1;2"; | ||||
| static const char _data_FX_mode_SM7[] PROGMEM = "Z💡SM7@Speed;;1;2"; | ||||
| static const char _data_FX_mode_SM6[] PROGMEM = "Z💡SM6@Speed;;1;2"; | ||||
| static const char _data_FX_mode_SM5[] PROGMEM = "Z💡SM5@Speed;;1;2"; | ||||
| static const char _data_FX_mode_SM4[] PROGMEM = "Z💡SM4@Speed;;1;2"; | ||||
| static const char _data_FX_mode_SM3[] PROGMEM = "Z💡SM3@Speed;;1;2"; | ||||
| static const char _data_FX_mode_SM2[] PROGMEM = "Z💡SM2@Speed;;1;2"; | ||||
| static const char _data_FX_mode_SM1[] PROGMEM = "Z💡SM1@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Big_Caleido[] PROGMEM = "Z💡Big_Caleido@Speed;;1;2"; | ||||
| static const char _data_FX_mode_RGB_Blobs5[] PROGMEM = "Z💡RGB_Blobs5@Speed;;1;2"; | ||||
| static const char _data_FX_mode_RGB_Blobs4[] PROGMEM = "Z💡RGB_Blobs4@Speed;;1;2"; | ||||
| static const char _data_FX_mode_RGB_Blobs3[] PROGMEM = "Z💡RGB_Blobs3@Speed;;1;2"; | ||||
| static const char _data_FX_mode_RGB_Blobs2[] PROGMEM = "Z💡RGB_Blobs2@Speed;;1;2"; | ||||
| static const char _data_FX_mode_RGB_Blobs[] PROGMEM = "Z💡RGB_Blobs@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Polar_Waves[] PROGMEM = "Z💡Polar_Waves@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Slow_Fade[] PROGMEM = "Z💡Slow_Fade@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Zoom[] PROGMEM = "Z💡Zoom@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Hot_Blob[] PROGMEM = "Z💡Hot_Blob@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Spiralus2[] PROGMEM = "Z💡Spiralus2@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Spiralus[] PROGMEM = "Z💡Spiralus@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Yves[] PROGMEM = "Z💡Yves@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Scaledemo1[] PROGMEM = "Z💡Scaledemo1@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Lava1[] PROGMEM = "Z💡Lava1@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Caleido3[] PROGMEM = "Z💡Caleido3@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Caleido2[] PROGMEM = "Z💡Caleido2@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Caleido1[] PROGMEM = "Z💡Caleido1@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Distance_Experiment[] PROGMEM = "Z💡Distance_Experiment@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Center_Field[] PROGMEM = "Z💡Center_Field@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Waves[] PROGMEM = "Z💡Waves@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Chasing_Spirals[] PROGMEM = "Z💡Chasing_Spirals@Speed;;1;2"; | ||||
| static const char _data_FX_mode_Rotating_Blob[] PROGMEM = "Z💡Rotating_Blob@Speed;;1;2"; | ||||
|  | ||||
|  | ||||
| class ANIMartRIXMod:public ANIMartRIX { | ||||
| 	public: | ||||
| 	void initEffect() { | ||||
| 	  if (SEGENV.call == 0) { | ||||
| 		init(SEGMENT.virtualWidth(), SEGMENT.virtualHeight(), false); | ||||
| 	  } | ||||
| 	  float speedFactor = 1.0; | ||||
| 	  if (SEGMENT.speed < 128) { | ||||
| 		speedFactor = (float) map(SEGMENT.speed,   0, 127, 1, 10) / 10.0f; | ||||
| 	  } | ||||
| 	  else{ | ||||
| 		speedFactor = map(SEGMENT.speed, 128, 255, 10, 100) / 10; | ||||
| 	  }  | ||||
| 	  setSpeedFactor(speedFactor); | ||||
| 	} | ||||
| 	void setPixelColor(int x, int y, rgb pixel) { | ||||
| 		SEGMENT.setPixelColorXY(x, y, CRGB(pixel.red, pixel.green, pixel.blue)); | ||||
| 	} | ||||
| 	void setPixelColor(int index, rgb pixel) { | ||||
| 		SEGMENT.setPixelColor(index, CRGB(pixel.red, pixel.green, pixel.blue)); | ||||
|   	} | ||||
|  | ||||
| 	// Add any extra custom effects not part of the ANIMartRIX libary here | ||||
| }; | ||||
| ANIMartRIXMod anim; | ||||
|  | ||||
| uint16_t mode_Module_Experiment10() { | ||||
| 	anim.initEffect();  | ||||
| 	anim.Module_Experiment10(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Module_Experiment9() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Module_Experiment9(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Module_Experiment8() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Module_Experiment8(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Module_Experiment7() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Module_Experiment7(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Module_Experiment6() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Module_Experiment6(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Module_Experiment5() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Module_Experiment5(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Module_Experiment4() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Module_Experiment4(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Zoom2() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Zoom2(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Module_Experiment3() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Module_Experiment3(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Module_Experiment2() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Module_Experiment2(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Module_Experiment1() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Module_Experiment1(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Parametric_Water() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Parametric_Water(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Water() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Water(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Complex_Kaleido_6() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Complex_Kaleido_6(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Complex_Kaleido_5() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Complex_Kaleido_5(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Complex_Kaleido_4() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Complex_Kaleido_4(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Complex_Kaleido_3() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Complex_Kaleido_3(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Complex_Kaleido_2() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Complex_Kaleido_2(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Complex_Kaleido() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Complex_Kaleido(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_SM10() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.SM10(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_SM9() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.SM9(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_SM8() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.SM8(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| // uint16_t mode_SM7() {  | ||||
| //	anim.initEffect();  | ||||
| // 	anim.SM7(); | ||||
| // | ||||
| //	return FRAMETIME; | ||||
| // } | ||||
| uint16_t mode_SM6() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.SM6(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_SM5() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.SM5(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_SM4() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.SM4(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_SM3() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.SM3(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_SM2() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.SM2(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_SM1() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.SM1(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Big_Caleido() {  | ||||
| 	anim.initEffect(); 	 | ||||
| 	anim.Big_Caleido(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_RGB_Blobs5() {  | ||||
| 	anim.initEffect(); 	 | ||||
| 	anim.RGB_Blobs5(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_RGB_Blobs4() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.RGB_Blobs4(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_RGB_Blobs3() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.RGB_Blobs3(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_RGB_Blobs2() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.RGB_Blobs2(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_RGB_Blobs() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.RGB_Blobs(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Polar_Waves() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Polar_Waves(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Slow_Fade() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Slow_Fade(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Zoom() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Zoom(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Hot_Blob() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Hot_Blob(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Spiralus2() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Spiralus2(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Spiralus() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Spiralus(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Yves() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Yves(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Scaledemo1() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Scaledemo1(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Lava1() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Lava1(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Caleido3() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Caleido3(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Caleido2() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Caleido2(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Caleido1() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Caleido1(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Distance_Experiment() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Distance_Experiment(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Center_Field() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Center_Field(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Waves() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Waves(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Chasing_Spirals() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Chasing_Spirals(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
| uint16_t mode_Rotating_Blob() {  | ||||
| 	anim.initEffect();  | ||||
| 	anim.Rotating_Blob(); | ||||
| 	return FRAMETIME; | ||||
| } | ||||
|  | ||||
|  | ||||
| class AnimartrixUsermod : public Usermod { | ||||
|   protected: | ||||
| 	bool enabled = false; //WLEDMM | ||||
| 	const char *_name; //WLEDMM | ||||
| 	bool initDone = false; //WLEDMM | ||||
| 	unsigned long lastTime = 0; //WLEDMM | ||||
|  | ||||
|   public: | ||||
|  | ||||
|     AnimartrixUsermod(const char *name, bool enabled) { | ||||
| 		this->_name = name; | ||||
| 		this->enabled = enabled; | ||||
| 	} //WLEDMM | ||||
| 	 | ||||
|  | ||||
|     void setup() { | ||||
|  | ||||
|       strip.addEffect(255, &mode_Module_Experiment10, _data_FX_mode_Module_Experiment10); | ||||
|       strip.addEffect(255, &mode_Module_Experiment9, _data_FX_mode_Module_Experiment9); | ||||
|       strip.addEffect(255, &mode_Module_Experiment8, _data_FX_mode_Module_Experiment8); | ||||
|       strip.addEffect(255, &mode_Module_Experiment7, _data_FX_mode_Module_Experiment7); | ||||
|       strip.addEffect(255, &mode_Module_Experiment6, _data_FX_mode_Module_Experiment6); | ||||
|       strip.addEffect(255, &mode_Module_Experiment5, _data_FX_mode_Module_Experiment5); | ||||
|       strip.addEffect(255, &mode_Module_Experiment4, _data_FX_mode_Module_Experiment4); | ||||
|       strip.addEffect(255, &mode_Zoom2, _data_FX_mode_Zoom2); | ||||
|       strip.addEffect(255, &mode_Module_Experiment3, _data_FX_mode_Module_Experiment3); | ||||
|       strip.addEffect(255, &mode_Module_Experiment2, _data_FX_mode_Module_Experiment2); | ||||
|       strip.addEffect(255, &mode_Module_Experiment1, _data_FX_mode_Module_Experiment1); | ||||
|       strip.addEffect(255, &mode_Parametric_Water, _data_FX_mode_Parametric_Water); | ||||
|       strip.addEffect(255, &mode_Water, _data_FX_mode_Water); | ||||
|       strip.addEffect(255, &mode_Complex_Kaleido_6, _data_FX_mode_Complex_Kaleido_6); | ||||
|       strip.addEffect(255, &mode_Complex_Kaleido_5, _data_FX_mode_Complex_Kaleido_5); | ||||
|       strip.addEffect(255, &mode_Complex_Kaleido_4, _data_FX_mode_Complex_Kaleido_4); | ||||
|       strip.addEffect(255, &mode_Complex_Kaleido_3, _data_FX_mode_Complex_Kaleido_3); | ||||
|       strip.addEffect(255, &mode_Complex_Kaleido_2, _data_FX_mode_Complex_Kaleido_2); | ||||
|       strip.addEffect(255, &mode_Complex_Kaleido, _data_FX_mode_Complex_Kaleido); | ||||
|       strip.addEffect(255, &mode_SM10, _data_FX_mode_SM10); | ||||
|       strip.addEffect(255, &mode_SM9, _data_FX_mode_SM9); | ||||
|       strip.addEffect(255, &mode_SM8, _data_FX_mode_SM8); | ||||
|       // strip.addEffect(255, &mode_SM7, _data_FX_mode_SM7); | ||||
|       strip.addEffect(255, &mode_SM6, _data_FX_mode_SM6); | ||||
|       strip.addEffect(255, &mode_SM5, _data_FX_mode_SM5); | ||||
|       strip.addEffect(255, &mode_SM4, _data_FX_mode_SM4); | ||||
|       strip.addEffect(255, &mode_SM3, _data_FX_mode_SM3); | ||||
|       strip.addEffect(255, &mode_SM2, _data_FX_mode_SM2); | ||||
|       strip.addEffect(255, &mode_SM1, _data_FX_mode_SM1); | ||||
|       strip.addEffect(255, &mode_Big_Caleido, _data_FX_mode_Big_Caleido); | ||||
|       strip.addEffect(255, &mode_RGB_Blobs5, _data_FX_mode_RGB_Blobs5); | ||||
|       strip.addEffect(255, &mode_RGB_Blobs4, _data_FX_mode_RGB_Blobs4); | ||||
|       strip.addEffect(255, &mode_RGB_Blobs3, _data_FX_mode_RGB_Blobs3); | ||||
|       strip.addEffect(255, &mode_RGB_Blobs2, _data_FX_mode_RGB_Blobs2); | ||||
|       strip.addEffect(255, &mode_RGB_Blobs, _data_FX_mode_RGB_Blobs); | ||||
|       strip.addEffect(255, &mode_Polar_Waves, _data_FX_mode_Polar_Waves); | ||||
|       strip.addEffect(255, &mode_Slow_Fade, _data_FX_mode_Slow_Fade); | ||||
|       strip.addEffect(255, &mode_Zoom, _data_FX_mode_Zoom); | ||||
|       strip.addEffect(255, &mode_Hot_Blob, _data_FX_mode_Hot_Blob); | ||||
|       strip.addEffect(255, &mode_Spiralus2, _data_FX_mode_Spiralus2); | ||||
|       strip.addEffect(255, &mode_Spiralus, _data_FX_mode_Spiralus); | ||||
|       strip.addEffect(255, &mode_Yves, _data_FX_mode_Yves); | ||||
|       strip.addEffect(255, &mode_Scaledemo1, _data_FX_mode_Scaledemo1); | ||||
|       strip.addEffect(255, &mode_Lava1, _data_FX_mode_Lava1); | ||||
|       strip.addEffect(255, &mode_Caleido3, _data_FX_mode_Caleido3); | ||||
|       strip.addEffect(255, &mode_Caleido2, _data_FX_mode_Caleido2); | ||||
|       strip.addEffect(255, &mode_Caleido1, _data_FX_mode_Caleido1); | ||||
|       strip.addEffect(255, &mode_Distance_Experiment, _data_FX_mode_Distance_Experiment); | ||||
|       strip.addEffect(255, &mode_Center_Field, _data_FX_mode_Center_Field); | ||||
|       strip.addEffect(255, &mode_Waves, _data_FX_mode_Waves); | ||||
|       strip.addEffect(255, &mode_Chasing_Spirals, _data_FX_mode_Chasing_Spirals); | ||||
|       strip.addEffect(255, &mode_Rotating_Blob, _data_FX_mode_Rotating_Blob); | ||||
|  | ||||
|       initDone = true; | ||||
|     } | ||||
|  | ||||
|     void loop() { | ||||
|       if (!enabled || strip.isUpdating()) return; | ||||
|  | ||||
|       // do your magic here | ||||
|       if (millis() - lastTime > 1000) { | ||||
|         //USER_PRINTLN("I'm alive!"); | ||||
|         lastTime = millis(); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     void addToJsonInfo(JsonObject& root) | ||||
|     { | ||||
|       char myStringBuffer[16]; // buffer for snprintf() | ||||
|       JsonObject user = root["u"]; | ||||
|       if (user.isNull()) user = root.createNestedObject("u"); | ||||
|  | ||||
|       JsonArray infoArr = user.createNestedArray(FPSTR(_name)); | ||||
|  | ||||
|       String uiDomString = F("Animartrix requires the Creative Commons Attribution License CC BY-NC 3.0"); | ||||
|       infoArr.add(uiDomString); | ||||
| 	} | ||||
|  | ||||
|     uint16_t getId() | ||||
|     { | ||||
|       return USERMOD_ID_ANIMARTRIX; | ||||
|     } | ||||
|  | ||||
| }; | ||||
|  | ||||
|  | ||||
|  | ||||
| @@ -1,63 +0,0 @@ | ||||
| # I2C 4 Line Display Usermod | ||||
|  | ||||
| First, thanks to the authors of the ssd11306_i2c_oled_u8g2 mod. | ||||
|  | ||||
| Provides a four line display using either | ||||
| 128x32 or 128x64 OLED displays. | ||||
| It can operate independently, but starts to provide | ||||
| a relatively complete on-device UI when paired with the  | ||||
| Rotary Encoder UI usermod. I strongly encourage you to use  | ||||
| them together. | ||||
|  | ||||
| [See the pair of usermods in action](https://www.youtube.com/watch?v=tITQY80rIOA) | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| Copy and update the example `platformio_override.ini.sample`  | ||||
| from the Rotary Encoder UI usermode folder to the root directory of your particular build. | ||||
| This file should be placed in the same directory as `platformio.ini`. | ||||
|  | ||||
| ### Define Your Options | ||||
|  | ||||
| * `USERMOD_FOUR_LINE_DISPLAY`  - define this to have this mod included wled00\usermods_list.cpp - also tells Rotary Encoder usermod, if installed, the display is available | ||||
| * `FLD_PIN_SCL`                - The display SCL pin, defaults to 5 | ||||
| * `FLD_PIN_SDA`                - The display SDA pin, defaults to 4 | ||||
|  | ||||
| All of the parameters can be configured via the Usermods settings page, including GPIO pins. | ||||
|  | ||||
| ### PlatformIO requirements | ||||
|  | ||||
| This usermod requires the `U8g2` and `Wire` libraries. See the  | ||||
| `platformio_override.ini.sample` found in the Rotary Encoder | ||||
| UI usermod folder for how to include these using `platformio_override.ini`. | ||||
|  | ||||
| ## Configuration | ||||
|  | ||||
| * `enabled` - enable/disable usermod | ||||
| * `pin` - GPIO pins used for display; I2C displays use Clk & Data; SPI displays can use SCK, MOSI, CS, DC & RST | ||||
| * `type` - display type in numeric format | ||||
|     * 1 = I2C SSD1306 128x32 | ||||
|     * 2 = I2C SH1106 128x32 | ||||
|     * 3 = I2C SSD1306 128x64 (4 double-height lines) | ||||
|     * 4 = I2C SSD1305 128x32 | ||||
|     * 5 = I2C SSD1305 128x64 (4 double-height lines) | ||||
|     * 6 = SPI SSD1306 128x32 | ||||
|     * 7 = SPI SSD1306 128x64 (4 double-height lines) | ||||
| * `contrast` - set display contrast (higher contrast may reduce display lifetime) | ||||
| * `refreshRateSec` - display refresh time in seconds | ||||
| * `screenTimeOutSec` - screen saver time-out in seconds | ||||
| * `flip` - flip/rotate display 180° | ||||
| * `sleepMode` - enable/disable screen saver | ||||
| * `clockMode` - enable/disable clock display in screen saver mode | ||||
| * `i2c-freq-kHz` - I2C clock frequency in kHz (may help reduce dropped frames, range: 400-3400) | ||||
|  | ||||
| ## Change Log | ||||
|  | ||||
| 2021-02 | ||||
| * First public release | ||||
|  | ||||
| 2021-04 | ||||
| * Adaptation for runtime configuration. | ||||
|  | ||||
| 2021-11 | ||||
| * Added configuration option description. | ||||
| @@ -1,742 +0,0 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
| #include <U8x8lib.h> // from https://github.com/olikraus/u8g2/ | ||||
|  | ||||
| // | ||||
| // Insired by the v1 usermod: ssd1306_i2c_oled_u8g2 | ||||
| // | ||||
| // v2 usermod for using 128x32 or 128x64 i2c | ||||
| // OLED displays to provide a four line display | ||||
| // for WLED. | ||||
| // | ||||
| // Dependencies | ||||
| // * This usermod REQUIRES the ModeSortUsermod | ||||
| // * This Usermod works best, by far, when coupled  | ||||
| //   with RotaryEncoderUIUsermod. | ||||
| // | ||||
| // Make sure to enable NTP and set your time zone in WLED Config | Time. | ||||
| // | ||||
| // REQUIREMENT: You must add the following requirements to | ||||
| // REQUIREMENT: "lib_deps" within platformio.ini / platformio_override.ini | ||||
| // REQUIREMENT: *  U8g2  (the version already in platformio.ini is fine) | ||||
| // REQUIREMENT: *  Wire | ||||
| // | ||||
|  | ||||
| //The SCL and SDA pins are defined here.  | ||||
| #ifndef FLD_PIN_SCL | ||||
|   #define FLD_PIN_SCL i2c_scl | ||||
| #endif | ||||
| #ifndef FLD_PIN_SDA | ||||
|   #define FLD_PIN_SDA i2c_sda | ||||
| #endif | ||||
| #ifndef FLD_PIN_CLOCKSPI | ||||
|   #define FLD_PIN_CLOCKSPI spi_sclk | ||||
| #endif | ||||
|   #ifndef FLD_PIN_DATASPI | ||||
|   #define FLD_PIN_DATASPI spi_mosi | ||||
| #endif    | ||||
| #ifndef FLD_PIN_CS | ||||
|   #define FLD_PIN_CS spi_cs | ||||
| #endif | ||||
| #ifdef ARDUINO_ARCH_ESP32 | ||||
|   #ifndef FLD_PIN_DC | ||||
|     #define FLD_PIN_DC 19 | ||||
|   #endif | ||||
|   #ifndef FLD_PIN_RESET | ||||
|     #define FLD_PIN_RESET 26 | ||||
|   #endif | ||||
| #else | ||||
|   #ifndef FLD_PIN_DC | ||||
|     #define FLD_PIN_DC 12 | ||||
|   #endif | ||||
|   #ifndef FLD_PIN_RESET | ||||
|     #define FLD_PIN_RESET 16 | ||||
|   #endif | ||||
| #endif | ||||
|  | ||||
| #ifndef FLD_TYPE | ||||
|   #ifndef FLD_SPI_DEFAULT | ||||
|     #define FLD_TYPE SSD1306 | ||||
|   #else | ||||
|     #define FLD_TYPE SSD1306_SPI | ||||
|   #endif | ||||
| #endif | ||||
|  | ||||
| // When to time out to the clock or blank the screen | ||||
| // if SLEEP_MODE_ENABLED. | ||||
| #define SCREEN_TIMEOUT_MS  60*1000    // 1 min | ||||
|  | ||||
| #define TIME_INDENT        0 | ||||
| #define DATE_INDENT        2 | ||||
|  | ||||
| // Minimum time between redrawing screen in ms | ||||
| #define USER_LOOP_REFRESH_RATE_MS 1000 | ||||
|  | ||||
| // Extra char (+1) for null | ||||
| #define LINE_BUFFER_SIZE            16+1 | ||||
|  | ||||
| typedef enum { | ||||
|   FLD_LINE_BRIGHTNESS = 0, | ||||
|   FLD_LINE_EFFECT_SPEED, | ||||
|   FLD_LINE_EFFECT_INTENSITY, | ||||
|   FLD_LINE_MODE, | ||||
|   FLD_LINE_PALETTE, | ||||
|   FLD_LINE_TIME | ||||
| } Line4Type; | ||||
|  | ||||
| typedef enum { | ||||
|   NONE = 0, | ||||
|   SSD1306,      // U8X8_SSD1306_128X32_UNIVISION_HW_I2C | ||||
|   SH1106,       // U8X8_SH1106_128X64_WINSTAR_HW_I2C | ||||
|   SSD1306_64,   // U8X8_SSD1306_128X64_NONAME_HW_I2C | ||||
|   SSD1305,      // U8X8_SSD1305_128X32_ADAFRUIT_HW_I2C | ||||
|   SSD1305_64,   // U8X8_SSD1305_128X64_ADAFRUIT_HW_I2C | ||||
|   SSD1306_SPI,  // U8X8_SSD1306_128X32_NONAME_HW_SPI | ||||
|   SSD1306_SPI64 // U8X8_SSD1306_128X64_NONAME_HW_SPI | ||||
| } DisplayType; | ||||
|  | ||||
| class FourLineDisplayUsermod : public Usermod { | ||||
|  | ||||
|   private: | ||||
|  | ||||
|     bool initDone = false; | ||||
|     unsigned long lastTime = 0; | ||||
|  | ||||
|     // HW interface & configuration | ||||
|     U8X8 *u8x8 = nullptr;           // pointer to U8X8 display object | ||||
|     #ifndef FLD_SPI_DEFAULT | ||||
|     int8_t ioPin[5] = {FLD_PIN_SCL, FLD_PIN_SDA, -1, -1, -1};        // I2C pins: SCL, SDA | ||||
|     uint32_t ioFrequency = 400000;  // in Hz (minimum is 100000, baseline is 400000 and maximum should be 3400000) | ||||
|     #else | ||||
|     int8_t ioPin[5] = {FLD_PIN_CLOCKSPI, FLD_PIN_DATASPI, FLD_PIN_CS, FLD_PIN_DC, FLD_PIN_RESET}; // SPI pins: CLK, MOSI, CS, DC, RST | ||||
|     uint32_t ioFrequency = 1000000;  // in Hz (minimum is 500kHz, baseline is 1MHz and maximum should be 20MHz) | ||||
|     #endif | ||||
|     DisplayType type = FLD_TYPE;    // display type | ||||
|     bool flip = false;              // flip display 180° | ||||
|     uint8_t contrast = 10;          // screen contrast | ||||
|     uint8_t lineHeight = 1;         // 1 row or 2 rows | ||||
|     uint32_t refreshRate = USER_LOOP_REFRESH_RATE_MS; // in ms | ||||
|     uint32_t screenTimeout = SCREEN_TIMEOUT_MS;       // in ms | ||||
|     bool sleepMode = true;          // allow screen sleep? | ||||
|     bool clockMode = false;         // display clock | ||||
|     bool enabled = true; | ||||
|  | ||||
|     // Next variables hold the previous known values to determine if redraw is | ||||
|     // required. | ||||
|     String knownSsid = ""; | ||||
|     IPAddress knownIp; | ||||
|     uint8_t knownBrightness = 0; | ||||
|     uint8_t knownEffectSpeed = 0; | ||||
|     uint8_t knownEffectIntensity = 0; | ||||
|     uint8_t knownMode = 0; | ||||
|     uint8_t knownPalette = 0; | ||||
|     uint8_t knownMinute = 99; | ||||
|     uint8_t knownHour = 99; | ||||
|  | ||||
|     bool displayTurnedOff = false; | ||||
|     unsigned long lastUpdate = 0; | ||||
|     unsigned long lastRedraw = 0; | ||||
|     unsigned long overlayUntil = 0; | ||||
|     Line4Type lineType = FLD_LINE_BRIGHTNESS; | ||||
|     // Set to 2 or 3 to mark lines 2 or 3. Other values ignored. | ||||
|     byte markLineNum = 0; | ||||
|  | ||||
|     // strings to reduce flash memory usage (used more than twice) | ||||
|     static const char _name[]; | ||||
|     static const char _enabled[]; | ||||
|     static const char _contrast[]; | ||||
|     static const char _refreshRate[]; | ||||
|     static const char _screenTimeOut[]; | ||||
|     static const char _flip[]; | ||||
|     static const char _sleepMode[]; | ||||
|     static const char _clockMode[]; | ||||
|     static const char _busClkFrequency[]; | ||||
|  | ||||
|     // If display does not work or looks corrupted check the | ||||
|     // constructor reference: | ||||
|     // https://github.com/olikraus/u8g2/wiki/u8x8setupcpp | ||||
|     // or check the gallery: | ||||
|     // https://github.com/olikraus/u8g2/wiki/gallery | ||||
|  | ||||
|   public: | ||||
|  | ||||
|     // gets called once at boot. Do all initialization that doesn't depend on | ||||
|     // network here | ||||
|     void setup() { | ||||
|       if (type == NONE || !enabled) return; | ||||
|  | ||||
|       bool isHW; | ||||
|       PinOwner po = PinOwner::UM_FourLineDisplay; | ||||
|       if (type == SSD1306_SPI || type == SSD1306_SPI64) { | ||||
|         isHW = (ioPin[0]==spi_sclk && ioPin[1]==spi_mosi); | ||||
|         if (isHW) po = PinOwner::HW_SPI;  // allow multiple allocations of HW I2C bus pins | ||||
|         PinManagerPinType pins[5] = { { ioPin[0], true }, { ioPin[1], true }, { ioPin[2], true }, { ioPin[3], true }, { ioPin[4], true }}; | ||||
|         if (!pinManager.allocateMultiplePins(pins, 5, po)) { type=NONE; return; } | ||||
|       } else { | ||||
|         isHW = (ioPin[0]==i2c_scl && ioPin[1]==i2c_sda); | ||||
|         if (isHW) po = PinOwner::HW_I2C;  // allow multiple allocations of HW I2C bus pins | ||||
|         PinManagerPinType pins[2] = { { ioPin[0], true }, { ioPin[1], true } }; | ||||
|         if (!pinManager.allocateMultiplePins(pins, 2, po)) { type=NONE; return; } | ||||
|       } | ||||
|  | ||||
|       DEBUG_PRINTLN(F("Allocating display.")); | ||||
|       switch (type) { | ||||
|         case SSD1306: | ||||
|           if (!isHW) u8x8 = (U8X8 *) new U8X8_SSD1306_128X32_UNIVISION_SW_I2C(ioPin[0], ioPin[1]); // SCL, SDA, reset | ||||
|           else       u8x8 = (U8X8 *) new U8X8_SSD1306_128X32_UNIVISION_HW_I2C(U8X8_PIN_NONE, ioPin[0], ioPin[1]); // Pins are Reset, SCL, SDA | ||||
|           lineHeight = 1; | ||||
|           break; | ||||
|         case SH1106: | ||||
|           if (!isHW) u8x8 = (U8X8 *) new U8X8_SH1106_128X64_WINSTAR_SW_I2C(ioPin[0], ioPin[1]); // SCL, SDA, reset | ||||
|           else       u8x8 = (U8X8 *) new U8X8_SH1106_128X64_WINSTAR_HW_I2C(U8X8_PIN_NONE, ioPin[0], ioPin[1]); // Pins are Reset, SCL, SDA | ||||
|           lineHeight = 2; | ||||
|           break; | ||||
|         case SSD1306_64: | ||||
|           if (!isHW) u8x8 = (U8X8 *) new U8X8_SSD1306_128X64_NONAME_SW_I2C(ioPin[0], ioPin[1]); // SCL, SDA, reset | ||||
|           else       u8x8 = (U8X8 *) new U8X8_SSD1306_128X64_NONAME_HW_I2C(U8X8_PIN_NONE, ioPin[0], ioPin[1]); // Pins are Reset, SCL, SDA | ||||
|           lineHeight = 2; | ||||
|           break; | ||||
|         case SSD1305: | ||||
|           if (!isHW) u8x8 = (U8X8 *) new U8X8_SSD1305_128X32_NONAME_SW_I2C(ioPin[0], ioPin[1]); // SCL, SDA, reset | ||||
|           else       u8x8 = (U8X8 *) new U8X8_SSD1305_128X32_ADAFRUIT_HW_I2C(U8X8_PIN_NONE, ioPin[0], ioPin[1]); // Pins are Reset, SCL, SDA | ||||
|           lineHeight = 1; | ||||
|           break; | ||||
|         case SSD1305_64: | ||||
|           if (!isHW) u8x8 = (U8X8 *) new U8X8_SSD1305_128X64_ADAFRUIT_SW_I2C(ioPin[0], ioPin[1]); // SCL, SDA, reset | ||||
|           else       u8x8 = (U8X8 *) new U8X8_SSD1305_128X64_ADAFRUIT_HW_I2C(U8X8_PIN_NONE, ioPin[0], ioPin[1]); // Pins are Reset, SCL, SDA | ||||
|           lineHeight = 2; | ||||
|           break; | ||||
|         case SSD1306_SPI: | ||||
|           if (!isHW)  u8x8 = (U8X8 *) new U8X8_SSD1306_128X32_UNIVISION_4W_SW_SPI(ioPin[0], ioPin[1], ioPin[2], ioPin[3], ioPin[4]); | ||||
|           else        u8x8 = (U8X8 *) new U8X8_SSD1306_128X32_UNIVISION_4W_HW_SPI(ioPin[2], ioPin[3], ioPin[4]); // Pins are cs, dc, reset | ||||
|           lineHeight = 1; | ||||
|           break; | ||||
|         case SSD1306_SPI64: | ||||
|           if (!isHW) u8x8 = (U8X8 *) new U8X8_SSD1306_128X64_NONAME_4W_SW_SPI(ioPin[0], ioPin[1], ioPin[2], ioPin[3], ioPin[4]); | ||||
|           else       u8x8 = (U8X8 *) new U8X8_SSD1306_128X64_NONAME_4W_HW_SPI(ioPin[2], ioPin[3], ioPin[4]); // Pins are cs, dc, reset | ||||
|           lineHeight = 2; | ||||
|           break; | ||||
|         default: | ||||
|           u8x8 = nullptr; | ||||
|       } | ||||
|  | ||||
|       if (nullptr == u8x8) { | ||||
|           DEBUG_PRINTLN(F("Display init failed.")); | ||||
|           pinManager.deallocateMultiplePins((const uint8_t*)ioPin, (type == SSD1306_SPI || type == SSD1306_SPI64) ? 5 : 2, po); | ||||
|           type = NONE; | ||||
|           return; | ||||
|       } | ||||
|  | ||||
|       initDone = true; | ||||
|       DEBUG_PRINTLN(F("Starting display.")); | ||||
|       /*if (!(type == SSD1306_SPI || type == SSD1306_SPI64))*/ u8x8->setBusClock(ioFrequency);  // can be used for SPI too | ||||
|       u8x8->begin(); | ||||
|       setFlipMode(flip); | ||||
|       setContrast(contrast); //Contrast setup will help to preserve OLED lifetime. In case OLED need to be brighter increase number up to 255 | ||||
|       setPowerSave(0); | ||||
|       drawString(0, 0, "Loading..."); | ||||
|     } | ||||
|  | ||||
|     // gets called every time WiFi is (re-)connected. Initialize own network | ||||
|     // interfaces here | ||||
|     void connected() {} | ||||
|  | ||||
|     /** | ||||
|      * Da loop. | ||||
|      */ | ||||
|     void loop() { | ||||
|       if (!enabled || millis() - lastUpdate < (clockMode?1000:refreshRate) || strip.isUpdating()) return; | ||||
|       lastUpdate = millis(); | ||||
|  | ||||
|       redraw(false); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Wrappers for screen drawing | ||||
|      */ | ||||
|     void setFlipMode(uint8_t mode) { | ||||
|       if (type == NONE || !enabled) return; | ||||
|       u8x8->setFlipMode(mode); | ||||
|     } | ||||
|     void setContrast(uint8_t contrast) { | ||||
|       if (type == NONE || !enabled) return; | ||||
|       u8x8->setContrast(contrast); | ||||
|     } | ||||
|     void drawString(uint8_t col, uint8_t row, const char *string, bool ignoreLH=false) { | ||||
|       if (type == NONE || !enabled) return; | ||||
|       u8x8->setFont(u8x8_font_chroma48medium8_r); | ||||
|       if (!ignoreLH && lineHeight==2) u8x8->draw1x2String(col, row, string); | ||||
|       else                            u8x8->drawString(col, row, string); | ||||
|     } | ||||
|     void draw2x2String(uint8_t col, uint8_t row, const char *string) { | ||||
|       if (type == NONE || !enabled) return; | ||||
|       u8x8->setFont(u8x8_font_chroma48medium8_r); | ||||
|       u8x8->draw2x2String(col, row, string); | ||||
|     } | ||||
|     void drawGlyph(uint8_t col, uint8_t row, char glyph, const uint8_t *font, bool ignoreLH=false) { | ||||
|       if (type == NONE || !enabled) return; | ||||
|       u8x8->setFont(font); | ||||
|       if (!ignoreLH && lineHeight==2) u8x8->draw1x2Glyph(col, row, glyph); | ||||
|       else                            u8x8->drawGlyph(col, row, glyph); | ||||
|     } | ||||
|     uint8_t getCols() { | ||||
|       if (type==NONE || !enabled) return 0; | ||||
|       return u8x8->getCols(); | ||||
|     } | ||||
|     void clear() { | ||||
|       if (type == NONE || !enabled) return; | ||||
|       u8x8->clear(); | ||||
|     } | ||||
|     void setPowerSave(uint8_t save) { | ||||
|       if (type == NONE || !enabled) return; | ||||
|       u8x8->setPowerSave(save); | ||||
|     } | ||||
|  | ||||
|     void center(String &line, uint8_t width) { | ||||
|       int len = line.length(); | ||||
|       if (len<width) for (byte i=(width-len)/2; i>0; i--) line = ' ' + line; | ||||
|       for (byte i=line.length(); i<width; i++) line += ' '; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Redraw the screen (but only if things have changed | ||||
|      * or if forceRedraw). | ||||
|      */ | ||||
|     void redraw(bool forceRedraw) { | ||||
|       static bool showName = false; | ||||
|       unsigned long now = millis(); | ||||
|  | ||||
|       if (type == NONE || !enabled) return; | ||||
|       if (overlayUntil > 0) { | ||||
|         if (now >= overlayUntil) { | ||||
|           // Time to display the overlay has elapsed. | ||||
|           overlayUntil = 0; | ||||
|           forceRedraw = true; | ||||
|         } else { | ||||
|           // We are still displaying the overlay | ||||
|           // Don't redraw. | ||||
|           return; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       // Check if values which are shown on display changed from the last time. | ||||
|       if (forceRedraw || | ||||
|           (((apActive) ? String(apSSID) : WiFi.SSID()) != knownSsid) || | ||||
|           (knownIp != (apActive ? IPAddress(4, 3, 2, 1) : Network.localIP())) || | ||||
|           (knownBrightness != bri) || | ||||
|           (knownEffectSpeed != effectSpeed) || | ||||
|           (knownEffectIntensity != effectIntensity) || | ||||
|           (knownMode != strip.getMainSegment().mode) || | ||||
|           (knownPalette != strip.getMainSegment().palette)) { | ||||
|         knownHour = 99;   // force time update | ||||
|         lastRedraw = now; // update lastRedraw marker | ||||
|       } else if (sleepMode && !displayTurnedOff && ((now - lastRedraw)/1000)%5 == 0) { | ||||
|         // change line every 5s | ||||
|         showName = !showName; | ||||
|         switch (lineType) { | ||||
|           case FLD_LINE_BRIGHTNESS: | ||||
|             lineType = FLD_LINE_EFFECT_SPEED; | ||||
|             break; | ||||
|           case FLD_LINE_MODE: | ||||
|             lineType = FLD_LINE_BRIGHTNESS; | ||||
|             break; | ||||
|           case FLD_LINE_PALETTE: | ||||
|             lineType = clockMode ? FLD_LINE_MODE : FLD_LINE_BRIGHTNESS; | ||||
|             break; | ||||
|           case FLD_LINE_EFFECT_SPEED: | ||||
|             lineType = FLD_LINE_EFFECT_INTENSITY; | ||||
|             break; | ||||
|           case FLD_LINE_EFFECT_INTENSITY: | ||||
|             lineType = FLD_LINE_PALETTE; | ||||
|             break; | ||||
|           default: | ||||
|             lineType = FLD_LINE_MODE; | ||||
|             break; | ||||
|         } | ||||
|         knownHour = 99; // force time update | ||||
|         // do not update lastRedraw marker if just switching row contenet | ||||
|       } else { | ||||
|         // Nothing to change. | ||||
|         // Turn off display after 3 minutes with no change. | ||||
|         if(sleepMode && !displayTurnedOff && (millis() - lastRedraw > screenTimeout)) { | ||||
|           // We will still check if there is a change in redraw() | ||||
|           // and turn it back on if it changed. | ||||
|           sleepOrClock(true); | ||||
|         } else if (displayTurnedOff && clockMode) { | ||||
|           showTime(); | ||||
|         } | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       // Turn the display back on | ||||
|       if (displayTurnedOff) sleepOrClock(false); | ||||
|  | ||||
|       // Update last known values. | ||||
|       knownSsid = apActive ? WiFi.softAPSSID() : WiFi.SSID(); | ||||
|       knownIp = apActive ? IPAddress(4, 3, 2, 1) : Network.localIP(); | ||||
|       knownBrightness = bri; | ||||
|       knownMode = strip.getMainSegment().mode; | ||||
|       knownPalette = strip.getMainSegment().palette; | ||||
|       knownEffectSpeed = effectSpeed; | ||||
|       knownEffectIntensity = effectIntensity; | ||||
|  | ||||
|       // Do the actual drawing | ||||
|       String line; | ||||
|       // First row with Wifi name | ||||
|       drawGlyph(0, 0, 80, u8x8_font_open_iconic_embedded_1x1); // home icon | ||||
|       line = knownSsid.substring(0, getCols() > 1 ? getCols() - 2 : 0); | ||||
|       center(line, getCols()-2); | ||||
|       drawString(1, 0, line.c_str()); | ||||
|       // Print `~` char to indicate that SSID is longer, than our display | ||||
|       if (knownSsid.length() > (int)getCols()-1) { | ||||
|         drawString(getCols() - 1, 0, "~"); | ||||
|       } | ||||
|  | ||||
|       // Second row with IP or Password | ||||
|       drawGlyph(0, lineHeight, 68, u8x8_font_open_iconic_embedded_1x1); // wifi icon | ||||
|       // Print password in AP mode and if led is OFF. | ||||
|       if (apActive && bri == 0) { | ||||
|         drawString(1, lineHeight, apPass); | ||||
|       } else { | ||||
|         // alternate IP address and server name | ||||
|         line = knownIp.toString(); | ||||
|         if (showName && strcmp(serverDescription, "WLED") != 0) { | ||||
|           line = serverDescription; | ||||
|         } | ||||
|         center(line, getCols()-1); | ||||
|         drawString(1, lineHeight, line.c_str()); | ||||
|       } | ||||
|  | ||||
|       // draw third and fourth row | ||||
|       drawLine(2, clockMode ? lineType : FLD_LINE_MODE); | ||||
|       drawLine(3, clockMode ? FLD_LINE_TIME : lineType); | ||||
|  | ||||
|       drawGlyph(0, 2*lineHeight, 66 + (bri > 0 ? 3 : 0), u8x8_font_open_iconic_weather_2x2); // sun/moon icon | ||||
|       //if (markLineNum>1) drawGlyph(2, markLineNum*lineHeight, 66, u8x8_font_open_iconic_arrow_1x1); // arrow icon | ||||
|     } | ||||
|  | ||||
|     void drawLine(uint8_t line, Line4Type lineType) { | ||||
|       char lineBuffer[LINE_BUFFER_SIZE]; | ||||
|       uint8_t printedChars; | ||||
|       switch(lineType) { | ||||
|         case FLD_LINE_BRIGHTNESS: | ||||
|           sprintf_P(lineBuffer, PSTR("Brightness %3d"), bri); | ||||
|           drawString(2, line*lineHeight, lineBuffer); | ||||
|           break; | ||||
|         case FLD_LINE_EFFECT_SPEED: | ||||
|           sprintf_P(lineBuffer, PSTR("FX Speed   %3d"), effectSpeed); | ||||
|           drawString(2, line*lineHeight, lineBuffer); | ||||
|           break; | ||||
|         case FLD_LINE_EFFECT_INTENSITY: | ||||
|           sprintf_P(lineBuffer, PSTR("FX Intens. %3d"), effectIntensity); | ||||
|           drawString(2, line*lineHeight, lineBuffer); | ||||
|           break; | ||||
|         case FLD_LINE_MODE: | ||||
|           printedChars = extractModeName(knownMode, JSON_mode_names, lineBuffer, LINE_BUFFER_SIZE-1); | ||||
|           for (;printedChars < getCols()-2 && printedChars < LINE_BUFFER_SIZE-3; printedChars++) lineBuffer[printedChars]=' '; | ||||
|           lineBuffer[printedChars] = 0; | ||||
|           drawString(2, line*lineHeight, lineBuffer); | ||||
|           break; | ||||
|         case FLD_LINE_PALETTE: | ||||
|           printedChars = extractModeName(knownPalette, JSON_palette_names, lineBuffer, LINE_BUFFER_SIZE-1); | ||||
|           for (;printedChars < getCols()-2 && printedChars < LINE_BUFFER_SIZE-3; printedChars++) lineBuffer[printedChars]=' '; | ||||
|           lineBuffer[printedChars] = 0; | ||||
|           drawString(2, line*lineHeight, lineBuffer); | ||||
|           break; | ||||
|         case FLD_LINE_TIME: | ||||
|         default: | ||||
|           showTime(false); | ||||
|           break; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * If there screen is off or in clock is displayed, | ||||
|      * this will return true. This allows us to throw away | ||||
|      * the first input from the rotary encoder but | ||||
|      * to wake up the screen. | ||||
|      */ | ||||
|     bool wakeDisplay() { | ||||
|       if (type == NONE || !enabled) return false; | ||||
|       knownHour = 99; | ||||
|       if (displayTurnedOff) { | ||||
|         // Turn the display back on | ||||
|         sleepOrClock(false); | ||||
|         redraw(true); | ||||
|         return true; | ||||
|       } | ||||
|       return false; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Allows you to show up to two lines as overlay for a | ||||
|      * period of time. | ||||
|      * Clears the screen and prints on the middle two lines. | ||||
|      */ | ||||
|     void overlay(const char* line1, const char *line2, long showHowLong) { | ||||
|       if (type == NONE || !enabled) return; | ||||
|  | ||||
|       if (displayTurnedOff) { | ||||
|         // Turn the display back on (includes clear()) | ||||
|         sleepOrClock(false); | ||||
|       } else { | ||||
|         clear(); | ||||
|       } | ||||
|  | ||||
|       // Print the overlay | ||||
|       if (line1) { | ||||
|         String buf = line1; | ||||
|         center(buf, getCols()); | ||||
|         drawString(0, 1*lineHeight, buf.c_str()); | ||||
|       } | ||||
|       if (line2) { | ||||
|         String buf = line2; | ||||
|         center(buf, getCols()); | ||||
|         drawString(0, 2*lineHeight, buf.c_str()); | ||||
|       } | ||||
|       overlayUntil = millis() + showHowLong; | ||||
|     } | ||||
|  | ||||
|     void setLineType(byte lT) { | ||||
|       lineType = (Line4Type) lT; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Line 3 or 4 (last two lines) can be marked with an | ||||
|      * arrow in the first column. Pass 2 or 3 to this to | ||||
|      * specify which line to mark with an arrow. | ||||
|      * Any other values are ignored. | ||||
|      */ | ||||
|     void setMarkLine(byte newMarkLineNum) { | ||||
|       if (newMarkLineNum == 2 || newMarkLineNum == 3) { | ||||
|         markLineNum = newMarkLineNum; | ||||
|       } | ||||
|       else { | ||||
|         markLineNum = 0; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Enable sleep (turn the display off) or clock mode. | ||||
|      */ | ||||
|     void sleepOrClock(bool enabled) { | ||||
|       clear(); | ||||
|       if (enabled) { | ||||
|         if (clockMode) showTime(); | ||||
|         else           setPowerSave(1); | ||||
|         displayTurnedOff = true; | ||||
|       } else { | ||||
|         setPowerSave(0); | ||||
|         displayTurnedOff = false; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Display the current date and time in large characters | ||||
|      * on the middle rows. Based 24 or 12 hour depending on | ||||
|      * the useAMPM configuration. | ||||
|      */ | ||||
|     void showTime(bool fullScreen = true) { | ||||
|       if (type == NONE || !enabled) return; | ||||
|       char lineBuffer[LINE_BUFFER_SIZE]; | ||||
|  | ||||
|       updateLocalTime(); | ||||
|       byte minuteCurrent = minute(localTime); | ||||
|       byte hourCurrent   = hour(localTime); | ||||
|       byte secondCurrent = second(localTime); | ||||
|       if (knownMinute == minuteCurrent && knownHour == hourCurrent) { | ||||
|         // Time hasn't changed. | ||||
|         if (!fullScreen) return; | ||||
|       } | ||||
|       knownMinute = minuteCurrent; | ||||
|       knownHour = hourCurrent; | ||||
|  | ||||
|       byte currentMonth = month(localTime); | ||||
|       sprintf_P(lineBuffer, PSTR("%s %2d "), monthShortStr(currentMonth), day(localTime)); | ||||
|       if (fullScreen) | ||||
|         draw2x2String(DATE_INDENT, lineHeight==1 ? 0 : lineHeight, lineBuffer); // adjust for 8 line displays | ||||
|       else | ||||
|         drawString(2, lineHeight*3, lineBuffer); | ||||
|  | ||||
|       byte showHour = hourCurrent; | ||||
|       boolean isAM = false; | ||||
|       if (useAMPM) { | ||||
|         if (showHour == 0) { | ||||
|           showHour = 12; | ||||
|           isAM = true; | ||||
|         }  | ||||
|         else if (showHour > 12) { | ||||
|           showHour -= 12; | ||||
|           isAM = false; | ||||
|         } | ||||
|         else { | ||||
|           isAM = true; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       sprintf_P(lineBuffer, (secondCurrent%2 || !fullScreen) ? PSTR("%2d:%02d") : PSTR("%2d %02d"), (useAMPM ? showHour : hourCurrent), minuteCurrent); | ||||
|       // For time, we always use LINE_HEIGHT of 2 since | ||||
|       // we are printing it big. | ||||
|       if (fullScreen) { | ||||
|         draw2x2String(TIME_INDENT+2, lineHeight*2, lineBuffer); | ||||
|         sprintf_P(lineBuffer, PSTR("%02d"), secondCurrent); | ||||
|         if (useAMPM) drawString(12+(fullScreen?0:2), lineHeight*2, (isAM ? "AM" : "PM"), true); | ||||
|         else         drawString(12, lineHeight*2+1, lineBuffer, true); // even with double sized rows print seconds in 1 line | ||||
|       } else { | ||||
|         drawString(9+(useAMPM?0:2), lineHeight*3, lineBuffer); | ||||
|         if (useAMPM) drawString(12+(fullScreen?0:2), lineHeight*3, (isAM ? "AM" : "PM"), true); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * addToJsonInfo() can be used to add custom entries to the /json/info part of the JSON API. | ||||
|      * Creating an "u" object allows you to add custom key/value pairs to the Info section of the WLED web UI. | ||||
|      * Below it is shown how this could be used for e.g. a light sensor | ||||
|      */ | ||||
|     //void addToJsonInfo(JsonObject& root) { | ||||
|       //JsonObject user = root["u"]; | ||||
|       //if (user.isNull()) user = root.createNestedObject("u"); | ||||
|       //JsonArray data = user.createNestedArray(F("4LineDisplay")); | ||||
|       //data.add(F("Loaded.")); | ||||
|     //} | ||||
|  | ||||
|     /* | ||||
|      * addToJsonState() can be used to add custom entries to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     //void addToJsonState(JsonObject& root) { | ||||
|     //} | ||||
|  | ||||
|     /* | ||||
|      * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     //void readFromJsonState(JsonObject& root) { | ||||
|     //  if (!initDone) return;  // prevent crash on boot applyPreset() | ||||
|     //} | ||||
|  | ||||
|     /* | ||||
|      * addToConfig() can be used to add custom persistent settings to the cfg.json file in the "um" (usermod) object. | ||||
|      * It will be called by WLED when settings are actually saved (for example, LED settings are saved) | ||||
|      * If you want to force saving the current state, use serializeConfig() in your loop(). | ||||
|      *  | ||||
|      * CAUTION: serializeConfig() will initiate a filesystem write operation. | ||||
|      * It might cause the LEDs to stutter and will cause flash wear if called too often. | ||||
|      * Use it sparingly and always in the loop, never in network callbacks! | ||||
|      *  | ||||
|      * addToConfig() will also not yet add your setting to one of the settings pages automatically. | ||||
|      * To make that work you still have to add the setting to the HTML, xml.cpp and set.cpp manually. | ||||
|      *  | ||||
|      * I highly recommend checking out the basics of ArduinoJson serialization and deserialization in order to use custom settings! | ||||
|      */ | ||||
|     void addToConfig(JsonObject& root) { | ||||
|       JsonObject top   = root.createNestedObject(FPSTR(_name)); | ||||
|       top[FPSTR(_enabled)]       = enabled; | ||||
|       JsonArray io_pin = top.createNestedArray("pin"); | ||||
|       for (byte i=0; i<5; i++) io_pin.add(ioPin[i]); | ||||
|       top["help4Pins"]           = F("Clk,Data,CS,DC,RST"); // help for Settings page | ||||
|       top["type"]                = type; | ||||
|       top["help4Type"]           = F("1=SSD1306,2=SH1106,3=SSD1306_128x64,4=SSD1305,5=SSD1305_128x64,6=SSD1306_SPI,7=SSD1306_SPI_128x64"); // help for Settings page | ||||
|       top[FPSTR(_flip)]          = (bool) flip; | ||||
|       top[FPSTR(_contrast)]      = contrast; | ||||
|       top[FPSTR(_refreshRate)]   = refreshRate/1000; | ||||
|       top[FPSTR(_screenTimeOut)] = screenTimeout/1000; | ||||
|       top[FPSTR(_sleepMode)]     = (bool) sleepMode; | ||||
|       top[FPSTR(_clockMode)]     = (bool) clockMode; | ||||
|       top[FPSTR(_busClkFrequency)] = ioFrequency/1000; | ||||
|       DEBUG_PRINTLN(F("4 Line Display config saved.")); | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * readFromConfig() can be used to read back the custom settings you added with addToConfig(). | ||||
|      * This is called by WLED when settings are loaded (currently this only happens once immediately after boot) | ||||
|      *  | ||||
|      * readFromConfig() is called BEFORE setup(). This means you can use your persistent values in setup() (e.g. pin assignments, buffer sizes), | ||||
|      * but also that if you want to write persistent values to a dynamic buffer, you'd need to allocate it here instead of in setup. | ||||
|      * If you don't know what that is, don't fret. It most likely doesn't affect your use case :) | ||||
|      */ | ||||
|     bool readFromConfig(JsonObject& root) { | ||||
|       bool needsRedraw    = false; | ||||
|       DisplayType newType = type; | ||||
|       int8_t newPin[5]; for (byte i=0; i<5; i++) newPin[i] = ioPin[i]; | ||||
|  | ||||
|       JsonObject top = root[FPSTR(_name)]; | ||||
|       if (top.isNull()) { | ||||
|         DEBUG_PRINT(FPSTR(_name)); | ||||
|         DEBUG_PRINTLN(F(": No config found. (Using defaults.)")); | ||||
|         return false; | ||||
|       } | ||||
|  | ||||
|       enabled       = top[FPSTR(_enabled)] | enabled; | ||||
|       newType       = top["type"] | newType; | ||||
|       for (byte i=0; i<5; i++) newPin[i] = top["pin"][i] | ioPin[i]; | ||||
|       flip          = top[FPSTR(_flip)] | flip; | ||||
|       contrast      = top[FPSTR(_contrast)] | contrast; | ||||
|       refreshRate   = (top[FPSTR(_refreshRate)] | refreshRate/1000) * 1000; | ||||
|       screenTimeout = (top[FPSTR(_screenTimeOut)] | screenTimeout/1000) * 1000; | ||||
|       sleepMode     = top[FPSTR(_sleepMode)] | sleepMode; | ||||
|       clockMode     = top[FPSTR(_clockMode)] | clockMode; | ||||
|       if (newType == SSD1306_SPI || newType == SSD1306_SPI64) | ||||
|         ioFrequency = min(20000, max(500, (int)(top[FPSTR(_busClkFrequency)] | ioFrequency/1000))) * 1000;  // limit frequency | ||||
|       else | ||||
|         ioFrequency = min(3400, max(100, (int)(top[FPSTR(_busClkFrequency)] | ioFrequency/1000))) * 1000;  // limit frequency | ||||
|  | ||||
|       DEBUG_PRINT(FPSTR(_name)); | ||||
|       if (!initDone) { | ||||
|         // first run: reading from cfg.json | ||||
|         for (byte i=0; i<5; i++) ioPin[i] = newPin[i]; | ||||
|         type = newType; | ||||
|         DEBUG_PRINTLN(F(" config loaded.")); | ||||
|       } else { | ||||
|         DEBUG_PRINTLN(F(" config (re)loaded.")); | ||||
|         // changing parameters from settings page | ||||
|         bool pinsChanged = false; | ||||
|         for (byte i=0; i<5; i++) if (ioPin[i] != newPin[i]) { pinsChanged = true; break; } | ||||
|         if (pinsChanged || type!=newType) { | ||||
|           if (type != NONE) delete u8x8; | ||||
|           PinOwner po = PinOwner::UM_FourLineDisplay; | ||||
|           bool isSPI = (type == SSD1306_SPI || type == SSD1306_SPI64); | ||||
|           if (isSPI) { | ||||
|             if (ioPin[0]==spi_sclk && ioPin[1]==spi_mosi) po = PinOwner::HW_SPI;  // allow multiple allocations of HW SPI bus pins | ||||
|             pinManager.deallocateMultiplePins((const uint8_t *)ioPin, 5, po); | ||||
|           } else { | ||||
|             if (ioPin[0]==i2c_scl && ioPin[1]==i2c_sda) po = PinOwner::HW_I2C;  // allow multiple allocations of HW I2C bus pins | ||||
|             pinManager.deallocateMultiplePins((const uint8_t *)ioPin, 2, po); | ||||
|           } | ||||
|           for (byte i=0; i<5; i++) ioPin[i] = newPin[i]; | ||||
|           if (ioPin[0]<0 || ioPin[1]<0) { // data & clock must be > -1 | ||||
|             type = NONE; | ||||
|             return true; | ||||
|           } else type = newType; | ||||
|           setup(); | ||||
|           needsRedraw |= true; | ||||
|         } | ||||
|         if (!(type == SSD1306_SPI || type == SSD1306_SPI64)) u8x8->setBusClock(ioFrequency); // can be used for SPI too | ||||
|         setContrast(contrast); | ||||
|         setFlipMode(flip); | ||||
|         if (needsRedraw && !wakeDisplay()) redraw(true); | ||||
|       } | ||||
|       // use "return !top["newestParameter"].isNull();" when updating Usermod with new features | ||||
|       return !top[FPSTR(_enabled)].isNull(); | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
|      * This could be used in the future for the system to determine whether your usermod is installed. | ||||
|      */ | ||||
|     uint16_t getId() { | ||||
|       return USERMOD_ID_FOUR_LINE_DISP; | ||||
|     } | ||||
| }; | ||||
|  | ||||
| // strings to reduce flash memory usage (used more than twice) | ||||
| const char FourLineDisplayUsermod::_name[]            PROGMEM = "4LineDisplay"; | ||||
| const char FourLineDisplayUsermod::_enabled[]         PROGMEM = "enabled"; | ||||
| const char FourLineDisplayUsermod::_contrast[]        PROGMEM = "contrast"; | ||||
| const char FourLineDisplayUsermod::_refreshRate[]     PROGMEM = "refreshRateSec"; | ||||
| const char FourLineDisplayUsermod::_screenTimeOut[]   PROGMEM = "screenTimeOutSec"; | ||||
| const char FourLineDisplayUsermod::_flip[]            PROGMEM = "flip"; | ||||
| const char FourLineDisplayUsermod::_sleepMode[]       PROGMEM = "sleepMode"; | ||||
| const char FourLineDisplayUsermod::_clockMode[]       PROGMEM = "clockMode"; | ||||
| const char FourLineDisplayUsermod::_busClkFrequency[] PROGMEM = "i2c-freq-kHz"; | ||||
| @@ -211,16 +211,16 @@ class FourLineDisplayUsermod : public Usermod { | ||||
|  | ||||
|     // gets called once at boot. Do all initialization that doesn't depend on | ||||
|     // network here | ||||
|     void setup(); | ||||
|     void setup() override; | ||||
|  | ||||
|     // gets called every time WiFi is (re-)connected. Initialize own network | ||||
|     // interfaces here | ||||
|     void connected(); | ||||
|     void connected() override; | ||||
|  | ||||
|     /** | ||||
|      * Da loop. | ||||
|      */ | ||||
|     void loop(); | ||||
|     void loop() override; | ||||
|  | ||||
|     //function to update lastredraw | ||||
|     inline void updateRedrawTime() { lastRedraw = millis(); } | ||||
| @@ -287,28 +287,28 @@ class FourLineDisplayUsermod : public Usermod { | ||||
|      */ | ||||
|     bool handleButton(uint8_t b); | ||||
|  | ||||
|     void onUpdateBegin(bool init); | ||||
|     void onUpdateBegin(bool init) override; | ||||
|  | ||||
|     /* | ||||
|      * addToJsonInfo() can be used to add custom entries to the /json/info part of the JSON API. | ||||
|      * Creating an "u" object allows you to add custom key/value pairs to the Info section of the WLED web UI. | ||||
|      * Below it is shown how this could be used for e.g. a light sensor | ||||
|      */ | ||||
|     //void addToJsonInfo(JsonObject& root); | ||||
|     //void addToJsonInfo(JsonObject& root) override; | ||||
|  | ||||
|     /* | ||||
|      * addToJsonState() can be used to add custom entries to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     //void addToJsonState(JsonObject& root); | ||||
|     //void addToJsonState(JsonObject& root) override; | ||||
|  | ||||
|     /* | ||||
|      * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     //void readFromJsonState(JsonObject& root); | ||||
|     //void readFromJsonState(JsonObject& root) override; | ||||
|  | ||||
|     void appendConfigData(); | ||||
|     void appendConfigData() override; | ||||
|  | ||||
|     /* | ||||
|      * addToConfig() can be used to add custom persistent settings to the cfg.json file in the "um" (usermod) object. | ||||
| @@ -324,7 +324,7 @@ class FourLineDisplayUsermod : public Usermod { | ||||
|      *  | ||||
|      * I highly recommend checking out the basics of ArduinoJson serialization and deserialization in order to use custom settings! | ||||
|      */ | ||||
|     void addToConfig(JsonObject& root); | ||||
|     void addToConfig(JsonObject& root) override; | ||||
|  | ||||
|     /* | ||||
|      * readFromConfig() can be used to read back the custom settings you added with addToConfig(). | ||||
| @@ -334,13 +334,13 @@ class FourLineDisplayUsermod : public Usermod { | ||||
|      * but also that if you want to write persistent values to a dynamic buffer, you'd need to allocate it here instead of in setup. | ||||
|      * If you don't know what that is, don't fret. It most likely doesn't affect your use case :) | ||||
|      */ | ||||
|     bool readFromConfig(JsonObject& root); | ||||
|     bool readFromConfig(JsonObject& root) override; | ||||
|  | ||||
|     /* | ||||
|      * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
|      * This could be used in the future for the system to determine whether your usermod is installed. | ||||
|      */ | ||||
|     uint16_t getId() { | ||||
|     uint16_t getId() override { | ||||
|       return USERMOD_ID_FOUR_LINE_DISP; | ||||
|     } | ||||
| }; | ||||
|   | ||||
| @@ -6,7 +6,7 @@ class klipper_percentage : public Usermod | ||||
| { | ||||
| private: | ||||
|   unsigned long lastTime = 0; | ||||
|   String ip = "192.168.25.207"; | ||||
|   String ip = F("0.0.0.0"); | ||||
|   WiFiClient wifiClient; | ||||
|   char errorMessage[100] = ""; | ||||
|   int printPercent = 0; | ||||
| @@ -30,7 +30,7 @@ private: | ||||
|     { | ||||
|       // Send HTTP request | ||||
|       client.println(F("GET /printer/objects/query?virtual_sdcard=progress HTTP/1.0")); | ||||
|       client.println("Host: " + ip); | ||||
|       client.print(F("Host: ")); client.println(ip); | ||||
|       client.println(F("Connection: close")); | ||||
|       if (client.println() == 0) | ||||
|       { | ||||
| @@ -41,7 +41,7 @@ private: | ||||
|         // Check HTTP status | ||||
|         char status[32] = {0}; | ||||
|         client.readBytesUntil('\r', status, sizeof(status)); | ||||
|         if (strcmp(status, "HTTP/1.1 200 OK") != 0) | ||||
|         if (strcmp_P(status, PSTR("HTTP/1.1 200 OK")) != 0) | ||||
|         { | ||||
|           strcat(errorMessage, PSTR("Unexpected response: ")); | ||||
|           strcat(errorMessage, status); | ||||
| @@ -86,11 +86,11 @@ public: | ||||
|               strcat(errorMessage, PSTR("deserializeJson() failed: ")); | ||||
|               strcat(errorMessage, error.c_str()); | ||||
|             } | ||||
|             printPercent = (int)(klipperDoc["result"]["status"]["virtual_sdcard"]["progress"].as<float>() * 100); | ||||
|             printPercent = (int)(klipperDoc[F("result")][F("status")][F("virtual_sdcard")][F("progress")].as<float>() * 100); | ||||
|  | ||||
|             DEBUG_PRINT("Percent: "); | ||||
|             DEBUG_PRINTLN((int)(klipperDoc["result"]["status"]["virtual_sdcard"]["progress"].as<float>() * 100)); | ||||
|             DEBUG_PRINT("LEDs: "); | ||||
|             DEBUG_PRINT(F("Percent: ")); | ||||
|             DEBUG_PRINTLN((int)(klipperDoc[F("result")][F("status")][F("virtual_sdcard")][F("progress")].as<float>() * 100)); | ||||
|             DEBUG_PRINT(F("LEDs: ")); | ||||
|             DEBUG_PRINTLN(direction == 2 ? (strip.getLengthTotal() / 2) * printPercent / 100 : strip.getLengthTotal() * printPercent / 100); | ||||
|           } | ||||
|           else | ||||
| @@ -106,10 +106,10 @@ public: | ||||
|  | ||||
|   void addToConfig(JsonObject &root) | ||||
|   { | ||||
|     JsonObject top = root.createNestedObject("Klipper Printing Percentage"); | ||||
|     top["Enabled"] = enabled; | ||||
|     top["Klipper IP"] = ip; | ||||
|     top["Direction"] = direction; | ||||
|     JsonObject top = root.createNestedObject(F("Klipper Printing Percentage")); | ||||
|     top[F("Enabled")] = enabled; | ||||
|     top[F("Klipper IP")] = ip; | ||||
|     top[F("Direction")] = direction; | ||||
|   } | ||||
|  | ||||
|   bool readFromConfig(JsonObject &root) | ||||
| @@ -117,12 +117,12 @@ public: | ||||
|     // default settings values could be set here (or below using the 3-argument getJsonValue()) instead of in the class definition or constructor | ||||
|     // setting them inside readFromConfig() is slightly more robust, handling the rare but plausible use case of single value being missing after boot (e.g. if the cfg.json was manually edited and a value was removed) | ||||
|  | ||||
|     JsonObject top = root["Klipper Printing Percentage"]; | ||||
|     JsonObject top = root[F("Klipper Printing Percentage")]; | ||||
|  | ||||
|     bool configComplete = !top.isNull(); | ||||
|     configComplete &= getJsonValue(top["Klipper IP"], ip); | ||||
|     configComplete &= getJsonValue(top["Enabled"], enabled); | ||||
|     configComplete &= getJsonValue(top["Direction"], direction); | ||||
|     configComplete &= getJsonValue(top[F("Klipper IP")], ip); | ||||
|     configComplete &= getJsonValue(top[F("Enabled")], enabled); | ||||
|     configComplete &= getJsonValue(top[F("Direction")], direction); | ||||
|     return configComplete; | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -1,33 +0,0 @@ | ||||
| # Mode Sort | ||||
|  | ||||
| v2 usermod that provides data about modes and | ||||
| palettes to other usermods. Notably it provides: | ||||
| * A direct method for a mode or palette name | ||||
| * Ability to retrieve mode and palette names in  | ||||
|   alphabetical order | ||||
|  | ||||
| ```char **getModesQStrings()``` | ||||
|  | ||||
| Provides a char* array (pointers) to the names of the | ||||
| palettes contained in JSON_mode_names, in the same order as  | ||||
| JSON_mode_names. These strings end in double quote (") | ||||
| (or \0 if there is a problem). | ||||
|  | ||||
| ```byte *getModesAlphaIndexes()``` | ||||
|  | ||||
| A byte array designating the indexes of names of the | ||||
| modes in alphabetical order. "Solid" will always remain  | ||||
| at the top of the list. | ||||
|  | ||||
| ```char **getPalettesQStrings()``` | ||||
|  | ||||
| Provides a char* array (pointers) to the names of the | ||||
| palettes contained in JSON_palette_names, in the same order as  | ||||
| JSON_palette_names. These strings end in double quote (") | ||||
| (or \0 if there is a problem). | ||||
|  | ||||
| ```byte *getPalettesAlphaIndexes()``` | ||||
|  | ||||
| A byte array designating the indexes of names of the | ||||
| palettes in alphabetical order. "Default" and those | ||||
| starting with "(" will always remain at the top of the list. | ||||
| @@ -1,244 +0,0 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| // | ||||
| // v2 usermod that provides data about modes and | ||||
| // palettes to other usermods. Notably it provides: | ||||
| // * A direct method for a mode or palette name | ||||
| // * Ability to retrieve mode and palette names in  | ||||
| //   alphabetical order | ||||
| //  | ||||
| // char **getModesQStrings() | ||||
| // Provides an array of char* (pointers) to the names of the | ||||
| // palettes within JSON_mode_names, in the same order as  | ||||
| // JSON_mode_names. These strings end in double quote (") | ||||
| // (or \0 if there is a problem). | ||||
| // | ||||
| // byte *getModesAlphaIndexes() | ||||
| // An array of byte designating the indexes of names of the | ||||
| // modes in alphabetical order. "Solid" will always remain  | ||||
| // at the front of the list. | ||||
| // | ||||
| // char **getPalettesQStrings() | ||||
| // Provides an array of char* (pointers) to the names of the | ||||
| // palettes within JSON_palette_names, in the same order as  | ||||
| // JSON_palette_names. These strings end in double quote (") | ||||
| // (or \0 if there is a problem). | ||||
| // | ||||
| // byte *getPalettesAlphaIndexes() | ||||
| // An array of byte designating the indexes of names of the | ||||
| // palettes in alphabetical order. "Default" and those | ||||
| // starting with "(" will always remain at the front of the list. | ||||
| // | ||||
|  | ||||
| // Number of modes at the start of the list to not sort | ||||
| #define MODE_SORT_SKIP_COUNT 1 | ||||
|  | ||||
| // Which list is being sorted | ||||
| char **listBeingSorted = nullptr; | ||||
|  | ||||
| /** | ||||
|  * Modes and palettes are stored as strings that | ||||
|  * end in a quote character. Compare two of them. | ||||
|  * We are comparing directly within either | ||||
|  * JSON_mode_names or JSON_palette_names. | ||||
|  */ | ||||
| int re_qstringCmp(const void *ap, const void *bp) { | ||||
|     char *a = listBeingSorted[*((byte *)ap)]; | ||||
|     char *b = listBeingSorted[*((byte *)bp)]; | ||||
|     int i = 0; | ||||
|     do { | ||||
|         char aVal = pgm_read_byte_near(a + i); | ||||
|         if (aVal >= 97 && aVal <= 122) { | ||||
|             // Lowercase | ||||
|             aVal -= 32; | ||||
|         } | ||||
|         char bVal = pgm_read_byte_near(b + i); | ||||
|         if (bVal >= 97 && bVal <= 122) { | ||||
|             // Lowercase | ||||
|             bVal -= 32; | ||||
|         } | ||||
|         // Relly we shouldn't ever get to '\0' | ||||
|         if (aVal == '"' || bVal == '"' || aVal == '\0' || bVal == '\0') { | ||||
|             // We're done. one is a substring of the other | ||||
|             // or something happenend and the quote didn't stop us. | ||||
|             if (aVal == bVal) { | ||||
|                 // Same value, probably shouldn't happen | ||||
|                 // with this dataset | ||||
|                 return 0; | ||||
|             } | ||||
|             else if (aVal == '"' || aVal == '\0') { | ||||
|                 return -1; | ||||
|             } | ||||
|             else { | ||||
|                 return 1; | ||||
|             } | ||||
|         } | ||||
|         if (aVal == bVal) { | ||||
|             // Same characters. Move to the next. | ||||
|             i++; | ||||
|             continue; | ||||
|         } | ||||
|         // We're done | ||||
|         if (aVal < bVal) { | ||||
|             return -1; | ||||
|         } | ||||
|         else { | ||||
|             return 1; | ||||
|         } | ||||
|     } while (true); | ||||
|     // We shouldn't get here. | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| class ModeSortUsermod : public Usermod { | ||||
| private: | ||||
|  | ||||
|     // Pointers the start of the mode names within JSON_mode_names | ||||
|     char **modes_qstrings = nullptr; | ||||
|  | ||||
|     // Array of mode indexes in alphabetical order. | ||||
|     byte *modes_alpha_indexes = nullptr; | ||||
|  | ||||
|     // Pointers the start of the palette names within JSON_palette_names | ||||
|     char **palettes_qstrings = nullptr; | ||||
|  | ||||
|     // Array of palette indexes in alphabetical order. | ||||
|     byte *palettes_alpha_indexes = nullptr; | ||||
|  | ||||
| public: | ||||
|     /** | ||||
|      * setup() is called once at boot. WiFi is not yet connected at this point. | ||||
|      * You can use it to initialize variables, sensors or similar. | ||||
|      */ | ||||
|     void setup() { | ||||
|         // Sort the modes and palettes on startup | ||||
|         // as they are guarantted to change. | ||||
|         sortModesAndPalettes(); | ||||
|     } | ||||
|  | ||||
|     char **getModesQStrings() { | ||||
|         return modes_qstrings; | ||||
|     } | ||||
|  | ||||
|     byte *getModesAlphaIndexes() { | ||||
|         return modes_alpha_indexes; | ||||
|     } | ||||
|  | ||||
|     char **getPalettesQStrings() { | ||||
|         return palettes_qstrings; | ||||
|     } | ||||
|  | ||||
|     byte *getPalettesAlphaIndexes() { | ||||
|         return palettes_alpha_indexes; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * This Usermod doesn't have anything for loop. | ||||
|      */ | ||||
|     void loop() {} | ||||
|  | ||||
|     /** | ||||
|      * Sort the modes and palettes to the index arrays | ||||
|      * modes_alpha_indexes and palettes_alpha_indexes. | ||||
|      */ | ||||
|     void sortModesAndPalettes() { | ||||
|         modes_qstrings = re_findModeStrings(JSON_mode_names, strip.getModeCount()); | ||||
|         modes_alpha_indexes = re_initIndexArray(strip.getModeCount()); | ||||
|         re_sortModes(modes_qstrings, modes_alpha_indexes, strip.getModeCount(), MODE_SORT_SKIP_COUNT); | ||||
|  | ||||
|         palettes_qstrings = re_findModeStrings(JSON_palette_names, strip.getPaletteCount()); | ||||
|         palettes_alpha_indexes = re_initIndexArray(strip.getPaletteCount()); | ||||
|  | ||||
|         int skipPaletteCount = 1; | ||||
|         while (true) { | ||||
|             // How many palette names start with '*' and should not be sorted? | ||||
|             // (Also skipping the first one, 'Default'). | ||||
|             if (pgm_read_byte_near(palettes_qstrings[skipPaletteCount]) == '*') { | ||||
|                 skipPaletteCount++; | ||||
|             } | ||||
|             else { | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|         re_sortModes(palettes_qstrings, palettes_alpha_indexes, strip.getPaletteCount(), skipPaletteCount); | ||||
|     } | ||||
|  | ||||
|     byte *re_initIndexArray(int numModes) { | ||||
|         byte *indexes = (byte *)malloc(sizeof(byte) * numModes); | ||||
|         for (byte i = 0; i < numModes; i++) { | ||||
|             indexes[i] = i; | ||||
|         } | ||||
|         return indexes; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Return an array of mode or palette names from the JSON string. | ||||
|      * They don't end in '\0', they end in '"'.  | ||||
|      */ | ||||
|     char **re_findModeStrings(const char json[], int numModes) { | ||||
|         char **modeStrings = (char **)malloc(sizeof(char *) * numModes); | ||||
|         uint8_t modeIndex = 0; | ||||
|         bool insideQuotes = false; | ||||
|         // advance past the mark for markLineNum that may exist. | ||||
|         char singleJsonSymbol; | ||||
|  | ||||
|         // Find the mode name in JSON | ||||
|         bool complete = false; | ||||
|         for (size_t i = 0; i < strlen_P(json); i++) { | ||||
|             singleJsonSymbol = pgm_read_byte_near(json + i); | ||||
|             if (singleJsonSymbol == '\0') break; | ||||
|             switch (singleJsonSymbol) { | ||||
|             case '"': | ||||
|                 insideQuotes = !insideQuotes; | ||||
|                 if (insideQuotes) { | ||||
|                     // We have a new mode or palette | ||||
|                     modeStrings[modeIndex] = (char *)(json + i + 1); | ||||
|                 } | ||||
|                 break; | ||||
|             case '[': | ||||
|                 break; | ||||
|             case ']': | ||||
|                 if (!insideQuotes) complete = true; | ||||
|                 break; | ||||
|             case ',': | ||||
|                 if (!insideQuotes) modeIndex++; | ||||
|             default: | ||||
|                 if (!insideQuotes) break; | ||||
|             } | ||||
|             if (complete) break; | ||||
|         } | ||||
|         return modeStrings; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|    * Sort either the modes or the palettes using quicksort. | ||||
|    */ | ||||
|     void re_sortModes(char **modeNames, byte *indexes, int count, int numSkip) { | ||||
|         listBeingSorted = modeNames; | ||||
|         qsort(indexes + numSkip, count - numSkip, sizeof(byte), re_qstringCmp); | ||||
|         listBeingSorted = nullptr; | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * addToJsonState() can be used to add custom entries to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void addToJsonState(JsonObject &root) {} | ||||
|  | ||||
|     /* | ||||
|      * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     void readFromJsonState(JsonObject &root) {} | ||||
|  | ||||
|     /* | ||||
|      * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
|      * This could be used in the future for the system to determine whether your usermod is installed. | ||||
|      */ | ||||
|     uint16_t getId() | ||||
|     { | ||||
|         return USERMOD_ID_MODE_SORT; | ||||
|     } | ||||
| }; | ||||
| @@ -1,48 +0,0 @@ | ||||
| [platformio] | ||||
| default_envs = d1_mini | ||||
| ; default_envs = esp32dev | ||||
|  | ||||
| [env:esp32dev] | ||||
| board = esp32dev | ||||
| platform = espressif32@3.2 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = | ||||
|     ${common.build_flags_esp32}  | ||||
|     -D USERMOD_MODE_SORT | ||||
|     -D USERMOD_FOUR_LINE_DISPLAY -D FLD_PIN_SCL=22 -D FLD_PIN_SDA=21 | ||||
|     -D USERMOD_ROTARY_ENCODER_UI -D ENCODER_DT_PIN=18 -D ENCODER_CLK_PIN=5 -D ENCODER_SW_PIN=19   | ||||
|     -D USERMOD_AUTO_SAVE -D AUTOSAVE_PRESET_NUM=1 | ||||
|     -D LEDPIN=16 -D BTNPIN=13 | ||||
| upload_speed = 460800 | ||||
| lib_ignore = | ||||
|   ESPAsyncTCP | ||||
|   ESPAsyncUDP | ||||
|  | ||||
| [env:d1_mini] | ||||
| board = d1_mini | ||||
| platform = ${common.platform_wled_default} | ||||
| platform_packages = ${common.platform_packages} | ||||
| upload_speed = 460800 | ||||
| board_build.ldscript = ${common.ldscript_4m1m} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = | ||||
|     ${common.build_flags_esp8266}  | ||||
|     -D USERMOD_MODE_SORT | ||||
|     -D USERMOD_FOUR_LINE_DISPLAY -D FLD_PIN_SCL=5 -D FLD_PIN_SDA=4 | ||||
|     -D USERMOD_ROTARY_ENCODER_UI -D ENCODER_DT_PIN=12 -D ENCODER_CLK_PIN=14 -D ENCODER_SW_PIN=13 | ||||
|     -D USERMOD_AUTO_SAVE -D AUTOSAVE_PRESET_NUM=1 | ||||
|     -D LEDPIN=3 -D BTNPIN=0 | ||||
| monitor_filters = esp8266_exception_decoder | ||||
|  | ||||
| [env] | ||||
| lib_deps = | ||||
|     fastled/FastLED @ 3.3.2 | ||||
|     NeoPixelBus @ 2.6.0 | ||||
|     ESPAsyncTCP @ 1.2.0 | ||||
|     ESPAsyncUDP | ||||
|     AsyncTCP @ 1.0.3 | ||||
|     IRremoteESP8266 @ 2.7.3 | ||||
|     https://github.com/lorol/LITTLEFS.git | ||||
|     https://github.com/Aircoookie/ESPAsyncWebServer.git @ ~2.0.0 | ||||
|     U8g2@~2.27.2 | ||||
|     Wire | ||||
| @@ -1,39 +0,0 @@ | ||||
| # Rotary Encoder UI Usermod | ||||
|  | ||||
| First, thanks to the authors of other Rotary Encoder usermods. | ||||
|  | ||||
| This usermod starts to provide a relatively complete on-device | ||||
| UI when paired with the Four Line Display usermod. I strongly | ||||
| encourage you to try them together. | ||||
|  | ||||
| [See the pair of usermods in action](https://www.youtube.com/watch?v=tITQY80rIOA) | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| Copy and update the example `platformio_override.ini.sample` to the root directory of your particular build. | ||||
| This file should be placed in the same directory as `platformio.ini`. | ||||
|  | ||||
| ### Define Your Options | ||||
|  | ||||
| * `USERMOD_ROTARY_ENCODER_UI`       - define this to have this user mod included wled00\usermods_list.cpp | ||||
| * `USERMOD_ROTARY_ENCODER_GPIO`     - define the GPIO function (INPUT, INPUT_PULLUP, etc...) | ||||
| * `USERMOD_FOUR_LINE_DISPLAY`       - define this to have this the Four Line Display mod included wled00\usermods_list.cpp | ||||
|                                         also tells this usermod that the display is available | ||||
|                                         (see the Four Line Display usermod `readme.md` for more details) | ||||
| * `ENCODER_DT_PIN`                    - defaults to 12 | ||||
| * `ENCODER_CLK_PIN`                 - defaults to 14 | ||||
| * `ENCODER_SW_PIN`                    - defaults to 13 | ||||
| * `USERMOD_ROTARY_ENCODER_GPIO`     - GPIO functionality: | ||||
|                                         `INPUT_PULLUP` to use internal pull-up | ||||
|                                         `INPUT` to use pull-up on the PCB | ||||
|  | ||||
| ### PlatformIO requirements | ||||
|  | ||||
| No special requirements. | ||||
|  | ||||
| Note: the Four Line Display usermod requires the libraries `U8g2` and `Wire`. | ||||
|  | ||||
| ## Change Log | ||||
|  | ||||
| 2021-02 | ||||
| * First public release | ||||
| @@ -1,496 +0,0 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| // | ||||
| // Inspired by the v1 usermods | ||||
| // * rotary_encoder_change_brightness | ||||
| // * rotary_encoder_change_effect | ||||
| // | ||||
| // v2 usermod that provides a rotary encoder-based UI. | ||||
| // | ||||
| // This usermod allows you to control: | ||||
| //  | ||||
| // * Brightness | ||||
| // * Selected Effect | ||||
| // * Effect Speed | ||||
| // * Effect Intensity | ||||
| // * Palette | ||||
| //  | ||||
| // Change between modes by pressing a button. | ||||
| // | ||||
| // Dependencies | ||||
| // * This usermod REQUIRES the ModeSortUsermod | ||||
| // * This Usermod works best coupled with  | ||||
| //   FourLineDisplayUsermod. | ||||
| // | ||||
|  | ||||
| #ifndef ENCODER_DT_PIN | ||||
| #define ENCODER_DT_PIN 12 | ||||
| #endif | ||||
|  | ||||
| #ifndef ENCODER_CLK_PIN | ||||
| #define ENCODER_CLK_PIN 14 | ||||
| #endif | ||||
|  | ||||
| #ifndef ENCODER_SW_PIN | ||||
| #define ENCODER_SW_PIN 13 | ||||
| #endif | ||||
|  | ||||
| #ifndef USERMOD_FOUR_LINE_DISPLAY | ||||
| // These constants won't be defined if we aren't using FourLineDisplay. | ||||
| #define FLD_LINE_BRIGHTNESS       0 | ||||
| #define FLD_LINE_MODE             0 | ||||
| #define FLD_LINE_EFFECT_SPEED     0 | ||||
| #define FLD_LINE_EFFECT_INTENSITY 0 | ||||
| #define FLD_LINE_PALETTE          0 | ||||
| #endif | ||||
|  | ||||
|  | ||||
| // The last UI state | ||||
| #define LAST_UI_STATE 4 | ||||
|  | ||||
|  | ||||
| class RotaryEncoderUIUsermod : public Usermod { | ||||
| private: | ||||
|   int fadeAmount = 10;             // Amount to change every step (brightness) | ||||
|   unsigned long currentTime; | ||||
|   unsigned long loopTime; | ||||
|   int8_t pinA = ENCODER_DT_PIN;       // DT from encoder | ||||
|   int8_t pinB = ENCODER_CLK_PIN;      // CLK from encoder | ||||
|   int8_t pinC = ENCODER_SW_PIN;       // SW from encoder | ||||
|   unsigned char select_state = 0;     // 0: brightness, 1: effect, 2: effect speed | ||||
|   unsigned char button_state = HIGH; | ||||
|   unsigned char prev_button_state = HIGH; | ||||
|    | ||||
| #ifdef USERMOD_FOUR_LINE_DISPLAY | ||||
|   FourLineDisplayUsermod *display; | ||||
| #else | ||||
|   void* display = nullptr; | ||||
| #endif | ||||
|  | ||||
|   byte *modes_alpha_indexes = nullptr; | ||||
|   byte *palettes_alpha_indexes = nullptr; | ||||
|  | ||||
|   unsigned char Enc_A; | ||||
|   unsigned char Enc_B; | ||||
|   unsigned char Enc_A_prev = 0; | ||||
|  | ||||
|   bool currentEffectAndPaletteInitialized = false; | ||||
|   uint8_t effectCurrentIndex = 0; | ||||
|   uint8_t effectPaletteIndex = 0; | ||||
|  | ||||
|   bool initDone = false; | ||||
|   bool enabled = true; | ||||
|  | ||||
|   // strings to reduce flash memory usage (used more than twice) | ||||
|   static const char _name[]; | ||||
|   static const char _enabled[]; | ||||
|   static const char _DT_pin[]; | ||||
|   static const char _CLK_pin[]; | ||||
|   static const char _SW_pin[]; | ||||
|  | ||||
| public: | ||||
|   /* | ||||
|      * setup() is called once at boot. WiFi is not yet connected at this point. | ||||
|      * You can use it to initialize variables, sensors or similar. | ||||
|      */ | ||||
|   void setup() | ||||
|   { | ||||
|     DEBUG_PRINTLN(F("Usermod Rotary Encoder init.")); | ||||
|     PinManagerPinType pins[3] = { { pinA, false }, { pinB, false }, { pinC, false } }; | ||||
|     if (!pinManager.allocateMultiplePins(pins, 3, PinOwner::UM_RotaryEncoderUI)) { | ||||
|       // BUG: configuring this usermod with conflicting pins | ||||
|       //      will cause it to de-allocate pins it does not own | ||||
|       //      (at second config) | ||||
|       //      This is the exact type of bug solved by pinManager | ||||
|       //      tracking the owner tags.... | ||||
|       pinA = pinB = pinC = -1; | ||||
|       enabled = false; | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     #ifndef USERMOD_ROTARY_ENCODER_GPIO | ||||
|       #define USERMOD_ROTARY_ENCODER_GPIO INPUT_PULLUP | ||||
|     #endif | ||||
|     pinMode(pinA, USERMOD_ROTARY_ENCODER_GPIO); | ||||
|     pinMode(pinB, USERMOD_ROTARY_ENCODER_GPIO); | ||||
|     pinMode(pinC, USERMOD_ROTARY_ENCODER_GPIO); | ||||
|  | ||||
|     currentTime = millis(); | ||||
|     loopTime = currentTime; | ||||
|  | ||||
|     ModeSortUsermod *modeSortUsermod = (ModeSortUsermod*) usermods.lookup(USERMOD_ID_MODE_SORT); | ||||
|     modes_alpha_indexes = modeSortUsermod->getModesAlphaIndexes(); | ||||
|     palettes_alpha_indexes = modeSortUsermod->getPalettesAlphaIndexes(); | ||||
|  | ||||
| #ifdef USERMOD_FOUR_LINE_DISPLAY     | ||||
|     // This Usermod uses FourLineDisplayUsermod for the best experience. | ||||
|     // But it's optional. But you want it. | ||||
|     display = (FourLineDisplayUsermod*) usermods.lookup(USERMOD_ID_FOUR_LINE_DISP); | ||||
|     if (display != nullptr) { | ||||
|       display->setLineType(FLD_LINE_BRIGHTNESS); | ||||
|       display->setMarkLine(3); | ||||
|     } | ||||
| #endif | ||||
|  | ||||
|     initDone = true; | ||||
|   } | ||||
|  | ||||
|   /* | ||||
|      * connected() is called every time the WiFi is (re)connected | ||||
|      * Use it to initialize network interfaces | ||||
|      */ | ||||
|   void connected() | ||||
|   { | ||||
|     //Serial.println("Connected to WiFi!"); | ||||
|   } | ||||
|  | ||||
|   /* | ||||
|      * loop() is called continuously. Here you can check for events, read sensors, etc. | ||||
|      *  | ||||
|      * Tips: | ||||
|      * 1. You can use "if (WLED_CONNECTED)" to check for a successful network connection. | ||||
|      *    Additionally, "if (WLED_MQTT_CONNECTED)" is available to check for a connection to an MQTT broker. | ||||
|      *  | ||||
|      * 2. Try to avoid using the delay() function. NEVER use delays longer than 10 milliseconds. | ||||
|      *    Instead, use a timer check as shown here. | ||||
|      */ | ||||
|   void loop() | ||||
|   { | ||||
|     if (!enabled) return; | ||||
|  | ||||
|     currentTime = millis(); // get the current elapsed time | ||||
|  | ||||
|     // Initialize effectCurrentIndex and effectPaletteIndex to | ||||
|     // current state. We do it here as (at least) effectCurrent | ||||
|     // is not yet initialized when setup is called. | ||||
|     if (!currentEffectAndPaletteInitialized) { | ||||
|       findCurrentEffectAndPalette(); | ||||
|     } | ||||
|  | ||||
|     if (currentTime >= (loopTime + 2)) // 2ms since last check of encoder = 500Hz | ||||
|     { | ||||
|       button_state = digitalRead(pinC); | ||||
|       if (prev_button_state != button_state) | ||||
|       { | ||||
|         if (button_state == LOW) | ||||
|         { | ||||
|           prev_button_state = button_state; | ||||
|  | ||||
|           char newState = select_state + 1; | ||||
|           if (newState > LAST_UI_STATE) newState = 0; | ||||
|            | ||||
|           bool changedState = true; | ||||
|           if (display != nullptr) { | ||||
|             switch(newState) { | ||||
|               case 0: | ||||
|                 changedState = changeState("Brightness", FLD_LINE_BRIGHTNESS, 3); | ||||
|                 break; | ||||
|               case 1: | ||||
|                 changedState = changeState("Select FX", FLD_LINE_MODE, 2); | ||||
|                 break; | ||||
|               case 2: | ||||
|                 changedState = changeState("FX Speed", FLD_LINE_EFFECT_SPEED, 3); | ||||
|                 break; | ||||
|               case 3: | ||||
|                 changedState = changeState("FX Intensity", FLD_LINE_EFFECT_INTENSITY, 3); | ||||
|                 break; | ||||
|               case 4: | ||||
|                 changedState = changeState("Palette", FLD_LINE_PALETTE, 3); | ||||
|                 break; | ||||
|             } | ||||
|           } | ||||
|           if (changedState) { | ||||
|             select_state = newState; | ||||
|           } | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|           prev_button_state = button_state; | ||||
|         } | ||||
|       } | ||||
|       int Enc_A = digitalRead(pinA); // Read encoder pins | ||||
|       int Enc_B = digitalRead(pinB); | ||||
|       if ((!Enc_A) && (Enc_A_prev)) | ||||
|       { // A has gone from high to low | ||||
|         if (Enc_B == HIGH) | ||||
|         { // B is high so clockwise | ||||
|           switch(select_state) { | ||||
|             case 0: | ||||
|               changeBrightness(true); | ||||
|               break; | ||||
|             case 1: | ||||
|               changeEffect(true); | ||||
|               break; | ||||
|             case 2: | ||||
|               changeEffectSpeed(true); | ||||
|               break; | ||||
|             case 3: | ||||
|               changeEffectIntensity(true); | ||||
|               break; | ||||
|             case 4: | ||||
|               changePalette(true); | ||||
|               break; | ||||
|           } | ||||
|         } | ||||
|         else if (Enc_B == LOW) | ||||
|         { // B is low so counter-clockwise | ||||
|           switch(select_state) { | ||||
|             case 0: | ||||
|               changeBrightness(false); | ||||
|               break; | ||||
|             case 1: | ||||
|               changeEffect(false); | ||||
|               break; | ||||
|             case 2: | ||||
|               changeEffectSpeed(false); | ||||
|               break; | ||||
|             case 3: | ||||
|               changeEffectIntensity(false); | ||||
|               break; | ||||
|             case 4: | ||||
|               changePalette(false); | ||||
|               break; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|       Enc_A_prev = Enc_A;     // Store value of A for next time | ||||
|       loopTime = currentTime; // Updates loopTime | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void findCurrentEffectAndPalette() { | ||||
|     currentEffectAndPaletteInitialized = true; | ||||
|     for (uint8_t i = 0; i < strip.getModeCount(); i++) { | ||||
|       //byte value = modes_alpha_indexes[i]; | ||||
|       if (modes_alpha_indexes[i] == effectCurrent) { | ||||
|         effectCurrentIndex = i; | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     for (uint8_t i = 0; i < strip.getPaletteCount(); i++) { | ||||
|       //byte value = palettes_alpha_indexes[i]; | ||||
|       if (palettes_alpha_indexes[i] == strip.getSegment(0).palette) { | ||||
|         effectPaletteIndex = i; | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   boolean changeState(const char *stateName, byte lineThreeMode, byte markedLine) { | ||||
| #ifdef USERMOD_FOUR_LINE_DISPLAY | ||||
|     if (display != nullptr) { | ||||
|       if (display->wakeDisplay()) { | ||||
|         // Throw away wake up input | ||||
|         return false; | ||||
|       } | ||||
|       display->overlay("Mode change", stateName, 1500); | ||||
|       display->setLineType(lineThreeMode); | ||||
|       display->setMarkLine(markedLine); | ||||
|     } | ||||
|   #endif | ||||
|     return true; | ||||
|   } | ||||
|  | ||||
|   void lampUdated() { | ||||
|     colorUpdated(CALL_MODE_BUTTON); | ||||
|     updateInterfaces(CALL_MODE_BUTTON); | ||||
|   } | ||||
|  | ||||
|   void changeBrightness(bool increase) { | ||||
| #ifdef USERMOD_FOUR_LINE_DISPLAY | ||||
|     if (display && display->wakeDisplay()) { | ||||
|       // Throw away wake up input | ||||
|       return; | ||||
|     } | ||||
| #endif | ||||
|     if (increase) { | ||||
|       bri = (bri + fadeAmount <= 255) ? (bri + fadeAmount) : 255; | ||||
|     } | ||||
|     else { | ||||
|       bri = (bri - fadeAmount >= 0) ? (bri - fadeAmount) : 0; | ||||
|     } | ||||
|     lampUdated(); | ||||
|   } | ||||
|  | ||||
|   void changeEffect(bool increase) { | ||||
| #ifdef USERMOD_FOUR_LINE_DISPLAY | ||||
|     if (display && display->wakeDisplay()) { | ||||
|       // Throw away wake up input | ||||
|       return; | ||||
|     } | ||||
| #endif | ||||
|     if (increase) { | ||||
|       effectCurrentIndex = (effectCurrentIndex + 1 >= strip.getModeCount()) ? 0 : (effectCurrentIndex + 1); | ||||
|     } | ||||
|     else { | ||||
|       effectCurrentIndex = (effectCurrentIndex - 1 < 0) ? (strip.getModeCount() - 1) : (effectCurrentIndex - 1); | ||||
|     } | ||||
|     effectCurrent = modes_alpha_indexes[effectCurrentIndex]; | ||||
|     lampUdated(); | ||||
|   } | ||||
|  | ||||
|   void changeEffectSpeed(bool increase) { | ||||
| #ifdef USERMOD_FOUR_LINE_DISPLAY | ||||
|     if (display && display->wakeDisplay()) { | ||||
|       // Throw away wake up input | ||||
|       return; | ||||
|     } | ||||
| #endif | ||||
|     if (increase) { | ||||
|       effectSpeed = (effectSpeed + fadeAmount <= 255) ? (effectSpeed + fadeAmount) : 255; | ||||
|     } | ||||
|     else { | ||||
|       effectSpeed = (effectSpeed - fadeAmount >= 0) ? (effectSpeed - fadeAmount) : 0; | ||||
|     } | ||||
|     lampUdated(); | ||||
|   } | ||||
|  | ||||
|   void changeEffectIntensity(bool increase) { | ||||
| #ifdef USERMOD_FOUR_LINE_DISPLAY | ||||
|     if (display && display->wakeDisplay()) { | ||||
|       // Throw away wake up input | ||||
|       return; | ||||
|     } | ||||
| #endif | ||||
|     if (increase) { | ||||
|       effectIntensity = (effectIntensity + fadeAmount <= 255) ? (effectIntensity + fadeAmount) : 255; | ||||
|     } | ||||
|     else { | ||||
|       effectIntensity = (effectIntensity - fadeAmount >= 0) ? (effectIntensity - fadeAmount) : 0; | ||||
|     } | ||||
|     lampUdated(); | ||||
|   } | ||||
|  | ||||
|   void changePalette(bool increase) { | ||||
| #ifdef USERMOD_FOUR_LINE_DISPLAY | ||||
|     if (display && display->wakeDisplay()) { | ||||
|       // Throw away wake up input | ||||
|       return; | ||||
|     } | ||||
| #endif | ||||
|     if (increase) { | ||||
|       effectPaletteIndex = (effectPaletteIndex + 1 >= strip.getPaletteCount()) ? 0 : (effectPaletteIndex + 1); | ||||
|     } | ||||
|     else { | ||||
|       effectPaletteIndex = (effectPaletteIndex - 1 < 0) ? (strip.getPaletteCount() - 1) : (effectPaletteIndex - 1); | ||||
|     } | ||||
|     effectPalette = palettes_alpha_indexes[effectPaletteIndex]; | ||||
|     lampUdated(); | ||||
|   } | ||||
|  | ||||
|   /* | ||||
|      * addToJsonInfo() can be used to add custom entries to the /json/info part of the JSON API. | ||||
|      * Creating an "u" object allows you to add custom key/value pairs to the Info section of the WLED web UI. | ||||
|      * Below it is shown how this could be used for e.g. a light sensor | ||||
|      */ | ||||
|   /* | ||||
|     void addToJsonInfo(JsonObject& root) | ||||
|     { | ||||
|       int reading = 20; | ||||
|       //this code adds "u":{"Light":[20," lux"]} to the info object | ||||
|       JsonObject user = root["u"]; | ||||
|       if (user.isNull()) user = root.createNestedObject("u"); | ||||
|       JsonArray lightArr = user.createNestedArray("Light"); //name | ||||
|       lightArr.add(reading); //value | ||||
|       lightArr.add(" lux"); //unit | ||||
|     } | ||||
|     */ | ||||
|  | ||||
|   /* | ||||
|      * addToJsonState() can be used to add custom entries to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|   void addToJsonState(JsonObject &root) | ||||
|   { | ||||
|     //root["user0"] = userVar0; | ||||
|   } | ||||
|  | ||||
|   /* | ||||
|      * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|   void readFromJsonState(JsonObject &root) | ||||
|   { | ||||
|     //userVar0 = root["user0"] | userVar0; //if "user0" key exists in JSON, update, else keep old value | ||||
|     //if (root["bri"] == 255) Serial.println(F("Don't burn down your garage!")); | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * addToConfig() (called from set.cpp) stores persistent properties to cfg.json | ||||
|    */ | ||||
|   void addToConfig(JsonObject &root) { | ||||
|     // we add JSON object: {"Rotary-Encoder":{"DT-pin":12,"CLK-pin":14,"SW-pin":13}} | ||||
|     JsonObject top = root.createNestedObject(FPSTR(_name)); // usermodname | ||||
|     top[FPSTR(_enabled)] = enabled; | ||||
|     top[FPSTR(_DT_pin)]  = pinA; | ||||
|     top[FPSTR(_CLK_pin)] = pinB; | ||||
|     top[FPSTR(_SW_pin)]  = pinC; | ||||
|     DEBUG_PRINTLN(F("Rotary Encoder config saved.")); | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * readFromConfig() is called before setup() to populate properties from values stored in cfg.json | ||||
|    * | ||||
|    * The function should return true if configuration was successfully loaded or false if there was no configuration. | ||||
|    */ | ||||
|   bool readFromConfig(JsonObject &root) { | ||||
|     // we look for JSON object: {"Rotary-Encoder":{"DT-pin":12,"CLK-pin":14,"SW-pin":13}} | ||||
|     JsonObject top = root[FPSTR(_name)]; | ||||
|     if (top.isNull()) { | ||||
|       DEBUG_PRINT(FPSTR(_name)); | ||||
|       DEBUG_PRINTLN(F(": No config found. (Using defaults.)")); | ||||
|       return false; | ||||
|     } | ||||
|     int8_t newDTpin  = top[FPSTR(_DT_pin)]  | pinA; | ||||
|     int8_t newCLKpin = top[FPSTR(_CLK_pin)] | pinB; | ||||
|     int8_t newSWpin  = top[FPSTR(_SW_pin)]  | pinC; | ||||
|  | ||||
|     enabled   = top[FPSTR(_enabled)] | enabled; | ||||
|  | ||||
|     DEBUG_PRINT(FPSTR(_name)); | ||||
|     if (!initDone) { | ||||
|       // first run: reading from cfg.json | ||||
|       pinA = newDTpin; | ||||
|       pinB = newCLKpin; | ||||
|       pinC = newSWpin; | ||||
|       DEBUG_PRINTLN(F(" config loaded.")); | ||||
|     } else { | ||||
|       DEBUG_PRINTLN(F(" config (re)loaded.")); | ||||
|       // changing parameters from settings page | ||||
|       if (pinA!=newDTpin || pinB!=newCLKpin || pinC!=newSWpin) { | ||||
|         pinManager.deallocatePin(pinA, PinOwner::UM_RotaryEncoderUI); | ||||
|         pinManager.deallocatePin(pinB, PinOwner::UM_RotaryEncoderUI); | ||||
|         pinManager.deallocatePin(pinC, PinOwner::UM_RotaryEncoderUI); | ||||
|         pinA = newDTpin; | ||||
|         pinB = newCLKpin; | ||||
|         pinC = newSWpin; | ||||
|         if (pinA<0 || pinB<0 || pinC<0) { | ||||
|           enabled = false; | ||||
|           return true; | ||||
|         } | ||||
|         setup(); | ||||
|       } | ||||
|     } | ||||
|     // use "return !top["newestParameter"].isNull();" when updating Usermod with new features | ||||
|     return !top[FPSTR(_enabled)].isNull(); | ||||
|   } | ||||
|  | ||||
|   /* | ||||
|      * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
|      * This could be used in the future for the system to determine whether your usermod is installed. | ||||
|      */ | ||||
|   uint16_t getId() | ||||
|   { | ||||
|     return USERMOD_ID_ROTARY_ENC_UI; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| // strings to reduce flash memory usage (used more than twice) | ||||
| const char RotaryEncoderUIUsermod::_name[]     PROGMEM = "Rotary-Encoder"; | ||||
| const char RotaryEncoderUIUsermod::_enabled[]  PROGMEM = "enabled"; | ||||
| const char RotaryEncoderUIUsermod::_DT_pin[]   PROGMEM = "DT-pin"; | ||||
| const char RotaryEncoderUIUsermod::_CLK_pin[]  PROGMEM = "CLK-pin"; | ||||
| const char RotaryEncoderUIUsermod::_SW_pin[]   PROGMEM = "SW-pin"; | ||||
| @@ -285,7 +285,7 @@ class RotaryEncoderUIUsermod : public Usermod { | ||||
|      * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
|      * This could be used in the future for the system to determine whether your usermod is installed. | ||||
|      */ | ||||
|     uint16_t getId() { return USERMOD_ID_ROTARY_ENC_UI; } | ||||
|     uint16_t getId() override { return USERMOD_ID_ROTARY_ENC_UI; } | ||||
|     /** | ||||
|      * Enable/Disable the usermod | ||||
|      */ | ||||
| @@ -300,7 +300,7 @@ class RotaryEncoderUIUsermod : public Usermod { | ||||
|      * setup() is called once at boot. WiFi is not yet connected at this point. | ||||
|      * You can use it to initialize variables, sensors or similar. | ||||
|      */ | ||||
|     void setup(); | ||||
|     void setup() override; | ||||
|  | ||||
|     /** | ||||
|      * connected() is called every time the WiFi is (re)connected | ||||
| @@ -311,11 +311,11 @@ class RotaryEncoderUIUsermod : public Usermod { | ||||
|     /** | ||||
|      * loop() is called continuously. Here you can check for events, read sensors, etc. | ||||
|      */ | ||||
|     void loop(); | ||||
|     void loop() override; | ||||
|  | ||||
| #ifndef WLED_DISABLE_MQTT | ||||
|     //bool onMqttMessage(char* topic, char* payload); | ||||
|     //void onMqttConnect(bool sessionPresent); | ||||
|     //bool onMqttMessage(char* topic, char* payload) override; | ||||
|     //void onMqttConnect(bool sessionPresent) override; | ||||
| #endif | ||||
|  | ||||
|     /** | ||||
| @@ -323,31 +323,31 @@ class RotaryEncoderUIUsermod : public Usermod { | ||||
|      * will prevent button working in a default way. | ||||
|      * Replicating button.cpp | ||||
|      */ | ||||
|     //bool handleButton(uint8_t b); | ||||
|     //bool handleButton(uint8_t b) override; | ||||
|  | ||||
|     /** | ||||
|      * addToJsonInfo() can be used to add custom entries to the /json/info part of the JSON API. | ||||
|      */ | ||||
|     //void addToJsonInfo(JsonObject &root); | ||||
|     //void addToJsonInfo(JsonObject &root) override; | ||||
|  | ||||
|     /** | ||||
|      * addToJsonState() can be used to add custom entries to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     //void addToJsonState(JsonObject &root); | ||||
|     //void addToJsonState(JsonObject &root) override; | ||||
|  | ||||
|     /** | ||||
|      * readFromJsonState() can be used to receive data clients send to the /json/state part of the JSON API (state object). | ||||
|      * Values in the state object may be modified by connected clients | ||||
|      */ | ||||
|     //void readFromJsonState(JsonObject &root); | ||||
|     //void readFromJsonState(JsonObject &root) override; | ||||
|  | ||||
|     /** | ||||
|      * provide the changeable values | ||||
|      */ | ||||
|     void addToConfig(JsonObject &root); | ||||
|     void addToConfig(JsonObject &root) override; | ||||
|  | ||||
|     void appendConfigData(); | ||||
|     void appendConfigData() override; | ||||
|  | ||||
|     /** | ||||
|      * restore the changeable values | ||||
| @@ -355,7 +355,7 @@ class RotaryEncoderUIUsermod : public Usermod { | ||||
|      *  | ||||
|      * The function should return true if configuration was successfully loaded or false if there was no configuration. | ||||
|      */ | ||||
|     bool readFromConfig(JsonObject &root); | ||||
|     bool readFromConfig(JsonObject &root) override; | ||||
|  | ||||
|     // custom methods | ||||
|     void displayNetworkInfo(); | ||||
|   | ||||
| @@ -66,9 +66,9 @@ class WireguardUsermod : public Usermod { | ||||
|     void addToConfig(JsonObject& root) { | ||||
|         JsonObject top = root.createNestedObject(F("WireGuard")); | ||||
|         top[F("host")] = endpoint_address; | ||||
|         top[F("port")] = endpoint_port; | ||||
|         top[F("ip")] = local_ip.toString(); | ||||
|         top[F("psk")] = preshared_key; | ||||
|         top["port"] = endpoint_port; | ||||
|         top["ip"] = local_ip.toString(); | ||||
|         top["psk"] = preshared_key; | ||||
|         top[F("pem")] = private_key; | ||||
|         top[F("pub")] = public_key; | ||||
|         top[F("tz")] = posix_tz; | ||||
| @@ -77,11 +77,11 @@ class WireguardUsermod : public Usermod { | ||||
|     bool readFromConfig(JsonObject& root) { | ||||
|         JsonObject top = root[F("WireGuard")]; | ||||
|  | ||||
|         if (top["host"].isNull() || top["port"].isNull() || top["ip"].isNull() || top["pem"].isNull() || top["pub"].isNull() || top["tz"].isNull()) { | ||||
|         if (top[F("host")].isNull() || top["port"].isNull() || top["ip"].isNull() || top[F("pem")].isNull() || top[F("pub")].isNull() || top[F("tz")].isNull()) { | ||||
|             is_enabled = false; | ||||
|             return false; | ||||
|         } else { | ||||
|             const char* host = top["host"]; | ||||
|             const char* host = top[F("host")]; | ||||
|             strncpy(endpoint_address, host, 100); | ||||
|  | ||||
|             const char* ip_s = top["ip"]; | ||||
| @@ -89,16 +89,16 @@ class WireguardUsermod : public Usermod { | ||||
|             sscanf(ip_s, "%u.%u.%u.%u", &ip[0], &ip[1], &ip[2], &ip[3]); | ||||
|             local_ip = IPAddress(ip[0], ip[1], ip[2], ip[3]); | ||||
|  | ||||
|             const char* pem = top["pem"]; | ||||
|             const char* pem = top[F("pem")]; | ||||
|             strncpy(private_key, pem, 45); | ||||
|  | ||||
|             const char* pub = top["pub"]; | ||||
|             const char* pub = top[F("pub")]; | ||||
|             strncpy(public_key, pub, 45); | ||||
|  | ||||
|             const char* tz = top["tz"]; | ||||
|             const char* tz = top[F("tz")]; | ||||
|             strncpy(posix_tz, tz, 150); | ||||
|  | ||||
|             endpoint_port = top["port"]; | ||||
|             endpoint_port = top[F("port")]; | ||||
|  | ||||
|             if (!top["psk"].isNull()) { | ||||
|                 const char* psk = top["psk"]; | ||||
|   | ||||
| @@ -325,8 +325,8 @@ public: | ||||
|     void addToConfig(JsonObject& root) | ||||
|     { | ||||
|       JsonObject modName = root.createNestedObject("id"); | ||||
|       modName["mdns"] = "wled-word-clock"; | ||||
|       modName["name"] = "WLED WORD CLOCK"; | ||||
|       modName[F("mdns")] = "wled-word-clock"; | ||||
|       modName[F("name")] = "WLED WORD CLOCK"; | ||||
|     } | ||||
|  | ||||
|     uint16_t getId() | ||||
|   | ||||
							
								
								
									
										705
									
								
								wled00/FX.cpp
									
									
									
									
									
								
							
							
						
						
									
										705
									
								
								wled00/FX.cpp
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										288
									
								
								wled00/FX.h
									
									
									
									
									
								
							
							
						
						
									
										288
									
								
								wled00/FX.h
									
									
									
									
									
								
							| @@ -62,10 +62,10 @@ | ||||
| //#define FRAMETIME        _frametime | ||||
| #define FRAMETIME        strip.getFrameTime() | ||||
|  | ||||
| /* each segment uses 52 bytes of SRAM memory, so if you're application fails because of | ||||
| /* 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 | ||||
|   #define MAX_NUM_SEGMENTS    12 | ||||
|   /* How much data bytes all segments combined may allocate */ | ||||
|   #define MAX_SEGMENT_DATA  5120 | ||||
| #else | ||||
| @@ -73,9 +73,13 @@ | ||||
|     #define MAX_NUM_SEGMENTS  32 | ||||
|   #endif | ||||
|   #if defined(ARDUINO_ARCH_ESP32S2) | ||||
|     #define MAX_SEGMENT_DATA  24576 | ||||
|     #if defined(BOARD_HAS_PSRAM) && defined(WLED_USE_PSRAM) | ||||
|       #define MAX_SEGMENT_DATA  MAX_NUM_SEGMENTS*1024 // 32k by default | ||||
|     #else | ||||
|       #define MAX_SEGMENT_DATA  MAX_NUM_SEGMENTS*768  // 24k by default | ||||
|     #endif | ||||
|   #else | ||||
|     #define MAX_SEGMENT_DATA  32767 | ||||
|     #define MAX_SEGMENT_DATA  MAX_NUM_SEGMENTS*1280 // 40k by default | ||||
|   #endif | ||||
| #endif | ||||
|  | ||||
| @@ -237,7 +241,7 @@ | ||||
| #define FX_MODE_CHUNCHUN               111 | ||||
| #define FX_MODE_DANCING_SHADOWS        112 | ||||
| #define FX_MODE_WASHING_MACHINE        113 | ||||
| // #define FX_MODE_CANDY_CANE             114  // removed in 0.14! | ||||
| #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) | ||||
| @@ -278,7 +282,7 @@ | ||||
| #define FX_MODE_RIPPLEPEAK             148 | ||||
| #define FX_MODE_2DFIRENOISE            149 | ||||
| #define FX_MODE_2DSQUAREDSWIRL         150 | ||||
| #define FX_MODE_2DFIRE2012             151 | ||||
| // #define FX_MODE_2DFIRE2012             151 | ||||
| #define FX_MODE_2DDNA                  152 | ||||
| #define FX_MODE_2DMATRIX               153 | ||||
| #define FX_MODE_2DMETABALLS            154 | ||||
| @@ -288,7 +292,7 @@ | ||||
| #define FX_MODE_GRAVFREQ               158 | ||||
| #define FX_MODE_DJLIGHT                159 | ||||
| #define FX_MODE_2DFUNKYPLANK           160 | ||||
| #define FX_MODE_2DCENTERBARS           161 | ||||
| //#define FX_MODE_2DCENTERBARS           161 | ||||
| #define FX_MODE_2DPULSER               162 | ||||
| #define FX_MODE_BLURZ                  163 | ||||
| #define FX_MODE_2DDRIFT                164 | ||||
| @@ -416,7 +420,8 @@ typedef struct Segment { | ||||
|     // perhaps this should be per segment, not static | ||||
|     static CRGBPalette16 _randomPalette;      // actual random palette | ||||
|     static CRGBPalette16 _newRandomPalette;   // target random palette | ||||
|     static unsigned long _lastPaletteChange;  // last random palette change time in millis() | ||||
|     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 | ||||
|     #ifndef WLED_DISABLE_MODE_BLEND | ||||
|     static bool          _modeBlend;          // mode/effect blending semaphore | ||||
|     #endif | ||||
| @@ -493,9 +498,9 @@ typedef struct Segment { | ||||
|  | ||||
|     ~Segment() { | ||||
|       #ifdef WLED_DEBUG | ||||
|       //Serial.printf("-- Destroying segment: %p\n", this); | ||||
|       //Serial.printf("-- Destroying segment: %p", this); | ||||
|       //if (name) Serial.printf(" %s (%p)", name, name); | ||||
|       //if (data) Serial.printf(" %d (%p)", (int)_dataLen, data); | ||||
|       //if (data) Serial.printf(" %d->(%p)", (int)_dataLen, data); | ||||
|       //Serial.println(); | ||||
|       #endif | ||||
|       if (name) { delete[] name; name = nullptr; } | ||||
| @@ -531,7 +536,7 @@ typedef struct Segment { | ||||
|     #endif | ||||
|     static void     handleRandomPalette(); | ||||
|  | ||||
|     void    setUp(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 segId = 255); | ||||
|     void    setUp(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); | ||||
|     bool    setColor(uint8_t slot, uint32_t c); //returns true if changed | ||||
|     void    setCCT(uint16_t k); | ||||
|     void    setOpacity(uint8_t o); | ||||
| @@ -543,9 +548,9 @@ typedef struct Segment { | ||||
|  | ||||
|     // runtime data functions | ||||
|     inline uint16_t dataSize(void) const { return _dataLen; } | ||||
|     bool allocateData(size_t len); | ||||
|     void deallocateData(void); | ||||
|     void resetIfRequired(void); | ||||
|     bool allocateData(size_t len);  // allocates effect data buffer in heap and clears it | ||||
|     void deallocateData(void);      // deallocates (frees) effect data buffer from heap | ||||
|     void resetIfRequired(void);     // 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. | ||||
| @@ -555,65 +560,65 @@ typedef struct Segment { | ||||
|     inline void markForReset(void) { reset = true; }  // setOption(SEG_OPTION_RESET, true) | ||||
|  | ||||
|     // transition functions | ||||
|     void     startTransition(uint16_t dur); // transition has to start before actual segment values change | ||||
|     void     stopTransition(void); | ||||
|     void     startTransition(uint16_t dur);     // transition has to start before actual segment values change | ||||
|     void     stopTransition(void);              // ends transition mode by destroying transition structure | ||||
|     void     handleTransition(void); | ||||
|     #ifndef WLED_DISABLE_MODE_BLEND | ||||
|     void     swapSegenv(tmpsegd_t &tmpSegD); | ||||
|     void     restoreSegenv(tmpsegd_t &tmpSegD); | ||||
|     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(tmpsegd_t &tmpSegD); // restores segment data from buffer, if buffer is not transition buffer, changed values are copied to transition buffer | ||||
|     #endif | ||||
|     uint16_t progress(void); //transition progression between 0-65535 | ||||
|     uint8_t  currentBri(bool useCct = false); | ||||
|     uint8_t  currentMode(void); | ||||
|     uint32_t currentColor(uint8_t slot); | ||||
|     uint16_t progress(void);                    // transition progression between 0-65535 | ||||
|     uint8_t  currentBri(bool useCct = false);   // current segment brightness/CCT (blended while in transition) | ||||
|     uint8_t  currentMode(void);                 // currently active effect/mode (while in transition) | ||||
|     uint32_t currentColor(uint8_t slot);        // currently active segment color (blended while in transition) | ||||
|     CRGBPalette16 &loadPalette(CRGBPalette16 &tgt, uint8_t pal); | ||||
|     CRGBPalette16 ¤tPalette(CRGBPalette16 &tgt, uint8_t paletteID); | ||||
|  | ||||
|     // 1D strip | ||||
|     uint16_t virtualLength(void) const; | ||||
|     void setPixelColor(int n, uint32_t c); // set relative pixel within segment with color | ||||
|     void setPixelColor(unsigned n, uint32_t c)                    { setPixelColor(int(n), c); } | ||||
|     void setPixelColor(int n, byte r, byte g, byte b, byte w = 0) { setPixelColor(n, RGBW32(r,g,b,w)); } // automatically inline | ||||
|     void setPixelColor(int n, CRGB c)                             { setPixelColor(n, RGBW32(c.r,c.g,c.b,0)); } // automatically inline | ||||
|     inline void setPixelColor(unsigned n, uint32_t c)                    { setPixelColor(int(n), c); } | ||||
|     inline void setPixelColor(int n, byte r, byte g, byte b, byte w = 0) { setPixelColor(n, RGBW32(r,g,b,w)); } | ||||
|     inline void setPixelColor(int n, CRGB c)                             { setPixelColor(n, RGBW32(c.r,c.g,c.b,0)); } | ||||
|     void setPixelColor(float i, uint32_t c, bool aa = true); | ||||
|     void setPixelColor(float i, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0, bool aa = true) { setPixelColor(i, RGBW32(r,g,b,w), aa); } | ||||
|     void setPixelColor(float i, CRGB c, bool aa = true)                                         { setPixelColor(i, RGBW32(c.r,c.g,c.b,0), aa); } | ||||
|     inline void setPixelColor(float i, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0, bool aa = true) { setPixelColor(i, RGBW32(r,g,b,w), aa); } | ||||
|     inline void setPixelColor(float i, CRGB c, bool aa = true)                                         { setPixelColor(i, RGBW32(c.r,c.g,c.b,0), aa); } | ||||
|     uint32_t getPixelColor(int i); | ||||
|     // 1D support functions (some implement 2D as well) | ||||
|     void blur(uint8_t); | ||||
|     void fill(uint32_t c); | ||||
|     void fade_out(uint8_t r); | ||||
|     void fadeToBlackBy(uint8_t fadeBy); | ||||
|     void blendPixelColor(int n, uint32_t color, uint8_t blend); | ||||
|     void blendPixelColor(int n, CRGB c, uint8_t blend)            { blendPixelColor(n, RGBW32(c.r,c.g,c.b,0), blend); } | ||||
|     void addPixelColor(int n, uint32_t color, bool fast = false); | ||||
|     void addPixelColor(int n, byte r, byte g, byte b, byte w = 0, bool fast = false) { addPixelColor(n, RGBW32(r,g,b,w), fast); } // automatically inline | ||||
|     void addPixelColor(int n, CRGB c, bool fast = false)          { addPixelColor(n, RGBW32(c.r,c.g,c.b,0), fast); } // automatically inline | ||||
|     void fadePixelColor(uint16_t n, uint8_t fade); | ||||
|     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 fast = false)  { setPixelColor(n, color_add(getPixelColor(n), color, fast)); } | ||||
|     inline void addPixelColor(int n, byte r, byte g, byte b, byte w = 0, bool fast = false) { addPixelColor(n, RGBW32(r,g,b,w), fast); } | ||||
|     inline void addPixelColor(int n, CRGB c, bool fast = false)          { addPixelColor(n, RGBW32(c.r,c.g,c.b,0), fast); } | ||||
|     inline void fadePixelColor(uint16_t n, uint8_t fade)                 { setPixelColor(n, color_fade(getPixelColor(n), fade, true)); } | ||||
|     uint32_t color_from_palette(uint16_t, bool mapping, bool wrap, uint8_t mcol, uint8_t pbri = 255); | ||||
|     uint32_t color_wheel(uint8_t pos); | ||||
|  | ||||
|     // 2D matrix | ||||
|     uint16_t virtualWidth(void)  const; | ||||
|     uint16_t virtualHeight(void) const; | ||||
|     uint16_t nrOfVStrips(void) const; | ||||
|     uint16_t virtualWidth(void)  const; // segment width in virtual pixels (accounts for groupping and spacing) | ||||
|     uint16_t virtualHeight(void) const; // segment height in virtual pixels (accounts for groupping and spacing) | ||||
|     uint16_t nrOfVStrips(void) const;   // returns number of virtual vertical strips in 2D matrix (used to expand 1D effects into 2D) | ||||
|   #ifndef WLED_DISABLE_2D | ||||
|     uint16_t XY(uint16_t x, uint16_t y); // support function to get relative index within segment | ||||
|     void setPixelColorXY(int x, int y, uint32_t c); // set relative pixel within segment with color | ||||
|     void setPixelColorXY(unsigned x, unsigned y, uint32_t c)               { setPixelColorXY(int(x), int(y), c); } | ||||
|     void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) { setPixelColorXY(x, y, RGBW32(r,g,b,w)); } // automatically inline | ||||
|     void setPixelColorXY(int x, int y, CRGB c)                             { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0)); } // automatically inline | ||||
|     inline void setPixelColorXY(unsigned x, unsigned y, uint32_t c)               { setPixelColorXY(int(x), int(y), c); } | ||||
|     inline void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) { setPixelColorXY(x, y, RGBW32(r,g,b,w)); } | ||||
|     inline void setPixelColorXY(int x, int y, CRGB c)                             { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0)); } | ||||
|     void setPixelColorXY(float x, float y, uint32_t c, bool aa = true); | ||||
|     void setPixelColorXY(float x, float y, byte r, byte g, byte b, byte w = 0, bool aa = true) { setPixelColorXY(x, y, RGBW32(r,g,b,w), aa); } | ||||
|     void setPixelColorXY(float x, float y, CRGB c, bool aa = true)                             { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), aa); } | ||||
|     inline void setPixelColorXY(float x, float y, byte r, byte g, byte b, byte w = 0, bool aa = true) { setPixelColorXY(x, y, RGBW32(r,g,b,w), aa); } | ||||
|     inline void setPixelColorXY(float x, float y, CRGB c, bool aa = true)                             { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), aa); } | ||||
|     uint32_t getPixelColorXY(uint16_t x, uint16_t y); | ||||
|     // 2D support functions | ||||
|     void blendPixelColorXY(uint16_t x, uint16_t y, uint32_t color, uint8_t blend); | ||||
|     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); } | ||||
|     void addPixelColorXY(int x, int y, uint32_t color, bool fast = false); | ||||
|     void addPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0, bool fast = false) { addPixelColorXY(x, y, RGBW32(r,g,b,w), fast); } // automatically inline | ||||
|     void addPixelColorXY(int x, int y, CRGB c, bool fast = false)                             { addPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), fast); } | ||||
|     void fadePixelColorXY(uint16_t x, uint16_t y, uint8_t fade); | ||||
|     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 fast = false)         { setPixelColorXY(x, y, color_add(getPixelColorXY(x,y), color, fast)); } | ||||
|     inline void addPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0, bool fast = false) { addPixelColorXY(x, y, RGBW32(r,g,b,w), fast); } | ||||
|     inline void addPixelColorXY(int x, int y, CRGB c, bool fast = false)                             { addPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0), fast); } | ||||
|     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(uint16_t i, bool vertical, fract8 blur_amount); // 1D box blur (with weight) | ||||
|     void blurRow(uint16_t row, fract8 blur_amount); | ||||
|     void blurCol(uint16_t col, fract8 blur_amount); | ||||
| @@ -623,43 +628,43 @@ typedef struct Segment { | ||||
|     void draw_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c); | ||||
|     void fill_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c); | ||||
|     void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, uint32_t c); | ||||
|     void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, CRGB c) { drawLine(x0, y0, x1, y1, RGBW32(c.r,c.g,c.b,0)); } // automatic inline | ||||
|     inline void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, CRGB c) { drawLine(x0, y0, x1, y1, RGBW32(c.r,c.g,c.b,0)); } // 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); | ||||
|     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 | ||||
|     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) { drawCharacter(chr, x, y, w, h, RGBW32(c.r,c.g,c.b,0), RGBW32(c2.r,c2.g,c2.b,0), rotate); } // automatic inline | ||||
|     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) { drawCharacter(chr, x, y, w, h, RGBW32(c.r,c.g,c.b,0), RGBW32(c2.r,c2.g,c2.b,0), rotate); } // automatic inline | ||||
|     void wu_pixel(uint32_t x, uint32_t y, CRGB c); | ||||
|     void blur1d(fract8 blur_amount); // blur all rows in 1 dimension | ||||
|     void blur2d(fract8 blur_amount) { blur(blur_amount); } | ||||
|     void fill_solid(CRGB c) { fill(RGBW32(c.r,c.g,c.b,0)); } | ||||
|     inline void blur2d(fract8 blur_amount) { blur(blur_amount); } | ||||
|     inline void fill_solid(CRGB c) { fill(RGBW32(c.r,c.g,c.b,0)); } | ||||
|     void nscale8(uint8_t scale); | ||||
|   #else | ||||
|     uint16_t XY(uint16_t x, uint16_t y)                                    { return x; } | ||||
|     void setPixelColorXY(int x, int y, uint32_t c)                         { setPixelColor(x, c); } | ||||
|     void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) { setPixelColor(x, RGBW32(r,g,b,w)); } | ||||
|     void setPixelColorXY(int x, int y, CRGB c)                             { setPixelColor(x, RGBW32(c.r,c.g,c.b,0)); } | ||||
|     void setPixelColorXY(float x, float y, uint32_t c, bool aa = true)     { setPixelColor(x, c, aa); } | ||||
|     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); } | ||||
|     void setPixelColorXY(float x, float y, CRGB c, bool aa = true)         { setPixelColor(x, RGBW32(c.r,c.g,c.b,0), aa); } | ||||
|     uint32_t getPixelColorXY(uint16_t x, uint16_t y)                       { return getPixelColor(x); } | ||||
|     void blendPixelColorXY(uint16_t x, uint16_t y, uint32_t c, uint8_t blend) { blendPixelColor(x, c, blend); } | ||||
|     void blendPixelColorXY(uint16_t x, uint16_t y, CRGB c, uint8_t blend)  { blendPixelColor(x, RGBW32(c.r,c.g,c.b,0), blend); } | ||||
|     void addPixelColorXY(int x, int y, uint32_t color, bool fast = false)  { addPixelColor(x, color, fast); } | ||||
|     void addPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0, bool fast = false) { addPixelColor(x, RGBW32(r,g,b,w), fast); } | ||||
|     void addPixelColorXY(int x, int y, CRGB c, bool fast = false)          { addPixelColor(x, RGBW32(c.r,c.g,c.b,0), fast); } | ||||
|     void fadePixelColorXY(uint16_t x, uint16_t y, uint8_t fade)            { fadePixelColor(x, fade); } | ||||
|     void box_blur(uint16_t i, bool vertical, fract8 blur_amount) {} | ||||
|     void blurRow(uint16_t row, fract8 blur_amount) {} | ||||
|     void blurCol(uint16_t col, fract8 blur_amount) {} | ||||
|     void moveX(int8_t delta, bool wrap = false) {} | ||||
|     void moveY(int8_t delta, bool wrap = false) {} | ||||
|     void move(uint8_t dir, uint8_t delta, bool wrap = false) {} | ||||
|     void fill_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c) {} | ||||
|     void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, uint32_t c) {} | ||||
|     void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, CRGB c) {} | ||||
|     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) {} | ||||
|     void drawCharacter(unsigned char chr, int16_t x, int16_t y, uint8_t w, uint8_t h, CRGB color) {} | ||||
|     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) {} | ||||
|     void wu_pixel(uint32_t x, uint32_t y, CRGB c) {} | ||||
|     inline uint16_t XY(uint16_t x, uint16_t y)                                    { return x; } | ||||
|     inline void setPixelColorXY(int x, int y, uint32_t c)                         { setPixelColor(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(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); } | ||||
|     inline uint32_t getPixelColorXY(uint16_t x, uint16_t 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 fast = false)  { addPixelColor(x, color, fast); } | ||||
|     inline void addPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0, bool fast = false) { addPixelColor(x, RGBW32(r,g,b,w), fast); } | ||||
|     inline void addPixelColorXY(int x, int y, CRGB c, bool fast = false)          { addPixelColor(x, RGBW32(c.r,c.g,c.b,0), fast); } | ||||
|     inline void fadePixelColorXY(uint16_t x, uint16_t y, uint8_t fade)            { fadePixelColor(x, fade); } | ||||
|     inline void box_blur(uint16_t i, bool vertical, fract8 blur_amount) {} | ||||
|     inline void blurRow(uint16_t row, fract8 blur_amount) {} | ||||
|     inline void blurCol(uint16_t col, fract8 blur_amount) {} | ||||
|     inline void moveX(int8_t delta, bool wrap = false) {} | ||||
|     inline void moveY(int8_t delta, bool wrap = false) {} | ||||
|     inline void move(uint8_t dir, uint8_t delta, bool wrap = false) {} | ||||
|     inline void fill_circle(uint16_t cx, uint16_t cy, uint8_t radius, CRGB c) {} | ||||
|     inline void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, uint32_t c) {} | ||||
|     inline void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1, CRGB c) {} | ||||
|     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) {} | ||||
|     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) {} | ||||
|     inline void wu_pixel(uint32_t x, uint32_t y, CRGB c) {} | ||||
|   #endif | ||||
| } segment; | ||||
| //static int segSize = sizeof(Segment); | ||||
| @@ -682,10 +687,7 @@ class WS2812FX {  // 96 bytes | ||||
|     WS2812FX() : | ||||
|       paletteFade(0), | ||||
|       paletteBlend(0), | ||||
|       milliampsPerLed(55), | ||||
|       cctBlending(0), | ||||
|       ablMilliampsMax(ABL_MILLIAMPS_DEFAULT), | ||||
|       currentMilliamps(0), | ||||
|       now(millis()), | ||||
|       timebase(0), | ||||
|       isMatrix(false), | ||||
| @@ -697,6 +699,7 @@ class WS2812FX {  // 96 bytes | ||||
|       _colors_t{0,0,0}, | ||||
|       _virtualSegmentLength(0), | ||||
|       // true private variables | ||||
|       _suspend(false), | ||||
|       _length(DEFAULT_LED_COUNT), | ||||
|       _brightness(DEFAULT_BRIGHTNESS), | ||||
|       _transitionDur(750), | ||||
| @@ -745,41 +748,42 @@ class WS2812FX {  // 96 bytes | ||||
|  | ||||
|     void | ||||
| #ifdef WLED_DEBUG | ||||
|       printSize(), | ||||
|       printSize(),                                // prints memory usage for strip components | ||||
| #endif | ||||
|       finalizeInit(), | ||||
|       service(void), | ||||
|       setMode(uint8_t segid, uint8_t m), | ||||
|       setColor(uint8_t slot, uint32_t c), | ||||
|       setCCT(uint16_t k), | ||||
|       setBrightness(uint8_t b, bool direct = false), | ||||
|       setRange(uint16_t i, uint16_t i2, uint32_t col), | ||||
|       setTransitionMode(bool t), | ||||
|       purgeSegments(bool force = false), | ||||
|       finalizeInit(),                             // initialises strip components | ||||
|       service(void),                              // executes effect functions when due and calls strip.show() | ||||
|       setMode(uint8_t segid, uint8_t m),          // sets effect/mode for given segment (high level API) | ||||
|       setColor(uint8_t slot, uint32_t c),         // sets color (in slot) for given segment (high level API) | ||||
|       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(void),                        // removes inactive segments from RAM (may incure penalty and memory fragmentation but reduces vector footprint) | ||||
|       setSegment(uint8_t n, uint16_t start, uint16_t stop, uint8_t grouping = 1, uint8_t spacing = 0, uint16_t offset = UINT16_MAX, uint16_t startY=0, uint16_t stopY=1), | ||||
|       setMainSegmentId(uint8_t n), | ||||
|       restartRuntime(), | ||||
|       resetSegments(), | ||||
|       makeAutoSegments(bool forceReset = false), | ||||
|       fixInvalidSegments(), | ||||
|       setPixelColor(int n, uint32_t c), | ||||
|       show(void), | ||||
|       setTargetFps(uint8_t fps); | ||||
|       resetSegments(),                            // marks all segments for reset | ||||
|       makeAutoSegments(bool forceReset = false),  // will create segments based on configured outputs | ||||
|       fixInvalidSegments(),                       // fixes incorrect segment configuration | ||||
|       setPixelColor(unsigned n, uint32_t c),      // paints absolute strip pixel with index n and color c | ||||
|       show(void),                                 // initiates LED output | ||||
|       setTargetFps(uint8_t fps), | ||||
|       addEffect(uint8_t id, mode_ptr mode_fn, const char *mode_name), // add effect to the list; defined in FX.cpp | ||||
|       setupEffectData(void);                      // add default effects to the list; defined in FX.cpp | ||||
|  | ||||
|     void setColor(uint8_t slot, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0) { setColor(slot, RGBW32(r,g,b,w)); } | ||||
|     void fill(uint32_t c) { for (int i = 0; i < getLengthTotal(); i++) setPixelColor(i, c); } // fill whole strip with color (inline) | ||||
|     void addEffect(uint8_t id, mode_ptr mode_fn, const char *mode_name); // add effect to the list; defined in FX.cpp | ||||
|     void setupEffectData(void); // add default effects to the list; defined in FX.cpp | ||||
|  | ||||
|     // outsmart the compiler :) by correctly overloading | ||||
|     inline void setPixelColor(int n, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0) { setPixelColor(n, RGBW32(r,g,b,w)); } | ||||
|     inline void setPixelColor(int n, CRGB c) { setPixelColor(n, c.red, c.green, c.blue); } | ||||
|     inline void trigger(void) { _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; } | ||||
|     inline void restartRuntime()          { for (Segment &seg : _segments) seg.markForReset(); } | ||||
|     inline void setTransitionMode(bool t) { for (Segment &seg : _segments) seg.startTransition(t ? _transitionDur : 0); } | ||||
|     inline void setColor(uint8_t slot, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0)    { setColor(slot, RGBW32(r,g,b,w)); } | ||||
|     inline void setPixelColor(unsigned n, uint8_t r, uint8_t g, uint8_t b, uint8_t w = 0) { setPixelColor(n, RGBW32(r,g,b,w)); } | ||||
|     inline void setPixelColor(unsigned n, CRGB c)                                         { setPixelColor(n, c.red, c.green, c.blue); } | ||||
|     inline void fill(uint32_t c)          { for (unsigned i = 0; i < getLengthTotal(); i++) setPixelColor(i, c); } // fill whole strip with color (inline) | ||||
|     inline void trigger(void)                                 { _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(void)                                 { _suspend = true; }    // will suspend (and canacel) strip.service() execution | ||||
|     inline void resume(void)                                  { _suspend = false; }   // will resume strip.service() execution | ||||
|  | ||||
|     bool | ||||
|       paletteFade, | ||||
|       checkSegmentAlignment(void), | ||||
|       hasRGBWBus(void), | ||||
|       hasCCTBus(void), | ||||
| @@ -787,49 +791,47 @@ class WS2812FX {  // 96 bytes | ||||
|       isUpdating(void), | ||||
|       deserializeMap(uint8_t n=0); | ||||
|  | ||||
|     inline bool isServicing(void) { return _isServicing; } | ||||
|     inline bool hasWhiteChannel(void) {return _hasWhiteChannel;} | ||||
|     inline bool isOffRefreshRequired(void) {return _isOffRefreshRequired;} | ||||
|     inline bool isServicing(void)          { return _isServicing; }           // returns true if strip.service() is executing | ||||
|     inline bool hasWhiteChannel(void)      { return _hasWhiteChannel; }       // returns true if strip contains separate white chanel | ||||
|     inline bool isOffRefreshRequired(void) { return _isOffRefreshRequired; }  // returns true if strip requires regular updates (i.e. TM1814 chipset) | ||||
|     inline bool isSuspended(void)          { return _suspend; }               // returns true if strip.service() execution is suspended | ||||
|     inline bool needsUpdate(void)          { return _triggered; }             // returns true if strip received a trigger() request | ||||
|  | ||||
|     uint8_t | ||||
|       paletteFade, | ||||
|       paletteBlend, | ||||
|       milliampsPerLed, | ||||
|       cctBlending, | ||||
|       getActiveSegmentsNum(void), | ||||
|       getFirstSelectedSegId(void), | ||||
|       getLastActiveSegmentId(void), | ||||
|       getActiveSegsLightCapabilities(bool selectedOnly = false), | ||||
|       setPixelSegment(uint8_t n); | ||||
|       getActiveSegsLightCapabilities(bool selectedOnly = false); | ||||
|  | ||||
|     inline uint8_t getBrightness(void) { return _brightness; } | ||||
|     inline uint8_t getMaxSegments(void) { return MAX_NUM_SEGMENTS; }  // returns maximum number of supported segments (fixed value) | ||||
|     inline uint8_t getSegmentsNum(void) { return _segments.size(); }  // returns currently present segments | ||||
|     inline uint8_t getCurrSegmentId(void) { return _segment_index; } | ||||
|     inline uint8_t getMainSegmentId(void) { return _mainSegment; } | ||||
|     inline uint8_t getPaletteCount() { return 13 + GRADIENT_PALETTE_COUNT; }  // will only return built-in palette count | ||||
|     inline uint8_t getTargetFps() { return _targetFps; } | ||||
|     inline uint8_t getModeCount() { return _modeCount; } | ||||
|     inline uint8_t getBrightness(void)    { return _brightness; }       // returns current strip brightness | ||||
|     inline uint8_t getMaxSegments(void)   { return MAX_NUM_SEGMENTS; }  // returns maximum number of supported segments (fixed value) | ||||
|     inline uint8_t getSegmentsNum(void)   { return _segments.size(); }  // returns currently present segments | ||||
|     inline uint8_t getCurrSegmentId(void) { return _segment_index; }    // returns current segment index (only valid while strip.isServicing()) | ||||
|     inline uint8_t getMainSegmentId(void) { return _mainSegment; }      // returns main segment index | ||||
|     inline uint8_t getPaletteCount()      { return 13 + GRADIENT_PALETTE_COUNT; }  // will only return built-in palette count | ||||
|     inline uint8_t getTargetFps()         { return _targetFps; }        // returns rough FPS value for las 2s interval | ||||
|     inline uint8_t getModeCount()         { return _modeCount; }        // returns number of registered modes/effects | ||||
|  | ||||
|     uint16_t | ||||
|       ablMilliampsMax, | ||||
|       currentMilliamps, | ||||
|       getLengthPhysical(void), | ||||
|       getLengthTotal(void), // will include virtual/nonexistent pixels in matrix | ||||
|       getFps(); | ||||
|       getFps(), | ||||
|       getMappedPixelIndex(uint16_t index); | ||||
|  | ||||
|     inline uint16_t getFrameTime(void) { return _frametime; } | ||||
|     inline uint16_t getMinShowDelay(void) { return MIN_SHOW_DELAY; } | ||||
|     inline uint16_t getLength(void) { return _length; } // 2D matrix may have less pixels than W*H | ||||
|     inline uint16_t getTransition(void) { return _transitionDur; } | ||||
|     inline uint16_t getFrameTime(void)    { return _frametime; }        // returns amount of time a frame should take (in ms) | ||||
|     inline uint16_t getMinShowDelay(void) { return MIN_SHOW_DELAY; }    // returns minimum amount of time strip.service() can be delayed (constant) | ||||
|     inline uint16_t getLength(void)       { return _length; }           // returns actual amount of LEDs on a strip (2D matrix may have less LEDs than W*H) | ||||
|     inline uint16_t getTransition(void)   { return _transitionDur; }    // returns currently set transition time (in ms) | ||||
|  | ||||
|     uint32_t | ||||
|       now, | ||||
|       timebase, | ||||
|       getPixelColor(uint16_t); | ||||
|  | ||||
|     inline uint32_t getLastShow(void) { return _lastShow; } | ||||
|     inline uint32_t segColor(uint8_t i) { return _colors_t[i]; } | ||||
|     inline uint32_t getLastShow(void)   { return _lastShow; }           // returns millis() timestamp of last strip.show() call | ||||
|     inline uint32_t segColor(uint8_t i) { return _colors_t[i]; }        // returns currently valid color (for slot i) AKA SEGCOLOR(); may be blended between two colors while in transition | ||||
|  | ||||
|     const char * | ||||
|       getModeData(uint8_t id = 0) { return (id && id<_modeCount) ? _modeData[id] : PSTR("Solid"); } | ||||
| @@ -838,9 +840,9 @@ class WS2812FX {  // 96 bytes | ||||
|       getModeDataSrc(void) { return &(_modeData[0]); } // vectors use arrays for underlying data | ||||
|  | ||||
|     Segment&        getSegment(uint8_t id); | ||||
|     inline Segment& getFirstSelectedSeg(void) { return _segments[getFirstSelectedSegId()]; } | ||||
|     inline Segment& getMainSegment(void)      { return _segments[getMainSegmentId()]; } | ||||
|     inline Segment* getSegments(void)         { return &(_segments[0]); } | ||||
|     inline Segment& getFirstSelectedSeg(void) { return _segments[getFirstSelectedSegId()]; }  // returns reference to first segment that is "selected" | ||||
|     inline Segment& getMainSegment(void)      { return _segments[getMainSegmentId()]; }       // returns reference to main segment | ||||
|     inline Segment* getSegments(void)         { return &(_segments[0]); }                     // returns pointer to segment vector structure (warning: use carefully) | ||||
|  | ||||
|   // 2D support (panels) | ||||
|     bool | ||||
| @@ -876,10 +878,10 @@ class WS2812FX {  // 96 bytes | ||||
|     std::vector<Panel> panel; | ||||
| #endif | ||||
|  | ||||
|     void setUpMatrix(); | ||||
|     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)   { setPixelColor(y * Segment::maxWidth + x, c); } | ||||
|     inline void setPixelColorXY(int x, int y, uint32_t c)   { setPixelColor((unsigned)(y * Segment::maxWidth + x), c); } | ||||
|     inline void setPixelColorXY(int x, int y, byte r, byte g, byte b, byte w = 0) { setPixelColorXY(x, y, RGBW32(r,g,b,w)); } | ||||
|     inline void setPixelColorXY(int x, int y, CRGB c)       { setPixelColorXY(x, y, RGBW32(c.r,c.g,c.b,0)); } | ||||
|  | ||||
| @@ -900,6 +902,8 @@ class WS2812FX {  // 96 bytes | ||||
|     friend class Segment; | ||||
|  | ||||
|   private: | ||||
|     volatile bool _suspend; | ||||
|  | ||||
|     uint16_t _length; | ||||
|     uint8_t  _brightness; | ||||
|     uint16_t _transitionDur; | ||||
| @@ -933,12 +937,10 @@ class WS2812FX {  // 96 bytes | ||||
|     uint16_t _qStart, _qStop, _qStartY, _qStopY; | ||||
|     uint8_t _qGrouping, _qSpacing; | ||||
|     uint16_t _qOffset; | ||||
|  | ||||
|     uint8_t | ||||
|       estimateCurrentAndLimitBri(void); | ||||
|  | ||||
| /* | ||||
|     void | ||||
|       setUpSegmentFromQueuedChanges(void); | ||||
| */ | ||||
| }; | ||||
|  | ||||
| extern const char JSON_mode_names[]; | ||||
|   | ||||
| @@ -36,14 +36,9 @@ | ||||
| // so matrix should disable regular ledmap processing | ||||
| void WS2812FX::setUpMatrix() { | ||||
| #ifndef WLED_DISABLE_2D | ||||
|   // erase old ledmap, just in case. | ||||
|   if (customMappingTable != nullptr) delete[] customMappingTable; | ||||
|   customMappingTable = nullptr; | ||||
|   customMappingSize = 0; | ||||
|  | ||||
|   // isMatrix is set in cfg.cpp or set.cpp | ||||
|   if (isMatrix) { | ||||
|     // calculate width dynamically because it will have gaps | ||||
|     // calculate width dynamically because it may have gaps | ||||
|     Segment::maxWidth = 1; | ||||
|     Segment::maxHeight = 1; | ||||
|     for (size_t i = 0; i < panel.size(); i++) { | ||||
| @@ -68,15 +63,17 @@ void WS2812FX::setUpMatrix() { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     customMappingTable = new uint16_t[Segment::maxWidth * Segment::maxHeight]; | ||||
|     customMappingSize = 0; // prevent use of mapping if anything goes wrong | ||||
|  | ||||
|     if (customMappingTable == nullptr) customMappingTable = new uint16_t[getLengthTotal()]; | ||||
|  | ||||
|     if (customMappingTable != nullptr) { | ||||
|       customMappingSize = Segment::maxWidth * Segment::maxHeight; | ||||
|       customMappingSize = getLengthTotal(); | ||||
|  | ||||
|       // fill with empty in case we don't fill the entire matrix | ||||
|       for (size_t i = 0; i< customMappingSize; i++) { | ||||
|         customMappingTable[i] = (uint16_t)-1; | ||||
|       } | ||||
|       unsigned matrixSize = Segment::maxWidth * Segment::maxHeight; | ||||
|       for (unsigned i = 0; i<matrixSize; i++) customMappingTable[i] = 0xFFFFU; | ||||
|       for (unsigned i = matrixSize; i<getLengthTotal(); i++) customMappingTable[i] = i; // trailing LEDs for ledmap (after matrix) if it exist | ||||
|  | ||||
|       // we will try to load a "gap" array (a JSON file) | ||||
|       // the array has to have the same amount of values as mapping array (or larger) | ||||
| @@ -94,14 +91,14 @@ void WS2812FX::setUpMatrix() { | ||||
|         DEBUG_PRINT(F("Reading LED gap from ")); | ||||
|         DEBUG_PRINTLN(fileName); | ||||
|         // read the array into global JSON buffer | ||||
|         if (readObjectFromFile(fileName, nullptr, &doc)) { | ||||
|         if (readObjectFromFile(fileName, nullptr, pDoc)) { | ||||
|           // the array is similar to ledmap, except it has only 3 values: | ||||
|           // -1 ... missing pixel (do not increase pixel count) | ||||
|           //  0 ... inactive pixel (it does count, but should be mapped out (-1)) | ||||
|           //  1 ... active pixel (it will count and will be mapped) | ||||
|           JsonArray map = doc.as<JsonArray>(); | ||||
|           JsonArray map = pDoc->as<JsonArray>(); | ||||
|           gapSize = map.size(); | ||||
|           if (!map.isNull() && gapSize >= customMappingSize) { // not an empty map | ||||
|           if (!map.isNull() && gapSize >= matrixSize) { // not an empty map | ||||
|             gapTable = new int8_t[gapSize]; | ||||
|             if (gapTable) for (size_t i = 0; i < gapSize; i++) { | ||||
|               gapTable[i] = constrain(map[i], -1, 1); | ||||
| @@ -136,7 +133,7 @@ void WS2812FX::setUpMatrix() { | ||||
|       DEBUG_PRINT(F("Matrix ledmap:")); | ||||
|       for (unsigned i=0; i<customMappingSize; i++) { | ||||
|         if (!(i%Segment::maxWidth)) DEBUG_PRINTLN(); | ||||
|         DEBUG_PRINTF("%4d,", customMappingTable[i]); | ||||
|         DEBUG_PRINTF_P(PSTR("%4d,"), customMappingTable[i]); | ||||
|       } | ||||
|       DEBUG_PRINTLN(); | ||||
|       #endif | ||||
| @@ -265,7 +262,7 @@ void Segment::setPixelColorXY(float x, float y, uint32_t col, bool aa) | ||||
| } | ||||
|  | ||||
| // returns RGBW values of pixel | ||||
| uint32_t Segment::getPixelColorXY(uint16_t x, uint16_t y) { | ||||
| uint32_t IRAM_ATTR Segment::getPixelColorXY(uint16_t x, uint16_t y) { | ||||
|   if (!isActive()) return 0; // not active | ||||
|   if (x >= virtualWidth() || y >= virtualHeight() || x<0 || y<0) return 0;  // if pixel would fall out of virtual segment just exit | ||||
|   if (reverse  ) x = virtualWidth()  - x - 1; | ||||
| @@ -277,23 +274,6 @@ uint32_t Segment::getPixelColorXY(uint16_t x, uint16_t y) { | ||||
|   return strip.getPixelColorXY(start + x, startY + y); | ||||
| } | ||||
|  | ||||
| // Blends the specified color with the existing pixel color. | ||||
| void Segment::blendPixelColorXY(uint16_t x, uint16_t y, uint32_t color, uint8_t blend) { | ||||
|   setPixelColorXY(x, y, color_blend(getPixelColorXY(x,y), color, blend)); | ||||
| } | ||||
|  | ||||
| // Adds the specified color with the existing pixel color perserving color balance. | ||||
| void Segment::addPixelColorXY(int x, int y, uint32_t color, bool fast) { | ||||
|   if (!isActive()) return; // not active | ||||
|   if (x >= virtualWidth() || y >= virtualHeight() || x<0 || y<0) return;  // if pixel would fall out of virtual segment just exit | ||||
|   setPixelColorXY(x, y, color_add(getPixelColorXY(x,y), color, fast)); | ||||
| } | ||||
|  | ||||
| void Segment::fadePixelColorXY(uint16_t x, uint16_t y, uint8_t fade) { | ||||
|   if (!isActive()) return; // not active | ||||
|   setPixelColorXY(x, y, color_fade(getPixelColorXY(x,y), fade, true)); | ||||
| } | ||||
|  | ||||
| // blurRow: perform a blur on a row of a rectangular matrix | ||||
| void Segment::blurRow(uint16_t row, fract8 blur_amount) { | ||||
|   if (!isActive() || blur_amount == 0) return; // not active | ||||
|   | ||||
| @@ -77,9 +77,10 @@ uint16_t Segment::_usedSegmentData = 0U; // amount of RAM all segments use for t | ||||
| uint16_t Segment::maxWidth = DEFAULT_LED_COUNT; | ||||
| uint16_t Segment::maxHeight = 1; | ||||
|  | ||||
| CRGBPalette16 Segment::_randomPalette = CRGBPalette16(DEFAULT_COLOR); | ||||
| CRGBPalette16 Segment::_newRandomPalette = CRGBPalette16(DEFAULT_COLOR); | ||||
| unsigned long Segment::_lastPaletteChange = 0; // perhaps it should be per segment | ||||
| CRGBPalette16 Segment::_randomPalette     = generateRandomPalette();  // was CRGBPalette16(DEFAULT_COLOR); | ||||
| CRGBPalette16 Segment::_newRandomPalette  = generateRandomPalette();  // was CRGBPalette16(DEFAULT_COLOR); | ||||
| uint16_t      Segment::_lastPaletteChange = 0; // perhaps it should be per segment | ||||
| uint16_t      Segment::_lastPaletteBlend  = 0; //in millis (lowest 16 bits only) | ||||
|  | ||||
| #ifndef WLED_DISABLE_MODE_BLEND | ||||
| bool Segment::_modeBlend = false; | ||||
| @@ -87,7 +88,7 @@ bool Segment::_modeBlend = false; | ||||
|  | ||||
| // copy constructor | ||||
| Segment::Segment(const Segment &orig) { | ||||
|   //DEBUG_PRINTF("-- Copy segment constructor: %p -> %p\n", &orig, this); | ||||
|   //DEBUG_PRINTF_P(PSTR("-- Copy segment constructor: %p -> %p\n"), &orig, this); | ||||
|   memcpy((void*)this, (void*)&orig, sizeof(Segment)); | ||||
|   _t = nullptr; // copied segment cannot be in transition | ||||
|   name = nullptr; | ||||
| @@ -99,7 +100,7 @@ Segment::Segment(const Segment &orig) { | ||||
|  | ||||
| // move constructor | ||||
| Segment::Segment(Segment &&orig) noexcept { | ||||
|   //DEBUG_PRINTF("-- Move segment constructor: %p -> %p\n", &orig, this); | ||||
|   //DEBUG_PRINTF_P(PSTR("-- Move segment constructor: %p -> %p\n"), &orig, this); | ||||
|   memcpy((void*)this, (void*)&orig, sizeof(Segment)); | ||||
|   orig._t   = nullptr; // old segment cannot be in transition any more | ||||
|   orig.name = nullptr; | ||||
| @@ -109,7 +110,7 @@ Segment::Segment(Segment &&orig) noexcept { | ||||
|  | ||||
| // copy assignment | ||||
| Segment& Segment::operator= (const Segment &orig) { | ||||
|   //DEBUG_PRINTF("-- Copying segment: %p -> %p\n", &orig, this); | ||||
|   //DEBUG_PRINTF_P(PSTR("-- Copying segment: %p -> %p\n"), &orig, this); | ||||
|   if (this != &orig) { | ||||
|     // clean destination | ||||
|     if (name) { delete[] name; name = nullptr; } | ||||
| @@ -129,7 +130,7 @@ Segment& Segment::operator= (const Segment &orig) { | ||||
|  | ||||
| // move assignment | ||||
| Segment& Segment::operator= (Segment &&orig) noexcept { | ||||
|   //DEBUG_PRINTF("-- Moving segment: %p -> %p\n", &orig, this); | ||||
|   //DEBUG_PRINTF_P(PSTR("-- Moving segment: %p -> %p\n"), &orig, this); | ||||
|   if (this != &orig) { | ||||
|     if (name) { delete[] name; name = nullptr; } // free old name | ||||
|     stopTransition(); | ||||
| @@ -143,40 +144,41 @@ Segment& Segment::operator= (Segment &&orig) noexcept { | ||||
|   return *this; | ||||
| } | ||||
|  | ||||
| bool Segment::allocateData(size_t len) { | ||||
| // allocates effect data buffer on heap and initialises (erases) it | ||||
| bool IRAM_ATTR Segment::allocateData(size_t len) { | ||||
|   if (len == 0) return false; // nothing to do | ||||
|   if (data && _dataLen >= len) {          // already allocated enough (reduce fragmentation) | ||||
|     if (call == 0) memset(data, 0, len);  // erase buffer if called during effect initialisation | ||||
|     return true; | ||||
|   } | ||||
|   //DEBUG_PRINTF("--   Allocating data (%d): %p\n", len, this); | ||||
|   deallocateData(); | ||||
|   if (len == 0) return false; // nothing to do | ||||
|   //DEBUG_PRINTF_P(PSTR("--   Allocating data (%d): %p\n", len, this); | ||||
|   deallocateData(); // if the old buffer was smaller release it first | ||||
|   if (Segment::getUsedSegmentData() + len > MAX_SEGMENT_DATA) { | ||||
|     // not enough memory | ||||
|     DEBUG_PRINT(F("!!! Effect RAM depleted: ")); | ||||
|     DEBUG_PRINTF("%d/%d !!!\n", len, Segment::getUsedSegmentData()); | ||||
|     DEBUG_PRINTF_P(PSTR("%d/%d !!!\n"), len, Segment::getUsedSegmentData()); | ||||
|     errorFlag = ERR_NORAM; | ||||
|     return false; | ||||
|   } | ||||
|   // do not use SPI RAM on ESP32 since it is slow | ||||
|   data = (byte*) malloc(len); | ||||
|   if (!data) { DEBUG_PRINTLN(F("!!! Allocation failed. !!!")); return false; } //allocation failed | ||||
|   data = (byte*)calloc(len, sizeof(byte)); | ||||
|   if (!data) { DEBUG_PRINTLN(F("!!! Allocation failed. !!!")); return false; } // allocation failed | ||||
|   Segment::addUsedSegmentData(len); | ||||
|   //DEBUG_PRINTF("---  Allocated data (%p): %d/%d -> %p\n", this, len, Segment::getUsedSegmentData(), data); | ||||
|   //DEBUG_PRINTF_P(PSTR("---  Allocated data (%p): %d/%d -> %p\n"), this, len, Segment::getUsedSegmentData(), data); | ||||
|   _dataLen = len; | ||||
|   memset(data, 0, len); | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| void Segment::deallocateData() { | ||||
| void IRAM_ATTR Segment::deallocateData() { | ||||
|   if (!data) { _dataLen = 0; return; } | ||||
|   //DEBUG_PRINTF("---  Released data (%p): %d/%d -> %p\n", this, _dataLen, Segment::getUsedSegmentData(), data); | ||||
|   //DEBUG_PRINTF_P(PSTR("---  Released data (%p): %d/%d -> %p\n"), this, _dataLen, Segment::getUsedSegmentData(), data); | ||||
|   if ((Segment::getUsedSegmentData() > 0) && (_dataLen > 0)) { // check that we don't have a dangling / inconsistent data pointer | ||||
|     free(data); | ||||
|   } else { | ||||
|     DEBUG_PRINT(F("---- Released data ")); | ||||
|     DEBUG_PRINTF("(%p): ", this); | ||||
|     DEBUG_PRINTF_P(PSTR("(%p): "), this); | ||||
|     DEBUG_PRINT(F("inconsistent UsedSegmentData ")); | ||||
|     DEBUG_PRINTF("(%d/%d)", _dataLen, Segment::getUsedSegmentData()); | ||||
|     DEBUG_PRINTF_P(PSTR("(%d/%d)"), _dataLen, Segment::getUsedSegmentData()); | ||||
|     DEBUG_PRINTLN(F(", cowardly refusing to free nothing.")); | ||||
|   } | ||||
|   data = nullptr; | ||||
| @@ -193,13 +195,13 @@ void Segment::deallocateData() { | ||||
|   */ | ||||
| void Segment::resetIfRequired() { | ||||
|   if (!reset) return; | ||||
|   //DEBUG_PRINTF("-- Segment reset: %p\n", this); | ||||
|   deallocateData(); | ||||
|   //DEBUG_PRINTF_P(PSTR("-- Segment reset: %p\n"), this); | ||||
|   if (data && _dataLen > 0) memset(data, 0, _dataLen);  // prevent heap fragmentation (just erase buffer instead of deallocateData()) | ||||
|   next_time = 0; step = 0; call = 0; aux0 = 0; aux1 = 0; | ||||
|   reset = false; | ||||
| } | ||||
|  | ||||
| CRGBPalette16 &Segment::loadPalette(CRGBPalette16 &targetPalette, uint8_t pal) { | ||||
| CRGBPalette16 IRAM_ATTR &Segment::loadPalette(CRGBPalette16 &targetPalette, uint8_t pal) { | ||||
|   if (pal < 245 && pal > GRADIENT_PALETTE_COUNT+13) pal = 0; | ||||
|   if (pal > 245 && (strip.customPalettes.size() == 0 || 255U-pal > strip.customPalettes.size()-1)) pal = 0; | ||||
|   //default palette. Differs depending on effect | ||||
| @@ -219,20 +221,9 @@ CRGBPalette16 &Segment::loadPalette(CRGBPalette16 &targetPalette, uint8_t pal) { | ||||
|   switch (pal) { | ||||
|     case 0: //default palette. Exceptions for specific effects above | ||||
|       targetPalette = PartyColors_p; break; | ||||
|     case 1: {//periodically replace palette with a random one | ||||
|       unsigned long timeSinceLastChange = millis() - _lastPaletteChange; | ||||
|       if (timeSinceLastChange > randomPaletteChangeTime * 1000U) { | ||||
|         _randomPalette = _newRandomPalette; | ||||
|         _newRandomPalette = CRGBPalette16( | ||||
|                         CHSV(random8(), random8(160, 255), random8(128, 255)), | ||||
|                         CHSV(random8(), random8(160, 255), random8(128, 255)), | ||||
|                         CHSV(random8(), random8(160, 255), random8(128, 255)), | ||||
|                         CHSV(random8(), random8(160, 255), random8(128, 255))); | ||||
|         _lastPaletteChange = millis(); | ||||
|         handleRandomPalette(); // do a 1st pass of blend | ||||
|       } | ||||
|       targetPalette = _randomPalette; | ||||
|       break;} | ||||
|     case 1: //randomly generated palette | ||||
|       targetPalette = _randomPalette; //random palette is generated at intervals in handleRandomPalette()  | ||||
|       break; | ||||
|     case 2: {//primary color only | ||||
|       CRGB prim = gamma32(colors[0]); | ||||
|       targetPalette = CRGBPalette16(prim); break;} | ||||
| @@ -293,7 +284,7 @@ void Segment::startTransition(uint16_t dur) { | ||||
|   _t = new Transition(dur); // no previous transition running | ||||
|   if (!_t) return; // failed to allocate data | ||||
|  | ||||
|   //DEBUG_PRINTF("-- Started transition: %p\n", this); | ||||
|   //DEBUG_PRINTF_P(PSTR("-- Started transition: %p (%p)\n"), this, _t); | ||||
|   loadPalette(_t->_palT, palette); | ||||
|   _t->_briT           = on ? opacity : 0; | ||||
|   _t->_cctT           = cct; | ||||
| @@ -306,7 +297,7 @@ void Segment::startTransition(uint16_t dur) { | ||||
|     if (_dataLen > 0 && data) { | ||||
|       _t->_segT._dataT = (byte *)malloc(_dataLen); | ||||
|       if (_t->_segT._dataT) { | ||||
|         //DEBUG_PRINTF("--  Allocated duplicate data (%d): %p\n", _dataLen, _t->_segT._dataT); | ||||
|         //DEBUG_PRINTF_P(PSTR("--  Allocated duplicate data (%d) for %p: %p\n"), _dataLen, this, _t->_segT._dataT); | ||||
|         memcpy(_t->_segT._dataT, data, _dataLen); | ||||
|         _t->_segT._dataLenT = _dataLen; | ||||
|       } | ||||
| @@ -320,11 +311,11 @@ void Segment::startTransition(uint16_t dur) { | ||||
| } | ||||
|  | ||||
| void Segment::stopTransition() { | ||||
|   //DEBUG_PRINTF("-- Stopping transition: %p\n", this); | ||||
|   if (isInTransition()) { | ||||
|     //DEBUG_PRINTF_P(PSTR("-- Stopping transition: %p\n"), this); | ||||
|     #ifndef WLED_DISABLE_MODE_BLEND | ||||
|     if (_t->_segT._dataT && _t->_segT._dataLenT > 0) { | ||||
|       //DEBUG_PRINTF("--  Released duplicate data (%d): %p\n", _t->_segT._dataLenT, _t->_segT._dataT); | ||||
|       //DEBUG_PRINTF_P(PSTR("--  Released duplicate data (%d) for %p: %p\n"), _t->_segT._dataLenT, this, _t->_segT._dataT); | ||||
|       free(_t->_segT._dataT); | ||||
|       _t->_segT._dataT = nullptr; | ||||
|       _t->_segT._dataLenT = 0; | ||||
| @@ -341,7 +332,7 @@ void Segment::handleTransition() { | ||||
| } | ||||
|  | ||||
| // transition progression between 0-65535 | ||||
| uint16_t Segment::progress() { | ||||
| uint16_t IRAM_ATTR Segment::progress() { | ||||
|   if (isInTransition()) { | ||||
|     unsigned long timeNow = millis(); | ||||
|     if (_t->_dur > 0 && timeNow - _t->_start < _t->_dur) return (timeNow - _t->_start) * 0xFFFFU / _t->_dur; | ||||
| @@ -351,7 +342,7 @@ uint16_t Segment::progress() { | ||||
|  | ||||
| #ifndef WLED_DISABLE_MODE_BLEND | ||||
| void Segment::swapSegenv(tmpsegd_t &tmpSeg) { | ||||
|   //DEBUG_PRINTF("--  Saving temp seg: %p (%p)\n", this, tmpSeg); | ||||
|   //DEBUG_PRINTF_P(PSTR("--  Saving temp seg: %p->(%p) [%d->%p]\n"), this, &tmpSeg, _dataLen, data); | ||||
|   tmpSeg._optionsT   = options; | ||||
|   for (size_t i=0; i<NUM_COLORS; i++) tmpSeg._colorT[i] = colors[i]; | ||||
|   tmpSeg._speedT     = speed; | ||||
| @@ -387,18 +378,17 @@ void Segment::swapSegenv(tmpsegd_t &tmpSeg) { | ||||
|     data      = _t->_segT._dataT; | ||||
|     _dataLen  = _t->_segT._dataLenT; | ||||
|   } | ||||
|   //DEBUG_PRINTF("--   temp seg data: %p (%d,%p)\n", this, _dataLen, data); | ||||
| } | ||||
|  | ||||
| void Segment::restoreSegenv(tmpsegd_t &tmpSeg) { | ||||
|   //DEBUG_PRINTF("--  Restoring temp seg: %p (%p)\n", this, tmpSeg); | ||||
|   //DEBUG_PRINTF_P(PSTR("--  Restoring temp seg: %p->(%p) [%d->%p]\n"), &tmpSeg, this, _dataLen, data); | ||||
|   if (_t && &(_t->_segT) != &tmpSeg) { | ||||
|     // update possibly changed variables to keep old effect running correctly | ||||
|     _t->_segT._aux0T = aux0; | ||||
|     _t->_segT._aux1T = aux1; | ||||
|     _t->_segT._stepT = step; | ||||
|     _t->_segT._callT = call; | ||||
|     //if (_t->_segT._dataT != data) DEBUG_PRINTF("---  data re-allocated: (%p) %p -> %p\n", this, _t->_segT._dataT, data); | ||||
|     //if (_t->_segT._dataT != data) DEBUG_PRINTF_P(PSTR("---  data re-allocated: (%p) %p -> %p\n"), this, _t->_segT._dataT, data); | ||||
|     _t->_segT._dataT = data; | ||||
|     _t->_segT._dataLenT = _dataLen; | ||||
|   } | ||||
| @@ -418,11 +408,10 @@ void Segment::restoreSegenv(tmpsegd_t &tmpSeg) { | ||||
|   call      = tmpSeg._callT; | ||||
|   data      = tmpSeg._dataT; | ||||
|   _dataLen  = tmpSeg._dataLenT; | ||||
|   //DEBUG_PRINTF("--   temp seg data: %p (%d,%p)\n", this, _dataLen, data); | ||||
| } | ||||
| #endif | ||||
|  | ||||
| uint8_t Segment::currentBri(bool useCct) { | ||||
| uint8_t IRAM_ATTR Segment::currentBri(bool useCct) { | ||||
|   uint32_t prog = progress(); | ||||
|   if (prog < 0xFFFFU) { | ||||
|     uint32_t curBri = (useCct ? cct : (on ? opacity : 0)) * prog; | ||||
| @@ -432,7 +421,7 @@ uint8_t Segment::currentBri(bool useCct) { | ||||
|   return (useCct ? cct : (on ? opacity : 0)); | ||||
| } | ||||
|  | ||||
| uint8_t Segment::currentMode() { | ||||
| uint8_t IRAM_ATTR Segment::currentMode() { | ||||
| #ifndef WLED_DISABLE_MODE_BLEND | ||||
|   uint16_t prog = progress(); | ||||
|   if (modeBlending && prog < 0xFFFFU) return _t->_modeT; | ||||
| @@ -440,7 +429,8 @@ uint8_t Segment::currentMode() { | ||||
|   return mode; | ||||
| } | ||||
|  | ||||
| uint32_t Segment::currentColor(uint8_t slot) { | ||||
| uint32_t IRAM_ATTR Segment::currentColor(uint8_t slot) { | ||||
|   if (slot >= NUM_COLORS) slot = 0; | ||||
| #ifndef WLED_DISABLE_MODE_BLEND | ||||
|   return isInTransition() ? color_blend(_t->_segT._colorT[slot], colors[slot], progress(), true) : colors[slot]; | ||||
| #else | ||||
| @@ -448,7 +438,7 @@ uint32_t Segment::currentColor(uint8_t slot) { | ||||
| #endif | ||||
| } | ||||
|  | ||||
| CRGBPalette16 &Segment::currentPalette(CRGBPalette16 &targetPalette, uint8_t pal) { | ||||
| CRGBPalette16 IRAM_ATTR &Segment::currentPalette(CRGBPalette16 &targetPalette, uint8_t pal) { | ||||
|   loadPalette(targetPalette, pal); | ||||
|   uint16_t prog = progress(); | ||||
|   if (strip.paletteFade && prog < 0xFFFFU) { | ||||
| @@ -462,15 +452,27 @@ CRGBPalette16 &Segment::currentPalette(CRGBPalette16 &targetPalette, uint8_t pal | ||||
|   return targetPalette; | ||||
| } | ||||
|  | ||||
| // relies on WS2812FX::service() to call it max every 8ms or more (MIN_SHOW_DELAY) | ||||
| // relies on WS2812FX::service() to call it for each frame | ||||
| void Segment::handleRandomPalette() { | ||||
|   // just do a blend; if the palettes are identical it will just compare 48 bytes (same as _randomPalette == _newRandomPalette) | ||||
|   // this will slowly blend _newRandomPalette into _randomPalette every 15ms or 8ms (depending on MIN_SHOW_DELAY) | ||||
|   // is it time to generate a new palette? | ||||
|   if ((millis()/1000U) - _lastPaletteChange > randomPaletteChangeTime) { | ||||
|         _newRandomPalette = useHarmonicRandomPalette ? generateHarmonicRandomPalette(_randomPalette) : generateRandomPalette(); | ||||
|         _lastPaletteChange = millis()/1000U; | ||||
|         _lastPaletteBlend = (uint16_t)(millis() & 0xFFFF)-512; // starts blending immediately | ||||
|   } | ||||
|  | ||||
|   // if palette transitions is enabled, blend it according to Transition Time (if longer than minimum given by service calls) | ||||
|   if (strip.paletteFade) { | ||||
|     // assumes that 128 updates are sufficient to blend a palette, so shift by 7 (can be more, can be less) | ||||
|     // in reality there need to be 255 blends to fully blend two entirely different palettes | ||||
|     if ((millis() & 0xFFFF) - _lastPaletteBlend < strip.getTransition() >> 7) return; // not yet time to fade, delay the update | ||||
|     _lastPaletteBlend = millis(); | ||||
|   } | ||||
|   nblendPaletteTowardPalette(_randomPalette, _newRandomPalette, 48); | ||||
| } | ||||
|  | ||||
| // segId is given when called from network callback, changes are queued if that segment is currently in its effect function | ||||
| void Segment::setUp(uint16_t i1, uint16_t i2, uint8_t grp, uint8_t spc, uint16_t ofs, uint16_t i1Y, uint16_t i2Y, uint8_t segId) { | ||||
| void Segment::setUp(uint16_t i1, uint16_t i2, uint8_t grp, uint8_t spc, uint16_t ofs, uint16_t i1Y, uint16_t i2Y) { | ||||
|   // return if neither bounds nor grouping have changed | ||||
|   bool boundsUnchanged = (start == i1 && stop == i2); | ||||
|   #ifndef WLED_DISABLE_2D | ||||
| @@ -563,35 +565,36 @@ void Segment::setOption(uint8_t n, bool val) { | ||||
| } | ||||
|  | ||||
| void Segment::setMode(uint8_t fx, bool loadDefaults) { | ||||
|   // skip reserved | ||||
|   while (fx < strip.getModeCount() && strncmp_P("RSVD", strip.getModeData(fx), 4) == 0) fx++; | ||||
|   if (fx >= strip.getModeCount()) fx = 0; // set solid mode | ||||
|   // if we have a valid mode & is not reserved | ||||
|   if (fx < strip.getModeCount() && strncmp_P("RSVD", strip.getModeData(fx), 4)) { | ||||
|     if (fx != mode) { | ||||
|   if (fx != mode) { | ||||
| #ifndef WLED_DISABLE_MODE_BLEND | ||||
|       if (modeBlending) startTransition(strip.getTransition()); // set effect transitions | ||||
|     if (modeBlending) startTransition(strip.getTransition()); // set effect transitions | ||||
| #endif | ||||
|       mode = fx; | ||||
|       // load default values from effect string | ||||
|       if (loadDefaults) { | ||||
|         int16_t sOpt; | ||||
|         sOpt = extractModeDefaults(fx, "sx");   speed     = (sOpt >= 0) ? sOpt : DEFAULT_SPEED; | ||||
|         sOpt = extractModeDefaults(fx, "ix");   intensity = (sOpt >= 0) ? sOpt : DEFAULT_INTENSITY; | ||||
|         sOpt = extractModeDefaults(fx, "c1");   custom1   = (sOpt >= 0) ? sOpt : DEFAULT_C1; | ||||
|         sOpt = extractModeDefaults(fx, "c2");   custom2   = (sOpt >= 0) ? sOpt : DEFAULT_C2; | ||||
|         sOpt = extractModeDefaults(fx, "c3");   custom3   = (sOpt >= 0) ? sOpt : DEFAULT_C3; | ||||
|         sOpt = extractModeDefaults(fx, "o1");   check1    = (sOpt >= 0) ? (bool)sOpt : false; | ||||
|         sOpt = extractModeDefaults(fx, "o2");   check2    = (sOpt >= 0) ? (bool)sOpt : false; | ||||
|         sOpt = extractModeDefaults(fx, "o3");   check3    = (sOpt >= 0) ? (bool)sOpt : false; | ||||
|         sOpt = extractModeDefaults(fx, "m12");  if (sOpt >= 0) map1D2D   = constrain(sOpt, 0, 7); else map1D2D = M12_Pixels;  // reset mapping if not defined (2D FX may not work) | ||||
|         sOpt = extractModeDefaults(fx, "si");   if (sOpt >= 0) soundSim  = constrain(sOpt, 0, 1); | ||||
|         sOpt = extractModeDefaults(fx, "rev");  if (sOpt >= 0) reverse   = (bool)sOpt; | ||||
|         sOpt = extractModeDefaults(fx, "mi");   if (sOpt >= 0) mirror    = (bool)sOpt; // NOTE: setting this option is a risky business | ||||
|         sOpt = extractModeDefaults(fx, "rY");   if (sOpt >= 0) reverse_y = (bool)sOpt; | ||||
|         sOpt = extractModeDefaults(fx, "mY");   if (sOpt >= 0) mirror_y  = (bool)sOpt; // NOTE: setting this option is a risky business | ||||
|         sOpt = extractModeDefaults(fx, "pal");  if (sOpt >= 0) setPalette(sOpt); //else setPalette(0); | ||||
|       } | ||||
|       markForReset(); | ||||
|       stateChanged = true; // send UDP/WS broadcast | ||||
|     mode = fx; | ||||
|     // load default values from effect string | ||||
|     if (loadDefaults) { | ||||
|       int16_t sOpt; | ||||
|       sOpt = extractModeDefaults(fx, "sx");  speed     = (sOpt >= 0) ? sOpt : DEFAULT_SPEED; | ||||
|       sOpt = extractModeDefaults(fx, "ix");  intensity = (sOpt >= 0) ? sOpt : DEFAULT_INTENSITY; | ||||
|       sOpt = extractModeDefaults(fx, "c1");  custom1   = (sOpt >= 0) ? sOpt : DEFAULT_C1; | ||||
|       sOpt = extractModeDefaults(fx, "c2");  custom2   = (sOpt >= 0) ? sOpt : DEFAULT_C2; | ||||
|       sOpt = extractModeDefaults(fx, "c3");  custom3   = (sOpt >= 0) ? sOpt : DEFAULT_C3; | ||||
|       sOpt = extractModeDefaults(fx, "o1");  check1    = (sOpt >= 0) ? (bool)sOpt : false; | ||||
|       sOpt = extractModeDefaults(fx, "o2");  check2    = (sOpt >= 0) ? (bool)sOpt : false; | ||||
|       sOpt = extractModeDefaults(fx, "o3");  check3    = (sOpt >= 0) ? (bool)sOpt : false; | ||||
|       sOpt = extractModeDefaults(fx, "m12"); if (sOpt >= 0) map1D2D   = constrain(sOpt, 0, 7); else map1D2D = M12_Pixels;  // reset mapping if not defined (2D FX may not work) | ||||
|       sOpt = extractModeDefaults(fx, "si");  if (sOpt >= 0) soundSim  = constrain(sOpt, 0, 3); | ||||
|       sOpt = extractModeDefaults(fx, "rev"); if (sOpt >= 0) reverse   = (bool)sOpt; | ||||
|       sOpt = extractModeDefaults(fx, "mi");  if (sOpt >= 0) mirror    = (bool)sOpt; // NOTE: setting this option is a risky business | ||||
|       sOpt = extractModeDefaults(fx, "rY");  if (sOpt >= 0) reverse_y = (bool)sOpt; | ||||
|       sOpt = extractModeDefaults(fx, "mY");  if (sOpt >= 0) mirror_y  = (bool)sOpt; // NOTE: setting this option is a risky business | ||||
|       sOpt = extractModeDefaults(fx, "pal"); if (sOpt >= 0) setPalette(sOpt); //else setPalette(0); | ||||
|     } | ||||
|     markForReset(); | ||||
|     stateChanged = true; // send UDP/WS broadcast | ||||
|   } | ||||
| } | ||||
|  | ||||
| @@ -606,21 +609,21 @@ void Segment::setPalette(uint8_t pal) { | ||||
| } | ||||
|  | ||||
| // 2D matrix | ||||
| uint16_t Segment::virtualWidth() const { | ||||
| uint16_t IRAM_ATTR Segment::virtualWidth() const { | ||||
|   uint16_t groupLen = groupLength(); | ||||
|   uint16_t vWidth = ((transpose ? height() : width()) + groupLen - 1) / groupLen; | ||||
|   if (mirror) vWidth = (vWidth + 1) /2;  // divide by 2 if mirror, leave at least a single LED | ||||
|   return vWidth; | ||||
| } | ||||
|  | ||||
| uint16_t Segment::virtualHeight() const { | ||||
| uint16_t IRAM_ATTR Segment::virtualHeight() const { | ||||
|   uint16_t groupLen = groupLength(); | ||||
|   uint16_t vHeight = ((transpose ? width() : height()) + groupLen - 1) / groupLen; | ||||
|   if (mirror_y) vHeight = (vHeight + 1) /2;  // divide by 2 if mirror, leave at least a single LED | ||||
|   return vHeight; | ||||
| } | ||||
|  | ||||
| uint16_t Segment::nrOfVStrips() const { | ||||
| uint16_t IRAM_ATTR Segment::nrOfVStrips() const { | ||||
|   uint16_t vLen = 1; | ||||
| #ifndef WLED_DISABLE_2D | ||||
|   if (is2D()) { | ||||
| @@ -635,7 +638,7 @@ uint16_t Segment::nrOfVStrips() const { | ||||
| } | ||||
|  | ||||
| // 1D strip | ||||
| uint16_t Segment::virtualLength() const { | ||||
| uint16_t IRAM_ATTR Segment::virtualLength() const { | ||||
| #ifndef WLED_DISABLE_2D | ||||
|   if (is2D()) { | ||||
|     uint16_t vW = virtualWidth(); | ||||
| @@ -753,10 +756,10 @@ void IRAM_ATTR Segment::setPixelColor(int i, uint32_t col) | ||||
|   uint32_t tmpCol = col; | ||||
|   // set all the pixels in the group | ||||
|   for (int j = 0; j < grouping; j++) { | ||||
|     uint16_t indexSet = i + ((reverse) ? -j : j); | ||||
|     unsigned indexSet = i + ((reverse) ? -j : j); | ||||
|     if (indexSet >= start && indexSet < stop) { | ||||
|       if (mirror) { //set the corresponding mirrored pixel | ||||
|         uint16_t indexMir = stop - indexSet + start - 1; | ||||
|         unsigned indexMir = stop - indexSet + start - 1; | ||||
|         indexMir += offset; // offset/phase | ||||
|         if (indexMir >= stop) indexMir -= len; // wrap | ||||
| #ifndef WLED_DISABLE_MODE_BLEND | ||||
| @@ -807,7 +810,7 @@ void Segment::setPixelColor(float i, uint32_t col, bool aa) | ||||
|   } | ||||
| } | ||||
|  | ||||
| uint32_t Segment::getPixelColor(int i) | ||||
| uint32_t IRAM_ATTR Segment::getPixelColor(int i) | ||||
| { | ||||
|   if (!isActive()) return 0; // not active | ||||
| #ifndef WLED_DISABLE_2D | ||||
| @@ -886,8 +889,7 @@ void Segment::refreshLightCapabilities() { | ||||
|   if (start < Segment::maxWidth * Segment::maxHeight) { | ||||
|     // we are withing 2D matrix (includes 1D segments) | ||||
|     for (int y = startY; y < stopY; y++) for (int x = start; x < stop; x++) { | ||||
|       uint16_t index = x + Segment::maxWidth * y; | ||||
|       if (index < strip.customMappingSize) index = strip.customMappingTable[index]; // convert logical address to physical | ||||
|       uint16_t index = strip.getMappedPixelIndex(x + Segment::maxWidth * y); // convert logical address to physical | ||||
|       if (index < 0xFFFFU) { | ||||
|         if (segStartIdx > index) segStartIdx = index; | ||||
|         if (segStopIdx  < index) segStopIdx  = index; | ||||
| @@ -900,8 +902,8 @@ void Segment::refreshLightCapabilities() { | ||||
|     segStopIdx  = stop; | ||||
|   } | ||||
|  | ||||
|   for (unsigned b = 0; b < busses.getNumBusses(); b++) { | ||||
|     Bus *bus = busses.getBus(b); | ||||
|   for (unsigned b = 0; b < BusManager::getNumBusses(); b++) { | ||||
|     Bus *bus = BusManager::getBus(b); | ||||
|     if (bus == nullptr || bus->getLength()==0) break; | ||||
|     if (!bus->isOk()) continue; | ||||
|     if (bus->getStart() >= segStopIdx) continue; | ||||
| @@ -936,22 +938,6 @@ void Segment::fill(uint32_t c) { | ||||
|   } | ||||
| } | ||||
|  | ||||
| // Blends the specified color with the existing pixel color. | ||||
| void Segment::blendPixelColor(int n, uint32_t color, uint8_t blend) { | ||||
|   setPixelColor(n, color_blend(getPixelColor(n), color, blend)); | ||||
| } | ||||
|  | ||||
| // Adds the specified color with the existing pixel color perserving color balance. | ||||
| void Segment::addPixelColor(int n, uint32_t color, bool fast) { | ||||
|   if (!isActive()) return; // not active | ||||
|   setPixelColor(n, color_add(getPixelColor(n), color, fast)); | ||||
| } | ||||
|  | ||||
| void Segment::fadePixelColor(uint16_t n, uint8_t fade) { | ||||
|   if (!isActive()) return; // not active | ||||
|   setPixelColor(n, color_fade(getPixelColor(n), fade, true)); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * fade out function, higher rate = quicker fade | ||||
|  */ | ||||
| @@ -1007,8 +993,7 @@ void Segment::fadeToBlackBy(uint8_t fadeBy) { | ||||
| /* | ||||
|  * blurs segment content, source: FastLED colorutils.cpp | ||||
|  */ | ||||
| void Segment::blur(uint8_t blur_amount) | ||||
| { | ||||
| void Segment::blur(uint8_t blur_amount) { | ||||
|   if (!isActive() || blur_amount == 0) return; // optimization: 0 means "don't blur" | ||||
| #ifndef WLED_DISABLE_2D | ||||
|   if (is2D()) { | ||||
| @@ -1043,16 +1028,17 @@ void Segment::blur(uint8_t blur_amount) | ||||
|  * Inspired by the Adafruit examples. | ||||
|  */ | ||||
| uint32_t Segment::color_wheel(uint8_t pos) { | ||||
|   if (palette) return color_from_palette(pos, false, true, 0); | ||||
|   if (palette) return color_from_palette(pos, false, true, 0); // perhaps "strip.paletteBlend < 2" should be better instead of "true" | ||||
|   uint8_t w = W(currentColor(0)); | ||||
|   pos = 255 - pos; | ||||
|   if (pos < 85) { | ||||
|     return ((uint32_t)(255 - pos * 3) << 16) | ((uint32_t)(0) << 8) | (pos * 3); | ||||
|     return RGBW32((255 - pos * 3), 0, (pos * 3), w); | ||||
|   } else if(pos < 170) { | ||||
|     pos -= 85; | ||||
|     return ((uint32_t)(0) << 16) | ((uint32_t)(pos * 3) << 8) | (255 - pos * 3); | ||||
|     return RGBW32(0, (pos * 3), (255 - pos * 3), w); | ||||
|   } else { | ||||
|     pos -= 170; | ||||
|     return ((uint32_t)(pos * 3) << 16) | ((uint32_t)(255 - pos * 3) << 8) | (0); | ||||
|     return RGBW32((pos * 3), (255 - pos * 3), 0, w); | ||||
|   } | ||||
| } | ||||
|  | ||||
| @@ -1065,26 +1051,21 @@ uint32_t Segment::color_wheel(uint8_t pos) { | ||||
|  * @param pbri Value to scale the brightness of the returned color by. Default is 255. (no scaling) | ||||
|  * @returns Single color from palette | ||||
|  */ | ||||
| uint32_t Segment::color_from_palette(uint16_t i, bool mapping, bool wrap, uint8_t mcol, uint8_t pbri) | ||||
| { | ||||
| uint32_t Segment::color_from_palette(uint16_t i, bool mapping, bool wrap, uint8_t mcol, uint8_t pbri) { | ||||
|   uint32_t color = gamma32(currentColor(mcol)); | ||||
|  | ||||
|   // default palette or no RGB support on segment | ||||
|   if ((palette == 0 && mcol < NUM_COLORS) || !_isRGB) { | ||||
|     uint32_t color = currentColor(mcol); | ||||
|     color = gamma32(color); | ||||
|     if (pbri == 255) return color; | ||||
|     return color_fade(color, pbri, true); | ||||
|   } | ||||
|   if ((palette == 0 && mcol < NUM_COLORS) || !_isRGB) return (pbri == 255) ? color : color_fade(color, pbri, true); | ||||
|  | ||||
|   uint8_t paletteIndex = i; | ||||
|   if (mapping && virtualLength() > 1) paletteIndex = (i*255)/(virtualLength() -1); | ||||
|   // paletteBlend: 0 - wrap when moving, 1 - always wrap, 2 - never wrap, 3 - none (undefined) | ||||
|   if (!wrap && strip.paletteBlend != 3) paletteIndex = scale8(paletteIndex, 240); //cut off blend at palette "end" | ||||
|   CRGBPalette16 curPal; | ||||
|   curPal = currentPalette(curPal, palette); | ||||
|   //if (isInTransition()) curPal = _t->_palT; | ||||
|   //else    loadPalette(curPal, palette); | ||||
|   CRGB fastled_col = ColorFromPalette(curPal, paletteIndex, pbri, (strip.paletteBlend == 3)? NOBLEND:LINEARBLEND); // NOTE: paletteBlend should be global | ||||
|  | ||||
|   return RGBW32(fastled_col.r, fastled_col.g, fastled_col.b, 0); | ||||
|   return RGBW32(fastled_col.r, fastled_col.g, fastled_col.b, W(color)); | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -1093,8 +1074,7 @@ uint32_t Segment::color_from_palette(uint16_t i, bool mapping, bool wrap, uint8_ | ||||
| /////////////////////////////////////////////////////////////////////////////// | ||||
|  | ||||
| //do not call this method from system context (network callback) | ||||
| void WS2812FX::finalizeInit(void) | ||||
| { | ||||
| void WS2812FX::finalizeInit(void) { | ||||
|   //reset segment runtimes | ||||
|   for (segment &seg : _segments) { | ||||
|     seg.markForReset(); | ||||
| @@ -1109,7 +1089,7 @@ void WS2812FX::finalizeInit(void) | ||||
|   _hasWhiteChannel = _isOffRefreshRequired = false; | ||||
|  | ||||
|   //if busses failed to load, add default (fresh install, FS issue, ...) | ||||
|   if (busses.getNumBusses() == 0) { | ||||
|   if (BusManager::getNumBusses() == 0) { | ||||
|     DEBUG_PRINTLN(F("No busses, init default")); | ||||
|     const uint8_t defDataPins[] = {DATA_PINS}; | ||||
|     const uint16_t defCounts[] = {PIXEL_COUNTS}; | ||||
| @@ -1122,13 +1102,13 @@ void WS2812FX::finalizeInit(void) | ||||
|       uint16_t count = defCounts[(i < defNumCounts) ? i : defNumCounts -1]; | ||||
|       prevLen += count; | ||||
|       BusConfig defCfg = BusConfig(DEFAULT_LED_TYPE, defPin, start, count, DEFAULT_LED_COLOR_ORDER, false, 0, RGBW_MODE_MANUAL_ONLY); | ||||
|       if (busses.add(defCfg) == -1) break; | ||||
|       if (BusManager::add(defCfg) == -1) break; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   _length = 0; | ||||
|   for (int i=0; i<busses.getNumBusses(); i++) { | ||||
|     Bus *bus = busses.getBus(i); | ||||
|   for (int i=0; i<BusManager::getNumBusses(); i++) { | ||||
|     Bus *bus = BusManager::getBus(i); | ||||
|     if (bus == nullptr) continue; | ||||
|     if (bus->getStart() + bus->getLength() > MAX_LEDS) break; | ||||
|     //RGBW mode is enabled if at least one of the strips is RGBW | ||||
| @@ -1146,29 +1126,28 @@ void WS2812FX::finalizeInit(void) | ||||
|     #endif | ||||
|   } | ||||
|  | ||||
|   if (isMatrix) setUpMatrix(); | ||||
|   else { | ||||
|     Segment::maxWidth  = _length; | ||||
|     Segment::maxHeight = 1; | ||||
|   } | ||||
|   Segment::maxWidth  = _length; | ||||
|   Segment::maxHeight = 1; | ||||
|  | ||||
|   //segments are created in makeAutoSegments(); | ||||
|   DEBUG_PRINTLN(F("Loading custom palettes")); | ||||
|   loadCustomPalettes(); // (re)load all custom palettes | ||||
|   DEBUG_PRINTLN(F("Loading custom ledmaps")); | ||||
|   deserializeMap();     // (re)load default ledmap | ||||
|   deserializeMap();     // (re)load default ledmap (will also setUpMatrix() if ledmap does not exist) | ||||
| } | ||||
|  | ||||
| void WS2812FX::service() { | ||||
|   unsigned long nowUp = millis(); // Be aware, millis() rolls over every 49 days | ||||
|   now = nowUp + timebase; | ||||
|   if (nowUp - _lastShow < MIN_SHOW_DELAY) return; | ||||
|   if (nowUp - _lastShow < MIN_SHOW_DELAY || _suspend) return; | ||||
|   bool doShow = false; | ||||
|  | ||||
|   _isServicing = true; | ||||
|   _segment_index = 0; | ||||
|   Segment::handleRandomPalette(); // move it into for loop when each segment has individual random palette | ||||
|  | ||||
|   for (segment &seg : _segments) { | ||||
|     if (_suspend) return; // immediately stop processing segments if suspend requested during service() | ||||
|  | ||||
|     // process transition (mode changes in the middle of transition) | ||||
|     seg.handleTransition(); | ||||
|     // reset the segment runtime data if needed | ||||
| @@ -1183,15 +1162,12 @@ void WS2812FX::service() { | ||||
|       uint16_t delay = FRAMETIME; | ||||
|  | ||||
|       if (!seg.freeze) { //only run effect function if not frozen | ||||
|         _virtualSegmentLength = seg.virtualLength(); | ||||
|         _colors_t[0] = seg.currentColor(0); | ||||
|         _colors_t[1] = seg.currentColor(1); | ||||
|         _colors_t[2] = seg.currentColor(2); | ||||
|         _virtualSegmentLength = seg.virtualLength(); //SEGLEN | ||||
|         _colors_t[0] = gamma32(seg.currentColor(0)); | ||||
|         _colors_t[1] = gamma32(seg.currentColor(1)); | ||||
|         _colors_t[2] = gamma32(seg.currentColor(2)); | ||||
|         seg.currentPalette(_currentPalette, seg.palette); // we need to pass reference | ||||
|  | ||||
|         if (!cctFromRgb || correctWB) busses.setSegmentCCT(seg.currentBri(true), correctWB); | ||||
|         for (int c = 0; c < NUM_COLORS; c++) _colors_t[c] = gamma32(_colors_t[c]); | ||||
|  | ||||
|         if (!cctFromRgb || correctWB) BusManager::setSegmentCCT(seg.currentBri(true), correctWB); | ||||
|         // Effect blending | ||||
|         // When two effects are being blended, each may have different segment data, this | ||||
|         // data needs to be saved first and then restored before running previous mode. | ||||
| @@ -1212,17 +1188,17 @@ void WS2812FX::service() { | ||||
|           Segment::modeBlend(false);          // unset semaphore | ||||
|         } | ||||
| #endif | ||||
|         if (seg.mode != FX_MODE_HALLOWEEN_EYES) seg.call++; | ||||
|         seg.call++; | ||||
|         if (seg.isInTransition() && delay > FRAMETIME) delay = FRAMETIME; // force faster updates during transition | ||||
|       } | ||||
|  | ||||
|       seg.next_time = nowUp + delay; | ||||
|     } | ||||
|     if (_segment_index == _queuedChangesSegId) setUpSegmentFromQueuedChanges(); | ||||
| //    if (_segment_index == _queuedChangesSegId) setUpSegmentFromQueuedChanges(); | ||||
|     _segment_index++; | ||||
|   } | ||||
|   _virtualSegmentLength = 0; | ||||
|   busses.setSegmentCCT(-1); | ||||
|   BusManager::setSegmentCCT(-1); | ||||
|   _isServicing = false; | ||||
|   _triggered = false; | ||||
|  | ||||
| @@ -1231,6 +1207,7 @@ void WS2812FX::service() { | ||||
|   #endif | ||||
|   if (doShow) { | ||||
|     yield(); | ||||
|     Segment::handleRandomPalette(); // slowly transtion random palette; move it into for loop when each segment has individual random palette | ||||
|     show(); | ||||
|   } | ||||
|   #ifdef WLED_DEBUG | ||||
| @@ -1238,98 +1215,16 @@ void WS2812FX::service() { | ||||
|   #endif | ||||
| } | ||||
|  | ||||
| void IRAM_ATTR WS2812FX::setPixelColor(int i, uint32_t col) | ||||
| { | ||||
|   if (i < customMappingSize) i = customMappingTable[i]; | ||||
| void IRAM_ATTR WS2812FX::setPixelColor(unsigned i, uint32_t col) { | ||||
|   i = getMappedPixelIndex(i); | ||||
|   if (i >= _length) return; | ||||
|   busses.setPixelColor(i, col); | ||||
|   BusManager::setPixelColor(i, col); | ||||
| } | ||||
|  | ||||
| uint32_t WS2812FX::getPixelColor(uint16_t i) | ||||
| { | ||||
|   if (i < customMappingSize) i = customMappingTable[i]; | ||||
| uint32_t IRAM_ATTR WS2812FX::getPixelColor(uint16_t i) { | ||||
|   i = getMappedPixelIndex(i); | ||||
|   if (i >= _length) return 0; | ||||
|   return busses.getPixelColor(i); | ||||
| } | ||||
|  | ||||
|  | ||||
| //DISCLAIMER | ||||
| //The following function attemps to calculate the current LED power usage, | ||||
| //and will limit the brightness to stay below a set amperage threshold. | ||||
| //It is NOT a measurement and NOT guaranteed to stay within the ablMilliampsMax margin. | ||||
| //Stay safe with high amperage and have a reasonable safety margin! | ||||
| //I am NOT to be held liable for burned down garages! | ||||
|  | ||||
| //fine tune power estimation constants for your setup | ||||
| #define MA_FOR_ESP        100 //how much mA does the ESP use (Wemos D1 about 80mA, ESP32 about 120mA) | ||||
|                               //you can set it to 0 if the ESP is powered by USB and the LEDs by external | ||||
|  | ||||
| uint8_t WS2812FX::estimateCurrentAndLimitBri() { | ||||
|   //power limit calculation | ||||
|   //each LED can draw up 195075 "power units" (approx. 53mA) | ||||
|   //one PU is the power it takes to have 1 channel 1 step brighter per brightness step | ||||
|   //so A=2,R=255,G=0,B=0 would use 510 PU per LED (1mA is about 3700 PU) | ||||
|   bool useWackyWS2815PowerModel = false; | ||||
|   byte actualMilliampsPerLed = milliampsPerLed; | ||||
|  | ||||
|   if (ablMilliampsMax < 150 || actualMilliampsPerLed == 0) { //0 mA per LED and too low numbers turn off calculation | ||||
|     currentMilliamps = 0; | ||||
|     return _brightness; | ||||
|   } | ||||
|  | ||||
|   if (milliampsPerLed == 255) { | ||||
|     useWackyWS2815PowerModel = true; | ||||
|     actualMilliampsPerLed = 12; // from testing an actual strip | ||||
|   } | ||||
|  | ||||
|   size_t powerBudget = (ablMilliampsMax - MA_FOR_ESP); //100mA for ESP power | ||||
|  | ||||
|   size_t pLen = 0; //getLengthPhysical(); | ||||
|   size_t powerSum = 0; | ||||
|   for (uint_fast8_t bNum = 0; bNum < busses.getNumBusses(); bNum++) { | ||||
|     Bus *bus = busses.getBus(bNum); | ||||
|     if (!IS_DIGITAL(bus->getType())) continue; //exclude non-digital network busses | ||||
|     uint16_t len = bus->getLength(); | ||||
|     pLen += len; | ||||
|     uint32_t busPowerSum = 0; | ||||
|     for (uint_fast16_t i = 0; i < len; i++) { //sum up the usage of each LED | ||||
|       uint32_t c = bus->getPixelColor(i); // always returns original or restored color without brightness scaling | ||||
|       byte r = R(c), g = G(c), b = B(c), w = W(c); | ||||
|  | ||||
|       if(useWackyWS2815PowerModel) { //ignore white component on WS2815 power calculation | ||||
|         busPowerSum += (MAX(MAX(r,g),b)) * 3; | ||||
|       } else { | ||||
|         busPowerSum += (r + g + b + w); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if (bus->hasWhite()) { //RGBW led total output with white LEDs enabled is still 50mA, so each channel uses less | ||||
|       busPowerSum *= 3; | ||||
|       busPowerSum >>= 2; //same as /= 4 | ||||
|     } | ||||
|     powerSum += busPowerSum; | ||||
|   } | ||||
|  | ||||
|   if (powerBudget > pLen) { //each LED uses about 1mA in standby, exclude that from power budget | ||||
|     powerBudget -= pLen; | ||||
|   } else { | ||||
|     powerBudget = 0; | ||||
|   } | ||||
|  | ||||
|   // powerSum has all the values of channels summed (max would be pLen*765 as white is excluded) so convert to milliAmps | ||||
|   powerSum = (powerSum * actualMilliampsPerLed) / 765; | ||||
|  | ||||
|   uint8_t newBri = _brightness; | ||||
|   if (powerSum * _brightness / 255 > powerBudget) { //scale brightness down to stay in current limit | ||||
|     float scale = (float)(powerBudget * 255) / (float)(powerSum * _brightness); | ||||
|     uint16_t scaleI = scale * 255; | ||||
|     uint8_t scaleB = (scaleI > 255) ? 255 : scaleI; | ||||
|     newBri = scale8(_brightness, scaleB) + 1; | ||||
|   } | ||||
|   currentMilliamps = (powerSum * newBri) / 255; | ||||
|   currentMilliamps += MA_FOR_ESP; //add power of ESP back to estimate | ||||
|   currentMilliamps += pLen; //add standby power (1mA/LED) back to estimate | ||||
|   return newBri; | ||||
|   return BusManager::getPixelColor(i); | ||||
| } | ||||
|  | ||||
| void WS2812FX::show(void) { | ||||
| @@ -1337,18 +1232,10 @@ void WS2812FX::show(void) { | ||||
|   show_callback callback = _callback; | ||||
|   if (callback) callback(); | ||||
|  | ||||
|   uint8_t newBri = estimateCurrentAndLimitBri(); | ||||
|   busses.setBrightness(newBri); // "repaints" all pixels if brightness changed | ||||
|  | ||||
|   // some buses send asynchronously and this method will return before | ||||
|   // all of the data has been sent. | ||||
|   // See https://github.com/Makuna/NeoPixelBus/wiki/ESP32-NeoMethods#neoesp32rmt-methods | ||||
|   busses.show(); | ||||
|  | ||||
|   // restore bus brightness to its original value | ||||
|   // this is done right after show, so this is only OK if LED updates are completed before show() returns | ||||
|   // or async show has a separate buffer (ESP32 RMT and I2S are ok) | ||||
|   if (newBri < _brightness) busses.setBrightness(_brightness); | ||||
|   BusManager::show(); | ||||
|  | ||||
|   unsigned long showNow = millis(); | ||||
|   size_t diff = showNow - _lastShow; | ||||
| @@ -1363,7 +1250,7 @@ void WS2812FX::show(void) { | ||||
|  * On some hardware (ESP32), strip updates are done asynchronously. | ||||
|  */ | ||||
| bool WS2812FX::isUpdating() { | ||||
|   return !busses.canAllShow(); | ||||
|   return !BusManager::canAllShow(); | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -1422,7 +1309,7 @@ void WS2812FX::setBrightness(uint8_t b, bool direct) { | ||||
|   } | ||||
|   // setting brightness with NeoPixelBusLg has no effect on already painted pixels, | ||||
|   // so we need to force an update to existing buffer | ||||
|   busses.setBrightness(b); | ||||
|   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 | ||||
| @@ -1437,8 +1324,7 @@ uint8_t WS2812FX::getActiveSegsLightCapabilities(bool selectedOnly) { | ||||
|   return totalLC; | ||||
| } | ||||
|  | ||||
| uint8_t WS2812FX::getFirstSelectedSegId(void) | ||||
| { | ||||
| uint8_t WS2812FX::getFirstSelectedSegId(void) { | ||||
|   size_t i = 0; | ||||
|   for (segment &seg : _segments) { | ||||
|     if (seg.isActive() && seg.isSelected()) return i; | ||||
| @@ -1479,8 +1365,8 @@ uint16_t WS2812FX::getLengthTotal(void) { | ||||
|  | ||||
| uint16_t WS2812FX::getLengthPhysical(void) { | ||||
|   uint16_t len = 0; | ||||
|   for (size_t b = 0; b < busses.getNumBusses(); b++) { | ||||
|     Bus *bus = busses.getBus(b); | ||||
|   for (size_t b = 0; b < BusManager::getNumBusses(); b++) { | ||||
|     Bus *bus = BusManager::getBus(b); | ||||
|     if (bus->getType() >= TYPE_NET_DDP_RGB) continue; //exclude non-physical network busses | ||||
|     len += bus->getLength(); | ||||
|   } | ||||
| @@ -1491,8 +1377,8 @@ uint16_t WS2812FX::getLengthPhysical(void) { | ||||
| //returns if there is an RGBW bus (supports RGB and White, not only white) | ||||
| //not influenced by auto-white mode, also true if white slider does not affect output white channel | ||||
| bool WS2812FX::hasRGBWBus(void) { | ||||
|   for (size_t b = 0; b < busses.getNumBusses(); b++) { | ||||
|     Bus *bus = busses.getBus(b); | ||||
|   for (size_t b = 0; b < BusManager::getNumBusses(); b++) { | ||||
|     Bus *bus = BusManager::getBus(b); | ||||
|     if (bus == nullptr || bus->getLength()==0) break; | ||||
|     if (bus->hasRGB() && bus->hasWhite()) return true; | ||||
|   } | ||||
| @@ -1501,8 +1387,8 @@ bool WS2812FX::hasRGBWBus(void) { | ||||
|  | ||||
| bool WS2812FX::hasCCTBus(void) { | ||||
|   if (cctFromRgb && !correctWB) return false; | ||||
|   for (size_t b = 0; b < busses.getNumBusses(); b++) { | ||||
|     Bus *bus = busses.getBus(b); | ||||
|   for (size_t b = 0; b < BusManager::getNumBusses(); b++) { | ||||
|     Bus *bus = BusManager::getBus(b); | ||||
|     if (bus == nullptr || bus->getLength()==0) break; | ||||
|     switch (bus->getType()) { | ||||
|       case TYPE_ANALOG_5CH: | ||||
| @@ -1513,18 +1399,18 @@ bool WS2812FX::hasCCTBus(void) { | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| void WS2812FX::purgeSegments(bool force) { | ||||
| void WS2812FX::purgeSegments() { | ||||
|   // remove all inactive segments (from the back) | ||||
|   int deleted = 0; | ||||
|   if (_segments.size() <= 1) return; | ||||
|   for (size_t i = _segments.size()-1; i > 0; i--) | ||||
|     if (_segments[i].stop == 0 || force) { | ||||
|     if (_segments[i].stop == 0) { | ||||
|       deleted++; | ||||
|       _segments.erase(_segments.begin() + i); | ||||
|     } | ||||
|   if (deleted) { | ||||
|     _segments.shrink_to_fit(); | ||||
|     /*if (_mainSegment >= _segments.size())*/ setMainSegmentId(0); | ||||
|     setMainSegmentId(0); | ||||
|   } | ||||
| } | ||||
|  | ||||
| @@ -1539,7 +1425,7 @@ void WS2812FX::setSegment(uint8_t segId, uint16_t i1, uint16_t i2, uint8_t group | ||||
|     appendSegment(Segment(0, strip.getLengthTotal())); | ||||
|     segId = getSegmentsNum()-1; // segments are added at the end of list | ||||
|   } | ||||
|  | ||||
| /* | ||||
|   if (_queuedChangesSegId == segId) _queuedChangesSegId = 255; // cancel queued change if already queued for this segment | ||||
|  | ||||
|   if (segId < getMaxSegments() && segId == getCurrSegmentId() && isServicing()) { // queue change to prevent concurrent access | ||||
| @@ -1551,21 +1437,19 @@ void WS2812FX::setSegment(uint8_t segId, uint16_t i1, uint16_t i2, uint8_t group | ||||
|     DEBUG_PRINT(F("Segment queued: ")); DEBUG_PRINTLN(segId); | ||||
|     return; // queued changes are applied immediately after effect function returns | ||||
|   } | ||||
|    | ||||
| */ | ||||
|   suspend(); | ||||
|   _segments[segId].setUp(i1, i2, grouping, spacing, offset, startY, stopY); | ||||
|   resume(); | ||||
|   if (segId > 0 && segId == getSegmentsNum()-1 && i2 <= i1) _segments.pop_back(); // if last segment was deleted remove it from vector | ||||
| } | ||||
|  | ||||
| /* | ||||
| void WS2812FX::setUpSegmentFromQueuedChanges() { | ||||
|   if (_queuedChangesSegId >= getSegmentsNum()) return; | ||||
|   getSegment(_queuedChangesSegId).setUp(_qStart, _qStop, _qGrouping, _qSpacing, _qOffset, _qStartY, _qStopY); | ||||
|   _segments[_queuedChangesSegId].setUp(_qStart, _qStop, _qGrouping, _qSpacing, _qOffset, _qStartY, _qStopY); | ||||
|   _queuedChangesSegId = 255; | ||||
| } | ||||
|  | ||||
| void WS2812FX::restartRuntime() { | ||||
|   for (segment &seg : _segments) seg.markForReset(); | ||||
| } | ||||
|  | ||||
| */ | ||||
| void WS2812FX::resetSegments() { | ||||
|   _segments.clear(); // destructs all Segment as part of clearing | ||||
|   #ifndef WLED_DISABLE_2D | ||||
| @@ -1574,6 +1458,7 @@ void WS2812FX::resetSegments() { | ||||
|   segment seg = Segment(0, _length); | ||||
|   #endif | ||||
|   _segments.push_back(seg); | ||||
|   _segments.shrink_to_fit(); // just in case ... | ||||
|   _mainSegment = 0; | ||||
| } | ||||
|  | ||||
| @@ -1592,8 +1477,8 @@ void WS2812FX::makeAutoSegments(bool forceReset) { | ||||
|     } | ||||
|     #endif | ||||
|  | ||||
|     for (size_t i = s; i < busses.getNumBusses(); i++) { | ||||
|       Bus* b = busses.getBus(i); | ||||
|     for (size_t i = s; i < BusManager::getNumBusses(); i++) { | ||||
|       Bus* b = BusManager::getBus(i); | ||||
|  | ||||
|       segStarts[s] = b->getStart(); | ||||
|       segStops[s]  = segStarts[s] + b->getLength(); | ||||
| @@ -1682,8 +1567,8 @@ void WS2812FX::fixInvalidSegments() { | ||||
| bool WS2812FX::checkSegmentAlignment() { | ||||
|   bool aligned = false; | ||||
|   for (segment &seg : _segments) { | ||||
|     for (unsigned b = 0; b<busses.getNumBusses(); b++) { | ||||
|       Bus *bus = busses.getBus(b); | ||||
|     for (unsigned b = 0; b<BusManager::getNumBusses(); b++) { | ||||
|       Bus *bus = BusManager::getBus(b); | ||||
|       if (seg.start == bus->getStart() && seg.stop == bus->getStart() + bus->getLength()) aligned = true; | ||||
|     } | ||||
|     if (seg.start == 0 && seg.stop == _length) aligned = true; | ||||
| @@ -1692,37 +1577,22 @@ bool WS2812FX::checkSegmentAlignment() { | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| //After this function is called, setPixelColor() will use that segment (offsets, grouping, ... will apply) | ||||
| //Note: If called in an interrupt (e.g. JSON API), original segment must be restored, | ||||
| //otherwise it can lead to a crash on ESP32 because _segment_index is modified while in use by the main thread | ||||
| uint8_t WS2812FX::setPixelSegment(uint8_t n) { | ||||
|   uint8_t prevSegId = _segment_index; | ||||
|   if (n < _segments.size()) { | ||||
|     _segment_index = n; | ||||
|     _virtualSegmentLength = _segments[_segment_index].virtualLength(); | ||||
|   } | ||||
|   return prevSegId; | ||||
| } | ||||
|  | ||||
| // used by analog clock overlay | ||||
| void WS2812FX::setRange(uint16_t i, uint16_t i2, uint32_t col) { | ||||
|   if (i2 < i) std::swap(i,i2); | ||||
|   for (unsigned x = i; x <= i2; x++) setPixelColor(x, col); | ||||
| } | ||||
|  | ||||
| void WS2812FX::setTransitionMode(bool t) { | ||||
|   for (segment &seg : _segments) seg.startTransition(t ? _transitionDur : 0); | ||||
| } | ||||
|  | ||||
| #ifdef WLED_DEBUG | ||||
| void WS2812FX::printSize() { | ||||
|   size_t size = 0; | ||||
|   for (const Segment &seg : _segments) size += seg.getSize(); | ||||
|   DEBUG_PRINTF("Segments: %d -> %uB\n", _segments.size(), size); | ||||
|   DEBUG_PRINTF("Modes: %d*%d=%uB\n", sizeof(mode_ptr), _mode.size(), (_mode.capacity()*sizeof(mode_ptr))); | ||||
|   DEBUG_PRINTF("Data: %d*%d=%uB\n", sizeof(const char *), _modeData.size(), (_modeData.capacity()*sizeof(const char *))); | ||||
|   DEBUG_PRINTF("Map: %d*%d=%uB\n", sizeof(uint16_t), (int)customMappingSize, customMappingSize*sizeof(uint16_t)); | ||||
|   DEBUG_PRINTF_P(PSTR("Segments: %d -> %uB\n"), _segments.size(), size); | ||||
|   DEBUG_PRINTF_P(PSTR("Modes: %d*%d=%uB\n"), sizeof(mode_ptr), _mode.size(), (_mode.capacity()*sizeof(mode_ptr))); | ||||
|   DEBUG_PRINTF_P(PSTR("Data: %d*%d=%uB\n"), sizeof(const char *), _modeData.size(), (_modeData.capacity()*sizeof(const char *))); | ||||
|   DEBUG_PRINTF_P(PSTR("Map: %d*%d=%uB\n"), sizeof(uint16_t), (int)customMappingSize, customMappingSize*sizeof(uint16_t)); | ||||
|   size = getLengthTotal(); | ||||
|   if (useGlobalLedBuffer) DEBUG_PRINTF("Buffer: %d*%u=%uB\n", sizeof(CRGB), size, size*sizeof(CRGB)); | ||||
|   if (useGlobalLedBuffer) DEBUG_PRINTF_P(PSTR("Buffer: %d*%u=%uB\n"), sizeof(CRGB), size, size*sizeof(CRGB)); | ||||
| } | ||||
| #endif | ||||
|  | ||||
| @@ -1751,7 +1621,7 @@ void WS2812FX::loadCustomPalettes() { | ||||
|               tcp[ j ] = (uint8_t) pal[ i ].as<int>(); // index | ||||
|               colorFromHexString(rgbw, pal[i+1].as<const char *>()); // will catch non-string entires | ||||
|               for (size_t c=0; c<3; c++) tcp[j+1+c] = gamma8(rgbw[c]); // only use RGB component | ||||
|               DEBUG_PRINTF("%d(%d) : %d %d %d\n", i, int(tcp[j]), int(tcp[j+1]), int(tcp[j+2]), int(tcp[j+3])); | ||||
|               DEBUG_PRINTF_P(PSTR("%d(%d) : %d %d %d\n"), i, int(tcp[j]), int(tcp[j+1]), int(tcp[j+2]), int(tcp[j+3])); | ||||
|             } | ||||
|           } else { | ||||
|             size_t palSize = MIN(pal.size(), 72); | ||||
| @@ -1761,7 +1631,7 @@ void WS2812FX::loadCustomPalettes() { | ||||
|               tcp[i+1] = gamma8((uint8_t) pal[i+1].as<int>()); // R | ||||
|               tcp[i+2] = gamma8((uint8_t) pal[i+2].as<int>()); // G | ||||
|               tcp[i+3] = gamma8((uint8_t) pal[i+3].as<int>()); // B | ||||
|               DEBUG_PRINTF("%d(%d) : %d %d %d\n", i, int(tcp[i]), int(tcp[i+1]), int(tcp[i+2]), int(tcp[i+3])); | ||||
|               DEBUG_PRINTF_P(PSTR("%d(%d) : %d %d %d\n"), i, int(tcp[i]), int(tcp[i+1]), int(tcp[i+2]), int(tcp[i+3])); | ||||
|             } | ||||
|           } | ||||
|           customPalettes.push_back(targetPalette.loadDynamicGradientPalette(tcp)); | ||||
| @@ -1785,46 +1655,44 @@ bool WS2812FX::deserializeMap(uint8_t n) { | ||||
|   strcat_P(fileName, PSTR(".json")); | ||||
|   bool isFile = WLED_FS.exists(fileName); | ||||
|  | ||||
|   if (!isFile) { | ||||
|     // erase custom mapping if selecting nonexistent ledmap.json (n==0) | ||||
|     if (!isMatrix && !n && customMappingTable != nullptr) { | ||||
|       customMappingSize = 0; | ||||
|       delete[] customMappingTable; | ||||
|       customMappingTable = nullptr; | ||||
|     } | ||||
|   customMappingSize = 0; // prevent use of mapping if anything goes wrong | ||||
|  | ||||
|   if (!isFile && n==0 && isMatrix) { | ||||
|     setUpMatrix(); | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   if (!requestJSONBufferLock(7)) return false; | ||||
|   if (!isFile || !requestJSONBufferLock(7)) return false; // this will trigger setUpMatrix() when called from wled.cpp | ||||
|  | ||||
|   if (!readObjectFromFile(fileName, nullptr, &doc)) { | ||||
|   if (!readObjectFromFile(fileName, nullptr, pDoc)) { | ||||
|     DEBUG_PRINT(F("ERROR Invalid ledmap in ")); DEBUG_PRINTLN(fileName); | ||||
|     releaseJSONBufferLock(); | ||||
|     return false; //if file does not exist just exit | ||||
|     return false; // if file does not load properly then exit | ||||
|   } | ||||
|  | ||||
|   DEBUG_PRINT(F("Reading LED map from ")); | ||||
|   DEBUG_PRINTLN(fileName); | ||||
|   DEBUG_PRINT(F("Reading LED map from ")); DEBUG_PRINTLN(fileName); | ||||
|  | ||||
|   // erase old custom ledmap | ||||
|   if (customMappingTable != nullptr) { | ||||
|     customMappingSize = 0; | ||||
|     delete[] customMappingTable; | ||||
|     customMappingTable = nullptr; | ||||
|   } | ||||
|   if (customMappingTable == nullptr) customMappingTable = new uint16_t[getLengthTotal()]; | ||||
|  | ||||
|   JsonArray map = doc[F("map")]; | ||||
|   JsonObject root = pDoc->as<JsonObject>(); | ||||
|   JsonArray map = root[F("map")]; | ||||
|   if (!map.isNull() && map.size()) {  // not an empty map | ||||
|     customMappingSize  = map.size(); | ||||
|     customMappingTable = new uint16_t[customMappingSize]; | ||||
|     for (unsigned i=0; i<customMappingSize; i++) { | ||||
|       customMappingTable[i] = (uint16_t) (map[i]<0 ? 0xFFFFU : map[i]); | ||||
|     } | ||||
|     customMappingSize = min((unsigned)map.size(), (unsigned)getLengthTotal()); | ||||
|     for (unsigned i=0; i<customMappingSize; i++) customMappingTable[i] = (uint16_t) (map[i]<0 ? 0xFFFFU : map[i]); | ||||
|   } | ||||
|  | ||||
|   releaseJSONBufferLock(); | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| uint16_t IRAM_ATTR WS2812FX::getMappedPixelIndex(uint16_t index) { | ||||
|   // convert logical address to physical | ||||
|   if (index < customMappingSize | ||||
|     && (realtimeMode == REALTIME_MODE_INACTIVE || realtimeRespectLedMaps)) index = customMappingTable[index]; | ||||
|  | ||||
|   return index; | ||||
| } | ||||
|  | ||||
|  | ||||
| WS2812FX* WS2812FX::instance = nullptr; | ||||
|  | ||||
|   | ||||
| @@ -12,7 +12,6 @@ | ||||
| //colors.cpp | ||||
| uint32_t colorBalanceFromKelvin(uint16_t kelvin, uint32_t rgb); | ||||
| uint16_t approximateKelvinFromRGB(uint32_t rgb); | ||||
| void colorRGBtoRGBW(byte* rgb); | ||||
|  | ||||
| //udp.cpp | ||||
| uint8_t realtimeBroadcast(uint8_t type, IPAddress client, uint16_t length, byte *buffer, uint8_t bri=255, bool isRGBW=false); | ||||
| @@ -32,10 +31,12 @@ uint8_t realtimeBroadcast(uint8_t type, IPAddress client, uint16_t length, byte | ||||
|   #define DEBUG_PRINT(x) DEBUGOUT.print(x) | ||||
|   #define DEBUG_PRINTLN(x) DEBUGOUT.println(x) | ||||
|   #define DEBUG_PRINTF(x...) DEBUGOUT.printf(x) | ||||
|   #define DEBUG_PRINTF_P(x...) DEBUGOUT.printf_P(x) | ||||
| #else | ||||
|   #define DEBUG_PRINT(x) | ||||
|   #define DEBUG_PRINTLN(x) | ||||
|   #define DEBUG_PRINTF(x...) | ||||
|   #define DEBUG_PRINTF_P(x...) | ||||
| #endif | ||||
|  | ||||
| //color mangling macros | ||||
| @@ -53,7 +54,8 @@ void ColorOrderMap::add(uint16_t start, uint16_t len, uint8_t colorOrder) { | ||||
|   if (len == 0) { | ||||
|     return; | ||||
|   } | ||||
|   if (colorOrder > COL_ORDER_MAX) { | ||||
|   // upper nibble contains W swap information | ||||
|   if ((colorOrder & 0x0F) > COL_ORDER_MAX) { | ||||
|     return; | ||||
|   } | ||||
|   _mappings[_count].start = start; | ||||
| @@ -63,12 +65,13 @@ void ColorOrderMap::add(uint16_t start, uint16_t len, uint8_t colorOrder) { | ||||
| } | ||||
|  | ||||
| uint8_t IRAM_ATTR ColorOrderMap::getPixelColorOrder(uint16_t pix, uint8_t defaultColorOrder) const { | ||||
|   if (_count == 0) return defaultColorOrder; | ||||
|   // upper nibble contains W swap information | ||||
|   uint8_t swapW = defaultColorOrder >> 4; | ||||
|   for (uint8_t i = 0; i < _count; i++) { | ||||
|     if (pix >= _mappings[i].start && pix < (_mappings[i].start + _mappings[i].len)) { | ||||
|       return _mappings[i].colorOrder | (swapW << 4); | ||||
|   if (_count > 0) { | ||||
|     // upper nibble contains W swap information | ||||
|     // when ColorOrderMap's upper nibble contains value >0 then swap information is used from it, otherwise global swap is used | ||||
|     for (unsigned i = 0; i < _count; i++) { | ||||
|       if (pix >= _mappings[i].start && pix < (_mappings[i].start + _mappings[i].len)) { | ||||
|         return _mappings[i].colorOrder | ((_mappings[i].colorOrder >> 4) ? 0 : (defaultColorOrder & 0xF0)); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   return defaultColorOrder; | ||||
| @@ -77,7 +80,7 @@ uint8_t IRAM_ATTR ColorOrderMap::getPixelColorOrder(uint16_t pix, uint8_t defaul | ||||
|  | ||||
| uint32_t Bus::autoWhiteCalc(uint32_t c) { | ||||
|   uint8_t aWM = _autoWhiteMode; | ||||
|   if (_gAWM != AW_GLOBAL_DISABLED) aWM = _gAWM; | ||||
|   if (_gAWM < AW_GLOBAL_DISABLED) aWM = _gAWM; | ||||
|   if (aWM == RGBW_MODE_MANUAL_ONLY) return c; | ||||
|   uint8_t w = W(c); | ||||
|   //ignore auto-white calculation if w>0 and mode DUAL (DUAL behaves as BRIGHTER if w==0) | ||||
| @@ -101,6 +104,8 @@ BusDigital::BusDigital(BusConfig &bc, uint8_t nr, const ColorOrderMap &com) | ||||
| : Bus(bc.type, bc.start, bc.autoWhite, bc.count, bc.reversed, (bc.refreshReq || bc.type == TYPE_TM1814)) | ||||
| , _skip(bc.skipAmount) //sacrificial pixels | ||||
| , _colorOrder(bc.colorOrder) | ||||
| , _milliAmpsPerLed(bc.milliAmpsPerLed) | ||||
| , _milliAmpsMax(bc.milliAmpsMax) | ||||
| , _colorOrderMap(com) | ||||
| { | ||||
|   if (!IS_DIGITAL(bc.type) || !bc.count) return; | ||||
| @@ -118,17 +123,92 @@ BusDigital::BusDigital(BusConfig &bc, uint8_t nr, const ColorOrderMap &com) | ||||
|   _iType = PolyBus::getI(bc.type, _pins, nr); | ||||
|   if (_iType == I_NONE) return; | ||||
|   if (bc.doubleBuffer && !allocData(bc.count * (Bus::hasWhite(_type) + 3*Bus::hasRGB(_type)))) return; //warning: hardcoded channel count | ||||
|   _buffering = bc.doubleBuffer; | ||||
|   //_buffering = bc.doubleBuffer; | ||||
|   uint16_t lenToCreate = bc.count; | ||||
|   if (bc.type == TYPE_WS2812_1CH_X3) lenToCreate = NUM_ICS_WS2812_1CH_3X(bc.count); // only needs a third of "RGB" LEDs for NeoPixelBus | ||||
|   _busPtr = PolyBus::create(_iType, _pins, lenToCreate + _skip, nr, _frequencykHz); | ||||
|   _valid = (_busPtr != nullptr); | ||||
|   DEBUG_PRINTF("%successfully inited strip %u (len %u) with type %u and pins %u,%u (itype %u)\n", _valid?"S":"Uns", nr, bc.count, bc.type, _pins[0], _pins[1], _iType); | ||||
|   DEBUG_PRINTF("%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], _pins[1], _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. | ||||
| //It is NOT a measurement and NOT guaranteed to stay within the ablMilliampsMax margin. | ||||
| //Stay safe with high amperage and have a reasonable safety margin! | ||||
| //I am NOT to be held liable for burned down garages or houses! | ||||
|  | ||||
| // To disable brightness limiter we either set output max current to 0 or single LED current to 0 | ||||
| uint8_t BusDigital::estimateCurrentAndLimitBri() { | ||||
|   bool useWackyWS2815PowerModel = false; | ||||
|   byte actualMilliampsPerLed = _milliAmpsPerLed; | ||||
|  | ||||
|   if (_milliAmpsMax < MA_FOR_ESP/BusManager::getNumBusses() || actualMilliampsPerLed == 0) { //0 mA per LED and too low numbers turn off calculation | ||||
|     return _bri; | ||||
|   } | ||||
|  | ||||
|   if (_milliAmpsPerLed == 255) { | ||||
|     useWackyWS2815PowerModel = true; | ||||
|     actualMilliampsPerLed = 12; // from testing an actual strip | ||||
|   } | ||||
|  | ||||
|   size_t powerBudget = (_milliAmpsMax - MA_FOR_ESP/BusManager::getNumBusses()); //80/120mA for ESP power | ||||
|   if (powerBudget > getLength()) { //each LED uses about 1mA in standby, exclude that from power budget | ||||
|     powerBudget -= getLength(); | ||||
|   } else { | ||||
|     powerBudget = 0; | ||||
|   } | ||||
|  | ||||
|   uint32_t busPowerSum = 0; | ||||
|   for (unsigned i = 0; i < getLength(); i++) {  //sum up the usage of each LED | ||||
|     uint32_t c = getPixelColor(i); // always returns original or restored color without brightness scaling | ||||
|     byte r = R(c), g = G(c), b = B(c), w = W(c); | ||||
|  | ||||
|     if (useWackyWS2815PowerModel) { //ignore white component on WS2815 power calculation | ||||
|       busPowerSum += (max(max(r,g),b)) * 3; | ||||
|     } else { | ||||
|       busPowerSum += (r + g + b + w); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (hasWhite()) { //RGBW led total output with white LEDs enabled is still 50mA, so each channel uses less | ||||
|     busPowerSum *= 3; | ||||
|     busPowerSum >>= 2; //same as /= 4 | ||||
|   } | ||||
|  | ||||
|   // powerSum has all the values of channels summed (max would be getLength()*765 as white is excluded) so convert to milliAmps | ||||
|   busPowerSum = (busPowerSum * actualMilliampsPerLed) / 765; | ||||
|   _milliAmpsTotal = busPowerSum * _bri / 255; | ||||
|  | ||||
|   uint8_t newBri = _bri; | ||||
|   if (busPowerSum * _bri / 255 > powerBudget) { //scale brightness down to stay in current limit | ||||
|     float scale = (float)(powerBudget * 255) / (float)(busPowerSum * _bri); | ||||
|     if (scale >= 1.0f) return _bri; | ||||
|     _milliAmpsTotal = ceilf((float)_milliAmpsTotal * scale); | ||||
|     uint8_t scaleB = min((int)(scale * 255), 255); | ||||
|     newBri = unsigned(_bri * scaleB) / 256 + 1; | ||||
|   } | ||||
|   return newBri; | ||||
| } | ||||
|  | ||||
| void BusDigital::show() { | ||||
|   _milliAmpsTotal = 0; | ||||
|   if (!_valid) return; | ||||
|   if (_buffering) { // should be _data != nullptr, but that causes ~20% FPS drop | ||||
|  | ||||
|   uint8_t newBri = estimateCurrentAndLimitBri();  // will fill _milliAmpsTotal | ||||
|   if (newBri < _bri) PolyBus::setBrightness(_busPtr, _iType, newBri); // limit brightness to stay within current limits | ||||
|  | ||||
|   if (_data) { // use _buffering this causes ~20% FPS drop | ||||
|     size_t channels = Bus::hasWhite(_type) + 3*Bus::hasRGB(_type); | ||||
|     for (size_t i=0; i<_len; i++) { | ||||
|       size_t offset = i*channels; | ||||
| @@ -152,8 +232,22 @@ void BusDigital::show() { | ||||
|     if (_skip) PolyBus::setPixelColor(_busPtr, _iType, 0, 0, _colorOrderMap.getPixelColorOrder(_start, _colorOrder)); // paint skipped pixels black | ||||
|     #endif | ||||
|     for (int i=1; i<_skip; i++) PolyBus::setPixelColor(_busPtr, _iType, i, 0, _colorOrderMap.getPixelColorOrder(_start, _colorOrder)); // paint skipped pixels black | ||||
|   } else { | ||||
|     if (newBri < _bri) { | ||||
|       uint16_t hwLen = _len; | ||||
|       if (_type == TYPE_WS2812_1CH_X3) hwLen = NUM_ICS_WS2812_1CH_3X(_len); // only needs a third of "RGB" LEDs for NeoPixelBus | ||||
|       for (unsigned i = 0; i < hwLen; i++) { | ||||
|         // use 0 as color order, actual order does not matter here as we just update the channel values as-is | ||||
|         uint32_t c = restoreColorLossy(PolyBus::getPixelColor(_busPtr, _iType, i, 0), _bri); | ||||
|         PolyBus::setPixelColor(_busPtr, _iType, i, c, 0); // repaint all pixels with new brightness | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   PolyBus::show(_busPtr, _iType, !_buffering); // faster if buffer consistency is not important | ||||
|   PolyBus::show(_busPtr, _iType, !_data); // faster if buffer consistency is not important (use !_buffering this causes 20% FPS drop) | ||||
|   // restore bus brightness to its original value | ||||
|   // this is done right after show, so this is only OK if LED updates are completed before show() returns | ||||
|   // or async show has a separate buffer (ESP32 RMT and I2S are ok) | ||||
|   if (newBri < _bri) PolyBus::setBrightness(_busPtr, _iType, _bri); | ||||
| } | ||||
|  | ||||
| bool BusDigital::canShow() { | ||||
| @@ -169,22 +263,8 @@ void BusDigital::setBrightness(uint8_t b) { | ||||
|     if (_pins[0] == LED_BUILTIN || _pins[1] == LED_BUILTIN) reinit(); | ||||
|   } | ||||
|   #endif | ||||
|   uint8_t prevBri = _bri; | ||||
|   Bus::setBrightness(b); | ||||
|   PolyBus::setBrightness(_busPtr, _iType, b); | ||||
|  | ||||
|   if (_buffering) return; | ||||
|  | ||||
|   // must update/repaint every LED in the NeoPixelBus buffer to the new brightness | ||||
|   // the only case where repainting is unnecessary is when all pixels are set after the brightness change but before the next show | ||||
|   // (which we can't rely on) | ||||
|   uint16_t hwLen = _len; | ||||
|   if (_type == TYPE_WS2812_1CH_X3) hwLen = NUM_ICS_WS2812_1CH_3X(_len); // only needs a third of "RGB" LEDs for NeoPixelBus | ||||
|   for (uint_fast16_t i = 0; i < hwLen; i++) { | ||||
|     // use 0 as color order, actual order does not matter here as we just update the channel values as-is | ||||
|     uint32_t c = restoreColorLossy(PolyBus::getPixelColor(_busPtr, _iType, i, 0),prevBri); | ||||
|     PolyBus::setPixelColor(_busPtr, _iType, i, c, 0); | ||||
|   } | ||||
| } | ||||
|  | ||||
| //If LEDs are skipped, it is possible to use the first as a status LED. | ||||
| @@ -200,7 +280,7 @@ void IRAM_ATTR BusDigital::setPixelColor(uint16_t pix, uint32_t c) { | ||||
|   if (!_valid) return; | ||||
|   if (Bus::hasWhite(_type)) c = autoWhiteCalc(c); | ||||
|   if (_cct >= 1900) c = colorBalanceFromKelvin(_cct, c); //color correction from CCT | ||||
|   if (_buffering) { // should be _data != nullptr, but that causes ~20% FPS drop | ||||
|   if (_data) { // use _buffering this causes ~20% FPS drop | ||||
|     size_t channels = Bus::hasWhite(_type) + 3*Bus::hasRGB(_type); | ||||
|     size_t offset = pix*channels; | ||||
|     if (Bus::hasRGB(_type)) { | ||||
| @@ -228,9 +308,9 @@ void IRAM_ATTR BusDigital::setPixelColor(uint16_t pix, uint32_t c) { | ||||
| } | ||||
|  | ||||
| // returns original color if global buffering is enabled, else returns lossly restored color from bus | ||||
| uint32_t BusDigital::getPixelColor(uint16_t pix) { | ||||
| uint32_t IRAM_ATTR BusDigital::getPixelColor(uint16_t pix) { | ||||
|   if (!_valid) return 0; | ||||
|   if (_buffering) { // should be _data != nullptr, but that causes ~20% FPS drop | ||||
|   if (_data) { // use _buffering this causes ~20% FPS drop | ||||
|     size_t channels = Bus::hasWhite(_type) + 3*Bus::hasRGB(_type); | ||||
|     size_t offset = pix*channels; | ||||
|     uint32_t c; | ||||
| @@ -261,7 +341,7 @@ uint32_t BusDigital::getPixelColor(uint16_t pix) { | ||||
|  | ||||
| uint8_t BusDigital::getPins(uint8_t* pinArray) { | ||||
|   uint8_t numPins = IS_2PIN(_type) ? 2 : 1; | ||||
|   for (uint8_t i = 0; i < numPins; i++) pinArray[i] = _pins[i]; | ||||
|   for (unsigned i = 0; i < numPins; i++) pinArray[i] = _pins[i]; | ||||
|   return numPins; | ||||
| } | ||||
|  | ||||
| @@ -296,16 +376,25 @@ BusPwm::BusPwm(BusConfig &bc) | ||||
|   _frequency = bc.frequency ? bc.frequency : WLED_PWM_FREQ; | ||||
|  | ||||
|   #ifdef ESP8266 | ||||
|   analogWriteRange(255);  //same range as one RGB channel | ||||
|   // duty cycle resolution (_depth) can be extracted from this formula: 1MHz > _frequency * 2^_depth | ||||
|   if      (_frequency > 1760) _depth =  8; | ||||
|   else if (_frequency >  880) _depth =  9; | ||||
|   else                        _depth = 10; // WLED_PWM_FREQ <= 880Hz | ||||
|   analogWriteRange((1<<_depth)-1); | ||||
|   analogWriteFreq(_frequency); | ||||
|   #else | ||||
|   _ledcStart = pinManager.allocateLedc(numPins); | ||||
|   if (_ledcStart == 255) { //no more free LEDC channels | ||||
|     deallocatePins(); return; | ||||
|   } | ||||
|   // duty cycle resolution (_depth) can be extracted from this formula: 80MHz > _frequency * 2^_depth | ||||
|   if      (_frequency > 78124) _depth =  9; | ||||
|   else if (_frequency > 39062) _depth = 10; | ||||
|   else if (_frequency > 19531) _depth = 11; | ||||
|   else                         _depth = 12; // WLED_PWM_FREQ <= 19531Hz | ||||
|   #endif | ||||
|  | ||||
|   for (uint8_t i = 0; i < numPins; i++) { | ||||
|   for (unsigned i = 0; i < numPins; i++) { | ||||
|     uint8_t currentPin = bc.pins[i]; | ||||
|     if (!pinManager.allocatePin(currentPin, true, PinOwner::BusPwm)) { | ||||
|       deallocatePins(); return; | ||||
| @@ -314,7 +403,7 @@ BusPwm::BusPwm(BusConfig &bc) | ||||
|     #ifdef ESP8266 | ||||
|     pinMode(_pins[i], OUTPUT); | ||||
|     #else | ||||
|     ledcSetup(_ledcStart + i, _frequency, 8); | ||||
|     ledcSetup(_ledcStart + i, _frequency, _depth); | ||||
|     ledcAttachPin(_pins[i], _ledcStart + i); | ||||
|     #endif | ||||
|   } | ||||
| @@ -381,12 +470,49 @@ uint32_t BusPwm::getPixelColor(uint16_t pix) { | ||||
|   return RGBW32(_data[0], _data[1], _data[2], _data[3]); | ||||
| } | ||||
|  | ||||
| #ifndef ESP8266 | ||||
| static const uint16_t cieLUT[256] = { | ||||
| 	0, 2, 4, 5, 7, 9, 11, 13, 15, 16,  | ||||
| 	18, 20, 22, 24, 26, 27, 29, 31, 33, 35,  | ||||
| 	34, 36, 37, 39, 41, 43, 45, 47, 49, 52,  | ||||
| 	54, 56, 59, 61, 64, 67, 69, 72, 75, 78,  | ||||
| 	81, 84, 87, 90, 94, 97, 100, 104, 108, 111,  | ||||
| 	115, 119, 123, 127, 131, 136, 140, 144, 149, 154,  | ||||
| 	158, 163, 168, 173, 178, 183, 189, 194, 200, 205,  | ||||
| 	211, 217, 223, 229, 235, 241, 247, 254, 261, 267,  | ||||
| 	274, 281, 288, 295, 302, 310, 317, 325, 333, 341,  | ||||
| 	349, 357, 365, 373, 382, 391, 399, 408, 417, 426,  | ||||
| 	436, 445, 455, 464, 474, 484, 494, 505, 515, 526,  | ||||
| 	536, 547, 558, 569, 580, 592, 603, 615, 627, 639,  | ||||
| 	651, 663, 676, 689, 701, 714, 727, 741, 754, 768,  | ||||
| 	781, 795, 809, 824, 838, 853, 867, 882, 897, 913,  | ||||
| 	928, 943, 959, 975, 991, 1008, 1024, 1041, 1058, 1075,  | ||||
| 	1092, 1109, 1127, 1144, 1162, 1180, 1199, 1217, 1236, 1255,  | ||||
| 	1274, 1293, 1312, 1332, 1352, 1372, 1392, 1412, 1433, 1454,  | ||||
| 	1475, 1496, 1517, 1539, 1561, 1583, 1605, 1628, 1650, 1673,  | ||||
| 	1696, 1719, 1743, 1767, 1791, 1815, 1839, 1864, 1888, 1913,  | ||||
| 	1939, 1964, 1990, 2016, 2042, 2068, 2095, 2121, 2148, 2176,  | ||||
| 	2203, 2231, 2259, 2287, 2315, 2344, 2373, 2402, 2431, 2461,  | ||||
| 	2491, 2521, 2551, 2581, 2612, 2643, 2675, 2706, 2738, 2770,  | ||||
| 	2802, 2835, 2867, 2900, 2934, 2967, 3001, 3035, 3069, 3104,  | ||||
| 	3138, 3174, 3209, 3244, 3280, 3316, 3353, 3389, 3426, 3463,  | ||||
| 	3501, 3539, 3576, 3615, 3653, 3692, 3731, 3770, 3810, 3850,  | ||||
| 	3890, 3930, 3971, 4012, 4053, 4095 | ||||
| }; | ||||
| #endif | ||||
|  | ||||
| void BusPwm::show() { | ||||
|   if (!_valid) return; | ||||
|   uint8_t numPins = NUM_PWM_PINS(_type); | ||||
|   for (uint8_t i = 0; i < numPins; i++) { | ||||
|     uint8_t scaled = (_data[i] * _bri) / 255; | ||||
|     if (_reversed) scaled = 255 - scaled; | ||||
|   unsigned maxBri = (1<<_depth) - 1; | ||||
|   #ifdef ESP8266 | ||||
|   unsigned pwmBri = (unsigned)(roundf(powf((float)_bri / 255.0f, 1.7f) * (float)maxBri + 0.5f)); // using gamma 1.7 to extrapolate PWM duty cycle | ||||
|   #else | ||||
|   unsigned pwmBri = cieLUT[_bri] >> (12 - _depth); // use CIE LUT | ||||
|   #endif | ||||
|   for (unsigned i = 0; i < numPins; i++) { | ||||
|     unsigned scaled = (_data[i] * pwmBri) / 255; | ||||
|     if (_reversed) scaled = maxBri - scaled; | ||||
|     #ifdef ESP8266 | ||||
|     analogWrite(_pins[i], scaled); | ||||
|     #else | ||||
| @@ -398,7 +524,7 @@ void BusPwm::show() { | ||||
| uint8_t BusPwm::getPins(uint8_t* pinArray) { | ||||
|   if (!_valid) return 0; | ||||
|   uint8_t numPins = NUM_PWM_PINS(_type); | ||||
|   for (uint8_t i = 0; i < numPins; i++) { | ||||
|   for (unsigned i = 0; i < numPins; i++) { | ||||
|     pinArray[i] = _pins[i]; | ||||
|   } | ||||
|   return numPins; | ||||
| @@ -406,7 +532,7 @@ uint8_t BusPwm::getPins(uint8_t* pinArray) { | ||||
|  | ||||
| void BusPwm::deallocatePins() { | ||||
|   uint8_t numPins = NUM_PWM_PINS(_type); | ||||
|   for (uint8_t i = 0; i < numPins; i++) { | ||||
|   for (unsigned i = 0; i < numPins; i++) { | ||||
|     pinManager.deallocatePin(_pins[i], PinOwner::BusPwm); | ||||
|     if (!pinManager.isPinOk(_pins[i])) continue; | ||||
|     #ifdef ESP8266 | ||||
| @@ -473,6 +599,10 @@ BusNetwork::BusNetwork(BusConfig &bc) | ||||
|       _rgbw = false; | ||||
|       _UDPtype = 2; | ||||
|       break; | ||||
|     case TYPE_NET_ARTNET_RGBW: | ||||
|       _rgbw = true; | ||||
|       _UDPtype = 2; | ||||
|       break; | ||||
|     case TYPE_NET_E131_RGB: | ||||
|       _rgbw = false; | ||||
|       _UDPtype = 1; | ||||
| @@ -512,7 +642,7 @@ void BusNetwork::show() { | ||||
| } | ||||
|  | ||||
| uint8_t BusNetwork::getPins(uint8_t* pinArray) { | ||||
|   for (uint8_t i = 0; i < 4; i++) { | ||||
|   for (unsigned i = 0; i < 4; i++) { | ||||
|     pinArray[i] = _client[i]; | ||||
|   } | ||||
|   return 4; | ||||
| @@ -527,29 +657,34 @@ void BusNetwork::cleanup() { | ||||
|  | ||||
| //utility to get the approx. memory usage of a given BusConfig | ||||
| uint32_t BusManager::memUsage(BusConfig &bc) { | ||||
|   uint8_t type = bc.type; | ||||
|   if (bc.type == TYPE_ONOFF || IS_PWM(bc.type)) return 5; | ||||
|  | ||||
|   uint16_t len = bc.count + bc.skipAmount; | ||||
|   if (type > 15 && type < 32) { // digital types | ||||
|     if (type == TYPE_UCS8903 || type == TYPE_UCS8904) len *= 2; // 16-bit LEDs | ||||
|   uint16_t channels = 3; | ||||
|   uint16_t multiplier = 1; | ||||
|   if (IS_DIGITAL(bc.type)) { // digital types | ||||
|     if (IS_16BIT(bc.type)) len *= 2; // 16-bit LEDs | ||||
|     #ifdef ESP8266 | ||||
|       if (bc.type > 28) channels = 4; //RGBW | ||||
|       if (bc.pins[0] == 3) { //8266 DMA uses 5x the mem | ||||
|         if (type > 28) return len*20; //RGBW | ||||
|         return len*15; | ||||
|         multiplier = 5; | ||||
|       } | ||||
|       if (type > 28) return len*4; //RGBW | ||||
|       return len*3; | ||||
|     #else //ESP32 RMT uses double buffer? | ||||
|       if (type > 28) return len*8; //RGBW | ||||
|       return len*6; | ||||
|     #else //ESP32 RMT uses double buffer, I2S uses 5x buffer | ||||
|       if (bc.type > 28) channels = 4; //RGBW | ||||
|       multiplier = 2; | ||||
|     #endif | ||||
|   } | ||||
|   if (type > 31 && type < 48) return 5; | ||||
|   return len*3; //RGB | ||||
|   if (IS_VIRTUAL(bc.type)) { | ||||
|     switch (bc.type) { | ||||
|       case TYPE_NET_DDP_RGBW: channels = 4; break; | ||||
|     } | ||||
|   } | ||||
|   return len * channels * multiplier; //RGB | ||||
| } | ||||
|  | ||||
| int BusManager::add(BusConfig &bc) { | ||||
|   if (getNumBusses() - getNumVirtualBusses() >= WLED_MAX_BUSSES) return -1; | ||||
|   if (bc.type >= TYPE_NET_DDP_RGB && bc.type < 96) { | ||||
|   if (IS_VIRTUAL(bc.type)) { | ||||
|     busses[numBusses] = new BusNetwork(bc); | ||||
|   } else if (IS_DIGITAL(bc.type)) { | ||||
|     busses[numBusses] = new BusDigital(bc, numBusses, colorOrderMap); | ||||
| @@ -566,24 +701,27 @@ void BusManager::removeAll() { | ||||
|   DEBUG_PRINTLN(F("Removing all.")); | ||||
|   //prevents crashes due to deleting busses while in use. | ||||
|   while (!canAllShow()) yield(); | ||||
|   for (uint8_t i = 0; i < numBusses; i++) delete busses[i]; | ||||
|   for (unsigned i = 0; i < numBusses; i++) delete busses[i]; | ||||
|   numBusses = 0; | ||||
| } | ||||
|  | ||||
| void BusManager::show() { | ||||
|   for (uint8_t i = 0; i < numBusses; i++) { | ||||
|   _milliAmpsUsed = 0; | ||||
|   for (unsigned i = 0; i < numBusses; i++) { | ||||
|     busses[i]->show(); | ||||
|     _milliAmpsUsed += busses[i]->getUsedCurrent(); | ||||
|   } | ||||
|   if (_milliAmpsUsed) _milliAmpsUsed += MA_FOR_ESP; | ||||
| } | ||||
|  | ||||
| void BusManager::setStatusPixel(uint32_t c) { | ||||
|   for (uint8_t i = 0; i < numBusses; i++) { | ||||
|   for (unsigned i = 0; i < numBusses; i++) { | ||||
|     busses[i]->setStatusPixel(c); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void IRAM_ATTR BusManager::setPixelColor(uint16_t pix, uint32_t c) { | ||||
|   for (uint8_t i = 0; i < numBusses; i++) { | ||||
|   for (unsigned i = 0; i < numBusses; i++) { | ||||
|     Bus* b = busses[i]; | ||||
|     uint16_t bstart = b->getStart(); | ||||
|     if (pix < bstart || pix >= bstart + b->getLength()) continue; | ||||
| @@ -592,7 +730,7 @@ void IRAM_ATTR BusManager::setPixelColor(uint16_t pix, uint32_t c) { | ||||
| } | ||||
|  | ||||
| void BusManager::setBrightness(uint8_t b) { | ||||
|   for (uint8_t i = 0; i < numBusses; i++) { | ||||
|   for (unsigned i = 0; i < numBusses; i++) { | ||||
|     busses[i]->setBrightness(b); | ||||
|   } | ||||
| } | ||||
| @@ -607,7 +745,7 @@ void BusManager::setSegmentCCT(int16_t cct, bool allowWBCorrection) { | ||||
| } | ||||
|  | ||||
| uint32_t BusManager::getPixelColor(uint16_t pix) { | ||||
|   for (uint8_t i = 0; i < numBusses; i++) { | ||||
|   for (unsigned i = 0; i < numBusses; i++) { | ||||
|     Bus* b = busses[i]; | ||||
|     uint16_t bstart = b->getStart(); | ||||
|     if (pix < bstart || pix >= bstart + b->getLength()) continue; | ||||
| @@ -617,7 +755,7 @@ uint32_t BusManager::getPixelColor(uint16_t pix) { | ||||
| } | ||||
|  | ||||
| bool BusManager::canAllShow() { | ||||
|   for (uint8_t i = 0; i < numBusses; i++) { | ||||
|   for (unsigned i = 0; i < numBusses; i++) { | ||||
|     if (!busses[i]->canShow()) return false; | ||||
|   } | ||||
|   return true; | ||||
| @@ -631,7 +769,7 @@ Bus* BusManager::getBus(uint8_t busNr) { | ||||
| //semi-duplicate of strip.getLengthTotal() (though that just returns strip._length, calculated in finalizeInit()) | ||||
| uint16_t BusManager::getTotalLength() { | ||||
|   uint16_t len = 0; | ||||
|   for (uint8_t i=0; i<numBusses; i++) len += busses[i]->getLength(); | ||||
|   for (unsigned i=0; i<numBusses; i++) len += busses[i]->getLength(); | ||||
|   return len; | ||||
| } | ||||
|  | ||||
| @@ -639,3 +777,11 @@ uint16_t BusManager::getTotalLength() { | ||||
| int16_t Bus::_cct = -1; | ||||
| uint8_t Bus::_cctBlend = 0; | ||||
| uint8_t Bus::_gAWM = 255; | ||||
|  | ||||
| uint16_t BusDigital::_milliAmpsTotal = 0; | ||||
|  | ||||
| uint8_t       BusManager::numBusses = 0; | ||||
| Bus*          BusManager::busses[WLED_MAX_BUSSES+WLED_MIN_VIRTUAL_BUSSES]; | ||||
| ColorOrderMap BusManager::colorOrderMap = {}; | ||||
| uint16_t      BusManager::_milliAmpsUsed = 0; | ||||
| uint16_t      BusManager::_milliAmpsMax = ABL_MILLIAMPS_DEFAULT; | ||||
| @@ -35,8 +35,10 @@ struct BusConfig { | ||||
|   uint8_t pins[5] = {LEDPIN, 255, 255, 255, 255}; | ||||
|   uint16_t frequency; | ||||
|   bool doubleBuffer; | ||||
|   uint8_t milliAmpsPerLed; | ||||
|   uint16_t milliAmpsMax; | ||||
|  | ||||
|   BusConfig(uint8_t busType, uint8_t* ppins, uint16_t pstart, uint16_t len = 1, uint8_t pcolorOrder = COL_ORDER_GRB, bool rev = false, uint8_t skip = 0, byte aw=RGBW_MODE_MANUAL_ONLY, uint16_t clock_kHz=0U, bool dblBfr=false) | ||||
|   BusConfig(uint8_t busType, uint8_t* ppins, uint16_t pstart, uint16_t len = 1, uint8_t pcolorOrder = COL_ORDER_GRB, bool rev = false, uint8_t skip = 0, byte aw=RGBW_MODE_MANUAL_ONLY, uint16_t clock_kHz=0U, bool dblBfr=false, uint8_t maPerLed=55, uint16_t maMax=ABL_MILLIAMPS_DEFAULT) | ||||
|   : count(len) | ||||
|   , start(pstart) | ||||
|   , colorOrder(pcolorOrder) | ||||
| @@ -45,6 +47,8 @@ struct BusConfig { | ||||
|   , autoWhite(aw) | ||||
|   , frequency(clock_kHz) | ||||
|   , doubleBuffer(dblBfr) | ||||
|   , milliAmpsPerLed(maPerLed) | ||||
|   , milliAmpsMax(maMax) | ||||
|   { | ||||
|     refreshReq = (bool) GET_BIT(busType,7); | ||||
|     type = busType & 0x7F;  // bit 7 may be/is hacked to include refresh info (1=refresh in off state, 0=no refresh) | ||||
| @@ -125,13 +129,15 @@ class Bus { | ||||
|     virtual void     setPixelColor(uint16_t pix, uint32_t c) = 0; | ||||
|     virtual uint32_t getPixelColor(uint16_t pix) { return 0; } | ||||
|     virtual void     setBrightness(uint8_t b)    { _bri = b; }; | ||||
|     virtual void     cleanup() = 0; | ||||
|     virtual uint8_t  getPins(uint8_t* pinArray)  { return 0; } | ||||
|     virtual uint16_t getLength()                 { return _len; } | ||||
|     virtual void     setColorOrder()             {} | ||||
|     virtual void     setColorOrder(uint8_t co)   {} | ||||
|     virtual uint8_t  getColorOrder()             { return COL_ORDER_RGB; } | ||||
|     virtual uint8_t  skippedLeds()               { return 0; } | ||||
|     virtual uint16_t getFrequency()              { return 0U; } | ||||
|     virtual uint16_t getLEDCurrent()             { return 0; } | ||||
|     virtual uint16_t getUsedCurrent()            { return 0; } | ||||
|     virtual uint16_t getMaxCurrent()             { return 0; } | ||||
|     inline  void     setReversed(bool reversed)  { _reversed = reversed; } | ||||
|     inline  uint16_t getStart()                  { return _start; } | ||||
|     inline  void     setStart(uint16_t start)    { _start = start; } | ||||
| @@ -200,17 +206,21 @@ class BusDigital : public Bus { | ||||
|     BusDigital(BusConfig &bc, uint8_t nr, const ColorOrderMap &com); | ||||
|     ~BusDigital() { cleanup(); } | ||||
|  | ||||
|     void show(); | ||||
|     bool canShow(); | ||||
|     void setBrightness(uint8_t b); | ||||
|     void setStatusPixel(uint32_t c); | ||||
|     void setPixelColor(uint16_t pix, uint32_t c); | ||||
|     void setColorOrder(uint8_t colorOrder); | ||||
|     uint32_t getPixelColor(uint16_t pix); | ||||
|     uint8_t  getColorOrder() { return _colorOrder; } | ||||
|     uint8_t  getPins(uint8_t* pinArray); | ||||
|     uint8_t  skippedLeds()   { return _skip; } | ||||
|     uint16_t getFrequency()  { return _frequencykHz; } | ||||
|     void show() override; | ||||
|     bool canShow() override; | ||||
|     void setBrightness(uint8_t b) override; | ||||
|     void setStatusPixel(uint32_t c) override; | ||||
|     void setPixelColor(uint16_t pix, uint32_t c) override; | ||||
|     void setColorOrder(uint8_t colorOrder) override; | ||||
|     uint32_t getPixelColor(uint16_t pix) override; | ||||
|     uint8_t  getColorOrder() override  { return _colorOrder; } | ||||
|     uint8_t  getPins(uint8_t* pinArray) override; | ||||
|     uint8_t  skippedLeds() override    { return _skip; } | ||||
|     uint16_t getFrequency() override   { return _frequencykHz; } | ||||
|     uint8_t  estimateCurrentAndLimitBri(); | ||||
|     uint16_t getLEDCurrent() override  { return _milliAmpsPerLed; } | ||||
|     uint16_t getUsedCurrent() override { return _milliAmpsTotal; } | ||||
|     uint16_t getMaxCurrent() override  { return _milliAmpsMax; } | ||||
|     void reinit(); | ||||
|     void cleanup(); | ||||
|  | ||||
| @@ -220,9 +230,12 @@ class BusDigital : public Bus { | ||||
|     uint8_t _pins[2]; | ||||
|     uint8_t _iType; | ||||
|     uint16_t _frequencykHz; | ||||
|     uint8_t _milliAmpsPerLed; | ||||
|     uint16_t _milliAmpsMax; | ||||
|     void * _busPtr; | ||||
|     const ColorOrderMap &_colorOrderMap; | ||||
|     bool _buffering; // temporary until we figure out why comparison "_data != nullptr" causes severe FPS drop | ||||
|  | ||||
|     static uint16_t _milliAmpsTotal; // is overwitten/recalculated on each show() | ||||
|  | ||||
|     inline uint32_t restoreColorLossy(uint32_t c, uint8_t restoreBri) { | ||||
|       if (restoreBri < 255) { | ||||
| @@ -242,11 +255,11 @@ class BusPwm : public Bus { | ||||
|     BusPwm(BusConfig &bc); | ||||
|     ~BusPwm() { cleanup(); } | ||||
|  | ||||
|     void setPixelColor(uint16_t pix, uint32_t c); | ||||
|     uint32_t getPixelColor(uint16_t pix); //does no index check | ||||
|     uint8_t  getPins(uint8_t* pinArray); | ||||
|     uint16_t getFrequency() { return _frequency; } | ||||
|     void show(); | ||||
|     void setPixelColor(uint16_t pix, uint32_t c) override; | ||||
|     uint32_t getPixelColor(uint16_t pix) override; //does no index check | ||||
|     uint8_t  getPins(uint8_t* pinArray) override; | ||||
|     uint16_t getFrequency() override { return _frequency; } | ||||
|     void show() override; | ||||
|     void cleanup() { deallocatePins(); } | ||||
|  | ||||
|   private: | ||||
| @@ -255,6 +268,7 @@ class BusPwm : public Bus { | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|     uint8_t _ledcStart; | ||||
|     #endif | ||||
|     uint8_t _depth; | ||||
|     uint16_t _frequency; | ||||
|  | ||||
|     void deallocatePins(); | ||||
| @@ -266,10 +280,10 @@ class BusOnOff : public Bus { | ||||
|     BusOnOff(BusConfig &bc); | ||||
|     ~BusOnOff() { cleanup(); } | ||||
|  | ||||
|     void setPixelColor(uint16_t pix, uint32_t c); | ||||
|     uint32_t getPixelColor(uint16_t pix); | ||||
|     uint8_t  getPins(uint8_t* pinArray); | ||||
|     void show(); | ||||
|     void setPixelColor(uint16_t pix, uint32_t c) override; | ||||
|     uint32_t getPixelColor(uint16_t pix) override; | ||||
|     uint8_t  getPins(uint8_t* pinArray) override; | ||||
|     void show() override; | ||||
|     void cleanup() { pinManager.deallocatePin(_pin, PinOwner::BusOnOff); } | ||||
|  | ||||
|   private: | ||||
| @@ -283,13 +297,13 @@ class BusNetwork : public Bus { | ||||
|     BusNetwork(BusConfig &bc); | ||||
|     ~BusNetwork() { cleanup(); } | ||||
|  | ||||
|     bool hasRGB()   { return true; } | ||||
|     bool hasWhite() { return _rgbw; } | ||||
|     bool canShow()  { return !_broadcastLock; } // this should be a return value from UDP routine if it is still sending data out | ||||
|     void setPixelColor(uint16_t pix, uint32_t c); | ||||
|     uint32_t getPixelColor(uint16_t pix); | ||||
|     uint8_t  getPins(uint8_t* pinArray); | ||||
|     void show(); | ||||
|     bool hasRGB() override   { return true; } | ||||
|     bool hasWhite() override { return _rgbw; } | ||||
|     bool canShow() override  { return !_broadcastLock; } // this should be a return value from UDP routine if it is still sending data out | ||||
|     void setPixelColor(uint16_t pix, uint32_t c) override; | ||||
|     uint32_t getPixelColor(uint16_t pix) override; | ||||
|     uint8_t  getPins(uint8_t* pinArray) override; | ||||
|     void show() override; | ||||
|     void cleanup(); | ||||
|  | ||||
|   private: | ||||
| @@ -303,39 +317,44 @@ class BusNetwork : public Bus { | ||||
|  | ||||
| class BusManager { | ||||
|   public: | ||||
|     BusManager() : numBusses(0) {}; | ||||
|     BusManager() {}; | ||||
|  | ||||
|     //utility to get the approx. memory usage of a given BusConfig | ||||
|     static uint32_t memUsage(BusConfig &bc); | ||||
|     static uint16_t currentMilliamps(void) { return _milliAmpsUsed; } | ||||
|     static uint16_t ablMilliampsMax(void)  { return _milliAmpsMax; } | ||||
|  | ||||
|     int add(BusConfig &bc); | ||||
|     static int add(BusConfig &bc); | ||||
|  | ||||
|     //do not call this method from system context (network callback) | ||||
|     void removeAll(); | ||||
|     static void removeAll(); | ||||
|  | ||||
|     void show(); | ||||
|     bool canAllShow(); | ||||
|     void setStatusPixel(uint32_t c); | ||||
|     void setPixelColor(uint16_t pix, uint32_t c); | ||||
|     void setBrightness(uint8_t b); | ||||
|     void setSegmentCCT(int16_t cct, bool allowWBCorrection = false); | ||||
|     uint32_t getPixelColor(uint16_t pix); | ||||
|     static void show(); | ||||
|     static bool canAllShow(); | ||||
|     static void setStatusPixel(uint32_t c); | ||||
|     static void setPixelColor(uint16_t pix, uint32_t c); | ||||
|     static void setBrightness(uint8_t b); | ||||
|     static void setSegmentCCT(int16_t cct, bool allowWBCorrection = false); | ||||
|     static void setMilliampsMax(uint16_t max) { _milliAmpsMax = max;} | ||||
|     static uint32_t getPixelColor(uint16_t pix); | ||||
|  | ||||
|     Bus* getBus(uint8_t busNr); | ||||
|     static Bus* getBus(uint8_t busNr); | ||||
|  | ||||
|     //semi-duplicate of strip.getLengthTotal() (though that just returns strip._length, calculated in finalizeInit()) | ||||
|     uint16_t getTotalLength(); | ||||
|     inline uint8_t getNumBusses() const { return numBusses; } | ||||
|     static uint16_t getTotalLength(); | ||||
|     static uint8_t getNumBusses() { return numBusses; } | ||||
|  | ||||
|     inline void                 updateColorOrderMap(const ColorOrderMap &com) { memcpy(&colorOrderMap, &com, sizeof(ColorOrderMap)); } | ||||
|     inline const ColorOrderMap& getColorOrderMap() const { return colorOrderMap; } | ||||
|     static void                 updateColorOrderMap(const ColorOrderMap &com) { memcpy(&colorOrderMap, &com, sizeof(ColorOrderMap)); } | ||||
|     static const ColorOrderMap& getColorOrderMap() { return colorOrderMap; } | ||||
|  | ||||
|   private: | ||||
|     uint8_t numBusses; | ||||
|     Bus* busses[WLED_MAX_BUSSES+WLED_MIN_VIRTUAL_BUSSES]; | ||||
|     ColorOrderMap colorOrderMap; | ||||
|     static uint8_t numBusses; | ||||
|     static Bus* busses[WLED_MAX_BUSSES+WLED_MIN_VIRTUAL_BUSSES]; | ||||
|     static ColorOrderMap colorOrderMap; | ||||
|     static uint16_t _milliAmpsUsed; | ||||
|     static uint16_t _milliAmpsMax; | ||||
|  | ||||
|     inline uint8_t getNumVirtualBusses() { | ||||
|     static uint8_t getNumVirtualBusses() { | ||||
|       int j = 0; | ||||
|       for (int i=0; i<numBusses; i++) if (busses[i]->getType() >= TYPE_NET_DDP_RGB && busses[i]->getType() < 96) j++; | ||||
|       return j; | ||||
|   | ||||
| @@ -63,6 +63,11 @@ | ||||
| #define I_8266_U1_UCS_4 54 | ||||
| #define I_8266_DM_UCS_4 55 | ||||
| #define I_8266_BB_UCS_4 56 | ||||
| //ESP8266 APA106 | ||||
| #define I_8266_U0_APA106_3 81 | ||||
| #define I_8266_U1_APA106_3 82 | ||||
| #define I_8266_DM_APA106_3 83 | ||||
| #define I_8266_BB_APA106_3 84 | ||||
|  | ||||
| /*** ESP32 Neopixel methods ***/ | ||||
| //RGB | ||||
| @@ -100,6 +105,10 @@ | ||||
| #define I_32_I0_UCS_4 61 | ||||
| #define I_32_I1_UCS_4 62 | ||||
| //Bit Bang theoratically possible, but very undesirable and not needed (no pin restrictions on RMT and I2S) | ||||
| #define I_32_RN_APA106_3 85 | ||||
| #define I_32_I0_APA106_3 86 | ||||
| #define I_32_I1_APA106_3 87 | ||||
| #define I_32_BB_APA106_3 88  // bitbangging on ESP32 not recommended | ||||
|  | ||||
| //APA102 | ||||
| #define I_HS_DOT_3 39 //hardware SPI | ||||
| @@ -162,6 +171,11 @@ | ||||
| #define B_8266_U1_UCS_4 NeoPixelBusLg<NeoRgbwUcs8904Feature, NeoEsp8266Uart1Ws2813Method, NeoGammaNullMethod>   //4 chan, esp8266, gpio2 | ||||
| #define B_8266_DM_UCS_4 NeoPixelBusLg<NeoRgbwUcs8904Feature, NeoEsp8266Dma800KbpsMethod, NeoGammaNullMethod>    //4 chan, esp8266, gpio3 | ||||
| #define B_8266_BB_UCS_4 NeoPixelBusLg<NeoRgbwUcs8904Feature, NeoEsp8266BitBang800KbpsMethod, NeoGammaNullMethod> //4 chan, esp8266, bb (any pin) | ||||
| //APA106 | ||||
| #define B_8266_U0_APA106_3 NeoPixelBusLg<NeoRbgFeature, NeoEsp8266Uart0Apa106Method, NeoGammaNullMethod> //3 chan, esp8266, gpio1 | ||||
| #define B_8266_U1_APA106_3 NeoPixelBusLg<NeoRbgFeature, NeoEsp8266Uart1Apa106Method, NeoGammaNullMethod> //3 chan, esp8266, gpio2 | ||||
| #define B_8266_DM_APA106_3 NeoPixelBusLg<NeoGrbFeature, NeoEsp8266DmaApa106Method, NeoGammaNullMethod>  //3 chan, esp8266, gpio3 | ||||
| #define B_8266_BB_APA106_3 NeoPixelBusLg<NeoGrbFeature, NeoEsp8266BitBangApa106Method, NeoGammaNullMethod> //3 chan, esp8266, bb (any pin but 16) | ||||
| #endif | ||||
|  | ||||
| /*** ESP32 Neopixel methods ***/ | ||||
| @@ -229,6 +243,14 @@ | ||||
| #define B_32_I1_UCS_4 NeoPixelBusLg<NeoRgbwUcs8904Feature, NeoEsp32I2s1800KbpsMethod, NeoGammaNullMethod> | ||||
| #endif | ||||
| //Bit Bang theoratically possible, but very undesirable and not needed (no pin restrictions on RMT and I2S) | ||||
| #define B_32_RN_APA106_3 NeoPixelBusLg<NeoGrbFeature, NeoEsp32RmtNApa106Method, NeoGammaNullMethod> | ||||
| #ifndef WLED_NO_I2S0_PIXELBUS | ||||
| #define B_32_I0_APA106_3 NeoPixelBusLg<NeoGrbFeature, NeoEsp32I2s0Apa106Method, NeoGammaNullMethod> | ||||
| #endif | ||||
| #ifndef WLED_NO_I2S1_PIXELBUS | ||||
| #define B_32_I1_APA106_3 NeoPixelBusLg<NeoGrbFeature, NeoEsp32I2s1Apa106Method, NeoGammaNullMethod> | ||||
| #endif | ||||
| //#define B_32_BB_APA106_3 NeoPixelBusLg<NeoGrbFeature, NeoEsp8266BitBangApa106Method, NeoGammaNullMethod> // NeoEsp8266BitBang800KbpsMethod | ||||
|  | ||||
| #endif | ||||
|  | ||||
| @@ -327,6 +349,10 @@ class PolyBus { | ||||
|       case I_8266_U1_UCS_4: (static_cast<B_8266_U1_UCS_4*>(busPtr))->Begin(); break; | ||||
|       case I_8266_DM_UCS_4: (static_cast<B_8266_DM_UCS_4*>(busPtr))->Begin(); break; | ||||
|       case I_8266_BB_UCS_4: (static_cast<B_8266_BB_UCS_4*>(busPtr))->Begin(); break; | ||||
|       case I_8266_U0_APA106_3: (static_cast<B_8266_U0_APA106_3*>(busPtr))->Begin(); break; | ||||
|       case I_8266_U1_APA106_3: (static_cast<B_8266_U1_APA106_3*>(busPtr))->Begin(); break; | ||||
|       case I_8266_DM_APA106_3: (static_cast<B_8266_DM_APA106_3*>(busPtr))->Begin(); break; | ||||
|       case I_8266_BB_APA106_3: (static_cast<B_8266_BB_APA106_3*>(busPtr))->Begin(); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_RN_NEO_3: (static_cast<B_32_RN_NEO_3*>(busPtr))->Begin(); break; | ||||
| @@ -379,7 +405,15 @@ class PolyBus { | ||||
|       case I_32_I1_UCS_4: (static_cast<B_32_I1_UCS_4*>(busPtr))->Begin(); break; | ||||
|       #endif | ||||
| //      case I_32_BB_UCS_4: (static_cast<B_32_BB_UCS_4*>(busPtr))->Begin(); break; | ||||
|       // ESP32 can (and should, to avoid inadvertently driving the chip select signal) specify the pins used for SPI, but only in begin() | ||||
|       case I_32_RN_APA106_3: (static_cast<B_32_RN_APA106_3*>(busPtr))->Begin(); break; | ||||
|       #ifndef WLED_NO_I2S0_PIXELBUS | ||||
|       case I_32_I0_APA106_3: (static_cast<B_32_I0_APA106_3*>(busPtr))->Begin(); break; | ||||
|       #endif | ||||
|       #ifndef WLED_NO_I2S1_PIXELBUS | ||||
|       case I_32_I1_APA106_3: (static_cast<B_32_I1_APA106_3*>(busPtr))->Begin(); break; | ||||
|       #endif | ||||
| //      case I_32_BB_APA106_3: (static_cast<B_32_BB_APA106_3*>(busPtr))->Begin(); break; | ||||
|       // ESP32 can (and should, to avoid inadvertantly driving the chip select signal) specify the pins used for SPI, but only in begin() | ||||
|       case I_HS_DOT_3: beginDotStar<B_HS_DOT_3*>(busPtr, pins[1], -1, pins[0], -1, clock_kHz); break; | ||||
|       case I_HS_LPD_3: beginDotStar<B_HS_LPD_3*>(busPtr, pins[1], -1, pins[0], -1, clock_kHz); break; | ||||
|       case I_HS_LPO_3: beginDotStar<B_HS_LPO_3*>(busPtr, pins[1], -1, pins[0], -1, clock_kHz); break; | ||||
| @@ -427,6 +461,10 @@ class PolyBus { | ||||
|       case I_8266_U1_UCS_4: busPtr = new B_8266_U1_UCS_4(len, pins[0]); break; | ||||
|       case I_8266_DM_UCS_4: busPtr = new B_8266_DM_UCS_4(len, pins[0]); break; | ||||
|       case I_8266_BB_UCS_4: busPtr = new B_8266_BB_UCS_4(len, pins[0]); break; | ||||
|       case I_8266_U0_APA106_3: busPtr = new B_8266_U0_APA106_3(len, pins[0]); break; | ||||
|       case I_8266_U1_APA106_3: busPtr = new B_8266_U1_APA106_3(len, pins[0]); break; | ||||
|       case I_8266_DM_APA106_3: busPtr = new B_8266_DM_APA106_3(len, pins[0]); break; | ||||
|       case I_8266_BB_APA106_3: busPtr = new B_8266_BB_APA106_3(len, pins[0]); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_RN_NEO_3: busPtr = new B_32_RN_NEO_3(len, pins[0], (NeoBusChannel)channel); break; | ||||
| @@ -479,6 +517,14 @@ class PolyBus { | ||||
|       case I_32_I1_UCS_4: busPtr = new B_32_I1_UCS_4(len, pins[0]); break; | ||||
|       #endif | ||||
| //      case I_32_BB_UCS_4: busPtr = new B_32_BB_UCS_4(len, pins[0], (NeoBusChannel)channel); break; | ||||
|       case I_32_RN_APA106_3: busPtr = new B_32_RN_APA106_3(len, pins[0], (NeoBusChannel)channel); break; | ||||
|       #ifndef WLED_NO_I2S0_PIXELBUS | ||||
|       case I_32_I0_APA106_3: busPtr = new B_32_I0_APA106_3(len, pins[0]); break; | ||||
|       #endif | ||||
|       #ifndef WLED_NO_I2S1_PIXELBUS | ||||
|       case I_32_I1_APA106_3: busPtr = new B_32_I1_APA106_3(len, pins[0]); break; | ||||
|       #endif | ||||
| //      case I_32_BB_APA106_3: busPtr = new B_32_BB_APA106_3(len, pins[0], (NeoBusChannel)channel); break; | ||||
|     #endif | ||||
|       // for 2-wire: pins[1] is clk, pins[0] is dat.  begin expects (len, clk, dat) | ||||
|       case I_HS_DOT_3: busPtr = new B_HS_DOT_3(len, pins[1], pins[0]); break; | ||||
| @@ -528,6 +574,10 @@ class PolyBus { | ||||
|       case I_8266_U1_UCS_4: (static_cast<B_8266_U1_UCS_4*>(busPtr))->Show(consistent); break; | ||||
|       case I_8266_DM_UCS_4: (static_cast<B_8266_DM_UCS_4*>(busPtr))->Show(consistent); break; | ||||
|       case I_8266_BB_UCS_4: (static_cast<B_8266_BB_UCS_4*>(busPtr))->Show(consistent); break; | ||||
|       case I_8266_U0_APA106_3: (static_cast<B_8266_U0_APA106_3*>(busPtr))->Show(consistent); break; | ||||
|       case I_8266_U1_APA106_3: (static_cast<B_8266_U1_APA106_3*>(busPtr))->Show(consistent); break; | ||||
|       case I_8266_DM_APA106_3: (static_cast<B_8266_DM_APA106_3*>(busPtr))->Show(consistent); break; | ||||
|       case I_8266_BB_APA106_3: (static_cast<B_8266_BB_APA106_3*>(busPtr))->Show(consistent); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_RN_NEO_3: (static_cast<B_32_RN_NEO_3*>(busPtr))->Show(consistent); break; | ||||
| @@ -580,6 +630,14 @@ class PolyBus { | ||||
|       case I_32_I1_UCS_4: (static_cast<B_32_I1_UCS_4*>(busPtr))->Show(consistent); break; | ||||
|       #endif | ||||
| //      case I_32_BB_UCS_4: (static_cast<B_32_BB_UCS_4*>(busPtr))->Show(consistent); break; | ||||
|       case I_32_RN_APA106_3: (static_cast<B_32_RN_APA106_3*>(busPtr))->Show(consistent); break; | ||||
|       #ifndef WLED_NO_I2S0_PIXELBUS | ||||
|       case I_32_I0_APA106_3: (static_cast<B_32_I0_APA106_3*>(busPtr))->Show(consistent); break; | ||||
|       #endif | ||||
|       #ifndef WLED_NO_I2S1_PIXELBUS | ||||
|       case I_32_I1_APA106_3: (static_cast<B_32_I1_APA106_3*>(busPtr))->Show(consistent); break; | ||||
|       #endif | ||||
| //      case I_32_BB_APA106_3: (static_cast<B_32_BB_APA106_3*>(busPtr))->Show(consistent); break; | ||||
|     #endif | ||||
|       case I_HS_DOT_3: (static_cast<B_HS_DOT_3*>(busPtr))->Show(consistent); break; | ||||
|       case I_SS_DOT_3: (static_cast<B_SS_DOT_3*>(busPtr))->Show(consistent); break; | ||||
| @@ -625,6 +683,10 @@ class PolyBus { | ||||
|       case I_8266_U0_UCS_4: return (static_cast<B_8266_U0_UCS_4*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_U1_UCS_4: return (static_cast<B_8266_U1_UCS_4*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_DM_UCS_4: return (static_cast<B_8266_DM_UCS_4*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_U0_APA106_3: return (static_cast<B_8266_U0_APA106_3*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_U1_APA106_3: return (static_cast<B_8266_U1_APA106_3*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_DM_APA106_3: return (static_cast<B_8266_DM_APA106_3*>(busPtr))->CanShow(); break; | ||||
|       case I_8266_BB_APA106_3: return (static_cast<B_8266_BB_APA106_3*>(busPtr))->CanShow(); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_RN_NEO_3: return (static_cast<B_32_RN_NEO_3*>(busPtr))->CanShow(); break; | ||||
| @@ -677,6 +739,14 @@ class PolyBus { | ||||
|       case I_32_I1_UCS_4: return (static_cast<B_32_I1_UCS_4*>(busPtr))->CanShow(); break; | ||||
|       #endif | ||||
| //      case I_32_BB_UCS_4: return (static_cast<B_32_BB_UCS_4*>(busPtr))->CanShow(); break; | ||||
|       case I_32_RN_APA106_3: return (static_cast<B_32_RN_APA106_3*>(busPtr))->CanShow(); break; | ||||
|       #ifndef WLED_NO_I2S0_PIXELBUS | ||||
|       case I_32_I0_APA106_3: return (static_cast<B_32_I0_APA106_3*>(busPtr))->CanShow(); break; | ||||
|       #endif | ||||
|       #ifndef WLED_NO_I2S1_PIXELBUS | ||||
|       case I_32_I1_APA106_3: return (static_cast<B_32_I1_APA106_3*>(busPtr))->CanShow(); break; | ||||
|       #endif | ||||
| //      case I_32_BB_APA106_3: return (static_cast<B_32_BB_APA106_3*>(busPtr))->CanShow(); break; | ||||
|     #endif | ||||
|       case I_HS_DOT_3: return (static_cast<B_HS_DOT_3*>(busPtr))->CanShow(); break; | ||||
|       case I_SS_DOT_3: return (static_cast<B_SS_DOT_3*>(busPtr))->CanShow(); break; | ||||
| @@ -747,6 +817,10 @@ class PolyBus { | ||||
|       case I_8266_U1_UCS_4: (static_cast<B_8266_U1_UCS_4*>(busPtr))->SetPixelColor(pix, Rgbw64Color(col)); break; | ||||
|       case I_8266_DM_UCS_4: (static_cast<B_8266_DM_UCS_4*>(busPtr))->SetPixelColor(pix, Rgbw64Color(col)); break; | ||||
|       case I_8266_BB_UCS_4: (static_cast<B_8266_BB_UCS_4*>(busPtr))->SetPixelColor(pix, Rgbw64Color(col)); break; | ||||
|       case I_8266_U0_APA106_3: (static_cast<B_8266_U0_APA106_3*>(busPtr))->SetPixelColor(pix, RgbColor(col)); break; | ||||
|       case I_8266_U1_APA106_3: (static_cast<B_8266_U1_APA106_3*>(busPtr))->SetPixelColor(pix, RgbColor(col)); break; | ||||
|       case I_8266_DM_APA106_3: (static_cast<B_8266_DM_APA106_3*>(busPtr))->SetPixelColor(pix, RgbColor(col)); break; | ||||
|       case I_8266_BB_APA106_3: (static_cast<B_8266_BB_APA106_3*>(busPtr))->SetPixelColor(pix, RgbColor(col)); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_RN_NEO_3: (static_cast<B_32_RN_NEO_3*>(busPtr))->SetPixelColor(pix, RgbColor(col)); break; | ||||
| @@ -799,6 +873,14 @@ class PolyBus { | ||||
|       case I_32_I1_UCS_4: (static_cast<B_32_I1_UCS_4*>(busPtr))->SetPixelColor(pix, Rgbw64Color(col)); break; | ||||
|       #endif | ||||
| //      case I_32_BB_UCS_4: (static_cast<B_32_BB_UCS_4*>(busPtr))->SetPixelColor(pix, Rgbw64Color(col)); break; | ||||
|       case I_32_RN_APA106_3: (static_cast<B_32_RN_APA106_3*>(busPtr))->SetPixelColor(pix, RgbColor(col)); break; | ||||
|       #ifndef WLED_NO_I2S0_PIXELBUS | ||||
|       case I_32_I0_APA106_3: (static_cast<B_32_I0_APA106_3*>(busPtr))->SetPixelColor(pix, RgbColor(col)); break; | ||||
|       #endif | ||||
|       #ifndef WLED_NO_I2S1_PIXELBUS | ||||
|       case I_32_I1_APA106_3: (static_cast<B_32_I1_APA106_3*>(busPtr))->SetPixelColor(pix, RgbColor(col)); break; | ||||
|       #endif | ||||
| //      case I_32_BB_APA106_3: (static_cast<B_32_BB_APA106_3*>(busPtr))->SetPixelColor(pix, RgbColor(col)); break; | ||||
|     #endif | ||||
|       case I_HS_DOT_3: (static_cast<B_HS_DOT_3*>(busPtr))->SetPixelColor(pix, RgbColor(col)); break; | ||||
|       case I_SS_DOT_3: (static_cast<B_SS_DOT_3*>(busPtr))->SetPixelColor(pix, RgbColor(col)); break; | ||||
| @@ -845,6 +927,10 @@ class PolyBus { | ||||
|       case I_8266_U1_UCS_4: (static_cast<B_8266_U1_UCS_4*>(busPtr))->SetLuminance(b); break; | ||||
|       case I_8266_DM_UCS_4: (static_cast<B_8266_DM_UCS_4*>(busPtr))->SetLuminance(b); break; | ||||
|       case I_8266_BB_UCS_4: (static_cast<B_8266_BB_UCS_4*>(busPtr))->SetLuminance(b); break; | ||||
|       case I_8266_U0_APA106_3: (static_cast<B_8266_U0_APA106_3*>(busPtr))->SetLuminance(b); break; | ||||
|       case I_8266_U1_APA106_3: (static_cast<B_8266_U1_APA106_3*>(busPtr))->SetLuminance(b); break; | ||||
|       case I_8266_DM_APA106_3: (static_cast<B_8266_DM_APA106_3*>(busPtr))->SetLuminance(b); break; | ||||
|       case I_8266_BB_APA106_3: (static_cast<B_8266_BB_APA106_3*>(busPtr))->SetLuminance(b); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_RN_NEO_3: (static_cast<B_32_RN_NEO_3*>(busPtr))->SetLuminance(b); break; | ||||
| @@ -897,6 +983,14 @@ class PolyBus { | ||||
|       case I_32_I1_UCS_4: (static_cast<B_32_I1_UCS_4*>(busPtr))->SetLuminance(b); break; | ||||
|       #endif | ||||
| //      case I_32_BB_UCS_4: (static_cast<B_32_BB_UCS_4*>(busPtr))->SetLuminance(b); break; | ||||
|       case I_32_RN_APA106_3: (static_cast<B_32_RN_APA106_3*>(busPtr))->SetLuminance(b); break; | ||||
|       #ifndef WLED_NO_I2S0_PIXELBUS | ||||
|       case I_32_I0_APA106_3: (static_cast<B_32_I0_APA106_3*>(busPtr))->SetLuminance(b); break; | ||||
|       #endif | ||||
|       #ifndef WLED_NO_I2S1_PIXELBUS | ||||
|       case I_32_I1_APA106_3: (static_cast<B_32_I1_APA106_3*>(busPtr))->SetLuminance(b); break; | ||||
|       #endif | ||||
| //      case I_32_BB_APA106_3: (static_cast<B_32_BB_APA106_3*>(busPtr))->SetLuminance(b); break; | ||||
|     #endif | ||||
|       case I_HS_DOT_3: (static_cast<B_HS_DOT_3*>(busPtr))->SetLuminance(b); break; | ||||
|       case I_SS_DOT_3: (static_cast<B_SS_DOT_3*>(busPtr))->SetLuminance(b); break; | ||||
| @@ -944,6 +1038,10 @@ class PolyBus { | ||||
|       case I_8266_U1_UCS_4: { Rgbw64Color c = (static_cast<B_8266_U1_UCS_4*>(busPtr))->GetPixelColor(pix); col = RGBW32(c.R>>8,c.G>>8,c.B>>8,c.W>>8); } break; | ||||
|       case I_8266_DM_UCS_4: { Rgbw64Color c = (static_cast<B_8266_DM_UCS_4*>(busPtr))->GetPixelColor(pix); col = RGBW32(c.R>>8,c.G>>8,c.B>>8,c.W>>8); } break; | ||||
|       case I_8266_BB_UCS_4: { Rgbw64Color c = (static_cast<B_8266_BB_UCS_4*>(busPtr))->GetPixelColor(pix); col = RGBW32(c.R>>8,c.G>>8,c.B>>8,c.W>>8); } break; | ||||
|       case I_8266_U0_APA106_3: col = (static_cast<B_8266_U0_APA106_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_U1_APA106_3: col = (static_cast<B_8266_U1_APA106_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_DM_APA106_3: col = (static_cast<B_8266_DM_APA106_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_8266_BB_APA106_3: col = (static_cast<B_8266_BB_APA106_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_RN_NEO_3: col = (static_cast<B_32_RN_NEO_3*>(busPtr))->GetPixelColor(pix); break; | ||||
| @@ -996,6 +1094,14 @@ class PolyBus { | ||||
|       case I_32_I1_UCS_4: { Rgbw64Color c = (static_cast<B_32_I1_UCS_4*>(busPtr))->GetPixelColor(pix); col = RGBW32(c.R>>8,c.G>>8,c.B>>8,c.W>>8); } break; | ||||
|       #endif | ||||
| //      case I_32_BB_UCS_4: col = (static_cast<B_32_BB_UCS_4*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_32_RN_APA106_3: col = (static_cast<B_32_RN_APA106_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       #ifndef WLED_NO_I2S0_PIXELBUS | ||||
|       case I_32_I0_APA106_3: col = (static_cast<B_32_I0_APA106_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       #endif | ||||
|       #ifndef WLED_NO_I2S1_PIXELBUS | ||||
|       case I_32_I1_APA106_3: col = (static_cast<B_32_I1_APA106_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       #endif | ||||
| //      case I_32_BB_APA106_3: col = (static_cast<B_32_BB_APA106_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|     #endif | ||||
|       case I_HS_DOT_3: col = (static_cast<B_HS_DOT_3*>(busPtr))->GetPixelColor(pix); break; | ||||
|       case I_SS_DOT_3: col = (static_cast<B_SS_DOT_3*>(busPtr))->GetPixelColor(pix); break; | ||||
| @@ -1061,6 +1167,10 @@ class PolyBus { | ||||
|       case I_8266_U1_UCS_4: delete (static_cast<B_8266_U1_UCS_4*>(busPtr)); break; | ||||
|       case I_8266_DM_UCS_4: delete (static_cast<B_8266_DM_UCS_4*>(busPtr)); break; | ||||
|       case I_8266_BB_UCS_4: delete (static_cast<B_8266_BB_UCS_4*>(busPtr)); break; | ||||
|       case I_8266_U0_APA106_3: delete (static_cast<B_8266_U0_APA106_3*>(busPtr)); break; | ||||
|       case I_8266_U1_APA106_3: delete (static_cast<B_8266_U1_APA106_3*>(busPtr)); break; | ||||
|       case I_8266_DM_APA106_3: delete (static_cast<B_8266_DM_APA106_3*>(busPtr)); break; | ||||
|       case I_8266_BB_APA106_3: delete (static_cast<B_8266_BB_APA106_3*>(busPtr)); break; | ||||
|     #endif | ||||
|     #ifdef ARDUINO_ARCH_ESP32 | ||||
|       case I_32_RN_NEO_3: delete (static_cast<B_32_RN_NEO_3*>(busPtr)); break; | ||||
| @@ -1113,6 +1223,14 @@ class PolyBus { | ||||
|       case I_32_I1_UCS_4: delete (static_cast<B_32_I1_UCS_4*>(busPtr)); break; | ||||
|       #endif | ||||
| //      case I_32_BB_UCS_4: delete (static_cast<B_32_BB_UCS_4*>(busPtr)); break; | ||||
|       case I_32_RN_APA106_3: delete (static_cast<B_32_RN_APA106_3*>(busPtr)); break; | ||||
|       #ifndef WLED_NO_I2S0_PIXELBUS | ||||
|       case I_32_I0_APA106_3: delete (static_cast<B_32_I0_APA106_3*>(busPtr)); break; | ||||
|       #endif | ||||
|       #ifndef WLED_NO_I2S1_PIXELBUS | ||||
|       case I_32_I1_APA106_3: delete (static_cast<B_32_I1_APA106_3*>(busPtr)); break; | ||||
|       #endif | ||||
| //      case I_32_BB_APA106_3: delete (static_cast<B_32_BB_APA106_3*>(busPtr)); break; | ||||
|     #endif | ||||
|       case I_HS_DOT_3: delete (static_cast<B_HS_DOT_3*>(busPtr)); break; | ||||
|       case I_SS_DOT_3: delete (static_cast<B_SS_DOT_3*>(busPtr)); break; | ||||
| @@ -1172,6 +1290,8 @@ class PolyBus { | ||||
|           return I_8266_U0_UCS_3 + offset; | ||||
|         case TYPE_UCS8904: | ||||
|           return I_8266_U0_UCS_4 + offset; | ||||
|         case TYPE_APA106: | ||||
|           return I_8266_U0_APA106_3 + offset; | ||||
|       } | ||||
|       #else //ESP32 | ||||
|       uint8_t offset = 0; //0 = RMT (num 0-7) 8 = I2S0 9 = I2S1 | ||||
| @@ -1210,6 +1330,8 @@ class PolyBus { | ||||
|           return I_32_RN_UCS_3 + offset; | ||||
|         case TYPE_UCS8904: | ||||
|           return I_32_RN_UCS_4 + offset; | ||||
|         case TYPE_APA106: | ||||
|           return I_32_RN_APA106_3 + offset; | ||||
|       } | ||||
|       #endif | ||||
|     } | ||||
|   | ||||
| @@ -21,7 +21,6 @@ void shortPressAction(uint8_t b) | ||||
|       case 1: ++effectCurrent %= strip.getModeCount(); stateChanged = true; colorUpdated(CALL_MODE_BUTTON); break; | ||||
|     } | ||||
|   } else { | ||||
|     unloadPlaylist(); // applying a preset unloads the playlist | ||||
|     applyPreset(macroButton[b], CALL_MODE_BUTTON_PRESET); | ||||
|   } | ||||
|  | ||||
| @@ -43,7 +42,6 @@ void longPressAction(uint8_t b) | ||||
|       case 1: bri += 8; stateUpdated(CALL_MODE_BUTTON); buttonPressedTime[b] = millis(); break; // repeatable action | ||||
|     } | ||||
|   } else { | ||||
|     unloadPlaylist(); // applying a preset unloads the playlist | ||||
|     applyPreset(macroLongPress[b], CALL_MODE_BUTTON_PRESET); | ||||
|   } | ||||
|  | ||||
| @@ -65,7 +63,6 @@ void doublePressAction(uint8_t b) | ||||
|       case 1: ++effectPalette %= strip.getPaletteCount(); colorUpdated(CALL_MODE_BUTTON); break; | ||||
|     } | ||||
|   } else { | ||||
|     unloadPlaylist(); // applying a preset unloads the playlist | ||||
|     applyPreset(macroDoublePress[b], CALL_MODE_BUTTON_PRESET); | ||||
|   } | ||||
|  | ||||
| @@ -97,8 +94,9 @@ bool isButtonPressed(uint8_t i) | ||||
|       if (digitalRead(pin) == HIGH) return true; | ||||
|       break; | ||||
|     case BTN_TYPE_TOUCH: | ||||
|     case BTN_TYPE_TOUCH_SWITCH: | ||||
|       #if defined(ARDUINO_ARCH_ESP32) && !defined(CONFIG_IDF_TARGET_ESP32C3) | ||||
|       if (touchRead(pin) <= touchThreshold) return true; | ||||
|       if (digitalPinToTouchChannel(btnPin[i]) >= 0 && touchRead(pin) <= touchThreshold) return true; | ||||
|       #endif | ||||
|       break; | ||||
|   } | ||||
| @@ -109,6 +107,7 @@ void handleSwitch(uint8_t b) | ||||
| { | ||||
|   // isButtonPressed() handles inverted/noninverted logic | ||||
|   if (buttonPressedBefore[b] != isButtonPressed(b)) { | ||||
|     DEBUG_PRINT(F("Switch: State changed ")); DEBUG_PRINTLN(b); | ||||
|     buttonPressedTime[b] = millis(); | ||||
|     buttonPressedBefore[b] = !buttonPressedBefore[b]; | ||||
|   } | ||||
| @@ -116,12 +115,15 @@ void handleSwitch(uint8_t b) | ||||
|   if (buttonLongPressed[b] == buttonPressedBefore[b]) return; | ||||
|  | ||||
|   if (millis() - buttonPressedTime[b] > WLED_DEBOUNCE_THRESHOLD) { //fire edge event only after 50ms without change (debounce) | ||||
|     DEBUG_PRINT(F("Switch: Activating ")); DEBUG_PRINTLN(b); | ||||
|     if (!buttonPressedBefore[b]) { // on -> off | ||||
|       DEBUG_PRINT(F("Switch: On -> Off ")); DEBUG_PRINTLN(b); | ||||
|       if (macroButton[b]) applyPreset(macroButton[b], CALL_MODE_BUTTON_PRESET); | ||||
|       else { //turn on | ||||
|         if (!bri) {toggleOnOff(); stateUpdated(CALL_MODE_BUTTON);} | ||||
|       } | ||||
|     } else {  // off -> on | ||||
|       DEBUG_PRINT(F("Switch: Off -> On ")); DEBUG_PRINTLN(b); | ||||
|       if (macroLongPress[b]) applyPreset(macroLongPress[b], CALL_MODE_BUTTON_PRESET); | ||||
|       else { //turn off | ||||
|         if (bri) {toggleOnOff(); stateUpdated(CALL_MODE_BUTTON);} | ||||
| @@ -153,6 +155,8 @@ void handleAnalog(uint8_t b) | ||||
|   static float filteredReading[WLED_MAX_BUTTONS] = {0.0f}; | ||||
|   uint16_t rawReading;    // raw value from analogRead, scaled to 12bit | ||||
|  | ||||
|   DEBUG_PRINT(F("Analog: Reading button ")); DEBUG_PRINTLN(b); | ||||
|  | ||||
|   #ifdef ESP8266 | ||||
|   rawReading = analogRead(A0) << 2;   // convert 10bit read to 12bit | ||||
|   #else | ||||
| @@ -171,7 +175,10 @@ void handleAnalog(uint8_t b) | ||||
|   // remove noise & reduce frequency of UI updates | ||||
|   if (abs(int(aRead) - int(oldRead[b])) <= POT_SENSITIVITY) return;  // no significant change in reading | ||||
|  | ||||
|   // Un-comment the next lines if you still see flickering related to potentiometer | ||||
|   DEBUG_PRINT(F("Analog: Raw = ")); DEBUG_PRINT(rawReading); | ||||
|   DEBUG_PRINT(F(" Filtered = ")); DEBUG_PRINTLN(aRead); | ||||
|  | ||||
|   // Unomment the next lines if you still see flickering related to potentiometer | ||||
|   // This waits until strip finishes updating (why: strip was not updating at the start of handleButton() but may have started during analogRead()?) | ||||
|   //unsigned long wait_started = millis(); | ||||
|   //while(strip.isUpdating() && (millis() - wait_started < STRIP_WAIT_TIME)) { | ||||
| @@ -182,6 +189,7 @@ void handleAnalog(uint8_t b) | ||||
|  | ||||
|   // if no macro for "short press" and "long press" is defined use brightness control | ||||
|   if (!macroButton[b] && !macroLongPress[b]) { | ||||
|     DEBUG_PRINT(F("Analog: Action = ")); DEBUG_PRINTLN(macroDoublePress[b]); | ||||
|     // if "double press" macro defines which option to change | ||||
|     if (macroDoublePress[b] >= 250) { | ||||
|       // global brightness | ||||
| @@ -217,6 +225,7 @@ void handleAnalog(uint8_t b) | ||||
|       updateInterfaces(CALL_MODE_BUTTON); | ||||
|     } | ||||
|   } else { | ||||
|     DEBUG_PRINTLN(F("Analog: No action")); | ||||
|     //TODO: | ||||
|     // we can either trigger a preset depending on the level (between short and long entries) | ||||
|     // or use it for RGBW direct control | ||||
| @@ -250,7 +259,7 @@ void handleButton() | ||||
|     } | ||||
|  | ||||
|     // button is not momentary, but switch. This is only suitable on pins whose on-boot state does not matter (NOT gpio0) | ||||
|     if (buttonType[b] == BTN_TYPE_SWITCH || buttonType[b] == BTN_TYPE_PIR_SENSOR) { | ||||
|     if (buttonType[b] == BTN_TYPE_SWITCH || buttonType[b] == BTN_TYPE_TOUCH_SWITCH || buttonType[b] == BTN_TYPE_PIR_SENSOR) { | ||||
|       handleSwitch(b); | ||||
|       continue; | ||||
|     } | ||||
| @@ -335,10 +344,10 @@ void handleButton() | ||||
| void esp32RMTInvertIdle() | ||||
| { | ||||
|   bool idle_out; | ||||
|   for (uint8_t u = 0; u < busses.getNumBusses(); u++) | ||||
|   for (uint8_t u = 0; u < BusManager::getNumBusses(); u++) | ||||
|   { | ||||
|     if (u > 7) return; // only 8 RMT channels, TODO: ESP32 variants have less RMT channels | ||||
|     Bus *bus = busses.getBus(u); | ||||
|     Bus *bus = BusManager::getBus(u); | ||||
|     if (!bus || bus->getLength()==0 || !IS_DIGITAL(bus->getType()) || IS_2PIN(bus->getType())) continue; | ||||
|     //assumes that bus number to rmt channel mapping stays 1:1 | ||||
|     rmt_channel_t ch = static_cast<rmt_channel_t>(u); | ||||
|   | ||||
							
								
								
									
										312
									
								
								wled00/cfg.cpp
									
									
									
									
									
								
							
							
						
						
									
										312
									
								
								wled00/cfg.cpp
									
									
									
									
									
								
							| @@ -31,25 +31,48 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|   getStringFromJson(cmDNS, id[F("mdns")], 33); | ||||
|   getStringFromJson(serverDescription, id[F("name")], 33); | ||||
|   getStringFromJson(alexaInvocationName, id[F("inv")], 33); | ||||
| #ifdef WLED_ENABLE_SIMPLE_UI | ||||
|   CJSON(simplifiedUI, id[F("sui")]); | ||||
|  | ||||
|   JsonObject nw = doc["nw"]; | ||||
| #ifndef WLED_DISABLE_ESPNOW | ||||
|   CJSON(enableESPNow, nw[F("espnow")]); | ||||
|   getStringFromJson(linked_remote, nw[F("linked_remote")], 13); | ||||
|   linked_remote[12] = '\0'; | ||||
| #endif | ||||
|  | ||||
|   JsonObject nw_ins_0 = doc["nw"]["ins"][0]; | ||||
|   getStringFromJson(clientSSID, nw_ins_0[F("ssid")], 33); | ||||
|   //int nw_ins_0_pskl = nw_ins_0[F("pskl")]; | ||||
|   //The WiFi PSK is normally not contained in the regular file for security reasons. | ||||
|   //If it is present however, we will use it | ||||
|   getStringFromJson(clientPass, nw_ins_0["psk"], 65); | ||||
|   size_t n = 0; | ||||
|   JsonArray nw_ins = nw["ins"]; | ||||
|   if (!nw_ins.isNull()) { | ||||
|     // as password are stored separately in wsec.json when reading configuration vector resize happens there, but for dynamic config we need to resize if necessary | ||||
|     if (nw_ins.size() > 1 && nw_ins.size() > multiWiFi.size()) multiWiFi.resize(nw_ins.size()); // resize constructs objects while resizing | ||||
|     for (JsonObject wifi : nw_ins) { | ||||
|       JsonArray ip = wifi["ip"]; | ||||
|       JsonArray gw = wifi["gw"]; | ||||
|       JsonArray sn = wifi["sn"]; | ||||
|       char ssid[33] = ""; | ||||
|       char pass[65] = ""; | ||||
|       IPAddress nIP = (uint32_t)0U, nGW = (uint32_t)0U, nSN = (uint32_t)0x00FFFFFF; // little endian | ||||
|       getStringFromJson(ssid, wifi[F("ssid")], 33); | ||||
|       getStringFromJson(pass, wifi["psk"], 65); // password is not normally present but if it is, use it | ||||
|       for (size_t i = 0; i < 4; i++) { | ||||
|         CJSON(nIP[i], ip[i]); | ||||
|         CJSON(nGW[i], gw[i]); | ||||
|         CJSON(nSN[i], sn[i]); | ||||
|       } | ||||
|       if (strlen(ssid) > 0) strlcpy(multiWiFi[n].clientSSID, ssid, 33); // this will keep old SSID intact if not present in JSON | ||||
|       if (strlen(pass) > 0) strlcpy(multiWiFi[n].clientPass, pass, 65); // this will keep old password intact if not present in JSON | ||||
|       multiWiFi[n].staticIP = nIP; | ||||
|       multiWiFi[n].staticGW = nGW; | ||||
|       multiWiFi[n].staticSN = nSN; | ||||
|       if (++n >= WLED_MAX_WIFI_COUNT) break; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   JsonArray nw_ins_0_ip = nw_ins_0["ip"]; | ||||
|   JsonArray nw_ins_0_gw = nw_ins_0["gw"]; | ||||
|   JsonArray nw_ins_0_sn = nw_ins_0["sn"]; | ||||
|  | ||||
|   for (byte i = 0; i < 4; i++) { | ||||
|     CJSON(staticIP[i], nw_ins_0_ip[i]); | ||||
|     CJSON(staticGateway[i], nw_ins_0_gw[i]); | ||||
|     CJSON(staticSubnet[i], nw_ins_0_sn[i]); | ||||
|   JsonArray dns = nw[F("dns")]; | ||||
|   if (!dns.isNull()) { | ||||
|     for (size_t i = 0; i < 4; i++) { | ||||
|       CJSON(dnsAddress[i], dns[i]); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   JsonObject ap = doc["ap"]; | ||||
| @@ -81,8 +104,9 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|   // initialize LED pins and lengths prior to other HW (except for ethernet) | ||||
|   JsonObject hw_led = hw["led"]; | ||||
|  | ||||
|   CJSON(strip.ablMilliampsMax, hw_led[F("maxpwr")]); | ||||
|   CJSON(strip.milliampsPerLed, hw_led[F("ledma")]); | ||||
|   uint16_t total = hw_led[F("total")] | strip.getLengthTotal(); | ||||
|   uint16_t ablMilliampsMax = hw_led[F("maxpwr")] | BusManager::ablMilliampsMax(); | ||||
|   BusManager::setMilliampsMax(ablMilliampsMax); | ||||
|   Bus::setGlobalAWMode(hw_led[F("rgbwm")] | AW_GLOBAL_DISABLED); | ||||
|   CJSON(correctWB, hw_led["cct"]); | ||||
|   CJSON(cctFromRgb, hw_led[F("cr")]); | ||||
| @@ -132,7 +156,7 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|  | ||||
|   if (fromFS || !ins.isNull()) { | ||||
|     uint8_t s = 0;  // bus iterator | ||||
|     if (fromFS) busses.removeAll(); // can't safely manipulate busses directly in network callback | ||||
|     if (fromFS) BusManager::removeAll(); // can't safely manipulate busses directly in network callback | ||||
|     uint32_t mem = 0, globalBufMem = 0; | ||||
|     uint16_t maxlen = 0; | ||||
|     bool busesChanged = false; | ||||
| @@ -156,20 +180,27 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|       uint8_t ledType = elm["type"] | TYPE_WS2812_RGB; | ||||
|       bool reversed = elm["rev"]; | ||||
|       bool refresh = elm["ref"] | false; | ||||
|       uint16_t freqkHz = elm[F("freq")] | 0;  // will be in kHz for DotStar and Hz for PWM (not yet implemented fully) | ||||
|       ledType |= refresh << 7; // hack bit 7 to indicate strip requires off refresh | ||||
|       uint16_t freqkHz = elm[F("freq")] | 0;  // will be in kHz for DotStar and Hz for PWM | ||||
|       uint8_t AWmode = elm[F("rgbwm")] | RGBW_MODE_MANUAL_ONLY; | ||||
|       uint8_t maPerLed = elm[F("ledma")] | 55; | ||||
|       uint16_t maMax = elm[F("maxpwr")] | (ablMilliampsMax * length) / total; // rough (incorrect?) per strip ABL calculation when no config exists | ||||
|       // To disable brightness limiter we either set output max current to 0 or single LED current to 0 (we choose output max current) | ||||
|       if ((ledType > TYPE_TM1814 && ledType < TYPE_WS2801) || ledType >= TYPE_NET_DDP_RGB) { // analog and virtual | ||||
|         maPerLed = 0; | ||||
|         maMax = 0; | ||||
|       } | ||||
|       ledType |= refresh << 7; // hack bit 7 to indicate strip requires off refresh | ||||
|       if (fromFS) { | ||||
|         BusConfig bc = BusConfig(ledType, pins, start, length, colorOrder, reversed, skipFirst, AWmode, freqkHz, useGlobalLedBuffer); | ||||
|         BusConfig bc = BusConfig(ledType, pins, start, length, colorOrder, reversed, skipFirst, AWmode, freqkHz, useGlobalLedBuffer, maPerLed, maMax); | ||||
|         mem += BusManager::memUsage(bc); | ||||
|         if (useGlobalLedBuffer && start + length > maxlen) { | ||||
|           maxlen = start + length; | ||||
|           globalBufMem = maxlen * 4; | ||||
|         } | ||||
|         if (mem + globalBufMem <= MAX_LED_MEMORY) if (busses.add(bc) == -1) break;  // finalization will be done in WLED::beginStrip() | ||||
|         if (mem + globalBufMem <= MAX_LED_MEMORY) if (BusManager::add(bc) == -1) break;  // finalization will be done in WLED::beginStrip() | ||||
|       } else { | ||||
|         if (busConfigs[s] != nullptr) delete busConfigs[s]; | ||||
|         busConfigs[s] = new BusConfig(ledType, pins, start, length, colorOrder, reversed, skipFirst, AWmode, freqkHz, useGlobalLedBuffer); | ||||
|         busConfigs[s] = new BusConfig(ledType, pins, start, length, colorOrder, reversed, skipFirst, AWmode, freqkHz, useGlobalLedBuffer, maPerLed, maMax); | ||||
|         busesChanged = true; | ||||
|       } | ||||
|       s++; | ||||
| @@ -177,7 +208,7 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|     doInitBusses = busesChanged; | ||||
|     // finalization done in beginStrip() | ||||
|   } | ||||
|   if (hw_led["rev"]) busses.getBus(0)->setReversed(true); //set 0.11 global reversed setting for first bus | ||||
|   if (hw_led["rev"]) BusManager::getBus(0)->setReversed(true); //set 0.11 global reversed setting for first bus | ||||
|  | ||||
|   // read color order map configuration | ||||
|   JsonArray hw_com = hw[F("com")]; | ||||
| @@ -192,14 +223,14 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|       com.add(start, len, colorOrder); | ||||
|       s++; | ||||
|     } | ||||
|     busses.updateColorOrderMap(com); | ||||
|     BusManager::updateColorOrderMap(com); | ||||
|   } | ||||
|  | ||||
|   // read multiple button configuration | ||||
|   JsonObject btn_obj = hw["btn"]; | ||||
|   bool pull = btn_obj[F("pull")] | (!disablePullUp); // if true, pullup is enabled | ||||
|   disablePullUp = !pull; | ||||
|   JsonArray hw_btn_ins = btn_obj[F("ins")]; | ||||
|   JsonArray hw_btn_ins = btn_obj["ins"]; | ||||
|   if (!hw_btn_ins.isNull()) { | ||||
|     for (uint8_t b = 0; b < WLED_MAX_BUTTONS; b++) { // deallocate existing button pins | ||||
|       pinManager.deallocatePin(btnPin[b], PinOwner::Button); // does nothing if trying to deallocate a pin with PinOwner != Button | ||||
| @@ -255,13 +286,22 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|     // new install/missing configuration (button 0 has defaults) | ||||
|     if (fromFS) { | ||||
|       // relies upon only being called once with fromFS == true, which is currently true. | ||||
|       uint8_t s = 0; | ||||
|       if (pinManager.allocatePin(btnPin[0], false, PinOwner::Button)) { // initialized to #define value BTNPIN, or zero if not defined(!) | ||||
|         ++s; // do not clear default button if allocated successfully | ||||
|       } | ||||
|       for (; s<WLED_MAX_BUTTONS; s++) { | ||||
|         btnPin[s]           = -1; | ||||
|         buttonType[s]       = BTN_TYPE_NONE; | ||||
|       for (size_t s = 0; s < WLED_MAX_BUTTONS; s++) { | ||||
|         if (buttonType[s] == BTN_TYPE_NONE || btnPin[s] < 0 || !pinManager.allocatePin(btnPin[s], false, PinOwner::Button)) { | ||||
|           btnPin[s]     = -1; | ||||
|           buttonType[s] = BTN_TYPE_NONE; | ||||
|         } | ||||
|         if (btnPin[s] >= 0) { | ||||
|           if (disablePullUp) { | ||||
|             pinMode(btnPin[s], INPUT); | ||||
|           } else { | ||||
|             #ifdef ESP32 | ||||
|             pinMode(btnPin[s], buttonType[s]==BTN_TYPE_PUSH_ACT_HIGH ? INPUT_PULLDOWN : INPUT_PULLUP); | ||||
|             #else | ||||
|             pinMode(btnPin[s], INPUT_PULLUP); | ||||
|             #endif | ||||
|           } | ||||
|         } | ||||
|         macroButton[s]      = 0; | ||||
|         macroLongPress[s]   = 0; | ||||
|         macroDoublePress[s] = 0; | ||||
| @@ -271,6 +311,7 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|   CJSON(touchThreshold,btn_obj[F("tt")]); | ||||
|   CJSON(buttonPublishMqtt,btn_obj["mqtt"]); | ||||
|  | ||||
|   #ifndef WLED_DISABLE_INFRARED | ||||
|   int hw_ir_pin = hw["ir"]["pin"] | -2; // 4 | ||||
|   if (hw_ir_pin > -2) { | ||||
|     pinManager.deallocatePin(irPin, PinOwner::IR); | ||||
| @@ -281,6 +322,7 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|     } | ||||
|   } | ||||
|   CJSON(irEnabled, hw["ir"]["type"]); | ||||
|   #endif | ||||
|   CJSON(irApplyToAllSelected, hw["ir"]["sel"]); | ||||
|  | ||||
|   JsonObject relay = hw[F("relay")]; | ||||
| @@ -364,6 +406,7 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|   strip.setTransition(fadeTransition ? transitionDelayDefault : 0); | ||||
|   CJSON(strip.paletteFade, light_tr["pal"]); | ||||
|   CJSON(randomPaletteChangeTime, light_tr[F("rpc")]); | ||||
|   CJSON(useHarmonicRandomPalette, light_tr[F("hrp")]); | ||||
|  | ||||
|   JsonObject light_nl = light["nl"]; | ||||
|   CJSON(nightlightMode, light_nl["mode"]); | ||||
| @@ -385,24 +428,25 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|   CJSON(udpPort, if_sync[F("port0")]); // 21324 | ||||
|   CJSON(udpPort2, if_sync[F("port1")]); // 65506 | ||||
|  | ||||
|   JsonObject if_sync_recv = if_sync["recv"]; | ||||
| #ifndef WLED_DISABLE_ESPNOW | ||||
|   CJSON(useESPNowSync, if_sync[F("espnow")]); | ||||
| #endif | ||||
|  | ||||
|   JsonObject if_sync_recv = if_sync[F("recv")]; | ||||
|   CJSON(receiveNotificationBrightness, if_sync_recv["bri"]); | ||||
|   CJSON(receiveNotificationColor, if_sync_recv["col"]); | ||||
|   CJSON(receiveNotificationEffects, if_sync_recv["fx"]); | ||||
|   CJSON(receiveGroups, if_sync_recv["grp"]); | ||||
|   CJSON(receiveSegmentOptions, if_sync_recv["seg"]); | ||||
|   CJSON(receiveSegmentBounds, if_sync_recv["sb"]); | ||||
|   //! following line might be a problem if called after boot | ||||
|   receiveNotifications = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects || receiveSegmentOptions); | ||||
|  | ||||
|   JsonObject if_sync_send = if_sync["send"]; | ||||
|   prev = notifyDirectDefault; | ||||
|   CJSON(notifyDirectDefault, if_sync_send[F("dir")]); | ||||
|   if (notifyDirectDefault != prev) notifyDirect = notifyDirectDefault; | ||||
|   JsonObject if_sync_send = if_sync[F("send")]; | ||||
|   CJSON(sendNotifications, if_sync_send["en"]); | ||||
|   sendNotificationsRT = sendNotifications; | ||||
|   CJSON(notifyDirect, if_sync_send[F("dir")]); | ||||
|   CJSON(notifyButton, if_sync_send["btn"]); | ||||
|   CJSON(notifyAlexa, if_sync_send["va"]); | ||||
|   CJSON(notifyHue, if_sync_send["hue"]); | ||||
|   CJSON(notifyMacro, if_sync_send["macro"]); | ||||
|   CJSON(syncGroups, if_sync_send["grp"]); | ||||
|   if (if_sync_send[F("twice")]) udpNumRetries = 1; // import setting from 0.13 and earlier | ||||
|   CJSON(udpNumRetries, if_sync_send["ret"]); | ||||
| @@ -412,13 +456,14 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|   CJSON(nodeBroadcastEnabled, if_nodes[F("bcast")]); | ||||
|  | ||||
|   JsonObject if_live = interfaces["live"]; | ||||
|   CJSON(receiveDirect, if_live["en"]); | ||||
|   CJSON(receiveDirect, if_live["en"]);  // UDP/Hyperion realtime | ||||
|   CJSON(useMainSegmentOnly, if_live[F("mso")]); | ||||
|   CJSON(realtimeRespectLedMaps, if_live[F("rlm")]); | ||||
|   CJSON(e131Port, if_live["port"]); // 5568 | ||||
|   if (e131Port == DDP_DEFAULT_PORT) e131Port = E131_DEFAULT_PORT; // prevent double DDP port allocation | ||||
|   CJSON(e131Multicast, if_live[F("mc")]); | ||||
|  | ||||
|   JsonObject if_live_dmx = if_live[F("dmx")]; | ||||
|   JsonObject if_live_dmx = if_live["dmx"]; | ||||
|   CJSON(e131Universe, if_live_dmx[F("uni")]); | ||||
|   CJSON(e131SkipOutOfSequence, if_live_dmx[F("seqskip")]); | ||||
|   CJSON(DMXAddress, if_live_dmx[F("addr")]); | ||||
| @@ -456,13 +501,6 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|   CJSON(retainMqttMsg, if_mqtt[F("rtn")]); | ||||
| #endif | ||||
|  | ||||
| #ifndef WLED_DISABLE_ESPNOW | ||||
|   JsonObject remote = doc["remote"]; | ||||
|   CJSON(enable_espnow_remote, remote[F("remote_enabled")]); | ||||
|   getStringFromJson(linked_remote, remote[F("linked_remote")], 13); | ||||
| #endif | ||||
|  | ||||
|  | ||||
| #ifndef WLED_DISABLE_HUESYNC | ||||
|   JsonObject if_hue = interfaces["hue"]; | ||||
|   CJSON(huePollingEnabled, if_hue["en"]); | ||||
| @@ -499,6 +537,7 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|   CJSON(analogClock12pixel, ol[F("o12pix")]); | ||||
|   CJSON(analogClock5MinuteMarks, ol[F("o5m")]); | ||||
|   CJSON(analogClockSecondsTrail, ol[F("osec")]); | ||||
|   CJSON(analogClockSolidBlack, ol[F("osb")]); | ||||
|  | ||||
|   //timed macro rules | ||||
|   JsonObject tm = doc[F("timers")]; | ||||
| @@ -588,6 +627,9 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | ||||
|   return (doc["sv"] | true); | ||||
| } | ||||
|  | ||||
|  | ||||
| static const char s_cfg_json[] PROGMEM = "/cfg.json"; | ||||
|  | ||||
| void deserializeConfigFromFS() { | ||||
|   bool success = deserializeConfigSec(); | ||||
|   if (!success) { //if file does not exist, try reading from EEPROM | ||||
| @@ -601,7 +643,7 @@ void deserializeConfigFromFS() { | ||||
|  | ||||
|   DEBUG_PRINTLN(F("Reading settings from /cfg.json...")); | ||||
|  | ||||
|   success = readObjectFromFile("/cfg.json", nullptr, &doc); | ||||
|   success = readObjectFromFile(s_cfg_json, nullptr, pDoc); | ||||
|   if (!success) { // if file does not exist, optionally try reading from EEPROM and then save defaults to FS | ||||
|     releaseJSONBufferLock(); | ||||
|     #ifdef WLED_ADD_EEPROM_SUPPORT | ||||
| @@ -622,7 +664,8 @@ void deserializeConfigFromFS() { | ||||
|  | ||||
|   // NOTE: This routine deserializes *and* applies the configuration | ||||
|   //       Therefore, must also initialize ethernet from this function | ||||
|   bool needsSave = deserializeConfig(doc.as<JsonObject>(), true); | ||||
|   JsonObject root = pDoc->as<JsonObject>(); | ||||
|   bool needsSave = deserializeConfig(root, true); | ||||
|   releaseJSONBufferLock(); | ||||
|  | ||||
|   if (needsSave) serializeConfig(); // usermods required new parameters | ||||
| @@ -635,39 +678,47 @@ void serializeConfig() { | ||||
|  | ||||
|   if (!requestJSONBufferLock(2)) return; | ||||
|  | ||||
|   JsonArray rev = doc.createNestedArray("rev"); | ||||
|   JsonObject root = pDoc->to<JsonObject>(); | ||||
|  | ||||
|   JsonArray rev = root.createNestedArray("rev"); | ||||
|   rev.add(1); //major settings revision | ||||
|   rev.add(0); //minor settings revision | ||||
|  | ||||
|   doc[F("vid")] = VERSION; | ||||
|   root[F("vid")] = VERSION; | ||||
|  | ||||
|   JsonObject id = doc.createNestedObject("id"); | ||||
|   JsonObject id = root.createNestedObject("id"); | ||||
|   id[F("mdns")] = cmDNS; | ||||
|   id[F("name")] = serverDescription; | ||||
|   id[F("inv")] = alexaInvocationName; | ||||
| #ifdef WLED_ENABLE_SIMPLE_UI | ||||
|   id[F("sui")] = simplifiedUI; | ||||
|  | ||||
|   JsonObject nw = root.createNestedObject("nw"); | ||||
| #ifndef WLED_DISABLE_ESPNOW | ||||
|   nw[F("espnow")] = enableESPNow; | ||||
|   nw[F("linked_remote")] = linked_remote; | ||||
| #endif | ||||
|  | ||||
|   JsonObject nw = doc.createNestedObject("nw"); | ||||
|  | ||||
|   JsonArray nw_ins = nw.createNestedArray("ins"); | ||||
|  | ||||
|   JsonObject nw_ins_0 = nw_ins.createNestedObject(); | ||||
|   nw_ins_0[F("ssid")] = clientSSID; | ||||
|   nw_ins_0[F("pskl")] = strlen(clientPass); | ||||
|  | ||||
|   JsonArray nw_ins_0_ip = nw_ins_0.createNestedArray("ip"); | ||||
|   JsonArray nw_ins_0_gw = nw_ins_0.createNestedArray("gw"); | ||||
|   JsonArray nw_ins_0_sn = nw_ins_0.createNestedArray("sn"); | ||||
|  | ||||
|   for (byte i = 0; i < 4; i++) { | ||||
|     nw_ins_0_ip.add(staticIP[i]); | ||||
|     nw_ins_0_gw.add(staticGateway[i]); | ||||
|     nw_ins_0_sn.add(staticSubnet[i]); | ||||
|   for (size_t n = 0; n < multiWiFi.size(); n++) { | ||||
|     JsonObject wifi = nw_ins.createNestedObject(); | ||||
|     wifi[F("ssid")] = multiWiFi[n].clientSSID; | ||||
|     wifi[F("pskl")] = strlen(multiWiFi[n].clientPass); | ||||
|     JsonArray wifi_ip = wifi.createNestedArray("ip"); | ||||
|     JsonArray wifi_gw = wifi.createNestedArray("gw"); | ||||
|     JsonArray wifi_sn = wifi.createNestedArray("sn"); | ||||
|     for (size_t i = 0; i < 4; i++) { | ||||
|       wifi_ip.add(multiWiFi[n].staticIP[i]); | ||||
|       wifi_gw.add(multiWiFi[n].staticGW[i]); | ||||
|       wifi_sn.add(multiWiFi[n].staticSN[i]); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   JsonObject ap = doc.createNestedObject("ap"); | ||||
|   JsonArray dns = nw.createNestedArray(F("dns")); | ||||
|   for (size_t i = 0; i < 4; i++) { | ||||
|     dns.add(dnsAddress[i]); | ||||
|   } | ||||
|  | ||||
|   JsonObject ap = root.createNestedObject("ap"); | ||||
|   ap[F("ssid")] = apSSID; | ||||
|   ap[F("pskl")] = strlen(apPass); | ||||
|   ap[F("chan")] = apChannel; | ||||
| @@ -680,12 +731,12 @@ void serializeConfig() { | ||||
|   ap_ip.add(2); | ||||
|   ap_ip.add(1); | ||||
|  | ||||
|   JsonObject wifi = doc.createNestedObject("wifi"); | ||||
|   JsonObject wifi = root.createNestedObject(F("wifi")); | ||||
|   wifi[F("sleep")] = !noWifiSleep; | ||||
|   wifi[F("phy")] = force802_3g; | ||||
|  | ||||
|   #ifdef WLED_USE_ETHERNET | ||||
|   JsonObject ethernet = doc.createNestedObject("eth"); | ||||
|   JsonObject ethernet = root.createNestedObject("eth"); | ||||
|   ethernet["type"] = ethernetType; | ||||
|   if (ethernetType != WLED_ETH_NONE && ethernetType < WLED_NUM_ETH_TYPES) { | ||||
|     JsonArray pins = ethernet.createNestedArray("pin"); | ||||
| @@ -708,12 +759,12 @@ void serializeConfig() { | ||||
|   } | ||||
|   #endif | ||||
|  | ||||
|   JsonObject hw = doc.createNestedObject("hw"); | ||||
|   JsonObject hw = root.createNestedObject(F("hw")); | ||||
|  | ||||
|   JsonObject hw_led = hw.createNestedObject("led"); | ||||
|   hw_led[F("total")] = strip.getLengthTotal(); //no longer read, but provided for compatibility on downgrade | ||||
|   hw_led[F("maxpwr")] = strip.ablMilliampsMax; | ||||
|   hw_led[F("ledma")] = strip.milliampsPerLed; | ||||
|   hw_led[F("total")] = strip.getLengthTotal(); //provided for compatibility on downgrade and per-output ABL | ||||
|   hw_led[F("maxpwr")] = BusManager::ablMilliampsMax(); | ||||
|   hw_led[F("ledma")] = 0; // no longer used | ||||
|   hw_led["cct"] = correctWB; | ||||
|   hw_led[F("cr")] = cctFromRgb; | ||||
|   hw_led[F("cb")] = strip.cctBlending; | ||||
| @@ -743,8 +794,8 @@ void serializeConfig() { | ||||
|  | ||||
|   JsonArray hw_led_ins = hw_led.createNestedArray("ins"); | ||||
|  | ||||
|   for (uint8_t s = 0; s < busses.getNumBusses(); s++) { | ||||
|     Bus *bus = busses.getBus(s); | ||||
|   for (uint8_t s = 0; s < BusManager::getNumBusses(); s++) { | ||||
|     Bus *bus = BusManager::getBus(s); | ||||
|     if (!bus || bus->getLength()==0) break; | ||||
|     JsonObject ins = hw_led_ins.createNestedObject(); | ||||
|     ins["start"] = bus->getStart(); | ||||
| @@ -760,10 +811,12 @@ void serializeConfig() { | ||||
|     ins["ref"] = bus->isOffRefreshRequired(); | ||||
|     ins[F("rgbwm")] = bus->getAutoWhiteMode(); | ||||
|     ins[F("freq")] = bus->getFrequency(); | ||||
|     ins[F("maxpwr")] = bus->getMaxCurrent(); | ||||
|     ins[F("ledma")] = bus->getLEDCurrent(); | ||||
|   } | ||||
|  | ||||
|   JsonArray hw_com = hw.createNestedArray(F("com")); | ||||
|   const ColorOrderMap& com = busses.getColorOrderMap(); | ||||
|   const ColorOrderMap& com = BusManager::getColorOrderMap(); | ||||
|   for (uint8_t s = 0; s < com.count(); s++) { | ||||
|     const ColorOrderMapEntry *entry = com.get(s); | ||||
|     if (!entry) break; | ||||
| @@ -796,8 +849,10 @@ void serializeConfig() { | ||||
|   hw_btn["mqtt"] = buttonPublishMqtt; | ||||
|  | ||||
|   JsonObject hw_ir = hw.createNestedObject("ir"); | ||||
|   #ifndef WLED_DISABLE_INFRARED | ||||
|   hw_ir["pin"] = irPin; | ||||
|   hw_ir["type"] = irEnabled;  // the byte 'irEnabled' does contain the IR-Remote Type ( 0=disabled ) | ||||
|   #endif | ||||
|   hw_ir["sel"] = irApplyToAllSelected; | ||||
|  | ||||
|   JsonObject hw_relay = hw.createNestedObject(F("relay")); | ||||
| @@ -818,7 +873,7 @@ void serializeConfig() { | ||||
|   //JsonObject hw_status = hw.createNestedObject("status"); | ||||
|   //hw_status["pin"] = -1; | ||||
|  | ||||
|   JsonObject light = doc.createNestedObject(F("light")); | ||||
|   JsonObject light = root.createNestedObject(F("light")); | ||||
|   light[F("scale-bri")] = briMultiplier; | ||||
|   light[F("pal-mode")] = strip.paletteBlend; | ||||
|   light[F("aseg")] = autoSegments; | ||||
| @@ -834,6 +889,7 @@ void serializeConfig() { | ||||
|   light_tr["dur"] = transitionDelayDefault / 100; | ||||
|   light_tr["pal"] = strip.paletteFade; | ||||
|   light_tr[F("rpc")] = randomPaletteChangeTime; | ||||
|   light_tr[F("hrp")] = useHarmonicRandomPalette; | ||||
|  | ||||
|   JsonObject light_nl = light.createNestedObject("nl"); | ||||
|   light_nl["mode"] = nightlightMode; | ||||
| @@ -841,18 +897,22 @@ void serializeConfig() { | ||||
|   light_nl[F("tbri")] = nightlightTargetBri; | ||||
|   light_nl["macro"] = macroNl; | ||||
|  | ||||
|   JsonObject def = doc.createNestedObject("def"); | ||||
|   JsonObject def = root.createNestedObject("def"); | ||||
|   def["ps"] = bootPreset; | ||||
|   def["on"] = turnOnAtBoot; | ||||
|   def["bri"] = briS; | ||||
|  | ||||
|   JsonObject interfaces = doc.createNestedObject("if"); | ||||
|   JsonObject interfaces = root.createNestedObject("if"); | ||||
|  | ||||
|   JsonObject if_sync = interfaces.createNestedObject("sync"); | ||||
|   if_sync[F("port0")] = udpPort; | ||||
|   if_sync[F("port1")] = udpPort2; | ||||
|  | ||||
|   JsonObject if_sync_recv = if_sync.createNestedObject("recv"); | ||||
| #ifndef WLED_DISABLE_ESPNOW | ||||
|   if_sync[F("espnow")] = useESPNowSync; | ||||
| #endif | ||||
|  | ||||
|   JsonObject if_sync_recv = if_sync.createNestedObject(F("recv")); | ||||
|   if_sync_recv["bri"] = receiveNotificationBrightness; | ||||
|   if_sync_recv["col"] = receiveNotificationColor; | ||||
|   if_sync_recv["fx"]  = receiveNotificationEffects; | ||||
| @@ -860,12 +920,12 @@ void serializeConfig() { | ||||
|   if_sync_recv["seg"] = receiveSegmentOptions; | ||||
|   if_sync_recv["sb"]  = receiveSegmentBounds; | ||||
|  | ||||
|   JsonObject if_sync_send = if_sync.createNestedObject("send"); | ||||
|   JsonObject if_sync_send = if_sync.createNestedObject(F("send")); | ||||
|   if_sync_send["en"] = sendNotifications; | ||||
|   if_sync_send[F("dir")] = notifyDirect; | ||||
|   if_sync_send["btn"] = notifyButton; | ||||
|   if_sync_send["va"] = notifyAlexa; | ||||
|   if_sync_send["hue"] = notifyHue; | ||||
|   if_sync_send["macro"] = notifyMacro; | ||||
|   if_sync_send["grp"] = syncGroups; | ||||
|   if_sync_send["ret"] = udpNumRetries; | ||||
|  | ||||
| @@ -874,8 +934,9 @@ void serializeConfig() { | ||||
|   if_nodes[F("bcast")] = nodeBroadcastEnabled; | ||||
|  | ||||
|   JsonObject if_live = interfaces.createNestedObject("live"); | ||||
|   if_live["en"] = receiveDirect; | ||||
|   if_live["en"] = receiveDirect; // UDP/Hyperion realtime | ||||
|   if_live[F("mso")] = useMainSegmentOnly; | ||||
|   if_live[F("rlm")] = realtimeRespectLedMaps; | ||||
|   if_live["port"] = e131Port; | ||||
|   if_live[F("mc")] = e131Multicast; | ||||
|  | ||||
| @@ -918,20 +979,13 @@ void serializeConfig() { | ||||
|   if_mqtt_topics[F("group")] = mqttGroupTopic; | ||||
| #endif | ||||
|  | ||||
| #ifndef WLED_DISABLE_ESPNOW | ||||
|   JsonObject remote = doc.createNestedObject(F("remote")); | ||||
|   remote[F("remote_enabled")] = enable_espnow_remote; | ||||
|   remote[F("linked_remote")] = linked_remote; | ||||
| #endif | ||||
|  | ||||
|  | ||||
| #ifndef WLED_DISABLE_HUESYNC | ||||
|   JsonObject if_hue = interfaces.createNestedObject("hue"); | ||||
|   if_hue["en"] = huePollingEnabled; | ||||
|   if_hue["id"] = huePollLightId; | ||||
|   if_hue[F("iv")] = huePollIntervalMs / 100; | ||||
|  | ||||
|   JsonObject if_hue_recv = if_hue.createNestedObject("recv"); | ||||
|   JsonObject if_hue_recv = if_hue.createNestedObject(F("recv")); | ||||
|   if_hue_recv["on"] = hueApplyOnOff; | ||||
|   if_hue_recv["bri"] = hueApplyBri; | ||||
|   if_hue_recv["col"] = hueApplyColor; | ||||
| @@ -951,7 +1005,7 @@ void serializeConfig() { | ||||
|   if_ntp[F("ln")] = longitude; | ||||
|   if_ntp[F("lt")] = latitude; | ||||
|  | ||||
|   JsonObject ol = doc.createNestedObject("ol"); | ||||
|   JsonObject ol = root.createNestedObject("ol"); | ||||
|   ol[F("clock")] = overlayCurrent; | ||||
|   ol[F("cntdwn")] = countdownMode; | ||||
|  | ||||
| @@ -960,8 +1014,9 @@ void serializeConfig() { | ||||
|   ol[F("o12pix")] = analogClock12pixel; | ||||
|   ol[F("o5m")] = analogClock5MinuteMarks; | ||||
|   ol[F("osec")] = analogClockSecondsTrail; | ||||
|   ol[F("osb")] = analogClockSolidBlack; | ||||
|  | ||||
|   JsonObject timers = doc.createNestedObject(F("timers")); | ||||
|   JsonObject timers = root.createNestedObject(F("timers")); | ||||
|  | ||||
|   JsonObject cntdwn = timers.createNestedObject(F("cntdwn")); | ||||
|   JsonArray goal = cntdwn.createNestedArray(F("goal")); | ||||
| @@ -989,14 +1044,14 @@ void serializeConfig() { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   JsonObject ota = doc.createNestedObject("ota"); | ||||
|   JsonObject ota = root.createNestedObject("ota"); | ||||
|   ota[F("lock")] = otaLock; | ||||
|   ota[F("lock-wifi")] = wifiLock; | ||||
|   ota[F("pskl")] = strlen(otaPass); | ||||
|   ota[F("aota")] = aOtaEnabled; | ||||
|  | ||||
|   #ifdef WLED_ENABLE_DMX | ||||
|   JsonObject dmx = doc.createNestedObject("dmx"); | ||||
|   JsonObject dmx = root.createNestedObject("dmx"); | ||||
|   dmx[F("chan")] = DMXChannels; | ||||
|   dmx[F("gap")] = DMXGap; | ||||
|   dmx["start"] = DMXStart; | ||||
| @@ -1010,36 +1065,50 @@ void serializeConfig() { | ||||
|   dmx[F("e131proxy")] = e131ProxyUniverse; | ||||
|   #endif | ||||
|  | ||||
|   JsonObject usermods_settings = doc.createNestedObject("um"); | ||||
|   JsonObject usermods_settings = root.createNestedObject("um"); | ||||
|   usermods.addToConfig(usermods_settings); | ||||
|  | ||||
|   File f = WLED_FS.open("/cfg.json", "w"); | ||||
|   if (f) serializeJson(doc, f); | ||||
|   File f = WLED_FS.open(FPSTR(s_cfg_json), "w"); | ||||
|   if (f) serializeJson(root, f); | ||||
|   f.close(); | ||||
|   releaseJSONBufferLock(); | ||||
|  | ||||
|   doSerializeConfig = false; | ||||
| } | ||||
|  | ||||
|  | ||||
| static const char s_wsec_json[] PROGMEM = "/wsec.json"; | ||||
|  | ||||
| //settings in /wsec.json, not accessible via webserver, for passwords and tokens | ||||
| bool deserializeConfigSec() { | ||||
|   DEBUG_PRINTLN(F("Reading settings from /wsec.json...")); | ||||
|  | ||||
|   if (!requestJSONBufferLock(3)) return false; | ||||
|  | ||||
|   bool success = readObjectFromFile("/wsec.json", nullptr, &doc); | ||||
|   bool success = readObjectFromFile(s_wsec_json, nullptr, pDoc); | ||||
|   if (!success) { | ||||
|     releaseJSONBufferLock(); | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   JsonObject nw_ins_0 = doc["nw"]["ins"][0]; | ||||
|   getStringFromJson(clientPass, nw_ins_0["psk"], 65); | ||||
|   JsonObject root = pDoc->as<JsonObject>(); | ||||
|  | ||||
|   JsonObject ap = doc["ap"]; | ||||
|   size_t n = 0; | ||||
|   JsonArray nw_ins = root["nw"]["ins"]; | ||||
|   if (!nw_ins.isNull()) { | ||||
|     if (nw_ins.size() > 1 && nw_ins.size() > multiWiFi.size()) multiWiFi.resize(nw_ins.size()); // resize constructs objects while resizing | ||||
|     for (JsonObject wifi : nw_ins) { | ||||
|       char pw[65] = ""; | ||||
|       getStringFromJson(pw, wifi["psk"], 65); | ||||
|       strlcpy(multiWiFi[n].clientPass, pw, 65); | ||||
|       if (++n >= WLED_MAX_WIFI_COUNT) break; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   JsonObject ap = root["ap"]; | ||||
|   getStringFromJson(apPass, ap["psk"] , 65); | ||||
|  | ||||
|   [[maybe_unused]] JsonObject interfaces = doc["if"]; | ||||
|   [[maybe_unused]] JsonObject interfaces = root["if"]; | ||||
|  | ||||
| #ifdef WLED_ENABLE_MQTT | ||||
|   JsonObject if_mqtt = interfaces["mqtt"]; | ||||
| @@ -1050,10 +1119,10 @@ bool deserializeConfigSec() { | ||||
|   getStringFromJson(hueApiKey, interfaces["hue"][F("key")], 47); | ||||
| #endif | ||||
|  | ||||
|   getStringFromJson(settingsPIN, doc["pin"], 5); | ||||
|   getStringFromJson(settingsPIN, root["pin"], 5); | ||||
|   correctPIN = !strlen(settingsPIN); | ||||
|  | ||||
|   JsonObject ota = doc["ota"]; | ||||
|   JsonObject ota = root["ota"]; | ||||
|   getStringFromJson(otaPass, ota[F("pwd")], 33); | ||||
|   CJSON(otaLock, ota[F("lock")]); | ||||
|   CJSON(wifiLock, ota[F("lock-wifi")]); | ||||
| @@ -1068,17 +1137,20 @@ void serializeConfigSec() { | ||||
|  | ||||
|   if (!requestJSONBufferLock(4)) return; | ||||
|  | ||||
|   JsonObject nw = doc.createNestedObject("nw"); | ||||
|   JsonObject root = pDoc->to<JsonObject>(); | ||||
|  | ||||
|   JsonObject nw = root.createNestedObject("nw"); | ||||
|  | ||||
|   JsonArray nw_ins = nw.createNestedArray("ins"); | ||||
|   for (size_t i = 0; i < multiWiFi.size(); i++) { | ||||
|     JsonObject wifi = nw_ins.createNestedObject(); | ||||
|     wifi[F("psk")] = multiWiFi[i].clientPass; | ||||
|   } | ||||
|  | ||||
|   JsonObject nw_ins_0 = nw_ins.createNestedObject(); | ||||
|   nw_ins_0["psk"] = clientPass; | ||||
|  | ||||
|   JsonObject ap = doc.createNestedObject("ap"); | ||||
|   JsonObject ap = root.createNestedObject("ap"); | ||||
|   ap["psk"] = apPass; | ||||
|  | ||||
|   [[maybe_unused]] JsonObject interfaces = doc.createNestedObject("if"); | ||||
|   [[maybe_unused]] JsonObject interfaces = root.createNestedObject("if"); | ||||
| #ifdef WLED_ENABLE_MQTT | ||||
|   JsonObject if_mqtt = interfaces.createNestedObject("mqtt"); | ||||
|   if_mqtt["psk"] = mqttPass; | ||||
| @@ -1088,16 +1160,16 @@ void serializeConfigSec() { | ||||
|   if_hue[F("key")] = hueApiKey; | ||||
| #endif | ||||
|  | ||||
|   doc["pin"] = settingsPIN; | ||||
|   root["pin"] = settingsPIN; | ||||
|  | ||||
|   JsonObject ota = doc.createNestedObject("ota"); | ||||
|   JsonObject ota = root.createNestedObject("ota"); | ||||
|   ota[F("pwd")] = otaPass; | ||||
|   ota[F("lock")] = otaLock; | ||||
|   ota[F("lock-wifi")] = wifiLock; | ||||
|   ota[F("aota")] = aOtaEnabled; | ||||
|  | ||||
|   File f = WLED_FS.open("/wsec.json", "w"); | ||||
|   if (f) serializeJson(doc, f); | ||||
|   File f = WLED_FS.open(FPSTR(s_wsec_json), "w"); | ||||
|   if (f) serializeJson(root, f); | ||||
|   f.close(); | ||||
|   releaseJSONBufferLock(); | ||||
| } | ||||
|   | ||||
| @@ -91,6 +91,115 @@ void setRandomColor(byte* rgb) | ||||
|   colorHStoRGB(lastRandomIndex*256,255,rgb); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * generates a random palette based on harmonic color theory | ||||
|  * takes a base palette as the input, it will choose one color of the base palette and keep it | ||||
|  */ | ||||
| CRGBPalette16 generateHarmonicRandomPalette(CRGBPalette16 &basepalette) | ||||
| { | ||||
|   CHSV palettecolors[4]; //array of colors for the new palette | ||||
|   uint8_t keepcolorposition = random8(4); //color position of current random palette to keep | ||||
|   palettecolors[keepcolorposition] = rgb2hsv_approximate(basepalette.entries[keepcolorposition*5]); //read one of the base colors of the current palette | ||||
|   palettecolors[keepcolorposition].hue += random8(10)-5; // +/- 5 randomness of base color | ||||
|   //generate 4 saturation and brightness value numbers | ||||
|   //only one saturation is allowed to be below 200 creating mostly vibrant colors | ||||
|   //only one brightness value number is allowed below 200, creating mostly bright palettes | ||||
|  | ||||
|   for (int i = 0; i < 3; i++) { //generate three high values | ||||
|     palettecolors[i].saturation = random8(200,255); | ||||
|     palettecolors[i].value = random8(220,255); | ||||
|   } | ||||
|   //allow one to be lower | ||||
|   palettecolors[3].saturation = random8(20,255); | ||||
|   palettecolors[3].value = random8(80,255); | ||||
|  | ||||
|   //shuffle the arrays | ||||
|   for (int i = 3; i > 0; i--) { | ||||
|     std::swap(palettecolors[i].saturation, palettecolors[random8(i + 1)].saturation); | ||||
|     std::swap(palettecolors[i].value, palettecolors[random8(i + 1)].value); | ||||
|   } | ||||
|  | ||||
|   //now generate three new hues based off of the hue of the chosen current color | ||||
|   uint8_t basehue = palettecolors[keepcolorposition].hue; | ||||
|   uint8_t harmonics[3]; //hues that are harmonic but still a little random | ||||
|   uint8_t type = random8(5); //choose a harmony type | ||||
|  | ||||
|   switch (type) { | ||||
|     case 0: // analogous | ||||
|       harmonics[0] = basehue + random8(30, 50); | ||||
|       harmonics[1] = basehue + random8(10, 30); | ||||
|       harmonics[2] = basehue - random8(10, 30); | ||||
|       break; | ||||
|  | ||||
|     case 1: // triadic | ||||
|       harmonics[0] = basehue + 113 + random8(15); | ||||
|       harmonics[1] = basehue + 233 + random8(15); | ||||
|       harmonics[2] = basehue -7 + random8(15); | ||||
|       break; | ||||
|  | ||||
|     case 2: // split-complementary | ||||
|       harmonics[0] = basehue + 145 + random8(10); | ||||
|       harmonics[1] = basehue + 205 + random8(10); | ||||
|       harmonics[2] = basehue - 5 + random8(10); | ||||
|       break; | ||||
|      | ||||
|     case 3: // square | ||||
|       harmonics[0] = basehue + 85 + random8(10); | ||||
|       harmonics[1] = basehue + 175 + random8(10); | ||||
|       harmonics[2] = basehue + 265 + random8(10); | ||||
|      break; | ||||
|  | ||||
|     case 4: // tetradic | ||||
|       harmonics[0] = basehue + 80 + random8(20); | ||||
|       harmonics[1] = basehue + 170 + random8(20); | ||||
|       harmonics[2] = basehue + random8(30)-15; | ||||
|      break; | ||||
|   } | ||||
|  | ||||
|   if (random8() < 128) { | ||||
|     //50:50 chance of shuffling hues or keep the color order | ||||
|     for (int i = 2; i > 0; i--) { | ||||
|       std::swap(harmonics[i], harmonics[random8(i + 1)]); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //now set the hues | ||||
|   int j = 0; | ||||
|   for (int i = 0; i < 4; i++) { | ||||
|     if (i==keepcolorposition) continue; //skip the base color | ||||
|     palettecolors[i].hue = harmonics[j]; | ||||
|     j++; | ||||
|   } | ||||
|  | ||||
|   bool makepastelpalette = false; | ||||
|   if (random8() < 25) { //~10% chance of desaturated 'pastel' colors | ||||
|     makepastelpalette = true; | ||||
|   } | ||||
|  | ||||
|   //apply saturation & gamma correction | ||||
|   CRGB RGBpalettecolors[4]; | ||||
|   for (int i = 0; i < 4; i++) { | ||||
|     if (makepastelpalette && palettecolors[i].saturation > 180) {  | ||||
|       palettecolors[i].saturation -= 160; //desaturate all four colors | ||||
|     }     | ||||
|     RGBpalettecolors[i] = (CRGB)palettecolors[i]; //convert to RGB | ||||
|     RGBpalettecolors[i] = gamma32(((uint32_t)RGBpalettecolors[i]) & 0x00FFFFFFU); //strip alpha from CRGB | ||||
|   } | ||||
|  | ||||
|   return CRGBPalette16(RGBpalettecolors[0], | ||||
|                        RGBpalettecolors[1], | ||||
|                        RGBpalettecolors[2], | ||||
|                        RGBpalettecolors[3]); | ||||
| } | ||||
|  | ||||
| CRGBPalette16 generateRandomPalette(void)  //generate fully random palette | ||||
| { | ||||
|   return CRGBPalette16(CHSV(random8(), random8(160, 255), random8(128, 255)), | ||||
|                        CHSV(random8(), random8(160, 255), random8(128, 255)), | ||||
|                        CHSV(random8(), random8(160, 255), random8(128, 255)), | ||||
|                        CHSV(random8(), random8(160, 255), random8(128, 255))); | ||||
| } | ||||
|  | ||||
| void colorHStoRGB(uint16_t hue, byte sat, byte* rgb) //hue, sat to rgb | ||||
| { | ||||
|   float h = ((float)hue)/65535.0f; | ||||
|   | ||||
| @@ -7,14 +7,35 @@ | ||||
|  | ||||
| #define GRADIENT_PALETTE_COUNT 58 | ||||
|  | ||||
| // You can define custom product info from build flags. | ||||
| // This is useful to allow API consumer to identify what type of WLED version | ||||
| // they are interacting with. Be aware that changing this might cause some third | ||||
| // party API consumers to consider this as a non-WLED device since the values | ||||
| // returned by the API and by MQTT will no longer be default. However, most | ||||
| // third party only uses mDNS to validate, so this is generally fine to change. | ||||
| // For example, Home Assistant will still work fine even with this value changed. | ||||
| // Use like this: | ||||
| // -D WLED_BRAND="\"Custom Brand\"" | ||||
| // -D WLED_PRODUCT_NAME="\"Custom Product\"" | ||||
| #ifndef WLED_BRAND | ||||
|   #define WLED_BRAND "WLED" | ||||
| #endif | ||||
| #ifndef WLED_PRODUCT_NAME | ||||
|   #define WLED_PRODUCT_NAME "FOSS" | ||||
| #endif | ||||
|  | ||||
| //Defaults | ||||
| #define DEFAULT_CLIENT_SSID "Your_Network" | ||||
| #define DEFAULT_AP_SSID     "WLED-AP" | ||||
| #define DEFAULT_AP_SSID     WLED_BRAND "-AP" | ||||
| #define DEFAULT_AP_PASS     "wled1234" | ||||
| #define DEFAULT_OTA_PASS    "wledota" | ||||
| #define DEFAULT_MDNS_NAME   "x" | ||||
|  | ||||
| //increase if you need more | ||||
| #ifndef WLED_MAX_WIFI_COUNT | ||||
|   #define WLED_MAX_WIFI_COUNT 3 | ||||
| #endif | ||||
|  | ||||
| #ifndef WLED_MAX_USERMODS | ||||
|   #ifdef ESP8266 | ||||
|     #define WLED_MAX_USERMODS 4 | ||||
| @@ -72,6 +93,11 @@ | ||||
|   #else | ||||
|     #define WLED_MAX_BUTTONS 4 | ||||
|   #endif | ||||
| #else | ||||
|   #if WLED_MAX_BUTTONS < 2 | ||||
|     #undef WLED_MAX_BUTTONS | ||||
|     #define WLED_MAX_BUTTONS 2 | ||||
|   #endif | ||||
| #endif | ||||
|  | ||||
| #ifdef ESP8266 | ||||
| @@ -152,12 +178,19 @@ | ||||
| #define USERMOD_ID_INTERNAL_TEMPERATURE  42     //Usermod "usermod_internal_temperature.h" | ||||
| #define USERMOD_ID_LDR_DUSK_DAWN         43     //Usermod "usermod_LDR_Dusk_Dawn_v2.h" | ||||
| #define USERMOD_ID_STAIRWAY_WIPE         44     //Usermod "stairway-wipe-usermod-v2.h" | ||||
| #define USERMOD_ID_ANIMARTRIX            45     //Usermod "usermod_v2_animartrix.h" | ||||
| #define USERMOD_ID_HTTP_PULL_LIGHT_CONTROL 46   //usermod "usermod_v2_HttpPullLightControl.h" | ||||
| #define USERMOD_ID_TETRISAI              47     //Usermod "usermod_v2_tetris.h" | ||||
|  | ||||
| //Access point behavior | ||||
| #define AP_BEHAVIOR_BOOT_NO_CONN          0     //Open AP when no connection after boot | ||||
| #define AP_BEHAVIOR_NO_CONN               1     //Open when no connection (either after boot or if connection is lost) | ||||
| #define AP_BEHAVIOR_ALWAYS                2     //Always open | ||||
| #define AP_BEHAVIOR_BUTTON_ONLY           3     //Only when button pressed for 6 sec | ||||
| #define AP_BEHAVIOR_TEMPORARY             4     //Open AP when no connection after boot but only temporary | ||||
| #ifndef WLED_AP_TIMEOUT | ||||
|   #define WLED_AP_TIMEOUT            300000     //Temporary AP timeout | ||||
| #endif | ||||
|  | ||||
| //Notifier callMode | ||||
| #define CALL_MODE_INIT           0     //no updates on init, can be used to disable updates | ||||
| @@ -226,7 +259,7 @@ | ||||
|  | ||||
| #define TYPE_NONE                 0            //light is not configured | ||||
| #define TYPE_RESERVED             1            //unused. Might indicate a "virtual" light | ||||
| //Digital types (data pin only) (16-31) | ||||
| //Digital types (data pin only) (16-39) | ||||
| #define TYPE_WS2812_1CH          18            //white-only chips (1 channel per IC) (unused) | ||||
| #define TYPE_WS2812_1CH_X3       19            //white-only chips (3 channels per IC) | ||||
| #define TYPE_WS2812_2CH_X3       20            //CCT chips (1st IC controls WW + CW of 1st zone and CW of 2nd zone, 2nd IC controls WW of 2nd zone and WW + CW of 3rd zone) | ||||
| @@ -236,11 +269,12 @@ | ||||
| #define TYPE_WS2811_400KHZ       24            //half-speed WS2812 protocol, used by very old WS2811 units | ||||
| #define TYPE_TM1829              25 | ||||
| #define TYPE_UCS8903             26 | ||||
| #define TYPE_UCS8904             29 | ||||
| #define TYPE_APA106              27 | ||||
| #define TYPE_UCS8904             29            //first RGBW digital type (hardcoded in busmanager.cpp, memUsage()) | ||||
| #define TYPE_SK6812_RGBW         30 | ||||
| #define TYPE_TM1814              31 | ||||
| //"Analog" types (PWM) (32-47) | ||||
| #define TYPE_ONOFF               40            //binary output (relays etc.) | ||||
| //"Analog" types (40-47) | ||||
| #define TYPE_ONOFF               40            //binary output (relays etc.; NOT PWM) | ||||
| #define TYPE_ANALOG_1CH          41            //single channel PWM. Uses value of brightest RGBW channel | ||||
| #define TYPE_ANALOG_2CH          42            //analog WW + CW | ||||
| #define TYPE_ANALOG_3CH          43            //analog RGB | ||||
| @@ -257,11 +291,15 @@ | ||||
| #define TYPE_NET_E131_RGB        81            //network E131 RGB bus (master broadcast bus, unused) | ||||
| #define TYPE_NET_ARTNET_RGB      82            //network ArtNet RGB bus (master broadcast bus, unused) | ||||
| #define TYPE_NET_DDP_RGBW        88            //network DDP RGBW bus (master broadcast bus) | ||||
| #define TYPE_NET_ARTNET_RGBW     89            //network ArtNet RGB bus (master broadcast bus, unused) | ||||
|  | ||||
| #define IS_DIGITAL(t) ((t) & 0x10) //digital are 16-31 and 48-63 | ||||
| #define IS_PWM(t)     ((t) > 40 && (t) < 46) | ||||
| #define NUM_PWM_PINS(t) ((t) - 40) //for analog PWM 41-45 only | ||||
| #define IS_2PIN(t)      ((t) > 47) | ||||
| #define IS_TYPE_VALID(t) ((t) > 15 && (t) < 128) | ||||
| #define IS_DIGITAL(t)    (((t) > 15 && (t) < 40) || ((t) > 47 && (t) < 64)) //digital are 16-39 and 48-63 | ||||
| #define IS_2PIN(t)       ((t) > 47 && (t) < 64) | ||||
| #define IS_16BIT(t)      ((t) == TYPE_UCS8903 || (t) == TYPE_UCS8904) | ||||
| #define IS_PWM(t)        ((t) > 40 && (t) < 46)     //does not include on/Off type | ||||
| #define NUM_PWM_PINS(t)  ((t) - 40)                 //for analog PWM 41-45 only | ||||
| #define IS_VIRTUAL(t)    ((t) >= 80 && (t) < 96)    //this was a poor choice a better would be 96-111 | ||||
|  | ||||
| //Color orders | ||||
| #define COL_ORDER_GRB             0           //GRB(w),defaut | ||||
| @@ -272,6 +310,10 @@ | ||||
| #define COL_ORDER_GBR             5 | ||||
| #define COL_ORDER_MAX             5 | ||||
|  | ||||
| //ESP-NOW | ||||
| #define ESP_NOW_STATE_UNINIT       0 | ||||
| #define ESP_NOW_STATE_ON           1 | ||||
| #define ESP_NOW_STATE_ERROR        2 | ||||
|  | ||||
| //Button type | ||||
| #define BTN_TYPE_NONE             0 | ||||
| @@ -283,21 +325,23 @@ | ||||
| #define BTN_TYPE_TOUCH            6 | ||||
| #define BTN_TYPE_ANALOG           7 | ||||
| #define BTN_TYPE_ANALOG_INVERTED  8 | ||||
| #define BTN_TYPE_TOUCH_SWITCH     9 | ||||
|  | ||||
| //Ethernet board types | ||||
| #define WLED_NUM_ETH_TYPES       11 | ||||
| #define WLED_NUM_ETH_TYPES        12 | ||||
|  | ||||
| #define WLED_ETH_NONE             0 | ||||
| #define WLED_ETH_WT32_ETH01       1 | ||||
| #define WLED_ETH_ESP32_POE        2 | ||||
| #define WLED_ETH_WESP32           3 | ||||
| #define WLED_ETH_QUINLED          4 | ||||
| #define WLED_ETH_TWILIGHTLORD     5 | ||||
| #define WLED_ETH_ESP32DEUX        6 | ||||
| #define WLED_ETH_ESP32ETHKITVE    7 | ||||
| #define WLED_ETH_QUINLED_OCTA     8 | ||||
| #define WLED_ETH_ABCWLEDV43ETH    9 | ||||
| #define WLED_ETH_SERG74          10 | ||||
| #define WLED_ETH_NONE              0 | ||||
| #define WLED_ETH_WT32_ETH01        1 | ||||
| #define WLED_ETH_ESP32_POE         2 | ||||
| #define WLED_ETH_WESP32            3 | ||||
| #define WLED_ETH_QUINLED           4 | ||||
| #define WLED_ETH_TWILIGHTLORD      5 | ||||
| #define WLED_ETH_ESP32DEUX         6 | ||||
| #define WLED_ETH_ESP32ETHKITVE     7 | ||||
| #define WLED_ETH_QUINLED_OCTA      8 | ||||
| #define WLED_ETH_ABCWLEDV43ETH     9 | ||||
| #define WLED_ETH_SERG74           10 | ||||
| #define WLED_ETH_ESP32_POE_WROVER 11 | ||||
|  | ||||
| //Hue error codes | ||||
| #define HUE_ERROR_INACTIVE        0 | ||||
| @@ -339,8 +383,10 @@ | ||||
| // WLED Error modes | ||||
| #define ERR_NONE         0  // All good :) | ||||
| #define ERR_DENIED       1  // Permission denied | ||||
| #define ERR_EEP_COMMIT   2  // Could not commit to EEPROM (wrong flash layout?) OBSOLETE | ||||
| #define ERR_CONCURRENCY  2  // Conurrency (client active) | ||||
| #define ERR_NOBUF        3  // JSON buffer was not released in time, request cannot be handled at this time | ||||
| #define ERR_NOT_IMPL     4  // Not implemented | ||||
| #define ERR_NORAM        8  // effect RAM depleted | ||||
| #define ERR_JSON         9  // JSON parsing failed (input too large?) | ||||
| #define ERR_FS_BEGIN    10  // Could not init filesystem (no partition?) | ||||
| #define ERR_FS_QUOTA    11  // The FS is full or the maximum file size is reached | ||||
| @@ -408,7 +454,7 @@ | ||||
| #ifdef ESP8266 | ||||
| #define SETTINGS_STACK_BUF_SIZE 2048 | ||||
| #else | ||||
| #define SETTINGS_STACK_BUF_SIZE 3608  // warning: quite a large value for stack | ||||
| #define SETTINGS_STACK_BUF_SIZE 3840  // warning: quite a large value for stack (640 * WLED_MAX_USERMODS) | ||||
| #endif | ||||
|  | ||||
| #ifdef WLED_USE_ETHERNET | ||||
| @@ -446,7 +492,11 @@ | ||||
| #ifdef ESP8266 | ||||
|   #define JSON_BUFFER_SIZE 10240 | ||||
| #else | ||||
|   #define JSON_BUFFER_SIZE 24576 | ||||
|   #if defined(ARDUINO_ARCH_ESP32S2) | ||||
|     #define JSON_BUFFER_SIZE 24576 | ||||
|   #else | ||||
|     #define JSON_BUFFER_SIZE 32767 | ||||
|   #endif | ||||
| #endif | ||||
|  | ||||
| //#define MIN_HEAP_SIZE (8k for AsyncWebServer) | ||||
|   | ||||
| @@ -544,6 +544,7 @@ | ||||
|         const json = await response.json(); | ||||
|         paletteArray.push(json); | ||||
|       } catch (error) { | ||||
|         cpalc--; //remove audio/dynamically generated palettes | ||||
|         console.error(`Error fetching JSON from ${url}: `, error); | ||||
|       } | ||||
|     } | ||||
|   | ||||
| @@ -20,8 +20,8 @@ | ||||
| 	--c-g: #2c1; | ||||
| 	--c-l: #48a; | ||||
| 	--c-y: #a90; | ||||
| 	--t-b: 0.5; | ||||
| 	--c-o: rgba(34, 34, 34, 0.9); | ||||
| 	--t-b: .5; | ||||
| 	--c-o: rgba(34, 34, 34, .9); | ||||
| 	--c-tb : rgba(34, 34, 34, var(--t-b)); | ||||
| 	--c-tba: rgba(102, 102, 102, var(--t-b)); | ||||
| 	--c-tbh: rgba(51, 51, 51, var(--t-b)); | ||||
| @@ -33,7 +33,8 @@ | ||||
| 	--bbp: 9px 0 7px 0; | ||||
| 	--bhd: none; | ||||
| 	--sgp: "block"; | ||||
| 	--bmt: 0px; | ||||
| 	--bmt: 0; | ||||
| 	--sti: 42px; | ||||
| } | ||||
|  | ||||
| html { | ||||
| @@ -88,7 +89,7 @@ a, a:visited { | ||||
| } | ||||
|  | ||||
| button { | ||||
| 	outline: none; | ||||
| 	outline: 0; | ||||
| 	cursor: pointer; | ||||
| } | ||||
|  | ||||
| @@ -219,7 +220,7 @@ button { | ||||
| .pop-c span { | ||||
| 	padding: 2px 6px; | ||||
| } | ||||
|    | ||||
|  | ||||
| .search-icon { | ||||
| 	position: absolute; | ||||
| 	top: 8px; | ||||
| @@ -238,7 +239,7 @@ button { | ||||
| .flr { | ||||
| 	color: var(--c-f); | ||||
| 	transform: rotate(0deg); | ||||
| 	transition: transform 0.3s; | ||||
| 	transition: transform .3s; | ||||
| 	position: absolute; | ||||
| 	top: 0; | ||||
| 	right: 0; | ||||
| @@ -258,13 +259,13 @@ button { | ||||
| #liveview { | ||||
| 	height: 4px; | ||||
| 	width: 100%; | ||||
| 	border: 0px; | ||||
| 	border: 0; | ||||
| } | ||||
|  | ||||
| #liveview2D { | ||||
| 	height: 90%; | ||||
| 	width: 90%; | ||||
| 	border: 0px; | ||||
| 	border: 0; | ||||
| 	position: absolute; | ||||
| 	top: 50%; | ||||
| 	left: 50%; | ||||
| @@ -287,8 +288,8 @@ button { | ||||
| .tab button { | ||||
| 	background-color: transparent; | ||||
| 	float: left; | ||||
| 	border: none; | ||||
| 	transition: color 0.3s, background-color 0.3s; | ||||
| 	border: 0; | ||||
| 	transition: color .3s, background-color .3s; | ||||
| 	font-size: 17px; | ||||
| 	color: var(--c-c); | ||||
| 	min-width: 44px; | ||||
| @@ -301,7 +302,7 @@ button { | ||||
|  | ||||
| .bot button { | ||||
| 	padding: var(--bbp); | ||||
| 	width:25%; | ||||
| 	width: 25%; | ||||
| 	margin: 0; | ||||
| } | ||||
|  | ||||
| @@ -336,18 +337,23 @@ button { | ||||
| 	width: 100%; | ||||
| 	width: calc(100%/var(--n)); | ||||
| 	box-sizing: border-box; | ||||
| 	border: 0px; | ||||
| 	overflow: auto; | ||||
| 	border: 0; | ||||
| 	overflow-y: auto; | ||||
| 	overflow-x: hidden; | ||||
| 	height: 100%; | ||||
| 	overscroll-behavior: none; | ||||
| 	padding: 0 4px; | ||||
| 	-webkit-overflow-scrolling: touch; | ||||
| } | ||||
|  | ||||
| #Segments, #Presets, #Effects, #Colors { | ||||
| 	font-size: 19px; | ||||
| 	padding: 4px 0 0; | ||||
| } | ||||
|  | ||||
| #segutil, #segutil2, #segcont, #putil, #pcont, #pql, #fx, #palw, | ||||
| .fnd { | ||||
| 	max-width: 280px; | ||||
| 	font-size: 19px; | ||||
| } | ||||
|  | ||||
| #putil, #segutil, #segutil2 { | ||||
| @@ -359,7 +365,7 @@ button { | ||||
| 	padding-top: 12px; | ||||
| } | ||||
|  | ||||
| #fx, #pql, #segcont, #pcont, #sliders, #picker, #qcs-w, #hexw, #pall, #ledmap, | ||||
| #fx, #pql, #segcont, #pcont, #sliders, #qcs-w, #hexw, #pall, #ledmap, | ||||
| .slider, .filter, .option, .segname, .pname, .fnd { | ||||
| 	margin: 0 auto; | ||||
| } | ||||
| @@ -368,6 +374,11 @@ button { | ||||
| 	padding: 5px 0 0; | ||||
| } | ||||
|  | ||||
| /* Quick load magin for simplified UI */ | ||||
| .simplified #pql, .simplified #palw, .simplified #fx { | ||||
| 	margin-bottom: 8px; | ||||
| } | ||||
|  | ||||
| .smooth { transition: transform	calc(var(--f, 1)*.5s) ease-out } | ||||
|  | ||||
| .tab-label { | ||||
| @@ -388,8 +399,8 @@ button { | ||||
| 	align-items: center; | ||||
| 	justify-content: center; | ||||
| 	z-index: 11; | ||||
| 	opacity: 0.95; | ||||
| 	transition: 0.7s; | ||||
| 	opacity: .95; | ||||
| 	transition: .7s; | ||||
| 	pointer-events: none; | ||||
| } | ||||
|  | ||||
| @@ -399,7 +410,7 @@ button { | ||||
| 	position: sticky !important; | ||||
| 	top: 0; | ||||
| 	z-index: 2; | ||||
|     margin: 0 auto auto; | ||||
| 	margin: 0 auto auto; | ||||
| } | ||||
|  | ||||
| .staybot { | ||||
| @@ -411,6 +422,7 @@ button { | ||||
| 	position: sticky; | ||||
| 	bottom: 0; | ||||
| 	max-width: 300px; | ||||
| 	z-index: 2; | ||||
| } | ||||
|  | ||||
| #sliders .labels { | ||||
| @@ -456,65 +468,56 @@ button { | ||||
| 	padding: 4px 2px; | ||||
| 	position: relative; | ||||
| 	opacity: 1; | ||||
| 	transition: opacity 0.5s linear, height 0.5s, transform 0.5s; | ||||
| 	transition: opacity .5s linear, height .25s, transform .25s; | ||||
| } | ||||
|  | ||||
| .filter { | ||||
| 	z-index: 1; | ||||
| 	overflow: hidden; | ||||
| 	/*overflow: visible;*/ | ||||
| 	border-radius: 0 0 16px 16px; | ||||
| 	max-width: 220px; | ||||
| 	height: 54px; | ||||
| 	line-height: 1.5; | ||||
| 	padding-bottom: 8px; | ||||
| 	pointer-events: none; | ||||
| } | ||||
|  | ||||
| /* Tooltip text */ | ||||
| .slider .tooltiptext, .option .tooltiptext { | ||||
| /* New tooltip */ | ||||
| .tooltip { | ||||
| 	position: absolute; | ||||
| 	opacity: 0; | ||||
| 	visibility: hidden; | ||||
| 	transition: opacity .25s ease, visibility .25s ease; | ||||
| 	background-color: var(--c-5); | ||||
| 	/*border: 2px solid var(--c-2);*/ | ||||
| 	box-shadow: 4px 4px 10px 4px var(--c-1); | ||||
| 	color: var(--c-f); | ||||
| 	text-align: center; | ||||
| 	padding: 4px 8px; | ||||
| 	padding: 8px 16px; | ||||
| 	border-radius: 6px; | ||||
|  | ||||
| 	/* Position the tooltip text */ | ||||
| 	width: 160px; | ||||
| 	position: absolute; | ||||
| 	z-index: 1; | ||||
| 	bottom: 80%; | ||||
| 	left: 50%; | ||||
| 	margin-left: -92px; | ||||
|  | ||||
| 	/* Ensure tooltip goes away when mouse leaves control */ | ||||
| 	pointer-events: none; | ||||
| } | ||||
|  | ||||
| 	/* Fade in tooltip */ | ||||
| 	opacity: 0; | ||||
| 	transition: opacity 0.75s; | ||||
| } | ||||
| .option .tooltiptext { | ||||
| 	bottom: 120%; | ||||
| } | ||||
| /* Tooltip arrow */ | ||||
| .slider .tooltiptext::after, .option .tooltiptext::after { | ||||
|  .tooltip::after { | ||||
| 	content: ""; | ||||
| 	position: absolute; | ||||
| 	top: 100%; | ||||
| 	left: 50%; | ||||
| 	margin-left: -5px; | ||||
| 	border-width: 5px; | ||||
| 	border-style: solid; | ||||
| 	border: 8px solid; | ||||
| 	border-color: var(--c-5) transparent transparent transparent; | ||||
| } | ||||
| /* Show the tooltip text when you mouse over the tooltip container */ | ||||
| .slider:hover .tooltiptext, .option .check:hover .tooltiptext { | ||||
| 	visibility: visible; | ||||
| 	top: 100%; | ||||
| 	left: calc(50% - 8px); | ||||
| 	z-index: 0; | ||||
|  } | ||||
|  | ||||
| .tooltip.visible { | ||||
| 	opacity: 1; | ||||
| 	visibility: visible; | ||||
| } | ||||
|  | ||||
| .fade { | ||||
| 	visibility: hidden; /* hide it */ | ||||
| 	opacity: 0; /* make it transparent */ | ||||
| 	transform: scaleY(0); /* shrink content */ | ||||
| 	height: 0px; /* force other elements to move */ | ||||
| 	height: 0; /* force other elements to move */ | ||||
| 	padding: 0; /* remove empty space */ | ||||
| } | ||||
|  | ||||
| @@ -542,24 +545,24 @@ button { | ||||
|  | ||||
| #toast.show { | ||||
| 	opacity: 1; | ||||
| 	animation: fadein 0.5s, fadein 0.5s 2.5s reverse; | ||||
| 	animation: fadein .5s, fadein .5s 2.5s reverse; | ||||
| } | ||||
|  | ||||
| #toast.error { | ||||
| 	opacity: 1; | ||||
| 	background-color: #b21; | ||||
| 	animation: fadein 0.5s; | ||||
| 	animation: fadein .5s; | ||||
| } | ||||
|  | ||||
| .modal { | ||||
| 	position:fixed; | ||||
| 	left: 0px; | ||||
| 	bottom: 0px; | ||||
| 	right: 0px; | ||||
| 	position: fixed; | ||||
| 	left: 0; | ||||
| 	bottom: 0; | ||||
| 	right: 0; | ||||
| 	top: calc(var(--th) - 1px); | ||||
| 	background-color: var(--c-o); | ||||
| 	transform: translateY(100%); | ||||
| 	transition: transform 0.4s; | ||||
| 	transition: transform .4s; | ||||
| 	padding: 8px; | ||||
| 	font-size: 20px; | ||||
| 	overflow: auto; | ||||
| @@ -620,12 +623,10 @@ button { | ||||
|   padding-bottom: 8px; | ||||
| } | ||||
|  | ||||
| #info .btn { | ||||
| .infobtn { | ||||
| 	margin: 5px; | ||||
| } | ||||
| #info table .btn, #nodes table .btn { | ||||
| 	margin: 0; | ||||
| } | ||||
|  | ||||
| #info div, #nodes div { | ||||
| 	max-width: 490px; | ||||
| 	margin: 0 auto; | ||||
| @@ -641,7 +642,7 @@ button { | ||||
| } | ||||
|  | ||||
| #heart { | ||||
| 	transition: color 0.9s; | ||||
| 	transition: color .9s; | ||||
| 	font-size: 16px; | ||||
| 	color: #f00; | ||||
| } | ||||
| @@ -720,7 +721,7 @@ input[type=range] { | ||||
| } | ||||
|  | ||||
| input[type=range]:focus { | ||||
| 	outline: none; | ||||
| 	outline: 0; | ||||
| } | ||||
| input[type=range]::-webkit-slider-runnable-track { | ||||
| 	width: 100%; | ||||
| @@ -743,7 +744,7 @@ input[type=range]::-moz-range-track { | ||||
| 	background-color: rgba(0, 0, 0, 0); | ||||
| } | ||||
| input[type=range]::-moz-range-thumb { | ||||
| 	border: 0px solid rgba(0, 0, 0, 0); | ||||
| 	border: 0 solid rgba(0, 0, 0, 0); | ||||
| 	height: 16px; | ||||
| 	width: 16px; | ||||
| 	border-radius: 50%; | ||||
| @@ -761,39 +762,43 @@ input[type=range]::-moz-range-thumb { | ||||
| } | ||||
|  | ||||
| #Colors .sliderwrap { | ||||
| 	margin: 4px 0 0; | ||||
| 	margin: 2px 0 0; | ||||
| } | ||||
|  | ||||
| /* Dynamically hide brightness slider label */ | ||||
| /* Dynamically hide labels */ | ||||
| .hd { | ||||
| 	display: var(--bhd); | ||||
| } | ||||
| /* Do not hide quick load label in simplified mode on small screen widths */ | ||||
| .simplified #pql .hd { | ||||
| 	display: var(--bhd) !important; | ||||
| } | ||||
|  | ||||
| #briwrap { | ||||
| 	min-width: 267px; | ||||
| 	min-width: 300px; | ||||
| 	float: right; | ||||
| 	margin-top: var(--bmt); | ||||
| } | ||||
|  | ||||
| #picker { | ||||
| 	margin-top: 8px !important; | ||||
| 	max-width: 260px; | ||||
| 	margin: 4px auto 0 !important; | ||||
| 	max-width: max-content; | ||||
| } | ||||
|  | ||||
| /* buttons */ | ||||
| .btn { | ||||
| 	padding: 8px; | ||||
| 	margin: 10px 4px; | ||||
| 	/*margin: 10px 4px;*/ | ||||
| 	width: 230px; | ||||
| 	font-size: 19px; | ||||
| 	color: var(--c-d); | ||||
| 	cursor: pointer; | ||||
| 	border-radius: 25px; | ||||
| 	transition-duration: 0.3s; | ||||
| 	transition-duration: .3s; | ||||
| 	-webkit-backface-visibility: hidden; | ||||
| 	-webkit-transform:translate3d(0,0,0); | ||||
| 	-webkit-transform: translate3d(0,0,0); | ||||
| 	backface-visibility: hidden; | ||||
| 	transform:translate3d(0,0,0); | ||||
| 	transform: translate3d(0,0,0); | ||||
| 	overflow: hidden; | ||||
| 	text-overflow: ellipsis; | ||||
| 	border: 1px solid var(--c-3); | ||||
| @@ -829,14 +834,14 @@ input[type=range]::-moz-range-thumb { | ||||
| 	text-overflow: clip; | ||||
| } | ||||
| .btn-xs { | ||||
| 	margin: 2px 0 0 0; | ||||
| } | ||||
| #putil .btn-xs { | ||||
| 	margin: 0; | ||||
| } | ||||
| #info .btn-xs { | ||||
| 	border: 1px solid var(--c-4); | ||||
| } | ||||
| #btns .btn-xs { | ||||
| 	margin: 0 4px; | ||||
| } | ||||
|  | ||||
| #putil .btn-s { | ||||
| 	width: 135px; | ||||
| @@ -855,6 +860,15 @@ input[type=range]::-moz-range-thumb { | ||||
| 	margin: 0; | ||||
| 	white-space: nowrap; | ||||
| } | ||||
| a.btn { | ||||
| 	display: block; | ||||
| 	white-space: nowrap; | ||||
| 	text-align: center; | ||||
| 	padding: 9px 32px 7px 24px; | ||||
| 	position: relative; | ||||
| 	box-sizing: border-box; | ||||
| 	line-height: 24px; | ||||
| } | ||||
|  | ||||
| /* Quick color select wrapper div */ | ||||
| #qcs-w { | ||||
| @@ -893,21 +907,18 @@ select { | ||||
| 	cursor: pointer; | ||||
| 	border: 0 solid var(--c-2); | ||||
| 	border-radius: 20px; | ||||
| 	transition-duration: 0.5s; | ||||
| 	transition-duration: .5s; | ||||
| 	-webkit-backface-visibility: hidden; | ||||
| 	-webkit-transform:translate3d(0,0,0); | ||||
|     -webkit-appearance: none; | ||||
|     -moz-appearance: none; | ||||
| 	-webkit-transform: translate3d(0,0,0); | ||||
| 	-webkit-appearance: none; | ||||
| 	-moz-appearance: none; | ||||
| 	backface-visibility: hidden; | ||||
| 	transform:translate3d(0,0,0); | ||||
| 	transform: translate3d(0,0,0); | ||||
| 	text-overflow: ellipsis; | ||||
| } | ||||
| #tt { | ||||
| 	text-align: center; | ||||
| } | ||||
| .cl { | ||||
| 	background-color: #000; | ||||
| } | ||||
| select.sel-p, select.sel-pl, select.sel-ple { | ||||
| 	margin: 5px 0; | ||||
| 	width: 100%; | ||||
| @@ -917,15 +928,15 @@ div.sel-p { | ||||
| 	position: relative; | ||||
| } | ||||
| div.sel-p:after { | ||||
|     content: ""; | ||||
|     position: absolute; | ||||
|     right: 10px; | ||||
|     top: 22px; | ||||
|     width: 0; | ||||
|     height: 0; | ||||
|     border-left: 8px solid transparent; | ||||
|     border-right: 8px solid transparent; | ||||
|     border-top: 8px solid var(--c-f); | ||||
| 	content: ""; | ||||
| 	position: absolute; | ||||
| 	right: 10px; | ||||
| 	top: 22px; | ||||
| 	width: 0; | ||||
| 	height: 0; | ||||
| 	border-left: 8px solid transparent; | ||||
| 	border-right: 8px solid transparent; | ||||
| 	border-top: 8px solid var(--c-f); | ||||
| } | ||||
| select.sel-ple { | ||||
| 	text-align: center; | ||||
| @@ -942,13 +953,13 @@ input[type=number], | ||||
| input[type=text] { | ||||
| 	background: var(--c-3); | ||||
| 	color: var(--c-f); | ||||
| 	border: 0px solid var(--c-2); | ||||
| 	border: 0 solid var(--c-2); | ||||
| 	border-radius: 10px; | ||||
| 	padding: 8px; | ||||
| 	/*margin: 6px 6px 6px 0;*/ | ||||
| 	font-size: 19px; | ||||
| 	transition: background-color 0.2s; | ||||
| 	outline: none; | ||||
| 	transition: background-color .2s; | ||||
| 	outline: 0; | ||||
| 	-webkit-appearance: textfield; | ||||
| 	-moz-appearance: textfield; | ||||
| 	appearance: textfield; | ||||
| @@ -988,7 +999,7 @@ textarea { | ||||
| 	height: 90px; | ||||
| 	border-radius: 5px; | ||||
| 	border: 2px solid var(--c-5); | ||||
| 	outline: none; | ||||
| 	outline: 0; | ||||
| 	resize: none; | ||||
| 	font-size: 19px; | ||||
| 	padding: 5px; | ||||
| @@ -1010,7 +1021,7 @@ textarea { | ||||
| 	width: 50px !important; | ||||
| } | ||||
|  | ||||
| .segname, .pname, .bname { | ||||
| .segname, .pname { | ||||
| 	white-space: nowrap; | ||||
| 	text-align: center; | ||||
| 	overflow: hidden; | ||||
| @@ -1020,9 +1031,6 @@ textarea { | ||||
| 	max-width: 170px; | ||||
| 	position: relative; | ||||
| } | ||||
| .bname { | ||||
| 	padding: 0 24px; | ||||
| } | ||||
|  | ||||
| .segname .flr, .pname .flr { | ||||
| 	transform: rotate(0deg); | ||||
| @@ -1041,7 +1049,7 @@ textarea { | ||||
| 	top: 1px; | ||||
| } | ||||
| .plname { | ||||
| 	top:0; | ||||
| 	top: 0; | ||||
| } | ||||
|  | ||||
| /* preset id number */ | ||||
| @@ -1057,27 +1065,24 @@ textarea { | ||||
| .newseg { | ||||
| 	cursor: default; | ||||
| } | ||||
|  | ||||
| /* | ||||
| .ic { | ||||
| 	padding: 6px 0 0 0; | ||||
| } | ||||
|  | ||||
| .xxs { | ||||
| */ | ||||
| /* color selector */ | ||||
| #csl button { | ||||
| 	width: 44px; | ||||
| 	height: 44px; | ||||
| 	margin: 5px; | ||||
| 	border: 2px solid var(--c-d) !important; | ||||
| 	background-color: #000; | ||||
| } | ||||
|  | ||||
| .xxs-w { | ||||
| /* selected color selector */ | ||||
| #csl .sl { | ||||
| 	margin: 2px; | ||||
| 	width: 50px; | ||||
| 	height: 50px; | ||||
| } | ||||
|  | ||||
| #csl .xxs { | ||||
| 	border: 2px solid var(--c-d) !important; | ||||
| } | ||||
| #csl .xxs-w { | ||||
| 	border-width: 5px !important; | ||||
| } | ||||
|  | ||||
| @@ -1121,8 +1126,8 @@ textarea { | ||||
| } | ||||
|  | ||||
| .revchkl { | ||||
| 	padding: 4px 0px 0px 35px; | ||||
| 	margin-bottom: 0px; | ||||
| 	padding: 4px 0 0 35px; | ||||
| 	margin-bottom: 0; | ||||
| 	margin-top: 8px; | ||||
| } | ||||
|  | ||||
| @@ -1218,9 +1223,9 @@ TD .checkmark, TD .radiomark { | ||||
| .seg, .pres { | ||||
| 	background-color: var(--c-2); | ||||
| 	/*color: var(--c-f);*/ /* seems to affect only the Add segment button, which should be same color as reset segments */ | ||||
| 	border: 0px solid var(--c-f); | ||||
| 	border: 0 solid var(--c-f); | ||||
| 	text-align: left; | ||||
| 	transition: background-color 0.5s; | ||||
| 	transition: background-color .5s; | ||||
| 	border-radius: 21px; | ||||
| } | ||||
|  | ||||
| @@ -1237,14 +1242,19 @@ TD .checkmark, TD .radiomark { | ||||
| /* checkmark labels */ | ||||
| .filter .fchkl, .option .ochkl { | ||||
| 	display: inline-block; | ||||
| 	min-width: 0.7em; | ||||
| 	padding: 1px 4px 4px 32px; | ||||
| 	min-width: .7em; | ||||
| 	padding: 1px 4px 1px 32px; | ||||
| 	text-align: left; | ||||
| 	line-height: 24px; | ||||
| 	vertical-align: middle; | ||||
| 	-webkit-filter: grayscale(100%); /* Safari 6.0 - 9.0 */ | ||||
| 	filter: grayscale(100%); | ||||
| } | ||||
| .filter .fchkl { | ||||
| 	margin: 0 4px; | ||||
| 	min-width: 20px; | ||||
| 	pointer-events: auto; | ||||
| } | ||||
|  | ||||
| .lbl-l { | ||||
| 	font-size: 13px; | ||||
| @@ -1263,27 +1273,30 @@ TD .checkmark, TD .radiomark { | ||||
| /* list wrapper */ | ||||
| .list { | ||||
| 	position: relative; | ||||
| 	transition: background-color 0.5s; | ||||
|     margin: auto auto 10px; | ||||
| 	transition: background-color .5s; | ||||
| 	margin: auto auto 10px; | ||||
| 	line-height: 24px; | ||||
| } | ||||
|  | ||||
| /* list item */ | ||||
| .lstI { | ||||
| 	align-items: center; | ||||
|     cursor: pointer; | ||||
| 	cursor: pointer; | ||||
| 	background-color: var(--c-2); | ||||
| 	overflow: hidden; | ||||
| 	position: -webkit-sticky; | ||||
| 	position: sticky; | ||||
| 	border-radius: 21px; | ||||
| 	margin: 13px auto 0; | ||||
| 	margin: 0 auto 12px; | ||||
| 	min-height: 40px; | ||||
| 	border: 1px solid var(--c-2); | ||||
| 	width: 100%; | ||||
| } | ||||
|  | ||||
| #segutil .lstI { | ||||
| 	margin-top: 0; | ||||
| /* Simplify segments */ | ||||
| .simplified #segcont .lstI { | ||||
| 	margin-top: 4px; | ||||
| 	min-height: unset; | ||||
| } | ||||
|  | ||||
| /* selected item/element */ | ||||
| @@ -1293,7 +1306,7 @@ TD .checkmark, TD .radiomark { | ||||
|  | ||||
| #segcont .seg:hover:not([class*="expanded"]), | ||||
| .lstI:hover:not([class*="expanded"]) { | ||||
|     background: var(--c-5); | ||||
| 	background: var(--c-5); | ||||
| } | ||||
|  | ||||
| .selected .checkmark, | ||||
| @@ -1313,7 +1326,7 @@ TD .checkmark, TD .radiomark { | ||||
| .lstI.sticky, | ||||
| .lstI.selected { | ||||
| 	z-index: 1; | ||||
| 	box-shadow: 0px 0px 10px 4px var(--c-1); | ||||
| 	box-shadow: 0 0 10px 4px var(--c-1); | ||||
| } | ||||
|  | ||||
| #pcont .selected:not([class*="expanded"]) { | ||||
| @@ -1321,20 +1334,27 @@ TD .checkmark, TD .radiomark { | ||||
| 	top: 42px; | ||||
| } | ||||
|  | ||||
| #fxlist .lstI.selected { | ||||
| 	top: 84px; | ||||
| } | ||||
|  | ||||
| #fxlist .lstI.sticky { | ||||
| 	top: 42px; | ||||
| } | ||||
|  | ||||
| #fxlist .lstI.selected, | ||||
| #pallist .lstI.selected { | ||||
| 	top: 84px; | ||||
| 	top: calc(var(--sti) + 42px); | ||||
| } | ||||
|  | ||||
| dialog::backdrop { | ||||
| 	backdrop-filter: blur(10px); | ||||
| 	-webkit-backdrop-filter: blur(10px); | ||||
| } | ||||
| dialog { | ||||
| 	max-height: 70%; | ||||
| 	border: 0; | ||||
| 	border-radius: 10px; | ||||
| 	background: linear-gradient(rgba(0, 0, 0, 0.15), rgba(0, 0, 0, 0.1)), var(--c-3); | ||||
| 	box-shadow: 4px 4px 10px 4px var(--c-1); | ||||
| 	color: var(--c-f); | ||||
| } | ||||
|  | ||||
| #fxlist .lstI.sticky, | ||||
| #pallist .lstI.sticky { | ||||
| 	top: 42px; | ||||
| 	top: var(--sti); | ||||
| } | ||||
|  | ||||
| /* list item content */ | ||||
| @@ -1370,8 +1390,8 @@ TD .checkmark, TD .radiomark { | ||||
| 	display: block; | ||||
| 	width: 100%; | ||||
| 	box-sizing: border-box; | ||||
|     padding: 8px 40px 8px 44px; | ||||
|     margin: 5px auto 0; | ||||
| 	padding: 8px 40px 8px 44px; | ||||
| 	margin: 4px auto 12px; | ||||
| 	text-align: left; | ||||
| 	border-radius: 21px; | ||||
| 	background: var(--c-2); | ||||
| @@ -1389,6 +1409,13 @@ TD .checkmark, TD .radiomark { | ||||
| 	background-color: var(--c-3); | ||||
| } | ||||
|  | ||||
| #fxFind.fnd input[type="text"] { | ||||
| 	margin-bottom: 0; | ||||
| } | ||||
| #fxFind { | ||||
| 	margin-bottom: 12px; | ||||
| } | ||||
|  | ||||
| /* segment & preset inner/expanded content */ | ||||
| .segin, | ||||
| .presin { | ||||
| @@ -1457,7 +1484,7 @@ TD .checkmark, TD .radiomark { | ||||
| } | ||||
| ::-webkit-scrollbar-thumb { | ||||
| 	background: var(--c-sb); | ||||
| 	opacity: 0.2; | ||||
| 	opacity: .2; | ||||
| 	border-radius: 5px; | ||||
| } | ||||
| ::-webkit-scrollbar-thumb:hover { | ||||
| @@ -1483,7 +1510,7 @@ TD .checkmark, TD .radiomark { | ||||
|  | ||||
| @media all and (max-width: 335px) { | ||||
| 	.sliderbubble { | ||||
|     	display: none; | ||||
| 		display: none; | ||||
|   	} | ||||
| } | ||||
|  | ||||
| @@ -1494,7 +1521,7 @@ TD .checkmark, TD .radiomark { | ||||
| 	#info .infobtn, #nodes .infobtn { | ||||
| 		width: 145px; | ||||
| 	} | ||||
| 	#info div, #nodes div { | ||||
| 	#info div, #nodes div, #nodes a.btn { | ||||
| 		max-width: 320px; | ||||
| 	} | ||||
| } | ||||
| @@ -1540,9 +1567,6 @@ TD .checkmark, TD .radiomark { | ||||
| 		max-width: 280px; | ||||
| 		font-size: 18px; | ||||
| 	} | ||||
| 	#picker { | ||||
| 		width: 230px; | ||||
| 	} | ||||
| 	#putil .btn-s { | ||||
| 		width: 114px; | ||||
| 	} | ||||
|   | ||||
| @@ -7,52 +7,9 @@ | ||||
| 	<meta content="yes" name="apple-mobile-web-app-capable"> | ||||
| 	<link rel="shortcut icon" href="data:image/x-icon;base64,AAABAAEAEBAAAAEAGACGAAAAFgAAAIlQTkcNChoKAAAADUlIRFIAAAAQAAAAEAgGAAAAH/P/YQAAAE1JREFUOI1j/P//PwOxgNGeAUMxE9G6cQCKDWAhpADZ2f8PMjBS3QW08QK20KaZC2gfC9hCnqouoNgARgY7zMxAyNlUdQHlXiAlO2MDAD63EVqNHAe0AAAAAElFTkSuQmCC"/> | ||||
| 	<title>WLED</title> | ||||
| 	<script> | ||||
| 	function feedback(){} | ||||
| 	// instead of including [script src="iro.js"][/script] and [script src="rangetouch.js"][/script] | ||||
| 	// (which would be inlined by nodeJS inliner during minimization and compression) we need to load them dynamically | ||||
| 	// the following is needed to load iro.js and rangetouch.js as consecutive requests to allow ESP8266 | ||||
| 	// to keep up with requests (if requests happent too fast some may not get processed) | ||||
| 	// it will also call onLoad() after last is loaded (it was removed from [body onload="onLoad()"]). | ||||
| 	var h  = document.getElementsByTagName('head')[0]; | ||||
| 	var l  = document.createElement('script'); | ||||
| 	l.type = 'application/javascript'; | ||||
| 	l.src = 'iro.js'; | ||||
| 	l.addEventListener('load', (e) => { | ||||
| 		// after iro is loaded initialize global variable | ||||
| 		cpick = new iro.ColorPicker("#picker", { | ||||
| 			width: 260, | ||||
| 			wheelLightness: false, | ||||
| 			wheelAngle: 270, | ||||
| 			wheelDirection: "clockwise", | ||||
| 			layout: [{ | ||||
| 				component: iro.ui.Wheel, | ||||
| 				options: {} | ||||
| 			}] | ||||
| 		}); | ||||
| 		cpick.on("input:end", () => {setColor(1);}); | ||||
| 		cpick.on("color:change", () => {updatePSliders()}); | ||||
| 		var l  = document.createElement('script'); | ||||
| 		l.type = 'application/javascript'; | ||||
| 		l.src = 'rangetouch.js'; | ||||
| 		l.addEventListener('load', (e) => { | ||||
| 			// after rangetouch is loaded initialize global variable | ||||
| 			ranges = RangeTouch.setup('input[type="range"]', {}); | ||||
| 			let stateCheck = setInterval(() => { | ||||
| 				if (document.readyState === 'complete') { | ||||
| 					clearInterval(stateCheck); | ||||
| 					// document ready, start processing UI | ||||
| 					onLoad(); | ||||
| 				} | ||||
| 			}, 100); | ||||
| 		}); | ||||
| 		setTimeout(()=>{h.appendChild(l)},100); | ||||
| 	}); | ||||
| 	setTimeout(()=>{h.appendChild(l)},100); | ||||
| 	</script> | ||||
| 	<link rel="stylesheet" href="index.css"> | ||||
| </head> | ||||
| <body> | ||||
| <body onload="onLoad()"> | ||||
|  | ||||
| <div id="cv" class="overlay">Loading WLED UI...</div> | ||||
| <noscript><div class="overlay" style="opacity:1;">Sorry, WLED UI needs JavaScript!</div></noscript> | ||||
| @@ -72,8 +29,8 @@ | ||||
| 		</div> | ||||
| 		<div id="briwrap"> | ||||
| 			<p class="hd">Brightness</p> | ||||
| 			<div class="slider" style="padding-right:32px;"> | ||||
| 				<i class="icons slider-icon" onclick="tglTheme()" style="transform: translate(-32px,5px);"></i> | ||||
| 			<div class="slider"> | ||||
| 				<i class="icons slider-icon" onclick="tglTheme()" style="transform: translateY(2px);"></i> | ||||
| 				<div class="sliderwrap il"> | ||||
| 					<input id="sliderBri" onchange="setBri()" oninput="updateTrail(this)" max="255" min="1" type="range" value="128" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| @@ -88,99 +45,92 @@ | ||||
| <div class ="container"> | ||||
| 	<div id="Colors" class="tabcontent"> | ||||
| 		<div id="picker" class="noslide"></div> | ||||
| 		<div id="hwrap" class="slider" style="margin-top: 20px;"> | ||||
| 			<div class="sliderwrap il"> | ||||
| 		<div id="hwrap" class="slider"> | ||||
| 			<div title="Hue" class="sliderwrap il"> | ||||
| 				<input id="sliderH" class="noslide" oninput="fromH()" onchange="setColor(0)" max="359" min="0" type="range" value="0" step="any"> | ||||
| 				<div class="sliderdisplay" style="background: linear-gradient(90deg, #f00 2%, #ff0 19%, #0f0 35%, #0ff 52%, #00f 68%, #f0f 85%, #f00)"></div> | ||||
| 			</div> | ||||
| 			<span class="tooltiptext">Hue</span> | ||||
| 		</div> | ||||
| 		<div id="swrap" class="slider"> | ||||
| 			<div class="sliderwrap il"> | ||||
| 			<div title="Saturation" class="sliderwrap il"> | ||||
| 				<input id="sliderS" class="noslide" oninput="fromS()" onchange="setColor(0)" max="100" min="0" type="range" value="100" step="any"> | ||||
| 				<div class="sliderdisplay" style="background: linear-gradient(90deg, #aaa 0%, #f00)"></div> | ||||
| 			</div> | ||||
| 			<span class="tooltiptext">Saturation</span> | ||||
| 		</div> | ||||
| 		<div id="vwrap" class="slider"> | ||||
| 			<div class="sliderwrap il"> | ||||
| 			<div title="Value/Brightness" class="sliderwrap il"> | ||||
| 				<input id="sliderV" class="noslide" oninput="fromV()" onchange="setColor(0)" max="100" min="0" type="range" value="100" step="any" /> | ||||
| 				<div class="sliderdisplay"></div> | ||||
| 			</div> | ||||
| 			<span class="tooltiptext">Value/Brightness</span> | ||||
| 		</div> | ||||
| 		<div id="kwrap" class="slider"> | ||||
| 			<div class="sliderwrap il"> | ||||
| 			<div title="Kelvin/Temperature" class="sliderwrap il"> | ||||
| 				<input id="sliderK" class="noslide" oninput="fromK()" onchange="setColor(0)" max="10091" min="1900" type="range" value="6550" /> | ||||
| 				<div class="sliderdisplay"></div> | ||||
| 			</div> | ||||
| 			<span class="tooltiptext">Kelvin/Temperature</span> | ||||
| 		</div> | ||||
| 		<div id="rgbwrap"> | ||||
| 			<!--p class="labels hd">RGB color</p--> | ||||
| 			<div id="rwrap" class="slider"> | ||||
| 				<div class="sliderwrap il"> | ||||
| 				<div title="Red channel" class="sliderwrap il"> | ||||
| 					<input id="sliderR" class="noslide" oninput="fromRgb()" onchange="setColor(0)" max="255" min="0" type="range" value="128" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 				<span class="tooltiptext">Red channel</span> | ||||
| 			</div> | ||||
| 			<div id="gwrap" class="slider"> | ||||
| 				<div class="sliderwrap il"> | ||||
| 				<div title="Green channel" class="sliderwrap il"> | ||||
| 					<input id="sliderG" class="noslide" oninput="fromRgb()" onchange="setColor(0)" max="255" min="0" type="range" value="128" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 				<span class="tooltiptext">Green channel</span> | ||||
| 			</div> | ||||
| 			<div id="bwrap" class="slider"> | ||||
| 				<div class="sliderwrap il"> | ||||
| 				<div title="Blue channel" class="sliderwrap il"> | ||||
| 					<input id="sliderB" class="noslide" oninput="fromRgb()" onchange="setColor(0)" max="255" min="0" type="range" value="128" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 				<span class="tooltiptext">Blue channel</span> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div id="wwrap" class="slider"> | ||||
| 			<!--p class="labels hd">White channel</p--> | ||||
| 			<div id="whibri" class="sliderwrap il"> | ||||
| 			<div id="whibri" title="White channel" class="sliderwrap il"> | ||||
| 				<input id="sliderW" class="noslide" oninput="fromW()" onchange="setColor(0)" max="255" min="0" type="range" value="128" /> | ||||
| 				<div class="sliderdisplay"></div> | ||||
| 			</div> | ||||
| 			<span class="tooltiptext">White channel</span> | ||||
| 		</div> | ||||
| 		<div id="wbal" class="slider"> | ||||
| 			<!--p class="labels hd">White balance</p--> | ||||
| 			<div class="sliderwrap il"> | ||||
| 			<div title="White balance" class="sliderwrap il"> | ||||
| 				<input id="sliderA" class="noslide" onchange="setBalance(this.value)" max="255" min="0" type="range" value="128" /> | ||||
| 				<div class="sliderdisplay"></div> | ||||
| 			</div> | ||||
| 			<span class="tooltiptext">White balance</span> | ||||
| 		</div> | ||||
| 		<div id="qcs-w"> | ||||
| 			<div class="qcs" onclick="pC('#ff0000');" title="Red" style="background-color:#ff0000;"></div> | ||||
| 			<div class="qcs" onclick="pC('#ffa000');" title="Orange" style="background-color:#ffa000;"></div> | ||||
| 			<div class="qcs" onclick="pC('#ffc800');" title="Yellow" style="background-color:#ffc800;"></div> | ||||
| 			<div class="qcs" onclick="pC('#ffe0a0');" title="Warm White" style="background-color:#ffe0a0;"></div> | ||||
| 			<div class="qcs" onclick="pC('#ffffff');" title="White" style="background-color:#ffffff;"></div> | ||||
| 			<div class="qcs qcsb" onclick="pC('#000000');" title="Black" style="background-color:#000000;"></div><br> | ||||
| 			<div class="qcs" onclick="pC('#ff00ff');" title="Pink" style="background-color:#ff00ff;"></div> | ||||
| 			<div class="qcs" onclick="pC('#0000ff');" title="Blue" style="background-color:#0000ff;"></div> | ||||
| 			<div class="qcs" onclick="pC('#00ffc8');" title="Cyan" style="background-color:#00ffc8;"></div> | ||||
| 			<div class="qcs" onclick="pC('#08ff00');" title="Green" style="background-color:#08ff00;"></div> | ||||
| 			<div class="qcs" onclick="pC('#ff0000');" style="background-color:#ff0000;"></div> | ||||
| 			<div class="qcs" onclick="pC('#ffa000');" style="background-color:#ffa000;"></div> | ||||
| 			<div class="qcs" onclick="pC('#ffc800');" style="background-color:#ffc800;"></div> | ||||
| 			<div class="qcs" onclick="pC('#ffe0a0');" style="background-color:#ffe0a0;"></div> | ||||
| 			<div class="qcs" onclick="pC('#ffffff');" style="background-color:#ffffff;"></div> | ||||
| 			<div class="qcs qcsb" onclick="pC('#000000');" style="background-color:#000000;"></div><br> | ||||
| 			<div class="qcs" onclick="pC('#ff00ff');" style="background-color:#ff00ff;"></div> | ||||
| 			<div class="qcs" onclick="pC('#0000ff');" style="background-color:#0000ff;"></div> | ||||
| 			<div class="qcs" onclick="pC('#00ffc8');" style="background-color:#00ffc8;"></div> | ||||
| 			<div class="qcs" onclick="pC('#08ff00');" style="background-color:#08ff00;"></div> | ||||
| 			<div class="qcs" onclick="pC('rnd');" title="Random" style="background:linear-gradient(to right, red, orange, yellow, green, blue, purple);transform: translateY(-11px);">R</div> | ||||
| 		</div> | ||||
| 		<div id="csl"> | ||||
| 			<button id="csl0" class="btn xxs cl" onclick="selectSlot(0);" data-r="0" data-g="0" data-b="0" data-w="0">1</button> | ||||
| 			<button id="csl1" class="btn xxs cl" onclick="selectSlot(1);" data-r="0" data-g="0" data-b="0" data-w="0">2</button> | ||||
| 			<button id="csl2" class="btn xxs cl" onclick="selectSlot(2);" data-r="0" data-g="0" data-b="0" data-w="0">3</button> | ||||
| 			<button id="csl0" title="Select slot" class="btn" onclick="selectSlot(0);" data-r="0" data-g="0" data-b="0" data-w="0">1</button> | ||||
| 			<button id="csl1" title="Select slot" class="btn" onclick="selectSlot(1);" data-r="0" data-g="0" data-b="0" data-w="0">2</button> | ||||
| 			<button id="csl2" title="Select slot" class="btn" onclick="selectSlot(2);" data-r="0" data-g="0" data-b="0" data-w="0">3</button> | ||||
| 		</div> | ||||
| 		<p class="labels h" id="cslLabel"></p> | ||||
| 		<div id="hexw"> | ||||
| 			<i class="icons sel-icon" onclick="tglRgb()"></i> | ||||
| 			<input id="hexc" type="text" class="noslide" onkeydown="hexEnter()" autocomplete="off" maxlength="8" /> | ||||
| 			<input id="hexc" title="Hex RGB" type="text" class="noslide" onkeydown="hexEnter()" autocomplete="off" maxlength="8" /> | ||||
| 			<button id="hexcnf" class="btn btn-xs" onclick="fromHex();"><i class="icons btn-icon"></i></button> | ||||
| 		</div> | ||||
| 		<p class="labels" id="pall"><i class="icons sel-icon" onclick="tglHex()"></i> Color palette</p> | ||||
| 		<div style="padding: 8px 0;" id="btns"> | ||||
| 			<button class="btn btn-xs" title="Pixel Magic Tool" type="button" id="pxmb" onclick="window.location.href=getURL('/pxmagic.htm')"><i class="icons btn-icon"></i></button> | ||||
| 			<button class="btn btn-xs" title="Add custom palette" type="button" onclick="window.location.href=getURL('/cpal.htm')"><i class="icons btn-icon"></i></button> | ||||
| 			<button class="btn btn-xs" title="Remove custom palette" type="button" id="rmPal" onclick="palettesData=null;localStorage.removeItem('wledPalx');requestJson({rmcpal:true});setTimeout(loadPalettes,250,loadPalettesData);"><i class="icons btn-icon"></i></button> | ||||
| 		</div> | ||||
| 		<p class="labels hd" id="pall"><i class="icons sel-icon" onclick="tglHex()"></i> Color palette</p> | ||||
| 		<div id="palw" class="il"> | ||||
| 			<div class="staytop fnd"> | ||||
| 				<input type="text" placeholder="Search" oninput="search(this,'pallist')" onfocus="search(this,'pallist')" /> | ||||
| @@ -189,7 +139,7 @@ | ||||
| 			</div> | ||||
| 			<div id="pallist" class="list"> | ||||
| 				<div class="lstI"> | ||||
| 					<label class="radio schkl" onclick="loadPalettes()">  | ||||
| 					<label class="radio schkl" onclick="loadPalettes()"> | ||||
| 						<div class="lstIcontent"> | ||||
| 							<span class="lstIname"> | ||||
| 								Loading... | ||||
| @@ -199,24 +149,45 @@ | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div style="padding-block: 10px;"> | ||||
| 			<button class="btn btn-xs" type="button" onclick="window.location.href=getURL('/pxmagic.htm')"><i class="icons btn-icon"></i></button> | ||||
| 			<button class="btn btn-xs" type="button" onclick="window.location.href=getURL('/cpal.htm')"><i class="icons btn-icon"></i></button> | ||||
| 			<button class="btn btn-xs" type="button" onclick="palettesData=null;localStorage.removeItem('wledPalx');requestJson({rmcpal:true});setTimeout(loadPalettes,250,loadPalettesData);"><i class="icons btn-icon"></i></button> | ||||
| 		</div> | ||||
| 	</div> | ||||
|  | ||||
| 	<div id="Effects" class="tabcontent"> | ||||
| 		<div id="fx"> | ||||
| 			<p class="labels hd" id="modeLabel">Effect mode</p> | ||||
| 			<div class="staytop fnd" id="fxFind"> | ||||
| 				<input type="text" placeholder="Search" oninput="search(this,'fxlist')" onfocus="search(this,'fxlist');gId('filters').classList.add('fade');" onblur="gId('filters').classList.remove('fade')"/> | ||||
| 			<div class="staytop fnd" id="fxFind" onmousedown="preventBlur(event);"> | ||||
| 				<input type="text" placeholder="Search" oninput="search(this,'fxlist')" onfocus="filterFocus(event);search(this,'fxlist');" onblur="filterFocus(event);"> | ||||
| 				<i class="icons clear-icon" onclick="clean(this);"></i> | ||||
| 				<i class="icons search-icon" onclick="gId('filters').classList.toggle('hide');" style="cursor:pointer;"></i> | ||||
| 				<i class="icons search-icon" style="cursor:pointer;"></i> | ||||
| 				<div id="filters" class="filter fade"> | ||||
| 					<label id="filterPal" title="Uses palette" class="check fchkl">🎨 | ||||
| 						<input type="checkbox" data-flt="🎨" onchange="filterFx();"> | ||||
| 						<span class="checkmark"></span> | ||||
| 					</label> | ||||
| 					<label id="filter0D" title="Single pixel" class="check fchkl">• | ||||
| 						<input type="checkbox" data-flt="•" onchange="filterFx();"> | ||||
| 						<span class="checkmark"></span> | ||||
| 					</label> | ||||
| 					<label id="filter1D" title="1D" class="check fchkl">⋮ | ||||
| 						<input type="checkbox" data-flt="⋮" onchange="filterFx();"> | ||||
| 						<span class="checkmark"></span> | ||||
| 					</label> | ||||
| 					<label id="filter2D" title="2D" class="check fchkl">▦ | ||||
| 						<input type="checkbox" data-flt="▦" onchange="filterFx();"> | ||||
| 						<span class="checkmark"></span> | ||||
| 					</label> | ||||
| 					<label id="filterVol" title="Volume" class="check fchkl">♪ | ||||
| 						<input type="checkbox" data-flt="♪" onchange="filterFx();"> | ||||
| 						<span class="checkmark"></span> | ||||
| 					</label> | ||||
| 					<label id="filterFreq" title="Frequency" class="check fchkl">♫ | ||||
| 						<input type="checkbox" data-flt="♫" onchange="filterFx();"> | ||||
| 						<span class="checkmark"></span> | ||||
| 					</label> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 			<div id="fxlist" class="list"> | ||||
| 				<div class="lstI"> | ||||
| 					<label class="radio schkl" onclick="loadFX()">  | ||||
| 					<label class="radio schkl" onclick="loadFX()"> | ||||
| 						<div class="lstIcontent"> | ||||
| 							<span class="lstIname"> | ||||
| 								Loading... | ||||
| @@ -227,87 +198,56 @@ | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div id="sliders"> | ||||
| 			<div id="filters" class="filter"> | ||||
| 				<label id="filterPal" class="check fchkl">🎨 | ||||
| 					<input type="checkbox" data-flt="🎨" onchange="filterFx(this)"> | ||||
| 					<span class="checkmark"></span> | ||||
| 				</label> | ||||
| 				<label id="filter0D" class="check fchkl hide">• | ||||
| 					<input type="checkbox" data-flt="•" onchange="filterFx(this)"> | ||||
| 					<span class="checkmark"></span> | ||||
| 				</label> | ||||
| 				<label id="filter1D" class="check fchkl">⋮ | ||||
| 					<input type="checkbox" data-flt="⋮" onchange="filterFx(this)"> | ||||
| 					<span class="checkmark"></span> | ||||
| 				</label> | ||||
| 				<label id="filter2D" class="check fchkl">▦ | ||||
| 					<input type="checkbox" data-flt="▦" onchange="filterFx(this)"> | ||||
| 					<span class="checkmark"></span> | ||||
| 				</label> | ||||
| 				<label id="filterVol" class="check fchkl">♪ | ||||
| 					<input type="checkbox" data-flt="♪" onchange="filterFx(this)"> | ||||
| 					<span class="checkmark"></span> | ||||
| 				</label> | ||||
| 				<label id="filterFreq" class="check fchkl">♫ | ||||
| 					<input type="checkbox" data-flt="♫" onchange="filterFx(this)"> | ||||
| 					<span class="checkmark"></span> | ||||
| 				</label> | ||||
| 			</div> | ||||
| 			<div id="slider0" class="slider"> | ||||
| 				<i class="icons slider-icon" onclick="tglFreeze()"></i> | ||||
| 				<div class="sliderwrap il"> | ||||
| 				<i class="icons slider-icon" title="Freeze" onclick="tglFreeze()"></i> | ||||
| 				<div title="Effect speed" class="sliderwrap il"> | ||||
| 					<input id="sliderSpeed" class="noslide" onchange="setSpeed()" oninput="updateTrail(this)" max="255" min="0" type="range" value="128" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 				<output class="sliderbubble"></output> | ||||
| 				<span id="sliderLabel0" class="tooltiptext">Effect speed</span> | ||||
| 			</div> | ||||
| 			<div id="slider1" class="slider"> | ||||
| 				<i class="icons slider-icon" onclick="tglLabels()"></i> | ||||
| 				<div class="sliderwrap il"> | ||||
| 				<i class="icons slider-icon" title="Toggle labels" onclick="tglLabels()"></i> | ||||
| 				<div title="Effect intensity" class="sliderwrap il"> | ||||
| 					<input id="sliderIntensity" class="noslide" onchange="setIntensity()" oninput="updateTrail(this)" max="255" min="0" type="range" value="128" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 				<output class="sliderbubble"></output> | ||||
| 				<span class="tooltiptext" id="sliderLabel1">Effect intensity</span> | ||||
| 			</div> | ||||
| 			<div id="slider2" class="slider hide"> | ||||
| 				<i class="icons slider-icon"></i> | ||||
| 				<div class="sliderwrap il"> | ||||
| 				<div title="Custom 1" class="sliderwrap il"> | ||||
| 					<input id="sliderC1" class="noslide" onchange="setCustom(1)" oninput="updateTrail(this)" max="255" min="0" type="range" value="0" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 				<output class="sliderbubble"></output> | ||||
| 				<span class="tooltiptext" id="sliderLabel2">Custom 1</span> | ||||
| 			</div> | ||||
| 			<div id="slider3" class="slider hide"> | ||||
| 				<i class="icons slider-icon"></i> | ||||
| 				<div class="sliderwrap il"> | ||||
| 				<div title="Custom 2" class="sliderwrap il"> | ||||
| 					<input id="sliderC2" class="noslide" onchange="setCustom(2)" oninput="updateTrail(this)" max="255" min="0" type="range" value="0" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 				<output class="sliderbubble"></output> | ||||
| 				<span class="tooltiptext" id="sliderLabel3">Custom 2</span> | ||||
| 			</div> | ||||
| 			<div id="slider4" class="slider hide"> | ||||
| 				<i class="icons slider-icon"></i> | ||||
| 				<div class="sliderwrap il"> | ||||
| 				<div title="Custom 3" class="sliderwrap il"> | ||||
| 					<input id="sliderC3" class="noslide" onchange="setCustom(3)" oninput="updateTrail(this)" max="31" min="0" type="range" value="0" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 				<output class="sliderbubble"></output> | ||||
| 				<span class="tooltiptext" id="sliderLabel4">Custom 3</span> | ||||
| 			</div> | ||||
| 			<div id="fxopt" class="option fade"> | ||||
| 				<label id="opt0" class="check ochkl hide"><i class="icons"></i><span class="tooltiptext" id="optLabel0">Check 1</span> | ||||
| 				<label id="opt0" title="Check 1" class="check ochkl hide"><i class="icons"></i> | ||||
| 					<input id="checkO1" type="checkbox" onchange="setOption(1, this.checked)"> | ||||
| 					<span class="checkmark"></span> | ||||
| 				</label> | ||||
| 				<label id="opt1" class="check ochkl hide"><i class="icons"></i><span class="tooltiptext" id="optLabel1">Check 2</span> | ||||
| 				<label id="opt1" title="Check 2" class="check ochkl hide"><i class="icons"></i> | ||||
| 					<input id="checkO2" type="checkbox" onchange="setOption(2, this.checked)"> | ||||
| 					<span class="checkmark"></span> | ||||
| 				</label> | ||||
| 				<label id="opt2" class="check ochkl hide"><i class="icons"></i><span class="tooltiptext" id="optLabel2">Check 3</span> | ||||
| 				<label id="opt2" title="Check 3" class="check ochkl hide"><i class="icons"></i> | ||||
| 					<input id="checkO3" type="checkbox" onchange="setOption(3, this.checked)"> | ||||
| 					<span class="checkmark"></span> | ||||
| 				</label> | ||||
| @@ -316,6 +256,7 @@ | ||||
| 	</div> | ||||
|  | ||||
| 	<div id="Segments" class="tabcontent"> | ||||
| 		<p class="labels hd" id="segLabel">Segments</p> | ||||
| 		<div id="segcont"> | ||||
| 			Loading... | ||||
| 		</div> | ||||
| @@ -368,7 +309,7 @@ | ||||
| 		<button class="btn infobtn" id="resetbtn" onclick="cnfReset()">Reboot WLED</button> | ||||
| 	</div> | ||||
| 	<br> | ||||
| 	<span class="h">Made with <span id="heart">❤︎</span> by Aircoookie and the <a href="https://wled.discourse.group/" target="_blank">WLED community</a></span> | ||||
| 	<span class="h">Made with <span id="heart">❤︎</span> by <a href="https://github.com/Aircoookie/" target="_blank">Aircoookie</a> and the <a href="https://wled.discourse.group/" target="_blank">WLED community</a></span> | ||||
| </div> | ||||
|  | ||||
| <div id="nodes" class="modal"> | ||||
| @@ -377,7 +318,7 @@ | ||||
| 	<div id="kn">Loading...</div> | ||||
| 	<div style="position:sticky;bottom:0;"> | ||||
| 		<button class="btn infobtn" onclick="loadNodes()">Refresh</button> | ||||
| 	</div>	 | ||||
| 	</div> | ||||
| </div> | ||||
|  | ||||
| <div id="mlv2D" class="modal"> | ||||
| @@ -395,6 +336,13 @@ | ||||
| </div> | ||||
|  | ||||
| <i id="roverstar" class="icons huge" onclick="setLor(0)"></i><br> | ||||
|  | ||||
| <!--  | ||||
| 	If you want to load iro.js and rangetouch.js as consecutive requests, you can do it like it was done in 0.14.0: | ||||
| 	https://github.com/Aircoookie/WLED/blob/v0.14.0/wled00/data/index.htm | ||||
| --> | ||||
| <script src="iro.js"></script> | ||||
| <script src="rangetouch.js"></script> | ||||
| <script src="index.js"></script> | ||||
| </body> | ||||
| </html> | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -6,23 +6,33 @@ | ||||
|   <meta name="theme-color" content="#222222"> | ||||
|   <title>WLED Live Preview</title> | ||||
|   <style> | ||||
|   body { | ||||
|     margin: 0; | ||||
|   } | ||||
|   #canv { | ||||
|     background: black; | ||||
|     filter: brightness(175%); | ||||
|     width: 100%; | ||||
|     height: 100%; | ||||
|     position: absolute; | ||||
|   } | ||||
|     body { | ||||
|       margin: 0; | ||||
|     } | ||||
|     #canv { | ||||
|       background: black; | ||||
|       filter: brightness(175%); | ||||
|       width: 100%; | ||||
|       height: 100%; | ||||
|       position: absolute; | ||||
|     } | ||||
|   </style> | ||||
|   <script> | ||||
|     var d = document; | ||||
|     var ws; | ||||
|     var tmout = null; | ||||
|     function update() // via HTTP (/json/live) | ||||
|     { | ||||
|       if (document.hidden) { | ||||
|     var c; | ||||
|     var ctx; | ||||
|     function draw(start, skip, leds, fill) { | ||||
|       c.width = d.documentElement.clientWidth; | ||||
|       let w = (c.width * skip) / (leds.length - start); | ||||
|       for (let i = start; i < leds.length; i += skip) { | ||||
|         ctx.fillStyle = fill(leds,i); | ||||
|         ctx.fillRect(Math.round((i - start) * w / skip), 0, Math.ceil(w), c.height); | ||||
|       } | ||||
|     } | ||||
|     function update() { // via HTTP (/json/live) | ||||
|       if (d.hidden) { | ||||
|         clearTimeout(tmout); | ||||
|         tmout = setTimeout(update, 250); | ||||
|         return; | ||||
| @@ -36,27 +46,20 @@ | ||||
|         return res.json(); | ||||
|       }) | ||||
|       .then(json => { | ||||
|         var str = "linear-gradient(90deg,"; | ||||
|         var len = json.leds.length; | ||||
|         for (i = 0; i < len; i++) { | ||||
|           var leddata = json.leds[i]; | ||||
|           if (leddata.length > 6) leddata = leddata.substring(2); | ||||
|           str += "#" + leddata; | ||||
|           if (i < len -1) str += "," | ||||
|         } | ||||
|         str += ")"; | ||||
|         document.getElementById("canv").style.background = str; | ||||
|         draw(0, 1, json.leds, (a,i) => "#" + ((a[i].length > 6) ? a[i].substring(2) : a[i])); | ||||
|         clearTimeout(tmout); | ||||
|         tmout = setTimeout(update, 40); | ||||
|       }) | ||||
|       .catch(function (error) { | ||||
|       .catch((error)=>{ | ||||
|         //console.error("Peek HTTP error:",error); | ||||
|         clearTimeout(tmout); | ||||
|         tmout = setTimeout(update, 2500); | ||||
|       }) | ||||
|     } | ||||
|     function S() { // Startup function (onload) | ||||
|       let wsOn = (window.location.href.indexOf("?ws") > 0); | ||||
|       if (!wsOn) {update(); return;} | ||||
|       c = d.getElementById('canv'); | ||||
|       ctx = c.getContext('2d'); | ||||
|       if (window.location.href.indexOf("?ws") == -1) {update(); return;} | ||||
|  | ||||
|       // Initialize WebSocket connection | ||||
|       try { | ||||
| @@ -86,15 +89,8 @@ | ||||
|           if (toString.call(e.data) === '[object ArrayBuffer]') { | ||||
|             let leds = new Uint8Array(event.data); | ||||
|             if (leds[0] != 76) return; //'L' | ||||
|             let str = "linear-gradient(90deg,"; | ||||
|             let len = leds.length; | ||||
|             let start = leds[1]==2 ? 4 : 2; // 1 = 1D, 2 = 1D/2D (leds[2]=w, leds[3]=h) | ||||
|             for (i = start; i < len; i+=3) { | ||||
|               str += `rgb(${leds[i]},${leds[i+1]},${leds[i+2]})`; | ||||
|               if (i < len -3) str += "," | ||||
|             } | ||||
|             str += ")"; | ||||
|             document.getElementById("canv").style.background = str; | ||||
|             // leds[1] = 1: 1D; leds[1] = 2: 1D/2D (leds[2]=w, leds[3]=h) | ||||
|             draw(leds[1]==2 ? 4 : 2, 3, leds, (a,i) => `rgb(${a[i]},${a[i+1]},${a[i+2]})`); | ||||
|           } | ||||
|         } catch (err) { | ||||
|           console.error("Peek WS error:",err); | ||||
| @@ -104,6 +100,6 @@ | ||||
|   </script> | ||||
| </head> | ||||
| <body onload="S()"> | ||||
|   <div id="canv"></div> | ||||
|   <canvas id="canv"></canvas> | ||||
| </body> | ||||
| </html> | ||||
| </html> | ||||
|   | ||||
| @@ -61,6 +61,7 @@ | ||||
|       } | ||||
|  | ||||
|       body { | ||||
|         margin: 0; | ||||
|         display: flex; | ||||
|         justify-content: center; | ||||
|         align-items: center; | ||||
| @@ -72,29 +73,24 @@ | ||||
|         width: 100%; | ||||
|       } | ||||
|  | ||||
|       form { | ||||
|         margin-bottom: 20px; | ||||
|       } | ||||
|  | ||||
|       small { | ||||
|         display: block; | ||||
|         font-weight: 400; | ||||
|         margin: 2px 0 5px; | ||||
|         color: var(--gray-light); | ||||
|         font-size: 12px; | ||||
|         font-size: 0.75rem; | ||||
|       } | ||||
|  | ||||
|       footer { | ||||
|         width: 100%; | ||||
|         margin: 0 auto 20px; | ||||
|         display: block; | ||||
|         text-align: center; | ||||
|         display: flex; | ||||
|         justify-content: center; | ||||
|         margin-block: 20px; | ||||
|       } | ||||
|  | ||||
|       a { | ||||
|         text-decoration: none; | ||||
|         color: var(--blue-light); | ||||
|         font-size: 12px; | ||||
|         font-size: 0.75rem; | ||||
|         font-weight: 600; | ||||
|       } | ||||
|  | ||||
| @@ -103,26 +99,13 @@ | ||||
|       } | ||||
|  | ||||
|       #wledEdit { | ||||
|         padding: 4px 8px; | ||||
|         padding: 1.5px 8px; | ||||
|         background: var(--blue-light); | ||||
|         margin-left: 6px; | ||||
|         display: inline-block; | ||||
|         border-radius: 4px; | ||||
|         color: var(--gray-light); | ||||
|       } | ||||
|  | ||||
|       .m-zero { | ||||
|         margin: 0 !important; | ||||
|       } | ||||
|  | ||||
|       .m-bottom { | ||||
|         margin-bottom: 10px !important; | ||||
|       } | ||||
|  | ||||
|       .m-top { | ||||
|         margin-top: 10px !important; | ||||
|       } | ||||
|  | ||||
|       .container { | ||||
|         width: 100%; | ||||
|         display: flex; | ||||
| @@ -133,7 +116,7 @@ | ||||
|  | ||||
|       .content { | ||||
|         width: min(768px, calc(100% - 40px)); | ||||
|         margin: 20px; | ||||
|         margin-inline: 20px; | ||||
|       } | ||||
|  | ||||
|       .row { | ||||
| @@ -143,32 +126,61 @@ | ||||
|         margin-top: 20px; | ||||
|       } | ||||
|  | ||||
|       .column { | ||||
|       .col { | ||||
|         flex-basis: calc(50% - 10px); | ||||
|         position: relative; | ||||
|         padding-inline: 5px; | ||||
|       } | ||||
|  | ||||
|       .column-full { | ||||
|       .col-full { | ||||
|         flex-basis: 100%; | ||||
|         position: relative; | ||||
|         padding-inline: 5px; | ||||
|       } | ||||
|  | ||||
|       .col-small { | ||||
|         flex-basis: 33.3333%; | ||||
|         position: relative; | ||||
|         padding-inline: 5px; | ||||
|       } | ||||
|  | ||||
|       .header { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         align-items: center; | ||||
|         padding-bottom: 20px; | ||||
|         padding-block: 20px; | ||||
|       } | ||||
|  | ||||
|       .header .brand { | ||||
|         width: 100%; | ||||
|         max-width: 200px; | ||||
|         height: 100%; | ||||
|         display: block; | ||||
|         outline: none; | ||||
|         border: 0; | ||||
|       .header .title { | ||||
|         font-size: 2.5rem; | ||||
|         line-height: 2.5rem; | ||||
|         font-weight: 800; | ||||
|         color: var(--gray-light); | ||||
|         padding-bottom: 5px; | ||||
|       } | ||||
|  | ||||
|       .header .subtitle { | ||||
|         font-size: 0.75rem; | ||||
|         color: var(--text); | ||||
|       } | ||||
|  | ||||
|       .header .rainbow { | ||||
|         background: linear-gradient( | ||||
|           to right, | ||||
|           #ef5350, | ||||
|           #f48fb1, | ||||
|           #7e57c2, | ||||
|           #2196f3, | ||||
|           #26c6da, | ||||
|           #43a047, | ||||
|           #eeff41, | ||||
|           #f9a825, | ||||
|           #ff5722 | ||||
|         ); | ||||
|         -webkit-background-clip: text; | ||||
|         -webkit-text-fill-color: transparent; | ||||
|         background-size: 200% 200%; | ||||
|         animation: rainbow 5s linear infinite; | ||||
|         transition: background-position 0.5s ease; | ||||
|       } | ||||
|  | ||||
|       label { | ||||
| @@ -190,15 +202,7 @@ | ||||
|         border: 1px solid var(--gray-medium); | ||||
|         outline: none; | ||||
|         color: var(--gray-light); | ||||
|         font-size: 14px; | ||||
|       } | ||||
|  | ||||
|       input[type="color"] { | ||||
|         width: 32px; | ||||
|         height: 32px; | ||||
|         cursor: pointer; | ||||
|         padding-inline: 1px; | ||||
|         outline: none; | ||||
|         font-size: 0.875rem; | ||||
|       } | ||||
|  | ||||
|       .input-group { | ||||
| @@ -208,7 +212,7 @@ | ||||
|       } | ||||
|  | ||||
|       .input-group input:not([type="range"]) { | ||||
|         border-radius: 8px 0 0 8px; | ||||
|         border-radius: 50px 0 0 50px; | ||||
|       } | ||||
|  | ||||
|       .input-group .input-description { | ||||
| @@ -220,15 +224,15 @@ | ||||
|         align-items: center; | ||||
|         color: var(--gray-dark); | ||||
|         background: var(--gray-light); | ||||
|         border-radius: 0px 8px 8px 0; | ||||
|         border-radius: 0px 50px 50px 0; | ||||
|         border: 1px solid var(--gray-light); | ||||
|         border-left: 0; | ||||
|         font-size: 14px; | ||||
|         line-height: 16px; | ||||
|         font-size: 0.875rem; | ||||
|         line-height: 1rem; | ||||
|       } | ||||
|  | ||||
|       .input-group .square { | ||||
|         border-radius: 8px !important; | ||||
|         border-radius: 50px !important; | ||||
|         margin-left: 10px; | ||||
|       } | ||||
|  | ||||
| @@ -242,13 +246,7 @@ | ||||
|  | ||||
|       textarea { | ||||
|         resize: none; | ||||
|         min-height: 200px; | ||||
|         border-radius: 8px; | ||||
|         overflow-x: hidden; | ||||
|       } | ||||
|  | ||||
|       .custom-select { | ||||
|         position: relative; | ||||
|       } | ||||
|  | ||||
|       .custom-select select { | ||||
| @@ -256,7 +254,7 @@ | ||||
|         -webkit-appearance: none; | ||||
|         -moz-appearance: none; | ||||
|         background-image: none; | ||||
|         padding-right: 20px; | ||||
|         padding-right: 39px; | ||||
|         cursor: pointer; | ||||
|       } | ||||
|  | ||||
| @@ -264,7 +262,7 @@ | ||||
|         content: ""; | ||||
|         position: absolute; | ||||
|         top: calc(50% + 6px); | ||||
|         right: 16px; | ||||
|         right: 21px; | ||||
|         transform: rotate(135deg); | ||||
|         width: 6px; | ||||
|         height: 6px; | ||||
| @@ -456,7 +454,7 @@ | ||||
|         background: var(--gray-medium); | ||||
|         border: 1px solid var(--gray-dark); | ||||
|         transition: all 0.5s ease-in-out; | ||||
|         font-size: 14px; | ||||
|         font-size: 0.875rem; | ||||
|         font-weight: 600; | ||||
|       } | ||||
|  | ||||
| @@ -512,7 +510,7 @@ | ||||
|         color: var(--error-dark); | ||||
|         padding-block: 4px; | ||||
|         font-weight: 600; | ||||
|         font-size: 12px; | ||||
|         font-size: 0.75rem; | ||||
|       } | ||||
|  | ||||
|       @media (max-width: 767px) { | ||||
| @@ -522,8 +520,9 @@ | ||||
|           margin: 0; | ||||
|         } | ||||
|  | ||||
|         .column, | ||||
|         .column-full { | ||||
|         .col, | ||||
|         .col-full, | ||||
|         .col-small { | ||||
|           flex-basis: 100%; | ||||
|           margin-top: 20px; | ||||
|           padding: 0; | ||||
| @@ -554,6 +553,15 @@ | ||||
|           transform: translateY(0); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       @keyframes rainbow { | ||||
|         0% { | ||||
|           background-position: 0% 0; | ||||
|         } | ||||
|         100% { | ||||
|           background-position: 200% 0; | ||||
|         } | ||||
|       } | ||||
|     </style> | ||||
|   </head> | ||||
|   <body> | ||||
| @@ -561,14 +569,21 @@ | ||||
|       <div class="content"> | ||||
|         <form id="formGenerate" novalidate> | ||||
|           <div class="header"> | ||||
|             <img src=" data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAMgAAACCCAYAAAADm4eUAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAACRBJREFUeNrsnYtx6joQhsWdWwBzGsAl0MGBSkILVABUQAtQCU4HLsG3gTN0kGvnyJmNIhnJT9n+vplMEr8f+q1dabVSCgCcrMa+gD9//uyKXw+xaP/r16+UVwMx8A+PAACBAEzLxCpMq49X2xSm1opXBNQgAJHy71xuxKdGMqAx4PvzOxe/TtTe8Qhkr39vi5+rWH4sfjJeDSxaINXXu/hymasyvuywGIEUAihriLX+91kU/qzDY3+02P1RibPOnLD001yK7c8TNqXM+4GRa5DSfNrpv1NhWn3VGMYyzCvAxBKm1lMLJ8YvbeUjSTZiXaavHxBIcOGqClEiViW6teSzNqnzNYrtDmLfvNj2ZtnsIv4+OQ5VnuO9+PktrsnLBHMsP+ifqqFhav5SLp7bRtwLDFiD7BwFNjGW1xWuN8M0u1lqn7MQlEsg7+V2Wpi7pb/w4lmUAjmLDxkCidnEisycUrSgQbQC0WbVm8X23xbrKpPn7jC3XnETNdab8eWU5tRKmE/VdVyN49xpUEAgbarxs6jGH9JUKtbta3ZNHGbQWix/b2Fa5GaNUbN9qrc1V/1HLYNAoJsa8Syd46rmM5Z/q+W0kNucp/XxYFiBZI6/beTCDPrWwSj2HfqFm/00Iec3GyRuluXK2KbJ/XV9PBhKIMUX7Biw7a0qRIZplr0wzfq8/ij7aQATK0aTKTGc+08Txuj3acOb8I9+mEc1ppTP8UwfESYukLtwyGMxExKL2ZQrd7+Pq1ZaicYC2YJ2eGEenQKv9+BYjkCmLpCGTbkAmFgDmVO2UJN1i0M26ssp9rmqn/FgkzE/Ech88fYpavp9THFVy119ObZRjlvH8Wr7lMwRgiOan5OGMekAoTWIT0+zQeOw7+Jc61AToklPthHGkphf4w6faeZRC51qgiv7oryuzkPzhdlomp/XYt1Tv6/9rASiwkectQn73jY4X5OEAomrwNpeYE1L0yvxHkXh+YjoXR97CpHZ1bzXyYOJBYCT/kmuhu8RTz2+tLlwZpuaqR8eZtWz53s0TeWsx3MikK6RYSwDnnPvUYjvPfZsH/uOPK7u0WKKHucQ9YyJBUANMg6eTrps0Wqa7fFSY74NacJe6s4dODYIgUBnZs45gmsoBXGe27NFIN3XGnIcS5+sRX9VHmtYhw5BSYQjP6nrRyDdIxPllS/9bqzvKtWO7D+6DPn1DjSVyns9xXT9CCQectP8IdUOJtYSzSqXGRFcK4gkESHhOxuP8KBWWSCnbiohkHFxmRHWWkW5MxvK9EJV+M5RFEifbI8u2maBbGoq3cR5Zeh+pu+teiYIBMIzG8ps+JYURFO531xfv6zBnlPoSEQg/SJzEeeMksTEioWyIN57qr7lsbMXZkQizJJUvQ51kbNrZQ4zSb0wt+T1ffNBPO6tbnavLkwlaS4+Ech49Jn98Mexa8yIUGpn15LrasytNvfuPH8XplKXkydNTSBXS8HIzJxYjvHVawUQKdaBR478S6Hza/zoQKoZr+08hrKM3yavE4xag9gK4Ejza7wjBhgTwt0BEAgAAgEYxgdZIjWDmwYb2DNCsjcn5tzxDeZX33fd1N7hHO/ec91TgwAgEAAEAoBAAHDSu3Xq2rJ2HKfV4KCaaIVFYQzEekVX6Uw3jpm30kUJRHXT4uHKHdx2HHVoa5VvpsLQpBFloXgf8R0d1PAtdwdlH4ezWppA5oRXpsIG8W6E8+CDACAQAAQCMCR9+iC7yCaQgZFxTIyaLFUgMRASQ/WgCPfOLnD7ujHyJXUtdp1ksJy1QEKC5aaYUmeEZ7ka+LTZi3eY1rzPXRcCwQcBQCAAHZpYjtCKzQjX5x0SwLXBkE56LA7rQXmGBHBtXtjEm3v6aKHz2bdKlh27QELZG0nNzsodX2Nuu4tBkOYIOo/7mKKj3WZkZOh89m2TZfeJ94hCYrGmw9azpY1EfBHWINA/Vx7B8NCKBbDUGsQxKCmE3xQRBDJnYnawbQ7zmwrr/ZVTD8B3cuWePz5FIJFj6y9pMET4Sb+L8/mWAmlrQeCDACAQAHyQRbIOMcswxxYqkJh6vB2DxC49JUYI7d1eUeQjEIguDOeI79u39acsgHTQIZDF8fRMoUMJwUkHAAQCgIkFL0xGm1+YRty69WZppStzIt8QSCS+irKHJYTmv/UlpKCGTstdcurgvF3fX6LcqX8OjmMgkBgovlRla9fe8iV+qB6mxNZfcq/COtK03G3vb++4j2hj5vBBABAIAAIBwAfpiK0jT6yNMjvHIGMuQpJddMipOO/J4i+sZvDuWvtISxXIemoOroWb8m99mlPIzKDvjlasiaIHBOWeNRMPDB8EAIEA4KQvAaaBRiBQz4lHMD+BXAaeMtjrfK4BWg2mgGvdlBpx82iUdPjuenkf+CAACAQAgQAgEIAYnPTWOU0DyHs836XBtfhwL37eKT6fuDLEZAO/u16gxWUkhmxhazKLl9n6U3OMtOXMVZhYAAgEYEE+CMTHtkVU7pbHh0DmzuTGcxSCvgaI8z5kOh9xjV9jZRgwFRERTTvtGz6UK3vLUv6i5tp5XsdYrYK1A7AQCPgKuhSCj5BChZ5qcaQD1hreaZsQCIzN+8ABsPggMDo3S40gx8XvPU20vjhq0+qBQGAscyw3TK6NWJ+OfH2Zvi5qkImRzvSckx0chkDi+vKmI4kkOlxNxHVhLcU+B/V3rnmTxk3ICARiJaSJuCJx7NO4CZlgRRiiNvgwaoFVzbZnH5NMHiMgGPNbwKfPdRGLBYCJBRPC5oOVLWCHmn1yZe/lPyEQmBu5FslTNMfuTIHoZdU2XzOBGSaUKZDPgM+QZmZMLIiNg/YnZHBmNdPXTSyT22z1/6YfUu4jRzteVeDAMWoQiMWRT7Q4flsc8s8awjKh549OSGO/tG3ibgQCsZA09Bl67YTExAJAIAAIBKZNrv421ab6/13ZkefIgi+5SOdd7/NAIDAryghgPS7EDAvZlM65dtA3lv3Kfe7G4rXYp9V4fJx0iJ2DsncSlk24T8c+VbOvSdnkmyEQWEKNkzbYLRt7LArAD7Rf8GHrx6jZ5yH289q39FeMfR5trwsfBAAAmvG/AAMAhiHUfzRdo4IAAAAASUVORK5CYII=" alt="Pixel Magic Tool" class="brand"> | ||||
|             <span class="title" | ||||
|               >PIXEL <span class="rainbow">MAGIC</span> TOOL</span | ||||
|             > | ||||
|             <span class="subtitle" | ||||
|               >It is a tool that converts any image into code in | ||||
|               <strong>JSON WLED</strong> format for | ||||
|               <strong>2D Matrix</strong> panels</span | ||||
|             > | ||||
|           </div> | ||||
|           <div class="row"> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="hostname">Hostname</label> | ||||
|               <input type="text" name="hostname" id="hostname" required /> | ||||
|             </div> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="name">Preset Name</label> | ||||
|               <input | ||||
|                 type="text" | ||||
| @@ -579,11 +594,10 @@ | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="row"> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <div class="custom-select"> | ||||
|                 <label for="pattern">Pattern</label> | ||||
|                 <select name="pattern" id="pattern" required> | ||||
|                   <option value="">Select a choice</option> | ||||
|                   <option value="1" title="['ffffff']">Individual</option> | ||||
|                   <option value="2" title="[0, 'ffffff']">Index</option> | ||||
|                   <option value="3" title="[0, 5, 'ffffff']" selected> | ||||
| @@ -592,11 +606,10 @@ | ||||
|                 </select> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <div class="custom-select"> | ||||
|                 <label for="output">Output</label> | ||||
|                 <select name="output" id="output" required> | ||||
|                   <option value="">Select a choice</option> | ||||
|                   <option value="json" selected>WLED JSON</option> | ||||
|                   <option value="ha">Home Assistant</option> | ||||
|                   <option value="curl">CURL</option> | ||||
| @@ -605,15 +618,15 @@ | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="row output" style="display: none"> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="device">Device</label> | ||||
|               <input type="text" name="device" id="device" required /> | ||||
|             </div> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="uniqueId">Unique Id</label> | ||||
|               <input type="text" name="uniqueId" id="uniqueId" required /> | ||||
|             </div> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="friendlyName">Friendly Name</label> | ||||
|               <input | ||||
|                 type="text" | ||||
| @@ -623,7 +636,7 @@ | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="row"> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <div class="custom-select"> | ||||
|                 <label for="segments">Segment Id</label> | ||||
|                 <select name="segments" id="segments"> | ||||
| @@ -633,7 +646,7 @@ | ||||
|                 </select> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="brightness">Brightness</label> | ||||
|               <div class="input-group"> | ||||
|                 <input | ||||
| @@ -655,7 +668,7 @@ | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="row"> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="animation">Animation</label> | ||||
|               <label class="switch"> | ||||
|                 <input | ||||
| @@ -666,7 +679,7 @@ | ||||
|                 <span class="switch-slider"></span> | ||||
|               </label> | ||||
|             </div> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="transparentImage">Transparent Image</label> | ||||
|               <label class="switch"> | ||||
|                 <input | ||||
| @@ -677,7 +690,7 @@ | ||||
|                 <span class="switch-slider"></span> | ||||
|               </label> | ||||
|             </div> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="resizeImage">Resize Image</label> | ||||
|               <label class="switch"> | ||||
|                 <input | ||||
| @@ -691,21 +704,21 @@ | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="row resizeImage"> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="width">Width</label> | ||||
|               <input type="number" name="width" id="width" value="16" /> | ||||
|             </div> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="height">Height</label> | ||||
|               <input type="number" name="height" id="height" value="16" /> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="row animation" style="display: none"> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="frames">Frames</label> | ||||
|               <input type="number" name="frames" id="frames" value="4" /> | ||||
|             </div> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="duration">Duration</label> | ||||
|               <div class="input-group"> | ||||
|                 <input | ||||
| @@ -722,7 +735,7 @@ | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="column" validate> | ||||
|             <div class="col" validate> | ||||
|               <label for="transition">Transition</label> | ||||
|               <div class="input-group"> | ||||
|                 <input | ||||
| @@ -741,7 +754,7 @@ | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="row transparentImage" style="display: none"> | ||||
|             <div class="column-full" validate> | ||||
|             <div class="col-full" validate> | ||||
|               <label for="color">Choose a color</label> | ||||
|               <small> | ||||
|                 Color that will replace the | ||||
| @@ -751,7 +764,7 @@ | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="row"> | ||||
|             <div class="column-full" validate> | ||||
|             <div class="col-full" validate> | ||||
|               <div class="custom-select"> | ||||
|                 <label for="images"> | ||||
|                   <span>Images upload to WLED</span> | ||||
| @@ -773,39 +786,47 @@ | ||||
|               type="file" | ||||
|               name="source" | ||||
|               id="source" | ||||
|               accept="image/jpg,image/jpeg,image/png,image/gif" | ||||
|               accept="image/*" | ||||
|               style="display: none" /> | ||||
|           </div> | ||||
|           <div class="row"> | ||||
|             <div class="column-full"> | ||||
|             <div class="col-full"> | ||||
|               <button type="button" class="button" id="btnGenerate"> | ||||
|                 Generate | ||||
|               </button> | ||||
|             </div> | ||||
|             <div class="col-small" id="gbth" style="display: none"> | ||||
|               <button | ||||
|                 type="button" | ||||
|                 class="button" | ||||
|                 onclick="window.location.href = WLED_URL;"> | ||||
|                 Back | ||||
|               </button> | ||||
|             </div> | ||||
|           </div> | ||||
|         </form> | ||||
|         <div id="preview" style="display: none"> | ||||
|           <div id="recreatedImage"></div> | ||||
|           <textarea name="response" id="response" readonly="readonly"> | ||||
|           <textarea name="response" id="response" rows="8" readonly="readonly"> | ||||
|           </textarea> | ||||
|           <div class="buttons"> | ||||
|             <div class="row"> | ||||
|               <div class="column"> | ||||
|               <div class="col"> | ||||
|                 <button type="button" class="button" id="btnCopyToClipboard"> | ||||
|                   Copy to Clipboard | ||||
|                 </button> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|               <div class="col"> | ||||
|                 <button type="button" class="button" id="btnSave">Save</button> | ||||
|               </div> | ||||
|               <div class="column"> | ||||
|               <div class="col"> | ||||
|                 <button type="button" class="button" id="btnDownloadPreset"> | ||||
|                   Download | ||||
|                 </button> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div class="row" id="simulate" style="display: none"> | ||||
|               <div class="column-full"> | ||||
|               <div class="col-full"> | ||||
|                 <button type="button" class="button" id="btnSimulatePreset"> | ||||
|                   Simulate | ||||
|                 </button> | ||||
| @@ -836,7 +857,7 @@ | ||||
|  | ||||
|     let WLED_URL = `${protocol}//${host}`; | ||||
|  | ||||
|     const hostname = element("hostname"); | ||||
|     const hostname = gId("hostname"); | ||||
|     hostname.value = host; | ||||
|  | ||||
|     hostname.addEventListener("blur", async () => { | ||||
| @@ -848,6 +869,9 @@ | ||||
|       hostnameLabel(); | ||||
|     }); | ||||
|  | ||||
|     gId("gbth").style.display = | ||||
|       window.location.protocol === "http:" ? "flex" : "none"; | ||||
|  | ||||
|     let jsonSaveWLED = []; | ||||
|     let jsonSendWLED = {}; | ||||
|  | ||||
| @@ -858,19 +882,19 @@ | ||||
|       hostnameLabel(); | ||||
|     })(); | ||||
|  | ||||
|     function element(id) { | ||||
|     function gId(id) { | ||||
|       return d.getElementById(id); | ||||
|     } | ||||
|  | ||||
|     function hostnameLabel() { | ||||
|       const link = element("wledEdit"); | ||||
|       const link = gId("wledEdit"); | ||||
|       link.href = WLED_URL + "/edit"; | ||||
|     } | ||||
|  | ||||
|     async function playlist() { | ||||
|       const { value: duration } = element("duration"); | ||||
|       const { value: transition } = element("transition"); | ||||
|       const { value: name } = element("name"); | ||||
|       const { value: duration } = gId("duration"); | ||||
|       const { value: transition } = gId("transition"); | ||||
|       const { value: name } = gId("name"); | ||||
|  | ||||
|       const urlPreset = `${WLED_URL}/presets.json`; | ||||
|       const url = `${WLED_URL}/json`; | ||||
| @@ -1003,7 +1027,7 @@ | ||||
|  | ||||
|     async function images() { | ||||
|       const url = `${WLED_URL}/edit?list=/`; | ||||
|       const select = element("images"); | ||||
|       const select = gId("images"); | ||||
|  | ||||
|       show(); | ||||
|  | ||||
| @@ -1049,9 +1073,9 @@ | ||||
|     } | ||||
|  | ||||
|     async function segments() { | ||||
|       const select = element("segments"); | ||||
|       const width = element("width"); | ||||
|       const height = element("height"); | ||||
|       const select = gId("segments"); | ||||
|       const width = gId("width"); | ||||
|       const height = gId("height"); | ||||
|  | ||||
|       show(); | ||||
|  | ||||
| @@ -1098,32 +1122,27 @@ | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     const dropzone = element("dropzone"); | ||||
|     const source = element("source"); | ||||
|  | ||||
|     dropzone.addEventListener("dragover", (e) => { | ||||
|     gId("dropzone").addEventListener("dragover", (e) => { | ||||
|       e.preventDefault(); | ||||
|     }); | ||||
|  | ||||
|     dropzone.addEventListener("drop", (e) => { | ||||
|     gId("dropzone").addEventListener("drop", (e) => { | ||||
|       e.preventDefault(); | ||||
|  | ||||
|       const dropzoneLabel = element("dropzoneLabel"); | ||||
|  | ||||
|       source.files = e.dataTransfer.files; | ||||
|  | ||||
|       const { name } = source.files[0]; | ||||
|       dropzoneLabel.textContent = `Image ${name} selected!`; | ||||
|       gId("dropzoneLabel").textContent = `Image ${name} selected!`; | ||||
|  | ||||
|       validate(e); | ||||
|     }); | ||||
|  | ||||
|     dropzone.addEventListener("click", () => { | ||||
|     gId("dropzone").addEventListener("click", () => { | ||||
|       source.click(); | ||||
|     }); | ||||
|  | ||||
|     source.addEventListener("change", (e) => { | ||||
|       const dropzoneLabel = element("dropzoneLabel"); | ||||
|     gId("source").addEventListener("change", (e) => { | ||||
|       const dropzoneLabel = gId("dropzoneLabel"); | ||||
|       const { value } = e.target; | ||||
|  | ||||
|       if (value) { | ||||
| @@ -1137,7 +1156,7 @@ | ||||
|       validate(e); | ||||
|     }); | ||||
|  | ||||
|     element("btnSimulatePreset").addEventListener("click", async () => { | ||||
|     gId("btnSimulatePreset").addEventListener("click", async () => { | ||||
|       const url = `${WLED_URL}/json/state`; | ||||
|  | ||||
|       const options = { | ||||
| @@ -1161,9 +1180,9 @@ | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     element("btnSave").addEventListener("click", async () => { | ||||
|       const { checked } = element("animation"); | ||||
|       const { value: name } = element("name"); | ||||
|     gId("btnSave").addEventListener("click", async () => { | ||||
|       const { checked } = gId("animation"); | ||||
|       const { value: name } = gId("name"); | ||||
|  | ||||
|       if (checked) { | ||||
|         await insert(jsonSaveWLED, true); | ||||
| @@ -1178,8 +1197,8 @@ | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     element("btnCopyToClipboard").addEventListener("click", async () => { | ||||
|       const response = element("response"); | ||||
|     gId("btnCopyToClipboard").addEventListener("click", async () => { | ||||
|       const response = gId("response"); | ||||
|  | ||||
|       response.select(); | ||||
|  | ||||
| @@ -1196,9 +1215,9 @@ | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     element("btnDownloadPreset").addEventListener("click", () => { | ||||
|       const { value: response } = element("response"); | ||||
|       const { value: output } = element("output"); | ||||
|     gId("btnDownloadPreset").addEventListener("click", () => { | ||||
|       const { value: response } = gId("response"); | ||||
|       const { value: output } = gId("output"); | ||||
|  | ||||
|       const timestamp = new Date().getTime(); | ||||
|       const filename = `WLED_${timestamp}`; | ||||
| @@ -1206,55 +1225,42 @@ | ||||
|       downloadFile(response, filename, output); | ||||
|     }); | ||||
|  | ||||
|     element("segments").addEventListener("change", (e) => { | ||||
|       const width = element("width"); | ||||
|       const height = element("height"); | ||||
|     gId("segments").addEventListener("change", (e) => { | ||||
|       const { width, height } = e.target.selectedOptions[0].dataset; | ||||
|  | ||||
|       const { width: w, height: h } = e.target.selectedOptions[0].dataset; | ||||
|  | ||||
|       width.value = w; | ||||
|       height.value = h; | ||||
|       gId("width").value = w; | ||||
|       gId("height").value = h; | ||||
|     }); | ||||
|  | ||||
|     element("output").addEventListener("change", (e) => { | ||||
|       const output = d.getElementsByClassName("output"); | ||||
|     gId("output").addEventListener("change", (e) => { | ||||
|       const { value } = e.target.selectedOptions[0]; | ||||
|  | ||||
|       Array.from(output).forEach(function (element) { | ||||
|         if (value === "ha") { | ||||
|           element.style.display = "flex"; | ||||
|         } else { | ||||
|           element.style.display = "none"; | ||||
|         } | ||||
|       }); | ||||
|       d.querySelector(".output").style.display = | ||||
|         value === "ha" ? "flex" : "none"; | ||||
|     }); | ||||
|  | ||||
|     element("brightnessValue").addEventListener("input", (e) => { | ||||
|       const brightness = element("brightness"); | ||||
|     gId("brightnessValue").addEventListener("input", (e) => { | ||||
|       const { value } = e.target; | ||||
|       const bri = value <= 255 ? value : 255; | ||||
|  | ||||
|       let bri = value <= 255 ? value : 255; | ||||
|  | ||||
|       brightness.value = bri; | ||||
|       gId("brightness").value = bri; | ||||
|       e.target.value = bri; | ||||
|     }); | ||||
|  | ||||
|     element("brightness").addEventListener("input", (e) => { | ||||
|       const brightnessValue = element("brightnessValue"); | ||||
|     gId("brightness").addEventListener("input", (e) => { | ||||
|       const { value } = e.target; | ||||
|  | ||||
|       brightnessValue.value = value; | ||||
|       gId("brightnessValue").value = value; | ||||
|     }); | ||||
|  | ||||
|     element("images").addEventListener("change", (e) => { | ||||
|       const dropzone = element("dropzone"); | ||||
|     gId("images").addEventListener("change", (e) => { | ||||
|       const dropzone = gId("dropzone"); | ||||
|       const { value } = e.target.selectedOptions[0]; | ||||
|  | ||||
|       if (!value) { | ||||
|         const dropzoneLabel = element("dropzoneLabel"); | ||||
|         const source = element("source"); | ||||
|         const source = gId("source"); | ||||
|  | ||||
|         dropzoneLabel.textContent = | ||||
|         gId("dropzoneLabel").textContent = | ||||
|           "Drag and drop a file here or click to select a local file"; | ||||
|         source.value = ""; | ||||
|  | ||||
| @@ -1264,33 +1270,23 @@ | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     element("transparentImage").addEventListener("change", (e) => { | ||||
|       const transparentImage = d.getElementsByClassName("transparentImage")[0]; | ||||
|     gId("transparentImage").addEventListener("change", (e) => { | ||||
|       const { checked } = e.target; | ||||
|  | ||||
|       if (checked) { | ||||
|         transparentImage.style.display = "flex"; | ||||
|       } else { | ||||
|         transparentImage.style.display = "none"; | ||||
|       } | ||||
|       d.querySelector(".transparentImage").style.display = checked | ||||
|         ? "flex" | ||||
|         : "none"; | ||||
|     }); | ||||
|  | ||||
|     element("resizeImage").addEventListener("change", (e) => { | ||||
|       const resizeImage = d.getElementsByClassName("resizeImage")[0]; | ||||
|       const pattern = element("pattern"); | ||||
|     gId("resizeImage").addEventListener("change", (e) => { | ||||
|       const { checked } = e.target; | ||||
|  | ||||
|       if (checked) { | ||||
|         resizeImage.style.display = "flex"; | ||||
|       } else { | ||||
|         resizeImage.style.display = "none"; | ||||
|       } | ||||
|       d.querySelector(".resizeImage").style.display = checked ? "flex" : "none"; | ||||
|     }); | ||||
|  | ||||
|     element("animation").addEventListener("change", (e) => { | ||||
|       const animation = d.getElementsByClassName("animation")[0]; | ||||
|       const pattern = element("pattern"); | ||||
|       const source = element("source"); | ||||
|     gId("animation").addEventListener("change", (e) => { | ||||
|       const animation = d.querySelector(".animation"); | ||||
|       const source = gId("source"); | ||||
|  | ||||
|       const { checked } = e.target; | ||||
|  | ||||
| @@ -1304,26 +1300,21 @@ | ||||
|         source.setAttribute("accept", "image/gif"); | ||||
|         animation.style.display = "flex"; | ||||
|       } else { | ||||
|         source.setAttribute( | ||||
|           "accept", | ||||
|           "image/jpg,image/jpeg,image/png,image/gif" | ||||
|         ); | ||||
|         source.setAttribute("accept", "image/*"); | ||||
|         animation.style.display = "none"; | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     element("btnGenerate").addEventListener("click", async (event) => { | ||||
|       const { checked } = element("animation"); | ||||
|  | ||||
|       const preview = element("preview"); | ||||
|       const recreatedImage = element("recreatedImage"); | ||||
|       const simulate = element("simulate"); | ||||
|     gId("btnGenerate").addEventListener("click", async (event) => { | ||||
|       const { checked } = gId("animation"); | ||||
|  | ||||
|       if (validate(event)) { | ||||
|         jsonSaveWLED.splice(0); | ||||
|  | ||||
|         preview.style.display = "block"; | ||||
|         recreatedImage.innerHTML = ""; | ||||
|         gId("preview").style.display = "block"; | ||||
|         gId("recreatedImage").innerHTML = ""; | ||||
|  | ||||
|         const simulate = gId("simulate"); | ||||
|  | ||||
|         if (checked) { | ||||
|           simulate.style.display = "none"; | ||||
| @@ -1335,36 +1326,41 @@ | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     function loadImage(src) { | ||||
|     async function createObjectURL(url) { | ||||
|       return fetch(url) | ||||
|         .then((response) => response.arrayBuffer()) | ||||
|         .then((buffer) => { | ||||
|           const binaryData = new Uint8Array(buffer); | ||||
|           const base64 = btoa(String.fromCharCode(...binaryData)); | ||||
|           return `data:image/png;base64,${base64}`; | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     function loadImage(url) { | ||||
|       return new Promise((resolve, reject) => { | ||||
|         const image = new Image(); | ||||
|  | ||||
|         image.addEventListener("load", function () { | ||||
|           resolve(image); | ||||
|         }); | ||||
|  | ||||
|         image.addEventListener("error", function () { | ||||
|           reject(new Error("Error loading image")); | ||||
|         }); | ||||
|  | ||||
|         image.src = src; | ||||
|         const img = new Image(); | ||||
|         img.onload = () => resolve(img); | ||||
|         img.onerror = (error) => reject(error); | ||||
|         img.src = url; | ||||
|       }); | ||||
|     } | ||||
|  | ||||
|     async function generate() { | ||||
|       const file = element("source").files[0]; | ||||
|       const file = gId("source").files[0]; | ||||
|  | ||||
|       const { value: images } = element("images"); | ||||
|       const { value: output } = element("output"); | ||||
|       const { value: images } = gId("images"); | ||||
|       const { value: output } = gId("output"); | ||||
|  | ||||
|       show(); | ||||
|  | ||||
|       try { | ||||
|         const response = element("response"); | ||||
|         const recreatedImage = element("recreatedImage"); | ||||
|  | ||||
|         const urlImage = !images ? URL.createObjectURL(file) : images; | ||||
|         const response = gId("response"); | ||||
|         const recreatedImage = gId("recreatedImage"); | ||||
|  | ||||
|         const urlImage = !images | ||||
|           ? URL.createObjectURL(file) | ||||
|           : await createObjectURL(images); | ||||
| 					 | ||||
|         const image = await loadImage(urlImage); | ||||
|         const { canvas, bri, id, i } = recreate(image); | ||||
|  | ||||
| @@ -1394,14 +1390,14 @@ | ||||
|     } | ||||
|  | ||||
|     async function generateAnimation() { | ||||
|       const file = element("source").files[0]; | ||||
|       const images = element("images"); | ||||
|       const response = element("response"); | ||||
|       const file = gId("source").files[0]; | ||||
|       const images = gId("images"); | ||||
|       const response = gId("response"); | ||||
|  | ||||
|       const { value: presetName } = element("name"); | ||||
|       const { value: amount } = element("frames"); | ||||
|       const { value: output } = element("output"); | ||||
|       const { value: duration } = element("duration"); | ||||
|       const { value: presetName } = gId("name"); | ||||
|       const { value: amount } = gId("frames"); | ||||
|       const { value: output } = gId("output"); | ||||
|       const { value: duration } = gId("duration"); | ||||
|  | ||||
|       const { text: imageName, value: imageValue } = images.selectedOptions[0]; | ||||
|  | ||||
| @@ -1410,7 +1406,7 @@ | ||||
|       try { | ||||
|         const body = new FormData(); | ||||
|  | ||||
|         if (imageValue === "upload") { | ||||
|         if (!imageValue) { | ||||
|           body.append("image", file); | ||||
|         } else { | ||||
|           const responseImage = await fetch(imageValue); | ||||
| @@ -1481,18 +1477,18 @@ | ||||
|     } | ||||
|  | ||||
|     function recreate(image) { | ||||
|       const { value: pattern } = element("pattern"); | ||||
|       const { value: segmentId } = element("segments"); | ||||
|       const { value: brightness } = element("brightness"); | ||||
|       const { value: inputWidth } = element("width"); | ||||
|       const { value: inputHeight } = element("height"); | ||||
|       const { checked: resizeImage } = element("resizeImage"); | ||||
|       const { value: pattern } = gId("pattern"); | ||||
|       const { value: segmentId } = gId("segments"); | ||||
|       const { value: brightness } = gId("brightness"); | ||||
|       const { value: inputWidth } = gId("width"); | ||||
|       const { value: inputHeight } = gId("height"); | ||||
|       const { checked: resizeImage } = gId("resizeImage"); | ||||
|  | ||||
|       const resizeWidth = parseInt(inputWidth); | ||||
|       const resizeHeight = parseInt(inputHeight); | ||||
|  | ||||
|       const { width: dataWidth, height: dataHeight } = | ||||
|         element("segments").selectedOptions[0].dataset; | ||||
|         gId("segments").selectedOptions[0].dataset; | ||||
|  | ||||
|       const segmentWidth = parseInt(dataWidth); | ||||
|       const segmentHeight = parseInt(dataHeight); | ||||
| @@ -1636,8 +1632,8 @@ | ||||
|     } | ||||
|  | ||||
|     function pixelColor(r, g, b, a) { | ||||
|       const { checked } = element("transparentImage"); | ||||
|       const { value } = element("color"); | ||||
|       const { checked } = gId("transparentImage"); | ||||
|       const { value } = gId("color"); | ||||
|  | ||||
|       if (a === 0) { | ||||
|         if (checked) { | ||||
| @@ -1733,10 +1729,10 @@ | ||||
|     } | ||||
|  | ||||
|     function yaml(jsonData) { | ||||
|       const { value: device } = element("device"); | ||||
|       const { value: friendly_name } = element("friendlyName"); | ||||
|       const { value: unique_id } = element("uniqueId"); | ||||
|       const { value: hostname } = element("hostname"); | ||||
|       const { value: device } = gId("device"); | ||||
|       const { value: friendly_name } = gId("friendlyName"); | ||||
|       const { value: unique_id } = gId("uniqueId"); | ||||
|       const { value: hostname } = gId("hostname"); | ||||
|  | ||||
|       if (device) { | ||||
|         const yamlData = { | ||||
| @@ -1755,7 +1751,7 @@ | ||||
|     } | ||||
|  | ||||
|     function curl(jsonData) { | ||||
|       const { value: hostname } = element("hostname"); | ||||
|       const { value: hostname } = gId("hostname"); | ||||
|       return `curl -X POST "http://${hostname}/json/state" -d '${jsonData}' -H "Content-Type: application/json"`; | ||||
|     } | ||||
|  | ||||
| @@ -1793,7 +1789,7 @@ | ||||
|       duration = 2000, | ||||
|       hideElement = "preview" | ||||
|     ) { | ||||
|       const hide = element(hideElement); | ||||
|       const hide = gId(hideElement); | ||||
|       const toast = d.createElement("div"); | ||||
|       const wait = 100; | ||||
|  | ||||
| @@ -1819,7 +1815,7 @@ | ||||
|  | ||||
|       toast.appendChild(progress); | ||||
|  | ||||
|       element("toast-container").appendChild(toast); | ||||
|       gId("toast-container").appendChild(toast); | ||||
|  | ||||
|       setTimeout(() => { | ||||
|         toast.style.opacity = 0; | ||||
| @@ -1847,7 +1843,7 @@ | ||||
|         } | ||||
|       }); | ||||
|  | ||||
|       const container = element(id); | ||||
|       const container = gId(id); | ||||
|       container.innerHTML = ""; | ||||
|       container.appendChild(carousel); | ||||
|  | ||||
| @@ -1895,7 +1891,7 @@ | ||||
|     } | ||||
|  | ||||
|     function show() { | ||||
|       const overlay = element("overlay"); | ||||
|       const overlay = gId("overlay"); | ||||
|       overlay.classList.add("loading"); | ||||
|       overlay.style.display = "block"; | ||||
|       overlay.style.cursor = "not-allowed"; | ||||
| @@ -1904,7 +1900,7 @@ | ||||
|     } | ||||
|  | ||||
|     function hide() { | ||||
|       const overlay = element("overlay"); | ||||
|       const overlay = gId("overlay"); | ||||
|       overlay.classList.remove("loading"); | ||||
|       overlay.style.display = "none"; | ||||
|       overlay.style.cursor = "default"; | ||||
| @@ -1915,7 +1911,7 @@ | ||||
|     function validate(event) { | ||||
|       event.preventDefault(); | ||||
|  | ||||
|       const form = element("formGenerate"); | ||||
|       const form = gId("formGenerate"); | ||||
|       const inputs = form.querySelectorAll("input, select, textarea"); | ||||
|  | ||||
|       let isValid = true; | ||||
|   | ||||
| @@ -40,9 +40,10 @@ | ||||
| 				let path = l.pathname; | ||||
| 				let paths = path.slice(1,path.endsWith('/')?-1:undefined).split("/"); | ||||
| 				if (paths.length > 1) { | ||||
| 					paths.pop(); // remove "settings" | ||||
| 					locproto = l.protocol; | ||||
| 					loc = true; | ||||
| 					locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths[0]; | ||||
| 					locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths.join('/'); | ||||
| 				} | ||||
| 			} | ||||
| 			loadJS(getURL('/settings/s.js?p=0'), false);	// If we set async false, file is loaded and executed, then next statement is processed | ||||
|   | ||||
| @@ -47,9 +47,11 @@ | ||||
| 			let path = l.pathname; | ||||
| 			let paths = path.slice(1,path.endsWith('/')?-1:undefined).split("/"); | ||||
| 			if (paths.length > 2) { | ||||
| 				paths.pop(); // remove "2d" | ||||
| 				paths.pop(); // remove "settings" | ||||
| 				locproto = l.protocol; | ||||
| 				loc = true; | ||||
| 				locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths[0]; | ||||
| 				locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths.join('/'); | ||||
| 			} | ||||
| 		} | ||||
| 		loadJS(getURL('/settings/s.js?p=10'), false);	// If we set async false, file is loaded and executed, then next statement is processed | ||||
|   | ||||
| @@ -66,9 +66,11 @@ | ||||
| 			let path = l.pathname; | ||||
| 			let paths = path.slice(1,path.endsWith('/')?-1:undefined).split("/"); | ||||
| 			if (paths.length > 2) { | ||||
| 				paths.pop(); // remove "dmx" | ||||
| 				paths.pop(); // remove "settings" | ||||
| 				locproto = l.protocol; | ||||
| 				loc = true; | ||||
| 				locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths[0]; | ||||
| 				locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths.join('/'); | ||||
| 			} | ||||
| 		} | ||||
| 		loadJS(getURL('/settings/s.js?p=7'), false);	// If we set async false, file is loaded and executed, then next statement is processed | ||||
|   | ||||
| @@ -16,6 +16,13 @@ | ||||
| 		function B(){window.open(getURL("/settings"),"_self");} | ||||
| 		function gId(n){return d.getElementById(n);} | ||||
| 		function off(n){d.getElementsByName(n)[0].value = -1;} | ||||
| 		// these functions correspond to C macros found in const.h | ||||
| 		function isPWM(t) { return t > 40 && t < 46; }                      // is PWM type | ||||
| 		function isAna(t) { return t == 40 || isPWM(t); }                   // is analog type | ||||
| 		function isDig(t) { return (t > 15 && t < 40) || isD2P(t); }        // is digital type | ||||
| 		function isD2P(t) { return t > 47 && t < 64; }                      // is digital 2 pin type | ||||
| 		function is16b(t) { return t == 26 || t == 29 }                     // is digital 16 bit type | ||||
| 		function isVir(t) { return t >= 80 && t < 96; }                     // is virtual type | ||||
| 		// https://www.educative.io/edpresso/how-to-dynamically-load-a-js-file-in-javascript | ||||
| 		function loadJS(FILE_URL, async = true) { | ||||
| 			let scE = d.createElement("script"); | ||||
| @@ -52,137 +59,171 @@ | ||||
| 			maxB = b; maxV = v; maxM = m; maxPB = p; maxL = l; | ||||
| 		} | ||||
| 		function pinsOK() { | ||||
| 			var LCs = d.Sf.querySelectorAll("#mLC input[name^=L]"); // input fields | ||||
| 			for (i=0; i<LCs.length; i++) { | ||||
| 				var nm = LCs[i].name.substring(0,2); | ||||
| 			var ok = true; | ||||
| 			var nList = d.Sf.querySelectorAll("#mLC input[name^=L]"); | ||||
| 			nList.forEach((LC,i)=>{ | ||||
| 				if (!ok) return; // prevent iteration after conflict | ||||
| 				let nm = LC.name.substring(0,2); | ||||
| 				let n = LC.name.substring(2); | ||||
| 				let t = parseInt(d.Sf["LT"+n].value, 10); // LED type SELECT | ||||
| 				// ignore IP address | ||||
| 				if (nm=="L0" || nm=="L1" || nm=="L2" || nm=="L3") { | ||||
| 					var n = LCs[i].name.substring(2); | ||||
| 					var t = parseInt(d.getElementsByName("LT"+n)[0].value, 10); // LED type SELECT | ||||
| 					if (t>=80) continue; | ||||
| 					if (t>=80) return; | ||||
| 				} | ||||
| 				//check for pin conflicts | ||||
| 				if (nm=="L0" || nm=="L1" || nm=="L2" || nm=="L3" || nm=="L4"/* || nm=="RL" || nm=="BT" || nm=="IR"*/) | ||||
| 					if (LCs[i].value!="" && LCs[i].value!="-1") { | ||||
| 						var p = d.rsvd.concat(d.um_p); // used pin array | ||||
| 				if (nm=="L0" || nm=="L1" || nm=="L2" || nm=="L3" || nm=="L4") | ||||
| 					if (LC.value!="" && LC.value!="-1") { | ||||
| 						let p = d.rsvd.concat(d.um_p); // used pin array | ||||
| 						d.Sf.querySelectorAll("select.pin").forEach((e)=>{if(e.value>-1)p.push(parseInt(e.value));}) // buttons, IR & relay | ||||
| 						if (p.some((e)=>e==parseInt(LCs[i].value))) { | ||||
| 						if (p.some((e)=>e==parseInt(LC.value))) { | ||||
| 							alert(`Sorry, pins ${JSON.stringify(p)} can't be used.`); | ||||
| 							LCs[i].value=""; | ||||
| 							LCs[i].focus(); | ||||
| 							return false; | ||||
| 						} | ||||
| 						else if (/*!(nm == "IR" || nm=="BT") &&*/ d.ro_gpio.some((e)=>e==parseInt(LCs[i].value))) { | ||||
| 							LC.value=""; | ||||
| 							LC.focus(); | ||||
| 							ok = false; | ||||
| 							return; | ||||
| 						} else if (d.ro_gpio.some((e)=>e==parseInt(LC.value))) { | ||||
| 							alert(`Sorry, pins ${JSON.stringify(d.ro_gpio)} are input only.`); | ||||
| 							LCs[i].value=""; | ||||
| 							LCs[i].focus(); | ||||
| 							return false; | ||||
| 							LC.value=""; | ||||
| 							LC.focus(); | ||||
| 							ok = false; | ||||
| 							return; | ||||
| 						} | ||||
| 						for (j=i+1; j<LCs.length; j++) { | ||||
| 							var n2 = LCs[j].name.substring(0,2); | ||||
| 							if (n2=="L0" || n2=="L1" || n2=="L2" || n2=="L3" || n2=="L4"/* || n2=="RL" || n2=="BT" || n2=="IR"*/) { | ||||
| 						for (j=i+1; j<nList.length; j++) { | ||||
| 							let n2 = nList[j].name.substring(0,2); | ||||
| 							if (n2=="L0" || n2=="L1" || n2=="L2" || n2=="L3" || n2=="L4") { | ||||
| 								if (n2.substring(0,1)==="L") { | ||||
| 									var m  = LCs[j].name.substring(2); | ||||
| 									var t2 = parseInt(d.getElementsByName("LT"+m)[0].value, 10); | ||||
| 									var m  = nList[j].name.substring(2); | ||||
| 									var t2 = parseInt(d.Sf["LT"+m].value, 10); | ||||
| 									if (t2>=80) continue; | ||||
| 								} | ||||
| 								if (LCs[j].value!="" && LCs[i].value==LCs[j].value) { | ||||
| 									alert(`Pin conflict between ${LCs[i].name}/${LCs[j].name}!`); | ||||
| 									LCs[j].value=""; | ||||
| 									LCs[j].focus(); | ||||
| 									return false; | ||||
| 								if (nList[j].value!="" && nList[i].value==nList[j].value) { | ||||
| 									alert(`Pin conflict between ${LC.name}/${nList[j].name}!`); | ||||
| 									nList[j].value=""; | ||||
| 									nList[j].focus(); | ||||
| 									ok = false; | ||||
| 									return; | ||||
| 								} | ||||
| 							} | ||||
| 						} | ||||
| 					} | ||||
| 			} | ||||
| 			return true; | ||||
| 			}); | ||||
| 			return ok; | ||||
| 		} | ||||
| 		function trySubmit(e) { | ||||
| 			d.Sf.data.value = ''; | ||||
| 			e.preventDefault(); | ||||
| 			if (!pinsOK()) {e.stopPropagation();return false;} // Prevent form submission and contact with server | ||||
| 			if (bquot > 100) {var msg = "Too many LEDs for me to handle!"; if (maxM < 10000) msg += "\n\rConsider using an ESP32."; alert(msg);} | ||||
| 			if (!d.Sf.ABL.checked || d.Sf.PPL.checked) d.Sf.MA.value = 0; // submit 0 as ABL (PPL will handle it) | ||||
| 			if (d.Sf.checkValidity()) d.Sf.submit(); //https://stackoverflow.com/q/37323914 | ||||
| 		} | ||||
| 		function enABL() | ||||
| 		{ | ||||
| 			var en = gId('able').checked; | ||||
| 			d.Sf.LA.value = (en) ? laprev:0; | ||||
| 			var en = d.Sf.ABL.checked; | ||||
| 			gId('abl').style.display = (en) ? 'inline':'none'; | ||||
| 			gId('psu2').style.display = (en) ? 'inline':'none'; | ||||
| 			if (d.Sf.LA.value > 0) setABL(); | ||||
| 			if (!en) d.Sf.PPL.checked = false; | ||||
| 			UI(); | ||||
| 		} | ||||
| 		function enLA() | ||||
| 		// enable per port limiter and calculate current | ||||
| 		function enPPL(sDI=0) | ||||
| 		{ | ||||
| 			var val = d.Sf.LAsel.value; | ||||
| 			d.Sf.LA.value = val; | ||||
| 			gId('LAdis').style.display = (val == 50) ? 'inline':'none'; | ||||
| 			UI(); | ||||
| 			const abl = d.Sf.ABL.checked; | ||||
| 			const ppl = d.Sf.PPL.checked; | ||||
| 			let sumMA = 0; | ||||
| 			d.Sf.MA.readonly = ppl; | ||||
| 			d.Sf.MA.min = abl && !ppl ? 250 : 0; | ||||
| 			gId("psuMA").style.display = ppl ? 'none' : 'inline'; | ||||
| 			gId("ppldis").style.display = ppl ? 'inline' : 'none'; | ||||
| 			// set PPL minimum value and clear actual PPL limit if ABL disabled | ||||
| 			d.Sf.querySelectorAll("#mLC input[name^=MA]").forEach((i,n)=>{ | ||||
| 				gId("PSU"+n).style.display = ppl ? "inline" : "none"; | ||||
| 				const t = parseInt(d.Sf["LT"+n].value); // LED type SELECT | ||||
| 				const c = parseInt(d.Sf["LC"+n].value); //get LED count | ||||
| 				i.min = ppl && !(isVir(t) || isAna(t)) ? 250 : 0; | ||||
| 				if (!abl || isVir(t) || isAna(t)) i.value = 0; | ||||
| 				else if (ppl) sumMA += parseInt(i.value,10); | ||||
| 				else if (sDI) i.value = Math.round(parseInt(d.Sf.MA.value,10)*c/sDI); | ||||
| 			}); | ||||
| 			if (ppl) d.Sf.MA.value = sumMA; // populate UI ABL value if PPL used | ||||
| 		} | ||||
| 		function enLA(s,n) | ||||
| 		{ | ||||
| 			const t = parseInt(d.Sf["LT"+n].value); // LED type SELECT | ||||
| 			gId('LAdis'+n).style.display = s.selectedIndex==5 ? "inline" : "none"; | ||||
| 			d.Sf["LA"+n].value = s.value==="0" ? 55 : s.value; | ||||
| 			d.Sf["LA"+n].min = (isVir(t) || isAna(t)) ? 0 : 1; | ||||
| 		} | ||||
| 		function setABL() | ||||
| 		{ | ||||
| 			gId('able').checked = true; | ||||
| 			d.Sf.LAsel.value = 50; | ||||
| 			switch (parseInt(d.Sf.LA.value)) { | ||||
| 				case 0: gId('able').checked = false; enABL(); break; | ||||
| 				case 30: d.Sf.LAsel.value = 30; break; | ||||
| 				case 35: d.Sf.LAsel.value = 35; break; | ||||
| 				case 55: d.Sf.LAsel.value = 55; break; | ||||
| 				case 255: d.Sf.LAsel.value = 255; break; | ||||
| 				default: gId('LAdis').style.display = 'inline'; | ||||
| 			} | ||||
| 			d.Sf.ABL.checked = parseInt(d.Sf.MA.value) > 0; | ||||
| 			// check if ABL is enabled (max mA entered per output) | ||||
| 			d.Sf.querySelectorAll("#mLC input[name^=MA]").forEach((i,n)=>{ | ||||
| 				if (parseInt(i.value) > 0) d.Sf.ABL.checked = true; | ||||
| 			}); | ||||
| 			// select appropriate LED current | ||||
| 			d.Sf.querySelectorAll("#mLC select[name^=LAsel]").forEach((sel,n)=>{ | ||||
| 				sel.value = 0; // set custom | ||||
| 				switch (parseInt(d.Sf["LA"+n].value)) { | ||||
| 					case 0: break; // disable ABL | ||||
| 					case 15: sel.value = 15; break; | ||||
| 					case 30: sel.value = 30; break; | ||||
| 					case 35: sel.value = 35; break; | ||||
| 					case 55: sel.value = 55; break; | ||||
| 					case 255: sel.value = 255; break; | ||||
| 				} | ||||
| 				enLA(sel,n); | ||||
| 			}); | ||||
| 			enABL(); | ||||
| 			gId('m1').innerHTML = maxM; | ||||
| 		} | ||||
| 		//returns mem usage | ||||
| 		function getMem(t, n) { | ||||
| 			if (isAna(t)) return 5;	// analog | ||||
| 			let len = parseInt(d.getElementsByName("LC"+n)[0].value); | ||||
| 			len += parseInt(d.getElementsByName("SL"+n)[0].value); // skipped LEDs are allocated too | ||||
| 			let dbl = 0; | ||||
| 			if (d.Sf.LD.checked) dbl = len * 4;	// double buffering | ||||
| 			if (t < 32) { | ||||
| 				if (t==26 || t==29) len *= 2; // 16 bit LEDs | ||||
| 			let ch = 3; | ||||
| 			let mul = 1; | ||||
| 			if (isDig(t)) { | ||||
| 				if (is16b(t)) len *= 2; // 16 bit LEDs | ||||
| 				if (t > 28 && t < 40) ch = 4; //RGBW | ||||
| 				if (maxM < 10000 && d.getElementsByName("L0"+n)[0].value == 3) { //8266 DMA uses 5x the mem | ||||
| 					if (t > 28) return len*20 + dbl; //RGBW | ||||
| 					return len*15 + dbl; | ||||
| 				} else if (maxM >= 10000) //ESP32 RMT uses double buffer? | ||||
| 				{ | ||||
| 					if (t > 28) return len*8 + dbl; //RGBW | ||||
| 					return len*6 + dbl; | ||||
| 					mul = 5; | ||||
| 				} | ||||
| 				if (t > 28) return len*4 + dbl; //RGBW | ||||
| 				return len*3 + dbl; | ||||
| 				if (maxM >= 10000) { //ESP32 RMT uses double buffer? | ||||
| 					mul = 2; | ||||
| 				} | ||||
| 				if (d.Sf.LD.checked) dbl = len * ch; // double buffering | ||||
| 			} | ||||
| 			if (t > 31 && t < 48) return 5;	// analog | ||||
| 			return len*3 + dbl; | ||||
| 			if (isVir(t) && t == 88) ch = 4; | ||||
| 			return len * ch * mul + dbl; | ||||
| 		} | ||||
|  | ||||
| 		function UI(change=false) | ||||
| 		{ | ||||
| 			let isRGBW = false, gRGBW = false, memu = 0; | ||||
|  | ||||
| 			gId('ampwarning').style.display = (d.Sf.MA.value > 7200) ? 'inline':'none'; | ||||
|  | ||||
| 			if (d.Sf.LA.value == 255) laprev = 12; | ||||
| 			else if (d.Sf.LA.value > 0) laprev = d.Sf.LA.value; | ||||
| 			let busMA = 0; | ||||
| 			let sLC = 0, sPC = 0, sDI = 0, maxLC = 0; | ||||
| 			const ablEN = d.Sf.ABL.checked; | ||||
|  | ||||
| 			// enable/disable LED fields | ||||
| 			d.Sf.querySelectorAll("#mLC select[name^=LT]").forEach((s)=>{ | ||||
| 				// is the field a LED type? | ||||
| 				var n = s.name.substring(2); | ||||
| 				var t = parseInt(s.value); | ||||
| 				gId("p0d"+n).innerHTML = (t>=80 && t<96) ? "IP address:" : (t > 49) ? "Data GPIO:" : (t > 41) ? "GPIOs:" : "GPIO:"; | ||||
| 				gId("p1d"+n).innerHTML = (t> 49 && t<64) ? "Clk GPIO:" : ""; | ||||
| 				gId("p0d"+n).innerHTML = isVir(t) ? "IP address:" : isD2P(t) ? "Data GPIO:" : (t > 41) ? "GPIOs:" : "GPIO:"; | ||||
| 				gId("p1d"+n).innerHTML = isD2P(t) ? "Clk GPIO:" : ""; | ||||
| 				gId("abl"+n).style.display = (!ablEN || isVir(t) || isAna(t)) ? "none" : "inline"; | ||||
| 				//var LK = d.getElementsByName("L1"+n)[0]; // clock pin | ||||
|  | ||||
| 				memu += getMem(t, n); // calc memory | ||||
|  | ||||
| 				// enumerate pins | ||||
| 				for (p=1; p<5; p++) { | ||||
| 					var LK = d.getElementsByName("L"+p+n)[0]; // secondary pins | ||||
| 					var LK = d.Sf["L"+p+n]; // secondary pins | ||||
| 					if (!LK) continue; | ||||
| 					if (((t>=80 && t<96) && p<4) || (t>49 && p==1) || (t>41 && t < 50 && (p+40 < t))) // TYPE_xxxx values from const.h | ||||
| 					if ((isVir(t) && p<4) || (isD2P(t) && p==1) || (isPWM(t) && (p+40 < t))) // TYPE_xxxx values from const.h | ||||
| 					{ | ||||
| 						// display pin field | ||||
| 						LK.style.display = "inline"; | ||||
| @@ -196,21 +237,23 @@ | ||||
| 				} | ||||
| 				if (change) { | ||||
| 					gId("rf"+n).checked = (gId("rf"+n).checked || t == 31); // LEDs require data in off state | ||||
| 					if (t > 31 && t < 48) d.getElementsByName("LC"+n)[0].value = 1; // for sanity change analog count just to 1 LED | ||||
| 					if (isAna(t)) d.Sf["LC"+n].value = 1; // for sanity change analog count just to 1 LED | ||||
| 					d.Sf["LA"+n].min = (isVir(t) || isAna(t)) ? 0 : 1; | ||||
| 					d.Sf["MA"+n].min = (isVir(t) || isAna(t)) ? 0 : 250; | ||||
| 				} | ||||
| 				gId("rf"+n).onclick = (t == 31) ? (()=>{return false}) : (()=>{});  // prevent change for TM1814 | ||||
| 				gRGBW |= isRGBW = ((t > 17 && t < 22) || (t > 28 && t < 32) || (t > 40 && t < 46 && t != 43) || t == 88); // RGBW checkbox, TYPE_xxxx values from const.h | ||||
| 				gId("co"+n).style.display = ((t >= 80 && t < 96) || (t >= 40 && t < 48)) ? "none":"inline";  // hide color order for PWM | ||||
| 				gId("dig"+n+"w").style.display = (t > 28 && t < 32) ? "inline":"none";  // show swap channels dropdown | ||||
| 				if (!(t > 28 && t < 32)) d.getElementsByName("WO"+n)[0].value = 0; // reset swapping | ||||
| 				gId("dig"+n+"c").style.display = (t >= 40 && t < 48) ? "none":"inline";  // hide count for analog | ||||
| 				gId("dig"+n+"r").style.display = (t >= 80 && t < 96) ? "none":"inline";  // hide reversed for virtual | ||||
| 				gId("dig"+n+"s").style.display = ((t >= 80 && t < 96) || (t >= 40 && t < 48)) ? "none":"inline";  // hide skip 1st for virtual & analog | ||||
| 				gId("dig"+n+"f").style.display = ((t >= 16 && t < 32) || (t >= 50 && t < 64)) ? "inline":"none";  // hide refresh | ||||
| 				gId("dig"+n+"a").style.display = (isRGBW && t != 40) ? "inline":"none";  // auto calculate white | ||||
| 				gId("dig"+n+"l").style.display = (t > 48 && t < 64) ? "inline":"none";  // bus clock speed | ||||
| 				gId("rev"+n).innerHTML = (t >= 40 && t < 48) ? "Inverted output":"Reversed (rotated 180°)";  // change reverse text for analog | ||||
| 				gId("psd"+n).innerHTML = (t >= 40 && t < 48) ? "Index:":"Start:";    // change analog start description | ||||
| 				gId("co"+n).style.display = (isVir(t) || isAna(t)) ? "none":"inline";  // hide color order for PWM | ||||
| 				gId("dig"+n+"w").style.display = (isDig(t) && isRGBW) ? "inline":"none";  // show swap channels dropdown | ||||
| 				if (!(isDig(t) && isRGBW)) d.Sf["WO"+n].value = 0; // reset swapping | ||||
| 				gId("dig"+n+"c").style.display = (isAna(t)) ? "none":"inline";  // hide count for analog | ||||
| 				gId("dig"+n+"r").style.display = (isVir(t)) ? "none":"inline";  // hide reversed for virtual | ||||
| 				gId("dig"+n+"s").style.display = (isVir(t) || isAna(t)) ? "none":"inline";  // hide skip 1st for virtual & analog | ||||
| 				gId("dig"+n+"f").style.display = (isDig(t)) ? "inline":"none";  // hide refresh | ||||
| 				gId("dig"+n+"a").style.display = (isRGBW) ? "inline":"none";  // auto calculate white | ||||
| 				gId("dig"+n+"l").style.display = (isD2P(t) || isPWM(t)) ? "inline":"none";  // bus clock speed / PWM speed (relative) (not On/Off) | ||||
| 				gId("rev"+n).innerHTML = isAna(t) ? "Inverted output":"Reversed (rotated 180°)";  // change reverse text for analog | ||||
| 				//gId("psd"+n).innerHTML = isAna(t) ? "Index:":"Start:";    // change analog start description | ||||
| 			}); | ||||
| 			// display global white channel overrides | ||||
| 			gId("wc").style.display = (gRGBW) ? 'inline':'none'; | ||||
| @@ -218,70 +261,72 @@ | ||||
| 				d.Sf.AW.selectedIndex = 0; | ||||
| 				d.Sf.CR.checked = false; | ||||
| 			} | ||||
| 			// check for pin conflicts | ||||
| 			var LCs = d.Sf.querySelectorAll("#mLC input[name^=L]"); // input fields | ||||
| 			var sLC = 0, sPC = 0, maxLC = 0; | ||||
| 			for (i=0; i<LCs.length; i++) { | ||||
| 				var nm = LCs[i].name.substring(0,2);  // field name | ||||
| 				var n  = LCs[i].name.substring(2);    // bus number | ||||
| 			// update start indexes, max values, calculate current, etc | ||||
| 			var nList = d.Sf.querySelectorAll("#mLC input[name^=L]"); | ||||
| 			nList.forEach((LC,i)=>{ | ||||
| 				let nm = LC.name.substring(0,2);  // field name | ||||
| 				let n  = LC.name.substring(2);    // bus number | ||||
| 				let t  = parseInt(d.Sf["LT"+n].value); // LED type SELECT | ||||
| 				// do we have a led count field | ||||
| 				if (nm=="LC") { | ||||
| 					var c=parseInt(LCs[i].value,10); //get LED count | ||||
| 					let c = parseInt(LC.value,10); //get LED count | ||||
| 					if (!customStarts || !startsDirty[n]) gId("ls"+n).value=sLC; //update start value | ||||
| 					gId("ls"+n).disabled = !customStarts; //enable/disable field editing | ||||
| 					if(c){ | ||||
| 						var s = parseInt(gId("ls"+n).value); //start value | ||||
| 					if (c) { | ||||
| 						let s = parseInt(gId("ls"+n).value); //start value | ||||
| 						if (s+c > sLC) sLC = s+c; //update total count | ||||
| 						if(c>maxLC)maxLC=c; //max per output | ||||
| 						var t = parseInt(d.getElementsByName("LT"+n)[0].value); // LED type SELECT | ||||
| 						if (t<80) sPC+=c; //virtual out busses do not count towards physical LEDs | ||||
| 						if (c > maxLC) maxLC = c; //max per output | ||||
| 						if (!isVir(t)) sPC += c; //virtual out busses do not count towards physical LEDs | ||||
| 						if (!(isVir(t) || isAna(t))) { | ||||
| 							sDI += c; // summarize digital LED count | ||||
| 							let maPL = parseInt(d.Sf["LA"+n].value); | ||||
| 							if (maPL == 255) maPL = 12; | ||||
| 							busMA += maPL*c; // summarize maximum bus current (calculated) | ||||
| 						} | ||||
| 					} // increase led count | ||||
| 					continue; | ||||
| 					return; | ||||
| 				} | ||||
| 				// do we have led pins for digital leds | ||||
| 				if (nm=="L0" || nm=="L1") { | ||||
| 					var lc=d.getElementsByName("LC"+n)[0]; | ||||
| 					lc.max=maxPB; // update max led count value | ||||
| 					d.Sf["LC"+n].max = maxPB; // update max led count value | ||||
| 				} | ||||
| 				// ignore IP address (stored in pins for virtual busses) | ||||
| 				if (nm=="L0" || nm=="L1" || nm=="L2" || nm=="L3") { | ||||
| 					var t = parseInt(d.getElementsByName("LT"+n)[0].value); // LED type SELECT | ||||
| 					if (t>=80) { | ||||
| 						LCs[i].max = 255; | ||||
| 						LCs[i].min = 0; | ||||
| 						LCs[i].style.color="#fff"; | ||||
| 						continue; // do not check conflicts | ||||
| 					if (isVir(t)) { | ||||
| 						LC.max = 255; | ||||
| 						LC.min = 0; | ||||
| 						LC.style.color="#fff"; | ||||
| 						return; // do not check conflicts | ||||
| 					} else { | ||||
| 						LCs[i].max = d.max_gpio; | ||||
| 						LCs[i].min = -1; | ||||
| 						LC.max = d.max_gpio; | ||||
| 						LC.min = -1; | ||||
| 					} | ||||
| 				} | ||||
| 				// check for pin conflicts | ||||
| 				if (nm=="L0" || nm=="L1" || nm=="L2" || nm=="L3" || nm=="L4"/* || nm=="RL" || nm=="BT" || nm=="IR"*/) | ||||
| 					if (LCs[i].value!="" && LCs[i].value!="-1") { | ||||
| 						var p = d.rsvd.concat(d.um_p); // used pin array | ||||
| 				// check for pin conflicts & color fields | ||||
| 				if (nm=="L0" || nm=="L1" || nm=="L2" || nm=="L3" || nm=="L4") | ||||
| 					if (LC.value!="" && LC.value!="-1") { | ||||
| 						let p = d.rsvd.concat(d.um_p); // used pin array | ||||
| 						d.Sf.querySelectorAll("select.pin").forEach((e)=>{if(e.value>-1)p.push(parseInt(e.value));}) // buttons, IR & relay | ||||
| 						for (j=0; j<LCs.length; j++) { | ||||
| 						for (j=0; j<nList.length; j++) { | ||||
| 							if (i==j) continue; | ||||
| 							var n2 = LCs[j].name.substring(0,2); | ||||
| 							if (n2=="L0" || n2=="L1" || n2=="L2" || n2=="L3" || n2=="L4"/* || n2=="RL" || n2=="BT" || n2=="IR"*/) { | ||||
| 							let n2 = nList[j].name.substring(0,2); | ||||
| 							if (n2=="L0" || n2=="L1" || n2=="L2" || n2=="L3" || n2=="L4") { | ||||
| 								if (n2.substring(0,1)==="L") { | ||||
| 									var m  = LCs[j].name.substring(2); | ||||
| 									var t2 = parseInt(d.getElementsByName("LT"+m)[0].value, 10); | ||||
| 									if (t2>=80) continue; | ||||
| 									let m  = nList[j].name.substring(2); | ||||
| 									let t2 = parseInt(d.Sf["LT"+m].value, 10); | ||||
| 									if (isVir(t2)) continue; | ||||
| 								} | ||||
| 								if (LCs[j].value!="" && LCs[j].value!="-1") p.push(parseInt(LCs[j].value,10));  // add current pin | ||||
| 								if (nList[j].value!="" && nList[j].value!="-1") p.push(parseInt(nList[j].value,10));  // add current pin | ||||
| 							} | ||||
| 						} | ||||
| 						// now check for conflicts | ||||
| 						if (p.some((e)=>e==parseInt(LCs[i].value))) LCs[i].style.color="red"; else LCs[i].style.color=d.ro_gpio.some((e)=>e==parseInt(LCs[i].value))?"orange":"#fff"; | ||||
| 						if (p.some((e)=>e==parseInt(LC.value))) LC.style.color = "red"; | ||||
| 						else LC.style.color = d.ro_gpio.some((e)=>e==parseInt(LC.value)) ? "orange" : "#fff"; | ||||
| 					} | ||||
| 				// check buttons, IR & relay | ||||
| 				//if (nm=="IR" || nm=="BT" || nm=="RL") { | ||||
| 				//	LCs[i].max = d.max_gpio; | ||||
| 				//	LCs[i].min = -1; | ||||
| 				//} | ||||
| 			} | ||||
| 			}); | ||||
| 			// distribute ABL current if not using PPL | ||||
| 			enPPL(sDI); | ||||
|  | ||||
| 			// update total led count | ||||
| 			gId("lc").textContent = sLC; | ||||
| 			gId("pc").textContent = (sLC == sPC) ? "":"(" + sPC + " physical)"; | ||||
| @@ -294,27 +339,19 @@ | ||||
| 			gId('ledwarning').style.color = (maxLC > Math.max(maxPB,800) || bquot > 100) ? 'red':'orange'; | ||||
| 			gId('wreason').innerHTML = (bquot > 80) ? "80% of max. LED memory" +(bquot>100 ? ` (<b>ERROR: Using over ${maxM}B!</b>)` : "") : "800 LEDs per output"; | ||||
| 			// calculate power | ||||
| 			var val = Math.ceil((100 + sPC * laprev)/500)/2; | ||||
| 			gId('ampwarning').style.display = (parseInt(d.Sf.MA.value,10) > 7200) ? 'inline':'none'; | ||||
| 			var val = Math.ceil((100 + busMA)/500)/2; | ||||
| 			val = (val > 5) ? Math.ceil(val) : val; | ||||
| 			var s = ""; | ||||
| 			var is12V = (d.Sf.LAsel.value == 30); | ||||
| 			var isWS2815 = (d.Sf.LAsel.value == 255); | ||||
| 			if (val < 1.02 && !is12V && !isWS2815) | ||||
| 			{ | ||||
| 				s = "ESP 5V pin with 1A USB supply"; | ||||
| 			} else | ||||
| 			{ | ||||
| 				s += is12V ? "12V ": isWS2815 ? "WS2815 12V " : "5V "; | ||||
| 				s += val; | ||||
| 				s += "A supply connected to LEDs"; | ||||
| 			} | ||||
| 			var val2 = Math.ceil((100 + sPC * laprev)/1500)/2; | ||||
| 			var s = "A power supply with total of "; | ||||
| 			s += val; | ||||
| 			s += "A is required."; | ||||
| 			var val2 = Math.ceil((100 + busMA)/1500)/2; | ||||
| 			val2 = (val2 > 5) ? Math.ceil(val2) : val2; | ||||
| 			var s2 = "(for most effects, ~"; | ||||
| 			s2 += val2; | ||||
| 			s2 += "A is enough)<br>"; | ||||
| 			gId('psu').innerHTML = s; | ||||
| 			gId('psu2').innerHTML = isWS2815 ? "" : s2; | ||||
| 			gId('psu2').innerHTML = s2; | ||||
| 			gId("json").style.display = d.Sf.IT.value==8 ? "" : "none"; | ||||
| 		} | ||||
| 		function lastEnd(i) { | ||||
| @@ -345,6 +382,7 @@ ${i+1}: | ||||
| <option value="24">400kHz</option>\ | ||||
| <option value="25">TM1829</option>\ | ||||
| <option value="26">UCS8903</option>\ | ||||
| <option value="27">APA106/PL9823</option>\ | ||||
| <option value="29">UCS8904 RGBW</option>\ | ||||
| <option value="50">WS2801</option>\ | ||||
| <option value="51">APA102</option>\ | ||||
| @@ -363,7 +401,20 @@ ${i+1}: | ||||
| <!--option value="81">E1.31 RGB (network)</option--> | ||||
| <option value="82">Art-Net RGB (network)</option> | ||||
| <option value="88">DDP RGBW (network)</option> | ||||
| <option value="89">Art-Net RGBW (network)</option> | ||||
| </select><br> | ||||
| <div id="abl${i}"> | ||||
| mA/LED: <select name="LAsel${i}" onchange="enLA(this,${i});UI();"> | ||||
| <option value="55" selected>55mA (typ. 5V WS281x)</option> | ||||
| <option value="35">35mA (eco WS2812)</option> | ||||
| <option value="30">30mA (typ. 12V)</option> | ||||
| <option value="255">12mA (WS2815)</option> | ||||
| <option value="15">15mA (seed/fairy pixels)</option> | ||||
| <option value="0">Custom</option> | ||||
| </select><br> | ||||
| <div id="LAdis${i}" style="display: none;">max. mA/LED: <input name="LA${i}" type="number" min="1" max="254" oninput="UI()"> mA<br></div> | ||||
| <div id="PSU${i}">PSU: <input name="MA${i}" type="number" class="xl" min="250" max="65000" oninput="UI()" value="250"> mA<br></div> | ||||
| </div> | ||||
| <div id="co${i}" style="display:inline">Color Order: | ||||
| <select name="CO${i}"> | ||||
| <option value="0">GRB</option> | ||||
| @@ -398,7 +449,9 @@ ${i+1}: | ||||
| 			gId("+").style.display = (i<maxB+maxV-1) ? "inline":"none"; | ||||
| 			gId("-").style.display = (i>0) ? "inline":"none"; | ||||
|  | ||||
| 			if (!init) UI(); | ||||
| 			if (!init) { | ||||
| 				UI(); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		function addCOM(start=0,len=1,co=0) { | ||||
| @@ -409,7 +462,7 @@ ${i+1}: | ||||
| <hr class="sml"> | ||||
| ${i+1}: Start: <input type="number" name="XS${i}" id="xs${i}" class="l starts" min="0" max="65535" value="${start}" oninput="UI();" required="">  | ||||
| Length: <input type="number" name="XC${i}" id="xc${i}" class="l" min="1" max="65535" value="${len}" required="" oninput="UI()"> | ||||
| <div style="display:inline">Color Order: | ||||
| <div>Color Order: | ||||
| <select id="xo${i}" name="XO${i}"> | ||||
| <option value="0">GRB</option> | ||||
| <option value="1">RGB</option> | ||||
| @@ -418,9 +471,16 @@ Length: <input type="number" name="XC${i}" id="xc${i}" class="l" min="1" max="65 | ||||
| <option value="4">BGR</option> | ||||
| <option value="5">GBR</option> | ||||
| </select> | ||||
| </div><br></div>`; | ||||
| Swap: <select id="xw${i}" name="XW${i}"> | ||||
| <option value="0">Use global</option> | ||||
| <option value="1">W & B</option> | ||||
| <option value="2">W & G</option> | ||||
| <option value="3">W & R</option> | ||||
| </select> | ||||
| </div></div>`; | ||||
| 			gId("com_entries").insertAdjacentHTML("beforeend", b); | ||||
| 			gId("xo"+i).value = co; | ||||
| 			gId("xo"+i).value = co & 0x0F; | ||||
| 			gId("xw"+i).value = co >> 4; | ||||
| 			btnCOM(i+1); | ||||
| 			UI(); | ||||
| 		} | ||||
| @@ -463,6 +523,7 @@ Length: <input type="number" name="XC${i}" id="xc${i}" class="l" min="1" max="65 | ||||
| 			c += `<option value="6" ${t==6?"selected":""}>Touch</option>`; | ||||
| 			c += `<option value="7" ${t==7?"selected":""}>Analog</option>`; | ||||
| 			c += `<option value="8" ${t==8?"selected":""}>Analog inverted</option>`; | ||||
| 			c += `<option value="9" ${t==9?"selected":""}>Touch (switch)</option>`; | ||||
| 			c += `</select>`; | ||||
| 			c += `<span style="cursor: pointer;" onclick="off('${bt}')"> ✕</span><br>`; | ||||
| 			gId("btns").innerHTML = c; | ||||
| @@ -669,9 +730,11 @@ Length: <input type="number" name="XC${i}" id="xc${i}" class="l" min="1" max="65 | ||||
| 				let path = l.pathname; | ||||
| 				let paths = path.slice(1,path.endsWith('/')?-1:undefined).split("/"); | ||||
| 				if (paths.length > 2) { | ||||
| 					paths.pop(); // remove "leds" | ||||
| 					paths.pop(); // remove "settings" | ||||
| 					locproto = l.protocol; | ||||
| 					loc = true; | ||||
| 					locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths[0]; | ||||
| 					locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths.join('/'); | ||||
| 				} | ||||
| 			} | ||||
| 			loadJS(getURL('/settings/s.js?p=2'), false);	// If we set async false, file is loaded and executed, then next statement is processed | ||||
| @@ -695,29 +758,23 @@ Length: <input type="number" name="XC${i}" id="xc${i}" class="l" min="1" max="65 | ||||
| 		<b><span id="psu">?</span></b><br> | ||||
| 		<span id="psu2"><br></span> | ||||
| 		<br> | ||||
| 		Enable automatic brightness limiter: <input type="checkbox" name="ABen" onchange="enABL()" id="able"><br> | ||||
| 		Enable automatic brightness limiter: <input type="checkbox" name="ABL" onchange="enABL()"><br> | ||||
| 		<div id="abl"> | ||||
| 			Maximum Current: <input name="MA" type="number" class="l" min="250" max="65000" oninput="UI()" required> mA<br> | ||||
| 			<i>Automatically limits brightness to stay close to the limit.<br> | ||||
| 				Keep at <1A if poweing LEDs directly from the ESP 5V pin!<br> | ||||
| 				Analog (PWM) and virtual LEDs cannot use automatic brightness limiter.<br></i> | ||||
| 			<div id="psuMA">Maximum PSU Current: <input name="MA" type="number" class="xl" min="250" max="65000" oninput="UI()" required> mA<br></div> | ||||
| 			Use per-output limiter: <input type="checkbox" name="PPL" onchange="UI()"><br> | ||||
| 			<div id="ppldis" style="display:none;"> | ||||
| 				<i>Make sure you enter correct values in each LED output.<br> | ||||
| 				If using multiple outputs with only one PSU, distribute its power proportionally amongst ouputs.</i><br> | ||||
| 			</div> | ||||
| 			<div id="ampwarning" class="warn" style="display: none;"> | ||||
| 				⚠ Your power supply provides high current.<br> | ||||
| 				To improve the safety of your setup,<br> | ||||
| 				please use thick cables,<br> | ||||
| 				multiple power injection points and a fuse!<br> | ||||
| 			</div> | ||||
| 			<i>Automatically limits brightness to stay close to the limit.<br> | ||||
| 			Keep at <1A if powering LEDs directly from the ESP 5V pin!<br> | ||||
| 			If you are using an external power supply, enter its rating.<br> | ||||
| 			(Current estimated usage: <span class="pow">unknown</span>)</i><br><br> | ||||
| 			LED voltage (Max. current for a single LED):<br> | ||||
| 			<select name="LAsel" onchange="enLA()"> | ||||
| 				<option value="55" selected>5V default (55mA)</option> | ||||
| 				<option value="35">5V efficient (35mA)</option> | ||||
| 				<option value="30">12V (30mA)</option> | ||||
| 				<option value="255">WS2815 (12mA)</option> | ||||
| 				<option value="50">Custom</option> | ||||
| 			</select><br> | ||||
| 			<span id="LAdis" style="display: none;">Custom max. current per LED: <input name="LA" type="number" min="0" max="255" id="la" oninput="UI()" required> mA<br></span> | ||||
| 			<i>Keep at default if you are unsure about your type of LEDs.</i><br> | ||||
| 		</div> | ||||
| 		<h3>Hardware setup</h3> | ||||
| 		<div id="mLC">LED outputs:</div> | ||||
| @@ -772,11 +829,14 @@ Length: <input type="number" name="XC${i}" id="xc${i}" class="l" min="1" max="65 | ||||
| 		Use Gamma value: <input name="GV" type="number" class="m" placeholder="2.8" min="1" max="3" step="0.1" required><br><br> | ||||
| 		Brightness factor: <input name="BF" type="number" class="m" min="1" max="255" required> % | ||||
| 		<h3>Transitions</h3> | ||||
| 		Crossfade: <input type="checkbox" name="TF"><br> | ||||
| 		Effect blending: <input type="checkbox" name="EB"><br> | ||||
| 		Transition Time: <input name="TD" type="number" class="xl" min="0" max="65500"> ms<br> | ||||
| 		Enable Palette transitions: <input type="checkbox" name="PF"><br> | ||||
| 		Enable transitions: <input type="checkbox" name="TF" onchange="gId('tran').style.display=this.checked?'inline':'none';"><br> | ||||
| 		<span id="tran"> | ||||
| 			Effect blending: <input type="checkbox" name="EB"><br> | ||||
| 			Transition Time: <input name="TD" type="number" class="xl" min="0" max="65500"> ms<br> | ||||
| 			Palette transitions: <input type="checkbox" name="PF"><br> | ||||
| 		</span> | ||||
| 		<i>Random Cycle</i> Palette Time: <input name="TP" type="number" class="m" min="1" max="255"> s<br> | ||||
| 		Use harmonic <i>Random Cycle</i> Palette: <input type="checkbox" name="TH"><br> | ||||
| 		<h3>Timed light</h3> | ||||
| 		Default Duration: <input name="TL" type="number" class="m" min="1" max="255" required> min<br> | ||||
| 		Default Target brightness: <input name="TB" type="number" class="m" min="0" max="255" required><br> | ||||
|   | ||||
| @@ -77,9 +77,11 @@ | ||||
| 				let path = l.pathname; | ||||
| 				let paths = path.slice(1,path.endsWith('/')?-1:undefined).split("/"); | ||||
| 				if (paths.length > 2) { | ||||
| 					paths.pop(); // remove "sec" | ||||
| 					paths.pop(); // remove "settings" | ||||
| 					locproto = l.protocol; | ||||
| 					loc = true; | ||||
| 					locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths[0]; | ||||
| 					locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths.join('/'); | ||||
| 				} | ||||
| 			} | ||||
| 			if (loc) { | ||||
| @@ -122,20 +124,20 @@ | ||||
| 		Enable ArduinoOTA: <input type="checkbox" name="AO"> | ||||
| 		<hr> | ||||
| 		<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.</div> | ||||
| 		For security reasons, passwords are not backed up. | ||||
| 		<a class="btn lnk" id="bckcfg" href="/presets.json" download="presets">Backup presets</a><br> | ||||
| 		<div>Restore presets<br><input type="file" name="data" accept=".json"> <button type="button" onclick="uploadFile(d.Sf.data,'/presets.json');">Upload</button><br></div><br> | ||||
| 		<a class="btn lnk" id="bckpresets" href="/cfg.json" download="cfg">Backup configuration</a><br> | ||||
| 		<div>Restore configuration<br><input type="file" name="data2" accept=".json"> <button type="button" onclick="uploadFile(d.Sf.data2,'/cfg.json');">Upload</button><br></div> | ||||
| 		<div class="warn">⚠ Restoring presets/configuration will OVERWRITE your current presets/configuration.<br> | ||||
| 		Incorrect configuration may require a factory reset or re-flashing of your ESP.</div> | ||||
| 		For security reasons, passwords are not backed up. | ||||
| 		<hr> | ||||
| 		<h3>About</h3> | ||||
| 		<a href="https://github.com/Aircoookie/WLED/" target="_blank">WLED</a> version ##VERSION##<!-- Autoreplaced from package.json --><br><br> | ||||
| 		<a href="https://github.com/Aircoookie/WLED/" target="_blank">WLED</a> version ##VERSION##<!-- Autoreplaced from package.json --><br><br> | ||||
| 		<a href="https://github.com/Aircoookie/WLED/wiki/Contributors-and-credits" target="_blank">Contributors, dependencies and special thanks</a><br> | ||||
| 		A huge thank you to everyone who helped me create WLED!<br><br> | ||||
| 		(c) 2016-2023 Christian Schwinne <br> | ||||
| 		<i>Licensed under the <a href="https://github.com/Aircoookie/WLED/blob/master/LICENSE" target="_blank">MIT license</a></i><br><br> | ||||
| 		(c) 2016-2024 Christian Schwinne <br> | ||||
| 		<i>Licensed under the <a href="https://github.com/Aircoookie/WLED/blob/master/LICENSE" target="_blank">MIT license</a></i><br><br> | ||||
| 		Server message: <span class="sip"> Response error! </span><hr> | ||||
| 		<div id="toast"></div> | ||||
| 		<button type="button" onclick="B()">Back</button><button type="submit">Save</button> | ||||
|   | ||||
							
								
								
									
										25
									
								
								wled00/data/settings_sync.htm
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
									
										25
									
								
								wled00/data/settings_sync.htm
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							| @@ -19,7 +19,7 @@ | ||||
| 		scE.setAttribute("type", "text/javascript"); | ||||
| 		scE.setAttribute("async", async); | ||||
| 		d.body.appendChild(scE); | ||||
| 		// success event  | ||||
| 		// success event | ||||
| 		scE.addEventListener("load", () => { | ||||
| 			//console.log("File loaded"); | ||||
| 			GetV();SetVal(); | ||||
| @@ -67,9 +67,11 @@ | ||||
| 			// detect reverse proxy | ||||
| 			let paths = l.pathname.slice(1,l.pathname.endsWith('/')?-1:undefined).split("/"); | ||||
| 			if (paths.length > 2) { | ||||
| 				paths.pop(); // remove "sync" | ||||
| 				paths.pop(); // remove "settings" | ||||
| 				locproto = l.protocol; | ||||
| 				loc = true; | ||||
| 				locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths[0]; | ||||
| 				locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths.join('/'); | ||||
| 			} | ||||
| 		} | ||||
| 		loadJS(getURL('/settings/s.js?p=4'), false);	// If we set async false, file is loaded and executed, then next statement is processed | ||||
| @@ -91,6 +93,12 @@ | ||||
| <h3>WLED Broadcast</h3> | ||||
| UDP Port: <input name="UP" type="number" min="1" max="65535" class="d5" required><br> | ||||
| 2nd Port: <input name="U2" type="number" min="1" max="65535" class="d5" required><br> | ||||
| <div id="NoESPNOW" class="hide"> | ||||
| <i class="warn">ESP-NOW support is disabled.<br></i> | ||||
| </div> | ||||
| <div id="ESPNOW"> | ||||
| Use ESP-NOW sync: <input type="checkbox" name="EN"><br><i>(in AP mode or no WiFi)</i><br> | ||||
| </div> | ||||
| <h3>Sync groups</h3> | ||||
| <input name="GS" id="GS" type="number" style="display: none;"><!-- hidden inputs for bitwise group checkboxes --> | ||||
| <input name="GR" id="GR" type="number" style="display: none;"> | ||||
| @@ -128,14 +136,16 @@ UDP Port: <input name="UP" type="number" min="1" max="65535" class="d5" required | ||||
| 		<td><input type="checkbox" id="R7" name="R7"></td> | ||||
| 		<td><input type="checkbox" id="R8" name="R8"></td> | ||||
| 	</tr> | ||||
| </table><br> | ||||
| Receive: <nowrap><input type="checkbox" name="RB">Brightness,</nowrap> <nowrap><input type="checkbox" name="RC">Color,</nowrap> <nowrap>and <input type="checkbox" name="RX">Effects</nowrap><br> | ||||
| <input type="checkbox" name="SO"> Segment options, <input type="checkbox" name="SG"> bounds<br> | ||||
| </table> | ||||
| <h3>Receive</h3> | ||||
| <nowrap><input type="checkbox" name="RB">Brightness,</nowrap> <nowrap><input type="checkbox" name="RC">Color,</nowrap> <nowrap>and <input type="checkbox" name="RX">Effects</nowrap><br> | ||||
| <input type="checkbox" name="SO"> Segment options, <input type="checkbox" name="SG"> bounds | ||||
| <h3>Send</h3> | ||||
| Enable Sync on start: <input type="checkbox" name="SS"><br> | ||||
| Send notifications on direct change: <input type="checkbox" name="SD"><br> | ||||
| Send notifications on button press or IR: <input type="checkbox" name="SB"><br> | ||||
| Send Alexa notifications: <input type="checkbox" name="SA"><br> | ||||
| Send Philips Hue change notifications: <input type="checkbox" name="SH"><br> | ||||
| Send Macro notifications: <input type="checkbox" name="SM"><br> | ||||
| UDP packet retransmissions: <input name="UR" type="number" min="0" max="30" class="d5" required><br><br> | ||||
| <i>Reboot required to apply changes. </i> | ||||
| <hr class="sml"> | ||||
| @@ -145,7 +155,8 @@ Make this instance discoverable: <input type="checkbox" name="NB"> | ||||
| <hr class="sml"> | ||||
| <h3>Realtime</h3> | ||||
| Receive UDP realtime: <input type="checkbox" name="RD"><br> | ||||
| Use main segment only: <input type="checkbox" name="MO"><br><br> | ||||
| Use main segment only: <input type="checkbox" name="MO"><br> | ||||
| Respect LED Maps: <input type="checkbox" name="RLM"><br><br> | ||||
| <i>Network DMX input</i><br> | ||||
| Type: | ||||
| <select name=DI onchange="SP(); adj();"> | ||||
|   | ||||
| @@ -45,9 +45,11 @@ | ||||
| 			let path = l.pathname; | ||||
| 			let paths = path.slice(1,path.endsWith('/')?-1:undefined).split("/"); | ||||
| 			if (paths.length > 2) { | ||||
| 				paths.pop(); // remove "time" | ||||
| 				paths.pop(); // remove "settings" | ||||
| 				locproto = l.protocol; | ||||
| 				loc = true; | ||||
| 				locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths[0]; | ||||
| 				locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths.join('/'); | ||||
| 			} | ||||
| 		} | ||||
| 		loadJS(getURL('/settings/s.js?p=5'), false);	// If we set async false, file is loaded and executed, then next statement is processed | ||||
| @@ -210,6 +212,7 @@ | ||||
| 			12h LED: <input name="OM" type="number" min="0" max="255" required><br> | ||||
| 			Show 5min marks: <input type="checkbox" name="O5"><br> | ||||
| 			Seconds (as trail): <input type="checkbox" name="OS"><br> | ||||
| 			Show clock overlay only if all LEDs are solid black: <input type="checkbox" name="OB"><br> | ||||
| 		</div> | ||||
| 		Countdown Mode: <input type="checkbox" name="CE"><br> | ||||
| 		Countdown Goal:<br> | ||||
|   | ||||
| @@ -7,7 +7,7 @@ | ||||
| 	<script> | ||||
| 	var d = document; | ||||
| 	var loc = false, locip, locproto = "http:"; | ||||
| 	var initial_ds, initial_st, initial_su; | ||||
| 	var initial_ds, initial_st, initial_su, oldUrl; | ||||
| 	var sett = null; | ||||
| 	var l = { | ||||
| 		"comp":{ | ||||
| @@ -27,6 +27,8 @@ | ||||
| 			"css": "Enable custom CSS", | ||||
| 			"hdays": "Enable custom Holidays list", | ||||
| 			"fxdef": "Use effect default parameters", | ||||
| 			"on": "Power button preset override for On", | ||||
| 			"off": "Power button preset override for Off", | ||||
| 			"idsort": "Sort presets by ID" | ||||
| 		}, | ||||
| 		"theme":{ | ||||
| @@ -36,7 +38,9 @@ | ||||
| 			}, | ||||
| 			"bg":{ | ||||
| 				"url":"BG image URL", | ||||
| 				"random":"Random BG image" | ||||
| 				"rnd":"Random BG image", | ||||
| 				"rndGrayscale":"Grayscale", | ||||
| 				"rndBlur":"Blur" | ||||
| 			}, | ||||
| 			"color":{ | ||||
| 				"bg":"BG HEX color" | ||||
| @@ -72,7 +76,7 @@ | ||||
| 	function addRec(s, path = "", label = null) | ||||
| 	{ | ||||
| 		var str = ""; | ||||
| 		for (i in s) | ||||
| 		for (let i in s) | ||||
| 		{ | ||||
| 			var fk = path + (path?'_':'') + i; | ||||
| 			if (isObject(s[i])) { | ||||
| @@ -114,8 +118,14 @@ | ||||
| 	function genForm(s) { | ||||
| 		var str = ""; | ||||
| 		str = addRec(s,"",l); | ||||
| 		oldUrl = ""; | ||||
| 		 | ||||
| 		gId('gen').innerHTML = str; | ||||
| 		if (gId('theme_bg_rnd').checked) { | ||||
| 			toggle("Image"); | ||||
| 		} else if (gId('theme_bg_url').value.startsWith('data:')) { | ||||
| 			gId("bg_url").classList.add("hide"); | ||||
| 		} else oldUrl = gId("theme_bg_url").value; | ||||
| 	} | ||||
| 	function GetLS() | ||||
| 	{ | ||||
| @@ -161,7 +171,7 @@ | ||||
| 	 | ||||
| 	function Save() { | ||||
| 		SetLS(); | ||||
| 		if (d.Sf.DS.value != initial_ds || d.Sf.ST.checked != initial_st || d.Sf.SU.checked != initial_su) d.Sf.submit(); | ||||
| 		if (d.Sf.DS.value != initial_ds || /*d.Sf.ST.checked != initial_st ||*/ d.Sf.SU.checked != initial_su) d.Sf.submit(); | ||||
| 	} | ||||
| 	 | ||||
| 	// https://www.educative.io/edpresso/how-to-dynamically-load-a-js-file-in-javascript | ||||
| @@ -176,7 +186,7 @@ | ||||
| 			//console.log("File loaded"); | ||||
| 			GetV();  | ||||
| 			initial_ds = d.Sf.DS.value; | ||||
| 			initial_st = d.Sf.ST.checked; | ||||
| 			//initial_st = d.Sf.ST.checked; | ||||
| 			initial_su = d.Sf.SU.checked; | ||||
| 			GetLS(); | ||||
| 		}); | ||||
| @@ -200,9 +210,11 @@ | ||||
| 			let path = l.pathname; | ||||
| 			let paths = path.slice(1,path.endsWith('/')?-1:undefined).split("/"); | ||||
| 			if (paths.length > 2) { | ||||
| 				paths.pop(); // remove "ui" | ||||
| 				paths.pop(); // remove "settings" | ||||
| 				locproto = l.protocol; | ||||
| 				loc = true; | ||||
| 				locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths[0]; | ||||
| 				locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths.join('/'); | ||||
| 			} | ||||
| 		} | ||||
| 		loadJS(getURL('/settings/s.js?p=3'), false);	// If we set async false, file is loaded and executed, then next statement is processed | ||||
| @@ -221,21 +233,38 @@ | ||||
| 	} | ||||
|  | ||||
| 	// random BG image | ||||
| 	function setRandomBg() { | ||||
| 		if (gId("theme_bg_random").checked) { | ||||
| 			gId("theme_bg_url").value = "https://picsum.photos/1920/1080"; | ||||
| 		} else { | ||||
| 			gId("theme_bg_url").value = ""; | ||||
| 	function randomBg() { | ||||
| 		let url = oldUrl; | ||||
| 		let t = "theme_bg_rnd"; | ||||
| 		if (gId(t).checked) { | ||||
| 			url = "https://picsum.photos/1920/1080"; | ||||
| 			if (gId(`${t}Grayscale`).checked) url += "?grayscale"; | ||||
| 			if (gId(`${t}Blur`).checked) url += (url.includes("?") ? "&" : "?") + "blur"; | ||||
| 			gId("theme_bg_img").value = ""; | ||||
| 			gId("bg_url").classList.remove("hide"); | ||||
| 		} | ||||
| 		 | ||||
| 		gId("theme_bg_url").value = url; | ||||
| 	} | ||||
| 	function checkRandomBg() { | ||||
| 		if (gId("theme_bg_url").value === "https://picsum.photos/1920/1080") { | ||||
| 			gId("theme_bg_random").checked = true; | ||||
| 		} else { | ||||
| 			gId("theme_bg_random").checked = false; | ||||
| 	// own BG image | ||||
| 	function ownBg(element) { | ||||
| 		const file = element.files[0]; | ||||
| 		const reader = new FileReader(); | ||||
| 		reader.onload = () => { | ||||
| 			gId("theme_bg_url").value = reader.result; | ||||
| 			gId("bg_url").classList.add("hide"); | ||||
| 			if (gId("theme_bg_rnd").checked) toggle("Image"); | ||||
| 			gId("theme_bg_rnd").checked = false; | ||||
| 		} | ||||
| 		reader.readAsDataURL(file); | ||||
| 	} | ||||
| 	function removeBgImg() { | ||||
| 		gId("theme_bg_url").value = ""; | ||||
| 		gId("theme_bg_img").value = ""; | ||||
| 		gId("bg_url").classList.remove("hide"); | ||||
| 		if (gId("theme_bg_rnd").checked) toggle("Image"); | ||||
| 		gId("theme_bg_rnd").checked = false; | ||||
| 	} | ||||
|  | ||||
| 	function uploadFile(fO,name) { | ||||
| 		var req = new XMLHttpRequest(); | ||||
| 		req.addEventListener('load', function(){showToast(this.responseText,this.status >= 400)}); | ||||
| @@ -260,11 +289,8 @@ | ||||
| 		</div> | ||||
| 		<h2>Web Setup</h2> | ||||
| 		Server description: <input type="text" name="DS" maxlength="32"><br> | ||||
| 		Sync button toggles both send and receive: <input type="checkbox" name="ST"><br> | ||||
| 		<div id="NoSimple" class="hide"> | ||||
| 			<i class="warn">This firmware build does not include simplified UI support.<br></i> | ||||
| 		</div> | ||||
| 		<div id="Simple">Enable simplified UI: <input type="checkbox" name="SU"><br></div> | ||||
| 		<!-- Sync button toggles both send and receive: <input type="checkbox" name="ST"><br> --> | ||||
| 		Enable simplified UI: <input type="checkbox" name="SU"><br> | ||||
| 		<i>The following UI customization settings are unique both to the WLED device and this browser.<br> | ||||
| 		You will need to set them again if using a different browser, device or WLED IP address.<br> | ||||
| 		Refresh the main UI to apply changes.</i><br> | ||||
| @@ -279,14 +305,26 @@ | ||||
| 		<span class="l"></span>: <input type="checkbox" id="comp_segpwr" class="agi cb"><br> | ||||
| 		<span class="l"></span>: <input type="checkbox" id="comp_segexp" class="agi cb"><br> | ||||
| 		<span class="l"></span>: <input type="checkbox" id="comp_fxdef" class="agi cb"><br> | ||||
| 		<span class="l"></span>: <input type="number" min=0 max=250 step=1 id="comp_on" class="agi"><br> | ||||
| 		<span class="l"></span>: <input type="number" min=0 max=250 step=1 id="comp_off" class="agi"><br> | ||||
| 		<span class="l"></span>: <input type="checkbox" id="comp_idsort" class="agi cb"><br> | ||||
| 		I hate dark mode: <input type="checkbox" id="dm" onchange="UI()"><br> | ||||
| 		<span id="idonthateyou" style="display:none"><i>Why would you? </i>🥺<br></span> | ||||
| 		<span class="l"></span>: <input type="number" min=0.0 max=1.0 step=0.01 id="theme_alpha_tab" class="agi"><br> | ||||
| 		<span class="l"></span>: <input type="number" min=0.0 max=1.0 step=0.01 id="theme_alpha_bg" class="agi"><br> | ||||
| 		<span class="l"></span>: <input type="text" id="theme_color_bg" maxlength="9" class="agi"><br> | ||||
| 		<span class="l">BG image URL</span>: <input type="text" id="theme_bg_url" class="agi" oninput="checkRandomBg()"><br> | ||||
| 		<span class="l">Random BG image</span>: <input type="checkbox" id="theme_bg_random" class="agi cb" onchange="setRandomBg()"><br> | ||||
| 		BG image: <input type="file" id="theme_bg_img" accept="image/*" onchange="ownBg(this)"> <input type="button" value="Remove" onclick="removeBgImg()"><br> | ||||
| 		<span class="l"></span>: <input type="checkbox" id="theme_bg_rnd" class="agi cb" onchange="randomBg();toggle('Image');"> | ||||
| 		<div id="Image"> | ||||
| 			<div id="bg_url"> | ||||
| 				<span class="l"></span>: <input type="text" id="theme_bg_url" class="agi"><br> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div id="NoImage" class="hide"> | ||||
| 			<h4>Random BG image settings</h4> | ||||
| 			<span class="l"></span>: <input type="checkbox" id="theme_bg_rndGrayscale" class="agi cb" onchange="randomBg()"><br> | ||||
| 			<span class="l"></span>: <input type="checkbox" id="theme_bg_rndBlur" class="agi cb" onchange="randomBg()"><br> | ||||
| 		</div> | ||||
| 		<input id="theme_base" class="agi" style="display:none"> | ||||
| 		<span class="l"></span>: <input type="checkbox" id="comp_css" class="agi cb"><br> | ||||
| 		<div id="skin">Custom CSS: <input type="file" name="data" accept=".css"> <input type="button" value="Upload" onclick="uploadFile(d.Sf.data,'/skin.css');"><br></div> | ||||
|   | ||||
| @@ -56,9 +56,11 @@ | ||||
| 			let path = l.pathname; | ||||
| 			let paths = path.slice(1,path.endsWith('/')?-1:undefined).split("/"); | ||||
| 			if (paths.length > 2) { | ||||
| 				paths.pop(); // remove "um" | ||||
| 				paths.pop(); // remove "settings" | ||||
| 				locproto = l.protocol; | ||||
| 				loc = true; | ||||
| 				locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths[0]; | ||||
| 				locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths.join('/'); | ||||
| 			} | ||||
| 		} | ||||
| 		ldS(); | ||||
| @@ -225,10 +227,10 @@ | ||||
| 		} else if (typeof(fld) === "number") sel.classList.add("pin"); // a hack to add a class | ||||
| 		let arr = d.getElementsByName(um); | ||||
| 		let idx = arr[0].type==="hidden"?1:0; // ignore hidden field | ||||
| 		if (arr.length > 2) { | ||||
| 		if (arr.length > 1+idx) { | ||||
| 			// we have array of values (usually pins) | ||||
| 			for (let i of arr) { | ||||
| 				if (i.type === "number") break; | ||||
| 				if (i.nodeName === "INPUT" && i.type === "number") break; | ||||
| 				idx++; | ||||
| 			} | ||||
| 		} | ||||
|   | ||||
| @@ -8,9 +8,10 @@ | ||||
| 		var d = document; | ||||
| 		var loc = false, locip, locproto = "http:"; | ||||
| 		var scanLoops = 0, preScanSSID = ""; | ||||
|  | ||||
| 		var maxNetworks = 3; | ||||
| 		function gId(e) { return d.getElementById(e); } | ||||
| 		function cE(e) { return d.createElement(e); } | ||||
| 		function toggle(el){gId(el).classList.toggle("hide"); gId('No'+el).classList.toggle("hide");} | ||||
| 		function H(){window.open("https://kno.wled.ge/features/settings/#wifi-settings");} | ||||
| 		function B(){window.open(getURL("/settings"),"_self");} | ||||
| 		function N() { | ||||
| @@ -51,13 +52,14 @@ | ||||
| 				} | ||||
| 				scanLoops = 0; | ||||
|  | ||||
| 				let cs = gId("CS"); | ||||
| 				if (cs) { | ||||
| 				let cs = d.querySelectorAll("#wifi_entries input[type=text]"); | ||||
| 				for (let input of (cs||[])) { | ||||
| 					let found = false; | ||||
| 					let select = cE("select"); | ||||
| 					select.setAttribute("id", "CS"); | ||||
| 					select.setAttribute("name", "CS"); | ||||
| 					select.setAttribute("onchange", "T()"); | ||||
| 					preScanSSID = cs.value; | ||||
| 					select.id = input.id; | ||||
| 					select.name = input.name; | ||||
| 					select.setAttribute("onchange", "T(this)"); | ||||
| 					preScanSSID = input.value; | ||||
|  | ||||
| 					for (let i = 0; i < select.children.length; i++) { | ||||
| 						select.removeChild(select.children[i]); | ||||
| @@ -69,8 +71,9 @@ | ||||
| 						option.setAttribute("value", networks[i].ssid); | ||||
| 						option.textContent = `${networks[i].ssid} (${networks[i].rssi} dBm)`; | ||||
|  | ||||
| 						if (networks[i].ssid === cs.value) { | ||||
| 						if (networks[i].ssid === input.value) { | ||||
| 							option.setAttribute("selected", "selected"); | ||||
| 							found = true; | ||||
| 						} | ||||
|  | ||||
| 						select.appendChild(option); | ||||
| @@ -78,10 +81,11 @@ | ||||
| 					const option = cE("option"); | ||||
|  | ||||
| 					option.setAttribute("value", "!Cs"); | ||||
| 					option.textContent = `Other network...`; | ||||
| 					option.textContent = "Other network..."; | ||||
| 					select.appendChild(option); | ||||
|  | ||||
| 					cs.replaceWith(select); | ||||
| 					if (input.value === "" || found) input.replaceWith(select); | ||||
| 					else select.remove();  | ||||
| 				} | ||||
|  | ||||
| 				button.disabled = false; | ||||
| @@ -89,17 +93,48 @@ | ||||
| 			}); | ||||
| 		} | ||||
| 		// replace WiFi select with custom SSID input field again | ||||
| 		function T() { | ||||
| 			let cs = gId("CS"); | ||||
| 		function T(cs) { | ||||
| 			if (!cs || cs.value != "!Cs") return; | ||||
| 			let input = cE("input"); | ||||
| 			input.type = "text"; | ||||
| 			input.id = "CS"; | ||||
| 			input.name ="CS"; | ||||
| 			input.id = cs.id; | ||||
| 			input.name = cs.name; | ||||
| 			input.setAttribute("maxlength",32); | ||||
| 			input.value = preScanSSID; | ||||
| 			cs.replaceWith(input); | ||||
| 		} | ||||
| 		function resetWiFi(maxN = undefined) { | ||||
| 			if (maxN) maxNetworks = maxN; | ||||
| 			let entries = gId("wifi_entries").children | ||||
| 			for (let i = entries.length; i > 0; i--) entries[i-1].remove(); | ||||
| 			btnWiFi(0); | ||||
| 		} | ||||
| 		function btnWiFi(i) { | ||||
| 			gId("wifi_add").style.display = (i<maxNetworks) ? "inline":"none"; | ||||
| 			gId("wifi_rem").style.display = (i>1) ? "inline":"none"; | ||||
| 		} | ||||
| 		function addWiFi(ssid="",pass="",ip=0,gw=0,sn=0x00ffffff) { // little endian | ||||
| 			var i = gId("wifi_entries").childNodes.length; | ||||
| 			if (i >= maxNetworks) return; | ||||
| 			var b = `<div id="net${i}"><hr class="sml"> | ||||
| Network name (SSID${i==0?", empty to not connect":""}):<br><input type="text" id="CS${i}" name="CS${i}" maxlength="32" value="${ssid}" ${i>0?"required":""}><br> | ||||
| Network password:<br><input type="password" name="PW${i}" maxlength="64" value="${pass}"><br> | ||||
| Static IP (leave at 0.0.0.0 for DHCP)${i==0?"<br>Also used by Ethernet":""}:<br> | ||||
| <input name="IP${i}0" type="number" class="s" min="0" max="255" value="${ip&0xFF}" required>.<input name="IP${i}1" type="number" class="s" min="0" max="255" value="${(ip>>8)&0xFF}" required>.<input name="IP${i}2" type="number" class="s" min="0" max="255" value="${(ip>>16)&0xFF}" required>.<input name="IP${i}3" type="number" class="s" min="0" max="255" value="${(ip>>24)&0xFF}" required><br> | ||||
| Static gateway:<br> | ||||
| <input name="GW${i}0" type="number" class="s" min="0" max="255" value="${gw&0xFF}" required>.<input name="GW${i}1" type="number" class="s" min="0" max="255" value="${(gw>>8)&0xFF}" required>.<input name="GW${i}2" type="number" class="s" min="0" max="255" value="${(gw>>16)&0xFF}" required>.<input name="GW${i}3" type="number" class="s" min="0" max="255" value="${(gw>>24)&0xFF}" required><br> | ||||
| Static subnet mask:<br> | ||||
| <input name="SN${i}0" type="number" class="s" min="0" max="255" value="${sn&0xFF}" required>.<input name="SN${i}1" type="number" class="s" min="0" max="255" value="${(sn>>8)&0xFF}" required>.<input name="SN${i}2" type="number" class="s" min="0" max="255" value="${(sn>>16)&0xFF}" required>.<input name="SN${i}3" type="number" class="s" min="0" max="255" value="${(sn>>24)&0xFF}" required></div>`; | ||||
| 			gId("wifi_entries").insertAdjacentHTML("beforeend", b); | ||||
| 			btnWiFi(i+1); | ||||
| 		} | ||||
| 		function remWiFi() { | ||||
| 			const entries = gId("wifi_entries").children; | ||||
| 			const i = entries.length; | ||||
| 			if (i < 2) return; | ||||
| 			entries[i-1].remove(); | ||||
| 			btnWiFi(i-1); | ||||
| 		} | ||||
| 		// https://www.educative.io/edpresso/how-to-dynamically-load-a-js-file-in-javascript | ||||
| 		function loadJS(FILE_URL, async = true) { | ||||
| 			let scE = cE("script"); | ||||
| @@ -132,9 +167,11 @@ | ||||
| 				let path = l.pathname; | ||||
| 				let paths = path.slice(1,path.endsWith('/')?-1:undefined).split("/"); | ||||
| 				if (paths.length > 2) { | ||||
| 					paths.pop(); // remove "wifi" | ||||
| 					paths.pop(); // remove "settings" | ||||
| 					locproto = l.protocol; | ||||
| 					loc = true; | ||||
| 					locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths[0]; | ||||
| 					locip = l.hostname + (l.port ? ":" + l.port : "") + "/" + paths.join('/'); | ||||
| 				} | ||||
| 			} | ||||
| 			loadJS(getURL('/settings/s.js?p=1'), false);	// If we set async false, file is loaded and executed, then next statement is processed | ||||
| @@ -155,24 +192,16 @@ | ||||
| 		<h2>WiFi setup</h2> | ||||
| 		<h3>Connect to existing network</h3> | ||||
| 		<button type="button" id="scan" onclick="N()">Scan</button><br> | ||||
| 		Network name (SSID, empty to not connect):<br> | ||||
| 		<input type="text" id="CS" name="CS" maxlength="32"><br> | ||||
| 		Network password: <br> <input type="password" name="CP" maxlength="63"><br> | ||||
| 		Static IP (leave at 0.0.0.0 for DHCP):<br> | ||||
| 		<input name="I0" type="number" class="s" min="0" max="255" required> . | ||||
| 		<input name="I1" type="number" class="s" min="0" max="255" required> . | ||||
| 		<input name="I2" type="number" class="s" min="0" max="255" required> . | ||||
| 		<input name="I3" type="number" class="s" min="0" max="255" required><br> | ||||
| 		Static gateway:<br> | ||||
| 		<input name="G0" type="number" class="s" min="0" max="255" required> . | ||||
| 		<input name="G1" type="number" class="s" min="0" max="255" required> . | ||||
| 		<input name="G2" type="number" class="s" min="0" max="255" required> . | ||||
| 		<input name="G3" type="number" class="s" min="0" max="255" required><br> | ||||
| 		Static subnet mask:<br> | ||||
| 		<input name="S0" type="number" class="s" min="0" max="255" required> . | ||||
| 		<input name="S1" type="number" class="s" min="0" max="255" required> . | ||||
| 		<input name="S2" type="number" class="s" min="0" max="255" required> . | ||||
| 		<input name="S3" type="number" class="s" min="0" max="255" required><br> | ||||
| 		<div id="wifi"> | ||||
| 			Wireless networks | ||||
| 			<div id="wifi_entries"></div> | ||||
| 			<hr class="sml"> | ||||
| 			<button type="button" id="wifi_add" onclick="addWiFi()">+</button> | ||||
| 			<button type="button" id="wifi_rem" onclick="remWiFi()">-</button><br> | ||||
| 		</div> | ||||
| 		DNS server address:<br> | ||||
| 		<input name="D0" type="number" class="s" min="0" max="255" required>.<input name="D1" type="number" class="s" min="0" max="255" required>.<input name="D2" type="number" class="s" min="0" max="255" required>.<input name="D3" type="number" class="s" min="0" max="255" required><br> | ||||
| 		<br> | ||||
| 		mDNS address (leave empty for no mDNS):<br> | ||||
| 		http:// <input type="text" name="CM" maxlength="32"> .local<br> | ||||
| 		Client IP: <span class="sip"> Not connected </span> <br> | ||||
| @@ -183,10 +212,12 @@ | ||||
| 		Access Point WiFi channel: <input name="AC" type="number" class="xs" min="1" max="13" required><br> | ||||
| 		AP opens: | ||||
| 		<select name="AB"> | ||||
| 		<option value="0">No connection after boot</option> | ||||
| 		<option value="1">Disconnected</option> | ||||
| 		<option value="2">Always</option> | ||||
| 		<option value="3">Never (not recommended)</option></select><br> | ||||
| 			<option value="0">No connection after boot</option> | ||||
| 			<option value="1">Disconnected</option> | ||||
| 			<option value="2">Always</option> | ||||
| 			<option value="3">Never (not recommended)</option> | ||||
| 			<option value="4">Temporary (no connection after boot)</option> | ||||
| 		</select><br> | ||||
| 		AP IP: <span class="sip"> Not active </span><br> | ||||
| 		<h3>Experimental</h3> | ||||
| 		Force 802.11g mode (ESP8266 only): <input type="checkbox" name="FG"><br> | ||||
| @@ -194,14 +225,16 @@ | ||||
| 		<i>Can help with connectivity issues.<br> | ||||
| 		Do not enable if WiFi is working correctly, increases power consumption.</i> | ||||
|  | ||||
| 		<div id="remd"> | ||||
| 			<h3>Wireless Remote</h3> | ||||
| 		<h3>ESP-NOW Wireless</h3> | ||||
| 		<div id="NoESPNOW" class="hide"> | ||||
| 			<i class="warn">This firmware build does not include ESP-NOW support.<br></i> | ||||
| 		</div> | ||||
| 		<div id="ESPNOW"> | ||||
| 			Enable ESP-NOW: <input type="checkbox" name="RE"><br> | ||||
| 			<i>Listen for events over ESP-NOW<br> | ||||
| 			Keep disabled if not using a remote, increases power consumption.<br></i> | ||||
| 		 | ||||
| 			Enable Remote: <input type="checkbox" name="RE"><br> | ||||
| 			Hardware MAC: <input type="text" name="RMAC"><br> | ||||
| 			Last Seen: <span class="rlid">None</span> <br> | ||||
| 			Keep disabled if not using a remote or wireless sync, increases power consumption.<br></i> | ||||
| 			Paired Remote MAC: <input type="text" name="RMAC" minlength="12" maxlength="12"><br> | ||||
| 			Last device seen: <span class="rlid" onclick="d.Sf.RMAC.value=this.textContent;" style="cursor:pointer;">None</span> <br> | ||||
| 		</div> | ||||
|  | ||||
| 		<div id="ethd"> | ||||
| @@ -210,7 +243,8 @@ | ||||
| 				<option value="0">None</option> | ||||
| 				<option value="9">ABC! WLED V43 & compatible</option> | ||||
| 				<option value="2">ESP32-POE</option> | ||||
| 				<option value="6">ESP32Deux</option> | ||||
| 				<option value="11">ESP32-POE-WROVER</option> | ||||
| 				<option value="6">ESP32Deux/RGB2Go Tetra</option> | ||||
| 				<option value="7">KIT-VE</option> | ||||
| 				<option value="8">QuinLED-Dig-Octa & T-ETH-POE</option> | ||||
| 				<option value="4">QuinLED-ESP32</option> | ||||
|   | ||||
| @@ -1,933 +0,0 @@ | ||||
| @font-face { | ||||
| 	font-family: "WIcons"; | ||||
| 	src: url(data:font/woff2;charset=utf-8;base64,d09GMgABAAAAAAnUAAsAAAAAE1AAAAmFAAGZmgAAAAAAAAAAAAAAAAAAAAAAAAAABmAAgXwRCAqcYJZIATYCJANwCzoABCAFgwYHIBs7D8iOwzgm3MXMnzZCktnjcbN+QlJLaJ3ulULplpW6UqWioeS91Jye0jUlJwZr5nTdE3LntdPvAg+ft/fbsLsGlNLuhlmQjKi7NPDEIgwTmP//a6mdl+SHUBhEIdHFxak7s4E/yzhJSjC7BQQLfDwopF/i6aqSElEFDXx8ZVWjy3rym4N6FlZQ4hu+nXsGIDMQF3gAxa14AgArtVMhfkgjfEAbiChwuSIwEUCmudPhiQdT6rvIjLSRZEwDhF9BIsooI53TIRIoIUD8kyNZI7UjAyMrR/aM/DwaOpozah9LGCsY2zN2YOzs2L3xqeNp4zXjq8bXT/hMBLj/53YDAIS+7u668n3H+HRPdZd1u3TzdRZdVMTfIl5HfKgd1b7Svqd9W9uprdP8QTOmeaz5TPORJlDDjHVjG0ANMQYsmRrKlmpyqV7kubIQC2GSIkFS+MneCJ48JJFVChQfuwKMp2yU9pmq1VKUR6ret0Gp0SjVYRRF+Xj7+OiUSk/GIzu1miHZWx+g8Y1RUktPmqIitRTXVNzzCtuFPKcH0zRBG+Y9/CnhBa20v5oHfsEUMgXMPEfO5ZcJx0FIPiVywgjb6MIuV+oZ4v2kk6/znIxDKrguM22y+bW8wUGqi7aL8fQJzwnCj8tIppdI9bYDSVJVCQInipW0HbtclcT7vCyLmXaSVrQSNMybaJJBh2PiXrXbgd6AbqecdDTO9EQEIeW0VPWQcdQ8ltPOEu+76q2IxUToJeWpfjQiHHH5AsADLj1bHgQxXsUoHfKYbg+CxCxC69eHcOvWheJ1l6b0nD7jG+bSA1dCZVxmw8ZJ/IYtxPtbJxlpQ/LGjSq00TmdNIZxrGel+y+rZJro+nUh3PrNIGwK6WrXNMV2xTeRWHSjScktLJfe1rc7spyvk3b6V4k48Sr3Am1Pv/QifhsI2uMvc863OiQQRNoedpPfHnSwcete+aDEE67cKzTgBlQgjpjgTDnJtGnX2qbmXJ6FOBLZ7wsr+JZzYnbjdbkCuEfU0HvlwqbtUgJ7zRXFNJsvSxlwz2WYta4xjri/fsulnnFVPyonpP0RL5oVNKkkfElG4csTDNAsgzC38G7gSKVgSZ7m/cEvKALmxKz//u7h6egHF7MrH4jJp/Zx4q32a8T71xnHVRCGlfFZNttd2FcUaay6e9PkhucyR0oPu1z1z/DB+8wixAFdMU1gnmB4xAw68pwHcWjlFrBnXxLjj63UGgvNGVGAJFzxFw+Womn7MAibVbu6leHRB5sc10fLtbrdr/JqV6Yr+ovwFtRHE7M4zG90qNB6YREoo51kFJabq3NeHVKdef/hsMFFSpt5m8XmJqDDAnR0c418mxmxrQzQuyPnspRwfAYkpthzr7gST1xNSf4WtBMM9DQT19uL+gb47gFLP3cT08F8I4dZxJl41Gsx9WHzLBOHzWjRS9NLCOUBCFQ+uGhB/V7ZzUwKESTmDriJ+UecdD/bFXFMLLsjgiAt4pp7ulpxb2tzE8I8xhyHODBK3SGg6QP12BiP3YMw2rDFtWUDXL+esnv3H9QxqfmbDnbMLjGUFpqqZbnWSg0lhWv9wU35qTHqP9zqUrL7kqKj8YjZzg01pb9+yQ8sXZpYxKGiFJTNsIwwpyR44gEOnV/+ennFdHD/2lQ3uS5y1qzIztXUNPE6odYJ0PqUiWJtgKGKMILY60dxeYynbb+sFKKqNn0Wz2rLtMbBQWPnYtmJa4WqFRob/9mmuycQVv7ifCNvXrlhzgDLDvAGA+8H5xjK948cDet+FaXfS+Lko/Wt+vScqarq6kZTbk4NaKqpObkEEpsac9L1rRNXJgPbrWyDdYje6tBQAztkbYC0wDe4UnNipmnZtInu/ujf6Kf7ve112Huf92Ev/7enB/+nP7pbrPiQJZbi0jCSpoN9UNPTkj7JMwpbWgopAbhtbOWkytAF3K+/qo0SASNW2G2bLfnshpB4a9dmz7/Hx//dc3OXNZ46YRyXUV2dYRsD97qKL79qazu+vSI1vPXT7375bWSGocBofD2eIRzJ0cMC0tenwQ0gfvuSdvd14f1uEooLPE3JJHL6uCd/n5n8d35UOKPn6nhr8kyrV3ad3nz2iTiNL414EnefL/JGLlWZtZWaqoEh4xSjvsGb/6m9raFlsLm4uHkQWlv7T/weZzjHHe7xZiUzpJ5WAWBLDNwRKxwRYnFoXGxcaKxN6DR8BNn2o9Nqmmutvra5TnIjXMBlmIFZ3yPYX3Mt9v5mmHuwYvvxPverL9eSvszXNjUXrkbqcGOVW2bEbDGKi3MLVTWzzWHF54Bu/2rA1qko6l9fFgVbBurfVBWFFlVW1ugxOwcs+8W//FcUZJieLl9WXA8eGL5crB7fhOMyxl8bjQWGjB1bW/ok6Ucqensr7F8H7utsmdqoHmz99rvyeE/Pz7u64mvVXLjyY8v8j5XhZeH3aPX75dpiO5eN/OzwcG7zkflt/sd5e7YcqbOowfRg22R5585at2vXX87W1Y0gQ079497eYT1EkyoEqMYABmHd8QvKGrRG6bJYTDCCZYGEWcm5G1jXM2i54Y9WtiBuklP57YtBZMAWlu2fYzDM7Q+5FmxKS3Oz5jwK6IactbWPowuQgNyHluKlaw9wnbOmtuajo/VSw9FrBSRwMcuUV2ZwFhh6s7hsqriWCsgA2s3nFcri4I7O+asxwxZbtLL03E9bhcR6Yz9mIbF0U96K0xGA7bx9y+l2//73j+H2i0EGd27uAVNI/WhCYuWqIDaYxads0lcVFV+dOlHmBx/qO7c6/uZX0tReUtJQv64y3adAvX6xDezAX/8Wm8Cgh/95O9OxsNCYnsXWQ+7pCz8/NMZ57ZAIGEdTw+ap8V+I3NUVe375wiv+lccqj172X7Yw5gJAUQGYPQ6QyxRfgeC+Qc5WnAMCAHFv6TJtet3pn/83b4YCAIBv35ofpTRyt5PjZEwT8KYAEQK8nFgBcE/yUwn2oqHSBKoEG7KZQLMpjo5uha/PI2yuBWOCTSDZajpqQ68+Za18jgGgYMT8nBhjKcFrKCYF6yKSZRLF5tR5YKhUzzNWM52mBvuPMiL7xPx4UaRgFiJZAVFscZ2HUIhUPcEaH5WWDvvmvdPfl5KaCvO8o1+fFCBb6hvuLz8lMROwfjPN8iar90RCCiRCJr3ugqHf6LqgUYYs5hzvu9tMIOUr/xpvRsNVvdZ/p+mB8n7V2Spo0T+aRhPpNhsNFOqxoE2u0suqTipgx58IJA0AAAA=) format('woff'); | ||||
| } | ||||
|  | ||||
| :root { | ||||
| 	--c-1: #111; | ||||
| 	--c-f: #fff; | ||||
| 	--c-2: #222; | ||||
| 	--c-3: #333; | ||||
| 	--c-4: #444; | ||||
| 	--c-5: #555; | ||||
| 	--c-6: #666; | ||||
| 	--c-8: #888; | ||||
| 	--c-b: #bbb; | ||||
| 	--c-c: #ccc; | ||||
| 	--c-e: #eee; | ||||
| 	--c-d: #ddd; | ||||
| 	--c-r: #e42; | ||||
| 	--c-g: #4e2; | ||||
| 	--c-l: #48a; | ||||
| 	--t-b: 0.5; | ||||
| 	--c-o: rgba(34, 34, 34, 0.9); | ||||
| 	--c-tb : rgba(34, 34, 34, var(--t-b)); | ||||
| 	--c-tba: rgba(102, 102, 102, var(--t-b)); | ||||
| 	--c-tbh: rgba(51, 51, 51, var(--t-b)); | ||||
| 	/*following are internal*/ | ||||
| 	--th: 70px; | ||||
| 	--tp: 70px; | ||||
| 	--bh: 63px; | ||||
| 	--tbp: 14px 8px 10px; | ||||
| 	--bbp: 9px 0 7px 0; | ||||
| 	--bhd: none; | ||||
| 	--bmt: 0px; | ||||
| } | ||||
|  | ||||
| html { | ||||
| 	touch-action: manipulation; | ||||
| } | ||||
|  | ||||
| body { | ||||
| 	margin: 0; | ||||
| 	background-color: var(--c-1); | ||||
| 	font-family: Helvetica, Verdana, sans-serif; | ||||
| 	font-size: 17px; | ||||
| 	color: var(--c-f); | ||||
| 	text-align: center; | ||||
| 	-webkit-touch-callout: none; | ||||
| 	-webkit-user-select: none; | ||||
| 	-moz-user-select: none; | ||||
| 	-ms-user-select: none; | ||||
| 	user-select: none; | ||||
| 	-webkit-tap-highlight-color: transparent; | ||||
| 	scrollbar-width: 6px; | ||||
| 	scrollbar-color: var(--c-sb) transparent; | ||||
| } | ||||
|  | ||||
| html, | ||||
| body { | ||||
| 	height: 100%; | ||||
| 	width: 100%; | ||||
| 	position: fixed; | ||||
| 	overscroll-behavior: none; | ||||
| } | ||||
|  | ||||
| #bg { | ||||
| 	height: 100vh; | ||||
| 	width: 100vw; | ||||
| 	position: fixed; | ||||
| 	z-index: -10; | ||||
| 	background-position: center; | ||||
| 	background-repeat: no-repeat; | ||||
| 	background-size: cover; | ||||
| 	opacity: 0; | ||||
| 	transition: opacity 2s; | ||||
| } | ||||
|  | ||||
| p { | ||||
| 	margin: 10px 0 2px 0; | ||||
| } | ||||
| a, p, a:visited { | ||||
| 	color: var(--c-d); | ||||
| } | ||||
| a, a:visited { | ||||
| 	text-decoration: none; | ||||
| } | ||||
|  | ||||
| button { | ||||
| 	outline: none; | ||||
| 	cursor: pointer; | ||||
| 	background-color: transparent; | ||||
| 	border: none; | ||||
| 	transition: color 0.3s, background-color 0.3s; | ||||
| 	font-size: 19px; | ||||
| 	color: var(--c-c); | ||||
| 	min-width: 40px; | ||||
| 	min-height: 40px; | ||||
| } | ||||
| button:hover { | ||||
|     background: var(--c-4); | ||||
| } | ||||
|  | ||||
| .label { | ||||
| 	margin: 0; | ||||
| 	padding: 6px 0 0; | ||||
| } | ||||
|  | ||||
| #namelabel { | ||||
| 	position: fixed; | ||||
| 	bottom: calc(var(--bh) + 6px); | ||||
| 	right: 4px; | ||||
| 	color: var(--c-6); | ||||
| 	cursor: pointer; | ||||
| 	writing-mode: vertical-rl; | ||||
| } | ||||
|  | ||||
| .wrapper { | ||||
| 	position: fixed; | ||||
| 	top: 0; | ||||
| 	left: 0; | ||||
| 	right: 0; | ||||
| 	background: var(--c-tb); | ||||
| 	z-index: 1; | ||||
| } | ||||
|  | ||||
| .center { | ||||
| 	margin: 0 auto; | ||||
| 	width: 320px; | ||||
| } | ||||
|  | ||||
| .icons { | ||||
| 	font-family: 'WIcons'; | ||||
| 	font-style: normal; | ||||
| 	font-size: 24px; | ||||
| 	line-height: 1; | ||||
| 	display: inline-block; | ||||
| 	margin: -2px 0 4px 0; | ||||
| 	text-shadow: -1px -1px 0 var(--c-3), 1px -1px 0 var(--c-3), -1px 1px 0 var(--c-3), 1px 1px 0 var(--c-3); | ||||
| } | ||||
|  | ||||
| .huge { | ||||
| 	font-size: 42px; | ||||
| } | ||||
|  | ||||
| .infot { | ||||
| 	table-layout: fixed; | ||||
| 	width: 100%; | ||||
| } | ||||
|  | ||||
| .keytd { | ||||
| 	text-align: left; | ||||
| 	padding-bottom: 8px; | ||||
| } | ||||
|  | ||||
| .valtd { | ||||
| 	text-align: right; | ||||
| 	padding-bottom: 8px; | ||||
| } | ||||
|  | ||||
| .valtd i { | ||||
| 	font-size: small; | ||||
| } | ||||
|  | ||||
| .slider-icon | ||||
| { | ||||
| 	transform: translate(4px,3px); | ||||
| 	color: var(--c-d); | ||||
| } | ||||
|  | ||||
| .il { | ||||
| 	display: inline-block; | ||||
| 	vertical-align: middle; | ||||
| } | ||||
|  | ||||
| .tab { | ||||
| 	background-color: transparent; | ||||
| 	color: var(--c-d); | ||||
| } | ||||
|  | ||||
| .tab button { | ||||
| 	background-color: transparent; | ||||
| 	float: left; | ||||
| 	border: none; | ||||
| 	transition: color 0.3s, background-color 0.3s; | ||||
| 	font-size: 17px; | ||||
| 	color: var(--c-c); | ||||
| 	min-width: 44px; | ||||
| } | ||||
|  | ||||
| .top button { | ||||
| 	padding: var(--tbp); | ||||
| 	margin: 0; | ||||
| } | ||||
|  | ||||
| .tab button:hover { | ||||
| 	background-color: var(--c-tbh); | ||||
| 	color: var(--c-e); | ||||
| } | ||||
|  | ||||
| .tab button.active { | ||||
| 	background-color: var(--c-tba) !important; | ||||
| 	color: var(--c-f); | ||||
| } | ||||
|  | ||||
| .active { | ||||
| 	background-color: var(--c-6) !important; | ||||
| 	color: var(--c-f); | ||||
| } | ||||
|  | ||||
| .container { | ||||
| 	width: 100%; | ||||
| 	height: calc(100% - var(--tp) - var(--bh)); | ||||
| 	margin-top: var(--tp); | ||||
| 	overscroll-behavior: none; | ||||
| } | ||||
|  | ||||
| .tabcontent { | ||||
| 	position: relative; | ||||
| 	width: 100%; | ||||
| 	box-sizing: border-box; | ||||
| 	border: 0px; | ||||
| 	overflow: auto; | ||||
| 	height: 100%; | ||||
| 	overscroll-behavior: none; | ||||
| } | ||||
|  | ||||
| .smooth { transition: transform	calc(var(--f, 1)*.5s) ease-out } | ||||
|  | ||||
| .tab-label { | ||||
| 	margin: 0 0 -5px 0; | ||||
| 	padding-bottom: 4px; | ||||
| } | ||||
|  | ||||
| .overlay { | ||||
| 	position: fixed; | ||||
| 	height: 100%; | ||||
| 	width: 100%; | ||||
| 	top: 0; | ||||
| 	left: 0; | ||||
| 	background-color: var(--c-3); | ||||
| 	font-size: 24px; | ||||
| 	display: flex; | ||||
| 	align-items: center; | ||||
| 	justify-content: center; | ||||
| 	z-index: 11; | ||||
| 	opacity: 0.95; | ||||
| 	transition: 0.7s; | ||||
| 	pointer-events: none; | ||||
| } | ||||
|  | ||||
| #toast { | ||||
| 	opacity: 0; | ||||
| 	background-color: var(--c-5); | ||||
| 	max-width: 90%; | ||||
| 	color: var(--c-f); | ||||
| 	text-align: center; | ||||
| 	border-radius: 5px; | ||||
| 	padding: 16px; | ||||
| 	position: fixed; | ||||
| 	z-index: 5; | ||||
| 	left: 50%; | ||||
| 	transform: translateX(-50%); | ||||
| 	bottom: calc(var(--bh) + 22px); | ||||
| 	font-size: 17px; | ||||
| 	pointer-events: none; | ||||
| } | ||||
|  | ||||
| #toast.show { | ||||
| 	opacity: 1; | ||||
| 	animation: fadein 0.5s, fadein 0.5s 2.5s reverse; | ||||
| } | ||||
|  | ||||
| #toast.error { | ||||
| 	opacity: 1; | ||||
| 	background-color: #b21; | ||||
| 	animation: fadein 0.5s; | ||||
| } | ||||
|  | ||||
| .modal { | ||||
| 	position:fixed; | ||||
| 	left: 0px; | ||||
| 	bottom: 0px; | ||||
| 	right: 0px; | ||||
| 	top: calc(var(--th) - 1px); | ||||
| 	background-color: var(--c-o); | ||||
| 	transform: translateY(100%); | ||||
| 	transition: transform 0.4s; | ||||
| 	padding: 8px; | ||||
| 	font-size: 20px; | ||||
| 	overflow: auto; | ||||
| } | ||||
|  | ||||
| #info, #nodes { | ||||
| 	z-index: 3; | ||||
| } | ||||
|  | ||||
| #rover { | ||||
| 	z-index: 2; | ||||
| } | ||||
|  | ||||
| #ndlt { | ||||
|   margin: 12px 0; | ||||
| } | ||||
|  | ||||
| #roverstar { | ||||
| 	position: fixed; | ||||
| 	top: calc(var(--th) + 5px); | ||||
| 	left: 1px; | ||||
| 	display: none; | ||||
| 	cursor: pointer; | ||||
| } | ||||
|  | ||||
| #connind { | ||||
| 	position: fixed; | ||||
| 	bottom: calc(var(--bh) + 5px); | ||||
| 	left: 4px; | ||||
| 	padding: 5px; | ||||
| 	border-radius: 5px; | ||||
| 	background-color: #a90; | ||||
| 	z-index: -2; | ||||
| } | ||||
|  | ||||
| #imgw { | ||||
| 	display: inline-block; | ||||
| 	margin: 8px; | ||||
| } | ||||
|  | ||||
| #kv, #kn { | ||||
| 	/*max-width: 490px;*/ | ||||
| 	display: inline-block; | ||||
| } | ||||
|  | ||||
| #info table, #nodes table { | ||||
| 	table-layout: fixed; | ||||
| 	width: 100%; | ||||
| } | ||||
|  | ||||
| #info td, #nodes td { | ||||
|   padding-bottom: 8px; | ||||
| } | ||||
|  | ||||
| #info .btn { | ||||
| 	margin: 5px; | ||||
| } | ||||
| #info table .btn, #nodes table .btn { | ||||
| 	margin: 0; | ||||
| 	width: 180px; | ||||
| } | ||||
| #info div, #nodes div { | ||||
| 	width: 490px; | ||||
| 	margin: 0 auto; | ||||
| } | ||||
|  | ||||
| #kn td { | ||||
|   padding-bottom: 12px; | ||||
| } | ||||
|  | ||||
| #heart { | ||||
| 	transition: color 0.9s; | ||||
| 	font-size: 16px; | ||||
| 	color: #f00; | ||||
| } | ||||
|  | ||||
| img { | ||||
| 	max-width: 100%; | ||||
| 	max-height: 100%; | ||||
| } | ||||
|  | ||||
| .wi { | ||||
| 	image-rendering: pixelated; | ||||
| 	image-rendering: crisp-edges; | ||||
| 	width: 210px; | ||||
| } | ||||
|  | ||||
| @keyframes fadein { | ||||
| 	from {bottom: 0; opacity: 0;} | ||||
| 	to {bottom: calc(var(--bh) + 22px); opacity: 1;} | ||||
| } | ||||
|  | ||||
| .sliderwrap { | ||||
| 	height: 30px; | ||||
| 	width: 250px; | ||||
| 	position: relative; | ||||
| 	margin: 4px 0; | ||||
| } | ||||
| #Colors .sliderwrap { | ||||
| 	width: 260px; | ||||
| 	margin: 10px 0 0; | ||||
| } | ||||
|  | ||||
| .sliderdisplay { | ||||
| 	content:''; | ||||
| 	position: absolute; | ||||
| 	top: 10px; left: 8px; right: 8px; | ||||
| 	height: 8px; | ||||
| 	background: var(--c-4); | ||||
| 	border-radius: 16px; | ||||
| 	pointer-events: none; | ||||
| 	z-index: -1; | ||||
| } | ||||
| #Colors .sliderdisplay { | ||||
| 	height: 28px; | ||||
| 	top: 0; bottom: 0; | ||||
| 	left: 0; right: 0; | ||||
| 	/*border: 1px solid var(--c-b);*/ | ||||
| } | ||||
| #rwrap .sliderdisplay { background: linear-gradient(90deg, #000 0%, #f00); } | ||||
| #gwrap .sliderdisplay { background: linear-gradient(90deg, #000 0%, #0f0); } | ||||
| #bwrap .sliderdisplay { background: linear-gradient(90deg, #000 0%, #00f); } | ||||
| #wwrap .sliderdisplay { background: linear-gradient(90deg, #000 0%, #fff); } | ||||
| #kwrap .sliderdisplay { background: linear-gradient(90deg, #ff8f1f 0%, #fff 50%, #cbdbff); } | ||||
| #wbal .sliderdisplay  { background: linear-gradient(90deg, #ff8f1f 0%, #fff 50%, #d4e0ff); } | ||||
|  | ||||
| .sliderbubble { | ||||
| 	width: 24px; | ||||
| 	position: relative; | ||||
| 	display: inline-block; | ||||
| 	border-radius: 10px; | ||||
| 	background: var(--c-3); | ||||
| 	color: var(--c-f); | ||||
| 	padding: 4px 4px 2px; | ||||
| 	font-size: 14px; | ||||
| 	right: 3px; | ||||
| 	transition: visibility 0.25s ease, opacity 0.25s ease; | ||||
| 	opacity: 0; | ||||
| 	visibility: hidden; | ||||
| } | ||||
|  | ||||
| output.sliderbubbleshow { | ||||
| 	visibility: visible; | ||||
| 	opacity: 1; | ||||
| } | ||||
|  | ||||
| .hidden { | ||||
| 	display: none; | ||||
| } | ||||
|  | ||||
| input[type=range] { | ||||
| 	-webkit-appearance: none; | ||||
| 	width: 100%; | ||||
| 	padding: 0; | ||||
| 	margin: 0; | ||||
| 	background-color: transparent; | ||||
| 	cursor: pointer; | ||||
| } | ||||
| #Colors input[type=range] { | ||||
| 	width: 252px; | ||||
| 	margin: 0; | ||||
| } | ||||
| input[type=range]::-webkit-slider-runnable-track { | ||||
| 	width: 100%; | ||||
| 	height: 30px; | ||||
| 	cursor: pointer; | ||||
| 	background: transparent; | ||||
| } | ||||
| input[type=range]::-webkit-slider-thumb { | ||||
| 	border: 2px solid #000; | ||||
| 	height: 20px; | ||||
| 	width: 20px; | ||||
| 	border-radius: 50%; | ||||
| 	background: var(--c-f); | ||||
| 	cursor: pointer; | ||||
| 	-webkit-appearance: none; | ||||
| 	margin-top: 4px; | ||||
| } | ||||
| input[type=range]::-moz-range-track { | ||||
| 	width: 100%; | ||||
| 	height: 30px; | ||||
| 	background-color: var(--c-0); | ||||
| } | ||||
| input[type=range]::-moz-range-thumb { | ||||
| 	border: 2px solid var(--c-3); | ||||
| 	height: 20px; | ||||
| 	width: 20px; | ||||
| 	border-radius: 50%; | ||||
| 	background: var(--c-f); | ||||
| 	transform: translateY(5px); | ||||
| } | ||||
| #Colors input[type=range]::-webkit-slider-thumb { | ||||
| 	border: 2px solid #000; | ||||
| } | ||||
| #Colors input[type=range]::-moz-range-thumb { | ||||
| 	border: 2px solid var(--c-1); | ||||
| } | ||||
|  | ||||
| #Presets .list { | ||||
| 	max-height: 215px; | ||||
| 	overflow-y: scroll; | ||||
| 	overflow-x: hidden; | ||||
| 	width: 280px; | ||||
|     margin: 0 0 0 20px; | ||||
|   	-ms-overflow-style: none; | ||||
| 	scrollbar-width: none; /* Firefox */ | ||||
| } | ||||
| /* Hide scrollbar for Chrome, Safari and Opera */ | ||||
| #Presets .list::-webkit-scrollbar { | ||||
| 	display: none; | ||||
| } | ||||
|  | ||||
| #Segments .sliderwrap{ | ||||
| 	width: 225px; | ||||
| } | ||||
|  | ||||
| #picker, #rgbwrap, #kwrap, #vwrap, #wwrap, #wbal { | ||||
| 	display: none; | ||||
| } | ||||
|  | ||||
| .hd { | ||||
| 	display: var(--bhd); | ||||
| } | ||||
|  | ||||
| #briwrap { | ||||
| 	float: right; | ||||
| 	margin-top: var(--bmt); | ||||
| } | ||||
|  | ||||
| #picker { | ||||
| 	width: 260px; | ||||
| } | ||||
|  | ||||
| #picker, #csl, #segcont { | ||||
| 	margin: 10px auto 0; | ||||
| } | ||||
|  | ||||
| .btn { | ||||
| 	margin: 10px auto 0; | ||||
| 	width: 280px; | ||||
| 	font-size: 19px; | ||||
| 	background-color: var(--c-3); | ||||
| 	color: var(--c-d); | ||||
| 	cursor: pointer; | ||||
| 	border: 1px solid var(--c-3); | ||||
| 	border-radius: 25px; | ||||
| 	transition-duration: 0.3s; | ||||
| 	-webkit-backface-visibility: hidden; | ||||
| 	-webkit-transform:translate3d(0,0,0); | ||||
| 	overflow: clip; | ||||
| 	text-overflow: clip; | ||||
| 	min-height: 40px; | ||||
| 	line-height: 40px; | ||||
| } | ||||
| .btn:hover { | ||||
| 	background-color: var(--c-4); | ||||
| 	border: 1px solid var(--c-4); | ||||
| } | ||||
|  | ||||
| .btn-xs { | ||||
| 	width: 42px; | ||||
| 	height: 42px; | ||||
| 	margin: 4px; | ||||
| 	padding: 0; | ||||
| } | ||||
|  | ||||
| #fxBtn, #palBtn { | ||||
| 	background-color: var(--c-2); | ||||
| 	border: 1px solid var(--c-2); | ||||
| } | ||||
| #fxBtn:hover, #palBtn:hover { | ||||
| 	background-color: var(--c-3); | ||||
| 	border: 1px solid var(--c-3); | ||||
| } | ||||
|  | ||||
| .btn-icon { | ||||
| 	margin-right: 8px; | ||||
| 	vertical-align: middle; | ||||
| 	display: inline-block; | ||||
| } | ||||
|  | ||||
| .qcs { | ||||
| 	margin: 2px; | ||||
| 	border-radius: 14px; | ||||
| 	display: inline-block; | ||||
| 	width: 28px; | ||||
| 	height: 28px; | ||||
| 	line-height: 28px;} | ||||
| .qcsb { | ||||
| 	width: 26px; | ||||
| 	height: 26px; | ||||
| 	line-height: 26px; | ||||
| 	border: 1px solid var(--c-f); | ||||
| } | ||||
| option { | ||||
| 	background-color: var(--c-3); | ||||
| 	color: var(--c-f); | ||||
| } | ||||
| input[type=number], input[type=text] { | ||||
| 	background: var(--c-3); | ||||
| 	color: var(--c-f); | ||||
| 	border: 0px solid var(--c-f); | ||||
| 	border-radius: 5px; | ||||
| 	padding: 8px; | ||||
| 	margin: 6px 6px 6px 0; | ||||
| 	font-size: 19px; | ||||
| 	transition: background-color 0.2s; | ||||
| 	outline: none; | ||||
| 	width: 50px; | ||||
| 	-webkit-appearance: textfield; | ||||
| 	-moz-appearance: textfield; | ||||
| 	appearance: textfield; | ||||
| } | ||||
|  | ||||
| ::selection { | ||||
| 	background: var(--c-b); | ||||
| } | ||||
|  | ||||
| input[type=number]:focus, input[type=text]:focus { | ||||
| 	background: var(--c-6); | ||||
| } | ||||
|  | ||||
| input[type=number]::-webkit-inner-spin-button, | ||||
| input[type=number]::-webkit-outer-spin-button { | ||||
| 	-webkit-appearance: none; | ||||
| } | ||||
|  | ||||
| .pid { | ||||
| 	position: absolute; | ||||
| 	top: 0px; | ||||
| 	left: 0px; | ||||
| 	padding: 12px 0px 0px 12px; | ||||
| 	font-size: 16px; | ||||
| 	width: 20px; | ||||
| 	text-align: center; | ||||
| 	color: var(--c-b); | ||||
| } | ||||
|  | ||||
| .xxs { | ||||
| 	border: 2px solid var(--c-e) !important; | ||||
| 	width: 44px; | ||||
| 	height: 44px; | ||||
| 	margin: 5px; | ||||
| 	padding: 0; | ||||
| } | ||||
|  | ||||
| .xxs-w { | ||||
| 	border-width: 4px !important; | ||||
| 	margin: 2px; | ||||
| 	width: 50px; | ||||
| 	height: 50px; | ||||
| 	padding: 0; | ||||
| } | ||||
|  | ||||
| .qcs, .xxs { | ||||
| 	text-shadow: -1px -1px 0 var(--c-6), 1px -1px 0 var(--c-6), -1px 1px 0 var(--c-6), 1px 1px 0 var(--c-6); | ||||
| } | ||||
|  | ||||
| .psts { | ||||
| 	color: var(--c-f); | ||||
| 	margin: 6px; | ||||
| } | ||||
|  | ||||
| .pwr { | ||||
| 	color: var(--c-6); | ||||
| 	cursor: pointer; | ||||
| } | ||||
|  | ||||
| .act { | ||||
| 	color: var(--c-f); | ||||
| } | ||||
|  | ||||
| .check, .radio { | ||||
| 	display: inline-block; | ||||
| 	position: relative; | ||||
| 	cursor: pointer; | ||||
| 	text-align: center; | ||||
| } | ||||
|  | ||||
| .schkl { | ||||
| 	width: 24px; | ||||
| 	top: -2px; | ||||
| } | ||||
|  | ||||
| .check input, .radio input { | ||||
| 	position: absolute; | ||||
| 	opacity: 0; | ||||
| 	cursor: pointer; | ||||
| 	height: 0; | ||||
| 	width: 0; | ||||
| } | ||||
|  | ||||
| .checkmark, .radiomark { | ||||
| 	position: absolute; | ||||
| 	top: 0; | ||||
| 	bottom: 0; | ||||
| 	left: 0; | ||||
| 	background-color: var(--c-3); | ||||
| 	border: 1px solid var(--c-2); | ||||
| } | ||||
|  | ||||
| .radiomark { | ||||
| 	height: 24px; | ||||
| 	width: 24px; | ||||
| 	border-radius: 50%; | ||||
| } | ||||
|  | ||||
| .checkmark { | ||||
| 	height: 25px; | ||||
| 	width: 25px; | ||||
| 	border-radius: 10px; | ||||
| } | ||||
|  | ||||
| .check:hover input ~ .checkmark { | ||||
| 	background-color: var(--c-4); | ||||
| } | ||||
|  | ||||
| .check input:checked ~ .checkmark { | ||||
| 	background-color: var(--c-6); | ||||
| } | ||||
|  | ||||
| .checkmark:after, .radiomark:after { | ||||
| 	content: ""; | ||||
| 	position: absolute; | ||||
| 	display: none; | ||||
| } | ||||
|  | ||||
| .check input:checked ~ .checkmark:after, .radio input:checked ~ .radiomark:after { | ||||
| 	display: block; | ||||
| } | ||||
|  | ||||
| .check .checkmark:after { | ||||
| 	left: 9px; | ||||
| 	top: 5px; | ||||
| 	width: 5px; | ||||
| 	height: 10px; | ||||
| 	border: solid var(--c-f); | ||||
| 	border-width: 0 3px 3px 0; | ||||
| 	-webkit-transform: rotate(45deg); | ||||
| 	-ms-transform: rotate(45deg); | ||||
| 	transform: rotate(45deg); | ||||
| } | ||||
|  | ||||
| .radio .radiomark:after { | ||||
| 	width: 12px; | ||||
| 	height: 12px; | ||||
| 	top: 50%; | ||||
| 	left: 50%; | ||||
| 	margin: -6px; | ||||
| 	border-radius: 50%; | ||||
| 	background: var(--c-f); | ||||
| } | ||||
|  | ||||
| .h { | ||||
| 	font-size: 13px; | ||||
| 	color: var(--c-b); | ||||
| } | ||||
|  | ||||
| .list { | ||||
| 	position: relative; | ||||
| 	width: 280px; | ||||
| 	transition: background-color 0.5s; | ||||
|     margin: auto auto 20px; | ||||
| 	font-size: 19px; | ||||
| 	line-height: 24px; | ||||
| } | ||||
|  | ||||
| .lstI { | ||||
|     cursor: pointer; | ||||
| 	background-color: var(--c-2); | ||||
| 	overflow: hidden; | ||||
| 	border-radius: 20px; | ||||
| 	display: block; | ||||
| 	position: relative; | ||||
| 	border: 1px solid var(--c-2); | ||||
| 	padding: 8px 10px; | ||||
| 	margin: 10px 0; | ||||
| 	min-height: 24px; | ||||
| } | ||||
|  | ||||
| .selected { /* has to be after .lstI */ | ||||
| 	background: var(--c-5); | ||||
| } | ||||
|  | ||||
| .lstI:hover { | ||||
|     background: var(--c-4); | ||||
| } | ||||
| /* | ||||
| .lstI:last-child { | ||||
| 	border: none; | ||||
|     border-radius: 0 0 20px 20px; | ||||
|     padding-bottom: 10px; | ||||
| } | ||||
| */ | ||||
| .lstIcontent { | ||||
| 	width: 100%; | ||||
| 	vertical-align: middle; | ||||
| 	padding: 0 20px 0 5px; | ||||
| 	text-align: left; | ||||
| } | ||||
|  | ||||
| .lstIname { | ||||
| 	white-space: nowrap; | ||||
| 	cursor: pointer; | ||||
| } | ||||
|  | ||||
| .lstIprev { | ||||
| 	width: 100%; | ||||
| 	height: 8px; | ||||
| 	position: absolute; | ||||
| 	bottom: 0; | ||||
| 	left: 0; | ||||
|   } | ||||
|  | ||||
| /* Dropdown Content (Hidden by Default) */ | ||||
| .dd-content { | ||||
| 	display: none; | ||||
| 	position: absolute; | ||||
| 	width: 284px; | ||||
| 	z-index: 1; | ||||
| 	height: 260px; | ||||
| 	overflow-y: scroll; | ||||
| 	overflow-x: hidden; | ||||
| 	padding: 0 18px; | ||||
| 	margin-top: 10px; | ||||
| 	-ms-overflow-style: none; | ||||
| 	scrollbar-width: none; /* Firefox */ | ||||
| } | ||||
| /* Hide scrollbar for Chrome, Safari and Opera */ | ||||
| .dd-content::-webkit-scrollbar { | ||||
| 	display: none; | ||||
| } | ||||
|  | ||||
| .fnd { | ||||
| 	position: sticky; | ||||
| 	top: 0; | ||||
| 	z-index: 1; | ||||
| 	width: 280px; | ||||
| 	margin: 0 auto; | ||||
| } | ||||
|  | ||||
| .search-icon { | ||||
| 	position: absolute; | ||||
| 	top: 10px; | ||||
| 	left: 13px; | ||||
| 	pointer-events: none; | ||||
| 	width: 24px; | ||||
| 	height: 24px; | ||||
| 	margin-top: -1px; | ||||
| 	z-index: 1; | ||||
| } | ||||
|  | ||||
| .clear-icon { | ||||
| 	position: absolute; | ||||
| 	display: none; | ||||
| 	top: 10px; | ||||
| 	right: 13px; | ||||
| 	cursor: pointer; | ||||
| 	margin-top: -1px; | ||||
| 	z-index: 1; | ||||
| } | ||||
|  | ||||
| input[type=text].fnd { | ||||
| 	display: block; | ||||
| 	width: 100%; | ||||
| 	box-sizing: border-box; | ||||
|     padding: 8px 48px 8px 48px; | ||||
|     margin: 5px auto 0; | ||||
| 	text-align: left; | ||||
| 	border-radius: 25px; | ||||
| 	background-color: var(--c-2); | ||||
|     border: 1px solid var(--c-4); | ||||
| } | ||||
|  | ||||
| input[type=text].fnd:focus { | ||||
| 	background-color: var(--c-4); | ||||
| } | ||||
|  | ||||
| input[type=text].fnd:not(:placeholder-shown), input[type=text].fnd:hover { | ||||
| 	background-color: var(--c-3); | ||||
| } | ||||
|  | ||||
| .h, .c { | ||||
| 	text-align: center; | ||||
| } | ||||
|  | ||||
| ::-webkit-scrollbar { | ||||
| 	width: 6px; | ||||
| } | ||||
| ::-webkit-scrollbar-track { | ||||
| 	background: transparent; | ||||
| } | ||||
| ::-webkit-scrollbar-thumb { | ||||
| 	background: var(--c-sb); | ||||
| 	opacity: 0.2; | ||||
| 	border-radius: 5px; | ||||
| } | ||||
| ::-webkit-scrollbar-thumb:hover { | ||||
| 	background: var(--c-sbh); | ||||
| } | ||||
|  | ||||
| @media not all and (hover: none) { | ||||
| 	.sliderwrap:hover + output.sliderbubble { | ||||
| 		visibility: visible; | ||||
| 		opacity: 1; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @media all and (max-width: 335px) { | ||||
| 	.sliderbubble { | ||||
|     	display: none; | ||||
|   	} | ||||
| } | ||||
|  | ||||
| @media all and (max-width: 550px) and (min-width: 374px) { | ||||
| 	#info .btn, #nodes .btn { | ||||
| 		width: 150px; | ||||
| 	} | ||||
| 	#info div, #nodes div { | ||||
| 		width: 320px; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @media all and (max-width: 540px) { | ||||
| 	.top button { | ||||
| 		width: 16.6%; | ||||
| 		padding: 8px 0 4px 0; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @media all and (min-width: 541px) and (max-width: 719px) { | ||||
| 	.top button { | ||||
| 		width: 14.2%; | ||||
| 		padding: 8px 0 4px 0; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @media all and (max-width: 719px) { | ||||
| 	.hd { | ||||
| 		display: none !important; | ||||
| 	} | ||||
| 	#briwrap { | ||||
| 		margin-top: 0px !important; | ||||
| 		float: none; | ||||
| 	} | ||||
| } | ||||
| @@ -1,263 +0,0 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
| <head> | ||||
| 	<meta name="viewport" content="width=device-width, initial-scale=1, minimum-scale=1"> | ||||
| 	<meta charset="utf-8"> | ||||
| 	<meta name="theme-color" content="#222222"> | ||||
| 	<meta content="yes" name="apple-mobile-web-app-capable"> | ||||
| 	<link rel="shortcut icon" href="data:image/x-icon;base64,AAABAAEAEBAAAAEAGACGAAAAFgAAAIlQTkcNChoKAAAADUlIRFIAAAAQAAAAEAgGAAAAH/P/YQAAAE1JREFUOI1j/P//PwOxgNGeAUMxE9G6cQCKDWAhpADZ2f8PMjBS3QW08QK20KaZC2gfC9hCnqouoNgARgY7zMxAyNlUdQHlXiAlO2MDAD63EVqNHAe0AAAAAElFTkSuQmCC"/> | ||||
| 	<title>WLED</title> | ||||
| 	<script> | ||||
| 	function feedback(){} | ||||
| 	// instead of including [script src="iro.js"][/script] and [script src="rangetouch.js"][/script] | ||||
| 	// (which would be inlined by nodeJS inliner during minimization and compression) we need to load them dynamically | ||||
| 	// the following is needed to load iro.js and rangetouch.js as consecutive requests to allow ESP8266 | ||||
| 	// to keep up with requests (if requests happent too fast some may not get processed) | ||||
| 	// it will also call onLoad() after last is loaded (it was removed from [body onload="onLoad()"]). | ||||
| 	var h  = document.getElementsByTagName('head')[0]; | ||||
| 	var l  = document.createElement('script'); | ||||
| 	l.type = 'application/javascript'; | ||||
| 	l.src = 'iro.js'; | ||||
| 	l.addEventListener('load', (e) => { | ||||
| 		// after iro is loaded initialize global variable | ||||
| 		cpick = new iro.ColorPicker("#picker", { | ||||
| 			width: 260, | ||||
| 			wheelLightness: false, | ||||
| 			wheelAngle: 270, | ||||
| 			wheelDirection: "clockwise", | ||||
| 			layout: [{ | ||||
| 				component: iro.ui.Wheel, | ||||
| 				options: {} | ||||
| 			}] | ||||
| 		}); | ||||
| 		cpick.on("input:end", () => {setColor(1);}); | ||||
| 		cpick.on("color:change", () => {updatePSliders()}); | ||||
| 		var l  = document.createElement('script'); | ||||
| 		l.type = 'application/javascript'; | ||||
| 		l.src = 'rangetouch.js'; | ||||
| 		l.addEventListener('load', (e) => { | ||||
| 			// after rangetouch is loaded initialize global variable | ||||
| 			ranges = RangeTouch.setup('input[type="range"]', {}); | ||||
| 			let stateCheck = setInterval(() => { | ||||
| 				if (document.readyState === 'complete') { | ||||
| 					clearInterval(stateCheck); | ||||
| 					// document ready, start processing UI | ||||
| 					onLoad(); | ||||
| 				} | ||||
| 			}, 100); | ||||
| 		}); | ||||
| 		setTimeout(function(){h.appendChild(l)},50); | ||||
| 	}); | ||||
| 	setTimeout(function(){h.appendChild(l)},50); | ||||
| 	</script> | ||||
| 	<link rel="stylesheet" href="simple.css"> | ||||
| </head> | ||||
| <body> | ||||
|  | ||||
| <div id="cv" class="overlay">Loading WLED UI...</div> | ||||
| <noscript><div class="overlay" style="opacity:1;">Sorry, WLED UI needs JavaScript!</div></noscript> | ||||
| <div id="bg"></div> | ||||
|  | ||||
| <div class="wrapper" id="top"> | ||||
| 	<div class="tab top"> | ||||
| 		<div class="btnwrap"> | ||||
| 			<button id="buttonPower" onclick="togglePower()"><i class="icons"></i><p class="tab-label">Power</p></button> | ||||
| 			<button id="buttonI" onclick="toggleInfo()"><i class="icons"></i><p class="tab-label">Info</p></button> | ||||
| 			<button id="buttonNodes" onclick="toggleNodes()"><i class="icons"></i><p class="tab-label">Nodes</p></button></div> | ||||
| 			<button onclick="window.location.href='/settings';"><i class="icons"></i><p class="tab-label">Config</p></button> | ||||
|             <button id="buttonCP" onclick="tglCP()"><i class="icons"></i><p class="tab-label">Expand</p></button> | ||||
|             <!--button id="buttonBri" onclick="tglBri()"><i class="icons"></i><p class="tab-label">Brightness</p></button--> | ||||
| 		</div> | ||||
| 		<div id="briwrap"> | ||||
| 			<p class="label hd">Global brightness</p> | ||||
| 			<div class="il"> | ||||
| 				<i class="icons slider-icon" onclick="tglTheme()"></i> | ||||
| 				<div class="sliderwrap il"> | ||||
| 					<input id="sliderBri" onchange="setBri()" oninput="updateTrail(this)" max="255" min="1" type="range" value="128" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 				<output class="sliderbubble"></output> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </div> | ||||
|  | ||||
| <div class ="container"> | ||||
|     <div class="tabcontent"> | ||||
| 		<div id="QuickLoad" class="center"> | ||||
| 			<p class="label h">Quick Load</p> | ||||
|         	<div id="pql"></div> | ||||
| 		</div> | ||||
|  | ||||
| 		<div id="QCS" class="center"> | ||||
| 			<p class="label h">Solid color</p> | ||||
| 			<div id="qcs-w" class="center"> | ||||
| 				<div class="qcs" onclick="pC('#ff0000');" title="Red" style="background-color:#ff0000;"></div> | ||||
| 				<div class="qcs" onclick="pC('#ffa000');" title="Orange" style="background-color:#ffa000;"></div> | ||||
| 				<div class="qcs" onclick="pC('#ffc800');" title="Yellow" style="background-color:#ffc800;"></div> | ||||
| 				<div class="qcs" onclick="pC('#ffe0a0');" title="Warm White" style="background-color:#ffe0a0;"></div> | ||||
| 				<div class="qcs" onclick="pC('#ffffff');" title="White" style="background-color:#ffffff;"></div> | ||||
| 				<div class="qcs qcsb" onclick="pC('#000000');" title="Black" style="background-color:#000000;"></div><br> | ||||
| 				<div class="qcs" onclick="pC('#ff00ff');" title="Pink" style="background-color:#ff00ff;"></div> | ||||
| 				<div class="qcs" onclick="pC('#0000ff');" title="Blue" style="background-color:#0000ff;"></div> | ||||
| 				<div class="qcs" onclick="pC('#00ffc8');" title="Cyan" style="background-color:#00ffc8;"></div> | ||||
| 				<div class="qcs" onclick="pC('#08ff00');" title="Green" style="background-color:#08ff00;"></div> | ||||
| 				<div class="qcs" onclick="pC('rnd');" title="Random" style="background:linear-gradient(to right, red, orange, yellow, green, blue, purple);transform: translateY(-11px);">R</div> | ||||
| 			</div> | ||||
| 		</div> | ||||
|  | ||||
| 		<div id="picker" class="center"></div> | ||||
|  | ||||
| 		<div id="Colors" class="center"> | ||||
| 			<div id="vwrap"> | ||||
| 				<!--p class="label h">Value</p--> | ||||
| 				<div class="sliderwrap il"> | ||||
| 					<input id="sliderV" class="noslide" oninput="fromV()" onchange="setColor(0)" max="100" min="0" type="range" value="100" step="any" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div><br> | ||||
| 			</div> | ||||
| 			<div id="kwrap"> | ||||
| 				<!--p class="label h">Temperature</p--> | ||||
| 				<div class="sliderwrap il"> | ||||
| 					<input id="sliderK" class="noslide" oninput="fromK()" onchange="setColor(0)" max="10091" min="1900" type="range" value="6550" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 			<div id="rgbwrap" class="center"> | ||||
| 				<p class="label h">RGB channels</p> | ||||
| 				<div id="rwrap" class="il"> | ||||
| 					<div class="sliderwrap il"> | ||||
| 						<input id="sliderR" class="noslide" onchange="fromRgb()" max="255" min="0" type="range" value="128" /> | ||||
| 						<div class="sliderdisplay"></div> | ||||
| 					</div> | ||||
| 				</div><br> | ||||
| 				<div id="gwrap" class="il"> | ||||
| 					<div class="sliderwrap il"> | ||||
| 						<input id="sliderG" class="noslide" onchange="fromRgb()" max="255" min="0" type="range" value="128" /> | ||||
| 						<div class="sliderdisplay"></div> | ||||
| 					</div> | ||||
| 				</div><br> | ||||
| 				<div id="bwrap" class="il"> | ||||
| 					<div class="sliderwrap il"> | ||||
| 						<input id="sliderB" class="noslide" onchange="fromRgb()" max="255" min="0" type="range" value="128" /> | ||||
| 						<div class="sliderdisplay"></div> | ||||
| 					</div> | ||||
| 				</div><br> | ||||
| 			</div> | ||||
| 			<div id="wwrap" class="center"> | ||||
| 				<p class="label h">White channel</p> | ||||
| 				<div class="sliderwrap il"> | ||||
| 					<input id="sliderW" class="noslide" onchange="setColor(0)" max="255" min="0" type="range" value="128" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 			<div id="wbal"> | ||||
| 				<p class="label h">White balance</p> | ||||
| 				<div class="sliderwrap il"> | ||||
| 					<input id="sliderA" class="noslide" onchange="setBalance(this.value)" max="255" min="0" type="range" value="128" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</div> | ||||
|  | ||||
| 		<div id="Slots" class="center"> | ||||
|         	<p class="label h">Color slots</p> | ||||
| 			<div id="csl" class="center" style="display: none;"> | ||||
| 				<button class="xxs btn" onclick="selectSlot(0);">1</button> | ||||
| 				<button class="xxs btn" onclick="selectSlot(1);">2</button> | ||||
| 				<button class="xxs btn" onclick="selectSlot(2);">3</button> | ||||
| 			</div> | ||||
| 		</div> | ||||
|  | ||||
| 		<div id="Segments" class="center"> | ||||
| 			<div id="segcont"></div> | ||||
| 		</div> | ||||
|  | ||||
| 		<div id="Presets" class="center"> | ||||
|         	<p class="label h">Presets</p> | ||||
| 			<div class="fnd"> | ||||
| 				<input type="text" class="fnd" placeholder="Search" oninput="search(this,'pcont')" onfocus="search(this)" /> | ||||
| 				<i class="icons clear-icon" onclick="clean(this);"></i> | ||||
| 				<i class="icons search-icon"></i> | ||||
| 			</div> | ||||
| 			<div id="pcont" class="list"></div> | ||||
| 		</div> | ||||
|  | ||||
| 		<div id="Effects" class="center"> | ||||
| 			<p class="label h">Effect</p> | ||||
| 			<div title="Effect speed"> | ||||
| 				<i class="icons slider-icon"></i> | ||||
| 				<div class="sliderwrap il"> | ||||
| 					<input id="sliderSpeed" onchange="setSpeed()" oninput="updateTrail(this)" max="255" min="0" type="range" value="128" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 				<output class="sliderbubble"></output> | ||||
| 			</div> | ||||
| 			<div title="Effect intensity"> | ||||
| 				<i class="icons slider-icon" onclick="tglLabels()"></i> | ||||
| 				<div class="sliderwrap il"> | ||||
| 					<input id="sliderIntensity" onchange="setIntensity()" oninput="updateTrail(this)" max="255" min="0" type="range" value="128" /> | ||||
| 					<div class="sliderdisplay"></div> | ||||
| 				</div> | ||||
| 				<output class="sliderbubble"></output> | ||||
| 			</div> | ||||
| 			<div style="padding-bottom:20px;"> | ||||
| 				<div onclick="tglFxDropdown()" class="c btn" id="fxBtn"><i class="icons"></i> Solid</div> | ||||
| 				<div onclick="tglPalDropdown()" class="c btn" id="palBtn"><i class="icons"></i>Default</div> | ||||
| 				<div id="fxDropdown" class="dd-content"> | ||||
| 					<div class="fnd"> | ||||
| 						<input type="text" class="fnd" placeholder="Search" oninput="search(this,'fxlist')" onfocus="search(this)" /> | ||||
| 						<i class="icons clear-icon" onclick="clean(this);"></i> | ||||
| 						<i class="icons search-icon"></i> | ||||
| 					</div> | ||||
| 					<div id="fxlist" class="list"> | ||||
| 						<div class="lstI" data-id="0" onClick="setEffect(0)"><a href="#0" onClick="setEffect(0)">Solid</a></div> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 				<div id="palDropdown" class="dd-content"> | ||||
| 					<div class="fnd"> | ||||
| 						<input type="text" class="fnd" placeholder="Search" oninput="search(this,'pallist')" onfocus="search(this)" /> | ||||
| 						<i class="icons clear-icon" onclick="clean(this);"></i> | ||||
| 						<i class="icons search-icon"></i> | ||||
| 					</div> | ||||
| 					<div id="pallist" class="list"> | ||||
| 						<div class="lstI" data-id="0" onClick="setPalette(0)"><a href="#0" onClick="setPalette(0)">Default</a><div class="lstIprev"></div></div> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 				<br> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </div> | ||||
|  | ||||
| <div id="connind"></div> | ||||
| <div id="toast"></div> | ||||
| <div id="namelabel" onclick="toggleNodes()"></div> | ||||
|  | ||||
| <div id="info" class="modal"> | ||||
| 	<div id="imgw"> | ||||
| 		<img class="wi" alt="" src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAB0AAAAFCAYAAAC5Fuf5AAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAABbSURBVChTlY9bDoAwDMNW7n9nwCipytQN4Z8tbrTHmDmF4oPzyldwRqp1SSdnV/NuZuzqerAByxXznBw3igkeFEfXyUuhK/yFM0CxJfyqXZEOc6/Sr9/bf7uIC5Nwd7orMvAPAAAAAElFTkSuQmCC" /> | ||||
| 	</div><br> | ||||
| 	<div id="kv">Loading...</div><br> | ||||
| 	<div> | ||||
| 	<button class="btn" onclick="requestJson()">Refresh</button> | ||||
| 	<button class="btn" onclick="toggleInfo()">Close Info</button> | ||||
| 	<button class="btn" onclick="toggleNodes()">Instance List</button> | ||||
| 	<button class="btn" id="resetbtn" onclick="cnfReset()">Reboot WLED</button> | ||||
| 	</div> | ||||
| 	<span class="h">Made with <span id="heart">❤︎</span> by Aircoookie and the <a href="https://wled.discourse.group/" target="_blank">WLED community</a></span> | ||||
| </div> | ||||
|  | ||||
| <div id="nodes" class="modal"> | ||||
| 	<div id="ndlt">WLED instances</div> | ||||
| 	<div id="kn">Loading...</div><br> | ||||
| 	<div> | ||||
| 	<button class="btn" onclick="loadNodes()">Refresh</button> | ||||
| 	<button class="btn" onclick="toggleNodes()">Close list</button> | ||||
| 	</div> | ||||
| </div> | ||||
|  | ||||
| <i id="roverstar" class="icons huge" onclick="setLor(0)"></i><br> | ||||
| <script src="simple.js"></script> | ||||
| </body> | ||||
| </html> | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -16,6 +16,9 @@ hr { | ||||
| hr.sml { | ||||
|   width: 260px; | ||||
| } | ||||
| h4 { | ||||
|   margin: 0; | ||||
| } | ||||
| a, a:hover { | ||||
|   color: #28f; | ||||
|   text-decoration: none; | ||||
| @@ -78,6 +81,7 @@ input:disabled { | ||||
| } | ||||
| input[type="text"], | ||||
| input[type="number"], | ||||
| input[type="password"], | ||||
| select { | ||||
|     font-size: medium; | ||||
|     margin: 2px; | ||||
|   | ||||
| @@ -69,14 +69,10 @@ void handleDMX() | ||||
| } | ||||
|  | ||||
| void initDMX() { | ||||
|  #ifdef ESP8266 | ||||
|  #if defined(ESP8266) || defined(CONFIG_IDF_TARGET_ESP32C3) || defined(CONFIG_IDF_TARGET_ESP32S2) | ||||
|   dmx.init(512);        // initialize with bus length | ||||
|  #else | ||||
|   dmx.initWrite(512);  // initialize with bus length | ||||
|  #endif | ||||
| } | ||||
|  | ||||
| #else | ||||
| void handleDMX() {} | ||||
| void initDMX() {} | ||||
| #endif | ||||
|   | ||||
| @@ -11,6 +11,7 @@ | ||||
| //DDP protocol support, called by handleE131Packet | ||||
| //handles RGB data only | ||||
| void handleDDPPacket(e131_packet_t* p) { | ||||
|   static bool ddpSeenPush = false;  // have we seen a push yet? | ||||
|   int lastPushSeq = e131LastSequenceNumber[0]; | ||||
|  | ||||
|   //reject late packets belonging to previous frame (assuming 4 packets max. before push) | ||||
| @@ -34,6 +35,7 @@ void handleDDPPacket(e131_packet_t* p) { | ||||
|   uint16_t c = 0; | ||||
|   if (p->flags & DDP_TIMECODE_FLAG) c = 4; //packet has timecode flag, we do not support it, but data starts 4 bytes later | ||||
|  | ||||
|   if (realtimeMode != REALTIME_MODE_DDP) ddpSeenPush = false; // just starting, no push yet | ||||
|   realtimeLock(realtimeTimeoutMs, REALTIME_MODE_DDP); | ||||
|  | ||||
|   if (!realtimeOverride || (realtimeMode && useMainSegmentOnly)) { | ||||
| @@ -44,7 +46,8 @@ void handleDDPPacket(e131_packet_t* p) { | ||||
|   } | ||||
|  | ||||
|   bool push = p->flags & DDP_PUSH_FLAG; | ||||
|   if (push) { | ||||
|   ddpSeenPush |= push; | ||||
|   if (!ddpSeenPush || push) { // if we've never seen a push, or this is one, render display | ||||
|     e131NewData = true; | ||||
|     byte sn = p->sequenceNum & 0xF; | ||||
|     if (sn) e131LastSequenceNumber[0] = sn; | ||||
| @@ -184,7 +187,6 @@ void handleE131Packet(e131_packet_t* p, IPAddress clientIP, byte protocol){ | ||||
|             // only apply preset if not in playlist, or playlist changed | ||||
|             (currentPlaylist < 0 || dmxValPreset != currentPlaylist)) {  | ||||
|           presetCycCurr = dmxValPreset; | ||||
|           unloadPlaylist(); // applying a preset unloads the playlist | ||||
|           applyPreset(dmxValPreset, CALL_MODE_NOTIFICATION); | ||||
|         } | ||||
|  | ||||
| @@ -225,11 +227,16 @@ void handleE131Packet(e131_packet_t* p, IPAddress clientIP, byte protocol){ | ||||
|           if (e131_data[dataOffset+3]   != seg.intensity) seg.intensity = e131_data[dataOffset+3]; | ||||
|           if (e131_data[dataOffset+4]   != seg.palette)   seg.setPalette(e131_data[dataOffset+4]); | ||||
|  | ||||
|           uint8_t segOption = (uint8_t)floor(e131_data[dataOffset+5]/64.0); | ||||
|           if (segOption == 0 && (seg.mirror  || seg.reverse )) {seg.setOption(SEG_OPTION_MIRROR, false); seg.setOption(SEG_OPTION_REVERSED, false);} | ||||
|           if (segOption == 1 && (seg.mirror  || !seg.reverse)) {seg.setOption(SEG_OPTION_MIRROR, false); seg.setOption(SEG_OPTION_REVERSED,  true);} | ||||
|           if (segOption == 2 && (!seg.mirror || seg.reverse )) {seg.setOption(SEG_OPTION_MIRROR,  true); seg.setOption(SEG_OPTION_REVERSED, false);} | ||||
|           if (segOption == 3 && (!seg.mirror || !seg.reverse)) {seg.setOption(SEG_OPTION_MIRROR,  true); seg.setOption(SEG_OPTION_REVERSED,  true);} | ||||
|           if ((e131_data[dataOffset+5] & 0b00000010) != seg.reverse_y) { seg.setOption(SEG_OPTION_REVERSED_Y, e131_data[dataOffset+5] & 0b00000010); } | ||||
|           if ((e131_data[dataOffset+5] & 0b00000100) != seg.mirror_y) { seg.setOption(SEG_OPTION_MIRROR_Y, e131_data[dataOffset+5] & 0b00000100); } | ||||
|           if ((e131_data[dataOffset+5] & 0b00001000) != seg.transpose) { seg.setOption(SEG_OPTION_TRANSPOSED, e131_data[dataOffset+5] & 0b00001000); } | ||||
|           if ((e131_data[dataOffset+5] & 0b00110000) / 8 != seg.map1D2D) { | ||||
|             seg.map1D2D = (e131_data[dataOffset+5] & 0b00110000) / 8; | ||||
|           } | ||||
|           // To maintain backwards compatibility with prior e1.31 values, reverse is fixed to mask 0x01000000 | ||||
|           if ((e131_data[dataOffset+5] & 0b01000000) != seg.reverse) { seg.setOption(SEG_OPTION_REVERSED, e131_data[dataOffset+5] & 0b01000000); } | ||||
|           // To maintain backwards compatibility with prior e1.31 values, mirror is fixed to mask 0x10000000 | ||||
|           if ((e131_data[dataOffset+5] & 0b10000000) != seg.mirror) { seg.setOption(SEG_OPTION_MIRROR, e131_data[dataOffset+5] & 0b10000000); } | ||||
|  | ||||
|           uint32_t colors[3]; | ||||
|           byte whites[3] = {0,0,0}; | ||||
| @@ -490,7 +497,7 @@ void prepareArtnetPollReply(ArtPollReply *reply) { | ||||
|   // Node is DHCP capable | ||||
|   // Node supports 15 bit Port-Address (Art-Net 3 or 4) | ||||
|   // Node is able to switch between ArtNet and sACN | ||||
|   reply->reply_status_2 = (staticIP[0] == 0) ? 0x1F : 0x1D; | ||||
|   reply->reply_status_2 = (multiWiFi[0].staticIP[0] == 0) ? 0x1F : 0x1D; | ||||
|  | ||||
|   // RDM is disabled | ||||
|   // Output style is continuous | ||||
|   | ||||
| @@ -48,6 +48,21 @@ bool getJsonValue(const JsonVariant& element, DestType& destination, const Defau | ||||
|   return true; | ||||
| } | ||||
|  | ||||
| typedef struct WiFiConfig { | ||||
|   char clientSSID[33]; | ||||
|   char clientPass[65]; | ||||
|   IPAddress staticIP; | ||||
|   IPAddress staticGW; | ||||
|   IPAddress staticSN; | ||||
|   WiFiConfig(const char *ssid="", const char *pass="", uint32_t ip=0, uint32_t gw=0, uint32_t subnet=0x00FFFFFF) // little endian | ||||
|   : staticIP(ip) | ||||
|   , staticGW(gw) | ||||
|   , staticSN(subnet) | ||||
|   { | ||||
|     strncpy(clientSSID, ssid, 32); clientSSID[32] = 0; | ||||
|     strncpy(clientPass, pass, 64); clientPass[64] = 0; | ||||
|   } | ||||
| } wifi_config; | ||||
|  | ||||
| //colors.cpp | ||||
| // similar to NeoPixelBus NeoGammaTableMethod but allows dynamic changes (superseded by NPB::NeoGammaDynamicTableMethod) | ||||
| @@ -65,6 +80,8 @@ class NeoGammaWLEDMethod { | ||||
| uint32_t color_blend(uint32_t,uint32_t,uint16_t,bool b16=false); | ||||
| uint32_t color_add(uint32_t,uint32_t, bool fast=false); | ||||
| uint32_t color_fade(uint32_t c1, uint8_t amount, bool video=false); | ||||
| CRGBPalette16 generateHarmonicRandomPalette(CRGBPalette16 &basepalette); | ||||
| CRGBPalette16 generateRandomPalette(void); | ||||
| inline uint32_t colorFromRgbw(byte* rgbw) { return uint32_t((byte(rgbw[3]) << 24) | (byte(rgbw[0]) << 16) | (byte(rgbw[1]) << 8) | (byte(rgbw[2]))); } | ||||
| void colorHStoRGB(uint16_t hue, byte sat, byte* rgb); //hue, sat to rgb | ||||
| void colorKtoRGB(uint16_t kelvin, byte* rgb); | ||||
| @@ -95,6 +112,10 @@ bool readObjectFromFileUsingId(const char* file, uint16_t id, JsonDocument* dest | ||||
| bool readObjectFromFile(const char* file, const char* key, JsonDocument* dest); | ||||
| void updateFSInfo(); | ||||
| void closeFile(); | ||||
| inline bool writeObjectToFileUsingId(const String &file, uint16_t id, JsonDocument* content) { return writeObjectToFileUsingId(file.c_str(), id, content); }; | ||||
| inline bool writeObjectToFile(const String &file, const char* key, JsonDocument* content) { return writeObjectToFile(file.c_str(), key, content); }; | ||||
| inline bool readObjectFromFileUsingId(const String &file, uint16_t id, JsonDocument* dest) { return readObjectFromFileUsingId(file.c_str(), id, dest); }; | ||||
| inline bool readObjectFromFile(const String &file, const char* key, JsonDocument* dest) { return readObjectFromFile(file.c_str(), key, dest); }; | ||||
|  | ||||
| //hue.cpp | ||||
| void handleHue(); | ||||
| @@ -208,6 +229,7 @@ void handlePlaylist(); | ||||
| void serializePlaylist(JsonObject obj); | ||||
|  | ||||
| //presets.cpp | ||||
| const char *getPresetsFileName(bool persistent = true); | ||||
| void initPresetsFile(); | ||||
| void handlePresets(); | ||||
| bool applyPreset(byte index, byte callMode = CALL_MODE_DIRECT_CHANGE); | ||||
| @@ -219,7 +241,7 @@ void deletePreset(byte index); | ||||
| bool getPresetName(byte index, String& name); | ||||
|  | ||||
| //remote.cpp | ||||
| void handleRemote(); | ||||
| void handleRemote(uint8_t *data, size_t len); | ||||
|  | ||||
| //set.cpp | ||||
| bool isAsterisksOnly(const char* str, byte maxLen); | ||||
| @@ -235,6 +257,10 @@ void handleNotifications(); | ||||
| void setRealtimePixel(uint16_t i, byte r, byte g, byte b, byte w); | ||||
| void refreshNodeList(); | ||||
| void sendSysInfoUDP(); | ||||
| #ifndef WLED_DISABLE_ESPNOW | ||||
| void espNowSentCB(uint8_t* address, uint8_t status); | ||||
| void espNowReceiveCB(uint8_t* address, uint8_t* data, uint8_t len, signed int rssi, bool broadcast); | ||||
| #endif | ||||
|  | ||||
| //network.cpp | ||||
| int getSignalQuality(int rssi); | ||||
| @@ -338,6 +364,7 @@ void userLoop(); | ||||
| 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 getBoolVal(JsonVariant elem, bool dflt); | ||||
| bool updateVal(const char* req, const char* key, byte* val, byte minv=0, byte maxv=255); | ||||
| bool oappend(const char* txt); // append new c string to temp buffer efficiently | ||||
| bool oappendi(int i);          // append new number to temp buffer efficiently | ||||
| @@ -407,15 +434,11 @@ void handleSerial(); | ||||
| void updateBaudRate(uint32_t rate); | ||||
|  | ||||
| //wled_server.cpp | ||||
| bool isIp(String str); | ||||
| String getFileContentType(String &filename); | ||||
| void createEditHandler(bool enable); | ||||
| bool captivePortal(AsyncWebServerRequest *request); | ||||
| void initServer(); | ||||
| void serveIndexOrWelcome(AsyncWebServerRequest *request); | ||||
| void serveIndex(AsyncWebServerRequest* request); | ||||
| String msgProcessor(const String& var); | ||||
| void serveMessage(AsyncWebServerRequest* request, uint16_t code, const String& headl, const String& subl="", byte optionT=255); | ||||
| String dmxProcessor(const String& var); | ||||
| void serveJsonError(AsyncWebServerRequest* request, uint16_t code, uint16_t error); | ||||
| void serveSettings(AsyncWebServerRequest* request, bool post = false); | ||||
| void serveSettingsJS(AsyncWebServerRequest* request); | ||||
|  | ||||
| @@ -426,7 +449,6 @@ void sendDataWs(AsyncWebSocketClient * client = nullptr); | ||||
|  | ||||
| //xml.cpp | ||||
| void XML_response(AsyncWebServerRequest *request, char* dest = nullptr); | ||||
| void URL_response(AsyncWebServerRequest *request); | ||||
| void getSettingsJS(byte subPage, char* dest); | ||||
|  | ||||
| #endif | ||||
|   | ||||
| @@ -226,7 +226,7 @@ bool appendObjectToFile(const char* key, JsonDocument* content, uint32_t s, uint | ||||
|  | ||||
|   if (pos == 0) //not found | ||||
|   { | ||||
|     DEBUGFS_PRINTLN("not }"); | ||||
|     DEBUGFS_PRINTLN(F("not }")); | ||||
|     f.seek(0); | ||||
|     while (bufferedFind("}",false)) //find last closing bracket in JSON if not last char | ||||
|     { | ||||
| @@ -272,8 +272,8 @@ bool writeObjectToFile(const char* file, const char* key, JsonDocument* content) | ||||
|   #endif | ||||
|  | ||||
|   size_t pos = 0; | ||||
|   f = WLED_FS.open(file, "r+"); | ||||
|   if (!f && !WLED_FS.exists(file)) f = WLED_FS.open(file, "w+"); | ||||
|   char fileName[129]; strncpy_P(fileName, file, 128); fileName[128] = 0; //use PROGMEM safe copy as FS.open() does not | ||||
|   f = WLED_FS.open(fileName, WLED_FS.exists(fileName) ? "r+" : "w+"); | ||||
|   if (!f) { | ||||
|     DEBUGFS_PRINTLN(F("Failed to open!")); | ||||
|     return false; | ||||
| @@ -340,7 +340,8 @@ bool readObjectFromFile(const char* file, const char* key, JsonDocument* dest) | ||||
|     DEBUGFS_PRINTF("Read from %s with key %s >>>\n", file, (key==nullptr)?"nullptr":key); | ||||
|     uint32_t s = millis(); | ||||
|   #endif | ||||
|   f = WLED_FS.open(file, "r"); | ||||
|   char fileName[129]; strncpy_P(fileName, file, 128); fileName[128] = 0; //use PROGMEM safe copy as FS.open() does not | ||||
|   f = WLED_FS.open(fileName, "r"); | ||||
|   if (!f) return false; | ||||
|  | ||||
|   if (key != nullptr && !bufferedFind(key)) //key does not exist in file | ||||
| @@ -374,36 +375,69 @@ void updateFSInfo() { | ||||
|   #endif | ||||
| } | ||||
|  | ||||
| #if defined(BOARD_HAS_PSRAM) && defined(WLED_USE_PSRAM) | ||||
| // caching presets in PSRAM may prevent occasional flashes seen when HomeAssitant polls WLED | ||||
| // original idea by @akaricchi (https://github.com/Akaricchi) | ||||
| // returns a pointer to the PSRAM buffer updates size parameter | ||||
| static const uint8_t *getPresetCache(size_t &size) { | ||||
|   static unsigned long presetsCachedTime; | ||||
|   static uint8_t *presetsCached; | ||||
|   static size_t presetsCachedSize; | ||||
|  | ||||
| //Un-comment any file types you need | ||||
| static String getContentType(AsyncWebServerRequest* request, String filename){ | ||||
|   if(request->hasArg("download")) return "application/octet-stream"; | ||||
|   else if(filename.endsWith(".htm")) return "text/html"; | ||||
|   else if(filename.endsWith(".html")) return "text/html"; | ||||
|   else if(filename.endsWith(".css")) return "text/css"; | ||||
|   else if(filename.endsWith(".js")) return "application/javascript"; | ||||
|   else if(filename.endsWith(".json")) return "application/json"; | ||||
|   else if(filename.endsWith(".png")) return "image/png"; | ||||
|   else if(filename.endsWith(".gif")) return "image/gif"; | ||||
|   else if(filename.endsWith(".jpg")) return "image/jpeg"; | ||||
|   else if(filename.endsWith(".ico")) return "image/x-icon"; | ||||
| //  else if(filename.endsWith(".xml")) return "text/xml"; | ||||
| //  else if(filename.endsWith(".pdf")) return "application/x-pdf"; | ||||
| //  else if(filename.endsWith(".zip")) return "application/x-zip"; | ||||
| //  else if(filename.endsWith(".gz")) return "application/x-gzip"; | ||||
|   return "text/plain"; | ||||
|   if (!psramFound()) { | ||||
|     size = 0; | ||||
|     return nullptr; | ||||
|   } | ||||
|  | ||||
|   if (presetsModifiedTime != presetsCachedTime) { | ||||
|     if (presetsCached) { | ||||
|       free(presetsCached); | ||||
|       presetsCached = nullptr; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (!presetsCached) { | ||||
|     File file = WLED_FS.open(FPSTR(getPresetsFileName()), "r"); | ||||
|     if (file) { | ||||
|       presetsCachedTime = presetsModifiedTime; | ||||
|       presetsCachedSize = 0; | ||||
|       presetsCached = (uint8_t*)ps_malloc(file.size() + 1); | ||||
|       if (presetsCached) { | ||||
|         presetsCachedSize = file.size(); | ||||
|         file.read(presetsCached, presetsCachedSize); | ||||
|         presetsCached[presetsCachedSize] = 0; | ||||
|         file.close(); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   size = presetsCachedSize; | ||||
|   return presetsCached; | ||||
| } | ||||
| #endif | ||||
|  | ||||
| bool handleFileRead(AsyncWebServerRequest* request, String path){ | ||||
|   DEBUG_PRINTLN("WS FileRead: " + path); | ||||
|   DEBUG_PRINT(F("WS FileRead: ")); DEBUG_PRINTLN(path); | ||||
|   if(path.endsWith("/")) path += "index.htm"; | ||||
|   if(path.indexOf("sec") > -1) return false; | ||||
|   String contentType = getContentType(request, path); | ||||
|   if(path.indexOf(F("sec")) > -1) return false; | ||||
|   String contentType = getFileContentType(path); | ||||
|   if(request->hasArg(F("download"))) contentType = F("application/octet-stream"); | ||||
|   /*String pathWithGz = path + ".gz"; | ||||
|   if(WLED_FS.exists(pathWithGz)){ | ||||
|     request->send(WLED_FS, pathWithGz, contentType); | ||||
|     return true; | ||||
|   }*/ | ||||
|   #if defined(BOARD_HAS_PSRAM) && defined(WLED_USE_PSRAM) | ||||
|   if (path.endsWith(FPSTR(getPresetsFileName()))) { | ||||
|     size_t psize; | ||||
|     const uint8_t *presets = getPresetCache(psize); | ||||
|     if (presets) { | ||||
|       AsyncWebServerResponse *response = request->beginResponse_P(200, contentType, presets, psize); | ||||
|       request->send(response); | ||||
|       return true; | ||||
|     } | ||||
|   } | ||||
|   #endif | ||||
|   if(WLED_FS.exists(path)) { | ||||
|     request->send(WLED_FS, path, contentType); | ||||
|     return true; | ||||
|   | ||||
| @@ -1,323 +0,0 @@ | ||||
| /* | ||||
|  * Binary array for the Web UI. | ||||
|  * gzip is used for smaller size and improved speeds. | ||||
|  *  | ||||
|  * Please see https://kno.wled.ge/advanced/custom-features/#changing-web-ui | ||||
|  * to find out how to easily modify the web UI source! | ||||
|  */ | ||||
|   | ||||
| // Autogenerated from wled00/data/cpal/cpal.htm, do not edit!! | ||||
| const uint16_t PAGE_cpal_L = 4970; | ||||
| const uint8_t PAGE_cpal[] PROGMEM = { | ||||
|   0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x13, 0xbd, 0x3b, 0xfd, 0x77, 0xdb, 0x36, | ||||
|   0x92, 0xbf, 0xe7, 0xaf, 0x40, 0x98, 0xd4, 0x21, 0x6b, 0x8a, 0x22, 0x29, 0x5b, 0xb2, 0x25, 0xd1, | ||||
|   0xdd, 0xd4, 0xc9, 0x9e, 0x73, 0xcf, 0x6e, 0xf2, 0x36, 0x3e, 0xb7, 0x3d, 0x9f, 0xf7, 0x99, 0x26, | ||||
|   0x21, 0x89, 0x0d, 0x45, 0x70, 0x41, 0x48, 0xb6, 0x2b, 0xeb, 0x7f, 0xbf, 0x19, 0x00, 0xa4, 0x48, | ||||
|   0x7d, 0x38, 0xc9, 0x75, 0xdf, 0xf5, 0xf9, 0x45, 0x20, 0x30, 0x18, 0x0c, 0x06, 0xf3, 0x89, 0x41, | ||||
|   0x87, 0x2f, 0xdf, 0x7d, 0x3c, 0xbd, 0xfc, 0xfd, 0xd3, 0x7b, 0x32, 0x11, 0xd3, 0xf4, 0x84, 0x0c, | ||||
|   0xcb, 0x1f, 0x1a, 0xc6, 0xf0, 0x33, 0xa5, 0x22, 0x24, 0x59, 0x38, 0xa5, 0x81, 0x31, 0x4f, 0xe8, | ||||
|   0x7d, 0xce, 0xb8, 0x30, 0xc8, 0x8b, 0x88, 0x65, 0x82, 0x66, 0x22, 0x30, 0xee, 0x93, 0x58, 0x4c, | ||||
|   0x82, 0x98, 0xce, 0x93, 0x88, 0xb6, 0xe4, 0x87, 0x9d, 0x64, 0x89, 0x48, 0xc2, 0xb4, 0x55, 0x44, | ||||
|   0x61, 0x4a, 0x03, 0xcf, 0x9e, 0x42, 0xc7, 0x74, 0x36, 0x2d, 0xbf, 0x8d, 0x12, 0xe9, 0x8b, 0x89, | ||||
|   0x10, 0x79, 0x8b, 0xfe, 0x6b, 0x96, 0xcc, 0x03, 0xe3, 0x34, 0x8c, 0x26, 0xb4, 0x75, 0x0a, 0x68, | ||||
|   0x39, 0x4b, 0x0d, 0x52, 0xe1, 0xcf, 0x58, 0x2b, 0xc2, 0x21, 0x9b, 0x40, 0xab, 0x10, 0x8c, 0x43, | ||||
|   0x6b, 0x3a, 0x2b, 0x44, 0x8b, 0xd3, 0x79, 0x98, 0x26, 0x71, 0x28, 0xe8, 0x76, 0x84, 0x9f, 0x78, | ||||
|   0x38, 0x9e, 0x86, 0x5b, 0x30, 0x55, 0xe0, 0x75, 0xe8, 0xf7, 0x0f, 0x79, 0xc2, 0x69, 0x51, 0x03, | ||||
|   0x77, 0x01, 0xee, 0xc5, 0x50, 0x24, 0x22, 0xa5, 0x27, 0xbf, 0x9e, 0xbf, 0x7f, 0x47, 0x4e, 0x61, | ||||
|   0x55, 0x36, 0x25, 0x9f, 0x60, 0x13, 0x42, 0x50, 0xf2, 0x3e, 0x4e, 0x80, 0x9a, 0x61, 0x5b, 0x41, | ||||
|   0x90, 0x61, 0x11, 0xf1, 0x24, 0x17, 0x44, 0x3c, 0xe6, 0xc0, 0x29, 0x41, 0x1f, 0x44, 0xfb, 0x8f, | ||||
|   0x70, 0x1e, 0xaa, 0x5e, 0xe3, 0xe4, 0xc5, 0x68, 0x96, 0x45, 0x22, 0x61, 0x19, 0x19, 0x7f, 0x88, | ||||
|   0x4d, 0x6a, 0x2d, 0x38, 0x15, 0x33, 0x9e, 0x91, 0xd8, 0x19, 0x53, 0xf1, 0x3e, 0xa5, 0x53, 0x58, | ||||
|   0xf3, 0xe7, 0x47, 0x39, 0xb4, 0xac, 0x40, 0xa3, 0xf7, 0x0d, 0xc8, 0x88, 0x53, 0xd8, 0xad, 0x06, | ||||
|   0x46, 0xc0, 0x79, 0xc8, 0x49, 0x1c, 0xc4, 0x2c, 0x9a, 0x61, 0xcf, 0x8b, 0x61, 0x5b, 0xad, 0x86, | ||||
|   0xc4, 0x88, 0x47, 0x20, 0xea, 0xc5, 0x1d, 0x8b, 0x1f, 0x17, 0x23, 0xd8, 0x51, 0x6b, 0x14, 0x4e, | ||||
|   0x93, 0xf4, 0xb1, 0xff, 0x96, 0xc3, 0xc1, 0xd8, 0x45, 0x98, 0x15, 0xad, 0x82, 0xf2, 0x64, 0x34, | ||||
|   0xb8, 0x0b, 0xa3, 0x2f, 0x63, 0xce, 0x66, 0x59, 0xdc, 0x8a, 0x58, 0xca, 0x78, 0xff, 0x95, 0xe7, | ||||
|   0x79, 0x03, 0x39, 0xa5, 0x48, 0xfe, 0xa4, 0x7d, 0xaf, 0x9b, 0x3f, 0x0c, 0xf4, 0x48, 0x1c, 0xc7, | ||||
|   0x83, 0x69, 0xc8, 0xc7, 0x49, 0xd6, 0x77, 0x89, 0xe7, 0xc2, 0x40, 0x9a, 0x64, 0xb4, 0x35, 0xa1, | ||||
|   0xc9, 0x78, 0x22, 0xfa, 0xce, 0xe1, 0xf2, 0x55, 0x1e, 0x72, 0x20, 0xa4, 0x85, 0x3c, 0x0c, 0x61, | ||||
|   0x88, 0x2f, 0x72, 0x56, 0x24, 0xb8, 0x95, 0x3e, 0xa7, 0x69, 0x28, 0x92, 0x39, 0x1d, 0x48, 0x11, | ||||
|   0xe9, 0x7b, 0xae, 0xfb, 0xc3, 0x40, 0x4f, 0xf4, 0x01, 0xd3, 0xf2, 0xd5, 0x1d, 0x13, 0xc0, 0xdd, | ||||
|   0xd3, 0xcd, 0x99, 0xe1, 0x5d, 0xc1, 0xd2, 0x99, 0xa0, 0x7a, 0xe9, 0x96, 0x60, 0x79, 0xff, 0x50, | ||||
|   0x4e, 0x19, 0xf3, 0x30, 0x4e, 0x70, 0xbd, 0x3b, 0xf6, 0xb0, 0xd8, 0xc4, 0x8b, 0xed, 0xa5, 0x23, | ||||
|   0x69, 0x6f, 0xc1, 0xdc, 0x2f, 0x94, 0xdb, 0xfa, 0x2b, 0x4f, 0x22, 0xf8, 0xd2, 0x9d, 0x5b, 0x56, | ||||
|   0xba, 0x63, 0x3c, 0x86, 0x71, 0x44, 0x3f, 0x2b, 0xfa, 0x1d, 0xd8, 0xe8, 0x06, 0x9b, 0x8a, 0x24, | ||||
|   0x9d, 0x53, 0xae, 0x21, 0xfb, 0x7e, 0xfe, 0x40, 0x60, 0x6e, 0x12, 0x13, 0x3e, 0xbe, 0x0b, 0xcd, | ||||
|   0xee, 0x91, 0xad, 0xfe, 0x9c, 0x43, 0x6b, 0xf0, 0x67, 0x2b, 0xc9, 0x62, 0xfa, 0xd0, 0xf7, 0x9b, | ||||
|   0xb4, 0x2c, 0x34, 0x95, 0x1d, 0xe4, 0xa3, 0x22, 0xbe, 0x07, 0x2d, 0xb5, 0xbb, 0x1f, 0x06, 0x82, | ||||
|   0xc3, 0x19, 0x8d, 0x18, 0x9f, 0xf6, 0x65, 0x0b, 0x98, 0x47, 0x7f, 0x37, 0x5b, 0x30, 0x62, 0x01, | ||||
|   0xc8, 0x2c, 0x9a, 0xb4, 0x42, 0x29, 0x22, 0xfd, 0x8c, 0x65, 0x74, 0xb9, 0x75, 0x5b, 0x1a, 0x7f, | ||||
|   0x6f, 0x03, 0xbd, 0x77, 0x88, 0x7c, 0x89, 0x29, 0x88, 0x31, 0xdd, 0xcd, 0x03, 0x3d, 0xfd, 0xb0, | ||||
|   0x9a, 0x8e, 0xad, 0x6f, 0x60, 0xcc, 0xab, 0xd1, 0x68, 0x54, 0xb2, 0xa5, 0x53, 0xb1, 0xe5, 0xd5, | ||||
|   0xf1, 0x9d, 0x7f, 0xe4, 0x1f, 0xc9, 0xf5, 0x7d, 0x1f, 0xf6, 0xb7, 0xc1, 0x15, 0x45, 0xfc, 0x6e, | ||||
|   0x42, 0xbc, 0x8a, 0x10, 0xaf, 0x22, 0x44, 0x36, 0xcb, 0x2d, 0x55, 0x28, 0xbd, 0x92, 0xcc, 0x9a, | ||||
|   0x40, 0x6f, 0x15, 0xf3, 0xa5, 0x73, 0x37, 0x03, 0xa1, 0xcb, 0xa2, 0x34, 0x2c, 0x8a, 0x45, 0x1e, | ||||
|   0xc6, 0x71, 0x92, 0x8d, 0xfb, 0x6e, 0x25, 0xe3, 0x03, 0x38, 0x61, 0x91, 0x80, 0xd1, 0x6a, 0x81, | ||||
|   0xa1, 0x19, 0x67, 0x7d, 0x25, 0xa2, 0x3b, 0x70, 0xad, 0x0b, 0x30, 0x29, 0xf2, 0x30, 0x5b, 0xc4, | ||||
|   0x49, 0x91, 0xa7, 0xe1, 0x63, 0x3f, 0xc9, 0xa4, 0xaa, 0x8c, 0x52, 0xfa, 0x30, 0x90, 0xc8, 0x5a, | ||||
|   0x89, 0xa0, 0xd3, 0xa2, 0x1f, 0x81, 0xf8, 0x82, 0x18, 0xd5, 0x58, 0x57, 0x53, 0x3d, 0x90, 0xaa, | ||||
|   0x75, 0x12, 0xa6, 0x49, 0x1c, 0xa7, 0x74, 0xf9, 0x2a, 0xc9, 0x46, 0xac, 0x42, 0x6e, 0x18, 0x03, | ||||
|   0xb4, 0x37, 0x1a, 0xe4, 0xab, 0x28, 0x37, 0x75, 0xb2, 0xa6, 0x59, 0x1b, 0x6a, 0x0d, 0x5c, 0xba, | ||||
|   0xe7, 0x61, 0x5e, 0xd7, 0xaf, 0xca, 0x06, 0x84, 0x33, 0xc1, 0x96, 0x7f, 0x9b, 0xd2, 0x38, 0x09, | ||||
|   0x89, 0x09, 0x56, 0x5e, 0xd9, 0xff, 0xfe, 0x91, 0x0b, 0x48, 0xac, 0x45, 0x7d, 0x9e, 0xec, 0x5a, | ||||
|   0x2e, 0x9d, 0x5c, 0x19, 0xcf, 0x45, 0x5d, 0xf5, 0xcb, 0xce, 0xff, 0xd0, 0xea, 0x5c, 0x2c, 0x90, | ||||
|   0x4d, 0x70, 0x8c, 0x35, 0xa0, 0x4d, 0xc9, 0xab, 0xa6, 0x15, 0x17, 0xc0, 0xef, 0xc5, 0x9a, 0x71, | ||||
|   0xa8, 0x99, 0x19, 0x09, 0x78, 0xc9, 0xf2, 0x72, 0xcd, 0x51, 0xa2, 0x0c, 0x14, 0xac, 0xf4, 0x17, | ||||
|   0xd9, 0xb6, 0xc6, 0x26, 0x58, 0xa6, 0xdc, 0xc2, 0x27, 0x69, 0x07, 0xab, 0x03, 0xda, 0x75, 0xec, | ||||
|   0x5b, 0x28, 0x5a, 0x3f, 0x8a, 0x7f, 0x2b, 0x85, 0x4a, 0xdc, 0x8b, 0x77, 0xc9, 0x7c, 0xab, 0x60, | ||||
|   0xea, 0xb5, 0x53, 0x3a, 0x6a, 0xe8, 0xbd, 0x3c, 0x23, 0x38, 0x63, 0xf1, 0x19, 0x24, 0xda, 0x76, | ||||
|   0x0a, 0x9a, 0xc5, 0xd8, 0x5a, 0x44, 0x33, 0x5e, 0x00, 0x25, 0x39, 0x4b, 0x90, 0xae, 0xe5, 0xc4, | ||||
|   0x5b, 0xd4, 0xe8, 0x71, 0xba, 0x9c, 0x4e, 0x97, 0xe8, 0x92, 0xa4, 0x27, 0x22, 0xc3, 0xb6, 0x0e, | ||||
|   0x27, 0xd0, 0x25, 0xc1, 0x4f, 0x9c, 0xcc, 0x49, 0x12, 0x43, 0xf8, 0x00, 0x32, 0x02, 0x4e, 0x17, | ||||
|   0x35, 0x50, 0x7f, 0xe8, 0xc1, 0x17, 0x72, 0x62, 0x60, 0x34, 0x78, 0xf8, 0x07, 0xb8, 0xe0, 0x64, | ||||
|   0xf4, 0x58, 0x72, 0x4b, 0xb3, 0x04, 0xa7, 0x4c, 0xbc, 0xed, 0x33, 0x36, 0xb9, 0x8e, 0xd0, 0xc5, | ||||
|   0x7c, 0x5c, 0x81, 0xab, 0x5d, 0x76, 0xd0, 0xcf, 0x95, 0x26, 0x19, 0xdb, 0x9a, 0x19, 0x5c, 0xf6, | ||||
|   0x40, 0x87, 0x41, 0x30, 0xfa, 0xf9, 0x99, 0x3d, 0x40, 0x64, 0x40, 0x5c, 0xd2, 0xf1, 0xe1, 0xcf, | ||||
|   0x38, 0x19, 0xe6, 0xa1, 0x98, 0x90, 0x17, 0xa3, 0x24, 0x4d, 0x03, 0xe3, 0x95, 0xeb, 0x76, 0xe0, | ||||
|   0x58, 0x0c, 0xf0, 0xc9, 0xc6, 0x45, 0x97, 0xf8, 0xfe, 0xe4, 0x68, 0x7e, 0x70, 0xd6, 0xfd, 0xf3, | ||||
|   0xc2, 0x3b, 0x20, 0xde, 0xc1, 0xe4, 0x60, 0x7e, 0x34, 0x69, 0x1d, 0xc0, 0xd7, 0x11, 0x38, 0xcf, | ||||
|   0xea, 0xcb, 0xf7, 0x49, 0x17, 0xe1, 0x26, 0xad, 0xa3, 0x3f, 0x8d, 0xf6, 0x09, 0x30, 0x6c, 0x3e, | ||||
|   0x3e, 0x79, 0x01, 0x24, 0x02, 0x8b, 0x25, 0x87, 0x90, 0x6f, 0xc6, 0xb3, 0x11, 0x08, 0x82, 0x4a, | ||||
|   0x0e, 0x7b, 0xf8, 0x2f, 0x30, 0xaf, 0x64, 0x21, 0x4e, 0x5f, 0x77, 0xc9, 0x46, 0x8d, 0xf9, 0x75, | ||||
|   0x07, 0x0a, 0x7b, 0xd1, 0x53, 0xeb, 0x18, 0xbe, 0xef, 0x10, 0x4a, 0xbc, 0xa5, 0x66, 0x62, 0x9c, | ||||
|   0xa8, 0x4e, 0xb6, 0xae, 0xab, 0x6b, 0x90, 0xa0, 0x9a, 0x95, 0x00, 0xe8, 0x4f, 0xd8, 0xff, 0xe9, | ||||
|   0x8c, 0x23, 0xdd, 0xe9, 0x23, 0x49, 0x32, 0x32, 0x2b, 0x28, 0x89, 0xd4, 0xde, 0x4b, 0x44, 0x64, | ||||
|   0x8d, 0xda, 0xbf, 0x4e, 0x34, 0x9a, 0x54, 0xb9, 0x72, 0x0a, 0x9e, 0x88, 0x40, 0xf4, 0x25, 0x26, | ||||
|   0x94, 0x94, 0x1c, 0x22, 0x54, 0xf2, 0x9a, 0x08, 0x46, 0xc0, 0x4d, 0x90, 0x8c, 0xde, 0x13, 0xa9, | ||||
|   0x87, 0xa4, 0x00, 0xef, 0x06, 0x81, 0x05, 0x02, 0xab, 0x19, 0xb2, 0x9b, 0xc6, 0x04, 0x58, 0x4a, | ||||
|   0xee, 0x68, 0xca, 0xee, 0x65, 0xaf, 0x02, 0xc3, 0xe9, 0xd1, 0x24, 0xcc, 0xc6, 0x94, 0x24, 0xa2, | ||||
|   0x50, 0xa0, 0x8e, 0x5e, 0x10, 0xa1, 0x9a, 0xf3, 0xc0, 0x9b, 0x81, 0xe5, 0xc7, 0x55, 0xcd, 0x30, | ||||
|   0x8b, 0x31, 0x30, 0x1d, 0x25, 0x7c, 0x6a, 0x21, 0x12, 0xe5, 0xbc, 0x1d, 0xf2, 0x31, 0x8b, 0x28, | ||||
|   0x19, 0x41, 0x78, 0x5d, 0x4c, 0x68, 0x6c, 0x03, 0x17, 0x4b, 0x4c, 0x21, 0xe7, 0x88, 0x21, 0xc2, | ||||
|   0x6d, 0x30, 0x32, 0xcb, 0x53, 0x16, 0xc6, 0x80, 0x10, 0xda, 0x38, 0x1a, 0xd3, 0x22, 0xc1, 0xb5, | ||||
|   0x8a, 0x94, 0x09, 0x87, 0x5c, 0x32, 0xb9, 0x3b, 0x42, 0x1f, 0x12, 0xe0, 0x51, 0x36, 0x2e, 0x79, | ||||
|   0x5c, 0xc7, 0x97, 0xd3, 0x2c, 0x4a, 0x52, 0x89, 0xd0, 0x81, 0xa8, 0x78, 0x93, 0xe9, 0xdf, 0xcf, | ||||
|   0x73, 0x29, 0x9d, 0x85, 0x00, 0x43, 0x15, 0x7d, 0xaa, 0xe4, 0xe5, 0x2b, 0xe2, 0x82, 0xe0, 0x3b, | ||||
|   0x45, 0xe6, 0xed, 0x3c, 0x4c, 0xd2, 0xf0, 0x2e, 0x05, 0x6e, 0x4b, 0xac, 0x5f, 0x93, 0x15, 0xf9, | ||||
|   0x33, 0x6c, 0x6b, 0x83, 0xa4, 0xc3, 0xf7, 0x17, 0xbb, 0xe2, 0x77, 0x8c, 0xb5, 0x4b, 0x69, 0x40, | ||||
|   0x2b, 0x80, 0x61, 0x7c, 0x53, 0x81, 0x2c, 0x3b, 0x82, 0x15, 0xa3, 0xa0, 0xe5, 0xd9, 0xf9, 0xc3, | ||||
|   0x29, 0x4b, 0x83, 0xc5, 0xd2, 0x16, 0xfa, 0x97, 0xd3, 0x48, 0x04, 0xb5, 0xe9, 0x18, 0xf5, 0xff, | ||||
|   0x8c, 0x21, 0x04, 0xf0, 0x1b, 0xce, 0x1f, 0x3a, 0xff, 0x01, 0x10, 0xa6, 0x65, 0x97, 0x30, 0xe7, | ||||
|   0x34, 0x1b, 0x43, 0x5e, 0x85, 0xf3, 0x1c, 0x95, 0x55, 0x4d, 0x3f, 0x8e, 0x46, 0x45, 0x70, 0x01, | ||||
|   0xf6, 0xc6, 0x91, 0xc1, 0x87, 0xd9, 0x04, 0x6d, 0xfb, 0x87, 0xdd, 0xb6, 0x6f, 0xb5, 0x0e, 0x6d, | ||||
|   0xbd, 0xed, 0xb7, 0x9c, 0x87, 0x8f, 0xc1, 0xf5, 0x4d, 0xf9, 0xfd, 0x0b, 0xa6, 0x70, 0xf0, 0x09, | ||||
|   0xf6, 0xe5, 0x73, 0x38, 0xa7, 0xc1, 0x1b, 0x69, 0x05, 0x1b, 0x46, 0xd0, 0x3f, 0x5c, 0x19, 0x41, | ||||
|   0x6c, 0xaf, 0xd9, 0x3c, 0xff, 0x00, 0xfe, 0x4a, 0x9b, 0x27, 0x4d, 0x1e, 0x7a, 0x21, 0x69, 0xed, | ||||
|   0x7c, 0xdf, 0xf6, 0xfc, 0xb7, 0x9e, 0x6b, 0x7b, 0x08, 0x08, 0x3f, 0xc4, 0xf3, 0x6d, 0xbf, 0xd9, | ||||
|   0xb3, 0x15, 0xa4, 0x09, 0x81, 0x20, 0x17, 0x3d, 0xf8, 0xe7, 0x1c, 0xc6, 0xbc, 0xde, 0x95, 0x77, | ||||
|   0x70, 0xe6, 0x75, 0xaf, 0x3c, 0xf7, 0xcc, 0xf3, 0xaf, 0x7a, 0xe7, 0x38, 0xf0, 0xdf, 0x95, 0x8d, | ||||
|   0x7c, 0x83, 0x3b, 0x41, 0x13, 0xf8, 0xef, 0xdd, 0x09, 0x12, 0x75, 0xda, 0x75, 0x0e, 0x7a, 0xb6, | ||||
|   0x0f, 0x14, 0x63, 0x43, 0x12, 0x7e, 0x8a, 0xf4, 0x38, 0x87, 0x1d, 0xa2, 0x86, 0x7c, 0xb5, 0xbf, | ||||
|   0x53, 0xd9, 0x87, 0x9f, 0x7e, 0x39, 0xee, 0x2b, 0x68, 0x3d, 0x55, 0x8f, 0x4b, 0xe8, 0x0b, 0xef, | ||||
|   0xd0, 0xf1, 0xec, 0x9e, 0xe3, 0xf6, 0x4e, 0xa1, 0xe5, 0x1f, 0xc8, 0x26, 0x81, 0x66, 0xe7, 0x08, | ||||
|   0x9a, 0x9e, 0x8f, 0xcd, 0x43, 0x68, 0xf9, 0x9d, 0x73, 0xaf, 0xeb, 0xf4, 0x7a, 0xf6, 0x91, 0x73, | ||||
|   0x08, 0x0b, 0xc0, 0x4f, 0x0f, 0xc6, 0x7a, 0xf6, 0xb1, 0x04, 0x97, 0x23, 0xc7, 0x8e, 0x7f, 0x74, | ||||
|   0x0e, 0xe0, 0xd0, 0xf4, 0x5c, 0xd9, 0xee, 0x00, 0x10, 0x40, 0xe2, 0xdc, 0x03, 0x6c, 0x22, 0x9a, | ||||
|   0x53, 0x68, 0x1e, 0xf9, 0x1a, 0xf7, 0x81, 0x73, 0xdc, 0xad, 0x56, 0x54, 0x64, 0x5c, 0xc0, 0x2c, | ||||
|   0xaf, 0x03, 0xb3, 0x8e, 0x3c, 0x44, 0xe6, 0x1d, 0x23, 0xb2, 0xa3, 0xde, 0xf9, 0x31, 0xf6, 0xc2, | ||||
|   0x42, 0xc7, 0x9d, 0x33, 0x04, 0xbb, 0x42, 0x34, 0xbd, 0xf3, 0x15, 0x70, 0xed, 0x0c, 0x06, 0x55, | ||||
|   0xae, 0x0a, 0x92, 0xfa, 0x71, 0x64, 0x62, 0xb6, 0xfa, 0xff, 0x26, 0xe9, 0xb5, 0x44, 0x39, 0x4d, | ||||
|   0xbe, 0x7c, 0xcc, 0xca, 0xe8, 0x4b, 0x25, 0xcd, 0x53, 0x36, 0xa7, 0x97, 0x3c, 0x2c, 0x26, 0x51, | ||||
|   0x98, 0x41, 0x8f, 0x0d, 0x76, 0xfb, 0xd4, 0xac, 0x21, 0xa5, 0x0e, 0x83, 0x65, 0xa8, 0xf8, 0xad, | ||||
|   0xdd, 0x44, 0xff, 0x23, 0xa0, 0xb7, 0x6a, 0x49, 0xb8, 0x9c, 0x47, 0x6d, 0x11, 0x18, 0x86, 0xb5, | ||||
|   0x00, 0x4d, 0x22, 0x1c, 0x35, 0x9c, 0x05, 0x2f, 0x3d, 0x08, 0xc5, 0xb2, 0x42, 0x90, 0xb0, 0xb1, | ||||
|   0xdd, 0x7f, 0xcd, 0x28, 0x7f, 0xfc, 0x0c, 0xf6, 0x39, 0x02, 0xcb, 0xfd, 0x36, 0x4d, 0x4d, 0xa3, | ||||
|   0x91, 0xf6, 0x19, 0xd6, 0x20, 0x19, 0x99, 0xa1, 0x03, 0xa9, 0xdd, 0xfb, 0x30, 0x9a, 0x98, 0xa6, | ||||
|   0xb0, 0xb9, 0x15, 0x9c, 0x2c, 0x04, 0xf2, 0xe9, 0xad, 0x10, 0x3c, 0x81, 0x20, 0x8d, 0x9a, 0x46, | ||||
|   0x1c, 0x8a, 0xb0, 0x25, 0xf8, 0x8c, 0x42, 0x50, 0x67, 0x58, 0x41, 0x40, 0xf7, 0xf6, 0x4c, 0x58, | ||||
|   0xd3, 0xb5, 0x96, 0xb0, 0x13, 0x27, 0x95, 0x94, 0x9e, 0x78, 0xbd, 0xb2, 0xd7, 0x66, 0x96, 0xba, | ||||
|   0x26, 0x40, 0xec, 0xf4, 0xc4, 0xdd, 0xdb, 0xa3, 0x43, 0xff, 0xf0, 0xd0, 0x82, 0x65, 0x4c, 0xb4, | ||||
|   0x5c, 0x59, 0xe0, 0x0d, 0xb2, 0x61, 0xe0, 0x75, 0xf7, 0xf6, 0xf8, 0x10, 0x9a, 0xfb, 0xfb, 0x96, | ||||
|   0x34, 0x60, 0x92, 0xb4, 0x0b, 0x45, 0xd9, 0x7e, 0x66, 0x3d, 0x3d, 0x99, 0x3c, 0xc8, 0xac, 0x01, | ||||
|   0x4d, 0xc1, 0xe3, 0xf2, 0x80, 0x0e, 0x0c, 0x23, 0x08, 0x04, 0x2c, 0x02, 0xbb, 0x7f, 0x65, 0xec, | ||||
|   0x9b, 0x5e, 0xb7, 0xd7, 0xeb, 0xf9, 0xde, 0xe1, 0x8f, 0x8a, 0x8f, 0xe0, 0x96, 0xd8, 0xd4, 0xb4, | ||||
|   0x86, 0x43, 0xd7, 0x72, 0x04, 0xfb, 0x0c, 0xc4, 0x67, 0x63, 0x80, 0xb1, 0x20, 0x14, 0x8e, 0x3f, | ||||
|   0x8b, 0x90, 0x0b, 0xb3, 0x6b, 0x1b, 0xae, 0x61, 0x59, 0x9a, 0x53, 0x69, 0x10, 0xbd, 0x37, 0x0d, | ||||
|   0x0c, 0x57, 0x80, 0x0d, 0xa9, 0x23, 0x2d, 0xb8, 0x34, 0x4b, 0x46, 0x83, 0x45, 0x76, 0xea, 0xa0, | ||||
|   0xb1, 0x6f, 0xd0, 0xc6, 0x57, 0x0b, 0x58, 0x30, 0x5e, 0xec, 0x66, 0x96, 0x4d, 0x9f, 0x01, 0x00, | ||||
|   0x9c, 0x86, 0x2d, 0x76, 0x00, 0x28, 0x79, 0x30, 0x94, 0xfc, 0x21, 0x0c, 0x1c, 0xfd, 0xfb, 0x39, | ||||
|   0x0a, 0x06, 0x38, 0x46, 0x0a, 0xb1, 0x13, 0xf0, 0x0b, 0x3d, 0xa2, 0x61, 0x43, 0x28, 0x92, 0xff, | ||||
|   0x7d, 0xc6, 0xc1, 0x33, 0xf2, 0x4f, 0x9c, 0xe5, 0x12, 0x1f, 0x9a, 0x1f, 0x07, 0x63, 0xe7, 0xe7, | ||||
|   0x25, 0xf7, 0x47, 0x6a, 0xed, 0xcb, 0x05, 0xf6, 0x0d, 0x30, 0x4b, 0x9a, 0x31, 0x89, 0x64, 0x4c, | ||||
|   0x92, 0xe5, 0x33, 0x81, 0x02, 0xe2, 0x28, 0x27, 0x24, 0x19, 0x60, 0xd8, 0x89, 0x33, 0x0f, 0xd3, | ||||
|   0x19, 0x0d, 0x04, 0xb4, 0x36, 0x58, 0xa6, 0xd2, 0x66, 0x04, 0xaa, 0x58, 0xf6, 0x49, 0x75, 0x35, | ||||
|   0x59, 0x96, 0x6c, 0xd9, 0x8c, 0x5a, 0xcf, 0x9e, 0xe5, 0x78, 0x89, 0x56, 0x2a, 0xcf, 0x76, 0x50, | ||||
|   0xbd, 0xef, 0x28, 0x3f, 0x4d, 0xbf, 0x94, 0xa7, 0x59, 0xd4, 0x4f, 0xb3, 0xd8, 0x45, 0x5a, 0x75, | ||||
|   0xa8, 0xc5, 0x3a, 0x85, 0x5b, 0x8f, 0xb6, 0x78, 0x66, 0x71, 0x96, 0xe2, 0xea, 0x00, 0x52, 0xe3, | ||||
|   0x75, 0x9d, 0xf1, 0x40, 0xf9, 0x8e, 0x11, 0x4d, 0x71, 0x5c, 0xa7, 0x98, 0xa3, 0x96, 0x70, 0xd4, | ||||
|   0x12, 0x90, 0xef, 0xb8, 0x4e, 0x7e, 0xe3, 0x6a, 0xc4, 0xb0, 0x63, 0x49, 0xb8, 0xea, 0xdc, 0x4a, | ||||
|   0x73, 0xbc, 0x9b, 0x66, 0x0a, 0xaa, 0xad, 0x66, 0x9e, 0xe2, 0xc6, 0xf1, 0xd2, 0x0e, 0xe1, 0x77, | ||||
|   0x90, 0xb9, 0x92, 0xa3, 0xd5, 0x1d, 0x83, 0x9c, 0x17, 0x68, 0x19, 0xa8, 0xf6, 0xbe, 0x6b, 0xbc, | ||||
|   0x6e, 0x89, 0xc2, 0x1c, 0xa2, 0xb6, 0xf8, 0x74, 0x92, 0xa4, 0xb1, 0x99, 0x58, 0x3b, 0x87, 0xd2, | ||||
|   0xdd, 0x43, 0xa0, 0x04, 0xee, 0xcb, 0x80, 0xef, 0xed, 0x01, 0x93, 0xe4, 0xef, 0x2e, 0xc0, 0xd8, | ||||
|   0xb2, 0xeb, 0xec, 0x9c, 0x86, 0x5f, 0xe8, 0x05, 0x7d, 0xc7, 0xc3, 0xb1, 0x89, 0x56, 0x06, 0xd5, | ||||
|   0xd9, 0x82, 0x73, 0xa3, 0xe2, 0x92, 0xb1, 0x54, 0x24, 0xb9, 0xe2, 0x62, 0x7d, 0xac, 0x29, 0x83, | ||||
|   0x66, 0xcd, 0xfc, 0xae, 0x8f, 0x2c, 0xd4, 0x51, 0xd2, 0xef, 0x34, 0xba, 0x1b, 0x11, 0x19, 0xdd, | ||||
|   0x30, 0xc1, 0x0a, 0x31, 0x93, 0x91, 0x1d, 0xbd, 0xe6, 0x37, 0x40, 0x99, 0xc3, 0x29, 0x84, 0xb3, | ||||
|   0x11, 0x6d, 0x1a, 0x4a, 0xbb, 0xa1, 0x67, 0x96, 0xa5, 0x78, 0x3f, 0xf8, 0xbe, 0x79, 0xba, 0x0f, | ||||
|   0x66, 0x6f, 0x3f, 0x51, 0x66, 0x4b, 0x5c, 0xcf, 0x0f, 0x3e, 0x63, 0xe4, 0x98, 0x55, 0xb9, 0x27, | ||||
|   0x09, 0xfb, 0x9c, 0x7b, 0xb1, 0xb3, 0xaf, 0xd8, 0xac, 0x50, 0x33, 0xf0, 0x3a, 0xbb, 0x81, 0xb5, | ||||
|   0x91, 0x85, 0xd7, 0x21, 0xb4, 0x96, 0x2b, 0xd1, 0x51, 0xca, 0x10, 0x18, 0x78, 0x9b, 0x10, 0xf2, | ||||
|   0x56, 0xd9, 0x6d, 0x42, 0xc6, 0x21, 0x53, 0x68, 0xc3, 0xfe, 0x78, 0xf7, 0x07, 0xba, 0x78, 0xe8, | ||||
|   0xe4, 0x09, 0x2d, 0x4c, 0x89, 0xcf, 0x5a, 0x1d, 0xc2, 0x35, 0xb8, 0xd8, 0x1b, 0x3c, 0x86, 0x26, | ||||
|   0xc6, 0xfd, 0xe0, 0xd6, 0x26, 0xaf, 0x17, 0x62, 0x09, 0xff, 0xd0, 0x65, 0xfe, 0x70, 0xbb, 0xb1, | ||||
|   0xe6, 0x7e, 0x60, 0x58, 0x46, 0x43, 0x84, 0xd7, 0x79, 0x16, 0x34, 0x27, 0xac, 0x64, 0x6b, 0xcd, | ||||
|   0x8c, 0x63, 0xdc, 0x40, 0x1d, 0xec, 0xc4, 0xaf, 0x70, 0x1c, 0x22, 0x50, 0x5d, 0x16, 0x95, 0x05, | ||||
|   0xda, 0x1e, 0x5e, 0x6c, 0x99, 0x68, 0x4b, 0x89, 0x70, 0x0a, 0x1e, 0xe9, 0x3b, 0xfb, 0x86, 0x64, | ||||
|   0x54, 0x42, 0x81, 0x32, 0x20, 0xcd, 0x45, 0x63, 0xa9, 0x7c, 0xc7, 0x4a, 0x73, 0x69, 0xa3, 0x9f, | ||||
|   0x25, 0xb3, 0xa6, 0x7e, 0x80, 0x00, 0xdd, 0xbf, 0x08, 0x5c, 0x9b, 0x7f, 0x53, 0x54, 0xc6, 0x02, | ||||
|   0xee, 0xc8, 0x03, 0xb3, 0x43, 0x68, 0x49, 0xab, 0x9a, 0x05, 0xac, 0x15, 0xee, 0x7b, 0xab, 0x50, | ||||
|   0x2f, 0xdd, 0x49, 0xd7, 0x00, 0x17, 0xe3, 0x81, 0x49, 0x03, 0xfa, 0xf4, 0x74, 0x0f, 0x09, 0x2c, | ||||
|   0xbb, 0x77, 0xd4, 0x88, 0x74, 0x69, 0x40, 0x34, 0x04, 0x05, 0xc5, 0xaf, 0x89, 0x98, 0x98, 0x86, | ||||
|   0xbc, 0xc6, 0x46, 0x3b, 0xfc, 0xf4, 0x44, 0x9d, 0x9c, 0x4b, 0xb0, 0x77, 0x74, 0x14, 0xce, 0x52, | ||||
|   0xa4, 0x43, 0x04, 0xfc, 0x27, 0xea, 0x48, 0x18, 0x5a, 0x5c, 0xbb, 0x37, 0xc8, 0x21, 0x00, 0xf8, | ||||
|   0xad, 0x4f, 0xcb, 0x16, 0x98, 0x51, 0x96, 0x4d, 0xd9, 0xac, 0xa0, 0xb3, 0x3c, 0x28, 0xe4, 0x97, | ||||
|   0x04, 0x07, 0x6a, 0x22, 0x9a, 0xd6, 0x7b, 0xc0, 0x4c, 0xe9, 0x4f, 0x09, 0x8e, 0x74, 0x07, 0xc9, | ||||
|   0x6a, 0x58, 0x7d, 0xaf, 0xd8, 0x97, 0x98, 0x5c, 0x71, 0x8d, 0x05, 0x10, 0x0d, 0xf1, 0x6f, 0xde, | ||||
|   0x08, 0x7b, 0x7a, 0xe2, 0x1b, 0x1b, 0x91, 0x1c, 0x49, 0x03, 0xf6, 0x13, 0xdf, 0xb6, 0x19, 0x5e, | ||||
|   0xb6, 0x06, 0xa2, 0x95, 0xc2, 0x9e, 0x53, 0x5b, 0x52, 0xf8, 0x89, 0x15, 0x1f, 0xaa, 0x68, 0x36, | ||||
|   0x10, 0x2d, 0x13, 0xd8, 0x0f, 0x2c, 0x01, 0x75, 0x85, 0x91, 0xba, 0xaa, 0x6e, 0x42, 0xb7, 0x33, | ||||
|   0x19, 0xbf, 0xda, 0x2c, 0x8d, 0x2f, 0x35, 0x3c, 0x7d, 0x5e, 0xf7, 0x35, 0x5a, 0x34, 0xdf, 0xba, | ||||
|   0xa9, 0x8c, 0xf8, 0x0a, 0xc3, 0xcb, 0x40, 0x0f, 0x60, 0x20, 0xa8, 0x41, 0xba, 0x07, 0x3f, 0x89, | ||||
|   0x49, 0x52, 0x7c, 0x94, 0x21, 0x53, 0xe0, 0xf6, 0x4b, 0x2c, 0xde, 0xb1, 0x5f, 0x1f, 0xe8, 0xf5, | ||||
|   0x6b, 0x1f, 0x1d, 0x29, 0xb6, 0xdb, 0xc2, 0xa4, 0x4c, 0x5a, 0x19, 0x8d, 0xa3, 0x1e, 0x1f, 0x69, | ||||
|   0x15, 0xfa, 0x3f, 0x59, 0x54, 0xb9, 0x48, 0x7d, 0xc5, 0xaf, 0x22, 0x6b, 0xf8, 0xf8, 0xbf, 0x80, | ||||
|   0x67, 0xcd, 0x3d, 0xec, 0x42, 0x43, 0x9f, 0x0d, 0x63, 0x4b, 0x6e, 0x6c, 0xfa, 0x4d, 0xba, 0xe9, | ||||
|   0x2f, 0x6b, 0xda, 0x5f, 0x80, 0x8f, 0xac, 0xab, 0x46, 0x36, 0x4b, 0xd3, 0x0d, 0xed, 0x68, 0x76, | ||||
|   0xa2, 0x82, 0x54, 0x3d, 0x2b, 0x1d, 0x69, 0x02, 0x55, 0x5d, 0x4b, 0x5a, 0x42, 0x81, 0x56, 0x64, | ||||
|   0x20, 0xb3, 0xb4, 0x04, 0x91, 0x5a, 0x11, 0xa4, 0x35, 0x5a, 0x36, 0x49, 0x5f, 0xac, 0xef, 0x5a, | ||||
|   0xd6, 0x4f, 0x0d, 0xfb, 0x16, 0xec, 0xfb, 0xb3, 0x82, 0xba, 0x24, 0x7d, 0xf2, 0x0c, 0x0c, 0x3a, | ||||
|   0x3d, 0x6b, 0x79, 0x5b, 0x63, 0x44, 0x33, 0xee, 0xd2, 0x76, 0x10, 0x03, 0xc0, 0x38, 0x99, 0x83, | ||||
|   0xba, 0xa2, 0x60, 0xbe, 0xab, 0x1d, 0x78, 0x50, 0xb7, 0xd4, 0x36, 0x8e, 0x9e, 0xae, 0x0e, 0x55, | ||||
|   0x06, 0x05, 0xeb, 0x33, 0x1a, 0x12, 0xa0, 0x56, 0x2b, 0x0f, 0x1f, 0x8e, 0x7d, 0x85, 0xa2, 0x2e, | ||||
|   0x9e, 0xdf, 0x8b, 0xa8, 0x92, 0xa2, 0x75, 0x74, 0xdf, 0x8a, 0x68, 0x87, 0x50, 0xda, 0x1c, 0x0e, | ||||
|   0x9d, 0x72, 0xa5, 0x99, 0xbf, 0x05, 0x9e, 0xab, 0x3b, 0x7e, 0x6b, 0xb0, 0x61, 0x97, 0x9b, 0x70, | ||||
|   0x1e, 0x5a, 0x8d, 0xf9, 0x7a, 0xf2, 0xef, 0xdf, 0x36, 0xf9, 0x71, 0xdf, 0xeb, 0xd8, 0x42, 0x46, | ||||
|   0xd5, 0x02, 0xfd, 0x87, 0x81, 0x1f, 0x19, 0x84, 0xcd, 0x67, 0x97, 0x17, 0xe7, 0xfa, 0x36, 0x67, | ||||
|   0xcb, 0x75, 0x0d, 0x79, 0x98, 0xa6, 0x59, 0x11, 0x18, 0x58, 0xb6, 0xef, 0xb7, 0xdb, 0xf7, 0xf7, | ||||
|   0xf7, 0xce, 0x7d, 0xc7, 0x61, 0x7c, 0xdc, 0xf6, 0x5d, 0xd7, 0xc5, 0xfb, 0x08, 0x83, 0xa8, 0xd7, | ||||
|   0x08, 0x06, 0x16, 0x55, 0x0d, 0xa2, 0xee, 0x7f, 0xf4, 0x97, 0xbe, 0xec, 0xd1, 0xb7, 0x44, 0x78, | ||||
|   0xe7, 0xd3, 0x7f, 0x75, 0x74, 0x04, 0x13, 0xdd, 0x01, 0x74, 0x72, 0xf6, 0x85, 0xf6, 0x09, 0x74, | ||||
|   0xe0, 0x7f, 0x65, 0x87, 0x2e, 0x67, 0x91, 0x16, 0x16, 0x57, 0x74, 0x57, 0x0c, 0xf4, 0x86, 0x78, | ||||
|   0xb3, 0xd6, 0x27, 0xae, 0xe3, 0xd9, 0xe4, 0x68, 0xa0, 0xae, 0xfb, 0x8f, 0xed, 0xce, 0xd5, 0xc1, | ||||
|   0xd9, 0xc1, 0x55, 0xf7, 0xec, 0xf0, 0xca, 0x3b, 0x7e, 0xeb, 0xdb, 0xbe, 0xbc, 0xd3, 0x72, 0x49, | ||||
|   0xcf, 0xf6, 0xbd, 0x33, 0xaf, 0x57, 0xeb, 0xc1, 0x7b, 0x96, 0x63, 0x00, 0xf4, 0x5d, 0x98, 0xe1, | ||||
|   0x1d, 0x5e, 0x75, 0xce, 0x8e, 0x2f, 0x7a, 0x76, 0xf7, 0x0c, 0xef, 0xbb, 0x8e, 0xcf, 0x7a, 0x57, | ||||
|   0x5d, 0x40, 0x76, 0x74, 0xe5, 0xf5, 0xce, 0x3c, 0xef, 0xea, 0x08, 0xc6, 0xf0, 0xd6, 0x45, 0x7e, | ||||
|   0x1e, 0xc2, 0xa7, 0xd7, 0xa9, 0xdf, 0x80, 0x09, 0x6d, 0x4e, 0xca, 0xca, 0x4f, 0x60, 0x94, 0x65, | ||||
|   0x53, 0xa3, 0x1a, 0x93, 0x76, 0x47, 0x1f, 0xae, 0xb2, 0xa9, 0xe5, 0x08, 0x44, 0x10, 0x7a, 0xe0, | ||||
|   0x77, 0x35, 0x10, 0x3b, 0x78, 0x19, 0xda, 0x88, 0xec, 0x21, 0xd6, 0x10, 0xcf, 0xa7, 0x37, 0xc2, | ||||
|   0x51, 0x25, 0x86, 0x5f, 0x58, 0x4c, 0x1d, 0x15, 0x15, 0xac, 0xa6, 0xae, 0xcb, 0xe7, 0x2e, 0xd0, | ||||
|   0x35, 0xb8, 0x1d, 0xca, 0xf3, 0xdc, 0xf4, 0x0d, 0x60, 0x6b, 0x5d, 0x87, 0xbf, 0x3a, 0x7b, 0xc7, | ||||
|   0xda, 0xdf, 0xb8, 0xea, 0x96, 0x24, 0x66, 0x7b, 0x54, 0xf8, 0x4c, 0xba, 0xd8, 0x0c, 0xaa, 0xbe, | ||||
|   0x12, 0xe6, 0x6d, 0xc4, 0xa0, 0x0b, 0xa9, 0x4d, 0xea, 0x66, 0x5a, 0x29, 0x16, 0x62, 0x00, 0x9b, | ||||
|   0x0c, 0x8a, 0x88, 0x3e, 0x1c, 0x7a, 0xd0, 0xc7, 0xcb, 0x1f, 0x53, 0xfe, 0xee, 0xdc, 0x1a, 0x0e, | ||||
|   0x22, 0xa5, 0xaa, 0xf3, 0x9b, 0x88, 0xad, 0x07, 0xaf, 0x93, 0x2f, 0xbf, 0xd6, 0x33, 0x35, 0x24, | ||||
|   0x48, 0xd6, 0xf6, 0x30, 0xae, 0x93, 0x5f, 0xb2, 0xa8, 0x65, 0x0d, 0xca, 0x3b, 0xb8, 0x5f, 0x51, | ||||
|   0xd1, 0x86, 0x5d, 0xd7, 0xfd, 0xa9, 0x94, 0x4d, 0x5d, 0x48, 0xc0, 0x57, 0x3b, 0x19, 0x35, 0xfa, | ||||
|   0x1b, 0xdd, 0xaa, 0x6e, 0x69, 0xd4, 0xd6, 0x0c, 0xd3, 0xe8, 0x3f, 0x3f, 0x7f, 0xfc, 0xc5, 0x54, | ||||
|   0x97, 0x74, 0x34, 0x78, 0xb3, 0x28, 0xcb, 0x08, 0x46, 0xff, 0xfa, 0xcd, 0x40, 0xbf, 0xa2, 0x59, | ||||
|   0xcb, 0x42, 0xc4, 0x5a, 0x12, 0x02, 0xa9, 0xa0, 0x4c, 0x42, 0x04, 0x86, 0x43, 0x26, 0x85, 0xdc, | ||||
|   0xc2, 0x46, 0x26, 0x42, 0x16, 0x82, 0x39, 0x88, 0x6d, 0xbc, 0x5e, 0x70, 0xa7, 0x80, 0xed, 0x53, | ||||
|   0xd3, 0xb3, 0x96, 0x06, 0x26, 0x23, 0x08, 0x73, 0xb3, 0x04, 0x55, 0xa8, 0x05, 0x90, 0xf2, 0xfd, | ||||
|   0x93, 0xa0, 0xff, 0x25, 0xcb, 0x2e, 0x78, 0x30, 0xaa, 0x00, 0x23, 0xc9, 0x5b, 0xd1, 0x69, 0xdf, | ||||
|   0xb6, 0x35, 0x81, 0x98, 0xda, 0x38, 0x7f, 0x14, 0x2c, 0xbb, 0x6d, 0x24, 0xbe, 0xd5, 0x1c, 0x48, | ||||
|   0x8d, 0x94, 0xff, 0xe2, 0x01, 0x56, 0x9e, 0x7e, 0xbb, 0x38, 0x3f, 0x03, 0x1b, 0xf8, 0x0f, 0x0a, | ||||
|   0x69, 0x6f, 0x21, 0x20, 0x64, 0xc7, 0xce, 0x9f, 0x53, 0x76, 0x07, 0x49, 0xd4, 0x8d, 0xbd, 0xc0, | ||||
|   0x00, 0xb5, 0x6f, 0x80, 0x12, 0xa7, 0x58, 0x3e, 0x02, 0x54, 0x6d, 0x44, 0x6d, 0x2c, 0x21, 0xd4, | ||||
|   0xde, 0x22, 0x79, 0xb8, 0x88, 0x61, 0x9b, 0x65, 0x02, 0xcc, 0xd0, 0x62, 0xb0, 0xb1, 0x14, 0x6e, | ||||
|   0x38, 0xfd, 0x22, 0x87, 0x3e, 0x7a, 0x49, 0x1f, 0x84, 0x6d, 0x90, 0x16, 0x31, 0xa4, 0x6e, 0x60, | ||||
|   0xf4, 0x2b, 0x66, 0x78, 0x43, 0xc6, 0x60, 0x37, 0x9f, 0x21, 0xe5, 0x0e, 0xc7, 0xa5, 0xfc, 0x7c, | ||||
|   0x10, 0x74, 0x0a, 0x87, 0x9d, 0xd2, 0xf8, 0x53, 0x98, 0x62, 0x4d, 0x44, 0x07, 0xcf, 0x08, 0x8a, | ||||
|   0xb4, 0x38, 0x13, 0x4e, 0x47, 0x81, 0xd1, 0x06, 0x72, 0xec, 0x6d, 0xe4, 0x50, 0xce, 0xf1, 0xce, | ||||
|   0x8b, 0xae, 0x91, 0x63, 0xbc, 0xc7, 0xfe, 0x3e, 0x91, 0xb7, 0x7b, 0x8d, 0x01, 0xf2, 0x59, 0x12, | ||||
|   0xd3, 0x5f, 0xa7, 0x0d, 0x43, 0x8f, 0x64, 0x4a, 0xd9, 0x4c, 0x98, 0x72, 0x73, 0x4b, 0xdb, 0xa3, | ||||
|   0x1d, 0x4b, 0xae, 0xca, 0xc0, 0xbc, 0x99, 0xc6, 0xa7, 0x8f, 0x9f, 0x2f, 0xe1, 0x74, 0xdb, 0x8a, | ||||
|   0xcf, 0x86, 0x8a, 0xd4, 0x43, 0xc9, 0xcb, 0xbf, 0x33, 0x3e, 0x7d, 0x07, 0x81, 0x45, 0x29, 0x34, | ||||
|   0xa1, 0x36, 0x89, 0x2a, 0xdc, 0x80, 0xdc, 0x1a, 0xaf, 0x10, 0xb9, 0xac, 0x84, 0x9b, 0xa1, 0x65, | ||||
|   0xbf, 0xf4, 0x96, 0x61, 0xf1, 0x98, 0x45, 0x64, 0xf5, 0xc6, 0x8b, 0x8a, 0x0f, 0xd9, 0x88, 0x81, | ||||
|   0x2c, 0x26, 0x23, 0x13, 0xc2, 0xa2, 0x60, 0xb5, 0x7d, 0x06, 0x27, 0x06, 0x3d, 0xe5, 0x15, 0xae, | ||||
|   0x6b, 0x09, 0xfe, 0x58, 0x69, 0x4a, 0x78, 0x1f, 0x26, 0x82, 0x8c, 0xa8, 0x00, 0x61, 0x2c, 0xfd, | ||||
|   0x9c, 0xb1, 0x0f, 0xe0, 0xfb, 0x86, 0x3c, 0xc4, 0xb6, 0x2c, 0x52, 0xa2, 0x16, 0x7d, 0x15, 0xb2, | ||||
|   0xaa, 0xc1, 0x02, 0xa5, 0x1a, 0x9a, 0x4a, 0x19, 0x83, 0x9c, 0xa4, 0x5e, 0x11, 0x52, 0x43, 0x42, | ||||
|   0x0f, 0xd9, 0x12, 0x63, 0x59, 0x8f, 0x33, 0x4d, 0x55, 0xd1, 0x82, 0x34, 0x05, 0x7f, 0xc1, 0x97, | ||||
|   0x0b, 0xab, 0x05, 0x82, 0x0f, 0x7b, 0x81, 0x65, 0xa9, 0x55, 0x1d, 0x91, 0x3c, 0x35, 0xbc, 0xdc, | ||||
|   0x92, 0x17, 0xc7, 0xcd, 0x5e, 0x03, 0x8c, 0x43, 0xc6, 0x04, 0x49, 0x62, 0x38, 0xe8, 0x64, 0xf4, | ||||
|   0x48, 0x90, 0x05, 0x10, 0xaa, 0xad, 0xb1, 0xac, 0xb9, 0x30, 0xe0, 0xae, 0x97, 0xb1, 0x34, 0xb7, | ||||
|   0x02, 0x77, 0x80, 0x17, 0xda, 0xa8, 0xdf, 0x90, 0x73, 0x0c, 0xc4, 0x30, 0xa0, 0x03, 0xb1, 0xbf, | ||||
|   0xbf, 0xb2, 0x34, 0xb7, 0x9a, 0x13, 0xaf, 0x17, 0xc0, 0x89, 0xe5, 0x4a, 0xbd, 0x84, 0x56, 0xaf, | ||||
|   0xc1, 0x8a, 0xd9, 0x4d, 0x16, 0xd2, 0x15, 0x97, 0xc4, 0x1a, 0x97, 0x14, 0x01, 0xf9, 0xac, 0x98, | ||||
|   0x40, 0x5e, 0xa8, 0xb7, 0x2e, 0xd6, 0xb7, 0x7e, 0x2b, 0xe5, 0x53, 0x21, 0xc3, 0x12, 0x2a, 0xaa, | ||||
|   0x2d, 0x19, 0x71, 0x36, 0x95, 0xd7, 0x16, 0x7d, 0x72, 0x0b, 0x12, 0xb3, 0x5c, 0x6e, 0xd9, 0xd2, | ||||
|   0xd0, 0x03, 0x43, 0xb3, 0xb9, 0x52, 0xb9, 0xfb, 0xfe, 0xb5, 0x6b, 0xf7, 0xca, 0x3f, 0x48, 0xcb, | ||||
|   0xaa, 0x8f, 0x9b, 0x65, 0x79, 0xbf, 0x23, 0x02, 0x5c, 0x0c, 0x2d, 0x79, 0x41, 0xcd, 0x86, 0x46, | ||||
|   0xa2, 0x14, 0xae, 0xa9, 0xa3, 0xac, 0x3e, 0x00, 0xf5, 0xc8, 0x46, 0xcd, 0x34, 0x2c, 0xb0, 0x83, | ||||
|   0x0f, 0x97, 0xa2, 0x0a, 0xbf, 0x60, 0x42, 0x1c, 0xf9, 0xda, 0xb3, 0x26, 0x24, 0x68, 0x56, 0xe8, | ||||
|   0x89, 0x7f, 0x70, 0x68, 0xe9, 0xcb, 0x4c, 0xa2, 0x00, 0x91, 0x02, 0x91, 0x64, 0x33, 0xba, 0x54, | ||||
|   0xb8, 0x78, 0xa0, 0xfb, 0xf1, 0x84, 0xb0, 0x2e, 0x31, 0xa8, 0xaf, 0xc3, 0x46, 0x84, 0xcb, 0x55, | ||||
|   0x5e, 0xaa, 0x8d, 0x26, 0x85, 0xfc, 0x05, 0xde, 0x3f, 0x3d, 0x1d, 0xbc, 0x0c, 0x02, 0xaa, 0x59, | ||||
|   0x62, 0x2d, 0x64, 0x71, 0xe5, 0x8e, 0xd3, 0xf0, 0xcb, 0x72, 0x85, 0x40, 0x20, 0x02, 0x6a, 0xc1, | ||||
|   0x7c, 0x23, 0x9b, 0x4d, 0xef, 0x20, 0x8a, 0x05, 0x9f, 0x06, 0xa6, 0x0e, 0x7a, 0xc5, 0xd3, 0x93, | ||||
|   0x18, 0xba, 0xf0, 0xcf, 0x09, 0xb0, 0xe8, 0xe9, 0xe9, 0xe5, 0x2f, 0x72, 0x1c, 0x16, 0xf8, 0x90, | ||||
|   0x09, 0x3a, 0x06, 0xb3, 0x22, 0xac, 0x06, 0xd2, 0x25, 0x12, 0xc1, 0xbe, 0xb2, 0x99, 0x30, 0xe0, | ||||
|   0xd7, 0x5c, 0x93, 0xd4, 0xf2, 0x6e, 0x90, 0x71, 0xf2, 0x1a, 0x34, 0x08, 0x21, 0x83, 0x5f, 0x89, | ||||
|   0xdc, 0xb5, 0xe3, 0x38, 0xe1, 0xcd, 0x80, 0x42, 0x67, 0x80, 0x07, 0xc4, 0xd5, 0x01, 0x82, 0x2e, | ||||
|   0x2c, 0x4b, 0x90, 0x75, 0x9f, 0x03, 0xcc, 0x76, 0xa6, 0x61, 0xbe, 0xba, 0xf3, 0x32, 0x17, 0xb0, | ||||
|   0x3e, 0xf8, 0xb8, 0x51, 0x1a, 0xe2, 0xdd, 0x07, 0xb2, 0x1f, 0xbe, 0xf0, 0x67, 0x69, 0xa1, 0x93, | ||||
|   0x2c, 0x18, 0x07, 0x2b, 0x86, 0x5e, 0x20, 0x38, 0xa1, 0xb2, 0x5f, 0xda, 0xd3, 0x14, 0x92, 0x9b, | ||||
|   0x29, 0x7a, 0x70, 0x53, 0x01, 0x43, 0x54, 0xbf, 0x29, 0x49, 0x40, 0x5e, 0xa9, 0x97, 0x30, 0x06, | ||||
|   0x78, 0x8c, 0x4b, 0x7c, 0xa5, 0x80, 0x0f, 0x61, 0x91, 0xa1, 0x55, 0x8d, 0x1c, 0xd0, 0x90, 0x69, | ||||
|   0x52, 0x14, 0xc9, 0x58, 0xc9, 0xed, 0x23, 0x9b, 0x71, 0x72, 0xc7, 0xd9, 0x7d, 0x01, 0x9c, 0x24, | ||||
|   0xbf, 0xb3, 0x19, 0x29, 0x26, 0x6c, 0x96, 0xc6, 0x24, 0xe7, 0xec, 0x2e, 0xbc, 0x4b, 0x1f, 0x89, | ||||
|   0x36, 0x8e, 0xfa, 0x4d, 0xc1, 0x34, 0x04, 0x39, 0x82, 0x30, 0x05, 0x96, 0xc9, 0x62, 0x82, 0x02, | ||||
|   0x00, 0xba, 0x24, 0x9f, 0x1d, 0xc0, 0x84, 0x9c, 0x72, 0x98, 0x30, 0xc2, 0x07, 0x14, 0xf8, 0x98, | ||||
|   0xa0, 0x5c, 0x53, 0x51, 0x81, 0x57, 0x84, 0x70, 0x4a, 0x60, 0xfe, 0xc1, 0x67, 0x92, 0x3b, 0x0a, | ||||
|   0x60, 0x54, 0x23, 0x47, 0x55, 0x9a, 0x50, 0x4e, 0x1d, 0x30, 0xd4, 0x17, 0x48, 0x1c, 0x7c, 0xcb, | ||||
|   0x49, 0x71, 0x85, 0xe4, 0x25, 0x58, 0xee, 0x72, 0xb2, 0x36, 0x1f, 0xef, 0x92, 0x79, 0x51, 0x0f, | ||||
|   0x94, 0xb6, 0x0e, 0x57, 0x07, 0xb8, 0xf1, 0x02, 0xd7, 0xa8, 0x99, 0x4e, 0xb1, 0x65, 0x78, 0xed, | ||||
|   0xcd, 0x02, 0x5a, 0x0e, 0xc5, 0x6f, 0xe4, 0x9b, 0x49, 0x9d, 0x08, 0x43, 0x28, 0x08, 0xab, 0x20, | ||||
|   0xae, 0x48, 0x52, 0x81, 0x49, 0x2f, 0x9e, 0x1a, 0x64, 0x66, 0x53, 0x69, 0x3b, 0xda, 0xff, 0xd4, | ||||
|   0xf8, 0xff, 0x27, 0x7e, 0xdd, 0x86, 0xe3, 0x5d, 0x93, 0x70, 0x6e, 0xad, 0x47, 0xd1, 0x95, 0xcd, | ||||
|   0xe3, 0x60, 0xf3, 0xf8, 0x70, 0x8b, 0x11, 0x19, 0xf0, 0x95, 0x11, 0x64, 0x41, 0x1d, 0xe0, 0x9a, | ||||
|   0xdf, 0xd8, 0x61, 0xb0, 0xfe, 0x74, 0x58, 0x67, 0xc0, 0xa1, 0x53, 0x7b, 0xac, 0x63, 0xec, 0x73, | ||||
|   0x3b, 0x54, 0x55, 0x10, 0xf4, 0xc6, 0xe8, 0x9a, 0x2b, 0x4e, 0x18, 0xa5, 0xad, 0xc9, 0x4a, 0x59, | ||||
|   0xfe, 0x42, 0x1f, 0x0b, 0x93, 0x59, 0x20, 0xf4, 0x80, 0x05, 0x9d, 0x22, 0xb8, 0x5b, 0xbc, 0x72, | ||||
|   0x97, 0xa9, 0x91, 0x32, 0x48, 0x85, 0xbc, 0x6c, 0x05, 0x83, 0x6f, 0xb2, 0xeb, 0xec, 0x66, 0x55, | ||||
|   0x02, 0xdc, 0x41, 0x4c, 0x5a, 0x27, 0xa6, 0x8c, 0xa1, 0x81, 0xa8, 0xaa, 0x42, 0xb6, 0x63, 0x9e, | ||||
|   0x2a, 0x7c, 0xad, 0x5e, 0xb6, 0xe1, 0x3e, 0x92, 0xf5, 0x7d, 0xd4, 0x86, 0x57, 0xd5, 0xab, 0x0d, | ||||
|   0x84, 0x55, 0x29, 0x4b, 0x3e, 0x35, 0xd1, 0xcf, 0xdc, 0x10, 0x5f, 0xe1, 0xe0, 0x23, 0x51, 0x08, | ||||
|   0x7a, 0x83, 0x52, 0xa6, 0xd0, 0xb3, 0x37, 0xa3, 0x3c, 0xf0, 0x0d, 0x58, 0x9d, 0xd9, 0x76, 0xb5, | ||||
|   0x61, 0x7c, 0xa6, 0xf8, 0x7e, 0x47, 0xbd, 0x68, 0xaa, 0xbd, 0x24, 0xc2, 0xe7, 0x37, 0x04, 0xf0, | ||||
|   0x63, 0x49, 0x6b, 0x95, 0xff, 0xea, 0xf7, 0x19, 0x76, 0xb1, 0xbe, 0x89, 0x8a, 0xa2, 0x72, 0x0b, | ||||
|   0xd1, 0xce, 0x2d, 0x44, 0x72, 0x0b, 0xe5, 0x9b, 0x3d, 0xdc, 0x42, 0xb4, 0x6d, 0x0b, 0x48, 0x38, | ||||
|   0xc4, 0x3a, 0xf8, 0x88, 0x42, 0xd2, 0x1f, 0xed, 0xb8, 0x9a, 0x39, 0x65, 0xf9, 0xa3, 0xa2, 0x16, | ||||
|   0xe2, 0xdf, 0x65, 0xa9, 0x7a, 0xb8, 0x05, 0xb5, 0x99, 0xdb, 0xa6, 0xe1, 0xc1, 0x72, 0x03, 0xda, | ||||
|   0xa3, 0xbd, 0xbd, 0x67, 0x11, 0xbe, 0x5e, 0x6c, 0x9b, 0xb4, 0x1d, 0x7d, 0xd4, 0x64, 0x10, 0x52, | ||||
|   0x0c, 0x7d, 0x6b, 0x0c, 0xaa, 0xf6, 0x8b, 0x75, 0xb4, 0xed, 0x92, 0x5c, 0xbd, 0x31, 0x45, 0x89, | ||||
|   0x03, 0x95, 0xca, 0x03, 0xc3, 0xa8, 0xf4, 0x8b, 0x82, 0x7e, 0xd1, 0x21, 0x4a, 0x6b, 0xa9, 0x57, | ||||
|   0x10, 0xdc, 0xfb, 0x56, 0x15, 0x30, 0xe0, 0x08, 0x3a, 0x0a, 0x74, 0x41, 0x4a, 0xba, 0xb1, 0x3a, | ||||
|   0xae, 0x5c, 0x90, 0x1a, 0xdb, 0xf7, 0x6e, 0xac, 0x1c, 0x92, 0x85, 0x57, 0xaf, 0x17, 0x55, 0x07, | ||||
|   0x56, 0x2e, 0x44, 0x1b, 0x5c, 0xc3, 0x8f, 0x9e, 0xeb, 0x2e, 0x7f, 0xb0, 0xc9, 0xad, 0x2c, 0xaf, | ||||
|   0x2f, 0x10, 0x4e, 0xbe, 0x16, 0xaf, 0xc3, 0x62, 0xad, 0x43, 0x7f, 0xf9, 0x8d, 0xaf, 0x0e, 0x7e, | ||||
|   0x79, 0xd6, 0x06, 0x2e, 0xcc, 0x3f, 0x7c, 0x88, 0x2c, 0x82, 0x5c, 0xe7, 0x25, 0xae, 0xdd, 0xf2, | ||||
|   0xb7, 0xd5, 0x11, 0x3f, 0x4c, 0xc1, 0x36, 0x07, 0xb7, 0xbb, 0xea, 0x33, 0xb8, 0x54, 0xbe, 0xb4, | ||||
|   0x6e, 0x4b, 0x1b, 0xa0, 0x2a, 0xa1, 0x1b, 0x4f, 0x5a, 0x41, 0x98, 0x4b, 0x9b, 0x10, 0x04, 0xd9, | ||||
|   0x4f, 0xc9, 0x5a, 0xb9, 0xb0, 0x1f, 0xe9, 0x75, 0xd5, 0xeb, 0xc9, 0x73, 0xbc, 0x7a, 0x30, 0xe4, | ||||
|   0xfb, 0x1b, 0x9b, 0xbf, 0x0c, 0x64, 0x10, 0xb9, 0xb7, 0xd7, 0x9c, 0x14, 0xe1, 0x7b, 0x86, 0xb5, | ||||
|   0x7a, 0x64, 0xb8, 0x5e, 0xbb, 0x6c, 0x2c, 0x4a, 0x1b, 0xa3, 0xa1, 0x05, 0xae, 0xb2, 0xd9, 0xb1, | ||||
|   0x5c, 0x99, 0xfe, 0xba, 0x84, 0x53, 0xbc, 0x17, 0xdd, 0xa8, 0x14, 0xbe, 0x91, 0x55, 0xf0, 0xeb, | ||||
|   0x24, 0xfe, 0x67, 0xb3, 0x7c, 0x7e, 0xf3, 0x66, 0x95, 0x17, 0x62, 0x36, 0x42, 0x77, 0xe5, 0xca, | ||||
|   0xba, 0xa4, 0xbb, 0x89, 0x18, 0x75, 0xb1, 0x86, 0x57, 0x5f, 0xc4, 0xdd, 0xd8, 0x64, 0x6d, 0xa0, | ||||
|   0x71, 0x7b, 0xdd, 0x18, 0x6e, 0xdc, 0xe0, 0x7d, 0x17, 0x41, 0x03, 0x15, 0x23, 0xd7, 0x22, 0xc3, | ||||
|   0x71, 0xcd, 0xb3, 0x19, 0xfb, 0xd4, 0xda, 0x76, 0x8f, 0x5a, 0x1a, 0x70, 0xc3, 0xb2, 0xd6, 0x94, | ||||
|   0x42, 0x6c, 0xd3, 0x08, 0x08, 0xf7, 0xaa, 0x60, 0x6f, 0xab, 0x52, 0x08, 0xa5, 0x11, 0x4c, 0x3e, | ||||
|   0x1e, 0x51, 0x1f, 0x4a, 0xf6, 0x59, 0x00, 0x92, 0x7f, 0xc9, 0xce, 0xe8, 0x83, 0xa9, 0xba, 0x6d, | ||||
|   0x21, 0x25, 0x5e, 0xfe, 0x74, 0x6e, 0x2c, 0x25, 0xd5, 0xf2, 0xe5, 0x0d, 0xb7, 0x59, 0x95, 0x44, | ||||
|   0x16, 0x11, 0x67, 0xc0, 0x59, 0xd7, 0x76, 0xeb, 0x67, 0x5c, 0xa1, 0x82, 0x00, 0xc9, 0xe6, 0x2b, | ||||
|   0x3f, 0x68, 0xd2, 0xe1, 0xd0, 0xeb, 0x42, 0x64, 0x38, 0x3c, 0x7a, 0xe2, 0xcd, 0x07, 0x2a, 0x3a, | ||||
|   0xb1, 0x43, 0xb2, 0x0c, 0xd7, 0xc0, 0x2b, 0x53, 0xb0, 0xa2, 0x66, 0xb7, 0xc5, 0xca, 0x10, 0x74, | ||||
|   0x9f, 0x2d, 0xab, 0x1c, 0xae, 0x5c, 0x7d, 0x57, 0xee, 0x8c, 0xb7, 0x1d, 0xcf, 0x00, 0x41, 0x12, | ||||
|   0x9d, 0xfc, 0x49, 0x4b, 0xb0, 0x46, 0x05, 0x7d, 0xe7, 0x4b, 0x87, 0xc6, 0x73, 0x26, 0xfd, 0x74, | ||||
|   0xc9, 0xd5, 0xbf, 0xf8, 0xac, 0x67, 0xe3, 0xaa, 0xa9, 0xfe, 0xff, 0xfe, 0xb4, 0xd5, 0xff, 0xbf, | ||||
|   0xf5, 0xbf, 0x04, 0xfa, 0x03, 0x25, 0xd7, 0x35, 0x00, 0x00 | ||||
| }; | ||||
							
								
								
									
										1145
									
								
								wled00/html_other.h
									
									
									
									
									
								
							
							
						
						
									
										1145
									
								
								wled00/html_other.h
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user