在 PyQt 窗口中无延迟地绘制 matplotlib 矩形

Posted

技术标签:

【中文标题】在 PyQt 窗口中无延迟地绘制 matplotlib 矩形【英文标题】:Draw matplotlib rectangle without lag in PyQt window 【发布时间】:2016-07-19 18:19:15 【问题描述】:

当我单击鼠标时,我想绘制一个框。当我单击按钮移动鼠标时,该框应更改。现在,问题是除非我停止鼠标移动,否则框不会更新,因此动作似乎有点滞后。如果不将情节放在 PyQt 环境中,它就可以正常工作。但不幸的是,我需要在 PyQt 窗口中使用它。

当按钮被点击并且它们立即更新时,我打印了他的鼠标坐标。

任何人都可以帮助或考虑替代结构吗?

import sys
from PyQt4 import QtGui

from matplotlib.backends.backend_qt4agg import FigureCanvasQTAgg as FigureCanvas
import matplotlib.pyplot as plt


from matplotlib.patches import Rectangle



class Annotate(object):
    def __init__(self, ax):
        self.ax = ax
        self.rect = Rectangle((0,0), 0, 0)
        self.x0 = None
        self.y0 = None
        self.x1 = None
        self.y1 = None
        self.ax.add_patch(self.rect)
        self.ax.figure.canvas.mpl_connect('button_press_event', self.on_press)
        self.ax.figure.canvas.mpl_connect('button_release_event', self.on_release)
        self.ax.figure.canvas.mpl_connect('motion_notify_event', self.on_motion)

        self.is_pressed = 0

    def on_motion(self, event):
        if self.is_pressed:
            self.x1 = event.xdata
            self.y1 = event.ydata 
            print(self.x1, self.y1)

            self.rect.set_width(self.x1 - self.x0)
            self.rect.set_height(self.y1 - self.y0)
            self.rect.set_xy((self.x0, self.y0))
            self.rect.figure.canvas.draw() 

    def on_press(self, event):
        self.is_pressed = 1
        self.x0 = event.xdata
        self.y0 = event.ydata

    def on_release(self, event):
        self.is_pressed = 0


class Window(QtGui.QDialog):
    def __init__(self, parent=None):
        super(Window, self).__init__(parent)
        self.figure = plt.figure(facecolor='white')
        self.canvas = FigureCanvas(self.figure)
        layout = QtGui.QVBoxLayout()
        layout.addWidget(self.canvas)
        self.setLayout(layout)

        ''' plot some random stuff '''
        ax = self.figure.add_subplot(111)
        self.ax = ax
        ax.plot()
        self.canvas.draw()

    def get_ax(self):
        return self.ax

if __name__ == '__main__':
    app = QtGui.QApplication(sys.argv)
    main = Window()
    ax = main.get_ax()
    main.show()
    a = Annotate(ax)
    sys.exit(app.exec_())

【问题讨论】:

【参考方案1】:

试试这个:

import sys
from PyQt5 import QtWidgets

from matplotlib.patches import Rectangle
from matplotlib.backends.backend_qt4agg import FigureCanvasQTAgg as FigureCanvas
import matplotlib.pyplot as plt    

class Annotate(object):
    def __init__(self, ax):
        self.ax = ax
        self.rect = Rectangle((0,0), 0, 0)
        self.x0 = None
        self.y0 = None
        self.x1 = None
        self.y1 = None
        self.ax.add_patch(self.rect)
        self.ax.figure.canvas.mpl_connect('button_press_event', self.on_press)
        self.ax.figure.canvas.mpl_connect('button_release_event', self.on_release)
        self.ax.figure.canvas.mpl_connect('motion_notify_event', self.on_motion)

        self.is_pressed = 0

    def on_motion(self, event):
        if self.is_pressed:
            self.x1 = event.xdata
            self.y1 = event.ydata 
            print(self.x1, self.y1)

            self.rect.set_width(self.x1 - self.x0)
            self.rect.set_height(self.y1 - self.y0)
            self.rect.set_xy((self.x0, self.y0))
            self.rect.figure.canvas.draw() 

    def on_press(self, event):
        self.is_pressed = 1
        self.x0 = event.xdata
        self.y0 = event.ydata

    def on_release(self, event):
        self.is_pressed = 0

class Window(QtWidgets.QDialog):
    def __init__(self, parent=None):
        super(Window, self).__init__(parent)
        self.figure = plt.figure(facecolor='white')
        self.canvas = FigureCanvas(self.figure)
        layout = QtWidgets.QVBoxLayout()
        layout.addWidget(self.canvas)
        self.setLayout(layout)

        ''' plot some random stuff '''
        ax = self.figure.add_subplot(111)
        self.ax = ax
        ax.plot()
        self.canvas.draw()

    def get_ax(self):
        return self.ax

if __name__ == '__main__':
    app = QtWidgets.QApplication(sys.argv)
    main = Window()
    ax = main.get_ax()
    main.show()
    a = Annotate(ax)
    sys.exit(app.exec_())

【讨论】:

以上是关于在 PyQt 窗口中无延迟地绘制 matplotlib 矩形的主要内容,如果未能解决你的问题,请参考以下文章

在 Objective-C 中无延迟播放音效

绘制多个图形时添加滚动条以在 PyQt5 中保留图形大小

在黑莓中无延迟播放音频

在 pyqt5 中绘制 Matplotlib 图

与 pyQt 一起使用时箱线图不显示

在 Pyqt 窗口中使用 glfw 窗口