diff --git a/geemap/core.py b/geemap/core.py
index 62027a3091..2eb50f200c 100644
--- a/geemap/core.py
+++ b/geemap/core.py
@@ -673,8 +673,14 @@ def _layer_manager(self) -> Optional[map_widgets.LayerManager]:
Optional[map_widgets.LayerManager]: The layer manager widget if found, else None.
"""
if toolbar_widget := self._toolbar:
- if isinstance(toolbar_widget.accessory_widget, map_widgets.LayerManager):
- return toolbar_widget.accessory_widget
+ return next(
+ (
+ widget
+ for widget in toolbar_widget.accessory_widgets
+ if isinstance(widget, map_widgets.LayerManager)
+ ),
+ None,
+ )
return self._find_widget_of_type(map_widgets.LayerManager)
@property
@@ -917,25 +923,6 @@ def add(self, obj: Any, position: str = "", **kwargs: Any) -> None:
else:
super().add(obj)
- def _on_toggle_toolbar_layers(self, is_open: bool) -> None:
- """Handles the toggle event for the toolbar layers.
-
- Args:
- is_open (bool): Whether the toolbar layers are open.
- """
- if is_open:
- if self._layer_manager:
- return
-
- layer_manager = map_widgets.LayerManager(self)
- layer_manager.header_hidden = True
- layer_manager.close_button_hidden = True
- layer_manager.refresh_layers()
- self._toolbar.accessory_widget = layer_manager
- else:
- self._toolbar.accessory_widget = None
- self.remove("layer_manager")
-
def _add_layer_manager(self, position: str, **kwargs: Any) -> None:
"""Adds a layer manager to the map.
@@ -943,9 +930,6 @@ def _add_layer_manager(self, position: str, **kwargs: Any) -> None:
position (str): The position to place the layer manager.
**kwargs (Any): Additional keyword arguments.
"""
- if self._layer_manager:
- return
-
layer_manager = map_widgets.LayerManager(self, **kwargs)
layer_manager.on_close = lambda: self.remove("layer_manager")
layer_manager.refresh_layers()
@@ -964,16 +948,21 @@ def _add_toolbar(self, position: str, **kwargs: Any) -> None:
if self._toolbar:
return
+ layer_manager = map_widgets.LayerManager(self)
+ layer_manager.header_hidden = True
+ layer_manager.close_button_hidden = True
+ layer_manager.refresh_layers()
+
toolbar_val = toolbar.Toolbar(
- self, self._toolbar_main_tools(), self._toolbar_extra_tools(), **kwargs
+ self,
+ self._toolbar_main_tools(),
+ self._toolbar_extra_tools(),
+ [layer_manager],
)
- toolbar_val.on_layers_toggled = self._on_toggle_toolbar_layers
toolbar_control = ipyleaflet.WidgetControl(
widget=toolbar_val, position=position
)
super().add(toolbar_control)
- # Enable the layer manager by default.
- toolbar_val.toggle_layers(True)
def _add_inspector(self, position: str, **kwargs: Any) -> None:
"""Adds an inspector to the map.
@@ -1255,65 +1244,66 @@ def _add_colorbar(
return control
def _open_help_page(
- self, host_map: "MapInterface", selected: bool, item: toolbar.Toolbar.Item
+ self, host_map: "MapInterface", selected: bool, item: toolbar.ToolbarItem
) -> None:
"""Opens the help page.
Args:
host_map (MapInterface): The host map.
selected (bool): Whether the item is selected.
- item (toolbar.Toolbar.Item): The toolbar item.
+ item (toolbar.ToolbarItem): The toolbar item.
"""
del host_map, item # Unused.
if selected:
coreutils.open_url("https://geemap.org")
- def _toolbar_main_tools(self) -> List[toolbar.Toolbar.Item]:
+ def _toolbar_main_tools(self) -> List[toolbar.ToolbarItem]:
"""Gets the main tools for the toolbar.
Returns:
- List[toolbar.Toolbar.Item]: The main tools for the toolbar.
+ List[toolbar.ToolbarItem]: The main tools for the toolbar.
"""
@toolbar._cleanup_toolbar_item
def inspector_tool_callback(
- map: Map, selected: bool, item: toolbar.Toolbar.Item
+ map: Map, selected: bool, item: toolbar.ToolbarItem
):
del selected, item # Unused.
map.add("inspector")
return map._inspector
@toolbar._cleanup_toolbar_item
- def basemap_tool_callback(map: Map, selected: bool, item: toolbar.Toolbar.Item):
+ def basemap_tool_callback(map: Map, selected: bool, item: toolbar.ToolbarItem):
del selected, item # Unused.
map.add("basemap_selector")
return map._basemap_selector
return [
- toolbar.Toolbar.Item(
+ toolbar.ToolbarItem(
icon="map",
tooltip="Basemap selector",
callback=basemap_tool_callback,
- reset=False,
),
- toolbar.Toolbar.Item(
- icon="info",
+ toolbar.ToolbarItem(
+ icon="point_scan",
tooltip="Inspector",
callback=inspector_tool_callback,
- reset=False,
),
- toolbar.Toolbar.Item(
- icon="question", tooltip="Get help", callback=self._open_help_page
+ toolbar.ToolbarItem(
+ icon="question_mark",
+ tooltip="Get help",
+ callback=self._open_help_page,
+ reset=True,
),
]
- def _toolbar_extra_tools(self) -> Optional[List[toolbar.Toolbar.Item]]:
+ def _toolbar_extra_tools(self) -> List[toolbar.ToolbarItem]:
"""Gets the extra tools for the toolbar.
Returns:
- Optional[List[toolbar.Toolbar.Item]]: The extra tools for the toolbar.
+ List[toolbar.ToolbarItem]: The extra tools for the toolbar.
"""
- return None
+ return []
def _control_config(self) -> Dict[str, List[str]]:
"""Gets the control configuration.
diff --git a/geemap/toolbar.py b/geemap/toolbar.py
index 0415ddf1ef..807f166f25 100644
--- a/geemap/toolbar.py
+++ b/geemap/toolbar.py
@@ -9,12 +9,16 @@
# *******************************************************************************#
import os
+import pathlib
from dataclasses import dataclass
+
+import anywidget
import ee
import ipyevents
import ipyleaflet
import ipywidgets as widgets
+import traitlets
from ipyfilechooser import FileChooser
from IPython.core.display import display
@@ -27,211 +31,137 @@
@map_widgets.Theme.apply
-class Toolbar(widgets.VBox):
- """A toolbar that can be added to the map."""
-
- @dataclass
- class Item:
- """A representation of an item in the toolbar.
+class ToolbarItem(anywidget.AnyWidget):
+ """A toolbar item widget for geemap."""
+
+ _esm = pathlib.Path(__file__).parent / "static" / "toolbar_item.js"
+ active = traitlets.Bool(False).tag(sync=True)
+ icon = traitlets.Unicode("").tag(sync=True)
+ # Note: "tooltip" is already defined on ipywidgets.Widget.
+ tooltip_text = traitlets.Unicode("").tag(sync=True)
+
+ def __init__(
+ self,
+ icon: str,
+ tooltip: str,
+ callback: Callable[[Any, bool, Any], None],
+ control: Optional[widgets.Widget] = None,
+ reset=False,
+ active=False,
+ ):
+ """A togglable, toolbar item.
- Attributes:
- icon: The icon to use for the item, from https://fontawesome.com/icons.
+ Args:
+ icon (str): The icon name to use, from https://fonts.google.com/icons.
tooltip: The tooltip text to show a user on hover.
callback: A callback function to execute when the item icon is clicked.
Its signature should be `callback(map, selected, item)`, where
`map` is the host map, `selected` is a boolean indicating if the
user selected or unselected the tool, and `item` is this object.
- reset: Whether to reset the selection after the callback has finished.
control: The control widget associated with this item. Used to
cleanup state when toggled off.
- toggle_button: The toggle button controlling the item.
+ reset: Whether to reset the selection after the callback has finished.
+ active: Whether the tool is currently active.
"""
+ super().__init__()
+ self.icon = icon
+ self.tooltip_text = tooltip
+ self.callback = callback
+ self.callback_wrapper = lambda *args: None
+ self.control = control
+ self.reset = reset
+ self.active = active
+
+ def toggle_off(self):
+ if self.active:
+ self.active = False
+
+ @traitlets.observe("active")
+ def _observe_value(self, change: Dict[str, Any]) -> None:
+ if (value := change.get("new")) is not None:
+ self.callback_wrapper(self.callback, value, self)
+ if self.active and self.reset:
+ self.active = False
- icon: str
- tooltip: str
- callback: Callable[[Any, bool, Any], None]
- reset: bool = True
- control: Optional[widgets.Widget] = None
- toggle_button: Optional[widgets.ToggleButton] = None
-
- def toggle_off(self):
- if self.toggle_button:
- self.toggle_button.value = False
- ICON_WIDTH = "32px"
- ICON_HEIGHT = "32px"
- NUM_COLS = 3
-
- _TOGGLE_TOOL_EXPAND_ICON = "plus"
- _TOGGLE_TOOL_EXPAND_TOOLTIP = "Expand toolbar"
- _TOGGLE_TOOL_COLLAPSE_ICON = "minus"
- _TOGGLE_TOOL_COLLAPSE_TOOLTIP = "Collapse toolbar"
+@map_widgets.Theme.apply
+class Toolbar(anywidget.AnyWidget):
+ """A toolbar that can be added to the map."""
- def __init__(self, host_map, main_tools, extra_tools=None):
+ _esm = pathlib.Path(__file__).parent / "static" / "toolbar.js"
+
+ # The accessory widget.
+ accessory_widgets = map_widgets.TypedTuple(
+ trait=traitlets.Instance(widgets.Widget),
+ help="The accessory widget",
+ ).tag(sync=True, **widgets.widget_serialization)
+
+ # The list of main tools.
+ main_tools = map_widgets.TypedTuple(
+ trait=traitlets.Instance(widgets.Widget),
+ help="List of main tools",
+ ).tag(sync=True, **widgets.widget_serialization)
+
+ # The list of extra tools.
+ extra_tools = map_widgets.TypedTuple(
+ trait=traitlets.Instance(widgets.Widget),
+ help="List of extra tools",
+ ).tag(sync=True, **widgets.widget_serialization)
+
+ # Whether the toolbar is expanded.
+ expanded = traitlets.Bool(False).tag(sync=True)
+
+ # The currently selected tab.
+ tab_index = traitlets.Int(0).tag(sync=True)
+
+ _TOGGLE_EXPAND_ICON = "add"
+ _TOGGLE_EXPAND_TOOLTIP = "Expand toolbar"
+ _TOGGLE_COLLAPSE_ICON = "remove"
+ _TOGGLE_COLLAPSE_TOOLTIP = "Collapse toolbar"
+
+ def __init__(
+ self,
+ host_map: "geemap.Map",
+ main_tools: List[ToolbarItem],
+ extra_tools: List[ToolbarItem],
+ accessory_widgets: List[widgets.DOMWidget],
+ ):
"""Adds a toolbar with `main_tools` and `extra_tools` to the `host_map`."""
+ super().__init__()
if not main_tools:
raise ValueError("A toolbar cannot be initialized without `main_tools`.")
self.host_map = host_map
- self.toggle_tool = Toolbar.Item(
- icon=self._TOGGLE_TOOL_EXPAND_ICON,
- tooltip=self._TOGGLE_TOOL_EXPAND_TOOLTIP,
+ self.toggle_widget = ToolbarItem(
+ icon=self._TOGGLE_EXPAND_ICON,
+ tooltip=self._TOGGLE_EXPAND_TOOLTIP,
callback=self._toggle_callback,
+ reset=True,
)
- self.on_layers_toggled = None
- self._accessory_widget = None
-
- if extra_tools:
- all_tools = main_tools + [self.toggle_tool] + extra_tools
- else:
- all_tools = main_tools
- icons = [tool.icon for tool in all_tools]
- tooltips = [tool.tooltip for tool in all_tools]
- callbacks = [tool.callback for tool in all_tools]
- resets = [tool.reset for tool in all_tools]
- self.num_collapsed_tools = len(main_tools) + (1 if extra_tools else 0)
- # -(-a//b) is the same as math.ceil(a/b)
- self.num_rows_expanded = -(-len(all_tools) // self.NUM_COLS)
- self.num_rows_collapsed = -(-self.num_collapsed_tools // self.NUM_COLS)
-
- self.all_widgets = [
- widgets.ToggleButton(
- layout=widgets.Layout(
- width="auto", height="auto", padding="0px 0px 0px 4px"
- ),
- button_style="primary",
- icon=icons[i],
- tooltip=tooltips[i],
- )
- for i in range(len(all_tools))
- ]
- self.toggle_widget = self.all_widgets[len(main_tools)] if extra_tools else None
-
- # We start with a collapsed grid of just the main tools and the toggle one.
- self.grid = widgets.GridBox(
- children=self.all_widgets[: self.num_collapsed_tools],
- layout=widgets.Layout(
- width="109px",
- grid_template_columns=(self.ICON_WIDTH + " ") * self.NUM_COLS,
- grid_template_rows=(self.ICON_HEIGHT + " ") * self.num_rows_collapsed,
- grid_gap="1px 1px",
- padding="5px",
- ),
- )
-
- def curry_callback(callback, should_reset_after, widget, item):
- def returned_callback(change):
- if change["type"] != "change":
- return
- callback(self.host_map, change["new"], item)
- if should_reset_after:
- widget.value = False
-
- return returned_callback
-
- for id, widget in enumerate(self.all_widgets):
- all_tools[id].toggle_button = widget
- widget.observe(
- curry_callback(callbacks[id], resets[id], widget, all_tools[id]),
- "value",
- )
-
- self.toolbar_button = widgets.ToggleButton(
- value=False,
- tooltip="Toolbar",
- icon="wrench",
- layout=widgets.Layout(
- width="28px", height="28px", padding="0px 0px 0px 4px"
- ),
- )
-
- self.layers_button = widgets.ToggleButton(
- value=False,
- tooltip="Layers",
- icon="server",
- layout=widgets.Layout(height="28px", width="72px"),
- )
-
- self.toolbar_header = widgets.HBox(
- layout=widgets.Layout(
- display="flex", justify_content="flex-end", align_items="center"
+ self.main_tools = main_tools + ([self.toggle_widget] if extra_tools else [])
+ self.extra_tools = extra_tools
+ for widget in self.main_tools + self.extra_tools:
+ widget.callback_wrapper = lambda callback, value, tool: callback(
+ self.host_map, value, tool
)
- )
- self.toolbar_header.children = [self.layers_button, self.toolbar_button]
- self.toolbar_footer = widgets.VBox()
- self.toolbar_footer.children = [self.grid]
-
- self.toolbar_button.observe(self._toolbar_btn_click, "value")
- self.layers_button.observe(self._layers_btn_click, "value")
-
- super().__init__(children=[self.toolbar_header])
+ self.accessory_widgets = accessory_widgets
def reset(self):
"""Resets the toolbar so that no widget is selected."""
- for widget in self.all_widgets:
+ for widget in self.main_tools + self.extra_tools:
widget.value = False
- def toggle_layers(self, enabled):
- self.layers_button.value = enabled
- self.on_layers_toggled(enabled)
- if enabled:
- self.toolbar_button.value = False
-
- def _reset_others(self, current):
- for other in self.all_widgets:
- if other is not current:
- other.value = False
-
def _toggle_callback(self, m, selected, item):
del m, item # unused
if not selected:
return
- if self.toggle_widget.icon == self._TOGGLE_TOOL_EXPAND_ICON:
- self.grid.layout.grid_template_rows = (
- self.ICON_HEIGHT + " "
- ) * self.num_rows_expanded
- self.grid.children = self.all_widgets
- self.toggle_widget.tooltip = self._TOGGLE_TOOL_COLLAPSE_TOOLTIP
- self.toggle_widget.icon = self._TOGGLE_TOOL_COLLAPSE_ICON
- elif self.toggle_widget.icon == self._TOGGLE_TOOL_COLLAPSE_ICON:
- self.grid.layout.grid_template_rows = (
- self.ICON_HEIGHT + " "
- ) * self.num_rows_collapsed
- self.grid.children = self.all_widgets[: self.num_collapsed_tools]
- self.toggle_widget.tooltip = self._TOGGLE_TOOL_EXPAND_TOOLTIP
- self.toggle_widget.icon = self._TOGGLE_TOOL_EXPAND_ICON
-
- def _toolbar_btn_click(self, change):
- if change["new"]:
- self.layers_button.value = False
- self.children = [self.toolbar_header, self.toolbar_footer]
- else:
- if not self.layers_button.value:
- self.children = [self.toolbar_header]
-
- def _layers_btn_click(self, change):
- # Allow callbacks to set accessory_widget to prevent flicker on click.
- if self.on_layers_toggled:
- self.on_layers_toggled(change["new"])
- if change["new"]:
- self.toolbar_button.value = False
- self.children = [self.toolbar_header, self.toolbar_footer]
- else:
- if not self.toolbar_button.value:
- self.children = [self.toolbar_header]
-
- @property
- def accessory_widget(self):
- """A widget that temporarily replaces the tool grid."""
- return self._accessory_widget
-
- @accessory_widget.setter
- def accessory_widget(self, value):
- """Sets the widget that temporarily replaces the tool grid."""
- self._accessory_widget = value
- if self._accessory_widget:
- self.toolbar_footer.children = [self._accessory_widget]
- else:
- self.toolbar_footer.children = [self.grid]
+ if self.toggle_widget.icon == self._TOGGLE_EXPAND_ICON:
+ self.expanded = True
+ self.toggle_widget.tooltip_text = self._TOGGLE_COLLAPSE_TOOLTIP
+ self.toggle_widget.icon = self._TOGGLE_COLLAPSE_ICON
+ elif self.toggle_widget.icon == self._TOGGLE_COLLAPSE_ICON:
+ self.expanded = False
+ self.toggle_widget.tooltip_text = self._TOGGLE_EXPAND_TOOLTIP
+ self.toggle_widget.icon = self._TOGGLE_EXPAND_ICON
def inspector_gui(m=None):
@@ -608,7 +538,6 @@ def ee_plot_gui(m, position="topright", **kwargs):
m (object): geemap.Map.
position (str, optional): Position of the widget. Defaults to "topright".
"""
-
close_btn = widgets.Button(
icon="times",
tooltip="Close the plot widget",
@@ -722,7 +651,7 @@ def handle_interaction(**kwargs):
dict_values = dict(zip(b_names, [dict_values_tmp[b] for b in b_names]))
generate_chart(dict_values, latlon)
except Exception as e:
- if hasattr(m, "_plot_widget"):
+ if hasattr(m, "_plot_widget") and m._plot_widget is not None:
m._plot_widget.clear_output()
with m._plot_widget:
print("No data for the clicked location.")
@@ -4492,101 +4421,90 @@ def _cog_stac_inspector_callback(map, selected, item):
main_tools = [
- Toolbar.Item(
- icon="info",
+ ToolbarItem(
+ icon="point_scan",
tooltip="Inspector",
callback=_inspector_tool_callback,
- reset=False,
),
- Toolbar.Item(
- icon="bar-chart",
+ ToolbarItem(
+ icon="bar_chart",
tooltip="Plotting",
callback=_plotting_tool_callback,
- reset=False,
),
- Toolbar.Item(
- icon="globe",
+ ToolbarItem(
+ icon="history",
tooltip="Create timelapse",
callback=_timelapse_tool_callback,
- reset=False,
),
- Toolbar.Item(
+ ToolbarItem(
icon="map",
tooltip="Change basemap",
callback=_basemap_tool_callback,
- reset=False,
),
- Toolbar.Item(
- icon="retweet",
+ ToolbarItem(
+ icon="code",
tooltip="Convert Earth Engine JavaScript to Python",
callback=_convert_js_tool_callback,
- reset=False,
),
]
extra_tools = [
- Toolbar.Item(
- icon="eraser",
+ ToolbarItem(
+ icon="ink_eraser",
tooltip="Remove all drawn features",
callback=lambda m, selected, _: m.remove_drawn_features() if selected else None,
+ reset=True,
),
- Toolbar.Item(
- icon="folder-open",
+ ToolbarItem(
+ icon="upload",
tooltip="Open local vector/raster data",
callback=_open_data_tool_callback,
- reset=False,
),
- Toolbar.Item(
- icon="gears",
+ ToolbarItem(
+ icon="manufacturing",
tooltip="WhiteboxTools for local geoprocessing",
callback=_whitebox_tool_callback,
- reset=False,
),
- Toolbar.Item(
- icon="google",
+ ToolbarItem(
+ icon="dns",
tooltip="GEE Toolbox for cloud computing",
callback=_gee_toolbox_tool_callback,
- reset=False,
),
- Toolbar.Item(
- icon="fast-forward",
+ ToolbarItem(
+ icon="fast_forward",
tooltip="Activate timeslider",
callback=_time_slider_tool_callback,
- reset=False,
),
- Toolbar.Item(
- icon="hand-o-up",
+ ToolbarItem(
+ icon="pan_tool_alt",
tooltip="Collect training samples",
callback=_collect_samples_tool_callback,
- reset=False,
),
- Toolbar.Item(
- icon="line-chart",
+ ToolbarItem(
+ icon="show_chart",
tooltip="Creating and plotting transects",
callback=_plot_transect_tool_callback,
- reset=False,
),
- Toolbar.Item(
- icon="random",
+ ToolbarItem(
+ icon="shuffle",
tooltip="Sankey plots",
callback=_sankee_tool_callback,
- reset=False,
),
- Toolbar.Item(
- icon="adjust",
+ ToolbarItem(
+ icon="image",
tooltip="Planet imagery",
callback=_split_basemaps_tool_callback,
),
- Toolbar.Item(
- icon="info-circle",
+ ToolbarItem(
+ icon="target",
tooltip="Get COG/STAC pixel value",
callback=_cog_stac_inspector_callback,
- reset=False,
),
- Toolbar.Item(
- icon="question",
+ ToolbarItem(
+ icon="question_mark",
tooltip="Get help",
callback=_open_help_page_callback,
+ reset=True,
),
]
diff --git a/js/basemap_selector.ts b/js/basemap_selector.ts
index 6f38f0a66e..e873749a28 100644
--- a/js/basemap_selector.ts
+++ b/js/basemap_selector.ts
@@ -52,7 +52,7 @@ export class BasemapSelector extends LitWidget<
@property({ type: Array }) basemaps: string[] = [];
@property({ type: String }) value: string = "";
- @query('select') selectElement!: HTMLSelectElement;
+ @query('select') selectElement!: HTMLSelectElement|null;
render(): TemplateResult {
return html`
diff --git a/js/ipywidgets_styles.ts b/js/ipywidgets_styles.ts
index fdb8162293..cb6c215263 100644
--- a/js/ipywidgets_styles.ts
+++ b/js/ipywidgets_styles.ts
@@ -57,6 +57,24 @@ export const legacyStyles = css`
line-height: var(--jp-widgets-inline-height);
}
+ .legacy-button.active {
+ background-color: var(--colab-primary-surface-color, --jp-layout-color3);
+ color: var(--jp-ui-font-color1);
+ box-shadow: 0 4px 5px 0 rgba(0, 0, 0, var(--md-shadow-key-penumbra-opacity)),
+ 0 1px 10px 0 rgba(0, 0, 0, var(--md-shadow-ambient-shadow-opacity)),
+ 0 2px 4px -1px rgba(0, 0, 0, var(--md-shadow-key-umbra-opacity));
+ }
+
+ .legacy-button.primary {
+ background-color: var(--jp-brand-color1);
+ color: var(--jp-ui-inverse-font-color1);
+ }
+
+ .legacy-button.primary.active {
+ background-color: var(--jp-brand-color0);
+ color: var(--jp-ui-inverse-font-color0);
+ }
+
.legacy-select {
-moz-appearance: none;
-webkit-appearance: none;
@@ -79,6 +97,5 @@ export const legacyStyles = css`
padding-left: calc(var(--jp-widgets-input-padding)* 2);
padding-right: 20px;
vertical-align: top;
-}
}
`;
diff --git a/js/layer_manager_row.ts b/js/layer_manager_row.ts
index bc2d4dbbca..33bcae4115 100644
--- a/js/layer_manager_row.ts
+++ b/js/layer_manager_row.ts
@@ -153,7 +153,7 @@ export class LayerManagerRow extends LitWidget<
class="legacy-button row-button settings-button"
@click="${this.onSettingsClicked}"
>
-
+ settings
diff --git a/js/tab_panel.ts b/js/tab_panel.ts
new file mode 100644
index 0000000000..cfddc70d0b
--- /dev/null
+++ b/js/tab_panel.ts
@@ -0,0 +1,183 @@
+import { html, css, nothing, LitElement, PropertyValues } from "lit";
+import { property, queryAll, queryAssignedElements } from "lit/decorators.js";
+import { legacyStyles } from "./ipywidgets_styles";
+import { classMap } from "lit/directives/class-map.js";
+import { materialStyles } from "./styles";
+import { styleMap } from "lit/directives/style-map.js";
+
+function convertToId(name: string | undefined): string {
+ return (name || "").trim().replace(" ", "-").toLowerCase();
+}
+
+/** The various modes. */
+export enum TabMode {
+ ALWAYS_SHOW,
+ HIDE_ON_SECOND_CLICK,
+}
+
+/** The tab configuration, as a string or Material Icon. */
+export interface Tab {
+ name: string | undefined,
+ icon: string | undefined,
+ width: number | undefined,
+}
+
+/**
+ * Defines the element which accepts N children, with a zero-based
+ * index determining which child to show, e.g.:
+ *
+ * Show when index is 0
+ * Show when index is 1
+ * Show when index is 2
+ *
+ */
+export class TabPanel extends LitElement {
+ static get componentName() {
+ return `tab-panel`;
+ }
+
+ static styles = [
+ legacyStyles,
+ materialStyles,
+ css`
+ ::slotted(*) {
+ display: none;
+ }
+
+ ::slotted(.show-tab) {
+ display: block;
+ }
+
+ .container {
+ padding: 0;
+ width: 100%;
+ }
+
+ .tab-container {
+ align-items: center;
+ display: flex;
+ flex-direction: row;
+ justify-content: flex-end;
+ }
+
+ .tab-container button {
+ border-radius: 5px;
+ height: 28px;
+ margin: 2px;
+ user-select: none;
+ }
+
+ .tab-container button.icon {
+ font-size: 16px;
+ width: 28px;
+ }
+
+ .tab-container button.name {
+ padding: 0 8px;
+ }
+ `,
+ ];
+
+ @property({ type: Array })
+ tabs: Tab[] = [];
+
+ @property({ type: Number })
+ index = 0;
+
+ @property({ type: Number })
+ mode: TabMode = TabMode.HIDE_ON_SECOND_CLICK;
+
+ /**
+ * The tab elements.
+ */
+ @queryAll(".tab") tabElements!: HTMLDivElement[];
+
+ /**
+ * The tab content element to show at a given index. Note that child
+ * elements are set to display block or none based on the index, and
+ * top-level text elements are ignored.
+ */
+ @queryAssignedElements() tabContentElements!: HTMLElement[];
+
+ render() {
+ return html`
+
+
+ ${this.renderTabs()}
+
+
+
+ `;
+ }
+
+ override update(changedProperties: PropertyValues) {
+ super.update(changedProperties);
+ if (changedProperties.has("index") && changedProperties.get("index") != null) {
+ this.updateSlotChildren();
+ }
+ }
+
+ private updateSlotChildren() {
+ if (!this.tabContentElements) {
+ return;
+ }
+ // Show the element at the current index.
+ this.tabContentElements.forEach((element: HTMLElement, i: number) => {
+ element.classList.remove("show-tab");
+
+ // Also add accessibility attributes.
+ const id = convertToId(this.tabs[i].name);
+ element.setAttribute("id", `tabpanel-${id}-${i}`);
+ element.setAttribute("role", "tabpanel");
+ element.setAttribute("aria-labelledby", `tab-${id}-${i}`);
+ });
+ this.tabContentElements[this.index]?.classList.add("show-tab");
+ }
+
+ private renderTabs() {
+ return this.tabs.map((tab: Tab, i: number) => {
+ const id = convertToId(this.tabs[i].name);
+ return html``;
+ });
+ }
+
+ private onTabClick(index: number) {
+ switch (this.mode) {
+ case TabMode.HIDE_ON_SECOND_CLICK:
+ // Hide the tab panel if clicked twice.
+ this.index = this.index === index ? -1 : index;
+ break;
+ case TabMode.ALWAYS_SHOW:
+ default:
+ this.index = index;
+ }
+ this.dispatchEvent(new CustomEvent("tab-changed", {
+ detail: index,
+ }));
+ }
+}
+
+// Without this check, there's a component registry issue when developing locally.
+if (!customElements.get(TabPanel.componentName)) {
+ customElements.define(TabPanel.componentName, TabPanel);
+}
diff --git a/js/toolbar.ts b/js/toolbar.ts
new file mode 100644
index 0000000000..7f3bd6f34d
--- /dev/null
+++ b/js/toolbar.ts
@@ -0,0 +1,141 @@
+import type { RenderProps } from "@anywidget/types";
+import { html, css } from "lit";
+import { property } from "lit/decorators.js";
+import { classMap } from "lit/directives/class-map.js";
+
+import { legacyStyles } from "./ipywidgets_styles";
+import { LitWidget } from "./lit_widget";
+import { materialStyles } from "./styles";
+import { loadFonts, updateChildren } from "./utils";
+
+import "./tab_panel";
+
+
+export interface ToolbarModel {
+ accessory_widgets: any;
+ main_tools: any;
+ extra_tools: any
+ expanded: boolean;
+ tab_index: number;
+}
+
+export class Toolbar extends LitWidget<
+ ToolbarModel,
+ Toolbar
+> {
+ static get componentName() {
+ return `toolbar-panel`;
+ }
+
+ static styles = [
+ legacyStyles,
+ materialStyles,
+ css`
+ .hide {
+ display: none;
+ }
+
+ .expanded {
+ display: block; !important
+ }
+
+ .tools-container {
+ padding: 4px;
+ }
+
+ slot[name="extra-tools"] {
+ margin-top: 4px;
+ }
+
+ ::slotted([slot="main-tools"]),
+ ::slotted([slot="extra-tools"]) {
+ align-items: center;
+ display: inline-grid;
+ grid-template-columns: auto auto auto;
+ grid-gap: 4px;
+ justify-items: center;
+ }
+ `,
+ ];
+
+ modelNameToViewName(): Map {
+ return new Map([
+ ["accessory_widgets", null],
+ ["main_tools", null],
+ ["extra_tools", null],
+ ["expanded", "expanded"],
+ ["tab_index", "tab_index"],
+ ]);
+ }
+
+ @property()
+ expanded: boolean = false;
+
+ @property()
+ tab_index: number = 0;
+
+ render() {
+ return html`
+ ) => {
+ this.tab_index = e.detail;
+ }}>
+
+
+
+
+
+
+
+
+ `;
+ }
+}
+
+// Without this check, there's a component registry issue when developing locally.
+if (!customElements.get(Toolbar.componentName)) {
+ customElements.define(Toolbar.componentName, Toolbar);
+}
+
+async function render({ model, el }: RenderProps) {
+ loadFonts();
+ const manager = document.createElement(Toolbar.componentName);
+ manager.model = model;
+ el.appendChild(manager);
+
+ const accessoryWidgetEl = document.createElement("div");
+ accessoryWidgetEl.slot = "accessory-widget";
+ manager.appendChild(accessoryWidgetEl);
+
+ updateChildren(accessoryWidgetEl, model, "accessory_widgets");
+ model.on("change:accessory_widgets", () => {
+ updateChildren(accessoryWidgetEl, model, "accessory_widgets");
+ });
+
+ const mainToolsEl = document.createElement("div");
+ mainToolsEl.slot = "main-tools";
+ manager.appendChild(mainToolsEl);
+
+ updateChildren(mainToolsEl, model, "main_tools");
+ model.on("change:main_tools", () => {
+ updateChildren(mainToolsEl, model, "main_tools");
+ });
+
+ const extraToolsEl = document.createElement("div");
+ extraToolsEl.slot = "extra-tools";
+ manager.appendChild(extraToolsEl);
+
+ updateChildren(extraToolsEl, model, "extra_tools");
+ model.on("change:extra_tools", () => {
+ updateChildren(extraToolsEl, model, "extra_tools");
+ });
+}
+
+export default { render };
diff --git a/js/toolbar_item.ts b/js/toolbar_item.ts
new file mode 100644
index 0000000000..24e4fe3939
--- /dev/null
+++ b/js/toolbar_item.ts
@@ -0,0 +1,85 @@
+import type { RenderProps } from "@anywidget/types";
+import { html, css } from "lit";
+import { property } from "lit/decorators.js";
+import { classMap } from 'lit/directives/class-map.js';
+
+import { legacyStyles } from './ipywidgets_styles';
+import { LitWidget } from "./lit_widget";
+import { materialStyles } from "./styles";
+import { loadFonts } from "./utils";
+
+export interface ToolbarItemModel {
+ active: boolean;
+ icon: string;
+ // Note: "tooltip" is already used by ipywidgets.
+ tooltip_text: string;
+}
+
+export class ToolbarItem extends LitWidget {
+ static get componentName() {
+ return `tool-button`;
+ }
+
+ static styles = [
+ legacyStyles,
+ materialStyles,
+ css`
+ button {
+ font-size: 16px !important;
+ height: 32px;
+ padding: 0px 0px 0px 4px;
+ width: 32px;
+ user-select: none;
+ }
+ `,
+ ];
+
+ modelNameToViewName(): Map {
+ return new Map([
+ ["active", "active"],
+ ["icon", "icon"],
+ ["tooltip_text", "tooltip_text"],
+ ]);
+ }
+
+ @property({ type: Boolean })
+ active: boolean = false;
+
+ @property({ type: String })
+ icon: string = '';
+
+ @property({ type: String })
+ tooltip_text: string = '';
+
+ render() {
+ return html`
+ `;
+ }
+
+ private onClick(_: Event) {
+ this.active = !this.active;
+ }
+}
+
+// Without this check, there's a component registry issue when developing locally.
+if (!customElements.get(ToolbarItem.componentName)) {
+ customElements.define(ToolbarItem.componentName, ToolbarItem);
+}
+
+async function render({ model, el }: RenderProps) {
+ loadFonts();
+ const manager = document.createElement(ToolbarItem.componentName);
+ manager.model = model;
+ el.appendChild(manager);
+}
+
+export default { render };
diff --git a/js/utils.ts b/js/utils.ts
index 5431eaf7a2..c93ac022bc 100644
--- a/js/utils.ts
+++ b/js/utils.ts
@@ -24,9 +24,13 @@ export function loadFonts() {
export async function updateChildren(
container: HTMLElement,
- model: AnyModel
+ model: AnyModel,
+ property = "children",
) {
- const children = model.get("children");
+ let children = model.get(property);
+ if (!Array.isArray(children)) {
+ children = [children]
+ }
const child_models = await unpackModels(children, model.widget_manager);
const child_views = await Promise.all(
child_models.map((model) => model.widget_manager.create_view(model))
diff --git a/tests/test_core.py b/tests/test_core.py
index cb1b44feb3..1d6261dae8 100644
--- a/tests/test_core.py
+++ b/tests/test_core.py
@@ -150,44 +150,17 @@ def test_add_toolbar(self):
self.assertEqual(len(self.core_map.controls), 1)
toolbar_control = self.core_map.controls[0].widget
- # Layer manager is selected and open by default.
- layer_button = utils.query_widget(
- toolbar_control, ipywidgets.ToggleButton, lambda c: c.tooltip == "Layers"
- )
- self.assertTrue(layer_button.value)
- self.assertIsNotNone(
- utils.query_widget(toolbar_control, map_widgets.LayerManager)
- )
- toolbar_button = utils.query_widget(
- toolbar_control, ipywidgets.ToggleButton, lambda c: c.tooltip == "Toolbar"
- )
- toolbar_button.value = True # Open the grid of tools.
- self.assertFalse(layer_button.value)
-
- tool_grid = utils.query_widget(toolbar_control, ipywidgets.GridBox).children
-
- self.assertEqual(len(tool_grid), 3)
- self.assertEqual(tool_grid[0].tooltip, "Basemap selector")
- self.assertEqual(tool_grid[1].tooltip, "Inspector")
- self.assertEqual(tool_grid[2].tooltip, "Get help")
-
- # Closing the toolbar button shows both buttons in the header.
- toolbar_button.value = False
- self.assertIsNotNone(
- utils.query_widget(
- toolbar_control,
- ipywidgets.ToggleButton,
- lambda c: c.tooltip == "Toolbar",
- )
- )
- self.assertIsNotNone(
- utils.query_widget(
- toolbar_control,
- ipywidgets.ToggleButton,
- lambda c: c.tooltip == "Layers",
- )
- )
+ # Layer manager is selected and open by default.
+ self.assertEqual(toolbar_control.tab_index, 0)
+ layer_button = toolbar_control.accessory_widgets[0]
+ self.assertIsInstance(layer_button, map_widgets.LayerManager)
+
+ toolbar_control.tab_index = 1
+ self.assertEqual(len(toolbar_control.main_tools), 3)
+ self.assertEqual(toolbar_control.main_tools[0].tooltip_text, "Basemap selector")
+ self.assertEqual(toolbar_control.main_tools[1].tooltip_text, "Inspector")
+ self.assertEqual(toolbar_control.main_tools[2].tooltip_text, "Get help")
def test_add_draw_control(self):
"""Tests adding and getting the draw widget."""
diff --git a/tests/test_toolbar.py b/tests/test_toolbar.py
index aed699e1d7..e8d093a014 100644
--- a/tests/test_toolbar.py
+++ b/tests/test_toolbar.py
@@ -10,39 +10,27 @@
import ipywidgets
import geemap
-from geemap.toolbar import Toolbar, _cleanup_toolbar_item
+from geemap.toolbar import Toolbar, ToolbarItem, _cleanup_toolbar_item
from tests import fake_map, utils
class TestToolbar(unittest.TestCase):
"""Tests for the Toolbar class in the `toolbar` module."""
- def _query_layers_button(self, toolbar):
- return utils.query_widget(
- toolbar, ipywidgets.ToggleButton, lambda c: c.tooltip == "Layers"
- )
-
- def _query_open_button(self, toolbar):
- return utils.query_widget(
- toolbar, ipywidgets.ToggleButton, lambda c: c.tooltip == "Toolbar"
- )
-
- def _query_tool_grid(self, toolbar):
- return utils.query_widget(toolbar, ipywidgets.GridBox, lambda c: True)
-
def setUp(self) -> None:
self.callback_calls = 0
self.last_called_with_selected = None
self.last_called_item = None
- self.item = Toolbar.Item(
+ self.item = ToolbarItem(
icon="info", tooltip="dummy item", callback=self.dummy_callback
)
- self.no_reset_item = Toolbar.Item(
+ self.reset_item = ToolbarItem(
icon="question",
tooltip="no reset item",
callback=self.dummy_callback,
- reset=False,
+ reset=True,
)
+ self.accessory_widgets = [ipywidgets.Text()]
return super().setUp()
def tearDown(self) -> None:
@@ -57,109 +45,62 @@ def dummy_callback(self, m, selected, item):
def test_no_tools_throws(self):
map = geemap.Map(ee_initialize=False)
- self.assertRaises(ValueError, Toolbar, map, [], [])
+ self.assertRaises(ValueError, Toolbar, map, [], [], self.accessory_widgets)
def test_only_main_tools_exist_if_no_extra_tools(self):
map = geemap.Map(ee_initialize=False)
- toolbar = Toolbar(map, [self.item], [])
- self.assertIsNone(toolbar.toggle_widget)
- self.assertEqual(len(toolbar.all_widgets), 1)
- self.assertEqual(toolbar.all_widgets[0].icon, "info")
- self.assertEqual(toolbar.all_widgets[0].tooltip, "dummy item")
- self.assertFalse(toolbar.all_widgets[0].value)
- self.assertEqual(toolbar.num_rows_collapsed, 1)
- self.assertEqual(toolbar.num_rows_expanded, 1)
+ toolbar = Toolbar(map, [self.item], [], self.accessory_widgets)
+ self.assertNotIn(toolbar.toggle_widget, toolbar.main_tools)
def test_all_tools_and_toggle_exist_if_extra_tools(self):
map = geemap.Map(ee_initialize=False)
- toolbar = Toolbar(map, [self.item], [self.no_reset_item])
+ toolbar = Toolbar(map, [self.item], [self.reset_item], self.accessory_widgets)
self.assertIsNotNone(toolbar.toggle_widget)
- self.assertEqual(len(toolbar.all_widgets), 3)
- self.assertEqual(toolbar.all_widgets[2].icon, "question")
- self.assertEqual(toolbar.all_widgets[2].tooltip, "no reset item")
- self.assertFalse(toolbar.all_widgets[2].value)
- self.assertEqual(toolbar.num_rows_collapsed, 1)
- self.assertEqual(toolbar.num_rows_expanded, 1)
-
- def test_has_correct_number_of_rows(self):
- map = geemap.Map(ee_initialize=False)
- toolbar = Toolbar(map, [self.item, self.item], [self.item, self.item])
- self.assertEqual(toolbar.num_rows_collapsed, 1)
- self.assertEqual(toolbar.num_rows_expanded, 2)
def test_toggle_expands_and_collapses(self):
map = geemap.Map(ee_initialize=False)
- toolbar = Toolbar(map, [self.item], [self.no_reset_item])
- self.assertEqual(len(toolbar.grid.children), 2)
+ toolbar = Toolbar(map, [self.item], [self.reset_item], self.accessory_widgets)
self.assertIsNotNone(toolbar.toggle_widget)
- toggle = toolbar.all_widgets[1]
- self.assertEqual(toggle.icon, "plus")
- self.assertEqual(toggle.tooltip, "Expand toolbar")
+ toggle = toolbar.toggle_widget
+ self.assertEqual(toggle.icon, "add")
+ self.assertEqual(toggle.tooltip_text, "Expand toolbar")
+ self.assertFalse(toolbar.expanded)
# Expand
- toggle.value = True
- self.assertEqual(len(toolbar.grid.children), 3)
- self.assertEqual(toggle.icon, "minus")
- self.assertEqual(toggle.tooltip, "Collapse toolbar")
+ toggle.active = True
+ self.assertTrue(toolbar.expanded)
+ self.assertEqual(toggle.icon, "remove")
+ self.assertEqual(toggle.tooltip_text, "Collapse toolbar")
# After expanding, button is unselected.
- self.assertFalse(toggle.value)
+ self.assertFalse(toggle.active)
# Collapse
- toggle.value = True
- self.assertEqual(len(toolbar.grid.children), 2)
- self.assertEqual(toggle.icon, "plus")
- self.assertEqual(toggle.tooltip, "Expand toolbar")
+ toggle.active = True
+ self.assertFalse(toolbar.expanded)
+ self.assertEqual(toggle.icon, "add")
+ self.assertEqual(toggle.tooltip_text, "Expand toolbar")
# After collapsing, button is unselected.
- self.assertFalse(toggle.value)
+ self.assertFalse(toggle.active)
def test_triggers_callbacks(self):
map = geemap.Map(ee_initialize=False)
- toolbar = Toolbar(map, [self.item, self.no_reset_item])
+ toolbar = Toolbar(map, [self.item, self.reset_item], [], self.accessory_widgets)
self.assertIsNone(self.last_called_with_selected)
self.assertIsNone(self.last_called_item)
- # Select first tool, which resets.
- toolbar.all_widgets[0].value = True
- self.assertFalse(self.last_called_with_selected) # was reset by callback
- self.assertEqual(self.callback_calls, 2)
- self.assertFalse(toolbar.all_widgets[0].value)
+ # Select first tool, which does not reset.
+ toolbar.main_tools[0].active = True
+ self.assertTrue(self.last_called_with_selected)
+ self.assertEqual(self.callback_calls, 1)
+ self.assertTrue(toolbar.main_tools[0].active)
self.assertEqual(self.item, self.last_called_item)
- # Select second tool, which does not reset.
- toolbar.all_widgets[1].value = True
- self.assertTrue(self.last_called_with_selected)
+ # Select second tool, which resets.
+ toolbar.main_tools[1].active = True
+ self.assertFalse(self.last_called_with_selected) # was reset by callback
self.assertEqual(self.callback_calls, 3)
- self.assertTrue(toolbar.all_widgets[1].value)
- self.assertEqual(self.no_reset_item, self.last_called_item)
-
- def test_layers_toggle_callback(self):
- """Verifies the on_layers_toggled callback is triggered."""
- map_fake = fake_map.FakeMap()
- toolbar = Toolbar(map_fake, [self.item, self.no_reset_item])
- self._query_open_button(toolbar).value = True
-
- self.assertIsNotNone(layers_button := self._query_layers_button(toolbar))
- on_toggled_mock = Mock()
- toolbar.on_layers_toggled = on_toggled_mock
- layers_button.value = True
-
- on_toggled_mock.assert_called_once()
-
- def test_accessory_widget(self):
- """Verifies the accessory widget replaces the tool grid."""
- map_fake = fake_map.FakeMap()
- toolbar = Toolbar(map_fake, [self.item, self.no_reset_item])
- self._query_open_button(toolbar).value = True
- self.assertIsNotNone(self._query_tool_grid(toolbar))
-
- toolbar.accessory_widget = ipywidgets.ToggleButton(tooltip="test-button")
-
- self.assertIsNone(self._query_tool_grid(toolbar))
- self.assertIsNotNone(
- utils.query_widget(
- toolbar, ipywidgets.ToggleButton, lambda c: c.tooltip == "test-button"
- )
- )
+ self.assertFalse(toolbar.main_tools[1].active)
+ self.assertEqual(self.reset_item, self.last_called_item)
@dataclass
class TestWidget:
@@ -177,24 +118,24 @@ def callback(m, selected, item):
widget.selected_count += 1
return widget
- item = Toolbar.Item(
+ item = ToolbarItem(
icon="info", tooltip="dummy item", callback=callback, reset=False
)
map_fake = fake_map.FakeMap()
- toolbar = Toolbar(map_fake, [item])
- toolbar.all_widgets[0].value = True
+ toolbar = Toolbar(map_fake, [item], [], self.accessory_widgets)
+ toolbar.main_tools[0].active = True
self.assertEqual(1, widget.selected_count)
self.assertEqual(0, widget.cleanup_count)
- toolbar.all_widgets[0].value = False
+ toolbar.main_tools[0].active = False
self.assertEqual(1, widget.selected_count)
self.assertEqual(1, widget.cleanup_count)
- toolbar.all_widgets[0].value = True
+ toolbar.main_tools[0].active = True
self.assertEqual(2, widget.selected_count)
self.assertEqual(1, widget.cleanup_count)
widget.cleanup()
self.assertEqual(2, widget.selected_count)
self.assertEqual(3, widget.cleanup_count)
- self.assertFalse(toolbar.all_widgets[0].value)
+ self.assertFalse(toolbar.main_tools[0].active)