下面的代码会创建一个包含QListView
(左)和QTableView
(右)的窗口。
两个视图共享相同的数据模型。
QListView
列出了字典键,例如' Animals',' Fish'和#39; Birds'当左侧视图的动物'单击项目右侧视图应显示:第一列:' Bison',第二列:' Panther'第三个:'大象'。
为了实现这一目标,已将QSortFilterProxyModel
分配给右侧QTableView
以过滤其上下文。
在左侧QListView上执行的每次鼠标单击都会触发onClick()
功能。此功能检查左侧视图的当前项目是什么。然后它从self.modelDict
字典中查询密钥的名称和密钥的对应值。
这是一个窗口截图:
显然代码并没有做到它应该做的事情。 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_())
答案 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
是枚举名称而不是值。