Переключение между кадрами в tkinter с помощью функций

Из ответа Брайана Окли на вопрос «Переключение между двумя кадрами в tkinter» я пытаюсь изменить работу кнопки на второй странице.

На первой странице с command=lambda: controller.show_frame(“StartPage”) все работает как надо.

На второй странице я хочу что-то добавить, а затем вернуться, но это не работает.

Почему мой обратный вызов не вызывается?

import tkinter as tk                # python 3
from tkinter import font  as tkfont # python 3

class SampleApp(tk.Tk):

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

        self.title_font = tkfont.Font(family='Helvetica', size=18, weight="bold", slant="italic")

        # 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)
        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 (StartPage, PageOne, PageTwo):
            page_name = F.__name__
            frame = F(parent=container, controller=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")

        self.show_frame("StartPage")

    def show_frame(self, page_name):
        '''Show a frame for the given page name'''
        frame = self.frames[page_name]
        frame.tkraise()


class StartPage(tk.Frame):

    def __init__(self, parent, controller):
        tk.Frame.__init__(self, parent)
        self.controller = controller
        label = tk.Label(self, text="This is the start page", font=controller.title_font)
        label.pack(side="top", fill="x", pady=10)

        button1 = tk.Button(self, text="Go to Page One",
                            command=lambda: controller.show_frame("PageOne"))
        button2 = tk.Button(self, text="Go to Page Two",
                            command=lambda: controller.show_frame("PageTwo"))
        button1.pack()
        button2.pack()


class PageOne(tk.Frame):

    def __init__(self, parent, controller):
        tk.Frame.__init__(self, parent)
        self.controller = controller
        label = tk.Label(self, text="This is page 1", font=controller.title_font)
        label.pack(side="top", fill="x", pady=10)
        button = tk.Button(self, text="Go to the start page",
                           command=lambda: controller.show_frame("StartPage"))
        button.pack()


class PageTwo(tk.Frame):

    def __init__(self, parent, controller):
        tk.Frame.__init__(self, parent)
        self.controller = controller
        label = tk.Label(self, text="This is page 2", font=controller.title_font)
        label.pack(side="top", fill="x", pady=10)
        button = tk.Button(self, text="Go to the start page",
                           command=self.go_to())
        button.pack()
    def go_to(self):
        # Add something to do            
        self.controller.show_frame("StartPage")


if __name__ == "__main__":
    app = SampleApp()
    app.mainloop()

person ogeretal    schedule 06.09.2017    source источник


Ответы (1)


Проблема в том, что аргумент ключевого слова command функции Button принимает функцию. Вы создаете свою кнопку с помощью tk.Button(self, ..., command=self.go_to()), поэтому команда self.go_to() оценивается как None.

Почему это так? go_to определяется следующим образом:

def go_to(self):
    # Add something to do 
    self.controller.show_frame("StartPage")

Поэтому, когда Python встречает command=self.go_to(), он вызывает go_to и передает возвращенное значение в параметр command. Это значение None (подразумевается отсутствием оператора return), поэтому ваша команда None, и ваша кнопка не действует.

Вам нужно убрать скобки, поэтому должно быть tk.Button(self, ..., command=self.go_to).

person Right leg    schedule 06.09.2017
comment
Спасибо. Это было перед моими глазами, и я не мог этого видеть. Спасибо - person ogeretal; 06.09.2017
comment
Только один вопрос: если go_to использует параметр, чтобы что-то сделать, а затем возвращается к StartPage, как вы это реализуете (потому что теперь нужны скобки)? - person ogeretal; 06.09.2017
comment
@ogeretal Либо оберните его вокруг функции без аргументов, либо используйте лямбду, например command=lambda: self.go_to(x, y). Эта лямбда означает функцию, которая не принимает аргументов и вызывает f с x и y, так что это функция, как и ожидалось. - person Right leg; 06.09.2017
comment
Спасибо еще раз. И да, вопрос дублируется - person ogeretal; 06.09.2017