我正在尝试从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_()
实际发生的是:
按钮无法点击,布局计算显然没有考虑到,但图像仍保留在原位。
根据Qt documentation,从布局中删除所有对象的正确方法是:
while ((child = layout->takeAt(0)) != 0) {
delete child;
}
这里我只想删除第三个按钮,所以我只需要调用takeAt(2)
,然后调用del b
来调用该项目的析构函数。按钮对象也是.pop
列表中的buttons
'd,以确保没有对该对象的剩余引用。我的代码与导致此类行为的Qt文档中的代码有何不同?
答案 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()
。它可以正确清理一切。