Qt中的拖拽(针对两个界面之间的拖动)

功能描述:
两个界面之间拖拽控件,在另一个界面生成相同的控件

.h文件
#include <QEvent>
#include <QDrag>
#include <QDragEnterEvent>
#include <QDragLeaveEvent>
#include <QDropEvent>
#include <QEvent>
#include <QMimeData>
#include <QVBoxLayout>
#include <QGridLayout>
#include <QPixmap>
#include "ui_TestBag.h"

class TestBag : public QWidget
{
    Q_OBJECT

public:
    TestBag(QWidget *parent = Q_NULLPTR);

protected:
	void dragEnterEvent(QDragEnterEvent* event);
	void dragMoveEvent(QDragMoveEvent* event);
	void dropEvent(QDropEvent* event);
	void mousePressEvent(QMouseEvent* event);
private:
    Ui::TestBagClass ui;

	QVBoxLayout* m_pVBoxLayout;
	//QGridLayout* m_pGrid
};

.cpp文件
#include "TestBag.h"
#include <QFile>
#include <QFileDialog>
#include <QPushButton>
#include <QTextStream>
#include <QLabel>
#include <QPainter>

TestBag::TestBag(QWidget *parent)
    : QWidget(parent)
{
    ui.setupUi(this);

	this->setMouseTracking(true);
	this->setAcceptDrops(true);

	QLabel* pLabel = new QLabel(this);
	pLabel->setText(tr("Test"));
	m_pVBoxLayout = new QVBoxLayout;
	m_pVBoxLayout->addWidget(pLabel);
	this->setLayout(m_pVBoxLayout);
	m_pVBoxLayout->setAlignment(Qt::AlignTop);
}

void TestBag::dragEnterEvent(QDragEnterEvent* event)
{
	if (event->mimeData()->hasFormat("application/x-dnditemdata")) {
		if (event->source() == this) {
			event->setDropAction(Qt::MoveAction);
			event->accept();
		}
		else
			event->acceptProposedAction();
	}
	else
		event->ignore();
}

void TestBag::dragMoveEvent(QDragMoveEvent* event)
{
	if (event->mimeData()->hasFormat("application/x-dnditemdata")){
		if (event->source() == this) {
			event->setDropAction(Qt::MoveAction);
			event->accept();
		}
		else
			event->acceptProposedAction();
	}
	else {
		event->ignore();
	}
}

void TestBag::dropEvent(QDropEvent* event)
{
	if (event->mimeData()->hasFormat("application/x-dnditemdata")) {
		QByteArray array = event->mimeData()->data("application/x-dnditemdata");
		QDataStream stream(&array, QIODevice::ReadOnly);
		QString text = "";
		QPoint offset;
		stream >> text >> offset;

		//移入的对象
		QLabel* pResult = new QLabel(this);
		pResult->setText(text);
		pResult->move(event->pos() - offset);
		pResult->show();
		//加入到本界面的布局中
		m_pVBoxLayout->addWidget(pResult);


		if (event->source() == this)
		{
			event->setDropAction(Qt::MoveAction);
			event->accept();
		}
		else
			event->acceptProposedAction();
	}
	else {
		event->ignore();
	}
}

void TestBag::mousePressEvent(QMouseEvent* event)
{
	QLabel* pDragButton = static_cast<QLabel*>(childAt(event->pos()));
	if (pDragButton == nullptr)
		return;

	//获取移动控件截图
	QPixmap image = QPixmap::grabWidget(pDragButton, pDragButton->rect());
	QString text = pDragButton->text();
	QByteArray array;
	QDataStream stream(&array, QIODevice::WriteOnly);
	stream << text << QPoint(event->pos() - pDragButton->pos());

	QMimeData* pMimeData = new QMimeData;
	pMimeData->setData("application/x-dnditemdata", array);

	//将拖动的控件截图跟着鼠标光标移动
	QDrag* drag = new QDrag(this);
	drag->setMimeData(pMimeData);
	drag->setPixmap(image);
	drag->setHotSpot(event->pos() - pDragButton->pos());

	if (drag->exec(Qt::CopyAction | Qt::MoveAction, Qt::CopyAction) == Qt::MoveAction) {
		pDragButton->close();
	}
	else {
		pDragButton->show();
		pDragButton->setText(text);
	}
}
  • 3
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: QDrag是Qt框架用于实现拖拽操作的类。当我们需要在两个见面之间拖拽时,可以通过以下步骤使用QDrag: 1. 首先,我们需要在QObject派生出的类重写mousePressEvent事件。在该事件,我们需要创建QDrag对象,并设置其相关属性。例如,我们可以使用setMimeData方法设置拖动的数据类型,使用setPixmap方法设置拖动时显示的图标。 2. 接下来,在mousePressEvent事件,我们还需要调用exec方法开始拖拽操作。exec方法将启动一个事件循环,直到拖拽结束为止。我们可以在exec方法设置拖拽操作的行为,例如设置Qt::MoveAction,表示移动操作(即拖拽源上的内容将被移动到目标位置)。 3. 当拖拽操作结束时,会触发dragLeaveEvent事件和dropEvent事件。我们可以在这两个事件分别处理拖拽离开和放下的操作。例如,在dropEvent事件,我们可以进行一些数据处理的操作,将拖拽的数据传输到目标位置。 总的来说,使用QDrag实现两个见面之间拖拽操作,我们需要重写mousePressEvent事件,创建并设置QDrag对象的相关属性,调用exec方法开始拖拽操作,并在dragLeaveEvent事件和dropEvent事件处理拖拽离开和放下的操作。这样,我们就可以实现Qt项目两个见面之间的简单拖拽操作。 ### 回答2: Qt是一个跨平台的GUI开发框架,其提供了丰富的控件和功能供开发者使用。QDrag是Qt的一个类,我们可以通过使用它来实现拖拽操作。 首先,我们需要在界面上创建两个控件,一个是源控件,一个是目标控件。接着,我们可以通过给源控件添加鼠标事件来实现拖拽操作。 当鼠标按下时,我们可以将源控件作为拖拽操作的起始点,同时创建一个QDrag对象。然后,我们通过将一些数据(例如文本信息或文件路径)与QDrag对象关联起来,使得在拖拽操作可以传递这些数据。 接着,我们可以通过调用QDrag对象的exec()函数开始拖拽操作。在拖拽过程,我们可以根据鼠标的位置更新目标控件的状态。当鼠标释放时,我们可以根据拖拽的结果来进行相应的操作。 需要注意的是,为了使得目标控件能够接受拖拽的数据,我们还需要在目标控件实现相关的事件处理函数。例如,在目标控件的dragEnterEvent()函数,我们可以指定当拖拽进入目标控件时的一些表现,例如改变鼠标的样式或者显示一些提示信息。 总结起来,通过使用Qt的QDrag类,我们可以很方便地实现拖拽操作。只需要在源控件实现鼠标事件并与QDrag对象关联,然后在目标控件实现拖拽事件的处理即可。这样,我们就可以在Qt灵活地处理拖拽操作,提升用户的交互体验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Pailugou

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值