25

我正在尝试从 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 文档,从布局中删除所有对象的正确方法是:

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

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

4

2 回答 2

42

超级简单的修复:

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()虽然使用。它正确地清理了一切。

于 2012-03-28T00:11:53.300 回答
0

'jdi' 提供的答案是有效的,尽管如果有人感兴趣,我尝试使用每个子小部件的循环来实现Qt 文档中建议的内容,并且我得到以下代码在 Python PySide6 中工作:

def delete():
    while ((child := layout.takeAt(0)) != None):
        child.widget().deleteLater()
于 2022-01-30T11:51:00.713 回答