NEWS
Kann man einem Datenpunkt eine flexible Werteliste zuweisen?
-
@mickym said in Kann man einem Datenpunkt eine flexible Werteliste zuweisen?:
Du musst halt die IDs der Räume wissen bzw. im Flow dann nachpflegen.
wenn das alles ist...
-
@schneidy76 So fertig.
Also wie gesagt Du erzeugst selbst eine Struktur die alle Räume als Logikewerte enthält und die Du mit check boxen in der VIS mit true oder false belegst.
Über iobroker-IN und Wildcard nach Räumen - überwachst Du mit einer Node alle Datenpunkte auf einmal:
Ich extrahiere den letzten Wert aus dem Pfad/topic und so baut sich wie eben schon gepostet ein Objekt auf das mit true oder false den Status enthält, ob der Raum gereinigt werden soll. (1. Debug ausgabe: Objekt Räume)
Im nächsten Schritt werden dann aus diesem Objekt die Räume extrahiert, die gereinigt werden sollen. (2. Debugausgabe: Array mit Räumen).
Anschliessend übersetzen wir die Räume in IDs.
In der ChangeNode übersetze IDs habe ich kurz eine Nachrichteneigenschaft roomIDs definiert, die Du halt einmalig pflegen musst:
Die ChangeNode übersetzt dann einfach in dem sie dort nachschaut die Namen der Räume durch IDs.
Deshalb löschen wir diese Eigenschaft auch wieder.
Die nächste Debug-Ausgabe enthält also nun ein Array mit den IDs, die zu reinigen sind.Anschließend setzen wir den Befehl und schicken Ihn an den Befehlsdatenpunkt:
Also eigentlich nicht so schwer - hier der Import:
-
@mickym das ist noch der einfache Teil
-
@schneidy76 Du kannst natürlich die ganzen Change Nodes in eine Change Node zusammenfassen - ich habe das nur zur Illustration und zu Debugging Zwecken in einzelne Change Nodes verteilt.
Achso - mach lieber noch eine Trigger Node hinten dran - damit der Staubi nicht mit jedem Klick ein Kommando erhält sondern erst wenn 10s keine Eingabe mehr erfolgt:
-
@mickym unglaublich, wie das immer hinzaubern tust...
Vielen Dank, das bekomme ich jetzt hin! -
@schneidy76 So hier nochmal mit trigger Node - die erst nach 10s das Kommando losschickt, wenn keine Eingabe mehr über VIS erfolgt:
Und im unteren Teil wurden die 3 Change Nodes in einer ChangeNode zusammengefasst:
-
Probiere halt mal den Datenpunkt aus, in dem Du das Kommando manuell in den Datenpunkt schreibst.
-
@mickym Mache ich. Vermutlich werde ich noch einen "Start" Button als Trigger hinzufügen.
Dann kann ich das Reinigen definiert auslösen. -
@schneidy76 Dann würde ich es aber über die List Node und Arrays holen - ich modifiziere mal den Flow entsprechend - dann kann man sich die JOIN Node und die TRIGGER Node sparen.
Wenn der Name richtig gepflegt ist kann man es direkt über den Namen anprechen - über die Id muss man halt wieder extrahieren.
So hier nun einen Flow - mit definiertem Trigger:
dann kann man zum Testen auch eine Inject Node nehmen.
Wenn der Name des Datenpunktes den Raumnamen enthält (siehe obere Change Node), dann kann man auch direkt auf das common Objekt zugreifen:
Ansonsten muss man es aus der _id extrahieren.
da muss ich auch immer bissi tüfteln.
Der Rest bleibt gleich - hier der Import:
-
Ich habe noch eine Ergänzung.
In der Doku steht ein Array für die Indizes:
Wenn Du nur 1 Raum selektierst, dann wird daraus in dem Flow ein skalarer Wert:
Wenn also in jedem Fall ein Array entstehen soll auch mit EINEM Element dann musst Du in der Change Node noch folgendes ergänzen:
also noch den Teil hinten ergänzen, dann kommt immer ein Array raus:
~>$append([])
also vollständig:
$.payload@$n.[roomIDs[Name=$n].id]~>$append([])
-
@mickym Hi, irgendwas habe ich kräftig vermasselt. Der Aufruf über den Trigger hat gerade ioBroker lahm gelegt...
Was ich versucht habe den ersten Flow mit dem Trigger zu kombinieren...[ { "id": "ee5a3152d31c300a", "type": "change", "z": "db97e840ec20025f", "name": "", "rules": [ { "t": "change", "p": "topic", "pt": "msg", "from": ".*\\/(.*)$", "fromt": "re", "to": "$1", "tot": "str" } ], "action": "", "property": "", "from": "", "to": "", "reg": false, "x": 390, "y": 100, "wires": [ [ "a5227d44b8b0b7de" ] ] }, { "id": "a5227d44b8b0b7de", "type": "join", "z": "db97e840ec20025f", "name": "", "mode": "custom", "build": "object", "property": "payload", "propertyType": "msg", "key": "topic", "joiner": "\\n", "joinerType": "str", "accumulate": true, "timeout": "", "count": "1", "reduceRight": false, "reduceExp": "", "reduceInit": "", "reduceInitType": "", "reduceFixup": "", "x": 570, "y": 100, "wires": [ [ "726d3392d952862d", "304b596c37bc5c42" ] ] }, { "id": "726d3392d952862d", "type": "debug", "z": "db97e840ec20025f", "name": "Objekt Räume", "active": true, "tosidebar": true, "console": false, "tostatus": false, "complete": "payload", "targetType": "msg", "statusVal": "", "statusType": "auto", "x": 760, "y": 40, "wires": [] }, { "id": "8ea80ed297a10f73", "type": "ioBroker out", "z": "db97e840ec20025f", "name": "send command", "topic": "mihome-vacuum.0.control.X_send_command", "ack": "false", "autoCreate": "false", "stateName": "", "role": "", "payloadType": "", "readonly": "", "stateUnit": "", "stateMin": "", "stateMax": "", "x": 1000, "y": 160, "wires": [] }, { "id": "46b3c4894b1c2958", "type": "change", "z": "db97e840ec20025f", "name": "", "rules": [ { "t": "set", "p": "payload", "pt": "msg", "to": "\"app_segment_clean;\" & payload", "tot": "jsonata" } ], "action": "", "property": "", "from": "", "to": "", "reg": false, "x": 790, "y": 200, "wires": [ [ "810b21206fb366fb", "8ea80ed297a10f73" ] ] }, { "id": "810b21206fb366fb", "type": "debug", "z": "db97e840ec20025f", "name": "Kommando", "active": true, "tosidebar": true, "console": false, "tostatus": false, "complete": "payload", "targetType": "msg", "statusVal": "", "statusType": "auto", "x": 990, "y": 200, "wires": [] }, { "id": "304b596c37bc5c42", "type": "change", "z": "db97e840ec20025f", "name": "Nur zu reinigende Räume", "rules": [ { "t": "set", "p": "payload", "pt": "msg", "to": "$each(payload, function($v, $k) {$v ? $k})", "tot": "jsonata" } ], "action": "", "property": "", "from": "", "to": "", "reg": false, "x": 770, "y": 100, "wires": [ [ "803ce32f1e9acca9", "c937041ba3fb937c" ] ] }, { "id": "803ce32f1e9acca9", "type": "debug", "z": "db97e840ec20025f", "name": "Nur zu reinigende Räume", "active": true, "tosidebar": true, "console": false, "tostatus": false, "complete": "payload", "targetType": "msg", "statusVal": "", "statusType": "auto", "x": 1050, "y": 40, "wires": [] }, { "id": "c937041ba3fb937c", "type": "change", "z": "db97e840ec20025f", "name": "Übersetze in IDs", "rules": [ { "t": "set", "p": "roomIDs", "pt": "msg", "to": "[{\"Name\":\"Wohnzimmer\",\"id\":16},{\"Name\":\"Esszimmer\",\"id\":18},{\"Name\":\"Küche\",\"id\":17},{\"Name\":\"Gang\",\"id\":22},{\"Name\":\"Bad\",\"id\":21},{\"Name\":\"Schlafzimmer\",\"id\":19},{\"Name\":\"Büro\",\"id\":20},{\"Name\":\"Treppenhaus\",\"id\":23},{\"Name\":\"Speiß\",\"id\":24}]", "tot": "json" }, { "t": "set", "p": "payload", "pt": "msg", "to": "$.payload@$n.[roomIDs[Name=$n].id]", "tot": "jsonata" }, { "t": "delete", "p": "roomIDs", "pt": "msg" } ], "action": "", "property": "", "from": "", "to": "", "reg": false, "x": 520, "y": 200, "wires": [ [ "46b3c4894b1c2958", "598044f418769ebd" ] ] }, { "id": "598044f418769ebd", "type": "debug", "z": "db97e840ec20025f", "name": "Array von zu reinigenden Räumen", "active": true, "tosidebar": true, "console": false, "tostatus": false, "complete": "payload", "targetType": "msg", "statusVal": "", "statusType": "auto", "x": 820, "y": 260, "wires": [] }, { "id": "edd702cba8f8c6b9", "type": "ioBroker in", "z": "db97e840ec20025f", "name": "Start Multiroom", "topic": "0_userdata.0.Räume.control.Start_Multiroom", "payloadType": "value", "onlyack": "", "func": "all", "gap": "", "fireOnStart": "false", "outFormat": "MQTT", "x": 100, "y": 60, "wires": [ [ "8d6e4906189268ed" ] ] }, { "id": "8d6e4906189268ed", "type": "ioBroker list", "z": "db97e840ec20025f", "name": "", "topic": "0_userdata.0.Räume.*", "objType": "state", "regex": "", "asArray": "true", "onlyIDs": "false", "withValues": "true", "x": 180, "y": 100, "wires": [ [ "ee5a3152d31c300a" ] ] } ]
Debug:26.9.2023, 17:45:36node: Objekt Räume Start_Multiroom : msg.payload : Object { Start_Multiroom: array[10] } 26.9.2023, 17:45:37node: Nur zu reinigende Räume Start_Multiroom : msg.payload : string[15] "Start_Multiroom" 26.9.2023, 17:45:38node: Array von zu reinigenden Räumen Start_Multiroom : msg.payload : undefined undefined 26.9.2023, 17:45:39node: Kommando Start_Multiroom : msg.payload : string[18] "app_segment_clean;" 26.9.2023, 17:45:39node: Objekt Räume Start_Multiroom : msg.payload : Object { Start_Multiroom: array[10] } 26.9.2023, 17:45:39node: Nur zu reinigende Räume Start_Multiroom : msg.payload : string[15] "Start_Multiroom" 26.9.2023, 17:45:39node: Array von zu reinigenden Räumen Start_Multiroom : msg.payload : undefined undefined 26.9.2023, 17:45:39node: Kommando Start_Multiroom : msg.payload : string[18] "app_segment_clean;" 26.9.2023, 17:45:39node: Objekt Räume Start_Multiroom : msg.payload : Object { Start_Multiroom: array[10] } 26.9.2023, 17:45:39node: Nur zu reinigende Räume Start_Multiroom : msg.payload : string[15] "Start_Multiroom" 26.9.2023, 17:45:39node: Array von zu reinigenden Räumen Start_Multiroom : msg.payload : undefined undefined 26.9.2023, 17:45:39node: Kommando Start_Multiroom : msg.payload : string[18] "app_segment_clean;" 26.9.2023, 17:46:06node: Objekt Räume Start_Multiroom : msg.payload : Object { Start_Multiroom: array[10] } 26.9.2023, 17:46:07node: Nur zu reinigende Räume Start_Multiroom : msg.payload : string[15] "Start_Multiroom" 26.9.2023, 17:46:08node: Array von zu reinigenden Räumen Start_Multiroom : msg.payload : undefined undefined 26.9.2023, 17:46:08node: Kommando Start_Multiroom : msg.payload : string[18] "app_segment_clean;" 26.9.2023, 17:46:08node: Objekt Räume Start_Multiroom : msg.payload : Object { Start_Multiroom: array[10] } 26.9.2023, 17:46:08node: Nur zu reinigende Räume Start_Multiroom : msg.payload : string[15] "Start_Multiroom" 26.9.2023, 17:46:08node: Array von zu reinigenden Räumen Start_Multiroom : msg.payload : undefined undefined 26.9.2023, 17:46:08node: Kommando Start_Multiroom : msg.payload : string[18] "app_segment_clean;" 26.9.2023, 17:46:08node: Objekt Räume Start_Multiroom : msg.payload : Object { Start_Multiroom: array[10] } 26.9.2023, 17:46:08node: Nur zu reinigende Räume Start_Multiroom : msg.payload : string[15] "Start_Multiroom" 26.9.2023, 17:46:08node: Array von zu reinigenden Räumen Start_Multiroom : msg.payload : undefined undefined 26.9.2023, 17:46:08node: Kommando Start_Multiroom : msg.payload : string[18] "app_segment_clean;"
-
@schneidy76 Nun dann deaktiviere halt mal die iobroker -out Node und prüge die Ausgabe bzw. poste alle Ausgaben aus den Denug nodes. Wenn das Array aus den zu reinigen Räumen undefinded ist, dann stimmt da ja schon was nicht. Du musst halt step für step vorgehen
-
@mickym habe den Fehler, mit der IN Node geht es. Nehme ich die IN Node als Trigger in Verbindung mit der GET Objects Node, schmiert es mir ab.
Also bleibt das hier, aber wo kann ich die Namen pflegen?
@schneidy76 Dann würde ich es aber über die List Node und Arrays holen - ich modifiziere mal den Flow entsprechend - dann kann man sich die JOIN Node und die TRIGGER Node sparen. Wenn der Name richtig gepflegt ist kann man es direkt über den Namen anprechen - über die Id muss man halt wieder extrahieren. ```
-
@schneidy76 sagte in Kann man einem Datenpunkt eine flexible Werteliste zuweisen?:
Also bleibt das hier, aber wo kann ich die Namen pflegen?
-
@mickym super, jetzt passt es. Nehme den 2. Weg über die IDs. Die Debug Ausgabe passt jetzt.
Habe noch eine Verzögerung rein (1msg/2sek) falls der Daumen nervös ist.Bastel mir nun noch ne Lösung, dass alle Räume wieder auf false gesetzt werden, wenn der Robi zurückfährt. Geht ja wahrscheinlich auch über eine Out Node ins Räume Verzeichnis.
Dann fehlt mir noch ein Popup, in der Vis, in der ich für jeden Raum den Reinigungsmodus definieren kann. Der Anfang ist geschafft!!!
Danke für Deine Geduld!
-
@mickym sagte in Kann man einem Datenpunkt eine flexible Werteliste zuweisen?:
Ich glaube das wirst Du Pech haben - man kann nicht mal mit Javascript die enums modifizieren.
Vielleicht verstehe ich dich falsch jedoch mit dieser Funktion:
async function addToEnum(enumName, newStateId) { if (!await existsObjectAsync(newStateId)) { log(newStateId + ' not exist!', 'warn') return false; } let myEnum = await getObjectAsync(enumName); if (myEnum) { let pos = myEnum.common.members.indexOf(newStateId); if (pos === -1) { try { myEnum.common.members.push(newStateId); myEnum.from = "system.adapter." + "0"; myEnum.ts = new Date().getTime(); await setObjectAsync(enumName, myEnum); return true; } catch (e) {log(e + ' add id: ' + newStateId,'error')} } } return false; }
kannst du einem enum ein neues Mitglied verpassen.