-
Hi, I think, my former post was to long to be read (sorry for that). I would like to know if there is a way to poke into an existing library function (source available) in order to display its progress in How do I have to combine the existing mechanisms (
Is this possible? Is there already a video describing this (I would think) common technical situation? |
Beta Was this translation helpful? Give feedback.
Replies: 7 comments 4 replies
-
When you instantiate a progress bar, you get a reactive If you prefer to count number of steps instead of percentages, there's also a Is this clear enough? # In your function that does the looping operation
pb = self.query("#your-awesome-progress-bar")
while doing_stuff:
progress_percentage = compute_progress()
pb.percentage = progress_percentage
# Do your own things |
Beta Was this translation helpful? Give feedback.
-
Hi @rodrigogiraoserrao , thanks for your quick answer. Unfortunately, this does not seem to work as easy as most of the examples demonstrate. In my case, there is no I would be happy if you could have a quick look at my former post, as it shows an example structure of a more complex setting. The goal is still just to be able to "monitor" the progress of a loop operation situated within another Python module. I don't want to bother you and the community, but I was hoping that Anyway, thanks. I might have a look at one of the real and bigger applications out there when I have more time. |
Beta Was this translation helpful? Give feedback.
-
I use callbacks fired from the thing doing the updating. The method containing your loop would have a param called Another option is to set up a watchdog loop that monitors the internal state of your application's progress (ie, check the object about once per second, and if the state changes, do something). I've actually used both of the above to equip an existing library/application with TUIs. |
Beta Was this translation helpful? Give feedback.
-
To achieve this, you must broadcast the progress of a certain program or module. Let's assume you have a logical processing unit called The first approach that comes to mind is to have class Tui(App):
def on_progress_changed(self, cur, total):
# Update progress as suggested by @rodrigogiraoserrao
pass
class Worker:
def reg_progress_handler(self, obj, fun):
fun(obj, cur, total) Alternatively, if you don't want from functools import partial
class EventHandler:
def __init__(self):
self.on_process_changed_handler = None
def set_on_process_changed(self, obj, fun):
self.on_process_changed_handler = partial(fun, obj)
def on_progress_changed(self, cur, total):
if self.on_process_changed_handler:
self.on_process_changed_handler(cur, total)
class Tui(App):
def __init__(self):
event_handler.set_on_process_changed(self, self.on_progress_changed)
super().__init__()
def on_progress_changed(self, cur, total):
# Update progress as suggested by @rodrigogiraoserrao
pass
class Worker:
# Call this function to update progress without depending on Tui
def handle_progress(self, cur, total):
event_handler.on_progress_changed(cur, total) For simplicity, I haven't included necessary safety checks (e.g., ensuring the callable is not Clearly, the second solution involves more code, but this is a small trade-off for achieving decoupling. |
Beta Was this translation helpful? Give feedback.
-
Hi @rodrigogiraoserrao , @comalice and @Smalldy . Thanky for your suggestions and ideas so far. Unfortunately, I still didn't manage to achieve what I intended. And I still think that my own brain just does not get the obvious... So, please have a look at the following sequence diagram: The hatched parts of the lifelines are those where the (single) processor does some work. Either for the GUI side or the LIB side. The module for "LIB" is included into the "GUI" module via
As I think, the essence will be to answer the question: "is is possible to stop/resume execution at will between modules"? can this be done? without asyncio? how else could this be done? |
Beta Was this translation helpful? Give feedback.
-
no "leave", no "resume". Just funtion call. A long work function will block your code. A short function will not block your code. |
Beta Was this translation helpful? Give feedback.
-
Hi @Toniolo-Marco . Sorry for the delay... I tried to write down a "template" showing how it worked for me (necessary includes are missing and the function name of the library function and other things would have to be adjusted to your needs): #
# application
#
class ExampleApp(App):
#
# compose the app GUI
#
def compose(self) -> ComposeResult:
#
# show progress bar
#
yield ProgressBar()
#
# show button
#
yield Button("button-label", id="button-id")
#
# show footer
#
yield Footer()
@on(Button.Pressed, "#"+"button-id")
@work(exclusive=True, thread=True)
def react_on_button_press_inside_worker_thread(self):
#
# get the widget to be updated
#
_widget = self.query_one(ProgressBar)
#
# prepare the loop operation
#
# get the worker object. this done is to check during the
# execution of the worker thread if it is still running. dunno,
# if this would really be necessary.
worker = get_current_worker()
# get number of iterations. normally, this should be a call
# into the library in order to determine the exact number
# of iterations to be performed within this worker thread
nIterations = 100
# set start conditions for widget
_widget.update(total=nIterations)
#
# loop around the library execution
#
# reset the iteration counter
_i = 0
# the list of items would result from a library call. but
# here, it is just a list of numbers. not real objects an would
# be the normal case
list_of_items = range(100)
# do the loop
for _item in list_of_items:
#
# update progress bar
#
if not worker.is_cancelled:
self.call_from_thread(
_widget.update, progress=_i,
)
#
# call lib function iteration
#
# here now the actual library function that has to
# do the job that is worth one progress bar tick.
sleep(1)
#library.evaluate_single_item(_item)
#
# increment item counter
#
_i = _i + 1
#
# show success message about files read
#
self.app.notify(f'successfully executed worker thread.')
#
# when started from CLI
#
if __name__ == "__main__":
# start gui
app = ExampleApp()
app.run() |
Beta Was this translation helpful? Give feedback.
Hi @Toniolo-Marco . Sorry for the delay...
I tried to write down a "template" showing how it worked for me (necessary includes are missing and the function name of the library function and other things would have to be adjusted to your needs):