You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
288 lines
9.4 KiB
288 lines
9.4 KiB
3 years ago
|
"""
|
||
|
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 https://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)
|