PySide:从布局中删除小部件

Eti*_*rot 21 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_()
Run Code Online (Sandbox Code Playgroud)

实际发生的是:

实际发生了什么

按钮是不可点击的,显然没有考虑布局计算,但其图像保持不变.

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

while ((child = layout->takeAt(0)) != 0) {
    delete child;
}
Run Code Online (Sandbox Code Playgroud)

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

jdi*_*jdi 32

超级简单的修复:

def deleteButton():
    b = layout.takeAt(2)
    buttons.pop(2)
    b.widget().deleteLater()
Run Code Online (Sandbox Code Playgroud)

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

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

def deleteButton():
    b = layout.takeAt(2)
    buttons.pop(2)
    w = b.widget()
    w.setParent(None)
Run Code Online (Sandbox Code Playgroud)

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

  • 只是为了添加一个非常好的答案:`Q*Layout`s永远不是小部件的`parent`,因为小部件的父级必须是另一个小部件而且`Q*Layout`不是从`QWidget`派生的.他们只是作为父母身份的转移代理人.即使您从布局中删除小部件,容器小部件仍然保留为父级,您的按钮仍然存在. (7认同)