1 """Crust combines the shell and filling into one control."""
3 __author__
= "Patrick K. O'Brien <pobrien@orbtech.com>"
5 __revision__
= "$Revision$"[11:-2]
16 from filling
import Filling
18 from shell
import Shell
19 from version
import VERSION
22 class Crust(wx
.SplitterWindow
):
23 """Crust based on SplitterWindow."""
26 revision
= __revision__
29 def __init__(self
, parent
, id=-1, pos
=wx
.DefaultPosition
,
30 size
=wx
.DefaultSize
, style
=wx
.SP_3D|wx
.SP_LIVE_UPDATE
,
31 name
='Crust Window', rootObject
=None, rootLabel
=None,
32 rootIsNamespace
=True, intro
='', locals=None,
34 startupScript
=None, execStartupScript
=True,
36 """Create Crust instance."""
37 wx
.SplitterWindow
.__init
__(self
, parent
, id, pos
, size
, style
, name
)
38 self
.shell
= Shell(parent
=self
, introText
=intro
,
39 locals=locals, InterpClass
=InterpClass
,
40 startupScript
=startupScript
,
41 execStartupScript
=execStartupScript
,
43 self
.editor
= self
.shell
44 if rootObject
is None:
45 rootObject
= self
.shell
.interp
.locals
46 self
.notebook
= wx
.Notebook(parent
=self
, id=-1)
47 self
.shell
.interp
.locals['notebook'] = self
.notebook
48 self
.filling
= Filling(parent
=self
.notebook
,
49 rootObject
=rootObject
,
51 rootIsNamespace
=rootIsNamespace
)
52 # Add 'filling' to the interpreter's locals.
53 self
.shell
.interp
.locals['filling'] = self
.filling
54 self
.notebook
.AddPage(page
=self
.filling
, text
='Namespace', select
=True)
56 self
.display
= Display(parent
=self
.notebook
)
57 self
.notebook
.AddPage(page
=self
.display
, text
='Display')
58 # Add 'pp' (pretty print) to the interpreter's locals.
59 self
.shell
.interp
.locals['pp'] = self
.display
.setItem
60 self
.display
.nbTab
= self
.notebook
.GetPageCount()-1
62 self
.calltip
= Calltip(parent
=self
.notebook
)
63 self
.notebook
.AddPage(page
=self
.calltip
, text
='Calltip')
65 self
.sessionlisting
= SessionListing(parent
=self
.notebook
)
66 self
.notebook
.AddPage(page
=self
.sessionlisting
, text
='History')
68 self
.dispatcherlisting
= DispatcherListing(parent
=self
.notebook
)
69 self
.notebook
.AddPage(page
=self
.dispatcherlisting
, text
='Dispatcher')
71 self
.SplitHorizontally(self
.shell
, self
.notebook
, -self
.sashoffset
)
72 self
.SetMinimumPaneSize(100)
74 self
.Bind(wx
.EVT_SIZE
, self
.SplitterOnSize
)
75 self
.Bind(wx
.EVT_SPLITTER_SASH_POS_CHANGED
, self
.OnChanged
)
78 def OnChanged(self
, event
):
79 """update sash offset from the bottom of the window"""
80 self
.sashoffset
= self
.GetSize().height
- event
.GetSashPosition()
84 # Make the splitter expand the top window when resized
85 def SplitterOnSize(self
, event
):
86 splitter
= event
.GetEventObject()
87 sz
= splitter
.GetSize()
88 splitter
.SetSashPosition(sz
.height
- self
.sashoffset
, True)
92 def LoadSettings(self
, config
):
93 self
.shell
.LoadSettings(config
)
94 self
.filling
.LoadSettings(config
)
96 pos
= config
.ReadInt('Sash/CrustPos', 400)
97 wx
.CallAfter(self
.SetSashPosition
, pos
)
98 def _updateSashPosValue():
100 self
.sashoffset
= sz
.height
- self
.GetSashPosition()
101 wx
.CallAfter(_updateSashPosValue
)
102 zoom
= config
.ReadInt('View/Zoom/Display', -99)
104 self
.display
.SetZoom(zoom
)
107 def SaveSettings(self
, config
):
108 self
.shell
.SaveSettings(config
)
109 self
.filling
.SaveSettings(config
)
111 config
.WriteInt('Sash/CrustPos', self
.GetSashPosition())
112 config
.WriteInt('View/Zoom/Display', self
.display
.GetZoom())
118 class Display(editwindow
.EditWindow
):
119 """STC used to display an object using Pretty Print."""
121 def __init__(self
, parent
, id=-1, pos
=wx
.DefaultPosition
,
123 style
=wx
.CLIP_CHILDREN | wx
.SUNKEN_BORDER
,
125 """Create Display instance."""
126 editwindow
.EditWindow
.__init
__(self
, parent
, id, pos
, size
, style
)
127 # Configure various defaults and user preferences.
128 self
.SetReadOnly(True)
129 self
.SetWrapMode(False)
131 dispatcher
.connect(receiver
=self
.push
, signal
='Interpreter.push')
133 def push(self
, command
, more
):
134 """Receiver for Interpreter.push signal."""
138 if not hasattr(self
, "item"):
140 self
.SetReadOnly(False)
141 text
= pprint
.pformat(self
.item
)
143 self
.SetReadOnly(True)
145 def setItem(self
, item
):
146 """Set item to pretty print in the notebook Display tab."""
149 if self
.GetParent().GetSelection() != self
.nbTab
:
150 focus
= wx
.Window
.FindFocus()
151 self
.GetParent().SetSelection(self
.nbTab
)
152 wx
.CallAfter(focus
.SetFocus
)
155 # TODO: Switch this to a editwindow.EditWindow
156 class Calltip(wx
.TextCtrl
):
157 """Text control containing the most recent shell calltip."""
159 def __init__(self
, parent
=None, id=-1):
160 style
= (wx
.TE_MULTILINE | wx
.TE_READONLY | wx
.TE_RICH2
)
161 wx
.TextCtrl
.__init
__(self
, parent
, id, style
=style
)
162 self
.SetBackgroundColour(wx
.Colour(255, 255, 208))
163 dispatcher
.connect(receiver
=self
.display
, signal
='Shell.calltip')
166 font
= wx
.Font(df
.GetPointSize(), wx
.TELETYPE
, wx
.NORMAL
, wx
.NORMAL
)
169 def display(self
, calltip
):
170 """Receiver for Shell.calltip signal."""
171 ## self.SetValue(calltip) # Caused refresh problem on Windows.
173 self
.AppendText(calltip
)
176 # TODO: Switch this to a editwindow.EditWindow
177 class SessionListing(wx
.TextCtrl
):
178 """Text control containing all commands for session."""
180 def __init__(self
, parent
=None, id=-1):
181 style
= (wx
.TE_MULTILINE | wx
.TE_READONLY |
182 wx
.TE_RICH2 | wx
.TE_DONTWRAP
)
183 wx
.TextCtrl
.__init
__(self
, parent
, id, style
=style
)
184 dispatcher
.connect(receiver
=self
.addHistory
, signal
="Shell.addHistory")
185 dispatcher
.connect(receiver
=self
.clearHistory
, signal
="Shell.clearHistory")
186 dispatcher
.connect(receiver
=self
.loadHistory
, signal
="Shell.loadHistory")
189 font
= wx
.Font(df
.GetPointSize(), wx
.TELETYPE
, wx
.NORMAL
, wx
.NORMAL
)
192 def loadHistory(self
, history
):
193 # preload the existing history, if any
196 self
.SetValue('\n'.join(hist
) + '\n')
197 self
.SetInsertionPointEnd()
199 def addHistory(self
, command
):
201 self
.SetInsertionPointEnd()
202 self
.AppendText(command
+ '\n')
204 def clearHistory(self
):
208 class DispatcherListing(wx
.TextCtrl
):
209 """Text control containing all dispatches for session."""
211 def __init__(self
, parent
=None, id=-1):
212 style
= (wx
.TE_MULTILINE | wx
.TE_READONLY |
213 wx
.TE_RICH2 | wx
.TE_DONTWRAP
)
214 wx
.TextCtrl
.__init
__(self
, parent
, id, style
=style
)
215 dispatcher
.connect(receiver
=self
.spy
)
218 font
= wx
.Font(df
.GetPointSize(), wx
.TELETYPE
, wx
.NORMAL
, wx
.NORMAL
)
221 def spy(self
, signal
, sender
):
222 """Receiver for Any signal from Any sender."""
223 text
= '%r from %s' % (signal
, sender
)
224 self
.SetInsertionPointEnd()
225 start
, end
= self
.GetSelection()
227 self
.SetSelection(0, 0)
228 self
.AppendText(text
+ '\n')
232 class CrustFrame(frame
.Frame
, frame
.ShellFrameMixin
):
233 """Frame containing all the PyCrust components."""
236 revision
= __revision__
239 def __init__(self
, parent
=None, id=-1, title
='PyCrust',
240 pos
=wx
.DefaultPosition
, size
=wx
.DefaultSize
,
241 style
=wx
.DEFAULT_FRAME_STYLE
,
242 rootObject
=None, rootLabel
=None, rootIsNamespace
=True,
243 locals=None, InterpClass
=None,
244 config
=None, dataDir
=None,
246 """Create CrustFrame instance."""
247 frame
.Frame
.__init
__(self
, parent
, id, title
, pos
, size
, style
)
248 frame
.ShellFrameMixin
.__init
__(self
, config
, dataDir
)
250 if size
== wx
.DefaultSize
:
251 self
.SetSize((800, 600))
253 intro
= 'PyCrust %s - The Flakiest Python Shell' % VERSION
254 self
.SetStatusText(intro
.replace('\n', ', '))
255 self
.crust
= Crust(parent
=self
, intro
=intro
,
256 rootObject
=rootObject
,
258 rootIsNamespace
=rootIsNamespace
,
260 InterpClass
=InterpClass
,
261 startupScript
=self
.startupScript
,
262 execStartupScript
=self
.execStartupScript
,
264 self
.shell
= self
.crust
.shell
266 # Override the filling so that status messages go to the status bar.
267 self
.crust
.filling
.tree
.setStatusText
= self
.SetStatusText
269 # Override the shell so that status messages go to the status bar.
270 self
.shell
.setStatusText
= self
.SetStatusText
272 self
.shell
.SetFocus()
276 def OnClose(self
, event
):
277 """Event handler for closing."""
279 self
.crust
.shell
.destroy()
283 def OnAbout(self
, event
):
284 """Display an About window."""
285 title
= 'About PyCrust'
286 text
= 'PyCrust %s\n\n' % VERSION
+ \
287 'Yet another Python shell, only flakier.\n\n' + \
288 'Half-baked by Patrick K. O\'Brien,\n' + \
289 'the other half is still in the oven.\n\n' + \
290 'Shell Revision: %s\n' % self
.shell
.revision
+ \
291 'Interpreter Revision: %s\n\n' % self
.shell
.interp
.revision
+ \
292 'Platform: %s\n' % sys
.platform
+ \
293 'Python Version: %s\n' % sys
.version
.split()[0] + \
294 'wxPython Version: %s\n' % wx
.VERSION_STRING
+ \
295 ('\t(%s)\n' % ", ".join(wx
.PlatformInfo
[1:]))
296 dialog
= wx
.MessageDialog(self
, text
, title
,
297 wx
.OK | wx
.ICON_INFORMATION
)
302 def OnHelp(self
, event
):
303 """Show a help dialog."""
304 frame
.ShellFrameMixin
.OnHelp(self
, event
)
307 def LoadSettings(self
):
308 if self
.config
is not None:
309 frame
.ShellFrameMixin
.LoadSettings(self
)
310 frame
.Frame
.LoadSettings(self
, self
.config
)
311 self
.crust
.LoadSettings(self
.config
)
314 def SaveSettings(self
, force
=False):
315 if self
.config
is not None:
316 frame
.ShellFrameMixin
.SaveSettings(self
)
317 if self
.autoSaveSettings
or force
:
318 frame
.Frame
.SaveSettings(self
, self
.config
)
319 self
.crust
.SaveSettings(self
.config
)
322 def DoSaveSettings(self
):
323 if self
.config
is not None:
324 self
.SaveSettings(force
=True)