blob: 365082d1b8d4f71fa00e3ea05dc20bac56b7b381 [file] [log] [blame]
Just van Rossum2d564fd2001-11-02 19:30:21 +00001# copyright 1997-2001 Just van Rossum, Letterror. just@letterror.com
Just van Rossum40f9b7b1999-01-30 22:39:17 +00002
3import Splash
4
5import FrameWork
Just van Rossum40f9b7b1999-01-30 22:39:17 +00006import Wapplication
7import W
8import os
Jack Jansene0ba0872002-03-29 21:23:47 +00009import sys
Jack Jansen815d2bf2002-01-21 23:00:52 +000010import MacOS
Jack Jansenfd0b00e2003-01-26 22:15:48 +000011import EasyDialogs
Jack Jansene7ee17c2003-02-06 22:32:35 +000012from Carbon import File
13from Carbon import Files
Just van Rossum40f9b7b1999-01-30 22:39:17 +000014
Jack Jansen815d2bf2002-01-21 23:00:52 +000015if MacOS.runtimemodel == 'macho':
16 ELIPSES = '...'
17else:
18 ELIPSES = '\xc9'
Just van Rossum40f9b7b1999-01-30 22:39:17 +000019
Just van Rossumbf0a9082002-02-04 12:48:06 +000020def runningOnOSX():
21 from gestalt import gestalt
22 gestaltMenuMgrAquaLayoutBit = 1 # menus have the Aqua 1.0 layout
23 gestaltMenuMgrAquaLayoutMask = (1L << gestaltMenuMgrAquaLayoutBit)
24 value = gestalt("menu") & gestaltMenuMgrAquaLayoutMask
25 return not not value
26
Jack Jansene7ee17c2003-02-06 22:32:35 +000027def getmodtime(file):
28 file = File.FSRef(file)
29 catinfo, d1, d2, d3 = file.FSGetCatalogInfo(Files.kFSCatInfoContentMod)
30 return catinfo.contentModDate
Just van Rossumbf0a9082002-02-04 12:48:06 +000031
Just van Rossum40f9b7b1999-01-30 22:39:17 +000032class PythonIDE(Wapplication.Application):
33
34 def __init__(self):
Just van Rossum979c5372002-07-12 16:50:32 +000035 self.preffilepath = os.path.join("Python", "PythonIDE preferences")
Just van Rossumf4b06811999-02-27 17:16:54 +000036 Wapplication.Application.__init__(self, 'Pide')
Jack Jansen5a6fdcd2001-08-25 12:15:04 +000037 from Carbon import AE
38 from Carbon import AppleEvents
Just van Rossum40f9b7b1999-01-30 22:39:17 +000039
40 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEOpenApplication,
41 self.ignoreevent)
Jack Jansen53ebe562001-03-08 23:09:32 +000042 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEReopenApplication,
43 self.ignoreevent)
Just van Rossum40f9b7b1999-01-30 22:39:17 +000044 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEPrintDocuments,
45 self.ignoreevent)
46 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEOpenDocuments,
47 self.opendocsevent)
48 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEQuitApplication,
49 self.quitevent)
50 import PyConsole, PyEdit
51 Splash.wait()
Just van Rossum7aa7f2e2002-10-20 17:14:28 +000052 if sys.platform == "darwin":
53 if sys.argv and sys.argv[0].startswith("-psn"):
54 home = os.getenv("HOME")
55 if home:
56 os.chdir(home)
Jack Jansene0ba0872002-03-29 21:23:47 +000057 # With -D option (OSX command line only) keep stderr, for debugging the IDE
58 # itself.
59 debug_stderr = None
Just van Rossum8bb61c82002-03-29 21:47:56 +000060 if len(sys.argv) >= 2 and sys.argv[1] == '-D':
Jack Jansene0ba0872002-03-29 21:23:47 +000061 debug_stderr = sys.stderr
62 del sys.argv[1]
Just van Rossum40f9b7b1999-01-30 22:39:17 +000063 PyConsole.installoutput()
64 PyConsole.installconsole()
Jack Jansene0ba0872002-03-29 21:23:47 +000065 if debug_stderr:
66 sys.stderr = debug_stderr
Just van Rossum40f9b7b1999-01-30 22:39:17 +000067 for path in sys.argv[1:]:
Just van Rossum62a103b2002-11-22 12:48:47 +000068 if path.startswith("-p"):
69 # process number added by the OS
70 continue
Just van Rossum40f9b7b1999-01-30 22:39:17 +000071 self.opendoc(path)
Just van Rossum62a103b2002-11-22 12:48:47 +000072 self.mainloop()
Just van Rossum40f9b7b1999-01-30 22:39:17 +000073
74 def makeusermenus(self):
75 m = Wapplication.Menu(self.menubar, "File")
76 newitem = FrameWork.MenuItem(m, "New", "N", 'new')
Jack Jansen815d2bf2002-01-21 23:00:52 +000077 openitem = FrameWork.MenuItem(m, "Open"+ELIPSES, "O", 'open')
Jack Jansen611b9f62003-02-05 15:41:09 +000078 openbynameitem = FrameWork.MenuItem(m, "Open File by Name"+ELIPSES, "D", 'openbyname')
Just van Rossum40f9b7b1999-01-30 22:39:17 +000079 FrameWork.Separator(m)
80 closeitem = FrameWork.MenuItem(m, "Close", "W", 'close')
81 saveitem = FrameWork.MenuItem(m, "Save", "S", 'save')
Jack Jansen815d2bf2002-01-21 23:00:52 +000082 saveasitem = FrameWork.MenuItem(m, "Save as"+ELIPSES, None, 'save_as')
Just van Rossum40f9b7b1999-01-30 22:39:17 +000083 FrameWork.Separator(m)
Jack Jansen815d2bf2002-01-21 23:00:52 +000084 saveasappletitem = FrameWork.MenuItem(m, "Save as Applet"+ELIPSES, None, 'save_as_applet')
Just van Rossumbf0a9082002-02-04 12:48:06 +000085 if not runningOnOSX():
86 # On OSX there's a special "magic" quit menu, so we shouldn't add
87 # it to the File menu.
88 FrameWork.Separator(m)
89 quititem = FrameWork.MenuItem(m, "Quit", "Q", 'quit')
Just van Rossum40f9b7b1999-01-30 22:39:17 +000090
91 m = Wapplication.Menu(self.menubar, "Edit")
92 undoitem = FrameWork.MenuItem(m, "Undo", 'Z', "undo")
93 FrameWork.Separator(m)
94 cutitem = FrameWork.MenuItem(m, "Cut", 'X', "cut")
95 copyitem = FrameWork.MenuItem(m, "Copy", "C", "copy")
96 pasteitem = FrameWork.MenuItem(m, "Paste", "V", "paste")
97 FrameWork.MenuItem(m, "Clear", None, "clear")
98 FrameWork.Separator(m)
99 selallitem = FrameWork.MenuItem(m, "Select all", "A", "selectall")
100 sellineitem = FrameWork.MenuItem(m, "Select line", "L", "selectline")
101 FrameWork.Separator(m)
Jack Jansen815d2bf2002-01-21 23:00:52 +0000102 finditem = FrameWork.MenuItem(m, "Find"+ELIPSES, "F", "find")
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000103 findagainitem = FrameWork.MenuItem(m, "Find again", 'G', "findnext")
104 enterselitem = FrameWork.MenuItem(m, "Enter search string", "E", "entersearchstring")
105 replaceitem = FrameWork.MenuItem(m, "Replace", None, "replace")
106 replacefinditem = FrameWork.MenuItem(m, "Replace & find again", 'T', "replacefind")
107 FrameWork.Separator(m)
108 shiftleftitem = FrameWork.MenuItem(m, "Shift left", "[", "shiftleft")
109 shiftrightitem = FrameWork.MenuItem(m, "Shift right", "]", "shiftright")
110
111 m = Wapplication.Menu(self.menubar, "Python")
112 runitem = FrameWork.MenuItem(m, "Run window", "R", 'run')
113 runselitem = FrameWork.MenuItem(m, "Run selection", None, 'runselection')
114 FrameWork.Separator(m)
Jack Jansen815d2bf2002-01-21 23:00:52 +0000115 moditem = FrameWork.MenuItem(m, "Module browser"+ELIPSES, "M", self.domenu_modulebrowser)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000116 FrameWork.Separator(m)
117 mm = FrameWork.SubMenu(m, "Preferences")
Jack Jansen815d2bf2002-01-21 23:00:52 +0000118 FrameWork.MenuItem(mm, "Set Scripts folder"+ELIPSES, None, self.do_setscriptsfolder)
119 FrameWork.MenuItem(mm, "Editor default settings"+ELIPSES, None, self.do_editorprefs)
120 FrameWork.MenuItem(mm, "Set default window font"+ELIPSES, None, self.do_setwindowfont)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000121
122 self.openwindowsmenu = Wapplication.Menu(self.menubar, 'Windows')
123 self.makeopenwindowsmenu()
124 self._menustocheck = [closeitem, saveitem, saveasitem, saveasappletitem,
125 undoitem, cutitem, copyitem, pasteitem,
126 selallitem, sellineitem,
127 finditem, findagainitem, enterselitem, replaceitem, replacefinditem,
128 shiftleftitem, shiftrightitem,
129 runitem, runselitem]
130
131 prefs = self.getprefs()
132 try:
Jack Jansene7ee17c2003-02-06 22:32:35 +0000133 fsr, d = File.Alias(rawdata=prefs.scriptsfolder).FSResolveAlias(None)
134 self.scriptsfolder = fsr.FSNewAliasMinimal()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000135 except:
Jack Jansene7ee17c2003-02-06 22:32:35 +0000136 path = os.path.join(os.getcwd(), "Mac", "IDE scripts")
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000137 if not os.path.exists(path):
Just van Rossum2028b591999-09-26 12:16:22 +0000138 path = os.path.join(os.getcwd(), "Scripts")
139 if not os.path.exists(path):
140 os.mkdir(path)
Jack Jansen815d2bf2002-01-21 23:00:52 +0000141 f = open(os.path.join(path, "Place your scripts here"+ELIPSES), "w")
Just van Rossum2028b591999-09-26 12:16:22 +0000142 f.close()
Jack Jansene7ee17c2003-02-06 22:32:35 +0000143 fsr = File.FSRef(path)
144 self.scriptsfolder = fsr.FSNewAliasMinimal()
145 self.scriptsfoldermodtime = getmodtime(fsr)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000146 else:
Jack Jansene7ee17c2003-02-06 22:32:35 +0000147 self.scriptsfoldermodtime = getmodtime(fsr)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000148 prefs.scriptsfolder = self.scriptsfolder.data
149 self._scripts = {}
150 self.scriptsmenu = None
151 self.makescriptsmenu()
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000152 self.makehelpmenu()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000153
154 def quitevent(self, theAppleEvent, theReply):
Jack Jansen5a6fdcd2001-08-25 12:15:04 +0000155 from Carbon import AE
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000156 AE.AEInteractWithUser(50000000)
157 self._quit()
158
159 def suspendresume(self, onoff):
160 if onoff:
Jack Jansene7ee17c2003-02-06 22:32:35 +0000161 fsr, changed = self.scriptsfolder.FSResolveAlias(None)
162 modtime = getmodtime(fsr)
163 if self.scriptsfoldermodtime <> modtime or changed:
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000164 self.scriptsfoldermodtime = modtime
165 W.SetCursor('watch')
166 self.makescriptsmenu()
167
168 def ignoreevent(self, theAppleEvent, theReply):
169 pass
170
171 def opendocsevent(self, theAppleEvent, theReply):
172 W.SetCursor('watch')
173 import aetools
174 parameters, args = aetools.unpackevent(theAppleEvent)
175 docs = parameters['----']
176 if type(docs) <> type([]):
177 docs = [docs]
178 for doc in docs:
Jack Jansene7ee17c2003-02-06 22:32:35 +0000179 fsr, a = doc.FSResolveAlias(None)
180 path = fsr.as_pathname()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000181 self.opendoc(path)
182
183 def opendoc(self, path):
Jack Jansene7ee17c2003-02-06 22:32:35 +0000184 fcreator, ftype = MacOS.GetCreatorAndType(path)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000185 if ftype == 'TEXT':
186 self.openscript(path)
Jack Jansene0ba0872002-03-29 21:23:47 +0000187 elif ftype == '\0\0\0\0' and path[-3:] == '.py':
188 self.openscript(path)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000189 else:
Just van Rossumdc3c6172001-06-19 21:37:33 +0000190 W.Message("Can't open file of type '%s'." % ftype)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000191
192 def getabouttext(self):
Jack Jansen815d2bf2002-01-21 23:00:52 +0000193 return "About Python IDE"+ELIPSES
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000194
195 def do_about(self, id, item, window, event):
196 Splash.about()
197
198 def do_setscriptsfolder(self, *args):
Jack Jansene7ee17c2003-02-06 22:32:35 +0000199 fsr = EasyDialogs.AskFolder(message="Select Scripts Folder",
200 wanted=File.FSRef)
201 if fsr:
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000202 prefs = self.getprefs()
Jack Jansene7ee17c2003-02-06 22:32:35 +0000203 alis = fsr.FSNewAliasMinimal()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000204 prefs.scriptsfolder = alis.data
205 self.scriptsfolder = alis
206 self.makescriptsmenu()
207 prefs.save()
208
209 def domenu_modulebrowser(self, *args):
210 W.SetCursor('watch')
211 import ModuleBrowser
212 ModuleBrowser.ModuleBrowser()
213
214 def domenu_open(self, *args):
Jack Jansenfd0b00e2003-01-26 22:15:48 +0000215 filename = EasyDialogs.AskFileForOpen(typeList=("TEXT",))
216 if filename:
217 self.openscript(filename)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000218
Jack Jansen611b9f62003-02-05 15:41:09 +0000219 def domenu_openbyname(self, *args):
220 # Open a file by name. If the clipboard contains a filename
221 # use that as the default.
222 from Carbon import Scrap
223 try:
224 sc = Scrap.GetCurrentScrap()
225 dft = sc.GetScrapFlavorData("TEXT")
226 except Scrap.Error:
227 dft = ""
228 else:
229 if not os.path.exists(dft):
230 dft = ""
231 filename = EasyDialogs.AskString("Open File Named:", default=dft, ok="Open")
232 if filename:
233 self.openscript(filename)
234
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000235 def domenu_new(self, *args):
236 W.SetCursor('watch')
237 import PyEdit
238 return PyEdit.Editor()
239
240 def makescriptsmenu(self):
241 W.SetCursor('watch')
242 if self._scripts:
243 for id, item in self._scripts.keys():
244 if self.menubar.menus.has_key(id):
245 m = self.menubar.menus[id]
246 m.delete()
247 self._scripts = {}
248 if self.scriptsmenu:
249 if hasattr(self.scriptsmenu, 'id') and self.menubar.menus.has_key(self.scriptsmenu.id):
250 self.scriptsmenu.delete()
251 self.scriptsmenu = FrameWork.Menu(self.menubar, "Scripts")
252 #FrameWork.MenuItem(self.scriptsmenu, "New script", None, self.domenu_new)
253 #self.scriptsmenu.addseparator()
Jack Jansene7ee17c2003-02-06 22:32:35 +0000254 fsr, d1 = self.scriptsfolder.FSResolveAlias(None)
255 self.scriptswalk(fsr.as_pathname(), self.scriptsmenu)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000256
257 def makeopenwindowsmenu(self):
258 for i in range(len(self.openwindowsmenu.items)):
259 self.openwindowsmenu.menu.DeleteMenuItem(1)
260 self.openwindowsmenu.items = []
261 windows = []
262 self._openwindows = {}
263 for window in self._windows.keys():
264 title = window.GetWTitle()
265 if not title:
266 title = "<no title>"
Jack Jansen34d11f02000-03-07 23:40:13 +0000267 windows.append((title, window))
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000268 windows.sort()
269 for title, window in windows:
270 if title == "Python Interactive": # ugly but useful hack by Joe Strout
271 shortcut = '0'
272 else:
273 shortcut = None
274 item = FrameWork.MenuItem(self.openwindowsmenu, title, shortcut, callback = self.domenu_openwindows)
275 self._openwindows[item.item] = window
276 self._openwindowscheckmark = 0
277 self.checkopenwindowsmenu()
278
279 def domenu_openwindows(self, id, item, window, event):
280 w = self._openwindows[item]
281 w.ShowWindow()
282 w.SelectWindow()
283
284 def domenu_quit(self):
285 self._quit()
286
287 def domenu_save(self, *args):
288 print "Save"
289
290 def _quit(self):
291 import PyConsole, PyEdit
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000292 for window in self._windows.values():
Just van Rossumd58c7461999-06-22 18:37:35 +0000293 try:
294 rv = window.close() # ignore any errors while quitting
295 except:
296 rv = 0 # (otherwise, we can get stuck!)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000297 if rv and rv > 0:
298 return
Just van Rossum62a103b2002-11-22 12:48:47 +0000299 try:
300 PyConsole.console.writeprefs()
301 PyConsole.output.writeprefs()
302 PyEdit.searchengine.writeprefs()
303 except:
304 # Write to __stderr__ so the msg end up in Console.app and has
305 # at least _some_ chance of getting read...
306 # But: this is a workaround for way more serious problems with
307 # the Python 2.2 Jaguar addon.
308 sys.__stderr__.write("*** PythonIDE: Can't write preferences ***\n")
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000309 self.quitting = 1
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000310
311 def makehelpmenu(self):
312 docs = self.installdocumentation()
313 self.helpmenu = m = self.gethelpmenu()
314 docitem = FrameWork.MenuItem(m, "Python Documentation", None, self.domenu_localdocs)
315 docitem.enable(docs)
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000316 finditem = FrameWork.MenuItem(m, "Lookup in Python Documentation", None, 'lookuppython')
317 finditem.enable(docs)
Jack Jansen7f677f42002-09-06 23:03:32 +0000318 if runningOnOSX():
319 FrameWork.Separator(m)
320 doc2item = FrameWork.MenuItem(m, "Apple Developer Documentation", None, self.domenu_appledocs)
321 find2item = FrameWork.MenuItem(m, "Lookup in Carbon Documentation", None, 'lookupcarbon')
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000322 FrameWork.Separator(m)
323 webitem = FrameWork.MenuItem(m, "Python Documentation on the Web", None, self.domenu_webdocs)
324 web2item = FrameWork.MenuItem(m, "Python on the Web", None, self.domenu_webpython)
325 web3item = FrameWork.MenuItem(m, "MacPython on the Web", None, self.domenu_webmacpython)
326
327 def domenu_localdocs(self, *args):
328 from Carbon import AH
Jack Jansenec694c32002-09-11 22:05:59 +0000329 AH.AHGotoPage("Python Help", None, None)
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000330
331 def domenu_appledocs(self, *args):
332 from Carbon import AH, AppleHelp
333 try:
334 AH.AHGotoMainTOC(AppleHelp.kAHTOCTypeDeveloper)
335 except AH.Error, arg:
336 if arg[0] == -50:
337 W.Message("Developer documentation not installed")
338 else:
339 W.Message("AppleHelp Error: %s" % `arg`)
340
341 def domenu_lookuppython(self, *args):
342 from Carbon import AH
343 searchstring = self._getsearchstring()
344 if not searchstring:
345 return
346 try:
347 AH.AHSearch("Python Help", searchstring)
348 except AH.Error, arg:
349 W.Message("AppleHelp Error: %s" % `arg`)
350
351 def domenu_lookupcarbon(self, *args):
352 from Carbon import AH
353 searchstring = self._getsearchstring()
354 if not searchstring:
355 return
356 try:
357 AH.AHSearch("Carbon", searchstring)
358 except AH.Error, arg:
359 W.Message("AppleHelp Error: %s" % `arg`)
360
361 def _getsearchstring(self):
362 import PyEdit
363 editor = PyEdit.findeditor(None, fromtop=1)
364 if editor:
365 text = editor.getselectedtext()
366 if text:
367 return text
368 # This is a cop-out. We should have disabled the menus
369 # if there is no selection, but the can_ methods only seem
370 # to work for Windows. Or not for the Help menu, maybe?
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000371 text = EasyDialogs.AskString("Search documentation for", ok="Search")
372 return text
373
374 def domenu_webdocs(self, *args):
375 import webbrowser
376 major, minor, micro, state, nano = sys.version_info
377 if state in ('alpha', 'beta'):
378 docversion = 'dev/doc/devel'
379 elif micro == 0:
380 docversion = 'doc/%d.%d' % (major, minor)
381 else:
382 docversion = 'doc/%d.%d.%d' % (major, minor, micro)
383 webbrowser.open("http://www.python.org/%s" % docversion)
384
385 def domenu_webpython(self, *args):
386 import webbrowser
387 webbrowser.open("http://www.python.org/")
388
389 def domenu_webmacpython(self, *args):
390 import webbrowser
391 webbrowser.open("http://www.cwi.nl/~jack/macpython.html")
392
393 def installdocumentation(self):
394 # This is rather much of a hack. Someone has to tell the Help Viewer
395 # about the Python documentation, so why not us. The documentation
396 # is located in the framework, but there's a symlink in Python.app.
397 # And as AHRegisterHelpBook wants a bundle (with the right bits in
398 # the plist file) we refer it to Python.app
399 python_app = os.path.join(sys.prefix, 'Resources/Python.app')
400 doc_source = os.path.join(python_app, 'Contents/Resources/English.lproj/Documentation')
401 if not os.path.isdir(doc_source):
402 return 0
403 try:
404 from Carbon import AH
405 AH.AHRegisterHelpBook(python_app)
406 except (ImportError, MacOS.Error), arg:
407 W.Message("Cannot register Python documentation: %s" % `arg`)
408 return 0
409 return 1
410
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000411
412PythonIDE()
413