blob: c79d9fcba55ff0bded506b279ccd2c67918ec230 [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')
Jack Jansen113af982003-02-12 12:47:56 +000085 FrameWork.Separator(m)
86 instmgritem = FrameWork.MenuItem(m, "Package Manager", None, 'openpackagemanager')
Just van Rossumbf0a9082002-02-04 12:48:06 +000087 if not runningOnOSX():
88 # On OSX there's a special "magic" quit menu, so we shouldn't add
89 # it to the File menu.
90 FrameWork.Separator(m)
91 quititem = FrameWork.MenuItem(m, "Quit", "Q", 'quit')
Just van Rossum40f9b7b1999-01-30 22:39:17 +000092
93 m = Wapplication.Menu(self.menubar, "Edit")
94 undoitem = FrameWork.MenuItem(m, "Undo", 'Z', "undo")
95 FrameWork.Separator(m)
96 cutitem = FrameWork.MenuItem(m, "Cut", 'X', "cut")
97 copyitem = FrameWork.MenuItem(m, "Copy", "C", "copy")
98 pasteitem = FrameWork.MenuItem(m, "Paste", "V", "paste")
99 FrameWork.MenuItem(m, "Clear", None, "clear")
100 FrameWork.Separator(m)
101 selallitem = FrameWork.MenuItem(m, "Select all", "A", "selectall")
102 sellineitem = FrameWork.MenuItem(m, "Select line", "L", "selectline")
103 FrameWork.Separator(m)
Jack Jansen815d2bf2002-01-21 23:00:52 +0000104 finditem = FrameWork.MenuItem(m, "Find"+ELIPSES, "F", "find")
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000105 findagainitem = FrameWork.MenuItem(m, "Find again", 'G', "findnext")
106 enterselitem = FrameWork.MenuItem(m, "Enter search string", "E", "entersearchstring")
107 replaceitem = FrameWork.MenuItem(m, "Replace", None, "replace")
108 replacefinditem = FrameWork.MenuItem(m, "Replace & find again", 'T', "replacefind")
109 FrameWork.Separator(m)
110 shiftleftitem = FrameWork.MenuItem(m, "Shift left", "[", "shiftleft")
111 shiftrightitem = FrameWork.MenuItem(m, "Shift right", "]", "shiftright")
112
113 m = Wapplication.Menu(self.menubar, "Python")
114 runitem = FrameWork.MenuItem(m, "Run window", "R", 'run')
115 runselitem = FrameWork.MenuItem(m, "Run selection", None, 'runselection')
116 FrameWork.Separator(m)
Jack Jansen815d2bf2002-01-21 23:00:52 +0000117 moditem = FrameWork.MenuItem(m, "Module browser"+ELIPSES, "M", self.domenu_modulebrowser)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000118 FrameWork.Separator(m)
119 mm = FrameWork.SubMenu(m, "Preferences")
Jack Jansen815d2bf2002-01-21 23:00:52 +0000120 FrameWork.MenuItem(mm, "Set Scripts folder"+ELIPSES, None, self.do_setscriptsfolder)
121 FrameWork.MenuItem(mm, "Editor default settings"+ELIPSES, None, self.do_editorprefs)
122 FrameWork.MenuItem(mm, "Set default window font"+ELIPSES, None, self.do_setwindowfont)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000123
124 self.openwindowsmenu = Wapplication.Menu(self.menubar, 'Windows')
125 self.makeopenwindowsmenu()
126 self._menustocheck = [closeitem, saveitem, saveasitem, saveasappletitem,
127 undoitem, cutitem, copyitem, pasteitem,
128 selallitem, sellineitem,
129 finditem, findagainitem, enterselitem, replaceitem, replacefinditem,
130 shiftleftitem, shiftrightitem,
131 runitem, runselitem]
132
133 prefs = self.getprefs()
134 try:
Jack Jansene7ee17c2003-02-06 22:32:35 +0000135 fsr, d = File.Alias(rawdata=prefs.scriptsfolder).FSResolveAlias(None)
136 self.scriptsfolder = fsr.FSNewAliasMinimal()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000137 except:
Jack Jansene7ee17c2003-02-06 22:32:35 +0000138 path = os.path.join(os.getcwd(), "Mac", "IDE scripts")
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000139 if not os.path.exists(path):
Just van Rossum2028b591999-09-26 12:16:22 +0000140 path = os.path.join(os.getcwd(), "Scripts")
141 if not os.path.exists(path):
142 os.mkdir(path)
Jack Jansen815d2bf2002-01-21 23:00:52 +0000143 f = open(os.path.join(path, "Place your scripts here"+ELIPSES), "w")
Just van Rossum2028b591999-09-26 12:16:22 +0000144 f.close()
Jack Jansene7ee17c2003-02-06 22:32:35 +0000145 fsr = File.FSRef(path)
146 self.scriptsfolder = fsr.FSNewAliasMinimal()
147 self.scriptsfoldermodtime = getmodtime(fsr)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000148 else:
Jack Jansene7ee17c2003-02-06 22:32:35 +0000149 self.scriptsfoldermodtime = getmodtime(fsr)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000150 prefs.scriptsfolder = self.scriptsfolder.data
151 self._scripts = {}
152 self.scriptsmenu = None
153 self.makescriptsmenu()
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000154 self.makehelpmenu()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000155
156 def quitevent(self, theAppleEvent, theReply):
Jack Jansen5a6fdcd2001-08-25 12:15:04 +0000157 from Carbon import AE
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000158 AE.AEInteractWithUser(50000000)
159 self._quit()
160
161 def suspendresume(self, onoff):
162 if onoff:
Jack Jansene7ee17c2003-02-06 22:32:35 +0000163 fsr, changed = self.scriptsfolder.FSResolveAlias(None)
164 modtime = getmodtime(fsr)
165 if self.scriptsfoldermodtime <> modtime or changed:
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000166 self.scriptsfoldermodtime = modtime
167 W.SetCursor('watch')
168 self.makescriptsmenu()
169
170 def ignoreevent(self, theAppleEvent, theReply):
171 pass
172
173 def opendocsevent(self, theAppleEvent, theReply):
174 W.SetCursor('watch')
175 import aetools
176 parameters, args = aetools.unpackevent(theAppleEvent)
177 docs = parameters['----']
178 if type(docs) <> type([]):
179 docs = [docs]
180 for doc in docs:
Jack Jansene7ee17c2003-02-06 22:32:35 +0000181 fsr, a = doc.FSResolveAlias(None)
182 path = fsr.as_pathname()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000183 self.opendoc(path)
184
185 def opendoc(self, path):
Jack Jansene7ee17c2003-02-06 22:32:35 +0000186 fcreator, ftype = MacOS.GetCreatorAndType(path)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000187 if ftype == 'TEXT':
188 self.openscript(path)
Jack Jansene0ba0872002-03-29 21:23:47 +0000189 elif ftype == '\0\0\0\0' and path[-3:] == '.py':
190 self.openscript(path)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000191 else:
Just van Rossumdc3c6172001-06-19 21:37:33 +0000192 W.Message("Can't open file of type '%s'." % ftype)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000193
194 def getabouttext(self):
Jack Jansen815d2bf2002-01-21 23:00:52 +0000195 return "About Python IDE"+ELIPSES
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000196
197 def do_about(self, id, item, window, event):
198 Splash.about()
199
200 def do_setscriptsfolder(self, *args):
Jack Jansene7ee17c2003-02-06 22:32:35 +0000201 fsr = EasyDialogs.AskFolder(message="Select Scripts Folder",
202 wanted=File.FSRef)
203 if fsr:
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000204 prefs = self.getprefs()
Jack Jansene7ee17c2003-02-06 22:32:35 +0000205 alis = fsr.FSNewAliasMinimal()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000206 prefs.scriptsfolder = alis.data
207 self.scriptsfolder = alis
208 self.makescriptsmenu()
209 prefs.save()
210
211 def domenu_modulebrowser(self, *args):
212 W.SetCursor('watch')
213 import ModuleBrowser
214 ModuleBrowser.ModuleBrowser()
215
216 def domenu_open(self, *args):
Jack Jansenfd0b00e2003-01-26 22:15:48 +0000217 filename = EasyDialogs.AskFileForOpen(typeList=("TEXT",))
218 if filename:
219 self.openscript(filename)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000220
Jack Jansen611b9f62003-02-05 15:41:09 +0000221 def domenu_openbyname(self, *args):
222 # Open a file by name. If the clipboard contains a filename
223 # use that as the default.
224 from Carbon import Scrap
225 try:
226 sc = Scrap.GetCurrentScrap()
227 dft = sc.GetScrapFlavorData("TEXT")
228 except Scrap.Error:
229 dft = ""
230 else:
231 if not os.path.exists(dft):
232 dft = ""
233 filename = EasyDialogs.AskString("Open File Named:", default=dft, ok="Open")
234 if filename:
235 self.openscript(filename)
236
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000237 def domenu_new(self, *args):
238 W.SetCursor('watch')
239 import PyEdit
240 return PyEdit.Editor()
241
242 def makescriptsmenu(self):
243 W.SetCursor('watch')
244 if self._scripts:
245 for id, item in self._scripts.keys():
246 if self.menubar.menus.has_key(id):
247 m = self.menubar.menus[id]
248 m.delete()
249 self._scripts = {}
250 if self.scriptsmenu:
251 if hasattr(self.scriptsmenu, 'id') and self.menubar.menus.has_key(self.scriptsmenu.id):
252 self.scriptsmenu.delete()
253 self.scriptsmenu = FrameWork.Menu(self.menubar, "Scripts")
254 #FrameWork.MenuItem(self.scriptsmenu, "New script", None, self.domenu_new)
255 #self.scriptsmenu.addseparator()
Jack Jansene7ee17c2003-02-06 22:32:35 +0000256 fsr, d1 = self.scriptsfolder.FSResolveAlias(None)
257 self.scriptswalk(fsr.as_pathname(), self.scriptsmenu)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000258
259 def makeopenwindowsmenu(self):
260 for i in range(len(self.openwindowsmenu.items)):
261 self.openwindowsmenu.menu.DeleteMenuItem(1)
262 self.openwindowsmenu.items = []
263 windows = []
264 self._openwindows = {}
265 for window in self._windows.keys():
266 title = window.GetWTitle()
267 if not title:
268 title = "<no title>"
Jack Jansen34d11f02000-03-07 23:40:13 +0000269 windows.append((title, window))
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000270 windows.sort()
271 for title, window in windows:
272 if title == "Python Interactive": # ugly but useful hack by Joe Strout
273 shortcut = '0'
274 else:
275 shortcut = None
276 item = FrameWork.MenuItem(self.openwindowsmenu, title, shortcut, callback = self.domenu_openwindows)
277 self._openwindows[item.item] = window
278 self._openwindowscheckmark = 0
279 self.checkopenwindowsmenu()
280
281 def domenu_openwindows(self, id, item, window, event):
282 w = self._openwindows[item]
283 w.ShowWindow()
284 w.SelectWindow()
285
286 def domenu_quit(self):
287 self._quit()
288
289 def domenu_save(self, *args):
290 print "Save"
291
292 def _quit(self):
293 import PyConsole, PyEdit
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000294 for window in self._windows.values():
Just van Rossumd58c7461999-06-22 18:37:35 +0000295 try:
296 rv = window.close() # ignore any errors while quitting
297 except:
298 rv = 0 # (otherwise, we can get stuck!)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000299 if rv and rv > 0:
300 return
Just van Rossum62a103b2002-11-22 12:48:47 +0000301 try:
302 PyConsole.console.writeprefs()
303 PyConsole.output.writeprefs()
304 PyEdit.searchengine.writeprefs()
305 except:
306 # Write to __stderr__ so the msg end up in Console.app and has
307 # at least _some_ chance of getting read...
308 # But: this is a workaround for way more serious problems with
309 # the Python 2.2 Jaguar addon.
310 sys.__stderr__.write("*** PythonIDE: Can't write preferences ***\n")
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000311 self.quitting = 1
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000312
Jack Jansen113af982003-02-12 12:47:56 +0000313 def domenu_openpackagemanager(self):
314 import PackageManager
315 PackageManager.PackageBrowser()
316 print "Done"
317
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000318 def makehelpmenu(self):
319 docs = self.installdocumentation()
320 self.helpmenu = m = self.gethelpmenu()
321 docitem = FrameWork.MenuItem(m, "Python Documentation", None, self.domenu_localdocs)
322 docitem.enable(docs)
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000323 finditem = FrameWork.MenuItem(m, "Lookup in Python Documentation", None, 'lookuppython')
324 finditem.enable(docs)
Jack Jansen7f677f42002-09-06 23:03:32 +0000325 if runningOnOSX():
326 FrameWork.Separator(m)
327 doc2item = FrameWork.MenuItem(m, "Apple Developer Documentation", None, self.domenu_appledocs)
328 find2item = FrameWork.MenuItem(m, "Lookup in Carbon Documentation", None, 'lookupcarbon')
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000329 FrameWork.Separator(m)
330 webitem = FrameWork.MenuItem(m, "Python Documentation on the Web", None, self.domenu_webdocs)
331 web2item = FrameWork.MenuItem(m, "Python on the Web", None, self.domenu_webpython)
332 web3item = FrameWork.MenuItem(m, "MacPython on the Web", None, self.domenu_webmacpython)
333
334 def domenu_localdocs(self, *args):
335 from Carbon import AH
Jack Jansenec694c32002-09-11 22:05:59 +0000336 AH.AHGotoPage("Python Help", None, None)
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000337
338 def domenu_appledocs(self, *args):
339 from Carbon import AH, AppleHelp
340 try:
341 AH.AHGotoMainTOC(AppleHelp.kAHTOCTypeDeveloper)
342 except AH.Error, arg:
343 if arg[0] == -50:
344 W.Message("Developer documentation not installed")
345 else:
346 W.Message("AppleHelp Error: %s" % `arg`)
347
348 def domenu_lookuppython(self, *args):
349 from Carbon import AH
350 searchstring = self._getsearchstring()
351 if not searchstring:
352 return
353 try:
354 AH.AHSearch("Python Help", searchstring)
355 except AH.Error, arg:
356 W.Message("AppleHelp Error: %s" % `arg`)
357
358 def domenu_lookupcarbon(self, *args):
359 from Carbon import AH
360 searchstring = self._getsearchstring()
361 if not searchstring:
362 return
363 try:
364 AH.AHSearch("Carbon", searchstring)
365 except AH.Error, arg:
366 W.Message("AppleHelp Error: %s" % `arg`)
367
368 def _getsearchstring(self):
369 import PyEdit
370 editor = PyEdit.findeditor(None, fromtop=1)
371 if editor:
372 text = editor.getselectedtext()
373 if text:
374 return text
375 # This is a cop-out. We should have disabled the menus
376 # if there is no selection, but the can_ methods only seem
377 # to work for Windows. Or not for the Help menu, maybe?
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000378 text = EasyDialogs.AskString("Search documentation for", ok="Search")
379 return text
380
381 def domenu_webdocs(self, *args):
382 import webbrowser
383 major, minor, micro, state, nano = sys.version_info
384 if state in ('alpha', 'beta'):
385 docversion = 'dev/doc/devel'
386 elif micro == 0:
387 docversion = 'doc/%d.%d' % (major, minor)
388 else:
389 docversion = 'doc/%d.%d.%d' % (major, minor, micro)
390 webbrowser.open("http://www.python.org/%s" % docversion)
391
392 def domenu_webpython(self, *args):
393 import webbrowser
394 webbrowser.open("http://www.python.org/")
395
396 def domenu_webmacpython(self, *args):
397 import webbrowser
398 webbrowser.open("http://www.cwi.nl/~jack/macpython.html")
399
400 def installdocumentation(self):
401 # This is rather much of a hack. Someone has to tell the Help Viewer
402 # about the Python documentation, so why not us. The documentation
403 # is located in the framework, but there's a symlink in Python.app.
404 # And as AHRegisterHelpBook wants a bundle (with the right bits in
405 # the plist file) we refer it to Python.app
406 python_app = os.path.join(sys.prefix, 'Resources/Python.app')
407 doc_source = os.path.join(python_app, 'Contents/Resources/English.lproj/Documentation')
408 if not os.path.isdir(doc_source):
409 return 0
410 try:
411 from Carbon import AH
412 AH.AHRegisterHelpBook(python_app)
413 except (ImportError, MacOS.Error), arg:
414 W.Message("Cannot register Python documentation: %s" % `arg`)
415 return 0
416 return 1
417
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000418
419PythonIDE()
420