""" A number of functions that enhance IDLE on macOS. """ from os.path import expanduser import plistlib from sys import platform # Used in _init_tk_type, changed by test. import tkinter ## Define functions that query the Mac graphics type. ## _tk_type and its initializer are private to this section. _tk_type = None def _init_tk_type(): """ Initializes OS X Tk variant values for isAquaTk(), isCarbonTk(), isCocoaTk(), and isXQuartz(). """ global _tk_type if platform == 'darwin': root = tkinter.Tk() ws = root.tk.call('tk', 'windowingsystem') if 'x11' in ws: _tk_type = "xquartz" elif 'aqua' not in ws: _tk_type = "other" elif 'AppKit' in root.tk.call('winfo', 'server', '.'): _tk_type = "cocoa" else: _tk_type = "carbon" root.destroy() else: _tk_type = "other" def isAquaTk(): """ Returns True if IDLE is using a native OS X Tk (Cocoa or Carbon). """ if not _tk_type: _init_tk_type() return _tk_type == "cocoa" or _tk_type == "carbon" def isCarbonTk(): """ Returns True if IDLE is using a Carbon Aqua Tk (instead of the newer Cocoa Aqua Tk). """ if not _tk_type: _init_tk_type() return _tk_type == "carbon" def isCocoaTk(): """ Returns True if IDLE is using a Cocoa Aqua Tk. """ if not _tk_type: _init_tk_type() return _tk_type == "cocoa" def isXQuartz(): """ Returns True if IDLE is using an OS X X11 Tk. """ if not _tk_type: _init_tk_type() return _tk_type == "xquartz" def tkVersionWarning(root): """ Returns a string warning message if the Tk version in use appears to be one known to cause problems with IDLE. 1. Apple Cocoa-based Tk 8.5.7 shipped with Mac OS X 10.6 is unusable. 2. Apple Cocoa-based Tk 8.5.9 in OS X 10.7 and 10.8 is better but can still crash unexpectedly. """ if isCocoaTk(): patchlevel = root.tk.call('info', 'patchlevel') if patchlevel not in ('8.5.7', '8.5.9'): return False return ("WARNING: The version of Tcl/Tk ({0}) in use may" " be unstable.\n" "Visit http://www.python.org/download/mac/tcltk/" " for current information.".format(patchlevel)) else: return False def readSystemPreferences(): """ Fetch the macOS system preferences. """ if platform != 'darwin': return None plist_path = expanduser('~/Library/Preferences/.GlobalPreferences.plist') try: with open(plist_path, 'rb') as plist_file: return plistlib.load(plist_file) except OSError: return None def preferTabsPreferenceWarning(): """ Warn if "Prefer tabs when opening documents" is set to "Always". """ if platform != 'darwin': return None prefs = readSystemPreferences() if prefs and prefs.get('AppleWindowTabbingMode') == 'always': return ( 'WARNING: The system preference "Prefer tabs when opening' ' documents" is set to "Always". This will cause various problems' ' with IDLE. For the best experience, change this setting when' ' running IDLE (via System Preferences -> Dock).' ) return None ## Fix the menu and related functions. def addOpenEventSupport(root, flist): """ This ensures that the application will respond to open AppleEvents, which makes is feasible to use IDLE as the default application for python files. """ def doOpenFile(*args): for fn in args: flist.open(fn) # The command below is a hook in aquatk that is called whenever the app # receives a file open event. The callback can have multiple arguments, # one for every file that should be opened. root.createcommand("::tk::mac::OpenDocument", doOpenFile) def hideTkConsole(root): try: root.tk.call('console', 'hide') except tkinter.TclError: # Some versions of the Tk framework don't have a console object pass def overrideRootMenu(root, flist): """ Replace the Tk root menu by something that is more appropriate for IDLE with an Aqua Tk. """ # The menu that is attached to the Tk root (".") is also used by AquaTk for # all windows that don't specify a menu of their own. The default menubar # contains a number of menus, none of which are appropriate for IDLE. The # Most annoying of those is an 'About Tck/Tk...' menu in the application # menu. # # This function replaces the default menubar by a mostly empty one, it # should only contain the correct application menu and the window menu. # # Due to a (mis-)feature of TkAqua the user will also see an empty Help # menu. from tkinter import Menu from idlelib import mainmenu from idlelib import window closeItem = mainmenu.menudefs[0][1][-2] # Remove the last 3 items of the file menu: a separator, close window and # quit. Close window will be reinserted just above the save item, where # it should be according to the HIG. Quit is in the application menu. del mainmenu.menudefs[0][1][-3:] mainmenu.menudefs[0][1].insert(6, closeItem) # Remove the 'About' entry from the help menu, it is in the application # menu del mainmenu.menudefs[-1][1][0:2] # Remove the 'Configure Idle' entry from the options menu, it is in the # application menu as 'Preferences' del mainmenu.menudefs[-2][1][0] menubar = Menu(root) root.configure(menu=menubar) menudict = {} menudict['window'] = menu = Menu(menubar, name='window', tearoff=0) menubar.add_cascade(label='Window', menu=menu, underline=0) def postwindowsmenu(menu=menu): end = menu.index('end') if end is None: end = -1 if end > 0: menu.delete(0, end) window.add_windows_to_menu(menu) window.register_callback(postwindowsmenu) def about_dialog(event=None): "Handle Help 'About IDLE' event." # Synchronize with editor.EditorWindow.about_dialog. from idlelib import help_about help_about.AboutDialog(root) def config_dialog(event=None): "Handle Options 'Configure IDLE' event." # Synchronize with editor.EditorWindow.config_dialog. from idlelib import configdialog # Ensure that the root object has an instance_dict attribute, # mirrors code in EditorWindow (although that sets the attribute # on an EditorWindow instance that is then passed as the first # argument to ConfigDialog) root.instance_dict = flist.inversedict configdialog.ConfigDialog(root, 'Settings') def help_dialog(event=None): "Handle Help 'IDLE Help' event." # Synchronize with editor.EditorWindow.help_dialog. from idlelib import help help.show_idlehelp(root) root.bind('<<about-idle>>', about_dialog) root.bind('<<open-config-dialog>>', config_dialog) root.createcommand('::tk::mac::ShowPreferences', config_dialog) if flist: root.bind('<<close-all-windows>>', flist.close_all_callback) # The binding above doesn't reliably work on all versions of Tk # on macOS. Adding command definition below does seem to do the # right thing for now. root.createcommand('exit', flist.close_all_callback) if isCarbonTk(): # for Carbon AquaTk, replace the default Tk apple menu menudict['application'] = menu = Menu(menubar, name='apple', tearoff=0) menubar.add_cascade(label='IDLE', menu=menu) mainmenu.menudefs.insert(0, ('application', [ ('About IDLE', '<<about-idle>>'), None, ])) if isCocoaTk(): # replace default About dialog with About IDLE one root.createcommand('tkAboutDialog', about_dialog) # replace default "Help" item in Help menu root.createcommand('::tk::mac::ShowHelp', help_dialog) # remove redundant "IDLE Help" from menu del mainmenu.menudefs[-1][1][0] def fixb2context(root): '''Removed bad AquaTk Button-2 (right) and Paste bindings. They prevent context menu access and seem to be gone in AquaTk8.6. See issue #24801. ''' root.unbind_class('Text', '<B2>') root.unbind_class('Text', '<B2-Motion>') root.unbind_class('Text', '<<PasteSelection>>') def setupApp(root, flist): """ Perform initial OS X customizations if needed. Called from pyshell.main() after initial calls to Tk() There are currently three major versions of Tk in use on OS X: 1. Aqua Cocoa Tk (native default since OS X 10.6) 2. Aqua Carbon Tk (original native, 32-bit only, deprecated) 3. X11 (supported by some third-party distributors, deprecated) There are various differences among the three that affect IDLE behavior, primarily with menus, mouse key events, and accelerators. Some one-time customizations are performed here. Others are dynamically tested throughout idlelib by calls to the isAquaTk(), isCarbonTk(), isCocoaTk(), isXQuartz() functions which are initialized here as well. """ if isAquaTk(): hideTkConsole(root) overrideRootMenu(root, flist) addOpenEventSupport(root, flist) fixb2context(root) if __name__ == '__main__': from unittest import main main('idlelib.idle_test.test_macosx', verbosity=2)
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
Icons | Folder | 0755 |
|
|
__pycache__ | Folder | 0755 |
|
|
idle_test | Folder | 0755 |
|
|
CREDITS.txt | File | 1.82 KB | 0644 |
|
ChangeLog | File | 55.04 KB | 0644 |
|
HISTORY.txt | File | 10.07 KB | 0644 |
|
NEWS.txt | File | 38.91 KB | 0644 |
|
NEWS2x.txt | File | 26.54 KB | 0644 |
|
README.txt | File | 9.37 KB | 0644 |
|
TODO.txt | File | 8.28 KB | 0644 |
|
__init__.py | File | 396 B | 0644 |
|
__main__.py | File | 159 B | 0644 |
|
_pyclbr.py | File | 14.84 KB | 0644 |
|
autocomplete.py | File | 9.11 KB | 0644 |
|
autocomplete_w.py | File | 19.36 KB | 0644 |
|
autoexpand.py | File | 3.14 KB | 0644 |
|
browser.py | File | 8.09 KB | 0644 |
|
calltip.py | File | 5.92 KB | 0644 |
|
calltip_w.py | File | 6.94 KB | 0644 |
|
codecontext.py | File | 10.24 KB | 0644 |
|
colorizer.py | File | 11.01 KB | 0644 |
|
config-extensions.def | File | 2.21 KB | 0644 |
|
config-highlight.def | File | 2.62 KB | 0644 |
|
config-keys.def | File | 10.52 KB | 0644 |
|
config-main.def | File | 3.05 KB | 0644 |
|
config.py | File | 37.97 KB | 0644 |
|
config_key.py | File | 13.09 KB | 0644 |
|
configdialog.py | File | 98.69 KB | 0644 |
|
debugger.py | File | 18.65 KB | 0644 |
|
debugger_r.py | File | 11.86 KB | 0644 |
|
debugobj.py | File | 3.96 KB | 0644 |
|
debugobj_r.py | File | 1.06 KB | 0644 |
|
delegator.py | File | 1.02 KB | 0644 |
|
dynoption.py | File | 1.97 KB | 0644 |
|
editor.py | File | 65.7 KB | 0644 |
|
extend.txt | File | 3.56 KB | 0644 |
|
filelist.py | File | 3.8 KB | 0644 |
|
grep.py | File | 6.58 KB | 0644 |
|
help.html | File | 53.82 KB | 0644 |
|
help.py | File | 11.06 KB | 0644 |
|
help_about.py | File | 8.77 KB | 0644 |
|
history.py | File | 3.95 KB | 0644 |
|
hyperparser.py | File | 12.58 KB | 0644 |
|
idle.py | File | 454 B | 0644 |
|
idle.pyw | File | 570 B | 0644 |
|
iomenu.py | File | 20.25 KB | 0644 |
|
macosx.py | File | 9.43 KB | 0644 |
|
mainmenu.py | File | 3.62 KB | 0644 |
|
multicall.py | File | 18.21 KB | 0644 |
|
outwin.py | File | 5.67 KB | 0644 |
|
paragraph.py | File | 7 KB | 0644 |
|
parenmatch.py | File | 7.04 KB | 0644 |
|
pathbrowser.py | File | 3.12 KB | 0644 |
|
percolator.py | File | 3.06 KB | 0644 |
|
pyparse.py | File | 19.65 KB | 0644 |
|
pyshell.py | File | 56.38 KB | 0755 |
|
query.py | File | 12.14 KB | 0644 |
|
redirector.py | File | 6.71 KB | 0644 |
|
replace.py | File | 7.33 KB | 0644 |
|
rpc.py | File | 20.64 KB | 0644 |
|
rstrip.py | File | 868 B | 0644 |
|
run.py | File | 16.87 KB | 0644 |
|
runscript.py | File | 7.66 KB | 0644 |
|
scrolledlist.py | File | 4.35 KB | 0644 |
|
search.py | File | 3.09 KB | 0644 |
|
searchbase.py | File | 7.28 KB | 0644 |
|
searchengine.py | File | 7.3 KB | 0644 |
|
squeezer.py | File | 13 KB | 0644 |
|
stackviewer.py | File | 4.35 KB | 0644 |
|
statusbar.py | File | 1.41 KB | 0644 |
|
textview.py | File | 5.98 KB | 0644 |
|
tooltip.py | File | 6.33 KB | 0644 |
|
tree.py | File | 14.74 KB | 0644 |
|
undo.py | File | 10.79 KB | 0644 |
|
window.py | File | 2.53 KB | 0644 |
|
zoomheight.py | File | 1.31 KB | 0644 |
|
zzdummy.py | File | 961 B | 0644 |
|