Qt为我们预定义了很多model,前面已经说过了QStringListModel、QDirModel(也算是Qt推荐使用的QFileSystemModel吧,这个在上一章最后重新加上了一段话,没有注意的朋友去看看哦)。今天我们要说的这个QSortFilterProxyModel并不能单独使用,看它的名字就会知道,它只是一个“代理”,真正的数据需要另外的一个model提供,并且它是用来排序和过滤的。所谓过滤,也就是说按照你输入的内容进行数据的筛选,很像Excel里面的过滤器。不过Qt提供的过滤功能是基于正则表达式的,因而功能强大。
 
我们从代码开始看起:
 
sortview.h
InBlock.gif#ifndef SORTVIEW_H
InBlock.gif#define SORTVIEW_H
InBlock.gif
InBlock.gif#include <QtGui>
InBlock.gif
InBlock.gif class SortView : public QWidget
InBlock.gif{
InBlock.gif        Q_OBJECT
InBlock.gif public:
InBlock.gif        SortView();
InBlock.gif
InBlock.gif private:
InBlock.gif        QListView *view;
InBlock.gif        QStringListModel *model;
InBlock.gif        QSortFilterProxyModel *modelProxy;
InBlock.gif        QComboBox *syntaxBox;
InBlock.gif
InBlock.gif private slots:
InBlock.gif         void filterChanged(QString text);
InBlock.gif};
InBlock.gif
InBlock.gif#endif // SORTVIEW_H
 
sortview.cpp
InBlock.gif#include "sortview.h"
InBlock.gif
InBlock.gifSortView::SortView()
InBlock.gif{
InBlock.gif        model = new QStringListModel(QColor::colorNames(), this);
InBlock.gif
InBlock.gif        modelProxy = new QSortFilterProxyModel( this);
InBlock.gif        modelProxy->setSourceModel(model);
InBlock.gif        modelProxy->setFilterKeyColumn(0);
InBlock.gif
InBlock.gif        view = new QListView( this);
InBlock.gif        view->setModel(modelProxy);
InBlock.gif
InBlock.gif        QLineEdit *filterInput = new QLineEdit;
InBlock.gif        QLabel *filterLabel = new QLabel(tr( "Filter"));
InBlock.gif        QHBoxLayout *filterLayout = new QHBoxLayout;
InBlock.gif        filterLayout->addWidget(filterLabel);
InBlock.gif        filterLayout->addWidget(filterInput);
InBlock.gif
InBlock.gif        syntaxBox = new QComboBox;
InBlock.gif        syntaxBox->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred);
InBlock.gif        syntaxBox->addItem(tr( "Regular expression"), QRegExp::RegExp);
InBlock.gif        syntaxBox->addItem(tr( "Wildcard"), QRegExp::Wildcard);
InBlock.gif        syntaxBox->addItem(tr( "Fixed string"), QRegExp::FixedString);
InBlock.gif        QLabel *syntaxLabel = new QLabel(tr( "Syntax"));
InBlock.gif        QHBoxLayout *syntaxLayout = new QHBoxLayout;
InBlock.gif        syntaxLayout->addWidget(syntaxLabel);
InBlock.gif        syntaxLayout->addWidget(syntaxBox);
InBlock.gif
InBlock.gif        QVBoxLayout *layout = new QVBoxLayout( this);
InBlock.gif        layout->addWidget(view);
InBlock.gif        layout->addLayout(filterLayout);
InBlock.gif        layout->addLayout(syntaxLayout);
InBlock.gif
InBlock.gif        connect(filterInput, SIGNAL(textChanged(QString)), this, SLOT(filterChanged(QString)));
InBlock.gif}
InBlock.gif
InBlock.gif void SortView::filterChanged(QString text)
InBlock.gif{
InBlock.gif        QRegExp::PatternSyntax syntax = QRegExp::PatternSyntax(
InBlock.gif                        syntaxBox->itemData(syntaxBox->currentIndex()).toInt());
InBlock.gif        QRegExp regExp(text, Qt::CaseInsensitive, syntax);
InBlock.gif        modelProxy->setFilterRegExp(regExp);
InBlock.gif}
 
至于main()函数的内容,由于和前面的代码几乎是一样的,这里就不再贴出来了。我们使用的是QColor::colorNames()函数提供的数据。这个函数返回值是一个QStringList类型的变量,可以给出预定义的颜色的名字。我们使用一个QStringListModel包装这个数据,这和前面的内容没有什么区别。然后创建一个QSortFilterProxyModel对象,使用它的setSourceModel()函数将前面定义的QStringListModel传进去,也就是我们需要对这个model进行代理。那么我们需要过滤哪一列呢?虽然QStringListModel只有一列,但是我们也需要使用setFilterKeyColumn()函数设置一下,以便让这个proxy知道要过滤的是第0列。最后重要的一点是,QListView的model必须设置为QSortFilterProxyModel,否则是看不到效果的。
 
下面的QLineEdit提供过滤数据的输入,这个没什么好说的。后面的QComboBox列出了三项:
 
InBlock.gifsyntaxBox->addItem(tr( "Regular expression"), QRegExp::RegExp);
InBlock.gifsyntaxBox->addItem(tr( "Wildcard"), QRegExp::Wildcard);
InBlock.gifsyntaxBox->addItem(tr( "Fixed string"), QRegExp::FixedString);
 
这是正则表达式的类型。正则表达式有一套通用的语法,但是对于不同的环境,正则表达式的规则可能是不一样的。第一个QregExp::RegExp提供了最一般的正则表达式语法,不过这个语法不支持贪婪限定符。这也是Qt默认的规则。如果你需要使用贪婪限定符,需要使用QRegExp::RegExp2,根据文档描述,这个将会是 Qt5 的默认规则。第二个是Unix下shell很常见的一种规则。第三个即固定表达式,也就是说基本上不使用正则表达式的。
 
我们使用connect()函数,将QLineEdit的textChanged()信号同slot连接起来。其中我们的slot函数如下所示:
 
InBlock.gif void SortView::filterChanged(QString text)
InBlock.gif{
InBlock.gif        QRegExp::PatternSyntax syntax = QRegExp::PatternSyntax(
InBlock.gif                        syntaxBox->itemData(syntaxBox->currentIndex()).toInt());
InBlock.gif        QRegExp regExp(text, Qt::CaseInsensitive, syntax);
InBlock.gif        modelProxy->setFilterRegExp(regExp);
InBlock.gif}
 
第一步,使用QComboBox的选择值创建一个QRegExp::PatternSyntax对象,然后利用这个语法规则构造一个正则表达式,注意我们在QLineEdit里面输入的内容是通过参数传递进来的,然后设置proxy的过滤器的表达式。好了,就这样运行一下看看效果吧!