我尝试从QGridLayout
中的指定行中删除小部件,如下所示:
void delete_grid_row(QGridLayout *layout, int row)
{
if (!layout || row < 0) return;
for (int i = 0; i < layout->columnCount(); ++i) {
QLayoutItem* item = layout->itemAtPosition(row, i);
if (!item) continue;
if (item->widget()) {
layout->removeWidget(item->widget());
} else {
layout->removeItem(item);
}
delete item;
}
}
但是当我调用它时,应用程序在第一次迭代中在delete item
上与SIGSEGV崩溃。有什么想法吗?
答案 0 :(得分:22)
从QGridLayout
删除行或列(甚至单个单元格)非常棘手。使用下面提供的代码。
首先,请注意QGridLayout::rowCount()
和QGridLayout::columnCount()
始终返回网格布局中内部分配的行和列的数量。例如,如果在新构造的网格布局上调用QGridLayout::addWidget(widget,5,7)
,则行计数将为6,列计数将为8,并且网格布局的所有单元格除了索引上的单元格(5,7)将是空的,因此在GUI中不可见。
请注意,遗憾的是不可能从网格布局中删除此类内部行或列。换句话说,网格布局的行数和列数始终只会增长,但永远不会缩小。
您可以做的是删除行或列的内容,这实际上会与删除行或列本身具有相同的视觉效果。但这当然意味着所有行数和列数都将保持不变。
那么如何清除行或列(或单元格)的内容?遗憾的是,这并不像看起来那么容易。
首先,您需要考虑是否只想从布局中删除小部件,或者是否还希望它们被删除。如果只从布局中删除小部件,则必须在之后将它们放回到不同的布局中,或者手动为它们提供合理的几何图形。如果窗口小部件也被删除,它们将从GUI中消失。提供的代码使用布尔参数来控制小部件删除。
接下来,您必须考虑布局单元格不仅可以包含窗口小部件,还可以包含嵌套布局,它本身可以包含嵌套布局,依此类推。您还需要处理跨越多行和多列的布局项。最后,有一些行和列属性,例如最小宽度和高度,它们不依赖于实际内容,但仍需要处理。
#include <QGridLayout>
#include <QWidget>
/**
* Utility class to remove the contents of a QGridLayout row, column or
* cell. If the deleteWidgets parameter is true, then the widgets become
* not only removed from the layout, but also deleted. Note that we won't
* actually remove any row or column itself from the layout, as this isn't
* possible. So the rowCount() and columnCount() will always stay the same,
* but the contents of the row, column or cell will be removed.
*/
class GridLayoutUtil {
public:
// Removes the contents of the given layout row.
static void removeRow(QGridLayout *layout, int row, bool deleteWidgets = true) {
remove(layout, row, -1, deleteWidgets);
layout->setRowMinimumHeight(row, 0);
layout->setRowStretch(row, 0);
}
// Removes the contents of the given layout column.
static void removeColumn(QGridLayout *layout, int column, bool deleteWidgets = true) {
remove(layout, -1, column, deleteWidgets);
layout->setColumnMinimumWidth(column, 0);
layout->setColumnStretch(column, 0);
}
// Removes the contents of the given layout cell.
static void removeCell(QGridLayout *layout, int row, int column, bool deleteWidgets = true) {
remove(layout, row, column, deleteWidgets);
}
private:
// Removes all layout items which span the given row and column.
static void remove(QGridLayout *layout, int row, int column, bool deleteWidgets) {
// We avoid usage of QGridLayout::itemAtPosition() here to improve performance.
for (int i = layout->count() - 1; i >= 0; i--) {
int r, c, rs, cs;
layout->getItemPosition(i, &r, &c, &rs, &cs);
if (
(row == -1 || (r <= row && r + rs > row)) &&
(column == -1 || (c <= column && c + cs > column))) {
// This layout item is subject to deletion.
QLayoutItem *item = layout->takeAt(i);
if (deleteWidgets) {
deleteChildWidgets(item);
}
delete item;
}
}
}
// Deletes all child widgets of the given layout item.
static void deleteChildWidgets(QLayoutItem *item) {
QLayout *layout = item->layout();
if (layout) {
// Process all child items recursively.
int itemCount = layout->count();
for (int i = 0; i < itemCount; i++) {
deleteChildWidgets(layout->itemAt(i));
}
}
delete item->widget();
}
};
答案 1 :(得分:3)
QGridLayout
本身正在管理QLayoutItem
。我相信您拨打removeWidget
该项目的那一刻将被删除。因此,您在该点有一个无效指针。试图用它做任何事情,而不仅仅是delete
,都会失败。
因此,只是不要删除它,你会没事的。