【问题标题】:Change ScrollHandDrag form left click to middle click PYQT5将 ScrollHandDrag 表单左键单击更改为中键单击 PYQT5
【发布时间】:2019-09-02 16:13:51
【问题描述】:

我很好奇是否有办法将QGraphicsView.ScrollHandDrag 更改为在用户放大点时使用中心按钮而不是鼠标左键进行平移。另外有没有办法对用户可以平移的位置没有任何限制?谢谢你的帮助。我在下面附上了一些代码。

from PyQt5.QtCore import *
from PyQt5.QtGui import *
from PyQt5.QtWidgets import *

from math import sqrt

class Point(QGraphicsItem):
    def __init__(self, x, y):
        super(Point, self).__init__()
        self.setFlag(QGraphicsItem.ItemIsSelectable, True)
        self.rectF = QRectF(0, 0, 30, 30)
        self.x=x
        self.y=y
        self._brush = QBrush(Qt.black)

    def setBrush(self, brush):
        self._brush = brush
        self.update()

    def boundingRect(self):
        return self.rectF

    def paint(self, painter=None, style=None, widget=None):
        painter.fillRect(self.rectF, self._brush)

    def hoverMoveEvent(self, event):
        point = event.pos().toPoint()
        print(point)
        QGraphicsItem.hoverMoveEvent(self, event)


class PhotoViewer(QGraphicsView):
    photoClicked = pyqtSignal(QPoint)

    def __init__(self, parent):
        super(PhotoViewer, self).__init__(parent)
        self._zoom = 0
        self._empty = True
        self._scene = QGraphicsScene(self)

        self.setTransformationAnchor(QGraphicsView.AnchorUnderMouse)
        self.setResizeAnchor(QGraphicsView.AnchorUnderMouse)
        self.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
        self.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
        self.setFrameShape(QFrame.NoFrame)
        self.area = float()
        self.setPoints()

    def setItems(self):
            self.data = {'x': [-2414943.8686, -2417160.6592, -2417160.6592, -2417856.1783, -2417054.7618, -2416009.9966, -2416012.5232, -2418160.8952, -2418160.8952, -2416012.5232, -2417094.7694, -2417094.7694], 'y': [10454269.7008,
     10454147.2672, 10454147.2672, 10453285.2456, 10452556.8132, 10453240.2808, 10455255.8752, 10455183.1912, 10455183.1912, 10455255.8752, 10456212.5959, 10456212.5959]}
            maxX = max(self.data['x'])
            minX = min(self.data['x'])
            maxY = max(self.data['y'])
            minY = min(self.data['y'])
            distance = sqrt((maxX-minX)**2+(maxY-minY)**2)

            self.area = QRectF(minX, minY, distance, distance)
            for i,x in enumerate(self.data['x']):
                x = self.data['x'][i]
                y = self.data['y'][i]
                p = Point(x,y)
                p.setPos(x,y)
                self._scene.addItem(p)
            self.setScene(self._scene)



    def fitInView(self, scale=True):
        rect = QRectF(self.area)
        if not rect.isNull():
            self.setSceneRect(rect)

            unity = self.transform().mapRect(QRectF(0, 0, 1, 1))
            self.scale(1 / unity.width(), 1 / unity.height())
            viewrect = self.viewport().rect()
            scenerect = self.transform().mapRect(rect)
            factor = min(viewrect.width() / scenerect.width(),
                             viewrect.height() / scenerect.height())
            self.scale(factor, factor)
            self._zoom = 0

    def setPoints(self):
        self._zoom = 0
        self.setItems()
        self.setDragMode(QGraphicsView.ScrollHandDrag)
        self.fitInView()

    def wheelEvent(self, event):
            if event.angleDelta().y() > 0:
                factor = 1.25
                self._zoom += 1
            else:
                factor = 0.8
                self._zoom -= 1
            if self._zoom > 0:
                self.scale(factor, factor)
            elif self._zoom == 0:
                self.fitInView()
            else:
                self._zoom = 0


    def mousePressEvent(self, event):
        if event.button() == Qt.MidButton:
            #Im Sure I have to do something here.
            pass
        super(PhotoViewer, self).mousePressEvent(event)


