在上一篇博文《C/C++ Qt TreeWidget 单层树形组件应用》中给大家演示了如何使用TreeWidget组件创建单层树形结构,并给这个树形组件增加了右键菜单功能,接下来将继续延申树形组件的使用,并实现对树形框多节点的各种操作。

常用树形框节点间的操作方法如下:

  • 节点遍历
  • 初始化节点
  • 单击双击节点
  • 添加根节点
  • 添加子节点
  • 修改选中节点
  • 删除选中节点
  • 枚举全部节点
  • 枚举选中节点
  • 获取节点子节点

简单的节点遍历: 首先我们还是使用TreeView组件实现一个简单的多层嵌套树结构,代码运行后,首先循环设置3个外层节点,接着循环内层节点,并将内层中的QStandardItem追加到外层上面。

  1. #include "mainwindow.h"
  2. #include "ui_mainwindow.h"
  3. #include <QTreeView>
  4. #include <QStandardItemModel>
  5. // By: LyShark
  6. // https://www.cnblogs.com/lyshark
  7. MainWindow::MainWindow(QWidget *parent) :QMainWindow(parent),ui(new Ui::MainWindow)
  8. {
  9. ui->setupUi(this);
  10. QStandardItemModel *tree = new QStandardItemModel(0,3,this);
  11. ui->treeView->setColumnWidth(0,50); // 设置第1列长度
  12. ui->treeView->setColumnWidth(1,200); // 设置第2列长度
  13. ui->treeView->setColumnWidth(2,200); // 设置第3列长度
  14. tree->setHeaderData(0, Qt::Horizontal, tr("序号"));
  15. tree->setHeaderData(1, Qt::Horizontal, tr("姓名"));
  16. tree->setHeaderData(2, Qt::Horizontal, tr("年龄"));
  17. ui->treeView->setModel(tree);
  18. for (int i = 0; i < 4; ++i)
  19. {
  20. // 设置3个外层节点
  21. QList<QStandardItem *> items;
  22. for (int i = 0; i < 3; ++i)
  23. {
  24. QStandardItem *item = new QStandardItem(QString("%0").arg(i));
  25. if (0 == i)
  26. item->setCheckable(true);
  27. items.push_back(item);
  28. }
  29. tree->appendRow(items);
  30. // 设置内层
  31. for (int i = 0; i < 2; ++i)
  32. {
  33. QList<QStandardItem *> childItems;
  34. for (int i = 0; i < 3; ++i)
  35. {
  36. QStandardItem *item = new QStandardItem(QString("lyshark"));
  37. if (0 == i)
  38. item->setCheckable(true);
  39. childItems.push_back(item);
  40. }
  41. items.at(0)->appendRow(childItems);
  42. }
  43. }
  44. }
  45. MainWindow::~MainWindow()
  46. {
  47. delete ui;
  48. }

代码运行效果如下:

