Hiding lines after showing a pyplot figure

前端 未结 3 1951
被撕碎了的回忆
被撕碎了的回忆 2020-12-04 22:33

I\'m using pyplot to display a line graph of up to 30 lines. I would like to add a way to quickly show and hide individual lines on the graph. Pyplot does have a menu wher

3条回答
  •  北海茫月
    2020-12-04 23:22

    If you'd like, you can hook up a callback to the legend that will show/hide lines when they're clicked. There's a simple example here: http://matplotlib.org/examples/event_handling/legend_picking.html

    Here's a "fancier" example that should work without needing to manually specify the relationship of the lines and legend markers (Also has a few more features).

    (Updated version in August 2019, as a response to repeated reports about this not working correctly; now it should! For the old version see version history)

    import numpy as np
    import matplotlib.pyplot as plt
    
    
    def main():
        x = np.arange(10)
        fig, ax = plt.subplots()
        for i in range(1, 31):
            ax.plot(x, i * x, label=r'$y={}x$'.format(i))
    
        ax.legend(loc='upper left', bbox_to_anchor=(1.05, 1),
                  ncol=2, borderaxespad=0)
        fig.subplots_adjust(right=0.55)
        fig.suptitle('Right-click to hide all\nMiddle-click to show all',
                     va='top', size='large')
    
        leg = interactive_legend()
        return fig, ax, leg
    
    def interactive_legend(ax=None):
        if ax is None:
            ax = plt.gca()
        if ax.legend_ is None:
            ax.legend()
    
        return InteractiveLegend(ax.get_legend())
    
    class InteractiveLegend(object):
        def __init__(self, legend):
            self.legend = legend
            self.fig = legend.axes.figure
    
            self.lookup_artist, self.lookup_handle = self._build_lookups(legend)
            self._setup_connections()
    
            self.update()
    
        def _setup_connections(self):
            for artist in self.legend.texts + self.legend.legendHandles:
                artist.set_picker(10) # 10 points tolerance
    
            self.fig.canvas.mpl_connect('pick_event', self.on_pick)
            self.fig.canvas.mpl_connect('button_press_event', self.on_click)
    
        def _build_lookups(self, legend):
            labels = [t.get_text() for t in legend.texts]
            handles = legend.legendHandles
            label2handle = dict(zip(labels, handles))
            handle2text = dict(zip(handles, legend.texts))
    
            lookup_artist = {}
            lookup_handle = {}
            for artist in legend.axes.get_children():
                if artist.get_label() in labels:
                    handle = label2handle[artist.get_label()]
                    lookup_handle[artist] = handle
                    lookup_artist[handle] = artist
                    lookup_artist[handle2text[handle]] = artist
    
            lookup_handle.update(zip(handles, handles))
            lookup_handle.update(zip(legend.texts, handles))
    
            return lookup_artist, lookup_handle
    
        def on_pick(self, event):
            handle = event.artist
            if handle in self.lookup_artist:
    
                artist = self.lookup_artist[handle]
                artist.set_visible(not artist.get_visible())
                self.update()
    
        def on_click(self, event):
            if event.button == 3:
                visible = False
            elif event.button == 2:
                visible = True
            else:
                return
    
            for artist in self.lookup_artist.values():
                artist.set_visible(visible)
            self.update()
    
        def update(self):
            for artist in self.lookup_artist.values():
                handle = self.lookup_handle[artist]
                if artist.get_visible():
                    handle.set_visible(True)
                else:
                    handle.set_visible(False)
            self.fig.canvas.draw()
    
        def show(self):
            plt.show()
    
    if __name__ == '__main__':
        fig, ax, leg = main()
        plt.show()
    

    This allows you to click on legend items to toggle their corresponding artists on/off. For example, you can go from this:

    enter image description here

    To this:

    enter image description here

提交回复
热议问题