Qt MVC结构之QItemSelectionModel模型介绍

QItemSelectionModel

Qt的MVC结构支持多个View共享同一个model,包括该model的选中状态等。我们可以通过设置QItemSelectionModel,来更改View的选中效果和显示效果。我们创建一个Qt Application项目,在MainWindow的头文件中添加一个QTbaleView*类型的成员 _table_view。然后在构造函数中为这个_table_view设置model

  1. MainWindow::MainWindow(QWidget *parent) :
  2. QMainWindow(parent),
  3. ui(new Ui::MainWindow)
  4. {
  5. ui->setupUi(this);
  6. QStandardItemModel * model = new QStandardItemModel(7,4,this);
  7. for(int row=0; row < 7; row++){
  8. for(int column = 0; column < 4; column++){
  9. QStandardItem * item = new QStandardItem(QString("%1").arg(row*4+column));
  10. model->setItem(row, column, item);
  11. }
  12. }
  13. _table_view = new QTableView;
  14. _table_view->setModel(model);
  15. setCentralWidget(_table_view);
  16. this->resize(800,800);
  17. }

我们创建了一个QStandardItemModel对象,然后为其设置几个Item,最后将这个model设置到tableview中。
我们可以为TableView设置选择的项目,接下来继续在构造函数中补充

  1. //获取视图的项目选择模型
  2. QItemSelectionModel * selection_model = _table_view->selectionModel();
  3. //定义左上角和右下角的索引
  4. QModelIndex topLeft;
  5. QModelIndex bottomRight;
  6. //根据上面两个索引选择项目
  7. //第1行1列
  8. topLeft = model->index(1,1,QModelIndex());
  9. //第5行2列
  10. bottomRight = model->index(5,2,QModelIndex());
  11. //设置选择区域
  12. QItemSelection selection(topLeft, bottomRight);
  13. //将选择的区域设置给选择模型, 设置 为选中状态
  14. selection_model->select(selection, QItemSelectionModel::Select);

我们从tableview中获取选择模型,然后从根节点下选择第1行1列作为左上索引,选择第5行2列作为右下索引。然后创建选择区域QItemSelection,最后将选择区域这只给选择模型。接下来为了我们为工具栏添加两个动作,为动作绑定两个槽函数,在MainWindow声明文件中添加两个函数的声明

  1. public slots:
  2. void getCurrentItemData();
  3. void toggleSelection();

在构造函数中将动作和槽函数绑定起来

  1. ui->mainToolBar->addAction(tr("当前项目"), this, &MainWindow::getCurrentItemData);
  2. ui->mainToolBar->addAction(tr("切换选择"), this, &MainWindow::toggleSelection);

实现这两个槽函数, getCurrentItemData获取当前条目的数据信息,toggleSelection实现切换选择的条目

  1. void MainWindow::getCurrentItemData(){
  2. auto currentData = _table_view->selectionModel()->currentIndex().data().toString();
  3. qDebug() <<tr("当前项目的内容") << currentData;
  4. }
  5. void MainWindow::toggleSelection(){
  6. //找到根节点下第0行0列的item的索引
  7. QModelIndex topLeft = _table_view->model()->index(0,0,QModelIndex());
  8. //获取根节点下最大的行号
  9. auto max_row = _table_view->model()->rowCount(QModelIndex());
  10. //获取根节点下最大的列号
  11. auto max_column = _table_view->model()->columnCount(QModelIndex());
  12. //根据列号和行号获取最右下角的item的索引
  13. QModelIndex bottomRight = _table_view->model()->index(max_row-1, max_column-1, QModelIndex());
  14. //设置选择区域
  15. QItemSelection curSelection(topLeft, bottomRight);
  16. _table_view->selectionModel()->select(curSelection, QItemSelectionModel::Toggle);
  17. }

通过点击切换选择,可以实现选择区域的切换,因为我们设置选择的类型为Toggle,系统会将选中的变为取消选中,将取消选中的变为选中。
切换前
https://cdn.llfc.club/1671529103099.jpg
切换后
https://cdn.llfc.club/1671529287147.jpg
因为默认当前的条目是第1行1列,所以getCurrentItemData会打印他的信息。每一个model都有当前条目和选择条目两个属性。

