键盘快捷键绑定,弹出上下文菜单和TypeErrors

时间:2015-03-26 20:45:00

标签: python types tkinter contextmenu keyboard-shortcuts

我总是觉得,当我在堆栈溢出上问一个问题,就像答案一样......应该是...... RTFM,即使我得到的每一个答案都是友好,耐心和接受的。经过一个下午试图找到答案后,我被困住了。我想打开一个Tkinter,文本框和绑定键(键盘快捷键)和一个菜单项,这样我就会有一个很好的弹出式菜单(也就是上下文菜单),比如Window&Note;见

Cut 
Copy 
Paste 
----- 
Select All

Ctrl + X Ctrl + C Ctrl + < kbd> V 和 Ctrl + A 作为键绑定(我还没弄清楚如何进行撤消)。问题是我无法同时获得键绑定和菜单弹出以使用相同的功能。如果我添加或删除&#34;事件&#34; select_all()定义的参数有效,但另一个没有。

from Tkinter import *

# Clears the clipboard and copies the selected text to the it
def copy():
    mainwin.clipboard_clear()
    mainwin.clipboard_append(mainwin.selection_get())

# Needed for the right click pop-up menu to work, goes with copy()
def popup(event):
    popupmenu.post(event.x_root, event.y_root)

# Selects all text, used with the "control a" keybinding
def select_all(event):
    textbox.tag_add(SEL, "1.0", END)
    textbox.mark_set(INSERT, "1.0")
    textbox.see(INSERT)
    return 'break'

# Start of the program
mainwin = Tk()

scrollbar = Scrollbar(mainwin)
scrollbar.pack(side=RIGHT, fill=Y)

textbox = Text(mainwin, yscrollcommand=scrollbar.set)
textbox.pack(side=LEFT, fill=BOTH)

scrollbar.config(command=textbox.yview)

# Key bindings for the Text widget
textbox.bind("<Control-Key-a>", select_all)

# Pop-up menu, with right click binding for the Text widget
popupmenu = Menu(mainwin, tearoff=0)
popupmenu.add_command(label="Copy", command=copy)
popupmenu.add_separator()
popupmenu.add_command(label="Select All", command=select_all)
textbox.bind("<Button-3>", popup)

mainloop()

它在键绑定上面写的方式有效,但是菜单项给了我:

TypeError: select_all() takes exactly 1 argument (0 given)

我可以写两个函数,但这似乎效率很低,并且无法解释为什么程序以这种方式运行。

1 个答案:

答案 0 :(得分:5)

这里的问题是,当您将函数绑定到用户输入事件(例如按键或鼠标单击),然后使用该事件调用它时,它会将事件发送到该函数。这非常有用,因为您可能希望将鼠标单击的位置传递给应该在画布上绘制点的函数。但是,如果单击或按键仅用作加速器,那么您将传递一个无用的事件。这本身并不是那么糟糕,但是如果你还想在没有按键或鼠标点击的情况下访问该功能 - 例如,通过菜单中的命令?这不会发送一个事件,你的功能正在期待一个。

有多种方法可以解决这个问题。

  1. 使用def select_all(event=None)而非def select_all(event)定义您的功能。这将允许您的函数期望0或1个参数,如果没有传递参数,则默认为event None。 (谢谢,@ Bryan。)

  2. 使用def select_all(*event)而非def select_all(event)定义您的功能。这将允许您的函数期望任意数量的位置参数,包括0或1.这个的通用名称是*args或“star args”。

  3. 将菜单命令与popupmenu.add_command(label="Select All", command=lambda: select_all(0))绑定。这定义了一个内联函数,它包含使用一个参数调用select_all函数。那个论点很垃圾,但你还是没用它,所以一切都很好。我认为在tkinter中你也可以使用command=select_all, 0绑定带有参数的命令,但lambda结构更受欢迎,因为它在tkinter之外很有用,实际上是方便的语言功能(例如,使用sorted(mylist, key=lambda x: x[1])按每个项目的第二个元素对可迭代项进行排序)。