Merge pull request #196 from MikeTheWatchGuy/Dev-latest
Added Enable/Disable to all input elements! Refresh of many Demo pro…
This commit is contained in:
commit
fe5c02f684
|
@ -8,15 +8,13 @@ states for buttons and changes the text color to show greyed-out (disabled) butt
|
||||||
sg.ChangeLookAndFeel('Dark')
|
sg.ChangeLookAndFeel('Dark')
|
||||||
sg.SetOptions(element_padding=(0,0))
|
sg.SetOptions(element_padding=(0,0))
|
||||||
|
|
||||||
StartButton = sg.ReadFormButton('Start', button_color=('white', 'black'))
|
|
||||||
StopButton = sg.ReadFormButton('Stop', button_color=('gray34','black'))
|
|
||||||
ResetButton = sg.ReadFormButton('Reset', button_color=('gray','firebrick3'))
|
|
||||||
SubmitButton = sg.ReadFormButton('Submit', button_color=('gray34','springgreen4'))
|
|
||||||
|
|
||||||
layout = [[sg.T('User:', pad=((3,0),0)), sg.OptionMenu(values = ('User 1', 'User 2'), size=(20,1)), sg.T('0', size=(8,1))],
|
layout = [[sg.T('User:', pad=((3,0),0)), sg.OptionMenu(values = ('User 1', 'User 2'), size=(20,1)), sg.T('0', size=(8,1))],
|
||||||
[sg.T('Customer:', pad=((3,0),0)), sg.OptionMenu(values=('Customer 1', 'Customer 2'), size=(20,1)), sg.T('1', size=(8,1))],
|
[sg.T('Customer:', pad=((3,0),0)), sg.OptionMenu(values=('Customer 1', 'Customer 2'), size=(20,1)), sg.T('1', size=(8,1))],
|
||||||
[sg.T('Notes:', pad=((3,0),0)), sg.In(size=(44,1), background_color='white', text_color='black')],
|
[sg.T('Notes:', pad=((3,0),0)), sg.In(size=(44,1), background_color='white', text_color='black')],
|
||||||
[StartButton, StopButton, ResetButton, SubmitButton]
|
[sg.ReadFormButton('Start', button_color=('white', 'black'), key='start'),
|
||||||
|
sg.ReadFormButton('Stop', button_color=('gray34', 'black'), key='stop'),
|
||||||
|
sg.ReadFormButton('Reset', button_color=('gray', 'firebrick3'), key='reset'),
|
||||||
|
sg.ReadFormButton('Submit', button_color=('gray34', 'springgreen4'), key='submit')]
|
||||||
]
|
]
|
||||||
|
|
||||||
form = sg.FlexForm("Time Tracker", default_element_size=(12,1), text_justification='r', auto_size_text=False, auto_size_buttons=False,
|
form = sg.FlexForm("Time Tracker", default_element_size=(12,1), text_justification='r', auto_size_text=False, auto_size_buttons=False,
|
||||||
|
@ -28,27 +26,27 @@ while True:
|
||||||
if button is None:
|
if button is None:
|
||||||
exit(69)
|
exit(69)
|
||||||
if button is 'Start':
|
if button is 'Start':
|
||||||
StartButton.Update(button_color=('gray34','black'))
|
form.FindElement('start').Update(button_color=('gray34','black'))
|
||||||
StopButton.Update(button_color=('white', 'black'))
|
form.FindElement('stop').Update(button_color=('white', 'black'))
|
||||||
ResetButton.Update(button_color=('white', 'firebrick3'))
|
form.FindElement('reset').Update(button_color=('white', 'firebrick3'))
|
||||||
recording = True
|
recording = True
|
||||||
elif button is 'Stop' and recording:
|
elif button is 'Stop' and recording:
|
||||||
StopButton.Update(button_color=('gray34','black'))
|
form.FindElement('stop').Update(button_color=('gray34','black'))
|
||||||
StartButton.Update(button_color=('white', 'black'))
|
form.FindElement('start').Update(button_color=('white', 'black'))
|
||||||
SubmitButton.Update(button_color=('white', 'springgreen4'))
|
form.FindElement('submit').Update(button_color=('white', 'springgreen4'))
|
||||||
recording = False
|
recording = False
|
||||||
have_data = True
|
have_data = True
|
||||||
elif button is 'Reset':
|
elif button is 'Reset':
|
||||||
StopButton.Update(button_color=('gray34','black'))
|
form.FindElement('stop').Update(button_color=('gray34','black'))
|
||||||
StartButton.Update(button_color=('white', 'black'))
|
form.FindElement('start').Update(button_color=('white', 'black'))
|
||||||
SubmitButton.Update(button_color=('gray34', 'springgreen4'))
|
form.FindElement('submit').Update(button_color=('gray34', 'springgreen4'))
|
||||||
ResetButton.Update(button_color=('gray34', 'firebrick3'))
|
form.FindElement('reset').Update(button_color=('gray34', 'firebrick3'))
|
||||||
recording = False
|
recording = False
|
||||||
have_data = False
|
have_data = False
|
||||||
elif button is 'Submit' and have_data:
|
elif button is 'Submit' and have_data:
|
||||||
StopButton.Update(button_color=('gray34','black'))
|
form.FindElement('stop').Update(button_color=('gray34','black'))
|
||||||
StartButton.Update(button_color=('white', 'black'))
|
form.FindElement('start').Update(button_color=('white', 'black'))
|
||||||
SubmitButton.Update(button_color=('gray34', 'springgreen4'))
|
form.FindElement('submit').Update(button_color=('gray34', 'springgreen4'))
|
||||||
ResetButton.Update(button_color=('gray34', 'firebrick3'))
|
form.FindElement('reset').Update(button_color=('gray34', 'firebrick3'))
|
||||||
recording = False
|
recording = False
|
||||||
|
|
||||||
|
|
|
@ -109,25 +109,34 @@ def AllWidgetsWithContext():
|
||||||
# sg.ChangeLookAndFeel('GreenTan')
|
# sg.ChangeLookAndFeel('GreenTan')
|
||||||
|
|
||||||
with sg.FlexForm('Everything bagel', default_element_size=(40, 1), grab_anywhere=False) as form:
|
with sg.FlexForm('Everything bagel', default_element_size=(40, 1), grab_anywhere=False) as form:
|
||||||
|
|
||||||
|
column1 = [[sg.Text('Column 1', background_color='#F7F3EC', justification='center', size=(10, 1))],
|
||||||
|
[sg.Spin(values=('Spin Box 1', '2', '3'), initial_value='Spin Box 1')],
|
||||||
|
[sg.Spin(values=('Spin Box 1', '2', '3'), initial_value='Spin Box 2')],
|
||||||
|
[sg.Spin(values=('Spin Box 1', '2', '3'), initial_value='Spin Box 3')]]
|
||||||
|
|
||||||
layout = [
|
layout = [
|
||||||
[sg.Text('All graphic widgets in one form!', size=(30, 1), font=("Helvetica", 25))],
|
[sg.Text('All graphic widgets in one form!', size=(30, 1), font=("Helvetica", 25))],
|
||||||
[sg.Text('Here is some text.... and a place to enter text')],
|
[sg.Text('Here is some text.... and a place to enter text')],
|
||||||
[sg.InputText()],
|
[sg.InputText('This is my text')],
|
||||||
[sg.Checkbox('My first checkbox!'), sg.Checkbox('My second checkbox!', default=True)],
|
[sg.Checkbox('Checkbox'), sg.Checkbox('My second checkbox!', default=True)],
|
||||||
[sg.Radio('My first Radio! ', "RADIO1", default=True), sg.Radio('My second Radio!', "RADIO1")],
|
[sg.Radio('My first Radio! ', "RADIO1", default=True), sg.Radio('My second Radio!', "RADIO1")],
|
||||||
[sg.Multiline(default_text='This is the default Text should you decide not to type anything', size=(35, 3)),
|
[sg.Multiline(default_text='This is the default Text should you decide not to type anything', size=(35, 3)),
|
||||||
sg.Multiline(default_text='A second multi-line', size=(35, 3))],
|
sg.Multiline(default_text='A second multi-line', size=(35, 3))],
|
||||||
[sg.InputCombo(('Combobox 1', 'Combobox 2'), size=(20, 3)),
|
[sg.InputCombo(('Combobox 1', 'Combobox 2'), size=(20, 1)),
|
||||||
sg.Slider(range=(1, 100), orientation='h', size=(34, 20), default_value=85)],
|
sg.Slider(range=(1, 100), orientation='h', size=(34, 20), default_value=85)],
|
||||||
[sg.Listbox(values=('Listbox 1', 'Listbox 2', 'Listbox 3', 'Listbox 4'), size=(30, 3)),
|
[sg.InputOptionMenu(('Menu Option 1', 'Menu Option 2', 'Menu Option 3'))],
|
||||||
|
[sg.Listbox(values=('Listbox 1', 'Listbox 2', 'Listbox 3'), size=(30, 3)),
|
||||||
sg.Slider(range=(1, 100), orientation='v', size=(5, 20), default_value=25),
|
sg.Slider(range=(1, 100), orientation='v', size=(5, 20), default_value=25),
|
||||||
sg.Slider(range=(1, 100), orientation='v', size=(5, 20), default_value=75),
|
sg.Slider(range=(1, 100), orientation='v', size=(5, 20), default_value=75),
|
||||||
sg.Slider(range=(1, 100), orientation='v', size=(5, 20), default_value=10),
|
sg.Slider(range=(1, 100), orientation='v', size=(5, 20), default_value=10),
|
||||||
sg.Spin(values=('Spin Box 1', '2', '3'), initial_value='Spin Box 1')],
|
sg.Column(column1, background_color='#F7F3EC')],
|
||||||
[sg.Text('_' * 80)],
|
[sg.Text('_' * 80)],
|
||||||
|
[sg.Text('Choose A Folder', size=(35, 1))],
|
||||||
[sg.Text('Your Folder', size=(15, 1), auto_size_text=False, justification='right'),
|
[sg.Text('Your Folder', size=(15, 1), auto_size_text=False, justification='right'),
|
||||||
sg.InputText('Default Folder'), sg.FolderBrowse()],
|
sg.InputText('Default Folder'), sg.FolderBrowse()],
|
||||||
[sg.Submit(), sg.Cancel(), sg.SimpleButton('Customized', button_color=('black', '#EDE5B7'))]]
|
[sg.Submit(), sg.Cancel()]
|
||||||
|
]
|
||||||
|
|
||||||
button, values = form.LayoutAndRead(layout)
|
button, values = form.LayoutAndRead(layout)
|
||||||
|
|
||||||
|
@ -137,31 +146,37 @@ def AllWidgetsNoContext():
|
||||||
"""
|
"""
|
||||||
import PySimpleGUI as sg
|
import PySimpleGUI as sg
|
||||||
|
|
||||||
# Green & tan color scheme
|
|
||||||
sg.ChangeLookAndFeel('GreenTan')
|
sg.ChangeLookAndFeel('GreenTan')
|
||||||
|
|
||||||
form = sg.FlexForm('Everything bagel', default_element_size=(40, 1), grab_anywhere=False)
|
form = sg.FlexForm('Everything bagel', default_element_size=(40, 1), grab_anywhere=False)
|
||||||
|
|
||||||
|
column1 = [[sg.Text('Column 1', background_color='#F7F3EC', justification='center', size=(10, 1))],
|
||||||
|
[sg.Spin(values=('Spin Box 1', '2', '3'), initial_value='Spin Box 1')],
|
||||||
|
[sg.Spin(values=('Spin Box 1', '2', '3'), initial_value='Spin Box 2')],
|
||||||
|
[sg.Spin(values=('Spin Box 1', '2', '3'), initial_value='Spin Box 3')]]
|
||||||
|
|
||||||
layout = [
|
layout = [
|
||||||
[sg.Text('All graphic widgets in one form!', size=(30, 1), font=("Helvetica", 25))],
|
[sg.Text('All graphic widgets in one form!', size=(30, 1), font=("Helvetica", 25))],
|
||||||
[sg.Text('Here is some text.... and a place to enter text')],
|
[sg.Text('Here is some text.... and a place to enter text')],
|
||||||
[sg.InputText('This is my text')],
|
[sg.InputText('This is my text')],
|
||||||
[sg.Checkbox('My first checkbox!'), sg.Checkbox('My second checkbox!', default=True)],
|
[sg.Checkbox('Checkbox'), sg.Checkbox('My second checkbox!', default=True)],
|
||||||
[sg.Radio('My first Radio! ', "RADIO1", default=True), sg.Radio('My second Radio!', "RADIO1")],
|
[sg.Radio('My first Radio! ', "RADIO1", default=True), sg.Radio('My second Radio!', "RADIO1")],
|
||||||
[sg.Multiline(default_text='This is the default Text should you decide not to type anything', size=(35, 3)),
|
[sg.Multiline(default_text='This is the default Text should you decide not to type anything', size=(35, 3)),
|
||||||
sg.Multiline(default_text='A second multi-line', size=(35, 3))],
|
sg.Multiline(default_text='A second multi-line', size=(35, 3))],
|
||||||
[sg.InputCombo(('Combobox 1', 'Combobox 2'), size=(20, 3)),
|
[sg.InputCombo(('Combobox 1', 'Combobox 2'), size=(20, 1)),
|
||||||
sg.Slider(range=(1, 100), orientation='h', size=(34, 20), default_value=85)],
|
sg.Slider(range=(1, 100), orientation='h', size=(34, 20), default_value=85)],
|
||||||
[sg.Listbox(values=('Listbox 1', 'Listbox 2', 'Listbox 3'), size=(30, 3)),
|
[sg.InputOptionMenu(('Menu Option 1', 'Menu Option 2', 'Menu Option 3'))],
|
||||||
sg.Slider(range=(1, 100), orientation='v', size=(5, 20), default_value=25),
|
[sg.Listbox(values=('Listbox 1', 'Listbox 2', 'Listbox 3'), size=(30, 3)),
|
||||||
sg.Slider(range=(1, 100), orientation='v', size=(5, 20), default_value=75),
|
sg.Slider(range=(1, 100), orientation='v', size=(5, 20), default_value=25),
|
||||||
sg.Slider(range=(1, 100), orientation='v', size=(5, 20), default_value=10),
|
sg.Slider(range=(1, 100), orientation='v', size=(5, 20), default_value=75),
|
||||||
sg.Spin(values=('Spin Box 1', '2', '3'), initial_value='Spin Box 1')],
|
sg.Slider(range=(1, 100), orientation='v', size=(5, 20), default_value=10),
|
||||||
[sg.Text('_' * 80)],
|
sg.Column(column1, background_color='#F7F3EC')],
|
||||||
[sg.Text('Choose A Folder', size=(35, 1))],
|
[sg.Text('_' * 80)],
|
||||||
[sg.Text('Your Folder', size=(15, 1), auto_size_text=False, justification='right'),
|
[sg.Text('Choose A Folder', size=(35, 1))],
|
||||||
sg.InputText('Default Folder'), sg.FolderBrowse()],
|
[sg.Text('Your Folder', size=(15, 1), auto_size_text=False, justification='right'),
|
||||||
[sg.Submit(), sg.Cancel(), sg.SimpleButton('Customized', button_color=('white', '#7E6C92'))]
|
sg.InputText('Default Folder'), sg.FolderBrowse()],
|
||||||
]
|
[sg.Submit(), sg.Cancel()]
|
||||||
|
]
|
||||||
|
|
||||||
button, values = form.LayoutAndRead(layout)
|
button, values = form.LayoutAndRead(layout)
|
||||||
|
|
||||||
|
@ -175,10 +190,9 @@ def NonBlockingWithUpdates():
|
||||||
|
|
||||||
form = sg.FlexForm('Running Timer')
|
form = sg.FlexForm('Running Timer')
|
||||||
# create a text element that will be updated periodically
|
# create a text element that will be updated periodically
|
||||||
text_element = sg.Text('', size=(10, 2), font=('Helvetica', 20), justification='center')
|
|
||||||
|
|
||||||
form_rows = [[sg.Text('Stopwatch', size=(20,2), justification='center')],
|
form_rows = [[sg.Text('Stopwatch', size=(20,2), justification='center')],
|
||||||
[text_element],
|
[ sg.Text('', size=(10, 2), font=('Helvetica', 20), justification='center', key='output')],
|
||||||
[sg.T(' ' * 5), sg.ReadFormButton('Start/Stop', focus=True), sg.Quit()]]
|
[sg.T(' ' * 5), sg.ReadFormButton('Start/Stop', focus=True), sg.Quit()]]
|
||||||
|
|
||||||
form.LayoutAndRead(form_rows, non_blocking=True)
|
form.LayoutAndRead(form_rows, non_blocking=True)
|
||||||
|
@ -193,7 +207,7 @@ def NonBlockingWithUpdates():
|
||||||
break
|
break
|
||||||
elif button == 'Start/Stop':
|
elif button == 'Start/Stop':
|
||||||
timer_running = not timer_running
|
timer_running = not timer_running
|
||||||
text_element.Update('{:02d}:{:02d}.{:02d}'.format((i // 100) // 60, (i // 100) % 60, i % 100))
|
form.FindElement('output').Update('{:02d}:{:02d}.{:02d}'.format((i // 100) // 60, (i // 100) % 60, i % 100))
|
||||||
|
|
||||||
time.sleep(.01)
|
time.sleep(.01)
|
||||||
# if the loop finished then need to close the form for the user
|
# if the loop finished then need to close the form for the user
|
||||||
|
@ -208,14 +222,13 @@ def NonBlockingWithContext():
|
||||||
import time
|
import time
|
||||||
|
|
||||||
with sg.FlexForm('Running Timer') as form:
|
with sg.FlexForm('Running Timer') as form:
|
||||||
text_element = sg.Text('', size=(10, 2), font=('Helvetica', 20), text_color='red', justification='center')
|
|
||||||
layout = [[sg.Text('Non blocking GUI with updates', justification='center')],
|
layout = [[sg.Text('Non blocking GUI with updates', justification='center')],
|
||||||
[text_element],
|
[sg.Text('', size=(10, 2), font=('Helvetica', 20), text_color='red', justification='center', key='output')],
|
||||||
[sg.T(' ' * 15), sg.Quit()]]
|
[sg.T(' ' * 15), sg.Quit()]]
|
||||||
form.LayoutAndRead(layout, non_blocking=True)
|
form.LayoutAndRead(layout, non_blocking=True)
|
||||||
|
|
||||||
for i in range(1, 500):
|
for i in range(1, 500):
|
||||||
text_element.Update('{:02d}:{:02d}.{:02d}'.format((i // 100) // 60, (i // 100) % 60, i % 100))
|
form.FindElement('output').Update('{:02d}:{:02d}.{:02d}'.format((i // 100) // 60, (i // 100) % 60, i % 100))
|
||||||
button, values = form.ReadNonBlocking()
|
button, values = form.ReadNonBlocking()
|
||||||
if values is None or button == 'Quit': # if user closed the window using X
|
if values is None or button == 'Quit': # if user closed the window using X
|
||||||
break
|
break
|
||||||
|
@ -451,16 +464,7 @@ def MachineLearning():
|
||||||
"""
|
"""
|
||||||
import PySimpleGUI as sg
|
import PySimpleGUI as sg
|
||||||
|
|
||||||
# Green & tan color scheme
|
sg.ChangeLookAndFeel('LightGreen')
|
||||||
sg.SetOptions(background_color='#9FB8AD',
|
|
||||||
text_element_background_color='#9FB8AD',
|
|
||||||
element_background_color='#9FB8AD',
|
|
||||||
input_elements_background_color='#F7F3EC',
|
|
||||||
button_color=('white', '#475841'),
|
|
||||||
border_width=0,
|
|
||||||
slider_border_width=0,
|
|
||||||
progress_meter_border_depth=0,
|
|
||||||
scrollbar_color='#F7F3EC')
|
|
||||||
|
|
||||||
sg.SetOptions(text_justification='right')
|
sg.SetOptions(text_justification='right')
|
||||||
|
|
||||||
|
@ -645,8 +649,6 @@ def InputElementUpdate():
|
||||||
"""
|
"""
|
||||||
import PySimpleGUI as g
|
import PySimpleGUI as g
|
||||||
|
|
||||||
# g.SetOptions(button_color=g.COLOR_SYSTEM_DEFAULT) # because some people like gray buttons
|
|
||||||
|
|
||||||
# Demonstrates a number of PySimpleGUI features including:
|
# Demonstrates a number of PySimpleGUI features including:
|
||||||
# Default element size
|
# Default element size
|
||||||
# auto_size_buttons
|
# auto_size_buttons
|
||||||
|
@ -655,17 +657,13 @@ def InputElementUpdate():
|
||||||
# Update of elements in form (Text, Input)
|
# Update of elements in form (Text, Input)
|
||||||
# do_not_clear of Input elements
|
# do_not_clear of Input elements
|
||||||
|
|
||||||
# create the 2 Elements we want to control outside the form
|
|
||||||
out_elem = g.Text('', size=(15, 1), font=('Helvetica', 18), text_color='red')
|
|
||||||
in_elem = g.Input(size=(10, 1), do_not_clear=True, key='input')
|
|
||||||
|
|
||||||
layout = [[g.Text('Enter Your Passcode')],
|
layout = [[g.Text('Enter Your Passcode')],
|
||||||
[in_elem],
|
[g.Input(size=(10, 1), do_not_clear=True, key='input')],
|
||||||
[g.ReadFormButton('1'), g.ReadFormButton('2'), g.ReadFormButton('3')],
|
[g.ReadFormButton('1'), g.ReadFormButton('2'), g.ReadFormButton('3')],
|
||||||
[g.ReadFormButton('4'), g.ReadFormButton('5'), g.ReadFormButton('6')],
|
[g.ReadFormButton('4'), g.ReadFormButton('5'), g.ReadFormButton('6')],
|
||||||
[g.ReadFormButton('7'), g.ReadFormButton('8'), g.ReadFormButton('9')],
|
[g.ReadFormButton('7'), g.ReadFormButton('8'), g.ReadFormButton('9')],
|
||||||
[g.ReadFormButton('Submit'), g.ReadFormButton('0'), g.ReadFormButton('Clear')],
|
[g.ReadFormButton('Submit'), g.ReadFormButton('0'), g.ReadFormButton('Clear')],
|
||||||
[out_elem],
|
[ g.Text('', size=(15, 1), font=('Helvetica', 18), text_color='red', key='output')],
|
||||||
]
|
]
|
||||||
|
|
||||||
form = g.FlexForm('Keypad', default_element_size=(5, 2), auto_size_buttons=False)
|
form = g.FlexForm('Keypad', default_element_size=(5, 2), auto_size_buttons=False)
|
||||||
|
@ -684,9 +682,9 @@ def InputElementUpdate():
|
||||||
keys_entered += button # add the new digit
|
keys_entered += button # add the new digit
|
||||||
elif button is 'Submit':
|
elif button is 'Submit':
|
||||||
keys_entered = values['input']
|
keys_entered = values['input']
|
||||||
out_elem.Update(keys_entered) # output the final string
|
form.FindElement('outpput').Update(keys_entered) # output the final string
|
||||||
|
|
||||||
in_elem.Update(keys_entered) # change the form to reflect current key string
|
form.FindElement('input').Update(keys_entered) # change the form to reflect current key string
|
||||||
|
|
||||||
|
|
||||||
def TableSimulation():
|
def TableSimulation():
|
||||||
|
@ -751,13 +749,12 @@ fig_dict = {'Simple Data Entry':SimpleDataEntry, 'Simple Entry Return Data as Di
|
||||||
'Table Simulation':TableSimulation, 'Tight Layout':TightLayout}
|
'Table Simulation':TableSimulation, 'Tight Layout':TightLayout}
|
||||||
|
|
||||||
|
|
||||||
# multiline_elem = sg.Multiline(size=(70,35),pad=(5,(3,90)))
|
|
||||||
# define the form layout
|
# define the form layout
|
||||||
listbox_values = [key for key in fig_dict.keys()]
|
listbox_values = [key for key in fig_dict.keys()]
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
# sg.ChangeLookAndFeel('Dark')
|
sg.ChangeLookAndFeel('Dark')
|
||||||
# sg.SetOptions(element_padding=(0,0))
|
sg.SetOptions(element_padding=(0,0))
|
||||||
|
|
||||||
col_listbox = [[sg.Listbox(values=listbox_values, size=(max(len(x) for x in listbox_values),min(len(listbox_values), 20)), change_submits=False, key='func')],
|
col_listbox = [[sg.Listbox(values=listbox_values, size=(max(len(x) for x in listbox_values),min(len(listbox_values), 20)), change_submits=False, key='func')],
|
||||||
[sg.ReadFormButton('Run', pad=(0,0)), sg.ReadFormButton('Show Code', button_color=('white', 'gray25'), pad=(0,0)), sg.Exit(button_color=('white', 'firebrick4'), pad=(0,0))]]
|
[sg.ReadFormButton('Run', pad=(0,0)), sg.ReadFormButton('Show Code', button_color=('white', 'gray25'), pad=(0,0)), sg.Exit(button_color=('white', 'firebrick4'), pad=(0,0))]]
|
||||||
|
|
|
@ -7,63 +7,62 @@ import time
|
||||||
You move it by grabbing anywhere on the window
|
You move it by grabbing anywhere on the window
|
||||||
Good example of how to do a non-blocking, polling program using PySimpleGUI
|
Good example of how to do a non-blocking, polling program using PySimpleGUI
|
||||||
Can be used to poll hardware when running on a Pi
|
Can be used to poll hardware when running on a Pi
|
||||||
|
|
||||||
NOTE - you will get a warning message printed when you exit using exit button.
|
NOTE - you will get a warning message printed when you exit using exit button.
|
||||||
It will look something like:
|
It will look something like:
|
||||||
invalid command name "1616802625480StopMove"
|
invalid command name "1616802625480StopMove"
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def Timer():
|
# ---------------- Create Form ----------------
|
||||||
# ---------------- Create Form ----------------
|
sg.ChangeLookAndFeel('Black')
|
||||||
sg.ChangeLookAndFeel('Black')
|
sg.SetOptions(element_padding=(0, 0))
|
||||||
sg.SetOptions(element_padding=(0,0))
|
# Make a form, but don't use context manager
|
||||||
# Make a form, but don't use context manager
|
# Create the form layout
|
||||||
# Create the form layout
|
form_rows = [[sg.Text('')],
|
||||||
form_rows = [[sg.Text('')],
|
[sg.Text('', size=(8, 2), font=('Helvetica', 20), justification='center', key='text')],
|
||||||
[sg.Text('', size=(8, 2), font=('Helvetica', 20), justification='center', key='text')],
|
[sg.ReadFormButton('Pause', key='button', button_color=('white', '#001480')),
|
||||||
[sg.ReadFormButton('Pause', key='button', button_color=('white', '#001480')), sg.ReadFormButton('Reset', button_color=('white', '#007339')), sg.Exit(button_color=('white','firebrick4'))]]
|
sg.ReadFormButton('Reset', button_color=('white', '#007339')),
|
||||||
# Layout the rows of the form and perform a read. Indicate the form is non-blocking!
|
sg.Exit(button_color=('white', 'firebrick4'))]]
|
||||||
form = sg.FlexForm('Running Timer', no_titlebar=True, auto_size_buttons=False, keep_on_top=True, grab_anywhere=True)
|
# Layout the rows of the form and perform a read. Indicate the form is non-blocking!
|
||||||
form.Layout(form_rows)
|
form = sg.FlexForm('Running Timer', no_titlebar=True, auto_size_buttons=False, keep_on_top=True, grab_anywhere=True)
|
||||||
#
|
form.Layout(form_rows)
|
||||||
# ---------------- main loop ----------------
|
#
|
||||||
current_time = 0
|
# ---------------- main loop ----------------
|
||||||
paused = False
|
current_time = 0
|
||||||
start_time = int(round(time.time()*100))
|
paused = False
|
||||||
while (True):
|
start_time = int(round(time.time() * 100))
|
||||||
# --------- Read and update window --------
|
while (True):
|
||||||
if not paused:
|
# --------- Read and update window --------
|
||||||
button, values = form.ReadNonBlocking()
|
if not paused:
|
||||||
current_time = int(round(time.time()*100)) - start_time
|
button, values = form.ReadNonBlocking()
|
||||||
else:
|
current_time = int(round(time.time() * 100)) - start_time
|
||||||
button, values = form.Read()
|
else:
|
||||||
# --------- Do Button Operations --------
|
button, values = form.Read()
|
||||||
if values is None or button == 'Exit':
|
# --------- Do Button Operations --------
|
||||||
break
|
if values is None or button == 'Exit':
|
||||||
if button is 'Reset':
|
break
|
||||||
start_time = int(round(time.time()*100))
|
if button is 'Reset':
|
||||||
current_time = 0
|
start_time = int(round(time.time() * 100))
|
||||||
paused_time = start_time
|
current_time = 0
|
||||||
elif button == 'Pause':
|
paused_time = start_time
|
||||||
paused = True
|
elif button == 'Pause':
|
||||||
paused_time = int(round(time.time()*100))
|
paused = True
|
||||||
element = form.FindElement('button')
|
paused_time = int(round(time.time() * 100))
|
||||||
element.Update(new_text='Run')
|
element = form.FindElement('button')
|
||||||
elif button == 'Run':
|
element.Update(text='Run')
|
||||||
paused = False
|
elif button == 'Run':
|
||||||
start_time = start_time + int(round(time.time()*100)) - paused_time
|
paused = False
|
||||||
element = form.FindElement('button')
|
start_time = start_time + int(round(time.time() * 100)) - paused_time
|
||||||
element.Update(new_text='Pause')
|
element = form.FindElement('button')
|
||||||
|
element.Update(text='Pause')
|
||||||
|
|
||||||
# --------- Display timer in window --------
|
# --------- Display timer in window --------
|
||||||
form.FindElement('text').Update('{:02d}:{:02d}.{:02d}'.format((current_time // 100) // 60,
|
form.FindElement('text').Update('{:02d}:{:02d}.{:02d}'.format((current_time // 100) // 60,
|
||||||
(current_time // 100) % 60,
|
(current_time // 100) % 60,
|
||||||
current_time % 100))
|
current_time % 100))
|
||||||
time.sleep(.01)
|
time.sleep(.01)
|
||||||
|
|
||||||
# --------- After loop --------
|
# --------- After loop --------
|
||||||
|
|
||||||
# Broke out of main loop. Close the window.
|
# Broke out of main loop. Close the window.
|
||||||
form.CloseNonBlockingForm()
|
form.CloseNonBlockingForm()
|
||||||
|
|
||||||
Timer()
|
|
||||||
|
|
|
@ -18,7 +18,7 @@ with sg.FlexForm("Keyboard Test", return_keyboard_events=True, use_default_focus
|
||||||
print(button, "exiting")
|
print(button, "exiting")
|
||||||
break
|
break
|
||||||
if len(button) == 1:
|
if len(button) == 1:
|
||||||
text_elem.Update(new_value='%s - %s'%(button, ord(button)))
|
text_elem.Update(value='%s - %s' % (button, ord(button)))
|
||||||
if button is not None:
|
if button is not None:
|
||||||
text_elem.Update(button)
|
text_elem.Update(button)
|
||||||
|
|
||||||
|
|
|
@ -1,43 +1,56 @@
|
||||||
import PySimpleGUI as sg
|
from tkinter import *
|
||||||
|
from random import randint
|
||||||
# g.SetOptions(button_color=g.COLOR_SYSTEM_DEFAULT) # because some people like gray buttons
|
import PySimpleGUI as g
|
||||||
|
from matplotlib.backends.backend_tkagg import FigureCanvasTkAgg, FigureCanvasAgg
|
||||||
# Demonstrates a number of PySimpleGUI features including:
|
from matplotlib.figure import Figure
|
||||||
# Default element size
|
import matplotlib.backends.tkagg as tkagg
|
||||||
# auto_size_buttons
|
import tkinter as Tk
|
||||||
# ReadFormButton
|
|
||||||
# Dictionary return values
|
|
||||||
# Update of elements in form (Text, Input)
|
|
||||||
# do_not_clear of Input elements
|
|
||||||
|
|
||||||
|
|
||||||
# create the 2 Elements we want to control outside the form
|
def main():
|
||||||
|
fig = Figure()
|
||||||
|
|
||||||
layout = [[sg.Text('Enter Your Passcode')],
|
ax = fig.add_subplot(111)
|
||||||
[sg.Input(size=(10, 1), do_not_clear=True, key='input')],
|
ax.set_xlabel("X axis")
|
||||||
[sg.ReadFormButton('1'), sg.ReadFormButton('2'), sg.ReadFormButton('3')],
|
ax.set_ylabel("Y axis")
|
||||||
[sg.ReadFormButton('4'), sg.ReadFormButton('5'), sg.ReadFormButton('6')],
|
ax.grid()
|
||||||
[sg.ReadFormButton('7'), sg.ReadFormButton('8'), sg.ReadFormButton('9')],
|
|
||||||
[sg.ReadFormButton('Submit'), sg.ReadFormButton('0'), sg.ReadFormButton('Clear')],
|
|
||||||
[sg.Text('', size=(15, 1), font=('Helvetica', 18), text_color='red', key='out')],
|
|
||||||
]
|
|
||||||
|
|
||||||
form = sg.FlexForm('Keypad', default_button_element_size=(5, 2), auto_size_buttons=False)
|
layout = [[g.Text('Animated Matplotlib', size=(40, 1), justification='center', font='Helvetica 20')],
|
||||||
form.Layout(layout)
|
[g.Canvas(size=(640, 480), key='canvas')],
|
||||||
|
[g.ReadFormButton('Exit', size=(10, 2), pad=((280, 0), 3), font='Helvetica 14')]]
|
||||||
|
|
||||||
# Loop forever reading the form's values, updating the Input field
|
# create the form and show it without the plot
|
||||||
keys_entered = ''
|
form = g.FlexForm('Demo Application - Embedding Matplotlib In PySimpleGUI')
|
||||||
while True:
|
form.Layout(layout)
|
||||||
button, values = form.Read() # read the form
|
form.ReadNonBlocking()
|
||||||
if button is None: # if the X button clicked, just exit
|
|
||||||
break
|
|
||||||
if button is 'Clear': # clear keys if clear button
|
|
||||||
keys_entered = ''
|
|
||||||
elif button in '1234567890':
|
|
||||||
keys_entered = values['input'] # get what's been entered so far
|
|
||||||
keys_entered += button # add the new digit
|
|
||||||
elif button is 'Submit':
|
|
||||||
keys_entered = values['input']
|
|
||||||
form.FindElement('out').Update(keys_entered) # output the final string
|
|
||||||
|
|
||||||
form.FindElement('input').Update(keys_entered) # change the form to reflect current key string
|
canvas_elem = form.FindElement('canvas')
|
||||||
|
|
||||||
|
graph = FigureCanvasTkAgg(fig, master=canvas_elem.TKCanvas)
|
||||||
|
canvas = canvas_elem.TKCanvas
|
||||||
|
|
||||||
|
dpts = [randint(0, 10) for x in range(10000)]
|
||||||
|
for i in range(len(dpts)):
|
||||||
|
button, values = form.ReadNonBlocking()
|
||||||
|
if button is 'Exit' or values is None:
|
||||||
|
exit(69)
|
||||||
|
|
||||||
|
ax.cla()
|
||||||
|
ax.grid()
|
||||||
|
|
||||||
|
ax.plot(range(20), dpts[i:i + 20], color='purple')
|
||||||
|
graph.draw()
|
||||||
|
figure_x, figure_y, figure_w, figure_h = fig.bbox.bounds
|
||||||
|
figure_w, figure_h = int(figure_w), int(figure_h)
|
||||||
|
photo = Tk.PhotoImage(master=canvas, width=figure_w, height=figure_h)
|
||||||
|
|
||||||
|
canvas.create_image(640 / 2, 480 / 2, image=photo)
|
||||||
|
|
||||||
|
figure_canvas_agg = FigureCanvasAgg(fig)
|
||||||
|
figure_canvas_agg.draw()
|
||||||
|
|
||||||
|
tkagg.blit(photo, figure_canvas_agg.get_renderer()._renderer, colormode=2)
|
||||||
|
# time.sleep(.1)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
|
@ -645,7 +645,7 @@ def main():
|
||||||
layout = [[canvas_elem, sg.ReadFormButton('Exit', pad=(0, (210, 0)))]]
|
layout = [[canvas_elem, sg.ReadFormButton('Exit', pad=(0, (210, 0)))]]
|
||||||
|
|
||||||
# create the form and show it without the plot
|
# create the form and show it without the plot
|
||||||
form = sg.FlexForm('Ping Graph', background_color='white')
|
form = sg.FlexForm('Ping Graph', background_color='white', grab_anywhere=True)
|
||||||
form.Layout(layout)
|
form.Layout(layout)
|
||||||
form.ReadNonBlocking()
|
form.ReadNonBlocking()
|
||||||
|
|
||||||
|
|
|
@ -4,9 +4,9 @@ import os
|
||||||
# Simple Image Browser based on PySimpleGUI
|
# Simple Image Browser based on PySimpleGUI
|
||||||
|
|
||||||
# Get the folder containing the images from the user
|
# Get the folder containing the images from the user
|
||||||
rc, folder = sg.GetPathBox('Image Browser', 'Image folder to open', default_path='')
|
folder = sg.PopupGetFolder('Image folder to open')
|
||||||
if rc is False or folder is '':
|
if folder is None:
|
||||||
sg.MsgBoxCancel('Cancelling')
|
sg.PopupCancel('Cancelling')
|
||||||
exit(0)
|
exit(0)
|
||||||
|
|
||||||
# get list of PNG files in folder
|
# get list of PNG files in folder
|
||||||
|
@ -14,52 +14,62 @@ png_files = [folder + '\\' + f for f in os.listdir(folder) if '.png' in f]
|
||||||
filenames_only = [f for f in os.listdir(folder) if '.png' in f]
|
filenames_only = [f for f in os.listdir(folder) if '.png' in f]
|
||||||
|
|
||||||
if len(png_files) == 0:
|
if len(png_files) == 0:
|
||||||
sg.MsgBox('No PNG images in folder')
|
sg.Popup('No PNG images in folder')
|
||||||
exit(0)
|
exit(0)
|
||||||
|
|
||||||
|
|
||||||
|
# define menu layout
|
||||||
|
menu = [['File', ['Open Folder', 'Exit']], ['Help', ['About',]]]
|
||||||
# create the form that also returns keyboard events
|
# create the form that also returns keyboard events
|
||||||
form = sg.FlexForm('Image Browser', return_keyboard_events=True, location=(0,0), use_default_focus=False )
|
form = sg.FlexForm('Image Browser', return_keyboard_events=True, location=(0,0), use_default_focus=False )
|
||||||
|
|
||||||
# make these 2 elements outside the layout because want to "update" them later
|
|
||||||
# initialize to the first PNG file in the list
|
|
||||||
image_elem = sg.Image(filename=png_files[0])
|
|
||||||
filename_display_elem = sg.Text(png_files[0], size=(80, 3))
|
|
||||||
file_num_display_elem = sg.Text('File 1 of {}'.format(len(png_files)), size=(15,1))
|
|
||||||
|
|
||||||
# define layout, show and read the form
|
# define layout, show and read the form
|
||||||
col = [[filename_display_elem],
|
col = [[sg.Text(png_files[0], size=(80, 3), key='filename')],
|
||||||
[image_elem],
|
[sg.Image(filename=png_files[0], key='image')],
|
||||||
[sg.ReadFormButton('Next', size=(8,2)), sg.ReadFormButton('Prev', size=(8,2)), file_num_display_elem]]
|
[sg.ReadFormButton('Next', size=(8,2)), sg.ReadFormButton('Prev', size=(8,2)),
|
||||||
|
sg.Text('File 1 of {}'.format(len(png_files)), size=(15,1), key='filenum')]]
|
||||||
|
|
||||||
col_files = [[sg.Listbox(values=filenames_only, size=(60,30), key='listbox')],
|
col_files = [[sg.Listbox(values=filenames_only, size=(60,30), key='listbox')],
|
||||||
[sg.ReadFormButton('Read')]]
|
[sg.ReadFormButton('Read')]]
|
||||||
layout = [[sg.Column(col_files), sg.Column(col)]]
|
layout = [[sg.Menu(menu)], [sg.Column(col_files), sg.Column(col)]]
|
||||||
button, values = form.LayoutAndRead(layout) # Shows form on screen
|
button, values = form.LayoutAndRead(layout) # Shows form on screen
|
||||||
|
|
||||||
# loop reading the user input and displaying image, filename
|
# loop reading the user input and displaying image, filename
|
||||||
i=0
|
i=0
|
||||||
while True:
|
while True:
|
||||||
|
|
||||||
# perform button and keyboard operations
|
# --------------------- Button & Keyboard ---------------------
|
||||||
if button is None:
|
if button is None:
|
||||||
break
|
break
|
||||||
elif button in ('Next', 'MouseWheel:Down', 'Down:40', 'Next:34') and i < len(png_files)-1:
|
elif button in ('Next', 'MouseWheel:Down', 'Down:40', 'Next:34') and i < len(png_files)-1:
|
||||||
i += 1
|
i += 1
|
||||||
elif button in ('Prev', 'MouseWheel:Up', 'Up:38', 'Prior:33') and i > 0:
|
elif button in ('Prev', 'MouseWheel:Up', 'Up:38', 'Prior:33') and i > 0:
|
||||||
i -= 1
|
i -= 1
|
||||||
|
elif button == 'Exit':
|
||||||
|
exit(69)
|
||||||
|
|
||||||
if button == 'Read':
|
filename = folder + '/' + values['listbox'][0] if button == 'Read' else png_files[i]
|
||||||
filename = folder + '\\' + values['listbox'][0]
|
|
||||||
# print(filename)
|
# ----------------- Menu choices -----------------
|
||||||
else:
|
if button == 'Open Folder':
|
||||||
filename = png_files[i]
|
newfolder = sg.PopupGetFolder('New folder', no_window=True)
|
||||||
|
if newfolder is None:
|
||||||
|
continue
|
||||||
|
folder = newfolder
|
||||||
|
png_files = [folder + '/' + f for f in os.listdir(folder) if '.png' in f]
|
||||||
|
filenames_only = [f for f in os.listdir(folder) if '.png' in f]
|
||||||
|
form.FindElement('listbox').Update(values=filenames_only)
|
||||||
|
form.Refresh()
|
||||||
|
i = 0
|
||||||
|
elif button == 'About':
|
||||||
|
sg.Popup('Demo PNG Viewer Program', 'Please give PySimpleGUI a try!')
|
||||||
|
|
||||||
# update window with new image
|
# update window with new image
|
||||||
image_elem.Update(filename=filename)
|
form.FindElement('image').Update(filename=filename)
|
||||||
# update window with filename
|
# update window with filename
|
||||||
filename_display_elem.Update(filename)
|
form.FindElement('filename').Update(filename)
|
||||||
# update page display
|
# update page display
|
||||||
file_num_display_elem.Update('File {} of {}'.format(i+1, len(png_files)))
|
form.FindElement('filenum').Update('File {} of {}'.format(i+1, len(png_files)))
|
||||||
|
|
||||||
# read the form
|
# read the form
|
||||||
button, values = form.Read()
|
button, values = form.Read()
|
||||||
|
|
|
@ -9,6 +9,8 @@ import hashlib
|
||||||
3. Type password into the GUI
|
3. Type password into the GUI
|
||||||
4. Copy and paste hash code form GUI into variable named login_password_hash
|
4. Copy and paste hash code form GUI into variable named login_password_hash
|
||||||
5. Run program again and test your login!
|
5. Run program again and test your login!
|
||||||
|
6. Are you paying attention? The first person that can post an issue on GitHub with the
|
||||||
|
matching password to the hash code in this example gets a $5 PayPal payment
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Use this GUI to get your password's hash code
|
# Use this GUI to get your password's hash code
|
||||||
|
@ -47,7 +49,7 @@ def PasswordMatches(password, hash):
|
||||||
return password_hash == hash
|
return password_hash == hash
|
||||||
|
|
||||||
|
|
||||||
login_password_hash = '5baa61e4c9b93f3f0682250b6cf8331b7ee68fd8'
|
login_password_hash = 'e5e9fa1ba31ecd1ae84f75caaa474f3a663f05f4'
|
||||||
password = sg.PopupGetText('Password', password_char='*')
|
password = sg.PopupGetText('Password', password_char='*')
|
||||||
if password == 'gui': # Remove when pasting into your program
|
if password == 'gui': # Remove when pasting into your program
|
||||||
HashGeneratorGUI() # Remove when pasting into your program
|
HashGeneratorGUI() # Remove when pasting into your program
|
||||||
|
|
|
@ -13,19 +13,4 @@ layout = [
|
||||||
|
|
||||||
button, values = form.LayoutAndRead(layout)
|
button, values = form.LayoutAndRead(layout)
|
||||||
|
|
||||||
print(button, values, values['name'], values['address'], values['phone'])
|
sg.Popup(button, values, values['name'], values['address'], values['phone'])
|
||||||
|
|
||||||
form = sg.FlexForm('Simple data entry form') # begin with a blank form
|
|
||||||
|
|
||||||
layout = [
|
|
||||||
[sg.Text('Please enter your Name, Address, Phone')],
|
|
||||||
[sg.Text('Name', size=(15, 1)), sg.InputText('name')],
|
|
||||||
[sg.Text('Address', size=(15, 1)), sg.InputText('address')],
|
|
||||||
[sg.Text('Phone', size=(15, 1)), sg.InputText('phone')],
|
|
||||||
[sg.Ok(), sg.Cancel()]
|
|
||||||
]
|
|
||||||
|
|
||||||
button, values = form.LayoutAndRead(layout)
|
|
||||||
print(values)
|
|
||||||
name, address, phone = values
|
|
||||||
sg.MsgBox(button, values[0], values[1], values[2])
|
|
||||||
|
|
|
@ -1,30 +1,68 @@
|
||||||
|
import csv
|
||||||
import PySimpleGUI as sg
|
import PySimpleGUI as sg
|
||||||
|
|
||||||
def TableSimulation():
|
def TableSimulation():
|
||||||
"""
|
"""
|
||||||
Display data in a table format
|
Display data in a table format
|
||||||
"""
|
"""
|
||||||
# sg.ChangeLookAndFeel('Dark')
|
|
||||||
sg.SetOptions(element_padding=(0,0))
|
sg.SetOptions(element_padding=(0,0))
|
||||||
layout = [[sg.T('Table Using Combos and Input Elements', font='Any 18')],
|
|
||||||
[sg.T('Row, Cal to change'),
|
menu_def = [['File', ['Open', 'Save', 'Exit']],
|
||||||
|
['Edit', ['Paste', ['Special', 'Normal',], 'Undo'],],
|
||||||
|
['Help', 'About...'],]
|
||||||
|
|
||||||
|
columm_layout = [[]]
|
||||||
|
|
||||||
|
MAX_ROWS = 60
|
||||||
|
MAX_COL = 10
|
||||||
|
for i in range(MAX_ROWS):
|
||||||
|
inputs = [sg.T('{}'.format(i), size=(4,1), justification='right')] + [sg.In(size=(10, 1), pad=(1, 1), justification='right', key=(i,j), do_not_clear=True) for j in range(MAX_COL)]
|
||||||
|
columm_layout.append(inputs)
|
||||||
|
|
||||||
|
layout = [ [sg.Menu(menu_def)],
|
||||||
|
[sg.T('Table Using Combos and Input Elements', font='Any 18')],
|
||||||
|
[sg.T('Type in a row, column and value. The form will update the values in realtime as you type'),
|
||||||
sg.In(key='inputrow', justification='right', size=(8,1), pad=(1,1), do_not_clear=True),
|
sg.In(key='inputrow', justification='right', size=(8,1), pad=(1,1), do_not_clear=True),
|
||||||
sg.In(key='inputcol', size=(8,1), pad=(1,1), justification='right', do_not_clear=True),
|
sg.In(key='inputcol', size=(8,1), pad=(1,1), justification='right', do_not_clear=True),
|
||||||
sg.In(key='value', size=(8,1), pad=(1,1), justification='right', do_not_clear=True)]]
|
sg.In(key='value', size=(8,1), pad=(1,1), justification='right', do_not_clear=True)],
|
||||||
|
[sg.Column(columm_layout, size=(815,600), scrollable=True)]]
|
||||||
for i in range(20):
|
|
||||||
inputs = [sg.In('{}{}'.format(i,j), size=(8, 1), pad=(1, 1), justification='right', key=(i,j), do_not_clear=True) for j in range(10)]
|
|
||||||
line = [sg.Combo(('Customer ID', 'Customer Name', 'Customer Info'))]
|
|
||||||
line.append(inputs)
|
|
||||||
layout.append(inputs)
|
|
||||||
|
|
||||||
form = sg.FlexForm('Table', return_keyboard_events=True, grab_anywhere=False)
|
form = sg.FlexForm('Table', return_keyboard_events=True, grab_anywhere=False)
|
||||||
form.Layout(layout)
|
form.Layout(layout)
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
button, values = form.Read()
|
button, values = form.Read()
|
||||||
if button is None:
|
# --- Process buttons --- #
|
||||||
|
if button is None or button == 'Exit':
|
||||||
break
|
break
|
||||||
|
elif button == 'About...':
|
||||||
|
sg.Popup('Demo of table capabilities')
|
||||||
|
elif button == 'Open':
|
||||||
|
filename = sg.PopupGetFile('filename to open', no_window=True, file_types=(("CSV Files","*.csv"),))
|
||||||
|
# --- populate table with file contents --- #
|
||||||
|
if filename is not None:
|
||||||
|
with open(filename, "r") as infile:
|
||||||
|
reader = csv.reader(infile)
|
||||||
|
try:
|
||||||
|
data = list(reader) # read everything else into a list of rows
|
||||||
|
except:
|
||||||
|
sg.PopupError('Error reading file')
|
||||||
|
continue
|
||||||
|
# clear the table
|
||||||
|
[form.FindElement((i,j)).Update('') for j in range(MAX_COL) for i in range(MAX_ROWS)]
|
||||||
|
|
||||||
|
for i, row in enumerate(data):
|
||||||
|
for j, item in enumerate(row):
|
||||||
|
location = (i,j)
|
||||||
|
try: # try the best we can at reading and filling the table
|
||||||
|
target_element = form.FindElement(location)
|
||||||
|
new_value = item
|
||||||
|
if target_element is not None and new_value != '':
|
||||||
|
target_element.Update(new_value)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
# if a valid table location entered, change that location's value
|
||||||
try:
|
try:
|
||||||
location = (int(values['inputrow']), int(values['inputcol']))
|
location = (int(values['inputrow']), int(values['inputcol']))
|
||||||
target_element = form.FindElement(location)
|
target_element = form.FindElement(location)
|
||||||
|
|
211
PySimpleGUI.py
211
PySimpleGUI.py
|
@ -282,11 +282,16 @@ class InputText(Element):
|
||||||
super().__init__(ELEM_TYPE_INPUT_TEXT, scale=scale, size=size, auto_size_text=auto_size_text, background_color=bg, text_color=fg, key=key, pad=pad)
|
super().__init__(ELEM_TYPE_INPUT_TEXT, scale=scale, size=size, auto_size_text=auto_size_text, background_color=bg, text_color=fg, key=key, pad=pad)
|
||||||
|
|
||||||
|
|
||||||
def Update(self, new_value):
|
def Update(self, value=None, disable=None):
|
||||||
try:
|
if disable is True:
|
||||||
self.TKStringVar.set(new_value)
|
self.TKEntry['state'] = 'disabled'
|
||||||
except: pass
|
elif disable is False:
|
||||||
self.DefaultText = new_value
|
self.TKEntry['state'] = 'normal'
|
||||||
|
if value is not None:
|
||||||
|
try:
|
||||||
|
self.TKStringVar.set(value)
|
||||||
|
except: pass
|
||||||
|
self.DefaultText = value
|
||||||
|
|
||||||
def Get(self):
|
def Get(self):
|
||||||
return self.TKStringVar.get()
|
return self.TKStringVar.get()
|
||||||
|
@ -317,23 +322,24 @@ class InputCombo(Element):
|
||||||
|
|
||||||
super().__init__(ELEM_TYPE_INPUT_COMBO, scale=scale, size=size, auto_size_text=auto_size_text, background_color=bg, text_color=fg, key=key, pad=pad)
|
super().__init__(ELEM_TYPE_INPUT_COMBO, scale=scale, size=size, auto_size_text=auto_size_text, background_color=bg, text_color=fg, key=key, pad=pad)
|
||||||
|
|
||||||
def Update(self, value=None, values=None, disabled=False):
|
def Update(self, value=None, values=None, disable=None):
|
||||||
if values is not None:
|
if values is not None:
|
||||||
try:
|
try:
|
||||||
self.TKCombo['values'] = values
|
self.TKCombo['values'] = values
|
||||||
self.TKCombo.current(0)
|
self.TKCombo.current(0)
|
||||||
except: pass
|
except: pass
|
||||||
self.Values = values
|
self.Values = values
|
||||||
|
for index, v in enumerate(self.Values):
|
||||||
self.TKCombo['state'] = 'disable' if disabled else 'enable'
|
if v == value:
|
||||||
|
try:
|
||||||
for index, v in enumerate(self.Values):
|
self.TKCombo.current(index)
|
||||||
if v == value:
|
except: pass
|
||||||
try:
|
self.DefaultValue = value
|
||||||
self.TKCombo.current(index)
|
break
|
||||||
except: pass
|
if disable == True:
|
||||||
self.DefaultValue = value
|
self.TKCombo['state'] = 'disable'
|
||||||
break
|
elif disable == False:
|
||||||
|
self.TKCombo['state'] = 'enable'
|
||||||
|
|
||||||
|
|
||||||
def __del__(self):
|
def __del__(self):
|
||||||
|
@ -365,14 +371,21 @@ class InputOptionMenu(Element):
|
||||||
|
|
||||||
super().__init__(ELEM_TYPE_INPUT_OPTION_MENU, scale=scale, size=size, auto_size_text=auto_size_text, background_color=bg, text_color=fg, key=key, pad=pad)
|
super().__init__(ELEM_TYPE_INPUT_OPTION_MENU, scale=scale, size=size, auto_size_text=auto_size_text, background_color=bg, text_color=fg, key=key, pad=pad)
|
||||||
|
|
||||||
def Update(self, value):
|
def Update(self, value=None, values=None, disable=None):
|
||||||
for index, v in enumerate(self.Values):
|
if values is not None:
|
||||||
if v == value:
|
self.Values = values
|
||||||
try:
|
if self.Values is not None:
|
||||||
self.TKStringVar.set(value)
|
for index, v in enumerate(self.Values):
|
||||||
except: pass
|
if v == value:
|
||||||
self.DefaultValue = value
|
try:
|
||||||
break
|
self.TKStringVar.set(value)
|
||||||
|
except: pass
|
||||||
|
self.DefaultValue = value
|
||||||
|
break
|
||||||
|
if disable == True:
|
||||||
|
self.TKOptionMenu['state'] = 'disabled'
|
||||||
|
elif disable == False:
|
||||||
|
self.TKOptionMenu['state'] = 'normal'
|
||||||
|
|
||||||
|
|
||||||
def __del__(self):
|
def __del__(self):
|
||||||
|
@ -415,12 +428,19 @@ class Listbox(Element):
|
||||||
|
|
||||||
super().__init__(ELEM_TYPE_INPUT_LISTBOX, scale=scale, size=size, auto_size_text=auto_size_text, font=font, background_color=bg, text_color=fg, key=key, pad=pad)
|
super().__init__(ELEM_TYPE_INPUT_LISTBOX, scale=scale, size=size, auto_size_text=auto_size_text, font=font, background_color=bg, text_color=fg, key=key, pad=pad)
|
||||||
|
|
||||||
def Update(self, values):
|
def Update(self, values=None, disable=None):
|
||||||
self.TKListbox.delete(0, 'end')
|
if disable == True:
|
||||||
for item in values:
|
self.TKListbox.configure(state='disabled')
|
||||||
self.TKListbox.insert(tk.END, item)
|
elif disable == False:
|
||||||
self.TKListbox.selection_set(0, 0)
|
self.TKListbox.configure(state='normal')
|
||||||
self.Values = values
|
if values is not None:
|
||||||
|
self.TKListbox.delete(0, 'end')
|
||||||
|
for item in values:
|
||||||
|
self.TKListbox.insert(tk.END, item)
|
||||||
|
self.TKListbox.selection_set(0, 0)
|
||||||
|
self.Values = values
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def SetValue(self, values):
|
def SetValue(self, values):
|
||||||
for index, item in enumerate(self.Values):
|
for index, item in enumerate(self.Values):
|
||||||
|
@ -466,14 +486,17 @@ class Radio(Element):
|
||||||
|
|
||||||
super().__init__(ELEM_TYPE_INPUT_RADIO, scale=scale , size=size, auto_size_text=auto_size_text, font=font, background_color=background_color, text_color=self.TextColor, key=key, pad=pad)
|
super().__init__(ELEM_TYPE_INPUT_RADIO, scale=scale , size=size, auto_size_text=auto_size_text, font=font, background_color=background_color, text_color=self.TextColor, key=key, pad=pad)
|
||||||
|
|
||||||
def Update(self, value):
|
def Update(self, value=None, disable=None):
|
||||||
if not value:
|
|
||||||
return
|
|
||||||
location = EncodeRadioRowCol(self.Position[0], self.Position[1])
|
location = EncodeRadioRowCol(self.Position[0], self.Position[1])
|
||||||
try:
|
if value is not None:
|
||||||
self.TKIntVar.set(location)
|
try:
|
||||||
except: pass
|
self.TKIntVar.set(location)
|
||||||
self.InitialState = value
|
except: pass
|
||||||
|
self.InitialState = value
|
||||||
|
if disable == True:
|
||||||
|
self.TKRadio['state'] = 'disabled'
|
||||||
|
elif disable == False:
|
||||||
|
self.TKRadio['state'] = 'normal'
|
||||||
|
|
||||||
def __del__(self):
|
def __del__(self):
|
||||||
try:
|
try:
|
||||||
|
@ -508,15 +531,16 @@ class Checkbox(Element):
|
||||||
def Get(self):
|
def Get(self):
|
||||||
return self.TKIntVar.get()
|
return self.TKIntVar.get()
|
||||||
|
|
||||||
def Update(self, value):
|
def Update(self, value=None, disable=None):
|
||||||
try:
|
if value is not None:
|
||||||
if value is None:
|
try:
|
||||||
self.TKCheckbutton.configure(state='disabled')
|
|
||||||
else:
|
|
||||||
self.TKCheckbutton.configure(state='normal')
|
|
||||||
self.TKIntVar.set(value)
|
self.TKIntVar.set(value)
|
||||||
except: pass
|
self.InitialState = value
|
||||||
self.InitialState = value
|
except: pass
|
||||||
|
if disable == True:
|
||||||
|
self.TKCheckbutton.configure(state='disabled')
|
||||||
|
elif disable == False:
|
||||||
|
self.TKCheckbutton.configure(state='normal')
|
||||||
|
|
||||||
|
|
||||||
def __del__(self):
|
def __del__(self):
|
||||||
|
@ -550,17 +574,21 @@ class Spin(Element):
|
||||||
super().__init__(ELEM_TYPE_INPUT_SPIN, scale, size, auto_size_text, font=font,background_color=bg, text_color=fg, key=key, pad=pad)
|
super().__init__(ELEM_TYPE_INPUT_SPIN, scale, size, auto_size_text, font=font,background_color=bg, text_color=fg, key=key, pad=pad)
|
||||||
return
|
return
|
||||||
|
|
||||||
def Update(self, new_value=None, new_values=None ):
|
def Update(self, value=None, values=None, disable=None):
|
||||||
if new_values != None:
|
if values != None:
|
||||||
old_value = self.TKStringVar.get()
|
old_value = self.TKStringVar.get()
|
||||||
self.Values = new_values
|
self.Values = values
|
||||||
self.TKSpinBox.configure(values=new_values)
|
self.TKSpinBox.configure(values=values)
|
||||||
self.TKStringVar.set(old_value)
|
self.TKStringVar.set(old_value)
|
||||||
if new_value is not None:
|
if value is not None:
|
||||||
try:
|
try:
|
||||||
self.TKStringVar.set(new_value)
|
self.TKStringVar.set(value)
|
||||||
except: pass
|
except: pass
|
||||||
self.DefaultValue = new_value
|
self.DefaultValue = value
|
||||||
|
if disable == True:
|
||||||
|
self.TKSpinBox.configure(state='disabled')
|
||||||
|
elif disable == False:
|
||||||
|
self.TKSpinBox.configure(state='normal')
|
||||||
|
|
||||||
|
|
||||||
def SpinChangedHandler(self, event):
|
def SpinChangedHandler(self, event):
|
||||||
|
@ -601,12 +629,17 @@ class Multiline(Element):
|
||||||
super().__init__(ELEM_TYPE_INPUT_MULTILINE, scale=scale, size=size, auto_size_text=auto_size_text, background_color=bg, text_color=fg, key=key, pad=pad)
|
super().__init__(ELEM_TYPE_INPUT_MULTILINE, scale=scale, size=size, auto_size_text=auto_size_text, background_color=bg, text_color=fg, key=key, pad=pad)
|
||||||
return
|
return
|
||||||
|
|
||||||
def Update(self, new_value):
|
def Update(self, value=None, disable=None):
|
||||||
try:
|
if value is not None:
|
||||||
self.TKText.delete('1.0', tk.END)
|
try:
|
||||||
self.TKText.insert(1.0, new_value)
|
self.TKText.delete('1.0', tk.END)
|
||||||
except: pass
|
self.TKText.insert(1.0, value)
|
||||||
self.DefaultText = new_value
|
except: pass
|
||||||
|
self.DefaultText = value
|
||||||
|
if disable == True:
|
||||||
|
self.TKText.configure(state='disabled')
|
||||||
|
elif disable == False:
|
||||||
|
self.TKText.configure(state='normal')
|
||||||
|
|
||||||
def Get(self):
|
def Get(self):
|
||||||
return self.TKText.get(1.0, tk.END)
|
return self.TKText.get(1.0, tk.END)
|
||||||
|
@ -641,11 +674,11 @@ class Text(Element):
|
||||||
super().__init__(ELEM_TYPE_TEXT, scale, size, auto_size_text, background_color=bg, font=font if font else DEFAULT_FONT, text_color=self.TextColor, pad=pad, key=key)
|
super().__init__(ELEM_TYPE_TEXT, scale, size, auto_size_text, background_color=bg, font=font if font else DEFAULT_FONT, text_color=self.TextColor, pad=pad, key=key)
|
||||||
return
|
return
|
||||||
|
|
||||||
def Update(self, new_value = None, background_color=None, text_color=None, font=None):
|
def Update(self, value = None, background_color=None, text_color=None, font=None):
|
||||||
if new_value is not None:
|
if value is not None:
|
||||||
self.DisplayText=new_value
|
self.DisplayText=value
|
||||||
stringvar = self.TKStringVar
|
stringvar = self.TKStringVar
|
||||||
stringvar.set(new_value)
|
stringvar.set(value)
|
||||||
if background_color is not None:
|
if background_color is not None:
|
||||||
self.TKText.configure(background=background_color)
|
self.TKText.configure(background=background_color)
|
||||||
if text_color is not None:
|
if text_color is not None:
|
||||||
|
@ -690,17 +723,18 @@ class TKProgressBar():
|
||||||
s.configure(str(length)+str(width)+"my.Vertical.TProgressbar", troughrelief=relief, borderwidth=border_width, thickness=width)
|
s.configure(str(length)+str(width)+"my.Vertical.TProgressbar", troughrelief=relief, borderwidth=border_width, thickness=width)
|
||||||
self.TKProgressBarForReal = ttk.Progressbar(root, maximum=self.Max, style=str(length)+str(width)+'my.Vertical.TProgressbar', length=length, orient=tk.VERTICAL, mode='determinate')
|
self.TKProgressBarForReal = ttk.Progressbar(root, maximum=self.Max, style=str(length)+str(width)+'my.Vertical.TProgressbar', length=length, orient=tk.VERTICAL, mode='determinate')
|
||||||
|
|
||||||
def Update(self, count, max=None):
|
def Update(self, count=None, max=None):
|
||||||
if max is not None:
|
if max is not None:
|
||||||
self.Max = max
|
self.Max = max
|
||||||
try:
|
try:
|
||||||
self.TKProgressBarForReal.config(maximum=max)
|
self.TKProgressBarForReal.config(maximum=max)
|
||||||
except:
|
except:
|
||||||
return False
|
return False
|
||||||
if count > self.Max: return False
|
if count is not None and count > self.Max: return False
|
||||||
try:
|
if count is not None:
|
||||||
self.TKProgressBarForReal['value'] = count
|
try:
|
||||||
except: return False
|
self.TKProgressBarForReal['value'] = count
|
||||||
|
except: return False
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def __del__(self):
|
def __del__(self):
|
||||||
|
@ -922,16 +956,23 @@ class Button(Element):
|
||||||
|
|
||||||
return
|
return
|
||||||
|
|
||||||
def Update(self, value=None, new_text=None, button_color=(None, None)):
|
def Update(self, value=None, text=None, button_color=(None, None), disable=None):
|
||||||
try:
|
try:
|
||||||
if new_text is not None:
|
if text is not None:
|
||||||
self.TKButton.configure(text=new_text)
|
self.TKButton.configure(text=text)
|
||||||
self.ButtonText = new_text
|
self.ButtonText = text
|
||||||
if button_color != (None, None):
|
if button_color != (None, None):
|
||||||
self.TKButton.config(foreground=button_color[0], background=button_color[1])
|
self.TKButton.config(foreground=button_color[0], background=button_color[1])
|
||||||
except:
|
except:
|
||||||
return
|
return
|
||||||
self.DefaultValue = value
|
if value is not None:
|
||||||
|
self.DefaultValue = value
|
||||||
|
if disable == True:
|
||||||
|
self.TKButton['state'] = 'disabled'
|
||||||
|
elif disable == False:
|
||||||
|
self.TKButton['state'] = 'normal'
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def __del__(self):
|
def __del__(self):
|
||||||
try:
|
try:
|
||||||
|
@ -1089,13 +1130,18 @@ class Slider(Element):
|
||||||
super().__init__(ELEM_TYPE_INPUT_SLIDER, scale=scale, size=size, font=font, background_color=background_color, text_color=text_color, key=key, pad=pad)
|
super().__init__(ELEM_TYPE_INPUT_SLIDER, scale=scale, size=size, font=font, background_color=background_color, text_color=text_color, key=key, pad=pad)
|
||||||
return
|
return
|
||||||
|
|
||||||
def Update(self, value, range=(None, None)):
|
def Update(self, value=None, range=(None, None), disable=None):
|
||||||
try:
|
if value is not None:
|
||||||
self.TKIntVar.set(value)
|
try:
|
||||||
if range != (None, None):
|
self.TKIntVar.set(value)
|
||||||
self.TKScale.config(from_ = range[0], to_ = range[1])
|
if range != (None, None):
|
||||||
except: pass
|
self.TKScale.config(from_ = range[0], to_ = range[1])
|
||||||
self.DefaultValue = value
|
except: pass
|
||||||
|
self.DefaultValue = value
|
||||||
|
if disable == True:
|
||||||
|
self.TKScale['state'] = 'disabled'
|
||||||
|
elif disable == False:
|
||||||
|
self.TKScale['state'] = 'normal'
|
||||||
|
|
||||||
def SliderChangedHandler(self, event):
|
def SliderChangedHandler(self, event):
|
||||||
# first, get the results table built
|
# first, get the results table built
|
||||||
|
@ -2737,8 +2783,7 @@ def StartupTK(my_flex_form):
|
||||||
|
|
||||||
my_flex_form.TKroot = root
|
my_flex_form.TKroot = root
|
||||||
# Make moveable window
|
# Make moveable window
|
||||||
if ((my_flex_form.NoTitleBar or my_flex_form.GrabAnywhere in (None, True)) and not my_flex_form.NonBlocking) or \
|
if (my_flex_form.GrabAnywhere is not False and not (my_flex_form.NonBlocking and my_flex_form.GrabAnywhere is not True)):
|
||||||
(my_flex_form.GrabAnywhere == True and my_flex_form.NonBlocking):
|
|
||||||
root.bind("<ButtonPress-1>", my_flex_form.StartMove)
|
root.bind("<ButtonPress-1>", my_flex_form.StartMove)
|
||||||
root.bind("<ButtonRelease-1>", my_flex_form.StopMove)
|
root.bind("<ButtonRelease-1>", my_flex_form.StopMove)
|
||||||
root.bind("<B1-Motion>", my_flex_form.OnMotion)
|
root.bind("<B1-Motion>", my_flex_form.OnMotion)
|
||||||
|
@ -3556,7 +3601,7 @@ def ObjToString(obj, extra=' '):
|
||||||
# Exits via an OK button2 press #
|
# Exits via an OK button2 press #
|
||||||
# Returns nothing #
|
# Returns nothing #
|
||||||
# ===================================================#
|
# ===================================================#
|
||||||
def Popup(*args, button_color=None, button_type=MSG_BOX_OK, auto_close=False, auto_close_duration=None, non_blocking=False, icon=DEFAULT_WINDOW_ICON, line_width=None, font=None, no_titlebar=False):
|
def Popup(*args, button_color=None, button_type=MSG_BOX_OK, auto_close=False, auto_close_duration=None, non_blocking=False, icon=DEFAULT_WINDOW_ICON, line_width=None, font=None, no_titlebar=False, keep_on_top=False):
|
||||||
'''
|
'''
|
||||||
Show message box. Displays one line per user supplied argument. Takes any Type of variable to display.
|
Show message box. Displays one line per user supplied argument. Takes any Type of variable to display.
|
||||||
:param args:
|
:param args:
|
||||||
|
@ -3578,7 +3623,7 @@ def Popup(*args, button_color=None, button_type=MSG_BOX_OK, auto_close=False, au
|
||||||
else:
|
else:
|
||||||
local_line_width = MESSAGE_BOX_LINE_WIDTH
|
local_line_width = MESSAGE_BOX_LINE_WIDTH
|
||||||
title = args_to_print[0] if args_to_print[0] is not None else 'None'
|
title = args_to_print[0] if args_to_print[0] is not None else 'None'
|
||||||
with FlexForm(title, auto_size_text=True, button_color=button_color, auto_close=auto_close, auto_close_duration=auto_close_duration, icon=icon, font=font, no_titlebar=no_titlebar) as form:
|
with FlexForm(title, auto_size_text=True, button_color=button_color, auto_close=auto_close, auto_close_duration=auto_close_duration, icon=icon, font=font, no_titlebar=no_titlebar, keep_on_top=keep_on_top) as form:
|
||||||
max_line_total, total_lines = 0,0
|
max_line_total, total_lines = 0,0
|
||||||
for message in args_to_print:
|
for message in args_to_print:
|
||||||
# fancy code to check if string and convert if not is not need. Just always convert to string :-)
|
# fancy code to check if string and convert if not is not need. Just always convert to string :-)
|
||||||
|
|
Loading…
Reference in New Issue