【问题标题】:PySide: Removing a widget from a layoutPySide:从布局中删除小部件
【发布时间】:2012-04-11 13:36:34
【问题描述】:

我正在尝试从 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,以确保没有对该对象的剩余引用。我的代码与 Qt 文档中会导致这种行为的代码有何不同?

【问题讨论】:

  • 我只想祝贺你提出了一个非常完善的问题。最近我看到了很多,它们只是一些没有上下文的句子,或者是我们需要阅读的大量代码。这有一个非常明确的问题,以及一个非常简洁和可运行的代码示例。你的图片也进一步澄清了这个问题。你展示了你已经尝试过的东西。太棒了!

标签: python qt pyside qlayout


【解决方案1】:

超级简单的修复:

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()。它可以正确清理所有内容。

【讨论】:

  • 完美,谢谢~我的头发也谢谢你,因为它不再被拉扯了。
  • 哈。是的,当我第一次开始使用有孩子来来去去的布局时,我也对此有所了解。
  • 只是为了添加一个非常好的答案:Q*Layouts 永远不是小部件的 parent,因为小部件的父级必须是另一个小部件,并且 Q*Layout 不是从 QWidget 派生的。他们只是充当父母身份的转移代理人。即使您从布局中删除小部件,容器小部件仍保留为父级,并且您的按钮仍然存在。
猜你喜欢
  • 2014-09-05
  • 2014-02-08
  • 1970-01-01
  • 1970-01-01
  • 2017-09-06
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多