假设我有一个列表变量datalist
,存储10,000个字符串实体。
QTableView
只需要显示其中一些实体。这就是为QTableView
分配QSortFilterProxyModel
进行所有过滤的原因。
在所有代理工作完成后,QTableView
“接收”要显示的25个实体(因此剩余的9,975个实体被“过滤掉”。
现在,我创建一个QLineEdit
作为搜索字段,用户可以在其中键入关键字以进一步缩小显示的25个实体(项目)的列表范围。为此,我将QLineEdit
的{{1}}信号链接到指定给QTableView Proxy的textChanged
方法。
我在这里看到的问题是代理模型需要返回到原来的10,000个实体长列表再次重新过滤。然后再次。再一次。
我想知道是否有可能创建第二个代理服务器来提取第一个已经过滤掉的代理服务器:25个实体而不是10,000个。
因此生成的模式如下所示:
filterAcceptsRow()
> datalist
> QAbstractTableModel
> QSortFilterProxyModel
> QSortFilterProxyModel
其中:
QTableView
是10,000个实体长列表变量。
datalist
是基础数据模型
QAbstractTableModel
是第一个执行最脏和最慢过滤作业的代理模型
QSortFilterProxyModel
是第二个代理模型,用于对第一个代理数据进行预过滤(用于按用户的关键字进行过滤)。
QSortFilterProxyModel
是一个QTableView本身,用于显示实体项目。
所以,问题是:这是一个有效的想法吗?
答案 0 :(得分:4)
以下代码使用两个ProxyModel过滤10,000个项目。有用...
from PyQt4.QtCore import *
from PyQt4.QtGui import *
import sys
class MyTableModel(QAbstractTableModel):
def __init__(self, parent=None, *args):
QAbstractTableModel.__init__(self, parent, *args)
self.items = [i for i in range(10000)]
def rowCount(self, parent):
return len(self.items)
def columnCount(self, parent):
return 1
def data(self, index, role):
if not index.isValid():
return QVariant()
elif role != Qt.DisplayRole:
return QVariant()
row=index.row()
column=index.column()
if row<len(self.items):
return QVariant(self.items[row])
else:
return QVariant()
class Proxy01(QSortFilterProxyModel):
def __init__(self):
super(Proxy01, self).__init__()
def filterAcceptsRow(self, row, parent):
sourceModel=self.sourceModel()
index=sourceModel.index(row, 0, parent)
name=sourceModel.data(index, Qt.DisplayRole).toString()
if name and not int(name)%10:
return True
return False
class Proxy02(QSortFilterProxyModel):
def __init__(self):
super(Proxy02, self).__init__()
self.keyword=None
def setKeyword(self, arg):
if arg: self.keyword=str(arg)
self.reset()
def filterAcceptsRow(self, row, parent):
sourceModel=self.sourceModel().sourceModel()
index=sourceModel.index(row, 0, parent)
name=sourceModel.data(index, Qt.DisplayRole).toString()
if self.keyword and name and not self.keyword.lower() in str(name).lower():
return False
return True
class MyWindow(QWidget):
def __init__(self, *args):
QWidget.__init__(self, *args)
self.tablemodel=MyTableModel(self)
self.proxy1=Proxy01()
self.proxy1.setSourceModel(self.tablemodel)
self.proxy2=Proxy02()
self.proxy2.setSourceModel(self.proxy1)
tableviewA=QTableView()
tableviewA.setModel(self.proxy2)
searchEdit=QLineEdit()
searchEdit.textChanged.connect(self.proxy2.setKeyword)
layout = QVBoxLayout(self)
layout.addWidget(tableviewA)
layout.addWidget(searchEdit)
self.setLayout(layout)
def test(self, arg):
print arg
if __name__ == "__main__":
app = QApplication(sys.argv)
w = MyWindow()
w.show()
sys.exit(app.exec_())
这是一种单一的代理方法。对于两个代理实现,按关键字搜索速度明显较慢:
from PyQt4.QtCore import *
from PyQt4.QtGui import *
import sys
class MyTableModel(QAbstractTableModel):
def __init__(self, parent=None, *args):
QAbstractTableModel.__init__(self, parent, *args)
self.items = [i for i in range(10000)]
def rowCount(self, parent):
return len(self.items)
def columnCount(self, parent):
return 1
def data(self, index, role):
if not index.isValid():
return QVariant()
elif role != Qt.DisplayRole:
return QVariant()
row=index.row()
column=index.column()
if row<len(self.items):
return QVariant(self.items[row])
else:
return QVariant()
class Proxy01(QSortFilterProxyModel):
def __init__(self):
super(Proxy01, self).__init__()
self.keyword=None
def setKeyword(self, arg):
if arg: self.keyword=str(arg)
self.reset()
def filterAcceptsRow(self, row, parent):
sourceModel=self.sourceModel()
index=sourceModel.index(row, 0, parent)
name=sourceModel.data(index, Qt.DisplayRole).toString()
if self.keyword and name and not self.keyword.lower() in str(name).lower():
return False
return True
class MyWindow(QWidget):
def __init__(self, *args):
QWidget.__init__(self, *args)
self.tablemodel=MyTableModel(self)
self.proxy1=Proxy01()
self.proxy1.setSourceModel(self.tablemodel)
tableviewA=QTableView()
tableviewA.setModel(self.proxy1)
searchEdit=QLineEdit()
searchEdit.textChanged.connect(self.proxy1.setKeyword)
layout = QVBoxLayout(self)
layout.addWidget(tableviewA)
layout.addWidget(searchEdit)
self.setLayout(layout)
def test(self, arg):
print arg
if __name__ == "__main__":
app = QApplication(sys.argv)
w = MyWindow()
w.show()
sys.exit(app.exec_())