当前位置: 首页 > news >正文

Qt拖拽事件,实现控件内项的相互拖拽

文章目录

  • 1拖拽演示
  • 2 步骤
  • 3 实现

这里主要以QTableview控件为例,实现表格内数据的相互拖拽。

1拖拽演示

在这里插入图片描述

2 步骤

自定以QTableView类,在自定义类中重写拖拽事件:

void dropEvent(QDropEvent *event);
void dragEnterEvent(QDragEnterEvent *event);
void dragMoveEvent(QDragMoveEvent *event);
void dragLeaveEvent(QDragLeaveEvent *event);

实现自定义的QTableView,在鼠标点击项的时候,构造一个图片,模拟出项被拖动的样子,通过
QMimeData来传递数据,QDrag会触发表格的进入事件,在进入事件内出入一个新的项,并通过发送信号的对象,来通过指针删除原来的项,大概思路就是这样。

3 实现

#include <QTableView>
#include <QStandardItemModel>
#include <QDropEvent>
#include <QMouseEvent>
#include <QDragEnterEvent>
#include<QByteArray>
class CTableView : public QTableView
{Q_OBJECTpublic:CTableView(QWidget *parent);~CTableView();
public:void insertNewItem(QString str);void setHorizontalHeaderName(QString _name);bool removeItem(QString str);void dropEvent(QDropEvent *event);void dragEnterEvent(QDragEnterEvent *event);void dragMoveEvent(QDragMoveEvent *event);void dragLeaveEvent(QDragLeaveEvent *event);
private:void slotItemPressed(const QModelIndex &index);
private:QStandardItemModel*		m_model;
};
#include "ctableview.h"
#include <QHeaderView>
#include <QPainter>
#include <QMimeData>
#include <QDrag>
#include <QDebug>
CTableView::CTableView(QWidget *parent): QTableView(parent)
{setAcceptDrops(true);m_model = new QStandardItemModel();this->setModel(m_model);this->verticalHeader()->setVisible(false);this->setEditTriggers(QAbstractItemView::NoEditTriggers);connect(this, &CTableView::pressed, this, &CTableView::slotItemPressed);
}CTableView::~CTableView()
{}void CTableView::insertNewItem(QString str)
{int t_row = m_model->rowCount();QStandardItem* t_item = new QStandardItem(str);m_model->setItem(t_row, 0, t_item);m_model->item(t_row, 0)->setTextAlignment(Qt::AlignHCenter);m_model->item(t_row, 0)->setBackground(QColor(187, 203, 233));
}void CTableView::setHorizontalHeaderName(QString _name)
{m_model->setHorizontalHeaderLabels(QStringList(_name));this->horizontalHeader()->setSectionResizeMode(QHeaderView::Stretch);// 设置表头调整模式为 Stretchthis->horizontalHeader()->setStretchLastSection(true);// 将最后一列拉伸到表格宽度this->horizontalHeader()->setStyleSheet("QHeaderView::section {""color: rgb(32,159,223);padding-left: 4px;border: 1px solid #6c6c6c;}");
}bool CTableView::removeItem(QString str)
{for (int row = 0; row < m_model->rowCount(); row++){if (m_model->item(row, 0)->text() == str){return m_model->removeRow(row);}}return false;
}void CTableView::dropEvent(QDropEvent *event)
{if (!event->mimeData()->hasText()) { // 修改判断条件return;}insertNewItem(event->mimeData()->text());CTableView* tableView = qobject_cast<CTableView*>(event->source());if (tableView == nullptr){return;}tableView->removeItem(event->mimeData()->text());
}void CTableView::dragEnterEvent(QDragEnterEvent *event)
{//这个函数的主要作用是根据拖动数据中是否包含文本,决定是否接受拖放操作,并将操作类型设置为建议类型。if (event->mimeData()->hasText()) { // 修改判断条件event->acceptProposedAction();}
}void CTableView::dragMoveEvent(QDragMoveEvent *event)
{if (event->mimeData()->hasText()) { // 修改判断条件event->acceptProposedAction();}
}void CTableView::dragLeaveEvent(QDragLeaveEvent *event)
{Q_UNUSED(event);
}void CTableView::slotItemPressed(const QModelIndex &index)
{if (!index.isValid()){return;}QStandardItem* item = m_model->item(index.row(), index.column());QPixmap* pixmap = new QPixmap(this->horizontalHeader()->sectionSize(0), this->verticalHeader()->sectionSize(0)); // 创建画布pixmap->fill(QColor(187, 203, 233)); // 填充绿色背景QPainter t_painter(this); // 创建 QPainter 对象t_painter.begin(pixmap);t_painter.drawText(QRectF(0, 0, 100, 20), Qt::AlignCenter, item->text()); // 在画布上绘制文字t_painter.end();QMimeData* t_mimeData = new QMimeData;t_mimeData->setText(item->text());QPoint hotSpot(pixmap->width() / 2, pixmap->height() / 2); // 设置热点位置为图片中心QDrag* t_drag = new QDrag(this);t_drag->setMimeData(t_mimeData);t_drag->setPixmap(*pixmap);t_drag->setHotSpot(hotSpot); // 设置热点位置t_drag->exec(Qt::CopyAction | Qt::MoveAction);delete t_drag;t_drag = nullptr;delete pixmap;pixmap = nullptr;
}

程序主窗口,new CTableView表格

#include "ctestdragitem.h"
#include <QGridLayout>
const int TableCount = 9;
CTestDragItem::CTestDragItem(QWidget *parent): QWidget(parent)
{ui.setupUi(this);QGridLayout* gridLayout = new QGridLayout(ui.widget);gridLayout->setSpacing(10);gridLayout->setContentsMargins(10, 10, 10, 10);for (int i=0; i< TableCount; i++){CTableView * tableView = new CTableView(ui.widget);tableView->setHorizontalHeaderName(QString("table%1").arg(i));tableView->insertNewItem(QString("item%1%2").arg(i / 3).arg(i % 3));gridLayout->addWidget(tableView, i / 3, i % 3);}}

通过以上方法就可以实现,表格数据的相互拖拽,这里只是举例了简单的用法,更复杂的操作也差不多,需要自己处理逻辑。

http://www.lryc.cn/news/293667.html

相关文章:

  • 基于MATLAB实现的OFDM仿真调制解调,BPSK、QPSK、4QAM、16QAM、32QAM,加性高斯白噪声信道、TDL瑞利衰落信道
  • Redis核心技术与实战【学习笔记】 - 21.Redis实现分布式锁
  • 17.Golang channel的基本定义及使用
  • Linux - iptables 防火墙
  • 如何在FBX剔除Lit.shader依赖
  • cesium-测量高度垂直距离
  • Adobe Illustrator CEP插件开发入门指南
  • 【Spring】自定义注解 + AOP 记录用户的使用日志
  • linux互斥锁:递归锁,非递归锁用法详解
  • MacOS安装dmg提示已文件已损坏的解决方法
  • 前端输入框简单实现检测@成员输入
  • 通过与chatGPT交流实现零样本事件抽取
  • 使用nodejs和html布局一个简单的视频播放网站,但是使用localhost:端口访问html无法加载视频
  • 【AG32VF407】国产MCU+FPGA Verilog双边沿检测输出方波
  • [晓理紫]每日论文分享(有中文摘要,源码或项目地址)--强化学习、模仿学习、机器人
  • 为什么说TiDB在线扩容对业务几乎没有影响
  • STM32--SPI通信协议(2)W25Q64简介
  • svn安装与搭建
  • 什么是缓存击穿、缓存穿透、缓存雪崩?
  • springboot153相亲网站
  • CMake生成osg的FFMPEG插件及Windows下不生成VS工程问题解决
  • 代码随想录算法训练营Day25 | 216.组合总和III、17.电话号码的字母组合
  • 故障诊断 | 一文解决,SVM支持向量机的故障诊断(Matlab)
  • 12.1 Web开发_DOMBOM:JS关联CSS(❤❤)
  • scoped样式隔离原理
  • 降价不是杀手锏,和府捞面打起“养生牌”
  • 在WORD中设置公式居中编号右对齐设置方式
  • 如何使用 Supabase Auth 在您的应用程序中设置身份验证
  • 带libc源码gdb动态调试(导入glibc库使得可执行文件动态调试时可看见调用库函数源码)
  • 初级通信工程师-通信动力与环境