|  | """ | 
|  | 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[-3][1][0:2] | 
|  | 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) |