Как бы я сделал метод, который запускается каждый раз, когда кадр отображается в 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)

Вы можете запускать что угодно, и это будет работать каждый кадр. Не лучшее решение, но тем не менее решение.

Другие вопросы по тегам