在 PyQt 中自定义 MPL

Posted

技术标签:

【中文标题】在 PyQt 中自定义 MPL【英文标题】:Customising MPL in PyQt 【发布时间】:2016-09-30 08:13:08 【问题描述】:

所以我一直在对 PyQt 和 MPL 进行大量研究,但我仍然遇到一些我无法解决的问题。

第一个是这个:如何改变FigureCanvas背景色(就是this image中的灰色部分)。 FigureCanvas 没有 facecolor 或 backgroundcolor 或我能想到或找到的任何其他属性。我可以改变坐标轴、绘图的颜色(这就是我得到你可以看到的黑色背景的方式),但不能改变画布本身。

第二个问题:我怎样才能摆脱在十字准线顶部仍然可见的鼠标指针?或者更好的是在光标进入 mpl FigureCanvas 小部件时更改光标(然后在离开时再次返回)。

编辑:这是我要求的一些代码。我通过以下方式从 Qt 调用 MPL 数字。

我的主窗口:

from PyQt5.QtCore import pyqtSlot as Slot
from PyQt5 import QtCore
from PyQt5 import QtWidgets
from PyQt5 import QtGui
from PyQt5 import uic

# MPL class
from pyqt5_mpl import mpl_figure

# Select Qt5 user interface.
qtCreatorFile = '<local filepath>'
Ui_MainWindow, QtBaseClass = uic.loadUiType(qtCreatorFile)

class main(QtWidgets.QMainWindow, Ui_MainWindow):

    def __init__(self):
        # Inititate UI.
        QtWidgets.QMainWindow.__init__(self)
        Ui_MainWindow.__init__(self)
        self.setupUi(self)

        # Create data structures.
        global a
        a = a_dataclass()

        # Create MPL Canvases (Layout->CreateMPLCanvas->AddWidget).
        # Create class instances for each figure.
        a.im1 = mpl_figure()
        a.im2 = mpl_figure()

        # Create Layouts and add widgets.
        a_layout = QtWidgets.QHBoxLayout(self.tab1)
        a_layout.addWidget(a.im1.canvas)
        a_layout.addWidget(a.im2.canvas)

        # Load images.
        a.im1.loadimage('image file path',a.pix)
        a.im2.loadimage('image file path',a.pix)
        # a.pix is dimensions for extent in plot.

if __name__ == "__main__":
    # QApp 
    app = QtWidgets.QApplication(sys.argv)
    # QWidget (MainWindow)
    window = main()
    window.show()
    sys.exit(app.exec_())

我的 mpl_figure 课:

import matplotlib as mpl
mpl.use('Qt5Agg')
import matplotlib.pyplot as plt
import numpy as np
from matplotlib.backends.backend_qt5agg import FigureCanvasQTAgg as FigureCanvas

class mpl_figure:

    def __init__(self):
        # Set up empty plot variables.
        self.image = None
        # Create a figure to plot on.
        fig = plt.figure()
        fig.set_facecolor('black') # This does nothing...
        # Create an axis in the figure.
        self.ax = fig.add_subplot(111, axisbg='black')
        # Create a canvas widget for Qt to use.
        self.canvas = FigureCanvas(fig)
        # Create cursor widget.
        self.cursor = mpl.widgets.Cursor(self.ax)
        # Refresh the canvas.
        self.canvas.draw()

    def loadimage(self,fn,pix):
        # Read in image.
        self.data = plt.imread(fn)
        # Calculate the extent.
        self.dims = np.array([0, pix[0]*np.shape(self.data)[1],0,pix[1]*np.shape(self.data)[0]])
        # Display the image.
        self.image = self.ax.imshow(self.data, cmap='gray', extent=self.dims)
        self.ax.set_autoscale_on(False)
        # Refresh the canvas.
        self.canvas.draw()
        # Start Callback ID
        self.cid = self.canvas.mpl_connect('button_press_event', self.onclick)

    def onclick(self,event):
        # Do stuff...
        pass

我尝试对 self.canvas 或 fig 进行更改的所有尝试。就其样式而言,它被渲染为静音,它什么也不做。至于在输入 mpl_widget 时更改光标(或隐藏它,它是指针),我只是不确定如何去做。我记得在 mpl 中看到了连接“event_leave_figure”的效果,但无法在它和 PyQt 之间建立联系

【问题讨论】:

请贴一些代码。 @HYRY 我已经添加了我的代码的 jist(去掉了很多不必要的东西)。如果您需要更多,请告诉我,希望对您有所帮助。 mpl_figure 不是 QtWidget 的子类,我认为mpl_layout.addWidget(mpl_widget) 会引发异常。 @HYRY 来自“matplotlib.backends.backend_qt5agg import FigureCanvasQTAgg as FigureCanvas”。它选择 Qt 作为 MPL 的后端,使其成为 QtWidget。 我无法运行代码。我认为您可以使用fig.set_facecolor() 更改背景颜色。您可以使用fig.canvas.mpl_connect() 将回调函数连接到“axes_enter_event”、“axes_leave_event”事件以隐藏和显示光标。 【参考方案1】:

感谢您提供MINIMAL WORKING example 并在提问前进行谷歌搜索。这总是可以节省很多时间。

所以下面的代码生成如下图。

import sys
from PyQt4 import QtGui, QtCore
import matplotlib 
import numpy as np
from matplotlib.backends.backend_qt4agg import FigureCanvasQTAgg as FigureCanvas
from matplotlib.figure import Figure


class ApplicationWindow(QtGui.QMainWindow):
    def __init__(self):
        QtGui.QMainWindow.__init__(self)
        self.setAttribute(QtCore.Qt.WA_DeleteOnClose)
        self.setWindowTitle("application main window")

        self.main_widget = QtGui.QWidget(self)
        self.setCentralWidget(self.main_widget)
        l = QtGui.QVBoxLayout(self.main_widget)

        self.fig = Figure(figsize=(5,3), dpi=100)
        self.ax = self.fig.add_subplot(111)
        self.canvas = FigureCanvas(self.fig)
        l.addWidget(self.canvas)

        t = np.arange(0.0, 3.0, 0.01)
        s = np.sin(2*np.pi*t)*np.exp(-t/7/np.pi)
        self.ax.plot(t, s, color="#0095a2", lw=3)

        #set some alpha
        self.fig.patch.set_alpha(0.5)
        #set color
        self.fig.patch.set_facecolor('#cd0e49')

        self.cursor = matplotlib.widgets.Cursor(self.ax)
        #set cursor
        #self.canvas.setCursor(QtGui.QCursor(QtCore.Qt.BlankCursor))
        #since I find Blank cursor very confusing, I will use a better one:
        self.canvas.setCursor(QtGui.QCursor(    QtCore.Qt.SizeAllCursor))



qApp = QtGui.QApplication(sys.argv)

aw = ApplicationWindow()
aw.show()
sys.exit(qApp.exec_())

【讨论】:

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

如何在 PyQt 中自定义 QCompleter 弹出窗口?

PyQt4自定义控件----导航栏控件

如何在Android中自定义动画

Google Colaboratory matplotlib 图表中的自定义字体

如何在 JavaScript 中自定义警报?

如何在 BlackBerry 中自定义 ListField?