确定当前在最上面的tkinter窗口

10
我用Python 2.7和Tkinter编写了一个应用程序。我创建了一个工具栏,其中包含几个按钮,打开相应的顶级窗口显示不同的选项。我使用ttk.Checkbutton,使用“toolbutton”样式来指示选项窗口是打开还是关闭。
问题在于,如果选择另一个窗口,则选项窗口将到后面。目前,如果再次选择工具按钮,则选项窗口将关闭。然而,我只想在选项窗口在顶部时关闭窗口。如果选项窗口没有在顶部,我想让窗口移到前面。
以下是我正在使用的部分代码:
class MainWindow:
    def __init__(self,application):
        self.mainframe=tk.Frame(application)
        application.geometry("900x600+30+30")

        self.otherOptionsSelect=tk.IntVar()
        self.otherOptions_Button=ttk.Checkbutton(application,style='Toolbutton',variable=self.otherOptionsSelect,
                                                onvalue=1, offvalue=0,image=self.optionsIcon, command=self.otherOptions)
    def otherOptions(self):

        if self.otherOptionsSelect.get()==0:
            self.otherOptions.destroy()
            return

        self.otherOptions=tk.Toplevel()
        self.otherOptions.title("IsoSurface Options")
        self.otherOptions.geometry("200x165+"+str(int(application.winfo_x())+555)+"+"+str(int(application.winfo_y())+230))

        self.otherOptApply_button=ttk.Button(self.otherOptions,text="Apply",command=self.showFrame)
        self.otherOptApply_button.place(x=20,y=80,width=50,height=30)

        self.otherOptClose_button=ttk.Button(self.otherOptions,text="Close",command=self.otherOptionsClose)
        self.otherOptClose_button.place(x=80,y=80,width=50,height=30)

    def otherOptionsClose(self):
        self.otherOptionsSelect.set(0)
        self.otherOptions.destroy()

这里是我编写的整个应用程序的图片: enter image description here 在上面的图片中,每个窗口都有它们各自的ttk.checkbutton。目前,切换checkbutton会打开或关闭窗口。然而,我真正想要的是,如果该窗口在应用程序前面,则将其关闭;如果该窗口在应用程序后面,则将其置于最前面。
希望这能解决一些问题。
提前致谢!

如果您选择一个复选框,那么带有复选框的窗口是否总是在顶部,或者您是否有一个窗口管理器,可以让您单击后台窗口上的按钮? - Bryan Oakley
也许我应该解释得更清楚一些。我有一个“主窗口”,上面有一个带有多个复选框的工具栏。每个复选框打开/关闭一个“toplevel”窗口,但是“主窗口”仍然处于活动状态。这些“toplevel”窗口并不总是在“主窗口”的顶部,它们可以到后台去。也许我会尝试发布一些图片。 - Onlyjus
那么,如果这些其他窗口之一在主窗口后面但在屏幕上,应该发生什么?它不是真正的顶部,所以你想删除它吗? - Bryan Oakley
也许可以在每个窗口上创建一个绑定,以便您可以跟踪它们重叠的顺序?即一个button-1绑定,用于跟踪焦点变化... - Niall Byrne
那可能并不总是有效。那只会给我每个窗口生成的顺序。可能性是“较旧”的窗口会在后面,但并非总是如此。 - Onlyjus
显示剩余2条评论
1个回答

11
事实上,可以检查窗口的堆叠顺序。使用Tkinter,您需要执行一些有趣的tcl evals来获取信息。我在TkDoc的Windows and Dialogs部分找到了答案,请向下滚动直到找到“Stacking Order”。这段代码让我感到困惑,直到我开始交互式地尝试它。我的测试代码如下:
import Tkinter as tk
root = tk.Tk()
root.title('root')
one = tk.Toplevel(root)
one.title('one')
two = tk.Toplevel(root)
two.title('two')

我随后调整了窗口,使得两个窗口在顶部,一个在它们下面,而根窗口在它们所有窗口的下方。在这种配置下,以下怪异现象可以告诉你窗口的相对层次:

root.tk.eval('wm stackorder '+str(two)+' isabove '+str(root))

返回1,意味着“是的,窗口2在窗口根上方。” 而以下内容:
root.tk.eval('wm stackorder '+str(root)+' isabove '+str(two))

返回0,意味着“不,窗口根不在窗口二上方。” 您还可以使用以下命令:

root.tk.eval('wm stackorder '+str(root))

以下是以奇怪的字符串形式返回完整的窗口层叠顺序:

'. .68400520L .68401032L'

当你运行以下命令时,这将开始变得有意义:

str(root)
str(one)
str(two)

并且找出根节点的内部名称为'.',其中一个是'.68400520L',另一个是'.68401032L'。你需要倒序阅读root.tk.eval('wm stackorder '+str(root))的输出,这样就可以知道两个节点在上面,一个在下面,而根节点在它们下面。


在Python 3.8中,tkinter不再使用随机数字串作为对小部件的引用,因此窗口堆叠顺序的输出将是:. .!toplevel .!toplevel2。根仍然是一个点。 - Luther

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接