Mostly rewritten. Instead of the old Run module and Debug module,

there are two new commands:

Import module (F5) imports or reloads the module and also adds its
name to the __main__ namespace.  This gets executed in the PyShell
window under control of its debug settings.

Run script (Control-F5) is similar but executes the contents of the
file directly in the __main__ namespace.
This commit is contained in:
Guido van Rossum 1999-04-22 22:27:40 +00:00
parent 3af507de11
commit 1f3de5d7b9
1 changed files with 46 additions and 107 deletions

View File

@ -3,88 +3,33 @@
This adds two commands (to the Edit menu, until there's a separate
Python menu):
- Run module (F5) is equivalent to either import or reload of the
- Import module (F5) is equivalent to either import or reload of the
current module. The window must have been saved previously. The
module only gets added to sys.modules, it doesn't get added to
anyone's namespace; you can import it in the shell if you need to. If
this generates any output to sys.stdout or sys.stderr, a new output
window is created to display that output. The two streams are
distinguished by their text color.
module is added to sys.modules, and is also added to the __main__
namespace. Output goes to the shell window.
- Debug module (Control-F5) does the same but executes the module's
code in the debugger.
When an unhandled exception occurs in Run module, the stack viewer is
popped up. This is not done in Debug module, because you've already
had an opportunity to view the stack. In either case, the variables
sys.last_type, sys.last_value, sys.last_traceback are set to the
exception info.
- Run module (Control-F5) does the same but executes the module's
code in the __main__ namespace.
"""
import sys
import os
import imp
import linecache
import traceback
import tkMessageBox
from OutputWindow import OutputWindow
# XXX These helper classes are more generally usable!
class OnDemandOutputWindow:
tagdefs = {
"stdout": {"foreground": "blue"},
"stderr": {"foreground": "#007700"},
}
def __init__(self, flist):
self.flist = flist
self.owin = None
def write(self, s, tags, mark):
if not self.owin:
self.setup()
self.owin.write(s, tags, mark)
def setup(self):
self.owin = owin = OutputWindow(self.flist)
text = owin.text
for tag, cnf in self.tagdefs.items():
if cnf:
apply(text.tag_configure, (tag,), cnf)
text.tag_raise('sel')
self.write = self.owin.write
class PseudoFile:
def __init__(self, owin, tags, mark="end"):
self.owin = owin
self.tags = tags
self.mark = mark
def write(self, s):
self.owin.write(s, self.tags, self.mark)
def writelines(self, l):
map(self.write, l)
def flush(self):
pass
class ScriptBinding:
keydefs = {
'<<run-module>>': ['<F5>'],
'<<debug-module>>': ['<Control-F5>'],
'<<import-module>>': ['<F5>'],
'<<run-script>>': ['<Control-F5>'],
}
menudefs = [
('edit', [None,
('Run module', '<<run-module>>'),
('Debug module', '<<debug-module>>'),
('Import module', '<<import-module>>'),
('Run script', '<<run-script>>'),
]
),
]
@ -96,7 +41,42 @@ def __init__(self, editwin):
self.flist = self.editwin.flist
self.root = self.flist.root
def run_module_event(self, event, debugger=None):
def import_module_event(self, event):
filename = self.getfilename()
if not filename:
return
modname, ext = os.path.splitext(os.path.basename(filename))
if sys.modules.has_key(modname):
mod = sys.modules[modname]
else:
mod = imp.new_module(modname)
sys.modules[modname] = mod
mod.__file__ = filename
setattr(sys.modules['__main__'], modname, mod)
dir = os.path.dirname(filename)
dir = os.path.normpath(os.path.abspath(dir))
if dir not in sys.path:
sys.path.insert(0, dir)
flist = self.editwin.flist
shell = flist.open_shell()
interp = shell.interp
interp.runcode("reload(%s)" % modname)
def run_script_event(self, event):
filename = self.getfilename()
if not filename:
return
flist = self.editwin.flist
shell = flist.open_shell()
interp = shell.interp
interp.execfile(filename)
def getfilename(self):
# Logic to make sure we have a saved filename
if not self.editwin.get_saved():
tkMessageBox.showerror("Not saved",
"Please save first!",
@ -110,45 +90,4 @@ def run_module_event(self, event, debugger=None):
master=self.editwin.text)
self.editwin.text.focus_set()
return
modname, ext = os.path.splitext(os.path.basename(filename))
if sys.modules.has_key(modname):
mod = sys.modules[modname]
else:
mod = imp.new_module(modname)
sys.modules[modname] = mod
mod.__file__ = filename
saveout = sys.stdout
saveerr = sys.stderr
owin = OnDemandOutputWindow(self.editwin.flist)
try:
sys.stderr = PseudoFile(owin, "stderr")
try:
sys.stdout = PseudoFile(owin, "stdout")
try:
if debugger:
debugger.run("execfile(%s)" % `filename`, mod.__dict__)
else:
execfile(filename, mod.__dict__)
except:
(sys.last_type, sys.last_value,
sys.last_traceback) = sys.exc_info()
linecache.checkcache()
traceback.print_exc()
if not debugger and \
self.editwin.getvar("<<toggle-jit-stack-viewer>>"):
from StackViewer import StackBrowser
sv = StackBrowser(self.root, self.flist)
finally:
sys.stdout = saveout
finally:
sys.stderr = saveerr
def debug_module_event(self, event):
import Debugger
debugger = Debugger.Debugger(self)
self.run_module_event(event, debugger)
def close_debugger(self):
# Method called by Debugger
# XXX This should be done differently
pass
return filename