2019-11-30 01:27:40 +00:00
|
|
|
#!/usr/bin/python3
|
|
|
|
import threading
|
|
|
|
import time
|
|
|
|
import PySimpleGUI as sg
|
|
|
|
|
|
|
|
"""
|
|
|
|
DESIGN PATTERN - Multithreaded Long Tasks GUI using shared global variables
|
|
|
|
Presents one method for running long-running operations in a PySimpleGUI environment.
|
|
|
|
The PySimpleGUI code, and thus the underlying GUI framework, runs as the primary, main thread
|
|
|
|
The "long work" is contained in the thread that is being started.
|
|
|
|
|
|
|
|
The PySimpleGUI code is structured just like a typical PySimpleGUI program. A layout defined,
|
|
|
|
a Window is created, and an event loop is executed.
|
|
|
|
"""
|
|
|
|
|
|
|
|
thread_done = 1
|
|
|
|
message = ''
|
|
|
|
thread_counter = 0
|
2019-11-30 01:53:26 +00:00
|
|
|
total = 100
|
|
|
|
progress = 0
|
2019-11-30 01:27:40 +00:00
|
|
|
|
|
|
|
def long_operation_thread(seconds):
|
|
|
|
"""
|
|
|
|
A worker thread that communicates with the GUI through a global variable
|
|
|
|
This thread can block for as long as it wants and the GUI will not be affected
|
|
|
|
:param seconds: (int) How long to sleep, the ultimate blocking call
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
|
2019-11-30 01:53:26 +00:00
|
|
|
global thread_done, message, thread_counter, progress
|
2019-11-30 01:27:40 +00:00
|
|
|
|
|
|
|
print('Starting thread - will sleep for {} seconds'.format(seconds))
|
|
|
|
thread_counter += 1
|
2019-11-30 01:53:26 +00:00
|
|
|
for i in range(int(seconds*10)):
|
|
|
|
time.sleep(.1) # sleep for a while
|
|
|
|
progress += total/(seconds*10)
|
|
|
|
|
2019-11-30 01:27:40 +00:00
|
|
|
message = f'***This is a message from the thread {thread_counter} ***'
|
|
|
|
thread_done = True
|
|
|
|
|
|
|
|
|
|
|
|
def the_gui():
|
|
|
|
"""
|
|
|
|
Starts and executes the GUI
|
|
|
|
Reads data from a global variable and displays
|
|
|
|
Returns when the user exits / closes the window
|
|
|
|
"""
|
2019-11-30 01:53:26 +00:00
|
|
|
global thread_done, message, progress
|
2019-11-30 01:27:40 +00:00
|
|
|
|
|
|
|
sg.change_look_and_feel('Light Brown 3')
|
|
|
|
|
|
|
|
layout = [[sg.Text('Long task to perform example')],
|
|
|
|
[sg.Output(size=(70, 12))],
|
|
|
|
[sg.Text('Number of seconds your task will take'),
|
|
|
|
sg.Input(key='-SECONDS-', size=(5, 1)),
|
|
|
|
sg.Button('Do Long Task', bind_return_key=True)],
|
2019-11-30 01:53:26 +00:00
|
|
|
[sg.Text('Work progress'), sg.ProgressBar(total, size=(20,20), orientation='h', key='-PROG-')],
|
2019-11-30 01:27:40 +00:00
|
|
|
[sg.Button('Click Me'), sg.Button('Exit')], ]
|
|
|
|
|
|
|
|
window = sg.Window('Multithreaded Window', layout)
|
|
|
|
|
|
|
|
# --------------------- EVENT LOOP ---------------------
|
|
|
|
while True:
|
|
|
|
event, values = window.read(timeout=100)
|
|
|
|
if event in (None, 'Exit'):
|
|
|
|
break
|
|
|
|
elif event.startswith('Do'):
|
|
|
|
seconds = float(values['-SECONDS-'])
|
|
|
|
print('Thread ALIVE! Long work....sending value of {} seconds'.format(seconds))
|
|
|
|
threading.Thread(target=long_operation_thread, args=(seconds, ), daemon=True).start()
|
|
|
|
elif event == 'Click Me':
|
|
|
|
print('Your GUI is alive and well')
|
|
|
|
# --------------- Check for incoming messages from threads ---------------
|
|
|
|
if thread_done is True:
|
|
|
|
print('The thread has finished!')
|
|
|
|
print(f'message = {message}')
|
2019-11-30 01:53:26 +00:00
|
|
|
# reset everything for the next run
|
2019-11-30 01:27:40 +00:00
|
|
|
thread_done = False
|
|
|
|
message = ''
|
2019-11-30 01:53:26 +00:00
|
|
|
progress = 0
|
|
|
|
window['-PROG-'].update_bar(total, total) # show the bar as maxed out
|
|
|
|
if progress != 0:
|
|
|
|
window['-PROG-'].update_bar(progress, total) # update the progress bar if non-zero
|
2019-11-30 01:27:40 +00:00
|
|
|
|
|
|
|
# if user exits the window, then close the window and exit the GUI func
|
|
|
|
window.close()
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
the_gui()
|
|
|
|
print('Exiting Program')
|