首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >如何在模型驱动下刷新QTableView

如何在模型驱动下刷新QTableView
EN

Stack Overflow用户
提问于 2014-12-21 17:51:45
回答 2查看 6.1K关注 0票数 0

下面的代码创建了由QTableView驱动的self.myModel (QAbstractTableModel)。“显示所有”self.checkBox链接到self.myModel.cbChanged()方法。

问:如何修改这段代码,使“QTableView”在选中复选框后立即刷新?

目标:当选中复选框时,我们希望显示奇数编号的项目。以及要隐藏的偶数物品。当复选框关闭(未选中)时,我们希望显示偶数项。所有奇数的物品都被藏起来了。

代码语言:javascript
复制
import sys, os
from PyQt4 import QtCore, QtGui
app=QtGui.QApplication(sys.argv)

class TableModel(QtCore.QAbstractTableModel):
    def __init__(self):
        QtCore.QAbstractTableModel.__init__(self)      

        self.items=['One','Two','Three','Four','Five','Six','Seven']
        self.cb_status=True

    def rowCount(self, parent=QtCore.QModelIndex()):   
        return len(self.items)
    def columnCount(self, index=QtCore.QModelIndex()):
        return 1

    def data(self, index, role):
        if not index.isValid() or not (0<=index.row()<len(self.items)):
            return QtCore.QVariant()

        item=str(self.items[index.row()])

        if role==QtCore.Qt.DisplayRole and self.cb_status:
            return item
        else:
            return QtCore.QVariant()

    def cbChanged(self, arg=None):
        self.cb_status=arg  

class Window(QtGui.QWidget):
    def __init__(self):
        super(Window, self).__init__()
        mainLayout=QtGui.QHBoxLayout()
        self.setLayout(mainLayout)   

        self.viewA=QtGui.QTableView()
        self.viewA.horizontalHeader().setResizeMode(QtGui.QHeaderView.Stretch) 

        self.myModel=TableModel()
        self.viewA.setModel(self.myModel)

        self.checkBox=QtGui.QCheckBox("Show All")  
        self.checkBox.stateChanged.connect(self.myModel.cbChanged)
        self.checkBox.setChecked(self.myModel.cb_status)

        mainLayout.addWidget(self.viewA)
        mainLayout.addWidget(self.checkBox)  
        self.show()   

view=Window() 
sys.exit(app.exec_())
EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2014-12-22 00:01:18

为此,您可以使用QSortFilterProxyModel类。这样,我们就不会篡改实际源模型的结构或它的数据。我们只是将主源映射到这个代理模型,视图使用该模型来显示过滤/排序的数据。我们可以随意地影响代理模型中的数据,而不需要篡改源模型的风险。

下面是对源代码进行修改以使用以下内容的代码:

代码语言:javascript
复制
import sys, os
from PyQt4 import QtCore, QtGui
app=QtGui.QApplication(sys.argv)

class TableModel(QtCore.QAbstractTableModel):
    def __init__(self):
        QtCore.QAbstractTableModel.__init__(self)
        self.items=['One','Two','Three','Four','Five','Six','Seven']

    def rowCount(self, parent=QtCore.QModelIndex()):   
        return len(self.items)
    def columnCount(self, index=QtCore.QModelIndex()):
        return 1

    def data(self, index, role):
        if not index.isValid() or not (0<=index.row()<len(self.items)):
            return QtCore.QVariant()

        item=str(self.items[index.row()])

        if role==QtCore.Qt.DisplayRole:
            return item
        else:
            return QtCore.QVariant()


class MySortFilterProxyModel(QtGui.QSortFilterProxyModel):
    def __init__(self):
        super(MySortFilterProxyModel, self).__init__()
        self.cb_status=True

    def cbChanged(self, arg=None):
        self.cb_status=arg
        print self.cb_status
        self.invalidateFilter()

    def filterAcceptsRow(self, sourceRow, sourceParent):
        print_when_odd_flag = self.cb_status
        is_odd = True
        index = self.sourceModel().index(sourceRow, 0, sourceParent)
        print "My Row Data: %s" % self.sourceModel().data(index, role=QtCore.Qt.DisplayRole)

        if (sourceRow + 1) % 2 == 0:
            is_odd = False

        if print_when_odd_flag:
            if is_odd:
                return True
            else:
                return False
        else:
            if not is_odd:
                return True
            else:
                return False


class Window(QtGui.QWidget):
    def __init__(self):
        super(Window, self).__init__()
        mainLayout=QtGui.QHBoxLayout()
        self.setLayout(mainLayout)   

        self.viewA=QtGui.QTableView()
        self.viewA.horizontalHeader().setResizeMode(QtGui.QHeaderView.Stretch) 

        self.myModel=TableModel()

        self.sortModel = MySortFilterProxyModel()
        self.sortModel.setSourceModel(self.myModel)

        self.viewA.setModel(self.sortModel)

        self.checkBox=QtGui.QCheckBox("Show All")  
        self.checkBox.stateChanged.connect(self.sortModel.cbChanged)
        self.checkBox.setChecked(self.sortModel.cb_status)

        mainLayout.addWidget(self.viewA)
        mainLayout.addWidget(self.checkBox)  
        self.show()   

view=Window() 
sys.exit(app.exec_())

如您所见,我已经从UI和主源模型中删除了所有连接。主源模型不关心是否设置了复选框。这使它保持解耦。它更干净。代理模型现在已经被赋予了这一责任。filterAcceptsRow()根据所显示的行的索引是奇数还是偶数,根据复选框状态来执行显示右行的主要任务。

我向它添加了几个print语句,以防您想要根据数据而不是索引来更改逻辑。

查看QSortFilterProxyModel和一些这里的例子上的文档。

希望这是有用的。

票数 5
EN

Stack Overflow用户

发布于 2022-04-01 06:51:58

QTableView实例具有viewport,它具有update方法,通过该方法我们可以更新表内容的

使用viewportviewport方法更新tableview的内容

代码语言:javascript
复制
viewport().update()

试着使用:

代码语言:javascript
复制
self.viewA.viewport().update()
票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/27592088

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档