blob: d93dc1cd327c6a8e3ecdda6055bd5d22a6b5d071 [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
Just van Rossum40f9b7b1999-01-30 22:39:17 +000010import macfs
Jack Jansen815d2bf2002-01-21 23:00:52 +000011import MacOS
Just van Rossum40f9b7b1999-01-30 22:39:17 +000012
Jack Jansen815d2bf2002-01-21 23:00:52 +000013if MacOS.runtimemodel == 'macho':
14 ELIPSES = '...'
15else:
16 ELIPSES = '\xc9'
Just van Rossum40f9b7b1999-01-30 22:39:17 +000017
Just van Rossumbf0a9082002-02-04 12:48:06 +000018def runningOnOSX():
19 from gestalt import gestalt
20 gestaltMenuMgrAquaLayoutBit = 1 # menus have the Aqua 1.0 layout
21 gestaltMenuMgrAquaLayoutMask = (1L << gestaltMenuMgrAquaLayoutBit)
22 value = gestalt("menu") & gestaltMenuMgrAquaLayoutMask
23 return not not value
24
25
Just van Rossum40f9b7b1999-01-30 22:39:17 +000026class PythonIDE(Wapplication.Application):
27
28 def __init__(self):
Just van Rossum979c5372002-07-12 16:50:32 +000029 self.preffilepath = os.path.join("Python", "PythonIDE preferences")
Just van Rossumf4b06811999-02-27 17:16:54 +000030 Wapplication.Application.__init__(self, 'Pide')
Jack Jansen5a6fdcd2001-08-25 12:15:04 +000031 from Carbon import AE
32 from Carbon import AppleEvents
Just van Rossum40f9b7b1999-01-30 22:39:17 +000033
34 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEOpenApplication,
35 self.ignoreevent)
Jack Jansen53ebe562001-03-08 23:09:32 +000036 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEReopenApplication,
37 self.ignoreevent)
Just van Rossum40f9b7b1999-01-30 22:39:17 +000038 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEPrintDocuments,
39 self.ignoreevent)
40 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEOpenDocuments,
41 self.opendocsevent)
42 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEQuitApplication,
43 self.quitevent)
44 import PyConsole, PyEdit
45 Splash.wait()
Jack Jansene0ba0872002-03-29 21:23:47 +000046 # With -D option (OSX command line only) keep stderr, for debugging the IDE
47 # itself.
48 debug_stderr = None
Just van Rossum8bb61c82002-03-29 21:47:56 +000049 if len(sys.argv) >= 2 and sys.argv[1] == '-D':
Jack Jansene0ba0872002-03-29 21:23:47 +000050 debug_stderr = sys.stderr
51 del sys.argv[1]
Just van Rossum40f9b7b1999-01-30 22:39:17 +000052 PyConsole.installoutput()
53 PyConsole.installconsole()
Jack Jansene0ba0872002-03-29 21:23:47 +000054 if debug_stderr:
55 sys.stderr = debug_stderr
Just van Rossum40f9b7b1999-01-30 22:39:17 +000056 for path in sys.argv[1:]:
57 self.opendoc(path)
Just van Rossum73efed22000-04-09 19:45:22 +000058 try:
Just van Rossuma1a33562000-10-20 06:37:11 +000059 import Wthreading
Just van Rossum73efed22000-04-09 19:45:22 +000060 except ImportError:
61 self.mainloop()
62 else:
Just van Rossuma1a33562000-10-20 06:37:11 +000063 if Wthreading.haveThreading:
64 self.mainthread = Wthreading.Thread("IDE event loop", self.mainloop)
65 self.mainthread.start()
66 #self.mainthread.setResistant(1)
67 Wthreading.run()
68 else:
69 self.mainloop()
Just van Rossum40f9b7b1999-01-30 22:39:17 +000070
71 def makeusermenus(self):
72 m = Wapplication.Menu(self.menubar, "File")
73 newitem = FrameWork.MenuItem(m, "New", "N", 'new')
Jack Jansen815d2bf2002-01-21 23:00:52 +000074 openitem = FrameWork.MenuItem(m, "Open"+ELIPSES, "O", 'open')
Just van Rossum40f9b7b1999-01-30 22:39:17 +000075 FrameWork.Separator(m)
76 closeitem = FrameWork.MenuItem(m, "Close", "W", 'close')
77 saveitem = FrameWork.MenuItem(m, "Save", "S", 'save')
Jack Jansen815d2bf2002-01-21 23:00:52 +000078 saveasitem = FrameWork.MenuItem(m, "Save as"+ELIPSES, None, 'save_as')
Just van Rossum40f9b7b1999-01-30 22:39:17 +000079 FrameWork.Separator(m)
Jack Jansen815d2bf2002-01-21 23:00:52 +000080 saveasappletitem = FrameWork.MenuItem(m, "Save as Applet"+ELIPSES, None, 'save_as_applet')
Just van Rossumbf0a9082002-02-04 12:48:06 +000081 if not runningOnOSX():
82 # On OSX there's a special "magic" quit menu, so we shouldn't add
83 # it to the File menu.
84 FrameWork.Separator(m)
85 quititem = FrameWork.MenuItem(m, "Quit", "Q", 'quit')
Just van Rossum40f9b7b1999-01-30 22:39:17 +000086
87 m = Wapplication.Menu(self.menubar, "Edit")
88 undoitem = FrameWork.MenuItem(m, "Undo", 'Z', "undo")
89 FrameWork.Separator(m)
90 cutitem = FrameWork.MenuItem(m, "Cut", 'X', "cut")
91 copyitem = FrameWork.MenuItem(m, "Copy", "C", "copy")
92 pasteitem = FrameWork.MenuItem(m, "Paste", "V", "paste")
93 FrameWork.MenuItem(m, "Clear", None, "clear")
94 FrameWork.Separator(m)
95 selallitem = FrameWork.MenuItem(m, "Select all", "A", "selectall")
96 sellineitem = FrameWork.MenuItem(m, "Select line", "L", "selectline")
97 FrameWork.Separator(m)
Jack Jansen815d2bf2002-01-21 23:00:52 +000098 finditem = FrameWork.MenuItem(m, "Find"+ELIPSES, "F", "find")
Just van Rossum40f9b7b1999-01-30 22:39:17 +000099 findagainitem = FrameWork.MenuItem(m, "Find again", 'G', "findnext")
100 enterselitem = FrameWork.MenuItem(m, "Enter search string", "E", "entersearchstring")
101 replaceitem = FrameWork.MenuItem(m, "Replace", None, "replace")
102 replacefinditem = FrameWork.MenuItem(m, "Replace & find again", 'T', "replacefind")
103 FrameWork.Separator(m)
104 shiftleftitem = FrameWork.MenuItem(m, "Shift left", "[", "shiftleft")
105 shiftrightitem = FrameWork.MenuItem(m, "Shift right", "]", "shiftright")
106
107 m = Wapplication.Menu(self.menubar, "Python")
108 runitem = FrameWork.MenuItem(m, "Run window", "R", 'run')
109 runselitem = FrameWork.MenuItem(m, "Run selection", None, 'runselection')
110 FrameWork.Separator(m)
Jack Jansen815d2bf2002-01-21 23:00:52 +0000111 moditem = FrameWork.MenuItem(m, "Module browser"+ELIPSES, "M", self.domenu_modulebrowser)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000112 FrameWork.Separator(m)
113 mm = FrameWork.SubMenu(m, "Preferences")
Jack Jansen815d2bf2002-01-21 23:00:52 +0000114 FrameWork.MenuItem(mm, "Set Scripts folder"+ELIPSES, None, self.do_setscriptsfolder)
115 FrameWork.MenuItem(mm, "Editor default settings"+ELIPSES, None, self.do_editorprefs)
116 FrameWork.MenuItem(mm, "Set default window font"+ELIPSES, None, self.do_setwindowfont)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000117
118 self.openwindowsmenu = Wapplication.Menu(self.menubar, 'Windows')
119 self.makeopenwindowsmenu()
120 self._menustocheck = [closeitem, saveitem, saveasitem, saveasappletitem,
121 undoitem, cutitem, copyitem, pasteitem,
122 selallitem, sellineitem,
123 finditem, findagainitem, enterselitem, replaceitem, replacefinditem,
124 shiftleftitem, shiftrightitem,
125 runitem, runselitem]
126
127 prefs = self.getprefs()
128 try:
129 fss, fss_changed = macfs.RawAlias(prefs.scriptsfolder).Resolve()
Just van Rossum68922f01999-02-25 22:33:05 +0000130 self.scriptsfolder = fss.NewAlias()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000131 except:
Just van Rossum2028b591999-09-26 12:16:22 +0000132 path = os.path.join(os.getcwd(), ":Mac:IDE scripts")
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000133 if not os.path.exists(path):
Just van Rossum2028b591999-09-26 12:16:22 +0000134 path = os.path.join(os.getcwd(), "Scripts")
135 if not os.path.exists(path):
136 os.mkdir(path)
Jack Jansen815d2bf2002-01-21 23:00:52 +0000137 f = open(os.path.join(path, "Place your scripts here"+ELIPSES), "w")
Just van Rossum2028b591999-09-26 12:16:22 +0000138 f.close()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000139 fss = macfs.FSSpec(path)
140 self.scriptsfolder = fss.NewAlias()
141 self.scriptsfoldermodtime = fss.GetDates()[1]
142 else:
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000143 self.scriptsfoldermodtime = fss.GetDates()[1]
144 prefs.scriptsfolder = self.scriptsfolder.data
145 self._scripts = {}
146 self.scriptsmenu = None
147 self.makescriptsmenu()
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000148 self.makehelpmenu()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000149
150 def quitevent(self, theAppleEvent, theReply):
Jack Jansen5a6fdcd2001-08-25 12:15:04 +0000151 from Carbon import AE
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000152 AE.AEInteractWithUser(50000000)
153 self._quit()
154
155 def suspendresume(self, onoff):
156 if onoff:
157 fss, fss_changed = self.scriptsfolder.Resolve()
158 modtime = fss.GetDates()[1]
159 if self.scriptsfoldermodtime <> modtime or fss_changed:
160 self.scriptsfoldermodtime = modtime
161 W.SetCursor('watch')
162 self.makescriptsmenu()
163
164 def ignoreevent(self, theAppleEvent, theReply):
165 pass
166
167 def opendocsevent(self, theAppleEvent, theReply):
168 W.SetCursor('watch')
169 import aetools
170 parameters, args = aetools.unpackevent(theAppleEvent)
171 docs = parameters['----']
172 if type(docs) <> type([]):
173 docs = [docs]
174 for doc in docs:
175 fss, a = doc.Resolve()
176 path = fss.as_pathname()
177 self.opendoc(path)
178
179 def opendoc(self, path):
180 fcreator, ftype = macfs.FSSpec(path).GetCreatorType()
181 if ftype == 'TEXT':
182 self.openscript(path)
Jack Jansene0ba0872002-03-29 21:23:47 +0000183 elif ftype == '\0\0\0\0' and path[-3:] == '.py':
184 self.openscript(path)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000185 else:
Just van Rossumdc3c6172001-06-19 21:37:33 +0000186 W.Message("Can't open file of type '%s'." % ftype)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000187
188 def getabouttext(self):
Jack Jansen815d2bf2002-01-21 23:00:52 +0000189 return "About Python IDE"+ELIPSES
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000190
191 def do_about(self, id, item, window, event):
192 Splash.about()
193
194 def do_setscriptsfolder(self, *args):
195 fss, ok = macfs.GetDirectory("Select Scripts Folder")
196 if ok:
197 prefs = self.getprefs()
198 alis = fss.NewAlias()
199 prefs.scriptsfolder = alis.data
200 self.scriptsfolder = alis
201 self.makescriptsmenu()
202 prefs.save()
203
204 def domenu_modulebrowser(self, *args):
205 W.SetCursor('watch')
206 import ModuleBrowser
207 ModuleBrowser.ModuleBrowser()
208
209 def domenu_open(self, *args):
210 fss, ok = macfs.StandardGetFile("TEXT")
211 if ok:
212 self.openscript(fss.as_pathname())
213
214 def domenu_new(self, *args):
215 W.SetCursor('watch')
216 import PyEdit
217 return PyEdit.Editor()
218
219 def makescriptsmenu(self):
220 W.SetCursor('watch')
221 if self._scripts:
222 for id, item in self._scripts.keys():
223 if self.menubar.menus.has_key(id):
224 m = self.menubar.menus[id]
225 m.delete()
226 self._scripts = {}
227 if self.scriptsmenu:
228 if hasattr(self.scriptsmenu, 'id') and self.menubar.menus.has_key(self.scriptsmenu.id):
229 self.scriptsmenu.delete()
230 self.scriptsmenu = FrameWork.Menu(self.menubar, "Scripts")
231 #FrameWork.MenuItem(self.scriptsmenu, "New script", None, self.domenu_new)
232 #self.scriptsmenu.addseparator()
233 fss, fss_changed = self.scriptsfolder.Resolve()
234 self.scriptswalk(fss.as_pathname(), self.scriptsmenu)
235
236 def makeopenwindowsmenu(self):
237 for i in range(len(self.openwindowsmenu.items)):
238 self.openwindowsmenu.menu.DeleteMenuItem(1)
239 self.openwindowsmenu.items = []
240 windows = []
241 self._openwindows = {}
242 for window in self._windows.keys():
243 title = window.GetWTitle()
244 if not title:
245 title = "<no title>"
Jack Jansen34d11f02000-03-07 23:40:13 +0000246 windows.append((title, window))
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000247 windows.sort()
248 for title, window in windows:
249 if title == "Python Interactive": # ugly but useful hack by Joe Strout
250 shortcut = '0'
251 else:
252 shortcut = None
253 item = FrameWork.MenuItem(self.openwindowsmenu, title, shortcut, callback = self.domenu_openwindows)
254 self._openwindows[item.item] = window
255 self._openwindowscheckmark = 0
256 self.checkopenwindowsmenu()
257
258 def domenu_openwindows(self, id, item, window, event):
259 w = self._openwindows[item]
260 w.ShowWindow()
261 w.SelectWindow()
262
263 def domenu_quit(self):
264 self._quit()
265
266 def domenu_save(self, *args):
267 print "Save"
268
269 def _quit(self):
270 import PyConsole, PyEdit
271 PyConsole.console.writeprefs()
272 PyConsole.output.writeprefs()
273 PyEdit.searchengine.writeprefs()
274 for window in self._windows.values():
Just van Rossumd58c7461999-06-22 18:37:35 +0000275 try:
276 rv = window.close() # ignore any errors while quitting
277 except:
278 rv = 0 # (otherwise, we can get stuck!)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000279 if rv and rv > 0:
280 return
281 self.quitting = 1
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000282
283 def makehelpmenu(self):
284 docs = self.installdocumentation()
285 self.helpmenu = m = self.gethelpmenu()
286 docitem = FrameWork.MenuItem(m, "Python Documentation", None, self.domenu_localdocs)
287 docitem.enable(docs)
288 doc2item = FrameWork.MenuItem(m, "Apple Developer Documentation", None, self.domenu_appledocs)
289 FrameWork.Separator(m)
290 finditem = FrameWork.MenuItem(m, "Lookup in Python Documentation", None, 'lookuppython')
291 finditem.enable(docs)
292 find2item = FrameWork.MenuItem(m, "Lookup in Carbon Documentation", None, 'lookupcarbon')
293 FrameWork.Separator(m)
294 webitem = FrameWork.MenuItem(m, "Python Documentation on the Web", None, self.domenu_webdocs)
295 web2item = FrameWork.MenuItem(m, "Python on the Web", None, self.domenu_webpython)
296 web3item = FrameWork.MenuItem(m, "MacPython on the Web", None, self.domenu_webmacpython)
297
298 def domenu_localdocs(self, *args):
299 from Carbon import AH
300 AH.AHGotoPage("Python Help", "index.html", "")
301
302 def domenu_appledocs(self, *args):
303 from Carbon import AH, AppleHelp
304 try:
305 AH.AHGotoMainTOC(AppleHelp.kAHTOCTypeDeveloper)
306 except AH.Error, arg:
307 if arg[0] == -50:
308 W.Message("Developer documentation not installed")
309 else:
310 W.Message("AppleHelp Error: %s" % `arg`)
311
312 def domenu_lookuppython(self, *args):
313 from Carbon import AH
314 searchstring = self._getsearchstring()
315 if not searchstring:
316 return
317 try:
318 AH.AHSearch("Python Help", searchstring)
319 except AH.Error, arg:
320 W.Message("AppleHelp Error: %s" % `arg`)
321
322 def domenu_lookupcarbon(self, *args):
323 from Carbon import AH
324 searchstring = self._getsearchstring()
325 if not searchstring:
326 return
327 try:
328 AH.AHSearch("Carbon", searchstring)
329 except AH.Error, arg:
330 W.Message("AppleHelp Error: %s" % `arg`)
331
332 def _getsearchstring(self):
333 import PyEdit
334 editor = PyEdit.findeditor(None, fromtop=1)
335 if editor:
336 text = editor.getselectedtext()
337 if text:
338 return text
339 # This is a cop-out. We should have disabled the menus
340 # if there is no selection, but the can_ methods only seem
341 # to work for Windows. Or not for the Help menu, maybe?
342 import EasyDialogs
343 text = EasyDialogs.AskString("Search documentation for", ok="Search")
344 return text
345
346 def domenu_webdocs(self, *args):
347 import webbrowser
348 major, minor, micro, state, nano = sys.version_info
349 if state in ('alpha', 'beta'):
350 docversion = 'dev/doc/devel'
351 elif micro == 0:
352 docversion = 'doc/%d.%d' % (major, minor)
353 else:
354 docversion = 'doc/%d.%d.%d' % (major, minor, micro)
355 webbrowser.open("http://www.python.org/%s" % docversion)
356
357 def domenu_webpython(self, *args):
358 import webbrowser
359 webbrowser.open("http://www.python.org/")
360
361 def domenu_webmacpython(self, *args):
362 import webbrowser
363 webbrowser.open("http://www.cwi.nl/~jack/macpython.html")
364
365 def installdocumentation(self):
366 # This is rather much of a hack. Someone has to tell the Help Viewer
367 # about the Python documentation, so why not us. The documentation
368 # is located in the framework, but there's a symlink in Python.app.
369 # And as AHRegisterHelpBook wants a bundle (with the right bits in
370 # the plist file) we refer it to Python.app
371 python_app = os.path.join(sys.prefix, 'Resources/Python.app')
372 doc_source = os.path.join(python_app, 'Contents/Resources/English.lproj/Documentation')
373 if not os.path.isdir(doc_source):
374 return 0
375 try:
376 from Carbon import AH
377 AH.AHRegisterHelpBook(python_app)
378 except (ImportError, MacOS.Error), arg:
379 W.Message("Cannot register Python documentation: %s" % `arg`)
380 return 0
381 return 1
382
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000383
384PythonIDE()
385