如何解决tkinter 菜单,多个条目总是调用相同的回调
我正在即时配置 tkinter 菜单选项,并传入菜单项和回调函数列表。
然而,无论实际选择哪个菜单项,都会调用相同的回调。我认为这与用于构建 lambda 的 cmd_func 变量有关,随着构建菜单选项的循环迭代而改变。随着循环迭代,lambda 中的值最终会发生变化,因此所有 lambda 最终都指向 cmd_func 所采用的最后一个值?但是我看不到如何解决问题? - 我确实尝试在构建菜单时枚举 menu_cmds,并使用索引 var 直接索引到数组 menu_cmds 中以获取函数值,而不是使用中间的 cmd_func var,但这没有帮助。
此代码复制了该问题;
from tkinter import *
class a:
def __init__(self):
self.root=Tk()
self.m=[ ('name_a',self.command_a),('name_b',self.command_b) ]
self.control_frame=Frame(self.root)
self.control_frame.pack(side=TOP,fill=X)
self.control=control(self.control_frame,self.m)
self.control.pack()
def command_a(self,data):
print("command_a %s,%s" % data)
def command_b(self,data):
print("command_b %s,%s" % data)
class control(Frame):
def __init__(self,parent,menu_cmds):
Frame.__init__(self,parent)
self.menu_cmds=menu_cmds
self.canvas=Canvas(self,width=800,height=400)
self.canvas.pack(side=LEFT,anchor=W)
self.canvas.create_rectangle(100,100,300,fill="#FF0000")
self.canvas.bind("<Button-3>",self.canvas_context_popup)
self.build_canvas_context_menu()
def build_canvas_context_menu(self):
self.canvas_context_menu=Menu(self.canvas,tearoff=0)
for menu_text,cmd_func in self.menu_cmds:
print("menu %s" % menu_text)
cmd=lambda : self.canvas_context_menu_select(cmd_func)
self.canvas_context_menu.add_command(label=menu_text,command=cmd)
def canvas_context_popup(self,event):
try:
self.popup_at_x=event.x_root-self.canvas.winfo_rootx()
self.popup_at_y=event.y_root-self.canvas.winfo_rooty()
self.canvas_context_menu.tk_popup(event.x_root,event.y_root,0)
finally:
self.canvas_context_menu.grab_release()
def canvas_context_menu_select(self,cmd_func):
x=self.popup_at_x
y=self.popup_at_y
cmd_func((x,y))
GUI = a()
GUI.root.geometry('1000x600')
GUI.root.update()
GUI.root.mainloop()
非常感谢任何帮助!
解决方法
上面的 manveti 和 furas 马上就收到了,谢谢!
解决方案是将 lambda 改为
cmd=lambda x=cmd_func : self.canvas_context_menu_select(x)
为了比我能给出的更好的解释;参考上面 manveti 链接的评论和信息。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。