Matplotlib连续按钮

时间:2017-10-18 18:00:01

标签: python python-3.x matplotlib

找不到任何关于如何将matplotlib中的checkbuttons'box'格式化为行与列格式的内容。我发现我可以移动实际的复选框和标签(在文本对象上使用get_position,在矩形对象上使用set_xy)但这似乎不是最好的方式,因为我正在移动它们是分开的,它们似乎没有使用相同的坐标系。

是否有更简单的方法可以将我的支票按钮从一列变成一行?

1 个答案:

答案 0 :(得分:3)

没有简单的方法可以使复选框显示在行而不是列中。原因是它们在轴坐标方面是硬编码的。当然你可以移动零碎的东西,但这听起来相当麻烦。

另一种选择是继承CheckButtons类并实现自己的布局。现在这样做时,需要考虑很多,比如标签和复选框之间有多少空间等。

另一方面,实际上有一个很好的matplotlib项目,可以让所有旋钮关于标签间距,填充等可用,并且适用于行和列:图例。

现在我在想为什么不将复选框生成为可点击的图例。这就是下面的代码。它子类CheckButtons并在其轴上创建一个图例。此图例将框作为句柄,将标签作为图例标签。 现在好了,您可以使用usual legend allows forfontsize, markerfirst, frameon, fancybox, shadow, framealpha, facecolor, edgecolor, mode, bbox_transform, title, borderpad, labelspacing, handlelength, handletextpad, borderaxespad, columnspacing的所有工具按照自己的喜好布局检查按钮。

这里最重要的是n_col,您可以将其设置为您想要的列数。

我从matplotlib页面获取了CheckButtons example并在其上使用了这个自定义Checkbuttons类。

import numpy as np
import matplotlib.pyplot as plt
from matplotlib.widgets import CheckButtons,AxesWidget

class PremiumCheckButtons(CheckButtons,AxesWidget):
    def __init__(self, ax, labels, actives, linecolor="k", showedge=True, **kw):
        AxesWidget.__init__(self, ax)

        ax.set_xticks([])
        ax.set_yticks([])
        ax.set_navigate(False)
        if not showedge:
            ax.axis("off")
        linekw = {'solid_capstyle': 'butt', "color" : linecolor}
        class Handler(object):
            def legend_artist(self, legend, orig_handle, fontsize, handlebox):
                x0, y0 = handlebox.xdescent, handlebox.ydescent
                height = handlebox.height
                self.line1 = plt.Line2D([x0,x0+height],[y0,y0+height], **linekw)
                self.line2 = plt.Line2D([x0,x0+height],[y0+height,y0], **linekw)
                self.rect = plt.Rectangle((x0,y0),height, height, 
                                          edgecolor="k", fill=False)
                handlebox.add_artist(self.rect)
                handlebox.add_artist(self.line1)
                handlebox.add_artist(self.line2)
                return [self.line1, self.line2, self.rect]

        self.box = ax.legend(handles = [object() for i in labels ], 
                             labels = labels,
                             handler_map={object: Handler()}, **kw)

        self.lines = [(h[0],h[1]) for h in self.box.legendHandles]
        self.rectangles = [h[2] for h in self.box.legendHandles]
        self.labels = self.box.texts

        for i,(l1,l2) in enumerate(self.lines):
            l1.set_visible(actives[i])
            l2.set_visible(actives[i])

        self.connect_event('button_press_event', self._clicked)

        self.cnt = 0
        self.observers = {}


t = np.arange(0.0, 2.0, 0.01)
s0 = np.sin(2*np.pi*t)
s1 = np.sin(4*np.pi*t)
s2 = np.sin(6*np.pi*t)

fig, (rax,ax) = plt.subplots(nrows=2, gridspec_kw=dict(height_ratios = [0.1,1]) )
l0, = ax.plot(t, s0, visible=False, lw=2)
l1, = ax.plot(t, s1, lw=2)
l2, = ax.plot(t, s2, lw=2)
plt.subplots_adjust(left=0.2)

check = PremiumCheckButtons(rax, ('2 Hz', '4 Hz', '6 Hz'), (False, True, True),
                           showedge = False, ncol=3)


def func(label):
    if label == '2 Hz':
        l0.set_visible(not l0.get_visible())
    elif label == '4 Hz':
        l1.set_visible(not l1.get_visible())
    elif label == '6 Hz':
        l2.set_visible(not l2.get_visible())
    fig.canvas.draw_idle()
check.on_clicked(func)

plt.show()

enter image description here

除了所有legend个参数之外,上面的PremiumCheckButtons类使用参数linecolor来设置复选框的x的颜色(默认为黑色)和showedgeshowedge可用于显示“图例”所在的轴的框架,并且可以打开以查看此轴以进行调试。例如。您需要确保完整的图例实际位于轴内,以便按钮可以点击。