""" Dialog for building Tkinter accelerator key bindings """ from Tkinter import * import tkMessageBox import string import sys class GetKeysDialog(Toplevel): def __init__(self,parent,title,action,currentKeySequences,_htest=False): """ action - string, the name of the virtual event these keys will be mapped to currentKeys - list, a list of all key sequence lists currently mapped to virtual events, for overlap checking _htest - bool, change box location when running htest """ Toplevel.__init__(self, parent) self.configure(borderwidth=5) self.resizable(height=FALSE,width=FALSE) self.title(title) self.transient(parent) self.grab_set() self.protocol("WM_DELETE_WINDOW", self.Cancel) self.parent = parent self.action=action self.currentKeySequences=currentKeySequences self.result='' self.keyString=StringVar(self) self.keyString.set('') self.SetModifiersForPlatform() # set self.modifiers, self.modifier_label self.modifier_vars = [] for modifier in self.modifiers: variable = StringVar(self) variable.set('') self.modifier_vars.append(variable) self.advanced = False self.CreateWidgets() self.LoadFinalKeyList() self.withdraw() #hide while setting geometry self.update_idletasks() self.geometry( "+%d+%d" % ( parent.winfo_rootx() + (parent.winfo_width()/2 - self.winfo_reqwidth()/2), parent.winfo_rooty() + ((parent.winfo_height()/2 - self.winfo_reqheight()/2) if not _htest else 150) ) ) #centre dialog over parent (or below htest box) self.deiconify() #geometry set, unhide self.wait_window() def CreateWidgets(self): frameMain = Frame(self,borderwidth=2,relief=SUNKEN) frameMain.pack(side=TOP,expand=TRUE,fill=BOTH) frameButtons=Frame(self) frameButtons.pack(side=BOTTOM,fill=X) self.buttonOK = Button(frameButtons,text='OK', width=8,command=self.OK) self.buttonOK.grid(row=0,column=0,padx=5,pady=5) self.buttonCancel = Button(frameButtons,text='Cancel', width=8,command=self.Cancel) self.buttonCancel.grid(row=0,column=1,padx=5,pady=5) self.frameKeySeqBasic = Frame(frameMain) self.frameKeySeqAdvanced = Frame(frameMain) self.frameControlsBasic = Frame(frameMain) self.frameHelpAdvanced = Frame(frameMain) self.frameKeySeqAdvanced.grid(row=0,column=0,sticky=NSEW,padx=5,pady=5) self.frameKeySeqBasic.grid(row=0,column=0,sticky=NSEW,padx=5,pady=5) self.frameKeySeqBasic.lift() self.frameHelpAdvanced.grid(row=1,column=0,sticky=NSEW,padx=5) self.frameControlsBasic.grid(row=1,column=0,sticky=NSEW,padx=5) self.frameControlsBasic.lift() self.buttonLevel = Button(frameMain,command=self.ToggleLevel, text='Advanced Key Binding Entry >>') self.buttonLevel.grid(row=2,column=0,stick=EW,padx=5,pady=5) labelTitleBasic = Label(self.frameKeySeqBasic, text="New keys for '"+self.action+"' :") labelTitleBasic.pack(anchor=W) labelKeysBasic = Label(self.frameKeySeqBasic,justify=LEFT, textvariable=self.keyString,relief=GROOVE,borderwidth=2) labelKeysBasic.pack(ipadx=5,ipady=5,fill=X) self.modifier_checkbuttons = {} column = 0 for modifier, variable in zip(self.modifiers, self.modifier_vars): label = self.modifier_label.get(modifier, modifier) check=Checkbutton(self.frameControlsBasic, command=self.BuildKeyString, text=label,variable=variable,onvalue=modifier,offvalue='') check.grid(row=0,column=column,padx=2,sticky=W) self.modifier_checkbuttons[modifier] = check column += 1 labelFnAdvice=Label(self.frameControlsBasic,justify=LEFT, text=\ "Select the desired modifier keys\n"+ "above, and the final key from the\n"+ "list on the right.\n\n" + "Use upper case Symbols when using\n" + "the Shift modifier. (Letters will be\n" + "converted automatically.)") labelFnAdvice.grid(row=1,column=0,columnspan=4,padx=2,sticky=W) self.listKeysFinal=Listbox(self.frameControlsBasic,width=15,height=10, selectmode=SINGLE) self.listKeysFinal.bind('<ButtonRelease-1>',self.FinalKeySelected) self.listKeysFinal.grid(row=0,column=4,rowspan=4,sticky=NS) scrollKeysFinal=Scrollbar(self.frameControlsBasic,orient=VERTICAL, command=self.listKeysFinal.yview) self.listKeysFinal.config(yscrollcommand=scrollKeysFinal.set) scrollKeysFinal.grid(row=0,column=5,rowspan=4,sticky=NS) self.buttonClear=Button(self.frameControlsBasic, text='Clear Keys',command=self.ClearKeySeq) self.buttonClear.grid(row=2,column=0,columnspan=4) labelTitleAdvanced = Label(self.frameKeySeqAdvanced,justify=LEFT, text="Enter new binding(s) for '"+self.action+"' :\n"+ "(These bindings will not be checked for validity!)") labelTitleAdvanced.pack(anchor=W) self.entryKeysAdvanced=Entry(self.frameKeySeqAdvanced, textvariable=self.keyString) self.entryKeysAdvanced.pack(fill=X) labelHelpAdvanced=Label(self.frameHelpAdvanced,justify=LEFT, text="Key bindings are specified using Tkinter keysyms as\n"+ "in these samples: <Control-f>, <Shift-F2>, <F12>,\n" "<Control-space>, <Meta-less>, <Control-Alt-Shift-X>.\n" "Upper case is used when the Shift modifier is present!\n\n" + "'Emacs style' multi-keystroke bindings are specified as\n" + "follows: <Control-x><Control-y>, where the first key\n" + "is the 'do-nothing' keybinding.\n\n" + "Multiple separate bindings for one action should be\n"+ "separated by a space, eg., <Alt-v> <Meta-v>." ) labelHelpAdvanced.grid(row=0,column=0,sticky=NSEW) def SetModifiersForPlatform(self): """Determine list of names of key modifiers for this platform. The names are used to build Tk bindings -- it doesn't matter if the keyboard has these keys, it matters if Tk understands them. The order is also important: key binding equality depends on it, so config-keys.def must use the same ordering. """ if sys.platform == "darwin": self.modifiers = ['Shift', 'Control', 'Option', 'Command'] else: self.modifiers = ['Control', 'Alt', 'Shift'] self.modifier_label = {'Control': 'Ctrl'} # short name def ToggleLevel(self): if self.buttonLevel.cget('text')[:8]=='Advanced': self.ClearKeySeq() self.buttonLevel.config(text='<< Basic Key Binding Entry') self.frameKeySeqAdvanced.lift() self.frameHelpAdvanced.lift() self.entryKeysAdvanced.focus_set() self.advanced = True else: self.ClearKeySeq() self.buttonLevel.config(text='Advanced Key Binding Entry >>') self.frameKeySeqBasic.lift() self.frameControlsBasic.lift() self.advanced = False def FinalKeySelected(self,event): self.BuildKeyString() def BuildKeyString(self): keyList = modifiers = self.GetModifiers() finalKey = self.listKeysFinal.get(ANCHOR) if finalKey: finalKey = self.TranslateKey(finalKey, modifiers) keyList.append(finalKey) self.keyString.set('<' + string.join(keyList,'-') + '>') def GetModifiers(self): modList = [variable.get() for variable in self.modifier_vars] return [mod for mod in modList if mod] def ClearKeySeq(self): self.listKeysFinal.select_clear(0,END) self.listKeysFinal.yview(MOVETO, '0.0') for variable in self.modifier_vars: variable.set('') self.keyString.set('') def LoadFinalKeyList(self): #these tuples are also available for use in validity checks self.functionKeys=('F1','F2','F3','F4','F5','F6','F7','F8','F9', 'F10','F11','F12') self.alphanumKeys=tuple(string.ascii_lowercase+string.digits) self.punctuationKeys=tuple('~!@#%^&*()_-+={}[]|;:,.<>/?') self.whitespaceKeys=('Tab','Space','Return') self.editKeys=('BackSpace','Delete','Insert') self.moveKeys=('Home','End','Page Up','Page Down','Left Arrow', 'Right Arrow','Up Arrow','Down Arrow') #make a tuple of most of the useful common 'final' keys keys=(self.alphanumKeys+self.punctuationKeys+self.functionKeys+ self.whitespaceKeys+self.editKeys+self.moveKeys) self.listKeysFinal.insert(END, *keys) def TranslateKey(self, key, modifiers): "Translate from keycap symbol to the Tkinter keysym" translateDict = {'Space':'space', '~':'asciitilde','!':'exclam','@':'at','#':'numbersign', '%':'percent','^':'asciicircum','&':'ampersand','*':'asterisk', '(':'parenleft',')':'parenright','_':'underscore','-':'minus', '+':'plus','=':'equal','{':'braceleft','}':'braceright', '[':'bracketleft',']':'bracketright','|':'bar',';':'semicolon', ':':'colon',',':'comma','.':'period','<':'less','>':'greater', '/':'slash','?':'question','Page Up':'Prior','Page Down':'Next', 'Left Arrow':'Left','Right Arrow':'Right','Up Arrow':'Up', 'Down Arrow': 'Down', 'Tab':'Tab'} if key in translateDict.keys(): key = translateDict[key] if 'Shift' in modifiers and key in string.ascii_lowercase: key = key.upper() key = 'Key-' + key return key def OK(self, event=None): if self.advanced or self.KeysOK(): # doesn't check advanced string yet self.result=self.keyString.get() self.grab_release() self.destroy() def Cancel(self, event=None): self.result='' self.grab_release() self.destroy() def KeysOK(self): '''Validity check on user's 'basic' keybinding selection. Doesn't check the string produced by the advanced dialog because 'modifiers' isn't set. ''' keys = self.keyString.get() keys.strip() finalKey = self.listKeysFinal.get(ANCHOR) modifiers = self.GetModifiers() # create a key sequence list for overlap check: keySequence = keys.split() keysOK = False title = 'Key Sequence Error' if not keys: tkMessageBox.showerror(title=title, parent=self, message='No keys specified.') elif not keys.endswith('>'): tkMessageBox.showerror(title=title, parent=self, message='Missing the final Key') elif (not modifiers and finalKey not in self.functionKeys + self.moveKeys): tkMessageBox.showerror(title=title, parent=self, message='No modifier key(s) specified.') elif (modifiers == ['Shift']) \ and (finalKey not in self.functionKeys + self.moveKeys + ('Tab', 'Space')): msg = 'The shift modifier by itself may not be used with'\ ' this key symbol.' tkMessageBox.showerror(title=title, parent=self, message=msg) elif keySequence in self.currentKeySequences: msg = 'This key combination is already in use.' tkMessageBox.showerror(title=title, parent=self, message=msg) else: keysOK = True return keysOK if __name__ == '__main__': from idlelib.idle_test.htest import run run(GetKeysDialog)
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
Icons | Folder | 0755 |
|
|
idle_test | Folder | 0755 |
|
|
AutoComplete.py | File | 8.75 KB | 0644 |
|
AutoComplete.pyc | File | 7.82 KB | 0644 |
|
AutoComplete.pyo | File | 7.82 KB | 0644 |
|
AutoCompleteWindow.py | File | 16.91 KB | 0644 |
|
AutoCompleteWindow.pyc | File | 12.19 KB | 0644 |
|
AutoCompleteWindow.pyo | File | 12.13 KB | 0644 |
|
AutoExpand.py | File | 3.32 KB | 0644 |
|
AutoExpand.pyc | File | 3.42 KB | 0644 |
|
AutoExpand.pyo | File | 3.42 KB | 0644 |
|
Bindings.py | File | 2.91 KB | 0644 |
|
Bindings.pyc | File | 4.58 KB | 0644 |
|
Bindings.pyo | File | 4.58 KB | 0644 |
|
CREDITS.txt | File | 1.82 KB | 0644 |
|
CallTipWindow.py | File | 5.92 KB | 0644 |
|
CallTipWindow.pyc | File | 5.99 KB | 0644 |
|
CallTipWindow.pyo | File | 5.99 KB | 0644 |
|
CallTips.py | File | 7.56 KB | 0644 |
|
CallTips.pyc | File | 7.94 KB | 0644 |
|
CallTips.pyo | File | 7.94 KB | 0644 |
|
ChangeLog | File | 55.07 KB | 0644 |
|
ClassBrowser.py | File | 6.83 KB | 0644 |
|
ClassBrowser.pyc | File | 9.28 KB | 0644 |
|
ClassBrowser.pyo | File | 9.28 KB | 0644 |
|
CodeContext.py | File | 8.15 KB | 0644 |
|
CodeContext.pyc | File | 6.5 KB | 0644 |
|
CodeContext.pyo | File | 6.46 KB | 0644 |
|
ColorDelegator.py | File | 9.53 KB | 0644 |
|
ColorDelegator.pyc | File | 8.69 KB | 0644 |
|
ColorDelegator.pyo | File | 8.69 KB | 0644 |
|
Debugger.py | File | 17.81 KB | 0644 |
|
Debugger.pyc | File | 17.13 KB | 0644 |
|
Debugger.pyo | File | 17.13 KB | 0644 |
|
Delegator.py | File | 665 B | 0644 |
|
Delegator.pyc | File | 1.24 KB | 0644 |
|
Delegator.pyo | File | 1.24 KB | 0644 |
|
EditorWindow.py | File | 63.96 KB | 0644 |
|
EditorWindow.pyc | File | 55.53 KB | 0644 |
|
EditorWindow.pyo | File | 55.43 KB | 0644 |
|
FileList.py | File | 3.63 KB | 0644 |
|
FileList.pyc | File | 3.93 KB | 0644 |
|
FileList.pyo | File | 3.9 KB | 0644 |
|
FormatParagraph.py | File | 7.12 KB | 0644 |
|
FormatParagraph.pyc | File | 6.97 KB | 0644 |
|
FormatParagraph.pyo | File | 6.97 KB | 0644 |
|
GrepDialog.py | File | 5.02 KB | 0644 |
|
GrepDialog.pyc | File | 6.27 KB | 0644 |
|
GrepDialog.pyo | File | 6.27 KB | 0644 |
|
HISTORY.txt | File | 10.08 KB | 0644 |
|
HyperParser.py | File | 10.25 KB | 0644 |
|
HyperParser.pyc | File | 6.52 KB | 0644 |
|
HyperParser.pyo | File | 6.52 KB | 0644 |
|
IOBinding.py | File | 21.4 KB | 0644 |
|
IOBinding.pyc | File | 18.1 KB | 0644 |
|
IOBinding.pyo | File | 18.1 KB | 0644 |
|
IdleHistory.py | File | 3.96 KB | 0644 |
|
IdleHistory.pyc | File | 3.96 KB | 0644 |
|
IdleHistory.pyo | File | 3.96 KB | 0644 |
|
MultiCall.py | File | 17.29 KB | 0644 |
|
MultiCall.pyc | File | 15.97 KB | 0644 |
|
MultiCall.pyo | File | 15.9 KB | 0644 |
|
MultiStatusBar.py | File | 1.32 KB | 0644 |
|
MultiStatusBar.pyc | File | 2.23 KB | 0644 |
|
MultiStatusBar.pyo | File | 2.23 KB | 0644 |
|
NEWS.txt | File | 46.14 KB | 0644 |
|
ObjectBrowser.py | File | 4.27 KB | 0644 |
|
ObjectBrowser.pyc | File | 6.9 KB | 0644 |
|
ObjectBrowser.pyo | File | 6.9 KB | 0644 |
|
OutputWindow.py | File | 4.47 KB | 0644 |
|
OutputWindow.pyc | File | 5.11 KB | 0644 |
|
OutputWindow.pyo | File | 5.11 KB | 0644 |
|
ParenMatch.py | File | 6.56 KB | 0644 |
|
ParenMatch.pyc | File | 6.96 KB | 0644 |
|
ParenMatch.pyo | File | 6.96 KB | 0644 |
|
PathBrowser.py | File | 2.94 KB | 0644 |
|
PathBrowser.pyc | File | 4.38 KB | 0644 |
|
PathBrowser.pyo | File | 4.38 KB | 0644 |
|
Percolator.py | File | 3.15 KB | 0644 |
|
Percolator.pyc | File | 4.5 KB | 0644 |
|
Percolator.pyo | File | 4.32 KB | 0644 |
|
PyParse.py | File | 19.05 KB | 0644 |
|
PyParse.pyc | File | 9.77 KB | 0644 |
|
PyParse.pyo | File | 9.34 KB | 0644 |
|
PyShell.py | File | 57.48 KB | 0755 |
|
PyShell.pyc | File | 51.59 KB | 0644 |
|
PyShell.pyo | File | 51.49 KB | 0644 |
|
README.txt | File | 7.71 KB | 0644 |
|
RemoteDebugger.py | File | 11.36 KB | 0644 |
|
RemoteDebugger.pyc | File | 15.94 KB | 0644 |
|
RemoteDebugger.pyo | File | 15.79 KB | 0644 |
|
RemoteObjectBrowser.py | File | 942 B | 0644 |
|
RemoteObjectBrowser.pyc | File | 2.1 KB | 0644 |
|
RemoteObjectBrowser.pyo | File | 2.1 KB | 0644 |
|
ReplaceDialog.py | File | 6.48 KB | 0644 |
|
ReplaceDialog.pyc | File | 7.57 KB | 0644 |
|
ReplaceDialog.pyo | File | 7.57 KB | 0644 |
|
RstripExtension.py | File | 1.03 KB | 0644 |
|
RstripExtension.pyc | File | 1.58 KB | 0644 |
|
RstripExtension.pyo | File | 1.58 KB | 0644 |
|
ScriptBinding.py | File | 8.26 KB | 0644 |
|
ScriptBinding.pyc | File | 8.01 KB | 0644 |
|
ScriptBinding.pyo | File | 8.01 KB | 0644 |
|
ScrolledList.py | File | 4.27 KB | 0644 |
|
ScrolledList.pyc | File | 6.33 KB | 0644 |
|
ScrolledList.pyo | File | 6.33 KB | 0644 |
|
SearchDialog.py | File | 2.57 KB | 0644 |
|
SearchDialog.pyc | File | 3.89 KB | 0644 |
|
SearchDialog.pyo | File | 3.89 KB | 0644 |
|
SearchDialogBase.py | File | 6.93 KB | 0644 |
|
SearchDialogBase.pyc | File | 8.26 KB | 0644 |
|
SearchDialogBase.pyo | File | 8.26 KB | 0644 |
|
SearchEngine.py | File | 7.29 KB | 0644 |
|
SearchEngine.pyc | File | 8.11 KB | 0644 |
|
SearchEngine.pyo | File | 8.11 KB | 0644 |
|
StackViewer.py | File | 4.33 KB | 0644 |
|
StackViewer.pyc | File | 6.25 KB | 0644 |
|
StackViewer.pyo | File | 6.25 KB | 0644 |
|
TODO.txt | File | 8.28 KB | 0644 |
|
ToolTip.py | File | 3.1 KB | 0644 |
|
ToolTip.pyc | File | 4.56 KB | 0644 |
|
ToolTip.pyo | File | 4.56 KB | 0644 |
|
TreeWidget.py | File | 14.68 KB | 0644 |
|
TreeWidget.pyc | File | 17.28 KB | 0644 |
|
TreeWidget.pyo | File | 17.28 KB | 0644 |
|
UndoDelegator.py | File | 10.53 KB | 0644 |
|
UndoDelegator.pyc | File | 13.24 KB | 0644 |
|
UndoDelegator.pyo | File | 13.24 KB | 0644 |
|
WidgetRedirector.py | File | 6.74 KB | 0644 |
|
WidgetRedirector.pyc | File | 7.59 KB | 0644 |
|
WidgetRedirector.pyo | File | 7.59 KB | 0644 |
|
WindowList.py | File | 2.42 KB | 0644 |
|
WindowList.pyc | File | 3.55 KB | 0644 |
|
WindowList.pyo | File | 3.55 KB | 0644 |
|
ZoomHeight.py | File | 1.27 KB | 0644 |
|
ZoomHeight.pyc | File | 1.61 KB | 0644 |
|
ZoomHeight.pyo | File | 1.61 KB | 0644 |
|
__init__.py | File | 288 B | 0644 |
|
__init__.pyc | File | 431 B | 0644 |
|
__init__.pyo | File | 431 B | 0644 |
|
aboutDialog.py | File | 6.85 KB | 0644 |
|
aboutDialog.pyc | File | 6.69 KB | 0644 |
|
aboutDialog.pyo | File | 6.69 KB | 0644 |
|
config-extensions.def | File | 2.9 KB | 0644 |
|
config-highlight.def | File | 2.46 KB | 0644 |
|
config-keys.def | File | 7.59 KB | 0644 |
|
config-main.def | File | 2.5 KB | 0644 |
|
configDialog.py | File | 64.41 KB | 0644 |
|
configDialog.pyc | File | 52.04 KB | 0644 |
|
configDialog.pyo | File | 52.04 KB | 0644 |
|
configHandler.py | File | 31.72 KB | 0644 |
|
configHandler.pyc | File | 28.67 KB | 0644 |
|
configHandler.pyo | File | 28.67 KB | 0644 |
|
configHelpSourceEdit.py | File | 6.53 KB | 0644 |
|
configHelpSourceEdit.pyc | File | 6.44 KB | 0644 |
|
configHelpSourceEdit.pyo | File | 6.44 KB | 0644 |
|
configSectionNameDialog.py | File | 3.95 KB | 0644 |
|
configSectionNameDialog.pyc | File | 4.32 KB | 0644 |
|
configSectionNameDialog.pyo | File | 4.32 KB | 0644 |
|
dynOptionMenuWidget.py | File | 1.94 KB | 0644 |
|
dynOptionMenuWidget.pyc | File | 2.72 KB | 0644 |
|
dynOptionMenuWidget.pyo | File | 2.72 KB | 0644 |
|
extend.txt | File | 3.56 KB | 0644 |
|
help.html | File | 41.42 KB | 0644 |
|
help.py | File | 10.78 KB | 0644 |
|
help.pyc | File | 11.98 KB | 0644 |
|
help.pyo | File | 11.98 KB | 0644 |
|
help.txt | File | 11.86 KB | 0644 |
|
idle.py | File | 453 B | 0644 |
|
idle.pyc | File | 410 B | 0644 |
|
idle.pyo | File | 410 B | 0644 |
|
idle.pyw | File | 563 B | 0644 |
|
idlever.py | File | 415 B | 0644 |
|
idlever.pyc | File | 578 B | 0644 |
|
idlever.pyo | File | 578 B | 0644 |
|
keybindingDialog.py | File | 12.18 KB | 0644 |
|
keybindingDialog.pyc | File | 11.89 KB | 0644 |
|
keybindingDialog.pyo | File | 11.89 KB | 0644 |
|
macosxSupport.py | File | 8.24 KB | 0644 |
|
macosxSupport.pyc | File | 8.16 KB | 0644 |
|
macosxSupport.pyo | File | 8.02 KB | 0644 |
|
rpc.py | File | 19.68 KB | 0644 |
|
rpc.pyc | File | 21.22 KB | 0644 |
|
rpc.pyo | File | 21.12 KB | 0644 |
|
run.py | File | 12.61 KB | 0644 |
|
run.pyc | File | 13.1 KB | 0644 |
|
run.pyo | File | 13.05 KB | 0644 |
|
tabbedpages.py | File | 18.01 KB | 0644 |
|
tabbedpages.pyc | File | 18.13 KB | 0644 |
|
tabbedpages.pyo | File | 18.13 KB | 0644 |
|
textView.py | File | 3.44 KB | 0644 |
|
textView.pyc | File | 3.93 KB | 0644 |
|
textView.pyo | File | 3.93 KB | 0644 |
|