from textual.app import App, ComposeResult from textual.containers import Center, Middle from textual.timer import Timer from textual.widgets import Footer, ProgressBar class IndeterminateProgressBar(App[None]): BINDINGS = [("s", "start", "Start")] progress_timer: Timer """Timer to simulate progress happening.""" def compose(self) -> ComposeResult: with Center(): with Middle(): yield ProgressBar() yield Footer() def on_mount(self) -> None: """Set up a timer to simulate progess happening.""" self.progress_timer = self.set_interval(1 / 10, self.make_progress, pause=True) def make_progress(self) -> None: """Called automatically to advance the progress bar.""" self.query_one(ProgressBar).advance(1) def action_start(self) -> None: """Start the progress tracking.""" self.query_one(ProgressBar).update(total=100) self.progress_timer.resume() def key_f(self) -> None: # Freeze time for the indeterminate progress bar. self.query_one(ProgressBar).query_one("#bar")._get_elapsed_time = lambda: 5 def key_t(self) -> None: # Freeze time to show always the same ETA. self.query_one(ProgressBar).query_one("#eta")._get_elapsed_time = lambda: 3.9 self.query_one(ProgressBar).update(total=100, progress=39) def key_u(self) -> None: self.query_one(ProgressBar).update(total=100, progress=100) if __name__ == "__main__": IndeterminateProgressBar().run()