blob: 765d9887dc23a4b1dbcf507b5fbbc8677cc9dfa5 [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
Jack Jansenfd0b00e2003-01-26 22:15:48 +000012import EasyDialogs
Just van Rossum40f9b7b1999-01-30 22:39:17 +000013
Jack Jansen815d2bf2002-01-21 23:00:52 +000014if MacOS.runtimemodel == 'macho':
15 ELIPSES = '...'
16else:
17 ELIPSES = '\xc9'
Just van Rossum40f9b7b1999-01-30 22:39:17 +000018
Just van Rossumbf0a9082002-02-04 12:48:06 +000019def runningOnOSX():
20 from gestalt import gestalt
21 gestaltMenuMgrAquaLayoutBit = 1 # menus have the Aqua 1.0 layout
22 gestaltMenuMgrAquaLayoutMask = (1L << gestaltMenuMgrAquaLayoutBit)
23 value = gestalt("menu") & gestaltMenuMgrAquaLayoutMask
24 return not not value
25
26
Just van Rossum40f9b7b1999-01-30 22:39:17 +000027class PythonIDE(Wapplication.Application):
28
29 def __init__(self):
Just van Rossum979c5372002-07-12 16:50:32 +000030 self.preffilepath = os.path.join("Python", "PythonIDE preferences")
Just van Rossumf4b06811999-02-27 17:16:54 +000031 Wapplication.Application.__init__(self, 'Pide')
Jack Jansen5a6fdcd2001-08-25 12:15:04 +000032 from Carbon import AE
33 from Carbon import AppleEvents
Just van Rossum40f9b7b1999-01-30 22:39:17 +000034
35 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEOpenApplication,
36 self.ignoreevent)
Jack Jansen53ebe562001-03-08 23:09:32 +000037 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEReopenApplication,
38 self.ignoreevent)
Just van Rossum40f9b7b1999-01-30 22:39:17 +000039 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEPrintDocuments,
40 self.ignoreevent)
41 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEOpenDocuments,
42 self.opendocsevent)
43 AE.AEInstallEventHandler(AppleEvents.kCoreEventClass, AppleEvents.kAEQuitApplication,
44 self.quitevent)
45 import PyConsole, PyEdit
46 Splash.wait()
Just van Rossum7aa7f2e2002-10-20 17:14:28 +000047 if sys.platform == "darwin":
48 if sys.argv and sys.argv[0].startswith("-psn"):
49 home = os.getenv("HOME")
50 if home:
51 os.chdir(home)
Jack Jansene0ba0872002-03-29 21:23:47 +000052 # With -D option (OSX command line only) keep stderr, for debugging the IDE
53 # itself.
54 debug_stderr = None
Just van Rossum8bb61c82002-03-29 21:47:56 +000055 if len(sys.argv) >= 2 and sys.argv[1] == '-D':
Jack Jansene0ba0872002-03-29 21:23:47 +000056 debug_stderr = sys.stderr
57 del sys.argv[1]
Just van Rossum40f9b7b1999-01-30 22:39:17 +000058 PyConsole.installoutput()
59 PyConsole.installconsole()
Jack Jansene0ba0872002-03-29 21:23:47 +000060 if debug_stderr:
61 sys.stderr = debug_stderr
Just van Rossum40f9b7b1999-01-30 22:39:17 +000062 for path in sys.argv[1:]:
Just van Rossum62a103b2002-11-22 12:48:47 +000063 if path.startswith("-p"):
64 # process number added by the OS
65 continue
Just van Rossum40f9b7b1999-01-30 22:39:17 +000066 self.opendoc(path)
Just van Rossum62a103b2002-11-22 12:48:47 +000067 self.mainloop()
Just van Rossum40f9b7b1999-01-30 22:39:17 +000068
69 def makeusermenus(self):
70 m = Wapplication.Menu(self.menubar, "File")
71 newitem = FrameWork.MenuItem(m, "New", "N", 'new')
Jack Jansen815d2bf2002-01-21 23:00:52 +000072 openitem = FrameWork.MenuItem(m, "Open"+ELIPSES, "O", 'open')
Jack Jansen611b9f62003-02-05 15:41:09 +000073 openbynameitem = FrameWork.MenuItem(m, "Open File by Name"+ELIPSES, "D", 'openbyname')
Just van Rossum40f9b7b1999-01-30 22:39:17 +000074 FrameWork.Separator(m)
75 closeitem = FrameWork.MenuItem(m, "Close", "W", 'close')
76 saveitem = FrameWork.MenuItem(m, "Save", "S", 'save')
Jack Jansen815d2bf2002-01-21 23:00:52 +000077 saveasitem = FrameWork.MenuItem(m, "Save as"+ELIPSES, None, 'save_as')
Just van Rossum40f9b7b1999-01-30 22:39:17 +000078 FrameWork.Separator(m)
Jack Jansen815d2bf2002-01-21 23:00:52 +000079 saveasappletitem = FrameWork.MenuItem(m, "Save as Applet"+ELIPSES, None, 'save_as_applet')
Just van Rossumbf0a9082002-02-04 12:48:06 +000080 if not runningOnOSX():
81 # On OSX there's a special "magic" quit menu, so we shouldn't add
82 # it to the File menu.
83 FrameWork.Separator(m)
84 quititem = FrameWork.MenuItem(m, "Quit", "Q", 'quit')
Just van Rossum40f9b7b1999-01-30 22:39:17 +000085
86 m = Wapplication.Menu(self.menubar, "Edit")
87 undoitem = FrameWork.MenuItem(m, "Undo", 'Z', "undo")
88 FrameWork.Separator(m)
89 cutitem = FrameWork.MenuItem(m, "Cut", 'X', "cut")
90 copyitem = FrameWork.MenuItem(m, "Copy", "C", "copy")
91 pasteitem = FrameWork.MenuItem(m, "Paste", "V", "paste")
92 FrameWork.MenuItem(m, "Clear", None, "clear")
93 FrameWork.Separator(m)
94 selallitem = FrameWork.MenuItem(m, "Select all", "A", "selectall")
95 sellineitem = FrameWork.MenuItem(m, "Select line", "L", "selectline")
96 FrameWork.Separator(m)
Jack Jansen815d2bf2002-01-21 23:00:52 +000097 finditem = FrameWork.MenuItem(m, "Find"+ELIPSES, "F", "find")
Just van Rossum40f9b7b1999-01-30 22:39:17 +000098 findagainitem = FrameWork.MenuItem(m, "Find again", 'G', "findnext")
99 enterselitem = FrameWork.MenuItem(m, "Enter search string", "E", "entersearchstring")
100 replaceitem = FrameWork.MenuItem(m, "Replace", None, "replace")
101 replacefinditem = FrameWork.MenuItem(m, "Replace & find again", 'T', "replacefind")
102 FrameWork.Separator(m)
103 shiftleftitem = FrameWork.MenuItem(m, "Shift left", "[", "shiftleft")
104 shiftrightitem = FrameWork.MenuItem(m, "Shift right", "]", "shiftright")
105
106 m = Wapplication.Menu(self.menubar, "Python")
107 runitem = FrameWork.MenuItem(m, "Run window", "R", 'run')
108 runselitem = FrameWork.MenuItem(m, "Run selection", None, 'runselection')
109 FrameWork.Separator(m)
Jack Jansen815d2bf2002-01-21 23:00:52 +0000110 moditem = FrameWork.MenuItem(m, "Module browser"+ELIPSES, "M", self.domenu_modulebrowser)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000111 FrameWork.Separator(m)
112 mm = FrameWork.SubMenu(m, "Preferences")
Jack Jansen815d2bf2002-01-21 23:00:52 +0000113 FrameWork.MenuItem(mm, "Set Scripts folder"+ELIPSES, None, self.do_setscriptsfolder)
114 FrameWork.MenuItem(mm, "Editor default settings"+ELIPSES, None, self.do_editorprefs)
115 FrameWork.MenuItem(mm, "Set default window font"+ELIPSES, None, self.do_setwindowfont)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000116
117 self.openwindowsmenu = Wapplication.Menu(self.menubar, 'Windows')
118 self.makeopenwindowsmenu()
119 self._menustocheck = [closeitem, saveitem, saveasitem, saveasappletitem,
120 undoitem, cutitem, copyitem, pasteitem,
121 selallitem, sellineitem,
122 finditem, findagainitem, enterselitem, replaceitem, replacefinditem,
123 shiftleftitem, shiftrightitem,
124 runitem, runselitem]
125
126 prefs = self.getprefs()
127 try:
128 fss, fss_changed = macfs.RawAlias(prefs.scriptsfolder).Resolve()
Just van Rossum68922f01999-02-25 22:33:05 +0000129 self.scriptsfolder = fss.NewAlias()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000130 except:
Just van Rossum2028b591999-09-26 12:16:22 +0000131 path = os.path.join(os.getcwd(), ":Mac:IDE scripts")
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000132 if not os.path.exists(path):
Just van Rossum2028b591999-09-26 12:16:22 +0000133 path = os.path.join(os.getcwd(), "Scripts")
134 if not os.path.exists(path):
135 os.mkdir(path)
Jack Jansen815d2bf2002-01-21 23:00:52 +0000136 f = open(os.path.join(path, "Place your scripts here"+ELIPSES), "w")
Just van Rossum2028b591999-09-26 12:16:22 +0000137 f.close()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000138 fss = macfs.FSSpec(path)
139 self.scriptsfolder = fss.NewAlias()
140 self.scriptsfoldermodtime = fss.GetDates()[1]
141 else:
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000142 self.scriptsfoldermodtime = fss.GetDates()[1]
143 prefs.scriptsfolder = self.scriptsfolder.data
144 self._scripts = {}
145 self.scriptsmenu = None
146 self.makescriptsmenu()
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000147 self.makehelpmenu()
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000148
149 def quitevent(self, theAppleEvent, theReply):
Jack Jansen5a6fdcd2001-08-25 12:15:04 +0000150 from Carbon import AE
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000151 AE.AEInteractWithUser(50000000)
152 self._quit()
153
154 def suspendresume(self, onoff):
155 if onoff:
156 fss, fss_changed = self.scriptsfolder.Resolve()
157 modtime = fss.GetDates()[1]
158 if self.scriptsfoldermodtime <> modtime or fss_changed:
159 self.scriptsfoldermodtime = modtime
160 W.SetCursor('watch')
161 self.makescriptsmenu()
162
163 def ignoreevent(self, theAppleEvent, theReply):
164 pass
165
166 def opendocsevent(self, theAppleEvent, theReply):
167 W.SetCursor('watch')
168 import aetools
169 parameters, args = aetools.unpackevent(theAppleEvent)
170 docs = parameters['----']
171 if type(docs) <> type([]):
172 docs = [docs]
173 for doc in docs:
174 fss, a = doc.Resolve()
175 path = fss.as_pathname()
176 self.opendoc(path)
177
178 def opendoc(self, path):
179 fcreator, ftype = macfs.FSSpec(path).GetCreatorType()
180 if ftype == 'TEXT':
181 self.openscript(path)
Jack Jansene0ba0872002-03-29 21:23:47 +0000182 elif ftype == '\0\0\0\0' and path[-3:] == '.py':
183 self.openscript(path)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000184 else:
Just van Rossumdc3c6172001-06-19 21:37:33 +0000185 W.Message("Can't open file of type '%s'." % ftype)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000186
187 def getabouttext(self):
Jack Jansen815d2bf2002-01-21 23:00:52 +0000188 return "About Python IDE"+ELIPSES
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000189
190 def do_about(self, id, item, window, event):
191 Splash.about()
192
193 def do_setscriptsfolder(self, *args):
Jack Jansenfd0b00e2003-01-26 22:15:48 +0000194 fss = EasyDialogs.AskFolder(message="Select Scripts Folder",
195 wanted=macfs.FSSpec)
196 if fss:
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000197 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):
Jack Jansenfd0b00e2003-01-26 22:15:48 +0000210 filename = EasyDialogs.AskFileForOpen(typeList=("TEXT",))
211 if filename:
212 self.openscript(filename)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000213
Jack Jansen611b9f62003-02-05 15:41:09 +0000214 def domenu_openbyname(self, *args):
215 # Open a file by name. If the clipboard contains a filename
216 # use that as the default.
217 from Carbon import Scrap
218 try:
219 sc = Scrap.GetCurrentScrap()
220 dft = sc.GetScrapFlavorData("TEXT")
221 except Scrap.Error:
222 dft = ""
223 else:
224 if not os.path.exists(dft):
225 dft = ""
226 filename = EasyDialogs.AskString("Open File Named:", default=dft, ok="Open")
227 if filename:
228 self.openscript(filename)
229
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000230 def domenu_new(self, *args):
231 W.SetCursor('watch')
232 import PyEdit
233 return PyEdit.Editor()
234
235 def makescriptsmenu(self):
236 W.SetCursor('watch')
237 if self._scripts:
238 for id, item in self._scripts.keys():
239 if self.menubar.menus.has_key(id):
240 m = self.menubar.menus[id]
241 m.delete()
242 self._scripts = {}
243 if self.scriptsmenu:
244 if hasattr(self.scriptsmenu, 'id') and self.menubar.menus.has_key(self.scriptsmenu.id):
245 self.scriptsmenu.delete()
246 self.scriptsmenu = FrameWork.Menu(self.menubar, "Scripts")
247 #FrameWork.MenuItem(self.scriptsmenu, "New script", None, self.domenu_new)
248 #self.scriptsmenu.addseparator()
249 fss, fss_changed = self.scriptsfolder.Resolve()
250 self.scriptswalk(fss.as_pathname(), self.scriptsmenu)
251
252 def makeopenwindowsmenu(self):
253 for i in range(len(self.openwindowsmenu.items)):
254 self.openwindowsmenu.menu.DeleteMenuItem(1)
255 self.openwindowsmenu.items = []
256 windows = []
257 self._openwindows = {}
258 for window in self._windows.keys():
259 title = window.GetWTitle()
260 if not title:
261 title = "<no title>"
Jack Jansen34d11f02000-03-07 23:40:13 +0000262 windows.append((title, window))
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000263 windows.sort()
264 for title, window in windows:
265 if title == "Python Interactive": # ugly but useful hack by Joe Strout
266 shortcut = '0'
267 else:
268 shortcut = None
269 item = FrameWork.MenuItem(self.openwindowsmenu, title, shortcut, callback = self.domenu_openwindows)
270 self._openwindows[item.item] = window
271 self._openwindowscheckmark = 0
272 self.checkopenwindowsmenu()
273
274 def domenu_openwindows(self, id, item, window, event):
275 w = self._openwindows[item]
276 w.ShowWindow()
277 w.SelectWindow()
278
279 def domenu_quit(self):
280 self._quit()
281
282 def domenu_save(self, *args):
283 print "Save"
284
285 def _quit(self):
286 import PyConsole, PyEdit
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000287 for window in self._windows.values():
Just van Rossumd58c7461999-06-22 18:37:35 +0000288 try:
289 rv = window.close() # ignore any errors while quitting
290 except:
291 rv = 0 # (otherwise, we can get stuck!)
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000292 if rv and rv > 0:
293 return
Just van Rossum62a103b2002-11-22 12:48:47 +0000294 try:
295 PyConsole.console.writeprefs()
296 PyConsole.output.writeprefs()
297 PyEdit.searchengine.writeprefs()
298 except:
299 # Write to __stderr__ so the msg end up in Console.app and has
300 # at least _some_ chance of getting read...
301 # But: this is a workaround for way more serious problems with
302 # the Python 2.2 Jaguar addon.
303 sys.__stderr__.write("*** PythonIDE: Can't write preferences ***\n")
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000304 self.quitting = 1
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000305
306 def makehelpmenu(self):
307 docs = self.installdocumentation()
308 self.helpmenu = m = self.gethelpmenu()
309 docitem = FrameWork.MenuItem(m, "Python Documentation", None, self.domenu_localdocs)
310 docitem.enable(docs)
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000311 finditem = FrameWork.MenuItem(m, "Lookup in Python Documentation", None, 'lookuppython')
312 finditem.enable(docs)
Jack Jansen7f677f42002-09-06 23:03:32 +0000313 if runningOnOSX():
314 FrameWork.Separator(m)
315 doc2item = FrameWork.MenuItem(m, "Apple Developer Documentation", None, self.domenu_appledocs)
316 find2item = FrameWork.MenuItem(m, "Lookup in Carbon Documentation", None, 'lookupcarbon')
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000317 FrameWork.Separator(m)
318 webitem = FrameWork.MenuItem(m, "Python Documentation on the Web", None, self.domenu_webdocs)
319 web2item = FrameWork.MenuItem(m, "Python on the Web", None, self.domenu_webpython)
320 web3item = FrameWork.MenuItem(m, "MacPython on the Web", None, self.domenu_webmacpython)
321
322 def domenu_localdocs(self, *args):
323 from Carbon import AH
Jack Jansenec694c32002-09-11 22:05:59 +0000324 AH.AHGotoPage("Python Help", None, None)
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000325
326 def domenu_appledocs(self, *args):
327 from Carbon import AH, AppleHelp
328 try:
329 AH.AHGotoMainTOC(AppleHelp.kAHTOCTypeDeveloper)
330 except AH.Error, arg:
331 if arg[0] == -50:
332 W.Message("Developer documentation not installed")
333 else:
334 W.Message("AppleHelp Error: %s" % `arg`)
335
336 def domenu_lookuppython(self, *args):
337 from Carbon import AH
338 searchstring = self._getsearchstring()
339 if not searchstring:
340 return
341 try:
342 AH.AHSearch("Python Help", searchstring)
343 except AH.Error, arg:
344 W.Message("AppleHelp Error: %s" % `arg`)
345
346 def domenu_lookupcarbon(self, *args):
347 from Carbon import AH
348 searchstring = self._getsearchstring()
349 if not searchstring:
350 return
351 try:
352 AH.AHSearch("Carbon", searchstring)
353 except AH.Error, arg:
354 W.Message("AppleHelp Error: %s" % `arg`)
355
356 def _getsearchstring(self):
357 import PyEdit
358 editor = PyEdit.findeditor(None, fromtop=1)
359 if editor:
360 text = editor.getselectedtext()
361 if text:
362 return text
363 # This is a cop-out. We should have disabled the menus
364 # if there is no selection, but the can_ methods only seem
365 # to work for Windows. Or not for the Help menu, maybe?
Jack Jansenb2d2bc92002-08-31 01:25:17 +0000366 text = EasyDialogs.AskString("Search documentation for", ok="Search")
367 return text
368
369 def domenu_webdocs(self, *args):
370 import webbrowser
371 major, minor, micro, state, nano = sys.version_info
372 if state in ('alpha', 'beta'):
373 docversion = 'dev/doc/devel'
374 elif micro == 0:
375 docversion = 'doc/%d.%d' % (major, minor)
376 else:
377 docversion = 'doc/%d.%d.%d' % (major, minor, micro)
378 webbrowser.open("http://www.python.org/%s" % docversion)
379
380 def domenu_webpython(self, *args):
381 import webbrowser
382 webbrowser.open("http://www.python.org/")
383
384 def domenu_webmacpython(self, *args):
385 import webbrowser
386 webbrowser.open("http://www.cwi.nl/~jack/macpython.html")
387
388 def installdocumentation(self):
389 # This is rather much of a hack. Someone has to tell the Help Viewer
390 # about the Python documentation, so why not us. The documentation
391 # is located in the framework, but there's a symlink in Python.app.
392 # And as AHRegisterHelpBook wants a bundle (with the right bits in
393 # the plist file) we refer it to Python.app
394 python_app = os.path.join(sys.prefix, 'Resources/Python.app')
395 doc_source = os.path.join(python_app, 'Contents/Resources/English.lproj/Documentation')
396 if not os.path.isdir(doc_source):
397 return 0
398 try:
399 from Carbon import AH
400 AH.AHRegisterHelpBook(python_app)
401 except (ImportError, MacOS.Error), arg:
402 W.Message("Cannot register Python documentation: %s" % `arg`)
403 return 0
404 return 1
405
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000406
407PythonIDE()
408