分享好友 编程语言首页 频道列表

Python PyQt5实现拖放效果的原理详解

Python  2023-02-09 03:430

PyQt5的拖放

拖放涉及到的主要的一些类如下所示:

Python PyQt5实现拖放效果的原理详解

一、拖放的基本原理

1.1 拖放的动作

拖放操作包括两个动作:

  • 拖动(drag)
  • 放下(drop 或称为放置)。

当被拖动时拖动的数据会被存储为 MIME 类型的对象, MIME 类型使用 QMimeData 类来描述。 MIME 类型通常由剪贴板和拖放系统使用,以识别不同类型的数据。

  • 拖动点(drag site):拖动的起始位置。
  • 放下点(drop site):被拖动的对象放下的位置,若部件不能接受拖动的对象, Qt 会改变光标的形状(一个禁用形状)来向用户进行说明。

1.2 拖动的启动和结束

启动拖放:

拖放通过调用 QDrag::exec()函数而启动,该函数是一个阻塞函数(但不会阻塞主事件循环),这意味着在拖放操作结束之前,不会返回该函数,调用 QDrag::exec()函数后, Qt 拥有对拖动对象的所有权,并会在必要时将其删除。

结束拖放:

当用户放下拖动或取消拖动操作时结束拖放。

1.3 拖放产生的过程和事件

启动拖放后,会使数据被拖动,这时需要按住鼠标按键才能拖动需要拖动的数据,松开鼠标按键时意味着拖动结束。

默认情况下,部件不接受放下事件。使用 QWidget::setAcceptDrops()函数可设置部件是否接受放下事件(即,拖放完成时发送的事件)。只有在部件接受放下事件的情形下,才会产生以下事件。

QDragEnterEvent:拖动进入事件

当拖动操作进入部件时,该事件被发送到部件,忽略该事件,将会导至后续的拖放事件不能被发送。 通常在该部件上光标会在外观上显示为禁用的图形。

QDragMoveEvnet:拖动移动事件

当拖动操作正在进行时,以及当具有焦点时按下键盘的修饰键(比如 Ctrl)时, 发送该事件, 要使部件能接收到该事件,则该部件必须接受 QDragEnterEvent 事件。

QDropEvent:放下事件

在完成拖放操作时发送该事件,即当用户在部件上放下一个对象时,发送此事件。要使部件能接收到该事件,则该部件必须接受 QDragEnterEvent事件,且不能忽略 QDragMoveEvnt 事件。

QLeaveEvent:当拖放操作离开部件时发送该事件

注意:要使部件能接收到该事件,必须要使拖动先进入该部件(即产生 QDragEnterEvent 事件),然后再离开该部件,才会产生 QLeaveEvent 事件。因很少使用该事件,因此本文不做重点介绍。

上文中提到的必须接受是指必须重新实现该事件的处理函数并接受该事件,不能忽略是指在处件事理函数中不明确调用 ignore()函数忽略该事件,这意味着可以不必重新实现该事件的处理函数。

以上事件产生的顺序为: QDragEnterEvent、 QDragMoveEvnet、 QDropEvent

Python PyQt5实现拖放效果的原理详解

1.4 编写拖放程序的步骤

在需要接受放下数据的部件上调用 QWidget::setAcceptDrops()函数以使该部件能接受拖放事件。

启动拖放: 通常在 mousePressEvent()或 mouseMoveEvent()函数中启动拖放,记住启动拖放就是调用 QDrag 对象的 exec()函数,因此也可以在 keyPressEvent()等函数中启动拖放(因很少这样做,所以本文不介绍这种情况下的拖放)。 在此步把需要拖动的数据保存在 QMimeData 对象中。

重新实现需要接受放下数据的部件的 dragEnterEvent()事件处理函数。

根据需要重新实现 dragMoveEvent 或 dropEvent()函数

1.5 简单的拖放示例代码

本示例程序示范了如何把数据从按钮A拖至按钮B:

from PyQt5.QtWidgets import QApplication, QWidget, QPushButton, QHBoxLayout
from PyQt5.QtCore import QMimeData, qDebug
from PyQt5 import QtGui
from PyQt5.QtGui import QDrag
import sys

​​​​​​​class MyButton(QPushButton):
    def __init__(self, text:str) -> None:
        super().__init__(text)
        
    def mousePressEvent(self, e: QtGui.QMouseEvent) -> None:
        '''
        在该事件中启动拖放
        '''
        # 将需要拖动的数据放入QMimeData对象中,该对象用于保存需要传递的数据
        # 数据的内容完全由程序员自行设定。通常为界面上所选择的内容。
        my_mime_data = QMimeData()
        
        # 这是QMimeData中存储的内容,即拖放的数据
        my_mime_data.setText(self.text())
        
        # 设置拖动的数据,该函数会获得QMimeData的所有权
        my_drag = QDrag(self)
        my_drag.setMimeData(my_mime_data)
        
        # 启动拖放
        my_drag.exec_()
        
    def dragEnterEvent(self, e: QtGui.QDragEnterEvent) -> None:
        '''
        处理是否接受拖动事件
        '''
        # 接受拖动进入事件
        e.accept()
        
        # 若忽略该事件,则不会再发送之后的事件,拖放至此结束,这会导致鼠标光标显示为禁用的图形
        # e.ignore()
        
    def dropEvent(self, e: QtGui.QDropEvent) -> None:
        '''
        处理拖动的数据(当然了,也可以不做任何处理)
        '''
        # 设置此部件的文本为拖动对象中的文本
        self.setText(e.mimeData().text())
        
        # 此事件不影响后续事件,可接受也可忽略
        # e.accept()
        # e.ignore()
        
