Compare commits
	
		
			316 Commits
		
	
	
		
			v1.3.12
			...
			492bf6cdb8
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 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 | 
							
								
								
									
										161
									
								
								.github/workflows/gitea-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										161
									
								
								.github/workflows/gitea-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -2,12 +2,25 @@ name: Gitea Release | |||||||
|  |  | ||||||
| on: | on: | ||||||
|   workflow_call: |   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: | jobs: | ||||||
|   create-release: |   create-release: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |     outputs: | ||||||
|  |       version: ${{ steps.get_version.outputs.VERSION }} | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4 |     - uses: actions/checkout@v4 | ||||||
|  |       with: | ||||||
|  |         fetch-depth: 0 | ||||||
|      |      | ||||||
|     - name: Set up Python |     - name: Set up Python | ||||||
|       uses: actions/setup-python@v4 |       uses: actions/setup-python@v4 | ||||||
| @@ -28,16 +41,16 @@ jobs: | |||||||
|       run: | |       run: | | ||||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) |         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||||
|          |          | ||||||
|         # Build firmware and SPIFFS |         # Build firmware and LittleFS | ||||||
|         echo "Building firmware and SPIFFS..." |         echo "Building firmware and LittleFS..." | ||||||
|         pio run -e esp32dev |         pio run -e esp32dev | ||||||
|         pio run -t buildfs |         pio run -t buildfs | ||||||
|          |          | ||||||
|         # Copy firmware binary |         # Copy firmware binary | ||||||
|         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/filaman_${VERSION}.bin |         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/upgrade_filaman_firmware_v${VERSION}.bin | ||||||
|          |          | ||||||
|         # Create SPIFFS binary |         # Create LittleFS binary - direct copy without header | ||||||
|         cp .pio/build/esp32dev/spiffs.bin .pio/build/esp32dev/webpage_${VERSION}.bin |         cp .pio/build/esp32dev/littlefs.bin .pio/build/esp32dev/upgrade_filaman_website_v${VERSION}.bin | ||||||
|          |          | ||||||
|         # Create full binary |         # Create full binary | ||||||
|         (cd .pio/build/esp32dev &&  |         (cd .pio/build/esp32dev &&  | ||||||
| @@ -50,7 +63,7 @@ jobs: | |||||||
|           0x1000 bootloader.bin \ |           0x1000 bootloader.bin \ | ||||||
|           0x8000 partitions.bin \ |           0x8000 partitions.bin \ | ||||||
|           0x10000 firmware.bin \ |           0x10000 firmware.bin \ | ||||||
|           0x390000 spiffs.bin) |           0x3D0000 littlefs.bin) | ||||||
|          |          | ||||||
|         # Verify file sizes |         # Verify file sizes | ||||||
|         echo "File sizes:" |         echo "File sizes:" | ||||||
| @@ -62,28 +75,134 @@ jobs: | |||||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) |         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||||
|         echo "VERSION=$VERSION" >> $GITHUB_OUTPUT |         echo "VERSION=$VERSION" >> $GITHUB_OUTPUT | ||||||
|        |        | ||||||
|     - name: Read CHANGELOG.md |     - name: Generate Release Notes | ||||||
|       id: changelog |       id: release_notes | ||||||
|       run: | |       run: | | ||||||
|         VERSION=${{ steps.get_version.outputs.VERSION }} |         # Get the latest tag | ||||||
|         CHANGELOG=$(awk "/## \\[$VERSION\\]/{p=1;print;next} /## \\[/{p=0} p" CHANGELOG.md) |         LATEST_TAG=$(git for-each-ref --sort=-creatordate --format '%(refname:short)' refs/tags | sed -n '2p') | ||||||
|         echo "CHANGES<<EOF" >> $GITHUB_OUTPUT |          | ||||||
|         echo "$CHANGELOG" >> $GITHUB_OUTPUT |         if [ -n "$LATEST_TAG" ]; then | ||||||
|         echo "EOF" >> $GITHUB_OUTPUT |           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 |     - 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: | |       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 }} |         VERSION=${{ steps.get_version.outputs.VERSION }} | ||||||
|         cd .pio/build/esp32dev |         cd .pio/build/esp32dev | ||||||
|          |          | ||||||
|         # Prepare files for upload |         # Debug-Ausgaben | ||||||
|         FILES="" |         echo "Debug: API URL: ${GITEA_API_URL}" | ||||||
|         for file in filaman_${VERSION}.bin webpage_${VERSION}.bin filaman_full_${VERSION}.bin; do |         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 |           if [ -f "$file" ]; then | ||||||
|             FILES="$FILES -a $file" |             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 |           fi | ||||||
|         done |         done | ||||||
|          |  | ||||||
|         # Create release using git command |  | ||||||
|         git tag -a "v${VERSION}" -m "Release ${VERSION}" |  | ||||||
|         git push origin "v${VERSION}" |  | ||||||
							
								
								
									
										176
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										176
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -3,7 +3,7 @@ name: GitHub Release | |||||||
| on: | on: | ||||||
|   workflow_call: |   workflow_call: | ||||||
|     secrets: |     secrets: | ||||||
|       GITHUB_TOKEN: |       RELEASE_TOKEN: | ||||||
|         description: 'GitHub token for release creation' |         description: 'GitHub token for release creation' | ||||||
|         required: true |         required: true | ||||||
|  |  | ||||||
| @@ -17,6 +17,8 @@ jobs: | |||||||
|       contents: write |       contents: write | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4 |     - uses: actions/checkout@v4 | ||||||
|  |       with: | ||||||
|  |         fetch-depth: 0 | ||||||
|      |      | ||||||
|     - name: Set up Python |     - name: Set up Python | ||||||
|       uses: actions/setup-python@v4 |       uses: actions/setup-python@v4 | ||||||
| @@ -33,42 +35,20 @@ jobs: | |||||||
|         sudo apt-get update |         sudo apt-get update | ||||||
|         sudo apt-get install xxd |         sudo apt-get install xxd | ||||||
|      |      | ||||||
|     - name: Check for Data changes |  | ||||||
|       id: check_data |  | ||||||
|       run: | |  | ||||||
|         git fetch --unshallow || true |  | ||||||
|         CHANGED_FILES=$(git diff --name-only HEAD^..HEAD) |  | ||||||
|         if echo "$CHANGED_FILES" | grep -q "^data/"; then |  | ||||||
|           echo "DATA_CHANGED=true" >> $GITHUB_OUTPUT |  | ||||||
|         else |  | ||||||
|           echo "DATA_CHANGED=false" >> $GITHUB_OUTPUT |  | ||||||
|         fi |  | ||||||
|  |  | ||||||
|     - name: Check for SPIFFS changes |  | ||||||
|       id: check_spiffs |  | ||||||
|       run: | |  | ||||||
|         git fetch --unshallow || true |  | ||||||
|         CHANGED_FILES=$(git diff --name-only HEAD^..HEAD) |  | ||||||
|         if echo "$CHANGED_FILES" | grep -q "^data/\|^html/"; then |  | ||||||
|           echo "SPIFFS_CHANGED=true" >> $GITHUB_OUTPUT |  | ||||||
|         else |  | ||||||
|           echo "SPIFFS_CHANGED=false" >> $GITHUB_OUTPUT |  | ||||||
|         fi |  | ||||||
|      |  | ||||||
|     - name: Build Firmware |     - name: Build Firmware | ||||||
|       run: | |       run: | | ||||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) |         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||||
|          |          | ||||||
|         # Always build firmware and SPIFFS |         # Always build firmware and LittleFS | ||||||
|         echo "Building firmware and SPIFFS..." |         echo "Building firmware and LittleFS..." | ||||||
|         pio run -e esp32dev |         pio run -e esp32dev | ||||||
|         pio run -t buildfs |         pio run -t buildfs | ||||||
|          |          | ||||||
|         # Copy firmware binary |         # Copy firmware binary | ||||||
|         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/filaman_${VERSION}.bin |         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/upgrade_filaman_firmware_v${VERSION}.bin | ||||||
|          |          | ||||||
|         # Always create SPIFFS binary |         # Create LittleFS binary - direct copy without header | ||||||
|         cp .pio/build/esp32dev/spiffs.bin .pio/build/esp32dev/webpage_${VERSION}.bin |         cp .pio/build/esp32dev/littlefs.bin .pio/build/esp32dev/upgrade_filaman_website_v${VERSION}.bin | ||||||
|          |          | ||||||
|         # Create full binary (always) |         # Create full binary (always) | ||||||
|         (cd .pio/build/esp32dev &&  |         (cd .pio/build/esp32dev &&  | ||||||
| @@ -81,7 +61,7 @@ jobs: | |||||||
|           0x1000 bootloader.bin \ |           0x1000 bootloader.bin \ | ||||||
|           0x8000 partitions.bin \ |           0x8000 partitions.bin \ | ||||||
|           0x10000 firmware.bin \ |           0x10000 firmware.bin \ | ||||||
|           0x390000 spiffs.bin) |           0x3D0000 littlefs.bin) | ||||||
|          |          | ||||||
|         # Verify file sizes |         # Verify file sizes | ||||||
|         echo "File sizes:" |         echo "File sizes:" | ||||||
| @@ -93,51 +73,113 @@ jobs: | |||||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) |         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||||
|         echo "VERSION=$VERSION" >> $GITHUB_OUTPUT |         echo "VERSION=$VERSION" >> $GITHUB_OUTPUT | ||||||
|        |        | ||||||
|     - name: Read CHANGELOG.md |     - name: Generate Release Notes | ||||||
|       id: changelog |       id: release_notes | ||||||
|       run: | |       run: | | ||||||
|         VERSION=${{ steps.get_version.outputs.VERSION }} |         # Get the latest tag | ||||||
|         CHANGELOG=$(awk "/## \\[$VERSION\\]/{p=1;print;next} /## \\[/{p=0} p" CHANGELOG.md) |         LATEST_TAG=$(git for-each-ref --sort=-creatordate --format '%(refname:short)' refs/tags | sed -n '2p') | ||||||
|         echo "CHANGES<<EOF" >> $GITHUB_OUTPUT |          | ||||||
|         echo "$CHANGELOG" >> $GITHUB_OUTPUT |         if [ -n "$LATEST_TAG" ]; then | ||||||
|         echo "EOF" >> $GITHUB_OUTPUT |           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 |     - name: Create GitHub Release | ||||||
|       env: |       env: | ||||||
|         GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |         GH_TOKEN: ${{ secrets.RELEASE_TOKEN }} | ||||||
|       run: | |       run: | | ||||||
|         VERSION=${{ steps.get_version.outputs.VERSION }} |         VERSION=${{ steps.get_version.outputs.VERSION }} | ||||||
|         cd .pio/build/esp32dev |         cd .pio/build/esp32dev | ||||||
|          |          | ||||||
|         # Create the release first |         # Create release with available files | ||||||
|         RELEASE_JSON=$(curl -L \ |         FILES_TO_UPLOAD="" | ||||||
|           -X POST \ |  | ||||||
|           -H "Accept: application/vnd.github+json" \ |  | ||||||
|           -H "Authorization: Bearer $GITHUB_TOKEN" \ |  | ||||||
|           -H "X-GitHub-Api-Version: 2022-11-28" \ |  | ||||||
|           "https://api.github.com/repos/$GITHUB_REPOSITORY/releases" \ |  | ||||||
|           -d "{ |  | ||||||
|             \"tag_name\":\"v${VERSION}\", |  | ||||||
|             \"name\":\"Release ${VERSION}\", |  | ||||||
|             \"body\":\"${{ steps.changelog.outputs.CHANGES }}\", |  | ||||||
|             \"draft\":false, |  | ||||||
|             \"prerelease\":false |  | ||||||
|           }") |  | ||||||
|          |          | ||||||
|         # Extract the upload URL from the response |         # Always add firmware | ||||||
|         UPLOAD_URL=$(echo "$RELEASE_JSON" | jq -r .upload_url | sed 's/{?name,label}//') |         if [ -f "upgrade_filaman_firmware_v${VERSION}.bin" ]; then | ||||||
|  |           FILES_TO_UPLOAD="$FILES_TO_UPLOAD upgrade_filaman_firmware_v${VERSION}.bin" | ||||||
|  |         fi | ||||||
|          |          | ||||||
|         # Upload the binary files |         # Add LittleFS and full binary only if they exist | ||||||
|         for file in filaman_${VERSION}.bin webpage_${VERSION}.bin filaman_full_${VERSION}.bin; do |         if [ -f "upgrade_filaman_website_v${VERSION}.bin" ]; then | ||||||
|           if [ -f "$file" ]; then |           FILES_TO_UPLOAD="$FILES_TO_UPLOAD upgrade_filaman_website_v${VERSION}.bin" | ||||||
|             echo "Uploading $file..." |         fi | ||||||
|             curl -L \ |          | ||||||
|               -X POST \ |         if [ -f "filaman_full_${VERSION}.bin" ]; then | ||||||
|               -H "Accept: application/vnd.github+json" \ |           FILES_TO_UPLOAD="$FILES_TO_UPLOAD filaman_full_${VERSION}.bin" | ||||||
|               -H "Authorization: Bearer $GITHUB_TOKEN" \ |         fi | ||||||
|               -H "X-GitHub-Api-Version: 2022-11-28" \ |          | ||||||
|               -H "Content-Type: application/octet-stream" \ |         # Create release with available files | ||||||
|               "${UPLOAD_URL}?name=${file}" \ |         if [ -n "$FILES_TO_UPLOAD" ]; then | ||||||
|               --data-binary "@${file}" |           gh release create "v${VERSION}" \ | ||||||
|           fi |             --title "Release ${VERSION}" \ | ||||||
|         done |             --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 | ||||||
							
								
								
									
										6
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -31,9 +31,11 @@ jobs: | |||||||
|     if: needs.detect-provider.outputs.provider == 'github' |     if: needs.detect-provider.outputs.provider == 'github' | ||||||
|     uses: ./.github/workflows/github-release.yml |     uses: ./.github/workflows/github-release.yml | ||||||
|     secrets: |     secrets: | ||||||
|       GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |       RELEASE_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |  | ||||||
|   gitea-release: |   gitea-release: | ||||||
|     needs: detect-provider |     needs: detect-provider | ||||||
|     if: needs.detect-provider.outputs.provider == 'gitea' |     if: needs.detect-provider.outputs.provider == 'gitea' | ||||||
|     uses: ./.github/workflows/gitea-release.yml |     uses: ./.github/workflows/gitea-release.yml | ||||||
|  |     secrets: | ||||||
|  |       GITEA_TOKEN: ${{ secrets.GITEA_TOKEN }} | ||||||
							
								
								
									
										54
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										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" |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										675
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										675
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,5 +1,680 @@ | |||||||
| # Changelog | # Changelog | ||||||
|  |  | ||||||
|  | ## [1.3.99] - 2025-02-28 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.3.99 | ||||||
|  | - update workflows to build firmware with LittleFS instead of SPIFFS | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.98] - 2025-02-28 | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.3.98 | ||||||
|  | - migrate from SPIFFS to LittleFS for file handling | ||||||
|  | - remove unused VSCode settings file | ||||||
|  | - remove commented-out spoolman and filaman data from api.cpp | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.97] - 2025-02-28 | ||||||
|  | ### Added | ||||||
|  | - füge Bestätigungsmeldung für Spool-Einstellung hinzu | ||||||
|  | - verbessere WLAN-Konfiguration und füge mDNS-Unterstützung hinzu | ||||||
|  | - aktualisiere OLED-Anzeige mit Versionsnummer und verbessere Textausrichtung | ||||||
|  | - füge regelmäßige WLAN-Verbindungsüberprüfung hinzu | ||||||
|  | - aktualisiere Schaltplan-Bild | ||||||
|  | - zeige Versionsnummer im OLED-Display an | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.3.97 | ||||||
|  | - entferne text-shadow von deaktivierten Schaltflächen | ||||||
|  | - füge Link zum Wiki für detaillierte Informationen über die Nutzung hinzu | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - Speichernutzung optimiert | ||||||
|  | - behebe doppelte http.end() Aufrufe in checkSpoolmanExtraFields | ||||||
|  | - optimiere Verzögerungen und Stackgrößen in NFC-Task-Funktionen | ||||||
|  | - entferne ungenutzte Bibliotheken und Debug-Ausgaben aus main.cpp | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.96] - 2025-02-25 | ||||||
|  | ### Added | ||||||
|  | - füge Unterstützung für Spoolman-Einstellungen hinzu und aktualisiere die Benutzeroberfläche | ||||||
|  | - entferne die sendAmsData-Funktion aus der API-Schnittstelle | ||||||
|  | - erweitere Bambu-Credentials um AutoSend-Zeit und aktualisiere die Benutzeroberfläche | ||||||
|  | - erweitere Bambu-Credentials mit AutoSend-Wartezeit und aktualisiere die Benutzeroberfläche | ||||||
|  | - add espRestart function and replace delay with vTaskDelay for OTA update process | ||||||
|  | - implement OTA update functionality with backup and restore for configurations | ||||||
|  | - add own_filaments.json and integrate custom filament loading in bambu.cpp | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update platformio.ini for version v1.3.96 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - aktualisiere Bedingungen für die AMS-Datenaktualisierung und entferne unnötige Aufrufe | ||||||
|  | - aktualisiere Bedingung für den Fortschritt der OTA-Update-Nachricht | ||||||
|  | - update auto set logic to check RFID tag before setting Bambu spool | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.95] - 2025-02-24 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.95 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - bind autoSendToBambu variable to checkbox in spoolman.html | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.94] - 2025-02-24 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.94 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - correct payload type check in NFC write event handling | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.93] - 2025-02-24 | ||||||
|  | ### Added | ||||||
|  | - implement auto send feature for Bambu spool management and update related configurations | ||||||
|  | - add debug mode instructions for Spoolman in README | ||||||
|  | - add wiring diagrams to README for PN532 I2C setup | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.93 | ||||||
|  | - simplify filament names in JSON configuration | ||||||
|  | - update findFilamentIdx to return structured result and improve type searching logic | ||||||
|  | - update README to reflect PN532 I2C configuration and remove SPI pin details | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - remove debug output from splitTextIntoLines and update weight display logic in scanRfidTask | ||||||
|  | - enhance weight display logic for negative values | ||||||
|  | - remove unnecessary CPU frequency configuration from setup function | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.92] - 2025-02-24 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.92 | ||||||
|  | - remove commented-out code in setBambuSpool function | ||||||
|  | - update installation instructions and formatting in README files | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - configure CPU frequency settings in setup function only for testing | ||||||
|  | - update comment to clarify NVS reading process | ||||||
|  | - adjust weight display logic to handle cases for weight less than 2 | ||||||
|  | - update weight display logic to handle negative and specific weight cases | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.91] - 2025-02-23 | ||||||
|  | ### Added | ||||||
|  | - update GitHub Actions workflow for FTP firmware upload with improved credential checks | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.91 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.90] - 2025-02-23 | ||||||
|  | ### Added | ||||||
|  | - update index.html for improved content structure and additional links | ||||||
|  | - improve UI for Spoolman and Bambu Lab printer credentials, enhancing layout and styling | ||||||
|  | - update README files with HSPI default PINs and add ESP32 pin diagram | ||||||
|  | - implement scale calibration checks and update start_scale function to return calibration status | ||||||
|  | - add FTP upload functionality to GitHub release workflow and update installation instructions in README | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.90 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - remove debug secrets check from Gitea release workflow | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.89] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.89 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - update Gitea release workflow to use vars for FTP credentials | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.88] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.88 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - update Gitea release workflow to use secrets for FTP credentials | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.87] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.87 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - enhance FTP upload workflow with credential checks and version output | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.86] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.86 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - streamline FTP credentials usage in Gitea release workflow | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.85] - 2025-02-23 | ||||||
|  | ### Added | ||||||
|  | - add FTP_USER variable for Gitea release workflow | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.85 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.84] - 2025-02-23 | ||||||
|  | ### Added | ||||||
|  | - add FTP_HOST variable for firmware upload in Gitea release workflow | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.84 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.83] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.83 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - correct variable interpolation for FTP credentials in Gitea release workflow | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.82] - 2025-02-23 | ||||||
|  | ### Added | ||||||
|  | - update Gitea release workflow to use variable interpolation for FTP credentials | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.82 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.81] - 2025-02-23 | ||||||
|  | ### Added | ||||||
|  | - update Gitea release workflow to use environment variables for FTP credentials and version | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.81 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.80] - 2025-02-23 | ||||||
|  | ### Added | ||||||
|  | - add FTP_USER and FTP_PASSWORD secrets for firmware upload in Gitea release workflow | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.80 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.79] - 2025-02-23 | ||||||
|  | ### Added | ||||||
|  | - add FTP_USER input for firmware upload in Gitea release workflow | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.79 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.78] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.78 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - change FTP protocol from FTPS to FTP for file upload in workflow | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.77] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.77 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - replace ncftp with lftp for secure firmware upload | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.76] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.76 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - replace FTP action with curl for secure firmware upload and install ncftp | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.75] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.75 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - update FTP user and enhance SSL options in gitea-release workflow | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.74] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.74 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - update password syntax in gitea-release workflow | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.73] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.73 | ||||||
|  | - update version to 1.3.72 in platformio.ini | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.72] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.72 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - update FTP options for Gitea release workflow | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.71] - 2025-02-23 | ||||||
|  | ### Added | ||||||
|  | - add FTP upload step for firmware in Gitea release workflow | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.71 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.70] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.70 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.69] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.69 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - update release note generation to use the second latest tag | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.68] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.68 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - update release note generation to include commit hash and author | ||||||
|  | - remove commented test line from platformio.ini | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.67] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.67 | ||||||
|  | - ci: update release note generation to use the latest tag | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.66] - 2025-02-23 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.66 | ||||||
|  | - ci: remove redundant git fetch for tags in release note generation | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.65] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.65 | ||||||
|  | - ci: improve release note generation by fetching tags and sorting unique commits | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.64] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.64 | ||||||
|  | - remove unnecessary closing tags from header.html | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.63] - 2025-02-22 | ||||||
|  | ### Added | ||||||
|  | - update update-form background and add glass border effect | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.63 | ||||||
|  | - update release note generation for initial release handling | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.62] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.62 | ||||||
|  | - update background colors and improve layout for update sections | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.61] - 2025-02-22 | ||||||
|  | ### Added | ||||||
|  | - update release notes generation to use previous tag for changes | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.61 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.60] - 2025-02-22 | ||||||
|  | ### Added | ||||||
|  | - remove automatic git push from changelog update script | ||||||
|  | - implement release notes generation with categorized changes since last tag | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.60 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.59] - 2025-02-22 | ||||||
|  | ### Added | ||||||
|  | - implement enhanced update progress handling and WebSocket notifications | ||||||
|  | - improve update progress reporting and enhance WebSocket notifications | ||||||
|  | - enhance update progress handling and add WebSocket closure notification | ||||||
|  | - implement WebSocket for update progress and enhance update response handling | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.59 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.58] - 2025-02-22 | ||||||
|  | ### Added | ||||||
|  | - implement backup and restore functionality for Bambu credentials and Spoolman URL | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.58 | ||||||
|  | - update upgrade page message and improve progress display logic | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.57] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.57 | ||||||
|  | - update header title to 'Filament Management Tool' in multiple HTML files | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.56] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.56 | ||||||
|  | - update header title and improve SPIFFS update error handling | ||||||
|  | - clarify comments in Gitea and GitHub release workflows | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.55] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.55 | ||||||
|  | - update component descriptions in README files | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.54] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.54 | ||||||
|  | - workflow: update SPIFFS binary creation to exclude header | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.53] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - version: update to version 1.3.53 | ||||||
|  | - update changelog for version 1.3.51 | ||||||
|  | - update changelog for version 1.3.51 | ||||||
|  | - workflow: update SPIFFS binary magic byte and revert version to 1.3.51 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.52] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.52 | ||||||
|  | - workflow: update SPIFFS binary creation to use correct chip revision (0xEB for Rev 3) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.51] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update changelog for version 1.3.51 | ||||||
|  | - workflow: update SPIFFS binary magic byte and revert version to 1.3.51 | ||||||
|  |  | ||||||
|  | ## [1.3.50] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.50 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.49] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.49 | ||||||
|  | - workflow: update SPIFFS binary header to use correct chip revision | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.48] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.48 | ||||||
|  | - workflow: update SPIFFS binary header for firmware release | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.47] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.47 | ||||||
|  | - workflow: optimize firmware and SPIFFS update process, improve progress handling and logging | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.46] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.46 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.45] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.45 | ||||||
|  | - workflow: update SPIFFS binary creation to include minimal header and adjust update validation logic | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.44] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.44 | ||||||
|  | - update header title to 'Hollo Lollo Trollo' | ||||||
|  | - update header title to 'Filament Management Tool' and improve update response messages | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.43] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.43 | ||||||
|  | - update header title to 'Hollo Lollo Trollo' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.42] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.42 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - correct path for SPIFFS binary creation in Gitea release workflow | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.41] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.41 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - remove redundant buffer size setting in NFC initialization | ||||||
|  | - update SPIFFS binary creation and enhance NFC buffer size | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.40] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.40 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - update SPIFFS binary header and enhance WebSocket error handling | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.39] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.39 | ||||||
|  | - workflow: update SPIFFS binary creation to set chip version to max supported | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.38] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.38 | ||||||
|  | - workflow: update SPIFFS binary creation with minimal ESP32 image header | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.37] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.37 | ||||||
|  | - workflow: update ESP32-WROOM image header for SPIFFS binary creation | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.36] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.36 | ||||||
|  | - partition: update SPIFFS binary header and offsets in workflow files | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.35] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.35 | ||||||
|  | - partition: update SPIFFS binary header and offsets in workflow files | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.34] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.34 | ||||||
|  | - partition: update SPIFFS binary creation and offsets in workflow files | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.33] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.33 | ||||||
|  | - partition: update spiffs offset and app sizes in partition files | ||||||
|  | - partition: update spiffs offset in partition files | ||||||
|  | - partition: update app sizes and offsets in partitions.csv | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.32] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.32 | ||||||
|  | - workflow: update magic byte for SPIFFS binary creation | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.31] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.31 | ||||||
|  | - workflow: remove unnecessary data and SPIFFS change checks from release workflows | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.30] - 2025-02-22 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.30 | ||||||
|  | - workflow: update Gitea and GitHub release workflows to create SPIFFS binary with magic byte | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.29] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.29 | ||||||
|  | - workflow: update Gitea release workflow to create release before file uploads | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.28] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.28 | ||||||
|  | - workflow: update Gitea release workflow to use file uploads with curl | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.27] - 2025-02-21 | ||||||
|  | ### Added | ||||||
|  | - workflow: add GITEA_TOKEN secret for Gitea API access in release workflows | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.27 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.26] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.26 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - workflow: improve Gitea release workflow with enhanced error handling and debug outputs | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.25] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.25 | ||||||
|  | - workflow: update Gitea release workflow to include RUNNER_NAME and improve error handling | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.24] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.24 | ||||||
|  | - workflow: rename update files to upgrade in GitHub release workflow | ||||||
|  | - workflow: aktualisiere bestehende Einträge im Changelog für vorhandene Versionen | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - workflow: improve Gitea release process with dynamic URL determination and debug outputs | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.23] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.23 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - workflow: enhance Gitea release process with debug outputs and API connection checks | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.22] - 2025-02-21 | ||||||
|  | ### Added | ||||||
|  | - workflow: improve Gitea release process with additional environment variables and error handling | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.22 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.21] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.21 | ||||||
|  | - workflow: enhance Gitea release process with API integration and token management | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.20] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.20 | ||||||
|  | - workflow: enable git tagging and pushing for Gitea releases | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.19] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.19 | ||||||
|  | - workflow: enable git push for version tagging in Gitea release | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.18] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - ACHTUNG: Installiere einmal das filaman_full.bin danach kannst du über die upgrade Files aktualisieren und deine Settings bleiben auch erhalten. | ||||||
|  | - ATTENTION: Install the filaman_full.bin once, then you can update via the upgrade files and your settings will also be retained. | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.18] - 2025-02-21 | ||||||
|  | ### Added | ||||||
|  | - add note about filaman_full.bin installation in changelog | ||||||
|  |  | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.18 | ||||||
|  | - update changelog for version 1.3.18 and enhance update script for existing entries | ||||||
|  |  | ||||||
|  | ## [1.3.17] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.17 | ||||||
|  | - ci: comment out git tag and push commands in gitea-release workflow | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.16] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.16 | ||||||
|  | - ci: update filenames for firmware and website binaries in release workflows | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.15] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.15 | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | - ci: fix missing 'fi' in GitHub release workflow script | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.14] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.14 | ||||||
|  | - ci: update GitHub release workflow to improve file upload handling | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [1.3.13] - 2025-02-21 | ||||||
|  | ### Changed | ||||||
|  | - update webpages for version v1.3.13 | ||||||
|  | - ci: update GitHub release workflow to use RELEASE_TOKEN for improved security | ||||||
|  |  | ||||||
|  |  | ||||||
| ## [1.3.12] - 2025-02-21 | ## [1.3.12] - 2025-02-21 | ||||||
| ### Changed | ### Changed | ||||||
| - update webpages for version v1.3.12 | - update webpages for version v1.3.12 | ||||||
|   | |||||||
							
								
								
									
										59
									
								
								README.de.md
									
									
									
									
									
								
							
							
						
						
									
										59
									
								
								README.de.md
									
									
									
									
									
								
							| @@ -6,10 +6,12 @@ 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) | ||||||
