tkinter
Wiele okien (widżety TopLevel)
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.