我正在尝试从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
来调用该项目的析构函数。按钮对象也是从buttons
列表中.pop
'd的,以确保没有对该对象的剩余引用。我的代码与Qt文档中会导致这种行为的代码有什么不同?
发布于 2012-03-28 08:11:53
超级简单的修复方法:
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()
。它正确地清理了所有的东西。
发布于 2022-01-30 11:51:01
'jdi‘提供的答案是有效的,尽管如果有人感兴趣,我尝试用每个子PySide6的循环来实现Qt Documentation中建议的东西,并且我在Python Widget中获得了以下代码:
def delete():
while ((child := layout.takeAt(0)) != None):
child.widget().deleteLater()
https://stackoverflow.com/questions/9899409
复制相似问题