Классы Python в Tkinter Grid Manager - PullRequest
1 голос
/ 20 октября 2011

У меня проблема с менеджером геометрии GRID Tkinter или с пользовательскими классами, или с обоими. Скорее всего оба. Вот код:

import sys
from tkinter import 

def main():
    app = App()
    app.master.title("Sample application")
    app.mainloop()

class App(Frame):
    def __init__(self, master=None):
        Frame.__init__(self, master)
        self.grid(sticky=N+S+E+W)

        top = self.winfo_toplevel()
        top.rowconfigure(0, weight=1)
        top.columnconfigure(0, weight=1)

        self.rowconfigure(1, weight=1)
        self.columnconfigure(0, weight=1)
        self.__createWidgets()

    def __createWidgets(self):
        self.f1 = Frame(self, height=100, width=200,
            bg='green')
        self.f1.grid(row=0, column=0, sticky=E+W)

        self.f2 = Frame( self, bg= "yellow", height=100, width=200 ) ############################# CHANGE to self.f2 = myTestFrame( self )
        self.f2.grid(row=1, sticky = N+S+E+W)

        self.f3 = Frame( self, bg = "cyan", height = 100, width = 200 )
        self.f3.grid(row=2, sticky = E+W)

        self.quitButton = Button ( self, text="Quit", command=self.quit )
        self.quitButton.grid(row=4, column=0, columnspan=100,
            sticky=E+W)

class myTestFrame( Frame ):
    def __init__( self, parent):
        Frame.__init__(self, None)

        self.myText = Text( self )
        self.myText.grid()

        #self.testFrame = Frame ( self, bg = "white", height = 100, width = 300 )
        #self.testFrame.grid()

if __name__ == "__main__":
    main()

`

Теперь вот проблема: этот фрагмент кода делает все, что я хочу. Здесь есть три кадра: верхний кадр self.f1, средний кадр self.f2, нижний кадр self.f3 и простая кнопка выхода. Размеры верхней (строка 0) и нижней (строка 2) изменяются по горизонтали (столбец 0), а строка средней рамки (строка 1, столбец 0) получает весь вес для изменения размера по горизонтали и вертикали. Все хорошо ... Пока я не попытаюсь реализовать свой собственный класс myTestFrame.

Всякий раз, когда я пытаюсь заполнить пользовательский кадр в self.f2, скажем, виджетом Text(), дела идут плохо. Он отображает текстовый виджет по умолчанию, но моё изменение размеров идет наперекосяк, размещение GRID сумасшедшее. Что здесь происходит?

EDIT !!! Ответ unutbu ниже, безусловно, помогает с проблемой класса. Теперь GRID-менеджер правильно размещает текстовый виджет в средней рамке self.f2, однако, даже когда я установил sticky в myText.grid(sticky = N+S+E+W), он не будет правильно расширяться. Поэтому я подумал следующее:

1) Установив размеры для корневого окна с помощью конструктора def __init__, равного class App, я установил его так, чтобы первый ряд и нулевой столбец окна приложения получали все выделенное пространство для изменения размера в self.rowconfigure( 1, weight = 1) и self.columnconfigure( 0, weight = 1).

2) Теперь, когда кадр self.f2 размещен в строке 1, он получит все изменения размера с помощью self.f2.grid(N+S+E+W).

3) Теперь я заполняю этот пользовательский кадр виджетом text(), который настроен на ВСЕ пространство в этом кадре, указав self.myText.grid( sticky = N+S+E+W).

Но виджет text() не заполняет self.f2! Когда я делаю это обычно без пользовательского класса, использование атрибута sticky = N+S+E+W работает правильно. Я правильно об этом думаю?

1 Ответ

4 голосов
/ 20 октября 2011

Frame.__init__ необходимо передать родительский элемент:

class myTestFrame( tk.Frame ):
    tk.Frame.__init__(self, parent, *args, **kwargs)

Пример запуска:

import sys
import Tkinter as tk
E=tk.E
W=tk.W
N=tk.N
S=tk.S
def main():
    app = App()
    app.master.title("Sample application")
    app.mainloop()

class App(tk.Frame):
    def __init__(self, master=None):
        tk.Frame.__init__(self, master)
        self.grid(sticky=N+S+E+W)

        top = self.winfo_toplevel()
        top.rowconfigure(0, weight=1)
        top.columnconfigure(0, weight=1)

        self.rowconfigure(1, weight=1)
        self.columnconfigure(0, weight=1)
        self.__createWidgets()

    def __createWidgets(self):
        self.f1 = tk.Frame(self, height=100, width=200, bg='green')
        self.f1.grid(row=0, sticky=E+W)

        self.f2 = myTestFrame(self, bg='yellow', height=100, width=200)

        self.f3 = tk.Frame( self, bg = "cyan", height = 100, width = 200)
        self.f3.grid(row=2, sticky=E+W)

        self.quitButton = tk.Button(self, text="Quit", command=self.quit)
        self.quitButton.grid(row=4, column=0, sticky=E+W)

class myTestFrame( tk.Frame ):
    def __init__(self, parent, cnf={}, **kw):
        tk.Frame.__init__(self, parent, cnf, **kw)
        self.grid(row=1, sticky=N+S+E+W)
        self.rowconfigure(0, weight=1)
        self.columnconfigure(0, weight=1)                
        self.myText = tk.Text(self)
        self.myText.grid(row=0, sticky=N+S+E+W)

if __name__ == "__main__":
    main()
...