使用 tkinter 将多个框架与小部件组合在一个窗口中

sri*_*rad 2 python user-interface tkinter

我正在尝试构建一个基本上只有一个窗口的 GUI。该窗口是不同框架的组合。我想要做的是将这些框架(每个框架进一步包含多个小部件)以某种格式排列在一起:

  • 第一帧(左上):窗口大小的一半(从上到左)
  • 第二帧(左下):从底部到左侧的窗口大小的一半
  • 第三帧(右上):窗口顶部尺寸的 1/3,窗口右侧尺寸的一半。
  • 第四帧(右下):窗户上剩余的空间。

下面,我为该问题添加了一个简单的简约代码。为了简单起见,我使用了相同的 FRAME 类并创建了该类的 4 个实例以添加到主窗口中。我使用 Tkinter 的网格管理器来组织框架中的小部件,然后再次使用网格管理器将这些框架打包到主窗口中。

import Tkinter as tk 

# Test frame class. The same frame is used to create 4 instances. 
class TestFrame(tk.Frame): 
    def __init__(self, master=None, *args, **kwargs):
        tk.Frame.__init__(self, *args, **kwargs)
        super().__init__(master)
        
        # Test Entry fields
        test_label_1                = tk.Label(master, text="Test Label 1").grid(row=0, column=0)
        test_label_1_entry          = tk.Entry(master, bd=5).grid(row=0, column=1)
        test_label_2                = tk.Label(master, text="Test Label 2").grid(row=0, column=2)
        test_label_2_entry          = tk.Entry(master, bd=5).grid(row=0, column=3)
    
        # Test checkbox fields
        test_checkbox_1_label       = tk.Label(master, text="Test check button 1").grid(row=2, column=0)
        test_checkbox_1_entry       = tk.Checkbutton(master, bd=5).grid(row=2, column=1)
        test_checkbox_2_label       = tk.Label(master, text="Test check button 2").grid(row=2, column=2)
        test_checkbox_2_entry       = tk.Checkbutton(master, bd=5).grid(row=2, column=3)
        
        # Test dropdowns
        test_dropdown_label         = tk.Label(master, text="Test dropdown 1").grid(row=3, column=0)
        list_of_options             = ["first option", "second option", "third option", "forth option"]
        first_option                = tk.StringVar(master)
        first_option.set(list_of_options[0])
        test_dropdown_label         = tk.OptionMenu(master, first_option, *list_of_options).grid(row=3, column=1)
        test_dropdown_label         = tk.Label(master, text="Test dropdown 1").grid(row=3, column=0)
        list_of_options             = ["first option", "second option", "third option", "forth option"]
        first_option                = tk.StringVar(master)
        first_option.set(list_of_options[0])
        test_dropdown_label         = tk.OptionMenu(master, first_option, *list_of_options).grid(row=3, column=1)
        pass 
    
    pass

class TestMainApplication(tk.Frame):

    def __init__(self, master, *args, **kwargs):
        tk.Frame.__init__(self, master, *args, **kwargs)
        super().__init__(master)

        title                       = "Test Application"
        self.grid()

        # Assign the weight to each row and column as 1. This is required to align the parameters in grid.
        for r in range(3):
            self.master.rowconfigure(r, weight=1)    
        for c in range(8):
            self.master.columnconfigure(c, weight=1)
        
        self.master.title(title)
        
        # Add two instances of the class TestFrame
        self.testFrame1            = TestFrame(master)
        self.testFrame2            = TestFrame(master)
        self.testFrame3            = TestFrame(master)
        self.testFrame4            = TestFrame(master)

        self.testFrame1.grid(row=0, column=0, rowspan=3, columnspan=3, sticky=tk.W+tk.E+tk.N+tk.S)
        self.testFrame2.grid(row=3, column=0, rowspan=3, columnspan=3, sticky=tk.W+tk.E+tk.N+tk.S)  
        self.testFrame3.grid(row=0, column=3, rowspan=2, columnspan=3, sticky=tk.W+tk.E+tk.N+tk.S)
        self.testFrame4.grid(row=2, column=3, rowspan=4, columnspan=3, sticky=tk.W+tk.E+tk.N+tk.S)
              
        pass 
    pass 


# Main application is created here. 
root                   = tk.Tk()
mainApplication        = TestMainApplication(root)
root.mainloop()
Run Code Online (Sandbox Code Playgroud)

我期望运行此代码后,窗口会分成 4 个部分,按照上面讨论的格式组织。然而,我看到的是所有框架都相互重叠。

环境:MacOS、tkinter 8.6、python 3.6.7

Mik*_*SMT 6

好的,这里有一些问题。

  1. 你的代码有点乱。请考虑使用 PEP8 标准。链接在这里

  2. 如果您打算对框架使用类继承性,为什么不对根窗口执行相同的操作呢?我已经编辑了您的代码以继承Tk().

  3. 重叠的原因是因为在你的TestFrame类中你告诉你的小部件它们属于master而不是self(即 TestFrame 类本身)。要纠正此问题,只需将masterTestFrame 中的所有调用替换为self.

  4. 您同时使用tk.Frame.__init__(self)super().__init__(master)。您实际上只需要super().__init__()在使用这种固有性时使用。

请参阅下面的示例代码:

import tkinter as tk


class TestFrame(tk.Frame):
    def __init__(self):
        super().__init__()
        list_of_options = ["first option", "second option", "third option", "forth option"]
        first_option = tk.StringVar(self)
        first_option.set(list_of_options[0])

        tk.Label(self, text="Test Label 1").grid(row=0, column=0)
        tk.Entry(self, bd=5).grid(row=0, column=1)
        tk.Label(self, text="Test Label 2").grid(row=0, column=2)
        tk.Entry(self, bd=5).grid(row=0, column=3)
        tk.Label(self, text="Test check button 1").grid(row=2, column=0)
        tk.Checkbutton(self, bd=5).grid(row=2, column=1)
        tk.Label(self, text="Test check button 2").grid(row=2, column=2)
        tk.Checkbutton(self, bd=5).grid(row=2, column=3)
        tk.Label(self, text="Test drop down 1").grid(row=3, column=0)
        tk.OptionMenu(self, first_option, *list_of_options).grid(row=3, column=1)


class TestMainApplication(tk.Tk):
    def __init__(self):
        super().__init__()
        self.title("Test Application")

        for r in range(3):
            self.rowconfigure(r, weight=1)
        for c in range(8):
            self.columnconfigure(c, weight=1)

        self.testFrame1 = TestFrame()
        self.testFrame2 = TestFrame()
        self.testFrame3 = TestFrame()
        self.testFrame4 = TestFrame()
        self.testFrame1.grid(row=0, column=0, rowspan=3, columnspan=3, sticky='nsew')
        self.testFrame2.grid(row=3, column=0, rowspan=3, columnspan=3, sticky='nsew')
        self.testFrame3.grid(row=0, column=3, rowspan=2, columnspan=3, sticky='nsew')
        self.testFrame4.grid(row=2, column=3, rowspan=4, columnspan=3, sticky='nsew')


TestMainApplication().mainloop()
Run Code Online (Sandbox Code Playgroud)