Как бы я сделал метод, который запускается каждый раз, когда кадр отображается в tkinter
У меня есть приложение графического интерфейса, которое имеет несколько окон и кнопок для перемещения вперед и назад. Для этого я использую контроллер и поднимаю фреймы наверх при каждом изменении окна. Вот мой код для контроллера и типичный кадр.
import Tkinter as tk # python
from tkFileDialog import askopenfilename, asksaveasfilename
from analyzer import Options
TITLE_FONT = ("Helvetica", 18, "bold")
class TennisProgram(tk.Tk):
def __init__(self, *args, **kwargs):
tk.Tk.__init__(self, *args, **kwargs)
# the container is where we'll stack a bunch of frames
# on top of each other, then the one we want visible
# will be raised above the others
container = tk.Frame(self)
#Allow the window to be resized
# container.resizable(width=True, height=True)
container.pack(side="top", fill="both", expand=True)
container.grid_rowconfigure(0, weight=1)
container.grid_columnconfigure(0, weight=1)
#List of options bundles. One bundle for each video
self.options_bundle_list = {}
self.frames = {}
#Init empty array to hold the list of files
#Placed in the container so that all views can access it
self.files = []
for F in (UploadPage, AnalysisOptionsPage, FormAnalysisOptions, MatchAnalysisOptions, MatchAnalysisOptionsPage2, OutputPage, ProcessingPage):
page_name = F.__name__
frame = F(container, self)
self.frames[page_name] = frame
# put all of the pages in the same location;
# the one on the top of the stacking order
# will be the one that is visible.
frame.grid(row=0, column=0, sticky="nsew")
# Name the window
self.title("Tennis Analyzer")
self.show_frame("UploadPage")
def show_frame(self, page_name):
'''Show a frame for the given page name'''
frame = self.frames[page_name]
frame.tkraise()
class UploadPage(tk.Frame):
#TODO Add logic to remove a file path from the list
def __init__(self, parent, controller):
tk.Frame.__init__(self, parent)
self.controller = controller
self.createView()
def createView(self):
label = tk.Label(self, text="Upload Videos for Analysis", font=TITLE_FONT)
label.pack(side="top", fill="x", pady=10)
#Listbox to display the list of files to be processed
self.path_list = tk.Listbox(self)
#Additional options allow listbox to expand as the window is resized
self.path_list.pack(fill=tk.BOTH ,expand=True)
for path in self.controller.files:
self.path_list.insert(tk.END, path)
add_vidoes_button = tk.Button(self, text="Add Videos",
command=self.choose_files)
continue_button = tk.Button(self, text="Continue",
command=lambda: self.controller.show_frame("AnalysisOptionsPage"))
add_vidoes_button.pack(side=tk.TOP)
continue_button.pack(side=tk.BOTTOM)
def choose_files_paths(self):
#don't want a full GUI, so keep the root window from appearing
#self.controller.withdraw()
#Get a file name from the user
filename = askopenfilename()
#Add it to the list of files to be processed
self.controller.files.append(filename)
self.path_list.insert(tk.END, filename)
Код, который я пишу в init, выполняется один раз и создает представление, но мне было интересно, можно ли было иметь функцию, которая запускается каждый раз, когда кадр поднимается, подобно функции onResume в Android. Я хотел бы сделать это в случае, если некоторые базовые данные изменились, как на этой странице загрузки. Например, что, если элемент удаляется из массива, который заполняет представление списка, я хотел бы иметь возможность обновить его.
2 ответа
Tkinter имеет события низкого уровня, такие как <Visibility>
а также <Map>
который должен срабатывать при изменении страниц. К сожалению, они не работают надежно на всех платформах.
Самое простое и надежное решение - создать собственное событие. Во-первых, изменить show_frame
отправить событие в окно, когда оно показано:
def show_frame(self, page_name):
...
frame.event_generate("<<ShowFrame>>")
Далее, каждая страница может привязываться к этому событию, если она должна быть уведомлена, когда она становится видимой:
class UploadPage(tk.Frame):
def __init__(self, parent, controller):
...
self.bind("<<ShowFrame>>", self.on_show_frame)
def on_show_frame(self, event):
print("I am being shown...")
Прошло немного времени, но я нашел решение вашей проблемы (если это кому-то нужно), которое не требует какой-то встроенной функции, а просто библиотека времени.
Я сделал простую в использовании функцию
def updatewindow(fps=60):
time.sleep(1/fps)
root.update()
#root.update_idletasks() # usually you don't need this
Это пример фрагмента, который использует эту функцию
from tkinter import *
import time
root=Tk()
root.title("Clock")
timelbl=Label(root,font=('helvetica',40,'bold'),fg='red',bg='black')
def updatewindow(fps=60):
time.sleep(1/fps)
root.update()
#root.update_idletasks() # usually you don't need this
timelbl.pack()
while True:
timelbl["text"]=time.strftime("%H:%M:%S")
updatewindow(fps=30)
Вы можете запускать что угодно, и это будет работать каждый кадр. Не лучшее решение, но тем не менее решение.