Python Tkinter画布自动调整大小

Python Tkinter画布自动调整大小,python,user-interface,callback,tkinter,Python,User Interface,Callback,Tkinter,所以在回答这个问题的过程中,我遇到了Tkinter的一些奇怪行为。我有一个类,它可以调整画布实例的大小以及在其上绘制的任何小部件。但是,当我运行代码时,无论初始窗口尺寸如何,窗口都会不断扩展,直到填满整个屏幕。发生这种情况后,窗口的行为与预期完全相同,可以正确调整对象的大小。该窗口仅在启动时扩展以填充屏幕 通过阅读Tkinter文档,我可以相信这可能是特定于平台的(但我没有任何证据) 我的问题是:为什么会发生这种情况?我怎样才能阻止它 代码如下: from Tkinter import * #

所以在回答这个问题的过程中,我遇到了Tkinter的一些奇怪行为。我有一个类,它可以调整
画布
实例的大小以及在其上绘制的任何小部件。但是,当我运行代码时,无论初始窗口尺寸如何,窗口都会不断扩展,直到填满整个屏幕。发生这种情况后,窗口的行为与预期完全相同,可以正确调整对象的大小。该窗口仅在启动时扩展以填充屏幕

通过阅读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()
从Tkinter导入*
#Canvas的一个子类,用于处理窗口的大小调整
类大小调整Canvas(画布):
定义初始值(自、父、**kwargs):
画布.\uuuuuuuuuuuuuuuuuuuuuu(自我、父母、**kwargs)
self.bind(“,self.on_resize)
self.height=self.winfo_reqheight()
self.width=self.winfo_reqwidth()
def on_resize(自我,事件):
#确定旧宽度/高度与新宽度/高度的比率
wscale=float(event.width)/self.width
hs刻度=浮动(事件高度)/自身高度
self.width=event.width
self.height=event.height
#调整画布的大小
self.config(宽度=self.width,高度=self.height)
#重新缩放使用“全部”标记标记的所有对象
自刻度(“全部”,0,0,wscale,hscale)
def main():
root=Tk()
myframe=框架(根)
myframe.pack(fill=BOTH,expand=YES)
mycanvas=ResizingCanvas(myframe,宽度=850,高度=400,bg=“红色”)
mycanvas.pack(fill=BOTH,expand=YES)
#向画布添加一些小部件
mycanvas.create_行(0,0,200,100)
mycanvas.create_line(0,100200,0,fill=“red”,dash=(4,4))
mycanvas.create_矩形(50、25、150、75,fill=“blue”)
#标记所有绘制的小部件
mycanvas.addtag_all(“全部”)
root.mainloop()
如果名称=“\uuuuu main\uuuuuuuu”:
main()

我们已经确定
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)
类大小调整Canvas(画布):
定义初始值(自、父、**kwargs):
画布.\uuuuuuuuuuuuuuuuuuuuuu(自我、父母、**kwargs)
打印self.winfo_reqwidth(),self.winfo_reqheight()#>>>854404
self.bind(“,self.on_resize)
def on_resize(自我,事件):
self.width=event.width#>>>854
self.height=event.height#>>>404
self.config(宽度=self.width,高度=self.height)
因此,
应该像这样操作:

  • 检测调整大小
  • 调用函数
  • 将画布设置为新大小
  • 但它是这样做的:

  • 检测调整大小
  • 调用函数
  • 将画布设置为新大小
  • 检测并调整大小,重复
  • 3点到4点之间发生了什么?好的,画布被设置为一个新的大小(它以前的大小+4),但在那之后,highlightthickness将实际大小更改为+4,这将触发无限循环,直到屏幕宽度被击中并断裂

    在这一点上,可以进行正常的大小调整,因为它只有一个大小(组合的高光和画布大小),并且功能正常。如果您添加了一个调整画布大小的按钮,并在画布停止扩展后按下它,它将调整大小,然后再次变得怪异并开始扩展


    我希望这能解释它。我不能100%确定这是100%正确,因此如果有人有任何更正,请放心。

    我认为最初的问题是on_resize()的递归行为,它绑定到配置事件,resize()还调用“self.config(width=self.width,height=self.height)”-这将再次触发配置事件。 删除“self.config(width=self.width,height=self.height)”将修复该问题


    显然,设置“highlightthickness=0”也可以解决问题-我想窗口布局管理器有一些神奇的功能,可以检测到窗口大小没有变化,因此停止递归调用_resize()

    画布在
    highlightthickness
    选项中添加四个额外像素,因此event.width/height正在尝试调整大小以跟上进度。请参阅:您应该把它作为一个答案,因为它解决了设置
    highlightthickness=0
    的问题。我仍然不完全清楚为什么会发生这种情况,因为我假设只调用一次
    config
    来设置此参数,这将导致一次调整大小。我98%确信这至少是96%正确的。您的解释可以解释为什么窗口在开始时一直在扩展,但无法解释为什么它从未尝试在调整大小时再次扩展。不管怎样,答案都很好。谢谢。我认为不同之处在于,当代码中的大小改变时,它只会改变小部件,highlightthickness会发生冲突,但是当手动调整窗口大小时,小部件和highlightthickness会一起改变。我不知道这是否有道理,这是有道理的。如果不深入研究Tkinter源代码(我不打算这么做),我想我不会完全理解它。