2018-09-27 20:24:09 +00:00
|
|
|
#!/usr/bin/env python
|
2019-10-23 20:10:03 +00:00
|
|
|
import PySimpleGUI as sg
|
2018-09-15 17:42:53 +00:00
|
|
|
from time import sleep
|
2018-09-24 22:01:00 +00:00
|
|
|
|
2020-05-07 10:22:59 +00:00
|
|
|
from docutils.nodes import title
|
|
|
|
|
2019-12-24 23:52:47 +00:00
|
|
|
sg.theme('Dark Blue 3')
|
2018-09-15 17:42:53 +00:00
|
|
|
|
|
|
|
"""
|
2019-10-23 20:10:03 +00:00
|
|
|
Demonstration of simple and multiple one_line_progress_meter's as well as the Progress Meter Element
|
2019-06-11 00:22:57 +00:00
|
|
|
|
2019-06-20 00:31:12 +00:00
|
|
|
There are 4 demos
|
2019-06-11 00:22:57 +00:00
|
|
|
1. Manually updated progress bar
|
|
|
|
2. Custom progress bar built into your window, updated in a loop
|
2019-10-23 20:10:03 +00:00
|
|
|
3. one_line_progress_meters, nested meters showing how 2 can be run at the same time.
|
|
|
|
4. An "iterable" style progress meter - a wrapper for one_line_progress_meters
|
2018-09-15 17:42:53 +00:00
|
|
|
|
|
|
|
If the software determined that a meter should be cancelled early,
|
|
|
|
calling OneLineProgresMeterCancel(key) will cancel the meter with the matching key
|
|
|
|
"""
|
|
|
|
|
2018-09-18 16:05:44 +00:00
|
|
|
|
|
|
|
"""
|
2019-06-11 00:22:57 +00:00
|
|
|
The simple case is that you want to add a single meter to your code. The one-line solution.
|
2019-10-23 20:10:03 +00:00
|
|
|
This demo function shows 3 different one_line_progress_meter tests
|
2019-06-11 00:22:57 +00:00
|
|
|
1. A horizontal with red and white bar colors
|
|
|
|
2. A vertical bar with default colors
|
|
|
|
3. A test showing 2 running at the same time
|
2018-09-18 16:05:44 +00:00
|
|
|
"""
|
|
|
|
|
2019-10-23 20:10:03 +00:00
|
|
|
|
2019-06-11 00:15:37 +00:00
|
|
|
def demo_one_line_progress_meter():
|
2018-09-28 23:21:44 +00:00
|
|
|
# Display a progress meter. Allow user to break out of loop using cancel button
|
2019-06-11 00:15:37 +00:00
|
|
|
for i in range(10000):
|
2019-10-23 20:10:03 +00:00
|
|
|
if not sg.one_line_progress_meter('My 1-line progress meter',
|
|
|
|
i+1, 10000,
|
|
|
|
'meter key',
|
|
|
|
'MY MESSAGE1',
|
|
|
|
'MY MESSAGE 2',
|
|
|
|
orientation='h',
|
2020-05-07 10:22:59 +00:00
|
|
|
no_titlebar=True,
|
|
|
|
grab_anywhere=True,
|
2019-10-23 20:10:03 +00:00
|
|
|
bar_color=('white', 'red')):
|
2019-06-11 00:15:37 +00:00
|
|
|
print('Hit the break')
|
|
|
|
break
|
|
|
|
for i in range(10000):
|
2019-10-23 20:10:03 +00:00
|
|
|
if not sg.one_line_progress_meter('My 1-line progress meter',
|
|
|
|
i+1, 10000,
|
|
|
|
'meter key',
|
|
|
|
'MY MESSAGE1',
|
|
|
|
'MY MESSAGE 2',
|
|
|
|
orientation='v'):
|
2019-06-11 00:15:37 +00:00
|
|
|
print('Hit the break')
|
2018-09-28 23:21:44 +00:00
|
|
|
break
|
2018-09-18 16:05:44 +00:00
|
|
|
|
2018-09-28 23:21:44 +00:00
|
|
|
layout = [
|
2019-10-23 20:10:03 +00:00
|
|
|
[sg.Text('One-Line Progress Meter Demo', font=('Any 18'))],
|
|
|
|
|
|
|
|
[sg.Text('Outer Loop Count', size=(15, 1), justification='r'),
|
|
|
|
sg.Input(default_text='100', size=(5, 1), key='CountOuter'),
|
|
|
|
sg.Text('Delay'), sg.Input(default_text='10', key='TimeOuter', size=(5, 1)), sg.Text('ms')],
|
|
|
|
|
|
|
|
[sg.Text('Inner Loop Count', size=(15, 1), justification='r'),
|
|
|
|
sg.Input(default_text='100', size=(5, 1), key='CountInner'),
|
|
|
|
sg.Text('Delay'), sg.Input(default_text='10', key='TimeInner', size=(5, 1)), sg.Text('ms')],
|
2018-09-18 16:05:44 +00:00
|
|
|
|
2019-10-23 20:10:03 +00:00
|
|
|
[sg.Button('Show', pad=((0, 0), 3), bind_return_key=True),
|
|
|
|
sg.Text('me the meters!')]
|
|
|
|
]
|
|
|
|
|
|
|
|
window = sg.Window('One-Line Progress Meter Demo', layout)
|
2018-09-18 16:05:44 +00:00
|
|
|
|
2018-09-28 23:21:44 +00:00
|
|
|
while True:
|
2019-10-23 20:10:03 +00:00
|
|
|
event, values = window.read()
|
2020-05-07 10:22:59 +00:00
|
|
|
if event == sg.WIN_CLOSED:
|
2018-09-28 23:21:44 +00:00
|
|
|
break
|
2018-10-15 20:07:23 +00:00
|
|
|
if event == 'Show':
|
2018-09-28 23:21:44 +00:00
|
|
|
max_outer = int(values['CountOuter'])
|
|
|
|
max_inner = int(values['CountInner'])
|
|
|
|
delay_inner = int(values['TimeInner'])
|
|
|
|
delay_outer = int(values['TimeOuter'])
|
|
|
|
for i in range(max_outer):
|
2019-10-23 20:10:03 +00:00
|
|
|
if not sg.one_line_progress_meter('Outer Loop', i+1, max_outer, 'outer'):
|
2018-09-28 23:21:44 +00:00
|
|
|
break
|
|
|
|
sleep(delay_outer/1000)
|
|
|
|
for j in range(max_inner):
|
2019-10-23 20:10:03 +00:00
|
|
|
if not sg.one_line_progress_meter('Inner Loop', j+1, max_inner, 'inner'):
|
2018-09-28 23:21:44 +00:00
|
|
|
break
|
|
|
|
sleep(delay_inner/1000)
|
2019-10-23 20:10:03 +00:00
|
|
|
window.close()
|
2018-09-15 17:42:53 +00:00
|
|
|
|
2019-06-11 00:15:37 +00:00
|
|
|
|
|
|
|
'''
|
|
|
|
Manually Updated Test
|
|
|
|
Here is an example for when you want to "sprinkle" progress bar updates in multiple
|
|
|
|
places within your source code and you're not running an event loop.
|
|
|
|
Note that UpdateBar is special compared to other Update methods. It also refreshes
|
|
|
|
the containing window and checks for window closure events
|
|
|
|
The sleep calls are here only for demonstration purposes. You should NOT be adding
|
|
|
|
these kinds of sleeps to a GUI based program normally.
|
|
|
|
'''
|
|
|
|
|
2019-10-23 20:10:03 +00:00
|
|
|
|
2019-06-11 00:15:37 +00:00
|
|
|
def manually_updated_meter_test():
|
|
|
|
# layout the form
|
|
|
|
layout = [[sg.Text('This meter is manually updated 4 times')],
|
2019-10-23 20:10:03 +00:00
|
|
|
[sg.ProgressBar(max_value=10, orientation='h', size=(20, 20), key='progress')]]
|
2019-06-11 00:15:37 +00:00
|
|
|
|
|
|
|
# create the form`
|
2019-10-23 20:10:03 +00:00
|
|
|
# must finalize since not running an event loop
|
|
|
|
window = sg.Window('Custom Progress Meter', layout, finalize=True)
|
2019-06-11 00:22:57 +00:00
|
|
|
|
2019-10-23 20:10:03 +00:00
|
|
|
# Get the element to make updating easier
|
|
|
|
progress_bar = window['progress']
|
2019-06-11 00:15:37 +00:00
|
|
|
|
|
|
|
# -------------------- Your Program Code --------------------
|
|
|
|
# Spot #1 to indicate progress
|
2019-10-23 20:10:03 +00:00
|
|
|
progress_bar.update_bar(1) # show 10% complete
|
2019-06-11 00:15:37 +00:00
|
|
|
sleep(2)
|
|
|
|
|
|
|
|
# more of your code.... perhaps pages and pages of code.
|
|
|
|
# Spot #2 to indicate progress
|
2019-10-23 20:10:03 +00:00
|
|
|
progress_bar.update_bar(2) # show 20% complete
|
2019-06-11 00:15:37 +00:00
|
|
|
sleep(2)
|
|
|
|
|
|
|
|
# more of your code.... perhaps pages and pages of code.
|
|
|
|
# Spot #3 to indicate progress
|
2019-10-23 20:10:03 +00:00
|
|
|
progress_bar.update_bar(6) # show 60% complete
|
2019-06-11 00:15:37 +00:00
|
|
|
sleep(2)
|
|
|
|
|
|
|
|
# more of your code.... perhaps pages and pages of code.
|
|
|
|
# Spot #4 to indicate progress
|
2019-10-23 20:10:03 +00:00
|
|
|
progress_bar.update_bar(9) # show 90% complete
|
2019-06-11 00:15:37 +00:00
|
|
|
sleep(2)
|
2019-10-23 20:10:03 +00:00
|
|
|
window.close()
|
2019-06-11 00:15:37 +00:00
|
|
|
|
|
|
|
|
2018-09-28 23:21:44 +00:00
|
|
|
'''
|
2019-06-11 00:15:37 +00:00
|
|
|
This function shows how to create a custom window with a custom progress bar and then
|
|
|
|
how to update the bar to indicate progress is being made
|
2018-09-28 23:21:44 +00:00
|
|
|
'''
|
2018-09-15 17:42:53 +00:00
|
|
|
|
2019-10-23 20:10:03 +00:00
|
|
|
|
2019-06-11 00:15:37 +00:00
|
|
|
def custom_meter_example():
|
2018-09-28 23:21:44 +00:00
|
|
|
# layout the form
|
2019-06-11 00:15:37 +00:00
|
|
|
layout = [[sg.Text('A typical custom progress meter')],
|
2019-10-23 20:10:03 +00:00
|
|
|
[sg.ProgressBar(1, orientation='h', size=(20, 20), key='progress')],
|
2018-09-28 23:21:44 +00:00
|
|
|
[sg.Cancel()]]
|
|
|
|
|
|
|
|
# create the form`
|
2019-10-23 20:10:03 +00:00
|
|
|
window = sg.Window('Custom Progress Meter', layout)
|
|
|
|
progress_bar = window['progress']
|
2018-09-28 23:21:44 +00:00
|
|
|
# loop that would normally do something useful
|
2019-06-11 00:15:37 +00:00
|
|
|
for i in range(10000):
|
2018-09-28 23:21:44 +00:00
|
|
|
# check to see if the cancel button was clicked and exit loop if clicked
|
2019-10-23 20:10:03 +00:00
|
|
|
event, values = window.read(timeout=0)
|
2018-10-29 00:01:03 +00:00
|
|
|
if event == 'Cancel' or event == None:
|
2018-09-28 23:21:44 +00:00
|
|
|
break
|
|
|
|
# update bar with loop value +1 so that bar eventually reaches the maximum
|
2019-10-23 20:10:03 +00:00
|
|
|
progress_bar.update_bar(i+1, 10000)
|
2018-09-28 23:21:44 +00:00
|
|
|
# done with loop... need to destroy the window as it's still open
|
2019-10-23 20:10:03 +00:00
|
|
|
window.close()
|
2018-09-15 17:42:53 +00:00
|
|
|
|
2019-06-11 00:15:37 +00:00
|
|
|
|
2019-06-20 00:31:12 +00:00
|
|
|
'''
|
2019-10-23 20:10:03 +00:00
|
|
|
A Wrapper for one_line_progress_meter that combines your iterable with a progress meter into a single interface. Two functions are provided
|
2019-06-20 00:56:50 +00:00
|
|
|
progess_bar - The basic wrapper
|
|
|
|
progress_bar_range - A "convienence function" if you wanted to specify like a range
|
2019-06-20 00:31:12 +00:00
|
|
|
'''
|
|
|
|
|
|
|
|
|
|
|
|
def progress_bar(key, iterable, *args, title='', **kwargs):
|
|
|
|
"""
|
|
|
|
Takes your iterable and adds a progress meter onto it
|
|
|
|
:param key: Progress Meter key
|
|
|
|
:param iterable: your iterable
|
|
|
|
:param args: To be shown in one line progress meter
|
|
|
|
:param title: Title shown in meter window
|
2019-10-23 20:10:03 +00:00
|
|
|
:param kwargs: Other arguments to pass to one_line_progress_meter
|
2019-06-20 00:31:12 +00:00
|
|
|
:return:
|
|
|
|
"""
|
2019-10-23 20:10:03 +00:00
|
|
|
sg.one_line_progress_meter(title, 0, len(iterable), key, *args, **kwargs)
|
2019-06-20 00:31:12 +00:00
|
|
|
for i, val in enumerate(iterable):
|
|
|
|
yield val
|
2019-10-23 20:10:03 +00:00
|
|
|
if not sg.one_line_progress_meter(title, i+1, len(iterable), key, *args, **kwargs):
|
2019-06-20 00:31:12 +00:00
|
|
|
break
|
|
|
|
|
|
|
|
|
2019-06-20 00:56:50 +00:00
|
|
|
def progress_bar_range(key, start, stop=None, step=1, *args, **kwargs):
|
2019-06-20 00:31:12 +00:00
|
|
|
"""
|
|
|
|
Acts like the range() function but with a progress meter built-into it
|
|
|
|
:param key: progess meter's key
|
|
|
|
:param start: low end of the range
|
|
|
|
:param stop: Uppder end of range
|
|
|
|
:param step:
|
|
|
|
:param args:
|
|
|
|
:param kwargs:
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
return progress_bar(key, range(start, stop, step), *args, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
# -------------------- Demo Usage --------------------
|
|
|
|
def demo_iterable_progress_bar():
|
2019-10-23 20:10:03 +00:00
|
|
|
# start with a list of 100 integers as the user's list
|
|
|
|
my_list = list(range(1000))
|
2019-06-20 00:31:12 +00:00
|
|
|
|
|
|
|
# first form takes an iterable and a key and will return a value from your iterable
|
|
|
|
# and bump the progress meter at the same time
|
2020-05-07 10:22:59 +00:00
|
|
|
for value in progress_bar('bar1', my_list, title='First bar Test'):
|
2019-06-20 00:31:12 +00:00
|
|
|
# do something useful with value, a value from your list.
|
|
|
|
print(value)
|
|
|
|
|
|
|
|
# Since the progress_bar is an iterator, you can use it within a list comprehension
|
2020-05-07 10:22:59 +00:00
|
|
|
my_list = [x for x in progress_bar('bar1', my_list, title='Second bar Test')]
|
2019-06-20 00:31:12 +00:00
|
|
|
|
2019-06-20 00:56:50 +00:00
|
|
|
|
2019-06-20 00:31:12 +00:00
|
|
|
demo_iterable_progress_bar()
|
2019-06-11 00:15:37 +00:00
|
|
|
manually_updated_meter_test()
|
|
|
|
custom_meter_example()
|
2019-10-23 20:10:03 +00:00
|
|
|
demo_one_line_progress_meter()
|