初始化树形节点: 首先在开始操作元素之前,我们可以在MainWindow::MainWindow中对树形节点进行简单的初始化,插入几个测试节点.

  1. #include "mainwindow.h"
  2. #include "ui_mainwindow.h"
  3. #include <iostream>
  4. #include <QTreeWidgetItem>
  5. #include <QString>
  6. // By: LyShark
  7. // https://www.cnblogs.com/lyshark
  8. MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
  9. {
  10. ui->setupUi(this);
  11. ui->treeWidget->clear();
  12. // 设置QTreeWidget的列数
  13. ui->treeWidget->setColumnCount(1);
  14. // 设置QTreeWidget标题隐藏
  15. ui->treeWidget->setHeaderHidden(true);
  16. // 创建QTreeWidget的朋友节点,父节点是tree
  17. QTreeWidgetItem *Friend = new QTreeWidgetItem(ui->treeWidget,QStringList(QString("朋友")));
  18. Friend->setIcon(0,QIcon(":/image/4.ico")); // 添加一个图标
  19. Friend->setFlags(Qt::ItemIsSelectable | Qt::ItemIsUserCheckable
  20. | Qt::ItemIsEnabled | Qt::ItemIsAutoTristate);
  21. Friend->setCheckState(0,Qt::Checked);
  22. // 给Friend添加一个子节点frd
  23. QTreeWidgetItem *frd = new QTreeWidgetItem(Friend);
  24. frd->setText(0,"www.lyshark.com");
  25. frd->setIcon(0,QIcon(tr(":/image/1.ico")));
  26. frd->setCheckState(0,Qt::Checked); // 默认选中状态
  27. QTreeWidgetItem *frs = new QTreeWidgetItem(Friend);
  28. frs->setText(0,"cdn.lyshark.com");
  29. frs->setIcon(0,QIcon(tr(":/image/1.ico")));
  30. frs->setCheckState(0,Qt::Unchecked); // 默认未选中
  31. // ----------------------------------------------------------
  32. // 创建名叫同学节点,父节点同样是tree
  33. QTreeWidgetItem * ClassMate = new QTreeWidgetItem(ui->treeWidget,QStringList(QString("同学")));
  34. ClassMate->setIcon(0,QIcon(":/image/5.ico")); // 添加一个图标
  35. ClassMate->setCheckState(0,Qt::Checked); // 默认选中
  36. //Fly是ClassMate的子节点
  37. QTreeWidgetItem *Fly = new QTreeWidgetItem(QStringList(QString("nas.lyshark.com")));
  38. Fly->setIcon(0,QIcon(tr(":/image/2.ico")));
  39. //创建子节点的另一种方法
  40. ClassMate->addChild(Fly);
  41. Fly->setCheckState(0,Qt::Checked); // 设置为选中
  42. QTreeWidgetItem *Fls = new QTreeWidgetItem(QStringList(QString("lyshark.cnblogs.com")));
  43. Fls->setIcon(0,QIcon(tr(":/image/2.ico")));
  44. ClassMate->addChild(Fls);
  45. Fls->setCheckState(0,Qt::Checked); // 设置为选中
  46. // ----------------------------------------------------------
  47. // 陌生人单独一栏
  48. QTreeWidgetItem *Strange = new QTreeWidgetItem(true);
  49. Strange->setText(0,"陌生人");
  50. Strange->setIcon(0,QIcon(":/image/6.ico")); // 添加一个图标
  51. ui->treeWidget->addTopLevelItem(ClassMate);
  52. ui->treeWidget->addTopLevelItem(Strange);
  53. // 增加文本到编辑框
  54. ui->plainTextEdit->appendPlainText("hello lyshark");
  55. //展开QTreeWidget的所有节点
  56. //ui->treeWidget->expandAll();
  57. //ui->treeWidget->resize(271,401);
  58. }
  59. MainWindow::~MainWindow()
  60. {
  61. delete ui;
  62. }

代码运行效果如下:

单击双击节点反馈: 当我们将鼠标停靠在指定节点内并点击时,我们需要触发treeWidget_itemDoubleClicked属性让其反馈该行标题等基本属性.

  1. // 当我们双击指定的成员时获取到该成员的名字
  2. void MainWindow::on_treeWidget_itemDoubleClicked(QTreeWidgetItem *item, int column)
  3. {
  4. QString str = item->text(column);
  5. std::cout << str.toStdString().data() << std::endl;
  6. ui->plainTextEdit->appendPlainText(str.toStdString().data());
  7. }
  8. // 当我们单击指定成员时获取数据
  9. void MainWindow::on_treeWidget_itemClicked(QTreeWidgetItem *item, int column)
  10. {
  11. QString str = item->text(column);
  12. std::cout << str.toStdString().data() << std::endl;
  13. ui->plainTextEdit->appendPlainText(str.toStdString().data());
  14. }

代码运行效果如下:

添加 父节点/子节点: 通过代码的方式当点击on_pushButton_clicked时分别实现增加一个父节点和一个子节点的功能。

  1. // 单击按钮添加新的父节点
  2. void MainWindow::on_pushButton_clicked()
  3. {
  4. QString NodeText = "新的父节点";
  5. QTreeWidgetItem *item = new QTreeWidgetItem(true);
  6. item->setText(0,NodeText);
  7. item->setIcon(0,QIcon(":/image/7.ico"));
  8. ui->treeWidget->addTopLevelItem(item);
  9. }
  10. // 单击按钮添加子节点
  11. void MainWindow::on_pushButton_4_clicked()
  12. {
  13. QTreeWidgetItem * item= ui->treeWidget->currentItem();
  14. if(item!=NULL)
  15. AddTreeNode(item,"新子节点","新子节点");
  16. else
  17. AddTreeRoot("新子节点","新子节点");
  18. }

代码运行效果如下:

删除选中节点: 首先选中要删除的指定节点,然后可以对该节点进行删除操作,删除子节点直接移除即可,删除父节点需要连同内部子节点一并删掉。

  1. // 删除选中的节点
  2. void MainWindow::on_pushButton_3_clicked()
  3. {
  4. QTreeWidgetItem *currentItem = ui->treeWidget->currentItem();
  5. if(currentItem == NULL)
  6. return;
  7. // 如果没有父节点则直接删除
  8. if(currentItem->parent() == NULL)
  9. {
  10. delete ui->treeWidget->takeTopLevelItem(ui->treeWidget->currentIndex().row());
  11. std::cout << ui->treeWidget->currentIndex().row() << std::endl;
  12. }
  13. else
  14. {
  15. // 如果有父节点就要用父节点的takeChild删除节点
  16. delete currentItem->parent()->takeChild(ui->treeWidget->currentIndex().row());
  17. }
  18. }

