概述
您只能将滚动条与一些小部件和根小部件相关联,而Frame不是该小部件组的一部分。
至少有几种方法可以做到这一点。如果您需要一组简单的垂直或水平小部件,您可以使用文本小部件和window_create添加小部件的方法。这种方法很简单,但不允许小部件的复杂布局。
一个更常见的通用解决方案是创建一个画布小部件并将滚动条与该小部件相关联。然后,在该画布中嵌入包含标签小部件的框架。确定框架的宽度/高度并将其输入到画布scrollregion选项中,以便滚动区域与框架的大小完全匹配。
为什么将小部件放在框架中而不是直接放在画布中?附加到画布的滚动条只能滚动使用其中一种create_方法创建的项目。您不能使用 、 或 滚动添加到画布pack的place项目grid。通过使用框架,您可以在框架内使用这些方法,然后create_window为框架调用一次。
直接在画布上绘制文本项目并不难,因此如果框架嵌入在画布中的解决方案看起来过于复杂,您可能需要重新考虑这种方法。由于您正在创建一个网格,因此每个文本项的坐标将非常容易计算,特别是如果每一行的高度相同(如果您使用单一字体,则可能是这样)。
要直接在画布上绘图,只需弄清楚您正在使用的字体的行高(并且有相应的命令)。然后,每个 y 坐标为row*(lineheight+spacing)。x 坐标将是基于每列中最宽项的固定数字。如果您为所有内容为其所在的列指定标签,则可以使用单个命令调整列中所有项目的 x 坐标和宽度。
面向对象的解决方案
这是使用面向对象方法的框架嵌入画布解决方案的示例:
import tkinter as tk
class Example(tk.Frame):
def __init__(self, parent):
tk.Frame.__init__(self, parent)
self.canvas = tk.Canvas(self, borderwidth=0, background="#ffffff")
self.frame = tk.Frame(self.canvas, background="#ffffff")
self.vsb = tk.Scrollbar(self, orient="vertical", command=self.canvas.yview)
self.canvas.configure(yscrollcommand=self.vsb.set)
self.vsb.pack(side="right", fill="y")
self.canvas.pack(side="left", fill="both", expand=True)
self.canvas.create_window((4,4), window=self.frame, anchor="nw",
tags="self.frame")
self.frame.bind("<Configure>", self.onFrameConfigure)
self.populate()
def populate(self):
'''Put in some fake data'''
for row in range(100):
tk.Label(self.frame, text="%s" % row, width=3, borderwidth="1",
relief="solid").grid(row=row, column=0)
t="this is the second column for row %s" %row
tk.Label(self.frame, text=t).grid(row=row, column=1)
def onFrameConfigure(self, event):
'''Reset the scroll region to encompass the inner frame'''
self.canvas.configure(scrollregion=self.canvas.bbox("all"))
if __name__ == "__main__":
root=tk.Tk()
example = Example(root)
example.pack(side="top", fill="both", expand=True)
root.mainloop()
程序解决方案
这是一个不使用类的解决方案:
import tkinter as tk
def populate(frame):
'''Put in some fake data'''
for row in range(100):
tk.Label(frame, text="%s" % row, width=3, borderwidth="1",
relief="solid").grid(row=row, column=0)
t="this is the second column for row %s" %row
tk.Label(frame, text=t).grid(row=row, column=1)
def onFrameConfigure(canvas):
'''Reset the scroll region to encompass the inner frame'''
canvas.configure(scrollregion=canvas.bbox("all"))
root = tk.Tk()
canvas = tk.Canvas(root, borderwidth=0, background="#ffffff")
frame = tk.Frame(canvas, background="#ffffff")
vsb = tk.Scrollbar(root, orient="vertical", command=canvas.yview)
canvas.configure(yscrollcommand=vsb.set)
vsb.pack(side="right", fill="y")
canvas.pack(side="left", fill="both", expand=True)
canvas.create_window((4,4), window=frame, anchor="nw")
frame.bind("<Configure>", lambda event, canvas=canvas: onFrameConfigure(canvas))
populate(frame)
root.mainloop()