forked from Pittini/iobroker-Batterienauswertung
-
Notifications
You must be signed in to change notification settings - Fork 0
/
batterienauswertung-V1.6.4.js
576 lines (528 loc) · 33.2 KB
/
batterienauswertung-V1.6.4.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
// Batterieüberwachungsskript Version 1.6.4 Stand 18.06.2020 - Git: https://github.com/Pittini/iobroker-Batterienauswertung - Forum: https://forum.iobroker.net/topic/31676/vorlage-generische-batteriestandsüberwachung-vis-ausgabe
//Überwacht Batteriespannungen beliebig vieler Geräte
//WICHTIG!!!
//Vorraussetzungen: Den Gerätechannels müssen Räume, sowie die Funktion "BatterieSpannung_xx" für jeden entsprechenden Batteriespannungs Datenpunkt zugewiesen sein.
//Bitte unbedingt Anleitung beachten
// Nach der Zuweisung unbedingt den JS Adpter neu starten!
//Grund Einstellungen
const praefix = "javascript.0.BatterieUeberwachung."; //Grundpfad für Script DPs
const logging = true; //Logging aktivieren?
const FunktionBaseName = "BatterieSpannung_"; //Name der Funktion welche für die Batterieüberwachung genutzt wird
const UseTelegram = false; // Sollen Nachrichten via Telegram gesendet werden?
const UseMail = false; // Sollen Nachrichten via Mail gesendet werden?
const UseAlexa = false; // Sollen Nachrichten via Alexa ausgegeben werden?
const AlexaId = ""; // Die Alexa Seriennummer
const UseSay = false; // Sollen Nachrichten via Say ausgegeben werden? Funktion des Authors, sollte bei Anwendern auf false gesetzt werden.
const UseEventLog = false; // Sollen Nachrichten ins Eventlog geschreiben werden? Funktion des Authors, sollte bei Anwendern auf false gesetzt werden.
const UsePopUp = false // Soll PopUp angezeigt werden? Funktion des Authors, sollte bei Anwendern auf false gesetzt werden.
const ProzMeansLive = true; //Zeigen Prozentwerte des Gerätedatenpunktes Batteriekapazität oder restliche Lebensdauer?
let DeadIsAfter = 180; // In Minuten - Zeit nach der ein Gerät als "tot" gewertet wird wenn keine Statusänderung (ts) erfolgte.
const NotifyDeadDevices = true; //Sollen auch "tote" Geräte gemeldet werden?
//Tabellen Einstellungen
const TblOkBgColor = "#4caf50"; //Hintergrundfarbe für Batteriestatus Ok
const TblInfoBgColor = "#ffc107"; //Hintergrundfarbe für Batteriestatus Info, also die leerste Batterie welche noch nicht das Limit unterschreitet
const TblWarnBgColor = "#f44336"; //Hintergrundfarbe für Batteriestatus Warnung, also jene Batterie welche unter das Limit kam.
const TblDeadBgColor = "grey"; //Hintergrundfarbe für Batterie/Geräte Status tot.
const HeadBgColor = "dimgrey"; //Hintergrundfarbe des Tabellenkopfes
const FontColor = "black"; //Textfarbe für Tabelleninhalt
const HeadFontColor = "white"; //Textfarbe für Tabellenkopf
const TblShowLfdCol = true; //Tabellenspalte mit laufender Nummer anzeigen?
const TblShowDeviceIDCol = true; //Tabellenspalte mit Geräte ID anzeigen?
const TblShowDeviceNameCol = true; //Tabellenspalte mit Gerätenamen anzeigen?
const TblShowRoomCol = true; //Tabellenspalte mit Raum anzeigen?
const TblShowUmaxCol = true; //Tabellenspalte mit Batterie Nennspannung anzeigen?
const TblShowUistCol = true; //Tabellenspalte mit aktueller Batteriespannung anzeigen?
const TblShowUlimitCol = true; //Tabellenspalte mit unterer Batterielimit Spannung anzeigen?
const TblShowProzbatCol = true; //Tabellenspalte mit Batteriestand in Prozent anzeigen?
const TblShowProzliveCol = true; //Tabellenspalte mit Restlebensdauer unter Berücksichtigung der Limitspannung in Prozent anzeigen? Beispiel: Batterie hat 3V Nennspannung, Limit ist bei 2V, aktueller Batteriestand ist 2.5V, dann wäre die Restlebensdauer 50%
const TblShowStatusCol = true; //Tabellenspalte mit Status ausgeben?
//Ab hier nix mehr ändern
let DpCount = 0; //Zähler für anzulegende Datenpunkte
const States = []; //States Array initialisieren
const Sensor = [] //Sensoren Array initialisieren
const SensorVal = []; //SensorDatenpunkte Werte Array initialisieren
const SensorUmax = []; //Sensoren Array Batteriespannungswerte Werte initialisieren
const SensorUProz = []; //Sensoren Array Spannung in Prozent initialisieren
const SensorLiveProz = []; //Sensoren Array verbleibende Lebendauer unter Berücksichtigung des min Limits, nicht zu verwechseln mit Batteriekapazität in %
const SensorState = []; //Statusarray, mögliche Werte ok,info,warn,dead
const BatteryMinLimit = []; // Das eingestellte Batterie min. limit
const BatteryMinLimitDp = []; //Array mit den generierten MinLimit Einstellungsdatenpunkten
const WelcheFunktionVerwenden = []; // Array mit allen Einträgen aus Funktionen welche den FunktionBaseName beinhalten
let AllBatterysOk = true;
let LastMessage = "";
let LastMessageSeparator = "<br>";
let NextExpectedLowBatt = "";
let EmptyBatCount = 0;
let DeadDeviceCount = 0;
//Datenpunkte anlegen in javascript.0.BatterieUeberwachung.
States[DpCount] = { id: praefix + "AllBatterysOk", initial: true, forceCreation: false, common: { read: true, write: false, name: "Alle Batterien Ok?", type: "boolean", role: "state", def: false } }; //
DpCount++;
States[DpCount] = { id: praefix + "LastMessage", initial: "", forceCreation: false, common: { read: true, write: false, name: "Letzte Warnmeldung", type: "string", role: "state", def: "" } }; //
DpCount++;
FillWelcheFunktionVerwenden(); //Vorab Funktionen mit Umax Spannungen einlesen da diese für ID und Namen der MinLimit States benötigt werden
for (let x = 0; x < WelcheFunktionVerwenden.length; x++) {
let dummy = WelcheFunktionVerwenden[x].slice(FunktionBaseName.length) //Letzten Zeichen aus Funktionsnamen extrahieren
let VoltInitial = CreateUmaxValueFromString(x) //Extrahierte Zeichen zu Kommazahl wandeln
VoltInitial = VoltInitial / 100 * 80; //Initialwert für Limit berechnen
if (logging) log("InitialSpannung " + x + " gesetzt auf 80%= " + VoltInitial);
States[DpCount] = { id: praefix + "BatteryMinLimit_" + dummy, initial: VoltInitial, forceCreation: false, common: { read: true, write: true, name: "Unteres Limit für Warnmeldung bei " + toFloat(dummy.slice(0, dummy.length - 1) + "." + dummy.slice(-1)) + "V Geräten", type: "number", role: "value", unit: "V", def: 2.6 } }; //
BatteryMinLimitDp[x] = "BatteryMinLimit_" + dummy;
DpCount++;
};
States[DpCount] = { id: praefix + "NextExpectedLowBatt", initial: "", forceCreation: false, common: { read: true, write: false, name: "Vorraussichtlich nächste zu wechselnde Batterie", type: "string", role: "state", def: "" } }; //
DpCount++;
States[DpCount] = { id: praefix + "OverviewTable", initial: "", forceCreation: false, common: { read: true, write: false, name: "Einfache HTML Übersichtstabelle", type: "string", role: "state", def: "" } }; //
DpCount++;
States[DpCount] = { id: praefix + "EmptyBatCount", initial: 0, forceCreation: false, common: { read: true, write: false, name: "Zähler für Anzahl der zu wechselnden Batterien", type: "number", role: "state", def: 0 } }; //
DpCount++;
States[DpCount] = { id: praefix + "DeadDeviceCount", initial: 0, forceCreation: false, common: { read: true, write: false, name: "Zähler für Anzahl der nicht mehr aktualisierenden Geräte", type: "number", role: "state", def: 0 } }; //
//Alle States anlegen, Main aufrufen wenn fertig
let numStates = States.length;
States.forEach(function (state) {
createState(state.id, state.initial, state.forceCreation, state.common, function () {
numStates--;
if (numStates === 0) {
if (logging) log("CreateStates fertig!");
main();
};
});
});
function CreateUmaxValueFromString(x) {
let dummy = WelcheFunktionVerwenden[x].slice(FunktionBaseName.length) //Aus der Funktionsbezeichnung die letzten Zeichen extrahieren= z.B. 33
return toFloat(dummy.slice(0, dummy.length - 1) + "." + dummy.slice(-1)) //Die extrahierten Zeichen zu einer Kommazahl wandeln= z.B. 3.3
}
function Init() {
if (logging) log("Reaching init()");
let counter = 0; //Zähler für Devices
let TempVal // Temporärer Sensorwert um nicht mehrere GetStates zu benötigen
let TempUnit //Einheit für Unterscheidung ob % vorliegen
let TempLiveWindow //Spannungsfensterbereich
let Funktionen = getEnums('functions'); //Alle Funktionen der Aufzählung in Array Funktionen übertragen
for (let x in Funktionen) { // loop ueber alle Funktionen
let Funktion = Funktionen[x].name; // Einzelne Funktion aus dem Array
if (typeof Funktion == 'object') Funktion = Funktion.de; //Wenn Rückgabewert ein Objekt ist, ist die Funktion mehrsprachig und es wird die deutsche Bezeichnug verwendet
let members = Funktionen[x].members; //Array mit allen Mitgliedern der Funktion erzeugen
for (let z = 0; z < WelcheFunktionVerwenden.length; z++) { //Loop über alle Funktions welche zu WelcheFunktionVerwenden passen
if (Funktion == WelcheFunktionVerwenden[z]) { //Wenn Function ist WelcheFunktionVerwenden (BatterieSpannung)
let Umax = CreateUmaxValueFromString(z) //Batteriesollspannung aus der Funktionsbezeichnung extrahieren
let BattMinLimitTemp = getState(praefix + "BatteryMinLimit_" + WelcheFunktionVerwenden[z].slice(FunktionBaseName.length)).val; //Temporäres (für den jeweiligen Schleifendurchlauf) MinLimit einlesen
if (typeof (BattMinLimitTemp == "string")) { //Falls MinLimit Wert String ist zu float wandeln
//log("BattMinLimit Value is String, trying to convert");
BattMinLimitTemp = parseFloat(BattMinLimitTemp);
if (typeof (BattMinLimitTemp == "number")) {
if (logging) log("BattMinLimit Value conversion - success");
};
};
let TempLiveWindow = Umax - BattMinLimitTemp;
for (let y in members) { // Loop über alle WelcheFunktionVerwenden Members
Sensor[counter] = members[y]; //Treffer in SenorIDarray einlesen
TempVal = getState(Sensor[counter]).val;//Wert vom Sensor in Tempval einlesen um wiederholte Getstates zu vermeiden
//if (typeof (TempVal) == undefined || typeof (TempVal) == null || TempVal == "") TempVal = 0; //Bei leeren Feldern 0 setzen um Fehler zu vermeiden
TempUnit = GetUnit(counter);
if (logging) log("Tempval=" + TempVal + " TempUnit=" + TempUnit + " TypeOf=" + typeof (TempVal))
switch (typeof (TempVal)) { //Wenn der Sensorwert bool ist (wenn nur LowBatt mit true/false vom Sensor gemeldet wird)
case "boolean": //Sensorval ist Bool
if (TempVal) { //Bei Lowbat=true
SensorVal[counter] = 0; //Batt wird als leer definiert und auf 0 gesetzt
SensorUProz[counter] = 0; //Prozentwerte aus Umax und Sensorwert errechnen
SensorLiveProz[counter] = 0; //Lebensprozent auf 0%
}
else {
SensorVal[counter] = Umax; //Batt wird als voll definiert und auf Umax gesetzt
SensorUProz[counter] = 100; //Prozentwerte aus Umax und Sensorwert errechnen
SensorLiveProz[counter] = 100; //Lebensprozent auf 100%
};
break;
case "number": //Sensorval ist Zahl
switch (TempUnit) { //Bei Zahlen nach Einheit unterscheiden um % Angaben mit zu verarbeiten
case "%": //Bei Datenpunkt Unit = %
//if (logging) log("unit= " + TempUnit + " should be %");
if (ProzMeansLive) { // Wenn die Prozentangabe bereits Lebensdauer zeigt (Einstellungsoption)
SensorLiveProz[counter] = TempVal; //Direkt zuweisen aus Sensorwert
//SensorUProz[counter] = (Umax - (TempLiveWindow / 100 * SensorLiveProz[counter])) / Umax * 100 //Errechne Batteriekapazität
//SensorVal[counter] = Umax / 100 * SensorUProz[counter]; //Errechne Spannung
SensorVal[counter] = Umax - TempLiveWindow + (TempLiveWindow / 100 * SensorLiveProz[counter]);
SensorUProz[counter] = (SensorVal[counter] / Umax) * 100 //Errechne Batteriekapazität
}
else if (!ProzMeansLive) { //Wenn die Prozentangabe Batteriekapazität darstellt (Einstellungsoption)
SensorUProz[counter] = TempVal; //Batteriekapazität in % bestimmen
SensorVal[counter] = Umax / 100 * SensorUProz[counter]; //Sensorwert aus Umax und Prozentwert bestimmen
SensorLiveProz[counter] = (SensorVal[counter] - BattMinLimitTemp) / (Umax - BattMinLimitTemp) * 100; //Restlebensdauer in % ermitteln
};
break;
default: // In allen anderen Fällen
SensorVal[counter] = TempVal; //Spannung ist Wert vom DP
SensorUProz[counter] = SensorVal[counter] / Umax * 100; //Prozentwerte aus Umax und Sensorwert errechnen
SensorLiveProz[counter] = (SensorVal[counter] - BattMinLimitTemp) / (Umax - BattMinLimitTemp) * 100; //Restlebensdauer in % ermitteln
};
break;
case "string": //Sensorval ist Text
if (TempVal == "ok") {
SensorVal[counter] = Umax; //Batt wird als voll definiert und auf Umax gesetzt
SensorUProz[counter] = 100; //Prozentwerte aus Umax und Sensorwert errechnen
SensorLiveProz[counter] = 100; //Lebensprozent auf 100%
}
else if (TempVal != "ok") { //Bei BatteryState != ok
SensorVal[counter] = 0; //Batt wird als leer definiert und 0.1 unter MinLimit gesetzt
SensorUProz[counter] = 0; //Prozentwerte aus Umax und Sensorwert errechnen
SensorLiveProz[counter] = 0; //Lebensprozent auf 0%
};
break;
default:
};
if (SensorLiveProz[counter] > 100) SensorLiveProz[counter] = 100; //Um bei übervollen Batterien mehr als 100% live zu vermeiden
SensorUmax[counter] = Umax; //Synchrones UmaxArray füllen
BatteryMinLimit[counter] = BattMinLimitTemp; //Limitarray füllen
if (logging) log(counter + " " + Funktion + ' found at ' + members[y] + " Umax= " + SensorUmax[counter] + " BattMinLimit=" + BattMinLimitTemp + " Val= " + SensorVal[counter] + " SensorProzent= " + SensorUProz[counter]);
counter++
};
};
};
};
}
function FillWelcheFunktionVerwenden() {
if (logging) log("Reaching FillWelcheFunktionVerwenden");
let z = 0;
let Funktionen = getEnums('functions'); //Alle Funktionen der Aufzählung in Array Funktionen übertragen
//let NumFunktionen = Funktionen.length
for (let x in Funktionen) { // loop ueber alle Funktionen
let Funktion = Funktionen[x].name; // Einzelne Funktion aus dem Array
if (typeof Funktion == 'object') Funktion = Funktion.de; //Wenn Rückgabewert ein Objekt ist, ist die Funktion mehrsprachig und es wird die deutsche Bezeichnung verwendet
let members = Funktionen[x].members; //Array mit allen Mitgliedern der Funktion erzeugen
if (Funktion.includes(FunktionBaseName)) {
WelcheFunktionVerwenden[z] = Funktion;
if (logging) log("Found Function " + WelcheFunktionVerwenden[z])
z++
};
};
}
function main() {
if (logging) log("Reaching main()");
Init(); //Alle Werte einlesen, Arrays füllen, fehlende Werte errechnen
CreateTrigger(); // Trigger erzeugen
CheckDeadBatt(); //Auf nicht mehr aktualisierende Geräte seit Zeit x (Einstellung) prüfen
CheckAllBatterys(); // Alle Batteriestände prüfen
CheckAllBatterysOk();
CheckNextLowBatt(); // Batterie mit niedrigster Spannung finden
MakeTable(); //HTML Tabelle erzeugen
Ticker(); //Startet Intervallprüfung für nicht aktualisierende Geräte
}
function Meldung(msg) {
if (logging) log("Reaching Meldung()");
if (UseSay) Say(msg);
if (UseTelegram) {
sendTo("telegram.0", "send", {
text: msg
});
};
if (UseMail) {
sendTo("email", {
html: msg
});
};
if (UseAlexa) {
if (AlexaId != "") setState("alexa2.0.Echo-Devices." + AlexaId + ".Commands.announcement"/*announcement*/, msg);
};
if (logging) log(msg);
if (UseEventLog) WriteEventLog(msg);
if (UsePopUp) ShowPopUp(true, msg, "Batterys", "red");
setState(praefix + "LastMessage", LastMessage); //Meldung in Datenpunkt LastMessage schreiben
}
function CheckDeadBatt() {
if (logging) log("Reaching CheckDeadBatt()");
let jetzt = new Date().getTime();
DeadDeviceCount = 0;
for (let x = 0; x < Sensor.length; x++) { //Alle Sensoren durchlaufen
if ((getState(Sensor[x]).ts + (DeadIsAfter * 60 * 1000)) < jetzt) { //Wenn letzte Aktualisierung + Karrenzzeit kleiner aktuelle Zeit
if (SensorState[x] != "dead") { //Wenn Sensor bei letzter Prüfung noch nicht tot.
if (NotifyDeadDevices) {
Meldung("Ausfall oder disconnect im " + GetRoom(x) + " bei Gerät " + getObject(GetParentId(Sensor[x])).common.name);
};
};
SensorState[x] = "dead"; //Status auf tot setzen
DeadDeviceCount++; //Zähler ehöhen
}
else if (SensorState[x] == "dead") { //Wenn Sensor als tot gelistet, aber wieder aktualisiert, Status prüfen
CheckBatterys(x);
};
};
setState(praefix + "DeadDeviceCount", DeadDeviceCount);
MakeTable();
}
function Ticker() {
setInterval(function () { // Wenn
CheckDeadBatt();
}, 60000);
}
function CheckNextLowBatt() { //Ermittelt die Batterie mit der geringsten Spannung, ignoriert Batterien welche das Limit bereits unterschritten haben da diese bereits in der LastMessage gemeldet werden
if (logging) log("Reaching CheckNextLowBatt()");
let LowestBattProz = 100; //Mit 100% initialisieren
let LowestBattIndex = 0;
for (let x = 0; x < Sensor.length; x++) { //Alle Sensoren durchlaufen
if (SensorState[x] != "warn" && SensorState[x] != "dead") SensorState[x] = "ok";
if (SensorVal[x] > BatteryMinLimit[x]) { // Nur Sensoren berücksichtigen die das min Limit noch nicht unterschritten haben
if (SensorLiveProz[x] < LowestBattProz) { //Wenn Sensorwert kleiner LowestBattProz, LowestBattVal auf neuen Wert setzen um das Gerät mit den wenigsten Prozent zu ermitteln
LowestBattProz = SensorLiveProz[x];
LowestBattIndex = x;
};
};
};
NextExpectedLowBatt = "Aktuell niedrigster Batteriestand (" + SensorVal[LowestBattIndex].toFixed(2) + "V): " + GetRoom(LowestBattIndex) + " bei Gerät " + getObject(GetParentId(Sensor[LowestBattIndex]), "common").common.name;
setState(praefix + "NextExpectedLowBatt", NextExpectedLowBatt);
SensorState[LowestBattIndex] = "info";
if (logging) log(NextExpectedLowBatt);
}
function CheckAllBatterysOk() {
if (logging) log("Reaching CheckAllBatterysOk - Lastmessage=" + LastMessage);
AllBatterysOk = true;
EmptyBatCount = 0;
for (let x = 0; x < Sensor.length; x++) { //Alle Sensoren durchlaufen
if (SensorVal[x] < BatteryMinLimit[x]) { // Nur Sensoren berücksichtigen die das min Limit unterschritten haben
//log(Sensor[x]);
AllBatterysOk = false;
EmptyBatCount++; //Alle Sensoren zählen welche das Batt min Limit unterschritten haben
};
};
if (DeadDeviceCount > 0) AllBatterysOk = false;
setState(praefix + "EmptyBatCount", EmptyBatCount);
setState(praefix + "AllBatterysOk", AllBatterysOk);
if (EmptyBatCount == 0 && DeadDeviceCount == 0) {
LastMessage = ""; //Lastmessage löschen
setState(praefix + "LastMessage", LastMessage); //Meldung in Datenpunkt LastMessage löschen
if (logging) log("Alle Batterien ok, Lastmessage gelöscht");
};
}
function CheckBatterys(x) { // Prüfung eines einzelnen Batteriestandes wenn getriggert
if (logging) log("Reaching CheckBatterys(" + x + ") Val=" + SensorVal[x] + " Limit=" + BatteryMinLimit[x]);
if (SensorVal[x] < BatteryMinLimit[x]) { //Wenn Min. Wert unterschritten
//LastMessage = "Batteriestand unter Limit im " + GetRoom(x) + " bei Gerät " + getObject(Sensor[x].substring(0, Sensor[x].lastIndexOf("."))).common.name;
LastMessage = "Batteriestand unter Limit im " + GetRoom(x) + " bei Gerät " + getObject(GetParentId(Sensor[x])).common.name;
Meldung(LastMessage);
SensorState[x] = "warn";
}
else {
SensorState[x] = "ok";
};
CheckAllBatterysOk();
CheckNextLowBatt();
CheckDeadBatt();
MakeTable();
}
function CheckAllBatterys() { // Prüfung aller Batteriestände bei Skriptstart
if (logging) log("Reaching CheckAllBatterys() found " + (Sensor.length) + " Devices");
for (let x = 0; x < Sensor.length; x++) { //Alle Sensoren durchlaufen
if (SensorState[x] == "dead") {
if (logging) log("Sensor[" + x + "] = ist ausgefallen oder disconnected");
LastMessage += "Ausfall oder disconnect im " + GetRoom(x) + " bei Gerät " + getObject(GetParentId(Sensor[x])).common.name + LastMessageSeparator;
}
else if (SensorVal[x] < BatteryMinLimit[x]) { //Wenn Min. Wert unterschritten
if (logging) log("SensorVal[" + x + "] = " + SensorVal[x] + "V, unterschreitet MinLimit von " + BatteryMinLimit[x] + " V");
LastMessage += "Batteriestand unter Limit im " + GetRoom(x) + " bei Gerät " + getObject(GetParentId(Sensor[x])).common.name + LastMessageSeparator;
SensorState[x] = "warn";
}
else {
if (SensorState[x] != "info" && SensorState[x] != "dead") SensorState[x] = "ok";
};
};
//CheckAllBatterysOk();
//log("Lastmessage=" + LastMessage);
LastMessage = LastMessage.substr(0, LastMessage.length - LastMessageSeparator.length); //letzten <br> Umbruch wieder entfernen
if (LastMessage != "") Meldung(LastMessage); // Wenn Lastmessage nicht leer, Nachricht ausgeben
}
function GetRoom(x) { // Raum eines Gerätes ermitteln
let room = getObject(Sensor[x], 'rooms').enumNames[0];
if (room == undefined) room = "Nicht zugewiesen";
if (typeof room == 'object') room = room.de;
room = room.replace(/_/g, " "); //Unterstriche durch Leerzeichen ersetzen
return room;
}
function GetUnit(x) {
let unit = getObject(Sensor[x], 'common').common.unit
return unit;
}
function GetParentId(Id) {
let parentDevicelId;
if (Id.indexOf("hm-rpc.") == -1 && Id.indexOf("shelly.") == -1) { //Wenn kein HM und kein shelly Adapter, eine Ebene zurück
parentDevicelId = Id.split(".").slice(0, -1).join(".");// Id an den Punkten in Array schreiben (split), das letzte Element von hinten entfernen (slice) und den Rest wieder zu String zusammensetzen
}
else { //Wenn HM dann zwei Ebenen zurück
parentDevicelId = Id.split(".").slice(0, -2).join(".");// Id an den Punkten in Array schreiben (split), die 2 letzten Elemente von hinten entfernen (slice) und den Rest wieder zu String zusammensetzen
};
//if (logging) log("Id= " + Id + " ParentDeviceId= " + parentDevicelId)
return parentDevicelId
}
function MakeTable() {
if (logging) log("Reaching MakeTable");
let BgColor = "";
let style0 = "style='border: 1px solid black; padding-left: 5px; padding-right: 5px; font-size:0.8em; font-weight: normal; text-align: left; color:" + FontColor + "; background-color:"
let style1 = "style='width: 40px; border: 1px solid black; padding-left: 5px; padding-right: 5px; font-size:0.8em; font-weight: normal; text-align: right; color:" + FontColor + "; background-color:"
let headstyle0 = "style='border: 1px solid black; padding-left: 5px; padding-right: 5px; height: 30px; font-size:1.0em; font-weight: bold; text-align: left; color:" + HeadFontColor + "; background-color:"
let headstyle1 = "style='width: 40px; border: 1px solid black; padding-left: 5px; padding-right: 5px; height: 30px; font-size:1.0em; font-weight: bold; text-align: center; color:" + HeadFontColor + "; background-color:"
let MyTableHead = "<table style='width:100%; border: 1px solid black; border-collapse: collapse;'><tr>";
let MyTable;
if (TblShowLfdCol) {
MyTableHead = MyTableHead + "<th " + headstyle0 + HeadBgColor + "'>lfd</th>";
};
if (TblShowDeviceIDCol) {
MyTableHead = MyTableHead + "<th " + headstyle0 + HeadBgColor + "'>Sensor ID</th>";
};
if (TblShowDeviceNameCol) {
MyTableHead = MyTableHead + "<th " + headstyle0 + HeadBgColor + "'>Sensor Name</th>";
};
if (TblShowRoomCol) {
MyTableHead = MyTableHead + "<th " + headstyle0 + HeadBgColor + "'>Raum</th>";
};
if (TblShowUmaxCol) {
MyTableHead = MyTableHead + "<th " + headstyle1 + HeadBgColor + "'>U<br>Nenn</th>";
};
if (TblShowUistCol) {
MyTableHead = MyTableHead + "<th " + headstyle1 + HeadBgColor + "'>U<br>Ist</th>";
};
if (TblShowUlimitCol) {
MyTableHead = MyTableHead + "<th " + headstyle1 + HeadBgColor + "'>U<br>Limit</th>";
};
if (TblShowProzbatCol) {
MyTableHead = MyTableHead + "<th " + headstyle1 + HeadBgColor + "'>%bat</th>";
};
if (TblShowProzliveCol) {
MyTableHead = MyTableHead + "<th " + headstyle1 + HeadBgColor + "'>%live</th>";
};
if (TblShowStatusCol) {
MyTableHead = MyTableHead + "<th " + headstyle1 + HeadBgColor + "'>Status</th>";
};
MyTableHead = MyTableHead + "</tr>";
MyTable = MyTableHead + "<tr>";
for (let x = 0; x < Sensor.length; x++) { //Alle Sensoren durchlaufen
switch (SensorState[x]) {
case "ok":
BgColor = TblOkBgColor;
break;
case "info":
BgColor = TblInfoBgColor;
break;
case "warn":
BgColor = TblWarnBgColor;
break;
case "dead":
BgColor = TblDeadBgColor;
break;
default:
};
MyTable = MyTable + "<tr>"
if (TblShowLfdCol) {
MyTable = MyTable + "<td " + style0 + BgColor + "'>" + (x + 1) + "</td>";
};
if (TblShowDeviceIDCol) {
MyTable = MyTable + "<td " + style0 + BgColor + "'>" + GetParentId(Sensor[x]) + "</td>";
};
if (TblShowDeviceNameCol) {
MyTable = MyTable + "<td " + style0 + BgColor + "'>" + getObject(GetParentId(Sensor[x]), "common").common.name + "</td>";
};
if (TblShowRoomCol) {
MyTable = MyTable + "<td " + style0 + BgColor + "'>" + GetRoom(x) + "</td>";
};
if (TblShowUmaxCol) {
MyTable = MyTable + "<td " + style1 + BgColor + "'>" + SensorUmax[x].toFixed(1) + "V</td>";
};
if (TblShowUistCol) {
MyTable = MyTable + "<td " + style1 + BgColor + "'>" + SensorVal[x].toFixed(2) + "V</td>";
};
if (TblShowUlimitCol) {
//log(BatteryMinLimit[x]+" hat type="+typeof(BatteryMinLimit[x]))
MyTable = MyTable + "<td " + style1 + BgColor + "'>" + BatteryMinLimit[x].toFixed(2) + "V</td>";
};
if (TblShowProzbatCol) {
if (typeof (SensorUProz[x]) == "number") {
MyTable = MyTable + "<td " + style1 + BgColor + "'>" + SensorUProz[x].toFixed(1) + "%</td>";
}
else {
MyTable = MyTable + "<td " + style1 + BgColor + "'>" + SensorUProz[x] + "</td>";
};
};
if (TblShowProzliveCol) {
if (typeof (SensorLiveProz[x]) == "number") {
MyTable = MyTable + "<td " + style1 + BgColor + "'>" + SensorLiveProz[x].toFixed(1) + "%</td>";
}
else {
MyTable = MyTable + "<td " + style1 + BgColor + "'>" + SensorLiveProz[x] + "</td>";
};
};
if (TblShowStatusCol) {
MyTable = MyTable + "<td " + style1 + BgColor + "'>" + SensorState[x] + "</td>";
};
MyTable = MyTable + "</tr>";
};
MyTable = MyTable + "</table>";
setState(praefix + "OverviewTable", MyTable);
//if (logging) log(MyTable);
}
//Trigger für Sensoren erzeugen
function CreateTrigger() {
for (let x = 0; x < Sensor.length; x++) { //Alle Sensoren durchlaufen
on(Sensor[x], function (dp) { //Trigger in Schleife erstellen
let TempVal = dp.state.val;
let TempUnit = GetUnit(x);
let TempLiveWindow = SensorUmax[x] - BatteryMinLimit[x]
switch (typeof (TempVal)) { //Wenn der Sensorwert bool ist (wenn nur LowBatt mit true/false vom Sensor gemeldet wird)
case "boolean": //Sensorval ist Bool
if (TempVal) { //Bei Lowbat=true
SensorVal[x] = 0; //Batt wird als leer definiert und 0.1 unter MinLimit gesetzt
SensorUProz[x] = 0; //Prozentwerte aus Umax und Sensorwert errechnen;
SensorLiveProz[x] = 0; //Lebensprozent auf 0%
}
else {
SensorVal[x] = SensorUmax[x]; //Batt wird als voll definiert und auf Umax gesetzt
SensorUProz[x] = 100; //Prozentwerte aus Umax und Sensorwert errechnen
SensorLiveProz[x] = 100; //Lebensprozent auf 100%
};
break;
case "number": //Sensorval ist Zahl
switch (TempUnit) { //Bei Zahlen nach Einheit unterscheiden um % Angaben mit zu verarbeiten
case "%": //Bei Datenpunkt Unit = %
//if (logging) log("unit= " + TempUnit + " should be %");
if (ProzMeansLive) { // Wenn die Prozentangabe bereits Lebensdauer zeigt (Einstellungsoption)
SensorLiveProz[x] = TempVal; //Direkt zuweisen aus Sensorwert
//SensorUProz[x] = (SensorUmax[x] - ((SensorUmax[x] - BatteryMinLimit[x]) / 100 * SensorLiveProz[x])) / SensorUmax[x] * 100 //Errechne Batteriekapazität
//SensorVal[x] = SensorUmax[x] / 100 * SensorUProz[x]; //Errechne Spannung
SensorVal[x] = SensorUmax[x] - TempLiveWindow + (TempLiveWindow / 100 * SensorLiveProz[x]); //Errechne Spannung
SensorUProz[x] = (SensorVal[x] / SensorUmax[x]) * 100 //Errechne Batteriekapazität
}
else { //Wenn die Prozentangabe Batteriekapazität darstellt (Einstellungsoption)
SensorUProz[x] = TempVal; //Batteriekapazität in % bestimmen
SensorVal[x] = SensorUmax[x] / 100 * SensorUProz[x]; //Sensorwert aus Umax und Prozentwert bestimmen
SensorLiveProz[x] = (SensorVal[x] - BatteryMinLimit[x]) / TempLiveWindow * 100; //Restlebensdauer in % ermitteln
};
break;
default: // In allen anderen Fällen
SensorVal[x] = TempVal; //Spannung ist Wert vom DP
SensorUProz[x] = SensorVal[x] / SensorUmax[x] * 100; //Prozentwerte aus Umax und Sensorwert errechnen
SensorLiveProz[x] = (SensorVal[x] - BatteryMinLimit[x]) / TempLiveWindow * 100; //Restlebensdauer in % ermitteln
};
break;
case "string": //Sensorval ist Text
if (TempVal == "ok") {
SensorVal[x] = SensorUmax[x]; //Batt wird als voll definiert und auf Umax gesetzt
SensorUProz[x] = 100; //Prozentwerte aus Umax und Sensorwert errechnen
SensorLiveProz[x] = 100; //Lebensprozent auf 100%
}
else if (TempVal != "ok") { //Bei BatteryState != ok
SensorVal[x] = 0; //Batt wird als leer definiert und 0.1 unter MinLimit gesetzt
SensorUProz[x] = 0; //Prozentwerte aus Umax und Sensorwert errechnen
SensorLiveProz[x] = 0; //Lebensprozent auf 0%
};
break;
default:
};
CheckBatterys(x); //Prüfen
});
};
for (let x = 0; x < WelcheFunktionVerwenden.length; x++) { //Alle Batteriefunktionen durchlaufen
on(praefix + BatteryMinLimitDp[x], function (dp) { //Trigger erstellen und auslösen wenn min Limit geändert wurde. Dann erneute Komplettprüfung aller Batteriestände
if (logging) log("Reaching Trigger for :" + praefix + BatteryMinLimitDp[x])
if (typeof (dp.state.val) != "number") {
log("MinLimit Wert keine Nummer, sondern " + typeof (dp.state.val), "warn");
};
main(); //Neuzuweisung des geänderten Limits an alle Geräte
});
};
}