如何使tkinter画布动态调整为窗口宽度?

52

我需要在tkinter中获取一个canvas,将其宽度设置为窗口的宽度,然后在用户调整窗口大小时动态重新调整canvas的大小。

有没有什么简单的方法可以实现这个功能?

3个回答

42

我认为添加一些额外的代码来扩展@fredtantini的答案是有必要的,因为它没有处理如何更新在Canvas上绘制的小部件的形状。

为此,您需要使用scale方法并标记所有小部件。下面是一个完整的示例。

from Tkinter import *

# a subclass of Canvas for dealing with resizing of windows
class ResizingCanvas(Canvas):
    def __init__(self,parent,**kwargs):
        Canvas.__init__(self,parent,**kwargs)
        self.bind("<Configure>", self.on_resize)
        self.height = self.winfo_reqheight()
        self.width = self.winfo_reqwidth()

    def on_resize(self,event):
        # determine the ratio of old width/height to new width/height
        wscale = float(event.width)/self.width
        hscale = float(event.height)/self.height
        self.width = event.width
        self.height = event.height
        # resize the canvas 
        self.config(width=self.width, height=self.height)
        # rescale all the objects tagged with the "all" tag
        self.scale("all",0,0,wscale,hscale)

def main():
    root = Tk()
    myframe = Frame(root)
    myframe.pack(fill=BOTH, expand=YES)
    mycanvas = ResizingCanvas(myframe,width=850, height=400, bg="red", highlightthickness=0)
    mycanvas.pack(fill=BOTH, expand=YES)

    # add some widgets to the canvas
    mycanvas.create_line(0, 0, 200, 100)
    mycanvas.create_line(0, 100, 200, 0, fill="red", dash=(4, 4))
    mycanvas.create_rectangle(50, 25, 150, 75, fill="blue")

    # tag all of the drawn widgets
    mycanvas.addtag_all("all")
    root.mainloop()

if __name__ == "__main__":
    main()

8
注意:在使用grid时,会导致窗口管理器崩溃! - steffen
2
现在:如果Canvas不是唯一的小部件,我该如何让它工作?在我的情况下,Canvas下面有个带按钮的Frame,该按钮被Canvas推出来了... - steffen
6
非常好。我想强调创建 ResizingCanvas 时 highlightthickness=0 的重要性,否则画布会不断扩展直到关闭程序。 - Plasma
1
非常感谢!但是似乎除了使用mycanvas.create_image(cur_image_width/2, cur_image_height/2, image=cur_image_bg)添加到画布的图像之外,所有对象都被重新缩放了。您知道如何使用self.scale函数重新调整图像大小吗? - schrottulk
1
这对于图像小部件不起作用。如果我们尝试使用mycanvas.create_image,那么似乎也不起作用。有什么解决方法吗? - Lakpa Tamang
显示剩余3条评论

25

你可以使用 .pack 几何管理器:

self.c=Canvas(…)
self.c.pack(fill="both", expand=True)

这应该能够解决问题。 如果您的画布在框架内,请对框架执行相同操作:

self.r = root
self.f = Frame(self.r)
self.f.pack(fill="both", expand=True)
self.c = Canvas(…)
self.c.pack(fill="both", expand=True)

了解更多信息请访问effbot

编辑:如果您不想要一个“全尺寸”画布,您可以将您的画布绑定到一个函数:

self.c.bind('<Configure>', self.resize)

def resize(self, event):
    w,h = event.width-100, event.height-100
    self.c.config(width=w, height=h)

请再次访问effbot以获取有关事件和绑定的信息。


你能在另一个使用 grid 的框架内使用 pack 吗? - greendino
如果任何一个框架的子元素使用了 pack,那么其他所有子元素也必须使用 pack - Billy

3
这样做就可以了 - 至少在我的Python版本中是这样。画布水平(sticky=E+W和rowconfigure)和垂直(sticky=N+S和columnconfigure)同时调整大小。我将画布背景设置为一种令人讨厌的颜色,但至少当它工作时可以看到。
from tkinter import *
root=Tk()
root.rowconfigure(0,weight=1)
root.columnconfigure(0,weight=1)
canv=Canvas(root, width=600, height=400, bg='#f0f0c0')
canv.grid(row=0, column=0, sticky=N+S+E+W)
root.mainloop()

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接