| import string | 
 | import sys | 
 | import os | 
 | from Tkinter import * | 
 | import linecache | 
 | from repr import Repr | 
 | from WindowList import ListedToplevel | 
 |  | 
 | from ScrolledList import ScrolledList | 
 |  | 
 |  | 
 | class StackBrowser: | 
 |  | 
 |     def __init__(self, root, flist, stack=None): | 
 |         self.top = top = ListedToplevel(root) | 
 |         top.protocol("WM_DELETE_WINDOW", self.close) | 
 |         top.bind("<Key-Escape>", self.close) | 
 |         top.wm_title("Stack viewer") | 
 |         top.wm_iconname("Stack") | 
 |         # Create help label | 
 |         self.helplabel = Label(top, | 
 |             text="Click once to view variables; twice for source", | 
 |             borderwidth=2, relief="groove") | 
 |         self.helplabel.pack(fill="x") | 
 |         # | 
 |         self.sv = StackViewer(top, flist, self) | 
 |         if stack is None: | 
 |             stack = get_stack() | 
 |         self.sv.load_stack(stack) | 
 |  | 
 |     def close(self, event=None): | 
 |         self.top.destroy() | 
 |  | 
 |     localsframe = None | 
 |     localsviewer = None | 
 |     localsdict = None | 
 |     globalsframe = None | 
 |     globalsviewer = None | 
 |     globalsdict = None | 
 |     curframe = None | 
 |  | 
 |     def show_frame(self, (frame, lineno)): | 
 |         if frame is self.curframe: | 
 |             return | 
 |         self.curframe = None | 
 |         if frame.f_globals is not self.globalsdict: | 
 |             self.show_globals(frame) | 
 |         self.show_locals(frame) | 
 |         self.curframe = frame | 
 |  | 
 |     def show_globals(self, frame): | 
 |         title = "Global Variables" | 
 |         if frame.f_globals.has_key("__name__"): | 
 |             try: | 
 |                 name = str(frame.f_globals["__name__"]) + "" | 
 |             except: | 
 |                 name = "" | 
 |             if name: | 
 |                 title = title + " in module " + name | 
 |         self.globalsdict = None | 
 |         if self.globalsviewer: | 
 |             self.globalsviewer.close() | 
 |         self.globalsviewer = None | 
 |         if not self.globalsframe: | 
 |             self.globalsframe = Frame(self.top) | 
 |         self.globalsdict = frame.f_globals | 
 |         self.globalsviewer = NamespaceViewer( | 
 |             self.globalsframe, | 
 |             title, | 
 |             self.globalsdict) | 
 |         self.globalsframe.pack(fill="both", side="bottom") | 
 |  | 
 |     def show_locals(self, frame): | 
 |         self.localsdict = None | 
 |         if self.localsviewer: | 
 |             self.localsviewer.close() | 
 |         self.localsviewer = None | 
 |         if frame.f_locals is not frame.f_globals: | 
 |             title = "Local Variables" | 
 |             code = frame.f_code | 
 |             funcname = code.co_name | 
 |             if funcname not in ("?", "", None): | 
 |                 title = title + " in " + funcname | 
 |             if not self.localsframe: | 
 |                 self.localsframe = Frame(self.top) | 
 |             self.localsdict = frame.f_locals | 
 |             self.localsviewer = NamespaceViewer( | 
 |                 self.localsframe, | 
 |                 title, | 
 |                 self.localsdict) | 
 |             self.localsframe.pack(fill="both", side="top") | 
 |         else: | 
 |             if self.localsframe: | 
 |                 self.localsframe.forget() | 
 |  | 
 |  | 
 | class StackViewer(ScrolledList): | 
 |  | 
 |     def __init__(self, master, flist, browser): | 
 |         ScrolledList.__init__(self, master, width=80) | 
 |         self.flist = flist | 
 |         self.browser = browser | 
 |         self.stack = [] | 
 |  | 
 |     def load_stack(self, stack, index=None): | 
 |         self.stack = stack | 
 |         self.clear() | 
 | ##        if len(stack) > 10: | 
 | ##            l["height"] = 10 | 
 | ##            self.topframe.pack(expand=1) | 
 | ##        else: | 
 | ##            l["height"] = len(stack) | 
 | ##            self.topframe.pack(expand=0) | 
 |         for i in range(len(stack)): | 
 |             frame, lineno = stack[i] | 
 |             try: | 
 |                 modname = frame.f_globals["__name__"] | 
 |             except: | 
 |                 modname = "?" | 
 |             code = frame.f_code | 
 |             filename = code.co_filename | 
 |             funcname = code.co_name | 
 |             sourceline = linecache.getline(filename, lineno) | 
 |             sourceline = string.strip(sourceline) | 
 |             if funcname in ("?", "", None): | 
 |                 item = "%s, line %d: %s" % (modname, lineno, sourceline) | 
 |             else: | 
 |                 item = "%s.%s(), line %d: %s" % (modname, funcname, | 
 |                                                  lineno, sourceline) | 
 |             if i == index: | 
 |                 item = "> " + item | 
 |             self.append(item) | 
 |         if index is not None: | 
 |             self.select(index) | 
 |  | 
 |     def popup_event(self, event): | 
 |         if self.stack: | 
 |             return ScrolledList.popup_event(self, event) | 
 |  | 
 |     def fill_menu(self): | 
 |         menu = self.menu | 
 |         menu.add_command(label="Go to source line", | 
 |                          command=self.goto_source_line) | 
 |         menu.add_command(label="Show stack frame", | 
 |                          command=self.show_stack_frame) | 
 |  | 
 |     def on_select(self, index): | 
 |         if 0 <= index < len(self.stack): | 
 |             self.browser.show_frame(self.stack[index]) | 
 |  | 
 |     def on_double(self, index): | 
 |         self.show_source(index) | 
 |  | 
 |     def goto_source_line(self): | 
 |         index = self.listbox.index("active") | 
 |         self.show_source(index) | 
 |  | 
 |     def show_stack_frame(self): | 
 |         index = self.listbox.index("active") | 
 |         if 0 <= index < len(self.stack): | 
 |             self.browser.show_frame(self.stack[index]) | 
 |  | 
 |     def show_source(self, index): | 
 |         if not (0 <= index < len(self.stack)): | 
 |             return | 
 |         frame, lineno = self.stack[index] | 
 |         code = frame.f_code | 
 |         filename = code.co_filename | 
 |         if os.path.isfile(filename): | 
 |             edit = self.flist.open(filename) | 
 |             if edit: | 
 |                 edit.gotoline(lineno) | 
 |  | 
 |  | 
 | def get_stack(t=None, f=None): | 
 |     if t is None: | 
 |         t = sys.last_traceback | 
 |     stack = [] | 
 |     if t and t.tb_frame is f: | 
 |         t = t.tb_next | 
 |     while f is not None: | 
 |         stack.append((f, f.f_lineno)) | 
 |         if f is self.botframe: | 
 |             break | 
 |         f = f.f_back | 
 |     stack.reverse() | 
 |     while t is not None: | 
 |         stack.append((t.tb_frame, t.tb_lineno)) | 
 |         t = t.tb_next | 
 |     return stack | 
 |  | 
 |  | 
 | def getexception(type=None, value=None): | 
 |     if type is None: | 
 |         type = sys.last_type | 
 |         value = sys.last_value | 
 |     if hasattr(type, "__name__"): | 
 |         type = type.__name__ | 
 |     s = str(type) | 
 |     if value is not None: | 
 |         s = s + ": " + str(value) | 
 |     return s | 
 |  | 
 |  | 
 | class NamespaceViewer: | 
 |  | 
 |     def __init__(self, master, title, dict=None): | 
 |         width = 0 | 
 |         height = 40 | 
 |         if dict: | 
 |             height = 20*len(dict) # XXX 20 == observed height of Entry widget | 
 |         self.master = master | 
 |         self.title = title | 
 |         self.repr = Repr() | 
 |         self.repr.maxstring = 60 | 
 |         self.repr.maxother = 60 | 
 |         self.frame = frame = Frame(master) | 
 |         self.frame.pack(expand=1, fill="both") | 
 |         self.label = Label(frame, text=title, borderwidth=2, relief="groove") | 
 |         self.label.pack(fill="x") | 
 |         self.vbar = vbar = Scrollbar(frame, name="vbar") | 
 |         vbar.pack(side="right", fill="y") | 
 |         self.canvas = canvas = Canvas(frame, | 
 |                                       height=min(300, max(40, height)), | 
 |                                       scrollregion=(0, 0, width, height)) | 
 |         canvas.pack(side="left", fill="both", expand=1) | 
 |         vbar["command"] = canvas.yview | 
 |         canvas["yscrollcommand"] = vbar.set | 
 |         self.subframe = subframe = Frame(canvas) | 
 |         self.sfid = canvas.create_window(0, 0, window=subframe, anchor="nw") | 
 |         self.load_dict(dict) | 
 |  | 
 |     dict = -1 | 
 |  | 
 |     def load_dict(self, dict, force=0, rpc_client=None): | 
 |         if dict is self.dict and not force: | 
 |             return | 
 |         subframe = self.subframe | 
 |         frame = self.frame | 
 |         for c in subframe.children.values(): | 
 |             c.destroy() | 
 |         self.dict = None | 
 |         if not dict: | 
 |             l = Label(subframe, text="None") | 
 |             l.grid(row=0, column=0) | 
 |         else: | 
 |             names = dict.keys() | 
 |             names.sort() | 
 |             row = 0 | 
 |             for name in names: | 
 |                 value = dict[name] | 
 |                 svalue = self.repr.repr(value) # repr(value) | 
 |                 # Strip extra quotes caused by calling repr on the (already) | 
 |                 # repr'd value sent across the RPC interface: | 
 |                 if rpc_client: | 
 |                     svalue = svalue[1:-1] | 
 |                 l = Label(subframe, text=name) | 
 |                 l.grid(row=row, column=0, sticky="nw") | 
 |     ##            l = Label(subframe, text=svalue, justify="l", wraplength=300) | 
 |                 l = Entry(subframe, width=0, borderwidth=0) | 
 |                 l.insert(0, svalue) | 
 |     ##            l["state"] = "disabled" | 
 |                 l.grid(row=row, column=1, sticky="nw") | 
 |                 row = row+1 | 
 |         self.dict = dict | 
 |         # XXX Could we use a <Configure> callback for the following? | 
 |         subframe.update_idletasks() # Alas! | 
 |         width = subframe.winfo_reqwidth() | 
 |         height = subframe.winfo_reqheight() | 
 |         canvas = self.canvas | 
 |         self.canvas["scrollregion"] = (0, 0, width, height) | 
 |         if height > 300: | 
 |             canvas["height"] = 300 | 
 |             frame.pack(expand=1) | 
 |         else: | 
 |             canvas["height"] = height | 
 |             frame.pack(expand=0) | 
 |  | 
 |     def close(self): | 
 |         self.frame.destroy() |