相关文章推荐
完美的馒头  ·  python QTreeWidget ...·  2 周前    · 
飘逸的饭卡  ·  进程监控工具supervisor ...·  3 月前    · 
打篮球的汽水  ·  Oracle ...·  7 月前    · 
宽容的刺猬  ·  Flink SQL No ...·  1 年前    · 

qt中用treewidget创建横向表格,实现两个treewidget能相互拖动item,并且只能拖动到到根节点的子节点同级节点

时间: 2024-02-01 10:13:26 浏览: 75
实现这个功能需要以下几个步骤: 1. 创建两个QTreeWidget,一个作为源,一个作为目标。 2. 设置源QTreeWidget为可拖动,目标QTreeWidget为可接受拖放。 ```cpp sourceWidget->setDragEnabled(true); targetWidget->setAcceptDrops(true); 3. 在源QTreeWidget的mousePressEvent事件中,记录被拖动的item和它的父节点。 ```cpp if (event->button() == Qt::LeftButton) { QTreeWidgetItem *item = sourceWidget->itemAt(event->pos()); if (item && item->parent()) { m_draggedItem = item; m_draggedParent = item->parent(); 4. 在源QTreeWidget的mouseMoveEvent事件中,如果鼠标移动的距离大于一定阈值,就开始拖拽操作。 ```cpp if (m_draggedItem && (event->buttons() & Qt::LeftButton)) { int distance = (event->pos() - m_pressPos).manhattanLength(); if (distance >= QApplication::startDragDistance()) { QDrag *drag = new QDrag(this); QMimeData *mimeData = new QMimeData; mimeData->setData("application/x-qabstractitemmodeldatalist", sourceWidget->model()->itemData(m_draggedItem).value<QByteArray>()); drag->setMimeData(mimeData); drag->exec(Qt::CopyAction | Qt::MoveAction, Qt::CopyAction); 5. 在目标QTreeWidget的dragEnterEvent和dragMoveEvent事件中,检查被拖动的item是否可以拖放到指定位置。如果不能,就设置禁止拖放的光标。 ```cpp if (event->mimeData()->hasFormat("application/x-qabstractitemmodeldatalist")) { QTreeWidgetItem *targetItem = this->itemAt(event->pos()); QTreeWidgetItem *parentItem = targetItem ? targetItem->parent() : this->invisibleRootItem(); if (parentItem == m_draggedParent) { event->accept(); } else { event->ignore(); this->setCursor(Qt::ForbiddenCursor); } else { event->ignore(); 6. 在目标QTreeWidget的dropEvent事件中,将被拖动的item插入到指定位置。 ```cpp if (event->mimeData()->hasFormat("application/x-qabstractitemmodeldatalist")) { QByteArray data = event->mimeData()->data("application/x-qabstractitemmodeldatalist"); QDataStream stream(&data, QIODevice::ReadOnly); QList<QVariant> itemData; QPoint offset; stream >> itemData >> offset; QTreeWidgetItem *targetItem = this->itemAt(event->pos()); QTreeWidgetItem *parentItem = targetItem ? targetItem->parent() : this->invisibleRootItem(); if (parentItem == m_draggedParent) { QTreeWidgetItem *newItem = new QTreeWidgetItem(parentItem, itemData); parentItem->insertChild(targetItem ? this->indexOfTopLevelItem(targetItem) : parentItem->childCount(), newItem); event->accept(); } else { event->ignore(); 完整的代码示例: ```cpp #include <QtWidgets> class TreeWidget : public QWidget { public: TreeWidget(QWidget *parent = nullptr) : QWidget(parent) { QVBoxLayout *layout = new QVBoxLayout(this); QHBoxLayout *hlayout = new QHBoxLayout; layout->addLayout(hlayout); m_sourceWidget = new QTreeWidget(this); m_sourceWidget->setColumnCount(1); m_sourceWidget->setHeaderHidden(true); m_sourceWidget->setDragEnabled(true); hlayout->addWidget(m_sourceWidget); m_targetWidget = new QTreeWidget(this); m_targetWidget->setColumnCount(1); m_targetWidget->setHeaderHidden(true); m_targetWidget->setAcceptDrops(true); hlayout->addWidget(m_targetWidget); QTreeWidgetItem *rootItem = new QTreeWidgetItem(m_sourceWidget, {"Root"}); QTreeWidgetItem *childItem = new QTreeWidgetItem(rootItem, {"Child"}); rootItem->addChild(childItem); connect(m_sourceWidget, &QTreeWidget::itemPressed, this, &TreeWidget::onSourceItemPressed); connect(m_targetWidget, &QTreeWidget::dragEnterEvent, this, &TreeWidget::onTargetDragEnter); connect(m_targetWidget, &QTreeWidget::dragMoveEvent, this, &TreeWidget::onTargetDragMove); connect(m_targetWidget, &QTreeWidget::dropEvent, this, &TreeWidget::onTargetDrop); void onSourceItemPressed(QTreeWidgetItem *item, int column) { Q_UNUSED(column) m_draggedItem = nullptr; m_draggedParent = nullptr; m_pressPos = QCursor::pos(); if (item && item->parent()) { m_draggedItem = item; m_draggedParent = item->parent(); void onTargetDragEnter(QDragEnterEvent *event) { if (event->mimeData()->hasFormat("application/x-qabstractitemmodeldatalist")) { QTreeWidgetItem *targetItem = m_targetWidget->itemAt(event->pos()); QTreeWidgetItem *parentItem = targetItem ? targetItem->parent() : m_targetWidget->invisibleRootItem(); if (parentItem == m_draggedParent) { event->accept(); } else { event->ignore(); m_targetWidget->setCursor(Qt::ForbiddenCursor); } else { event->ignore(); void onTargetDragMove(QDragMoveEvent *event) { if (event->mimeData()->hasFormat("application/x-qabstractitemmodeldatalist")) { QTreeWidgetItem *targetItem = m_targetWidget->itemAt(event->pos()); QTreeWidgetItem *parentItem = targetItem ? targetItem->parent() : m_targetWidget->invisibleRootItem(); if (parentItem == m_draggedParent) { event->accept(); } else { event->ignore(); m_targetWidget->setCursor(Qt::ForbiddenCursor); } else { event->ignore(); void onTargetDrop(QDropEvent *event) { if (event->mimeData()->hasFormat("application/x-qabstractitemmodeldatalist")) { QByteArray data = event->mimeData()->data("application/x-qabstractitemmodeldatalist"); QDataStream stream(&data, QIODevice::ReadOnly); QList<QVariant> itemData; QPoint offset; stream >> itemData >> offset; QTreeWidgetItem *targetItem = m_targetWidget->itemAt(event->pos()); QTreeWidgetItem *parentItem = targetItem ? targetItem->parent() : m_targetWidget->invisibleRootItem(); if (parentItem == m_draggedParent) { QTreeWidgetItem *newItem = new QTreeWidgetItem(parentItem, itemData); parentItem->insertChild(targetItem ? m_targetWidget->indexOfTopLevelItem(targetItem) : parentItem->childCount(), newItem); event->accept(); } else { event->ignore(); } else { event->ignore(); private: QTreeWidget *m_sourceWidget; QTreeWidget *m_targetWidget; QTreeWidgetItem *m_draggedItem; QTreeWidgetItem *m_draggedParent; QPoint m_pressPos; int main(int argc, char *argv[]) { QApplication app(argc, argv); TreeWidget widget; widget.show(); return app.exec(); ```

相关推荐

最新推荐

recommend-type

Qt自定义图形实现拖拽效果

在Qt框架中,自定义图形实现拖拽效果是一种常见的交互设计,用于增强用户界面的可操作性。本篇将详细介绍如何在Qt中创建一个可拖动的自定义图形,并提供相关的代码示例。 首先,我们需要创建一个自定义的图形视图类...
recommend-type

pyqt5对用qt designer设计的窗体实现弹出子窗口的示例

在这个示例中,我们创建了两个窗体:一个是主窗体(MainWindow),另一个是子窗体(Dialog)。主窗体包含一个QPushButton,而子窗体同样也包含一个QPushButton。 1. 主窗体(MainWindow)的创建: 在Qt Designer中...
recommend-type

Qt图形图像开发之曲线图表模块QChart库一个chart中显示两条曲线详细方法与实例

Qt图形图像开发之曲线图表模块QChart库一个chart中显示两条曲线详细方法与实例 在 Qt 图形图像开发中,QChart 库是一个非常强大的图表库,能够帮助开发者快速创建各种类型的图表。今天,我们将介绍如何在 QChart 库...
recommend-type

支持向量机非线性回归通用matlab程序【含Matlab源码】【Matlab精品】..zip

### 支持向量机非线性回归通用MATLAB程序解析 #### 一、概述 本文将详细介绍一个基于MATLAB的支持向量机(SVM)非线性回归的通用程序。该程序采用支持向量机方法来实现数据的非线性回归,并通过不同的核函数设置来适应不同类型的数据分布。此外,该程序还提供了数据预处理的方法,使得用户能够更加方便地应用此程序解决实际问题。 #### 二、核心功能与原理 ##### 1. 支持向量机(SVM) 支持向量机是一种监督学习模型,主要用于分类和回归分析。对于非线性回归任务,SVM通过引入核技巧(kernel trick)将原始低维空间中的非线性问题转换为高维空间中的线性问题,从而实现有效的非线性建模。 ##### 2. 核函数 核函数的选择直接影响到模型的性能。本程序内置了三种常用的核函数: - **线性核函数**:`K(x, y) = x'y` - **多项式核函数**:`K(x, y) = (x'y + 1)^d` - **径向基函数(RBF)**:`K(x, y) = exp(-γ|x - y|^2)` 其中RBF核函数被广泛应用于非线性问题中,因为它可以处理非常复杂的非线性关系。本程序默认使用的是RBF核函数,参数`D`用于控制高斯核函数的宽度。 ##### 3. 数据预处理 虽然程序本身没有直接涉及数据预处理的过程,但在实际应用中,对数据进行适当的预处理是非常重要的。常见的预处理步骤包括归一化、缺失值处理等。 ##### 4. 模型参数 - **Epsilon**: ε-insensitive loss function的ε值,控制回归带宽。 - **C**: 松弛变量的惩罚系数,控制模型复杂度与过拟合的风险之间的平衡。 #### 三、程序实现细节 ##### 1. 函数输入与输出 - **输入**: - `X`: 输入特征矩阵,维度为(n, l),其中n是特征数量,l是样本数量。 - `Y`: 目标值向量,长度为l。 - `Epsilon`: 回归带宽。 - `C`: 松弛变量的惩罚系数。 - `D`: RBF核函数的参数。 - **输出**: - `Alpha1`: 正的拉格朗日乘子向量。 - `Alpha2`: 负的拉格朗日乘子向量。 - `Alpha`: 拉格朗日乘子向量。 - `Flag`: 标记向量,表示每个样本的类型。 - `B`: 偏置项。 ##### 2. 核心代码解析 程序首先计算所有样本间的核矩阵`K`,然后构建二次规划问题并求解得到拉格朗日乘子向量。根据拉格朗日乘子的值确定支持向量,并计算偏置项`B`。 - **核矩阵计算**:采用RBF核函数,通过`exp(-(sum((xi-xj).^2)/D))`计算任意两个样本之间的相似度。 - **二次规划**:构建目标函数和约束条件,使用`quadprog`函数求解最小化问题。 - **支持向量识别**:根据拉格朗日乘子的大小判断每个样本是否为支持向量,并据此计算偏置项`B`。 #### 四、程序扩展与优化 - **多核函数支持**:可以通过增加更多的核函数选项,提高程序的灵活性。 - **自动调参**:实现参数自动选择的功能,例如通过交叉验证选择最优的`Epsilon`和`C`值。 - **并行计算**:利用MATLAB的并行计算工具箱加速计算过程,特别是当样本量很大时。 #### 五、应用场景 该程序适用于需要进行非线性回归预测的场景,如经济预测、天气预报等领域。通过调整核函数和参数,可以有效应对各种类型的非线性问题。 ### 总结 本程序提供了一个支持向量机非线性回归的完整实现框架,通过灵活的核函数设置和参数调整,能够有效地处理非线性问题。对于需要进行回归预测的应用场景,这是一个非常实用且强大的工具。
recommend-type

Pascal语言自动转换功能详解:基础到高级

自动转换功能是Pascal编程语言中的一个重要特性,特别是在处理文本文件操作时。Pascal语言允许程序员在读取文本文件时,无需显式地进行类型转换,因为其内部机制会自动将字符型的文件元素转换为与目标变量匹配的数据类型,如整型、实型或字符串型。这种自动转换在简化代码编写的同时,提高了效率,使得程序员可以专注于逻辑结构的设计。 在Pascal的基础教程中,第一章介绍初识Pascal语言,强调了编程在信息学奥林匹克竞赛中的重要性,要求参赛者掌握高级语言如Pascal。Pascal语言由瑞士苏黎世联邦工业大学的N.沃思教授设计,最初版本发布于1971年,并在后续得到了标准化,成为一种结构化、系统化的编程语言。 Pascal的特点包括但不限于: 1. **结构化**:Pascal语言基于ALGOL60发展而来,遵循模块化和结构化的编程原则,通过分块结构(如if嵌套、case语句、循环结构等)来组织代码,使得程序逻辑清晰易懂。 2. **系统性**:作为系统程序设计语言,它可以用于编写操作系统级的软件,如编译器,体现了其广泛的应用范围。 3. **易学易用**:Pascal语言的设计目标是使编程过程简单,编译器通常提供简洁的语法和易于理解的错误提示,便于初学者快速上手。 4. **类型安全**:自动转换功能确保了数据类型的兼容性,减少了类型错误的可能性,但同时也要求开发者在理解数据类型的前提下正确地使用变量。 5. **强大的功能**:尽管Pascal在70年代就已出现,但它仍具备较强的实用性,支持一维和多维数组、字符数组与字符串处理、枚举类型、子界和集合,以及过程与函数等高级概念。 6. **文件操作**:文件操作是Pascal的重要部分,允许程序员在程序中读写文本和二进制文件,这对于处理数据输入输出非常关键。 7. **附录扩展**:教程中还提供了丰富的补充材料,如字符串函数和数学函数列表,fillchar的使用技巧,调试技巧,以及不同的退出语句用法,有助于深入理解和实践Pascal。 Pascal的自动转换功能是其编程灵活性和高效性的一个体现,而Pascal语言本身则因其结构化、系统性和易用性,成为了初学者学习算法设计和系统编程的理想选择。通过理解并熟练运用这些特性,开发者能够更好地构建和维护复杂的程序。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

探索CMake编译OpenCV的替代方案:评估优缺点,选择最佳构建工具

![探索CMake编译OpenCV的替代方案:评估优缺点,选择最佳构建工具](https://atlas.pingcode.com/files/public/667929b44164a54a6ffb494d?x-oss-process=image/auto-orient,1/format,webp) # 1. CMake编译OpenCV的替代方案概述 CMake是一个流行的跨平台构建系统,用于编译和构建OpenCV等大型C++项目。然而,对于某些项目和用例,CMake可能存在局限性。本文探讨了CMake的替代方案,这些替代方案提供了不同的优点和功能,以满足各种编译和构建需求。 这些替代方案
recommend-type

uniapp defineProps

`uni-app defineProps` 是 `uni-app` 中用于在组件之间传递数据的一种方式。它允许开发者将一组属性作为参数从父组件传入到子组件,这样可以使得子组件能够访问并利用这些信息来定制其外观、功能等。 ### 使用场景 当你希望在组件间共享数据并且这种数据不会频繁改变时,`defineProps` 非常有用。例如,在构建应用的某个部分时,需要基于一些静态设置渲染界面元素,如颜色方案、标题文本或其他配置信息。 ### 示例 假设你有一个名为 `ThemeComponent.vue` 的组件,它需要接收主题背景色作为属性: ```javascript <template
recommend-type

Pascal语言基础:文本文件与机器视觉算法入门

"文本文件-机器视觉算法与应用01" 在PASCAL编程语言中,文件操作是一个重要的组成部分,用于存储和读取数据。文件分为三类:文本文件、有类型文件和无类型文件。以下是这些文件类型的详细说明: 1. **文本文件**:也称为正文文件或行文文件,它们是以人类可读的形式存在的,是人机交互的基础。文本文件通常包含ASCII字符,可以通过文字编辑器如DOS的`edit`或Turbo Pascal的内置编辑器创建、查看和修改。PASCAL程序也可以在运行时动态创建文本文件。 文本文件的操作包括: - **定义文件**:在PASCAL中,需要先定义文件变量,指定文件类型和打开模式(如只读、写入或追加)。 - **建立联系**:通过`assign`函数将内部文件名与实际磁盘上的文件路径关联起来。 - **打开文件**:使用`open`函数打开已分配的文件。 - **读写操作**:使用`read`和`write`语句对文件进行读写操作,或者使用`readln`和`writeln`处理整行数据。 - **关闭文件**:确保在完成操作后使用`close`函数关闭文件,以释放系统资源。 2. **有类型文件**:这类文件可以是顺序或随机访问的,它们通常用于存储结构化数据,如整数、浮点数或自定义数据类型。在PASCAL中,需要声明文件类型,并且可以指定每个记录的大小。 3. **无类型文件**:同样支持顺序或随机访问,但不预先定义数据类型,允许更灵活的数据存储。 学习PASCAL语言的过程中,会涉及到各种基本语法和结构,如: - **赋值语句**:用于给变量赋值,如`var x: integer; x := 10;` - **输出语句**:`write`和`writeln`用于输出数据到屏幕。 - **分支结构**:`if...then`和`case`语句用于根据条件执行不同代码块。 - **循环结构**:`for`、`while`和`repeat...until`循环控制流程。 - **数组**:一维和多维数组用于存储一组相同类型的数据。 - **字符串处理**:PASCAL提供了字符串处理函数,如截取、连接等。 - **过程与函数**:封装代码逻辑,实现模块化编程。 - **指针**:动态数据类型,允许直接操作内存地址。 - **文件操作**:如上述文本文件的读写,以及有类型和无类型文件的处理。