Tkinter画布自动调整大小

6
在回答这个问题的过程中,我遇到了一些有关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()

1
画布在 highlightthickness 选项中添加了四个额外的像素,因此 event.width/height 尝试调整大小以跟上。请参见:https://dev59.com/GmfWa4cB1Zd3GeqPf1Ex - atlasologist
你应该将其作为答案放置,因为它解决了设置highlightthickness=0的问题。然而,我仍然不完全清楚为什么会发生这种情况,因为我认为config只会被调用一次来设置此参数,并且这将导致单个调整大小。 - ebarr
2个回答

5
我们已经确定,highlightthicknessCanvas的一个选项,它是这种行为的罪魁祸首,将其设置为0可以解决问题。
我认为出现这种情况的原因如下:
来自http://effbot.org/tkinterbook/tkinter-events-and-bindings.htm Configure 部件的大小(或位置,在某些平台上)发生了变化。新的大小由传递给回调函数的事件对象的宽度和高度属性提供。 这是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. 将画布设置为新大小
但它正在执行以下操作:
1. 检测调整大小 2. 调用函数 3. 将画布设置为新大小 4. 检测调整大小,重复以上步骤
在第三步和第四步之间发生了什么?嗯,画布被设置为新的大小(其先前大小+4),但在那之后,highlightthickness更改了实际大小加上了4。这导致<Configure>无限循环直到屏幕宽度被占满并中断。
在那一点上,正常调整大小就可以发生,因为它只有一个尺寸可供使用(结合了高亮和画布大小),并且可以正常工作。如果您添加了一个按钮来调整画布大小,并在画布停止扩展后按下它,它将重新调整大小,然后再次变得奇怪并开始扩展。
希望我已经解释清楚了。我不100%确定这是100%正确的,所以如果有任何错误,请随意指出。

1
我有98%的把握这至少96%是正确的。你的解释很有道理,可以解释为什么窗口在开始时会不断扩展,但它并不能解释为什么在调整大小后它不再尝试扩展。无论如何,回答得很好。谢谢。 - ebarr
我认为不同之处在于当代码中更改尺寸时,它只会更改小部件,而highlightthickness会产生冲突,但是当手动调整窗口大小时,小部件和highlightthickness将一起更改。我不知道这是否有任何意义。 - atlasologist
这是有道理的。我认为如果不深入研究Tkinter源代码(而我不打算这样做),我将无法完全理解它。 - ebarr

0

我认为问题的根源在于on_resize()的递归行为绑定到Configure事件上,而resize()也调用了“self.config(width=self.width, height=self.height)” - 这将再次触发Configure事件。 删除“self.config(width=self.width, height=self.height)”将解决问题。

显然,设置“highlightthickness=0”也可以解决问题 - 我猜测窗口布局管理器有一些魔法检测到窗口大小没有变化,因此停止递归调用on_resize()。


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