当前位置: 首页 > 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。

相关文章:

  • 文件上传之使用一个属性接收多个文件
  • Python课设-学生信息管理系统
  • 申请高德地图API的Key并加以接口调用的方法
  • 深入分析 Android Service (完)
  • WordPress plugin MStore API SQL注入漏洞复现(CVE-2023-3077)
  • axios七大特性
  • Golang 协程和 Channel 的协同工作
  • Facebook开户 | Facebook二不限户
  • java —— 匿名内部类与 Lambda 表达式
  • python-最接近target的值
  • R语言绘图 --- 柱状图(Biorplot 开发日志 --- 3)
  • Tree——输出项目的文件结构(Linux)
  • 多项目的.net core解决方案(项目间引用)如何使用Docker部署
  • keepalived安装文档
  • Python知识点11---高阶函数
  • @angular/forms 源码解析之双向绑定
  • 【跃迁之路】【444天】程序员高效学习方法论探索系列(实验阶段201-2018.04.25)...
  • Cookie 在前端中的实践
  • Iterator 和 for...of 循环
  • Java编程基础24——递归练习
  • java中的hashCode
  • MobX
  • QQ浏览器x5内核的兼容性问题
  • react-core-image-upload 一款轻量级图片上传裁剪插件
  • Spring Cloud中负载均衡器概览
  • SpringCloud(第 039 篇)链接Mysql数据库,通过JpaRepository编写数据库访问
  • Web设计流程优化:网页效果图设计新思路
  • 测试如何在敏捷团队中工作?
  • 第十八天-企业应用架构模式-基本模式
  • 前端临床手札——文件上传
  • 使用 @font-face
  • 数据仓库的几种建模方法
  • 听说你叫Java(二)–Servlet请求
  • 我的面试准备过程--容器(更新中)
  • 一个SAP顾问在美国的这些年
  • #宝哥教你#查看jquery绑定的事件函数
  • $NOIp2018$劝退记
  • (板子)A* astar算法,AcWing第k短路+八数码 带注释
  • (第27天)Oracle 数据泵转换分区表
  • (二)WCF的Binding模型
  • (每日持续更新)信息系统项目管理(第四版)(高级项目管理)考试重点整理第3章 信息系统治理(一)
  • (原創) 如何刪除Windows Live Writer留在本機的文章? (Web) (Windows Live Writer)
  • (转) Android中ViewStub组件使用
  • (转)IIS6 ASP 0251超过响应缓冲区限制错误的解决方法
  • (转)nsfocus-绿盟科技笔试题目
  • (转)创业家杂志:UCWEB天使第一步
  • .gitignore文件_Git:.gitignore
  • .NET Framework 和 .NET Core 在默认情况下垃圾回收(GC)机制的不同(局部变量部分)
  • .NET Micro Framework初体验(二)
  • .NET/C# 反射的的性能数据,以及高性能开发建议(反射获取 Attribute 和反射调用方法)
  • .Net7 环境安装配置
  • .NET中的Event与Delegates,从Publisher到Subscriber的衔接!
  • @synthesize和@dynamic分别有什么作用?
  • [ web基础篇 ] Burp Suite 爆破 Basic 认证密码
  • [.net 面向对象程序设计进阶] (19) 异步(Asynchronous) 使用异步创建快速响应和可伸缩性的应用程序...