NEWS
Bitte Programmierhilfe
-
@paul53 sagte in Bitte Programmierhilfe:
Nur noch am Rande, weil es im Text anders steht, als es programmiert ist. Ich beziehe mich auf die ursprünglichen Quelltexte!
Im Text steht "meine Klimaanlage soll eingeschaltet werden, wenn ein Grenzwert (hier 23 Grad) überschritten wird und abgeschaltet werden, wenn der untere Grenzwert (22 Grad) unterschritten wird."// Sollwert 22 Wohnzimmer ................ const Sollwert = 22; //Erlaubten Schwankungsbereich der Temperatur setzen const maxSoll = Sollwert + 1; const minSoll = Sollwert - 1;
Daraus ergibt sich der maxSoll mit 23 und minSoll mit 21!
Und über
if (AktTemperatur > maxSoll) setState(klimEin, true); else if (AktTemperatur < minSoll) setState(klimAus, true);
Hier wird > maxSoll abgefragt, was ja passt, weil eingeschaltet werden soll, wenn der Ist-Wert > 23° ist. Da ist es nur davon abhängig, was als nächste Stufe kommt. Wenn der Istwert ganzzahlig ist, wird bei 24° eingeschaltet.
Aber < minSoll ist dann < 21°. Wenn es also ganzzahlige Werte sind würde bei erreichen von 20° abgeschaltet. Selbst wenn es FP-Werte sind und in 0,1-Schritten ausgegeben werden, würde noch >1° Differenz zur Anforderung bestehen!Ich weiß, das ist für die Viele nur Erbsenzählerei.
-
@skorpil sagte: Ich hätte gedacht, daß ich die beiden "on Anweisungen" ineinander verschachteln muß, also 1. erst prüfen, ob die Automatic eingeschaltet ist und dann 2. auf Änderungen der Temperatur reagieren.
Man darf keine Trigger ineinander verschachteln!
@skorpil sagte in Bitte Programmierhilfe:
Wenn Automaticmodus AUS, dann keine Reaktion auf Temp Änderung.
In Zeile 21 wird der Automatikmodus geprüft.
-
Vielen Dank für die Erklärung und die Hinweise! Ihr seid klasse Java Script Lehrer. Habt ein schönes Wochenende
-
@paul53 ich komme noch einmal zurück auf dieses Thema. Ich möchte noch eine Prüfung einbauen, ob die Klimaanlage wirklich den Infrarotbefehl empfangen hat (unidirektional) und ob die Klimaanlage wirklich ein- resp. ausgeschaltet wurde. Dazu würde ich den Zwischenstecker nutzen, der ja die Powerwerte zeigt:
const Stromwert = 'hm-rpc.0.OEQ1544610.2.POWER'/*Zw Stecker 3 Wohnzimmer mM(WERTE):2 POWER*/
wenn dieser Wert größer ist als 100, dann ist die Klimaanlage wirklich eingeschaltet. Ist der Wert unter 100, dann wurde der IR Befehl offenbar nicht empfangen und sollte nochmals gesenet werden. Max. drei Versuche.
Du hast vor eienm Jahr gesagt "Bei IR stört es nicht, wenn höchstens alle 2,5 Minuten gesendet wird". Diese 2,5 Minuten sollte man auch einbauen/ berücksichtigen.
Ich komme aber nicht wirklich weiter mit dem Einbau dieser Prüfung. Vor allem auch, ob die Klimaanalge auch den Auschalbefehl richtig empfangen hat.
Würdest Du @paul53 mir bittehier noch einmalunter die Arme greifen?
-
@skorpil sagte: noch einmalunter die Arme greifen?
Poste bitte das aktuelle Skript in Code tags.
-
// ************Deklarationen********; // const KlimAut = 'hm-rega.0.9668'/*KlimaAutomatic*/; const AktTemperaturChan = 'hm-rpc.0.JEQ0709626.1.TEMPERATURE'/*Thermostat Wohnzimmer 0 TEMPERATURE*/; const Stromwert = 'hm-rpc.0.OEQ1544610.2.POWER'/*Zw Stecker 3 Wohnzimmer mM(WERTE):2 POWER*/; // Schaltbefehle const klimEin = 'broadlink2.0.RM:Broadlink-RMPROPLUS-f4-3c-2b.L.Klima Weiden EIN'/*Klima Weiden EIN*/; const klimAus = 'broadlink2.0.RM:Broadlink-RMPROPLUS-f4-3c-2b.L.Klima Weiden AUS'/*Klima Weiden AUS*/; // Sollwert 22 Wohnzimmer ................ const Sollwert = 22; // Erlaubten Schwankungsbereich der Temperatur setzen const maxSoll = Sollwert + 1; const minSoll = Sollwert - 1; // Anzahl der Wiederholungen und Intervallzeit const maxWiederholungen = 3; const intervallzeit = 150000; // 2,5 Minuten in Millisekunden // Statusvariable für das Script let scriptGestartet = false; //****************Programm*********; on(KlimAut, function (dp) { if (dp.state.val) { starteScript(); } else { stoppeScript(); } }); function starteScript() { if (!scriptGestartet) { scriptGestartet = true; pruefeTemperaturUndStromverbrauch(); } } function stoppeScript() { if (scriptGestartet) { scriptGestartet = false; clearInterval(pruefeStromverbrauchIntervall); } } function pruefeTemperaturUndStromverbrauch() { let temperatur = getState(AktTemperaturChan).val; let stromverbrauch = getState(Stromwert).val; if (temperatur > maxSoll) { setState(klimEin, true); pruefeStromverbrauch(); } else if (temperatur < minSoll) { setState(klimAus, true); pruefeStromverbrauch(); } else if (stromverbrauch > 100) { stoppeScript(); } } function pruefeStromverbrauch() { let wiederholungen = 0; let stromverbrauchPruefen = setInterval(function() { let stromverbrauch = getState(Stromwert).val; if (stromverbrauch > 100) { clearInterval(stromverbrauchPruefen); } else { wiederholungen++; if (wiederholungen > maxWiederholungen) { clearInterval(stromverbrauchPruefen); console.log('Schaltbefehl hat nicht funktioniert'); } else { console.log('Schaltbefehl wiederholen'); if (getState(klimEin).val) { setState(klimEin, true); } else if (getState(klimAus).val) { setState(klimAus, true); } } } }, intervallzeit); } // Initialisierung des Skripts basierend auf dem aktuellen Status von KlimAut if (getState(KlimAut).val) { starteScript(); } else { stoppeScript(); }
-
@skorpil
Das ist zu kompliziert, da nur der Automatikschalter triggert, aber nicht die Temperatur und nicht der Stromverbrauch. In ioBroker verwendet man Trigger auf Ereignisse. -
@paul53 ich hab schon weitergearbeitet. Was sagst Du hierzu?
// ************Deklarationen********; const KlimAut = 'hm-rega.0.9668'/*KlimaAutomatic*/; const AktTemperaturChan = 'hm-rpc.0.JEQ0709626.1.TEMPERATURE'/*Thermostat Wohnzimmer 0 TEMPERATURE*/; const Stromwert = 'hm-rpc.0.OEQ1544610.2.POWER'/*Zw Stecker 3 Wohnzimmer mM(WERTE):2 POWER*/; // Schaltbefehle const klimEin = 'broadlink2.0.RM:Broadlink-RMPROPLUS-f4-3c-2b.L.Klima Weiden EIN'/*Klima Weiden EIN*/; const klimAus = 'broadlink2.0.RM:Broadlink-RMPROPLUS-f4-3c-2b.L.Klima Weiden AUS'/*Klima Weiden AUS*/; // Sollwert 23 Wohnzimmer ................ const Sollwert = 23; //Erlaubten Schwankungsbereich der Temperatur setzen const maxSoll = Sollwert + 1; const minSoll = Sollwert - 1; // Anzahl der Wiederholungen und Intervallzeit const maxWiederholungen = 3; const intervallzeit = 150000; // 2,5 Minuten in Millisekunden //****************Programm*********; on(KlimAut, function (dp) { if(!dp.state.val) setState(klimAus, true); else if(getState(AktTemperaturChan).val > maxSoll) { setState(klimEin, true); pruefeStromverbrauch(); } }); on(AktTemperaturChan, function(dp) { // Aktuelle Temperatur von Wohnzimmer ermitteln let AktTemperatur = dp.state.val; if(getState(KlimAut).val) { // Bei aktueller Temperatur > Sollwert + 1.0 Aktor ausschalten // Bei aktueller Temperatur < Sollwert - 1.0 Aktor einschalten if (AktTemperatur > maxSoll) { setState(klimEin, true); pruefeStromverbrauch(); } else if (AktTemperatur < minSoll) { setState(klimAus, true); pruefeStromverbrauch(); } } }); function pruefeStromverbrauch() { let wiederholungen = 0; let stromverbrauchPruefen = setInterval(function() { let stromverbrauch = getState(Stromwert).val; if (stromverbrauch > 100) { clearInterval(stromverbrauchPruefen); } else { wiederholungen++; if (wiederholungen > maxWiederholungen) { clearInterval(stromverbrauchPruefen); console.log('Schaltbefehl hat nicht funktioniert'); } else { console.log('Schaltbefehl wiederholen'); if (getState(klimEin).val) { setState(klimEin, true); } else if (getState(klimAus).val) { setState(klimAus, true); } } } }, intervallzeit); }
-
@skorpil
Schon besser, aber noch zu kompliziert.
Vorschlag (ohne Begrenzung der Anzahl Wiederholungen):// ************Deklarationen********; // const KlimAut = 'hm-rega.0.9668'/*KlimaAutomatic*/; const AktTemperaturChan = 'hm-rpc.0.JEQ0709626.1.TEMPERATURE'/*Thermostat Wohnzimmer 0 TEMPERATURE*/; const Stromwert = 'hm-rpc.0.OEQ1544610.2.POWER'/*Zw Stecker 3 Wohnzimmer mM(WERTE):2 POWER*/; // Schaltbefehle const klimEin = 'broadlink2.0.RM:Broadlink-RMPROPLUS-f4-3c-2b.L.Klima Weiden EIN'/*Klima Weiden EIN*/; const klimAus = 'broadlink2.0.RM:Broadlink-RMPROPLUS-f4-3c-2b.L.Klima Weiden AUS'/*Klima Weiden AUS*/; // Sollwert 22 Wohnzimmer ................ const Sollwert = 22; // Erlaubten Schwankungsbereich der Temperatur setzen const maxSoll = Sollwert + 1; const minSoll = Sollwert - 1; // Anzahl der Wiederholungen // const maxWiederholungen = 3; // Statusvariablen für das Script var temp = getState(AktTemperaturChan).val; var auto = getState(KlimAut).val; var istEin = getState(Stromwert).val > 100; // var wiederholugen = 0; //****************Programm*********; function Regler() { if(auto) { if (temp >= maxSoll) { if (!istEin) setState(klimEin, true); } else if (temp <= minSoll && istEin) { setState(klimAus, true); } } else if (istEin) { setState(klimAus, true); } } Regler(); // Skriptstart on(KlimAut, function (dp) { auto = dp.state.val; Regler(); }); on({id: AktTemperaturChan}, function(dp) { // triggert alle 150 s temp = dp.state.val; Regler(); }); on(Stromwert, function(dp) { istEin = dp.state.val > 100; });
Es wird solange alle 2,5 Minuten wiederholt, bis die Stromaufnahme passt.
-
Zeile 37-39 würde ich nach 46 verschieben und in 9 oder 10 ist ein Tippfehler vom TE
-
@ticaki sagte: Zeile 37-39 würde ich nach 46 verschieben
Dort wird nicht wiederholt und auch nicht der Skriptstart berücksichtigt.
-
-
@paul53
Richtig, wieso sollte das Skript schalten, wenn schalten deaktiviert wurde? -
@ticaki sagte: wieso sollte das Skript schalten, wenn schalten deaktiviert wurde?
Wenn "Auto" deaktiviert wird, soll es ausschalten (falls eingeschaltet), was evtl. nicht sofort klappt.
@skorpil sagte in Bitte Programmierhilfe:
ob die Klimaanalge auch den Auschalbefehl richtig empfangen hat.
-
Das hab ich dann später auch gelesen. Die Automatik sollte aber wirklich nicht noch unendlich rumschalten daher:
const retrys = 3 var autoAusZ = getState(KlimAut).val ? retrys : 0; function Regler() { if(auto) { if (temp >= maxSoll) { if (!istEin) setState(klimEin, true); } else if (temp <= minSoll && istEin) { setState(klimAus, true); } } else if (autoAusZ-- > 0 && istEin) { setState(klimAus, true); } } Regler(); // Sktiptstart on(KlimAut, function (dp) { auto = dp.state.val; autoAusZ = retrys; Regler(); });
10000 Jahre bis zum überlauf von autoAusZ
-
@ticaki sagte: Die Automatik sollte aber wirklich nicht noch unendlich rumschalten daher:
Diese Wiederholungsbegrenzung kann man natürlich ergänzen.
-
was genau ist jetzt der Unterschied zwischen dem Vorschlag von ticaki:
const retrys = 3 var autoAusZ = getState(KlimAut).val ? retrys : 0; function Regler() { if(auto) { if (temp >= maxSoll) { if (!istEin) setState(klimEin, true); } else if (temp <= minSoll && istEin) { setState(klimAus, true); } } else if (autoAusZ-- > 0 && istEin) { setState(klimAus, true); } } Regler(); // Sktiptstart on(KlimAut, function (dp) { auto = dp.state.val; autoAusZ = retrys; Regler(); });
und meinem Script (siehe oben):
// ************Deklarationen********; const KlimAut = 'hm-rega.0.9668'/*KlimaAutomatic*/; const AktTemperaturChan = 'hm-rpc.0.JEQ0709626.1.TEMPERATURE'/*Thermostat Wohnzimmer 0 TEMPERATURE*/; const Stromwert = 'hm-rpc.0.OEQ1544610.2.POWER'/*Zw Stecker 3 Wohnzimmer mM(WERTE):2 POWER*/; // Schaltbefehle const klimEin = 'broadlink2.0.RM:Broadlink-RMPROPLUS-f4-3c-2b.L.Klima Weiden EIN'/*Klima Weiden EIN*/; const klimAus = 'broadlink2.0.RM:Broadlink-RMPROPLUS-f4-3c-2b.L.Klima Weiden AUS'/*Klima Weiden AUS*/; // Sollwert 23 Wohnzimmer ................ const Sollwert = 23; //Erlaubten Schwankungsbereich der Temperatur setzen const maxSoll = Sollwert + 1; const minSoll = Sollwert - 1; // Anzahl der Wiederholungen und Intervallzeit const maxWiederholungen = 3; const intervallzeit = 150000; // 2,5 Minuten in Millisekunden //****************Programm*********; on(KlimAut, function (dp) { if(!dp.state.val) setState(klimAus, true); else if(getState(AktTemperaturChan).val > maxSoll) { setState(klimEin, true); pruefeStromverbrauch(); } }); on(AktTemperaturChan, function(dp) { // Aktuelle Temperatur von Wohnzimmer ermitteln let AktTemperatur = dp.state.val; if(getState(KlimAut).val) { // Bei aktueller Temperatur > Sollwert + 1.0 Aktor ausschalten // Bei aktueller Temperatur < Sollwert - 1.0 Aktor einschalten if (AktTemperatur > maxSoll) { setState(klimEin, true); pruefeStromverbrauch(); } else if (AktTemperatur < minSoll) { setState(klimAus, true); pruefeStromverbrauch(); } } }); function pruefeStromverbrauch() { let wiederholungen = 0; let stromverbrauchPruefen = setInterval(function() { let stromverbrauch = getState(Stromwert).val; if (stromverbrauch > 100) { clearInterval(stromverbrauchPruefen); } else { wiederholungen++; if (wiederholungen > maxWiederholungen) { clearInterval(stromverbrauchPruefen); console.log('Schaltbefehl hat nicht funktioniert'); } else { console.log('Schaltbefehl wiederholen'); if (getState(klimEin).val) { setState(klimEin, true); } else if (getState(klimAus).val) { setState(klimAus, true); } } } }, intervallzeit); }
Danke für Eure Hilfe
-
- Deine Bedingungen werden erfüllt, in deinem Skript wird nur 1 mal der Ausschaltbefehl geprüft.
- wenn pruefeStr... 2 mal so aufgerufen wird das beide Interval gleichzeitig laufen, wird clearInterval im besten Fall den jüngsten beenden und der erste läuft unendlich weiter. k.A. da das nicht funktionieren kann, hab ichs nicht getestet. selbiges gilt für wiederholungen... eine Variable für Intervalle die mehrfach aufgerufen werden können ist quark. Ob sie es sollen ist was anderes wie ob sie es können
if (getState(klimEin).val) { setState(klimEin, true); } else if (getState(klimAus).val) { setState(klimAus, true); }
mal neben her... wer setzt klimEin auf false?
EDIT:
Ich würde den Infrarotteil wohl so lösen:const ON = 1 const OFF = 2 var target_state = 0 var current_state = 0 function stromverbrauch(){ // hier gucken ob an return // true oder false } setInterval(function(){ if ( target_state == current_state) return if (target_state == ON) { if (!stromverbrauch) { //sende einschaltsignal } else { current_state = ON } } if (target_state == OFF) { if (stromverbrauch ) { //sende auschaltsignal } else { current_state = OFF } } },10000)
Noch ein EDIT:
über target_state steuert man das dann. Vielleicht noch ein Zähler der bei auto off hoch zählt und den return am anfang auslöstif ( (target_state == current_state && auto) || (!auto && acount++ > 3)) // der zählt nur hoch wenn auto == false nur noch zurücksetzen beim on(auto)
Noch was:
so kannst du die Zahl der aufrufe insgesamt begrenzenvar retrys = 0; function setTargetState(val) { target_state = val retrys = 0; } setInterval(function(){ if (target_state == current_state || retrys++ > 3) return
dann steuert man über setTargetState(ON)
-
@ticaki sagte: wer setzt klimEin auf false?
Die meisten Button sind immer true und reagieren auf Aktualisierung des Zeitstempels.
-
@paul53
ok. Er fragt auf true ab und das ist ja dann immer true?