import asyncio from rich.text import Text from textual.app import ComposeResult from textual.screen import ModalScreen from textual.widgets import RichLog, Button, Static from textual.containers import Vertical, Horizontal class OperationLog(ModalScreen[None]): BINDINGS = [("escape", "close", "Close")] def __init__(self, title: str = "Operation Output"): super().__init__() self._title = title self._running = True def compose(self) -> ComposeResult: with Vertical(id="op-log"): with Horizontal(id="ol-header"): yield Static(self._title, id="ol-title") yield Static("⏳", id="ol-spinner") yield RichLog(id="ol-log", wrap=True, highlight=True, markup=True) yield Button("Close", variant="primary", id="ol-close") def on_button_pressed(self, event: Button.Pressed) -> None: self.dismiss(None) def action_close(self) -> None: self.dismiss(None) def _write_to_log(self, log: RichLog, text: str) -> None: if "[" in text and "[/" in text: log.write(Text.from_markup(text)) else: log.write(text) def finish(self) -> None: self._running = False try: self.query_one("#ol-spinner", Static).update("✅") except Exception: pass def write(self, text: str) -> None: try: log = self.query_one("#ol-log", RichLog) self._write_to_log(log, text) except Exception: pass