将功能从Tkinter帧调用到另一个

时间:2018-02-11 11:40:51

标签: python function class tkinter frame

我有一个页面,其中显示了一些客户详细信息。所以我创建了一个名为"客户详细信息"使用我需要的所有标签,并将这些标签的文本设置为变量。太糟糕了,它不起作用。标签是使用__init__方法创建的,因此我无法更新"他们,因为init只在开始时被调用。所以我想的是创建一个包含所有标签的新功能,当我有必要时我会调用该功能......这就是问题所在。我无法在其他tk.Frame中调用某个函数。以下代码是代码的简化版本。

import tkinter as tk
from tkinter import ttk

class Myapp(tk.Tk):

    def __init__(self, *args, **kwargs):
        tk.Tk.__init__(self, *args, **kwargs)

        container = ttk.Frame(self, borderwidth=10, relief="sunken", width=200, height=100)
        container.pack(side="top", fill="both", expand=True)
        container.grid_rowconfigure(0, weight=1)
        container.grid_columnconfigure(0, weight=1)

        self.frames = {}
        for F in (HomePage, PageOne):
            frame = F(container, self)
            self.frames[F] = frame
            frame.grid(row=0, column=0, sticky="N,S,E,W")
        self.show_frame(HomePage)

    def show_frame(self, cont):
        frame = self.frames[cont]
        frame.tkraise()


class HomePage(tk.Frame):

    def __init__(self, parent, controller):
        tk.Frame.__init__(self, parent)
        label = ttk.Label(self, text="HomePage")
        label.pack()
        button1 = ttk.Button(self, text="Quit",
                            command=lambda: quit())
        button1.pack()
        button2 = ttk.Button(self, text="Call Function in the other page/class to show the label",
                            command=lambda: PageOne.function()) # this is to do it from an other class. I can't do this
        button2.pack()
        button3 = ttk.Button(self, text="Page One",
                            command=lambda: controller.show_frame(PageOne))
        button3.pack()


class PageOne(tk.Frame):

    def __init__(self, parent, controller):
        tk.Frame.__init__(self, parent)
        label = ttk.Label(self, text="PageOne")
        label.pack()
        button1 = ttk.Button(self, text="Quit",
                            command=lambda: quit())
        button1.pack()
        button2 = ttk.Button(self, text="Call Function, in local it works..",
                            command=lambda: self.function())#this is to do it in local
        button2.pack()
        button3 = ttk.Button(self, text="HomePage",
                            command=lambda: controller.show_frame(HomePage))
        button3.pack()

    def function(self):
        label1 = ttk.Label(self, text="It Worked!")
        label1.pack()


app = Myapp()
app.mainloop()

2 个答案:

答案 0 :(得分:2)

要在另一个对象上调用方法,您需要对该对象的引用。您为管理不同页面而复制的代码旨在简化这一过程,但缺少获取页面实例的功能。

所以,首先想到你需要做的是在控制器上添加一个get_page方法:

class Myapp(tk.Tk):
    ...
    def get_page(self, page_class):
        return self.frames[page_class]

通过它,您可以获取页面实例,并且可以使用页面实例调用该方法。

接下来,您需要保留对控制器的引用,以便您可以从其他功能调用它:

class PageOne(tk.Frame):
    def __init__(self, parent, controller):
        self.controller = controller
        ...

最后,您现在可以使用控制器来获取页面,并使用您可以调用该功能的页面。

我的建议是不要使用lambda,除非你绝对需要它,在这种情况下你不需要。使用适当的函数而不是lambda时,编写和调试代码要容易得多。

例如:

class PageOne(tk.Frame):
    def __init__(self, parent, controller):
        ...
        button2 = ttk.Button(..., command=self.do_button)
        ...

    def do_button(self):
        page = self.controller.get_page(PageOne)
        page.function()

答案 1 :(得分:0)

您可以考虑保存对您需要更新的GUI字段的引用,例如,MyApp类中的dict。这样,您可以从类中的任何位置访问它们,无论实际GUI元素放置在何处。

GUI编程通常会产生相当混乱的类,使用dict来跟踪元素减少混乱。其他人喜欢为以后需要访问的每个元素保留单独的类属性。