繁体   English   中英

获取拖放目标 pyqt5

[英]Getting drag and drop target pyqt5

我正在 pyqt5 中使用拖放 GUI 并尝试获取拖放操作的目标小部件,但是当我尝试 QDrag object 的目标()function 时,我返回 < main .MainWindow object at 0x0000025FDAC09EE0> 并且我不知道如何使用它。 我想访问 QGridLayout 中小部件的索引,以便我可以让两个小部件交换位置。

这是我的代码:

import sys

from PyQt5.QtCore import Qt, QMimeData
from PyQt5.QtGui import QDrag
from PyQt5 import QtWidgets
from PyQt5.QtWidgets import QPushButton, QWidget, QApplication, QGridLayout, QScrollArea, QMainWindow, QSlider


class Stroj:
    
    def __init__(self, rok, naziv, trajanje):
        self.rok = rok
        self.naziv = naziv
        self.trajanje = trajanje
        

class Button(QPushButton):
    drag = 0

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

    def mouseMoveEvent(self, e):

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

        mimeData = QMimeData()
        mimeData.setText(self.text())

        self.drag = QDrag(self)
        self.drag.setMimeData(mimeData)
        self.drag.setPixmap(self.grab())
        self.drag.setHotSpot(self.rect().center())

        dropAction = self.drag.exec_(Qt.MoveAction)


class MainWindow(QMainWindow):
    layout = QGridLayout()
    btns = []
    snd = ""
    i = 0
    j = 0

    def __init__(self):
        super().__init__()

        self.initUI()
        
    def initUI(self):

        self.setAcceptDrops(True)
        self.scroll = QScrollArea()
        self.widget = QWidget()
        self.drag = QDrag(self)
        
        SL = []
        for x in range(30):
            self.btns.append(x)
        
        for x in range(30):
            
            self.btns[x] = Button(str(x), self)
            self.btns[x].setSizePolicy(QtWidgets.QSizePolicy.Minimum, QtWidgets.QSizePolicy.Minimum)
            self.layout.addWidget(self.btns[x], self.i, self.j)
            if(self.j > 5):
                self.j = 0
                self.i += 1
            else:
                self.j += 1
        
        self.widget.setLayout(self.layout)
        self.scroll.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOn)
        self.scroll.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOn)
        self.scroll.setWidgetResizable(True)
        self.scroll.setWidget(self.widget)

        self.setCentralWidget(self.scroll)

        self.setWindowTitle('Raspored')
        self.setGeometry(350, 75, 950, 750)

    def dragEnterEvent(self, e):
        self.snd = e.mimeData().text()
        e.accept()
        
    def dragMoveEvent(self, e):
        e.accept()

    def dropEvent(self, e):
        sender = self.snd
        position = e.pos()
        position.setX(int(position.x() - self.btns[int(sender)].width() / 2))
        position.setY(int(position.y() - self.btns[int(sender)].height() / 2))
        self.btns[int(sender)].move(position)
        print(self.layout.indexOf(e.source()))
        print(e.source().drag.target())

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


def main():
    
    app = QApplication(sys.argv)
    main = MainWindow()
    main.show()
    app.exec_()


if __name__ == '__main__':
    main()

放置事件的目标始终是接收放置操作的小部件,因此很明显,如果您从主 window 实例拦截事件,您将获得主 window 作为目标。 如果您需要在特定的 position 找到小部件,则需要使用QApplication.widgetAt(pos)

在下面的示例中,根据给定的代码进行了修改,仅当源是Button实例并且目标不相同时,我才接受 dragEnter/dragMove 事件。 然后我使用布局中的 position 切换这些按钮。

class MainWindow(QMainWindow):
    # ...
    def dragEnterEvent(self, e):
        e.accept()
        
    def dragMoveEvent(self, e):
        source = e.source()
        target = QApplication.widgetAt(self.mapToGlobal(e.pos()))
        if (isinstance(e.source(), Button) and isinstance(target, Button) and target != source):
            e.accept()
        else:
            e.ignore()

    def dropEvent(self, e):
        source = e.source()
        target = QApplication.widgetAt(self.mapToGlobal(e.pos()))
        if (not isinstance(source, Button) or not isinstance(target, Button) 
            or target == source):
                return
        layout = self.widget.layout()

        sourceIndex = layout.indexOf(source)
        sourcePos = layout.getItemPosition(sourceIndex)

        targetIndex = layout.indexOf(target)
        targetPos = layout.getItemPosition(targetIndex)
        
        layout.addWidget(source, *targetPos)
        layout.addWidget(target, *sourcePos)

        e.accept()

考虑到这是一个非常简单的实现:您还应该确保小部件实际上是同一个 window 的子项并且处于相同的布局中。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM