]>
Commit | Line | Data |
---|---|---|
1 | #---------------------------------------------------------------------------- | |
2 | # Name: DebuggerService.py | |
3 | # Purpose: Debugger Service for Python. | |
4 | # | |
5 | # Author: Matt Fryer | |
6 | # | |
7 | # Created: 12/9/04 | |
8 | # CVS-ID: $Id$ | |
9 | # Copyright: (c) 2004-2005 ActiveGrid, Inc. | |
10 | # License: wxWindows License | |
11 | #---------------------------------------------------------------------------- | |
12 | ||
13 | import wx | |
14 | import wx.lib.intctrl | |
15 | import wx.lib.docview | |
16 | import wx.lib.dialogs | |
17 | import wx.gizmos | |
18 | import wx._core | |
19 | import wx.lib.pydocview | |
20 | import Service | |
21 | import STCTextEditor | |
22 | import CodeEditor | |
23 | import PythonEditor | |
24 | from IDE import ACTIVEGRID_BASE_IDE | |
25 | if not ACTIVEGRID_BASE_IDE: | |
26 | import ProcessModelEditor | |
27 | import wx.lib.scrolledpanel as scrolled | |
28 | import sys | |
29 | import time | |
30 | import SimpleXMLRPCServer | |
31 | import xmlrpclib | |
32 | import os | |
33 | import threading | |
34 | import process | |
35 | import Queue | |
36 | import SocketServer | |
37 | import ProjectEditor | |
38 | import types | |
39 | from xml.dom.minidom import parse, parseString | |
40 | import bz2 | |
41 | import pickle | |
42 | import DebuggerHarness | |
43 | import traceback | |
44 | import StringIO | |
45 | if wx.Platform == '__WXMSW__': | |
46 | import win32api | |
47 | _WINDOWS = True | |
48 | else: | |
49 | _WINDOWS = False | |
50 | _ = wx.GetTranslation | |
51 | ||
52 | _VERBOSE = False | |
53 | _WATCHES_ON = False | |
54 | ||
55 | # Class to read from stdout or stderr and write the result to a text control. | |
56 | # Args: file=file-like object | |
57 | # callback_function= function that takes a single argument, the line of text | |
58 | # read. | |
59 | class OutputReaderThread(threading.Thread): | |
60 | def __init__(self, file, callback_function, callbackOnExit=None, accumulate=True): | |
61 | threading.Thread.__init__(self) | |
62 | self._file = file | |
63 | self._callback_function = callback_function | |
64 | self._keepGoing = True | |
65 | self._lineCount = 0 | |
66 | self._accumulate = accumulate | |
67 | self._callbackOnExit = callbackOnExit | |
68 | ||
69 | def run(self): | |
70 | file = self._file | |
71 | start = time.time() | |
72 | output = "" | |
73 | while self._keepGoing: | |
74 | try: | |
75 | # This could block--how to handle that? | |
76 | text = file.readline() | |
77 | if text == '' or text == None: | |
78 | self._keepGoing = False | |
79 | elif not self._accumulate: | |
80 | self._callback_function(text) | |
81 | else: | |
82 | # Should use a buffer? StringIO? | |
83 | output += text | |
84 | # Seems as though the read blocks if we got an error, so, to be | |
85 | # sure that at least some of the exception gets printed, always | |
86 | # send the first hundred lines back as they come in. | |
87 | if self._lineCount < 100: | |
88 | self._callback_function(output) | |
89 | self._lineCount += 1 | |
90 | output = "" | |
91 | elif time.time() - start > 0.25: | |
92 | try: | |
93 | self._callback_function(output) | |
94 | except wx._core.PyDeadObjectError: | |
95 | # GUI was killed while we were blocked. | |
96 | self._keepGoing = False | |
97 | start = time.time() | |
98 | output = "" | |
99 | except: | |
100 | tp, val, tb = sys.exc_info() | |
101 | print "Exception in OutputReaderThread.run():", tp, val | |
102 | self._keepGoing = False | |
103 | if self._callbackOnExit: | |
104 | try: | |
105 | self._callbackOnExit() | |
106 | except wx._core.PyDeadObjectError: | |
107 | pass | |
108 | if _VERBOSE: print "Exiting OutputReaderThread" | |
109 | ||
110 | def AskToStop(self): | |
111 | self._keepGoing = False | |
112 | ||
113 | import wx.lib.newevent | |
114 | (UpdateTextEvent, EVT_UPDATE_STDTEXT) = wx.lib.newevent.NewEvent() | |
115 | (UpdateErrorEvent, EVT_UPDATE_ERRTEXT) = wx.lib.newevent.NewEvent() | |
116 | ||
117 | class Executor: | |
118 | ||
119 | def GetPythonExecutablePath(): | |
120 | config = wx.ConfigBase_Get() | |
121 | path = config.Read("ActiveGridPythonLocation") | |
122 | if path: | |
123 | return path | |
124 | wx.MessageBox(_("To proceed I need to know the location of the python.exe you would like to use.\nTo set this, go to Tools-->Options and use the 'Python' tab to enter a value.\n"), _("Python Executable Location Unknown")) | |
125 | return None | |
126 | GetPythonExecutablePath = staticmethod(GetPythonExecutablePath) | |
127 | ||
128 | def __init__(self, fileName, wxComponent, arg1=None, arg2=None, arg3=None, arg4=None, arg5=None, arg6=None, arg7=None, arg8=None, arg9=None, callbackOnExit=None): | |
129 | self._fileName = fileName | |
130 | self._stdOutCallback = self.OutCall | |
131 | self._stdErrCallback = self.ErrCall | |
132 | self._callbackOnExit = callbackOnExit | |
133 | self._wxComponent = wxComponent | |
134 | path = Executor.GetPythonExecutablePath() | |
135 | self._cmd = '"' + path + '" -u \"' + fileName + '\"' | |
136 | #Better way to do this? Quotes needed for windows file paths. | |
137 | if(arg1 != None): | |
138 | self._cmd += ' \"' + arg1 + '\"' | |
139 | if(arg2 != None): | |
140 | self._cmd += ' \"' + arg2 + '\"' | |
141 | if(arg3 != None): | |
142 | self._cmd += ' \"' + arg3 + '\"' | |
143 | if(arg4 != None): | |
144 | self._cmd += ' \"' + arg4 + '\"' | |
145 | if(arg5 != None): | |
146 | self._cmd += ' \"' + arg5 + '\"' | |
147 | if(arg6 != None): | |
148 | self._cmd += ' \"' + arg6 + '\"' | |
149 | if(arg7 != None): | |
150 | self._cmd += ' \"' + arg7 + '\"' | |
151 | if(arg8 != None): | |
152 | self._cmd += ' \"' + arg8 + '\"' | |
153 | if(arg9 != None): | |
154 | self._cmd += ' \"' + arg9 + '\"' | |
155 | ||
156 | self._stdOutReader = None | |
157 | self._stdErrReader = None | |
158 | self._process = None | |
159 | ||
160 | def OutCall(self, text): | |
161 | evt = UpdateTextEvent(value = text) | |
162 | wx.PostEvent(self._wxComponent, evt) | |
163 | ||
164 | def ErrCall(self, text): | |
165 | evt = UpdateErrorEvent(value = text) | |
166 | wx.PostEvent(self._wxComponent, evt) | |
167 | ||
168 | def Execute(self, arguments, startIn=None, environment=None): | |
169 | if not startIn: | |
170 | startIn = str(os.getcwd()) | |
171 | startIn = os.path.abspath(startIn) | |
172 | command = self._cmd + ' ' + arguments | |
173 | #stdinput = process.IOBuffer() | |
174 | #self._process = process.ProcessProxy(command, mode='b', cwd=startIn, stdin=stdinput) | |
175 | self._process = process.ProcessOpen(command, mode='b', cwd=startIn, env=environment) | |
176 | # Kick off threads to read stdout and stderr and write them | |
177 | # to our text control. | |
178 | self._stdOutReader = OutputReaderThread(self._process.stdout, self._stdOutCallback, callbackOnExit=self._callbackOnExit) | |
179 | self._stdOutReader.start() | |
180 | self._stdErrReader = OutputReaderThread(self._process.stderr, self._stdErrCallback, accumulate=False) | |
181 | self._stdErrReader.start() | |
182 | ||
183 | ||
184 | def DoStopExecution(self): | |
185 | if(self._process != None): | |
186 | self._process.kill() | |
187 | self._process.close() | |
188 | self._process = None | |
189 | if(self._stdOutReader != None): | |
190 | self._stdOutReader.AskToStop() | |
191 | if(self._stdErrReader != None): | |
192 | self._stdErrReader.AskToStop() | |
193 | ||
194 | class RunCommandUI(wx.Panel): | |
195 | ||
196 | def __init__(self, parent, id, fileName): | |
197 | wx.Panel.__init__(self, parent, id) | |
198 | self._noteBook = parent | |
199 | ||
200 | self.KILL_PROCESS_ID = wx.NewId() | |
201 | self.CLOSE_TAB_ID = wx.NewId() | |
202 | ||
203 | self.Bind(wx.EVT_END_PROCESS, self.OnProcessEnded) | |
204 | ||
205 | # GUI Initialization follows | |
206 | sizer = wx.BoxSizer(wx.HORIZONTAL) | |
207 | self._tb = tb = wx.ToolBar(self, -1, wx.DefaultPosition, (30,1000), wx.TB_VERTICAL| wx.TB_FLAT, "Runner" ) | |
208 | tb.SetToolBitmapSize((16,16)) | |
209 | sizer.Add(tb, 0, wx.EXPAND |wx.ALIGN_LEFT|wx.ALL, 1) | |
210 | ||
211 | close_bmp = getCloseBitmap() | |
212 | tb.AddSimpleTool( self.CLOSE_TAB_ID, close_bmp, _('Close Window')) | |
213 | wx.EVT_TOOL(self, self.CLOSE_TAB_ID, self.OnToolClicked) | |
214 | ||
215 | stop_bmp = getStopBitmap() | |
216 | tb.AddSimpleTool(self.KILL_PROCESS_ID, stop_bmp, _("Stop the Run.")) | |
217 | wx.EVT_TOOL(self, self.KILL_PROCESS_ID, self.OnToolClicked) | |
218 | ||
219 | tb.Realize() | |
220 | self._textCtrl = STCTextEditor.TextCtrl(self, wx.NewId()) #id) | |
221 | sizer.Add(self._textCtrl, 1, wx.ALIGN_LEFT|wx.ALL|wx.EXPAND, 1) | |
222 | self._textCtrl.SetViewLineNumbers(False) | |
223 | self._textCtrl.SetReadOnly(True) | |
224 | if wx.Platform == '__WXMSW__': | |
225 | font = "Courier New" | |
226 | else: | |
227 | font = "Courier" | |
228 | self._textCtrl.SetFont(wx.Font(9, wx.DEFAULT, wx.NORMAL, wx.NORMAL, faceName = font)) | |
229 | self._textCtrl.SetFontColor(wx.BLACK) | |
230 | self._textCtrl.StyleClearAll() | |
231 | ||
232 | #Disabling for now...may interfere with file open. wx.stc.EVT_STC_DOUBLECLICK(self._textCtrl, self._textCtrl.GetId(), self.OnDoubleClick) | |
233 | ||
234 | self.SetSizer(sizer) | |
235 | sizer.Fit(self) | |
236 | ||
237 | # Executor initialization | |
238 | self._executor = Executor(fileName, self, callbackOnExit=self.ExecutorFinished) | |
239 | self.Bind(EVT_UPDATE_STDTEXT, self.AppendText) | |
240 | self.Bind(EVT_UPDATE_ERRTEXT, self.AppendErrorText) | |
241 | ||
242 | def __del__(self): | |
243 | self._executor.DoStopExecution() | |
244 | ||
245 | def Execute(self, initialArgs, startIn, environment): | |
246 | self._executor.Execute(initialArgs, startIn, environment) | |
247 | ||
248 | def ExecutorFinished(self): | |
249 | self._tb.EnableTool(self.KILL_PROCESS_ID, False) | |
250 | nb = self.GetParent() | |
251 | for i in range(0,nb.GetPageCount()): | |
252 | if self == nb.GetPage(i): | |
253 | text = nb.GetPageText(i) | |
254 | newText = text.replace("Running", "Finished") | |
255 | nb.SetPageText(i, newText) | |
256 | break | |
257 | ||
258 | def StopExecution(self): | |
259 | self.Unbind(EVT_UPDATE_STDTEXT) | |
260 | self.Unbind(EVT_UPDATE_ERRTEXT) | |
261 | self._executor.DoStopExecution() | |
262 | ||
263 | def AppendText(self, event): | |
264 | self._textCtrl.SetReadOnly(False) | |
265 | self._textCtrl.AddText(event.value) | |
266 | self._textCtrl.ScrollToLine(self._textCtrl.GetLineCount()) | |
267 | self._textCtrl.SetReadOnly(True) | |
268 | ||
269 | def AppendErrorText(self, event): | |
270 | self._textCtrl.SetReadOnly(False) | |
271 | self._textCtrl.SetFontColor(wx.RED) | |
272 | self._textCtrl.StyleClearAll() | |
273 | self._textCtrl.AddText(event.value) | |
274 | self._textCtrl.ScrollToLine(self._textCtrl.GetLineCount()) | |
275 | self._textCtrl.SetFontColor(wx.BLACK) | |
276 | self._textCtrl.StyleClearAll() | |
277 | self._textCtrl.SetReadOnly(True) | |
278 | ||
279 | #------------------------------------------------------------------------------ | |
280 | # Event handling | |
281 | #----------------------------------------------------------------------------- | |
282 | ||
283 | def OnToolClicked(self, event): | |
284 | id = event.GetId() | |
285 | ||
286 | if id == self.KILL_PROCESS_ID: | |
287 | self._executor.DoStopExecution() | |
288 | ||
289 | elif id == self.CLOSE_TAB_ID: | |
290 | self._executor.DoStopExecution() | |
291 | index = self._noteBook.GetSelection() | |
292 | self._noteBook.GetPage(index).Show(False) | |
293 | self._noteBook.RemovePage(index) | |
294 | ||
295 | def OnDoubleClick(self, event): | |
296 | # Looking for a stack trace line. | |
297 | lineText, pos = self._textCtrl.GetCurLine() | |
298 | fileBegin = lineText.find("File \"") | |
299 | fileEnd = lineText.find("\", line ") | |
300 | lineEnd = lineText.find(", in ") | |
301 | if lineText == "\n" or fileBegin == -1 or fileEnd == -1 or lineEnd == -1: | |
302 | # Check the line before the one that was clicked on | |
303 | lineNumber = self._textCtrl.GetCurrentLine() | |
304 | if(lineNumber == 0): | |
305 | return | |
306 | lineText = self._textCtrl.GetLine(lineNumber - 1) | |
307 | fileBegin = lineText.find("File \"") | |
308 | fileEnd = lineText.find("\", line ") | |
309 | lineEnd = lineText.find(", in ") | |
310 | if lineText == "\n" or fileBegin == -1 or fileEnd == -1 or lineEnd == -1: | |
311 | return | |
312 | ||
313 | filename = lineText[fileBegin + 6:fileEnd] | |
314 | lineNum = int(lineText[fileEnd + 8:lineEnd]) | |
315 | ||
316 | foundView = None | |
317 | openDocs = wx.GetApp().GetDocumentManager().GetDocuments() | |
318 | for openDoc in openDocs: | |
319 | if openDoc.GetFilename() == filename: | |
320 | foundView = openDoc.GetFirstView() | |
321 | break | |
322 | ||
323 | if not foundView: | |
324 | doc = wx.GetApp().GetDocumentManager().CreateDocument(filename, wx.lib.docview.DOC_SILENT) | |
325 | foundView = doc.GetFirstView() | |
326 | ||
327 | if foundView: | |
328 | foundView.GetFrame().SetFocus() | |
329 | foundView.Activate() | |
330 | foundView.GotoLine(lineNum) | |
331 | startPos = foundView.PositionFromLine(lineNum) | |
332 | ||
333 | # FACTOR THIS INTO DocManager | |
334 | openDocs = wx.GetApp().GetDocumentManager().GetDocuments() | |
335 | for openDoc in openDocs: | |
336 | if(isinstance(openDoc.GetFirstView(), CodeEditor.CodeView)): | |
337 | openDoc.GetFirstView().GetCtrl().ClearCurrentLineMarkers() | |
338 | ||
339 | foundView.GetCtrl().MarkerAdd(lineNum -1, CodeEditor.CodeCtrl.CURRENT_LINE_MARKER_NUM) | |
340 | ||
341 | def OnProcessEnded(self, evt): | |
342 | self._executor.DoStopExecution() | |
343 | ||
344 | DEFAULT_PORT = 32032 | |
345 | DEFAULT_HOST = 'localhost' | |
346 | PORT_COUNT = 21 | |
347 | ||
348 | class DebugCommandUI(wx.Panel): | |
349 | debuggerPortList = None | |
350 | debuggers = [] | |
351 | ||
352 | def NotifyDebuggersOfBreakpointChange(): | |
353 | for debugger in DebugCommandUI.debuggers: | |
354 | debugger.BreakPointChange() | |
355 | ||
356 | NotifyDebuggersOfBreakpointChange = staticmethod(NotifyDebuggersOfBreakpointChange) | |
357 | ||
358 | def DebuggerRunning(): | |
359 | for debugger in DebugCommandUI.debuggers: | |
360 | if debugger._executor: | |
361 | return True | |
362 | return False | |
363 | DebuggerRunning = staticmethod(DebuggerRunning) | |
364 | ||
365 | def ShutdownAllDebuggers(): | |
366 | for debugger in DebugCommandUI.debuggers: | |
367 | debugger.StopExecution() | |
368 | ||
369 | ShutdownAllDebuggers = staticmethod(ShutdownAllDebuggers) | |
370 | ||
371 | def GetAvailablePort(): | |
372 | for index in range( 0, len(DebugCommandUI.debuggerPortList)): | |
373 | port = DebugCommandUI.debuggerPortList[index] | |
374 | if DebugCommandUI.PortAvailable(port): | |
375 | DebugCommandUI.debuggerPortList.pop(index) | |
376 | return port | |
377 | wx.MessageBox(_("Out of ports for debugging! Please restart the application builder.\nIf that does not work, check for and remove running instances of python."), _("Out of Ports")) | |
378 | assert False, "Out of ports for debugger." | |
379 | ||
380 | GetAvailablePort = staticmethod(GetAvailablePort) | |
381 | ||
382 | def ReturnPortToPool(port): | |
383 | config = wx.ConfigBase_Get() | |
384 | startingPort = config.ReadInt("DebuggerStartingPort", DEFAULT_PORT) | |
385 | if port in range(startingPort, startingPort + PORT_COUNT): | |
386 | DebugCommandUI.debuggerPortList.append(port) | |
387 | ||
388 | ReturnPortToPool = staticmethod(ReturnPortToPool) | |
389 | ||
390 | def PortAvailable(port): | |
391 | config = wx.ConfigBase_Get() | |
392 | hostname = config.Read("DebuggerHostName", DEFAULT_HOST) | |
393 | try: | |
394 | server = AGXMLRPCServer((hostname, port)) | |
395 | server.server_close() | |
396 | if _VERBOSE: print "Port ", str(port), " available." | |
397 | return True | |
398 | except: | |
399 | tp,val,tb = sys.exc_info() | |
400 | if _VERBOSE: traceback.print_exception(tp, val, tb) | |
401 | if _VERBOSE: print "Port ", str(port), " unavailable." | |
402 | return False | |
403 | ||
404 | PortAvailable = staticmethod(PortAvailable) | |
405 | ||
406 | def NewPortRange(): | |
407 | config = wx.ConfigBase_Get() | |
408 | startingPort = config.ReadInt("DebuggerStartingPort", DEFAULT_PORT) | |
409 | DebugCommandUI.debuggerPortList = range(startingPort, startingPort + PORT_COUNT) | |
410 | NewPortRange = staticmethod(NewPortRange) | |
411 | ||
412 | def __init__(self, parent, id, command, service): | |
413 | # Check for ports before creating the panel. | |
414 | if not DebugCommandUI.debuggerPortList: | |
415 | DebugCommandUI.NewPortRange() | |
416 | self._debuggerPort = str(DebugCommandUI.GetAvailablePort()) | |
417 | self._guiPort = str(DebugCommandUI.GetAvailablePort()) | |
418 | self._debuggerBreakPort = str(DebugCommandUI.GetAvailablePort()) | |
419 | ||
420 | wx.Panel.__init__(self, parent, id) | |
421 | ||
422 | self._parentNoteBook = parent | |
423 | self._command = command | |
424 | self._textCtrl = None | |
425 | self._service = service | |
426 | self._executor = None | |
427 | self.STEP_ID = wx.NewId() | |
428 | self.CONTINUE_ID = wx.NewId() | |
429 | self.STEP_OUT_ID = wx.NewId() | |
430 | self.NEXT_ID = wx.NewId() | |
431 | self.KILL_PROCESS_ID = wx.NewId() | |
432 | self.CLOSE_WINDOW_ID = wx.NewId() | |
433 | self.BREAK_INTO_DEBUGGER_ID = wx.NewId() | |
434 | self.CLEAR_ID = wx.NewId() | |
435 | self.ADD_WATCH_ID = wx.NewId() | |
436 | sizer = wx.BoxSizer(wx.VERTICAL) | |
437 | self._tb = tb = wx.ToolBar(self, -1, wx.DefaultPosition, (1000,30), wx.TB_HORIZONTAL| wx.NO_BORDER| wx.TB_FLAT| wx.TB_TEXT, "Debugger" ) | |
438 | sizer.Add(tb, 0, wx.EXPAND |wx.ALIGN_LEFT|wx.ALL, 1) | |
439 | tb.SetToolBitmapSize((16,16)) | |
440 | ||
441 | close_bmp = getCloseBitmap() | |
442 | tb.AddSimpleTool( self.CLOSE_WINDOW_ID, close_bmp, _('Close Window')) | |
443 | wx.EVT_TOOL(self, self.CLOSE_WINDOW_ID, self.StopAndRemoveUI) | |
444 | ||
445 | stop_bmp = getStopBitmap() | |
446 | tb.AddSimpleTool( self.KILL_PROCESS_ID, stop_bmp, _("Stop Debugging")) | |
447 | wx.EVT_TOOL(self, self.KILL_PROCESS_ID, self.StopExecution) | |
448 | ||
449 | tb.AddSeparator() | |
450 | ||
451 | break_bmp = getBreakBitmap() | |
452 | tb.AddSimpleTool( self.BREAK_INTO_DEBUGGER_ID, break_bmp, _("Break into Debugger")) | |
453 | wx.EVT_TOOL(self, self.BREAK_INTO_DEBUGGER_ID, self.BreakExecution) | |
454 | ||
455 | tb.AddSeparator() | |
456 | ||
457 | continue_bmp = getContinueBitmap() | |
458 | tb.AddSimpleTool( self.CONTINUE_ID, continue_bmp, _("Continue Execution")) | |
459 | wx.EVT_TOOL(self, self.CONTINUE_ID, self.OnContinue) | |
460 | ||
461 | tb.AddSeparator() | |
462 | ||
463 | next_bmp = getNextBitmap() | |
464 | tb.AddSimpleTool( self.NEXT_ID, next_bmp, _("Step to next line")) | |
465 | wx.EVT_TOOL(self, self.NEXT_ID, self.OnNext) | |
466 | ||
467 | step_bmp = getStepInBitmap() | |
468 | tb.AddSimpleTool( self.STEP_ID, step_bmp, _("Step in")) | |
469 | wx.EVT_TOOL(self, self.STEP_ID, self.OnSingleStep) | |
470 | ||
471 | stepOut_bmp = getStepReturnBitmap() | |
472 | tb.AddSimpleTool(self.STEP_OUT_ID, stepOut_bmp, _("Stop at function return")) | |
473 | wx.EVT_TOOL(self, self.STEP_OUT_ID, self.OnStepOut) | |
474 | ||
475 | tb.AddSeparator() | |
476 | if _WATCHES_ON: | |
477 | watch_bmp = getAddWatchBitmap() | |
478 | tb.AddSimpleTool(self.ADD_WATCH_ID, watch_bmp, _("Add a watch")) | |
479 | wx.EVT_TOOL(self, self.ADD_WATCH_ID, self.OnAddWatch) | |
480 | tb.AddSeparator() | |
481 | ||
482 | clear_bmp = getClearOutputBitmap() | |
483 | tb.AddSimpleTool(self.CLEAR_ID, clear_bmp, _("Clear output pane")) | |
484 | wx.EVT_TOOL(self, self.CLEAR_ID, self.OnClearOutput) | |
485 | ||
486 | tb.Realize() | |
487 | self.framesTab = None | |
488 | self.DisableWhileDebuggerRunning() | |
489 | self._notebook = wx.Notebook(self, -1, wx.DefaultPosition, wx.DefaultSize, wx.LB_DEFAULT, "Debugger") | |
490 | sizer.Add(self._notebook, 1, wx.ALIGN_LEFT|wx.ALL|wx.EXPAND, 1) | |
491 | self.consoleTab = self.MakeConsoleTab(self._notebook, wx.NewId(), None) | |
492 | self.framesTab = self.MakeFramesTab(self._notebook, wx.NewId(), None) | |
493 | self.breakPointsTab = self.MakeBreakPointsTab(self._notebook, wx.NewId(), None) | |
494 | self._notebook.AddPage(self.consoleTab, "Output") | |
495 | self._notebook.AddPage(self.framesTab, "Frames") | |
496 | self._notebook.AddPage(self.breakPointsTab, "Break Points") | |
497 | ||
498 | self._statusBar = wx.StatusBar( self, -1) | |
499 | self._statusBar.SetFieldsCount(1) | |
500 | sizer.Add(self._statusBar, 0, wx.EXPAND |wx.ALIGN_LEFT|wx.ALL, 1) | |
501 | ||
502 | self.SetStatusText("Starting debug...") | |
503 | ||
504 | self.SetSizer(sizer) | |
505 | sizer.Fit(self) | |
506 | config = wx.ConfigBase_Get() | |
507 | self._debuggerHost = self._guiHost = config.Read("DebuggerHostName", DEFAULT_HOST) | |
508 | url = 'http://' + self._debuggerHost + ':' + self._debuggerPort + '/' | |
509 | self._breakURL = 'http://' + self._debuggerHost + ':' + self._debuggerBreakPort + '/' | |
510 | self._callback = DebuggerCallback(self._guiHost, self._guiPort, url, self._breakURL, self) | |
511 | if DebuggerHarness.__file__.find('library.zip') > 0: | |
512 | try: | |
513 | fname = DebuggerHarness.__file__ | |
514 | parts = fname.split('library.zip') | |
515 | path = os.path.join(parts[0],'activegrid', 'tool', 'DebuggerHarness.py') | |
516 | except: | |
517 | tp, val, tb = sys.exc_info() | |
518 | traceback.print_exception(tp, val, tb) | |
519 | ||
520 | else: | |
521 | print "Starting debugger on these ports: %s, %s, %s" % (str(self._debuggerPort) , str(self._guiPort) , str(self._debuggerBreakPort)) | |
522 | path = DebuggerService.ExpandPath(DebuggerHarness.__file__) | |
523 | self._executor = Executor(path, self, self._debuggerHost, \ | |
524 | self._debuggerPort, self._debuggerBreakPort, self._guiHost, self._guiPort, self._command, callbackOnExit=self.ExecutorFinished) | |
525 | ||
526 | self.Bind(EVT_UPDATE_STDTEXT, self.AppendText) | |
527 | self.Bind(EVT_UPDATE_ERRTEXT, self.AppendErrorText) | |
528 | DebugCommandUI.debuggers.append(self) | |
529 | self._stopped = False | |
530 | ||
531 | def OnSingleStep(self, event): | |
532 | self._callback.SingleStep() | |
533 | ||
534 | def OnContinue(self, event): | |
535 | self._callback.Continue() | |
536 | ||
537 | def OnStepOut(self, event): | |
538 | self._callback.Return() | |
539 | ||
540 | def OnNext(self, event): | |
541 | self._callback.Next() | |
542 | ||
543 | def BreakPointChange(self): | |
544 | if not self._stopped: | |
545 | self._callback.pushBreakpoints() | |
546 | self.breakPointsTab.PopulateBPList() | |
547 | ||
548 | def __del__(self): | |
549 | if self in DebugCommandUI.debuggers: | |
550 | DebugCommandUI.debuggers.remove(self) | |
551 | ||
552 | def SwitchToOutputTab(self): | |
553 | self._notebook.SetSelection(0) | |
554 | ||
555 | def DisableWhileDebuggerRunning(self): | |
556 | self._tb.EnableTool(self.STEP_ID, False) | |
557 | self._tb.EnableTool(self.CONTINUE_ID, False) | |
558 | self._tb.EnableTool(self.STEP_OUT_ID, False) | |
559 | self._tb.EnableTool(self.NEXT_ID, False) | |
560 | self._tb.EnableTool(self.BREAK_INTO_DEBUGGER_ID, True) | |
561 | if _WATCHES_ON: | |
562 | self._tb.EnableTool(self.ADD_WATCH_ID, False) | |
563 | openDocs = wx.GetApp().GetDocumentManager().GetDocuments() | |
564 | for openDoc in openDocs: | |
565 | if(isinstance(openDoc.GetFirstView(), CodeEditor.CodeView)): | |
566 | openDoc.GetFirstView().GetCtrl().ClearCurrentLineMarkers() | |
567 | if self.framesTab: | |
568 | self.framesTab.ClearWhileRunning() | |
569 | #wx.GetApp().ProcessPendingEvents() #Yield(True) | |
570 | ||
571 | def EnableWhileDebuggerStopped(self): | |
572 | self._tb.EnableTool(self.STEP_ID, True) | |
573 | self._tb.EnableTool(self.CONTINUE_ID, True) | |
574 | self._tb.EnableTool(self.STEP_OUT_ID, True) | |
575 | self._tb.EnableTool(self.NEXT_ID, True) | |
576 | if _WATCHES_ON: | |
577 | self._tb.EnableTool(self.ADD_WATCH_ID, True) | |
578 | self._tb.EnableTool(self.BREAK_INTO_DEBUGGER_ID, False) | |
579 | #if _WINDOWS: | |
580 | # wx.GetApp().GetTopWindow().RequestUserAttention() | |
581 | ||
582 | def ExecutorFinished(self): | |
583 | if _VERBOSE: print "In ExectorFinished" | |
584 | try: | |
585 | self.DisableAfterStop() | |
586 | except wx._core.PyDeadObjectError: | |
587 | pass | |
588 | try: | |
589 | nb = self.GetParent() | |
590 | for i in range(0, nb.GetPageCount()): | |
591 | if self == nb.GetPage(i): | |
592 | text = nb.GetPageText(i) | |
593 | newText = text.replace("Debugging", "Finished") | |
594 | nb.SetPageText(i, newText) | |
595 | if _VERBOSE: print "In ExectorFinished, changed tab title." | |
596 | break | |
597 | except: | |
598 | if _VERBOSE: print "In ExectorFinished, got exception" | |
599 | ||
600 | def DisableAfterStop(self): | |
601 | self.DisableWhileDebuggerRunning() | |
602 | self._tb.EnableTool(self.BREAK_INTO_DEBUGGER_ID, False) | |
603 | self._tb.EnableTool(self.KILL_PROCESS_ID, False) | |
604 | ||
605 | def SynchCurrentLine(self, filename, lineNum): | |
606 | # FACTOR THIS INTO DocManager | |
607 | self.DeleteCurrentLineMarkers() | |
608 | ||
609 | # Filename will be <string> if we're in a bit of code that was executed from | |
610 | # a string (rather than a file). I haven't been able to get the original string | |
611 | # for display. | |
612 | if filename == '<string>': | |
613 | return | |
614 | foundView = None | |
615 | openDocs = wx.GetApp().GetDocumentManager().GetDocuments() | |
616 | for openDoc in openDocs: | |
617 | # This ugliness to prevent comparison failing because the drive letter | |
618 | # gets lowercased occasionally. Don't know why that happens or why it | |
619 | # only happens occasionally. | |
620 | if DebuggerService.ComparePaths(openDoc.GetFilename(),filename): | |
621 | foundView = openDoc.GetFirstView() | |
622 | break | |
623 | ||
624 | if not foundView: | |
625 | if _VERBOSE: | |
626 | print "filename=", filename | |
627 | doc = wx.GetApp().GetDocumentManager().CreateDocument(DebuggerService.ExpandPath(filename), wx.lib.docview.DOC_SILENT) | |
628 | foundView = doc.GetFirstView() | |
629 | ||
630 | if foundView: | |
631 | foundView.GetFrame().SetFocus() | |
632 | foundView.Activate() | |
633 | foundView.GotoLine(lineNum) | |
634 | startPos = foundView.PositionFromLine(lineNum) | |
635 | ||
636 | foundView.GetCtrl().MarkerAdd(lineNum -1, CodeEditor.CodeCtrl.CURRENT_LINE_MARKER_NUM) | |
637 | ||
638 | def DeleteCurrentLineMarkers(self): | |
639 | openDocs = wx.GetApp().GetDocumentManager().GetDocuments() | |
640 | for openDoc in openDocs: | |
641 | if(isinstance(openDoc.GetFirstView(), CodeEditor.CodeView)): | |
642 | openDoc.GetFirstView().GetCtrl().ClearCurrentLineMarkers() | |
643 | ||
644 | def LoadFramesListXML(self, framesXML): | |
645 | self.framesTab.LoadFramesListXML(framesXML) | |
646 | ||
647 | def SetStatusText(self, text): | |
648 | self._statusBar.SetStatusText(text,0) | |
649 | ||
650 | def Execute(self, initialArgs, startIn, environment): | |
651 | self._callback.start() | |
652 | self._executor.Execute(initialArgs, startIn, environment) | |
653 | self._callback.waitForRPC() | |
654 | ||
655 | def BreakExecution(self, event): | |
656 | self._callback.BreakExecution() | |
657 | ||
658 | ||
659 | def StopExecution(self, event): | |
660 | self._stopped = True | |
661 | self.DisableAfterStop() | |
662 | try: | |
663 | self._callback.ServerClose() | |
664 | except: | |
665 | pass | |
666 | try: | |
667 | if self._executor: | |
668 | self._executor.DoStopExecution() | |
669 | self._executor = None | |
670 | except: | |
671 | pass | |
672 | self.DeleteCurrentLineMarkers() | |
673 | DebugCommandUI.ReturnPortToPool(self._debuggerPort) | |
674 | DebugCommandUI.ReturnPortToPool(self._guiPort) | |
675 | DebugCommandUI.ReturnPortToPool(self._debuggerBreakPort) | |
676 | ||
677 | def StopAndRemoveUI(self, event): | |
678 | self.StopExecution(None) | |
679 | if self in DebugCommandUI.debuggers: | |
680 | DebugCommandUI.debuggers.remove(self) | |
681 | index = self._parentNoteBook.GetSelection() | |
682 | self._parentNoteBook.GetPage(index).Show(False) | |
683 | self._parentNoteBook.RemovePage(index) | |
684 | ||
685 | def GetConsoleTextControl(self): | |
686 | return self._textCtrl | |
687 | ||
688 | def OnClearOutput(self, event): | |
689 | self._textCtrl.SetReadOnly(False) | |
690 | self._textCtrl.ClearAll() | |
691 | self._textCtrl.SetReadOnly(True) | |
692 | ||
693 | def OnAddWatch(self, event): | |
694 | if self.framesTab: | |
695 | self.framesTab.OnWatch(event) | |
696 | ||
697 | def MakeConsoleTab(self, parent, id, debugger): | |
698 | panel = wx.Panel(parent, id) | |
699 | sizer = wx.BoxSizer(wx.HORIZONTAL) | |
700 | self._textCtrl = STCTextEditor.TextCtrl(panel, wx.NewId()) | |
701 | sizer.Add(self._textCtrl, 1, wx.ALIGN_LEFT|wx.ALL|wx.EXPAND, 1) | |
702 | self._textCtrl.SetViewLineNumbers(False) | |
703 | self._textCtrl.SetReadOnly(True) | |
704 | if wx.Platform == '__WXMSW__': | |
705 | font = "Courier New" | |
706 | else: | |
707 | font = "Courier" | |
708 | self._textCtrl.SetFont(wx.Font(9, wx.DEFAULT, wx.NORMAL, wx.NORMAL, faceName = font)) | |
709 | self._textCtrl.SetFontColor(wx.BLACK) | |
710 | self._textCtrl.StyleClearAll() | |
711 | panel.SetSizer(sizer) | |
712 | sizer.Fit(panel) | |
713 | ||
714 | return panel | |
715 | ||
716 | def MakeFramesTab(self, parent, id, debugger): | |
717 | panel = FramesUI(parent, id, self) | |
718 | return panel | |
719 | ||
720 | def MakeBreakPointsTab(self, parent, id, debugger): | |
721 | panel = BreakpointsUI(parent, id, self) | |
722 | return panel | |
723 | ||
724 | def AppendText(self, event): | |
725 | self._textCtrl.SetReadOnly(False) | |
726 | self._textCtrl.AddText(event.value) | |
727 | self._textCtrl.ScrollToLine(self._textCtrl.GetLineCount()) | |
728 | self._textCtrl.SetReadOnly(True) | |
729 | ||
730 | def AppendErrorText(self, event): | |
731 | self._textCtrl.SetReadOnly(False) | |
732 | self._textCtrl.SetFontColor(wx.RED) | |
733 | self._textCtrl.StyleClearAll() | |
734 | self._textCtrl.AddText(event.value) | |
735 | self._textCtrl.ScrollToLine(self._textCtrl.GetLineCount()) | |
736 | self._textCtrl.SetFontColor(wx.BLACK) | |
737 | self._textCtrl.StyleClearAll() | |
738 | self._textCtrl.SetReadOnly(True) | |
739 | ||
740 | class BreakpointsUI(wx.Panel): | |
741 | def __init__(self, parent, id, ui): | |
742 | wx.Panel.__init__(self, parent, id) | |
743 | self._ui = ui | |
744 | self.currentItem = None | |
745 | self.clearBPID = wx.NewId() | |
746 | self.Bind(wx.EVT_MENU, self.ClearBreakPoint, id=self.clearBPID) | |
747 | self.syncLineID = wx.NewId() | |
748 | self.Bind(wx.EVT_MENU, self.SyncBPLine, id=self.syncLineID) | |
749 | ||
750 | sizer = wx.BoxSizer(wx.VERTICAL) | |
751 | p1 = self | |
752 | self._bpListCtrl = wx.ListCtrl(p1, -1, pos=wx.DefaultPosition, size=(1000,1000), style=wx.LC_REPORT) | |
753 | sizer.Add(self._bpListCtrl, 1, wx.ALIGN_LEFT|wx.ALL|wx.EXPAND, 1) | |
754 | self._bpListCtrl.InsertColumn(0, "File") | |
755 | self._bpListCtrl.InsertColumn(1, "Line") | |
756 | self._bpListCtrl.InsertColumn(2, "Path") | |
757 | self._bpListCtrl.SetColumnWidth(0, 150) | |
758 | self._bpListCtrl.SetColumnWidth(1, 50) | |
759 | self._bpListCtrl.SetColumnWidth(2, 450) | |
760 | self._bpListCtrl.Bind(wx.EVT_LIST_ITEM_RIGHT_CLICK, self.OnListRightClick) | |
761 | self.Bind(wx.EVT_LIST_ITEM_SELECTED, self.ListItemSelected, self._bpListCtrl) | |
762 | self.Bind(wx.EVT_LIST_ITEM_DESELECTED, self.ListItemDeselected, self._bpListCtrl) | |
763 | ||
764 | self.PopulateBPList() | |
765 | ||
766 | p1.SetSizer(sizer) | |
767 | sizer.Fit(p1) | |
768 | p1.Layout() | |
769 | ||
770 | def PopulateBPList(self): | |
771 | list = self._bpListCtrl | |
772 | list.DeleteAllItems() | |
773 | ||
774 | bps = wx.GetApp().GetService(DebuggerService).GetMasterBreakpointDict() | |
775 | index = 0 | |
776 | for fileName in bps.keys(): | |
777 | shortFile = os.path.basename(fileName) | |
778 | lines = bps[fileName] | |
779 | if lines: | |
780 | for line in lines: | |
781 | list.InsertStringItem(index, shortFile) | |
782 | list.SetStringItem(index, 1, str(line)) | |
783 | list.SetStringItem(index, 2, fileName) | |
784 | ||
785 | def OnListRightClick(self, event): | |
786 | menu = wx.Menu() | |
787 | item = wx.MenuItem(menu, self.clearBPID, "Clear Breakpoint") | |
788 | menu.AppendItem(item) | |
789 | item = wx.MenuItem(menu, self.syncLineID, "Goto Source Line") | |
790 | menu.AppendItem(item) | |
791 | self.PopupMenu(menu, event.GetPosition()) | |
792 | menu.Destroy() | |
793 | ||
794 | def SyncBPLine(self, event): | |
795 | if self.currentItem != -1: | |
796 | list = self._bpListCtrl | |
797 | fileName = list.GetItem(self.currentItem, 2).GetText() | |
798 | lineNumber = list.GetItem(self.currentItem, 1).GetText() | |
799 | self._ui.SynchCurrentLine( fileName, int(lineNumber) ) | |
800 | ||
801 | def ClearBreakPoint(self, event): | |
802 | if self.currentItem >= 0: | |
803 | list = self._bpListCtrl | |
804 | fileName = list.GetItem(self.currentItem, 2).GetText() | |
805 | lineNumber = list.GetItem(self.currentItem, 1).GetText() | |
806 | wx.GetApp().GetService(DebuggerService).OnToggleBreakpoint(None, line=int(lineNumber) -1, fileName=fileName ) | |
807 | ||
808 | def ListItemSelected(self, event): | |
809 | self.currentItem = event.m_itemIndex | |
810 | ||
811 | def ListItemDeselected(self, event): | |
812 | self.currentItem = -1 | |
813 | ||
814 | class Watch: | |
815 | CODE_ALL_FRAMES = 1 | |
816 | CODE_THIS_BLOCK = 2 | |
817 | CODE_THIS_LINE = 4 | |
818 | CODE_RUN_ONCE = 8 | |
819 | ||
820 | def __init__(self, name, command, show_code=CODE_ALL_FRAMES): | |
821 | self._name = name | |
822 | self._command = command | |
823 | self._show_code = show_code | |
824 | ||
825 | class WatchDialog(wx.Dialog): | |
826 | WATCH_ALL_FRAMES = "Watch in all frames" | |
827 | WATCH_THIS_FRAME = "Watch in this frame only" | |
828 | WATCH_ONCE = "Watch once and delete" | |
829 | def __init__(self, parent, title, chain): | |
830 | wx.Dialog.__init__(self, parent, -1, title, style=wx.DEFAULT_DIALOG_STYLE) | |
831 | self._chain = chain | |
832 | self.label_2 = wx.StaticText(self, -1, "Watch Name:") | |
833 | self._watchNameTextCtrl = wx.TextCtrl(self, -1, "") | |
834 | self.label_3 = wx.StaticText(self, -1, "eval(", style=wx.ALIGN_RIGHT) | |
835 | self._watchValueTextCtrl = wx.TextCtrl(self, -1, "") | |
836 | self.label_4 = wx.StaticText(self, -1, ",frame.f_globals, frame.f_locals)") | |
837 | self.radio_box_1 = wx.RadioBox(self, -1, "Watch Information", choices=[WatchDialog.WATCH_ALL_FRAMES, WatchDialog.WATCH_THIS_FRAME, WatchDialog.WATCH_ONCE], majorDimension=0, style=wx.RA_SPECIFY_ROWS) | |
838 | ||
839 | self._okButton = wx.Button(self, wx.ID_OK, "OK", size=(75,-1)) | |
840 | self._okButton.SetDefault() | |
841 | self._okButton.SetHelpText(_("The OK button completes the dialog")) | |
842 | def OnOkClick(event): | |
843 | if self._watchNameTextCtrl.GetValue() == "": | |
844 | wx.MessageBox(_("You must enter a name for the watch."), _("Add Watch")) | |
845 | return | |
846 | if self._watchValueTextCtrl.GetValue() == "": | |
847 | wx.MessageBox(_("You must enter some code to run for the watch."), _("Add Watch")) | |
848 | return | |
849 | self.EndModal(wx.ID_OK) | |
850 | self.Bind(wx.EVT_BUTTON, OnOkClick, self._okButton) | |
851 | ||
852 | self._cancelButton = wx.Button(self, wx.ID_CANCEL, _("Cancel"), size=(75,-1)) | |
853 | self._cancelButton.SetHelpText(_("The Cancel button cancels the dialog.")) | |
854 | ||
855 | self.__set_properties() | |
856 | self.__do_layout() | |
857 | ||
858 | def GetSettings(self): | |
859 | return self._watchNameTextCtrl.GetValue(), self._watchValueTextCtrl.GetValue(), self.GetSendFrame(), self.GetRunOnce() | |
860 | ||
861 | def GetSendFrame(self): | |
862 | return (WatchDialog.WATCH_ALL_FRAMES != self.radio_box_1.GetStringSelection()) | |
863 | ||
864 | def GetRunOnce(self): | |
865 | return (WatchDialog.WATCH_ONCE == self.radio_box_1.GetStringSelection()) | |
866 | ||
867 | def __set_properties(self): | |
868 | self.SetTitle("Add a Watch") | |
869 | #self.SetSize((400, 250)) | |
870 | self.radio_box_1.SetSelection(0) | |
871 | ||
872 | def __do_layout(self): | |
873 | sizer_1 = wx.BoxSizer(wx.VERTICAL) | |
874 | grid_sizer_4 = wx.FlexGridSizer(1, 3, 5, 5) | |
875 | grid_sizer_2 = wx.FlexGridSizer(1, 2, 5, 5) | |
876 | grid_sizer_2.Add(self.label_2, 0, wx.ALIGN_CENTER_VERTICAL|wx.FIXED_MINSIZE, 0) | |
877 | grid_sizer_2.Add(self._watchNameTextCtrl, 0, wx.EXPAND, 0) | |
878 | grid_sizer_2.AddGrowableCol(1) | |
879 | sizer_1.Add(grid_sizer_2, 1, wx.EXPAND, 0) | |
880 | grid_sizer_4.Add(self.label_3, 0, wx.ALIGN_CENTER_VERTICAL|wx.FIXED_MINSIZE, 0) | |
881 | grid_sizer_4.Add(self._watchValueTextCtrl, 0, wx.EXPAND, 0) | |
882 | grid_sizer_4.AddGrowableCol(1) | |
883 | grid_sizer_4.Add(self.label_4, 0, wx.ALIGN_CENTER_VERTICAL|wx.FIXED_MINSIZE, 0) | |
884 | sizer_1.Add(grid_sizer_4, 0, wx.EXPAND, 0) | |
885 | sizer_1.Add(self.radio_box_1, 0, wx.EXPAND, 0) | |
886 | ||
887 | box = wx.BoxSizer(wx.HORIZONTAL) | |
888 | box.Add(self._okButton, 0, wx.ALIGN_RIGHT|wx.ALL, 5) | |
889 | box.Add(self._cancelButton, 0, wx.ALIGN_RIGHT|wx.ALL, 5) | |
890 | sizer_1.Add(box, 1, wx.EXPAND, 0) | |
891 | self.SetAutoLayout(True) | |
892 | self.SetSizer(sizer_1) | |
893 | self.Layout() | |
894 | ||
895 | class FramesUI(wx.SplitterWindow): | |
896 | def __init__(self, parent, id, ui): | |
897 | wx.SplitterWindow.__init__(self, parent, id, style = wx.SP_3D) | |
898 | self._ui = ui | |
899 | sizer = wx.BoxSizer(wx.VERTICAL) | |
900 | self._p1 = p1 = wx.ScrolledWindow(self, -1) | |
901 | p1.Bind(wx.EVT_SIZE, self.OnSize) | |
902 | ||
903 | self._framesListCtrl = wx.ListCtrl(p1, -1, pos=wx.DefaultPosition, size=(250,150), style=wx.LC_REPORT) | |
904 | sizer.Add(self._framesListCtrl, 1, wx.ALIGN_LEFT|wx.ALL|wx.EXPAND, 1) | |
905 | self._framesListCtrl.InsertColumn(0, "Frame") | |
906 | self._framesListCtrl.SetColumnWidth(0, 250) | |
907 | self._framesListCtrl.Bind(wx.EVT_LIST_ITEM_RIGHT_CLICK, self.OnListRightClick) | |
908 | self.Bind(wx.EVT_LIST_ITEM_SELECTED, self.ListItemSelected, self._framesListCtrl) | |
909 | self.Bind(wx.EVT_LIST_ITEM_DESELECTED, self.ListItemDeselected, self._framesListCtrl) | |
910 | ||
911 | sizer2 = wx.BoxSizer(wx.VERTICAL) | |
912 | self._p2 = p2 = wx.ScrolledWindow(self, -1) | |
913 | p2.Bind(wx.EVT_SIZE, self.OnSize) | |
914 | ||
915 | self._treeCtrl = wx.gizmos.TreeListCtrl(p2, -1, size=(530,250), style=wx.TR_DEFAULT_STYLE| wx.TR_FULL_ROW_HIGHLIGHT) | |
916 | self._treeCtrl.Bind(wx.EVT_TREE_ITEM_RIGHT_CLICK, self.OnRightClick) | |
917 | sizer2.Add(self._framesListCtrl, 1, wx.ALIGN_LEFT|wx.ALL|wx.EXPAND, 1) | |
918 | tree = self._treeCtrl | |
919 | tree.AddColumn("Thing") | |
920 | tree.AddColumn("Value") | |
921 | tree.SetMainColumn(0) # the one with the tree in it... | |
922 | tree.SetColumnWidth(0, 175) | |
923 | tree.SetColumnWidth(1, 355) | |
924 | self._root = tree.AddRoot("Frame") | |
925 | tree.SetItemText(self._root, "", 1) | |
926 | ||
927 | self.SetMinimumPaneSize(20) | |
928 | self.SplitVertically(p1, p2, 250) | |
929 | self.currentItem = None | |
930 | self.Layout() | |
931 | ||
932 | def OnRightClick(self, event): | |
933 | #Refactor this... | |
934 | self._introspectItem = event.GetItem() | |
935 | self._parentChain = self.GetItemChain(event.GetItem()) | |
936 | watchOnly = len(self._parentChain) < 1 | |
937 | if not _WATCHES_ON and watchOnly: | |
938 | return | |
939 | menu = wx.Menu() | |
940 | if not watchOnly: | |
941 | if not hasattr(self, "introspectID"): | |
942 | self.introspectID = wx.NewId() | |
943 | self.Bind(wx.EVT_MENU, self.OnIntrospect, id=self.introspectID) | |
944 | item = wx.MenuItem(menu, self.introspectID, "Attempt Introspection") | |
945 | menu.AppendItem(item) | |
946 | menu.AppendSeparator() | |
947 | if _WATCHES_ON: | |
948 | if not hasattr(self, "watchID"): | |
949 | self.watchID = wx.NewId() | |
950 | self.Bind(wx.EVT_MENU, self.OnWatch, id=self.watchID) | |
951 | item = wx.MenuItem(menu, self.watchID, "Create a Watch") | |
952 | menu.AppendItem(item) | |
953 | menu.AppendSeparator() | |
954 | if not watchOnly: | |
955 | if not hasattr(self, "viewID"): | |
956 | self.viewID = wx.NewId() | |
957 | self.Bind(wx.EVT_MENU, self.OnView, id=self.viewID) | |
958 | item = wx.MenuItem(menu, self.viewID, "View in Dialog") | |
959 | menu.AppendItem(item) | |
960 | offset = wx.Point(x=0, y=20) | |
961 | menuSpot = event.GetPoint() + offset | |
962 | self._treeCtrl.PopupMenu(menu, menuSpot) | |
963 | menu.Destroy() | |
964 | self._parentChain = None | |
965 | self._introspectItem = None | |
966 | ||
967 | def GetItemChain(self, item): | |
968 | parentChain = [] | |
969 | if item: | |
970 | if _VERBOSE: print 'Exploding: %s' % self._treeCtrl.GetItemText(item, 0) | |
971 | while item != self._root: | |
972 | text = self._treeCtrl.GetItemText(item, 0) | |
973 | if _VERBOSE: print "Appending ", text | |
974 | parentChain.append(text) | |
975 | item = self._treeCtrl.GetItemParent(item) | |
976 | parentChain.reverse() | |
977 | return parentChain | |
978 | ||
979 | def OnView(self, event): | |
980 | title = self._treeCtrl.GetItemText(self._introspectItem,0) | |
981 | value = self._treeCtrl.GetItemText(self._introspectItem,1) | |
982 | dlg = wx.lib.dialogs.ScrolledMessageDialog(self, value, title, style=wx.DD_DEFAULT_STYLE | wx.RESIZE_BORDER) | |
983 | dlg.Show() | |
984 | ||
985 | def OnWatch(self, event): | |
986 | try: | |
987 | if hasattr(self, '_parentChain'): | |
988 | wd = WatchDialog(wx.GetApp().GetTopWindow(), "Add a Watch", self._parentChain) | |
989 | else: | |
990 | wd = WatchDialog(wx.GetApp().GetTopWindow(), "Add a Watch", None) | |
991 | if wd.ShowModal() == wx.ID_OK: | |
992 | name, text, send_frame, run_once = wd.GetSettings() | |
993 | if send_frame: | |
994 | frameNode = self._stack[int(self.currentItem)] | |
995 | message = frameNode.getAttribute("message") | |
996 | else: | |
997 | message = "" | |
998 | binType = self._ui._callback._debuggerServer.add_watch(name, text, message, run_once) | |
999 | xmldoc = bz2.decompress(binType.data) | |
1000 | domDoc = parseString(xmldoc) | |
1001 | nodeList = domDoc.getElementsByTagName('watch') | |
1002 | if len(nodeList) == 1: | |
1003 | watchValue = nodeList.item(0).getAttribute("message") | |
1004 | except: | |
1005 | tp, val, tb = sys.exc_info() | |
1006 | traceback.print_exception(tp, val, tb) | |
1007 | ||
1008 | def OnIntrospect(self, event): | |
1009 | wx.GetApp().GetTopWindow().SetCursor(wx.StockCursor(wx.CURSOR_WAIT)) | |
1010 | ||
1011 | try: | |
1012 | list = self._framesListCtrl | |
1013 | frameNode = self._stack[int(self.currentItem)] | |
1014 | message = frameNode.getAttribute("message") | |
1015 | binType = self._ui._callback._debuggerServer.attempt_introspection(message, self._parentChain) | |
1016 | xmldoc = bz2.decompress(binType.data) | |
1017 | ||
1018 | domDoc = parseString(xmldoc) | |
1019 | nodeList = domDoc.getElementsByTagName('replacement') | |
1020 | replacementNode = nodeList.item(0) | |
1021 | if len(replacementNode.childNodes): | |
1022 | thingToWalk = replacementNode.childNodes.item(0) | |
1023 | tree = self._treeCtrl | |
1024 | parent = tree.GetItemParent(self._introspectItem) | |
1025 | treeNode = self.AppendSubTreeFromNode(thingToWalk, thingToWalk.getAttribute('name'), parent, insertBefore=self._introspectItem) | |
1026 | tree.Delete(self._introspectItem) | |
1027 | except: | |
1028 | tp,val,tb = sys.exc_info() | |
1029 | traceback.print_exception(tp, val, tb) | |
1030 | ||
1031 | wx.GetApp().GetTopWindow().SetCursor(wx.StockCursor(wx.CURSOR_DEFAULT)) | |
1032 | ||
1033 | def OnSize(self, event): | |
1034 | self._treeCtrl.SetSize(self._p2.GetSize()) | |
1035 | w,h = self._p1.GetClientSizeTuple() | |
1036 | self._framesListCtrl.SetDimensions(0, 0, w, h) | |
1037 | ||
1038 | def ClearWhileRunning(self): | |
1039 | list = self._framesListCtrl | |
1040 | list.DeleteAllItems() | |
1041 | tree = self._treeCtrl | |
1042 | tree.Hide() | |
1043 | ||
1044 | def OnListRightClick(self, event): | |
1045 | if not hasattr(self, "syncFrameID"): | |
1046 | self.syncFrameID = wx.NewId() | |
1047 | self.Bind(wx.EVT_MENU, self.OnSyncFrame, id=self.syncFrameID) | |
1048 | menu = wx.Menu() | |
1049 | item = wx.MenuItem(menu, self.syncFrameID, "Goto Source Line") | |
1050 | menu.AppendItem(item) | |
1051 | self.PopupMenu(menu, event.GetPosition()) | |
1052 | menu.Destroy() | |
1053 | ||
1054 | def OnSyncFrame(self, event): | |
1055 | list = self._framesListCtrl | |
1056 | frameNode = self._stack[int(self.currentItem)] | |
1057 | file = frameNode.getAttribute("file") | |
1058 | line = frameNode.getAttribute("line") | |
1059 | self._ui.SynchCurrentLine( file, int(line) ) | |
1060 | ||
1061 | def LoadFramesListXML(self, framesXML): | |
1062 | wx.GetApp().GetTopWindow().SetCursor(wx.StockCursor(wx.CURSOR_WAIT)) | |
1063 | try: | |
1064 | domDoc = parseString(framesXML) | |
1065 | list = self._framesListCtrl | |
1066 | list.DeleteAllItems() | |
1067 | self._stack = [] | |
1068 | nodeList = domDoc.getElementsByTagName('frame') | |
1069 | frame_count = -1 | |
1070 | for index in range(0, nodeList.length): | |
1071 | frameNode = nodeList.item(index) | |
1072 | message = frameNode.getAttribute("message") | |
1073 | list.InsertStringItem(index, message) | |
1074 | self._stack.append(frameNode) | |
1075 | frame_count += 1 | |
1076 | list.Select(frame_count) | |
1077 | self._p1.FitInside() | |
1078 | frameNode = nodeList.item(index) | |
1079 | file = frameNode.getAttribute("file") | |
1080 | line = frameNode.getAttribute("line") | |
1081 | self._ui.SynchCurrentLine( file, int(line) ) | |
1082 | except: | |
1083 | tp,val,tb=sys.exc_info() | |
1084 | traceback.print_exception(tp, val, tb) | |
1085 | ||
1086 | wx.GetApp().GetTopWindow().SetCursor(wx.StockCursor(wx.CURSOR_DEFAULT)) | |
1087 | ||
1088 | def ListItemDeselected(self, event): | |
1089 | pass | |
1090 | ||
1091 | def ListItemSelected(self, event): | |
1092 | self.currentItem = event.m_itemIndex | |
1093 | frameNode = self._stack[int(self.currentItem)] | |
1094 | self.PopulateTreeFromFrameNode(frameNode) | |
1095 | # Temporarily doing this to test out automatically swicting to source line. | |
1096 | self.OnSyncFrame(None) | |
1097 | ||
1098 | def PopulateTreeFromFrameNode(self, frameNode): | |
1099 | tree = self._treeCtrl | |
1100 | tree.Show(True) | |
1101 | root = self._root | |
1102 | tree.DeleteChildren(root) | |
1103 | children = frameNode.childNodes | |
1104 | firstChild = None | |
1105 | for index in range(0, children.length): | |
1106 | subNode = children.item(index) | |
1107 | treeNode = self.AppendSubTreeFromNode(subNode, subNode.getAttribute('name'), root) | |
1108 | if not firstChild: | |
1109 | firstChild = treeNode | |
1110 | tree.Expand(root) | |
1111 | tree.Expand(firstChild) | |
1112 | self._p2.FitInside() | |
1113 | ||
1114 | def AppendSubTreeFromNode(self, node, name, parent, insertBefore=None): | |
1115 | tree = self._treeCtrl | |
1116 | if insertBefore != None: | |
1117 | treeNode = tree.InsertItem(parent, insertBefore, name) | |
1118 | else: | |
1119 | treeNode = tree.AppendItem(parent, name) | |
1120 | children = node.childNodes | |
1121 | if children.length == 0: | |
1122 | tree.SetItemText(treeNode, self.StripOuterSingleQuotes(node.getAttribute("value")), 1) | |
1123 | for index in range(0, children.length): | |
1124 | subNode = children.item(index) | |
1125 | if self.HasChildren(subNode): | |
1126 | self.AppendSubTreeFromNode(subNode, subNode.getAttribute("name"), treeNode) | |
1127 | else: | |
1128 | name = subNode.getAttribute("name") | |
1129 | value = self.StripOuterSingleQuotes(subNode.getAttribute("value")) | |
1130 | n = tree.AppendItem(treeNode, name) | |
1131 | tree.SetItemText(n, value, 1) | |
1132 | return treeNode | |
1133 | ||
1134 | def StripOuterSingleQuotes(self, string): | |
1135 | if string.startswith("'") and string.endswith("'"): | |
1136 | return string[1:-1] | |
1137 | elif type(string) == types.UnicodeType: | |
1138 | return string[1:-1] | |
1139 | else: | |
1140 | return string | |
1141 | ||
1142 | def HasChildren(self, node): | |
1143 | try: | |
1144 | return node.childNodes.length > 0 | |
1145 | except: | |
1146 | tp,val,tb=sys.exc_info() | |
1147 | return False | |
1148 | ||
1149 | class DebuggerView(Service.ServiceView): | |
1150 | ||
1151 | #---------------------------------------------------------------------------- | |
1152 | # Overridden methods | |
1153 | #---------------------------------------------------------------------------- | |
1154 | ||
1155 | def __init__(self, service): | |
1156 | Service.ServiceView.__init__(self, service) | |
1157 | ||
1158 | def _CreateControl(self, parent, id): | |
1159 | return None | |
1160 | ||
1161 | #------------------------------------------------------------------------------ | |
1162 | # Event handling | |
1163 | #----------------------------------------------------------------------------- | |
1164 | ||
1165 | def OnToolClicked(self, event): | |
1166 | self.GetFrame().ProcessEvent(event) | |
1167 | ||
1168 | #------------------------------------------------------------------------------ | |
1169 | # Class methods | |
1170 | #----------------------------------------------------------------------------- | |
1171 | ||
1172 | class Interaction: | |
1173 | def __init__(self, message, framesXML, info=None, quit=False): | |
1174 | self._framesXML = framesXML | |
1175 | self._message = message | |
1176 | self._info = info | |
1177 | self._quit = quit | |
1178 | ||
1179 | def getFramesXML(self): | |
1180 | return self._framesXML | |
1181 | ||
1182 | def getMessage(self): | |
1183 | return self._message | |
1184 | ||
1185 | def getInfo(self): | |
1186 | return self._info | |
1187 | ||
1188 | def getQuit(self): | |
1189 | return self._quit | |
1190 | ||
1191 | class AGXMLRPCServer(SimpleXMLRPCServer.SimpleXMLRPCServer): | |
1192 | def __init__(self, address, logRequests=0): | |
1193 | SimpleXMLRPCServer.SimpleXMLRPCServer.__init__(self, address, logRequests=logRequests) | |
1194 | ||
1195 | class RequestHandlerThread(threading.Thread): | |
1196 | def __init__(self, queue, address): | |
1197 | threading.Thread.__init__(self) | |
1198 | self._keepGoing = True | |
1199 | self._queue = queue | |
1200 | self._address = address | |
1201 | self._server = AGXMLRPCServer(self._address,logRequests=0) | |
1202 | self._server.register_function(self.interaction) | |
1203 | self._server.register_function(self.quit) | |
1204 | self._server.register_function(self.dummyOperation) | |
1205 | if _VERBOSE: print "RequestHandlerThread on fileno %s" % str(self._server.fileno()) | |
1206 | ||
1207 | def run(self): | |
1208 | while self._keepGoing: | |
1209 | try: | |
1210 | self._server.handle_request() | |
1211 | except: | |
1212 | tp, val, tb = sys.exc_info() | |
1213 | traceback.print_exception(tp, val, tb) | |
1214 | self._keepGoing = False | |
1215 | if _VERBOSE: print "Exiting Request Handler Thread." | |
1216 | ||
1217 | def interaction(self, message, frameXML, info): | |
1218 | if _VERBOSE: print "In RequestHandlerThread.interaction -- adding to queue" | |
1219 | interaction = Interaction(message, frameXML, info) | |
1220 | self._queue.put(interaction) | |
1221 | return "" | |
1222 | ||
1223 | def quit(self): | |
1224 | interaction = Interaction(None, None, info=None, quit=True) | |
1225 | self._queue.put(interaction) | |
1226 | return "" | |
1227 | ||
1228 | def dummyOperation(self): | |
1229 | return "" | |
1230 | ||
1231 | def AskToStop(self): | |
1232 | self._keepGoing = False | |
1233 | if type(self._server) is not types.NoneType: | |
1234 | try: | |
1235 | # This is a really ugly way to make sure this thread isn't blocked in | |
1236 | # handle_request. | |
1237 | url = 'http://' + self._address[0] + ':' + str(self._address[1]) + '/' | |
1238 | tempServer = xmlrpclib.ServerProxy(url, allow_none=1) | |
1239 | tempServer.dummyOperation() | |
1240 | except: | |
1241 | tp, val, tb = sys.exc_info() | |
1242 | traceback.print_exception(tp, val, tb) | |
1243 | self._server.server_close() | |
1244 | ||
1245 | ||
1246 | class RequestBreakThread(threading.Thread): | |
1247 | def __init__(self, server, interrupt=False, pushBreakpoints=False, breakDict=None, kill=False): | |
1248 | threading.Thread.__init__(self) | |
1249 | self._server = server | |
1250 | ||
1251 | self._interrupt = interrupt | |
1252 | self._pushBreakpoints = pushBreakpoints | |
1253 | self._breakDict = breakDict | |
1254 | self._kill = kill | |
1255 | ||
1256 | def run(self): | |
1257 | try: | |
1258 | if _VERBOSE: print "RequestBreakThread, before call" | |
1259 | if self._interrupt: | |
1260 | self._server.break_requested() | |
1261 | if self._pushBreakpoints: | |
1262 | self._server.update_breakpoints(xmlrpclib.Binary(pickle.dumps(self._breakDict))) | |
1263 | if self._kill: | |
1264 | try: | |
1265 | self._server.die() | |
1266 | except: | |
1267 | pass | |
1268 | if _VERBOSE: print "RequestBreakThread, after call" | |
1269 | except: | |
1270 | tp,val,tb = sys.exc_info() | |
1271 | traceback.print_exception(tp, val, tb) | |
1272 | ||
1273 | class DebuggerOperationThread(threading.Thread): | |
1274 | def __init__(self, function): | |
1275 | threading.Thread.__init__(self) | |
1276 | self._function = function | |
1277 | ||
1278 | def run(self): | |
1279 | if _VERBOSE: print "In DOT, before call" | |
1280 | try: | |
1281 | self._function() | |
1282 | except: | |
1283 | tp,val,tb = sys.exc_info() | |
1284 | traceback.print_exception(tp, val, tb) | |
1285 | if _VERBOSE: print "In DOT, after call" | |
1286 | ||
1287 | class DebuggerCallback: | |
1288 | ||
1289 | def __init__(self, host, port, debugger_url, break_url, debuggerUI): | |
1290 | if _VERBOSE: print "+++++++ Creating server on port, ", str(port) | |
1291 | ||
1292 | self._queue = Queue.Queue(50) | |
1293 | self._host = host | |
1294 | self._port = int(port) | |
1295 | threading._VERBOSE = _VERBOSE | |
1296 | self._serverHandlerThread = RequestHandlerThread(self._queue, (self._host, self._port)) | |
1297 | ||
1298 | self._debugger_url = debugger_url | |
1299 | self._debuggerServer = None | |
1300 | self._waiting = False | |
1301 | self._service = wx.GetApp().GetService(DebuggerService) | |
1302 | self._debuggerUI = debuggerUI | |
1303 | self._break_url = break_url | |
1304 | self._breakServer = None | |
1305 | self._firstInteraction = True | |
1306 | self._pendingBreak = False | |
1307 | ||
1308 | def start(self): | |
1309 | self._serverHandlerThread.start() | |
1310 | ||
1311 | def ServerClose(self): | |
1312 | rbt = RequestBreakThread(self._breakServer, kill=True) | |
1313 | rbt.start() | |
1314 | self.setWaiting(False) | |
1315 | if self._serverHandlerThread: | |
1316 | self._serverHandlerThread.AskToStop() | |
1317 | self._serverHandlerThread = None | |
1318 | ||
1319 | def BreakExecution(self): | |
1320 | rbt = RequestBreakThread(self._breakServer, interrupt=True) | |
1321 | rbt.start() | |
1322 | ||
1323 | def SingleStep(self): | |
1324 | self._debuggerUI.DisableWhileDebuggerRunning() | |
1325 | #dot = DebuggerOperationThread(self._debuggerServer.set_step) | |
1326 | #dot.start() | |
1327 | self._debuggerServer.set_step() # Figure out where to set allowNone | |
1328 | self.waitForRPC() | |
1329 | ||
1330 | def Next(self): | |
1331 | self._debuggerUI.DisableWhileDebuggerRunning() | |
1332 | #dot = DebuggerOperationThread(self._debuggerServer.set_next) | |
1333 | #dot.start() | |
1334 | self._debuggerServer.set_next() | |
1335 | self.waitForRPC() | |
1336 | ||
1337 | def Continue(self): | |
1338 | self._debuggerUI.DisableWhileDebuggerRunning() | |
1339 | #dot = DebuggerOperationThread(self._debuggerServer.set_continue) | |
1340 | #dot.start() | |
1341 | self._debuggerServer.set_continue() | |
1342 | self.waitForRPC() | |
1343 | ||
1344 | def Return(self): | |
1345 | self._debuggerUI.DisableWhileDebuggerRunning() | |
1346 | #dot = DebuggerOperationThread(self._debuggerServer.set_return) | |
1347 | #dot.start() | |
1348 | self._debuggerServer.set_return() | |
1349 | self.waitForRPC() | |
1350 | ||
1351 | def setWaiting(self, value): | |
1352 | self._waiting = value | |
1353 | ||
1354 | def getWaiting(self): | |
1355 | return self._waiting | |
1356 | ||
1357 | def readQueue(self): | |
1358 | if self._queue.qsize(): | |
1359 | try: | |
1360 | item = self._queue.get_nowait() | |
1361 | if item.getQuit(): | |
1362 | self.interaction(None, None, None, True) | |
1363 | else: | |
1364 | data = bz2.decompress(item.getFramesXML().data) | |
1365 | self.interaction(item.getMessage().data, data, item.getInfo(), False) | |
1366 | except Queue.Empty: | |
1367 | pass | |
1368 | ||
1369 | def pushBreakpoints(self): | |
1370 | rbt = RequestBreakThread(self._breakServer, pushBreakpoints=True, breakDict=self._service.GetMasterBreakpointDict()) | |
1371 | rbt.start() | |
1372 | ||
1373 | ||
1374 | def waitForRPC(self): | |
1375 | self.setWaiting(True) | |
1376 | while self.getWaiting(): | |
1377 | try: | |
1378 | self.readQueue() | |
1379 | import time | |
1380 | time.sleep(0.02) | |
1381 | except: | |
1382 | tp, val, tb = sys.exc_info() | |
1383 | traceback.print_exception(tp, val, tb) | |
1384 | wx.GetApp().Yield(True) | |
1385 | if _VERBOSE: print "Exiting waitForRPC." | |
1386 | ||
1387 | def interaction(self, message, frameXML, info, quit): | |
1388 | ||
1389 | #This method should be hit as the debugger starts. | |
1390 | if self._firstInteraction: | |
1391 | self._firstInteraction = False | |
1392 | self._debuggerServer = xmlrpclib.ServerProxy(self._debugger_url, allow_none=1) | |
1393 | self._breakServer = xmlrpclib.ServerProxy(self._break_url, allow_none=1) | |
1394 | self.pushBreakpoints() | |
1395 | self.setWaiting(False) | |
1396 | if _VERBOSE: print "+"*40 | |
1397 | if(quit): | |
1398 | self._debuggerUI.StopExecution(None) | |
1399 | return "" | |
1400 | if(info != ""): | |
1401 | if _VERBOSE: print "Hit interaction with exception" | |
1402 | #self._debuggerUI.StopExecution(None) | |
1403 | #self._debuggerUI.SetStatusText("Got exception: " + str(info)) | |
1404 | self._debuggerUI.SwitchToOutputTab() | |
1405 | else: | |
1406 | if _VERBOSE: print "Hit interaction no exception" | |
1407 | self._debuggerUI.SetStatusText(message) | |
1408 | self._debuggerUI.LoadFramesListXML(frameXML) | |
1409 | self._debuggerUI.EnableWhileDebuggerStopped() | |
1410 | if _VERBOSE: print "+"*40 | |
1411 | ||
1412 | class DebuggerService(Service.Service): | |
1413 | ||
1414 | #---------------------------------------------------------------------------- | |
1415 | # Constants | |
1416 | #---------------------------------------------------------------------------- | |
1417 | TOGGLE_BREAKPOINT_ID = wx.NewId() | |
1418 | CLEAR_ALL_BREAKPOINTS = wx.NewId() | |
1419 | RUN_ID = wx.NewId() | |
1420 | DEBUG_ID = wx.NewId() | |
1421 | ||
1422 | def ComparePaths(first, second): | |
1423 | one = DebuggerService.ExpandPath(first) | |
1424 | two = DebuggerService.ExpandPath(second) | |
1425 | if _WINDOWS: | |
1426 | return one.lower() == two.lower() | |
1427 | else: | |
1428 | return one == two | |
1429 | ComparePaths = staticmethod(ComparePaths) | |
1430 | ||
1431 | # Make sure we're using an expanded path on windows. | |
1432 | def ExpandPath(path): | |
1433 | if _WINDOWS: | |
1434 | try: | |
1435 | return win32api.GetLongPathName(path) | |
1436 | except: | |
1437 | print "Cannot get long path for %s" % path | |
1438 | ||
1439 | return path | |
1440 | ||
1441 | ExpandPath = staticmethod(ExpandPath) | |
1442 | ||
1443 | #---------------------------------------------------------------------------- | |
1444 | # Overridden methods | |
1445 | #---------------------------------------------------------------------------- | |
1446 | ||
1447 | def __init__(self, serviceName, embeddedWindowLocation = wx.lib.pydocview.EMBEDDED_WINDOW_LEFT): | |
1448 | Service.Service.__init__(self, serviceName, embeddedWindowLocation) | |
1449 | self.BREAKPOINT_DICT_STRING = "MasterBreakpointDict" | |
1450 | config = wx.ConfigBase_Get() | |
1451 | pickledbps = config.Read(self.BREAKPOINT_DICT_STRING) | |
1452 | if pickledbps: | |
1453 | try: | |
1454 | self._masterBPDict = pickle.loads(pickledbps.encode('ascii')) | |
1455 | except: | |
1456 | tp, val, tb = sys.exc_info() | |
1457 | traceback.print_exception(tp,val,tb) | |
1458 | self._masterBPDict = {} | |
1459 | else: | |
1460 | self._masterBPDict = {} | |
1461 | ||
1462 | def OnCloseFrame(self, event): | |
1463 | # IS THIS THE RIGHT PLACE? | |
1464 | try: | |
1465 | config = wx.ConfigBase_Get() | |
1466 | config.Write(self.BREAKPOINT_DICT_STRING, pickle.dumps(self._masterBPDict)) | |
1467 | except: | |
1468 | tp,val,tb = sys.exc_info() | |
1469 | traceback.print_exception(tp, val, tb) | |
1470 | return True | |
1471 | ||
1472 | def _CreateView(self): | |
1473 | return DebuggerView(self) | |
1474 | ||
1475 | ||
1476 | #---------------------------------------------------------------------------- | |
1477 | # Service specific methods | |
1478 | #---------------------------------------------------------------------------- | |
1479 | ||
1480 | def InstallControls(self, frame, menuBar = None, toolBar = None, statusBar = None, document = None): | |
1481 | #Service.Service.InstallControls(self, frame, menuBar, toolBar, statusBar, document) | |
1482 | ||
1483 | config = wx.ConfigBase_Get() | |
1484 | ||
1485 | debuggerMenu = wx.Menu() | |
1486 | if not menuBar.FindItemById(DebuggerService.CLEAR_ALL_BREAKPOINTS): | |
1487 | ||
1488 | debuggerMenu.Append(DebuggerService.RUN_ID, _("&Run...\tCtrl+R"), _("Runs a file")) | |
1489 | wx.EVT_MENU(frame, DebuggerService.RUN_ID, frame.ProcessEvent) | |
1490 | wx.EVT_UPDATE_UI(frame, DebuggerService.RUN_ID, frame.ProcessUpdateUIEvent) | |
1491 | ||
1492 | debuggerMenu.Append(DebuggerService.DEBUG_ID, _("&Debug...\tCtrl+D"), _("Debugs a file")) | |
1493 | wx.EVT_MENU(frame, DebuggerService.DEBUG_ID, frame.ProcessEvent) | |
1494 | wx.EVT_UPDATE_UI(frame, DebuggerService.DEBUG_ID, frame.ProcessUpdateUIEvent) | |
1495 | ||
1496 | debuggerMenu.AppendSeparator() | |
1497 | ||
1498 | debuggerMenu.Append(DebuggerService.TOGGLE_BREAKPOINT_ID, _("&Toggle Breakpoint...\tCtrl+B"), _("Toggle a breakpoint")) | |
1499 | wx.EVT_MENU(frame, DebuggerService.TOGGLE_BREAKPOINT_ID, self.ProcessEvent) | |
1500 | wx.EVT_UPDATE_UI(frame, DebuggerService.TOGGLE_BREAKPOINT_ID, self.ProcessUpdateUIEvent) | |
1501 | ||
1502 | debuggerMenu.Append(DebuggerService.CLEAR_ALL_BREAKPOINTS, _("&Clear All Breakpoints"), _("Clear All Breakpoints")) | |
1503 | wx.EVT_MENU(frame, DebuggerService.CLEAR_ALL_BREAKPOINTS, self.ProcessEvent) | |
1504 | wx.EVT_UPDATE_UI(frame, DebuggerService.CLEAR_ALL_BREAKPOINTS, self.ProcessUpdateUIEvent) | |
1505 | ||
1506 | ||
1507 | viewMenuIndex = menuBar.FindMenu(_("&Project")) | |
1508 | menuBar.Insert(viewMenuIndex + 1, debuggerMenu, _("&Run")) | |
1509 | ||
1510 | return True | |
1511 | ||
1512 | ||
1513 | ||
1514 | #---------------------------------------------------------------------------- | |
1515 | # Event Processing Methods | |
1516 | #---------------------------------------------------------------------------- | |
1517 | ||
1518 | def ProcessEventBeforeWindows(self, event): | |
1519 | return False | |
1520 | ||
1521 | ||
1522 | def ProcessEvent(self, event): | |
1523 | if Service.Service.ProcessEvent(self, event): | |
1524 | return True | |
1525 | ||
1526 | an_id = event.GetId() | |
1527 | if an_id == DebuggerService.TOGGLE_BREAKPOINT_ID: | |
1528 | self.OnToggleBreakpoint(event) | |
1529 | return True | |
1530 | elif an_id == DebuggerService.CLEAR_ALL_BREAKPOINTS: | |
1531 | self.ClearAllBreakpoints() | |
1532 | return True | |
1533 | elif an_id == DebuggerService.RUN_ID: | |
1534 | self.OnRunProject(event) | |
1535 | return True | |
1536 | elif an_id == DebuggerService.DEBUG_ID: | |
1537 | self.OnDebugProject(event) | |
1538 | return True | |
1539 | return False | |
1540 | ||
1541 | def ProcessUpdateUIEvent(self, event): | |
1542 | if Service.Service.ProcessUpdateUIEvent(self, event): | |
1543 | return True | |
1544 | ||
1545 | an_id = event.GetId() | |
1546 | if an_id == DebuggerService.TOGGLE_BREAKPOINT_ID: | |
1547 | currentView = self.GetDocumentManager().GetCurrentView() | |
1548 | event.Enable(isinstance(currentView, PythonEditor.PythonView)) | |
1549 | return True | |
1550 | elif an_id == DebuggerService.CLEAR_ALL_BREAKPOINTS: | |
1551 | event.Enable(self.HasBreakpointsSet()) | |
1552 | return True | |
1553 | elif an_id == DebuggerService.RUN_ID: | |
1554 | event.Enable(self.HasAnyFiles()) | |
1555 | return True | |
1556 | elif an_id == DebuggerService.DEBUG_ID: | |
1557 | event.Enable(self.HasAnyFiles()) | |
1558 | return True | |
1559 | else: | |
1560 | return False | |
1561 | ||
1562 | #---------------------------------------------------------------------------- | |
1563 | # Class Methods | |
1564 | #---------------------------------------------------------------------------- | |
1565 | ||
1566 | def OnDebugProject(self, event): | |
1567 | if not Executor.GetPythonExecutablePath(): | |
1568 | return | |
1569 | if DebugCommandUI.DebuggerRunning(): | |
1570 | wx.MessageBox(_("A debugger is already running. Please shut down the other debugger first."), _("Debugger Running")) | |
1571 | return | |
1572 | self.ShowWindow(True) | |
1573 | projectService = wx.GetApp().GetService(ProjectEditor.ProjectService) | |
1574 | project = projectService.GetView().GetDocument() | |
1575 | dlg = CommandPropertiesDialog(self.GetView().GetFrame(), 'Debug Python File', projectService, None, pythonOnly=True, okButtonName="Debug", debugging=True) | |
1576 | if dlg.ShowModal() == wx.ID_OK: | |
1577 | fileToDebug, initialArgs, startIn, isPython, environment = dlg.GetSettings() | |
1578 | dlg.Destroy() | |
1579 | else: | |
1580 | dlg.Destroy() | |
1581 | return | |
1582 | self.PromptToSaveFiles() | |
1583 | ||
1584 | shortFile = os.path.basename(fileToDebug) | |
1585 | fileToDebug = DebuggerService.ExpandPath(fileToDebug) | |
1586 | try: | |
1587 | page = DebugCommandUI(Service.ServiceView.bottomTab, -1, str(fileToDebug), self) | |
1588 | count = Service.ServiceView.bottomTab.GetPageCount() | |
1589 | Service.ServiceView.bottomTab.AddPage(page, "Debugging: " + shortFile) | |
1590 | Service.ServiceView.bottomTab.SetSelection(count) | |
1591 | page.Execute(initialArgs, startIn, environment) | |
1592 | except: | |
1593 | pass | |
1594 | ||
1595 | def HasAnyFiles(self): | |
1596 | docs = wx.GetApp().GetDocumentManager().GetDocuments() | |
1597 | return len(docs) > 0 | |
1598 | ||
1599 | def PromptToSaveFiles(self, running=True): | |
1600 | filesModified = False | |
1601 | docs = wx.GetApp().GetDocumentManager().GetDocuments() | |
1602 | for doc in docs: | |
1603 | if doc.IsModified(): | |
1604 | filesModified = True | |
1605 | break | |
1606 | if filesModified: | |
1607 | frame = self.GetView().GetFrame() | |
1608 | if running: | |
1609 | yesNoMsg = wx.MessageDialog(frame, | |
1610 | _("Files have been modified.\nWould you like to save all files before running?"), | |
1611 | _("Run"), | |
1612 | wx.YES_NO | |
1613 | ) | |
1614 | else: | |
1615 | yesNoMsg = wx.MessageDialog(frame, | |
1616 | _("Files have been modified.\nWould you like to save all files before debugging?"), | |
1617 | _("Debug"), | |
1618 | wx.YES_NO | |
1619 | ) | |
1620 | if yesNoMsg.ShowModal() == wx.ID_YES: | |
1621 | docs = wx.GetApp().GetDocumentManager().GetDocuments() | |
1622 | for doc in docs: | |
1623 | doc.Save() | |
1624 | ||
1625 | def OnExit(self): | |
1626 | DebugCommandUI.ShutdownAllDebuggers() | |
1627 | ||
1628 | def OnRunProject(self, event): | |
1629 | if not Executor.GetPythonExecutablePath(): | |
1630 | return | |
1631 | projectService = wx.GetApp().GetService(ProjectEditor.ProjectService) | |
1632 | project = projectService.GetView().GetDocument() | |
1633 | dlg = CommandPropertiesDialog(self.GetView().GetFrame(), 'Run', projectService, None) | |
1634 | if dlg.ShowModal() == wx.ID_OK: | |
1635 | fileToRun, initialArgs, startIn, isPython, environment = dlg.GetSettings() | |
1636 | ||
1637 | ||
1638 | dlg.Destroy() | |
1639 | else: | |
1640 | dlg.Destroy() | |
1641 | return | |
1642 | self.PromptToSaveFiles() | |
1643 | # This will need to change when we can run more than .py and .bpel files. | |
1644 | if not isPython: | |
1645 | projectService.RunProcessModel(fileToRun) | |
1646 | return | |
1647 | ||
1648 | self.ShowWindow(True) | |
1649 | shortFile = os.path.basename(fileToRun) | |
1650 | page = RunCommandUI(Service.ServiceView.bottomTab, -1, str(fileToRun)) | |
1651 | count = Service.ServiceView.bottomTab.GetPageCount() | |
1652 | Service.ServiceView.bottomTab.AddPage(page, "Running: " + shortFile) | |
1653 | Service.ServiceView.bottomTab.SetSelection(count) | |
1654 | page.Execute(initialArgs, startIn, environment) | |
1655 | ||
1656 | def OnToggleBreakpoint(self, event, line=-1, fileName=None): | |
1657 | if not fileName: | |
1658 | view = wx.GetApp().GetDocumentManager().GetCurrentView() | |
1659 | # Test to make sure we aren't the project view. | |
1660 | if not hasattr(view, 'MarkerExists'): | |
1661 | return | |
1662 | fileName = wx.GetApp().GetDocumentManager().GetCurrentDocument().GetFilename() | |
1663 | if line < 0: | |
1664 | line = view.GetCtrl().GetCurrentLine() | |
1665 | if self.BreakpointSet(fileName, line + 1): | |
1666 | self.ClearBreak(fileName, line + 1) | |
1667 | else: | |
1668 | self.SetBreak(fileName, line + 1) | |
1669 | # Now refresh all the markers icons in all the open views. | |
1670 | self.ClearAllBreakpointMarkers() | |
1671 | self.SetAllBreakpointMarkers() | |
1672 | ||
1673 | def SilentToggleBreakpoint(self, fileName, line): | |
1674 | found = False | |
1675 | for lineNumber in self.GetBreakpointList(fileName): | |
1676 | if int(lineNumber) == int(line): | |
1677 | found = True | |
1678 | break | |
1679 | if found: | |
1680 | self.SetBreak(fileName, line) | |
1681 | else: | |
1682 | self.ClearBreak(fileName, line) | |
1683 | ||
1684 | def SetBreak(self, fileName, line): | |
1685 | expandedName = DebuggerService.ExpandPath(fileName) | |
1686 | if not self._masterBPDict.has_key(expandedName): | |
1687 | self._masterBPDict[expandedName] = [line] | |
1688 | else: | |
1689 | self._masterBPDict[expandedName] += [line] | |
1690 | # If we're already debugging, pass this bp off to the DebuggerCallback | |
1691 | self.NotifyDebuggersOfBreakpointChange() | |
1692 | ||
1693 | def NotifyDebuggersOfBreakpointChange(self): | |
1694 | DebugCommandUI.NotifyDebuggersOfBreakpointChange() | |
1695 | ||
1696 | def GetBreakpointList(self, fileName): | |
1697 | expandedName = DebuggerService.ExpandPath(fileName) | |
1698 | if not self._masterBPDict.has_key(expandedName): | |
1699 | return [] | |
1700 | else: | |
1701 | return self._masterBPDict[expandedName] | |
1702 | ||
1703 | def BreakpointSet(self, fileName, line): | |
1704 | expandedName = DebuggerService.ExpandPath(fileName) | |
1705 | if not self._masterBPDict.has_key(expandedName): | |
1706 | return False | |
1707 | else: | |
1708 | newList = [] | |
1709 | for number in self._masterBPDict[expandedName]: | |
1710 | if(int(number) == int(line)): | |
1711 | return True | |
1712 | return False | |
1713 | ||
1714 | def ClearBreak(self, fileName, line): | |
1715 | expandedName = DebuggerService.ExpandPath(fileName) | |
1716 | if not self._masterBPDict.has_key(expandedName): | |
1717 | print "In ClearBreak: no key" | |
1718 | return | |
1719 | else: | |
1720 | newList = [] | |
1721 | for number in self._masterBPDict[expandedName]: | |
1722 | if(int(number) != int(line)): | |
1723 | newList.append(number) | |
1724 | self._masterBPDict[expandedName] = newList | |
1725 | self.NotifyDebuggersOfBreakpointChange() | |
1726 | ||
1727 | def HasBreakpointsSet(self): | |
1728 | for key, value in self._masterBPDict.items(): | |
1729 | if len(value) > 0: | |
1730 | return True | |
1731 | return False | |
1732 | ||
1733 | def ClearAllBreakpoints(self): | |
1734 | self._masterBPDict = {} | |
1735 | self.NotifyDebuggersOfBreakpointChange() | |
1736 | self.ClearAllBreakpointMarkers() | |
1737 | ||
1738 | def ClearAllBreakpointMarkers(self): | |
1739 | openDocs = wx.GetApp().GetDocumentManager().GetDocuments() | |
1740 | for openDoc in openDocs: | |
1741 | if(isinstance(openDoc.GetFirstView(), CodeEditor.CodeView)): | |
1742 | openDoc.GetFirstView().MarkerDeleteAll(CodeEditor.CodeCtrl.BREAKPOINT_MARKER_NUM) | |
1743 | ||
1744 | def GetMasterBreakpointDict(self): | |
1745 | return self._masterBPDict | |
1746 | ||
1747 | def SetAllBreakpointMarkers(self): | |
1748 | openDocs = wx.GetApp().GetDocumentManager().GetDocuments() | |
1749 | for openDoc in openDocs: | |
1750 | if(isinstance(openDoc.GetFirstView(), CodeEditor.CodeView)): | |
1751 | self.SetCurrentBreakpointMarkers(openDoc.GetFirstView()) | |
1752 | ||
1753 | def SetCurrentBreakpointMarkers(self, view): | |
1754 | if isinstance(view, CodeEditor.CodeView) and hasattr(view, 'GetDocument'): | |
1755 | view.MarkerDeleteAll(CodeEditor.CodeCtrl.BREAKPOINT_MARKER_NUM) | |
1756 | for linenum in self.GetBreakpointList(view.GetDocument().GetFilename()): | |
1757 | view.MarkerAdd(lineNum=int(linenum) - 1, marker_index=CodeEditor.CodeCtrl.BREAKPOINT_MARKER_NUM) | |
1758 | ||
1759 | class DebuggerOptionsPanel(wx.Panel): | |
1760 | ||
1761 | ||
1762 | def __init__(self, parent, id): | |
1763 | wx.Panel.__init__(self, parent, id) | |
1764 | SPACE = 10 | |
1765 | config = wx.ConfigBase_Get() | |
1766 | localHostStaticText = wx.StaticText(self, -1, _("Local Host Name:")) | |
1767 | self._LocalHostTextCtrl = wx.TextCtrl(self, -1, config.Read("DebuggerHostName", DEFAULT_HOST), size = (150, -1)) | |
1768 | portNumberStaticText = wx.StaticText(self, -1, _("Port Range:")) | |
1769 | dashStaticText = wx.StaticText(self, -1, _("through to")) | |
1770 | startingPort=config.ReadInt("DebuggerStartingPort", DEFAULT_PORT) | |
1771 | self._PortNumberTextCtrl = wx.lib.intctrl.IntCtrl(self, -1, startingPort, size = (50, -1)) | |
1772 | self._PortNumberTextCtrl.SetMin(1)#What are real values? | |
1773 | self._PortNumberTextCtrl.SetMax(65514) #What are real values? | |
1774 | self.Bind(wx.lib.intctrl.EVT_INT, self.MinPortChange, self._PortNumberTextCtrl) | |
1775 | ||
1776 | self._EndPortNumberTextCtrl = wx.lib.intctrl.IntCtrl(self, -1, startingPort + PORT_COUNT, size = (50, -1)) | |
1777 | self._EndPortNumberTextCtrl.SetMin(22)#What are real values? | |
1778 | self._EndPortNumberTextCtrl.SetMax(65535)#What are real values? | |
1779 | self._EndPortNumberTextCtrl.Enable( False ) | |
1780 | debuggerPanelBorderSizer = wx.BoxSizer(wx.VERTICAL) | |
1781 | debuggerPanelSizer = wx.GridBagSizer(hgap = 5, vgap = 5) | |
1782 | debuggerPanelSizer.Add( localHostStaticText, (0,0), flag=wx.ALIGN_CENTER_VERTICAL|wx.ALIGN_LEFT) | |
1783 | debuggerPanelSizer.Add( self._LocalHostTextCtrl, (0,1), (1,3), flag=wx.EXPAND|wx.ALIGN_CENTER) | |
1784 | debuggerPanelSizer.Add( portNumberStaticText, (1,0), flag=wx.ALIGN_LEFT|wx.ALIGN_CENTER_VERTICAL) | |
1785 | debuggerPanelSizer.Add( self._PortNumberTextCtrl, (1,1), flag=wx.ALIGN_CENTER) | |
1786 | debuggerPanelSizer.Add( dashStaticText, (1,2), flag=wx.ALIGN_CENTER) | |
1787 | debuggerPanelSizer.Add( self._EndPortNumberTextCtrl, (1,3), flag=wx.ALIGN_CENTER) | |
1788 | FLUSH_PORTS_ID = wx.NewId() | |
1789 | self._flushPortsButton = wx.Button(self, FLUSH_PORTS_ID, "Reset Port List") | |
1790 | wx.EVT_BUTTON(parent, FLUSH_PORTS_ID, self.FlushPorts) | |
1791 | debuggerPanelSizer.Add(self._flushPortsButton, (2,2), (1,2), flag=wx.ALIGN_RIGHT) | |
1792 | ||
1793 | debuggerPanelBorderSizer.Add(debuggerPanelSizer, 0, wx.ALL, SPACE) | |
1794 | self.SetSizer(debuggerPanelBorderSizer) | |
1795 | self.Layout() | |
1796 | parent.AddPage(self, _("Debugger")) | |
1797 | ||
1798 | def FlushPorts(self, event): | |
1799 | if self._PortNumberTextCtrl.IsInBounds(): | |
1800 | config = wx.ConfigBase_Get() | |
1801 | config.WriteInt("DebuggerStartingPort", self._PortNumberTextCtrl.GetValue()) | |
1802 | DebugCommandUI.NewPortRange() | |
1803 | else: | |
1804 | wx.MessageBox(_("The starting port is not valid. Please change the value and try again.", "Invalid Starting Port Number")) | |
1805 | ||
1806 | def MinPortChange(self, event): | |
1807 | self._EndPortNumberTextCtrl.Enable( True ) | |
1808 | self._EndPortNumberTextCtrl.SetValue( self._PortNumberTextCtrl.GetValue() + PORT_COUNT) | |
1809 | self._EndPortNumberTextCtrl.Enable( False ) | |
1810 | ||
1811 | def OnOK(self, optionsDialog): | |
1812 | config = wx.ConfigBase_Get() | |
1813 | config.Write("DebuggerHostName", self._LocalHostTextCtrl.GetValue()) | |
1814 | if self._PortNumberTextCtrl.IsInBounds(): | |
1815 | config.WriteInt("DebuggerStartingPort", self._PortNumberTextCtrl.GetValue()) | |
1816 | ||
1817 | class CommandPropertiesDialog(wx.Dialog): | |
1818 | ||
1819 | def __init__(self, parent, title, projectService, currentProjectDocument, pythonOnly=False, okButtonName="Run", debugging=False): | |
1820 | if _WINDOWS: | |
1821 | wx.Dialog.__init__(self, parent, -1, title) | |
1822 | else: | |
1823 | wx.Dialog.__init__(self, parent, -1, title, size=(390,270)) | |
1824 | self._projService = projectService | |
1825 | self._pmext = None | |
1826 | self._pyext = None | |
1827 | for template in self._projService.GetDocumentManager().GetTemplates(): | |
1828 | if not ACTIVEGRID_BASE_IDE and template.GetDocumentType() == ProcessModelEditor.ProcessModelDocument: | |
1829 | self._pmext = template.GetDefaultExtension() | |
1830 | if template.GetDocumentType() == PythonEditor.PythonDocument: | |
1831 | self._pyext = template.GetDefaultExtension() | |
1832 | self._pythonOnly = pythonOnly | |
1833 | ||
1834 | self._currentProj = currentProjectDocument | |
1835 | projStaticText = wx.StaticText(self, -1, _("Project:")) | |
1836 | fileStaticText = wx.StaticText(self, -1, _("File:")) | |
1837 | argsStaticText = wx.StaticText(self, -1, _("Arguments:")) | |
1838 | startInStaticText = wx.StaticText(self, -1, _("Start in:")) | |
1839 | pythonPathStaticText = wx.StaticText(self, -1, _("PYTHONPATH:")) | |
1840 | postpendStaticText = _("Postpend win32api path") | |
1841 | cpPanelBorderSizer = wx.BoxSizer(wx.VERTICAL) | |
1842 | self._projectNameList, self._projectDocumentList, selectedIndex = self.GetProjectList() | |
1843 | self._projList = wx.Choice(self, -1, (200,-1), choices=self._projectNameList) | |
1844 | self.Bind(wx.EVT_CHOICE, self.EvtListBox, self._projList) | |
1845 | HALF_SPACE = 5 | |
1846 | flexGridSizer = wx.FlexGridSizer(cols = 3, vgap = 10, hgap = 10) | |
1847 | ||
1848 | flexGridSizer.Add(projStaticText, 0, flag=wx.ALIGN_CENTER_VERTICAL|wx.ALIGN_LEFT) | |
1849 | flexGridSizer.Add(self._projList, 1, flag=wx.EXPAND) | |
1850 | flexGridSizer.Add(wx.StaticText(parent, -1, ""), 0) | |
1851 | ||
1852 | flexGridSizer.Add(fileStaticText, 0, flag=wx.ALIGN_CENTER_VERTICAL|wx.ALIGN_LEFT) | |
1853 | self._fileList = wx.Choice(self, -1, (200,-1)) | |
1854 | self.Bind(wx.EVT_CHOICE, self.OnFileSelected, self._fileList) | |
1855 | flexGridSizer.Add(self._fileList, 1, flag=wx.EXPAND) | |
1856 | flexGridSizer.Add(wx.StaticText(parent, -1, ""), 0) | |
1857 | ||
1858 | config = wx.ConfigBase_Get() | |
1859 | self._lastArguments = config.Read("LastRunArguments") | |
1860 | self._argsEntry = wx.TextCtrl(self, -1, str(self._lastArguments)) | |
1861 | self._argsEntry.SetToolTipString(str(self._lastArguments)) | |
1862 | def TextChanged(event): | |
1863 | self._argsEntry.SetToolTipString(event.GetString()) | |
1864 | self.Bind(wx.EVT_TEXT, TextChanged, self._argsEntry) | |
1865 | ||
1866 | flexGridSizer.Add(argsStaticText, 0, flag=wx.ALIGN_CENTER_VERTICAL|wx.ALIGN_LEFT) | |
1867 | flexGridSizer.Add(self._argsEntry, 1, flag=wx.EXPAND) | |
1868 | flexGridSizer.Add(wx.StaticText(parent, -1, ""), 0) | |
1869 | ||
1870 | flexGridSizer.Add(startInStaticText, 0, flag=wx.ALIGN_CENTER_VERTICAL|wx.ALIGN_LEFT) | |
1871 | self._lastStartIn = config.Read("LastRunStartIn") | |
1872 | if not self._lastStartIn: | |
1873 | self._lastStartIn = str(os.getcwd()) | |
1874 | self._startEntry = wx.TextCtrl(self, -1, self._lastStartIn) | |
1875 | self._startEntry.SetToolTipString(self._lastStartIn) | |
1876 | def TextChanged2(event): | |
1877 | self._startEntry.SetToolTipString(event.GetString()) | |
1878 | self.Bind(wx.EVT_TEXT, TextChanged2, self._startEntry) | |
1879 | ||
1880 | flexGridSizer.Add(self._startEntry, 1, wx.EXPAND) | |
1881 | self._findDir = wx.Button(self, -1, _("Browse..."), size=(60,-1)) | |
1882 | self.Bind(wx.EVT_BUTTON, self.OnFindDirClick, self._findDir) | |
1883 | flexGridSizer.Add(self._findDir, 0, wx.RIGHT, 10) | |
1884 | ||
1885 | flexGridSizer.Add(pythonPathStaticText, 0, flag=wx.ALIGN_CENTER_VERTICAL|wx.ALIGN_LEFT) | |
1886 | if os.environ.has_key('PYTHONPATH'): | |
1887 | startval = os.environ['PYTHONPATH'] | |
1888 | else: | |
1889 | startval = "" | |
1890 | self._lastPythonPath = config.Read("LastPythonPath", startval) | |
1891 | self._pythonPathEntry = wx.TextCtrl(self, -1, self._lastPythonPath) | |
1892 | self._pythonPathEntry.SetToolTipString(self._lastPythonPath) | |
1893 | flexGridSizer.Add(self._pythonPathEntry, 1, wx.EXPAND) | |
1894 | flexGridSizer.Add(wx.StaticText(parent, -1, ""), 0) | |
1895 | flexGridSizer.Add(wx.StaticText(parent, -1, ""), 0) | |
1896 | if debugging: | |
1897 | self._postpendCheckBox = wx.CheckBox(self, -1, postpendStaticText) | |
1898 | checked = bool(config.ReadInt("PythonPathPostpend", 1)) | |
1899 | self._postpendCheckBox.SetValue(checked) | |
1900 | flexGridSizer.Add(self._postpendCheckBox, 1, wx.EXPAND) | |
1901 | flexGridSizer.Add(wx.StaticText(parent, -1, ""), 0) | |
1902 | cpPanelBorderSizer.Add(flexGridSizer, 0, wx.ALL, 10) | |
1903 | ||
1904 | box = wx.BoxSizer(wx.HORIZONTAL) | |
1905 | self._okButton = wx.Button(self, wx.ID_OK, okButtonName, size=(75,-1)) | |
1906 | self._okButton.SetDefault() | |
1907 | self._okButton.SetHelpText(_("The ") + okButtonName + _(" button completes the dialog")) | |
1908 | box.Add(self._okButton, 0, wx.ALIGN_RIGHT|wx.ALL, 5) | |
1909 | self.Bind(wx.EVT_BUTTON, self.OnOKClick, self._okButton) | |
1910 | btn = wx.Button(self, wx.ID_CANCEL, _("Cancel"), size=(75,-1)) | |
1911 | btn.SetHelpText(_("The Cancel button cancels the dialog.")) | |
1912 | box.Add(btn, 0, wx.ALIGN_RIGHT|wx.ALL, 5) | |
1913 | cpPanelBorderSizer.Add(box, 0, wx.ALIGN_RIGHT|wx.BOTTOM, 5) | |
1914 | ||
1915 | self.SetSizer(cpPanelBorderSizer) | |
1916 | if _WINDOWS: | |
1917 | self.GetSizer().Fit(self) | |
1918 | ||
1919 | self.Layout() | |
1920 | ||
1921 | # Set up selections based on last values used. | |
1922 | self._fileNameList = None | |
1923 | self._selectedFileIndex = 0 | |
1924 | lastProject = config.Read("LastRunProject") | |
1925 | lastFile = config.Read("LastRunFile") | |
1926 | ||
1927 | if lastProject in self._projectNameList: | |
1928 | selectedIndex = self._projectNameList.index(lastProject) | |
1929 | elif selectedIndex < 0: | |
1930 | selectedIndex = 0 | |
1931 | self._projList.Select(selectedIndex) | |
1932 | self._selectedProjectIndex = selectedIndex | |
1933 | self._selectedProjectDocument = self._projectDocumentList[selectedIndex] | |
1934 | self.PopulateFileList(self._selectedProjectDocument, lastFile) | |
1935 | ||
1936 | def OnOKClick(self, event): | |
1937 | startIn = self._startEntry.GetValue() | |
1938 | fileToRun = self._fileList.GetStringSelection() | |
1939 | if not fileToRun: | |
1940 | wx.MessageBox(_("You must select a file to proceed. Note that not all projects have files that can be run or debugged.")) | |
1941 | return | |
1942 | isPython = fileToRun.endswith(self._pyext) | |
1943 | if isPython and not os.path.exists(startIn): | |
1944 | wx.MessageBox(_("Starting directory does not exist. Please change this value.")) | |
1945 | return | |
1946 | config = wx.ConfigBase_Get() | |
1947 | config.Write("LastRunProject", self._projectNameList[self._selectedProjectIndex]) | |
1948 | config.Write("LastRunFile", fileToRun) | |
1949 | # Don't update the arguments or starting directory unless we're runing python. | |
1950 | if isPython: | |
1951 | config.Write("LastRunArguments", self._argsEntry.GetValue()) | |
1952 | config.Write("LastRunStartIn", self._startEntry.GetValue()) | |
1953 | config.Write("LastPythonPath",self._pythonPathEntry.GetValue()) | |
1954 | if hasattr(self, "_postpendCheckBox"): | |
1955 | config.WriteInt("PythonPathPostpend", int(self._postpendCheckBox.GetValue())) | |
1956 | ||
1957 | self.EndModal(wx.ID_OK) | |
1958 | ||
1959 | def GetSettings(self): | |
1960 | filename = self._fileNameList[self._selectedFileIndex] | |
1961 | args = self._argsEntry.GetValue() | |
1962 | startIn = self._startEntry.GetValue() | |
1963 | isPython = filename.endswith(self._pyext) | |
1964 | env = os.environ | |
1965 | if hasattr(self, "_postpendCheckBox"): | |
1966 | postpend = self._postpendCheckBox.GetValue() | |
1967 | else: | |
1968 | postpend = False | |
1969 | if postpend: | |
1970 | env['PYTHONPATH'] = self._pythonPathEntry.GetValue() + os.pathsep + os.path.join(os.getcwd(), "3rdparty", "pywin32") | |
1971 | else: | |
1972 | env['PYTHONPATH'] = self._pythonPathEntry.GetValue() | |
1973 | ||
1974 | return filename, args, startIn, isPython, env | |
1975 | ||
1976 | def OnFileSelected(self, event): | |
1977 | self._selectedFileIndex = self._fileList.GetSelection() | |
1978 | self.EnableForFileType(event.GetString()) | |
1979 | ||
1980 | def EnableForFileType(self, fileName): | |
1981 | show = fileName.endswith(self._pyext) | |
1982 | self._startEntry.Enable(show) | |
1983 | self._findDir.Enable(show) | |
1984 | self._argsEntry.Enable(show) | |
1985 | ||
1986 | if not show: | |
1987 | self._lastStartIn = self._startEntry.GetValue() | |
1988 | self._startEntry.SetValue("") | |
1989 | self._lastArguments = self._argsEntry.GetValue() | |
1990 | self._argsEntry.SetValue("") | |
1991 | else: | |
1992 | self._startEntry.SetValue(self._lastStartIn) | |
1993 | self._argsEntry.SetValue(self._lastArguments) | |
1994 | ||
1995 | ||
1996 | ||
1997 | def OnFindDirClick(self, event): | |
1998 | dlg = wx.DirDialog(self, "Choose a starting directory:", self._startEntry.GetValue(), | |
1999 | style=wx.DD_DEFAULT_STYLE|wx.DD_NEW_DIR_BUTTON) | |
2000 | ||
2001 | if dlg.ShowModal() == wx.ID_OK: | |
2002 | self._startEntry.SetValue(dlg.GetPath()) | |
2003 | ||
2004 | dlg.Destroy() | |
2005 | ||
2006 | def EvtListBox(self, event): | |
2007 | if event.GetString(): | |
2008 | index = self._projectNameList.index(event.GetString()) | |
2009 | self._selectedProjectDocument = self._projectDocumentList[index] | |
2010 | self._selectedProjectIndex = index | |
2011 | self.PopulateFileList(self._selectedProjectDocument) | |
2012 | ||
2013 | def FilterFileList(self, list): | |
2014 | if self._pythonOnly: | |
2015 | files = filter(lambda f: f.endswith(self._pyext), list) | |
2016 | else: | |
2017 | files = filter(lambda f: (self._pmext and f.endswith(self._pmext)) or f.endswith(self._pyext), list) | |
2018 | return files | |
2019 | ||
2020 | def PopulateFileList(self, project, shortNameToSelect=None): | |
2021 | self._fileNameList = self.FilterFileList(project.GetFiles()[:]) | |
2022 | self._fileList.Clear() | |
2023 | if not self._fileNameList: | |
2024 | return | |
2025 | self._fileNameList.sort(lambda a, b: cmp(os.path.basename(a).lower(), os.path.basename(b).lower())) | |
2026 | strings = map(lambda file: os.path.basename(file), self._fileNameList) | |
2027 | for index in range(0, len(strings)): | |
2028 | if shortNameToSelect == strings[index]: | |
2029 | self._selectedFileIndex = index | |
2030 | break | |
2031 | self._fileList.Hide() | |
2032 | self._fileList.AppendItems(strings) | |
2033 | self._fileList.Show() | |
2034 | if self._selectedFileIndex not in range(0, len(strings)) : self._selectedFileIndex = 0 | |
2035 | self._fileList.SetSelection(self._selectedFileIndex) | |
2036 | self.EnableForFileType(strings[self._selectedFileIndex]) | |
2037 | ||
2038 | def GetProjectList(self): | |
2039 | docList = [] | |
2040 | nameList = [] | |
2041 | found = False | |
2042 | index = -1 | |
2043 | count = 0 | |
2044 | for document in self._projService.GetDocumentManager().GetDocuments(): | |
2045 | if document.GetDocumentTemplate().GetDocumentType() == ProjectEditor.ProjectDocument and len(document.GetFiles()): | |
2046 | docList.append(document) | |
2047 | nameList.append(os.path.basename(document.GetFilename())) | |
2048 | if document == self._currentProj: | |
2049 | found = True | |
2050 | index = count | |
2051 | count += 1 | |
2052 | return nameList, docList, index | |
2053 | #---------------------------------------------------------------------- | |
2054 | from wx import ImageFromStream, BitmapFromImage | |
2055 | from wx import EmptyIcon | |
2056 | import cStringIO | |
2057 | #---------------------------------------------------------------------- | |
2058 | def getBreakData(): | |
2059 | return \ | |
2060 | '\x89PNG\r\n\x1a\n\x00\x00\x00\rIHDR\x00\x00\x00\x10\x00\x00\x00\x11\x08\x06\ | |
2061 | \x00\x00\x00\xd4\xaf,\xc4\x00\x00\x00\x04sBIT\x08\x08\x08\x08|\x08d\x88\x00\ | |
2062 | \x00\x00\x97IDAT8\x8d\xbdSA\x0e\xc3 \x0c\x8b\x13\xfe=\x1e^\xe2\x1dF\xbb\x8c\ | |
2063 | \xd2\x0c\xa9\xda,q\x88\x05\x8e\x1d\x00P\x93;\xd0[\xa7W\x04\xe8\x8d\x0f\xdfxU\ | |
2064 | c%\x02\xbd\xbd\x05HQ+Xv\xb0\xa3VN\xf9\xd4\x01\xbd\x11j\x18\x1d\x00\x10\xa8AD\ | |
2065 | \xa4\xa4\xd6_\x9b\x19\xbb\x03\xd8c|\x8f\x00\xe0\x93\xa8g>\x15 C\xee:\xe7\x8f\ | |
2066 | \x08\xdesj\xcf\xe6\xde(\xddn\xec\x18f0w\xe0m;\x8d\x9b\xe4\xb1\xd4\n\xa6\x0e2\ | |
2067 | \xc4{\x1f\xeb\xdf?\xe5\xff\t\xa8\x1a$\x0cg\xac\xaf\xb0\xf4\x992<\x01\xec\xa0\ | |
2068 | U9V\xf9\x18\xc8\x00\x00\x00\x00IEND\xaeB`\x82' | |
2069 | ||
2070 | def getBreakBitmap(): | |
2071 | return BitmapFromImage(getBreakImage()) | |
2072 | ||
2073 | def getBreakImage(): | |
2074 | stream = cStringIO.StringIO(getBreakData()) | |
2075 | return ImageFromStream(stream) | |
2076 | ||
2077 | def getBreakIcon(): | |
2078 | icon = EmptyIcon() | |
2079 | icon.CopyFromBitmap(getBreakBitmap()) | |
2080 | return icon | |
2081 | ||
2082 | #---------------------------------------------------------------------- | |
2083 | def getClearOutputData(): | |
2084 | return \ | |
2085 | '\x89PNG\r\n\x1a\n\x00\x00\x00\rIHDR\x00\x00\x00\x10\x00\x00\x00\x10\x08\x06\ | |
2086 | \x00\x00\x00\x1f\xf3\xffa\x00\x00\x00\x04sBIT\x08\x08\x08\x08|\x08d\x88\x00\ | |
2087 | \x00\x00\xb4IDAT8\x8d\xa5\x92\xdd\r\x03!\x0c\x83\xbf\xa0n\xd4\x9d\xda5\xb81\ | |
2088 | \xbaS\xbb\x12\xee\x03?\xe5\x08\xe5N\xba\xbc Db\xec\xd8p\xb1l\xb8\xa7\x83\xfe\ | |
2089 | \xb0\x02H\x92F\xc0_\xa3\x99$\x99\x99\xedznc\xe36\x81\x88\x98"\xb2\x02\xa2\ | |
2090 | \x1e\xc4Q\x9aUD\x161\xcd\xde\x1c\x83\x15\x084)\x8d\xc5)\x06\xab\xaaZ\x92\xee\ | |
2091 | \xce\x11W\xdbGD\x0cIT\x06\xe7\x00\xdeY\xfe\xcc\x89\x06\xf0\xf2\x99\x00\xe0\ | |
2092 | \x91\x7f\xab\x83\xed\xa4\xc8\xafK\x0c\xcf\x92\x83\x99\x8d\xe3p\xef\xe4\xa1\ | |
2093 | \x0b\xe57j\xc8:\x06\t\x08\x87.H\xb2n\xa8\xc9\xa9\x12vQ\xfeG"\xe3\xacw\x00\ | |
2094 | \x10$M\xd3\x86_\xf0\xe5\xfc\xb4\xfa\x02\xcb\x13j\x10\xc5\xd7\x92D\x00\x00\ | |
2095 | \x00\x00IEND\xaeB`\x82' | |
2096 | ||
2097 | def getClearOutputBitmap(): | |
2098 | return BitmapFromImage(getClearOutputImage()) | |
2099 | ||
2100 | def getClearOutputImage(): | |
2101 | stream = cStringIO.StringIO(getClearOutputData()) | |
2102 | return ImageFromStream(stream) | |
2103 | ||
2104 | def getClearOutputIcon(): | |
2105 | icon = EmptyIcon() | |
2106 | icon.CopyFromBitmap(getClearOutputBitmap()) | |
2107 | return icon | |
2108 | ||
2109 | #---------------------------------------------------------------------- | |
2110 | def getCloseData(): | |
2111 | return \ | |
2112 | '\x89PNG\r\n\x1a\n\x00\x00\x00\rIHDR\x00\x00\x00\x12\x00\x00\x00\x12\x08\x06\ | |
2113 | \x00\x00\x00V\xce\x8eW\x00\x00\x00\x04sBIT\x08\x08\x08\x08|\x08d\x88\x00\x00\ | |
2114 | \x00\x86IDAT8\x8d\xed\x90\xb1\r\x80 \x10E_"c\xd80\x02-\x138\x87;8\x8f\x8d\ | |
2115 | \x8b\xb0\x02\xa5\xad\rS\x88\xcd\x11) \x82\xb6\xbe\xea\xf2\xc9\xbd\xfc\x03~\ | |
2116 | \xdeb\x81\xb9\x90\xabJ^%\x00N\x849\x0e\xf0\x85\xbc\x8a\x12YZR2\xc7\x1eIB\xcb\ | |
2117 | \xb2\xcb\x9at\x9d\x95c\xa5Y|\x92\x0c\x0f\xa2\r8e\x1e\x81\x1d8z\xdb8\xee?Ig\ | |
2118 | \xfa\xb7\x92)\xcb\xacd\x01XZ$QD\xba\xf0\xa6\x80\xd5\x18cZ\x1b\x95$?\x1f\xb9\ | |
2119 | \x00\x1d\x94\x1e*e_\x8a~\x00\x00\x00\x00IEND\xaeB`\x82' | |
2120 | ||
2121 | def getCloseBitmap(): | |
2122 | return BitmapFromImage(getCloseImage()) | |
2123 | ||
2124 | def getCloseImage(): | |
2125 | stream = cStringIO.StringIO(getCloseData()) | |
2126 | return ImageFromStream(stream) | |
2127 | ||
2128 | def getCloseIcon(): | |
2129 | icon = EmptyIcon() | |
2130 | icon.CopyFromBitmap(getCloseBitmap()) | |
2131 | return icon | |
2132 | ||
2133 | #---------------------------------------------------------------------- | |
2134 | def getContinueData(): | |
2135 | return \ | |
2136 | '\x89PNG\r\n\x1a\n\x00\x00\x00\rIHDR\x00\x00\x00\x10\x00\x00\x00\x10\x08\x06\ | |
2137 | \x00\x00\x00\x1f\xf3\xffa\x00\x00\x00\x04sBIT\x08\x08\x08\x08|\x08d\x88\x00\ | |
2138 | \x00\x00\x9eIDAT8\x8d\xbd\x92\xd1\r\x83@\x0cC\xed\x8a\x11X\xac\x1b\xddF,\xd6\ | |
2139 | \x11\x90\xdc\x0f\x9a\x93s)\x14Z\xa9\x91\x10\n\x97\xbc\xd89\x80?\x84\x1a\xa4\ | |
2140 | \x83\xfc\x1c$\x1e)7\xdf<Y0\xaf\x0b\xe6\xf5\x1d\xa1\xb5\x13C\x03 !\xaa\xfd\ | |
2141 | \xed\n:mr\xc0\x1d\x8f\xc9\x9a!\t$\xe5\xd3I\xe2\xe5B$\x99\x00[\x01\xe8\xc5\ | |
2142 | \xd9G\xfaN`\xd8\x81I\xed\x8c\xb19\x94\x8d\xcbL\x00;t\xcf\x9fwPh\xdb\x0e\xe8\ | |
2143 | \xd3,\x17\x8b\xc7\x9d\xbb>\x8a \xec5\x94\tc\xc4\x12\xab\x94\xeb\x7fkWr\xc9B%\ | |
2144 | \xfc\xd2\xfcM<\x01\xf6tn\x12O3c\xe6\x00\x00\x00\x00IEND\xaeB`\x82' | |
2145 | ||
2146 | def getContinueBitmap(): | |
2147 | return BitmapFromImage(getContinueImage()) | |
2148 | ||
2149 | def getContinueImage(): | |
2150 | stream = cStringIO.StringIO(getContinueData()) | |
2151 | return ImageFromStream(stream) | |
2152 | ||
2153 | def getContinueIcon(): | |
2154 | icon = EmptyIcon() | |
2155 | icon.CopyFromBitmap(getContinueBitmap()) | |
2156 | return icon | |
2157 | ||
2158 | #---------------------------------------------------------------------- | |
2159 | def getNextData(): | |
2160 | return \ | |
2161 | '\x89PNG\r\n\x1a\n\x00\x00\x00\rIHDR\x00\x00\x00\x10\x00\x00\x00\x10\x08\x06\ | |
2162 | \x00\x00\x00\x1f\xf3\xffa\x00\x00\x00\x04sBIT\x08\x08\x08\x08|\x08d\x88\x00\ | |
2163 | \x00\x00\x8eIDAT8\x8d\xa5SA\x12\xc4 \x08K\xb0\xff\xde\xe9\xbf\xb7\xa6\x87\ | |
2164 | \x1d:\xba\xa2tZn(\x84`"i\x05obk\x13\xd5CmN+\xcc\x00l\xd6\x0c\x00\xf5\xf8\x0e\ | |
2165 | gK\x06\x00 \xa5=k\x00\x00\xb0\xb2]\xd4?5f\xb1\xdb\xaf\xc6\xa2\xcb\xa8\xf0?\ | |
2166 | \x1c\x98\xae\x82\xbf\x81\xa4\x8eA\x16\xe1\n\xd1\xa4\x19\xb3\xe9\n\xce\xe8\ | |
2167 | \xf1\n\x9eg^\x18\x18\x90\xec<\x11\xf9#\x04XMZ\x19\xaac@+\x94\xd4\x99)SeP\xa1\ | |
2168 | )\xd6\x1dI\xe7*\xdc\xf4\x03\xdf~\xe7\x13T^Q?:X\x19d\x00\x00\x00\x00IEND\xaeB\ | |
2169 | `\x82' | |
2170 | ||
2171 | def getNextBitmap(): | |
2172 | return BitmapFromImage(getNextImage()) | |
2173 | ||
2174 | def getNextImage(): | |
2175 | stream = cStringIO.StringIO(getNextData()) | |
2176 | return ImageFromStream(stream) | |
2177 | ||
2178 | def getNextIcon(): | |
2179 | icon = EmptyIcon() | |
2180 | icon.CopyFromBitmap(getNextBitmap()) | |
2181 | return icon | |
2182 | ||
2183 | #---------------------------------------------------------------------- | |
2184 | def getStepInData(): | |
2185 | return \ | |
2186 | '\x89PNG\r\n\x1a\n\x00\x00\x00\rIHDR\x00\x00\x00\x10\x00\x00\x00\x10\x08\x06\ | |
2187 | \x00\x00\x00\x1f\xf3\xffa\x00\x00\x00\x04sBIT\x08\x08\x08\x08|\x08d\x88\x00\ | |
2188 | \x00\x00\x87IDAT8\x8d\xadSA\x12\x84 \x0ck\x8a\xffv\xfc\xb74{X\xeb0P@\x07s\ | |
2189 | \x84\xa4$\x01\x00M\xb2\x02]R\x8b\xc86\xda\xdc\xedd\xb4~\xe8\x86\xc6\x01-\x93\ | |
2190 | \x96\xd9#\xf6\x06\xc3;p1I\xd1\x14\x0b#|\x17aF\xec\r\xeeF\xa0eB\xd34\xca\xd0A\ | |
2191 | ]j\x84\xa6\x03\x00""\xb7\xb0tRZ\xf7x\xb7\x83\x91]\xcb\x7fa\xd9\x89\x0fC\xfd\ | |
2192 | \x94\x9d|9\x99^k\x13\xa1 \xb3\x16\x0f#\xd4\x88N~\x14\xe1-\x96\x7f\xe3\x0f\ | |
2193 | \x11\x91UC\x0cX\'\x1e\x00\x00\x00\x00IEND\xaeB`\x82' | |
2194 | ||
2195 | def getStepInBitmap(): | |
2196 | return BitmapFromImage(getStepInImage()) | |
2197 | ||
2198 | def getStepInImage(): | |
2199 | stream = cStringIO.StringIO(getStepInData()) | |
2200 | return ImageFromStream(stream) | |
2201 | ||
2202 | def getStepInIcon(): | |
2203 | icon = EmptyIcon() | |
2204 | icon.CopyFromBitmap(getStepInBitmap()) | |
2205 | return icon | |
2206 | ||
2207 | #---------------------------------------------------------------------- | |
2208 | def getStopData(): | |
2209 | return \ | |
2210 | "\x89PNG\r\n\x1a\n\x00\x00\x00\rIHDR\x00\x00\x00\x10\x00\x00\x00\x10\x08\x06\ | |
2211 | \x00\x00\x00\x1f\xf3\xffa\x00\x00\x00\x04sBIT\x08\x08\x08\x08|\x08d\x88\x00\ | |
2212 | \x00\x00FIDAT8\x8d\xed\x91\xc1\t\xc00\x0c\x03O!\x0b\xa9\xfb\xef\xa6\xfcB\xa1\ | |
2213 | N\t\xf4Wr\xa0\x8f\xb1\x0f\x81\xe1\x97\xe4-\xb6}_V%\xc8\xc2, \t\x92\xe6]\xfbZ\ | |
2214 | \xf7\x08\xa0W\xc3\xea5\xdb\rl_IX\xe5\xf0d\x00\xfa\x8d#\x7f\xc4\xf7'\xab\x00\ | |
2215 | \x00\x00\x00IEND\xaeB`\x82" | |
2216 | ||
2217 | def getStopBitmap(): | |
2218 | return BitmapFromImage(getStopImage()) | |
2219 | ||
2220 | def getStopImage(): | |
2221 | stream = cStringIO.StringIO(getStopData()) | |
2222 | return ImageFromStream(stream) | |
2223 | ||
2224 | def getStopIcon(): | |
2225 | icon = EmptyIcon() | |
2226 | icon.CopyFromBitmap(getStopBitmap()) | |
2227 | return icon | |
2228 | ||
2229 | #---------------------------------------------------------------------- | |
2230 | def getStepReturnData(): | |
2231 | return \ | |
2232 | "\x89PNG\r\n\x1a\n\x00\x00\x00\rIHDR\x00\x00\x00\x10\x00\x00\x00\x10\x08\x06\ | |
2233 | \x00\x00\x00\x1f\xf3\xffa\x00\x00\x00\x04sBIT\x08\x08\x08\x08|\x08d\x88\x00\ | |
2234 | \x00\x00\x8dIDAT8\x8d\xa5S\xd1\x0e\xc4 \x08\xa3\xb0\xff\xbe\xdc\x7fO\xba'6\ | |
2235 | \xf1\xf44\xb3O$Phk\x04\xd4d\x07\xba\xc5\x16\x91#\nza\xdb\x84\x1a\xa2\xfe\xf8\ | |
2236 | \x99\xfa_=p+\xe8\x91ED\xbc<\xa4 \xb4\x0b\x01\xb5{\x01\xf9\xbbG-\x13\x87\x16f\ | |
2237 | \x84\xbf\x16V\xb0l\x01@\no\x86\xae\x82Q\xa8=\xa4\x0c\x80\xe70\xbd\x10jh\xbd\ | |
2238 | \x07R\x06#\xc9^N\xb6\xde\x03)\x83\x18\xaeU\x90\x9c>a\xb2P\r\xb3&/Y\xa8\xd1^^\ | |
2239 | \xb6\xf0\x16\xdb\xbf\xf1\x02\x81\xa5TK\x1d\x07\xde\x92\x00\x00\x00\x00IEND\ | |
2240 | \xaeB`\x82" | |
2241 | ||
2242 | def getStepReturnBitmap(): | |
2243 | return BitmapFromImage(getStepReturnImage()) | |
2244 | ||
2245 | def getStepReturnImage(): | |
2246 | stream = cStringIO.StringIO(getStepReturnData()) | |
2247 | return ImageFromStream(stream) | |
2248 | ||
2249 | def getStepReturnIcon(): | |
2250 | icon = EmptyIcon() | |
2251 | icon.CopyFromBitmap(getStepReturnBitmap()) | |
2252 | return icon | |
2253 | ||
2254 | def getAddWatchData(): | |
2255 | return \ | |
2256 | '\x89PNG\r\n\x1a\n\x00\x00\x00\rIHDR\x00\x00\x00\x10\x00\x00\x00\x10\x08\x06\ | |
2257 | \x00\x00\x00\x1f\xf3\xffa\x00\x00\x00\x04sBIT\x08\x08\x08\x08|\x08d\x88\x00\ | |
2258 | \x00\x00\x85IDAT8\x8dc`\x18h\xc0\x88.\xd0\xc0\xf0\xff?*\x9f\x11C\rN\x80\xae\ | |
2259 | \x19\x97\x18\xd1\x9a\x896\x84\x18[p\xa9aA\xe6\xfc7f\xc0P\xc4x\x163\x9cp\x1a0\ | |
2260 | \xeb,!w\x100 \x1dK\xac\x10\r\x08\x05".yFL\x85\x8c\x18b\xa8|Ty\xa2\x13\x92\'\ | |
2261 | \xc3\xe4\xff\x9f\x18\x1e3\xb82t\xa2\x88\x13\xedg.\x06aa&\x06VV\x7f\x86\xb9\ | |
2262 | \xcfU\x19\xbc\xb0\xba\x86h\xe0\xc8\xd0\xfc\xbf\x80\xe1>q)\x94\xe6\x00\x00\ | |
2263 | \x85\x923_\xd22\xa4\xcd\x00\x00\x00\x00IEND\xaeB`\x82' | |
2264 | ||
2265 | def getAddWatchBitmap(): | |
2266 | return BitmapFromImage(getAddWatchImage()) | |
2267 | ||
2268 | def getAddWatchImage(): | |
2269 | stream = cStringIO.StringIO(getAddWatchData()) | |
2270 | return ImageFromStream(stream) | |
2271 | ||
2272 | def getAddWatchIcon(): | |
2273 | icon = EmptyIcon() | |
2274 | icon.CopyFromBitmap(getAddWatchBitmap()) | |
2275 | return icon |