new HW with integrated OLED (Wifi Kit 8) and some fixes
This commit is contained in:
nightflyer88 2020-09-27 12:16:27 +02:00 committed by GitHub
commit 6cc51bb9ef
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 2817 additions and 191 deletions

View File

@ -4,18 +4,32 @@
(c) 2019 by M. Lehmann
------------------------------------------------------------------
*/
#define CGSCALE_VERSION "2.0"
#define CGSCALE_VERSION "2.11"
/*
******************************************************************
history:
V2.11 18.08.20 code is now compatible with standard OLED displays
and original code base (default pw length = 32)
V2.1 18.07.20 added support for ESP8266 based Wifi Kit 8
(by Pulsar07/ (https://heltec.org/project/wifi-kit-8/)
R.Stransky is a ESP8266 with
a build in OLED 128x32
battery connector with charging management
reset and GPIO0 button
support for a tare button (PIN_TARE_BUTTON)
bug fixed: wifi password now with up to 64 chars
bug fixed: wifi data (ssid/passwd) with special
character (e.g. +) is now supported
for specified battery type, voltage is displayed
using uncompressed html files makes WEB GUI much faster
V2.01 29.01.20 small bug fixes with AVR
V2.0 26.01.20 Webpage rewritten, no bootstrap framework needed
add translation to webpage (en, de)
optimized for measuring with landinggears
updated to ArduinoJson V6
firmware update over web interface
V1.2.1 31.03.19 small bug fixed
V1.2.1 31.03.19 small bug fixed
values in model database are rounded
mDNS and OTA did not work in AP mode
V1.2 23.02.19 Add OTA (over the air update)
@ -87,7 +101,7 @@
#endif
// HX711 constructor array (dout pin, sck pint):
HX711_ADC LoadCell[]{HX711_ADC(PIN_LOADCELL1_DOUT, PIN_LOADCELL1_PD_SCK),HX711_ADC(PIN_LOADCELL2_DOUT, PIN_LOADCELL2_PD_SCK),HX711_ADC(PIN_LOADCELL3_DOUT, PIN_LOADCELL3_PD_SCK)};
HX711_ADC LoadCell[] {HX711_ADC(PIN_LOADCELL1_DOUT, PIN_LOADCELL1_PD_SCK), HX711_ADC(PIN_LOADCELL2_DOUT, PIN_LOADCELL2_PD_SCK), HX711_ADC(PIN_LOADCELL3_DOUT, PIN_LOADCELL3_PD_SCK)};
// webserver constructor
#if defined(ESP8266)
@ -142,6 +156,12 @@ bool updateMenu = true;
int menuPage = 0;
String errMsg[5];
int errMsgCnt = 0;
int oledDisplayHeight;
int oledDisplayWidth;
const uint8_t *oledFontLarge;
const uint8_t *oledFontNormal;
const uint8_t *oledFontSmall;
const uint8_t *oledFontTiny;
#if defined(ESP8266)
String updateMsg = "";
bool wifiSTAmode = true;
@ -156,6 +176,200 @@ void(* resetCPU) (void) = 0;
void resetCPU() {}
#endif
void initOLED() {
oledDisplay.begin();
oledDisplayHeight = oledDisplay.getDisplayHeight();
oledDisplayWidth = oledDisplay.getDisplayWidth();
printConsole(T_BOOT, "init OLED display: " + String(oledDisplayWidth) + String("x") + String(oledDisplayHeight));
oledFontLarge = u8g2_font_helvR12_tr;
oledFontNormal = u8g2_font_helvR10_tr;
oledFontSmall = u8g2_font_5x7_tr;
oledFontTiny = u8g2_font_4x6_tr;
if (oledDisplayHeight <= 32) {
oledFontLarge = u8g2_font_helvR10_tr;
oledFontNormal = u8g2_font_6x12_tr;
}
int ylineHeight = oledDisplayHeight / 3;
oledDisplay.setFont(oledFontNormal);
oledDisplay.firstPage();
do {
oledDisplay.setFont(oledFontLarge);
if (oledDisplayHeight <= 32) {
oledDisplay.drawXBMP(5, 0, 18, 18, CGImage);
} else {
oledDisplay.drawXBMP(20, 12, 18, 18, CGImage);
}
oledDisplay.setFont(oledFontLarge);
if (oledDisplayHeight <= 32) {
oledDisplay.setCursor(30, 12);
} else {
oledDisplay.setCursor(45, 28);
}
oledDisplay.print(F("CG scale"));
oledDisplay.setFont(oledFontSmall);
if (oledDisplayHeight <= 32) {
oledDisplay.setCursor(30, 22);
} else {
oledDisplay.setCursor(35, 55);
}
oledDisplay.print(F("Version: "));
oledDisplay.print(CGSCALE_VERSION);
if (oledDisplayHeight <= 32) {
oledDisplay.setCursor(5, 31);
} else {
oledDisplay.setCursor(20, 64);
}
oledDisplay.print(F("(c) 2019 M.Lehmann et al."));
} while ( oledDisplay.nextPage() );
}
void printOLED(String aLine1, String aLine2, String aLine3 = String(""));
void printOLED(String aLine1, String aLine2, String aLine3) {
int ylineHeight = oledDisplayHeight / 3;
oledDisplay.firstPage();
do {
oledDisplay.setFont(oledFontNormal);
oledDisplay.setCursor(0, ylineHeight * 1);
oledDisplay.print(aLine1);
oledDisplay.setCursor(0, ylineHeight * 2);
oledDisplay.print(aLine2);
if (aLine3 == "") {
oledDisplay.drawLine(0, ylineHeight * 2 + 2, oledDisplayWidth, ylineHeight * 2 + 2);
oledDisplay.setFont(oledFontTiny);
oledDisplay.setCursor(0, oledDisplayHeight);
oledDisplay.print("IP:" + WiFi.localIP().toString());
String signature = "CG scale: V" + String(CGSCALE_VERSION);
oledDisplay.setCursor(oledDisplayWidth - oledDisplay.getStrWidth(signature.c_str()), oledDisplayHeight);
oledDisplay.print(signature);
} else {
oledDisplay.setCursor(0, oledDisplayHeight);
oledDisplay.print(aLine3);
}
} while ( oledDisplay.nextPage() );
}
void printScaleOLED() {
// print to display
char buff1[8];
char buff[12];
char buff2[8];
int pos_weightTotal = 7;
int pos_CG_length = 28;
if (nLoadcells == 2) {
pos_weightTotal = 17;
pos_CG_length = 45;
if (batType == 0) {
pos_weightTotal = 12;
pos_CG_length = 40;
}
}
oledDisplay.firstPage();
do {
if (errMsgCnt == 0) {
// print battery
if (batType > B_OFF) {
oledDisplay.drawXBMP(48, 1, 12, 6, batteryImage);
float percentVolt = percentBat(batVolt / batCells);
dtostrf(percentVolt, 3, 0, buff);
oledDisplay.drawBox(49, 2, (percentVolt / (100 / 8)), 4);
oledDisplay.setFont(oledFontSmall);
oledDisplay.setCursor(78 - oledDisplay.getStrWidth(buff), 7);
if (batType > B_VOLT) {
dtostrf(percentVolt, 3, 0, buff);
oledDisplay.print(buff);
oledDisplay.print(F("%/"));
}
dtostrf(batVolt, 2, 2, buff);
oledDisplay.print(buff);
oledDisplay.print(F("V"));
}
// print total weight
oledDisplay.setFont(oledFontNormal);
dtostrf(weightTotal, 7, 1, buff);
if (oledDisplayHeight <= 32) {
oledDisplay.setCursor(1, 18);
oledDisplay.print(F("M = "));
} else {
oledDisplay.drawXBMP(2, pos_weightTotal, 18, 18, weightImage);
oledDisplay.setCursor(93 - oledDisplay.getStrWidth(buff), pos_weightTotal + 17);
}
oledDisplay.print(buff);
oledDisplay.print(F(" g"));
// print CG longitudinal axis
dtostrf(CG_length, 7, 1, buff);
if (oledDisplayHeight <= 32) {
oledDisplay.setCursor(1, 32);
oledDisplay.print(F("CG = "));
} else {
oledDisplay.drawXBMP(2, pos_CG_length, 18, 18, CGImage);
oledDisplay.setCursor(93 - oledDisplay.getStrWidth(buff), pos_CG_length + 16);
}
oledDisplay.print(buff);
oledDisplay.print(F(" mm"));
// print CG transverse axis
if (nLoadcells == 3) {
if (oledDisplayHeight <= 32) {
oledDisplay.setCursor(78, 32);
oledDisplay.print(F("LR="));
dtostrf(CG_trans, 3, 0, buff);
} else {
oledDisplay.drawXBMP(2, 47, 18, 18, CGtransImage);
oledDisplay.setCursor(93 - oledDisplay.getStrWidth(buff), 64);
dtostrf(CG_trans, 7, 1, buff);
}
oledDisplay.print(buff);
oledDisplay.print(F(" mm"));
}
} else {
oledDisplay.setFont(oledFontSmall);
for (int i = 1; i <= errMsgCnt; i++) {
oledDisplay.setCursor(0, 7 * i);
oledDisplay.print(errMsg[i]);
}
}
} while ( oledDisplay.nextPage() );
}
#ifdef PIN_TARE_BUTTON
void handleTareBtn() {
static unsigned long lastTaraBtn = 0;
if ((millis() - lastTaraBtn) > 20) {
lastTaraBtn = millis();
static int tareBtnCnt = 0;
if (digitalRead(PIN_TARE_BUTTON)) {
tareBtnCnt = 0;
} else {
tareBtnCnt++;
if (tareBtnCnt > 10) {
Serial.println("tare button pressed");
printOLED("TARE ==>", " tare load cells ...");
// avoid keybounce
tareBtnCnt = -1000;
tareLoadcells();
delay(2000);
}
}
}
}
#endif
// save calibration factor
void saveCalFactor(int nLC) {
@ -167,7 +381,7 @@ void saveCalFactor(int nLC) {
}
void updateLoadcells(){
void updateLoadcells() {
for (int i = LC1; i <= LC3; i++) {
if (i < nLoadcells) {
LoadCell[i].update();
@ -176,7 +390,7 @@ void updateLoadcells(){
}
void tareLoadcells(){
void tareLoadcells() {
for (int i = LC1; i <= LC3; i++) {
if (i < nLoadcells) {
LoadCell[i].tare();
@ -185,7 +399,7 @@ void tareLoadcells(){
}
void printNewValueText(){
void printNewValueText() {
Serial.print(F("Set new value:"));
}
@ -210,7 +424,7 @@ bool runAutoCalibrate() {
calFactorLoadcell[i] = calFactorLoadcell[i] / (toWeightLoadCell[i] / weightLoadCell[i]);
saveCalFactor(i);
}
// finish
Serial.println(F("done"));
}
@ -250,7 +464,7 @@ int percentBat(float cellVoltage) {
break;
}
}
float cellempty = pgm_read_float( &percentList[batTypeArray][0][0]);
float cellfull = pgm_read_float( &percentList[batTypeArray][elementCount][0]);
@ -262,9 +476,9 @@ int percentBat(float cellVoltage) {
for (int i = 0; i <= elementCount; i++) {
float curVolt = pgm_read_float(&percentList[batTypeArray][i][0]);
if (curVolt >= cellVoltage && i > 0) {
float lastVolt = pgm_read_float(&percentList[batTypeArray][i-1][0]);
float lastVolt = pgm_read_float(&percentList[batTypeArray][i - 1][0]);
float curPercent = pgm_read_float(&percentList[batTypeArray][i][1]);
float lastPercent = pgm_read_float(&percentList[batTypeArray][i-1][1]);
float lastPercent = pgm_read_float(&percentList[batTypeArray][i - 1][1]);
result = float((cellVoltage - lastVolt) / (curVolt - lastVolt)) * (curPercent - lastPercent) + lastPercent;
break;
}
@ -279,12 +493,13 @@ int percentBat(float cellVoltage) {
void setup() {
// init serial
Serial.begin(9600);
Serial.begin(115200);
Serial.println();
delay(1000);
#if defined(ESP8266)
printConsole(T_BOOT, "startup CG scale V" + String(CGSCALE_VERSION));
// init filesystem
SPIFFS.begin();
EEPROM.begin(EEPROM_SIZE);
@ -367,24 +582,7 @@ void setup() {
#endif
// init OLED display
#if defined(ESP8266)
printConsole(T_BOOT, "init OLED display");
#endif
oledDisplay.begin();
oledDisplay.firstPage();
do {
oledDisplay.drawXBMP(20, 12, 18, 18, CGImage);
oledDisplay.setFont(u8g2_font_helvR12_tr);
oledDisplay.setCursor(45, 28);
oledDisplay.print(F("CG scale"));
oledDisplay.setFont(u8g2_font_5x7_tr);
oledDisplay.setCursor(35, 55);
oledDisplay.print(F("Version: "));
oledDisplay.print(CGSCALE_VERSION);
oledDisplay.setCursor(20, 64);
oledDisplay.print(F("(c) 2019 M. Lehmann"));
} while ( oledDisplay.nextPage() );
initOLED();
// init & tare Loadcells
for (int i = LC1; i <= LC3; i++) {
@ -392,7 +590,7 @@ void setup() {
LoadCell[i].begin();
LoadCell[i].setCalFactor(calFactorLoadcell[i]);
#if defined(ESP8266)
printConsole(T_BOOT, "init Loadcell " + String(i+1));
printConsole(T_BOOT, "init Loadcell " + String(i + 1));
#endif
}
}
@ -408,28 +606,32 @@ void setup() {
#if defined(ESP8266)
printConsole(T_BOOT, "Wifi: STA mode - connecing with: " + String(ssid_STA));
// Start by connecting to a WiFi network
WiFi.persistent(false);
WiFi.mode(WIFI_STA);
WiFi.begin(ssid_STA, password_STA);
printConsole(T_BOOT, "Wifi: STA mode - connect with: " + String(ssid_STA));
long timeoutWiFi = millis();
while (WiFi.status() != WL_CONNECTED) {
delay(500);
Serial.print(".");
if (WiFi.status() == WL_NO_SSID_AVAIL) {
printConsole(T_ERROR, "Wifi: No SSID available");
printConsole(T_ERROR, "\nWifi: No SSID available");
break;
} else if (WiFi.status() == WL_CONNECT_FAILED) {
printConsole(T_ERROR, "Wifi: Connection failed");
printConsole(T_ERROR, "\nWifi: Connection failed");
break;
} else if ((millis() - timeoutWiFi) > TIMEOUT_CONNECT) {
printConsole(T_ERROR, "Wifi: Timeout");
printConsole(T_ERROR, "\nWifi: Timeout");
break;
}
}
if (WiFi.status() != WL_CONNECTED) {
// if WiFi not connected, switch to access point mode
wifiSTAmode = false;
@ -442,7 +644,7 @@ void setup() {
printConsole(T_RUN, "Wifi: Connected, IP: " + String(WiFi.localIP().toString()));
}
// Set Hostname
String hostname = "disabled";
#if ENABLE_MDNS
@ -452,40 +654,22 @@ void setup() {
if (!MDNS.begin(hostname, WiFi.localIP())) {
hostname = "mDNS failed";
printConsole(T_ERROR, "Wifi: " + hostname);
}else{
} else {
hostname += ".local";
printConsole(T_RUN, "Wifi: " + hostname);
}
#endif
if (wifiSTAmode) {
printOLED("WiFi: " + String(ssid_STA),
"Host: " + String(hostname),
"IP : " + WiFi.localIP().toString());
} else {
printOLED("WiFi: " + String(ssid_AP),
"Host: " + String(hostname),
"IP : " + WiFi.softAPIP().toString());
}
// print wifi status
oledDisplay.firstPage();
do {
oledDisplay.setFont(u8g2_font_5x7_tr);
oledDisplay.setCursor(0, 14);
oledDisplay.print(F("WiFi:"));
oledDisplay.setCursor(0, 39);
oledDisplay.print(F("Host:"));
oledDisplay.setCursor(0, 64);
oledDisplay.print(F("IP:"));
oledDisplay.setFont(u8g2_font_helvR10_tr);
oledDisplay.setCursor(28, 14);
if (wifiSTAmode) {
oledDisplay.print(ssid_STA);
} else {
oledDisplay.print(ssid_AP);
}
oledDisplay.setCursor(28, 39);
oledDisplay.print(hostname);
oledDisplay.setCursor(28, 64);
if (wifiSTAmode) {
oledDisplay.print(WiFi.localIP());
} else {
oledDisplay.print(WiFi.softAPIP());
}
} while ( oledDisplay.nextPage() );
delay(3000);
// When the client requests data
@ -520,10 +704,10 @@ void setup() {
printConsole(T_RUN, "Webserver is up and running");
// init OTA (over the air update)
if(enableOTA){
if (enableOTA) {
ArduinoOTA.setHostname(ssid_AP);
ArduinoOTA.setPassword(password_AP);
ArduinoOTA.onStart([]() {
String type;
if (ArduinoOTA.getCommand() == U_FLASH) {
@ -535,16 +719,16 @@ void setup() {
updateMsg = "Updating " + type;
printConsole(T_UPDATE, type);
});
ArduinoOTA.onEnd([]() {
updateMsg = "successful..";
printUpdateProgress(100, 100);
});
ArduinoOTA.onProgress([](unsigned int progress, unsigned int total) {
printUpdateProgress(progress, total);
});
ArduinoOTA.onError([](ota_error_t error) {
if (error == OTA_AUTH_ERROR) {
updateMsg = "Auth Failed";
@ -563,14 +747,14 @@ void setup() {
ArduinoOTA.begin();
printConsole(T_RUN, "OTA is up and running");
}
// https update
// https update
httpsClient.setInsecure();
if(enableUpdate){
if (enableUpdate) {
// check for update
httpsUpdate(PROBE_UPDATE);
}
#endif
}
@ -584,12 +768,16 @@ void loop() {
MDNS.update();
#endif
if(enableOTA){
if (enableOTA) {
ArduinoOTA.handle();
}
server.handleClient();
#endif
#ifdef PIN_TARE_BUTTON
handleTareBtn();
#endif
updateLoadcells();
// update loadcell values
@ -607,6 +795,7 @@ void loop() {
}
}
// update display and serial menu
if ((millis() - lastTimeMenu) > UPDATE_INTERVAL_OLED_MENU) {
@ -623,9 +812,9 @@ void loop() {
CG_length = ((weightLoadCell[LC2] * model.distance[X2]) / weightTotal) + model.distance[X1];
#if defined(ESP8266)
if(model.mechanicsType == 2){
if (model.mechanicsType == 2) {
CG_length = ((weightLoadCell[LC2] * model.distance[X2]) / weightTotal) - model.distance[X1];
}else if(model.mechanicsType == 3){
} else if (model.mechanicsType == 3) {
CG_length = ((weightLoadCell[LC2] * model.distance[X2]) / weightTotal) * -1 + model.distance[X1];
}
#endif
@ -643,80 +832,9 @@ void loop() {
// read battery voltage
if (batType > B_OFF) {
batVolt = (analogRead(VOLTAGE_PIN) / 1024.0) * V_REF * ((resistor[R1] + resistor[R2]) / resistor[R2]) / 1000.0;
#if ENABLE_PERCENTLIST
if (batType > B_VOLT) {
batVolt = percentBat(batVolt / batCells);
}
#endif
}
// print to display
char buff[8];
int pos_weightTotal = 7;
int pos_CG_length = 28;
if (nLoadcells == 2) {
pos_weightTotal = 17;
pos_CG_length = 45;
if (batType == 0) {
pos_weightTotal = 12;
pos_CG_length = 40;
}
}
oledDisplay.firstPage();
do {
if (errMsgCnt == 0) {
// print battery
if (batType > B_OFF) {
oledDisplay.drawXBMP(88, 1, 12, 6, batteryImage);
if (batType == B_VOLT) {
dtostrf(batVolt, 2, 2, buff);
} else {
dtostrf(batVolt, 3, 0, buff);
oledDisplay.drawBox(89, 2, (batVolt / (100 / 8)), 4);
}
oledDisplay.setFont(u8g2_font_5x7_tr);
oledDisplay.setCursor(123 - oledDisplay.getStrWidth(buff), 7);
oledDisplay.print(buff);
if (batType == B_VOLT) {
oledDisplay.print(F("V"));
} else {
oledDisplay.print(F("%"));
}
}
// print total weight
oledDisplay.drawXBMP(2, pos_weightTotal, 18, 18, weightImage);
dtostrf(weightTotal, 5, 1, buff);
oledDisplay.setFont(u8g2_font_helvR12_tr);
oledDisplay.setCursor(93 - oledDisplay.getStrWidth(buff), pos_weightTotal + 17);
oledDisplay.print(buff);
oledDisplay.print(F(" g"));
// print CG longitudinal axis
oledDisplay.drawXBMP(2, pos_CG_length, 18, 18, CGImage);
dtostrf(CG_length, 5, 1, buff);
oledDisplay.setCursor(93 - oledDisplay.getStrWidth(buff), pos_CG_length + 16);
oledDisplay.print(buff);
oledDisplay.print(F(" mm"));
// print CG transverse axis
if (nLoadcells == 3) {
oledDisplay.drawXBMP(2, 47, 18, 18, CGtransImage);
dtostrf(CG_trans, 5, 1, buff);
oledDisplay.setCursor(93 - oledDisplay.getStrWidth(buff), 64);
oledDisplay.print(buff);
oledDisplay.print(F(" mm"));
}
} else {
oledDisplay.setFont(u8g2_font_5x7_tr);
for (int i = 1; i <= errMsgCnt; i++) {
oledDisplay.setCursor(0, 7 * i);
oledDisplay.print(errMsg[i]);
}
}
} while ( oledDisplay.nextPage() );
printScaleOLED();
// serial connection
if (Serial) {
@ -839,7 +957,7 @@ void loop() {
switch (menuPage)
{
case MENU_HOME: {
Serial.print(F("\n\n********************************************\nCG scale by M.Lehmann - V"));
Serial.print(F("\n\n********************************************\nCG scale by M.Lehmann et al. - V"));
Serial.print(CGSCALE_VERSION);
Serial.print(F("\n\n"));
@ -851,7 +969,7 @@ void loop() {
for (int i = X1; i <= X3; i++) {
Serial.print(MENU_DISTANCE_X1 + i);
Serial.print(F(" - Set distance X"));
Serial.print(i+1);
Serial.print(i + 1);
Serial.print(F(" ("));
Serial.print(model.distance[i]);
Serial.print(F("mm)\n"));
@ -872,9 +990,9 @@ void loop() {
for (int i = LC1; i <= LC3; i++) {
Serial.print(MENU_LOADCELL1_CALIBRATION_FACTOR + i);
if((MENU_LOADCELL1_CALIBRATION_FACTOR + i) < 10) Serial.print(F(" "));
if ((MENU_LOADCELL1_CALIBRATION_FACTOR + i) < 10) Serial.print(F(" "));
Serial.print(F(" - Set calibration factor of load cell "));
Serial.print(i+1);
Serial.print(i + 1);
Serial.print(F(" ("));
Serial.print(calFactorLoadcell[i]);
Serial.print(F(")\n"));
@ -883,7 +1001,7 @@ void loop() {
for (int i = R1; i <= R2; i++) {
Serial.print(MENU_RESISTOR_R1 + i);
Serial.print(F(" - Set value of resistor R"));
Serial.print(i+1);
Serial.print(i + 1);
Serial.print(F(" ("));
Serial.print(resistor[i]);
Serial.print(F("ohm)\n"));
@ -1112,10 +1230,16 @@ void getValue() {
response += buff;
response += "V";
} else {
dtostrf(batVolt, 5, 0, buff);
float percentVolt = percentBat(batVolt / batCells);
dtostrf(percentVolt, 5, 0, buff);
response += buff;
response += "%";
response += "%/";
dtostrf(batVolt, 5, 2, buff);
response += buff;
response += "V";
}
Serial.print("send response: ");
Serial.println(response);
server.send(200, "text/html", response);
}
@ -1138,9 +1262,13 @@ void getRawValue() {
response += buff;
response += "V";
} else {
dtostrf(batVolt, 5, 0, buff);
float percentVolt = percentBat(batVolt / batCells);
dtostrf(percentVolt, 5, 0, buff);
response += buff;
response += "%";
response += "%/";
dtostrf(batVolt, 5, 2, buff);
response += buff;
response += "V";
}
server.send(200, "text/html", response);
}
@ -1234,14 +1362,14 @@ void getWiFiNetworks() {
bool ssidSTAavailable = false;
String response = "";
int n = WiFi.scanNetworks();
if (n > 0) {
for (int i = 0; i < n; ++i) {
response += WiFi.SSID(i);
if(WiFi.SSID(i) == ssid_STA) ssidSTAavailable = true;
if (WiFi.SSID(i) == ssid_STA) ssidSTAavailable = true;
if (i < n - 1) response += "&";
}
if(!ssidSTAavailable){
if (!ssidSTAavailable) {
response += "&";
response += ssid_STA;
}
@ -1293,7 +1421,7 @@ void saveParameter() {
EEPROM.put(P_ENABLE_OTA, enableOTA);
EEPROM.commit();
if(model.name != ""){
if (model.name != "") {
saveModelJson(model.name);
}
@ -1554,7 +1682,7 @@ bool deleteModelJson(String modelName) {
void writeModelData(JsonObject object) {
char buff[8];
String stringBuff;
dtostrf(weightTotal, 5, 1, buff);
stringBuff = buff;
stringBuff.trim();
@ -1579,14 +1707,13 @@ void writeModelData(JsonObject object) {
// print update progress screen
void printUpdateProgress(unsigned int progress, unsigned int total) {
printConsole(T_UPDATE, updateMsg);
oledDisplay.firstPage();
do {
oledDisplay.setFont(u8g2_font_helvR08_tr);
oledDisplay.setFont(oledFontSmall);
oledDisplay.setCursor(0, 12);
oledDisplay.print(updateMsg);
oledDisplay.setFont(u8g2_font_5x7_tr);
oledDisplay.setCursor(107, 35);
oledDisplay.printf("%u%%\r", (progress / (total / 100)));
@ -1611,10 +1738,10 @@ char * TimeToString(unsigned long t)
return str;
}
void printConsole(int t, String msg){
void printConsole(int t, String msg) {
Serial.print(TimeToString(millis()));
Serial.print(" [");
switch(t) {
switch (t) {
case T_BOOT:
Serial.print("BOOT");
break;
@ -1640,7 +1767,7 @@ void printConsole(int t, String msg){
// https update
bool httpsUpdate(uint8_t command){
bool httpsUpdate(uint8_t command) {
if (!httpsClient.connect(HOST, HTTPS_PORT)) {
printConsole(T_ERROR, "Wifi: connection to GIT failed");
return false;
@ -1648,44 +1775,44 @@ bool httpsUpdate(uint8_t command){
const char * headerKeys[] = {"Location"} ;
const size_t numberOfHeaders = 1;
HTTPClient https;
https.setUserAgent("cgscale");
https.setRedirectLimit(0);
https.setFollowRedirects(true);
String url = "https://" + String(HOST) + String(URL);
if (https.begin(httpsClient, url)) {
if (https.begin(httpsClient, url)) {
https.collectHeaders(headerKeys, numberOfHeaders);
printConsole(T_HTTPS, "GET: " + url);
int httpCode = https.GET();
if (httpCode > 0) {
if (httpCode > 0) {
// response
if (httpCode == HTTP_CODE_FOUND) {
if (httpCode == HTTP_CODE_FOUND) {
String newUrl = https.header("Location");
gitVersion = newUrl.substring(newUrl.lastIndexOf('/')+2).toFloat();
if(gitVersion > String(CGSCALE_VERSION).toFloat()){
gitVersion = newUrl.substring(newUrl.lastIndexOf('/') + 2).toFloat();
if (gitVersion > String(CGSCALE_VERSION).toFloat()) {
printConsole(T_UPDATE, "Firmware update available: V" + String(gitVersion));
}else{
} else {
printConsole(T_UPDATE, "Firmware version found on GitHub: V" + String(gitVersion) + " - current firmware is up to date");
}
}else if (httpCode == HTTP_CODE_OK || httpCode == HTTP_CODE_MOVED_PERMANENTLY) {
} else if (httpCode == HTTP_CODE_OK || httpCode == HTTP_CODE_MOVED_PERMANENTLY) {
Serial.println(https.getString());
}else{
} else {
printConsole(T_ERROR, "HTTPS: GET... failed, " + https.errorToString(httpCode));
https.end();
return false;
}
}else {
} else {
return false;
}
https.end();
}else {
https.end();
} else {
printConsole(T_ERROR, "Wifi: Unable to connect");
return false;
}
return true;
}

View File

@ -7,7 +7,7 @@ Schwerpunktwaage auf Arduinobasis zum auswiegen von Flugmodellen. Es werden rela
Die wichtigsten Funktionen:
- unterstützt Waagen mit 2 oder 3 Wiegezellen
- unterstützt ESP8266 und Arduino mit ATmega328, ATmega32u4
- unterstützt ESP8266 (auch Wifi Kit 8) und Arduino mit ATmega328, ATmega32u4
- automatische Kalibrierung anhand eines Referenzobjekts, dadurch kein mühsames eruieren der Kalibrierwerte
- Anzeige durch OLED Display
- Batteriespannnung kann gemessen werden

880
data/index.html Executable file
View File

@ -0,0 +1,880 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
<meta name="description" content="">
<meta name="author" content="M. Lehmann">
<title>CG scale by M. Lehmann</title>
<style>
body {
font-family: "Trebuchet MS", Arial, Helvetica, sans-serif;
background: #DCDCDC;
margin:0;
}
/***** remove standard-styles *****/
input, select, textarea {
-webkit-appearance: none;
-moz-appearance: none;
appearance: none;
border:none;
border-radius: 4px;
font-size: 1em;
width: 100%
}
/***** navbar *****/
.navbar {
list-style-type: none;
margin: 0;
padding: 0;
top: 0;
width: 100%;
background-color: #555;
position: fixed;
overflow: auto;
z-index: 10;
}
.navbar logo {
float: left;
display: block;
padding: 12px;
color: white;
text-decoration: none;
font-size: 32px;
}
.navbar a {
float: right;
display: block;
padding: 12px;
color: white;
text-decoration: none;
font-size: 32px;
}
.navbar a:hover {
background-color: #000;
}
.active {
background-color: #1E90FF;
}
.navbar .icon {
display: none;
}
.navbar .badge {
position: absolute;
top: 0px;
right: 0px;
padding: 5px 8px;
border-radius: 30%;
background: red;
color: white;
font-size: 20px;
}
@media screen and (max-width: 600px) {
.navbar a:not(:first-child) {display: none;}
.navbar a.icon {
float: right;
display: block;
}
.navbar.responsive {position: relative;}
.navbar.responsive .icon {
position: absolute;
right: 0;
top: 0;
height: auto;
}
.navbar.responsive a {
float: none;
display: block;
text-align: left;
}
.navbar.responsive logo{display: none;}
}
/***** tab *****/
* {
box-sizing: border-box;
}
.wrapper {
width: 100%;
margin: 0 auto;
}
.tabs {
position: relative;
margin: 3rem 0;
background: #1abc9c;
}
.tabs::before,
.tabs::after {
content: "";
display: table;
}
.tabs::after {
clear: both;
}
.tab {
float: left;
}
.tab-switch {
display: none;
}
.tab-label {
position: relative;
display: block;
line-height: 2.75em;
height: 3em;
padding: 0 1.618em;
background: #1abc9c;
border-right: 0.125rem solid #16a085;
color: #2c3e50;
cursor: pointer;
top: 0;
transition: all 0.25s;
}
.tab-label:hover {
top: -0.25rem;
transition: top 0.25s;
}
.tab-content {
height: 1320px;
width: 100%;
position: absolute;
z-index: 1;
top: 2.75em;
left: 0;
padding: 1.618rem;
background: #FFFFFF;
color: #2c3e50;
opacity: 0;
transition: all 0.35s;
}
.tab-switch:checked + .tab-label {
background: #FFFFFF;
color: #2c3e50;
border-bottom: 0;
transition: all 0.35s;
z-index: 1;
top: -0.0625rem;
}
.tab-switch:checked + label + .tab-content {
z-index: 2;
opacity: 1;
transition: all 0.35s;
}
/***** button *****/
.button {
border: none;
border-radius: 4px;
color: white;
padding: 7px 16px;
text-align: center;
text-decoration: none;
display: inline-block;
font-size: 16px;
margin: 4px 2px;
-webkit-transition-duration: 0.4s; /* Safari */
transition-duration: 0.4s;
cursor: pointer;
}
.tareButton {
background-color: white;
color: black;
border: 2px solid #778899;
}
.tareButton:hover {
background-color: #778899;
color: white;
}
.saveButton {
background-color: white;
color: black;
border: 2px solid #1abc9c;
}
.saveButton:hover {
background-color: #1abc9c;
color: white;
}
/***** Labels *****/
big{
display: inline-block;
padding: 10px 2px;
padding-top: 15px;
font-size: 16px;
color: #3c3c3c;
}
.big_short{
width: 12%;
text-align: left;
color: #808080;
}
small{
display: inline-block;
padding: 10px 2px;
font-size: 12px;
color: #808080;
}
.titlebar {
margin-left:30px;
margin-top:95px;
font-size:25px;
float: left;
width:40%;
}
.buttonbar {
margin-right:20px;
margin-top:85px;
float: right;
text-align:right;
width:50%;
}
.main_container {
padding-left:20px;
padding-right: 20px;
padding-top:105px;
padding-bottom: 1300px
}
.basic_img {
height:50px;
padding-left:30px;
vertical-align:middle
}
.basic_text_cont{
padding-left:20px;
vertical-align:middle;
float: right;
width: 70%;
}
.basic_value{
color:#000000;
font-size:40px;
width: 100%;
}
.basic_value_saved{
color:#808080;
font-size:17px;
width: 100%;
}
@media screen and (max-width: 600px) {
.titlebar {
width:100%;
}
.buttonbar {
margin-left:20px;
margin-top:20px;
float: left;
text-align:left;
width:100%;
}
.main_container {
padding-top:165px;
}
}
/***** Input *****/
input{
background-color:white;
border: 1px solid #bbb;
padding:.75em 1em .5em 1em;
font-size: 16px;
color: #3c3c3c;
}
.input_short{
float: right;
width: 86%;
}
input:active,
/***** selectbox *****/
select {
/*width:100%;*/
border: 1px solid #bbb;
padding:.75em 1em .5em 1em;
background-color:white;
background-image:url('');
background-position: right;
background-repeat: no-repeat;
font-size: 16px;
color: #3c3c3c;
}
/* Hide browser-styling in IE10 */
select::-ms-expand {
display:none;
}
/* Hide custom-icons in lower versions of Internet Explorer (< IE10). */
.lt-ie10 select {
background-image: none;
}
</style>
<script>
var CG_trans_visible = true;
var batVolt_visible = true;
var curModelName = "";
var targetCGmin = 0;
var targetCGmax = 0;
var mType = 0;
var imgFW190 = new Image();
imgFW190.src = "fw190.png";
var translation;
var languages = {
"en":{
"measure":"Measure",
"tareBtn":"Tare",
"saveBtn":"Save",
"tabConfig":"Config",
"gearTyp":"Landing gear type:",
"gearTyp2":"2 point with tailspur",
"gearTyp3":"3 point with nosewheel",
"distances":"Distances in mm:",
"tabBasic":"Basic",
"weightTotalSaved":"not saved",
"CG_lengthSaved":"not saved",
"CG_transSaved":"not saved",
"tabExpert":"Extended",
"targetCG":"Target CG in mm:"
},
"de":{
"measure":"Messen",
"tareBtn":"Tare",
"saveBtn":"Speichern",
"tabConfig":"Konfig",
"gearTyp":"Fahrwerktyp:",
"gearTyp2":"2 Punkt mit Hecksporn",
"gearTyp3":"3 Punkt mit Bugrad",
"distances":"Distanzen in mm:",
"tabBasic":"Basis",
"weightTotalSaved":"nicht gespeichert",
"CG_lengthSaved":"nicht gespeichert",
"CG_transSaved":"nicht gespeichert",
"tabExpert":"Erweitert",
"targetCG":"Soll CG in mm:"
}
};
function setTranslation(defaultLang) {
var lang = navigator.languages ? navigator.languages[0] : navigator.language;
lang = lang.substr(0, 2);
translation = languages[lang] || languages[defaultLang];
for (var item in translation) {
var docElement = document.getElementById(item);
if(docElement){
docElement.innerHTML = translation[item];
}
}
}
function getHead(){
var request = new XMLHttpRequest();
request.onreadystatechange = function(){
if(this.readyState == 4 && this.status == 200){
if(this.responseText != null){
var responseString = this.responseText;
var value = responseString.split("&");
document.getElementById("ssid").innerHTML = value[0];
document.getElementById("errMsg").innerHTML = value[1];
document.getElementById("cgscaleVersion").innerHTML = value[2];
// check if new version available
if(parseFloat(value[3]) > parseFloat(value[2])){
var a = document.getElementById('settingsBtn');
a.insertAdjacentHTML('beforeend', '<span class="badge">1</span>');
}
}
}
}
request.open("GET", "getHead", true);
request.send();
}
function getParameter(){
var request = new XMLHttpRequest();
request.onreadystatechange = function(){
if(this.readyState == 4 && this.status == 200){
if(this.responseText != null){
var responseString = this.responseText;
var value = responseString.split("&");
if(value[0] == "2"){
CG_trans_visible = false;
document.getElementById("cglrContainer").outerHTML = "";
}
if(value[11] == "0"){
batVolt_visible = false;
document.getElementById("batContainer").outerHTML = "";
}
curModelName = value[17];
if (curModelName != "") {
document.getElementById("modelName").innerHTML = curModelName;
setValue("distanceX1", value[1]);
setValue("distanceX2", value[2]);
setValue("distanceX3", value[3]);
document.getElementById("weightTotalSaved").innerHTML = value[18];
document.getElementById("CG_lengthSaved").innerHTML = value[19];
if(CG_trans_visible == true){
document.getElementById("CG_transSaved").innerHTML = value[20];
}
setValue("targetCGmin", value[21]);
setValue("targetCGmax", value[22]);
targetCGmin = parseFloat(value[21]);
targetCGmax = parseFloat(value[22]);
mType = parseInt(value[23]);
if(mType == 0){
document.getElementById("tab-2").checked = true;
}else if(mType == 1){
document.getElementById("tab-config").hidden = false;
document.getElementById("tab-1").checked = true;
document.getElementById("gearTyp").outerHTML = "";
document.getElementById("gearTypeImg").src = "CG_scale_mechanics.png";
}else if(mType > 1){
document.getElementById("tab-config").hidden = false;
document.getElementById("tab-1").checked = true;
document.getElementById("mechanicsType").hidden = false;
setValue("mechanicsType", value[23]);
changeGearImg();
}
}
}
}
}
request.open("GET", "getParameter", true);
request.send();
}
function getValue(){
var request = new XMLHttpRequest();
request.onreadystatechange = function(){
if(this.readyState == 4 && this.status == 200){
if(this.responseText != null){
var responseString = this.responseText;
var value = responseString.split("&");
document.getElementById("weightTotal").innerHTML = value[0];
document.getElementById("CG_length").innerHTML = value[1];
if(CG_trans_visible == true){
document.getElementById("CG_trans").innerHTML = value[2];
}
if(batVolt_visible == true){
drawBattery(value[3]);
}
drawExpert(value[0],value[1],value[2]);
}
}
}
request.open("GET", "getValue", true);
request.send();
setTimeout('getValue()', 1200);
}
function saveModel(){
if (curModelName != "") {
var data = "?";
data += "modelname=" + curModelName;
data += "&targetCGmin=" + document.getElementById("targetCGmin").value;
data += "&targetCGmax=" + document.getElementById("targetCGmax").value;
if(parseInt(mType) > 0){
data += "&distanceX1=" + document.getElementById("distanceX1").value;
data += "&distanceX2=" + document.getElementById("distanceX2").value;
data += "&distanceX3=" + document.getElementById("distanceX3").value;
}
data += "&mechanicsType=" + mType;
var request = new XMLHttpRequest();
request.onreadystatechange = function(){
if(this.readyState == 4 && this.status == 200){
alert(curModelName + " saved");
location.reload();
}else if(this.readyState == 4 && this.status == 404){
alert("Model not saved !");
}
}
request.open("GET", "saveModel" + data, true);
request.send();
}
}
function setValue(id, value)
{
var element = document.getElementById(id);
element.value = value;
}
function tare(){
document.getElementById("weightTotal").innerHTML = "TARE";
var request = new XMLHttpRequest();
request.onreadystatechange = function(){
if(this.readyState == 4 && this.status == 404){
alert("Tare failed !");
}
}
request.open("GET", "tare", true);
request.send();
}
function drawBattery(value){
document.getElementById("batValue").innerHTML = value;
var canvas = document.getElementById('batCanvas');
var ctx = canvas.getContext('2d');
ctx.clearRect(0,0,canvas.width, canvas.height);
// battery icon
var r = 5;
var l = 40;
var h = 20;
var bl = 3;
var bh = 3;
var ox = 2;
var oy = 2;
// battery fill
var fr = 2;
var fl = 34; //100%
var fh = 14;
var fx = 5;
var fy = 5;
var percentpos = value.indexOf('%');
if(percentpos != -1){
var percent = Number(value.substring(0, percentpos));
fl = fl/100*percent;
}else{
fl = 0;
}
// draw icon
ctx.beginPath();
ctx.lineWidth = 2;
ctx.moveTo(ox, oy+r);
ctx.arc(ox+r, oy+r, r, (Math.PI / 180) * 180, (Math.PI / 180) * 270, false);
ctx.lineTo(ox+(l-r), oy);
ctx.arc(ox+(l-r), oy+r, r, (Math.PI / 180) * 270, 0, false);
ctx.lineTo(ox+l, h/2-bh+oy);
ctx.lineTo(ox+l+bl, h/2-bh+oy);
ctx.lineTo(ox+l+bl, h/2+bh+oy);
ctx.lineTo(ox+l, h/2+bh+oy);
ctx.lineTo(ox+l, oy+h-r);
ctx.arc(ox+(l-r), oy+h-r, r, 0, (Math.PI / 180) * 90, false);
ctx.lineTo(ox+r, oy+h);
ctx.arc(ox+r, oy+h-r, r, (Math.PI / 180) * 90, (Math.PI / 180) * 180, false);
ctx.lineTo(ox, oy+r);
ctx.strokeStyle = "#FFFFFF";
ctx.stroke();
// draw filled value
if(fl>1){
ctx.beginPath();
ctx.fillStyle = "#FFFFFF";
ctx.moveTo(fx, fy+fr);
ctx.arc(fx+fr, fy+fr, fr, (Math.PI / 180) * 180, (Math.PI / 180) * 270, false);
ctx.lineTo(fx+(fl-fr), fy);
ctx.arc(fx+(fl-fr), fy+fr, fr, (Math.PI / 180) * 270, 0, false);
ctx.lineTo(fx+fl, fy+fh-fr);
ctx.arc(fx+(fl-fr), fy+fh-fr, fr, 0, (Math.PI / 180) * 90, false);
ctx.lineTo(fx+fr, fy+fh);
ctx.arc(fx+fr, fy+fh-fr, fr, (Math.PI / 180) * 90, (Math.PI / 180) * 180, false);
ctx.lineTo(fx, fy+fr);
ctx.fill();
}
}
function drawExpert(weight,cg,cglr){
var tolerance = (targetCGmax-targetCGmin)/2;
var centerVal = parseFloat(cg) - (tolerance + targetCGmin);
var canvas = document.getElementById("expertCanvas");
var ctx = canvas.getContext("2d");
ctx.clearRect(0,0,canvas.width, canvas.height);
// print profil
var pHeight = 400;
var pHeightOffset = 170;
ctx.beginPath();
ctx.lineWidth = 3;
ctx.moveTo(canvas.width/1000*5, pHeight/500*100+pHeightOffset);
ctx.bezierCurveTo(canvas.width/1000*20, pHeight/500*-40+pHeightOffset, canvas.width/1000*1000, pHeight/500*95+pHeightOffset, canvas.width/1000*995, pHeight/500*100+pHeightOffset);
ctx.moveTo(canvas.width/1000*5, pHeight/500*100+pHeightOffset);
ctx.bezierCurveTo(canvas.width/1000*5, pHeight/500*140+pHeightOffset, canvas.width/1000*1000, pHeight/500*104+pHeightOffset, canvas.width/1000*995, pHeight/500*100+pHeightOffset);
ctx.stroke();
// print difference
ctx.font = "25px Trebuchet MS";
var strCenterVal = centerVal.toFixed(1)+"mm";
var maxProgresPx = 100;
var progresPx = (maxProgresPx / 2 / tolerance) * centerVal;
if(progresPx > 0){
if(progresPx < maxProgresPx/2){
ctx.fillStyle = "#00b300";
}else{
ctx.fillStyle = "#FF0000";
if(progresPx > maxProgresPx){
progresPx = maxProgresPx;
}
}
ctx.fillRect(225, 125+pHeightOffset, progresPx, 25);
ctx.fillStyle = "#808080";
ctx.textAlign = "left";
ctx.fillText(strCenterVal, 232 + progresPx, 147+pHeightOffset);
}else if(progresPx < 0){
if(progresPx > maxProgresPx / 2 * -1){
ctx.fillStyle = "#00b300";
}else{
ctx.fillStyle = "#FF0000";
if(progresPx < maxProgresPx * -1){
progresPx = maxProgresPx * -1;
}
}
ctx.fillRect(225 + progresPx, 125+pHeightOffset, progresPx * -1, 25);
ctx.fillStyle = "#808080";
ctx.textAlign = "right";
ctx.fillText(strCenterVal, 218 + progresPx, 147+pHeightOffset);
}else{
ctx.fillStyle = "#808080";
ctx.textAlign = "left";
ctx.fillText(strCenterVal, 190, 147+pHeightOffset);
}
// print CG
var cgimg = document.getElementById("cgimg");
ctx.drawImage(cgimg, 198, 180+pHeightOffset,55,55);
ctx.font = "42px Trebuchet MS";
ctx.fillStyle = "#000000";
ctx.textAlign = "left";
ctx.fillText(cg, 300, 224+pHeightOffset);
// print weight
var weightimg = document.getElementById("weightimg");
ctx.drawImage(weightimg, 198, 50,55,55);
ctx.font = "42px Trebuchet MS";
ctx.fillStyle = "#000000";
ctx.textAlign = "left";
ctx.fillText(weight, 300, 94);
// print CG L-R
if(CG_trans_visible == true){
var cglrHeightOffset = 550;
ctx.drawImage(imgFW190, 50, cglrHeightOffset);
// print difference
ctx.font = "25px Trebuchet MS";
maxProgresPx = 100;
progresPx = (maxProgresPx / 2 / tolerance) * parseFloat(cglr);
if(progresPx > 0){
ctx.fillStyle = "#FF0000";
if(progresPx > maxProgresPx){
progresPx = maxProgresPx;
}
ctx.fillRect(362, cglrHeightOffset+175, progresPx, 25);
ctx.fillStyle = "#808080";
ctx.textAlign = "left";
ctx.fillText(cglr, 369 + progresPx, cglrHeightOffset+197);
}else if(progresPx < 0){
ctx.fillStyle = "#FF0000";
if(progresPx < maxProgresPx * -1){
progresPx = maxProgresPx * -1;
}
ctx.fillRect(362 + progresPx, cglrHeightOffset+175, progresPx * -1, 25);
ctx.fillStyle = "#808080";
ctx.textAlign = "right";
ctx.fillText(cglr, 355 + progresPx, cglrHeightOffset+197);
}else{
ctx.fillStyle = "#808080";
ctx.textAlign = "left";
ctx.fillText(cglr, 320, cglrHeightOffset+197);
}
}
}
function mobileNavbar() {
var x = document.getElementById("Navbar");
if (x.className === "navbar") {
x.className += " responsive";
} else {
x.className = "navbar";
}
}
function changeGearImg() {
mType = document.getElementById("mechanicsType").value;
if(mType == 2){
document.getElementById("gearTypeImg").src="pc6.png";
}else if(mType == 3){
document.getElementById("gearTypeImg").src="pc7.png";
}
}
</script>
</head>
<body>
<div class="navbar" id="Navbar">
<logo id="batContainer">
<canvas id="batCanvas" height="24" width="50" style="vertical-align:middle;"></canvas>
<span id="batValue" style="color:#FFFF; padding-right:20px; vertical-align:middle; font-size:17px;">-</span>
</logo>
<logo id="ssid" style="vertical-align:middle; font-size:32px;"></logo>
<a id="settingsBtn" href="/settings.html"><img height="32" src=""></a>
<a href="/models.html"><img height="32" src=""></a>
<a class="active" href="/index.html"><img height="32" src=""></a>
<a href="javascript:void(0);" class="icon" onclick="mobileNavbar()">
<img height="32px" src="">
</a>
</div>
<div>
<div class="titlebar">
<span style="color: #808080;" id="measure"></span><span id="modelName" style="padding-left:10px; color: #1abc9c;"></span>
</div>
<div class="buttonbar">
<button type="button" class="button tareButton" onclick="tare()" id="tareBtn"></button>
<button type="button" class="button saveButton" onclick="saveModel()" id="saveBtn"></button>
</div>
</div>
<div class="wrapper main_container">
<div class="tabs">
<div class="tab" hidden id="tab-config">
<input type="radio" name="css-tabs" id="tab-1" class="tab-switch">
<label for="tab-1" class="tab-label" id="tabConfig"></label>
<div class="tab-content">
<div>
<big id="gearTyp"></big>
<select hidden id="mechanicsType" onchange="changeGearImg()">
<option value="2" id="gearTyp2"></option>
<option value="3" id="gearTyp3"></option>
</select>
</div>
<div style="margin-top:50px; margin-left: 30px; margin-bottom: 50px;">
<img style="max-width: 100%" id="gearTypeImg" src="">
</div>
<div>
<big id="distances"></big>
</div>
<div style="margin-bottom: 10px">
<big class="big_short">X1:</big>
<input type="text" id="distanceX1" class="input_short">
</div>
<div style="margin-bottom: 10px">
<big class="big_short">X2:</big>
<input type="text" id="distanceX2" class="input_short">
</div>
<div>
<big class="big_short">X3:</big>
<input type="text" id="distanceX3" class="input_short">
</div>
</div>
</div>
<div class="tab">
<input type="radio" name="css-tabs" id="tab-2" class="tab-switch">
<label for="tab-2" class="tab-label" id="tabBasic"></label>
<div class="tab-content">
<div style="padding-bottom:50px; padding-top: 30px;">
<img id="weightimg" class="basic_img" src="">
<div class="basic_text_cont" >
<div class="basic_value" id="weightTotal">-</div>
<div class="basic_value_saved" id="weightTotalSaved"></div>
</div>
</div>
<div style="padding-bottom:50px">
<img id="cgimg" class="basic_img" src="">
<div class="basic_text_cont" >
<div class="basic_value" id="CG_length">-</div>
<div class="basic_value_saved" id="CG_lengthSaved"></div>
</div>
</div>
<div id="cglrContainer">
<img class="basic_img" src="">
<div class="basic_text_cont" >
<div class="basic_value" id="CG_trans">-</div>
<div class="basic_value_saved" id="CG_transSaved"></div>
</div>
</div>
<div style="color:#FF0000; padding-left:30px; padding-top:20px; font-size:17px;">
<span id="errMsg"></span>
</div>
</div>
</div>
<div class="tab">
<input type="radio" name="css-tabs" id="tab-3" class="tab-switch">
<label for="tab-3" class="tab-label" id="tabExpert"></label>
<div class="tab-content">
<div>
<big id="targetCG"></big>
</div>
<div style="margin-bottom: 10px">
<big class="big_short">min:</big>
<input type="text" id="targetCGmin" class="input_short" onchange="targetCGmin = parseFloat(this.value)">
</div>
<div>
<big class="big_short">max:</big>
<input type="text" id="targetCGmax" class="input_short" onchange="targetCGmax = parseFloat(this.value)">
</div>
<div style="margin-top:50px; text-align: center;">
<canvas id="expertCanvas" width="700" height="1000" style="max-width: 100%"></canvas>
</div>
</div>
</div>
</div>
</div>
<div style="color:#808080; text-align:center; font-size:12px;">(c) 2019 M. Lehmann - Version: <span id="cgscaleVersion">--</span></div>
<script>
setTranslation("en");
drawExpert("-","-","-");
getHead();
getParameter();
getValue();
</script>
</body>
</html>