捕获选择条目的变化

当我们点击鼠标选择一个条目时,当前条目就变为该条目,选择的条目也变为该条目。我们可以通过选择模型发出的selectionChanged信号,获取选择了那些条目的索引,以及取消选择了哪些条目的索引。我们可以通过currentChanged获取当前的条目索引以及变化之前的条目索引。现在MainWindow中添加两个槽函数的声明

  1. public slots:
  2. void updateSelection(const QItemSelection& selected, const QItemSelection& deselected);
  3. void changeCurrent(const QModelIndex& current, const QModelIndex& previous);

两个槽函数的参数和信号的参数匹配。我们实现changeCurrent,当选择模型的当前索引变化时,打印变化前后的索引行号,列号。

  1. void MainWindow::changeCurrent(const QModelIndex& current, const QModelIndex& previous){
  2. qDebug() << tr("move(%1, %2) to (%3, %4)").arg(previous.row())
  3. .arg(previous.column()).arg(current.row()).arg(current.column());
  4. }

然后我们实现选择模型的选择条目变化时更新条目的数据

  1. void MainWindow::updateSelection(const QItemSelection& selected, const QItemSelection& deselected){
  2. QModelIndex index;
  3. QModelIndexList list = selected.indexes();
  4. //为现在的选择项目设置值
  5. for(int i =0; i < list.size(); i++){
  6. QString text = QString("(%1, %2)").arg(list[i].row()).arg(list[i].column());
  7. _table_view->model()->setData(list[i], text);
  8. }
  9. list = deselected.indexes();
  10. foreach(index, list){
  11. _table_view->model()->setData(index,"");
  12. }
  13. }

在MainWindow的构造函数中添加信号和槽函数的链接

  1. //选择模型的选择条目更改后触发updateSelection函数
  2. connect(selection_model, &QItemSelectionModel::selectionChanged, this, &MainWindow::updateSelection);
  3. //选择模型的当前项目更改后,关联changeCurrent函数
  4. connect(selection_model, &QItemSelectionModel::currentChanged, this, &MainWindow::changeCurrent);

点击某个条目,其他条目的数据都变为空,这个条目被选中,内容修改为行列号
https://cdn.llfc.club/1671530820828.jpg

多个View共享模型

MVC结构最大的好处就是可以通过多个View共享同一个model,这个model的选中状态改变时,多个View的显示效果一致。
在构造函数中添加另一个QTableView

  1. QTableView * tableView2;
  2. tableView2 = new QTableView();
  3. tableView2->setWindowTitle("tableView2");
  4. tableView2->resize(400,300);
  5. tableView2->setModel(model);
  6. tableView2->setSelectionModel(selection_model);
  7. tableView2->show();

可以看到两个View显示的选中状态是一致的
https://cdn.llfc.club/1671531054879.jpg

源码链接

源码链接
https://gitee.com/secondtonone1/qt-learning-notes

热门评论

热门文章

  1. vscode搭建windows C++开发环境

    喜欢(596) 浏览(83438)
  2. 聊天项目(28) 分布式服务通知好友申请

    喜欢(507) 浏览(6096)
  3. 使用hexo搭建个人博客

    喜欢(533) 浏览(11903)
  4. Linux环境搭建和编码

    喜欢(594) 浏览(13547)
  5. Qt环境搭建

    喜欢(517) 浏览(24707)

