git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@19793 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
437 lines
16 KiB
Python
437 lines
16 KiB
Python
"""PyCrust Filling is the gui tree control through which a user can
|
|
navigate the local namespace or any object."""
|
|
|
|
__author__ = "Patrick K. O'Brien <pobrien@orbtech.com>"
|
|
__cvsid__ = "$Id$"
|
|
__revision__ = "$Revision$"[11:-2]
|
|
|
|
from wxPython import wx
|
|
from wxPython import stc
|
|
from version import VERSION
|
|
import dispatcher
|
|
import inspect
|
|
import introspect
|
|
import keyword
|
|
import sys
|
|
import types
|
|
|
|
try:
|
|
True
|
|
except NameError:
|
|
True = 1==1
|
|
False = 1==0
|
|
|
|
COMMONTYPES = [getattr(types, t) for t in dir(types) \
|
|
if not t.startswith('_') \
|
|
and t not in ('ClassType', 'InstanceType', 'ModuleType')]
|
|
|
|
DOCTYPES = ('BuiltinFunctionType', 'BuiltinMethodType', 'ClassType',
|
|
'FunctionType', 'GeneratorType', 'InstanceType',
|
|
'LambdaType', 'MethodType', 'ModuleType',
|
|
'UnboundMethodType', 'method-wrapper')
|
|
|
|
SIMPLETYPES = [getattr(types, t) for t in dir(types) \
|
|
if not t.startswith('_') and t not in DOCTYPES]
|
|
|
|
try:
|
|
COMMONTYPES.append(type(''.__repr__)) # Method-wrapper in version 2.2.x.
|
|
except AttributeError:
|
|
pass
|
|
|
|
|
|
class FillingTree(wx.wxTreeCtrl):
|
|
"""PyCrust FillingTree based on wxTreeCtrl."""
|
|
|
|
name = 'PyCrust Filling Tree'
|
|
revision = __revision__
|
|
|
|
def __init__(self, parent, id=-1, pos=wx.wxDefaultPosition,
|
|
size=wx.wxDefaultSize, style=wx.wxTR_DEFAULT_STYLE,
|
|
rootObject=None, rootLabel=None, rootIsNamespace=0,
|
|
static=False):
|
|
"""Create a PyCrust FillingTree instance."""
|
|
wx.wxTreeCtrl.__init__(self, parent, id, pos, size, style)
|
|
self.rootIsNamespace = rootIsNamespace
|
|
import __main__
|
|
if rootObject is None:
|
|
rootObject = __main__.__dict__
|
|
self.rootIsNamespace = 1
|
|
if rootObject is __main__.__dict__ and rootLabel is None:
|
|
rootLabel = 'locals()'
|
|
if not rootLabel:
|
|
rootLabel = 'Ingredients'
|
|
rootData = wx.wxTreeItemData(rootObject)
|
|
self.item = self.root = self.AddRoot(rootLabel, -1, -1, rootData)
|
|
self.SetItemHasChildren(self.root, self.hasChildren(rootObject))
|
|
wx.EVT_TREE_ITEM_EXPANDING(self, self.GetId(), self.OnItemExpanding)
|
|
wx.EVT_TREE_ITEM_COLLAPSED(self, self.GetId(), self.OnItemCollapsed)
|
|
wx.EVT_TREE_SEL_CHANGED(self, self.GetId(), self.OnSelChanged)
|
|
wx.EVT_TREE_ITEM_ACTIVATED(self, self.GetId(), self.OnItemActivated)
|
|
if not static:
|
|
dispatcher.connect(receiver=self.push, signal='Interpreter.push')
|
|
|
|
def push(self, command, more):
|
|
"""Receiver for Interpreter.push signal."""
|
|
self.display()
|
|
|
|
def OnItemExpanding(self, event):
|
|
"""Add children to the item."""
|
|
busy = wx.wxBusyCursor()
|
|
item = event.GetItem()
|
|
if self.IsExpanded(item):
|
|
return
|
|
self.addChildren(item)
|
|
# self.SelectItem(item)
|
|
|
|
def OnItemCollapsed(self, event):
|
|
"""Remove all children from the item."""
|
|
busy = wx.wxBusyCursor()
|
|
item = event.GetItem()
|
|
# self.CollapseAndReset(item)
|
|
# self.DeleteChildren(item)
|
|
# self.SelectItem(item)
|
|
|
|
def OnSelChanged(self, event):
|
|
"""Display information about the item."""
|
|
busy = wx.wxBusyCursor()
|
|
self.item = event.GetItem()
|
|
self.display()
|
|
|
|
def OnItemActivated(self, event):
|
|
"""Launch a DirFrame."""
|
|
item = event.GetItem()
|
|
text = self.getFullName(item)
|
|
obj = self.GetPyData(item)
|
|
frame = FillingFrame(parent=self, size=(600, 100), rootObject=obj,
|
|
rootLabel=text, rootIsNamespace=False)
|
|
frame.Show()
|
|
|
|
def hasChildren(self, obj):
|
|
"""Return true if object has children."""
|
|
if self.getChildren(obj):
|
|
return True
|
|
else:
|
|
return False
|
|
|
|
def getChildren(self, obj):
|
|
"""Return dictionary with attributes or contents of object."""
|
|
busy = wx.wxBusyCursor()
|
|
otype = type(obj)
|
|
if otype is types.DictType \
|
|
or str(otype)[17:23] == 'BTrees' and hasattr(obj, 'keys'):
|
|
return obj
|
|
d = {}
|
|
if otype is types.ListType or otype is types.TupleType:
|
|
for n in range(len(obj)):
|
|
key = '[' + str(n) + ']'
|
|
d[key] = obj[n]
|
|
if otype not in COMMONTYPES:
|
|
for key in introspect.getAttributeNames(obj):
|
|
# Believe it or not, some attributes can disappear,
|
|
# such as the exc_traceback attribute of the sys
|
|
# module. So this is nested in a try block.
|
|
try:
|
|
d[key] = getattr(obj, key)
|
|
except:
|
|
pass
|
|
return d
|
|
|
|
def addChildren(self, item):
|
|
self.DeleteChildren(item)
|
|
obj = self.GetPyData(item)
|
|
children = self.getChildren(obj)
|
|
if not children:
|
|
return
|
|
keys = children.keys()
|
|
keys.sort(lambda x, y: cmp(x.lower(), y.lower()))
|
|
for key in keys:
|
|
itemtext = str(key)
|
|
# Show string dictionary items with single quotes, except
|
|
# for the first level of items, if they represent a
|
|
# namespace.
|
|
if type(obj) is types.DictType \
|
|
and type(key) is types.StringType \
|
|
and (item != self.root \
|
|
or (item == self.root and not self.rootIsNamespace)):
|
|
itemtext = repr(key)
|
|
child = children[key]
|
|
data = wx.wxTreeItemData(child)
|
|
branch = self.AppendItem(parent=item, text=itemtext, data=data)
|
|
self.SetItemHasChildren(branch, self.hasChildren(child))
|
|
|
|
def display(self):
|
|
item = self.item
|
|
if self.IsExpanded(item):
|
|
self.addChildren(item)
|
|
self.setText('')
|
|
obj = self.GetPyData(item)
|
|
if obj is None: # Windows bug fix.
|
|
return
|
|
self.SetItemHasChildren(item, self.hasChildren(obj))
|
|
otype = type(obj)
|
|
text = ''
|
|
text += self.getFullName(item)
|
|
text += '\n\nType: ' + str(otype)
|
|
try:
|
|
value = str(obj)
|
|
except:
|
|
value = ''
|
|
if otype is types.StringType or otype is types.UnicodeType:
|
|
value = repr(obj)
|
|
text += '\n\nValue: ' + value
|
|
if otype not in SIMPLETYPES:
|
|
try:
|
|
text += '\n\nDocstring:\n\n"""' + \
|
|
inspect.getdoc(obj).strip() + '"""'
|
|
except:
|
|
pass
|
|
if otype is types.InstanceType:
|
|
try:
|
|
text += '\n\nClass Definition:\n\n' + \
|
|
inspect.getsource(obj.__class__)
|
|
except:
|
|
pass
|
|
else:
|
|
try:
|
|
text += '\n\nSource Code:\n\n' + \
|
|
inspect.getsource(obj)
|
|
except:
|
|
pass
|
|
self.setText(text)
|
|
|
|
def getFullName(self, item, partial=''):
|
|
"""Return a syntactically proper name for item."""
|
|
name = self.GetItemText(item)
|
|
parent = None
|
|
obj = None
|
|
if item != self.root:
|
|
parent = self.GetItemParent(item)
|
|
obj = self.GetPyData(parent)
|
|
# Apply dictionary syntax to dictionary items, except the root
|
|
# and first level children of a namepace.
|
|
if (type(obj) is types.DictType \
|
|
or str(type(obj))[17:23] == 'BTrees' \
|
|
and hasattr(obj, 'keys')) \
|
|
and ((item != self.root and parent != self.root) \
|
|
or (parent == self.root and not self.rootIsNamespace)):
|
|
name = '[' + name + ']'
|
|
# Apply dot syntax to multipart names.
|
|
if partial:
|
|
if partial[0] == '[':
|
|
name += partial
|
|
else:
|
|
name += '.' + partial
|
|
# Repeat for everything but the root item
|
|
# and first level children of a namespace.
|
|
if (item != self.root and parent != self.root) \
|
|
or (parent == self.root and not self.rootIsNamespace):
|
|
name = self.getFullName(parent, partial=name)
|
|
return name
|
|
|
|
def setText(self, text):
|
|
"""Display information about the current selection."""
|
|
|
|
# This method will likely be replaced by the enclosing app to
|
|
# do something more interesting, like write to a text control.
|
|
print text
|
|
|
|
def setStatusText(self, text):
|
|
"""Display status information."""
|
|
|
|
# This method will likely be replaced by the enclosing app to
|
|
# do something more interesting, like write to a status bar.
|
|
print text
|
|
|
|
|
|
if wx.wxPlatform == '__WXMSW__':
|
|
faces = { 'times' : 'Times New Roman',
|
|
'mono' : 'Courier New',
|
|
'helv' : 'Lucida Console',
|
|
'lucida' : 'Lucida Console',
|
|
'other' : 'Comic Sans MS',
|
|
'size' : 10,
|
|
'lnsize' : 9,
|
|
'backcol': '#FFFFFF',
|
|
}
|
|
else: # GTK
|
|
faces = { 'times' : 'Times',
|
|
'mono' : 'Courier',
|
|
'helv' : 'Helvetica',
|
|
'other' : 'new century schoolbook',
|
|
'size' : 12,
|
|
'lnsize' : 10,
|
|
'backcol': '#FFFFFF',
|
|
}
|
|
|
|
|
|
class FillingText(stc.wxStyledTextCtrl):
|
|
"""PyCrust FillingText based on wxStyledTextCtrl."""
|
|
|
|
name = 'PyCrust Filling Text'
|
|
revision = __revision__
|
|
|
|
def __init__(self, parent, id=-1, pos=wx.wxDefaultPosition,
|
|
size=wx.wxDefaultSize, style=wx.wxCLIP_CHILDREN,
|
|
static=False):
|
|
"""Create a PyCrust FillingText instance."""
|
|
stc.wxStyledTextCtrl.__init__(self, parent, id, pos, size, style)
|
|
# Configure various defaults and user preferences.
|
|
self.config()
|
|
dispatcher.connect(receiver=self.fontsizer, signal='FontIncrease')
|
|
dispatcher.connect(receiver=self.fontsizer, signal='FontDecrease')
|
|
dispatcher.connect(receiver=self.fontsizer, signal='FontDefault')
|
|
if not static:
|
|
dispatcher.connect(receiver=self.push, signal='Interpreter.push')
|
|
|
|
def push(self, command, more):
|
|
"""Receiver for Interpreter.push signal."""
|
|
self.Refresh()
|
|
|
|
def fontsizer(self, signal):
|
|
"""Receiver for Font* signals."""
|
|
size = self.GetZoom()
|
|
if signal == 'FontIncrease':
|
|
size += 1
|
|
elif signal == 'FontDecrease':
|
|
size -= 1
|
|
elif signal == 'FontDefault':
|
|
size = 0
|
|
self.SetZoom(size)
|
|
|
|
def config(self):
|
|
"""Configure shell based on user preferences."""
|
|
self.SetMarginWidth(1, 0)
|
|
|
|
self.SetLexer(stc.wxSTC_LEX_PYTHON)
|
|
self.SetKeyWords(0, ' '.join(keyword.kwlist))
|
|
|
|
self.setStyles(faces)
|
|
self.SetViewWhiteSpace(0)
|
|
self.SetTabWidth(4)
|
|
self.SetUseTabs(0)
|
|
self.SetReadOnly(1)
|
|
try:
|
|
self.SetWrapMode(1)
|
|
except AttributeError:
|
|
pass
|
|
|
|
def setStyles(self, faces):
|
|
"""Configure font size, typeface and color for lexer."""
|
|
|
|
# Default style
|
|
self.StyleSetSpec(stc.wxSTC_STYLE_DEFAULT,
|
|
"face:%(mono)s,size:%(size)d" % faces)
|
|
|
|
self.StyleClearAll()
|
|
|
|
# Built in styles
|
|
self.StyleSetSpec(stc.wxSTC_STYLE_LINENUMBER,
|
|
"back:#C0C0C0,face:%(mono)s,size:%(lnsize)d" % faces)
|
|
self.StyleSetSpec(stc.wxSTC_STYLE_CONTROLCHAR,
|
|
"face:%(mono)s" % faces)
|
|
self.StyleSetSpec(stc.wxSTC_STYLE_BRACELIGHT,
|
|
"fore:#0000FF,back:#FFFF88")
|
|
self.StyleSetSpec(stc.wxSTC_STYLE_BRACEBAD,
|
|
"fore:#FF0000,back:#FFFF88")
|
|
|
|
# Python styles
|
|
self.StyleSetSpec(stc.wxSTC_P_DEFAULT,
|
|
"face:%(mono)s" % faces)
|
|
self.StyleSetSpec(stc.wxSTC_P_COMMENTLINE,
|
|
"fore:#007F00,face:%(mono)s" % faces)
|
|
self.StyleSetSpec(stc.wxSTC_P_NUMBER,
|
|
"")
|
|
self.StyleSetSpec(stc.wxSTC_P_STRING,
|
|
"fore:#7F007F,face:%(mono)s" % faces)
|
|
self.StyleSetSpec(stc.wxSTC_P_CHARACTER,
|
|
"fore:#7F007F,face:%(mono)s" % faces)
|
|
self.StyleSetSpec(stc.wxSTC_P_WORD,
|
|
"fore:#00007F,bold")
|
|
self.StyleSetSpec(stc.wxSTC_P_TRIPLE,
|
|
"fore:#7F0000")
|
|
self.StyleSetSpec(stc.wxSTC_P_TRIPLEDOUBLE,
|
|
"fore:#000033,back:#FFFFE8")
|
|
self.StyleSetSpec(stc.wxSTC_P_CLASSNAME,
|
|
"fore:#0000FF,bold")
|
|
self.StyleSetSpec(stc.wxSTC_P_DEFNAME,
|
|
"fore:#007F7F,bold")
|
|
self.StyleSetSpec(stc.wxSTC_P_OPERATOR,
|
|
"")
|
|
self.StyleSetSpec(stc.wxSTC_P_IDENTIFIER,
|
|
"")
|
|
self.StyleSetSpec(stc.wxSTC_P_COMMENTBLOCK,
|
|
"fore:#7F7F7F")
|
|
self.StyleSetSpec(stc.wxSTC_P_STRINGEOL,
|
|
"fore:#000000,face:%(mono)s,back:#E0C0E0,eolfilled" % faces)
|
|
|
|
def SetText(self, *args, **kwds):
|
|
self.SetReadOnly(0)
|
|
stc.wxStyledTextCtrl.SetText(self, *args, **kwds)
|
|
self.SetReadOnly(1)
|
|
|
|
|
|
class Filling(wx.wxSplitterWindow):
|
|
"""PyCrust Filling based on wxSplitterWindow."""
|
|
|
|
name = 'PyCrust Filling'
|
|
revision = __revision__
|
|
|
|
def __init__(self, parent, id=-1, pos=wx.wxDefaultPosition,
|
|
size=wx.wxDefaultSize, style=wx.wxSP_3D,
|
|
name='Filling Window', rootObject=None,
|
|
rootLabel=None, rootIsNamespace=0, static=False):
|
|
"""Create a PyCrust Filling instance."""
|
|
wx.wxSplitterWindow.__init__(self, parent, id, pos, size, style, name)
|
|
self.tree = FillingTree(parent=self, rootObject=rootObject,
|
|
rootLabel=rootLabel,
|
|
rootIsNamespace=rootIsNamespace,
|
|
static=static)
|
|
self.text = FillingText(parent=self, static=static)
|
|
self.SplitVertically(self.tree, self.text, 200)
|
|
self.SetMinimumPaneSize(1)
|
|
# Override the filling so that descriptions go to FillingText.
|
|
self.tree.setText = self.text.SetText
|
|
# Display the root item.
|
|
## self.tree.SelectItem(self.tree.root)
|
|
self.tree.display()
|
|
|
|
|
|
class FillingFrame(wx.wxFrame):
|
|
"""Frame containing the PyCrust filling, or namespace tree component."""
|
|
|
|
name = 'PyCrust Filling Frame'
|
|
revision = __revision__
|
|
|
|
def __init__(self, parent=None, id=-1, title='PyFilling',
|
|
pos=wx.wxDefaultPosition, size=wx.wxDefaultSize,
|
|
style=wx.wxDEFAULT_FRAME_STYLE, rootObject=None,
|
|
rootLabel=None, rootIsNamespace=0, static=False):
|
|
"""Create a PyCrust FillingFrame instance."""
|
|
wx.wxFrame.__init__(self, parent, id, title, pos, size, style)
|
|
intro = 'PyFilling - The Tastiest Namespace Inspector'
|
|
self.CreateStatusBar()
|
|
self.SetStatusText(intro)
|
|
import images
|
|
self.SetIcon(images.getPyCrustIcon())
|
|
self.filling = Filling(parent=self, rootObject=rootObject,
|
|
rootLabel=rootLabel,
|
|
rootIsNamespace=rootIsNamespace,
|
|
static=static)
|
|
# Override so that status messages go to the status bar.
|
|
self.filling.tree.setStatusText = self.SetStatusText
|
|
|
|
|
|
class App(wx.wxApp):
|
|
"""PyFilling standalone application."""
|
|
|
|
def OnInit(self):
|
|
wx.wxInitAllImageHandlers()
|
|
self.fillingFrame = FillingFrame()
|
|
self.fillingFrame.Show(True)
|
|
self.SetTopWindow(self.fillingFrame)
|
|
return True
|
|
|
|
|
|
|
|
|