将可检查组合框的文本显示到 QTableWidget
Display the text of a checkable combobox to a QTableWidget
我的代码有一个带有复选框的组合框,每次我选中一个复选框时,我想在 QTableWidget 行中显示文本,如果未选中复选框,我想从 QTableWidget 中删除该行。
尽管最初使用带有 QCombobox 的 QTableWidget 管理逻辑看起来很简单,但从长远来看可能会很复杂。一个更优雅的解决方案是使用将由 QComboBox 和 QTableView 共享的模型,其中过滤由 QSortFilterProxyModel 完成,当共享两个元素时,同一模型将看到 QTableWidget 中的复选框,该复选框已被委托删除。
from PyQt5 import QtCore, QtGui, QtWidgets
class CheckedFilterProxyModel(QtCore.QSortFilterProxyModel):
def filterAcceptsRow(self, source_row, source_parent):
index = self.sourceModel().index(source_row, 0, source_parent)
return index.data(QtCore.Qt.CheckStateRole) == QtCore.Qt.Checked
class RemoveCheckBoxDelegate(QtWidgets.QStyledItemDelegate):
def initStyleOption(self, option, index):
super(RemoveCheckBoxDelegate, self).initStyleOption(option, index)
option.features &= ~QtWidgets.QStyleOptionViewItem.HasCheckIndicator
class Widget(QtWidgets.QWidget):
def __init__(self, parent=None):
super(Widget, self).__init__(parent)
self.m_combobox = QtWidgets.QComboBox()
self.m_tableview = QtWidgets.QTableView()
self.m_model = QtGui.QStandardItemModel(0, 2)
self.m_proxy = CheckedFilterProxyModel(self)
self.m_proxy.setSourceModel(self.m_model)
self.m_combobox.setModel(self.m_model)
self.m_tableview.setModel(self.m_proxy)
delegate = RemoveCheckBoxDelegate(self)
self.m_tableview.setItemDelegateForColumn(0, delegate)
lay = QtWidgets.QVBoxLayout(self)
lay.addWidget(self.m_combobox)
lay.addWidget(self.m_tableview)
options = ["Test1", "Test2", "Test3"]
self.fill_combobox(options)
def fill_combobox(self, options):
self.m_model.clear()
for option in options:
item = QtGui.QStandardItem(option)
item.setFlags(QtCore.Qt.ItemIsUserCheckable | QtCore.Qt.ItemIsEnabled)
item.setData(QtCore.Qt.Unchecked, QtCore.Qt.CheckStateRole)
items = [item] + [
QtGui.QStandardItem() for _ in range(self.m_model.rowCount() - 1)
]
self.m_model.appendRow(items)
if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
w = Widget()
w.show()
sys.exit(app.exec_())
我的代码有一个带有复选框的组合框,每次我选中一个复选框时,我想在 QTableWidget 行中显示文本,如果未选中复选框,我想从 QTableWidget 中删除该行。
尽管最初使用带有 QCombobox 的 QTableWidget 管理逻辑看起来很简单,但从长远来看可能会很复杂。一个更优雅的解决方案是使用将由 QComboBox 和 QTableView 共享的模型,其中过滤由 QSortFilterProxyModel 完成,当共享两个元素时,同一模型将看到 QTableWidget 中的复选框,该复选框已被委托删除。
from PyQt5 import QtCore, QtGui, QtWidgets
class CheckedFilterProxyModel(QtCore.QSortFilterProxyModel):
def filterAcceptsRow(self, source_row, source_parent):
index = self.sourceModel().index(source_row, 0, source_parent)
return index.data(QtCore.Qt.CheckStateRole) == QtCore.Qt.Checked
class RemoveCheckBoxDelegate(QtWidgets.QStyledItemDelegate):
def initStyleOption(self, option, index):
super(RemoveCheckBoxDelegate, self).initStyleOption(option, index)
option.features &= ~QtWidgets.QStyleOptionViewItem.HasCheckIndicator
class Widget(QtWidgets.QWidget):
def __init__(self, parent=None):
super(Widget, self).__init__(parent)
self.m_combobox = QtWidgets.QComboBox()
self.m_tableview = QtWidgets.QTableView()
self.m_model = QtGui.QStandardItemModel(0, 2)
self.m_proxy = CheckedFilterProxyModel(self)
self.m_proxy.setSourceModel(self.m_model)
self.m_combobox.setModel(self.m_model)
self.m_tableview.setModel(self.m_proxy)
delegate = RemoveCheckBoxDelegate(self)
self.m_tableview.setItemDelegateForColumn(0, delegate)
lay = QtWidgets.QVBoxLayout(self)
lay.addWidget(self.m_combobox)
lay.addWidget(self.m_tableview)
options = ["Test1", "Test2", "Test3"]
self.fill_combobox(options)
def fill_combobox(self, options):
self.m_model.clear()
for option in options:
item = QtGui.QStandardItem(option)
item.setFlags(QtCore.Qt.ItemIsUserCheckable | QtCore.Qt.ItemIsEnabled)
item.setData(QtCore.Qt.Unchecked, QtCore.Qt.CheckStateRole)
items = [item] + [
QtGui.QStandardItem() for _ in range(self.m_model.rowCount() - 1)
]
self.m_model.appendRow(items)
if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
w = Widget()
w.show()
sys.exit(app.exec_())