python - 具有可点击和可选择点的 PyQt4 散点图

标签 python matplotlib pyqt4 clickable scatter-plot

我正在尝试以某种方式创建一个包含大约 1000 个数据点的散点图,以便可以通过使用鼠标单击它们来选择每个数据点,这将打开一个上下文菜单,允许用户删除或更改数据点的颜色。我一直在关注 matplotlib 的教程,并查看了 Draggable Rectangle Exercise,但我遇到了困难。我正在使用 matplotlib.patches.Circle 类来表示每个数据点,但我无法让“包含”方法与“button_press_event”一起正常工作。主要是,似乎没有与每个 Circle 对象关联的“ Canvas ”对象。我在第 16 行收到以下错误:

AttributeError: 'NoneType' object has no attribute 'canvas'

代码如下:

#!/usr/bin/python -tt

import sys
import numpy
#import matplotlib.pyplot
from PyQt4 import QtGui, QtCore
from matplotlib.figure import Figure
from matplotlib.backends.backend_qt4agg import FigureCanvasQTAgg as FigureCanvas
from matplotlib.patches import Circle

class SelectablePoint:
    def __init__(self, xy, label):
        self.point = Circle( (xy[0], xy[0]), .005 )
        self.label = label
        self.point.figure
        self.cidpress = self.point.figure.canvas.mpl_connect('button_press_event', self.onClick)

    def onClick(self, e):

        print e.xdata, e.ydata
        #print(dir(self.point))
        print self.point.center
        print self.point.contains(e)[0]
        #if self.point.contains(e)[0]:
        #    print self.label


class ScatterPlot(FigureCanvas):
    '''
    classdocs
    '''

    def __init__(self, parent=None):
        '''
        Constructor
        '''

        self.fig = Figure()
        FigureCanvas.__init__(self, self.fig)

        self.axes = self.fig.add_subplot(111)
        #x = numpy.arange(0.0, 3.0, 0.1)
        #y = numpy.cos(2*numpy.pi*x)
        x = [.5]
        y = [.5]

        #scatterplot = self.axes.scatter(x,y)

        for i in range(len(x)):
            c = Circle( (x[i], y[i]), .05 )
            self.axes.add_patch(c)
            #SelectablePoint( (x[i],y[i]), 'label for: ' + str(i), self.figure.canvas )
            SelectablePoint( (x[i],y[i]), 'label for: ' + str(i) )
            #self.axes.add_artist(c)

class MainContainer(QtGui.QMainWindow):

    def __init__(self):
        QtGui.QMainWindow.__init__(self)
        self.resize(900,600)
        self.setWindowTitle('Scatter Plot')

        sp = ScatterPlot(self)
        self.setCentralWidget(sp)

        self.center()

    def center(self):
        # Get the resolution of the screen
        screen = QtGui.QDesktopWidget().screenGeometry()

        # Get the size of widget
        size = self.geometry()
        self.move( (screen.width() - size.width())/2, (screen.height() - size.height())/2 )

if __name__ == '__main__':
    app = QtGui.QApplication(sys.argv)
    b = MainContainer()
    b.show()
    sys.exit(app.exec_())

我不确定我是否以正确的方式处理这个问题,或者我是否应该查看另一个图形模块,但我查看了 gnuplot 和 chaco,我觉得 matplotlib 更适合我的问题.还有其他推荐吗?非常感谢。

** 我的解决方案 **

这是我目前所做的工作。单击散点图中的点时,它只是将每个数据点的标签输出到标准输出。

#!/usr/bin/python -tt

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

class SelectablePoint:
    def __init__(self, xy, label, fig):
        self.point = Circle( (xy[0], xy[1]), .25, figure=fig)
        self.label = label
        self.cidpress = self.point.figure.canvas.mpl_connect('button_press_event', self.onClick)

    def onClick(self, e):
        if self.point.contains(e)[0]:
            print self.label


class ScatterPlot(FigureCanvas):
    '''
    classdocs
    '''

    def __init__(self, parent=None):
        '''
        Constructor
        '''

        self.fig = Figure()
        FigureCanvas.__init__(self, self.fig)

        self.axes = self.fig.add_subplot(111)
        xlim = [0,7]
        ylim = [0,7]
        self.axes.set_xlim(xlim)
        self.axes.set_ylim(ylim)
        self.axes.set_aspect( 1 )

        x = [1, 1.2, 3, 4, 5, 6]
        y = [1, 1.2, 3, 4, 5, 6]
        labels = ['1', '2', '3', '4', '5', '6']

        for i in range(len(x)):
            sp = SelectablePoint( (x[i],y[i]), labels[i], self.fig)
            self.axes.add_artist(sp.point)

class MainContainer(QtGui.QMainWindow):

    def __init__(self):
        QtGui.QMainWindow.__init__(self)
        self.resize(900,600)
        self.setWindowTitle('Scatter Plot')

        sp = ScatterPlot(self)
        self.setCentralWidget(sp)

        self.center()

    def center(self):
        # Get the resolution of the screen
        screen = QtGui.QDesktopWidget().screenGeometry()

        # Get the size of widget
        size = self.geometry()
        self.move( (screen.width() - size.width())/2, (screen.height() - size.height())/2 )

if __name__ == '__main__':
    app = QtGui.QApplication(sys.argv)
    b = MainContainer()
    b.show()
    sys.exit(app.exec_())

最佳答案

糟糕,我正在为每个数据点创建两个 patches.Circle 实例。将第一个实例传递到我的 SelectablePoint 类后,一切正常。

关于python - 具有可点击和可选择点的 PyQt4 散点图,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5055972/

相关文章:

matplotlib - matplotlib 中的散点图未更新 xlim 和 ylim

python - pyqt 的信号与直接调用方法

python - 如何在 Pyspark 中创建虚拟(0 字节)HDFS 文件

python - networkx 中的 nx.get_node_attributes 返回一个空字典

python - 如何确定 matplotlib 中的屏幕大小

python - Pycharm:Python Qt代码代码补全

python - PyQt - 减少小部件*扩展*布局中的边距和间距

python - pylibjpeg-libjpeg 未安装

python - 在构建编码器/解码器模型之前训练自动编码器是否有效?

python - 在 Matplotlib 3D 图中更改旋转中心