sync and merge master
							
								
								
									
										22
									
								
								.devcontainer/Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,22 @@ | ||||
| # See here for image contents: https://github.com/microsoft/vscode-dev-containers/tree/v0.148.1/containers/python-3/.devcontainer/base.Dockerfile | ||||
|  | ||||
| # [Choice] Python version: 3, 3.9, 3.8, 3.7, 3.6 | ||||
| ARG VARIANT="3" | ||||
| FROM mcr.microsoft.com/vscode/devcontainers/python:0-${VARIANT} | ||||
|  | ||||
| # [Option] Install Node.js | ||||
| ARG INSTALL_NODE="true" | ||||
| ARG NODE_VERSION="lts/*" | ||||
| RUN if [ "${INSTALL_NODE}" = "true" ]; then su vscode -c "source /usr/local/share/nvm/nvm.sh && nvm install ${NODE_VERSION} 2>&1"; fi | ||||
|  | ||||
| # [Optional] If your pip requirements rarely change, uncomment this section to add them to the image. | ||||
| # COPY requirements.txt /tmp/pip-tmp/ | ||||
| # RUN pip3 --disable-pip-version-check --no-cache-dir install -r /tmp/pip-tmp/requirements.txt \ | ||||
| #    && rm -rf /tmp/pip-tmp | ||||
|  | ||||
| # [Optional] Uncomment this section to install additional OS packages. | ||||
| # RUN apt-get update && export DEBIAN_FRONTEND=noninteractive \ | ||||
| #     && apt-get -y install --no-install-recommends <your-package-list-here> | ||||
|  | ||||
| # [Optional] Uncomment this line to install global node packages. | ||||
| # RUN su vscode -c "source /usr/local/share/nvm/nvm.sh && npm install -g <your-package-here>" 2>&1 | ||||
							
								
								
									
										47
									
								
								.devcontainer/devcontainer.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,47 @@ | ||||
| { | ||||
| 	"name": "Python 3", | ||||
| 	"build": { | ||||
| 		"dockerfile": "Dockerfile", | ||||
| 		"context": "..", | ||||
| 		"args": {  | ||||
| 			// Update 'VARIANT' to pick a Python version: 3, 3.6, 3.7, 3.8, 3.9 | ||||
| 			"VARIANT": "3", | ||||
| 			// Options | ||||
| 			"INSTALL_NODE": "true", | ||||
| 			"NODE_VERSION": "lts/*" | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	// Set *default* container specific settings.json values on container create. | ||||
| 	"settings": {  | ||||
| 		"terminal.integrated.shell.linux": "/bin/bash", | ||||
| 		"python.pythonPath": "/usr/local/bin/python", | ||||
| 		"python.linting.enabled": true, | ||||
| 		"python.linting.pylintEnabled": true, | ||||
| 		"python.formatting.autopep8Path": "/usr/local/py-utils/bin/autopep8", | ||||
| 		"python.formatting.blackPath": "/usr/local/py-utils/bin/black", | ||||
| 		"python.formatting.yapfPath": "/usr/local/py-utils/bin/yapf", | ||||
| 		"python.linting.banditPath": "/usr/local/py-utils/bin/bandit", | ||||
| 		"python.linting.flake8Path": "/usr/local/py-utils/bin/flake8", | ||||
| 		"python.linting.mypyPath": "/usr/local/py-utils/bin/mypy", | ||||
| 		"python.linting.pycodestylePath": "/usr/local/py-utils/bin/pycodestyle", | ||||
| 		"python.linting.pydocstylePath": "/usr/local/py-utils/bin/pydocstyle", | ||||
| 		"python.linting.pylintPath": "/usr/local/py-utils/bin/pylint" | ||||
| 	}, | ||||
|  | ||||
| 	// Add the IDs of extensions you want installed when the container is created. | ||||
| 	"extensions": [ | ||||
| 		"ms-python.python", | ||||
| 		"platformio.platformio-ide" | ||||
| 	], | ||||
|  | ||||
| 	// Use 'forwardPorts' to make a list of ports inside the container available locally. | ||||
| 	// "forwardPorts": [], | ||||
|  | ||||
| 	// Use 'postCreateCommand' to run commands after the container is created. | ||||
| 	"postCreateCommand": "npm install", | ||||
|  | ||||
| 	// Comment out connect as root instead. More info: https://aka.ms/vscode-remote/containers/non-root. | ||||
| 	"remoteUser": "vscode" | ||||
| } | ||||
|  | ||||
							
								
								
									
										27
									
								
								.github/ISSUE_TEMPLATE/bug.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,27 @@ | ||||
| --- | ||||
| name: Bug | ||||
| about: Noticed an issue with your lights? | ||||
| title: '' | ||||
| labels: bug | ||||
| assignees: '' | ||||
|  | ||||
| --- | ||||
|  | ||||
| **Describe the bug** | ||||
| A clear and concise description of what the bug is. Please quickly search existing issues first! | ||||
|  | ||||
| **To Reproduce** | ||||
| Steps to reproduce the behavior, if consistently possible | ||||
|  | ||||
| **Expected behavior** | ||||
| A clear and concise description of what you expected to happen. | ||||
|  | ||||
| **WLED version** | ||||
|  - Board: [e.g. Wemos D1, ESP32 dev] | ||||
|  - Version [e.g. 0.10.0, dev200603] | ||||
|  - Format [e.g. Binary, self-compiled] | ||||
|  | ||||
| **Additional context** | ||||
| Anything else you'd like to say about the problem? | ||||
|  | ||||
| Thank you for your help! | ||||
							
								
								
									
										22
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,22 @@ | ||||
| --- | ||||
| name: Feature request | ||||
| about: Suggest an improvement idea for WLED! | ||||
| title: '' | ||||
| labels: enhancement | ||||
| assignees: '' | ||||
|  | ||||
| --- | ||||
|  | ||||
| **Is your feature request related to a problem? Please describe.** | ||||
| A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] | ||||
|  | ||||
| **Describe the solution you'd like** | ||||
| A clear and concise description of what you want to happen. | ||||
|  | ||||
| **Describe alternatives you've considered** | ||||
| A clear and concise description of any alternative solutions or features you've considered. | ||||
|  | ||||
| **Additional context** | ||||
| Add any other context or screenshots about the feature request here. | ||||
|  | ||||
| Thank you for your ideas for making WLED better! | ||||
							
								
								
									
										19
									
								
								.github/ISSUE_TEMPLATE/question.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | ||||
| --- | ||||
| name: Question | ||||
| about: Have a question about using WLED? | ||||
| title: '' | ||||
| labels: question | ||||
| assignees: '' | ||||
|  | ||||
| --- | ||||
|  | ||||
| **Take a look at the wiki and FAQ, perhaps your question is already answered!** | ||||
| [FAQ](https://github.com/Aircoookie/WLED/wiki/FAQ) | ||||
|  | ||||
| **Please consider asking your question on the WLED forum or Discord** | ||||
| [Forum](https://wled.discourse.group/) | ||||
| [Discord](https://discord.gg/KuqP7NE) | ||||
| [What to post where?](https://github.com/Aircoookie/WLED/issues/658) | ||||
|  | ||||
| **If you do not like to use these platforms, delete this template and ask away!** | ||||
| Please keep in mind though that the issue section is generally not the preferred place for general questions. | ||||
							
								
								
									
										20
									
								
								.github/stale.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | ||||
| # Number of days of inactivity before an issue becomes stale | ||||
| daysUntilStale: 120 | ||||
| # Number of days of inactivity before a stale issue is closed | ||||
| daysUntilClose: 7 | ||||
| # Issues with these labels will never be considered stale | ||||
| exemptLabels: | ||||
|   - pinned | ||||
|   - keep | ||||
|   - enhancement | ||||
|   - confirmed | ||||
| # Label to use when marking an issue as stale | ||||
| staleLabel: stale | ||||
| # Comment to post when marking an issue as stale. Set to `false` to disable | ||||
| markComment: > | ||||
|   Hey! This issue has been open for quite some time without any new comments now. | ||||
|   It will be closed automatically in a week if no further activity occurs. | ||||
|    | ||||
|   Thank you for using WLED! | ||||
| # Comment to post when closing a stale issue. Set to `false` to disable | ||||
| closeComment: false | ||||
							
								
								
									
										31
									
								
								.github/workflows/wled-ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,31 @@ | ||||
| name: PlatformIO CI | ||||
|  | ||||
| on: [push, pull_request] | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|  | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|     - name: Cache pip | ||||
|       uses: actions/cache@v2 | ||||
|       with: | ||||
|         path: ~/.cache/pip | ||||
|         key: ${{ runner.os }}-pip-${{ hashFiles('**/requirements.txt') }} | ||||
|         restore-keys: | | ||||
|           ${{ runner.os }}-pip- | ||||
|     - name: Cache PlatformIO | ||||
|       uses: actions/cache@v2 | ||||
|       with: | ||||
|         path: ~/.platformio | ||||
|         key: ${{ runner.os }}-${{ hashFiles('**/lockfiles') }} | ||||
|     - name: Set up Python | ||||
|       uses: actions/setup-python@v2 | ||||
|     - name: Install PlatformIO | ||||
|       run: | | ||||
|         python -m pip install --upgrade pip | ||||
|         pip install --upgrade platformio | ||||
|     - name: Run PlatformIO | ||||
|       run: pio run | ||||
							
								
								
									
										9
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,7 +1,16 @@ | ||||
| .pio | ||||
| .cache | ||||
| .pioenvs | ||||
| .piolibdeps | ||||
| .vscode | ||||
| !.vscode/extensions.json | ||||
| /wled00/Release | ||||
| /wled00/extLibs | ||||
| /platformio_override.ini | ||||
| /wled00/my_config.h | ||||
| /build_output | ||||
| .DS_Store | ||||
| .gitignore | ||||
| .clang-format | ||||
| node_modules | ||||
| .idea | ||||
|   | ||||
							
								
								
									
										5
									
								
								.gitpod.Dockerfile
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | ||||
| FROM gitpod/workspace-full | ||||
|                      | ||||
| USER gitpod | ||||
|  | ||||
| RUN pip3 install -U platformio | ||||
							
								
								
									
										12
									
								
								.gitpod.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | ||||
| tasks: | ||||
|   - command: platformio run | ||||
|  | ||||
| image: | ||||
|   file: .gitpod.Dockerfile | ||||
|  | ||||
| vscode: | ||||
|   extensions: | ||||
|     - ms-vscode.cpptools@0.26.3:u3GsZ5PK12Ddr79vh4TWgQ== | ||||
|     - eamodio.gitlens@10.2.1:e0IYyp0efFqVsrZwsIe8CA== | ||||
|     - Atishay-Jain.All-Autocomplete@0.0.23:fbZNfSpnd8XkAHGfAPS2rA== | ||||
|     - 2gua.rainbow-brackets@0.0.6:Tbu8dTz0i+/bgcKQTQ5b8g== | ||||
							
								
								
									
										35
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						| @@ -1,35 +0,0 @@ | ||||
| # Continuous Integration (CI) is the practice, in software | ||||
| # engineering, of merging all developer working copies with a shared mainline | ||||
| # several times a day < https://docs.platformio.org/page/ci/index.html > | ||||
| # | ||||
| # Documentation: | ||||
| # | ||||
| # * Travis CI Embedded Builds with PlatformIO | ||||
| #   < https://docs.travis-ci.com/user/integration/platformio/ > | ||||
| # | ||||
| # * PlatformIO integration with Travis CI | ||||
| #   < https://docs.platformio.org/page/ci/travis.html > | ||||
| # | ||||
| # * User Guide for `platformio ci` command | ||||
| #   < https://docs.platformio.org/page/userguide/cmd_ci.html > | ||||
| # | ||||
| # | ||||
| # Please choose one of the following templates (proposed below) and uncomment | ||||
| # it (remove "# " before each line) or use own configuration according to the | ||||
| # Travis CI documentation (see above). | ||||
| # | ||||
|  | ||||
| language: python | ||||
| python: | ||||
|     - "2.7" | ||||
| sudo: false | ||||
| cache: | ||||
|     directories: | ||||
|         - "~/.platformio" | ||||
| env: | ||||
|     - PLATFORMIO_CI_SRC=wled00 | ||||
| install: | ||||
|     - pip install -U platformio | ||||
|     - platformio update | ||||
| script: | ||||
|     - platformio ci --project-conf=./platformio.ini | ||||
							
								
								
									
										12
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,7 +1,7 @@ | ||||
| { | ||||
| 	// See http://go.microsoft.com/fwlink/?LinkId=827846 | ||||
| 	// for the documentation about the extensions.json format | ||||
| 	"recommendations": [ | ||||
| 		"platformio.platformio-ide" | ||||
| 	] | ||||
| } | ||||
|     // See http://go.microsoft.com/fwlink/?LinkId=827846 | ||||
|     // for the documentation about the extensions.json format | ||||
|     "recommendations": [ | ||||
|         "platformio.platformio-ide" | ||||
|     ] | ||||
| } | ||||
|   | ||||
							
								
								
									
										42
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,42 @@ | ||||
| { | ||||
|   "version": "2.0.0", | ||||
|   "tasks": [ | ||||
|     { | ||||
|       "label": "Build: HTML and binary", | ||||
|       "dependsOn": [ | ||||
|         "Build: HTML only", | ||||
|         "Build: binary only" | ||||
|       ], | ||||
|       "dependsOrder": "sequence", | ||||
|       "problemMatcher": [ | ||||
|         "$platformio", | ||||
|       ], | ||||
|     }, | ||||
|     { | ||||
|       "type": "PlatformIO", | ||||
|       "label": "Build: binary only", | ||||
|       "task": "Build", | ||||
|       "group": { | ||||
|         "kind": "build", | ||||
|         "isDefault": true, | ||||
|       }, | ||||
|       "problemMatcher": [ | ||||
|         "$platformio" | ||||
|       ], | ||||
|       "presentation": { | ||||
|         "panel": "shared" | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "type": "npm", | ||||
|       "script": "build", | ||||
|       "group": "build", | ||||
|       "problemMatcher": [], | ||||
|       "label": "Build: HTML only", | ||||
|       "detail": "npm run build", | ||||
|       "presentation": { | ||||
|         "panel": "shared" | ||||
|       } | ||||
|     } | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										608
									
								
								CHANGELOG.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,608 @@ | ||||
| ## WLED changelog | ||||
|  | ||||
| ### Builds after release 0.12.0 | ||||
|  | ||||
| #### Build 2104030 | ||||
|  | ||||
| -   Fixed ESP32 crash on Drip effect with reversed segment (#1854) | ||||
| -   Added flag `WLED_DISABLE_BROWNOUT_DET` to disable ESP32 brownout detector (off by default) | ||||
|  | ||||
| ### WLED release 0.12.0 | ||||
|  | ||||
| #### Build 2104020 | ||||
|  | ||||
| -   Allow clearing button/IR/relay pin on platforms that don't support negative numbers | ||||
| -   Removed AUX pin | ||||
| -   Hid some easter eggs, only to be found at easter | ||||
|  | ||||
| ### Development versions between 0.11.1 and 0.12.0 releases | ||||
|  | ||||
| #### Build 2103310 | ||||
|  | ||||
| -   Version bump to 0.12.0 "Hikari" | ||||
| -   Fixed LED settings submission in iOS app | ||||
|  | ||||
| #### Build 2103300 | ||||
|  | ||||
| -   Version bump to 0.12.0-b5 "Hikari" | ||||
| -   Update to core espressif32@3.2 | ||||
| -   Fixed IR pin not configurable | ||||
|  | ||||
| #### Build 2103290 | ||||
|  | ||||
| -   Version bump to 0.12.0-b4 "Hikari" | ||||
| -   Experimental use of espressif32@3.1.1 | ||||
| -   Fixed RGBW mode disabled after LED settings saved | ||||
| -   Fixed infrared support not compiled in if IRPIN is not defined | ||||
|  | ||||
| #### Build 2103230 | ||||
|  | ||||
| -   Fixed current estimation | ||||
|  | ||||
| #### Build 2103220 | ||||
|  | ||||
| -   Version bump to 0.12.0-b2 "Hikari" | ||||
| -   Worked around an issue causing a critical decrease in framerate (wled.cpp l.240 block) | ||||
| -   Bump to Espalexa v2.7.0, fixing discovery | ||||
|  | ||||
| #### Build 2103210 | ||||
|  | ||||
| -   Version bump to 0.12.0-b1 "Hikari" | ||||
| -   More colors visible on Palette preview | ||||
| -   Fixed chevron icon not included | ||||
| -   Fixed color order override | ||||
| -   Cleanup | ||||
|  | ||||
| #### Build 2103200 | ||||
|  | ||||
| -   Version bump to 0.12.0-b0 "Hikari" | ||||
| -   Added palette preview and search (PR #1637) | ||||
| -   Added Reverse checkbox for PWM busses - reverses logic level for on | ||||
| -   Fixed various problems with the Playlist feature (PR #1724) | ||||
| -   Replaced "Layer" icon with "i" icon for Info button | ||||
| -   Chunchun effect more fitting for various segment lengths (PR #1804) | ||||
| -   Removed global reverse (in favor of individual bus reverse) | ||||
| -   Removed some unused icons from UI icon font | ||||
|  | ||||
| #### Build 2103130 | ||||
|  | ||||
| -   Added options for Auto Node discovery | ||||
| -   Optimized strings (no string both F() and raw) | ||||
|  | ||||
| #### Build 2103090 | ||||
|  | ||||
| -   Added Auto Node discovery (PR #1683) | ||||
| -   Added tooltips to quick color selectors for accessibility | ||||
|  | ||||
| #### Build 2103060 | ||||
|  | ||||
| -   Auto start field population in bus config | ||||
|  | ||||
| #### Build 2103050 | ||||
|  | ||||
| -   Fixed incorrect over-memory indication in LED settings on ESP32 | ||||
|  | ||||
| #### Build 2103041 | ||||
|  | ||||
| -   Added destructor for BusPwm (fixes #1789) | ||||
|  | ||||
| #### Build 2103040 | ||||
|  | ||||
| -   Fixed relay mode inverted when upgrading from 0.11.0 | ||||
| -   Fixed no more than 2 pins per bus configurable in UI | ||||
| -   Changed to non-linear IR brightness steps (PR #1742) | ||||
| -   Fixed various warnings (PR #1744) | ||||
| -   Added UDP DNRGBW Mode (PR #1704) | ||||
| -   Added dynamic LED mapping with ledmap.json file (PR #1738) | ||||
| -   Added support for QuinLED-ESP32-Ethernet board | ||||
| -   Added support for WESP32 ethernet board (PR #1764) | ||||
| -   Added Caching for main UI (PR #1704) | ||||
| -   Added Tetrix mode (PR #1729) | ||||
| -   Added memory check on Bus creation | ||||
|  | ||||
| #### Build 2102050 | ||||
|  | ||||
| -   Version bump to 0.12.0-a0 "Hikari" | ||||
| -   Added FPS indication in info | ||||
| -   Bumped max outputs from 7 to 10 busses for ESP32 | ||||
|  | ||||
| #### Build 2101310 | ||||
|  | ||||
| -   First alpha configurable multipin | ||||
|  | ||||
| #### Build 2101130 | ||||
|  | ||||
| -   Added color transitions for all segments and slots and for segment brightness | ||||
| -   Fixed bug that prevented setting a boot preset higher than 25 | ||||
|  | ||||
| #### Build 2101040 | ||||
|  | ||||
| -   Replaced Red & Blue effect with Aurora effect (PR #1589) | ||||
| -   Fixed HTTP changing segments uncommanded (#1618) | ||||
| -   Updated copyright year and contributor page link | ||||
|  | ||||
| #### Build 2012311 | ||||
|  | ||||
| -   Fixed Countdown mode | ||||
|  | ||||
| #### Build 2012310 | ||||
|  | ||||
| -   (Hopefully actually) fixed display of usermod values in info screen | ||||
|  | ||||
| #### Build 2012240 | ||||
|  | ||||
| -   Fixed display of usermod values in info screen | ||||
| -   4 more effects now use FRAMETIME | ||||
| -   Remove unsupported environments from platformio.ini | ||||
|  | ||||
| #### Build 2012210 | ||||
|  | ||||
| -   Split index.htm in separate CSS + JS files (PR #1542) | ||||
| -   Minify UI HTML, saving >1.5kB flash | ||||
| -   Fixed JShint warnings | ||||
|  | ||||
| #### Build 2012180 | ||||
|  | ||||
| -   Boot brightness 0 will now use the brightness from preset | ||||
| -   Add iOS scrolling momentum (from PR #1528) | ||||
|  | ||||
| ### WLED release 0.11.1 | ||||
|  | ||||
| #### Build 2012180 | ||||
|  | ||||
| -   Release of WLED 0.11.1 "Mirai" | ||||
| -   Fixed AP hide not saving (fixes #1520) | ||||
| -   Fixed MQTT password re-transmitted to HTML | ||||
| -   Hide Update buttons while uploading, accept .bin | ||||
| -   Make sure AP password is at least 8 characters long | ||||
|  | ||||
| ### Development versions after 0.11.0 release | ||||
|  | ||||
| #### Build 2012160 | ||||
|  | ||||
| -   Bump Espalexa to 2.5.0, fixing discovery (PR Espalexa/#152, originally PR #1497) | ||||
|  | ||||
| #### Build 2012150 | ||||
|  | ||||
| -   Added Blends FX (PR #1491) | ||||
| -   Fixed an issue that made it impossible to deactivate timed presets | ||||
|  | ||||
| #### Build 2012140 | ||||
|  | ||||
| -   Added Preset ID quick display option (PR #1462) | ||||
| -   Fixed LEDs not turning on when using gamma correct brightness and LEDPIN 2 (default) | ||||
| -   Fixed notifier applying main segment to selected segments on notification with FX/Col disabled  | ||||
|  | ||||
| #### Build 2012130 | ||||
|  | ||||
| -   Fixed RGBW mode not saved between reboots (fixes #1457) | ||||
| -   Added brightness scaling in palette function for default (PR #1484) | ||||
|  | ||||
| #### Build 2012101 | ||||
|  | ||||
| -   Fixed preset cycle default duration rounded down to nearest 10sec interval (#1458) | ||||
| -   Enabled E1.31/DDP/Art-Net in AP mode | ||||
|  | ||||
| #### Build 2012100 | ||||
|  | ||||
| -   Fixed multi-segment preset cycle | ||||
| -   Fixed EEPROM (pre-0.11 settings) not cleared on factory reset | ||||
| -   Fixed an issue with intermittent crashes on FX change (PR #1465) | ||||
| -   Added function to know if strip is updating (PR #1466) | ||||
| -   Fixed using colorwheel sliding the UI (PR #1459) | ||||
| -   Fixed analog clock settings not saving (PR #1448) | ||||
| -   Added Temperature palette (PR #1430) | ||||
| -   Added Candy cane FX (PR #1445) | ||||
|  | ||||
| #### Build 2012020 | ||||
|  | ||||
| -   UDP `parsePacket()` with sync disabled (#1390) | ||||
| -   Added Multi RGBW DMX mode (PR #1383) | ||||
|  | ||||
| #### Build 2012010 | ||||
|  | ||||
| -   Fixed compilation for analog (PWM) LEDs | ||||
|  | ||||
| ### WLED version 0.11.0 | ||||
|  | ||||
| #### Build 2011290 | ||||
|  | ||||
| -   Release of WLED 0.11.0 "Mirai" | ||||
| -   Workaround for weird empty %f Espalexa issue | ||||
| -   Fixed crash on saving preset with HTTP API `PS` | ||||
| -   Improved performance for color changes in non-main segment | ||||
|  | ||||
| #### Build 2011270 | ||||
|  | ||||
| -   Added tooltips for speed and intensity sliders (PR #1378) | ||||
| -   Moved color order to NpbWrapper.h | ||||
| -   Added compile time define to override the color order for a specific range | ||||
|  | ||||
| #### Build 2011260 | ||||
|  | ||||
| -   Add `live` property to state, allowing toggling of realtime (not incl. in state resp.) | ||||
| -   PIO environment changes | ||||
|  | ||||
| #### Build 2011230 | ||||
|  | ||||
| -   Version bump to 0.11.0 "Mirai" | ||||
| -   Improved preset name sorting | ||||
| -   Fixed Preset cycle not working beyond preset 16 | ||||
|  | ||||
| ### Development versions between 0.10.2 and 0.11.0 releases | ||||
|  | ||||
| #### Build 2011220 | ||||
|  | ||||
| -   Fixed invalid save when modifying preset before refresh (might be related to #1361) | ||||
| -   Fixed brightness factor ignored on realtime timeout (fixes #1363) | ||||
| -   Fixed Phase and Chase effects with LED counts >256 (PR #1366) | ||||
|  | ||||
| #### Build 2011210 | ||||
|  | ||||
| -   Fixed Brightness slider beneath color wheel not working (fixes #1360) | ||||
| -   Fixed invalid UI state after saving modified preset | ||||
|  | ||||
| #### Build 2011200 | ||||
|  | ||||
| -   Added HEX color receiving to JSON API with `"col":["RRGGBBWW"]` format | ||||
| -   Moved Kelvin color receiving in JSON API from `"col":[[val]]` to `"col":[val]` format | ||||
|     _Notice:_ This is technically a breaking change. Since no release was made since the introduction and the Kelvin property was not previously documented in the wiki, | ||||
|     impact should be minimal.  | ||||
| -   BTNPIN can now be disabled by setting to -1 (fixes #1237) | ||||
|  | ||||
| #### Build 2011180 | ||||
|  | ||||
| -   Platformio.ini updates and streamlining (PR #1266) | ||||
| -   my_config.h custom compile settings system (not yet used for much, adapted from PR #1266) | ||||
| -   Added Hawaii timezone (HST) | ||||
| -   Linebreak after 5 quick select buttons | ||||
|  | ||||
| #### Build 2011154 | ||||
|  | ||||
| -   Fixed RGBW saved incorrectly | ||||
| -   Fixed pmt caching requesting /presets.json too often | ||||
| -   Fixed deEEP not copying the first segment of EEPROM preset 16 | ||||
|  | ||||
| #### Build 2011153 | ||||
|  | ||||
| -   Fixed an ESP32 end-of-file issue | ||||
| -   Fixed strip.isRgbw not read from cfg.json | ||||
|  | ||||
| #### Build 2011152 | ||||
|  | ||||
| -   Version bump to 0.11.0p "Mirai" | ||||
| -   Increased max. num of segments to 12 (ESP8266) / 16 (ESP32) | ||||
| -   Up to 250 presets stored in the `presets.json` file in filesystem | ||||
| -   Complete overhaul of the Presets UI tab | ||||
| -   Updated iro.js to v5 (fixes black color wheel) | ||||
| -   Added white temperature slider to color wheel | ||||
| -   Add JSON settings serialization/deserialization to cfg.json and wsec.json | ||||
| -   Added deEEP to convert the EEPROM settings and presets to files | ||||
| -   Playlist support - JSON only for now | ||||
| -   New v2 usermod methods `addToConfig()` and `readFromConfig()` (see EXAMPLE_v2 for doc) | ||||
| -   Added Ethernet support for ESP32 (PR #1316) | ||||
| -   IP addresses are now handled by the `Network` class | ||||
| -   New `esp32_poe` PIO environment | ||||
| -   Use EspAsyncWebserver Aircoookie fork v.2.0.0 (hiding wsec.json) | ||||
| -   Removed `WLED_DISABLE_FILESYSTEM` and `WLED_ENABLE_FS_SERVING` defines as they are now required | ||||
| -   Added pin manager | ||||
| -   UI performance improvements (no drop shadows) | ||||
| -   More explanatory error messages in UI | ||||
| -   Improved candle brightness | ||||
| -   Return remaining nightlight time `nl.rem` in JSON API (PR #1302) | ||||
| -   UI sends timestamp with every command, allowing for timed presets without using NTP | ||||
| -   Added gamma calculation (yet unused) | ||||
| -   Added LED type definitions to const.h (yet unused) | ||||
| -   Added nicer 404 page | ||||
| -   Removed `NP` and `MS=` macro HTTP API commands | ||||
| -   Removed macros from Time settings | ||||
|  | ||||
| #### Build 2011120 | ||||
|  | ||||
| -   Added the ability for the /api MQTT topic to receive JSON API payloads | ||||
|  | ||||
| #### Build 2011040 | ||||
|  | ||||
| -   Inversed Rain direction (fixes #1147) | ||||
|  | ||||
| #### Build 2011010 | ||||
|  | ||||
| -   Re-added previous C9 palette | ||||
| -   Renamed new C9 palette | ||||
|  | ||||
| #### Build 2010290 | ||||
|  | ||||
| -   Colorful effect now supports palettes | ||||
| -   Added C9 2 palette (#1291) | ||||
| -   Improved C9 palette brightness by 12% | ||||
| -   Disable onboard LED if LEDs are off (PR #1245) | ||||
| -   Added optional status LED (PR #1264) | ||||
| -   Realtime max. brightness now honors brightness factor (fixes #1271) | ||||
| -   Updated ArduinoJSON to 6.17.0 | ||||
|  | ||||
| #### Build 2010020 | ||||
|  | ||||
| -   Fixed interaction of `T` and `NL` HTTP API commands (#1214) | ||||
| -   Fixed an issue where Sunrise mode nightlight does not activate if toggled on simultaneously  | ||||
|  | ||||
| #### Build 2009291 | ||||
|  | ||||
| -   Fixed MQTT bootloop (no F() macro, #1199) | ||||
|  | ||||
| #### Build 2009290 | ||||
|  | ||||
| -   Added basic DDP protocol support | ||||
| -   Added Washing Machine effect (PR #1208) | ||||
|  | ||||
| #### Build 2009260 | ||||
|  | ||||
| -   Added Loxone parser (PR #1185) | ||||
| -   Added support for kelvin input via `K=` HTTP and `"col":[[val]]` JSON API calls | ||||
|     _Notice:_ `"col":[[val]]` removed in build 2011200, use `"col":[val]` | ||||
| -   Added supplementary UDP socket (#1205) | ||||
| -   TMP2.net receivable by default | ||||
| -   UDP sockets accept HTTP and JSON API commands | ||||
| -   Fixed missing timezones (#1201) | ||||
|  | ||||
| #### Build 2009202 | ||||
|  | ||||
| -   Fixed LPD8806 compilation | ||||
|  | ||||
| #### Build 2009201 | ||||
|  | ||||
| -   Added support for preset cycle toggling using CY=2 | ||||
| -   Added ESP32 touch pin support (#1190) | ||||
| -   Fixed modem sleep on ESP8266 (#1184) | ||||
|  | ||||
| #### Build 2009200 | ||||
|  | ||||
| -   Increased available heap memory by 4kB | ||||
| -   Use F() macro for the majority of strings | ||||
| -   Restructure timezone code | ||||
| -   Restructured settings saved code | ||||
| -   Updated ArduinoJSON to 6.16.1 | ||||
|  | ||||
| #### Build 2009170 | ||||
|  | ||||
| -   New WLED logo on Welcome screen (#1164) | ||||
| -   Fixed 170th pixel dark in E1.31 | ||||
|  | ||||
| #### Build 2009100 | ||||
|  | ||||
| -   Fixed sunrise mode not reinitializing | ||||
| -   Fixed passwords not clearable | ||||
|  | ||||
| #### Build 2009070 | ||||
|  | ||||
| -   New Segments are now initialized with default speed and intensity | ||||
|  | ||||
| #### Build 2009030 | ||||
|  | ||||
| -   Fixed bootloop if mDNS is used on builds without OTA support | ||||
|  | ||||
| ### WLED version 0.10.2 | ||||
|  | ||||
| #### Build 2008310 | ||||
|  | ||||
| -   Added new logo | ||||
| -   Maximum GZIP compression (#1126) | ||||
| -   Enable WebSockets by default | ||||
|  | ||||
| ### Development versions between 0.10.0 and 0.10.2 releases | ||||
|  | ||||
| #### Build 2008300 | ||||
|  | ||||
| -   Added new UI customization options to UI settings | ||||
| -   Added Dancing Shadows effect (#1108) | ||||
| -   Preset cycle is now paused if lights turned off or nightlight active | ||||
| -   Removed `esp01` and `esp01_ota` envs from travis build (need too much flash) | ||||
|  | ||||
| #### Build 2008290 | ||||
|  | ||||
| -   Added individual LED control support to JSON API | ||||
| -   Added internal Segment Freeze/Pause option | ||||
|  | ||||
| #### Build 2008250 | ||||
|  | ||||
| -   Made `platformio_override.ini` example easier to use by including the `default_envs` property | ||||
| -   FastLED uses `now` as timer, so effects using e.g. `beatsin88()` will sync correctly | ||||
| -   Extended the speed range of Pacifica effect | ||||
| -   Improved TPM2.net receiving (#1100) | ||||
| -   Fixed exception on empty MQTT payload (#1101) | ||||
|  | ||||
| #### Build 2008200 | ||||
|  | ||||
| -   Added segment mirroring to web UI | ||||
| -   Fixed segment mirroring when in reverse mode | ||||
|  | ||||
| #### Build 2008140 | ||||
|  | ||||
| -   Removed verbose live mode info from `<ds>` in HTTP API response | ||||
|  | ||||
| #### Build 2008100 | ||||
|  | ||||
| -   Fixed Auto White mode setting (fixes #1088) | ||||
|  | ||||
| #### Build 2008070 | ||||
|  | ||||
| -   Added segment mirroring (`mi` property) (#1017) | ||||
| -   Fixed DMX settings page not displayed (#1070) | ||||
| -   Fixed ArtNet multi universe and improve code style (#1076) | ||||
| -   Renamed global var `local` to `localTime` (#1078) | ||||
|  | ||||
| #### Build 2007190 | ||||
|  | ||||
| -   Fixed hostname containing illegal characters (#1035) | ||||
|  | ||||
| #### Build 2006251 | ||||
|  | ||||
| -   Added `SV=2` to HTTP API, allow selecting single segment only | ||||
|  | ||||
| #### Build 2006250 | ||||
|  | ||||
| -   Fix Alexa not turning off white channel (fixes #1012) | ||||
|  | ||||
| #### Build 2006220 | ||||
|  | ||||
| -   Added Sunrise nightlight mode | ||||
| -   Added Chunchun effect | ||||
| -   Added `LO` (live override) command to HTTP API | ||||
| -   Added `mode` to `nl` object of JSON state API, deprecating `fade` | ||||
| -   Added light color scheme support to web UI (click sun next to brightness slider) | ||||
| -   Added option to hide labels in web UI (click flame icon next to intensity slider) | ||||
| -   Added hex color input (click palette icon next to palette select) (resolves #506) | ||||
| -   Added support for RGB sliders (need to set in localstorage) | ||||
| -   Added support for custom background color or image (need to set in localstorage) | ||||
| -   Added option to hide bottom tab bar in PC mode (need to set in localstorage) | ||||
| -   Fixed transition lag with multiple segments (fixes #985) | ||||
| -   Changed Nightlight wording (resolves #940) | ||||
|  | ||||
| #### Build 2006060 | ||||
|  | ||||
| -   Added five effects by Andrew Tuline (Phased, Phased Noise, Sine, Noise Pal and Twinkleup) | ||||
| -   Added two new effects by Aircoookie (Sunrise and Flow) | ||||
| -   Added US-style sequence to traffic light effect | ||||
| -   Merged pull request #964 adding 9 key IR remote | ||||
|  | ||||
| #### Build 2005280 | ||||
|  | ||||
| -   Added v2 usermod API | ||||
| -   Added v2 example usermod `usermod_v2_example` in the usermods folder as prelimary documentation | ||||
| -   Added DS18B20 Temperature usermod with Info page support | ||||
| -   Disabled MQTT on ESP01 build to make room in flash | ||||
|  | ||||
| #### Build 2005230 | ||||
|  | ||||
| -   Fixed TPM2 | ||||
|  | ||||
| #### Build 2005220 | ||||
|  | ||||
| -   Added TPM2.NET protocol support (need to set WLED broadcast UDP port to 65506) | ||||
| -   Added TPM2 protocol support via Serial | ||||
| -   Support up to 6553 seconds preset cycle durations (backend, NOT yet in UI) | ||||
| -   Merged pull request #591 fixing WS2801 color order | ||||
| -   Merged pull request #858 adding fully featured travis builds | ||||
| -   Merged pull request #862 adding DMX proxy feature | ||||
|  | ||||
| #### Build 2005100 | ||||
|  | ||||
| -   Update to Espalexa v2.4.6 (+1.6kB free heap memory) | ||||
| -   Added `m5atom` PlatformIO environment | ||||
|  | ||||
| #### Build 2005090 | ||||
|  | ||||
| -   Default to ESP8266 Arduino core v2.7.1 in PlatformIO | ||||
| -   Fixed Preset Slot 16 always indicating as empty (#891) | ||||
| -   Disabled Alexa emulation by default (causes bootloop for some users) | ||||
| -   Added BWLT11 and SHOJO_PCB defines to NpbWrapper | ||||
| -   Merged pull request #898 adding Solid Glitter effect | ||||
|  | ||||
| ### WLED version 0.10.0 | ||||
|  | ||||
| #### Build 2005030 | ||||
|  | ||||
| -   DMX Single RGW and Single DRGB modes now support an additional white channel | ||||
| -   Improved palettes derived from set colors and changed their names | ||||
|  | ||||
| ### Development versions between 0.9.1 and 0.10.0 release | ||||
|  | ||||
| #### Build 2005020 | ||||
|  | ||||
| -   Added ACST and ACST/ACDT timezones | ||||
|  | ||||
| #### Build 2005010 | ||||
|  | ||||
| -   Added module info page to web UI | ||||
| -   Added realtime override functionality to web UI | ||||
| -   Added individial segment power and brightness to web UI | ||||
| -   Added feature to one-click select single segment only by tapping segment name | ||||
| -   Removed palette jumping to default if color is changed | ||||
|  | ||||
| #### Build 2004300 | ||||
|  | ||||
| -   Added realtime override option and `lor` JSON property | ||||
| -   Added `lm` (live mode) and `lip` (live IP) properties to info in JSON API | ||||
| -   Added reset commands to APIs | ||||
| -   Added `json/si`, returning state and info, but no FX or Palette lists | ||||
| -   Added rollover detection to millis(). Can track uptimes longer than 49 days | ||||
| -   Attempted to fix Wifi issues with Unifi brand APs | ||||
|  | ||||
| #### Build 2004230 | ||||
|  | ||||
| -   Added brightness and power for individual segments | ||||
| -   Added `on` and `bri` properties to Segment object in JSON API | ||||
| -   Added `C3` an `SB` commands to HTTP get API | ||||
| -   Merged pull request #865 for 5CH_Shojo_PCB environment | ||||
|  | ||||
| #### Build 2004220 | ||||
|  | ||||
| -   Added Candle Multi effect | ||||
| -   Added Palette capability to Pacifica effect | ||||
|  | ||||
| #### Build 2004190 | ||||
|  | ||||
| -   Added TM1814 type LED defines | ||||
|  | ||||
| #### Build 2004120 | ||||
|  | ||||
| -   Added Art-Net support | ||||
| -   Added OTA platform to platformio.ini | ||||
|  | ||||
| #### Build 2004100 | ||||
|  | ||||
| -   Fixed DMX output compilation | ||||
| -   Added DMX start LED setting | ||||
|  | ||||
| #### Build 2004061 | ||||
|  | ||||
| -   Fixed RBG and BGR getPixelColor (#825) | ||||
| -   Improved formatting | ||||
|  | ||||
| #### Build 2004060 | ||||
|  | ||||
| -   Consolidated global variables in wled.h | ||||
|  | ||||
| #### Build 2003300 | ||||
|  | ||||
| -   Major change of project structure from .ino to .cpp and func_declare.h | ||||
|  | ||||
| #### Build 2003262 | ||||
|  | ||||
| -   Fixed compilation for Analog LEDs | ||||
| -   Fixed sync settings network port fields too small | ||||
|  | ||||
| #### Build 2003261 | ||||
|  | ||||
| -   Fixed live preview not displaying whole light if over 255 LEDs | ||||
|  | ||||
| #### Build 2003251 | ||||
|  | ||||
| -   Added Pacifica effect (tentative, doesn't yet support other colors) | ||||
| -   Added Atlantica palette | ||||
| -   Fixed ESP32 build of Espalexa | ||||
|  | ||||
| #### Build 2003222 | ||||
|  | ||||
| -   Fixed Alexa Whites on non-RGBW lights (bump Espalexa to 2.4.5) | ||||
|  | ||||
| #### Build 2003221 | ||||
|  | ||||
| -   Moved Cronixie driver from FX library to drawOverlay handler | ||||
|  | ||||
| #### Build 2003211 | ||||
|  | ||||
| -   Added custom mapping compile define to FX_fcn.h | ||||
| -   Merged pull request #784 by @TravisDean: Fixed initialization bug when toggling skip first | ||||
| -   Added link to youtube videos by Room31 to readme | ||||
|  | ||||
| #### Build 2003141 | ||||
|  | ||||
| -   Fixed color of main segment returned in JSON API during transition not being target color (closes #765) | ||||
| -   Fixed arlsLock() being called after pixels set in E1.31 (closes #772) | ||||
| -   Fixed HTTP API calls not having an effect if no segment selected (now applies to main segment) | ||||
|  | ||||
| #### Build 2003121 | ||||
|  | ||||
| -   Created changelog.md - make tracking changes to code easier | ||||
| -   Merged pull request #766 by @pille: Fix E1.31 out-of sequence detection | ||||
|  | ||||
							
								
								
									
										76
									
								
								CODE_OF_CONDUCT.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,76 @@ | ||||
| # Contributor Covenant Code of Conduct | ||||
|  | ||||
| ## Our Pledge | ||||
|  | ||||
| In the interest of fostering an open and welcoming environment, we as | ||||
| contributors and maintainers pledge to making participation in our project and | ||||
| our community a harassment-free experience for everyone, regardless of age, body | ||||
| size, disability, ethnicity, sex characteristics, gender identity and expression, | ||||
| level of experience, education, socio-economic status, nationality, personal | ||||
| appearance, race, religion, or sexual identity and orientation. | ||||
|  | ||||
| ## Our Standards | ||||
|  | ||||
| Examples of behavior that contributes to creating a positive environment | ||||
| include: | ||||
|  | ||||
| * Using welcoming and inclusive language | ||||
| * Being respectful of differing viewpoints and experiences | ||||
| * Gracefully accepting constructive criticism | ||||
| * Focusing on what is best for the community | ||||
| * Showing empathy towards other community members | ||||
|  | ||||
| Examples of unacceptable behavior by participants include: | ||||
|  | ||||
| * The use of sexualized language or imagery and unwelcome sexual attention or | ||||
|  advances | ||||
| * Trolling, insulting/derogatory comments, and personal or political attacks | ||||
| * Public or private harassment | ||||
| * Publishing others' private information, such as a physical or electronic | ||||
|  address, without explicit permission | ||||
| * Other conduct which could reasonably be considered inappropriate in a | ||||
|  professional setting | ||||
|  | ||||
| ## Our Responsibilities | ||||
|  | ||||
| Project maintainers are responsible for clarifying the standards of acceptable | ||||
| behavior and are expected to take appropriate and fair corrective action in | ||||
| response to any instances of unacceptable behavior. | ||||
|  | ||||
| Project maintainers have the right and responsibility to remove, edit, or | ||||
| reject comments, commits, code, wiki edits, issues, and other contributions | ||||
| that are not aligned to this Code of Conduct, or to ban temporarily or | ||||
| permanently any contributor for other behaviors that they deem inappropriate, | ||||
| threatening, offensive, or harmful. | ||||
|  | ||||
| ## Scope | ||||
|  | ||||
| This Code of Conduct applies both within project spaces and in public spaces | ||||
| when an individual is representing the project or its community. Examples of | ||||
| representing a project or community include using an official project e-mail | ||||
| address, posting via an official social media account, or acting as an appointed | ||||
| representative at an online or offline event. Representation of a project may be | ||||
| further defined and clarified by project maintainers. | ||||
|  | ||||
| ## Enforcement | ||||
|  | ||||
| Instances of abusive, harassing, or otherwise unacceptable behavior may be | ||||
| reported by contacting the project team at dev.aircoookie@gmail.com. All | ||||
| complaints will be reviewed and investigated and will result in a response that | ||||
| is deemed necessary and appropriate to the circumstances. The project team is | ||||
| obligated to maintain confidentiality with regard to the reporter of an incident. | ||||
| Further details of specific enforcement policies may be posted separately. | ||||
|  | ||||
| Project maintainers who do not follow or enforce the Code of Conduct in good | ||||
| faith may face temporary or permanent repercussions as determined by other | ||||
| members of the project's leadership. | ||||
|  | ||||
| ## Attribution | ||||
|  | ||||
| This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4, | ||||
| available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html | ||||
|  | ||||
| [homepage]: https://www.contributor-covenant.org | ||||
|  | ||||
| For answers to common questions about this code of conduct, see | ||||
| https://www.contributor-covenant.org/faq | ||||
							
								
								
									
										5
									
								
								images/Readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | ||||
| ### Additional Logos | ||||
|  | ||||
| Additional awesome logos for WLED can be found here [Aircoookie/Akemi](https://github.com/Aircoookie/Akemi). | ||||
|  | ||||
| <img src="https://github.com/Aircoookie/Akemi/blob/master/akemi/001_cheerful.png"> | ||||
							
								
								
									
										
											BIN
										
									
								
								images/macbook-pro-space-gray-on-the-wooden-table.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 489 KiB | 
							
								
								
									
										
											BIN
										
									
								
								images/walking-with-iphone-x.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 500 KiB | 
							
								
								
									
										
											BIN
										
									
								
								images/wled_logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 40 KiB | 
							
								
								
									
										
											BIN
										
									
								
								images/wled_logo_akemi.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 22 KiB | 
							
								
								
									
										
											BIN
										
									
								
								images/wled_logo_clean.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 602 B | 
							
								
								
									
										2312
									
								
								package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										31
									
								
								package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,31 @@ | ||||
| { | ||||
|   "name": "wled", | ||||
|   "version": "0.12.0", | ||||
|   "description": "Tools for WLED project", | ||||
|   "main": "tools/cdata.js", | ||||
|   "directories": { | ||||
|     "lib": "lib", | ||||
|     "test": "test" | ||||
|   }, | ||||
|   "scripts": { | ||||
|     "build": "node tools/cdata.js", | ||||
|     "dev": "nodemon -e js,html,htm,css,png,jpg,gif,ico,js -w tools/ -w wled00/data/ -x node tools/cdata.js" | ||||
|   }, | ||||
|   "repository": { | ||||
|     "type": "git", | ||||
|     "url": "git+https://github.com/Aircoookie/WLED.git" | ||||
|   }, | ||||
|   "author": "", | ||||
|   "license": "ISC", | ||||
|   "bugs": { | ||||
|     "url": "https://github.com/Aircoookie/WLED/issues" | ||||
|   }, | ||||
|   "homepage": "https://github.com/Aircoookie/WLED#readme", | ||||
|   "dependencies": { | ||||
|     "clean-css": "^4.2.3", | ||||
|     "html-minifier-terser": "^5.1.1", | ||||
|     "inliner": "^1.13.1", | ||||
|     "nodemon": "^2.0.4", | ||||
|     "zlib": "^1.0.5" | ||||
|   } | ||||
| } | ||||
							
								
								
									
										23
									
								
								pio-scripts/gzip-firmware.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,23 @@ | ||||
| Import('env') | ||||
| import os | ||||
| import shutil | ||||
| import gzip | ||||
|  | ||||
| OUTPUT_DIR = "build_output{}".format(os.path.sep) | ||||
|  | ||||
| def bin_gzip(source, target, env): | ||||
|     variant = str(target[0]).split(os.path.sep)[2] | ||||
|      | ||||
|     # create string with location and file names based on variant | ||||
|     bin_file = "{}firmware{}{}.bin".format(OUTPUT_DIR, os.path.sep, variant) | ||||
|     gzip_file = "{}firmware{}{}.bin.gz".format(OUTPUT_DIR, os.path.sep, variant) | ||||
|  | ||||
|     # check if new target files exist and remove if necessary | ||||
|     if os.path.isfile(gzip_file): os.remove(gzip_file) | ||||
|  | ||||
|     # write gzip firmware file | ||||
|     with open(bin_file,"rb") as fp: | ||||
|         with gzip.open(gzip_file, "wb", compresslevel = 9) as f: | ||||
|             shutil.copyfileobj(fp, f) | ||||
|  | ||||
| env.AddPostAction("$BUILD_DIR/${PROGNAME}.bin", [bin_gzip]) | ||||
							
								
								
									
										34
									
								
								pio-scripts/name-firmware.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,34 @@ | ||||
| Import('env') | ||||
| import os | ||||
| import shutil | ||||
|  | ||||
| OUTPUT_DIR = "build_output{}".format(os.path.sep) | ||||
|  | ||||
| def bin_rename_copy(source, target, env): | ||||
|     variant = str(target[0]).split(os.path.sep)[2] | ||||
|      | ||||
|     # check if output directories exist and create if necessary | ||||
|     if not os.path.isdir(OUTPUT_DIR): | ||||
|         os.mkdir(OUTPUT_DIR) | ||||
|  | ||||
|     for d in ['firmware', 'map']: | ||||
|         if not os.path.isdir("{}{}".format(OUTPUT_DIR, d)): | ||||
|             os.mkdir("{}{}".format(OUTPUT_DIR, d)) | ||||
|  | ||||
|     # create string with location and file names based on variant | ||||
|     map_file = "{}map{}{}.map".format(OUTPUT_DIR, os.path.sep, variant) | ||||
|     bin_file = "{}firmware{}{}.bin".format(OUTPUT_DIR, os.path.sep, variant) | ||||
|  | ||||
|     # check if new target files exist and remove if necessary | ||||
|     for f in [map_file, bin_file]: | ||||
|         if os.path.isfile(f): | ||||
|             os.remove(f) | ||||
|  | ||||
|     # copy firmware.bin to firmware/<variant>.bin | ||||
|     shutil.copy(str(target[0]), bin_file) | ||||
|  | ||||
|     # copy firmware.map to map/<variant>.map | ||||
|     if os.path.isfile("firmware.map"): | ||||
|         shutil.move("firmware.map", map_file) | ||||
|  | ||||
| env.AddPostAction("$BUILD_DIR/${PROGNAME}.bin", [bin_rename_copy]) | ||||
							
								
								
									
										9
									
								
								pio-scripts/obj-dump.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | ||||
| # Little convenience script to get an object dump | ||||
|  | ||||
| Import('env') | ||||
|  | ||||
| def obj_dump_after_elf(source, target, env): | ||||
|     print("Create firmware.asm") | ||||
|     env.Execute("xtensa-lx106-elf-objdump "+ "-D " + str(target[0]) + " > "+ "${PROGNAME}.asm") | ||||
|      | ||||
| env.AddPostAction("$BUILD_DIR/${PROGNAME}.elf", [obj_dump_after_elf]) | ||||
							
								
								
									
										15
									
								
								pio-scripts/strip-floats.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| Import('env') | ||||
|  | ||||
| # | ||||
| # Dump build environment (for debug) | ||||
| #print env.Dump() | ||||
| # | ||||
|  | ||||
| flags = " ".join(env['LINKFLAGS']) | ||||
| flags = flags.replace("-u _printf_float", "") | ||||
| flags = flags.replace("-u _scanf_float", "") | ||||
| newflags = flags.split() | ||||
|  | ||||
| env.Replace( | ||||
|   LINKFLAGS=newflags | ||||
| ) | ||||
							
								
								
									
										9
									
								
								pio-scripts/user_config_copy.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | ||||
| Import('env') | ||||
| import os | ||||
| import shutil | ||||
|  | ||||
| # copy WLED00/my_config_sample.h to WLED00/my_config.h | ||||
| if os.path.isfile("wled00/my_config.h"): | ||||
|     print ("*** use existing my_config.h ***") | ||||
| else:  | ||||
|     shutil.copy("wled00/my_config_sample.h", "wled00/my_config.h") | ||||
							
								
								
									
										683
									
								
								platformio.ini
									
									
									
									
									
								
							
							
						
						| @@ -2,258 +2,507 @@ | ||||
| ; Please visit documentation: https://docs.platformio.org/page/projectconf.html | ||||
|  | ||||
| [platformio] | ||||
| src_dir = ./wled00 | ||||
| data_dir = ./wled00/data | ||||
| ;lib_extra_dirs = ./wled00/src | ||||
| lib_dir = ./wled00/src | ||||
| ; Please uncomment one of the 5 lines below to select your board | ||||
| default_envs = nodemcuv2 | ||||
| ; default_envs = esp01 | ||||
| ; default_envs = esp01_1m | ||||
| # ------------------------------------------------------------------------------ | ||||
| # ENVIRONMENTS | ||||
| # | ||||
| # Please uncomment one of the lines below to select your board(s) | ||||
| # ------------------------------------------------------------------------------ | ||||
|  | ||||
| # Travis CI binaries (comment this out with a ';' when building for your own board) | ||||
| ;default_envs = travis_esp8266, travis_esp32 | ||||
|  | ||||
| # Release binaries | ||||
| default_envs = nodemcuv2, esp01_1m_full, esp32dev, esp32_eth | ||||
|  | ||||
| # Single binaries (uncomment your board) | ||||
| ; default_envs = nodemcuv2 | ||||
| ; 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_4CH_H801 | ||||
| ; default_envs = esp8285_5CH_H801 | ||||
| ; default_envs = d1_mini_5CH_Shojo_PCB | ||||
| ; default_envs = wemos_shield_esp32 | ||||
| ; default_envs = m5atom | ||||
| ; default_envs = esp32_eth | ||||
|  | ||||
| src_dir  = ./wled00 | ||||
| data_dir = ./wled00/data | ||||
| build_cache_dir = ~/.buildcache | ||||
| extra_configs = | ||||
|   platformio_override.ini | ||||
|  | ||||
| [common] | ||||
| framework = arduino | ||||
| monitor_speed = 115200 | ||||
| board_build.flash_mode = dout | ||||
| upload_speed = 115200 | ||||
| upload_speed_fast = 921600 | ||||
| build_flags = | ||||
|   -w ; supresses all C/C++ warnings | ||||
|   ; -D VERSION=0.8.5 | ||||
|   ; -D DEBUG | ||||
|   #build_flags for the IRremoteESP8266 library (enabled decoders have to appear here) | ||||
|   -D _IR_ENABLE_DEFAULT_=false  | ||||
|   -D DECODE_HASH=true  | ||||
|   -D DECODE_NEC=true | ||||
|   -D DECODE_SONY=true  | ||||
|   -D DECODE_SAMSUNG=true | ||||
|   -D DECODE_LG=true | ||||
|  | ||||
| # TODO replace libs in /lib with managed libs in here if possible. | ||||
| # If they are not changed it's just a matter of setting the correct version and change the import statement | ||||
| lib_deps_external = | ||||
|   #Blynk@0.5.4(changed) | ||||
|   #E131@1.0.0(changed) | ||||
|   FastLED@3.3.2 | ||||
|   NeoPixelBus@2.5.6 | ||||
|   ESPAsyncTCP@1.2.0 | ||||
|   ESPAsyncUDP@697c75a025 | ||||
|   AsyncTCP@1.0.3 | ||||
|   Esp Async WebServer@1.2.0 | ||||
|   #ArduinoJson@5.13.5 | ||||
|   #IRremoteESP8266@2.7.2 | ||||
|   https://github.com/crankyoldgit/IRremoteESP8266.git | ||||
|   #Time@1.5 | ||||
|   #Timezone@1.2.1 | ||||
|   #For use SSD1306 OLED display uncomment following | ||||
|   #U8g2@~2.27.2 | ||||
|   #For Dallas sensor uncomment following 2 lines | ||||
|   #DallasTemperature@~3.8.0 | ||||
|   #OneWire@~2.3.5 | ||||
| [common:esp8266] | ||||
| # ------------------------------------------------------------------------------ | ||||
| # PLATFORM: | ||||
| #   !! DO NOT confuse platformio's ESP8266 development platform with Arduino core for ESP8266 | ||||
| #   We use Arduino Core 2.5.0 (platformIO 2.0.4) as default | ||||
| # | ||||
| #   arduino core 2.3.0 = platformIO 1.5.0 | ||||
| #   arduino core 2.4.0 = platformIO 1.6.0 | ||||
| #   arduino core 2.4.1 = platformIO 1.7.3 | ||||
| #   arduino core 2.4.2 = platformIO 1.8.0 | ||||
| #   arduino core 2.5.0 = platformIO 2.0.4 | ||||
| #   arduino core stage = platformIO feature#stage | ||||
| #   arduino core 2.6.3 = platformIO 2.3.2 | ||||
| #   arduino core 2.7.0 = platformIO 2.5.0 | ||||
| # ------------------------------------------------------------------------------ | ||||
| arduino_core_2_3_0 = espressif8266@1.5.0 | ||||
| arduino_core_2_4_0 = espressif8266@1.6.0 | ||||
| arduino_core_2_4_1 = espressif8266@1.7.3 | ||||
| arduino_core_2_4_2 = espressif8266@1.8.0 | ||||
| arduino_core_2_5_0 = espressif8266@2.0.4 | ||||
| arduino_core_2_5_2 = espressif8266@2.2.3 | ||||
| arduino_core_2_6_1 = espressif8266@2.3.0 | ||||
| arduino_core_2_6_2 = espressif8266@2.3.1 | ||||
| arduino_core_stage = https://github.com/platformio/platform-espressif8266.git#feature/stage | ||||
| platform = ${common:esp8266.arduino_core_2_6_2} | ||||
| build_flags = | ||||
|   -D ESP8266 | ||||
|   -D PIO_FRAMEWORK_ARDUINO_LWIP2_HIGHER_BANDWIDTH | ||||
|   -Wl,-Teagle.flash.4m1m.ld  ;;;; Required for core > v2.5.0 or staging version 4MB Flash 3MB SPIFFs | ||||
| arduino_core_2_6_3 = espressif8266@2.3.3 | ||||
| arduino_core_2_7_4 = espressif8266@2.6.2 | ||||
|  | ||||
| [common:esp8266_1M] | ||||
| platform = espressif8266@1.8.0 | ||||
| build_flags = | ||||
|   -D PIO_FRAMEWORK_ARDUINO_LWIP2_HIGHER_BANDWIDTH_LOW_FLASH | ||||
|   -Wl,-Teagle.flash.1m0.ld  ;;;; Compile with no SPIFFS to leave space for OTA | ||||
|   ; -D WLED_DISABLE_MOBILE_UI | ||||
|   ; -D WLED_DISABLE_OTA | ||||
|   ; -D WLED_DISABLE_ALEXA | ||||
|   -D WLED_DISABLE_BLYNK | ||||
|   -D WLED_DISABLE_CRONIXIE | ||||
|   ; -D WLED_DISABLE_HUESYNC | ||||
|   ; -D WLED_DISABLE_INFRARED | ||||
| # Development platforms | ||||
| arduino_core_develop = https://github.com/platformio/platform-espressif8266#develop | ||||
| arduino_core_git = https://github.com/platformio/platform-espressif8266#feature/stage | ||||
|  | ||||
| [common:esp8266_512k] | ||||
| platform = espressif8266@1.8.0 | ||||
| build_flags = | ||||
|   -D PIO_FRAMEWORK_ARDUINO_LWIP2_HIGHER_BANDWIDTH_LOW_FLASH | ||||
|   -Wl,-Teagle.flash.512k0.ld  ;;;; Compile with no SPIFFS | ||||
|   ; -D WLED_DISABLE_MOBILE_UI | ||||
|   -D WLED_DISABLE_OTA | ||||
|   ; -D WLED_DISABLE_ALEXA | ||||
|   -D WLED_DISABLE_BLYNK | ||||
|   -D WLED_DISABLE_CRONIXIE | ||||
|   -D WLED_DISABLE_HUESYNC | ||||
|   ; -D WLED_DISABLE_INFRARED | ||||
| # Platform to use for ESP8266 | ||||
| platform_wled_default = ${common.arduino_core_2_7_4} | ||||
| # We use 2.7.4.7 for all, includes PWM flicker fix and Wstring optimization | ||||
| platform_packages = tasmota/framework-arduinoespressif8266 @ 3.20704.7 | ||||
|                     platformio/toolchain-xtensa @ ~2.40802.200502 | ||||
|                     platformio/tool-esptool @ ~1.413.0 | ||||
|                     platformio/tool-esptoolpy @ ~1.30000.0 | ||||
|  | ||||
| [common:esp32] | ||||
| platform = espressif32@1.11.1 | ||||
| build_flags = | ||||
|   -D PIO_FRAMEWORK_ARDUINO_LWIP2_HIGHER_BANDWIDTH | ||||
|   -D ARDUINO_ARCH_ESP32 | ||||
| # ------------------------------------------------------------------------------ | ||||
| # FLAGS: DEBUG | ||||
| # | ||||
| # ------------------------------------------------------------------------------ | ||||
| debug_flags = -D DEBUG=1 -D WLED_DEBUG -DDEBUG_ESP_WIFI -DDEBUG_ESP_HTTP_CLIENT -DDEBUG_ESP_HTTP_UPDATE -DDEBUG_ESP_HTTP_SERVER -DDEBUG_ESP_UPDATER -DDEBUG_ESP_OTA -DDEBUG_TLS_MEM | ||||
| #if needed (for memleaks etc) also add; -DDEBUG_ESP_OOM -include "umm_malloc/umm_malloc_cfg.h" | ||||
| #-DDEBUG_ESP_CORE is not working right now | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # FLAGS: ldscript (available ldscripts at https://github.com/esp8266/Arduino/tree/master/tools/sdk/ld) | ||||
| #    ldscript_512k ( 512 KB) =  487 KB sketch, 4 KB eeprom,      no spiffs, 16 KB reserved | ||||
| #    ldscript_1m0m (1024 KB) =  999 KB sketch, 4 KB eeprom,      no spiffs, 16 KB reserved | ||||
| #    ldscript_2m1m (2048 KB) = 1019 KB sketch, 4 KB eeprom, 1004 KB spiffs, 16 KB reserved | ||||
| #    ldscript_4m1m (4096 KB) = 1019 KB sketch, 4 KB eeprom, 1002 KB spiffs, 16 KB reserved, 2048 KB empty/ota? | ||||
| #    ldscript_4m3m (4096 KB) = 1019 KB sketch, 4 KB eeprom, 3040 KB spiffs, 16 KB reserved | ||||
| # | ||||
| # Available lwIP variants (macros): | ||||
| #    -DPIO_FRAMEWORK_ARDUINO_LWIP_HIGHER_BANDWIDTH  = v1.4 Higher Bandwidth (default) | ||||
| #    -DPIO_FRAMEWORK_ARDUINO_LWIP2_LOW_MEMORY       = v2 Lower Memory | ||||
| #    -DPIO_FRAMEWORK_ARDUINO_LWIP2_HIGHER_BANDWIDTH = v2 Higher Bandwidth | ||||
| #    -DPIO_FRAMEWORK_ARDUINO_LWIP2_HIGHER_BANDWIDTH_LOW_FLASH | ||||
| # | ||||
| # BearSSL performance: | ||||
| #  When building with -DSECURE_CLIENT=SECURE_CLIENT_BEARSSL, please add `board_build.f_cpu = 160000000` to the environment configuration | ||||
| # | ||||
| # BearSSL ciphers: | ||||
| #   When building on core >= 2.5, you can add the build flag -DBEARSSL_SSL_BASIC in order to build BearSSL with a limited set of ciphers: | ||||
| #     TLS_RSA_WITH_AES_128_CBC_SHA256 / AES128-SHA256 | ||||
| #     TLS_RSA_WITH_AES_256_CBC_SHA256 / AES256-SHA256 | ||||
| #     TLS_RSA_WITH_AES_128_CBC_SHA / AES128-SHA | ||||
| #     TLS_RSA_WITH_AES_256_CBC_SHA / AES256-SHA | ||||
| #  This reduces the OTA size with ~45KB, so it's especially useful on low memory boards (512k/1m). | ||||
| # ------------------------------------------------------------------------------ | ||||
| build_flags = | ||||
|   -DMQTT_MAX_PACKET_SIZE=1024 | ||||
|   -DSECURE_CLIENT=SECURE_CLIENT_BEARSSL | ||||
|   -DBEARSSL_SSL_BASIC | ||||
|   -D CORE_DEBUG_LEVEL=0 | ||||
|   -D NDEBUG | ||||
|   #build_flags for the IRremoteESP8266 library (enabled decoders have to appear here) | ||||
|   -D _IR_ENABLE_DEFAULT_=false | ||||
|   -D DECODE_HASH=true | ||||
|   -D DECODE_NEC=true | ||||
|   -D DECODE_SONY=true | ||||
|   -D DECODE_SAMSUNG=true | ||||
|   -D DECODE_LG=true | ||||
|   -DWLED_USE_MY_CONFIG | ||||
|   ; -D USERMOD_SENSORSTOMQTT | ||||
|  | ||||
| build_unflags = | ||||
|  | ||||
| # enables all features for travis CI | ||||
| build_flags_all_features = | ||||
|   -D WLED_USE_ANALOG_LED | ||||
|   -D WLED_USE_H801 | ||||
|   -D WLED_ENABLE_5CH_LEDS | ||||
|   -D WLED_ENABLE_ADALIGHT | ||||
|   -D WLED_ENABLE_DMX | ||||
|   -D WLED_ENABLE_MQTT | ||||
|   -D WLED_ENABLE_WEBSOCKETS | ||||
|  | ||||
| build_flags_esp8266 = ${common.build_flags}  ${esp8266.build_flags} | ||||
| build_flags_esp32   = ${common.build_flags}  ${esp32.build_flags} | ||||
|  | ||||
| ldscript_1m128k = eagle.flash.1m128.ld | ||||
| ldscript_2m512k = eagle.flash.2m512.ld | ||||
| ldscript_2m1m = eagle.flash.2m1m.ld | ||||
| ldscript_4m1m = eagle.flash.4m1m.ld | ||||
|  | ||||
| [esp8266] | ||||
| build_flags = | ||||
|   -DESP8266 | ||||
|   -DFP_IN_IROM | ||||
| ; NONOSDK22x_190703 = 2.2.2-dev(38a443e) | ||||
|   -DPIO_FRAMEWORK_ARDUINO_ESPRESSIF_SDK22x_190703 | ||||
| ; lwIP 2 - Higher Bandwidth no Features | ||||
| ;  -DPIO_FRAMEWORK_ARDUINO_LWIP2_HIGHER_BANDWIDTH_LOW_FLASH | ||||
| ; lwIP 1.4 - Higher Bandwidth (Aircoookie has) | ||||
|    -DPIO_FRAMEWORK_ARDUINO_LWIP_HIGHER_BANDWIDTH | ||||
| ; VTABLES in Flash | ||||
|   -DVTABLES_IN_FLASH | ||||
| ; restrict to minimal mime-types | ||||
|   -DMIMETYPE_MINIMAL | ||||
|  | ||||
| [esp32] | ||||
| build_flags = -g | ||||
|   -DARDUINO_ARCH_ESP32 | ||||
|   -DCONFIG_LITTLEFS_FOR_IDF_3_2 | ||||
|  | ||||
| [scripts_defaults] | ||||
| extra_scripts             = pio-scripts/name-firmware.py | ||||
|                             pio-scripts/gzip-firmware.py | ||||
|                             pio-scripts/strip-floats.py | ||||
|                             pio-scripts/user_config_copy.py | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # COMMON SETTINGS: | ||||
| # ------------------------------------------------------------------------------ | ||||
| [env] | ||||
| framework = arduino | ||||
| board_build.flash_mode = dout | ||||
| monitor_speed = 115200 | ||||
| # slow upload speed (comment this out with a ';' when building for development use) | ||||
| upload_speed = 115200 | ||||
| # fast upload speed (remove ';' when building for development use) | ||||
| ; upload_speed = 921600 | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # LIBRARIES: required dependencies | ||||
| #   Please note that we don't always use the latest version of a library. | ||||
| # | ||||
| #   The following libraries have been included (and some of them changd) in the source: | ||||
| #     ArduinoJson@5.13.5, Blynk@0.5.4(changed), E131@1.0.0(changed), Time@1.5, Timezone@1.2.1 | ||||
| # ------------------------------------------------------------------------------ | ||||
| lib_compat_mode = strict | ||||
| 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.2 | ||||
|   #For use of the TTGO T-Display ESP32 Module with integrated TFT display uncomment the following line | ||||
|     #TFT_eSPI | ||||
|   #For use SSD1306 OLED display uncomment following | ||||
|     #U8g2@~2.27.2 | ||||
|   #For Dallas sensor uncomment following 2 lines | ||||
|     #OneWire@~2.3.5 | ||||
|     #milesburton/DallasTemperature@^3.9.0 | ||||
|   #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 | ||||
|  | ||||
| lib_ignore = | ||||
|     AsyncTCP | ||||
|  | ||||
| extra_scripts             = ${scripts_defaults.extra_scripts} | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # WLED BUILDS | ||||
| # ------------------------------------------------------------------------------ | ||||
|  | ||||
| # see: http://docs.platformio.org/en/latest/platforms/espressif8266.html | ||||
| [env:nodemcuv2] | ||||
| board = nodemcuv2 | ||||
| platform = ${common:esp8266.platform} | ||||
| monitor_speed = ${common.monitor_speed} | ||||
| upload_speed = ${common.upload_speed} | ||||
| framework = ${common.framework} | ||||
| build_flags = | ||||
|   ${common.build_flags} | ||||
|   ${common:esp8266.build_flags} | ||||
| lib_deps = | ||||
|   ${common.lib_deps_external} | ||||
| lib_compat_mode = strict | ||||
| lib_ignore = AsynTCP | ||||
| 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} | ||||
|  | ||||
| [env:d1_mini] | ||||
| board = d1_mini | ||||
| platform = ${common:esp8266.platform} | ||||
| monitor_speed = ${common.monitor_speed} | ||||
| upload_speed = ${common.upload_speed} | ||||
| framework = ${common.framework} | ||||
| build_flags = | ||||
|   ${common.build_flags} | ||||
|   ${common:esp8266.build_flags} | ||||
| lib_deps = | ||||
|   ${common.lib_deps_external} | ||||
| lib_compat_mode = strict | ||||
| lib_ignore = AsynTCP | ||||
|  | ||||
| [env:esp01_1m] | ||||
| [env:esp01_1m_full] | ||||
| board = esp01_1m | ||||
| platform = ${common:esp8266_1M.platform} | ||||
| monitor_speed = ${common.monitor_speed} | ||||
| upload_speed = ${common.upload_speed} | ||||
| framework = ${common.framework} | ||||
| build_flags = | ||||
|   ${common.build_flags} | ||||
|   ${common:esp8266_1M.build_flags} | ||||
|   # disable IR because there is no pin for it | ||||
|   -D WLED_DISABLE_INFRARED | ||||
| lib_deps = | ||||
|   ${common.lib_deps_external} | ||||
| lib_compat_mode = strict | ||||
| lib_ignore = AsynTCP | ||||
|  | ||||
| [env:esp01] | ||||
| board = esp01 | ||||
| platform = ${common:esp8266_512k.platform} | ||||
| monitor_speed = ${common.monitor_speed} | ||||
| upload_speed = ${common.upload_speed} | ||||
| framework = ${common.framework} | ||||
| build_flags = | ||||
|   ${common.build_flags} | ||||
|   ${common:esp8266_512k.build_flags} | ||||
|   -D WLED_DISABLE_INFRARED | ||||
| lib_deps = | ||||
|   ${common.lib_deps_external} | ||||
| lib_compat_mode = strict | ||||
| lib_ignore = AsynTCP | ||||
| 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 | ||||
|  | ||||
| [env:esp07] | ||||
| board = esp07 | ||||
| platform = ${common:esp8266.platform} | ||||
| monitor_speed = ${common.monitor_speed} | ||||
| upload_speed = ${common.upload_speed} | ||||
| framework = ${common.framework} | ||||
| build_flags = | ||||
|   ${common.build_flags} | ||||
|   ${common:esp8266.build_flags} | ||||
| lib_deps = | ||||
|   ${common.lib_deps_external} | ||||
| lib_compat_mode = strict | ||||
| lib_ignore = AsynTCP | ||||
| 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} | ||||
|  | ||||
| [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} | ||||
| 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} | ||||
|  | ||||
| [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 | ||||
|  | ||||
| # see: http://docs.platformio.org/en/latest/platforms/espressif32.html | ||||
| [env:esp32dev] | ||||
| board = esp32dev | ||||
| platform = ${common:esp32.platform} | ||||
| monitor_speed = ${common.monitor_speed} | ||||
| upload_speed = ${common.upload_speed_fast} | ||||
| framework = ${common.framework} | ||||
| build_flags = | ||||
|   ${common.build_flags} | ||||
|   ${common:esp32.build_flags} | ||||
| lib_deps = | ||||
|   ${common.lib_deps_external} | ||||
| platform = espressif32@3.2 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} | ||||
| lib_ignore = | ||||
|   ESPAsyncTCP | ||||
|   ESPAsyncUDP | ||||
|  | ||||
| [env:esp32_eth] | ||||
| board = esp32-poe | ||||
| platform = espressif32@3.2 | ||||
| upload_speed = 921600 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} -D RLYPIN=-1 -D WLED_USE_ETHERNET -D BTNPIN=-1 | ||||
| lib_ignore = | ||||
|   ESPAsyncTCP | ||||
|   ESPAsyncUDP | ||||
| lib_compat_mode = strict | ||||
|  | ||||
| [env:esp8285_4CH_MagicHome] | ||||
| board = esp8285 | ||||
| platform = ${common:esp8266_1M.platform} | ||||
| monitor_speed = ${common.monitor_speed} | ||||
| upload_speed = ${common.upload_speed} | ||||
| framework = ${common.framework} | ||||
| build_flags = | ||||
|   ${common.build_flags} | ||||
|   ${common:esp8266_1M.build_flags} | ||||
|   -D WLED_DISABLE_HUESYNC | ||||
|   -D WLED_USE_ANALOG_LEDS | ||||
| lib_deps = | ||||
|   ${common.lib_deps_external} | ||||
| lib_compat_mode = strict | ||||
| lib_ignore = AsynTCP | ||||
| 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 -D WLED_USE_ANALOG_LEDS | ||||
|  | ||||
| [env:esp8285_4CH_H801] | ||||
| board = esp8285 | ||||
| platform = ${common:esp8266_1M.platform} | ||||
| monitor_speed = ${common.monitor_speed} | ||||
| upload_speed = ${common.upload_speed} | ||||
| framework = ${common.framework} | ||||
| build_flags = | ||||
|   ${common.build_flags} | ||||
|   ${common:esp8266_1M.build_flags} | ||||
|   -D WLED_DISABLE_HUESYNC | ||||
|   -D WLED_USE_ANALOG_LEDS | ||||
|   -D WLED_USE_H801 | ||||
| lib_deps = | ||||
|   ${common.lib_deps_external} | ||||
| lib_compat_mode = strict | ||||
| lib_ignore = AsynTCP | ||||
| 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 -D WLED_USE_ANALOG_LEDS -D WLED_USE_H801 | ||||
|  | ||||
| [env:esp8285_5CH_H801] | ||||
| board = esp8285 | ||||
| platform = ${common:esp8266_1M.platform} | ||||
| monitor_speed = ${common.monitor_speed} | ||||
| upload_speed = ${common.upload_speed} | ||||
| framework = ${common.framework} | ||||
| build_flags = | ||||
|   ${common.build_flags} | ||||
|   ${common:esp8266_1M.build_flags} | ||||
|   -D WLED_DISABLE_HUESYNC | ||||
|   -D WLED_USE_ANALOG_LEDS | ||||
|   -D WLED_USE_H801 | ||||
|   -D WLED_ENABLE_5CH_LEDS  | ||||
| lib_deps = | ||||
|   ${common.lib_deps_external} | ||||
| lib_compat_mode = strict | ||||
| lib_ignore = AsynTCP | ||||
|    | ||||
| 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 -D WLED_USE_ANALOG_LEDS -D WLED_USE_H801 -D WLED_ENABLE_5CH_LEDS | ||||
|  | ||||
| [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_ANALOG_LEDS -D WLED_USE_SHOJO_PCB -D WLED_ENABLE_5CH_LEDS | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # 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} | ||||
|  | ||||
| [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} | ||||
|  | ||||
| [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 | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # custom board configurations | ||||
| # ------------------------------------------------------------------------------ | ||||
|  | ||||
| [env:custom_LEDPIN_4] | ||||
| 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=4 -D IRPIN=5 | ||||
|  | ||||
| [env:custom_LEDPIN_16] | ||||
| 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=16 | ||||
|  | ||||
|  | ||||
| [env:custom_LEDPIN_3] | ||||
| 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=3 | ||||
|  | ||||
| [env:custom_APA102] | ||||
| 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 USE_APA102 | ||||
|  | ||||
| [env:custom_WS2801] | ||||
| 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 USE_WS2801 | ||||
|  | ||||
| [env:custom32_LEDPIN_16] | ||||
| board = esp32dev | ||||
| platform = espressif32@3.2 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} -D LEDPIN=16 -D RLYPIN=19 | ||||
| lib_ignore = | ||||
|   ESPAsyncTCP | ||||
|   ESPAsyncUDP | ||||
|  | ||||
| [env:custom32_APA102] | ||||
| board = esp32dev | ||||
| platform = espressif32@3.2 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} -D USE_APA102 | ||||
| lib_ignore = | ||||
|   ESPAsyncTCP | ||||
|   ESPAsyncUDP | ||||
|  | ||||
| [env:custom32_TOUCHPIN_T0] | ||||
| board = esp32dev | ||||
| platform = espressif32@3.2 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} -D TOUCHPIN=T0 | ||||
| lib_ignore = | ||||
|   ESPAsyncTCP | ||||
|   ESPAsyncUDP | ||||
|  | ||||
| [env:wemos_shield_esp32] | ||||
| board = esp32dev | ||||
| platform = espressif32@3.2 | ||||
| upload_port = /dev/cu.SLAB_USBtoUART | ||||
| monitor_port = /dev/cu.SLAB_USBtoUART | ||||
| upload_speed = 460800 | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} -D LEDPIN=16 -D RLYPIN=19 -D BTNPIN=17 | ||||
| lib_ignore = | ||||
|   ESPAsyncTCP | ||||
|   ESPAsyncUDP | ||||
|  | ||||
| [env:m5atom] | ||||
| board = esp32dev | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} -D LEDPIN=27 -D BTNPIN=39 | ||||
| lib_ignore = | ||||
| 	ESPAsyncTCP | ||||
| 	ESPAsyncUDP | ||||
| platform = espressif32@3.2 | ||||
|  | ||||
| [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 | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # travis test board configurations | ||||
| # ------------------------------------------------------------------------------ | ||||
|  | ||||
| [env:travis_esp8266] | ||||
| extends = env:d1_mini | ||||
| build_type = debug | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} ${common.debug_flags} ${common.build_flags_all_features} | ||||
|  | ||||
| [env:travis_esp32] | ||||
| extends = env:esp32dev | ||||
| ; build_type = debug | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp32} ${common.debug_flags} ${common.build_flags_all_features} | ||||
|  | ||||
| # ------------------------------------------------------------------------------ | ||||
| # codm pixel controller board configurations | ||||
| # ------------------------------------------------------------------------------ | ||||
|  | ||||
| [env:codm-controller-0.4] | ||||
| 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} -D LEDPIN=3 | ||||
|  | ||||
| [env:codm-controller-0.4-WS2801] | ||||
| 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} -D USE_WS2801 -D CLKPIN=13 -D DATAPIN=3 | ||||
|  | ||||
| [env:codm-controller-0.4-APA102] | ||||
| 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} -D USE_APA102 -D CLKPIN=13 -D DATAPIN=3 | ||||
|  | ||||
| [env:codm-controller-0.5] | ||||
| 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} | ||||
|  | ||||
| [env:codm-controller-0.5-WS2801] | ||||
| 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} -D USE_WS2801 #-D CLKPIN=0 -D DATAPIN=2 | ||||
|  | ||||
| [env:codm-controller-0.5-APA102] | ||||
| 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} -D USE_APA102 #-D CLKPIN=0 -D DATAPIN=2 | ||||
|   | ||||
							
								
								
									
										50
									
								
								platformio_override.ini.sample
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,50 @@ | ||||
| # 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} | ||||
| build_unflags = ${common.build_unflags} | ||||
| build_flags = ${common.build_flags_esp8266} | ||||
| ; ********************************************************************* | ||||
| ; *** Use custom settings from file my_config.h | ||||
|    -DWLED_USE_MY_CONFIG | ||||
| ; ********************************************************************* | ||||
| ;  -D WLED_DISABLE_OTA | ||||
| ;  -D WLED_DISABLE_ALEXA | ||||
| ;  -D WLED_DISABLE_BLYNK | ||||
| ;  -D WLED_DISABLE_CRONIXIE | ||||
| ;  -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), uncomment the following | ||||
| ; PWM pins 5,12,13,15 are used with Magic Home LED Controller (default) | ||||
|    -D WLED_USE_ANALOG_LEDS | ||||
| ; for the H801 controller (PINs 15,13,12,14 (W2 = 04)) uncomment this | ||||
| ;   -D WLED_USE_H801 | ||||
| ; for the BW-LT11 controller (PINs 12,4,14,5 ) uncomment this | ||||
| ;   -D WLED_USE_BWLT11 | ||||
| ; and to enable channel 5 for RGBW-CT led strips this | ||||
| ;   -D WLED_USE_5CH_LEDS | ||||
							
								
								
									
										92
									
								
								readme.md
									
									
									
									
									
								
							
							
						
						| @@ -1,69 +1,101 @@ | ||||
|     | ||||
| <p align="center"> | ||||
|   <img src="/images/wled_logo_akemi.png"> | ||||
|   <a href="https://github.com/Aircoookie/WLED/releases"><img src="https://img.shields.io/github/release/Aircoookie/WLED.svg?style=flat-square"></a> | ||||
|   <a href="https://raw.githubusercontent.com/Aircoookie/WLED/master/LICENSE"><img src="https://img.shields.io/github/license/Aircoookie/wled?color=blue&style=flat-square"></a> | ||||
|   <a href="https://wled.discourse.group"><img src="https://img.shields.io/discourse/topics?colorB=blue&label=forum&server=https%3A%2F%2Fwled.discourse.group%2F&style=flat-square"></a> | ||||
|   <a href="https://discord.gg/KuqP7NE"><img src="https://img.shields.io/discord/473448917040758787.svg?colorB=blue&label=discord&style=flat-square"></a> | ||||
|   <a href="https://github.com/Aircoookie/WLED/wiki"><img src="https://img.shields.io/badge/quick_start-wiki-blue.svg?style=flat-square"></a> | ||||
|   <a href="https://github.com/Aircoookie/WLED-App"><img src="https://img.shields.io/badge/app-wled-blue.svg?style=flat-square"></a> | ||||
|   <a href="https://gitpod.io/#https://github.com/Aircoookie/WLED"><img src="https://img.shields.io/badge/Gitpod-ready--to--code-blue?style=flat-square&logo=gitpod"></a> | ||||
|  | ||||
| [](https://github.com/Aircoookie/WLED/releases) | ||||
| [](https://wled.discourse.group) | ||||
| [](https://discord.gg/KuqP7NE) | ||||
| [](https://github.com/Aircoookie/WLED/wiki) | ||||
| [](https://github.com/Aircoookie/WLED-App) | ||||
|   </p> | ||||
|    | ||||
| # Welcome to my project WLED! ✨ | ||||
|  | ||||
| ## Welcome to my project WLED! | ||||
| A fast and feature-rich implementation of an ESP8266/ESP32 webserver to control NeoPixel (WS2812B, WS2811, SK6812) LEDs or also SPI based chipsets like the WS2801 and APA102! | ||||
|  | ||||
| A fast and feature-rich implementation of an ESP8266/ESP32 webserver to control NeoPixel (WS2812B, WS2811, SK6812, APA102) LEDs! | ||||
|  | ||||
| ### Features: | ||||
| - WS2812FX library integrated for almost 100 special effects   | ||||
| - FastLED noise effects and palettes   | ||||
| ## ⚙️ Features | ||||
| - WS2812FX library integrated for over 100 special effects   | ||||
| - FastLED noise effects and 50 palettes   | ||||
| - Modern UI with color, effect and segment controls   | ||||
| - Segments to set different effects and colors to parts of the LEDs   | ||||
| - Settings page - configuration over network   | ||||
| - Access Point and station mode - automatic failsafe AP   | ||||
| - Up to 10 LED outputs per instance | ||||
| - Support for RGBW strips   | ||||
| - 16 user presets to save and load colors/effects easily, supports cycling through them.   | ||||
| - Macro functions to automatically execute API calls   | ||||
| - Up to 250 user presets to save and load colors/effects easily, supports cycling through them.   | ||||
| - Presets can be used to automatically execute API calls   | ||||
| - Nightlight function (gradually dims down)   | ||||
| - Full OTA software updatability (HTTP + ArduinoOTA), password protectable   | ||||
| - Configurable analog clock + support for the Cronixie kit by Diamex   | ||||
| - Configurable Auto Brightness limit for safer operation   | ||||
| - Filesystem-based config for easier backup of presets and settings   | ||||
|  | ||||
| ### Supported light control interfaces: | ||||
| - WLED app for Android and iOS   | ||||
| ## 💡 Supported light control interfaces | ||||
| - WLED app for [Android](https://play.google.com/store/apps/details?id=com.aircoookie.WLED) and [iOS](https://apps.apple.com/us/app/wled/id1475695033) | ||||
| - JSON and HTTP request APIs   | ||||
| - MQTT   | ||||
| - Blynk IoT   | ||||
| - E1.31   | ||||
| - Hyperion   | ||||
| - E1.31, Art-Net, DDP and TPM2.net | ||||
| - [Hyperion](https://github.com/hyperion-project/hyperion.ng) | ||||
| - UDP realtime   | ||||
| - Alexa voice control (including dimming and color)   | ||||
| - Sync to Philips hue lights   | ||||
| - Adalight (PC ambilight via serial)   | ||||
| - Adalight (PC ambilight via serial) and TPM2   | ||||
| - Sync color of multiple WLED devices (UDP notifier)   | ||||
| - Infrared remotes (24-key RGB, receiver required)   | ||||
| - Simple timers/schedules (time from NTP, timezones/DST supported)   | ||||
|  | ||||
| ### Quick start guide and documentation: | ||||
| ## 📲 Quick start guide and documentation | ||||
|  | ||||
| See the [wiki](https://github.com/Aircoookie/WLED/wiki)! | ||||
|  | ||||
| DrZzs has made some excellent video guides:   | ||||
| [Introduction, hardware and installation](https://www.youtube.com/watch?v=tXvtxwK3jRk)   | ||||
| [Settings, tips and tricks](https://www.youtube.com/watch?v=6eCE2BpLaUQ)   | ||||
| [On this page](https://github.com/Aircoookie/WLED/wiki/Learning-the-ropes) you can find excellent tutorials made by the community and helpful tools to help you get your new lamp up and running! | ||||
|  | ||||
| If you'd rather read, here is a very [detailed step-by-step beginner tutorial](https://tynick.com/blog/11-03-2019/getting-started-with-wled-on-esp8266/) by tynick!   | ||||
| ## 🖼️ Images | ||||
| <img src="/images/macbook-pro-space-gray-on-the-wooden-table.jpg" width="50%"><img src="/images/walking-with-iphone-x.jpg" width="50%"> | ||||
|  | ||||
| ### Other | ||||
| ## 💾 Compatible LED Strips | ||||
| Type | Voltage | Comments | ||||
| |---|---|---| | ||||
| WS2812B | 5v | | ||||
| WS2813 | 5v |  | ||||
| SK6812 | 5v | RGBW | ||||
| APA102 | 5v | C/D | ||||
| WS2801 | 5v | C/D | ||||
| LPD8806 | 5v | C/D | ||||
| TM1814 | 12v | RGBW | ||||
| WS2811 | 12v | 3-LED segments | ||||
| WS2815 | 12v |  | ||||
| GS8208 | 12v | | ||||
| Analog/non-addressable | any | Requires additional circuitry | ||||
|  | ||||
| ## 🧊 Compatible PC RGB Fans and ARGB accessories | ||||
| Brand | Model | Comments | ||||
| |---|---|---| | ||||
| Corsair | HD120 Fan | Uses WS2812B, data-in only | ||||
| PCCOOLER | Moonlight 5-pack Fans | Uses WS2812B, includes Data-out connector to keep each fan uniquely addressable if wired in series like traditional LED strips | ||||
| Any | 5v 3-pin ARGB for PC | Any PC RGB device that supports the 5v 3-pin ARGB motherboard header should work fine with WLED. All the major motherboard vendors support the Corsair HD120 and PCCOOLER fans listed, so we can safely assume any device that supports motherboard ARGB 5V 3-Pin standard will work with WLED. | ||||
|  | ||||
|  | ||||
| ## ✌️ Other | ||||
|  | ||||
| Licensed under the MIT license   | ||||
| Credits [here](https://github.com/Aircoookie/WLED/wiki/Contributors-&-About)! | ||||
|  | ||||
| Uses Linearicons by Perxis! | ||||
|  | ||||
| Join the Discord [server](https://discord.gg/KuqP7NE) to discuss everything about WLED!   | ||||
| Join the Discord server to discuss everything about WLED! | ||||
|  | ||||
| <a href="https://discord.gg/KuqP7NE"><img src="https://discordapp.com/api/guilds/473448917040758787/widget.png?style=banner2" width="25%"></a> | ||||
|  | ||||
| Check out the WLED [Discourse forum](https://wled.discourse.group)!   | ||||
| You can also send me mails to [dev.aircoookie@gmail.com](mailto:dev.aircoookie@gmail.com), but please only do so if you want to talk to me privately.   | ||||
| If WLED really brightens up your every day, you can [](https://paypal.me/aircoookie) | ||||
|  | ||||
| *Disclaimer:*    | ||||
| If you are sensitive to photoeleptic seizures it is not recommended that you use this software.   | ||||
| In case you still want to try, don't use strobe, lighting or noise modes or high effect speed settings. | ||||
| As per the MIT license, i assume no liability for any damage to you or any other person or equipment.   | ||||
|  | ||||
| *Disclaimer:*    | ||||
| If you are sensitive to photosensitive epilepsy it is not recommended that you use this software.   | ||||
| In case you still want to try, don't use strobe, lighting or noise modes or high effect speed settings. | ||||
| As per the MIT license, I assume no liability for any damage to you or any other person or equipment.   | ||||
|  | ||||
|   | ||||
							
								
								
									
										424
									
								
								tools/cdata.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,424 @@ | ||||
| /** | ||||
|  * Writes compressed C arrays of data files (web interface) | ||||
|  * How to use it? | ||||
|  * | ||||
|  * 1) Install Node 11+ and npm | ||||
|  * 2) npm install | ||||
|  * 3) npm run build | ||||
|  * | ||||
|  * If you change data folder often, you can run it in monitoring mode (it will recompile and update *.h on every file change) | ||||
|  * | ||||
|  * > npm run dev | ||||
|  * | ||||
|  * How it works? | ||||
|  * | ||||
|  * It uses NodeJS packages to inline, minify and GZIP files. See writeHtmlGzipped and writeChunks invocations at the bottom of the page. | ||||
|  */ | ||||
|  | ||||
| const fs = require("fs"); | ||||
| const packageJson = require("../package.json"); | ||||
|  | ||||
| /** | ||||
|  * | ||||
|  */ | ||||
| function hexdump(buffer) { | ||||
|   let lines = []; | ||||
|  | ||||
|   for (let i = 0; i < buffer.length; i += 16) { | ||||
|     let block = buffer.slice(i, i + 16); // cut buffer into blocks of 16 | ||||
|     let hexArray = []; | ||||
|  | ||||
|     for (let value of block) { | ||||
|       hexArray.push("0x" + value.toString(16).padStart(2, "0")); | ||||
|     } | ||||
|  | ||||
|     let hexString = hexArray.join(", "); | ||||
|     let line = `  ${hexString}`; | ||||
|     lines.push(line); | ||||
|   } | ||||
|  | ||||
|   return lines.join(",\n"); | ||||
| } | ||||
|  | ||||
| const inliner = require("inliner"); | ||||
| const zlib = require("zlib"); | ||||
|  | ||||
| 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); | ||||
|   } | ||||
|  | ||||
|   let version = packageJson.version; | ||||
|   if (version) { | ||||
|     html = strReplace(html, "##VERSION##", version); | ||||
|   } | ||||
|  | ||||
|   return html; | ||||
| } | ||||
|  | ||||
| function writeHtmlGzipped(sourceFile, resultFile) { | ||||
|   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; | ||||
|     } | ||||
|  | ||||
|     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://github.com/Aircoookie/WLED/wiki/Add-own-functionality#web-ui | ||||
|  * to find out how to easily modify the web UI source! | ||||
|  */ | ||||
|   | ||||
| // Autogenerated from ${sourceFile}, do not edit!! | ||||
| const uint16_t PAGE_index_L = ${result.length}; | ||||
| const uint8_t PAGE_index[] PROGMEM = { | ||||
| ${array} | ||||
| }; | ||||
| `; | ||||
|       console.info("Writing " + resultFile); | ||||
|       fs.writeFileSync(resultFile, src); | ||||
|     }); | ||||
|   }); | ||||
| } | ||||
|  | ||||
| const CleanCSS = require("clean-css"); | ||||
| const MinifyHTML = require("html-minifier-terser").minify; | ||||
|  | ||||
| function filter(str, type) { | ||||
|   str = adoptVersionAndRepo(str); | ||||
|  | ||||
|   if (type === undefined) { | ||||
|     return str; | ||||
|   } else if (type == "css-minify") { | ||||
|     return new CleanCSS({}).minify(str).styles; | ||||
|   } 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; | ||||
|   } | ||||
| } | ||||
|  | ||||
| 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 || "" | ||||
|     }"; | ||||
|  | ||||
| `; | ||||
|     return s.mangle ? s.mangle(chunk) : 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 s.mangle ? s.mangle(chunk) : 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://github.com/Aircoookie/WLED/wiki/Add-own-functionality#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 | ||||
|       ); | ||||
|     } | ||||
|   }); | ||||
|   console.info("Writing " + src.length + " characters into " + resultFile); | ||||
|   fs.writeFileSync(resultFile, src); | ||||
| } | ||||
|  | ||||
| writeHtmlGzipped("wled00/data/index.htm", "wled00/html_ui.h"); | ||||
|  | ||||
| writeChunks( | ||||
|   "wled00/data", | ||||
|   [ | ||||
|     { | ||||
|       file: "style.css", | ||||
|       name: "PAGE_settingsCss", | ||||
|       prepend: "=====(<style>", | ||||
|       append: "</style>)=====", | ||||
|       method: "plaintext", | ||||
|       filter: "css-minify", | ||||
|     }, | ||||
|     { | ||||
|       file: "settings.htm", | ||||
|       name: "PAGE_settings", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|       mangle: (str) => | ||||
|         str | ||||
|           .replace("%", "%%") | ||||
|           .replace(/User Interface\<\/button\>\<\/form\>/gms, "User Interface\<\/button\>\<\/form\>%DMXMENU%"), | ||||
|     }, | ||||
|     { | ||||
|       file: "settings_wifi.htm", | ||||
|       name: "PAGE_settings_wifi", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|       mangle: (str) => | ||||
|         str | ||||
|           .replace(/\<link rel="stylesheet".*\>/gms, "") | ||||
|           .replace(/\<style\>.*\<\/style\>/gms, "%CSS%%SCSS%") | ||||
|           .replace( | ||||
|             /function GetV().*\<\/script\>/gms, | ||||
|             "function GetV() {var d=document;\n" | ||||
|           ), | ||||
|     }, | ||||
|     { | ||||
|       file: "settings_leds.htm", | ||||
|       name: "PAGE_settings_leds", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|       mangle: (str) => | ||||
|         str | ||||
|           .replace(/\<link rel="stylesheet".*\>/gms, "") | ||||
|           .replace(/\<style\>.*\<\/style\>/gms, "%CSS%%SCSS%") | ||||
|           .replace( | ||||
|             /function GetV().*\<\/script\>/gms, | ||||
|             "function GetV() {var d=document;\n" | ||||
|           ), | ||||
|     }, | ||||
|     { | ||||
|       file: "settings_dmx.htm", | ||||
|       name: "PAGE_settings_dmx", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|       mangle: (str) => { | ||||
|         const nocss = str | ||||
|           .replace(/\<link rel="stylesheet".*\>/gms, "") | ||||
|           .replace(/\<style\>.*\<\/style\>/gms, "%CSS%%SCSS%") | ||||
|           .replace( | ||||
|             /function GetV().*\<\/script\>/gms, | ||||
|             "function GetV() {var d=document;\n" | ||||
|           ); | ||||
|         return ` | ||||
| #ifdef WLED_ENABLE_DMX | ||||
| ${nocss} | ||||
| #else | ||||
| const char PAGE_settings_dmx[] PROGMEM = R"=====()====="; | ||||
| #endif | ||||
| `; | ||||
|       }, | ||||
|     }, | ||||
|     { | ||||
|       file: "settings_ui.htm", | ||||
|       name: "PAGE_settings_ui", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|       mangle: (str) => | ||||
|         str | ||||
|           .replace(/\<link rel="stylesheet".*\>/gms, "") | ||||
|           .replace(/\<style\>.*\<\/style\>/gms, "%CSS%%SCSS%") | ||||
|           .replace( | ||||
|             /function GetV().*\<\/script\>/gms, | ||||
|             "function GetV() {var d=document;\n" | ||||
|           ), | ||||
|     }, | ||||
|     { | ||||
|       file: "settings_sync.htm", | ||||
|       name: "PAGE_settings_sync", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|       mangle: (str) => | ||||
|         str | ||||
|           .replace(/\<link rel="stylesheet".*\>/gms, "") | ||||
|           .replace(/\<style\>.*\<\/style\>/gms, "%CSS%%SCSS%") | ||||
|           .replace(/function GetV().*\<\/script\>/gms, "function GetV() {\n"), | ||||
|     }, | ||||
|     { | ||||
|       file: "settings_time.htm", | ||||
|       name: "PAGE_settings_time", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|       mangle: (str) => | ||||
|         str | ||||
|           .replace(/\<link rel="stylesheet".*\>/gms, "") | ||||
|           .replace(/\<style\>.*\<\/style\>/gms, "%CSS%%SCSS%") | ||||
|           .replace(/function GetV().*\<\/script\>/gms, "function GetV() {\n"), | ||||
|     }, | ||||
|     { | ||||
|       file: "settings_sec.htm", | ||||
|       name: "PAGE_settings_sec", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|       mangle: (str) => | ||||
|         str | ||||
|           .replace(/\<link rel="stylesheet".*\>/gms, "") | ||||
|           .replace(/\<style\>.*\<\/style\>/gms, "%CSS%%SCSS%") | ||||
|           .replace( | ||||
|             /function GetV().*\<\/script\>/gms, | ||||
|             "function GetV() {var d=document;\n" | ||||
|           ), | ||||
|     }, | ||||
|   ], | ||||
|   "wled00/html_settings.h" | ||||
| ); | ||||
|  | ||||
| writeChunks( | ||||
|   "wled00/data", | ||||
|   [ | ||||
|     { | ||||
|       file: "usermod.htm", | ||||
|       name: "PAGE_usermod", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|       mangle: (str) => | ||||
|         str.replace(/fetch\("http\:\/\/.*\/win/gms, 'fetch("/win'), | ||||
|     }, | ||||
|     { | ||||
|       file: "msg.htm", | ||||
|       name: "PAGE_msg", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|       mangle: (str) => str.replace(/\<h2\>.*\<\/body\>/gms, "<h2>%MSG%</body>"), | ||||
|     }, | ||||
|     { | ||||
|       file: "dmxmap.htm", | ||||
|       name: "PAGE_dmxmap", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|       mangle: (str) => ` | ||||
| #ifdef WLED_ENABLE_DMX | ||||
| ${str.replace(/function FM\(\)[ ]?\{/gms, "function FM() {%DMXVARS%\n")} | ||||
| #else | ||||
| const char PAGE_dmxmap[] PROGMEM = R"=====()====="; | ||||
| #endif | ||||
| `, | ||||
|     }, | ||||
|     { | ||||
|       file: "update.htm", | ||||
|       name: "PAGE_update", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|     }, | ||||
|     { | ||||
|       file: "welcome.htm", | ||||
|       name: "PAGE_welcome", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|     }, | ||||
|     { | ||||
|       file: "liveview.htm", | ||||
|       name: "PAGE_liveview", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|     }, | ||||
|     { | ||||
|       file: "liveviewws.htm", | ||||
|       name: "PAGE_liveviewws", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|     }, | ||||
|     { | ||||
|       file: "404.htm", | ||||
|       name: "PAGE_404", | ||||
|       prepend: "=====(", | ||||
|       append: ")=====", | ||||
|       method: "plaintext", | ||||
|       filter: "html-minify", | ||||
|     }, | ||||
|     { | ||||
|       file: "favicon.ico", | ||||
|       name: "favicon", | ||||
|       method: "binary", | ||||
|     }, | ||||
|   ], | ||||
|   "wled00/html_other.h" | ||||
| ); | ||||
							
								
								
									
										427
									
								
								usermods/Animated_Staircase/Animated_Staircase.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,427 @@ | ||||
| /* | ||||
|  * Usermod for detecting people entering/leaving a staircase and switching the | ||||
|  * staircase on/off. | ||||
|  * | ||||
|  * Edit the Animated_Staircase_config.h file to compile this usermod for your | ||||
|  * specific configuration. | ||||
|  *  | ||||
|  * See the accompanying README.md file for more info. | ||||
|  */ | ||||
| #pragma once | ||||
| #include "wled.h" | ||||
| #include "Animated_Staircase_config.h" | ||||
| #define USERMOD_ID_ANIMATED_STAIRCASE 1011 | ||||
|  | ||||
| /* Initial configuration (available in API and stored in flash) */ | ||||
| bool enabled = true;                   // Enable this usermod | ||||
| unsigned long segment_delay_ms = 150;  // Time between switching each segment | ||||
| unsigned long on_time_ms = 5 * 1000;   // The time for the light to stay on | ||||
| #ifndef TOP_PIR_PIN | ||||
| unsigned int topMaxTimeUs = 1749;  // default echo timout, top | ||||
| #endif | ||||
| #ifndef BOTTOM_PIR_PIN | ||||
| unsigned int bottomMaxTimeUs = 1749;  // default echo timout, bottom | ||||
| #endif | ||||
|  | ||||
| // Time between checking of the sensors | ||||
| const int scanDelay = 50; | ||||
|  | ||||
| class Animated_Staircase : public Usermod { | ||||
|  private: | ||||
|   // Lights on or off. | ||||
|   // Flipping this will start a transition. | ||||
|   bool on = false; | ||||
|  | ||||
|   // Swipe direction for current transition | ||||
| #define SWIPE_UP true | ||||
| #define SWIPE_DOWN false | ||||
|   bool swipe = SWIPE_UP; | ||||
|  | ||||
|   // Indicates which Sensor was seen last (to determine | ||||
|   // the direction when swiping off) | ||||
| #define LOWER false | ||||
| #define UPPER true | ||||
|   bool lastSensor = LOWER; | ||||
|  | ||||
|   // Time of the last transition action | ||||
|   unsigned long lastTime = 0; | ||||
|  | ||||
|   // Time of the last sensor check | ||||
|   unsigned long lastScanTime = 0; | ||||
|  | ||||
|   // Last time the lights were switched on or off | ||||
|   unsigned long lastSwitchTime = 0; | ||||
|  | ||||
|   // segment id between onIndex and offIndex are on. | ||||
|   // controll the swipe by setting/moving these indices around. | ||||
|   // onIndex must be less than or equal to offIndex | ||||
|   byte onIndex = 0; | ||||
|   byte offIndex = 0; | ||||
|  | ||||
|   // The maximum number of configured segments. | ||||
|   // Dynamically updated based on user configuration. | ||||
|   byte maxSegmentId = 1; | ||||
|   byte mainSegmentId = 0; | ||||
|  | ||||
|   bool saveState = false; | ||||
|  | ||||
|   // These values are used by the API to read the | ||||
|   // last sensor state, or trigger a sensor | ||||
|   // through the API | ||||
|   bool topSensorRead = false; | ||||
|   bool topSensorWrite = false; | ||||
|   bool bottomSensorRead = false; | ||||
|   bool bottomSensorWrite = false;   | ||||
|  | ||||
|   void updateSegments() { | ||||
|     mainSegmentId = strip.getMainSegmentId(); | ||||
|     WS2812FX::Segment mainsegment = strip.getSegment(mainSegmentId); | ||||
|     WS2812FX::Segment* segments = strip.getSegments(); | ||||
|     for (int i = 0; i < MAX_NUM_SEGMENTS; i++, segments++) { | ||||
|       if (!segments->isActive()) { | ||||
|         maxSegmentId = i - 1; | ||||
|         break; | ||||
|       } | ||||
|  | ||||
|       if (i >= onIndex && i < offIndex) { | ||||
|         segments->setOption(SEG_OPTION_ON, 1, 1); | ||||
|  | ||||
|         // We may need to copy mode and colors from segment 0 to make sure | ||||
|         // changes are propagated even when the config is changed during a wipe | ||||
|         // segments->mode = mainsegment.mode; | ||||
|         // segments->colors[0] = mainsegment.colors[0]; | ||||
|       } else { | ||||
|         segments->setOption(SEG_OPTION_ON, 0, 1); | ||||
|       } | ||||
|       // Always mark segments as "transitional", we are animating the staircase | ||||
|       segments->setOption(SEG_OPTION_TRANSITIONAL, 1, 1); | ||||
|     } | ||||
|     colorUpdated(NOTIFIER_CALL_MODE_DIRECT_CHANGE); | ||||
|   } | ||||
|  | ||||
|   /* | ||||
|    * Detects if an object is within ultrasound range. | ||||
|    * signalPin: The pin where the pulse is sent | ||||
|    * echoPin:   The pin where the echo is received | ||||
|    * maxTimeUs: Detection timeout in microseconds. If an echo is | ||||
|    *            received within this time, an object is detected | ||||
|    *            and the function will return true. | ||||
|    * | ||||
|    * The speed of sound is 343 meters per second at 20 degress Celcius. | ||||
|    * Since the sound has to travel back and forth, the detection | ||||
|    * distance for the sensor in cm is (0.0343 * maxTimeUs) / 2. | ||||
|    * | ||||
|    * For practical reasons, here are some useful distances: | ||||
|    * | ||||
|    * Distance =	maxtime | ||||
|    *     5 cm =  292 uS | ||||
|    *    10 cm =  583 uS | ||||
|    *    20 cm = 1166 uS | ||||
|    *    30 cm = 1749 uS | ||||
|    *    50 cm = 2915 uS | ||||
|    *   100 cm = 5831 uS | ||||
|    */ | ||||
|   bool ultrasoundRead(uint8_t signalPin, | ||||
|                       uint8_t echoPin, | ||||
|                       unsigned int maxTimeUs) { | ||||
|     digitalWrite(signalPin, HIGH); | ||||
|     delayMicroseconds(10); | ||||
|     digitalWrite(signalPin, LOW); | ||||
|     return pulseIn(echoPin, HIGH, maxTimeUs) > 0; | ||||
|   } | ||||
|  | ||||
|   void checkSensors() { | ||||
|     if ((millis() - lastScanTime) > scanDelay) { | ||||
|       lastScanTime = millis(); | ||||
|  | ||||
| #ifdef BOTTOM_PIR_PIN | ||||
|       bottomSensorRead = bottomSensorWrite || (digitalRead(BOTTOM_PIR_PIN) == HIGH); | ||||
| #else | ||||
|       bottomSensorRead = bottomSensorWrite || ultrasoundRead(BOTTOM_TRIGGER_PIN, BOTTOM_ECHO_PIN, bottomMaxTimeUs); | ||||
| #endif | ||||
|  | ||||
| #ifdef TOP_PIR_PIN | ||||
|       topSensorRead = topSensorWrite || (digitalRead(TOP_PIR_PIN) == HIGH); | ||||
| #else | ||||
|       topSensorRead = topSensorWrite || ultrasoundRead(TOP_TRIGGER_PIN, TOP_ECHO_PIN, topMaxTimeUs); | ||||
| #endif | ||||
|  | ||||
|       // Values read, reset the flags for next API call | ||||
|       topSensorWrite = false; | ||||
|       bottomSensorWrite = false; | ||||
|  | ||||
|       if (topSensorRead != bottomSensorRead) { | ||||
|         lastSwitchTime = millis(); | ||||
|  | ||||
|         if (on) { | ||||
|           lastSensor = topSensorRead; | ||||
|         } else { | ||||
|           // If the bottom sensor triggered, we need to swipe up, ON | ||||
|           swipe = bottomSensorRead; | ||||
|  | ||||
|           if (swipe) { | ||||
|             Serial.println("ON -> Swipe up."); | ||||
|           } else { | ||||
|             Serial.println("ON -> Swipe down."); | ||||
|           } | ||||
|  | ||||
|           if (onIndex == offIndex) { | ||||
|             // Position the indices for a correct on-swipe | ||||
|             if (swipe == SWIPE_UP) { | ||||
|               onIndex = mainSegmentId; | ||||
|             } else { | ||||
|               onIndex = maxSegmentId+1; | ||||
|             } | ||||
|             offIndex = onIndex; | ||||
|           } | ||||
|           on = true; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void autoPowerOff() { | ||||
|     if (on && ((millis() - lastSwitchTime) > on_time_ms)) { | ||||
|       // Swipe OFF in the direction of the last sensor detection | ||||
|       swipe = lastSensor; | ||||
|       on = false; | ||||
|  | ||||
|       if (swipe) { | ||||
|         Serial.println("OFF -> Swipe up."); | ||||
|       } else { | ||||
|         Serial.println("OFF -> Swipe down."); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void updateSwipe() { | ||||
|     if ((millis() - lastTime) > segment_delay_ms) { | ||||
|       lastTime = millis(); | ||||
|  | ||||
|       byte oldOnIndex = onIndex; | ||||
|       byte oldOffIndex = offIndex; | ||||
|  | ||||
|       if (on) { | ||||
|         // Turn on all segments | ||||
|         onIndex = MAX(mainSegmentId, onIndex - 1); | ||||
|         offIndex = MIN(maxSegmentId + 1, offIndex + 1); | ||||
|       } else { | ||||
|         if (swipe == SWIPE_UP) { | ||||
|           onIndex = MIN(offIndex, onIndex + 1); | ||||
|         } else { | ||||
|           offIndex = MAX(onIndex, offIndex - 1); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       updateSegments(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void writeSettingsToJson(JsonObject& root) { | ||||
|     JsonObject staircase = root["staircase"]; | ||||
|     if (staircase.isNull()) { | ||||
|       staircase = root.createNestedObject("staircase"); | ||||
|     } | ||||
|     staircase["enabled"] = enabled; | ||||
|     staircase["segment-delay-ms"] = segment_delay_ms; | ||||
|     staircase["on-time-s"] = on_time_ms / 1000; | ||||
|  | ||||
| #ifdef TOP_TRIGGER_PIN | ||||
|     staircase["top-echo-us"] = topMaxTimeUs; | ||||
| #endif | ||||
| #ifdef BOTTOM_TRIGGER_PIN | ||||
|     staircase["bottom-echo-us"] = bottomMaxTimeUs; | ||||
| #endif | ||||
|   } | ||||
|  | ||||
|   void writeSensorsToJson(JsonObject& root) { | ||||
|     JsonObject staircase = root["staircase"]; | ||||
|     if (staircase.isNull()) { | ||||
|       staircase = root.createNestedObject("staircase"); | ||||
|     } | ||||
|     staircase["top-sensor"] = topSensorRead; | ||||
|     staircase["bottom-sensor"] = bottomSensorRead; | ||||
|   } | ||||
|  | ||||
|   bool readSettingsFromJson(JsonObject& root) { | ||||
|     JsonObject staircase = root["staircase"]; | ||||
|     bool changed = false; | ||||
|  | ||||
|     bool shouldEnable = staircase["enabled"] | enabled; | ||||
|     if (shouldEnable != enabled) { | ||||
|       enable(shouldEnable); | ||||
|       changed = true; | ||||
|     } | ||||
|  | ||||
|     unsigned long c_segment_delay_ms = staircase["segment-delay-ms"] | segment_delay_ms; | ||||
|     if (c_segment_delay_ms != segment_delay_ms) { | ||||
|       segment_delay_ms = c_segment_delay_ms; | ||||
|       changed = true; | ||||
|     } | ||||
|  | ||||
|     unsigned long c_on_time_ms = (staircase["on-time-s"] | (on_time_ms / 1000)) * 1000; | ||||
|     if (c_on_time_ms != on_time_ms) { | ||||
|       on_time_ms = c_on_time_ms; | ||||
|       changed = true; | ||||
|     } | ||||
|  | ||||
| #ifdef TOP_TRIGGER_PIN | ||||
|     unsigned int c_topMaxTimeUs = staircase["top-echo-us"] | topMaxTimeUs; | ||||
|     if (c_topMaxTimeUs != topMaxTimeUs) { | ||||
|       topMaxTimeUs = c_topMaxTimeUs; | ||||
|       changed = true; | ||||
|     } | ||||
| #endif | ||||
| #ifdef BOTTOM_TRIGGER_PIN | ||||
|     unsigned int c_bottomMaxTimeUs = staircase["bottom-echo-us"] | bottomMaxTimeUs; | ||||
|     if (c_bottomMaxTimeUs != bottomMaxTimeUs) { | ||||
|       bottomMaxTimeUs = c_bottomMaxTimeUs; | ||||
|       changed = true; | ||||
|     } | ||||
| #endif | ||||
|  | ||||
|     return changed; | ||||
|   } | ||||
|  | ||||
|   void readSensorsFromJson(JsonObject& root) { | ||||
|     JsonObject staircase = root["staircase"]; | ||||
|     bottomSensorWrite = bottomSensorRead || (staircase["bottom-sensor"].as<bool>()); | ||||
|     topSensorWrite = topSensorRead || (staircase["top-sensor"].as<bool>()); | ||||
|   } | ||||
|  | ||||
|   void enable(bool enable) { | ||||
|     if (enable) { | ||||
|       Serial.println("Animated Staircase enabled."); | ||||
|       Serial.print("Delay between steps: "); | ||||
|       Serial.print(segment_delay_ms, DEC); | ||||
|       Serial.print(" milliseconds.\nStairs switch off after: "); | ||||
|       Serial.print(on_time_ms / 1000, DEC); | ||||
|       Serial.println(" seconds."); | ||||
|  | ||||
| #ifdef BOTTOM_PIR_PIN | ||||
|       pinMode(BOTTOM_PIR_PIN, INPUT); | ||||
| #else | ||||
|       pinMode(BOTTOM_TRIGGER_PIN, OUTPUT); | ||||
|       pinMode(BOTTOM_ECHO_PIN, INPUT); | ||||
| #endif | ||||
|  | ||||
| #ifdef TOP_PIR_PIN | ||||
|       pinMode(TOP_PIR_PIN, INPUT); | ||||
| #else | ||||
|       pinMode(TOP_TRIGGER_PIN, OUTPUT); | ||||
|       pinMode(TOP_ECHO_PIN, INPUT); | ||||
| #endif | ||||
|     } else { | ||||
|       // Restore segment options | ||||
|       WS2812FX::Segment mainsegment = strip.getSegment(mainSegmentId); | ||||
|       WS2812FX::Segment* segments = strip.getSegments(); | ||||
|       for (int i = 0; i < MAX_NUM_SEGMENTS; i++, segments++) { | ||||
|         if (!segments->isActive()) { | ||||
|           maxSegmentId = i - 1; | ||||
|           break; | ||||
|         } | ||||
|         segments->setOption(SEG_OPTION_ON, 1, 1); | ||||
|       } | ||||
|       colorUpdated(NOTIFIER_CALL_MODE_DIRECT_CHANGE); | ||||
|       Serial.println("Animated Staircase disabled."); | ||||
|     } | ||||
|     enabled = enable; | ||||
|   } | ||||
|  | ||||
|  public: | ||||
|   void setup() { enable(enabled); } | ||||
|  | ||||
|   void loop() { | ||||
|     // Write changed settings from to flash (see readFromJsonState()) | ||||
|     if (saveState) { | ||||
|       serializeConfig(); | ||||
|       saveState = false; | ||||
|     } | ||||
|  | ||||
|     if (!enabled) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     checkSensors(); | ||||
|     autoPowerOff(); | ||||
|     updateSwipe(); | ||||
|  | ||||
|   } | ||||
|  | ||||
|   uint16_t getId() { return USERMOD_ID_ANIMATED_STAIRCASE; } | ||||
|  | ||||
|   /* | ||||
|    * Shows configuration settings to the json API. This object looks like: | ||||
|    * | ||||
|    * "staircase" : { | ||||
|    *   "enabled" : true | ||||
|    *   "segment-delay-ms" : 150, | ||||
|    *   "on-time-s" : 5 | ||||
|    * } | ||||
|    * | ||||
|    */ | ||||
|   void addToJsonState(JsonObject& root) { | ||||
|     writeSettingsToJson(root); | ||||
|     writeSensorsToJson(root); | ||||
|     Serial.println("Staircase config exposed in API."); | ||||
|   } | ||||
|  | ||||
|   /* | ||||
|    * Reads configuration settings from the json API. | ||||
|    * See void addToJsonState(JsonObject& root) | ||||
|    */ | ||||
|   void readFromJsonState(JsonObject& root) { | ||||
|     // The call to serializeConfig() must be done in the main loop, | ||||
|     // so we set a flag to signal the main loop to save state. | ||||
|     saveState = readSettingsFromJson(root); | ||||
|     readSensorsFromJson(root); | ||||
|     Serial.println("Staircase config read from API."); | ||||
|   } | ||||
|  | ||||
|   /* | ||||
|    * Writes the configuration to internal flash memory. | ||||
|    */ | ||||
|   void addToConfig(JsonObject& root) { | ||||
|     writeSettingsToJson(root); | ||||
|     Serial.println("Staircase config saved."); | ||||
|   } | ||||
|  | ||||
|   /* | ||||
|    * Reads the configuration to internal flash memory before setup() is called. | ||||
|    */ | ||||
|   void readFromConfig(JsonObject& root) { | ||||
|     readSettingsFromJson(root); | ||||
|     Serial.println("Staircase config loaded."); | ||||
|   } | ||||
|  | ||||
|   /* | ||||
|    * Shows the delay between steps and power-off time in the "info" | ||||
|    * tab of the web-UI. | ||||
|    */ | ||||
|   void addToJsonInfo(JsonObject& root) { | ||||
|     JsonObject staircase = root["u"]; | ||||
|     if (staircase.isNull()) { | ||||
|       staircase = root.createNestedObject("u"); | ||||
|     } | ||||
|  | ||||
|     if (enabled) { | ||||
|       JsonArray usermodEnabled = | ||||
|           staircase.createNestedArray("Staircase enabled");  // name | ||||
|       usermodEnabled.add("yes");                             // value | ||||
|  | ||||
|       JsonArray segmentDelay = | ||||
|           staircase.createNestedArray("Delay between stairs");  // name | ||||
|       segmentDelay.add(segment_delay_ms);                       // value | ||||
|       segmentDelay.add(" milliseconds");                        // unit | ||||
|  | ||||
|       JsonArray onTime = | ||||
|           staircase.createNestedArray("Power-off stairs after");  // name | ||||
|       onTime.add(on_time_ms / 1000);                              // value | ||||
|       onTime.add(" seconds");                                     // unit | ||||
|     } else { | ||||
|       JsonArray usermodEnabled = | ||||
|           staircase.createNestedArray("Staircase enabled");  // name | ||||
|       usermodEnabled.add("no");                              // value | ||||
|     } | ||||
|   } | ||||
| }; | ||||
							
								
								
									
										21
									
								
								usermods/Animated_Staircase/Animated_Staircase_config.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | ||||
| /* | ||||
|  * Animated_Staircase compiletime confguration. | ||||
|  * | ||||
|  * Please see README.md on how to change this file. | ||||
|  */ | ||||
|  | ||||
| // Please change the pin numbering below to match your board. | ||||
| #define TOP_PIR_PIN D5 | ||||
| #define BOTTOM_PIR_PIN D6 | ||||
|  | ||||
| // Or uncumment and a pir and use an ultrasound HC-SR04 sensor, | ||||
| // see README.md for details | ||||
| #ifndef TOP_PIR_PIN | ||||
| #define TOP_TRIGGER_PIN D2 | ||||
| #define TOP_ECHO_PIN D3 | ||||
| #endif | ||||
|  | ||||
| #ifndef BOTTOM_PIR_PIN | ||||
| #define BOTTOM_TRIGGER_PIN D4 | ||||
| #define BOTTOM_ECHO_PIN D5 | ||||
| #endif | ||||
							
								
								
									
										203
									
								
								usermods/Animated_Staircase/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,203 @@ | ||||
| # Usermod Animated Staircase | ||||
| This usermod makes your staircase look cool by switching it on with an animation. It uses | ||||
| PIR or ultrasonic sensors at the top and bottom of your stairs to: | ||||
|  | ||||
| - Light up the steps in your walking direction, leading the way. | ||||
| - Switch off the steps after you, in the direction of the last detected movement. | ||||
| - Always switch on when one of the sensors detects movement, even if an effect | ||||
|   is still running. It can therewith handle multiple people on the stairs gracefully. | ||||
|  | ||||
| The Animated Staircase can be controlled by the WLED API. Change settings such as | ||||
| speed, on/off time and distance settings by sending an HTTP request, see below. | ||||
|  | ||||
| ## WLED integration | ||||
| To include this usermod in your WLED setup, you have to be able to [compile WLED from source](https://github.com/Aircoookie/WLED/wiki/Compiling-WLED). | ||||
|  | ||||
| Before compiling, you have to make the following modifications: | ||||
|  | ||||
| Edit `usermods_list.cpp`: | ||||
| 1. Open `wled00/usermods_list.cpp` | ||||
| 2. add `#include "../usermods/Animated_Staircase/Animated_Staircase.h"` to the top of the file | ||||
| 3. add `usermods.add(new Animated_Staircase());` to the end of the `void registerUsermods()` function. | ||||
|  | ||||
| Edit `Animated_Staircase_config.h`: | ||||
| 1. Open `usermods/Animated_Staircase/Animated_Staircase_config.h`  | ||||
| 2. To use PIR sensors, change these lines to match your setup: | ||||
|    Using D7 and D6 pin notation as used on several boards: | ||||
|    | ||||
|    ```cpp | ||||
|      #define TOP_PIR_PIN    D7 | ||||
|      #define BOTTOM_PIR_PIN D6 | ||||
|    ``` | ||||
|     | ||||
|    Or using GPIO numbering for pins 25 and 26: | ||||
|    ```cpp | ||||
|      #define TOP_PIR_PIN    26 | ||||
|      #define BOTTOM_PIR_PIN 25 | ||||
|    ``` | ||||
|  | ||||
|    To use Ultrasonic HC-SR04 sensors instead of (one of the) PIR sensors, | ||||
|    uncomment one of the PIR sensor lines and adjust the pin numbers for the | ||||
|    connected Ultrasonic sensor. In the example below we use an Ultrasonic | ||||
|    sensor at the bottom of the stairs: | ||||
|  | ||||
|    ```cpp | ||||
|    #define TOP_PIR_PIN 32 | ||||
|    //#define BOTTOM_PIR_PIN D6 /* This PIR sensor is disabled   */ | ||||
|  | ||||
|    #ifndef TOP_PIR_PIN | ||||
|    #define TOP_SIGNAL_PIN D2 | ||||
|    #define TOP_ECHO_PIN   D3 | ||||
|    #endif | ||||
|  | ||||
|    #ifndef BOTTOM_PIR_PIN      /* If the bottom PIR is disabled, */ | ||||
|    #define BOTTOM_SIGNAL_PIN 25 /* This Ultrasonic sensor is used */ | ||||
|    #define BOTTOM_ECHO_PIN   26 | ||||
|    #endif | ||||
|    ``` | ||||
|  | ||||
| After these modifications, compile and upload your WLED binary to your board | ||||
| and check the WLED info page to see if this usermod is enabled. | ||||
|  | ||||
| ## Hardware installation | ||||
| 1. Stick the LED strip under each step of the stairs. | ||||
| 2. Connect the ESP8266 pin D4 or ESP32 pin D2 to the first LED data pin at the bottom step | ||||
|    of your stairs. | ||||
| 3. Connect the data-out pin at the end of each strip per step to the data-in pin on the  | ||||
|    other end of the next step, creating one large virtual LED strip. | ||||
| 4. Mount sensors of choice at the bottom and top of the stairs and connect them to the ESP. | ||||
| 5. To make sure all LEDs get enough power and have your staircase lighted evenly, power each | ||||
|    step from one side, using at least AWG14 or 2.5mm^2 cable. Don't connect them serial as you | ||||
|    do for the datacable! | ||||
|  | ||||
| You _may_ need to use 10k pull-down resistors on the selected PIR pins, depending on the sensor. | ||||
|  | ||||
| ## WLED configuration | ||||
| 1. In the WLED UI, confgure a segment for each step. The lowest step of the stairs is the  | ||||
|    lowest segment id.  | ||||
| 2. Save your segments into a preset.  | ||||
| 3. Ideally, add the preset in the config > LED setup menu to the "apply  | ||||
|    preset **n** at boot" setting. | ||||
|  | ||||
| ## Changing behavior through API | ||||
| The Staircase settings can be changed through the WLED JSON api. | ||||
|  | ||||
| **NOTE:** We are using [curl](https://curl.se/) to send HTTP POSTs to the WLED API. | ||||
| If you're using Windows and want to use the curl commands, replace the `\` with a `^` | ||||
| or remove them and put everything on one line. | ||||
|  | ||||
|  | ||||
| | Setting          | Description                                                   | Default | | ||||
| |------------------|---------------------------------------------------------------|---------| | ||||
| | enabled          | Enable or disable the usermod                                 | true    | | ||||
| | segment-delay-ms | Delay (milliseconds) between switching on/off each step       | 150     | | ||||
| | on-time-s        | Time (seconds) the stairs stay lit after last detection       | 5       | | ||||
| | bottom-echo-us   | Detection range of ultrasonic sensor                          | 1749    | | ||||
| | bottomsensor     | Manually trigger a down to up animation via API               | false   |  | ||||
| | topsensor        | Manually trigger an up to down animation via API              | false   | | ||||
|  | ||||
|  | ||||
| To read the current settings, open a browser to `http://xxx.xxx.xxx.xxx/json/state` (use your WLED  | ||||
| device IP address). The device will respond with a json object containing all WLED settings.  | ||||
| The staircase settings and sensor states are inside the WLED status element: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "state": { | ||||
|         "staircase": { | ||||
|             "enabled": true, | ||||
|             "segment-delay-ms": 150, | ||||
|             "on-time-s": 5, | ||||
|             "bottomsensor": false, | ||||
|             "topsensor": false | ||||
|         }, | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Enable/disable the usermod | ||||
| By disabling the usermod you will be able to keep the LED's on, independent from the sensor | ||||
| activity. This enables to play with the lights without the usermod switching them on or off. | ||||
|  | ||||
| To disable the usermod: | ||||
|  | ||||
| ```bash | ||||
| curl -X POST -H "Content-Type: application/json" \ | ||||
|      -d {"staircase":{"enabled":false}} \ | ||||
|      xxx.xxx.xxx.xxx/json/state | ||||
| ``` | ||||
|  | ||||
| To enable the usermod again, use `"enabled":true`. | ||||
|  | ||||
| ### Changing animation parameters | ||||
| To change the delay between the steps to (for example) 100 milliseconds and the on-time to | ||||
| 10 seconds: | ||||
|  | ||||
| ```bash | ||||
| curl -X POST -H "Content-Type: application/json" \ | ||||
|      -d '{"staircase":{"segment-delay-ms":100,"on-time-s":10}}' \ | ||||
|      xxx.xxx.xxx.xxx/json/state | ||||
| ``` | ||||
|  | ||||
| ### Changing detection range of the ultrasonic HC-SR04 sensor | ||||
| When an ultrasonic sensor is enabled in `Animated_Staircase_config.h`, you'll see a  | ||||
| `bottom-echo-us` setting appear in the json api: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "state": { | ||||
|         "staircase": { | ||||
|             "enabled": true, | ||||
|             "segment-delay-ms": 150, | ||||
|             "on-time-s": 5, | ||||
|             "bottom-echo-us": 1749 | ||||
|         }, | ||||
| } | ||||
| ``` | ||||
|  | ||||
| If the HC-SR04 sensor detects an echo within 1749 microseconds (corresponding to ~30 cm  | ||||
| detection range from the sensor), it will trigger switching on the staircase. This setting  | ||||
| can be changed through the API with an HTTP POST: | ||||
|  | ||||
| ```bash | ||||
| curl -X POST -H "Content-Type: application/json" \ | ||||
|      -d '{"staircase":{"bottom-echo-us":1166}}' \ | ||||
|      xxx.xxx.xxx.xxx/json/state | ||||
| ``` | ||||
|  | ||||
| Calculating the detection range can be performed as follows: The speed of sound is 343m/s at 20  | ||||
| degrees Centigrade. Since the sound has to travel back and forth, the detection range for the | ||||
| sensor in cm is (0.0343 * maxTimeUs) / 2. To get you started, please find delays and distances below: | ||||
|  | ||||
| | Distance | Detection time  | | ||||
| |---------:|----------------:| | ||||
| |     5 cm |          292 uS | | ||||
| |    10 cm |          583 uS | | ||||
| |    20 cm |         1166 uS | | ||||
| |    30 cm |         1749 uS | | ||||
| |    50 cm |         2915 uS | | ||||
| |   100 cm |         5831 uS | | ||||
|  | ||||
| **Please note:** that using an HC-SR04 sensor, particularly when detecting echos at longer | ||||
| distances creates delays in the WLED software, and _might_ introduce timing hickups in your animations or | ||||
| a less responsive web interface. It is therefore advised to keep the detection time as short as possible. | ||||
|  | ||||
| ### Animation triggering through the API | ||||
| Instead of stairs activation by one of the sensors, you can also trigger the animation through | ||||
| the API. To simulate triggering the bottom sensor, use: | ||||
|  | ||||
| ```bash | ||||
| curl -X POST -H "Content-Type: application/json" \ | ||||
|      -d '{"staircase":{"bottomsensor":true}}' \ | ||||
|      xxx.xxx.xxx.xxx/json/state | ||||
| ``` | ||||
|  | ||||
| Likewise, to trigger the top sensor, use: | ||||
|  | ||||
| ```bash | ||||
| curl -X POST -H "Content-Type: application/json" \ | ||||
|      -d '{"staircase":{"topsensor":true}}' \ | ||||
|      xxx.xxx.xxx.xxx/json/state | ||||
| ``` | ||||
|  | ||||
| Have fun with this usermod.<br/> | ||||
| www.rolfje.com | ||||
							
								
								
									
										5
									
								
								usermods/Artemis_reciever/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | ||||
| Usermod to allow WLED to receive via UDP port from RGB.NET (and therefore add as a device to be controlled within artemis on PC) | ||||
|  | ||||
| This is only a very simple code to support a single led strip, it does not support the full function of the RGB.NET sketch for esp8266 only what is needed to be used with Artemis. It will show as a ws281x device in artemis when you provide the correct hostname or ip. Artemis queries the number of LEDs via the web interface (/config) but communication to set the LEDs is all done via the UDP interface. | ||||
|  | ||||
| To install, copy the usermod.cpp file to wled00 folder and recompile | ||||
							
								
								
									
										93
									
								
								usermods/Artemis_reciever/usermod.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,93 @@ | ||||
| /* | ||||
|  *          RGB.NET (artemis) receiver | ||||
|  *           | ||||
|  * This works via the UDP, http is not supported apart from reporting LED count | ||||
|  *  | ||||
|  *  | ||||
|  */ | ||||
| #include "wled.h" | ||||
| #include <WiFiUdp.h> | ||||
|  | ||||
| WiFiUDP UDP; | ||||
| const unsigned int RGBNET_localUdpPort = 1872; // local port to listen on | ||||
| unsigned char RGBNET_packet[770]; | ||||
| long lastTime = 0; | ||||
| int delayMs = 10; | ||||
| bool isRGBNETUDPEnabled; | ||||
|  | ||||
| void RGBNET_readValues() { | ||||
|    | ||||
|   int RGBNET_packetSize = UDP.parsePacket(); | ||||
|   if (RGBNET_packetSize) { | ||||
|     // receive incoming UDP packets | ||||
|     int sequenceNumber = UDP.read(); | ||||
|     int channel = UDP.read(); | ||||
|  | ||||
|     //channel data is not used we only supports one channel | ||||
|     int len = UDP.read(RGBNET_packet, ledCount*3); | ||||
|     if(len==0){ | ||||
|       return; | ||||
|     } | ||||
|      | ||||
|     for (int i = 0; i < len; i=i+3) { | ||||
|       strip.setPixelColor(i/3, RGBNET_packet[i], RGBNET_packet[i+1], RGBNET_packet[i+2], 0); | ||||
|     }  | ||||
|     //strip.show();   | ||||
|   } | ||||
| } | ||||
|  | ||||
| //update LED strip | ||||
| void RGBNET_show() { | ||||
|   strip.show(); | ||||
|   lastTime = millis(); | ||||
| } | ||||
|  | ||||
| //This function provides a json with info on the number of LEDs connected | ||||
| // it is needed by artemis to know how many LEDs to display on the surface | ||||
| void handleConfig(AsyncWebServerRequest *request) | ||||
| { | ||||
|   String config = (String)"{\ | ||||
|   \"channels\": [\ | ||||
|     {\ | ||||
|       \"channel\": 1,\ | ||||
|       \"leds\": " + ledCount + "\ | ||||
|     },\ | ||||
|     {\ | ||||
|       \"channel\": 2,\ | ||||
|       \"leds\": " + "0" + "\ | ||||
|     },\ | ||||
|     {\ | ||||
|       \"channel\": 3,\ | ||||
|       \"leds\": " + "0" + "\ | ||||
|     },\ | ||||
|     {\ | ||||
|       \"channel\": 4,\ | ||||
|       \"leds\": " + "0" + "\ | ||||
|     }\ | ||||
|   ]\ | ||||
| }"; | ||||
|   request->send(200, "application/json", config); | ||||
| } | ||||
|  | ||||
|  | ||||
| void userSetup() | ||||
| { | ||||
|   server.on("/config", HTTP_GET, [](AsyncWebServerRequest *request){  | ||||
|     handleConfig(request); | ||||
|   }); | ||||
| } | ||||
|  | ||||
| void userConnected() | ||||
| { | ||||
|   // new wifi, who dis? | ||||
|   UDP.begin(RGBNET_localUdpPort); | ||||
|   isRGBNETUDPEnabled = true; | ||||
| } | ||||
|  | ||||
| void userLoop() | ||||
| { | ||||
|   RGBNET_readValues(); | ||||
|     if (millis()-lastTime > delayMs) { | ||||
|       RGBNET_show(); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										40
									
								
								usermods/BME280_v2/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,40 @@ | ||||
| Hello! I have written a v2 usermod for the BME280/BMP280 sensor based on the [existing v1 usermod](https://github.com/Aircoookie/WLED/blob/master/usermods/Wemos_D1_mini%2BWemos32_mini_shield/usermod_bme280.cpp). It is not just a refactor, there are many changes which I made to fit my use case, and I hope they will fit the use cases of others as well! Most notably, this usermod is *just* for the BME280 and does not control a display like in the v1 usermod designed for the WeMos shield.  | ||||
|  | ||||
| - Requires libraries `BME280@~3.0.0` (by [finitespace](https://github.com/finitespace/BME280)) and `Wire`. Please add these under `lib_deps` in your `platform.ini` (or `platform_override.ini`). | ||||
| - Data is published over MQTT so make sure you've enabled the MQTT sync interface. | ||||
| - This usermod also writes to serial (GPIO1 on ESP8266). Please make sure nothing else listening on the serial TX pin of your board will get confused by log messages! | ||||
|  | ||||
| To enable, compile with `USERMOD_BME280` defined (i.e. `platformio_override.ini`) | ||||
| ```ini | ||||
| build_flags = | ||||
|   ${common.build_flags_esp8266} | ||||
|   -D USERMOD_BME280 | ||||
| ``` | ||||
| or define `USERMOD_BME280` in `my_config.h` | ||||
| ```c++ | ||||
| #define USERMOD_BME280 | ||||
| ``` | ||||
|  | ||||
| Changes include: | ||||
| - Adjustable measure intervals | ||||
|   - Temperature and pressure have separate intervals due to pressure not frequently changing at any constant altitude | ||||
| - Adjustment of number of decimal places in published sensor values | ||||
|   - Separate adjustment for temperature, humidity and pressure values | ||||
|   - Values are rounded to the specified number of decimal places | ||||
| - Pressure measured in units of hPa instead of Pa | ||||
| - Calculation of heat index (apparent temperature) and dew point | ||||
|   - These, along with humidity measurements, are disabled if the sensor is a BMP280 | ||||
| - 16x oversampling of sensor during measurement | ||||
| - Values are only published if they are different from the previous value | ||||
| - Values are published on startup (continually until the MQTT broker acknowledges a successful publication) | ||||
|  | ||||
| Adjustments are made through preprocessor definitions at the start of the class definition. | ||||
|  | ||||
| MQTT topics are as follows: | ||||
| Measurement type | MQTT topic | ||||
| --- | --- | ||||
| Temperature | `<deviceTopic>/temperature` | ||||
| Humidity | `<deviceTopic>/humidity` | ||||
| Pressure | `<deviceTopic>/pressure` | ||||
| Heat index | `<deviceTopic>/heat_index` | ||||
| Dew point | `<deviceTopic>/dew_point` | ||||
							
								
								
									
										212
									
								
								usermods/BME280_v2/usermod_bme280.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,212 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
| #include <Arduino.h> | ||||
| #include <Wire.h> | ||||
| #include <BME280I2C.h>               // BME280 sensor | ||||
| #include <EnvironmentCalculations.h> // BME280 extended measurements | ||||
|  | ||||
| class UsermodBME280 : public Usermod | ||||
| { | ||||
| private: | ||||
| // User-defined configuration | ||||
| #define Celsius               // Show temperature mesaurement in Celcius. Comment out for Fahrenheit | ||||
| #define TemperatureDecimals 1 // Number of decimal places in published temperaure values | ||||
| #define HumidityDecimals 0    // Number of decimal places in published humidity values | ||||
| #define PressureDecimals 2    // Number of decimal places in published pressure values | ||||
| #define TemperatureInterval 5 // Interval to measure temperature (and humidity, dew point if available) in seconds | ||||
| #define PressureInterval 300  // Interval to measure pressure in seconds | ||||
|  | ||||
| // Sanity checks | ||||
| #if !defined(TemperatureDecimals) || TemperatureDecimals < 0 | ||||
|   #define TemperatureDecimals 0 | ||||
| #endif | ||||
| #if !defined(HumidityDecimals) || HumidityDecimals < 0 | ||||
|   #define HumidityDecimals 0 | ||||
| #endif | ||||
| #if !defined(PressureDecimals) || PressureDecimals < 0 | ||||
|   #define PressureDecimals 0 | ||||
| #endif | ||||
| #if !defined(TemperatureInterval) || TemperatureInterval < 0 | ||||
|   #define TemperatureInterval 1 | ||||
| #endif | ||||
| #if !defined(PressureInterval) || PressureInterval < 0 | ||||
|   #define PressureInterval TemperatureInterval | ||||
| #endif | ||||
|  | ||||
| #ifdef ARDUINO_ARCH_ESP32 // ESP32 boards | ||||
|   uint8_t SCL_PIN = 22; | ||||
|   uint8_t SDA_PIN = 21; | ||||
| #else // ESP8266 boards | ||||
|   uint8_t SCL_PIN = 5; | ||||
|   uint8_t SDA_PIN = 4; | ||||
|   //uint8_t RST_PIN = 16; // Uncoment for Heltec WiFi-Kit-8 | ||||
| #endif | ||||
|  | ||||
|   // BME280 sensor settings | ||||
|   BME280I2C::Settings settings{ | ||||
|       BME280::OSR_X16, // Temperature oversampling x16 | ||||
|       BME280::OSR_X16, // Humidity oversampling x16 | ||||
|       BME280::OSR_X16, // Pressure oversampling x16 | ||||
|       // Defaults | ||||
|       BME280::Mode_Forced, | ||||
|       BME280::StandbyTime_1000ms, | ||||
|       BME280::Filter_Off, | ||||
|       BME280::SpiEnable_False, | ||||
|       BME280I2C::I2CAddr_0x76 // I2C address. I2C specific. Default 0x76 | ||||
|   }; | ||||
|  | ||||
|   BME280I2C bme{settings}; | ||||
|  | ||||
|   uint8_t SensorType; | ||||
|  | ||||
|   // Measurement timers | ||||
|   long timer; | ||||
|   long lastTemperatureMeasure = 0; | ||||
|   long lastPressureMeasure = 0; | ||||
|  | ||||
|   // Current sensor values | ||||
|   float SensorTemperature; | ||||
|   float SensorHumidity; | ||||
|   float SensorHeatIndex; | ||||
|   float SensorDewPoint; | ||||
|   float SensorPressure; | ||||
|   // Track previous sensor values | ||||
|   float lastTemperature; | ||||
|   float lastHumidity; | ||||
|   float lastHeatIndex; | ||||
|   float lastDewPoint; | ||||
|   float lastPressure; | ||||
|  | ||||
|   // Store packet IDs of MQTT publications | ||||
|   uint16_t mqttTemperaturePub = 0; | ||||
|   uint16_t mqttPressurePub = 0; | ||||
|  | ||||
|   void UpdateBME280Data(int SensorType) | ||||
|   { | ||||
|     float _temperature, _humidity, _pressure; | ||||
|     #ifdef Celsius | ||||
|       BME280::TempUnit tempUnit(BME280::TempUnit_Celsius); | ||||
|       EnvironmentCalculations::TempUnit envTempUnit(EnvironmentCalculations::TempUnit_Celsius); | ||||
|     #else | ||||
|       BME280::TempUnit tempUnit(BME280::TempUnit_Fahrenheit); | ||||
|       EnvironmentCalculations::TempUnit envTempUnit(EnvironmentCalculations::TempUnit_Fahrenheit); | ||||
|     #endif | ||||
|     BME280::PresUnit presUnit(BME280::PresUnit_hPa); | ||||
|  | ||||
|     bme.read(_pressure, _temperature, _humidity, tempUnit, presUnit); | ||||
|  | ||||
|     SensorTemperature = _temperature; | ||||
|     SensorHumidity = _humidity; | ||||
|     SensorPressure = _pressure; | ||||
|     if (SensorType == 1) | ||||
|     { | ||||
|       SensorHeatIndex = EnvironmentCalculations::HeatIndex(_temperature, _humidity, envTempUnit); | ||||
|       SensorDewPoint = EnvironmentCalculations::DewPoint(_temperature, _humidity, envTempUnit); | ||||
|     } | ||||
|   } | ||||
|  | ||||
| public: | ||||
|   void setup() | ||||
|   { | ||||
|     Wire.begin(SDA_PIN, SCL_PIN); | ||||
|  | ||||
|     if (!bme.begin()) | ||||
|     { | ||||
|       SensorType = 0; | ||||
|       Serial.println("Could not find BME280I2C sensor!"); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|       switch (bme.chipModel()) | ||||
|       { | ||||
|       case BME280::ChipModel_BME280: | ||||
|         SensorType = 1; | ||||
|         Serial.println("Found BME280 sensor! Success."); | ||||
|         break; | ||||
|       case BME280::ChipModel_BMP280: | ||||
|         SensorType = 2; | ||||
|         Serial.println("Found BMP280 sensor! No Humidity available."); | ||||
|         break; | ||||
|       default: | ||||
|         SensorType = 0; | ||||
|         Serial.println("Found UNKNOWN sensor! Error!"); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void loop() | ||||
|   { | ||||
|     // BME280 sensor MQTT publishing | ||||
|     // Check if sensor present and MQTT Connected, otherwise it will crash the MCU | ||||
|     if (SensorType != 0 && mqtt != nullptr) | ||||
|     { | ||||
|       // Timer to fetch new temperature, humidity and pressure data at intervals | ||||
|       timer = millis(); | ||||
|  | ||||
|       if (timer - lastTemperatureMeasure >= TemperatureInterval * 1000 || mqttTemperaturePub == 0) | ||||
|       { | ||||
|         lastTemperatureMeasure = timer; | ||||
|  | ||||
|         UpdateBME280Data(SensorType); | ||||
|  | ||||
|         float Temperature = roundf(SensorTemperature * pow(10, TemperatureDecimals)) / pow(10, TemperatureDecimals); | ||||
|         float Humidity, HeatIndex, DewPoint; | ||||
|  | ||||
|         // If temperature has changed since last measure, create string populated with device topic | ||||
|         // from the UI and values read from sensor, then publish to broker | ||||
|         if (Temperature != lastTemperature) | ||||
|         { | ||||
|           String topic = String(mqttDeviceTopic) + "/temperature"; | ||||
|           mqttTemperaturePub = mqtt->publish(topic.c_str(), 0, false, String(Temperature, TemperatureDecimals).c_str()); | ||||
|         } | ||||
|  | ||||
|         lastTemperature = Temperature; // Update last sensor temperature for next loop | ||||
|  | ||||
|         if (SensorType == 1) // Only if sensor is a BME280 | ||||
|         { | ||||
|           Humidity = roundf(SensorHumidity * pow(10, HumidityDecimals)) / pow(10, HumidityDecimals); | ||||
|           HeatIndex = roundf(SensorHeatIndex * pow(10, TemperatureDecimals)) / pow(10, TemperatureDecimals); | ||||
|           DewPoint = roundf(SensorDewPoint * pow(10, TemperatureDecimals)) / pow(10, TemperatureDecimals); | ||||
|  | ||||
|           if (Humidity != lastHumidity) | ||||
|           { | ||||
|             String topic = String(mqttDeviceTopic) + "/humidity"; | ||||
|             mqtt->publish(topic.c_str(), 0, false, String(Humidity, HumidityDecimals).c_str()); | ||||
|           } | ||||
|  | ||||
|           if (HeatIndex != lastHeatIndex) | ||||
|           { | ||||
|             String topic = String(mqttDeviceTopic) + "/heat_index"; | ||||
|             mqtt->publish(topic.c_str(), 0, false, String(HeatIndex, TemperatureDecimals).c_str()); | ||||
|           } | ||||
|  | ||||
|           if (DewPoint != lastDewPoint) | ||||
|           { | ||||
|             String topic = String(mqttDeviceTopic) + "/dew_point"; | ||||
|             mqtt->publish(topic.c_str(), 0, false, String(DewPoint, TemperatureDecimals).c_str()); | ||||
|           } | ||||
|  | ||||
|           lastHumidity = Humidity; | ||||
|           lastHeatIndex = HeatIndex; | ||||
|           lastDewPoint = DewPoint; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       if (timer - lastPressureMeasure >= PressureInterval * 1000 || mqttPressurePub == 0) | ||||
|       { | ||||
|         lastPressureMeasure = timer; | ||||
|  | ||||
|         float Pressure = roundf(SensorPressure * pow(10, PressureDecimals)) / pow(10, PressureDecimals); | ||||
|  | ||||
|         if (Pressure != lastPressure) | ||||
|         { | ||||
|           String topic = String(mqttDeviceTopic) + "/pressure"; | ||||
|           mqttPressurePub = mqtt->publish(topic.c_str(), 0, true, String(Pressure, PressureDecimals).c_str()); | ||||
|         } | ||||
|  | ||||
|         lastPressure = Pressure; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| }; | ||||
							
								
								
									
										22
									
								
								usermods/DHT/platformio_override.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,22 @@ | ||||
| ; Options | ||||
| ; ------- | ||||
| ; USERMOD_DHT                      - define this to have this user mod included wled00\usermods_list.cpp | ||||
| ; USERMOD_DHT_DHTTYPE              - DHT model: 11, 21, 22 for DHT11, DHT21, or DHT22, defaults to 22/DHT22 | ||||
| ; USERMOD_DHT_PIN                  - pin to which DTH is connected, defaults to Q2 pin on QuinLed Dig-Uno's board | ||||
| ; USERMOD_DHT_CELSIUS              - define this to report temperatures in degrees celsious, otherwise fahrenheit will be reported | ||||
| ; USERMOD_DHT_MEASUREMENT_INTERVAL - the number of milliseconds between measurements, defaults to 60 seconds | ||||
| ; USERMOD_DHT_FIRST_MEASUREMENT_AT - the number of milliseconds after boot to take first measurement, defaults to 90 seconds | ||||
| ; USERMOD_DHT_STATS                - For debug, report delay stats | ||||
|  | ||||
| [env:d1_mini_usermod_dht_C] | ||||
| extends = env:d1_mini | ||||
| build_flags = ${env:d1_mini.build_flags} -D USERMOD_DHT -D USERMOD_DHT_CELSIUS | ||||
| lib_deps = ${env.lib_deps} | ||||
|     https://github.com/alwynallan/DHT_nonblocking | ||||
|  | ||||
| [env:custom32_LEDPIN_16_usermod_dht_C] | ||||
| extends = env:custom32_LEDPIN_16 | ||||
| build_flags = ${env:custom32_LEDPIN_16.build_flags} -D USERMOD_DHT -D USERMOD_DHT_CELSIUS -D USERMOD_DHT_STATS | ||||
| lib_deps = ${env.lib_deps} | ||||
|     https://github.com/alwynallan/DHT_nonblocking | ||||
|  | ||||
							
								
								
									
										41
									
								
								usermods/DHT/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,41 @@ | ||||
| # DHT Temperature/Humidity sensor usermod | ||||
|  | ||||
| This usermod will read from an attached DHT22 or DHT11 humidity and temperature sensor. | ||||
| The sensor readings are displayed in the Info section of the web UI. | ||||
|  | ||||
| If sensor is not detected after a while (10 update intervals), this usermod will be disabled. | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| Copy the example `platformio_override.ini` to the root directory.  This file should be placed in the same directory as `platformio.ini`. | ||||
|  | ||||
| ### Define Your Options | ||||
|  | ||||
| * `USERMOD_DHT`                      - define this to have this user mod included wled00\usermods_list.cpp | ||||
| * `USERMOD_DHT_DHTTYPE`              - DHT model: 11, 21, 22 for DHT11, DHT21, or DHT22, defaults to 22/DHT22 | ||||
| * `USERMOD_DHT_PIN`                  - pin to which DTH is connected, defaults to Q2 pin on QuinLed Dig-Uno's board | ||||
| * `USERMOD_DHT_CELSIUS`              - define this to report temperatures in degrees celsious, otherwise fahrenheit will be reported | ||||
| * `USERMOD_DHT_MEASUREMENT_INTERVAL` - the number of milliseconds between measurements, defaults to 60 seconds | ||||
| * `USERMOD_DHT_FIRST_MEASUREMENT_AT` - the number of milliseconds after boot to take first measurement, defaults to 90 seconds | ||||
| * `USERMOD_DHT_STATS`                - For debug, report delay stats | ||||
|  | ||||
| ## Project link | ||||
|  | ||||
| * [QuinLED-Dig-Uno](https://quinled.info/2018/09/15/quinled-dig-uno/) - Project link | ||||
|  | ||||
| ### PlatformIO requirements | ||||
|  | ||||
| If you are using `platformio_override.ini`, you should be able to refresh the task list and see your custom task, for example `env:d1_mini_usermod_dht_C`. If not, you can add the libraries and dependencies into `platformio.ini` as you see fit. | ||||
|  | ||||
|  | ||||
| ## Change Log | ||||
|  | ||||
| 2020-02-04 | ||||
| * Change default QuinLed pin to Q2 | ||||
| * Instead of trying to keep updates at constant cadence, space readings out by measurement interval; hope this helps to avoid occasional bursts of readings with errors | ||||
| * Add some more (optional) stats | ||||
| 2020-02-03 | ||||
| * Due to poor readouts on ESP32 with previous DHT library, rewrote to use https://github.com/alwynallan/DHT_nonblocking | ||||
| * The new library serializes/delays up to 5ms for the sensor readout   | ||||
| 2020-02-02  | ||||
| * Created | ||||
							
								
								
									
										216
									
								
								usermods/DHT/usermod_dht.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,216 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| #include <dht_nonblocking.h> | ||||
|  | ||||
| // USERMOD_DHT_DHTTYPE: | ||||
| //   11   // DHT 11 | ||||
| //   21   // DHT 21 | ||||
| //   22   // DHT 22  (AM2302), AM2321 *** default | ||||
| #ifndef USERMOD_DHT_DHTTYPE | ||||
| #define USERMOD_DHT_DHTTYPE 22 | ||||
| #endif | ||||
|  | ||||
| #if USERMOD_DHT_DHTTYPE == 11 | ||||
| #define DHTTYPE DHT_TYPE_11 | ||||
| #elif USERMOD_DHT_DHTTYPE == 21 | ||||
| #define DHTTYPE DHT_TYPE_21 | ||||
| #elif USERMOD_DHT_DHTTYPE == 22 | ||||
| #define DHTTYPE DHT_TYPE_22 | ||||
| #endif | ||||
|  | ||||
| // Connect pin 1 (on the left) of the sensor to +5V | ||||
| //   NOTE: If using a board with 3.3V logic like an Arduino Due connect pin 1 | ||||
| //   to 3.3V instead of 5V! | ||||
| // Connect pin 2 of the sensor to whatever your DHTPIN is | ||||
| //   NOTE: Pin defaults below are for QuinLed Dig-Uno's Q2 on the board | ||||
| // Connect pin 4 (on the right) of the sensor to GROUND | ||||
| //   NOTE: If using a bare sensor (AM*), Connect a 10K resistor from pin 2 | ||||
| //   (data) to pin 1 (power) of the sensor. DHT* boards have the pullup already | ||||
|  | ||||
| #ifdef USERMOD_DHT_PIN | ||||
| #define DHTPIN USERMOD_DHT_PIN | ||||
| #else | ||||
| #ifdef ARDUINO_ARCH_ESP32 | ||||
| #define DHTPIN 21 | ||||
| #else //ESP8266 boards | ||||
| #define DHTPIN 4 | ||||
| #endif | ||||
| #endif | ||||
|  | ||||
| // the frequency to check sensor, 1 minute | ||||
| #ifndef USERMOD_DHT_MEASUREMENT_INTERVAL | ||||
| #define USERMOD_DHT_MEASUREMENT_INTERVAL 60000 | ||||
| #endif | ||||
|  | ||||
| // how many seconds after boot to take first measurement, 90 seconds | ||||
| // 90 gives enough time to OTA update firmware if this crashses | ||||
| #ifndef USERMOD_DHT_FIRST_MEASUREMENT_AT | ||||
| #define USERMOD_DHT_FIRST_MEASUREMENT_AT 90000 | ||||
| #endif | ||||
|  | ||||
| // from COOLDOWN_TIME in dht_nonblocking.cpp | ||||
| #define DHT_TIMEOUT_TIME  10000 | ||||
|  | ||||
| DHT_nonblocking dht_sensor(DHTPIN, DHTTYPE); | ||||
|  | ||||
| class UsermodDHT : public Usermod { | ||||
|   private: | ||||
|     unsigned long nextReadTime = 0; | ||||
|     unsigned long lastReadTime = 0; | ||||
|     float humidity, temperature = 0; | ||||
|     bool initializing = true; | ||||
|     bool disabled = false; | ||||
|     #ifdef USERMOD_DHT_STATS | ||||
|     unsigned long nextResetStatsTime = 0; | ||||
|     uint16_t updates = 0; | ||||
|     uint16_t clean_updates = 0; | ||||
|     uint16_t errors = 0; | ||||
|     unsigned long maxDelay = 0; | ||||
|     unsigned long currentIteration = 0; | ||||
|     unsigned long maxIteration = 0; | ||||
|     #endif | ||||
|  | ||||
|   public: | ||||
|     void setup() { | ||||
|       nextReadTime = millis() + USERMOD_DHT_FIRST_MEASUREMENT_AT; | ||||
|       lastReadTime = millis(); | ||||
|       #ifdef USERMOD_DHT_STATS | ||||
|       nextResetStatsTime = millis() + 60*60*1000; | ||||
|       #endif | ||||
|     } | ||||
|  | ||||
|     void loop() { | ||||
|       if (disabled) { | ||||
|         return; | ||||
|       } | ||||
|       if (millis() < nextReadTime) { | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       #ifdef USERMOD_DHT_STATS | ||||
|       if (millis() >= nextResetStatsTime) { | ||||
|         nextResetStatsTime += 60*60*1000; | ||||
|         errors = 0; | ||||
|         updates = 0; | ||||
|         clean_updates = 0; | ||||
|       } | ||||
|       unsigned long dcalc = millis(); | ||||
|       if (currentIteration == 0) { | ||||
|         currentIteration = millis(); | ||||
|       } | ||||
|       #endif | ||||
|  | ||||
|       float tempC; | ||||
|       if (dht_sensor.measure(&tempC, &humidity)) { | ||||
|         #ifdef USERMOD_DHT_CELSIUS | ||||
|         temperature = tempC; | ||||
|         #else | ||||
|         temperature = tempC * 9 / 5 + 32; | ||||
|         #endif | ||||
|  | ||||
|         nextReadTime = millis() + USERMOD_DHT_MEASUREMENT_INTERVAL; | ||||
|         lastReadTime = millis(); | ||||
|         initializing = false; | ||||
|          | ||||
|         #ifdef USERMOD_DHT_STATS | ||||
|         unsigned long icalc = millis() - currentIteration; | ||||
|         if (icalc > maxIteration) { | ||||
|           maxIteration = icalc; | ||||
|         } | ||||
|         if (icalc > DHT_TIMEOUT_TIME) { | ||||
|           errors += icalc/DHT_TIMEOUT_TIME; | ||||
|         } else { | ||||
|           clean_updates += 1; | ||||
|         } | ||||
|         updates += 1; | ||||
|         currentIteration = 0; | ||||
|  | ||||
|         #endif | ||||
|       } | ||||
|  | ||||
|       #ifdef USERMOD_DHT_STATS | ||||
|       dcalc = millis() - dcalc; | ||||
|       if (dcalc > maxDelay) { | ||||
|         maxDelay = dcalc; | ||||
|       }  | ||||
|       #endif | ||||
|  | ||||
|       if (((millis() - lastReadTime) > 10*USERMOD_DHT_MEASUREMENT_INTERVAL)) { | ||||
|         disabled = true; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     void addToJsonInfo(JsonObject& root) { | ||||
|       if (disabled) { | ||||
|         return; | ||||
|       } | ||||
|       JsonObject user = root["u"]; | ||||
|       if (user.isNull()) user = root.createNestedObject("u"); | ||||
|  | ||||
|       JsonArray temp = user.createNestedArray("Temperature"); | ||||
|       JsonArray hum = user.createNestedArray("Humidity"); | ||||
|  | ||||
|       #ifdef USERMOD_DHT_STATS | ||||
|       JsonArray next = user.createNestedArray("next"); | ||||
|       if (nextReadTime >= millis()) { | ||||
|         next.add((nextReadTime - millis()) / 1000); | ||||
|         next.add(" sec until read"); | ||||
|       } else { | ||||
|         next.add((millis() - nextReadTime) / 1000); | ||||
|         next.add(" sec active reading"); | ||||
|       } | ||||
|  | ||||
|       JsonArray last = user.createNestedArray("last"); | ||||
|       last.add((millis() - lastReadTime) / 60000); | ||||
|       last.add(" min since read"); | ||||
|  | ||||
|       JsonArray err = user.createNestedArray("errors"); | ||||
|       err.add(errors); | ||||
|       err.add(" Errors"); | ||||
|  | ||||
|       JsonArray upd = user.createNestedArray("updates"); | ||||
|       upd.add(updates); | ||||
|       upd.add(" Updates"); | ||||
|  | ||||
|       JsonArray cupd = user.createNestedArray("cleanUpdates"); | ||||
|       cupd.add(clean_updates); | ||||
|       cupd.add(" Updates"); | ||||
|  | ||||
|       JsonArray iter = user.createNestedArray("maxIter"); | ||||
|       iter.add(maxIteration); | ||||
|       iter.add(" ms"); | ||||
|  | ||||
|       JsonArray delay = user.createNestedArray("maxDelay"); | ||||
|       delay.add(maxDelay); | ||||
|       delay.add(" ms"); | ||||
|       #endif | ||||
|  | ||||
|       if (initializing) { | ||||
|         // if we haven't read the sensor yet, let the user know | ||||
|         // that we are still waiting for the first measurement | ||||
|         temp.add((nextReadTime - millis()) / 1000); | ||||
|         temp.add(" sec until read"); | ||||
|         hum.add((nextReadTime - millis()) / 1000); | ||||
|         hum.add(" sec until read"); | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       hum.add(humidity); | ||||
|       hum.add("%"); | ||||
|  | ||||
|       temp.add(temperature); | ||||
|       #ifdef USERMOD_DHT_CELSIUS | ||||
|       temp.add("°C"); | ||||
|       #else | ||||
|       temp.add("°F"); | ||||
|       #endif | ||||
|     } | ||||
|     | ||||
|     uint16_t getId() | ||||
|     { | ||||
|       return USERMOD_ID_DHT; | ||||
|     } | ||||
|  | ||||
| }; | ||||
							
								
								
									
										19
									
								
								usermods/ESP32_TouchBrightnessControl/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | ||||
| # ESP32 Touch Brightness Control | ||||
|  | ||||
| Toggle On/Off with a long press (800ms) | ||||
| Switch through 5 brightness levels (defined in usermod_touchbrightness.h, values 0-255) with a short (100ms) touch | ||||
|  | ||||
| ## Installation  | ||||
|  | ||||
| Copy 'usermod_touchbrightness.h' to the wled00 directory.   | ||||
| in 'usermod_list.cpp' add this: | ||||
|  | ||||
| > #include "usermod_touchbrightness.h" | ||||
| above "void registerUsermods()" | ||||
|  | ||||
| and | ||||
|  | ||||
| > usermods.add(new TouchBrightnessControl()); | ||||
| inside the "registerUsermods()" function | ||||
|  | ||||
|  | ||||
| @@ -0,0 +1,89 @@ | ||||
| // | ||||
| //  usermod_touchbrightness.h | ||||
| //  github.com/aircoookie/WLED | ||||
| // | ||||
| //  Created by Justin Kühner on 14.09.2020. | ||||
| //  Copyright © 2020 NeariX. All rights reserved. | ||||
| //  https://github.com/NeariX67/ | ||||
| //  Discord: @NeariX#4799 | ||||
|  | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| #define threshold 40                    //Increase value if touches falsely accur. Decrease value if actual touches are not recognized | ||||
| #define touchPin T0                     //T0 = D4 / GPIO4 | ||||
|  | ||||
| //Define the 5 brightness levels | ||||
| //Long press to turn off / on | ||||
| #define brightness1 51 | ||||
| #define brightness2 102 | ||||
| #define brightness3 153 | ||||
| #define brightness4 204 | ||||
| #define brightness5 255 | ||||
|  | ||||
|  | ||||
| #ifdef ESP32 | ||||
|  | ||||
|  | ||||
| class TouchBrightnessControl : public Usermod { | ||||
|   private: | ||||
|     unsigned long lastTime = 0;         //Interval | ||||
|     unsigned long lastTouch = 0;        //Timestamp of last Touch | ||||
|     unsigned long lastRelease = 0;      //Timestamp of last Touch release | ||||
|     boolean released = true;            //current Touch state (touched/released) | ||||
|     uint16_t touchReading = 0;          //sensor reading, maybe use uint8_t??? | ||||
|     uint16_t touchDuration = 0;         //duration of last touch | ||||
|   public: | ||||
|    | ||||
|     void setup() { | ||||
|       lastTouch = millis(); | ||||
|       lastRelease = millis(); | ||||
|       lastTime = millis(); | ||||
|     } | ||||
|  | ||||
|     void loop() { | ||||
|       if (millis() - lastTime >= 50) {                           //Check every 50ms if a touch occurs | ||||
|         lastTime = millis(); | ||||
|         touchReading = touchRead(touchPin);                      //Read touch sensor on pin T0 (GPIO4 / D4) | ||||
|          | ||||
|         if(touchReading < threshold && released) {               //Touch started | ||||
|           released = false; | ||||
|           lastTouch = millis(); | ||||
|         } | ||||
|         else if(touchReading >= threshold && !released) {        //Touch released | ||||
|           released = true; | ||||
|           lastRelease = millis(); | ||||
|           touchDuration = lastRelease - lastTouch;               //Calculate duration | ||||
|         } | ||||
|          | ||||
|         //Serial.println(touchDuration); | ||||
|  | ||||
|         if(touchDuration >= 800 && released) {                   //Toggle power if button press is longer than 800ms | ||||
|           touchDuration = 0;                                     //Reset touch duration to avoid multiple actions on same touch | ||||
|           toggleOnOff(); | ||||
|           colorUpdated(2);                                       //Refresh values | ||||
|         } | ||||
|         else if(touchDuration >= 100 && released) {              //Switch to next brightness if touch is between 100 and 800ms | ||||
|           touchDuration = 0;                                     //Reset touch duration to avoid multiple actions on same touch | ||||
|           if(bri < brightness1) { | ||||
|             bri = brightness1; | ||||
|           } else if(bri >= brightness1 && bri < brightness2) { | ||||
|             bri = brightness2; | ||||
|           } else if(bri >= brightness2 && bri < brightness3) { | ||||
|             bri = brightness3; | ||||
|           } else if(bri >= brightness3 && bri < brightness4) { | ||||
|             bri = brightness4; | ||||
|           } else if(bri >= brightness4 && bri < brightness5) { | ||||
|             bri = brightness5; | ||||
|           } else if(bri >= brightness5) { | ||||
|             bri = brightness1; | ||||
|           } | ||||
|           colorUpdated(2);                                       //Refresh values | ||||
|         } | ||||
|          | ||||
|       } | ||||
|     } | ||||
| }; | ||||
| #endif | ||||
							
								
								
									
										10
									
								
								usermods/EXAMPLE_v2/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | ||||
| # Usermods API v2 example usermod | ||||
|  | ||||
| In this usermod file you can find the documentation on how to take advantage of the new version 2 usermods! | ||||
|  | ||||
| ## Installation  | ||||
|  | ||||
| Copy `usermod_v2_example.h` to the wled00 directory.   | ||||
| Uncomment the corresponding lines in `usermods_list.cpp` and compile!   | ||||
| _(You shouldn't need to actually install this, it does nothing useful)_ | ||||
|  | ||||
							
								
								
									
										155
									
								
								usermods/EXAMPLE_v2/usermod_v2_example.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,155 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * Usermods allow you to add own functionality to WLED more easily | ||||
|  * See: https://github.com/Aircoookie/WLED/wiki/Add-own-functionality | ||||
|  *  | ||||
|  * This is an example for a v2 usermod. | ||||
|  * v2 usermods are class inheritance based and can (but don't have to) implement more functions, each of them is shown in this example. | ||||
|  * Multiple v2 usermods can be added to one compilation easily. | ||||
|  *  | ||||
|  * Creating a usermod: | ||||
|  * This file serves as an example. If you want to create a usermod, it is recommended to use usermod_v2_empty.h from the usermods folder as a template. | ||||
|  * Please remember to rename the class and file to a descriptive name. | ||||
|  * You may also use multiple .h and .cpp files. | ||||
|  *  | ||||
|  * Using a 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 | ||||
|  */ | ||||
|  | ||||
| //class name. Use something descriptive and leave the ": public Usermod" part :) | ||||
| class MyExampleUsermod : public Usermod { | ||||
|   private: | ||||
|     //Private class members. You can declare variables and functions only accessible to your usermod here | ||||
|     unsigned long lastTime = 0; | ||||
|   public: | ||||
|     //Functions called by WLED | ||||
|  | ||||
|     /* | ||||
|      * 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() { | ||||
|       //Serial.println("Hello from my usermod!"); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * 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 (millis() - lastTime > 1000) { | ||||
|         //Serial.println("I'm alive!"); | ||||
|         lastTime = millis(); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * 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() 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("exampleUsermod"); | ||||
|       top["great"] = userVar0; //save this var persistently whenever settings are 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 :) | ||||
|      */ | ||||
|     void readFromConfig(JsonObject& root) | ||||
|     { | ||||
|       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) | ||||
|     } | ||||
|  | ||||
|     | ||||
|     /* | ||||
|      * 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_EXAMPLE; | ||||
|     } | ||||
|  | ||||
|    //More methods can be added in the future, this example will then be extended. | ||||
|    //Your usermod will remain compatible as it does not need to implement all methods from the Usermod base class! | ||||
| }; | ||||
| Before Width: | Height: | Size: 150 KiB After Width: | Height: | Size: 136 KiB | 
| Before Width: | Height: | Size: 235 KiB After Width: | Height: | Size: 201 KiB | 
| @@ -1,30 +1,40 @@ | ||||
| # Almost universal controller board for outdoor applications | ||||
| This usermod is using ideas from @mrVanboy and @400killer | ||||
|  | ||||
| Installation of file: Copy and replace file in wled00 directory. | ||||
|  | ||||
| For BME280 sensor use usermod_bme280.cpp. Copy to wled00 and rename to usermod.cpp | ||||
|  | ||||
| ## Project repository | ||||
| -   [Original repository](https://github.com/srg74/Controller-for-WLED-firmware) - Main controller repository | ||||
| ## Features | ||||
| *   SSD1306 128x32 and 128x64 I2C OLED display | ||||
| *   On screen IP address, SSID and controller status (e.g. ON or OFF, recent effect) | ||||
| *   Auto display shutoff for saving display lifetime | ||||
| *   Dallas temperature sensor | ||||
| *   Reporting temperature to MQTT broker | ||||
| -   SSD1306 128x32 and 128x64 I2C OLED display | ||||
| -   On screen IP address, SSID and controller status (e.g. ON or OFF, recent effect) | ||||
| -   Auto display shutoff for saving display lifetime | ||||
| -   Dallas temperature sensor | ||||
| -   Reporting temperature to MQTT broker | ||||
|  | ||||
| ## Hardware | ||||
|  | ||||
|  | ||||
| ## Functionality checked with | ||||
| *   ESP-07S | ||||
| *   PlatformIO | ||||
| *   SSD1306 128x32 I2C OLED display | ||||
| *   DS18B20 (temperature sensor) | ||||
| *   KY-022 (infrared receiver) | ||||
| *   Push button (N.O. momentary switch) | ||||
| -   ESP-07S | ||||
| -   PlatformIO | ||||
| -   SSD1306 128x32 I2C OLED display | ||||
| -   DS18B20 (temperature sensor) | ||||
| -   BME280 (temperature, humidity and pressure sensor) | ||||
| -   KY-022 (infrared receiver) | ||||
| -   Push button (N.O. momentary switch) | ||||
|  | ||||
| ### Platformio requirements | ||||
| Uncomment `U8g2@~2.27.3`,`DallasTemperature@~3.8.0`,`OneWire@~2.3.5 under` `[common]` section in `platformio.ini`: | ||||
| For Dallas sensor uncomment `U8g2@~2.27.3`,`DallasTemperature@~3.8.0`,`OneWire@~2.3.5 under` `[common]` section in `platformio.ini`: | ||||
| ```ini | ||||
| # platformio.ini | ||||
| ... | ||||
| [platformio] | ||||
| ... | ||||
| default_envs = esp07 | ||||
| ; default_envs = d1_mini | ||||
| ... | ||||
| [common] | ||||
| ... | ||||
| lib_deps_external = | ||||
| @@ -36,3 +46,23 @@ lib_deps_external = | ||||
|   OneWire@~2.3.5 | ||||
| ... | ||||
| ``` | ||||
|  | ||||
| For BME280 sensor uncomment `U8g2@~2.27.3`,`BME280@~3.0.0 under` `[common]` section in `platformio.ini`: | ||||
| ```ini | ||||
| # platformio.ini | ||||
| ... | ||||
| [platformio] | ||||
| ... | ||||
| default_envs = esp07 | ||||
| ; default_envs = d1_mini | ||||
| ... | ||||
| [common] | ||||
| ... | ||||
| lib_deps_external = | ||||
|   ... | ||||
|   #For use SSD1306 OLED display uncomment following | ||||
|   U8g2@~2.27.3 | ||||
|   #For BME280 sensor uncomment following | ||||
|   BME280@~3.0.0 | ||||
| ... | ||||
| ``` | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| #include "wled.h" | ||||
| #include <Arduino.h> | ||||
| #include <U8x8lib.h> // from https://github.com/olikraus/u8g2/
 | ||||
| #include <DallasTemperature.h> //Dallastemperature sensor
 | ||||
| //The SCL and SDA pins are defined here. 
 | ||||
| @@ -25,6 +27,7 @@ U8X8_SSD1306_128X32_UNIVISION_HW_I2C u8x8(U8X8_PIN_NONE, U8X8_PIN_SCL, U8X8_PIN_ | ||||
| void userSetup() { | ||||
|   sensor.begin(); //Start Dallas temperature sensor
 | ||||
|   u8x8.begin(); | ||||
|   //u8x8.setFlipMode(1); //Uncoment if using WLED Wemos shield 
 | ||||
|   u8x8.setPowerSave(0); | ||||
|   u8x8.setContrast(10); //Contrast setup will help to preserve OLED lifetime. In case OLED need to be brighter increase number up to 255
 | ||||
|   u8x8.setFont(u8x8_font_chroma48medium8_r); | ||||
							
								
								
									
										266
									
								
								usermods/Enclosure_with_OLED_temp_ESP07/usermod_bme280.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,266 @@ | ||||
| #include "wled.h" | ||||
| #include <Arduino.h> | ||||
| #include <U8x8lib.h> // from https://github.com/olikraus/u8g2/ | ||||
| #include <Wire.h> | ||||
| #include <BME280I2C.h> //BME280 sensor | ||||
|  | ||||
| void UpdateBME280Data(); | ||||
|  | ||||
| #define Celsius // Show temperature mesaurement in Celcius otherwise is in Fahrenheit  | ||||
| BME280I2C bme;    // Default : forced mode, standby time = 1000 ms | ||||
|                   // Oversampling = pressure ×1, temperature ×1, humidity ×1, filter off, | ||||
|  | ||||
| #ifdef ARDUINO_ARCH_ESP32 //ESP32 boards | ||||
| uint8_t SCL_PIN = 22; | ||||
| uint8_t SDA_PIN = 21; | ||||
| #else //ESP8266 boards | ||||
| uint8_t SCL_PIN = 5; | ||||
| uint8_t SDA_PIN = 4; | ||||
| // uint8_t RST_PIN = 16; // Uncoment for Heltec WiFi-Kit-8 | ||||
| #endif | ||||
|  | ||||
| //The SCL and SDA pins are defined here. | ||||
| //ESP8266 Wemos D1 mini board use SCL=5 SDA=4 while ESP32 Wemos32 mini board use SCL=22 SDA=21 | ||||
| #define U8X8_PIN_SCL SCL_PIN | ||||
| #define U8X8_PIN_SDA SDA_PIN | ||||
| //#define U8X8_PIN_RESET RST_PIN // Uncoment for Heltec WiFi-Kit-8 | ||||
|  | ||||
| // 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 | ||||
| // --> First choise of cheap I2C OLED 128X32 0.91" | ||||
| U8X8_SSD1306_128X32_UNIVISION_HW_I2C u8x8(U8X8_PIN_NONE, U8X8_PIN_SCL, U8X8_PIN_SDA); // Pins are Reset, SCL, SDA | ||||
| // --> Second choise of cheap I2C OLED 128X64 0.96" or 1.3" | ||||
| //U8X8_SSD1306_128X64_NONAME_HW_I2C u8x8(U8X8_PIN_NONE, U8X8_PIN_SCL, U8X8_PIN_SDA); // Pins are Reset, SCL, SDA | ||||
| // --> Third choise of Heltec WiFi-Kit-8 OLED 128X32 0.91" | ||||
| //U8X8_SSD1306_128X32_UNIVISION_HW_I2C u8x8(U8X8_PIN_RESET, U8X8_PIN_SCL, U8X8_PIN_SDA); // Constructor for Heltec WiFi-Kit-8 | ||||
| // gets called once at boot. Do all initialization that doesn't depend on network here | ||||
|  | ||||
| // BME280 sensor timer | ||||
| long tempTimer = millis(); | ||||
| long lastMeasure = 0; | ||||
|  | ||||
| float SensorPressure(NAN); | ||||
| float SensorTemperature(NAN); | ||||
| float SensorHumidity(NAN); | ||||
|  | ||||
| void userSetup() { | ||||
|   u8x8.begin(); | ||||
|   u8x8.setPowerSave(0); | ||||
|   u8x8.setFlipMode(1); | ||||
|   u8x8.setContrast(10); //Contrast setup will help to preserve OLED lifetime. In case OLED need to be brighter increase number up to 255 | ||||
|   u8x8.setFont(u8x8_font_chroma48medium8_r); | ||||
|   u8x8.drawString(0, 0, "Loading..."); | ||||
|   Wire.begin(SDA_PIN,SCL_PIN); | ||||
|  | ||||
| while(!bme.begin()) | ||||
|   { | ||||
|     Serial.println("Could not find BME280I2C sensor!"); | ||||
|     delay(1000); | ||||
|   } | ||||
| switch(bme.chipModel()) | ||||
|   { | ||||
|     case BME280::ChipModel_BME280: | ||||
|       Serial.println("Found BME280 sensor! Success."); | ||||
|       break; | ||||
|     case BME280::ChipModel_BMP280: | ||||
|       Serial.println("Found BMP280 sensor! No Humidity available."); | ||||
|       break; | ||||
|     default: | ||||
|       Serial.println("Found UNKNOWN sensor! Error!"); | ||||
|   } | ||||
| } | ||||
|  | ||||
| // gets called every time WiFi is (re-)connected. Initialize own network | ||||
| // interfaces here | ||||
| void userConnected() {} | ||||
|  | ||||
| // needRedraw marks if redraw is required to prevent often redrawing. | ||||
| bool needRedraw = true; | ||||
|  | ||||
| // Next variables hold the previous known values to determine if redraw is | ||||
| // required. | ||||
| String knownSsid = ""; | ||||
| IPAddress knownIp; | ||||
| uint8_t knownBrightness = 0; | ||||
| uint8_t knownMode = 0; | ||||
| uint8_t knownPalette = 0; | ||||
|  | ||||
| long lastUpdate = 0; | ||||
| long lastRedraw = 0; | ||||
| bool displayTurnedOff = false; | ||||
| // How often we are redrawing screen | ||||
| #define USER_LOOP_REFRESH_RATE_MS 5000 | ||||
|  | ||||
| void userLoop() { | ||||
|  | ||||
| // BME280 sensor MQTT publishing | ||||
|   tempTimer = millis();   | ||||
| // Timer to publish new sensor data every 60 seconds | ||||
|   if (tempTimer - lastMeasure > 60000)  | ||||
|   { | ||||
|     lastMeasure = tempTimer;     | ||||
|  | ||||
| // Check if MQTT Connected, otherwise it will crash the 8266 | ||||
|     if (mqtt != nullptr) | ||||
|     { | ||||
|       UpdateBME280Data(); | ||||
|       float board_temperature = SensorTemperature; | ||||
|       float board_pressure = SensorPressure; | ||||
|       float board_humidity = SensorHumidity; | ||||
|  | ||||
| // Create string populated with user defined device topic from the UI, and the read temperature, humidity and pressure. Then publish to MQTT server. | ||||
|       String t = String(mqttDeviceTopic); | ||||
|       t += "/temperature"; | ||||
|       mqtt->publish(t.c_str(), 0, true, String(board_temperature).c_str()); | ||||
|       String p = String(mqttDeviceTopic); | ||||
|       p += "/pressure"; | ||||
|       mqtt->publish(p.c_str(), 0, true, String(board_pressure).c_str()); | ||||
|       String h = String(mqttDeviceTopic); | ||||
|       h += "/humidity"; | ||||
|       mqtt->publish(h.c_str(), 0, true, String(board_humidity).c_str()); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // Check if we time interval for redrawing passes. | ||||
|   if (millis() - lastUpdate < USER_LOOP_REFRESH_RATE_MS) { | ||||
|     return; | ||||
|   } | ||||
|   lastUpdate = millis(); | ||||
|    | ||||
|   // Turn off display after 3 minutes with no change. | ||||
|   if(!displayTurnedOff && millis() - lastRedraw > 3*60*1000) { | ||||
|     u8x8.setPowerSave(1); | ||||
|     displayTurnedOff = true; | ||||
|   } | ||||
|  | ||||
|   // Check if values which are shown on display changed from the last time. | ||||
|   if (((apActive) ? String(apSSID) : WiFi.SSID()) != knownSsid) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownIp != (apActive ? IPAddress(4, 3, 2, 1) : WiFi.localIP())) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownBrightness != bri) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownMode != strip.getMode()) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownPalette != strip.getSegment(0).palette) { | ||||
|     needRedraw = true; | ||||
|   } | ||||
|  | ||||
|   if (!needRedraw) { | ||||
|     return; | ||||
|   } | ||||
|   needRedraw = false; | ||||
|    | ||||
|   if (displayTurnedOff) | ||||
|   { | ||||
|     u8x8.setPowerSave(0); | ||||
|     displayTurnedOff = false; | ||||
|   } | ||||
|   lastRedraw = millis(); | ||||
|  | ||||
|   // Update last known values. | ||||
|   #if defined(ESP8266) | ||||
|   knownSsid = apActive ? WiFi.softAPSSID() : WiFi.SSID(); | ||||
|   #else | ||||
|   knownSsid = WiFi.SSID(); | ||||
|   #endif | ||||
|   knownIp = apActive ? IPAddress(4, 3, 2, 1) : WiFi.localIP(); | ||||
|   knownBrightness = bri; | ||||
|   knownMode = strip.getMode(); | ||||
|   knownPalette = strip.getSegment(0).palette; | ||||
|   u8x8.clear(); | ||||
|   u8x8.setFont(u8x8_font_chroma48medium8_r); | ||||
|  | ||||
|   // First row with Wifi name | ||||
|   u8x8.setCursor(1, 0); | ||||
|   u8x8.print(knownSsid.substring(0, u8x8.getCols() > 1 ? u8x8.getCols() - 2 : 0)); | ||||
|   // Print `~` char to indicate that SSID is longer, than owr dicplay | ||||
|   if (knownSsid.length() > u8x8.getCols()) | ||||
|     u8x8.print("~"); | ||||
|  | ||||
|   // Second row with IP or Psssword | ||||
|   u8x8.setCursor(1, 1); | ||||
|   // Print password in AP mode and if led is OFF. | ||||
|   if (apActive && bri == 0) | ||||
|     u8x8.print(apPass); | ||||
|   else | ||||
|     u8x8.print(knownIp); | ||||
|  | ||||
|   // Third row with mode name | ||||
|   u8x8.setCursor(2, 2); | ||||
|   uint8_t qComma = 0; | ||||
|   bool insideQuotes = false; | ||||
|   uint8_t printedChars = 0; | ||||
|   char singleJsonSymbol; | ||||
|  | ||||
|   // Find the mode name in JSON | ||||
|   for (size_t i = 0; i < strlen_P(JSON_mode_names); i++) { | ||||
|     singleJsonSymbol = pgm_read_byte_near(JSON_mode_names + i); | ||||
|     switch (singleJsonSymbol) { | ||||
|     case '"': | ||||
|       insideQuotes = !insideQuotes; | ||||
|       break; | ||||
|     case '[': | ||||
|     case ']': | ||||
|       break; | ||||
|     case ',': | ||||
|       qComma++; | ||||
|     default: | ||||
|       if (!insideQuotes || (qComma != knownMode)) | ||||
|         break; | ||||
|       u8x8.print(singleJsonSymbol); | ||||
|       printedChars++; | ||||
|     } | ||||
|     if ((qComma > knownMode) || (printedChars > u8x8.getCols() - 2)) | ||||
|       break; | ||||
|   } | ||||
|   // Fourth row with palette name | ||||
|   u8x8.setCursor(2, 3); | ||||
|   qComma = 0; | ||||
|   insideQuotes = false; | ||||
|   printedChars = 0; | ||||
|   // Looking for palette name in JSON. | ||||
|   for (size_t i = 0; i < strlen_P(JSON_palette_names); i++) { | ||||
|     singleJsonSymbol = pgm_read_byte_near(JSON_palette_names + i); | ||||
|     switch (singleJsonSymbol) { | ||||
|     case '"': | ||||
|       insideQuotes = !insideQuotes; | ||||
|       break; | ||||
|     case '[': | ||||
|     case ']': | ||||
|       break; | ||||
|     case ',': | ||||
|       qComma++; | ||||
|     default: | ||||
|       if (!insideQuotes || (qComma != knownPalette)) | ||||
|         break; | ||||
|       u8x8.print(singleJsonSymbol); | ||||
|       printedChars++; | ||||
|     } | ||||
|     if ((qComma > knownMode) || (printedChars > u8x8.getCols() - 2)) | ||||
|       break; | ||||
|   } | ||||
|  | ||||
|   u8x8.setFont(u8x8_font_open_iconic_embedded_1x1); | ||||
|   u8x8.drawGlyph(0, 0, 80); // wifi icon | ||||
|   u8x8.drawGlyph(0, 1, 68); // home icon | ||||
|   u8x8.setFont(u8x8_font_open_iconic_weather_2x2); | ||||
|   u8x8.drawGlyph(0, 2, 66 + (bri > 0 ? 3 : 0)); // sun/moon icon | ||||
| } | ||||
|  | ||||
| void UpdateBME280Data() { | ||||
|   float temp(NAN), hum(NAN), pres(NAN); | ||||
| #ifdef Celsius | ||||
|   BME280::TempUnit tempUnit(BME280::TempUnit_Celsius); | ||||
| #else | ||||
|   BME280::TempUnit tempUnit(BME280::TempUnit_Fahrenheit); | ||||
| #endif | ||||
|   BME280::PresUnit presUnit(BME280::PresUnit_Pa); | ||||
|   bme.read(pres, temp, hum, tempUnit, presUnit); | ||||
|   SensorTemperature=temp; | ||||
|   SensorHumidity=hum; | ||||
|   SensorPressure=pres; | ||||
| } | ||||
							
								
								
									
										69
									
								
								usermods/Fix_unreachable_netservices_v2/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,69 @@ | ||||
| # Fix unreachable net services V2 | ||||
|  | ||||
| **Attention: This usermod compiles only for ESP8266** | ||||
|  | ||||
| This usermod-v2 modification performs a ping request to the local IP address every 60 seconds. By this procedure the net services of WLED remains accessible in some problematic WLAN environments. | ||||
|  | ||||
| The modification works with static or DHCP IP address configuration. | ||||
|  | ||||
| _Story:_ | ||||
|  | ||||
| Unfortunately, with all ESP projects where a web server or other network services are running, I have the problem that after some time the web server is no longer accessible.  Now I found out that the connection is at least reestablished when a ping request is executed by the device. | ||||
|  | ||||
| With this modification, in the worst case, the network functions are not available for 60 seconds until the next ping request. | ||||
|  | ||||
| ## Webinterface | ||||
|  | ||||
| The number of pings and reconnects is displayed on the info page in the web interface. | ||||
| The ping delay can be changed. Changes persist after a reboot. | ||||
|  | ||||
| ## JSON API | ||||
|  | ||||
| The usermod supports the following state changes: | ||||
|  | ||||
| | JSON key    | Value range      | Description                     | | ||||
| |-------------|------------------|---------------------------------| | ||||
| | PingDelayMs | 5000 to 18000000 | Deactivdate/activate the sensor | | ||||
|  | ||||
|  Changes also persist after a reboot. | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| 1. Copy the file `usermod_Fix_unreachable_netservices.h` to the `wled00` directory. | ||||
| 2. Register the usermod by adding `#include "usermod_Fix_unreachable_netservices.h"` in the top and `registerUsermod(new FixUnreachableNetServices());` in the bottom of `usermods_list.cpp`. | ||||
|  | ||||
| Example **usermods_list.cpp**: | ||||
|  | ||||
| ```cpp | ||||
| #include "wled.h" | ||||
| /* | ||||
|  * Register your v2 usermods here! | ||||
|  *   (for v1 usermods using just usermod.cpp, you can ignore this file) | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * Add/uncomment your usermod filename here (and once more below) | ||||
|  * || || || | ||||
|  * \/ \/ \/ | ||||
|  */ | ||||
| //#include "usermod_v2_example.h" | ||||
| //#include "usermod_temperature.h" | ||||
| //#include "usermod_v2_empty.h" | ||||
| #include  "usermod_Fix_unreachable_netservices.h" | ||||
|  | ||||
| void registerUsermods() | ||||
| { | ||||
|   /* | ||||
|    * Add your usermod class name here | ||||
|    * || || || | ||||
|    * \/ \/ \/ | ||||
|    */ | ||||
|   //usermods.add(new MyExampleUsermod()); | ||||
|   //usermods.add(new UsermodTemperature()); | ||||
|   //usermods.add(new UsermodRenameMe()); | ||||
|   usermods.add(new FixUnreachableNetServices()); | ||||
|  | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Hopefully I can help someone with that - @gegu | ||||
| @@ -0,0 +1,168 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
| #if defined(ESP32) | ||||
| #warning "Usermod FixUnreachableNetServices works only with ESP8266 builds" | ||||
| class FixUnreachableNetServices : public Usermod | ||||
| { | ||||
| }; | ||||
| #endif | ||||
|  | ||||
| #if defined(ESP8266) | ||||
| #include <ping.h> | ||||
|  | ||||
| /* | ||||
|  * This usermod performs a ping request to the local IP address every 60 seconds.  | ||||
|  * By this procedure the net services of WLED remains accessible in some problematic WLAN environments. | ||||
|  *  | ||||
|  * Usermods allow you to add own functionality to WLED more easily | ||||
|  * See: https://github.com/Aircoookie/WLED/wiki/Add-own-functionality | ||||
|  *  | ||||
|  * v2 usermods are class inheritance based and can (but don't have to) implement more functions, each of them is shown in this example. | ||||
|  * Multiple v2 usermods can be added to one compilation easily. | ||||
|  *  | ||||
|  * Creating a usermod: | ||||
|  * This file serves as an example. If you want to create a usermod, it is recommended to use usermod_v2_empty.h from the usermods folder as a template. | ||||
|  * Please remember to rename the class and file to a descriptive name. | ||||
|  * You may also use multiple .h and .cpp files. | ||||
|  *  | ||||
|  * Using a 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 | ||||
|  */ | ||||
|  | ||||
| class FixUnreachableNetServices : public Usermod | ||||
| { | ||||
| private: | ||||
|   //Private class members. You can declare variables and functions only accessible to your usermod here | ||||
|   unsigned long m_lastTime = 0; | ||||
|  | ||||
|   // declare required variables | ||||
|   unsigned long m_pingDelayMs = 60000; | ||||
|   unsigned long m_connectedWiFi = 0; | ||||
|   ping_option m_pingOpt; | ||||
|   unsigned int m_pingCount = 0; | ||||
|   bool m_updateConfig = false; | ||||
|  | ||||
| public: | ||||
|   //Functions called by WLED | ||||
|  | ||||
|   /** | ||||
|    * 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() | ||||
|   { | ||||
|     //Serial.println("Hello from my usermod!"); | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * connected() is called every time the WiFi is (re)connected | ||||
|    * Use it to initialize network interfaces | ||||
|    */ | ||||
|   void connected() | ||||
|   { | ||||
|     //Serial.println("Connected to WiFi!"); | ||||
|  | ||||
|     ++m_connectedWiFi; | ||||
|  | ||||
|     // initialize ping_options structure | ||||
|     memset(&m_pingOpt, 0, sizeof(struct ping_option)); | ||||
|     m_pingOpt.count = 1; | ||||
|     m_pingOpt.ip = WiFi.localIP(); | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * loop | ||||
|    */ | ||||
|   void loop() | ||||
|   { | ||||
|     if (m_connectedWiFi > 0 && millis() - m_lastTime > m_pingDelayMs) | ||||
|     { | ||||
|       ping_start(&m_pingOpt); | ||||
|       m_lastTime = millis(); | ||||
|       ++m_pingCount; | ||||
|     } | ||||
|     if (m_updateConfig) | ||||
|     { | ||||
|       serializeConfig(); | ||||
|       m_updateConfig = false; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * 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) | ||||
|   { | ||||
|     //this code adds "u":{"⚡ Ping fix pings": m_pingCount} to the info object | ||||
|     JsonObject user = root["u"]; | ||||
|     if (user.isNull()) | ||||
|       user = root.createNestedObject("u"); | ||||
|  | ||||
|     String uiDomString = "⚡ Ping fix pings<span style=\"display:block;padding-left:25px;\">\ | ||||
| Delay <input type=\"number\" min=\"5\" max=\"300\" value=\""; | ||||
|     uiDomString += (unsigned long)(m_pingDelayMs / 1000); | ||||
|     uiDomString += "\" onchange=\"requestJson({PingDelay:parseInt(this.value)});\">sec</span>"; | ||||
|  | ||||
|     JsonArray infoArr = user.createNestedArray(uiDomString); //name | ||||
|     infoArr.add(m_pingCount);                                              //value | ||||
|  | ||||
|     //this code adds "u":{"⚡ Reconnects": m_connectedWiFi - 1} to the info object | ||||
|     infoArr = user.createNestedArray("⚡ Reconnects"); //name | ||||
|     infoArr.add(m_connectedWiFi - 1);                        //value | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * 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["PingDelay"] = (m_pingDelayMs/1000); | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * 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["PingDelay"] != nullptr) | ||||
|     { | ||||
|       m_pingDelayMs = (1000 * max(1UL, min(300UL, root["PingDelay"].as<unsigned long>()))); | ||||
|       m_updateConfig = true; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * provide the changeable values | ||||
|    */ | ||||
|   void addToConfig(JsonObject &root) | ||||
|   { | ||||
|     JsonObject top = root.createNestedObject("FixUnreachableNetServices"); | ||||
|     top["PingDelayMs"] = m_pingDelayMs; | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * restore the changeable values | ||||
|    */ | ||||
|   void readFromConfig(JsonObject &root) | ||||
|   { | ||||
|     JsonObject top = root["FixUnreachableNetServices"]; | ||||
|     m_pingDelayMs = top["PingDelayMs"] | m_pingDelayMs; | ||||
|     m_pingDelayMs = max(5000UL, min(18000000UL, m_pingDelayMs)); | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * 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_FIXNETSERVICES; | ||||
|   } | ||||
| }; | ||||
| #endif | ||||
							
								
								
									
										9
									
								
								usermods/PIR_sensor_mqtt_v1/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | ||||
| # PIR sensor with MQTT | ||||
|  | ||||
| This simple usermod allows attaching a PIR sensor like the AM312 and publish the readings over MQTT. A message is sent when motion is detected as well as when motion has stopped. | ||||
|  | ||||
| This usermod has only been tested with the AM312 sensor though should work for any other PIR sensor. Note that this does not control the LED strip directly, it only publishes MQTT readings for use with other integrations like Home Assistant. | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| Copy and replace the file `usermod.cpp` in wled00 directory. | ||||
							
								
								
									
										55
									
								
								usermods/PIR_sensor_mqtt_v1/usermod.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,55 @@ | ||||
| #include "wled.h" | ||||
| /* | ||||
|  * This v1 usermod file allows you to add own functionality to WLED more easily | ||||
|  * See: https://github.com/Aircoookie/WLED/wiki/Add-own-functionality | ||||
|  * EEPROM bytes 2750+ are reserved for your custom use case. (if you extend #define EEPSIZE in const.h) | ||||
|  * If you just need 8 bytes, use 2551-2559 (you do not need to increase EEPSIZE) | ||||
|  *  | ||||
|  * Consider the v2 usermod API if you need a more advanced feature set! | ||||
|  */ | ||||
|  | ||||
| //Use userVar0 and userVar1 (API calls &U0=,&U1=, uint16_t) | ||||
|  | ||||
| // PIR sensor pin | ||||
| const int MOTION_PIN = 16; | ||||
|  // MQTT topic for sensor values | ||||
| const char MQTT_TOPIC[] = "/motion"; | ||||
|  | ||||
| int prevState = LOW; | ||||
|  | ||||
| //gets called once at boot. Do all initialization that doesn't depend on network here | ||||
| void userSetup() | ||||
| { | ||||
|   pinMode(MOTION_PIN, INPUT); | ||||
| } | ||||
|  | ||||
| //gets called every time WiFi is (re-)connected. Initialize own network interfaces here | ||||
| void userConnected() | ||||
| { | ||||
|  | ||||
| } | ||||
|  | ||||
| void publishMqtt(String state) | ||||
| { | ||||
|   //Check if MQTT Connected, otherwise it will crash the 8266 | ||||
|   if (mqtt != nullptr){ | ||||
|     char subuf[38]; | ||||
|     strcpy(subuf, mqttDeviceTopic); | ||||
|     strcat(subuf, MQTT_TOPIC); | ||||
|     mqtt->publish(subuf, 0, true, state.c_str()); | ||||
|   } | ||||
| } | ||||
|  | ||||
| //loop. You can use "if (WLED_CONNECTED)" to check for successful connection | ||||
| void userLoop() | ||||
| { | ||||
|   if (digitalRead(MOTION_PIN) == HIGH && prevState == LOW) { // Motion detected | ||||
|     publishMqtt("ON"); | ||||
|     prevState = HIGH; | ||||
|   }  | ||||
|   if (digitalRead(MOTION_PIN) == LOW && prevState == HIGH) {  // Motion stopped | ||||
|     publishMqtt("OFF"); | ||||
|     prevState = LOW; | ||||
|   } | ||||
| } | ||||
|  | ||||
							
								
								
									
										347
									
								
								usermods/PIR_sensor_switch/PIR_Highlight_Standby
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,347 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * -------------------- | ||||
|  * Rawframe edit: | ||||
|  * - TESTED ON WLED VS.0.10.1 - WHERE ONLY PRESET 16 SAVES SEGMENTS - some macros may not be needed if this changes. | ||||
|  * - Code has been modified as my usage changed, as such it has poor use of functions vs if thens, but feel free to change it for me :) | ||||
|  *  | ||||
|  * Edited to SWITCH between two lighting scenes/modes : STANDBY and HIGHLIGHT | ||||
|  *  | ||||
|  * Usage: | ||||
|  *  - Standby is the default mode and Highlight is activated when the PIR detects activity. | ||||
|  *  - PIR delay now set to same value as Nightlight feature on boot but otherwise controlled as normal. | ||||
|  *  - Standby and Highlight brightness can be set on the fly (default values set on boot via macros calling presets). | ||||
|  *  - Macros are used to set Standby and Highlight states (macros can load saved presets etc). | ||||
|  *  | ||||
|  *    - Macro short button press   =  Highlight state default (used on boot only and sets default brightness). | ||||
|  *    - Macro double button press  =  Standby state default   (used on boot only and sets default brightness). | ||||
|  *    - Macro long button press    =  Highlight state         (after boot). | ||||
|  *    - Macro 16                   =  Standby state           (after boot). | ||||
|  * | ||||
|  *    ! It is advised not to set 'Apply preset at boot' or a boot macro (that activates a preset) as we will call our own macros on boot. | ||||
|  *  | ||||
|  *  - When the strip is off before PIR activates the strip will return to off for Standby mode, and vice versa. | ||||
|  *  - When the strip is turned off while in Highlight mode, it will return to standby mode. (This behaviour could be changed easily if for some reason you wanted the lights to go out when the pir is activated). | ||||
|  *  - Macros can be chained so you could do almost anything, such as have standby mode also turn on the nightlight function with a new time delay. | ||||
|  *  | ||||
|  * Segment Notes: | ||||
|  * - It's easier to save the segment selections in preset than apply via macro while we a limited to preset 16. (Ie, instead of selecting sections at the point of activating standby/highlight modes).  | ||||
|  * - Because only preset 16 saves segments, for now we are having to use addiotional macros to control segments where they are involved. Macros can be chained so this works but it would be better if macros also accepted json-api commands. (Testing http api segement behaviour of SS with SB left me a little confused). | ||||
|  *  | ||||
|  * Future: | ||||
|  *  - Maybe a second timer/timetable that turns on/off standby mode also after set inactivity period / date & times. For now this can be achieved others ways so may not be worth eating more processing power. | ||||
|  *  | ||||
|  * -------------------- | ||||
|  *  | ||||
|  * This usermod handles PIR sensor states. | ||||
|  * The strip will be switched on and the off timer will be resetted when the sensor goes HIGH.  | ||||
|  * When the sensor state goes LOW, the off timer is started and when it expires, the strip is switched off.  | ||||
|  *  | ||||
|  *  | ||||
|  * Usermods allow you to add own functionality to WLED more easily | ||||
|  * See: https://github.com/Aircoookie/WLED/wiki/Add-own-functionality | ||||
|  *  | ||||
|  * v2 usermods are class inheritance based and can (but don't have to) implement more functions, each of them is shown in this example. | ||||
|  * Multiple v2 usermods can be added to one compilation easily. | ||||
|  *  | ||||
|  * Creating a usermod: | ||||
|  * This file serves as an example. If you want to create a usermod, it is recommended to use usermod_v2_empty.h from the usermods folder as a template. | ||||
|  * Please remember to rename the class and file to a descriptive name. | ||||
|  * You may also use multiple .h and .cpp files. | ||||
|  *  | ||||
|  * Using a 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 | ||||
|  */ | ||||
|  | ||||
| class PIRsensorSwitch : public Usermod { | ||||
|   private: | ||||
|     // PIR sensor pin | ||||
|     const uint8_t PIRsensorPin = 13; // D7 on D1 mini | ||||
|     // notification mode for colorUpdated() | ||||
|     const byte NotifyUpdateMode = NOTIFIER_CALL_MODE_NO_NOTIFY; // NOTIFIER_CALL_MODE_DIRECT_CHANGE | ||||
|     // 1 min delay before switch off after the sensor state goes LOW | ||||
|     uint32_t m_switchOffDelay = 60000; | ||||
|     // off timer start time | ||||
|     uint32_t m_offTimerStart = 0; | ||||
|     // current PIR sensor pin state | ||||
|     byte m_PIRsensorPinState = LOW; | ||||
|     // PIR sensor enabled - ISR attached | ||||
|     bool m_PIRenabled = true; | ||||
|     // temp standby brightness store. initial value set as nightlight default target brightness | ||||
|     byte briStandby _INIT(nightlightTargetBri); | ||||
|     // temp hightlight brightness store. initial value set as current brightness | ||||
|     byte briHighlight _INIT(bri); | ||||
|     // highlight active/deactive monitor  | ||||
|     bool highlightActive = false; | ||||
|     // wled on/off state in standby mode | ||||
|     bool standbyoff = false; | ||||
|  | ||||
|     /* | ||||
|      * return or change if new PIR sensor state is available | ||||
|      */ | ||||
|     static volatile bool newPIRsensorState(bool changeState = false, bool newState = false) { | ||||
|       static volatile bool s_PIRsensorState = false; | ||||
|       if (changeState) { | ||||
|         s_PIRsensorState = newState; | ||||
|       } | ||||
|       return s_PIRsensorState; | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * PIR sensor state has changed | ||||
|      */ | ||||
|     static void IRAM_ATTR ISR_PIRstateChange() { | ||||
|       newPIRsensorState(true, true); | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * switch strip on/off | ||||
|      */ | ||||
|     // now allowing adjustable standby and highlight brightness | ||||
|     void switchStrip(bool switchOn) { | ||||
|       //if (switchOn && bri == 0) { | ||||
|       if (switchOn) { // **pir sensor is on and activated**  | ||||
|         //bri = briLast; | ||||
|         if (bri != 0) { // is WLED currently on | ||||
|           if (highlightActive) { // and is Highlight already on | ||||
|             briHighlight = bri; // then update highlight brightness with current brightness | ||||
|           } | ||||
|           else { | ||||
|             briStandby = bri; // else update standby brightness with current brightness | ||||
|           } | ||||
|         } | ||||
|         else { // WLED is currently off | ||||
|           if (!highlightActive) { // and Highlight is not already on | ||||
|             briStandby = briLast; // then update standby brightness with last active brightness (before turned off) | ||||
|             standbyoff = true; | ||||
|           } | ||||
|           else { // and Highlight is already on | ||||
|             briHighlight = briLast; // then set hightlight brightness to last active brightness (before turned off) | ||||
|           } | ||||
|         } | ||||
|         applyMacro(16); // apply highlight lighting without brightness | ||||
|         if (bri != briHighlight) {  | ||||
|           bri = briHighlight; // set current highlight brightness to last set highlight brightness | ||||
|         } | ||||
|         colorUpdated(NotifyUpdateMode); | ||||
|         highlightActive = true; // flag highlight is on | ||||
|       }     | ||||
|       else { // **pir timer has elapsed** | ||||
|         //briLast = bri; | ||||
|         //bri = 0; | ||||
|         if (bri != 0) { // is WLED currently on | ||||
|           briHighlight = bri; // update highlight brightness with current brightness | ||||
|           if (!standbyoff) { //  | ||||
|             bri = briStandby; // set standby brightness to last set standby brightness | ||||
|           } | ||||
|           else { //  | ||||
|             briLast = briStandby; // set standby off brightness | ||||
|             bri = 0; // set power off in standby | ||||
|             standbyoff = false; // turn off flag | ||||
|           } | ||||
|           applyMacro(macroLongPress); // apply standby lighting without brightness | ||||
|         } | ||||
|         else { // WLED is currently off | ||||
|           briHighlight = briLast; // set last active brightness (before turned off) to highlight lighting brightness | ||||
|           if (!standbyoff) { //  | ||||
|             bri = briStandby; // set standby brightness to last set standby brightness | ||||
|           } | ||||
|           else { //  | ||||
|             briLast = briStandby; // set standby off brightness  | ||||
|             bri = 0; // set power off in standby | ||||
|             standbyoff = false; // turn off flag  | ||||
|           } | ||||
|           applyMacro(macroLongPress); // apply standby lighting without brightness | ||||
|         } | ||||
|         colorUpdated(NotifyUpdateMode); | ||||
|         highlightActive = false; // flag highlight is off | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * Read and update PIR sensor state. | ||||
|      * Initilize/reset switch off timer | ||||
|      */ | ||||
|     bool updatePIRsensorState() { | ||||
|       if (newPIRsensorState()) { | ||||
|         m_PIRsensorPinState = digitalRead(PIRsensorPin); | ||||
|          | ||||
|         if (m_PIRsensorPinState == HIGH) { | ||||
|           m_offTimerStart = 0; | ||||
|           switchStrip(true); | ||||
|         } | ||||
|         else if (bri != 0) { | ||||
|           // start switch off timer | ||||
|           m_offTimerStart = millis(); | ||||
|         } | ||||
|         newPIRsensorState(true, false); | ||||
|         return true; | ||||
|       } | ||||
|       return false; | ||||
|     } | ||||
|  | ||||
|     /*  | ||||
|      * switch off the strip if the delay has elapsed  | ||||
|      */ | ||||
|     bool handleOffTimer() { | ||||
|       if (m_offTimerStart > 0) { | ||||
|         if ((millis() - m_offTimerStart > m_switchOffDelay) || bri == 0 ) { // now also checking for manual power off during highlight mode | ||||
|         switchStrip(false); | ||||
|         m_offTimerStart = 0;         | ||||
|         return true; | ||||
|         } | ||||
|       } | ||||
|       return false; | ||||
|     } | ||||
|  | ||||
|   public: | ||||
|     //Functions called by WLED | ||||
|  | ||||
|     /* | ||||
|      * 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() { | ||||
|       // PIR Sensor mode INPUT_PULLUP | ||||
|       pinMode(PIRsensorPin, INPUT_PULLUP); | ||||
|       // assign interrupt function and set CHANGE mode | ||||
|       attachInterrupt(digitalPinToInterrupt(PIRsensorPin), ISR_PIRstateChange, CHANGE); | ||||
|       // set delay to nightlight default duration on boot (after which json PIRoffSec overides if needed) | ||||
|       m_switchOffDelay = (nightlightDelayMins*60000); | ||||
|       applyMacro(macroButton); // apply default highlight lighting | ||||
|       briHighlight = bri; | ||||
|       applyMacro(macroDoublePress); // apply default standby lighting with brightness | ||||
|       briStandby = bri; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * connected() is called every time the WiFi is (re)connected | ||||
|      * Use it to initialize network interfaces | ||||
|      */ | ||||
|     void connected() { | ||||
|  | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * loop() is called continuously. Here you can check for events, read sensors, etc. | ||||
|      */ | ||||
|     void loop() { | ||||
|       if (!updatePIRsensorState()) { | ||||
|         handleOffTimer(); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * 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) | ||||
|     { | ||||
|       //this code adds "u":{"⏲ PIR sensor state":uiDomString} to the info object | ||||
|       // the value contains a button to toggle the sensor enabled/disabled | ||||
|       JsonObject user = root["u"]; | ||||
|       if (user.isNull()) user = root.createNestedObject("u"); | ||||
|  | ||||
|       JsonArray infoArr = user.createNestedArray("⏲ PIR sensor state"); //name | ||||
|       String uiDomString = "<button class=\"btn infobtn\" onclick=\"requestJson({PIRenabled:"; | ||||
|       String sensorStateInfo; | ||||
|  | ||||
|       // PIR sensor state | ||||
|       if (m_PIRenabled) { | ||||
|         uiDomString += "false"; | ||||
|         sensorStateInfo = (m_PIRsensorPinState != LOW ? "active" : "inactive"); //value | ||||
|       } else { | ||||
|         uiDomString += "true"; | ||||
|         sensorStateInfo = "Disabled !"; | ||||
|       } | ||||
|       uiDomString += "});return false;\">"; | ||||
|       uiDomString +=  sensorStateInfo; | ||||
|       uiDomString += "</button>"; | ||||
|       infoArr.add(uiDomString); //value | ||||
|  | ||||
|       //this code adds "u":{"⏲ switch off timer":uiDomString} to the info object | ||||
|       infoArr = user.createNestedArray("⏲ switch off timer"); //name | ||||
|  | ||||
|       // off timer | ||||
|       if (m_offTimerStart > 0) { | ||||
|         uiDomString = ""; | ||||
|         unsigned int offSeconds = (m_switchOffDelay - (millis() - m_offTimerStart)) / 1000; | ||||
|         if (offSeconds >= 3600) { | ||||
|           uiDomString += (offSeconds / 3600);  | ||||
|           uiDomString += " hours ";  | ||||
|           offSeconds %= 3600; | ||||
|         } | ||||
|         if (offSeconds >= 60) { | ||||
|           uiDomString += (offSeconds / 60);  | ||||
|           offSeconds %= 60; | ||||
|         } else if (uiDomString.length() > 0){ | ||||
|           uiDomString += 0;  | ||||
|         } | ||||
|         if (uiDomString.length() > 0){ | ||||
|           uiDomString += " min "; | ||||
|         } | ||||
|         uiDomString += (offSeconds);  | ||||
|         infoArr.add(uiDomString + " sec"); | ||||
|       } else { | ||||
|         infoArr.add("inactive"); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * 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 | ||||
|      * Add "PIRenabled" to json state. This can be used to disable/enable the sensor. | ||||
|      * Add "PIRoffSec" to json state. This can be used to adjust <m_switchOffDelay> milliseconds . | ||||
|      */ | ||||
|     void addToJsonState(JsonObject& root) | ||||
|     { | ||||
|       root["PIRenabled"] = m_PIRenabled; | ||||
|       root["PIRoffSec"] = (m_switchOffDelay / 1000); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * 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 | ||||
|      * Read "PIRenabled" from json state and switch enable/disable the PIR sensor. | ||||
|      * Read "PIRoffSec" from json state and adjust <m_switchOffDelay> milliseconds . | ||||
|      */ | ||||
|     void readFromJsonState(JsonObject& root) | ||||
|     { | ||||
|       if (root["PIRoffSec"] != nullptr) { | ||||
|         m_switchOffDelay = (1000 * max(60UL, min(43200UL, root["PIRoffSec"].as<unsigned long>()))); | ||||
|       } | ||||
|        | ||||
|       if (root["PIRenabled"] != nullptr) { | ||||
|         if (root["PIRenabled"] && !m_PIRenabled) { | ||||
|           attachInterrupt(digitalPinToInterrupt(PIRsensorPin), ISR_PIRstateChange, CHANGE); | ||||
|           newPIRsensorState(true, true); | ||||
|         }  | ||||
|         else if(m_PIRenabled) { | ||||
|           detachInterrupt(PIRsensorPin); | ||||
|         } | ||||
|         m_PIRenabled = root["PIRenabled"];           | ||||
|       } | ||||
|     } | ||||
|      | ||||
|     | ||||
|     /* | ||||
|      * 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; | ||||
|     } | ||||
|  | ||||
|    //More methods can be added in the future, this example will then be extended. | ||||
|    //Your usermod will remain compatible as it does not need to implement all methods from the Usermod base class! | ||||
| }; | ||||
							
								
								
									
										109
									
								
								usermods/PIR_sensor_switch/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,109 @@ | ||||
| # PIR sensor switch | ||||
|  | ||||
| This usermod-v2 modification allows the connection of a PIR sensor to switch on the LED strip when motion is detected. The switch-off occurs ten minutes after no more motion is detected. | ||||
|  | ||||
| _Story:_ | ||||
|  | ||||
| I use the PIR Sensor to automatically turn on the WLED analog clock in my home office room when I am there. | ||||
| The LED strip is switched [using a relay](https://github.com/Aircoookie/WLED/wiki/Control-a-relay-with-WLED) to keep the power consumption low when it is switched off. | ||||
|  | ||||
| ## Webinterface | ||||
|  | ||||
| The info page in the web interface shows the items below | ||||
|  | ||||
| - the state of the sensor. By clicking on the state the sensor can be deactivated/activated. Changes persist after a reboot. | ||||
| **I recommend to deactivate the sensor before an OTA update and activate it again afterwards**. | ||||
| - the remaining time of the off timer.  | ||||
|  | ||||
| ## JSON API | ||||
|  | ||||
| The usermod supports the following state changes: | ||||
|  | ||||
| | JSON key   | Value range | Description                     | | ||||
| |------------|-------------|---------------------------------| | ||||
| | PIRenabled | bool        | Deactivdate/activate the sensor | | ||||
| | PIRoffSec  | 60 to 43200 | Off timer seconds               | | ||||
|  | ||||
|  Changes also persist after a reboot. | ||||
|  | ||||
| ## Sensor connection | ||||
|  | ||||
| My setup uses an HC-SR501 sensor, a HC-SR505 should also work. | ||||
|  | ||||
| The usermod uses GPIO13 (D1 mini pin D7) for the sensor signal.  | ||||
| [This example page](http://www.esp8266learning.com/wemos-mini-pir-sensor-example.php) describes how to connect the sensor. | ||||
|  | ||||
| Use the potentiometers on the sensor to set the time-delay to the minimum and the sensitivity to about half, or slightly above. | ||||
|  | ||||
| ## Usermod installation | ||||
|  | ||||
| 1. Copy the file `usermod_PIR_sensor_switch.h` to the `wled00` directory. | ||||
| 2. Register the usermod by adding `#include "usermod_PIR_sensor_switch.h"` in the top and `registerUsermod(new PIRsensorSwitch());` in the bottom of `usermods_list.cpp`. | ||||
|  | ||||
| Example **usermods_list.cpp**: | ||||
|  | ||||
| ```cpp | ||||
| #include "wled.h" | ||||
| /* | ||||
|  * Register your v2 usermods here! | ||||
|  *   (for v1 usermods using just usermod.cpp, you can ignore this file) | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * Add/uncomment your usermod filename here (and once more below) | ||||
|  * || || || | ||||
|  * \/ \/ \/ | ||||
|  */ | ||||
| //#include "usermod_v2_example.h" | ||||
| //#include "usermod_temperature.h" | ||||
| //#include "usermod_v2_empty.h" | ||||
| #include "usermod_PIR_sensor_switch.h" | ||||
|  | ||||
| void registerUsermods() | ||||
| { | ||||
|   /* | ||||
|    * Add your usermod class name here | ||||
|    * || || || | ||||
|    * \/ \/ \/ | ||||
|    */ | ||||
|   //usermods.add(new MyExampleUsermod()); | ||||
|   //usermods.add(new UsermodTemperature()); | ||||
|   //usermods.add(new UsermodRenameMe()); | ||||
|   usermods.add(new PIRsensorSwitch()); | ||||
|  | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## API to enable/disable the PIR sensor from outside. For example from another usermod. | ||||
|  | ||||
| The class provides the static method `PIRsensorSwitch* PIRsensorSwitch::GetInstance()` to get a pointer to the usermod object. | ||||
|  | ||||
| To query or change the PIR sensor state the methods `bool PIRsensorEnabled()` and `void EnablePIRsensor(bool enable)` are available.  | ||||
|  | ||||
| ### There are two options to get access to the usermod instance: | ||||
|  | ||||
| 1. Include `usermod_PIR_sensor_switch.h` **before** you include the other usermod in `usermods_list.cpp' | ||||
|  | ||||
| or | ||||
|  | ||||
| 2. Use `#include "usermod_PIR_sensor_switch.h"` at the top of the `usermod.h` where you need it. | ||||
|  | ||||
| **Example usermod.h :** | ||||
| ```cpp | ||||
| #include "wled.h" | ||||
|  | ||||
| #include "usermod_PIR_sensor_switch.h" | ||||
|  | ||||
| class MyUsermod : public Usermod { | ||||
|   //... | ||||
|  | ||||
|   void togglePIRSensor() { | ||||
|     if (PIRsensorSwitch::GetInstance() != nullptr) { | ||||
|       PIRsensorSwitch::GetInstance()->EnablePIRsensor(!PIRsensorSwitch::GetInstance()->PIRsensorEnabled()); | ||||
|     } | ||||
|   } | ||||
|   //... | ||||
| }; | ||||
| ``` | ||||
|  | ||||
| Have fun - @gegu | ||||
							
								
								
									
										366
									
								
								usermods/PIR_sensor_switch/usermod_PIR_sensor_switch.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,366 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| /* | ||||
|  * This usermod handles PIR sensor states. | ||||
|  * The strip will be switched on and the off timer will be resetted when the sensor goes HIGH.  | ||||
|  * When the sensor state goes LOW, the off timer is started and when it expires, the strip is switched off.  | ||||
|  *  | ||||
|  *  | ||||
|  * Usermods allow you to add own functionality to WLED more easily | ||||
|  * See: https://github.com/Aircoookie/WLED/wiki/Add-own-functionality | ||||
|  *  | ||||
|  * v2 usermods are class inheritance based and can (but don't have to) implement more functions, each of them is shown in this example. | ||||
|  * Multiple v2 usermods can be added to one compilation easily. | ||||
|  *  | ||||
|  * Creating a usermod: | ||||
|  * This file serves as an example. If you want to create a usermod, it is recommended to use usermod_v2_empty.h from the usermods folder as a template. | ||||
|  * Please remember to rename the class and file to a descriptive name. | ||||
|  * You may also use multiple .h and .cpp files. | ||||
|  *  | ||||
|  * Using a 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 | ||||
|  */ | ||||
|  | ||||
| class PIRsensorSwitch : public Usermod | ||||
| { | ||||
| public: | ||||
|   /** | ||||
|    * constructor | ||||
|    */ | ||||
|   PIRsensorSwitch() | ||||
|   { | ||||
|     // set static instance pointer | ||||
|     PIRsensorSwitchInstance(this); | ||||
|   } | ||||
|   /** | ||||
|    * desctructor | ||||
|    */ | ||||
|   ~PIRsensorSwitch() | ||||
|   { | ||||
|     PIRsensorSwitchInstance(nullptr, true); | ||||
|     ; | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * return the instance pointer of the class | ||||
|    */ | ||||
|   static PIRsensorSwitch *GetInstance() { return PIRsensorSwitchInstance(); } | ||||
|  | ||||
|   /** | ||||
|    * Enable/Disable the PIR sensor | ||||
|    */ | ||||
|   void EnablePIRsensor(bool enable) { m_PIRenabled = enable; } | ||||
|   /** | ||||
|    * Get PIR sensor enabled/disabled state | ||||
|    */ | ||||
|   bool PIRsensorEnabled() { return m_PIRenabled; } | ||||
|  | ||||
| private: | ||||
|   // PIR sensor pin | ||||
|   const uint8_t PIRsensorPin = 13; // D7 on D1 mini | ||||
|   // notification mode for colorUpdated() | ||||
|   const byte NotifyUpdateMode = NOTIFIER_CALL_MODE_NO_NOTIFY; // NOTIFIER_CALL_MODE_DIRECT_CHANGE | ||||
|   // delay before switch off after the sensor state goes LOW | ||||
|   uint32_t m_switchOffDelay = 600000; | ||||
|   // off timer start time | ||||
|   uint32_t m_offTimerStart = 0; | ||||
|   // current PIR sensor pin state | ||||
|   byte m_PIRsensorPinState = LOW; | ||||
|   // PIR sensor enabled - ISR attached | ||||
|   bool m_PIRenabled = true; | ||||
|   // state if serializeConfig() should be called | ||||
|   bool m_updateConfig = false; | ||||
|  | ||||
|   /** | ||||
|    * return or change if new PIR sensor state is available | ||||
|    */ | ||||
|   static volatile bool newPIRsensorState(bool changeState = false, bool newState = false); | ||||
|  | ||||
|   /** | ||||
|    * PIR sensor state has changed | ||||
|    */ | ||||
|   static void IRAM_ATTR ISR_PIRstateChange(); | ||||
|  | ||||
|   /** | ||||
|    * Set/get instance pointer | ||||
|    */ | ||||
|   static PIRsensorSwitch *PIRsensorSwitchInstance(PIRsensorSwitch *pInstance = nullptr, bool bRemoveInstance = false); | ||||
|  | ||||
|   /** | ||||
|    * switch strip on/off | ||||
|    */ | ||||
|   void switchStrip(bool switchOn) | ||||
|   { | ||||
|     if (switchOn && bri == 0) | ||||
|     { | ||||
|       bri = briLast; | ||||
|       colorUpdated(NotifyUpdateMode); | ||||
|     } | ||||
|     else if (!switchOn && bri != 0) | ||||
|     { | ||||
|       briLast = bri; | ||||
|       bri = 0; | ||||
|       colorUpdated(NotifyUpdateMode); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * Read and update PIR sensor state. | ||||
|    * Initilize/reset switch off timer | ||||
|    */ | ||||
|   bool updatePIRsensorState() | ||||
|   { | ||||
|     if (newPIRsensorState()) | ||||
|     { | ||||
|       m_PIRsensorPinState = digitalRead(PIRsensorPin); | ||||
|  | ||||
|       if (m_PIRsensorPinState == HIGH) | ||||
|       { | ||||
|         m_offTimerStart = 0; | ||||
|         switchStrip(true); | ||||
|       } | ||||
|       else if (bri != 0) | ||||
|       { | ||||
|         // start switch off timer | ||||
|         m_offTimerStart = millis(); | ||||
|       } | ||||
|       newPIRsensorState(true, false); | ||||
|       return true; | ||||
|     } | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * switch off the strip if the delay has elapsed  | ||||
|    */ | ||||
|   bool handleOffTimer() | ||||
|   { | ||||
|     if (m_offTimerStart > 0 && millis() - m_offTimerStart > m_switchOffDelay) | ||||
|     { | ||||
|       if (m_PIRenabled == true) | ||||
|       { | ||||
|         switchStrip(false); | ||||
|       } | ||||
|       m_offTimerStart = 0; | ||||
|       return true; | ||||
|     } | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
| public: | ||||
|   //Functions called by WLED | ||||
|  | ||||
|   /** | ||||
|    * 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() | ||||
|   { | ||||
|     // PIR Sensor mode INPUT_PULLUP | ||||
|     pinMode(PIRsensorPin, INPUT_PULLUP); | ||||
|     if (m_PIRenabled) | ||||
|     { | ||||
|       // assign interrupt function and set CHANGE mode | ||||
|       attachInterrupt(digitalPinToInterrupt(PIRsensorPin), ISR_PIRstateChange, CHANGE); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * connected() is called every time the WiFi is (re)connected | ||||
|    * Use it to initialize network interfaces | ||||
|    */ | ||||
|   void connected() | ||||
|   { | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * loop() is called continuously. Here you can check for events, read sensors, etc. | ||||
|    */ | ||||
|   void loop() | ||||
|   { | ||||
|     if (!updatePIRsensorState()) | ||||
|     { | ||||
|       handleOffTimer(); | ||||
|       if (m_updateConfig) | ||||
|       { | ||||
|         serializeConfig(); | ||||
|         m_updateConfig = false; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * 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) | ||||
|   { | ||||
|     //this code adds "u":{"⏲ PIR sensor state":uiDomString} to the info object | ||||
|     // the value contains a button to toggle the sensor enabled/disabled | ||||
|     JsonObject user = root["u"]; | ||||
|     if (user.isNull()) | ||||
|       user = root.createNestedObject("u"); | ||||
|  | ||||
|     JsonArray infoArr = user.createNestedArray("⏲ PIR sensor state"); //name | ||||
|     String uiDomString = "<button class=\"btn infobtn\" onclick=\"requestJson({PIRenabled:"; | ||||
|     String sensorStateInfo; | ||||
|  | ||||
|     // PIR sensor state | ||||
|     if (m_PIRenabled) | ||||
|     { | ||||
|       uiDomString += "false"; | ||||
|       sensorStateInfo = (m_PIRsensorPinState != LOW ? "active" : "inactive"); //value | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|       uiDomString += "true"; | ||||
|       sensorStateInfo = "Disabled !"; | ||||
|     } | ||||
|     uiDomString += "});return false;\">"; | ||||
|     uiDomString += sensorStateInfo; | ||||
|     uiDomString += "</button>"; | ||||
|     infoArr.add(uiDomString); //value | ||||
|  | ||||
|     //this code adds "u":{"⏲ switch off timer":uiDomString} to the info object | ||||
|     uiDomString = "⏲ switch off timer<span style=\"display:block;padding-left:25px;\">\ | ||||
| after <input type=\"number\" min=\"1\" max=\"720\" value=\""; | ||||
|     uiDomString += (m_switchOffDelay / 60000); | ||||
|     uiDomString += "\" onchange=\"requestJson({PIRoffSec:parseInt(this.value)*60});\">min</span>"; | ||||
|     infoArr = user.createNestedArray(uiDomString); //name | ||||
|  | ||||
|     // off timer | ||||
|     if (m_offTimerStart > 0) | ||||
|     { | ||||
|       uiDomString = ""; | ||||
|       unsigned int offSeconds = (m_switchOffDelay - (millis() - m_offTimerStart)) / 1000; | ||||
|       if (offSeconds >= 3600) | ||||
|       { | ||||
|         uiDomString += (offSeconds / 3600); | ||||
|         uiDomString += " hours "; | ||||
|         offSeconds %= 3600; | ||||
|       } | ||||
|       if (offSeconds >= 60) | ||||
|       { | ||||
|         uiDomString += (offSeconds / 60); | ||||
|         offSeconds %= 60; | ||||
|       } | ||||
|       else if (uiDomString.length() > 0) | ||||
|       { | ||||
|         uiDomString += 0; | ||||
|       } | ||||
|       if (uiDomString.length() > 0) | ||||
|       { | ||||
|         uiDomString += " min "; | ||||
|       } | ||||
|       uiDomString += (offSeconds); | ||||
|       infoArr.add(uiDomString + " sec"); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|       infoArr.add("inactive"); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * 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 | ||||
|    * Add "PIRenabled" to json state. This can be used to disable/enable the sensor. | ||||
|    * Add "PIRoffSec" to json state. This can be used to adjust <m_switchOffDelay> milliseconds. | ||||
|    */ | ||||
|   void addToJsonState(JsonObject &root) | ||||
|   { | ||||
|     root["PIRenabled"] = m_PIRenabled; | ||||
|     root["PIRoffSec"] = (m_switchOffDelay / 1000); | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * 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 | ||||
|    * Read "PIRenabled" from json state and switch enable/disable the PIR sensor. | ||||
|    * Read "PIRoffSec" from json state and adjust <m_switchOffDelay> milliseconds. | ||||
|    */ | ||||
|   void readFromJsonState(JsonObject &root) | ||||
|   { | ||||
|     if (root["PIRoffSec"] != nullptr) | ||||
|     { | ||||
|       m_switchOffDelay = (1000 * max(60UL, min(43200UL, root["PIRoffSec"].as<unsigned long>()))); | ||||
|       m_updateConfig = true; | ||||
|     } | ||||
|  | ||||
|     if (root["PIRenabled"] != nullptr) | ||||
|     { | ||||
|       if (root["PIRenabled"] && !m_PIRenabled) | ||||
|       { | ||||
|         attachInterrupt(digitalPinToInterrupt(PIRsensorPin), ISR_PIRstateChange, CHANGE); | ||||
|         newPIRsensorState(true, true); | ||||
|       } | ||||
|       else if (m_PIRenabled) | ||||
|       { | ||||
|         detachInterrupt(PIRsensorPin); | ||||
|       } | ||||
|       m_PIRenabled = root["PIRenabled"]; | ||||
|       m_updateConfig = true; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * provide the changeable values | ||||
|    */ | ||||
|   void addToConfig(JsonObject &root) | ||||
|   { | ||||
|     JsonObject top = root.createNestedObject("PIRsensorSwitch"); | ||||
|     top["PIRenabled"] = m_PIRenabled; | ||||
|     top["PIRoffSec"] = m_switchOffDelay; | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * restore the changeable values | ||||
|    */ | ||||
|   void readFromConfig(JsonObject &root) | ||||
|   { | ||||
|     JsonObject top = root["PIRsensorSwitch"]; | ||||
|     m_PIRenabled = (top["PIRenabled"] != nullptr ? top["PIRenabled"] : true); | ||||
|     m_switchOffDelay = top["PIRoffSec"] | m_switchOffDelay; | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * 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; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| ////////////////////////////////////////////////////// | ||||
| // PIRsensorSwitch static method implementations | ||||
|  | ||||
| volatile bool PIRsensorSwitch::newPIRsensorState(bool changeState, bool newState) | ||||
| { | ||||
|   static volatile bool s_PIRsensorState = false; | ||||
|   if (changeState) | ||||
|   { | ||||
|     s_PIRsensorState = newState; | ||||
|   } | ||||
|   return s_PIRsensorState; | ||||
| } | ||||
|  | ||||
| void IRAM_ATTR PIRsensorSwitch::ISR_PIRstateChange() | ||||
| { | ||||
|   newPIRsensorState(true, true); | ||||
| } | ||||
|  | ||||
| PIRsensorSwitch *PIRsensorSwitch::PIRsensorSwitchInstance(PIRsensorSwitch *pInstance, bool bRemoveInstance) | ||||
| { | ||||
|   static PIRsensorSwitch *s_pPIRsensorSwitch = nullptr; | ||||
|   if (pInstance != nullptr || bRemoveInstance) | ||||
|   { | ||||
|     s_pPIRsensorSwitch = pInstance; | ||||
|   } | ||||
|   return s_pPIRsensorSwitch; | ||||
| } | ||||
| @@ -1,8 +1,34 @@ | ||||
| These files allow WLED 0.8.6 to report the temp sensor on the Quinled board to MQTT. I use it to report the board temp to Home Assistant via MQTT, so it will send notifications if something happens and the board start to heat up. | ||||
| # QuinLED Dig Uno board | ||||
| 
 | ||||
| These files allow WLED 0.9.1 to report the temp sensor on the Quinled board to MQTT. I use it to report the board temp to Home Assistant via MQTT, so it will send notifications if something happens and the board start to heat up. | ||||
| This code uses Aircookie's WLED software. It has a premade file for user modifications. I use it to publish the temperature from the dallas temperature sensor on the Quinled board. The entries for the top of the WLED00 file, initializes the required libraries, and variables for the sensor. The .ino file waits for 60 seconds, and checks to see if the MQTT server is connected (thanks Aircoookie). It then poles the sensor, and published it using the MQTT service already running, using the main topic programmed in the WLED UI. | ||||
| 
 | ||||
| To install: | ||||
| Installation of file: Copy and replace file in wled00 directory | ||||
| 
 | ||||
| Add the entries in the WLED00 file to the top of the same file from Aircoookies WLED. | ||||
| Replace the WLED06_usermod.ino file in Aircoookies WLED folder. | ||||
| ## Project link | ||||
| 
 | ||||
| * [QuinLED-Dig-Uno](https://quinled.info/2018/09/15/quinled-dig-uno/) - Project link | ||||
| 
 | ||||
| ### Platformio requirements | ||||
| 
 | ||||
| Uncomment `DallasTemperature@~3.8.0`,`OneWire@~2.3.5 under` `[common]` section in `platformio.ini`: | ||||
| 
 | ||||
| ```ini | ||||
| # platformio.ini | ||||
| ... | ||||
| [platformio] | ||||
| ... | ||||
| ; default_envs = esp07 | ||||
| default_envs = d1_mini | ||||
| ... | ||||
| [common] | ||||
| ... | ||||
| lib_deps_external = | ||||
|   ... | ||||
|   #For use SSD1306 OLED display uncomment following | ||||
|   U8g2@~2.27.3 | ||||
|   #For Dallas sensor uncomment following 2 lines | ||||
|   DallasTemperature@~3.8.0 | ||||
|   OneWire@~2.3.5 | ||||
| ... | ||||
| ``` | ||||
| @@ -1,8 +1,21 @@ | ||||
| //starts Dallas Temp service on boot
 | ||||
| #include <Arduino.h> | ||||
| #include "wled.h" | ||||
| //Intiating code for QuinLED Dig-Uno temp sensor
 | ||||
| //Uncomment Celsius if that is your prefered temperature scale
 | ||||
| #include <DallasTemperature.h> //Dallastemperature sensor
 | ||||
| #ifdef ARDUINO_ARCH_ESP32 //ESP32 boards
 | ||||
| OneWire oneWire(18); | ||||
| #else //ESP8266 boards
 | ||||
| OneWire oneWire(14); | ||||
| #endif | ||||
| DallasTemperature sensor(&oneWire); | ||||
| long temptimer = millis(); | ||||
| long lastMeasure = 0; | ||||
| #define Celsius // Show temperature mesaurement in Celcius otherwise is in Fahrenheit
 | ||||
| void userSetup() | ||||
| { | ||||
| // Start the DS18B20 sensor
 | ||||
|   sensors.begin();  | ||||
|   sensor.begin(); | ||||
| } | ||||
|      | ||||
| //gets called every time WiFi is (re-)connected. Initialize own network interfaces here
 | ||||
| @@ -21,11 +34,11 @@ void userLoop() | ||||
|      | ||||
| //Check if MQTT Connected, otherwise it will crash the 8266
 | ||||
|     if (mqtt != nullptr){ | ||||
|       sensors.requestTemperatures(); | ||||
|       sensor.requestTemperatures(); | ||||
| 
 | ||||
| //Gets prefered temperature scale based on selection in definitions section
 | ||||
|       #ifdef Celsius | ||||
|       float board_temperature = sensors.getTempCByIndex(0); | ||||
|       float board_temperature = sensor.getTempCByIndex(0); | ||||
|       #else | ||||
|       float board_temperature = sensors.getTempFByIndex(0); | ||||
|       #endif | ||||
| @@ -1,8 +0,0 @@ | ||||
| //Intiating code for QuinLED Dig-Uno temp sensor | ||||
| //Uncomment Celsius if that is your prefered temperature scale | ||||
| #include <DallasTemperature.h> | ||||
| OneWire oneWire(14);  | ||||
| DallasTemperature sensors(&oneWire); | ||||
| long temptimer = millis(); | ||||
| long lastMeasure = 0; | ||||
| //#define Celsius | ||||
							
								
								
									
										91
									
								
								usermods/TTGO-T-Display/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,91 @@ | ||||
| # TTGO T-Display ESP32 with 240x135 TFT via SPI with TFT_eSPI | ||||
| This usermod allows use of the TTGO T-Display ESP32 module with integrated 240x135 display | ||||
| for controlling WLED and showing the following information:  | ||||
| * Current SSID | ||||
| * IP address if obtained | ||||
|   * If connected to a network, current brightness % is shown  | ||||
|   * in AP mode AP IP and password are shown | ||||
| * Current effect | ||||
| * Current palette | ||||
| * Estimated current in mA is shown (NOTE: for this to be a reasonable value, the correct LED type must be specified in the LED Prefs section) | ||||
|  | ||||
| Button pin is mapped to the onboard button next to the side actuated reset button of the TTGO T-Display board. | ||||
|  | ||||
| I have designed a 3D printed case around this board and an ["ElectroCookie"](https://amzn.to/2WCNeeA) project board, a [level shifter](https://amzn.to/3hbKu18), a [buck regulator](https://amzn.to/3mLMy0W), and a DC [power jack](https://amzn.to/3phj9NZ).  I use 12V WS2815 LED strips for my projects, and power them with 12V power supplies, so the regulator drops the voltage to the 5V level I need to power the ESP module and the level shifter.  If there is any interest in this case, which elevates the board and display on some custom extended headers to make place the screen at the top of the enclosure (with accessible buttons), let me know, and I could post the STL files.  It is a bit tricky to get the height correct, so I also designed a one-time use 3D printed solder fixture to set the board in the right location and at the correct height for the housing.  (It is one-time use because it has to be cut off after soldering to be able to remove it).  I didn't think the effort to make it in multiple pieces was worthwhile. | ||||
|  | ||||
| Usermod based on a rework of the ssd1306_i2c_oled_u8g2 usermod from the WLED repo. | ||||
|  | ||||
| ## Hardware | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Github reference for TTGO-Tdisplay | ||||
|  | ||||
| * [TTGO T-Display](https://github.com/Xinyuan-LilyGO/TTGO-T-Display) | ||||
|  | ||||
| ## Requirements | ||||
| Functionality checked with: | ||||
| * TTGO T-Display | ||||
| * PlatformIO | ||||
| * Group of 4 individual Neopixels from Adafruit, and a several full strings of 12v WS2815 LEDs. | ||||
| * The hardware design shown above should be limited to shorter strings.  For larger strings, I use a different setup with a dedicated 12v power supply and power them directly off the supply (in addition to dropping the 12v supply down to 5v with a buck regulator for the ESP module and level shifter). | ||||
|  | ||||
| ## Setup Needed: | ||||
| * As with all usermods, copy the usermod.cpp file from the TTGO-T-Display usermod folder to the wled00 folder (replacing the default usermod.cpp file). | ||||
|  | ||||
| ## Platformio Requirements | ||||
| ### Platformio.ini changes | ||||
| Under the root folder of the project, in the `platformio.ini` file, uncomment the `TFT_eSPI` line within the [common] section, under `lib_deps`: | ||||
| ```ini | ||||
| # platformio.ini | ||||
| ... | ||||
| [common] | ||||
| ... | ||||
| lib_deps = | ||||
|     ... | ||||
|   #For use of the TTGO T-Display ESP32 Module with integrated TFT display uncomment the following line   | ||||
|     #TFT_eSPI | ||||
| ... | ||||
| ``` | ||||
|  | ||||
| Also, while in the `platformio.ini` file, you must change the environment setup to build for just the esp32dev platform as follows: | ||||
|  | ||||
| Comment out the line described below: | ||||
| ```ini | ||||
| # Travis CI binaries (comment this out when building for single board) | ||||
| ; default_envs = travis_esp8266, esp01, esp01_1m_ota, travis_esp32 | ||||
| ``` | ||||
| and UNCOMMENT the following line in the 'Single binaries' section: | ||||
| ```ini | ||||
| default_envs = esp32dev | ||||
| ``` | ||||
| Save the `platformio.ini` file.  Once this is saved, the required library files should be automatically downloaded for modifications in a later step. | ||||
|  | ||||
| ### Platformio_overrides.ini (added) | ||||
| Copy the `platformio_overrides.ini` file which is contained in the `usermods/TTGO-T-Display/` folder into the root of your project folder. This file contains an override that remaps the button pin of WLED to use the on-board button to the right of the USB-C connector (when viewed with the port oriented downward - see hardware photo). | ||||
|  | ||||
| ### TFT_eSPI Library Adjustments (board selection) | ||||
| We need to modify a file in the `TFT_eSPI` library to select the correct board.  If you followed the directions to modify and save the `platformio.ini` file above, the `User_Setup_Select.h` file can be found in the `/.pio/libdeps/esp32dev/TFT_eSPI_ID1559` folder. | ||||
|  | ||||
| Modify the  `User_Setup_Select.h` file as follows: | ||||
| * Comment out the following line (which is the 'default' setup file): | ||||
| ```ini | ||||
| //#include <User_Setup.h>           // Default setup is root library folder | ||||
| ``` | ||||
| * Uncomment the following line (which points to the setup file for the TTGO T-Display): | ||||
| ```ini | ||||
| #include <User_Setups/Setup25_TTGO_T_Display.h>    // Setup file for ESP32 and TTGO T-Display ST7789V SPI bus TFT | ||||
| ``` | ||||
|  | ||||
| Run the build and it should complete correctly.  If you see a failure like this: | ||||
| ```ini | ||||
| xtensa-esp32-elf-g++: error: wled00\wled00.ino.cpp: No such file or directory | ||||
| xtensa-esp32-elf-g++: fatal error: no input files | ||||
| ``` | ||||
| Just try building again - I find that sometimes this happens on the first build attempt and subsequent attempts will build correctly. | ||||
|  | ||||
| ## Arduino IDE | ||||
| - UNTESTED | ||||
							
								
								
									
										
											BIN
										
									
								
								usermods/TTGO-T-Display/assets/ttgo-tdisplay-enclosure1a.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 708 KiB | 
							
								
								
									
										
											BIN
										
									
								
								usermods/TTGO-T-Display/assets/ttgo-tdisplay-enclosure2a.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 848 KiB | 
							
								
								
									
										
											BIN
										
									
								
								usermods/TTGO-T-Display/assets/ttgo-tdisplay-enclosure3a.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.1 MiB | 
							
								
								
									
										
											BIN
										
									
								
								usermods/TTGO-T-Display/assets/ttgo-tdisplay-enclosure4a.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 876 KiB | 
							
								
								
									
										
											BIN
										
									
								
								usermods/TTGO-T-Display/assets/ttgo_hardware1.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 613 KiB | 
							
								
								
									
										8
									
								
								usermods/TTGO-T-Display/platformio_override.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | ||||
| [env:esp32dev] | ||||
| build_flags = ${common.build_flags_esp32}  | ||||
| ; PIN defines - uncomment and change, if needed: | ||||
| ;  -D LEDPIN=2 | ||||
|    -D BTNPIN=35 | ||||
| ;  -D IRPIN=4 | ||||
| ;  -D RLYPIN=12 | ||||
| ;  -D RLYMDE=1 | ||||
							
								
								
									
										238
									
								
								usermods/TTGO-T-Display/usermod.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,238 @@ | ||||
|  | ||||
| /* | ||||
|  * This file allows you to add own functionality to WLED more easily | ||||
|  * See: https://github.com/Aircoookie/WLED/wiki/Add-own-functionality | ||||
|  * EEPROM bytes 2750+ are reserved for your custom use case. (if you extend #define EEPSIZE in const.h) | ||||
|  * bytes 2400+ are currently ununsed, but might be used for future wled features | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * Pin 2 of the TTGO T-Display serves as the data line for the LED string. | ||||
|  * Pin 35 is set up as the button pin in the platformio_overrides.ini file. | ||||
|  * The button can be set up via the macros section in the web interface. | ||||
|  * I use the button to cycle between presets. | ||||
|  * The Pin 35 button is the one on the RIGHT side of the USB-C port on the board, | ||||
|  * when the port is oriented downwards.  See readme.md file for photo. | ||||
|  * The display is set up to turn off after 5 minutes, and turns on automatically  | ||||
|  * when a change in the dipslayed info is detected (within a 5 second interval). | ||||
|  */ | ||||
|   | ||||
|  | ||||
| //Use userVar0 and userVar1 (API calls &U0=,&U1=, uint16_t) | ||||
|  | ||||
| #include "wled.h" | ||||
| #include <TFT_eSPI.h> | ||||
| #include <SPI.h> | ||||
| #include "WiFi.h" | ||||
| #include <Wire.h> | ||||
|  | ||||
| #ifndef TFT_DISPOFF | ||||
| #define TFT_DISPOFF 0x28 | ||||
| #endif | ||||
|  | ||||
| #ifndef TFT_SLPIN | ||||
| #define TFT_SLPIN   0x10 | ||||
| #endif | ||||
|  | ||||
| #define TFT_MOSI            19 | ||||
| #define TFT_SCLK            18 | ||||
| #define TFT_CS              5 | ||||
| #define TFT_DC              16 | ||||
| #define TFT_RST             23 | ||||
|  | ||||
| #define TFT_BL          4  // Display backlight control pin | ||||
| #define ADC_EN          14  // Used for enabling battery voltage measurements - not used in this program | ||||
|  | ||||
| TFT_eSPI tft = TFT_eSPI(135, 240); // Invoke custom library | ||||
|  | ||||
| //gets called once at boot. Do all initialization that doesn't depend on network here | ||||
| void userSetup() { | ||||
|     Serial.begin(115200); | ||||
|     Serial.println("Start"); | ||||
|     tft.init(); | ||||
|     tft.setRotation(3);  //Rotation here is set up for the text to be readable with the port on the left. Use 1 to flip. | ||||
|     tft.fillScreen(TFT_BLACK); | ||||
|     tft.setTextSize(2); | ||||
|     tft.setTextColor(TFT_WHITE); | ||||
|     tft.setCursor(1, 10); | ||||
|     tft.setTextDatum(MC_DATUM); | ||||
|     tft.setTextSize(3); | ||||
|     tft.print("Loading..."); | ||||
|  | ||||
|     if (TFT_BL > 0) { // TFT_BL has been set in the TFT_eSPI library in the User Setup file TTGO_T_Display.h | ||||
|          pinMode(TFT_BL, OUTPUT); // Set backlight pin to output mode | ||||
|          digitalWrite(TFT_BL, HIGH); // Turn backlight on.  | ||||
|     } | ||||
|  | ||||
|     // tft.setRotation(3); | ||||
| } | ||||
|  | ||||
| // gets called every time WiFi is (re-)connected. Initialize own network | ||||
| // interfaces here | ||||
| void userConnected() {} | ||||
|  | ||||
| // needRedraw marks if redraw is required to prevent often redrawing. | ||||
| bool needRedraw = true; | ||||
|  | ||||
| // Next variables hold the previous known values to determine if redraw is | ||||
| // required. | ||||
| String knownSsid = ""; | ||||
| IPAddress knownIp; | ||||
| uint8_t knownBrightness = 0; | ||||
| uint8_t knownMode = 0; | ||||
| uint8_t knownPalette = 0; | ||||
| uint8_t tftcharwidth = 19;  // Number of chars that fit on screen with text size set to 2 | ||||
|  | ||||
| long lastUpdate = 0; | ||||
| long lastRedraw = 0; | ||||
| bool displayTurnedOff = false; | ||||
| // How often we are redrawing screen | ||||
| #define USER_LOOP_REFRESH_RATE_MS 5000 | ||||
|  | ||||
| void userLoop() { | ||||
|  | ||||
|   // Check if we time interval for redrawing passes. | ||||
|   if (millis() - lastUpdate < USER_LOOP_REFRESH_RATE_MS) { | ||||
|     return; | ||||
|   } | ||||
|   lastUpdate = millis(); | ||||
|    | ||||
|   // Turn off display after 5 minutes with no change. | ||||
|    if(!displayTurnedOff && millis() - lastRedraw > 5*60*1000) { | ||||
|     digitalWrite(TFT_BL, LOW); // Turn backlight off.  | ||||
|     displayTurnedOff = true; | ||||
|   }  | ||||
|  | ||||
|   // Check if values which are shown on display changed from the last time. | ||||
|   if (((apActive) ? String(apSSID) : WiFi.SSID()) != knownSsid) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownIp != (apActive ? IPAddress(4, 3, 2, 1) : WiFi.localIP())) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownBrightness != bri) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownMode != strip.getMode()) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownPalette != strip.getSegment(0).palette) { | ||||
|     needRedraw = true; | ||||
|   } | ||||
|  | ||||
|   if (!needRedraw) { | ||||
|     return; | ||||
|   } | ||||
|   needRedraw = false; | ||||
|    | ||||
|   if (displayTurnedOff) | ||||
|   { | ||||
|     digitalWrite(TFT_BL, TFT_BACKLIGHT_ON); // Turn backlight on. | ||||
|     displayTurnedOff = false; | ||||
|   } | ||||
|   lastRedraw = millis(); | ||||
|  | ||||
|   // Update last known values. | ||||
|   #if defined(ESP8266) | ||||
|   knownSsid = apActive ? WiFi.softAPSSID() : WiFi.SSID(); | ||||
|   #else | ||||
|   knownSsid = WiFi.SSID(); | ||||
|   #endif | ||||
|   knownIp = apActive ? IPAddress(4, 3, 2, 1) : WiFi.localIP(); | ||||
|   knownBrightness = bri; | ||||
|   knownMode = strip.getMode(); | ||||
|   knownPalette = strip.getSegment(0).palette; | ||||
|  | ||||
|   tft.fillScreen(TFT_BLACK); | ||||
|   tft.setTextSize(2); | ||||
|   // First row with Wifi name | ||||
|   tft.setCursor(1, 1); | ||||
|   tft.print(knownSsid.substring(0, tftcharwidth > 1 ? tftcharwidth - 1 : 0)); | ||||
|   // Print `~` char to indicate that SSID is longer, than our dicplay | ||||
|   if (knownSsid.length() > tftcharwidth) | ||||
|     tft.print("~"); | ||||
|  | ||||
|   // Second row with AP IP and Password or IP | ||||
|   tft.setTextSize(2); | ||||
|   tft.setCursor(1, 24); | ||||
|   // Print AP IP and password in AP mode or knownIP if AP not active. | ||||
|   // if (apActive && bri == 0) | ||||
|   //   tft.print(apPass); | ||||
|   // else | ||||
|   //   tft.print(knownIp); | ||||
|  | ||||
|   if (apActive) { | ||||
|     tft.print("AP IP: "); | ||||
|     tft.print(knownIp); | ||||
|     tft.setCursor(1,46); | ||||
|     tft.print("AP Pass:"); | ||||
|     tft.print(apPass); | ||||
|   } | ||||
|   else { | ||||
|     tft.print("IP: "); | ||||
|     tft.print(knownIp); | ||||
|     tft.setCursor(1,46); | ||||
|     //tft.print("Signal Strength: "); | ||||
|     //tft.print(i.wifi.signal); | ||||
|     tft.print("Brightness: "); | ||||
|     tft.print(((float(bri)/255)*100)); | ||||
|     tft.print("%"); | ||||
|   } | ||||
|  | ||||
|   // Third row with mode name | ||||
|   tft.setCursor(1, 68); | ||||
|   uint8_t qComma = 0; | ||||
|   bool insideQuotes = false; | ||||
|   uint8_t printedChars = 0; | ||||
|   char singleJsonSymbol; | ||||
|   // Find the mode name in JSON | ||||
|   for (size_t i = 0; i < strlen_P(JSON_mode_names); i++) { | ||||
|     singleJsonSymbol = pgm_read_byte_near(JSON_mode_names + i); | ||||
|     switch (singleJsonSymbol) { | ||||
|     case '"': | ||||
|       insideQuotes = !insideQuotes; | ||||
|       break; | ||||
|     case '[': | ||||
|     case ']': | ||||
|       break; | ||||
|     case ',': | ||||
|       qComma++; | ||||
|     default: | ||||
|       if (!insideQuotes || (qComma != knownMode)) | ||||
|         break; | ||||
|       tft.print(singleJsonSymbol); | ||||
|       printedChars++; | ||||
|     } | ||||
|     if ((qComma > knownMode) || (printedChars > tftcharwidth - 1)) | ||||
|       break; | ||||
|   } | ||||
|   // Fourth row with palette name | ||||
|   tft.setCursor(1, 90); | ||||
|   qComma = 0; | ||||
|   insideQuotes = false; | ||||
|   printedChars = 0; | ||||
|   // Looking for palette name in JSON. | ||||
|   for (size_t i = 0; i < strlen_P(JSON_palette_names); i++) { | ||||
|     singleJsonSymbol = pgm_read_byte_near(JSON_palette_names + i); | ||||
|     switch (singleJsonSymbol) { | ||||
|     case '"': | ||||
|       insideQuotes = !insideQuotes; | ||||
|       break; | ||||
|     case '[': | ||||
|     case ']': | ||||
|       break; | ||||
|     case ',': | ||||
|       qComma++; | ||||
|     default: | ||||
|       if (!insideQuotes || (qComma != knownPalette)) | ||||
|         break; | ||||
|       tft.print(singleJsonSymbol); | ||||
|       printedChars++; | ||||
|     } | ||||
|     // The following is modified from the code from the u8g2/u8g8 based code (knownPalette was knownMode) | ||||
|     if ((qComma > knownPalette) || (printedChars > tftcharwidth - 1)) | ||||
|       break; | ||||
|   } | ||||
|   // Fifth row with estimated mA usage | ||||
|   tft.setCursor(1, 112); | ||||
|   // Print estimated milliamp usage (must specify the LED type in LED prefs for this to be a reasonable estimate). | ||||
|   tft.print(strip.currentMilliamps); | ||||
|   tft.print("mA (estimated)"); | ||||
|    | ||||
| } | ||||
							
								
								
									
										13
									
								
								usermods/Temperature/platformio_override.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,13 @@ | ||||
| ; Options | ||||
| ; ------- | ||||
| ; USERMOD_DALLASTEMPERATURE                      - define this to have this user mod included wled00\usermods_list.cpp | ||||
| ; USERMOD_DALLASTEMPERATURE_CELSIUS              - define this to report temperatures in degrees celsius, otherwise fahrenheit will be reported | ||||
| ; USERMOD_DALLASTEMPERATURE_MEASUREMENT_INTERVAL - the number of milliseconds between measurements, defaults to 60 seconds | ||||
| ; USERMOD_DALLASTEMPERATURE_FIRST_MEASUREMENT_AT - the number of milliseconds after boot to take first measurement, defaults to 20 seconds | ||||
| ; | ||||
| [env:d1_mini_usermod_dallas_temperature_C] | ||||
| extends = env:d1_mini | ||||
| build_flags = ${common.build_flags_esp8266} -D USERMOD_DALLASTEMPERATURE -D USERMOD_DALLASTEMPERATURE_CELSIUS | ||||
| lib_deps = ${env.lib_deps} | ||||
|     milesburton/DallasTemperature@^3.9.0 | ||||
|     OneWire@~2.3.5 | ||||
							
								
								
									
										58
									
								
								usermods/Temperature/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,58 @@ | ||||
| # Temperature usermod | ||||
|  | ||||
| Based on the excellent `QuinLED_Dig_Uno_Temp_MQTT` by srg74 and 400killer!   | ||||
| This usermod will read from an attached DS18B20 temperature sensor (as available on the QuinLED Dig-Uno)   | ||||
| The temperature is displayed both in the Info section of the web UI as well as published to the `/temperature` MQTT topic if enabled.   | ||||
| This usermod will be expanded with support for different sensor types in the future. | ||||
|  | ||||
| If temperature sensor is not detected during boot, this usermod will be disabled. | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| Copy the example `platformio_override.ini` to the root directory.  This file should be placed in the same directory as `platformio.ini`. | ||||
|  | ||||
| ### Define Your Options | ||||
|  | ||||
| * `USERMOD_DALLASTEMPERATURE`                      - define this to have this user mod included wled00\usermods_list.cpp | ||||
| * `USERMOD_DALLASTEMPERATURE_CELSIUS`              - define this to report temperatures in degrees celsious, otherwise fahrenheit will be reported | ||||
| * `USERMOD_DALLASTEMPERATURE_MEASUREMENT_INTERVAL` - the number of milliseconds between measurements, defaults to 60 seconds | ||||
| * `USERMOD_DALLASTEMPERATURE_FIRST_MEASUREMENT_AT` - the number of milliseconds after boot to take first measurement, defaults to 20 seconds | ||||
|  | ||||
| ## Project link | ||||
|  | ||||
| * [QuinLED-Dig-Uno](https://quinled.info/2018/09/15/quinled-dig-uno/) - Project link | ||||
|  | ||||
| ### PlatformIO requirements | ||||
|  | ||||
| If you are using `platformio_override.ini`, you should be able to refresh the task list and see your custom task, for example `env:d1_mini_usermod_dallas_temperature_C`. | ||||
|  | ||||
|  | ||||
| If you are not using `platformio_override.ini`, you might have to uncomment `DallasTemperature@~3.8.0`,`OneWire@~2.3.5 under` `[common]` section in `platformio.ini`: | ||||
|  | ||||
| ```ini | ||||
| # platformio.ini | ||||
| ... | ||||
| [platformio] | ||||
| ... | ||||
| ; default_envs = esp07 | ||||
| default_envs = d1_mini | ||||
| ... | ||||
| [common] | ||||
| ... | ||||
| lib_deps = | ||||
|   ... | ||||
|   #For use SSD1306 OLED display uncomment following | ||||
|   U8g2@~2.27.3 | ||||
|   #For Dallas sensor uncomment following 2 lines | ||||
|   DallasTemperature@~3.8.0 | ||||
|   OneWire@~2.3.5 | ||||
| ... | ||||
| ``` | ||||
|  | ||||
| ## Change Log | ||||
|  | ||||
| 2020-09-12  | ||||
| * Changed to use async, non-blocking implementation | ||||
| * Do not report low temperatures that indicate an error to mqtt | ||||
| * Disable plugin if temperature sensor not detected | ||||
| * Report the number of seconds until the first read in the info screen instead of sensor error | ||||
							
								
								
									
										173
									
								
								usermods/Temperature/usermod_temperature.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,173 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| #include <DallasTemperature.h> //DS18B20 | ||||
|  | ||||
| //Pin defaults for QuinLed Dig-Uno | ||||
| #ifndef TEMPERATURE_PIN | ||||
| #ifdef ARDUINO_ARCH_ESP32 | ||||
| #define TEMPERATURE_PIN 18 | ||||
| #else //ESP8266 boards | ||||
| #define TEMPERATURE_PIN 14 | ||||
| #endif | ||||
| #endif | ||||
|  | ||||
| // the frequency to check temperature, 1 minute | ||||
| #ifndef USERMOD_DALLASTEMPERATURE_MEASUREMENT_INTERVAL | ||||
| #define USERMOD_DALLASTEMPERATURE_MEASUREMENT_INTERVAL 60000 | ||||
| #endif | ||||
|  | ||||
| // how many seconds after boot to take first measurement, 20 seconds | ||||
| #ifndef USERMOD_DALLASTEMPERATURE_FIRST_MEASUREMENT_AT | ||||
| #define USERMOD_DALLASTEMPERATURE_FIRST_MEASUREMENT_AT 20000  | ||||
| #endif | ||||
|  | ||||
| OneWire oneWire(TEMPERATURE_PIN); | ||||
| DallasTemperature sensor(&oneWire); | ||||
|  | ||||
| class UsermodTemperature : public Usermod { | ||||
|   private: | ||||
|     // The device's unique 64-bit serial code stored in on-board ROM. | ||||
|     // Reading directly from the sensor device address is faster than | ||||
|     // reading from index. When reading by index, DallasTemperature | ||||
|     // must first look up the device address at the specified index. | ||||
|     DeviceAddress sensorDeviceAddress; | ||||
|     // set last reading as "40 sec before boot", so first reading is taken after 20 sec | ||||
|     unsigned long lastMeasurement = UINT32_MAX - (USERMOD_DALLASTEMPERATURE_MEASUREMENT_INTERVAL - USERMOD_DALLASTEMPERATURE_FIRST_MEASUREMENT_AT); | ||||
|     // last time requestTemperatures was called | ||||
|     // used to determine when we can read the sensors temperature | ||||
|     // we have to wait at least 93.75 ms after requestTemperatures() is called | ||||
|     unsigned long lastTemperaturesRequest; | ||||
|     float temperature = -100; // default to -100, DS18B20 only goes down to -50C | ||||
|     // indicates requestTemperatures has been called but the sensor measurement is not complete | ||||
|     bool waitingForConversion = false; | ||||
|     // flag to indicate we have finished the first getTemperature call | ||||
|     // allows this library to report to the user how long until the first | ||||
|     // measurement | ||||
|     bool getTemperatureComplete = false; | ||||
|     // flag set at startup if DS18B20 sensor not found, avoids trying to keep getting | ||||
|     // temperature if flashed to a board without a sensor attached | ||||
|     bool disabled = false; | ||||
|  | ||||
|     void requestTemperatures() { | ||||
|         // there is requestTemperaturesByAddress however it  | ||||
|         // appears to do more work,  | ||||
|         // TODO: measure exection time difference | ||||
|         sensor.requestTemperatures();  | ||||
|         lastTemperaturesRequest = millis(); | ||||
|         waitingForConversion = true; | ||||
|     } | ||||
|  | ||||
|     void getTemperature() { | ||||
|       if (strip.isUpdating()) return; | ||||
|       #ifdef USERMOD_DALLASTEMPERATURE_CELSIUS | ||||
|       temperature = sensor.getTempC(sensorDeviceAddress); | ||||
|       #else | ||||
|       temperature = sensor.getTempF(sensorDeviceAddress); | ||||
|       #endif | ||||
|  | ||||
|       lastMeasurement = millis(); | ||||
|       waitingForConversion = false; | ||||
|       getTemperatureComplete = true; | ||||
|     } | ||||
|  | ||||
|   public: | ||||
|  | ||||
|  | ||||
|     void setup() { | ||||
|       sensor.begin(); | ||||
|  | ||||
|       // get the unique 64-bit serial code stored in on-board ROM | ||||
|       // if getAddress returns false, the sensor was not found | ||||
|       disabled = !sensor.getAddress(sensorDeviceAddress, 0); | ||||
|  | ||||
|       if (!disabled) { | ||||
|         DEBUG_PRINTLN(F("Dallas Temperature found")); | ||||
|         // set the resolution for this specific device | ||||
|         sensor.setResolution(sensorDeviceAddress, 9, true); | ||||
|         // do not block waiting for reading | ||||
|         sensor.setWaitForConversion(false); | ||||
|         // allocate pin & prevent other use | ||||
|         if (!pinManager.allocatePin(TEMPERATURE_PIN,false)) | ||||
|           disabled = true; | ||||
|       } else { | ||||
|         DEBUG_PRINTLN(F("Dallas Temperature not found")); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     void loop() { | ||||
|       if (disabled || strip.isUpdating()) return; | ||||
|        | ||||
|       unsigned long now = millis(); | ||||
|  | ||||
|       // check to see if we are due for taking a measurement | ||||
|       // lastMeasurement will not be updated until the conversion | ||||
|       // is complete the the reading is finished | ||||
|       if (now - lastMeasurement < USERMOD_DALLASTEMPERATURE_MEASUREMENT_INTERVAL) return; | ||||
|  | ||||
|       // we are due for a measurement, if we are not already waiting  | ||||
|       // for a conversion to complete, then make a new request for temps | ||||
|       if (!waitingForConversion) | ||||
|       { | ||||
|         requestTemperatures(); | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       // we were waiting for a conversion to complete, have we waited log enough? | ||||
|       if (now - lastTemperaturesRequest >= 94 /* 93.75ms per the datasheet */) | ||||
|       { | ||||
|         getTemperature(); | ||||
|   | ||||
|         if (WLED_MQTT_CONNECTED) { | ||||
|           char subuf[38]; | ||||
|           strcpy(subuf, mqttDeviceTopic); | ||||
|           if (-100 <= temperature) { | ||||
|             // 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")); | ||||
|             mqtt->publish(subuf, 0, true, String(temperature).c_str()); | ||||
|           } else { | ||||
|             // publish something else to indicate status? | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     void addToJsonInfo(JsonObject& root) { | ||||
|       // dont add temperature to info if we are disabled | ||||
|       if (disabled) return; | ||||
|  | ||||
|       JsonObject user = root[F("u")]; | ||||
|       if (user.isNull()) user = root.createNestedObject(F("u")); | ||||
|  | ||||
|       JsonArray temp = user.createNestedArray(F("Temperature")); | ||||
|  | ||||
|       if (!getTemperatureComplete) { | ||||
|         // if we haven't read the sensor yet, let the user know | ||||
|         // that we are still waiting for the first measurement | ||||
|         temp.add((USERMOD_DALLASTEMPERATURE_FIRST_MEASUREMENT_AT - millis()) / 1000); | ||||
|         temp.add(F(" sec until read")); | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       if (temperature <= -100) { | ||||
|         temp.add(0); | ||||
|         temp.add(F(" Sensor Error!")); | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       temp.add(temperature); | ||||
|       #ifdef USERMOD_DALLASTEMPERATURE_CELSIUS | ||||
|       temp.add(F("°C")); | ||||
|       #else | ||||
|       temp.add(F("°F")); | ||||
|       #endif | ||||
|     } | ||||
|  | ||||
|     uint16_t getId() | ||||
|     { | ||||
|       return USERMOD_ID_TEMPERATURE; | ||||
|     } | ||||
| }; | ||||
							
								
								
									
										31
									
								
								usermods/Temperature/usermods_list.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,31 @@ | ||||
| #include "wled.h" | ||||
| /* | ||||
|  * Register your v2 usermods here! | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * Add/uncomment your usermod filename here (and once more below) | ||||
|  * || || || | ||||
|  * \/ \/ \/ | ||||
|  */ | ||||
| //#include "usermod_v2_example.h" | ||||
| #ifdef USERMOD_DALLASTEMPERATURE | ||||
| #include "../usermods/Temperature/usermod_temperature.h" | ||||
| #endif | ||||
|  | ||||
| //#include "usermod_v2_empty.h" | ||||
|  | ||||
| void registerUsermods() | ||||
| { | ||||
|   /* | ||||
|    * Add your usermod class name here | ||||
|    * || || || | ||||
|    * \/ \/ \/ | ||||
|    */ | ||||
|   //usermods.add(new MyExampleUsermod()); | ||||
| #ifdef USERMOD_DALLASTEMPERATURE | ||||
|   usermods.add(new UsermodTemperature()); | ||||
| #endif | ||||
|  | ||||
|   //usermods.add(new UsermodRenameMe()); | ||||
| } | ||||
							
								
								
									
										15
									
								
								usermods/UserModv2_SunRiseAndSet/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| WLED v2 UserMod for running macros at sunrise and sunset. | ||||
|  | ||||
| At the time of this text, this user mod requires code to be changed to set certain variables: | ||||
|     1. To reflect the user's graphical location (latitude/longitude) used for calculating apparent sunrise/sunset | ||||
|     2. To specify which macros will be run at sunrise and/or sunset. (defaults to 15 at sunrise and 16 at sunset) | ||||
|     3. To optionally provide an offset from sunrise/sunset, in minutes (max of +/- 2 hours), when the macro will be run. | ||||
|  | ||||
| In addition, WLED must be configured to get time from NTP (and the time must be retrieved via NTP.) | ||||
|  | ||||
| Please open the UserMod_SunRiseAndSet.h file for instructions on what needs to be changed, where to copy files, etc. | ||||
|  | ||||
| If this usermod proves useful enough, the code might eventually be updated to allow prompting for the required information | ||||
| via the web interface and to store settings in EEPROM instead of hard-coding in the .h file. | ||||
|  | ||||
| This usermod has only been tested on the esp32dev platform, but there's no reason it wouldn't work on other platforms. | ||||
							
								
								
									
										166
									
								
								usermods/UserModv2_SunRiseAndSet/UserMod_SunRiseAndSet.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,166 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
| #include <Dusk2Dawn.h> | ||||
|  | ||||
| /* | ||||
|  *  | ||||
|  * REQUIREMENTS: | ||||
|  *      The Dusk2Dawn library must be installed.  This can be found at https://github.com/dmkishi/Dusk2Dawn.  The 1.0.1 version of this library found via | ||||
|  *      Arduino or platformio library managers is buggy and won't compile.  The latest version from github should be used. | ||||
|  *  | ||||
|  *      NTP must be enabled and functional.  It simply makes no sense to have events on sunrise/sunset when an accurate time isn't available. | ||||
|  *  | ||||
|  *      The user's geographical latitude and longitude must be configured (in decimal, not degrees/minutes/etc) using m_fLatitude and m_fLongitude | ||||
|  *  | ||||
|  *      if desired, an offset of up to +/- 2 hours can be specified for each of sunrise/sunset using m_sunriseOffset and m_sunsetOffset (defaults to 0) | ||||
|  *  | ||||
|  *      The specific macro to run at sunrise and/or sunset can be changed using m_sunriseMacro and m_sunsetMacro. (defaults to 15 and 16) | ||||
|  *  | ||||
|  *      From the Dusk2Dawn library: | ||||
|  *          HINT: An easy way to find the longitude and latitude for any location is | ||||
|  *          to find the spot in Google Maps, right click the place on the map, and | ||||
|  *          select "What's here?". At the bottom, you’ll see a card with the | ||||
|  *          coordinates. | ||||
|  *  | ||||
|  *      Once configured, copy UserMod_SunRiseAndSet.h to the sketch file (the same folder as wled00.ino exists),  | ||||
|  *      and then edit "usermods_list.cpp": | ||||
|  *          Add '#include "UserMod_SunRiseAndSet.h"' in the 'includes' area | ||||
|  *          Add 'usermods.add(new UserMod_SunRiseAndSet());' in the registerUsermods() area | ||||
|  *  | ||||
|  */ | ||||
|  | ||||
| class UserMod_SunRiseAndSet : public Usermod  | ||||
| { | ||||
| private:     | ||||
|  | ||||
|     /**** USER SETTINGS ****/ | ||||
|  | ||||
|     float m_fLatitude = 40.6;       // latitude where sunrise/set are calculated | ||||
|     float m_fLongitude = -79.80;    // longitude where sunrise/set are calculated | ||||
|     int8_t m_sunriseOffset = 0;     // offset from sunrise, in minutes, when macro should be run (negative for before sunrise, positive for after sunrise) | ||||
|     int8_t m_sunsetOffset = 0;      // offset from sunset, in minutes, when macro should be run (negative for before sunset, positive for after sunset) | ||||
|     uint8_t m_sunriseMacro = 15;    // macro number to run at sunrise | ||||
|     uint8_t m_sunsetMacro = 16;     // macro number to run at sunset | ||||
|  | ||||
|     /**** END OF USER SETTINGS.  DO NOT EDIT BELOW THIS LINE! ****/ | ||||
|      | ||||
|  | ||||
|     Dusk2Dawn *m_pD2D = NULL;         // this must be dynamically allocated in order for parameters to be loaded from EEPROM | ||||
|      | ||||
|     int m_nUserSunrise = -1;        // time, in minutes from midnight, of sunrise     | ||||
|     int m_nUserSunset = -1;         // time, in minutes from midnight, of sunset | ||||
|  | ||||
|     byte m_nLastRunMinute = -1;     // indicates what minute the userloop was last run - used so that the code only runs once per minute | ||||
|  | ||||
| public: | ||||
|  | ||||
|     virtual void setup(void) | ||||
|     { | ||||
|         /* TODO:  From EEPROM, load the following variables: | ||||
|         * | ||||
|         *   int16_t latitude16 = 4060;      // user provided latitude, multiplied by 100 and rounded | ||||
|         *   int16_t longitude16 = -7980;    // user provided longitude, multiplied by 100 and rounded. | ||||
|         *   int8_t  sunrise_offset = 0;     // number of minutes to offset the sunrise macro trigger (positive for minutes after sunrise, negative for minutes before) | ||||
|         *   int8_t  sunset_offset = 0;     // number of minutes to offset the sunset macro trigger (positive for minutes after sunset, negative for minutes before) | ||||
|         *  | ||||
|         *   then: | ||||
|         *       m_fLatitude = (float)latitude / 100.0; | ||||
|         *       m_fLongitude = (float)longitude / 100.0; | ||||
|         *       m_sunriseOffset = sunrise_offset; | ||||
|         *       m_sunsetOffset = sunset_offset; | ||||
|         */ | ||||
|  | ||||
|         if ((0.0 != m_fLatitude) || (0.0 != m_fLongitude)) | ||||
|         { | ||||
|             m_pD2D = new Dusk2Dawn (m_fLatitude, m_fLongitude, 0 /* UTC */); | ||||
|             // can't really check for failures.  if the alloc fails, the mod just doesn't work. | ||||
|         }         | ||||
|     }         | ||||
|  | ||||
|     void loop(void)  | ||||
|     { | ||||
|         // without NTP, or a configured lat/long, none of this stuff is going to work...   | ||||
|         // As an alternative, need to figure out how to determine if the user has manually set the clock or not. | ||||
|         if (m_pD2D && (999000000L != ntpLastSyncTime)) | ||||
|         { | ||||
|             // to prevent needing to import all the timezone stuff from other modules, work completely in UTC | ||||
|             time_t timeUTC = now(); | ||||
|             tmElements_t tmNow; | ||||
|             breakTime(timeUTC, tmNow); | ||||
|             int nCurMinute = tmNow.Minute; | ||||
|  | ||||
|             if (m_nLastRunMinute != nCurMinute) //only check once a new minute begins | ||||
|             { | ||||
|                 m_nLastRunMinute = nCurMinute; | ||||
|                 int numMinutes = (60 * tmNow.Hour) + m_nLastRunMinute; // how many minutes into the day are we? | ||||
|  | ||||
|                 // check to see if sunrise/sunset should be re-determined.  Only do this if neither sunrise nor sunset  | ||||
|                 // are set.  That happens when the device has just stated, and after both sunrise/sunset have already run. | ||||
|                 if ((-1 == m_nUserSunrise) && (-1 == m_nUserSunset)) | ||||
|                 {                 | ||||
|                     m_nUserSunrise = m_pD2D->sunrise(tmNow.Year + 1970, tmNow.Month, tmNow.Day, false) % 1440; | ||||
|                     m_nUserSunset = m_pD2D->sunset(tmNow.Year + 1970, tmNow.Month, tmNow.Day, false) % 1440; | ||||
|                     if (m_nUserSunrise > numMinutes) // has sunrise already passed?  if so, recompute for tomorrow | ||||
|                     { | ||||
|                         breakTime(timeUTC + (60*60*24), tmNow); | ||||
|                         m_nUserSunrise = m_pD2D->sunrise(tmNow.Year + 1970, tmNow.Month, tmNow.Day, false) % 1440; | ||||
|                         if (m_nUserSunset > numMinutes) // if sunset has also passed, recompute that as well | ||||
|                         { | ||||
|                             m_nUserSunset = m_pD2D->sunset(tmNow.Year + 1970, tmNow.Month, tmNow.Day, false) % 1440; | ||||
|                         }                     | ||||
|                     } | ||||
|                     // offset by user provided values.  becuase the offsets are signed bytes, the max offset is just over 2 hours. | ||||
|                     m_nUserSunrise += m_sunriseOffset; | ||||
|                     m_nUserSunset += m_sunsetOffset; | ||||
|                 } | ||||
|  | ||||
|                 if (numMinutes == m_nUserSunrise)     // Good Morning! | ||||
|                 { | ||||
|                     if (m_sunriseMacro) | ||||
|                         applyMacro(m_sunriseMacro); // run macro 15 | ||||
|                     m_nUserSunrise = -1; | ||||
|                 } | ||||
|                 else if (numMinutes == m_nUserSunset)  // Good Night! | ||||
|                 { | ||||
|                     if (m_sunsetMacro)                     | ||||
|                         applyMacro(m_sunsetMacro); // run macro 16 | ||||
|                     m_nUserSunset = -1; | ||||
|                 } | ||||
|             } // if (m_nLastRunMinute != nCurMinute) | ||||
|         }  // if (m_pD2D && (999000000L != ntpLastSyncTime)) | ||||
|     } | ||||
|  | ||||
|    void addToJsonState(JsonObject& root) | ||||
|     { | ||||
|         JsonObject user = root["SunRiseAndSet"]; | ||||
|         if (user.isNull()) user = root.createNestedObject("SunRiseAndSet"); | ||||
|  | ||||
|         char buf[10]; | ||||
|         if (-1 != m_nUserSunrise) | ||||
|         { | ||||
|             snprintf(buf, 10, "%02d:%02d UTC", m_nUserSunrise / 60, m_nUserSunrise % 60); | ||||
|             user["rise"] = buf; | ||||
|         } | ||||
|         if (-1 != m_nUserSunset) | ||||
|         { | ||||
|             snprintf(buf, 10, "%02d:%02d UTC", m_nUserSunset / 60, m_nUserSunset % 60); | ||||
|             user["set"] = buf; | ||||
|         } | ||||
|         JsonObject vars = user.createNestedObject("vars"); | ||||
|         vars["lat"] = m_fLatitude; | ||||
|         vars["long"] = m_fLongitude; | ||||
|         vars["rise_mac"] = m_sunriseMacro; | ||||
|         vars["set_mac"] = m_sunsetMacro; | ||||
|         vars["rise_off"] = m_sunriseOffset; | ||||
|         vars["set_off"] = m_sunsetOffset; | ||||
|     } | ||||
|  | ||||
|     ~UserMod_SunRiseAndSet(void) | ||||
|     { | ||||
|         if (m_pD2D) delete m_pD2D; | ||||
|     } | ||||
| }; | ||||
|  | ||||
|  | ||||
|  | ||||
							
								
								
									
										71
									
								
								usermods/Wemos_D1_mini+Wemos32_mini_shield/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,71 @@ | ||||
| # Wemos D1 mini and Wemos32 mini shield | ||||
| -   Installation of file: Copy and replace file in wled00 directory | ||||
| -   For BME280 sensor use usermod_bme280.cpp. Copy to wled00 and rename to usermod.cpp | ||||
| -   Added third choice of controller Heltec WiFi-Kit-8. Totally DIY but with OLED display. | ||||
| ## Project repository | ||||
| -   [Original repository](https://github.com/srg74/WLED-wemos-shield) - WLED Wemos shield repository | ||||
| -   [Wemos shield project Wiki](https://github.com/srg74/WLED-wemos-shield/wiki) | ||||
| -   [Precompiled WLED firmware](https://github.com/srg74/WLED-wemos-shield/tree/master/resources/Firmware) | ||||
| ## Features | ||||
| -   SSD1306 128x32 or 128x64 I2C OLED display | ||||
| -   On screen IP address, SSID and controller status (e.g. ON or OFF, recent effect) | ||||
| -   Auto display shutoff for saving display lifetime | ||||
| -   Dallas temperature sensor | ||||
| -   Reporting temperature to MQTT broker | ||||
| -   Relay for energy saving | ||||
|  | ||||
| ## Hardware | ||||
|  | ||||
|  | ||||
| ## Functionality checked with | ||||
|  | ||||
| -   Wemos D1 mini original v3.1 and clones | ||||
| -   Wemos32 mini | ||||
| -   PlatformIO | ||||
| -   SSD1306 128x32 I2C OLED display | ||||
| -   DS18B20 (temperature sensor) | ||||
| -   BME280 (temperature, humidity and pressure sensor) | ||||
| -   Push button (N.O. momentary switch) | ||||
|  | ||||
| ### Platformio requirements | ||||
|  | ||||
| For Dallas sensor uncomment `U8g2@~2.27.3`,`DallasTemperature@~3.8.0`,`OneWire@~2.3.5 under` `[common]` section in `platformio.ini`: | ||||
| ```ini | ||||
| # platformio.ini | ||||
| ... | ||||
| [platformio] | ||||
| ... | ||||
| ; default_envs = esp07 | ||||
| default_envs = d1_mini | ||||
| ... | ||||
| [common] | ||||
| ... | ||||
| lib_deps_external = | ||||
|   ... | ||||
|   #For use SSD1306 OLED display uncomment following | ||||
|   U8g2@~2.27.3 | ||||
|   #For Dallas sensor uncomment following 2 lines | ||||
|   DallasTemperature@~3.8.0 | ||||
|   OneWire@~2.3.5 | ||||
| ... | ||||
| ``` | ||||
|  | ||||
| For BME280 sensor uncomment `U8g2@~2.27.3`,`BME280@~3.0.0 under` `[common]` section in `platformio.ini`: | ||||
| ```ini | ||||
| # platformio.ini | ||||
| ... | ||||
| [platformio] | ||||
| ... | ||||
| ; default_envs = esp07 | ||||
| default_envs = d1_mini | ||||
| ... | ||||
| [common] | ||||
| ... | ||||
| lib_deps_external = | ||||
|   ... | ||||
|   #For use SSD1306 OLED display uncomment following | ||||
|   U8g2@~2.27.3 | ||||
|   #For BME280 sensor uncomment following | ||||
|   BME280@~3.0.0 | ||||
| ... | ||||
| ``` | ||||
							
								
								
									
										246
									
								
								usermods/Wemos_D1_mini+Wemos32_mini_shield/usermod.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,246 @@ | ||||
| #include "wled.h" | ||||
| #include <Arduino.h> | ||||
| #include <U8x8lib.h> // from https://github.com/olikraus/u8g2/ | ||||
| #include <OneWire.h> // Dallas temperature sensor | ||||
|  | ||||
| //Dallas sensor quick reading. Credit to - Author: Peter Scargill, August 17th, 2013 | ||||
| int16_t Dallas(int x, byte start) | ||||
| { | ||||
|     OneWire DallasSensor(x); | ||||
|     byte i; | ||||
|     byte data[2]; | ||||
|     int16_t result; | ||||
|     do | ||||
|     { | ||||
|       DallasSensor.reset(); | ||||
|       DallasSensor.write(0xCC); | ||||
|       DallasSensor.write(0xBE); | ||||
|       for ( i = 0; i < 2; i++) data[i] = DallasSensor.read(); | ||||
|       result=(data[1]<<8)|data[0]; | ||||
|       result>>=4; if (data[1]&128) result|=61440; | ||||
|       if (data[0]&8) ++result; | ||||
|       DallasSensor.reset(); | ||||
|       DallasSensor.write(0xCC); | ||||
|       DallasSensor.write(0x44,1); | ||||
|       if (start) delay(1000); | ||||
|     } while (start--); | ||||
|       return result; | ||||
| } | ||||
| #ifdef ARDUINO_ARCH_ESP32 | ||||
| uint8_t SCL_PIN = 22;  | ||||
| uint8_t SDA_PIN = 21; | ||||
| uint8_t DALLAS_PIN =23; | ||||
| #else | ||||
| uint8_t SCL_PIN = 5; | ||||
| uint8_t SDA_PIN = 4; | ||||
| uint8_t DALLAS_PIN =13; | ||||
| // uint8_t RST_PIN = 16; // Uncoment for Heltec WiFi-Kit-8 | ||||
| #endif | ||||
|  | ||||
| //The SCL and SDA pins are defined here. | ||||
| //ESP8266 Wemos D1 mini board use SCL=5 SDA=4 while ESP32 Wemos32 mini board use SCL=22 SDA=21 | ||||
| #define U8X8_PIN_SCL SCL_PIN | ||||
| #define U8X8_PIN_SDA SDA_PIN | ||||
| //#define U8X8_PIN_RESET RST_PIN // Uncoment for Heltec WiFi-Kit-8 | ||||
|  | ||||
| // Dallas sensor reading timer | ||||
| long temptimer = millis(); | ||||
| long lastMeasure = 0; | ||||
| #define Celsius // Show temperature mesaurement in Celcius otherwise is in Fahrenheit  | ||||
|  | ||||
| // 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 | ||||
| // --> First choise of cheap I2C OLED 128X32 0.91" | ||||
| U8X8_SSD1306_128X32_UNIVISION_HW_I2C u8x8(U8X8_PIN_NONE, U8X8_PIN_SCL, U8X8_PIN_SDA); // Pins are Reset, SCL, SDA | ||||
| // --> Second choise of cheap I2C OLED 128X64 0.96" or 1.3" | ||||
| //U8X8_SSD1306_128X64_NONAME_HW_I2C u8x8(U8X8_PIN_NONE, U8X8_PIN_SCL, U8X8_PIN_SDA); // Pins are Reset, SCL, SDA | ||||
| // --> Third choise of Heltec WiFi-Kit-8 OLED 128X32 0.91" | ||||
| //U8X8_SSD1306_128X32_UNIVISION_HW_I2C u8x8(U8X8_PIN_RESET, U8X8_PIN_SCL, U8X8_PIN_SDA); // Constructor for Heltec WiFi-Kit-8 | ||||
| // gets called once at boot. Do all initialization that doesn't depend on network here | ||||
| void userSetup() { | ||||
| //Serial.begin(115200); | ||||
|  | ||||
|   Dallas (DALLAS_PIN,1); | ||||
|   u8x8.begin(); | ||||
|   u8x8.setPowerSave(0); | ||||
|   u8x8.setFlipMode(1); | ||||
|   u8x8.setContrast(10); //Contrast setup will help to preserve OLED lifetime. In case OLED need to be brighter increase number up to 255 | ||||
|   u8x8.setFont(u8x8_font_chroma48medium8_r); | ||||
|   u8x8.drawString(0, 0, "Loading..."); | ||||
| } | ||||
|  | ||||
| // gets called every time WiFi is (re-)connected. Initialize own network | ||||
| // interfaces here | ||||
| void userConnected() {} | ||||
|  | ||||
| // needRedraw marks if redraw is required to prevent often redrawing. | ||||
| bool needRedraw = true; | ||||
|  | ||||
| // Next variables hold the previous known values to determine if redraw is | ||||
| // required. | ||||
| String knownSsid = ""; | ||||
| IPAddress knownIp; | ||||
| uint8_t knownBrightness = 0; | ||||
| uint8_t knownMode = 0; | ||||
| uint8_t knownPalette = 0; | ||||
|  | ||||
| long lastUpdate = 0; | ||||
| long lastRedraw = 0; | ||||
| bool displayTurnedOff = false; | ||||
| // How often we are redrawing screen | ||||
| #define USER_LOOP_REFRESH_RATE_MS 5000 | ||||
|  | ||||
| void userLoop() { | ||||
|  | ||||
| //----> Dallas temperature sensor MQTT publishing | ||||
|   temptimer = millis();   | ||||
| // Timer to publishe new temperature every 60 seconds | ||||
|   if (temptimer - lastMeasure > 60000)  | ||||
|   { | ||||
|     lastMeasure = temptimer;     | ||||
| //Check if MQTT Connected, otherwise it will crash the 8266 | ||||
|     if (mqtt != nullptr) | ||||
|     { | ||||
| //      Serial.println(Dallas(DALLAS_PIN,0)); | ||||
| //Gets prefered temperature scale based on selection in definitions section | ||||
|         #ifdef Celsius | ||||
|         int16_t board_temperature = Dallas(DALLAS_PIN,0); | ||||
|         #else | ||||
|         int16_t board_temperature = (Dallas(DALLAS_PIN,0)* 1.8 + 32); | ||||
|         #endif | ||||
| //Create character string populated with user defined device topic from the UI, and the read temperature. Then publish to MQTT server. | ||||
|         String t = String(mqttDeviceTopic); | ||||
|         t += "/temperature"; | ||||
|         mqtt->publish(t.c_str(), 0, true, String(board_temperature).c_str()); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // Check if we time interval for redrawing passes. | ||||
|   if (millis() - lastUpdate < USER_LOOP_REFRESH_RATE_MS) { | ||||
|     return; | ||||
|   } | ||||
|   lastUpdate = millis(); | ||||
|    | ||||
|   // Turn off display after 3 minutes with no change. | ||||
|   if(!displayTurnedOff && millis() - lastRedraw > 3*60*1000) { | ||||
|     u8x8.setPowerSave(1); | ||||
|     displayTurnedOff = true; | ||||
|   } | ||||
|  | ||||
|   // Check if values which are shown on display changed from the last time. | ||||
|   if (((apActive) ? String(apSSID) : WiFi.SSID()) != knownSsid) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownIp != (apActive ? IPAddress(4, 3, 2, 1) : WiFi.localIP())) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownBrightness != bri) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownMode != strip.getMode()) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownPalette != strip.getSegment(0).palette) { | ||||
|     needRedraw = true; | ||||
|   } | ||||
|  | ||||
|   if (!needRedraw) { | ||||
|     return; | ||||
|   } | ||||
|   needRedraw = false; | ||||
|    | ||||
|   if (displayTurnedOff) | ||||
|   { | ||||
|     u8x8.setPowerSave(0); | ||||
|     displayTurnedOff = false; | ||||
|   } | ||||
|   lastRedraw = millis(); | ||||
|  | ||||
|   // Update last known values. | ||||
|   #ifdef ARDUINO_ARCH_ESP32 | ||||
|   knownSsid = WiFi.SSID(); | ||||
|   #else | ||||
|   knownSsid = apActive ? WiFi.softAPSSID() : WiFi.SSID(); | ||||
|   #endif | ||||
|   knownIp = apActive ? IPAddress(4, 3, 2, 1) : WiFi.localIP(); | ||||
|   knownBrightness = bri; | ||||
|   knownMode = strip.getMode(); | ||||
|   knownPalette = strip.getSegment(0).palette; | ||||
|   u8x8.clear(); | ||||
|   u8x8.setFont(u8x8_font_chroma48medium8_r); | ||||
|  | ||||
|   // First row with Wifi name | ||||
|   u8x8.setCursor(1, 0); | ||||
|   u8x8.print(knownSsid.substring(0, u8x8.getCols() > 1 ? u8x8.getCols() - 2 : 0)); | ||||
|   // Print `~` char to indicate that SSID is longer, than owr dicplay | ||||
|   if (knownSsid.length() > u8x8.getCols()) | ||||
|     u8x8.print("~"); | ||||
|  | ||||
|   // Second row with IP or Psssword | ||||
|   u8x8.setCursor(1, 1); | ||||
|   // Print password in AP mode and if led is OFF. | ||||
|   if (apActive && bri == 0) | ||||
|     u8x8.print(apPass); | ||||
|   else | ||||
|     u8x8.print(knownIp); | ||||
|  | ||||
|   // Third row with mode name | ||||
|   u8x8.setCursor(2, 2); | ||||
|   uint8_t qComma = 0; | ||||
|   bool insideQuotes = false; | ||||
|   uint8_t printedChars = 0; | ||||
|   char singleJsonSymbol; | ||||
|  | ||||
|   // Find the mode name in JSON | ||||
|   for (size_t i = 0; i < strlen_P(JSON_mode_names); i++) { | ||||
|     singleJsonSymbol = pgm_read_byte_near(JSON_mode_names + i); | ||||
|     switch (singleJsonSymbol) { | ||||
|     case '"': | ||||
|       insideQuotes = !insideQuotes; | ||||
|       break; | ||||
|     case '[': | ||||
|     case ']': | ||||
|       break; | ||||
|     case ',': | ||||
|       qComma++; | ||||
|     default: | ||||
|       if (!insideQuotes || (qComma != knownMode)) | ||||
|         break; | ||||
|       u8x8.print(singleJsonSymbol); | ||||
|       printedChars++; | ||||
|     } | ||||
|     if ((qComma > knownMode) || (printedChars > u8x8.getCols() - 2)) | ||||
|       break; | ||||
|   } | ||||
|   // Fourth row with palette name | ||||
|   u8x8.setCursor(2, 3); | ||||
|   qComma = 0; | ||||
|   insideQuotes = false; | ||||
|   printedChars = 0; | ||||
|   // Looking for palette name in JSON. | ||||
|   for (size_t i = 0; i < strlen_P(JSON_palette_names); i++) { | ||||
|     singleJsonSymbol = pgm_read_byte_near(JSON_palette_names + i); | ||||
|     switch (singleJsonSymbol) { | ||||
|     case '"': | ||||
|       insideQuotes = !insideQuotes; | ||||
|       break; | ||||
|     case '[': | ||||
|     case ']': | ||||
|       break; | ||||
|     case ',': | ||||
|       qComma++; | ||||
|     default: | ||||
|       if (!insideQuotes || (qComma != knownPalette)) | ||||
|         break; | ||||
|       u8x8.print(singleJsonSymbol); | ||||
|       printedChars++; | ||||
|     } | ||||
|     if ((qComma > knownMode) || (printedChars > u8x8.getCols() - 2)) | ||||
|       break; | ||||
|   } | ||||
|  | ||||
|   u8x8.setFont(u8x8_font_open_iconic_embedded_1x1); | ||||
|   u8x8.drawGlyph(0, 0, 80); // wifi icon | ||||
|   u8x8.drawGlyph(0, 1, 68); // home icon | ||||
|   u8x8.setFont(u8x8_font_open_iconic_weather_2x2); | ||||
|   u8x8.drawGlyph(0, 2, 66 + (bri > 0 ? 3 : 0)); // sun/moon icon | ||||
| } | ||||
							
								
								
									
										268
									
								
								usermods/Wemos_D1_mini+Wemos32_mini_shield/usermod_bme280.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,268 @@ | ||||
| #include "wled.h" | ||||
| #include <Arduino.h> | ||||
| #include <U8x8lib.h> // from https://github.com/olikraus/u8g2/ | ||||
| #include <Wire.h> | ||||
| #include <BME280I2C.h> //BME280 sensor | ||||
|  | ||||
| void UpdateBME280Data(); | ||||
|  | ||||
| #define Celsius // Show temperature mesaurement in Celcius otherwise is in Fahrenheit  | ||||
| BME280I2C bme;    // Default : forced mode, standby time = 1000 ms | ||||
|                   // Oversampling = pressure ×1, temperature ×1, humidity ×1, filter off, | ||||
|  | ||||
| #ifdef ARDUINO_ARCH_ESP32 //ESP32 boards | ||||
| uint8_t SCL_PIN = 22; | ||||
| uint8_t SDA_PIN = 21; | ||||
| #else //ESP8266 boards | ||||
| uint8_t SCL_PIN = 5; | ||||
| uint8_t SDA_PIN = 4; | ||||
| // uint8_t RST_PIN = 16; // Uncoment for Heltec WiFi-Kit-8 | ||||
| #endif | ||||
|  | ||||
| //The SCL and SDA pins are defined here. | ||||
| //ESP8266 Wemos D1 mini board use SCL=5 SDA=4 while ESP32 Wemos32 mini board use SCL=22 SDA=21 | ||||
| #define U8X8_PIN_SCL SCL_PIN | ||||
| #define U8X8_PIN_SDA SDA_PIN | ||||
| //#define U8X8_PIN_RESET RST_PIN // Uncoment for Heltec WiFi-Kit-8 | ||||
|  | ||||
| // 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 | ||||
| // --> First choise of cheap I2C OLED 128X32 0.91" | ||||
| U8X8_SSD1306_128X32_UNIVISION_HW_I2C u8x8(U8X8_PIN_NONE, U8X8_PIN_SCL, U8X8_PIN_SDA); // Pins are Reset, SCL, SDA | ||||
| // --> Second choise of cheap I2C OLED 128X64 0.96" or 1.3" | ||||
| //U8X8_SSD1306_128X64_NONAME_HW_I2C u8x8(U8X8_PIN_NONE, U8X8_PIN_SCL, U8X8_PIN_SDA); // Pins are Reset, SCL, SDA | ||||
| // --> Third choise of Heltec WiFi-Kit-8 OLED 128X32 0.91" | ||||
| //U8X8_SSD1306_128X32_UNIVISION_HW_I2C u8x8(U8X8_PIN_RESET, U8X8_PIN_SCL, U8X8_PIN_SDA); // Constructor for Heltec WiFi-Kit-8 | ||||
| // gets called once at boot. Do all initialization that doesn't depend on network here | ||||
|  | ||||
| // BME280 sensor timer | ||||
| long tempTimer = millis(); | ||||
| long lastMeasure = 0; | ||||
|  | ||||
| float SensorPressure(NAN); | ||||
| float SensorTemperature(NAN); | ||||
| float SensorHumidity(NAN); | ||||
|  | ||||
| void userSetup() { | ||||
|   u8x8.begin(); | ||||
|   u8x8.setPowerSave(0); | ||||
|   u8x8.setFlipMode(1); | ||||
|   u8x8.setContrast(10); //Contrast setup will help to preserve OLED lifetime. In case OLED need to be brighter increase number up to 255 | ||||
|   u8x8.setFont(u8x8_font_chroma48medium8_r); | ||||
|   u8x8.drawString(0, 0, "Loading..."); | ||||
|   Wire.begin(SDA_PIN,SCL_PIN); | ||||
|  | ||||
| while(!bme.begin()) | ||||
|   { | ||||
|     Serial.println("Could not find BME280I2C sensor!"); | ||||
|     delay(1000); | ||||
|   } | ||||
| switch(bme.chipModel()) | ||||
|   { | ||||
|     case BME280::ChipModel_BME280: | ||||
|       Serial.println("Found BME280 sensor! Success."); | ||||
|       break; | ||||
|     case BME280::ChipModel_BMP280: | ||||
|       Serial.println("Found BMP280 sensor! No Humidity available."); | ||||
|       break; | ||||
|     default: | ||||
|       Serial.println("Found UNKNOWN sensor! Error!"); | ||||
|   } | ||||
| } | ||||
|  | ||||
| // gets called every time WiFi is (re-)connected. Initialize own network | ||||
| // interfaces here | ||||
| void userConnected() {} | ||||
|  | ||||
| // needRedraw marks if redraw is required to prevent often redrawing. | ||||
| bool needRedraw = true; | ||||
|  | ||||
| // Next variables hold the previous known values to determine if redraw is | ||||
| // required. | ||||
| String knownSsid = ""; | ||||
| IPAddress knownIp; | ||||
| uint8_t knownBrightness = 0; | ||||
| uint8_t knownMode = 0; | ||||
| uint8_t knownPalette = 0; | ||||
|  | ||||
| long lastUpdate = 0; | ||||
| long lastRedraw = 0; | ||||
| bool displayTurnedOff = false; | ||||
| // How often we are redrawing screen | ||||
| #define USER_LOOP_REFRESH_RATE_MS 5000 | ||||
|  | ||||
| void userLoop() { | ||||
|  | ||||
| // BME280 sensor MQTT publishing | ||||
|   tempTimer = millis();   | ||||
| // Timer to publish new sensor data every 60 seconds | ||||
|   if (tempTimer - lastMeasure > 60000)  | ||||
|   { | ||||
|     lastMeasure = tempTimer;     | ||||
|  | ||||
| // Check if MQTT Connected, otherwise it will crash the 8266 | ||||
|     if (mqtt != nullptr) | ||||
|     { | ||||
|       UpdateBME280Data(); | ||||
|       float board_temperature = SensorTemperature; | ||||
|       float board_pressure = SensorPressure; | ||||
|       float board_humidity = SensorHumidity; | ||||
|  | ||||
| // Create string populated with user defined device topic from the UI, and the read temperature, humidity and pressure. Then publish to MQTT server. | ||||
|       String t = String(mqttDeviceTopic); | ||||
|       t += "/temperature"; | ||||
|       mqtt->publish(t.c_str(), 0, true, String(board_temperature).c_str()); | ||||
|       String p = String(mqttDeviceTopic); | ||||
|       p += "/pressure"; | ||||
|       mqtt->publish(p.c_str(), 0, true, String(board_pressure).c_str()); | ||||
|       String h = String(mqttDeviceTopic); | ||||
|       h += "/humidity"; | ||||
|       mqtt->publish(h.c_str(), 0, true, String(board_humidity).c_str()); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // Check if we time interval for redrawing passes. | ||||
|   if (millis() - lastUpdate < USER_LOOP_REFRESH_RATE_MS) { | ||||
|     return; | ||||
|   } | ||||
|   lastUpdate = millis(); | ||||
|    | ||||
|   // Turn off display after 3 minutes with no change. | ||||
|   if(!displayTurnedOff && millis() - lastRedraw > 3*60*1000) { | ||||
|     u8x8.setPowerSave(1); | ||||
|     displayTurnedOff = true; | ||||
|   } | ||||
|  | ||||
|   // Check if values which are shown on display changed from the last time. | ||||
|   if (((apActive) ? String(apSSID) : WiFi.SSID()) != knownSsid) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownIp != (apActive ? IPAddress(4, 3, 2, 1) : WiFi.localIP())) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownBrightness != bri) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownMode != strip.getMode()) { | ||||
|     needRedraw = true; | ||||
|   } else if (knownPalette != strip.getSegment(0).palette) { | ||||
|     needRedraw = true; | ||||
|   } | ||||
|  | ||||
|   if (!needRedraw) { | ||||
|     return; | ||||
|   } | ||||
|   needRedraw = false; | ||||
|    | ||||
|   if (displayTurnedOff) | ||||
|   { | ||||
|     u8x8.setPowerSave(0); | ||||
|     displayTurnedOff = false; | ||||
|   } | ||||
|   lastRedraw = millis(); | ||||
|  | ||||
|   // Update last known values. | ||||
|   #if defined(ESP8266) | ||||
|   knownSsid = apActive ? WiFi.softAPSSID() : WiFi.SSID(); | ||||
|   #else | ||||
|   knownSsid = WiFi.SSID(); | ||||
|   #endif | ||||
|   knownIp = apActive ? IPAddress(4, 3, 2, 1) : WiFi.localIP(); | ||||
|   knownBrightness = bri; | ||||
|   knownMode = strip.getMode(); | ||||
|   knownPalette = strip.getSegment(0).palette; | ||||
|   u8x8.clear(); | ||||
|   u8x8.setFont(u8x8_font_chroma48medium8_r); | ||||
|  | ||||
|   // First row with Wifi name | ||||
|   u8x8.setCursor(1, 0); | ||||
|   u8x8.print(knownSsid.substring(0, u8x8.getCols() > 1 ? u8x8.getCols() - 2 : 0)); | ||||
|   // Print `~` char to indicate that SSID is longer, than owr dicplay | ||||
|   if (knownSsid.length() > u8x8.getCols()) | ||||
|     u8x8.print("~"); | ||||
|  | ||||
|   // Second row with IP or Psssword | ||||
|   u8x8.setCursor(1, 1); | ||||
|   // Print password in AP mode and if led is OFF. | ||||
|   if (apActive && bri == 0) | ||||
|     u8x8.print(apPass); | ||||
|   else | ||||
|     u8x8.print(knownIp); | ||||
|  | ||||
|   // Third row with mode name | ||||
|   u8x8.setCursor(2, 2); | ||||
|   uint8_t qComma = 0; | ||||
|   bool insideQuotes = false; | ||||
|   uint8_t printedChars = 0; | ||||
|   char singleJsonSymbol; | ||||
|  | ||||
|   // Find the mode name in JSON | ||||
|   for (size_t i = 0; i < strlen_P(JSON_mode_names); i++) { | ||||
|     singleJsonSymbol = pgm_read_byte_near(JSON_mode_names + i); | ||||
|     switch (singleJsonSymbol) { | ||||
|     case '"': | ||||
|       insideQuotes = !insideQuotes; | ||||
|       break; | ||||
|     case '[': | ||||
|     case ']': | ||||
|       break; | ||||
|     case ',': | ||||
|       qComma++; | ||||
|     default: | ||||
|       if (!insideQuotes || (qComma != knownMode)) | ||||
|         break; | ||||
|       u8x8.print(singleJsonSymbol); | ||||
|       printedChars++; | ||||
|     } | ||||
|     if ((qComma > knownMode) || (printedChars > u8x8.getCols() - 2)) | ||||
|       break; | ||||
|   } | ||||
|   // Fourth row with palette name | ||||
|   u8x8.setCursor(2, 3); | ||||
|   qComma = 0; | ||||
|   insideQuotes = false; | ||||
|   printedChars = 0; | ||||
|   // Looking for palette name in JSON. | ||||
|   for (size_t i = 0; i < strlen_P(JSON_palette_names); i++) { | ||||
|     singleJsonSymbol = pgm_read_byte_near(JSON_palette_names + i); | ||||
|     switch (singleJsonSymbol) { | ||||
|     case '"': | ||||
|       insideQuotes = !insideQuotes; | ||||
|       break; | ||||
|     case '[': | ||||
|     case ']': | ||||
|       break; | ||||
|     case ',': | ||||
|       qComma++; | ||||
|     default: | ||||
|       if (!insideQuotes || (qComma != knownPalette)) | ||||
|         break; | ||||
|       u8x8.print(singleJsonSymbol); | ||||
|       printedChars++; | ||||
|     } | ||||
|     if ((qComma > knownMode) || (printedChars > u8x8.getCols() - 2)) | ||||
|       break; | ||||
|   } | ||||
|  | ||||
|   u8x8.setFont(u8x8_font_open_iconic_embedded_1x1); | ||||
|   u8x8.drawGlyph(0, 0, 80); // wifi icon | ||||
|   u8x8.drawGlyph(0, 1, 68); // home icon | ||||
|   u8x8.setFont(u8x8_font_open_iconic_weather_2x2); | ||||
|   u8x8.drawGlyph(0, 2, 66 + (bri > 0 ? 3 : 0)); // sun/moon icon | ||||
| } | ||||
|  | ||||
| void UpdateBME280Data() { | ||||
|   float temp(NAN), hum(NAN), pres(NAN); | ||||
| #ifdef Celsius | ||||
|   BME280::TempUnit tempUnit(BME280::TempUnit_Celsius); | ||||
|   BME280::PresUnit presUnit(BME280::PresUnit_Pa); | ||||
|   bme.read(pres, temp, hum, tempUnit, presUnit); | ||||
| #else | ||||
|   BME280::TempUnit tempUnit(BME280::TempUnit_Fahrenheit); | ||||
|   BME280::PresUnit presUnit(BME280::PresUnit_Pa); | ||||
|   bme.read(pres, temp, hum, tempUnit, presUnit); | ||||
| #endif | ||||
|   SensorTemperature=temp; | ||||
|   SensorHumidity=hum; | ||||
|   SensorPressure=pres; | ||||
| } | ||||
							
								
								
									
										12
									
								
								usermods/battery_keypad_controller/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | ||||
| # Battery powered controller with keypad | ||||
|  | ||||
| I'm using this controller for a festival totem. Runs on 3 18650 Cells, can deliver >5A current. | ||||
|  | ||||
| Via keypad one can select 8 presets, change effect, effect speed, effect intensity and palette. Brightness can be | ||||
| adjusted with a potentiometer. | ||||
|  | ||||
| ## Pictures | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
							
								
								
									
										
											BIN
										
									
								
								usermods/battery_keypad_controller/assets/bat-key-ctrl-1.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 299 KiB | 
							
								
								
									
										
											BIN
										
									
								
								usermods/battery_keypad_controller/assets/bat-key-ctrl-2.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 236 KiB | 
							
								
								
									
										
											BIN
										
									
								
								usermods/battery_keypad_controller/assets/bat-key-ctrl-3.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 323 KiB | 
							
								
								
									
										151
									
								
								usermods/battery_keypad_controller/wled06_usermod.ino
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,151 @@ | ||||
| /* | ||||
|  *  WLED usermod for keypad and brightness-pot. | ||||
|  *  3'2020 https://github.com/hobbyquaker | ||||
|  */ | ||||
|  | ||||
| #include <Keypad.h> | ||||
| const byte keypad_rows = 4; | ||||
| const byte keypad_cols = 4; | ||||
| char keypad_keys[keypad_rows][keypad_cols] = { | ||||
|         {'1', '2', '3', 'A'}, | ||||
|         {'4', '5', '6', 'B'}, | ||||
|         {'7', '8', '9', 'C'}, | ||||
|         {'*', '0', '#', 'D'} | ||||
| }; | ||||
|  | ||||
| byte keypad_colPins[keypad_rows] = {D3, D2, D1, D0}; | ||||
| byte keypad_rowPins[keypad_cols] = {D7, D6, D5, D4}; | ||||
|  | ||||
| Keypad myKeypad = Keypad(makeKeymap(keypad_keys), keypad_rowPins, keypad_colPins, keypad_rows, keypad_cols); | ||||
|  | ||||
| void userSetup() | ||||
| { | ||||
|  | ||||
| } | ||||
|  | ||||
| void userConnected() | ||||
| { | ||||
|  | ||||
| } | ||||
|  | ||||
| long lastTime = 0; | ||||
| int delayMs = 20; //we want to do something every 2 seconds | ||||
|  | ||||
| void userLoop() | ||||
| { | ||||
|     if (millis()-lastTime > delayMs) | ||||
|     { | ||||
|  | ||||
|         long analog = analogRead(0); | ||||
|         int new_bri = 1; | ||||
|         if (analog > 900) { | ||||
|             new_bri = 255; | ||||
|         } else if (analog > 30) { | ||||
|             new_bri = dim8_video(map(analog, 31, 900, 16, 255)); | ||||
|         } | ||||
|         if (bri != new_bri) { | ||||
|             bri = new_bri; | ||||
|             colorUpdated(1); | ||||
|  | ||||
|         } | ||||
|  | ||||
|         char myKey = myKeypad.getKey(); | ||||
|         if (myKey != NULL) { | ||||
|             switch (myKey) { | ||||
|                 case '1': | ||||
|                     applyPreset(1); | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|                 case '2': | ||||
|                     applyPreset(2); | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|                 case '3': | ||||
|                     applyPreset(3); | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|                 case '4': | ||||
|                     applyPreset(4); | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|                 case '5': | ||||
|                     applyPreset(5); | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|                 case '6': | ||||
|                     applyPreset(6); | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|                 case 'A': | ||||
|                     applyPreset(7); | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|                 case 'B': | ||||
|                     applyPreset(8); | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|  | ||||
|                 case '7': | ||||
|                     effectCurrent += 1; | ||||
|                     if (effectCurrent >= MODE_COUNT) effectCurrent = 0; | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|                 case '*': | ||||
|                     effectCurrent -= 1; | ||||
|                     if (effectCurrent < 0) effectCurrent = (MODE_COUNT-1); | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|  | ||||
|                 case '8': | ||||
|                     if (effectSpeed < 240) { | ||||
|                         effectSpeed += 12; | ||||
|                     } else if (effectSpeed < 255) { | ||||
|                         effectSpeed += 1; | ||||
|                     } | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|                 case '0': | ||||
|                     if (effectSpeed > 15) { | ||||
|                         effectSpeed -= 12; | ||||
|                     } else if (effectSpeed > 0) { | ||||
|                         effectSpeed -= 1; | ||||
|                     } | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|  | ||||
|                 case '9': | ||||
|                     if (effectIntensity < 240) { | ||||
|                         effectIntensity += 12; | ||||
|                     } else if (effectIntensity < 255) { | ||||
|                         effectIntensity += 1; | ||||
|                     } | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|                 case '#': | ||||
|                     if (effectIntensity > 15) { | ||||
|                         effectIntensity -= 12; | ||||
|                     } else if (effectIntensity > 0) { | ||||
|                         effectIntensity -= 1; | ||||
|                     } | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|  | ||||
|                 case 'C': | ||||
|                     effectPalette += 1; | ||||
|                     if (effectPalette >= 50) effectPalette = 0; | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|                 case 'D': | ||||
|                     effectPalette -= 1; | ||||
|                     if (effectPalette <= 0) effectPalette = 50; | ||||
|                     colorUpdated(NOTIFIER_CALL_MODE_FX_CHANGED); | ||||
|                     break; | ||||
|  | ||||
|             } | ||||
|  | ||||
|         } | ||||
|  | ||||
|         lastTime = millis(); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| /* | ||||
|  * This file allows you to add own functionality to WLED more easily | ||||
|  * See: https://github.com/Aircoookie/WLED/wiki/Add-own-functionality | ||||
|  * EEPROM bytes 2750+ are reserved for your custom use case. (if you extend #define EEPSIZE in wled01_eeprom.h) | ||||
|  * EEPROM bytes 2750+ are reserved for your custom use case. (if you extend #define EEPSIZE in wled_eeprom.h) | ||||
|  * bytes 2400+ are currently ununsed, but might be used for future wled features | ||||
|  */ | ||||
|  | ||||
|   | ||||
							
								
								
									
										81
									
								
								usermods/buzzer/usermod_v2_buzzer.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,81 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
| #include "Arduino.h" | ||||
|  | ||||
| #include <deque> | ||||
|  | ||||
| #define USERMOD_ID_BUZZER 900 | ||||
| #ifndef USERMOD_BUZZER_PIN | ||||
| #define USERMOD_BUZZER_PIN GPIO_NUM_32 | ||||
| #endif | ||||
|  | ||||
| /* | ||||
|  * Usermods allow you to add own functionality to WLED more easily | ||||
|  * See: https://github.com/Aircoookie/WLED/wiki/Add-own-functionality | ||||
|  *  | ||||
|  * Using a 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 | ||||
|  */ | ||||
|  | ||||
| class BuzzerUsermod : public Usermod { | ||||
|   private: | ||||
|     unsigned long lastTime_ = 0; | ||||
|     unsigned long delay_ = 0; | ||||
|     std::deque<std::pair<uint8_t, unsigned long>> sequence_ {}; | ||||
|   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() { | ||||
|       // Setup the pin, and default to LOW | ||||
|       pinMode(USERMOD_BUZZER_PIN, OUTPUT); | ||||
|       digitalWrite(USERMOD_BUZZER_PIN, LOW); | ||||
|  | ||||
|       // Beep on startup | ||||
|       sequence_.push_back({ HIGH, 50 }); | ||||
|       sequence_.push_back({ LOW, 0 }); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * connected() is called every time the WiFi is (re)connected | ||||
|      * Use it to initialize network interfaces | ||||
|      */ | ||||
|     void connected() { | ||||
|       // Double beep on WiFi | ||||
|       sequence_.push_back({ LOW, 100 }); | ||||
|       sequence_.push_back({ HIGH, 50 }); | ||||
|       sequence_.push_back({ LOW, 30 }); | ||||
|       sequence_.push_back({ HIGH, 50 }); | ||||
|       sequence_.push_back({ LOW, 0 }); | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * loop() is called continuously. Here you can check for events, read sensors, etc. | ||||
|      */ | ||||
|     void loop() { | ||||
|       if (sequence_.size() < 1) return; // Wait until there is a sequence | ||||
|       if (millis() - lastTime_ <= delay_) return; // Wait until delay has elapsed | ||||
|  | ||||
|       auto event = sequence_.front(); | ||||
|       sequence_.pop_front(); | ||||
|  | ||||
|       digitalWrite(USERMOD_BUZZER_PIN, event.first); | ||||
|       delay_ = event.second; | ||||
|  | ||||
|       lastTime_ = millis(); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /* | ||||
|      * 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_BUZZER; | ||||
|     } | ||||
| }; | ||||
							
								
								
									
										515
									
								
								usermods/esp32_multistrip/NpbWrapper.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,515 @@ | ||||
| //this code is a modified version of https://github.com/Makuna/NeoPixelBus/issues/103 | ||||
| #ifndef NpbWrapper_h | ||||
| #define NpbWrapper_h | ||||
|  | ||||
| // make sure we're using esp32 platform | ||||
| #ifndef ARDUINO_ARCH_ESP32 | ||||
|   #error This version of NbpWrapper.h only works with ESP32 hardware. | ||||
| #endif | ||||
|  | ||||
| #ifndef NUM_STRIPS | ||||
|   #error Need to define number of LED strips using build flag -D NUM_STRIPS=4 for 4 LED strips | ||||
| #endif | ||||
|  | ||||
| #ifndef PIXEL_COUNTS | ||||
|   #error Need to define pixel counts using build flag -D PIXEL_COUNTS="25, 25, 25, 25" for 4 LED strips with 25 LEDs each | ||||
| #endif | ||||
|  | ||||
| #ifndef DATA_PINS | ||||
|   #error Need to define data pins using build flag -D DATA_PINS="1, 2, 3, 4" if LED strips are on data pins 1, 2, 3, and 4 | ||||
| #endif | ||||
|  | ||||
| // //PIN CONFIGURATION | ||||
| #ifndef LEDPIN | ||||
|   #define LEDPIN 1  // Legacy pin def required by some other portions of code. This pin is not used do drive LEDs. | ||||
| #endif | ||||
|  | ||||
| #ifndef IRPIN | ||||
|   #define IRPIN -1  //infrared pin (-1 to disable)  MagicHome: 4, H801 Wifi: 0 | ||||
| #endif | ||||
|  | ||||
| #ifndef RLYPIN | ||||
|   #define RLYPIN -1  //pin for relay, will be set HIGH if LEDs are on (-1 to disable). Also usable for standby leds, triggers,... | ||||
| #endif | ||||
|  | ||||
| #ifndef AUXPIN | ||||
|   #define AUXPIN -1  //debug auxiliary output pin (-1 to disable) | ||||
| #endif | ||||
|  | ||||
| #ifndef RLYMDE | ||||
|   #define RLYMDE 1  //mode for relay, 0: LOW if LEDs are on 1: HIGH if LEDs are on | ||||
| #endif | ||||
|  | ||||
| #include <NeoPixelBrightnessBus.h> | ||||
| #include "const.h" | ||||
|  | ||||
| const uint8_t numStrips = NUM_STRIPS;  // max 8 strips allowed on esp32 | ||||
| const uint16_t pixelCounts[numStrips] = {PIXEL_COUNTS}; // number of pixels on each strip | ||||
| const uint8_t dataPins[numStrips] = {DATA_PINS}; // change these pins based on your board | ||||
|  | ||||
| #define PIXELFEATURE3 NeoGrbFeature | ||||
| #define PIXELFEATURE4 NeoGrbwFeature | ||||
|  | ||||
| // ESP32 has 8 RMT interfaces available, each of which can drive a strip of pixels | ||||
| // Convenience #defines for creating NeoPixelBrightnessBus on each RMT interface for both GRB and GRBW LED strips | ||||
| #define NeoPixelBrightnessBusGrbRmt0 NeoPixelBrightnessBus<PIXELFEATURE3, NeoEsp32Rmt0Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbRmt1 NeoPixelBrightnessBus<PIXELFEATURE3, NeoEsp32Rmt1Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbRmt2 NeoPixelBrightnessBus<PIXELFEATURE3, NeoEsp32Rmt2Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbRmt3 NeoPixelBrightnessBus<PIXELFEATURE3, NeoEsp32Rmt3Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbRmt4 NeoPixelBrightnessBus<PIXELFEATURE3, NeoEsp32Rmt4Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbRmt5 NeoPixelBrightnessBus<PIXELFEATURE3, NeoEsp32Rmt5Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbRmt6 NeoPixelBrightnessBus<PIXELFEATURE3, NeoEsp32Rmt6Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbRmt7 NeoPixelBrightnessBus<PIXELFEATURE3, NeoEsp32Rmt7Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbwRmt0 NeoPixelBrightnessBus<PIXELFEATURE4, NeoEsp32Rmt0Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbwRmt1 NeoPixelBrightnessBus<PIXELFEATURE4, NeoEsp32Rmt1Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbwRmt2 NeoPixelBrightnessBus<PIXELFEATURE4, NeoEsp32Rmt2Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbwRmt3 NeoPixelBrightnessBus<PIXELFEATURE4, NeoEsp32Rmt3Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbwRmt4 NeoPixelBrightnessBus<PIXELFEATURE4, NeoEsp32Rmt4Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbwRmt5 NeoPixelBrightnessBus<PIXELFEATURE4, NeoEsp32Rmt5Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbwRmt6 NeoPixelBrightnessBus<PIXELFEATURE4, NeoEsp32Rmt6Ws2812xMethod> | ||||
| #define NeoPixelBrightnessBusGrbwRmt7 NeoPixelBrightnessBus<PIXELFEATURE4, NeoEsp32Rmt7Ws2812xMethod> | ||||
|  | ||||
| enum NeoPixelType | ||||
| { | ||||
|   NeoPixelType_None = 0, | ||||
|   NeoPixelType_Grb  = 1, | ||||
|   NeoPixelType_Grbw = 2, | ||||
|   NeoPixelType_End  = 3 | ||||
| }; | ||||
|  | ||||
| class NeoPixelWrapper | ||||
| { | ||||
| public: | ||||
|   NeoPixelWrapper() : | ||||
|     _type(NeoPixelType_None) | ||||
|   { | ||||
|     // On initialization fill in the pixelStripStartIdx array with the beginning index of each strip | ||||
|     // relative to th entire array. | ||||
|     uint16_t totalPixels = 0; | ||||
|     for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|     { | ||||
|       pixelStripStartIdx[idx] = totalPixels; | ||||
|       totalPixels += pixelCounts[idx]; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   ~NeoPixelWrapper() | ||||
|   { | ||||
|     cleanup(); | ||||
|   } | ||||
|  | ||||
|   void Begin(NeoPixelType type, uint16_t pixelCount) | ||||
|   { | ||||
|  | ||||
|     cleanup(); | ||||
|  | ||||
|     _type = type; | ||||
|  | ||||
|     switch (_type) | ||||
|     { | ||||
|       case NeoPixelType_Grb: | ||||
|       { | ||||
|         for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|         { | ||||
|           switch (idx) | ||||
|           { | ||||
|             case 0: pGrb0 = new NeoPixelBrightnessBusGrbRmt0(pixelCounts[idx], dataPins[idx]); pGrb0->Begin(); break; | ||||
|             case 1: pGrb1 = new NeoPixelBrightnessBusGrbRmt1(pixelCounts[idx], dataPins[idx]); pGrb1->Begin(); break; | ||||
|             case 2: pGrb2 = new NeoPixelBrightnessBusGrbRmt2(pixelCounts[idx], dataPins[idx]); pGrb2->Begin(); break; | ||||
|             case 3: pGrb3 = new NeoPixelBrightnessBusGrbRmt3(pixelCounts[idx], dataPins[idx]); pGrb3->Begin(); break; | ||||
|             case 4: pGrb4 = new NeoPixelBrightnessBusGrbRmt4(pixelCounts[idx], dataPins[idx]); pGrb4->Begin(); break; | ||||
|             case 5: pGrb5 = new NeoPixelBrightnessBusGrbRmt5(pixelCounts[idx], dataPins[idx]); pGrb5->Begin(); break; | ||||
|             case 6: pGrb6 = new NeoPixelBrightnessBusGrbRmt6(pixelCounts[idx], dataPins[idx]); pGrb6->Begin(); break; | ||||
|             case 7: pGrb7 = new NeoPixelBrightnessBusGrbRmt7(pixelCounts[idx], dataPins[idx]); pGrb7->Begin(); break; | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|  | ||||
|       case NeoPixelType_Grbw: | ||||
|       { | ||||
|         for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|         { | ||||
|           switch (idx) | ||||
|           { | ||||
|             case 0: pGrbw0 = new NeoPixelBrightnessBusGrbwRmt0(pixelCounts[idx], dataPins[idx]); pGrbw0->Begin(); break; | ||||
|             case 1: pGrbw1 = new NeoPixelBrightnessBusGrbwRmt1(pixelCounts[idx], dataPins[idx]); pGrbw1->Begin(); break; | ||||
|             case 2: pGrbw2 = new NeoPixelBrightnessBusGrbwRmt2(pixelCounts[idx], dataPins[idx]); pGrbw2->Begin(); break; | ||||
|             case 3: pGrbw3 = new NeoPixelBrightnessBusGrbwRmt3(pixelCounts[idx], dataPins[idx]); pGrbw3->Begin(); break; | ||||
|             case 4: pGrbw4 = new NeoPixelBrightnessBusGrbwRmt4(pixelCounts[idx], dataPins[idx]); pGrbw4->Begin(); break; | ||||
|             case 5: pGrbw5 = new NeoPixelBrightnessBusGrbwRmt5(pixelCounts[idx], dataPins[idx]); pGrbw5->Begin(); break; | ||||
|             case 6: pGrbw6 = new NeoPixelBrightnessBusGrbwRmt6(pixelCounts[idx], dataPins[idx]); pGrbw6->Begin(); break; | ||||
|             case 7: pGrbw7 = new NeoPixelBrightnessBusGrbwRmt7(pixelCounts[idx], dataPins[idx]); pGrbw7->Begin(); break; | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void Show() | ||||
|   { | ||||
|     switch (_type) | ||||
|     { | ||||
|       case NeoPixelType_Grb: | ||||
|       { | ||||
|         for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|         { | ||||
|           switch (idx) | ||||
|           { | ||||
|             case 0: pGrb0->Show(); break; | ||||
|             case 1: pGrb1->Show(); break; | ||||
|             case 2: pGrb2->Show(); break; | ||||
|             case 3: pGrb3->Show(); break; | ||||
|             case 4: pGrb4->Show(); break; | ||||
|             case 5: pGrb5->Show(); break; | ||||
|             case 6: pGrb6->Show(); break; | ||||
|             case 7: pGrb7->Show(); break; | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|       case NeoPixelType_Grbw: | ||||
|       { | ||||
|         for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|         { | ||||
|           switch (idx) | ||||
|           { | ||||
|             case 0: pGrbw0->Show(); break; | ||||
|             case 1: pGrbw1->Show(); break; | ||||
|             case 2: pGrbw2->Show(); break; | ||||
|             case 3: pGrbw3->Show(); break; | ||||
|             case 4: pGrbw4->Show(); break; | ||||
|             case 5: pGrbw5->Show(); break; | ||||
|             case 6: pGrbw6->Show(); break; | ||||
|             case 7: pGrbw7->Show(); break; | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   bool CanShow() | ||||
|   { | ||||
|     bool canShow = true; | ||||
|     switch (_type) | ||||
|     { | ||||
|       case NeoPixelType_Grb: | ||||
|       { | ||||
|         for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|         { | ||||
|           switch (idx) | ||||
|           { | ||||
|             case 0: canShow &= pGrb0->CanShow(); break; | ||||
|             case 1: canShow &= pGrb1->CanShow(); break; | ||||
|             case 2: canShow &= pGrb2->CanShow(); break; | ||||
|             case 3: canShow &= pGrb3->CanShow(); break; | ||||
|             case 4: canShow &= pGrb4->CanShow(); break; | ||||
|             case 5: canShow &= pGrb5->CanShow(); break; | ||||
|             case 6: canShow &= pGrb6->CanShow(); break; | ||||
|             case 7: canShow &= pGrb7->CanShow(); break; | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|       case NeoPixelType_Grbw: | ||||
|       { | ||||
|         for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|         { | ||||
|           switch (idx) | ||||
|           { | ||||
|             case 0: canShow &= pGrbw0->CanShow(); break; | ||||
|             case 1: canShow &= pGrbw1->CanShow(); break; | ||||
|             case 2: canShow &= pGrbw2->CanShow(); break; | ||||
|             case 3: canShow &= pGrbw3->CanShow(); break; | ||||
|             case 4: canShow &= pGrbw4->CanShow(); break; | ||||
|             case 5: canShow &= pGrbw5->CanShow(); break; | ||||
|             case 6: canShow &= pGrbw6->CanShow(); break; | ||||
|             case 7: canShow &= pGrbw7->CanShow(); break; | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|     return canShow; | ||||
|   } | ||||
|  | ||||
|   void SetPixelColorRaw(uint16_t indexPixel, RgbwColor c) | ||||
|   { | ||||
|     // figure out which strip this pixel index is on | ||||
|     uint8_t stripIdx = 0; | ||||
|     for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|     { | ||||
|       if (indexPixel >= pixelStripStartIdx[idx]) | ||||
|       { | ||||
|         stripIdx = idx; | ||||
|       } | ||||
|       else | ||||
|       { | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|     // subtract strip start index so we're addressing just this strip instead of all pixels on all strips | ||||
|     indexPixel -= pixelStripStartIdx[stripIdx]; | ||||
|     switch (_type) | ||||
|     { | ||||
|       case NeoPixelType_Grb: | ||||
|       { | ||||
|         RgbColor rgb = RgbColor(c.R, c.G, c.B); | ||||
|         switch (stripIdx) | ||||
|         { | ||||
|           case 0: pGrb0->SetPixelColor(indexPixel, rgb); break; | ||||
|           case 1: pGrb1->SetPixelColor(indexPixel, rgb); break; | ||||
|           case 2: pGrb2->SetPixelColor(indexPixel, rgb); break; | ||||
|           case 3: pGrb3->SetPixelColor(indexPixel, rgb); break; | ||||
|           case 4: pGrb4->SetPixelColor(indexPixel, rgb); break; | ||||
|           case 5: pGrb5->SetPixelColor(indexPixel, rgb); break; | ||||
|           case 6: pGrb6->SetPixelColor(indexPixel, rgb); break; | ||||
|           case 7: pGrb7->SetPixelColor(indexPixel, rgb); break; | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|       case NeoPixelType_Grbw: | ||||
|       { | ||||
|         switch (stripIdx) | ||||
|         { | ||||
|           case 0: pGrbw0->SetPixelColor(indexPixel, c); break; | ||||
|           case 1: pGrbw1->SetPixelColor(indexPixel, c); break; | ||||
|           case 2: pGrbw2->SetPixelColor(indexPixel, c); break; | ||||
|           case 3: pGrbw3->SetPixelColor(indexPixel, c); break; | ||||
|           case 4: pGrbw4->SetPixelColor(indexPixel, c); break; | ||||
|           case 5: pGrbw5->SetPixelColor(indexPixel, c); break; | ||||
|           case 6: pGrbw6->SetPixelColor(indexPixel, c); break; | ||||
|           case 7: pGrbw7->SetPixelColor(indexPixel, c); break; | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void SetPixelColor(uint16_t indexPixel, RgbwColor c) | ||||
|   { | ||||
|     /* | ||||
|     Set pixel color with necessary color order conversion. | ||||
|     */ | ||||
|  | ||||
|     RgbwColor col; | ||||
|  | ||||
|     uint8_t co = _colorOrder; | ||||
|     #ifdef COLOR_ORDER_OVERRIDE | ||||
|     if (indexPixel >= COO_MIN && indexPixel < COO_MAX) co = COO_ORDER; | ||||
|     #endif | ||||
|  | ||||
|     //reorder channels to selected order | ||||
|     switch (co) | ||||
|     { | ||||
|       case  0: col.G = c.G; col.R = c.R; col.B = c.B; break; //0 = GRB, default | ||||
|       case  1: col.G = c.R; col.R = c.G; col.B = c.B; break; //1 = RGB, common for WS2811 | ||||
|       case  2: col.G = c.B; col.R = c.R; col.B = c.G; break; //2 = BRG | ||||
|       case  3: col.G = c.R; col.R = c.B; col.B = c.G; break; //3 = RBG | ||||
|       case  4: col.G = c.B; col.R = c.G; col.B = c.R; break; //4 = BGR | ||||
|       default: col.G = c.G; col.R = c.B; col.B = c.R; break; //5 = GBR | ||||
|     } | ||||
|     col.W = c.W; | ||||
|  | ||||
|     SetPixelColorRaw(indexPixel, col); | ||||
|   } | ||||
|  | ||||
|   void SetBrightness(byte b) | ||||
|   { | ||||
|     switch (_type) | ||||
|     { | ||||
|       case NeoPixelType_Grb: | ||||
|       { | ||||
|         for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|         { | ||||
|           switch (idx) | ||||
|           { | ||||
|             case 0: pGrb0->SetBrightness(b); break; | ||||
|             case 1: pGrb1->SetBrightness(b); break; | ||||
|             case 2: pGrb2->SetBrightness(b); break; | ||||
|             case 3: pGrb3->SetBrightness(b); break; | ||||
|             case 4: pGrb4->SetBrightness(b); break; | ||||
|             case 5: pGrb5->SetBrightness(b); break; | ||||
|             case 6: pGrb6->SetBrightness(b); break; | ||||
|             case 7: pGrb7->SetBrightness(b); break; | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|       case NeoPixelType_Grbw: | ||||
|       { | ||||
|         for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|         { | ||||
|           switch (idx) | ||||
|           { | ||||
|             case 0: pGrbw0->SetBrightness(b); break; | ||||
|             case 1: pGrbw1->SetBrightness(b); break; | ||||
|             case 2: pGrbw2->SetBrightness(b); break; | ||||
|             case 3: pGrbw3->SetBrightness(b); break; | ||||
|             case 4: pGrbw4->SetBrightness(b); break; | ||||
|             case 5: pGrbw5->SetBrightness(b); break; | ||||
|             case 6: pGrbw6->SetBrightness(b); break; | ||||
|             case 7: pGrbw7->SetBrightness(b); break; | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void SetColorOrder(byte colorOrder) | ||||
|   { | ||||
|     _colorOrder = colorOrder; | ||||
|   } | ||||
|  | ||||
|   uint8_t GetColorOrder() | ||||
|   { | ||||
|     return _colorOrder; | ||||
|   } | ||||
|  | ||||
|   RgbwColor GetPixelColorRaw(uint16_t indexPixel) const | ||||
|   { | ||||
|     // figure out which strip this pixel index is on | ||||
|     uint8_t stripIdx = 0; | ||||
|     for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|     { | ||||
|       if (indexPixel >= pixelStripStartIdx[idx]) | ||||
|       { | ||||
|         stripIdx = idx; | ||||
|       } | ||||
|       else | ||||
|       { | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|     // subtract strip start index so we're addressing just this strip instead of all pixels on all strips | ||||
|     indexPixel -= pixelStripStartIdx[stripIdx]; | ||||
|     switch (_type) | ||||
|     { | ||||
|     case NeoPixelType_Grb: | ||||
|     { | ||||
|       switch (stripIdx) | ||||
|       { | ||||
|         case 0: return pGrb0->GetPixelColor(indexPixel); | ||||
|         case 1: return pGrb1->GetPixelColor(indexPixel); | ||||
|         case 2: return pGrb2->GetPixelColor(indexPixel); | ||||
|         case 3: return pGrb3->GetPixelColor(indexPixel); | ||||
|         case 4: return pGrb4->GetPixelColor(indexPixel); | ||||
|         case 5: return pGrb5->GetPixelColor(indexPixel); | ||||
|         case 6: return pGrb6->GetPixelColor(indexPixel); | ||||
|         case 7: return pGrb7->GetPixelColor(indexPixel); | ||||
|       } | ||||
|       break; | ||||
|     } | ||||
|     case NeoPixelType_Grbw: | ||||
|       switch (stripIdx) | ||||
|       { | ||||
|         case 0: return pGrbw0->GetPixelColor(indexPixel); | ||||
|         case 1: return pGrbw1->GetPixelColor(indexPixel); | ||||
|         case 2: return pGrbw2->GetPixelColor(indexPixel); | ||||
|         case 3: return pGrbw3->GetPixelColor(indexPixel); | ||||
|         case 4: return pGrbw4->GetPixelColor(indexPixel); | ||||
|         case 5: return pGrbw5->GetPixelColor(indexPixel); | ||||
|         case 6: return pGrbw6->GetPixelColor(indexPixel); | ||||
|         case 7: return pGrbw7->GetPixelColor(indexPixel); | ||||
|       } | ||||
|       break; | ||||
|     } | ||||
|     return 0; | ||||
|   } | ||||
|  | ||||
|   // NOTE: Due to feature differences, some support RGBW but the method name | ||||
|   // here needs to be unique, thus GetPixeColorRgbw | ||||
|   uint32_t GetPixelColorRgbw(uint16_t indexPixel) const | ||||
|   { | ||||
|     RgbwColor col = GetPixelColorRaw(indexPixel); | ||||
|     uint8_t co = _colorOrder; | ||||
|     #ifdef COLOR_ORDER_OVERRIDE | ||||
|     if (indexPixel >= COO_MIN && indexPixel < COO_MAX) co = COO_ORDER; | ||||
|     #endif | ||||
|  | ||||
|     switch (co) | ||||
|     { | ||||
|       //                    W               G              R               B | ||||
|       case  0: return ((col.W << 24) | (col.G << 8) | (col.R << 16) | (col.B)); //0 = GRB, default | ||||
|       case  1: return ((col.W << 24) | (col.R << 8) | (col.G << 16) | (col.B)); //1 = RGB, common for WS2811 | ||||
|       case  2: return ((col.W << 24) | (col.B << 8) | (col.R << 16) | (col.G)); //2 = BRG | ||||
|       case  3: return ((col.W << 24) | (col.B << 8) | (col.G << 16) | (col.R)); //3 = RBG | ||||
|       case  4: return ((col.W << 24) | (col.R << 8) | (col.B << 16) | (col.G)); //4 = BGR | ||||
|       case  5: return ((col.W << 24) | (col.G << 8) | (col.B << 16) | (col.R)); //5 = GBR | ||||
|     } | ||||
|  | ||||
|     return 0; | ||||
|  | ||||
|   } | ||||
|    | ||||
|  | ||||
| private: | ||||
|   NeoPixelType _type; | ||||
|   byte _colorOrder = 0; | ||||
|  | ||||
|   uint16_t pixelStripStartIdx[numStrips]; | ||||
|  | ||||
|   // pointers for every possible type for up to 8 strips | ||||
|   NeoPixelBrightnessBusGrbRmt0 *pGrb0; | ||||
|   NeoPixelBrightnessBusGrbRmt1 *pGrb1; | ||||
|   NeoPixelBrightnessBusGrbRmt2 *pGrb2; | ||||
|   NeoPixelBrightnessBusGrbRmt3 *pGrb3; | ||||
|   NeoPixelBrightnessBusGrbRmt4 *pGrb4; | ||||
|   NeoPixelBrightnessBusGrbRmt5 *pGrb5; | ||||
|   NeoPixelBrightnessBusGrbRmt6 *pGrb6; | ||||
|   NeoPixelBrightnessBusGrbRmt7 *pGrb7; | ||||
|   NeoPixelBrightnessBusGrbwRmt0 *pGrbw0; | ||||
|   NeoPixelBrightnessBusGrbwRmt1 *pGrbw1; | ||||
|   NeoPixelBrightnessBusGrbwRmt2 *pGrbw2; | ||||
|   NeoPixelBrightnessBusGrbwRmt3 *pGrbw3; | ||||
|   NeoPixelBrightnessBusGrbwRmt4 *pGrbw4; | ||||
|   NeoPixelBrightnessBusGrbwRmt5 *pGrbw5; | ||||
|   NeoPixelBrightnessBusGrbwRmt6 *pGrbw6; | ||||
|   NeoPixelBrightnessBusGrbwRmt7 *pGrbw7; | ||||
|  | ||||
|   void cleanup() | ||||
|   { | ||||
|     switch (_type) | ||||
|     { | ||||
|       case NeoPixelType_Grb: | ||||
|       { | ||||
|         for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|         { | ||||
|           switch (idx) | ||||
|           { | ||||
|             case 0: delete pGrb0; pGrb0 = NULL; break; | ||||
|             case 1: delete pGrb1; pGrb1 = NULL; break; | ||||
|             case 2: delete pGrb2; pGrb2 = NULL; break; | ||||
|             case 3: delete pGrb3; pGrb3 = NULL; break; | ||||
|             case 4: delete pGrb4; pGrb4 = NULL; break; | ||||
|             case 5: delete pGrb5; pGrb5 = NULL; break; | ||||
|             case 6: delete pGrb6; pGrb6 = NULL; break; | ||||
|             case 7: delete pGrb7; pGrb7 = NULL; break; | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|       } | ||||
|       case NeoPixelType_Grbw: | ||||
|       { | ||||
|         for (uint8_t idx = 0; idx < numStrips; idx++) | ||||
|         { | ||||
|           switch (idx) | ||||
|           { | ||||
|             case 0: delete pGrbw0; pGrbw0 = NULL; break; | ||||
|             case 1: delete pGrbw1; pGrbw1 = NULL; break; | ||||
|             case 2: delete pGrbw2; pGrbw2 = NULL; break; | ||||
|             case 3: delete pGrbw3; pGrbw3 = NULL; break; | ||||
|             case 4: delete pGrbw4; pGrbw4 = NULL; break; | ||||
|             case 5: delete pGrbw5; pGrbw5 = NULL; break; | ||||
|             case 6: delete pGrbw6; pGrbw6 = NULL; break; | ||||
|             case 7: delete pGrbw7; pGrbw7 = NULL; break; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| }; | ||||
| #endif | ||||
							
								
								
									
										22
									
								
								usermods/esp32_multistrip/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,22 @@ | ||||
| # esp32_multistrip | ||||
|  | ||||
| This usermod enables up to 8 data pins to be used from an esp32 module to drive separate LED strands. This only works with one-wire LEDs like the WS2812. | ||||
|  | ||||
| The esp32 RMT hardware is used for data output. See here for hardware driver implementation details: https://github.com/Makuna/NeoPixelBus/wiki/ESP32-NeoMethods#neoesp32rmt-methods | ||||
|  | ||||
| Pass the following variables to the compiler as build flags: | ||||
|  | ||||
|  - `ESP32_MULTISTRIP` | ||||
|    - Define this to use usermod NpbWrapper.h instead of default one in WLED. | ||||
|  - `NUM_STRIPS` | ||||
|    - Number of strips in use | ||||
|  - `PIXEL_COUNTS` | ||||
|    - List of pixel counts in each strip | ||||
|  - `DATA_PINS` | ||||
|    - List of data pins each strip is attached to. There may be board-specific restrictions on which pins can be used for RTM. | ||||
|  | ||||
| From the perspective of WLED software, the LEDs are addressed as one long strand. The modified NbpWrapper.h file addresses the appropriate strand from the overall LED index based on the number of LEDs defined in each strand. | ||||
|  | ||||
| See `platformio_override.ini` for example configuration. | ||||
|  | ||||
| Tested on low cost ESP-WROOM-32 dev boards from Amazon, such as those sold by KeeYees. | ||||
							
								
								
									
										16
									
								
								usermods/esp32_multistrip/platformio_override.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,16 @@ | ||||
| ; Example platformio_override.ini that shows how to configure your environment to use the multistrip usermod. | ||||
| ; Copy this file to the base wled directory that contains platformio.ini. | ||||
| ; Multistrip requires ESP32 because it has many more pins that can be used as LED outputs. | ||||
| ; Need to define NUM_STRIPS, PIXEL_COUNTS, and DATA_PINS as shown below. | ||||
|  | ||||
| [platformio] | ||||
| default_envs = esp32_multistrip | ||||
|  | ||||
| [env:esp32_multistrip] | ||||
| extends=env:esp32dev | ||||
| build_flags = ${env:esp32dev.build_flags} | ||||
|   -D ESP32_MULTISTRIP  ; define this variable to use ESP32_MULTISTRIP usermod | ||||
|   -D NUM_STRIPS=4  ; number of pixel strips in use | ||||
|   -D PIXEL_COUNTS="50, 50, 50, 50"  ; number of pixels in each strip | ||||
|   -D DATA_PINS="25, 26, 32, 33"  ; esp32 pins used for each pixel strip. available pins depends on esp32 module. | ||||
|    | ||||
							
								
								
									
										94
									
								
								usermods/mpu6050_imu/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,94 @@ | ||||
| # MPU-6050 Six-Axis (Gyro + Accelerometer) Driver | ||||
|  | ||||
| This usermod-v2 modification allows the connection of a MPU-6050 IMU sensor to | ||||
| allow for effects that are controlled by the orientation or motion of the WLED Device. | ||||
|  | ||||
| The MPU6050 has a built in "Digital Motion Processor" which does a lot of the heavy | ||||
| lifting in integrating the gyro and accel measurements to get potentially more | ||||
| useful gravity vector and orientation output. | ||||
|  | ||||
| It is pretty straightforward to comment out some of the variables being read off the device if they're not needed to save CPU/Mem/Bandwidth. | ||||
|  | ||||
| _Story:_ | ||||
|  | ||||
| As a memento to a long trip I was on, I built an icosahedron globe. I put lights inside to indicate cities I travelled to. | ||||
|  | ||||
| I wanted to integrate an IMU to allow either on-board, or off-board effects that would | ||||
| react to the globes orientation. See the blog post on building it <https://www.robopenguins.com/icosahedron-travel-globe/> or a video demo <https://youtu.be/zYjybxHBsHM> . | ||||
|  | ||||
| ## Adding Dependencies | ||||
|  | ||||
| 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) | ||||
|  | ||||
| For example: | ||||
|  | ||||
| ``` | ||||
| lib_compat_mode = soft | ||||
| lib_deps = | ||||
|     FastLED@3.3.2 | ||||
|     NeoPixelBus@2.5.7 | ||||
|     ESPAsyncTCP@1.2.0 | ||||
|     ESPAsyncUDP@697c75a025 | ||||
|     AsyncTCP@1.0.3 | ||||
|     Esp Async WebServer@1.2.0 | ||||
|     IRremoteESP8266@2.7.3 | ||||
|     I2Cdevlib-MPU6050@fbde122cc5 | ||||
| ``` | ||||
|  | ||||
| ## Wiring | ||||
|  | ||||
| The connections needed to the MPU6050 are as follows: | ||||
| ``` | ||||
|   VCC     VU (5V USB)   Not available on all boards so use 3.3V if needed. | ||||
|   GND     G             Ground | ||||
|   SCL     D1 (GPIO05)   I2C clock | ||||
|   SDA     D2 (GPIO04)   I2C data | ||||
|   XDA     not connected | ||||
|   XCL     not connected | ||||
|   AD0     not connected | ||||
|   INT     D8 (GPIO15)   Interrupt pin | ||||
| ``` | ||||
|  | ||||
| You could probably modify the code not to need an interrupt, but I used the | ||||
| setup directly from the example. | ||||
|  | ||||
| ## JSON API | ||||
|  | ||||
| This code adds: | ||||
| ```json | ||||
| "u":{ | ||||
|   "IMU":{ | ||||
|     "Quat":        [w, x, y, z], | ||||
|     "Euler":       [psi, theta, phi], | ||||
|     "Gyro":        [x, y, z], | ||||
|     "Accel":       [x, y, z], | ||||
|     "RealAccel":   [x, y, z], | ||||
|     "WorldAccel":  [x, y, z], | ||||
|     "Gravity":     [x, y, z], | ||||
|     "Orientation": [yaw, pitch, roll] | ||||
|   } | ||||
| } | ||||
| ``` | ||||
| to the info object | ||||
|  | ||||
| ## Usermod installation | ||||
|  | ||||
| 1. Copy the file `usermod_mpu6050_imu.h` to the `wled00` directory. | ||||
| 2. Register the usermod by adding `#include "usermod_mpu6050_imu.h.h"` in the top and `registerUsermod(new MPU6050Driver());` in the bottom of `usermods_list.cpp`. | ||||
|  | ||||
| Example **usermods_list.cpp**: | ||||
|  | ||||
| ```cpp | ||||
| #include "wled.h" | ||||
|  | ||||
| #include "usermod_mpu6050_imu.h" | ||||
|  | ||||
| void registerUsermods() | ||||
| { | ||||
|   usermods.add(new MPU6050Driver()); | ||||
| } | ||||
| ``` | ||||
							
								
								
									
										274
									
								
								usermods/mpu6050_imu/usermod_mpu6050_imu.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,274 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
|  | ||||
| /* This driver reads quaternion data from the MPU6060 and adds it to the JSON | ||||
|    This example is adapted from: | ||||
|    https://github.com/jrowberg/i2cdevlib/tree/master/Arduino/MPU6050/examples/MPU6050_DMP6_ESPWiFi | ||||
|  | ||||
|    Tested with a d1 mini esp-12f | ||||
|  | ||||
|   GY-521  NodeMCU | ||||
|   MPU6050 devkit 1.0 | ||||
|   board   Lolin         Description | ||||
|   ======= ==========    ==================================================== | ||||
|   VCC     VU (5V USB)   Not available on all boards so use 3.3V if needed. | ||||
|   GND     G             Ground | ||||
|   SCL     D1 (GPIO05)   I2C clock | ||||
|   SDA     D2 (GPIO04)   I2C data | ||||
|   XDA     not connected | ||||
|   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  | ||||
|      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) | ||||
|   5. Wire up the MPU6050 as detailed above. | ||||
| */ | ||||
|  | ||||
| #include "I2Cdev.h" | ||||
|  | ||||
| #include "MPU6050_6Axis_MotionApps20.h" | ||||
| //#include "MPU6050.h" // not necessary if using MotionApps include file | ||||
|  | ||||
| // Arduino Wire library is required if I2Cdev I2CDEV_ARDUINO_WIRE implementation | ||||
| // is used in I2Cdev.h | ||||
| #if I2CDEV_IMPLEMENTATION == I2CDEV_ARDUINO_WIRE | ||||
|     #include "Wire.h" | ||||
| #endif | ||||
|  | ||||
| // ================================================================ | ||||
| // ===               INTERRUPT DETECTION ROUTINE                === | ||||
| // ================================================================ | ||||
|  | ||||
| volatile bool mpuInterrupt = false;     // indicates whether MPU interrupt pin has gone high | ||||
| void IRAM_ATTR dmpDataReady() { | ||||
|     mpuInterrupt = true; | ||||
| } | ||||
|  | ||||
|  | ||||
| class MPU6050Driver : public Usermod { | ||||
|   private: | ||||
|     MPU6050 mpu; | ||||
|      | ||||
|     // MPU control/status vars | ||||
|     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 | ||||
|     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 | ||||
|     VectorInt16 aa;         // [x, y, z]            accel sensor measurements | ||||
|     VectorInt16 gy;         // [x, y, z]            gyro sensor measurements | ||||
|     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 | ||||
|  | ||||
|     static const int INTERRUPT_PIN = 15; // use pin 15 on ESP8266 | ||||
|  | ||||
|   public: | ||||
|     //Functions called by WLED | ||||
|  | ||||
|     /* | ||||
|      * setup() is called once at boot. WiFi is not yet connected at this point. | ||||
|      */ | ||||
|     void setup() { | ||||
|       // join I2C bus (I2Cdev library doesn't do this automatically) | ||||
|       #if I2CDEV_IMPLEMENTATION == I2CDEV_ARDUINO_WIRE | ||||
|         Wire.begin(); | ||||
|         Wire.setClock(400000); // 400kHz I2C clock. Comment this line if having compilation difficulties | ||||
|       #elif I2CDEV_IMPLEMENTATION == I2CDEV_BUILTIN_FASTWIRE | ||||
|         Fastwire::setup(400, true); | ||||
|       #endif | ||||
|  | ||||
|       // initialize device | ||||
|       Serial.println(F("Initializing I2C devices...")); | ||||
|       mpu.initialize(); | ||||
|       pinMode(INTERRUPT_PIN, INPUT); | ||||
|  | ||||
|       // verify connection | ||||
|       Serial.println(F("Testing device connections...")); | ||||
|       Serial.println(mpu.testConnection() ? F("MPU6050 connection successful") : F("MPU6050 connection failed")); | ||||
|  | ||||
|       // load and configure the DMP | ||||
|       Serial.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 | ||||
|  | ||||
|       // make sure it worked (returns 0 if so) | ||||
|       if (devStatus == 0) { | ||||
|         // turn on the DMP, now that it's ready | ||||
|         Serial.println(F("Enabling DMP...")); | ||||
|         mpu.setDMPEnabled(true); | ||||
|  | ||||
|         // enable Arduino interrupt detection | ||||
|         Serial.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 | ||||
|         Serial.println(F("DMP ready! Waiting for first interrupt...")); | ||||
|         dmpReady = true; | ||||
|  | ||||
|         // get expected DMP packet size for later comparison | ||||
|         packetSize = mpu.dmpGetFIFOPacketSize(); | ||||
|       } else { | ||||
|         // ERROR! | ||||
|         // 1 = initial memory load failed | ||||
|         // 2 = DMP configuration updates failed | ||||
|         // (if it's going to break, usually the code will be 1) | ||||
|         Serial.print(F("DMP Initialization failed (code ")); | ||||
|         Serial.print(devStatus); | ||||
|         Serial.println(F(")")); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * 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. | ||||
|      */ | ||||
|     void loop() { | ||||
|       // if programming failed, don't try to do anything | ||||
|       if (!dmpReady) return; | ||||
|  | ||||
|       // wait for MPU interrupt or extra packet(s) available | ||||
|       if (!mpuInterrupt && fifoCount < packetSize) return; | ||||
|  | ||||
|       // reset interrupt flag and get INT_STATUS byte | ||||
|       mpuInterrupt = false; | ||||
|       mpuIntStatus = mpu.getIntStatus(); | ||||
|  | ||||
|       // get 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(); | ||||
|         Serial.println(F("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(); | ||||
|  | ||||
|         // read a packet from FIFO | ||||
|         mpu.getFIFOBytes(fifoBuffer, packetSize); | ||||
|  | ||||
|         // track FIFO count here in case there is > 1 packet available | ||||
|         // (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); | ||||
|         mpu.dmpGetEuler(euler, &qat); | ||||
|         mpu.dmpGetGravity(&gravity, &qat); | ||||
|         mpu.dmpGetGyro(&gy, fifoBuffer); | ||||
|         mpu.dmpGetAccel(&aa, fifoBuffer); | ||||
|         mpu.dmpGetLinearAccel(&aaReal, &aa, &gravity); | ||||
|         mpu.dmpGetLinearAccelInWorld(&aaWorld, &aaReal, &qat); | ||||
|         mpu.dmpGetYawPitchRoll(ypr, &qat, &gravity); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|  | ||||
|  | ||||
|     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 imu_meas = user.createNestedObject("IMU"); | ||||
|       JsonArray quat_json = imu_meas.createNestedArray("Quat"); | ||||
|       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"); | ||||
|       euler_json.add(euler[0]); | ||||
|       euler_json.add(euler[1]); | ||||
|       euler_json.add(euler[2]); | ||||
|       JsonArray accel_json = imu_meas.createNestedArray("Accel"); | ||||
|       accel_json.add(aa.x); | ||||
|       accel_json.add(aa.y); | ||||
|       accel_json.add(aa.z); | ||||
|       JsonArray gyro_json = imu_meas.createNestedArray("Gyro"); | ||||
|       gyro_json.add(gy.x); | ||||
|       gyro_json.add(gy.y); | ||||
|       gyro_json.add(gy.z); | ||||
|       JsonArray world_json = imu_meas.createNestedArray("WorldAccel"); | ||||
|       world_json.add(aaWorld.x); | ||||
|       world_json.add(aaWorld.y); | ||||
|       world_json.add(aaWorld.z); | ||||
|       JsonArray real_json = imu_meas.createNestedArray("RealAccel"); | ||||
|       real_json.add(aaReal.x); | ||||
|       real_json.add(aaReal.y); | ||||
|       real_json.add(aaReal.z); | ||||
|       JsonArray grav_json = imu_meas.createNestedArray("Gravity"); | ||||
|       grav_json.add(gravity.x); | ||||
|       grav_json.add(gravity.y); | ||||
|       grav_json.add(gravity.z); | ||||
|       JsonArray orient_json = imu_meas.createNestedArray("Orientation"); | ||||
|       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) Serial.println(F("Don't burn down your garage!")); | ||||
|     } | ||||
|      | ||||
|     | ||||
|     /* | ||||
|      * getId() allows you to optionally give your V2 usermod an unique ID (please define it in const.h!). | ||||
|      */ | ||||
|     uint16_t getId() | ||||
|     { | ||||
|       return USERMOD_ID_IMU; | ||||
|     } | ||||
|  | ||||
| }; | ||||
							
								
								
									
										50
									
								
								usermods/mqtt_switch_v2/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,50 @@ | ||||
| # MQTT controllable switches | ||||
| This usermod allows controlling switches (e.g. relays) via MQTT. | ||||
|  | ||||
| ## Usermod installation | ||||
|  | ||||
| 1. Copy the file `usermod_mqtt_switch.h` to the `wled00` directory. | ||||
| 2. Register the usermod by adding `#include "usermod_mqtt_switch.h"` in the top and `registerUsermod(new UsermodMqttSwitch());` in the bottom of `usermods_list.cpp`. | ||||
|  | ||||
|  | ||||
| Example `usermods_list.cpp`: | ||||
|  | ||||
| ``` | ||||
| #include "wled.h" | ||||
| #include "usermod_mqtt_switch.h" | ||||
|  | ||||
| void registerUsermods() | ||||
| { | ||||
|   usermods.add(new UsermodMqttSwitch()); | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## Define pins | ||||
| Add a define for MQTTSWITCHPINS to platformio_override.ini. | ||||
| The following example defines 3 switches connected to the GPIO pins 13, 5 and 2: | ||||
|  | ||||
| ``` | ||||
| [env:livingroom] | ||||
| board = esp12e | ||||
| platform = ${common.platform_wled_default} | ||||
| board_build.ldscript = ${common.ldscript_4m1m} | ||||
| build_flags = ${common.build_flags_esp8266}  | ||||
|    -D LEDPIN=3 | ||||
|    -D BTNPIN=4 | ||||
|    -D RLYPIN=12 | ||||
|    -D RLYMDE=1 | ||||
|    -D STATUSPIN=15 | ||||
|    -D MQTTSWITCHPINS="13, 5, 2" | ||||
| ``` | ||||
|  | ||||
| Pins can be inverted by setting `MQTTSWITCHINVERT`. For example `-D MQTTSWITCHINVERT="false, false, true"` would invert the switch on pin 2 in the previous example. | ||||
|  | ||||
| The default state after booting before any MQTT message can be set by `MQTTSWITCHDEFAULTS`. For example `-D MQTTSWITCHDEFAULTS="ON, OFF, OFF"` would power on the switch on pin 13 and power off switches on pins 5 and 2. | ||||
|     | ||||
| ## MQTT topics | ||||
| This usermod listens on `[mqttDeviceTopic]/switch/0/set` (where 0 is replaced with the index of the switch) for commands. Anything starting with `ON` turns on the switch, everything else turns it off. | ||||
| Feedback about the current state is provided at `[mqttDeviceTopic]/switch/0/state`. | ||||
|  | ||||
| ### Home Assistant auto-discovery | ||||
| Auto-discovery information is automatically published and you shoudn't have to do anything to register the switches in Home Assistant. | ||||
|   | ||||
							
								
								
									
										157
									
								
								usermods/mqtt_switch_v2/usermod_mqtt_switch.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,157 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "wled.h" | ||||
| #ifndef WLED_ENABLE_MQTT | ||||
| #error "This user mod requires MQTT to be enabled." | ||||
| #endif | ||||
|  | ||||
| #ifndef MQTTSWITCHPINS | ||||
| #error "Please define MQTTSWITCHPINS in platformio_override.ini. e.g. -D MQTTSWITCHPINS="12, 0, 2" " | ||||
| // The following define helps Eclipse's C++ parser but is never used in production due to the #error statement on the line before | ||||
| #define MQTTSWITCHPINS 12, 0, 2 | ||||
| #endif | ||||
|  | ||||
| // Default behavior: All outputs active high | ||||
| #ifndef MQTTSWITCHINVERT | ||||
| #define MQTTSWITCHINVERT | ||||
| #endif | ||||
|  | ||||
| // Default behavior: All outputs off | ||||
| #ifndef MQTTSWITCHDEFAULTS | ||||
| #define MQTTSWITCHDEFAULTS | ||||
| #endif | ||||
|  | ||||
| static const uint8_t switchPins[] = { MQTTSWITCHPINS }; | ||||
| //This is a hack to get the number of pins defined by the user | ||||
| #define NUM_SWITCH_PINS (sizeof(switchPins)) | ||||
| static const bool switchInvert[NUM_SWITCH_PINS] = { MQTTSWITCHINVERT}; | ||||
| //Make settings in config file more readable | ||||
| #define ON 1 | ||||
| #define OFF 0 | ||||
| static const bool switchDefaults[NUM_SWITCH_PINS] = { MQTTSWITCHDEFAULTS}; | ||||
| #undef ON | ||||
| #undef OFF | ||||
|  | ||||
| class UsermodMqttSwitch: public Usermod | ||||
| { | ||||
| private: | ||||
|     bool mqttInitialized; | ||||
|     bool switchState[NUM_SWITCH_PINS]; | ||||
|  | ||||
| public: | ||||
|     UsermodMqttSwitch() : | ||||
|             mqttInitialized(false) | ||||
|     { | ||||
|     } | ||||
|  | ||||
|     void setup() | ||||
|     { | ||||
|         for (int pinNr = 0; pinNr < NUM_SWITCH_PINS; pinNr++) { | ||||
|             setState(pinNr, switchDefaults[pinNr]); | ||||
|             pinMode(switchPins[pinNr], OUTPUT); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void loop() | ||||
|     { | ||||
|         if (!mqttInitialized) { | ||||
|             mqttInit(); | ||||
|             return; // Try again in next loop iteration | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void mqttInit() | ||||
|     { | ||||
|         if (!mqtt) | ||||
|             return; | ||||
|         mqtt->onMessage( | ||||
|                 std::bind(&UsermodMqttSwitch::onMqttMessage, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3, std::placeholders::_4, | ||||
|                           std::placeholders::_5, std::placeholders::_6)); | ||||
|         mqtt->onConnect(std::bind(&UsermodMqttSwitch::onMqttConnect, this, std::placeholders::_1)); | ||||
|         mqttInitialized = true; | ||||
|     } | ||||
|  | ||||
|     void onMqttConnect(bool sessionPresent); | ||||
|  | ||||
|     void onMqttMessage(char *topic, char *payload, AsyncMqttClientMessageProperties properties, size_t len, size_t index, size_t total); | ||||
|     void updateState(uint8_t pinNr); | ||||
|  | ||||
|     void setState(uint8_t pinNr, bool active) | ||||
|     { | ||||
|         if (pinNr > NUM_SWITCH_PINS) | ||||
|             return; | ||||
|         switchState[pinNr] = active; | ||||
|         digitalWrite((char) switchPins[pinNr], (char) (switchInvert[pinNr] ? !active : active)); | ||||
|         updateState(pinNr); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| inline void UsermodMqttSwitch::onMqttConnect(bool sessionPresent) | ||||
| { | ||||
|     if (mqttDeviceTopic[0] == 0) | ||||
|         return; | ||||
|  | ||||
|     for (int pinNr = 0; pinNr < NUM_SWITCH_PINS; pinNr++) { | ||||
|         char buf[128]; | ||||
|         StaticJsonDocument<1024> json; | ||||
|         sprintf(buf, "%s Switch %d", serverDescription, pinNr + 1); | ||||
|         json[F("name")] = buf; | ||||
|  | ||||
|         sprintf(buf, "%s/switch/%d", mqttDeviceTopic, pinNr); | ||||
|         json["~"] = buf; | ||||
|         strcat(buf, "/set"); | ||||
|         mqtt->subscribe(buf, 0); | ||||
|  | ||||
|         json[F("stat_t")] = "~/state"; | ||||
|         json[F("cmd_t")] = "~/set"; | ||||
|         json[F("pl_off")] = F("OFF"); | ||||
|         json[F("pl_on")] = F("ON"); | ||||
|  | ||||
|         char uid[16]; | ||||
|         sprintf(uid, "%s_sw%d", escapedMac.c_str(), pinNr); | ||||
|         json[F("unique_id")] = uid; | ||||
|  | ||||
|         strcpy(buf, mqttDeviceTopic); | ||||
|         strcat(buf, "/status"); | ||||
|         json[F("avty_t")] = buf; | ||||
|         json[F("pl_avail")] = F("online"); | ||||
|         json[F("pl_not_avail")] = F("offline"); | ||||
|         //TODO: dev | ||||
|         sprintf(buf, "homeassistant/switch/%s/config", uid); | ||||
|         char json_str[1024]; | ||||
|         size_t payload_size = serializeJson(json, json_str); | ||||
|         mqtt->publish(buf, 0, true, json_str, payload_size); | ||||
|         updateState(pinNr); | ||||
|     } | ||||
| } | ||||
|  | ||||
| inline void UsermodMqttSwitch::onMqttMessage(char *topic, char *payload, AsyncMqttClientMessageProperties properties, size_t len, size_t index, size_t total) | ||||
| { | ||||
|     //Note: Payload is not necessarily null terminated. Check "len" instead. | ||||
|     for (int pinNr = 0; pinNr < NUM_SWITCH_PINS; pinNr++) { | ||||
|         char buf[64]; | ||||
|         sprintf(buf, "%s/switch/%d/set", mqttDeviceTopic, pinNr); | ||||
|         if (strcmp(topic, buf) == 0) { | ||||
|             //Any string starting with "ON" is interpreted as ON, everything else as OFF | ||||
|             setState(pinNr, len >= 2 && payload[0] == 'O' && payload[1] == 'N'); | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| inline void UsermodMqttSwitch::updateState(uint8_t pinNr) | ||||
| { | ||||
|     if (!mqttInitialized) | ||||
|         return; | ||||
|  | ||||
|     if (pinNr > NUM_SWITCH_PINS) | ||||
|         return; | ||||
|  | ||||
|     char buf[64]; | ||||
|     sprintf(buf, "%s/switch/%d/state", mqttDeviceTopic, pinNr); | ||||
|     if (switchState[pinNr]) { | ||||
|         mqtt->publish(buf, 0, false, "ON"); | ||||
|     } else { | ||||
|         mqtt->publish(buf, 0, false, "OFF"); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										11
									
								
								usermods/photoresistor_sensor_mqtt_v1/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | ||||
| # Photoresister sensor with MQTT | ||||
|  | ||||
| This simple usermod allows attaching a photoresistor sensor like the KY-018 and publish the readings in percentage over MQTT. The frequency of MQTT messages can be modified, and there is a threshold value that can be set so that significant changes in the readings can be published immediately instead of waiting for the next update. This was found to be a good compromise between spamming MQTT messages and delayed updates. | ||||
|  | ||||
| I also found it useful to limit the frequency of analog pin reads because otherwise the board hangs. | ||||
|  | ||||
| This usermod has only been tested with the KY-018 sensor though should work for any other analog pin sensor. Note that this does not control the LED strip directly, it only publishes MQTT readings for use with other integrations like Home Assistant. | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| Copy and replace the file `usermod.cpp` in wled00 directory. | ||||
							
								
								
									
										70
									
								
								usermods/photoresistor_sensor_mqtt_v1/usermod.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,70 @@ | ||||
| #include "wled.h" | ||||
| /* | ||||
|  * This v1 usermod file allows you to add own functionality to WLED more easily | ||||
|  * See: https://github.com/Aircoookie/WLED/wiki/Add-own-functionality | ||||
|  * EEPROM bytes 2750+ are reserved for your custom use case. (if you extend #define EEPSIZE in const.h) | ||||
|  * If you just need 8 bytes, use 2551-2559 (you do not need to increase EEPSIZE) | ||||
|  *  | ||||
|  * Consider the v2 usermod API if you need a more advanced feature set! | ||||
|  */ | ||||
|  | ||||
| //Use userVar0 and userVar1 (API calls &U0=,&U1=, uint16_t) | ||||
|  | ||||
| const int LIGHT_PIN = A0; // define analog pin | ||||
| const long UPDATE_MS = 30000; // Upper threshold between mqtt messages | ||||
| const char MQTT_TOPIC[] = "/light"; // MQTT topic for sensor values | ||||
| const int CHANGE_THRESHOLD = 5; // Change threshold in percentage to send before UPDATE_MS | ||||
|  | ||||
| // variables | ||||
| long lastTime = 0; | ||||
| long timeDiff = 0; | ||||
| long readTime = 0; | ||||
| int lightValue = 0;  | ||||
| float lightPercentage = 0; | ||||
| float lastPercentage = 0; | ||||
|  | ||||
| //gets called once at boot. Do all initialization that doesn't depend on network here | ||||
| void userSetup() | ||||
| { | ||||
|   pinMode(LIGHT_PIN, INPUT); | ||||
| } | ||||
|  | ||||
| //gets called every time WiFi is (re-)connected. Initialize own network interfaces here | ||||
| void userConnected() | ||||
| { | ||||
|  | ||||
| } | ||||
|  | ||||
| void publishMqtt(float state) | ||||
| { | ||||
|   //Check if MQTT Connected, otherwise it will crash the 8266 | ||||
|   if (mqtt != nullptr){ | ||||
|     char subuf[38]; | ||||
|     strcpy(subuf, mqttDeviceTopic); | ||||
|     strcat(subuf, MQTT_TOPIC); | ||||
|     mqtt->publish(subuf, 0, true, String(state).c_str()); | ||||
|   } | ||||
| } | ||||
|  | ||||
| //loop. You can use "if (WLED_CONNECTED)" to check for successful connection | ||||
| void userLoop() | ||||
| { | ||||
|    // Read only every 500ms, otherwise it causes the board to hang | ||||
|   if (millis() - readTime > 500) | ||||
|   { | ||||
|     readTime = millis(); | ||||
|     timeDiff = millis() - lastTime; | ||||
|      | ||||
|     // Convert value to percentage | ||||
|     lightValue = analogRead(LIGHT_PIN); | ||||
|     lightPercentage = ((float)lightValue * -1 + 1024)/(float)1024 *100; | ||||
|      | ||||
|     // Send MQTT message on significant change or after UPDATE_MS | ||||
|     if (abs(lightPercentage - lastPercentage) > CHANGE_THRESHOLD || timeDiff > UPDATE_MS)  | ||||
|     { | ||||
|       publishMqtt(lightPercentage); | ||||
|       lastTime = millis(); | ||||
|       lastPercentage = lightPercentage; | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										23
									
								
								usermods/project_cars_shiftlight/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,23 @@ | ||||
| ### Shift Light for Project Cars | ||||
|  | ||||
| Turn your WLED lights into a rev light and shift indicator for Project Cars. | ||||
|  | ||||
| It is pretty straight forward to use. | ||||
|  | ||||
| 1. Make sure, your WLED device and your PC/console are on the same network and can talk to each other | ||||
|  | ||||
| 2. Go to the gameplay settings menu in PCARS and enable UDP. There are 9 numbers you can choose from. This is the refresh rate. The lower the number, the better. But you might run into problems at faster rates. | ||||
|  | ||||
| | Number | Updates/Second | | ||||
| | ------ | -------------- | | ||||
| | 1      | 60             | | ||||
| | 2      | 50             | | ||||
| | 3      | 40             | | ||||
| | 4      | 30             | | ||||
| | 5      | 20             | | ||||
| | 6      | 15             | | ||||
| | 7      | 10             | | ||||
| | 8      | 05             | | ||||
| | 9      | 1              | | ||||
|  | ||||
| 3. once you enter a race, WLED should automatically shift to PCARS mode. Done. | ||||
							
								
								
									
										95
									
								
								usermods/project_cars_shiftlight/wled06_usermod.ino
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,95 @@ | ||||
| /* | ||||
|  *          Car rev display and shift indicator for Project Cars | ||||
|  *           | ||||
|  * This works via the UDP telemetry function. You'll need to enable it in the settings of the game. | ||||
|  * I've had good results with settings around 5 (20 fps). | ||||
|  *  | ||||
|  */ | ||||
| const uint8_t PCARS_dimcolor = 20; | ||||
| WiFiUDP UDP; | ||||
| const unsigned int PCARS_localUdpPort = 5606; // local port to listen on | ||||
| char PCARS_packet[2048]; | ||||
|  | ||||
| char PCARS_tempChar[2]; // Temporary array for u16 conversion | ||||
|  | ||||
| u16 PCARS_RPM; | ||||
| u16 PCARS_maxRPM; | ||||
|  | ||||
| long PCARS_lastRead = millis() - 2001; | ||||
| float PCARS_rpmRatio; | ||||
|  | ||||
| void PCARS_readValues() { | ||||
|  | ||||
|   int PCARS_packetSize = UDP.parsePacket(); | ||||
|   if (PCARS_packetSize) { | ||||
|     int len = UDP.read(PCARS_packet, PCARS_packetSize); | ||||
|     if (len > 0) { | ||||
|       PCARS_packet[len] = 0; | ||||
|     } | ||||
|     if (len == 1367) { // Telemetry packet. Ignoring everything else. | ||||
|       PCARS_lastRead = millis(); | ||||
|  | ||||
|       realtimeLock(realtimeTimeoutMs, REALTIME_MODE_GENERIC); | ||||
|       // current RPM | ||||
|       memcpy(&PCARS_tempChar, &PCARS_packet[124], 2); | ||||
|       PCARS_RPM = (PCARS_tempChar[1] << 8) + PCARS_tempChar[0]; | ||||
|  | ||||
|       // max RPM | ||||
|       memcpy(&PCARS_tempChar, &PCARS_packet[126], 2); | ||||
|       PCARS_maxRPM = (PCARS_tempChar[1] << 8) + PCARS_tempChar[0]; | ||||
|  | ||||
|       if (PCARS_maxRPM) { | ||||
|         PCARS_rpmRatio = constrain((float)PCARS_RPM / (float)PCARS_maxRPM, 0, 1); | ||||
|       } else { | ||||
|         PCARS_rpmRatio = 0.0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| void PCARS_buildcolorbars() { | ||||
|   boolean activated = false; | ||||
|   float ledratio = 0; | ||||
|  | ||||
|   for (uint16_t i = 0; i < ledCount; i++) { | ||||
|     if (PCARS_rpmRatio < .95 || (millis() % 100 > 70 )) { | ||||
|  | ||||
|       ledratio = (float)i / (float)ledCount; | ||||
|       if (ledratio < PCARS_rpmRatio) { | ||||
|         activated = true; | ||||
|       } else { | ||||
|         activated = false; | ||||
|       } | ||||
|       if (ledratio > 0.66) { | ||||
|         setRealtimePixel(i, 0, 0, PCARS_dimcolor + ((255 - PCARS_dimcolor)*activated), 0); | ||||
|       } else if (ledratio > 0.33) { | ||||
|         setRealtimePixel(i, PCARS_dimcolor + ((255 - PCARS_dimcolor)*activated), 0, 0, 0); | ||||
|       } else { | ||||
|         setRealtimePixel(i, 0, PCARS_dimcolor + ((255 - PCARS_dimcolor)*activated), 0, 0); | ||||
|       } | ||||
|     } | ||||
|     else { | ||||
|       setRealtimePixel(i, 0, 0, 0, 0); | ||||
|  | ||||
|     } | ||||
|   } | ||||
|   colorUpdated(5); | ||||
|   strip.show(); | ||||
| } | ||||
|  | ||||
| void userSetup() | ||||
| { | ||||
|   UDP.begin(PCARS_localUdpPort); | ||||
| } | ||||
|  | ||||
| void userConnected() | ||||
| { | ||||
|   // new wifi, who dis? | ||||
| } | ||||
|  | ||||
| void userLoop() | ||||
| { | ||||
|   PCARS_readValues(); | ||||
|   if (PCARS_lastRead > millis() - 2000) { | ||||
|     PCARS_buildcolorbars(); | ||||
|   } | ||||
| } | ||||
| @@ -0,0 +1,37 @@ | ||||
| # QuinLED-Dig-Quad Preassembled Unofficial Build | ||||
|  | ||||
| This usermod targets the [Preassembled QuinLED-Dig-Quad](https://quinled.info/pre-assembled-quinled-dig-quad/). Tested on board revision v1r6b, | ||||
| and includes the following features: | ||||
|  | ||||
|  * **Multi-channel Support** - enabling use of LED1, LED2, LED3, LED4 pins to work using segments | ||||
|  * **Temperature Sensor Support** - pulls readings from the built-in temperature sensor and adds the reading to the *Info* page in the UI | ||||
|  | ||||
| ## Background | ||||
|  | ||||
| As a starting point, you should check out this awesome video from Quindor: [How to compile WLED yourself](https://quinled.info/2020/12/22/livestream-wled-compile/). The usermod you are reading now just provides some shortcuts for parts of what were covered in that video. | ||||
|  | ||||
| ## Build Firmware with Multi-channel and Temp Support | ||||
|  | ||||
| 1. Copy the `platformio_override.ini` file to the project's root directory | ||||
| 1. If using VS Code with the PlatformIO plugin like in the video, you will now see this new project task listed in the PLATFORMIO panel at the bottom as `env:QL-DigQuad-Pre-v0.1` (you probably need to hit the refresh button)  | ||||
|  | ||||
|    <img src="images/pio-screenshot.png" width="400px"/> | ||||
|  | ||||
| 1. Edit this file from the root directory as needed: | ||||
|  | ||||
|    <img src="images/params.png" width="400px"/> | ||||
|  | ||||
|    * `PIXEL_COUNTS` may need to be adjusted for your set-up. E.g. I have lots of LEDs in Channel 1, but that's probably unusual for most | ||||
|    * `DATA_PINS` may need to be changed to "16,3,1,26" instead of "16,1,3,26" apparently depending on the board revision or some such | ||||
|  | ||||
| 1. Build the mod (e.g. click `Build` from the project task circled above) and update your firmware using the `QL-DigQuad-Pre-v0.1` file, e.g. using _Manual OTA_ from the Config menu. Based on the video and my own experience, you might need to build twice 🤷♂️. | ||||
|  | ||||
| ## Observing Temperature | ||||
|  | ||||
| Hopefully you can now see the Temperature listed in the Info page. If not, use Chrome Developer Tools to find the current temperature | ||||
|  | ||||
| 1. Open the Developer Tools Console | ||||
| 2. Enter `lastinfo.u.Temperature` to view the Temperature array | ||||
|  | ||||
|    <img src="images/json-temp.png" width="300px"/> | ||||
|  | ||||
| After Width: | Height: | Size: 296 KiB | 
| After Width: | Height: | Size: 69 KiB | 
 fishbone-git
					fishbone-git