NEWS
Lichtsteuerung (langsam dimmen lassen) [gelöst]
-
Moin!
Ich versuche hier seit Tagen, eine Funktion in meiner Lichtsteuerung zu realisieren, bei der die Helligkeit meiner Glühbirnen und Panels (momentan Tradfri, aber soll prinzipiell Herstellerunabhängig funktionieren - daher die Funktion) langsam in Richtung Zielwert gehen soll.
Das ganze funktioniert auch, der interval stoppt aber nicht. Insbesondere, wenn ich die Funktion zweimal starte, mit unterschiedlichen Parametern, gibt es Probleme. Geht das so nicht?EDIT:
Führe ich dieses Script mit den beiden Aufrufen aus, dimmt das Licht korrekt und dann kommt die Endlosmeldung "Sonstiger Fall - Stoppe" - die sollte doch normalerweise nur einmal kommen, dann wäre es gestoppt?EDIT 10.04.:
dank @ticaki ist das Problem gelöst. Hier der Code:
-
@azmo sagte:
wenn ich die Funktion zweimal starte, mit unterschiedlichen Parametern, gibt es Probleme. Geht das so nicht?
Das Problem ist, dass die Variablen DimmAktuell und dimmintervall globale Variablen sind. Zwei Funktionsaufrufe bedingen also eine gegenseitige Beeinflussung. Mit DimmAktuell läßt sich das Problem leicht umgehen, indem man auch diese als Übergabeparameter definiert. Wie man aber in Blockly verhindert, dass die Intervall-Variable global ist, weiß ich nicht.
-
Ich sehe folgende Problem.
- Du erhöhst/verringerst einen unbekannten Wert um 0,5 und erwartest, dass er irgendwann einen bestimmten Wert bis auf die letzten Nachkommastelle trifft. Besser wäre
if (dimmziel > dimmwert + 0.5) if (dimmziel < dimmwert - 0.5)
und den letzten Schritt im Sonst auszuführen. Einen Bereich von +/- 0.5 wirst du mit dem Code immer treffen.
- Mit den Interval Funktionen kannst du nur genau einen Interval (mit dem selben Namen)stoppen. Du kannst zwar "endlos" viele startet aber nur der letzte wird durch die Stop - Funktion abgebrochen. Mit Javascript ist das kein Problem zu umgehen, hab aber keine Ahnung wie und ob man das mit
Blockly machen kann.
Lösung für 2. wäre: Du speicherst das Gerät und den Zielwert in einem Array als Objekt oder weiteres Array. Und arbeitest diese Liste in deinem Interval ab.
Bei deconz gibt es einen Datenpunkt "transitiontime" gibt es diesen auch für Tradfri? Wenn ja, kannst du über diesen Wert dimmen.
Editiert
-
Hi.
Danke für die Antworten!@paul53 sagte in Lichtsteuerung (langsam dimmen lassen):
Wie man aber in Blockly verhindert, dass die Intervall-Variable global ist, weiß ich nicht.
@ticaki sagte in Lichtsteuerung (langsam dimmen lassen):
Mit den Interval Funktionen kannst du nur genau einen Interval stoppen.
Mir scheint, dass da der Knackpunkt liegt, oder? Das ist natürlich richtig nervig..
Weiß jemand anderes vielleicht, ob und wie das geht? Ich habe keinerlei Erfahrung mit Javascript, kann da jemand helfen bzw. wäre es einfach möglich, so eine Umgehung einzubinden?@ticaki sagte in Lichtsteuerung (langsam dimmen lassen):
Bei deconz gibt es einen Datenpunkt "transitiontime" gibt es diesen auch für Tradfri? Wenn ja, kannst du über diesen Wert dimmen.
Den gibt es, das funktioniert aber seltsamerweise und leider nicht. Auch mit der Option im Adapter, dass der Wert nicht zurückgesetzt wird, hat eine Veränderung einfach keinen Effekt auf die Dimmzeit.
@ticaki sagte in Lichtsteuerung (langsam dimmen lassen):
Ich sehe folgende Problem.
- Du erhöhst/verringerst einen unbekannten Wert um 0,5 und erwartest, dass er irgendwann einen bestimmten Wert bis auf die letzten Nachkommastelle trifft.
Ok, ich glaube ich verstehe was du meinst aber das sollte doch eigentlich in dem Script nicht passieren, oder? Wie ich es verstehe, sollte er einmal einmal am Anfang den realen Wert abfragen, gerundet in die Zielvariable schreiben und diese dann immer um 0,5 verändern und ins Gerät schreiben, bis der Zielwert (in der Variable, vom Gerät weiß er ja dann nichts mehr) erreicht ist. Das Ding ist, dass die Tradfris leider nie genau den Wert ansteuern, den sie sollen sondern der Hub ihn noch einmal zu korrieren scheint (z.B. wird aus 10 dann 10,2 oder so etwas seltsames). Mit dieser Vorgehensweise wollte ich das umgehen und einfach den "realen" Wert ignorieren. Mit reicht, wenn er ungefähr getroffen wird und das sollte ja mit dem Schreiben des angenommenen Wertes in jedem Durchgang gegeben sein (und nur dieser wird ja am Ende überprüft). Oder habe ich da falsch gedacht? Vorher (mit einer Abfrage des tatsächlichen Wertes in jedem Intervall) kam es zu Problemen, weil der Wert z.B. in einem der Schritte von 10 auf 10,5 geschaltet werden sollte und direkt danach aber auf 10 gesprungen ist - und dann wieder auf 10,5 geschaltet wurde usw.
Dank erst einmal... Das Problem bleibt leider bestehen
Weiß jemand vielleicht eine andere Lösung? Ich würde das gerne elegant machen, so dass ich nicht für jede Lampe und jeden Parameter (Helligkeit/Farbe) einen einzelnen Intervall schreiben muss. -
@azmo
Punkt 1. Hast recht, hab das runden nicht gesehen.In Punkt 2. hab ich dir doch die Lösung geschrieben?
Hab das gerade mit Blockly gebaut und es sieht schrecklich aus -
@ticaki sagte in Lichtsteuerung (langsam dimmen lassen):
@azmo
Punkt 1. Hast recht, hab das runden nicht gesehen.In Punkt 2. hab ich dir doch die Lösung geschrieben?
Hab das gerade mit Blockly gebaut und es sieht schrecklich ausOhgott, danke aber das ist ja dann richtig schrecklich.
Ich finde das unglaublich ärgerlich, dass das nicht so einfach geht...
Kann man das statt des zickigen Intervalls vielleicht mit Schleifen lösen? Da war aber auch etwas mit dem Timing was ich gelesen hatte meine ich... Wird da nicht immer alles gleichzeitig gemacht oder sowas komisches?grübel
-
@azmo sagte:
Wird da nicht immer alles gleichzeitig gemacht oder sowas komisches?
Ja, quasi gleichzeitig.
-
Ich hab doch ne Idee. Dazu mußt du aber alle Variablen in der Funktion übergeben, zum stoppen eine Funktion aufrufen und für jede Lampe die du hinzufügst beide Funktionen um einen Eintrag erweitern. i ist ein Zähler den du beim aufruf setzten mußt, m,j,k sind deine verwendeten Variablen. Für 3 Lampen sieht es dann so aus.
etwas tun ist deine aktuelle Funktion mit entsprechenden Parametern, das hier könnte gehen. Bin mir aber nicht sicher.
Schleifen gehe nicht, da du diese nicht pausieren kannst. Aufrufe vorwegnehmen geht auch nicht, da die Verzögerung soweit ich sehe nicht mit Variablen gesetzt werden kann.
-
@ticaki sagte in Lichtsteuerung (langsam dimmen lassen):
Ich hab doch ne Idee. Dazu mußt du aber alle Variablen in der Funktion übergeben, zum stoppen eine Funktion aufrufen und für jede Lampe die du hinzufügst beide Funktionen um einen Eintrag erweitern. i ist ein Zähler den du beim aufruf setzten mußt, m,j,k sind deine verwendeten Variablen. Für 3 Lampen sieht es dann so aus.
etwas tun ist deine aktuelle Funktion mit entsprechenden Parametern, das hier könnte gehen. Bin mir aber nicht sicher.
Schleifen gehe nicht, da du diese nicht pausieren kannst. Aufrufe vorwegnehmen geht auch nicht, da die Verzögerung soweit ich sehe nicht mit Variablen gesetzt werden kann.
Hey, das sieht interessant aus!
Danke, das probiere ich heute Abend aus wenn ich mich dransetzen kann. Gute Idee.EDIT:
Ich vermute da könnte man i ja auch jeweils +1 bzw -1 setzen wenn ein Intervall gestartet bzw. gestoppt wird... Interessant. Das müsste im Idealfall dann variabel sein damit ich nicht doch für jede Lampe/jeden Parameter einen Intervall basteln bzw. im Vorfeld eine Zahl festlegen muss... Hm. -
Ich habe mit Tradfri (und dem Tradfri-Adapter) auch lange an einem Sonnenuntergang gebastelt.
Das dumme bei Tradfri ist, dass zum einen die transition time nicht beliebig hoch sein darf, und wenn noch weitere Paramter wie Helligkeit, Farbe, Sättigung) übergeben werden sollen nur dann vernünftig abgearbeitet werden, wenn die transition time = 0 ist.
Ich musste also auch mit möglichst vielen kleinsten Schritten arbeiten, damit es halbwegs weich ist.
-
@azmo sagte in Lichtsteuerung (langsam dimmen lassen):
Ich vermute da fehlt noch das i-1 wenn ein Intervall gestoppt wurde...
Soweit wie du hab ich nicht nicht durchdacht. Ob i automatisch gesetzt werden kann, weiß ich nicht.
Mit Javascript ohne Blockly muß man sich auf jeden Fall nicht so einen abbrechen
-
Mit Javascript ohne Blockly muß man sich auf jeden Fall nicht so einen abbrechen
Den Eindruck habe ich auch.
Mir missfällt noch immer, dass nicht einfach innerhalb der Funktion alles lokal bleiben kann und diese so einfach mehrfach mit verschiedenen Parametern ausgeführt werden kann. Das wäre so schön übersichtlich. Verstehe ich richtig, dass es am Intervall liegt, der global ausgeführt und so mehrfach neu gestartet wird bzw wo laufende Intervalle neue variablen bekommen durch den start eines neuen?
Müsste es dann nicht möglich sein, WENN ich eine art sleep/pause für eine Sekunde einbaue (gibt es doch sicher einen Weg?), eine Schleife sozusagen händisch via Funktion nachzubauen und so ohne den für mich so eigentlich nutzlosen Intervall das ganze ausführen zu lassen? Eine pause, sollte sie kompliziert zu basteln sein wie ich vermute (habe etwas recherchiert) sollte sich doch notfalls via Blockly JS Funktion einbauen lassen?Naja ich schlafe mal eine Nacht drüber... Das muss doch gehen!
-
@azmo
Ich klappere das mal ab:Blockly soll einfach sein, globale Variablen sind einfach, da man in einem Blockly von überall alles ändern kann.
Verstehe ich richtig, dass es am Intervall liegt, der global ausgeführt und so mehrfach neu gestartet wird bzw wo laufende Intervalle neue variablen bekommen durch den start eines neuen?
Es geht darum einen Interval von ausserhalb des Intervals zu beenden. Wenn du einen Funktion ausrufst und ihre als Parameter den Namen des Intervals übergibst solltest du den Interval von innerhalb des Intervals beenden können. Wenn du aber für Lampe X 2 * die Funktion aufrufst hast du keine Möglichkeit den 1. Interval zu beenden.
Ich kenne mich mit async und await nicht wirklich aus.
Blockly JS Funktion muß ich mir mal ansehen.
EDIT: Oha
Ich könnte dir heute Abend 2-3 Funktionen zusammenstellen die den Verwaltungskram machen, dann könntest du bei deinem ursprünglichen Plan bleiben.Funktionen:
- Speichere aktuellen Interval für ID
- Beende Interval für ID
-
Das wäre ja total cool!
Schön, dass es noch Hoffnung gibt. Ich forsche heute nach Feierabend auch noch einmal weiter nach.Danke!
-
Das hier sollte funktionieren. Hab es aber nicht getestet.
Der Intervall muß Intervall heißen. Da ich nicht weiß was alles im XML Code ist, hab ich den JS Code für die Funktionen auch gepostet
Code saveInterval:
if (database === undefined) database = {}; database[id] = Intervall; return database;
Code clearInterval()
if (database === undefined) return {}; if (database[id] === undefined) return database; if (database[id]) { clearInterval(database[id]); database[id] = null; } return database;
Obj-Id ist die Lampe die du schaltest oder eine andere eindeutige Zeichenkette
-
@azmo
Ich habe gerade das hier mit einem Array "timers" getestet: -
Hi.
Danke für die Antworten, ich habe mich leider erst heute dransetzen können.
Momentan teste ich die Idee von @ticaki .
Ich habe versucht, den Code soweit zu verstehen, komme jedoch nicht so recht weiter. Anscheinend wird der Intervall irgendwie umbenannt, dynamisch nach obj-id?
Die Frage ist nur: wie führe ich dann meine Aktionen darin aus? Muss ich die JS Funktionen erweitern oder bin ich auf dem richtigen Weg? (siehe Screenshot).Ich bin gerade ein wenig ratlos... Vermutlich mache ich etwas offensichtliches falsch.
Wie man sieht, wird die Debug Info korrekt ausgegeben, der Intervall scheint aber nicht zu starten bzw. nur einmal zu laufen. Was ist verkehrt?
EDIT:
Die Schleife scheint zu laufen, aber irgendwie alle Minute. Hier der Log, diese Ausgaben wurde ohne weitere Schalteraktion erzeugt:10.4.2020, 11:53:01.009 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: database: [object Object] DimmAktuell: 100 10.4.2020, 11:54:01.009 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: database: [object Object] DimmAktuell: 100 10.4.2020, 11:55:01.012 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: database: [object Object] DimmAktuell: 100
Irgendwie ist er von 1 (realer Wert des Objekts zu Beginn, jetzt ist der reale Wert 3,4) auf 100 gesprungen.
Wenn ich den Script neu starte und den Schalter wieder betätige (vorher Zielwert auf 50 gesetzt):10.4.2020, 11:58:43.428 [info ]: javascript.0 (1293) Start javascript script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest 10.4.2020, 11:58:43.435 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: registered 1 subscription and 1 schedule 10.4.2020, 11:58:52.494 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: test 10.4.2020, 11:58:53.497 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: database: [object Object] DimmAktuell: 3 10.4.2020, 11:59:01.010 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: database: [object Object] DimmAktuell: 100 10.4.2020, 12:00:01.011 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: database: [object Object] DimmAktuell: 100 10.4.2020, 12:01:01.007 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: database: [object Object] DimmAktuell: 100
-
Ja ist nicht ganz richtig. Die Funktion clearInterval ist das selbe wie stoppe zyklische Ausführung. Benutze es dort wo du den Interval beenden willst.
Du mußt nicht in Code arbeiten. Ich erstelle ein Objekt in dem ich zu jedem Datenpunkt, den du übermittels die Referenz auf den Interval abspeichere. Womit das hier nicht klar kommt, ist wenn du 2 Intervals auf dem selben Datenpunkt zur gleichen Zeit startest.
Habs mittlerweile auch getestet und es funktioniert.
! -
@ticaki
Ah danke!Damit scheint es zu funktionieren. Den Verdacht hatte ich auch...
Er dimmt jetzt korrekt hoch bzw. runter.
Wenn er es erreicht, wird gestoppt wie geplant.Aber: Wenn ich im angefügten Script den Schalter umlege:
dann springt das Licht in der Farbe von 0 auf 100 und es passiert folgendes:
10.4.2020, 12:26:32.138 [info ]: javascript.0 (1293) Start javascript script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest 10.4.2020, 12:26:32.144 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: registered 1 subscription and 0 schedules 10.4.2020, 12:26:33.994 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: test 10.4.2020, 12:26:34.997 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.colorTemperature DimmAktuell: 100 10.4.2020, 12:26:34.998 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: Intervall Beenden 10.4.2020, 12:26:34.999 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.brightness DimmAktuell: 100 10.4.2020, 12:26:35.997 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.colorTemperature DimmAktuell: 99.5 10.4.2020, 12:26:35.999 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.brightness DimmAktuell: 100 10.4.2020, 12:26:36.998 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.colorTemperature DimmAktuell: 99.5 10.4.2020, 12:26:37.000 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.brightness DimmAktuell: 100 10.4.2020, 12:26:37.999 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.colorTemperature DimmAktuell: 99.5 10.4.2020, 12:26:38.001 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.brightness DimmAktuell: 100 10.4.2020, 12:26:38.999 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.colorTemperature DimmAktuell: 99.5 10.4.2020, 12:26:39.003 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.brightness DimmAktuell: 100 10.4.2020, 12:26:40.000 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.colorTemperature DimmAktuell: 99.5 10.4.2020, 12:26:40.002 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.brightness DimmAktuell: 100 10.4.2020, 12:26:41.000 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.colorTemperature DimmAktuell: 99.5 10.4.2020, 12:26:41.002 [info ]: javascript.0 (1293) script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest: obj-id: tradfri.0.L-65546.lightbulb.brightness DimmAktuell: 100 10.4.2020, 12:26:41.180 [info ]: javascript.0 (1293) Stop script script.js.Zeitsteuerung.Lichtsteuerung_Dimmertest
Jetzt bin ich verwirrt, bestimmt etwas offensichtliches.
Die Werte scheinen zu konkurrieren? Einer hoch, einer runter? Es liegt also vermutlich an den Variablen DimmAktuell und Dimmziel? Sind die nicht in der Funktion lokal? Oder auch global? Argh! -
@azmo
Das ist nicht dein FehlerBlocky definiert ALLE Variablen global. Um dein Problem zu beheben übergebe Dimmaktuell (Auf gleiche Schreibweise achten ich habs falsch denke ich) der Funktion:
Damit wird innerhalb der Funktion Dimmaktuell lokal und nicht jeder Interval überschreibt die. Deshalb hatte ich obj-id, j, k angegeben. Alle Variablen die du verwendest müssen als Parameter rein kommen.