PySimpleGUI/DemoPrograms/Demo_Threaded_Work.py

118 lines
4.6 KiB
Python

#!/usr/bin/python3
import queue
import threading
import time
import PySimpleGUI as sg
"""
Demo on how to add a long-running item to your PySimpleGUI Event Loop
If you want to do something that takes a long time, and you do it in the
main event loop, you'll quickly begin to see messages from windows that your
program has hung, asking if you want to kill it.
The problem is not that your problem is hung, the problem is that you are
not calling Read or Refresh often enough.
One way through this, shown here, is to put your long work into a thread that
is spun off, allowed to work, and then gets back to the GUI when it's done working
on that task.
If you have multiple long tasks to run, then you'll want a more sophisticated
format to your messages going back to the GUI so you'll know which task finished
"""
# Put your....
###### ######## ## ##
## ## ## ## ## ##
## ## ## ## ##
## ######## ## ##
## ## ## ##
## ## ## ## ##
####### ## #######
#### ## ## ######## ######## ## ## ###### #### ## ## ########
## ### ## ## ## ### ## ## ## ## ## ## ##
## #### ## ## ## #### ## ## ## ## ## ##
## ## ## ## ## ###### ## ## ## ###### ## ## ## ######
## ## #### ## ## ## #### ## ## ## ## ##
## ## ### ## ## ## ### ## ## ## ## ## ##
#### ## ## ## ######## ## ## ###### #### ### ########
###### ####### ######## ########
## ## ## ## ## ## ##
## ## ## ## ## ##
## ## ## ## ## ######
## ## ## ## ## ##
## ## ## ## ## ## ##
####### ####### ######## ########
# Here in this thread
def worker_thread(thread_name, gui_queue):
print('Starting thread - {} '.format(thread_name))
# this is our "long running function call"
time.sleep(5) # sleep for a while
print('Ending thread - {} '.format(thread_name))
# at the end of the work, before exiting, send a message back to the GUI indicating end
# in this case, we're using a simple string
gui_queue.put('{} - done'.format(thread_name)) # put a message into queue for GUI
######## ## ## ####
## ## ## ## ##
## ## ## ##
## #### ## ## ##
## ## ## ## ##
## ## ## ## ##
######## ######### ####
def the_gui(gui_queue):
layout = [[sg.Text('Multithreaded Work Example')],
[sg.Text('', size=(25, 1), key='_OUTPUT_')],
# [sg.Output(size=(40,6))],
[sg.Button('Go'), sg.Button('Exit')], ]
window = sg.Window('Multithreaded Window').Layout(layout)
# --------------------- EVENT LOOP ---------------------
count = 0
while True:
event, values = window.Read(timeout=100) # wait for up to 100 ms for a GUI event
if event is None or event == 'Exit':
break
if event == 'Go': # clicking "Go" starts a long running work item by starting thread
window.Element('_OUTPUT_').Update('Starting long work %s'%count)
# STARTING long run by starting a thread
threading.Thread(target=worker_thread, args=('Thread %s'%count, gui_queue,), daemon=True).start()
count += 1
# --------------- Read next message coming in from threads ---------------
try:
message = gui_queue.get_nowait() # see if something has been posted to Queue
except queue.Empty: # get_nowait() will get exception when Queue is empty
message = None # nothing in queue so do nothing
# if message received from queue, display the message in the Window
if message is not None:
# this is the place you would execute code at ENDING of long running task
window.Element('_OUTPUT_').Update(message)
window.Refresh() # do a refresh because could be showing multiple messages before next Read
# if user exits the window, then close the window and exit the GUI func
window.Close()
## ## ### #### ## ##
### ### ## ## ## ### ##
#### #### ## ## ## #### ##
## ### ## ## ## ## ## ## ##
## ## ######### ## ## ####
## ## ## ## ## ## ###
## ## ## ## #### ## ##
if __name__ == '__main__':
# -- Create a Queue to communicate with GUI --
gui_queue = queue.Queue() # queue used to communicate between the gui and the threads
the_gui(gui_queue)
print('Exiting Program')