散景图上的多个滑块

时间:2017-12-25 03:18:25

标签: python plot slider bokeh

我正在尝试使用散景滑块绘制几个圆圈。我知道页面https://bokeh.pydata.org/en/latest/docs/gallery/slider.html上有一个例子,但是当我尝试实现它时,我得到一个图形和滑块,但图形没有更新。我想这是回调的问题。

这不是我的代码:

import numpy as np
from bokeh.layouts import row, widgetbox
from bokeh.models import CustomJS, Slider
from bokeh.plotting import figure, output_file, show, ColumnDataSource

L_1 = 150
L_2 = 100
w = 0
h = 250
r1 = L_1 + L_2
r2 = L_1 - L_2
t = np.linspace(0, 2 * np.pi, 200)
sint = np.sin(t)
cost = np.cos(t)
x1 = r1 * cost
y1 = r1 * sint
x2 = r2 * cost
y2 = r2 * sint
x3 = r1 * cost + w
y3 = r1 * sint + h
x4 = r2 * cost + w
y4 = r2 * sint + h

source = ColumnDataSource(data=dict(x1=x1,y1=y1,x2=x2,y2=y2,x3=x3,y3=y3,x4=x4,y4=y4,t=t))

plot = figure(plot_width=400, plot_height=400)

plot.line('x1', 'y1', source=source, line_width=3, line_alpha=0.6)
plot.line('x2', 'y2', source=source, line_width=3, line_alpha=0.6)
plot.line('x3', 'y3', source=source, line_width=3, line_alpha=0.6)
plot.line('x4', 'y4', source=source, line_width=3, line_alpha=0.6)

callback = CustomJS(args=dict(source=source), code="""
    var data = source.data;
    var L_1 = l1.value;
    var L_2 = l2.value;
    var w = w.value;
    var h = h.value;
    var r1 = L_1 + L_2;
    var r2 = L_1 - L_2;
    t = data['t'];
    for (i = 0; i < t.length; i++) {
        sint = Math.sin(t[i]);
        cost = Math.cos(t[i]);
        y1[i] = r1 * sint;
        x1[i] = r1 * cost;
        y2[i] = r2 * sint;
        x2[i] = r2 * cost;
        y3[i] = r1 * sint + w;
        x3[i] = r1 * cost + h;
        y4[i] = r2 * sint + w;
        x4[i] = r2 * cost + h;
    }
    source.change.emit();
""")

L_1_slider = Slider(start=0, end=300, value=200, step=1,
                    title="L1", callback=callback)
callback.args["l1"] = L_1_slider

L_2_slider = Slider(start=0, end=300, value=150, step=1,
                    title="L2", callback=callback)
callback.args["l2"] = L_2_slider

w_slider = Slider(start=0, end=600, value=0, step=1,
                  title="w", callback=callback)
callback.args["w"] = w_slider

h_slider = Slider(start=0, end=600, value=0, step=.1,
                   title="h", callback=callback)
callback.args["h"] = h_slider

layout = row(
    plot,
    widgetbox(L_1_slider, L_2_slider, w_slider, h_slider),
)

output_file("slider2.html", title="slider.py example")

show(layout)

1 个答案:

答案 0 :(得分:3)

你是对的,这是回调的问题。在当前状态下,您可以检查浏览器控制台 - 它将显示y1未定义。这是因为回调中没有定义变量为y1(和y2等......)。然而,它存储在列数据源中;所以你可以按如下方式更新回调:(或者在for循环中赋值之前定义所有变量)。

callback = CustomJS(args=dict(source=source), code="""
    var data = source.data;
    var L_1 = l1.value;
    var L_2 = l2.value;
    var w = w.value;
    var h = h.value;
    var r1 = L_1 + L_2;
    var r2 = L_1 - L_2;
    t = data['t'];
    for (i = 0; i < t.length; i++) {
        sint = Math.sin(t[i]);
        cost = Math.cos(t[i]);
        data.y1[i] = r1 * sint;
        data.x1[i] = r1 * cost;
        data.y2[i] = r2 * sint;
        data.x2[i] = r2 * cost;
        data.y3[i] = r1 * sint + w;
        data.x3[i] = r1 * cost + h;
        data.y4[i] = r2 * sint + w;
        data.x4[i] = r2 * cost + h;
    }
    source.change.emit();
""")