当前位置: 首页 > news >正文

【Qt】【模型视图架构】代理模型

文章目录

  • 代理模型简单介绍
  • QSortFilterProxyModel类简单介绍
    • 排序
    • 过滤
    • 子类化

代理模型简单介绍

代理模型的作用是可以将一个模型中的数据进行排序或者过滤,然后提供给视图进行显示。

如下所示,创建一个源模型、一个代理模型,界面上创建一个列表视图;设置代理模型的源模型,并设置列表视图的模型为代理模型。

    // 源数据模型模型QStringList list;list << "yafei" << "yafeilinux" << "Qt" << "Qt Creator";QStringListModel *listModel = new QStringListModel(list, this);filterModel = new QSortFilterProxyModel;// 为代理模型设置源模型filterModel->setSourceModel(listModel);// 列表视图设置数据模型ui->listView->setModel(filterModel);

进行过滤时可以调用代理模型的setFilterRegExp()、setFilterRegularExpression()等函数。如下:

  filterModel->setFilterRegularExpression(ui->lineEdit->text());

QSortFilterProxyModel类简单介绍

QSortFilterProxy类支持对另一个模型进行排序和过滤。

代理模型通过将源模型的索引映射到对应新的索引,从而转换源模型的结构。对于视图来说,这种方法允许对给定的源模型进行重构,而不需要对底层数据进行任何转换,也不需要在内存中复制数据。

比如要对自定义模型提供的项进行排序和筛选。不进行排序和筛选的情况下,用于设置模型和视图的代码如下:

    QTreeView *treeView = new QTreeView(this);MyItemModel *myModel = new MyItemModel(this);treeView->setModel(myModel);

为MyItemModel添加排序和筛选支持的话,需要创建一个QSortFilterProxyModel对象,并调用setSourceModel()函数设置源模型,在视图上设置代理模型对象,如下:

    QTreeView *treeView = new QTreeView(this);MyItemModel *myModel = new MyItemModel(this);QSortFilterProxyModel *filterModel = new QSortFilterProxyModel;filterModel->setSourceModel(myModel);treeView->setModel(filterModel);

如果仅有以上的代码,此时视图显示既不会排序也不会筛选,上述代码只是提供了排序和筛选功能的支持。还需要调用setSort*setFilter*开头的函数进行排序和筛选。

代理模型做的任何更改都会应用于源模型

QSortFilterProxyModel作为源模型的包装器,如果需要将源QModelIndexes转换为已经排序或过滤的模型的索引,或者反之,可以使用mapToSource()、mapFromSource()、mapSelectionToSource()和mapSelectionFromSource()等函数。

默认情况下,只要源模型发生更改,模型就会动态地对数据进行重新排列和重新筛选。可以通过设置dynamicSortFilter属性更改此行为。

排序

QTreeView和QTableView类具有sortingEnabled属性,该属性控制用户是否可以通过单击视图的水平标题对视图进行排序。

设置sortingEnabled属性为true,单击标题部分会根据该列对项目进行排序,重复单击,用户可以在升序和降序之间交替排列。

视图调用模型上的sort()虚函数重新排序模型中的数据。对于自定义模型类,可以重新实现**sort(),**也可以使用QSortFilterProxyModel来包装自定义的模型,因为QSortFilterProxyModel提供了一个通用的sort()函数,它对项的setRole()进行操作,并识别几种数据类型,包括int、QString和QDateTime。

对于层次模型,排序递归应用于所有子项。

默认情况下,字符串比较区分大小写,可以通过设置sortCaseSensityvity属性更改。

自定义的排序行为通过子类化QSortFilterProxyModel并重新实现用于比较项的lessThan()函数来实现。

排序的另一种方法是禁用对视图的排序,并将某个顺序强加给用户,这是通过显式调用sort()函数来完成的。

当设置参数按列-1排序时,将返回到基础源模型的排列顺序。

过滤

除了排序之外,QSortFilterProxyModel还可以用于隐藏与某个筛选器不匹配的项。

该筛选器使用QRegExp对象指定,并应用于给定列的每个项的filterRole(),默认情况下为Qt::DisplayRole。QRegExp对象可用于匹配正则表达式、通配符模式或固定字符串。

