在拖放中识别一个实例

identify an instance in drag-and-drop

我努力了解如何在拖放中识别 class 的实例,但我需要一些帮助。

下面我从“http://zetcode.com/gui/pyqt5/dragdrop/”中添加一个示例来解释。

#!/usr/bin/python3
# -*- coding: utf-8 -*-

"""
ZetCode PyQt5 tutorial

In this program, we can press on a button with a left mouse
click or drag and drop the button with  the right mouse click. 

Author: Jan Bodnar
Website: zetcode.com
Last edited: August 2017
"""

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

class Button(QPushButton):

    def __init__(self, title, parent):
        super().__init__(title, parent)


    def mouseMoveEvent(self, e):

        if e.buttons() != Qt.RightButton:
            return

        mimeData = QMimeData()

        drag = QDrag(self)
        drag.setMimeData(mimeData)
        drag.setHotSpot(e.pos() - self.rect().topLeft())

        dropAction = drag.exec_(Qt.MoveAction)


    def mousePressEvent(self, e):

        super().mousePressEvent(e)

        if e.button() == Qt.LeftButton:
            print('press')


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_() 

这是我的问题:

如果我实例化第二个按钮

    self.button2 = Button('Button2', self)
    self.button2.move(100, 165)

我也可以移动按钮 2,但在 DropEvent 中移动了第一个按钮 "button"。

如何在 DropEvent 中识别我移动 "button2" 并放置按钮 2?

所以:问题是:如何识别,是哪个按钮。 (我只找到拖放示例,如何用另一个相同类型的元素替换标签或按钮)

另外一个问题:创建一个新实例适用于此类代码。如果我在 QT5 设计器中创建按钮,则不可能在此代码中将它们作为此类 class 按钮的实例。 因此,在 QT5 设计器中定义拖放,将列表中的项目拖放至另一个列表是可行的,但我无法识别我拖了哪个元素

您总是在放置事件中使用 self.button,因此它指的是同一个按钮。相反,您可以使用 QDropEvent.source() 来获取正确的小部件。

def dropEvent(self, e):
    position = e.pos()
    e.source().move(position) # source widget

    e.setDropAction(Qt.MoveAction)
    e.accept()

如果你从中拖动的对象是一个 QWidget,那么你可以通过 QDropEvent source() 方法获取它,而 QPushButton 是一个 QWidget,所以它适用于这种情况:

def dropEvent(self, e):
    if e.source() is not None:
        position = e.pos()
        e.source().move(position)
        e.setDropAction(Qt.MoveAction)
        e.accept()

记得验证它不是 None 因为源可能是另一个 window 而不是 QWidget 所以你会得到一个异常


如果您希望通过 Qt Designer 添加的按钮具有相同的功能,那么最简单的解决方案是升级按钮,SO 中有许多示例展示了如何做到这一点:

  • ,等等