繁体   English   中英

Tkinter画布自动调整大小

[英]Tkinter canvas resizing automatically

所以在回答这个问题的过程中,我遇到了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()

我们已经建立了highlightthickness的一种选择Canvas ,是针对此行为的罪魁祸首,并将其设置为0修复该问题。

这就是为什么(我认为)它会发生的原因:

来自http://effbot.org/tkinterbook/tkinter-events-and-bindings.htm

配置 窗口小部件更改大小(或在某些平台上的位置)。 新大小在传递给回调的事件对象的width和height属性中提供。

这是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. 检测调整大小,重复

3到4之间发生了什么? 好吧,Canvas被设置为一个新的大小(之前的大小+4),但之后,highlightthickness将实际大小更改为+4,这将在无限循环中触发<Configure>直到屏幕宽度被击中它打破了。

此时,可以进行正常的调整大小,因为它只有一个尺寸可以处理(组合的高光和画布大小),并且它正常运行。 如果你添加了一个按钮来调整画布大小并在画布停止展开后按下它,它会调整大小,然后再次变得怪异并开始扩展。

我希望能解释一下。 我不是100%确定这是100%正确,所以如果有人有更正,请随意。

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

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

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM