【Qt笔记】QSortFilterProxyModel

从本章开始,咱们将逐步了解有关自定义模型的相关内容。尽管前面咱们曾经介绍过 Qt 提供的几个内置模型:QStringListModelQFileSystemModel,但对于变幻无穷的需求而言,这些显然是远远不够的。因而,Qt 也容许咱们对模型进行自定义。正则表达式

在正式开始介绍自定义模形以前,咱们先来了解一个新的类:QSortFilterProxyModel。之因此将这个类放在这里,是由于在必定程序上,咱们可使用QSortFilterProxyModel得到一些可能必须自定义才能达到的效果。QSortFilterProxyModel并不能单独使用。顾名思义,它是一个“代理”,其真正的数据须要另外的一个模型提供。它的做用是对数据进行排序和过滤。排序很好理解,而过滤,则是按照输入的内容对数据及进行筛选,很像 Excel 里面的过滤器。不过 Qt 提供的过滤功能是基于正则表达式的,功能很强大。shell

 

下面咱们根据代码来了解下QSortFilterProxyModel的使用:express

#ifndef SORTVIEW_H
#define SORTVIEW_H

#include <QWidget>
#include <QListView>
#include <QStringListModel>
#include <QSortFilterProxyModel>
#include <QComboBox>

class SortView : public QWidget
{
    Q_OBJECT
public:
    SortView();

private:
    QListView *view;
    QStringListModel *model;
    QSortFilterProxyModel *modelProxy;
    QComboBox *syntaxBox;

private slots:
    void filterChanged(const QString &text);
};

#endif // SORTVIEW_H

头文件中,咱们声明了一个类SortView,继承自QWidget。它有四个成员变量以及一个私有槽函数。函数

#include "sortview.h"

#include <QLineEdit>
#include <QLabel>
#include <QHBoxLayout>

SortView::SortView()
{
    model = new QStringListModel(QColor::colorNames(), this);

    modelProxy = new QSortFilterProxyModel(this);
    modelProxy->setSourceModel(model);
    modelProxy->setFilterKeyColumn(0);

    view = new QListView(this);
    view->setModel(modelProxy);

    QLineEdit *filterInput = new QLineEdit;
    QLabel *filterLabel = new QLabel(tr("Filter"));
    QHBoxLayout *filterLayout = new QHBoxLayout;
    filterLayout->addWidget(filterLabel);
    filterLayout->addWidget(filterInput);

    syntaxBox = new QComboBox;
    syntaxBox->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred);
    syntaxBox->addItem(tr("Regular expression"), QRegExp::RegExp);
    syntaxBox->addItem(tr("Wildcard"), QRegExp::Wildcard);
    syntaxBox->addItem(tr("Fixed string"), QRegExp::FixedString);
    QLabel *syntaxLabel = new QLabel(tr("Syntax"));
    QHBoxLayout *syntaxLayout = new QHBoxLayout;
    syntaxLayout->addWidget(syntaxLabel);
    syntaxLayout->addWidget(syntaxBox);

    QVBoxLayout *layout = new QVBoxLayout(this);
    layout->addWidget(view);
    layout->addLayout(filterLayout);
    layout->addLayout(syntaxLayout);

    connect(filterInput, SIGNAL(textChanged(QString)),
            this, SLOT(filterChanged(QString)));
}

在构造函数中,咱们首先建立一个QStringListModel对象,其内容是 Qt 预约义的全部颜色的名字(利用QColor::colorNames()获取)。而后是QSortFilterProxyModel对象,咱们将其原模型设置为刚刚建立的 model,也就是要为这个 model 进行代理;而后将FilterKeyColumn设置为 0,也就是仅仅对第一列进行过滤。咱们使用一个QStringListModel包装这个数据,这和前面的内容没有什么区别。而后建立一个QSortFilterProxyModel对象,使用它的setSourceModel()函数将前面定义的QStringListModel传进去,也就是咱们须要对这个 model 进行代理。最后重要的一点是,QListView的数据源必须设置为QSortFilterProxyModel,而不是最开始的 model 对象。工具

做为过滤选项,syntaxBox 添加了三个数据项:this

syntaxBox->addItem(tr("Regular expression"), QRegExp::RegExp);
syntaxBox->addItem(tr("Wildcard"), QRegExp::Wildcard);
syntaxBox->addItem(tr("Fixed string"), QRegExp::FixedString);

这正是正则表达式的几种类型。正则表达式本身有一套相对通用的语法,可是对于不一样的语言环境(例如 Java、C# 和 Python),其具体定义可能会略有差异。这里咱们使用的是 Qt 本身的正则表达式处理工具(C++ 自己并无解析正则表达式的机制,虽然 boost 提供了一套)。第一个QregExp::RegExp提供了最通常的正则表达式语法,但这个语法不支持贪婪限定符。这也是 Qt 默认的规则;若是须要使用贪婪限定符,须要使用QRegExp::RegExp2。尽管在 Qt4 的文档中声明,QRegExp::RegExp2将会做为 Qt5 的默认规则,但其实并非这样。第二个咱们提供的是 Unix shell 常见的一种规则,使用通配符处理。第三个即固定表达式,也就是说基本上不使用正则表达式。代理

接下来咱们看看 filterChanged() 函数的实现:code

void SortView::filterChanged(const QString &text)
{
    QRegExp::PatternSyntax syntax = QRegExp::PatternSyntax(
                syntaxBox->itemData(syntaxBox->currentIndex()).toInt());
    QRegExp regExp(text, Qt::CaseInsensitive, syntax);
    modelProxy->setFilterRegExp(regExp);
}

在这段代码中,首先使用QComboBox的选择值建立一个QRegExp::PatternSyntax对象;而后利用这个语法规则构造一个正则表达式,注意咱们在QLineEdit里面输入的内容是经过参数传递进来的,而后设置数据模型代理的过滤表达式。下面能够运行一下看看结果:对象

上图中,咱们输入的是 gr[ae]y 做为正则表达式。这是说,咱们但愿获取这样一个颜色的名字:它的名字中有这样的四个字母,第一个字母是 g,第二个字母是 r,第三个字母要么是 a,要么是 e,第四个字母是 y。若是找到符合条件的名字,就要把它过滤出来,显示到列表中,不符合条件的所有不显示。咱们的程序正是这样的结果。若是你对这个正则表达式不熟悉,请自行查阅有关正则表达式的内容。排序

相关文章
相关标签/搜索