2

我在某些单元格中有带有 CheckBoxes 的 QTableWidget。我想在我使用表格中的数据时禁用用户在表格单元格上执行鼠标单击(因此他不能更改复选框状态)一段时间。我已经尝试过table.setDisabled(1),但这会禁用整个表,我需要启用滚动。

任何帮助,将不胜感激。

编辑 更准确地说:表格中最多可以有 15x3000 个单元格,填充文本(可编辑)、复选框(可检查)、svg 图形(双击时打开其他窗口)或一些自定义小部件(也有可点击或可编辑部分)。我需要在 1 秒 - 10 秒的时间间隔内禁止用户单击或双击单元格(因此他不能更改其中任何一个)(解决方案必须是快速的,而不是遍历所有项目),但我需要滚动条启用和正常的表可见性。

4

3 回答 3

1

实现此目的的一种方法是继承 QTableWidgetItem 并重新实现setData方法。这样,您可以控制项目是否接受某些角色的值。

要控制所有项目的“可检查性”,您可以向子类添加一个类属性,只要将检查状态角色的值传递给 setData,就可以对其进行测试。

下面是子类的样子:

class TableWidgetItem(QtGui.QTableWidgetItem):
    _blocked = True

    @classmethod
    def blocked(cls):
        return cls._checkable

    @classmethod
    def setBlocked(cls, checkable):
        cls._checkable = bool(checkable)

    def setData(self, role, value):
        if role != QtCore.Qt.CheckStateRole or self.checkable():
            QtGui.QTableWidgetItem.setData(self, role, value)

并且所有项目的“可检查性”都将被禁用,如下所示:

    TableWidgetItem.setCheckable(False)

更新

可以通过为单元小部件添加通用包装类来扩展上述想法。

下面的类将阻止对表格小部件项目的文本和检查状态的更改,以及通过事件过滤器对单元格小部件的一系列键盘和鼠标事件(可以根据需要阻止其他事件)。

需要像这样创建单元格小部件:

    widget = CellWidget(self.table, QtGui.QLineEdit())
    self.table.setCellWidget(row, column, widget)

然后像这样访问:

    widget = self.table.cellWidget().widget()

整个表的阻塞将像这样打开:

    TableWidgetItem.setBlocked(True)
    CellWidget.setBlocked(True)
    # or Blockable.setBlocked(True)

以下是课程:

class Blockable(object):
    _blocked = False

    @classmethod
    def blocked(cls):
        return cls._blocked

    @classmethod
    def setBlocked(cls, blocked):
        cls._blocked = bool(blocked)

class TableWidgetItem(Blockable, QtGui.QTableWidgetItem):
    def setData(self, role, value):
        if (not self.blocked() or (
            role != QtCore.Qt.EditRole and
            role != QtCore.Qt.CheckStateRole)):
            QtGui.QTableWidgetItem.setData(self, role, value)

class CellWidget(Blockable, QtGui.QWidget):
    def __init__(self, parent, widget):
        QtGui.QWidget.__init__(self, parent)
        self._widget = widget
        layout = QtGui.QVBoxLayout(self)
        layout.setContentsMargins(0, 0, 0, 0)
        layout.addWidget(widget)
        widget.setParent(self)
        widget.installEventFilter(self)
        if hasattr(widget, 'viewport'):
            widget.viewport().installEventFilter(self)
        widget.show()

    def widget(self):
        return self._widget

    def eventFilter(self, widget, event):
        if self.blocked():
            etype = event.type()
            if (etype == QtCore.QEvent.KeyPress or
                etype == QtCore.QEvent.KeyRelease or
                etype == QtCore.QEvent.MouseButtonPress or
                etype == QtCore.QEvent.MouseButtonRelease or
                etype == QtCore.QEvent.MouseButtonDblClick or
                etype == QtCore.QEvent.ContextMenu or
                etype == QtCore.QEvent.Wheel):
                return True
        return QtGui.QWidget.eventFilter(self, widget, event)
于 2014-01-08T22:39:19.177 回答
0

只需遍历所有QStandardItems 并更改不应更改的项目的标志值。
您可以使用标志:Qt::ItemIsEditable或/和Qt::ItemIsEnabled

于 2014-01-08T16:36:11.697 回答
0

如果您不想禁用 QCheckBoxes 以外的其他项目,则需要禁用项目本身而不是整个表。有关详细信息,请参阅下面的 python 代码:

'''
    Iterate through all the check boxes in the standard items
    and make sure the enabled flag is cleared so that the items are disabled
'''
for standardItem in standardItems:
    standardItem.setFlags(standardItem.flags() & ~Qt.ItemIsEnabled)

在这里您可以找到相应的文档:

无效 QTableWidgetItem::setFlags(Qt::ItemFlags 标志)

将项目的标志设置为给定的标志。这些决定是否可以选择或修改项目。

于 2014-01-08T20:06:18.020 回答