代码运行效果如下:

修改指定节点名称: 单击后将指定节点修改为Modify并将图标设置为新的

  1. // 修改节点
  2. // By: LyShark
  3. // https://www.cnblogs.com/lyshark
  4. void MainWindow::on_pushButton_2_clicked()
  5. {
  6. // 得到当前节点
  7. QTreeWidgetItem *currentItem = ui->treeWidget->currentItem();
  8. if(currentItem == NULL)
  9. return;
  10. // 修改选中项
  11. for(int x=0;x<currentItem->columnCount();x++)
  12. {
  13. currentItem->setText(x,tr("Modify") + QString::number(x));
  14. currentItem->setIcon(x,QIcon(":/image/1.ico"));
  15. }
  16. }

代码运行效果如下:

枚举所有节点元素: 枚举当前Tree中的所有节点元素,并将结果输出到右侧编辑框内。

  1. // 枚举所有节点
  2. // By: LyShark
  3. // https://www.cnblogs.com/lyshark
  4. // 枚举所有节点
  5. void MainWindow::on_pushButton_5_clicked()
  6. {
  7. // 获取到全部的根节点数量
  8. int size = ui->treeWidget->topLevelItemCount();
  9. QTreeWidgetItem *child;
  10. for(int x=0;x<size;x++)
  11. {
  12. // 输出所有父节点
  13. child = ui->treeWidget->topLevelItem(x);
  14. std::cout << "all root = "<< child->text(0).toStdString().data() << std::endl;
  15. ui->plainTextEdit->appendPlainText(child->text(0).toStdString().data());
  16. // 得到所有子节点计数
  17. int childCount = child->childCount();
  18. // std::cout << "all child count = " << childCount << std::endl;
  19. // 输出根节点下面的子节点
  20. for(int y=0;y<childCount;++y)
  21. {
  22. QTreeWidgetItem *grandson = child->child(y);
  23. std::cout << "--> sub child = "<< grandson->text(0).toStdString().data() << std::endl;
  24. ui->plainTextEdit->appendPlainText(grandson->text(0).toStdString().data());
  25. }
  26. }
  27. }

代码运行效果如下:

枚举选中节点元素: 枚举当前Tree中选中节点的元素,并将结果输出到右侧编辑框内。

  1. // 枚举所有的 【选中】节点
  2. // https://www.cnblogs.com/lyshark
  3. void MainWindow::on_pushButton_7_clicked()
  4. {
  5. // 获取到全部的根节点数量
  6. int size = ui->treeWidget->topLevelItemCount();
  7. QTreeWidgetItem *child;
  8. for(int x=0;x<size;x++)
  9. {
  10. // 输出所有父节点
  11. child = ui->treeWidget->topLevelItem(x);
  12. // 得到所有子节点计数
  13. int childCount = child->childCount();
  14. // 输出根节点下面的子节点
  15. for(int y=0;y<childCount;++y)
  16. {
  17. QTreeWidgetItem *grandson = child->child(y);
  18. // 判断是否选中,如果选中输出父节点与子节点
  19. if(Qt::Checked == grandson->checkState(0))
  20. {
  21. std::cout << "root -> " << child->text(0).toStdString().data()
  22. << "--> sub child = "<< grandson->text(0).toStdString().data() << std::endl;
  23. ui->plainTextEdit->appendPlainText(grandson->text(0).toStdString().data());
  24. }
  25. }
  26. }
  27. }

代码运行效果如下:

获取选中子节点的父节点: 获取子节点的父节点ID,然后根据ID得到子节点名字。

  1. void MainWindow::on_pushButton_6_clicked()
  2. {
  3. // 取所有的父节点
  4. QTreeWidgetItem *currentItem = ui->treeWidget->currentItem()->parent();
  5. int root_count = ui->treeWidget->indexOfTopLevelItem(currentItem);
  6. std::cout << "root Count = " << root_count << std::endl;
  7. if(root_count != -1)
  8. {
  9. // 指定序号对应的父节点名字
  10. QTreeWidgetItem *child;
  11. child = ui->treeWidget->topLevelItem(root_count);
  12. std::cout << "root name= "<< child->text(0).toStdString().data() << std::endl;
  13. ui->plainTextEdit->appendPlainText(child->text(0).toStdString().data());
  14. }
  15. }

代码运行效果如下:

C/C++ Qt TreeWidget 嵌套节点操作技巧的更多相关文章

  1. 分享基于.NET动态编译&Newtonsoft.Json封装实现JSON转换器(JsonConverter)原理及JSON操作技巧

    看文章标题就知道,本文的主题就是关于JSON,JSON转换器(JsonConverter)具有将C#定义的类源代码直接转换成对应的JSON字符串,以及将JSON字符串转换成对应的C#定义的类源代码,而 ...

  2. 深度解析JQuery Dom元素操作技巧

    深度解析JQuery Dom元素操作技巧 DOM是一种与浏览器.平台.语言无关的接口,使用该接口可以轻松访问页面中所有的标准组件,这篇文章给大家介绍了JQuery dom元素操作方法,写的十分的全面细 ...

  3. 用ArcGIS?37个Arcmap常用操作技巧可能帮到您

    1. 要素的剪切与延伸 实用工具 TASK 任务栏 Extend/Trim feature 剪切所得内容与你画线的方向有关. 2. 自动捕捉跟踪工具 点击Editor工具栏中Snapping来打开Sn ...

  4. 深入理解DOM节点操作

    × 目录 [1]创建节点 [2]插入节点 [3]移除节点[4]替换节点[5]复制节点 前面的话 一般地,提起操作会想到“增删改查”这四个字,而DOM节点操作也类似地对应于此,接下来将详细介绍DOM的节 ...

  5. 用Excel做出比肩任务管理软件的操作技巧

    用Excel做出比肩任务管理软件的操作技巧 在项目管理中,网上有各种各样的工具可以选择,到底用哪个,曾一度困扰着我.我是一个有轻度强迫症的人,总是喜欢试用各种各样的系统,以比较他们之间的不同,试图选择 ...

  6. 深入学习jQuery节点操作

    × 目录 [1]创建节点 [2]插入节点 [3]删除节点[4]复制节点[5]替换节点[6]包裹节点 前面的话 DOM节点操作包括创建节点.插入节点.移除节点.替换节点和复制节点.jQuery也有类似的 ...

  7. 第32课 Qt中的文件操作

    1. Qt的中IO操作 (1)Qt中IO操作的处理方式 ①Qt通过统一的接口简化了文件和外部设备的操作方式 ②Qt中的文件被看作一种特殊的外部设备 ③Qt中的文件操作与外部设备的操作相同 (2)IO操 ...

  8. Query节点操作,jQuery插入节点,jQuery删除节点,jQuery Dom操作

    一.创建节点 var box = $('<div>节点</div>'); //创建一个节点,或者var box = "<div>节点</div> ...

  9. 第6章 DOM节点操作

    一.创建节点 为了使页面更加智能化,有时我们想动态的在 html 结构页面添加一个元素标签,那么 在插入之前首先要做的动作就是:创建节点. varbox=$('<div id="box ...

随机推荐

  1. FastAPI 学习之路(三)

    系列文章: FastAPI 学习之路(一)fastapi--高性能web开发框架 FastAPI 学习之路(二) 之前的文章分享了如何去创建一个简单的路径的请求.那么我们这次分享的如何在请求路径中,增 ...

  2. Python技法3:匿名函数、回调函数和高阶函数

    1.定义匿名或内联函数 如果我们想提供一个短小的回调函数供sort()这样的函数用,但不想用def这样的语句编写一个单行的函数,我们可以借助lambda表达式来编写"内联"式的函数 ...

  3. DOS命令和快捷键

  4. 2021.9.20考试总结[NOIP模拟57]

    (换个编辑器代码就SB地不自动折叠了.. T1 2A 考察快读的写法. $code:$ T1 #include<bits/stdc++.h> #define scanf SCANF=sca ...

  5. AtCoder Beginner Contest 220部分题(G,H)题解

    刚开始的时候被E题卡住了,不过发现是个数学题后就开始使劲推式子,幸运的是推出来了,之后的F题更是树形DP换根的模板吧,就草草的过了,看了一眼G,随便口胡了一下,赶紧打代码,毕竟时间不多了,最后也没打完 ...

  6. hdu 2201 熊猫阿波的故事(简单概率。。)

    题意: 阿波上了飞机,飞机上有座位1,2,....,N.第i个乘客本应坐在第i个座位上. 可是阿波随便找了个座位就坐了下来,接下来大家也都随便找了个座位坐了下来. 问:第i个乘客坐到原座位的概率是多少 ...

  7. js this指向汇总

    this指向 普通函数  window 定时器函数         window 事件函数 事件源 箭头函数 父function中的this,没有就是window 对象函数 对象本身 构造函数 实例化 ...

  8. (一)lamp 环境搭建之编译安装apache

    apache的编译安装: 安装步骤大概参考:http://www.cnblogs.com/iyoule/archive/2013/10/24/3385540.html 简单的将分为三步: (1)安装a ...

  9. element-UI 中的upload组件如何添加token?

    <el-upload :show-file-list="false" :on-error="errmsg" :headers="headers& ...

  10. NOIP模拟92(多校25)

    前言 所以说这次是 HZOI 多校联测巅峰????(题目,数据过水??) T1 石子合并 解题思路 签到题. 发现我们可以给每个数字附一个正负号,每个数字的贡献就是它本身乘上这个符号. 发现至少应该有 ...