PyQt - 通过隐藏行来进行QTableView搜索

时间:2013-12-13 10:10:08

标签: python qt filter pyqt

这是我的问题。我有一个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_()

1 个答案:

答案 0 :(得分:3)

您应该查看QSortFilterProxyModel

不是直接在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))