python - Matplotlib 导航工具栏嵌入 pyqt5 窗口 - 重置原始 View 崩溃

标签 python matplotlib navigation pyqt5 interactive

我正在尝试使用嵌入到 pyqt5 窗口中的 matplotlib 的交互式绘图功能,mpl 版本为:2.1.1

我在这里提供了 UI 文件的最小示例:

https://ufile.io/8irs7

和Python代码:

import matplotlib
matplotlib.use("Qt5Agg")
import sys

from matplotlib.backends.backend_qt5agg import FigureCanvasQTAgg as FigureCanvas
from matplotlib.backends.backend_qt5agg import NavigationToolbar2QT as NavigationToolbar
from matplotlib.figure import Figure

import pandas as pd
import numpy as np

from PyQt5 import  QtWidgets, uic  # works for pyqt5
from PyQt5.Qt import  QVBoxLayout


# import numpy as np
qtCreatorFile = "window.ui" # Enter file here.
# qtCreatorFile = "pssga.ui"


Ui_MainWindow, QtBaseClass = uic.loadUiType(qtCreatorFile)

class PlotSeries(QtWidgets.QMainWindow, Ui_MainWindow):

    def __init__(self):
        QtWidgets.QMainWindow.__init__(self)
        Ui_MainWindow.__init__(self)
        self.setupUi(self)

                #create figure
        self.fig=Figure()

        self.axes = self.fig.add_subplot(111)
        self.axes.grid()



        self.canvas=FigureCanvas(self.fig)
        #add plot toolbar from matplotlib
        self.toolbar = NavigationToolbar(self.canvas, self)

        #create new layout
        layout = QVBoxLayout()

        layout.addWidget(self.canvas)
        layout.addWidget(self.toolbar)

        #add layout to qwidget
        self.plotlayout=self.widgetplot.setLayout(layout)

        #draw the canvas ! 
        self.canvas.draw()
        #now let's call widget 
        self.plot_basegraph()


    def plot_basegraph(self):
        #create simple time series 
        for i in range(0,5): 
            ts=pd.Series(np.random.randn(2000), index=pd.date_range('1/1/2015',freq='h', periods=2000),name="series "+str(i))
            if i==0:
                self.stackTSPD=ts.to_frame()
            else:
                self.stackTSPD=self.stackTSPD.join(ts.to_frame(),how='inner')
        print(matplotlib.__version__)

        self.axes.clear()
        self.axes.grid()

        for i in self.stackTSPD.columns.values:
            t=self.stackTSPD.index.to_pydatetime()

            y=self.stackTSPD[i].values
            self.axes.plot(t,y, label=i)


if __name__ == "__main__":
    app = QtWidgets.QApplication(sys.argv)
    window = PlotSeries()
    window.show()
    sys.exit(app.exec_())

放大时间序列,平移按预期工作,但是当我尝试按“Home”时 - 重置原始 View 它崩溃了:

Traceback (most recent call last):
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/backends/backend_qt5agg.py", line 155, in __draw_idle_agg
    self.draw()
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/backends/backend_qt5agg.py", line 127, in draw
    super(FigureCanvasQTAggBase, self).draw()
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/backends/backend_agg.py", line 430, in draw
    self.figure.draw(self.renderer)
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py", line 55, in draw_wrapper
    return draw(artist, renderer, *args, **kwargs)
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/figure.py", line 1299, in draw
    renderer, self, artists, self.suppressComposite)
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/image.py", line 138, in _draw_list_compositing_images
    a.draw(renderer)
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py", line 55, in draw_wrapper
    return draw(artist, renderer, *args, **kwargs)
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/axes/_base.py", line 2437, in draw
    mimage._draw_list_compositing_images(renderer, self, artists)
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/image.py", line 138, in _draw_list_compositing_images
    a.draw(renderer)
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/artist.py", line 55, in draw_wrapper
    return draw(artist, renderer, *args, **kwargs)
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py", line 1133, in draw
    ticks_to_draw = self._update_ticks(renderer)
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py", line 974, in _update_ticks
    tick_tups = list(self.iter_ticks())
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/axis.py", line 917, in iter_ticks
    majorLocs = self.major.locator()
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/dates.py", line 1061, in __call__
    self.refresh()
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/dates.py", line 1081, in refresh
    dmin, dmax = self.viewlim_to_dt()
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/dates.py", line 839, in viewlim_to_dt
    return num2date(vmin, self.tz), num2date(vmax, self.tz)
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/dates.py", line 445, in num2date
    return _from_ordinalf(x, tz)
  File "/usr/local/lib/python3.5/dist-packages/matplotlib/dates.py", line 260, in _from_ordinalf
    dt = datetime.datetime.fromordinal(ix).replace(tzinfo=UTC)
