Tkinter画布自动调整大小

6 投票
2 回答
13007 浏览
提问于 2025-04-18 01:10

在回答这个问题的过程中,我发现了Tkinter的一些奇怪行为。我有一个类,它可以调整一个Canvas实例和上面绘制的任何小部件的大小。但是当我运行代码时,无论最初窗口的大小是多少,窗口都会不断扩大,直到填满整个屏幕。等到这个情况发生后,窗口的行为就完全正常了,可以正确调整里面的对象大小。这个窗口只在启动时会扩展到填满屏幕。

根据我阅读的Tkinter文档,我觉得这可能和平台有关(不过我没有证据)。

我的问题是:为什么会这样?我该如何阻止这种情况?

代码如下:

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")
    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()

2 个回答

0

我觉得最初的问题在于on_resize()这个函数的递归行为。这个函数是绑定在“配置”事件上的,而resize()函数又调用了“self.config(width=self.width, height=self.height)”,这会再次触发“配置”事件。

如果去掉“self.config(width=self.width, height=self.height)”,问题就能解决。

显然,把“highlightthickness”设置为0也能解决这个问题。我猜测窗口的布局管理器有一些神奇的机制,可以检测到窗口大小没有变化,因此停止了on_resize()的递归调用。

5

我们已经确认了,highlightthickness这个选项是导致这个问题的原因,把它设置为0就能解决这个问题。

我认为这是为什么会发生的原因:

根据这个链接

配置
小部件的大小(或者在某些平台上是位置)发生了变化。新的大小会在传递给回调函数的事件对象的宽度和高度属性中提供。

这是Canvas子类的简化版本:

class ResizingCanvas(Canvas):
    def __init__(self,parent,**kwargs):
        Canvas.__init__(self,parent,**kwargs)
        print self.winfo_reqwidth(),self.winfo_reqheight() #>>>854, 404
        self.bind("<Configure>", self.on_resize)

    def on_resize(self,event):
        self.width = event.width   #>>>854
        self.height = event.height #>>>404
        self.config(width=self.width, height=self.height)

所以,<Configure>应该这样工作:

  1. 检测到大小变化
  2. 调用函数
  3. 将Canvas设置为新大小

但实际上它是这样做的:

  1. 检测到大小变化
  2. 调用函数
  3. 将Canvas设置为新大小
  4. 再次检测到大小变化,重复以上步骤

在第3步和第4步之间发生了什么呢?实际上,Canvas被设置为一个新大小(之前的大小加4),但之后,highlightthickness又把实际大小改成了这个新大小加4,这就导致<Configure>不断触发,形成一个无尽的循环,直到屏幕宽度达到极限,才会停止。

到那时,正常的大小调整就可以进行,因为它只需要基于一个大小(highlight和canvas的组合大小)来工作,这样就能正常运作。如果你添加了一个按钮来调整canvas的大小,并在canvas停止扩展后按下它,它会调整大小,但之后又会变得奇怪,开始再次扩展。

希望这样能解释清楚。我不敢保证这100%正确,如果有人有更正意见,欢迎提出。

撰写回答