python3+PyQt5重新實(shí)現(xiàn)自定義數(shù)據(jù)拖放處理
本文分成兩部分,第一部分通過python3+PyQt5實(shí)現(xiàn)自定義數(shù)據(jù)的拖放操作。第二部分則對(duì)第一部分的程序進(jìn)行修改,增加拖放操作時(shí),菜單提示是否移動(dòng)或拷貝,還有可以通過ctrl鍵盤來設(shè)置移動(dòng)過程中拷貝源而非會(huì)將源刪除。
自定義數(shù)據(jù)MIME數(shù)據(jù)類型QMimeData,MIME是一種用于處理具有多個(gè)組成部分的自定義數(shù)據(jù)的標(biāo)準(zhǔn)化格式。MIME數(shù)據(jù)由一個(gè)數(shù)據(jù)類型和一個(gè)子類型構(gòu)成–例如,text/plain,text/html,image/png,要處理自定義MIME數(shù)據(jù),就必須要選用一種自定義數(shù)據(jù)類型和一種子類型,然后將數(shù)據(jù)封裝到QMimeData對(duì)象中。本例子中,我們創(chuàng)建端為application/x-icon-and-text類型的新MIME數(shù)據(jù)。
注:
dragEnterEvent這是一個(gè)拖拽事件的函數(shù),我們把文件拖拽進(jìn)程序界面打開,之前必須setAcceptDrops(true)了以后拖拽,但是只設(shè)置acceptDrops還不夠,還需要在dragEnterEvent事件中對(duì)拖入的對(duì)象進(jìn)行篩選,判斷mimeData的類型是否是你能處理的,如果是,則調(diào)用event.acceptProposedAction()放行。拖放結(jié)束后會(huì)產(chǎn)生dropEvent事件,在那里進(jìn)行最后的放置操作??傊@是拖拽事件函數(shù)的一個(gè)篩選事件并放置的函數(shù)。
第一部分:
#!/usr/bin/env python3
import os
import sys
from PyQt5.QtCore import (QByteArray, QDataStream, QIODevice, QMimeData,
QPoint, QSize, Qt)
from PyQt5.QtWidgets import (QApplication, QDialog,QGridLayout,
QLineEdit, QListWidget,QListWidgetItem, QWidget)
from PyQt5.QtGui import QIcon,QColor,QPainter,QFontMetricsF,QDrag
class DropLineEdit(QLineEdit):
def __init__(self, parent=None):
super(DropLineEdit, self).__init__(parent)
self.setAcceptDrops(True)
def dragEnterEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.accept()
else:
event.ignore()
def dragMoveEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.setDropAction(Qt.CopyAction)
event.accept()
else:
event.ignore()
def dropEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
data = event.mimeData().data("application/x-icon-and-text")
stream = QDataStream(data, QIODevice.ReadOnly)
text = ""
#stream >> text
text=stream.readQString()
self.setText(text)
event.setDropAction(Qt.CopyAction)
event.accept()
else:
event.ignore()
class DnDListWidget(QListWidget):
def __init__(self, parent=None):
super(DnDListWidget, self).__init__(parent)
self.setAcceptDrops(True)
self.setDragEnabled(True)
def dragEnterEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.accept()
else:
event.ignore()
def dragMoveEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.setDropAction(Qt.MoveAction)
event.accept()
else:
event.ignore()
def dropEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
data = event.mimeData().data("application/x-icon-and-text")
stream = QDataStream(data, QIODevice.ReadOnly)
text = ""
icon = QIcon()
#stream >> text >> icon
text=stream.readQString()
stream >> icon
item = QListWidgetItem(text, self)
item.setIcon(icon)
event.setDropAction(Qt.MoveAction)
event.accept()
else:
event.ignore()
def startDrag(self, dropActions):
item = self.currentItem()
icon = item.icon()
data = QByteArray()
stream = QDataStream(data, QIODevice.WriteOnly)
#stream << item.text() << icon
stream.writeQString(item.text())
stream << icon
mimeData = QMimeData()
mimeData.setData("application/x-icon-and-text", data)
drag = QDrag(self)
drag.setMimeData(mimeData)
pixmap = icon.pixmap(24, 24)
drag.setHotSpot(QPoint(12, 12))
drag.setPixmap(pixmap)
if drag.exec(Qt.MoveAction) == Qt.MoveAction:
self.takeItem(self.row(item))
class DnDWidget(QWidget):
def __init__(self, text, icon=QIcon(), parent=None):
super(DnDWidget, self).__init__(parent)
self.setAcceptDrops(True)
self.text = text
self.icon = icon
def minimumSizeHint(self):
fm = QFontMetricsF(self.font())
if self.icon.isNull():
return QSize(fm.width(self.text), fm.height() * 1.5)
return QSize(34 + fm.width(self.text), max(34, fm.height() * 1.5))
def paintEvent(self, event):
height = QFontMetricsF(self.font()).height()
painter = QPainter(self)
painter.setRenderHint(QPainter.Antialiasing)
painter.setRenderHint(QPainter.TextAntialiasing)
painter.fillRect(self.rect(), QColor(Qt.yellow).lighter())
if self.icon.isNull():
painter.drawText(10, height, self.text)
else:
pixmap = self.icon.pixmap(24, 24)
painter.drawPixmap(0, 5, pixmap)
painter.drawText(34, height,
self.text + " (Drag to or from me!)")
def dragEnterEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.accept()
else:
event.ignore()
def dragMoveEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.setDropAction(Qt.CopyAction)
event.accept()
else:
event.ignore()
def dropEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
data = event.mimeData().data("application/x-icon-and-text")
stream = QDataStream(data, QIODevice.ReadOnly)
self.text = ""
self.icon = QIcon()
#stream >> self.text >> self.icon
self.text=stream.readQString()
stream>>self.icon
event.setDropAction(Qt.CopyAction)
event.accept()
self.updateGeometry()
self.update()
else:
event.ignore()
def mouseMoveEvent(self, event):
self.startDrag()
QWidget.mouseMoveEvent(self, event)
def startDrag(self):
icon = self.icon
if icon.isNull():
return
data = QByteArray()
stream = QDataStream(data, QIODevice.WriteOnly)
#stream << self.text << icon
stream.writeQString(self.text)
stream<<icon
mimeData = QMimeData()
mimeData.setData("application/x-icon-and-text", data)
drag = QDrag(self)
drag.setMimeData(mimeData)
pixmap = icon.pixmap(24, 24)
drag.setHotSpot(QPoint(12, 12))
drag.setPixmap(pixmap)
drag.exec(Qt.CopyAction)
class Form(QDialog):
def __init__(self, parent=None):
super(Form, self).__init__(parent)
dndListWidget = DnDListWidget()
path = os.path.dirname(__file__)
for image in sorted(os.listdir(os.path.join(path, "images"))):
if image.endswith(".png"):
item = QListWidgetItem(image.split(".")[0].capitalize())
item.setIcon(QIcon(os.path.join(path,
"images/{0}".format(image))))
dndListWidget.addItem(item)
dndIconListWidget = DnDListWidget()
dndIconListWidget.setViewMode(QListWidget.IconMode)
dndWidget = DnDWidget("Drag to me!")
dropLineEdit = DropLineEdit()
layout = QGridLayout()
layout.addWidget(dndListWidget, 0, 0)
layout.addWidget(dndIconListWidget, 0, 1)
layout.addWidget(dndWidget, 1, 0)
layout.addWidget(dropLineEdit, 1, 1)
self.setLayout(layout)
self.setWindowTitle("Custom Drag and Drop")
if __name__ == "__main__":
app = QApplication(sys.argv)
form = Form()
form.show()
app.exec_()
運(yùn)行結(jié)果:

