Merge pull request #1475 from PySimpleGUI/Dev-latest
PySimpleGUIdebugger - match latest PyPI 1.7
This commit is contained in:
commit
b23e2df4ba
|
@ -1,18 +1,19 @@
|
||||||
import PySimpleGUI as sg
|
import PySimpleGUI as sg
|
||||||
import PySimpleGUIdebugger
|
import PySimpleGUIdebugger # STEP 1
|
||||||
|
|
||||||
"""
|
"""
|
||||||
Demo program that shows you how to integrate the PySimpleGUI Debugger
|
Demo program that shows you how to integrate the PySimpleGUI Debugger
|
||||||
into your program.
|
into your program.
|
||||||
|
In this example, the debugger is not started initiallly. You click the "Debug" button to launch it
|
||||||
There are THREE steps, and they are copy and pastes.
|
There are THREE steps, and they are copy and pastes.
|
||||||
1. At the top of your app to debug add
|
1. At the top of your app to debug add
|
||||||
import PySimpleGUIdebugger
|
import PySimpleGUIdebugger
|
||||||
2. Initialize the debugger by calling:
|
2. Initialize the debugger at the start of your program by calling:
|
||||||
PySimpleGUIdebugger.initialize()
|
PySimpleGUIdebugger.initialize()
|
||||||
2. At the top of your app's event loop add
|
3. At the top of your app's Event Loop add:
|
||||||
PySimpleGUIdebugger.refresh(locals(), globals())
|
PySimpleGUIdebugger.refresh(locals(), globals())
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
layout = [
|
layout = [
|
||||||
[sg.T('A typical PSG application')],
|
[sg.T('A typical PSG application')],
|
||||||
[sg.In(key='_IN_')],
|
[sg.In(key='_IN_')],
|
||||||
|
@ -20,22 +21,26 @@ layout = [
|
||||||
[sg.Radio('a',1, key='_R1_'), sg.Radio('b',1, key='_R2_'), sg.Radio('c',1, key='_R3_')],
|
[sg.Radio('a',1, key='_R1_'), sg.Radio('b',1, key='_R2_'), sg.Radio('c',1, key='_R3_')],
|
||||||
[sg.Combo(['c1', 'c2', 'c3'], size=(6,3), key='_COMBO_')],
|
[sg.Combo(['c1', 'c2', 'c3'], size=(6,3), key='_COMBO_')],
|
||||||
[sg.Output(size=(50,6))],
|
[sg.Output(size=(50,6))],
|
||||||
[sg.Ok(), sg.Exit()],
|
[sg.Ok(), sg.Exit(), sg.B('Debug')],
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
window = sg.Window('This is your Application Window', layout)
|
window = sg.Window('This is your Application Window', layout)
|
||||||
# Variables that we'll use to demonstrate the debugger's features
|
|
||||||
counter = 0
|
counter = 0
|
||||||
timeout = 100
|
timeout = 100
|
||||||
PySimpleGUIdebugger.initialize()
|
debug_started = False
|
||||||
|
|
||||||
while True: # Event Loop
|
while True: # Your Event Loop
|
||||||
PySimpleGUIdebugger.refresh(locals(), globals()) # call the debugger to refresh the items being shown
|
if debug_started:
|
||||||
|
debug_started = PySimpleGUIdebugger.refresh(locals(), globals()) # STEP 3 - refresh debugger
|
||||||
event, values = window.Read(timeout=timeout)
|
event, values = window.Read(timeout=timeout)
|
||||||
if event in (None, 'Exit'):
|
if event in (None, 'Exit'):
|
||||||
break
|
break
|
||||||
elif event == 'Ok':
|
elif event == 'Ok':
|
||||||
print('You clicked Ok.... this is where print output goes')
|
print('You clicked Ok.... this is where print output goes')
|
||||||
|
elif event == 'Debug' and not debug_started:
|
||||||
|
PySimpleGUIdebugger.initialize() # STEP 2
|
||||||
|
debug_started = True
|
||||||
counter += 1
|
counter += 1
|
||||||
window.Element('_OUT_').Update(values['_IN_'])
|
window.Element('_OUT_').Update(values['_IN_'])
|
||||||
|
window.Close()
|
||||||
|
|
|
@ -16,10 +16,10 @@ WIDTH_RESULTS = 36
|
||||||
|
|
||||||
# done purely for testing / show
|
# done purely for testing / show
|
||||||
def func(x=''):
|
def func(x=''):
|
||||||
return f'return value from func()={x}'
|
return 'return value from func()={}'.format(x)
|
||||||
|
|
||||||
|
|
||||||
def _init():
|
def non_user_init():
|
||||||
global watcher_window
|
global watcher_window
|
||||||
sg.ChangeLookAndFeel(COLOR_SCHEME)
|
sg.ChangeLookAndFeel(COLOR_SCHEME)
|
||||||
def InVar(key1, key2):
|
def InVar(key1, key2):
|
||||||
|
@ -44,15 +44,18 @@ def _init():
|
||||||
window.Element('_INTERACTIVE_').SetFocus()
|
window.Element('_INTERACTIVE_').SetFocus()
|
||||||
watcher_window = window
|
watcher_window = window
|
||||||
sg.ChangeLookAndFeel('SystemDefault')
|
sg.ChangeLookAndFeel('SystemDefault')
|
||||||
|
|
||||||
return window
|
return window
|
||||||
|
|
||||||
def _event_once(_window, mylocals, myglobals):
|
def _event_once(mylocals, myglobals):
|
||||||
|
global myrc, watcher_window
|
||||||
|
if not watcher_window:
|
||||||
|
return
|
||||||
_window = watcher_window
|
_window = watcher_window
|
||||||
global myrc
|
_event, _values = _window.Read(timeout=1)
|
||||||
_event, _values = _window.Read(timeout=100)
|
|
||||||
if _event in (None, 'Exit'):
|
if _event in (None, 'Exit'):
|
||||||
return False
|
_window.Close()
|
||||||
|
watcher_window = None
|
||||||
|
return
|
||||||
cmd = _values['_INTERACTIVE_']
|
cmd = _values['_INTERACTIVE_']
|
||||||
if _event == 'Run':
|
if _event == 'Run':
|
||||||
_runCommand(cmd=cmd, window=_window)
|
_runCommand(cmd=cmd, window=_window)
|
||||||
|
@ -61,7 +64,7 @@ def _event_once(_window, mylocals, myglobals):
|
||||||
_window.Element('_OUTPUT_').Update(">>> {}\n".format(cmd), append=True, autoscroll=True)
|
_window.Element('_OUTPUT_').Update(">>> {}\n".format(cmd), append=True, autoscroll=True)
|
||||||
expression = """
|
expression = """
|
||||||
global myrc
|
global myrc
|
||||||
PSGdebugger.myrc = {} """.format(cmd)
|
PySimpleGUIdebugger.PySimpleGUIdebugger.myrc = {} """.format(cmd)
|
||||||
try:
|
try:
|
||||||
exec(expression, myglobals, mylocals)
|
exec(expression, myglobals, mylocals)
|
||||||
_window.Element('_OUTPUT_').Update('{}\n'.format(myrc),append=True, autoscroll=True)
|
_window.Element('_OUTPUT_').Update('{}\n'.format(myrc),append=True, autoscroll=True)
|
||||||
|
@ -72,23 +75,23 @@ PSGdebugger.myrc = {} """.format(cmd)
|
||||||
elif _event.endswith('_DETAIL_'):
|
elif _event.endswith('_DETAIL_'):
|
||||||
expression = """
|
expression = """
|
||||||
global myrc
|
global myrc
|
||||||
PSGdebugger.myrc = {} """.format(_values[f'_VAR{_event[4]}_'])
|
PySimpleGUIdebugger.PySimpleGUIdebugger.myrc = {} """.format(_values['_VAR{}_'.format(_event[4])])
|
||||||
try:
|
try:
|
||||||
exec(expression, myglobals, mylocals)
|
exec(expression, myglobals, mylocals)
|
||||||
sg.PopupScrolled(myrc)
|
sg.PopupScrolled(str(_values['_VAR{}_'.format(_event[4])]) + '\n' + str(myrc))
|
||||||
except:
|
except:
|
||||||
print('Detail failed')
|
print('Detail failed')
|
||||||
|
|
||||||
# -------------------- Process the "watch list" ------------------
|
# -------------------- Process the "watch list" ------------------
|
||||||
for i in range(1, 7):
|
for i in range(1, 7):
|
||||||
key = f'_VAR{i}_'
|
key = '_VAR{}_'.format(i)
|
||||||
out_key = f'_VAR{i}_CHANGED_'
|
out_key = '_VAR{}_CHANGED_'.format(i)
|
||||||
myrc =''
|
myrc =''
|
||||||
if _window.Element(key):
|
if _window.Element(key):
|
||||||
if _values[key]:
|
if _values[key]:
|
||||||
expression = """
|
expression = """
|
||||||
global myrc
|
global myrc
|
||||||
PSGdebugger.myrc = {} """.format(_values[key])
|
PySimpleGUIdebugger.PySimpleGUIdebugger.myrc = {} """.format(_values[key])
|
||||||
try:
|
try:
|
||||||
exec(expression, myglobals, mylocals)
|
exec(expression, myglobals, mylocals)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
@ -96,7 +99,7 @@ PSGdebugger.myrc = {} """.format(_values[key])
|
||||||
_window.Element(out_key).Update(myrc)
|
_window.Element(out_key).Update(myrc)
|
||||||
else:
|
else:
|
||||||
_window.Element(out_key).Update('')
|
_window.Element(out_key).Update('')
|
||||||
|
return
|
||||||
|
|
||||||
|
|
||||||
def _runCommand(cmd, timeout=None, window=None):
|
def _runCommand(cmd, timeout=None, window=None):
|
||||||
|
@ -118,8 +121,11 @@ def _runCommand(cmd, timeout=None, window=None):
|
||||||
return (retval, output)
|
return (retval, output)
|
||||||
|
|
||||||
def refresh(locals, globals):
|
def refresh(locals, globals):
|
||||||
|
_event_once(locals, globals)
|
||||||
|
|
||||||
|
def initialize():
|
||||||
global watcher_window
|
global watcher_window
|
||||||
_event_once(watcher_window, locals, globals)
|
watcher_window = non_user_init()
|
||||||
|
|
||||||
myrc = ''
|
myrc = ''
|
||||||
watcher_window = _init()
|
watcher_window = None
|
||||||
|
|
|
@ -7,16 +7,16 @@
|
||||||
|
|
||||||
# PySimpleGUIdebugger
|
# PySimpleGUIdebugger
|
||||||
|
|
||||||
PySimpleGUI now has it's own built-in, sorta, debugger.
|
A "debugger" that's based on PySimpleGUI. It was developed to help debug PySimpleGUI based programs, but it can be used to debug any program. The only requirement is that a `refresh()` function be called on a "periodic basis".
|
||||||
|
|
||||||
What you can do with this "debugger" is:
|
What you can do with this "debugger" is:
|
||||||
* Set "watch points" that update in realtime
|
* Set "watch points" that update in realtime
|
||||||
* Write expressions that update in realtime
|
* Write expressions / code that update in realtime
|
||||||
* Use a REPL style prompt to type in "code" / modify variables
|
* Use a REPL style prompt to type in "code" / modify variables
|
||||||
|
|
||||||
All of this is done using a window secondary and separate from your primary application window.
|
All of this is done using a window secondary and separate from your primary application window.
|
||||||
|
|
||||||
Check out this video as a guide. The user's window is the smaller one one top. The PySimpleGUIdebugger is the green window on the buttom. You can watch variables, evaluate expressions, even xecute code.
|
Check out this video as a guide. The user's window is the smaller one one top. The PySimpleGUIdebugger is the green window on the buttom. You can watch variables, evaluate expressions, even execute code.
|
||||||
|
|
||||||
![PSG Debugger2](https://user-images.githubusercontent.com/13696193/58362085-3ead8f00-7e61-11e9-9439-e77e9a059dbc.gif)
|
![PSG Debugger2](https://user-images.githubusercontent.com/13696193/58362085-3ead8f00-7e61-11e9-9439-e77e9a059dbc.gif)
|
||||||
|
|
||||||
|
@ -98,14 +98,73 @@ This "refresh" call that must be added to your event loop. Your `window.Read` c
|
||||||
Add this line to the top of your event loop.
|
Add this line to the top of your event loop.
|
||||||
`PySimpleGUIdebugger.refresh(locals(), globals())`
|
`PySimpleGUIdebugger.refresh(locals(), globals())`
|
||||||
|
|
||||||
|
### Using in "when needed"
|
||||||
|
|
||||||
|
The Demo Program was recently updated so that instead of launching with the Debugger window immediately shown, the program launches with the Debugger not started. With this new code, you can open and close the Debugger as many times as you wish.
|
||||||
|
|
||||||
|
Here is the code, based on the code shown previously in this readme, that has a "Debug" button
|
||||||
|
|
||||||
|
```python
|
||||||
|
import PySimpleGUI as sg
|
||||||
|
import PySimpleGUIdebugger # STEP 1
|
||||||
|
|
||||||
|
"""
|
||||||
|
Demo program that shows you how to integrate the PySimpleGUI Debugger
|
||||||
|
into your program.
|
||||||
|
In this example, the debugger is not started initiallly. You click the "Debug" button to launch it
|
||||||
|
There are THREE steps, and they are copy and pastes.
|
||||||
|
1. At the top of your app to debug add
|
||||||
|
import PySimpleGUIdebugger
|
||||||
|
2. Initialize the debugger at the start of your program by calling:
|
||||||
|
PySimpleGUIdebugger.initialize()
|
||||||
|
3. At the top of your app's Event Loop add:
|
||||||
|
PySimpleGUIdebugger.refresh(locals(), globals())
|
||||||
|
"""
|
||||||
|
|
||||||
|
layout = [
|
||||||
|
[sg.T('A typical PSG application')],
|
||||||
|
[sg.In(key='_IN_')],
|
||||||
|
[sg.T(' ', key='_OUT_')],
|
||||||
|
[sg.Radio('a',1, key='_R1_'), sg.Radio('b',1, key='_R2_'), sg.Radio('c',1, key='_R3_')],
|
||||||
|
[sg.Combo(['c1', 'c2', 'c3'], size=(6,3), key='_COMBO_')],
|
||||||
|
[sg.Output(size=(50,6))],
|
||||||
|
[sg.Ok(), sg.Exit(), sg.B('Debug')],
|
||||||
|
]
|
||||||
|
|
||||||
|
window = sg.Window('This is your Application Window', layout)
|
||||||
|
|
||||||
|
counter = 0
|
||||||
|
timeout = 100
|
||||||
|
debug_started = False
|
||||||
|
|
||||||
|
while True: # Your Event Loop
|
||||||
|
if debug_started:
|
||||||
|
debug_started = PySimpleGUIdebugger.refresh(locals(), globals()) # STEP 3 - refresh debugger
|
||||||
|
event, values = window.Read(timeout=timeout)
|
||||||
|
if event in (None, 'Exit'):
|
||||||
|
break
|
||||||
|
elif event == 'Ok':
|
||||||
|
print('You clicked Ok.... this is where print output goes')
|
||||||
|
elif event == 'Debug' and not debug_started:
|
||||||
|
PySimpleGUIdebugger.initialize() # STEP 2
|
||||||
|
debug_started = True
|
||||||
|
counter += 1
|
||||||
|
window.Element('_OUT_').Update(values['_IN_'])
|
||||||
|
window.Close()
|
||||||
|
```
|
||||||
|
|
||||||
|
This puts the launching of the debugger firmly into the control of the program being debugged. Want debugger help? Then press the debug button.
|
||||||
|
|
||||||
|
In the future I want to add a "hotkey" or some other trivial way of launching a debugger from any program that has is running PySimpleGUI. The only one with real trouble wit this will be the PySimpleGUIWeb one as multiple web windows gets a bit cluttered.
|
||||||
|
|
||||||
|
|
||||||
## Requirements
|
## Requirements
|
||||||
|
|
||||||
You'll need to have PySimpleGUI installed.
|
**You'll need to have PySimpleGUI installed.**
|
||||||
|
|
||||||
The debugger itself is written using PySimpleGUI, the tkinter version. It could be changed to use Qt for example.
|
The newest PyPI installation should automatically install PySimpleGUI for you now.
|
||||||
|
|
||||||
|
The debugger itself is written using PySimpleGUI, the tkinter version. It could be changed to use Qt for example by modifying the pip installed version.
|
||||||
|
|
||||||
## What's it good for, when should it be used??
|
## What's it good for, when should it be used??
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue