Почему стиль Matplotlib.rcParams.update не обновляет цвет лица PyQt5? - PullRequest
0 голосов
/ 24 августа 2018

Я использую стили matplotlib и пытаюсь динамически менять стили, когда стиль выбирается в выпадающем списке. Я использую Matplotlib 2.2.3, Python 3.6.6, PyQt5, Windows 10. Но когда я выбираю стиль dark_background, цвет лица фигуры и цвет лица осей не меняются. Вот анимация: dark_bakground error Это код:

Файл IHMDrawDates.py, созданный с помощью pyuic5:

from PyQt5 import QtCore, QtGui, QtWidgets

class Ui_MplMainWindow(object):
    def setupUi(self, MplMainWindow):
        MplMainWindow.setObjectName("MplMainWindow")
        MplMainWindow.resize(628, 416)
        self.centralwidget = QtWidgets.QWidget(MplMainWindow)
        self.centralwidget.setObjectName("centralwidget")
        self.gridLayout_2 = QtWidgets.QGridLayout(self.centralwidget)
        self.gridLayout_2.setObjectName("gridLayout_2")
        self.mpl = MplWidgetTest(self.centralwidget)
        sizePolicy = QtWidgets.QSizePolicy(QtWidgets.QSizePolicy.Expanding, QtWidgets.QSizePolicy.Expanding)
        sizePolicy.setHorizontalStretch(0)
        sizePolicy.setVerticalStretch(0)
        sizePolicy.setHeightForWidth(self.mpl.sizePolicy().hasHeightForWidth())
        self.mpl.setSizePolicy(sizePolicy)
        self.mpl.setObjectName("mpl")
        self.gridLayout_2.addWidget(self.mpl, 0, 0, 1, 1)
        self.groupBox = QtWidgets.QGroupBox(self.centralwidget)
        self.groupBox.setMaximumSize(QtCore.QSize(95, 16777215))
        self.groupBox.setObjectName("groupBox")
        self.gridLayout = QtWidgets.QGridLayout(self.groupBox)
        self.gridLayout.setObjectName("gridLayout")
        self.buttonDrawDate = QtWidgets.QPushButton(self.groupBox)
        self.buttonDrawDate.setMaximumSize(QtCore.QSize(75, 16777215))
        self.buttonDrawDate.setObjectName("buttonDrawDate")
        self.gridLayout.addWidget(self.buttonDrawDate, 1, 0, 1, 1)
        self.buttonErase = QtWidgets.QPushButton(self.groupBox)
        self.buttonErase.setMaximumSize(QtCore.QSize(75, 16777215))
        self.buttonErase.setObjectName("buttonErase")
        self.gridLayout.addWidget(self.buttonErase, 2, 0, 1, 1)
        spacerItem = QtWidgets.QSpacerItem(20, 40, QtWidgets.QSizePolicy.Minimum, QtWidgets.QSizePolicy.Expanding)
        self.gridLayout.addItem(spacerItem, 3, 0, 1, 1)
        self.comboTema = QtWidgets.QComboBox(self.groupBox)
        self.comboTema.setObjectName("comboTema")
        self.gridLayout.addWidget(self.comboTema, 0, 0, 1, 1)
        self.gridLayout_2.addWidget(self.groupBox, 0, 1, 1, 1)
        MplMainWindow.setCentralWidget(self.centralwidget)
        self.menubar = QtWidgets.QMenuBar(MplMainWindow)
        self.menubar.setGeometry(QtCore.QRect(0, 0, 628, 21))
        self.menubar.setObjectName("menubar")
        MplMainWindow.setMenuBar(self.menubar)
        self.statusbar = QtWidgets.QStatusBar(MplMainWindow)
        self.statusbar.setObjectName("statusbar")
        MplMainWindow.setStatusBar(self.statusbar)

        self.retranslateUi(MplMainWindow)
        QtCore.QMetaObject.connectSlotsByName(MplMainWindow)

    def retranslateUi(self, MplMainWindow):
        _translate = QtCore.QCoreApplication.translate
        MplMainWindow.setWindowTitle(_translate("MplMainWindow", "MainWindow"))
        self.groupBox.setTitle(_translate("MplMainWindow", "GroupBox"))
        self.buttonDrawDate.setText(_translate("MplMainWindow", "Draw"))
        self.buttonErase.setText(_translate("MplMainWindow", "Erase"))

from mplwidgettest import MplWidgetTest

mplwidgettest.py файл, который содержит метод setTema для обновления стиля matplotlib. Когда я печатаю переменную rcParams, она содержит axes.facecolor: black, но она не применяется

from PyQt5.QtWidgets import QSizePolicy, QWidget, QVBoxLayout
from matplotlib.figure import Figure
from matplotlib.backends.backend_qt5agg import (
    FigureCanvasQTAgg as FigureCanvas,
    NavigationToolbar2QT as NavigationToolbar)
import matplotlib as mplib


class MplCanvas(FigureCanvas):
    """Class to represent the FigureCanvas widget"""
    def __init__(self):
        # setup Matplotlib Figure and Axis
        mplib.rcParams.update(mplib.rcParamsDefault)
        mplib.style.use('bmh')
        self.fig = Figure()
        self.ax = self.fig.add_subplot(111)
        # initialization of the canvas
        FigureCanvas.__init__(self, self.fig)
        # we define the widget as expandable
        FigureCanvas.setSizePolicy(self,
                                   QSizePolicy.Expanding,
                                   QSizePolicy.Expanding)
        # notify the system of updated policy
        FigureCanvas.updateGeometry(self)

    def setTema(self, tema='classic'):
        print(tema)
        mplib.style.use(tema)
        print(mplib.rcParams)
        mplib.rcParams.update(mplib.rcParams)


