在 Python3/tkinter 中如何在用户单击 Toplevel 窗口上的关闭按钮时进行拦截

Vic*_*gos 2 oop user-interface tkinter python-3.x

我想要一个主窗口和一个或多个Toplevel()可以按需打开的窗口。我能够创建窗户,甚至摧毁它们。

但是,我试图在主窗口中实现一个按钮,可以打开和关闭第二个窗口(第二个窗口应该始终是唯一的,即永远不会同时打开两次)。这就是我现在所拥有的,在摆弄了一下之后:

#!/usr/bin/python3

import tkinter as tk
from tkinter import ttk
import tkinter.font


class baseApp(ttk.Frame):
    def __init__(self,master,*args,**kwargs):
        super().__init__(master,*args,**kwargs)
        self.master = master

        self.mainframe = ttk.Frame(master)
        self.topframe = ttk.Frame(self.mainframe, padding="5 8 5 5")

        self.topframe.pack(side=tk.TOP, fill=tk.X)
        self.mainframe.pack(side=tk.TOP, expand=True, fill=tk.BOTH)


class App(baseApp):
    def __init__(self,master,*args,**kwargs):
        super().__init__(master,*args,**kwargs)
        self.master = master
        self.button1 = ttk.Button(self.topframe,text="One",command=self.button_one)
        self.btn_remessas = ttk.Button(self.topframe,text="Open/close Toplevel window",command=self.create_window1)
        self.button1.grid(row=0,column=0)
        self.btn_remessas.grid(row=0,column=1)
        self.topframe.pack(side=tk.TOP, fill=tk.X)
        self.mainframe.pack(side=tk.TOP, expand=True, fill=tk.BOTH)

    def create_window1(self):
        if current_state.window2_open == False:
            self.newWindow2 = tk.Toplevel(self.master)
            self.newWindow2.geometry('600x500+680+0')
            self.newWindow2.title('Second window')
            self.janela_remessas = SecondWindow(self.newWindow2)
            current_state.window2_open = True              
        else:
            self.newWindow2.destroy()
            root.update_idletasks()
            current_state.window2_open = False

    def button_one(self):
        print("button 1 pressed")


class SecondWindow:
    def __init__(self,master,*args,**kwargs):
        #super().__init__(master,*args,**kwargs)
        self.mainframe = ttk.Frame(master, padding="5 8 5 5")
        self.topframe = ttk.Frame(self.mainframe)
        self.button1 = ttk.Button(self.topframe,text="button",command=self.button_function)
        self.button1.pack()                
        self.topframe.pack(side=tk.TOP, fill=tk.X)
        self.mainframe.pack(side=tk.TOP, expand=True, fill=tk.BOTH)

    def button_function(self, *event):
        print("user just pressed button")

    def close_window(self, *event): #Please fix me!
        current_state.window2_open = False
        self.destroy()


class AppStatus:
    def __init__(self):
        self.window2_open = False  


if __name__ == "__main__": 
    root = tk.Tk() 
    app = App(root) 
    current_state = AppStatus()
    root.configure(background='grey95')
    root.title('Application window')
    root.geometry('1000x760+0+0')
    root.bind_all("<Mod2-q>", exit)
    root.mainloop()
Run Code Online (Sandbox Code Playgroud)

现在,如果用户单击关闭窗口按钮或相应的键盘快捷键,应用程序将不知道第二个窗口不再存在,因此当我们按下按钮打开/关闭窗口时,什么也没有发生。此外,如果我们一直按下打开/关闭按钮,有时按钮不会打开第二个窗口。我究竟做错了什么?

pat*_*yts 5

您想使用wm_protocol顶级小部件的方法。特别是WM_DELETE_WINDOW协议。

>>> import tkinter as tk
>>> root = tk.Tk()
>>> dlg = tk.Toplevel(root)
>>> dlg.wm_title("dialog")
''
>>> root.wm_protocol("WM_DELETE_WINDOW", lambda: print("close root"))
''
>>> dlg.wm_protocol("WM_DELETE_WINDOW", lambda: print("close dialog"))
''
>>> root.mainloop()
close dialog
close root
Run Code Online (Sandbox Code Playgroud)

当我单击窗口框架关闭按钮(大红色 X)时输出最后几行。当我单击它时,这现在不会退出,并且Alt-F4在任一 Tk 窗口上也被调用。

Tk的文档有更多的话要说。我找到的 python 文档似乎相当稀疏​​。