Python Tkinter-CLose窗口事件的协议处理程序

时间:2018-10-23 06:58:17

标签: python python-3.x tkinter window protocols

如果用户单击窗口右上角的X按钮,如何在顶级窗口中覆盖退出功能?我试过了这段代码,没什么大不了的。

def chat():
global root, private_chat, chat_list_window, chat_list, ip, port, address,username
global private_chat_window, private_chat_messages, private_chat_label, private_chat_entry
address = (ip,int(port))


invited_client = chat_list[len(chat_list) -1].get(chat_list[len(chat_list) -1].curselection())

private_chat_window.append(tk.Toplevel(root))
length = len(private_chat_window)
private_chat_window[length-1].title("Private Chat with '" +  invited_client[2] + "'" )


private_chat_messages.append(Listbox(private_chat_window[length -1], width=80, height=30))
private_chat_messages[length-1].grid(row=1,column=1, columnspan=3,rowspan=1)
private_chat_messages[length-1].insert(END, "Inviting client '" + invited_client[2] + "'for a private chat...")


private_chat_label.append(ttk.Label(private_chat_window[length -1], text='Message Entry Box : '))
private_chat_label[length-1].grid(row=2,column=1)


private_chat_entry.append(ttk.Entry(private_chat_window[length -1] , width=60))
private_chat_entry[length-1].grid(row=2, column=2, columnspan = 2)    


daemon = Pyro4.Daemon()       
uri_str = daemon.register(Messaging())      

private_chat_window[length-1].bind('<KeyPress>', lambda event: private_chat_onkeypress(event,private_chat_messages[length -1], private_chat_entry[length -1], uri_str = uri_str) )
chat_list_window[len(chat_list_window)-1].destroy()

s.sendto(str.encode(":::PrivateChatInvitation:::" + '<' + username + '><' +  str(uri_str) + '>' + json.dumps(invited_client)) , address)


Thread(target = daemon.requestLoop).start()
Thread(target=check_successful_connection(uri_str, private_chat_messages[length -1])).start()

#EXIT
private_chat_window[length -1].protocol("WM_DELETE_WINDOW", on_closing(private_chat_window[length -1]))
private_chat_window[length -1].mainloop()


def on_closing(window_instance):

    if messagebox.askokcancel("Quit", "Do you want to quit?"):
         window_instance.destroy()

协议处理程序在顶层窗口上不起作用,但是当附加到根主窗口时,它可以正常工作。下面是代码:

def render_gui():     全球根     全局prev_key     全局列表框,label1,entry1,用户名     prev_key = 0

root = tk.Tk()
root.title("Welcome Client '" + username + "'" )


listbox = Listbox(root, width=80, height=30)      
listbox.grid(row=1,column=1, columnspan=5,rowspan=3)


vsb = tk.Scrollbar(root, orient="vertical", command=listbox.yview)
hsb = tk.Scrollbar(root, orient="horizontal", command=listbox.xview)
vsb.grid(row=1, column=4,rowspan = 3, sticky='ns')
hsb.grid(row=3, column=0,rowspan = 1, columnspan=5, sticky = 'wes')


#Menubar

menubar = Menu(root)
filemenu = Menu(menubar,tearoff=0)
filemenu.add_command(label="Private Chat",command=private_chat_threader) #Hmppp
menubar.add_cascade(label="Options", menu= filemenu)
root.config(menu=menubar)

listbox.configure(yscrollcommand=vsb.set,xscrollcommand=hsb.set)

label1 = ttk.Label(root, text='Message Entry Box : ')
label1.grid(row=4,column=1)

entry1 = ttk.Entry(root, width=60)
entry1.grid(row=4, column=2)    

root.geometry('490x520')
root.bind('<KeyPress>', onKeyPress)

root.protocol("WM_DELETE_WINDOW", on_closing(root))
root.mainloop()

请帮助我。谢谢。

1 个答案:

答案 0 :(得分:1)

我看不到您的代码如何在根窗口或Toplevel上工作。 protocol方法要求您传入 callable (即:对函数的引用)。相反,您要立即调用on_closing(root),然后将结果传递给协议处理程序。

您需要像这样呼叫protocol

private_chat_window[length -1].protocol("WM_DELETE_WINDOW", on_closing)

如果需要传递参数,则可以像使用任何其他回调一样使用lambdafunctools.partial。例如:

private_chat_window[length-1].protocol(
    "WM_DELETE_WINDOW", 
    lambda: on_closing(private_chat_window[length -1])
)
相关问题