繁体   English   中英

如何覆盖pyqt4 QGraphicsView.mousePressEvent?

[英]How can I override a pyqt4 QGraphicsView.mousePressEvent?

我正在尝试在pyqt4中调整照片查看器(我从本网站上的另一个问题获得了代码)。 默认情况下,鼠标滚轮会缩放,鼠标左键用于平移/拖动(不知道究竟要调用的内容)图像。 我想在左键单击图像时添加获取像素信息的功能,因此我打算使用一个按钮将“平移”模式更改为“像素信息”模式。 我尝试这样做的方法如下,但它没有按照我的意图行事。

当我使用第一个GUI按钮加载图像并放大图像时,如果我不平移/拖动图像,则单击第二个GUI按钮将禁用平移/拖动,当我单击图像时,提供像素我点击的像素的信息。 这是我的预期。

但是,如果我放大然后平移/拖动图像,当我单击第二个GUI按钮时,它不再禁用平移/拖动并且不打印像素信息。

由于某种原因,我不明白(我希望有人可以启发我)在缩放后拖动图像似乎锁定了平移/拖动方法而不允许我更改它。 谁能告诉我为什么。

from PyQt4 import QtCore, QtGui

class PhotoViewer(QtGui.QGraphicsView):
    def __init__(self, parent):
        super(PhotoViewer, self).__init__(parent)
        self._zoom = 0
        self._scene = QtGui.QGraphicsScene(self)
        self._photo = QtGui.QGraphicsPixmapItem()
        self._scene.addItem(self._photo)
        self.setScene(self._scene)
        self.setTransformationAnchor(QtGui.QGraphicsView.AnchorUnderMouse)
        self.setResizeAnchor(QtGui.QGraphicsView.AnchorUnderMouse)
        self.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff)
        self.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff)
        self.setBackgroundBrush(QtGui.QBrush(QtGui.QColor(30, 30, 30)))
        self.setFrameShape(QtGui.QFrame.NoFrame)

    def fitInView(self):
        rect = QtCore.QRectF(self._photo.pixmap().rect())
        if not rect.isNull():
            unity = self.transform().mapRect(QtCore.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.centerOn(rect.center())
            self._zoom = 0

    def setPhoto(self, pixmap=None):
        self._zoom = 0
        if pixmap and not pixmap.isNull():
            self.setDragMode(QtGui.QGraphicsView.ScrollHandDrag)
            self._photo.setPixmap(pixmap)
            self.fitInView()
        else:
            self.setDragMode(QtGui.QGraphicsView.NoDrag)
            self._photo.setPixmap(QtGui.QPixmap())

    def wheelEvent(self, event):
        if not self._photo.pixmap().isNull():
            if event.delta() > 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 printPixInfo(self,event):
        print event.pos()

class Window(QtGui.QWidget):
    def __init__(self):
        super(Window, self).__init__()
        self.viewer = PhotoViewer(self)
        # 'Load image' button
        self.btnLoad = QtGui.QToolButton(self)
        self.btnLoad.setText('Load image')
        self.btnLoad.clicked.connect(self.loadImage)
        # Button to change from drag/pan to getting pixel info
        self.btnPixInfo = QtGui.QToolButton(self)
        self.btnPixInfo.setText('Enter pixel info mode')
        self.btnPixInfo.clicked.connect(self.pixInfo)
        # Arrange layout
        VBlayout = QtGui.QVBoxLayout(self)
        VBlayout.addWidget(self.viewer)
        HBlayout = QtGui.QHBoxLayout()
        HBlayout.setAlignment(QtCore.Qt.AlignLeft)
        HBlayout.addWidget(self.btnLoad)
        HBlayout.addWidget(self.btnPixInfo)
        VBlayout.addLayout(HBlayout)

    def loadImage(self):
        self.viewer.setPhoto(QtGui.QPixmap('pic.jpg'))

    def pixInfo(self):
        print 'Now in pixel info mode'
        self.viewer._photo.mousePressEvent = self.viewer.printPixInfo

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

我还想知道,一旦我成功更改了鼠标单击方法,返回鼠标左键单击返回默认的平移/拖动行为的语法是什么。

我是一个python新手,所以尽量不要批评我太多。 谢谢。

问题是,通过完全覆盖mousePressEvent您将丢失所有默认行为(例如,使用鼠标进行平移)。 最好使用自定义信号通知何时单击照片。 由于图形视图已经有用于更改拖动模式的API,因此您可以添加用于切换模式的插槽:

class PhotoViewer(QtGui.QGraphicsView):
    photoClicked = QtCore.pyqtSignal(QtCore.QPoint)    
    ...

    def toggleDragMode(self):
        if self.dragMode() == QtGui.QGraphicsView.ScrollHandDrag:
            self.setDragMode(QtGui.QGraphicsView.NoDrag)
        elif not self._photo.pixmap().isNull():
            self.setDragMode(QtGui.QGraphicsView.ScrollHandDrag)

    def mousePressEvent(self, event):
        if self._photo.isUnderMouse():
            self.photoClicked.emit(QtCore.QPoint(event.pos()))  
        # keep the default behaviour
        super(PhotoViewer, self).mousePressEvent(event)

然后可以使用这两个新API为您提供所需的功能:

class Window(QtGui.QWidget):
    def __init__(self):
        ...
        # add a line-edit to display the position info
        self.editPixInfo = QtGui.QLineEdit(self)
        self.editPixInfo.setReadOnly(True)
        self.viewer.photoClicked.connect(self.photoClicked)
        ...
        HBlayout.addWidget(self.editPixInfo)

    def pixInfo(self):
        # switch modes
        self.viewer.toggleDragMode()

    def photoClicked(self, pos):
        # handle photo-click signals
        if self.viewer.dragMode() == QtGui.QGraphicsView.NoDrag:
            self.editPixInfo.setText('%d, %d' % (pos.x(), pos.y()))

暂无
暂无

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

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