class Window(QWidget):
    def __init__(self):
        super(Window, self).__init__()
        self.viewer = PhotoViewer(self)
        # 'Load image' button
        self.btnLoad = QToolButton(self)
        self.btnLoad.setText('Load Points')
        self.btnLoad.clicked.connect(self.loadPoints)

        VBlayout = QVBoxLayout(self)
        VBlayout.addWidget(self.viewer)
        HBlayout = QHBoxLayout()
        HBlayout.setAlignment(Qt.AlignLeft)
        HBlayout.addWidget(self.btnLoad)

        VBlayout.addLayout(HBlayout)

    def loadPoints(self):
        self.viewer.setPoints()




if __name__ == '__main__':
    import sys
    app = QApplication(sys.argv)
    window = Window()
    window.setGeometry(500, 300, 800, 600)
    window.show()
    sys.exit(app.exec_())

【问题讨论】:

    标签: python python-3.x pyqt pyqt5 qgraphicsview


    【解决方案1】:

    是的,您可以更改平移光标。

    请将您的代码部分更改为这些。

    def mousePressEvent(self, event):
    
        if event.button() == Qt.MidButton:
            self.viewport().setCursor(Qt.ClosedHandCursor)  
            self.original_event = event
            handmade_event = QMouseEvent(QEvent.MouseButtonPress,QPointF(event.pos()),Qt.LeftButton,event.buttons(),Qt.KeyboardModifiers())
            self.mousePressEvent(handmade_event)
            #Im Sure I have to do something here.
    
        super(PhotoViewer, self).mousePressEvent(event)
    def mouseReleaseEvent(self, event):
        if event.button() == Qt.MidButton:
            #for changing back to Qt.OpenHandCursor
            self.viewport().setCursor(Qt.OpenHandCursor)
            handmade_event = QMouseEvent(QEvent.MouseButtonRelease,QPointF(event.pos()),Qt.LeftButton,event.buttons(),Qt.KeyboardModifiers())
            self.mouseReleaseEvent(handmade_event)
        super(PhotoViewer, self).mouseReleaseEvent(event)
    

    说明

    用于更改QGraphicsView上的鼠标光标图片,基本上,

    QGraphicsView.vieport().setCursor()
    

    为什么Qt.LeftButton默认可以改变光标?

    我认为这是默认设置。可能Qt的开发人员预测用户希望通过按左键平移屏幕,此时光标形状为OpenHandCursor

    因此,您可能无法在 LeftButton 的情况下保持光标“打开”,反之亦然。 它嵌入在super(PhotoViewer, self).mousePressEvent(event) 我们将无法连接内容。 所以你不想让LeftMouse改变,你写return,你在这里写其他命令。

    手工活动对象

    如果你推送MidButton,你会自己创建LeftButton MouseEvent, 然后你将它传递给 mousePressEvent。 mousePressEvent 接受它,并像处理 leftButton 事件一样处理。

    如果你只把手工码放在mousePressEvent里,你将无法逃脱陷阱。所以你也把手工码放在mouseReleaseEvent中。

    完美。这取决于您在事件处理程序中编写的命令。 如果你不想让 leftButton 平移,你可以在里面做同样的事情。 也就是说你可以在LeftButtonMidButton之间切换事件!

    【讨论】:

    • 这太棒了!谢谢!谢谢你的解释!
    • 有没有办法让我可以让鼠标左键单击做一些完全不同的事情?当我尝试将Qt.LeftButton 添加到另一个if 语句时,我得到RecursionError: maximum recursion depth exceeded while calling a Python object 或者这个问题在新帖子中会更好吗?
    • @laxer 在新帖子中会更好。1.可能,这是其他单独问题的类型。2.仅在评论上很难说。
    猜你喜欢
    • 2015-10-11
    • 1970-01-01
    • 1970-01-01
    • 2017-07-31
    • 1970-01-01
    • 2019-04-12
    • 1970-01-01
    • 2015-03-03
    • 2019-05-03
    相关资源
    最近更新 更多