如下:

   filterModel->setFilterRegExp(QRegExp(".png", Qt::CaseInsensitive, QRegExp::FixedString));  // 设置过滤器filterModel->setFilterKeyColumn(1);  // 设置过滤器作用的列

对于层次模型来说,过滤器递归地应用于所有子级。如果父项与筛选器不匹配,则不会显示其任何子项。

比较常见的用法是让用户在QLineEdit中指定过滤器的正则表达式、通配符模式或固定字符串,并将textChanged()信号连接到setFilterRegularExpression()、setFilterWildcard()或setFilterFixedString()来重新应用过滤器。

对于自定义的过滤行为,可以重新实现filterAcceptsRow()和filterAcceptColumn()函数。

如果处理大量的过滤,并且必须重复调用invalidFilter(),根据模型的实现情况,使用reset()可能会更高效。但是需要注意,reset()会将代理模型返回到其原始状态,从而丢失选择信息,并将导致代理模型重新填充。

子类化

由于QSortFilterProxyModel及其子类是从QAbstractItemModel派生的,因此关于将正常模型子类化的许多相同建议也适用于代理模型。

另外需要注意的是,该类中函数的许多默认实现是为了调用相关源模型中的等效函数而编写的。

对于具有更复杂行为的源模型,可能需要覆盖这种简单的代理机制,比如,如果源模型提供了一个自定义的hasChild()实现,那么也应该在代理模型中提供一个。

子类化时可能需要实现一下几个函数:

virtual bool filterAcceptsColumn(int source_column, const QModelIndex &source_parent) const;
virtual bool filterAcceptsRow(int source_row, const QModelIndex &source_parent) const;
virtual bool lessThan(const QModelIndex &source_left, const QModelIndex &source_right) const;
  • filterAcceptsColumn():如果给定的source_column和source_parent指示的列中的项应该包含在模型中,返回true,否则返回false。
  • filterAcceptsRow():如果给定的source_row和source_parent指示的行中的项应该包含在模型中,返回true,否则返回false。
  • lessThan():如果给定索引source_left引用的项的值小于给定索引source_right引用的项值,返回true,否则返回false。

注意1:使用Qt5,通过QRegularExpression类改进了对正则表达式的支持。Qt5以后的版本应考虑弃用QRegExp API,使用QRegularExpression API。

http://www.lryc.cn/news/360321.html

相关文章:

  • Flutter 中的 IconTheme 小部件:全面指南
  • virtualbox虚拟机、centos7安装增强工具
  • Kotlin 泛型
  • Tomcat 面试题(一)
  • 跟踪一个Pytorch Module在训练过程中的内存分配情况
  • LeetCode 2965.找出缺失和重复的数字:小数据?我选择暴力(附优化方法清单:O(1)空间方法×3)
  • 【运维】VMware Workstation 虚拟机内无网络的解决办法(或许可行)
  • 如何使用Dora SDK完成Fragment流式切换和非流式切换
  • 低代码开发平台(Low-code Development Platform)的模块组成部分
  • Java网络编程(上)
  • Spring Kafka 之 @KafkaListener 注解详解
  • 【量算分析工具-贴地距离】GeoServer改造Springboot番外系列九
  • 【linux】(1)文件操作及vi
  • 【5】MySQL数据库备份-XtraBackup - 全量备份
  • 数据治理-数据标准演示
  • 基于Chisel的FPGA流水灯设计
  • LabVIEW齿轮调制故障检测系统
  • AI帮写:探索国内AI写作工具的创新与实用性
  • n后问题 回溯笔记
  • 简述Java中实现Socket通信的步骤
  • Asp.Net Core 实现分片下载的最简单方式
  • [Mac软件]Leech for Mac v3.2 - 轻量级mac下载工具
  • 留给“端侧大模型”的时间不多了
  • Pytest框架中的Setup和Teardown功能
  • yolov10/v8 loss详解
  • Typescript高级: 深入理解infer关键字
  • JQC-3FF-S-Z 继电器模块使用(arduino)
  • 黑马一站制造数仓实战2
  • 网络I/O模型
  • Docker 简介和安装