Compare commits
	
		
			91 Commits
		
	
	
		
			v1.4.0
			...
			07a919b6ba
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 07a919b6ba | |||
| 8618b90e33 | |||
| 57723b5354 | |||
| d2be752175 | |||
| 97a050ace8 | |||
| 367e692c74 | |||
| 926a21249b | |||
| 2635c19667 | |||
|  | 6cc4efca0a | ||
| 1484a6b0da | |||
| b5f0472af4 | |||
| 95c1bc823c | |||
| 491ba7f526 | |||
| 56d7d8596c | |||
| 1044e91a0a | |||
| e459b53472 | |||
| 024056cb7d | |||
| e040a736b0 | |||
| 72b6b349c6 | |||
| 190e952ec4 | |||
| 89620a7f00 | |||
| 536950eeb3 | |||
|  | fe4d2d7479 | ||
| 43719aac41 | |||
| 16d0079f7a | |||
| 48b9bf7076 | |||
| b6bd4cb9ad | |||
| e89bb1d547 | |||
| f25789d703 | |||
|  | 65d8cd675f | ||
| 9dfe75ffa2 | |||
| 68cdd8ab40 | |||
| 1069781931 | |||
| eada54eff2 | |||
| 48301ade36 | |||
| 76e0b20393 | |||
| a765b39896 | |||
| d68f6c4a89 | |||
| 1702e2396e | |||
| af23b07df1 | |||
| dd7ba3bf5d | |||
| a818dcd3c0 | |||
| b5279b167a | |||
| a09fd4fda4 | |||
| e4fe08f54c | |||
| 3eac0e5ac4 | |||
| 24d91693d9 | |||
| 94c26590c8 | |||
| 4559bae066 | |||
| cdb2d16cf9 | |||
| cd71949c82 | |||
| 6cd280389d | |||
|  | daf27820b1 | ||
|  | dd7fbe1119 | ||
|  | dc2ddb47eb | ||
|  | 6bb8f565e6 | ||
|  | ec60ca88f1 | ||
|  | 17664acf9e | ||
|  | 18f7454a76 | ||
|  | e7b5917888 | ||
|  | 5c57968ba9 | ||
|  | 795c926c1f | ||
|  | 8735a9740c | ||
|  | 02d0adc6bf | ||
|  | 24067666ed | ||
|  | 9264333eda | ||
|  | 66216d57ae | ||
|  | 5100a669b0 | ||
|  | 4ad89b68a7 | ||
|  | 758acaff9f | ||
|  | fed96b9c58 | ||
|  | 2d072ee09a | ||
|  | b55b6e3fd5 | ||
|  | 238b928236 | ||
|  | 24ce0ca6df | ||
|  | 3cf934b920 | ||
|  | f68ea3edb0 | ||
|  | 16321c9461 | ||
|  | f9530f6d9a | ||
| 83f2f0834d | |||
| 6632aa8f95 | |||
| 8a558c3121 | |||
|  | d434fde92e | ||
| 5afb60df32 | |||
| 3394e6eb01 | |||
| 3818c2c059 | |||
| 0afc543b5f | |||
| adee46e3fc | |||
| 1db74867e6 | |||
| 0f24a63d32 | |||
| 3640809502 | 
							
								
								
									
										142
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						| @@ -1,5 +1,147 @@ | |||||||
| # Changelog | # Changelog | ||||||
|  |  | ||||||
|  | ## [1.4.9] - 2025-03-29 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.9 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - enhance HTTP method handling in sendToApi function | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.8] - 2025-03-29 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.8 | ||||||
|  | - Merge pull request #30 from janecker/main | ||||||
|  | - Merge branch 'testing' into main | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - improve HTTP client configuration and clear update documents after API calls | ||||||
|  | - Fixes memory leak in HTTPClient by disabling connection reuse | ||||||
|  | - update reload logic after removing and saving Bambu credentials for better cache handling | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.7] - 2025-03-27 | ||||||
|  | ### Added | ||||||
|  | - add forced cache refresh after removing and saving Bambu credentials | ||||||
|  | - add functionality to remove Bambu credentials and update API handling | ||||||
|  | - add rfid_bambu.html and update bambu connection handling | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.7 | ||||||
|  | - Merge branch 'testing' | ||||||
|  | - update remove button for Bambu credentials with red background | ||||||
|  | - Merge pull request #28 from tugsi/main | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - handle Bambu connection state by introducing bambuDisabled flag | ||||||
|  | - Fix rfid.js-Failure with X1-Series, if you wanna send a Spool to AMS:  - Uncaught TypeError: Cannot read properties of undefined (reading 'replace')     at handleSpoolIn (rfid.js:493:67)     at HTMLButtonElement.onclick ((Index):1:1) handleSpoolIn	@	rfid.js:493 onclick	@	(Index):1 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.6] - 2025-03-26 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.6 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - handle potential undefined value for tray_info_idx in handleSpoolIn function, by @tugsi | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.5] - 2025-03-25 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.5 | ||||||
|  | - Merge branch 'testing' | ||||||
|  | - remove unused request_topic subscription and reduce MQTT task stack size | ||||||
|  | - Merge pull request #26 from tugsi/main | ||||||
|  | - rename report_topic to topic and update MQTT subscription logic, switched publish topic to request | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - increase MQTT buffer size and adjust task stack size | ||||||
|  | - Fix BufferSize for larger JSONs from X-Series | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.4] - 2025-03-23 | ||||||
|  | ### Added | ||||||
|  | - add error handling for missing vendor IDs in filament data | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.4 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - adjust weight threshold for tare check to allow negative values | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.3] - 2025-03-23 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.3 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.2] - 2025-03-23 | ||||||
|  | ### Added | ||||||
|  | - add WiFi connection check and restart Bambu if not connected | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.2 | ||||||
|  | - increase stack size for BambuMqtt task | ||||||
|  | - update Discord Link | ||||||
|  | - update Discord Link | ||||||
|  | - remove commented-out subscription topic in MQTT setup | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - use unique client ID for MQTT connection to avoid conflicts | ||||||
|  | - reload page after firmware update completion | ||||||
|  | - increase WiFi connection timeout from 5 to 10 seconds | ||||||
|  | - ensure valid URL format and remove trailing slash in setupWebserver | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.4.1] - 2025-03-10 | ||||||
|  | ### Added | ||||||
|  | - added new .step, now with correct individual parts | ||||||
|  | - added changelog | ||||||
|  | - Add files via upload | ||||||
|  | - added .stp files of modifications | ||||||
|  | - added merged picture | ||||||
|  | - added pictures of components bought from AliE | ||||||
|  | - Add files via upload | ||||||
|  | - added pictures for heat insert location | ||||||
|  | - added pictures showing heat insert location | ||||||
|  | - remove unnecessary delay in MQTT setup and add delay before restart | ||||||
|  | - add new 3D print file for Filaman scale | ||||||
|  | - added Discord Server | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.4.1 | ||||||
|  | - refactor length calculation to convert total length to meters before formatting | ||||||
|  | - Merge pull request #16 from spitzbirne32/main | ||||||
|  | - improved housing to show display better | ||||||
|  | - removed CAD, as they were all duplicates | ||||||
|  | - typo in AliE link | ||||||
|  | - Delete usermod/spitzbirne32/STL/README.md | ||||||
|  | - Update README.md | ||||||
|  | - moved pictures of parts into dedicated folders | ||||||
|  | - Update README.md | ||||||
|  | - Update README.md | ||||||
|  | - Update README.md | ||||||
|  | - Delete usermod/spitzbirne32/STL/ScaleTop_Heatinsert_Location_usermod_spitzbirne32_.png | ||||||
|  | - Delete usermod/spitzbirne32/STL/Housing_Heatinsert_Location_usermod_spitzbirne32_.png | ||||||
|  | - created folders | ||||||
|  | - Update README.md | ||||||
|  | - Update README.md | ||||||
|  | - Create README.md | ||||||
|  | - Update README.md | ||||||
|  | - Update README.md | ||||||
|  | - Create README.md | ||||||
|  | - Merge pull request #15 from ManuelW77/main | ||||||
|  | - Merge pull request #14 from janecker/scale-calibration-rework | ||||||
|  | - Reworks the scale calibration handling | ||||||
|  | - remove redundant scale calibration checks and enhance task management | ||||||
|  | - enhance AMS data handling and streamline spool auto-setting logic | ||||||
|  | - adjust stack size and improve scale calibration logic | ||||||
|  | - update labels and input types for better clarity and functionality | ||||||
|  | - update documentation for clarity and accuracy | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - correct typo in console log for total length | ||||||
|  |  | ||||||
|  |  | ||||||
| ## [1.4.0] - 2025-03-01 | ## [1.4.0] - 2025-03-01 | ||||||
| ### Added | ### Added | ||||||
| - add support for Spoolman Octoprint Plugin in README files | - add support for Spoolman Octoprint Plugin in README files | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ Das System integriert sich nahtlos mit der [Spoolman](https://github.com/Donkie/ | |||||||
| Weitere Bilder finden Sie im [img Ordner](/img/) | Weitere Bilder finden Sie im [img Ordner](/img/) | ||||||
| oder auf meiner Website: [FilaMan Website](https://www.filaman.app)   | oder auf meiner Website: [FilaMan Website](https://www.filaman.app)   | ||||||
| Deutsches Erklärvideo: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaOHU)   | Deutsches Erklärvideo: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaOHU)   | ||||||
|  | Discord Server: [https://discord.gg/my7Gvaxj2v](https://discord.gg/my7Gvaxj2v) | ||||||
|  |  | ||||||
| ### Es gibt jetzt auch ein Wiki, dort sind nochmal alle Funktionen beschrieben: [Wiki](https://github.com/ManuelW77/Filaman/wiki) | ### Es gibt jetzt auch ein Wiki, dort sind nochmal alle Funktionen beschrieben: [Wiki](https://github.com/ManuelW77/Filaman/wiki) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,12 +6,14 @@ FilaMan is a filament management system for 3D printing. It uses ESP32 hardware | |||||||
| Users can manage filament spools, monitor the status of the Automatic Material System (AMS) and make settings via a web interface.  | 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. | 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. | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| More Images can be found in the [img Folder](/img/)   | More Images can be found in the [img Folder](/img/)   | ||||||
| or my website:[FilaMan Website](https://www.filaman.app)   | or my website: [FilaMan Website](https://www.filaman.app)   | ||||||
| german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaOHU)   | german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaOHU)   | ||||||
|  | Discord Server: [https://discord.gg/my7Gvaxj2v](https://discord.gg/my7Gvaxj2v) | ||||||
|  |  | ||||||
| ### Now more detailed informations about the usage: [Wiki](https://github.com/ManuelW77/Filaman/wiki) | ### Now more detailed informations about the usage: [Wiki](https://github.com/ManuelW77/Filaman/wiki) | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										15297
									
								
								_3D Print Files/FilaMan-Waage.step
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								_3D Print Files/Filaman-Waage.f3z
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -141,15 +141,6 @@ | |||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
|         <!-- Rechte Spalte --> |  | ||||||
|         <div class="column"> |  | ||||||
|             <div class="feature-box"> |  | ||||||
|                 <h2>Bambu AMS</h2> |  | ||||||
|                 <div id="amsDataContainer"> |  | ||||||
|                     <div class="amsData" id="amsData">Wait for AMS-Data...</div> |  | ||||||
|                 </div> |  | ||||||
|             </div> |  | ||||||
|         </div> |  | ||||||
|     </div> |     </div> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -208,7 +208,7 @@ document.addEventListener('spoolmanError', function(event) { | |||||||
|     showNotification(`Spoolman Error: ${event.detail.message}`, false); |     showNotification(`Spoolman Error: ${event.detail.message}`, false); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| document.addEventListener('filamentSelected', function(event) { | document.addEventListener('filamentSelected', function (event) { | ||||||
|     updateNfcInfo(); |     updateNfcInfo(); | ||||||
|     // Zeige Spool-Buttons wenn ein Filament ausgewählt wurde |     // Zeige Spool-Buttons wenn ein Filament ausgewählt wurde | ||||||
|     const selectedText = document.getElementById("selected-filament").textContent; |     const selectedText = document.getElementById("selected-filament").textContent; | ||||||
| @@ -490,7 +490,7 @@ function handleSpoolIn(amsId, trayId) { | |||||||
|             nozzle_temp_max: parseInt(maxTemp), |             nozzle_temp_max: parseInt(maxTemp), | ||||||
|             type: selectedSpool.filament.material, |             type: selectedSpool.filament.material, | ||||||
|             brand: selectedSpool.filament.vendor.name, |             brand: selectedSpool.filament.vendor.name, | ||||||
|             tray_info_idx: selectedSpool.filament.extra.bambu_idx.replace(/['"]+/g, '').trim(), |             tray_info_idx: selectedSpool.filament.extra.bambu_idx?.replace(/['"]+/g, '').trim() || '', | ||||||
|             cali_idx: "-1"  // Default-Wert setzen |             cali_idx: "-1"  // Default-Wert setzen | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
|   | |||||||
							
								
								
									
										160
									
								
								html/rfid_bambu.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,160 @@ | |||||||
|  | <!-- 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> | ||||||
|  |  | ||||||
|  |         <!-- Rechte Spalte --> | ||||||
|  |         <div class="column"> | ||||||
|  |             <div class="feature-box"> | ||||||
|  |                 <h2>Bambu AMS</h2> | ||||||
|  |                 <div id="amsDataContainer"> | ||||||
|  |                     <div class="amsData" id="amsData">Wait for AMS-Data...</div> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |     </div> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | <script src="spoolman.js"></script> | ||||||
|  | <script src="rfid.js"></script> | ||||||
|  |  | ||||||
|  | </body> | ||||||
|  | </html> | ||||||
| @@ -57,6 +57,31 @@ | |||||||
|             toggleOctoFields(); |             toggleOctoFields(); | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|  |         function removeBambuCredentials() { | ||||||
|  |             fetch('/api/bambu?remove=true') | ||||||
|  |                 .then(response => response.json()) | ||||||
|  |                 .then(data => { | ||||||
|  |                     if (data.success) { | ||||||
|  |                         document.getElementById('bambuIp').value = ''; | ||||||
|  |                         document.getElementById('bambuSerial').value = ''; | ||||||
|  |                         document.getElementById('bambuCode').value = ''; | ||||||
|  |                         document.getElementById('autoSend').checked = false; | ||||||
|  |                         document.getElementById('autoSendTime').value = ''; | ||||||
|  |                         document.getElementById('bambuStatusMessage').innerText = 'Bambu Credentials removed!'; | ||||||
|  |                         // Reload with forced cache refresh after short delay | ||||||
|  |                         setTimeout(() => { | ||||||
|  |                             window.location.reload(true); | ||||||
|  |                             window.location.href = '/'; | ||||||
|  |                         }, 1500); | ||||||
|  |                     } else { | ||||||
|  |                         document.getElementById('bambuStatusMessage').innerText = 'Error while removing Bambu Credentials.'; | ||||||
|  |                     } | ||||||
|  |                 }) | ||||||
|  |                 .catch(error => { | ||||||
|  |                     document.getElementById('bambuStatusMessage').innerText = 'Error while removing: ' + error.message; | ||||||
|  |                 }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         function checkSpoolmanInstance() { |         function checkSpoolmanInstance() { | ||||||
|             const url = document.getElementById('spoolmanUrl').value; |             const url = document.getElementById('spoolmanUrl').value; | ||||||
|             const spoolmanOctoEnabled = document.getElementById('spoolmanOctoEnabled').checked; |             const spoolmanOctoEnabled = document.getElementById('spoolmanOctoEnabled').checked; | ||||||
| @@ -89,6 +114,11 @@ | |||||||
|                 .then(data => { |                 .then(data => { | ||||||
|                     if (data.healthy) { |                     if (data.healthy) { | ||||||
|                         document.getElementById('bambuStatusMessage').innerText = 'Bambu Credentials saved!'; |                         document.getElementById('bambuStatusMessage').innerText = 'Bambu Credentials saved!'; | ||||||
|  |                         // Reload with forced cache refresh after short delay | ||||||
|  |                         setTimeout(() => { | ||||||
|  |                             window.location.reload(true); | ||||||
|  |                             window.location.href = '/'; | ||||||
|  |                         }, 1500); | ||||||
|                     } else { |                     } else { | ||||||
|                         document.getElementById('bambuStatusMessage').innerText = 'Error while saving Bambu Credentials.'; |                         document.getElementById('bambuStatusMessage').innerText = 'Error while saving Bambu Credentials.'; | ||||||
|                     } |                     } | ||||||
| @@ -154,14 +184,15 @@ | |||||||
|                     <p>If activated, FilaMan will automatically update the next filled tray with the last scanned and weighed spool.</p> |                     <p>If activated, FilaMan will automatically update the next filled tray with the last scanned and weighed spool.</p> | ||||||
|                     <div class="input-group" style="display: flex; margin-bottom: 0;"> |                     <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="autoSend" style="width: 250px; margin-right: 5px;">Auto Send to Bambu:</label> | ||||||
|                         <label for="autoSendTime" style="width: 250px; margin-right: 5px;">Wait time in Seconds:</label> |                         <label for="autoSendTime" style="width: 250px; margin-right: 5px;">Wait for Spool in Sec:</label> | ||||||
|                     </div> |                     </div> | ||||||
|                     <div class="input-group" style="display: flex;"> |                     <div class="input-group" style="display: flex;"> | ||||||
|                         <input type="checkbox" id="autoSend" {{autoSendToBambu}} style="width: 190px; margin-right: 10px;"> |                         <input type="checkbox" id="autoSend" {{autoSendToBambu}} style="width: 190px; margin-right: 10px;"> | ||||||
|                         <input type="text" id="autoSendTime" placeholder="Time to wait for new Spool" value="{{autoSendTime}}" style="width: 100px;"> |                         <input type="number" min="60" id="autoSendTime" placeholder="Time to wait" value="{{autoSendTime}}" style="width: 100px;"> | ||||||
|                     </div> |                     </div> | ||||||
|  |  | ||||||
|                     <button style="margin: 0;" onclick="saveBambuCredentials()">Save Bambu Credentials</button> |                     <button style="margin: 0;" onclick="saveBambuCredentials()">Save Bambu Credentials</button> | ||||||
|  |                     <button style="margin: 0; background-color: red;" onclick="removeBambuCredentials()">Remove Credentials</button> | ||||||
|                     <p id="bambuStatusMessage"></p> |                     <p id="bambuStatusMessage"></p> | ||||||
|                 </div> |                 </div> | ||||||
|             </div> |             </div> | ||||||
|   | |||||||
| @@ -86,10 +86,10 @@ function populateVendorDropdown(data, selectedSmId = null) { | |||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     // Nach der Schleife: Formatierung der Gesamtlänge |     // Nach der Schleife: Formatierung der Gesamtlänge | ||||||
|     console.log("Total Lenght: ", totalLength); |     const lengthInM = totalLength / 1000;  // erst in m umrechnen | ||||||
|     const formattedLength = totalLength > 1000  |     const formattedLength = lengthInM > 1000  | ||||||
|         ? (totalLength / 1000).toFixed(2) + " km"  |         ? (lengthInM / 1000).toFixed(2) + " km"  | ||||||
|         : totalLength.toFixed(2) + " m"; |         : lengthInM.toFixed(2) + " m"; | ||||||
|  |  | ||||||
|     // Formatierung des Gesamtgewichts (von g zu kg zu t) |     // Formatierung des Gesamtgewichts (von g zu kg zu t) | ||||||
|     const weightInKg = totalWeight / 1000;  // erst in kg umrechnen |     const weightInKg = totalWeight / 1000;  // erst in kg umrechnen | ||||||
| @@ -147,6 +147,13 @@ function updateFilamentDropdown(selectedSmId = null) { | |||||||
|  |  | ||||||
|     if (vendorId) { |     if (vendorId) { | ||||||
|         const filteredFilaments = spoolsData.filter(spool => { |         const filteredFilaments = spoolsData.filter(spool => { | ||||||
|  |             if (!spool?.filament?.vendor?.id) { | ||||||
|  |                 console.log('Problem aufgetreten bei: ', spool?.filament?.vendor); | ||||||
|  |                 console.log('Problematische Spulen:',  | ||||||
|  |                     spoolsData.filter(spool => !spool?.filament?.vendor?.id)); | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |  | ||||||
|             const hasValidNfcId = spool.extra &&  |             const hasValidNfcId = spool.extra &&  | ||||||
|                                  spool.extra.nfc_id &&  |                                  spool.extra.nfc_id &&  | ||||||
|                                  spool.extra.nfc_id !== '""' &&  |                                  spool.extra.nfc_id !== '""' &&  | ||||||
| @@ -240,18 +247,6 @@ 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"); |  | ||||||
| }; |  | ||||||
| */ |  | ||||||
|  |  | ||||||
| // Event Listener | // Event Listener | ||||||
| document.addEventListener('DOMContentLoaded', () => { | document.addEventListener('DOMContentLoaded', () => { | ||||||
|     initSpoolman(); |     initSpoolman(); | ||||||
|   | |||||||
| @@ -188,14 +188,18 @@ label { | |||||||
|     font-weight: bold;  |     font-weight: bold;  | ||||||
| } | } | ||||||
|  |  | ||||||
| input[type="text"], input[type="submit"] {  | input[type="text"], input[type="submit"], input[type="number"] {  | ||||||
|     padding: 10px;  |     padding: 10px;  | ||||||
|     border: 1px solid #ccc;  |     border: 1px solid #ccc;  | ||||||
|     border-radius: 5px;  |     border-radius: 5px;  | ||||||
|     font-size: 16px;  |     font-size: 16px;  | ||||||
| } | } | ||||||
|  |  | ||||||
| input[type="text"]:focus {  | input[type="number"] {  | ||||||
|  |     width: 108px !important;  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | input[type="text"]:focus, input[type="number"]:focus {  | ||||||
|     border-color: #007bff;  |     border-color: #007bff;  | ||||||
|     outline: none;  |     outline: none;  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -129,6 +129,7 @@ | |||||||
|                             if (data.status === 'success' || lastReceivedProgress >= 98) { |                             if (data.status === 'success' || lastReceivedProgress >= 98) { | ||||||
|                                 clearTimeout(wsReconnectTimer); |                                 clearTimeout(wsReconnectTimer); | ||||||
|                                 setTimeout(() => { |                                 setTimeout(() => { | ||||||
|  |                                     window.location.reload(true); | ||||||
|                                     window.location.href = '/'; |                                     window.location.href = '/'; | ||||||
|                                 }, 30000); |                                 }, 30000); | ||||||
|                             } |                             } | ||||||
| @@ -148,6 +149,7 @@ | |||||||
|                         status.style.display = 'block'; |                         status.style.display = 'block'; | ||||||
|                         clearTimeout(wsReconnectTimer); |                         clearTimeout(wsReconnectTimer); | ||||||
|                         setTimeout(() => { |                         setTimeout(() => { | ||||||
|  |                             window.location.reload(true); | ||||||
|                             window.location.href = '/'; |                             window.location.href = '/'; | ||||||
|                         }, 30000); |                         }, 30000); | ||||||
|                     } else { |                     } else { | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								img/7-enable.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 52 KiB | 
| @@ -9,7 +9,7 @@ | |||||||
| ; https://docs.platformio.org/page/projectconf.html | ; https://docs.platformio.org/page/projectconf.html | ||||||
|  |  | ||||||
| [common] | [common] | ||||||
| version = "1.4.0" | version = "1.4.9" | ||||||
| to_old_version = "1.4.0" | to_old_version = "1.4.0" | ||||||
|  |  | ||||||
| ## | ## | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								src/api.cpp
									
									
									
									
									
								
							
							
						
						| @@ -94,13 +94,16 @@ void sendToApi(void *parameter) { | |||||||
|     String octoToken = params->octoToken;     |     String octoToken = params->octoToken;     | ||||||
|  |  | ||||||
|     HTTPClient http; |     HTTPClient http; | ||||||
|  |     http.setReuse(false); | ||||||
|  |  | ||||||
|     http.begin(spoolsUrl); |     http.begin(spoolsUrl); | ||||||
|     http.addHeader("Content-Type", "application/json"); |     http.addHeader("Content-Type", "application/json"); | ||||||
|     if (octoEnabled && octoToken != "") http.addHeader("X-Api-Key", octoToken); |     if (octoEnabled && octoToken != "") http.addHeader("X-Api-Key", octoToken); | ||||||
|  |  | ||||||
|     int httpCode = http.PUT(updatePayload); |     int httpCode; | ||||||
|     if (httpType == "PATCH") httpCode = http.PATCH(updatePayload); |     if (httpType == "PATCH") httpCode = http.PATCH(updatePayload); | ||||||
|     if (httpType == "POST") httpCode = http.POST(updatePayload); |     else if (httpType == "POST") httpCode = http.POST(updatePayload); | ||||||
|  |     else httpCode = http.PUT(updatePayload); | ||||||
|  |  | ||||||
|     if (httpCode == HTTP_CODE_OK) { |     if (httpCode == HTTP_CODE_OK) { | ||||||
|         Serial.println("Spoolman erfolgreich aktualisiert"); |         Serial.println("Spoolman erfolgreich aktualisiert"); | ||||||
| @@ -111,6 +114,7 @@ void sendToApi(void *parameter) { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     http.end(); |     http.end(); | ||||||
|  |     vTaskDelay(50 / portTICK_PERIOD_MS); | ||||||
|  |  | ||||||
|     // Speicher freigeben |     // Speicher freigeben | ||||||
|     delete params; |     delete params; | ||||||
| @@ -165,6 +169,8 @@ bool updateSpoolTagId(String uidString, const char* payload) { | |||||||
|         NULL                      // Task-Handle (nicht benötigt) |         NULL                      // Task-Handle (nicht benötigt) | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |     updateDoc.clear(); | ||||||
|  |  | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -201,6 +207,8 @@ uint8_t updateSpoolWeight(String spoolId, uint16_t weight) { | |||||||
|         NULL                      // Task-Handle (nicht benötigt) |         NULL                      // Task-Handle (nicht benötigt) | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |     updateDoc.clear(); | ||||||
|  |  | ||||||
|     return 1; |     return 1; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -238,6 +246,8 @@ bool updateSpoolOcto(int spoolId) { | |||||||
|         NULL                      // Task-Handle (nicht benötigt) |         NULL                      // Task-Handle (nicht benötigt) | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |     updateDoc.clear(); | ||||||
|  |  | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ PubSubClient client(sslClient); | |||||||
|  |  | ||||||
| TaskHandle_t BambuMqttTask; | TaskHandle_t BambuMqttTask; | ||||||
|  |  | ||||||
| String report_topic = ""; | String topic = ""; | ||||||
| //String request_topic = ""; | //String request_topic = ""; | ||||||
| const char* bambu_username = "bblp"; | const char* bambu_username = "bblp"; | ||||||
| const char* bambu_ip = nullptr; | const char* bambu_ip = nullptr; | ||||||
| @@ -27,6 +27,7 @@ const char* bambu_serialnr = nullptr; | |||||||
| String g_bambu_ip = ""; | String g_bambu_ip = ""; | ||||||
| String g_bambu_accesscode = ""; | String g_bambu_accesscode = ""; | ||||||
| String g_bambu_serialnr = ""; | String g_bambu_serialnr = ""; | ||||||
|  | bool bambuDisabled = false; | ||||||
|  |  | ||||||
| bool bambu_connected = false; | bool bambu_connected = false; | ||||||
| bool autoSendToBambu = false; | bool autoSendToBambu = false; | ||||||
| @@ -37,6 +38,32 @@ int ams_count = 0; | |||||||
| String amsJsonData;  // Speichert das fertige JSON für WebSocket-Clients | String amsJsonData;  // Speichert das fertige JSON für WebSocket-Clients | ||||||
| AMSData ams_data[MAX_AMS];  // Definition des Arrays; | AMSData ams_data[MAX_AMS];  // Definition des Arrays; | ||||||
|  |  | ||||||
|  | bool removeBambuCredentials() { | ||||||
|  |     if (BambuMqttTask) { | ||||||
|  |         vTaskDelete(BambuMqttTask); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     if (!removeJsonValue("/bambu_credentials.json")) { | ||||||
|  |         Serial.println("Fehler beim Löschen der Bambu-Credentials."); | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |     // Löschen der globalen Variablen | ||||||
|  |     g_bambu_ip = ""; | ||||||
|  |     g_bambu_accesscode = ""; | ||||||
|  |     g_bambu_serialnr = ""; | ||||||
|  |     bambu_ip = nullptr; | ||||||
|  |     bambu_accesscode = nullptr; | ||||||
|  |     bambu_serialnr = nullptr; | ||||||
|  |     autoSendToBambu = false; | ||||||
|  |     autoSetToBambuSpoolId = 0; | ||||||
|  |     ams_count = 0; | ||||||
|  |     amsJsonData = ""; | ||||||
|  |  | ||||||
|  |     bambuDisabled = true; | ||||||
|  |  | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  |  | ||||||
| bool saveBambuCredentials(const String& ip, const String& serialnr, const String& accesscode, bool autoSend, const String& autoSendTime) { | bool saveBambuCredentials(const String& ip, const String& serialnr, const String& accesscode, bool autoSend, const String& autoSendTime) { | ||||||
|     if (BambuMqttTask) { |     if (BambuMqttTask) { | ||||||
|         vTaskDelete(BambuMqttTask); |         vTaskDelete(BambuMqttTask); | ||||||
| @@ -91,7 +118,7 @@ bool loadBambuCredentials() { | |||||||
|         bambu_accesscode = g_bambu_accesscode.c_str(); |         bambu_accesscode = g_bambu_accesscode.c_str(); | ||||||
|         bambu_serialnr = g_bambu_serialnr.c_str(); |         bambu_serialnr = g_bambu_serialnr.c_str(); | ||||||
|  |  | ||||||
|         report_topic = "device/" + String(bambu_serialnr) + "/report"; |         topic = "device/" + String(bambu_serialnr); | ||||||
|         //request_topic = "device/" + String(bambu_serialnr) + "/request"; |         //request_topic = "device/" + String(bambu_serialnr) + "/request"; | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| @@ -199,7 +226,7 @@ FilamentResult findFilamentIdx(String brand, String type) { | |||||||
| bool sendMqttMessage(const String& payload) { | bool sendMqttMessage(const String& payload) { | ||||||
|     Serial.println("Sending MQTT message"); |     Serial.println("Sending MQTT message"); | ||||||
|     Serial.println(payload); |     Serial.println(payload); | ||||||
|     if (client.publish(report_topic.c_str(), payload.c_str()))  |     if (client.publish((String(topic) + "/request").c_str(), payload.c_str()))  | ||||||
|     { |     { | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| @@ -341,7 +368,7 @@ void updateAmsWsData(JsonDocument& doc, JsonArray& amsArray, int& ams_count, Jso | |||||||
|             ams_data[i].trays[j].tray_color = trayObj["tray_color"].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_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].nozzle_temp_max = trayObj["nozzle_temp_max"].as<int>(); | ||||||
|             //ams_data[i].trays[j].setting_id = trayObj["setting_id"].as<String>(); |             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>(); |             ams_data[i].trays[j].cali_idx = trayObj["cali_idx"].as<String>(); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -425,16 +452,8 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | |||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Wenn bambu auto set spool aktiv und eine spule erkannt und mqtt meldung das neue spule im ams |  | ||||||
|     if (autoSendToBambu && autoSetToBambuSpoolId > 0 &&  |  | ||||||
|         doc["print"]["command"].as<String>() == "push_status" && doc["print"]["ams"]["tray_pre"].as<uint8_t>() |  | ||||||
|         && !doc["print"]["ams"]["ams"].as<JsonArray>()) |  | ||||||
|     { |  | ||||||
|         autoSetSpool(autoSetToBambuSpoolId, doc["print"]["ams"]["tray_pre"].as<uint8_t>()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // Prüfen, ob "print->upgrade_state" und "print.ams.ams" existieren |     // Prüfen, ob "print->upgrade_state" und "print.ams.ams" existieren | ||||||
|     if (doc["print"]["upgrade_state"].is<JsonObject>())  |     if (doc["print"]["upgrade_state"].is<JsonObject>() || (doc["print"]["command"].is<String>() && doc["print"]["command"] == "push_status"))  | ||||||
|     { |     { | ||||||
|         // Prüfen ob AMS-Daten vorhanden sind |         // Prüfen ob AMS-Daten vorhanden sind | ||||||
|         if (!doc["print"]["ams"].is<JsonObject>() || !doc["print"]["ams"]["ams"].is<JsonArray>())  |         if (!doc["print"]["ams"].is<JsonObject>() || !doc["print"]["ams"]["ams"].is<JsonArray>())  | ||||||
| @@ -479,6 +498,12 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | |||||||
|                     (trayObj["setting_id"].as<String>() != "" && trayObj["setting_id"].as<String>() != ams_data[storedIndex].trays[j].setting_id) || |                     (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) { |                     trayObj["cali_idx"].as<String>() != ams_data[storedIndex].trays[j].cali_idx) { | ||||||
|                     hasChanges = true; |                     hasChanges = true; | ||||||
|  |  | ||||||
|  |                     if (autoSendToBambu && autoSetToBambuSpoolId > 0 && hasChanges) | ||||||
|  |                     { | ||||||
|  |                         autoSetSpool(autoSetToBambuSpoolId, ams_data[storedIndex].trays[j].id); | ||||||
|  |                     } | ||||||
|  |  | ||||||
|                     break; |                     break; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -497,6 +522,11 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | |||||||
|                         (vtTray["setting_id"].as<String>() != "" && vtTray["setting_id"].as<String>() != ams_data[i].trays[0].setting_id) || |                         (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)) { |                         (vtTray["tray_type"].as<String>() != "" && vtTray["cali_idx"].as<String>() != ams_data[i].trays[0].cali_idx)) { | ||||||
|                         hasChanges = true; |                         hasChanges = true; | ||||||
|  |  | ||||||
|  |                         if (autoSendToBambu && autoSetToBambuSpoolId > 0 && hasChanges) | ||||||
|  |                         { | ||||||
|  |                             autoSetSpool(autoSetToBambuSpoolId, 254); | ||||||
|  |                         } | ||||||
|                     } |                     } | ||||||
|                     break; |                     break; | ||||||
|                 } |                 } | ||||||
| @@ -550,10 +580,11 @@ void reconnect() { | |||||||
|         oledShowTopRow(); |         oledShowTopRow(); | ||||||
|  |  | ||||||
|         // Attempt to connect |         // Attempt to connect | ||||||
|         if (client.connect(bambu_serialnr, bambu_username, bambu_accesscode)) { |         String clientId = String(bambu_serialnr) + "_" + String(random(0, 100)); | ||||||
|  |         if (client.connect(clientId.c_str(), bambu_username, bambu_accesscode)) { | ||||||
|             Serial.println("MQTT re/connected"); |             Serial.println("MQTT re/connected"); | ||||||
|  |  | ||||||
|             client.subscribe(report_topic.c_str()); |             client.subscribe((String(topic) + "/report").c_str()); | ||||||
|             bambu_connected = true; |             bambu_connected = true; | ||||||
|             oledShowTopRow(); |             oledShowTopRow(); | ||||||
|         } else { |         } else { | ||||||
| @@ -600,30 +631,27 @@ void mqtt_loop(void * parameter) { | |||||||
| bool setupMqtt() { | bool setupMqtt() { | ||||||
|     // Wenn Bambu Daten vorhanden |     // Wenn Bambu Daten vorhanden | ||||||
|     bool success = loadBambuCredentials(); |     bool success = loadBambuCredentials(); | ||||||
|     vTaskDelay(100 / portTICK_PERIOD_MS); |  | ||||||
|  |  | ||||||
|     if (!success) { |     if (!success) { | ||||||
|         Serial.println("Failed to load Bambu credentials"); |         bambuDisabled = true; | ||||||
|         oledShowMessage("Bambu Credentials Missing"); |  | ||||||
|         vTaskDelay(2000 / portTICK_PERIOD_MS); |  | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (success && bambu_ip != "" && bambu_accesscode != "" && bambu_serialnr != "")  |     if (success && bambu_ip != "" && bambu_accesscode != "" && bambu_serialnr != "")  | ||||||
|     { |     { | ||||||
|  |         bambuDisabled = false; | ||||||
|         sslClient.setCACert(root_ca); |         sslClient.setCACert(root_ca); | ||||||
|         sslClient.setInsecure(); |         sslClient.setInsecure(); | ||||||
|         client.setServer(bambu_ip, 8883); |         client.setServer(bambu_ip, 8883); | ||||||
|  |  | ||||||
|         // Verbinden mit dem MQTT-Server |         // Verbinden mit dem MQTT-Server | ||||||
|         bool connected = true; |         bool connected = true; | ||||||
|         if (client.connect(bambu_serialnr, bambu_username, bambu_accesscode))  |         String clientId = String(bambu_serialnr) + "_" + String(random(0, 100)); | ||||||
|  |         if (client.connect(clientId.c_str(), bambu_username, bambu_accesscode))  | ||||||
|         { |         { | ||||||
|             client.setCallback(mqtt_callback); |             client.setCallback(mqtt_callback); | ||||||
|             client.setBufferSize(5120); |             client.setBufferSize(15488); | ||||||
|             // Optional: Topic abonnieren |             client.subscribe((String(topic) + "/report").c_str()); | ||||||
|             client.subscribe(report_topic.c_str()); |  | ||||||
|             //client.subscribe(request_topic.c_str()); |  | ||||||
|             Serial.println("MQTT-Client initialisiert"); |             Serial.println("MQTT-Client initialisiert"); | ||||||
|  |  | ||||||
|             oledShowMessage("Bambu Connected"); |             oledShowMessage("Bambu Connected"); | ||||||
| @@ -652,10 +680,7 @@ bool setupMqtt() { | |||||||
|     }  |     }  | ||||||
|     else  |     else  | ||||||
|     { |     { | ||||||
|         Serial.println("Fehler: Keine MQTT-Daten vorhanden"); |         bambuDisabled = true; | ||||||
|         oledShowMessage("Bambu Credentials Missing"); |  | ||||||
|         oledShowTopRow(); |  | ||||||
|         vTaskDelay(2000 / portTICK_PERIOD_MS); |  | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|     return true; |     return true; | ||||||
| @@ -664,6 +689,7 @@ bool setupMqtt() { | |||||||
| void bambu_restart() { | void bambu_restart() { | ||||||
|     if (BambuMqttTask) { |     if (BambuMqttTask) { | ||||||
|         vTaskDelete(BambuMqttTask); |         vTaskDelete(BambuMqttTask); | ||||||
|  |         delay(10); | ||||||
|     } |     } | ||||||
|     setupMqtt(); |     setupMqtt(); | ||||||
| } | } | ||||||
| @@ -30,7 +30,9 @@ extern int ams_count; | |||||||
| extern AMSData ams_data[MAX_AMS]; | extern AMSData ams_data[MAX_AMS]; | ||||||
| extern bool autoSendToBambu; | extern bool autoSendToBambu; | ||||||
| extern int autoSetToBambuSpoolId; | extern int autoSetToBambuSpoolId; | ||||||
|  | extern bool bambuDisabled; | ||||||
|  |  | ||||||
|  | bool removeBambuCredentials(); | ||||||
| bool loadBambuCredentials(); | bool loadBambuCredentials(); | ||||||
| bool saveBambuCredentials(const String& bambu_ip, const String& bambu_serialnr, const String& bambu_accesscode, const bool autoSend, const String& autoSendTime); | bool saveBambuCredentials(const String& bambu_ip, const String& bambu_serialnr, const String& bambu_accesscode, const bool autoSend, const String& autoSendTime); | ||||||
| bool setupMqtt(); | bool setupMqtt(); | ||||||
|   | |||||||
| @@ -1,6 +1,20 @@ | |||||||
| #include "commonFS.h" | #include "commonFS.h" | ||||||
| #include <LittleFS.h> | #include <LittleFS.h> | ||||||
|  |  | ||||||
|  | bool removeJsonValue(const char* filename) { | ||||||
|  |     File file = LittleFS.open(filename, "r"); | ||||||
|  |     if (!file) { | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |     file.close(); | ||||||
|  |     if (!LittleFS.remove(filename)) { | ||||||
|  |         Serial.print("Fehler beim Löschen der Datei: "); | ||||||
|  |         Serial.println(filename); | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  |  | ||||||
| bool saveJsonValue(const char* filename, const JsonDocument& doc) { | bool saveJsonValue(const char* filename, const JsonDocument& doc) { | ||||||
|     File file = LittleFS.open(filename, "w"); |     File file = LittleFS.open(filename, "w"); | ||||||
|     if (!file) { |     if (!file) { | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ | |||||||
| #include <ArduinoJson.h> | #include <ArduinoJson.h> | ||||||
| #include <LittleFS.h> | #include <LittleFS.h> | ||||||
|  |  | ||||||
|  | bool removeJsonValue(const char* filename); | ||||||
| bool saveJsonValue(const char* filename, const JsonDocument& doc); | bool saveJsonValue(const char* filename, const JsonDocument& doc); | ||||||
| bool loadJsonValue(const char* filename, JsonDocument& doc); | bool loadJsonValue(const char* filename, JsonDocument& doc); | ||||||
| void initializeFileSystem(); | void initializeFileSystem(); | ||||||
|   | |||||||
							
								
								
									
										62
									
								
								src/main.cpp
									
									
									
									
									
								
							
							
						
						| @@ -13,6 +13,9 @@ | |||||||
| #include "esp_task_wdt.h" | #include "esp_task_wdt.h" | ||||||
| #include "commonFS.h" | #include "commonFS.h" | ||||||
|  |  | ||||||
|  | bool mainTaskWasPaused = 0; | ||||||
|  | uint8_t scaleTareCounter = 0; | ||||||
|  |  | ||||||
| // ##### SETUP ##### | // ##### SETUP ##### | ||||||
| void setup() { | void setup() { | ||||||
|   Serial.begin(115200); |   Serial.begin(115200); | ||||||
| @@ -45,22 +48,7 @@ void setup() { | |||||||
|   // NFC Reader |   // NFC Reader | ||||||
|   startNfc(); |   startNfc(); | ||||||
|  |  | ||||||
|   uint8_t scaleCalibrated = start_scale(); |   start_scale(); | ||||||
|   if (scaleCalibrated == 3) { |  | ||||||
|     oledShowMessage("Scale not calibrated!"); |  | ||||||
|     for (uint16_t i = 0; i < 50000; i++) { |  | ||||||
|       yield(); |  | ||||||
|       vTaskDelay(pdMS_TO_TICKS(1)); |  | ||||||
|       esp_task_wdt_reset(); |  | ||||||
|     } |  | ||||||
|   } else if (scaleCalibrated == 0) { |  | ||||||
|     oledShowMessage("HX711 not found"); |  | ||||||
|     for (uint16_t i = 0; i < 50000; i++) { |  | ||||||
|       yield(); |  | ||||||
|       vTaskDelay(pdMS_TO_TICKS(1)); |  | ||||||
|       esp_task_wdt_reset(); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   // WDT initialisieren mit 10 Sekunden Timeout |   // WDT initialisieren mit 10 Sekunden Timeout | ||||||
|   bool panic = true; // Wenn true, löst ein WDT-Timeout einen System-Panik aus |   bool panic = true; // Wenn true, löst ein WDT-Timeout einen System-Panik aus | ||||||
| @@ -68,9 +56,6 @@ void setup() { | |||||||
|  |  | ||||||
|   // Aktuellen Task (loopTask) zum Watchdog hinzufügen |   // Aktuellen Task (loopTask) zum Watchdog hinzufügen | ||||||
|   esp_task_wdt_add(NULL); |   esp_task_wdt_add(NULL); | ||||||
|  |  | ||||||
|   // Optional: Andere Tasks zum Watchdog hinzufügen, falls nötig |  | ||||||
|   // esp_task_wdt_add(task_handle); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -107,12 +92,19 @@ void loop() { | |||||||
|   unsigned long currentMillis = millis(); |   unsigned long currentMillis = millis(); | ||||||
|  |  | ||||||
|   // Überprüfe regelmäßig die WLAN-Verbindung |   // Überprüfe regelmäßig die WLAN-Verbindung | ||||||
|   if (intervalElapsed(currentMillis, lastWifiCheckTime, wifiCheckInterval)) { |   if (intervalElapsed(currentMillis, lastWifiCheckTime, wifiCheckInterval))  | ||||||
|  |   { | ||||||
|     checkWiFiConnection(); |     checkWiFiConnection(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Wenn Bambu auto set Spool aktiv |   // Wenn Bambu auto set Spool aktiv | ||||||
|   if (autoSendToBambu && autoSetToBambuSpoolId > 0) { |   if (autoSendToBambu && autoSetToBambuSpoolId > 0)  | ||||||
|  |   { | ||||||
|  |     if (!bambuDisabled && !bambu_connected)  | ||||||
|  |     { | ||||||
|  |       bambu_restart(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     if (intervalElapsed(currentMillis, lastAutoSetBambuAmsTime, autoSetBambuAmsInterval))  |     if (intervalElapsed(currentMillis, lastAutoSetBambuAmsTime, autoSetBambuAmsInterval))  | ||||||
|     { |     { | ||||||
|       if (hasReadRfidTag == 0) |       if (hasReadRfidTag == 0) | ||||||
| @@ -147,9 +139,17 @@ void loop() { | |||||||
|   }  |   }  | ||||||
|  |  | ||||||
|   // Ausgabe der Waage auf Display |   // Ausgabe der Waage auf Display | ||||||
|   if (pauseMainTask == 0 && weight != lastWeight && hasReadRfidTag == 0 && (!autoSendToBambu || autoSetToBambuSpoolId == 0)) |   if(pauseMainTask == 0) | ||||||
|   { |   { | ||||||
|     (weight < 2) ? ((weight < -2) ? oledShowMessage("!! -0") : oledShowWeight(0)) : oledShowWeight(weight); |     if (mainTaskWasPaused || (weight != lastWeight && hasReadRfidTag == 0 && (!autoSendToBambu || autoSetToBambuSpoolId == 0))) | ||||||
|  |     { | ||||||
|  |       (weight < 2) ? ((weight < -2) ? oledShowMessage("!! -0") : oledShowWeight(0)) : oledShowWeight(weight); | ||||||
|  |     } | ||||||
|  |     mainTaskWasPaused = false; | ||||||
|  |   } | ||||||
|  |   else | ||||||
|  |   { | ||||||
|  |     mainTaskWasPaused = true; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -159,13 +159,22 @@ void loop() { | |||||||
|     lastWeightReadTime = currentMillis; |     lastWeightReadTime = currentMillis; | ||||||
|  |  | ||||||
|     // Prüfen ob die Waage korrekt genullt ist |     // Prüfen ob die Waage korrekt genullt ist | ||||||
|     if ((weight > 0 && weight < 5) || weight < 0) |     if ((weight > 0 && weight < 5) || weight < -1) | ||||||
|     { |     { | ||||||
|       scale_tare_counter++; |       if(scaleTareCounter < 5) | ||||||
|  |       { | ||||||
|  |         scaleTareCounter++; | ||||||
|  |       } | ||||||
|  |       else | ||||||
|  |       { | ||||||
|  |         scaleTareRequest = true; | ||||||
|  |         scaleTareCounter = 0; | ||||||
|  |       } | ||||||
|  |        | ||||||
|     } |     } | ||||||
|     else |     else | ||||||
|     { |     { | ||||||
|       scale_tare_counter = 0; |       scaleTareCounter = 0; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Prüfen ob das Gewicht gleich bleibt und dann senden |     // Prüfen ob das Gewicht gleich bleibt und dann senden | ||||||
| @@ -210,6 +219,5 @@ void loop() { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|    |    | ||||||
|   yield(); |  | ||||||
|   esp_task_wdt_reset(); |   esp_task_wdt_reset(); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ TaskHandle_t ScaleTask; | |||||||
| int16_t weight = 0; | int16_t weight = 0; | ||||||
|  |  | ||||||
| uint8_t weigthCouterToApi = 0; | uint8_t weigthCouterToApi = 0; | ||||||
| uint8_t scale_tare_counter = 0; | bool scaleTareRequest = false; | ||||||
| uint8_t pauseMainTask = 0; | uint8_t pauseMainTask = 0; | ||||||
| uint8_t scaleCalibrated = 1; | uint8_t scaleCalibrated = 1; | ||||||
|  |  | ||||||
| @@ -34,30 +34,32 @@ void scale_loop(void * parameter) { | |||||||
|   Serial.println("++++++++++++++++++++++++++++++"); |   Serial.println("++++++++++++++++++++++++++++++"); | ||||||
|   Serial.println("Scale Loop started"); |   Serial.println("Scale Loop started"); | ||||||
|   Serial.println("++++++++++++++++++++++++++++++"); |   Serial.println("++++++++++++++++++++++++++++++"); | ||||||
|  |  | ||||||
|   for(;;) { |   for(;;) { | ||||||
|     if (scale.is_ready())  |     if (scale.is_ready())  | ||||||
|     { |     { | ||||||
|       // Waage nochmal Taren, wenn zu lange Abweichung |       // Waage nochmal Taren, wenn zu lange Abweichung | ||||||
|       if (scale_tare_counter >= 5)  |       if (scaleTareRequest == true)  | ||||||
|       { |       { | ||||||
|  |         Serial.println("Re-Tare scale"); | ||||||
|         scale.tare(); |         scale.tare(); | ||||||
|         scale_tare_counter = 0; |         scaleTareRequest = false; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       weight = round(scale.get_units()); |       weight = round(scale.get_units()); | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     vTaskDelay(pdMS_TO_TICKS(100)); // Verzögerung, um die CPU nicht zu überlasten |     vTaskDelay(pdMS_TO_TICKS(100)); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| uint8_t start_scale() { | void start_scale() { | ||||||
|   Serial.println("Prüfe Calibration Value"); |   Serial.println("Prüfe Calibration Value"); | ||||||
|   long calibrationValue; |   float calibrationValue; | ||||||
|  |  | ||||||
|   // NVS lesen |   // NVS lesen | ||||||
|   preferences.begin(NVS_NAMESPACE, true); // true = readonly |   preferences.begin(NVS_NAMESPACE, true); // true = readonly | ||||||
|   calibrationValue = preferences.getLong(NVS_KEY_CALIBRATION, defaultScaleCalibrationValue); |   calibrationValue = preferences.getFloat(NVS_KEY_CALIBRATION, defaultScaleCalibrationValue); | ||||||
|   preferences.end(); |   preferences.end(); | ||||||
|  |  | ||||||
|   Serial.print("Read Scale Calibration Value "); |   Serial.print("Read Scale Calibration Value "); | ||||||
| @@ -68,6 +70,13 @@ uint8_t start_scale() { | |||||||
|   if (isnan(calibrationValue) || calibrationValue < 1) { |   if (isnan(calibrationValue) || calibrationValue < 1) { | ||||||
|     calibrationValue = defaultScaleCalibrationValue; |     calibrationValue = defaultScaleCalibrationValue; | ||||||
|     scaleCalibrated = 0; |     scaleCalibrated = 0; | ||||||
|  |  | ||||||
|  |     oledShowMessage("Scale not calibrated!"); | ||||||
|  |     for (uint16_t i = 0; i < 50000; i++) { | ||||||
|  |       yield(); | ||||||
|  |       vTaskDelay(pdMS_TO_TICKS(1)); | ||||||
|  |       esp_task_wdt_reset(); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   oledShowMessage("Scale Tare Please remove all"); |   oledShowMessage("Scale Tare Please remove all"); | ||||||
| @@ -90,7 +99,7 @@ uint8_t start_scale() { | |||||||
|   BaseType_t result = xTaskCreatePinnedToCore( |   BaseType_t result = xTaskCreatePinnedToCore( | ||||||
|     scale_loop, /* Function to implement the task */ |     scale_loop, /* Function to implement the task */ | ||||||
|     "ScaleLoop", /* Name of the task */ |     "ScaleLoop", /* Name of the task */ | ||||||
|     10000,  /* Stack size in words */ |     2048,  /* Stack size in words */ | ||||||
|     NULL,  /* Task input parameter */ |     NULL,  /* Task input parameter */ | ||||||
|     scaleTaskPrio,  /* Priority of the task */ |     scaleTaskPrio,  /* Priority of the task */ | ||||||
|     &ScaleTask,  /* Task handle. */ |     &ScaleTask,  /* Task handle. */ | ||||||
| @@ -101,20 +110,21 @@ uint8_t start_scale() { | |||||||
|   } else { |   } else { | ||||||
|       Serial.println("ScaleLoop-Task erfolgreich erstellt"); |       Serial.println("ScaleLoop-Task erfolgreich erstellt"); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return (scaleCalibrated == 1) ? 1 : 3; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| uint8_t calibrate_scale() { | uint8_t calibrate_scale() { | ||||||
|   long newCalibrationValue; |   uint8_t returnState = 0; | ||||||
|  |   float newCalibrationValue; | ||||||
|  |  | ||||||
|  |   vTaskSuspend(RfidReaderTask); | ||||||
|  |   vTaskSuspend(ScaleTask); | ||||||
|  |  | ||||||
|   //vTaskSuspend(RfidReaderTask); |  | ||||||
|   vTaskDelete(RfidReaderTask); |  | ||||||
|   pauseBambuMqttTask = true; |   pauseBambuMqttTask = true; | ||||||
|   pauseMainTask = 1; |   pauseMainTask = 1; | ||||||
|    |    | ||||||
|   if (scale.wait_ready_timeout(1000)) |   if (scale.wait_ready_timeout(1000)) | ||||||
|   { |   { | ||||||
|  |      | ||||||
|     scale.set_scale(); |     scale.set_scale(); | ||||||
|     oledShowMessage("Step 1 empty Scale"); |     oledShowMessage("Step 1 empty Scale"); | ||||||
|  |  | ||||||
| @@ -136,7 +146,7 @@ uint8_t calibrate_scale() { | |||||||
|       esp_task_wdt_reset(); |       esp_task_wdt_reset(); | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     long newCalibrationValue = scale.get_units(10); |     float newCalibrationValue = scale.get_units(10); | ||||||
|     Serial.print("Result: "); |     Serial.print("Result: "); | ||||||
|     Serial.println(newCalibrationValue); |     Serial.println(newCalibrationValue); | ||||||
|  |  | ||||||
| @@ -149,28 +159,33 @@ uint8_t calibrate_scale() { | |||||||
|  |  | ||||||
|       // Speichern mit NVS |       // Speichern mit NVS | ||||||
|       preferences.begin(NVS_NAMESPACE, false); // false = readwrite |       preferences.begin(NVS_NAMESPACE, false); // false = readwrite | ||||||
|       preferences.putLong(NVS_KEY_CALIBRATION, newCalibrationValue); |       preferences.putFloat(NVS_KEY_CALIBRATION, newCalibrationValue); | ||||||
|       preferences.end(); |       preferences.end(); | ||||||
|  |  | ||||||
|       // Verifizieren |       // Verifizieren | ||||||
|       preferences.begin(NVS_NAMESPACE, true); |       preferences.begin(NVS_NAMESPACE, true); | ||||||
|       long verifyValue = preferences.getLong(NVS_KEY_CALIBRATION, 0); |       float verifyValue = preferences.getFloat(NVS_KEY_CALIBRATION, 0); | ||||||
|       preferences.end(); |       preferences.end(); | ||||||
|  |  | ||||||
|       Serial.print("Verified stored value: "); |       Serial.print("Verified stored value: "); | ||||||
|       Serial.println(verifyValue); |       Serial.println(verifyValue); | ||||||
|  |  | ||||||
|       Serial.println("End calibration, revome weight"); |       Serial.println("End calibration, remove weight"); | ||||||
|  |  | ||||||
|       oledShowMessage("Remove weight"); |       oledShowMessage("Remove weight"); | ||||||
|  |  | ||||||
|  |       scale.set_scale(newCalibrationValue); | ||||||
|       for (uint16_t i = 0; i < 2000; i++) { |       for (uint16_t i = 0; i < 2000; i++) { | ||||||
|         yield(); |         yield(); | ||||||
|         vTaskDelay(pdMS_TO_TICKS(1)); |         vTaskDelay(pdMS_TO_TICKS(1)); | ||||||
|         esp_task_wdt_reset(); |         esp_task_wdt_reset(); | ||||||
|       } |       } | ||||||
|        |        | ||||||
|       oledShowMessage("Calibration done"); |       oledShowMessage("Scale calibrated"); | ||||||
|  |  | ||||||
|  |       // For some reason it is not possible to re-tare the scale here, it will result in a wdt timeout. Instead let the scale loop do the taring | ||||||
|  |       //scale.tare(); | ||||||
|  |       scaleTareRequest = true; | ||||||
|  |  | ||||||
|       for (uint16_t i = 0; i < 2000; i++) { |       for (uint16_t i = 0; i < 2000; i++) { | ||||||
|         yield(); |         yield(); | ||||||
| @@ -178,8 +193,9 @@ uint8_t calibrate_scale() { | |||||||
|         esp_task_wdt_reset(); |         esp_task_wdt_reset(); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       //ESP.restart(); |       returnState = 1; | ||||||
|     } |     } | ||||||
|  |     | ||||||
|     else |     else | ||||||
|     { |     { | ||||||
|       { |       { | ||||||
| @@ -192,7 +208,7 @@ uint8_t calibrate_scale() { | |||||||
|           vTaskDelay(pdMS_TO_TICKS(1)); |           vTaskDelay(pdMS_TO_TICKS(1)); | ||||||
|           esp_task_wdt_reset(); |           esp_task_wdt_reset(); | ||||||
|         } |         } | ||||||
|         return 0; |         returnState = 0; | ||||||
|       } |       } | ||||||
|     }  |     }  | ||||||
|   } |   } | ||||||
| @@ -207,17 +223,13 @@ uint8_t calibrate_scale() { | |||||||
|       vTaskDelay(pdMS_TO_TICKS(1)); |       vTaskDelay(pdMS_TO_TICKS(1)); | ||||||
|       esp_task_wdt_reset(); |       esp_task_wdt_reset(); | ||||||
|     } |     } | ||||||
|     return 0; |     returnState = 0; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   oledShowMessage("Scale Ready"); |   vTaskResume(RfidReaderTask); | ||||||
|  |   vTaskResume(ScaleTask); | ||||||
|    |  | ||||||
|   Serial.println("starte Scale Task"); |  | ||||||
|   start_scale(); |  | ||||||
|  |  | ||||||
|   pauseBambuMqttTask = false; |   pauseBambuMqttTask = false; | ||||||
|   pauseMainTask = 0; |   pauseMainTask = 0; | ||||||
|  |  | ||||||
|   return 1; |   return returnState; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -12,7 +12,7 @@ uint8_t tareScale(); | |||||||
| extern HX711 scale; | extern HX711 scale; | ||||||
| extern int16_t weight; | extern int16_t weight; | ||||||
| extern uint8_t weigthCouterToApi; | extern uint8_t weigthCouterToApi; | ||||||
| extern uint8_t scale_tare_counter; | extern uint8_t scaleTareRequest; | ||||||
| extern uint8_t pauseMainTask; | extern uint8_t pauseMainTask; | ||||||
| extern uint8_t scaleCalibrated; | extern uint8_t scaleCalibrated; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -29,7 +29,7 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | |||||||
|     if (type == WS_EVT_CONNECT) { |     if (type == WS_EVT_CONNECT) { | ||||||
|         Serial.println("Neuer Client verbunden!"); |         Serial.println("Neuer Client verbunden!"); | ||||||
|         // Sende die AMS-Daten an den neuen Client |         // Sende die AMS-Daten an den neuen Client | ||||||
|         sendAmsData(client); |         if (!bambuDisabled) sendAmsData(client); | ||||||
|         sendNfcData(client); |         sendNfcData(client); | ||||||
|         foundNfcTag(client, 0); |         foundNfcTag(client, 0); | ||||||
|         sendWriteResult(client, 3); |         sendWriteResult(client, 3); | ||||||
| @@ -213,7 +213,10 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|     // Route für RFID |     // Route für RFID | ||||||
|     server.on("/", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|         Serial.println("Anfrage für /rfid erhalten"); |         Serial.println("Anfrage für /rfid erhalten"); | ||||||
|         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/rfid.html.gz", "text/html"); |          | ||||||
|  |         String page = (bambuDisabled) ? "/rfid.html.gz" : "/rfid_bambu.html.gz"; | ||||||
|  |         AsyncWebServerResponse *response = request->beginResponse(LittleFS, page, "text/html"); | ||||||
|  |          | ||||||
|         response->addHeader("Content-Encoding", "gzip"); |         response->addHeader("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); |         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |         request->send(response); | ||||||
| @@ -286,6 +289,14 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         String url = request->getParam("url")->value(); |         String url = request->getParam("url")->value(); | ||||||
|  |         if (url.indexOf("http://") == -1 && url.indexOf("https://") == -1) { | ||||||
|  |             url = "http://" + url; | ||||||
|  |         } | ||||||
|  |         // Remove trailing slash if exists | ||||||
|  |         if (url.length() > 0 && url.charAt(url.length()-1) == '/') { | ||||||
|  |             url = url.substring(0, url.length()-1); | ||||||
|  |         } | ||||||
|  |          | ||||||
|         bool octoEnabled = (request->getParam("octoEnabled")->value() == "true") ? true : false; |         bool octoEnabled = (request->getParam("octoEnabled")->value() == "true") ? true : false; | ||||||
|         String octoUrl = request->getParam("octoUrl")->value(); |         String octoUrl = request->getParam("octoUrl")->value(); | ||||||
|         String octoToken = (request->getParam("octoToken")->value() != "") ? request->getParam("octoToken")->value() : ""; |         String octoToken = (request->getParam("octoToken")->value() != "") ? request->getParam("octoToken")->value() : ""; | ||||||
| @@ -300,8 +311,17 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|         request->send(200, "application/json", jsonResponse); |         request->send(200, "application/json", jsonResponse); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     // Route für das Überprüfen der Spoolman-Instanz |     // Route für das Überprüfen der Bambu-Instanz | ||||||
|     server.on("/api/bambu", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/api/bambu", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|  |         if (request->hasParam("remove")) { | ||||||
|  |             if (removeBambuCredentials()) { | ||||||
|  |                 request->send(200, "application/json", "{\"success\": true}"); | ||||||
|  |             } else { | ||||||
|  |                 request->send(500, "application/json", "{\"success\": false, \"error\": \"Fehler beim Löschen der Bambu-Credentials\"}"); | ||||||
|  |             } | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         if (!request->hasParam("bambu_ip") || !request->hasParam("bambu_serialnr") || !request->hasParam("bambu_accesscode")) { |         if (!request->hasParam("bambu_ip") || !request->hasParam("bambu_serialnr") || !request->hasParam("bambu_accesscode")) { | ||||||
|             request->send(400, "application/json", "{\"success\": false, \"error\": \"Missing parameter\"}"); |             request->send(400, "application/json", "{\"success\": false, \"error\": \"Missing parameter\"}"); | ||||||
|             return; |             return; | ||||||
|   | |||||||
| @@ -59,7 +59,7 @@ void initWiFi() { | |||||||
|   if(wm_nonblocking) wm.setConfigPortalBlocking(false); |   if(wm_nonblocking) wm.setConfigPortalBlocking(false); | ||||||
|   //wm.setConfigPortalTimeout(320); // Portal nach 5min schließen |   //wm.setConfigPortalTimeout(320); // Portal nach 5min schließen | ||||||
|   wm.setWiFiAutoReconnect(true); |   wm.setWiFiAutoReconnect(true); | ||||||
|   wm.setConnectTimeout(5); |   wm.setConnectTimeout(10); | ||||||
|  |  | ||||||
|   oledShowTopRow(); |   oledShowTopRow(); | ||||||
|   oledShowMessage("WiFi Setup"); |   oledShowMessage("WiFi Setup"); | ||||||
|   | |||||||
							
								
								
									
										6432
									
								
								usermod/spitzbirne32/CAD/Base_usermod_spitzbirne32.stp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										16385
									
								
								usermod/spitzbirne32/CAD/FilaMan-Scale_usermod_spitzbirne32.stp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										5278
									
								
								usermod/spitzbirne32/CAD/Housing_usermod_spitzbirne32.stp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										0
									
								
								usermod/spitzbirne32/CAD/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										4888
									
								
								usermod/spitzbirne32/CAD/ScaleTop_usermod_spitzbirne32.stp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 540 KiB | 
| After Width: | Height: | Size: 525 KiB | 
| After Width: | Height: | Size: 7.9 MiB | 
| After Width: | Height: | Size: 183 KiB | 
							
								
								
									
										12
									
								
								usermod/spitzbirne32/Images/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | |||||||
|  | ## **Heat insert location** | ||||||
|  |  | ||||||
|  | Housing:  | ||||||
|  | - every hole is made to fit a heat insert | ||||||
|  |  | ||||||
|  |  | ||||||
|  | --- | ||||||
|  | Scale top:  | ||||||
|  | - two heat inserts for the NFC Reader | ||||||
|  |  | ||||||
|  |    | ||||||
|  |  | ||||||
| After Width: | Height: | Size: 491 KiB | 
| After Width: | Height: | Size: 834 KiB | 
							
								
								
									
										
											BIN
										
									
								
								usermod/spitzbirne32/Images/Showcase_usermod_spitzbirne32.gif
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.9 MiB | 
							
								
								
									
										69
									
								
								usermod/spitzbirne32/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,69 @@ | |||||||
|  | ## Modifications | ||||||
|  |  | ||||||
|  | To reduce costs, components were sourced from AliExpress instead of Amazon. However, differences in dimensions and mounting hole spacing necessitated adjustments to the 3D-printed parts. Additionally M3 heat inserts were used to limit M4 screws to a minimum. | ||||||
|  |  | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | List of parts that were used: | ||||||
|  | - Display: https://aliexpress.com/item/1005007389730469.html | ||||||
|  | - Scale(5KG with HX711): https://aliexpress.com/item/1005006827930173.html | ||||||
|  | - NFC Reader: https://aliexpress.com/item/1005005973913526.html | ||||||
|  | - NFC Chips: https://aliexpress.com/item/1005006332360160.html | ||||||
|  | - [VORON](https://vorondesign.com/) Heat Inserts M3 OD5mm L4mm: https://aliexpress.com/item/1005003582355741.html  - make sure to select the correct size | ||||||
|  |  | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | - **Parts are designed to be printed in ABS/ASA.** Shrinking compensation not needed. | ||||||
|  |  | ||||||
|  | - **Display and Scale Adjustments:** The AliExpress-sourced display and scale had different dimensions and hole spacings compared to the Amazon versions. The 3D models were modified to accommodate these differences, ensuring proper fit and functionality. | ||||||
|  |    -  measurement of my Display & Scale to check if your parts will fit can be found in the images folder | ||||||
|  |  | ||||||
|  | - **Screw Size and Heat Inserts:** All holes originally designed for M4 screws were resized to fit M3 screws. Standard VORON heat inserts were incorporated to provide durable threading. This change standardizes the hardware and simplifies assembly. | ||||||
|  |  | ||||||
|  | - **Display Mounting:** The display is now mounted using M3 screws with VORON heat inserts. The display's mounting holes need to be drilled to 3mm to accommodate the M3 screws. | ||||||
|  |  | ||||||
|  | - **Scale Top Surface:** The top surface of the scale was modified to allow M3 socket head cap screws to sit flush with the 3D-printed part. This design ensures that the filament spool rests flat without interference. | ||||||
|  |  | ||||||
|  | - **NFC Reader Mounting:** The NFC reader is also secured using M3 screws and VORON heat inserts, maintaining consistency across all components. | ||||||
|  |  | ||||||
|  | - **Scale Base Mounting:** The only M4 screws required are for attaching the metal part of the scale to its base. | ||||||
|  |  | ||||||
|  | ## Benefits of Modifications | ||||||
|  |  | ||||||
|  | - **Cost Reduction:** Sourcing components from AliExpress offers a more affordable alternative to Amazon, making the project more accessible. | ||||||
|  |  | ||||||
|  | - **Standardized Hardware:** Using M3 screws and [VORON](https://vorondesign.com/) heat inserts throughout the assembly simplifies the build process and reduces the variety of required hardware. | ||||||
|  |  | ||||||
|  | - **Enhanced Compatibility:** Adjustments to the 3D models ensure compatibility with readily available components, accommodating variations in part dimensions. | ||||||
|  |  | ||||||
|  | ## Assembly Instructions | ||||||
|  |  | ||||||
|  | 1. **Component Preparation:** | ||||||
|  |    - Carefully drill the display's mounting holes to 3mm to fit M3 screws. | ||||||
|  |  | ||||||
|  | 2. **Heat Insert Installation:** | ||||||
|  |    - install VORON M3 heat inserts into the designated holes in the 3D-printed housing/case for the ESP32 and Scale top → [heat insert location pictures](./Images/README.md) | ||||||
|  |  | ||||||
|  | 3. **Component Mounting:** | ||||||
|  |    - Attach the display, scale, and NFC reader to their respective mounts using M3 screws. | ||||||
|  |    - Secure the metal part of the scale to its base using M4 screws. | ||||||
|  |  | ||||||
|  | 4. **Final Assembly:** | ||||||
|  |    - Assemble all components according to the original FilaMan instructions, ensuring that all modified parts fit correctly and function as intended. | ||||||
|  |  | ||||||
|  | For detailed assembly guides and additional resources, refer to the [original FilaMan documentation](https://github.com/ManuelW77/Filaman). | ||||||
|  |  | ||||||
|  | ## Conclusion | ||||||
|  |  | ||||||
|  | These modifications to the FilaMan project provide a cost-effective and standardized approach to building a filament management system. By sourcing components from AliExpress and adjusting the 3D models accordingly, users can achieve the same functionality at a reduced cost, with the added benefit of using uniform hardware throughout the assembly. | ||||||
|  |  | ||||||
|  | ## Changelog | ||||||
|  |  | ||||||
|  | ### Version 1.0 - 2025-03-04 | ||||||
|  | - Initial release of modifications for AliExpress-sourced components. | ||||||
|  | - Adjusted 3D models to fit different display and scale dimensions. | ||||||
|  | - Replaced M4 screws with M3 screws and integrated VORON heat inserts. | ||||||
|  | - Modified display mounting, requiring drilling to 3mm for M3 screws. | ||||||
|  | - Adjusted scale top surface for flush screw placement. | ||||||
|  | - Standardized NFC reader mounting with M3 screws and VORON heat inserts. | ||||||
|  | - Retained M4 screws only for metal scale attachment. | ||||||