pyqt:删除所有项目并添加新项目时正确更新 QGridLayout

use*_*611 2 python pyqt qgridlayout pyqt5

我有一个包含 3 列的 QGridLayout,像这样创建

self.my_grid = QGridLayout()
self.my_grid.setColumnStretch(2, 0)
Run Code Online (Sandbox Code Playgroud)

用 10 个项目填充网格后,它看起来如下所示:

X X X
X X X
X X X
X
Run Code Online (Sandbox Code Playgroud)

在某些情况下,网格项目会发生变化,我尝试删除所有项目并添加新项目:

def updateGrid(self, taskList):

    # delete existing items from grid
    for i in reversed(range(self.my_grid.count())):
        widgetToRemove = self.my_grid.itemAt(i).widget()
        widgetToRemove.setParent(None)
        widgetToRemove.deleteLater()

    # add new buttons
    for task in tasklist:
        btn = QPushButton(task)
        self.my_grid.addWidget(btn)
Run Code Online (Sandbox Code Playgroud)

但随后它会显示一个空单元格,如下所示:

  X X
X X X
X X X
X X
Run Code Online (Sandbox Code Playgroud)

再次调用 updateGrid 后,它有两个空单元格,如下所示:

    X
X X X
X X X
X X X
Run Code Online (Sandbox Code Playgroud)

最后,在再次强制更新后,我得到了我所期望的结果:

X X X
X X X
X X X
X
Run Code Online (Sandbox Code Playgroud)

如何避免出现空单元格?

ekh*_*oro 5

QGridLayout当小部件被删除时,和类QFormLayout不会删除它们的行。空行不占用任何垂直空间,但如果您添加小部件而不指定行和列,它们将放置在最后一个空列旁边。在这些布局中,无法将行数和/或列数重置为零。

在您的示例中,最后一行在零列中有一个项目。因此,在下一次迭代中,第一个小部件将添加到第一列。然后在下一次迭代中,最后一行将有两个项目,因此第一个小部件将放置在第二列 - 依此类推,直到回到开始的位置。

要解决此问题,您可以重新使用现有的空行和列,并更新网格,如下所示:

def updateGrid(self, taskList):
    for i in reversed(range(self.my_grid.count())):
        widgetToRemove = self.my_grid.itemAt(i).widget()
        widgetToRemove.setParent(None)
        widgetToRemove.deleteLater()
    try:
        index = 0
        for row in range(self.my_grid.rowCount()):
            for column in range(self.my_grid.columnCount()):
                btn = QPushButton(taskList[index])
                self.my_grid.addWidget(btn, row, column)
                index += 1
    except IndexError:
        pass
Run Code Online (Sandbox Code Playgroud)