NEWS
Roborock S5 "cleaning area" per Tastendruck?
-
@0018
Siehe Skript:// Koordiaten zählen von unten links (0,0) nach oben rechts (51200,51200) // Die folgenden Koordinaten gehen von nicht rotierter Karte (Winkel 0) aus
Winkel 0 heißt "Robo links vom Dock". Wenn das bei dir anders ist, müsstest du in Zeile 119ff die Interpretation der Winkel ändern. Beispiel: Sauger steht bei dir normalerweise rechts, dann 180 => 0, 270 => 90, 90 => 270, 0 => 180.
-
@kaiserm Hey! Vielen Dank für das Blocky! Finde ich eine tolle Funktion! Habe ich gestern auch mal direkt nachgebaut, allerdings funktioniert es bei mir leider nicht. Jetzt hoffe ich hier auf etwas Hilfe ...
Im Protokoll kommt die Meldung:09:35:03.244 info javascript.0 (1571) Start javascript script.js.common.Leere_den_Staubbehälter 09:35:03.247 info javascript.0 (1571) script.js.common.Leere_den_Staubbehälter: registered 1 subscription and 0 schedules
Der Roborock fährt weder los noch gibt die Alexa den Speak-Befehl aus. Allerdings gibt Sie als Antwort auf meine Spracheingabe die Bestätigung "ok" bzw. den "Pieps" aus.
Das Blocky scheint mir identisch nachgebaut zu sein. Aus folgenden Pfaden kommen die Variabeln:
-
summary: Alexa2/0/History/summary
-
speak: Alexa2/0/Echo-Devices/Geräte-NUMMER/Commands/speak
-
go to Point: Mihome-vacuum/0/control/goto
-
Die Koordianten kommen aus Flolevac (Vgl. @AlCalzone )
Screenshot Blocky mit Protokoll:
Oder muss ich noch etwas ganz anderes machen, bzw. haben damit das Script überhaupt funktioniert? Bin leider ein Noob, daher komme ich leider nicht weiter ...
Was ich auch noch probiert habe: Ich habe dem Script einen "smartName" gegeben, damit dieses im Cloud-Adapter als "Smart Gerät" auftaucht und dann in den Alexa-Routinen per Sprachsteuerung integrierbar wird. Auch um synonyme Befehle definieren zu können. Ist das richtig, bzw. notwendig mit dem Blocky-Script? Immerhin ist doch der speak-Befehl eine Bedingung im Script ...
Könnte mir bitte jemand helfen?
-
-
@AlCalzone Sauber! Das Script funktioniert nun auch bei mir hervorragend! Besten Dank!
-
@AlCalzone sagte in Roborock S5 "cleaning area" per Tastendruck?:
@kaiserm Also wie versprochen:
Muss als TypeScript angelegt werden und benötigt das zusätzliche Paket "axios":
Bekomme bei mir folgende Meldung:
javascript.0 2019-12-07 00:24:41.947 warn ERROR: Cannot find module 'axios'. javascript.0 2019-12-07 00:24:41.947 warn ^ javascript.0 2019-12-07 00:24:41.947 warn import axios from "axios"; javascript.0 2019-12-07 00:24:41.947 warn (2094) script.js.common.4_MiRobot.MiRobot_Zone_neu: TypeScript compilation had errors:
Habe im JS Instanz das Paket axios hinzugefügt, aber die Meldung kommt trotzdem noch. Muss ich das noch irgendwo anders eintragen?
Gibt es einen Grund warum der Sauger erst zu einem Spot fährt, bevor die Zonenreinigung startet?
-
@0018 sagte in Roborock S5 "cleaning area" per Tastendruck?:
Habe im JS Instanz das Paket axios hinzugefügt, aber die Meldung kommt trotzdem noch. Muss ich das noch irgendwo anders eintragen?
Ich bin mir nicht sicher, ob der aktuelle Adapter damit schon released ist. Wenn es die Option "Activate syntax help for these npm modules:" gibt, trag da auch noch axios ein. Ansonsten folgende Zeile über den Import schreiben:
// @ts-ignore
Gibt es einen Grund warum der Sauger erst zu einem Spot fährt, bevor die Zonenreinigung startet?
Ja, das ist leiser als wenn er im Reinigungs-Modus in die Zone fährt.
-
Hey @AlCalzone kann man dein Script denn auch so anpassen/erweitern, dass es zusätzlich zu den Räumen auch Positionen gibt, die nur angefahren werden könnten? Um z.B. den Staubbehälter vor dem Mülleimer leeren zu können? "Fahre Staubsauger zum Mülleimer"
Zudem wäre es denke ich auch interessant wenn man die Saugstufen in Abhängigkeit machen könnten. Im Bad könnte ich beispielsweise immer eine starke Stufe vertragen da hier unser Katzenklo steht. Oder kann man es in Abhängikeit vom Raum so einstellen, dass der ZoneCleanup auch mehrfach ausgeführt wird? Über ein paar Scripterweiterungen würde ich micht mega freuen -
@Mor9oth Gute Ideen. Eventuell komme ich morgen dazu - aber keine Garantie!
-
@AlCalzone Freut mich sehr
-
@AlCalzone sagte in Roborock S5 "cleaning area" per Tastendruck?:
Beispiel: Sauger steht bei dir normalerweise rechts, dann 180 => 0, 270 => 90, 90 => 270, 0 => 180.Ja genau, mein Sauger steht rechts vom Dock. Habe jetzt nachdem ich die Daten angepasst habe, leider das Problem das meine Bereiche nicht mehr mit den Koordinaten aus der Valetudo Map
übereinstimmen. Muss ich hier auch noch was drehen? Den die Zonenkordination sind ja dann auch gedreht oder? -
@0018 Inwiefern stimmt es nicht mehr überein?
-
@AlCalzone Damit das ganze besser zu erklären ist hier mit Screenshots.
-
So sieht meine Karte in Valetudo und die dazugehörigen Koordinaten der Zone1 aus:
-
Aus meiner Sicht steht mein Robo rechts vom Dock, daher folgende Anpassung deines Scriptes:
const rooms: Record<string, Rectangle[]> = { "Test": [[26397, 22791, 29302, 24138]], };
ab Zeile 119:
// Valetudo zählt von oben links nach unten rechts, d.h. die Y-Koordinaten // sind entgegengesetzt der mathematischen Definition let c2r = [robot[0] - charger[0], charger[1] - robot[1]]; const angle = Math.atan2(c2r[1], c2r[0]) * 180 / Math.PI; log("angle = " + angle); if (angle <= 45 && angle >= -45) { log("Sauger steht rechts"); return 0; } else if (angle > 45 && angle < 135) { log("Sauger steht oben"); return 90; } else if (angle < -45 && angle > -135) { log("Sauger steht unten"); return 270; } else { return 180; } }
-
Wenn ich dann die Zonnenreinigung starte, sieht das auf der Vaeltuda Map dann wie folgt aus:
-
Als test mal mit dem Standard:
// Valetudo zählt von oben links nach unten rechts, d.h. die Y-Koordinaten // sind entgegengesetzt der mathematischen Definition let c2r = [robot[0] - charger[0], charger[1] - robot[1]]; const angle = Math.atan2(c2r[1], c2r[0]) * 180 / Math.PI; log("angle = " + angle); if (angle <= 45 && angle >= -45) { log("Sauger steht rechts"); return 180; } else if (angle > 45 && angle < 135) { log("Sauger steht oben"); return 270; } else if (angle < -45 && angle > -135) { log("Sauger steht unten"); return 90; } else { return 0; } }
ergibt folgendes Ergebniss:
Habe auch schon 90 + 270 probiert, leider auch ohne Erfolg.
Im Script habe ich mal den Angle mit geloggt und bei den Versuchen kommt immer ähnliches raus....javascript.0 2019-12-09 21:12:16.299 info (2094) script.js.common.4_MiRobot.MiRobot_Zone_neu: angle = -2.981461219982192 javascript.0 2019-12-09 21:06:13.153 info (2094) script.js.common.4_MiRobot.MiRobot_Zone_neu: angle = 2.8912695962205643 javascript.0 2019-12-09 20:56:30.228 info (2094) script.js.common.4_MiRobot.MiRobot_Zone_neu: angle = -1.461310090881229
Vielleicht siehst du ja auf Anhieb woran es liegt?
-
-
@0018 sagte in Roborock S5 "cleaning area" per Tastendruck?:
Vielleicht siehst du ja auf Anhieb woran es liegt?
Auf dem 1. Bild im Dock sieht es für mich so aus als zeigt der Sauger nach links, d.h. Winkel 0°. Hast du die Koordinaten direkt aus Valetudo? Dann müssen ggf. die Y-Koordinaten aller Zonen invertiert werden, da Valetudo anscheinend anders zählt als der Sauger selbst:
y(für Skript) = 51200 - y(Valetudo)
-
@AlCalzone Ja, die Koordinaten sind direkt aus Valetudo unter http://IPdesSaugers/zone
Habe jetzt die Y-Koordinaten nach Vorgabe umgerechnet und jetzt stimmen die Zonen überein.
Der Tipp mit// @ts-ignore
hat auch funktioniert. Scheint also auf den ersten Blick erstmal alles zu passen. Ich werde es jetzt die nächsten Wochen mal testen.
Vielen Dank!
-
@AlCalzone Erstmal danke für dein Script
Leider scheint es bei mir nicht zu funktionieren, laut Log bleibt es bei
script.js.Xiaomi_Sauger.Zonenreinigung_Kartenrotation: Teste Kartenorientierung... stehen.
Ich nutze den mihome-vacuum 1.9.2 mit Valetudo RE 0.8.1, der Datenpunkt mapRotated bleibt auf unkown(-1). -
@Chris_71 Hast du die Zugangsdaten und Hostnamen bzw. IP korrekt eingetragen?
Kannst du mit diesen Zugangsdaten im Browser die folgende Adresse öffnen?http://<Adresse-Des-Saugers>/api/map/latest
-
@Chris_71 sagte in Roborock S5 "cleaning area" per Tastendruck?:
Valetudo RE 0.8.1
Ich glaube das liegt an Valetudo RE 0.8.1. Dort bekommt man über /api/map/latest nichts mehr angezeigt, sondern es wird eine Datei ohne Endung heruntergeladen. Der Inhalt ist mit dem Notepad++ nicht lesbar.
Seit meinem Update von Valetudo auf Valetudo RE 0.8.1 habe ich das gleiche Problem. Siehe auch hier von @base im Beitrag hier
-
@0018 Kannst du vielleicht mal im Browser schauen, welche Adresse diese Version lädt, um an die Map zu kommen?
-
@AlCalzone Da passiert das gleiche das auch @0018 geschrieben hat.
Eine Karte kann man über /api/simple_map abrufen (PNG-Datei). -
@Chris_71 dann muss man die Karte anders decoden. Ich brauche nicht die Pixel sondern die Koordinaten von sauger und Station um die rotation zu erkennen
@Meistertr hast du ne Idee wie sich das genau geändert hat?
-
@AlCalzone: Hey! Ich hoffe, du kannst mir helfen! Ich nutze dein Script und plötzlich, seit ca. einer Woche, funktioniert es nicht mehr. Ich bekomme plötzlich im Log:
State value to set for "mihome-vacuum.0.info.water_box" has to be type "string" but received type "boolean"
sobald ich einen der Räume ansteuere. (per Sprache oder per Alexa (Gerät/Button).
Geändert habe ich nichts. Die relevanten Adapter scheinen auch alle richtig angebunden zu sein. Siehe Screenshot:
Die Verbindung zum Roborock habe ich mit dem "Find" erfolgreich getestet. Muss ich was am script verändern, um die Raumsteuerung wieder zum Laufen zu bringen?
Dein Script lief bei mir Jahre so:
// Hier den State eintragen, der den aktuellen Sauger-Status angibt const idVacuumState = "mihome-vacuum.0.info.state"; // Koordiaten zählen von unten links (0,0) nach oben rechts (51200,51200) // Die folgenden Koordinaten gehen von nicht rotierter Karte (Winkel 0) aus const rooms: Record<string, Rectangle[]> = { "Küche": [[21386,18786,23486,22936]], "Flur": [[18323,22873,23623,24523]], "Wohnzimmer": [[23470,18841,27620,24341]], "Schlafzimmer": [[18235,24479,22285,28829]], "Bad": [[18273,19241,20723,23091]], "Arbeitszimmer": [[22413,24357,27463,28657]], "Wohnbereich": [[23470,18841,27620,24341],[21386,18786,23486,22936]], "Wohnung": [[23470,18841,27620,24341],[21386,18786,23486,22936],[18323,22873,23623,24523],[18273,19241,20723,23091],[18235,24479,22285,28829]] }; // =============================================================================== type Rectangle = [number, number, number, number]; for (const room of Object.keys(rooms) as (keyof typeof rooms)[]) { createState(`Staubsauger.${room}`, { type: "boolean", read: true, write: true, role: "switch", name: `${room} saugen`, }); on({ id: `javascript.${instance}.Staubsauger.${room}`, val: true, ack: false }, async () => { if (getState(idVacuumState).val !== 8) await cancelCurrentAction(); cleanRoom(room); }); } createState(`Staubsauger.stop`, { type: "boolean", read: true, write: true, role: "switch", name: `Staubsauger anhalten`, }); on({ id: `javascript.${instance}.Staubsauger.stop`, val: true, ack: false }, () => { stopCleanup(); }); // Reset all control states when the vacuum starts charging on({ id: idVacuumState, val: 8 /* charging */ }, (obj) => { for (const room of Object.keys(rooms) as (keyof typeof rooms)[]) { setState(`Staubsauger.${room}`, false, true); } setState(`Staubsauger.stop`, false, true); }); async function cancelCurrentAction(): Promise<void> { setState("mihome-vacuum.0.control.pause", true); // wait for the "paused" status before going home await waitForPauseOrSleep(); } async function stopCleanup(): Promise<void> { log(`Saugvorgang abgebrochen!`); setState(`Staubsauger.stop`, true, true); if (getState(idVacuumState).val !== 8) { await cancelCurrentAction(); await setStateAsync('mihome-vacuum.0.control.home', true); // wait for the "charging" status before resolving await waitFor(idVacuumState, 8); } log(`Staubsauger ist in der Basis`); setState(`Staubsauger.stop`, false, true); } async function cleanRoom(room: keyof typeof rooms): Promise<void> { log(`Saugvorgang für ${room} gestartet!`); setState(`Staubsauger.${room}`, true, true); const roomCoords = rooms[room]; const coords = roomCoords[0]; const targetCoords = [ ((coords[0] + coords[2]) / 2).toFixed(0), ((coords[1] + coords[3]) / 2).toFixed(0), ] // go to center of first zone const gotoString = targetCoords.join(","); await setStateAsync("mihome-vacuum.0.control.goTo", gotoString); log(`Fahre zur Mitte von ${room}`); await wait(10000); await waitForPauseOrSleep(); if (getState("Staubsauger.stop").val) return; const zoneCleanString = roomCoords.map(zone => { return "[" + zone.concat(1).map(coord => coord.toString()).join(",") + "]"; }).join(","); log("Starte Zonenreinigung..."); await setStateAsync("mihome-vacuum.0.control.zoneClean", zoneCleanString); // wait for the cleanup to finish await waitFor(idVacuumState, 6); } async function waitForPauseOrSleep(): Promise<void> { log("Warte auf Zustand schlafen oder Pause..."); switch (getState(idVacuumState).val) { case 3: case 10: log(" => Zustand bereits aktiv!"); return; default: await Promise.race([ waitFor(idVacuumState, 10), waitFor(idVacuumState, 3), ]); log(" => Zustand erreicht!"); } } function wait(ms: number): Promise<void> { return new Promise(resolve => { setTimeout(resolve, ms); }); } function waitFor(stateID: string, value: any): Promise<void> { return new Promise(resolve => { const handler = (obj: iobJS.ChangedStateObject) => { if (obj.newState.val === value) { unsubscribe(handler); resolve(); } } subscribe(stateID, handler); }) } function setStateAsync(id: string, state: any): Promise<void> { return new Promise(res => { setState(id, state, () => res()); }); }