v2.11
new HW with integrated OLED (Wifi Kit 8) and some fixes
This commit is contained in:
commit
6cc51bb9ef
489
CG_scale.ino
489
CG_scale.ino
@ -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;
|
||||
}
|
||||
|
||||
|
@ -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
880
data/index.html
Executable 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
531
data/models.html
Executable 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
1062
data/settings.html
Executable file
File diff suppressed because it is too large
Load Diff
Binary file not shown.
@ -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"
|
||||
|
Loading…
x
Reference in New Issue
Block a user