Пользовательский инструмент BokehJS для переключения видимости легенды

Мое приложение боке имеет дело с сеткой из нескольких фигур, каждая из которых показывает несколько глифов. Чтобы улучшить читаемость, я хотел бы иметь возможность скрывать/показывать легенды цифр при нажатии на кнопку. Хотя это показалось мне идеальным примером для кнопки инструмента на панели инструментов, такой как «сохранить» и «сбросить», эта функция еще не реализована в боке.

Я нашел несколько советов о том, как самостоятельно реализовать собственный инструмент, см. здесь, здесь, здесь, здесь или здесь. Пример того, как добавить собственный значок, показан здесь.

Это то, что я получил до сих пор: main.py:

from bokeh.layouts import gridplot
from bokeh.models import ColumnDataSource, Tool
from bokeh.plotting import figure

from os.path import dirname
from jinja2 import FileSystemLoader, Environment
from bokeh.embed.standalone import file_html
from bokeh.resources import CDN

class LegendToggleTool(Tool):
    __implementation__ = """
        import {ActionTool, ActionToolView} from "models/tools/actions/action_tool"
        import * as p from "core/properties"
        
        export class LegendToggleToolView extends ActionToolView {
            model: LegendToggleTool
            
            doit(): void {
                for(const r of this.plot_view.model.panels){
                    if (r.type=="Legend"){
                        r.visible = !r.visible
                    }
                }
            }
        }     
        
        export namespace LegendToggleTool {
          export type Attrs = p.AttrsOf<Props>
        
          export type Props = ActionTool.Props
        }
        
        export interface LegendToggleTool extends LegendToggleTool.Attrs {}
        
        export class LegendToggleTool extends ActionTool {
            properties: LegendToggleTool.Props
            __view_type__: LegendToggleToolView
            
            constructor(attrs?: Partial<LegendToggleTool.Attrs>) {
                super(attrs)
            }
            
            static init_LegendToggleTool(): void {
                this.prototype.default_view = LegendToggleToolView
                this.register_alias("legendtoggle", () => new LegendToggleTool())
            }
        
            tool_name = "LegendToggle"
            icon      = "legend-toggle-icon"
        }
        """

env = Environment(loader=FileSystemLoader(dirname(__file__)))
template = env.get_template('template.html')

source01 = ColumnDataSource(data=dict(x=[0,1,2,3,4], y=[0,1,2,3,4],z=[4,3,2,1,0]))
source02 = ColumnDataSource(data=dict(x=[0,3,1,6,1], y=[0,1,2,3,4],z=[4,3,2,1,0]))

fig01 = figure(x_range=(0, 10), y_range=(0, 10),tools=[LegendToggleTool()])
fig01.line('x', 'y', source=source01, legend_label = 'line_01')
fig01.line('x', 'z', source=source01, legend_label = 'line_02')

fig02 = figure(x_range=(0, 10), y_range=(0, 10),tools=[LegendToggleTool()])
fig02.line('x', 'y', source=source02, legend_label = 'line_03')
fig02.line('x', 'z', source=source02, legend_label = 'line_04')

with open('out.html', 'w') as f:
    f.write(file_html(gridplot([[fig01,fig02]]), resources=CDN, template=template))

template.html (в том же каталоге, скопировано из здесь ):

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="utf-8">
    <title>{{ title|e if title else "Bokeh Plot" }}</title>
    {{ bokeh_css }}
    {{ bokeh_js }}
    <style>
        html {
            width: 100%;
            height: 100%;
        }

        body {
            width: 90%;
            height: 100%;
            margin: auto;
        }

        .legend-toggle-icon {
            background-image: url();
        }
    </style>
</head>
<body>
{{ plot_div|indent(8) }}
{{ plot_script|indent(8) }}
</body>
</html>

Запуск main.py приводит к созданию выходного файла «out.html». Переключение видимости легенд работает должным образом, но значок остается пустым. Итак, вот мои вопросы:

  1. Как сделать, чтобы иконка отображалась?
  2. Допустим, я хотел запустить это через bokeh serve --show <dirname>, как мне организовать приведенный выше код внутри папки <dirname>?

person Alperino    schedule 23.09.2020    source источник
comment
в консоли есть ошибки?   -  person Void Spirit    schedule 23.09.2020
comment
Хорошая мысль, есть одна: DevTools failed to load SourceMap: Could not load content for file:///D:/python/bokeh/legend_tool/main.py:LegendToggleTool.js.map: System error: net::ERR_FILE_NOT_FOUND   -  person Alperino    schedule 23.09.2020