tkinter:在画布上使用滚动条

问题描述 投票:0回答:3

我正在尝试使画布可滚动。然而,一旦我尝试设置滚动条来使用画布,tkinter 似乎完全忽略了我最初为画布设置的尺寸。我尝试将它们全部打包在一个框架中,设置画布以填充框架,然后设置框架大小,但这会出现相同的问题,除非我将框架也设置为填充窗口,这不是我想要的。基本上,我想要一个固定大小的画布,上面有滚动条。我当前的代码如下所示(在 python 3.1 中):

from tkinter import *
root=Tk()
frame=Frame(root,width=300,height=300)
frame.grid(row=0,column=0)
canvas=Canvas(frame,bg='#FFFFFF',width=300,height=300,scrollregion=(0,0,500,500))
hbar=Scrollbar(canvas,orient=HORIZONTAL)
hbar.pack(side=BOTTOM,fill=X)
hbar.config(command=canvas.xview)
vbar=Scrollbar(canvas,orient=VERTICAL)
vbar.pack(side=RIGHT,fill=Y)
vbar.config(command=canvas.yview)
canvas.config(width=300,height=300)
canvas.config(xscrollcommand=hbar.set, yscrollcommand=vbar.set)
canvas.pack(side=LEFT,expand=True,fill=BOTH)

root.mainloop()
python tkinter scrollbar tkinter-canvas
3个回答
40
投票

您的滚动条需要将 Frame 作为父级,而不是 Canvas:

from tkinter import *
root=Tk()
frame=Frame(root,width=300,height=300)
frame.pack(expand=True, fill=BOTH) #.grid(row=0,column=0)
canvas=Canvas(frame,bg='#FFFFFF',width=300,height=300,scrollregion=(0,0,500,500))
hbar=Scrollbar(frame,orient=HORIZONTAL)
hbar.pack(side=BOTTOM,fill=X)
hbar.config(command=canvas.xview)
vbar=Scrollbar(frame,orient=VERTICAL)
vbar.pack(side=RIGHT,fill=Y)
vbar.config(command=canvas.yview)
canvas.config(width=300,height=300)
canvas.config(xscrollcommand=hbar.set, yscrollcommand=vbar.set)
canvas.pack(side=LEFT,expand=True,fill=BOTH)

root.mainloop()

之所以有效,是因为

pack
的工作原理。默认情况下,它将尝试缩小(或增大)容器以完全适合其子容器。因为在原始示例中滚动条是画布的子项,所以画布会缩小以适应。

如果您希望滚动条出现在画布内,技巧是使用额外的框架。将画布和滚动条放置在此内部框架中,关闭画布的边框并打开框架的边框。将框架的背景设置为与画布相同,就会出现滚动条位于画布内部的情况。


0
投票

滚动条也可以在没有框架的情况下工作,通过在声明滚动条时传递画布而不是框架

vbar=Scrollbar(canvas,orient=VERTICAL)


-1
投票

当我使用 .pack 选项时,以下代码有效。但使用 .place 选项时,它不起作用

from tkinter import ttk
import tkinter as tk



root = tk.Tk()
container = ttk.Frame(root)
container.pack()
canvas = tk.Canvas(container)
scrollbar = ttk.Scrollbar(container, orient="vertical", command=canvas.yview)
scrollable_frame = ttk.Frame(canvas)

scrollable_frame.bind(
    "<Configure>",
    lambda e: canvas.configure(
        scrollregion=canvas.bbox("all")
    )
)

canvas.create_window((0, 0), window=scrollable_frame, anchor="nw")

canvas.configure(yscrollcommand=scrollbar.set)


x=ttk.Label(scrollable_frame, text="Sample scrolling label")
x.place(x=50,y=50)


canvas.pack(side="left", fill="both", expand=True)
scrollbar.pack(side="right", fill="y")

root.mainloop()

为什么我使用 .place 时不起作用?

© www.soinside.com 2019 - 2024. All rights reserved.