class MplWidgetTest(QWidget):
    """Widget defined in Qt Designer"""
    def __init__(self, parent=None):
        # initialization of Qt MainWindow widget
        QWidget.__init__(self, parent)
        # set the canvas to the Matplotlib widget
        self.canvas = MplCanvas()
        # create a NavigatioToolbar
        self.ntb = NavigationToolbar(self.canvas, self)
        # create a vertical box layout
        self.vbl = QVBoxLayout()
        # add mpl widget to vertical box
        self.vbl.addWidget(self.canvas)
        # add NavigationToolBar to vertical box
        self.vbl.addWidget(self.ntb)
        # set the layout to th vertical box
        self.setLayout(self.vbl)

mainMplWidget.py Файл, который вызывает два предыдущих файла и содержит метод 'cambiarTema', который изменяет стиль matplotlib

import sys
from IHMDrawDates import Ui_MplMainWindow
from PyQt5.QtWidgets import QMainWindow, QApplication
from PyQt5.QtGui import QCursor
import numpy as np


class DesignerMainWindow(QMainWindow, Ui_MplMainWindow):

    def __init__(self, parent=None):
        super(DesignerMainWindow, self).__init__(parent)
        self.setupUi(self)
        self.posX = []
        self.posY = []
        temas = ['bmh', 'classic', 'grayscale', 'seaborn-bright',
                 'ggplot', 'dark_background']
        self.comboTema.addItems(temas)
        # connect the signals with the slots
        self.buttonDrawDate.clicked.connect(self.drawDate)
        self.buttonErase.clicked.connect(self.eraseDate)
        self.comboTema.currentIndexChanged.connect(self.cambiarTema)

    def cambiarTema(self):
        tema = self.comboTema.currentText()
        self.mpl.canvas.setTema(str(tema))
        self.mpl.canvas.ax.clear()
        self.mpl.canvas.draw()
        self.mpl.canvas.flush_events()

    def drawDate(self):
        x = np.arange(0, 100, 0.1)
        y = np.sin(x)
        self.mpl.canvas.ax.plot(x, y)
        self.mpl.canvas.ax.relim()
        self.mpl.canvas.ax.autoscale(True)
        self.mpl.ntb.update()
        self.mpl.ntb.push_current()
        self.mpl.canvas.draw()

    def eraseDate(self):
        self.mpl.canvas.ax.clear()
        self.mpl.ntb.update()
        self.mpl.ntb.push_current()
        self.mpl.canvas.draw()

if __name__ == '__main__':
    app = 0
    app = QApplication(sys.argv)
    dmw = DesignerMainWindow()
    # show it
    dmw.show()
    sys.exit(app.exec_())

Ответы [ 2 ]

0 голосов
/ 31 августа 2018

Ниже объясняется, почему установка нового стиля не приведет к обновлению существующего рисунка.

matplotlib.rcParams - это словарь, в котором хранятся определенные параметры по умолчанию, которые должны использоваться при создании объектов в matplotlib. rcParams.update обновит этот словарь; update - это метод объекта python dict. matplotlib.style.use - это ярлык для обновления rcParams, он загружает соответствующие параметры из файла или словаря.

Эти параметры затем используются при создании нового объекта. В упрощенном виде это будет выглядеть как

def create_object(arg1, parameter1=None):
    if not parameter1:
        # use the default from the rcParams
        parameter1 = rcParams["parameter1"]
    obj = MatplotlibObject(arg1, parameter1=parameter1)
    return obj

Такая функция используется при создании объектов. Однако после создания такого объекта он не будет запускаться снова, если что-то в rcParams изменится.

obj = create_object(1)
rcParams.update({"parameter1" : "New Value"})
# at this point, obj would not know about the new parameter1 value

Как уже отмечалось, есть два варианта.

  • Изменить параметр вручную . Вы можете изменить параметр вручную после создания объекта.

    obj = create_object(1)
    obj.set_parameter1("New Value")
    
  • Создайте заново объект. Вы можете удалить существующий объект, обновить rcParams и воссоздать объект так, чтобы он использовал новое значение.

    obj = create_object(1)
    del obj
    rcParams.update({"parameter1" : "New Value"})
    obj = create_object(1)
    
0 голосов
/ 31 августа 2018

Единственное решение, которое я нашел до сих пор, это добавить эти две строки в метод 'setTema' файла mplwidgettest.py

self.fig.set_facecolor(mplib.rcParams['figure.facecolor'])
self.ax.set_facecolor(mplib.rcParams['axes.facecolor'])

Полный класс:

class MplCanvas(FigureCanvas):
    """Class to represent the FigureCanvas widget"""
    def __init__(self):
        # setup Matplotlib Figure and Axis
        mplib.rcParams.update(mplib.rcParamsDefault)
        mplib.style.use('bmh')
        self.fig = Figure()
        self.ax = self.fig.add_subplot(111)
        # initialization of the canvas
        FigureCanvas.__init__(self, self.fig)
        # we define the widget as expandable
        FigureCanvas.setSizePolicy(self,
                                   QSizePolicy.Expanding,
                                   QSizePolicy.Expanding)
        # notify the system of updated policy
        FigureCanvas.updateGeometry(self)

    def setTema(self, tema='classic'):
        mplib.rcParams.update(mplib.rcParamsDefault)
        mplib.style.use(tema)
        self.fig.set_facecolor(mplib.rcParams['figure.facecolor'])
        self.ax.set_facecolor(mplib.rcParams['axes.facecolor'])

Анимация результата: matplot dark style

Однако я не думаю, что каждый из атрибутов фигуры должен быть изменен, если они должны быть определены в стиле. Благословенный матплотлиб: - (

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...