Compare commits
	
		
			399 Commits
		
	
	
		
			8a558c3121
			...
			v1.5.12-be
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 54275f2ac9 | |||
| fbd9cb66f1 | |||
| f1cdd3f41d | |||
| d897817020 | |||
| 686eb22232 | |||
| a2816da654 | |||
| cc8f1cfd7b | |||
| d195f76d5e | |||
| 6bed3b086c | |||
| 3dd4b82710 | |||
| bc41205f15 | |||
| f450d1efdf | |||
| 6e94092a74 | |||
| ece510099e | |||
| 1f01af4da9 | |||
| c5d24d5972 | |||
| 48556b9519 | |||
| 2ac8effe04 | |||
| 4e58407af8 | |||
| d776956c5e | |||
| 25233f70d5 | |||
| b4584364d6 | |||
| 33ea062773 | |||
| 771b0a4839 | |||
| c48003e1b2 | |||
| 83dec4c876 | |||
| dca9ef8d08 | |||
| 513e02b867 | |||
| 99babe2b4a | |||
| c17ab2c434 | |||
| ec7386922e | |||
| 1eb81fad5d | |||
| 9d406e3428 | |||
| 5c2db22a90 | |||
| 164c7b2af5 | |||
| cd1c93c485 | |||
| 15219fa1e4 | |||
| 206db69e6d | |||
| 9e67af7343 | |||
| 9e58b042c8 | |||
| 55200d31cd | |||
| 65967ca047 | |||
| 86e5f7e48a | |||
| e4d1ba6c1c | |||
| 7ccdde8489 | |||
| 88598611c5 | |||
| 619979ab14 | |||
| 377f4bc146 | |||
| 174c48f734 | |||
| 7cbd34bc91 | |||
| fdeb6d5b61 | |||
| f7484f635e | |||
| fb7dca38f0 | |||
| 90ce30215f | |||
|  | 69ae5cab5f | ||
|  | 5fa93f2695 | ||
|  | 0e00fd8b91 | ||
|  | 4706152022 | ||
|  | accb02ab80 | ||
|  | 5509d98969 | ||
|  | d7ee52ba1f | ||
|  | a7c99d3f26 | ||
|  | 0a02912e4a | ||
|  | 89a5728cc0 | ||
|  | f133a1b321 | ||
|  | b95497aec2 | ||
| 876e9c62d8 | |||
| 765cb5319d | |||
| 9a9ed175dd | |||
| a156cac18e | |||
|  | 5b04c2eb80 | ||
|  | 09f4c43f89 | ||
|  | b94db80321 | ||
|  | ec0e544f30 | ||
|  | d815733550 | ||
|  | b6d82c8afe | ||
|  | afef544c66 | ||
|  | 97a1368747 | ||
|  | 6b6aec07b3 | ||
|  | 85a9bcf8bd | ||
|  | 852a2f4c69 | ||
|  | c450df59aa | ||
|  | 4b81703e38 | ||
|  | 722ef421cb | ||
| 7ba0c4f933 | |||
| b0cd731c5a | |||
| d0b793a300 | |||
| f022bee578 | |||
| 7c320a87fe | |||
| 3286b64836 | |||
| 0777b6371d | |||
| 739fe7e764 | |||
| fcdf91071c | |||
| 5f8953a19d | |||
| ffb1117150 | |||
| c919eeb848 | |||
| c317610229 | |||
| 43177c670e | |||
| 73c3457f40 | |||
| 1b50694f5f | |||
| cf62e12aa4 | |||
| 48edde8557 | |||
|  | b583ef71ad | ||
|  | cb5d8ac10a | ||
|  | b991f2ee27 | ||
|  | bf48c6d4e1 | ||
|  | e2e0a23f0a | ||
|  | 5d2d5e9ee1 | ||
|  | 537f452601 | ||
|  | 7e76612bb4 | ||
|  | faaffee391 | ||
|  | f038020042 | ||
|  | d536181a73 | ||
|  | 8343fe887b | ||
| e38220739d | |||
| 3bb6c1caf5 | |||
| fc48d6e67c | |||
| 37df07f102 | |||
| aeb61ba462 | |||
| 8484c1310b | |||
|  | 7f25f3e14f | ||
|  | fd7b4c25b3 | ||
| 150a178038 | |||
| d490b116b9 | |||
| 8b43f34a86 | |||
| 5bc6192b6f | |||
| 7a85ce6a04 | |||
| 2202d9a1aa | |||
| 68fa1e77a1 | |||
| 7dbca0ab87 | |||
| 9c06fe6725 | |||
| 24b3521f83 | |||
| 1cf392c1cd | |||
| 6c9f290bac | |||
|  | 69d6ba4bcb | ||
|  | eab937d6ca | ||
|  | 21ec4e0ff3 | ||
|  | 27ef8399e4 | ||
| c2a09b21a0 | |||
| 2920159f32 | |||
| 0937a9e9f0 | |||
| 2e19bccfa9 | |||
| 818b8387c0 | |||
| 859e89431e | |||
| 3f2beb6f54 | |||
| 6dc26ca51f | |||
| 56248ff2cb | |||
| 0becae7ed6 | |||
| 6a4945666e | |||
| 3d31833f50 | |||
| 97d1519489 | |||
| 599cc47443 | |||
| f608c4a19b | |||
| b1f7923770 | |||
| aa2eb91d64 | |||
| c78c20979d | |||
| 35d2445c6c | |||
| e79c522e46 | |||
| 537607ed40 | |||
| cf8cce72a5 | |||
| 7e330dca1a | |||
| 0b356609d1 | |||
| d943d15c0a | |||
| 01f1e123ac | |||
| a345b76cd2 | |||
| 012f91851e | |||
| 836e48bde2 | |||
| 9ed3c70c01 | |||
| a6a8c69aee | |||
| e23f3a2151 | |||
| ddb4cd8e53 | |||
| f73306f0b9 | |||
| d45313a3ff | |||
| a450d4bd1a | |||
| 70350e19f8 | |||
| d48d994c00 | |||
| 7613effccf | |||
| 32bb85f897 | |||
| 7280d5be7f | |||
| e9d32ee060 | |||
| ada4a84942 | |||
| aba28422bd | |||
| e32aa6ec51 | |||
| 4a55620d39 | |||
| 04a18469b5 | |||
| 7b18266534 | |||
| 1c4d5f3874 | |||
| d81acb2b61 | |||
| a2eb57cd7a | |||
| 8c7fc159d3 | |||
| 1c619c5bcb | |||
| 476d3e82e2 | |||
| 2e05651f88 | |||
| 3c294a135f | |||
| f1b803a3c1 | |||
| bb751b6289 | |||
| 5c4ba9f0ba | |||
| 7fd01bd1b9 | |||
| 19d70301f5 | |||
| fad84e12c8 | |||
| 4fa21d3c0e | |||
| 696efc4d79 | |||
| f22a01127c | |||
| 29868e7101 | |||
| 92d377713d | |||
| 823db6157c | |||
| 8732c81bb9 | |||
| 458cc4eaf2 | |||
| e7bbf45a9f | |||
| 83d14b32d1 | |||
| a8ce964add | |||
| 2bf7c9fb7d | |||
|  | 69f01d1e57 | ||
|  | ac8adca84d | ||
|  | 99231786a5 | ||
|  | c701149c64 | ||
| 8536b4f8fa | |||
| 07a919b6ba | |||
| c84c5fa734 | |||
| 8618b90e33 | |||
| 2a60e149b9 | |||
| 57723b5354 | |||
| 7e486191b7 | |||
| d2be752175 | |||
| 610479bc5a | |||
| 97a050ace8 | |||
| b7fa53da7e | |||
| 367e692c74 | |||
| 629b4276cf | |||
| 926a21249b | |||
| cb15dae87e | |||
| 2635c19667 | |||
|  | abed1c9806 | ||
|  | 6cc4efca0a | ||
| db1f33c2b6 | |||
| 1484a6b0da | |||
| 174a58906c | |||
| b5f0472af4 | |||
| 20cc9b196b | |||
| 95c1bc823c | |||
| ff80b05502 | |||
| 491ba7f526 | |||
| edfdef53f4 | |||
| 56d7d8596c | |||
| 89a3fed7a9 | |||
| 1044e91a0a | |||
| f44173824f | |||
| e459b53472 | |||
| 169d73bfc0 | |||
| 024056cb7d | |||
| c78f36d21a | |||
| e040a736b0 | |||
| 054bc43f65 | |||
| 72b6b349c6 | |||
| 31c41576ee | |||
| 190e952ec4 | |||
| 3a744bc1e6 | |||
| 89620a7f00 | |||
| 42f76fc20a | |||
| 536950eeb3 | |||
|  | 124f326670 | ||
|  | fe4d2d7479 | ||
| af34ce45dd | |||
| 43719aac41 | |||
| c0cb3ff5c9 | |||
| 16d0079f7a | |||
| 4c754d84ff | |||
| 48b9bf7076 | |||
| d2c85018f5 | |||
| b6bd4cb9ad | |||
| 8dac49ea9e | |||
| e89bb1d547 | |||
| 5365c0e1b9 | |||
| f25789d703 | |||
|  | 4abe9d6d33 | ||
|  | 65d8cd675f | ||
| e5d0334714 | |||
| 9dfe75ffa2 | |||
| 16364cbd86 | |||
| 68cdd8ab40 | |||
| 1b63ab668f | |||
| 1069781931 | |||
| f67ef8e905 | |||
| eada54eff2 | |||
| a490b77860 | |||
| 48301ade36 | |||
| 52d063b619 | |||
| 76e0b20393 | |||
| d5c005d6f7 | |||
| a765b39896 | |||
| 68866f1632 | |||
| d68f6c4a89 | |||
| a4200e469d | |||
| 1702e2396e | |||
| e5e14dfc99 | |||
| af23b07df1 | |||
| 863d591a17 | |||
| dd7ba3bf5d | |||
| 69675f3c06 | |||
| a818dcd3c0 | |||
| 2ae3df1aab | |||
| b5279b167a | |||
| 3910da9fb5 | |||
| a09fd4fda4 | |||
| 26d53929ac | |||
| e4fe08f54c | |||
| 64e3461264 | |||
| 3eac0e5ac4 | |||
| bc04db91b8 | |||
| 24d91693d9 | |||
| f500f8bd11 | |||
| 94c26590c8 | |||
| 84391faffd | |||
| 4559bae066 | |||
| aae93de7dd | |||
| cdb2d16cf9 | |||
| 0f847a2731 | |||
| cd71949c82 | |||
| aa7fc7e64b | |||
| 6cd280389d | |||
|  | e0f5f48cc4 | ||
|  | daf27820b1 | ||
|  | 0b79891f83 | ||
|  | dd7fbe1119 | ||
|  | 11c5ca3383 | ||
|  | dc2ddb47eb | ||
|  | e3c3b3f42d | ||
|  | 6bb8f565e6 | ||
|  | 8db7765e7e | ||
|  | ec60ca88f1 | ||
|  | dc97740ddc | ||
|  | 17664acf9e | ||
|  | ababe8b842 | ||
|  | 18f7454a76 | ||
|  | 62bcbb2ae8 | ||
|  | e7b5917888 | ||
|  | 62330a3fd8 | ||
|  | 5c57968ba9 | ||
|  | 4556730c6e | ||
|  | 795c926c1f | ||
|  | c92a8b0957 | ||
|  | 8735a9740c | ||
|  | b08da071c2 | ||
|  | 02d0adc6bf | ||
|  | 9c949e74e8 | ||
|  | 24067666ed | ||
|  | 17fcf765fd | ||
|  | 9264333eda | ||
|  | 95a03f92e2 | ||
|  | 66216d57ae | ||
|  | d9e69d8c14 | ||
|  | 5100a669b0 | ||
|  | 1ec09ebf3a | ||
|  | 4ad89b68a7 | ||
|  | 7ef0cc44d5 | ||
|  | 758acaff9f | ||
|  | fe962b2bfa | ||
|  | fed96b9c58 | ||
|  | aec07f3c6d | ||
|  | 2d072ee09a | ||
|  | b5cb5b17ea | ||
|  | b55b6e3fd5 | ||
|  | c3e7758920 | ||
|  | 238b928236 | ||
|  | 66395028a6 | ||
|  | 24ce0ca6df | ||
|  | 64403b9599 | ||
|  | 3cf934b920 | ||
|  | ebf6688701 | ||
|  | f68ea3edb0 | ||
|  | 073a5f4539 | ||
|  | 16321c9461 | ||
|  | 69bd5c3eb2 | ||
|  | f9530f6d9a | ||
| a328fbc6a6 | |||
| 83f2f0834d | |||
| 6f52cd1686 | |||
| 6632aa8f95 | |||
| c1122ad87d | |||
|  | 1aeced76a2 | ||
|  | d434fde92e | ||
| 967ec35c6a | |||
| f60113aa83 | |||
| 63a7398979 | |||
| 40cb504251 | |||
| 41a4f8af4a | |||
| e122224472 | |||
| 5d3a8d971f | |||
| e62e5e7062 | |||
| 726a60882d | |||
| 2918b4ca77 | |||
| 955ba0f001 | |||
| 8cf7dc0b77 | |||
| 33e4b371ed | |||
| fd832d8808 | |||
| c0e213a4ac | |||
| bcc00f711b | |||
| 78f336d5d7 | |||
| ee7f8ff517 | 
							
								
								
									
										42
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,42 @@ | |||||||
|  | .pio | ||||||
|  | .vscode/ | ||||||
|  | .aider* | ||||||
|  | .DS_Store | ||||||
|  | ._* | ||||||
|  | **/.DS_Store | ||||||
|  | **/.Spotlight-V100 | ||||||
|  | **/.Trashes | ||||||
|  | **/.fseventsd | ||||||
|  | .AppleDouble | ||||||
|  | **/.DS_Store | ||||||
|  | **/.Spotlight-V100 | ||||||
|  | **/.Trashes | ||||||
|  | **/.fseventsd | ||||||
|  | .AppleDouble | ||||||
|  | .aider.chat.history.md | ||||||
|  | .aider.input.history | ||||||
|  | .DS_Store | ||||||
|  | .gitignore | ||||||
|  | .aider.tags.cache.v3/cache.db | ||||||
|  | .aider.tags.cache.v3/cache.db-shm | ||||||
|  | .aider.tags.cache.v3/cache.db-wal | ||||||
|  | .vscode/c_cpp_properties.json | ||||||
|  | .vscode/launch.json | ||||||
|  | .vscode/ipch | ||||||
|  | .vscode/extensions.json | ||||||
|  | .vscode/launch.json | ||||||
|  | include/README | ||||||
|  | lib/README | ||||||
|  | test/README | ||||||
|  | .aider* | ||||||
|  | data/* | ||||||
|  | !data/ | ||||||
|  | !data/.gitkeep | ||||||
|  | # important | ||||||
|  | html/bambu_credentials.json | ||||||
|  | html/spoolman_url.json | ||||||
|  | _local/* | ||||||
|  | website/* | ||||||
|  | release.sh | ||||||
|  | .github/copilot-instructions.md | ||||||
|  | data | ||||||
							
								
								
									
										595
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						| @@ -1,5 +1,600 @@ | |||||||
| # Changelog | # Changelog | ||||||
|  |  | ||||||
|  | ## [1.5.12-beta5] - 2025-08-29 | ||||||
|  | ### Added | ||||||
|  | - enhance NDEF decoding with detailed validation and debugging output | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for beta version v1.5.12-beta5 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.12-beta4] - 2025-08-29 | ||||||
|  | ### Added | ||||||
|  | - enhance NDEF decoding to validate structure and extract JSON payload | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for beta version v1.5.12-beta4 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.12-beta3] - 2025-08-29 | ||||||
|  | ### Added | ||||||
|  | - add logging for decoded JSON data in NFC processing | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for beta version v1.5.12-beta3 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.12-beta2] - 2025-08-29 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for beta version v1.5.12-beta2 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - enhance filament creation logic to include dynamic comments based on payload | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.12-beta1] - 2025-08-28 | ||||||
|  | ### Added | ||||||
|  | - implement filament and spool creation in Spoolman API | ||||||
|  | - Add JSON structure comments for filament and spool creation | ||||||
|  | - Add vendor and filament management to API; implement recycling factory handling in NFC | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for beta version v1.5.12-beta1 | ||||||
|  | - Merge branch 'main' into recyclingfabrik | ||||||
|  | - Merge branch 'main' into recyclingfabrik | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.12] - 2025-08-28 | ||||||
|  | ### Added | ||||||
|  | - add numbering to update sections in upgrade.html refactor: improve readability of checkSpoolmanInstance function | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.5.12 | ||||||
|  | - clean up library dependencies in platformio.ini | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.11-beta4] - 2025-08-28 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for beta version v1.5.11-beta4 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - remove debug logging from checkSpoolmanInstance function | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.11-beta3] - 2025-08-28 | ||||||
|  | ### Added | ||||||
|  | - add logging for spoolman status in checkSpoolmanInstance function | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for beta version v1.5.11-beta3 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.11-beta2] - 2025-08-28 | ||||||
|  | ### Added | ||||||
|  | - add logging for healthy spoolman instance check | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for beta version v1.5.11-beta2 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.11-beta1] - 2025-08-28 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for beta version v1.5.11-beta1 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - adjust spacing in loop structure and enable tare function in scale loop | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.11] - 2025-08-27 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.5.11 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.10] - 2025-08-27 | ||||||
|  | ### Added | ||||||
|  | - improve weight processing logic and add auto-send feature for Bambu spool ID | ||||||
|  | - improve weight processing logic and add auto-send feature for Bambu spool ID | ||||||
|  | - Adds a link to the spool in spoolman when reading a spool tag | ||||||
|  | - Fixes types and some issues in the new graphics | ||||||
|  | - Adds ENABLE_HEAP_DEBUGGING define as comment to the build flags | ||||||
|  | - Adds data directory and further .vscode files to to .gitignore | ||||||
|  | - Introduces new heap debugging feature and fixes some memory leaks in website feature | ||||||
|  | - Fixes some issues with the new location tags | ||||||
|  | - Adds new feature to write and read location tags | ||||||
|  | - Adds slight debouncing to the scale loop weight logic | ||||||
|  | - add loadcell desc. | ||||||
|  | - implement multi-color filament display and styles for dropdown options | ||||||
|  | - add remaining weight logging for PUT requests and improve error reporting in sendToApi function | ||||||
|  | - add remaining weight logging and display after successful spool update | ||||||
|  | - add weight field to update payload in updateSpoolTagId function | ||||||
|  | - add auto-tare functionality and update scale handling based on touch sensor connection | ||||||
|  | - add touch sensor connection check and update logic | ||||||
|  | - add manual tare functionality for scale | ||||||
|  | - add debounce handling for TTP223 touch sensor | ||||||
|  | - add TTP223 touch sensor support and wiring configuration | ||||||
|  | - Renamed states of NFC state machine and introduced new state machine for spoolman API | ||||||
|  | - add forced cache refresh after removing and saving Bambu credentials | ||||||
|  | - add functionality to remove Bambu credentials and update API handling | ||||||
|  | - add rfid_bambu.html and update bambu connection handling | ||||||
|  | - add error handling for missing vendor IDs in filament data | ||||||
|  | - add WiFi connection check and restart Bambu if not connected | ||||||
|  | - added new .step, now with correct individual parts | ||||||
|  | - added changelog | ||||||
|  | - Add files via upload | ||||||
|  | - added .stp files of modifications | ||||||
|  | - added merged picture | ||||||
|  | - added pictures of components bought from AliE | ||||||
|  | - Add files via upload | ||||||
|  | - added pictures for heat insert location | ||||||
|  | - added pictures showing heat insert location | ||||||
|  | - remove unnecessary delay in MQTT setup and add delay before restart | ||||||
|  | - add new 3D print file for Filaman scale | ||||||
|  | - added Discord Server | ||||||
|  | - add support for Spoolman Octoprint Plugin in README files | ||||||
|  | - add OctoPrint integration with configurable fields and update functionality | ||||||
|  | - add version comparison function and check for outdated versions before updates | ||||||
|  | - remove unused version and protocol fields from JSON output; add error message for insufficient memory | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.5.10 | ||||||
|  | - Merge branch 'main' of https://gitlab.fire-devils.org/3D-Druck/Filaman | ||||||
|  | - Changed Amazon Link for PN532 | ||||||
|  | - Changed Amazon Link for PN532 | ||||||
|  | - update changelog and header for version v1.5.9 | ||||||
|  | - update platformio.ini for version v1.5.9 | ||||||
|  | - Enhance API to support weight updates after successful spool tag updates | ||||||
|  | - update changelog and header for version v1.5.8 | ||||||
|  | - update platformio.ini for version v1.5.8 | ||||||
|  | - Merge pull request #45 from janecker/nfc_write_improvements | ||||||
|  | - Introduces periodic Spoolman Healthcheck | ||||||
|  | - Improves init - NFC reading now only starts after boot is finished | ||||||
|  | - Further improvements on NFC writing | ||||||
|  | - Merge pull request #44 from janecker/graphics_rework | ||||||
|  | - Graphic rework of the NFC writing process | ||||||
|  | - Remove unused parameter of sendNfcData() | ||||||
|  | - Reworks startup graphics and timings | ||||||
|  | - update changelog and header for version v1.5.7 | ||||||
|  | - update platformio.ini for version v1.5.7 | ||||||
|  | - clean up unused variables and improve .gitignore entries | ||||||
|  | - update changelog and header for version v1.5.6 | ||||||
|  | - update webpages for version v1.5.6 | ||||||
|  | - update platformio.ini for version v1.5.6 | ||||||
|  | - Merge pull request #42 from janecker/configuration_nvs_storage | ||||||
|  | - Merge branch 'main' into configuration_nvs_storage | ||||||
|  | - Changes configuration storage of spoolman and bambu values | ||||||
|  | - update changelog and header for version v1.5.5 | ||||||
|  | - update platformio.ini for version v1.5.5 | ||||||
|  | - update changelog and header for version v1.5.4 | ||||||
|  | - update platformio.ini for version v1.5.4 | ||||||
|  | - Merge branch 'main' of github.com:ManuelW77/Filaman | ||||||
|  | - Merge pull request #39 from janecker/location_tags | ||||||
|  | - Merge pull request #38 from janecker/scale_debouncing | ||||||
|  | - update changelog and header for version v1.5.3 | ||||||
|  | - update platformio.ini for version v1.5.3 | ||||||
|  | - Affiliate Links | ||||||
|  | - update changelog and header for version v1.5.2 | ||||||
|  | - update platformio.ini for version v1.5.2 | ||||||
|  | - update changelog and header for version v1.5.1 | ||||||
|  | - update version to 1.5.1 and improve OTA update handling with task management | ||||||
|  | - update changelog and header for version v1.4.14 | ||||||
|  | - update platformio.ini for version v1.4.14 | ||||||
|  | - update changelog and header for version v1.4.13 | ||||||
|  | - update platformio.ini for version v1.4.13 | ||||||
|  | - update changelog and header for version v1.4.12 | ||||||
|  | - update platformio.ini for version v1.4.12 | ||||||
|  | - update README files to clarify PN532 DIP switch settings | ||||||
|  | - update changelog and header for version v1.4.11 | ||||||
|  | - update platformio.ini for version v1.4.11 | ||||||
|  | - Merge branch 'main' of github.com:ManuelW77/Filaman | ||||||
|  | - update changelog and header for version v1.4.10 | ||||||
|  | - update platformio.ini for version v1.4.10 | ||||||
|  | - Merge pull request #31 from janecker/nfc_rework | ||||||
|  | - Introducing enum for handling the NFC state to improve code readability | ||||||
|  | - update changelog and header for version v1.4.9 | ||||||
|  | - update platformio.ini for version v1.4.9 | ||||||
|  | - update changelog and header for version v1.4.8 | ||||||
|  | - update platformio.ini for version v1.4.8 | ||||||
|  | - Merge pull request #30 from janecker/main | ||||||
|  | - Merge branch 'testing' into main | ||||||
|  | - update changelog and header for version v1.4.7 | ||||||
|  | - update platformio.ini for version v1.4.7 | ||||||
|  | - Merge branch 'testing' | ||||||
|  | - update remove button for Bambu credentials with red background | ||||||
|  | - Merge pull request #28 from tugsi/main | ||||||
|  | - update changelog and header for version v1.4.6 | ||||||
|  | - update platformio.ini for version v1.4.6 | ||||||
|  | - update changelog and header for version v1.4.5 | ||||||
|  | - update platformio.ini for version v1.4.5 | ||||||
|  | - Merge branch 'testing' | ||||||
|  | - remove unused request_topic subscription and reduce MQTT task stack size | ||||||
|  | - Merge pull request #26 from tugsi/main | ||||||
|  | - rename report_topic to topic and update MQTT subscription logic, switched publish topic to request | ||||||
|  | - update changelog and header for version v1.4.4 | ||||||
|  | - update platformio.ini for version v1.4.4 | ||||||
|  | - update changelog and header for version v1.4.3 | ||||||
|  | - update platformio.ini for version v1.4.3 | ||||||
|  | - update changelog and header for version v1.4.2 | ||||||
|  | - update platformio.ini for version v1.4.2 | ||||||
|  | - increase stack size for BambuMqtt task | ||||||
|  | - update Discord Link | ||||||
|  | - update Discord Link | ||||||
|  | - remove commented-out subscription topic in MQTT setup | ||||||
|  | - update changelog and header for version v1.4.1 | ||||||
|  | - update platformio.ini for version v1.4.1 | ||||||
|  | - refactor length calculation to convert total length to meters before formatting | ||||||
|  | - Merge pull request #16 from spitzbirne32/main | ||||||
|  | - improved housing to show display better | ||||||
|  | - removed CAD, as they were all duplicates | ||||||
|  | - typo in AliE link | ||||||
|  | - Delete usermod/spitzbirne32/STL/README.md | ||||||
|  | - Update README.md | ||||||
|  | - moved pictures of parts into dedicated folders | ||||||
|  | - Update README.md | ||||||
|  | - Update README.md | ||||||
|  | - Update README.md | ||||||
|  | - Delete usermod/spitzbirne32/STL/ScaleTop_Heatinsert_Location_usermod_spitzbirne32_.png | ||||||
|  | - Delete usermod/spitzbirne32/STL/Housing_Heatinsert_Location_usermod_spitzbirne32_.png | ||||||
|  | - created folders | ||||||
|  | - Update README.md | ||||||
|  | - Update README.md | ||||||
|  | - Create README.md | ||||||
|  | - Update README.md | ||||||
|  | - Update README.md | ||||||
|  | - Create README.md | ||||||
|  | - Merge pull request #15 from ManuelW77/main | ||||||
|  | - Merge pull request #14 from janecker/scale-calibration-rework | ||||||
|  | - Reworks the scale calibration handling | ||||||
|  | - remove redundant scale calibration checks and enhance task management | ||||||
|  | - enhance AMS data handling and streamline spool auto-setting logic | ||||||
|  | - adjust stack size and improve scale calibration logic | ||||||
|  | - update labels and input types for better clarity and functionality | ||||||
|  | - update documentation for clarity and accuracy | ||||||
|  | - update changelog and header for version v1.4.0 | ||||||
|  | - update NFC tag references to include NTAG213 and clarify storage capacity | ||||||
|  | - bump version to 1.4.0 | ||||||
|  | - remove unused version and protocol fields from NFC data packet | ||||||
|  | - sort vendors alphabetically in the dropdown list | ||||||
|  | - Merge pull request #10 from janecker/nfc-improvements | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - Fixes issue that scale not calibrated message was not shown | ||||||
|  | - Improves NFC writing workaround and removes debug output | ||||||
|  | - Fixes typos in upgrade page | ||||||
|  | - Reworks graphics of tag reading and some api fixes | ||||||
|  | - Replaces usage of String with const char* in heap debug function | ||||||
|  | - Merge pull request #41 from janecker/memory_leak_fixes | ||||||
|  | - Fixes compiler warnings in nfc | ||||||
|  | - Memory leak fixes in api and nfc, location tag fix | ||||||
|  | - Merge pull request #40 from janecker/location_bambu_fix | ||||||
|  | - uncomment monitor_port configuration in platformio.ini | ||||||
|  | - update spool weight conditionally based on NFC ID | ||||||
|  | - update weight field in update payload to only include values greater than 10 | ||||||
|  | - increase stack size for sendToApi task to improve stability | ||||||
|  | - adjust tare weight tolerance to ignore deviations of 2g | ||||||
|  | - improve weight stability check before sending to API | ||||||
|  | - update touch sensor connection logic to correctly identify connection status | ||||||
|  | - update TTP223 pin configuration and adjust touch sensor logic | ||||||
|  | - enhance HTTP method handling in sendToApi function | ||||||
|  | - improve HTTP client configuration and clear update documents after API calls | ||||||
|  | - Fixes memory leak in HTTPClient by disabling connection reuse | ||||||
|  | - update reload logic after removing and saving Bambu credentials for better cache handling | ||||||
|  | - handle Bambu connection state by introducing bambuDisabled flag | ||||||
|  | - handle potential undefined value for tray_info_idx in handleSpoolIn function, by @tugsi | ||||||
|  | - Fix rfid.js-Failure with X1-Series, if you wanna send a Spool to AMS:  - Uncaught TypeError: Cannot read properties of undefined (reading 'replace')     at handleSpoolIn (rfid.js:493:67)     at HTMLButtonElement.onclick ((Index):1:1) handleSpoolIn	@	rfid.js:493 onclick	@	(Index):1 | ||||||
|  | - increase MQTT buffer size and adjust task stack size | ||||||
|  | - Fix BufferSize for larger JSONs from X-Series | ||||||
|  | - adjust weight threshold for tare check to allow negative values | ||||||
|  | - use unique client ID for MQTT connection to avoid conflicts | ||||||
|  | - reload page after firmware update completion | ||||||
|  | - increase WiFi connection timeout from 5 to 10 seconds | ||||||
|  | - ensure valid URL format and remove trailing slash in setupWebserver | ||||||
|  | - correct typo in console log for total length | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.9] - 2025-08-11 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.5.9 | ||||||
|  | - Enhance API to support weight updates after successful spool tag updates | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.8] - 2025-08-10 | ||||||
|  | ### Added | ||||||
|  | - Adds a link to the spool in spoolman when reading a spool tag | ||||||
|  | - Fixes types and some issues in the new graphics | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.5.8 | ||||||
|  | - Merge pull request #45 from janecker/nfc_write_improvements | ||||||
|  | - Introduces periodic Spoolman Healthcheck | ||||||
|  | - Improves init - NFC reading now only starts after boot is finished | ||||||
|  | - Further improvements on NFC writing | ||||||
|  | - Merge pull request #44 from janecker/graphics_rework | ||||||
|  | - Graphic rework of the NFC writing process | ||||||
|  | - Remove unused parameter of sendNfcData() | ||||||
|  | - Reworks startup graphics and timings | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - Fixes issue that scale not calibrated message was not shown | ||||||
|  | - Improves NFC writing workaround and removes debug output | ||||||
|  | - Fixes typos in upgrade page | ||||||
|  | - Reworks graphics of tag reading and some api fixes | ||||||
|  | - Replaces usage of String with const char* in heap debug function | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.7] - 2025-07-28 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.5.7 | ||||||
|  | - clean up unused variables and improve .gitignore entries | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.6] - 2025-07-28 | ||||||
|  | ### Added | ||||||
|  | - Adds ENABLE_HEAP_DEBUGGING define as comment to the build flags | ||||||
|  | - Adds data directory and further .vscode files to to .gitignore | ||||||
|  | - Introduces new heap debugging feature and fixes some memory leaks in website feature | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.5.6 | ||||||
|  | - update platformio.ini for version v1.5.6 | ||||||
|  | - Merge pull request #42 from janecker/configuration_nvs_storage | ||||||
|  | - Merge branch 'main' into configuration_nvs_storage | ||||||
|  | - Changes configuration storage of spoolman and bambu values | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - Merge pull request #41 from janecker/memory_leak_fixes | ||||||
|  | - Fixes compiler warnings in nfc | ||||||
|  | - Memory leak fixes in api and nfc, location tag fix | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.5] - 2025-07-22 | ||||||
|  | ### Added | ||||||
|  | - Fixes some issues with the new location tags | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.5.5 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - Merge pull request #40 from janecker/location_bambu_fix | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.4] - 2025-07-22 | ||||||
|  | ### Added | ||||||
|  | - Adds new feature to write and read location tags | ||||||
|  | - Adds slight debouncing to the scale loop weight logic | ||||||
|  | - add loadcell desc. | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.5.4 | ||||||
|  | - Merge branch 'main' of github.com:ManuelW77/Filaman | ||||||
|  | - Merge pull request #39 from janecker/location_tags | ||||||
|  | - Merge pull request #38 from janecker/scale_debouncing | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - uncomment monitor_port configuration in platformio.ini | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.3] - 2025-04-25 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.5.3 | ||||||
|  | - Affiliate Links | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - update spool weight conditionally based on NFC ID | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.2] - 2025-04-23 | ||||||
|  | ### Added | ||||||
|  | - implement multi-color filament display and styles for dropdown options | ||||||
|  | - add remaining weight logging for PUT requests and improve error reporting in sendToApi function | ||||||
|  | - add remaining weight logging and display after successful spool update | ||||||
|  | - add weight field to update payload in updateSpoolTagId function | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.5.2 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - update weight field in update payload to only include values greater than 10 | ||||||
|  | - increase stack size for sendToApi task to improve stability | ||||||
|  | - adjust tare weight tolerance to ignore deviations of 2g | ||||||
|  | - improve weight stability check before sending to API | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.5.1] - 2025-03-30 | ||||||
|  | ### Changed | ||||||
|  | - update version to 1.5.1 and improve OTA update handling with task management | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.14] - 2025-03-30 | ||||||
|  | ### Added | ||||||
|  | - add auto-tare functionality and update scale handling based on touch sensor connection | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.14 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.13] - 2025-03-30 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.13 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - update touch sensor connection logic to correctly identify connection status | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.12] - 2025-03-30 | ||||||
|  | ### Added | ||||||
|  | - add touch sensor connection check and update logic | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.12 | ||||||
|  | - update README files to clarify PN532 DIP switch settings | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.11] - 2025-03-30 | ||||||
|  | ### Added | ||||||
|  | - Renamed states of NFC state machine and introduced new state machine for spoolman API | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.11 | ||||||
|  | - Merge branch 'main' of github.com:ManuelW77/Filaman | ||||||
|  | - Merge pull request #31 from janecker/nfc_rework | ||||||
|  | - Introducing enum for handling the NFC state to improve code readability | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.10] - 2025-03-30 | ||||||
|  | ### Added | ||||||
|  | - add manual tare functionality for scale | ||||||
|  | - add debounce handling for TTP223 touch sensor | ||||||
|  | - add TTP223 touch sensor support and wiring configuration | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.10 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - update TTP223 pin configuration and adjust touch sensor logic | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.9] - 2025-03-29 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.9 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - enhance HTTP method handling in sendToApi function | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.8] - 2025-03-29 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.8 | ||||||
|  | - Merge pull request #30 from janecker/main | ||||||
|  | - Merge branch 'testing' into main | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - improve HTTP client configuration and clear update documents after API calls | ||||||
|  | - Fixes memory leak in HTTPClient by disabling connection reuse | ||||||
|  | - update reload logic after removing and saving Bambu credentials for better cache handling | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.7] - 2025-03-27 | ||||||
|  | ### Added | ||||||
|  | - add forced cache refresh after removing and saving Bambu credentials | ||||||
|  | - add functionality to remove Bambu credentials and update API handling | ||||||
|  | - add rfid_bambu.html and update bambu connection handling | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.7 | ||||||
|  | - Merge branch 'testing' | ||||||
|  | - update remove button for Bambu credentials with red background | ||||||
|  | - Merge pull request #28 from tugsi/main | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - handle Bambu connection state by introducing bambuDisabled flag | ||||||
|  | - Fix rfid.js-Failure with X1-Series, if you wanna send a Spool to AMS:  - Uncaught TypeError: Cannot read properties of undefined (reading 'replace')     at handleSpoolIn (rfid.js:493:67)     at HTMLButtonElement.onclick ((Index):1:1) handleSpoolIn	@	rfid.js:493 onclick	@	(Index):1 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.6] - 2025-03-26 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.6 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - handle potential undefined value for tray_info_idx in handleSpoolIn function, by @tugsi | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.5] - 2025-03-25 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.5 | ||||||
|  | - Merge branch 'testing' | ||||||
|  | - remove unused request_topic subscription and reduce MQTT task stack size | ||||||
|  | - Merge pull request #26 from tugsi/main | ||||||
|  | - rename report_topic to topic and update MQTT subscription logic, switched publish topic to request | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - increase MQTT buffer size and adjust task stack size | ||||||
|  | - Fix BufferSize for larger JSONs from X-Series | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.4] - 2025-03-23 | ||||||
|  | ### Added | ||||||
|  | - add error handling for missing vendor IDs in filament data | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.4 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - adjust weight threshold for tare check to allow negative values | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.3] - 2025-03-23 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.3 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.2] - 2025-03-23 | ||||||
|  | ### Added | ||||||
|  | - add WiFi connection check and restart Bambu if not connected | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.2 | ||||||
|  | - increase stack size for BambuMqtt task | ||||||
|  | - update Discord Link | ||||||
|  | - update Discord Link | ||||||
|  | - remove commented-out subscription topic in MQTT setup | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - use unique client ID for MQTT connection to avoid conflicts | ||||||
|  | - reload page after firmware update completion | ||||||
|  | - increase WiFi connection timeout from 5 to 10 seconds | ||||||
|  | - ensure valid URL format and remove trailing slash in setupWebserver | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.1] - 2025-03-10 | ||||||
|  | ### Added | ||||||
|  | - added new .step, now with correct individual parts | ||||||
|  | - added changelog | ||||||
|  | - Add files via upload | ||||||
|  | - added .stp files of modifications | ||||||
|  | - added merged picture | ||||||
|  | - added pictures of components bought from AliE | ||||||
|  | - Add files via upload | ||||||
|  | - added pictures for heat insert location | ||||||
|  | - added pictures showing heat insert location | ||||||
|  | - remove unnecessary delay in MQTT setup and add delay before restart | ||||||
|  | - add new 3D print file for Filaman scale | ||||||
|  | - added Discord Server | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.1 | ||||||
|  | - refactor length calculation to convert total length to meters before formatting | ||||||
|  | - Merge pull request #16 from spitzbirne32/main | ||||||
|  | - improved housing to show display better | ||||||
|  | - removed CAD, as they were all duplicates | ||||||
|  | - typo in AliE link | ||||||
|  | - Delete usermod/spitzbirne32/STL/README.md | ||||||
|  | - Update README.md | ||||||
|  | - moved pictures of parts into dedicated folders | ||||||
|  | - Update README.md | ||||||
|  | - Update README.md | ||||||
|  | - Update README.md | ||||||
|  | - Delete usermod/spitzbirne32/STL/ScaleTop_Heatinsert_Location_usermod_spitzbirne32_.png | ||||||
|  | - Delete usermod/spitzbirne32/STL/Housing_Heatinsert_Location_usermod_spitzbirne32_.png | ||||||
|  | - created folders | ||||||
|  | - Update README.md | ||||||
|  | - Update README.md | ||||||
|  | - Create README.md | ||||||
|  | - Update README.md | ||||||
|  | - Update README.md | ||||||
|  | - Create README.md | ||||||
|  | - Merge pull request #15 from ManuelW77/main | ||||||
|  | - Merge pull request #14 from janecker/scale-calibration-rework | ||||||
|  | - Reworks the scale calibration handling | ||||||
|  | - remove redundant scale calibration checks and enhance task management | ||||||
|  | - enhance AMS data handling and streamline spool auto-setting logic | ||||||
|  | - adjust stack size and improve scale calibration logic | ||||||
|  | - update labels and input types for better clarity and functionality | ||||||
|  | - update documentation for clarity and accuracy | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - correct typo in console log for total length | ||||||
|  |  | ||||||
|  |  | ||||||
| ## [1.4.0] - 2025-03-01 | ## [1.4.0] - 2025-03-01 | ||||||
| ### Added | ### Added | ||||||
| - add support for Spoolman Octoprint Plugin in README files | - add support for Spoolman Octoprint Plugin in README files | ||||||
|   | |||||||
							
								
								
									
										41
									
								
								README.de.md
									
									
									
									
									
								
							
							
						
						| @@ -9,7 +9,7 @@ Das System integriert sich nahtlos mit der [Spoolman](https://github.com/Donkie/ | |||||||
| Weitere Bilder finden Sie im [img Ordner](/img/) | Weitere Bilder finden Sie im [img Ordner](/img/) | ||||||
| oder auf meiner Website: [FilaMan Website](https://www.filaman.app)   | oder auf meiner Website: [FilaMan Website](https://www.filaman.app)   | ||||||
| Deutsches Erklärvideo: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaOHU)   | Deutsches Erklärvideo: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaOHU)   | ||||||
| Discord Server: [https://discord.gg/vMAx2gf5](https://discord.gg/vMAx2gf5) | Discord Server: [https://discord.gg/my7Gvaxj2v](https://discord.gg/my7Gvaxj2v) | ||||||
|  |  | ||||||
| ### Es gibt jetzt auch ein Wiki, dort sind nochmal alle Funktionen beschrieben: [Wiki](https://github.com/ManuelW77/Filaman/wiki) | ### Es gibt jetzt auch ein Wiki, dort sind nochmal alle Funktionen beschrieben: [Wiki](https://github.com/ManuelW77/Filaman/wiki) | ||||||
|  |  | ||||||
| @@ -54,20 +54,23 @@ Discord Server: [https://discord.gg/vMAx2gf5](https://discord.gg/vMAx2gf5) | |||||||
|  |  | ||||||
| ## Hardware-Anforderungen | ## Hardware-Anforderungen | ||||||
|  |  | ||||||
| ### Komponenten | ### Komponenten (Affiliate Links) | ||||||
| - **ESP32 Entwicklungsboard:** Jede ESP32-Variante. | - **ESP32 Development Board:** Any ESP32 variant. | ||||||
| [Amazon Link](https://amzn.eu/d/aXThslf) | [Amazon Link](https://amzn.to/3FHea6D) | ||||||
| - **HX711 5kg Wägezellen-Verstärker:** Für Gewichtsmessung. | - **HX711 5kg Load Cell Amplifier:** For weight measurement. | ||||||
| [Amazon Link](https://amzn.eu/d/06A0DLb) | [Amazon Link](https://amzn.to/4ja1KTe) | ||||||
| - **OLED 0.96 Zoll I2C weiß/gelb Display:** 128x64 SSD1306. | - **OLED 0.96 Zoll I2C white/yellow Display:** 128x64 SSD1306. | ||||||
| [Amazon Link](https://amzn.eu/d/0AuBp2c) | [Amazon Link](https://amzn.to/445aaa9) | ||||||
| - **PN532 NFC NXP RFID-Modul V3:** Für NFC-Tag-Operationen. | - **PN532 NFC NXP RFID-Modul V3:** For NFC tag operations. | ||||||
| [Amazon Link](https://amzn.eu/d/jfIuQXb) | [Amazon Link](https://amzn.eu/d/gy9vaBX) | ||||||
| - **NFC Tags NTAG213 NTA215:** RFID Tag | - **NFC Tags NTAG213 NTAG215:** RFID Tag | ||||||
| [Amazon Link](https://amzn.eu/d/9Z6mXc1) | [Amazon Link](https://amzn.to/3E071xO) | ||||||
|  | - **TTP223 Touch Sensor (optional):** For reTARE per Button/Touch | ||||||
|  | [Amazon Link](https://amzn.to/4hTChMK) | ||||||
|  |  | ||||||
| ### Pin-Konfiguration |  | ||||||
| | Komponente        | ESP32 Pin | | ### Pin Konfiguration | ||||||
|  | | Component          | ESP32 Pin | | ||||||
| |-------------------|-----------| | |-------------------|-----------| | ||||||
| | HX711 DOUT        | 16        | | | HX711 DOUT        | 16        | | ||||||
| | HX711 SCK         | 17        | | | HX711 SCK         | 17        | | ||||||
| @@ -77,14 +80,22 @@ Discord Server: [https://discord.gg/vMAx2gf5](https://discord.gg/vMAx2gf5) | |||||||
| | PN532 RESET       | 33        | | | PN532 RESET       | 33        | | ||||||
| | PN532 SDA         | 21        | | | PN532 SDA         | 21        | | ||||||
| | PN532 SCL         | 22        | | | PN532 SCL         | 22        | | ||||||
|  | | TTP223 I/O        | 25        | | ||||||
|  |  | ||||||
| **Achte darauf, dass am PN532 die DIP-Schalter auf I2C gestellt sind** | **!! Achte darauf, dass am PN532 die DIP-Schalter auf I2C gestellt sind**   | ||||||
|  | **Nutze den 3V Pin vom ESP für den Touch Sensor** | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | *Die Wägezelle wird bei den meisten HX711 Modulen folgendermaßen verkabelt:   | ||||||
|  | E+ rot   | ||||||
|  | E- schwarz   | ||||||
|  | A- weiß   | ||||||
|  | A+ grün* | ||||||
|  |  | ||||||
| ## Software-Abhängigkeiten | ## Software-Abhängigkeiten | ||||||
|  |  | ||||||
| ### ESP32-Bibliotheken | ### ESP32-Bibliotheken | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -13,7 +13,7 @@ The system integrates seamlessly with [Bambulab](https://bambulab.com/en-us) 3D | |||||||
| More Images can be found in the [img Folder](/img/)   | More Images can be found in the [img Folder](/img/)   | ||||||
| or my website: [FilaMan Website](https://www.filaman.app)   | or my website: [FilaMan Website](https://www.filaman.app)   | ||||||
| german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaOHU)   | german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaOHU)   | ||||||
| Discord Server: [https://discord.gg/vMAx2gf5](https://discord.gg/vMAx2gf5) | Discord Server: [https://discord.gg/my7Gvaxj2v](https://discord.gg/my7Gvaxj2v) | ||||||
|  |  | ||||||
| ### Now more detailed informations about the usage: [Wiki](https://github.com/ManuelW77/Filaman/wiki) | ### Now more detailed informations about the usage: [Wiki](https://github.com/ManuelW77/Filaman/wiki) | ||||||
|  |  | ||||||
| @@ -58,17 +58,19 @@ Discord Server: [https://discord.gg/vMAx2gf5](https://discord.gg/vMAx2gf5) | |||||||
|  |  | ||||||
| ## Hardware Requirements | ## Hardware Requirements | ||||||
|  |  | ||||||
| ### Components | ### Components (Affiliate Links) | ||||||
| - **ESP32 Development Board:** Any ESP32 variant. | - **ESP32 Development Board:** Any ESP32 variant. | ||||||
| [Amazon Link](https://amzn.eu/d/aXThslf) | [Amazon Link](https://amzn.to/3FHea6D) | ||||||
| - **HX711 5kg Load Cell Amplifier:** For weight measurement. | - **HX711 5kg Load Cell Amplifier:** For weight measurement. | ||||||
| [Amazon Link](https://amzn.eu/d/06A0DLb) | [Amazon Link](https://amzn.to/4ja1KTe) | ||||||
| - **OLED 0.96 Zoll I2C white/yellow Display:** 128x64 SSD1306. | - **OLED 0.96 Zoll I2C white/yellow Display:** 128x64 SSD1306. | ||||||
| [Amazon Link](https://amzn.eu/d/0AuBp2c) | [Amazon Link](https://amzn.to/445aaa9) | ||||||
| - **PN532 NFC NXP RFID-Modul V3:** For NFC tag operations. | - **PN532 NFC NXP RFID-Modul V3:** For NFC tag operations. | ||||||
| [Amazon Link](https://amzn.eu/d/jfIuQXb) | [Amazon Link](https://amzn.eu/d/gy9vaBX) | ||||||
| - **NFC Tags NTAG213 NTAG215:** RFID Tag | - **NFC Tags NTAG213 NTAG215:** RFID Tag | ||||||
| [Amazon Link](https://amzn.eu/d/9Z6mXc1) | [Amazon Link](https://amzn.to/3E071xO) | ||||||
|  | - **TTP223 Touch Sensor (optional):** For reTARE per Button/Touch | ||||||
|  | [Amazon Link](https://amzn.to/4hTChMK) | ||||||
|  |  | ||||||
|  |  | ||||||
| ### Pin Configuration | ### Pin Configuration | ||||||
| @@ -82,14 +84,22 @@ Discord Server: [https://discord.gg/vMAx2gf5](https://discord.gg/vMAx2gf5) | |||||||
| | PN532 RESET       | 33        | | | PN532 RESET       | 33        | | ||||||
| | PN532 SDA         | 21        | | | PN532 SDA         | 21        | | ||||||
| | PN532 SCL         | 22        | | | PN532 SCL         | 22        | | ||||||
|  | | TTP223 I/O        | 25        | | ||||||
|  |  | ||||||
| **Make sure that the DIP switches on the PN532 are set to I2C** | **!! Make sure that the DIP switches on the PN532 are set to I2C**   | ||||||
|  | **Use the 3V pin from the ESP for the touch sensor** | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | *The load cell is connected to most HX711 modules as follows:   | ||||||
|  | E+ red   | ||||||
|  | E- black   | ||||||
|  | A- white   | ||||||
|  | A+ green* | ||||||
|  |  | ||||||
| ## Software Dependencies | ## Software Dependencies | ||||||
|  |  | ||||||
| ### ESP32 Libraries | ### ESP32 Libraries | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								html/.DS_Store
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										1
									
								
								html/bambu_credentials.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | {"bambu_ip": "192.168.1.14", "bambu_accesscode": "22772584", "bambu_serialnr": "01P00C492600230","autoSendToBambu":true,"autoSendTime": 60} | ||||||
| @@ -139,17 +139,20 @@ | |||||||
|                 <p id="nfcInfo" class="nfc-status"></p> |                 <p id="nfcInfo" class="nfc-status"></p> | ||||||
|                 <button id="writeNfcButton" class="btn btn-primary hidden" onclick="writeNfcTag()">Write Tag</button> |                 <button id="writeNfcButton" class="btn btn-primary hidden" onclick="writeNfcTag()">Write Tag</button> | ||||||
|             </div> |             </div> | ||||||
|  |  | ||||||
|  |             <div class="feature-box"> | ||||||
|  |                 <h2>Spoolman Locations</h2> | ||||||
|  |                 <label for="locationSelect">Location:</label> | ||||||
|  |                 <div style="display: flex; justify-content: space-between; align-items: center;"> | ||||||
|  |                     <select id="locationSelect" class="styled-select"> | ||||||
|  |                         <option value="">Please choose...</option> | ||||||
|  |                     </select> | ||||||
|  |                 </div> | ||||||
|  |                 <p id="nfcInfoLocation" class="nfc-status"></p> | ||||||
|  |                 <button id="writeLocationNfcButton" class="btn btn-primary hidden" onclick="writeLocationNfcTag()">Write Location Tag</button> | ||||||
|  |             </div> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
|         <!-- Rechte Spalte --> |  | ||||||
|         <div class="column"> |  | ||||||
|             <div class="feature-box"> |  | ||||||
|                 <h2>Bambu AMS</h2> |  | ||||||
|                 <div id="amsDataContainer"> |  | ||||||
|                     <div class="amsData" id="amsData">Wait for AMS-Data...</div> |  | ||||||
|                 </div> |  | ||||||
|             </div> |  | ||||||
|         </div> |  | ||||||
|     </div> |     </div> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										93
									
								
								html/rfid.js
									
									
									
									
									
								
							
							
						
						| @@ -7,6 +7,7 @@ let heartbeatTimer = null; | |||||||
| let lastHeartbeatResponse = Date.now(); | let lastHeartbeatResponse = Date.now(); | ||||||
| const HEARTBEAT_TIMEOUT = 20000; | const HEARTBEAT_TIMEOUT = 20000; | ||||||
| let reconnectTimer = null; | let reconnectTimer = null; | ||||||
|  | let spoolDetected = false; | ||||||
|  |  | ||||||
| // WebSocket Funktionen | // WebSocket Funktionen | ||||||
| function startHeartbeat() { | function startHeartbeat() { | ||||||
| @@ -208,27 +209,13 @@ document.addEventListener('spoolmanError', function(event) { | |||||||
|     showNotification(`Spoolman Error: ${event.detail.message}`, false); |     showNotification(`Spoolman Error: ${event.detail.message}`, false); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| document.addEventListener('filamentSelected', function(event) { | document.addEventListener('filamentSelected', function (event) { | ||||||
|     updateNfcInfo(); |     updateNfcInfo(); | ||||||
|     // Zeige Spool-Buttons wenn ein Filament ausgewählt wurde |     // Zeige Spool-Buttons wenn ein Filament ausgewählt wurde | ||||||
|     const selectedText = document.getElementById("selected-filament").textContent; |     const selectedText = document.getElementById("selected-filament").textContent; | ||||||
|     updateSpoolButtons(selectedText !== "Please choose..."); |     updateSpoolButtons(selectedText !== "Please choose..."); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| // Hilfsfunktion für kontrastreiche Textfarbe |  | ||||||
| function getContrastColor(hexcolor) { |  | ||||||
|     // Konvertiere Hex zu RGB |  | ||||||
|     const r = parseInt(hexcolor.substr(0,2),16); |  | ||||||
|     const g = parseInt(hexcolor.substr(2,2),16); |  | ||||||
|     const b = parseInt(hexcolor.substr(4,2),16); |  | ||||||
|      |  | ||||||
|     // Berechne Helligkeit (YIQ Formel) |  | ||||||
|     const yiq = ((r*299)+(g*587)+(b*114))/1000; |  | ||||||
|      |  | ||||||
|     // Return schwarz oder weiß basierend auf Helligkeit |  | ||||||
|     return (yiq >= 128) ? '#000000' : '#FFFFFF'; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function updateNfcInfo() { | function updateNfcInfo() { | ||||||
|     const selectedText = document.getElementById("selected-filament").textContent; |     const selectedText = document.getElementById("selected-filament").textContent; | ||||||
|     const nfcInfo = document.getElementById("nfcInfo"); |     const nfcInfo = document.getElementById("nfcInfo"); | ||||||
| @@ -490,7 +477,7 @@ function handleSpoolIn(amsId, trayId) { | |||||||
|             nozzle_temp_max: parseInt(maxTemp), |             nozzle_temp_max: parseInt(maxTemp), | ||||||
|             type: selectedSpool.filament.material, |             type: selectedSpool.filament.material, | ||||||
|             brand: selectedSpool.filament.vendor.name, |             brand: selectedSpool.filament.vendor.name, | ||||||
|             tray_info_idx: selectedSpool.filament.extra.bambu_idx.replace(/['"]+/g, '').trim(), |             tray_info_idx: selectedSpool.filament.extra.bambu_idx?.replace(/['"]+/g, '').trim() || '', | ||||||
|             cali_idx: "-1"  // Default-Wert setzen |             cali_idx: "-1"  // Default-Wert setzen | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
| @@ -522,12 +509,15 @@ function updateNfcStatusIndicator(data) { | |||||||
|     if (data.found === 0) { |     if (data.found === 0) { | ||||||
|         // Kein NFC Tag gefunden |         // Kein NFC Tag gefunden | ||||||
|         indicator.className = 'status-circle'; |         indicator.className = 'status-circle'; | ||||||
|  |         spoolDetected = false; | ||||||
|     } else if (data.found === 1) { |     } else if (data.found === 1) { | ||||||
|         // NFC Tag erfolgreich gelesen |         // NFC Tag erfolgreich gelesen | ||||||
|         indicator.className = 'status-circle success'; |         indicator.className = 'status-circle success'; | ||||||
|  |         spoolDetected = true; | ||||||
|     } else { |     } else { | ||||||
|         // Fehler beim Lesen |         // Fehler beim Lesen | ||||||
|         indicator.className = 'status-circle error'; |         indicator.className = 'status-circle error'; | ||||||
|  |         spoolDetected = true; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -569,7 +559,10 @@ function updateNfcData(data) { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     // HTML für die Datenanzeige erstellen |     // HTML für die Datenanzeige erstellen | ||||||
|     let html = ` |     let html = ""; | ||||||
|  |  | ||||||
|  |     if(data.sm_id){ | ||||||
|  |         html = ` | ||||||
|         <div class="nfc-card-data" style="margin-top: 10px;"> |         <div class="nfc-card-data" style="margin-top: 10px;"> | ||||||
|             <p><strong>Brand:</strong> ${data.brand || 'N/A'}</p> |             <p><strong>Brand:</strong> ${data.brand || 'N/A'}</p> | ||||||
|             <p><strong>Type:</strong> ${data.type || 'N/A'} ${data.color_hex ? `<span style=" |             <p><strong>Type:</strong> ${data.type || 'N/A'} ${data.color_hex ? `<span style=" | ||||||
| @@ -585,7 +578,24 @@ function updateNfcData(data) { | |||||||
|         `; |         `; | ||||||
|  |  | ||||||
|         // Spoolman ID anzeigen |         // Spoolman ID anzeigen | ||||||
|     html += `<p><strong>Spoolman ID:</strong> ${data.sm_id || 'No Spoolman ID'}</p>`; |         html += `<p><strong>Spoolman ID:</strong> ${data.sm_id} (<a href="${spoolmanUrl}/spool/show/${data.sm_id}">Open in Spoolman</a>)</p>`; | ||||||
|  |      } | ||||||
|  |      else if(data.location) | ||||||
|  |      { | ||||||
|  |         html = ` | ||||||
|  |         <div class="nfc-card-data" style="margin-top: 10px;"> | ||||||
|  |             <p><strong>Location:</strong> ${data.location || 'N/A'}</p> | ||||||
|  |         `; | ||||||
|  |      } | ||||||
|  |      else | ||||||
|  |      { | ||||||
|  |         html = ` | ||||||
|  |         <div class="nfc-card-data" style="margin-top: 10px;"> | ||||||
|  |             <p><strong>Unknown tag</strong></p> | ||||||
|  |         `; | ||||||
|  |      } | ||||||
|  |  | ||||||
|  |      | ||||||
|  |  | ||||||
|     // Nur wenn eine sm_id vorhanden ist, aktualisiere die Dropdowns |     // Nur wenn eine sm_id vorhanden ist, aktualisiere die Dropdowns | ||||||
|     if (data.sm_id) { |     if (data.sm_id) { | ||||||
| @@ -612,6 +622,7 @@ function updateNfcData(data) { | |||||||
| } | } | ||||||
|  |  | ||||||
| function writeNfcTag() { | function writeNfcTag() { | ||||||
|  |     if(!spoolDetected || confirm("Are you sure you want to overwrite the Tag?") == true){ | ||||||
|         const selectedText = document.getElementById("selected-filament").textContent; |         const selectedText = document.getElementById("selected-filament").textContent; | ||||||
|         if (selectedText === "Please choose...") { |         if (selectedText === "Please choose...") { | ||||||
|             alert('Please select a Spool first.'); |             alert('Please select a Spool first.'); | ||||||
| @@ -654,15 +665,47 @@ function writeNfcTag() { | |||||||
|             writeButton.textContent = "Writing"; |             writeButton.textContent = "Writing"; | ||||||
|             socket.send(JSON.stringify({ |             socket.send(JSON.stringify({ | ||||||
|                 type: 'writeNfcTag', |                 type: 'writeNfcTag', | ||||||
|  |                 tagType: 'spool', | ||||||
|                 payload: nfcData |                 payload: nfcData | ||||||
|             })); |             })); | ||||||
|         } else { |         } else { | ||||||
|             alert('Not connected to Server. Please check connection.'); |             alert('Not connected to Server. Please check connection.'); | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function writeLocationNfcTag() { | ||||||
|  |     if(!spoolDetected || confirm("Are you sure you want to overwrite the Tag?") == true){ | ||||||
|  |         const selectedText = document.getElementById("locationSelect").value; | ||||||
|  |         if (selectedText === "Please choose...") { | ||||||
|  |             alert('Please select a location first.'); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         // Erstelle das NFC-Datenpaket mit korrekten Datentypen | ||||||
|  |         const nfcData = { | ||||||
|  |             location: String(selectedText) | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         if (socket?.readyState === WebSocket.OPEN) { | ||||||
|  |             const writeButton = document.getElementById("writeLocationNfcButton"); | ||||||
|  |             writeButton.classList.add("writing"); | ||||||
|  |             writeButton.textContent = "Writing"; | ||||||
|  |             socket.send(JSON.stringify({ | ||||||
|  |                 type: 'writeNfcTag', | ||||||
|  |                 tagType: 'location', | ||||||
|  |                 payload: nfcData | ||||||
|  |             })); | ||||||
|  |         } else { | ||||||
|  |             alert('Not connected to Server. Please check connection.'); | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| function handleWriteNfcTagResponse(success) { | function handleWriteNfcTagResponse(success) { | ||||||
|     const writeButton = document.getElementById("writeNfcButton"); |     const writeButton = document.getElementById("writeNfcButton"); | ||||||
|  |     const writeLocationButton = document.getElementById("writeLocationNfcButton"); | ||||||
|  |     if(writeButton.classList.contains("writing")){ | ||||||
|         writeButton.classList.remove("writing"); |         writeButton.classList.remove("writing"); | ||||||
|         writeButton.classList.add(success ? "success" : "error"); |         writeButton.classList.add(success ? "success" : "error"); | ||||||
|         writeButton.textContent = success ? "Write success" : "Write failed"; |         writeButton.textContent = success ? "Write success" : "Write failed"; | ||||||
| @@ -671,6 +714,20 @@ function handleWriteNfcTagResponse(success) { | |||||||
|             writeButton.classList.remove("success", "error"); |             writeButton.classList.remove("success", "error"); | ||||||
|             writeButton.textContent = "Write Tag"; |             writeButton.textContent = "Write Tag"; | ||||||
|         }, 5000); |         }, 5000); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if(writeLocationButton.classList.contains("writing")){ | ||||||
|  |         writeLocationButton.classList.remove("writing"); | ||||||
|  |         writeLocationButton.classList.add(success ? "success" : "error"); | ||||||
|  |         writeLocationButton.textContent = success ? "Write success" : "Write failed"; | ||||||
|  |  | ||||||
|  |         setTimeout(() => { | ||||||
|  |             writeLocationButton.classList.remove("success", "error"); | ||||||
|  |             writeLocationButton.textContent = "Write Location Tag"; | ||||||
|  |         }, 5000); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |      | ||||||
| } | } | ||||||
|  |  | ||||||
| function showNotification(message, isSuccess) { | function showNotification(message, isSuccess) { | ||||||
|   | |||||||
							
								
								
									
										172
									
								
								html/rfid_bambu.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,172 @@ | |||||||
|  | <!-- head --><!DOCTYPE html> | ||||||
|  | <html lang="en"> | ||||||
|  | <head> | ||||||
|  |     <meta charset="UTF-8"> | ||||||
|  |     <meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||||||
|  |     <title>FilaMan - Filament Management Tool</title> | ||||||
|  |     <link rel="icon" type="image/png" href="/favicon.ico"> | ||||||
|  |     <link rel="stylesheet" href="style.css"> | ||||||
|  |     <script> | ||||||
|  |         fetch('/api/version') | ||||||
|  |             .then(response => response.json()) | ||||||
|  |             .then(data => { | ||||||
|  |                 const versionSpan = document.querySelector('.version'); | ||||||
|  |                 if (versionSpan) { | ||||||
|  |                     versionSpan.textContent = 'v' + data.version; | ||||||
|  |                 } | ||||||
|  |             }) | ||||||
|  |             .catch(error => console.error('Error fetching version:', error)); | ||||||
|  |     </script> | ||||||
|  | </head> | ||||||
|  | <body> | ||||||
|  |     <div class="navbar"> | ||||||
|  |         <div style="display: flex; align-items: center; gap: 2rem;"> | ||||||
|  |             <img src="/logo.png" alt="FilaMan Logo" class="logo"> | ||||||
|  |             <div class="logo-text"> | ||||||
|  |                 <h1>FilaMan<span class="version"></span></h1> | ||||||
|  |                 <h4>Filament Management Tool</h4> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |         <nav style="display: flex; gap: 1rem;"> | ||||||
|  |             <a href="/">Start</a> | ||||||
|  |             <a href="/waage">Scale</a> | ||||||
|  |             <a href="/spoolman">Spoolman/Bambu</a> | ||||||
|  |             <a href="/about">About</a> | ||||||
|  |             <a href="/upgrade">Upgrade</a> | ||||||
|  |         </nav> | ||||||
|  |         <div class="status-container"> | ||||||
|  |             <div class="status-item"> | ||||||
|  |                 <span class="status-dot" id="bambuDot"></span>B | ||||||
|  |             </div> | ||||||
|  |             <div class="status-item"> | ||||||
|  |                 <span class="status-dot" id="spoolmanDot"></span>S | ||||||
|  |             </div> | ||||||
|  |             <div class="ram-status" id="ramStatus"></div> | ||||||
|  |         </div> | ||||||
|  |     </div> | ||||||
|  |  | ||||||
|  | <!-- head --> | ||||||
|  |  | ||||||
|  | <div class="connection-status hidden"> | ||||||
|  |     <div class="spinner"></div> | ||||||
|  |     <span>Connection lost. Trying to reconnect...</span> | ||||||
|  | </div> | ||||||
|  | <div class="content"> | ||||||
|  |     <div class="three-column-layout"> | ||||||
|  |         <!-- Linke Spalte --> | ||||||
|  |         <div class="column"> | ||||||
|  |             <div class="feature-box"> | ||||||
|  |                 <div class="statistics-header"> | ||||||
|  |                     <h2>Statistics</h2> | ||||||
|  |                     <button id="refreshSpoolman" class="refresh-button"> | ||||||
|  |                         <span>Refresh Spoolman</span> | ||||||
|  |                     </button> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="statistics-column"> | ||||||
|  |                     <h3>Spools</h3> | ||||||
|  |                         <div class="spool-stat" style="display: flex; justify-content: center; align-items: center;"> | ||||||
|  |                             <span class="stat-label">total:</span> | ||||||
|  |                             <span class="stat-value" id="totalSpools"></span> | ||||||
|  |                             <div style="width: auto;"></div> | ||||||
|  |                             <span class="stat-label">without Tag:</span> | ||||||
|  |                             <span class="stat-value" id="spoolsWithoutTag"></span> | ||||||
|  |                         </div> | ||||||
|  |                 </div> | ||||||
|  |  | ||||||
|  |                 <div class="statistics-grid"> | ||||||
|  |                     <div class="statistics-column"> | ||||||
|  |                         <h3>Overview</h3> | ||||||
|  |                         <ul class="statistics-list"> | ||||||
|  |                             <li> | ||||||
|  |                                 <span class="stat-label">Manufacturer:</span> | ||||||
|  |                                 <span class="stat-value" id="totalVendors"></span> | ||||||
|  |                             </li> | ||||||
|  |                             <li> | ||||||
|  |                                 <span class="stat-label">Weight:</span> | ||||||
|  |                                 <span class="stat-value"><span id="totalWeight"></span></span> | ||||||
|  |                             </li> | ||||||
|  |                             <li> | ||||||
|  |                                 <span class="stat-label">Length:</span> | ||||||
|  |                                 <span class="stat-value"><span id="totalLength"></span></span> | ||||||
|  |                             </li> | ||||||
|  |                         </ul> | ||||||
|  |                     </div> | ||||||
|  |                     <div class="statistics-column"> | ||||||
|  |                         <h3>Materials</h3> | ||||||
|  |                         <ul class="statistics-list" id="materialsList"> | ||||||
|  |                             <!-- Wird dynamisch befüllt --> | ||||||
|  |                         </ul> | ||||||
|  |                     </div> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |             <div class="feature-box"> | ||||||
|  |                 <div class="nfc-header"> | ||||||
|  |                     <h2>NFC-Tag</h2> | ||||||
|  |                     <span id="nfcStatusIndicator" class="status-circle"></span> | ||||||
|  |                 </div> | ||||||
|  |                 <div class="nfc-status-display"></div> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|  |         <!-- Mittlere Spalte --> | ||||||
|  |         <div class="column"> | ||||||
|  |             <div class="feature-box"> | ||||||
|  |                 <h2>Spoolman Spools</h2> | ||||||
|  |                 <label for="vendorSelect">Manufacturer:</label> | ||||||
|  |                 <div style="display: flex; justify-content: space-between; align-items: center;"> | ||||||
|  |                     <select id="vendorSelect" class="styled-select"> | ||||||
|  |                         <option value="">Please choose...</option> | ||||||
|  |                     </select> | ||||||
|  |                     <label style="margin-left: 10px;"> | ||||||
|  |                         <input type="checkbox" id="onlyWithoutSmId" checked onchange="updateFilamentDropdown()"> | ||||||
|  |                         Only Spools without SM ID | ||||||
|  |                     </label> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |  | ||||||
|  |             <div id="filamentSection" class="feature-box hidden"> | ||||||
|  |                 <label>Spool / Filament:</label> | ||||||
|  |                 <div class="custom-dropdown"> | ||||||
|  |                     <div class="dropdown-button" onclick="toggleFilamentDropdown()"> | ||||||
|  |                         <div class="selected-color" id="selected-color"></div> | ||||||
|  |                         <span id="selected-filament">Please choose...</span> | ||||||
|  |                         <span class="dropdown-arrow">▼</span> | ||||||
|  |                     </div> | ||||||
|  |                     <div class="dropdown-content" id="filament-dropdown-content"> | ||||||
|  |                         <!-- Optionen werden dynamisch hinzugefügt --> | ||||||
|  |                     </div> | ||||||
|  |                 </div> | ||||||
|  |                 <p id="nfcInfo" class="nfc-status"></p> | ||||||
|  |                 <button id="writeNfcButton" class="btn btn-primary hidden" onclick="writeNfcTag()">Write Tag</button> | ||||||
|  |             </div> | ||||||
|  |  | ||||||
|  |             <div class="feature-box"> | ||||||
|  |                 <h2>Spoolman Locations</h2> | ||||||
|  |                 <label for="locationSelect">Location:</label> | ||||||
|  |                 <div style="display: flex; justify-content: space-between; align-items: center;"> | ||||||
|  |                     <select id="locationSelect" class="styled-select"> | ||||||
|  |                         <option value="">Please choose...</option> | ||||||
|  |                     </select> | ||||||
|  |                 </div> | ||||||
|  |                 <p id="nfcInfoLocation" class="nfc-status"></p> | ||||||
|  |                 <button id="writeLocationNfcButton" class="btn btn-primary hidden" onclick="writeLocationNfcTag()">Write Location Tag</button> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|  |         <!-- Rechte Spalte --> | ||||||
|  |         <div class="column"> | ||||||
|  |             <div class="feature-box"> | ||||||
|  |                 <h2>Bambu AMS</h2> | ||||||
|  |                 <div id="amsDataContainer"> | ||||||
|  |                     <div class="amsData" id="amsData">Wait for AMS-Data...</div> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |     </div> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | <script src="spoolman.js"></script> | ||||||
|  | <script src="rfid.js"></script> | ||||||
|  |  | ||||||
|  | </body> | ||||||
|  | </html> | ||||||
| @@ -57,6 +57,31 @@ | |||||||
|             toggleOctoFields(); |             toggleOctoFields(); | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|  |         function removeBambuCredentials() { | ||||||
|  |             fetch('/api/bambu?remove=true') | ||||||
|  |                 .then(response => response.json()) | ||||||
|  |                 .then(data => { | ||||||
|  |                     if (data.success) { | ||||||
|  |                         document.getElementById('bambuIp').value = ''; | ||||||
|  |                         document.getElementById('bambuSerial').value = ''; | ||||||
|  |                         document.getElementById('bambuCode').value = ''; | ||||||
|  |                         document.getElementById('autoSend').checked = false; | ||||||
|  |                         document.getElementById('autoSendTime').value = ''; | ||||||
|  |                         document.getElementById('bambuStatusMessage').innerText = 'Bambu Credentials removed!'; | ||||||
|  |                         // Reload with forced cache refresh after short delay | ||||||
|  |                         setTimeout(() => { | ||||||
|  |                             window.location.reload(true); | ||||||
|  |                             window.location.href = '/'; | ||||||
|  |                         }, 1500); | ||||||
|  |                     } else { | ||||||
|  |                         document.getElementById('bambuStatusMessage').innerText = 'Error while removing Bambu Credentials.'; | ||||||
|  |                     } | ||||||
|  |                 }) | ||||||
|  |                 .catch(error => { | ||||||
|  |                     document.getElementById('bambuStatusMessage').innerText = 'Error while removing: ' + error.message; | ||||||
|  |                 }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         function checkSpoolmanInstance() { |         function checkSpoolmanInstance() { | ||||||
|             const url = document.getElementById('spoolmanUrl').value; |             const url = document.getElementById('spoolmanUrl').value; | ||||||
|             const spoolmanOctoEnabled = document.getElementById('spoolmanOctoEnabled').checked; |             const spoolmanOctoEnabled = document.getElementById('spoolmanOctoEnabled').checked; | ||||||
| @@ -89,6 +114,11 @@ | |||||||
|                 .then(data => { |                 .then(data => { | ||||||
|                     if (data.healthy) { |                     if (data.healthy) { | ||||||
|                         document.getElementById('bambuStatusMessage').innerText = 'Bambu Credentials saved!'; |                         document.getElementById('bambuStatusMessage').innerText = 'Bambu Credentials saved!'; | ||||||
|  |                         // Reload with forced cache refresh after short delay | ||||||
|  |                         setTimeout(() => { | ||||||
|  |                             window.location.reload(true); | ||||||
|  |                             window.location.href = '/'; | ||||||
|  |                         }, 1500); | ||||||
|                     } else { |                     } else { | ||||||
|                         document.getElementById('bambuStatusMessage').innerText = 'Error while saving Bambu Credentials.'; |                         document.getElementById('bambuStatusMessage').innerText = 'Error while saving Bambu Credentials.'; | ||||||
|                     } |                     } | ||||||
| @@ -116,20 +146,20 @@ | |||||||
|  |  | ||||||
|         <div class="card"> |         <div class="card"> | ||||||
|             <div class="card-body"> |             <div class="card-body"> | ||||||
|                 <h5 class="card-title">Set URL/IP to your Spoolman-Instanz</h5> |                 <h5 class="card-title">Set URL/IP to your Spoolman instance</h5> | ||||||
|                 <input type="text" id="spoolmanUrl" placeholder="http://ip-or-url-of-your-spoolman-instanz:port"> |                 <input type="text" id="spoolmanUrl" onkeydown="if(event.keyCode == 13) document.getElementById('btnSaveSpoolmanUrl').click()" placeholder="http://ip-or-url-of-your-spoolman-instance:port"> | ||||||
|                 <h5 class="card-title">If you want to enable sending Spool to Spoolman Octoprint Plugin:</h5> |                 <h5 class="card-title">If you want to enable sending the spool to the Spoolman Octoprint plugin:</h5> | ||||||
|                 <p> |                 <p> | ||||||
|                     <input type="checkbox" id="spoolmanOctoEnabled" {{spoolmanOctoEnabled}} onchange="toggleOctoFields()"> Send to Octo-Plugin |                     <input type="checkbox" id="spoolmanOctoEnabled" {{spoolmanOctoEnabled}} onchange="toggleOctoFields()"> Send to Octo-Plugin | ||||||
|                 </p> |                 </p> | ||||||
|                 <div id="octoFields" style="display: none;"> |                 <div id="octoFields" style="display: none;"> | ||||||
|                     <p> |                     <p> | ||||||
|                         <input type="text" id="spoolmanOctoUrl" placeholder="http://ip-or-url-of-your-octoprint-instanz:port" value="{{spoolmanOctoUrl}}"> |                         <input type="text" id="spoolmanOctoUrl" placeholder="http://ip-or-url-of-your-octoprint-instance:port" value="{{spoolmanOctoUrl}}"> | ||||||
|                         <input type="text" id="spoolmanOctoToken" placeholder="Your Octoprint Token" value="{{spoolmanOctoToken}}"> |                         <input type="text" id="spoolmanOctoToken" placeholder="Your Octoprint Token" value="{{spoolmanOctoToken}}"> | ||||||
|                     </p> |                     </p> | ||||||
|                 </div> |                 </div> | ||||||
|                  |                  | ||||||
|                 <button onclick="checkSpoolmanInstance()">Save Spoolman URL</button> |                 <button id="btnSaveSpoolmanUrl" onclick="checkSpoolmanInstance()">Save Spoolman URL</button> | ||||||
|                 <p id="statusMessage"></p> |                 <p id="statusMessage"></p> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
| @@ -139,16 +169,16 @@ | |||||||
|                 <h5 class="card-title">Bambu Lab Printer Credentials</h5> |                 <h5 class="card-title">Bambu Lab Printer Credentials</h5> | ||||||
|                 <div class="bambu-settings"> |                 <div class="bambu-settings"> | ||||||
|                     <div class="input-group"> |                     <div class="input-group"> | ||||||
|                         <label for="bambuIp">Bambu Drucker IP-Adresse:</label> |                         <label for="bambuIp">Bambu Printer IP Address:</label> | ||||||
|                         <input type="text" id="bambuIp" placeholder="192.168.1.xxx" value="{{bambuIp}}"> |                         <input type="text" id="bambuIp" placeholder="192.168.1.xxx" value="{{bambuIp}}"> | ||||||
|                     </div> |                     </div> | ||||||
|                     <div class="input-group"> |                     <div class="input-group"> | ||||||
|                         <label for="bambuSerial">Drucker Seriennummer:</label> |                         <label for="bambuSerial">Printer Serial Number:</label> | ||||||
|                         <input type="text" id="bambuSerial" placeholder="BBLXXXXXXXX" value="{{bambuSerial}}"> |                         <input type="text" id="bambuSerial" placeholder="BBLXXXXXXXX" value="{{bambuSerial}}"> | ||||||
|                     </div> |                     </div> | ||||||
|                     <div class="input-group"> |                     <div class="input-group"> | ||||||
|                         <label for="bambuCode">Access Code:</label> |                         <label for="bambuCode">Access Code:</label> | ||||||
|                         <input type="text" id="bambuCode" placeholder="Access Code vom Drucker" value="{{bambuCode}}"> |                         <input type="text" id="bambuCode" placeholder="Access Code of the printer" value="{{bambuCode}}"> | ||||||
|                     </div> |                     </div> | ||||||
|                     <hr> |                     <hr> | ||||||
|                     <p>If activated, FilaMan will automatically update the next filled tray with the last scanned and weighed spool.</p> |                     <p>If activated, FilaMan will automatically update the next filled tray with the last scanned and weighed spool.</p> | ||||||
| @@ -162,6 +192,7 @@ | |||||||
|                     </div> |                     </div> | ||||||
|  |  | ||||||
|                     <button style="margin: 0;" onclick="saveBambuCredentials()">Save Bambu Credentials</button> |                     <button style="margin: 0;" onclick="saveBambuCredentials()">Save Bambu Credentials</button> | ||||||
|  |                     <button style="margin: 0; background-color: red;" onclick="removeBambuCredentials()">Remove Credentials</button> | ||||||
|                     <p id="bambuStatusMessage"></p> |                     <p id="bambuStatusMessage"></p> | ||||||
|                 </div> |                 </div> | ||||||
|             </div> |             </div> | ||||||
|   | |||||||
							
								
								
									
										137
									
								
								html/spoolman.js
									
									
									
									
									
								
							
							
						
						| @@ -1,6 +1,7 @@ | |||||||
| // Globale Variablen | // Globale Variablen | ||||||
| let spoolmanUrl = ''; | let spoolmanUrl = ''; | ||||||
| let spoolsData = []; | let spoolsData = []; | ||||||
|  | let locationData = []; | ||||||
|  |  | ||||||
| // Hilfsfunktionen für Datenmanipulation | // Hilfsfunktionen für Datenmanipulation | ||||||
| function processSpoolData(data) { | function processSpoolData(data) { | ||||||
| @@ -86,10 +87,10 @@ function populateVendorDropdown(data, selectedSmId = null) { | |||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     // Nach der Schleife: Formatierung der Gesamtlänge |     // Nach der Schleife: Formatierung der Gesamtlänge | ||||||
|     console.log("Total Length: ", totalLength); |     const lengthInM = totalLength / 1000;  // erst in m umrechnen | ||||||
|     const formattedLength = totalLength > 1000  |     const formattedLength = lengthInM > 1000  | ||||||
|         ? (totalLength / 1000).toFixed(2) + " km"  |         ? (lengthInM / 1000).toFixed(2) + " km"  | ||||||
|         : totalLength.toFixed(2) + " m"; |         : lengthInM.toFixed(2) + " m"; | ||||||
|  |  | ||||||
|     // Formatierung des Gesamtgewichts (von g zu kg zu t) |     // Formatierung des Gesamtgewichts (von g zu kg zu t) | ||||||
|     const weightInKg = totalWeight / 1000;  // erst in kg umrechnen |     const weightInKg = totalWeight / 1000;  // erst in kg umrechnen | ||||||
| @@ -133,6 +134,26 @@ function populateVendorDropdown(data, selectedSmId = null) { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // Dropdown-Funktionen | ||||||
|  | function populateLocationDropdown(data) { | ||||||
|  |     const locationSelect = document.getElementById("locationSelect"); | ||||||
|  |     if (!locationSelect) { | ||||||
|  |         console.error('locationSelect Element nicht gefunden'); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     locationSelect.innerHTML = '<option value="">Bitte wählen...</option>'; | ||||||
|  |     // Dropdown mit gefilterten Herstellern befüllen - alphabetisch sortiert | ||||||
|  |     Object.entries(data) | ||||||
|  |         .sort(([, nameA], [, nameB]) => nameA.localeCompare(nameB)) // Sort vendors alphabetically by name | ||||||
|  |         .forEach(([id, name]) => { | ||||||
|  |             const option = document.createElement("option"); | ||||||
|  |             option.value = name; | ||||||
|  |             option.textContent = name; | ||||||
|  |             locationSelect.appendChild(option); | ||||||
|  |         }); | ||||||
|  | } | ||||||
|  |  | ||||||
| function updateFilamentDropdown(selectedSmId = null) { | function updateFilamentDropdown(selectedSmId = null) { | ||||||
|     const vendorId = document.getElementById("vendorSelect").value; |     const vendorId = document.getElementById("vendorSelect").value; | ||||||
|     const dropdownContentInner = document.getElementById("filament-dropdown-content"); |     const dropdownContentInner = document.getElementById("filament-dropdown-content"); | ||||||
| @@ -147,6 +168,13 @@ function updateFilamentDropdown(selectedSmId = null) { | |||||||
|  |  | ||||||
|     if (vendorId) { |     if (vendorId) { | ||||||
|         const filteredFilaments = spoolsData.filter(spool => { |         const filteredFilaments = spoolsData.filter(spool => { | ||||||
|  |             if (!spool?.filament?.vendor?.id) { | ||||||
|  |                 console.log('Problem aufgetreten bei: ', spool?.filament?.vendor); | ||||||
|  |                 console.log('Problematische Spulen:',  | ||||||
|  |                     spoolsData.filter(spool => !spool?.filament?.vendor?.id)); | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |  | ||||||
|             const hasValidNfcId = spool.extra &&  |             const hasValidNfcId = spool.extra &&  | ||||||
|                                  spool.extra.nfc_id &&  |                                  spool.extra.nfc_id &&  | ||||||
|                                  spool.extra.nfc_id !== '""' &&  |                                  spool.extra.nfc_id !== '""' &&  | ||||||
| @@ -162,9 +190,32 @@ function updateFilamentDropdown(selectedSmId = null) { | |||||||
|             option.setAttribute("data-value", spool.filament.id); |             option.setAttribute("data-value", spool.filament.id); | ||||||
|             option.setAttribute("data-nfc-id", spool.extra.nfc_id || ""); |             option.setAttribute("data-nfc-id", spool.extra.nfc_id || ""); | ||||||
|              |              | ||||||
|  |  | ||||||
|  |             // Generate color representation based on filament type (single or multi color) | ||||||
|  |             let colorHTML = ''; | ||||||
|  |              | ||||||
|  |             // Check if this is a multicolor filament | ||||||
|  |             if (spool.filament.multi_color_hexes) { | ||||||
|  |                 // Parse multi color hexes from comma-separated string | ||||||
|  |                 const colors = spool.filament.multi_color_hexes.replace(/#/g, '').split(','); | ||||||
|  |                  | ||||||
|  |                 // Determine the display style based on direction | ||||||
|  |                 const direction = spool.filament.multi_color_direction || 'coaxial'; | ||||||
|  |                  | ||||||
|  |                 // Generate color circles for each color | ||||||
|  |                 colorHTML = '<div class="option-colors">'; | ||||||
|  |                 colors.forEach(color => { | ||||||
|  |                     colorHTML += `<div class="option-color multi-color ${direction}" style="background-color: #${color}"></div>`; | ||||||
|  |                 }); | ||||||
|  |                 colorHTML += '</div>'; | ||||||
|  |             } else { | ||||||
|  |                 // Single color filament | ||||||
|                 const colorHex = spool.filament.color_hex || 'FFFFFF'; |                 const colorHex = spool.filament.color_hex || 'FFFFFF'; | ||||||
|  |                 colorHTML = `<div class="option-color" style="background-color: #${colorHex}"></div>`; | ||||||
|  |             } | ||||||
|  |              | ||||||
|             option.innerHTML = ` |             option.innerHTML = ` | ||||||
|                 <div class="option-color" style="background-color: #${colorHex}"></div> |                 ${colorHTML} | ||||||
|                 <span>${spool.id} | ${spool.filament.name} (${spool.filament.material})</span> |                 <span>${spool.id} | ${spool.filament.name} (${spool.filament.material})</span> | ||||||
|             `; |             `; | ||||||
|              |              | ||||||
| @@ -178,12 +229,41 @@ function updateFilamentDropdown(selectedSmId = null) { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | function updateLocationSelect(){ | ||||||
|  |     const writeLocationNfcButton = document.getElementById('writeLocationNfcButton'); | ||||||
|  |     if(writeLocationNfcButton){ | ||||||
|  |         writeLocationNfcButton.classList.remove("hidden"); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| function selectFilament(spool) { | function selectFilament(spool) { | ||||||
|     const selectedColor = document.getElementById("selected-color"); |     const selectedColor = document.getElementById("selected-color"); | ||||||
|     const selectedText = document.getElementById("selected-filament"); |     const selectedText = document.getElementById("selected-filament"); | ||||||
|     const dropdownContent = document.getElementById("filament-dropdown-content"); |     const dropdownContent = document.getElementById("filament-dropdown-content"); | ||||||
|      |      | ||||||
|  |     // Update the selected color display | ||||||
|  |     if (spool.filament.multi_color_hexes) { | ||||||
|  |         // Handle multicolor filament display in the selection header | ||||||
|  |         const colors = spool.filament.multi_color_hexes.replace(/#/g, '').split(','); | ||||||
|  |         const direction = spool.filament.multi_color_direction || 'coaxial'; | ||||||
|  |          | ||||||
|  |         // Replace the single color div with multiple color divs | ||||||
|  |         selectedColor.innerHTML = ''; | ||||||
|  |         colors.forEach(color => { | ||||||
|  |             const colorDiv = document.createElement('div'); | ||||||
|  |             colorDiv.className = `color-segment multi-color ${direction}`; | ||||||
|  |             colorDiv.style.backgroundColor = `#${color}`; | ||||||
|  |             selectedColor.appendChild(colorDiv); | ||||||
|  |         }); | ||||||
|  |         // Add multiple color class to the container | ||||||
|  |         selectedColor.classList.add('multi-color-container'); | ||||||
|  |     } else { | ||||||
|  |         // Single color filament - reset to default display | ||||||
|  |         selectedColor.innerHTML = ''; | ||||||
|  |         selectedColor.classList.remove('multi-color-container'); | ||||||
|         selectedColor.style.backgroundColor = `#${spool.filament.color_hex || 'FFFFFF'}`; |         selectedColor.style.backgroundColor = `#${spool.filament.color_hex || 'FFFFFF'}`; | ||||||
|  |     } | ||||||
|  |      | ||||||
|     selectedText.textContent = `${spool.id} | ${spool.filament.name} (${spool.filament.material})`; |     selectedText.textContent = `${spool.id} | ${spool.filament.name} (${spool.filament.material})`; | ||||||
|     dropdownContent.classList.remove("show"); |     dropdownContent.classList.remove("show"); | ||||||
|      |      | ||||||
| @@ -213,6 +293,14 @@ async function initSpoolman() { | |||||||
|         document.dispatchEvent(new CustomEvent('spoolDataLoaded', {  |         document.dispatchEvent(new CustomEvent('spoolDataLoaded', {  | ||||||
|             detail: spoolsData  |             detail: spoolsData  | ||||||
|         })); |         })); | ||||||
|  |          | ||||||
|  |         locationData = await fetchLocationData(); | ||||||
|  |          | ||||||
|  |         document.dispatchEvent(new CustomEvent('locationDataLoaded', {  | ||||||
|  |             detail: locationData  | ||||||
|  |         })); | ||||||
|  |  | ||||||
|  |  | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|         console.error('Fehler beim Initialisieren von Spoolman:', error); |         console.error('Fehler beim Initialisieren von Spoolman:', error); | ||||||
|         document.dispatchEvent(new CustomEvent('spoolmanError', {  |         document.dispatchEvent(new CustomEvent('spoolmanError', {  | ||||||
| @@ -240,17 +328,24 @@ async function fetchSpoolData() { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| /* | async function fetchLocationData() { | ||||||
| // Exportiere Funktionen |     try { | ||||||
| window.getSpoolData = () => spoolsData; |         if (!spoolmanUrl) { | ||||||
| window.reloadSpoolData = initSpoolman; |             throw new Error('Spoolman URL ist nicht initialisiert'); | ||||||
| window.populateVendorDropdown = populateVendorDropdown; |         } | ||||||
| window.updateFilamentDropdown = updateFilamentDropdown; |          | ||||||
| window.toggleFilamentDropdown = () => { |         const response = await fetch(`${spoolmanUrl}/api/v1/location`); | ||||||
|     const content = document.getElementById("filament-dropdown-content"); |         if (!response.ok) { | ||||||
|     content.classList.toggle("show"); |             throw new Error(`HTTP error! status: ${response.status}`); | ||||||
| }; |         } | ||||||
| */ |          | ||||||
|  |         const data = await response.json(); | ||||||
|  |         return data; | ||||||
|  |     } catch (error) { | ||||||
|  |         console.error('Fehler beim Abrufen der Location-Daten:', error); | ||||||
|  |         return []; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| // Event Listener | // Event Listener | ||||||
| document.addEventListener('DOMContentLoaded', () => { | document.addEventListener('DOMContentLoaded', () => { | ||||||
| @@ -261,6 +356,11 @@ document.addEventListener('DOMContentLoaded', () => { | |||||||
|         vendorSelect.addEventListener('change', () => updateFilamentDropdown()); |         vendorSelect.addEventListener('change', () => updateFilamentDropdown()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     const locationSelect = document.getElementById('locationSelect'); | ||||||
|  |     if (locationSelect) { | ||||||
|  |         locationSelect.addEventListener('change', () => updateLocationSelect()); | ||||||
|  |     } | ||||||
|  |      | ||||||
|     const onlyWithoutSmId = document.getElementById('onlyWithoutSmId'); |     const onlyWithoutSmId = document.getElementById('onlyWithoutSmId'); | ||||||
|     if (onlyWithoutSmId) { |     if (onlyWithoutSmId) { | ||||||
|         onlyWithoutSmId.addEventListener('change', () => { |         onlyWithoutSmId.addEventListener('change', () => { | ||||||
| @@ -273,6 +373,10 @@ document.addEventListener('DOMContentLoaded', () => { | |||||||
|         populateVendorDropdown(event.detail); |         populateVendorDropdown(event.detail); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|  |     document.addEventListener('locationDataLoaded', (event) => { | ||||||
|  |         populateLocationDropdown(event.detail); | ||||||
|  |     }); | ||||||
|  |      | ||||||
|     window.onclick = function(event) { |     window.onclick = function(event) { | ||||||
|         if (!event.target.closest('.custom-dropdown')) { |         if (!event.target.closest('.custom-dropdown')) { | ||||||
|             const dropdowns = document.getElementsByClassName("dropdown-content"); |             const dropdowns = document.getElementsByClassName("dropdown-content"); | ||||||
| @@ -302,6 +406,7 @@ window.getSpoolData = () => spoolsData; | |||||||
| window.setSpoolData = (data) => { spoolsData = data; }; | window.setSpoolData = (data) => { spoolsData = data; }; | ||||||
| window.reloadSpoolData = initSpoolman; | window.reloadSpoolData = initSpoolman; | ||||||
| window.populateVendorDropdown = populateVendorDropdown; | window.populateVendorDropdown = populateVendorDropdown; | ||||||
|  | window.populateLocationDropdown = populateLocationDropdown; | ||||||
| window.updateFilamentDropdown = updateFilamentDropdown; | window.updateFilamentDropdown = updateFilamentDropdown; | ||||||
| window.toggleFilamentDropdown = () => { | window.toggleFilamentDropdown = () => { | ||||||
|     const content = document.getElementById("filament-dropdown-content"); |     const content = document.getElementById("filament-dropdown-content"); | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								html/spoolman_url.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | {"url": "http://192.168.1.5:7912", "octoEnabled": true, "octoUrl": "http://192.168.1.17:5001", "octoToken": "O5zZ58mXRAyeGpVEj2ZZj-UPAPqJ2N7JgtD36mw1M4g"} | ||||||
| @@ -759,6 +759,50 @@ a:hover { | |||||||
|     flex-shrink: 0; |     flex-shrink: 0; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /* Multi-color filament styles */ | ||||||
|  | .option-colors { | ||||||
|  |     display: flex; | ||||||
|  |     flex-shrink: 0; | ||||||
|  |     gap: 2px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .multi-color { | ||||||
|  |     width: 14px; | ||||||
|  |     height: 14px; | ||||||
|  |     border-radius: 50%; | ||||||
|  |     border: 1px solid #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Coaxial pattern (horizontal stripes) */ | ||||||
|  | .multi-color.coaxial { | ||||||
|  |     border-radius: 50%; | ||||||
|  |     position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Longitudinal pattern (vertical stripes) */ | ||||||
|  | .multi-color.longitudinal { | ||||||
|  |     border-radius: 50%; | ||||||
|  |     position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Container for multiple colors in selected display */ | ||||||
|  | .multi-color-container { | ||||||
|  |     display: flex !important; | ||||||
|  |     background: none !important; | ||||||
|  |     border: none !important; | ||||||
|  |     gap: 2px; | ||||||
|  |     align-items: center; | ||||||
|  |     justify-content: flex-start; | ||||||
|  |     width: auto !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .color-segment { | ||||||
|  |     width: 16px; | ||||||
|  |     height: 16px; | ||||||
|  |     border-radius: 50%; | ||||||
|  |     border: 1px solid #333; | ||||||
|  | } | ||||||
|  |  | ||||||
| .notification { | .notification { | ||||||
|     position: fixed; |     position: fixed; | ||||||
|     top: 20px; |     top: 20px; | ||||||
| @@ -927,31 +971,35 @@ input[type="submit"]:disabled, | |||||||
| } | } | ||||||
|  |  | ||||||
| /* Schreib-Button */ | /* Schreib-Button */ | ||||||
| #writeNfcButton { | #writeNfcButton, #writeLocationNfcButton { | ||||||
|     background-color: #007bff; |     background-color: #007bff; | ||||||
|     color: white; |     color: white; | ||||||
|     transition: background-color 0.3s, color 0.3s; |     transition: background-color 0.3s, color 0.3s; | ||||||
|     width: 160px; |     width: 160px; | ||||||
| } | } | ||||||
|  |  | ||||||
| #writeNfcButton.writing { | #writeNfcButton.writing, #writeLocationNfcButton.writing { | ||||||
|     background-color: #ffc107; |     background-color: #ffc107; | ||||||
|     color: black; |     color: black; | ||||||
|     width: 160px; |     width: 160px; | ||||||
| } | } | ||||||
|  |  | ||||||
| #writeNfcButton.success { | #writeNfcButton.success, #writeLocationNfcButton.success { | ||||||
|     background-color: #28a745; |     background-color: #28a745; | ||||||
|     color: white; |     color: white; | ||||||
|     width: 160px; |     width: 160px; | ||||||
| } | } | ||||||
|  |  | ||||||
| #writeNfcButton.error { | #writeNfcButton.error, #writeLocationNfcButton.error { | ||||||
|     background-color: #dc3545; |     background-color: #dc3545; | ||||||
|     color: white; |     color: white; | ||||||
|     width: 160px; |     width: 160px; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | #writeLocationNfcButton{ | ||||||
|  |     width: 250px; | ||||||
|  | } | ||||||
|  |  | ||||||
| @keyframes dots { | @keyframes dots { | ||||||
|     0% { content: ""; } |     0% { content: ""; } | ||||||
|     33% { content: "."; } |     33% { content: "."; } | ||||||
| @@ -959,7 +1007,7 @@ input[type="submit"]:disabled, | |||||||
|     100% { content: "..."; } |     100% { content: "..."; } | ||||||
| } | } | ||||||
|  |  | ||||||
| #writeNfcButton.writing::after { | #writeNfcButton.writing::after, #writeLocationNfcButton.writing::after { | ||||||
|     content: "..."; |     content: "..."; | ||||||
|     animation: dots 1s steps(3, end) infinite; |     animation: dots 1s steps(3, end) infinite; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -56,8 +56,8 @@ | |||||||
|  |  | ||||||
|         <div class="update-options"> |         <div class="update-options"> | ||||||
|             <div class="update-section"> |             <div class="update-section"> | ||||||
|                 <h2>Firmware Update</h2> |                 <h2>1) Firmware Update</h2> | ||||||
|                 <p>Upload a new firmware file (filaman_*.bin)</p> |                 <p>Upload a new firmware file (upgrade_filaman_firmware_*.bin)</p> | ||||||
|                 <div class="update-form"> |                 <div class="update-form"> | ||||||
|                     <form id="firmwareForm" enctype='multipart/form-data' data-type="firmware"> |                     <form id="firmwareForm" enctype='multipart/form-data' data-type="firmware"> | ||||||
|                         <input type='file' name='update' accept='.bin' required> |                         <input type='file' name='update' accept='.bin' required> | ||||||
| @@ -67,8 +67,8 @@ | |||||||
|             </div> |             </div> | ||||||
|  |  | ||||||
|             <div class="update-section"> |             <div class="update-section"> | ||||||
|                 <h2>Webpage Update</h2> |                 <h2>2) Webpage Update</h2> | ||||||
|                 <p>Upload a new webpage file (webpage_*.bin)</p> |                 <p>Upload a new webpage file (upgrade_filaman_website_*.bin)</p> | ||||||
|                 <div class="update-form"> |                 <div class="update-form"> | ||||||
|                     <form id="webpageForm" enctype='multipart/form-data' data-type="webpage"> |                     <form id="webpageForm" enctype='multipart/form-data' data-type="webpage"> | ||||||
|                         <input type='file' name='update' accept='.bin' required> |                         <input type='file' name='update' accept='.bin' required> | ||||||
| @@ -129,6 +129,7 @@ | |||||||
|                             if (data.status === 'success' || lastReceivedProgress >= 98) { |                             if (data.status === 'success' || lastReceivedProgress >= 98) { | ||||||
|                                 clearTimeout(wsReconnectTimer); |                                 clearTimeout(wsReconnectTimer); | ||||||
|                                 setTimeout(() => { |                                 setTimeout(() => { | ||||||
|  |                                     window.location.reload(true); | ||||||
|                                     window.location.href = '/'; |                                     window.location.href = '/'; | ||||||
|                                 }, 30000); |                                 }, 30000); | ||||||
|                             } |                             } | ||||||
| @@ -148,6 +149,7 @@ | |||||||
|                         status.style.display = 'block'; |                         status.style.display = 'block'; | ||||||
|                         clearTimeout(wsReconnectTimer); |                         clearTimeout(wsReconnectTimer); | ||||||
|                         setTimeout(() => { |                         setTimeout(() => { | ||||||
|  |                             window.location.reload(true); | ||||||
|                             window.location.href = '/'; |                             window.location.href = '/'; | ||||||
|                         }, 30000); |                         }, 30000); | ||||||
|                     } else { |                     } else { | ||||||
|   | |||||||
| @@ -55,6 +55,7 @@ | |||||||
|                 <h5 class="card-title">Sacle Calibration</h5> |                 <h5 class="card-title">Sacle Calibration</h5> | ||||||
|                 <button id="calibrateBtn" class="btn btn-primary">Calibrate Scale</button> |                 <button id="calibrateBtn" class="btn btn-primary">Calibrate Scale</button> | ||||||
|                 <button id="tareBtn" class="btn btn-secondary">Tare Scale</button> |                 <button id="tareBtn" class="btn btn-secondary">Tare Scale</button> | ||||||
|  |                    Enable Auto-TARE <input type="checkbox" id="autoTareCheckbox" onchange="setAutoTare(this.checked);" {{autoTare}}> | ||||||
|                 <div id="statusMessage" class="mt-3"></div> |                 <div id="statusMessage" class="mt-3"></div> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
| @@ -140,6 +141,15 @@ | |||||||
|             })); |             })); | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
|  |         // Add auto-tare function | ||||||
|  |         function setAutoTare(enabled) { | ||||||
|  |             ws.send(JSON.stringify({ | ||||||
|  |                 type: 'scale', | ||||||
|  |                 payload: 'setAutoTare', | ||||||
|  |                 enabled: enabled | ||||||
|  |             })); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         // WebSocket-Verbindung beim Laden der Seite initiieren |         // WebSocket-Verbindung beim Laden der Seite initiieren | ||||||
|         connectWebSocket(); |         connectWebSocket(); | ||||||
|     </script> |     </script> | ||||||
|   | |||||||
| @@ -9,8 +9,8 @@ | |||||||
| ; https://docs.platformio.org/page/projectconf.html | ; https://docs.platformio.org/page/projectconf.html | ||||||
|  |  | ||||||
| [common] | [common] | ||||||
| version = "1.4.0" | version = "1.5.12-beta5" | ||||||
| to_old_version = "1.4.0" | to_old_version = "1.5.0" | ||||||
|  |  | ||||||
| ## | ## | ||||||
| [env:esp32dev] | [env:esp32dev] | ||||||
| @@ -18,14 +18,12 @@ platform = espressif32 | |||||||
| board = esp32dev | board = esp32dev | ||||||
| framework = arduino | framework = arduino | ||||||
| monitor_speed = 115200 | monitor_speed = 115200 | ||||||
|  | #monitor_port = /dev/cu.usbmodem01 | ||||||
|  |  | ||||||
| lib_deps = | lib_deps = | ||||||
|     tzapu/WiFiManager @ ^2.0.17 |     tzapu/WiFiManager @ ^2.0.17 | ||||||
|     https://github.com/me-no-dev/ESPAsyncWebServer.git#master |     https://github.com/me-no-dev/ESPAsyncWebServer.git#master | ||||||
|     #me-no-dev/AsyncTCP @ ^1.1.1 |  | ||||||
|     https://github.com/esphome/AsyncTCP.git |     https://github.com/esphome/AsyncTCP.git | ||||||
|     #mathieucarbou/ESPAsyncWebServer @ ^3.6.0 |  | ||||||
|     #esp32async/AsyncTCP @ ^3.3.5 |  | ||||||
|     bogde/HX711 @ ^0.7.5 |     bogde/HX711 @ ^0.7.5 | ||||||
|     adafruit/Adafruit SSD1306 @ ^2.5.13 |     adafruit/Adafruit SSD1306 @ ^2.5.13 | ||||||
|     adafruit/Adafruit GFX Library @ ^1.11.11 |     adafruit/Adafruit GFX Library @ ^1.11.11 | ||||||
| @@ -35,7 +33,6 @@ lib_deps = | |||||||
|     digitaldragon/SSLClient @ ^1.3.2 |     digitaldragon/SSLClient @ ^1.3.2 | ||||||
|      |      | ||||||
| ; Enable SPIFFS upload | ; Enable SPIFFS upload | ||||||
| #board_build.filesystem = spiffs |  | ||||||
| board_build.filesystem = littlefs | board_build.filesystem = littlefs | ||||||
| ; Update partition settings | ; Update partition settings | ||||||
| board_build.partitions = partitions.csv | board_build.partitions = partitions.csv | ||||||
| @@ -51,6 +48,7 @@ build_flags = | |||||||
|     -mtext-section-literals |     -mtext-section-literals | ||||||
|     -DVERSION=\"${common.version}\" |     -DVERSION=\"${common.version}\" | ||||||
|     -DTOOLDVERSION=\"${common.to_old_version}\" |     -DTOOLDVERSION=\"${common.to_old_version}\" | ||||||
|  |     #-DENABLE_HEAP_DEBUGGING | ||||||
|     -DASYNCWEBSERVER_REGEX |     -DASYNCWEBSERVER_REGEX | ||||||
|     #-DCORE_DEBUG_LEVEL=3 |     #-DCORE_DEBUG_LEVEL=3 | ||||||
|     -DCONFIG_ARDUHAL_LOG_COLORS=1 |     -DCONFIG_ARDUHAL_LOG_COLORS=1 | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ def copy_file(input_file, output_file): | |||||||
|  |  | ||||||
| def should_compress(file): | def should_compress(file): | ||||||
|      # Skip compression for spoolman.html |      # Skip compression for spoolman.html | ||||||
|     if file == 'spoolman.html': |     if file == 'spoolman.html' or file == 'waage.html': | ||||||
|         return False |         return False | ||||||
|     # Komprimiere nur bestimmte Dateitypen |     # Komprimiere nur bestimmte Dateitypen | ||||||
|     return file.endswith(('.js', '.png', '.css', '.html')) |     return file.endswith(('.js', '.png', '.css', '.html')) | ||||||
|   | |||||||
							
								
								
									
										825
									
								
								src/api.cpp
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										25
									
								
								src/api.h
									
									
									
									
									
								
							
							
						
						| @@ -6,22 +6,45 @@ | |||||||
| #include "website.h" | #include "website.h" | ||||||
| #include "display.h" | #include "display.h" | ||||||
| #include <ArduinoJson.h> | #include <ArduinoJson.h> | ||||||
|  | typedef enum { | ||||||
|  |     API_INIT, | ||||||
|  |     API_IDLE, | ||||||
|  |     API_TRANSMITTING | ||||||
|  | } spoolmanApiStateType; | ||||||
|  |  | ||||||
|  | typedef enum { | ||||||
|  |     API_REQUEST_OCTO_SPOOL_UPDATE, | ||||||
|  |     API_REQUEST_BAMBU_UPDATE, | ||||||
|  |     API_REQUEST_SPOOL_TAG_ID_UPDATE, | ||||||
|  |     API_REQUEST_SPOOL_WEIGHT_UPDATE, | ||||||
|  |     API_REQUEST_SPOOL_LOCATION_UPDATE, | ||||||
|  |     API_REQUEST_VENDOR_CREATE, | ||||||
|  |     API_REQUEST_VENDOR_CHECK, | ||||||
|  |     API_REQUEST_FILAMENT_CHECK, | ||||||
|  |     API_REQUEST_FILAMENT_CREATE, | ||||||
|  |     API_REQUEST_SPOOL_CREATE | ||||||
|  | } SpoolmanApiRequestType; | ||||||
|  |  | ||||||
|  | extern volatile spoolmanApiStateType spoolmanApiState; | ||||||
| extern bool spoolman_connected; | extern bool spoolman_connected; | ||||||
| extern String spoolmanUrl; | extern String spoolmanUrl; | ||||||
| extern bool octoEnabled; | extern bool octoEnabled; | ||||||
|  | extern bool sendOctoUpdate; | ||||||
| extern String octoUrl; | extern String octoUrl; | ||||||
| extern String octoToken; | extern String octoToken; | ||||||
|  | extern bool spoolmanConnected; | ||||||
|  |  | ||||||
| bool checkSpoolmanInstance(const String& url); | bool checkSpoolmanInstance(); | ||||||
| bool saveSpoolmanUrl(const String& url, bool octoOn, const String& octoWh, const String& octoTk); | bool saveSpoolmanUrl(const String& url, bool octoOn, const String& octoWh, const String& octoTk); | ||||||
| String loadSpoolmanUrl(); // Neue Funktion zum Laden der URL | String loadSpoolmanUrl(); // Neue Funktion zum Laden der URL | ||||||
| bool checkSpoolmanExtraFields(); // Neue Funktion zum Überprüfen der Extrafelder | bool checkSpoolmanExtraFields(); // Neue Funktion zum Überprüfen der Extrafelder | ||||||
| JsonDocument fetchSingleSpoolInfo(int spoolId); // API-Funktion für die Webseite | JsonDocument fetchSingleSpoolInfo(int spoolId); // API-Funktion für die Webseite | ||||||
| bool updateSpoolTagId(String uidString, const char* payload); // Neue Funktion zum Aktualisieren eines Spools | bool updateSpoolTagId(String uidString, const char* payload); // Neue Funktion zum Aktualisieren eines Spools | ||||||
| uint8_t updateSpoolWeight(String spoolId, uint16_t weight); // Neue Funktion zum Aktualisieren des Gewichts | uint8_t updateSpoolWeight(String spoolId, uint16_t weight); // Neue Funktion zum Aktualisieren des Gewichts | ||||||
|  | uint8_t updateSpoolLocation(String spoolId, String location); | ||||||
| bool initSpoolman(); // Neue Funktion zum Initialisieren von Spoolman | bool initSpoolman(); // Neue Funktion zum Initialisieren von Spoolman | ||||||
| bool updateSpoolBambuData(String payload); // Neue Funktion zum Aktualisieren der Bambu-Daten | bool updateSpoolBambuData(String payload); // Neue Funktion zum Aktualisieren der Bambu-Daten | ||||||
| bool updateSpoolOcto(int spoolId); // Neue Funktion zum Aktualisieren der Octo-Daten | bool updateSpoolOcto(int spoolId); // Neue Funktion zum Aktualisieren der Octo-Daten | ||||||
|  | bool createBrandFilament(JsonDocument& payload, String uidString); | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|   | |||||||
							
								
								
									
										167
									
								
								src/bambu.cpp
									
									
									
									
									
								
							
							
						
						| @@ -10,6 +10,7 @@ | |||||||
| #include "esp_task_wdt.h" | #include "esp_task_wdt.h" | ||||||
| #include "config.h" | #include "config.h" | ||||||
| #include "display.h" | #include "display.h" | ||||||
|  | #include <Preferences.h> | ||||||
|  |  | ||||||
| WiFiClient espClient; | WiFiClient espClient; | ||||||
| SSLClient sslClient(&espClient); | SSLClient sslClient(&espClient); | ||||||
| @@ -17,49 +18,69 @@ PubSubClient client(sslClient); | |||||||
|  |  | ||||||
| TaskHandle_t BambuMqttTask; | TaskHandle_t BambuMqttTask; | ||||||
|  |  | ||||||
| String report_topic = ""; | bool bambuDisabled = false; | ||||||
| //String request_topic = ""; |  | ||||||
| const char* bambu_username = "bblp"; |  | ||||||
| const char* bambu_ip = nullptr; |  | ||||||
| const char* bambu_accesscode = nullptr; |  | ||||||
| const char* bambu_serialnr = nullptr; |  | ||||||
|  |  | ||||||
| String g_bambu_ip = ""; |  | ||||||
| String g_bambu_accesscode = ""; |  | ||||||
| String g_bambu_serialnr = ""; |  | ||||||
|  |  | ||||||
| bool bambu_connected = false; | bool bambu_connected = false; | ||||||
| bool autoSendToBambu = false; |  | ||||||
| int autoSetToBambuSpoolId = 0; | int autoSetToBambuSpoolId = 0; | ||||||
|  |  | ||||||
|  | BambuCredentials bambuCredentials; | ||||||
|  |  | ||||||
| // Globale Variablen für AMS-Daten | // Globale Variablen für AMS-Daten | ||||||
| int ams_count = 0; | int ams_count = 0; | ||||||
| String amsJsonData;  // Speichert das fertige JSON für WebSocket-Clients | String amsJsonData;  // Speichert das fertige JSON für WebSocket-Clients | ||||||
| AMSData ams_data[MAX_AMS];  // Definition des Arrays; | AMSData ams_data[MAX_AMS];  // Definition des Arrays; | ||||||
|  |  | ||||||
|  | bool removeBambuCredentials() { | ||||||
|  |     if (BambuMqttTask) { | ||||||
|  |         vTaskDelete(BambuMqttTask); | ||||||
|  |         BambuMqttTask = NULL; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     Preferences preferences; | ||||||
|  |     preferences.begin(NVS_NAMESPACE_BAMBU, false); // false = readwrite | ||||||
|  |     preferences.remove(NVS_KEY_BAMBU_IP); | ||||||
|  |     preferences.remove(NVS_KEY_BAMBU_SERIAL); | ||||||
|  |     preferences.remove(NVS_KEY_BAMBU_ACCESSCODE); | ||||||
|  |     preferences.remove(NVS_KEY_BAMBU_AUTOSEND_ENABLE); | ||||||
|  |     preferences.remove(NVS_KEY_BAMBU_AUTOSEND_TIME); | ||||||
|  |     preferences.end(); | ||||||
|  |  | ||||||
|  |     // Löschen der globalen Variablen | ||||||
|  |     bambuCredentials.ip = ""; | ||||||
|  |     bambuCredentials.serial = ""; | ||||||
|  |     bambuCredentials.accesscode = ""; | ||||||
|  |     bambuCredentials.autosend_enable = false; | ||||||
|  |     bambuCredentials.autosend_time = BAMBU_DEFAULT_AUTOSEND_TIME; | ||||||
|  |  | ||||||
|  |     autoSetToBambuSpoolId = 0; | ||||||
|  |     ams_count = 0; | ||||||
|  |     amsJsonData = ""; | ||||||
|  |  | ||||||
|  |     bambuDisabled = true; | ||||||
|  |  | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  |  | ||||||
| bool saveBambuCredentials(const String& ip, const String& serialnr, const String& accesscode, bool autoSend, const String& autoSendTime) { | bool saveBambuCredentials(const String& ip, const String& serialnr, const String& accesscode, bool autoSend, const String& autoSendTime) { | ||||||
|     if (BambuMqttTask) { |     if (BambuMqttTask) { | ||||||
|         vTaskDelete(BambuMqttTask); |         vTaskDelete(BambuMqttTask); | ||||||
|  |         BambuMqttTask = NULL; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     JsonDocument doc; |     bambuCredentials.ip = ip.c_str(); | ||||||
|     doc["bambu_ip"] = ip; |     bambuCredentials.serial = serialnr.c_str(); | ||||||
|     doc["bambu_accesscode"] = accesscode; |     bambuCredentials.accesscode = accesscode.c_str(); | ||||||
|     doc["bambu_serialnr"] = serialnr; |     bambuCredentials.autosend_enable = autoSend; | ||||||
|     doc["autoSendToBambu"] = autoSend; |     bambuCredentials.autosend_time = autoSendTime.toInt(); | ||||||
|     doc["autoSendTime"] = (autoSendTime != "") ? autoSendTime.toInt() : autoSetBambuAmsCounter; |  | ||||||
|  |  | ||||||
|     if (!saveJsonValue("/bambu_credentials.json", doc)) { |     Preferences preferences; | ||||||
|         Serial.println("Fehler beim Speichern der Bambu-Credentials."); |     preferences.begin(NVS_NAMESPACE_BAMBU, false); // false = readwrite | ||||||
|         return false; |     preferences.putString(NVS_KEY_BAMBU_IP, bambuCredentials.ip); | ||||||
|     } |     preferences.putString(NVS_KEY_BAMBU_SERIAL, bambuCredentials.serial); | ||||||
|  |     preferences.putString(NVS_KEY_BAMBU_ACCESSCODE, bambuCredentials.accesscode); | ||||||
|     // Dynamische Speicherallokation für die globalen Pointer |     preferences.putBool(NVS_KEY_BAMBU_AUTOSEND_ENABLE, bambuCredentials.autosend_enable); | ||||||
|     bambu_ip = ip.c_str(); |     preferences.putInt(NVS_KEY_BAMBU_AUTOSEND_TIME, bambuCredentials.autosend_time); | ||||||
|     bambu_accesscode = accesscode.c_str(); |     preferences.end(); | ||||||
|     bambu_serialnr = serialnr.c_str(); |  | ||||||
|     autoSendToBambu = autoSend; |  | ||||||
|     autoSetBambuAmsCounter = autoSendTime.toInt(); |  | ||||||
|  |  | ||||||
|     vTaskDelay(100 / portTICK_PERIOD_MS); |     vTaskDelay(100 / portTICK_PERIOD_MS); | ||||||
|     if (!setupMqtt()) return false; |     if (!setupMqtt()) return false; | ||||||
| @@ -68,35 +89,36 @@ bool saveBambuCredentials(const String& ip, const String& serialnr, const String | |||||||
| } | } | ||||||
|  |  | ||||||
| bool loadBambuCredentials() { | bool loadBambuCredentials() { | ||||||
|     JsonDocument doc; |     Preferences preferences; | ||||||
|     if (loadJsonValue("/bambu_credentials.json", doc) && doc["bambu_ip"].is<String>()) { |     preferences.begin(NVS_NAMESPACE_BAMBU, true); | ||||||
|         // Temporäre Strings für die Werte |     String ip = preferences.getString(NVS_KEY_BAMBU_IP, ""); | ||||||
|         String ip = doc["bambu_ip"].as<String>(); |     String serial = preferences.getString(NVS_KEY_BAMBU_SERIAL, ""); | ||||||
|         String code = doc["bambu_accesscode"].as<String>(); |     String code = preferences.getString(NVS_KEY_BAMBU_ACCESSCODE, ""); | ||||||
|         String serial = doc["bambu_serialnr"].as<String>(); |     bool autosendEnable = preferences.getBool(NVS_KEY_BAMBU_AUTOSEND_ENABLE, false); | ||||||
|  |     int autosendTime = preferences.getInt(NVS_KEY_BAMBU_AUTOSEND_TIME, BAMBU_DEFAULT_AUTOSEND_TIME); | ||||||
|  |     preferences.end(); | ||||||
|  |  | ||||||
|         g_bambu_ip = ip; |     if(ip != ""){ | ||||||
|         g_bambu_accesscode = code; |         bambuCredentials.ip = ip.c_str(); | ||||||
|         g_bambu_serialnr = serial; |         bambuCredentials.serial = serial.c_str(); | ||||||
|  |         bambuCredentials.accesscode = code.c_str(); | ||||||
|  |         bambuCredentials.autosend_enable = autosendEnable; | ||||||
|  |         bambuCredentials.autosend_time = autosendTime; | ||||||
|  |  | ||||||
|         if (doc["autoSendToBambu"].is<bool>()) autoSendToBambu = doc["autoSendToBambu"].as<bool>(); |         Serial.println("credentials loaded loadCredentials!"); | ||||||
|         if (doc["autoSendTime"].is<int>()) autoSetBambuAmsCounter = doc["autoSendTime"].as<int>(); |         Serial.println(bambuCredentials.ip); | ||||||
|  |         Serial.println(bambuCredentials.serial); | ||||||
|  |         Serial.println(bambuCredentials.accesscode); | ||||||
|  |         Serial.println(String(bambuCredentials.autosend_enable)); | ||||||
|  |         Serial.println(String(bambuCredentials.autosend_time)); | ||||||
|  |  | ||||||
|         ip.trim(); |  | ||||||
|         code.trim(); |  | ||||||
|         serial.trim(); |  | ||||||
|  |  | ||||||
|         // Dynamische Speicherallokation für die globalen Pointer |  | ||||||
|         bambu_ip = g_bambu_ip.c_str(); |  | ||||||
|         bambu_accesscode = g_bambu_accesscode.c_str(); |  | ||||||
|         bambu_serialnr = g_bambu_serialnr.c_str(); |  | ||||||
|  |  | ||||||
|         report_topic = "device/" + String(bambu_serialnr) + "/report"; |  | ||||||
|         //request_topic = "device/" + String(bambu_serialnr) + "/request"; |  | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |     else | ||||||
|  |     { | ||||||
|         Serial.println("Keine gültigen Bambu-Credentials gefunden."); |         Serial.println("Keine gültigen Bambu-Credentials gefunden."); | ||||||
|         return false; |         return false; | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| struct FilamentResult { | struct FilamentResult { | ||||||
| @@ -199,7 +221,7 @@ FilamentResult findFilamentIdx(String brand, String type) { | |||||||
| bool sendMqttMessage(const String& payload) { | bool sendMqttMessage(const String& payload) { | ||||||
|     Serial.println("Sending MQTT message"); |     Serial.println("Sending MQTT message"); | ||||||
|     Serial.println(payload); |     Serial.println(payload); | ||||||
|     if (client.publish(report_topic.c_str(), payload.c_str()))  |     if (client.publish(("device/"+bambuCredentials.serial+"/request").c_str(), payload.c_str()))  | ||||||
|     { |     { | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| @@ -472,7 +494,7 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | |||||||
|                     trayObj["cali_idx"].as<String>() != ams_data[storedIndex].trays[j].cali_idx) { |                     trayObj["cali_idx"].as<String>() != ams_data[storedIndex].trays[j].cali_idx) { | ||||||
|                     hasChanges = true; |                     hasChanges = true; | ||||||
|  |  | ||||||
|                     if (autoSendToBambu && autoSetToBambuSpoolId > 0 && hasChanges) |                     if (bambuCredentials.autosend_enable && autoSetToBambuSpoolId > 0 && hasChanges) | ||||||
|                     { |                     { | ||||||
|                         autoSetSpool(autoSetToBambuSpoolId, ams_data[storedIndex].trays[j].id); |                         autoSetSpool(autoSetToBambuSpoolId, ams_data[storedIndex].trays[j].id); | ||||||
|                     } |                     } | ||||||
| @@ -496,7 +518,7 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | |||||||
|                         (vtTray["tray_type"].as<String>() != "" && vtTray["cali_idx"].as<String>() != ams_data[i].trays[0].cali_idx)) { |                         (vtTray["tray_type"].as<String>() != "" && vtTray["cali_idx"].as<String>() != ams_data[i].trays[0].cali_idx)) { | ||||||
|                         hasChanges = true; |                         hasChanges = true; | ||||||
|  |  | ||||||
|                         if (autoSendToBambu && autoSetToBambuSpoolId > 0 && hasChanges) |                         if (bambuCredentials.autosend_enable && autoSetToBambuSpoolId > 0 && hasChanges) | ||||||
|                         { |                         { | ||||||
|                             autoSetSpool(autoSetToBambuSpoolId, 254); |                             autoSetSpool(autoSetToBambuSpoolId, 254); | ||||||
|                         } |                         } | ||||||
| @@ -553,10 +575,11 @@ void reconnect() { | |||||||
|         oledShowTopRow(); |         oledShowTopRow(); | ||||||
|  |  | ||||||
|         // Attempt to connect |         // Attempt to connect | ||||||
|         if (client.connect(bambu_serialnr, bambu_username, bambu_accesscode)) { |         String clientId = bambuCredentials.serial + "_" + String(random(0, 100)); | ||||||
|  |         if (client.connect(clientId.c_str(), BAMBU_USERNAME, bambuCredentials.accesscode.c_str())) { | ||||||
|             Serial.println("MQTT re/connected"); |             Serial.println("MQTT re/connected"); | ||||||
|  |  | ||||||
|             client.subscribe(report_topic.c_str()); |             client.subscribe(("device/"+bambuCredentials.serial+"/report").c_str()); | ||||||
|             bambu_connected = true; |             bambu_connected = true; | ||||||
|             oledShowTopRow(); |             oledShowTopRow(); | ||||||
|         } else { |         } else { | ||||||
| @@ -572,6 +595,7 @@ void reconnect() { | |||||||
|                 Serial.println("Disable Bambu MQTT Task after 5 retries"); |                 Serial.println("Disable Bambu MQTT Task after 5 retries"); | ||||||
|                 //vTaskSuspend(BambuMqttTask); |                 //vTaskSuspend(BambuMqttTask); | ||||||
|                 vTaskDelete(BambuMqttTask); |                 vTaskDelete(BambuMqttTask); | ||||||
|  |                 BambuMqttTask = NULL; | ||||||
|                 break; |                 break; | ||||||
|             } |             } | ||||||
|  |  | ||||||
| @@ -602,30 +626,24 @@ void mqtt_loop(void * parameter) { | |||||||
|  |  | ||||||
| bool setupMqtt() { | bool setupMqtt() { | ||||||
|     // Wenn Bambu Daten vorhanden |     // Wenn Bambu Daten vorhanden | ||||||
|     bool success = loadBambuCredentials(); |     //bool success = loadBambuCredentials(); | ||||||
|  |  | ||||||
|     if (!success) { |     if (bambuCredentials.ip != "" && bambuCredentials.accesscode != "" && bambuCredentials.serial != "")  | ||||||
|         Serial.println("Failed to load Bambu credentials"); |  | ||||||
|         oledShowMessage("Bambu Credentials Missing"); |  | ||||||
|         vTaskDelay(2000 / portTICK_PERIOD_MS); |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (success && bambu_ip != "" && bambu_accesscode != "" && bambu_serialnr != "")  |  | ||||||
|     { |     { | ||||||
|  |         oledShowProgressBar(4, 7, DISPLAY_BOOT_TEXT, "Bambu init"); | ||||||
|  |         bambuDisabled = false; | ||||||
|         sslClient.setCACert(root_ca); |         sslClient.setCACert(root_ca); | ||||||
|         sslClient.setInsecure(); |         sslClient.setInsecure(); | ||||||
|         client.setServer(bambu_ip, 8883); |         client.setServer(bambuCredentials.ip.c_str(), 8883); | ||||||
|  |  | ||||||
|         // Verbinden mit dem MQTT-Server |         // Verbinden mit dem MQTT-Server | ||||||
|         bool connected = true; |         bool connected = true; | ||||||
|         if (client.connect(bambu_serialnr, bambu_username, bambu_accesscode))  |         String clientId = String(bambuCredentials.serial) + "_" + String(random(0, 100)); | ||||||
|  |         if (client.connect(bambuCredentials.ip.c_str(), BAMBU_USERNAME, bambuCredentials.accesscode.c_str()))  | ||||||
|         { |         { | ||||||
|             client.setCallback(mqtt_callback); |             client.setCallback(mqtt_callback); | ||||||
|             client.setBufferSize(5120); |             client.setBufferSize(15488); | ||||||
|             // Optional: Topic abonnieren |             client.subscribe(("device/"+bambuCredentials.serial+"/report").c_str()); | ||||||
|             client.subscribe(report_topic.c_str()); |  | ||||||
|             //client.subscribe(request_topic.c_str()); |  | ||||||
|             Serial.println("MQTT-Client initialisiert"); |             Serial.println("MQTT-Client initialisiert"); | ||||||
|  |  | ||||||
|             oledShowMessage("Bambu Connected"); |             oledShowMessage("Bambu Connected"); | ||||||
| @@ -648,24 +666,25 @@ bool setupMqtt() { | |||||||
|             vTaskDelay(2000 / portTICK_PERIOD_MS); |             vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||||
|             connected = false; |             connected = false; | ||||||
|             oledShowTopRow(); |             oledShowTopRow(); | ||||||
|  |             autoSetToBambuSpoolId = 0; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (!connected) return false; |         if (!connected) return false; | ||||||
|     }  |     }  | ||||||
|     else  |     else  | ||||||
|     { |     { | ||||||
|         Serial.println("Fehler: Keine MQTT-Daten vorhanden"); |         bambuDisabled = true; | ||||||
|         oledShowMessage("Bambu Credentials Missing"); |  | ||||||
|         oledShowTopRow(); |  | ||||||
|         vTaskDelay(2000 / portTICK_PERIOD_MS); |  | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| void bambu_restart() { | void bambu_restart() { | ||||||
|  |     Serial.println("Bambu restart"); | ||||||
|  |  | ||||||
|     if (BambuMqttTask) { |     if (BambuMqttTask) { | ||||||
|         vTaskDelete(BambuMqttTask); |         vTaskDelete(BambuMqttTask); | ||||||
|  |         BambuMqttTask = NULL; | ||||||
|         delay(10); |         delay(10); | ||||||
|     } |     } | ||||||
|     setupMqtt(); |     setupMqtt(); | ||||||
|   | |||||||
							
								
								
									
										13
									
								
								src/bambu.h
									
									
									
									
									
								
							
							
						
						| @@ -16,6 +16,14 @@ struct TrayData { | |||||||
|     String cali_idx; |     String cali_idx; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | struct BambuCredentials { | ||||||
|  |     String ip; | ||||||
|  |     String serial; | ||||||
|  |     String accesscode; | ||||||
|  |     bool autosend_enable; | ||||||
|  |     int autosend_time; | ||||||
|  | }; | ||||||
|  |  | ||||||
| #define MAX_AMS 17  // 16 normale AMS + 1 externe Spule | #define MAX_AMS 17  // 16 normale AMS + 1 externe Spule | ||||||
| extern String amsJsonData;  // Für die vorbereiteten JSON-Daten | extern String amsJsonData;  // Für die vorbereiteten JSON-Daten | ||||||
|  |  | ||||||
| @@ -28,9 +36,12 @@ extern bool bambu_connected; | |||||||
|  |  | ||||||
| extern int ams_count; | extern int ams_count; | ||||||
| extern AMSData ams_data[MAX_AMS]; | extern AMSData ams_data[MAX_AMS]; | ||||||
| extern bool autoSendToBambu; | //extern bool autoSendToBambu; | ||||||
| extern int autoSetToBambuSpoolId; | extern int autoSetToBambuSpoolId; | ||||||
|  | extern bool bambuDisabled; | ||||||
|  | extern BambuCredentials bambuCredentials; | ||||||
|  |  | ||||||
|  | bool removeBambuCredentials(); | ||||||
| bool loadBambuCredentials(); | bool loadBambuCredentials(); | ||||||
| bool saveBambuCredentials(const String& bambu_ip, const String& bambu_serialnr, const String& bambu_accesscode, const bool autoSend, const String& autoSendTime); | bool saveBambuCredentials(const String& bambu_ip, const String& bambu_serialnr, const String& bambu_accesscode, const bool autoSend, const String& autoSendTime); | ||||||
| bool setupMqtt(); | bool setupMqtt(); | ||||||
|   | |||||||
| @@ -1,6 +1,20 @@ | |||||||
| #include "commonFS.h" | #include "commonFS.h" | ||||||
| #include <LittleFS.h> | #include <LittleFS.h> | ||||||
|  |  | ||||||
|  | bool removeJsonValue(const char* filename) { | ||||||
|  |     File file = LittleFS.open(filename, "r"); | ||||||
|  |     if (!file) { | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |     file.close(); | ||||||
|  |     if (!LittleFS.remove(filename)) { | ||||||
|  |         Serial.print("Fehler beim Löschen der Datei: "); | ||||||
|  |         Serial.println(filename); | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  |  | ||||||
| bool saveJsonValue(const char* filename, const JsonDocument& doc) { | bool saveJsonValue(const char* filename, const JsonDocument& doc) { | ||||||
|     File file = LittleFS.open(filename, "w"); |     File file = LittleFS.open(filename, "w"); | ||||||
|     if (!file) { |     if (!file) { | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ | |||||||
| #include <ArduinoJson.h> | #include <ArduinoJson.h> | ||||||
| #include <LittleFS.h> | #include <LittleFS.h> | ||||||
|  |  | ||||||
|  | bool removeJsonValue(const char* filename); | ||||||
| bool saveJsonValue(const char* filename, const JsonDocument& doc); | bool saveJsonValue(const char* filename, const JsonDocument& doc); | ||||||
| bool loadJsonValue(const char* filename, JsonDocument& doc); | bool loadJsonValue(const char* filename, JsonDocument& doc); | ||||||
| void initializeFileSystem(); | void initializeFileSystem(); | ||||||
|   | |||||||
| @@ -16,20 +16,20 @@ const uint8_t LOADCELL_DOUT_PIN = 16; //16; | |||||||
| const uint8_t LOADCELL_SCK_PIN = 17; //17; | const uint8_t LOADCELL_SCK_PIN = 17; //17; | ||||||
| const uint8_t calVal_eepromAdress = 0; | const uint8_t calVal_eepromAdress = 0; | ||||||
| const uint16_t SCALE_LEVEL_WEIGHT = 500; | const uint16_t SCALE_LEVEL_WEIGHT = 500; | ||||||
| uint16_t defaultScaleCalibrationValue = 430; |  | ||||||
| // ***** HX711 | // ***** HX711 | ||||||
|  |  | ||||||
|  | // ***** TTP223 (Touch Sensor) | ||||||
|  | // TTP223 circuit wiring | ||||||
|  | const uint8_t TTP223_PIN = 25; | ||||||
|  | // ***** TTP223 | ||||||
|  |  | ||||||
|  |  | ||||||
| // ***** Display | // ***** Display | ||||||
| // Declaration for an SSD1306 display connected to I2C (SDA, SCL pins) |  | ||||||
| // On an ESP32:   21(SDA),  22(SCL) |  | ||||||
| const int8_t OLED_RESET = -1; // Reset pin # (or -1 if sharing Arduino reset pin) |  | ||||||
| const uint8_t SCREEN_ADDRESS = 0x3C; ///< See datasheet for Address; 0x3D for 128x64, 0x3C for 128x32 |  | ||||||
| const uint8_t SCREEN_WIDTH = 128; // OLED display width, in pixels |  | ||||||
| const uint8_t SCREEN_HEIGHT = 64; // OLED display height, in pixels |  | ||||||
| const uint8_t OLED_TOP_START = 0; | const uint8_t OLED_TOP_START = 0; | ||||||
| const uint8_t OLED_TOP_END = 16; | const uint8_t OLED_TOP_END = 16; | ||||||
| const uint8_t OLED_DATA_START = 17; | const uint8_t OLED_DATA_START = 17; | ||||||
| const uint8_t OLED_DATA_END = SCREEN_HEIGHT; | const uint8_t OLED_DATA_END = SCREEN_HEIGHT; | ||||||
|  |  | ||||||
| // ***** Display | // ***** Display | ||||||
|  |  | ||||||
| // ***** Webserver | // ***** Webserver | ||||||
| @@ -40,8 +40,6 @@ const uint8_t webserverPort = 80; | |||||||
| const char* apiUrl = "/api/v1"; | const char* apiUrl = "/api/v1"; | ||||||
| // ***** API | // ***** API | ||||||
|  |  | ||||||
| // ***** Bambu Auto Set Spool |  | ||||||
| uint8_t autoSetBambuAmsCounter = 60; |  | ||||||
| // ***** Bambu Auto Set Spool | // ***** Bambu Auto Set Spool | ||||||
|  |  | ||||||
| // ***** Task Prios | // ***** Task Prios | ||||||
|   | |||||||
							
								
								
									
										42
									
								
								src/config.h
									
									
									
									
									
								
							
							
						
						| @@ -3,6 +3,40 @@ | |||||||
|  |  | ||||||
| #include <Arduino.h> | #include <Arduino.h> | ||||||
|  |  | ||||||
|  | #define BAMBU_DEFAULT_AUTOSEND_TIME         60 | ||||||
|  |  | ||||||
|  | #define NVS_NAMESPACE_API                   "api" | ||||||
|  | #define NVS_KEY_SPOOLMAN_URL                "spoolmanUrl" | ||||||
|  | #define NVS_KEY_OCTOPRINT_ENABLED           "octoEnabled" | ||||||
|  | #define NVS_KEY_OCTOPRINT_URL               "octoUrl" | ||||||
|  | #define NVS_KEY_OCTOPRINT_TOKEN             "octoToken" | ||||||
|  |  | ||||||
|  | #define NVS_NAMESPACE_BAMBU                 "bambu" | ||||||
|  | #define NVS_KEY_BAMBU_IP                    "bambuIp" | ||||||
|  | #define NVS_KEY_BAMBU_ACCESSCODE            "bambuCode" | ||||||
|  | #define NVS_KEY_BAMBU_SERIAL                "bambuSerial" | ||||||
|  | #define NVS_KEY_BAMBU_AUTOSEND_ENABLE       "autosendEnable" | ||||||
|  | #define NVS_KEY_BAMBU_AUTOSEND_TIME         "autosendTime" | ||||||
|  |  | ||||||
|  | #define NVS_NAMESPACE_SCALE                 "scale" | ||||||
|  | #define NVS_KEY_CALIBRATION                 "cal_value" | ||||||
|  | #define NVS_KEY_AUTOTARE                    "auto_tare" | ||||||
|  | #define SCALE_DEFAULT_CALIBRATION_VALUE     430.0f; | ||||||
|  |  | ||||||
|  | #define BAMBU_USERNAME                      "bblp" | ||||||
|  |  | ||||||
|  | #define OLED_RESET                          -1      // Reset pin # (or -1 if sharing Arduino reset pin) | ||||||
|  | #define SCREEN_ADDRESS                      0x3CU   // See datasheet for Address; 0x3D for 128x64, 0x3C for 128x32 | ||||||
|  | #define SCREEN_WIDTH                        128U | ||||||
|  | #define SCREEN_HEIGHT                       64U | ||||||
|  | #define SCREEN_TOP_BAR_HEIGHT               16U | ||||||
|  | #define SCREEN_PROGRESS_BAR_HEIGHT          12U | ||||||
|  | #define DISPLAY_BOOT_TEXT                   "FilaMan" | ||||||
|  |  | ||||||
|  | #define WIFI_CHECK_INTERVAL                 60000U | ||||||
|  | #define DISPLAY_UPDATE_INTERVAL             1000U | ||||||
|  | #define SPOOLMAN_HEALTHCHECK_INTERVAL       60000U | ||||||
|  |  | ||||||
| extern const uint8_t PN532_IRQ; | extern const uint8_t PN532_IRQ; | ||||||
| extern const uint8_t PN532_RESET; | extern const uint8_t PN532_RESET; | ||||||
|  |  | ||||||
| @@ -11,10 +45,8 @@ extern const uint8_t LOADCELL_SCK_PIN; | |||||||
| extern const uint8_t calVal_eepromAdress; | extern const uint8_t calVal_eepromAdress; | ||||||
| extern const uint16_t SCALE_LEVEL_WEIGHT; | extern const uint16_t SCALE_LEVEL_WEIGHT; | ||||||
|  |  | ||||||
| extern const int8_t OLED_RESET; | extern const uint8_t TTP223_PIN; | ||||||
| extern const uint8_t SCREEN_ADDRESS; |  | ||||||
| extern const uint8_t SCREEN_WIDTH; |  | ||||||
| extern const uint8_t SCREEN_HEIGHT; |  | ||||||
| extern const uint8_t OLED_TOP_START; | extern const uint8_t OLED_TOP_START; | ||||||
| extern const uint8_t OLED_TOP_END; | extern const uint8_t OLED_TOP_END; | ||||||
| extern const uint8_t OLED_DATA_START; | extern const uint8_t OLED_DATA_START; | ||||||
| @@ -23,7 +55,7 @@ extern const uint8_t OLED_DATA_END; | |||||||
| extern const char* apiUrl; | extern const char* apiUrl; | ||||||
| extern const uint8_t webserverPort; | extern const uint8_t webserverPort; | ||||||
|  |  | ||||||
| extern uint8_t autoSetBambuAmsCounter; |  | ||||||
|  |  | ||||||
| extern const unsigned char wifi_on[]; | extern const unsigned char wifi_on[]; | ||||||
| extern const unsigned char wifi_off[]; | extern const unsigned char wifi_off[]; | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								src/debug.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | |||||||
|  | #include <Arduino.h> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifdef ENABLE_HEAP_DEBUGGING | ||||||
|  |     #define HEAP_DEBUG_MESSAGE(location) printHeapDebugData(location); | ||||||
|  | #else | ||||||
|  |     #define HEAP_DEBUG_MESSAGE(location)  | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | inline void printHeapDebugData(const char *location){ | ||||||
|  |     Serial.println("Heap: " + String(ESP.getMinFreeHeap()/1024) + "\t" + String(ESP.getFreeHeap()/1024) + "\t" + String(ESP.getMaxAllocHeap()/1024) + "\t" + location); | ||||||
|  | } | ||||||
| @@ -2,10 +2,12 @@ | |||||||
| #include "api.h" | #include "api.h" | ||||||
| #include <vector> | #include <vector> | ||||||
| #include "icons.h" | #include "icons.h" | ||||||
|  | #include "main.h" | ||||||
|  |  | ||||||
| Adafruit_SSD1306 display(SCREEN_WIDTH, SCREEN_HEIGHT, &Wire, OLED_RESET); | Adafruit_SSD1306 display(SCREEN_WIDTH, SCREEN_HEIGHT, &Wire, OLED_RESET); | ||||||
|  |  | ||||||
| bool wifiOn = false; | bool wifiOn = false; | ||||||
|  | bool iconToggle = false; | ||||||
|  |  | ||||||
| void setupDisplay() { | void setupDisplay() { | ||||||
|     if (!display.begin(SSD1306_SWITCHCAPVCC, SCREEN_ADDRESS)) { |     if (!display.begin(SSD1306_SWITCHCAPVCC, SCREEN_ADDRESS)) { | ||||||
| @@ -14,15 +16,10 @@ void setupDisplay() { | |||||||
|     } |     } | ||||||
|     display.setTextColor(WHITE); |     display.setTextColor(WHITE); | ||||||
|     display.clearDisplay(); |     display.clearDisplay(); | ||||||
|     display.display(); |  | ||||||
|  |  | ||||||
|     // Show initial display buffer contents on the screen -- |  | ||||||
|     // the library initializes this with an Adafruit splash screen. |  | ||||||
|     display.setTextColor(WHITE); |  | ||||||
|     display.display(); |  | ||||||
|     oledShowTopRow(); |     oledShowTopRow(); | ||||||
|     oledShowMessage("FilaMan v" + String(VERSION)); |     oledShowProgressBar(0, 7, DISPLAY_BOOT_TEXT, "Display init"); | ||||||
|     vTaskDelay(2000 / portTICK_PERIOD_MS); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void oledclearline() { | void oledclearline() { | ||||||
| @@ -45,14 +42,14 @@ void oledcleardata() { | |||||||
|     //display.display(); |     //display.display(); | ||||||
| } | } | ||||||
|  |  | ||||||
| int oled_center_h(String text) { | int oled_center_h(const String &text) { | ||||||
|     int16_t x1, y1; |     int16_t x1, y1; | ||||||
|     uint16_t w, h; |     uint16_t w, h; | ||||||
|     display.getTextBounds(text, 0, 0, &x1, &y1, &w, &h); |     display.getTextBounds(text, 0, 0, &x1, &y1, &w, &h); | ||||||
|     return (SCREEN_WIDTH - w) / 2; |     return (SCREEN_WIDTH - w) / 2; | ||||||
| } | } | ||||||
|  |  | ||||||
| int oled_center_v(String text) { | int oled_center_v(const String &text) { | ||||||
|     int16_t x1, y1; |     int16_t x1, y1; | ||||||
|     uint16_t w, h; |     uint16_t w, h; | ||||||
|     display.getTextBounds(text, 0, OLED_DATA_START, &x1, &y1, &w, &h); |     display.getTextBounds(text, 0, OLED_DATA_START, &x1, &y1, &w, &h); | ||||||
| @@ -60,7 +57,7 @@ int oled_center_v(String text) { | |||||||
|     return OLED_DATA_START + ((OLED_DATA_END - OLED_DATA_START - h) / 2); |     return OLED_DATA_START + ((OLED_DATA_END - OLED_DATA_START - h) / 2); | ||||||
| } | } | ||||||
|  |  | ||||||
| std::vector<String> splitTextIntoLines(String text, uint8_t textSize) { | std::vector<String> splitTextIntoLines(const String &text, uint8_t textSize) { | ||||||
|     std::vector<String> lines; |     std::vector<String> lines; | ||||||
|     display.setTextSize(textSize); |     display.setTextSize(textSize); | ||||||
|      |      | ||||||
| @@ -120,7 +117,7 @@ std::vector<String> splitTextIntoLines(String text, uint8_t textSize) { | |||||||
|     return lines; |     return lines; | ||||||
| } | } | ||||||
|  |  | ||||||
| void oledShowMultilineMessage(String message, uint8_t size) { | void oledShowMultilineMessage(const String &message, uint8_t size) { | ||||||
|     std::vector<String> lines; |     std::vector<String> lines; | ||||||
|     int maxLines = 3;  // Maximale Anzahl Zeilen für size 2 |     int maxLines = 3;  // Maximale Anzahl Zeilen für size 2 | ||||||
|      |      | ||||||
| @@ -148,7 +145,7 @@ void oledShowMultilineMessage(String message, uint8_t size) { | |||||||
|     display.display(); |     display.display(); | ||||||
| } | } | ||||||
|  |  | ||||||
| void oledShowMessage(String message, uint8_t size) { | void oledShowMessage(const String &message, uint8_t size) { | ||||||
|     oledcleardata(); |     oledcleardata(); | ||||||
|     display.setTextSize(size); |     display.setTextSize(size); | ||||||
|     display.setTextWrap(false); |     display.setTextWrap(false); | ||||||
| @@ -171,22 +168,46 @@ void oledShowMessage(String message, uint8_t size) { | |||||||
| void oledShowTopRow() { | void oledShowTopRow() { | ||||||
|     oledclearline(); |     oledclearline(); | ||||||
|  |  | ||||||
|  |     display.setTextSize(1); | ||||||
|  |     display.setCursor(0, 4); | ||||||
|  |     display.print("v"); | ||||||
|  |     display.print(VERSION); | ||||||
|  |  | ||||||
|  |     iconToggle = !iconToggle; | ||||||
|  |  | ||||||
|  |     // Do not show status indicators during boot | ||||||
|  |     if(!booting){ | ||||||
|  |         if(bambuDisabled == false) { | ||||||
|             if (bambu_connected == 1) { |             if (bambu_connected == 1) { | ||||||
|                 display.drawBitmap(50, 0, bitmap_bambu_on , 16, 16, WHITE); |                 display.drawBitmap(50, 0, bitmap_bambu_on , 16, 16, WHITE); | ||||||
|             } else { |             } else { | ||||||
|         display.drawBitmap(50, 0, bitmap_off , 16, 16, WHITE); |                 if(iconToggle){ | ||||||
|  |                     display.drawBitmap(50, 0, bitmap_bambu_on , 16, 16, WHITE); | ||||||
|  |                     display.drawLine(50, 15, 66, 0, WHITE); | ||||||
|  |                     display.drawLine(51, 15, 67, 0, WHITE); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     if (spoolman_connected == 1) { |         if (spoolmanConnected) { | ||||||
|             display.drawBitmap(80, 0, bitmap_spoolman_on , 16, 16, WHITE); |             display.drawBitmap(80, 0, bitmap_spoolman_on , 16, 16, WHITE); | ||||||
|         } else { |         } else { | ||||||
|         display.drawBitmap(80, 0, bitmap_off , 16, 16, WHITE); |             if(iconToggle){ | ||||||
|  |                 display.drawBitmap(80, 0, bitmap_spoolman_on , 16, 16, WHITE); | ||||||
|  |                 display.drawLine(80, 15, 96, 0, WHITE); | ||||||
|  |                 display.drawLine(81, 15, 97, 0, WHITE); | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (wifiOn == 1) { |         if (wifiOn == 1) { | ||||||
|             display.drawBitmap(107, 0, wifi_on , 16, 16, WHITE); |             display.drawBitmap(107, 0, wifi_on , 16, 16, WHITE); | ||||||
|         } else { |         } else { | ||||||
|         display.drawBitmap(107, 0, wifi_off , 16, 16, WHITE); |             if(iconToggle){ | ||||||
|  |                 display.drawBitmap(107, 0, wifi_on , 16, 16, WHITE); | ||||||
|  |                 display.drawLine(107, 15, 123, 0, WHITE); | ||||||
|  |                 display.drawLine(108, 15, 124, 0, WHITE); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     display.display(); |     display.display(); | ||||||
| @@ -214,6 +235,27 @@ void oledShowIcon(const char* icon) { | |||||||
|     display.display(); |     display.display(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | void oledShowProgressBar(const uint8_t step, const uint8_t numSteps, const char* largeText, const char* statusMessage){ | ||||||
|  |     assert(step <= numSteps); | ||||||
|  |  | ||||||
|  |     // clear data and bar area | ||||||
|  |     display.fillRect(0, OLED_DATA_START, SCREEN_WIDTH, SCREEN_HEIGHT-16, BLACK); | ||||||
|  |  | ||||||
|  |      | ||||||
|  |     display.setTextWrap(false); | ||||||
|  |     display.setTextSize(2); | ||||||
|  |     display.setCursor(0, OLED_DATA_START+4); | ||||||
|  |     display.print(largeText); | ||||||
|  |     display.setTextSize(1); | ||||||
|  |     display.setCursor(0, OLED_DATA_END-SCREEN_PROGRESS_BAR_HEIGHT-10); | ||||||
|  |     display.print(statusMessage); | ||||||
|  |  | ||||||
|  |     const int barLength = ((SCREEN_WIDTH-2)*step)/numSteps; | ||||||
|  |     display.drawRoundRect(0, SCREEN_HEIGHT-SCREEN_PROGRESS_BAR_HEIGHT, SCREEN_WIDTH, 12, 6, WHITE); | ||||||
|  |     display.fillRoundRect(1, SCREEN_HEIGHT-SCREEN_PROGRESS_BAR_HEIGHT+1, barLength, 10, 6, WHITE); | ||||||
|  |     display.display(); | ||||||
|  | } | ||||||
|  |  | ||||||
| void oledShowWeight(uint16_t weight) { | void oledShowWeight(uint16_t weight) { | ||||||
|     // Display Gewicht |     // Display Gewicht | ||||||
|     oledcleardata(); |     oledcleardata(); | ||||||
|   | |||||||
| @@ -13,11 +13,13 @@ extern bool wifiOn; | |||||||
| void setupDisplay(); | void setupDisplay(); | ||||||
| void oledclearline(); | void oledclearline(); | ||||||
| void oledcleardata(); | void oledcleardata(); | ||||||
| int oled_center_h(String text); | int oled_center_h(const String &text); | ||||||
| int oled_center_v(String text); | int oled_center_v(const String &text); | ||||||
|  |  | ||||||
|  | void oledShowProgressBar(const uint8_t step, const uint8_t numSteps, const char* largeText, const char* statusMessage); | ||||||
|  |  | ||||||
| void oledShowWeight(uint16_t weight); | void oledShowWeight(uint16_t weight); | ||||||
| void oledShowMessage(String message, uint8_t size = 2); | void oledShowMessage(const String &message, uint8_t size = 2); | ||||||
| void oledShowTopRow(); | void oledShowTopRow(); | ||||||
| void oledShowIcon(const char* icon); | void oledShowIcon(const char* icon); | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										143
									
								
								src/main.cpp
									
									
									
									
									
								
							
							
						
						| @@ -13,6 +13,11 @@ | |||||||
| #include "esp_task_wdt.h" | #include "esp_task_wdt.h" | ||||||
| #include "commonFS.h" | #include "commonFS.h" | ||||||
|  |  | ||||||
|  | bool mainTaskWasPaused = 0; | ||||||
|  | uint8_t scaleTareCounter = 0; | ||||||
|  | bool touchSensorConnected = false; | ||||||
|  | bool booting = true; | ||||||
|  |  | ||||||
| // ##### SETUP ##### | // ##### SETUP ##### | ||||||
| void setup() { | void setup() { | ||||||
|   Serial.begin(115200); |   Serial.begin(115200); | ||||||
| @@ -36,7 +41,6 @@ void setup() { | |||||||
|   setupWebserver(server); |   setupWebserver(server); | ||||||
|  |  | ||||||
|   // Spoolman API |   // Spoolman API | ||||||
|   // api.cpp |  | ||||||
|   initSpoolman(); |   initSpoolman(); | ||||||
|  |  | ||||||
|   // Bambu MQTT |   // Bambu MQTT | ||||||
| @@ -45,32 +49,24 @@ void setup() { | |||||||
|   // NFC Reader |   // NFC Reader | ||||||
|   startNfc(); |   startNfc(); | ||||||
|  |  | ||||||
|   uint8_t scaleCalibrated = start_scale(); |   // Touch Sensor | ||||||
|   if (scaleCalibrated == 3) { |   pinMode(TTP223_PIN, INPUT_PULLUP); | ||||||
|     oledShowMessage("Scale not calibrated!"); |   if (digitalRead(TTP223_PIN) == LOW)  | ||||||
|     for (uint16_t i = 0; i < 50000; i++) { |   { | ||||||
|       yield(); |     Serial.println("Touch Sensor is connected"); | ||||||
|       vTaskDelay(pdMS_TO_TICKS(1)); |     touchSensorConnected = true; | ||||||
|       esp_task_wdt_reset(); |  | ||||||
|     } |  | ||||||
|   } else if (scaleCalibrated == 0) { |  | ||||||
|     oledShowMessage("HX711 not found"); |  | ||||||
|     for (uint16_t i = 0; i < 50000; i++) { |  | ||||||
|       yield(); |  | ||||||
|       vTaskDelay(pdMS_TO_TICKS(1)); |  | ||||||
|       esp_task_wdt_reset(); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   // Scale | ||||||
|  |   start_scale(touchSensorConnected); | ||||||
|  |  | ||||||
|   // WDT initialisieren mit 10 Sekunden Timeout |   // WDT initialisieren mit 10 Sekunden Timeout | ||||||
|   bool panic = true; // Wenn true, löst ein WDT-Timeout einen System-Panik aus |   bool panic = true; // Wenn true, löst ein WDT-Timeout einen System-Panik aus | ||||||
|   esp_task_wdt_init(10, panic); |   esp_task_wdt_init(10, panic); | ||||||
|  |  | ||||||
|  |   booting = false; | ||||||
|   // Aktuellen Task (loopTask) zum Watchdog hinzufügen |   // Aktuellen Task (loopTask) zum Watchdog hinzufügen | ||||||
|   esp_task_wdt_add(NULL); |   esp_task_wdt_add(NULL); | ||||||
|  |  | ||||||
|   // Optional: Andere Tasks zum Watchdog hinzufügen, falls nötig |  | ||||||
|   // esp_task_wdt_add(task_handle); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -99,29 +95,61 @@ uint8_t autoAmsCounter = 0; | |||||||
| uint8_t weightSend = 0; | uint8_t weightSend = 0; | ||||||
| int16_t lastWeight = 0; | int16_t lastWeight = 0; | ||||||
|  |  | ||||||
|  | // WIFI check variables | ||||||
| unsigned long lastWifiCheckTime = 0; | unsigned long lastWifiCheckTime = 0; | ||||||
| const unsigned long wifiCheckInterval = 60000; // Überprüfe alle 60 Sekunden (60000 ms) | unsigned long lastTopRowUpdateTime = 0; | ||||||
|  | unsigned long lastSpoolmanHealcheckTime = 0; | ||||||
|  |  | ||||||
|  | // Button debounce variables | ||||||
|  | unsigned long lastButtonPress = 0; | ||||||
|  | const unsigned long debounceDelay = 500; // 500 ms debounce delay | ||||||
|  |  | ||||||
| // ##### PROGRAM START ##### | // ##### PROGRAM START ##### | ||||||
| void loop() { | void loop() { | ||||||
|   unsigned long currentMillis = millis(); |   unsigned long currentMillis = millis(); | ||||||
|  |  | ||||||
|  |   // Überprüfe den Status des Touch Sensors | ||||||
|  |   if (touchSensorConnected && digitalRead(TTP223_PIN) == HIGH && currentMillis - lastButtonPress > debounceDelay)  | ||||||
|  |   { | ||||||
|  |     lastButtonPress = currentMillis; | ||||||
|  |     scaleTareRequest = true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   // Überprüfe regelmäßig die WLAN-Verbindung |   // Überprüfe regelmäßig die WLAN-Verbindung | ||||||
|   if (intervalElapsed(currentMillis, lastWifiCheckTime, wifiCheckInterval)) { |   if (intervalElapsed(currentMillis, lastWifiCheckTime, WIFI_CHECK_INTERVAL))  | ||||||
|  |   { | ||||||
|     checkWiFiConnection(); |     checkWiFiConnection(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   // Periodic display update | ||||||
|  |   if (intervalElapsed(currentMillis, lastTopRowUpdateTime, DISPLAY_UPDATE_INTERVAL))  | ||||||
|  |   { | ||||||
|  |     oledShowTopRow(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Periodic spoolman health check | ||||||
|  |   if (intervalElapsed(currentMillis, lastSpoolmanHealcheckTime, SPOOLMAN_HEALTHCHECK_INTERVAL))  | ||||||
|  |   { | ||||||
|  |     checkSpoolmanInstance(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   // Wenn Bambu auto set Spool aktiv |   // Wenn Bambu auto set Spool aktiv | ||||||
|   if (autoSendToBambu && autoSetToBambuSpoolId > 0) { |   if (bambuCredentials.autosend_enable && autoSetToBambuSpoolId > 0)  | ||||||
|  |   { | ||||||
|  |     if (!bambuDisabled && !bambu_connected)  | ||||||
|  |     { | ||||||
|  |       bambu_restart(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     if (intervalElapsed(currentMillis, lastAutoSetBambuAmsTime, autoSetBambuAmsInterval))  |     if (intervalElapsed(currentMillis, lastAutoSetBambuAmsTime, autoSetBambuAmsInterval))  | ||||||
|     { |     { | ||||||
|       if (hasReadRfidTag == 0) |       if (nfcReaderState == NFC_IDLE) | ||||||
|       { |       { | ||||||
|         lastAutoSetBambuAmsTime = currentMillis; |         lastAutoSetBambuAmsTime = currentMillis; | ||||||
|         oledShowMessage("Auto Set         " + String(autoSetBambuAmsCounter - autoAmsCounter) + "s"); |         oledShowMessage("Auto Set         " + String(bambuCredentials.autosend_time - autoAmsCounter) + "s"); | ||||||
|         autoAmsCounter++; |         autoAmsCounter++; | ||||||
|  |  | ||||||
|         if (autoAmsCounter >= autoSetBambuAmsCounter)  |         if (autoAmsCounter >= bambuCredentials.autosend_time)  | ||||||
|         { |         { | ||||||
|           autoSetToBambuSpoolId = 0; |           autoSetToBambuSpoolId = 0; | ||||||
|           autoAmsCounter = 0; |           autoAmsCounter = 0; | ||||||
| @@ -135,31 +163,40 @@ void loop() { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Wenn Waage nicht Kalibriert |   // If scale is not calibrated, only show a warning | ||||||
|   if (scaleCalibrated == 3)  |   if (!scaleCalibrated)  | ||||||
|   { |   { | ||||||
|     oledShowMessage("Scale not calibrated!"); |     // Do not show the warning if the calibratin process is onging | ||||||
|     vTaskDelay(5000 / portTICK_PERIOD_MS); |     if(!scaleCalibrationActive){ | ||||||
|     yield(); |       oledShowMessage("Scale not calibrated"); | ||||||
|     esp_task_wdt_reset(); |       vTaskDelay(1000 / portTICK_PERIOD_MS); | ||||||
|      |  | ||||||
|     return; |  | ||||||
|     } |     } | ||||||
|  |   }  | ||||||
|  |   else  | ||||||
|  |   { | ||||||
|     // Ausgabe der Waage auf Display |     // Ausgabe der Waage auf Display | ||||||
|   if (pauseMainTask == 0 && weight != lastWeight && hasReadRfidTag == 0 && (!autoSendToBambu || autoSetToBambuSpoolId == 0)) |     if(pauseMainTask == 0) | ||||||
|  |     { | ||||||
|  |       if (mainTaskWasPaused || (weight != lastWeight && nfcReaderState == NFC_IDLE && (!bambuCredentials.autosend_enable || autoSetToBambuSpoolId == 0))) | ||||||
|       { |       { | ||||||
|         (weight < 2) ? ((weight < -2) ? oledShowMessage("!! -0") : oledShowWeight(0)) : oledShowWeight(weight); |         (weight < 2) ? ((weight < -2) ? oledShowMessage("!! -0") : oledShowWeight(0)) : oledShowWeight(weight); | ||||||
|       } |       } | ||||||
|  |       mainTaskWasPaused = false; | ||||||
|  |     } | ||||||
|  |     else | ||||||
|  |     { | ||||||
|  |       mainTaskWasPaused = true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     // Wenn Timer abgelaufen und nicht gerade ein RFID-Tag geschrieben wird |     // Wenn Timer abgelaufen und nicht gerade ein RFID-Tag geschrieben wird | ||||||
|   if (currentMillis - lastWeightReadTime >= weightReadInterval && hasReadRfidTag < 3) |     if (currentMillis - lastWeightReadTime >= weightReadInterval && nfcReaderState < NFC_WRITING) | ||||||
|     { |     { | ||||||
|       lastWeightReadTime = currentMillis; |       lastWeightReadTime = currentMillis; | ||||||
|  |  | ||||||
|       // Prüfen ob die Waage korrekt genullt ist |       // Prüfen ob die Waage korrekt genullt ist | ||||||
|     if ((weight > 0 && weight < 5) || weight < 0) |       // Abweichung von 2g ignorieren | ||||||
|  |       if (autoTare && (weight > 2 && weight < 7) || weight < -2) | ||||||
|       { |       { | ||||||
|         scale_tare_counter++; |         scale_tare_counter++; | ||||||
|       } |       } | ||||||
| @@ -169,7 +206,7 @@ void loop() { | |||||||
|       } |       } | ||||||
|  |  | ||||||
|       // Prüfen ob das Gewicht gleich bleibt und dann senden |       // Prüfen ob das Gewicht gleich bleibt und dann senden | ||||||
|     if (weight == lastWeight && weight > 5) |       if (abs(weight - lastWeight) <= 2 && weight > 5) | ||||||
|       { |       { | ||||||
|         weigthCouterToApi++; |         weigthCouterToApi++; | ||||||
|       }  |       }  | ||||||
| @@ -181,7 +218,8 @@ void loop() { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     // reset weight counter after writing tag |     // reset weight counter after writing tag | ||||||
|   if (currentMillis - lastWeightReadTime >= weightReadInterval && hasReadRfidTag > 1) |     // TBD: what exactly is the logic behind this? | ||||||
|  |     if (currentMillis - lastWeightReadTime >= weightReadInterval && nfcReaderState != NFC_IDLE && nfcReaderState != NFC_READ_SUCCESS) | ||||||
|     { |     { | ||||||
|       weigthCouterToApi = 0; |       weigthCouterToApi = 0; | ||||||
|     } |     } | ||||||
| @@ -189,18 +227,19 @@ void loop() { | |||||||
|     lastWeight = weight; |     lastWeight = weight; | ||||||
|  |  | ||||||
|     // Wenn ein Tag mit SM id erkannte wurde und der Waage Counter anspricht an SM Senden |     // Wenn ein Tag mit SM id erkannte wurde und der Waage Counter anspricht an SM Senden | ||||||
|   if (spoolId != "" && weigthCouterToApi > 3 && weightSend == 0 && hasReadRfidTag == 1) { |     if (activeSpoolId != "" && weigthCouterToApi > 3 && weightSend == 0 && nfcReaderState == NFC_READ_SUCCESS && tagProcessed == false && spoolmanApiState == API_IDLE)  | ||||||
|     oledShowIcon("loading"); |  | ||||||
|     if (updateSpoolWeight(spoolId, weight))  |  | ||||||
|     { |     { | ||||||
|       oledShowIcon("success"); |       // set the current tag as processed to prevent it beeing processed again | ||||||
|       vTaskDelay(2000 / portTICK_PERIOD_MS); |       tagProcessed = true; | ||||||
|       weightSend = 1; |  | ||||||
|       autoSetToBambuSpoolId = spoolId.toInt(); |  | ||||||
|  |  | ||||||
|       if (octoEnabled)  |       if (updateSpoolWeight(activeSpoolId, weight))  | ||||||
|       { |       { | ||||||
|         updateSpoolOcto(autoSetToBambuSpoolId); |         weightSend = 1; | ||||||
|  |          | ||||||
|  |         // Set Bambu spool ID for auto-send if enabled | ||||||
|  |         if (bambuCredentials.autosend_enable)  | ||||||
|  |         { | ||||||
|  |           autoSetToBambuSpoolId = activeSpoolId.toInt(); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       else |       else | ||||||
| @@ -210,6 +249,12 @@ void loop() { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   yield(); |     if(octoEnabled && sendOctoUpdate && spoolmanApiState == API_IDLE) | ||||||
|  |     { | ||||||
|  |       updateSpoolOcto(autoSetToBambuSpoolId); | ||||||
|  |       sendOctoUpdate = false; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |    | ||||||
|   esp_task_wdt_reset(); |   esp_task_wdt_reset(); | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								src/main.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | |||||||
|  | #ifndef MAIN_H | ||||||
|  | #define MAIN_H | ||||||
|  |  | ||||||
|  | #include <Arduino.h> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | extern bool booting; | ||||||
|  |  | ||||||
|  | #endif | ||||||
							
								
								
									
										387
									
								
								src/nfc.cpp
									
									
									
									
									
								
							
							
						
						| @@ -7,6 +7,8 @@ | |||||||
| #include "api.h" | #include "api.h" | ||||||
| #include "esp_task_wdt.h" | #include "esp_task_wdt.h" | ||||||
| #include "scale.h" | #include "scale.h" | ||||||
|  | #include "bambu.h" | ||||||
|  | #include "main.h" | ||||||
|  |  | ||||||
| //Adafruit_PN532 nfc(PN532_SCK, PN532_MISO, PN532_MOSI, PN532_SS); | //Adafruit_PN532 nfc(PN532_SCK, PN532_MISO, PN532_MOSI, PN532_SS); | ||||||
| Adafruit_PN532 nfc(PN532_IRQ, PN532_RESET); | Adafruit_PN532 nfc(PN532_IRQ, PN532_RESET); | ||||||
| @@ -14,11 +16,20 @@ Adafruit_PN532 nfc(PN532_IRQ, PN532_RESET); | |||||||
| TaskHandle_t RfidReaderTask; | TaskHandle_t RfidReaderTask; | ||||||
|  |  | ||||||
| JsonDocument rfidData; | JsonDocument rfidData; | ||||||
| String spoolId = ""; | String activeSpoolId = ""; | ||||||
|  | String lastSpoolId = ""; | ||||||
| String nfcJsonData = ""; | String nfcJsonData = ""; | ||||||
|  | bool tagProcessed = false; | ||||||
| volatile bool pauseBambuMqttTask = false; | volatile bool pauseBambuMqttTask = false; | ||||||
|  | volatile bool nfcReadingTaskSuspendRequest = false; | ||||||
|  | volatile bool nfcReadingTaskSuspendState = false; | ||||||
|  |  | ||||||
| volatile uint8_t hasReadRfidTag = 0; | struct NfcWriteParameterType { | ||||||
|  |   bool tagType; | ||||||
|  |   char* payload; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | volatile nfcReaderStateType nfcReaderState = NFC_IDLE; | ||||||
| // 0 = nicht gelesen | // 0 = nicht gelesen | ||||||
| // 1 = erfolgreich gelesen | // 1 = erfolgreich gelesen | ||||||
| // 2 = fehler beim Lesen | // 2 = fehler beim Lesen | ||||||
| @@ -28,7 +39,6 @@ volatile uint8_t hasReadRfidTag = 0; | |||||||
| // 6 = reading | // 6 = reading | ||||||
| // ***** PN532 | // ***** PN532 | ||||||
|  |  | ||||||
|  |  | ||||||
| // ##### Funktionen für RFID ##### | // ##### Funktionen für RFID ##### | ||||||
| void payloadToJson(uint8_t *data) { | void payloadToJson(uint8_t *data) { | ||||||
|     const char* startJson = strchr((char*)data, '{'); |     const char* startJson = strchr((char*)data, '{'); | ||||||
| @@ -64,6 +74,8 @@ void payloadToJson(uint8_t *data) { | |||||||
|         Serial.print("deserializeJson() failed: "); |         Serial.print("deserializeJson() failed: "); | ||||||
|         Serial.println(error.f_str()); |         Serial.println(error.f_str()); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       doc.clear(); | ||||||
|     } else { |     } else { | ||||||
|         Serial.println("Kein gültiger JSON-Inhalt gefunden oder fehlerhafte Formatierung."); |         Serial.println("Kein gültiger JSON-Inhalt gefunden oder fehlerhafte Formatierung."); | ||||||
|         //writeJsonToTag("{\"version\":\"1.0\",\"protocol\":\"NFC\",\"color_hex\":\"#FFFFFF\",\"type\":\"Example\",\"min_temp\":10,\"max_temp\":30,\"brand\":\"BrandName\"}"); |         //writeJsonToTag("{\"version\":\"1.0\",\"protocol\":\"NFC\",\"color_hex\":\"#FFFFFF\",\"type\":\"Example\",\"min_temp\":10,\"max_temp\":30,\"brand\":\"BrandName\"}"); | ||||||
| @@ -191,19 +203,175 @@ uint8_t ntag2xx_WriteNDEF(const char *payload) { | |||||||
|   return 1; |   return 1; | ||||||
| } | } | ||||||
|  |  | ||||||
| bool decodeNdefAndReturnJson(const byte* encodedMessage) { | bool decodeNdefAndReturnJson(const byte* encodedMessage, String uidString) { | ||||||
|   byte typeLength = encodedMessage[3]; |   oledShowProgressBar(1, octoEnabled?5:4, "Reading", "Decoding data"); | ||||||
|   byte payloadLength = encodedMessage[4]; |  | ||||||
|  |   // Debug: Print first 32 bytes of the raw data | ||||||
|  |   Serial.println("Raw NDEF data (first 32 bytes):"); | ||||||
|  |   for (int i = 0; i < 32; i++) { | ||||||
|  |     if (encodedMessage[i] < 0x10) Serial.print("0"); | ||||||
|  |     Serial.print(encodedMessage[i], HEX); | ||||||
|  |     Serial.print(" "); | ||||||
|  |     if ((i + 1) % 16 == 0) Serial.println(); | ||||||
|  |   } | ||||||
|  |   Serial.println(); | ||||||
|  |  | ||||||
|  |   // Look for the NDEF TLV structure starting from the beginning | ||||||
|  |   int tlvOffset = 0; | ||||||
|  |   bool foundNdefTlv = false; | ||||||
|  |    | ||||||
|  |   // Search for NDEF TLV (0x03) in the first few bytes | ||||||
|  |   for (int i = 0; i < 16; i++) { | ||||||
|  |     if (encodedMessage[i] == 0x03) { | ||||||
|  |       tlvOffset = i; | ||||||
|  |       foundNdefTlv = true; | ||||||
|  |       Serial.print("Found NDEF TLV at offset: "); | ||||||
|  |       Serial.println(tlvOffset); | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (!foundNdefTlv) { | ||||||
|  |     Serial.println("No NDEF TLV found in tag data"); | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Get the NDEF message length from TLV | ||||||
|  |   uint16_t ndefMessageLength = 0; | ||||||
|  |   int ndefRecordOffset = 0; | ||||||
|  |    | ||||||
|  |   if (encodedMessage[tlvOffset + 1] == 0xFF) { | ||||||
|  |     // Extended length format: next 2 bytes contain the actual length | ||||||
|  |     ndefMessageLength = (encodedMessage[tlvOffset + 2] << 8) | encodedMessage[tlvOffset + 3]; | ||||||
|  |     ndefRecordOffset = tlvOffset + 4; // Skip TLV tag + 0xFF + 2 length bytes | ||||||
|  |     Serial.print("NDEF Message Length (extended): "); | ||||||
|  |   } else { | ||||||
|  |     // Standard length format: single byte contains the length | ||||||
|  |     ndefMessageLength = encodedMessage[tlvOffset + 1]; | ||||||
|  |     ndefRecordOffset = tlvOffset + 2; // Skip TLV tag + 1 length byte | ||||||
|  |     Serial.print("NDEF Message Length (standard): "); | ||||||
|  |   } | ||||||
|  |   Serial.println(ndefMessageLength); | ||||||
|  |  | ||||||
|  |   // Get pointer to NDEF record | ||||||
|  |   const byte* ndefRecord = &encodedMessage[ndefRecordOffset]; | ||||||
|  |    | ||||||
|  |   // Parse NDEF record header | ||||||
|  |   byte recordHeader = ndefRecord[0]; | ||||||
|  |   byte typeLength = ndefRecord[1]; | ||||||
|  |    | ||||||
|  |   Serial.print("NDEF Record Header: 0x"); | ||||||
|  |   Serial.println(recordHeader, HEX); | ||||||
|  |   Serial.print("Type Length: "); | ||||||
|  |   Serial.println(typeLength); | ||||||
|  |  | ||||||
|  |   // Determine payload length (can be 1 or 4 bytes depending on SR flag) | ||||||
|  |   uint32_t payloadLength = 0; | ||||||
|  |   byte payloadLengthBytes = 1; | ||||||
|  |   byte payloadLengthOffset = 2; | ||||||
|  |    | ||||||
|  |   // Check if Short Record (SR) flag is set (bit 4) | ||||||
|  |   if (recordHeader & 0x10) { // SR flag | ||||||
|  |     payloadLength = ndefRecord[2]; | ||||||
|  |     payloadLengthBytes = 1; | ||||||
|  |     payloadLengthOffset = 2; | ||||||
|  |   } else { | ||||||
|  |     // Long record format (4 bytes for payload length) | ||||||
|  |     payloadLength = (ndefRecord[2] << 24) | (ndefRecord[3] << 16) |  | ||||||
|  |                    (ndefRecord[4] << 8) | ndefRecord[5]; | ||||||
|  |     payloadLengthBytes = 4; | ||||||
|  |     payloadLengthOffset = 2; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Serial.print("Payload Length: "); | ||||||
|  |   Serial.println(payloadLength); | ||||||
|  |   Serial.print("Payload Length Bytes: "); | ||||||
|  |   Serial.println(payloadLengthBytes); | ||||||
|  |  | ||||||
|  |   // Check for ID field (if IL flag is set) | ||||||
|  |   byte idLength = 0; | ||||||
|  |   if (recordHeader & 0x08) { // IL flag | ||||||
|  |     idLength = ndefRecord[payloadLengthOffset + payloadLengthBytes]; | ||||||
|  |     Serial.print("ID Length: "); | ||||||
|  |     Serial.println(idLength); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Calculate offset to payload | ||||||
|  |   byte payloadOffset = 1 + 1 + payloadLengthBytes + typeLength + idLength; | ||||||
|  |    | ||||||
|  |   Serial.print("Calculated payload offset: "); | ||||||
|  |   Serial.println(payloadOffset); | ||||||
|  |  | ||||||
|  |   // Verify we have enough data | ||||||
|  |   if (payloadOffset + payloadLength > ndefMessageLength) { | ||||||
|  |     Serial.println("Invalid NDEF structure - payload extends beyond message"); | ||||||
|  |     Serial.print("Payload offset + length: "); | ||||||
|  |     Serial.print(payloadOffset + payloadLength); | ||||||
|  |     Serial.print(", NDEF message length: "); | ||||||
|  |     Serial.println(ndefMessageLength); | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Print the record type for debugging | ||||||
|  |   Serial.print("Record Type: "); | ||||||
|  |   for (int i = 0; i < typeLength; i++) { | ||||||
|  |     Serial.print((char)ndefRecord[1 + 1 + payloadLengthBytes + i]); | ||||||
|  |   } | ||||||
|  |   Serial.println(); | ||||||
|  |  | ||||||
|   nfcJsonData = ""; |   nfcJsonData = ""; | ||||||
|  |  | ||||||
|   for (int i = 2; i < payloadLength+2; i++)  |   // Extract JSON payload with validation | ||||||
|   { |   uint32_t actualJsonLength = 0; | ||||||
|     nfcJsonData += (char)encodedMessage[3 + typeLength + i]; |   for (uint32_t i = 0; i < payloadLength; i++) { | ||||||
|  |     byte currentByte = ndefRecord[payloadOffset + i]; | ||||||
|  |      | ||||||
|  |     // Stop at null terminator or if we find the end of JSON | ||||||
|  |     if (currentByte == 0x00) { | ||||||
|  |       Serial.print("Found null terminator at position: "); | ||||||
|  |       Serial.println(i); | ||||||
|  |       break; | ||||||
|     } |     } | ||||||
|      |      | ||||||
|  |     // Only add printable characters and common JSON characters | ||||||
|  |     if (currentByte >= 32 && currentByte <= 126) { | ||||||
|  |       nfcJsonData += (char)currentByte; | ||||||
|  |       actualJsonLength++; | ||||||
|  |     } else { | ||||||
|  |       Serial.print("Skipping non-printable byte at position "); | ||||||
|  |       Serial.print(i); | ||||||
|  |       Serial.print(": 0x"); | ||||||
|  |       Serial.println(currentByte, HEX); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     // Check if we've reached the end of a JSON object | ||||||
|  |     if (currentByte == '}') { | ||||||
|  |       // Count opening and closing braces to detect complete JSON | ||||||
|  |       int braceCount = 0; | ||||||
|  |       for (uint32_t j = 0; j <= i; j++) { | ||||||
|  |         if (ndefRecord[payloadOffset + j] == '{') braceCount++; | ||||||
|  |         else if (ndefRecord[payloadOffset + j] == '}') braceCount--; | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       if (braceCount == 0) { | ||||||
|  |         Serial.print("Found complete JSON object at position: "); | ||||||
|  |         Serial.println(i); | ||||||
|  |         actualJsonLength = i + 1; | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Serial.print("Actual JSON length extracted: "); | ||||||
|  |   Serial.println(actualJsonLength); | ||||||
|  |   Serial.println("Decoded JSON Data:"); | ||||||
|  |   Serial.println(nfcJsonData); | ||||||
|  |    | ||||||
|  |   // Trim any trailing whitespace or invalid characters | ||||||
|  |   nfcJsonData.trim(); | ||||||
|  |  | ||||||
|   // JSON-Dokument verarbeiten |   // JSON-Dokument verarbeiten | ||||||
|   JsonDocument doc;  // Passen Sie die Größe an den JSON-Inhalt an |   JsonDocument doc; | ||||||
|   DeserializationError error = deserializeJson(doc, nfcJsonData); |   DeserializationError error = deserializeJson(doc, nfcJsonData); | ||||||
|   if (error)  |   if (error)  | ||||||
|   { |   { | ||||||
| @@ -215,52 +383,89 @@ bool decodeNdefAndReturnJson(const byte* encodedMessage) { | |||||||
|   }  |   }  | ||||||
|   else  |   else  | ||||||
|   { |   { | ||||||
|  |     // If spoolman is unavailable, there is no point in continuing | ||||||
|  |     if(spoolmanConnected){ | ||||||
|       // Sende die aktualisierten AMS-Daten an alle WebSocket-Clients |       // Sende die aktualisierten AMS-Daten an alle WebSocket-Clients | ||||||
|       Serial.println("JSON-Dokument erfolgreich verarbeitet"); |       Serial.println("JSON-Dokument erfolgreich verarbeitet"); | ||||||
|       Serial.println(doc.as<String>()); |       Serial.println(doc.as<String>()); | ||||||
|     if (doc["sm_id"] != "")  |       if (doc["sm_id"].is<String>() && doc["sm_id"] != "" && doc["sm_id"] != "0") | ||||||
|       { |       { | ||||||
|  |         oledShowProgressBar(2, octoEnabled?5:4, "Spool Tag", "Weighing"); | ||||||
|         Serial.println("SPOOL-ID gefunden: " + doc["sm_id"].as<String>()); |         Serial.println("SPOOL-ID gefunden: " + doc["sm_id"].as<String>()); | ||||||
|       spoolId = doc["sm_id"].as<String>(); |         activeSpoolId = doc["sm_id"].as<String>(); | ||||||
|  |         lastSpoolId = activeSpoolId; | ||||||
|  |       } | ||||||
|  |       else if(doc["location"].is<String>() && doc["location"] != "") | ||||||
|  |       { | ||||||
|  |         Serial.println("Location Tag found!"); | ||||||
|  |         String location = doc["location"].as<String>(); | ||||||
|  |         if(lastSpoolId != ""){ | ||||||
|  |           updateSpoolLocation(lastSpoolId, location); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |           Serial.println("Location update tag scanned without scanning spool before!"); | ||||||
|  |           oledShowProgressBar(1, 1, "Failure", "Scan spool first"); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       // Brand Filament not registered to Spoolman | ||||||
|  |       else if ((!doc["sm_id"].is<String>() || (doc["sm_id"].is<String>() && (doc["sm_id"] == "0" || doc["sm_id"] == ""))) | ||||||
|  |               && doc["brand"].is<String>() && doc["artnr"].is<String>()) | ||||||
|  |       { | ||||||
|  |         doc["sm_id"] = "0"; // Ensure sm_id is set to 0 | ||||||
|  |         // If no sm_id is present but the brand is Brand Filament then | ||||||
|  |         // create a new spool, maybe brand too, in Spoolman | ||||||
|  |         Serial.println("New Brand Filament Tag found!"); | ||||||
|  |         createBrandFilament(doc, uidString); | ||||||
|       } |       } | ||||||
|       else  |       else  | ||||||
|       { |       { | ||||||
|         Serial.println("Keine SPOOL-ID gefunden."); |         Serial.println("Keine SPOOL-ID gefunden."); | ||||||
|       spoolId = ""; |         activeSpoolId = ""; | ||||||
|       oledShowMessage("Unknown Spool"); |         oledShowProgressBar(1, 1, "Failure", "Unkown tag"); | ||||||
|       vTaskDelay(2000 / portTICK_PERIOD_MS); |       } | ||||||
|  |     }else{ | ||||||
|  |       oledShowProgressBar(octoEnabled?5:4, octoEnabled?5:4, "Failure!", "Spoolman unavailable"); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   doc.clear(); | ||||||
|  |  | ||||||
|   return true; |   return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| void writeJsonToTag(void *parameter) { | void writeJsonToTag(void *parameter) { | ||||||
|   const char* payload = (const char*)parameter; |   NfcWriteParameterType* params = (NfcWriteParameterType*)parameter; | ||||||
|  |  | ||||||
|   // Gib die erstellte NDEF-Message aus |   // Gib die erstellte NDEF-Message aus | ||||||
|   Serial.println("Erstelle NDEF-Message..."); |   Serial.println("Erstelle NDEF-Message..."); | ||||||
|   Serial.println(payload); |   Serial.println(params->payload); | ||||||
|  |  | ||||||
|   hasReadRfidTag = 3; |   nfcReaderState = NFC_WRITING; | ||||||
|   vTaskSuspend(RfidReaderTask); |  | ||||||
|   vTaskDelay(50 / portTICK_PERIOD_MS); |   // First request the reading task to be suspended and than wait until it responds | ||||||
|  |   nfcReadingTaskSuspendRequest = true; | ||||||
|  |   while(nfcReadingTaskSuspendState == false){ | ||||||
|  |     vTaskDelay(100 / portTICK_PERIOD_MS); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   //pauseBambuMqttTask = true; |   //pauseBambuMqttTask = true; | ||||||
|   // aktualisieren der Website wenn sich der Status ändert |   // aktualisieren der Website wenn sich der Status ändert | ||||||
|   sendNfcData(nullptr); |   sendNfcData(); | ||||||
|   vTaskDelay(100 / portTICK_PERIOD_MS); |   vTaskDelay(100 / portTICK_PERIOD_MS); | ||||||
|   oledShowMessage("Waiting for NFC-Tag"); |  | ||||||
|    |  | ||||||
|   // Wait 10sec for tag |   // Wait 10sec for tag | ||||||
|   uint8_t success = 0; |   uint8_t success = 0; | ||||||
|   String uidString = ""; |   String uidString = ""; | ||||||
|   for (uint16_t i = 0; i < 20; i++) { |   for (uint16_t i = 0; i < 20; i++) { | ||||||
|     uint8_t uid[] = { 0, 0, 0, 0, 0, 0, 0 };  // Buffer to store the returned UID |     uint8_t uid[] = { 0, 0, 0, 0, 0, 0, 0 };  // Buffer to store the returned UID | ||||||
|     uint8_t uidLength; |     uint8_t uidLength; | ||||||
|     success = nfc.readPassiveTargetID(PN532_MIFARE_ISO14443A, uid, &uidLength, 500); |     // yield before potentially waiting for 400ms | ||||||
|  |     yield(); | ||||||
|  |     esp_task_wdt_reset(); | ||||||
|  |     success = nfc.readPassiveTargetID(PN532_MIFARE_ISO14443A, uid, &uidLength, 400); | ||||||
|     if (success) { |     if (success) { | ||||||
|       for (uint8_t i = 0; i < uidLength; i++) { |       for (uint8_t i = 0; i < uidLength; i++) { | ||||||
|  |         //TBD: Rework to remove all the string operations | ||||||
|         uidString += String(uid[i], HEX); |         uidString += String(uid[i], HEX); | ||||||
|         if (i < uidLength - 1) { |         if (i < uidLength - 1) { | ||||||
|             uidString += ":"; // Optional: Trennzeichen hinzufügen |             uidString += ":"; // Optional: Trennzeichen hinzufügen | ||||||
| @@ -270,8 +475,6 @@ void writeJsonToTag(void *parameter) { | |||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (i == 0) oledShowMessage("Waiting for NFC-Tag"); |  | ||||||
|  |  | ||||||
|     yield(); |     yield(); | ||||||
|     esp_task_wdt_reset(); |     esp_task_wdt_reset(); | ||||||
|     vTaskDelay(pdMS_TO_TICKS(1)); |     vTaskDelay(pdMS_TO_TICKS(1)); | ||||||
| @@ -279,29 +482,37 @@ void writeJsonToTag(void *parameter) { | |||||||
|  |  | ||||||
|   if (success) |   if (success) | ||||||
|   { |   { | ||||||
|     oledShowIcon("transfer"); |     oledShowProgressBar(1, 3, "Write Tag", "Writing"); | ||||||
|  |  | ||||||
|     // Schreibe die NDEF-Message auf den Tag |     // Schreibe die NDEF-Message auf den Tag | ||||||
|     success = ntag2xx_WriteNDEF(payload); |     success = ntag2xx_WriteNDEF(params->payload); | ||||||
|     if (success)  |     if (success)  | ||||||
|     { |     { | ||||||
|         Serial.println("NDEF-Message erfolgreich auf den Tag geschrieben"); |         Serial.println("NDEF-Message erfolgreich auf den Tag geschrieben"); | ||||||
|         //oledShowMessage("NFC-Tag written"); |         //oledShowMessage("NFC-Tag written"); | ||||||
|         oledShowIcon("success"); |         //vTaskDelay(1000 / portTICK_PERIOD_MS); | ||||||
|         vTaskDelay(1000 / portTICK_PERIOD_MS); |         nfcReaderState = NFC_WRITE_SUCCESS; | ||||||
|         hasReadRfidTag = 5; |  | ||||||
|         // aktualisieren der Website wenn sich der Status ändert |         // aktualisieren der Website wenn sich der Status ändert | ||||||
|         sendNfcData(nullptr); |         sendNfcData(); | ||||||
|         pauseBambuMqttTask = false; |         pauseBambuMqttTask = false; | ||||||
|          |          | ||||||
|         if (updateSpoolTagId(uidString, payload)) { |         if(params->tagType){ | ||||||
|  |           // TBD: should this be simplified? | ||||||
|  |           if (updateSpoolTagId(uidString, params->payload) && params->tagType) { | ||||||
|  |              | ||||||
|  |           }else{ | ||||||
|  |             // Potentially handle errors | ||||||
|  |           } | ||||||
|  |         }else{ | ||||||
|  |           oledShowProgressBar(1, 1, "Write Tag", "Done!"); | ||||||
|  |         } | ||||||
|         uint8_t uid[] = { 0, 0, 0, 0, 0, 0, 0 };  // Buffer to store the returned UID |         uint8_t uid[] = { 0, 0, 0, 0, 0, 0, 0 };  // Buffer to store the returned UID | ||||||
|         uint8_t uidLength; |         uint8_t uidLength; | ||||||
|           oledShowIcon("success"); |         yield(); | ||||||
|           while (nfc.readPassiveTargetID(PN532_MIFARE_ISO14443A, uid, &uidLength, 500)) { |         esp_task_wdt_reset(); | ||||||
|  |         while (nfc.readPassiveTargetID(PN532_MIFARE_ISO14443A, uid, &uidLength, 400)) { | ||||||
|           yield(); |           yield(); | ||||||
|         }  |         }  | ||||||
|         } |  | ||||||
|            |  | ||||||
|         vTaskResume(RfidReaderTask); |         vTaskResume(RfidReaderTask); | ||||||
|         vTaskDelay(500 / portTICK_PERIOD_MS);         |         vTaskDelay(500 / portTICK_PERIOD_MS);         | ||||||
|     }  |     }  | ||||||
| @@ -310,40 +521,49 @@ void writeJsonToTag(void *parameter) { | |||||||
|         Serial.println("Fehler beim Schreiben der NDEF-Message auf den Tag"); |         Serial.println("Fehler beim Schreiben der NDEF-Message auf den Tag"); | ||||||
|         oledShowIcon("failed"); |         oledShowIcon("failed"); | ||||||
|         vTaskDelay(2000 / portTICK_PERIOD_MS); |         vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||||
|         hasReadRfidTag = 4; |         nfcReaderState = NFC_WRITE_ERROR; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   else |   else | ||||||
|   { |   { | ||||||
|     Serial.println("Fehler: Kein Tag zu schreiben gefunden."); |     Serial.println("Fehler: Kein Tag zu schreiben gefunden."); | ||||||
|     oledShowMessage("No NFC-Tag found"); |     oledShowProgressBar(1, 1, "Failure!", "No tag found"); | ||||||
|     vTaskDelay(2000 / portTICK_PERIOD_MS); |     vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||||
|     hasReadRfidTag = 0; |     nfcReaderState = NFC_IDLE; | ||||||
|   } |   } | ||||||
|    |    | ||||||
|   sendWriteResult(nullptr, success); |   sendWriteResult(nullptr, success); | ||||||
|   sendNfcData(nullptr); |   sendNfcData(); | ||||||
|  |  | ||||||
|   vTaskResume(RfidReaderTask); |   nfcReadingTaskSuspendRequest = false; | ||||||
|   pauseBambuMqttTask = false; |   pauseBambuMqttTask = false; | ||||||
|  |  | ||||||
|  |   free(params->payload); | ||||||
|  |   delete params; | ||||||
|  |  | ||||||
|   vTaskDelete(NULL); |   vTaskDelete(NULL); | ||||||
| } | } | ||||||
|  |  | ||||||
| void startWriteJsonToTag(const char* payload) { | void startWriteJsonToTag(const bool isSpoolTag, const char* payload) { | ||||||
|   char* payloadCopy = strdup(payload); |   NfcWriteParameterType* parameters = new NfcWriteParameterType(); | ||||||
|  |   parameters->tagType = isSpoolTag; | ||||||
|  |   parameters->payload = strdup(payload); | ||||||
|    |    | ||||||
|   // Task nicht mehrfach starten |   // Task nicht mehrfach starten | ||||||
|   if (hasReadRfidTag != 3) { |   if (nfcReaderState == NFC_IDLE || nfcReaderState == NFC_READ_ERROR || nfcReaderState == NFC_READ_SUCCESS) { | ||||||
|  |     oledShowProgressBar(0, 1, "Write Tag", "Place tag now"); | ||||||
|     // Erstelle die Task |     // Erstelle die Task | ||||||
|     xTaskCreate( |     xTaskCreate( | ||||||
|         writeJsonToTag,        // Task-Funktion |         writeJsonToTag,        // Task-Funktion | ||||||
|         "WriteJsonToTagTask",       // Task-Name |         "WriteJsonToTagTask",       // Task-Name | ||||||
|         5115,                        // Stackgröße in Bytes |         5115,                        // Stackgröße in Bytes | ||||||
|         (void*)payloadCopy,         // Parameter |         (void*)parameters,         // Parameter | ||||||
|         rfidWriteTaskPrio,           // Priorität |         rfidWriteTaskPrio,           // Priorität | ||||||
|         NULL                         // Task-Handle (nicht benötigt) |         NULL                         // Task-Handle (nicht benötigt) | ||||||
|     ); |     ); | ||||||
|  |   }else{ | ||||||
|  |     oledShowProgressBar(0, 1, "FAILURE", "NFC busy!"); | ||||||
|  |     // TBD: Add proper error handling (website) | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -351,27 +571,43 @@ void scanRfidTask(void * parameter) { | |||||||
|   Serial.println("RFID Task gestartet"); |   Serial.println("RFID Task gestartet"); | ||||||
|   for(;;) { |   for(;;) { | ||||||
|     // Wenn geschrieben wird Schleife aussetzen |     // Wenn geschrieben wird Schleife aussetzen | ||||||
|     if (hasReadRfidTag != 3) |     if (nfcReaderState != NFC_WRITING && !nfcReadingTaskSuspendRequest && !booting) | ||||||
|     { |     { | ||||||
|  |       nfcReadingTaskSuspendState = false; | ||||||
|       yield(); |       yield(); | ||||||
|  |  | ||||||
|       uint8_t success; |       uint8_t success; | ||||||
|       uint8_t uid[] = { 0, 0, 0, 0, 0, 0, 0 };  // Buffer to store the returned UID |       uint8_t uid[] = { 0, 0, 0, 0, 0, 0, 0 };  // Buffer to store the returned UID | ||||||
|       uint8_t uidLength; |       uint8_t uidLength; | ||||||
|  |  | ||||||
|       success = nfc.readPassiveTargetID(PN532_MIFARE_ISO14443A, uid, &uidLength, 1000); |       success = nfc.readPassiveTargetID(PN532_MIFARE_ISO14443A, uid, &uidLength, 500); | ||||||
|  |  | ||||||
|       foundNfcTag(nullptr, success); |       foundNfcTag(nullptr, success); | ||||||
|        |        | ||||||
|       if (success && hasReadRfidTag != 1) |       // As long as there is still a tag on the reader, do not try to read it again | ||||||
|  |       if (success && nfcReaderState == NFC_IDLE) | ||||||
|       { |       { | ||||||
|  |         // Set the current tag as not processed | ||||||
|  |         tagProcessed = false; | ||||||
|  |  | ||||||
|         // Display some basic information about the card |         // Display some basic information about the card | ||||||
|         Serial.println("Found an ISO14443A card"); |         Serial.println("Found an ISO14443A card"); | ||||||
|  |  | ||||||
|         hasReadRfidTag = 6; |         nfcReaderState = NFC_READING; | ||||||
|  |  | ||||||
|         oledShowIcon("transfer"); |         oledShowProgressBar(0, octoEnabled?5:4, "Reading", "Detecting tag"); | ||||||
|         vTaskDelay(500 / portTICK_PERIOD_MS); |  | ||||||
|  |         //vTaskDelay(500 / portTICK_PERIOD_MS); | ||||||
|  |  | ||||||
|  |         // create Tag UID string | ||||||
|  |         String uidString = ""; | ||||||
|  |         for (uint8_t i = 0; i < uidLength; i++) { | ||||||
|  |           //TBD: Rework to remove all the string operations | ||||||
|  |           uidString += String(uid[i], HEX); | ||||||
|  |           if (i < uidLength - 1) { | ||||||
|  |               uidString += ":"; // Optional: Trennzeichen hinzufügen | ||||||
|  |           } | ||||||
|  |         } | ||||||
|          |          | ||||||
|         if (uidLength == 7) |         if (uidLength == 7) | ||||||
|         { |         { | ||||||
| @@ -402,57 +638,76 @@ void scanRfidTask(void * parameter) { | |||||||
|               vTaskDelay(pdMS_TO_TICKS(1)); |               vTaskDelay(pdMS_TO_TICKS(1)); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             if (!decodeNdefAndReturnJson(data))  |             if (!decodeNdefAndReturnJson(data, uidString))  | ||||||
|             { |             { | ||||||
|               oledShowMessage("NFC-Tag unknown"); |               oledShowProgressBar(1, 1, "Failure", "Unknown tag"); | ||||||
|               vTaskDelay(2000 / portTICK_PERIOD_MS); |               nfcReaderState = NFC_READ_ERROR; | ||||||
|               hasReadRfidTag = 2; |  | ||||||
|             } |             } | ||||||
|             else  |             else  | ||||||
|             { |             { | ||||||
|               hasReadRfidTag = 1; |               nfcReaderState = NFC_READ_SUCCESS; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             free(data); |             free(data); | ||||||
|           } |           } | ||||||
|           else |           else | ||||||
|           { |           { | ||||||
|             oledShowMessage("NFC-Tag read error"); |             oledShowProgressBar(1, 1, "Failure", "Tag read error"); | ||||||
|             hasReadRfidTag = 2; |             nfcReaderState = NFC_READ_ERROR; | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
|         else |         else | ||||||
|         { |         { | ||||||
|  |           //TBD: Show error here?! | ||||||
|  |           oledShowProgressBar(1, 1, "Failure", "Unkown tag type"); | ||||||
|           Serial.println("This doesn't seem to be an NTAG2xx tag (UUID length != 7 bytes)!"); |           Serial.println("This doesn't seem to be an NTAG2xx tag (UUID length != 7 bytes)!"); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       if (!success && hasReadRfidTag > 0) |       if (!success && nfcReaderState != NFC_IDLE && !nfcReadingTaskSuspendRequest) | ||||||
|       { |       { | ||||||
|         hasReadRfidTag = 0; |         nfcReaderState = NFC_IDLE; | ||||||
|         //uidString = ""; |         //uidString = ""; | ||||||
|         nfcJsonData = ""; |         nfcJsonData = ""; | ||||||
|  |         activeSpoolId = ""; | ||||||
|         Serial.println("Tag entfernt"); |         Serial.println("Tag entfernt"); | ||||||
|         if (!autoSendToBambu) oledShowWeight(weight); |         if (!bambuCredentials.autosend_enable) oledShowWeight(weight); | ||||||
|  |       } | ||||||
|  |       // Reset state after successful read when tag is removed | ||||||
|  |       else if (!success && nfcReaderState == NFC_READ_SUCCESS) | ||||||
|  |       { | ||||||
|  |         nfcReaderState = NFC_IDLE; | ||||||
|  |         Serial.println("Tag nach erfolgreichem Lesen entfernt - bereit für nächsten Tag"); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // Add a longer pause after successful reading to prevent immediate re-reading | ||||||
|  |       if (nfcReaderState == NFC_READ_SUCCESS) { | ||||||
|  |         Serial.println("Tag erfolgreich gelesen - warte 5 Sekunden vor nächstem Scan"); | ||||||
|  |         vTaskDelay(5000 / portTICK_PERIOD_MS); // 5 second pause | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       // aktualisieren der Website wenn sich der Status ändert |       // aktualisieren der Website wenn sich der Status ändert | ||||||
|       sendNfcData(nullptr); |       sendNfcData(); | ||||||
|  |     } | ||||||
|  |     else | ||||||
|  |     { | ||||||
|  |       nfcReadingTaskSuspendState = true; | ||||||
|  |       Serial.println("NFC Reading disabled"); | ||||||
|  |       vTaskDelay(1000 / portTICK_PERIOD_MS); | ||||||
|     } |     } | ||||||
|     yield(); |     yield(); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| void startNfc() { | void startNfc() { | ||||||
|  |   oledShowProgressBar(5, 7, DISPLAY_BOOT_TEXT, "NFC init"); | ||||||
|   nfc.begin();                                           // Beginne Kommunikation mit RFID Leser |   nfc.begin();                                           // Beginne Kommunikation mit RFID Leser | ||||||
|   delay(1000); |   delay(1000); | ||||||
|   unsigned long versiondata = nfc.getFirmwareVersion();  // Lese Versionsnummer der Firmware aus |   unsigned long versiondata = nfc.getFirmwareVersion();  // Lese Versionsnummer der Firmware aus | ||||||
|   if (! versiondata) {                                   // Wenn keine Antwort kommt |   if (! versiondata) {                                   // Wenn keine Antwort kommt | ||||||
|     Serial.println("Kann kein RFID Board finden !");            // Sende Text "Kann kein..." an seriellen Monitor |     Serial.println("Kann kein RFID Board finden !");            // Sende Text "Kann kein..." an seriellen Monitor | ||||||
|     //delay(5000); |  | ||||||
|     //ESP.restart(); |  | ||||||
|     oledShowMessage("No RFID Board found"); |     oledShowMessage("No RFID Board found"); | ||||||
|     delay(2000); |     vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||||
|   } |   } | ||||||
|   else { |   else { | ||||||
|     Serial.print("Chip PN5 gefunden"); Serial.println((versiondata >> 24) & 0xFF, HEX); // Sende Text und Versionsinfos an seriellen |     Serial.print("Chip PN5 gefunden"); Serial.println((versiondata >> 24) & 0xFF, HEX); // Sende Text und Versionsinfos an seriellen | ||||||
|   | |||||||
							
								
								
									
										20
									
								
								src/nfc.h
									
									
									
									
									
								
							
							
						
						| @@ -3,14 +3,28 @@ | |||||||
|  |  | ||||||
| #include <Arduino.h> | #include <Arduino.h> | ||||||
|  |  | ||||||
|  | typedef enum{ | ||||||
|  |     NFC_IDLE, | ||||||
|  |     NFC_READING, | ||||||
|  |     NFC_READ_SUCCESS, | ||||||
|  |     NFC_READ_ERROR, | ||||||
|  |     NFC_WRITING, | ||||||
|  |     NFC_WRITE_SUCCESS, | ||||||
|  |     NFC_WRITE_ERROR | ||||||
|  | } nfcReaderStateType; | ||||||
|  |  | ||||||
| void startNfc(); | void startNfc(); | ||||||
| void scanRfidTask(void * parameter); | void scanRfidTask(void * parameter); | ||||||
| void startWriteJsonToTag(const char* payload); | void startWriteJsonToTag(const bool isSpoolTag, const char* payload); | ||||||
|  |  | ||||||
| extern TaskHandle_t RfidReaderTask; | extern TaskHandle_t RfidReaderTask; | ||||||
| extern String nfcJsonData; | extern String nfcJsonData; | ||||||
| extern String spoolId; | extern String activeSpoolId; | ||||||
| extern volatile uint8_t hasReadRfidTag; | extern String lastSpoolId; | ||||||
|  | extern volatile nfcReaderStateType nfcReaderState; | ||||||
| extern volatile bool pauseBambuMqttTask; | extern volatile bool pauseBambuMqttTask; | ||||||
|  | extern bool tagProcessed; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
							
								
								
									
										35
									
								
								src/ota.cpp
									
									
									
									
									
								
							
							
						
						| @@ -1,6 +1,10 @@ | |||||||
| #include <Arduino.h> | #include <Arduino.h> | ||||||
| #include <website.h> | #include <website.h> | ||||||
| #include <commonFS.h> | #include <commonFS.h> | ||||||
|  | #include "scale.h" | ||||||
|  | #include "bambu.h" | ||||||
|  | #include "nfc.h" | ||||||
|  |  | ||||||
|  |  | ||||||
| // Globale Variablen für Config Backups hinzufügen | // Globale Variablen für Config Backups hinzufügen | ||||||
| String bambuCredentialsBackup; | String bambuCredentialsBackup; | ||||||
| @@ -151,6 +155,25 @@ void handleUpdate(AsyncWebServer &server) { | |||||||
|  |  | ||||||
|     updateHandler->onUpload([](AsyncWebServerRequest *request, String filename, |     updateHandler->onUpload([](AsyncWebServerRequest *request, String filename, | ||||||
|                              size_t index, uint8_t *data, size_t len, bool final) { |                              size_t index, uint8_t *data, size_t len, bool final) { | ||||||
|  |  | ||||||
|  |         // Disable all Tasks | ||||||
|  |         if (BambuMqttTask != NULL)  | ||||||
|  |         { | ||||||
|  |             Serial.println("Delete BambuMqttTask"); | ||||||
|  |             vTaskDelete(BambuMqttTask); | ||||||
|  |             BambuMqttTask = NULL; | ||||||
|  |         } | ||||||
|  |         if (ScaleTask) { | ||||||
|  |             Serial.println("Delete ScaleTask"); | ||||||
|  |             vTaskDelete(ScaleTask); | ||||||
|  |             ScaleTask = NULL; | ||||||
|  |         } | ||||||
|  |         if (RfidReaderTask) { | ||||||
|  |             Serial.println("Delete RfidReaderTask"); | ||||||
|  |             vTaskDelete(RfidReaderTask); | ||||||
|  |             RfidReaderTask = NULL; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         if (!index) { |         if (!index) { | ||||||
|             updateTotalSize = request->contentLength(); |             updateTotalSize = request->contentLength(); | ||||||
|             updateWritten = 0; |             updateWritten = 0; | ||||||
| @@ -159,9 +182,9 @@ void handleUpdate(AsyncWebServer &server) { | |||||||
|             if (isSpiffsUpdate) { |             if (isSpiffsUpdate) { | ||||||
|                 // Backup vor dem Update |                 // Backup vor dem Update | ||||||
|                 sendUpdateProgress(0, "backup", "Backing up configurations..."); |                 sendUpdateProgress(0, "backup", "Backing up configurations..."); | ||||||
|                 delay(200); |                 vTaskDelay(200 / portTICK_PERIOD_MS); | ||||||
|                 backupJsonConfigs(); |                 backupJsonConfigs(); | ||||||
|                 delay(200); |                 vTaskDelay(200 / portTICK_PERIOD_MS); | ||||||
|                  |                  | ||||||
|                 const esp_partition_t *partition = esp_partition_find_first(ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_SPIFFS, NULL); |                 const esp_partition_t *partition = esp_partition_find_first(ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_SPIFFS, NULL); | ||||||
|                 if (!partition || !Update.begin(partition->size, U_SPIFFS)) { |                 if (!partition || !Update.begin(partition->size, U_SPIFFS)) { | ||||||
| @@ -169,14 +192,14 @@ void handleUpdate(AsyncWebServer &server) { | |||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|                 sendUpdateProgress(5, "starting", "Starting SPIFFS update..."); |                 sendUpdateProgress(5, "starting", "Starting SPIFFS update..."); | ||||||
|                 delay(200); |                 vTaskDelay(200 / portTICK_PERIOD_MS); | ||||||
|             } else { |             } else { | ||||||
|                 if (!Update.begin(updateTotalSize)) { |                 if (!Update.begin(updateTotalSize)) { | ||||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"Update initialization failed\"}"); |                     request->send(400, "application/json", "{\"success\":false,\"message\":\"Update initialization failed\"}"); | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|                 sendUpdateProgress(0, "starting", "Starting firmware update..."); |                 sendUpdateProgress(0, "starting", "Starting firmware update..."); | ||||||
|                 delay(200); |                 vTaskDelay(200 / portTICK_PERIOD_MS); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -201,8 +224,8 @@ void handleUpdate(AsyncWebServer &server) { | |||||||
|             static int lastProgress = -1; |             static int lastProgress = -1; | ||||||
|             if (currentProgress != lastProgress && (currentProgress % 10 == 0 || final)) { |             if (currentProgress != lastProgress && (currentProgress % 10 == 0 || final)) { | ||||||
|                 sendUpdateProgress(currentProgress, "uploading"); |                 sendUpdateProgress(currentProgress, "uploading"); | ||||||
|                 oledShowMessage("Update: " + String(currentProgress) + "%"); |                 oledShowProgressBar(currentProgress, 100, "Update", "Download"); | ||||||
|                 delay(50); |                 vTaskDelay(50 / portTICK_PERIOD_MS); | ||||||
|                 lastProgress = currentProgress; |                 lastProgress = currentProgress; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|   | |||||||
							
								
								
									
										151
									
								
								src/scale.cpp
									
									
									
									
									
								
							
							
						
						| @@ -15,14 +15,27 @@ int16_t weight = 0; | |||||||
|  |  | ||||||
| uint8_t weigthCouterToApi = 0; | uint8_t weigthCouterToApi = 0; | ||||||
| uint8_t scale_tare_counter = 0; | uint8_t scale_tare_counter = 0; | ||||||
|  | bool scaleTareRequest = false; | ||||||
| uint8_t pauseMainTask = 0; | uint8_t pauseMainTask = 0; | ||||||
| uint8_t scaleCalibrated = 1; | bool scaleCalibrated; | ||||||
|  | bool autoTare = true; | ||||||
| Preferences preferences; | bool scaleCalibrationActive = false; | ||||||
| const char* NVS_NAMESPACE = "scale"; |  | ||||||
| const char* NVS_KEY_CALIBRATION = "cal_value"; |  | ||||||
|  |  | ||||||
| // ##### Funktionen für Waage ##### | // ##### Funktionen für Waage ##### | ||||||
|  | uint8_t setAutoTare(bool autoTareValue) { | ||||||
|  |   Serial.print("Set AutoTare to "); | ||||||
|  |   Serial.println(autoTareValue); | ||||||
|  |   autoTare = autoTareValue; | ||||||
|  |  | ||||||
|  |   // Speichern mit NVS | ||||||
|  |   Preferences preferences; | ||||||
|  |   preferences.begin(NVS_NAMESPACE_SCALE, false); // false = readwrite | ||||||
|  |   preferences.putBool(NVS_KEY_AUTOTARE, autoTare); | ||||||
|  |   preferences.end(); | ||||||
|  |  | ||||||
|  |   return 1; | ||||||
|  | } | ||||||
|  |  | ||||||
| uint8_t tareScale() { | uint8_t tareScale() { | ||||||
|   Serial.println("Tare scale"); |   Serial.println("Tare scale"); | ||||||
|   scale.tare(); |   scale.tare(); | ||||||
| @@ -34,30 +47,64 @@ void scale_loop(void * parameter) { | |||||||
|   Serial.println("++++++++++++++++++++++++++++++"); |   Serial.println("++++++++++++++++++++++++++++++"); | ||||||
|   Serial.println("Scale Loop started"); |   Serial.println("Scale Loop started"); | ||||||
|   Serial.println("++++++++++++++++++++++++++++++"); |   Serial.println("++++++++++++++++++++++++++++++"); | ||||||
|  |  | ||||||
|  |   scale.tare(); | ||||||
|  |  | ||||||
|   for(;;) { |   for(;;) { | ||||||
|     if (scale.is_ready())  |     if (scale.is_ready())  | ||||||
|     { |     { | ||||||
|       // Waage nochmal Taren, wenn zu lange Abweichung |       // Waage automatisch Taren, wenn zu lange Abweichung | ||||||
|       if (scale_tare_counter >= 5)  |       if (autoTare && scale_tare_counter >= 5)  | ||||||
|       { |       { | ||||||
|  |         Serial.println("Auto Tare scale"); | ||||||
|         scale.tare(); |         scale.tare(); | ||||||
|         scale_tare_counter = 0; |         scale_tare_counter = 0; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       weight = round(scale.get_units()); |       // Waage manuell Taren | ||||||
|  |       if (scaleTareRequest == true)  | ||||||
|  |       { | ||||||
|  |         Serial.println("Re-Tare scale"); | ||||||
|  |         oledShowMessage("TARE Scale"); | ||||||
|  |         vTaskDelay(pdMS_TO_TICKS(1000)); | ||||||
|  |         scale.tare(); | ||||||
|  |         vTaskDelay(pdMS_TO_TICKS(1000)); | ||||||
|  |         oledShowWeight(0); | ||||||
|  |         scaleTareRequest = false; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // Only update weight if median changed more than 1 | ||||||
|  |       int16_t newWeight = round(scale.get_units()); | ||||||
|  |       if(abs(weight-newWeight) > 1){ | ||||||
|  |         weight = newWeight; | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     vTaskDelay(pdMS_TO_TICKS(100)); |     vTaskDelay(pdMS_TO_TICKS(100)); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| uint8_t start_scale() { | void start_scale(bool touchSensorConnected) { | ||||||
|   Serial.println("Prüfe Calibration Value"); |   Serial.println("Prüfe Calibration Value"); | ||||||
|   long calibrationValue; |   float calibrationValue; | ||||||
|  |  | ||||||
|   // NVS lesen |   // NVS lesen | ||||||
|   preferences.begin(NVS_NAMESPACE, true); // true = readonly |   Preferences preferences; | ||||||
|   calibrationValue = preferences.getLong(NVS_KEY_CALIBRATION, defaultScaleCalibrationValue); |   preferences.begin(NVS_NAMESPACE_SCALE, true); // true = readonly | ||||||
|  |   if(preferences.isKey(NVS_KEY_CALIBRATION)){ | ||||||
|  |     calibrationValue = preferences.getFloat(NVS_KEY_CALIBRATION); | ||||||
|  |     scaleCalibrated = true; | ||||||
|  |   }else{ | ||||||
|  |     calibrationValue = SCALE_DEFAULT_CALIBRATION_VALUE; | ||||||
|  |     scaleCalibrated = false; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   // auto Tare | ||||||
|  |   // Wenn Touch Sensor verbunden, dann autoTare auf false setzen | ||||||
|  |   // Danach prüfen was in NVS gespeichert ist | ||||||
|  |   autoTare = (touchSensorConnected) ? false : true; | ||||||
|  |   autoTare = preferences.getBool(NVS_KEY_AUTOTARE, autoTare); | ||||||
|  |  | ||||||
|   preferences.end(); |   preferences.end(); | ||||||
|  |  | ||||||
|   Serial.print("Read Scale Calibration Value "); |   Serial.print("Read Scale Calibration Value "); | ||||||
| @@ -65,12 +112,7 @@ uint8_t start_scale() { | |||||||
|  |  | ||||||
|   scale.begin(LOADCELL_DOUT_PIN, LOADCELL_SCK_PIN); |   scale.begin(LOADCELL_DOUT_PIN, LOADCELL_SCK_PIN); | ||||||
|  |  | ||||||
|   if (isnan(calibrationValue) || calibrationValue < 1) { |   oledShowProgressBar(6, 7, DISPLAY_BOOT_TEXT, "Tare scale"); | ||||||
|     calibrationValue = defaultScaleCalibrationValue; |  | ||||||
|     scaleCalibrated = 0; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   oledShowMessage("Scale Tare Please remove all"); |  | ||||||
|   for (uint16_t i = 0; i < 2000; i++) { |   for (uint16_t i = 0; i < 2000; i++) { | ||||||
|     yield(); |     yield(); | ||||||
|     vTaskDelay(pdMS_TO_TICKS(1)); |     vTaskDelay(pdMS_TO_TICKS(1)); | ||||||
| @@ -80,7 +122,7 @@ uint8_t start_scale() { | |||||||
|   if (scale.wait_ready_timeout(1000)) |   if (scale.wait_ready_timeout(1000)) | ||||||
|   { |   { | ||||||
|     scale.set_scale(calibrationValue); // this value is obtained by calibrating the scale with known weights; see the README for details |     scale.set_scale(calibrationValue); // this value is obtained by calibrating the scale with known weights; see the README for details | ||||||
|     scale.tare(); |     //scale.tare(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Display Gewicht |   // Display Gewicht | ||||||
| @@ -101,23 +143,25 @@ uint8_t start_scale() { | |||||||
|   } else { |   } else { | ||||||
|       Serial.println("ScaleLoop-Task erfolgreich erstellt"); |       Serial.println("ScaleLoop-Task erfolgreich erstellt"); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return (scaleCalibrated == 1) ? 1 : 3; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| uint8_t calibrate_scale() { | uint8_t calibrate_scale() { | ||||||
|   long newCalibrationValue; |   uint8_t returnState = 0; | ||||||
|  |   float newCalibrationValue; | ||||||
|  |  | ||||||
|  |   scaleCalibrationActive = true; | ||||||
|  |  | ||||||
|  |   vTaskSuspend(RfidReaderTask); | ||||||
|  |   vTaskSuspend(ScaleTask); | ||||||
|  |  | ||||||
|   //vTaskSuspend(RfidReaderTask); |  | ||||||
|   vTaskDelete(RfidReaderTask); |  | ||||||
|   vTaskDelete(ScaleTask); |  | ||||||
|   pauseBambuMqttTask = true; |   pauseBambuMqttTask = true; | ||||||
|   pauseMainTask = 1; |   pauseMainTask = 1; | ||||||
|    |    | ||||||
|   if (scale.wait_ready_timeout(1000)) |   if (scale.wait_ready_timeout(1000)) | ||||||
|   { |   { | ||||||
|  |      | ||||||
|     scale.set_scale(); |     scale.set_scale(); | ||||||
|     oledShowMessage("Step 1 empty Scale"); |     oledShowProgressBar(0, 3, "Scale Cal.", "Empty Scale"); | ||||||
|  |  | ||||||
|     for (uint16_t i = 0; i < 5000; i++) { |     for (uint16_t i = 0; i < 5000; i++) { | ||||||
|       yield(); |       yield(); | ||||||
| @@ -129,7 +173,7 @@ uint8_t calibrate_scale() { | |||||||
|     Serial.println("Tare done..."); |     Serial.println("Tare done..."); | ||||||
|     Serial.print("Place a known weight on the scale..."); |     Serial.print("Place a known weight on the scale..."); | ||||||
|  |  | ||||||
|     oledShowMessage("Step 2 Place the weight"); |     oledShowProgressBar(1, 3, "Scale Cal.", "Place the weight"); | ||||||
|  |  | ||||||
|     for (uint16_t i = 0; i < 5000; i++) { |     for (uint16_t i = 0; i < 5000; i++) { | ||||||
|       yield(); |       yield(); | ||||||
| @@ -137,7 +181,7 @@ uint8_t calibrate_scale() { | |||||||
|       esp_task_wdt_reset(); |       esp_task_wdt_reset(); | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     long newCalibrationValue = scale.get_units(10); |     float newCalibrationValue = scale.get_units(10); | ||||||
|     Serial.print("Result: "); |     Serial.print("Result: "); | ||||||
|     Serial.println(newCalibrationValue); |     Serial.println(newCalibrationValue); | ||||||
|  |  | ||||||
| @@ -149,21 +193,33 @@ uint8_t calibrate_scale() { | |||||||
|       Serial.println(newCalibrationValue); |       Serial.println(newCalibrationValue); | ||||||
|  |  | ||||||
|       // Speichern mit NVS |       // Speichern mit NVS | ||||||
|       preferences.begin(NVS_NAMESPACE, false); // false = readwrite |       Preferences preferences; | ||||||
|       preferences.putLong(NVS_KEY_CALIBRATION, newCalibrationValue); |       preferences.begin(NVS_NAMESPACE_SCALE, false); // false = readwrite | ||||||
|  |       preferences.putFloat(NVS_KEY_CALIBRATION, newCalibrationValue); | ||||||
|       preferences.end(); |       preferences.end(); | ||||||
|  |  | ||||||
|       // Verifizieren |       // Verifizieren | ||||||
|       preferences.begin(NVS_NAMESPACE, true); |       preferences.begin(NVS_NAMESPACE_SCALE, true); | ||||||
|       long verifyValue = preferences.getLong(NVS_KEY_CALIBRATION, 0); |       float verifyValue = preferences.getFloat(NVS_KEY_CALIBRATION, 0); | ||||||
|       preferences.end(); |       preferences.end(); | ||||||
|  |  | ||||||
|       Serial.print("Verified stored value: "); |       Serial.print("Verified stored value: "); | ||||||
|       Serial.println(verifyValue); |       Serial.println(verifyValue); | ||||||
|  |  | ||||||
|       Serial.println("End calibration, revome weight"); |       oledShowProgressBar(2, 3, "Scale Cal.", "Remove weight"); | ||||||
|  |  | ||||||
|       oledShowMessage("Remove weight"); |       scale.set_scale(newCalibrationValue); | ||||||
|  |       for (uint16_t i = 0; i < 2000; i++) { | ||||||
|  |         yield(); | ||||||
|  |         vTaskDelay(pdMS_TO_TICKS(1)); | ||||||
|  |         esp_task_wdt_reset(); | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       oledShowProgressBar(3, 3, "Scale Cal.", "Completed"); | ||||||
|  |  | ||||||
|  |       // For some reason it is not possible to re-tare the scale here, it will result in a wdt timeout. Instead let the scale loop do the taring | ||||||
|  |       //scale.tare(); | ||||||
|  |       scaleTareRequest = true; | ||||||
|  |  | ||||||
|       for (uint16_t i = 0; i < 2000; i++) { |       for (uint16_t i = 0; i < 2000; i++) { | ||||||
|         yield(); |         yield(); | ||||||
| @@ -171,28 +227,21 @@ uint8_t calibrate_scale() { | |||||||
|         esp_task_wdt_reset(); |         esp_task_wdt_reset(); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       oledShowMessage("Calibration done"); |       scaleCalibrated = true; | ||||||
|  |       returnState = 1; | ||||||
|       for (uint16_t i = 0; i < 2000; i++) { |  | ||||||
|         yield(); |  | ||||||
|         vTaskDelay(pdMS_TO_TICKS(1)); |  | ||||||
|         esp_task_wdt_reset(); |  | ||||||
|       } |  | ||||||
|     } |     } | ||||||
|     else |     else | ||||||
|     { |  | ||||||
|     { |     { | ||||||
|       Serial.println("Calibration value is invalid. Please recalibrate."); |       Serial.println("Calibration value is invalid. Please recalibrate."); | ||||||
|  |  | ||||||
|         oledShowMessage("Calibration ERROR Try again"); |       oledShowProgressBar(3, 3, "Failure", "Calibration error"); | ||||||
|  |  | ||||||
|       for (uint16_t i = 0; i < 50000; i++) { |       for (uint16_t i = 0; i < 50000; i++) { | ||||||
|         yield(); |         yield(); | ||||||
|         vTaskDelay(pdMS_TO_TICKS(1)); |         vTaskDelay(pdMS_TO_TICKS(1)); | ||||||
|         esp_task_wdt_reset(); |         esp_task_wdt_reset(); | ||||||
|       } |       } | ||||||
|         return 0; |       returnState = 0; | ||||||
|       } |  | ||||||
|     }  |     }  | ||||||
|   } |   } | ||||||
|   else  |   else  | ||||||
| @@ -206,16 +255,14 @@ uint8_t calibrate_scale() { | |||||||
|       vTaskDelay(pdMS_TO_TICKS(1)); |       vTaskDelay(pdMS_TO_TICKS(1)); | ||||||
|       esp_task_wdt_reset(); |       esp_task_wdt_reset(); | ||||||
|     } |     } | ||||||
|     return 0; |     returnState = 0; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   oledShowMessage("Scale Ready"); |   vTaskResume(RfidReaderTask); | ||||||
|    |   vTaskResume(ScaleTask); | ||||||
|   Serial.println("restart Scale Task"); |  | ||||||
|   start_scale(); |  | ||||||
|  |  | ||||||
|   pauseBambuMqttTask = false; |   pauseBambuMqttTask = false; | ||||||
|   pauseMainTask = 0; |   pauseMainTask = 0; | ||||||
|  |   scaleCalibrationActive = false; | ||||||
|  |  | ||||||
|   return 1; |   return returnState; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -4,8 +4,8 @@ | |||||||
| #include <Arduino.h> | #include <Arduino.h> | ||||||
| #include "HX711.h" | #include "HX711.h" | ||||||
|  |  | ||||||
|  | uint8_t setAutoTare(bool autoTareValue); | ||||||
| uint8_t start_scale(); | uint8_t start_scale(bool touchSensorConnected); | ||||||
| uint8_t calibrate_scale(); | uint8_t calibrate_scale(); | ||||||
| uint8_t tareScale(); | uint8_t tareScale(); | ||||||
|  |  | ||||||
| @@ -13,8 +13,11 @@ extern HX711 scale; | |||||||
| extern int16_t weight; | extern int16_t weight; | ||||||
| extern uint8_t weigthCouterToApi; | extern uint8_t weigthCouterToApi; | ||||||
| extern uint8_t scale_tare_counter; | extern uint8_t scale_tare_counter; | ||||||
|  | extern uint8_t scaleTareRequest; | ||||||
| extern uint8_t pauseMainTask; | extern uint8_t pauseMainTask; | ||||||
| extern uint8_t scaleCalibrated; | extern bool scaleCalibrated; | ||||||
|  | extern bool autoTare; | ||||||
|  | extern bool scaleCalibrationActive; | ||||||
|  |  | ||||||
| extern TaskHandle_t ScaleTask; | extern TaskHandle_t ScaleTask; | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										148
									
								
								src/website.cpp
									
									
									
									
									
								
							
							
						
						| @@ -10,6 +10,9 @@ | |||||||
| #include <Update.h> | #include <Update.h> | ||||||
| #include "display.h" | #include "display.h" | ||||||
| #include "ota.h" | #include "ota.h" | ||||||
|  | #include "config.h" | ||||||
|  | #include "debug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
| #ifndef VERSION | #ifndef VERSION | ||||||
|   #define VERSION "1.1.0" |   #define VERSION "1.1.0" | ||||||
| @@ -22,17 +25,22 @@ AsyncWebServer server(webserverPort); | |||||||
| AsyncWebSocket ws("/ws"); | AsyncWebSocket ws("/ws"); | ||||||
|  |  | ||||||
| uint8_t lastSuccess = 0; | uint8_t lastSuccess = 0; | ||||||
| uint8_t lastHasReadRfidTag = 0; | nfcReaderStateType lastnfcReaderState = NFC_IDLE; | ||||||
|  |  | ||||||
|  |  | ||||||
| void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventType type, void *arg, uint8_t *data, size_t len) { | void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventType type, void *arg, uint8_t *data, size_t len) { | ||||||
|  |     HEAP_DEBUG_MESSAGE("onWsEvent begin"); | ||||||
|     if (type == WS_EVT_CONNECT) { |     if (type == WS_EVT_CONNECT) { | ||||||
|         Serial.println("Neuer Client verbunden!"); |         Serial.println("Neuer Client verbunden!"); | ||||||
|         // Sende die AMS-Daten an den neuen Client |         // Sende die AMS-Daten an den neuen Client | ||||||
|         sendAmsData(client); |         if (!bambuDisabled) sendAmsData(client); | ||||||
|         sendNfcData(client); |         sendNfcData(); | ||||||
|         foundNfcTag(client, 0); |         foundNfcTag(client, 0); | ||||||
|         sendWriteResult(client, 3); |         sendWriteResult(client, 3); | ||||||
|  |  | ||||||
|  |         // Clean up dead connections | ||||||
|  |         (*server).cleanupClients(); | ||||||
|  |         Serial.println("Currently connected number of clients: " + String((*server).getClients().size())); | ||||||
|     } else if (type == WS_EVT_DISCONNECT) { |     } else if (type == WS_EVT_DISCONNECT) { | ||||||
|         Serial.println("Client getrennt."); |         Serial.println("Client getrennt."); | ||||||
|     } else if (type == WS_EVT_ERROR) { |     } else if (type == WS_EVT_ERROR) { | ||||||
| @@ -40,9 +48,15 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | |||||||
|     } else if (type == WS_EVT_PONG) { |     } else if (type == WS_EVT_PONG) { | ||||||
|         Serial.printf("WebSocket Client #%u pong\n", client->id()); |         Serial.printf("WebSocket Client #%u pong\n", client->id()); | ||||||
|     } else if (type == WS_EVT_DATA) { |     } else if (type == WS_EVT_DATA) { | ||||||
|         String message = String((char*)data); |  | ||||||
|         JsonDocument doc; |         JsonDocument doc; | ||||||
|         deserializeJson(doc, message); |         DeserializationError error = deserializeJson(doc, (char*)data, len); | ||||||
|  |         //String message = String((char*)data); | ||||||
|  |         //deserializeJson(doc, message); | ||||||
|  |  | ||||||
|  |         if (error) { | ||||||
|  |             Serial.println("JSON deserialization failed: " + String(error.c_str())); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         if (doc["type"] == "heartbeat") { |         if (doc["type"] == "heartbeat") { | ||||||
|             // Sende Heartbeat-Antwort |             // Sende Heartbeat-Antwort | ||||||
| @@ -50,7 +64,7 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | |||||||
|                 "\"type\":\"heartbeat\"," |                 "\"type\":\"heartbeat\"," | ||||||
|                 "\"freeHeap\":" + String(ESP.getFreeHeap()/1024) + "," |                 "\"freeHeap\":" + String(ESP.getFreeHeap()/1024) + "," | ||||||
|                 "\"bambu_connected\":" + String(bambu_connected) + "," |                 "\"bambu_connected\":" + String(bambu_connected) + "," | ||||||
|                 "\"spoolman_connected\":" + String(spoolman_connected) + "" |                 "\"spoolman_connected\":" + String(spoolmanConnected) + "" | ||||||
|                 "}"); |                 "}"); | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -59,7 +73,8 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | |||||||
|                 // Versuche NFC-Daten zu schreiben |                 // Versuche NFC-Daten zu schreiben | ||||||
|                 String payloadString; |                 String payloadString; | ||||||
|                 serializeJson(doc["payload"], payloadString); |                 serializeJson(doc["payload"], payloadString); | ||||||
|                 startWriteJsonToTag(payloadString.c_str()); |  | ||||||
|  |                 startWriteJsonToTag((doc["tagType"] == "spool") ? true : false, payloadString.c_str()); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -73,6 +88,10 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | |||||||
|                 success = calibrate_scale(); |                 success = calibrate_scale(); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  |             if (doc["payload"] == "setAutoTare") { | ||||||
|  |                 success = setAutoTare(doc["enabled"].as<bool>()); | ||||||
|  |             } | ||||||
|  |  | ||||||
|             if (success) { |             if (success) { | ||||||
|                 ws.textAll("{\"type\":\"scale\",\"payload\":\"success\"}"); |                 ws.textAll("{\"type\":\"scale\",\"payload\":\"success\"}"); | ||||||
|             } else { |             } else { | ||||||
| @@ -107,7 +126,9 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | |||||||
|         else { |         else { | ||||||
|             Serial.println("Unbekannter WebSocket-Typ: " + doc["type"].as<String>()); |             Serial.println("Unbekannter WebSocket-Typ: " + doc["type"].as<String>()); | ||||||
|         } |         } | ||||||
|  |         doc.clear(); | ||||||
|     } |     } | ||||||
|  |     HEAP_DEBUG_MESSAGE("onWsEvent end"); | ||||||
| } | } | ||||||
|  |  | ||||||
| // Funktion zum Laden und Ersetzen des Headers in einer HTML-Datei | // Funktion zum Laden und Ersetzen des Headers in einer HTML-Datei | ||||||
| @@ -134,39 +155,36 @@ void sendWriteResult(AsyncWebSocketClient *client, uint8_t success) { | |||||||
| void foundNfcTag(AsyncWebSocketClient *client, uint8_t success) { | void foundNfcTag(AsyncWebSocketClient *client, uint8_t success) { | ||||||
|     if (success == lastSuccess) return; |     if (success == lastSuccess) return; | ||||||
|     ws.textAll("{\"type\":\"nfcTag\", \"payload\":{\"found\": " + String(success) + "}}"); |     ws.textAll("{\"type\":\"nfcTag\", \"payload\":{\"found\": " + String(success) + "}}"); | ||||||
|     sendNfcData(nullptr); |     sendNfcData(); | ||||||
|     lastSuccess = success; |     lastSuccess = success; | ||||||
| } | } | ||||||
|  |  | ||||||
| void sendNfcData(AsyncWebSocketClient *client) { | void sendNfcData() { | ||||||
|     if (lastHasReadRfidTag == hasReadRfidTag) return; |     if (lastnfcReaderState == nfcReaderState) return; | ||||||
|     if (hasReadRfidTag == 0) { |     // TBD: Why is there no status for reading the tag? | ||||||
|  |     switch(nfcReaderState){ | ||||||
|  |         case NFC_IDLE: | ||||||
|             ws.textAll("{\"type\":\"nfcData\", \"payload\":{}}"); |             ws.textAll("{\"type\":\"nfcData\", \"payload\":{}}"); | ||||||
|     } |             break; | ||||||
|     else if (hasReadRfidTag == 1) { |         case NFC_READ_SUCCESS: | ||||||
|             ws.textAll("{\"type\":\"nfcData\", \"payload\":" + nfcJsonData + "}"); |             ws.textAll("{\"type\":\"nfcData\", \"payload\":" + nfcJsonData + "}"); | ||||||
|     } |             break; | ||||||
|     else if (hasReadRfidTag == 2) |         case NFC_READ_ERROR: | ||||||
|     { |  | ||||||
|             ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"error\":\"Empty Tag or Data not readable\"}}"); |             ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"error\":\"Empty Tag or Data not readable\"}}"); | ||||||
|     } |             break; | ||||||
|     else if (hasReadRfidTag == 3) |         case NFC_WRITING: | ||||||
|     { |  | ||||||
|             ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"info\":\"Schreibe Tag...\"}}"); |             ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"info\":\"Schreibe Tag...\"}}"); | ||||||
|     } |             break; | ||||||
|     else if (hasReadRfidTag == 4) |         case NFC_WRITE_SUCCESS: | ||||||
|     { |  | ||||||
|         ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"error\":\"Error writing to Tag\"}}"); |  | ||||||
|     } |  | ||||||
|     else if (hasReadRfidTag == 5) |  | ||||||
|     { |  | ||||||
|             ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"info\":\"Tag erfolgreich geschrieben\"}}"); |             ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"info\":\"Tag erfolgreich geschrieben\"}}"); | ||||||
|     } |             break; | ||||||
|     else  |         case NFC_WRITE_ERROR: | ||||||
|     { |             ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"error\":\"Error writing to Tag\"}}"); | ||||||
|  |             break; | ||||||
|  |         case DEFAULT: | ||||||
|             ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"error\":\"Something went wrong\"}}"); |             ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"error\":\"Something went wrong\"}}"); | ||||||
|     } |     } | ||||||
|     lastHasReadRfidTag = hasReadRfidTag; |     lastnfcReaderState = nfcReaderState; | ||||||
| } | } | ||||||
|  |  | ||||||
| void sendAmsData(AsyncWebSocketClient *client) { | void sendAmsData(AsyncWebSocketClient *client) { | ||||||
| @@ -176,6 +194,7 @@ void sendAmsData(AsyncWebSocketClient *client) { | |||||||
| } | } | ||||||
|  |  | ||||||
| void setupWebserver(AsyncWebServer &server) { | void setupWebserver(AsyncWebServer &server) { | ||||||
|  |     oledShowProgressBar(2, 7, DISPLAY_BOOT_TEXT, "Webserver init"); | ||||||
|     // Deaktiviere alle Debug-Ausgaben |     // Deaktiviere alle Debug-Ausgaben | ||||||
|     Serial.setDebugOutput(false); |     Serial.setDebugOutput(false); | ||||||
|      |      | ||||||
| @@ -192,6 +211,9 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|     Serial.print("Geladene Spoolman-URL: "); |     Serial.print("Geladene Spoolman-URL: "); | ||||||
|     Serial.println(spoolmanUrl); |     Serial.println(spoolmanUrl); | ||||||
|  |  | ||||||
|  |     // Load Bamb credentials: | ||||||
|  |     loadBambuCredentials(); | ||||||
|  |  | ||||||
|     // Route für about |     // Route für about | ||||||
|     server.on("/about", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/about", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|         Serial.println("Anfrage für /about erhalten"); |         Serial.println("Anfrage für /about erhalten"); | ||||||
| @@ -204,16 +226,23 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|     // Route für Waage |     // Route für Waage | ||||||
|     server.on("/waage", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/waage", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|         Serial.println("Anfrage für /waage erhalten"); |         Serial.println("Anfrage für /waage erhalten"); | ||||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/waage.html.gz", "text/html"); |         //AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/waage.html.gz", "text/html"); | ||||||
|         response->addHeader("Content-Encoding", "gzip"); |         //response->addHeader("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); |         //response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |  | ||||||
|  |         String html = loadHtmlWithHeader("/waage.html"); | ||||||
|  |         html.replace("{{autoTare}}", (autoTare) ? "checked" : ""); | ||||||
|  |  | ||||||
|  |         request->send(200, "text/html", html); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     // Route für RFID |     // Route für RFID | ||||||
|     server.on("/", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|         Serial.println("Anfrage für /rfid erhalten"); |         Serial.println("Anfrage für /rfid erhalten"); | ||||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/rfid.html.gz", "text/html"); |          | ||||||
|  |         String page = (bambuDisabled) ? "/rfid.html.gz" : "/rfid_bambu.html.gz"; | ||||||
|  |         AsyncWebServerResponse *response = request->beginResponse(LittleFS, page, "text/html"); | ||||||
|  |          | ||||||
|         response->addHeader("Content-Encoding", "gzip"); |         response->addHeader("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); |         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |         request->send(response); | ||||||
| @@ -244,31 +273,11 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|         html.replace("{{spoolmanOctoUrl}}", (octoUrl != "") ? octoUrl : ""); |         html.replace("{{spoolmanOctoUrl}}", (octoUrl != "") ? octoUrl : ""); | ||||||
|         html.replace("{{spoolmanOctoToken}}", (octoToken != "") ? octoToken : ""); |         html.replace("{{spoolmanOctoToken}}", (octoToken != "") ? octoToken : ""); | ||||||
|  |  | ||||||
|         JsonDocument doc; |         html.replace("{{bambuIp}}", bambuCredentials.ip);             | ||||||
|         if (loadJsonValue("/bambu_credentials.json", doc) && doc["bambu_ip"].is<String>())  |         html.replace("{{bambuSerial}}", bambuCredentials.serial); | ||||||
|         { |         html.replace("{{bambuCode}}", bambuCredentials.accesscode ? bambuCredentials.accesscode : ""); | ||||||
|             String bambuIp = doc["bambu_ip"].as<String>(); |         html.replace("{{autoSendToBambu}}", bambuCredentials.autosend_enable ? "checked" : ""); | ||||||
|             String bambuSerial = doc["bambu_serialnr"].as<String>(); |         html.replace("{{autoSendTime}}", (bambuCredentials.autosend_time != 0) ? String(bambuCredentials.autosend_time) : String(BAMBU_DEFAULT_AUTOSEND_TIME)); | ||||||
|             String bambuCode = doc["bambu_accesscode"].as<String>(); |  | ||||||
|             autoSendToBambu = doc["autoSendToBambu"].as<bool>(); |  | ||||||
|             bambuIp.trim(); |  | ||||||
|             bambuSerial.trim(); |  | ||||||
|             bambuCode.trim(); |  | ||||||
|  |  | ||||||
|             html.replace("{{bambuIp}}", bambuIp ? bambuIp : "");             |  | ||||||
|             html.replace("{{bambuSerial}}", bambuSerial ? bambuSerial : ""); |  | ||||||
|             html.replace("{{bambuCode}}", bambuCode ? bambuCode : ""); |  | ||||||
|             html.replace("{{autoSendToBambu}}", autoSendToBambu ? "checked" : ""); |  | ||||||
|             html.replace("{{autoSendTime}}", String(autoSetBambuAmsCounter)); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             html.replace("{{bambuIp}}", ""); |  | ||||||
|             html.replace("{{bambuSerial}}", ""); |  | ||||||
|             html.replace("{{bambuCode}}", ""); |  | ||||||
|             html.replace("{{autoSendToBambu}}", ""); |  | ||||||
|             html.replace("{{autoSendTime}}", String(autoSetBambuAmsCounter)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         request->send(200, "text/html", html); |         request->send(200, "text/html", html); | ||||||
|     }); |     }); | ||||||
| @@ -286,6 +295,14 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         String url = request->getParam("url")->value(); |         String url = request->getParam("url")->value(); | ||||||
|  |         if (url.indexOf("http://") == -1 && url.indexOf("https://") == -1) { | ||||||
|  |             url = "http://" + url; | ||||||
|  |         } | ||||||
|  |         // Remove trailing slash if exists | ||||||
|  |         if (url.length() > 0 && url.charAt(url.length()-1) == '/') { | ||||||
|  |             url = url.substring(0, url.length()-1); | ||||||
|  |         } | ||||||
|  |          | ||||||
|         bool octoEnabled = (request->getParam("octoEnabled")->value() == "true") ? true : false; |         bool octoEnabled = (request->getParam("octoEnabled")->value() == "true") ? true : false; | ||||||
|         String octoUrl = request->getParam("octoUrl")->value(); |         String octoUrl = request->getParam("octoUrl")->value(); | ||||||
|         String octoToken = (request->getParam("octoToken")->value() != "") ? request->getParam("octoToken")->value() : ""; |         String octoToken = (request->getParam("octoToken")->value() != "") ? request->getParam("octoToken")->value() : ""; | ||||||
| @@ -300,8 +317,17 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|         request->send(200, "application/json", jsonResponse); |         request->send(200, "application/json", jsonResponse); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     // Route für das Überprüfen der Spoolman-Instanz |     // Route für das Überprüfen der Bambu-Instanz | ||||||
|     server.on("/api/bambu", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/api/bambu", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|  |         if (request->hasParam("remove")) { | ||||||
|  |             if (removeBambuCredentials()) { | ||||||
|  |                 request->send(200, "application/json", "{\"success\": true}"); | ||||||
|  |             } else { | ||||||
|  |                 request->send(500, "application/json", "{\"success\": false, \"error\": \"Fehler beim Löschen der Bambu-Credentials\"}"); | ||||||
|  |             } | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         if (!request->hasParam("bambu_ip") || !request->hasParam("bambu_serialnr") || !request->hasParam("bambu_accesscode")) { |         if (!request->hasParam("bambu_ip") || !request->hasParam("bambu_serialnr") || !request->hasParam("bambu_accesscode")) { | ||||||
|             request->send(400, "application/json", "{\"success\": false, \"error\": \"Missing parameter\"}"); |             request->send(400, "application/json", "{\"success\": false, \"error\": \"Missing parameter\"}"); | ||||||
|             return; |             return; | ||||||
|   | |||||||
| @@ -24,7 +24,7 @@ void setupWebserver(AsyncWebServer &server); | |||||||
|  |  | ||||||
| // WebSocket-Funktionen | // WebSocket-Funktionen | ||||||
| void sendAmsData(AsyncWebSocketClient *client); | void sendAmsData(AsyncWebSocketClient *client); | ||||||
| void sendNfcData(AsyncWebSocketClient *client); | void sendNfcData(); | ||||||
| void foundNfcTag(AsyncWebSocketClient *client, uint8_t success); | void foundNfcTag(AsyncWebSocketClient *client, uint8_t success); | ||||||
| void sendWriteResult(AsyncWebSocketClient *client, uint8_t success); | void sendWriteResult(AsyncWebSocketClient *client, uint8_t success); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -59,10 +59,9 @@ void initWiFi() { | |||||||
|   if(wm_nonblocking) wm.setConfigPortalBlocking(false); |   if(wm_nonblocking) wm.setConfigPortalBlocking(false); | ||||||
|   //wm.setConfigPortalTimeout(320); // Portal nach 5min schließen |   //wm.setConfigPortalTimeout(320); // Portal nach 5min schließen | ||||||
|   wm.setWiFiAutoReconnect(true); |   wm.setWiFiAutoReconnect(true); | ||||||
|   wm.setConnectTimeout(5); |   wm.setConnectTimeout(10); | ||||||
|  |  | ||||||
|   oledShowTopRow(); |   oledShowProgressBar(1, 7, DISPLAY_BOOT_TEXT, "WiFi init"); | ||||||
|   oledShowMessage("WiFi Setup"); |  | ||||||
|    |    | ||||||
|   //bool res = wm.autoConnect("FilaMan"); // anonymous ap |   //bool res = wm.autoConnect("FilaMan"); // anonymous ap | ||||||
|   if(!wm.autoConnect("FilaMan")) { |   if(!wm.autoConnect("FilaMan")) { | ||||||
| @@ -80,9 +79,6 @@ void initWiFi() { | |||||||
|     Serial.println(WiFi.localIP()); |     Serial.println(WiFi.localIP()); | ||||||
|  |  | ||||||
|     oledShowTopRow(); |     oledShowTopRow(); | ||||||
|     display.display(); |  | ||||||
|  |  | ||||||
|     vTaskDelay(500 / portTICK_PERIOD_MS); |  | ||||||
|  |  | ||||||
|     // mDNS |     // mDNS | ||||||
|     startMDNS(); |     startMDNS(); | ||||||
|   | |||||||
							
								
								
									
										6432
									
								
								usermod/spitzbirne32/CAD/Base_usermod_spitzbirne32.stp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										16385
									
								
								usermod/spitzbirne32/CAD/FilaMan-Scale_usermod_spitzbirne32.stp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										5278
									
								
								usermod/spitzbirne32/CAD/Housing_usermod_spitzbirne32.stp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										0
									
								
								usermod/spitzbirne32/CAD/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										4888
									
								
								usermod/spitzbirne32/CAD/ScaleTop_usermod_spitzbirne32.stp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 540 KiB | 
| After Width: | Height: | Size: 525 KiB | 
| After Width: | Height: | Size: 7.9 MiB | 
| After Width: | Height: | Size: 183 KiB | 
							
								
								
									
										12
									
								
								usermod/spitzbirne32/Images/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | |||||||
|  | ## **Heat insert location** | ||||||
|  |  | ||||||
|  | Housing:  | ||||||
|  | - every hole is made to fit a heat insert | ||||||
|  |  | ||||||
|  |  | ||||||
|  | --- | ||||||
|  | Scale top:  | ||||||
|  | - two heat inserts for the NFC Reader | ||||||
|  |  | ||||||
|  |    | ||||||
|  |  | ||||||
| After Width: | Height: | Size: 491 KiB | 
| After Width: | Height: | Size: 834 KiB | 
							
								
								
									
										
											BIN
										
									
								
								usermod/spitzbirne32/Images/Showcase_usermod_spitzbirne32.gif
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.9 MiB | 
							
								
								
									
										69
									
								
								usermod/spitzbirne32/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,69 @@ | |||||||
|  | ## Modifications | ||||||
|  |  | ||||||
|  | To reduce costs, components were sourced from AliExpress instead of Amazon. However, differences in dimensions and mounting hole spacing necessitated adjustments to the 3D-printed parts. Additionally M3 heat inserts were used to limit M4 screws to a minimum. | ||||||
|  |  | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | List of parts that were used: | ||||||
|  | - Display: https://aliexpress.com/item/1005007389730469.html | ||||||
|  | - Scale(5KG with HX711): https://aliexpress.com/item/1005006827930173.html | ||||||
|  | - NFC Reader: https://aliexpress.com/item/1005005973913526.html | ||||||
|  | - NFC Chips: https://aliexpress.com/item/1005006332360160.html | ||||||
|  | - [VORON](https://vorondesign.com/) Heat Inserts M3 OD5mm L4mm: https://aliexpress.com/item/1005003582355741.html  - make sure to select the correct size | ||||||
|  |  | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | - **Parts are designed to be printed in ABS/ASA.** Shrinking compensation not needed. | ||||||
|  |  | ||||||
|  | - **Display and Scale Adjustments:** The AliExpress-sourced display and scale had different dimensions and hole spacings compared to the Amazon versions. The 3D models were modified to accommodate these differences, ensuring proper fit and functionality. | ||||||
|  |    -  measurement of my Display & Scale to check if your parts will fit can be found in the images folder | ||||||
|  |  | ||||||
|  | - **Screw Size and Heat Inserts:** All holes originally designed for M4 screws were resized to fit M3 screws. Standard VORON heat inserts were incorporated to provide durable threading. This change standardizes the hardware and simplifies assembly. | ||||||
|  |  | ||||||
|  | - **Display Mounting:** The display is now mounted using M3 screws with VORON heat inserts. The display's mounting holes need to be drilled to 3mm to accommodate the M3 screws. | ||||||
|  |  | ||||||
|  | - **Scale Top Surface:** The top surface of the scale was modified to allow M3 socket head cap screws to sit flush with the 3D-printed part. This design ensures that the filament spool rests flat without interference. | ||||||
|  |  | ||||||
|  | - **NFC Reader Mounting:** The NFC reader is also secured using M3 screws and VORON heat inserts, maintaining consistency across all components. | ||||||
|  |  | ||||||
|  | - **Scale Base Mounting:** The only M4 screws required are for attaching the metal part of the scale to its base. | ||||||
|  |  | ||||||
|  | ## Benefits of Modifications | ||||||
|  |  | ||||||
|  | - **Cost Reduction:** Sourcing components from AliExpress offers a more affordable alternative to Amazon, making the project more accessible. | ||||||
|  |  | ||||||
|  | - **Standardized Hardware:** Using M3 screws and [VORON](https://vorondesign.com/) heat inserts throughout the assembly simplifies the build process and reduces the variety of required hardware. | ||||||
|  |  | ||||||
|  | - **Enhanced Compatibility:** Adjustments to the 3D models ensure compatibility with readily available components, accommodating variations in part dimensions. | ||||||
|  |  | ||||||
|  | ## Assembly Instructions | ||||||
|  |  | ||||||
|  | 1. **Component Preparation:** | ||||||
|  |    - Carefully drill the display's mounting holes to 3mm to fit M3 screws. | ||||||
|  |  | ||||||
|  | 2. **Heat Insert Installation:** | ||||||
|  |    - install VORON M3 heat inserts into the designated holes in the 3D-printed housing/case for the ESP32 and Scale top → [heat insert location pictures](./Images/README.md) | ||||||
|  |  | ||||||
|  | 3. **Component Mounting:** | ||||||
|  |    - Attach the display, scale, and NFC reader to their respective mounts using M3 screws. | ||||||
|  |    - Secure the metal part of the scale to its base using M4 screws. | ||||||
|  |  | ||||||
|  | 4. **Final Assembly:** | ||||||
|  |    - Assemble all components according to the original FilaMan instructions, ensuring that all modified parts fit correctly and function as intended. | ||||||
|  |  | ||||||
|  | For detailed assembly guides and additional resources, refer to the [original FilaMan documentation](https://github.com/ManuelW77/Filaman). | ||||||
|  |  | ||||||
|  | ## Conclusion | ||||||
|  |  | ||||||
|  | These modifications to the FilaMan project provide a cost-effective and standardized approach to building a filament management system. By sourcing components from AliExpress and adjusting the 3D models accordingly, users can achieve the same functionality at a reduced cost, with the added benefit of using uniform hardware throughout the assembly. | ||||||
|  |  | ||||||
|  | ## Changelog | ||||||
|  |  | ||||||
|  | ### Version 1.0 - 2025-03-04 | ||||||
|  | - Initial release of modifications for AliExpress-sourced components. | ||||||
|  | - Adjusted 3D models to fit different display and scale dimensions. | ||||||
|  | - Replaced M4 screws with M3 screws and integrated VORON heat inserts. | ||||||
|  | - Modified display mounting, requiring drilling to 3mm for M3 screws. | ||||||
|  | - Adjusted scale top surface for flush screw placement. | ||||||
|  | - Standardized NFC reader mounting with M3 screws and VORON heat inserts. | ||||||
|  | - Retained M4 screws only for metal scale attachment. | ||||||