PyQt拖拽事件——mimeData的使用(一)

实现思路

1、简要介绍QMimeData
2、QMimeData的用例1:在QT实现输入框的文字拖拽
3、QMimeData的用例2:在QT实现按钮拖动

两个用例的实现效果如下:

用例1:
在这里插入图片描述

用例2:
在这里插入图片描述

1、简要介绍QMimeData

一、QDrag

首先是创建QDrag,可以在mousePressEvent、mouseMoveEvent、dragMoveEvent中创建。

QDrag在exec前,一定要设置QMimeData,否则不会开始拖拽操作。

QMimeData在拖拽中非常有用,可以用来保存拖拽操作附带的信息,比如字符串、文件或者图片,同时也可以用来验证其所保存的信息格式,并以此来判断是否可接收。

另外要注意,在windows下,QDrag::exec()是个同步操作,要在exec()返回后,才会继续执行下面的代码。

二、drag相关事件

首先,当需要一个控件接收drag和drop,就要先调用控件的方法:setAcceptDrops(True)

qt中一共有三个drag相关事件,dragEnterEvent、dragMoveEvent、dragLeaveEvent。这三个事件触发条件类似鼠标移入,鼠标移动,鼠标移出。当鼠标拖拽进入控件触发dragEnterEvent,在控件内拖拽移动触发dragMoveEvent,鼠标拖拽离开控件触发dragLeaveEvent。

三、dropEvent

当drag为accept状态,然后释放鼠标,就会产生dropEvent。我们可以在这个事件里处理本次拖拽附带的Mime信息。

四、拖放关键逻辑图
在这里插入图片描述

2、QMimeData的用例1:在QT实现输入框的文字拖拽

# -*- coding: utf-8 -*-
import sys

from PyQt5.QtCore import Qt, QMimeData
from PyQt5.QtGui import QDrag
from PyQt5.QtWidgets import QWidget, QLineEdit, QApplication, QSplitter, QHBoxLayout


class MyLineEdit(QLineEdit):
    def __init__(self, parent):
        super().__init__(parent)
        self.setAcceptDrops(True)

    def dragMoveEvent(self, event):
        drag = QDrag(self)
        mime = QMimeData()
        drag.setMimeData(mime)
        drag.exec(Qt.CopyAction)

    def dragEnterEvent(self, event):
        if event.mimeData().hasText():
            event.accept()
        else:
            event.ignore()

    def dropEvent(self, event):
        self.setText(event.mimeData().text())
        event.source().setText("")


class SimpleDrag(QWidget):
    def __init__(self):
        super().__init__()

        self.initUI()

    def initUI(self):
        hlayout = QHBoxLayout(self)
        edit1 = MyLineEdit(self)
        edit1.setDragEnabled(True)
        edit2 = MyLineEdit(self)
        edit2.setDragEnabled(True)

        splitter = QSplitter(Qt.Horizontal)
        splitter.addWidget(edit1)
        splitter.addWidget(edit2)
        hlayout.addWidget(splitter)
        self.setLayout(hlayout)
        self.setWindowTitle('简易的拖动事件')


if __name__ == '__main__':
    app = QApplication(sys.argv)
    ex = SimpleDrag()
    ex.show()
    app.exec_()

关键解析:

在自定义控件中:

1、我们创建了一个继承自Qt的QLineEdit的输入框

2、在dragMoveEvent中创建了QDrag,并且设置了drag的mimeData,接着对QDrag调用exec方法

3、在dragEnterEvent中接收了该事件 即对应代码的 event.accept()

4、在dropEvent 中 对事件进行了放的处理

在主窗口中:

1、设置该窗口可以接收拖拽事件setDragEnabled(True)
这就完美对应上面的QMimeData的使用啦

3、QMimeData的用例2:在QT实现按钮拖动

# -*- coding: utf-8 -*-
import sys
from PyQt5.QtWidgets import QPushButton, QWidget, QApplication
from PyQt5.QtCore import Qt, QMimeData
from PyQt5.QtGui import QDrag


class Button(QPushButton):
    def __init__(self, title, parent):
        super().__init__(title, parent)

    def mouseMoveEvent(self, e):
        if e.buttons() != Qt.LeftButton:
            return
        mimeData = QMimeData()
        drag = QDrag(self)
        drag.setMimeData(mimeData)
        drag.setHotSpot(e.pos() - self.rect().topLeft())
        drag.exec_(Qt.MoveAction)


class Example(QWidget):
    def __init__(self):
        super().__init__()
        self.initUI()

    def initUI(self):
        self.setAcceptDrops(True)
        self.button = Button('Button', self)
        self.button.move(100, 65)
        self.setWindowTitle('Click or Move')
        self.setGeometry(300, 300, 280, 150)

    def dragEnterEvent(self, e):
        e.accept()

    def dropEvent(self, e):
        position = e.pos()
        self.button.move(position)
        e.setDropAction(Qt.MoveAction)
        e.accept()


if __name__ == '__main__':
    app = QApplication(sys.argv)
    ex = Example()
    ex.show()
    app.exec_()

关键解析:

在自定义控件中:

1、我们创建了一个继承自Qt的QPushButton的按钮

2、在mouseMoveEvent中创建了QDrag,并且设置了drag的mimeData,接着对QDrag调用exec方法

在主窗口中:

1、设置该窗口可以接收拖拽事件setDragEnabled(True)

2、在dropEvent 中 对事件进行了放的处理,改变按钮的位置

1、在dragEnterEvent中接收了该事件 即对应代码的 event.accept()

第二个例子跟第一个有点不一样,因为第一个例子中,放的事件给到输入框 MyLineEdit

而第二个例子中,此时接收放事件的控件是主窗口 Example(QWidget)

PS.后面还有一篇复杂的关于拖拽的使用,只是例子更为复杂,原理还是一样的

  • 11
    点赞
  • 49
    收藏
    觉得还不错? 一键收藏
  • 5
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值