ValueError: ordinal must be >= 1

当我尝试将此时间序列图嵌入到 pyqt5 形式中时,这种行为就开始了。当绘图在新的 GTK 窗口中单独打开时,它工作没有任何问题。我附上了带有工作主页按钮的示例,该按钮在非 pyqt 窗口中打开...

'''
Created on 16. jan. 2018

@author: gregor
'''

import matplotlib as mpl
import matplotlib.pyplot as plt
# matplotlib.use("Qt5Agg")


# from matplotlib.backends.backend_qt5agg import FigureCanvasQTAgg as FigureCanvas
# from matplotlib.backends.backend_qt5agg import NavigationToolbar2QT as NavigationToolbar
# from matplotlib.figure import Figure

import pandas as pd
import numpy as np





class PlotSeries():

    def __init__(self):


        self.fig,self.axes = plt.subplots()

        self.axes.grid()



#         self.canvas=FigureCanvas(self.fig)
        #add plot toolbar from matplotlib
#         self.toolbar = NavigationToolbar(self.canvas, self)

        #create new layout
#         layout = QVBoxLayout()
#         
#         layout.addWidget(self.canvas)
#         layout.addWidget(self.toolbar)
#         
#         #add layout to qwidget
#         self.plotlayout=self.widgetplot.setLayout(layout)
#         
#         #draw the canvas ! 
#         self.canvas.draw()
        #now let's call widget 
        self.plot_basegraph()


    def plot_basegraph(self):
        #create simple time series 
        for i in range(0,5): 
            ts=pd.Series(np.random.randn(2000), index=pd.date_range('1/1/2015',freq='h', periods=2000),name="series "+str(i))
            if i==0:
                self.stackTSPD=ts.to_frame()
            else:
                self.stackTSPD=self.stackTSPD.join(ts.to_frame(),how='inner')
        print(mpl.__version__)

        self.axes.clear()
        self.axes.grid()

        for i in self.stackTSPD.columns.values:
            t=self.stackTSPD.index.to_pydatetime()

            y=self.stackTSPD[i].values
            self.axes.plot_date(t,y, label=i)
        plt.show()



if __name__ == "__main__":
 PlotSeries()  

谢谢!

最佳答案

简短回答:删除行self.canvas.draw()

说明:
您可以在 Canvas 包含任何数据之前绘制 Canvas 。但由于此时 Canvas 已经配备了工具栏,因此“Home”状态是轴没有数据的状态。

这本身并不是问题,但一旦绘制日期就会成为问题。
空轴的默认初始限制是(0,1)。因此,“Home”状态将引用这些限制,并且单击按钮后将尝试恢复这些限制。
然而与此同时,轴心已被告知要保留日期。日期范围从(1,无论什么)。因此 0 在日期轴上不是有效数字。

解决方案就很简单了。只需从代码中删除 self.canvas.draw() 行即可。无论如何,它无法完成任何所需的任务。

关于python - Matplotlib 导航工具栏嵌入 pyqt5 窗口 - 重置原始 View 崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48264553/

相关文章:

python - 为什么 matplotlib.PatchCollection 弄乱了补丁的颜色?

css - 具有动态 Logo /元素高度的水平导航中的 100% child 高度

javascript - 上一个和下一个按钮

python - "download_slot"在 scrapy 中是如何工作的

python - 有没有一种简单的方法可以在 Python 列表中表示 base64 字母表?

python - 通过特定组合将句子拆分为句子列表

python - Matplotlib 给出错误

python - 异常类

python - 将 Pandas 数据框处理成 fiddle 图

html - 导航栏,列表项之间的可见空间