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()虽然使用.它可以正确清理一切.
| 归档时间: |
|
| 查看次数: |
12141 次 |
| 最近记录: |