| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | """Class browser.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | XXX TO DO: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - reparse when source changed (maybe just a button would be OK?) | 
					
						
							|  |  |  |     (or recheck on window popup) | 
					
						
							|  |  |  | - add popup menu with more options (e.g. doc strings, base classes, imports) | 
					
						
							|  |  |  | - show function argument list? (have to do pattern matching on source) | 
					
						
							|  |  |  | - should the classes and methods lists also be in the module's menu bar? | 
					
						
							|  |  |  | - add base classes to class browser tree | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import os | 
					
						
							|  |  |  | import pyclbr | 
					
						
							| 
									
										
										
										
											2016-08-31 00:50:55 -04:00
										 |  |  | import sys | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-31 00:50:55 -04:00
										 |  |  | from idlelib.config import idleConf | 
					
						
							| 
									
										
										
										
											2016-05-28 13:22:31 -04:00
										 |  |  | from idlelib import pyshell | 
					
						
							|  |  |  | from idlelib.tree import TreeNode, TreeItem, ScrolledCanvas | 
					
						
							| 
									
										
										
										
											2016-08-31 00:50:55 -04:00
										 |  |  | from idlelib.windows import ListedToplevel | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-17 01:31:35 -04:00
										 |  |  | file_open = None  # Method...Item and Class...Item use this. | 
					
						
							| 
									
										
										
										
											2016-05-28 13:22:31 -04:00
										 |  |  | # Normally pyshell.flist.open, but there is no pyshell.flist for htest. | 
					
						
							| 
									
										
										
										
											2014-10-17 01:31:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | class ClassBrowser: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-24 18:48:18 -04:00
										 |  |  |     def __init__(self, flist, name, path, _htest=False): | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         # XXX This API should change, if the file doesn't end in ".py" | 
					
						
							|  |  |  |         # XXX the code here is bogus! | 
					
						
							| 
									
										
										
										
											2014-05-24 18:48:18 -04:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         _htest - bool, change box when location running htest. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2014-10-17 01:31:35 -04:00
										 |  |  |         global file_open | 
					
						
							|  |  |  |         if not _htest: | 
					
						
							| 
									
										
										
										
											2016-05-28 13:22:31 -04:00
										 |  |  |             file_open = pyshell.flist.open | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         self.name = name | 
					
						
							|  |  |  |         self.file = os.path.join(path[0], self.name + ".py") | 
					
						
							| 
									
										
										
										
											2014-05-24 18:48:18 -04:00
										 |  |  |         self._htest = _htest | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         self.init(flist) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def close(self, event=None): | 
					
						
							|  |  |  |         self.top.destroy() | 
					
						
							|  |  |  |         self.node.destroy() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def init(self, flist): | 
					
						
							|  |  |  |         self.flist = flist | 
					
						
							|  |  |  |         # reset pyclbr | 
					
						
							|  |  |  |         pyclbr._modules.clear() | 
					
						
							|  |  |  |         # create top | 
					
						
							|  |  |  |         self.top = top = ListedToplevel(flist.root) | 
					
						
							|  |  |  |         top.protocol("WM_DELETE_WINDOW", self.close) | 
					
						
							|  |  |  |         top.bind("<Escape>", self.close) | 
					
						
							| 
									
										
										
										
											2014-05-24 18:48:18 -04:00
										 |  |  |         if self._htest: # place dialog below parent if running htest | 
					
						
							|  |  |  |             top.geometry("+%d+%d" % | 
					
						
							|  |  |  |                 (flist.root.winfo_rootx(), flist.root.winfo_rooty() + 200)) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         self.settitle() | 
					
						
							|  |  |  |         top.focus_set() | 
					
						
							|  |  |  |         # create scrolled canvas | 
					
						
							| 
									
										
										
										
											2015-11-12 15:02:57 -05:00
										 |  |  |         theme = idleConf.CurrentTheme() | 
					
						
							| 
									
										
										
										
											2004-03-08 18:15:31 +00:00
										 |  |  |         background = idleConf.GetHighlight(theme, 'normal')['background'] | 
					
						
							|  |  |  |         sc = ScrolledCanvas(top, bg=background, highlightthickness=0, takefocus=1) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         sc.frame.pack(expand=1, fill="both") | 
					
						
							|  |  |  |         item = self.rootnode() | 
					
						
							|  |  |  |         self.node = node = TreeNode(sc.canvas, None, item) | 
					
						
							|  |  |  |         node.update() | 
					
						
							|  |  |  |         node.expand() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def settitle(self): | 
					
						
							|  |  |  |         self.top.wm_title("Class Browser - " + self.name) | 
					
						
							|  |  |  |         self.top.wm_iconname("Class Browser") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def rootnode(self): | 
					
						
							|  |  |  |         return ModuleBrowserTreeItem(self.file) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ModuleBrowserTreeItem(TreeItem): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, file): | 
					
						
							|  |  |  |         self.file = file | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def GetText(self): | 
					
						
							|  |  |  |         return os.path.basename(self.file) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def GetIconName(self): | 
					
						
							|  |  |  |         return "python" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def GetSubList(self): | 
					
						
							|  |  |  |         sublist = [] | 
					
						
							|  |  |  |         for name in self.listclasses(): | 
					
						
							|  |  |  |             item = ClassBrowserTreeItem(name, self.classes, self.file) | 
					
						
							|  |  |  |             sublist.append(item) | 
					
						
							|  |  |  |         return sublist | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def OnDoubleClick(self): | 
					
						
							|  |  |  |         if os.path.normcase(self.file[-3:]) != ".py": | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         if not os.path.exists(self.file): | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2016-05-28 13:22:31 -04:00
										 |  |  |         pyshell.flist.open(self.file) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def IsExpandable(self): | 
					
						
							|  |  |  |         return os.path.normcase(self.file[-3:]) == ".py" | 
					
						
							| 
									
										
										
										
											2001-07-12 23:54:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |     def listclasses(self): | 
					
						
							|  |  |  |         dir, file = os.path.split(self.file) | 
					
						
							|  |  |  |         name, ext = os.path.splitext(file) | 
					
						
							|  |  |  |         if os.path.normcase(ext) != ".py": | 
					
						
							|  |  |  |             return [] | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             dict = pyclbr.readmodule_ex(name, [dir] + sys.path) | 
					
						
							| 
									
										
										
										
											2014-07-01 18:52:37 -04:00
										 |  |  |         except ImportError: | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |             return [] | 
					
						
							|  |  |  |         items = [] | 
					
						
							|  |  |  |         self.classes = {} | 
					
						
							|  |  |  |         for key, cl in dict.items(): | 
					
						
							|  |  |  |             if cl.module == name: | 
					
						
							|  |  |  |                 s = key | 
					
						
							| 
									
										
										
										
											2003-01-19 02:37:41 +00:00
										 |  |  |                 if hasattr(cl, 'super') and cl.super: | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |                     supers = [] | 
					
						
							|  |  |  |                     for sup in cl.super: | 
					
						
							|  |  |  |                         if type(sup) is type(''): | 
					
						
							|  |  |  |                             sname = sup | 
					
						
							|  |  |  |                         else: | 
					
						
							|  |  |  |                             sname = sup.name | 
					
						
							|  |  |  |                             if sup.module != cl.module: | 
					
						
							|  |  |  |                                 sname = "%s.%s" % (sup.module, sname) | 
					
						
							|  |  |  |                         supers.append(sname) | 
					
						
							| 
									
										
										
										
											2002-09-15 22:09:16 +00:00
										 |  |  |                     s = s + "(%s)" % ", ".join(supers) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |                 items.append((cl.lineno, s)) | 
					
						
							|  |  |  |                 self.classes[s] = cl | 
					
						
							|  |  |  |         items.sort() | 
					
						
							|  |  |  |         list = [] | 
					
						
							|  |  |  |         for item, s in items: | 
					
						
							|  |  |  |             list.append(s) | 
					
						
							|  |  |  |         return list | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ClassBrowserTreeItem(TreeItem): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, name, classes, file): | 
					
						
							|  |  |  |         self.name = name | 
					
						
							|  |  |  |         self.classes = classes | 
					
						
							|  |  |  |         self.file = file | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             self.cl = self.classes[self.name] | 
					
						
							|  |  |  |         except (IndexError, KeyError): | 
					
						
							|  |  |  |             self.cl = None | 
					
						
							|  |  |  |         self.isfunction = isinstance(self.cl, pyclbr.Function) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def GetText(self): | 
					
						
							|  |  |  |         if self.isfunction: | 
					
						
							|  |  |  |             return "def " + self.name + "(...)" | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return "class " + self.name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def GetIconName(self): | 
					
						
							|  |  |  |         if self.isfunction: | 
					
						
							|  |  |  |             return "python" | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return "folder" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def IsExpandable(self): | 
					
						
							|  |  |  |         if self.cl: | 
					
						
							| 
									
										
										
										
											2003-01-20 04:49:37 +00:00
										 |  |  |             try: | 
					
						
							|  |  |  |                 return not not self.cl.methods | 
					
						
							|  |  |  |             except AttributeError: | 
					
						
							|  |  |  |                 return False | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def GetSubList(self): | 
					
						
							|  |  |  |         if not self.cl: | 
					
						
							|  |  |  |             return [] | 
					
						
							|  |  |  |         sublist = [] | 
					
						
							|  |  |  |         for name in self.listmethods(): | 
					
						
							|  |  |  |             item = MethodBrowserTreeItem(name, self.cl, self.file) | 
					
						
							|  |  |  |             sublist.append(item) | 
					
						
							|  |  |  |         return sublist | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def OnDoubleClick(self): | 
					
						
							|  |  |  |         if not os.path.exists(self.file): | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2014-10-17 01:31:35 -04:00
										 |  |  |         edit = file_open(self.file) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         if hasattr(self.cl, 'lineno'): | 
					
						
							|  |  |  |             lineno = self.cl.lineno | 
					
						
							|  |  |  |             edit.gotoline(lineno) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def listmethods(self): | 
					
						
							|  |  |  |         if not self.cl: | 
					
						
							|  |  |  |             return [] | 
					
						
							|  |  |  |         items = [] | 
					
						
							|  |  |  |         for name, lineno in self.cl.methods.items(): | 
					
						
							|  |  |  |             items.append((lineno, name)) | 
					
						
							|  |  |  |         items.sort() | 
					
						
							|  |  |  |         list = [] | 
					
						
							|  |  |  |         for item, name in items: | 
					
						
							|  |  |  |             list.append(name) | 
					
						
							|  |  |  |         return list | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MethodBrowserTreeItem(TreeItem): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, name, cl, file): | 
					
						
							|  |  |  |         self.name = name | 
					
						
							|  |  |  |         self.cl = cl | 
					
						
							|  |  |  |         self.file = file | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def GetText(self): | 
					
						
							|  |  |  |         return "def " + self.name + "(...)" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def GetIconName(self): | 
					
						
							|  |  |  |         return "python" # XXX | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def IsExpandable(self): | 
					
						
							|  |  |  |         return 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def OnDoubleClick(self): | 
					
						
							|  |  |  |         if not os.path.exists(self.file): | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2014-10-17 01:31:35 -04:00
										 |  |  |         edit = file_open(self.file) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |         edit.gotoline(self.cl.methods[self.name]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-24 18:48:18 -04:00
										 |  |  | def _class_browser(parent): #Wrapper for htest | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  |     try: | 
					
						
							|  |  |  |         file = __file__ | 
					
						
							|  |  |  |     except NameError: | 
					
						
							|  |  |  |         file = sys.argv[0] | 
					
						
							|  |  |  |         if sys.argv[1:]: | 
					
						
							|  |  |  |             file = sys.argv[1] | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             file = sys.argv[0] | 
					
						
							|  |  |  |     dir, file = os.path.split(file) | 
					
						
							|  |  |  |     name = os.path.splitext(file)[0] | 
					
						
							| 
									
										
										
										
											2016-05-28 13:22:31 -04:00
										 |  |  |     flist = pyshell.PyShellFileList(parent) | 
					
						
							| 
									
										
										
										
											2014-10-17 01:31:35 -04:00
										 |  |  |     global file_open | 
					
						
							|  |  |  |     file_open = flist.open | 
					
						
							| 
									
										
										
										
											2014-05-24 18:48:18 -04:00
										 |  |  |     ClassBrowser(flist, name, [dir], _htest=True) | 
					
						
							| 
									
										
										
										
											2000-08-15 01:13:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							| 
									
										
										
										
											2014-05-24 18:48:18 -04:00
										 |  |  |     from idlelib.idle_test.htest import run | 
					
						
							|  |  |  |     run(_class_browser) |