Qt4用子类化ProxyModel和子类化MainWindow实现全表筛选,中文排序和复制粘贴

目录

1 需求

2 子类化ProxyModel实现全表筛选

3 字符串列表实现中文排序

3.1 Qt5中文排序

3.2 Qt4排序

4 表格的复制粘贴

5 应用

5.1 新建工程

5.2 新建proxymodel类

5.3 完成主程序构造函数及点击事件,双击事件的槽函数

5.4 借助字符串列表操作完成分类列表视图显示和模型挂载


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

代码解析:

生成子类需要注意构造函数,虽然是空的,但在new一个新对象时,需要挂载在主界面this上。这样在析构时可以随着主界面一起释放内存。

此外,重写filterAcceptsRow即可,无需更改其他函数。

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;
}

代码解析:

filterAcceptsRow函数的输入参数为当前行号,当前模型索引和源模型。输出参数是布尔型,如果为真则显示,假则不显示。

函数的思路是:对当前行内的信息进行判断,符合要求则显示,不符合要求则不显示。根据我们的全表搜索需求,在当前行应该对行内所有单元进行遍历,逐个查看其是否包含搜索文本。如果包含则应返回真。

这段子类化代码在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); //正序排序

还有另一个写法,是用QCollator实现。这个类Qt4也没有。具体代码如下:

// 创建一个中文字符串列表
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 应用

下面根据上面提到的技术,编写了一个标准规范浏览器,目的是实现标准规范文件的分类展示,同时还能实现全表搜索。

5.1 新建工程

创建main window工程,并在main函数里添加支持utf8编码的语句

代码为:

#include <QtGui/QApplication>
#include "mainwindow.h"
#include <QTextCodec>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);

    // 文本编码规定
    QTextCodec::setCodecForTr(QTextCodec::codecForName("UTF8"));
    QTextCodec::setCodecForLocale(QTextCodec::codecForName("UTF8"));
    QTextCodec::setCodecForCStrings(QTextCodec::codecForName("UTF8"));


    MainWindow w;
    w.show();

    return a.exec();
}

5.2 新建proxymodel类

新建c++类,并完成函数重写:

5.3 完成主程序构造函数及点击事件,双击事件的槽函数

这个程序设计了两个点击事件,分别为:

  • 点击列表显示对应内容;
  • 双击表格打开对应文件。

实现思路为:


// 列表点击事件
void MainWindow::on_list_clicked(const QModelIndex &index)
{
    int row = index.row();
    QString item = list->stringList().at(row);
    if(item!="全部文档")
    {
        proxyModel->setFilterRegExp(QRegExp(item, Qt::CaseInsensitive, QRegExp::FixedString));
    }
    else
    {
        proxyModel->setFilterRegExp(QRegExp("", Qt::CaseInsensitive, QRegExp::FixedString));
    }
}


// 表格双击事件
void MainWindow::on_table_doubleClicked(const QModelIndex &index)
{
    int row = ui->table->currentIndex().row();
    QString fileName = proxyModel->data(proxyModel->index(row,4)).toString();
    QString path = "box\\"+fileName.replace(QRegExp("/"),"\\");
    QProcess::execute("explorer \""+path+"\"");
}

使用QProcess加参数的方法来打开文件。为了避免文件路径有空格,需要在后面加上双引号。

5.4 借助字符串列表操作完成分类列表视图显示和模型挂载

分类信息处于表格第4列,用分号分割,因此列表创建思路为:

  • 逐行读取第4列信息,按分号分割,添加到总列表;
  • 总列表去重复;
  • 总列表去空值;
  • 总列表排序(中文排序)
  • 总列表添加表头;
  • 总列表挂载至模型并在视图显示。

下面是具体代码:

   // 解析分类列并形成列表
    QStringList items;
    for(i = 0; i<model->rowCount(); i++)
    {
        QString item = model->data(model->index(i,3)).toString();
        QStringList cell = item.split(";");
        for(int j = 0; j<cell.count(); j++)
            items<<cell.at(j);

    }

    // 去掉重复
    QStringList keys;
    keys = items.toSet().toList();

    // 去掉空值
    keys.removeAll(QString(""));

    // 中文排序
    keys = sort(keys);

    // 插入头部
    keys.insert(0,"全部文档");

    // 关联列表模型
    list->setStringList(keys);

这里有个问题需要注意:proxymodel指定列后,和多列筛选是否冲突?目前测试是不冲突。

通过在构造函数中读取数据库,将文件加载到表格中,再对分类进行解析,经过去重复,排序,形成左侧分类栏。点击分类可实现proxymodel的筛选展示。在搜索工具条也可以实现全表搜索。

  • 12
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Intimes

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值