blob: 27613f2cb42dc7c566d274e64e3d4b39bf532d45 [file] [log] [blame]
Just van Rossum40f9b7b1999-01-30 22:39:17 +00001import FrameWork
Jack Jansen5a6fdcd2001-08-25 12:15:04 +00002from Carbon import Win
3from Carbon import Qd
4from Carbon import Evt
Just van Rossum40f9b7b1999-01-30 22:39:17 +00005import MacOS
Jack Jansendc4d9252001-08-27 10:55:41 +00006from Carbon import Events
Just van Rossum40f9b7b1999-01-30 22:39:17 +00007import traceback
8from types import *
Jack Jansen5a6fdcd2001-08-25 12:15:04 +00009from Carbon import Menu; MenuToolbox = Menu; del Menu
Jack Jansene9c03582002-04-03 21:52:10 +000010import macresource
Just van Rossum40f9b7b1999-01-30 22:39:17 +000011
Just van Rossum40144012002-02-04 12:52:44 +000012if hasattr(Win, "FrontNonFloatingWindow"):
13 MyFrontWindow = Win.FrontNonFloatingWindow
14else:
15 MyFrontWindow = Win.FrontWindow
16
17
Just van Rossumf376ef02001-11-18 14:12:43 +000018KILLUNKNOWNWINDOWS = 0 # Set to 0 for debugging.
Just van Rossum40f9b7b1999-01-30 22:39:17 +000019
20class Application(FrameWork.Application):
21
22 def __init__(self, signature='Pyth'):
Jack Jansene9c03582002-04-03 21:52:10 +000023 # Open our resource file, if it is not open yet
24 macresource.need('CURS', 468, "Widgets.rsrc")
Just van Rossum40f9b7b1999-01-30 22:39:17 +000025 import W
26 W.setapplication(self, signature)
27 FrameWork.Application.__init__(self)
28 self._suspended = 0
29 self.quitting = 0
30 self.debugger_quitting = 1
31 self.DebuggerQuit = 'DebuggerQuitDummyException'
32 self._idlefuncs = []
33 # map certain F key codes to equivalent command-letter combos (JJS)
34 self.fkeymaps = {122:"z", 120:"x", 99:"c", 118:"v"}
35
Jack Jansene5c94742001-03-08 23:10:37 +000036 def mainloop(self, mask=FrameWork.everyEvent, wait=None):
Just van Rossum40f9b7b1999-01-30 22:39:17 +000037 import W
38 self.quitting = 0
Jack Jansen815d2bf2002-01-21 23:00:52 +000039 if hasattr(MacOS, 'EnableAppswitch'):
40 saveyield = MacOS.EnableAppswitch(-1)
Just van Rossum40f9b7b1999-01-30 22:39:17 +000041 try:
42 while not self.quitting:
43 try:
44 self.do1event(mask, wait)
45 except W.AlertError, detail:
Jack Jansen815d2bf2002-01-21 23:00:52 +000046 if hasattr(MacOS, 'EnableAppswitch'):
47 MacOS.EnableAppswitch(-1)
Just van Rossum40f9b7b1999-01-30 22:39:17 +000048 W.Message(detail)
49 except self.DebuggerQuit:
Jack Jansen815d2bf2002-01-21 23:00:52 +000050 if hasattr(MacOS, 'EnableAppswitch'):
51 MacOS.EnableAppswitch(-1)
Just van Rossum40f9b7b1999-01-30 22:39:17 +000052 except:
Jack Jansen815d2bf2002-01-21 23:00:52 +000053 if hasattr(MacOS, 'EnableAppswitch'):
54 MacOS.EnableAppswitch(-1)
Just van Rossum40f9b7b1999-01-30 22:39:17 +000055 import PyEdit
56 PyEdit.tracebackwindow.traceback()
57 finally:
Jack Jansen815d2bf2002-01-21 23:00:52 +000058 if hasattr(MacOS, 'EnableAppswitch'):
59 MacOS.EnableAppswitch(1)
Just van Rossum40f9b7b1999-01-30 22:39:17 +000060
Jack Jansene5c94742001-03-08 23:10:37 +000061 def debugger_mainloop(self, mask=FrameWork.everyEvent, wait=None):
Just van Rossum40f9b7b1999-01-30 22:39:17 +000062 import W
63 self.debugger_quitting = 0
Jack Jansen815d2bf2002-01-21 23:00:52 +000064 if hasattr(MacOS, 'EnableAppswitch'):
65 saveyield = MacOS.EnableAppswitch(-1)
Just van Rossum40f9b7b1999-01-30 22:39:17 +000066 try:
67 while not self.quitting and not self.debugger_quitting:
68 try:
69 self.do1event(mask, wait)
70 except W.AlertError, detail:
71 W.Message(detail)
72 except:
73 import PyEdit
74 PyEdit.tracebackwindow.traceback()
75 finally:
Jack Jansen815d2bf2002-01-21 23:00:52 +000076 if hasattr(MacOS, 'EnableAppswitch'):
77 MacOS.EnableAppswitch(saveyield)
Just van Rossum40f9b7b1999-01-30 22:39:17 +000078
79 def breathe(self, wait=1):
80 import W
81 ok, event = Evt.WaitNextEvent(FrameWork.updateMask |
82 FrameWork.mDownMask | FrameWork.osMask |
83 FrameWork.activMask,
84 wait)
85 if ok:
86 (what, message, when, where, modifiers) = event
87 #print FrameWork.eventname[what]
88 if FrameWork.eventname[what] == 'mouseDown':
89 partcode, wid = Win.FindWindow(where)
90 if FrameWork.partname[partcode] <> 'inDesk':
91 return
92 else:
93 W.SetCursor('watch')
94 self.dispatch(event)
95
96 def refreshwindows(self, wait=1):
97 import W
98 while 1:
99 ok, event = Evt.WaitNextEvent(FrameWork.updateMask, wait)
100 if not ok:
101 break
102 self.dispatch(event)
103
104 def addidlefunc(self, func):
105 self._idlefuncs.append(func)
106
107 def removeidlefunc(self, func):
108 self._idlefuncs.remove(func)
109
110 def idle(self, event):
111 if not self._suspended:
112 if not self.do_frontWindowMethod("idle", event):
113 Qd.InitCursor()
114 if self._idlefuncs:
115 for func in self._idlefuncs:
116 try:
117 func()
118 except:
119 import sys
120 sys.stderr.write("exception in idle function %s; killed:\n" % `func`)
121 traceback.print_exc()
122 self._idlefuncs.remove(func)
123 break
124
125 def do_frontWindowMethod(self, attr, *args):
Just van Rossum40144012002-02-04 12:52:44 +0000126 wid = MyFrontWindow()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000127 if wid and self._windows.has_key(wid):
128 window = self._windows[wid]
129 if hasattr(window, attr):
130 handler = getattr(window, attr)
131 apply(handler, args)
132 return 1
133
134 def appendwindow(self, wid, window):
135 self._windows[wid] = window
136 self.makeopenwindowsmenu()
137
138 def removewindow(self, wid):
139 del self._windows[wid]
140 self.makeopenwindowsmenu()
141
Just van Rossumd4bdbf11999-02-26 21:30:19 +0000142 def makeopenwindowsmenu(self):
143 # dummy; could be the full version from PythonIDEMain.py
144 self._openwindows = {}
145 self._openwindowscheckmark = 0
146 if not hasattr(self, "_menustocheck"):
147 self._menustocheck = []
148
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000149 def do_key(self, event):
150 (what, message, when, where, modifiers) = event
151 ch = chr(message & FrameWork.charCodeMask)
152 rest = message & ~FrameWork.charCodeMask
153 keycode = (message & FrameWork.keyCodeMask) >> 8
154 if keycode in self.fkeymaps.keys(): # JJS
155 ch = self.fkeymaps[keycode]
156 modifiers = modifiers | FrameWork.cmdKey
Just van Rossum40144012002-02-04 12:52:44 +0000157 wid = MyFrontWindow()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000158 if modifiers & FrameWork.cmdKey and not modifiers & FrameWork.shiftKey:
159 if wid and self._windows.has_key(wid):
160 self.checkmenus(self._windows[wid])
161 else:
162 self.checkmenus(None)
163 event = (what, ord(ch) | rest, when, where, modifiers)
164 result = MenuToolbox.MenuKey(ord(ch))
165 id = (result>>16) & 0xffff # Hi word
166 item = result & 0xffff # Lo word
167 if id:
168 self.do_rawmenu(id, item, None, event)
169 return # here! we had a menukey!
170 #else:
171 # print "XXX Command-" +`ch`
172 # See whether the front window wants it
173 if wid and self._windows.has_key(wid):
174 window = self._windows[wid]
175 try:
176 do_char = window.do_char
177 except AttributeError:
178 do_char = self.do_char
179 do_char(ch, event)
180 # else it wasn't for us, sigh...
181
182 def do_inMenuBar(self, partcode, window, event):
183 Qd.InitCursor()
184 (what, message, when, where, modifiers) = event
185 self.checkopenwindowsmenu()
Just van Rossum40144012002-02-04 12:52:44 +0000186 wid = MyFrontWindow()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000187 if wid and self._windows.has_key(wid):
188 self.checkmenus(self._windows[wid])
189 else:
190 self.checkmenus(None)
191 result = MenuToolbox.MenuSelect(where)
192 id = (result>>16) & 0xffff # Hi word
193 item = result & 0xffff # Lo word
194 self.do_rawmenu(id, item, window, event)
195
196 def do_updateEvt(self, event):
197 (what, message, when, where, modifiers) = event
198 wid = Win.WhichWindow(message)
199 if wid and self._windows.has_key(wid):
200 window = self._windows[wid]
201 window.do_rawupdate(wid, event)
202 else:
Jack Jansendc4d9252001-08-27 10:55:41 +0000203 if KILLUNKNOWNWINDOWS and wid:
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000204 wid.HideWindow()
205 import sys
206 sys.stderr.write("XXX killed unknown (crashed?) Python window.\n")
207 else:
Jack Jansenb9e66612002-03-21 22:36:57 +0000208 if hasattr(MacOS, 'HandleEvent'):
209 MacOS.HandleEvent(event)
210 else:
211 print 'Unexpected updateEvent:', event
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000212
213 def suspendresume(self, onoff):
214 pass
215
216 def do_suspendresume(self, event):
217 self._suspended = not event[1] & 1
218 FrameWork.Application.do_suspendresume(self, event)
219
220 def checkopenwindowsmenu(self):
221 if self._openwindowscheckmark:
Jack Jansenafd0aa62001-01-29 13:29:47 +0000222 self.openwindowsmenu.menu.CheckMenuItem(self._openwindowscheckmark, 0)
Just van Rossum40144012002-02-04 12:52:44 +0000223 window = MyFrontWindow()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000224 if window:
225 for item, wid in self._openwindows.items():
226 if wid == window:
227 #self.pythonwindowsmenuitem.check(1)
Jack Jansenafd0aa62001-01-29 13:29:47 +0000228 self.openwindowsmenu.menu.CheckMenuItem(item, 1)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000229 self._openwindowscheckmark = item
230 break
231 else:
232 self._openwindowscheckmark = 0
233 #if self._openwindows:
234 # self.pythonwindowsmenuitem.enable(1)
235 #else:
236 # self.pythonwindowsmenuitem.enable(0)
237
238 def checkmenus(self, window):
239 for item in self._menustocheck:
240 callback = item.menu.items[item.item-1][2]
241 if type(callback) <> StringType:
242 item.enable(1)
243 elif hasattr(window, "domenu_" + callback):
244 if hasattr(window, "can_" + callback):
245 canhandler = getattr(window, "can_" + callback)
246 if canhandler(item):
247 item.enable(1)
248 else:
249 item.enable(0)
250 else:
251 item.enable(1)
252 else:
253 item.enable(0)
254
255 def enablemenubar(self, onoff):
256 for m in self.menubar.menus.values():
257 if onoff:
Jack Jansenafd0aa62001-01-29 13:29:47 +0000258 m.menu.EnableMenuItem(0)
Just van Rossum4798c4e1999-02-27 17:15:32 +0000259 elif m.menu.GetMenuItemText(3) <> 'Cut': # ew...
Jack Jansenafd0aa62001-01-29 13:29:47 +0000260 m.menu.DisableMenuItem(0)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000261 MenuToolbox.DrawMenuBar()
262
263 def makemenubar(self):
264 self.menubar = MenuBar(self)
265 FrameWork.AppleMenu(self.menubar, self.getabouttext(), self.do_about)
266 self.makeusermenus()
Just van Rossum7171df31999-10-30 11:46:18 +0000267
268 def scriptswalk(self, top, menu, done=None):
269 if done is None:
270 done = {}
271 if done.has_key(top):
272 return
273 done[top] = 1
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000274 import os, macfs, string
275 try:
276 names = os.listdir(top)
277 except os.error:
278 FrameWork.MenuItem(menu, '(Scripts Folder not found)', None, None)
279 return
280 savedir = os.getcwd()
281 os.chdir(top)
282 for name in names:
Just van Rossum6dc2b571999-12-23 21:50:26 +0000283 if name == "CVS":
284 continue
285 try:
286 fss, isdir, isalias = macfs.ResolveAliasFile(name)
287 except:
288 # maybe a broken alias
289 continue
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000290 path = fss.as_pathname()
Just van Rossum7171df31999-10-30 11:46:18 +0000291 if done.has_key(path):
292 continue
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000293 name = string.strip(name)
294 if name[-3:] == '---':
295 menu.addseparator()
296 elif isdir:
297 submenu = FrameWork.SubMenu(menu, name)
Just van Rossum7171df31999-10-30 11:46:18 +0000298 self.scriptswalk(path, submenu, done)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000299 else:
300 creator, type = fss.GetCreatorType()
301 if type == 'TEXT':
302 if name[-3:] == '.py':
303 name = name[:-3]
304 item = FrameWork.MenuItem(menu, name, None, self.domenu_script)
305 self._scripts[(menu.id, item.item)] = path
Just van Rossum7171df31999-10-30 11:46:18 +0000306 done[path] = 1
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000307 os.chdir(savedir)
308
309 def domenu_script(self, id, item, window, event):
310 (what, message, when, where, modifiers) = event
311 path = self._scripts[(id, item)]
312 import os
313 if not os.path.exists(path):
314 self.makescriptsmenu()
315 import W
316 raise W.AlertError, "File not found."
317 if ord(Evt.GetKeys()[7]) & 4:
318 self.openscript(path)
319 else:
320 import W, MacOS, sys
321 W.SetCursor("watch")
322 sys.argv = [path]
323 #cwd = os.getcwd()
324 #os.chdir(os.path.dirname(path) + ':')
325 try:
326 # xxx if there is a script window for this file,
327 # exec in that window's namespace.
328 # xxx what to do when it's not saved???
329 # promt to save?
Jack Jansen815d2bf2002-01-21 23:00:52 +0000330 if hasattr(MacOS, 'EnableAppswitch'):
331 MacOS.EnableAppswitch(0)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000332 execfile(path, {'__name__': '__main__', '__file__': path})
333 except W.AlertError, detail:
Jack Jansen815d2bf2002-01-21 23:00:52 +0000334 if hasattr(MacOS, 'EnableAppswitch'):
335 MacOS.EnableAppswitch(-1)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000336 raise W.AlertError, detail
337 except KeyboardInterrupt:
Jack Jansen815d2bf2002-01-21 23:00:52 +0000338 if hasattr(MacOS, 'EnableAppswitch'):
339 MacOS.EnableAppswitch(-1)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000340 except:
Jack Jansen815d2bf2002-01-21 23:00:52 +0000341 if hasattr(MacOS, 'EnableAppswitch'):
342 MacOS.EnableAppswitch(-1)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000343 import PyEdit
344 PyEdit.tracebackwindow.traceback(1)
345 else:
Jack Jansen815d2bf2002-01-21 23:00:52 +0000346 if hasattr(MacOS, 'EnableAppswitch'):
347 MacOS.EnableAppswitch(-1)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000348 #os.chdir(cwd)
349
350 def openscript(self, filename, lineno=None, charoffset=0, modname=""):
351 import os, PyEdit, W
352 editor = self.getscript(filename)
353 if editor:
354 editor.select()
355 elif os.path.exists(filename):
356 editor = PyEdit.Editor(filename)
357 elif filename[-3:] == '.py' or filename[-4:] == '.pyc':
358 import imp
359 if not modname:
360 if filename[-1] == 'c':
361 modname = os.path.basename(filename)[:-4]
362 else:
363 modname = os.path.basename(filename)[:-3]
364 try:
365 # XXX This does not work correctly with packages!
366 # XXX The docs say we should do it manually, pack, then sub, then sub2 etc.
367 # XXX It says we should use imp.load_module(), but that *reloads* a package,
368 # XXX and that's the last thing we want here.
369 f, filename, (suff, mode, dummy) = imp.find_module(modname)
370 except ImportError:
Just van Rossumdc3c6172001-06-19 21:37:33 +0000371 raise W.AlertError, "Can't find file for \"%s\"" % modname
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000372 else:
373 if not f:
Just van Rossumdc3c6172001-06-19 21:37:33 +0000374 raise W.AlertError, "Can't find file for \"%s\"" % modname
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000375 f.close()
376 if suff == '.py':
377 self.openscript(filename, lineno, charoffset)
378 return
379 else:
Just van Rossumdc3c6172001-06-19 21:37:33 +0000380 raise W.AlertError, "Can't find file for \"%s\"" % modname
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000381 else:
Just van Rossumdc3c6172001-06-19 21:37:33 +0000382 raise W.AlertError, "Can't find file \"%s\"" % filename
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000383 if lineno is not None:
384 editor.selectline(lineno, charoffset)
385 return editor
386
387 def getscript(self, filename):
388 if filename[:1] == '<' and filename[-1:] == '>':
389 filename = filename[1:-1]
390 import string
391 lowpath = string.lower(filename)
392 for wid, window in self._windows.items():
Just van Rossum7171df31999-10-30 11:46:18 +0000393 if hasattr(window, "path") and type(window.path) == StringType and \
394 lowpath == string.lower(window.path):
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000395 return window
396 elif hasattr(window, "path") and filename == wid.GetWTitle():
397 return window
398
399 def getprefs(self):
400 import MacPrefs
401 return MacPrefs.GetPrefs(self.preffilepath)
Just van Rossum2a759091999-09-26 12:18:19 +0000402
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000403 def do_editorprefs(self, *args):
404 import PyEdit
405 PyEdit.EditorDefaultSettings()
406
Just van Rossum2a759091999-09-26 12:18:19 +0000407 def do_setwindowfont(self, *args):
408 import FontSettings, W
409 prefs = self.getprefs()
410 settings = FontSettings.FontDialog(prefs.defaultfont)
411 if settings:
412 prefs.defaultfont, tabsettings = settings
413 raise W.AlertError, "Note that changes will only affect new windows!"
414
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000415
416
417class MenuBar(FrameWork.MenuBar):
418
419 possibleIDs = range(10, 256)
420
421 def getnextid(self):
422 id = self.possibleIDs[0]
423 del self.possibleIDs[0]
424 return id
425
426 def __init__(self, parent = None):
427 self.bar = MenuToolbox.GetMenuBar()
428 MenuToolbox.ClearMenuBar()
429 self.menus = {}
430 self.parent = parent
431
432 def dispatch(self, id, item, window, event):
433 if self.menus.has_key(id):
434 self.menus[id].dispatch(id, item, window, event)
435
436 def delmenu(self, id):
437 MenuToolbox.DeleteMenu(id)
438 if id in self.possibleIDs:
439 print "XXX duplicate menu ID!", id
440 self.possibleIDs.append(id)
441
442
443class Menu(FrameWork.Menu):
444
445 def dispatch(self, id, item, window, event):
446 title, shortcut, callback, kind = self.items[item-1]
447 if type(callback) == StringType:
448 callback = self._getmenuhandler(callback)
449 if callback:
450 import W
451 W.CallbackCall(callback, 0, id, item, window, event)
452
453 def _getmenuhandler(self, callback):
454 menuhandler = None
Just van Rossum40144012002-02-04 12:52:44 +0000455 wid = MyFrontWindow()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000456 if wid and self.bar.parent._windows.has_key(wid):
457 window = self.bar.parent._windows[wid]
458 if hasattr(window, "domenu_" + callback):
459 menuhandler = getattr(window, "domenu_" + callback)
460 elif hasattr(self.bar.parent, "domenu_" + callback):
461 menuhandler = getattr(self.bar.parent, "domenu_" + callback)
462 elif hasattr(self.bar.parent, "domenu_" + callback):
463 menuhandler = getattr(self.bar.parent, "domenu_" + callback)
464 return menuhandler
465