0

我想模拟自定义小部件的 QDockWidget/QToolBar 行为,mousePressEvent从一个位置取消固定小部件并将其固定到另一个位置。就像将上框架放置到“子”位置:

在此处输入图像描述

我可以使用以下代码取消固定它:

class DraggableSpectralFrame(SpectralFrame):
    def __init__(self, parent=None):
        super().__init__(parent)
        self.setMouseTracking(True)
        self.setWindowFlags(Qt.Window | Qt.FramelessWindowHint)
        self.oldPos = QPointF(self.pos())
        self.local= QPointF(self.pos())
        self.floating = False

    def mousePressEvent(self, event):
        self.oldPos = event.globalPos()
        if self.hoverRect.contains(event.localPos()) and \
                event.button() == Qt.LeftButton:
            self.floating = True
            self.local = event.localPos()
            self.setParent(None)
            self.show()
            self.move((self.oldPos - self.local).toPoint())


    def mouseMoveEvent(self, event):
        self.oldPos = event.globalPos()
        if self.floating:
            self.move((self.oldPos - self.local).toPoint())

在此处输入图像描述

变形为窗口并以正确的方式移动窗口后我失去按下的功能的问题DraggableSpectralFrame我必须再次单击/按下。没有额外的点击窗口仍然会对光标移动做出反应(由于self.floatingTrue),但很容易失去对急剧移动的关注。

可能有更简单的方法来实现这个想法,如果是这样的话 - 请告诉我!

4

1 回答 1

2

实现拖放的一种方法是使用QDrag对象在拖动时显示小部件。对于应该接受拖动小部件的小部件,您需要设置为 True并acceptDrops至少重新实现. 这是一个非常简单的例子,如何做到这一点。在这个例子中,“Move me”标签可以在两个窗口之间来回拖动。dragEnterEventdropEvent

from PyQt5 import QtWidgets, QtCore, QtGui


class DropFrame(QtWidgets.QWidget):
    def __init__(self, parent=None):
        super().__init__(parent)
        self.vlayout = QtWidgets.QVBoxLayout(self)
        self.setAcceptDrops(True)

    def dragEnterEvent(self, event):
        if isinstance(event.source(), DraggableLabel):
            event.accept()

    def dropEvent(self, event):
        event.accept()
        widget = event.source()
        if widget:
            self.vlayout.addWidget(widget)


class DraggableLabel(QtWidgets.QLabel):
    def __init__(self, text, parent=None):
        super().__init__(text, parent)
        self.setFrameShape(self.Box)

    def mousePressEvent(self, event):
        self.drag = QtGui.QDrag(self)
        self.mime_data = QtCore.QMimeData()
        self.drag.setMimeData(self.mime_data)
        # capture image of self to use as pixmap while dragging
        self.pixmap = self.grab()
        self.drag.setPixmap(self.pixmap)

        self.hide()
        self.drag.exec(QtCore.Qt.MoveAction)
        self.show()


if __name__ == "__main__":
    app = QtWidgets.QApplication([])
    widget1 = DropFrame()
    label = DraggableLabel('Move me')
    widget1.layout().addWidget(label)
    widget1.show()

    widget2 = DropFrame()
    widget2.show()
    widget2.move(widget1.pos().x(),widget1.pos().y()+150)
    app.exec()
于 2021-02-22T07:47:01.343 回答