目录
1 需求
2 子类化ProxyModel实现全表筛选
3 字符串列表实现中文排序
3.1 Qt5中文排序
3.2 Qt4排序
4 表格的复制粘贴
5 应用
1 需求
模型视图编程是Qt开发的基本功,其中有几个关键问题需要解决:
- 全表筛选,或者说多列搜索
- 中文排序问题
- 表格内容的复制粘贴
下面就这两个问题进行阐述。
2 子类化ProxyModel实现全表筛选
QSortFilterProxyModel是对模型功能的补充,可用于实现排序,筛选等。但是其筛选功能只能对某列进行,代码如下:
proxyModel->setFilterKeyColumn(3);
指定列搜索没法达到全表筛选的功能需求,为达到这一点,需要子类化QSortFilterProxyModel,并重写filterAcceptsRow函数。
下面是代码:
proxymodel.h
#ifndef PROXYMODEL_H
#define PROXYMODEL_H
#include <QSortFilterProxyModel>
class ProxyModel : public QSortFilterProxyModel
{
Q_OBJECT
public:
ProxyModel(QObject *parent);
protected:
bool filterAcceptsRow(int source_row, const QModelIndex &source_parent) const;
};
#endif // PROXYMODEL_H
proxymodel.cpp
#include "proxymodel.h"
ProxyModel::ProxyModel(QObject *parent)
: QSortFilterProxyModel(parent)
{
}
// 重写filterAcceptsRow成员函数 实现全表查询 只要该行有1个以上单元符合条件就显示
bool ProxyModel::filterAcceptsRow(int source_row, const QModelIndex &source_parent) const
{
// 获取源模型的列数
int colCount = sourceModel()->columnCount();
// 循环该行每1列
QString cell;
for(int i = 0; i<colCount; i++)
{
cell = sourceModel()->index(source_row, i, source_parent).data(Qt::DisplayRole).toString();
// 若该列符合条件 则返回真
if(cell.contains(this->filterRegExp()))
return true;
}
// 若各列都不为真 则返回假
return false;
}
这段子类化代码在CSDN其他博客上都有相关描述,但很多还重写了一些其他函数,单纯从全表筛选角度看,是没有必要的,只要重写filterAcceptsRow函数即可。基本逻辑为:
先获取源模型有多少列。
遍历列,逐个判断当前单元是否符合正则化要求,如果符合,直接返回真,就是说这一行是要显示的。
如果遍历完都没有符合项,就返回假,说明这一行不符合要求,不用显示。
之后在视图上,就完成了全表筛选。
如果要筛选指定列,或者共同项,对这个逻辑做代换即可,比如不要遍历所有列,只检查指定列;或者将判断条件从或||变成和&&
3 字符串列表实现中文排序
Qt自带的排序功能只能实现数字和字母排序,要实现中文排序,对QT4和QT5有两种不同的实现方法。
3.1 Qt5中文排序
可以借助QLocale类(没经过验证)
具体代码如下:
QLocale loc(QLocale::Chinese, QLocale::China);
loc.languageToString(QLocale::Chinese);
QCollator qoc(loc);
qSort(listData.begin(), listData.end(), qoc); //正序排序
还有另一个写法:
// 创建一个中文字符串列表
QStringList list;
list << "赵" << "钱" << "孙" << "李" << "周" << "吴" << "郑" << "王";
// 使用 QCollator 进行排序
QCollator collator;
collator.setNumericMode(true); // 数字模式
collator.setCaseSensitivity(Qt::CaseInsensitive); // 不区分大小写
std::sort(list.begin(), list.end(), collator);
这两个写法都因为没装qt5环境,没经过验证,但思路是可行的。
3.2 Qt4排序
Qt4因为没这个库,只能通过写子程序的方式实现。
代码如下:
// 中文排序
QStringList MainWindow::sort(QStringList stringList)
{
QMap<QByteArray,QString> barryMap;
QTextCodec* codec = QTextCodec::codecForName("GBK");
if(codec)
{
for(int i=0; i<stringList.count(); i++)
{
QString text = stringList.at(i);
if( isContainsHz(text))
{
QByteArray barr = codec->fromUnicode(text);
barryMap.insert(barr, text);
}
else
{
barryMap.insert(text.toLatin1(), text);
}
}
}
stringList.clear();
stringList = barryMap.values();
return stringList;
}
// 中文排序 子程序
bool MainWindow::isContainsHz(const QString text)
{
return text.contains( QRegExp("[\\x4e00-\\x9fa5]+") );
}
这个排序可实现对QStringList的中文排序,注意是字符串列表而不是模型。在对QStringList排序后,还需要加载到模型中代入视图显示。
4 表格的复制粘贴
tableview的复制粘贴需要进行子类化,但我们编写小程序时,如果不子类化tableview,也可以直接写在mainwindow里,这里需要对用到的模型变量稍微做下修改即可。
代码如下:
// 实现ctrl+c ctrl+v 选中单元复制粘贴
void MainWindow::keyPressEvent(QKeyEvent *keyEvent)
{
if(keyEvent->matches(QKeySequence::Copy))//复制
{
QModelIndexList indexList = ui->table->selectionModel()->selectedIndexes();
if(indexList.isEmpty())
return;
int startRow = indexList.first().row();
int endRow = indexList.last().row();
int startCol = indexList.first().column();
int endCol = indexList.last().column();
QStringList clipboardTextList;
for(int i = startRow;i <= endRow;i++)
{
QStringList rowText;
for(int j = startCol;j <= endCol;j++)
{
rowText.append(model->data(model->index(i,j)).toString());
}
clipboardTextList.append(rowText.join("\t"));
}
QString clipboardText = clipboardTextList.join("\n" );
QApplication::clipboard()->setText(clipboardText);
}
else if (keyEvent->matches(QKeySequence::Paste))
{
QString clipboardText = QApplication::clipboard()->text();
if(clipboardText.isEmpty())
return;
QStringList rowTextList = clipboardText.split('\n');
if(rowTextList.last().isEmpty())//从word或者excel复制的内容后面可能会带'\n',导致split出来后面有个空字符串。
rowTextList.removeLast();
QModelIndexList indexList = ui->table->selectionModel()->selectedIndexes();
if(indexList.isEmpty())
return;
QModelIndex startIndex = indexList.first();
for(int i = 0;i < rowTextList.size();i++)
{
QStringList itemTextList = rowTextList.at(i).split('\t');
for(int j = 0;j < itemTextList.size();j++)
{
QModelIndex curIndex = model->index(i + startIndex.row(),j + startIndex.column());
if(curIndex.isValid())
{
model->setData(curIndex,itemTextList.at(j));
}
}
}
}
}
用的时候,要根据自己的程序把里面的共有变量进行替换。
5 应用
下面根据上面提到的技术,编写了一个标准规范浏览器,目的是实现标准规范文件的分类展示,同时还能实现全表搜索。
通过在构造函数中读取数据库,将文件加载到表格中,再对分类进行解析,经过去重复,排序,形成左侧分类栏。点击分类可实现proxymodel的筛选展示。在搜索工具条也可以实现全表搜索。