UNPKG

@quartic/bokehjs

Version:

Interactive, novel data visualization

63 lines (49 loc) 1.8 kB
namespace TappyScatter { import plt = Bokeh.Plotting; const {range, zip} = Bokeh.LinAlg; Bokeh.set_log_level("info"); Bokeh.logger.info(`Bokeh ${Bokeh.version}`); const random = (function() { let seed = 1 return function() { // Park-Miller LCG seed = (seed*48271) % 2147483647 // 1 <= seed < 2^31 - 1 return (seed - 1) / 2147483646 } })() const M = 100 const xx: Array<number> = [] const yy: Array<number> = [] for (let y = 0; y <= M; y += 4) { for (let x = 0; x <= M; x += 4) { xx.push(x) yy.push(y) } } const N = xx.length const indices = range(N).map((i) => i.toString()) const radii = range(N).map((i) => random()*0.4 + 1.7) const colors: Array<string> = [] for (const [r, g] of zip(xx.map((x) => 50 + 2*x), yy.map((y) => 30 + 2*y))) colors.push(plt.color(r, g, 150)) const source = new Bokeh.ColumnDataSource({ data: {x: xx, y: yy, radius: radii, colors: colors } }) const tools = "pan,crosshair,wheel_zoom,box_zoom,reset,tap,save" const p = plt.figure({title: "Tappy Scatter", tools: tools}) const circles = p.circle({field: "x"}, {field: "y"}, {source: source, radius: radii, fill_color: colors, fill_alpha: 0.6, line_color: null}) p.text({field: "x"}, {field: "y"}, indices, {source: source, alpha: 0.5, text_font_size: "5pt", text_baseline: "middle", text_align: "center"}) const tap = p.toolbar.select_one(Bokeh.TapTool) tap.renderers = [circles] tap.callback = (ds) => { const indices = ds.selected['1d'].indices if (indices.length == 1) console.log(`Selected index: ${indices[0]}`) else if (indices.length > 1) console.log(`Selected indices: ${indices.join(', ')}`) else console.log("Nothing selected") } plt.show(p) }