Esempio n. 1
0
 def test_on_change_emits_state(self):
     key = "k"
     value = "*.nc"
     controls = db.Controls(self.database, patterns=[(key, value)])
     callback = unittest.mock.Mock()
     controls.subscribe(callback)
     controls.on_change('pattern')(None, None, value)
     callback.assert_called_once_with(db.State(pattern=value))
Esempio n. 2
0
 def test_render_state_configures_variable_menu(self):
     controls = db.Controls(self.database)
     state = db.State(variables=["mslp"])
     controls.render(state)
     result = controls.dropdowns["variable"].menu
     expect = ["mslp"]
     self.assert_label_equal(expect, result)
     self.assert_value_equal(expect, result)
Esempio n. 3
0
 def test_next_pressure_given_pressures_returns_first_element(self):
     value = 950
     controls = db.Controls(self.database,
                            state=db.State(pressures=[value]))
     controls.on_click('pressure', 'next')()
     result = controls.state.pressure
     expect = value
     self.assertEqual(expect, result)
Esempio n. 4
0
 def test_send_sets_state_variables(self):
     self.database.insert_variable("a.nc", "air_temperature")
     self.database.insert_variable("b.nc", "mslp")
     controls = db.Controls(self.database)
     controls.state = db.State(pattern="b.nc")
     controls.send(self.blank_message)
     result = controls.state.variables
     expect = ["mslp"]
     self.assertEqual(expect, result)
Esempio n. 5
0
 def test_state_change_given_previous_initial_time_message(self):
     state = db.State(initial_times=["2019-01-01 00:00:00"])
     message = db.Message.button("initial_time", "previous")
     result = db.Controls(self.database).modify(state, message)
     expect = db.State(initial_times=["2019-01-01 00:00:00"],
                       initial_time="2019-01-01 00:00:00")
     self.assertEqual(expect.initial_time, result.initial_time)
     self.assertEqual(expect.initial_times, result.initial_times)
     self.assertEqual(expect, result)
Esempio n. 6
0
 def test_next_pressure_given_current_pressure(self):
     pressure = 950
     pressures = [1000, 950, 800]
     controls = db.Controls(self.database,
                            state=db.State(pressures=pressures,
                                           pressure=pressure))
     controls.on_click('pressure', 'next')()
     result = controls.state.pressure
     expect = 800
     self.assertEqual(expect, result)
Esempio n. 7
0
 def test_render_state_configures_variable_menu(self):
     self.database.insert_variable("a.nc", "air_temperature")
     self.database.insert_variable("b.nc", "mslp")
     controls = db.Controls(self.database)
     state = db.State(pattern="b.nc")
     controls.render(state)
     result = controls.dropdowns["variable"].menu
     expect = ["mslp"]
     self.assert_label_equal(expect, result)
     self.assert_value_equal(expect, result)
Esempio n. 8
0
def main():
    args = parse_args()
    with open(args.config_file) as stream:
        config = load_config(stream)
    database = db.Database.connect(args.database)
    controls = db.Controls(database, patterns=config.patterns)
    controls.subscribe(print)

    locator = db.Locator.connect(args.database)
    text = db.View(text="Hello, world!", locator=locator)
    controls.subscribe(text.on_state)

    document = bokeh.plotting.curdoc()
    document.add_root(controls.layout)
    document.add_root(text.div)