Binary file not shown.

531
data/models.html Executable file
View File

@ -0,0 +1,531 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
<meta name="description" content="">
<meta name="author" content="M. Lehmann">
<title>CG scale by M. Lehmann</title>
<style>
body {
font-family: "Trebuchet MS", Arial, Helvetica, sans-serif;
background: #DCDCDC;
margin:0;
}
/***** navbar *****/
.navbar {
list-style-type: none;
margin: 0;
padding: 0;
top: 0;
width: 100%;
background-color: #555;
position: fixed;
overflow: auto;
z-index: 10;
}
.navbar logo {
float: left;
display: block;
padding: 12px;
color: white;
text-decoration: none;
font-size: 32px;
}
.navbar a {
float: right;
display: block;
padding: 12px;
color: white;
text-decoration: none;
font-size: 32px;
}
.navbar a:hover {
background-color: #000;
}
.active {
background-color: #1E90FF;
}
.navbar .icon {
display: none;
}
.navbar .badge {
position: absolute;
top: 0px;
right: 0px;
padding: 5px 8px;
border-radius: 30%;
background: red;
color: white;
font-size: 20px;
}
@media screen and (max-width: 600px) {
.navbar a:not(:first-child) {display: none;}
.navbar a.icon {
float: right;
display: block;
}
.navbar.responsive {position: relative;}
.navbar.responsive .icon {
position: absolute;
right: 0;
top: 0;
height: auto;
}
.navbar.responsive a {
float: none;
display: block;
text-align: left;
}
.navbar.responsive logo{display: none;}
}
/***** Table *****/
table {
border-collapse: collapse;
width: 100%;
cursor: pointer;
}
th {
padding: 8px;
text-align: left;
border-bottom: 1px solid #ddd;
background-color:#1abc9c;
}
td {
height: 40px;
padding: 8px;
text-align: left;
border-bottom: 1px solid #ddd;
background-color:#ffffff;
}
tr:hover td{background-color:#f5f5f5;}
/***** button *****/
.button {
border: none;
border-radius: 3px;
color: white;
padding: 3px 10px;
text-align: center;
text-decoration: none;
display: inline-block;
font-size: 16px;
margin: 4px 2px;
-webkit-transition-duration: 0.4s; /* Safari */
transition-duration: 0.4s;
cursor: pointer;
}
.openButton {
background-color: white;
color: black;
border: 2px solid #1abc9c;
}
.openButton:hover {
background-color: #1abc9c;
color: white;
}
.deleteButton {
background-color: white;
color: black;
border: 2px solid #DC143C;
}
.deleteButton:hover {
background-color: #DC143C;
color: white;
}
.newButton {
padding: 7px 16px;
background-color: white;
color: black;
border: 2px solid #778899;
}
.newButton:hover {
background-color: #778899;
color: white;
}
/***** Labels *****/
.titlebar {
margin-left:30px;
margin-top:95px;
font-size:25px;
float: left;
width:40%;
}
.buttonbar {
margin-right:20px;
margin-top:85px;
float: right;
text-align:right;
width:50%;
}
.main_container {
padding-left:20px;
padding-right: 20px;
padding-top:152px;
}
@media screen and (max-width: 600px) {
.titlebar {
width:100%;
}
.buttonbar {
margin-left:20px;
margin-top:20px;
float: left;
text-align:left;
width:100%;
}
.main_container {
padding-top:212px;
}
}
</style>
<script>
var CG_trans_visible = true;
var batVolt_visible = true;
var curModelName = "";
var translation;
var languages = {
"en":{
"modeltitle":"Model",
"saveBtn":"New model",
"thName":"Name"
},
"de":{
"modeltitle":"Modell",
"saveBtn":"Neues Modell",
"thName":"Name"
}
};
function setTranslation(defaultLang) {
var lang = navigator.languages ? navigator.languages[0] : navigator.language;
lang = lang.substr(0, 2);
translation = languages[lang] || languages[defaultLang];
for (var item in translation) {
var docElement = document.getElementById(item);
if(docElement){
docElement.innerHTML = translation[item];
}
}
}
function getHead(){
var request = new XMLHttpRequest();
request.onreadystatechange = function(){
if(this.readyState == 4 && this.status == 200){
if(this.responseText != null){
var responseString = this.responseText;
var value = responseString.split("&");
document.getElementById("ssid").innerHTML = value[0];
document.getElementById("cgscaleVersion").innerHTML = value[2];
// check if new version available
if(parseFloat(value[3]) > parseFloat(value[2])){
var a = document.getElementById('settingsBtn');
a.insertAdjacentHTML('beforeend', '<span class="badge">1</span>');
}
}
}
}
request.open("GET", "getHead", true);
request.send();
}
function getParameter(){
var request = new XMLHttpRequest();
request.onreadystatechange = function(){
if(this.readyState == 4 && this.status == 200){
if(this.responseText != null){
var responseString = this.responseText;
var value = responseString.split("&");
if(value[0] == "2"){
CG_trans_visible = false;
}
if(value[11] == "0"){
batVolt_visible = false;
document.getElementById("batContainer").outerHTML = "";
}
curModelName = value[17];
if (curModelName != "") {
document.getElementById("modelName").innerHTML = curModelName;
}
}
}
}
request.open("GET", "getParameter", true);
request.send();
}
function getValue(){
var request = new XMLHttpRequest();
request.onreadystatechange = function(){
if(this.readyState == 4 && this.status == 200){
if(this.responseText != null){
var responseString = this.responseText;
var value = responseString.split("&");
if(batVolt_visible == true){
drawBattery(value[3]);
}
}
}
}
request.open("GET", "getValue", true);
request.send();
setTimeout('getValue()', 5000);
}
function getModels(){
var request = new XMLHttpRequest();
request.onreadystatechange = function(){
if(this.readyState == 4 && this.status == 200){
var root = JSON.parse(this.responseText);
var table = document.getElementById("models");
if(CG_trans_visible == false){
document.getElementById("cgLRrow").outerHTML = "";
}
for (name in root) {
var row = table.insertRow();
var cellName = row.insertCell(0);
var cellWeight = row.insertCell(1);
var cellCG = row.insertCell(2);
cellName.innerHTML = name;
cellWeight.innerHTML = root[name].wt;
cellCG.innerHTML = root[name].cg;
if(CG_trans_visible == true){
var cellcglr = row.insertCell(3);
var cellButton = row.insertCell(4);
cellcglr.innerHTML = root[name].cglr;
}else{
var cellButton = row.insertCell(3);
}
if(name != curModelName){
var htmlbutton = '<div><button type="button" class="button openButton" onclick="openModel(\'';
htmlbutton += name;
htmlbutton += '\')"><img style="height:20px; vertical-align:middle" src=""></button><button type="button" class="button deleteButton" onclick="deleteModel(\'';
htmlbutton += name;
htmlbutton += '\')"><img style="height:20px; vertical-align:middle" src=""></button></div>';
cellButton.innerHTML = htmlbutton;
}else{
row.setAttribute("class", "table-active");
}
}
}
}
request.open("GET", "models.json", true);
request.send();
}
function saveModel(){
var modelname = prompt("Please enter new model name:");
if (modelname != "") {
var request = new XMLHttpRequest();
request.onreadystatechange = function(){
if(this.readyState == 4 && this.status == 200){
location.reload();
}else if(this.readyState == 4 && this.status == 404){
alert("Model not saved !");
}
}
request.open("GET", "saveModel?modelname=" + modelname, true);
request.send();
}
}
function openModel(modelname){
if (modelname != "") {
var request = new XMLHttpRequest();
request.onreadystatechange = function(){
if(this.readyState == 4 && this.status == 200){
location.reload();
}else if(this.readyState == 4 && this.status == 404){
alert("Model not open !");
}
}
request.open("GET", "openModel?modelname=" + modelname, true);
request.send();
}
}
function deleteModel(modelname){
if (modelname != "" && confirm("Do you want to delete: " + modelname + "?")){
var request = new XMLHttpRequest();
request.onreadystatechange = function(){
if(this.readyState == 4 && this.status == 200){
location.reload();
}else if(this.readyState == 4 && this.status == 404){
alert("Model not deleted !");
}
}
request.open("GET", "deleteModel?modelname=" + modelname, true);
request.send();
}
}
function download(file)
{
window.location=file;
}
function drawBattery(value){
document.getElementById("batValue").innerHTML = value;
var canvas = document.getElementById('batCanvas');
var ctx = canvas.getContext('2d');
ctx.clearRect(0,0,canvas.width, canvas.height);
// battery icon
var r = 5;
var l = 40;
var h = 20;
var bl = 3;
var bh = 3;
var ox = 2;
var oy = 2;
// battery fill
var fr = 2;
var fl = 34; //100%
var fh = 14;
var fx = 5;
var fy = 5;
var percentpos = value.indexOf('%');
if(percentpos != -1){
var percent = Number(value.substring(0, percentpos));
fl = fl/100*percent;
}else{
fl = 0;
}
// draw icon
ctx.beginPath();
ctx.lineWidth = 2;
ctx.moveTo(ox, oy+r);
ctx.arc(ox+r, oy+r, r, (Math.PI / 180) * 180, (Math.PI / 180) * 270, false);
ctx.lineTo(ox+(l-r), oy);
ctx.arc(ox+(l-r), oy+r, r, (Math.PI / 180) * 270, 0, false);
ctx.lineTo(ox+l, h/2-bh+oy);
ctx.lineTo(ox+l+bl, h/2-bh+oy);
ctx.lineTo(ox+l+bl, h/2+bh+oy);
ctx.lineTo(ox+l, h/2+bh+oy);
ctx.lineTo(ox+l, oy+h-r);
ctx.arc(ox+(l-r), oy+h-r, r, 0, (Math.PI / 180) * 90, false);
ctx.lineTo(ox+r, oy+h);
ctx.arc(ox+r, oy+h-r, r, (Math.PI / 180) * 90, (Math.PI / 180) * 180, false);
ctx.lineTo(ox, oy+r);
ctx.strokeStyle = "#FFFFFF";
ctx.stroke();
// draw filled value
if(fl>1){
ctx.beginPath();
ctx.fillStyle = "#FFFFFF";
ctx.moveTo(fx, fy+fr);
ctx.arc(fx+fr, fy+fr, fr, (Math.PI / 180) * 180, (Math.PI / 180) * 270, false);
ctx.lineTo(fx+(fl-fr), fy);
ctx.arc(fx+(fl-fr), fy+fr, fr, (Math.PI / 180) * 270, 0, false);
ctx.lineTo(fx+fl, fy+fh-fr);
ctx.arc(fx+(fl-fr), fy+fh-fr, fr, 0, (Math.PI / 180) * 90, false);
ctx.lineTo(fx+fr, fy+fh);
ctx.arc(fx+fr, fy+fh-fr, fr, (Math.PI / 180) * 90, (Math.PI / 180) * 180, false);
ctx.lineTo(fx, fy+fr);
ctx.fill();
}
}
function mobileNavbar() {
var x = document.getElementById("Navbar");
if (x.className === "navbar") {
x.className += " responsive";
} else {
x.className = "navbar";
}
}
</script>
</head>
<body>
<div class="navbar" id="Navbar">
<logo id="batContainer">
<canvas id="batCanvas" height="24" width="50" style="vertical-align:middle;"></canvas>
<span id="batValue" style="color:#FFFF; padding-right:20px; vertical-align:middle; font-size:17px;">-</span>
</logo>
<logo id="ssid" style="vertical-align:middle; font-size:32px;"></logo>
<a id="settingsBtn" href="/settings.html"><img height="32" src=""></a>
<a class="active" href="/models.html"><img height="32" src=""></a>
<a href="/index.html"><img height="32" src=""></a>
<a href="javascript:void(0);" class="icon" onclick="mobileNavbar()">
<img height="32px" src="">
</a>
</div>
<div>
<div class="titlebar">
<span style="color: #808080;" id="modeltitle"></span><span id="modelName" style="padding-left:10px; color: #1abc9c;"></span>
</div>
<div class="buttonbar">
<button type="button" class="button newButton" onclick="saveModel()" id="saveBtn"></button>
</div>
</div>
<div class="main_container">
<table id="models">
<tr>
<th id="thName"></th>
<th><img style="height:30px; vertical-align:middle" src=""></th>
<th><img style="height:30px; vertical-align:middle" src=""></th>
<th id="cgLRrow"><img style="height:30px; vertical-align:middle" src=""></th>
<th></th>
</tr>
</table>
</div>
<div style="color:#808080; text-align:center; font-size:12px; padding-top:30px;">(c) 2019 M. Lehmann - Version: <span id="cgscaleVersion">--</span></div>
<script>
setTranslation("en");
getHead();
getParameter();
getModels();
getValue();
</script>
</body>
</html>

Binary file not shown.

1062
data/settings.html Executable file

File diff suppressed because it is too large Load Diff

Binary file not shown.

View File

@ -45,15 +45,35 @@ CG scale with 3 Loadcells:
*/
#define PIN_LOADCELL1_DOUT D6
#define PIN_LOADCELL1_PD_SCK D5
#define PIN_LOADCELL2_DOUT D2
#define PIN_LOADCELL2_PD_SCK D1
// Wifi Kit 8 (https://heltec.org/project/wifi-kit-8/)
// is a ESP8266 based board, with integrated OLED and battery management
#define WIFI_KIT_8 1
#ifdef WIFI_KIT_8
#define PIN_LOADCELL1_DOUT D6
#define PIN_LOADCELL1_PD_SCK D7
#define PIN_LOADCELL3_DOUT D7
#define PIN_LOADCELL3_PD_SCK D0
#define PIN_LOADCELL2_DOUT D3
#define PIN_LOADCELL2_PD_SCK D8
#define PIN_LOADCELL3_DOUT D0
#define PIN_LOADCELL3_PD_SCK D0
// D3 can be used in parallel to the load cell with Wifi Kit 8
#define PIN_TARE_BUTTON D3
#define MAX_SSID_PW_LENGHT 64
#else
#define PIN_LOADCELL1_DOUT D6
#define PIN_LOADCELL1_PD_SCK D5
#define PIN_LOADCELL2_DOUT D2
#define PIN_LOADCELL2_PD_SCK D1
#define PIN_LOADCELL3_DOUT D7
#define PIN_LOADCELL3_PD_SCK D0
#endif
// **** Measurement settings ****
@ -81,9 +101,13 @@ CG scale with 3 Loadcells:
// Please UNCOMMENT the display used
U8G2_SH1106_128X64_NONAME_1_HW_I2C oledDisplay(U8G2_R0, /* reset=*/ U8X8_PIN_NONE, /* clock=*/ D3, /* data=*/ D4);
//U8G2_SSD1306_128X64_NONAME_1_HW_I2C oledDisplay(U8G2_R0, /* reset=*/ U8X8_PIN_NONE, /* clock=*/ D3, /* data=*/ D4);
#ifdef WIFI_KIT_8
// Wifi Kit 8 has a fixed wired 128x32 display
U8G2_SSD1306_128X32_UNIVISION_1_HW_I2C oledDisplay(U8G2_R0, /* reset=*/ 16, /* clock=*/ 5, /* data=*/ 4);
#else
U8G2_SH1106_128X64_NONAME_1_HW_I2C oledDisplay(U8G2_R0, /* reset=*/ U8X8_PIN_NONE, /* clock=*/ D3, /* data=*/ D4);
//U8G2_SSD1306_128X64_NONAME_1_HW_I2C oledDisplay(U8G2_R0, /* reset=*/ U8X8_PIN_NONE, /* clock=*/ D3, /* data=*/ D4);
#endif
// **** Voltage measurement settings ****
@ -129,7 +153,9 @@ U8G2_SH1106_128X64_NONAME_1_HW_I2C oledDisplay(U8G2_R0, /* reset=*/ U8X8_PIN_NON
// **** Wifi settings ****
#ifndef MAX_SSID_PW_LENGHT
#define MAX_SSID_PW_LENGHT 32
#endif
// Station mode: connect to available network
#define SSID_STA "myWiFi"