首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >获取拖放目标pyqt5

获取拖放目标pyqt5
EN

Stack Overflow用户
提问于 2020-08-31 18:30:32
回答 1查看 449关注 0票数 0

我正在使用pyqt5中的拖放GUI,并试图获取拖放操作的目标小部件,但是当我尝试QDrag对象的QDrag()函数时,我在0x0000025FDAC09EE0>返回了0x0000025FDAC09EE0>对象,我不知道如何使用它。我希望访问QGridLayout中的小部件的索引,以便使这两个小部件交换位置。

这是我的代码:

代码语言:javascript
运行
复制
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()
EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2020-08-31 21:46:19

drop事件的目标始终是接收drop操作的小部件,因此很明显,如果从主窗口实例拦截事件,那么主窗口就会成为目标。如果需要在特定位置找到小部件,则需要使用QApplication.widgetAt(pos)

在下面的示例中,从给定的代码中修改,只有当源是Button实例且目标不相同时,我才接受dragEnter/dragMove事件。然后我用它们在布局中的位置来切换这些按钮。

代码语言:javascript
运行
复制
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()

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

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/63676300

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档