Compare commits
	
		
			1230 Commits
		
	
	
		
			124580f82a
			...
			v2.0.2-bet
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 77fbacc681 | |||
| b4f1fc3b0a | |||
| 3a82175bb6 | |||
| b80184bf23 | |||
| 0f63880d1f | |||
| 0baa1d286e | |||
| 100328b1d6 | |||
| 9ec5bca652 | |||
| 1dba2b2f23 | |||
| cca0bd9dbe | |||
| 818094c36e | |||
| 4cf3858d0a | |||
| 66eef2242b | |||
| 87288e606b | |||
| 9ae9e80dcd | |||
| f2b38a5a99 | |||
| ab005b3dd1 | |||
| e537c6ec07 | |||
| bec769e95a | |||
| 5cc58927a6 | |||
| afde3f5f81 | |||
| 6800c88bb2 | |||
| 6172242f24 | |||
| 7f4b3b8d90 | |||
| 7a15424bc7 | |||
| 039a29fa3c | |||
| 6cccf3d603 | |||
| 693ee839e5 | |||
| 0bf383ecd9 | |||
| 6451d91c59 | |||
| 8d82e221b5 | |||
| bf63ecd594 | |||
| 0daa3a148b | |||
| 602642c203 | |||
| 458bd2e67b | |||
| e6a5cb29a9 | |||
| 6502bb7185 | |||
| 63fafa2463 | |||
| f664e85933 | |||
| 7bf9868d79 | |||
| b9e488d675 | |||
| 2e3fc19741 | |||
| 4d84169b29 | |||
| 10aeb9bc52 | |||
| 00b9bc08af | |||
| dfe9e4dbe9 | |||
| 79eacae225 | |||
| d5d7358f58 | |||
| 9b362b3c73 | |||
| bc51956793 | |||
| 5666a58da2 | |||
| a35f15eca5 | |||
| f28b34e427 | |||
| 9215560558 | |||
| 7f6bce1699 | |||
| 2a4f8bb679 | |||
| 480e2da23e | |||
| ba22602767 | |||
| b2c68d5aac | |||
| 52a7f6b5b6 | |||
| 4cce9f8d5d | |||
| f0eced8585 | |||
| 02e31878ee | |||
| 7ff499f984 | |||
| fcd637cc30 | |||
| 587485d0de | |||
| e0cc99e993 | |||
| d9a8388ac7 | |||
| cb77112976 | |||
| 1c0ddb52ba | |||
| 17f03e9472 | |||
| 213b9c099c | |||
| 687e57b77a | |||
| aea11e0c06 | |||
| bd8f4606c6 | |||
| ac91e71c14 | |||
| 0d3503f4f1 | |||
| 1460c6e5f9 | |||
| fef7e5aa4b | |||
| bda8c3dd98 | |||
| 8702469020 | |||
| 2a0f999f3b | |||
| c89adb6256 | |||
| 1f21954703 | |||
| 3e59ce1366 | |||
| 1f880fc8f1 | |||
| 69bf5f90fa | |||
| 382caeaced | |||
| 47bdf022ec | |||
| 02febfa943 | |||
| 257f4df800 | |||
| bff6e72219 | |||
| 26e905050d | |||
| 046f770a52 | |||
| 2587227e78 | |||
| 0f19dc4f46 | |||
| 721dac1ead | |||
| 08abd1a37f | |||
| da78861613 | |||
| 9231a303f3 | |||
| d12e766cd7 | |||
| af7bc23703 | |||
| de39892f64 | |||
| 40cb835e51 | |||
| eb9d9e74f4 | |||
| d8af3f45e5 | |||
| 96bb8f9c7c | |||
| b8b6893cd0 | |||
| 0a246c1fe4 | |||
| 965ea5da1e | |||
| b8b6f637f2 | |||
| 12044b657b | |||
| 95433b4842 | |||
| 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 | |||
| 8a558c3121 | |||
|  | 1aeced76a2 | ||
|  | d434fde92e | ||
| 967ec35c6a | |||
| 5afb60df32 | |||
| f60113aa83 | |||
| 3394e6eb01 | |||
| 63a7398979 | |||
| 3818c2c059 | |||
| 40cb504251 | |||
| 0afc543b5f | |||
| 41a4f8af4a | |||
| adee46e3fc | |||
| e122224472 | |||
| 1db74867e6 | |||
| 5d3a8d971f | |||
| 0f24a63d32 | |||
| e62e5e7062 | |||
| 3640809502 | |||
| 726a60882d | |||
| 289d5357be | |||
| 2918b4ca77 | |||
| 315530d1ea | |||
| 955ba0f001 | |||
| f36773a4c4 | |||
| 8cf7dc0b77 | |||
| b35163936f | |||
| 33e4b371ed | |||
| 7a2c9d6d17 | |||
| fd832d8808 | |||
| eb2a8dc128 | |||
| c0e213a4ac | |||
| bec2c91331 | |||
| bcc00f711b | |||
| c6e727de06 | |||
| 78f336d5d7 | |||
| 3253e7d407 | |||
| ee7f8ff517 | |||
| bce2ad2ed8 | |||
|  | 0eff29ef4a | ||
| 492bf6cdb8 | |||
| b0317f4001 | |||
| 58ff6458b0 | |||
| d9c40f5124 | |||
| 68bc31e29a | |||
| 9b23ac5fd2 | |||
| d31bff14c3 | |||
| 150f92484a | |||
| fa74832fb9 | |||
| 2eab3db77d | |||
| 0a1bf22f7e | |||
| d58244c1f8 | |||
| db626ea516 | |||
| fd8f7685a1 | |||
| 944b156528 | |||
| 76100593cc | |||
| 732d590344 | |||
| 46cd953b80 | |||
| c645035bbe | |||
| 9e76620cd3 | |||
| faddda6201 | |||
| de9c1706c0 | |||
| 9f7ee13e78 | |||
| cf3f6f6741 | |||
| b87d43c64e | |||
| 3d0411e3c1 | |||
| 9c61b708aa | |||
| 90f800d042 | |||
| a7b1721e1d | |||
| e4825d2905 | |||
| c1733848d3 | |||
| 484c95523d | |||
| 8499613215 | |||
| 08f37186b4 | |||
| 2948a35fa8 | |||
| 730724fe58 | |||
| 714b7065e7 | |||
| 2d8aec515d | |||
| b245a206ce | |||
| f1489e75cc | |||
| d9ae829503 | |||
| 2247b8ed6c | |||
| d70b187bf9 | |||
| 1ade007473 | |||
| 0af14e2f7d | |||
| de67cdbff3 | |||
| 98fce15ccc | |||
| ab417ba64b | |||
| 320057bc49 | |||
| 9007a65fc2 | |||
| 2214f5f5de | |||
| 5c5846c52c | |||
| 517fa37a3d | |||
| aaa7a6ee9c | |||
| a0b8639488 | |||
| a16c05287e | |||
| ecb35a97bd | |||
| ba968611ec | |||
| 6bd11ddce3 | |||
| 3eb313e61a | |||
| aad35dc296 | |||
| 85ac636b1e | |||
| 6f1804c3fe | |||
| 89716920dc | |||
| 78b5078651 | |||
| 6098c3b052 | |||
| e7537f94d4 | |||
| 37717392d0 | |||
| c6da28ad6f | |||
| d6e38a4e73 | |||
| 4e0d9353c8 | |||
| 7059826659 | |||
| 41faa8bb1c | |||
| b38e3fa5ef | |||
| 5280d7e341 | |||
| 2f95c66d39 | |||
| df1b87465c | |||
| 84f1420999 | |||
| b14dd5475d | |||
| 975845421b | |||
| 044ddbe0eb | |||
| c385544d67 | |||
| c6cfd85687 | |||
| 84632322e2 | |||
| 86e55a8696 | |||
| d2b40daaca | |||
| 9d58cbc31c | |||
| d09aeaf47c | |||
| 9fb82fe51e | |||
| 5e0e2c5f6b | |||
| a8460503ff | |||
| 6700a1761f | |||
| 7207f36e06 | |||
| e79bee3381 | |||
| c3918f075b | |||
| 0c384219c5 | |||
| 42b9daf4be | |||
| 13a771682f | |||
| f79f87bf09 | |||
| 9fe3f6c0ff | |||
| 55e89948bb | |||
| 6c5e8c4d07 | |||
| 4f79700d74 | |||
| 1b4fecf409 | |||
| 89a6101d97 | |||
| ee45a74fee | |||
| db365aba3c | |||
| 63cdfaee6c | |||
| eb2e360c35 | |||
| 7d578640e2 | |||
| b006533a91 | |||
| 9fa7526623 | |||
| dfbb2fbd9b | |||
| 0302158449 | |||
| 68c385f9d7 | |||
| 9a8bd58cb3 | |||
| 0d8b8918c1 | |||
| a892b854b5 | |||
| 0f02f6c848 | |||
| 96c054827e | |||
| f93eedf775 | |||
| 68a10dfeb2 | |||
| 632b7a089e | |||
| c0e3650bf4 | |||
| 8e3dfc93f7 | |||
| 5016285dce | |||
| 9b1a232fde | |||
| 37e79b7a49 | |||
| 6bd23f31c1 | |||
| 3099e9ded9 | |||
| 4952ad3150 | |||
| 2055da9962 | |||
| 459a31cad3 | |||
| 4b1930209b | |||
| 7dde07b5ab | |||
| 33a5406248 | |||
| b016a31ff0 | |||
| 19bc4927e4 | |||
| cd55cb86ba | |||
| 8ab16b351b | |||
| 400a37d3ac | |||
| eb4f809435 | |||
| 1148947b8e | |||
| 3b01336999 | |||
| 44614b58dc | |||
| ed8d618272 | |||
| cd2ac54e98 | |||
| 92f675b24c | |||
| c342877558 | |||
| f5743cbd7b | |||
| 8a62597705 | |||
| 374721d1e5 | |||
| ea6f708c6e | |||
| 78169dfdb1 | |||
| 074bfb658d | |||
| 989076e794 | |||
| aa0d056d10 | |||
| cd619b8f2a | |||
| 6d8358cbb9 | |||
| 1f3a67634f | |||
| 09969b644e | |||
| deb7abd102 | |||
| 1b059c35f1 | |||
| e098d71f6f | |||
| 4b25b72b2e | |||
| 5c59016f94 | |||
| d2da501b94 | |||
| 4135073623 | |||
| fe7b57fe0e | |||
| c1ae6b7295 | |||
| 9eee89fac7 | |||
| 8c5e7e26ac | |||
| 7b52066378 | |||
| d5afa38ded | |||
| cf50baba2d | |||
| aa9e7da94b | |||
| 71cd3ba4fc | |||
| 73e240e879 | |||
| 0d34e1d718 | |||
| 84cc8beb9b | |||
| fd70e3179d | |||
| c553640ad8 | |||
| 807eca3c43 | |||
| b52730bf67 | |||
| 9a59b91e88 | |||
| a5af4013d8 | |||
| e54ce58ec4 | |||
| 142eafd232 | |||
| 63ab9e0993 | |||
| aaa5506d40 | |||
| 8037adc045 | |||
| 6e7c728cd8 | |||
| 3fe8271344 | |||
| f2bc6eab92 | |||
| 37df492339 | |||
| c4b425403f | |||
| 73244689dd | |||
| 27296104d2 | |||
| 5f99773897 | |||
| 7416285fb9 | |||
| 85928e358d | |||
| 092b4fd8ec | |||
| 399645a2b3 | |||
| 164bb241b7 | |||
| e564c6eeae | |||
| 4288dd0cd4 | |||
| 37d43b2d7d | |||
| adb354ddcd | |||
| 15d5e5edce | |||
| c6edf30245 | |||
| 65ac207f36 | |||
| 698abbd669 | |||
| 04a7c2cce3 | |||
| 78f54b72fd | |||
| f4eee9af91 | |||
| cad14b3bc2 | |||
| 312f75fc5f | |||
| b8714e93e2 | |||
| cd9da0fe4f | |||
| 2b620ef5ed | |||
| 3f63a01b8b | |||
| 22bb16b6a4 | |||
| 53ceee7816 | |||
| d48b002806 | |||
| dd905b6c6e | |||
| 77b9eda110 | |||
| 32a6e9dcd3 | |||
| 6cd5539e60 | |||
| 903b697912 | |||
| 72c2fb70c2 | |||
| f2f3f0ab9f | |||
| c07692c218 | |||
| a184903b66 | |||
| af1640383d | |||
| c00e54b145 | |||
| f6c92c686b | |||
| b8db01529b | |||
| 55db6d76ab | |||
| a18749a1ff | |||
| 1811fd9159 | |||
| b550760427 | |||
| c5033acadc | |||
| 7de4189c83 | |||
| f43f2a15b2 | |||
| 858192c6cb | |||
| e2bd39922d | |||
| c86cc7173e | |||
| 16362e66a3 | |||
| 48d9ba8f71 | |||
| e2bea5a0c3 | |||
| 3e11f65188 | |||
| df59c42c8a | |||
| abe1d7c930 | |||
| ca614c3cc4 | |||
| 5153374093 | |||
| 66db4d7a85 | |||
| 90e71922b1 | |||
| e8e5c0bd3d | |||
| 7e53e1ccb0 | |||
| e49e812b13 | |||
| b1e0fcfadf | |||
| 31ef3ac8df | |||
| 8cf3f87c89 | |||
| c446188311 | |||
| 8e2a8d597d | |||
| 7d3b1c34f6 | |||
| b95c61118b | |||
| 0dfb158959 | |||
| 75c774bb24 | |||
| cf80adb43c | |||
| 36d50cbe7f | |||
| 9148d207c7 | |||
| 5f6fef9448 | |||
| 946202de0e | |||
| 41a3717347 | |||
| 255c820439 | |||
| aef3ba77ba | |||
| 2592c3a497 | |||
| a48c5dfef0 | |||
| 00554d0b09 | |||
| 05a91cd8d8 | |||
| 7cf113eaff | |||
| 44d27adab2 | |||
| e0a2dff5fe | |||
| 519a089684 | |||
| ef053bb2b6 | |||
| 0a91c7b269 | |||
| 875d9d2b70 | |||
| 52840b9b0b | |||
| da1fc7678f | |||
| 982bb5aa21 | |||
| 007737db13 | |||
| 17e5949201 | |||
| 6a57186091 | |||
| babd3f47a0 | |||
| 5372fe10fe | |||
| e0c9d90892 | |||
| e5f5d1961b | |||
| 31a960fb9e | |||
| 3c2e75b77a | |||
| 367143c456 | |||
| fbde4b764f | |||
| e57f4216d4 | |||
| b8beb992d6 | |||
| 4234b2254e | |||
| b8faf79163 | |||
| d35afaff46 | |||
| a8a00372b5 | |||
| 72f4eab588 | |||
| afa4eddc00 | |||
| b0888e7e63 | |||
| 238a84a8a2 | |||
| 59cc00ca13 | |||
| ab083f5f57 | |||
| c111573206 | |||
| 52b2494e52 | |||
| 069ec2d7a1 | |||
| 94e35ae86e | |||
| d71e3d8184 | |||
| bb166aa29f | |||
| 0d718023f8 | |||
| b16781043f | |||
| dff184ff25 | |||
| 0ce281221d | |||
| bc26c160e8 | |||
| c25f41db75 | |||
| e107c17f50 | |||
| 85b9d03ebd | |||
| 17b188626a | |||
| a534c5f872 | |||
| 93f7582790 | |||
| 46acc63756 | |||
| 67a9e1bdce | |||
| 2b75b64b4a | |||
| 8d003295e7 | |||
| f89500946a | |||
| 14e745ff06 | |||
| d058397fa2 | |||
| 622f5403a7 | |||
| 92b78a86dd | |||
| ec399390e8 | |||
| 909c4e9b5e | |||
| f4b20bfffd | |||
| 78464215a9 | |||
| 4365f0463a | |||
| 727bc0e760 | |||
| 04604013eb | |||
| cf5fc5f6f1 | |||
| 945a4ccce6 | |||
| 7cf9e2d145 | |||
| 9db4e338ea | |||
| dea6ca2c66 | |||
| e224e72e41 | |||
| 306c517da7 | |||
| 0337bbabe0 | |||
| bde14e50e0 | |||
| 9c656a9bd0 | |||
| eae552017d | |||
| a77918da41 | |||
| 262dad38a6 | |||
| cfc9f103cf | |||
| 0117302672 | |||
| 1de283b62f | |||
| f1eb78eb38 | |||
| 8a65b86475 | |||
| a3aef819c8 | |||
| a62b5ec933 | |||
| 1a8cf7a58f | |||
| b0b3d41c84 | |||
| 38b68aecfc | |||
| 4992f5f433 | |||
| 5cbbe1d231 | |||
| 9b29460d64 | |||
| dd14d475b7 | |||
| 9e6cd3b451 | |||
| c1be6ca582 | |||
| 265ff0c787 | |||
| 67eca82ac5 | |||
| 568db90db0 | |||
| 2dfd53d64a | |||
| 262a2fcbd4 | |||
| 3770de15d3 | |||
| 75a74ec9bd | |||
| 979adcbb14 | |||
| 2dd563a178 | |||
| 767c217c25 | |||
| c07689e15a | |||
| d6ca69fd19 | |||
| 60553255b8 | |||
| 8199b283c0 | |||
| d774ce0d09 | |||
| 4a44eda5c4 | |||
| c43ca20d8d | |||
| 21ba35cd19 | |||
| 62273320e9 | |||
| b8e4af4e4d | |||
| 513d74fdb0 | |||
| df884e7668 | |||
| 8182b5f684 | |||
| 4477537cec | |||
| 44ba7df34f | |||
| 54744a06dd | |||
| cefa81030b | |||
| 62052927d2 | |||
| 933a84f8ce | |||
| db3c19ff2e | |||
| ae9eb4cc6b | |||
| 89d40832c5 | |||
| c161216c04 | |||
| 6a016b6ac4 | |||
| 44dd485e17 | |||
| d41f0f3e67 | |||
| 484058515e | |||
| f552b492cf | |||
| c3040b3c29 | |||
| d7ba67085d | |||
| 48efb9e21a | |||
| e983ba6e44 | |||
| 37171d6eca | |||
| ebb08a7a66 | |||
| b5330af351 | |||
| 4919d34484 | |||
| 2da641d604 | |||
| ce413965c7 | |||
| 3fafed930e | |||
| e1c604ee8d | |||
| 11bbfb7db6 | |||
| 71d8f7ec5a | |||
| f4518e4a36 | |||
| 62d9596d08 | |||
| e27e95d291 | |||
| b7651ad50d | |||
| f1937e2977 | |||
| ad5ddf713c | |||
| ccb494f843 | |||
| 17307d8f03 | |||
| e5240a9572 | |||
| 06ebf105cf | |||
| 118e099fc5 | |||
| 8edd50f786 | |||
| b85325a747 | |||
| e1e0352beb | |||
| 8a93cccfce | |||
| c374069f36 | |||
| 59cd7c177d | |||
| 45088b5838 | |||
| 1b9c79b559 | |||
| 37e1e861d3 | |||
| cce39319d9 | |||
| 6391054c23 | |||
| 52cf46d7f8 | |||
| 84b05e48ce | |||
| 5c41d864c1 | |||
| 5dc3563da6 | |||
| 9e1b2943d6 | |||
| 7b89b04621 | |||
| e140f8e003 | |||
| 3d0bdde476 | |||
| 3ac7d6b4f7 | |||
| 5f52775984 | |||
| 463eaf4b6f | |||
| 4bf6b11d3a | |||
| b0c4af7c4e | |||
| 249e896ea4 | |||
| c74f587fff | |||
| 7a7ee72585 | |||
| 3dd5fbc585 | |||
| ed9c1487ed | |||
| d8756421a1 | |||
| d92c78f9d0 | |||
| 2d19ea745f | |||
| 13779cc9d7 | |||
| b6d5a8a00b | |||
| f6319e79f0 | |||
| 6f24630a7d | |||
| 4475d21218 | |||
| 01a926a38d | |||
| 6b966c02b3 | |||
| 1450e1ad2e | |||
| 3102a6c217 | |||
| d5b2b2746d | |||
| 7e776d4816 | |||
| e84b2973c5 | |||
| 5793dc1a1f | |||
| 1732491c48 | |||
| 0500bb6951 | |||
| ef9ef7257a | |||
| e86fd229dc | |||
| b940a166da | |||
| c857e16de2 | |||
| 8b2a537b72 | |||
| 88ec151c4c | |||
| 0a203f02eb | |||
| 04b59f4809 | |||
| b31861af67 | |||
| ce3b423dc1 | |||
| 05f275142f | |||
| 72af54bd20 | |||
| a1e30a3b7f | |||
| 8f8322e629 | |||
| 1e386c49af | |||
| 3070d75d07 | |||
| 34ee9badea | |||
| 6594af9540 | |||
| c21bd2d4ec | |||
| e5f895b560 | |||
| bb0f50ce1d | |||
| 499a08aa75 | |||
| 745d960f62 | |||
| 531d0fe9a2 | |||
| 2af23e8084 | |||
| cd204fdaa8 | |||
| d04ad42b5b | |||
| 8d80a8fb5a | |||
| 77032bf9ae | |||
| 3967366ae6 | |||
| cd391378c2 | |||
| 3fc3a689cb | |||
| bd5b32a3b0 | |||
| a5c999234c | |||
| f64d3e51ce | |||
| a650b7d860 | |||
| 14e98072aa | |||
| 0bc4c0565f | |||
| 4aa8e844ac | |||
| d7135fddc8 | |||
| 6e3e978839 | |||
| 33b0e79c18 | |||
| 787d878e83 | |||
| 1e24179b82 | |||
| 67027840f4 | |||
| 2755e9c863 | |||
| 7ae26fb0a9 | |||
| 98bcf98f1e | |||
| bc8ac32fee | |||
| ba74eca21d | |||
| d4e1cf1322 | |||
| 5170784e44 | |||
| ac323167d0 | |||
| 003de5bc00 | |||
| 83595cfadf | |||
| cfe21d63d7 | |||
| 63e72076ed | |||
| 2a011ee244 | |||
| 84dd2bd40b | |||
| 1449cacc0e | |||
| f1c2b2eb87 | |||
| 0f1a3b1d5b | |||
| 6190cf04db | |||
| 680aed0e10 | |||
| fadb122d28 | |||
| 4347f89c16 | |||
| 9856757e49 | |||
| afb7b5f42b | |||
| ba8506247b | |||
| 61f3f90d6d | |||
| a143527dd0 | |||
| f2c9818e61 | |||
| 9e23af29ff | |||
| 095885442a | |||
| 359c9b5a6d | |||
| b16a7a4c17 | |||
| b4ea175757 | |||
| ca4671a7dd | |||
| 7a4b1a934f | |||
| 77d90bbe83 | |||
| c99d802184 | |||
| ca15d2abe4 | |||
| bddb48ba9a | |||
| f4b67e52b8 | |||
| 00f6a4b0ae | |||
| ab3937a69d | |||
| 0c4bae48d4 | |||
| b4b17cb999 | |||
| 5ff3864d9d | |||
| 9b8736d35f | |||
| cf0ba20637 | |||
| e06c0b9a76 | |||
| 4cb370ddff | |||
| 0b9c1711da | |||
| e2449030c5 | |||
| 5a91e87afa | |||
| 5e66c3bd45 | |||
| 61c82f796f | |||
| 7661e0eb20 | |||
| c06e6de89a | |||
| 262bed84df | |||
| c710e5d7f8 | |||
| 5d39f04786 | |||
| d955e26b82 | |||
| c20e7de5c3 | |||
| 55c3b3688a | |||
| fea5932125 | |||
| 04f557939d | |||
| 58549a53e4 | |||
| 64a46e3e7d | |||
| c2bd0982ab | |||
| e0b8da079e | |||
| 8c88827a2d | |||
| 6128bc2827 | |||
| 6d21a8ec52 | |||
| 2ee4eca7dd | |||
| 9eb64e06f0 | |||
| 693ef4e26c | |||
| 11c099eba2 | |||
| c7c122a505 | |||
| d1bb4b7c83 | |||
| dc5118587c | |||
| b13f2c4eee | |||
| 6d16c35e62 | |||
| c94ebf5e33 | |||
| 9203706a20 | |||
| c252f48a49 | |||
| 228dcacb1e | |||
| 9de4ed9ee9 | |||
| 4d8a6fb943 | |||
| 2cab24403e | |||
| 8b246e180b | |||
| b5c014db86 | |||
| 3c783c9844 | |||
| 175d614d1f | |||
| 678a286af1 | |||
| f877f43d90 | |||
| 3cd0798186 | |||
| 2a67d8f67c | |||
| 240795a2d0 | |||
| 4e384d777e | |||
| 03cbf82275 | |||
| 0c1a222636 | |||
| 8716b4ad73 | |||
| 6fcfefec8f | |||
| b696a79f4b | |||
| 61ed765d87 | |||
| 2703689e4e | |||
| b24c50722f | |||
| 3ec23a9f79 | |||
| 75fe6b55ad | |||
| fa2f980312 | |||
| 7964f1cd77 | |||
| dd611df9f5 | |||
| 0ccc67f4b2 | |||
| a027dfb54e | |||
| 07741f2a52 | |||
| 3e9b89f69b | |||
| bf67a635f6 | |||
| 84cccd5014 | |||
| e983b0fcfd | |||
| e01bb9b1f9 | |||
| 17b6051da0 | |||
| 84617ebc53 | |||
| 8ac6ba57a4 | |||
| cc17140dc4 | |||
| 8741216374 | |||
| 7bdebeab03 | |||
| ab33f423c8 | |||
| 15600ceac5 | |||
| 7ccacec68f | |||
| 2f34a0ca4e | |||
| 2fb2a2f183 | |||
| 269f54b2b3 | |||
| 1c1043ac75 | |||
| d1f22c78f7 | |||
| 3e1490cafc | |||
| dc82c04a17 | |||
| c6fd5f8ad5 | |||
| a55cce854e | |||
| c21be92e98 | |||
| d4348944fc | |||
| f1bd896cf1 | |||
| 9ace9f0567 | |||
| 0379c4c45f | |||
| 49f36bb25b | |||
| a05cde8669 | |||
| 64e5a171b6 | |||
| bec403ec1b | |||
| ccc810ab7e | |||
| 61dadc0aa1 | |||
| a2ee9a98a6 | |||
| ae96c729b1 | |||
| 1d337dd990 | |||
| cfa6c12d65 | |||
| b2aeae5815 | |||
| 5cdbc5f2cc | |||
| 7a5415ad80 | |||
| 36fd271d0c | |||
| a39e199c79 | |||
| b7004f5b71 | |||
| e5bf05ee43 | |||
| f01a42d850 | |||
| 04101d6cfa | |||
| d46d5e14d2 | |||
| be0a23e736 | |||
| c46553434b | |||
| 89a88463ef | |||
| 3e8e19e8dc | |||
| d220479709 | |||
| f49bbdb664 | |||
| 412127fce5 | 
							
								
								
									
										208
									
								
								.github/workflows/gitea-release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,208 @@ | ||||
| name: Gitea Release | ||||
|  | ||||
| on: | ||||
|   workflow_call: | ||||
|     secrets: | ||||
|       GITEA_TOKEN: | ||||
|         description: 'Token für Gitea API-Zugriff' | ||||
|         required: true | ||||
|  | ||||
|     outputs: | ||||
|       version: | ||||
|         description: 'The version that was released' | ||||
|         value: ${{ jobs.create-release.outputs.version }} | ||||
|  | ||||
| jobs: | ||||
|   create-release: | ||||
|     runs-on: ubuntu-latest | ||||
|     outputs: | ||||
|       version: ${{ steps.get_version.outputs.VERSION }} | ||||
|     steps: | ||||
|     - uses: actions/checkout@v4 | ||||
|       with: | ||||
|         fetch-depth: 0 | ||||
|      | ||||
|     - name: Set up Python | ||||
|       uses: actions/setup-python@v4 | ||||
|       with: | ||||
|         python-version: '3.x' | ||||
|      | ||||
|     - name: Install PlatformIO | ||||
|       run: | | ||||
|         python -m pip install --upgrade pip | ||||
|         pip install --upgrade platformio esptool | ||||
|      | ||||
|     - name: Install xxd | ||||
|       run: | | ||||
|         sudo apt-get update | ||||
|         sudo apt-get install xxd | ||||
|      | ||||
|     - name: Build Firmware | ||||
|       run: | | ||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||
|          | ||||
|         # Build firmware and LittleFS | ||||
|         echo "Building firmware and LittleFS..." | ||||
|         pio run -e esp32dev | ||||
|         pio run -t buildfs | ||||
|          | ||||
|         # Copy firmware binary | ||||
|         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/upgrade_filaman_firmware_v${VERSION}.bin | ||||
|          | ||||
|         # Create LittleFS binary - direct copy without header | ||||
|         cp .pio/build/esp32dev/littlefs.bin .pio/build/esp32dev/upgrade_filaman_website_v${VERSION}.bin | ||||
|          | ||||
|         # Create full binary | ||||
|         (cd .pio/build/esp32dev &&  | ||||
|         esptool.py --chip esp32 merge_bin \ | ||||
|           --fill-flash-size 4MB \ | ||||
|           --flash_mode dio \ | ||||
|           --flash_freq 40m \ | ||||
|           --flash_size 4MB \ | ||||
|           -o filaman_full_${VERSION}.bin \ | ||||
|           0x1000 bootloader.bin \ | ||||
|           0x8000 partitions.bin \ | ||||
|           0x10000 firmware.bin \ | ||||
|           0x3D0000 littlefs.bin) | ||||
|          | ||||
|         # Verify file sizes | ||||
|         echo "File sizes:" | ||||
|         (cd .pio/build/esp32dev && ls -lh *.bin) | ||||
|      | ||||
|     - name: Get version from platformio.ini | ||||
|       id: get_version | ||||
|       run: | | ||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||
|         echo "VERSION=$VERSION" >> $GITHUB_OUTPUT | ||||
|        | ||||
|     - name: Generate Release Notes | ||||
|       id: release_notes | ||||
|       run: | | ||||
|         # Get the latest tag | ||||
|         LATEST_TAG=$(git for-each-ref --sort=-creatordate --format '%(refname:short)' refs/tags | sed -n '2p') | ||||
|          | ||||
|         if [ -n "$LATEST_TAG" ]; then | ||||
|           echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "Changes since ${LATEST_TAG}:" >> $GITHUB_OUTPUT | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           # Get all commits since last release with commit hash and author | ||||
|           echo "### Added" >> $GITHUB_OUTPUT | ||||
|           git log ${LATEST_TAG}..HEAD --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - (feat|add|new)' | sed 's/^[a-f0-9]* - feat: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Fixed" >> $GITHUB_OUTPUT | ||||
|           git log ${LATEST_TAG}..HEAD --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - fix' | sed 's/^[a-f0-9]* - fix: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Changed" >> $GITHUB_OUTPUT | ||||
|           git log ${LATEST_TAG}..HEAD --pretty=format:"%h - %s (%an)" | grep -ivE '^[a-f0-9]+ - (feat|fix|add|new)' | sed 's/^[a-f0-9]* - /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|         else | ||||
|           # First release | ||||
|           echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "Initial Release" >> $GITHUB_OUTPUT | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           # Add all commits for initial release | ||||
|           echo "### Added" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - (feat|add|new)' | sed 's/^[a-f0-9]* - feat: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Fixed" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - fix' | sed 's/^[a-f0-9]* - fix: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Changed" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:"%h - %s (%an)" | grep -ivE '^[a-f0-9]+ - (feat|fix|add|new)' | sed 's/^[a-f0-9]* - /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|         fi | ||||
|  | ||||
|     - name: Determine Gitea URL | ||||
|       id: gitea_url | ||||
|       run: | | ||||
|         echo "Debug Environment:" | ||||
|         echo "GITHUB_SERVER_URL=${GITHUB_SERVER_URL:-not set}" | ||||
|         echo "GITEA_SERVER_URL=${GITEA_SERVER_URL:-not set}" | ||||
|         echo "GITHUB_REPOSITORY=${GITHUB_REPOSITORY:-not set}" | ||||
|         echo "GITEA_REPOSITORY=${GITEA_REPOSITORY:-not set}" | ||||
|         echo "RUNNER_NAME=${RUNNER_NAME:-not set}" | ||||
|          | ||||
|         # Set API URL based on environment | ||||
|         if [ -n "${GITEA_ACTIONS}" ] || [ -n "${GITEA_REPOSITORY}" ] || [[ "${RUNNER_NAME}" == *"gitea"* ]]; then | ||||
|           GITEA_API_URL="${GITHUB_SERVER_URL}" | ||||
|           GITEA_REPO=$(echo "${GITHUB_REPOSITORY}" | cut -d'/' -f2) | ||||
|           GITEA_OWNER=$(echo "${GITHUB_REPOSITORY}" | cut -d'/' -f1) | ||||
|         else | ||||
|           echo "Error: This workflow is only for Gitea" | ||||
|           exit 1 | ||||
|         fi | ||||
|          | ||||
|         echo "GITEA_API_URL=${GITEA_API_URL}" >> $GITHUB_OUTPUT | ||||
|         echo "GITEA_REPO=${GITEA_REPO}" >> $GITHUB_OUTPUT | ||||
|         echo "GITEA_OWNER=${GITEA_OWNER}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|     - name: Create Gitea Release | ||||
|       env: | ||||
|         GITEA_TOKEN: ${{ secrets.GITEA_TOKEN }} | ||||
|         GITEA_API_URL: ${{ steps.gitea_url.outputs.GITEA_API_URL }} | ||||
|         GITEA_REPO: ${{ steps.gitea_url.outputs.GITEA_REPO }} | ||||
|         GITEA_OWNER: ${{ steps.gitea_url.outputs.GITEA_OWNER }} | ||||
|       run: | | ||||
|         # Debug Token (nur Länge ausgeben für Sicherheit) | ||||
|         echo "Debug: Token length: ${#GITEA_TOKEN}" | ||||
|         if [ -z "$GITEA_TOKEN" ]; then | ||||
|           echo "Error: GITEA_TOKEN is empty" | ||||
|           exit 1 | ||||
|         fi | ||||
|  | ||||
|         VERSION=${{ steps.get_version.outputs.VERSION }} | ||||
|         cd .pio/build/esp32dev | ||||
|          | ||||
|         # Debug-Ausgaben | ||||
|         echo "Debug: API URL: ${GITEA_API_URL}" | ||||
|         echo "Debug: Repository: ${GITEA_OWNER}/${GITEA_REPO}" | ||||
|          | ||||
|         # Erstelle zuerst den Release ohne Dateien | ||||
|         echo "Debug: Creating release..." | ||||
|         RELEASE_DATA="{\"tag_name\":\"v${VERSION}\",\"name\":\"v${VERSION}\",\"body\":\"${{ steps.release_notes.outputs.CHANGES }}\"}" | ||||
|          | ||||
|         RELEASE_RESPONSE=$(curl -s -w "\n%{http_code}" \ | ||||
|           -X POST \ | ||||
|           -H "Authorization: token ${GITEA_TOKEN}" \ | ||||
|           -H "Content-Type: application/json" \ | ||||
|           -d "${RELEASE_DATA}" \ | ||||
|           "${GITEA_API_URL}/api/v1/repos/${GITEA_OWNER}/${GITEA_REPO}/releases") | ||||
|          | ||||
|         RELEASE_STATUS=$(echo "$RELEASE_RESPONSE" | tail -n1) | ||||
|         RELEASE_BODY=$(echo "$RELEASE_RESPONSE" | head -n -1) | ||||
|          | ||||
|         if [ "$RELEASE_STATUS" != "201" ]; then | ||||
|           echo "Error: Failed to create release" | ||||
|           echo "Response: $RELEASE_BODY" | ||||
|           exit 1 | ||||
|         fi | ||||
|          | ||||
|         # Extrahiere die Release-ID aus der Antwort | ||||
|         RELEASE_ID=$(echo "$RELEASE_BODY" | grep -o '"id":[0-9]*' | cut -d':' -f2) | ||||
|          | ||||
|         # Lade die Dateien einzeln hoch | ||||
|         for file in upgrade_filaman_firmware_v${VERSION}.bin upgrade_filaman_website_v${VERSION}.bin filaman_full_${VERSION}.bin; do | ||||
|           if [ -f "$file" ]; then | ||||
|             echo "Debug: Uploading $file..." | ||||
|             UPLOAD_RESPONSE=$(curl -s -w "\n%{http_code}" \ | ||||
|               -X POST \ | ||||
|               -H "Authorization: token ${GITEA_TOKEN}" \ | ||||
|               -H "Content-Type: application/octet-stream" \ | ||||
|               --data-binary @"$file" \ | ||||
|               "${GITEA_API_URL}/api/v1/repos/${GITEA_OWNER}/${GITEA_REPO}/releases/${RELEASE_ID}/assets?name=${file}") | ||||
|              | ||||
|             UPLOAD_STATUS=$(echo "$UPLOAD_RESPONSE" | tail -n1) | ||||
|             if [ "$UPLOAD_STATUS" != "201" ]; then | ||||
|               echo "Warning: Failed to upload $file" | ||||
|               echo "Response: $(echo "$UPLOAD_RESPONSE" | head -n -1)" | ||||
|             else | ||||
|               echo "Successfully uploaded $file" | ||||
|             fi | ||||
|           fi | ||||
|         done | ||||
							
								
								
									
										185
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,185 @@ | ||||
| name: GitHub Release | ||||
|  | ||||
| on: | ||||
|   workflow_call: | ||||
|     secrets: | ||||
|       RELEASE_TOKEN: | ||||
|         description: 'GitHub token for release creation' | ||||
|         required: true | ||||
|  | ||||
| permissions: | ||||
|   contents: write | ||||
|  | ||||
| jobs: | ||||
|   create-release: | ||||
|     runs-on: ubuntu-latest | ||||
|     permissions: | ||||
|       contents: write | ||||
|     steps: | ||||
|     - uses: actions/checkout@v4 | ||||
|       with: | ||||
|         fetch-depth: 0 | ||||
|      | ||||
|     - name: Set up Python | ||||
|       uses: actions/setup-python@v4 | ||||
|       with: | ||||
|         python-version: '3.x' | ||||
|      | ||||
|     - name: Install PlatformIO | ||||
|       run: | | ||||
|         python -m pip install --upgrade pip | ||||
|         pip install --upgrade platformio esptool | ||||
|      | ||||
|     - name: Install xxd | ||||
|       run: | | ||||
|         sudo apt-get update | ||||
|         sudo apt-get install xxd | ||||
|      | ||||
|     - name: Build Firmware | ||||
|       run: | | ||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||
|          | ||||
|         # Always build firmware and LittleFS | ||||
|         echo "Building firmware and LittleFS..." | ||||
|         pio run -e esp32dev | ||||
|         pio run -t buildfs | ||||
|          | ||||
|         # Copy firmware binary | ||||
|         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/upgrade_filaman_firmware_v${VERSION}.bin | ||||
|          | ||||
|         # Create LittleFS binary - direct copy without header | ||||
|         cp .pio/build/esp32dev/littlefs.bin .pio/build/esp32dev/upgrade_filaman_website_v${VERSION}.bin | ||||
|          | ||||
|         # Create full binary (always) | ||||
|         (cd .pio/build/esp32dev &&  | ||||
|         esptool.py --chip esp32 merge_bin \ | ||||
|           --fill-flash-size 4MB \ | ||||
|           --flash_mode dio \ | ||||
|           --flash_freq 40m \ | ||||
|           --flash_size 4MB \ | ||||
|           -o filaman_full_${VERSION}.bin \ | ||||
|           0x1000 bootloader.bin \ | ||||
|           0x8000 partitions.bin \ | ||||
|           0x10000 firmware.bin \ | ||||
|           0x3D0000 littlefs.bin) | ||||
|          | ||||
|         # Verify file sizes | ||||
|         echo "File sizes:" | ||||
|         (cd .pio/build/esp32dev && ls -lh *.bin) | ||||
|      | ||||
|     - name: Get version from platformio.ini | ||||
|       id: get_version | ||||
|       run: | | ||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||
|         echo "VERSION=$VERSION" >> $GITHUB_OUTPUT | ||||
|        | ||||
|     - name: Generate Release Notes | ||||
|       id: release_notes | ||||
|       run: | | ||||
|         # Get the latest tag | ||||
|         LATEST_TAG=$(git for-each-ref --sort=-creatordate --format '%(refname:short)' refs/tags | sed -n '2p') | ||||
|          | ||||
|         if [ -n "$LATEST_TAG" ]; then | ||||
|           echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "Changes since ${LATEST_TAG}:" >> $GITHUB_OUTPUT | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           # Get all commits since last release with commit hash and author | ||||
|           echo "### Added" >> $GITHUB_OUTPUT | ||||
|           git log ${LATEST_TAG}..HEAD --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - (feat|add|new)' | sed 's/^[a-f0-9]* - feat: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Fixed" >> $GITHUB_OUTPUT | ||||
|           git log ${LATEST_TAG}..HEAD --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - fix' | sed 's/^[a-f0-9]* - fix: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Changed" >> $GITHUB_OUTPUT | ||||
|           git log ${LATEST_TAG}..HEAD --pretty=format:"%h - %s (%an)" | grep -ivE '^[a-f0-9]+ - (feat|fix|add|new)' | sed 's/^[a-f0-9]* - /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|         else | ||||
|           # First release | ||||
|           echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "Initial Release" >> $GITHUB_OUTPUT | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           # Add all commits for initial release | ||||
|           echo "### Added" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - (feat|add|new)' | sed 's/^[a-f0-9]* - feat: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Fixed" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - fix' | sed 's/^[a-f0-9]* - fix: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Changed" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:"%h - %s (%an)" | grep -ivE '^[a-f0-9]+ - (feat|fix|add|new)' | sed 's/^[a-f0-9]* - /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|         fi | ||||
|  | ||||
|     - name: Create GitHub Release | ||||
|       env: | ||||
|         GH_TOKEN: ${{ secrets.RELEASE_TOKEN }} | ||||
|       run: | | ||||
|         VERSION=${{ steps.get_version.outputs.VERSION }} | ||||
|         cd .pio/build/esp32dev | ||||
|          | ||||
|         # Create release with available files | ||||
|         FILES_TO_UPLOAD="" | ||||
|          | ||||
|         # Always add firmware | ||||
|         if [ -f "upgrade_filaman_firmware_v${VERSION}.bin" ]; then | ||||
|           FILES_TO_UPLOAD="$FILES_TO_UPLOAD upgrade_filaman_firmware_v${VERSION}.bin" | ||||
|         fi | ||||
|          | ||||
|         # Add LittleFS and full binary only if they exist | ||||
|         if [ -f "upgrade_filaman_website_v${VERSION}.bin" ]; then | ||||
|           FILES_TO_UPLOAD="$FILES_TO_UPLOAD upgrade_filaman_website_v${VERSION}.bin" | ||||
|         fi | ||||
|          | ||||
|         if [ -f "filaman_full_${VERSION}.bin" ]; then | ||||
|           FILES_TO_UPLOAD="$FILES_TO_UPLOAD filaman_full_${VERSION}.bin" | ||||
|         fi | ||||
|          | ||||
|         # Create release with available files | ||||
|         if [ -n "$FILES_TO_UPLOAD" ]; then | ||||
|           gh release create "v${VERSION}" \ | ||||
|             --title "Release ${VERSION}" \ | ||||
|             --notes "${{ steps.release_notes.outputs.CHANGES }}" \ | ||||
|             $FILES_TO_UPLOAD | ||||
|         else | ||||
|           echo "Error: No files found to upload" | ||||
|           exit 1 | ||||
|         fi | ||||
|  | ||||
|     - name: Install lftp | ||||
|       run: sudo apt-get install -y lftp | ||||
|          | ||||
|     - name: Upload Firmware via FTP | ||||
|       if: success() | ||||
|       env: | ||||
|         FTP_PASSWORD: ${{ vars.FTP_PASSWORD }} | ||||
|         FTP_USER: ${{ vars.FTP_USER }} | ||||
|         FTP_HOST: ${{ vars.FTP_HOST }} | ||||
|         VERSION: ${{ steps.get_version.outputs.VERSION }} | ||||
|       run: | | ||||
|         echo "Environment variables:" | ||||
|         env | grep -E '^FTP_' | while read -r line; do | ||||
|           var_name=$(echo "$line" | cut -d= -f1) | ||||
|           var_value=$(echo "$line" | cut -d= -f2-) | ||||
|           echo "$var_name is $(if [ -n "$var_value" ]; then echo "set"; else echo "empty"; fi)" | ||||
|         done | ||||
|          | ||||
|         cd .pio/build/esp32dev | ||||
|         if [ -n "$FTP_USER" ] && [ -n "$FTP_PASSWORD" ] && [ -n "$FTP_HOST" ]; then | ||||
|           echo "All FTP credentials are present, attempting upload..." | ||||
|           lftp -c "set ssl:verify-certificate no; \ | ||||
|                     set ftp:ssl-protect-data true; \ | ||||
|                     set ftp:ssl-force true; \ | ||||
|                     set ssl:check-hostname false; \ | ||||
|                     set ftp:ssl-auth TLS; \ | ||||
|                     open -u $FTP_USER,$FTP_PASSWORD $FTP_HOST; \ | ||||
|                     put -O / filaman_full_${VERSION}.bin -o filaman_full.bin" | ||||
|         else | ||||
|           echo "Error: Some FTP credentials are missing" | ||||
|           exit 1 | ||||
|         fi | ||||
							
								
								
									
										41
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,41 @@ | ||||
| name: Release Workflow | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     tags: | ||||
|       - 'v*' | ||||
|  | ||||
| permissions: | ||||
|   contents: write | ||||
|  | ||||
| jobs: | ||||
|   detect-provider: | ||||
|     runs-on: ubuntu-latest | ||||
|     outputs: | ||||
|       provider: ${{ steps.provider.outputs.provider }} | ||||
|     steps: | ||||
|       - name: Determine CI Provider | ||||
|         id: provider | ||||
|         shell: bash | ||||
|         run: | | ||||
|           if [ -n "${GITEA_ACTIONS}" ] || [ -n "${GITEA_REPOSITORY}" ] || [[ "${RUNNER_NAME}" == *"gitea"* ]]; then | ||||
|             echo "provider=gitea" >> "$GITHUB_OUTPUT" | ||||
|           else | ||||
|             echo "provider=github" >> "$GITHUB_OUTPUT" | ||||
|           fi | ||||
|  | ||||
|   github-release: | ||||
|     needs: detect-provider | ||||
|     permissions: | ||||
|       contents: write | ||||
|     if: needs.detect-provider.outputs.provider == 'github' | ||||
|     uses: ./.github/workflows/github-release.yml | ||||
|     secrets: | ||||
|       RELEASE_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
|   gitea-release: | ||||
|     needs: detect-provider | ||||
|     if: needs.detect-provider.outputs.provider == 'gitea' | ||||
|     uses: ./.github/workflows/gitea-release.yml | ||||
|     secrets: | ||||
|       GITEA_TOKEN: ${{ secrets.GITEA_TOKEN }} | ||||
							
								
								
									
										43
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,43 @@ | ||||
| .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 | ||||
| wiki | ||||
							
								
								
									
										54
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,54 +0,0 @@ | ||||
| { | ||||
|     "files.associations": { | ||||
|         "algorithm": "cpp", | ||||
|         "vector": "cpp", | ||||
|         "cmath": "cpp", | ||||
|         "array": "cpp", | ||||
|         "atomic": "cpp", | ||||
|         "*.tcc": "cpp", | ||||
|         "bitset": "cpp", | ||||
|         "cctype": "cpp", | ||||
|         "clocale": "cpp", | ||||
|         "cstdarg": "cpp", | ||||
|         "cstddef": "cpp", | ||||
|         "cstdint": "cpp", | ||||
|         "cstdio": "cpp", | ||||
|         "cstdlib": "cpp", | ||||
|         "cstring": "cpp", | ||||
|         "ctime": "cpp", | ||||
|         "cwchar": "cpp", | ||||
|         "cwctype": "cpp", | ||||
|         "deque": "cpp", | ||||
|         "unordered_map": "cpp", | ||||
|         "unordered_set": "cpp", | ||||
|         "exception": "cpp", | ||||
|         "functional": "cpp", | ||||
|         "iterator": "cpp", | ||||
|         "map": "cpp", | ||||
|         "memory": "cpp", | ||||
|         "memory_resource": "cpp", | ||||
|         "numeric": "cpp", | ||||
|         "optional": "cpp", | ||||
|         "random": "cpp", | ||||
|         "regex": "cpp", | ||||
|         "string": "cpp", | ||||
|         "string_view": "cpp", | ||||
|         "system_error": "cpp", | ||||
|         "tuple": "cpp", | ||||
|         "type_traits": "cpp", | ||||
|         "utility": "cpp", | ||||
|         "fstream": "cpp", | ||||
|         "initializer_list": "cpp", | ||||
|         "iomanip": "cpp", | ||||
|         "iosfwd": "cpp", | ||||
|         "istream": "cpp", | ||||
|         "limits": "cpp", | ||||
|         "new": "cpp", | ||||
|         "ostream": "cpp", | ||||
|         "sstream": "cpp", | ||||
|         "stdexcept": "cpp", | ||||
|         "streambuf": "cpp", | ||||
|         "cinttypes": "cpp", | ||||
|         "typeinfo": "cpp" | ||||
|     } | ||||
| } | ||||
							
								
								
									
										3363
									
								
								CHANGELOG.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										21
									
								
								LICENSE.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | ||||
| MIT License | ||||
|  | ||||
| Copyright (c) 2025 Manuel Weiser | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
| in the Software without restriction, including without limitation the rights | ||||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| copies of the Software, and to permit persons to whom the Software is | ||||
| furnished to do so, subject to the following conditions: | ||||
|  | ||||
| The above copyright notice and this permission notice shall be included in all | ||||
| copies or substantial portions of the Software. | ||||
|  | ||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
| SOFTWARE. | ||||
							
								
								
									
										227
									
								
								README.de.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,227 @@ | ||||
| # FilaMan - Filament Management System | ||||
|  | ||||
| FilaMan ist ein Filament-Managementsystem für den 3D-Druck. Es verwendet ESP32-Hardware für Gewichtsmessungen und NFC-Tag-Management.  | ||||
| Benutzer können Filamentspulen verwalten, den Status des Automatic Material System (AMS) von Bablulab Druckern überwachen und Einstellungen über eine Weboberfläche vornehmen.  | ||||
| Das System integriert sich nahtlos mit der [Spoolman](https://github.com/Donkie/Spoolman) Filamentverwaltung, zusätzlich mit [Bambulab](https://bambulab.com/en-us) 3D-Druckern und sowie dem [Openspool](https://github.com/spuder/OpenSpool) NFC-TAG Format. | ||||
|  | ||||
|  | ||||
|  | ||||
| Weitere Bilder finden Sie im [img Ordner](/img/) | ||||
| oder auf meiner Website: [FilaMan Website](https://www.filaman.app)   | ||||
| Deutsches Erklärvideo: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaOHU)   | ||||
| 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) | ||||
|  | ||||
| ### ESP32 Hardware-Funktionen | ||||
| - **Gewichtsmessung:** Verwendung einer Wägezelle mit HX711-Verstärker für präzise Gewichtsverfolgung. | ||||
| - **NFC-Tag Lesen/Schreiben:** PN532-Modul zum Lesen und Schreiben von Filamentdaten auf NFC-Tags. | ||||
| - **OLED-Display:** Zeigt aktuelles Gewicht, Verbindungsstatus (WiFi, Bambu Lab, Spoolman). | ||||
| - **WLAN-Konnektivität:** WiFiManager für einfache Netzwerkkonfiguration. | ||||
| - **MQTT-Integration:** Verbindet sich mit Bambu Lab Drucker für AMS-Steuerung. | ||||
| - **NFC-Tag NTAG213 NTAG215:** Verwendung von NTAG213, besser NTAG215 wegen ausreichendem Speicherplatz auf dem Tag | ||||
|  | ||||
| ### Weboberflächen-Funktionen | ||||
| - **Echtzeit-Updates:** WebSocket-Verbindung für Live-Daten-Updates. | ||||
| - **NFC-Tag-Verwaltung:**  | ||||
|     - Filamentdaten auf NFC-Tags schreiben. | ||||
|     - Verwendet das NFC-Tag-Format von [Openspool](https://github.com/spuder/OpenSpool) | ||||
|     - Ermöglicht automatische Spulenerkennung im AMS | ||||
|     - **Hersteller Tag Unterstützung:** Automatische Erstellung von Spoolman-Einträgen aus Hersteller NFC-Tags ([Mehr erfahren](README_ManufacturerTags_DE.md)) | ||||
| - **Bambulab AMS-Integration:**  | ||||
|   - Anzeige der aktuellen AMS-Fachbelegung. | ||||
|   - Zuordnung von Filamenten zu AMS-Slots. | ||||
|   - Unterstützung für externe Spulenhalter. | ||||
| - **Spoolman-Integration:** | ||||
|   - Auflistung verfügbarer Filamentspulen. | ||||
|   - Filtern und Auswählen von Filamenten. | ||||
|   - Automatische Aktualisierung der Spulengewichte. | ||||
|   - Verfolgung von NFC-Tag-Zuweisungen. | ||||
|   - Unterstützt das Spoolman Octoprint Plugin | ||||
|  | ||||
| ### Wenn Sie meine Arbeit unterstützen möchten, freue ich mich über einen Kaffee | ||||
|  | ||||
| <a href="https://www.buymeacoffee.com/manuelw" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/v2/default-yellow.png" alt="Buy Me A Coffee" style="height: 30px !important;width: 108px !important;" ></a> | ||||
|  | ||||
| ## Hersteller Tags Unterstützung | ||||
|  | ||||
| 🎉 **Aufregende Neuigkeiten!** FilaMan unterstützt jetzt **Hersteller Tags** - NFC-Tags, die direkt von Filament-Herstellern vorprogrammiert geliefert werden! | ||||
|  | ||||
| ### Erster Hersteller-Partner: RecyclingFabrik | ||||
|  | ||||
| Wir freuen uns anzukündigen, dass [**RecyclingFabrik**](https://www.recyclingfabrik.de) der **erste Filament-Hersteller** sein wird, der FilaMan unterstützt, indem sie NFC-Tags im FilaMan-Format auf ihren Spulen anbieten! | ||||
|  | ||||
| **Demnächst verfügbar:** RecyclingFabrik-Spulen werden NFC-Tags enthalten, die sich automatisch in Ihr FilaMan-System integrieren, manuelle Einrichtung überflüssig machen und perfekte Kompatibilität gewährleisten. | ||||
|  | ||||
| ### Wie Hersteller Tags funktionieren | ||||
|  | ||||
| Wenn Sie zum ersten Mal einen Hersteller NFC-Tag scannen: | ||||
| 1. **Automatische Markenerkennung:** FilaMan erkennt den Hersteller und erstellt die Marke in Spoolman | ||||
| 2. **Filament-Typ Erstellung:** Alle Materialspezifikationen werden automatisch hinzugefügt | ||||
| 3. **Spulen-Registrierung:** Ihre spezifische Spule wird mit korrektem Gewicht und Spezifikationen registriert | ||||
| 4. **Zukünftige Schnellerkennung:** Nachfolgende Scans verwenden Fast-Path-Erkennung für sofortige Gewichtsmessung | ||||
|  | ||||
| **Für detaillierte technische Informationen:** [Hersteller Tags Dokumentation](README_ManufacturerTags_DE.md) | ||||
|  | ||||
| ### Vorteile für Benutzer | ||||
| - ✅ **Null manuelle Einrichtung** - Einfach scannen und wiegen | ||||
| - ✅ **Perfekte Datengenauigkeit** - Hersteller-verifizierte Spezifikationen | ||||
| - ✅ **Sofortige Integration** - Nahtlose Spoolman-Kompatibilität | ||||
| - ✅ **Zukunftssicher** - Tags funktionieren mit jedem FilaMan-kompatiblen System | ||||
|  | ||||
| ## Detaillierte Funktionalität | ||||
|  | ||||
| ### ESP32-Funktionalität | ||||
| - **Druckaufträge steuern und überwachen:** Der ESP32 kommuniziert mit dem Bambu Lab Drucker. | ||||
| - **Drucker-Kommunikation:** Nutzt MQTT für Echtzeit-Kommunikation mit dem Drucker. | ||||
| - **Benutzerinteraktionen:** Das OLED-Display bietet sofortiges Feedback zum Systemstatus. | ||||
|  | ||||
| ### Weboberflächen-Funktionalität | ||||
| - **Benutzerinteraktionen:** Die Weboberfläche ermöglicht Benutzern die Interaktion mit dem System. | ||||
| - **UI-Elemente:** Enthält Dropdown-Menüs für Hersteller und Filamente, Buttons zum Beschreiben von NFC-Tags und Echtzeit-Statusanzeigen. | ||||
|  | ||||
| ## Hardware-Anforderungen | ||||
|  | ||||
| ### Komponenten (Affiliate Links) | ||||
| - **ESP32 Development Board:** Any ESP32 variant. | ||||
| [Amazon Link](https://amzn.to/3FHea6D) | ||||
| - **HX711 5kg Load Cell Amplifier:** For weight measurement. | ||||
| [Amazon Link](https://amzn.to/4ja1KTe) | ||||
| - **OLED 0.96 Zoll I2C white/yellow Display:** 128x64 SSD1306. | ||||
| [Amazon Link](https://amzn.to/445aaa9) | ||||
| - **PN532 NFC NXP RFID-Modul V3:** For NFC tag operations. | ||||
| [Amazon Link](https://amzn.eu/d/gy9vaBX) | ||||
| - **NFC Tags NTAG213 NTAG215:** RFID Tag | ||||
| [Amazon Link](https://amzn.to/3E071xO) | ||||
| - **TTP223 Touch Sensor (optional):** For reTARE per Button/Touch | ||||
| [Amazon Link](https://amzn.to/4hTChMK) | ||||
|  | ||||
|  | ||||
| ### Pin Konfiguration | ||||
| | Component          | ESP32 Pin | | ||||
| |-------------------|-----------| | ||||
| | HX711 DOUT        | 16        | | ||||
| | HX711 SCK         | 17        | | ||||
| | OLED SDA          | 21        | | ||||
| | OLED SCL          | 22        | | ||||
| | PN532 IRQ         | 32        | | ||||
| | PN532 RESET       | 33        | | ||||
| | PN532 SDA         | 21        | | ||||
| | PN532 SCL         | 22        | | ||||
| | TTP223 I/O        | 25        | | ||||
|  | ||||
| **!! 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 | ||||
|  | ||||
| ### ESP32-Bibliotheken | ||||
| - `WiFiManager`: Netzwerkkonfiguration | ||||
| - `ESPAsyncWebServer`: Webserver-Funktionalität | ||||
| - `ArduinoJson`: JSON-Verarbeitung | ||||
| - `PubSubClient`: MQTT-Kommunikation | ||||
| - `Adafruit_PN532`: NFC-Funktionalität | ||||
| - `Adafruit_SSD1306`: OLED-Display-Steuerung | ||||
| - `HX711`: Wägezellen-Kommunikation | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| ### Voraussetzungen | ||||
| - **Software:** | ||||
|   - [PlatformIO](https://platformio.org/) in VS Code | ||||
|   - [Spoolman](https://github.com/Donkie/Spoolman) Instanz | ||||
| - **Hardware:** | ||||
|   - ESP32 Entwicklungsboard | ||||
|   - HX711 Wägezellen-Verstärker | ||||
|   - Wägezelle (Gewichtssensor) | ||||
|   - OLED Display (128x64 SSD1306) | ||||
|   - PN532 NFC Modul | ||||
|   - Verbindungskabel | ||||
|  | ||||
| ## Wichtiger Hinweis | ||||
| Du musst Spoolman auf DEBUG Modus setzten, da man bisher in Spoolman keine CORS Domains setzen kann! | ||||
|  | ||||
| ``` | ||||
| # Enable debug mode | ||||
| # If enabled, the client will accept requests from any host | ||||
| # This can be useful when developing, but is also a security risk | ||||
| # Default: FALSE | ||||
| #SPOOLMAN_DEBUG_MODE=TRUE | ||||
| ``` | ||||
|  | ||||
| ## Schritt-für-Schritt Installation | ||||
| ### Einfache Installation | ||||
| 1. **Gehe auf [FilaMan Installer](https://www.filaman.app/installer.html)** | ||||
|  | ||||
| 2. **Stecke dein ESP an den Rechner und klicke Connect** | ||||
|  | ||||
| 3. **Wähle dein Device Port und klicke Intall** | ||||
|  | ||||
| 4. **Ersteinrichtung:** | ||||
|     - Mit dem "FilaMan" WLAN-Zugangspunkt verbinden. | ||||
|     - WLAN-Einstellungen über das Konfigurationsportal vornehmen. | ||||
|     - Weboberfläche unter `http://filaman.local` oder der IP-Adresse aufrufen. | ||||
|  | ||||
| ### Compile by yourself | ||||
| 1. **Repository klonen:** | ||||
|     ```bash | ||||
|     git clone https://github.com/ManuelW77/Filaman.git | ||||
|     cd FilaMan | ||||
|     ``` | ||||
| 2. **Abhängigkeiten installieren:** | ||||
|     ```bash | ||||
|     pio lib install | ||||
|     ``` | ||||
| 3. **ESP32 flashen:** | ||||
|     ```bash | ||||
|     pio run --target upload | ||||
|     ``` | ||||
| 4. **Ersteinrichtung:** | ||||
|     - Mit dem "FilaMan" WLAN-Zugangspunkt verbinden. | ||||
|     - WLAN-Einstellungen über das Konfigurationsportal vornehmen. | ||||
|     - Weboberfläche unter `http://filaman.local` oder der IP-Adresse aufrufen. | ||||
|  | ||||
| ## Dokumentation | ||||
|  | ||||
| ### Relevante Links | ||||
| - [PlatformIO Dokumentation](https://docs.platformio.org/) | ||||
| - [Spoolman Dokumentation](https://github.com/Donkie/Spoolman) | ||||
| - [Bambu Lab Drucker Dokumentation](https://www.bambulab.com/) | ||||
|  | ||||
| ### Tutorials und Beispiele | ||||
| - [PlatformIO erste Schritte](https://docs.platformio.org/en/latest/tutorials/espressif32/arduino_debugging_unit_testing.html) | ||||
| - [ESP32 Webserver Tutorial](https://randomnerdtutorials.com/esp32-web-server-arduino-ide/) | ||||
|  | ||||
| ## Lizenz | ||||
|  | ||||
| Dieses Projekt ist unter der MIT-Lizenz lizenziert. Siehe [LICENSE](LICENSE) Datei für Details. | ||||
|  | ||||
| ## Materialien | ||||
|  | ||||
| ### Nützliche Ressourcen | ||||
| - [ESP32 Offizielle Dokumentation](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/) | ||||
| - [Arduino Bibliotheken](https://www.arduino.cc/en/Reference/Libraries) | ||||
| - [NFC Tag Informationen](https://learn.adafruit.com/adafruit-pn532-rfid-nfc/overview) | ||||
|  | ||||
| ### Community und Support | ||||
| - [PlatformIO Community](https://community.platformio.org/) | ||||
| - [Arduino Forum](https://forum.arduino.cc/) | ||||
| - [ESP32 Forum](https://www.esp32.com/) | ||||
|  | ||||
| ## Verfügbarkeit | ||||
|  | ||||
| Der Code kann getestet und die Anwendung kann vom [GitHub Repository](https://github.com/ManuelW77/Filaman) heruntergeladen werden. | ||||
|  | ||||
| ### Wenn Sie meine Arbeit unterstützen möchten, freue ich mich über einen Kaffee | ||||
| <a href="https://www.buymeacoffee.com/manuelw" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/v2/default-yellow.png" alt="Buy Me A Coffee" style="height: 30px !important;width: 108px !important;" ></a> | ||||
							
								
								
									
										210
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,10 +1,21 @@ | ||||
| # FilaMan - Filament Management System | ||||
|  | ||||
| A comprehensive filament management system combining ESP32-based hardware for weight measurement and NFC tag reading/writing with a web interface for managing filament spools in conjunction with Bambu Lab AMS and Spoolman. | ||||
| [Deutsche Version](README.de.md) | ||||
|  | ||||
| ## Project Overview | ||||
| FilaMan is a filament management system for 3D printing. It uses ESP32 hardware for weight measurement and NFC tag management.  | ||||
| Users can manage filament spools, monitor the status of the Automatic Material System (AMS) and make settings via a web interface.  | ||||
| The system integrates seamlessly with [Bambulab](https://bambulab.com/en-us) 3D printers and [Spoolman](https://github.com/Donkie/Spoolman) filament management as well as the [Openspool](https://github.com/spuder/OpenSpool) NFC-TAG format. | ||||
|  | ||||
| FilaMan is designed to streamline the management of filament spools for 3D printing. The system consists of an ESP32 microcontroller that handles weight measurement and NFC tag operations, and a web interface that allows users to manage filament spools, monitor AMS (Automatic Material System) status, and interact with Spoolman and Bambu Lab printers. | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| More Images can be found in the [img Folder](/img/)   | ||||
| or my website: [FilaMan Website](https://www.filaman.app)   | ||||
| german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaOHU)   | ||||
| Discord Server: [https://discord.gg/my7Gvaxj2v](https://discord.gg/my7Gvaxj2v) | ||||
|  | ||||
| ### Now more detailed informations about the usage: [Wiki](https://github.com/ManuelW77/Filaman/wiki) | ||||
|  | ||||
| ### ESP32 Hardware Features | ||||
| - **Weight Measurement:** Using a load cell with HX711 amplifier for precise weight tracking. | ||||
| @@ -12,13 +23,16 @@ FilaMan is designed to streamline the management of filament spools for 3D print | ||||
| - **OLED Display:** Shows current weight, connection status (WiFi, Bambu Lab, Spoolman). | ||||
| - **WiFi Connectivity:** WiFiManager for easy network configuration. | ||||
| - **MQTT Integration:** Connects to Bambu Lab printer for AMS control. | ||||
| - **Data Persistence:** Stores calibration data in EEPROM. | ||||
| - **Watchdog Timer:** Ensures system stability. | ||||
| - **NFC-Tag NTAG213 NTAG215:** Use NTAG213, better NTAG215 because of enaught space on the Tag | ||||
|  | ||||
| ### Web Interface Features | ||||
| - **Real-time Updates:** WebSocket connection for live data updates. | ||||
| - **NFC Tag Management:** Write filament data to NFC tags. | ||||
| - **AMS Integration:**  | ||||
| - **NFC Tag Management:**  | ||||
| 	- Write filament data to NFC tags. | ||||
| 	- uses NFC-Tag Format of [Openspool](https://github.com/spuder/OpenSpool) | ||||
| 	- so you can use it with automatic Spool detection in AMS | ||||
| 	- **Manufacturer Tag Support:** Automatic creation of Spoolman entries from manufacturer NFC tags ([Learn more](README_ManufacturerTags_EN.md)) | ||||
| - **Bambulab AMS Integration:**  | ||||
|   - Display current AMS tray contents. | ||||
|   - Assign filaments to AMS slots. | ||||
|   - Support for external spool holder. | ||||
| @@ -27,6 +41,37 @@ FilaMan is designed to streamline the management of filament spools for 3D print | ||||
|   - Filter and select filaments. | ||||
|   - Update spool weights automatically. | ||||
|   - Track NFC tag assignments. | ||||
|   - Supports Spoolman Octoprint Plugin | ||||
|  | ||||
| ### If you want to support my work, i would be happy to get a coffe | ||||
|  | ||||
| <a href="https://www.buymeacoffee.com/manuelw" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/v2/default-yellow.png" alt="Buy Me A Coffee" style="height: 30px !important;width: 108px !important;" ></a> | ||||
|  | ||||
| ## Manufacturer Tags Support | ||||
|  | ||||
| 🎉 **Exciting News!** FilaMan now supports **Manufacturer Tags** - NFC tags that come pre-programmed directly from filament manufacturers! | ||||
|  | ||||
| ### First Manufacturer Partner: RecyclingFabrik | ||||
|  | ||||
| We're thrilled to announce that [**RecyclingFabrik**](https://www.recyclingfabrik.de) will be the **first filament manufacturer** to support FilaMan by offering NFC tags in the FilaMan format on their spools! | ||||
|  | ||||
| **Coming Soon:** RecyclingFabrik spools will include NFC tags that automatically integrate with your FilaMan system, eliminating manual setup and ensuring perfect compatibility. | ||||
|  | ||||
| ### How Manufacturer Tags Work | ||||
|  | ||||
| When you scan a manufacturer NFC tag for the first time: | ||||
| 1. **Automatic Brand Detection:** FilaMan recognizes the manufacturer and creates the brand in Spoolman | ||||
| 2. **Filament Type Creation:** All material specifications are automatically added | ||||
| 3. **Spool Registration:** Your specific spool is registered with proper weight and specifications | ||||
| 4. **Future Fast Recognition:** Subsequent scans use fast-path detection for instant weight measurement | ||||
|  | ||||
| **For detailed technical information:** [Manufacturer Tags Documentation](README_ManufacturerTags_EN.md) | ||||
|  | ||||
| ### Benefits for Users | ||||
| - ✅ **Zero Manual Setup** - Just scan and weigh | ||||
| - ✅ **Perfect Data Accuracy** - Manufacturer-verified specifications | ||||
| - ✅ **Instant Integration** - Seamless Spoolman compatibility | ||||
| - ✅ **Future-Proof** - Tags work with any FilaMan-compatible system | ||||
|  | ||||
| ## Detailed Functionality | ||||
|  | ||||
| @@ -39,13 +84,67 @@ FilaMan is designed to streamline the management of filament spools for 3D print | ||||
| - **User Interactions:** The web interface allows users to interact with the system, select filaments, write NFC tags, and monitor AMS status. | ||||
| - **UI Elements:** Includes dropdowns for selecting manufacturers and filaments, buttons for writing NFC tags, and real-time status indicators. | ||||
|  | ||||
| ## Installation | ||||
| ## Hardware Requirements | ||||
|  | ||||
| ### Prerequisites | ||||
| ### Components (Affiliate Links) | ||||
| - **ESP32 Development Board:** Any ESP32 variant. | ||||
| [Amazon Link](https://amzn.to/3FHea6D) | ||||
| - **HX711 5kg Load Cell Amplifier:** For weight measurement. | ||||
| [Amazon Link](https://amzn.to/4ja1KTe) | ||||
| - **OLED 0.96 Zoll I2C white/yellow Display:** 128x64 SSD1306. | ||||
| [Amazon Link](https://amzn.to/445aaa9) | ||||
| - **PN532 NFC NXP RFID-Modul V3:** For NFC tag operations. | ||||
| [Amazon Link](https://amzn.eu/d/gy9vaBX) | ||||
| - **NFC Tags NTAG213 NTAG215:** RFID Tag | ||||
| [Amazon Link](https://amzn.to/3E071xO) | ||||
| - **TTP223 Touch Sensor (optional):** For reTARE per Button/Touch | ||||
| [Amazon Link](https://amzn.to/4hTChMK) | ||||
|  | ||||
|  | ||||
| ### Pin Configuration | ||||
| | Component          | ESP32 Pin | | ||||
| |-------------------|-----------| | ||||
| | HX711 DOUT        | 16        | | ||||
| | HX711 SCK         | 17        | | ||||
| | OLED SDA          | 21        | | ||||
| | OLED SCL          | 22        | | ||||
| | PN532 IRQ         | 32        | | ||||
| | PN532 RESET       | 33        | | ||||
| | PN532 SDA         | 21        | | ||||
| | PN532 SCL         | 22        | | ||||
| | TTP223 I/O        | 25        | | ||||
|  | ||||
| **!! 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 | ||||
|  | ||||
| ### ESP32 Libraries | ||||
| - `WiFiManager`: Network configuration | ||||
| - `ESPAsyncWebServer`: Web server functionality | ||||
| - `ArduinoJson`: JSON parsing and creation | ||||
| - `PubSubClient`: MQTT communication | ||||
| - `Adafruit_PN532`: NFC functionality | ||||
| - `Adafruit_SSD1306`: OLED display control | ||||
| - `HX711`: Load cell communication | ||||
|  | ||||
| ### Installation | ||||
|  | ||||
| ## Prerequisites | ||||
| - **Software:** | ||||
|   - [PlatformIO](https://platformio.org/) in VS Code | ||||
|   - [Spoolman](https://github.com/Donkie/Spoolman) instance | ||||
|   - Bambu Lab printer (optional for AMS integration) | ||||
| - **Hardware:** | ||||
|   - ESP32 Development Board | ||||
|   - HX711 Load Cell Amplifier | ||||
| @@ -54,10 +153,35 @@ FilaMan is designed to streamline the management of filament spools for 3D print | ||||
|   - PN532 NFC Module | ||||
|   - Connecting wires | ||||
|  | ||||
| ### Step-by-Step Installation | ||||
| ## Important Note | ||||
| You have to activate Spoolman in debug mode, because you are not able to set CORS Domains in Spoolman yet. | ||||
|  | ||||
| ``` | ||||
| # Enable debug mode | ||||
| # If enabled, the client will accept requests from any host | ||||
| # This can be useful when developing, but is also a security risk | ||||
| # Default: FALSE | ||||
| #SPOOLMAN_DEBUG_MODE=TRUE | ||||
| ``` | ||||
|  | ||||
|  | ||||
| ## Step-by-Step Installation | ||||
| ### Easy Installation | ||||
| 1. **Go to [FilaMan Installer](https://www.filaman.app/installer.html)** | ||||
|  | ||||
| 2. **Plug you device in and push Connect button** | ||||
|  | ||||
| 3. **Select your Device Port and push Intall** | ||||
|  | ||||
| 4. **Initial Setup:** | ||||
|     - Connect to the "FilaMan" WiFi access point. | ||||
|     - Configure WiFi settings through the captive portal. | ||||
|     - Access the web interface at `http://filaman.local` or the IP address. | ||||
|  | ||||
| ### Compile by yourself | ||||
| 1. **Clone the Repository:** | ||||
|     ```bash | ||||
|     git clone https://github.com/yourusername/FilaMan.git | ||||
|     git clone https://github.com/ManuelW77/Filaman.git | ||||
|     cd FilaMan | ||||
|     ``` | ||||
| 2. **Install Dependencies:** | ||||
| @@ -73,63 +197,6 @@ FilaMan is designed to streamline the management of filament spools for 3D print | ||||
|     - Configure WiFi settings through the captive portal. | ||||
|     - Access the web interface at `http://filaman.local` or the IP address. | ||||
|  | ||||
| ## Hardware Requirements | ||||
|  | ||||
| ### Components | ||||
| - **ESP32 Development Board:** Any ESP32 variant. | ||||
| - **HX711 Load Cell Amplifier:** For weight measurement. | ||||
| - **Load Cell:** Weight sensor. | ||||
| - **OLED Display:** 128x64 SSD1306. | ||||
| - **PN532 NFC Module:** For NFC tag operations. | ||||
| - **Connecting Wires:** For connections. | ||||
|  | ||||
| ### Pin Configuration | ||||
| | Component          | ESP32 Pin | | ||||
| |-------------------|-----------| | ||||
| | HX711 DOUT        | 16        | | ||||
| | HX711 SCK         | 17        | | ||||
| | OLED SDA          | 21        | | ||||
| | OLED SCL          | 22        | | ||||
| | PN532 IRQ         | 32        | | ||||
| | PN532 RESET       | 33        | | ||||
|  | ||||
| ## Software Dependencies | ||||
|  | ||||
| ### ESP32 Libraries | ||||
| - `WiFiManager`: Network configuration | ||||
| - `ESPAsyncWebServer`: Web server functionality | ||||
| - `ArduinoJson`: JSON parsing and creation | ||||
| - `PubSubClient`: MQTT communication | ||||
| - `Adafruit_PN532`: NFC functionality | ||||
| - `Adafruit_SSD1306`: OLED display control | ||||
| - `HX711`: Load cell communication | ||||
|  | ||||
| ### External Services | ||||
| - **Bambu Lab Printer:** For AMS integration. | ||||
| - **Spoolman:** For filament management. | ||||
|  | ||||
| ## API Communication | ||||
|  | ||||
| ### Spoolman Integration | ||||
| The system communicates with Spoolman using its REST API for: | ||||
| - Fetching spool information. | ||||
| - Updating spool weights. | ||||
| - Managing NFC tag assignments. | ||||
|  | ||||
| ### Data Format | ||||
| ```json | ||||
| { | ||||
|   "version": "2.0", | ||||
|   "protocol": "openspool", | ||||
|   "color_hex": "FFFFFF", | ||||
|   "type": "PLA", | ||||
|   "min_temp": 200, | ||||
|   "max_temp": 220, | ||||
|   "brand": "Vendor", | ||||
|   "sm_id": "1234" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## Documentation | ||||
|  | ||||
| ### Relevant Links | ||||
| @@ -159,4 +226,7 @@ This project is licensed under the MIT License. See the [LICENSE](LICENSE) file | ||||
|  | ||||
| ## Availability | ||||
|  | ||||
| The code can be tested and the application can be downloaded from the [GitHub repository](https://github.com/yourusername/FilaMan). | ||||
| The code can be tested and the application can be downloaded from the [GitHub repository](https://github.com/ManuelW77/Filaman). | ||||
|  | ||||
| ### If you want to support my work, i would be happy to get a coffe | ||||
| <a href="https://www.buymeacoffee.com/manuelw" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/v2/default-yellow.png" alt="Buy Me A Coffee" style="height: 30px !important;width: 108px !important;" ></a> | ||||
							
								
								
									
										159
									
								
								README_ManufacturerTags_DE.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,159 @@ | ||||
| # Hersteller Tags - Deutsche Dokumentation | ||||
|  | ||||
| ## Überblick | ||||
|  | ||||
| Das FilaMan NFC-System unterstützt **Hersteller Tags**, die es Filament-Produzenten ermöglichen, standardisierte NFC-Tags für ihre Produkte zu erstellen. Beim Scannen dieser Tags werden automatisch die notwendigen Einträge in Spoolman (Marke, Filament-Typ und Spule) erstellt, ohne dass eine manuelle Einrichtung erforderlich ist. | ||||
|  | ||||
| ## Funktionsweise der Hersteller Tags | ||||
|  | ||||
| ### Ablauf | ||||
|  | ||||
| 1. **Tag-Erkennung**: Wenn ein Tag ohne `sm_id` gescannt wird, prüft das System auf Hersteller Tag Format | ||||
| 2. **Marken-Erstellung/Suche**: Das System sucht die Marke in Spoolman oder erstellt sie, falls sie nicht existiert | ||||
| 3. **Filament-Typ-Erstellung/Suche**: Der Filament-Typ wird basierend auf Marke, Material und Spezifikationen erstellt oder gefunden | ||||
| 4. **Spulen-Erstellung**: Ein neuer Spulen-Eintrag wird automatisch mit der Tag-UID als Referenz erstellt | ||||
| 5. **Tag-Update**: Der Tag wird mit der neuen Spoolman Spulen-ID (`sm_id`) aktualisiert | ||||
|  | ||||
| ### Warum Hersteller Tags verwenden? | ||||
|  | ||||
| - **Automatische Integration**: Keine manuelle Dateneingabe erforderlich | ||||
| - **Standardisiertes Format**: Konsistente Produktinformationen verschiedener Hersteller | ||||
| - **Lagerverwaltung**: Automatische Erstellung vollständiger Spoolman-Einträge | ||||
| - **Rückverfolgbarkeit**: Direkte Verbindung zwischen physischem Produkt und digitalem Inventar | ||||
|  | ||||
| ## Tag-Format Spezifikation | ||||
|  | ||||
| ### JSON-Struktur | ||||
|  | ||||
| Hersteller Tags müssen eine JSON-Payload mit spezifischen Feldern enthalten, die **kurze Schlüssel** verwenden, um die Tag-Größe zu minimieren: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "b": "Marke/Hersteller Name", | ||||
|     "an": "Artikelnummer", | ||||
|     "t": "Filament Typ (PLA, PETG, etc)", | ||||
|     "c": "Filament Farbe ohne # (FF5733)", | ||||
|     "mc": "Optional Mehrfarben-Filament Farben ohne # (FF0000,00FF00,0000FF)", | ||||
|     "mcd": "Optional Mehrfarben-Richtung als Wort (coaxial, longitudinal)", | ||||
|     "cn": "Farbname (rot, Blaubeere, Arktisches Blau)", | ||||
|     "et": "Extruder Temp als Zahl in C° (230)", | ||||
|     "bt": "Bett Temp als Zahl in C° (60)", | ||||
|     "di": "Durchmesser als Float (1.75)", | ||||
|     "de": "Dichte als Float (1.24)", | ||||
|     "sw": "Leeres Spulengewicht als Zahl in g (180)", | ||||
|     "u": "URL zum Filament mit der Artikelnummer" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Pflichtfelder | ||||
|  | ||||
| - **`b`** (brand): Hersteller/Markenname | ||||
| - **`an`** (article number): Eindeutige Produktkennung | ||||
| - **`t`** (type): Materialtyp (PLA, PETG, ABS, etc.) | ||||
| - **`c`** (color): Hex-Farbcode ohne # | ||||
| - **`cn`** (color name): Lesbare Farbbezeichnung | ||||
| - **`et`** (extruder temp): Empfohlene Extruder-Temperatur in Celsius | ||||
| - **`bt`** (bed temp): Empfohlene Bett-Temperatur in Celsius | ||||
| - **`di`** (diameter): Filamentdurchmesser in mm | ||||
| - **`de`** (density): Materialdichte in g/cm³ | ||||
| - **`sw`** (spool weight): Leeres Spulengewicht in Gramm | ||||
|  | ||||
| ### Optionale Felder | ||||
|  | ||||
| - **`mc`** (multicolor): Komma-getrennte Hex-Farben für Mehrfarben-Filamente | ||||
| - **`mcd`** (multicolor direction): Richtung für Mehrfarben (coaxial, longitudinal) | ||||
| - **`u`** (url): Produkt-URL mit direktem Link zum Artikel zB für Nachbestellung | ||||
|  | ||||
| ### Beispiel Tag | ||||
|  | ||||
| ```json | ||||
| {"b":"Recycling Fabrik","an":"FX1_PETG-S175-1000-DAEM00055","t":"PETG","c":"FF5733","cn":"Lebendiges Orange","et":"230","bt":"70","di":"1.75","de":"1.24","sw":"180","u":"https://www.recyclingfabrik.com/search?q="} | ||||
| ``` | ||||
|  | ||||
| ## Implementierungsrichtlinien | ||||
|  | ||||
| ### Für Hersteller | ||||
|  | ||||
| 1. **Tag-Kodierung**: NDEF-Format mit MIME-Typ `application/json` verwenden | ||||
| 2. **Datenminimierung**: Kompaktes JSON-Format für Tag-Größenbegrenzungen nutzen | ||||
| 3. **Qualitätskontrolle**: Sicherstellen, dass alle Pflichtfelder vorhanden und korrekt formatiert sind | ||||
| 4. **Testen**: Tags vor der Produktion mit dem FilaMan-System verifizieren | ||||
|  | ||||
| ### Tag-Größe Überlegungen | ||||
|  | ||||
| - **NTAG213**: 144 Bytes Nutzerdaten (geeignet für einfache Tags) | ||||
| - **NTAG215**: 504 Bytes Nutzerdaten (empfohlen für umfassende Daten) | ||||
| - **NTAG216**: 888 Bytes Nutzerdaten (maximale Kompatibilität) | ||||
|  | ||||
| ### Best Practices | ||||
|  | ||||
| - Markennamen über alle Produkte hinweg konsistent halten | ||||
| - Standardisierte Materialtypnamen verwenden (PLA, PETG, ABS, etc.) | ||||
| - Genaue Temperaturempfehlungen angeben | ||||
| - Aussagekräftige Farbnamen für bessere Benutzererfahrung verwenden | ||||
| - Tags vor Massenproduktion mit dem FilaMan-System testen | ||||
|  | ||||
| ## System-Integration | ||||
|  | ||||
| ### Spoolman Datenbankstruktur | ||||
|  | ||||
| Bei der Verarbeitung eines Hersteller Tags erstellt das System: | ||||
|  | ||||
| 1. **Lieferanten-Eintrag**: Markeninformationen in der Spoolman Lieferanten-Datenbank | ||||
| 2. **Filament-Eintrag**: Materialspezifikationen und Eigenschaften | ||||
| 3. **Spulen-Eintrag**: Einzelne Spule mit Gewicht und NFC-Tag-Referenz | ||||
|  | ||||
| ### Fast-Path Erkennung | ||||
|  | ||||
| Sobald ein Tag verarbeitet und mit `sm_id` aktualisiert wurde, nutzt er das Fast-Path-Erkennungssystem für schnelle nachfolgende Scans. | ||||
|  | ||||
| ## Fehlerbehebung | ||||
|  | ||||
| ### Häufige Probleme | ||||
|  | ||||
| - **Tag zu klein**: NTAG215 oder NTAG216 für größere JSON-Payloads verwenden | ||||
| - **Fehlende Felder**: Sicherstellen, dass alle Pflichtfelder vorhanden sind | ||||
| - **Ungültiges Format**: JSON-Syntax und Feldtypen überprüfen | ||||
| - **Spoolman-Verbindung**: Sicherstellen, dass FilaMan mit der Spoolman API verbinden kann | ||||
|  | ||||
| ### Validierung | ||||
|  | ||||
| Das System validiert: | ||||
|  | ||||
| - JSON-Format Korrektheit | ||||
| - Vorhandensein der Pflichtfelder | ||||
| - Datentyp-Konformität | ||||
| - Tag-Größe Kompatibilität | ||||
|  | ||||
| ## Technische Details | ||||
|  | ||||
| ### Verarbeitungsalgorithmus | ||||
|  | ||||
| 1. Tag-Scan erkennt kein `sm_id` Feld | ||||
| 2. System prüft auf `b` (Marke) und `an` (Artikelnummer) Felder | ||||
| 3. `checkVendor()` erstellt oder findet Marke in Spoolman | ||||
| 4. `checkFilament()` erstellt oder findet Filament-Typ | ||||
| 5. `createSpool()` erstellt neuen Spulen-Eintrag | ||||
| 6. Tag wird mit neuer `sm_id` aktualisiert | ||||
|  | ||||
| ### Fehlerbehandlung | ||||
|  | ||||
| - Graceful Fallback bei Netzwerkproblemen | ||||
| - Detaillierte Protokollierung für Debugging | ||||
| - Benutzer-Feedback bei fehlgeschlagenen Operationen | ||||
| - Wiederholungsmechanismen für temporäre Fehler | ||||
|  | ||||
| ### Systemverhalten | ||||
|  | ||||
| #### Bei fehlendem sm_id: | ||||
| - System prüft auf `b` (brand) und `an` (artnr) Felder | ||||
| - Falls vorhanden → Hersteller Tag erkannt | ||||
| - Automatische Erstellung von Lieferant, Filament und Spule in Spoolman | ||||
| - Tag wird mit neuer `sm_id` beschrieben | ||||
|  | ||||
| #### Bei vorhandenem sm_id: | ||||
| - Fast-Path Erkennung für bekannte Spulen | ||||
| - Sofortige Gewichtsmessung ohne vollständige Tag-Analyse | ||||
| - Optimierte Performance für häufig verwendete Tags | ||||
|  | ||||
| Dieses System ermöglicht eine nahtlose Integration von Hersteller-Filamentprodukten in das FilaMan-Ökosystem unter Beibehaltung von Datenkonsistenz und Benutzererfahrung. | ||||
							
								
								
									
										145
									
								
								README_ManufacturerTags_EN.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,145 @@ | ||||
| # Manufacturer Tags - English Documentation | ||||
|  | ||||
| ## Overview | ||||
|  | ||||
| The FilaMan NFC system supports **Manufacturer Tags** that allow filament producers to create standardized NFC tags for their products. When scanned, these tags automatically create the necessary entries in Spoolman (brand, filament type, and spool) without requiring manual setup. | ||||
|  | ||||
| ## How Manufacturer Tags Work | ||||
|  | ||||
| ### Process Flow | ||||
|  | ||||
| 1. **Tag Detection**: When a tag without `sm_id` is scanned, the system checks for manufacturer tag format | ||||
| 2. **Brand Creation/Lookup**: The system searches for the brand in Spoolman or creates it if it doesn't exist | ||||
| 3. **Filament Type Creation/Lookup**: The filament type is created or found based on brand, material, and specifications | ||||
| 4. **Spool Creation**: A new spool entry is automatically created with the tag's UID as reference | ||||
| 5. **Tag Update**: The tag is updated with the new Spoolman spool ID (`sm_id`) | ||||
|  | ||||
| ### Why Use Manufacturer Tags? | ||||
|  | ||||
| - **Automatic Integration**: No manual data entry required | ||||
| - **Standardized Format**: Consistent product information across different manufacturers | ||||
| - **Inventory Management**: Automatic creation of complete Spoolman entries | ||||
| - **Traceability**: Direct link between physical product and digital inventory | ||||
|  | ||||
| ## Tag Format Specification | ||||
|  | ||||
| ### JSON Structure | ||||
|  | ||||
| Manufacturer tags must contain a JSON payload with specific fields using **short keys** to minimize tag size: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "b": "Brand/Vendor Name", | ||||
|     "an": "Article Number", | ||||
|     "t": "Filament Type (PLA, PETG, etc)", | ||||
|     "c": "Filament Color without # (FF5733)", | ||||
|     "mc": "Optional Multicolor Filament Colors without # (FF0000,00FF00,0000FF)", | ||||
|     "mcd": "Optional Multicolor Direction as Word (coaxial, longitudinal)", | ||||
|     "cn": "Color Name (red, Blueberry, Arctic Blue)", | ||||
|     "et": "Extruder Temp as Number in C° (230)", | ||||
|     "bt": "Bed Temp as Number in C° (60)", | ||||
|     "di": "Diameter as Float (1.75)", | ||||
|     "de": "Density as Float (1.24)", | ||||
|     "sw": "Empty Spool Weight as Number in g (180)", | ||||
|     "u": "URL to get the Filament with the Article Number" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Required Fields | ||||
|  | ||||
| - **`b`** (brand): Manufacturer/brand name | ||||
| - **`an`** (article number): Unique product identifier | ||||
| - **`t`** (type): Material type (PLA, PETG, ABS, etc.) | ||||
| - **`c`** (color): Hex color code without # | ||||
| - **`cn`** (color name): Human-readable color name | ||||
| - **`et`** (extruder temp): Recommended extruder temperature in Celsius | ||||
| - **`bt`** (bed temp): Recommended bed temperature in Celsius | ||||
| - **`di`** (diameter): Filament diameter in mm | ||||
| - **`de`** (density): Material density in g/cm³ | ||||
| - **`sw`** (spool weight): Empty spool weight in grams | ||||
|  | ||||
| ### Optional Fields | ||||
|  | ||||
| - **`mc`** (multicolor): Comma-separated hex colors for multicolor filaments | ||||
| - **`mcd`** (multicolor direction): Direction for multicolor (coaxial, longitudinal) | ||||
| - **`u`** (url): Product URL with direct link to the article e.g. for reordering | ||||
|  | ||||
| ### Example Tag | ||||
|  | ||||
| ```json | ||||
| {"b":"Recycling Fabrik","an":"FX1_PETG-S175-1000-DAEM00055","t":"PETG","c":"FF5733","cn":"Vibrant Orange","et":"230","bt":"70","di":"1.75","de":"1.24","sw":"180","u":"https://www.recyclingfabrik.com/search?q="} | ||||
| ``` | ||||
|  | ||||
| ## Implementation Guidelines | ||||
|  | ||||
| ### For Manufacturers | ||||
|  | ||||
| 1. **Tag Encoding**: Use NDEF format with MIME type `application/json` | ||||
| 2. **Data Minimization**: Use the compact JSON format to fit within tag size limits | ||||
| 3. **Quality Control**: Ensure all required fields are present and correctly formatted | ||||
| 4. **Testing**: Verify tags work with FilaMan system before production | ||||
|  | ||||
| ### Tag Size Considerations | ||||
|  | ||||
| - **NTAG213**: 144 bytes user data (suitable for basic tags) | ||||
| - **NTAG215**: 504 bytes user data (recommended for comprehensive data) | ||||
| - **NTAG216**: 888 bytes user data (maximum compatibility) | ||||
|  | ||||
| ### Best Practices | ||||
|  | ||||
| - Keep brand names consistent across all products | ||||
| - Use standardized material type names (PLA, PETG, ABS, etc.) | ||||
| - Provide accurate temperature recommendations | ||||
| - Include meaningful color names for user experience | ||||
| - Test tags with the FilaMan system before mass production | ||||
|  | ||||
| ## System Integration | ||||
|  | ||||
| ### Spoolman Database Structure | ||||
|  | ||||
| When a manufacturer tag is processed, the system creates: | ||||
|  | ||||
| 1. **Vendor Entry**: Brand information in Spoolman vendor database | ||||
| 2. **Filament Entry**: Material specifications and properties | ||||
| 3. **Spool Entry**: Individual spool with weight and NFC tag reference | ||||
|  | ||||
| ### Fast-Path Recognition | ||||
|  | ||||
| Once a tag is processed and updated with `sm_id`, it uses the fast-path recognition system for quick subsequent scans. | ||||
|  | ||||
| ## Troubleshooting | ||||
|  | ||||
| ### Common Issues | ||||
|  | ||||
| - **Tag Too Small**: Use NTAG215 or NTAG216 for larger JSON payloads | ||||
| - **Missing Fields**: Ensure all required fields are present | ||||
| - **Invalid Format**: Verify JSON syntax and field types | ||||
| - **Spoolman Connection**: Ensure FilaMan can connect to Spoolman API | ||||
|  | ||||
| ### Validation | ||||
|  | ||||
| The system validates: | ||||
| - JSON format correctness | ||||
| - Required field presence | ||||
| - Data type compliance | ||||
| - Tag size compatibility | ||||
|  | ||||
| ## Technical Details | ||||
|  | ||||
| ### Processing Algorithm | ||||
|  | ||||
| 1. Tag scan detects no `sm_id` field | ||||
| 2. System checks for `b` (brand) and `an` (article number) fields | ||||
| 3. `checkVendor()` creates or finds brand in Spoolman | ||||
| 4. `checkFilament()` creates or finds filament type | ||||
| 5. `createSpool()` creates new spool entry | ||||
| 6. Tag is updated with new `sm_id` | ||||
|  | ||||
| ### Error Handling | ||||
|  | ||||
| - Graceful fallback for network issues | ||||
| - Detailed logging for debugging | ||||
| - User feedback for failed operations | ||||
| - Retry mechanisms for temporary failures | ||||
|  | ||||
| This system enables seamless integration of manufacturer filament products into the FilaMan ecosystem while maintaining data consistency and user experience. | ||||
							
								
								
									
										468
									
								
								WIKI_DE.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,468 @@ | ||||
| # FilaMan Wiki - Deutsch | ||||
|  | ||||
| ## Inhaltsverzeichnis | ||||
|  | ||||
| 1. [Überblick](#überblick) | ||||
| 2. [Installation](#installation) | ||||
| 3. [Hardware-Anforderungen](#hardware-anforderungen) | ||||
| 4. [Ersteinrichtung](#ersteinrichtung) | ||||
| 5. [Konfiguration](#konfiguration) | ||||
| 6. [Benutzung](#benutzung) | ||||
| 7. [NFC-Tags](#nfc-tags) | ||||
| 8. [Bambu Lab Integration](#bambu-lab-integration) | ||||
| 9. [Spoolman Integration](#spoolman-integration) | ||||
| 10. [Octoprint Integration](#octoprint-integration) | ||||
| 11. [Hersteller Tags](#hersteller-tags) | ||||
| 12. [Fehlerbehebung](#fehlerbehebung) | ||||
| 13. [Support](#support) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Überblick | ||||
|  | ||||
| FilaMan ist ein umfassendes Filament-Managementsystem für 3D-Drucker, das auf ESP32-Hardware basiert. Es bietet Gewichtsmessung, NFC-Tag-Management und nahtlose Integration mit Spoolman und Bambu Lab 3D-Druckern. | ||||
|  | ||||
| ### Hauptfunktionen | ||||
|  | ||||
| - **Präzise Gewichtsmessung** mit HX711 Wägezellen-Verstärker | ||||
| - **NFC-Tag Lesen und Schreiben** für Filament-Identifikation | ||||
| - **OLED-Display** für Status-Anzeigen | ||||
| - **WiFi-Konnektivität** mit einfacher Konfiguration | ||||
| - **Webbasierte Benutzeroberfläche** mit Echtzeit-Updates | ||||
| - **Spoolman-Integration** für Lagerverwaltung | ||||
| - **Bambu Lab AMS-Steuerung** via MQTT | ||||
| - **Openspool NFC-Format** Kompatibilität | ||||
| - **Hersteller Tag Unterstützung** für automatische Einrichtung | ||||
|  | ||||
| ### Systemvoraussetzungen | ||||
|  | ||||
| - **ESP32 Development Board** | ||||
| - **Spoolman Instanz** (erforderlich für volle Funktionalität) | ||||
| - **WiFi-Netzwerk** | ||||
| - **Webbrowser** (Chrome/Firefox/Safari) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| ### Einfache Installation (Empfohlen) | ||||
|  | ||||
| 1. **Öffnen Sie den [FilaMan Web-Installer](https://www.filaman.app/installer.html)** | ||||
|    - Verwenden Sie einen Chrome-basierten Browser | ||||
|  | ||||
| 2. **ESP32 vorbereiten** | ||||
|    - Verbinden Sie den ESP32 über USB mit Ihrem Computer | ||||
|    - Klicken Sie auf "Connect" | ||||
|  | ||||
| 3. **Port auswählen** | ||||
|    - Wählen Sie den entsprechenden USB-Port aus | ||||
|    - Bestätigen Sie die Auswahl | ||||
|  | ||||
| 4. **Installation starten** | ||||
|    - Klicken Sie auf "FilaMan installieren" | ||||
|    - Warten Sie, bis der Installationsvorgang abgeschlossen ist | ||||
|  | ||||
| ### Manuelle Kompilierung | ||||
|  | ||||
| Für erfahrene Benutzer mit PlatformIO: | ||||
|  | ||||
| ```bash | ||||
| git clone https://github.com/ManuelW77/Filaman.git | ||||
| cd FilaMan/esp32 | ||||
| pio lib install | ||||
| pio run --target upload | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Hardware-Anforderungen | ||||
|  | ||||
| ### Erforderliche Komponenten | ||||
|  | ||||
| | Komponente | Beschreibung | Amazon Link (Affiliate) | | ||||
| |------------|--------------|-------------------------| | ||||
| | ESP32 Development Board | Jede ESP32-Variante | [Amazon](https://amzn.to/3FHea6D) | | ||||
| | HX711 + Wägezelle | 5kg Load Cell Amplifier | [Amazon](https://amzn.to/4ja1KTe) | | ||||
| | OLED Display | 0.96" I2C 128x64 SSD1306 | [Amazon](https://amzn.to/445aaa9) | | ||||
| | PN532 NFC Modul | V3 RFID-Modul | [Amazon](https://amzn.eu/d/gy9vaBX) | | ||||
| | NFC Tags | NTAG213/NTAG215 | [Amazon](https://amzn.to/3E071xO) | | ||||
| | TTP223 Touch Sensor | Optional für Tara-Funktion | [Amazon](https://amzn.to/4hTChMK) | | ||||
|  | ||||
| ### Pin-Konfiguration | ||||
|  | ||||
| | Komponente | ESP32 Pin | Funktion | | ||||
| |------------|-----------|----------| | ||||
| | HX711 DOUT | 16 | Datenausgang Wägezelle | | ||||
| | HX711 SCK | 17 | Takt Wägezelle | | ||||
| | OLED SDA | 21 | I2C Daten | | ||||
| | OLED SCL | 22 | I2C Takt | | ||||
| | PN532 IRQ | 32 | Interrupt | | ||||
| | PN532 RESET | 33 | Reset | | ||||
| | PN532 SDA | 21 | I2C Daten (geteilt) | | ||||
| | PN532 SCL | 22 | I2C Takt (geteilt) | | ||||
| | TTP223 I/O | 25 | Touch-Sensor (optional) | | ||||
|  | ||||
| ### Wichtige Hinweise | ||||
|  | ||||
| - **PN532 DIP-Schalter** müssen auf I2C-Modus eingestellt sein | ||||
| - **3V Pin** vom ESP32 für Touch-Sensor verwenden | ||||
| - **Wägezellen-Verkabelung**: E+ (rot), E- (schwarz), A- (weiß), A+ (grün) | ||||
|  | ||||
|  | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Ersteinrichtung | ||||
|  | ||||
| ### Nach der Installation | ||||
|  | ||||
| 1. **ESP32 Neustart** | ||||
|    - Das System erstellt automatisch einen WiFi-Hotspot "FilaMan" | ||||
|  | ||||
| 2. **WiFi-Konfiguration** | ||||
|    - Verbinden Sie sich mit dem "FilaMan" Netzwerk | ||||
|    - Öffnen Sie einen Browser (automatisches Portal oder http://192.168.4.1) | ||||
|    - Konfigurieren Sie Ihre WiFi-Zugangsdaten | ||||
|  | ||||
| 3. **Erster Zugriff** | ||||
|    - Nach erfolgreicher WiFi-Verbindung ist das System unter http://filaman.local erreichbar | ||||
|    - Alternativ über die vom Router zugewiesene IP-Adresse | ||||
|  | ||||
| ### Spoolman Vorbereitung | ||||
|  | ||||
| **Wichtiger Hinweis**: Spoolman muss im Debug-Modus laufen: | ||||
|  | ||||
| ```env | ||||
| # In der .env Datei von Spoolman auskommentieren: | ||||
| SPOOLMAN_DEBUG_MODE=TRUE | ||||
| ``` | ||||
|  | ||||
| Dies ist erforderlich, da Spoolman noch keine CORS-Domain-Konfiguration unterstützt. | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Konfiguration | ||||
|  | ||||
| ### Waagen-Kalibrierung | ||||
|  | ||||
| 1. **Kalibrierung starten** | ||||
|    - Gehen Sie zur "Scale" (Waage) Seite | ||||
|    - Bereiten Sie ein 500g Referenzgewicht vor (z.B. Wasserglas) | ||||
|  | ||||
| 2. **Kalibrierungsschritte** | ||||
|    - Folgen Sie den Anweisungen auf dem Display | ||||
|    - Legen Sie das Gewicht auf, wenn gefordert | ||||
|    - Warten Sie, bis die Kalibrierung abgeschlossen ist | ||||
|  | ||||
| 3. **Validierung** | ||||
|    - Testen Sie die Genauigkeit mit bekannten Gewichten | ||||
|    - Bei Bedarf "Tare Scale" für Nullstellung verwenden | ||||
|  | ||||
| ### Spoolman-Verbindung | ||||
|  | ||||
| 1. **Spoolman-URL eingeben** | ||||
|    - Gehen Sie zur "Spoolman/Bambu" Seite | ||||
|    - Geben Sie die vollständige URL Ihrer Spoolman-Instanz ein | ||||
|    - Format: `http://spoolman-server:7912` | ||||
|  | ||||
| 2. **Verbindung testen** | ||||
|    - Das System prüft automatisch die Verbindung | ||||
|    - Erfolgreiche Verbindung wird durch grünen Status angezeigt | ||||
|  | ||||
| ### Bambu Lab Drucker (optional) | ||||
|  | ||||
| 1. **Drucker-Einstellungen** | ||||
|    - Öffnen Sie das Einstellungsmenü auf Ihrem Bambu-Drucker | ||||
|    - Notieren Sie sich die folgenden Daten: | ||||
|      - IP-Adresse des Druckers | ||||
|      - Access Code | ||||
|      - Serial Number | ||||
|  | ||||
| 2. **FilaMan Konfiguration** | ||||
|    - Geben Sie die Drucker-Daten in der "Spoolman/Bambu" Seite ein | ||||
|    - Aktivieren Sie "Auto Send to Bambu" für automatische AMS-Zuordnung | ||||
|  | ||||
| 3. **Auto-Send Timeout** | ||||
|    - Konfigurieren Sie die Wartezeit für automatische Spulen-Erkennung | ||||
|    - Empfohlener Wert: 10-30 Sekunden | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Benutzung | ||||
|  | ||||
| ### Grundlegende Bedienung | ||||
|  | ||||
| 1. **Filament wiegen** | ||||
|    - Platzieren Sie die Spule auf der Waage | ||||
|    - Das Gewicht wird automatisch auf dem Display und in der Weboberfläche angezeigt | ||||
|  | ||||
| 2. **NFC-Tag scannen** | ||||
|    - Halten Sie den Tag in die Nähe des PN532-Moduls | ||||
|    - Bei erkannten Tags wird die Spulen-Information angezeigt | ||||
|    - Das Gewicht wird automatisch in Spoolman aktualisiert | ||||
|  | ||||
| 3. **Status-Überwachung** | ||||
|    - **OLED-Display** zeigt aktuelles Gewicht und Verbindungsstatus | ||||
|    - **Weboberfläche** bietet detaillierte Informationen und Steuerung | ||||
|  | ||||
| ### Weboberfläche Navigation | ||||
|  | ||||
| - **Startseite**: Hauptfunktionen und aktueller Status | ||||
| - **Scale**: Waagen-Kalibrierung und -Einstellungen | ||||
| - **Spoolman/Bambu**: System-Konfiguration | ||||
| - **Statistics**: Nutzungsstatistiken (falls aktiviert) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## NFC-Tags | ||||
|  | ||||
| ### Unterstützte Tag-Typen | ||||
|  | ||||
| - **NTAG213**: 144 Bytes (grundlegende Funktionen) | ||||
| - **NTAG215**: 504 Bytes (empfohlen) | ||||
| - **NTAG216**: 888 Bytes (erweiterte Funktionen) | ||||
|  | ||||
| ### Tag beschreiben | ||||
|  | ||||
| 1. **Spule in Spoolman vorbereiten** | ||||
|    - Erstellen Sie eine neue Spule in Spoolman | ||||
|    - Stellen Sie sicher, dass alle erforderlichen Daten eingegeben sind | ||||
|  | ||||
| 2. **Tag-Beschreibung starten** | ||||
|    - Wählen Sie die Spule aus der Liste | ||||
|    - Klicken Sie auf "Write Tag" | ||||
|    - Das Display zeigt "Waiting for Tag" | ||||
|  | ||||
| 3. **Tag auflegen** | ||||
|    - Platzieren Sie den NFC-Tag auf dem PN532-Modul | ||||
|    - Warten Sie auf die Bestätigung | ||||
|  | ||||
| 4. **Erfolgsmeldung** | ||||
|    - Bei erfolgreichem Beschreiben wird ein Häkchen angezeigt | ||||
|    - Der Tag ist nun mit der Spoolman-Spule verknüpft | ||||
|  | ||||
| ### Tag lesen | ||||
|  | ||||
| 1. **Tag scannen** | ||||
|    - Platzieren Sie die Spule mit dem NFC-Tag auf die Waage über dem NFC-Reader | ||||
|    - Bei Problemen beim Lesen: Spule etwas anders positionieren (nicht ganz an den Rand) | ||||
|    - Die Spulen-Information wird automatisch geladen | ||||
|  | ||||
| 2. **Automatische Updates** | ||||
|    - Das aktuelle Gewicht wird in Spoolman übertragen | ||||
|    - Die Spule wird in der Weboberfläche automatisch ausgewählt | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Bambu Lab Integration | ||||
|  | ||||
| ### AMS (Automatic Material System) | ||||
|  | ||||
| 1. **AMS-Status anzeigen** | ||||
|    - Die Weboberfläche zeigt den aktuellen Zustand aller AMS-Fächer | ||||
|    - Beladene Fächer werden mit Filament-Informationen angezeigt | ||||
|  | ||||
| 2. **Filament manuell zuordnen** | ||||
|    - Wählen Sie eine Spule aus der Spoolman-Liste | ||||
|    - Klicken Sie auf das entsprechende AMS-Fach-Symbol | ||||
|    - Das Filament wird dem Fach zugeordnet | ||||
|  | ||||
| 3. **Automatische Zuordnung** | ||||
|    - Nach dem Wiegen mit aktiviertem "Auto Send to Bambu" | ||||
|    - Das System wartet auf neue Spulen im AMS | ||||
|    - Kalibrierte Filamente werden automatisch zugeordnet | ||||
|  | ||||
| ### Bambu Studio Integration | ||||
|  | ||||
| 1. **Filament-Profile synchronisieren** | ||||
|    - Kalibrieren Sie Filamente in Bambu Studio | ||||
|    - Verwenden Sie Device → AMS → Bleistift-Symbol → Auswählen | ||||
|  | ||||
| 2. **Setting-IDs speichern** | ||||
|    - FilaMan erkennt verfügbare Setting-IDs automatisch | ||||
|    - Klicken Sie auf "Settings in Spoolman speichern" | ||||
|    - Die Profile werden für zukünftige Drucke verwendet | ||||
|  | ||||
| ### Verbindung wiederherstellen | ||||
|  | ||||
| - Bei Verbindungsproblemen klicken Sie den roten Punkt in der Menüleiste | ||||
| - Das System stellt automatisch eine neue Verbindung her | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Spoolman Integration | ||||
|  | ||||
| ### Automatische Funktionen | ||||
|  | ||||
| 1. **Spulen-Synchronisation** | ||||
|    - Automatische Übertragung von Gewichtsänderungen | ||||
|    - Echtzeit-Updates der Spulen-Daten | ||||
|  | ||||
| 2. **Extra-Felder** | ||||
|    - FilaMan erstellt automatisch erforderliche benutzerdefinierte Felder | ||||
|    - NFC-Tag-UID wird als Referenz gespeichert | ||||
|  | ||||
| 3. **Filterung** | ||||
|    - "Nur Spulen ohne NFC-Tag anzeigen" für einfache Tag-Zuordnung | ||||
|    - Kategorisierung nach Herstellern und Materialtypen | ||||
|  | ||||
| ### Spoolman Octoprint Plugin | ||||
|  | ||||
| Für Octoprint-Benutzer ist eine automatische Spulen-Zuordnung verfügbar: | ||||
|  | ||||
| 1. **Plugin installieren** | ||||
|    ``` | ||||
|    https://github.com/ManuelW77/OctoPrint-Spoolman-Filaman/archive/refs/heads/master.zip | ||||
|    ``` | ||||
|  | ||||
| 2. **FilaMan konfigurieren** | ||||
|    - Aktivieren Sie "Send to Octo-Plugin" | ||||
|    - Geben Sie Octoprint-URL und API-Key ein | ||||
|  | ||||
| 3. **Automatische Zuordnung** | ||||
|    - Nach dem Wiegen wird die Spule automatisch in Octoprint aktiviert | ||||
|    - Unterstützt aktuell nur Tool0 (erste Düse) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Hersteller Tags | ||||
|  | ||||
| ### Überblick | ||||
|  | ||||
| Hersteller Tags ermöglichen es Filament-Produzenten, vorkonfigurierte NFC-Tags zu liefern, die automatisch alle notwendigen Einträge in Spoolman erstellen. | ||||
|  | ||||
| ### Erste Schritte mit Hersteller Tags | ||||
|  | ||||
| 1. **Tag scannen** | ||||
|    - Platzieren Sie die Spule mit dem Hersteller-Tag auf die Waage über dem NFC-Reader | ||||
|    - Bei Problemen beim Lesen: Spule etwas anders positionieren (nicht ganz an den Rand) | ||||
|    - Das System erkennt automatisch das Hersteller-Format | ||||
|  | ||||
| 2. **Automatische Erstellung** | ||||
|    - **Marke** wird in Spoolman angelegt (falls nicht vorhanden) | ||||
|    - **Filament-Typ** wird mit allen Spezifikationen erstellt | ||||
|    - **Spule** wird automatisch registriert | ||||
|  | ||||
| 3. **Zukünftige Scans** | ||||
|    - Nach der ersten Einrichtung nutzen Tags das Fast-Path-System | ||||
|    - Sofortige Gewichtsmessung ohne erneute Einrichtung | ||||
|  | ||||
| ### Unterstützte Hersteller | ||||
|  | ||||
| - **RecyclingFabrik**: Erster offizieller Partner | ||||
| - Weitere Hersteller folgen | ||||
|  | ||||
| ### Vorteile | ||||
|  | ||||
| - ✅ **Null manuelle Einrichtung** | ||||
| - ✅ **Perfekte Datengenauigkeit** | ||||
| - ✅ **Sofortige Integration** | ||||
| - ✅ **Zukunftssicher** | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Fehlerbehebung | ||||
|  | ||||
| ### Häufige Probleme | ||||
|  | ||||
| #### WiFi-Verbindung | ||||
|  | ||||
| **Problem**: Kann nicht mit FilaMan-Hotspot verbinden | ||||
| - Lösung: Stellen Sie sicher, dass der ESP32 gestartet ist | ||||
| - Alternative: Manuell zu http://192.168.4.1 navigieren | ||||
|  | ||||
| **Problem**: Weboberfläche nicht erreichbar | ||||
| - Lösung: Prüfen Sie die IP-Adresse im Router | ||||
| - Alternative: Verwenden Sie http://filaman.local | ||||
|  | ||||
| #### Waage | ||||
|  | ||||
| **Problem**: Ungenaue Gewichtsmessungen | ||||
| - Lösung: Kalibrierung wiederholen | ||||
| - Tipp: Verwenden Sie "Tare Scale" für Nullstellung | ||||
|  | ||||
| **Problem**: Wägezelle reagiert nicht | ||||
| - Lösung: Überprüfen Sie die Verkabelung (E+, E-, A+, A-) | ||||
| - Tipp: Testen Sie mit einem Multimeter | ||||
|  | ||||
| #### NFC-Tags | ||||
|  | ||||
| **Problem**: Tag wird nicht erkannt | ||||
| - Lösung: Überprüfen Sie die PN532 DIP-Schalter (I2C-Modus) | ||||
| - Tipp: Spule etwas anders auf der Waage positionieren (nicht ganz an den Rand) | ||||
|  | ||||
| **Problem**: Tag kann nicht beschrieben werden | ||||
| - Lösung: Verwenden Sie NTAG215 für bessere Kompatibilität | ||||
| - Tipp: Stellen Sie sicher, dass der Tag nicht schreibgeschützt ist | ||||
|  | ||||
| #### Spoolman | ||||
|  | ||||
| **Problem**: Verbindung zu Spoolman schlägt fehl | ||||
| - Lösung: Aktivieren Sie SPOOLMAN_DEBUG_MODE=TRUE | ||||
| - Tipp: Überprüfen Sie die URL-Formatierung | ||||
|  | ||||
| **Problem**: Spulen werden nicht angezeigt | ||||
| - Lösung: Stellen Sie sicher, dass Spoolman läuft | ||||
| - Tipp: Prüfen Sie die Netzwerk-Firewall-Einstellungen | ||||
|  | ||||
| #### Bambu Lab | ||||
|  | ||||
| **Problem**: Drucker verbindet nicht | ||||
| - Lösung: Überprüfen Sie Access Code und IP-Adresse | ||||
| - Tipp: Stellen Sie sicher, dass der Drucker im LAN-Modus ist | ||||
|  | ||||
| **Problem**: AMS-Status wird nicht angezeigt | ||||
| - Lösung: Prüfen Sie die MQTT-Verbindung | ||||
| - Hinweis: Bambu kann die API jederzeit schließen | ||||
|  | ||||
| ### Debug-Informationen | ||||
|  | ||||
| Falls Sie Probleme haben, können Sie diese Schritte zur Diagnose verwenden: | ||||
|  | ||||
| #### Serieller Monitor (für Entwickler) | ||||
| - Verbinden Sie den ESP32 über USB mit Ihrem Computer | ||||
| - Öffnen Sie einen seriellen Monitor (z.B. Arduino IDE) mit 115200 Baud | ||||
| - Sie sehen detaillierte Log-Nachrichten des Systems | ||||
|  | ||||
| #### Browser-Konsole | ||||
| - Öffnen Sie die Weboberfläche von FilaMan | ||||
| - Drücken Sie F12 um die Entwicklertools zu öffnen   | ||||
| - Schauen Sie in der Konsole nach Fehlermeldungen | ||||
|  | ||||
| #### Neustart bei anhaltenden Problemen | ||||
| 1. ESP32 vom Strom trennen | ||||
| 2. 10 Sekunden warten | ||||
| 3. Wieder anschließen | ||||
| 4. 30 Sekunden für vollständigen Start warten | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Support | ||||
|  | ||||
| ### Community | ||||
|  | ||||
| - **Discord Server**: [https://discord.gg/my7Gvaxj2v](https://discord.gg/my7Gvaxj2v) | ||||
| - **GitHub Issues**: [Filaman Repository](https://github.com/ManuelW77/Filaman/issues) | ||||
| - **YouTube Kanal**: [Deutsches Erklärvideo](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaOHU) | ||||
|  | ||||
| ### Dokumentation | ||||
|  | ||||
| - **Offizielle Website**: [www.filaman.app](https://www.filaman.app) | ||||
| - **GitHub Wiki**: [Detaillierte Dokumentation](https://github.com/ManuelW77/Filaman/wiki) | ||||
| - **Hardware-Referenz**: ESP32 Pinout-Diagramme in `/img/` | ||||
|  | ||||
| ### Entwicklung unterstützen | ||||
|  | ||||
| Wenn Sie das Projekt unterstützen möchten: | ||||
|  | ||||
| [](https://www.buymeacoffee.com/manuelw) | ||||
|  | ||||
| ### Lizenz | ||||
|  | ||||
| Dieses Projekt ist unter der MIT-Lizenz veröffentlicht. Siehe [LICENSE](LICENSE.txt) für Details. | ||||
|  | ||||
| --- | ||||
|  | ||||
| **Letzte Aktualisierung**: August 2025 | ||||
| **Version**: 2.0 | ||||
| **Maintainer**: Manuel W. | ||||
							
								
								
									
										746
									
								
								WIKI_EN.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,746 @@ | ||||
| # FilaMan Wiki - English | ||||
|  | ||||
| ## Table of Contents | ||||
|  | ||||
| 1. [Overview](#overview) | ||||
| 2. [Installation](#installation) | ||||
| 3. [Hardware Requirements](#hardware-requirements) | ||||
| 4. [Initial Setup](#initial-setup) | ||||
| 5. [Configuration](#configuration) | ||||
| 6. [Usage](#usage) | ||||
| 7. [NFC Tags](#nfc-tags) | ||||
| 8. [Bambu Lab Integration](#bambu-lab-integration) | ||||
| 9. [Spoolman Integration](#spoolman-integration) | ||||
| 10. [Octoprint Integration](#octoprint-integration) | ||||
| 11. [Manufacturer Tags](#manufacturer-tags) | ||||
| 12. [Troubleshooting](#troubleshooting) | ||||
| 13. [Support](#support) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Overview | ||||
|  | ||||
| FilaMan is a comprehensive filament management system for 3D printers based on ESP32 hardware. It provides weight measurement, NFC tag management, and seamless integration with Spoolman and Bambu Lab 3D printers. | ||||
|  | ||||
| ### Key Features | ||||
|  | ||||
| - **Precise weight measurement** with HX711 load cell amplifier | ||||
| - **NFC tag reading and writing** for filament identification | ||||
| - **OLED display** for status information | ||||
| - **WiFi connectivity** with easy configuration | ||||
| - **Web-based user interface** with real-time updates | ||||
| - **Spoolman integration** for inventory management | ||||
| - **Bambu Lab AMS control** via MQTT | ||||
| - **OpenSpool NFC format** compatibility | ||||
| - **Manufacturer tag support** for automatic setup | ||||
|  | ||||
| ### System Requirements | ||||
|  | ||||
| - **ESP32 Development Board** | ||||
| - **Spoolman Instance** (required for full functionality) | ||||
| - **WiFi Network** | ||||
| - **Web Browser** (Chrome/Firefox/Safari) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| ### Easy Installation (Recommended) | ||||
|  | ||||
| 1. **Open the [FilaMan Web Installer](https://www.filaman.app/installer.html)** | ||||
|    - Use a Chrome-based browser | ||||
|  | ||||
| 2. **Prepare ESP32** | ||||
|    - Connect ESP32 via USB to your computer | ||||
|    - Click "Connect" | ||||
|  | ||||
| 3. **Select Port** | ||||
|    - Choose the appropriate USB port | ||||
|    - Confirm selection | ||||
|  | ||||
| 4. **Start Installation** | ||||
|    - Click "Install FilaMan" | ||||
|    - Wait for installation to complete | ||||
|  | ||||
| ### Manual Compilation | ||||
|  | ||||
| For advanced users with PlatformIO: | ||||
|  | ||||
| ```bash | ||||
| git clone https://github.com/ManuelW77/Filaman.git | ||||
| cd FilaMan/esp32 | ||||
| pio lib install | ||||
| pio run --target upload | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Hardware Requirements | ||||
|  | ||||
| ### Required Components | ||||
|  | ||||
| | Component | Description | Amazon Link (Affiliate) | | ||||
| |-----------|-------------|-------------------------| | ||||
| | ESP32 Development Board | Any ESP32 variant | [Amazon](https://amzn.to/3FHea6D) | | ||||
| | HX711 + Load Cell | 5kg Load Cell Amplifier | [Amazon](https://amzn.to/4ja1KTe) | | ||||
| | OLED Display | 0.96" I2C 128x64 SSD1306 | [Amazon](https://amzn.to/445aaa9) | | ||||
| | PN532 NFC Module | V3 RFID Module | [Amazon](https://amzn.eu/d/gy9vaBX) | | ||||
| | NFC Tags | NTAG213/NTAG215 | [Amazon](https://amzn.to/3E071xO) | | ||||
| | TTP223 Touch Sensor | Optional for tare function | [Amazon](https://amzn.to/4hTChMK) | | ||||
|  | ||||
| ### Pin Configuration | ||||
|  | ||||
| | Component | ESP32 Pin | Function | | ||||
| |-----------|-----------|----------| | ||||
| | HX711 DOUT | 16 | Load cell data output | | ||||
| | HX711 SCK | 17 | Load cell clock | | ||||
| | OLED SDA | 21 | I2C data | | ||||
| | OLED SCL | 22 | I2C clock | | ||||
| | PN532 IRQ | 32 | Interrupt | | ||||
| | PN532 RESET | 33 | Reset | | ||||
| | PN532 SDA | 21 | I2C data (shared) | | ||||
| | PN532 SCL | 22 | I2C clock (shared) | | ||||
| | TTP223 I/O | 25 | Touch sensor (optional) | | ||||
|  | ||||
| ### Important Notes | ||||
|  | ||||
| - **PN532 DIP switches** must be set to I2C mode | ||||
| - **3V pin** from ESP32 for touch sensor | ||||
| - **Load cell wiring**: E+ (red), E- (black), A- (white), A+ (green) | ||||
|  | ||||
|  | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Initial Setup | ||||
|  | ||||
| ### After Installation | ||||
|  | ||||
| 1. **ESP32 Restart** | ||||
|    - System automatically creates a WiFi hotspot "FilaMan" | ||||
|  | ||||
| 2. **WiFi Configuration** | ||||
|    - Connect to the "FilaMan" network | ||||
|    - Open browser (automatic portal or <http://192.168.4.1>) | ||||
|    - Configure your WiFi credentials | ||||
|  | ||||
| 3. **First Access** | ||||
|    - After successful WiFi connection, access system at <http://filaman.local> | ||||
|    - Alternative: Use IP address assigned by router | ||||
|  | ||||
| ### Spoolman Preparation | ||||
|  | ||||
| **Important Note**: Spoolman must run in debug mode: | ||||
|  | ||||
| ```env | ||||
| # Uncomment in Spoolman's .env file: | ||||
| SPOOLMAN_DEBUG_MODE=TRUE | ||||
| ``` | ||||
|  | ||||
| This is required as Spoolman doesn't support CORS domain configuration yet. | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Configuration | ||||
|  | ||||
| ### Scale Calibration | ||||
|  | ||||
| 1. **Start Calibration** | ||||
|    - Go to "Scale" page | ||||
|    - Prepare a 500g reference weight (e.g., water glass) | ||||
|  | ||||
| 2. **Calibration Steps** | ||||
|    - Follow instructions on display | ||||
|    - Place weight when prompted | ||||
|    - Wait for calibration to complete | ||||
|  | ||||
| 3. **Validation** | ||||
|    - Test accuracy with known weights | ||||
|    - Use "Tare Scale" for zero adjustment if needed | ||||
|  | ||||
| ### Spoolman Connection | ||||
|  | ||||
| 1. **Enter Spoolman URL** | ||||
|    - Go to "Spoolman/Bambu" page | ||||
|    - Enter complete URL of your Spoolman instance | ||||
|    - Format: `http://spoolman-server:7912` | ||||
|  | ||||
| 2. **Test Connection** | ||||
|    - System automatically checks connection | ||||
|    - Successful connection shown by green status | ||||
|  | ||||
| ### Bambu Lab Printer (Optional) | ||||
|  | ||||
| 1. **Printer Settings** | ||||
|    - Open settings menu on your Bambu printer | ||||
|    - Note the following data: | ||||
|      - Printer IP address | ||||
|      - Access Code | ||||
|      - Serial Number | ||||
|  | ||||
| 2. **FilaMan Configuration** | ||||
|    - Enter printer data on "Spoolman/Bambu" page | ||||
|    - Enable "Auto Send to Bambu" for automatic AMS assignment | ||||
|  | ||||
| 3. **Auto-Send Timeout** | ||||
|    - Configure waiting time for automatic spool detection | ||||
|    - Recommended value: 10-30 seconds | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Usage | ||||
|  | ||||
| ### Basic Operation | ||||
|  | ||||
| 1. **Weigh Filament** | ||||
|    - Place spool on scale | ||||
|    - Weight automatically displayed on screen and web interface | ||||
|  | ||||
| 2. **Scan NFC Tag** | ||||
|    - Hold tag near PN532 module | ||||
|    - Recognized tags display spool information | ||||
|    - Weight automatically updated in Spoolman | ||||
|  | ||||
| 3. **Status Monitoring** | ||||
|    - **OLED Display** shows current weight and connection status | ||||
|    - **Web Interface** provides detailed information and control | ||||
|  | ||||
| ### Web Interface Navigation | ||||
|  | ||||
| - **Home**: Main functions and current status | ||||
| - **Scale**: Scale calibration and settings | ||||
| - **Spoolman/Bambu**: System configuration | ||||
| - **Statistics**: Usage statistics (if enabled) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## NFC Tags | ||||
|  | ||||
| ### Supported Tag Types | ||||
|  | ||||
| - **NTAG213**: 144 bytes (basic functions) | ||||
| - **NTAG215**: 504 bytes (recommended) | ||||
| - **NTAG216**: 888 bytes (extended functions) | ||||
|  | ||||
| ### Writing Tags | ||||
|  | ||||
| 1. **Prepare Spool in Spoolman** | ||||
|    - Create new spool in Spoolman | ||||
|    - Ensure all required data is entered | ||||
|  | ||||
| 2. **Start Tag Writing** | ||||
|    - Select spool from list | ||||
|    - Click "Write Tag" | ||||
|    - Display shows "Waiting for Tag" | ||||
|  | ||||
| 3. **Place Tag** | ||||
|    - Position NFC tag on PN532 module | ||||
|    - Wait for confirmation | ||||
|  | ||||
| 4. **Success Message** | ||||
|    - Successful writing shows checkmark | ||||
|    - Tag is now linked to Spoolman spool | ||||
|  | ||||
| ### Reading Tags | ||||
|  | ||||
| 1. **Scan Tag** | ||||
|    - Place the spool with NFC tag on the scale over the NFC reader | ||||
|    - If reading fails: Reposition spool slightly (not completely at the edge) | ||||
|    - Spool information automatically loaded | ||||
|  | ||||
| 2. **Automatic Updates** | ||||
|    - Current weight transferred to Spoolman | ||||
|    - Spool automatically selected in web interface | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Bambu Lab Integration | ||||
|  | ||||
| ### AMS (Automatic Material System) | ||||
|  | ||||
| 1. **Display AMS Status** | ||||
|    - Web interface shows current state of all AMS slots | ||||
|    - Loaded slots display filament information | ||||
|  | ||||
| 2. **Manual Filament Assignment** | ||||
|    - Select spool from Spoolman list | ||||
|    - Click corresponding AMS slot icon | ||||
|    - Filament assigned to slot | ||||
|  | ||||
| 3. **Automatic Assignment** | ||||
|    - After weighing with "Auto Send to Bambu" enabled | ||||
|    - System waits for new spools in AMS | ||||
|    - Calibrated filaments automatically assigned | ||||
|  | ||||
| ### Bambu Studio Integration | ||||
|  | ||||
| 1. **Sync Filament Profiles** | ||||
|    - Calibrate filaments in Bambu Studio | ||||
|    - Use Device → AMS → Pencil icon → Select | ||||
|  | ||||
| 2. **Save Setting IDs** | ||||
|    - FilaMan automatically detects available setting IDs | ||||
|    - Click "Save Settings to Spoolman" | ||||
|    - Profiles used for future prints | ||||
|  | ||||
| ### Restore Connection | ||||
|  | ||||
| - For connection issues, click red dot in menu bar | ||||
| - System automatically establishes new connection | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Spoolman Integration | ||||
|  | ||||
| ### Automatic Functions | ||||
|  | ||||
| 1. **Spool Synchronization** | ||||
|    - Automatic transfer of weight changes | ||||
|    - Real-time updates of spool data | ||||
|  | ||||
| 2. **Extra Fields** | ||||
|    - FilaMan automatically creates required custom fields | ||||
|    - NFC tag UID stored as reference | ||||
|  | ||||
| 3. **Filtering** | ||||
|    - "Show only spools without NFC tag" for easy tag assignment | ||||
|    - Categorization by manufacturers and material types | ||||
|  | ||||
| ### Spoolman Octoprint Plugin | ||||
|  | ||||
| For Octoprint users, automatic spool assignment is available: | ||||
|  | ||||
| 1. **Install Plugin** | ||||
|  | ||||
|    ```text | ||||
|    https://github.com/ManuelW77/OctoPrint-Spoolman-Filaman/archive/refs/heads/master.zip | ||||
|    ``` | ||||
|  | ||||
| 2. **Configure FilaMan** | ||||
|    - Enable "Send to Octo-Plugin" | ||||
|    - Enter Octoprint URL and API key | ||||
|  | ||||
| 3. **Automatic Assignment** | ||||
|    - After weighing, spool automatically activated in Octoprint | ||||
|    - Currently supports only Tool0 (first nozzle) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Manufacturer Tags | ||||
|  | ||||
| ### Overview | ||||
|  | ||||
| Manufacturer tags allow filament producers to provide pre-configured NFC tags that automatically create all necessary entries in Spoolman. | ||||
|  | ||||
| ### Getting Started with Manufacturer Tags | ||||
|  | ||||
| 1. **Scan Tag** | ||||
|    - Place spool with manufacturer tag on the scale over the NFC reader | ||||
|    - If reading fails: Reposition spool slightly (not completely at the edge) | ||||
|    - System automatically recognizes manufacturer format | ||||
|  | ||||
| 2. **Automatic Creation** | ||||
|    - **Brand** created in Spoolman (if not present) | ||||
|    - **Filament type** created with all specifications | ||||
|    - **Spool** automatically registered | ||||
|  | ||||
| 3. **Future Scans** | ||||
|    - After initial setup, tags use fast-path system | ||||
|    - Immediate weight measurement without re-setup | ||||
|  | ||||
| ### Supported Manufacturers | ||||
|  | ||||
| - **RecyclingFabrik**: First official partner | ||||
| - More manufacturers coming soon | ||||
|  | ||||
| ### Benefits | ||||
|  | ||||
| - ✅ **Zero manual setup** | ||||
| - ✅ **Perfect data accuracy** | ||||
| - ✅ **Instant integration** | ||||
| - ✅ **Future-proof** | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Troubleshooting | ||||
|  | ||||
| ### Common Issues | ||||
|  | ||||
| #### WiFi Connection | ||||
|  | ||||
| **Issue**: Cannot connect to FilaMan hotspot | ||||
|  | ||||
| - Solution: Ensure ESP32 is started | ||||
| - Alternative: Manually navigate to <http://192.168.4.1> | ||||
|  | ||||
| **Issue**: Web interface not accessible | ||||
|  | ||||
| - Solution: Check IP address in router | ||||
| - Alternative: Use <http://filaman.local> | ||||
|  | ||||
| #### Scale | ||||
|  | ||||
| **Issue**: Inaccurate weight measurements | ||||
|  | ||||
| - Solution: Repeat calibration | ||||
| - Tip: Use "Tare Scale" for zero adjustment | ||||
|  | ||||
| **Issue**: Load cell not responding | ||||
|  | ||||
| - Solution: Check wiring (E+, E-, A+, A-) | ||||
| - Tip: Test with multimeter | ||||
|  | ||||
| #### NFC Tags | ||||
|  | ||||
| **Issue**: Tag not recognized | ||||
|  | ||||
| - Solution: Check PN532 DIP switches (I2C mode) | ||||
| - Tip: Reposition spool slightly on scale (not completely at the edge) | ||||
|  | ||||
| **Issue**: Cannot write tag | ||||
|  | ||||
| - Solution: Use NTAG215 for better compatibility | ||||
| - Tip: Ensure tag is not write-protected | ||||
|  | ||||
| #### Spoolman | ||||
|  | ||||
| **Issue**: Connection to Spoolman fails | ||||
|  | ||||
| - Solution: Enable SPOOLMAN_DEBUG_MODE=TRUE | ||||
| - Tip: Check URL formatting | ||||
|  | ||||
| **Issue**: Spools not displayed | ||||
|  | ||||
| - Solution: Ensure Spoolman is running | ||||
| - Tip: Check network firewall settings | ||||
|  | ||||
| #### Bambu Lab | ||||
|  | ||||
| **Issue**: Printer won't connect | ||||
|  | ||||
| - Solution: Check access code and IP address | ||||
| - Tip: Ensure printer is in LAN mode | ||||
|  | ||||
| **Issue**: AMS status not displayed | ||||
|  | ||||
| - Solution: Check MQTT connection | ||||
| - Note: Bambu may close API at any time | ||||
|  | ||||
| ### Debug Information | ||||
|  | ||||
| If you have problems, you can use these steps for diagnosis: | ||||
|  | ||||
| #### Serial Monitor (for developers) | ||||
|  | ||||
| - Connect the ESP32 via USB to your computer | ||||
| - Open a serial monitor (e.g., Arduino IDE) with 115200 baud | ||||
| - You will see detailed log messages from the system | ||||
|  | ||||
| #### Browser Console | ||||
|  | ||||
| - Open the FilaMan web interface | ||||
| - Press F12 to open developer tools | ||||
| - Check the console for error messages | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Maintenance and Updates | ||||
|  | ||||
| ### Firmware Update | ||||
|  | ||||
| 1. **Via Web Interface**: Access `http://filaman.local/upgrade.html` | ||||
| 2. **Select firmware file** (.bin format) | ||||
| 3. **Upload** - System restarts automatically | ||||
| 4. **Configuration preserved** - Settings remain intact | ||||
|  | ||||
| ### System Reset | ||||
|  | ||||
| For persistent issues: | ||||
|  | ||||
| 1. Disconnect ESP32 from power | ||||
| 2. Wait 10 seconds | ||||
| 3. Reconnect | ||||
| 4. Wait 30 seconds for complete startup | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Support and Information | ||||
|  | ||||
| **Manufacturer**: Your Company Name | ||||
| **Maintainer**: Manuel W. | ||||
|  | ||||
| ### Scale Technology | ||||
|  | ||||
| #### Weight Stabilization | ||||
|  | ||||
| The system uses multiple filters for precise measurements: | ||||
|  | ||||
| ```cpp | ||||
| // Moving Average Filter with 8 values | ||||
| #define MOVING_AVERAGE_SIZE 8 | ||||
| // Low-Pass Filter for smoothing | ||||
| #define LOW_PASS_ALPHA 0.3f | ||||
| // Thresholds for updates | ||||
| #define DISPLAY_THRESHOLD 0.3f    // Display update | ||||
| #define API_THRESHOLD 1.5f        // API actions | ||||
| ``` | ||||
|  | ||||
| #### Calibration Algorithm | ||||
|  | ||||
| 1. **System Pause**: All tasks are temporarily paused | ||||
| 2. **Zero Setting**: Tare scale without weight | ||||
| 3. **Reference Measurement**: 500g weight for 10 measurements | ||||
| 4. **Calculation**: `newValue = rawValue / SCALE_LEVEL_WEIGHT` | ||||
| 5. **NVS Storage**: Permanent value with verification | ||||
| 6. **Filter Reset**: New baseline for stabilization | ||||
|  | ||||
| #### Auto-Tare Logic | ||||
|  | ||||
| ```cpp | ||||
| // Conditions for Auto-Tare | ||||
| if (autoTare && (weight > 2 && weight < 7) || weight < -2) { | ||||
|     scale_tare_counter++; | ||||
|     if (scale_tare_counter >= 5) { | ||||
|         // Automatic zero setting | ||||
|         scale.tare(); | ||||
|         resetWeightFilter(); | ||||
|     } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### NFC Technology | ||||
|  | ||||
| #### PN532 Communication | ||||
|  | ||||
| - **Interface**: I2C at 400kHz | ||||
| - **IRQ Pin**: Interrupt-based tag detection | ||||
| - **Reset Handling**: Automatic recovery from communication errors | ||||
| - **DIP Switches**: Must be set to I2C mode (00) | ||||
|  | ||||
| #### NDEF Implementation | ||||
|  | ||||
| ```json | ||||
| // FilaMan Spoolman Format (with sm_id) | ||||
| { | ||||
|   "sm_id": "123", | ||||
|   "color": "#FF5733", | ||||
|   "type": "PLA",  | ||||
|   "brand": "Example Brand" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| #### Manufacturer Tag Schema | ||||
|  | ||||
| Compact JSON format for storage efficiency: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "b": "RecyclingFabrik",           // brand | ||||
|   "an": "FX1_PLA-S175-1000-RED",  // article number | ||||
|   "t": "PLA",                      // type | ||||
|   "c": "FF0000",                   // color (hex without #) | ||||
|   "cn": "Red",                     // color name | ||||
|   "et": "210",                     // extruder temp | ||||
|   "bt": "60",                      // bed temp | ||||
|   "di": "1.75",                    // diameter | ||||
|   "de": "1.24",                    // density | ||||
|   "sw": "240",                      // spool weight | ||||
|   "u": "https://www.yoururl.com/search?q=" // URL used vor Brand Link and Filament Link | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Display System | ||||
|  | ||||
| #### OLED Architecture (SSD1306) | ||||
|  | ||||
| - **Resolution**: 128x64 pixels monochrome | ||||
| - **Areas**: | ||||
|   - Status bar: 0-16 pixels (version, icons) | ||||
|   - Main area: 17-64 pixels (weight, messages) | ||||
| - **Update Interval**: 1 second for status line | ||||
|  | ||||
| #### Icon System | ||||
|  | ||||
| Bitmap icons for various states: | ||||
|  | ||||
| ```cpp | ||||
| // Status Icons (16x16 pixels) | ||||
| - icon_success: Checkmark for successful operations | ||||
| - icon_failed: X for errors | ||||
| - icon_transfer: Arrow for data transmission | ||||
| - icon_loading: Loading circle for ongoing operations | ||||
|  | ||||
| // Connection Icons with strikethrough indicator | ||||
| - wifi_on/wifi_off: WLAN status | ||||
| - bambu_on: Bambu Lab connection | ||||
| - spoolman_on: Spoolman API status | ||||
| ``` | ||||
|  | ||||
| ### API Integration | ||||
|  | ||||
| #### Spoolman REST API | ||||
|  | ||||
| FilaMan interacts with the following endpoints: | ||||
|  | ||||
| ```http | ||||
| GET  /api/v1/spool/          # List spools | ||||
| POST /api/v1/spool/          # Create new spool | ||||
| PUT  /api/v1/spool/{id}/     # Update spool | ||||
|  | ||||
| GET  /api/v1/vendor/         # List vendors | ||||
| POST /api/v1/vendor/         # Create new vendor | ||||
|  | ||||
| GET  /api/v1/filament/       # List filaments | ||||
| POST /api/v1/filament/       # Create new filament | ||||
| ``` | ||||
|  | ||||
| #### Request Handling | ||||
|  | ||||
| ```cpp | ||||
| // Sequential API processing | ||||
| enum spoolmanApiStateType { | ||||
|     API_IDLE = 0, | ||||
|     API_PROCESSING = 1, | ||||
|     API_ERROR = 2 | ||||
| }; | ||||
| ``` | ||||
|  | ||||
| Prevents simultaneous API calls and deadlocks. | ||||
|  | ||||
| #### Weight Update Logic | ||||
|  | ||||
| ```cpp | ||||
| // Conditions for Spoolman update | ||||
| if (activeSpoolId != "" &&  | ||||
|     weigthCouterToApi > 3 &&    // 3+ stable measurements | ||||
|     weightSend == 0 &&          // Not yet sent | ||||
|     weight > 5 &&               // Minimum weight 5g | ||||
|     spoolmanApiState == API_IDLE) { | ||||
|     updateSpoolWeight(activeSpoolId, weight); | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Bambu Lab MQTT | ||||
|  | ||||
| #### Connection Parameters | ||||
|  | ||||
| ```cpp | ||||
| // SSL/TLS Configuration | ||||
| #define BAMBU_PORT 8883 | ||||
| #define BAMBU_USERNAME "bblp" | ||||
|  | ||||
| // Topic Structure | ||||
| String topic = "device/" + bambu_serial + "/report"; | ||||
| String request_topic = "device/" + bambu_serial + "/request"; | ||||
| ``` | ||||
|  | ||||
| #### AMS Data Structure | ||||
|  | ||||
| ```cpp | ||||
| struct AMSData { | ||||
|     String tray_id; | ||||
|     String tray_type; | ||||
|     String tray_color; | ||||
|     String tray_material; | ||||
|     String setting_id; | ||||
|     String tray_info_idx; | ||||
|     bool has_spool; | ||||
| }; | ||||
| ``` | ||||
|  | ||||
| #### Auto-Send Mechanism | ||||
|  | ||||
| ```cpp | ||||
| // After tag recognition | ||||
| if (bambuCredentials.autosend_enable) { | ||||
|     autoSetToBambuSpoolId = activeSpoolId.toInt(); | ||||
|     // Countdown starts automatically | ||||
|     // Waits for new spool in AMS | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### WebSocket Communication | ||||
|  | ||||
| #### Message Types | ||||
|  | ||||
| ```javascript | ||||
| // Client → Server | ||||
| { | ||||
|   "type": "writeNfcTag", | ||||
|   "tagType": "spool", | ||||
|   "payload": { /* JSON data */ } | ||||
| } | ||||
|  | ||||
| { | ||||
|   "type": "scale", | ||||
|   "payload": "tare|calibrate|setAutoTare", | ||||
|   "enabled": true | ||||
| } | ||||
|  | ||||
| // Server → Client | ||||
| { | ||||
|   "type": "heartbeat", | ||||
|   "freeHeap": 245, | ||||
|   "bambu_connected": true, | ||||
|   "spoolman_connected": true | ||||
| } | ||||
|  | ||||
| { | ||||
|   "type": "amsData", | ||||
|   "data": [ /* AMS array */ ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| #### Connection Management | ||||
|  | ||||
| - **Auto-Reconnect**: Client-side reconnection | ||||
| - **Heartbeat**: Every 30 seconds for connection monitoring | ||||
| - **Cleanup**: Automatic removal of dead connections | ||||
|  | ||||
| ### Watchdog and Error Handling | ||||
|  | ||||
| #### System Watchdog | ||||
|  | ||||
| ```cpp | ||||
| // WDT Configuration | ||||
| esp_task_wdt_init(10, true);  // 10s timeout, panic on overflow | ||||
| esp_task_wdt_add(NULL);       // Add current task | ||||
| ``` | ||||
|  | ||||
| #### Error Recovery | ||||
|  | ||||
| - **NFC Reset**: Automatic PN532 restart on communication errors | ||||
| - **MQTT Reconnect**: Bambu Lab connection automatically restored | ||||
| - **WiFi Monitoring**: Connection check every 60 seconds | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Support | ||||
|  | ||||
| ### Community | ||||
|  | ||||
| - **Discord Server**: [https://discord.gg/my7Gvaxj2v](https://discord.gg/my7Gvaxj2v) | ||||
| - **GitHub Issues**: [Filaman Repository](https://github.com/ManuelW77/Filaman/issues) | ||||
| - **YouTube Channel**: [German explanation video](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaOHU) | ||||
|  | ||||
| ### Documentation | ||||
|  | ||||
| - **Official Website**: [www.filaman.app](https://www.filaman.app) | ||||
| - **GitHub Wiki**: [Detailed documentation](https://github.com/ManuelW77/Filaman/wiki) | ||||
| - **Hardware Reference**: ESP32 pinout diagrams in `/img/` | ||||
|  | ||||
| ### Support Development | ||||
|  | ||||
| If you'd like to support the project: | ||||
|  | ||||
| [](https://www.buymeacoffee.com/manuelw) | ||||
|  | ||||
| ### License | ||||
|  | ||||
| This project is released under the MIT License. See [LICENSE](LICENSE.txt) for details. | ||||
|  | ||||
| --- | ||||
|  | ||||
| **Last Updated**: August 2025   | ||||
| **Version**: 2.0   | ||||
| **Maintainer**: Manuel W. | ||||
							
								
								
									
										15297
									
								
								_3D Print Files/FilaMan-Waage.step
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								_3D Print Files/Filaman-Waage.f3z
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											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} | ||||
| @@ -1,7 +1,31 @@ | ||||
| { | ||||
|     "GFU99": "Generic TPU", | ||||
|     "GFN99": "Generic PA", | ||||
|     "GFN98": "Generic PA-CF", | ||||
|     "GFU99": "TPU", | ||||
|     "GFN99": "PA", | ||||
|     "GFN98": "PA-CF", | ||||
|     "GFL99": "PLA", | ||||
|     "GFL96": "PLA Silk", | ||||
|     "GFL98": "PLA-CF", | ||||
|     "GFL95": "PLA High Speed", | ||||
|     "GFG99": "PETG", | ||||
|     "GFG98": "PETG-CF", | ||||
|     "GFG97": "PCTG", | ||||
|     "GFB99": "ABS", | ||||
|     "GFC99": "PC", | ||||
|     "GFB98": "ASA", | ||||
|     "GFS99": "PVA", | ||||
|     "GFS98": "HIPS", | ||||
|     "GFT98": "PPS-CF", | ||||
|     "GFT97": "PPS", | ||||
|     "GFN97": "PPA-CF", | ||||
|     "GFN96": "PPA-GF", | ||||
|     "GFP99": "PE", | ||||
|     "GFP98": "PE-CF", | ||||
|     "GFP97": "PP", | ||||
|     "GFP96": "PP-CF", | ||||
|     "GFP95": "PP-GF", | ||||
|     "GFR99": "EVA", | ||||
|     "GFR98": "PHA", | ||||
|     "GFS97": "BVOH", | ||||
|     "GFA01": "Bambu PLA Matte", | ||||
|     "GFA00": "Bambu PLA Basic", | ||||
|     "GFA09": "Bambu PLA Tough", | ||||
| @@ -13,15 +37,11 @@ | ||||
|     "GFL03": "eSUN PLA+", | ||||
|     "GFL01": "PolyTerra PLA", | ||||
|     "GFL00": "PolyLite PLA", | ||||
|     "GFL99": "Generic PLA", | ||||
|     "GFL96": "Generic PLA Silk", | ||||
|     "GFL98": "Generic PLA-CF", | ||||
|     "GFA50": "Bambu PLA-CF", | ||||
|     "GFS02": "Bambu Support For PLA", | ||||
|     "GFA11": "Bambu PLA Aero", | ||||
|     "GFL04": "Overture PLA", | ||||
|     "GFL05": "Overture Matte PLA", | ||||
|     "GFL95": "Generic PLA High Speed", | ||||
|     "GFA12": "Bambu PLA Glow", | ||||
|     "GFA13": "Bambu PLA Dynamic", | ||||
|     "GFA15": "Bambu PLA Galaxy", | ||||
| @@ -30,41 +50,21 @@ | ||||
|     "GFU00": "Bambu TPU 95A HF", | ||||
|     "GFG00": "Bambu PETG Basic", | ||||
|     "GFT01": "Bambu PET-CF", | ||||
|     "GFG99": "Generic PETG", | ||||
|     "GFG98": "Generic PETG-CF", | ||||
|     "GFG50": "Bambu PETG-CF", | ||||
|     "GFG60": "PolyLite PETG", | ||||
|     "GFG01": "Bambu PETG Translucent", | ||||
|     "GFG97": "Generic PCTG", | ||||
|     "GFB00": "Bambu ABS", | ||||
|     "GFB99": "Generic ABS", | ||||
|     "GFB60": "PolyLite ABS", | ||||
|     "GFB50": "Bambu ABS-GF", | ||||
|     "GFC00": "Bambu PC", | ||||
|     "GFC99": "Generic PC", | ||||
|     "GFB98": "Generic ASA", | ||||
|     "GFB01": "Bambu ASA", | ||||
|     "GFB61": "PolyLite ASA", | ||||
|     "GFB02": "Bambu ASA-Aero", | ||||
|     "GFS99": "Generic PVA", | ||||
|     "GFS04": "Bambu PVA", | ||||
|     "GFS01": "Bambu Support G", | ||||
|     "GFN03": "Bambu PA-CF", | ||||
|     "GFN04": "Bambu PAHT-CF", | ||||
|     "GFS03": "Bambu Support For PA/PET", | ||||
|     "GFN05": "Bambu PA6-CF", | ||||
|     "GFN08": "Bambu PA6-GF", | ||||
|     "GFS98": "Generic HIPS", | ||||
|     "GFT98": "Generic PPS-CF", | ||||
|     "GFT97": "Generic PPS", | ||||
|     "GFN97": "Generic PPA-CF", | ||||
|     "GFN96": "Generic PPA-GF", | ||||
|     "GFP99": "Generic PE", | ||||
|     "GFP98": "Generic PE-CF", | ||||
|     "GFP97": "Generic PP", | ||||
|     "GFP96": "Generic PP-CF", | ||||
|     "GFP95": "Generic PP-GF", | ||||
|     "GFR99": "Generic EVA", | ||||
|     "GFR98": "Generic PHA", | ||||
|     "GFS97": "Generic BVOH" | ||||
|     "GFN08": "Bambu PA6-GF" | ||||
| } | ||||
| @@ -6,39 +6,34 @@ | ||||
|     <title>FilaMan - Filament Management Tool</title> | ||||
|     <link rel="icon" type="image/png" href="/favicon.ico"> | ||||
|     <link rel="stylesheet" href="style.css"> | ||||
|     <style> | ||||
|         .status-container { | ||||
|             float: right; | ||||
|             display: flex; | ||||
|             gap: 10px; | ||||
|             align-items: center; | ||||
|             margin-right: 10px; | ||||
|         } | ||||
|         .status-dot { | ||||
|             width: 8px; | ||||
|             height: 8px; | ||||
|             border-radius: 50%; | ||||
|             display: inline-block; | ||||
|             margin-right: 4px; | ||||
|         } | ||||
|         .status-item { | ||||
|             display: flex; | ||||
|             align-items: center; | ||||
|             font-size: 0.8em; | ||||
|             color: #fff; | ||||
|         } | ||||
|         .online { background-color: #2ecc71; } | ||||
|         .offline { background-color: #e74c3c; } | ||||
|         .ram-status { color: #fff; font-size: 0.8em; } | ||||
|     </style> | ||||
|     <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"> | ||||
|         <img src="/logo.png" alt="FilaMan Logo" class="logo"> | ||||
|         <a href="/">Start</a> | ||||
|         <a href="/waage">Scale</a> | ||||
|         <a href="/spoolman">Spoolman/Bambu</a> | ||||
|         <a href="/about">About</a> | ||||
|         <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 | ||||
| @@ -49,3 +44,4 @@ | ||||
|             <div class="ram-status" id="ramStatus"></div> | ||||
|         </div> | ||||
|     </div> | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,53 @@ | ||||
| {{header}} | ||||
| <!-- 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> | ||||
|  | ||||
|     <div class="container"> | ||||
| <!-- head --> | ||||
|  | ||||
|     <div class="content"> | ||||
|         <h1>FilaMan</h1> | ||||
|         <p>Filament Management Tool</p> | ||||
|         <p>Your smart solution for <strong>Filament Management</strong> in 3D printing.</p> | ||||
| @@ -8,10 +55,11 @@ | ||||
|         <h2>About FilaMan</h2> | ||||
|         <p> | ||||
|             FilaMan is a tool designed to simplify filament spool management. It allows you to identify and weigh filament spools,  | ||||
|             automatically sync data with the self-hosted <a href="https://github.com/Donkie/Spoolman" target="_blank">Spoolman</a> platform,  | ||||
|             and ensure compatibility with <a href="https://github.com/spuder/OpenSpool" target="_blank">OpenSpool</a> for Bambu printers. | ||||
|             automatically sync data with the self-hosted <a href="https://github.com/Donkie/Spoolman" target="_blank">Spoolman</a> platform. | ||||
|         </p> | ||||
|  | ||||
|         <p>Get more information at <a href="https://www.filaman.app" target="_blank">https://www.filaman.app</a> and <a href="https://github.com/ManuelW77/Filaman" target="_blank">https://github.com/ManuelW77/Filaman</a>.</p> | ||||
|  | ||||
|         <div class="features"> | ||||
|             <div class="feature"> | ||||
|                 <h3>Spool Identification</h3> | ||||
| @@ -26,12 +74,6 @@ | ||||
|                 <p>Works with OpenSpool to recognize and activate spools on Bambu printers.</p> | ||||
|             </div> | ||||
|         </div> | ||||
|  | ||||
|         <h2>Future Plans</h2> | ||||
|         <p> | ||||
|             We are working on expanding compatibility to support smaller NFC tags like NTag213  | ||||
|             and developing custom software to enhance the OpenSpool experience. | ||||
|         </p> | ||||
|     </div> | ||||
| </body> | ||||
| </html> | ||||
|   | ||||
							
								
								
									
										31
									
								
								html/own_filaments.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,31 @@ | ||||
| { | ||||
|     "TPU": "GFU99", | ||||
|     "PA": "GFN99", | ||||
|     "PA-CF": "GFN98", | ||||
|     "PLA": "GFL99", | ||||
|     "PLA Silk": "GFL96", | ||||
|     "PLA-CF": "GFL98", | ||||
|     "PLA High Speed": "GFL95", | ||||
|     "PETG": "GFG99", | ||||
|     "PETG-CF": "GFG98", | ||||
|     "PCTG": "GFG97", | ||||
|     "ABS": "GFB99", | ||||
|     "ABS+HS": "GFB99", | ||||
|     "PC": "GFC99", | ||||
|     "PC/ABS": "GFC99", | ||||
|     "ASA": "GFB98", | ||||
|     "PVA": "GFS99", | ||||
|     "HIPS": "GFS98", | ||||
|     "PPS-CF": "GFT98", | ||||
|     "PPS": "GFT97", | ||||
|     "PPA-CF": "GFN97", | ||||
|     "PPA-GF": "GFN96", | ||||
|     "PE": "GFP99", | ||||
|     "PE-CF": "GFP98", | ||||
|     "PP": "GFP97", | ||||
|     "PP-CF": "GFP96", | ||||
|     "PP-GF": "GFP95", | ||||
|     "EVA": "GFR99", | ||||
|     "PHA": "GFR98", | ||||
|     "BVOH": "GFS97" | ||||
| } | ||||
| @@ -1,4 +1,52 @@ | ||||
| {{header}} | ||||
| <!-- 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> | ||||
| @@ -35,11 +83,11 @@ | ||||
|                             </li> | ||||
|                             <li> | ||||
|                                 <span class="stat-label">Weight:</span> | ||||
|                                 <span class="stat-value"><span id="totalWeight"></span> kg</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> m</span> | ||||
|                                 <span class="stat-value"><span id="totalLength"></span></span> | ||||
|                             </li> | ||||
|                         </ul> | ||||
|                     </div> | ||||
| @@ -64,7 +112,6 @@ | ||||
|         <div class="column"> | ||||
|             <div class="feature-box"> | ||||
|                 <h2>Spoolman Spools</h2> | ||||
|                 <h2>1. select Manufacturer</h2> | ||||
|                 <label for="vendorSelect">Manufacturer:</label> | ||||
|                 <div style="display: flex; justify-content: space-between; align-items: center;"> | ||||
|                     <select id="vendorSelect" class="styled-select"> | ||||
| @@ -78,7 +125,6 @@ | ||||
|             </div> | ||||
|  | ||||
|             <div id="filamentSection" class="feature-box hidden"> | ||||
|                 <h2>2. Select Spool</h2> | ||||
|                 <label>Spool / Filament:</label> | ||||
|                 <div class="custom-dropdown"> | ||||
|                     <div class="dropdown-button" onclick="toggleFilamentDropdown()"> | ||||
| @@ -93,17 +139,20 @@ | ||||
|                 <p id="nfcInfo" class="nfc-status"></p> | ||||
|                 <button id="writeNfcButton" class="btn btn-primary hidden" onclick="writeNfcTag()">Write 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> | ||||
|                 <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> | ||||
| </div> | ||||
|  | ||||
|   | ||||
							
								
								
									
										386
									
								
								html/rfid.js
									
									
									
									
									
								
							
							
						
						| @@ -7,6 +7,7 @@ let heartbeatTimer = null; | ||||
| let lastHeartbeatResponse = Date.now(); | ||||
| const HEARTBEAT_TIMEOUT = 20000; | ||||
| let reconnectTimer = null; | ||||
| let spoolDetected = false; | ||||
|  | ||||
| // WebSocket Funktionen | ||||
| function startHeartbeat() { | ||||
| @@ -112,14 +113,51 @@ function initWebSocket() { | ||||
|  | ||||
|                 if (bambuDot) { | ||||
|                     bambuDot.className = 'status-dot ' + (data.bambu_connected ? 'online' : 'offline'); | ||||
|                     // Add click handler only when offline | ||||
|                     if (!data.bambu_connected) { | ||||
|                         bambuDot.style.cursor = 'pointer'; | ||||
|                         bambuDot.onclick = function() { | ||||
|                             if (socket && socket.readyState === WebSocket.OPEN) { | ||||
|                                 socket.send(JSON.stringify({ | ||||
|                                     type: 'reconnect', | ||||
|                                     payload: 'bambu' | ||||
|                                 })); | ||||
|                             } | ||||
|                         }; | ||||
|                     } else { | ||||
|                         bambuDot.style.cursor = 'default'; | ||||
|                         bambuDot.onclick = null; | ||||
|                     } | ||||
|                 } | ||||
|                 if (spoolmanDot) { | ||||
|                     spoolmanDot.className = 'status-dot ' + (data.spoolman_connected ? 'online' : 'offline'); | ||||
|                     // Add click handler only when offline | ||||
|                     if (!data.spoolman_connected) { | ||||
|                         spoolmanDot.style.cursor = 'pointer'; | ||||
|                         spoolmanDot.onclick = function() { | ||||
|                             if (socket && socket.readyState === WebSocket.OPEN) { | ||||
|                                 socket.send(JSON.stringify({ | ||||
|                                     type: 'reconnect', | ||||
|                                     payload: 'spoolman' | ||||
|                                 })); | ||||
|                             } | ||||
|                         }; | ||||
|                     } else { | ||||
|                         spoolmanDot.style.cursor = 'default'; | ||||
|                         spoolmanDot.onclick = null; | ||||
|                     } | ||||
|                 } | ||||
|                 if (ramStatus) { | ||||
|                     ramStatus.textContent = `${data.freeHeap}k`; | ||||
|                 } | ||||
|             } | ||||
|             else if (data.type === 'setSpoolmanSettings') { | ||||
|                 if (data.payload == 'success') { | ||||
|                     showNotification(`Spoolman Settings set successfully`, true); | ||||
|                 } else { | ||||
|                     showNotification(`Error setting Spoolman Settings`, false); | ||||
|                 } | ||||
|             } | ||||
|         }; | ||||
|     } catch (error) { | ||||
|         isConnected = false; | ||||
| @@ -171,27 +209,13 @@ document.addEventListener('spoolmanError', function(event) { | ||||
|     showNotification(`Spoolman Error: ${event.detail.message}`, false); | ||||
| }); | ||||
|  | ||||
| document.addEventListener('filamentSelected', function(event) { | ||||
| document.addEventListener('filamentSelected', function (event) { | ||||
|     updateNfcInfo(); | ||||
|     // Zeige Spool-Buttons wenn ein Filament ausgewählt wurde | ||||
|     const selectedText = document.getElementById("selected-filament").textContent; | ||||
|     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() { | ||||
|     const selectedText = document.getElementById("selected-filament").textContent; | ||||
|     const nfcInfo = document.getElementById("nfcInfo"); | ||||
| @@ -209,16 +233,6 @@ function updateNfcInfo() { | ||||
|         `${spool.id} | ${spool.filament.name} (${spool.filament.material})` === selectedText | ||||
|     ); | ||||
|  | ||||
|     if (selectedSpool && selectedSpool.extra.nfc_id) { | ||||
|         nfcInfo.textContent = "NFC Tag assigned"; | ||||
|         nfcInfo.classList.add("nfc-success"); | ||||
|         nfcInfo.classList.remove("nfc-error"); | ||||
|     } else { | ||||
|         nfcInfo.textContent = "No NFC-Tag assigned"; | ||||
|         nfcInfo.classList.add("nfc-error"); | ||||
|         nfcInfo.classList.remove("nfc-success"); | ||||
|     } | ||||
|  | ||||
|     if (selectedSpool) { | ||||
|         writeButton.classList.remove("hidden"); | ||||
|     } else { | ||||
| @@ -236,7 +250,7 @@ function displayAmsData(amsData) { | ||||
|          | ||||
|         const trayHTML = ams.tray.map(tray => { | ||||
|             // Prüfe ob überhaupt Daten vorhanden sind | ||||
|             const relevantFields = ['tray_type', 'tray_sub_brands', 'tray_info_idx', 'setting_id']; | ||||
|             const relevantFields = ['tray_type', 'tray_sub_brands', 'tray_info_idx', 'setting_id', 'cali_idx']; | ||||
|             const hasAnyContent = relevantFields.some(field =>  | ||||
|                 tray[field] !== null &&  | ||||
|                 tray[field] !== undefined &&  | ||||
| @@ -244,24 +258,47 @@ function displayAmsData(amsData) { | ||||
|                 tray[field] !== 'null' | ||||
|             ); | ||||
|  | ||||
|             if (!hasAnyContent) { | ||||
|                 return ` | ||||
|                     <div class="tray"> | ||||
|                         <p><b>Tray ${tray.id}</b></p> | ||||
|                         <p>Empty</p> | ||||
|                     </div> | ||||
|                     <hr>`; | ||||
|             } | ||||
|             // Bestimme den Anzeigenamen für das Tray | ||||
|             const trayDisplayName = (ams.ams_id === 255) ? 'External' : `Tray ${tray.id}`; | ||||
|  | ||||
|             // Nur für nicht-leere Trays den Button-HTML erstellen | ||||
|             const buttonHtml = ` | ||||
|                 <button class="spool-button" onclick="handleSpoolIn(${ams.ams_id}, ${tray.id})"  | ||||
|                         style="position: absolute; top: 5px; left: 5px;  | ||||
|                         style="position: absolute; top: -30px; left: -15px;  | ||||
|                                background: none; border: none; padding: 0;  | ||||
|                                cursor: pointer; display: none;"> | ||||
|                     <img src="spool_in.png" alt="Spool In" style="width: 48px; height: 48px;"> | ||||
|                 </button>`; | ||||
|              | ||||
|                         // Nur für nicht-leere Trays den Button-HTML erstellen | ||||
|             const outButtonHtml = ` | ||||
|                 <button class="spool-button" onclick="handleSpoolOut()"  | ||||
|                         style="position: absolute; top: -35px; right: -15px;  | ||||
|                                background: none; border: none; padding: 0;  | ||||
|                                cursor: pointer; display: block;"> | ||||
|                     <img src="spool_in.png" alt="Spool In" style="width: 48px; height: 48px; transform: rotate(180deg) scaleX(-1);"> | ||||
|                 </button>`; | ||||
|  | ||||
|             const spoolmanButtonHtml = ` | ||||
|                 <button class="spool-button" onclick="handleSpoolmanSettings('${tray.tray_info_idx}', '${tray.setting_id}', '${tray.cali_idx}', '${tray.nozzle_temp_min}', '${tray.nozzle_temp_max}')"  | ||||
|                         style="position: absolute; bottom: 0px; right: 0px;  | ||||
|                                background: none; border: none; padding: 0;  | ||||
|                                cursor: pointer; display: none;"> | ||||
|                     <img src="set_spoolman.png" alt="Spool In" style="width: 38px; height: 38px;"> | ||||
|                 </button>`; | ||||
|  | ||||
|             if (!hasAnyContent) { | ||||
|                 return ` | ||||
|                     <div class="tray"> | ||||
|                         <p class="tray-head">${trayDisplayName}</p> | ||||
|                         <p> | ||||
|                             ${(ams.ams_id === 255 && tray.tray_type === '') ? buttonHtml : ''} | ||||
|                             Empty | ||||
|                         </p> | ||||
|                     </div> | ||||
|                     <hr>`; | ||||
|             } | ||||
|  | ||||
|             // Generiere den Type mit Color-Box zusammen | ||||
|             const typeWithColor = tray.tray_type ?  | ||||
|                 `<p>Typ: ${tray.tray_type} ${tray.tray_color ? `<span style=" | ||||
| @@ -277,8 +314,9 @@ function displayAmsData(amsData) { | ||||
|             // Array mit restlichen Tray-Eigenschaften | ||||
|             const trayProperties = [ | ||||
|                 { key: 'tray_sub_brands', label: 'Sub Brands' }, | ||||
|                 { key: 'tray_info_idx', label: 'Filament Index' }, | ||||
|                 { key: 'setting_id', label: 'Setting ID' } | ||||
|                 { key: 'tray_info_idx', label: 'Filament IDX' }, | ||||
|                 { key: 'setting_id', label: 'Setting ID' }, | ||||
|                 { key: 'cali_idx', label: 'Calibration IDX' } | ||||
|             ]; | ||||
|  | ||||
|             // Nur gültige Felder anzeigen | ||||
| @@ -289,7 +327,13 @@ function displayAmsData(amsData) { | ||||
|                     tray[prop.key] !== '' && | ||||
|                     tray[prop.key] !== 'null' | ||||
|                 ) | ||||
|                 .map(prop => `<p>${prop.label}: ${tray[prop.key]}</p>`) | ||||
|                 .map(prop => { | ||||
|                     // Spezielle Behandlung für setting_id | ||||
|                     if (prop.key === 'cali_idx' && tray[prop.key] === '-1') { | ||||
|                         return `<p>${prop.label}: not calibrated</p>`; | ||||
|                     } | ||||
|                     return `<p>${prop.label}: ${tray[prop.key]}</p>`; | ||||
|                 }) | ||||
|                 .join(''); | ||||
|  | ||||
|             // Temperaturen nur anzeigen, wenn beide nicht 0 sind | ||||
| @@ -297,20 +341,19 @@ function displayAmsData(amsData) { | ||||
|                 ? `<p>Nozzle Temp: ${tray.nozzle_temp_min}°C - ${tray.nozzle_temp_max}°C</p>` | ||||
|                 : ''; | ||||
|  | ||||
|             // Bestimme den Anzeigenamen für das Tray | ||||
|             const trayDisplayName = (ams.ams_id === 255) ? 'External' : `Tray ${tray.id}`; | ||||
|  | ||||
|             return ` | ||||
|                 <div class="tray" ${tray.tray_color ? `style="border-left: 4px solid #${tray.tray_color};"` : 'style="border-left: 4px solid #007bff;"'}> | ||||
|                     <div style="position: relative;"> | ||||
|                         ${buttonHtml} | ||||
|                         <p><b>${trayDisplayName}</b></p> | ||||
|                         <p class="tray-head">${trayDisplayName}</p> | ||||
|                         ${typeWithColor} | ||||
|                         ${trayDetails} | ||||
|                         ${tempHTML} | ||||
|                         ${(ams.ams_id === 255 && tray.tray_type !== '') ? outButtonHtml : ''} | ||||
|                         ${(tray.setting_id != "" && tray.setting_id != "null") ? spoolmanButtonHtml : ''} | ||||
|                     </div> | ||||
|                 </div> | ||||
|                 <hr>`; | ||||
|                      | ||||
|                 </div>`; | ||||
|         }).join(''); | ||||
|  | ||||
|         const amsInfo = ` | ||||
| @@ -333,6 +376,60 @@ function updateSpoolButtons(show) { | ||||
|     }); | ||||
| } | ||||
|  | ||||
| function handleSpoolmanSettings(tray_info_idx, setting_id, cali_idx, nozzle_temp_min, nozzle_temp_max) { | ||||
|     // Hole das ausgewählte Filament | ||||
|     const selectedText = document.getElementById("selected-filament").textContent; | ||||
|  | ||||
|     // Finde die ausgewählte Spule in den Daten | ||||
|     const selectedSpool = spoolsData.find(spool =>  | ||||
|         `${spool.id} | ${spool.filament.name} (${spool.filament.material})` === selectedText | ||||
|     ); | ||||
|  | ||||
|     const payload = { | ||||
|         type: 'setSpoolmanSettings', | ||||
|         payload: { | ||||
|             filament_id: selectedSpool.filament.id, | ||||
|             tray_info_idx: tray_info_idx, | ||||
|             setting_id: setting_id, | ||||
|             cali_idx: cali_idx, | ||||
|             temp_min: nozzle_temp_min, | ||||
|             temp_max: nozzle_temp_max | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     try { | ||||
|         socket.send(JSON.stringify(payload)); | ||||
|         showNotification(`Setting send to Spoolman`, true); | ||||
|     } catch (error) { | ||||
|         console.error("Error while sending settings to Spoolman:", error); | ||||
|         showNotification("Error while sending!", false); | ||||
|     } | ||||
| } | ||||
|  | ||||
| function handleSpoolOut() { | ||||
|     // Erstelle Payload | ||||
|     const payload = { | ||||
|         type: 'setBambuSpool', | ||||
|         payload: { | ||||
|             amsId: 255, | ||||
|             trayId: 254, | ||||
|             color: "FFFFFF", | ||||
|             nozzle_temp_min: 0, | ||||
|             nozzle_temp_max: 0, | ||||
|             type: "", | ||||
|             brand: "" | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     try { | ||||
|         socket.send(JSON.stringify(payload)); | ||||
|         showNotification(`External Spool removed. Pls wait`, true); | ||||
|     } catch (error) { | ||||
|         console.error("Fehler beim Senden der WebSocket Nachricht:", error); | ||||
|         showNotification("Error while sending!", false); | ||||
|     } | ||||
| } | ||||
|  | ||||
| // Neue Funktion zum Behandeln des Spool-In-Klicks | ||||
| function handleSpoolIn(amsId, trayId) { | ||||
|     // Prüfe WebSocket Verbindung zuerst | ||||
| @@ -379,19 +476,30 @@ function handleSpoolIn(amsId, trayId) { | ||||
|             nozzle_temp_min: parseInt(minTemp), | ||||
|             nozzle_temp_max: parseInt(maxTemp), | ||||
|             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() || '', | ||||
|             cali_idx: "-1"  // Default-Wert setzen | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     // Debug logging | ||||
|     console.log("Sende WebSocket Nachricht:", payload); | ||||
|     // Prüfe, ob der Key cali_idx vorhanden ist und setze ihn | ||||
|     if (selectedSpool.filament.extra.bambu_cali_id) { | ||||
|         payload.payload.cali_idx = selectedSpool.filament.extra.bambu_cali_id.replace(/['"]+/g, '').trim(); | ||||
|     } | ||||
|  | ||||
|     // Prüfe, ob der Key bambu_setting_id vorhanden ist | ||||
|     if (selectedSpool.filament.extra.bambu_setting_id) { | ||||
|         payload.payload.bambu_setting_id = selectedSpool.filament.extra.bambu_setting_id.replace(/['"]+/g, '').trim(); | ||||
|     } | ||||
|  | ||||
|     console.log("Spool-In Payload:", payload); | ||||
|  | ||||
|     try { | ||||
|         socket.send(JSON.stringify(payload)); | ||||
|         showNotification(`Spool set in AMS ${amsId} Tray ${trayId}. Pls wait`, true); | ||||
|     } catch (error) { | ||||
|         console.error("Fehler beim Senden der WebSocket Nachricht:", error); | ||||
|         showNotification("Fehler beim Senden der Daten", false); | ||||
|         showNotification("Error while sending", false); | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -401,12 +509,15 @@ function updateNfcStatusIndicator(data) { | ||||
|     if (data.found === 0) { | ||||
|         // Kein NFC Tag gefunden | ||||
|         indicator.className = 'status-circle'; | ||||
|         spoolDetected = false; | ||||
|     } else if (data.found === 1) { | ||||
|         // NFC Tag erfolgreich gelesen | ||||
|         indicator.className = 'status-circle success'; | ||||
|         spoolDetected = true; | ||||
|     } else { | ||||
|         // Fehler beim Lesen | ||||
|         indicator.className = 'status-circle error'; | ||||
|         spoolDetected = true; | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -441,15 +552,18 @@ function updateNfcData(data) { | ||||
|             } | ||||
|  | ||||
|         } else { | ||||
|             nfcDataDiv.innerHTML = '<div style="margin-top: 10px;"></div>'; | ||||
|             nfcDataDiv.innerHTML = '<div class="info-message-inner" style="margin-top: 10px;"></div>'; | ||||
|         } | ||||
|         nfcStatusContainer.appendChild(nfcDataDiv); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // HTML für die Datenanzeige erstellen | ||||
|     let html = ` | ||||
|         <div style="margin-top: 10px;"> | ||||
|     let html = ""; | ||||
|  | ||||
|     if(data.sm_id){ | ||||
|         html = ` | ||||
|         <div class="nfc-card-data" style="margin-top: 10px;"> | ||||
|             <p><strong>Brand:</strong> ${data.brand || 'N/A'}</p> | ||||
|             <p><strong>Type:</strong> ${data.type || 'N/A'} ${data.color_hex ? `<span style=" | ||||
|                 background-color: #${data.color_hex};  | ||||
| @@ -461,10 +575,27 @@ function updateNfcData(data) { | ||||
|                 border-radius: 3px; | ||||
|                 margin-left: 5px; | ||||
|             "></span>` : ''}</p> | ||||
|     `; | ||||
|         `; | ||||
|  | ||||
|         // Spoolman ID anzeigen | ||||
|         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> | ||||
|         `; | ||||
|      } | ||||
|  | ||||
|      | ||||
|     // Spoolman ID anzeigen | ||||
|     html += `<p><strong>Spoolman ID:</strong> ${data.sm_id || 'No Spoolman ID'}</p>`; | ||||
|  | ||||
|     // Nur wenn eine sm_id vorhanden ist, aktualisiere die Dropdowns | ||||
|     if (data.sm_id) { | ||||
| @@ -491,67 +622,112 @@ function updateNfcData(data) { | ||||
| } | ||||
|  | ||||
| function writeNfcTag() { | ||||
|     const selectedText = document.getElementById("selected-filament").textContent; | ||||
|     if (selectedText === "Please choose...") { | ||||
|         alert('Please select a Spool first.'); | ||||
|         return; | ||||
|     if(!spoolDetected || confirm("Are you sure you want to overwrite the Tag?") == true){ | ||||
|         const selectedText = document.getElementById("selected-filament").textContent; | ||||
|         if (selectedText === "Please choose...") { | ||||
|             alert('Please select a Spool first.'); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         const spoolsData = window.getSpoolData(); | ||||
|         const selectedSpool = spoolsData.find(spool =>  | ||||
|             `${spool.id} | ${spool.filament.name} (${spool.filament.material})` === selectedText | ||||
|         ); | ||||
|  | ||||
|         if (!selectedSpool) { | ||||
|             alert('Ausgewählte Spule konnte nicht gefunden werden.'); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Temperaturwerte korrekt extrahieren | ||||
|         let minTemp = "175"; | ||||
|         let maxTemp = "275"; | ||||
|          | ||||
|         if (Array.isArray(selectedSpool.filament.nozzle_temperature) &&  | ||||
|             selectedSpool.filament.nozzle_temperature.length >= 2) { | ||||
|             minTemp = String(selectedSpool.filament.nozzle_temperature[0]); | ||||
|             maxTemp = String(selectedSpool.filament.nozzle_temperature[1]); | ||||
|         } | ||||
|  | ||||
|         // Erstelle das NFC-Datenpaket mit korrekten Datentypen | ||||
|         const nfcData = { | ||||
|             color_hex: selectedSpool.filament.color_hex || "FFFFFF", | ||||
|             type: selectedSpool.filament.material, | ||||
|             min_temp: minTemp, | ||||
|             max_temp: maxTemp, | ||||
|             brand: selectedSpool.filament.vendor.name, | ||||
|             sm_id: String(selectedSpool.id) // Konvertiere zu String | ||||
|         }; | ||||
|  | ||||
|         if (socket?.readyState === WebSocket.OPEN) { | ||||
|             const writeButton = document.getElementById("writeNfcButton"); | ||||
|             writeButton.classList.add("writing"); | ||||
|             writeButton.textContent = "Writing"; | ||||
|             socket.send(JSON.stringify({ | ||||
|                 type: 'writeNfcTag', | ||||
|                 tagType: 'spool', | ||||
|                 payload: nfcData | ||||
|             })); | ||||
|         } else { | ||||
|             alert('Not connected to Server. Please check connection.'); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
|     const spoolsData = window.getSpoolData(); | ||||
|     const selectedSpool = spoolsData.find(spool =>  | ||||
|         `${spool.id} | ${spool.filament.name} (${spool.filament.material})` === selectedText | ||||
|     ); | ||||
| 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 (!selectedSpool) { | ||||
|         alert('Ausgewählte Spule konnte nicht gefunden werden.'); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // Temperaturwerte korrekt extrahieren | ||||
|     let minTemp = "175"; | ||||
|     let maxTemp = "275"; | ||||
|      | ||||
|     if (Array.isArray(selectedSpool.filament.nozzle_temperature) &&  | ||||
|         selectedSpool.filament.nozzle_temperature.length >= 2) { | ||||
|         minTemp = String(selectedSpool.filament.nozzle_temperature[0]); | ||||
|         maxTemp = String(selectedSpool.filament.nozzle_temperature[1]); | ||||
|     } | ||||
|  | ||||
|     // Erstelle das NFC-Datenpaket mit korrekten Datentypen | ||||
|     const nfcData = { | ||||
|         version: "2.0", | ||||
|         protocol: "openspool", | ||||
|         color_hex: selectedSpool.filament.color_hex || "FFFFFF", | ||||
|         type: selectedSpool.filament.material, | ||||
|         min_temp: minTemp, | ||||
|         max_temp: maxTemp, | ||||
|         brand: selectedSpool.filament.vendor.name, | ||||
|         sm_id: String(selectedSpool.id) // Konvertiere zu String | ||||
|     }; | ||||
|  | ||||
|     if (socket?.readyState === WebSocket.OPEN) { | ||||
|         const writeButton = document.getElementById("writeNfcButton"); | ||||
|         writeButton.classList.add("writing"); | ||||
|         writeButton.textContent = "Writing"; | ||||
|         socket.send(JSON.stringify({ | ||||
|             type: 'writeNfcTag', | ||||
|             payload: nfcData | ||||
|         })); | ||||
|     } else { | ||||
|         alert('Not connected to Server. Please check connection.'); | ||||
|         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) { | ||||
|     const writeButton = document.getElementById("writeNfcButton"); | ||||
|     writeButton.classList.remove("writing"); | ||||
|     writeButton.classList.add(success ? "success" : "error"); | ||||
|     writeButton.textContent = success ? "Write success" : "Write failed"; | ||||
|     const writeLocationButton = document.getElementById("writeLocationNfcButton"); | ||||
|     if(writeButton.classList.contains("writing")){ | ||||
|         writeButton.classList.remove("writing"); | ||||
|         writeButton.classList.add(success ? "success" : "error"); | ||||
|         writeButton.textContent = success ? "Write success" : "Write failed"; | ||||
|  | ||||
|         setTimeout(() => { | ||||
|             writeButton.classList.remove("success", "error"); | ||||
|             writeButton.textContent = "Write Tag"; | ||||
|         }, 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); | ||||
|     } | ||||
|  | ||||
|      | ||||
|     setTimeout(() => { | ||||
|         writeButton.classList.remove("success", "error"); | ||||
|         writeButton.textContent = "Write Tag"; | ||||
|     }, 5000); | ||||
| } | ||||
|  | ||||
| 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> | ||||
							
								
								
									
										
											BIN
										
									
								
								html/set_spoolman.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 9.2 KiB | 
| @@ -1,14 +1,94 @@ | ||||
| {{header}} | ||||
| <!-- 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 --> | ||||
|  | ||||
|     <script> | ||||
|         window.onload = function() { | ||||
|             if (spoolmanUrl && spoolmanUrl.trim() !== "") { | ||||
|                 document.getElementById('spoolmanUrl').value = spoolmanUrl; | ||||
|             } | ||||
|              | ||||
|             // Initialize OctoPrint fields visibility | ||||
|             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() { | ||||
|             const url = document.getElementById('spoolmanUrl').value; | ||||
|             fetch(`/api/checkSpoolman?url=${encodeURIComponent(url)}`) | ||||
|             const spoolmanOctoEnabled = document.getElementById('spoolmanOctoEnabled').checked; | ||||
|             const spoolmanOctoUrl = document.getElementById('spoolmanOctoUrl').value; | ||||
|             const spoolmanOctoToken = document.getElementById('spoolmanOctoToken').value; | ||||
|              | ||||
|             fetch(`/api/checkSpoolman?url=${encodeURIComponent(url)}&octoEnabled=${spoolmanOctoEnabled}&octoUrl=${spoolmanOctoUrl}&octoToken=${spoolmanOctoToken}`) | ||||
|                 .then(response => response.json()) | ||||
|                 .then(data => { | ||||
|                     if (data.healthy) { | ||||
| @@ -26,18 +106,19 @@ | ||||
|             const ip = document.getElementById('bambuIp').value; | ||||
|             const serial = document.getElementById('bambuSerial').value; | ||||
|             const code = document.getElementById('bambuCode').value; | ||||
|             const autoSend = document.getElementById('autoSend').checked; | ||||
|             const autoSendTime = document.getElementById('autoSendTime').value; | ||||
|  | ||||
|             fetch(`/api/bambu?bambu_ip=${encodeURIComponent(ip)}&bambu_serialnr=${encodeURIComponent(serial)}&bambu_accesscode=${encodeURIComponent(code)}`) | ||||
|             fetch(`/api/bambu?bambu_ip=${encodeURIComponent(ip)}&bambu_serialnr=${encodeURIComponent(serial)}&bambu_accesscode=${encodeURIComponent(code)}&autoSend=${autoSend}&autoSendTime=${autoSendTime}`) | ||||
|                 .then(response => response.json()) | ||||
|                 .then(data => { | ||||
|                     if (data.healthy) { | ||||
|                         document.getElementById('bambuStatusMessage').innerText = 'Bambu Credentials saved!'; | ||||
|                         // Erstelle und zeige den Reboot-Button | ||||
|                         const rebootBtn = document.createElement('button'); | ||||
|                         rebootBtn.innerText = 'Reboot now'; | ||||
|                         rebootBtn.className = 'reboot-button'; | ||||
|                         rebootBtn.onclick = () => window.location.href = '/reboot'; | ||||
|                         document.getElementById('bambuStatusMessage').appendChild(rebootBtn); | ||||
|                         // Reload with forced cache refresh after short delay | ||||
|                         setTimeout(() => { | ||||
|                             window.location.reload(true); | ||||
|                             window.location.href = '/'; | ||||
|                         }, 1500); | ||||
|                     } else { | ||||
|                         document.getElementById('bambuStatusMessage').innerText = 'Error while saving Bambu Credentials.'; | ||||
|                     } | ||||
| @@ -46,6 +127,15 @@ | ||||
|                     document.getElementById('bambuStatusMessage').innerText = 'Error while saving: ' + error.message; | ||||
|                 }); | ||||
|         } | ||||
|  | ||||
|         /** | ||||
|          * Controls visibility of OctoPrint configuration fields based on checkbox state | ||||
|          * Called on page load and when checkbox changes | ||||
|          */ | ||||
|         function toggleOctoFields() { | ||||
|             const octoEnabled = document.getElementById('spoolmanOctoEnabled').checked; | ||||
|             document.getElementById('octoFields').style.display = octoEnabled ? 'block' : 'none'; | ||||
|         } | ||||
|     </script> | ||||
|     <script> | ||||
|         var spoolmanUrl = "{{spoolmanUrl}}"; | ||||
| @@ -53,27 +143,59 @@ | ||||
|      | ||||
|     <div class="content"> | ||||
|         <h1>Spoolman API URL / Bambu Credentials</h1> | ||||
|         <label for="spoolmanUrl">Set URL/IP to your Spoolman-Instanz:</label> | ||||
|         <input type="text" id="spoolmanUrl" placeholder="http://ip-or-url-of-your-spoolman-instanz:port"> | ||||
|         <button onclick="checkSpoolmanInstance()">Save Spoolman URL</button> | ||||
|         <p id="statusMessage"></p> | ||||
|  | ||||
|         <h2>Bambu Lab Printer Credentials</h2> | ||||
|         <div class="bambu-settings"> | ||||
|             <div class="input-group"> | ||||
|                 <label for="bambuIp">Bambu Drucker IP-Adresse:</label> | ||||
|                 <input type="text" id="bambuIp" placeholder="192.168.1.xxx" value="{{bambuIp}}"> | ||||
|         <div class="card"> | ||||
|             <div class="card-body"> | ||||
|                 <h5 class="card-title">Set URL/IP to your Spoolman instance</h5> | ||||
|                 <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 the spool to the Spoolman Octoprint plugin:</h5> | ||||
|                 <p> | ||||
|                     <input type="checkbox" id="spoolmanOctoEnabled" {{spoolmanOctoEnabled}} onchange="toggleOctoFields()"> Send to Octo-Plugin | ||||
|                 </p> | ||||
|                 <div id="octoFields" style="display: none;"> | ||||
|                     <p> | ||||
|                         <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}}"> | ||||
|                     </p> | ||||
|                 </div> | ||||
|                  | ||||
|                 <button id="btnSaveSpoolmanUrl" onclick="checkSpoolmanInstance()">Save Spoolman URL</button> | ||||
|                 <p id="statusMessage"></p> | ||||
|             </div> | ||||
|             <div class="input-group"> | ||||
|                 <label for="bambuSerial">Drucker Seriennummer:</label> | ||||
|                 <input type="text" id="bambuSerial" placeholder="BBLXXXXXXXX" value="{{bambuSerial}}"> | ||||
|         </div> | ||||
|  | ||||
|         <div class="card"> | ||||
|             <div class="card-body"> | ||||
|                 <h5 class="card-title">Bambu Lab Printer Credentials</h5> | ||||
|                 <div class="bambu-settings"> | ||||
|                     <div class="input-group"> | ||||
|                         <label for="bambuIp">Bambu Printer IP Address:</label> | ||||
|                         <input type="text" id="bambuIp" placeholder="192.168.1.xxx" value="{{bambuIp}}"> | ||||
|                     </div> | ||||
|                     <div class="input-group"> | ||||
|                         <label for="bambuSerial">Printer Serial Number:</label> | ||||
|                         <input type="text" id="bambuSerial" placeholder="BBLXXXXXXXX" value="{{bambuSerial}}"> | ||||
|                     </div> | ||||
|                     <div class="input-group"> | ||||
|                         <label for="bambuCode">Access Code:</label> | ||||
|                         <input type="text" id="bambuCode" placeholder="Access Code of the printer" value="{{bambuCode}}"> | ||||
|                     </div> | ||||
|                     <hr> | ||||
|                     <p>If activated, FilaMan will automatically update the next filled tray with the last scanned and weighed spool.</p> | ||||
|                     <div class="input-group" style="display: flex; margin-bottom: 0;"> | ||||
|                         <label for="autoSend" style="width: 250px; margin-right: 5px;">Auto Send to Bambu:</label> | ||||
|                         <label for="autoSendTime" style="width: 250px; margin-right: 5px;">Wait for Spool in Sec:</label> | ||||
|                     </div> | ||||
|                     <div class="input-group" style="display: flex;"> | ||||
|                         <input type="checkbox" id="autoSend" {{autoSendToBambu}} style="width: 190px; margin-right: 10px;"> | ||||
|                         <input type="number" min="60" id="autoSendTime" placeholder="Time to wait" value="{{autoSendTime}}" style="width: 100px;"> | ||||
|                     </div> | ||||
|  | ||||
|                     <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> | ||||
|                 </div> | ||||
|             </div> | ||||
|             <div class="input-group"> | ||||
|                 <label for="bambuCode">Access Code:</label> | ||||
|                 <input type="text" id="bambuCode" placeholder="Access Code vom Drucker" value="{{bambuCode}}"> | ||||
|             </div> | ||||
|             <button onclick="saveBambuCredentials()">Save Bambu Credentials</button> | ||||
|             <p id="bambuStatusMessage"></p> | ||||
|         </div> | ||||
|     </div> | ||||
| </body> | ||||
|   | ||||
							
								
								
									
										170
									
								
								html/spoolman.js
									
									
									
									
									
								
							
							
						
						| @@ -1,6 +1,7 @@ | ||||
| // Globale Variablen | ||||
| let spoolmanUrl = ''; | ||||
| let spoolsData = []; | ||||
| let locationData = []; | ||||
|  | ||||
| // Hilfsfunktionen für Datenmanipulation | ||||
| function processSpoolData(data) { | ||||
| @@ -61,9 +62,6 @@ function populateVendorDropdown(data, selectedSmId = null) { | ||||
|             totalLength += spool.remaining_length; | ||||
|         } | ||||
|  | ||||
|         console.log("Länge gesamt: " + spool.remaining_length); | ||||
|         console.log("Gewicht gesamt" + spool.remaining_weight); | ||||
|  | ||||
|         const vendor = spool.filament.vendor; | ||||
|          | ||||
|         const hasValidNfcId = spool.extra &&  | ||||
| @@ -88,13 +86,27 @@ function populateVendorDropdown(data, selectedSmId = null) { | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     // Dropdown mit gefilterten Herstellern befüllen | ||||
|     Object.entries(filteredVendors).forEach(([id, name]) => { | ||||
|         const option = document.createElement("option"); | ||||
|         option.value = id; | ||||
|         option.textContent = name; | ||||
|         vendorSelect.appendChild(option); | ||||
|     }); | ||||
|     // Nach der Schleife: Formatierung der Gesamtlänge | ||||
|     const lengthInM = totalLength / 1000;  // erst in m umrechnen | ||||
|     const formattedLength = lengthInM > 1000  | ||||
|         ? (lengthInM / 1000).toFixed(2) + " km"  | ||||
|         : lengthInM.toFixed(2) + " m"; | ||||
|  | ||||
|     // Formatierung des Gesamtgewichts (von g zu kg zu t) | ||||
|     const weightInKg = totalWeight / 1000;  // erst in kg umrechnen | ||||
|     const formattedWeight = weightInKg > 1000  | ||||
|         ? (weightInKg / 1000).toFixed(2) + " t"  | ||||
|         : weightInKg.toFixed(2) + " kg"; | ||||
|  | ||||
|     // Dropdown mit gefilterten Herstellern befüllen - alphabetisch sortiert | ||||
|     Object.entries(filteredVendors) | ||||
|         .sort(([, nameA], [, nameB]) => nameA.localeCompare(nameB)) // Sort vendors alphabetically by name | ||||
|         .forEach(([id, name]) => { | ||||
|             const option = document.createElement("option"); | ||||
|             option.value = id; | ||||
|             option.textContent = name; | ||||
|             vendorSelect.appendChild(option); | ||||
|         }); | ||||
|  | ||||
|     document.getElementById("totalSpools").textContent = totalSpools; | ||||
|     document.getElementById("spoolsWithoutTag").textContent = spoolsWithoutTag; | ||||
| @@ -102,8 +114,8 @@ function populateVendorDropdown(data, selectedSmId = null) { | ||||
|     document.getElementById("totalVendors").textContent = Object.keys(allVendors).length; | ||||
|      | ||||
|     // Neue Statistiken hinzufügen | ||||
|     document.getElementById("totalWeight").textContent = (totalWeight / 1000).toFixed(2); | ||||
|     document.getElementById("totalLength").textContent = (totalLength / 1000).toFixed(2); | ||||
|     document.getElementById("totalWeight").textContent = formattedWeight; | ||||
|     document.getElementById("totalLength").textContent = formattedLength; | ||||
|  | ||||
|     // Material-Statistiken zum DOM hinzufügen | ||||
|     const materialsList = document.getElementById("materialsList"); | ||||
| @@ -112,7 +124,7 @@ function populateVendorDropdown(data, selectedSmId = null) { | ||||
|         .sort(([,a], [,b]) => b - a) // Sortiere nach Anzahl absteigend | ||||
|         .forEach(([material, count]) => { | ||||
|             const li = document.createElement("li"); | ||||
|             li.textContent = `${material}: ${count} ${count === 1 ? 'Spule' : 'Spulen'}`; | ||||
|             li.textContent = `${material}: ${count} ${count === 1 ? 'Spool' : 'Spools'}`; | ||||
|             materialsList.appendChild(li); | ||||
|         }); | ||||
|  | ||||
| @@ -122,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) { | ||||
|     const vendorId = document.getElementById("vendorSelect").value; | ||||
|     const dropdownContentInner = document.getElementById("filament-dropdown-content"); | ||||
| @@ -136,6 +168,13 @@ function updateFilamentDropdown(selectedSmId = null) { | ||||
|  | ||||
|     if (vendorId) { | ||||
|         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 &&  | ||||
|                                  spool.extra.nfc_id &&  | ||||
|                                  spool.extra.nfc_id !== '""' &&  | ||||
| @@ -151,9 +190,32 @@ function updateFilamentDropdown(selectedSmId = null) { | ||||
|             option.setAttribute("data-value", spool.filament.id); | ||||
|             option.setAttribute("data-nfc-id", spool.extra.nfc_id || ""); | ||||
|              | ||||
|             const colorHex = spool.filament.color_hex || 'FFFFFF'; | ||||
|  | ||||
|             // 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'; | ||||
|                 colorHTML = `<div class="option-color" style="background-color: #${colorHex}"></div>`; | ||||
|             } | ||||
|              | ||||
|             option.innerHTML = ` | ||||
|                 <div class="option-color" style="background-color: #${colorHex}"></div> | ||||
|                 ${colorHTML} | ||||
|                 <span>${spool.id} | ${spool.filament.name} (${spool.filament.material})</span> | ||||
|             `; | ||||
|              | ||||
| @@ -167,12 +229,41 @@ function updateFilamentDropdown(selectedSmId = null) { | ||||
|     } | ||||
| } | ||||
|  | ||||
| function updateLocationSelect(){ | ||||
|     const writeLocationNfcButton = document.getElementById('writeLocationNfcButton'); | ||||
|     if(writeLocationNfcButton){ | ||||
|         writeLocationNfcButton.classList.remove("hidden"); | ||||
|     } | ||||
| } | ||||
|  | ||||
| function selectFilament(spool) { | ||||
|     const selectedColor = document.getElementById("selected-color"); | ||||
|     const selectedText = document.getElementById("selected-filament"); | ||||
|     const dropdownContent = document.getElementById("filament-dropdown-content"); | ||||
|      | ||||
|     selectedColor.style.backgroundColor = `#${spool.filament.color_hex || 'FFFFFF'}`; | ||||
|     // 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'}`; | ||||
|     } | ||||
|      | ||||
|     selectedText.textContent = `${spool.id} | ${spool.filament.name} (${spool.filament.material})`; | ||||
|     dropdownContent.classList.remove("show"); | ||||
|      | ||||
| @@ -202,6 +293,14 @@ async function initSpoolman() { | ||||
|         document.dispatchEvent(new CustomEvent('spoolDataLoaded', {  | ||||
|             detail: spoolsData  | ||||
|         })); | ||||
|          | ||||
|         locationData = await fetchLocationData(); | ||||
|          | ||||
|         document.dispatchEvent(new CustomEvent('locationDataLoaded', {  | ||||
|             detail: locationData  | ||||
|         })); | ||||
|  | ||||
|  | ||||
|     } catch (error) { | ||||
|         console.error('Fehler beim Initialisieren von Spoolman:', error); | ||||
|         document.dispatchEvent(new CustomEvent('spoolmanError', {  | ||||
| @@ -229,17 +328,24 @@ async function fetchSpoolData() { | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* | ||||
| // Exportiere Funktionen | ||||
| window.getSpoolData = () => spoolsData; | ||||
| window.reloadSpoolData = initSpoolman; | ||||
| window.populateVendorDropdown = populateVendorDropdown; | ||||
| window.updateFilamentDropdown = updateFilamentDropdown; | ||||
| window.toggleFilamentDropdown = () => { | ||||
|     const content = document.getElementById("filament-dropdown-content"); | ||||
|     content.classList.toggle("show"); | ||||
| }; | ||||
| */ | ||||
| async function fetchLocationData() { | ||||
|     try { | ||||
|         if (!spoolmanUrl) { | ||||
|             throw new Error('Spoolman URL ist nicht initialisiert'); | ||||
|         } | ||||
|          | ||||
|         const response = await fetch(`${spoolmanUrl}/api/v1/location`); | ||||
|         if (!response.ok) { | ||||
|             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 | ||||
| document.addEventListener('DOMContentLoaded', () => { | ||||
| @@ -250,6 +356,11 @@ document.addEventListener('DOMContentLoaded', () => { | ||||
|         vendorSelect.addEventListener('change', () => updateFilamentDropdown()); | ||||
|     } | ||||
|  | ||||
|     const locationSelect = document.getElementById('locationSelect'); | ||||
|     if (locationSelect) { | ||||
|         locationSelect.addEventListener('change', () => updateLocationSelect()); | ||||
|     } | ||||
|      | ||||
|     const onlyWithoutSmId = document.getElementById('onlyWithoutSmId'); | ||||
|     if (onlyWithoutSmId) { | ||||
|         onlyWithoutSmId.addEventListener('change', () => { | ||||
| @@ -262,6 +373,10 @@ document.addEventListener('DOMContentLoaded', () => { | ||||
|         populateVendorDropdown(event.detail); | ||||
|     }); | ||||
|  | ||||
|     document.addEventListener('locationDataLoaded', (event) => { | ||||
|         populateLocationDropdown(event.detail); | ||||
|     }); | ||||
|      | ||||
|     window.onclick = function(event) { | ||||
|         if (!event.target.closest('.custom-dropdown')) { | ||||
|             const dropdowns = document.getElementsByClassName("dropdown-content"); | ||||
| @@ -291,6 +406,7 @@ window.getSpoolData = () => spoolsData; | ||||
| window.setSpoolData = (data) => { spoolsData = data; }; | ||||
| window.reloadSpoolData = initSpoolman; | ||||
| window.populateVendorDropdown = populateVendorDropdown; | ||||
| window.populateLocationDropdown = populateLocationDropdown; | ||||
| window.updateFilamentDropdown = updateFilamentDropdown; | ||||
| window.toggleFilamentDropdown = () => { | ||||
|     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"} | ||||
							
								
								
									
										510
									
								
								html/style.css
									
									
									
									
									
								
							
							
						
						| @@ -1,53 +1,153 @@ | ||||
| /* Allgemeine Stile */ | ||||
| :root { | ||||
|     --primary-color: #63bb67;     /* Hauptfarbe Grün */ | ||||
|     --primary-light: #8cd590;     /* Helleres Grün */ | ||||
|     --background-green: #28902D;  /* Dunkleres Hintergrund-Grün */ | ||||
|     --text-color: #1e293b;       /* Dunkelgrau für Text */ | ||||
|     --inner-box-bg: #63bb67;     /* Grüner Hintergrund für innere Boxen */ | ||||
|     --inner-text-color: #ffffff; /* Weißer Text in Boxen */ | ||||
|     --stat-value-color: #ffd700; /* Gelber Text für Werte */ | ||||
|     --header-bg: #28902D; /* Hintergrundfarbe für den Header */ | ||||
|     --header-border: #f7208c; /* Pinke Randfarbe für den Header */ | ||||
|     --accent-color: #d51274; /* Pink für Akzentfarben */ | ||||
|     --header-text: #40e8b7; /* Türkisfarbener Text */ | ||||
|     --background-light: #63bb67;  /* Helleres Grün für Container */ | ||||
|     --border-color: rgba(255, 255, 255, 0.1); /* Subtile Borders */ | ||||
|     --card-background: rgba(255, 255, 255, 0.15); | ||||
|     --card-shadow: 0 4px 15px rgba(31, 41, 55, 0.1); | ||||
|     --glass-bg: rgba(255, 255, 255, 0.15); | ||||
|     --glass-border: 1px solid rgba(255, 255, 255, 0.2); | ||||
|     --glass-blur: blur(12px); | ||||
|     --glass-shadow: 0 8px 32px 0 rgba(0, 0, 0, 0.1); | ||||
| } | ||||
|  | ||||
| body {  | ||||
|     font-family: Arial, sans-serif;  | ||||
|     font-family: 'Inter', -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif; | ||||
|     margin: 0;  | ||||
|     padding: 0;  | ||||
|     background-color: #f8f9fa;  | ||||
|     color: #333; | ||||
|     background: var(--background-green); | ||||
|     background-image: radial-gradient(circle at center, var(--primary-color) 0%, var(--background-green) 100%); | ||||
|     color: var(--text-color); | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     align-items: center; | ||||
|     min-height: 100vh; | ||||
|     text-align: center; | ||||
| } | ||||
|  | ||||
| /* Header und Navigation */ | ||||
| .navbar {  | ||||
|     background: var(--header-bg); | ||||
|     border: 2px solid var(--header-border); | ||||
|     width: calc(100% - 4rem); | ||||
|     max-width: 1400px; | ||||
|     display: flex; | ||||
|     justify-content: space-between; | ||||
|     align-items: center; | ||||
|     padding: 0 1.5rem; | ||||
|     box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1); | ||||
|     position: fixed; | ||||
|     top: 1rem; | ||||
|     left: 50%; | ||||
|     transform: translateX(-50%); | ||||
|     z-index: 1000; | ||||
|     border-radius: 1rem; | ||||
|     height: 80px; | ||||
| } | ||||
|  | ||||
| .logo { | ||||
|     height: 40px;  /* Anpassen an die Navbar-Höhe */ | ||||
|     height: 70px; | ||||
|     width: auto; | ||||
|     margin-right: 15px; | ||||
|     margin-left: 10px; | ||||
| } | ||||
|  | ||||
| /* Navigationsleiste */ | ||||
| .navbar {  | ||||
|     background-color: #007bff;  | ||||
|     width: 100%; | ||||
|     display: flex; | ||||
|     justify-content: center; /* Zentriert die Navigation */ | ||||
|     padding: 10px 0; | ||||
|     box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1); | ||||
|     position: fixed; | ||||
|     top: 0; | ||||
|     left: 0; | ||||
|     z-index: 1000; | ||||
| } | ||||
|  | ||||
| .navbar a {  | ||||
|     display: inline-block;  | ||||
|     color: white;  | ||||
|     text-align: center;  | ||||
|     padding: 14px 20px;  | ||||
|     color: var(--inner-text-color); | ||||
|     text-decoration: none; | ||||
|     font-weight: bold; | ||||
|     transition: background 0.3s, color 0.3s; | ||||
|     cursor: pointer !important; /* Wichtig: cursor-Definition für Nav-Links */ | ||||
|     font-weight: 600; | ||||
|     padding: 0.5rem 1rem; | ||||
|     border-radius: 0.75rem; | ||||
|     background: rgba(255, 255, 255, 0.05); | ||||
|     border: 1px solid transparent; | ||||
|     transition: all 0.3s; | ||||
| } | ||||
|  | ||||
| .navbar a:hover {  | ||||
|     background-color: #0056b3;  | ||||
|     color: #fff; | ||||
|     cursor: pointer !important; | ||||
|     background-color: rgba(255, 255, 255, 0.1); | ||||
|     transform: translateY(-1px); | ||||
| } | ||||
|  | ||||
| /* Logo und Text im Header */ | ||||
| .logo-text { | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     align-items: flex-start; | ||||
|     justify-content: center; | ||||
| } | ||||
|  | ||||
| .logo-text h4 { | ||||
|     font-size: 1rem; | ||||
|     color: var(--header-text); | ||||
|     margin: 0 !important; | ||||
| } | ||||
|  | ||||
| .logo-text h1 { | ||||
|     font-size: 1.75rem; | ||||
|     font-weight: 600; | ||||
|     color: var(--header-border); | ||||
|     margin: 5px !important; | ||||
|     text-shadow: 1px 1px 0 var(--accent-color); | ||||
| } | ||||
|  | ||||
| /* Status Container in der Navbar */ | ||||
| .status-container { | ||||
|     background: var(--inner-box-bg) !important; | ||||
|     border-radius: 1rem; | ||||
|     padding: 0.5rem 1rem; | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|     gap: 1rem; | ||||
| } | ||||
|  | ||||
| .status-item { | ||||
|     background: rgba(0, 0, 0, 0.1) !important; | ||||
|     color: var(--inner-text-color) !important; | ||||
|     padding: 0.25rem 0.75rem; | ||||
|     border-radius: 0.5rem; | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|     gap: 0.5rem; | ||||
| } | ||||
|  | ||||
| .status-dot { | ||||
|     width: 8px; | ||||
|     height: 8px; | ||||
|     border-radius: 50%; | ||||
|     display: inline-block; | ||||
|     margin-right: 4px; | ||||
| } | ||||
|  | ||||
| .online {  | ||||
|     background-color: #63bb67; | ||||
| } | ||||
|  | ||||
| .offline {  | ||||
|     background-color: #dc2626; | ||||
| } | ||||
|  | ||||
| .status-dot.offline { | ||||
|     cursor: pointer; | ||||
| } | ||||
|  | ||||
| .status-dot.offline:hover { | ||||
|     opacity: 0.8; | ||||
|     transform: scale(1.1); | ||||
| } | ||||
|  | ||||
| .ram-status {  | ||||
|     color: var(--inner-text-color); | ||||
|     font-size: 0.9em; | ||||
|     padding: 0.25rem 0.75rem; | ||||
|     background: rgba(99, 187, 103, 0.1); | ||||
|     border-radius: 0.5rem; | ||||
| } | ||||
|  | ||||
| /* Inhalt */ | ||||
| @@ -68,11 +168,12 @@ h1 { | ||||
| } | ||||
|  | ||||
| h3 { | ||||
|     color: #007bff; | ||||
|     color: #007bff !important; | ||||
|     font-size: 24px; | ||||
|     margin-top: 5px; | ||||
|     margin-bottom: 5px; | ||||
|     font-weight: bold; | ||||
|     text-align: center; | ||||
| } | ||||
|  | ||||
| /* Formulare */ | ||||
| @@ -87,14 +188,18 @@ label { | ||||
|     font-weight: bold;  | ||||
| } | ||||
|  | ||||
| input[type="text"], input[type="submit"] {  | ||||
| input[type="text"], input[type="submit"], input[type="number"] {  | ||||
|     padding: 10px;  | ||||
|     border: 1px solid #ccc;  | ||||
|     border-radius: 5px;  | ||||
|     font-size: 16px;  | ||||
| } | ||||
|  | ||||
| input[type="text"]:focus {  | ||||
| input[type="number"] {  | ||||
|     width: 108px !important;  | ||||
| } | ||||
|  | ||||
| input[type="text"]:focus, input[type="number"]:focus {  | ||||
|     border-color: #007bff;  | ||||
|     outline: none;  | ||||
| } | ||||
| @@ -145,11 +250,11 @@ button:hover { | ||||
| } | ||||
| .feature { | ||||
|     flex: 1; | ||||
|     padding: 20px; | ||||
|     background-color: #f9f9f9; | ||||
|     padding: 0 20px; | ||||
|     border-radius: 8px; | ||||
|     margin: 0 10px; | ||||
|     margin: 0 10px 10px; | ||||
|     box-shadow: 0 4px 8px rgba(0, 0, 0, 0.05); | ||||
|     border: var(--glass-border); | ||||
| } | ||||
| .feature h3 { | ||||
|     font-size: 1.4rem; | ||||
| @@ -158,12 +263,12 @@ button:hover { | ||||
| } | ||||
| .feature p { | ||||
|     font-size: 1rem; | ||||
|     color: #555; | ||||
|     color: #ffffff; | ||||
| } | ||||
|  | ||||
| p { | ||||
|     font-size: 1rem; | ||||
|     color: #555; | ||||
|     color: #ffffff; | ||||
| } | ||||
|  | ||||
| a { | ||||
| @@ -178,9 +283,10 @@ a:hover { | ||||
|  | ||||
| /* Karten-Stil für optische Trennung */ | ||||
| .card { | ||||
|     background: #f9f9f9; | ||||
|     background: var(--primary-color); | ||||
|     width: 500px; | ||||
|     padding: 15px; | ||||
|     margin: 20px 0; | ||||
|     margin: 20px auto; | ||||
|     border-radius: 8px; | ||||
|     box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); | ||||
| } | ||||
| @@ -233,6 +339,13 @@ a:hover { | ||||
|     margin-top: 10px; | ||||
| } | ||||
|  | ||||
| .nfc-card-data { | ||||
|     padding-left: 20px !important; | ||||
|     color: white !important; | ||||
|     font-weight: bold !important; | ||||
|     margin: 0 auto !important; | ||||
| } | ||||
|  | ||||
| .nfc-success { | ||||
|     color: green; | ||||
| } | ||||
| @@ -247,7 +360,7 @@ a:hover { | ||||
|     display: flex; | ||||
|     justify-content: space-between; | ||||
|     gap: 20px; | ||||
|     margin-top: 20px; | ||||
|     margin-top: 5px; | ||||
|     width: 100%; | ||||
| } | ||||
|  | ||||
| @@ -257,17 +370,21 @@ a:hover { | ||||
| } | ||||
|  | ||||
| .feature-box { | ||||
|     background: white; | ||||
|     padding: 5px 20px 20px 20px; | ||||
|     border-radius: 8px; | ||||
|     box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); | ||||
|     margin-bottom: 20px; | ||||
|     background: var(--inner-box-bg); | ||||
|     border: 1px solid rgba(99, 187, 103, 0.2); | ||||
|     border-radius: 0.5rem; | ||||
|     padding: 1rem; | ||||
|     color: var(--inner-text-color); | ||||
|     margin: 10px 0 0 0; | ||||
| } | ||||
|  | ||||
| .feature-box h2 { | ||||
|     color: #007bff; | ||||
|     color: var(--inner-text-color); | ||||
|     font-size: 1.4rem; | ||||
|     margin-bottom: 15px; | ||||
|     margin-bottom: 1rem; | ||||
|     margin-top: 0; | ||||
|     color: var(--accent-color); | ||||
|     text-align: center; | ||||
| } | ||||
|  | ||||
| .feature-box ul { | ||||
| @@ -285,27 +402,24 @@ a:hover { | ||||
|     width: 95%; | ||||
|     max-width: 1400px; | ||||
|     margin: 0 auto; | ||||
|     padding-top: 60px; | ||||
|     padding-top: 100px; | ||||
|     padding-bottom: 20px;; | ||||
| } | ||||
|  | ||||
| .tray { | ||||
|     background: #ffffff; | ||||
|     border-radius: 8px; | ||||
|     padding: 15px; | ||||
|     margin: 10px 0; | ||||
|     color: var(--inner-text-color); | ||||
|     background: var(--background-green); | ||||
|     border: 1px solid var(--border-color); | ||||
|     border-radius: 8px; | ||||
|     box-shadow: 0 2px 4px rgba(0, 0, 0, 0.05); | ||||
|     border-left: 4px solid #ffffff; | ||||
| } | ||||
|  | ||||
| .tray p { | ||||
|     margin: 5px 0; | ||||
| } | ||||
|  | ||||
| .tray b { | ||||
|     color: #007bff; | ||||
| } | ||||
|  | ||||
| /* Responsive Design */ | ||||
| @media (max-width: 1024px) { | ||||
|     .three-column-layout { | ||||
| @@ -345,7 +459,7 @@ a:hover { | ||||
|  | ||||
| .nfc-data { | ||||
|     padding: 10px; | ||||
|     background-color: #f8f9fa; | ||||
|     background-color: var(--primary-color); | ||||
|     border-radius: 4px; | ||||
|     margin-top: 5px; | ||||
|     width: 100%; | ||||
| @@ -371,11 +485,15 @@ a:hover { | ||||
|  | ||||
| .info-message { | ||||
|     padding: 10px; | ||||
|     background-color: #fff3f3; | ||||
|     background-color: var(--header-bg); | ||||
|     border-radius: 4px; | ||||
|     border-left: 4px solid #39d82e; | ||||
| } | ||||
|  | ||||
| .info-message-inner { | ||||
|     background-color: var(--header-bg) !important; | ||||
| } | ||||
|  | ||||
| .nfc-header { | ||||
|     display: grid; | ||||
|     grid-template-columns: 40px 1fr 40px; | ||||
| @@ -387,6 +505,7 @@ a:hover { | ||||
|     margin: 0; | ||||
|     grid-column: 2; | ||||
|     text-align: center; | ||||
|     color: var(--accent-color); | ||||
| } | ||||
|  | ||||
| .nfc-header .status-circle { | ||||
| @@ -567,6 +686,7 @@ a:hover { | ||||
|     width: 100%; | ||||
|     font-family: inherit; | ||||
|     cursor: default; /* Container selbst soll normalen Cursor haben */ | ||||
|     color: black; | ||||
| } | ||||
|  | ||||
| .dropdown-button { | ||||
| @@ -639,23 +759,69 @@ a:hover { | ||||
|     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 { | ||||
|     position: fixed; | ||||
|     top: 20px; | ||||
|     right: 20px; | ||||
|     padding: 15px 25px; | ||||
|     border-radius: 4px; | ||||
|     color: white; | ||||
|     color: black; | ||||
|     z-index: 1000; | ||||
|     animation: slideIn 0.3s ease-out; | ||||
| } | ||||
|  | ||||
| .notification.success { | ||||
|     background-color: #28a745; | ||||
|     color: black !important; | ||||
| } | ||||
|  | ||||
| .notification.error { | ||||
|     background-color: #dc3545; | ||||
|     color: white !important; | ||||
| } | ||||
|  | ||||
| .notification.fade-out { | ||||
| @@ -677,23 +843,20 @@ a:hover { | ||||
| /* Neue Styles für die Statistiken */ | ||||
| .statistics-grid { | ||||
|     display: grid; | ||||
|     grid-template-columns: 1fr 1fr; | ||||
|     gap: 20px; | ||||
|     grid-template-columns: repeat(auto-fit, minmax(150px, 1fr)); | ||||
|     gap: 1rem; | ||||
|     margin-top: 15px; | ||||
| } | ||||
|  | ||||
| .statistics-column { | ||||
|     background: #f8f9fa; | ||||
|     padding: 0; | ||||
|     border-radius: 8px; | ||||
|     box-shadow: 0 2px 4px rgba(0,0,0,0.05); | ||||
|     background: var(--inner-box-bg); | ||||
|     border: 1px solid rgba(99, 187, 103, 0.2); | ||||
|     border-radius: 0.5rem; | ||||
| } | ||||
|  | ||||
| .statistics-column h3 { | ||||
|     color: #007bff; | ||||
|     margin-bottom: 5px; | ||||
|     padding-bottom: 8px; | ||||
|     border-bottom: 2px solid #e9ecef; | ||||
|     color: var(--inner-text-color); | ||||
|     border-bottom: 1px solid rgba(99, 187, 103, 0.2); | ||||
|     font-size: 1.1rem; | ||||
| } | ||||
|  | ||||
| @@ -706,8 +869,8 @@ a:hover { | ||||
| .statistics-list li { | ||||
|     display: flex; | ||||
|     justify-content: space-between; | ||||
|     padding: 8px 5px 0 5px; | ||||
|     border-bottom: 1px solid #e9ecef; | ||||
|     padding: 0.5rem; | ||||
|     border-bottom: 1px solid rgba(99, 187, 103, 0.1); | ||||
| } | ||||
|  | ||||
| .statistics-list li:last-child { | ||||
| @@ -715,13 +878,12 @@ a:hover { | ||||
| } | ||||
|  | ||||
| .stat-label { | ||||
|     color: #495057; | ||||
|     font-weight: 500; | ||||
|     color: var(--inner-text-color); | ||||
| } | ||||
|  | ||||
| .stat-value { | ||||
|     font-weight: bold; | ||||
|     color: #007bff; | ||||
|     color: var(--stat-value-color); | ||||
|     font-weight: 500; | ||||
| } | ||||
|  | ||||
| /* Responsive Design Anpassung */ | ||||
| @@ -772,17 +934,18 @@ a:hover { | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|     gap: 8px; | ||||
|     background: var(--inner-box-bg); | ||||
|     border: 1px solid rgba(99, 187, 103, 0.2); | ||||
|     border-radius: 0.5rem; | ||||
| } | ||||
|  | ||||
| .spool-stat .stat-label { | ||||
|     color: #495057; | ||||
|     font-weight: 500; | ||||
|     white-space: nowrap; | ||||
|     color: var(--inner-text-color); | ||||
| } | ||||
|  | ||||
| .spool-stat .stat-value { | ||||
|     font-weight: bold; | ||||
|     color: #007bff; | ||||
|     color: var(--stat-value-color); | ||||
|     font-weight: 500; | ||||
| } | ||||
|  | ||||
| /* Buttons und klickbare Elemente */ | ||||
| @@ -808,31 +971,35 @@ input[type="submit"]:disabled, | ||||
| } | ||||
|  | ||||
| /* Schreib-Button */ | ||||
| #writeNfcButton { | ||||
| #writeNfcButton, #writeLocationNfcButton { | ||||
|     background-color: #007bff; | ||||
|     color: white; | ||||
|     transition: background-color 0.3s, color 0.3s; | ||||
|     width: 160px; | ||||
| } | ||||
|  | ||||
| #writeNfcButton.writing { | ||||
| #writeNfcButton.writing, #writeLocationNfcButton.writing { | ||||
|     background-color: #ffc107; | ||||
|     color: black; | ||||
|     width: 160px; | ||||
| } | ||||
|  | ||||
| #writeNfcButton.success { | ||||
| #writeNfcButton.success, #writeLocationNfcButton.success { | ||||
|     background-color: #28a745; | ||||
|     color: white; | ||||
|     width: 160px; | ||||
| } | ||||
|  | ||||
| #writeNfcButton.error { | ||||
| #writeNfcButton.error, #writeLocationNfcButton.error { | ||||
|     background-color: #dc3545; | ||||
|     color: white; | ||||
|     width: 160px; | ||||
| } | ||||
|  | ||||
| #writeLocationNfcButton{ | ||||
|     width: 250px; | ||||
| } | ||||
|  | ||||
| @keyframes dots { | ||||
|     0% { content: ""; } | ||||
|     33% { content: "."; } | ||||
| @@ -840,28 +1007,13 @@ input[type="submit"]:disabled, | ||||
|     100% { content: "..."; } | ||||
| } | ||||
|  | ||||
| #writeNfcButton.writing::after { | ||||
| #writeNfcButton.writing::after, #writeLocationNfcButton.writing::after { | ||||
|     content: "..."; | ||||
|     animation: dots 1s steps(3, end) infinite; | ||||
| } | ||||
|  | ||||
| .reboot-button { | ||||
|     background-color: #ff0000; | ||||
|     color: white; | ||||
|     padding: 10px 20px; | ||||
|     border: none; | ||||
|     border-radius: 4px; | ||||
|     margin-left: 10px; | ||||
|     cursor: pointer; | ||||
| } | ||||
|  | ||||
| .reboot-button:hover { | ||||
|     background-color: #cc0000; | ||||
| } | ||||
|  | ||||
| /* Bambu Settings Erweiterung */ | ||||
| .bambu-settings { | ||||
|     background: white; | ||||
|     padding: 20px; | ||||
|     border-radius: 8px; | ||||
|     box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); | ||||
| @@ -880,7 +1032,7 @@ input[type="submit"]:disabled, | ||||
| } | ||||
|  | ||||
| .bambu-settings .input-group input { | ||||
|     width: 100%; | ||||
|     width: 95%; | ||||
| } | ||||
|  | ||||
| #bambuStatusMessage { | ||||
| @@ -891,11 +1043,165 @@ input[type="submit"]:disabled, | ||||
|     justify-content: center; | ||||
| } | ||||
|  | ||||
| .amsData { | ||||
|     border-color: black !important; | ||||
|     border-width: 1px !important; | ||||
| } | ||||
|  | ||||
| .tray { | ||||
|     position: relative; | ||||
| } | ||||
|  | ||||
| .tray-head { | ||||
|     color: var(--stat-value-color) !important; | ||||
|     text-align: center !important; | ||||
|     font-weight: bold !important; | ||||
| } | ||||
|  | ||||
| .spool-button:hover { | ||||
|     opacity: 0.8; | ||||
| } | ||||
|  | ||||
| .version { | ||||
|     font-size: 0.4em; | ||||
|     color: #000; | ||||
|     vertical-align: middle; | ||||
|     margin-left: 0.5rem; | ||||
|     text-shadow: none !important; | ||||
| } | ||||
|  | ||||
| .progress-container { | ||||
|     width: 100%; | ||||
|     margin: 20px 0; | ||||
|     display: none; | ||||
|     background: #f0f0f0; | ||||
|     border-radius: 4px; | ||||
|     overflow: hidden; | ||||
| } | ||||
| .progress-bar { | ||||
|     width: 0%; | ||||
|     height: 24px; | ||||
|     background-color: #4CAF50; | ||||
|     text-align: center; | ||||
|     line-height: 24px; | ||||
|     color: white; | ||||
|     transition: width 0.3s ease-in-out; | ||||
|     font-weight: bold; | ||||
| } | ||||
| .status { | ||||
|     margin: 10px 0; | ||||
|     padding: 15px; | ||||
|     border-radius: 4px; | ||||
|     display: none; | ||||
| } | ||||
| .error {  | ||||
|     background-color: #ffebee;  | ||||
|     color: #c62828;  | ||||
|     border: 1px solid #ef9a9a; | ||||
| } | ||||
| .success {  | ||||
|     background-color: #e8f5e9;  | ||||
|     color: #2e7d32;  | ||||
|     border: 1px solid #a5d6a7; | ||||
| } | ||||
| .update-form { | ||||
|     background: var(--primary-color); | ||||
|     box-shadow: 0 4px 8px rgba(0, 0, 0, 0.05); | ||||
|     border: var(--glass-border); | ||||
|     padding: 20px; | ||||
|     border-radius: 8px; | ||||
|     margin: 0 auto; | ||||
|     width: 400px; | ||||
|     text-align: center; | ||||
| } | ||||
| .update-form input[type="file"] { | ||||
|     margin-bottom: 15px; | ||||
|     width: 80%; | ||||
|     padding: 8px; | ||||
|     border: 1px solid #ddd; | ||||
|     border-radius: 4px; | ||||
|     background-color: #4CAF50; | ||||
| } | ||||
| .update-form input[type="submit"] { | ||||
|     background-color: #4CAF50; | ||||
|     color: white; | ||||
|     padding: 10px 20px; | ||||
|     border: none; | ||||
|     border-radius: 4px; | ||||
|     cursor: pointer; | ||||
|     font-size: 16px; | ||||
|     transition: background-color 0.3s; | ||||
| } | ||||
| .update-form input[type="submit"]:hover { | ||||
|     background-color: #45a049; | ||||
| } | ||||
| .update-form input[type="submit"]:disabled { | ||||
|     background-color: #cccccc; | ||||
|     cursor: not-allowed; | ||||
| } | ||||
| .warning { | ||||
|     background-color: var(--primary-color); | ||||
|     border: 1px solid #ffe0b2; | ||||
|     margin: 20px auto; | ||||
|     border-radius: 4px; | ||||
|     max-width: 600px; | ||||
|     text-align: center; | ||||
|     color: #e65100; | ||||
|     padding: 15px; | ||||
| } | ||||
|  | ||||
| .update-options { | ||||
|     display: flex; | ||||
|     gap: 2rem; | ||||
|     margin: 2rem 0; | ||||
| } | ||||
| .update-section { | ||||
|     flex: 1; | ||||
|     background: var(--background-green); | ||||
|     padding: 1.5rem; | ||||
|     border-radius: 8px; | ||||
| } | ||||
| .update-section h2 { | ||||
|     margin-top: 0; | ||||
|     color: #333; | ||||
| } | ||||
| .update-section p { | ||||
|     color: #666; | ||||
|     margin-bottom: 1rem; | ||||
| } | ||||
| .progress-container { | ||||
|     margin: 20px 0; | ||||
|     background: #f0f0f0; | ||||
|     border-radius: 4px; | ||||
|     overflow: hidden; | ||||
| } | ||||
| .progress-bar { | ||||
|     width: 0; | ||||
|     height: 20px; | ||||
|     background: #4CAF50; | ||||
|     transition: width 0.3s ease-in-out; | ||||
|     text-align: center; | ||||
|     line-height: 20px; | ||||
|     color: white; | ||||
| } | ||||
| .status { | ||||
|     margin-top: 20px; | ||||
|     padding: 10px; | ||||
|     border-radius: 4px; | ||||
|     display: none; | ||||
| } | ||||
| .status.success { | ||||
|     background: #e8f5e9; | ||||
|     color: #2e7d32; | ||||
| } | ||||
| .status.error { | ||||
|     background: #ffebee; | ||||
|     color: #c62828; | ||||
| } | ||||
| .warning { | ||||
|     background: #fff3e0; | ||||
|     color: #e65100; | ||||
|     padding: 15px; | ||||
|     border-radius: 4px; | ||||
|     margin-bottom: 20px; | ||||
| } | ||||
							
								
								
									
										255
									
								
								html/upgrade.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,255 @@ | ||||
| <!-- 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="content"> | ||||
|         <h1>Firmware Upgrade</h1> | ||||
|  | ||||
|         <div class="warning"> | ||||
|             <strong>Warning:</strong> Do not power off the device during update. | ||||
|         </div> | ||||
|  | ||||
|         <div class="update-options"> | ||||
|             <div class="update-section"> | ||||
|                 <h2>1) Firmware Update</h2> | ||||
|                 <p>Upload a new firmware file (upgrade_filaman_firmware_*.bin)</p> | ||||
|                 <div class="update-form"> | ||||
|                     <form id="firmwareForm" enctype='multipart/form-data' data-type="firmware"> | ||||
|                         <input type='file' name='update' accept='.bin' required> | ||||
|                         <input type='submit' value='Start Firmware Update'> | ||||
|                     </form> | ||||
|                 </div> | ||||
|             </div> | ||||
|  | ||||
|             <div class="update-section"> | ||||
|                 <h2>2) Webpage Update</h2> | ||||
|                 <p>Upload a new webpage file (upgrade_filaman_website_*.bin)</p> | ||||
|                 <div class="update-form"> | ||||
|                     <form id="webpageForm" enctype='multipart/form-data' data-type="webpage"> | ||||
|                         <input type='file' name='update' accept='.bin' required> | ||||
|                         <input type='submit' value='Start Webpage Update'> | ||||
|                     </form> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|  | ||||
|         <div class="progress-container" style="display: none;"> | ||||
|             <div class="progress-bar">0%</div> | ||||
|         </div> | ||||
|         <div class="status"></div> | ||||
|     </div> | ||||
|  | ||||
|     <script> | ||||
|         // Hide status indicators during update | ||||
|         const statusContainer = document.querySelector('.status-container'); | ||||
|         if (statusContainer) { | ||||
|             statusContainer.style.display = 'none'; | ||||
|         } | ||||
|  | ||||
|         const progress = document.querySelector('.progress-bar'); | ||||
|         const progressContainer = document.querySelector('.progress-container'); | ||||
|         const status = document.querySelector('.status'); | ||||
|         let updateInProgress = false; | ||||
|         let lastReceivedProgress = 0; | ||||
|  | ||||
|         // WebSocket Handling | ||||
|         let ws = null; | ||||
|         let wsReconnectTimer = null; | ||||
|  | ||||
|         function connectWebSocket() { | ||||
|             ws = new WebSocket('ws://' + window.location.host + '/ws'); | ||||
|              | ||||
|             ws.onmessage = function(event) { | ||||
|                 try { | ||||
|                     const data = JSON.parse(event.data); | ||||
|                     if (data.type === "updateProgress" && updateInProgress) { | ||||
|                         // Zeige Fortschrittsbalken | ||||
|                         progressContainer.style.display = 'block'; | ||||
|                          | ||||
|                         // Aktualisiere den Fortschritt nur wenn er größer ist | ||||
|                         const newProgress = parseInt(data.progress); | ||||
|                         if (!isNaN(newProgress) && newProgress >= lastReceivedProgress) { | ||||
|                             progress.style.width = newProgress + '%'; | ||||
|                             progress.textContent = newProgress + '%'; | ||||
|                             lastReceivedProgress = newProgress; | ||||
|                         } | ||||
|                          | ||||
|                         // Zeige Status-Nachricht | ||||
|                         if (data.message || data.status) { | ||||
|                             status.textContent = data.message || getStatusMessage(data.status); | ||||
|                             status.className = 'status success'; | ||||
|                             status.style.display = 'block'; | ||||
|                              | ||||
|                             // Starte Reload wenn Update erfolgreich | ||||
|                             if (data.status === 'success' || lastReceivedProgress >= 98) { | ||||
|                                 clearTimeout(wsReconnectTimer); | ||||
|                                 setTimeout(() => { | ||||
|                                     window.location.reload(true); | ||||
|                                     window.location.href = '/'; | ||||
|                                 }, 30000); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                 } catch (e) { | ||||
|                     console.error('WebSocket message error:', e); | ||||
|                 } | ||||
|             }; | ||||
|  | ||||
|             ws.onclose = function() { | ||||
|                 if (updateInProgress) { | ||||
|                     // Wenn der Fortschritt hoch genug ist, gehen wir von einem erfolgreichen Update aus | ||||
|                     if (lastReceivedProgress >= 85) { | ||||
|                         status.textContent = "Update appears successful! Device is restarting... Page will reload in 30 seconds."; | ||||
|                         status.className = 'status success'; | ||||
|                         status.style.display = 'block'; | ||||
|                         clearTimeout(wsReconnectTimer); | ||||
|                         setTimeout(() => { | ||||
|                             window.location.reload(true); | ||||
|                             window.location.href = '/'; | ||||
|                         }, 30000); | ||||
|                     } else { | ||||
|                         // Versuche Reconnect bei niedrigem Fortschritt | ||||
|                         wsReconnectTimer = setTimeout(connectWebSocket, 1000); | ||||
|                     } | ||||
|                 } | ||||
|             }; | ||||
|  | ||||
|             ws.onerror = function(err) { | ||||
|                 console.error('WebSocket error:', err); | ||||
|                 if (updateInProgress && lastReceivedProgress >= 85) { | ||||
|                     status.textContent = "Update appears successful! Device is restarting... Page will reload in 30 seconds."; | ||||
|                     status.className = 'status success'; | ||||
|                     status.style.display = 'block'; | ||||
|                     setTimeout(() => { | ||||
|                         window.location.href = '/'; | ||||
|                     }, 30000); | ||||
|                 } | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         // Initial WebSocket connection | ||||
|         connectWebSocket(); | ||||
|  | ||||
|         function getStatusMessage(status) { | ||||
|             switch(status) { | ||||
|                 case 'starting': return 'Starting update...'; | ||||
|                 case 'uploading': return 'Uploading...'; | ||||
|                 case 'finalizing': return 'Finalizing update...'; | ||||
|                 case 'restoring': return 'Restoring configurations...'; | ||||
|                 case 'preparing': return 'Preparing for restart...'; | ||||
|                 case 'success': return 'Update successful! Device is restarting... Page will reload in 30 seconds.'; | ||||
|                 default: return 'Updating...'; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         function handleUpdate(e) { | ||||
|             e.preventDefault(); | ||||
|             const form = e.target; | ||||
|             const file = form.update.files[0]; | ||||
|             const updateType = form.dataset.type; | ||||
|              | ||||
|             if (!file) { | ||||
|                 alert('Please select a file.'); | ||||
|                 return; | ||||
|             } | ||||
|              | ||||
|             // Validate file name pattern | ||||
|             if (updateType === 'firmware' && !file.name.startsWith('upgrade_filaman_firmware_')) { | ||||
|                 alert('Please select a valid firmware file (upgrade_filaman_firmware_*.bin)'); | ||||
|                 return; | ||||
|             } | ||||
|             if (updateType === 'webpage' && !file.name.startsWith('upgrade_filaman_website_')) { | ||||
|                 alert('Please select a valid webpage file (upgrade_filaman_website_*.bin)'); | ||||
|                 return; | ||||
|             } | ||||
|              | ||||
|             // Reset UI | ||||
|             updateInProgress = true; | ||||
|             progressContainer.style.display = 'block'; | ||||
|             status.style.display = 'none'; | ||||
|             status.className = 'status'; | ||||
|             progress.style.width = '0%'; | ||||
|             progress.textContent = '0%'; | ||||
|              | ||||
|             // Disable submit buttons | ||||
|             document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = true); | ||||
|  | ||||
|             // Send update | ||||
|             const xhr = new XMLHttpRequest(); | ||||
|             xhr.open('POST', '/update', true); | ||||
|              | ||||
|             xhr.onload = function() { | ||||
|                 if (xhr.status !== 200 && !progress.textContent.startsWith('100')) { | ||||
|                     status.textContent = "Update failed: " + (xhr.responseText || "Unknown error"); | ||||
|                     status.className = 'status error'; | ||||
|                     status.style.display = 'block'; | ||||
|                     updateInProgress = false; | ||||
|                     document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = false); | ||||
|                 } | ||||
|             }; | ||||
|              | ||||
|             xhr.onerror = function() { | ||||
|                 if (!progress.textContent.startsWith('100')) { | ||||
|                     status.textContent = "Network error during update"; | ||||
|                     status.className = 'status error'; | ||||
|                     status.style.display = 'block'; | ||||
|                     updateInProgress = false; | ||||
|                     document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = false); | ||||
|                 } | ||||
|             }; | ||||
|  | ||||
|             const formData = new FormData(); | ||||
|             formData.append('update', file); | ||||
|             xhr.send(formData); | ||||
|         } | ||||
|  | ||||
|         document.getElementById('firmwareForm').addEventListener('submit', handleUpdate); | ||||
|         document.getElementById('webpageForm').addEventListener('submit', handleUpdate); | ||||
|     </script> | ||||
| </body> | ||||
| </html> | ||||
| @@ -1,4 +1,52 @@ | ||||
| {{header}} | ||||
| <!-- 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="content"> | ||||
|         <h1>Scale Configuration Page</h1> | ||||
|          | ||||
| @@ -7,6 +55,7 @@ | ||||
|                 <h5 class="card-title">Sacle Calibration</h5> | ||||
|                 <button id="calibrateBtn" class="btn btn-primary">Calibrate 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> | ||||
|         </div> | ||||
| @@ -92,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 | ||||
|         connectWebSocket(); | ||||
|     </script> | ||||
|   | ||||
| @@ -1,4 +1,52 @@ | ||||
| {{header}} | ||||
| <!-- 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="content"> | ||||
|         <h1>WiFi Configuration Page</h1> | ||||
|         <form action="/setToken" method="post"> | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								img/7-enable.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 52 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/ESP32-SPI-Pins.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 143 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/IMG_2589.jpeg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 136 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/IMG_2590.jpeg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 143 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/Schaltplan.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 283 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/display_1.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 571 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/display_2.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 470 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/display_3.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 535 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/display_4.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 534 KiB | 
| Before Width: | Height: | Size: 255 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/scale_1.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 735 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/scale_2.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 835 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/scale_3.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 814 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/scale_4.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 648 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/scale_side.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 594 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/scale_trans.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 665 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/web_1.jpeg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 488 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/web_ams.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 221 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/web_nfc.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 107 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/web_spoolm_to_ams.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 318 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/web_spoolman.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 156 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/web_statistics.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 110 KiB | 
							
								
								
									
										6
									
								
								partitions.csv
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,6 @@ | ||||
| # Name,   Type, SubType,    Offset,   Size,     Flags | ||||
| nvs,      data, nvs,       0x9000,   0x5000, | ||||
| otadata,  data, ota,       0xe000,   0x2000, | ||||
| app0,     app,  ota_0,     0x10000,  0x1E0000, | ||||
| app1,     app,  ota_1,     0x1F0000, 0x1E0000, | ||||
| spiffs,   data, spiffs,    0x3D0000, 0x30000, | ||||
| 
 | 
							
								
								
									
										73
									
								
								platformio.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,73 @@ | ||||
| ; PlatformIO Project Configuration File | ||||
| ; | ||||
| ;   Build options: build flags, source filter | ||||
| ;   Upload options: custom upload port, speed and extra flags | ||||
| ;   Library options: dependencies, extra library storages | ||||
| ;   Advanced options: extra scripting | ||||
| ; | ||||
| ; Please visit documentation for the other options and examples | ||||
| ; https://docs.platformio.org/page/projectconf.html | ||||
|  | ||||
| [common] | ||||
| version = "2.0.2-beta3" | ||||
| to_old_version = "1.5.10" | ||||
|  | ||||
| ## | ||||
| [env:esp32dev] | ||||
| platform = espressif32 | ||||
| board = esp32dev | ||||
| framework = arduino | ||||
| monitor_speed = 115200 | ||||
| #monitor_port = /dev/cu.usbmodem01 | ||||
|  | ||||
| lib_deps = | ||||
|     tzapu/WiFiManager @ ^2.0.17 | ||||
|     https://github.com/me-no-dev/ESPAsyncWebServer.git#master | ||||
|     https://github.com/esphome/AsyncTCP.git | ||||
|     bogde/HX711 @ ^0.7.5 | ||||
|     adafruit/Adafruit SSD1306 @ ^2.5.13 | ||||
|     adafruit/Adafruit GFX Library @ ^1.11.11 | ||||
|     adafruit/Adafruit PN532 @ ^1.3.3 | ||||
|     bblanchon/ArduinoJson @ ^7.3.0 | ||||
|     knolleary/PubSubClient @ ^2.8 | ||||
|     digitaldragon/SSLClient @ ^1.3.2 | ||||
|      | ||||
| ; Enable SPIFFS upload | ||||
| board_build.filesystem = littlefs | ||||
| ; Update partition settings | ||||
| board_build.partitions = partitions.csv | ||||
| board_upload.flash_size = 4MB | ||||
| board_build.flash_mode = dio | ||||
| board_upload.flash_freq = "40m" | ||||
|  | ||||
| build_flags =  | ||||
|     -Os | ||||
|     -ffunction-sections | ||||
|     -fdata-sections | ||||
|     #-DNDEBUG | ||||
|     -mtext-section-literals | ||||
|     -DVERSION=\"${common.version}\" | ||||
|     -DTOOLDVERSION=\"${common.to_old_version}\" | ||||
|     #-DENABLE_HEAP_DEBUGGING | ||||
|     -DASYNCWEBSERVER_REGEX | ||||
|     #-DCORE_DEBUG_LEVEL=3 | ||||
|     -DCONFIG_ARDUHAL_LOG_COLORS=1 | ||||
|     #-DOTA_DEBUG=1 | ||||
|     -DCONFIG_OPTIMIZATION_LEVEL_DEBUG=1 | ||||
|     -DBOOT_APP_PARTITION_OTA_0=1 | ||||
|     -DCONFIG_LWIP_TCP_MSL=60000 | ||||
|     -DCONFIG_LWIP_TCP_RCV_BUF_DEFAULT=4096 | ||||
|     -DCONFIG_LWIP_MAX_ACTIVE_TCP=16 | ||||
|      | ||||
| extra_scripts =  | ||||
|     scripts/extra_script.py | ||||
|     ${env:buildfs.extra_scripts} | ||||
|  | ||||
| [env:buildfs] | ||||
| extra_scripts = | ||||
|     pre:scripts/combine_html.py  ; Combine header with HTML files | ||||
|     scripts/gzip_files.py       ; Compress files for SPIFFS | ||||
|  | ||||
| [platformio] | ||||
| default_envs = esp32dev | ||||
|  | ||||
							
								
								
									
										34
									
								
								scripts/combine_html.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,34 @@ | ||||
| Import("env") | ||||
| import os | ||||
| import re | ||||
|  | ||||
| def combine_html_files(source, target, env): | ||||
|     print("COMBINE HTML FILES") | ||||
|      | ||||
|     html_dir = "./html" | ||||
|     header_file = os.path.join(html_dir, "header.html") | ||||
|      | ||||
|     # Read header content | ||||
|     with open(header_file, 'r') as f: | ||||
|         header_content = f.read() | ||||
|      | ||||
|     # Process all HTML files except header.html | ||||
|     for filename in os.listdir(html_dir): | ||||
|         if filename.endswith('.html') and filename != 'header.html': | ||||
|             file_path = os.path.join(html_dir, filename) | ||||
|              | ||||
|             # Read content | ||||
|             with open(file_path, 'r') as f: | ||||
|                 content = f.read() | ||||
|              | ||||
|             # Replace content between head comments with header content | ||||
|             pattern = r'(<!-- head -->).*?(<!-- head -->)' | ||||
|             new_content = re.sub(pattern, r'\1' + header_content + r'\2', content, flags=re.DOTALL) | ||||
|              | ||||
|             # Write back combined content | ||||
|             with open(file_path, 'w') as f: | ||||
|                 f.write(new_content) | ||||
|             print(f"Combined header with {filename}") | ||||
|  | ||||
| # Register the script to run before building SPIFFS | ||||
| env.AddPreAction("buildfs", combine_html_files) | ||||
| @@ -13,8 +13,11 @@ def copy_file(input_file, output_file): | ||||
|     shutil.copy2(input_file, output_file) | ||||
| 
 | ||||
| def should_compress(file): | ||||
|      # Skip compression for spoolman.html | ||||
|     if file == 'spoolman.html' or file == 'waage.html': | ||||
|         return False | ||||
|     # Komprimiere nur bestimmte Dateitypen | ||||
|     return file.endswith(('.js', '.png', '.css')) | ||||
|     return file.endswith(('.js', '.png', '.css', '.html')) | ||||
| 
 | ||||
| def main(source_dir, target_dir): | ||||
|     for root, dirs, files in os.walk(source_dir): | ||||
							
								
								
									
										25
									
								
								scripts/pre_build.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,25 @@ | ||||
| Import("env") | ||||
| import os | ||||
|  | ||||
| def replace_version(source, target, env): | ||||
|     # Get version from common section | ||||
|     version = env.GetProjectConfig().get("common", "version").strip('"') | ||||
|     header_file = "./html/header.html" | ||||
|      | ||||
|     with open(header_file, 'r') as file: | ||||
|         content = file.read() | ||||
|      | ||||
|     # Replace version in header.html using string manipulation instead of regex | ||||
|     search = '<h1>FilaMan<span class="version">v' | ||||
|     end = '</span>' | ||||
|     start_pos = content.find(search) | ||||
|     if start_pos != -1: | ||||
|         start_pos += len(search) | ||||
|         end_pos = content.find(end, start_pos) | ||||
|         if end_pos != -1: | ||||
|             content = content[:start_pos] + version + content[end_pos:] | ||||
|      | ||||
|     with open(header_file, 'w') as file: | ||||
|         file.write(content) | ||||
|  | ||||
| env.AddPreAction("buildfs", replace_version) | ||||
							
								
								
									
										39
									
								
								scripts/pre_spiffs.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,39 @@ | ||||
| Import("env") | ||||
|  | ||||
| board_config = env.BoardConfig() | ||||
|  | ||||
| # Calculate SPIFFS size based on partition table | ||||
| SPIFFS_START = 0x310000  # From partitions.csv | ||||
| SPIFFS_SIZE = 0xE0000   # From partitions.csv | ||||
| SPIFFS_PAGE = 256 | ||||
| SPIFFS_BLOCK = 4096 | ||||
|  | ||||
| env.Replace( | ||||
|     MKSPIFFSTOOL="mkspiffs", | ||||
|     SPIFFSBLOCKSZ=SPIFFS_BLOCK, | ||||
|     SPIFFSBLOCKSIZE=SPIFFS_BLOCK, | ||||
|     SPIFFSSTART=SPIFFS_START, | ||||
|     SPIFFSEND=SPIFFS_START + SPIFFS_SIZE, | ||||
|     SPIFFSPAGESZ=SPIFFS_PAGE, | ||||
|     SPIFFSSIZE=SPIFFS_SIZE | ||||
| ) | ||||
|  | ||||
| # Wiederverwendung der replace_version Funktion | ||||
| exec(open("./scripts/pre_build.py").read()) | ||||
|  | ||||
| # Bind to SPIFFS build | ||||
| env.AddPreAction("buildfs", replace_version) | ||||
|  | ||||
| import os | ||||
| import shutil | ||||
| from SCons.Script import DefaultEnvironment | ||||
|  | ||||
| env = DefaultEnvironment() | ||||
|  | ||||
| # Format SPIFFS partition before uploading new files | ||||
| spiffs_dir = os.path.join(env.subst("$BUILD_DIR"), "spiffs") | ||||
| if os.path.exists(spiffs_dir): | ||||
|     shutil.rmtree(spiffs_dir) | ||||
| os.makedirs(spiffs_dir) | ||||
|  | ||||
| print("SPIFFS partition formatted.") | ||||
							
								
								
									
										155
									
								
								scripts/update_changelog.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,155 @@ | ||||
| import os | ||||
| import re | ||||
| import subprocess | ||||
| from datetime import datetime | ||||
|  | ||||
| def get_version(): | ||||
|     script_dir = os.path.dirname(os.path.abspath(__file__)) | ||||
|     project_dir = os.path.dirname(script_dir) | ||||
|     platformio_path = os.path.join(project_dir, 'platformio.ini') | ||||
|      | ||||
|     with open(platformio_path, 'r') as f: | ||||
|         content = f.read() | ||||
|         version_match = re.search(r'version\s*=\s*"([^"]+)"', content) | ||||
|         return version_match.group(1) if version_match else None | ||||
|  | ||||
| def get_last_tag(): | ||||
|     """Get the last non-beta tag for changelog generation""" | ||||
|     try: | ||||
|         # Get all tags sorted by version | ||||
|         result = subprocess.run(['git', 'tag', '-l', '--sort=-version:refname'],  | ||||
|                               capture_output=True, text=True) | ||||
|         if result.returncode != 0: | ||||
|             return None | ||||
|              | ||||
|         tags = result.stdout.strip().split('\n') | ||||
|          | ||||
|         # Find the first (newest) non-beta tag | ||||
|         for tag in tags: | ||||
|             if tag and not '-beta' in tag.lower(): | ||||
|                 print(f"Using last stable tag for changelog: {tag}") | ||||
|                 return tag | ||||
|          | ||||
|         # Fallback: if no non-beta tags found, use the newest tag | ||||
|         print("No stable tags found, using newest tag") | ||||
|         if tags and tags[0]: | ||||
|             return tags[0] | ||||
|         return None | ||||
|     except subprocess.CalledProcessError: | ||||
|         return None | ||||
|  | ||||
| def categorize_commit(commit_msg): | ||||
|     """Categorize commit messages based on conventional commits""" | ||||
|     lower_msg = commit_msg.lower() | ||||
|      | ||||
|     # Check for breaking changes first | ||||
|     if ('!' in commit_msg and any(x in lower_msg for x in ['feat!', 'fix!', 'chore!', 'refactor!'])) or \ | ||||
|        'breaking change' in lower_msg or 'breaking:' in lower_msg: | ||||
|         return 'Breaking Changes' | ||||
|     elif any(x in lower_msg for x in ['feat', 'add', 'new']): | ||||
|         return 'Added' | ||||
|     elif any(x in lower_msg for x in ['fix', 'bug']): | ||||
|         return 'Fixed' | ||||
|     else: | ||||
|         return 'Changed' | ||||
|  | ||||
| def get_changes_from_git(): | ||||
|     """Get changes from git commits since last tag""" | ||||
|     changes = { | ||||
|         'Breaking Changes': [], | ||||
|         'Added': [], | ||||
|         'Changed': [], | ||||
|         'Fixed': [] | ||||
|     } | ||||
|      | ||||
|     last_tag = get_last_tag() | ||||
|      | ||||
|     # Get commits since last tag | ||||
|     git_log_command = ['git', 'log', '--pretty=format:%s'] | ||||
|     if last_tag: | ||||
|         git_log_command.append(f'{last_tag}..HEAD') | ||||
|      | ||||
|     try: | ||||
|         result = subprocess.run(git_log_command, capture_output=True, text=True) | ||||
|         commits = result.stdout.strip().split('\n') | ||||
|          | ||||
|         # Filter out empty commits and categorize | ||||
|         for commit in commits: | ||||
|             if commit: | ||||
|                 category = categorize_commit(commit) | ||||
|                 # Clean up commit message | ||||
|                 clean_msg = re.sub(r'^(feat|fix|chore|docs|style|refactor|perf|test)(\(.*\))?!?:', '', commit).strip() | ||||
|                 # Remove BREAKING CHANGE prefix if present | ||||
|                 clean_msg = re.sub(r'^breaking change:\s*', '', clean_msg, flags=re.IGNORECASE).strip() | ||||
|                 changes[category].append(clean_msg) | ||||
|                  | ||||
|     except subprocess.CalledProcessError: | ||||
|         print("Error: Failed to get git commits") | ||||
|         return None | ||||
|      | ||||
|     return changes | ||||
|  | ||||
| def update_changelog(): | ||||
|     print("Starting changelog update...") | ||||
|     version = get_version() | ||||
|     print(f"Current version: {version}") | ||||
|     today = datetime.now().strftime('%Y-%m-%d') | ||||
|      | ||||
|     script_dir = os.path.dirname(os.path.abspath(__file__)) | ||||
|     project_dir = os.path.dirname(script_dir) | ||||
|     changelog_path = os.path.join(project_dir, 'CHANGELOG.md') | ||||
|      | ||||
|     # Get changes from git | ||||
|     changes = get_changes_from_git() | ||||
|     if not changes: | ||||
|         print("No changes found or error occurred") | ||||
|         return | ||||
|      | ||||
|     # Create changelog entry | ||||
|     changelog_entry = f"## [{version}] - {today}\n" | ||||
|     for section, entries in changes.items(): | ||||
|         if entries:  # Only add sections that have entries | ||||
|             changelog_entry += f"### {section}\n" | ||||
|             for entry in entries: | ||||
|                 changelog_entry += f"- {entry}\n" | ||||
|             changelog_entry += "\n" | ||||
|      | ||||
|     if not os.path.exists(changelog_path): | ||||
|         with open(changelog_path, 'w') as f: | ||||
|             f.write(f"# Changelog\n\n{changelog_entry}") | ||||
|         print(f"Created new changelog file with version {version}") | ||||
|     else: | ||||
|         with open(changelog_path, 'r') as f: | ||||
|             content = f.read() | ||||
|          | ||||
|         if f"[{version}]" not in content: | ||||
|             updated_content = content.replace("# Changelog\n", f"# Changelog\n\n{changelog_entry}") | ||||
|             with open(changelog_path, 'w') as f: | ||||
|                 f.write(updated_content) | ||||
|             print(f"Added new version {version} to changelog") | ||||
|         else: | ||||
|             # Version existiert bereits, aktualisiere die bestehenden Einträge | ||||
|             version_pattern = f"## \\[{version}\\] - \\d{{4}}-\\d{{2}}-\\d{{2}}" | ||||
|             next_version_pattern = "## \\[.*?\\] - \\d{4}-\\d{2}-\\d{2}" | ||||
|              | ||||
|             # Finde den Start der aktuellen Version | ||||
|             version_match = re.search(version_pattern, content) | ||||
|             if version_match: | ||||
|                 version_start = version_match.start() | ||||
|                 # Suche nach der nächsten Version | ||||
|                 next_version_match = re.search(next_version_pattern, content[version_start + 1:]) | ||||
|                  | ||||
|                 if next_version_match: | ||||
|                     # Ersetze den Inhalt zwischen aktueller und nächster Version | ||||
|                     next_version_pos = version_start + 1 + next_version_match.start() | ||||
|                     updated_content = content[:version_start] + changelog_entry + content[next_version_pos:] | ||||
|                 else: | ||||
|                     # Wenn keine nächste Version existiert, ersetze bis zum Ende | ||||
|                     updated_content = content[:version_start] + changelog_entry + "\n" | ||||
|                  | ||||
|                 with open(changelog_path, 'w') as f: | ||||
|                     f.write(updated_content) | ||||
|                 print(f"Updated entries for version {version}") | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     update_changelog() | ||||
							
								
								
									
										1391
									
								
								src/api.cpp
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										39
									
								
								src/api.h
									
									
									
									
									
								
							
							
						
						| @@ -6,19 +6,46 @@ | ||||
| #include "website.h" | ||||
| #include "display.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 String spoolmanUrl; | ||||
| extern bool octoEnabled; | ||||
| extern bool sendOctoUpdate; | ||||
| extern String octoUrl; | ||||
| extern String octoToken; | ||||
| extern bool spoolmanConnected; | ||||
| extern uint16_t updateOctoSpoolId; | ||||
|  | ||||
| bool checkSpoolmanInstance(const String& url); | ||||
| bool saveSpoolmanUrl(const String& url); | ||||
| bool checkSpoolmanInstance(); | ||||
| bool saveSpoolmanUrl(const String& url, bool octoOn, const String& octoWh, const String& octoTk); | ||||
| String loadSpoolmanUrl(); // Neue Funktion zum Laden der URL | ||||
| bool checkSpoolmanExtraFields(); // Neue Funktion zum Überprüfen der Extrafelder | ||||
| JsonDocument fetchSpoolsForWebsite(); // API-Funktion für die Webseite | ||||
| JsonDocument fetchAllSpoolsInfo(); | ||||
| void sendAmsData(AsyncWebSocketClient *client); // Neue Funktion zum Senden von AMS-Daten | ||||
| uint8_t updateSpoolTagId(String uidString, const char* payload); // Neue Funktion zum Aktualisieren eines Spools | ||||
| JsonDocument fetchSingleSpoolInfo(int spoolId); // API-Funktion für die Webseite | ||||
| 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 updateSpoolLocation(String spoolId, String location); | ||||
| bool initSpoolman(); // Neue Funktion zum Initialisieren von Spoolman | ||||
| bool updateSpoolBambuData(String payload); // Neue Funktion zum Aktualisieren der Bambu-Daten | ||||
| bool updateSpoolOcto(int spoolId); // Neue Funktion zum Aktualisieren der Octo-Daten | ||||
| bool createBrandFilament(JsonDocument& payload, String uidString); | ||||
|  | ||||
| #endif | ||||
|   | ||||
							
								
								
									
										643
									
								
								src/bambu.cpp
									
									
									
									
									
								
							
							
						
						| @@ -10,6 +10,7 @@ | ||||
| #include "esp_task_wdt.h" | ||||
| #include "config.h" | ||||
| #include "display.h" | ||||
| #include <Preferences.h> | ||||
|  | ||||
| WiFiClient espClient; | ||||
| SSLClient sslClient(&espClient); | ||||
| @@ -17,30 +18,70 @@ PubSubClient client(sslClient); | ||||
|  | ||||
| TaskHandle_t BambuMqttTask; | ||||
|  | ||||
| String report_topic = ""; | ||||
| //String request_topic = ""; | ||||
| const char* bambu_username = "bblp"; | ||||
| const char* bambu_ip = nullptr; | ||||
| const char* bambu_accesscode = nullptr; | ||||
| const char* bambu_serialnr = nullptr; | ||||
| bool bambuDisabled = false; | ||||
|  | ||||
| bool bambu_connected = false; | ||||
| uint16_t autoSetToBambuSpoolId = 0; | ||||
|  | ||||
| BambuCredentials bambuCredentials; | ||||
|  | ||||
| // Globale Variablen für AMS-Daten | ||||
| int ams_count = 0; | ||||
| 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 saveBambuCredentials(const String& bambu_ip, const String& bambu_serialnr, const String& bambu_accesscode) { | ||||
|     JsonDocument doc; | ||||
|     doc["bambu_ip"] = bambu_ip; | ||||
|     doc["bambu_accesscode"] = bambu_accesscode; | ||||
|     doc["bambu_serialnr"] = bambu_serialnr; | ||||
|  | ||||
|     if (!saveJsonValue("/bambu_credentials.json", doc)) { | ||||
|         Serial.println("Fehler beim Speichern der Bambu-Credentials."); | ||||
|         return false; | ||||
| 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) { | ||||
|     if (BambuMqttTask) { | ||||
|         vTaskDelete(BambuMqttTask); | ||||
|         BambuMqttTask = NULL; | ||||
|     } | ||||
|  | ||||
|     bambuCredentials.ip = ip.c_str(); | ||||
|     bambuCredentials.serial = serialnr.c_str(); | ||||
|     bambuCredentials.accesscode = accesscode.c_str(); | ||||
|     bambuCredentials.autosend_enable = autoSend; | ||||
|     bambuCredentials.autosend_time = autoSendTime.toInt(); | ||||
|  | ||||
|     Preferences preferences; | ||||
|     preferences.begin(NVS_NAMESPACE_BAMBU, false); // false = readwrite | ||||
|     preferences.putString(NVS_KEY_BAMBU_IP, bambuCredentials.ip); | ||||
|     preferences.putString(NVS_KEY_BAMBU_SERIAL, bambuCredentials.serial); | ||||
|     preferences.putString(NVS_KEY_BAMBU_ACCESSCODE, bambuCredentials.accesscode); | ||||
|     preferences.putBool(NVS_KEY_BAMBU_AUTOSEND_ENABLE, bambuCredentials.autosend_enable); | ||||
|     preferences.putInt(NVS_KEY_BAMBU_AUTOSEND_TIME, bambuCredentials.autosend_time); | ||||
|     preferences.end(); | ||||
|  | ||||
|     vTaskDelay(100 / portTICK_PERIOD_MS); | ||||
|     if (!setupMqtt()) return false; | ||||
|  | ||||
| @@ -48,43 +89,81 @@ bool saveBambuCredentials(const String& bambu_ip, const String& bambu_serialnr, | ||||
| } | ||||
|  | ||||
| bool loadBambuCredentials() { | ||||
|     JsonDocument doc; | ||||
|     if (loadJsonValue("/bambu_credentials.json", doc) && doc.containsKey("bambu_ip")) { | ||||
|         // Temporäre Strings für die Werte | ||||
|         String ip = doc["bambu_ip"].as<String>(); | ||||
|         String code = doc["bambu_accesscode"].as<String>(); | ||||
|         String serial = doc["bambu_serialnr"].as<String>(); | ||||
|     Preferences preferences; | ||||
|     preferences.begin(NVS_NAMESPACE_BAMBU, true); | ||||
|     String ip = preferences.getString(NVS_KEY_BAMBU_IP, ""); | ||||
|     String serial = preferences.getString(NVS_KEY_BAMBU_SERIAL, ""); | ||||
|     String code = preferences.getString(NVS_KEY_BAMBU_ACCESSCODE, ""); | ||||
|     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(); | ||||
|  | ||||
|         ip.trim(); | ||||
|         code.trim(); | ||||
|         serial.trim(); | ||||
|     if(ip != ""){ | ||||
|         bambuCredentials.ip = ip.c_str(); | ||||
|         bambuCredentials.serial = serial.c_str(); | ||||
|         bambuCredentials.accesscode = code.c_str(); | ||||
|         bambuCredentials.autosend_enable = autosendEnable; | ||||
|         bambuCredentials.autosend_time = autosendTime; | ||||
|  | ||||
|         // Dynamische Speicherallokation für die globalen Pointer | ||||
|         bambu_ip = strdup(ip.c_str()); | ||||
|         bambu_accesscode = strdup(code.c_str()); | ||||
|         bambu_serialnr = strdup(serial.c_str()); | ||||
|         Serial.println("credentials loaded loadCredentials!"); | ||||
|         Serial.println(bambuCredentials.ip); | ||||
|         Serial.println(bambuCredentials.serial); | ||||
|         Serial.println(bambuCredentials.accesscode); | ||||
|         Serial.println(String(bambuCredentials.autosend_enable)); | ||||
|         Serial.println(String(bambuCredentials.autosend_time)); | ||||
|  | ||||
|         report_topic = "device/" + String(bambu_serialnr) + "/report"; | ||||
|         //request_topic = "device/" + String(bambu_serialnr) + "/request"; | ||||
|         return true; | ||||
|     } | ||||
|     Serial.println("Keine gültigen Bambu-Credentials gefunden."); | ||||
|     return false; | ||||
|     else | ||||
|     { | ||||
|         Serial.println("Keine gültigen Bambu-Credentials gefunden."); | ||||
|         return false; | ||||
|     } | ||||
| } | ||||
|  | ||||
| String findFilamentIdx(String brand, String type) { | ||||
| struct FilamentResult { | ||||
|     String key; | ||||
|     String type; | ||||
| }; | ||||
|  | ||||
| FilamentResult findFilamentIdx(String brand, String type) { | ||||
|     // JSON-Dokument für die Filament-Daten erstellen | ||||
|     JsonDocument doc; | ||||
|      | ||||
|     // Laden der own_filaments.json | ||||
|     String ownFilament = ""; | ||||
|     if (!loadJsonValue("/own_filaments.json", doc))  | ||||
|     { | ||||
|         Serial.println("Fehler beim Laden der eigenen Filament-Daten"); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         // Durchsuche direkt nach dem Type als Schlüssel | ||||
|         if (doc[type].is<String>()) { | ||||
|             ownFilament = doc[type].as<String>(); | ||||
|         } | ||||
|         doc.clear(); | ||||
|     } | ||||
|     doc.clear(); | ||||
|  | ||||
|     // Laden der bambu_filaments.json | ||||
|     if (!loadJsonValue("/bambu_filaments.json", doc)) { | ||||
|     if (!loadJsonValue("/bambu_filaments.json", doc))  | ||||
|     { | ||||
|         Serial.println("Fehler beim Laden der Filament-Daten"); | ||||
|         return "GFL99"; // Fallback auf Generic PLA | ||||
|         return {"GFL99", "PLA"}; // Fallback auf Generic PLA | ||||
|     } | ||||
|  | ||||
|     String searchKey; | ||||
|     // Wenn eigener Typ | ||||
|     if (ownFilament != "") | ||||
|     { | ||||
|         if (doc[ownFilament].is<String>())  | ||||
|         { | ||||
|             return {ownFilament, doc[ownFilament].as<String>()}; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // 1. Suche nach Brand + Type Kombination | ||||
|     // 1. Erst versuchen wir die exakte Brand + Type Kombination zu finden | ||||
|     String searchKey; | ||||
|     if (brand == "Bambu" || brand == "Bambulab") { | ||||
|         searchKey = "Bambu " + type; | ||||
|     } else if (brand == "PolyLite") { | ||||
| @@ -100,26 +179,49 @@ String findFilamentIdx(String brand, String type) { | ||||
|     // Durchsuche alle Einträge nach der Brand + Type Kombination | ||||
|     for (JsonPair kv : doc.as<JsonObject>()) { | ||||
|         if (kv.value().as<String>() == searchKey) { | ||||
|             return kv.key().c_str(); | ||||
|             return {kv.key().c_str(), kv.value().as<String>()}; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // 2. Wenn nicht gefunden, suche nach Generic + Type | ||||
|     searchKey = "Generic " + type; | ||||
|     // 2. Wenn nicht gefunden, zerlege den type String in Wörter und suche nach jedem Wort | ||||
|     // Sammle alle vorhandenen Filamenttypen aus der JSON | ||||
|     std::vector<String> knownTypes; | ||||
|     for (JsonPair kv : doc.as<JsonObject>()) { | ||||
|         if (kv.value().as<String>() == searchKey) { | ||||
|             return kv.key().c_str(); | ||||
|         String value = kv.value().as<String>(); | ||||
|         // Extrahiere den Typ ohne Markennamen | ||||
|         if (value.indexOf(" ") != -1) { | ||||
|             value = value.substring(value.indexOf(" ") + 1); | ||||
|         } | ||||
|         if (!value.isEmpty()) { | ||||
|             knownTypes.push_back(value); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Zerlege den Input-Type in Wörter | ||||
|     String typeStr = type; | ||||
|     typeStr.trim(); | ||||
|      | ||||
|     // Durchsuche für jedes bekannte Filament, ob es im Input vorkommt | ||||
|     for (const String& knownType : knownTypes) { | ||||
|         if (typeStr.indexOf(knownType) != -1) { | ||||
|             // Suche nach diesem Typ in der Original-JSON | ||||
|             for (JsonPair kv : doc.as<JsonObject>()) { | ||||
|                 String value = kv.value().as<String>(); | ||||
|                 if (value.indexOf(knownType) != -1) { | ||||
|                     return {kv.key().c_str(), knownType}; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // 3. Wenn immer noch nichts gefunden, gebe GFL99 zurück (Generic PLA) | ||||
|     return "GFL99"; | ||||
|     return {"GFL99", "PLA"}; | ||||
| } | ||||
|  | ||||
| bool sendMqttMessage(String payload) { | ||||
| bool sendMqttMessage(const String& payload) { | ||||
|     Serial.println("Sending MQTT message"); | ||||
|     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; | ||||
|     } | ||||
| @@ -128,43 +230,8 @@ bool sendMqttMessage(String payload) { | ||||
| } | ||||
|  | ||||
| bool setBambuSpool(String payload) { | ||||
|     /* payload | ||||
|     //// set Spool | ||||
|     { | ||||
|         "print": { | ||||
|             "sequence_id": 0, | ||||
|             "command": "ams_filament_setting", | ||||
|             "ams_id": 0, // AMS ID 0-3 oder externe Spule 255 | ||||
|             "tray_id": 0, // Tray ID 0-3 oder externe Spule 254 | ||||
|             "tray_color": "000000FF", | ||||
|             "nozzle_temp_min": 170, | ||||
|             "nozzle_temp_max": 200, | ||||
|             "tray_type": "PETG", | ||||
|             "setting_id": "", | ||||
|             "tray_info_idx": "GFG99" | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|  | ||||
|     //// Remove Spool | ||||
|     { | ||||
|         "print":{ | ||||
|             "ams_id":255, | ||||
|             "command":"ams_filament_setting", | ||||
|             "nozzle_temp_max": 0, | ||||
|             "nozzle_temp_min": 0, | ||||
|             "sequence_id": 0, | ||||
|             "setting_id": "", | ||||
|             "tray_color": "FFFFFFFF", | ||||
|             "tray_id": 254, | ||||
|             "tray_info_idx": "", | ||||
|             "tray_type": "", | ||||
|         } | ||||
|     } | ||||
|     */ | ||||
|  | ||||
|     Serial.println("Setting spool"); | ||||
|     Serial.println("Spool settings in"); | ||||
|     Serial.println(payload); | ||||
|  | ||||
|     // Parse the JSON | ||||
|     JsonDocument doc; | ||||
| @@ -182,21 +249,32 @@ bool setBambuSpool(String payload) { | ||||
|     int minTemp = doc["nozzle_temp_min"]; | ||||
|     int maxTemp = doc["nozzle_temp_max"]; | ||||
|     String type = doc["type"].as<String>(); | ||||
|     (type == "PLA+") ? type = "PLA" : type; | ||||
|     String brand = doc["brand"].as<String>(); | ||||
|     String tray_info_idx = findFilamentIdx(brand, type); | ||||
|     String tray_info_idx = (doc["tray_info_idx"].as<String>() != "-1") ? doc["tray_info_idx"].as<String>() : ""; | ||||
|     if (tray_info_idx == "") { | ||||
|         if (brand != "" && type != "") { | ||||
|             FilamentResult result = findFilamentIdx(brand, type); | ||||
|             tray_info_idx = result.key; | ||||
|             type = result.type;  // Aktualisiere den type mit dem gefundenen Basistyp | ||||
|         } | ||||
|     } | ||||
|     String setting_id = doc["bambu_setting_id"].as<String>(); | ||||
|     String cali_idx = doc["cali_idx"].as<String>(); | ||||
|  | ||||
|     doc.clear(); | ||||
|  | ||||
|     doc["print"]["sequence_id"] = 0; | ||||
|     doc["print"]["sequence_id"] = "0"; | ||||
|     doc["print"]["command"] = "ams_filament_setting"; | ||||
|     doc["print"]["ams_id"] = amsId < 200 ? amsId-1 : 255; | ||||
|     doc["print"]["tray_id"] = trayId < 200 ? trayId-1 : 254; | ||||
|     doc["print"]["ams_id"] = amsId < 200 ? amsId : 255; | ||||
|     doc["print"]["tray_id"] = trayId < 200 ? trayId : 254; | ||||
|     doc["print"]["tray_color"] = color.length() == 8 ? color : color+"FF"; | ||||
|     doc["print"]["nozzle_temp_min"] = minTemp; | ||||
|     doc["print"]["nozzle_temp_max"] = maxTemp; | ||||
|     doc["print"]["tray_type"] = type; | ||||
|     doc["print"]["setting_id"] = ""; | ||||
|     //doc["print"]["cali_idx"] = (cali_idx != "") ? cali_idx : ""; | ||||
|     doc["print"]["tray_info_idx"] = tray_info_idx; | ||||
|     doc["print"]["setting_id"] = setting_id; | ||||
|      | ||||
|     // Serialize the JSON | ||||
|     String output; | ||||
| @@ -211,12 +289,149 @@ bool setBambuSpool(String payload) { | ||||
|         return false; | ||||
|     } | ||||
|      | ||||
|     doc.clear(); | ||||
|     yield(); | ||||
|  | ||||
|     if (cali_idx != "") { | ||||
|         yield(); | ||||
|         doc["print"]["sequence_id"] = "0"; | ||||
|         doc["print"]["command"] = "extrusion_cali_sel"; | ||||
|         doc["print"]["filament_id"] = tray_info_idx; | ||||
|         doc["print"]["nozzle_diameter"] = "0.4"; | ||||
|         doc["print"]["cali_idx"] = cali_idx.toInt(); | ||||
|         doc["print"]["tray_id"] = trayId < 200 ? trayId : 254; | ||||
|         //doc["print"]["ams_id"] = amsId < 200 ? amsId : 255; | ||||
|  | ||||
|         // Serialize the JSON | ||||
|         String output; | ||||
|         serializeJson(doc, output); | ||||
|  | ||||
|         if (sendMqttMessage(output)) { | ||||
|             Serial.println("Extrusion calibration successfully set"); | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             Serial.println("Failed to set extrusion calibration"); | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         doc.clear(); | ||||
|         yield(); | ||||
|     } | ||||
|  | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| void autoSetSpool(int spoolId, uint8_t trayId) { | ||||
|     // wenn neue spule erkannt und autoSetToBambu > 0 | ||||
|     JsonDocument spoolInfo = fetchSingleSpoolInfo(spoolId); | ||||
|  | ||||
|     if (!spoolInfo.isNull()) | ||||
|     { | ||||
|         // AMS und TRAY id ergänzen | ||||
|         spoolInfo["amsId"] = 0; | ||||
|         spoolInfo["trayId"] = trayId; | ||||
|  | ||||
|         Serial.println("Auto set spool"); | ||||
|         Serial.println(spoolInfo.as<String>()); | ||||
|  | ||||
|         setBambuSpool(spoolInfo.as<String>()); | ||||
|  | ||||
|         oledShowMessage("Spool set"); | ||||
|     } | ||||
|  | ||||
|     // id wieder zurücksetzen damit abgeschlossen | ||||
|     autoSetToBambuSpoolId = 0; | ||||
| } | ||||
|  | ||||
| void updateAmsWsData(JsonDocument& doc, JsonArray& amsArray, int& ams_count, JsonObject& vtTray) { | ||||
|     // Fortfahren mit der bestehenden Verarbeitung, da Änderungen gefunden wurden | ||||
|     ams_count = amsArray.size(); | ||||
|          | ||||
|     for (int i = 0; i < ams_count && i < 16; i++) { | ||||
|         JsonObject amsObj = amsArray[i]; | ||||
|         JsonArray trayArray = amsObj["tray"].as<JsonArray>(); | ||||
|  | ||||
|         ams_data[i].ams_id = i; // Setze die AMS-ID | ||||
|         for (int j = 0; j < trayArray.size() && j < 4; j++) { // Annahme: Maximal 4 Trays pro AMS | ||||
|             JsonObject trayObj = trayArray[j]; | ||||
|  | ||||
|             ams_data[i].trays[j].id = trayObj["id"].as<uint8_t>(); | ||||
|             ams_data[i].trays[j].tray_info_idx = trayObj["tray_info_idx"].as<String>(); | ||||
|             ams_data[i].trays[j].tray_type = trayObj["tray_type"].as<String>(); | ||||
|             ams_data[i].trays[j].tray_sub_brands = trayObj["tray_sub_brands"].as<String>(); | ||||
|             ams_data[i].trays[j].tray_color = trayObj["tray_color"].as<String>(); | ||||
|             ams_data[i].trays[j].nozzle_temp_min = trayObj["nozzle_temp_min"].as<int>(); | ||||
|             ams_data[i].trays[j].nozzle_temp_max = trayObj["nozzle_temp_max"].as<int>(); | ||||
|             if (trayObj["tray_type"].as<String>() == "") ams_data[i].trays[j].setting_id = ""; | ||||
|             ams_data[i].trays[j].cali_idx = trayObj["cali_idx"].as<String>(); | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     // Setze ams_count auf die Anzahl der normalen AMS | ||||
|     ams_count = amsArray.size(); | ||||
|  | ||||
|     // Wenn externe Spule vorhanden, füge sie hinzu | ||||
|     if (doc["print"]["vt_tray"].is<JsonObject>()) { | ||||
|         //JsonObject vtTray = doc["print"]["vt_tray"]; | ||||
|         int extIdx = ams_count;  // Index für externe Spule | ||||
|         ams_data[extIdx].ams_id = 255;  // Spezielle ID für externe Spule | ||||
|         ams_data[extIdx].trays[0].id = 254;  // Spezielle ID für externes Tray | ||||
|         ams_data[extIdx].trays[0].tray_info_idx = vtTray["tray_info_idx"].as<String>(); | ||||
|         ams_data[extIdx].trays[0].tray_type = vtTray["tray_type"].as<String>(); | ||||
|         ams_data[extIdx].trays[0].tray_sub_brands = vtTray["tray_sub_brands"].as<String>(); | ||||
|         ams_data[extIdx].trays[0].tray_color = vtTray["tray_color"].as<String>(); | ||||
|         ams_data[extIdx].trays[0].nozzle_temp_min = vtTray["nozzle_temp_min"].as<int>(); | ||||
|         ams_data[extIdx].trays[0].nozzle_temp_max = vtTray["nozzle_temp_max"].as<int>(); | ||||
|  | ||||
|         if (doc["print"]["vt_tray"]["tray_type"].as<String>() != "") | ||||
|         { | ||||
|             //ams_data[extIdx].trays[0].setting_id = vtTray["setting_id"].as<String>(); | ||||
|             ams_data[extIdx].trays[0].cali_idx = vtTray["cali_idx"].as<String>(); | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             ams_data[extIdx].trays[0].setting_id = ""; | ||||
|             ams_data[extIdx].trays[0].cali_idx = ""; | ||||
|         } | ||||
|         ams_count++;  // Erhöhe ams_count für die externe Spule | ||||
|     } | ||||
|  | ||||
|     // Erstelle JSON für WebSocket-Clients | ||||
|     JsonDocument wsDoc; | ||||
|     JsonArray wsArray = wsDoc.to<JsonArray>(); | ||||
|  | ||||
|     for (int i = 0; i < ams_count; i++) { | ||||
|         JsonObject amsObj = wsArray.add<JsonObject>(); | ||||
|         amsObj["ams_id"] = ams_data[i].ams_id; | ||||
|  | ||||
|         JsonArray trays = amsObj["tray"].to<JsonArray>(); | ||||
|         int maxTrays = (ams_data[i].ams_id == 255) ? 1 : 4; | ||||
|          | ||||
|         for (int j = 0; j < maxTrays; j++) { | ||||
|             JsonObject trayObj = trays.add<JsonObject>(); | ||||
|             trayObj["id"] = ams_data[i].trays[j].id; | ||||
|             trayObj["tray_info_idx"] = ams_data[i].trays[j].tray_info_idx; | ||||
|             trayObj["tray_type"] = ams_data[i].trays[j].tray_type; | ||||
|             trayObj["tray_sub_brands"] = ams_data[i].trays[j].tray_sub_brands; | ||||
|             trayObj["tray_color"] = ams_data[i].trays[j].tray_color; | ||||
|             trayObj["nozzle_temp_min"] = ams_data[i].trays[j].nozzle_temp_min; | ||||
|             trayObj["nozzle_temp_max"] = ams_data[i].trays[j].nozzle_temp_max; | ||||
|             trayObj["setting_id"] = ams_data[i].trays[j].setting_id; | ||||
|             trayObj["cali_idx"] = ams_data[i].trays[j].cali_idx; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     serializeJson(wsArray, amsJsonData); | ||||
|     wsDoc.clear(); | ||||
|     Serial.println("AMS data updated"); | ||||
|     sendAmsData(nullptr); | ||||
| } | ||||
|  | ||||
| // init | ||||
| void mqtt_callback(char* topic, byte* payload, unsigned int length) { | ||||
|     String message; | ||||
|      | ||||
|     for (int i = 0; i < length; i++) { | ||||
|         message += (char)payload[i]; | ||||
|     } | ||||
| @@ -224,16 +439,20 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | ||||
|     // JSON-Dokument parsen | ||||
|     JsonDocument doc; | ||||
|     DeserializationError error = deserializeJson(doc, message); | ||||
|     if (error) { | ||||
|     message = ""; | ||||
|     if (error)  | ||||
|     { | ||||
|         Serial.print("Fehler beim Parsen des JSON: "); | ||||
|         Serial.println(error.c_str()); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // Prüfen, ob "print->upgrade_state" und "print.ams.ams" existieren | ||||
|     if (doc["print"].containsKey("upgrade_state")) { | ||||
|     if (doc["print"]["upgrade_state"].is<JsonObject>() || (doc["print"]["command"].is<String>() && doc["print"]["command"] == "push_status"))  | ||||
|     { | ||||
|         // Prüfen ob AMS-Daten vorhanden sind | ||||
|         if (!doc["print"].containsKey("ams") || !doc["print"]["ams"].containsKey("ams")) { | ||||
|         if (!doc["print"]["ams"].is<JsonObject>() || !doc["print"]["ams"]["ams"].is<JsonArray>())  | ||||
|         { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
| @@ -265,139 +484,102 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | ||||
|             // Vergleiche die Trays | ||||
|             for (int j = 0; j < trayArray.size() && j < 4 && !hasChanges; j++) { | ||||
|                 JsonObject trayObj = trayArray[j]; | ||||
|  | ||||
|                 if (trayObj["tray_type"].as<String>() == "") ams_data[storedIndex].trays[j].setting_id = ""; | ||||
|                 if (trayObj["setting_id"].isNull()) trayObj["setting_id"] = ""; | ||||
|                 if (trayObj["tray_info_idx"].as<String>() != ams_data[storedIndex].trays[j].tray_info_idx || | ||||
|                     trayObj["tray_type"].as<String>() != ams_data[storedIndex].trays[j].tray_type || | ||||
|                     trayObj["tray_color"].as<String>() != ams_data[storedIndex].trays[j].tray_color) { | ||||
|                     trayObj["tray_color"].as<String>() != ams_data[storedIndex].trays[j].tray_color || | ||||
|                     (trayObj["setting_id"].as<String>() != "" && trayObj["setting_id"].as<String>() != ams_data[storedIndex].trays[j].setting_id) || | ||||
|                     trayObj["cali_idx"].as<String>() != ams_data[storedIndex].trays[j].cali_idx) { | ||||
|                     hasChanges = true; | ||||
|  | ||||
|                     if (bambuCredentials.autosend_enable && autoSetToBambuSpoolId > 0 && hasChanges) | ||||
|                     { | ||||
|                         autoSetSpool(autoSetToBambuSpoolId, ams_data[storedIndex].trays[j].id); | ||||
|                     } | ||||
|  | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Prüfe die externe Spule | ||||
|         if (!hasChanges && doc["print"].containsKey("vt_tray")) { | ||||
|             JsonObject vtTray = doc["print"]["vt_tray"]; | ||||
|             bool foundExternal = false; | ||||
|              | ||||
|         JsonObject vtTray = doc["print"]["vt_tray"]; | ||||
|         if (doc["print"]["vt_tray"].is<JsonObject>()) { | ||||
|             for (int i = 0; i < ams_count; i++) { | ||||
|                 if (ams_data[i].ams_id == 255) { | ||||
|                     foundExternal = true; | ||||
|                     if (vtTray["tray_type"].as<String>() == "") ams_data[i].trays[0].setting_id = ""; | ||||
|                     if (vtTray["setting_id"].isNull()) vtTray["setting_id"] = ""; | ||||
|                     if (vtTray["tray_info_idx"].as<String>() != ams_data[i].trays[0].tray_info_idx || | ||||
|                         vtTray["tray_type"].as<String>() != ams_data[i].trays[0].tray_type || | ||||
|                         vtTray["tray_color"].as<String>() != ams_data[i].trays[0].tray_color) { | ||||
|                         vtTray["tray_color"].as<String>() != ams_data[i].trays[0].tray_color || | ||||
|                         (vtTray["setting_id"].as<String>() != "" && vtTray["setting_id"].as<String>() != ams_data[i].trays[0].setting_id) || | ||||
|                         (vtTray["tray_type"].as<String>() != "" && vtTray["cali_idx"].as<String>() != ams_data[i].trays[0].cali_idx)) { | ||||
|                         hasChanges = true; | ||||
|  | ||||
|                         if (bambuCredentials.autosend_enable && autoSetToBambuSpoolId > 0 && hasChanges) | ||||
|                         { | ||||
|                             autoSetSpool(autoSetToBambuSpoolId, 254); | ||||
|                         } | ||||
|                     } | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|             if (!foundExternal) hasChanges = true; | ||||
|         } | ||||
|  | ||||
|         if (!hasChanges) return; | ||||
|  | ||||
|         // Fortfahren mit der bestehenden Verarbeitung, da Änderungen gefunden wurden | ||||
|         ams_count = amsArray.size(); | ||||
|         updateAmsWsData(doc, amsArray, ams_count, vtTray); | ||||
|     } | ||||
|      | ||||
|         // Restlicher bestehender Code... | ||||
|         for (int i = 0; i < ams_count && i < 16; i++) { | ||||
|             JsonObject amsObj = amsArray[i]; | ||||
|             JsonArray trayArray = amsObj["tray"].as<JsonArray>(); | ||||
|  | ||||
|             ams_data[i].ams_id = i; // Setze die AMS-ID | ||||
|             for (int j = 0; j < trayArray.size() && j < 4; j++) { // Annahme: Maximal 4 Trays pro AMS | ||||
|                 JsonObject trayObj = trayArray[j]; | ||||
|  | ||||
|                 ams_data[i].trays[j].id = trayObj["id"].as<uint8_t>(); | ||||
|                 ams_data[i].trays[j].tray_info_idx = trayObj["tray_info_idx"].as<String>(); | ||||
|                 ams_data[i].trays[j].tray_type = trayObj["tray_type"].as<String>(); | ||||
|                 ams_data[i].trays[j].tray_sub_brands = trayObj["tray_sub_brands"].as<String>(); | ||||
|                 ams_data[i].trays[j].tray_color = trayObj["tray_color"].as<String>(); | ||||
|                 ams_data[i].trays[j].nozzle_temp_min = trayObj["nozzle_temp_min"].as<int>(); | ||||
|                 ams_data[i].trays[j].nozzle_temp_max = trayObj["nozzle_temp_max"].as<int>(); | ||||
|                 ams_data[i].trays[j].setting_id = trayObj["setting_id"].as<String>(); | ||||
|             } | ||||
|         } | ||||
|         //Serial.println("----------------"); | ||||
|         //Serial.println(); | ||||
|  | ||||
|         // Sende die aktualisierten AMS-Daten an alle WebSocket-Clients | ||||
|         sendAmsData(nullptr); | ||||
|  | ||||
|         // Verarbeite erst die normalen AMS-Daten | ||||
|         for (int i = 0; i < amsArray.size() && i < 16; i++) { | ||||
|             JsonObject amsObj = amsArray[i]; | ||||
|             JsonArray trayArray = amsObj["tray"].as<JsonArray>(); | ||||
|  | ||||
|             ams_data[i].ams_id = amsObj["id"].as<uint8_t>(); | ||||
|             for (int j = 0; j < trayArray.size() && j < 4; j++) { | ||||
|                 JsonObject trayObj = trayArray[j]; | ||||
|                 ams_data[i].trays[j].id = trayObj["id"].as<uint8_t>(); | ||||
|                 ams_data[i].trays[j].tray_info_idx = trayObj["tray_info_idx"].as<String>(); | ||||
|                 // ... weitere Tray-Daten ... | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         // Setze ams_count auf die Anzahl der normalen AMS | ||||
|         ams_count = amsArray.size(); | ||||
|  | ||||
|         // Wenn externe Spule vorhanden, füge sie hinzu | ||||
|         if (doc["print"].containsKey("vt_tray")) { | ||||
|             JsonObject vtTray = doc["print"]["vt_tray"]; | ||||
|             int extIdx = ams_count;  // Index für externe Spule | ||||
|             ams_data[extIdx].ams_id = 255;  // Spezielle ID für externe Spule | ||||
|             ams_data[extIdx].trays[0].id = 254;  // Spezielle ID für externes Tray | ||||
|             ams_data[extIdx].trays[0].tray_info_idx = vtTray["tray_info_idx"].as<String>(); | ||||
|             ams_data[extIdx].trays[0].tray_type = vtTray["tray_type"].as<String>(); | ||||
|             ams_data[extIdx].trays[0].tray_sub_brands = vtTray["tray_sub_brands"].as<String>(); | ||||
|             ams_data[extIdx].trays[0].tray_color = vtTray["tray_color"].as<String>(); | ||||
|             ams_data[extIdx].trays[0].nozzle_temp_min = vtTray["nozzle_temp_min"].as<int>(); | ||||
|             ams_data[extIdx].trays[0].nozzle_temp_max = vtTray["nozzle_temp_max"].as<int>(); | ||||
|             ams_data[extIdx].trays[0].setting_id = vtTray["setting_id"].as<String>(); | ||||
|             ams_count++;  // Erhöhe ams_count für die externe Spule | ||||
|         } | ||||
|  | ||||
|         // Sende die aktualisierten AMS-Daten | ||||
|         sendAmsData(nullptr); | ||||
|  | ||||
|         // Erstelle JSON für WebSocket-Clients | ||||
|         JsonDocument wsDoc; | ||||
|         JsonArray wsArray = wsDoc.to<JsonArray>(); | ||||
|     // Neue Bedingung für ams_filament_setting | ||||
|     if (doc["print"]["command"] == "ams_filament_setting") { | ||||
|         int amsId = doc["print"]["ams_id"].as<int>(); | ||||
|         int trayId = doc["print"]["tray_id"].as<int>(); | ||||
|         String settingId = (doc["print"]["setting_id"].is<String>()) ? doc["print"]["setting_id"].as<String>() : ""; | ||||
|  | ||||
|         // Finde das entsprechende AMS und Tray | ||||
|         for (int i = 0; i < ams_count; i++) { | ||||
|             JsonObject amsObj = wsArray.createNestedObject(); | ||||
|             amsObj["ams_id"] = ams_data[i].ams_id; | ||||
|             if (ams_data[i].ams_id == amsId) { | ||||
|                 if (trayId == 254) | ||||
|                 { | ||||
|                     // Suche AMS mit ID 255 (externe Spule) | ||||
|                     for (int j = 0; j < ams_count; j++) { | ||||
|                         if (ams_data[j].ams_id == 255) { | ||||
|                             ams_data[j].trays[0].setting_id = settingId; | ||||
|                             break; | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     ams_data[i].trays[trayId].setting_id = settingId; | ||||
|                 } | ||||
|                 | ||||
|             JsonArray trays = amsObj.createNestedArray("tray"); | ||||
|             int maxTrays = (ams_data[i].ams_id == 255) ? 1 : 4; | ||||
|              | ||||
|             for (int j = 0; j < maxTrays; j++) { | ||||
|                 JsonObject trayObj = trays.createNestedObject(); | ||||
|                 trayObj["id"] = ams_data[i].trays[j].id; | ||||
|                 trayObj["tray_info_idx"] = ams_data[i].trays[j].tray_info_idx; | ||||
|                 trayObj["tray_type"] = ams_data[i].trays[j].tray_type; | ||||
|                 trayObj["tray_sub_brands"] = ams_data[i].trays[j].tray_sub_brands; | ||||
|                 trayObj["tray_color"] = ams_data[i].trays[j].tray_color; | ||||
|                 trayObj["nozzle_temp_min"] = ams_data[i].trays[j].nozzle_temp_min; | ||||
|                 trayObj["nozzle_temp_max"] = ams_data[i].trays[j].nozzle_temp_max; | ||||
|                 trayObj["setting_id"] = ams_data[i].trays[j].setting_id; | ||||
|                 // Sende an WebSocket Clients | ||||
|                 Serial.println("Filament setting updated"); | ||||
|                 sendAmsData(nullptr); | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         serializeJson(wsArray, amsJsonData); | ||||
|         sendAmsData(nullptr); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void reconnect() { | ||||
|     // Loop until we're reconnected | ||||
|     uint8_t retries = 0; | ||||
|     while (!client.connected()) { | ||||
|         Serial.print("Attempting MQTT connection..."); | ||||
|         Serial.println("Attempting MQTT re/connection..."); | ||||
|         bambu_connected = false; | ||||
|         oledShowTopRow(); | ||||
|  | ||||
|         // Attempt to connect | ||||
|         if (client.connect(bambu_serialnr, bambu_username, bambu_accesscode)) { | ||||
|             Serial.println("... re-connected"); | ||||
|             // ... and resubscribe | ||||
|             client.subscribe(report_topic.c_str()); | ||||
|         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"); | ||||
|  | ||||
|             client.subscribe(("device/"+bambuCredentials.serial+"/report").c_str()); | ||||
|             bambu_connected = true; | ||||
|             oledShowTopRow(); | ||||
|         } else { | ||||
| @@ -406,16 +588,24 @@ void reconnect() { | ||||
|             Serial.println(" try again in 5 seconds"); | ||||
|             bambu_connected = false; | ||||
|             oledShowTopRow(); | ||||
|             // Wait 5 seconds before retrying | ||||
|             delay(5000); | ||||
|              | ||||
|             yield(); | ||||
|             vTaskDelay(5000 / portTICK_PERIOD_MS); | ||||
|             if (retries > 5) { | ||||
|                 Serial.println("Disable Bambu MQTT Task after 5 retries"); | ||||
|                 //vTaskSuspend(BambuMqttTask); | ||||
|                 vTaskDelete(BambuMqttTask); | ||||
|                 BambuMqttTask = NULL; | ||||
|                 break; | ||||
|             } | ||||
|  | ||||
|             retries++; | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| void mqtt_loop(void * parameter) { | ||||
|     oledShowMessage("Bambu Connected"); | ||||
|     bambu_connected = true; | ||||
|     oledShowTopRow(); | ||||
|     Serial.println("Bambu MQTT Task gestartet"); | ||||
|     for(;;) { | ||||
|         if (pauseBambuMqttTask) { | ||||
|             vTaskDelay(10000); | ||||
| @@ -428,59 +618,74 @@ void mqtt_loop(void * parameter) { | ||||
|             vTaskDelay(100); | ||||
|         } | ||||
|         client.loop(); | ||||
|         yield(); | ||||
|         esp_task_wdt_reset(); | ||||
|         vTaskDelay(100); | ||||
|     } | ||||
| } | ||||
|  | ||||
| bool setupMqtt() { | ||||
|     // Wenn Bambu Daten vorhanden | ||||
|     bool success = loadBambuCredentials(); | ||||
|     vTaskDelay(100 / portTICK_PERIOD_MS); | ||||
|     //bool success = loadBambuCredentials(); | ||||
|  | ||||
|     if (!success) { | ||||
|         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 != "") { | ||||
|     if (bambuCredentials.ip != "" && bambuCredentials.accesscode != "" && bambuCredentials.serial != "")  | ||||
|     { | ||||
|         oledShowProgressBar(4, 7, DISPLAY_BOOT_TEXT, "Bambu init"); | ||||
|         bambuDisabled = false; | ||||
|         sslClient.setCACert(root_ca); | ||||
|         sslClient.setInsecure(); | ||||
|         client.setServer(bambu_ip, 8883); | ||||
|         client.setServer(bambuCredentials.ip.c_str(), 8883); | ||||
|  | ||||
|         // Verbinden mit dem MQTT-Server | ||||
|         if (client.connect(bambu_serialnr, bambu_username, bambu_accesscode)) { | ||||
|         bool connected = true; | ||||
|         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.setBufferSize(5120); | ||||
|             // Optional: Topic abonnieren | ||||
|             client.subscribe(report_topic.c_str()); | ||||
|             //client.subscribe(request_topic.c_str()); | ||||
|             client.setBufferSize(15488); | ||||
|             client.subscribe(("device/"+bambuCredentials.serial+"/report").c_str()); | ||||
|             Serial.println("MQTT-Client initialisiert"); | ||||
|  | ||||
|             oledShowMessage("Bambu Connected"); | ||||
|             bambu_connected = true; | ||||
|             oledShowTopRow(); | ||||
|  | ||||
|             xTaskCreatePinnedToCore( | ||||
|                 mqtt_loop, /* Function to implement the task */ | ||||
|                 "BambuMqtt", /* Name of the task */ | ||||
|                 10000,  /* Stack size in words */ | ||||
|                 8192,  /* Stack size in words */ | ||||
|                 NULL,  /* Task input parameter */ | ||||
|                 mqttTaskPrio,  /* Priority of the task */ | ||||
|                 &BambuMqttTask,  /* Task handle. */ | ||||
|                 mqttTaskCore); /* Core where the task should run */ | ||||
|  | ||||
|         } else { | ||||
|         }  | ||||
|         else  | ||||
|         { | ||||
|             Serial.println("Fehler: Konnte sich nicht beim MQTT-Server anmelden"); | ||||
|             oledShowMessage("Bambu Connection Failed"); | ||||
|             oledShowTopRow(); | ||||
|             vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||
|             return false; | ||||
|             connected = false; | ||||
|             oledShowTopRow(); | ||||
|             autoSetToBambuSpoolId = 0; | ||||
|         } | ||||
|     } else { | ||||
|         Serial.println("Fehler: Keine MQTT-Daten vorhanden"); | ||||
|         oledShowMessage("Bambu Credentials Missing"); | ||||
|         oledShowTopRow(); | ||||
|         vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||
|  | ||||
|         if (!connected) return false; | ||||
|     }  | ||||
|     else  | ||||
|     { | ||||
|         bambuDisabled = true; | ||||
|         return false; | ||||
|     } | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| void bambu_restart() { | ||||
|     Serial.println("Bambu restart"); | ||||
|  | ||||
|     if (BambuMqttTask) { | ||||
|         vTaskDelete(BambuMqttTask); | ||||
|         BambuMqttTask = NULL; | ||||
|         delay(10); | ||||
|     } | ||||
|     setupMqtt(); | ||||
| } | ||||
							
								
								
									
										18
									
								
								src/bambu.h
									
									
									
									
									
								
							
							
						
						| @@ -13,6 +13,15 @@ struct TrayData { | ||||
|     int nozzle_temp_min; | ||||
|     int nozzle_temp_max; | ||||
|     String setting_id; | ||||
|     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 | ||||
| @@ -27,11 +36,18 @@ extern bool bambu_connected; | ||||
|  | ||||
| extern int ams_count; | ||||
| extern AMSData ams_data[MAX_AMS]; | ||||
| //extern bool autoSendToBambu; | ||||
| extern uint16_t autoSetToBambuSpoolId; | ||||
| extern bool bambuDisabled; | ||||
| extern BambuCredentials bambuCredentials; | ||||
|  | ||||
| bool removeBambuCredentials(); | ||||
| bool loadBambuCredentials(); | ||||
| bool saveBambuCredentials(const String& bambu_ip, const String& bambu_serialnr, const String& bambu_accesscode); | ||||
| bool saveBambuCredentials(const String& bambu_ip, const String& bambu_serialnr, const String& bambu_accesscode, const bool autoSend, const String& autoSendTime); | ||||
| bool setupMqtt(); | ||||
| void mqtt_loop(void * parameter); | ||||
| bool setBambuSpool(String payload); | ||||
| void bambu_restart(); | ||||
|  | ||||
| extern TaskHandle_t BambuMqttTask; | ||||
| #endif | ||||
|   | ||||
| @@ -1,24 +1,40 @@ | ||||
| #include "commonFS.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) { | ||||
|     File file = SPIFFS.open(filename, "w"); | ||||
|     File file = LittleFS.open(filename, "w"); | ||||
|     if (!file) { | ||||
|         Serial.print("Fehler beim Öffnen der Datei zum Schreiben: "); | ||||
|         Serial.println(filename); | ||||
|         return false; | ||||
|     } | ||||
|     return true; | ||||
|  | ||||
|     if (serializeJson(doc, file) == 0) { | ||||
|         Serial.println("Fehler beim Serialisieren von JSON."); | ||||
|         file.close(); | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     file.close(); | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| bool loadJsonValue(const char* filename, JsonDocument& doc) { | ||||
|     File file = SPIFFS.open(filename, "r"); | ||||
|     File file = LittleFS.open(filename, "r"); | ||||
|     if (!file) { | ||||
|         Serial.print("Fehler beim Öffnen der Datei zum Lesen: "); | ||||
|         Serial.println(filename); | ||||
| @@ -34,23 +50,12 @@ bool loadJsonValue(const char* filename, JsonDocument& doc) { | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| bool initializeSPIFFS() { | ||||
|     // Erster Versuch | ||||
|     if (SPIFFS.begin(true)) { | ||||
|         Serial.println("SPIFFS mounted successfully."); | ||||
|         return true; | ||||
| void initializeFileSystem() { | ||||
|     if (!LittleFS.begin(true)) { | ||||
|         Serial.println("LittleFS Mount Failed"); | ||||
|         return; | ||||
|     } | ||||
|      | ||||
|     // Formatierung versuchen | ||||
|     Serial.println("Failed to mount SPIFFS. Formatting..."); | ||||
|     SPIFFS.format(); | ||||
|      | ||||
|     // Zweiter Versuch nach Formatierung | ||||
|     if (SPIFFS.begin(true)) { | ||||
|         Serial.println("SPIFFS formatted and mounted successfully."); | ||||
|         return true; | ||||
|     } | ||||
|      | ||||
|     Serial.println("SPIFFS initialization failed completely."); | ||||
|     return false; | ||||
|     Serial.printf("LittleFS Total: %u bytes\n", LittleFS.totalBytes()); | ||||
|     Serial.printf("LittleFS Used: %u bytes\n", LittleFS.usedBytes()); | ||||
|     Serial.printf("LittleFS Free: %u bytes\n", LittleFS.totalBytes() - LittleFS.usedBytes()); | ||||
| } | ||||
| @@ -2,11 +2,12 @@ | ||||
| #define COMMONFS_H | ||||
|  | ||||
| #include <Arduino.h> | ||||
| #include <SPIFFS.h> | ||||
| #include <ArduinoJson.h> | ||||
| #include <LittleFS.h> | ||||
|  | ||||
| bool removeJsonValue(const char* filename); | ||||
| bool saveJsonValue(const char* filename, const JsonDocument& doc); | ||||
| bool loadJsonValue(const char* filename, JsonDocument& doc); | ||||
| bool initializeSPIFFS(); | ||||
| void initializeFileSystem(); | ||||
|  | ||||
| #endif | ||||
|   | ||||
| @@ -16,20 +16,20 @@ const uint8_t LOADCELL_DOUT_PIN = 16; //16; | ||||
| const uint8_t LOADCELL_SCK_PIN = 17; //17; | ||||
| const uint8_t calVal_eepromAdress = 0; | ||||
| const uint16_t SCALE_LEVEL_WEIGHT = 500; | ||||
| uint16_t defaultScaleCalibrationValue = 430; | ||||
| // ***** HX711 | ||||
|  | ||||
| // ***** TTP223 (Touch Sensor) | ||||
| // TTP223 circuit wiring | ||||
| const uint8_t TTP223_PIN = 25; | ||||
| // ***** TTP223 | ||||
|  | ||||
|  | ||||
| // ***** 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_END = 16; | ||||
| const uint8_t OLED_DATA_START = 17; | ||||
| const uint8_t OLED_DATA_END = SCREEN_HEIGHT; | ||||
|  | ||||
| // ***** Display | ||||
|  | ||||
| // ***** Webserver | ||||
| @@ -40,6 +40,8 @@ const uint8_t webserverPort = 80; | ||||
| const char* apiUrl = "/api/v1"; | ||||
| // ***** API | ||||
|  | ||||
| // ***** Bambu Auto Set Spool | ||||
|  | ||||
| // ***** Task Prios | ||||
| uint8_t rfidTaskCore = 1; | ||||
| uint8_t rfidTaskPrio = 1; | ||||
|   | ||||
							
								
								
									
										42
									
								
								src/config.h
									
									
									
									
									
								
							
							
						
						| @@ -3,6 +3,40 @@ | ||||
|  | ||||
| #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_RESET; | ||||
|  | ||||
| @@ -11,10 +45,8 @@ extern const uint8_t LOADCELL_SCK_PIN; | ||||
| extern const uint8_t calVal_eepromAdress; | ||||
| extern const uint16_t SCALE_LEVEL_WEIGHT; | ||||
|  | ||||
| extern const int8_t OLED_RESET; | ||||
| extern const uint8_t SCREEN_ADDRESS; | ||||
| extern const uint8_t SCREEN_WIDTH; | ||||
| extern const uint8_t SCREEN_HEIGHT; | ||||
| extern const uint8_t TTP223_PIN; | ||||
|  | ||||
| extern const uint8_t OLED_TOP_START; | ||||
| extern const uint8_t OLED_TOP_END; | ||||
| extern const uint8_t OLED_DATA_START; | ||||
| @@ -23,6 +55,8 @@ extern const uint8_t OLED_DATA_END; | ||||
| extern const char* apiUrl; | ||||
| extern const uint8_t webserverPort; | ||||
|  | ||||
|  | ||||
|  | ||||
| extern const unsigned char wifi_on[]; | ||||
| extern const unsigned char wifi_off[]; | ||||
| extern const unsigned char cloud_on[]; | ||||
|   | ||||
							
								
								
									
										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 <vector> | ||||
| #include "icons.h" | ||||
| #include "main.h" | ||||
|  | ||||
| Adafruit_SSD1306 display(SCREEN_WIDTH, SCREEN_HEIGHT, &Wire, OLED_RESET); | ||||
|  | ||||
| bool wifiOn = false; | ||||
| bool iconToggle = false; | ||||
|  | ||||
| void setupDisplay() { | ||||
|     if (!display.begin(SSD1306_SWITCHCAPVCC, SCREEN_ADDRESS)) { | ||||
| @@ -14,15 +16,10 @@ void setupDisplay() { | ||||
|     } | ||||
|     display.setTextColor(WHITE); | ||||
|     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(); | ||||
|     delay(1000); // Pause for 2 seconds | ||||
|  | ||||
|     oledShowTopRow(); | ||||
|     delay(2000); | ||||
|     oledShowProgressBar(0, 7, DISPLAY_BOOT_TEXT, "Display init"); | ||||
| } | ||||
|  | ||||
| void oledclearline() { | ||||
| @@ -45,14 +42,14 @@ void oledcleardata() { | ||||
|     //display.display(); | ||||
| } | ||||
|  | ||||
| int oled_center_h(String text) { | ||||
| int oled_center_h(const String &text) { | ||||
|     int16_t x1, y1; | ||||
|     uint16_t w, h; | ||||
|     display.getTextBounds(text, 0, 0, &x1, &y1, &w, &h); | ||||
|     return (SCREEN_WIDTH - w) / 2; | ||||
| } | ||||
|  | ||||
| int oled_center_v(String text) { | ||||
| int oled_center_v(const String &text) { | ||||
|     int16_t x1, y1; | ||||
|     uint16_t 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); | ||||
| } | ||||
|  | ||||
| std::vector<String> splitTextIntoLines(String text, uint8_t textSize) { | ||||
| std::vector<String> splitTextIntoLines(const String &text, uint8_t textSize) { | ||||
|     std::vector<String> lines; | ||||
|     display.setTextSize(textSize); | ||||
|      | ||||
| @@ -117,11 +114,10 @@ std::vector<String> splitTextIntoLines(String text, uint8_t textSize) { | ||||
|         lines.push_back(currentLine); | ||||
|     } | ||||
|      | ||||
|     Serial.println(lines.size()); | ||||
|     return lines; | ||||
| } | ||||
|  | ||||
| void oledShowMultilineMessage(String message, uint8_t size) { | ||||
| void oledShowMultilineMessage(const String &message, uint8_t size) { | ||||
|     std::vector<String> lines; | ||||
|     int maxLines = 3;  // Maximale Anzahl Zeilen für size 2 | ||||
|      | ||||
| @@ -140,15 +136,16 @@ void oledShowMultilineMessage(String message, uint8_t size) { | ||||
|     int totalHeight = lines.size() * lineHeight; | ||||
|     int startY = OLED_DATA_START + ((OLED_DATA_END - OLED_DATA_START - totalHeight) / 2); | ||||
|      | ||||
|     uint8_t lineDistance = (lines.size() == 2) ? 5 : 0; | ||||
|     for (size_t i = 0; i < lines.size(); i++) { | ||||
|         display.setCursor(oled_center_h(lines[i]), startY + (i * lineHeight)); | ||||
|         display.setCursor(oled_center_h(lines[i]), startY + (i * lineHeight) + (i == 1 ? lineDistance : 0)); | ||||
|         display.print(lines[i]); | ||||
|     } | ||||
|      | ||||
|     display.display(); | ||||
| } | ||||
|  | ||||
| void oledShowMessage(String message, uint8_t size) { | ||||
| void oledShowMessage(const String &message, uint8_t size) { | ||||
|     oledcleardata(); | ||||
|     display.setTextSize(size); | ||||
|     display.setTextWrap(false); | ||||
| @@ -171,22 +168,46 @@ void oledShowMessage(String message, uint8_t size) { | ||||
| void oledShowTopRow() { | ||||
|     oledclearline(); | ||||
|  | ||||
|     if (bambu_connected == 1) { | ||||
|         display.drawBitmap(50, 0, bitmap_bambu_on , 16, 16, WHITE); | ||||
|     } else { | ||||
|         display.drawBitmap(50, 0, bitmap_off , 16, 16, WHITE); | ||||
|     } | ||||
|     display.setTextSize(1); | ||||
|     display.setCursor(0, 4); | ||||
|     display.print("v"); | ||||
|     display.print(VERSION); | ||||
|  | ||||
|     if (spoolman_connected == 1) { | ||||
|         display.drawBitmap(80, 0, bitmap_spoolman_on , 16, 16, WHITE); | ||||
|     } else { | ||||
|         display.drawBitmap(80, 0, bitmap_off , 16, 16, WHITE); | ||||
|     } | ||||
|     iconToggle = !iconToggle; | ||||
|  | ||||
|     if (wifiOn == 1) { | ||||
|         display.drawBitmap(107, 0, wifi_on , 16, 16, WHITE); | ||||
|     } else { | ||||
|         display.drawBitmap(107, 0, wifi_off , 16, 16, WHITE); | ||||
|     // Do not show status indicators during boot | ||||
|     if(!booting){ | ||||
|         if(bambuDisabled == false) { | ||||
|             if (bambu_connected == 1) { | ||||
|                 display.drawBitmap(50, 0, bitmap_bambu_on , 16, 16, WHITE); | ||||
|             } else { | ||||
|                 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 (spoolmanConnected) { | ||||
|             display.drawBitmap(80, 0, bitmap_spoolman_on , 16, 16, WHITE); | ||||
|         } else { | ||||
|             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) { | ||||
|             display.drawBitmap(107, 0, wifi_on , 16, 16, WHITE); | ||||
|         } else { | ||||
|             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(); | ||||
| @@ -214,6 +235,27 @@ void oledShowIcon(const char* icon) { | ||||
|     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) { | ||||
|     // Display Gewicht | ||||
|     oledcleardata(); | ||||
|   | ||||
| @@ -13,11 +13,13 @@ extern bool wifiOn; | ||||
| void setupDisplay(); | ||||
| void oledclearline(); | ||||
| void oledcleardata(); | ||||
| int oled_center_h(String text); | ||||
| int oled_center_v(String text); | ||||
| int oled_center_h(const 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 oledShowMessage(String message, uint8_t size = 2); | ||||
| void oledShowMessage(const String &message, uint8_t size = 2); | ||||
| void oledShowTopRow(); | ||||
| void oledShowIcon(const char* icon); | ||||
|  | ||||
|   | ||||
							
								
								
									
										311
									
								
								src/main.cpp
									
									
									
									
									
								
							
							
						
						| @@ -1,10 +1,8 @@ | ||||
| #include <Arduino.h> | ||||
| #include <WiFi.h> | ||||
| #include <DNSServer.h> | ||||
| #include <WiFiManager.h> | ||||
| #include <ESPmDNS.h> | ||||
| #include <Wire.h> | ||||
| #include <WiFi.h> | ||||
|  | ||||
| #include "wlan.h" | ||||
| #include "config.h" | ||||
| #include "website.h" | ||||
| #include "api.h" | ||||
| @@ -15,18 +13,23 @@ | ||||
| #include "esp_task_wdt.h" | ||||
| #include "commonFS.h" | ||||
|  | ||||
| // ***** WIFI initialisieren | ||||
| WiFiManager wm; | ||||
| bool wm_nonblocking = false; | ||||
| void initWiFi(); | ||||
| // ################### Functions | ||||
| bool mainTaskWasPaused = 0; | ||||
| uint8_t scaleTareCounter = 0; | ||||
| bool touchSensorConnected = false; | ||||
| bool booting = true; | ||||
|  | ||||
| // ##### SETUP ##### | ||||
| void setup() { | ||||
|   Serial.begin(115200); | ||||
|  | ||||
|   uint64_t chipid; | ||||
|  | ||||
|   chipid = ESP.getEfuseMac(); //The chip ID is essentially its MAC address(length: 6 bytes). | ||||
|   Serial.printf("ESP32 Chip ID = %04X", (uint16_t)(chipid >> 32)); //print High 2 bytes | ||||
|   Serial.printf("%08X\n", (uint32_t)chipid); //print Low 4bytes. | ||||
|  | ||||
|   // Initialize SPIFFS | ||||
|   initializeSPIFFS(); | ||||
|   initializeFileSystem(); | ||||
|  | ||||
|   // Start Display | ||||
|   setupDisplay(); | ||||
| @@ -35,157 +38,247 @@ void setup() { | ||||
|   initWiFi(); | ||||
|  | ||||
|   // Webserver | ||||
|   Serial.println("Starte Webserver"); | ||||
|   setupWebserver(server); | ||||
|  | ||||
|   // Spoolman API | ||||
|   // api.cpp | ||||
|   initSpoolman(); | ||||
|  | ||||
|   // Bambu MQTT | ||||
|   // bambu.cpp | ||||
|   setupMqtt(); | ||||
|  | ||||
|   // mDNS | ||||
|   Serial.println("Starte MDNS"); | ||||
|   if (!MDNS.begin("filaman")) {   // Set the hostname to "esp32.local" | ||||
|     Serial.println("Error setting up MDNS responder!"); | ||||
|     while(1) { | ||||
|       delay(1000); | ||||
|     } | ||||
|   } | ||||
|   Serial.println("mDNS responder started"); | ||||
|    | ||||
|   // NFC Reader | ||||
|   startNfc(); | ||||
|  | ||||
|   start_scale(); | ||||
|   // Touch Sensor | ||||
|   pinMode(TTP223_PIN, INPUT_PULLUP); | ||||
|   if (digitalRead(TTP223_PIN) == LOW)  | ||||
|   { | ||||
|     Serial.println("Touch Sensor is connected"); | ||||
|     touchSensorConnected = true; | ||||
|   } | ||||
|  | ||||
|   // Scale | ||||
|   start_scale(touchSensorConnected); | ||||
|  | ||||
|   // WDT initialisieren mit 10 Sekunden Timeout | ||||
|   bool panic = true; // Wenn true, löst ein WDT-Timeout einen System-Panik aus | ||||
|   esp_task_wdt_init(10, panic); | ||||
|  | ||||
|   booting = false; | ||||
|   // Aktuellen Task (loopTask) zum Watchdog hinzufügen | ||||
|   esp_task_wdt_add(NULL); | ||||
|  | ||||
|   // Optional: Andere Tasks zum Watchdog hinzufügen, falls nötig | ||||
|   // esp_task_wdt_add(task_handle); | ||||
| } | ||||
|  | ||||
|  | ||||
| /** | ||||
|  * Safe interval check that handles millis() overflow | ||||
|  * @param currentTime Current millis() value | ||||
|  * @param lastTime Last recorded time | ||||
|  * @param interval Desired interval in milliseconds | ||||
|  * @return True if interval has elapsed | ||||
|  */ | ||||
| bool intervalElapsed(unsigned long currentTime, unsigned long &lastTime, unsigned long interval) { | ||||
|   if (currentTime - lastTime >= interval || currentTime < lastTime) { | ||||
|     lastTime = currentTime; | ||||
|     return true; | ||||
|   } | ||||
|   return false; | ||||
| } | ||||
|  | ||||
| unsigned long lastWeightReadTime = 0; | ||||
| const unsigned long weightReadInterval = 1000; // 1 second | ||||
|  | ||||
| unsigned long lastAutoSetBambuAmsTime = 0; | ||||
| const unsigned long autoSetBambuAmsInterval = 1000; // 1 second | ||||
| uint8_t autoAmsCounter = 0; | ||||
|  | ||||
| uint8_t weightSend = 0; | ||||
| int16_t lastWeight = 0; | ||||
| uint8_t wifiErrorCounter = 0; | ||||
|  | ||||
| // WIFI check variables | ||||
| unsigned long lastWifiCheckTime = 0; | ||||
| 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 ##### | ||||
| void loop() { | ||||
|   // Überprüfe den WLAN-Status | ||||
|   if (WiFi.status() != WL_CONNECTED) { | ||||
|     wifiErrorCounter++; | ||||
|     wifiOn = false; | ||||
|   } else { | ||||
|     wifiErrorCounter = 0; | ||||
|     wifiOn = true; | ||||
|   } | ||||
|   if (wifiErrorCounter > 20) ESP.restart(); | ||||
|  | ||||
|   unsigned long currentMillis = millis(); | ||||
|  | ||||
|   // Falls WifiManager im nicht blockenden Modus ist | ||||
|   //if(wm_nonblocking) wm.process(); | ||||
|  | ||||
|   // Ausgabe der Waage auf Display | ||||
|   if (pauseMainTask == 0 && weight != lastWeight && hasReadRfidTag == 0) | ||||
|   // Überprüfe den Status des Touch Sensors | ||||
|   if (touchSensorConnected && digitalRead(TTP223_PIN) == HIGH && currentMillis - lastButtonPress > debounceDelay)  | ||||
|   { | ||||
|     (weight < 0) ? oledShowMessage("!! -1") : oledShowWeight(weight); | ||||
|     lastButtonPress = currentMillis; | ||||
|     scaleTareRequest = true; | ||||
|   } | ||||
|  | ||||
|   // Wenn Timer abgelaufen und nicht gerade ein RFID-Tag geschrieben wird | ||||
|   if (currentMillis - lastWeightReadTime >= weightReadInterval && hasReadRfidTag < 3) | ||||
|   // Überprüfe regelmäßig die WLAN-Verbindung | ||||
|   if (intervalElapsed(currentMillis, lastWifiCheckTime, WIFI_CHECK_INTERVAL))  | ||||
|   { | ||||
|     lastWeightReadTime = currentMillis; | ||||
|     checkWiFiConnection(); | ||||
|   } | ||||
|  | ||||
|     // Prüfen ob die Waage korrekt genullt ist | ||||
|     if ((weight > 0 && weight < 5) || weight < 0) | ||||
|   // 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 | ||||
|   if (bambuCredentials.autosend_enable && autoSetToBambuSpoolId > 0)  | ||||
|   { | ||||
|     if (!bambuDisabled && !bambu_connected)  | ||||
|     { | ||||
|       scale_tare_counter++; | ||||
|       bambu_restart(); | ||||
|     } | ||||
|  | ||||
|     if (intervalElapsed(currentMillis, lastAutoSetBambuAmsTime, autoSetBambuAmsInterval))  | ||||
|     { | ||||
|       if (nfcReaderState == NFC_IDLE) | ||||
|       { | ||||
|         lastAutoSetBambuAmsTime = currentMillis; | ||||
|         oledShowMessage("Auto Set         " + String(bambuCredentials.autosend_time - autoAmsCounter) + "s"); | ||||
|         autoAmsCounter++; | ||||
|  | ||||
|         if (autoAmsCounter >= bambuCredentials.autosend_time)  | ||||
|         { | ||||
|           autoSetToBambuSpoolId = 0; | ||||
|           autoAmsCounter = 0; | ||||
|           oledShowWeight(weight); | ||||
|         } | ||||
|       } | ||||
|       else | ||||
|       { | ||||
|         autoAmsCounter = 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // If scale is not calibrated, only show a warning | ||||
|   if (!scaleCalibrated)  | ||||
|   { | ||||
|     // Do not show the warning if the calibratin process is onging | ||||
|     if(!scaleCalibrationActive){ | ||||
|       oledShowMessage("Scale not calibrated"); | ||||
|       vTaskDelay(1000 / portTICK_PERIOD_MS); | ||||
|     } | ||||
|   }  | ||||
|   else  | ||||
|   { | ||||
|     // Ausgabe der Waage auf Display | ||||
|     if(pauseMainTask == 0) | ||||
|     { | ||||
|       // Use filtered weight for smooth display, but still check API weight for significant changes | ||||
|       int16_t displayWeight = getFilteredDisplayWeight(); | ||||
|       if (mainTaskWasPaused || (weight != lastWeight && nfcReaderState == NFC_IDLE && (!bambuCredentials.autosend_enable || autoSetToBambuSpoolId == 0))) | ||||
|       { | ||||
|         (displayWeight < 2) ? ((displayWeight < -2) ? oledShowMessage("!! -0") : oledShowWeight(0)) : oledShowWeight(displayWeight); | ||||
|       } | ||||
|       mainTaskWasPaused = false; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|       scale_tare_counter = 0; | ||||
|       mainTaskWasPaused = true; | ||||
|     } | ||||
|  | ||||
|     // Prüfen ob das Gewicht gleich bleibt und dann senden | ||||
|     if (weight == lastWeight && weight > 5) | ||||
|  | ||||
|     // Wenn Timer abgelaufen und nicht gerade ein RFID-Tag geschrieben wird | ||||
|     if (currentMillis - lastWeightReadTime >= weightReadInterval && nfcReaderState < NFC_WRITING) | ||||
|     { | ||||
|       weigthCouterToApi++; | ||||
|       lastWeightReadTime = currentMillis; | ||||
|  | ||||
|       // Prüfen ob die Waage korrekt genullt ist | ||||
|       // Abweichung von 2g ignorieren | ||||
|       if (autoTare && (weight > 2 && weight < 7) || weight < -2) | ||||
|       { | ||||
|         scale_tare_counter++; | ||||
|       } | ||||
|       else | ||||
|       { | ||||
|         scale_tare_counter = 0; | ||||
|       } | ||||
|  | ||||
|       // Prüfen ob das Gewicht gleich bleibt und dann senden | ||||
|       if (abs(weight - lastWeight) <= 2 && weight > 5) | ||||
|       { | ||||
|         weigthCouterToApi++; | ||||
|       }  | ||||
|       else  | ||||
|       { | ||||
|         weigthCouterToApi = 0; | ||||
|         weightSend = 0; | ||||
|       } | ||||
|     } | ||||
|     else  | ||||
|  | ||||
|     // reset weight counter after writing tag | ||||
|     if (currentMillis - lastWeightReadTime >= weightReadInterval && nfcReaderState != NFC_IDLE && nfcReaderState != NFC_READ_SUCCESS) | ||||
|     { | ||||
|       weigthCouterToApi = 0; | ||||
|       weightSend = 0; | ||||
|     } | ||||
|   } | ||||
|   // reset weight counter after writing tag | ||||
|   if (currentMillis - lastWeightReadTime >= weightReadInterval && hasReadRfidTag > 1) | ||||
|   { | ||||
|     weigthCouterToApi = 0; | ||||
|   } | ||||
|      | ||||
|   lastWeight = weight; | ||||
|     lastWeight = weight; | ||||
|  | ||||
|   // Wenn ein Tag mit SM id erkannte wurde und der Waage Counter anspricht an SM Senden | ||||
|   if (spoolId != "" && weigthCouterToApi > 5 && weightSend == 0 && hasReadRfidTag == 1) { | ||||
|     oledShowIcon("loading"); | ||||
|     if (updateSpoolWeight(spoolId, weight))  | ||||
|     // Wenn ein Tag mit SM id erkannte wurde und der Waage Counter anspricht an SM Senden | ||||
|     if (activeSpoolId != "" && weigthCouterToApi > 3 && weightSend == 0 && nfcReaderState == NFC_READ_SUCCESS && tagProcessed == false && spoolmanApiState == API_IDLE)  | ||||
|     { | ||||
|       oledShowIcon("success"); | ||||
|       vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||
|       weightSend = 1; | ||||
|       // set the current tag as processed to prevent it beeing processed again | ||||
|       tagProcessed = true; | ||||
|  | ||||
|       if (updateSpoolWeight(activeSpoolId, weight))  | ||||
|       { | ||||
|         weightSend = 1; | ||||
|          | ||||
|         // Set Bambu spool ID for auto-send if enabled | ||||
|         if (bambuCredentials.autosend_enable)  | ||||
|         { | ||||
|           autoSetToBambuSpoolId = activeSpoolId.toInt(); | ||||
|         } | ||||
|         if (octoEnabled)  | ||||
|         { | ||||
|           updateOctoSpoolId = activeSpoolId.toInt(); | ||||
|         } | ||||
|       } | ||||
|       else | ||||
|       { | ||||
|         oledShowIcon("failed"); | ||||
|         vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||
|       } | ||||
|     } | ||||
|     else | ||||
|  | ||||
|     // Handle successful tag write: Send weight to Spoolman but NEVER auto-send to Bambu | ||||
|     if (activeSpoolId != "" && weigthCouterToApi > 3 && weightSend == 0 && nfcReaderState == NFC_WRITE_SUCCESS && tagProcessed == false && spoolmanApiState == API_IDLE)  | ||||
|     { | ||||
|       oledShowIcon("failed"); | ||||
|       vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||
|       // set the current tag as processed to prevent it beeing processed again | ||||
|       tagProcessed = true; | ||||
|  | ||||
|       if (updateSpoolWeight(activeSpoolId, weight))  | ||||
|       { | ||||
|         weightSend = 1; | ||||
|         Serial.println("Tag written: Weight sent to Spoolman, but NO auto-send to Bambu"); | ||||
|         // INTENTIONALLY do NOT set autoSetToBambuSpoolId here to prevent Bambu auto-send | ||||
|       } | ||||
|       else | ||||
|       { | ||||
|         oledShowIcon("failed"); | ||||
|         vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if(octoEnabled && sendOctoUpdate && spoolmanApiState == API_IDLE) | ||||
|     { | ||||
|       updateSpoolOcto(updateOctoSpoolId); | ||||
|       sendOctoUpdate = false; | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   yield(); | ||||
|   esp_task_wdt_reset(); | ||||
| } | ||||
|  | ||||
| // ##### Funktionen für Konfiguration ##### | ||||
| void initWiFi() { | ||||
|   WiFi.mode(WIFI_STA); // explicitly set mode, esp defaults to STA+AP | ||||
|  | ||||
|   if(wm_nonblocking) wm.setConfigPortalBlocking(false); | ||||
|   wm.setConfigPortalTimeout(320); // Portal nach 5min schließen | ||||
|  | ||||
|   oledShowTopRow(); | ||||
|   oledShowMessage("WiFi Setup"); | ||||
|    | ||||
|   bool res; | ||||
|   // res = wm.autoConnect(); // auto generated AP name from chipid | ||||
|   res = wm.autoConnect("FilaMan"); // anonymous ap | ||||
|   // res = wm.autoConnect("spoolman","password"); // password protected ap | ||||
|  | ||||
|   if(!res) { | ||||
|     Serial.println("Failed to connect or hit timeout"); | ||||
|     // ESP.restart(); | ||||
|     oledShowTopRow(); | ||||
|     oledShowMessage("WiFi not connected Check Portal"); | ||||
|   }  | ||||
|   else { | ||||
|     wifiOn = true; | ||||
|  | ||||
|     //if you get here you have connected to the WiFi     | ||||
|     Serial.println("connected...yeey :)"); | ||||
|     Serial.print("IP address: "); | ||||
|     Serial.println(WiFi.localIP()); | ||||
|  | ||||
|     oledShowTopRow(); | ||||
|     display.display(); | ||||
|   } | ||||
| } | ||||
| // ##### Funktionen für Konfiguration Ende ##### | ||||
|   | ||||
							
								
								
									
										9
									
								
								src/main.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | ||||
| #ifndef MAIN_H | ||||
| #define MAIN_H | ||||
|  | ||||
| #include <Arduino.h> | ||||
|  | ||||
|  | ||||
| extern bool booting; | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										2052
									
								
								src/nfc.cpp
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										23
									
								
								src/nfc.h
									
									
									
									
									
								
							
							
						
						| @@ -3,14 +3,31 @@ | ||||
|  | ||||
| #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 scanRfidTask(void * parameter); | ||||
| void startWriteJsonToTag(const char* payload); | ||||
| void startWriteJsonToTag(const bool isSpoolTag, const char* payload); | ||||
| bool quickSpoolIdCheck(String uidString); | ||||
| bool readCompleteJsonForFastPath(); // Read complete JSON data for fast-path web interface display | ||||
|  | ||||
| extern TaskHandle_t RfidReaderTask; | ||||
| extern String nfcJsonData; | ||||
| extern String spoolId; | ||||
| extern volatile uint8_t hasReadRfidTag; | ||||
| extern String activeSpoolId; | ||||
| extern String lastSpoolId; | ||||
| extern volatile nfcReaderStateType nfcReaderState; | ||||
| extern volatile bool pauseBambuMqttTask; | ||||
| extern volatile bool nfcWriteInProgress; | ||||
| extern bool tagProcessed; | ||||
|  | ||||
|  | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										266
									
								
								src/ota.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,266 @@ | ||||
| #include <Arduino.h> | ||||
| #include <website.h> | ||||
| #include <commonFS.h> | ||||
| #include "scale.h" | ||||
| #include "bambu.h" | ||||
| #include "nfc.h" | ||||
|  | ||||
|  | ||||
| // Globale Variablen für Config Backups hinzufügen | ||||
| String bambuCredentialsBackup; | ||||
| String spoolmanUrlBackup; | ||||
|  | ||||
| // Globale Variable für den Update-Typ | ||||
| static int currentUpdateCommand = 0; | ||||
|  | ||||
| // Globale Update-Variablen | ||||
| static size_t updateTotalSize = 0; | ||||
| static size_t updateWritten = 0; | ||||
| static bool isSpiffsUpdate = false; | ||||
|  | ||||
| /** | ||||
|  * Compares two version strings and determines if version1 is less than version2 | ||||
|  *  | ||||
|  * @param version1 First version string (format: x.y.z) | ||||
|  * @param version2 Second version string (format: x.y.z) | ||||
|  * @return true if version1 is less than version2 | ||||
|  */ | ||||
| bool isVersionLessThan(const String& version1, const String& version2) { | ||||
|     int major1 = 0, minor1 = 0, patch1 = 0; | ||||
|     int major2 = 0, minor2 = 0, patch2 = 0; | ||||
|      | ||||
|     // Parse version1 | ||||
|     sscanf(version1.c_str(), "%d.%d.%d", &major1, &minor1, &patch1); | ||||
|      | ||||
|     // Parse version2 | ||||
|     sscanf(version2.c_str(), "%d.%d.%d", &major2, &minor2, &patch2); | ||||
|      | ||||
|     // Compare major version | ||||
|     if (major1 < major2) return true; | ||||
|     if (major1 > major2) return false; | ||||
|      | ||||
|     // Major versions equal, compare minor | ||||
|     if (minor1 < minor2) return true; | ||||
|     if (minor1 > minor2) return false; | ||||
|      | ||||
|     // Minor versions equal, compare patch | ||||
|     return patch1 < patch2; | ||||
| } | ||||
|  | ||||
| void backupJsonConfigs() { | ||||
|     // Bambu Credentials backup | ||||
|     if (LittleFS.exists("/bambu_credentials.json")) { | ||||
|         File file = LittleFS.open("/bambu_credentials.json", "r"); | ||||
|         if (file) { | ||||
|             bambuCredentialsBackup = file.readString(); | ||||
|             file.close(); | ||||
|             Serial.println("Bambu credentials backed up"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Spoolman URL backup | ||||
|     if (LittleFS.exists("/spoolman_url.json")) { | ||||
|         File file = LittleFS.open("/spoolman_url.json", "r"); | ||||
|         if (file) { | ||||
|             spoolmanUrlBackup = file.readString(); | ||||
|             file.close(); | ||||
|             Serial.println("Spoolman URL backed up"); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| void restoreJsonConfigs() { | ||||
|     // Restore Bambu credentials | ||||
|     if (bambuCredentialsBackup.length() > 0) { | ||||
|         File file = LittleFS.open("/bambu_credentials.json", "w"); | ||||
|         if (file) { | ||||
|             file.print(bambuCredentialsBackup); | ||||
|             file.close(); | ||||
|             Serial.println("Bambu credentials restored"); | ||||
|         } | ||||
|         bambuCredentialsBackup = ""; // Clear backup | ||||
|     } | ||||
|  | ||||
|     // Restore Spoolman URL | ||||
|     if (spoolmanUrlBackup.length() > 0) { | ||||
|         File file = LittleFS.open("/spoolman_url.json", "w"); | ||||
|         if (file) { | ||||
|             file.print(spoolmanUrlBackup); | ||||
|             file.close(); | ||||
|             Serial.println("Spoolman URL restored"); | ||||
|         } | ||||
|         spoolmanUrlBackup = ""; // Clear backup | ||||
|     } | ||||
| } | ||||
|  | ||||
| void espRestart() { | ||||
|     yield(); | ||||
|     vTaskDelay(5000 / portTICK_PERIOD_MS); | ||||
|  | ||||
|     ESP.restart(); | ||||
| } | ||||
|  | ||||
|  | ||||
| void sendUpdateProgress(int progress, const char* status = nullptr, const char* message = nullptr) { | ||||
|     static int lastSentProgress = -1; | ||||
|      | ||||
|     // Verhindere zu häufige Updates | ||||
|     if (progress == lastSentProgress && !status && !message) { | ||||
|         return; | ||||
|     } | ||||
|      | ||||
|     String progressMsg = "{\"type\":\"updateProgress\",\"progress\":" + String(progress); | ||||
|     if (status) { | ||||
|         progressMsg += ",\"status\":\"" + String(status) + "\""; | ||||
|     } | ||||
|     if (message) { | ||||
|         progressMsg += ",\"message\":\"" + String(message) + "\""; | ||||
|     } | ||||
|     progressMsg += "}"; | ||||
|      | ||||
|     if (progress >= 100) { | ||||
|         // Sende die Nachricht nur einmal für den Abschluss | ||||
|         ws.textAll("{\"type\":\"updateProgress\",\"progress\":100,\"status\":\"success\",\"message\":\"Update successful! Restarting device...\"}"); | ||||
|         delay(50); | ||||
|     } | ||||
|  | ||||
|     // Sende die Nachricht mehrmals mit Verzögerung für wichtige Updates | ||||
|     if (status || abs(progress - lastSentProgress) >= 10 || progress == 100) { | ||||
|         for (int i = 0; i < 2; i++) { | ||||
|             ws.textAll(progressMsg); | ||||
|             delay(100);  // Längerer Delay zwischen Nachrichten | ||||
|         } | ||||
|     } else { | ||||
|         ws.textAll(progressMsg); | ||||
|         delay(50); | ||||
|     } | ||||
|      | ||||
|     lastSentProgress = progress; | ||||
| } | ||||
|  | ||||
| void handleUpdate(AsyncWebServer &server) { | ||||
|     AsyncCallbackWebHandler* updateHandler = new AsyncCallbackWebHandler(); | ||||
|     updateHandler->setUri("/update"); | ||||
|     updateHandler->setMethod(HTTP_POST); | ||||
|      | ||||
|     // Check if current version is less than defined TOOLVERSION before proceeding with update | ||||
|     if (isVersionLessThan(VERSION, TOOLDVERSION)) { | ||||
|         updateHandler->onRequest([](AsyncWebServerRequest *request) { | ||||
|             request->send(400, "application/json",  | ||||
|                 "{\"success\":false,\"message\":\"Your current version is too old. Please perform a full upgrade.\"}"); | ||||
|         }); | ||||
|         server.addHandler(updateHandler); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     updateHandler->onUpload([](AsyncWebServerRequest *request, String filename, | ||||
|                              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) { | ||||
|             updateTotalSize = request->contentLength(); | ||||
|             updateWritten = 0; | ||||
|             isSpiffsUpdate = (filename.indexOf("website") > -1); | ||||
|              | ||||
|             if (isSpiffsUpdate) { | ||||
|                 // Backup vor dem Update | ||||
|                 sendUpdateProgress(0, "backup", "Backing up configurations..."); | ||||
|                 vTaskDelay(200 / portTICK_PERIOD_MS); | ||||
|                 backupJsonConfigs(); | ||||
|                 vTaskDelay(200 / portTICK_PERIOD_MS); | ||||
|                  | ||||
|                 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)) { | ||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"Update initialization failed\"}"); | ||||
|                     return; | ||||
|                 } | ||||
|                 sendUpdateProgress(5, "starting", "Starting SPIFFS update..."); | ||||
|                 vTaskDelay(200 / portTICK_PERIOD_MS); | ||||
|             } else { | ||||
|                 if (!Update.begin(updateTotalSize)) { | ||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"Update initialization failed\"}"); | ||||
|                     return; | ||||
|                 } | ||||
|                 sendUpdateProgress(0, "starting", "Starting firmware update..."); | ||||
|                 vTaskDelay(200 / portTICK_PERIOD_MS); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (len) { | ||||
|             if (Update.write(data, len) != len) { | ||||
|                 request->send(400, "application/json", "{\"success\":false,\"message\":\"Write failed\"}"); | ||||
|                 return; | ||||
|             } | ||||
|              | ||||
|             updateWritten += len; | ||||
|             int currentProgress; | ||||
|              | ||||
|             // Berechne den Fortschritt basierend auf dem Update-Typ | ||||
|             if (isSpiffsUpdate) { | ||||
|                 // SPIFFS: 5-75% für Upload | ||||
|                 currentProgress = 6 + (updateWritten * 100) / updateTotalSize; | ||||
|             } else { | ||||
|                 // Firmware: 0-100% für Upload | ||||
|                 currentProgress = 1 + (updateWritten * 100) / updateTotalSize; | ||||
|             } | ||||
|              | ||||
|             static int lastProgress = -1; | ||||
|             if (currentProgress != lastProgress && (currentProgress % 10 == 0 || final)) { | ||||
|                 sendUpdateProgress(currentProgress, "uploading"); | ||||
|                 oledShowProgressBar(currentProgress, 100, "Update", "Download"); | ||||
|                 vTaskDelay(50 / portTICK_PERIOD_MS); | ||||
|                 lastProgress = currentProgress; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (final) { | ||||
|             if (Update.end(true)) { | ||||
|                 if (isSpiffsUpdate) { | ||||
|                     restoreJsonConfigs(); | ||||
|                 } | ||||
|             } else { | ||||
|                 request->send(400, "application/json", "{\"success\":false,\"message\":\"Update finalization failed\"}"); | ||||
|             } | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     updateHandler->onRequest([](AsyncWebServerRequest *request) { | ||||
|         if (Update.hasError()) { | ||||
|             request->send(400, "application/json", "{\"success\":false,\"message\":\"Update failed\"}"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Erste 100% Nachricht | ||||
|         ws.textAll("{\"type\":\"updateProgress\",\"progress\":100,\"status\":\"success\",\"message\":\"Update successful! Restarting device...\"}"); | ||||
|         vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||
|          | ||||
|         AsyncWebServerResponse *response = request->beginResponse(200, "application/json",  | ||||
|             "{\"success\":true,\"message\":\"Update successful! Restarting device...\"}"); | ||||
|         response->addHeader("Connection", "close"); | ||||
|         request->send(response); | ||||
|          | ||||
|         // Zweite 100% Nachricht zur Sicherheit | ||||
|         ws.textAll("{\"type\":\"updateProgress\",\"progress\":100,\"status\":\"success\",\"message\":\"Update successful! Restarting device...\"}"); | ||||
|          | ||||
|         espRestart(); | ||||
|     }); | ||||
|  | ||||
|     server.addHandler(updateHandler); | ||||
| } | ||||
							
								
								
									
										9
									
								
								src/ota.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | ||||
| #ifndef OTA_H | ||||
| #define OTA_H | ||||
|  | ||||
| #include <ArduinoOTA.h> | ||||
| #include <ESPAsyncWebServer.h> | ||||
|  | ||||
| void handleUpdate(AsyncWebServer &server); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										327
									
								
								src/scale.cpp
									
									
									
									
									
								
							
							
						
						| @@ -3,10 +3,9 @@ | ||||
| #include <ArduinoJson.h> | ||||
| #include "config.h" | ||||
| #include "HX711.h" | ||||
| #include <EEPROM.h> | ||||
| #include "display.h" | ||||
| #include "nfc.h" | ||||
| #include "esp_task_wdt.h" | ||||
| #include <Preferences.h> | ||||
|  | ||||
| HX711 scale; | ||||
|  | ||||
| @@ -14,14 +13,135 @@ TaskHandle_t ScaleTask; | ||||
|  | ||||
| int16_t weight = 0; | ||||
|  | ||||
| // Weight stabilization variables | ||||
| #define MOVING_AVERAGE_SIZE 8           // Reduced from 20 to 8 for faster response | ||||
| #define LOW_PASS_ALPHA 0.3f            // Increased from 0.15 to 0.3 for faster tracking | ||||
| #define DISPLAY_THRESHOLD 0.3f         // Reduced from 0.5 to 0.3g for more responsive display | ||||
| #define API_THRESHOLD 1.5f             // Reduced from 2.0 to 1.5g for faster API actions | ||||
| #define MEASUREMENT_INTERVAL_MS 30     // Reduced from 50ms to 30ms for faster updates | ||||
|  | ||||
| float weightBuffer[MOVING_AVERAGE_SIZE]; | ||||
| uint8_t bufferIndex = 0; | ||||
| bool bufferFilled = false; | ||||
| float filteredWeight = 0.0f; | ||||
| int16_t lastDisplayedWeight = 0; | ||||
| int16_t lastStableWeight = 0;        // For API/action triggering | ||||
| unsigned long lastMeasurementTime = 0; | ||||
|  | ||||
| uint8_t weigthCouterToApi = 0; | ||||
| uint8_t scale_tare_counter = 0; | ||||
| bool scaleTareRequest = false; | ||||
| uint8_t pauseMainTask = 0; | ||||
| bool scaleCalibrated; | ||||
| bool autoTare = true; | ||||
| bool scaleCalibrationActive = false; | ||||
|  | ||||
| // ##### Weight stabilization functions ##### | ||||
|  | ||||
| /** | ||||
|  * Reset weight filter buffer - call after tare or calibration | ||||
|  */ | ||||
| void resetWeightFilter() { | ||||
|   bufferIndex = 0; | ||||
|   bufferFilled = false; | ||||
|   filteredWeight = 0.0f; | ||||
|   lastDisplayedWeight = 0; | ||||
|   lastStableWeight = 0;            // Reset stable weight for API actions | ||||
|    | ||||
|   // Initialize buffer with zeros | ||||
|   for (int i = 0; i < MOVING_AVERAGE_SIZE; i++) { | ||||
|     weightBuffer[i] = 0.0f; | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Calculate moving average from weight buffer | ||||
|  */ | ||||
| float calculateMovingAverage() { | ||||
|   float sum = 0.0f; | ||||
|   int count = bufferFilled ? MOVING_AVERAGE_SIZE : bufferIndex; | ||||
|    | ||||
|   for (int i = 0; i < count; i++) { | ||||
|     sum += weightBuffer[i]; | ||||
|   } | ||||
|    | ||||
|   return (count > 0) ? sum / count : 0.0f; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Apply low-pass filter to smooth weight readings | ||||
|  * Uses exponential smoothing: y_new = alpha * x_new + (1-alpha) * y_old | ||||
|  */ | ||||
| float applyLowPassFilter(float newValue) { | ||||
|   filteredWeight = LOW_PASS_ALPHA * newValue + (1.0f - LOW_PASS_ALPHA) * filteredWeight; | ||||
|   return filteredWeight; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Process new weight reading with stabilization | ||||
|  * Returns stabilized weight value | ||||
|  */ | ||||
| int16_t processWeightReading(float rawWeight) { | ||||
|   // Add to moving average buffer | ||||
|   weightBuffer[bufferIndex] = rawWeight; | ||||
|   bufferIndex = (bufferIndex + 1) % MOVING_AVERAGE_SIZE; | ||||
|    | ||||
|   if (bufferIndex == 0) { | ||||
|     bufferFilled = true; | ||||
|   } | ||||
|    | ||||
|   // Calculate moving average | ||||
|   float avgWeight = calculateMovingAverage(); | ||||
|    | ||||
|   // Apply low-pass filter | ||||
|   float smoothedWeight = applyLowPassFilter(avgWeight); | ||||
|    | ||||
|   // Round to nearest gram | ||||
|   int16_t newWeight = round(smoothedWeight); | ||||
|    | ||||
|   // Update displayed weight if display threshold is reached | ||||
|   if (abs(newWeight - lastDisplayedWeight) >= DISPLAY_THRESHOLD) { | ||||
|     lastDisplayedWeight = newWeight; | ||||
|   } | ||||
|    | ||||
|   // Update global weight for API actions only if stable threshold is reached | ||||
|   int16_t weightToReturn = weight; // Default: keep current weight | ||||
|    | ||||
|   if (abs(newWeight - lastStableWeight) >= API_THRESHOLD) { | ||||
|     lastStableWeight = newWeight; | ||||
|     weightToReturn = newWeight; | ||||
|   } | ||||
|    | ||||
|   return weightToReturn; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get current filtered weight for display purposes | ||||
|  * This returns the smoothed weight even if it hasn't triggered API actions | ||||
|  */ | ||||
| int16_t getFilteredDisplayWeight() { | ||||
|   return lastDisplayedWeight; | ||||
| } | ||||
|  | ||||
| // ##### 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() { | ||||
|   Serial.println("Tare scale"); | ||||
|   scale.tare(); | ||||
|   scaleTareRequest == true; | ||||
|    | ||||
|   return 1; | ||||
| } | ||||
| @@ -30,53 +150,111 @@ void scale_loop(void * parameter) { | ||||
|   Serial.println("++++++++++++++++++++++++++++++"); | ||||
|   Serial.println("Scale Loop started"); | ||||
|   Serial.println("++++++++++++++++++++++++++++++"); | ||||
|   for(;;) { | ||||
|     if (scale.is_ready())  | ||||
|     { | ||||
|       // Waage nochmal Taren, wenn zu lange Abweichung | ||||
|       if (scale_tare_counter >= 5)  | ||||
|       { | ||||
|         scale.tare(); | ||||
|         scale_tare_counter = 0; | ||||
|       } | ||||
|  | ||||
|       weight = round(scale.get_units()); | ||||
|   scaleTareRequest == true; | ||||
|   // Initialize weight filter | ||||
|   resetWeightFilter(); | ||||
|   lastMeasurementTime = millis(); | ||||
|  | ||||
|   for(;;) { | ||||
|     unsigned long currentTime = millis(); | ||||
|      | ||||
|     // Only measure at defined intervals to reduce noise | ||||
|     if (currentTime - lastMeasurementTime >= MEASUREMENT_INTERVAL_MS) { | ||||
|       if (scale.is_ready())  | ||||
|       { | ||||
|         // Waage automatisch Taren, wenn zu lange Abweichung | ||||
|         if (autoTare && scale_tare_counter >= 5)  | ||||
|         { | ||||
|           Serial.println("Auto Tare scale"); | ||||
|           scale.tare(); | ||||
|           resetWeightFilter(); // Reset filter after auto tare | ||||
|           scale_tare_counter = 0; | ||||
|         } | ||||
|  | ||||
|         // Waage manuell Taren | ||||
|         if (scaleTareRequest == true)  | ||||
|         { | ||||
|           Serial.println("Re-Tare scale"); | ||||
|           oledShowMessage("TARE Scale"); | ||||
|           vTaskDelay(pdMS_TO_TICKS(1000)); | ||||
|           scale.tare(); | ||||
|           resetWeightFilter(); // Reset filter after manual tare | ||||
|           vTaskDelay(pdMS_TO_TICKS(1000)); | ||||
|           oledShowWeight(0); | ||||
|           scaleTareRequest = false; | ||||
|         } | ||||
|  | ||||
|         // Get raw weight reading | ||||
|         float rawWeight = scale.get_units(); | ||||
|          | ||||
|         // Process weight with stabilization | ||||
|         int16_t stabilizedWeight = processWeightReading(rawWeight); | ||||
|          | ||||
|         // Update global weight variable only if it changed significantly (for API actions) | ||||
|         if (stabilizedWeight != weight) { | ||||
|           weight = stabilizedWeight; | ||||
|         } | ||||
|          | ||||
|         // Debug output for monitoring (can be removed in production) | ||||
|         static unsigned long lastDebugTime = 0; | ||||
|         if (currentTime - lastDebugTime > 2000) { // Print every 2 seconds | ||||
|           lastDebugTime = currentTime; | ||||
|         } | ||||
|          | ||||
|         lastMeasurementTime = currentTime; | ||||
|       } | ||||
|     } | ||||
|      | ||||
|     vTaskDelay(pdMS_TO_TICKS(100)); // Verzögerung, um die CPU nicht zu überlasten | ||||
|     vTaskDelay(pdMS_TO_TICKS(10)); // Shorter delay for more responsive loop | ||||
|   } | ||||
| } | ||||
|  | ||||
| void start_scale() { | ||||
| void start_scale(bool touchSensorConnected) { | ||||
|   Serial.println("Prüfe Calibration Value"); | ||||
|   long calibrationValue; // calibration value (see example file "Calibration.ino") | ||||
|   //calibrationValue = 696.0; // uncomment this if you want to set the calibration value in the sketch | ||||
|   float calibrationValue; | ||||
|  | ||||
|   EEPROM.begin(512); | ||||
|   EEPROM.get(calVal_eepromAdress, calibrationValue); // uncomment this if you want to fetch the calibration value from eeprom | ||||
|   // NVS lesen | ||||
|   Preferences preferences; | ||||
|   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; | ||||
|   } | ||||
|    | ||||
|   //calibrationValue = EEPROM.read(calVal_eepromAdress); | ||||
|   // 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(); | ||||
|  | ||||
|   Serial.print("Read Scale Calibration Value "); | ||||
|   Serial.println(calibrationValue); | ||||
|  | ||||
|   scale.begin(LOADCELL_DOUT_PIN, LOADCELL_SCK_PIN); | ||||
|  | ||||
|   if (isnan(calibrationValue) || calibrationValue < 1) calibrationValue = defaultScaleCalibrationValue; | ||||
|  | ||||
|   oledShowMessage("Scale Tare Please remove all"); | ||||
|   for (uint16_t i = 0; i < 2000; i++) { | ||||
|   oledShowProgressBar(6, 7, DISPLAY_BOOT_TEXT, "Tare scale"); | ||||
|   for (uint16_t i = 0; i < 3000; i++) { | ||||
|     yield(); | ||||
|     vTaskDelay(pdMS_TO_TICKS(1)); | ||||
|     esp_task_wdt_reset(); | ||||
|   } | ||||
|  | ||||
|   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.tare(); | ||||
|   while(!scale.is_ready()) { | ||||
|     vTaskDelay(pdMS_TO_TICKS(5000)); | ||||
|   } | ||||
|  | ||||
|   scale.set_scale(calibrationValue); | ||||
|   //vTaskDelay(pdMS_TO_TICKS(5000)); | ||||
|  | ||||
|   // Initialize weight stabilization filter | ||||
|   resetWeightFilter(); | ||||
|  | ||||
|   // Display Gewicht | ||||
|   oledShowWeight(0); | ||||
|  | ||||
| @@ -84,7 +262,7 @@ void start_scale() { | ||||
|   BaseType_t result = xTaskCreatePinnedToCore( | ||||
|     scale_loop, /* Function to implement the task */ | ||||
|     "ScaleLoop", /* Name of the task */ | ||||
|     10000,  /* Stack size in words */ | ||||
|     2048,  /* Stack size in words */ | ||||
|     NULL,  /* Task input parameter */ | ||||
|     scaleTaskPrio,  /* Priority of the task */ | ||||
|     &ScaleTask,  /* Task handle. */ | ||||
| @@ -98,17 +276,22 @@ void start_scale() { | ||||
| } | ||||
|  | ||||
| uint8_t calibrate_scale() { | ||||
|   long newCalibrationValue; | ||||
|   uint8_t returnState = 0; | ||||
|   float newCalibrationValue; | ||||
|  | ||||
|   scaleCalibrationActive = true; | ||||
|  | ||||
|   vTaskSuspend(RfidReaderTask); | ||||
|   vTaskSuspend(ScaleTask); | ||||
|  | ||||
|   //vTaskSuspend(RfidReaderTask); | ||||
|   vTaskDelete(RfidReaderTask); | ||||
|   pauseBambuMqttTask = true; | ||||
|   pauseMainTask = 1; | ||||
|    | ||||
|   if (scale.wait_ready_timeout(1000)) | ||||
|   { | ||||
|      | ||||
|     scale.set_scale(); | ||||
|     oledShowMessage("Step 1 empty Scale"); | ||||
|     oledShowProgressBar(0, 3, "Scale Cal.", "Empty Scale"); | ||||
|  | ||||
|     for (uint16_t i = 0; i < 5000; i++) { | ||||
|       yield(); | ||||
| @@ -120,7 +303,7 @@ uint8_t calibrate_scale() { | ||||
|     Serial.println("Tare done..."); | ||||
|     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++) { | ||||
|       yield(); | ||||
| @@ -128,7 +311,7 @@ uint8_t calibrate_scale() { | ||||
|       esp_task_wdt_reset(); | ||||
|     } | ||||
|      | ||||
|     long newCalibrationValue = scale.get_units(10); | ||||
|     float newCalibrationValue = scale.get_units(10); | ||||
|     Serial.print("Result: "); | ||||
|     Serial.println(newCalibrationValue); | ||||
|  | ||||
| @@ -138,22 +321,36 @@ uint8_t calibrate_scale() { | ||||
|     { | ||||
|       Serial.print("New calibration value has been set to: "); | ||||
|       Serial.println(newCalibrationValue); | ||||
|       Serial.print("Save this value to EEPROM adress "); | ||||
|       Serial.println(calVal_eepromAdress); | ||||
|  | ||||
|       //EEPROM.put(calVal_eepromAdress, newCalibrationValue); | ||||
|       EEPROM.put(calVal_eepromAdress, newCalibrationValue); | ||||
|       EEPROM.commit(); | ||||
|       // Speichern mit NVS | ||||
|       Preferences preferences; | ||||
|       preferences.begin(NVS_NAMESPACE_SCALE, false); // false = readwrite | ||||
|       preferences.putFloat(NVS_KEY_CALIBRATION, newCalibrationValue); | ||||
|       preferences.end(); | ||||
|  | ||||
|       EEPROM.get(calVal_eepromAdress, newCalibrationValue); | ||||
|       //newCalibrationValue = EEPROM.read(calVal_eepromAdress); | ||||
|       // Verifizieren | ||||
|       preferences.begin(NVS_NAMESPACE_SCALE, true); | ||||
|       float verifyValue = preferences.getFloat(NVS_KEY_CALIBRATION, 0); | ||||
|       preferences.end(); | ||||
|  | ||||
|       Serial.print("Read Value "); | ||||
|       Serial.println(newCalibrationValue); | ||||
|       Serial.print("Verified stored value: "); | ||||
|       Serial.println(verifyValue); | ||||
|  | ||||
|       Serial.println("End calibration, revome weight"); | ||||
|       oledShowProgressBar(2, 3, "Scale Cal.", "Remove weight"); | ||||
|  | ||||
|       oledShowMessage("Remove weight"); | ||||
|       scale.set_scale(newCalibrationValue); | ||||
|       resetWeightFilter(); // Reset filter after calibration | ||||
|       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++) { | ||||
|         yield(); | ||||
| @@ -161,30 +358,21 @@ uint8_t calibrate_scale() { | ||||
|         esp_task_wdt_reset(); | ||||
|       } | ||||
|  | ||||
|       oledShowMessage("Calibration done"); | ||||
|  | ||||
|       for (uint16_t i = 0; i < 2000; i++) { | ||||
|         yield(); | ||||
|         vTaskDelay(pdMS_TO_TICKS(1)); | ||||
|         esp_task_wdt_reset(); | ||||
|       } | ||||
|  | ||||
|       //ESP.restart(); | ||||
|       scaleCalibrated = true; | ||||
|       returnState = 1; | ||||
|     } | ||||
|     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++) { | ||||
|           yield(); | ||||
|           vTaskDelay(pdMS_TO_TICKS(1)); | ||||
|           esp_task_wdt_reset(); | ||||
|         } | ||||
|         return 0; | ||||
|       for (uint16_t i = 0; i < 50000; i++) { | ||||
|         yield(); | ||||
|         vTaskDelay(pdMS_TO_TICKS(1)); | ||||
|         esp_task_wdt_reset(); | ||||
|       } | ||||
|       returnState = 0; | ||||
|     }  | ||||
|   } | ||||
|   else  | ||||
| @@ -198,17 +386,14 @@ uint8_t calibrate_scale() { | ||||
|       vTaskDelay(pdMS_TO_TICKS(1)); | ||||
|       esp_task_wdt_reset(); | ||||
|     } | ||||
|     return 0; | ||||
|     returnState = 0; | ||||
|   } | ||||
|  | ||||
|   oledShowMessage("Scale Ready"); | ||||
|  | ||||
|    | ||||
|   Serial.println("starte Scale Task"); | ||||
|   start_scale(); | ||||
|  | ||||
|   vTaskResume(RfidReaderTask); | ||||
|   vTaskResume(ScaleTask); | ||||
|   pauseBambuMqttTask = false; | ||||
|   pauseMainTask = 0; | ||||
|   scaleCalibrationActive = false; | ||||
|  | ||||
|   return 1; | ||||
|   return returnState; | ||||
| } | ||||
|   | ||||
							
								
								
									
										17
									
								
								src/scale.h
									
									
									
									
									
								
							
							
						
						| @@ -4,15 +4,28 @@ | ||||
| #include <Arduino.h> | ||||
| #include "HX711.h" | ||||
|  | ||||
|  | ||||
| void start_scale(); | ||||
| uint8_t setAutoTare(bool autoTareValue); | ||||
| uint8_t start_scale(bool touchSensorConnected); | ||||
| uint8_t calibrate_scale(); | ||||
| uint8_t tareScale(); | ||||
|  | ||||
| // Weight stabilization functions | ||||
| void resetWeightFilter(); | ||||
| float calculateMovingAverage(); | ||||
| float applyLowPassFilter(float newValue); | ||||
| int16_t processWeightReading(float rawWeight); | ||||
| int16_t getFilteredDisplayWeight(); | ||||
|  | ||||
| extern HX711 scale; | ||||
| extern int16_t weight; | ||||
| extern uint8_t weigthCouterToApi; | ||||
| extern uint8_t scale_tare_counter; | ||||
| extern uint8_t scaleTareRequest; | ||||
| extern uint8_t pauseMainTask; | ||||
| extern bool scaleCalibrated; | ||||
| extern bool autoTare; | ||||
| extern bool scaleCalibrationActive; | ||||
|  | ||||
| extern TaskHandle_t ScaleTask; | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										304
									
								
								src/website.cpp
									
									
									
									
									
								
							
							
						
						| @@ -3,35 +3,60 @@ | ||||
| #include "api.h" | ||||
| #include <ArduinoJson.h> | ||||
| #include <ESPAsyncWebServer.h> | ||||
| //#include <AsyncWebSocket.h> | ||||
| #include "bambu.h" | ||||
| #include "nfc.h" | ||||
| #include "scale.h" | ||||
| #include "esp_task_wdt.h" | ||||
| #include <Update.h> | ||||
| #include "display.h" | ||||
| #include "ota.h" | ||||
| #include "config.h" | ||||
| #include "debug.h" | ||||
|  | ||||
|  | ||||
| #ifndef VERSION | ||||
|   #define VERSION "1.1.0" | ||||
| #endif | ||||
|  | ||||
| // Cache-Control Header definieren | ||||
| #define CACHE_CONTROL "max-age=31536000" // Cache für 1 Jahr | ||||
| #define CACHE_CONTROL "max-age=604800" // Cache für 1 Woche | ||||
|  | ||||
| AsyncWebServer server(webserverPort); | ||||
| AsyncWebSocket ws("/ws"); | ||||
|  | ||||
| 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) { | ||||
|     HEAP_DEBUG_MESSAGE("onWsEvent begin"); | ||||
|     if (type == WS_EVT_CONNECT) { | ||||
|         Serial.println("Neuer Client verbunden!"); | ||||
|         // Sende die AMS-Daten an den neuen Client | ||||
|         sendAmsData(client); | ||||
|         sendNfcData(client); | ||||
|         foundNfcTag(client, hasReadRfidTag); | ||||
|         sendWriteResult(client, 0); | ||||
|         if (!bambuDisabled) sendAmsData(client); | ||||
|         sendNfcData(); | ||||
|         foundNfcTag(client, 0); | ||||
|         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) { | ||||
|         Serial.println("Client getrennt."); | ||||
|     } else if (type == WS_EVT_ERROR) { | ||||
|         Serial.printf("WebSocket Client #%u error(%u): %s\n", client->id(), *((uint16_t*)arg), (char*)data); | ||||
|     } else if (type == WS_EVT_PONG) { | ||||
|         Serial.printf("WebSocket Client #%u pong\n", client->id()); | ||||
|     } else if (type == WS_EVT_DATA) { | ||||
|         String message = String((char*)data); | ||||
|         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") { | ||||
|             // Sende Heartbeat-Antwort | ||||
| @@ -39,16 +64,17 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | ||||
|                 "\"type\":\"heartbeat\"," | ||||
|                 "\"freeHeap\":" + String(ESP.getFreeHeap()/1024) + "," | ||||
|                 "\"bambu_connected\":" + String(bambu_connected) + "," | ||||
|                 "\"spoolman_connected\":" + String(spoolman_connected) + "" | ||||
|                 "\"spoolman_connected\":" + String(spoolmanConnected) + "" | ||||
|                 "}"); | ||||
|         } | ||||
|  | ||||
|         else if (doc["type"] == "writeNfcTag") { | ||||
|             if (doc.containsKey("payload")) { | ||||
|             if (doc["payload"].is<JsonObject>()) { | ||||
|                 // Versuche NFC-Daten zu schreiben | ||||
|                 String payloadString; | ||||
|                 serializeJson(doc["payload"], payloadString); | ||||
|                 startWriteJsonToTag(payloadString.c_str()); | ||||
|  | ||||
|                 startWriteJsonToTag((doc["tagType"] == "spool") ? true : false, payloadString.c_str()); | ||||
|             } | ||||
|         } | ||||
|  | ||||
| @@ -62,6 +88,10 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | ||||
|                 success = calibrate_scale(); | ||||
|             } | ||||
|  | ||||
|             if (doc["payload"] == "setAutoTare") { | ||||
|                 success = setAutoTare(doc["enabled"].as<bool>()); | ||||
|             } | ||||
|  | ||||
|             if (success) { | ||||
|                 ws.textAll("{\"type\":\"scale\",\"payload\":\"success\"}"); | ||||
|             } else { | ||||
| @@ -69,37 +99,50 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         else if (doc["type"] == "reconnect") { | ||||
|             if (doc["payload"] == "bambu") { | ||||
|                 bambu_restart(); | ||||
|             } | ||||
|  | ||||
|             if (doc["payload"] == "spoolman") { | ||||
|                 initSpoolman(); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         else if (doc["type"] == "setBambuSpool") { | ||||
|             Serial.println(doc["payload"].as<String>()); | ||||
|             setBambuSpool(doc["payload"]); | ||||
|         } | ||||
|  | ||||
|         else if (doc["type"] == "setSpoolmanSettings") { | ||||
|             Serial.println(doc["payload"].as<String>()); | ||||
|             if (updateSpoolBambuData(doc["payload"].as<String>())) { | ||||
|                 ws.textAll("{\"type\":\"setSpoolmanSettings\",\"payload\":\"success\"}"); | ||||
|             } else { | ||||
|                 ws.textAll("{\"type\":\"setSpoolmanSettings\",\"payload\":\"error\"}"); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         else { | ||||
|             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 | ||||
| String loadHtmlWithHeader(const char* filename) { | ||||
|     if (!SPIFFS.exists(filename) || !SPIFFS.exists("/header.html")) { | ||||
|     Serial.println("Lade HTML-Datei: " + String(filename)); | ||||
|     if (!LittleFS.exists(filename)) { | ||||
|         Serial.println("Fehler: Datei nicht gefunden!"); | ||||
|         return "Fehler: Datei nicht gefunden!"; | ||||
|     } | ||||
|  | ||||
|     // Lade den Header | ||||
|     File headerFile = SPIFFS.open("/header.html", "r"); | ||||
|     String header = headerFile.readString(); | ||||
|     headerFile.close(); | ||||
|  | ||||
|     // Lade die Hauptdatei | ||||
|     File file = SPIFFS.open(filename, "r"); | ||||
|     File file = LittleFS.open(filename, "r"); | ||||
|     String html = file.readString(); | ||||
|     file.close(); | ||||
|  | ||||
|     // Ersetze den Platzhalter mit dem Header | ||||
|     html.replace("{{header}}", header); | ||||
|  | ||||
|     return html; | ||||
| } | ||||
|  | ||||
| @@ -112,86 +155,100 @@ void sendWriteResult(AsyncWebSocketClient *client, uint8_t success) { | ||||
| void foundNfcTag(AsyncWebSocketClient *client, uint8_t success) { | ||||
|     if (success == lastSuccess) return; | ||||
|     ws.textAll("{\"type\":\"nfcTag\", \"payload\":{\"found\": " + String(success) + "}}"); | ||||
|     sendNfcData(nullptr); | ||||
|     sendNfcData(); | ||||
|     lastSuccess = success; | ||||
| } | ||||
|  | ||||
| void sendNfcData(AsyncWebSocketClient *client) { | ||||
|     if (lastHasReadRfidTag == hasReadRfidTag) return; | ||||
|     if (hasReadRfidTag == 0) { | ||||
|         ws.textAll("{\"type\":\"nfcData\", \"payload\":{}}"); | ||||
| void sendNfcData() { | ||||
|     if (lastnfcReaderState == nfcReaderState) return; | ||||
|     // TBD: Why is there no status for reading the tag? | ||||
|     switch(nfcReaderState){ | ||||
|         case NFC_IDLE: | ||||
|             ws.textAll("{\"type\":\"nfcData\", \"payload\":{}}"); | ||||
|             break; | ||||
|         case NFC_READ_SUCCESS: | ||||
|             ws.textAll("{\"type\":\"nfcData\", \"payload\":" + nfcJsonData + "}"); | ||||
|             break; | ||||
|         case NFC_READ_ERROR: | ||||
|             ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"error\":\"Empty Tag or Data not readable\"}}"); | ||||
|             break; | ||||
|         case NFC_WRITING: | ||||
|             ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"info\":\"Schreibe Tag...\"}}"); | ||||
|             break; | ||||
|         case NFC_WRITE_SUCCESS: | ||||
|             ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"info\":\"Tag erfolgreich geschrieben\"}}"); | ||||
|             break; | ||||
|         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\"}}"); | ||||
|     } | ||||
|     else if (hasReadRfidTag == 1) { | ||||
|         ws.textAll("{\"type\":\"nfcData\", \"payload\":" + nfcJsonData + "}"); | ||||
|     } | ||||
|     else if (hasReadRfidTag == 2) | ||||
|     { | ||||
|         ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"error\":\"Empty Tag or Data not readable\"}}"); | ||||
|     } | ||||
|     else if (hasReadRfidTag == 3) | ||||
|     { | ||||
|         ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"info\":\"Schreibe Tag...\"}}"); | ||||
|     } | ||||
|     else if (hasReadRfidTag == 4) | ||||
|     { | ||||
|         ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"error\":\"Error writing to Tag\"}}"); | ||||
|     } | ||||
|     else if (hasReadRfidTag == 5) | ||||
|     { | ||||
|         ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"info\":\"Tag erfolgreich geschrieben\"}}"); | ||||
|     } | ||||
|     else  | ||||
|     { | ||||
|         ws.textAll("{\"type\":\"nfcData\", \"payload\":{\"error\":\"Something went wrong\"}}"); | ||||
|     } | ||||
|     lastHasReadRfidTag = hasReadRfidTag; | ||||
|     lastnfcReaderState = nfcReaderState; | ||||
| } | ||||
|  | ||||
| void sendAmsData(AsyncWebSocketClient *client) { | ||||
|     if (ams_count > 0) { | ||||
|         ws.textAll("{\"type\":\"amsData\", \"payload\":" + amsJsonData + "}"); | ||||
|         ws.textAll("{\"type\":\"amsData\",\"payload\":" + amsJsonData + "}"); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void setupWebserver(AsyncWebServer &server) { | ||||
|     oledShowProgressBar(2, 7, DISPLAY_BOOT_TEXT, "Webserver init"); | ||||
|     // Deaktiviere alle Debug-Ausgaben | ||||
|     Serial.setDebugOutput(false); | ||||
|      | ||||
|     // WebSocket-Optimierungen | ||||
|     ws.onEvent(onWsEvent); | ||||
|     ws.enable(true); | ||||
|  | ||||
|     // Konfiguriere Server für große Uploads | ||||
|     server.onRequestBody([](AsyncWebServerRequest *request, uint8_t *data, size_t len, size_t index, size_t total){}); | ||||
|     server.onFileUpload([](AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final){}); | ||||
|  | ||||
|     // Lade die Spoolman-URL beim Booten | ||||
|     spoolmanUrl = loadSpoolmanUrl(); | ||||
|     Serial.print("Geladene Spoolman-URL: "); | ||||
|     Serial.println(spoolmanUrl); | ||||
|  | ||||
|     // Route für die Startseite | ||||
|     // Load Bamb credentials: | ||||
|     loadBambuCredentials(); | ||||
|  | ||||
|     // Route für about | ||||
|     server.on("/about", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         Serial.println("Anfrage für / erhalten"); | ||||
|         String html = loadHtmlWithHeader("/index.html"); | ||||
|         request->send(200, "text/html", html); | ||||
|         Serial.println("Anfrage für /about erhalten"); | ||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/index.html.gz", "text/html"); | ||||
|         response->addHeader("Content-Encoding", "gzip"); | ||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||
|         request->send(response); | ||||
|     }); | ||||
|  | ||||
|     // Route für Waage | ||||
|     server.on("/waage", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         Serial.println("Anfrage für /waage erhalten"); | ||||
|         //AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/waage.html.gz", "text/html"); | ||||
|         //response->addHeader("Content-Encoding", "gzip"); | ||||
|         //response->addHeader("Cache-Control", CACHE_CONTROL); | ||||
|  | ||||
|         String html = loadHtmlWithHeader("/waage.html"); | ||||
|         html.replace("{{autoTare}}", (autoTare) ? "checked" : ""); | ||||
|  | ||||
|         request->send(200, "text/html", html); | ||||
|     }); | ||||
|  | ||||
|     // Route für RFID | ||||
|     server.on("/", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         Serial.println("Anfrage für /rfid erhalten"); | ||||
|         String html = loadHtmlWithHeader("/rfid.html"); | ||||
|         request->send(200, "text/html", 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("Cache-Control", CACHE_CONTROL); | ||||
|         request->send(response); | ||||
|         Serial.println("RFID-Seite gesendet"); | ||||
|     }); | ||||
|  | ||||
|     /* | ||||
|     // Neue API-Route für das Abrufen der Spool-Daten | ||||
|     server.on("/api/spools", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         Serial.println("API-Aufruf: /api/spools"); | ||||
|         JsonDocument spoolsData = fetchSpoolsForWebsite(); | ||||
|         String response; | ||||
|         serializeJson(spoolsData, response); | ||||
|         request->send(200, "application/json", response); | ||||
|     }); | ||||
|     */ | ||||
|  | ||||
|     server.on("/api/url", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         Serial.println("API-Aufruf: /api/url"); | ||||
|         String jsonResponse = "{\"spoolman_url\": \"" + String(spoolmanUrl) + "\"}"; | ||||
| @@ -201,22 +258,26 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|     // Route für WiFi | ||||
|     server.on("/wifi", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         Serial.println("Anfrage für /wifi erhalten"); | ||||
|         String html = loadHtmlWithHeader("/wifi.html"); | ||||
|         request->send(200, "text/html", html); | ||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/wifi.html.gz", "text/html"); | ||||
|         response->addHeader("Content-Encoding", "gzip"); | ||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||
|         request->send(response); | ||||
|     }); | ||||
|  | ||||
|     // Route für Spoolman Setting | ||||
|     server.on("/spoolman", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         Serial.println("Anfrage für /spoolman erhalten"); | ||||
|         String html = loadHtmlWithHeader("/spoolman.html"); | ||||
|         html.replace("{{spoolmanUrl}}", spoolmanUrl); | ||||
|         html.replace("{{spoolmanUrl}}", (spoolmanUrl != "") ? spoolmanUrl : ""); | ||||
|         html.replace("{{spoolmanOctoEnabled}}", octoEnabled ? "checked" : ""); | ||||
|         html.replace("{{spoolmanOctoUrl}}", (octoUrl != "") ? octoUrl : ""); | ||||
|         html.replace("{{spoolmanOctoToken}}", (octoToken != "") ? octoToken : ""); | ||||
|  | ||||
|         JsonDocument doc; | ||||
|         if (loadJsonValue("/bambu_credentials.json", doc) && doc.containsKey("bambu_ip")) { | ||||
|             html.replace("{{bambuIp}}", doc["bambu_ip"].as<String>() ? doc["bambu_ip"].as<String>() : "");             | ||||
|             html.replace("{{bambuSerial}}", doc["bambu_serialnr"].as<String>() ? doc["bambu_serialnr"].as<String>() : ""); | ||||
|             html.replace("{{bambuCode}}", doc["bambu_accesscode"].as<String>() ? doc["bambu_accesscode"].as<String>() : ""); | ||||
|         }    | ||||
|         html.replace("{{bambuIp}}", bambuCredentials.ip);             | ||||
|         html.replace("{{bambuSerial}}", bambuCredentials.serial); | ||||
|         html.replace("{{bambuCode}}", bambuCredentials.accesscode ? bambuCredentials.accesscode : ""); | ||||
|         html.replace("{{autoSendToBambu}}", bambuCredentials.autosend_enable ? "checked" : ""); | ||||
|         html.replace("{{autoSendTime}}", (bambuCredentials.autosend_time != 0) ? String(bambuCredentials.autosend_time) : String(BAMBU_DEFAULT_AUTOSEND_TIME)); | ||||
|  | ||||
|         request->send(200, "text/html", html); | ||||
|     }); | ||||
| @@ -228,17 +289,45 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         String url = request->getParam("url")->value(); | ||||
|         url.trim(); | ||||
|         if (request->getParam("octoEnabled")->value() == "true" && (!request->hasParam("octoUrl") || !request->hasParam("octoToken"))) { | ||||
|             request->send(400, "application/json", "{\"success\": false, \"error\": \"Missing OctoPrint URL or Token parameter\"}"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         bool healthy = saveSpoolmanUrl(url); | ||||
|         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; | ||||
|         String octoUrl = request->getParam("octoUrl")->value(); | ||||
|         String octoToken = (request->getParam("octoToken")->value() != "") ? request->getParam("octoToken")->value() : ""; | ||||
|  | ||||
|         url.trim(); | ||||
|         octoUrl.trim(); | ||||
|         octoToken.trim(); | ||||
|          | ||||
|         bool healthy = saveSpoolmanUrl(url, octoEnabled, octoUrl, octoToken); | ||||
|         String jsonResponse = "{\"healthy\": " + String(healthy ? "true" : "false") + "}"; | ||||
|  | ||||
|         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){ | ||||
|         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")) { | ||||
|             request->send(400, "application/json", "{\"success\": false, \"error\": \"Missing parameter\"}"); | ||||
|             return; | ||||
| @@ -247,11 +336,20 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|         String bambu_ip = request->getParam("bambu_ip")->value(); | ||||
|         String bambu_serialnr = request->getParam("bambu_serialnr")->value(); | ||||
|         String bambu_accesscode = request->getParam("bambu_accesscode")->value(); | ||||
|         bool autoSend = (request->getParam("autoSend")->value() == "true") ? true : false; | ||||
|         String autoSendTime = request->getParam("autoSendTime")->value(); | ||||
|          | ||||
|         bambu_ip.trim(); | ||||
|         bambu_serialnr.trim(); | ||||
|         bambu_accesscode.trim(); | ||||
|         autoSendTime.trim(); | ||||
|  | ||||
|         bool success = saveBambuCredentials(bambu_ip, bambu_serialnr, bambu_accesscode); | ||||
|         if (bambu_ip.length() == 0 || bambu_serialnr.length() == 0 || bambu_accesscode.length() == 0) { | ||||
|             request->send(400, "application/json", "{\"success\": false, \"error\": \"Empty parameter\"}"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         bool success = saveBambuCredentials(bambu_ip, bambu_serialnr, bambu_accesscode, autoSend, autoSendTime); | ||||
|  | ||||
|         request->send(200, "application/json", "{\"healthy\": " + String(success ? "true" : "false") + "}"); | ||||
|     }); | ||||
| @@ -264,7 +362,7 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|     // Route für das Laden der CSS-Datei | ||||
|     server.on("/style.css", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         Serial.println("Lade style.css"); | ||||
|         AsyncWebServerResponse *response = request->beginResponse(SPIFFS, "/style.css.gz", "text/css"); | ||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/style.css.gz", "text/css"); | ||||
|         response->addHeader("Content-Encoding", "gzip"); | ||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||
|         request->send(response); | ||||
| @@ -273,7 +371,7 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|  | ||||
|     // Route für das Logo | ||||
|     server.on("/logo.png", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         AsyncWebServerResponse *response = request->beginResponse(SPIFFS, "/logo.png.gz", "image/png"); | ||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/logo.png.gz", "image/png"); | ||||
|         response->addHeader("Content-Encoding", "gzip"); | ||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||
|         request->send(response); | ||||
| @@ -282,7 +380,7 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|  | ||||
|     // Route für Favicon | ||||
|     server.on("/favicon.ico", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         AsyncWebServerResponse *response = request->beginResponse(SPIFFS, "/favicon.ico", "image/x-icon"); | ||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/favicon.ico", "image/x-icon"); | ||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||
|         request->send(response); | ||||
|         Serial.println("favicon.ico gesendet"); | ||||
| @@ -290,17 +388,26 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|  | ||||
|     // Route für spool_in.png | ||||
|     server.on("/spool_in.png", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         AsyncWebServerResponse *response = request->beginResponse(SPIFFS, "/spool_in.png.gz", "image/png"); | ||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/spool_in.png.gz", "image/png"); | ||||
|         response->addHeader("Content-Encoding", "gzip"); | ||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||
|         request->send(response); | ||||
|         Serial.println("spool_in.png gesendet"); | ||||
|     }); | ||||
|  | ||||
|     // Route für set_spoolman.png | ||||
|     server.on("/set_spoolman.png", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/set_spoolman.png.gz", "image/png"); | ||||
|         response->addHeader("Content-Encoding", "gzip"); | ||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||
|         request->send(response); | ||||
|         Serial.println("set_spoolman.png gesendet"); | ||||
|     }); | ||||
|  | ||||
|     // Route für JavaScript Dateien | ||||
|     server.on("/spoolman.js", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         Serial.println("Anfrage für /spoolman.js erhalten"); | ||||
|         AsyncWebServerResponse *response = request->beginResponse(SPIFFS, "/spoolman.js.gz", "text/javascript"); | ||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/spoolman.js.gz", "text/javascript"); | ||||
|         response->addHeader("Content-Encoding", "gzip"); | ||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||
|         request->send(response); | ||||
| @@ -309,13 +416,30 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|  | ||||
|     server.on("/rfid.js", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         Serial.println("Anfrage für /rfid.js erhalten"); | ||||
|         AsyncWebServerResponse *response = request->beginResponse(SPIFFS,"/rfid.js.gz", "text/javascript"); | ||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS,"/rfid.js.gz", "text/javascript"); | ||||
|         response->addHeader("Content-Encoding", "gzip"); | ||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||
|         request->send(response); | ||||
|         Serial.println("RFID.js gesendet"); | ||||
|     }); | ||||
|  | ||||
|     // Vereinfachter Update-Handler | ||||
|     server.on("/upgrade", HTTP_GET, [](AsyncWebServerRequest *request) { | ||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/upgrade.html.gz", "text/html"); | ||||
|         response->addHeader("Content-Encoding", "gzip"); | ||||
|         response->addHeader("Cache-Control", "no-store"); | ||||
|         request->send(response); | ||||
|     }); | ||||
|  | ||||
|     // Update-Handler registrieren | ||||
|     handleUpdate(server); | ||||
|  | ||||
|     server.on("/api/version", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         String fm_version = VERSION; | ||||
|         String jsonResponse = "{\"version\": \""+ fm_version +"\"}"; | ||||
|         request->send(200, "application/json", jsonResponse); | ||||
|     }); | ||||
|  | ||||
|     // Fehlerbehandlung für nicht gefundene Seiten | ||||
|     server.onNotFound([](AsyncWebServerRequest *request){ | ||||
|         Serial.print("404 - Nicht gefunden: "); | ||||
|   | ||||
| @@ -6,8 +6,8 @@ | ||||
| #include "commonFS.h" | ||||
| #include "api.h" | ||||
| #include <ArduinoJson.h> | ||||
| #include <ESPAsyncWebServer.h> | ||||
| #include <AsyncWebSocket.h> | ||||
| #include <Update.h> | ||||
| #include <AsyncTCP.h> | ||||
| #include "bambu.h" | ||||
| #include "nfc.h" | ||||
| #include "scale.h" | ||||
| @@ -17,9 +17,14 @@ extern String spoolmanUrl; | ||||
| extern AsyncWebServer server; | ||||
| extern AsyncWebSocket ws; | ||||
|  | ||||
| // Server-Initialisierung und Handler | ||||
| void initWebServer(); | ||||
| void handleBody(AsyncWebServerRequest *request, uint8_t *data, size_t len, size_t index, size_t total); | ||||
| void setupWebserver(AsyncWebServer &server); | ||||
|  | ||||
| // WebSocket-Funktionen | ||||
| void sendAmsData(AsyncWebSocketClient *client); | ||||
| void sendNfcData(AsyncWebSocketClient *client); | ||||
| void sendNfcData(); | ||||
| void foundNfcTag(AsyncWebSocketClient *client, uint8_t success); | ||||
| void sendWriteResult(AsyncWebSocketClient *client, uint8_t success); | ||||
|  | ||||
|   | ||||
							
								
								
									
										125
									
								
								src/wlan.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,125 @@ | ||||
| #include <Arduino.h> | ||||
| #include "wlan.h" | ||||
| #include <WiFi.h> | ||||
| #include <esp_wifi.h> | ||||
| #include <WiFiManager.h> | ||||
| #include <DNSServer.h> | ||||
| #include <ESPmDNS.h> | ||||
| #include "display.h" | ||||
| #include "config.h" | ||||
|  | ||||
| WiFiManager wm; | ||||
| bool wm_nonblocking = false; | ||||
| uint8_t wifiErrorCounter = 0; | ||||
|  | ||||
| void wifiSettings() { | ||||
|     // Optimierte WiFi-Einstellungen | ||||
|     WiFi.mode(WIFI_STA); // explicitly set mode, esp defaults to STA+AP | ||||
|     WiFi.setSleep(false); // disable sleep mode | ||||
|     WiFi.setHostname("FilaMan"); | ||||
|     esp_wifi_set_ps(WIFI_PS_NONE); | ||||
|      | ||||
|     // Maximale Sendeleistung | ||||
|     WiFi.setTxPower(WIFI_POWER_19_5dBm); // Set maximum transmit power | ||||
|    | ||||
|     // Optimiere TCP/IP Stack | ||||
|     esp_wifi_set_protocol(WIFI_IF_STA, WIFI_PROTOCOL_11B | WIFI_PROTOCOL_11G | WIFI_PROTOCOL_11N); | ||||
|      | ||||
|     // Aktiviere WiFi-Roaming für bessere Stabilität | ||||
|     esp_wifi_set_rssi_threshold(-80); | ||||
| } | ||||
|  | ||||
| void startMDNS() { | ||||
|   if (!MDNS.begin("filaman")) { | ||||
|     Serial.println("Error setting up MDNS responder!"); | ||||
|     while(1) { | ||||
|       vTaskDelay(1000 / portTICK_PERIOD_MS); | ||||
|     } | ||||
|   } | ||||
|   Serial.println("mDNS responder started"); | ||||
| } | ||||
|  | ||||
| void configModeCallback (WiFiManager *myWiFiManager) { | ||||
|   Serial.println("Entered config mode"); | ||||
|   oledShowTopRow(); | ||||
|   oledShowMessage("WiFi Config Mode"); | ||||
| } | ||||
|  | ||||
| void initWiFi() { | ||||
|   // load Wifi settings | ||||
|   wifiSettings(); | ||||
|  | ||||
|   wm.setAPCallback(configModeCallback); | ||||
|  | ||||
|   wm.setSaveConfigCallback([]() { | ||||
|     Serial.println("Configurations updated"); | ||||
|     ESP.restart(); | ||||
|   }); | ||||
|  | ||||
|   if(wm_nonblocking) wm.setConfigPortalBlocking(false); | ||||
|   //wm.setConfigPortalTimeout(320); // Portal nach 5min schließen | ||||
|   wm.setWiFiAutoReconnect(true); | ||||
|   wm.setConnectTimeout(10); | ||||
|  | ||||
|   oledShowProgressBar(1, 7, DISPLAY_BOOT_TEXT, "WiFi init"); | ||||
|    | ||||
|   //bool res = wm.autoConnect("FilaMan"); // anonymous ap | ||||
|   if(!wm.autoConnect("FilaMan")) { | ||||
|     Serial.println("Failed to connect or hit timeout"); | ||||
|     // ESP.restart(); | ||||
|     oledShowTopRow(); | ||||
|     oledShowMessage("WiFi not connected Check Portal"); | ||||
|   }  | ||||
|   else { | ||||
|     wifiOn = true; | ||||
|  | ||||
|     //if you get here you have connected to the WiFi     | ||||
|     Serial.println("connected...yeey :)"); | ||||
|     Serial.print("IP address: "); | ||||
|     Serial.println(WiFi.localIP()); | ||||
|  | ||||
|     oledShowTopRow(); | ||||
|  | ||||
|     // mDNS | ||||
|     startMDNS(); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void checkWiFiConnection() { | ||||
|   if (WiFi.status() != WL_CONNECTED)  | ||||
|   { | ||||
|     Serial.println("WiFi connection lost. Reconnecting..."); | ||||
|     wifiOn = false; | ||||
|     oledShowTopRow(); | ||||
|     oledShowMessage("WiFi reconnecting"); | ||||
|     WiFi.reconnect(); // Versuche, die Verbindung wiederherzustellen | ||||
|     vTaskDelay(5000 / portTICK_PERIOD_MS); // Warte 5 Sekunden, bevor erneut geprüft wird | ||||
|     if (WiFi.status() != WL_CONNECTED)  | ||||
|     { | ||||
|       Serial.println("Failed to reconnect. Restarting WiFi..."); | ||||
|       WiFi.disconnect(); | ||||
|       Serial.println("WiFi disconnected."); | ||||
|       vTaskDelay(1000 / portTICK_PERIOD_MS); | ||||
|       wifiErrorCounter++; | ||||
|  | ||||
|       //wifiSettings(); | ||||
|       WiFi.reconnect(); | ||||
|       Serial.println("WiFi reconnecting..."); | ||||
|       WiFi.waitForConnectResult(); | ||||
|     }  | ||||
|     else  | ||||
|     { | ||||
|       Serial.println("WiFi reconnected."); | ||||
|       wifiErrorCounter = 0; | ||||
|       wifiOn = true; | ||||
|       oledShowTopRow(); | ||||
|       startMDNS(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (wifiErrorCounter >= 5)  | ||||
|   { | ||||
|     Serial.println("Too many WiFi errors. Restarting..."); | ||||
|     ESP.restart(); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										9
									
								
								src/wlan.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | ||||
| #ifndef WLAN_H | ||||
| #define WLAN_H | ||||
|  | ||||
| #include <Arduino.h> | ||||
|  | ||||
| void initWiFi(); | ||||
| void checkWiFiConnection(); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										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
									
								
							
							
						
						
							
								
								
									
										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. | ||||