最新评论

  1. 类和对象 陈宇航:支持!!!!
  2. 聊天项目(7) visualstudio配置grpc diablorrr:cmake文件得改一下 find_package(Boost REQUIRED COMPONENTS system filesystem),要加上filesystem。在target_link_libraries中也同样加上
  3. boost::asio之socket的创建和连接 项空月:发现一些错别字 :每隔vector存储  是不是是每个. asio::mutable_buffers_1 o或者    是不是多打了个o
  4. Qt MVC结构之QItemDelegate介绍 胡歌-此生不换:gpt, google
  5. interface应用 secondtonone1:interface是万能类型,但是使用时要转换为实际类型来使用。interface丰富了go的多态特性,也降低了传统面向对象语言的耦合性。
  6. 无锁并发队列 TenThousandOne:_head  和 _tail  替换为原子变量。那里pop的逻辑,val = _data[h] 可以移到循环外面吗
  7. string类 WangQi888888:确实错了,应该是!isspace(sind[index]). 否则不进入循环,还是原来的字符串“some string”
  8. 堆排序 secondtonone1:堆排序非常实用,定时器就是这个原理制作的。
  9. 答疑汇总(thread,async源码分析) Yagus:如果引用计数为0,则会执行 future 的析构进而等待任务执行完成,那么看到的输出将是 这边应该不对吧,std::future析构只在这三种情况都满足的时候才回block: 1.共享状态是std::async 创造的(类型是_Task_async_state) 2.共享状态没有ready 3.这个future是共享状态的最后一个引用 这边共享状态类型是“_Package_state”,引用计数即使为0也不应该block啊
  10. visual studio配置boost库 一giao里我离giaogiao:请问是修改成这样吗:.\b2.exe toolset=MinGW
  11. 处理网络粘包问题 zyouth: //消息的长度小于头部规定的长度,说明数据未收全,则先将部分消息放到接收节点里 if (bytes_transferred < data_len) { memcpy(_recv_msg_node->_data + _recv_msg_node->_cur_len, _data + copy_len, bytes_transferred); _recv_msg_node->_cur_len += bytes_transferred; ::memset(_data, 0, MAX_LENGTH); _socket.async_read_some(boost::asio::buffer(_data, MAX_LENGTH), std::bind(&CSession::HandleRead, this, std::placeholders::_1, std::placeholders::_2, shared_self)); //头部处理完成 _b_head_parse = true; return; } 把_b_head_parse = true;放在_socket.async_read_some前面是不是更好
  12. 构造函数 secondtonone1:构造函数是类的基础知识,要着重掌握
  13. 聊天项目(9) redis服务搭建 pro_lin:redis线程池的析构函数,除了pop出队列,还要free掉redis连接把
  14. protobuf配置和使用 熊二:你可以把dll放到系统目录,也可以配置环境变量,还能把dll丢到lib里
  15. C++ 并发三剑客future, promise和async Yunfei:大佬您好,如果这个线程池中加入的异步任务的形参如果有右值引用,这个commit中的返回类型推导和bind绑定就会出现问题,请问实际工程中,是不是不会用到这种任务,如果用到了,应该怎么解决?
  16. 聊天项目(15) 客户端实现TCP管理者 lkx:已经在&QTcpSocket::readyRead 回调函数中做了处理了的。
  17. 网络编程学习方法和图书推荐 Corleone:啥程度可以找工作
  18. 创建项目和编译 secondtonone1:谢谢支持
  19. Qt 对话框 Spade2077:QDialog w(); //这里是不是不需要带括号
  20. 再谈单例模式 secondtonone1:是的,C++11以后返回局部static变量对象能保证线程安全了。
  21. 利用栅栏实现同步 Dzher:作者你好!我觉得 std::thread a(write_x); std::thread b(write_y); std::thread c(read_x_then_y); std::thread d(read_y_then_x); 这个例子中的assert fail并不会发生,原子变量设定了非relaxed内存序后一个线程的原子变量被写入,那么之后的读取一定会被同步的,c和d线程中只可能同时发生一个z++未执行的情况,最终z不是1就是2了,我测试了很多次都没有assert,请问我这个观点有什么错误,谢谢!
  22. 面试题汇总(一) secondtonone1:看到网络上经常提问的go的问题,做了一下汇总,结合自己的经验给出的答案,如有纰漏,望指正批评。
  23. 解决博客回复区被脚本注入的问题 secondtonone1:走到现在我忽然明白一个道理,无论工作也好生活也罢,最重要的是开心,即使一份安稳的工作不能给我带来事业上的积累也要合理的舍弃,所以我还是想去做喜欢的方向。

个人公众号

个人微信