class MyWidget(QWidget):
    def __init__(self, parent=None) -> None:
        super().__init__(parent)
        self.__init_ui()
        
    def __init_ui(self):
        btn_a = MyButton('AAA')
        btn_b = MyButton('BBB')
        btn_a.setAcceptDrops(False)
        btn_b.setAcceptDrops(True)
        
        layout = QHBoxLayout()
        layout.addWidget(btn_a)
        layout.addWidget(btn_b)
        self.setLayout(layout)
        
if __name__ == '__main__':
    app = QApplication(sys.argv)
    my_widget = MyWidget()
    my_widget.show()
    sys.exit(app.exec_())

运行效果如下:

原始状态:

Python PyQt5实现拖放效果的原理详解

在按钮AAA上按下鼠标左键不动并拖动到图示位置,由于主窗口不接受放下事件,因此光标显示为禁用的状态

Python PyQt5实现拖放效果的原理详解

拖动AAA到按钮BBB上时,会发送QDragEnterEvent事件,同时光标改变形状,表示BBB按钮可以接受拖动的数据,

在按钮BBB上释放鼠标时,此时发送QDropEvent事件,按钮BBB的文本被修改为拖动对象中保存的数据。

Python PyQt5实现拖放效果的原理详解

至此,拖动结束。

原文地址:https://blog.csdn.net/hubing_hust/article/details/128072839

查看更多关于【Python】的文章

展开全文
相关推荐
反对 0
举报 0
评论 0
图文资讯
热门推荐
优选好物
更多热点专题
更多推荐文章
如何在Abaqus的python中调用Matlab程序
目录1. 确定版本信息2. 备份python3. 设置环境变量4. 安装程序5. 调试运行参考资料Abaqus2018操作系统Win10 64位Python版本2.7(路径C:\SIMULIA\CAE\2018\win_b64\tools\SMApy\python2.7)2. 备份python将上述的“python2.7”文件夹复制出来,避免因操作错误

0评论2023-03-16608

sf02_选择排序算法Java Python rust 实现
Java 实现package common;public class SimpleArithmetic {/** * 选择排序 * 输入整形数组:a[n] 【4、5、3、7】 * 1. 取数组编号为i(i属于[0 , n-2])的数组值 a[i],即第一重循环 * 2. 假定a[i]为数组a[k](k属于[i,n-1])中的最小值a[min],即执行初始化 min =i

0评论2023-02-09407

Python vs Ruby: 谁是最好的 web 开发语言?
Python 和 Ruby 都是目前用来开发 websites、web-based apps 和 web services 的流行编程语言之一。 这两种语言在许多方面有相似之处。它们都是高级的面向对象的编程语言,都是交互式脚本语言、都提供标准库且支持持久化。但是,Python 和 Ruby 的解决方法却

0评论2023-02-09819

Python+Sklearn实现异常检测
目录离群检测 与 新奇检测Sklearn 中支持的方法孤立森林 IsolationForestLocal Outlier FactorOneClassSVMElliptic Envelope离群检测 与 新奇检测很多应用场景都需要能够确定样本是否属于与现有的分布,或者应该被视为不同的分布。离群检测(Outlier detectio

0评论2023-02-09736

Python异常与错误处理详细讲解 python的异常
基础知识优先使用异常捕获LBYL(look before you leap): 在执行一个可能出错的操作时,先做一些关键的条件判断,仅当满足条件时才进行操作。EAFP(eaiser to ask for forgiveness than permission): 不做事前检查,直接执行操作。后者更优: 代码简洁,效率更高

0评论2023-02-09962

Python多线程与同步机制浅析
目录线程实现Thread类函数方式继承方式同步机制同步锁Lock条件变量Condition信号量Semaphore事件Event屏障BarrierGIL全局解释器锁线程实现Python中线程有两种方式:函数或者用类来包装线程对象。threading模块中包含了丰富的多线程支持功能:threading.curren

0评论2023-02-09409

python基础之reverse和reversed函数的介绍及使用
目录一、reverse二、reversed附:Python中reverse和reversed反转列表的操作方法总结一、reversereverse()是python中列表的一个内置方法(在字典、字符串和元组中没有这个内置方法),用于列表中数据的反转例子:lista = [1, 2, 3, 4]lista.reverse()print(lista

0评论2023-02-09878

Python多进程并发与同步机制超详细讲解
目录多进程僵尸进程Process类函数方式继承方式同步机制状态管理Managers在《多线程与同步》中介绍了多线程及存在的问题,而通过使用多进程而非线程可有效地绕过全局解释器锁。 因此,通过multiprocessing模块可充分地利用多核CPU的资源。多进程多进程是通过mu

0评论2023-02-09469

Python进程间通讯与进程池超详细讲解 python进程池的作用
目录进程间通讯队列Queue管道Pipe进程池Pool在《多进程并发与同步》中介绍了进程创建与信息共享,除此之外python还提供了更方便的进程间通讯方式。进程间通讯multiprocessing中提供了Pipe(一对一)和Queue(多对多)用于进程间通讯。队列Queue队列是一个可用

0评论2023-02-09797

Python PyMuPDF实现PDF与图片和PPT相互转换
目录安装与简介MuPDFPyMuPDFPyMuPDF使用元数据页面Page代码示例PDF转图片图片转PDFPDF转PPT文章目录 安装与简介MuPDFPyMuPDF PyMuPDF使用元数据页面Page 代码示例PDF转图片图片转PDFPDF转PPTPyMuPDF提供了PDF及流行图片处理接口。安装与简介安装:pip install

0评论2023-02-09349

更多推荐