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
This commit is contained in:
Rainer Stransky 2020-07-18 12:31:16 +02:00
parent 1ce3c18ada
commit 3cf22bb65a
9 changed files with 2759 additions and 135 deletions

View File

@ -4,11 +4,23 @@
(c) 2019 by M. Lehmann
------------------------------------------------------------------
*/
#define CGSCALE_VERSION "2.0"
#define CGSCALE_VERSION "2.1"
/*
******************************************************************
history:
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.0 26.01.20 Webpage rewritten, no bootstrap framework needed
add translation to webpage (en, de)
optimized for measuring with landinggears
@ -155,6 +167,206 @@ void(* resetCPU) (void) = 0;
void resetCPU() {}
#endif
void initOLED() {
oledDisplay.begin();
const uint8_t *font;
font = u8g2_font_6x12_tr;
int displayHeight = oledDisplay.getDisplayHeight();
int displayWidth = oledDisplay.getDisplayWidth();
if (displayHeight <= 32) {
font = u8g2_font_6x12_tr;
}
int ylineHeight = displayHeight/3;
oledDisplay.firstPage();
do {
if (displayHeight <= 32) {
oledDisplay.drawXBMP(5, 0, 18, 18, CGImage);
} else {
oledDisplay.drawXBMP(20, 12, 18, 18, CGImage);
}
oledDisplay.setFont(u8g2_font_helvR12_tr);
if (displayHeight <= 32) {
oledDisplay.setCursor(30, 12);
} else {
oledDisplay.setCursor(45, 28);
}
oledDisplay.print(F("CG scale"));
oledDisplay.setFont(u8g2_font_5x7_tr);
if (displayHeight <= 32) {
oledDisplay.setCursor(30, 22);
} else {
oledDisplay.setCursor(35, 55);
}
oledDisplay.print(F("Version: "));
oledDisplay.print(CGSCALE_VERSION);
if (displayHeight <= 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) {
const uint8_t *font;
font = u8g2_font_6x12_tr;
int displayHeight = oledDisplay.getDisplayHeight();
int displayWidth = oledDisplay.getDisplayWidth();
font = u8g2_font_helvR10_tr;
if (displayHeight <= 32) {
font = u8g2_font_6x12_tr;
}
int ylineHeight = displayHeight/3;
oledDisplay.firstPage();
do {
oledDisplay.setFont(u8g2_font_6x12_tr);
oledDisplay.setCursor(0, ylineHeight*1);
oledDisplay.print(aLine1);
oledDisplay.setCursor(0, ylineHeight*2);
oledDisplay.print(aLine2);
if (aLine3 == "") {
oledDisplay.drawLine(0, ylineHeight*2 + 2, displayWidth, ylineHeight*2+2);
oledDisplay.setFont(u8g2_font_4x6_tr);
oledDisplay.setCursor(0, displayHeight);
String signature = "CG scale: V" + String(CGSCALE_VERSION);
oledDisplay.setCursor(displayWidth - oledDisplay.getStrWidth(signature.c_str()), displayHeight);
oledDisplay.print(signature);
} else {
oledDisplay.setCursor(0, displayHeight);
oledDisplay.print(aLine3);
}
} while ( oledDisplay.nextPage() );
}
void printScaleOLED() {
// 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;
}
}
const uint8_t *font;
int linestart = 14;
int linedist = 25;
int col0=0;
int col1=28;
font = u8g2_font_helvR12_tr;
if (oledDisplay.getDisplayHeight() <=32) {
font = u8g2_font_6x12_tr;
linestart = 8;
linedist = 12;
col0=0;
col1=28;
}
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(u8g2_font_5x7_tr);
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(font);
if (oledDisplay.getDisplayHeight() <= 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);
}
dtostrf(weightTotal, 5, 1, buff);
oledDisplay.print(buff);
oledDisplay.print(F("g"));
// print CG longitudinal axis
if (oledDisplay.getDisplayHeight() <=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);
}
dtostrf(CG_length, 5, 1, buff);
oledDisplay.print(buff);
oledDisplay.print(F("mm"));
// print CG transverse axis
if (nLoadcells == 3) {
if (oledDisplay.getDisplayHeight() <=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, 5, 1, buff);
}
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() );
}
#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) {
@ -278,8 +490,9 @@ 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));
@ -367,23 +580,9 @@ void setup() {
// init OLED display
#if defined(ESP8266)
printConsole(T_BOOT, "init OLED display");
printConsole(T_BOOT, "init OLED display: " + String(oledDisplay.getDisplayWidth())+ String("x") + String(oledDisplay.getDisplayHeight()));
#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++) {
@ -408,6 +607,7 @@ void setup() {
#if defined(ESP8266)
// Start by connecting to a WiFi network
WiFi.persistent(false);
WiFi.mode(WIFI_STA);
WiFi.begin(ssid_STA, password_STA);
@ -417,18 +617,20 @@ void setup() {
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;
@ -457,34 +659,16 @@ void setup() {
}
#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
@ -589,6 +773,10 @@ void loop() {
server.handleClient();
#endif
#ifdef PIN_TARE_BUTTON
handleTareBtn();
#endif
updateLoadcells();
// update loadcell values
@ -606,6 +794,7 @@ void loop() {
}
}
// update display and serial menu
if ((millis() - lastTimeMenu) > UPDATE_INTERVAL_OLED_MENU) {
@ -642,80 +831,10 @@ 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;
}
}
printScaleOLED();
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() );
// serial connection
if (Serial) {
@ -838,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"));
@ -1111,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);
}
@ -1137,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);
}

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,33 @@ 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
#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 +99,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 +151,7 @@ U8G2_SH1106_128X64_NONAME_1_HW_I2C oledDisplay(U8G2_R0, /* reset=*/ U8X8_PIN_NON
// **** Wifi settings ****
#define MAX_SSID_PW_LENGHT 32
#define MAX_SSID_PW_LENGHT 64
// Station mode: connect to available network
#define SSID_STA "myWiFi"