diff --git a/miio/__init__.py b/miio/__init__.py index d304229d5..f108e507c 100644 --- a/miio/__init__.py +++ b/miio/__init__.py @@ -42,6 +42,7 @@ from miio.pwzn_relay import PwznRelay from miio.toiletlid import Toiletlid from miio.vacuum import Vacuum, VacuumException +from miio.vacuum_tui import VacuumTUI from miio.vacuumcontainers import ( CleaningDetails, CleaningSummary, diff --git a/miio/vacuum_cli.py b/miio/vacuum_cli.py index 73c3b1943..916e8ab7a 100644 --- a/miio/vacuum_cli.py +++ b/miio/vacuum_cli.py @@ -231,6 +231,13 @@ def manual(vac: miio.Vacuum): # if not vac.manual_mode and command : +@manual.command() +@pass_dev +def tui(vac: miio.Vacuum): + """TUI for the manual mode.""" + miio.VacuumTUI.run(vac) + + @manual.command() @pass_dev def start(vac: miio.Vacuum): # noqa: F811 # redef of start diff --git a/miio/vacuum_tui.py b/miio/vacuum_tui.py new file mode 100644 index 000000000..e8d661f6a --- /dev/null +++ b/miio/vacuum_tui.py @@ -0,0 +1,95 @@ +try: + import curses +except ImportError: + pass + +import enum +from typing import Tuple, Type + +from .vacuum import Vacuum + + +class Control(enum.Enum): + + Quit = "q" + Forward = "l" + ForwardFast = "L" + Backward = "k" + BackwardFast = "K" + Left = "j" + LeftFast = "J" + Right = ";" + RightFast = ":" + + +class VacuumTUI: + def __init__(self, vac: Vacuum): + self.vac = vac + self.rot = 0 + self.rot_delta = 30 + self.rot_min = Vacuum.MANUAL_ROTATION_MIN + self.rot_max = Vacuum.MANUAL_ROTATION_MAX + self.vel = 0.0 + self.vel_delta = 0.1 + self.vel_min = Vacuum.MANUAL_VELOCITY_MIN + self.vel_max = Vacuum.MANUAL_VELOCITY_MAX + self.dur = 10 * 1000 + + @classmethod + def run(cls: Type["VacuumTUI"], vac: Vacuum) -> None: + vac.manual_start() + try: + tui = cls(vac=vac) + curses.wrapper(tui.main) + finally: + vac.manual_stop() + + def main(self, screen) -> None: + done = False + while not done: + key = screen.getkey() + text, done = self.handle(key) + screen.clear() + screen.addstr(text) + screen.refresh() + + def handle(self, key: str) -> Tuple[str, bool]: + try: + ctl = Control(key) + except ValueError as e: + return "ignoring %s: %s\n" % (key, e), False + + if ctl == Control.Quit: + return "exiting normally\n", True + + self.control(ctl) + return self.info(), False + + def control(self, ctl: Control) -> None: + if ctl == Control.Forward: + self.vel = min(self.vel + self.vel_delta, self.vel_max) + elif ctl == Control.ForwardFast: + self.vel = 0 if self.vel < 0 else self.vel_max + + elif ctl == Control.Backward: + self.vel = max(self.vel - self.vel_delta, self.vel_min) + elif ctl == Control.BackwardFast: + self.vel = 0 if self.vel > 0 else self.vel_min + + elif ctl == Control.Left: + self.rot = min(self.rot + self.rot_delta, self.rot_max) + elif ctl == Control.LeftFast: + self.rot = 0 if self.rot < 0 else self.rot_max + + elif ctl == Control.Right: + self.rot = max(self.rot - self.rot_delta, self.rot_min) + elif ctl == Control.RightFast: + self.rot = 0 if self.rot > 0 else self.rot_min + + else: + raise RuntimeError("unreachable") + + self.vac.manual_control(rotation=self.rot, velocity=self.vel, duration=self.dur) + + def info(self) -> str: + return f"rotation={self.rot}\nvelocity={self.vel}\n"