第二部分:
#!/usr/bin/env python3
import os
import sys
from PyQt5.QtCore import (QByteArray, QDataStream, QIODevice, QMimeData,
QPoint, QSize, Qt)
from PyQt5.QtWidgets import (QApplication, QDialog,QGridLayout,
QLineEdit, QListWidget,QListWidgetItem, QWidget,QMenu)
from PyQt5.QtGui import QIcon,QColor,QPainter,QFontMetricsF,QDrag,QCursor
class DropLineEdit(QLineEdit):
def __init__(self, parent=None):
super(DropLineEdit, self).__init__(parent)
self.setAcceptDrops(True)
def dragEnterEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.accept()
else:
event.ignore()
def dragMoveEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.setDropAction(Qt.CopyAction)
event.accept()
else:
event.ignore()
def dropEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
data = event.mimeData().data("application/x-icon-and-text")
stream = QDataStream(data, QIODevice.ReadOnly)
text = ""
text=stream.readQString()
self.setText(text)
event.setDropAction(Qt.CopyAction)
event.accept()
else:
event.ignore()
class DnDMenuListWidget(QListWidget):
def __init__(self, parent=None):
super(DnDMenuListWidget, self).__init__(parent)
self.setAcceptDrops(True)
self.setDragEnabled(True)
self.dropAction = Qt.CopyAction
def dragEnterEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.accept()
else:
event.ignore()
def dragMoveEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.setDropAction(Qt.MoveAction)
event.accept()
else:
event.ignore()
def dropEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
data = event.mimeData().data("application/x-icon-and-text")
stream = QDataStream(data, QIODevice.ReadOnly)
text = ""
icon = QIcon()
text=stream.readQString()
stream>>icon
menu = QMenu(self)
menu.addAction("&Copy", self.setCopyAction)
menu.addAction("&Move", self.setMoveAction)
if menu.exec_(QCursor.pos()):
item = QListWidgetItem(text, self)
item.setIcon(icon)
event.setDropAction(self.dropAction)
event.accept()
return
else:
event.setDropAction(Qt.IgnoreAction)
event.ignore()
def setCopyAction(self):
self.dropAction = Qt.CopyAction
def setMoveAction(self):
self.dropAction = Qt.MoveAction
def startDrag(self, dropActions):
item = self.currentItem()
icon = item.icon()
data = QByteArray()
stream = QDataStream(data, QIODevice.WriteOnly)
stream.writeQString(item.text())
stream<<icon
mimeData = QMimeData()
mimeData.setData("application/x-icon-and-text", data)
drag = QDrag(self)
drag.setMimeData(mimeData)
pixmap = icon.pixmap(24, 24)
drag.setHotSpot(QPoint(12, 12))
drag.setPixmap(pixmap)
if (drag.exec(Qt.MoveAction|Qt.CopyAction) == Qt.MoveAction):
self.takeItem(self.row(item))
class DnDCtrlListWidget(QListWidget):
def __init__(self, parent=None):
super(DnDCtrlListWidget, self).__init__(parent)
self.setAcceptDrops(True)
self.setDragEnabled(True)
def dragEnterEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.accept()
else:
event.ignore()
def dragMoveEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
action = Qt.MoveAction
if event.keyboardModifiers() & Qt.ControlModifier:
action = Qt.CopyAction
event.setDropAction(action)
event.accept()
else:
event.ignore()
def dropEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
data = event.mimeData().data("application/x-icon-and-text")
stream = QDataStream(data, QIODevice.ReadOnly)
text = ""
icon = QIcon()
text=stream.readQString()
stream>>icon
item = QListWidgetItem(text, self)
item.setIcon(icon)
action = Qt.MoveAction
if event.keyboardModifiers() & Qt.ControlModifier:
action = Qt.CopyAction
event.setDropAction(action)
event.accept()
else:
event.ignore()
def startDrag(self, dropActions):
item = self.currentItem()
icon = item.icon()
data = QByteArray()
stream = QDataStream(data, QIODevice.WriteOnly)
stream.writeQString(item.text())
stream<<icon
mimeData = QMimeData()
mimeData.setData("application/x-icon-and-text", data)
drag = QDrag(self)
drag.setMimeData(mimeData)
pixmap = icon.pixmap(24, 24)
drag.setHotSpot(QPoint(12, 12))
drag.setPixmap(pixmap)
if (drag.exec(Qt.MoveAction|Qt.CopyAction) == Qt.MoveAction):
self.takeItem(self.row(item))
class DnDWidget(QWidget):
def __init__(self, text, icon=QIcon(), parent=None):
super(DnDWidget, self).__init__(parent)
self.setAcceptDrops(True)
self.text = text
self.icon = icon
def minimumSizeHint(self):
fm = QFontMetricsF(self.font())
if self.icon.isNull():
return QSize(fm.width(self.text), fm.height() * 1.5)
return QSize(34 + fm.width(self.text), max(34, fm.height() * 1.5))
def paintEvent(self, event):
height = QFontMetricsF(self.font()).height()
painter = QPainter(self)
painter.setRenderHint(QPainter.Antialiasing)
painter.setRenderHint(QPainter.TextAntialiasing)
painter.fillRect(self.rect(), QColor(Qt.yellow).lighter())
if self.icon.isNull():
painter.drawText(10, height, self.text)
else:
pixmap = self.icon.pixmap(24, 24)
painter.drawPixmap(0, 5, pixmap)
painter.drawText(34, height,
self.text + " (Drag to or from me!)")
def dragEnterEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.accept()
else:
event.ignore()
def dragMoveEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
event.setDropAction(Qt.CopyAction)
event.accept()
else:
event.ignore()
def dropEvent(self, event):
if event.mimeData().hasFormat("application/x-icon-and-text"):
data = event.mimeData().data("application/x-icon-and-text")
stream = QDataStream(data, QIODevice.ReadOnly)
self.text = ""
self.icon = QIcon()
self.text=stream.readQString()
stream>>self.icon
event.setDropAction(Qt.CopyAction)
event.accept()
self.updateGeometry()
self.update()
else:
event.ignore()
def mouseMoveEvent(self, event):
self.startDrag()
QWidget.mouseMoveEvent(self, event)
def startDrag(self):
icon = self.icon
if icon.isNull():
return
data = QByteArray()
stream = QDataStream(data, QIODevice.WriteOnly)
stream.writeQString(self.text)
stream<<icon
mimeData = QMimeData()
mimeData.setData("application/x-icon-and-text", data)
drag = QDrag(self)
drag.setMimeData(mimeData)
pixmap = icon.pixmap(24, 24)
drag.setHotSpot(QPoint(12, 12))
drag.setPixmap(pixmap)
drag.exec(Qt.CopyAction)
class Form(QDialog):
def __init__(self, parent=None):
super(Form, self).__init__(parent)
dndListWidget = DnDMenuListWidget()
path = os.path.dirname(__file__)
for image in sorted(os.listdir(os.path.join(path, "images"))):
if image.endswith(".png"):
item = QListWidgetItem(image.split(".")[0].capitalize())
item.setIcon(QIcon(os.path.join(path,
"images/{0}".format(image))))
dndListWidget.addItem(item)
dndIconListWidget = DnDCtrlListWidget()
dndIconListWidget.setViewMode(QListWidget.IconMode)
dndWidget = DnDWidget("Drag to me!")
dropLineEdit = DropLineEdit()
layout = QGridLayout()
layout.addWidget(dndListWidget, 0, 0)
layout.addWidget(dndIconListWidget, 0, 1)
layout.addWidget(dndWidget, 1, 0)
layout.addWidget(dropLineEdit, 1, 1)
self.setLayout(layout)
self.setWindowTitle("Custom Drag and Drop")
if __name__ == "__main__":
app = QApplication(sys.argv)
form = Form()
form.show()
app.exec_()
運(yùn)行結(jié)果:

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
python開發(fā)實(shí)時(shí)可視化儀表盤的示例
這篇文章主要介紹了python開發(fā)實(shí)時(shí)可視化儀表盤的示例,幫助大家更好的理解和學(xué)習(xí)使用python,感興趣的朋友可以了解下2021-05-05
Python讀取圖片屬性信息的實(shí)現(xiàn)方法
這篇文章介紹了利用Python讀取圖片屬性信息的方法,讀取的內(nèi)容包括GPS 信息、圖片分辨率、圖片像素、設(shè)備商、拍攝設(shè)備等,有需要的朋友們可以參考借鑒。2016-09-09
python自動(dòng)發(fā)郵件庫yagmail的示例代碼
本篇文章主要介紹了python自動(dòng)發(fā)郵件庫yagmail的示例代碼,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-02-02
已安裝Pytorch卻提示no?moudle?named?'torch'(沒有名稱為torch
這篇文章主要給大家介紹了關(guān)于已安裝Pytorch卻提示no?moudle?named?'torch'(沒有名稱為torch的模塊)的相關(guān)資料,當(dāng)提示"No module named 'torch'"時(shí),可能是由于安裝的Pytorch版本與當(dāng)前環(huán)境不匹配導(dǎo)致的,需要的朋友可以參考下2023-11-11
Python importlib動(dòng)態(tài)導(dǎo)入模塊實(shí)現(xiàn)代碼
這篇文章主要介紹了Python importlib動(dòng)態(tài)導(dǎo)入模塊實(shí)現(xiàn)代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04
Python編譯成.so文件進(jìn)行加密后調(diào)用的實(shí)現(xiàn)
今天小編就為大家分享一篇Python編譯成.so文件進(jìn)行加密后調(diào)用的實(shí)現(xiàn),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2019-12-12
如何使用Python實(shí)現(xiàn)CartPole游戲
在深度強(qiáng)化學(xué)習(xí)內(nèi)容的介紹中,提出了CartPole游戲進(jìn)行深度強(qiáng)化學(xué)習(xí),現(xiàn)在提供一種用Python簡(jiǎn)單實(shí)現(xiàn)Cart Pole游戲的方法,感興趣的朋友跟隨小編一起看看吧2024-07-07

