MarcSkovMadsen's picture
update
2c150e9
raw
history blame
9.91 kB
import dask.dataframe as dd
import holoviews as hv
import numpy as np
import pandas as pd
import panel as pn
import param
from holoviews.operation.datashader import dynspread, rasterize
from utils import (
DATASETS,
DATASHADER_LOGO,
DATASHADER_URL,
DESCRIPTION,
ESA_EASTING,
ESA_NORTHING,
MAJOR_TOM_LOGO,
MAJOR_TOM_LYRICS,
MAJOR_TOM_PICTURE,
MAJOR_TOM_REF_URL,
META_DATA_COLUMNS,
PANEL_LOGO,
PANEL_URL,
get_closest_rows,
get_image,
get_meta_data,
)
class DatasetInput(pn.viewable.Viewer):
value = param.Selector(objects=DATASETS, allow_None=False, label="Dataset")
data = param.DataFrame(allow_None=False)
def __panel__(self):
return pn.widgets.RadioButtonGroup.from_param(
self.param.value, button_style="outline"
)
@pn.depends("value", watch=True, on_init=True)
def _update_data(self):
self.data = pn.cache(get_meta_data)(dataset=self.value)
class MapInput(pn.viewable.Viewer):
data = param.DataFrame(allow_refs=True, allow_None=False)
data_in_view = param.DataFrame(allow_None=False)
data_selected = param.DataFrame(allow_None=False)
_plot = param.Parameter(allow_None=False)
_pointer_x = param.Parameter(allow_None=False)
_pointer_y = param.Parameter(allow_None=False)
_range_xy = param.Parameter(allow_None=False)
_tap = param.Parameter(allow_None=False)
updating = param.Boolean()
def __panel__(self):
return pn.Column(
pn.pane.HoloViews(
self._plot, height=550, width=800, loading=self.param.updating
),
self._description,
)
@param.depends("data", watch=True, on_init=True)
def _handle_data_dask_change(self):
with self.param.update(updating=True):
data_dask = dd.from_pandas(self.data).persist()
points = hv.Points(
data_dask, kdims=["centre_easting", "centre_northing"], vdims=[]
)
rangexy = hv.streams.RangeXY(source=points)
tap = hv.streams.Tap(source=points, x=ESA_EASTING, y=ESA_NORTHING)
agg = rasterize(
points, link_inputs=True, x_sampling=0.0001, y_sampling=0.0001
)
dyn = dynspread(agg)
dyn.opts(cmap="kr_r", colorbar=True)
pointerx = hv.streams.PointerX(x=ESA_EASTING, source=points)
pointery = hv.streams.PointerY(y=ESA_NORTHING, source=points)
vline = hv.DynamicMap(lambda x: hv.VLine(x), streams=[pointerx])
hline = hv.DynamicMap(lambda y: hv.HLine(y), streams=[pointery])
tiles = hv.Tiles(
"https://tile.openstreetmap.org/{Z}/{X}/{Y}.png", name="OSM"
).opts(xlabel="Longitude", ylabel="Latitude")
self.param.update(
_plot=tiles * agg * dyn * hline * vline,
_pointer_x=pointerx,
_pointer_y=pointery,
_range_xy=rangexy,
_tap=tap,
)
update_viewed = pn.bind(
self._update_data_in_view,
rangexy.param.x_range,
rangexy.param.y_range,
watch=True,
)
update_viewed()
update_selected = pn.bind(
self._update_data_selected, tap.param.x, tap.param.y, watch=True
)
update_selected()
def _update_data_in_view(self, x_range, y_range):
if not x_range or not y_range:
self.data_in_view = self.data
return
data = self.data
data = data[
(data.centre_easting.between(*x_range))
& (data.centre_northing.between(*y_range))
]
self.data_in_view = data.reset_index(drop=True)
def _update_data_selected(self, tap_x, tap_y):
self.data_selected = get_closest_rows(self.data, tap_x, tap_y)
@pn.depends("data_in_view")
def _description(self):
return f"Rows: {len(self.data_in_view):,}"
class ImageInput(pn.viewable.Viewer):
data = param.DataFrame(allow_refs=True, allow_None=False)
column_name = param.Selector(
default="Thumbnail", objects=list(META_DATA_COLUMNS), label="Image Type"
)
updating = param.Boolean()
meta_data = param.DataFrame()
image = param.Parameter()
plot = param.Parameter()
_timestamp = param.Selector(label="Timestamp", objects=[None])
def __panel__(self):
return pn.Column(
pn.Row(
pn.widgets.RadioButtonGroup.from_param(
self.param._timestamp,
button_style="outline",
align="end",
),
pn.widgets.Select.from_param(
self.param.column_name, disabled=self.param.updating
),
),
pn.Tabs(
pn.pane.HoloViews(
self.param.plot,
loading=self.param.updating,
height=800,
width=800,
name="Interactive Image",
),
pn.pane.Image(
self.param.image,
name="Static Image",
loading=self.param.updating,
width=800,
),
pn.widgets.Tabulator(
self.param.meta_data,
name="Meta Data",
loading=self.param.updating,
disabled=True,
),
pn.pane.Markdown(self.code, name="Code"),
dynamic=True,
),
)
@pn.depends("data", watch=True, on_init=True)
def _update_timestamp(self):
if self.data.empty:
default_value = None
options = [None]
print("empty options")
else:
options = sorted(self.data["timestamp"].unique())
default_value = options[0]
print("options", options)
self.param._timestamp.objects = options
if not self._timestamp in options:
self._timestamp = default_value
@property
def column(self):
return META_DATA_COLUMNS[self.column_name]
@pn.depends("_timestamp", "column_name", watch=True, on_init=True)
def _update_plot(self):
if self.data.empty or not self._timestamp:
self.meta_data = self.data.T
self.image = None
self.plot = hv.RGB(np.array([]))
else:
with self.param.update(updating=True):
row = self.data[self.data.timestamp == self._timestamp].iloc[0]
self.meta_data = pd.DataFrame(row)
self.image = image = pn.cache(get_image)(row, self.column)
image_array = np.array(image)
if image_array.ndim == 2:
self.plot = hv.Image(image_array).opts(
cmap="gray_r", xaxis=None, yaxis=None, colorbar=True
)
else:
self.plot = hv.RGB(image_array).opts(xaxis=None, yaxis=None)
@pn.depends("meta_data", "column_name")
def code(self):
if self.meta_data.empty:
return ""
parquet_url = self.meta_data.T["parquet_url"].iloc[0]
parquet_row = self.meta_data.T["parquet_row"].iloc[0]
return f"""\
```python
from io import BytesIO
import holoviews as hv
import numpy as np
import panel as pn
import pyarrow.parquet as pq
from fsspec.parquet import open_parquet_file
from PIL import Image
pn.extension()
parquet_url = "{parquet_url}"
parquet_row = {parquet_row}
column = "{self.column}"
with open_parquet_file(parquet_url, columns=[column]) as f:
with pq.ParquetFile(f) as pf:
first_row_group = pf.read_row_group(parquet_row, columns=[column])
stream = BytesIO(first_row_group[column][0].as_py())
image = Image.open(stream)
image_array = np.array(image)
if image_array.ndim==2:
plot = hv.Image(image_array).opts(cmap="gray", colorbar=True)
else:
plot = hv.RGB(image_array)
plot.opts(xaxis=None, yaxis=None)
pn.panel(plot).servable()
```
"""
class App(param.Parameterized):
sidebar = param.Parameter()
main = param.Parameter()
def __init__(self, **params):
super().__init__(**params)
self.sidebar = self._create_sidebar()
self.main = pn.FlexBox(
pn.Column(
pn.Row(
pn.indicators.LoadingSpinner(value=True, size=50),
"**Loading data...**",
),
MAJOR_TOM_LYRICS,
)
)
pn.state.onload(self._update_main)
def _create_sidebar(self):
return pn.Column(
pn.pane.Image(
MAJOR_TOM_LOGO, link_url=MAJOR_TOM_REF_URL, sizing_mode="stretch_width"
),
pn.pane.Image(
MAJOR_TOM_PICTURE,
link_url=MAJOR_TOM_REF_URL,
sizing_mode="stretch_width",
),
DESCRIPTION,
pn.pane.Image(PANEL_LOGO, link_url=PANEL_URL, width=200, margin=(10, 20)),
pn.pane.Image(
DATASHADER_LOGO, link_url=DATASHADER_URL, width=200, margin=(10, 20)
),
)
def _create_main_content(self):
dataset = DatasetInput()
map_input = MapInput(data=dataset.param.data)
image_input = ImageInput(data=map_input.param.data_selected)
return pn.Column(dataset, map_input), image_input
def _update_main(self):
self.main[:] = list(self._create_main_content())
pn.extension("tabulator", design="fast")
app = App()
pn.template.FastListTemplate(
title="Major TOM Explorer",
main=[app.main],
sidebar=[app.sidebar],
main_layout=None,
accent="#003247", # "#A01346"
).servable()