pyqtgraph 自定义缩放问题

Posted

技术标签:

【中文标题】pyqtgraph 自定义缩放问题【英文标题】:pyqtgraph custom scaling issue 【发布时间】:2017-02-03 14:41:28 【问题描述】:

我使用 pyqtgraph 库进行绘图。我真的很喜欢情节上的鼠标交互(缩放,平移,...)。

对于我的一些情节,我想在滚动鼠标滚轮时更改缩放行为。标准实现是同时在 x 和 y 方向上进行缩放。在这些图上沿 x 方向缩放没有意义,所以我想禁用它。我尝试了以下方法:

###################################################################
#                                                                 #
#                     PLOTTING A LIVE GRAPH                       #
#                  ----------------------------                   #
#                                                                 #
###################################################################

import sys
import os
from PyQt4 import QtGui
from PyQt4 import QtCore
import pyqtgraph as pg
import numpy as np

# Override the pg.ViewBox class to add custom
# implementations to the wheelEvent
class CustomViewBox(pg.ViewBox):
    def __init__(self, *args, **kwds):
        pg.ViewBox.__init__(self, *args, **kwds)
        #self.setMouseMode(self.RectMode)


    def wheelEvent(self, ev, axis=None):
        # 1. Pass on the wheelevent to the superclass, such
        #    that the standard zoomoperation can be executed.
        pg.ViewBox.wheelEvent(ev,axis)

        # 2. Reset the x-axis to its original limits
        #
        # [code not yet written]
        #

class CustomMainWindow(QtGui.QMainWindow):

    def __init__(self):

        super(CustomMainWindow, self).__init__()

        # 1. Define look and feel of this window
        self.setGeometry(300, 300, 800, 400)
        self.setWindowTitle("pyqtgraph example")

        self.FRAME_A = QtGui.QFrame(self)
        self.FRAME_A.setStyleSheet("QWidget  background-color: %s " % QtGui.QColor(210,210,235,255).name())
        self.LAYOUT_A = QtGui.QHBoxLayout()
        self.FRAME_A.setLayout(self.LAYOUT_A)
        self.setCentralWidget(self.FRAME_A)

        # 2. Create the PlotWidget(QGraphicsView)
        # ----------------------------------------
        self.vb = CustomViewBox()
        self.plotWidget = pg.PlotWidget(viewBox=self.vb, name='myPlotWidget')
        self.LAYOUT_A.addWidget(self.plotWidget)
        self.plotWidget.setLabel('left', 'Value', units='V')
        self.plotWidget.setLabel('bottom', 'Time', units='s')
        self.plotWidget.setXRange(0, 10)
        self.plotWidget.setYRange(0, 100)

        # 3. Get the PlotItem from the PlotWidget
        # ----------------------------------------
        self.plotItem = self.plotWidget.getPlotItem()

        # 4. Get the PlotDataItem from the PlotItem
        # ------------------------------------------
        # The plot() function adds a new plot and returns it.
        # The function can be called on self.plotWidget or self.plotItem
        self.plotDataItem = self.plotItem.plot()
        self.plotDataItem.setPen((255, 240, 240))
        self.plotDataItem.setShadowPen(pg.mkPen((70, 70, 30), width=2, cosmetic=True))


        # 5. Create the x and y arrays
        # -----------------------------
        n = np.linspace(0, 499, 500)
        self.y = 50 + 5 * (np.sin(n / 8.3)) + 7 * (np.sin(n / 7.5)) - 5 * (np.sin(n / 1.5))
        self.x = 10 * n / len(n)
        self.plotDataItem.setData(x=self.x, y=self.y)

        self.show()


if __name__== '__main__':
    app = QtGui.QApplication(sys.argv)
    QtGui.QApplication.setStyle(QtGui.QStyleFactory.create('Plastique'))
    myGUI = CustomMainWindow()


    sys.exit(app.exec_())

只需将此代码复制粘贴到一个新的 Python 文件中,您应该会得到以下输出:

不幸的是,每次 mouseWheel 事件都会弹出一条错误消息:

Traceback (most recent call last):
  File "pyTest.py", line 26, in wheelEvent
    pg.ViewBox.wheelEvent(ev,axis)
  File "C:\Anaconda3\lib\site-packages\pyqtgraph\graphicsItems\ViewBox\ViewBox.py", line 1206, in wheelEvent
    mask = np.array(self.state['mouseEnabled'], dtype=np.float)
AttributeError: 'QGraphicsSceneWheelEvent' object has no attribute 'state'

我的系统如下:

Python 版本:3.5.2 PyQt 版本:4.11.4 Qt 版本:4.8.7 pyqtgraph 版本:0.9.10

【问题讨论】:

【参考方案1】:

我的同事指出,在覆盖wheelEvent 函数时,我必须添加self 作为第一个参数:

# Override the pg.ViewBox class to add custom
# implementations to the wheelEvent
class CustomViewBox(pg.ViewBox):
    def __init__(self, *args, **kwds):
        pg.ViewBox.__init__(self, *args, **kwds)
        #self.setMouseMode(self.RectMode)


    def wheelEvent(self, ev, axis=None):
        print(str(self.viewRange()))

        # 1. Pass on the wheelevent to the superclass, such
        #    that the standard zoomoperation can be executed.
        pg.ViewBox.wheelEvent(self,ev,axis) # <- To override the function
                                                 properly, one should add
                                                 'self' as first argument

        # 2. Reset the x-axis to its original limits
        self.setXRange(0,10)

现在可以了。但唯一的缺点是以下代码行:

    # 2. Reset the x-axis to its original limits
    self.setXRange(0,10)

这样做会更好:

def wheelEvent(self, ev, axis=None):
    # 1. Determine initial x-range
    initialRange = self.viewRange()

    # 2. Call the superclass method for zooming in
    pg.ViewBox.wheelEvent(self,ev,axis)

    # 3. Reset the x-axis to its original limits
    self.setXRange(initialRange[0][0],initialRange[0][1])

问题在于函数 self.viewRange() 不返回 [0,10] 而是返回 [-0.37, 10.37]。 viewBox 在左侧和右侧添加了一些边距。如果你继续这样做,最终这些利润会随着时间的推移而漂移: [-0.37, 10.37] -&gt; [-0.74, 10.74] -&gt; ...

【讨论】:

【参考方案2】:

您可以简单地禁用 x 轴上的鼠标行为,而不是使用自己的鼠标事件处理程序创建自定义 ViewBox,使用:

self.plotWidget.setMouseEnabled(x=False)

【讨论】:

以上是关于pyqtgraph 自定义缩放问题的主要内容,如果未能解决你的问题,请参考以下文章

pyqtgraph中带有箭头的自定义LinearRegionItem

传单:是不是可以自定义缩放级别?

如何缩放视图并将自定义边框保持在原始大小\缩放?

自定义视图在缩放后重绘时如何防止“反弹”效果?

Android Maps Library V2 缩放控件自定义位置

Matplotlib imshow与自定义色彩映射(例如自定义地图和自定义缩放)