Compare commits
10 Commits
v1.5.7
...
recyclingf
Author | SHA1 | Date | |
---|---|---|---|
3e04f385cb | |||
2eac7bbd1d | |||
a156cac18e | |||
|
09f4c43f89 | ||
|
ec0e544f30 | ||
|
b6d82c8afe | ||
|
97a1368747 | ||
|
85a9bcf8bd | ||
|
c450df59aa | ||
|
722ef421cb |
@@ -660,6 +660,7 @@ function writeNfcTag() {
|
|||||||
writeButton.textContent = "Writing";
|
writeButton.textContent = "Writing";
|
||||||
socket.send(JSON.stringify({
|
socket.send(JSON.stringify({
|
||||||
type: 'writeNfcTag',
|
type: 'writeNfcTag',
|
||||||
|
tagType: 'spool',
|
||||||
payload: nfcData
|
payload: nfcData
|
||||||
}));
|
}));
|
||||||
} else {
|
} else {
|
||||||
@@ -684,6 +685,7 @@ function writeLocationNfcTag() {
|
|||||||
writeButton.textContent = "Writing";
|
writeButton.textContent = "Writing";
|
||||||
socket.send(JSON.stringify({
|
socket.send(JSON.stringify({
|
||||||
type: 'writeNfcTag',
|
type: 'writeNfcTag',
|
||||||
|
tagType: 'location',
|
||||||
payload: nfcData
|
payload: nfcData
|
||||||
}));
|
}));
|
||||||
} else {
|
} else {
|
||||||
|
@@ -57,7 +57,7 @@
|
|||||||
<div class="update-options">
|
<div class="update-options">
|
||||||
<div class="update-section">
|
<div class="update-section">
|
||||||
<h2>Firmware Update</h2>
|
<h2>Firmware Update</h2>
|
||||||
<p>Upload a new firmware file (filaman_*.bin)</p>
|
<p>Upload a new firmware file (upgrade_filaman_firmware_*.bin)</p>
|
||||||
<div class="update-form">
|
<div class="update-form">
|
||||||
<form id="firmwareForm" enctype='multipart/form-data' data-type="firmware">
|
<form id="firmwareForm" enctype='multipart/form-data' data-type="firmware">
|
||||||
<input type='file' name='update' accept='.bin' required>
|
<input type='file' name='update' accept='.bin' required>
|
||||||
@@ -68,7 +68,7 @@
|
|||||||
|
|
||||||
<div class="update-section">
|
<div class="update-section">
|
||||||
<h2>Webpage Update</h2>
|
<h2>Webpage Update</h2>
|
||||||
<p>Upload a new webpage file (webpage_*.bin)</p>
|
<p>Upload a new webpage file (upgrade_filaman_website_*.bin)</p>
|
||||||
<div class="update-form">
|
<div class="update-form">
|
||||||
<form id="webpageForm" enctype='multipart/form-data' data-type="webpage">
|
<form id="webpageForm" enctype='multipart/form-data' data-type="webpage">
|
||||||
<input type='file' name='update' accept='.bin' required>
|
<input type='file' name='update' accept='.bin' required>
|
||||||
|
369
src/api.cpp
369
src/api.cpp
@@ -9,8 +9,14 @@ volatile spoolmanApiStateType spoolmanApiState = API_INIT;
|
|||||||
//bool spoolman_connected = false;
|
//bool spoolman_connected = false;
|
||||||
String spoolmanUrl = "";
|
String spoolmanUrl = "";
|
||||||
bool octoEnabled = false;
|
bool octoEnabled = false;
|
||||||
|
bool sendOctoUpdate = false;
|
||||||
String octoUrl = "";
|
String octoUrl = "";
|
||||||
String octoToken = "";
|
String octoToken = "";
|
||||||
|
uint16_t remainingWeight = 0;
|
||||||
|
uint16_t createdVendorId = 0; // Store ID of newly created vendor
|
||||||
|
uint16_t foundVendorId = 0; // Store ID of found vendor
|
||||||
|
uint16_t foundFilamentId = 0; // Store ID of found filament
|
||||||
|
bool spoolmanConnected = false;
|
||||||
|
|
||||||
struct SendToApiParams {
|
struct SendToApiParams {
|
||||||
SpoolmanApiRequestType requestType;
|
SpoolmanApiRequestType requestType;
|
||||||
@@ -111,6 +117,7 @@ void sendToApi(void *parameter) {
|
|||||||
int httpCode;
|
int httpCode;
|
||||||
if (httpType == "PATCH") httpCode = http.PATCH(updatePayload);
|
if (httpType == "PATCH") httpCode = http.PATCH(updatePayload);
|
||||||
else if (httpType == "POST") httpCode = http.POST(updatePayload);
|
else if (httpType == "POST") httpCode = http.POST(updatePayload);
|
||||||
|
else if (httpType == "GET") httpCode = http.GET();
|
||||||
else httpCode = http.PUT(updatePayload);
|
else httpCode = http.PUT(updatePayload);
|
||||||
|
|
||||||
if (httpCode == HTTP_CODE_OK) {
|
if (httpCode == HTTP_CODE_OK) {
|
||||||
@@ -124,23 +131,107 @@ void sendToApi(void *parameter) {
|
|||||||
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 {
|
||||||
if (requestType == API_REQUEST_SPOOL_WEIGHT_UPDATE) {
|
switch(requestType){
|
||||||
uint16_t remaining_weight = doc["remaining_weight"].as<float>();
|
case API_REQUEST_SPOOL_WEIGHT_UPDATE:
|
||||||
|
remainingWeight = doc["remaining_weight"].as<uint16_t>();
|
||||||
Serial.print("Aktuelles Gewicht: ");
|
Serial.print("Aktuelles Gewicht: ");
|
||||||
Serial.println(remaining_weight);
|
Serial.println(remainingWeight);
|
||||||
oledShowMessage("Remaining: " + String(remaining_weight) + "g");
|
//oledShowMessage("Remaining: " + String(remaining_weight) + "g");
|
||||||
|
if(!octoEnabled){
|
||||||
|
// TBD: Do not use Strings...
|
||||||
|
oledShowProgressBar(1, 1, "Spool Tag", ("Done: " + String(remainingWeight) + " g remain").c_str());
|
||||||
|
remainingWeight = 0;
|
||||||
|
}else{
|
||||||
|
// ocoto is enabled, trigger octo update
|
||||||
|
sendOctoUpdate = true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case API_REQUEST_SPOOL_LOCATION_UPDATE:
|
||||||
|
oledShowProgressBar(1, 1, "Loc. Tag", "Done!");
|
||||||
|
break;
|
||||||
|
case API_REQUEST_SPOOL_TAG_ID_UPDATE:
|
||||||
|
oledShowProgressBar(1, 1, "Write Tag", "Done!");
|
||||||
|
break;
|
||||||
|
case API_REQUEST_OCTO_SPOOL_UPDATE:
|
||||||
|
// TBD: Do not use Strings...
|
||||||
|
oledShowProgressBar(5, 5, "Spool Tag", ("Done: " + String(remainingWeight) + " g remain").c_str());
|
||||||
|
remainingWeight = 0;
|
||||||
|
break;
|
||||||
|
case API_REQUEST_VENDOR_CREATE:
|
||||||
|
Serial.println("Vendor successfully created!");
|
||||||
|
createdVendorId = doc["id"].as<uint16_t>();
|
||||||
|
Serial.print("Created Vendor ID: ");
|
||||||
|
Serial.println(createdVendorId);
|
||||||
|
oledShowProgressBar(1, 1, "Vendor", "Created!");
|
||||||
|
break;
|
||||||
|
case API_REQUEST_VENDOR_CHECK:
|
||||||
|
if (doc.isNull() || doc.size() == 0) {
|
||||||
|
Serial.println("Vendor not found in response");
|
||||||
|
foundVendorId = 0;
|
||||||
|
} else {
|
||||||
|
foundVendorId = doc[0]["id"].as<uint16_t>();
|
||||||
|
Serial.print("Found Vendor ID: ");
|
||||||
|
Serial.println(foundVendorId);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case API_REQUEST_FILAMENT_CHECK:
|
||||||
|
if (doc.isNull() || doc.size() == 0) {
|
||||||
|
Serial.println("Filament not found in response");
|
||||||
|
foundFilamentId = 0;
|
||||||
|
} else {
|
||||||
|
foundFilamentId = doc[0]["id"].as<uint16_t>();
|
||||||
|
Serial.print("Found Filament ID: ");
|
||||||
|
Serial.println(foundFilamentId);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
doc.clear();
|
||||||
|
} else if (httpCode == HTTP_CODE_CREATED) {
|
||||||
|
Serial.println("Spoolman erfolgreich erstellt");
|
||||||
|
|
||||||
|
// Parse response for created resources
|
||||||
|
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 {
|
||||||
|
switch(requestType){
|
||||||
|
case API_REQUEST_VENDOR_CREATE:
|
||||||
|
Serial.println("Vendor successfully created!");
|
||||||
|
createdVendorId = doc["id"].as<uint16_t>();
|
||||||
|
Serial.print("Created Vendor ID: ");
|
||||||
|
Serial.println(createdVendorId);
|
||||||
|
oledShowProgressBar(1, 1, "Vendor", "Created!");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
// Handle other create operations if needed
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
else if ( requestType == API_REQUEST_SPOOL_LOCATION_UPDATE) {
|
|
||||||
oledShowMessage("Location updated!");
|
|
||||||
}
|
|
||||||
|
|
||||||
vTaskDelay(3000 / portTICK_PERIOD_MS);
|
|
||||||
}
|
}
|
||||||
doc.clear();
|
doc.clear();
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
|
switch(requestType){
|
||||||
|
case API_REQUEST_SPOOL_WEIGHT_UPDATE:
|
||||||
|
case API_REQUEST_SPOOL_LOCATION_UPDATE:
|
||||||
|
case API_REQUEST_SPOOL_TAG_ID_UPDATE:
|
||||||
|
oledShowProgressBar(1, 1, "Failure!", "Spoolman update");
|
||||||
|
break;
|
||||||
|
case API_REQUEST_OCTO_SPOOL_UPDATE:
|
||||||
|
oledShowProgressBar(1, 1, "Failure!", "Octoprint update");
|
||||||
|
break;
|
||||||
|
case API_REQUEST_BAMBU_UPDATE:
|
||||||
|
oledShowProgressBar(1, 1, "Failure!", "Bambu update");
|
||||||
|
break;
|
||||||
|
case API_REQUEST_VENDOR_CREATE:
|
||||||
|
oledShowProgressBar(1, 1, "Failure!", "Vendor create");
|
||||||
|
break;
|
||||||
|
}
|
||||||
Serial.println("Fehler beim Senden an Spoolman! HTTP Code: " + String(httpCode));
|
Serial.println("Fehler beim Senden an Spoolman! HTTP Code: " + String(httpCode));
|
||||||
oledShowMessage("Spoolman update failed");
|
|
||||||
|
// TBD: really required?
|
||||||
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -155,6 +246,8 @@ void sendToApi(void *parameter) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool updateSpoolTagId(String uidString, const char* payload) {
|
bool updateSpoolTagId(String uidString, const char* payload) {
|
||||||
|
oledShowProgressBar(2, 3, "Write Tag", "Update Spoolman");
|
||||||
|
|
||||||
JsonDocument doc;
|
JsonDocument doc;
|
||||||
DeserializationError error = deserializeJson(doc, payload);
|
DeserializationError error = deserializeJson(doc, payload);
|
||||||
|
|
||||||
@@ -185,7 +278,7 @@ bool updateSpoolTagId(String uidString, const char* payload) {
|
|||||||
Serial.print("Update Payload: ");
|
Serial.print("Update Payload: ");
|
||||||
Serial.println(updatePayload);
|
Serial.println(updatePayload);
|
||||||
|
|
||||||
SendToApiParams* params = new SendToApiParams();
|
SendToApiParams* params = new SendToApiParams();
|
||||||
if (params == nullptr) {
|
if (params == nullptr) {
|
||||||
Serial.println("Fehler: Kann Speicher für Task-Parameter nicht allokieren.");
|
Serial.println("Fehler: Kann Speicher für Task-Parameter nicht allokieren.");
|
||||||
return false;
|
return false;
|
||||||
@@ -208,13 +301,15 @@ bool updateSpoolTagId(String uidString, const char* payload) {
|
|||||||
updateDoc.clear();
|
updateDoc.clear();
|
||||||
|
|
||||||
// Update Spool weight
|
// Update Spool weight
|
||||||
if (weight > 10) updateSpoolWeight(doc["sm_id"].as<String>(), weight);
|
//TBD: how to handle this with spool and locatin tags? Also potential parallel access again
|
||||||
|
//if (weight > 10) updateSpoolWeight(doc["sm_id"].as<String>(), weight);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t updateSpoolWeight(String spoolId, uint16_t weight) {
|
uint8_t updateSpoolWeight(String spoolId, uint16_t weight) {
|
||||||
HEAP_DEBUG_MESSAGE("updateSpoolWeight begin");
|
HEAP_DEBUG_MESSAGE("updateSpoolWeight begin");
|
||||||
|
oledShowProgressBar(3, octoEnabled?5:4, "Spool Tag", "Spoolman update");
|
||||||
String spoolsUrl = spoolmanUrl + apiUrl + "/spool/" + spoolId + "/measure";
|
String spoolsUrl = spoolmanUrl + apiUrl + "/spool/" + spoolId + "/measure";
|
||||||
Serial.print("Update Spule mit URL: ");
|
Serial.print("Update Spule mit URL: ");
|
||||||
Serial.println(spoolsUrl);
|
Serial.println(spoolsUrl);
|
||||||
@@ -230,6 +325,7 @@ uint8_t updateSpoolWeight(String spoolId, uint16_t weight) {
|
|||||||
|
|
||||||
SendToApiParams* params = new SendToApiParams();
|
SendToApiParams* params = new SendToApiParams();
|
||||||
if (params == nullptr) {
|
if (params == nullptr) {
|
||||||
|
// TBD: reset ESP instead of showing a message
|
||||||
Serial.println("Fehler: Kann Speicher für Task-Parameter nicht allokieren.");
|
Serial.println("Fehler: Kann Speicher für Task-Parameter nicht allokieren.");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -257,6 +353,8 @@ uint8_t updateSpoolWeight(String spoolId, uint16_t weight) {
|
|||||||
uint8_t updateSpoolLocation(String spoolId, String location){
|
uint8_t updateSpoolLocation(String spoolId, String location){
|
||||||
HEAP_DEBUG_MESSAGE("updateSpoolLocation begin");
|
HEAP_DEBUG_MESSAGE("updateSpoolLocation begin");
|
||||||
|
|
||||||
|
oledShowProgressBar(3, octoEnabled?5:4, "Loc. Tag", "Spoolman update");
|
||||||
|
|
||||||
String spoolsUrl = spoolmanUrl + apiUrl + "/spool/" + spoolId;
|
String spoolsUrl = spoolmanUrl + apiUrl + "/spool/" + spoolId;
|
||||||
Serial.print("Update Spule mit URL: ");
|
Serial.print("Update Spule mit URL: ");
|
||||||
Serial.println(spoolsUrl);
|
Serial.println(spoolsUrl);
|
||||||
@@ -280,6 +378,7 @@ uint8_t updateSpoolLocation(String spoolId, String location){
|
|||||||
params->spoolsUrl = spoolsUrl;
|
params->spoolsUrl = spoolsUrl;
|
||||||
params->updatePayload = updatePayload;
|
params->updatePayload = updatePayload;
|
||||||
|
|
||||||
|
if(spoolmanApiState == API_IDLE){
|
||||||
// Erstelle die Task
|
// Erstelle die Task
|
||||||
BaseType_t result = xTaskCreate(
|
BaseType_t result = xTaskCreate(
|
||||||
sendToApi, // Task-Funktion
|
sendToApi, // Task-Funktion
|
||||||
@@ -290,6 +389,10 @@ uint8_t updateSpoolLocation(String spoolId, String location){
|
|||||||
NULL // Task-Handle (nicht benötigt)
|
NULL // Task-Handle (nicht benötigt)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
}else{
|
||||||
|
Serial.println("Not spawning new task, API still active!");
|
||||||
|
}
|
||||||
|
|
||||||
updateDoc.clear();
|
updateDoc.clear();
|
||||||
|
|
||||||
HEAP_DEBUG_MESSAGE("updateSpoolLocation end");
|
HEAP_DEBUG_MESSAGE("updateSpoolLocation end");
|
||||||
@@ -297,6 +400,8 @@ uint8_t updateSpoolLocation(String spoolId, String location){
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool updateSpoolOcto(int spoolId) {
|
bool updateSpoolOcto(int spoolId) {
|
||||||
|
oledShowProgressBar(4, octoEnabled?5:4, "Spool Tag", "Octoprint update");
|
||||||
|
|
||||||
String spoolsUrl = octoUrl + "/plugin/Spoolman/selectSpool";
|
String spoolsUrl = octoUrl + "/plugin/Spoolman/selectSpool";
|
||||||
Serial.print("Update Spule in Octoprint mit URL: ");
|
Serial.print("Update Spule in Octoprint mit URL: ");
|
||||||
Serial.println(spoolsUrl);
|
Serial.println(spoolsUrl);
|
||||||
@@ -387,6 +492,238 @@ bool updateSpoolBambuData(String payload) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// #### Filament Fabrik
|
||||||
|
uint16_t checkVendor(String vendor) {
|
||||||
|
// Check if vendor exists using task system
|
||||||
|
foundVendorId = 0; // Reset previous value
|
||||||
|
|
||||||
|
String spoolsUrl = spoolmanUrl + apiUrl + "/vendor?name=" + vendor;
|
||||||
|
Serial.print("Check vendor with URL: ");
|
||||||
|
Serial.println(spoolsUrl);
|
||||||
|
|
||||||
|
SendToApiParams* params = new SendToApiParams();
|
||||||
|
if (params == nullptr) {
|
||||||
|
Serial.println("Fehler: Kann Speicher für Task-Parameter nicht allokieren.");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
params->requestType = API_REQUEST_VENDOR_CHECK;
|
||||||
|
params->httpType = "GET";
|
||||||
|
params->spoolsUrl = spoolsUrl;
|
||||||
|
params->updatePayload = ""; // Empty for GET request
|
||||||
|
|
||||||
|
// Check if API is idle before creating task
|
||||||
|
if(spoolmanApiState == API_IDLE){
|
||||||
|
// Erstelle die Task
|
||||||
|
BaseType_t result = xTaskCreate(
|
||||||
|
sendToApi, // Task-Funktion
|
||||||
|
"SendToApiTask", // Task-Name
|
||||||
|
6144, // Stackgröße in Bytes
|
||||||
|
(void*)params, // Parameter
|
||||||
|
0, // Priorität
|
||||||
|
NULL // Task-Handle (nicht benötigt)
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
Serial.println("Not spawning new task, API still active!");
|
||||||
|
delete params;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Wait for task completion
|
||||||
|
while(spoolmanApiState != API_IDLE) {
|
||||||
|
vTaskDelay(100 / portTICK_PERIOD_MS);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if vendor was found
|
||||||
|
if (foundVendorId == 0) {
|
||||||
|
Serial.println("Vendor not found, creating new vendor...");
|
||||||
|
uint16_t vendorId = createVendor(vendor);
|
||||||
|
if (vendorId == 0) {
|
||||||
|
Serial.println("Failed to create vendor, returning 0.");
|
||||||
|
return 0; // Failed to create vendor
|
||||||
|
} else {
|
||||||
|
Serial.println("Vendor created with ID: " + String(vendorId));
|
||||||
|
checkFilament(vendorId);
|
||||||
|
return vendorId;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Serial.println("Vendor found: " + vendor);
|
||||||
|
Serial.print("Vendor ID: ");
|
||||||
|
Serial.println(foundVendorId);
|
||||||
|
return foundVendorId;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16_t createVendor(String vendor) {
|
||||||
|
// Create new vendor in Spoolman database using task system
|
||||||
|
// Note: Due to async nature, the ID will be stored in createdVendorId global variable
|
||||||
|
createdVendorId = 0; // Reset previous value
|
||||||
|
|
||||||
|
String spoolsUrl = spoolmanUrl + apiUrl + "/vendor";
|
||||||
|
Serial.print("Create vendor with URL: ");
|
||||||
|
Serial.println(spoolsUrl);
|
||||||
|
|
||||||
|
// Create JSON payload for vendor creation
|
||||||
|
JsonDocument vendorDoc;
|
||||||
|
vendorDoc["name"] = vendor;
|
||||||
|
vendorDoc["comment"] = "automatically generated";
|
||||||
|
vendorDoc["empty_spool_weight"] = 180;
|
||||||
|
vendorDoc["external_id"] = vendor;
|
||||||
|
|
||||||
|
String vendorPayload;
|
||||||
|
serializeJson(vendorDoc, vendorPayload);
|
||||||
|
Serial.print("Vendor Payload: ");
|
||||||
|
Serial.println(vendorPayload);
|
||||||
|
|
||||||
|
SendToApiParams* params = new SendToApiParams();
|
||||||
|
if (params == nullptr) {
|
||||||
|
Serial.println("Fehler: Kann Speicher für Task-Parameter nicht allokieren.");
|
||||||
|
vendorDoc.clear();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
params->requestType = API_REQUEST_VENDOR_CREATE;
|
||||||
|
params->httpType = "POST";
|
||||||
|
params->spoolsUrl = spoolsUrl;
|
||||||
|
params->updatePayload = vendorPayload;
|
||||||
|
|
||||||
|
// Check if API is idle before creating task
|
||||||
|
if(spoolmanApiState == API_IDLE){
|
||||||
|
// Erstelle die Task
|
||||||
|
BaseType_t result = xTaskCreate(
|
||||||
|
sendToApi, // Task-Funktion
|
||||||
|
"SendToApiTask", // Task-Name
|
||||||
|
6144, // Stackgröße in Bytes
|
||||||
|
(void*)params, // Parameter
|
||||||
|
0, // Priorität
|
||||||
|
NULL // Task-Handle (nicht benötigt)
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
Serial.println("Not spawning new task, API still active!");
|
||||||
|
delete params;
|
||||||
|
vendorDoc.clear();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
vendorDoc.clear();
|
||||||
|
|
||||||
|
// Wait for task completion and return the created vendor ID
|
||||||
|
// Note: createdVendorId will be set by sendToApi when response is received
|
||||||
|
while(spoolmanApiState != API_IDLE) {
|
||||||
|
vTaskDelay(100 / portTICK_PERIOD_MS);
|
||||||
|
}
|
||||||
|
|
||||||
|
return createdVendorId;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16_t checkFilament(uint16_t vendorId) {
|
||||||
|
// Check if filament exists using task system
|
||||||
|
foundFilamentId = 0; // Reset previous value
|
||||||
|
|
||||||
|
String spoolsUrl = spoolmanUrl + apiUrl + "/filament?vendor.id=" + String(vendorId);
|
||||||
|
Serial.print("Check filament with URL: ");
|
||||||
|
Serial.println(spoolsUrl);
|
||||||
|
|
||||||
|
SendToApiParams* params = new SendToApiParams();
|
||||||
|
if (params == nullptr) {
|
||||||
|
Serial.println("Fehler: Kann Speicher für Task-Parameter nicht allokieren.");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
params->requestType = API_REQUEST_FILAMENT_CHECK;
|
||||||
|
params->httpType = "GET";
|
||||||
|
params->spoolsUrl = spoolsUrl;
|
||||||
|
params->updatePayload = ""; // Empty for GET request
|
||||||
|
|
||||||
|
// Check if API is idle before creating task
|
||||||
|
if(spoolmanApiState == API_IDLE){
|
||||||
|
// Erstelle die Task
|
||||||
|
BaseType_t result = xTaskCreate(
|
||||||
|
sendToApi, // Task-Funktion
|
||||||
|
"SendToApiTask", // Task-Name
|
||||||
|
6144, // Stackgröße in Bytes
|
||||||
|
(void*)params, // Parameter
|
||||||
|
0, // Priorität
|
||||||
|
NULL // Task-Handle (nicht benötigt)
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
Serial.println("Not spawning new task, API still active!");
|
||||||
|
delete params;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Wait for task completion
|
||||||
|
while(spoolmanApiState != API_IDLE) {
|
||||||
|
vTaskDelay(100 / portTICK_PERIOD_MS);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if filament was found
|
||||||
|
if (foundFilamentId == 0) {
|
||||||
|
Serial.println("Filament not found, creating new filament...");
|
||||||
|
uint16_t filamentId = createFilament();
|
||||||
|
if (filamentId == 0) {
|
||||||
|
Serial.println("Failed to create filament, returning 0.");
|
||||||
|
return 0; // Failed to create filament
|
||||||
|
} else {
|
||||||
|
Serial.println("Filament created with ID: " + String(filamentId));
|
||||||
|
checkSpool();
|
||||||
|
return filamentId;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Serial.println("Filament found for vendor ID: " + String(vendorId));
|
||||||
|
Serial.print("Filament ID: ");
|
||||||
|
Serial.println(foundFilamentId);
|
||||||
|
return foundFilamentId;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool createFilament() {
|
||||||
|
// {
|
||||||
|
// "name": "PolyTerra Charcoal Black",
|
||||||
|
// "vendor_id": 0,
|
||||||
|
// "material": "PLA",
|
||||||
|
// "price": 20,
|
||||||
|
// "density": 1.24,
|
||||||
|
// "diameter": 1.75,
|
||||||
|
// "weight": 1000,
|
||||||
|
// "spool_weight": 140,
|
||||||
|
// "article_number": "PM70820",
|
||||||
|
// "comment": "automatically generated",
|
||||||
|
// "settings_extruder_temp": 210,
|
||||||
|
// "settings_bed_temp": 60,
|
||||||
|
// "color_hex": "FF0000",
|
||||||
|
// "multi_color_hexes": "FF0000,00FF00,0000FF",
|
||||||
|
// "multi_color_direction": "coaxial",
|
||||||
|
// "external_id": "polymaker_pla_polysonicblack_1000_175",
|
||||||
|
// "extra": {
|
||||||
|
// "nozzle_temperature": "string"
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16_t checkSpool() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
bool createSpool() {
|
||||||
|
// Implement specific handling for Spool creation
|
||||||
|
// {
|
||||||
|
// "first_used": "2019-08-24T14:15:22Z",
|
||||||
|
// "last_used": "2019-08-24T14:15:22Z",
|
||||||
|
// "filament_id": 0,
|
||||||
|
// "price": 20,
|
||||||
|
// "initial_weight": 200,
|
||||||
|
// "spool_weight": 200,
|
||||||
|
// "remaining_weight": 800,
|
||||||
|
// "used_weight": 200,
|
||||||
|
// "location": "Shelf A",
|
||||||
|
// "lot_nr": "52342",
|
||||||
|
// "comment": "",
|
||||||
|
// "archived": false,
|
||||||
|
// "extra": {
|
||||||
|
// "nfc_id": "string"
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
}
|
||||||
|
|
||||||
// #### Spoolman init
|
// #### Spoolman init
|
||||||
bool checkSpoolmanExtraFields() {
|
bool checkSpoolmanExtraFields() {
|
||||||
HTTPClient http;
|
HTTPClient http;
|
||||||
@@ -551,9 +888,6 @@ bool checkSpoolmanInstance(const String& url) {
|
|||||||
|
|
||||||
if (httpCode > 0) {
|
if (httpCode > 0) {
|
||||||
if (httpCode == HTTP_CODE_OK) {
|
if (httpCode == HTTP_CODE_OK) {
|
||||||
oledShowMessage("Spoolman available");
|
|
||||||
vTaskDelay(1000 / portTICK_PERIOD_MS);
|
|
||||||
|
|
||||||
String payload = http.getString();
|
String payload = http.getString();
|
||||||
JsonDocument doc;
|
JsonDocument doc;
|
||||||
DeserializationError error = deserializeJson(doc, payload);
|
DeserializationError error = deserializeJson(doc, payload);
|
||||||
@@ -564,6 +898,7 @@ bool checkSpoolmanInstance(const String& url) {
|
|||||||
if (!checkSpoolmanExtraFields()) {
|
if (!checkSpoolmanExtraFields()) {
|
||||||
Serial.println("Fehler beim Überprüfen der Extrafelder.");
|
Serial.println("Fehler beim Überprüfen der Extrafelder.");
|
||||||
|
|
||||||
|
// TBD
|
||||||
oledShowMessage("Spoolman Error creating Extrafields");
|
oledShowMessage("Spoolman Error creating Extrafields");
|
||||||
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
||||||
|
|
||||||
@@ -572,6 +907,7 @@ bool checkSpoolmanInstance(const String& url) {
|
|||||||
|
|
||||||
spoolmanApiState = API_IDLE;
|
spoolmanApiState = API_IDLE;
|
||||||
oledShowTopRow();
|
oledShowTopRow();
|
||||||
|
spoolmanConnected = true;
|
||||||
return strcmp(status, "healthy") == 0;
|
return strcmp(status, "healthy") == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -617,6 +953,7 @@ String loadSpoolmanUrl() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool initSpoolman() {
|
bool initSpoolman() {
|
||||||
|
oledShowProgressBar(3, 7, DISPLAY_BOOT_TEXT, "Spoolman init");
|
||||||
spoolmanUrl = loadSpoolmanUrl();
|
spoolmanUrl = loadSpoolmanUrl();
|
||||||
spoolmanUrl.trim();
|
spoolmanUrl.trim();
|
||||||
if (spoolmanUrl == "") {
|
if (spoolmanUrl == "") {
|
||||||
|
17
src/api.h
17
src/api.h
@@ -17,15 +17,23 @@ typedef enum {
|
|||||||
API_REQUEST_BAMBU_UPDATE,
|
API_REQUEST_BAMBU_UPDATE,
|
||||||
API_REQUEST_SPOOL_TAG_ID_UPDATE,
|
API_REQUEST_SPOOL_TAG_ID_UPDATE,
|
||||||
API_REQUEST_SPOOL_WEIGHT_UPDATE,
|
API_REQUEST_SPOOL_WEIGHT_UPDATE,
|
||||||
API_REQUEST_SPOOL_LOCATION_UPDATE
|
API_REQUEST_SPOOL_LOCATION_UPDATE,
|
||||||
|
API_REQUEST_VENDOR_CREATE,
|
||||||
|
API_REQUEST_VENDOR_CHECK,
|
||||||
|
API_REQUEST_FILAMENT_CHECK
|
||||||
} SpoolmanApiRequestType;
|
} SpoolmanApiRequestType;
|
||||||
|
|
||||||
extern volatile spoolmanApiStateType spoolmanApiState;
|
extern volatile spoolmanApiStateType spoolmanApiState;
|
||||||
extern bool spoolman_connected;
|
extern bool spoolman_connected;
|
||||||
extern String spoolmanUrl;
|
extern String spoolmanUrl;
|
||||||
extern bool octoEnabled;
|
extern bool octoEnabled;
|
||||||
|
extern bool sendOctoUpdate;
|
||||||
extern String octoUrl;
|
extern String octoUrl;
|
||||||
extern String octoToken;
|
extern String octoToken;
|
||||||
|
extern uint16_t createdVendorId; // ID of newly created vendor
|
||||||
|
extern uint16_t foundVendorId; // ID of found vendor
|
||||||
|
extern uint16_t foundFilamentId; // ID of found filament
|
||||||
|
extern bool spoolmanConnected;
|
||||||
|
|
||||||
bool checkSpoolmanInstance(const String& url);
|
bool checkSpoolmanInstance(const String& url);
|
||||||
bool saveSpoolmanUrl(const String& url, bool octoOn, const String& octoWh, const String& octoTk);
|
bool saveSpoolmanUrl(const String& url, bool octoOn, const String& octoWh, const String& octoTk);
|
||||||
@@ -38,5 +46,12 @@ uint8_t updateSpoolLocation(String spoolId, String location);
|
|||||||
bool initSpoolman(); // Neue Funktion zum Initialisieren von Spoolman
|
bool initSpoolman(); // Neue Funktion zum Initialisieren von Spoolman
|
||||||
bool updateSpoolBambuData(String payload); // Neue Funktion zum Aktualisieren der Bambu-Daten
|
bool updateSpoolBambuData(String payload); // Neue Funktion zum Aktualisieren der Bambu-Daten
|
||||||
bool updateSpoolOcto(int spoolId); // Neue Funktion zum Aktualisieren der Octo-Daten
|
bool updateSpoolOcto(int spoolId); // Neue Funktion zum Aktualisieren der Octo-Daten
|
||||||
|
uint16_t checkVendor(String vendor); // Check if vendor exists, return ID
|
||||||
|
uint16_t createVendor(String vendor); // Create vendor, return ID
|
||||||
|
uint16_t checkFilament(); // Check if filament exists, return ID
|
||||||
|
bool createFilament(); // Create filament
|
||||||
|
uint16_t checkSpool(); // Check if spool exists, return ID
|
||||||
|
bool createSpool(); // Create spool
|
||||||
|
void createFilamentFabrik(JsonDocument payload);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@@ -627,6 +627,7 @@ bool setupMqtt() {
|
|||||||
|
|
||||||
if (bambuCredentials.ip != "" && bambuCredentials.accesscode != "" && bambuCredentials.serial != "")
|
if (bambuCredentials.ip != "" && bambuCredentials.accesscode != "" && bambuCredentials.serial != "")
|
||||||
{
|
{
|
||||||
|
oledShowProgressBar(4, 7, DISPLAY_BOOT_TEXT, "Bambu init");
|
||||||
bambuDisabled = false;
|
bambuDisabled = false;
|
||||||
sslClient.setCACert(root_ca);
|
sslClient.setCACert(root_ca);
|
||||||
sslClient.setInsecure();
|
sslClient.setInsecure();
|
||||||
|
@@ -26,16 +26,11 @@ const uint8_t TTP223_PIN = 25;
|
|||||||
|
|
||||||
|
|
||||||
// ***** Display
|
// ***** Display
|
||||||
// Declaration for an SSD1306 display connected to I2C (SDA, SCL pins)
|
|
||||||
// On an ESP32: 21(SDA), 22(SCL)
|
|
||||||
const int8_t OLED_RESET = -1; // Reset pin # (or -1 if sharing Arduino reset pin)
|
|
||||||
const uint8_t SCREEN_ADDRESS = 0x3C; ///< See datasheet for Address; 0x3D for 128x64, 0x3C for 128x32
|
|
||||||
const uint8_t SCREEN_WIDTH = 128; // OLED display width, in pixels
|
|
||||||
const uint8_t SCREEN_HEIGHT = 64; // OLED display height, in pixels
|
|
||||||
const uint8_t OLED_TOP_START = 0;
|
const uint8_t OLED_TOP_START = 0;
|
||||||
const uint8_t OLED_TOP_END = 16;
|
const uint8_t OLED_TOP_END = 16;
|
||||||
const uint8_t OLED_DATA_START = 17;
|
const uint8_t OLED_DATA_START = 17;
|
||||||
const uint8_t OLED_DATA_END = SCREEN_HEIGHT;
|
const uint8_t OLED_DATA_END = SCREEN_HEIGHT;
|
||||||
|
|
||||||
// ***** Display
|
// ***** Display
|
||||||
|
|
||||||
// ***** Webserver
|
// ***** Webserver
|
||||||
|
12
src/config.h
12
src/config.h
@@ -25,6 +25,14 @@
|
|||||||
|
|
||||||
#define BAMBU_USERNAME "bblp"
|
#define BAMBU_USERNAME "bblp"
|
||||||
|
|
||||||
|
#define OLED_RESET -1 // Reset pin # (or -1 if sharing Arduino reset pin)
|
||||||
|
#define SCREEN_ADDRESS 0x3CU // See datasheet for Address; 0x3D for 128x64, 0x3C for 128x32
|
||||||
|
#define SCREEN_WIDTH 128U
|
||||||
|
#define SCREEN_HEIGHT 64U
|
||||||
|
#define SCREEN_TOP_BAR_HEIGHT 16U
|
||||||
|
#define SCREEN_PROGRESS_BAR_HEIGHT 12U
|
||||||
|
#define DISPLAY_BOOT_TEXT "FilaMan"
|
||||||
|
|
||||||
|
|
||||||
extern const uint8_t PN532_IRQ;
|
extern const uint8_t PN532_IRQ;
|
||||||
extern const uint8_t PN532_RESET;
|
extern const uint8_t PN532_RESET;
|
||||||
@@ -36,10 +44,6 @@ extern const uint16_t SCALE_LEVEL_WEIGHT;
|
|||||||
|
|
||||||
extern const uint8_t TTP223_PIN;
|
extern const uint8_t TTP223_PIN;
|
||||||
|
|
||||||
extern const int8_t OLED_RESET;
|
|
||||||
extern const uint8_t SCREEN_ADDRESS;
|
|
||||||
extern const uint8_t SCREEN_WIDTH;
|
|
||||||
extern const uint8_t SCREEN_HEIGHT;
|
|
||||||
extern const uint8_t OLED_TOP_START;
|
extern const uint8_t OLED_TOP_START;
|
||||||
extern const uint8_t OLED_TOP_END;
|
extern const uint8_t OLED_TOP_END;
|
||||||
extern const uint8_t OLED_DATA_START;
|
extern const uint8_t OLED_DATA_START;
|
||||||
|
@@ -7,6 +7,6 @@
|
|||||||
#define HEAP_DEBUG_MESSAGE(location)
|
#define HEAP_DEBUG_MESSAGE(location)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
inline void printHeapDebugData(String location){
|
inline void printHeapDebugData(const char *location){
|
||||||
Serial.println("Heap: " + String(ESP.getMinFreeHeap()/1024) + "\t" + String(ESP.getFreeHeap()/1024) + "\t" + String(ESP.getMaxAllocHeap()/1024) + "\t" + location);
|
Serial.println("Heap: " + String(ESP.getMinFreeHeap()/1024) + "\t" + String(ESP.getFreeHeap()/1024) + "\t" + String(ESP.getMaxAllocHeap()/1024) + "\t" + location);
|
||||||
}
|
}
|
@@ -2,10 +2,12 @@
|
|||||||
#include "api.h"
|
#include "api.h"
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include "icons.h"
|
#include "icons.h"
|
||||||
|
#include "main.h"
|
||||||
|
|
||||||
Adafruit_SSD1306 display(SCREEN_WIDTH, SCREEN_HEIGHT, &Wire, OLED_RESET);
|
Adafruit_SSD1306 display(SCREEN_WIDTH, SCREEN_HEIGHT, &Wire, OLED_RESET);
|
||||||
|
|
||||||
bool wifiOn = false;
|
bool wifiOn = false;
|
||||||
|
bool iconToggle = false;
|
||||||
|
|
||||||
void setupDisplay() {
|
void setupDisplay() {
|
||||||
if (!display.begin(SSD1306_SWITCHCAPVCC, SCREEN_ADDRESS)) {
|
if (!display.begin(SSD1306_SWITCHCAPVCC, SCREEN_ADDRESS)) {
|
||||||
@@ -14,15 +16,10 @@ void setupDisplay() {
|
|||||||
}
|
}
|
||||||
display.setTextColor(WHITE);
|
display.setTextColor(WHITE);
|
||||||
display.clearDisplay();
|
display.clearDisplay();
|
||||||
display.display();
|
|
||||||
|
|
||||||
// Show initial display buffer contents on the screen --
|
|
||||||
// the library initializes this with an Adafruit splash screen.
|
|
||||||
display.setTextColor(WHITE);
|
|
||||||
display.display();
|
|
||||||
oledShowTopRow();
|
oledShowTopRow();
|
||||||
oledShowMessage("FilaMan v" + String(VERSION));
|
oledShowProgressBar(0, 7, DISPLAY_BOOT_TEXT, "Display init");
|
||||||
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void oledclearline() {
|
void oledclearline() {
|
||||||
@@ -45,14 +42,14 @@ void oledcleardata() {
|
|||||||
//display.display();
|
//display.display();
|
||||||
}
|
}
|
||||||
|
|
||||||
int oled_center_h(String text) {
|
int oled_center_h(const String &text) {
|
||||||
int16_t x1, y1;
|
int16_t x1, y1;
|
||||||
uint16_t w, h;
|
uint16_t w, h;
|
||||||
display.getTextBounds(text, 0, 0, &x1, &y1, &w, &h);
|
display.getTextBounds(text, 0, 0, &x1, &y1, &w, &h);
|
||||||
return (SCREEN_WIDTH - w) / 2;
|
return (SCREEN_WIDTH - w) / 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
int oled_center_v(String text) {
|
int oled_center_v(const String &text) {
|
||||||
int16_t x1, y1;
|
int16_t x1, y1;
|
||||||
uint16_t w, h;
|
uint16_t w, h;
|
||||||
display.getTextBounds(text, 0, OLED_DATA_START, &x1, &y1, &w, &h);
|
display.getTextBounds(text, 0, OLED_DATA_START, &x1, &y1, &w, &h);
|
||||||
@@ -60,7 +57,7 @@ int oled_center_v(String text) {
|
|||||||
return OLED_DATA_START + ((OLED_DATA_END - OLED_DATA_START - h) / 2);
|
return OLED_DATA_START + ((OLED_DATA_END - OLED_DATA_START - h) / 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<String> splitTextIntoLines(String text, uint8_t textSize) {
|
std::vector<String> splitTextIntoLines(const String &text, uint8_t textSize) {
|
||||||
std::vector<String> lines;
|
std::vector<String> lines;
|
||||||
display.setTextSize(textSize);
|
display.setTextSize(textSize);
|
||||||
|
|
||||||
@@ -120,7 +117,7 @@ std::vector<String> splitTextIntoLines(String text, uint8_t textSize) {
|
|||||||
return lines;
|
return lines;
|
||||||
}
|
}
|
||||||
|
|
||||||
void oledShowMultilineMessage(String message, uint8_t size) {
|
void oledShowMultilineMessage(const String &message, uint8_t size) {
|
||||||
std::vector<String> lines;
|
std::vector<String> lines;
|
||||||
int maxLines = 3; // Maximale Anzahl Zeilen für size 2
|
int maxLines = 3; // Maximale Anzahl Zeilen für size 2
|
||||||
|
|
||||||
@@ -148,7 +145,7 @@ void oledShowMultilineMessage(String message, uint8_t size) {
|
|||||||
display.display();
|
display.display();
|
||||||
}
|
}
|
||||||
|
|
||||||
void oledShowMessage(String message, uint8_t size) {
|
void oledShowMessage(const String &message, uint8_t size) {
|
||||||
oledcleardata();
|
oledcleardata();
|
||||||
display.setTextSize(size);
|
display.setTextSize(size);
|
||||||
display.setTextWrap(false);
|
display.setTextWrap(false);
|
||||||
@@ -171,22 +168,46 @@ void oledShowMessage(String message, uint8_t size) {
|
|||||||
void oledShowTopRow() {
|
void oledShowTopRow() {
|
||||||
oledclearline();
|
oledclearline();
|
||||||
|
|
||||||
if (bambu_connected == 1) {
|
display.setTextSize(1);
|
||||||
display.drawBitmap(50, 0, bitmap_bambu_on , 16, 16, WHITE);
|
display.setCursor(0, 4);
|
||||||
} else {
|
display.print("v");
|
||||||
display.drawBitmap(50, 0, bitmap_off , 16, 16, WHITE);
|
display.print(VERSION);
|
||||||
}
|
|
||||||
|
|
||||||
if (spoolmanApiState != API_INIT) {
|
iconToggle = !iconToggle;
|
||||||
display.drawBitmap(80, 0, bitmap_spoolman_on , 16, 16, WHITE);
|
|
||||||
} else {
|
|
||||||
display.drawBitmap(80, 0, bitmap_off , 16, 16, WHITE);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (wifiOn == 1) {
|
// Do not show status indicators during boot
|
||||||
display.drawBitmap(107, 0, wifi_on , 16, 16, WHITE);
|
if(!booting){
|
||||||
} else {
|
if(bambuDisabled == false) {
|
||||||
display.drawBitmap(107, 0, wifi_off , 16, 16, WHITE);
|
if (bambu_connected == 1) {
|
||||||
|
display.drawBitmap(50, 0, bitmap_bambu_on , 16, 16, WHITE);
|
||||||
|
} else {
|
||||||
|
if(iconToggle){
|
||||||
|
display.drawBitmap(50, 0, bitmap_bambu_on , 16, 16, WHITE);
|
||||||
|
display.drawLine(50, 15, 66, 0, WHITE);
|
||||||
|
display.drawLine(51, 15, 67, 0, WHITE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (spoolmanConnected) {
|
||||||
|
display.drawBitmap(80, 0, bitmap_spoolman_on , 16, 16, WHITE);
|
||||||
|
} else {
|
||||||
|
if(iconToggle){
|
||||||
|
display.drawBitmap(80, 0, bitmap_spoolman_on , 16, 16, WHITE);
|
||||||
|
display.drawLine(80, 15, 96, 0, WHITE);
|
||||||
|
display.drawLine(81, 15, 97, 0, WHITE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (wifiOn == 1) {
|
||||||
|
display.drawBitmap(107, 0, wifi_on , 16, 16, WHITE);
|
||||||
|
} else {
|
||||||
|
if(iconToggle){
|
||||||
|
display.drawBitmap(107, 0, wifi_on , 16, 16, WHITE);
|
||||||
|
display.drawLine(107, 15, 123, 0, WHITE);
|
||||||
|
display.drawLine(108, 15, 124, 0, WHITE);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
display.display();
|
display.display();
|
||||||
@@ -214,6 +235,27 @@ void oledShowIcon(const char* icon) {
|
|||||||
display.display();
|
display.display();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void oledShowProgressBar(const uint8_t step, const uint8_t numSteps, const char* largeText, const char* statusMessage){
|
||||||
|
assert(step <= numSteps);
|
||||||
|
|
||||||
|
// clear data and bar area
|
||||||
|
display.fillRect(0, OLED_DATA_START, SCREEN_WIDTH, SCREEN_HEIGHT-16, BLACK);
|
||||||
|
|
||||||
|
|
||||||
|
display.setTextWrap(false);
|
||||||
|
display.setTextSize(2);
|
||||||
|
display.setCursor(0, OLED_DATA_START+4);
|
||||||
|
display.print(largeText);
|
||||||
|
display.setTextSize(1);
|
||||||
|
display.setCursor(0, OLED_DATA_END-SCREEN_PROGRESS_BAR_HEIGHT-10);
|
||||||
|
display.print(statusMessage);
|
||||||
|
|
||||||
|
const int barLength = ((SCREEN_WIDTH-2)*step)/numSteps;
|
||||||
|
display.drawRoundRect(0, SCREEN_HEIGHT-SCREEN_PROGRESS_BAR_HEIGHT, SCREEN_WIDTH, 12, 6, WHITE);
|
||||||
|
display.fillRoundRect(1, SCREEN_HEIGHT-SCREEN_PROGRESS_BAR_HEIGHT+1, barLength, 10, 6, WHITE);
|
||||||
|
display.display();
|
||||||
|
}
|
||||||
|
|
||||||
void oledShowWeight(uint16_t weight) {
|
void oledShowWeight(uint16_t weight) {
|
||||||
// Display Gewicht
|
// Display Gewicht
|
||||||
oledcleardata();
|
oledcleardata();
|
||||||
|
@@ -13,11 +13,13 @@ extern bool wifiOn;
|
|||||||
void setupDisplay();
|
void setupDisplay();
|
||||||
void oledclearline();
|
void oledclearline();
|
||||||
void oledcleardata();
|
void oledcleardata();
|
||||||
int oled_center_h(String text);
|
int oled_center_h(const String &text);
|
||||||
int oled_center_v(String text);
|
int oled_center_v(const String &text);
|
||||||
|
|
||||||
|
void oledShowProgressBar(const uint8_t step, const uint8_t numSteps, const char* largeText, const char* statusMessage);
|
||||||
|
|
||||||
void oledShowWeight(uint16_t weight);
|
void oledShowWeight(uint16_t weight);
|
||||||
void oledShowMessage(String message, uint8_t size = 2);
|
void oledShowMessage(const String &message, uint8_t size = 2);
|
||||||
void oledShowTopRow();
|
void oledShowTopRow();
|
||||||
void oledShowIcon(const char* icon);
|
void oledShowIcon(const char* icon);
|
||||||
|
|
||||||
|
33
src/main.cpp
33
src/main.cpp
@@ -16,6 +16,7 @@
|
|||||||
bool mainTaskWasPaused = 0;
|
bool mainTaskWasPaused = 0;
|
||||||
uint8_t scaleTareCounter = 0;
|
uint8_t scaleTareCounter = 0;
|
||||||
bool touchSensorConnected = false;
|
bool touchSensorConnected = false;
|
||||||
|
bool booting = true;
|
||||||
|
|
||||||
// ##### SETUP #####
|
// ##### SETUP #####
|
||||||
void setup() {
|
void setup() {
|
||||||
@@ -63,6 +64,7 @@ void setup() {
|
|||||||
bool panic = true; // Wenn true, löst ein WDT-Timeout einen System-Panik aus
|
bool panic = true; // Wenn true, löst ein WDT-Timeout einen System-Panik aus
|
||||||
esp_task_wdt_init(10, panic);
|
esp_task_wdt_init(10, panic);
|
||||||
|
|
||||||
|
booting = false;
|
||||||
// Aktuellen Task (loopTask) zum Watchdog hinzufügen
|
// Aktuellen Task (loopTask) zum Watchdog hinzufügen
|
||||||
esp_task_wdt_add(NULL);
|
esp_task_wdt_add(NULL);
|
||||||
}
|
}
|
||||||
@@ -118,6 +120,12 @@ void loop() {
|
|||||||
checkWiFiConnection();
|
checkWiFiConnection();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Periodic display update
|
||||||
|
if (intervalElapsed(currentMillis, lastWifiCheckTime, 1000))
|
||||||
|
{
|
||||||
|
oledShowTopRow();
|
||||||
|
}
|
||||||
|
|
||||||
// Wenn Bambu auto set Spool aktiv
|
// Wenn Bambu auto set Spool aktiv
|
||||||
if (bambuCredentials.autosend_enable && autoSetToBambuSpoolId > 0)
|
if (bambuCredentials.autosend_enable && autoSetToBambuSpoolId > 0)
|
||||||
{
|
{
|
||||||
@@ -212,19 +220,14 @@ void loop() {
|
|||||||
lastWeight = weight;
|
lastWeight = weight;
|
||||||
|
|
||||||
// Wenn ein Tag mit SM id erkannte wurde und der Waage Counter anspricht an SM Senden
|
// Wenn ein Tag mit SM id erkannte wurde und der Waage Counter anspricht an SM Senden
|
||||||
if (activeSpoolId != "" && weigthCouterToApi > 3 && weightSend == 0 && nfcReaderState == NFC_READ_SUCCESS) {
|
if (activeSpoolId != "" && weigthCouterToApi > 3 && weightSend == 0 && nfcReaderState == NFC_READ_SUCCESS && tagProcessed == false && spoolmanApiState == API_IDLE) {
|
||||||
oledShowIcon("loading");
|
// set the current tag as processed to prevent it beeing processed again
|
||||||
|
tagProcessed = true;
|
||||||
|
|
||||||
if (updateSpoolWeight(activeSpoolId, weight))
|
if (updateSpoolWeight(activeSpoolId, weight))
|
||||||
{
|
{
|
||||||
oledShowIcon("success");
|
|
||||||
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
|
||||||
weightSend = 1;
|
weightSend = 1;
|
||||||
autoSetToBambuSpoolId = activeSpoolId.toInt();
|
|
||||||
|
|
||||||
if (octoEnabled)
|
|
||||||
{
|
|
||||||
updateSpoolOcto(autoSetToBambuSpoolId);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -232,6 +235,16 @@ void loop() {
|
|||||||
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(sendOctoUpdate && spoolmanApiState == API_IDLE){
|
||||||
|
autoSetToBambuSpoolId = activeSpoolId.toInt();
|
||||||
|
|
||||||
|
if(octoEnabled)
|
||||||
|
{
|
||||||
|
updateSpoolOcto(autoSetToBambuSpoolId);
|
||||||
|
}
|
||||||
|
sendOctoUpdate = false;
|
||||||
|
}
|
||||||
|
|
||||||
esp_task_wdt_reset();
|
esp_task_wdt_reset();
|
||||||
}
|
}
|
||||||
|
9
src/main.h
Normal file
9
src/main.h
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
#ifndef MAIN_H
|
||||||
|
#define MAIN_H
|
||||||
|
|
||||||
|
#include <Arduino.h>
|
||||||
|
|
||||||
|
|
||||||
|
extern bool booting;
|
||||||
|
|
||||||
|
#endif
|
184
src/nfc.cpp
184
src/nfc.cpp
@@ -18,8 +18,14 @@ JsonDocument rfidData;
|
|||||||
String activeSpoolId = "";
|
String activeSpoolId = "";
|
||||||
String lastSpoolId = "";
|
String lastSpoolId = "";
|
||||||
String nfcJsonData = "";
|
String nfcJsonData = "";
|
||||||
|
bool tagProcessed = false;
|
||||||
volatile bool pauseBambuMqttTask = false;
|
volatile bool pauseBambuMqttTask = false;
|
||||||
|
|
||||||
|
struct NfcWriteParameterType {
|
||||||
|
bool tagType;
|
||||||
|
char* payload;
|
||||||
|
};
|
||||||
|
|
||||||
volatile nfcReaderStateType nfcReaderState = NFC_IDLE;
|
volatile nfcReaderStateType nfcReaderState = NFC_IDLE;
|
||||||
// 0 = nicht gelesen
|
// 0 = nicht gelesen
|
||||||
// 1 = erfolgreich gelesen
|
// 1 = erfolgreich gelesen
|
||||||
@@ -31,6 +37,11 @@ volatile nfcReaderStateType nfcReaderState = NFC_IDLE;
|
|||||||
// ***** PN532
|
// ***** PN532
|
||||||
|
|
||||||
|
|
||||||
|
// ##### Recycling Fabrik #####
|
||||||
|
bool isRecyclingFabrik(const char* brand) {
|
||||||
|
return strcmp(brand, "Recycling Fabrik") == 0;
|
||||||
|
}
|
||||||
|
|
||||||
// ##### Funktionen für RFID #####
|
// ##### Funktionen für RFID #####
|
||||||
void payloadToJson(uint8_t *data) {
|
void payloadToJson(uint8_t *data) {
|
||||||
const char* startJson = strchr((char*)data, '{');
|
const char* startJson = strchr((char*)data, '{');
|
||||||
@@ -51,7 +62,13 @@ void payloadToJson(uint8_t *data) {
|
|||||||
int min_temp = doc["min_temp"];
|
int min_temp = doc["min_temp"];
|
||||||
int max_temp = doc["max_temp"];
|
int max_temp = doc["max_temp"];
|
||||||
const char* brand = doc["brand"];
|
const char* brand = doc["brand"];
|
||||||
|
|
||||||
|
// Recycling Fabrik
|
||||||
|
if (isRecyclingFabrik(brand)) {
|
||||||
|
// TODO: Implement specific handling for Recycling Fabrik
|
||||||
|
Serial.println("Recycling Fabrik erkannt.");
|
||||||
|
}
|
||||||
|
|
||||||
Serial.println();
|
Serial.println();
|
||||||
Serial.println("-----------------");
|
Serial.println("-----------------");
|
||||||
Serial.println("JSON-Parsed Data:");
|
Serial.println("JSON-Parsed Data:");
|
||||||
@@ -196,6 +213,8 @@ uint8_t ntag2xx_WriteNDEF(const char *payload) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool decodeNdefAndReturnJson(const byte* encodedMessage) {
|
bool decodeNdefAndReturnJson(const byte* encodedMessage) {
|
||||||
|
oledShowProgressBar(1, octoEnabled?5:4, "Reading", "Decoding data");
|
||||||
|
|
||||||
byte typeLength = encodedMessage[3];
|
byte typeLength = encodedMessage[3];
|
||||||
byte payloadLength = encodedMessage[4];
|
byte payloadLength = encodedMessage[4];
|
||||||
|
|
||||||
@@ -207,7 +226,7 @@ bool decodeNdefAndReturnJson(const byte* encodedMessage) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// JSON-Dokument verarbeiten
|
// JSON-Dokument verarbeiten
|
||||||
JsonDocument doc; // Passen Sie die Größe an den JSON-Inhalt an
|
JsonDocument doc;
|
||||||
DeserializationError error = deserializeJson(doc, nfcJsonData);
|
DeserializationError error = deserializeJson(doc, nfcJsonData);
|
||||||
if (error)
|
if (error)
|
||||||
{
|
{
|
||||||
@@ -219,35 +238,48 @@ bool decodeNdefAndReturnJson(const byte* encodedMessage) {
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Sende die aktualisierten AMS-Daten an alle WebSocket-Clients
|
// If spoolman is unavailable, there is no point in continuing
|
||||||
Serial.println("JSON-Dokument erfolgreich verarbeitet");
|
if(spoolmanConnected){
|
||||||
Serial.println(doc.as<String>());
|
// Sende die aktualisierten AMS-Daten an alle WebSocket-Clients
|
||||||
if (doc["sm_id"].is<String>() && doc["sm_id"] != "")
|
Serial.println("JSON-Dokument erfolgreich verarbeitet");
|
||||||
{
|
Serial.println(doc.as<String>());
|
||||||
Serial.println("SPOOL-ID gefunden: " + doc["sm_id"].as<String>());
|
if (doc["sm_id"].is<String>() && doc["sm_id"] != "" && doc["sm_id"] != "0")
|
||||||
activeSpoolId = doc["sm_id"].as<String>();
|
|
||||||
lastSpoolId = activeSpoolId;
|
|
||||||
}
|
|
||||||
else if(doc["location"].is<String>() && doc["location"] != "")
|
|
||||||
{
|
|
||||||
Serial.println("Location Tag found!");
|
|
||||||
String location = doc["location"].as<String>();
|
|
||||||
if(lastSpoolId != ""){
|
|
||||||
updateSpoolLocation(lastSpoolId, location);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
{
|
||||||
Serial.println("Location update tag scanned without scanning spool before!");
|
oledShowProgressBar(2, octoEnabled?5:4, "Spool Tag", "Weighing");
|
||||||
oledShowMessage("No spool scanned before!");
|
Serial.println("SPOOL-ID gefunden: " + doc["sm_id"].as<String>());
|
||||||
}
|
activeSpoolId = doc["sm_id"].as<String>();
|
||||||
|
lastSpoolId = activeSpoolId;
|
||||||
|
|
||||||
}
|
Serial.println("Api state: " + String(spoolmanApiState));
|
||||||
else
|
}
|
||||||
{
|
else if(doc["location"].is<String>() && doc["location"] != "")
|
||||||
Serial.println("Keine SPOOL-ID gefunden.");
|
{
|
||||||
activeSpoolId = "";
|
Serial.println("Location Tag found!");
|
||||||
oledShowMessage("Unknown Spool");
|
String location = doc["location"].as<String>();
|
||||||
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
if(lastSpoolId != ""){
|
||||||
|
updateSpoolLocation(lastSpoolId, location);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Serial.println("Location update tag scanned without scanning spool before!");
|
||||||
|
oledShowProgressBar(1, 1, "Failure", "Scan spool first");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Recycling Fabrik
|
||||||
|
else if (isRecyclingFabrik(doc["type"].as<String>().c_str())) {
|
||||||
|
// If no sm_id is present but the brand is Recycling Fabrik then
|
||||||
|
// create a new spool, maybe brand too, in Spoolman
|
||||||
|
Serial.println("Recycling Fabrik Tag found!");
|
||||||
|
createFilamentFabrik(doc);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Serial.println("Keine SPOOL-ID gefunden.");
|
||||||
|
activeSpoolId = "";
|
||||||
|
oledShowProgressBar(1, 1, "Failure", "Unkown tag");
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
oledShowProgressBar(octoEnabled?5:4, octoEnabled?5:4, "Failure!", "Spoolman unavailable");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -257,11 +289,11 @@ bool decodeNdefAndReturnJson(const byte* encodedMessage) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void writeJsonToTag(void *parameter) {
|
void writeJsonToTag(void *parameter) {
|
||||||
const char* payload = (const char*)parameter;
|
NfcWriteParameterType* params = (NfcWriteParameterType*)parameter;
|
||||||
|
|
||||||
// Gib die erstellte NDEF-Message aus
|
// Gib die erstellte NDEF-Message aus
|
||||||
Serial.println("Erstelle NDEF-Message...");
|
Serial.println("Erstelle NDEF-Message...");
|
||||||
Serial.println(payload);
|
Serial.println(params->payload);
|
||||||
|
|
||||||
nfcReaderState = NFC_WRITING;
|
nfcReaderState = NFC_WRITING;
|
||||||
vTaskSuspend(RfidReaderTask);
|
vTaskSuspend(RfidReaderTask);
|
||||||
@@ -269,19 +301,24 @@ void writeJsonToTag(void *parameter) {
|
|||||||
|
|
||||||
//pauseBambuMqttTask = true;
|
//pauseBambuMqttTask = true;
|
||||||
// aktualisieren der Website wenn sich der Status ändert
|
// aktualisieren der Website wenn sich der Status ändert
|
||||||
sendNfcData(nullptr);
|
sendNfcData();
|
||||||
vTaskDelay(100 / portTICK_PERIOD_MS);
|
vTaskDelay(100 / portTICK_PERIOD_MS);
|
||||||
oledShowMessage("Waiting for NFC-Tag");
|
Serial.println("CP 1");
|
||||||
|
|
||||||
// Wait 10sec for tag
|
// Wait 10sec for tag
|
||||||
uint8_t success = 0;
|
uint8_t success = 0;
|
||||||
String uidString = "";
|
String uidString = "";
|
||||||
for (uint16_t i = 0; i < 20; i++) {
|
for (uint16_t i = 0; i < 20; i++) {
|
||||||
|
Serial.println("CP 2");
|
||||||
uint8_t uid[] = { 0, 0, 0, 0, 0, 0, 0 }; // Buffer to store the returned UID
|
uint8_t uid[] = { 0, 0, 0, 0, 0, 0, 0 }; // Buffer to store the returned UID
|
||||||
uint8_t uidLength;
|
uint8_t uidLength;
|
||||||
success = nfc.readPassiveTargetID(PN532_MIFARE_ISO14443A, uid, &uidLength, 500);
|
// yield before potentially waiting for 400ms
|
||||||
|
yield();
|
||||||
|
esp_task_wdt_reset();
|
||||||
|
success = nfc.readPassiveTargetID(PN532_MIFARE_ISO14443A, uid, &uidLength, 400);
|
||||||
if (success) {
|
if (success) {
|
||||||
|
Serial.println("CP 3.1");
|
||||||
for (uint8_t i = 0; i < uidLength; i++) {
|
for (uint8_t i = 0; i < uidLength; i++) {
|
||||||
|
//TBD: Rework to remove all the string operations
|
||||||
uidString += String(uid[i], HEX);
|
uidString += String(uid[i], HEX);
|
||||||
if (i < uidLength - 1) {
|
if (i < uidLength - 1) {
|
||||||
uidString += ":"; // Optional: Trennzeichen hinzufügen
|
uidString += ":"; // Optional: Trennzeichen hinzufügen
|
||||||
@@ -289,10 +326,10 @@ void writeJsonToTag(void *parameter) {
|
|||||||
}
|
}
|
||||||
foundNfcTag(nullptr, success);
|
foundNfcTag(nullptr, success);
|
||||||
break;
|
break;
|
||||||
|
}else{
|
||||||
|
Serial.println("CP 3.2");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (i == 0) oledShowMessage("Waiting for NFC-Tag");
|
|
||||||
|
|
||||||
yield();
|
yield();
|
||||||
esp_task_wdt_reset();
|
esp_task_wdt_reset();
|
||||||
vTaskDelay(pdMS_TO_TICKS(1));
|
vTaskDelay(pdMS_TO_TICKS(1));
|
||||||
@@ -300,29 +337,37 @@ void writeJsonToTag(void *parameter) {
|
|||||||
|
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
oledShowIcon("transfer");
|
oledShowProgressBar(1, 3, "Write Tag", "Writing");
|
||||||
|
|
||||||
// Schreibe die NDEF-Message auf den Tag
|
// Schreibe die NDEF-Message auf den Tag
|
||||||
success = ntag2xx_WriteNDEF(payload);
|
success = ntag2xx_WriteNDEF(params->payload);
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
Serial.println("NDEF-Message erfolgreich auf den Tag geschrieben");
|
Serial.println("NDEF-Message erfolgreich auf den Tag geschrieben");
|
||||||
//oledShowMessage("NFC-Tag written");
|
//oledShowMessage("NFC-Tag written");
|
||||||
oledShowIcon("success");
|
//vTaskDelay(1000 / portTICK_PERIOD_MS);
|
||||||
vTaskDelay(1000 / portTICK_PERIOD_MS);
|
|
||||||
nfcReaderState = NFC_WRITE_SUCCESS;
|
nfcReaderState = NFC_WRITE_SUCCESS;
|
||||||
// aktualisieren der Website wenn sich der Status ändert
|
// aktualisieren der Website wenn sich der Status ändert
|
||||||
sendNfcData(nullptr);
|
sendNfcData();
|
||||||
pauseBambuMqttTask = false;
|
pauseBambuMqttTask = false;
|
||||||
|
|
||||||
if (updateSpoolTagId(uidString, payload)) {
|
if(params->tagType){
|
||||||
uint8_t uid[] = { 0, 0, 0, 0, 0, 0, 0 }; // Buffer to store the returned UID
|
// TBD: should this be simplified?
|
||||||
uint8_t uidLength;
|
if (updateSpoolTagId(uidString, params->payload) && params->tagType) {
|
||||||
oledShowIcon("success");
|
|
||||||
while (nfc.readPassiveTargetID(PN532_MIFARE_ISO14443A, uid, &uidLength, 500)) {
|
}else{
|
||||||
yield();
|
// Potentially handle errors
|
||||||
}
|
}
|
||||||
|
}else{
|
||||||
|
oledShowProgressBar(1, 1, "Write Tag", "Done!");
|
||||||
}
|
}
|
||||||
|
uint8_t uid[] = { 0, 0, 0, 0, 0, 0, 0 }; // Buffer to store the returned UID
|
||||||
|
uint8_t uidLength;
|
||||||
|
yield();
|
||||||
|
esp_task_wdt_reset();
|
||||||
|
while (nfc.readPassiveTargetID(PN532_MIFARE_ISO14443A, uid, &uidLength, 400)) {
|
||||||
|
yield();
|
||||||
|
}
|
||||||
vTaskResume(RfidReaderTask);
|
vTaskResume(RfidReaderTask);
|
||||||
vTaskDelay(500 / portTICK_PERIOD_MS);
|
vTaskDelay(500 / portTICK_PERIOD_MS);
|
||||||
}
|
}
|
||||||
@@ -337,13 +382,13 @@ void writeJsonToTag(void *parameter) {
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
Serial.println("Fehler: Kein Tag zu schreiben gefunden.");
|
Serial.println("Fehler: Kein Tag zu schreiben gefunden.");
|
||||||
oledShowMessage("No NFC-Tag found");
|
oledShowProgressBar(1, 1, "Failure!", "No tag found");
|
||||||
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
||||||
nfcReaderState = NFC_IDLE;
|
nfcReaderState = NFC_IDLE;
|
||||||
}
|
}
|
||||||
|
|
||||||
sendWriteResult(nullptr, success);
|
sendWriteResult(nullptr, success);
|
||||||
sendNfcData(nullptr);
|
sendNfcData();
|
||||||
|
|
||||||
vTaskResume(RfidReaderTask);
|
vTaskResume(RfidReaderTask);
|
||||||
pauseBambuMqttTask = false;
|
pauseBambuMqttTask = false;
|
||||||
@@ -351,20 +396,26 @@ void writeJsonToTag(void *parameter) {
|
|||||||
vTaskDelete(NULL);
|
vTaskDelete(NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
void startWriteJsonToTag(const char* payload) {
|
void startWriteJsonToTag(const bool isSpoolTag, const char* payload) {
|
||||||
char* payloadCopy = strdup(payload);
|
NfcWriteParameterType* parameters = new NfcWriteParameterType();
|
||||||
|
parameters->tagType = isSpoolTag;
|
||||||
|
parameters->payload = strdup(payload);
|
||||||
|
|
||||||
// Task nicht mehrfach starten
|
// Task nicht mehrfach starten
|
||||||
if (nfcReaderState != NFC_WRITING) {
|
if (nfcReaderState == NFC_IDLE) {
|
||||||
|
oledShowProgressBar(0, 1, "Write Tag", "Place tag now");
|
||||||
// Erstelle die Task
|
// Erstelle die Task
|
||||||
xTaskCreate(
|
xTaskCreate(
|
||||||
writeJsonToTag, // Task-Funktion
|
writeJsonToTag, // Task-Funktion
|
||||||
"WriteJsonToTagTask", // Task-Name
|
"WriteJsonToTagTask", // Task-Name
|
||||||
5115, // Stackgröße in Bytes
|
5115, // Stackgröße in Bytes
|
||||||
(void*)payloadCopy, // Parameter
|
(void*)parameters, // Parameter
|
||||||
rfidWriteTaskPrio, // Priorität
|
rfidWriteTaskPrio, // Priorität
|
||||||
NULL // Task-Handle (nicht benötigt)
|
NULL // Task-Handle (nicht benötigt)
|
||||||
);
|
);
|
||||||
|
}else{
|
||||||
|
oledShowProgressBar(0, 1, "FAILURE", "NFC busy!");
|
||||||
|
// TBD: Add proper error handling (website)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -384,14 +435,19 @@ void scanRfidTask(void * parameter) {
|
|||||||
|
|
||||||
foundNfcTag(nullptr, success);
|
foundNfcTag(nullptr, success);
|
||||||
|
|
||||||
if (success && nfcReaderState != NFC_READ_SUCCESS)
|
// As long as there is still a tag on the reader, do not try to read it again
|
||||||
|
if (success && nfcReaderState == NFC_IDLE)
|
||||||
{
|
{
|
||||||
|
// Set the current tag as not processed
|
||||||
|
tagProcessed = false;
|
||||||
|
|
||||||
// Display some basic information about the card
|
// Display some basic information about the card
|
||||||
Serial.println("Found an ISO14443A card");
|
Serial.println("Found an ISO14443A card");
|
||||||
|
|
||||||
nfcReaderState = NFC_READING;
|
nfcReaderState = NFC_READING;
|
||||||
|
|
||||||
oledShowIcon("transfer");
|
oledShowProgressBar(0, octoEnabled?5:4, "Reading", "Detecting tag");
|
||||||
|
|
||||||
vTaskDelay(500 / portTICK_PERIOD_MS);
|
vTaskDelay(500 / portTICK_PERIOD_MS);
|
||||||
|
|
||||||
if (uidLength == 7)
|
if (uidLength == 7)
|
||||||
@@ -425,8 +481,7 @@ void scanRfidTask(void * parameter) {
|
|||||||
|
|
||||||
if (!decodeNdefAndReturnJson(data))
|
if (!decodeNdefAndReturnJson(data))
|
||||||
{
|
{
|
||||||
oledShowMessage("NFC-Tag unknown");
|
oledShowProgressBar(1, 1, "Failure", "Unknown tag");
|
||||||
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
|
||||||
nfcReaderState = NFC_READ_ERROR;
|
nfcReaderState = NFC_READ_ERROR;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -438,12 +493,14 @@ void scanRfidTask(void * parameter) {
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
oledShowMessage("NFC-Tag read error");
|
oledShowProgressBar(1, 1, "Failure", "Tag read error");
|
||||||
nfcReaderState = NFC_READ_ERROR;
|
nfcReaderState = NFC_READ_ERROR;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
//TBD: Show error here?!
|
||||||
|
oledShowProgressBar(1, 1, "Failure", "Unkown tag type");
|
||||||
Serial.println("This doesn't seem to be an NTAG2xx tag (UUID length != 7 bytes)!");
|
Serial.println("This doesn't seem to be an NTAG2xx tag (UUID length != 7 bytes)!");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -459,22 +516,21 @@ void scanRfidTask(void * parameter) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// aktualisieren der Website wenn sich der Status ändert
|
// aktualisieren der Website wenn sich der Status ändert
|
||||||
sendNfcData(nullptr);
|
sendNfcData();
|
||||||
}
|
}
|
||||||
yield();
|
yield();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void startNfc() {
|
void startNfc() {
|
||||||
|
oledShowProgressBar(5, 7, DISPLAY_BOOT_TEXT, "NFC init");
|
||||||
nfc.begin(); // Beginne Kommunikation mit RFID Leser
|
nfc.begin(); // Beginne Kommunikation mit RFID Leser
|
||||||
delay(1000);
|
delay(1000);
|
||||||
unsigned long versiondata = nfc.getFirmwareVersion(); // Lese Versionsnummer der Firmware aus
|
unsigned long versiondata = nfc.getFirmwareVersion(); // Lese Versionsnummer der Firmware aus
|
||||||
if (! versiondata) { // Wenn keine Antwort kommt
|
if (! versiondata) { // Wenn keine Antwort kommt
|
||||||
Serial.println("Kann kein RFID Board finden !"); // Sende Text "Kann kein..." an seriellen Monitor
|
Serial.println("Kann kein RFID Board finden !"); // Sende Text "Kann kein..." an seriellen Monitor
|
||||||
//delay(5000);
|
|
||||||
//ESP.restart();
|
|
||||||
oledShowMessage("No RFID Board found");
|
oledShowMessage("No RFID Board found");
|
||||||
delay(2000);
|
vTaskDelay(2000 / portTICK_PERIOD_MS);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
Serial.print("Chip PN5 gefunden"); Serial.println((versiondata >> 24) & 0xFF, HEX); // Sende Text und Versionsinfos an seriellen
|
Serial.print("Chip PN5 gefunden"); Serial.println((versiondata >> 24) & 0xFF, HEX); // Sende Text und Versionsinfos an seriellen
|
||||||
|
@@ -15,7 +15,7 @@ typedef enum{
|
|||||||
|
|
||||||
void startNfc();
|
void startNfc();
|
||||||
void scanRfidTask(void * parameter);
|
void scanRfidTask(void * parameter);
|
||||||
void startWriteJsonToTag(const char* payload);
|
void startWriteJsonToTag(const bool isSpoolTag, const char* payload);
|
||||||
|
|
||||||
extern TaskHandle_t RfidReaderTask;
|
extern TaskHandle_t RfidReaderTask;
|
||||||
extern String nfcJsonData;
|
extern String nfcJsonData;
|
||||||
@@ -23,6 +23,7 @@ extern String activeSpoolId;
|
|||||||
extern String lastSpoolId;
|
extern String lastSpoolId;
|
||||||
extern volatile nfcReaderStateType nfcReaderState;
|
extern volatile nfcReaderStateType nfcReaderState;
|
||||||
extern volatile bool pauseBambuMqttTask;
|
extern volatile bool pauseBambuMqttTask;
|
||||||
|
extern bool tagProcessed;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@@ -224,7 +224,7 @@ void handleUpdate(AsyncWebServer &server) {
|
|||||||
static int lastProgress = -1;
|
static int lastProgress = -1;
|
||||||
if (currentProgress != lastProgress && (currentProgress % 10 == 0 || final)) {
|
if (currentProgress != lastProgress && (currentProgress % 10 == 0 || final)) {
|
||||||
sendUpdateProgress(currentProgress, "uploading");
|
sendUpdateProgress(currentProgress, "uploading");
|
||||||
oledShowMessage("Update: " + String(currentProgress) + "%");
|
oledShowProgressBar(currentProgress, 100, "Update", "Download");
|
||||||
vTaskDelay(50 / portTICK_PERIOD_MS);
|
vTaskDelay(50 / portTICK_PERIOD_MS);
|
||||||
lastProgress = currentProgress;
|
lastProgress = currentProgress;
|
||||||
}
|
}
|
||||||
|
@@ -115,7 +115,7 @@ void start_scale(bool touchSensorConnected) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
oledShowMessage("Scale Tare Please remove all");
|
oledShowProgressBar(6, 7, DISPLAY_BOOT_TEXT, "Tare scale");
|
||||||
for (uint16_t i = 0; i < 2000; i++) {
|
for (uint16_t i = 0; i < 2000; i++) {
|
||||||
yield();
|
yield();
|
||||||
vTaskDelay(pdMS_TO_TICKS(1));
|
vTaskDelay(pdMS_TO_TICKS(1));
|
||||||
@@ -162,7 +162,7 @@ uint8_t calibrate_scale() {
|
|||||||
{
|
{
|
||||||
|
|
||||||
scale.set_scale();
|
scale.set_scale();
|
||||||
oledShowMessage("Step 1 empty Scale");
|
oledShowProgressBar(0, 3, "Scale Cal.", "Empty Scale");
|
||||||
|
|
||||||
for (uint16_t i = 0; i < 5000; i++) {
|
for (uint16_t i = 0; i < 5000; i++) {
|
||||||
yield();
|
yield();
|
||||||
@@ -174,7 +174,7 @@ uint8_t calibrate_scale() {
|
|||||||
Serial.println("Tare done...");
|
Serial.println("Tare done...");
|
||||||
Serial.print("Place a known weight on the scale...");
|
Serial.print("Place a known weight on the scale...");
|
||||||
|
|
||||||
oledShowMessage("Step 2 Place the weight");
|
oledShowProgressBar(1, 3, "Scale Cal.", "Place the weight");
|
||||||
|
|
||||||
for (uint16_t i = 0; i < 5000; i++) {
|
for (uint16_t i = 0; i < 5000; i++) {
|
||||||
yield();
|
yield();
|
||||||
@@ -207,9 +207,7 @@ uint8_t calibrate_scale() {
|
|||||||
Serial.print("Verified stored value: ");
|
Serial.print("Verified stored value: ");
|
||||||
Serial.println(verifyValue);
|
Serial.println(verifyValue);
|
||||||
|
|
||||||
Serial.println("End calibration, remove weight");
|
oledShowProgressBar(2, 3, "Scale Cal.", "Remove weight");
|
||||||
|
|
||||||
oledShowMessage("Remove weight");
|
|
||||||
|
|
||||||
scale.set_scale(newCalibrationValue);
|
scale.set_scale(newCalibrationValue);
|
||||||
for (uint16_t i = 0; i < 2000; i++) {
|
for (uint16_t i = 0; i < 2000; i++) {
|
||||||
@@ -218,7 +216,7 @@ uint8_t calibrate_scale() {
|
|||||||
esp_task_wdt_reset();
|
esp_task_wdt_reset();
|
||||||
}
|
}
|
||||||
|
|
||||||
oledShowMessage("Scale calibrated");
|
oledShowProgressBar(3, 3, "Scale Cal.", "Completed");
|
||||||
|
|
||||||
// For some reason it is not possible to re-tare the scale here, it will result in a wdt timeout. Instead let the scale loop do the taring
|
// For some reason it is not possible to re-tare the scale here, it will result in a wdt timeout. Instead let the scale loop do the taring
|
||||||
//scale.tare();
|
//scale.tare();
|
||||||
@@ -232,21 +230,18 @@ uint8_t calibrate_scale() {
|
|||||||
|
|
||||||
returnState = 1;
|
returnState = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
{
|
Serial.println("Calibration value is invalid. Please recalibrate.");
|
||||||
Serial.println("Calibration value is invalid. Please recalibrate.");
|
|
||||||
|
|
||||||
oledShowMessage("Calibration ERROR Try again");
|
oledShowProgressBar(3, 3, "Failure", "Calibration error");
|
||||||
|
|
||||||
for (uint16_t i = 0; i < 50000; i++) {
|
for (uint16_t i = 0; i < 50000; i++) {
|
||||||
yield();
|
yield();
|
||||||
vTaskDelay(pdMS_TO_TICKS(1));
|
vTaskDelay(pdMS_TO_TICKS(1));
|
||||||
esp_task_wdt_reset();
|
esp_task_wdt_reset();
|
||||||
}
|
|
||||||
returnState = 0;
|
|
||||||
}
|
}
|
||||||
|
returnState = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@@ -34,7 +34,7 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp
|
|||||||
Serial.println("Neuer Client verbunden!");
|
Serial.println("Neuer Client verbunden!");
|
||||||
// Sende die AMS-Daten an den neuen Client
|
// Sende die AMS-Daten an den neuen Client
|
||||||
if (!bambuDisabled) sendAmsData(client);
|
if (!bambuDisabled) sendAmsData(client);
|
||||||
sendNfcData(client);
|
sendNfcData();
|
||||||
foundNfcTag(client, 0);
|
foundNfcTag(client, 0);
|
||||||
sendWriteResult(client, 3);
|
sendWriteResult(client, 3);
|
||||||
|
|
||||||
@@ -52,8 +52,6 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp
|
|||||||
JsonDocument doc;
|
JsonDocument doc;
|
||||||
deserializeJson(doc, message);
|
deserializeJson(doc, message);
|
||||||
|
|
||||||
bool spoolmanConnected = (spoolmanApiState != API_INIT);
|
|
||||||
|
|
||||||
if (doc["type"] == "heartbeat") {
|
if (doc["type"] == "heartbeat") {
|
||||||
// Sende Heartbeat-Antwort
|
// Sende Heartbeat-Antwort
|
||||||
ws.text(client->id(), "{"
|
ws.text(client->id(), "{"
|
||||||
@@ -69,7 +67,8 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp
|
|||||||
// Versuche NFC-Daten zu schreiben
|
// Versuche NFC-Daten zu schreiben
|
||||||
String payloadString;
|
String payloadString;
|
||||||
serializeJson(doc["payload"], payloadString);
|
serializeJson(doc["payload"], payloadString);
|
||||||
startWriteJsonToTag(payloadString.c_str());
|
|
||||||
|
startWriteJsonToTag((doc["tagType"] == "spool") ? true : false, payloadString.c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -150,11 +149,11 @@ void sendWriteResult(AsyncWebSocketClient *client, uint8_t success) {
|
|||||||
void foundNfcTag(AsyncWebSocketClient *client, uint8_t success) {
|
void foundNfcTag(AsyncWebSocketClient *client, uint8_t success) {
|
||||||
if (success == lastSuccess) return;
|
if (success == lastSuccess) return;
|
||||||
ws.textAll("{\"type\":\"nfcTag\", \"payload\":{\"found\": " + String(success) + "}}");
|
ws.textAll("{\"type\":\"nfcTag\", \"payload\":{\"found\": " + String(success) + "}}");
|
||||||
sendNfcData(nullptr);
|
sendNfcData();
|
||||||
lastSuccess = success;
|
lastSuccess = success;
|
||||||
}
|
}
|
||||||
|
|
||||||
void sendNfcData(AsyncWebSocketClient *client) {
|
void sendNfcData() {
|
||||||
if (lastnfcReaderState == nfcReaderState) return;
|
if (lastnfcReaderState == nfcReaderState) return;
|
||||||
// TBD: Why is there no status for reading the tag?
|
// TBD: Why is there no status for reading the tag?
|
||||||
switch(nfcReaderState){
|
switch(nfcReaderState){
|
||||||
@@ -189,6 +188,7 @@ void sendAmsData(AsyncWebSocketClient *client) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void setupWebserver(AsyncWebServer &server) {
|
void setupWebserver(AsyncWebServer &server) {
|
||||||
|
oledShowProgressBar(2, 7, DISPLAY_BOOT_TEXT, "Webserver init");
|
||||||
// Deaktiviere alle Debug-Ausgaben
|
// Deaktiviere alle Debug-Ausgaben
|
||||||
Serial.setDebugOutput(false);
|
Serial.setDebugOutput(false);
|
||||||
|
|
||||||
|
@@ -24,7 +24,7 @@ void setupWebserver(AsyncWebServer &server);
|
|||||||
|
|
||||||
// WebSocket-Funktionen
|
// WebSocket-Funktionen
|
||||||
void sendAmsData(AsyncWebSocketClient *client);
|
void sendAmsData(AsyncWebSocketClient *client);
|
||||||
void sendNfcData(AsyncWebSocketClient *client);
|
void sendNfcData();
|
||||||
void foundNfcTag(AsyncWebSocketClient *client, uint8_t success);
|
void foundNfcTag(AsyncWebSocketClient *client, uint8_t success);
|
||||||
void sendWriteResult(AsyncWebSocketClient *client, uint8_t success);
|
void sendWriteResult(AsyncWebSocketClient *client, uint8_t success);
|
||||||
|
|
||||||
|
@@ -61,8 +61,7 @@ void initWiFi() {
|
|||||||
wm.setWiFiAutoReconnect(true);
|
wm.setWiFiAutoReconnect(true);
|
||||||
wm.setConnectTimeout(10);
|
wm.setConnectTimeout(10);
|
||||||
|
|
||||||
oledShowTopRow();
|
oledShowProgressBar(1, 7, DISPLAY_BOOT_TEXT, "WiFi init");
|
||||||
oledShowMessage("WiFi Setup");
|
|
||||||
|
|
||||||
//bool res = wm.autoConnect("FilaMan"); // anonymous ap
|
//bool res = wm.autoConnect("FilaMan"); // anonymous ap
|
||||||
if(!wm.autoConnect("FilaMan")) {
|
if(!wm.autoConnect("FilaMan")) {
|
||||||
@@ -80,9 +79,6 @@ void initWiFi() {
|
|||||||
Serial.println(WiFi.localIP());
|
Serial.println(WiFi.localIP());
|
||||||
|
|
||||||
oledShowTopRow();
|
oledShowTopRow();
|
||||||
display.display();
|
|
||||||
|
|
||||||
vTaskDelay(500 / portTICK_PERIOD_MS);
|
|
||||||
|
|
||||||
// mDNS
|
// mDNS
|
||||||
startMDNS();
|
startMDNS();
|
||||||
|
Reference in New Issue
Block a user