Szukaj…


Różnica między Tk a Toplevel

Tk jest absolutnym katalogiem głównym aplikacji, jest to pierwszy widget, który musi zostać utworzony, a GUI wyłączy się, gdy zostanie zniszczony.

Toplevel to okno w aplikacji, zamknięcie okna zniszczy wszystkie widgety podrzędne umieszczone w tym oknie {1}, ale nie zamknie programu.

try:
    import tkinter as tk #python3
except ImportError:
    import Tkinter as tk #python2

#root application, can only have one of these.
root = tk.Tk() 

#put a label in the root to identify the window.
label1 = tk.Label(root, text="""this is root
closing this window will shut down app""")
label1.pack()

#you can make as many Toplevels as you like
extra_window = tk.Toplevel(root)
label2 = tk.Label(extra_window, text="""this is extra_window
closing this will not affect root""")
label2.pack()

root.mainloop()

Jeśli twój program python reprezentuje tylko jedną aplikację (którą prawie zawsze będzie), to powinieneś mieć tylko jedną instancję Tk , ale możesz utworzyć tyle okien Toplevel ile chcesz.

try:
    import tkinter as tk #python3
except ImportError:
    import Tkinter as tk #python2

def generate_new_window():
    window = tk.Toplevel()
    label = tk.Label(window, text="a generic Toplevel window")
    label.pack()

root = tk.Tk()

spawn_window_button = tk.Button(root,
                                text="make a new window!",
                                command=generate_new_window)
spawn_window_button.pack()

root.mainloop()

{1}: jeśli najwyższy poziom ( A = Toplevel(root) ) jest rodzicem innego najwyższego poziomu ( B = Toplevel(A) ), wówczas zamknięcie okna A spowoduje również zamknięcie okna B.

układanie stosu okien (metoda .lift)

Najbardziej podstawowym przypadkiem do podniesienia określonego okna nad inne, po prostu wywołaj .lift() w tym oknie ( Toplevel lub Tk )

import tkinter as tk #import Tkinter as tk #change to commented for python2

root = tk.Tk()

for i in range(4):
    #make a window with a label
    window = tk.Toplevel(root)
    label = tk.Label(window,text="window {}".format(i))
    label.pack()
    #add a button to root to lift that window
    button = tk.Button(root, text = "lift window {}".format(i), command=window.lift)
    button.grid(row=i)

root.mainloop()

Jeśli jednak okno zostanie zniszczone podczas próby jego podniesienia, pojawi się następujący błąd:

Exception in Tkinter callback
Traceback (most recent call last):
  File "/.../tkinter/__init__.py", line 1549, in __call__
    return self.func(*args)
  File "/.../tkinter/__init__.py", line 785, in tkraise
    self.tk.call('raise', self._w, aboveThis)
_tkinter.TclError: bad window path name ".4385637096"

Często, gdy próbujemy umieścić określone okno przed użytkownikiem, ale zostało ono zamknięte, dobrą alternatywą jest odtworzenie tego okna:

import tkinter as tk #import Tkinter as tk #change to commented for python2

dialog_window = None

def create_dialog():
    """creates the dialog window
  ** do not call if dialog_window is already open, this will
     create a duplicate without handling the other
if you are unsure if it already exists or not use show_dialog()"""
    global dialog_window
    dialog_window = tk.Toplevel(root)
    label1 = tk.Label(dialog_window,text="this is the dialog window")
    label1.pack()
    #put other widgets
    dialog_window.lift() #ensure it appears above all others, probably will do this anyway

def show_dialog():
    """lifts the dialog_window if it exists or creates a new one otherwise"""
    #this can be refactored to only have one call to create_dialog()
    #but sometimes extra code will be wanted the first time it is created
    if dialog_window is None:
        create_dialog()
        return
    try:
        dialog_window.lift()
    except tk.TclError:
        #window was closed, create a new one.
        create_dialog()
    
    
root = tk.Tk()

dialog_button = tk.Button(root,
                          text="show dialog_window",
                          command=show_dialog)
dialog_button.pack()
root.mainloop()

W ten sposób funkcja show_dialog pokaże okno dialogowe, czy istnieje, czy nie, należy również pamiętać, że można wywołać .winfo_exists() aby sprawdzić, czy istnieje, przed próbą podniesienia okna zamiast zawijania go w try:except .

Istnieje również metoda .lower() , która działa w taki sam sposób jak metoda .lift() , z wyjątkiem opuszczania okna na stosie:

import tkinter as tk #import Tkinter as tk #change to commented for python2

root = tk.Tk()
root.title("ROOT")
extra = tk.Toplevel()
label = tk.Label(extra, text="extra window")
label.pack()

lower_button = tk.Button(root,
                         text="lower this window",
                         command=root.lower)
lower_button.pack()

root.mainloop()

Zauważysz, że obniża się nawet pod innymi aplikacjami, aby tylko obniżyć poniżej określonego okna, możesz przekazać je do metody .lower() , podobnie można to zrobić również za pomocą metody .lift() , aby podnieść okno ponad inną jeden.



Modified text is an extract of the original Stack Overflow Documentation
Licencjonowany na podstawie CC BY-SA 3.0
Nie związany z Stack Overflow