PySide:从布局中删除小部件

时间:2012-03-28 00:02:50

标签: python qt pyside qlayout

我正在尝试从PySide应用程序中的布局中删除Qt小部件。

这是一个最小的例子。它是一个包含5个按钮的小部件,中间的按钮应该在单击时自行删除:

import sys
from PySide import QtGui

app = QtGui.QApplication(sys.argv)
widget = QtGui.QWidget()
layout = QtGui.QVBoxLayout()
buttons = [QtGui.QPushButton(str(x)) for x in xrange(5)]

def deleteButton():
    b = layout.takeAt(2)
    buttons.pop(2)
    del b
buttons[2].clicked.connect(deleteButton)

map(layout.addWidget, buttons)
widget.setLayout(layout)
widget.show()
app.exec_()

实际发生的是:

What actually happens

按钮无法点击,布局计算显然没有考虑到,但图像仍保留在原位。

根据Qt documentation,从布局中删除所有对象的正确方法是:

while ((child = layout->takeAt(0)) != 0) {
    delete child;
}

这里我只想删除第三个按钮,所以我只需要调用takeAt(2),然后调用del b来调用该项目的析构函数。按钮对象也是.pop列表中的buttons'd,以确保没有对该对象的剩余引用。我的代码与导致此类行为的Qt文档中的代码有何不同?

1 个答案:

答案 0 :(得分:31)

超级简单的修复:

def deleteButton():
    b = layout.takeAt(2)
    buttons.pop(2)
    b.widget().deleteLater()

首先必须确保您正在寻址实际按钮,而不是从布局返回的QWidgetItem,然后调用deleteLater(),这将告诉Qt在此插槽结束后控制返回事件后销毁小部件环。

另一个例子说明了问题发生的原因。即使您采用布局项,底层窗口小部件仍然是原始布局窗口小部件的主要部分。

def deleteButton():
    b = layout.takeAt(2)
    buttons.pop(2)
    w = b.widget()
    w.setParent(None)

这不是首选方式,因为它仍然会使对象的清理模糊不清。但它表明清除父级允许它离开可视显示。但请使用deleteLater()。它可以正确清理一切。