Esempio n. 9
0
def main():
    args = parse_args.parse_args()
    database = db.Database.connect(args.database)
    with open(args.config_file) as stream:
        config = parse_args.load_config(stream)

    # Access latest files
    data.FILE_DB.sync()

    # Full screen map
    lon_range = (0, 30)
    lat_range = (0, 30)
    x_range, y_range = geo.web_mercator(
        lon_range,
        lat_range)
    figure = bokeh.plotting.figure(
        x_range=x_range,
        y_range=y_range,
        x_axis_type="mercator",
        y_axis_type="mercator",
        active_scroll="wheel_zoom")
    tile = bokeh.models.WMTSTileSource(
        url="https://maps.wikimedia.org/osm-intl/{Z}/{X}/{Y}.png",
        attribution=""
    )

    figures = [figure]
    for _ in range(2):
        f = bokeh.plotting.figure(
            x_range=figure.x_range,
            y_range=figure.y_range,
            x_axis_type="mercator",
            y_axis_type="mercator",
            active_scroll="wheel_zoom")
        figures.append(f)

    for f in figures:
        f.axis.visible = False
        f.toolbar.logo = None
        f.toolbar_location = None
        f.min_border = 0
        f.add_tile(tile)

    figure_row = bokeh.layouts.row(*figures,
            sizing_mode="stretch_both")
    figure_row.children = [figures[0]]  # Trick to keep correct sizing modes

    figure_drop = bokeh.models.Dropdown(
            label="Figure",
            menu=[(str(i), str(i)) for i in [1, 2, 3]])

    def on_change(attr, old, new):
        if int(new) == 1:
            figure_row.children = [
                    figures[0]]
        elif int(new) == 2:
            figure_row.children = [
                    figures[0],
                    figures[1]]
        elif int(new) == 3:
            figure_row.children = [
                    figures[0],
                    figures[1],
                    figures[2]]

    figure_drop.on_change("value", on_change)

    color_mapper = bokeh.models.LinearColorMapper(
            low=0,
            high=1,
            palette=bokeh.palettes.Plasma[256])
    for figure in figures:
        colorbar = bokeh.models.ColorBar(
            color_mapper=color_mapper,
            orientation="horizontal",
            background_fill_alpha=0.,
            location="bottom_center",
            major_tick_line_color="black",
            bar_line_color="black")
        figure.add_layout(colorbar, 'center')

    for name, pattern in config.patterns:
        if name not in data.LOADERS:
            locator = db.Locator(
                database.connection,
                directory=args.directory)
            loader = data.DBLoader(name, pattern, locator)
            data.add_loader(name, loader)

    renderers = {}
    viewers = {}
    for name, loader in data.LOADERS.items():
        if isinstance(loader, rdt.Loader):
            viewer = rdt.View(loader)
        elif isinstance(loader, earth_networks.Loader):
            viewer = earth_networks.View(loader)
        elif isinstance(loader, data.GPM):
            viewer = view.GPMView(loader, color_mapper)
        elif isinstance(loader, satellite.EIDA50):
            viewer = view.EIDA50(loader, color_mapper)
        else:
            viewer = view.UMView(loader, color_mapper)
        viewers[name] = viewer
        renderers[name] = [
                viewer.add_figure(f)
                for f in figures]
    artist = Artist(viewers, renderers)
    renderers = []
    for _, r in artist.renderers.items():
        renderers += r

    image_sources = []
    for name, viewer in artist.viewers.items():
        if isinstance(viewer, (view.UMView, view.GPMView, view.EIDA50)):
            image_sources.append(viewer.source)

    # image_loaders = []
    # for name, loader in data.LOADERS.items():
    #     if isinstance(loader, (data.UMLoader, data.GPM)):
    #         image_loaders.append(loader)

    # Lakes
    for figure in figures:
        add_feature(figure, data.LAKES, color="lightblue")

    features = []
    for figure in figures:
        features += [
            add_feature(figure, data.COASTLINES),
            add_feature(figure, data.BORDERS)]

    # Disputed borders
    for figure in figures:
        add_feature(figure, data.DISPUTED, color="red")

    toggle = bokeh.models.CheckboxButtonGroup(
            labels=["Coastlines"],
            active=[0],
            width=135)

    def on_change(attr, old, new):
        if len(new) == 1:
            for feature in features:
                feature.visible = True
        else:
            for feature in features:
                feature.visible = False

    toggle.on_change("active", on_change)

    dropdown = bokeh.models.Dropdown(
            label="Color",
            menu=[
                ("Black", "black"),
                ("White", "white")],
            width=50)
    autolabel(dropdown)

    def on_change(attr, old, new):
        for feature in features:
            feature.glyph.line_color = new

    dropdown.on_change("value", on_change)

    slider = bokeh.models.Slider(
        start=0,
        end=1,
        step=0.1,
        value=1.0,
        show_value=False)
    custom_js = bokeh.models.CustomJS(
            args=dict(renderers=renderers),
            code="""
            renderers.forEach(function (r) {
                r.glyph.global_alpha = cb_obj.value
            })
            """)
    slider.js_on_change("value", custom_js)

    colors_controls = colors.Controls(
            color_mapper, "Plasma", 256)

    mapper_limits = MapperLimits(image_sources, color_mapper)

    menu = [(n, n) for n in data.FILE_DB.names]
    for k, _ in config.patterns:
        menu.append((k, k))

    image_controls = images.Controls(menu)

    def on_change(attr, old, new):
        if int(new) == 1:
            image_controls.labels = ["Show"]
        elif int(new) == 2:
            image_controls.labels = ["L", "R"]
        elif int(new) == 3:
            image_controls.labels = ["L", "C", "R"]

    figure_drop.on_change("value", on_change)

    image_controls.subscribe(artist.on_visible)

    div = bokeh.models.Div(text="", width=10)
    border_row = bokeh.layouts.row(
        bokeh.layouts.column(toggle),
        bokeh.layouts.column(div),
        bokeh.layouts.column(dropdown))

    # Add prototype database controls
    controls = db.Controls(database, patterns=config.patterns)
    locator = db.Locator(
        database.connection,
        directory=args.directory)
    text = db.View(text="", locator=locator)
    controls.subscribe(text.on_state)
    controls.subscribe(artist.on_state)

    tabs = bokeh.models.Tabs(tabs=[
        bokeh.models.Panel(
            child=bokeh.layouts.column(
                bokeh.models.Div(text="Navigate:"),
                controls.layout,
                bokeh.models.Div(text="Compare:"),
                bokeh.layouts.row(figure_drop),
                image_controls.column,
                text.div),
            title="Control"
        ),
        bokeh.models.Panel(
            child=bokeh.layouts.column(
                border_row,
                bokeh.layouts.row(slider),
                colors_controls.layout,
                bokeh.layouts.row(mapper_limits.low_input),
                bokeh.layouts.row(mapper_limits.high_input),
                bokeh.layouts.row(mapper_limits.checkbox),
                ),
            title="Settings")
        ])

    # Series sub-figure widget
    series_figure = bokeh.plotting.figure(
                plot_width=400,
                plot_height=200,
                x_axis_type="datetime",
                toolbar_location=None,
                border_fill_alpha=0)
    series_figure.toolbar.logo = None
    series_row = bokeh.layouts.row(
            series_figure,
            name="series")

    def place_marker(figure, source):
        figure.circle(
                x="x",
                y="y",
                color="red",
                source=source)
        def cb(event):
            source.data = {
                    "x": [event.x],
                    "y": [event.y]}
        return cb

    marker_source = bokeh.models.ColumnDataSource({
            "x": [],
            "y": []})

    series = Series(series_figure)
    controls.subscribe(series.on_state)
    for f in figures:
        f.on_event(bokeh.events.Tap, series.on_tap)
        f.on_event(bokeh.events.Tap, place_marker(f, marker_source))


    # Minimise controls to ease navigation
    compact_button = bokeh.models.Button(
            label="Compact")
    compact_minus = bokeh.models.Button(label="-", width=50)
    compact_plus = bokeh.models.Button(label="+", width=50)
    compact_navigation = bokeh.layouts.column(
            compact_button,
            bokeh.layouts.row(
                compact_minus,
                compact_plus,
                width=100))
    control_root = bokeh.layouts.column(
            compact_button,
            tabs,
            name="controls")

    display = "large"
    def on_compact():
        nonlocal display
        if display == "large":
            control_root.height = 100
            control_root.width = 120
            compact_button.width = 100
            compact_button.label = "Expand"
            control_root.children = [
                    compact_navigation]
            display = "compact"
        else:
            control_root.height = 500
            control_root.width = 300
            compact_button.width = 300
            compact_button.label = "Compact"
            control_root.children = [compact_button, tabs]
            display = "large"

    compact_button.on_click(on_compact)

    document = bokeh.plotting.curdoc()
    document.title = "FOREST"
    document.add_root(control_root)
    document.add_root(series_row)
    document.add_root(figure_row)
Esempio n. 10
0
 def setUp(self):
     self.database = db.Database.connect(":memory:")
     self.controls = db.Controls(self.database)
     self.blank_message = db.Message('blank', None)
Esempio n. 11
0
 def test_state_change_given_dropdown_message(self):
     state = db.State()
     message = db.Message.dropdown("pressure", "1000")
     result = db.Controls(self.database).modify(state, message)
     expect = db.State(pressure=1000.)
     self.assertEqual(expect, result)
Esempio n. 12
0
 def test_next_pressure_given_pressures_none(self):
     controls = db.Controls(self.database, state=db.State())
     controls.on_click('pressure', 'next')()
     result = controls.state.pressure
     expect = None
     self.assertEqual(expect, result)
Esempio n. 13
0
 def setUp(self):
     self.database = db.Database.connect(":memory:")
     self.controls = db.Controls(self.database)