记录下使用QListView遇到的各种问题

QListView可以用来以列表的形式展示数据,在Qt中使用model/View结构来管理数据与视图的关系,model负责数据的存取,数据的交互通过delegate来实现。

添加数据模型

QT提供了一些现成的models用于处理数据项:
QStringListModel 用于存储简单的QString列表。
QStandardItemModel 管理复杂的树型结构数据项,每项都可以包含任意数据。
QDirModel 提供本地文件系统中的文件与目录信息。
QSqlQueryModel, QSqlTableModel,QSqlRelationTableModel用来访问数据库。

使用Qt自带的模型类QStandardItemModel即可。模型中的每个数据项都有一个与之对应的role来存储某一类数据。需要存取自定义数据可以使用UserRole,UserRole+1...

对于自定义数据类型,如果要使用QVariant,就必须使用Q_DECLARE_METATYPE注册。

struct ItemData{QString name;QString tel;
};Q_DECLARE_METATYPE(ItemData)

单一数据存取

//存
Item->setData(itemStatus,Qt::UserRole);  // 单一存取//取
ItemStatus status = (ItemStatus)(index.data(Qt::UserRole).toInt());

结构体数据存取

//存
Item->setData(QVariant::fromValue(itemData),Qt::UserRole+1);//整体存取//取
QVariant variant = index.data(Qt::UserRole+1);
ItemData data = variant.value<ItemData>();

自定义delegate

模型的交互和绘制通过自定义delegate来实现,暂时没用到交互,先说下item的绘制。继承了QStyledItemDelegate后,重写paint函数处理item的样式,以及sizeHint函数返回item的大小:

绘制item

void ItemDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
{if(index.isValid()){painter->save();ItemStatus status = (ItemStatus)(index.data(Qt::UserRole).toInt());QVariant variant = index.data(Qt::UserRole+1);ItemData data = variant.value<ItemData>();QStyleOptionViewItem viewOption(option);//用来在视图中画一个itemQRectF rect;rect.setX(option.rect.x());rect.setY(option.rect.y());rect.setWidth( option.rect.width()-1);rect.setHeight(option.rect.height()-1);//QPainterPath画圆角矩形const qreal radius = 7;QPainterPath path;path.moveTo(rect.topRight() - QPointF(radius, 0));path.lineTo(rect.topLeft() + QPointF(radius, 0));path.quadTo(rect.topLeft(), rect.topLeft() + QPointF(0, radius));path.lineTo(rect.bottomLeft() + QPointF(0, -radius));path.quadTo(rect.bottomLeft(), rect.bottomLeft() + QPointF(radius, 0));path.lineTo(rect.bottomRight() - QPointF(radius, 0));path.quadTo(rect.bottomRight(), rect.bottomRight() + QPointF(0, -radius));path.lineTo(rect.topRight() + QPointF(0, radius));path.quadTo(rect.topRight(), rect.topRight() + QPointF(-radius, -0));if(option.state.testFlag(QStyle::State_Selected)){painter->setPen(QPen(Qt::blue));painter->setBrush(QColor(229, 241, 255));painter->drawPath(path);}else if(option.state.testFlag(QStyle::State_MouseOver)){painter->setPen(QPen(Qt::green));painter->setBrush(Qt::NoBrush);painter->drawPath(path);}else{painter->setPen(QPen(Qt::gray));painter->setBrush(Qt::NoBrush);painter->drawPath(path);}//绘制数据位置QRect NameRect = QRect(rect.left() +10, rect.top()+10, rect.width()-30, 20);QRect circle = QRect(NameRect.right(), rect.top()+10, 10, 10);QRect telRect = QRect(rect.left() +10, rect.bottom()-25, rect.width()-10, 20);switch (status) {case S_RED:painter->setBrush(Qt::red);painter->setPen(QPen(Qt::red));break;case S_BLUE:painter->setBrush(Qt::blue);painter->setPen(QPen(Qt::blue));break;case S_YELLOW:painter->setBrush(Qt::yellow);painter->setPen(QPen(Qt::yellow));break;}painter->drawEllipse(circle);     //画圆圈painter->setPen(QPen(Qt::black));painter->setFont(QFont("Times", 12, QFont::Bold));painter->drawText(NameRect,Qt::AlignLeft,data.name); //绘制名字painter->setPen(QPen(Qt::gray));painter->setFont(QFont("Times", 10));painter->drawText(telRect,Qt::AlignLeft,data.tel); //绘制电话painter->restore();}
}QSize ItemDelegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const
{return QSize(160, 60);
}

设置item不同状态下的样式

在paint函数中,还可以获得当前item的状态,并设置不同的样式:

 if(option.state.testFlag(QStyle::State_Selected)) //选中状态{painter->setPen(QPen(Qt::blue));painter->setBrush(QColor(229, 241, 255));painter->drawPath(path);}else if(option.state.testFlag(QStyle::State_MouseOver))//鼠标划过状态{painter->setPen(QPen(Qt::green));painter->setBrush(Qt::NoBrush);painter->drawPath(path);}else{painter->setPen(QPen(Qt::gray));painter->setBrush(Qt::NoBrush);painter->drawPath(path);}

设置好模型后,再对QListView进行下属性设置:

    ui->listView->setItemDelegate(m_delegate);       //为视图设置委托ui->listView->setSpacing(15);                   //为视图设置控件间距ui->listView->setModel(m_model);                  //为委托设置模型ui->listView->setViewMode(QListView::IconMode); //设置Item图标显示ui->listView->setDragEnabled(false);

模型的数据和展示都处理好后,运行效果如下:

listview展示

过滤item

Qt中提供了一个方便处理模型排序和过滤的类QSortFilterProxyModel,通过他可以非常方便的处理我们的model。将QListView展示的model设置成代理模型:

    ui->listView->setItemDelegate(m_delegate);       //为视图设置委托ui->listView->setSpacing(15);                   //为视图设置控件间距m_proxyModel = new QSortFilterProxyModel(ui->listView);m_proxyModel->setSourceModel(m_model);m_proxyModel->setFilterRole(Qt::UserRole);m_proxyModel->setDynamicSortFilter(true);ui->listView->setModel(m_proxyModel);                  //为委托设置模型ui->listView->setViewMode(QListView::IconMode); //设置Item图标显示ui->listView->setDragEnabled(false);            //控件不允许拖动

其中,m_proxyModel->setFilterRole(Qt::UserRole);设置根据模型的某一项数据来处理模型的过滤。proxyModel可以设置过滤的方式,根据QString或者正则表达式来过滤:

m_proxyModel->setFilterFixedString(QString::number(S_RED));//根据字符串过滤
m_proxyModel->setFilterRegExp(QRegExp("^[0|2]$")); //根据正则表达式过滤

过滤模型

获取选中item

对于列表中item的操作,可以是在delegate中处理交互事件,也可以通过QListView获取到所有选中item的QModelIndex,然后对模型本身进行修改。这里我选择的后者:

QModelIndexList modelIndexList = ui->listView->selectionModel()->selectedIndexes();

设置多选

将QListView的selectionBehavior设置成MultiSelection即可。

对于多选的时候,模型的修改有一个坑。在设置了代理模型后,由于开启了动态排序模式,如果修改代理模型的数据,在第一个item修改数据后可能就不在当前过滤模型中,会被过滤掉,后面的item的QModelIndex就会变化,导致后续的修改失败。

dynamicSortFilter : bool
This property holds whether the proxy model is dynamically sorted and filtered whenever the contents of the source model change.
Note that you should not update the source model through the proxy model when dynamicSortFilter is true. For instance, if you set the proxy model on a QComboBox, then using functions that update the model, e.g., addItem(), will not work as expected. An alternative is to set dynamicSortFilter to false and call sort() after adding items to the QComboBox.
The default value is true.

有两个方法处理这个坑,一是不修改代理模型,修改源模型的数据。二是在修改模型数据的时候关闭代理模型的动态排序功能。

修改数据

QModelIndexList sourceIndexList;foreach (QModelIndex modelIndex, modelIndexList){sourceIndexList<<m_proxyModel->mapToSource(modelIndex); //获取源model的modelIndex}//    g_proxyModel->setDynamicSortFilter(false);foreach (QModelIndex sourceIndex, sourceIndexList){ItemStatus status = (ItemStatus)(sourceIndex.data(Qt::UserRole).toInt());qDebug() << "Index : " << sourceIndex.row();switch (status) {case S_RED:redNum--;break;case S_BLUE:blueNum--;break;case S_YELLOW:yellowNum--;break;}status = S_RED;redNum++;m_model->setData(sourceIndex,status,Qt::UserRole);}
//    g_proxyModel->setDynamicSortFilter(true);

最终效果

Demo在这里:->Github链接地址。

Qt之QListView使用相关推荐

  1. qt中QListView的用法和QModelIndex的使用

    使用QTreeView,对于很多函数中针对item的唯一标识QModelIndex的使用,记录下两种对于QModelIdex的使用 1,树形结构的item设置为选中 QModelIndex rootI ...

  2. Qt实现QListView自定义Item界面——仿QQ好友界面

    一直都认为,用最通俗的语言,讲解最深刻的技术,是每一个技术交流者应该考虑的事情,今天朋友问我,好友列表该怎么实现.我想起之前上网查阅的时候,发现网上介绍这块的内容甚少,而且讲解的不够好,于是,本着互相 ...

  3. 【Qt】QListView

    例子 QStringListModel *model = new QStringListModel(this); ui->listView->setModel(model); model- ...

  4. git 使用writer_GitHub - Vpredictor/WriterFly: [QT/C++] 写作天下,为作家创造世界而生,执云作笔,诉尽平生意。...

    写作天下 简介 为作家们创造世界而诞生,执云作笔,诉尽平生意. 集简约UI与人性化AI于一体的码字工具,无论是小说.作文.日记.报告,都能轻松驾驭. QQ交流群:705849222 特点 已有功能: ...

  5. C++Qt开发——Mode View(模型视图)

    Model/View(模型/视图)结构 简介 Model/View(模型/视图)结构是 Qt 中用界面组件显示与编辑数据的一种结构,视图(View)是显示和编辑数据的界面组件,模型(Model)是视图 ...

  6. 【QT的音乐播放器(简单版)】

    基于QT的音乐播放器 前言 一.主体效果 二.主要技术点: 1. mp3的ID3V2格式文件解析:作者.歌手.时长.专辑图片等 1.1 需要工具: 1.2 ID3V2文件格式 1.3 mp3ID3V2 ...

  7. 基于Qt的仿酷狗音乐播放器设计(二)

    简述 在上一文"基于Qt的仿酷狗音乐播放器设计(一)"中,博主给出了仿酷狗界面的部分内容,在本文中将继续分析酷狗界面,并作出相应的分析. 下面我们来看一下酷狗界面中的左侧滑动页控制 ...

  8. 开源项目推荐:本人收集的有关Qt的GitHub/Gitee开源项目(★精品收藏★)

    尊重作者,支持原创,如需转载,请附上原地址:开源项目推荐:Qt有关的GitHub/Gitee开源项目(★精品收藏★)_$firecat全宏的代码足迹$-CSDN博客_qt开源项目https://lib ...

  9. Python Qt GUI设计:QTableView、QListView、QListWidet、QTableWidget、QTreeWidget和QTreeWidgetltem表格和树类(提升篇—1)

    目录 1.QTableView类 2.QListView类 3.QListWidet类 4.QTableWidget类 5.QTreeWidget和QTreeWidgetltem类 表格与树解决的问题 ...

最新文章

  1. es6 日期字符串转日期_量化数据预处理-中文日期(含)转英文日期
  2. SQL语句-exec执行
  3. WPF如何获得变量异步回调函数时产生的异步回调
  4. 一文说说这十多年来计算机玩摄影的历史
  5. 进程用户态 上下文切换需要保存哪些_漫话性能:CPU上下文切换
  6. 杭电OJ-1062_Text Reverse
  7. 关于Entity Framework中的Attached报错相关解决方案的总结
  8. 推动运营商安全标准体系建设
  9. 目录代码php_php获取某个目录大小的代码
  10. Fixjs——显示交互基类InteractiveObject
  11. Bailian2686 打印完数【暴力】
  12. python画三维图-Python基于matplotlib实现绘制三维图形功能示例
  13. 一篇牛B的 纹理映射 大全
  14. 计算机如何安装pdf,pdf虚拟打印机是什么?怎么安装到电脑里
  15. SpringBoot2 学习5集成Thymeleaf
  16. 自学unity,该不该阻止?
  17. 2018迅雷校园招聘客户端在线笔试B卷---输入一个有符号整数,输出该整数的反转值。
  18. 苹果linux桌面文件夹,Linux下打造仿Mac系统桌面
  19. android 访问内网ip_android 获取局域网IP与MAC 地址 毫秒级(详解)
  20. 物联网应用,档案管理首当其冲

热门文章

  1. 七天LLVM零基础入门(Linux版本)------总结
  2. cygwin下各盘挂载点
  3. 整理 被appstore 拒绝审核通过的原因
  4. VMware虚拟机里centos7下安装mysql5.6并授权远程连接Navicat
  5. Linux 的 ps 命令 查看系统进程
  6. 【Oracle】RAC集群中的命令
  7. JavaWeb项目启动时,tomcat会启动两次的原因(之一)和解决方案
  8. 【VUE】vue在vue-cli3环境下基于axios解决跨域问题
  9. 解决IE9下JQuery的ajax失效的问题
  10. git clone 解决Permission Denied (publickey)问题