这是我的问题。我有一个QTableView显示一些数据(在模型中设置)和一个QLineEdit Widget,我想用它来搜索所有显示的行中的文本。预期的行为应该是:我在QLineEdit中键入一些文本,QTableView更新本身隐藏所有不包含该数据的行。
问题是,我该如何实现?我发现一个名为hideRows()的QTableView成员函数似乎是正确的选择,但我无法弄清楚我应该如何遍历所有数据以及放置该方法的位置。它应该包含在模型或对话框中? (这实际上是我第一次使用模型,所以我抓住了它们的工作原理)
另外我需要实现导出功能(csv,html或其他),但只需要显示当前显示的行(那些没有隐藏的行)。这可能吗?
感谢您的任何建议。
这是我的代码,直到现在:
from PyQt4.QtCore import *
from PyQt4.QtGui import *
import sys
import Android_extractor
import ui_android_dialog
class recordsTableModel(QAbstractTableModel):
def __init__(self, records, parent = None):
QAbstractTableModel.__init__(self, parent)
self.__records = records
def rowCount(self, parent):
return len(self.__records)
def columnCount(self, parent):
return len(self.__records[0])
def flags(self, index):
return Qt.ItemIsEnabled | Qt.ItemIsSelectable
def data(self, index, role):
if role == Qt.EditRole:
row = index.row()
column = index.column()
return self.__colors[row][column].name()
if role == Qt.DisplayRole:
row = index.row()
column = index.column()
value = self.__records[row][column]
return value
def headerData(self, section, orientation, role):
if role == Qt.DisplayRole:
if orientation == Qt.Horizontal:
return self.__records[0]._fields[section]
class AndroidDialog(QDialog, ui_android_dialog.Ui_androidDialog):
def __init__(self, parent=None):
super(AndroidDialog, self).__init__(parent)
self.setupUi(self)
self.buttonMapper = QSignalMapper(self)
self.buttonMapper.setMapping(self.contactsToolButton, 0)
self.buttonMapper.setMapping(self.groupsToolButton, 1)
self.buttonMapper.setMapping(self.chatsessionToolButton, 2)
self.buttonMapper.setMapping(self.messageToolButton, 3)
self.contactsToolButton.clicked.connect(self.buttonMapper.map)
self.groupsToolButton.clicked.connect(self.buttonMapper.map)
self.chatsessionToolButton.clicked.connect(self.buttonMapper.map)
self.messageToolButton.clicked.connect(self.buttonMapper.map)
self.buttonMapper.mapped.connect(self.dataStackedWidget.setCurrentIndex)
self.newQuery = Android_extractor.AndroidQuery(sys.argv[1])
self.contacts = self.newQuery.getContacts()
self.groups = self.newQuery.getGroups()
self.chats = self.newQuery.getChats()
self.contactsTableView.setModel(recordsTableModel(self.contacts))
self.contactsTableView.resizeColumnsToContents()
self.contactsTableView.resizeRowsToContents()
self.groupsTableView.setModel(recordsTableModel(self.groups))
self.groupsTableView.resizeColumnsToContents()
self.groupsTableView.resizeRowsToContents()
self.chatsessionTableView.setModel(recordsTableModel(self.chats))
self.chatsessionTableView.resizeColumnsToContents()
self.chatsessionTableView.resizeRowsToContents()
app = QApplication(sys.argv)
form = AndroidDialog()
form.show()
app.exec_()
答案 0 :(得分:3)
不是直接在tableview上设置自定义模型,而是在代理上设置为Source模型,然后在视图上设置代理模型。
self.proxyModelContact = QSortFilterProxyModel(self)
self.proxyModelContact.setSourceModel(recordsTableModel(self.contacts))
self.contactsTableView.setModel(self.proxyModelContact)
QSortFilterProxyModel
提供了两种方法:
setFilterRegExp(pattern)
允许您在视图上设置正则表达式过滤器(即只显示与模式匹配的项目)
setFilterKeyColumn(index)
允许您定义将用于进行过滤的列(如果index = -1,将查看所有列)。
您只需将您的有关行的textChanged
信号链接到将更新过滤器的插槽,例如:
def onTextChanged(self, text):
self.proxyModelContact.setFilterRegExp(str(text))