Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 1 | """Class browser. |
Guido van Rossum | 7f5c9ef | 1998-10-13 16:31:03 +0000 | [diff] [blame] | 2 | |
| 3 | XXX TO DO: |
Guido van Rossum | 504b0bf | 1999-01-02 21:28:54 +0000 | [diff] [blame] | 4 | |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 5 | - reparse when source changed (maybe just a button would be OK?) |
| 6 | (or recheck on window popup) |
Guido van Rossum | 7f5c9ef | 1998-10-13 16:31:03 +0000 | [diff] [blame] | 7 | - add popup menu with more options (e.g. doc strings, base classes, imports) |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 8 | - show function argument list? (have to do pattern matching on source) |
Guido van Rossum | 7f5c9ef | 1998-10-13 16:31:03 +0000 | [diff] [blame] | 9 | - should the classes and methods lists also be in the module's menu bar? |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 10 | - add base classes to class browser tree |
Guido van Rossum | 7f5c9ef | 1998-10-13 16:31:03 +0000 | [diff] [blame] | 11 | """ |
| 12 | |
Guido van Rossum | e6fae1c | 1998-10-15 23:27:08 +0000 | [diff] [blame] | 13 | import os |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 14 | import sys |
Guido van Rossum | 439c467 | 1998-10-13 03:59:57 +0000 | [diff] [blame] | 15 | import pyclbr |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 16 | |
| 17 | import PyShell |
Guido van Rossum | 504b0bf | 1999-01-02 21:28:54 +0000 | [diff] [blame] | 18 | from WindowList import ListedToplevel |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 19 | from TreeWidget import TreeNode, TreeItem, ScrolledCanvas |
Guido van Rossum | e6fae1c | 1998-10-15 23:27:08 +0000 | [diff] [blame] | 20 | |
Guido van Rossum | 439c467 | 1998-10-13 03:59:57 +0000 | [diff] [blame] | 21 | class ClassBrowser: |
Guido van Rossum | 504b0bf | 1999-01-02 21:28:54 +0000 | [diff] [blame] | 22 | |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 23 | def __init__(self, flist, name, path): |
Guido van Rossum | f801f3b | 2000-02-15 18:02:11 +0000 | [diff] [blame] | 24 | # XXX This API should change, if the file doesn't end in ".py" |
| 25 | # XXX the code here is bogus! |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 26 | self.name = name |
| 27 | self.file = os.path.join(path[0], self.name + ".py") |
| 28 | self.init(flist) |
Guido van Rossum | 504b0bf | 1999-01-02 21:28:54 +0000 | [diff] [blame] | 29 | |
Guido van Rossum | dc5066e | 1999-01-11 14:46:06 +0000 | [diff] [blame] | 30 | def close(self, event=None): |
Guido van Rossum | 7f5c9ef | 1998-10-13 16:31:03 +0000 | [diff] [blame] | 31 | self.top.destroy() |
Guido van Rossum | 5af0df5 | 1999-06-25 17:08:19 +0000 | [diff] [blame] | 32 | self.node.destroy() |
Guido van Rossum | 504b0bf | 1999-01-02 21:28:54 +0000 | [diff] [blame] | 33 | |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 34 | def init(self, flist): |
Guido van Rossum | e6fae1c | 1998-10-15 23:27:08 +0000 | [diff] [blame] | 35 | self.flist = flist |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 36 | # reset pyclbr |
| 37 | pyclbr._modules.clear() |
| 38 | # create top |
| 39 | self.top = top = ListedToplevel(flist.root) |
| 40 | top.protocol("WM_DELETE_WINDOW", self.close) |
| 41 | top.bind("<Escape>", self.close) |
| 42 | self.settitle() |
| 43 | top.focus_set() |
| 44 | # create scrolled canvas |
| 45 | sc = ScrolledCanvas(top, bg="white", highlightthickness=0, takefocus=1) |
| 46 | sc.frame.pack(expand=1, fill="both") |
| 47 | item = self.rootnode() |
Guido van Rossum | 5af0df5 | 1999-06-25 17:08:19 +0000 | [diff] [blame] | 48 | self.node = node = TreeNode(sc.canvas, None, item) |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 49 | node.update() |
| 50 | node.expand() |
Guido van Rossum | 504b0bf | 1999-01-02 21:28:54 +0000 | [diff] [blame] | 51 | |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 52 | def settitle(self): |
| 53 | self.top.wm_title("Class Browser - " + self.name) |
| 54 | self.top.wm_iconname("Class Browser") |
| 55 | |
| 56 | def rootnode(self): |
| 57 | return ModuleBrowserTreeItem(self.file) |
| 58 | |
| 59 | class ModuleBrowserTreeItem(TreeItem): |
| 60 | |
| 61 | def __init__(self, file): |
| 62 | self.file = file |
| 63 | |
| 64 | def GetText(self): |
| 65 | return os.path.basename(self.file) |
| 66 | |
| 67 | def GetIconName(self): |
| 68 | return "python" |
| 69 | |
| 70 | def GetSubList(self): |
| 71 | sublist = [] |
| 72 | for name in self.listclasses(): |
| 73 | item = ClassBrowserTreeItem(name, self.classes, self.file) |
| 74 | sublist.append(item) |
| 75 | return sublist |
| 76 | |
| 77 | def OnDoubleClick(self): |
| 78 | if os.path.normcase(self.file[-3:]) != ".py": |
| 79 | return |
| 80 | if not os.path.exists(self.file): |
| 81 | return |
| 82 | PyShell.flist.open(self.file) |
| 83 | |
| 84 | def IsExpandable(self): |
| 85 | return os.path.normcase(self.file[-3:]) == ".py" |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 86 | |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 87 | def listclasses(self): |
| 88 | dir, file = os.path.split(self.file) |
| 89 | name, ext = os.path.splitext(file) |
| 90 | if os.path.normcase(ext) != ".py": |
| 91 | return [] |
| 92 | try: |
Guido van Rossum | 4431b0f | 1999-06-10 15:19:14 +0000 | [diff] [blame] | 93 | dict = pyclbr.readmodule_ex(name, [dir] + sys.path) |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 94 | except ImportError, msg: |
| 95 | return [] |
Guido van Rossum | 7f5c9ef | 1998-10-13 16:31:03 +0000 | [diff] [blame] | 96 | items = [] |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 97 | self.classes = {} |
| 98 | for key, cl in dict.items(): |
| 99 | if cl.module == name: |
| 100 | s = key |
Raymond Hettinger | 18acea7 | 2003-01-18 22:53:36 +0000 | [diff] [blame] | 101 | if hasattr(cl, "super") and cl.super: |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 102 | supers = [] |
| 103 | for sup in cl.super: |
| 104 | if type(sup) is type(''): |
| 105 | sname = sup |
| 106 | else: |
| 107 | sname = sup.name |
| 108 | if sup.module != cl.module: |
| 109 | sname = "%s.%s" % (sup.module, sname) |
| 110 | supers.append(sname) |
Walter Dörwald | aaab30e | 2002-09-11 20:36:02 +0000 | [diff] [blame] | 111 | s = s + "(%s)" % ", ".join(supers) |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 112 | items.append((cl.lineno, s)) |
| 113 | self.classes[s] = cl |
Guido van Rossum | 439c467 | 1998-10-13 03:59:57 +0000 | [diff] [blame] | 114 | items.sort() |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 115 | list = [] |
| 116 | for item, s in items: |
| 117 | list.append(s) |
| 118 | return list |
Guido van Rossum | 504b0bf | 1999-01-02 21:28:54 +0000 | [diff] [blame] | 119 | |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 120 | class ClassBrowserTreeItem(TreeItem): |
| 121 | |
| 122 | def __init__(self, name, classes, file): |
| 123 | self.name = name |
| 124 | self.classes = classes |
| 125 | self.file = file |
Guido van Rossum | 4431b0f | 1999-06-10 15:19:14 +0000 | [diff] [blame] | 126 | try: |
| 127 | self.cl = self.classes[self.name] |
| 128 | except (IndexError, KeyError): |
| 129 | self.cl = None |
| 130 | self.isfunction = isinstance(self.cl, pyclbr.Function) |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 131 | |
| 132 | def GetText(self): |
Guido van Rossum | 4431b0f | 1999-06-10 15:19:14 +0000 | [diff] [blame] | 133 | if self.isfunction: |
| 134 | return "def " + self.name + "(...)" |
| 135 | else: |
| 136 | return "class " + self.name |
| 137 | |
| 138 | def GetIconName(self): |
| 139 | if self.isfunction: |
| 140 | return "python" |
| 141 | else: |
| 142 | return "folder" |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 143 | |
| 144 | def IsExpandable(self): |
Raymond Hettinger | 96bf0d7 | 2003-01-22 13:29:00 +0000 | [diff] [blame^] | 145 | try: |
| 146 | return bool(self.cl.methods) |
| 147 | except AttributeError: |
| 148 | return False |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 149 | |
| 150 | def GetSubList(self): |
Guido van Rossum | 4431b0f | 1999-06-10 15:19:14 +0000 | [diff] [blame] | 151 | if not self.cl: |
| 152 | return [] |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 153 | sublist = [] |
| 154 | for name in self.listmethods(): |
Guido van Rossum | 4431b0f | 1999-06-10 15:19:14 +0000 | [diff] [blame] | 155 | item = MethodBrowserTreeItem(name, self.cl, self.file) |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 156 | sublist.append(item) |
| 157 | return sublist |
| 158 | |
| 159 | def OnDoubleClick(self): |
| 160 | if not os.path.exists(self.file): |
| 161 | return |
| 162 | edit = PyShell.flist.open(self.file) |
Guido van Rossum | 4431b0f | 1999-06-10 15:19:14 +0000 | [diff] [blame] | 163 | if hasattr(self.cl, 'lineno'): |
| 164 | lineno = self.cl.lineno |
| 165 | edit.gotoline(lineno) |
Guido van Rossum | 7f5c9ef | 1998-10-13 16:31:03 +0000 | [diff] [blame] | 166 | |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 167 | def listmethods(self): |
Guido van Rossum | 4431b0f | 1999-06-10 15:19:14 +0000 | [diff] [blame] | 168 | if not self.cl: |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 169 | return [] |
Guido van Rossum | 7f5c9ef | 1998-10-13 16:31:03 +0000 | [diff] [blame] | 170 | items = [] |
Guido van Rossum | 4431b0f | 1999-06-10 15:19:14 +0000 | [diff] [blame] | 171 | for name, lineno in self.cl.methods.items(): |
Guido van Rossum | 7f5c9ef | 1998-10-13 16:31:03 +0000 | [diff] [blame] | 172 | items.append((lineno, name)) |
| 173 | items.sort() |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 174 | list = [] |
Guido van Rossum | 7f5c9ef | 1998-10-13 16:31:03 +0000 | [diff] [blame] | 175 | for item, name in items: |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 176 | list.append(name) |
| 177 | return list |
Guido van Rossum | 439c467 | 1998-10-13 03:59:57 +0000 | [diff] [blame] | 178 | |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 179 | class MethodBrowserTreeItem(TreeItem): |
Guido van Rossum | 504b0bf | 1999-01-02 21:28:54 +0000 | [diff] [blame] | 180 | |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 181 | def __init__(self, name, cl, file): |
| 182 | self.name = name |
| 183 | self.cl = cl |
| 184 | self.file = file |
Guido van Rossum | 504b0bf | 1999-01-02 21:28:54 +0000 | [diff] [blame] | 185 | |
Guido van Rossum | ec9cca7 | 1999-06-01 18:21:31 +0000 | [diff] [blame] | 186 | def GetText(self): |
| 187 | return "def " + self.name + "(...)" |
| 188 | |
| 189 | def GetIconName(self): |
| 190 | return "python" # XXX |
| 191 | |
| 192 | def IsExpandable(self): |
| 193 | return 0 |
| 194 | |
| 195 | def OnDoubleClick(self): |
| 196 | if not os.path.exists(self.file): |
| 197 | return |
| 198 | edit = PyShell.flist.open(self.file) |
| 199 | edit.gotoline(self.cl.methods[self.name]) |
| 200 | |
| 201 | def main(): |
| 202 | try: |
| 203 | file = __file__ |
| 204 | except NameError: |
| 205 | file = sys.argv[0] |
| 206 | if sys.argv[1:]: |
| 207 | file = sys.argv[1] |
| 208 | else: |
| 209 | file = sys.argv[0] |
| 210 | dir, file = os.path.split(file) |
| 211 | name = os.path.splitext(file)[0] |
| 212 | ClassBrowser(PyShell.flist, name, [dir]) |
| 213 | if sys.stdin is sys.__stdin__: |
| 214 | mainloop() |
| 215 | |
| 216 | if __name__ == "__main__": |
| 217 | main() |