|  |  | ||||||
|  | ### Es gibt jetzt auch ein Wiki, dort sind nochmal alle Funktionen beschrieben: [Wiki](https://github.com/ManuelW77/Filaman/wiki) | ||||||
|  |  | ||||||
| ### ESP32 Hardware-Funktionen | ### ESP32 Hardware-Funktionen | ||||||
| - **Gewichtsmessung:** Verwendung einer Wägezelle mit HX711-Verstärker für präzise Gewichtsverfolgung. | - **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. | - **NFC-Tag Lesen/Schreiben:** PN532-Modul zum Lesen und Schreiben von Filamentdaten auf NFC-Tags. | ||||||
| @@ -53,14 +55,14 @@ Deutsches Erklärvideo: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaO | |||||||
| ### Komponenten | ### Komponenten | ||||||
| - **ESP32 Entwicklungsboard:** Jede ESP32-Variante. | - **ESP32 Entwicklungsboard:** Jede ESP32-Variante. | ||||||
| [Amazon Link](https://amzn.eu/d/aXThslf) | [Amazon Link](https://amzn.eu/d/aXThslf) | ||||||
| - **HX711 Wägezellen-Verstärker:** Für Gewichtsmessung. | - **HX711 5kg Wägezellen-Verstärker:** Für Gewichtsmessung. | ||||||
| [Amazon Link](https://amzn.eu/d/1wZ4v0x) | [Amazon Link](https://amzn.eu/d/06A0DLb) | ||||||
| - **OLED Display:** 128x64 SSD1306. | - **OLED 0.96 Zoll I2C weiß/gelb Display:** 128x64 SSD1306. | ||||||
| [Amazon Link](https://amzn.eu/d/dozAYDU) | [Amazon Link](https://amzn.eu/d/0AuBp2c) | ||||||
| - **PN532 NFC Modul:** Für NFC-Tag-Operationen. | - **PN532 NFC NXP RFID-Modul V3:** Für NFC-Tag-Operationen. | ||||||
| [Amazon Link](https://amzn.eu/d/8205DDh) | [Amazon Link](https://amzn.eu/d/jfIuQXb) | ||||||
| - **NFC-Tag:** NTAG215 | - **NFC Tags Ntag215:** RFID Tag | ||||||
| [Amazon Link](https://amzn.eu/d/fywy4c4) | [Amazon Link](https://amzn.eu/d/9Z6mXc1) | ||||||
|  |  | ||||||
| ### Pin-Konfiguration | ### Pin-Konfiguration | ||||||
| | Komponente        | ESP32 Pin | | | Komponente        | ESP32 Pin | | ||||||
| @@ -71,10 +73,15 @@ Deutsches Erklärvideo: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaO | |||||||
| | OLED SCL          | 22        | | | OLED SCL          | 22        | | ||||||
| | PN532 IRQ         | 32        | | | PN532 IRQ         | 32        | | ||||||
| | PN532 RESET       | 33        | | | PN532 RESET       | 33        | | ||||||
| | PN532 SCK         | 14        | | | PN532 SDA         | 21        | | ||||||
| | PN532 MOSI        | 13        | | | PN532 SCL         | 22        | | ||||||
| | PN532 MISO        | 12        | |  | ||||||
| | PN532 CS/SS       | 15        | | **Achte darauf, dass am PN532 die DIP-Schalter auf I2C gestellt sind** | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Software-Abhängigkeiten | ## Software-Abhängigkeiten | ||||||
|  |  | ||||||
| @@ -101,7 +108,31 @@ Deutsches Erklärvideo: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaO | |||||||
|   - PN532 NFC Modul |   - PN532 NFC Modul | ||||||
|   - Verbindungskabel |   - Verbindungskabel | ||||||
|  |  | ||||||
| ### Schritt-für-Schritt Installation | ## 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:** | 1. **Repository klonen:** | ||||||
|     ```bash |     ```bash | ||||||
|     git clone https://github.com/ManuelW77/Filaman.git |     git clone https://github.com/ManuelW77/Filaman.git | ||||||
|   | |||||||
							
								
								
									
										81
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										81
									
								
								README.md
									
									
									
									
									
								
							| @@ -13,6 +13,8 @@ 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) | ||||||
|  |  | ||||||
|  | ### Now more detailed informations about the usage: [Wiki](https://github.com/ManuelW77/Filaman/wiki) | ||||||
|  |  | ||||||
| ### ESP32 Hardware Features | ### ESP32 Hardware Features | ||||||
| - **Weight Measurement:** Using a load cell with HX711 amplifier for precise weight tracking. | - **Weight Measurement:** Using a load cell with HX711 amplifier for precise weight tracking. | ||||||
| - **NFC Tag Reading/Writing:** PN532 module for reading and writing filament data to NFC tags. | - **NFC Tag Reading/Writing:** PN532 module for reading and writing filament data to NFC tags. | ||||||
| @@ -56,14 +58,14 @@ german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62z | |||||||
| ### Components | ### Components | ||||||
| - **ESP32 Development Board:** Any ESP32 variant. | - **ESP32 Development Board:** Any ESP32 variant. | ||||||
| [Amazon Link](https://amzn.eu/d/aXThslf) | [Amazon Link](https://amzn.eu/d/aXThslf) | ||||||
| - **HX711 Load Cell Amplifier:** For weight measurement. | - **HX711 5kg Load Cell Amplifier:** For weight measurement. | ||||||
| [Amazon Link](https://amzn.eu/d/1wZ4v0x) | [Amazon Link](https://amzn.eu/d/06A0DLb) | ||||||
| - **OLED Display:** 128x64 SSD1306. | - **OLED 0.96 Zoll I2C white/yellow Display:** 128x64 SSD1306. | ||||||
| [Amazon Link](https://amzn.eu/d/dozAYDU) | [Amazon Link](https://amzn.eu/d/0AuBp2c) | ||||||
| - **PN532 NFC Module:** For NFC tag operations. | - **PN532 NFC NXP RFID-Modul V3:** For NFC tag operations. | ||||||
| [Amazon Link](https://amzn.eu/d/8205DDh) | [Amazon Link](https://amzn.eu/d/jfIuQXb) | ||||||
| - **NFC-Tag:** NTAG215 | - **NFC Tags Ntag215:** RFID Tag | ||||||
| [Amazon Link](https://amzn.eu/d/fywy4c4) | [Amazon Link](https://amzn.eu/d/9Z6mXc1) | ||||||
|  |  | ||||||
|  |  | ||||||
| ### Pin Configuration | ### Pin Configuration | ||||||
| @@ -75,10 +77,15 @@ german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62z | |||||||
| | OLED SCL          | 22        | | | OLED SCL          | 22        | | ||||||
| | PN532 IRQ         | 32        | | | PN532 IRQ         | 32        | | ||||||
| | PN532 RESET       | 33        | | | PN532 RESET       | 33        | | ||||||
| | PN532 SCK  	    | 14        | | | PN532 SDA         | 21        | | ||||||
| | PN532 MOSI    	| 13        | | | PN532 SCL         | 22        | | ||||||
| | PN532 MISO       	| 12        | |  | ||||||
| | PN532 CS/SS       | 15        | | **Make sure that the DIP switches on the PN532 are set to I2C** | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Software Dependencies | ## Software Dependencies | ||||||
|  |  | ||||||
| @@ -91,9 +98,9 @@ german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62z | |||||||
| - `Adafruit_SSD1306`: OLED display control | - `Adafruit_SSD1306`: OLED display control | ||||||
| - `HX711`: Load cell communication | - `HX711`: Load cell communication | ||||||
|  |  | ||||||
| ## Installation | ### Installation | ||||||
|  |  | ||||||
| ### Prerequisites | ## Prerequisites | ||||||
| - **Software:** | - **Software:** | ||||||
|   - [PlatformIO](https://platformio.org/) in VS Code |   - [PlatformIO](https://platformio.org/) in VS Code | ||||||
|   - [Spoolman](https://github.com/Donkie/Spoolman) instance |   - [Spoolman](https://github.com/Donkie/Spoolman) instance | ||||||
| @@ -105,7 +112,32 @@ german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62z | |||||||
|   - PN532 NFC Module |   - PN532 NFC Module | ||||||
|   - Connecting wires |   - 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:** | 1. **Clone the Repository:** | ||||||
|     ```bash |     ```bash | ||||||
|     git clone https://github.com/ManuelW77/Filaman.git |     git clone https://github.com/ManuelW77/Filaman.git | ||||||
| @@ -124,25 +156,6 @@ german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62z | |||||||
|     - Configure WiFi settings through the captive portal. |     - Configure WiFi settings through the captive portal. | ||||||
|     - Access the web interface at `http://filaman.local` or the IP address. |     - Access the web interface at `http://filaman.local` or the IP address. | ||||||
|  |  | ||||||
| ## GitHub Actions Configuration |  | ||||||
|  |  | ||||||
| ### Required Secrets for Gitea Releases |  | ||||||
|  |  | ||||||
| When using Gitea as your repository host, you need to configure the following secrets in your repository: |  | ||||||
|  |  | ||||||
| - `GITEA_API_URL`: The base URL of your Gitea instance, including protocol (e.g., `https://git.example.com`) |  | ||||||
| - `GITEA_TOKEN`: Your Gitea access token with permissions to create releases |  | ||||||
| - `GITEA_REPOSITORY`: The repository name in format `owner/repo` (e.g., `username/filaman`) |  | ||||||
|  |  | ||||||
| Example values: |  | ||||||
| ``` |  | ||||||
| GITEA_API_URL=https://git.example.com |  | ||||||
| GITEA_TOKEN=abcdef1234567890 |  | ||||||
| GITEA_REPOSITORY=username/filaman |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| Make sure to set these secrets in your repository settings under Settings > Secrets and Variables > Actions. |  | ||||||
|  |  | ||||||
| ## Documentation | ## Documentation | ||||||
|  |  | ||||||
| ### Relevant Links | ### Relevant Links | ||||||
|   | |||||||
| @@ -1,7 +1,31 @@ | |||||||
| { | { | ||||||
|     "GFU99": "Generic TPU", |     "GFU99": "TPU", | ||||||
|     "GFN99": "Generic PA", |     "GFN99": "PA", | ||||||
|     "GFN98": "Generic PA-CF", |     "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", |     "GFA01": "Bambu PLA Matte", | ||||||
|     "GFA00": "Bambu PLA Basic", |     "GFA00": "Bambu PLA Basic", | ||||||
|     "GFA09": "Bambu PLA Tough", |     "GFA09": "Bambu PLA Tough", | ||||||
| @@ -13,15 +37,11 @@ | |||||||
|     "GFL03": "eSUN PLA+", |     "GFL03": "eSUN PLA+", | ||||||
|     "GFL01": "PolyTerra PLA", |     "GFL01": "PolyTerra PLA", | ||||||
|     "GFL00": "PolyLite PLA", |     "GFL00": "PolyLite PLA", | ||||||
|     "GFL99": "Generic PLA", |  | ||||||
|     "GFL96": "Generic PLA Silk", |  | ||||||
|     "GFL98": "Generic PLA-CF", |  | ||||||
|     "GFA50": "Bambu PLA-CF", |     "GFA50": "Bambu PLA-CF", | ||||||
|     "GFS02": "Bambu Support For PLA", |     "GFS02": "Bambu Support For PLA", | ||||||
|     "GFA11": "Bambu PLA Aero", |     "GFA11": "Bambu PLA Aero", | ||||||
|     "GFL04": "Overture PLA", |     "GFL04": "Overture PLA", | ||||||
|     "GFL05": "Overture Matte PLA", |     "GFL05": "Overture Matte PLA", | ||||||
|     "GFL95": "Generic PLA High Speed", |  | ||||||
|     "GFA12": "Bambu PLA Glow", |     "GFA12": "Bambu PLA Glow", | ||||||
|     "GFA13": "Bambu PLA Dynamic", |     "GFA13": "Bambu PLA Dynamic", | ||||||
|     "GFA15": "Bambu PLA Galaxy", |     "GFA15": "Bambu PLA Galaxy", | ||||||
| @@ -30,41 +50,21 @@ | |||||||
|     "GFU00": "Bambu TPU 95A HF", |     "GFU00": "Bambu TPU 95A HF", | ||||||
|     "GFG00": "Bambu PETG Basic", |     "GFG00": "Bambu PETG Basic", | ||||||
|     "GFT01": "Bambu PET-CF", |     "GFT01": "Bambu PET-CF", | ||||||
|     "GFG99": "Generic PETG", |  | ||||||
|     "GFG98": "Generic PETG-CF", |  | ||||||
|     "GFG50": "Bambu PETG-CF", |     "GFG50": "Bambu PETG-CF", | ||||||
|     "GFG60": "PolyLite PETG", |     "GFG60": "PolyLite PETG", | ||||||
|     "GFG01": "Bambu PETG Translucent", |     "GFG01": "Bambu PETG Translucent", | ||||||
|     "GFG97": "Generic PCTG", |  | ||||||
|     "GFB00": "Bambu ABS", |     "GFB00": "Bambu ABS", | ||||||
|     "GFB99": "Generic ABS", |  | ||||||
|     "GFB60": "PolyLite ABS", |     "GFB60": "PolyLite ABS", | ||||||
|     "GFB50": "Bambu ABS-GF", |     "GFB50": "Bambu ABS-GF", | ||||||
|     "GFC00": "Bambu PC", |     "GFC00": "Bambu PC", | ||||||
|     "GFC99": "Generic PC", |  | ||||||
|     "GFB98": "Generic ASA", |  | ||||||
|     "GFB01": "Bambu ASA", |     "GFB01": "Bambu ASA", | ||||||
|     "GFB61": "PolyLite ASA", |     "GFB61": "PolyLite ASA", | ||||||
|     "GFB02": "Bambu ASA-Aero", |     "GFB02": "Bambu ASA-Aero", | ||||||
|     "GFS99": "Generic PVA", |  | ||||||
|     "GFS04": "Bambu PVA", |     "GFS04": "Bambu PVA", | ||||||
|     "GFS01": "Bambu Support G", |     "GFS01": "Bambu Support G", | ||||||
|     "GFN03": "Bambu PA-CF", |     "GFN03": "Bambu PA-CF", | ||||||
|     "GFN04": "Bambu PAHT-CF", |     "GFN04": "Bambu PAHT-CF", | ||||||
|     "GFS03": "Bambu Support For PA/PET", |     "GFS03": "Bambu Support For PA/PET", | ||||||
|     "GFN05": "Bambu PA6-CF", |     "GFN05": "Bambu PA6-CF", | ||||||
|     "GFN08": "Bambu PA6-GF", |     "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" |  | ||||||
| } | } | ||||||
| @@ -44,6 +44,4 @@ | |||||||
|             <div class="ram-status" id="ramStatus"></div> |             <div class="ram-status" id="ramStatus"></div> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
| </body> |  | ||||||
| </html> |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -44,12 +44,10 @@ | |||||||
|             <div class="ram-status" id="ramStatus"></div> |             <div class="ram-status" id="ramStatus"></div> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
| </body> |  | ||||||
| </html> |  | ||||||
|  |  | ||||||
| <!-- head --> | <!-- head --> | ||||||
|  |  | ||||||
|     <div class="container"> |     <div class="content"> | ||||||
|         <h1>FilaMan</h1> |         <h1>FilaMan</h1> | ||||||
|         <p>Filament Management Tool</p> |         <p>Filament Management Tool</p> | ||||||
|         <p>Your smart solution for <strong>Filament Management</strong> in 3D printing.</p> |         <p>Your smart solution for <strong>Filament Management</strong> in 3D printing.</p> | ||||||
| @@ -57,10 +55,11 @@ | |||||||
|         <h2>About FilaMan</h2> |         <h2>About FilaMan</h2> | ||||||
|         <p> |         <p> | ||||||
|             FilaMan is a tool designed to simplify filament spool management. It allows you to identify and weigh filament spools,  |             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,  |             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. |  | ||||||
|         </p> |         </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="features"> | ||||||
|             <div class="feature"> |             <div class="feature"> | ||||||
|                 <h3>Spool Identification</h3> |                 <h3>Spool Identification</h3> | ||||||
| @@ -75,12 +74,6 @@ | |||||||
|                 <p>Works with OpenSpool to recognize and activate spools on Bambu printers.</p> |                 <p>Works with OpenSpool to recognize and activate spools on Bambu printers.</p> | ||||||
|             </div> |             </div> | ||||||
|         </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> |     </div> | ||||||
| </body> | </body> | ||||||
| </html> | </html> | ||||||
|   | |||||||
							
								
								
									
										31
									
								
								html/own_filaments.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										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" | ||||||
|  | } | ||||||
| @@ -44,8 +44,6 @@ | |||||||
|             <div class="ram-status" id="ramStatus"></div> |             <div class="ram-status" id="ramStatus"></div> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
| </body> |  | ||||||
| </html> |  | ||||||
|  |  | ||||||
| <!-- head --> | <!-- head --> | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										46
									
								
								html/rfid.js
									
									
									
									
									
								
							
							
						
						
									
										46
									
								
								html/rfid.js
									
									
									
									
									
								
							| @@ -150,6 +150,13 @@ function initWebSocket() { | |||||||
|                     ramStatus.textContent = `${data.freeHeap}k`; |                     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) { |     } catch (error) { | ||||||
|         isConnected = false; |         isConnected = false; | ||||||
| @@ -285,6 +292,14 @@ function displayAmsData(amsData) { | |||||||
|                     <img src="spool_in.png" alt="Spool In" style="width: 48px; height: 48px; transform: rotate(180deg) scaleX(-1);"> |                     <img src="spool_in.png" alt="Spool In" style="width: 48px; height: 48px; transform: rotate(180deg) scaleX(-1);"> | ||||||
|                 </button>`; |                 </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) { |             if (!hasAnyContent) { | ||||||
|                 return ` |                 return ` | ||||||
|                     <div class="tray"> |                     <div class="tray"> | ||||||
| @@ -348,6 +363,7 @@ function displayAmsData(amsData) { | |||||||
|                         ${trayDetails} |                         ${trayDetails} | ||||||
|                         ${tempHTML} |                         ${tempHTML} | ||||||
|                         ${(ams.ams_id === 255 && tray.tray_type !== '') ? outButtonHtml : ''} |                         ${(ams.ams_id === 255 && tray.tray_type !== '') ? outButtonHtml : ''} | ||||||
|  |                         ${(tray.setting_id != "" && tray.setting_id != "null") ? spoolmanButtonHtml : ''} | ||||||
|                     </div> |                     </div> | ||||||
|                      |                      | ||||||
|                 </div>`; |                 </div>`; | ||||||
| @@ -373,6 +389,36 @@ 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() { | function handleSpoolOut() { | ||||||
|     // Erstelle Payload |     // Erstelle Payload | ||||||
|     const payload = { |     const payload = { | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								html/set_spoolman.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								html/set_spoolman.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 9.2 KiB | 
| @@ -44,8 +44,6 @@ | |||||||
|             <div class="ram-status" id="ramStatus"></div> |             <div class="ram-status" id="ramStatus"></div> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
| </body> |  | ||||||
| </html> |  | ||||||
|  |  | ||||||
| <!-- head --> | <!-- head --> | ||||||
|  |  | ||||||
| @@ -76,8 +74,10 @@ | |||||||
|             const ip = document.getElementById('bambuIp').value; |             const ip = document.getElementById('bambuIp').value; | ||||||
|             const serial = document.getElementById('bambuSerial').value; |             const serial = document.getElementById('bambuSerial').value; | ||||||
|             const code = document.getElementById('bambuCode').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(response => response.json()) | ||||||
|                 .then(data => { |                 .then(data => { | ||||||
|                     if (data.healthy) { |                     if (data.healthy) { | ||||||
| @@ -97,27 +97,47 @@ | |||||||
|      |      | ||||||
|     <div class="content"> |     <div class="content"> | ||||||
|         <h1>Spoolman API URL / Bambu Credentials</h1> |         <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="card"> | ||||||
|         <div class="bambu-settings"> |             <div class="card-body"> | ||||||
|             <div class="input-group"> |                 <h5 class="card-title">Set URL/IP to your Spoolman-Instanz</h5> | ||||||
|                 <label for="bambuIp">Bambu Drucker IP-Adresse:</label> |                 <input type="text" id="spoolmanUrl" placeholder="http://ip-or-url-of-your-spoolman-instanz:port"> | ||||||
|                 <input type="text" id="bambuIp" placeholder="192.168.1.xxx" value="{{bambuIp}}"> |                 <button onclick="checkSpoolmanInstance()">Save Spoolman URL</button> | ||||||
|  |                 <p id="statusMessage"></p> | ||||||
|             </div> |             </div> | ||||||
|             <div class="input-group"> |         </div> | ||||||
|                 <label for="bambuSerial">Drucker Seriennummer:</label> |  | ||||||
|                 <input type="text" id="bambuSerial" placeholder="BBLXXXXXXXX" value="{{bambuSerial}}"> |         <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 Drucker IP-Adresse:</label> | ||||||
|  |                         <input type="text" id="bambuIp" placeholder="192.168.1.xxx" value="{{bambuIp}}"> | ||||||
|  |                     </div> | ||||||
|  |                     <div class="input-group"> | ||||||
|  |                         <label for="bambuSerial">Drucker Seriennummer:</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 vom Drucker" 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 time in Seconds:</label> | ||||||
|  |                     </div> | ||||||
|  |                     <div class="input-group" style="display: flex;"> | ||||||
|  |                         <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;"> | ||||||
|  |                     </div> | ||||||
|  |  | ||||||
|  |                     <button style="margin: 0;" onclick="saveBambuCredentials()">Save Bambu Credentials</button> | ||||||
|  |                     <p id="bambuStatusMessage"></p> | ||||||
|  |                 </div> | ||||||
|             </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> | ||||||
|     </div> |     </div> | ||||||
| </body> | </body> | ||||||
|   | |||||||
| @@ -279,9 +279,10 @@ a:hover { | |||||||
|  |  | ||||||
| /* Karten-Stil für optische Trennung */ | /* Karten-Stil für optische Trennung */ | ||||||
| .card { | .card { | ||||||
|     background: #f9f9f9; |     background: var(--primary-color); | ||||||
|  |     width: 500px; | ||||||
|     padding: 15px; |     padding: 15px; | ||||||
|     margin: 20px 0; |     margin: 20px auto; | ||||||
|     border-radius: 8px; |     border-radius: 8px; | ||||||
|     box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); |     box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); | ||||||
| } | } | ||||||
| @@ -760,17 +761,19 @@ a:hover { | |||||||
|     right: 20px; |     right: 20px; | ||||||
|     padding: 15px 25px; |     padding: 15px 25px; | ||||||
|     border-radius: 4px; |     border-radius: 4px; | ||||||
|     color: white; |     color: black; | ||||||
|     z-index: 1000; |     z-index: 1000; | ||||||
|     animation: slideIn 0.3s ease-out; |     animation: slideIn 0.3s ease-out; | ||||||
| } | } | ||||||
|  |  | ||||||
| .notification.success { | .notification.success { | ||||||
|     background-color: #28a745; |     background-color: #28a745; | ||||||
|  |     color: black !important; | ||||||
| } | } | ||||||
|  |  | ||||||
| .notification.error { | .notification.error { | ||||||
|     background-color: #dc3545; |     background-color: #dc3545; | ||||||
|  |     color: white !important; | ||||||
| } | } | ||||||
|  |  | ||||||
| .notification.fade-out { | .notification.fade-out { | ||||||
| @@ -959,7 +962,6 @@ input[type="submit"]:disabled, | |||||||
|  |  | ||||||
| /* Bambu Settings Erweiterung */ | /* Bambu Settings Erweiterung */ | ||||||
| .bambu-settings { | .bambu-settings { | ||||||
|     background: white; |  | ||||||
|     padding: 20px; |     padding: 20px; | ||||||
|     border-radius: 8px; |     border-radius: 8px; | ||||||
|     box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); |     box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); | ||||||
| @@ -1013,6 +1015,7 @@ input[type="submit"]:disabled, | |||||||
|     color: #000; |     color: #000; | ||||||
|     vertical-align: middle; |     vertical-align: middle; | ||||||
|     margin-left: 0.5rem; |     margin-left: 0.5rem; | ||||||
|  |     text-shadow: none !important; | ||||||
| } | } | ||||||
|  |  | ||||||
| .progress-container { | .progress-container { | ||||||
| @@ -1051,9 +1054,10 @@ input[type="submit"]:disabled, | |||||||
| } | } | ||||||
| .update-form { | .update-form { | ||||||
|     background: var(--primary-color); |     background: var(--primary-color); | ||||||
|  |     box-shadow: 0 4px 8px rgba(0, 0, 0, 0.05); | ||||||
|  |     border: var(--glass-border); | ||||||
|     padding: 20px; |     padding: 20px; | ||||||
|     border-radius: 8px; |     border-radius: 8px; | ||||||
|     box-shadow: 0 2px 4px rgba(0,0,0,0.1); |  | ||||||
|     margin: 0 auto; |     margin: 0 auto; | ||||||
|     width: 400px; |     width: 400px; | ||||||
|     text-align: center; |     text-align: center; | ||||||
| @@ -1064,7 +1068,7 @@ input[type="submit"]:disabled, | |||||||
|     padding: 8px; |     padding: 8px; | ||||||
|     border: 1px solid #ddd; |     border: 1px solid #ddd; | ||||||
|     border-radius: 4px; |     border-radius: 4px; | ||||||
|     background: white; |     background-color: #4CAF50; | ||||||
| } | } | ||||||
| .update-form input[type="submit"] { | .update-form input[type="submit"] { | ||||||
|     background-color: #4CAF50; |     background-color: #4CAF50; | ||||||
| @@ -1086,10 +1090,66 @@ input[type="submit"]:disabled, | |||||||
| .warning { | .warning { | ||||||
|     background-color: var(--primary-color); |     background-color: var(--primary-color); | ||||||
|     border: 1px solid #ffe0b2; |     border: 1px solid #ffe0b2; | ||||||
|     color: white; |  | ||||||
|     padding: 15px; |  | ||||||
|     margin: 20px auto; |     margin: 20px auto; | ||||||
|     border-radius: 4px; |     border-radius: 4px; | ||||||
|     max-width: 600px; |     max-width: 600px; | ||||||
|     text-align: center; |     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; | ||||||
| } | } | ||||||
| @@ -44,8 +44,6 @@ | |||||||
|             <div class="ram-status" id="ramStatus"></div> |             <div class="ram-status" id="ramStatus"></div> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
| </body> |  | ||||||
| </html> |  | ||||||
|  |  | ||||||
| <!-- head --> | <!-- head --> | ||||||
|      |      | ||||||
| @@ -86,64 +84,6 @@ | |||||||
|         <div class="status"></div> |         <div class="status"></div> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|     <style> |  | ||||||
|         .update-options { |  | ||||||
|             display: flex; |  | ||||||
|             gap: 2rem; |  | ||||||
|             margin: 2rem 0; |  | ||||||
|         } |  | ||||||
|         .update-section { |  | ||||||
|             flex: 1; |  | ||||||
|             background: #f5f5f5; |  | ||||||
|             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; |  | ||||||
|         } |  | ||||||
|     </style> |  | ||||||
|  |  | ||||||
|     <script> |     <script> | ||||||
|         // Hide status indicators during update |         // Hide status indicators during update | ||||||
|         const statusContainer = document.querySelector('.status-container'); |         const statusContainer = document.querySelector('.status-container'); | ||||||
| @@ -154,98 +94,151 @@ | |||||||
|         const progress = document.querySelector('.progress-bar'); |         const progress = document.querySelector('.progress-bar'); | ||||||
|         const progressContainer = document.querySelector('.progress-container'); |         const progressContainer = document.querySelector('.progress-container'); | ||||||
|         const status = document.querySelector('.status'); |         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.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.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) { |         function handleUpdate(e) { | ||||||
|             e.preventDefault(); |             e.preventDefault(); | ||||||
|             const form = e.target; |             const form = e.target; | ||||||
|             const file = form.update.files[0]; |             const file = form.update.files[0]; | ||||||
|             const updateType = form.dataset.type; |             const updateType = form.dataset.type; | ||||||
|  |              | ||||||
|             if (!file) { |             if (!file) { | ||||||
|                 alert('Please select a file.'); |                 alert('Please select a file.'); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|  |              | ||||||
|             // Validate file name pattern |             // Validate file name pattern | ||||||
|             if (updateType === 'firmware' && !file.name.startsWith('filaman_')) { |             if (updateType === 'firmware' && !file.name.startsWith('upgrade_filaman_firmware_')) { | ||||||
|                 alert('Please select a valid firmware file (filaman_*.bin)'); |                 alert('Please select a valid firmware file (upgrade_filaman_firmware_*.bin)'); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             if (updateType === 'webpage' && !file.name.startsWith('webpage_')) { |             if (updateType === 'webpage' && !file.name.startsWith('upgrade_filaman_website_')) { | ||||||
|                 alert('Please select a valid webpage file (webpage_*.bin)'); |                 alert('Please select a valid webpage file (upgrade_filaman_website_*.bin)'); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|              |              | ||||||
|  |             // Reset UI | ||||||
|  |             updateInProgress = true; | ||||||
|             progressContainer.style.display = 'block'; |             progressContainer.style.display = 'block'; | ||||||
|             status.style.display = 'none'; |             status.style.display = 'none'; | ||||||
|             status.className = 'status'; |             status.className = 'status'; | ||||||
|  |  | ||||||
|             // Reset progress bar |  | ||||||
|             progress.style.width = '0%'; |             progress.style.width = '0%'; | ||||||
|             progress.textContent = '0%'; |             progress.textContent = '0%'; | ||||||
|  |              | ||||||
|             // Disable both forms during update |             // Disable submit buttons | ||||||
|             document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = true); |             document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = true); | ||||||
|  |  | ||||||
|  |             // Send update | ||||||
|             const xhr = new XMLHttpRequest(); |             const xhr = new XMLHttpRequest(); | ||||||
|             xhr.open('POST', '/update', true); |             xhr.open('POST', '/update', true); | ||||||
|             xhr.upload.onprogress = (e) => { |              | ||||||
|                 if (e.lengthComputable) { |  | ||||||
|                     const percentComplete = (e.loaded / e.total) * 100; |  | ||||||
|                     progress.style.width = percentComplete + '%'; |  | ||||||
|                     progress.textContent = Math.round(percentComplete) + '%'; |  | ||||||
|                 } |  | ||||||
|             }; |  | ||||||
|  |  | ||||||
|             xhr.onload = function() { |             xhr.onload = function() { | ||||||
|                 try { |                 if (xhr.status !== 200 && !progress.textContent.startsWith('100')) { | ||||||
|                     let response = this.responseText; |                     status.textContent = "Update failed: " + (xhr.responseText || "Unknown error"); | ||||||
|                     try { |                     status.className = 'status error'; | ||||||
|                         const jsonResponse = JSON.parse(response); |  | ||||||
|                         response = jsonResponse.message; |  | ||||||
|                          |  | ||||||
|                         if (jsonResponse.restart) { |  | ||||||
|                             status.textContent = response + " Redirecting in 20 seconds..."; |  | ||||||
|                             let countdown = 20; |  | ||||||
|                             const timer = setInterval(() => { |  | ||||||
|                                 countdown--; |  | ||||||
|                                 if (countdown <= 0) { |  | ||||||
|                                     clearInterval(timer); |  | ||||||
|                                     window.location.href = '/'; |  | ||||||
|                                 } else { |  | ||||||
|                                     status.textContent = response + ` Redirecting in ${countdown} seconds...`; |  | ||||||
|                                 } |  | ||||||
|                             }, 1000); |  | ||||||
|                         } |  | ||||||
|                     } catch (e) { |  | ||||||
|                         if (!isNaN(response)) { |  | ||||||
|                             const percent = parseInt(response); |  | ||||||
|                             progress.style.width = percent + '%'; |  | ||||||
|                             progress.textContent = percent + '%'; |  | ||||||
|                             return; |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                      |  | ||||||
|                     status.textContent = response; |  | ||||||
|                     status.classList.add(xhr.status === 200 ? 'success' : 'error'); |  | ||||||
|                     status.style.display = 'block'; |  | ||||||
|                      |  | ||||||
|                     if (xhr.status !== 200) { |  | ||||||
|                         document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = false); |  | ||||||
|                     } |  | ||||||
|                 } catch (error) { |  | ||||||
|                     status.textContent = 'Error: ' + error.message; |  | ||||||
|                     status.classList.add('error'); |  | ||||||
|                     status.style.display = 'block'; |                     status.style.display = 'block'; | ||||||
|  |                     updateInProgress = false; | ||||||
|                     document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = false); |                     document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = false); | ||||||
|                 } |                 } | ||||||
|             }; |             }; | ||||||
|  |              | ||||||
|             xhr.onerror = function() { |             xhr.onerror = function() { | ||||||
|                 status.textContent = 'Update failed: Network error'; |                 if (!progress.textContent.startsWith('100')) { | ||||||
|                 status.classList.add('error'); |                     status.textContent = "Network error during update"; | ||||||
|                 status.style.display = 'block'; |                     status.className = 'status error'; | ||||||
|                 document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = false); |                     status.style.display = 'block'; | ||||||
|  |                     updateInProgress = false; | ||||||
|  |                     document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = false); | ||||||
|  |                 } | ||||||
|             }; |             }; | ||||||
|  |  | ||||||
|             const formData = new FormData(); |             const formData = new FormData(); | ||||||
|   | |||||||
| @@ -44,8 +44,6 @@ | |||||||
|             <div class="ram-status" id="ramStatus"></div> |             <div class="ram-status" id="ramStatus"></div> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
| </body> |  | ||||||
| </html> |  | ||||||
|  |  | ||||||
| <!-- head --> | <!-- head --> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -44,8 +44,6 @@ | |||||||
|             <div class="ram-status" id="ramStatus"></div> |             <div class="ram-status" id="ramStatus"></div> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
| </body> |  | ||||||
| </html> |  | ||||||
|  |  | ||||||
| <!-- head --> | <!-- head --> | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								img/ESP32-SPI-Pins.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								img/ESP32-SPI-Pins.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 143 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/IMG_2589.jpeg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								img/IMG_2589.jpeg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 136 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/IMG_2590.jpeg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								img/IMG_2590.jpeg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 143 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/Schaltplan.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								img/Schaltplan.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 283 KiB | 
| @@ -1,6 +1,6 @@ | |||||||
| # Name,   Type, SubType,    Offset,   Size,     Flags | # Name,   Type, SubType,    Offset,   Size,     Flags | ||||||
| nvs,      data, nvs,       0x9000,   0x5000, | nvs,      data, nvs,       0x9000,   0x5000, | ||||||
| otadata,  data, ota,       0xe000,   0x2000, | otadata,  data, ota,       0xe000,   0x2000, | ||||||
| app0,     app,  ota_0,     0x10000,  0x180000, | app0,     app,  ota_0,     0x10000,  0x1E0000, | ||||||
| app1,     app,  ota_1,     0x190000, 0x180000, | app1,     app,  ota_1,     0x1F0000, 0x1E0000, | ||||||
| spiffs,   data, spiffs,    0x310000, 0xE0000, | spiffs,   data, spiffs,    0x3D0000, 0x30000, | ||||||
| 
 | 
| @@ -9,8 +9,8 @@ | |||||||
| ; https://docs.platformio.org/page/projectconf.html | ; https://docs.platformio.org/page/projectconf.html | ||||||
|  |  | ||||||
| [common] | [common] | ||||||
| version = "1.3.12" | version = "1.3.99" | ||||||
|  | ## | ||||||
| [env:esp32dev] | [env:esp32dev] | ||||||
| platform = espressif32 | platform = espressif32 | ||||||
| board = esp32dev | board = esp32dev | ||||||
| @@ -33,7 +33,8 @@ lib_deps = | |||||||
|     digitaldragon/SSLClient @ ^1.3.2 |     digitaldragon/SSLClient @ ^1.3.2 | ||||||
|      |      | ||||||
| ; Enable SPIFFS upload | ; Enable SPIFFS upload | ||||||
| board_build.filesystem = spiffs | #board_build.filesystem = spiffs | ||||||
|  | board_build.filesystem = littlefs | ||||||
| ; Update partition settings | ; Update partition settings | ||||||
| board_build.partitions = partitions.csv | board_build.partitions = partitions.csv | ||||||
| board_upload.flash_size = 4MB | board_upload.flash_size = 4MB | ||||||
| @@ -44,20 +45,16 @@ build_flags = | |||||||
|     -Os |     -Os | ||||||
|     -ffunction-sections |     -ffunction-sections | ||||||
|     -fdata-sections |     -fdata-sections | ||||||
|     -DNDEBUG |     #-DNDEBUG | ||||||
|     -mtext-section-literals |     -mtext-section-literals | ||||||
|     -DVERSION=\"${common.version}\" |     -DVERSION=\"${common.version}\" | ||||||
|     -DASYNCWEBSERVER_REGEX |     -DASYNCWEBSERVER_REGEX | ||||||
|     -DCORE_DEBUG_LEVEL=3 |     #-DCORE_DEBUG_LEVEL=3 | ||||||
|     -DCONFIG_ARDUHAL_LOG_COLORS=1 |     -DCONFIG_ARDUHAL_LOG_COLORS=1 | ||||||
|     -DOTA_DEBUG=1 |     #-DOTA_DEBUG=1 | ||||||
|     -DCONFIG_OPTIMIZATION_LEVEL_DEBUG=1 |     -DCONFIG_OPTIMIZATION_LEVEL_DEBUG=1 | ||||||
|     -DCONFIG_ESP32_PANIC_PRINT_REBOOT |  | ||||||
|     -DBOOT_APP_PARTITION_OTA_0=1 |     -DBOOT_APP_PARTITION_OTA_0=1 | ||||||
|     -DCONFIG_LOG_DEFAULT_LEVEL=3 |  | ||||||
|     -DCONFIG_LWIP_TCP_MSL=60000 |     -DCONFIG_LWIP_TCP_MSL=60000 | ||||||
|     -DCONFIG_LWIP_TCP_WND_DEFAULT=8192 |  | ||||||
|     -DCONFIG_LWIP_TCP_SND_BUF_DEFAULT=4096 |  | ||||||
|     -DCONFIG_LWIP_TCP_RCV_BUF_DEFAULT=4096 |     -DCONFIG_LWIP_TCP_RCV_BUF_DEFAULT=4096 | ||||||
|     -DCONFIG_LWIP_MAX_ACTIVE_TCP=16 |     -DCONFIG_LWIP_MAX_ACTIVE_TCP=16 | ||||||
|      |      | ||||||
|   | |||||||
| @@ -64,29 +64,10 @@ def get_changes_from_git(): | |||||||
|      |      | ||||||
|     return changes |     return changes | ||||||
|  |  | ||||||
| def push_changes(version): |  | ||||||
|     """Push changes to upstream""" |  | ||||||
|     try: |  | ||||||
|         # Stage the CHANGELOG.md |  | ||||||
|         subprocess.run(['git', 'add', 'CHANGELOG.md'], check=True) |  | ||||||
|          |  | ||||||
|         # Commit the changelog |  | ||||||
|         commit_msg = f"docs: update changelog for version {version}" |  | ||||||
|         subprocess.run(['git', 'commit', '-m', commit_msg], check=True) |  | ||||||
|          |  | ||||||
|         # Push to origin (local) |  | ||||||
|         subprocess.run(['git', 'push', 'origin'], check=True) |  | ||||||
|         print("Successfully pushed to origin") |  | ||||||
|          |  | ||||||
|     except subprocess.CalledProcessError as e: |  | ||||||
|         print(f"Error during git operations: {e}") |  | ||||||
|         return False |  | ||||||
|     return True |  | ||||||
|  |  | ||||||
| def update_changelog(): | def update_changelog(): | ||||||
|     print("Starting changelog update...")  # Add this line |     print("Starting changelog update...") | ||||||
|     version = get_version() |     version = get_version() | ||||||
|     print(f"Current version: {version}")   # Add this line |     print(f"Current version: {version}") | ||||||
|     today = datetime.now().strftime('%Y-%m-%d') |     today = datetime.now().strftime('%Y-%m-%d') | ||||||
|      |      | ||||||
|     script_dir = os.path.dirname(os.path.abspath(__file__)) |     script_dir = os.path.dirname(os.path.abspath(__file__)) | ||||||
| @@ -111,7 +92,7 @@ def update_changelog(): | |||||||
|     if not os.path.exists(changelog_path): |     if not os.path.exists(changelog_path): | ||||||
|         with open(changelog_path, 'w') as f: |         with open(changelog_path, 'w') as f: | ||||||
|             f.write(f"# Changelog\n\n{changelog_entry}") |             f.write(f"# Changelog\n\n{changelog_entry}") | ||||||
|         push_changes(version) |         print(f"Created new changelog file with version {version}") | ||||||
|     else: |     else: | ||||||
|         with open(changelog_path, 'r') as f: |         with open(changelog_path, 'r') as f: | ||||||
|             content = f.read() |             content = f.read() | ||||||
| @@ -120,9 +101,30 @@ def update_changelog(): | |||||||
|             updated_content = content.replace("# Changelog\n", f"# Changelog\n\n{changelog_entry}") |             updated_content = content.replace("# Changelog\n", f"# Changelog\n\n{changelog_entry}") | ||||||
|             with open(changelog_path, 'w') as f: |             with open(changelog_path, 'w') as f: | ||||||
|                 f.write(updated_content) |                 f.write(updated_content) | ||||||
|             push_changes(version) |             print(f"Added new version {version} to changelog") | ||||||
|         else: |         else: | ||||||
|             print(f"Version {version} already exists in changelog") |             # 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__": | if __name__ == "__main__": | ||||||
|     update_changelog() |     update_changelog() | ||||||
							
								
								
									
										189
									
								
								src/api.cpp
									
									
									
									
									
								
							
							
						
						
									
										189
									
								
								src/api.cpp
									
									
									
									
									
								
							| @@ -12,34 +12,9 @@ struct SendToApiParams { | |||||||
|     String updatePayload; |     String updatePayload; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| /* | JsonDocument fetchSingleSpoolInfo(int spoolId) { | ||||||
|     // Spoolman Data |  | ||||||
|     { |  | ||||||
|         "version":"1.0", |  | ||||||
|         "protocol":"openspool", |  | ||||||
|         "color_hex":"AF7933", |  | ||||||
|         "type":"ABS", |  | ||||||
|         "min_temp":175, |  | ||||||
|         "max_temp":275, |  | ||||||
|         "brand":"Overture" |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // FilaMan Data |  | ||||||
|     { |  | ||||||
|         "version":"1.0", |  | ||||||
|         "protocol":"openspool", |  | ||||||
|         "color_hex":"AF7933", |  | ||||||
|         "type":"ABS", |  | ||||||
|         "min_temp":175, |  | ||||||
|         "max_temp":275, |  | ||||||
|         "brand":"Overture", |  | ||||||
|         "sm_id":  |  | ||||||
|     } |  | ||||||
| */ |  | ||||||
|  |  | ||||||
| JsonDocument fetchSpoolsForWebsite() { |  | ||||||
|     HTTPClient http; |     HTTPClient http; | ||||||
|     String spoolsUrl = spoolmanUrl + apiUrl + "/spool"; |     String spoolsUrl = spoolmanUrl + apiUrl + "/spool/" + spoolId; | ||||||
|  |  | ||||||
|     Serial.print("Rufe Spool-Daten von: "); |     Serial.print("Rufe Spool-Daten von: "); | ||||||
|     Serial.println(spoolsUrl); |     Serial.println(spoolsUrl); | ||||||
| @@ -56,84 +31,45 @@ JsonDocument fetchSpoolsForWebsite() { | |||||||
|             Serial.print("Fehler beim Parsen der JSON-Antwort: "); |             Serial.print("Fehler beim Parsen der JSON-Antwort: "); | ||||||
|             Serial.println(error.c_str()); |             Serial.println(error.c_str()); | ||||||
|         } else { |         } else { | ||||||
|             JsonArray spools = doc.as<JsonArray>(); |             String filamentType = doc["filament"]["material"].as<String>(); | ||||||
|             JsonArray filteredSpools = filteredDoc.to<JsonArray>(); |             String filamentBrand = doc["filament"]["vendor"]["name"].as<String>(); | ||||||
|  |  | ||||||
|             for (JsonObject spool : spools) { |             int nozzle_temp_min = 0; | ||||||
|                 JsonObject filteredSpool = filteredSpools.add<JsonObject>(); |             int nozzle_temp_max = 0; | ||||||
|                 filteredSpool["extra"]["nfc_id"] = spool["extra"]["nfc_id"]; |             if (doc["filament"]["extra"]["nozzle_temperature"].is<String>()) { | ||||||
|  |                 String tempString = doc["filament"]["extra"]["nozzle_temperature"].as<String>(); | ||||||
|  |                 tempString.replace("[", ""); | ||||||
|  |                 tempString.replace("]", ""); | ||||||
|  |                 int commaIndex = tempString.indexOf(','); | ||||||
|  |                  | ||||||
|  |                 if (commaIndex != -1) { | ||||||
|  |                     nozzle_temp_min = tempString.substring(0, commaIndex).toInt(); | ||||||
|  |                     nozzle_temp_max = tempString.substring(commaIndex + 1).toInt(); | ||||||
|  |                 } | ||||||
|  |             }  | ||||||
|  |  | ||||||
|                 JsonObject filament = filteredSpool["filament"].to<JsonObject>(); |             String filamentColor = doc["filament"]["color_hex"].as<String>(); | ||||||
|                 filament["sm_id"] = spool["id"]; |             filamentColor.toUpperCase(); | ||||||
|                 filament["id"] = spool["filament"]["id"]; |  | ||||||
|                 filament["name"] = spool["filament"]["name"]; |  | ||||||
|                 filament["material"] = spool["filament"]["material"]; |  | ||||||
|                 filament["color_hex"] = spool["filament"]["color_hex"]; |  | ||||||
|                 filament["nozzle_temperature"] = spool["filament"]["extra"]["nozzle_temperature"]; // [190,230] |  | ||||||
|                 filament["price_meter"] = spool["filament"]["extra"]["price_meter"]; |  | ||||||
|                 filament["price_gramm"] = spool["filament"]["extra"]["price_gramm"]; |  | ||||||
|  |  | ||||||
|                 JsonObject vendor = filament["vendor"].to<JsonObject>(); |             String tray_info_idx = doc["filament"]["extra"]["bambu_idx"].as<String>(); | ||||||
|                 vendor["id"] = spool["filament"]["vendor"]["id"]; |             tray_info_idx.replace("\"", ""); | ||||||
|                 vendor["name"] = spool["filament"]["vendor"]["name"]; |              | ||||||
|             } |             String cali_idx = doc["filament"]["extra"]["bambu_cali_id"].as<String>(); // "\"153\"" | ||||||
|         } |             cali_idx.replace("\"", ""); | ||||||
|     } else { |              | ||||||
|         Serial.print("Fehler beim Abrufen der Spool-Daten. HTTP-Code: "); |             String bambu_setting_id = doc["filament"]["extra"]["bambu_setting_id"].as<String>(); // "\"PFUSf40e9953b40d3d\"" | ||||||
|         Serial.println(httpCode); |             bambu_setting_id.replace("\"", ""); | ||||||
|     } |  | ||||||
|  |  | ||||||
|     http.end(); |             doc.clear(); | ||||||
|     return filteredDoc; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| JsonDocument fetchAllSpoolsInfo() { |             filteredDoc["color"] = filamentColor; | ||||||
|     HTTPClient http; |             filteredDoc["type"] = filamentType; | ||||||
|     String spoolsUrl = spoolmanUrl + apiUrl + "/spool"; |             filteredDoc["nozzle_temp_min"] = nozzle_temp_min; | ||||||
|  |             filteredDoc["nozzle_temp_max"] = nozzle_temp_max; | ||||||
|     Serial.print("Rufe Spool-Daten von: "); |             filteredDoc["brand"] = filamentBrand; | ||||||
|     Serial.println(spoolsUrl); |             filteredDoc["tray_info_idx"] = tray_info_idx; | ||||||
|  |             filteredDoc["cali_idx"] = cali_idx; | ||||||
|     http.begin(spoolsUrl); |             filteredDoc["bambu_setting_id"] = bambu_setting_id; | ||||||
|     int httpCode = http.GET(); |  | ||||||
|  |  | ||||||
|     JsonDocument filteredDoc; |  | ||||||
|     if (httpCode == HTTP_CODE_OK) { |  | ||||||
|         String payload = http.getString(); |  | ||||||
|         JsonDocument doc; |  | ||||||
|         DeserializationError error = deserializeJson(doc, payload); |  | ||||||
|         if (error) { |  | ||||||
|             Serial.print("Fehler beim Parsen der JSON-Antwort: "); |  | ||||||
|             Serial.println(error.c_str()); |  | ||||||
|         } else { |  | ||||||
|             JsonArray spools = doc.as<JsonArray>(); |  | ||||||
|             JsonArray filteredSpools = filteredDoc.to<JsonArray>(); |  | ||||||
|  |  | ||||||
|             for (JsonObject spool : spools) { |  | ||||||
|                 JsonObject filteredSpool = filteredSpools.add<JsonObject>(); |  | ||||||
|                 filteredSpool["price"] = spool["price"]; |  | ||||||
|                 filteredSpool["remaining_weight"] = spool["remaining_weight"]; |  | ||||||
|                 filteredSpool["used_weight"] = spool["used_weight"]; |  | ||||||
|                 filteredSpool["extra"]["nfc_id"] = spool["extra"]["nfc_id"]; |  | ||||||
|  |  | ||||||
|                 JsonObject filament = filteredSpool["filament"].to<JsonObject>(); |  | ||||||
|                 filament["id"] = spool["filament"]["id"]; |  | ||||||
|                 filament["name"] = spool["filament"]["name"]; |  | ||||||
|                 filament["material"] = spool["filament"]["material"]; |  | ||||||
|                 filament["density"] = spool["filament"]["density"]; |  | ||||||
|                 filament["diameter"] = spool["filament"]["diameter"]; |  | ||||||
|                 filament["spool_weight"] = spool["filament"]["spool_weight"]; |  | ||||||
|                 filament["color_hex"] = spool["filament"]["color_hex"]; |  | ||||||
|  |  | ||||||
|                 JsonObject vendor = filament["vendor"].to<JsonObject>(); |  | ||||||
|                 vendor["id"] = spool["filament"]["vendor"]["id"]; |  | ||||||
|                 vendor["name"] = spool["filament"]["vendor"]["name"]; |  | ||||||
|  |  | ||||||
|                 JsonObject extra = filament["extra"].to<JsonObject>(); |  | ||||||
|                 extra["nozzle_temperature"] = spool["filament"]["extra"]["nozzle_temperature"]; |  | ||||||
|                 extra["price_gramm"] = spool["filament"]["extra"]["price_gramm"]; |  | ||||||
|                 extra["price_meter"] = spool["filament"]["extra"]["price_meter"]; |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|     } else { |     } else { | ||||||
|         Serial.print("Fehler beim Abrufen der Spool-Daten. HTTP-Code: "); |         Serial.print("Fehler beim Abrufen der Spool-Daten. HTTP-Code: "); | ||||||
| @@ -161,9 +97,9 @@ void sendToApi(void *parameter) { | |||||||
|     if (httpType == "PATCH") httpCode = http.PATCH(updatePayload); |     if (httpType == "PATCH") httpCode = http.PATCH(updatePayload); | ||||||
|  |  | ||||||
|     if (httpCode == HTTP_CODE_OK) { |     if (httpCode == HTTP_CODE_OK) { | ||||||
|         Serial.println("Gewicht der Spule erfolgreich aktualisiert"); |         Serial.println("Spoolman erfolgreich aktualisiert"); | ||||||
|     } else { |     } else { | ||||||
|         Serial.println("Fehler beim Aktualisieren des Gewichts der Spule"); |         Serial.println("Fehler beim Senden an Spoolman!"); | ||||||
|         oledShowMessage("Spoolman update failed"); |         oledShowMessage("Spoolman update failed"); | ||||||
|         vTaskDelay(2000 / portTICK_PERIOD_MS); |         vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||||
|     } |     } | ||||||
| @@ -262,6 +198,52 @@ uint8_t updateSpoolWeight(String spoolId, uint16_t weight) { | |||||||
|     return 1; |     return 1; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | bool updateSpoolBambuData(String payload) { | ||||||
|  |     JsonDocument doc; | ||||||
|  |     DeserializationError error = deserializeJson(doc, payload); | ||||||
|  |     if (error) { | ||||||
|  |         Serial.print("Fehler beim JSON-Parsing: "); | ||||||
|  |         Serial.println(error.c_str()); | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     String spoolsUrl = spoolmanUrl + apiUrl + "/filament/" + doc["filament_id"].as<String>(); | ||||||
|  |     Serial.print("Update Spule mit URL: "); | ||||||
|  |     Serial.println(spoolsUrl); | ||||||
|  |  | ||||||
|  |     JsonDocument updateDoc; | ||||||
|  |     updateDoc["extra"]["bambu_setting_id"] = "\"" + doc["setting_id"].as<String>() + "\""; | ||||||
|  |     updateDoc["extra"]["bambu_cali_id"] = "\"" + doc["cali_idx"].as<String>() + "\""; | ||||||
|  |     updateDoc["extra"]["bambu_idx"] = "\"" + doc["tray_info_idx"].as<String>() + "\""; | ||||||
|  |     updateDoc["extra"]["nozzle_temperature"] = "[" + doc["temp_min"].as<String>() + "," + doc["temp_max"].as<String>() + "]"; | ||||||
|  |  | ||||||
|  |     String updatePayload; | ||||||
|  |     serializeJson(updateDoc, updatePayload); | ||||||
|  |     Serial.print("Update Payload: "); | ||||||
|  |     Serial.println(updatePayload); | ||||||
|  |  | ||||||
|  |     SendToApiParams* params = new SendToApiParams(); | ||||||
|  |     if (params == nullptr) { | ||||||
|  |         Serial.println("Fehler: Kann Speicher für Task-Parameter nicht allokieren."); | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |     params->httpType = "PATCH"; | ||||||
|  |     params->spoolsUrl = spoolsUrl; | ||||||
|  |     params->updatePayload = updatePayload; | ||||||
|  |  | ||||||
|  |     // Erstelle die Task | ||||||
|  |     BaseType_t result = xTaskCreate( | ||||||
|  |         sendToApi,                // Task-Funktion | ||||||
|  |         "SendToApiTask",          // Task-Name | ||||||
|  |         4096,                     // Stackgröße in Bytes | ||||||
|  |         (void*)params,            // Parameter | ||||||
|  |         0,                        // Priorität | ||||||
|  |         NULL                      // Task-Handle (nicht benötigt) | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  |  | ||||||
| // #### Spoolman init | // #### Spoolman init | ||||||
| bool checkSpoolmanExtraFields() { | bool checkSpoolmanExtraFields() { | ||||||
|     HTTPClient http; |     HTTPClient http; | ||||||
| @@ -403,12 +385,13 @@ bool checkSpoolmanExtraFields() { | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         http.end();   |  | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     Serial.println("-------- ENDE Prüfe Felder --------"); |     Serial.println("-------- ENDE Prüfe Felder --------"); | ||||||
|     Serial.println(); |     Serial.println(); | ||||||
|  |  | ||||||
|  |     http.end(); | ||||||
|  |  | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -14,11 +14,10 @@ bool checkSpoolmanInstance(const String& url); | |||||||
| bool saveSpoolmanUrl(const String& url); | bool saveSpoolmanUrl(const String& url); | ||||||
| String loadSpoolmanUrl(); // Neue Funktion zum Laden der URL | String loadSpoolmanUrl(); // Neue Funktion zum Laden der URL | ||||||
| bool checkSpoolmanExtraFields(); // Neue Funktion zum Überprüfen der Extrafelder | bool checkSpoolmanExtraFields(); // Neue Funktion zum Überprüfen der Extrafelder | ||||||
| JsonDocument fetchSpoolsForWebsite(); // API-Funktion für die Webseite | JsonDocument fetchSingleSpoolInfo(int spoolId); // API-Funktion für die Webseite | ||||||
| JsonDocument fetchAllSpoolsInfo(); |  | ||||||
| void sendAmsData(AsyncWebSocketClient *client); // Neue Funktion zum Senden von AMS-Daten |  | ||||||
| bool updateSpoolTagId(String uidString, const char* payload); // Neue Funktion zum Aktualisieren eines Spools | bool updateSpoolTagId(String uidString, const char* payload); // Neue Funktion zum Aktualisieren eines Spools | ||||||
| uint8_t updateSpoolWeight(String spoolId, uint16_t weight); // Neue Funktion zum Aktualisieren des Gewichts | uint8_t updateSpoolWeight(String spoolId, uint16_t weight); // Neue Funktion zum Aktualisieren des Gewichts | ||||||
| bool initSpoolman(); // Neue Funktion zum Initialisieren von Spoolman | bool initSpoolman(); // Neue Funktion zum Initialisieren von Spoolman | ||||||
|  | bool updateSpoolBambuData(String payload); // Neue Funktion zum Aktualisieren der Bambu-Daten | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|   | |||||||
							
								
								
									
										431
									
								
								src/bambu.cpp
									
									
									
									
									
								
							
							
						
						
									
										431
									
								
								src/bambu.cpp
									
									
									
									
									
								
							| @@ -23,14 +23,21 @@ const char* bambu_username = "bblp"; | |||||||
| const char* bambu_ip = nullptr; | const char* bambu_ip = nullptr; | ||||||
| const char* bambu_accesscode = nullptr; | const char* bambu_accesscode = nullptr; | ||||||
| const char* bambu_serialnr = nullptr; | const char* bambu_serialnr = nullptr; | ||||||
|  |  | ||||||
|  | String g_bambu_ip = ""; | ||||||
|  | String g_bambu_accesscode = ""; | ||||||
|  | String g_bambu_serialnr = ""; | ||||||
|  |  | ||||||
| bool bambu_connected = false; | bool bambu_connected = false; | ||||||
|  | bool autoSendToBambu = false; | ||||||
|  | int autoSetToBambuSpoolId = 0; | ||||||
|  |  | ||||||
| // Globale Variablen für AMS-Daten | // Globale Variablen für AMS-Daten | ||||||
| int ams_count = 0; | int ams_count = 0; | ||||||
| String amsJsonData;  // Speichert das fertige JSON für WebSocket-Clients | String amsJsonData;  // Speichert das fertige JSON für WebSocket-Clients | ||||||
| AMSData ams_data[MAX_AMS];  // Definition des Arrays | AMSData ams_data[MAX_AMS];  // Definition des Arrays; | ||||||
|  |  | ||||||
| bool saveBambuCredentials(const String& ip, const String& serialnr, const String& accesscode) { | bool saveBambuCredentials(const String& ip, const String& serialnr, const String& accesscode, bool autoSend, const String& autoSendTime) { | ||||||
|     if (BambuMqttTask) { |     if (BambuMqttTask) { | ||||||
|         vTaskDelete(BambuMqttTask); |         vTaskDelete(BambuMqttTask); | ||||||
|     } |     } | ||||||
| @@ -39,6 +46,8 @@ bool saveBambuCredentials(const String& ip, const String& serialnr, const String | |||||||
|     doc["bambu_ip"] = ip; |     doc["bambu_ip"] = ip; | ||||||
|     doc["bambu_accesscode"] = accesscode; |     doc["bambu_accesscode"] = accesscode; | ||||||
|     doc["bambu_serialnr"] = serialnr; |     doc["bambu_serialnr"] = serialnr; | ||||||
|  |     doc["autoSendToBambu"] = autoSend; | ||||||
|  |     doc["autoSendTime"] = (autoSendTime != "") ? autoSendTime.toInt() : autoSetBambuAmsCounter; | ||||||
|  |  | ||||||
|     if (!saveJsonValue("/bambu_credentials.json", doc)) { |     if (!saveJsonValue("/bambu_credentials.json", doc)) { | ||||||
|         Serial.println("Fehler beim Speichern der Bambu-Credentials."); |         Serial.println("Fehler beim Speichern der Bambu-Credentials."); | ||||||
| @@ -49,6 +58,8 @@ bool saveBambuCredentials(const String& ip, const String& serialnr, const String | |||||||
|     bambu_ip = ip.c_str(); |     bambu_ip = ip.c_str(); | ||||||
|     bambu_accesscode = accesscode.c_str(); |     bambu_accesscode = accesscode.c_str(); | ||||||
|     bambu_serialnr = serialnr.c_str(); |     bambu_serialnr = serialnr.c_str(); | ||||||
|  |     autoSendToBambu = autoSend; | ||||||
|  |     autoSetBambuAmsCounter = autoSendTime.toInt(); | ||||||
|  |  | ||||||
|     vTaskDelay(100 / portTICK_PERIOD_MS); |     vTaskDelay(100 / portTICK_PERIOD_MS); | ||||||
|     if (!setupMqtt()) return false; |     if (!setupMqtt()) return false; | ||||||
| @@ -64,14 +75,21 @@ bool loadBambuCredentials() { | |||||||
|         String code = doc["bambu_accesscode"].as<String>(); |         String code = doc["bambu_accesscode"].as<String>(); | ||||||
|         String serial = doc["bambu_serialnr"].as<String>(); |         String serial = doc["bambu_serialnr"].as<String>(); | ||||||
|  |  | ||||||
|  |         g_bambu_ip = ip; | ||||||
|  |         g_bambu_accesscode = code; | ||||||
|  |         g_bambu_serialnr = serial; | ||||||
|  |  | ||||||
|  |         if (doc["autoSendToBambu"].is<bool>()) autoSendToBambu = doc["autoSendToBambu"].as<bool>(); | ||||||
|  |         if (doc["autoSendTime"].is<int>()) autoSetBambuAmsCounter = doc["autoSendTime"].as<int>(); | ||||||
|  |  | ||||||
|         ip.trim(); |         ip.trim(); | ||||||
|         code.trim(); |         code.trim(); | ||||||
|         serial.trim(); |         serial.trim(); | ||||||
|  |  | ||||||
|         // Dynamische Speicherallokation für die globalen Pointer |         // Dynamische Speicherallokation für die globalen Pointer | ||||||
|         bambu_ip = strdup(ip.c_str()); |         bambu_ip = g_bambu_ip.c_str(); | ||||||
|         bambu_accesscode = strdup(code.c_str()); |         bambu_accesscode = g_bambu_accesscode.c_str(); | ||||||
|         bambu_serialnr = strdup(serial.c_str()); |         bambu_serialnr = g_bambu_serialnr.c_str(); | ||||||
|  |  | ||||||
|         report_topic = "device/" + String(bambu_serialnr) + "/report"; |         report_topic = "device/" + String(bambu_serialnr) + "/report"; | ||||||
|         //request_topic = "device/" + String(bambu_serialnr) + "/request"; |         //request_topic = "device/" + String(bambu_serialnr) + "/request"; | ||||||
| @@ -81,19 +99,49 @@ bool loadBambuCredentials() { | |||||||
|     return false; |     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 |     // JSON-Dokument für die Filament-Daten erstellen | ||||||
|     JsonDocument doc; |     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 |     // 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"); |         Serial.println("Fehler beim Laden der Filament-Daten"); | ||||||
|         return "GFL99"; // Fallback auf Generic PLA |         return {"GFL99", "PLA"}; // Fallback auf Generic PLA | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     // Wenn eigener Typ | ||||||
|  |     if (ownFilament != "") | ||||||
|  |     { | ||||||
|  |         if (doc[ownFilament].is<String>())  | ||||||
|  |         { | ||||||
|  |             return {ownFilament, doc[ownFilament].as<String>()}; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // 1. Erst versuchen wir die exakte Brand + Type Kombination zu finden | ||||||
|     String searchKey; |     String searchKey; | ||||||
|      |  | ||||||
|     // 1. Suche nach Brand + Type Kombination |  | ||||||
|     if (brand == "Bambu" || brand == "Bambulab") { |     if (brand == "Bambu" || brand == "Bambulab") { | ||||||
|         searchKey = "Bambu " + type; |         searchKey = "Bambu " + type; | ||||||
|     } else if (brand == "PolyLite") { |     } else if (brand == "PolyLite") { | ||||||
| @@ -109,23 +157,46 @@ String findFilamentIdx(String brand, String type) { | |||||||
|     // Durchsuche alle Einträge nach der Brand + Type Kombination |     // Durchsuche alle Einträge nach der Brand + Type Kombination | ||||||
|     for (JsonPair kv : doc.as<JsonObject>()) { |     for (JsonPair kv : doc.as<JsonObject>()) { | ||||||
|         if (kv.value().as<String>() == searchKey) { |         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 |     // 2. Wenn nicht gefunden, zerlege den type String in Wörter und suche nach jedem Wort | ||||||
|     searchKey = "Generic " + type; |     // Sammle alle vorhandenen Filamenttypen aus der JSON | ||||||
|  |     std::vector<String> knownTypes; | ||||||
|     for (JsonPair kv : doc.as<JsonObject>()) { |     for (JsonPair kv : doc.as<JsonObject>()) { | ||||||
|         if (kv.value().as<String>() == searchKey) { |         String value = kv.value().as<String>(); | ||||||
|             return kv.key().c_str(); |         // 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) |     // 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("Sending MQTT message"); | ||||||
|     Serial.println(payload); |     Serial.println(payload); | ||||||
|     if (client.publish(report_topic.c_str(), payload.c_str()))  |     if (client.publish(report_topic.c_str(), payload.c_str()))  | ||||||
| @@ -156,15 +227,22 @@ bool setBambuSpool(String payload) { | |||||||
|     int minTemp = doc["nozzle_temp_min"]; |     int minTemp = doc["nozzle_temp_min"]; | ||||||
|     int maxTemp = doc["nozzle_temp_max"]; |     int maxTemp = doc["nozzle_temp_max"]; | ||||||
|     String type = doc["type"].as<String>(); |     String type = doc["type"].as<String>(); | ||||||
|  |     (type == "PLA+") ? type = "PLA" : type; | ||||||
|     String brand = doc["brand"].as<String>(); |     String brand = doc["brand"].as<String>(); | ||||||
|     String tray_info_idx = (doc["tray_info_idx"].as<String>() != "-1") ? doc["tray_info_idx"].as<String>() : ""; |     String tray_info_idx = (doc["tray_info_idx"].as<String>() != "-1") ? doc["tray_info_idx"].as<String>() : ""; | ||||||
|     if (tray_info_idx == "") tray_info_idx = (brand != "" && type != "") ? findFilamentIdx(brand, type) : ""; |     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 setting_id = doc["bambu_setting_id"].as<String>(); | ||||||
|     String cali_idx = doc["cali_idx"].as<String>(); |     String cali_idx = doc["cali_idx"].as<String>(); | ||||||
|  |  | ||||||
|     doc.clear(); |     doc.clear(); | ||||||
|  |  | ||||||
|     doc["print"]["sequence_id"] = 0; |     doc["print"]["sequence_id"] = "0"; | ||||||
|     doc["print"]["command"] = "ams_filament_setting"; |     doc["print"]["command"] = "ams_filament_setting"; | ||||||
|     doc["print"]["ams_id"] = amsId < 200 ? amsId : 255; |     doc["print"]["ams_id"] = amsId < 200 ? amsId : 255; | ||||||
|     doc["print"]["tray_id"] = trayId < 200 ? trayId : 254; |     doc["print"]["tray_id"] = trayId < 200 ? trayId : 254; | ||||||
| @@ -172,10 +250,10 @@ bool setBambuSpool(String payload) { | |||||||
|     doc["print"]["nozzle_temp_min"] = minTemp; |     doc["print"]["nozzle_temp_min"] = minTemp; | ||||||
|     doc["print"]["nozzle_temp_max"] = maxTemp; |     doc["print"]["nozzle_temp_max"] = maxTemp; | ||||||
|     doc["print"]["tray_type"] = type; |     doc["print"]["tray_type"] = type; | ||||||
|     doc["print"]["cali_idx"] = (cali_idx != "") ? cali_idx : ""; |     //doc["print"]["cali_idx"] = (cali_idx != "") ? cali_idx : ""; | ||||||
|     doc["print"]["tray_info_idx"] = tray_info_idx; |     doc["print"]["tray_info_idx"] = tray_info_idx; | ||||||
|     doc["print"]["setting_id"] = setting_id; |     doc["print"]["setting_id"] = setting_id; | ||||||
|  |      | ||||||
|     // Serialize the JSON |     // Serialize the JSON | ||||||
|     String output; |     String output; | ||||||
|     serializeJson(doc, output); |     serializeJson(doc, output); | ||||||
| @@ -194,13 +272,13 @@ bool setBambuSpool(String payload) { | |||||||
|  |  | ||||||
|     if (cali_idx != "") { |     if (cali_idx != "") { | ||||||
|         yield(); |         yield(); | ||||||
|         doc["print"]["sequence_id"] = 0; |         doc["print"]["sequence_id"] = "0"; | ||||||
|         doc["print"]["command"] = "extrusion_cali_sel"; |         doc["print"]["command"] = "extrusion_cali_sel"; | ||||||
|         doc["print"]["filament_id"] = tray_info_idx; |         doc["print"]["filament_id"] = tray_info_idx; | ||||||
|         doc["print"]["nozzle_diameter"] = "0.4"; |         doc["print"]["nozzle_diameter"] = "0.4"; | ||||||
|         doc["print"]["cali_idx"] = cali_idx.toInt(); |         doc["print"]["cali_idx"] = cali_idx.toInt(); | ||||||
|         doc["print"]["tray_id"] = trayId < 200 ? trayId : 254; |         doc["print"]["tray_id"] = trayId < 200 ? trayId : 254; | ||||||
|         doc["print"]["ams_id"] = amsId < 200 ? amsId : 255; |         //doc["print"]["ams_id"] = amsId < 200 ? amsId : 255; | ||||||
|  |  | ||||||
|         // Serialize the JSON |         // Serialize the JSON | ||||||
|         String output; |         String output; | ||||||
| @@ -218,44 +296,120 @@ bool setBambuSpool(String payload) { | |||||||
|         doc.clear(); |         doc.clear(); | ||||||
|         yield(); |         yield(); | ||||||
|     } |     } | ||||||
| /* |  | ||||||
|     if (setting_id != "") { |  | ||||||
|         yield(); |  | ||||||
|         doc["print"]["sequence_id"] = 0; |  | ||||||
|         doc["print"]["command"] = "ams_filament_setting"; |  | ||||||
|         doc["print"]["nozzle_temp_min"] = minTemp; |  | ||||||
|         doc["print"]["nozzle_temp_max"] = maxTemp; |  | ||||||
|         doc["print"]["setting_id"] = setting_id; |  | ||||||
|         doc["print"]["tray_color"] = color.length() == 8 ? color : color+"FF"; |  | ||||||
|         doc["print"]["ams_id"] = amsId < 200 ? amsId : 255; |  | ||||||
|         doc["print"]["tray_id"] = trayId < 200 ? trayId : 254; |  | ||||||
|         doc["print"]["tray_info_idx"] = tray_info_idx; |  | ||||||
|         doc["print"]["tray_type"] = type; |  | ||||||
|  |  | ||||||
|         // Serialize the JSON |     return true; | ||||||
|         String output; | } | ||||||
|         serializeJson(doc, output); |  | ||||||
|  |  | ||||||
|         if (sendMqttMessage(output)) { | void autoSetSpool(int spoolId, uint8_t trayId) { | ||||||
|             Serial.println("Filament Setting successfully set"); |     // 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>(); | ||||||
|  |             //ams_data[i].trays[j].setting_id = trayObj["setting_id"].as<String>(); | ||||||
|  |             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 |         else | ||||||
|         { |         { | ||||||
|             Serial.println("Failed to set Filament setting"); |             ams_data[extIdx].trays[0].setting_id = ""; | ||||||
|             return false; |             ams_data[extIdx].trays[0].cali_idx = ""; | ||||||
|         } |         } | ||||||
|  |         ams_count++;  // Erhöhe ams_count für die externe Spule | ||||||
|         doc.clear(); |  | ||||||
|         yield(); |  | ||||||
|     } |     } | ||||||
| */ |  | ||||||
|  |  | ||||||
|     return true; |     // 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 | // init | ||||||
| void mqtt_callback(char* topic, byte* payload, unsigned int length) { | void mqtt_callback(char* topic, byte* payload, unsigned int length) { | ||||||
|     String message; |     String message; | ||||||
|  |      | ||||||
|     for (int i = 0; i < length; i++) { |     for (int i = 0; i < length; i++) { | ||||||
|         message += (char)payload[i]; |         message += (char)payload[i]; | ||||||
|     } |     } | ||||||
| @@ -263,16 +417,28 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | |||||||
|     // JSON-Dokument parsen |     // JSON-Dokument parsen | ||||||
|     JsonDocument doc; |     JsonDocument doc; | ||||||
|     DeserializationError error = deserializeJson(doc, message); |     DeserializationError error = deserializeJson(doc, message); | ||||||
|     if (error) { |     message = ""; | ||||||
|  |     if (error)  | ||||||
|  |     { | ||||||
|         Serial.print("Fehler beim Parsen des JSON: "); |         Serial.print("Fehler beim Parsen des JSON: "); | ||||||
|         Serial.println(error.c_str()); |         Serial.println(error.c_str()); | ||||||
|         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<String>()) { |     if (doc["print"]["upgrade_state"].is<JsonObject>())  | ||||||
|  |     { | ||||||
|         // Prüfen ob AMS-Daten vorhanden sind |         // Prüfen ob AMS-Daten vorhanden sind | ||||||
|         if (!doc["print"]["ams"].is<String>() || !doc["print"]["ams"]["ams"].is<String>()) { |         if (!doc["print"]["ams"].is<JsonObject>() || !doc["print"]["ams"]["ams"].is<JsonArray>())  | ||||||
|  |         { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -304,9 +470,13 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | |||||||
|             // Vergleiche die Trays |             // Vergleiche die Trays | ||||||
|             for (int j = 0; j < trayArray.size() && j < 4 && !hasChanges; j++) { |             for (int j = 0; j < trayArray.size() && j < 4 && !hasChanges; j++) { | ||||||
|                 JsonObject trayObj = trayArray[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 || |                 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_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) { |                     trayObj["cali_idx"].as<String>() != ams_data[storedIndex].trays[j].cali_idx) { | ||||||
|                     hasChanges = true; |                     hasChanges = true; | ||||||
|                     break; |                     break; | ||||||
| @@ -315,143 +485,55 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         // Prüfe die externe Spule |         // Prüfe die externe Spule | ||||||
|         if (!hasChanges && doc["print"]["vt_tray"].is<String>()) { |         JsonObject vtTray = doc["print"]["vt_tray"]; | ||||||
|             JsonObject vtTray = doc["print"]["vt_tray"]; |         if (doc["print"]["vt_tray"].is<JsonObject>()) { | ||||||
|             bool foundExternal = false; |  | ||||||
|              |  | ||||||
|             for (int i = 0; i < ams_count; i++) { |             for (int i = 0; i < ams_count; i++) { | ||||||
|                 if (ams_data[i].ams_id == 255) { |                 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 || |                     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_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["cali_idx"].as<String>() != ams_data[i].trays[0].cali_idx) { |                         (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; |                         hasChanges = true; | ||||||
|                     } |                     } | ||||||
|                     break; |                     break; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             if (!foundExternal) hasChanges = true; |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (!hasChanges) return; |         if (!hasChanges) return; | ||||||
|  |  | ||||||
|         // Fortfahren mit der bestehenden Verarbeitung, da Änderungen gefunden wurden |         updateAmsWsData(doc, amsArray, ams_count, vtTray); | ||||||
|         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>(); |  | ||||||
|                 ams_data[i].trays[j].setting_id = trayObj["setting_id"].as<String>(); |  | ||||||
|                 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<String>()) { |  | ||||||
|             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_data[extIdx].trays[0].cali_idx = vtTray["cali_idx"].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>(); |  | ||||||
|  |  | ||||||
|         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); |  | ||||||
|         sendAmsData(nullptr); |  | ||||||
|     } |     } | ||||||
|  |      | ||||||
|     // Neue Bedingung für ams_filament_setting |     // Neue Bedingung für ams_filament_setting | ||||||
|     else if (doc["print"]["command"] == "ams_filament_setting") { |     if (doc["print"]["command"] == "ams_filament_setting") { | ||||||
|         int amsId = doc["print"]["ams_id"].as<int>(); |         int amsId = doc["print"]["ams_id"].as<int>(); | ||||||
|         int trayId = doc["print"]["tray_id"].as<int>(); |         int trayId = doc["print"]["tray_id"].as<int>(); | ||||||
|         String settingId = doc["print"]["setting_id"].as<String>(); |         String settingId = (doc["print"]["setting_id"].is<String>()) ? doc["print"]["setting_id"].as<String>() : ""; | ||||||
|          |  | ||||||
|         // Finde das entsprechende AMS und Tray |         // Finde das entsprechende AMS und Tray | ||||||
|         for (int i = 0; i < ams_count; i++) { |         for (int i = 0; i < ams_count; i++) { | ||||||
|             if (ams_data[i].ams_id == amsId) { |             if (ams_data[i].ams_id == amsId) { | ||||||
|                 // Update setting_id im entsprechenden Tray |                 if (trayId == 254) | ||||||
|                 ams_data[i].trays[trayId].setting_id = settingId; |                 { | ||||||
|                  |                     // Suche AMS mit ID 255 (externe Spule) | ||||||
|                 // Erstelle neues JSON für WebSocket-Clients |                     for (int j = 0; j < ams_count; j++) { | ||||||
|                 JsonDocument wsDoc; |                         if (ams_data[j].ams_id == 255) { | ||||||
|                 JsonArray wsArray = wsDoc.to<JsonArray>(); |                             ams_data[j].trays[0].setting_id = settingId; | ||||||
|  |                             break; | ||||||
|                 for (int j = 0; j < ams_count; j++) { |                         } | ||||||
|                     JsonObject amsObj = wsArray.add<JsonObject>(); |  | ||||||
|                     amsObj["ams_id"] = ams_data[j].ams_id; |  | ||||||
|  |  | ||||||
|                     JsonArray trays = amsObj["tray"].to<JsonArray>(); |  | ||||||
|                     int maxTrays = (ams_data[j].ams_id == 255) ? 1 : 4; |  | ||||||
|                      |  | ||||||
|                     for (int k = 0; k < maxTrays; k++) { |  | ||||||
|                         JsonObject trayObj = trays.add<JsonObject>(); |  | ||||||
|                         trayObj["id"] = ams_data[j].trays[k].id; |  | ||||||
|                         trayObj["tray_info_idx"] = ams_data[j].trays[k].tray_info_idx; |  | ||||||
|                         trayObj["tray_type"] = ams_data[j].trays[k].tray_type; |  | ||||||
|                         trayObj["tray_sub_brands"] = ams_data[j].trays[k].tray_sub_brands; |  | ||||||
|                         trayObj["tray_color"] = ams_data[j].trays[k].tray_color; |  | ||||||
|                         trayObj["nozzle_temp_min"] = ams_data[j].trays[k].nozzle_temp_min; |  | ||||||
|                         trayObj["nozzle_temp_max"] = ams_data[j].trays[k].nozzle_temp_max; |  | ||||||
|                         trayObj["setting_id"] = ams_data[j].trays[k].setting_id; |  | ||||||
|                         trayObj["cali_idx"] = ams_data[j].trays[k].cali_idx; |  | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|  |                 else | ||||||
|                 // Aktualisiere das globale amsJsonData |                 { | ||||||
|                 amsJsonData = ""; |                     ams_data[i].trays[trayId].setting_id = settingId; | ||||||
|                 serializeJson(wsArray, amsJsonData); |                 } | ||||||
|                  |                 | ||||||
|                 // Sende an WebSocket Clients |                 // Sende an WebSocket Clients | ||||||
|  |                 Serial.println("Filament setting updated"); | ||||||
|                 sendAmsData(nullptr); |                 sendAmsData(nullptr); | ||||||
|                 break; |                 break; | ||||||
|             } |             } | ||||||
| @@ -461,15 +543,16 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | |||||||
|  |  | ||||||
| void reconnect() { | void reconnect() { | ||||||
|     // Loop until we're reconnected |     // Loop until we're reconnected | ||||||
|  |     uint8_t retries = 0; | ||||||
|     while (!client.connected()) { |     while (!client.connected()) { | ||||||
|         Serial.print("Attempting MQTT connection..."); |         Serial.println("Attempting MQTT re/connection..."); | ||||||
|         bambu_connected = false; |         bambu_connected = false; | ||||||
|         oledShowTopRow(); |         oledShowTopRow(); | ||||||
|  |  | ||||||
|         // Attempt to connect |         // Attempt to connect | ||||||
|         if (client.connect(bambu_serialnr, bambu_username, bambu_accesscode)) { |         if (client.connect(bambu_serialnr, bambu_username, bambu_accesscode)) { | ||||||
|             Serial.println("... re-connected"); |             Serial.println("MQTT re/connected"); | ||||||
|             // ... and resubscribe |  | ||||||
|             client.subscribe(report_topic.c_str()); |             client.subscribe(report_topic.c_str()); | ||||||
|             bambu_connected = true; |             bambu_connected = true; | ||||||
|             oledShowTopRow(); |             oledShowTopRow(); | ||||||
| @@ -479,14 +562,23 @@ void reconnect() { | |||||||
|             Serial.println(" try again in 5 seconds"); |             Serial.println(" try again in 5 seconds"); | ||||||
|             bambu_connected = false; |             bambu_connected = false; | ||||||
|             oledShowTopRow(); |             oledShowTopRow(); | ||||||
|             // Wait 5 seconds before retrying |              | ||||||
|             yield(); |             yield(); | ||||||
|             vTaskDelay(5000 / portTICK_PERIOD_MS); |             vTaskDelay(5000 / portTICK_PERIOD_MS); | ||||||
|  |             if (retries > 5) { | ||||||
|  |                 Serial.println("Disable Bambu MQTT Task after 5 retries"); | ||||||
|  |                 //vTaskSuspend(BambuMqttTask); | ||||||
|  |                 vTaskDelete(BambuMqttTask); | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             retries++; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| void mqtt_loop(void * parameter) { | void mqtt_loop(void * parameter) { | ||||||
|  |     Serial.println("Bambu MQTT Task gestartet"); | ||||||
|     for(;;) { |     for(;;) { | ||||||
|         if (pauseBambuMqttTask) { |         if (pauseBambuMqttTask) { | ||||||
|             vTaskDelay(10000); |             vTaskDelay(10000); | ||||||
| @@ -500,6 +592,7 @@ void mqtt_loop(void * parameter) { | |||||||
|         } |         } | ||||||
|         client.loop(); |         client.loop(); | ||||||
|         yield(); |         yield(); | ||||||
|  |         esp_task_wdt_reset(); | ||||||
|         vTaskDelay(100); |         vTaskDelay(100); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -540,7 +633,7 @@ bool setupMqtt() { | |||||||
|             xTaskCreatePinnedToCore( |             xTaskCreatePinnedToCore( | ||||||
|                 mqtt_loop, /* Function to implement the task */ |                 mqtt_loop, /* Function to implement the task */ | ||||||
|                 "BambuMqtt", /* Name of the task */ |                 "BambuMqtt", /* Name of the task */ | ||||||
|                 10000,  /* Stack size in words */ |                 8192,  /* Stack size in words */ | ||||||
|                 NULL,  /* Task input parameter */ |                 NULL,  /* Task input parameter */ | ||||||
|                 mqttTaskPrio,  /* Priority of the task */ |                 mqttTaskPrio,  /* Priority of the task */ | ||||||
|                 &BambuMqttTask,  /* Task handle. */ |                 &BambuMqttTask,  /* Task handle. */ | ||||||
|   | |||||||
| @@ -28,9 +28,11 @@ extern bool bambu_connected; | |||||||
|  |  | ||||||
| extern int ams_count; | extern int ams_count; | ||||||
| extern AMSData ams_data[MAX_AMS]; | extern AMSData ams_data[MAX_AMS]; | ||||||
|  | extern bool autoSendToBambu; | ||||||
|  | extern int autoSetToBambuSpoolId; | ||||||
|  |  | ||||||
| bool loadBambuCredentials(); | 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(); | bool setupMqtt(); | ||||||
| void mqtt_loop(void * parameter); | void mqtt_loop(void * parameter); | ||||||
| bool setBambuSpool(String payload); | bool setBambuSpool(String payload); | ||||||
|   | |||||||
| @@ -1,8 +1,8 @@ | |||||||
| #include "commonFS.h" | #include "commonFS.h" | ||||||
| #include <SPIFFS.h> | #include <LittleFS.h> | ||||||
|  |  | ||||||
| bool saveJsonValue(const char* filename, const JsonDocument& doc) { | bool saveJsonValue(const char* filename, const JsonDocument& doc) { | ||||||
|     File file = SPIFFS.open(filename, "w"); |     File file = LittleFS.open(filename, "w"); | ||||||
|     if (!file) { |     if (!file) { | ||||||
|         Serial.print("Fehler beim Öffnen der Datei zum Schreiben: "); |         Serial.print("Fehler beim Öffnen der Datei zum Schreiben: "); | ||||||
|         Serial.println(filename); |         Serial.println(filename); | ||||||
| @@ -20,7 +20,7 @@ bool saveJsonValue(const char* filename, const JsonDocument& doc) { | |||||||
| } | } | ||||||
|  |  | ||||||
| bool loadJsonValue(const char* filename, JsonDocument& doc) { | bool loadJsonValue(const char* filename, JsonDocument& doc) { | ||||||
|     File file = SPIFFS.open(filename, "r"); |     File file = LittleFS.open(filename, "r"); | ||||||
|     if (!file) { |     if (!file) { | ||||||
|         Serial.print("Fehler beim Öffnen der Datei zum Lesen: "); |         Serial.print("Fehler beim Öffnen der Datei zum Lesen: "); | ||||||
|         Serial.println(filename); |         Serial.println(filename); | ||||||
| @@ -36,12 +36,12 @@ bool loadJsonValue(const char* filename, JsonDocument& doc) { | |||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| void initializeSPIFFS() { | void initializeFileSystem() { | ||||||
|     if (!SPIFFS.begin(true, "/spiffs", 10, "spiffs")) { |     if (!LittleFS.begin(true)) { | ||||||
|         Serial.println("SPIFFS Mount Failed"); |         Serial.println("LittleFS Mount Failed"); | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
|     Serial.printf("SPIFFS Total: %u bytes\n", SPIFFS.totalBytes()); |     Serial.printf("LittleFS Total: %u bytes\n", LittleFS.totalBytes()); | ||||||
|     Serial.printf("SPIFFS Used: %u bytes\n", SPIFFS.usedBytes()); |     Serial.printf("LittleFS Used: %u bytes\n", LittleFS.usedBytes()); | ||||||
|     Serial.printf("SPIFFS Free: %u bytes\n", SPIFFS.totalBytes() - SPIFFS.usedBytes()); |     Serial.printf("LittleFS Free: %u bytes\n", LittleFS.totalBytes() - LittleFS.usedBytes()); | ||||||
| } | } | ||||||
| @@ -2,11 +2,11 @@ | |||||||
| #define COMMONFS_H | #define COMMONFS_H | ||||||
|  |  | ||||||
| #include <Arduino.h> | #include <Arduino.h> | ||||||
| #include <SPIFFS.h> |  | ||||||
| #include <ArduinoJson.h> | #include <ArduinoJson.h> | ||||||
|  | #include <LittleFS.h> | ||||||
|  |  | ||||||
| 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 initializeSPIFFS(); | void initializeFileSystem(); | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|   | |||||||
| @@ -40,6 +40,10 @@ const uint8_t webserverPort = 80; | |||||||
| const char* apiUrl = "/api/v1"; | const char* apiUrl = "/api/v1"; | ||||||
| // ***** API | // ***** API | ||||||
|  |  | ||||||
|  | // ***** Bambu Auto Set Spool | ||||||
|  | uint8_t autoSetBambuAmsCounter = 60; | ||||||
|  | // ***** Bambu Auto Set Spool | ||||||
|  |  | ||||||
| // ***** Task Prios | // ***** Task Prios | ||||||
| uint8_t rfidTaskCore = 1; | uint8_t rfidTaskCore = 1; | ||||||
| uint8_t rfidTaskPrio = 1; | uint8_t rfidTaskPrio = 1; | ||||||
|   | |||||||
| @@ -23,6 +23,8 @@ extern const uint8_t OLED_DATA_END; | |||||||
| extern const char* apiUrl; | extern const char* apiUrl; | ||||||
| extern const uint8_t webserverPort; | extern const uint8_t webserverPort; | ||||||
|  |  | ||||||
|  | extern uint8_t autoSetBambuAmsCounter; | ||||||
|  |  | ||||||
| extern const unsigned char wifi_on[]; | extern const unsigned char wifi_on[]; | ||||||
| extern const unsigned char wifi_off[]; | extern const unsigned char wifi_off[]; | ||||||
| extern const unsigned char cloud_on[]; | extern const unsigned char cloud_on[]; | ||||||
|   | |||||||
| @@ -20,9 +20,9 @@ void setupDisplay() { | |||||||
|     // the library initializes this with an Adafruit splash screen. |     // the library initializes this with an Adafruit splash screen. | ||||||
|     display.setTextColor(WHITE); |     display.setTextColor(WHITE); | ||||||
|     display.display(); |     display.display(); | ||||||
|     delay(1000); // Pause for 2 seconds |  | ||||||
|     oledShowTopRow(); |     oledShowTopRow(); | ||||||
|     delay(2000); |     oledShowMessage("FilaMan v" + String(VERSION)); | ||||||
|  |     vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||||
| } | } | ||||||
|  |  | ||||||
| void oledclearline() { | void oledclearline() { | ||||||
| @@ -117,7 +117,6 @@ std::vector<String> splitTextIntoLines(String text, uint8_t textSize) { | |||||||
|         lines.push_back(currentLine); |         lines.push_back(currentLine); | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     Serial.println(lines.size()); |  | ||||||
|     return lines; |     return lines; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -140,8 +139,9 @@ void oledShowMultilineMessage(String message, uint8_t size) { | |||||||
|     int totalHeight = lines.size() * lineHeight; |     int totalHeight = lines.size() * lineHeight; | ||||||
|     int startY = OLED_DATA_START + ((OLED_DATA_END - OLED_DATA_START - totalHeight) / 2); |     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++) { |     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.print(lines[i]); | ||||||
|     } |     } | ||||||
|      |      | ||||||
|   | |||||||
							
								
								
									
										129
									
								
								src/main.cpp
									
									
									
									
									
								
							
							
						
						
									
										129
									
								
								src/main.cpp
									
									
									
									
									
								
							| @@ -1,6 +1,4 @@ | |||||||
| #include <Arduino.h> | #include <Arduino.h> | ||||||
| #include <DNSServer.h> |  | ||||||
| #include <ESPmDNS.h> |  | ||||||
| #include <Wire.h> | #include <Wire.h> | ||||||
| #include <WiFi.h> | #include <WiFi.h> | ||||||
|  |  | ||||||
| @@ -19,8 +17,14 @@ | |||||||
| void setup() { | void setup() { | ||||||
|   Serial.begin(115200); |   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 |   // Initialize SPIFFS | ||||||
|   initializeSPIFFS(); |   initializeFileSystem(); | ||||||
|  |  | ||||||
|   // Start Display |   // Start Display | ||||||
|   setupDisplay(); |   setupDisplay(); | ||||||
| @@ -29,7 +33,6 @@ void setup() { | |||||||
|   initWiFi(); |   initWiFi(); | ||||||
|  |  | ||||||
|   // Webserver |   // Webserver | ||||||
|   Serial.println("Starte Webserver"); |  | ||||||
|   setupWebserver(server); |   setupWebserver(server); | ||||||
|  |  | ||||||
|   // Spoolman API |   // Spoolman API | ||||||
| @@ -37,22 +40,27 @@ void setup() { | |||||||
|   initSpoolman(); |   initSpoolman(); | ||||||
|  |  | ||||||
|   // Bambu MQTT |   // Bambu MQTT | ||||||
|   // bambu.cpp |  | ||||||
|   setupMqtt(); |   setupMqtt(); | ||||||
|  |  | ||||||
|   // mDNS |   // NFC Reader | ||||||
|   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"); |  | ||||||
|    |  | ||||||
|   startNfc(); |   startNfc(); | ||||||
|  |  | ||||||
|   start_scale(); |   uint8_t scaleCalibrated = 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 | ||||||
| @@ -66,45 +74,85 @@ void setup() { | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * 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; | unsigned long lastWeightReadTime = 0; | ||||||
| const unsigned long weightReadInterval = 1000; // 1 second | const unsigned long weightReadInterval = 1000; // 1 second | ||||||
|  |  | ||||||
| unsigned long lastAmsSendTime = 0; | unsigned long lastAutoSetBambuAmsTime = 0; | ||||||
| const unsigned long amsSendInterval = 60000; // 1 minute | const unsigned long autoSetBambuAmsInterval = 1000; // 1 second | ||||||
|  | uint8_t autoAmsCounter = 0; | ||||||
|  |  | ||||||
| uint8_t weightSend = 0; | uint8_t weightSend = 0; | ||||||
| int16_t lastWeight = 0; | int16_t lastWeight = 0; | ||||||
| uint8_t wifiErrorCounter = 0; |  | ||||||
|  | unsigned long lastWifiCheckTime = 0; | ||||||
|  | const unsigned long wifiCheckInterval = 60000; // Überprüfe alle 60 Sekunden (60000 ms) | ||||||
|  |  | ||||||
| // ##### PROGRAM START ##### | // ##### PROGRAM START ##### | ||||||
| void loop() { | 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(); |   unsigned long currentMillis = millis(); | ||||||
|  |  | ||||||
|   // Send AMS Data min every Minute |   // Überprüfe regelmäßig die WLAN-Verbindung | ||||||
|   if (currentMillis - lastAmsSendTime >= amsSendInterval) { |   if (intervalElapsed(currentMillis, lastWifiCheckTime, wifiCheckInterval)) { | ||||||
|     lastAmsSendTime = currentMillis; |     checkWiFiConnection(); | ||||||
|     sendAmsData(nullptr); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Ausgabe der Waage auf Display |   // Wenn Bambu auto set Spool aktiv | ||||||
|   if (pauseMainTask == 0 && weight != lastWeight && hasReadRfidTag == 0) |   if (autoSendToBambu && autoSetToBambuSpoolId > 0) { | ||||||
|   { |     if (intervalElapsed(currentMillis, lastAutoSetBambuAmsTime, autoSetBambuAmsInterval))  | ||||||
|     (weight < 0) ? oledShowMessage("!! -1") : oledShowWeight(weight); |     { | ||||||
|  |       if (hasReadRfidTag == 0) | ||||||
|  |       { | ||||||
|  |         lastAutoSetBambuAmsTime = currentMillis; | ||||||
|  |         oledShowMessage("Auto Set         " + String(autoSetBambuAmsCounter - autoAmsCounter) + "s"); | ||||||
|  |         autoAmsCounter++; | ||||||
|  |  | ||||||
|  |         if (autoAmsCounter >= autoSetBambuAmsCounter)  | ||||||
|  |         { | ||||||
|  |           autoSetToBambuSpoolId = 0; | ||||||
|  |           autoAmsCounter = 0; | ||||||
|  |           oledShowWeight(weight); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       else | ||||||
|  |       { | ||||||
|  |         autoAmsCounter = 0; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   // Wenn Waage nicht Kalibriert | ||||||
|  |   if (scaleCalibrated == 3)  | ||||||
|  |   { | ||||||
|  |     oledShowMessage("Scale not calibrated!"); | ||||||
|  |     vTaskDelay(5000 / portTICK_PERIOD_MS); | ||||||
|  |     yield(); | ||||||
|  |     esp_task_wdt_reset(); | ||||||
|  |      | ||||||
|  |     return; | ||||||
|  |   }  | ||||||
|  |  | ||||||
|  |   // Ausgabe der Waage auf Display | ||||||
|  |   if (pauseMainTask == 0 && weight != lastWeight && hasReadRfidTag == 0 && (!autoSendToBambu || autoSetToBambuSpoolId == 0)) | ||||||
|  |   { | ||||||
|  |     (weight < 2) ? ((weight < -2) ? oledShowMessage("!! -0") : oledShowWeight(0)) : oledShowWeight(weight); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|   // Wenn Timer abgelaufen und nicht gerade ein RFID-Tag geschrieben wird |   // Wenn Timer abgelaufen und nicht gerade ein RFID-Tag geschrieben wird | ||||||
|   if (currentMillis - lastWeightReadTime >= weightReadInterval && hasReadRfidTag < 3) |   if (currentMillis - lastWeightReadTime >= weightReadInterval && hasReadRfidTag < 3) | ||||||
|   { |   { | ||||||
| @@ -148,6 +196,7 @@ void loop() { | |||||||
|       oledShowIcon("success"); |       oledShowIcon("success"); | ||||||
|       vTaskDelay(2000 / portTICK_PERIOD_MS); |       vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||||
|       weightSend = 1; |       weightSend = 1; | ||||||
|  |       autoSetToBambuSpoolId = spoolId.toInt(); | ||||||
|     } |     } | ||||||
|     else |     else | ||||||
|     { |     { | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								src/nfc.cpp
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								src/nfc.cpp
									
									
									
									
									
								
							| @@ -238,12 +238,14 @@ void writeJsonToTag(void *parameter) { | |||||||
|  |  | ||||||
|   hasReadRfidTag = 3; |   hasReadRfidTag = 3; | ||||||
|   vTaskSuspend(RfidReaderTask); |   vTaskSuspend(RfidReaderTask); | ||||||
|   vTaskDelay(500 / portTICK_PERIOD_MS); |   vTaskDelay(50 / portTICK_PERIOD_MS); | ||||||
|  |  | ||||||
|   //pauseBambuMqttTask = true; |   //pauseBambuMqttTask = true; | ||||||
|   // aktualisieren der Website wenn sich der Status ändert |   // aktualisieren der Website wenn sich der Status ändert | ||||||
|   sendNfcData(nullptr); |   sendNfcData(nullptr); | ||||||
|  |   vTaskDelay(100 / portTICK_PERIOD_MS); | ||||||
|   oledShowMessage("Waiting for NFC-Tag"); |   oledShowMessage("Waiting for NFC-Tag"); | ||||||
|  |    | ||||||
|   // Wait 10sec for tag |   // Wait 10sec for tag | ||||||
|   uint8_t success = 0; |   uint8_t success = 0; | ||||||
|   String uidString = ""; |   String uidString = ""; | ||||||
| @@ -331,7 +333,7 @@ void startWriteJsonToTag(const char* payload) { | |||||||
|     xTaskCreate( |     xTaskCreate( | ||||||
|         writeJsonToTag,        // Task-Funktion |         writeJsonToTag,        // Task-Funktion | ||||||
|         "WriteJsonToTagTask",       // Task-Name |         "WriteJsonToTagTask",       // Task-Name | ||||||
|         4096,                        // Stackgröße in Bytes |         5115,                        // Stackgröße in Bytes | ||||||
|         (void*)payloadCopy,         // Parameter |         (void*)payloadCopy,         // Parameter | ||||||
|         rfidWriteTaskPrio,           // Priorität |         rfidWriteTaskPrio,           // Priorität | ||||||
|         NULL                         // Task-Handle (nicht benötigt) |         NULL                         // Task-Handle (nicht benötigt) | ||||||
| @@ -420,7 +422,7 @@ void scanRfidTask(void * parameter) { | |||||||
|         //uidString = ""; |         //uidString = ""; | ||||||
|         nfcJsonData = ""; |         nfcJsonData = ""; | ||||||
|         Serial.println("Tag entfernt"); |         Serial.println("Tag entfernt"); | ||||||
|         oledShowWeight(0); |         if (!autoSendToBambu) oledShowWeight(weight); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       // aktualisieren der Website wenn sich der Status ändert |       // aktualisieren der Website wenn sich der Status ändert | ||||||
| @@ -456,7 +458,7 @@ void startNfc() { | |||||||
|     BaseType_t result = xTaskCreatePinnedToCore( |     BaseType_t result = xTaskCreatePinnedToCore( | ||||||
|       scanRfidTask, /* Function to implement the task */ |       scanRfidTask, /* Function to implement the task */ | ||||||
|       "RfidReader", /* Name of the task */ |       "RfidReader", /* Name of the task */ | ||||||
|       10000,  /* Stack size in words */ |       5115,  /* Stack size in words */ | ||||||
|       NULL,  /* Task input parameter */ |       NULL,  /* Task input parameter */ | ||||||
|       rfidTaskPrio,  /* Priority of the task */ |       rfidTaskPrio,  /* Priority of the task */ | ||||||
|       &RfidReaderTask,  /* Task handle. */ |       &RfidReaderTask,  /* Task handle. */ | ||||||
|   | |||||||
							
								
								
									
										205
									
								
								src/ota.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										205
									
								
								src/ota.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,205 @@ | |||||||
|  | #include <Arduino.h> | ||||||
|  | #include <website.h> | ||||||
|  | #include <commonFS.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; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 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); | ||||||
|  |      | ||||||
|  |     updateHandler->onUpload([](AsyncWebServerRequest *request, String filename, | ||||||
|  |                              size_t index, uint8_t *data, size_t len, bool final) { | ||||||
|  |         if (!index) { | ||||||
|  |             updateTotalSize = request->contentLength(); | ||||||
|  |             updateWritten = 0; | ||||||
|  |             isSpiffsUpdate = (filename.indexOf("website") > -1); | ||||||
|  |              | ||||||
|  |             if (isSpiffsUpdate) { | ||||||
|  |                 // Backup vor dem Update | ||||||
|  |                 sendUpdateProgress(0, "backup", "Backing up configurations..."); | ||||||
|  |                 delay(200); | ||||||
|  |                 backupJsonConfigs(); | ||||||
|  |                 delay(200); | ||||||
|  |                  | ||||||
|  |                 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..."); | ||||||
|  |                 delay(200); | ||||||
|  |             } else { | ||||||
|  |                 if (!Update.begin(updateTotalSize)) { | ||||||
|  |                     request->send(400, "application/json", "{\"success\":false,\"message\":\"Update initialization failed\"}"); | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |                 sendUpdateProgress(0, "starting", "Starting firmware update..."); | ||||||
|  |                 delay(200); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         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"); | ||||||
|  |                 oledShowMessage("Update: " + String(currentProgress) + "%"); | ||||||
|  |                 delay(50); | ||||||
|  |                 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
									
								
							
							
						
						
									
										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 | ||||||
| @@ -16,6 +16,7 @@ int16_t weight = 0; | |||||||
| uint8_t weigthCouterToApi = 0; | uint8_t weigthCouterToApi = 0; | ||||||
| uint8_t scale_tare_counter = 0; | uint8_t scale_tare_counter = 0; | ||||||
| uint8_t pauseMainTask = 0; | uint8_t pauseMainTask = 0; | ||||||
|  | uint8_t scaleCalibrated = 1; | ||||||
|  |  | ||||||
| Preferences preferences; | Preferences preferences; | ||||||
| const char* NVS_NAMESPACE = "scale"; | const char* NVS_NAMESPACE = "scale"; | ||||||
| @@ -50,11 +51,11 @@ void scale_loop(void * parameter) { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| void start_scale() { | uint8_t start_scale() { | ||||||
|   Serial.println("Prüfe Calibration Value"); |   Serial.println("Prüfe Calibration Value"); | ||||||
|   long calibrationValue; |   long calibrationValue; | ||||||
|  |  | ||||||
|   // NVS |   // 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.getLong(NVS_KEY_CALIBRATION, defaultScaleCalibrationValue); | ||||||
|   preferences.end(); |   preferences.end(); | ||||||
| @@ -64,7 +65,10 @@ void start_scale() { | |||||||
|  |  | ||||||
|   scale.begin(LOADCELL_DOUT_PIN, LOADCELL_SCK_PIN); |   scale.begin(LOADCELL_DOUT_PIN, LOADCELL_SCK_PIN); | ||||||
|  |  | ||||||
|   if (isnan(calibrationValue) || calibrationValue < 1) calibrationValue = defaultScaleCalibrationValue; |   if (isnan(calibrationValue) || calibrationValue < 1) { | ||||||
|  |     calibrationValue = defaultScaleCalibrationValue; | ||||||
|  |     scaleCalibrated = 0; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   oledShowMessage("Scale Tare Please remove all"); |   oledShowMessage("Scale Tare Please remove all"); | ||||||
|   for (uint16_t i = 0; i < 2000; i++) { |   for (uint16_t i = 0; i < 2000; i++) { | ||||||
| @@ -97,6 +101,8 @@ void 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() { | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ | |||||||
| #include "HX711.h" | #include "HX711.h" | ||||||
|  |  | ||||||
|  |  | ||||||
| void start_scale(); | uint8_t start_scale(); | ||||||
| uint8_t calibrate_scale(); | uint8_t calibrate_scale(); | ||||||
| uint8_t tareScale(); | uint8_t tareScale(); | ||||||
|  |  | ||||||
| @@ -14,6 +14,7 @@ extern int16_t weight; | |||||||
| extern uint8_t weigthCouterToApi; | extern uint8_t weigthCouterToApi; | ||||||
| extern uint8_t scale_tare_counter; | extern uint8_t scale_tare_counter; | ||||||
| extern uint8_t pauseMainTask; | extern uint8_t pauseMainTask; | ||||||
|  | extern uint8_t scaleCalibrated; | ||||||
|  |  | ||||||
| extern TaskHandle_t ScaleTask; | extern TaskHandle_t ScaleTask; | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										230
									
								
								src/website.cpp
									
									
									
									
									
								
							
							
						
						
									
										230
									
								
								src/website.cpp
									
									
									
									
									
								
							| @@ -8,6 +8,8 @@ | |||||||
| #include "scale.h" | #include "scale.h" | ||||||
| #include "esp_task_wdt.h" | #include "esp_task_wdt.h" | ||||||
| #include <Update.h> | #include <Update.h> | ||||||
|  | #include "display.h" | ||||||
|  | #include "ota.h" | ||||||
|  |  | ||||||
| #ifndef VERSION | #ifndef VERSION | ||||||
|   #define VERSION "1.1.0" |   #define VERSION "1.1.0" | ||||||
| @@ -22,6 +24,7 @@ AsyncWebSocket ws("/ws"); | |||||||
| uint8_t lastSuccess = 0; | uint8_t lastSuccess = 0; | ||||||
| uint8_t lastHasReadRfidTag = 0; | uint8_t lastHasReadRfidTag = 0; | ||||||
|  |  | ||||||
|  |  | ||||||
| void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventType type, void *arg, uint8_t *data, size_t len) { | void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventType type, void *arg, uint8_t *data, size_t len) { | ||||||
|     if (type == WS_EVT_CONNECT) { |     if (type == WS_EVT_CONNECT) { | ||||||
|         Serial.println("Neuer Client verbunden!"); |         Serial.println("Neuer Client verbunden!"); | ||||||
| @@ -32,11 +35,15 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | |||||||
|         sendWriteResult(client, 3); |         sendWriteResult(client, 3); | ||||||
|     } else if (type == WS_EVT_DISCONNECT) { |     } else if (type == WS_EVT_DISCONNECT) { | ||||||
|         Serial.println("Client getrennt."); |         Serial.println("Client getrennt."); | ||||||
|  |     } else if (type == WS_EVT_ERROR) { | ||||||
|  |         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) { |     } else if (type == WS_EVT_DATA) { | ||||||
|         String message = String((char*)data); |         String message = String((char*)data); | ||||||
|         JsonDocument doc; |         JsonDocument doc; | ||||||
|         deserializeJson(doc, message); |         deserializeJson(doc, message); | ||||||
|          |  | ||||||
|         if (doc["type"] == "heartbeat") { |         if (doc["type"] == "heartbeat") { | ||||||
|             // Sende Heartbeat-Antwort |             // Sende Heartbeat-Antwort | ||||||
|             ws.text(client->id(), "{" |             ws.text(client->id(), "{" | ||||||
| @@ -48,7 +55,7 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         else if (doc["type"] == "writeNfcTag") { |         else if (doc["type"] == "writeNfcTag") { | ||||||
|             if (doc["payload"].is<String>()) { |             if (doc["payload"].is<JsonObject>()) { | ||||||
|                 // Versuche NFC-Daten zu schreiben |                 // Versuche NFC-Daten zu schreiben | ||||||
|                 String payloadString; |                 String payloadString; | ||||||
|                 serializeJson(doc["payload"], payloadString); |                 serializeJson(doc["payload"], payloadString); | ||||||
| @@ -88,6 +95,15 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | |||||||
|             setBambuSpool(doc["payload"]); |             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 { |         else { | ||||||
|             Serial.println("Unbekannter WebSocket-Typ: " + doc["type"].as<String>()); |             Serial.println("Unbekannter WebSocket-Typ: " + doc["type"].as<String>()); | ||||||
|         } |         } | ||||||
| @@ -97,12 +113,12 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | |||||||
| // Funktion zum Laden und Ersetzen des Headers in einer HTML-Datei | // Funktion zum Laden und Ersetzen des Headers in einer HTML-Datei | ||||||
| String loadHtmlWithHeader(const char* filename) { | String loadHtmlWithHeader(const char* filename) { | ||||||
|     Serial.println("Lade HTML-Datei: " + String(filename)); |     Serial.println("Lade HTML-Datei: " + String(filename)); | ||||||
|     if (!SPIFFS.exists(filename)) { |     if (!LittleFS.exists(filename)) { | ||||||
|         Serial.println("Fehler: Datei nicht gefunden!"); |         Serial.println("Fehler: Datei nicht gefunden!"); | ||||||
|         return "Fehler: Datei nicht gefunden!"; |         return "Fehler: Datei nicht gefunden!"; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     File file = SPIFFS.open(filename, "r"); |     File file = LittleFS.open(filename, "r"); | ||||||
|     String html = file.readString(); |     String html = file.readString(); | ||||||
|     file.close(); |     file.close(); | ||||||
|  |  | ||||||
| @@ -160,6 +176,9 @@ void sendAmsData(AsyncWebSocketClient *client) { | |||||||
| } | } | ||||||
|  |  | ||||||
| void setupWebserver(AsyncWebServer &server) { | void setupWebserver(AsyncWebServer &server) { | ||||||
|  |     // Deaktiviere alle Debug-Ausgaben | ||||||
|  |     Serial.setDebugOutput(false); | ||||||
|  |      | ||||||
|     // WebSocket-Optimierungen |     // WebSocket-Optimierungen | ||||||
|     ws.onEvent(onWsEvent); |     ws.onEvent(onWsEvent); | ||||||
|     ws.enable(true); |     ws.enable(true); | ||||||
| @@ -176,7 +195,7 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|     // Route für about |     // Route für about | ||||||
|     server.on("/about", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/about", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|         Serial.println("Anfrage für /about erhalten"); |         Serial.println("Anfrage für /about erhalten"); | ||||||
|         AsyncWebServerResponse *response = request->beginResponse(SPIFFS, "/index.html.gz", "text/html"); |         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/index.html.gz", "text/html"); | ||||||
|         response->addHeader("Content-Encoding", "gzip"); |         response->addHeader("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); |         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |         request->send(response); | ||||||
| @@ -185,7 +204,7 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|     // Route für Waage |     // Route für Waage | ||||||
|     server.on("/waage", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/waage", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|         Serial.println("Anfrage für /waage erhalten"); |         Serial.println("Anfrage für /waage erhalten"); | ||||||
|         AsyncWebServerResponse *response = request->beginResponse(SPIFFS, "/waage.html.gz", "text/html"); |         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/waage.html.gz", "text/html"); | ||||||
|         response->addHeader("Content-Encoding", "gzip"); |         response->addHeader("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); |         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |         request->send(response); | ||||||
| @@ -194,24 +213,13 @@ 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(SPIFFS, "/rfid.html.gz", "text/html"); |         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/rfid.html.gz", "text/html"); | ||||||
|         response->addHeader("Content-Encoding", "gzip"); |         response->addHeader("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); |         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |         request->send(response); | ||||||
|         Serial.println("RFID-Seite gesendet"); |         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){ |     server.on("/api/url", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|         Serial.println("API-Aufruf: /api/url"); |         Serial.println("API-Aufruf: /api/url"); | ||||||
|         String jsonResponse = "{\"spoolman_url\": \"" + String(spoolmanUrl) + "\"}"; |         String jsonResponse = "{\"spoolman_url\": \"" + String(spoolmanUrl) + "\"}"; | ||||||
| @@ -221,7 +229,7 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|     // Route für WiFi |     // Route für WiFi | ||||||
|     server.on("/wifi", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/wifi", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|         Serial.println("Anfrage für /wifi erhalten"); |         Serial.println("Anfrage für /wifi erhalten"); | ||||||
|         AsyncWebServerResponse *response = request->beginResponse(SPIFFS, "/wifi.html.gz", "text/html"); |         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/wifi.html.gz", "text/html"); | ||||||
|         response->addHeader("Content-Encoding", "gzip"); |         response->addHeader("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); |         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |         request->send(response); | ||||||
| @@ -234,10 +242,12 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|         html.replace("{{spoolmanUrl}}", spoolmanUrl); |         html.replace("{{spoolmanUrl}}", spoolmanUrl); | ||||||
|  |  | ||||||
|         JsonDocument doc; |         JsonDocument doc; | ||||||
|         if (loadJsonValue("/bambu_credentials.json", doc) && doc["bambu_ip"].is<String>()) { |         if (loadJsonValue("/bambu_credentials.json", doc) && doc["bambu_ip"].is<String>())  | ||||||
|  |         { | ||||||
|             String bambuIp = doc["bambu_ip"].as<String>(); |             String bambuIp = doc["bambu_ip"].as<String>(); | ||||||
|             String bambuSerial = doc["bambu_serialnr"].as<String>(); |             String bambuSerial = doc["bambu_serialnr"].as<String>(); | ||||||
|             String bambuCode = doc["bambu_accesscode"].as<String>(); |             String bambuCode = doc["bambu_accesscode"].as<String>(); | ||||||
|  |             autoSendToBambu = doc["autoSendToBambu"].as<bool>(); | ||||||
|             bambuIp.trim(); |             bambuIp.trim(); | ||||||
|             bambuSerial.trim(); |             bambuSerial.trim(); | ||||||
|             bambuCode.trim(); |             bambuCode.trim(); | ||||||
| @@ -245,7 +255,17 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|             html.replace("{{bambuIp}}", bambuIp ? bambuIp : "");             |             html.replace("{{bambuIp}}", bambuIp ? bambuIp : "");             | ||||||
|             html.replace("{{bambuSerial}}", bambuSerial ? bambuSerial : ""); |             html.replace("{{bambuSerial}}", bambuSerial ? bambuSerial : ""); | ||||||
|             html.replace("{{bambuCode}}", bambuCode ? bambuCode : ""); |             html.replace("{{bambuCode}}", bambuCode ? bambuCode : ""); | ||||||
|         }    |             html.replace("{{autoSendToBambu}}", autoSendToBambu ? "checked" : ""); | ||||||
|  |             html.replace("{{autoSendTime}}", String(autoSetBambuAmsCounter)); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             html.replace("{{bambuIp}}", ""); | ||||||
|  |             html.replace("{{bambuSerial}}", ""); | ||||||
|  |             html.replace("{{bambuCode}}", ""); | ||||||
|  |             html.replace("{{autoSendToBambu}}", ""); | ||||||
|  |             html.replace("{{autoSendTime}}", String(autoSetBambuAmsCounter)); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         request->send(200, "text/html", html); |         request->send(200, "text/html", html); | ||||||
|     }); |     }); | ||||||
| @@ -276,16 +296,20 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|         String bambu_ip = request->getParam("bambu_ip")->value(); |         String bambu_ip = request->getParam("bambu_ip")->value(); | ||||||
|         String bambu_serialnr = request->getParam("bambu_serialnr")->value(); |         String bambu_serialnr = request->getParam("bambu_serialnr")->value(); | ||||||
|         String bambu_accesscode = request->getParam("bambu_accesscode")->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_ip.trim(); | ||||||
|         bambu_serialnr.trim(); |         bambu_serialnr.trim(); | ||||||
|         bambu_accesscode.trim(); |         bambu_accesscode.trim(); | ||||||
|  |         autoSendTime.trim(); | ||||||
|  |  | ||||||
|         if (bambu_ip.length() == 0 || bambu_serialnr.length() == 0 || bambu_accesscode.length() == 0) { |         if (bambu_ip.length() == 0 || bambu_serialnr.length() == 0 || bambu_accesscode.length() == 0) { | ||||||
|             request->send(400, "application/json", "{\"success\": false, \"error\": \"Empty parameter\"}"); |             request->send(400, "application/json", "{\"success\": false, \"error\": \"Empty parameter\"}"); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         bool success = saveBambuCredentials(bambu_ip, bambu_serialnr, bambu_accesscode); |         bool success = saveBambuCredentials(bambu_ip, bambu_serialnr, bambu_accesscode, autoSend, autoSendTime); | ||||||
|  |  | ||||||
|         request->send(200, "application/json", "{\"healthy\": " + String(success ? "true" : "false") + "}"); |         request->send(200, "application/json", "{\"healthy\": " + String(success ? "true" : "false") + "}"); | ||||||
|     }); |     }); | ||||||
| @@ -298,7 +322,7 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|     // Route für das Laden der CSS-Datei |     // Route für das Laden der CSS-Datei | ||||||
|     server.on("/style.css", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/style.css", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|         Serial.println("Lade style.css"); |         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("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); |         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |         request->send(response); | ||||||
| @@ -307,7 +331,7 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|  |  | ||||||
|     // Route für das Logo |     // Route für das Logo | ||||||
|     server.on("/logo.png", HTTP_GET, [](AsyncWebServerRequest *request){ |     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("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); |         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |         request->send(response); | ||||||
| @@ -316,7 +340,7 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|  |  | ||||||
|     // Route für Favicon |     // Route für Favicon | ||||||
|     server.on("/favicon.ico", HTTP_GET, [](AsyncWebServerRequest *request){ |     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); |         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |         request->send(response); | ||||||
|         Serial.println("favicon.ico gesendet"); |         Serial.println("favicon.ico gesendet"); | ||||||
| @@ -324,17 +348,26 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|  |  | ||||||
|     // Route für spool_in.png |     // Route für spool_in.png | ||||||
|     server.on("/spool_in.png", HTTP_GET, [](AsyncWebServerRequest *request){ |     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("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); |         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |         request->send(response); | ||||||
|         Serial.println("spool_in.png gesendet"); |         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 |     // Route für JavaScript Dateien | ||||||
|     server.on("/spoolman.js", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/spoolman.js", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|         Serial.println("Anfrage für /spoolman.js erhalten"); |         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("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); |         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |         request->send(response); | ||||||
| @@ -343,7 +376,7 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|  |  | ||||||
|     server.on("/rfid.js", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/rfid.js", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|         Serial.println("Anfrage für /rfid.js erhalten"); |         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("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); |         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||||
|         request->send(response); |         request->send(response); | ||||||
| @@ -352,72 +385,14 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|  |  | ||||||
|     // Vereinfachter Update-Handler |     // Vereinfachter Update-Handler | ||||||
|     server.on("/upgrade", HTTP_GET, [](AsyncWebServerRequest *request) { |     server.on("/upgrade", HTTP_GET, [](AsyncWebServerRequest *request) { | ||||||
|         AsyncWebServerResponse *response = request->beginResponse(SPIFFS, "/upgrade.html.gz", "text/html"); |         AsyncWebServerResponse *response = request->beginResponse(LittleFS, "/upgrade.html.gz", "text/html"); | ||||||
|         response->addHeader("Content-Encoding", "gzip"); |         response->addHeader("Content-Encoding", "gzip"); | ||||||
|         response->addHeader("Cache-Control", "no-store"); |         response->addHeader("Cache-Control", "no-store"); | ||||||
|         request->send(response); |         request->send(response); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     // Update-Handler mit verbesserter Fehlerbehandlung |     // Update-Handler registrieren | ||||||
|     server.on("/update", HTTP_POST,  |     handleUpdate(server); | ||||||
|         [](AsyncWebServerRequest *request) { |  | ||||||
|             // Nach Update-Abschluss |  | ||||||
|             bool success = !Update.hasError(); |  | ||||||
|             AsyncWebServerResponse *response = request->beginResponse( |  | ||||||
|                 success ? 200 : 400, |  | ||||||
|                 "application/json", |  | ||||||
|                 success ? "{\"success\":true,\"message\":\"Update successful\"}"  |  | ||||||
|                        : "{\"success\":false,\"message\":\"Update failed\"}" |  | ||||||
|             ); |  | ||||||
|             response->addHeader("Connection", "close"); |  | ||||||
|             request->send(response); |  | ||||||
|              |  | ||||||
|             if (success) { |  | ||||||
|                 delay(500); |  | ||||||
|                 ESP.restart(); |  | ||||||
|             } |  | ||||||
|         }, |  | ||||||
|         [](AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final) { |  | ||||||
|             static size_t updateSize = 0; |  | ||||||
|             static int command = 0; |  | ||||||
|  |  | ||||||
|             if (!index) { |  | ||||||
|                 updateSize = request->contentLength(); |  | ||||||
|                 command = (filename.indexOf("spiffs") > -1) ? U_SPIFFS : U_FLASH; |  | ||||||
|                 Serial.printf("Update Start: %s\nSize: %u\nCommand: %d\n", filename.c_str(), updateSize, command); |  | ||||||
|  |  | ||||||
|                 if (!Update.begin(updateSize, command)) { |  | ||||||
|                     Serial.printf("Update Begin Error: "); |  | ||||||
|                     Update.printError(Serial); |  | ||||||
|                     String errorMsg = String("Update begin failed: ") + Update.errorString(); |  | ||||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"" + errorMsg + "\"}"); |  | ||||||
|                     return; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if (len) { |  | ||||||
|                 if (Update.write(data, len) != len) { |  | ||||||
|                     Serial.printf("Update Write Error: "); |  | ||||||
|                     Update.printError(Serial); |  | ||||||
|                     String errorMsg = String("Write failed: ") + Update.errorString(); |  | ||||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"" + errorMsg + "\"}"); |  | ||||||
|                     return; |  | ||||||
|                 } |  | ||||||
|                 Serial.printf("Progress: %u/%u\r", index + len, updateSize); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if (final) { |  | ||||||
|                 if (!Update.end(true)) { |  | ||||||
|                     Serial.printf("Update End Error: "); |  | ||||||
|                     Update.printError(Serial); |  | ||||||
|                     String errorMsg = String("Update end failed: ") + Update.errorString(); |  | ||||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"" + errorMsg + "\"}"); |  | ||||||
|                     return; |  | ||||||
|                 } |  | ||||||
|                 Serial.printf("\nUpdate Success: %uB\n", index+len); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     server.on("/api/version", HTTP_GET, [](AsyncWebServerRequest *request){ |     server.on("/api/version", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|         String fm_version = VERSION; |         String fm_version = VERSION; | ||||||
| @@ -441,80 +416,3 @@ void setupWebserver(AsyncWebServer &server) { | |||||||
|     server.begin(); |     server.begin(); | ||||||
|     Serial.println("Webserver gestartet"); |     Serial.println("Webserver gestartet"); | ||||||
| } | } | ||||||
|  |  | ||||||
| void handleOTAUpload(AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final) { |  | ||||||
|     static bool isSpiffsUpdate = false; |  | ||||||
|     if (!index) { |  | ||||||
|         // Start eines neuen Uploads |  | ||||||
|         Serial.println("Update Start: " + filename); |  | ||||||
|          |  | ||||||
|         // Überprüfe den Dateityp basierend auf dem Dateinamen |  | ||||||
|         bool isFirmware = filename.startsWith("filaman_"); |  | ||||||
|         isSpiffsUpdate = filename.startsWith("webpage_"); |  | ||||||
|          |  | ||||||
|         if (!isFirmware && !isSpiffsUpdate) { |  | ||||||
|             request->send(400, "application/json", "{\"message\":\"Invalid file type. File must start with 'filaman_' or 'webpage_'\"}"); |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // Wähle den Update-Typ basierend auf dem Dateinamen |  | ||||||
|         if (isSpiffsUpdate) { |  | ||||||
|             if (!Update.begin(SPIFFS.totalBytes(), U_SPIFFS)) { |  | ||||||
|                 Update.printError(Serial); |  | ||||||
|                 request->send(400, "application/json", "{\"message\":\"SPIFFS Update failed: " + String(Update.errorString()) + "\"}"); |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|             // Backup JSON configs before SPIFFS update |  | ||||||
|             backupJsonConfigs(); |  | ||||||
|         } else { |  | ||||||
|             if (!Update.begin(UPDATE_SIZE_UNKNOWN, U_FLASH)) { |  | ||||||
|                 Update.printError(Serial); |  | ||||||
|                 request->send(400, "application/json", "{\"message\":\"Firmware Update failed: " + String(Update.errorString()) + "\"}"); |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (Update.write(data, len) != len) { |  | ||||||
|         Update.printError(Serial); |  | ||||||
|         request->send(400, "application/json", "{\"message\":\"Write failed: " + String(Update.errorString()) + "\"}"); |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (final) { |  | ||||||
|         if (!Update.end(true)) { |  | ||||||
|             Update.printError(Serial); |  | ||||||
|             request->send(400, "application/json", "{\"message\":\"Update failed: " + String(Update.errorString()) + "\"}"); |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (isSpiffsUpdate) { |  | ||||||
|             // Restore JSON configs after SPIFFS update |  | ||||||
|             restoreJsonConfigs(); |  | ||||||
|         } |  | ||||||
|         request->send(200, "application/json", "{\"message\":\"Update successful!\", \"restart\": true}"); |  | ||||||
|         delay(500); |  | ||||||
|         ESP.restart(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void backupJsonConfigs() { |  | ||||||
|     const char* configs[] = {"/bambu_credentials.json", "/spoolman_url.json"}; |  | ||||||
|     for (const char* config : configs) { |  | ||||||
|         if (SPIFFS.exists(config)) { |  | ||||||
|             String backupPath = String(config) + ".bak"; |  | ||||||
|             SPIFFS.remove(backupPath); |  | ||||||
|             SPIFFS.rename(config, backupPath); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void restoreJsonConfigs() { |  | ||||||
|     const char* configs[] = {"/bambu_credentials.json", "/spoolman_url.json"}; |  | ||||||
|     for (const char* config : configs) { |  | ||||||
|         String backupPath = String(config) + ".bak"; |  | ||||||
|         if (SPIFFS.exists(backupPath)) { |  | ||||||
|             SPIFFS.remove(config); |  | ||||||
|             SPIFFS.rename(backupPath, config); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -19,7 +19,6 @@ extern AsyncWebSocket ws; | |||||||
|  |  | ||||||
| // Server-Initialisierung und Handler | // Server-Initialisierung und Handler | ||||||
| void initWebServer(); | void initWebServer(); | ||||||
| void handleUpload(AsyncWebServerRequest *request, String filename, size_t index, uint8_t *data, size_t len, bool final); |  | ||||||
| void handleBody(AsyncWebServerRequest *request, uint8_t *data, size_t len, size_t index, size_t total); | void handleBody(AsyncWebServerRequest *request, uint8_t *data, size_t len, size_t index, size_t total); | ||||||
| void setupWebserver(AsyncWebServer &server); | void setupWebserver(AsyncWebServer &server); | ||||||
|  |  | ||||||
| @@ -29,8 +28,4 @@ void sendNfcData(AsyncWebSocketClient *client); | |||||||
| void foundNfcTag(AsyncWebSocketClient *client, uint8_t success); | void foundNfcTag(AsyncWebSocketClient *client, uint8_t success); | ||||||
| void sendWriteResult(AsyncWebSocketClient *client, uint8_t success); | void sendWriteResult(AsyncWebSocketClient *client, uint8_t success); | ||||||
|  |  | ||||||
| // Upgrade-Funktionen |  | ||||||
| void backupJsonConfigs(); |  | ||||||
| void restoreJsonConfigs(); |  | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|   | |||||||
							
								
								
									
										134
									
								
								src/wlan.cpp
									
									
									
									
									
								
							
							
						
						
									
										134
									
								
								src/wlan.cpp
									
									
									
									
									
								
							| @@ -3,16 +3,20 @@ | |||||||
| #include <WiFi.h> | #include <WiFi.h> | ||||||
| #include <esp_wifi.h> | #include <esp_wifi.h> | ||||||
| #include <WiFiManager.h> | #include <WiFiManager.h> | ||||||
|  | #include <DNSServer.h> | ||||||
|  | #include <ESPmDNS.h> | ||||||
| #include "display.h" | #include "display.h" | ||||||
| #include "config.h" | #include "config.h" | ||||||
|  |  | ||||||
| WiFiManager wm; | WiFiManager wm; | ||||||
| bool wm_nonblocking = false; | bool wm_nonblocking = false; | ||||||
|  | uint8_t wifiErrorCounter = 0; | ||||||
|  |  | ||||||
| void initWiFi() { | void wifiSettings() { | ||||||
|     // Optimierte WiFi-Einstellungen |     // Optimierte WiFi-Einstellungen | ||||||
|     WiFi.mode(WIFI_STA); // explicitly set mode, esp defaults to STA+AP |     WiFi.mode(WIFI_STA); // explicitly set mode, esp defaults to STA+AP | ||||||
|     WiFi.setSleep(false); // disable sleep mode |     WiFi.setSleep(false); // disable sleep mode | ||||||
|  |     WiFi.setHostname("FilaMan"); | ||||||
|     esp_wifi_set_ps(WIFI_PS_NONE); |     esp_wifi_set_ps(WIFI_PS_NONE); | ||||||
|      |      | ||||||
|     // Maximale Sendeleistung |     // Maximale Sendeleistung | ||||||
| @@ -23,33 +27,103 @@ void initWiFi() { | |||||||
|      |      | ||||||
|     // Aktiviere WiFi-Roaming für bessere Stabilität |     // Aktiviere WiFi-Roaming für bessere Stabilität | ||||||
|     esp_wifi_set_rssi_threshold(-80); |     esp_wifi_set_rssi_threshold(-80); | ||||||
|    | } | ||||||
|     if(wm_nonblocking) wm.setConfigPortalBlocking(false); |  | ||||||
|     wm.setConfigPortalTimeout(320); // Portal nach 5min schließen | void startMDNS() { | ||||||
|    |   if (!MDNS.begin("filaman")) { | ||||||
|     oledShowTopRow(); |     Serial.println("Error setting up MDNS responder!"); | ||||||
|     oledShowMessage("WiFi Setup"); |     while(1) { | ||||||
|      |       vTaskDelay(1000 / portTICK_PERIOD_MS); | ||||||
|     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(); |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |   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(5); | ||||||
|  |  | ||||||
|  |   oledShowTopRow(); | ||||||
|  |   oledShowMessage("WiFi Setup"); | ||||||
|  |    | ||||||
|  |   //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(); | ||||||
|  |     display.display(); | ||||||
|  |  | ||||||
|  |     vTaskDelay(500 / portTICK_PERIOD_MS); | ||||||
|  |  | ||||||
|  |     // 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(); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -4,5 +4,6 @@ | |||||||
| #include <Arduino.h> | #include <Arduino.h> | ||||||
|  |  | ||||||
| void initWiFi(); | void initWiFi(); | ||||||
|  | void checkWiFiConnection(); | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
		Reference in New Issue
	
	Block a user