4

我有一台可以理解多种串行协议的设备。在开发过程中,我创建了简单的 Tkinter UI 来使用协议。每个协议都有一个新的 UI。由于协议有很多命令,我在一个可滚动的画布中实现了整个 UI,以便在较小的显示器上使用时可以滚动它。单独的 UI 工作正常,我现在正尝试将单独的 UI 组合成一个选项卡式 UI。

每个 UI 的共同元素是串行端口选择器,我将其分离出来并放入单独的顶部框架中。然后我实现了一个笔记本,并将每个协议 UI 放入每个选项卡的框架中。

但我无法正确控制大小:我希望将根窗口宽度固定为任何协议框架或串行选择器框架的最大宽度,并禁用水平调整大小。我希望串行选择器始终存在,并且在垂直调整窗口大小时不受影响(仅调整笔记本大小/滚动)。

以下是我到目前为止所拥有的。所有的部分都存在,但笔记本没有填满整个窗口宽度,并且在调整窗口大小时笔记本不会调整大小(调整大小只会增加空白空间)。

def main():
    ## Main window
    root = Tkinter.Tk()
    root.title("Multi-Protocol UI")

    ## Grid sizing behavior in window
    root.grid_rowconfigure(0, weight=0)
    root.grid_rowconfigure(1, weight=1)
    root.grid_columnconfigure(0, weight=1)
    root.grid_columnconfigure(1, weight=0)

    ## Window content
    upper = ttk.Frame(root)   # Serial port selector
    upper.grid(row=0)
    upper.grid_rowconfigure(0, weight=0)
    upper.grid_columnconfigure(0, weight=1)
    upper.grid_columnconfigure(1, weight=0)
    lower = ttk.Frame(root)   # For protocols
    lower.grid(row=1)
    lower.grid(row=1, sticky='nswe')
    lower.grid_rowconfigure(0, weight=1)
    lower.grid_columnconfigure(0, weight=1)
    lower.grid_columnconfigure(1, weight=0)

    # Setup serial control frame
    serial = SerialFrame(master=upper)  # Serial port selector widget + Open button

    ## Protocol GUIs are large: Use a form within a scrollable canvas.
    cnv = Tkinter.Canvas(lower)
    cnv.grid(row=0, column=0, sticky='nswe')
    cnv.grid_rowconfigure(0, weight=1)
    cnv.grid_columnconfigure(0, weight=1)
    cnv.grid_columnconfigure(1, weight=0)
    # Scrollbar for canvas
    vScroll = Tkinter.Scrollbar(
        lower, orient=Tkinter.VERTICAL, command=cnv.yview)
    vScroll.grid(row=0, column=1, sticky='ns')
    cnv.configure(yscrollcommand=vScroll.set)
    # Frame in canvas
    window = Tkinter.Frame(cnv)
    window.grid()
    # Put the frame in the canvas's scrollable zone
    cnv.create_window(0, 0, window=window, anchor='nw')

    # Setup the notebook (tabs) within the scrollable window
    notebook = ttk.Notebook(window)
    frame1 = ttk.Frame(notebook)
    frame2 = ttk.Frame(notebook)
    notebook.add(frame1, text="ProtoA")
    notebook.add(frame2, text="ProtoB")
    notebook.grid(row=0, column=0, sticky='nswe')

    # Create tab frames
    protoA = ProtoAFrame(master=frame1)
    protoA.grid()
    protoA.update_idletasks()
    protoB = ProtoBFrame(master=frame2)
    protoB.grid()
    protoB.update_idletasks()

    ## Update display to get correct dimensions
    root.update_idletasks()
    window.update_idletasks()
    ## Configure size of canvas's scrollable zone
    cnv.configure(scrollregion=(0, 0, window.winfo_width(), window.winfo_height()))

    # Not resizable in width:
    root.resizable(width=0, height=1)

    ## Go!
    root.mainloop()

如何锁定顶部串行框架,将笔记本扩展至全宽,并强制调整窗口大小以仅影响笔记本框架?我是 Tkinter 新手,所以如果我错过了一些“明显”的东西,请保持温柔。

蒂亚!

4

1 回答 1

6

您有嵌套在小部件内的小部件。即使您将笔记本设置为在其容器中正确展开,您也需要确保每个向上的容器也正确展开。你没有这样做。例如,问问自己画布是否已设置为在lower.

既然你刚开始,这是我推荐的。与其试图一次把所有事情都做好,不如选择“分而治之”的方法。首先,为 GUI 的主要区域创建框架。除了那些框架,什么都不做,并让它们的调整大小行为完全符合您的要求。在此阶段为每个小部件赋予自己的颜色会有所帮助,这样您就可以清楚地看到小部件的位置。您以后可以随时更改颜色。此外,如果您只有几个小部件,或者所有小部件都是水平或垂直方向的,那么打包通常比网格更容易使用。

例如:

upper.pack(side="top", fill="x", expand=False)
lower.pack(side="bottom", fill="both", expand=True)

一旦你对这些主要部分进行了适当的调整,就可以只解决其中一个领域的问题了。选择一个区域,然后做同样的事情:添加它的子小部件,或者如果您在一个区域中有区域,则细分为框架。一旦你有这个工作,泡沫,冲洗,重复。

于 2012-11-19T17:50:04.193 回答