在 PyQt4 中使用 FigureCanvas 创建多个弹出图

Create several popup plots with FigureCanvas in PyQt4

以下代码创建了一个带有按钮和进度条的小部件。当按下 Button 且 Progressbar 达到 100% 时,显示 3 个图(Matplotlib.pyplot):

import sys
from PyQt4.QtGui import *
from PyQt4.QtCore import *
import matplotlib.pyplot as plt
import pandas as pd

class App(QMainWindow):

    def __init__(self):
        super(App, self).__init__()
        self.setGeometry(500, 300, 820, 350)
        self.setWindowTitle("Program")

        #Buttons
        btnposx = 30
        btnposy = 50

        self.btn4 = QPushButton('Load', self)
        self.btn4.move(btnposx,btnposy+220)       
        self.btn4.released.connect(self.thread)

        #ProgressBar
        self.pb = QProgressBar(self)
        self.pb.move(btnposx+150,btnposy+220)
        self.pb.resize(470,27)        

        self.show()

    @pyqtSlot(float)
    def load(self, val):
        self.pb.setValue(val)

    @pyqtSlot(object)
    def plot(self, pq):
        pq.plot(grid = 1)
        plt.show()

    def thread(self):
        self.thread_ = Thread()
        self.thread_.pb_signal.connect(self.load, Qt.QueuedConnection)
        self.thread_.plot_signal.connect(self.plot, Qt.QueuedConnection)
        self.thread_.start()

class Thread(QThread):
    pb_signal = pyqtSignal(float)
    plot_signal = pyqtSignal(object)

    def __init__(self, *args, **kwargs):
        QThread.__init__(self, *args, **kwargs)

    def __del__(self):
        self.wait()

    @pyqtSlot()    
    def run(self):       
        val = 0
        self.pb_signal.emit(20)
        l = range(50000000)
        for i in l:
            val += 1
        self.pb_signal.emit(60)
        self.pb_signal.emit(100)
        pq = pd.DataFrame(data = {'col1':[1,2,3,4,5,6], 'col2':[6,5,4,3,2,1]})
        self.plot_signal.emit(pq)
        self.plot_signal.emit(pq)
        self.plot_signal.emit(pq)        
        return

if __name__ == '__main__':
    app = QApplication(sys.argv)
    ex = App()
    ex.show()
    sys.exit(app.exec_())

如何使用 FigureCanvas 做完全相同的事情?我不想制作一个 window,因为嵌入了 3 个图,但嵌入了 3 个独立的图形。网上关于如何使用FigureCanvas的资料非常少

FigureCanvas 是一个专门的 QWidget,可以包含 matplotlib 的 Figure,在那个 Figure 中你可以画图,所以解决方案是为每个发射创建一个,此外,绘图函数有一个额外的参数接收将在其中绘制的轴,因此您必须传递在图内创建的轴:

import sys
from PyQt4.QtGui import *
from PyQt4.QtCore import *
import pandas as pd
from matplotlib.backends.backend_qt4agg import (
        FigureCanvas, NavigationToolbar2QT as NavigationToolbar)
from matplotlib.figure import Figure

class App(QMainWindow):

    def __init__(self):
        super(App, self).__init__()
        self.setGeometry(500, 300, 820, 350)
        self.setWindowTitle("Program")

        #Buttons
        btnposx = 30
        btnposy = 50

        self.btn4 = QPushButton('Load', self)
        self.btn4.move(btnposx,btnposy+220)       
        self.btn4.released.connect(self.thread)

        #ProgressBar
        self.pb = QProgressBar(self)
        self.pb.move(btnposx+150,btnposy+220)
        self.pb.resize(470,27)        
        self.canvas = []
        self.show()

    @pyqtSlot(float)
    def load(self, val):
        self.pb.setValue(val)

    @pyqtSlot(object)
    def plot(self, pq):
        cv = FigureCanvas(Figure(figsize=(5, 3)))
        ax = cv.figure.subplots()
        pq.plot(grid = 1, ax=ax)
        cv.show()
        # avoid garbage collector
        self.canvas.append(cv)

    def thread(self):
        self.thread_ = Thread()
        self.thread_.pb_signal.connect(self.load, Qt.QueuedConnection)
        self.thread_.plot_signal.connect(self.plot, Qt.QueuedConnection)
        self.thread_.start()

class Thread(QThread):
    pb_signal = pyqtSignal(float)
    plot_signal = pyqtSignal(object)

    def __del__(self):
        self.wait()

    @pyqtSlot()    
    def run(self):       
        val = 0
        self.pb_signal.emit(20)
        l = range(50000000)
        for i in l:
            val += 1
        self.pb_signal.emit(60)
        self.pb_signal.emit(100)
        pq = pd.DataFrame(data = {'col1':[1,2,3,4,5,6], 'col2':[6,5,4,3,2,1]})
        self.plot_signal.emit(pq)
        self.plot_signal.emit(pq)
        self.plot_signal.emit(pq)        

if __name__ == '__main__':
    app = QApplication(sys.argv)
    ex = App()
    ex.show()
    sys.exit(app.exec_())

注意:只创建了列表以防止垃圾收集器删除图