QSortFilterProxyModel的工作原理

时间:2014-09-17 19:37:17

标签: python qt pyqt

下面的代码会创建一个包含QListView(左)和QTableView(右)的窗口。 两个视图共享相同的数据模型。

QListView列出了字典键,例如' Animals',' Fish'和#39; Birds'当左侧视图的动物'单击项目右侧视图应显示:第一列:' Bison',第二列:' Panther'第三个:'大象'。

为了实现这一目标,已将QSortFilterProxyModel分配给右侧QTableView以过滤其上下文。 在左侧QListView上执行的每次鼠标单击都会触发onClick()功能。此功能检查左侧视图的当前项目是什么。然后它从self.modelDict字典中查询密钥的名称和密钥的对应值。

这是一个窗口截图: enter image description here

显然代码并没有做到它应该做的事情。 QSortFilterProxyModel确实显示了正确的"项目" ...因此两个视图都是同步的。这很好。但右侧表视图在每列中显示相同的键名:"动物","动物","动物"。而目标再一次是显示从.data()方法收到的字典中获取的动物列表:

value=self.dataModel.data(index, QtCore.Qt.ItemDataRole)

其中value是字典,例如:

{1:'Bison',2:'Panther',3:'Elephant'}

请告知。

import os,sys
from PyQt4 import QtCore, QtGui
app=QtGui.QApplication(sys.argv)
elements={'Animals':{1:'Bison',2:'Panther',3:'Elephant'},'Birds':{1:'Duck',2:'Hawk',3:'Pigeon'},'Fish':{1:'Shark',2:'Salmon',3:'Piranha'}}

class DataModel(QtCore.QAbstractTableModel):
    def __init__(self):
        QtCore.QAbstractTableModel.__init__(self)
        self.modelDict={}    
        self.items=[]    
    def rowCount(self, parent=QtCore.QModelIndex()):
        return len(self.items)   
    def columnCount(self, index=QtCore.QModelIndex()):
        return 3
    def data(self, index, role):
        if not index.isValid() or not (0<=index.row()<len(self.items)): return QtCore.QVariant()
        if role==QtCore.Qt.DisplayRole: return self.items[index.row()]
        if role==QtCore.Qt.ItemDataRole: return self.modelDict.get(str(index.data().toString()))

    def addItem(self, itemName=None, column=0):
        totalItems=self.rowCount()+1
        self.beginInsertRows(QtCore.QModelIndex(), totalItems, column)
        if not itemName:            itemName='Item %s'%self.rowCount()
        self.items.append(itemName)
        self.endInsertRows()

    def buildItems(self):
        for key in self.modelDict:
            index=QtCore.QModelIndex()
            self.addItem(key) 

class ProxyModel(QtGui.QSortFilterProxyModel):
    def __init__(self, parent=None):
        super(ProxyModel, self).__init__(parent)

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

        self.dataModel=DataModel()
        self.dataModel.modelDict=elements
        self.dataModel.buildItems() 

        self.proxyModel=ProxyModel()
        self.proxyModel.setFilterKeyColumn(0)    
        self.proxyModel.setSourceModel(self.dataModel)

        self.viewA=QtGui.QListView()
        self.viewA.setModel(self.dataModel)
        self.viewA.clicked.connect(self.onClick)          
        self.viewB=QtGui.QTableView() 
        self.viewB.setModel(self.proxyModel)

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

    def onClick(self):
        index=self.viewA.currentIndex()
        key=self.dataModel.data(index, QtCore.Qt.DisplayRole)  
        value=self.dataModel.data(index, QtCore.Qt.ItemDataRole)        
        self.proxyModel.setFilterRegExp('%s'%key)
        print 'onClick(): key: %s'%type('%s'%key)

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

1 个答案:

答案 0 :(得分:1)

DisplayRole应为不同的行返回不同的数据 根据你的代码:

如果使用data调用index(1,0),则会返回Animals 如果使用data调用index(1,1),则会返回Animals 如果使用data调用index(1,2),则会返回Animals

相反它应该是

如果使用data调用index(1,0),则会返回Animals 如果使用data调用index(1,1),则会返回Bison 如果使用data调用index(1,2),则会返回Panther 如果使用data调用index(1,3),则会返回Elephant 共4列。

要从表格视图隐藏第一列组名,请使用hideColumn(0)

PS。 ItemDataRole是枚举名称而不是值。