plotlymap module¶
Canvas
¶
The widgets.HBox containing the map and a toolbar.
Source code in leafmap/plotlymap.py
class Canvas:
"""The widgets.HBox containing the map and a toolbar."""
def __init__(
self,
map,
map_min_width: Optional[str] = "90%",
map_max_width: Optional[str] = "98%",
map_refresh: Optional[bool] = False,
**kwargs,
) -> None:
"""Initialize the Canvas.
Args:
map (go.FigureWidget): The map to display.
map_min_width (str, optional): The minimum width of the map. Defaults to '90%'.
map_max_width (str, optional): The maximum width of the map. Defaults to '98%'.
map_refresh (bool, optional): Whether to refresh the map when the map is resized. Defaults to False.
"""
from .toolbar import plotly_toolbar
map_widget = widgets.Output(layout=widgets.Layout(width=map_max_width))
with map_widget:
display(map)
self.map = map
self.map_min_width = map_min_width
self.map_max_width = map_max_width
self.map_refresh = map_refresh
self.map_widget = map_widget
container_widget = widgets.VBox()
self.container_widget = container_widget
toolbar_widget = plotly_toolbar(self)
sidebar_widget = widgets.VBox([toolbar_widget, container_widget])
canvas = widgets.HBox([map_widget, sidebar_widget])
self.canvas = canvas
self.toolbar_widget = toolbar_widget
def toolbar_reset(self):
"""Reset the toolbar so that no tool is selected."""
if hasattr(self, "toolbar"):
toolbar_grid = self.toolbar
for tool in toolbar_grid.children:
tool.value = False
__init__(self, map, map_min_width='90%', map_max_width='98%', map_refresh=False, **kwargs)
special
¶
Initialize the Canvas.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
map |
go.FigureWidget |
The map to display. |
required |
map_min_width |
str |
The minimum width of the map. Defaults to '90%'. |
'90%' |
map_max_width |
str |
The maximum width of the map. Defaults to '98%'. |
'98%' |
map_refresh |
bool |
Whether to refresh the map when the map is resized. Defaults to False. |
False |
Source code in leafmap/plotlymap.py
def __init__(
self,
map,
map_min_width: Optional[str] = "90%",
map_max_width: Optional[str] = "98%",
map_refresh: Optional[bool] = False,
**kwargs,
) -> None:
"""Initialize the Canvas.
Args:
map (go.FigureWidget): The map to display.
map_min_width (str, optional): The minimum width of the map. Defaults to '90%'.
map_max_width (str, optional): The maximum width of the map. Defaults to '98%'.
map_refresh (bool, optional): Whether to refresh the map when the map is resized. Defaults to False.
"""
from .toolbar import plotly_toolbar
map_widget = widgets.Output(layout=widgets.Layout(width=map_max_width))
with map_widget:
display(map)
self.map = map
self.map_min_width = map_min_width
self.map_max_width = map_max_width
self.map_refresh = map_refresh
self.map_widget = map_widget
container_widget = widgets.VBox()
self.container_widget = container_widget
toolbar_widget = plotly_toolbar(self)
sidebar_widget = widgets.VBox([toolbar_widget, container_widget])
canvas = widgets.HBox([map_widget, sidebar_widget])
self.canvas = canvas
self.toolbar_widget = toolbar_widget
toolbar_reset(self)
¶
Reset the toolbar so that no tool is selected.
Source code in leafmap/plotlymap.py
def toolbar_reset(self):
"""Reset the toolbar so that no tool is selected."""
if hasattr(self, "toolbar"):
toolbar_grid = self.toolbar
for tool in toolbar_grid.children:
tool.value = False
Map (FigureWidget)
¶
The Map class inherits the Plotly FigureWidget class. More info at https://plotly.com/python/figurewidget.
Source code in leafmap/plotlymap.py
class Map(go.FigureWidget):
"""The Map class inherits the Plotly FigureWidget class. More info at https://plotly.com/python/figurewidget."""
def __init__(
self,
center: Optional[Tuple[float, float]] = (20, 0),
zoom: Optional[float] = 1,
basemap: Optional[str] = "open-street-map",
height: Optional[int] = 600,
**kwargs,
) -> None:
"""Initializes a map. More info at https://plotly.com/python/mapbox-layers/
Args:
center (tuple, optional): Center of the map. Defaults to (20, 0).
zoom (float, optional): Zoom level of the map. Defaults to 1.
basemap (str, optional): Can be one of string from "open-street-map", "carto-positron", "carto-darkmatter", "stamen-terrain", "stamen-toner" or "stamen-watercolor" . Defaults to 'open-street-map'.
height (int, optional): Height of the map. Defaults to 600.
"""
super().__init__(**kwargs)
self.add_scattermapbox()
self.update_layout(
{
"mapbox": {
"style": basemap,
"center": {"lat": center[0], "lon": center[1]},
"zoom": zoom,
},
"margin": {"r": 0, "t": 0, "l": 0, "b": 0},
"height": height,
}
)
def show(
self,
toolbar: Optional[bool] = True,
map_min_width: Optional[str] = "91%",
map_max_width: Optional[str] = "98%",
refresh: Optional[bool] = False,
**kwargs,
) -> None:
"""Shows the map.
Args:
toolbar (bool, optional): Whether to show the toolbar. Defaults to True.
map_min_width (str, optional): The minimum width of the map. Defaults to '91%'.
map_max_width (str, optional): The maximum width of the map. Defaults to '98%'.
refresh (bool, optional): Whether to refresh the map when the map is resized. Defaults to False.
Returns:
Canvas: Map canvas.
"""
if not toolbar:
super().show(**kwargs)
else:
canvas = Canvas(
self,
map_min_width=map_min_width,
map_max_width=map_max_width,
map_refresh=refresh,
)
return canvas.canvas
def clear_controls(self):
"""Removes all controls from the map."""
config = {
"scrollZoom": True,
"displayModeBar": False,
"editable": True,
"showLink": False,
"displaylogo": False,
}
self.show(toolbar=False, config=config)
def add_controls(self, controls: List) -> None:
"""Adds controls to the map.
Args:
controls (list): List of controls to add, e.g., ['drawline', 'drawopenpath', 'drawclosedpath', 'drawcircle', 'drawrect', 'eraseshape'] See https://bit.ly/33Tmqxr
"""
if isinstance(controls, str):
controls = [controls]
elif not isinstance(controls, list):
raise ValueError(
"Controls must be a string or a list of strings. See https://bit.ly/33Tmqxr"
)
self.update_layout(modebar_add=controls)
def remove_controls(self, controls: List):
"""Removes controls to the map.
Args:
controls (list): List of controls to remove, e.g., ["zoomin", "zoomout", "toimage", "pan", "resetview"]. See https://bit.ly/3Jk7wkb
"""
if isinstance(controls, str):
controls = [controls]
elif not isinstance(controls, list):
raise ValueError(
"Controls must be a string or a list of strings. See https://bit.ly/3Jk7wkb"
)
self.update_layout(modebar_remove=controls)
def set_center(self, lat: float, lon: float, zoom: Optional[float] = None) -> None:
"""Sets the center of the map.
Args:
lat (float): Latitude.
lon (float): Longitude.
zoom (int, optional): Zoom level of the map. Defaults to None.
"""
self.update_layout(
mapbox=dict(
center=dict(lat=lat, lon=lon),
zoom=zoom if zoom is not None else self.layout.mapbox.zoom,
)
)
def add_basemap(self, basemap: Optional[str] = "ROADMAP") -> None:
"""Adds a basemap to the map.
Args:
basemap (str, optional): Can be one of string from basemaps. Defaults to 'ROADMAP'.
"""
if basemap not in basemaps:
raise ValueError(
f"Basemap {basemap} not found. Choose from {','.join(basemaps.keys())}"
)
if basemap in self.get_tile_layers():
self.remove_basemap(basemap)
layers = list(self.layout.mapbox.layers) + [basemaps[basemap]]
self.update_layout(mapbox_layers=layers)
def remove_basemap(self, name: str) -> None:
"""Removes a basemap from the map.
Args:
name (str): Name of the basemap to remove.
"""
layers = list(self.layout.mapbox.layers)
layers = [layer for layer in layers if layer["name"] != name]
self.layout.mapbox.layers = layers
def add_mapbox_layer(self, style: Union[str, Dict], access_token=None) -> None:
"""Adds a mapbox layer to the map.
Args:
layer (str | dict): Layer to add. Can be "basic", "streets", "outdoors", "light", "dark", "satellite", or "satellite-streets". See https://plotly.com/python/mapbox-layers/ and https://docs.mapbox.com/mapbox-gl-js/style-spec/
access_token (str, optional): The Mapbox Access token. It can be set as an environment variable "MAPBOX_TOKEN". Defaults to None.
"""
if access_token is None:
access_token = os.environ.get("MAPBOX_TOKEN")
self.update_layout(
mapbox_style=style, mapbox_layers=[], mapbox_accesstoken=access_token
)
def add_layer(self, layer, name: Optional[str] = None, **kwargs):
"""Adds a layer to the map.
Args:
layer (plotly.graph_objects): Layer to add.
name (str, optional): Name of the layer. Defaults to None.
"""
if isinstance(name, str):
layer.name = name
self.add_trace(layer, **kwargs)
def remove_layer(self, name: str) -> None:
"""Removes a layer from the map.
Args:
name (str): Name of the layer to remove.
"""
if name in self.get_data_layers():
self.data = [layer for layer in self.data if layer.name != name]
elif name in self.get_tile_layers():
self.layout.mapbox.layers = [
layer for layer in self.layout.mapbox.layers if layer["name"] != name
]
def clear_layers(self, clear_basemap: Optional[bool] = False):
"""Clears all layers from the map.
Args:
clear_basemap (bool, optional): If True, clears the basemap. Defaults to False.
"""
if clear_basemap:
self.data = []
else:
if len(self.data) > 1:
self.data = self.data[:1]
def get_layers(self) -> Dict:
"""Returns a dictionary of all layers in the map.
Returns:
dict: A dictionary of all layers in the map.
"""
layers = {}
for layer in self.layout.mapbox.layers:
if layer["name"] is not None:
layers[layer["name"]] = layer
for layer in self.data:
if layer.name is not None and layer.name != "trace 0":
layers[layer.name] = layer
return layers
def get_tile_layers(self) -> Dict:
"""Returns a dictionary of tile layers in the map.
Returns:
dict: A dictionary of tile layers in the map.
"""
layers = {}
for layer in self.layout.mapbox.layers:
if layer["name"] is not None:
layers[layer["name"]] = layer
return layers
def get_data_layers(self) -> Dict:
"""Returns a dictionary of data layers in the map.
Returns:
dict: A dictionary of data layers in the map.
"""
layers = {}
for layer in self.data:
if layer.name is not None and layer.name != "trace 0":
layers[layer.name] = layer
return layers
def find_layer_index(self, name: str) -> int:
"""Finds the index of a layer.
Args:
name (str): Name of the layer to find.
Returns:
int: Index of the layer.
"""
for i, layer in enumerate(self.data):
if layer.name == name:
return i
for i, layer in enumerate(self.layout.mapbox.layers):
if layer["name"] == name:
return i
return None
def set_layer_visibility(self, name: str, show: Optional[bool] = True) -> None:
"""Sets the visibility of a layer.
Args:
name (str): Name of the layer to set.
show (bool, optional): If True, shows the layer. Defaults to True.
"""
if name in self.get_tile_layers():
index = self.find_layer_index(name)
self.layout.mapbox.layers[index].visible = show
elif name in self.get_data_layers():
index = self.find_layer_index(name)
self.data[index].visible = show
else:
print(f"Layer {name} not found.")
def set_layer_opacity(self, name: str, opacity: Optional[float] = 1) -> None:
"""Sets the visibility of a layer.
Args:
name (str): Name of the layer to set.
opacity (float, optional): Opacity of the layer. Defaults to 1.
"""
if name in self.get_tile_layers():
index = self.find_layer_index(name)
self.layout.mapbox.layers[index].opacity = opacity
elif name in self.get_data_layers():
index = self.find_layer_index(name)
layer = self.data[index]
if hasattr(layer, "opacity"):
layer.opacity = opacity
elif hasattr(layer, "marker"):
layer.marker.opacity = opacity
else:
print(f"Layer {name} not found.")
def add_tile_layer(
self,
url: str,
name: Optional[str] = "TileLayer",
attribution: Optional[str] = "",
opacity: Optional[float] = 1.0,
**kwargs,
) -> None:
"""Adds a TileLayer to the map.
Args:
url (str): The URL of the tile layer.
name (str, optional): Name of the layer. Defaults to 'TileLayer'.
attribution (str): The attribution to use. Defaults to "".
opacity (float, optional): The opacity of the layer. Defaults to 1.
"""
layer = {
"below": "traces",
"sourcetype": "raster",
"sourceattribution": attribution,
"source": [url],
"opacity": opacity,
"name": name,
}
layers = list(self.layout.mapbox.layers) + [layer]
self.update_layout(mapbox_layers=layers)
def add_cog_layer(
self,
url: str,
name: Optional[str] = "Untitled",
attribution: Optional[str] = "",
opacity: Optional[float] = 1.0,
bands: Optional[List] = None,
titiler_endpoint: Optional[str] = None,
**kwargs,
) -> None:
"""Adds a COG TileLayer to the map.
Args:
url (str): The URL of the COG tile layer, e.g., 'https://github.com/opengeos/data/releases/download/raster/Libya-2023-07-01.tif'
name (str, optional): The layer name to use for the layer. Defaults to 'Untitled'.
attribution (str, optional): The attribution to use. Defaults to ''.
opacity (float, optional): The opacity of the layer. Defaults to 1.
bands (list, optional): The bands to use. Defaults to None.
titiler_endpoint (str, optional): Titiler endpoint. Defaults to "https://titiler.xyz".
**kwargs: Arbitrary keyword arguments, including bidx, expression, nodata, unscale, resampling, rescale,
color_formula, colormap, colormap_name, return_mask. See https://developmentseed.org/titiler/endpoints/cog/
and https://cogeotiff.github.io/rio-tiler/colormap/. To select a certain bands, use bidx=[1, 2, 3].
apply a rescaling to multiple bands, use something like `rescale=["164,223","130,211","99,212"]`.
"""
tile_url = cog_tile(url, bands, titiler_endpoint, **kwargs)
center = cog_center(url, titiler_endpoint) # (lon, lat)
self.add_tile_layer(tile_url, name, attribution, opacity)
self.set_center(lon=center[0], lat=center[1], zoom=10)
def add_stac_layer(
self,
url: str = None,
collection: str = None,
items: str = None,
assets: str = None,
bands: List = None,
titiler_endpoint: Optional[str] = None,
name: Optional[str] = "STAC Layer",
attribution: Optional[str] = "",
opacity: Optional[float] = 1.0,
**kwargs,
) -> None:
"""Adds a STAC TileLayer to the map.
Args:
url (str): HTTP URL to a STAC item, e.g., https://canada-spot-ortho.s3.amazonaws.com/canada_spot_orthoimages/canada_spot5_orthoimages/S5_2007/S5_11055_6057_20070622/S5_11055_6057_20070622.json
collection (str): The Microsoft Planetary Computer STAC collection ID, e.g., landsat-8-c2-l2.
items (str): The Microsoft Planetary Computer STAC item ID, e.g., LC08_L2SP_047027_20201204_02_T1.
assets (str | list): The Microsoft Planetary Computer STAC asset ID, e.g., ["SR_B7", "SR_B5", "SR_B4"].
bands (list): A list of band names, e.g., ["SR_B7", "SR_B5", "SR_B4"]
titiler_endpoint (str, optional): Titiler endpoint, e.g., "https://titiler.xyz", "planetary-computer", "pc". Defaults to None.
name (str, optional): The layer name to use for the layer. Defaults to 'STAC Layer'.
attribution (str, optional): The attribution to use. Defaults to ''.
opacity (float, optional): The opacity of the layer. Defaults to 1.
"""
tile_url = stac_tile(
url, collection, item, assets, bands, titiler_endpoint, **kwargs
)
center = stac_center(url, collection, item, titiler_endpoint)
self.add_tile_layer(tile_url, name, attribution, opacity)
self.set_center(lon=center[0], lat=center[1], zoom=10)
def add_mosaic_layer(
self,
url: str,
titiler_endpoint: Optional[str] = None,
name: Optional[str] = "Mosaic Layer",
attribution: Optional[str] = "",
opacity: Optional[float] = 1.0,
**kwargs,
) -> None:
"""Adds a STAC TileLayer to the map.
Args:
url (str): HTTP URL to a MosaicJSON.
titiler_endpoint (str, optional): Titiler endpoint, e.g., "https://titiler.xyz". Defaults to None.
name (str, optional): The layer name to use for the layer. Defaults to 'Mosaic Layer'.
attribution (str, optional): The attribution to use. Defaults to ''.
opacity (float, optional): The opacity of the layer. Defaults to 1.
"""
tile_url = mosaic_tile(url, titiler_endpoint, **kwargs)
center = mosaic_info(url, titiler_endpoint)["center"]
self.add_tile_layer(tile_url, name, attribution, opacity)
self.set_center(lon=center[0], lat=center[1], zoom=10)
def add_planet_by_month(
self,
year: Optional[int] = 2016,
month: Optional[int] = 1,
api_key: Optional[str] = None,
token_name: Optional[str] = "PLANET_API_KEY",
name: Optional[str] = None,
attribution: str = "",
opacity: Optional[float] = 1.0,
) -> None:
"""Adds Planet global mosaic by month to the map. To get a Planet API key, see https://developers.planet.com/quickstart/apis/
Args:
year (int, optional): The year of Planet global mosaic, must be >=2016. Defaults to 2016.
month (int, optional): The month of Planet global mosaic, must be 1-12. Defaults to 1.
api_key (str, optional): The Planet API key. Defaults to None.
token_name (str, optional): The environment variable name of the API key. Defaults to "PLANET_API_KEY".
name (str, optional): Name of the layer. Defaults to 'TileLayer'.
attribution (str): The attribution to use. Defaults to "".
opacity (float, optional): The opacity of the layer. Defaults to 1.
"""
if name is None:
name = str(year) + "-" + str(month).zfill(2)
tile_url = planet_by_month(year, month, api_key, token_name)
self.add_tile_layer(
tile_url, name=name, attribution=attribution, opacity=opacity
)
def add_planet_by_quarter(
self,
year: Optional[int] = 2016,
quarter: Optional[int] = 1,
api_key: Optional[str] = None,
token_name: Optional[str] = "PLANET_API_KEY",
name: Optional[str] = None,
attribution: str = "",
opacity: Optional[float] = 1.0,
) -> None:
"""Adds Planet global mosaic by month to the map. To get a Planet API key, see https://developers.planet.com/quickstart/apis/
Args:
year (int, optional): The year of Planet global mosaic, must be >=2016. Defaults to 2016.
quarter (int, optional): The quarter of Planet global mosaic, must be 1-4. Defaults to 1.
api_key (str, optional): The Planet API key. Defaults to None.
token_name (str, optional): The environment variable name of the API key. Defaults to "PLANET_API_KEY".
name (str, optional): Name of the layer. Defaults to 'TileLayer'.
attribution (str): The attribution to use. Defaults to "".
opacity (float, optional): The opacity of the layer. Defaults to 1.
"""
if name is None:
name = str(year) + "-" + "q" + str(quarter)
tile_url = planet_by_quarter(year, quarter, api_key, token_name)
self.add_tile_layer(
tile_url, name=name, attribution=attribution, opacity=opacity
)
def save(
self,
file: str,
format: Optional[str] = None,
width: Optional[int] = None,
height: Optional[int] = None,
scale: Optional[int] = None,
**kwargs,
) -> None:
"""Convert a map to a static image and write it to a file or writeable object
Args:
file (str): A string representing a local file path or a writeable object (e.g. a pathlib.Path object or an open file descriptor)
format (str, optional): The desired image format. One of png, jpg, jpeg, webp, svg, pdf, eps. Defaults to None.
width (int, optional): The width of the exported image in layout pixels. If the `scale` property is 1.0, this will also be the width of the exported image in physical pixels.. Defaults to None.
height (int, optional): The height of the exported image in layout pixels. If the `scale` property is 1.0, this will also be the height of the exported image in physical pixels.. Defaults to None.
scale (int, optional): The scale factor to use when exporting the figure. A scale factor larger than 1.0 will increase the image resolution with respect to the figure's layout pixel dimensions. Whereas as scale factor of less than 1.0 will decrease the image resolution.. Defaults to None.
"""
self.write_image(
file, format=format, width=width, height=height, scale=scale, **kwargs
)
def add_choropleth_map(
self,
data: str,
name: Optional[str] = None,
z: Optional[str] = None,
colorscale: Optional[str] = "Viridis",
**kwargs,
) -> None:
"""Adds a choropleth map to the map.
Args:
data (str): File path to vector data, e.g., https://raw.githubusercontent.com/opengeos/leafmap/master/examples/data/countries.geojson
name (str, optional): Name of the layer. Defaults to None.
z (str, optional): Z value of the data. Defaults to None.
colorscale (str, optional): Color scale of the data. Defaults to "Viridis".
"""
check_package("geopandas")
import json
import geopandas as gpd
gdf = gpd.read_file(data).to_crs(epsg=4326)
geojson = json.loads(gdf.to_json())
self.add_choroplethmapbox(
geojson=geojson,
locations=gdf.index,
z=gdf[z],
name=name,
colorscale=colorscale,
**kwargs,
)
def add_scatter_plot_demo(self, **kwargs) -> None:
"""Adds a scatter plot to the map."""
lons = np.random.random(1000) * 360.0
lats = np.random.random(1000) * 180.0 - 90.0
z = np.random.random(1000) * 50.0
self.add_scattermapbox(
lon=lons, lat=lats, marker={"color": z}, name="Random points", **kwargs
)
def add_heatmap(
self,
data: Union[str, DataFrame],
latitude: Optional[str] = "latitude",
longitude: Optional[str] = "longitude",
z: Optional[str] = "value",
radius: Optional[int] = 10,
colorscale: Optional[str] = None,
name: Optional[str] = "Heat map",
**kwargs,
) -> None:
"""Adds a heat map to the map. Reference: https://plotly.com/python/mapbox-density-heatmaps
Args:
data (str | pd.DataFrame): File path or HTTP URL to the input file or a . For example, https://raw.githubusercontent.com/plotly/datasets/master/earthquakes-23k.csv
latitude (str, optional): The column name of latitude. Defaults to "latitude".
longitude (str, optional): The column name of longitude. Defaults to "longitude".
z (str, optional): The column name of z values. Defaults to "value".
radius (int, optional): Radius of each “point” of the heatmap. Defaults to 25.
colorscale (str, optional): Color scale of the data, e.g., Viridis. See https://plotly.com/python/builtin-colorscales. Defaults to None.
name (str, optional): Layer name to use. Defaults to "Heat map".
"""
if isinstance(data, str):
df = pd.read_csv(data)
elif isinstance(data, pd.DataFrame):
df = data
else:
raise ValueError("data must be a DataFrame or a file path.")
heatmap = go.Densitymapbox(
lat=df[latitude],
lon=df[longitude],
z=df[z],
radius=radius,
colorscale=colorscale,
name=name,
**kwargs,
)
self.add_trace(heatmap)
def add_heatmap_demo(self, **kwargs) -> None:
"""Adds a heatmap to the map."""
quakes = pd.read_csv(
"https://raw.githubusercontent.com/plotly/datasets/master/earthquakes-23k.csv"
)
heatmap = go.Densitymapbox(
lat=quakes.Latitude,
lon=quakes.Longitude,
z=quakes.Magnitude,
radius=10,
name="Earthquake",
**kwargs,
)
self.add_basemap("Esri.WorldTopoMap")
self.add_trace(heatmap)
def add_gdf_demo(
self,
gdf,
label_col: Optional[str],
color_col: Optional[str],
color_continuous_scale: Optional[str] = "Viridis",
**kwargs,
) -> None:
check_package("geopandas", "https://geopandas.org")
import geopandas as gpd
geojson_url = str(gdf)
if isinstance(gdf, str):
gdf = gpd.read_file(gdf).to_crs(epsg=4326)
fig = go.Choroplethmapbox(
geojson=geojson_url,
featureidkey="properties.{}".format(label_col),
locations=gdf[label_col],
z=gdf[color_col],
autocolorscale=False,
colorscale=color_continuous_scale,
marker_line_color="peachpuff",
colorbar=dict(
title={"text": "Legend"},
thickness=15,
len=0.35,
bgcolor="rgba(255,255,255,0.6)",
xanchor="left",
x=0.02,
yanchor="bottom",
y=0.05,
),
)
self.add_trace(fig)
def add_gdf(
self,
gdf,
label_col: Optional[str] = None,
color_col: Optional[str] = None,
labels=None,
opacity: Optional[float] = 1.0,
zoom: Optional[int] = None,
color_continuous_scale: Optional[str] = "Viridis",
**kwargs,
) -> None:
"""Adds a GeoDataFrame to the map.
Args:
gdf (GeoDataFrame): A GeoDataFrame.
label_col (str, optional): The column name of locations. Defaults to None.
color_col (str, optional): The column name of color. Defaults to None.
"""
check_package("geopandas", "https://geopandas.org")
import geopandas as gpd
if isinstance(gdf, str):
gdf = gpd.read_file(gdf)
if not isinstance(gdf, gpd.GeoDataFrame):
raise ValueError("gdf must be a GeoDataFrame.")
gdf = gdf.to_crs(epsg=4326)
# geom_type = gdf_geom_type(gdf)
center_lon, center_lat = gdf_centroid(gdf)
if isinstance(label_col, str):
gdf = gdf.set_index(label_col)
if label_col == color_col:
gdf[label_col] = gdf.index
label_col = gdf.index
elif label_col is None:
label_col = gdf.index
if isinstance(color_col, str):
if color_col not in gdf.columns:
raise ValueError(
f"color must be a column name in the GeoDataFrame. Can be one of {','.join(gdf.columns)} "
)
fig = px.choropleth_mapbox(
gdf,
geojson=gdf.geometry,
locations=label_col,
color=color_col,
color_continuous_scale=color_continuous_scale,
opacity=opacity,
labels=labels,
# mapbox_style="carto-positron",
**kwargs,
)
self.add_traces(fig.data)
self.set_center(center_lat, center_lon, zoom)
def add_geojson_layer(
self,
geojson_in: Union[str, Dict],
name: str,
color: Optional[str] = "blue",
opacity: Optional[float] = 1,
) -> None:
"""Prepare proper and give style for different type of Geometry
Args:
in_geojson (str | dict): The file path or http URL to the input GeoJSON or a dictionary containing the geojson.
name (str): Name for the Layer
color (str, optional): Plain name for color (e.g: blue) or color code (e.g: #FF0000)
opacity(float, optional): opacity of the layer in Map
"""
import json
import requests
if isinstance(geojson_in, dict):
data = geojson_in
elif geojson_in.startswith("http"):
data = requests.get(geojson_in).json()
elif geojson_in.lower().endswith((".json", ".geojson")):
with open(geojson_in) as fp:
data = json.load(fp)
else:
data = geojson_in
""" Only Checking Geometry of first feature( todo : handle multiple type of Geometry in same geojson ) """
first_feature = data["features"][0]
geometry_type = first_feature["geometry"]["type"]
if geometry_type.lower() in ["polygon", "multipolygon"]:
type = "fill"
elif geometry_type.lower() in ["linstring", "multilinestring"]:
type = "line"
elif geometry_type.lower() in ["point", "multipoint"]:
type = "circle"
else:
type = "fill"
self.add_geojson(data, name, type, color, opacity)
def add_geojson(
self,
data: Dict,
name: str,
type: Optional[str],
color: Optional[str],
opacity: Optional[float],
) -> None:
"""Add layers to the Map
Args:
data (dict): Geojson in Dict form
name (str): Name for the Layer
color (str, optional): Plain name for color (e.g: blue) or color code (e.g: #FF0000)
opacity(float, optional): opacity of the layer in Map
"""
new_layer = {
"source": data,
"name": name,
"type": type,
"opacity": opacity,
"color": color,
}
if type == "circle":
new_layer["circle"] = {"radius": 5}
existing_layers = list(self.layout.mapbox.layers)
existing_layers.append(new_layer)
self.update_layout(mapbox={"layers": tuple(existing_layers)})
__init__(self, center=(20, 0), zoom=1, basemap='open-street-map', height=600, **kwargs)
special
¶
Initializes a map. More info at https://plotly.com/python/mapbox-layers/
Parameters:
Name | Type | Description | Default |
---|---|---|---|
center |
tuple |
Center of the map. Defaults to (20, 0). |
(20, 0) |
zoom |
float |
Zoom level of the map. Defaults to 1. |
1 |
basemap |
str |
Can be one of string from "open-street-map", "carto-positron", "carto-darkmatter", "stamen-terrain", "stamen-toner" or "stamen-watercolor" . Defaults to 'open-street-map'. |
'open-street-map' |
height |
int |
Height of the map. Defaults to 600. |
600 |
Source code in leafmap/plotlymap.py
def __init__(
self,
center: Optional[Tuple[float, float]] = (20, 0),
zoom: Optional[float] = 1,
basemap: Optional[str] = "open-street-map",
height: Optional[int] = 600,
**kwargs,
) -> None:
"""Initializes a map. More info at https://plotly.com/python/mapbox-layers/
Args:
center (tuple, optional): Center of the map. Defaults to (20, 0).
zoom (float, optional): Zoom level of the map. Defaults to 1.
basemap (str, optional): Can be one of string from "open-street-map", "carto-positron", "carto-darkmatter", "stamen-terrain", "stamen-toner" or "stamen-watercolor" . Defaults to 'open-street-map'.
height (int, optional): Height of the map. Defaults to 600.
"""
super().__init__(**kwargs)
self.add_scattermapbox()
self.update_layout(
{
"mapbox": {
"style": basemap,
"center": {"lat": center[0], "lon": center[1]},
"zoom": zoom,
},
"margin": {"r": 0, "t": 0, "l": 0, "b": 0},
"height": height,
}
)
add_basemap(self, basemap='ROADMAP')
¶
Adds a basemap to the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
basemap |
str |
Can be one of string from basemaps. Defaults to 'ROADMAP'. |
'ROADMAP' |
Source code in leafmap/plotlymap.py
def add_basemap(self, basemap: Optional[str] = "ROADMAP") -> None:
"""Adds a basemap to the map.
Args:
basemap (str, optional): Can be one of string from basemaps. Defaults to 'ROADMAP'.
"""
if basemap not in basemaps:
raise ValueError(
f"Basemap {basemap} not found. Choose from {','.join(basemaps.keys())}"
)
if basemap in self.get_tile_layers():
self.remove_basemap(basemap)
layers = list(self.layout.mapbox.layers) + [basemaps[basemap]]
self.update_layout(mapbox_layers=layers)
add_choropleth_map(self, data, name=None, z=None, colorscale='Viridis', **kwargs)
¶
Adds a choropleth map to the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
data |
str |
File path to vector data, e.g., https://raw.githubusercontent.com/opengeos/leafmap/master/examples/data/countries.geojson |
required |
name |
str |
Name of the layer. Defaults to None. |
None |
z |
str |
Z value of the data. Defaults to None. |
None |
colorscale |
str |
Color scale of the data. Defaults to "Viridis". |
'Viridis' |
Source code in leafmap/plotlymap.py
def add_choropleth_map(
self,
data: str,
name: Optional[str] = None,
z: Optional[str] = None,
colorscale: Optional[str] = "Viridis",
**kwargs,
) -> None:
"""Adds a choropleth map to the map.
Args:
data (str): File path to vector data, e.g., https://raw.githubusercontent.com/opengeos/leafmap/master/examples/data/countries.geojson
name (str, optional): Name of the layer. Defaults to None.
z (str, optional): Z value of the data. Defaults to None.
colorscale (str, optional): Color scale of the data. Defaults to "Viridis".
"""
check_package("geopandas")
import json
import geopandas as gpd
gdf = gpd.read_file(data).to_crs(epsg=4326)
geojson = json.loads(gdf.to_json())
self.add_choroplethmapbox(
geojson=geojson,
locations=gdf.index,
z=gdf[z],
name=name,
colorscale=colorscale,
**kwargs,
)
add_cog_layer(self, url, name='Untitled', attribution='', opacity=1.0, bands=None, titiler_endpoint=None, **kwargs)
¶
Adds a COG TileLayer to the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
url |
str |
The URL of the COG tile layer, e.g., 'https://github.com/opengeos/data/releases/download/raster/Libya-2023-07-01.tif' |
required |
name |
str |
The layer name to use for the layer. Defaults to 'Untitled'. |
'Untitled' |
attribution |
str |
The attribution to use. Defaults to ''. |
'' |
opacity |
float |
The opacity of the layer. Defaults to 1. |
1.0 |
bands |
list |
The bands to use. Defaults to None. |
None |
titiler_endpoint |
str |
Titiler endpoint. Defaults to "https://titiler.xyz". |
None |
**kwargs |
Arbitrary keyword arguments, including bidx, expression, nodata, unscale, resampling, rescale,
color_formula, colormap, colormap_name, return_mask. See https://developmentseed.org/titiler/endpoints/cog/
and https://cogeotiff.github.io/rio-tiler/colormap/. To select a certain bands, use bidx=[1, 2, 3].
apply a rescaling to multiple bands, use something like |
{} |
Source code in leafmap/plotlymap.py
def add_cog_layer(
self,
url: str,
name: Optional[str] = "Untitled",
attribution: Optional[str] = "",
opacity: Optional[float] = 1.0,
bands: Optional[List] = None,
titiler_endpoint: Optional[str] = None,
**kwargs,
) -> None:
"""Adds a COG TileLayer to the map.
Args:
url (str): The URL of the COG tile layer, e.g., 'https://github.com/opengeos/data/releases/download/raster/Libya-2023-07-01.tif'
name (str, optional): The layer name to use for the layer. Defaults to 'Untitled'.
attribution (str, optional): The attribution to use. Defaults to ''.
opacity (float, optional): The opacity of the layer. Defaults to 1.
bands (list, optional): The bands to use. Defaults to None.
titiler_endpoint (str, optional): Titiler endpoint. Defaults to "https://titiler.xyz".
**kwargs: Arbitrary keyword arguments, including bidx, expression, nodata, unscale, resampling, rescale,
color_formula, colormap, colormap_name, return_mask. See https://developmentseed.org/titiler/endpoints/cog/
and https://cogeotiff.github.io/rio-tiler/colormap/. To select a certain bands, use bidx=[1, 2, 3].
apply a rescaling to multiple bands, use something like `rescale=["164,223","130,211","99,212"]`.
"""
tile_url = cog_tile(url, bands, titiler_endpoint, **kwargs)
center = cog_center(url, titiler_endpoint) # (lon, lat)
self.add_tile_layer(tile_url, name, attribution, opacity)
self.set_center(lon=center[0], lat=center[1], zoom=10)
add_controls(self, controls)
¶
Adds controls to the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
controls |
list |
List of controls to add, e.g., ['drawline', 'drawopenpath', 'drawclosedpath', 'drawcircle', 'drawrect', 'eraseshape'] See https://bit.ly/33Tmqxr |
required |
Source code in leafmap/plotlymap.py
def add_controls(self, controls: List) -> None:
"""Adds controls to the map.
Args:
controls (list): List of controls to add, e.g., ['drawline', 'drawopenpath', 'drawclosedpath', 'drawcircle', 'drawrect', 'eraseshape'] See https://bit.ly/33Tmqxr
"""
if isinstance(controls, str):
controls = [controls]
elif not isinstance(controls, list):
raise ValueError(
"Controls must be a string or a list of strings. See https://bit.ly/33Tmqxr"
)
self.update_layout(modebar_add=controls)
add_gdf(self, gdf, label_col=None, color_col=None, labels=None, opacity=1.0, zoom=None, color_continuous_scale='Viridis', **kwargs)
¶
Adds a GeoDataFrame to the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
gdf |
GeoDataFrame |
A GeoDataFrame. |
required |
label_col |
str |
The column name of locations. Defaults to None. |
None |
color_col |
str |
The column name of color. Defaults to None. |
None |
Source code in leafmap/plotlymap.py
def add_gdf(
self,
gdf,
label_col: Optional[str] = None,
color_col: Optional[str] = None,
labels=None,
opacity: Optional[float] = 1.0,
zoom: Optional[int] = None,
color_continuous_scale: Optional[str] = "Viridis",
**kwargs,
) -> None:
"""Adds a GeoDataFrame to the map.
Args:
gdf (GeoDataFrame): A GeoDataFrame.
label_col (str, optional): The column name of locations. Defaults to None.
color_col (str, optional): The column name of color. Defaults to None.
"""
check_package("geopandas", "https://geopandas.org")
import geopandas as gpd
if isinstance(gdf, str):
gdf = gpd.read_file(gdf)
if not isinstance(gdf, gpd.GeoDataFrame):
raise ValueError("gdf must be a GeoDataFrame.")
gdf = gdf.to_crs(epsg=4326)
# geom_type = gdf_geom_type(gdf)
center_lon, center_lat = gdf_centroid(gdf)
if isinstance(label_col, str):
gdf = gdf.set_index(label_col)
if label_col == color_col:
gdf[label_col] = gdf.index
label_col = gdf.index
elif label_col is None:
label_col = gdf.index
if isinstance(color_col, str):
if color_col not in gdf.columns:
raise ValueError(
f"color must be a column name in the GeoDataFrame. Can be one of {','.join(gdf.columns)} "
)
fig = px.choropleth_mapbox(
gdf,
geojson=gdf.geometry,
locations=label_col,
color=color_col,
color_continuous_scale=color_continuous_scale,
opacity=opacity,
labels=labels,
# mapbox_style="carto-positron",
**kwargs,
)
self.add_traces(fig.data)
self.set_center(center_lat, center_lon, zoom)
add_geojson(self, data, name, type, color, opacity)
¶
Add layers to the Map
Parameters:
Name | Type | Description | Default |
---|---|---|---|
data |
dict |
Geojson in Dict form |
required |
name |
str |
Name for the Layer |
required |
color |
str |
Plain name for color (e.g: blue) or color code (e.g: #FF0000) |
required |
opacity(float, |
optional |
opacity of the layer in Map |
required |
Source code in leafmap/plotlymap.py
def add_geojson(
self,
data: Dict,
name: str,
type: Optional[str],
color: Optional[str],
opacity: Optional[float],
) -> None:
"""Add layers to the Map
Args:
data (dict): Geojson in Dict form
name (str): Name for the Layer
color (str, optional): Plain name for color (e.g: blue) or color code (e.g: #FF0000)
opacity(float, optional): opacity of the layer in Map
"""
new_layer = {
"source": data,
"name": name,
"type": type,
"opacity": opacity,
"color": color,
}
if type == "circle":
new_layer["circle"] = {"radius": 5}
existing_layers = list(self.layout.mapbox.layers)
existing_layers.append(new_layer)
self.update_layout(mapbox={"layers": tuple(existing_layers)})
add_geojson_layer(self, geojson_in, name, color='blue', opacity=1)
¶
Prepare proper and give style for different type of Geometry
Parameters:
Name | Type | Description | Default |
---|---|---|---|
in_geojson |
str | dict |
The file path or http URL to the input GeoJSON or a dictionary containing the geojson. |
required |
name |
str |
Name for the Layer |
required |
color |
str |
Plain name for color (e.g: blue) or color code (e.g: #FF0000) |
'blue' |
opacity(float, |
optional |
opacity of the layer in Map |
required |
Source code in leafmap/plotlymap.py
def add_geojson_layer(
self,
geojson_in: Union[str, Dict],
name: str,
color: Optional[str] = "blue",
opacity: Optional[float] = 1,
) -> None:
"""Prepare proper and give style for different type of Geometry
Args:
in_geojson (str | dict): The file path or http URL to the input GeoJSON or a dictionary containing the geojson.
name (str): Name for the Layer
color (str, optional): Plain name for color (e.g: blue) or color code (e.g: #FF0000)
opacity(float, optional): opacity of the layer in Map
"""
import json
import requests
if isinstance(geojson_in, dict):
data = geojson_in
elif geojson_in.startswith("http"):
data = requests.get(geojson_in).json()
elif geojson_in.lower().endswith((".json", ".geojson")):
with open(geojson_in) as fp:
data = json.load(fp)
else:
data = geojson_in
""" Only Checking Geometry of first feature( todo : handle multiple type of Geometry in same geojson ) """
first_feature = data["features"][0]
geometry_type = first_feature["geometry"]["type"]
if geometry_type.lower() in ["polygon", "multipolygon"]:
type = "fill"
elif geometry_type.lower() in ["linstring", "multilinestring"]:
type = "line"
elif geometry_type.lower() in ["point", "multipoint"]:
type = "circle"
else:
type = "fill"
self.add_geojson(data, name, type, color, opacity)
add_heatmap(self, data, latitude='latitude', longitude='longitude', z='value', radius=10, colorscale=None, name='Heat map', **kwargs)
¶
Adds a heat map to the map. Reference: https://plotly.com/python/mapbox-density-heatmaps
Parameters:
Name | Type | Description | Default |
---|---|---|---|
data |
str | pd.DataFrame |
File path or HTTP URL to the input file or a . For example, https://raw.githubusercontent.com/plotly/datasets/master/earthquakes-23k.csv |
required |
latitude |
str |
The column name of latitude. Defaults to "latitude". |
'latitude' |
longitude |
str |
The column name of longitude. Defaults to "longitude". |
'longitude' |
z |
str |
The column name of z values. Defaults to "value". |
'value' |
radius |
int |
Radius of each “point” of the heatmap. Defaults to 25. |
10 |
colorscale |
str |
Color scale of the data, e.g., Viridis. See https://plotly.com/python/builtin-colorscales. Defaults to None. |
None |
name |
str |
Layer name to use. Defaults to "Heat map". |
'Heat map' |
Source code in leafmap/plotlymap.py
def add_heatmap(
self,
data: Union[str, DataFrame],
latitude: Optional[str] = "latitude",
longitude: Optional[str] = "longitude",
z: Optional[str] = "value",
radius: Optional[int] = 10,
colorscale: Optional[str] = None,
name: Optional[str] = "Heat map",
**kwargs,
) -> None:
"""Adds a heat map to the map. Reference: https://plotly.com/python/mapbox-density-heatmaps
Args:
data (str | pd.DataFrame): File path or HTTP URL to the input file or a . For example, https://raw.githubusercontent.com/plotly/datasets/master/earthquakes-23k.csv
latitude (str, optional): The column name of latitude. Defaults to "latitude".
longitude (str, optional): The column name of longitude. Defaults to "longitude".
z (str, optional): The column name of z values. Defaults to "value".
radius (int, optional): Radius of each “point” of the heatmap. Defaults to 25.
colorscale (str, optional): Color scale of the data, e.g., Viridis. See https://plotly.com/python/builtin-colorscales. Defaults to None.
name (str, optional): Layer name to use. Defaults to "Heat map".
"""
if isinstance(data, str):
df = pd.read_csv(data)
elif isinstance(data, pd.DataFrame):
df = data
else:
raise ValueError("data must be a DataFrame or a file path.")
heatmap = go.Densitymapbox(
lat=df[latitude],
lon=df[longitude],
z=df[z],
radius=radius,
colorscale=colorscale,
name=name,
**kwargs,
)
self.add_trace(heatmap)
add_heatmap_demo(self, **kwargs)
¶
Adds a heatmap to the map.
Source code in leafmap/plotlymap.py
def add_heatmap_demo(self, **kwargs) -> None:
"""Adds a heatmap to the map."""
quakes = pd.read_csv(
"https://raw.githubusercontent.com/plotly/datasets/master/earthquakes-23k.csv"
)
heatmap = go.Densitymapbox(
lat=quakes.Latitude,
lon=quakes.Longitude,
z=quakes.Magnitude,
radius=10,
name="Earthquake",
**kwargs,
)
self.add_basemap("Esri.WorldTopoMap")
self.add_trace(heatmap)
add_layer(self, layer, name=None, **kwargs)
¶
Adds a layer to the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
layer |
plotly.graph_objects |
Layer to add. |
required |
name |
str |
Name of the layer. Defaults to None. |
None |
Source code in leafmap/plotlymap.py
def add_layer(self, layer, name: Optional[str] = None, **kwargs):
"""Adds a layer to the map.
Args:
layer (plotly.graph_objects): Layer to add.
name (str, optional): Name of the layer. Defaults to None.
"""
if isinstance(name, str):
layer.name = name
self.add_trace(layer, **kwargs)
add_mapbox_layer(self, style, access_token=None)
¶
Adds a mapbox layer to the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
layer |
str | dict |
Layer to add. Can be "basic", "streets", "outdoors", "light", "dark", "satellite", or "satellite-streets". See https://plotly.com/python/mapbox-layers/ and https://docs.mapbox.com/mapbox-gl-js/style-spec/ |
required |
access_token |
str |
The Mapbox Access token. It can be set as an environment variable "MAPBOX_TOKEN". Defaults to None. |
None |
Source code in leafmap/plotlymap.py
def add_mapbox_layer(self, style: Union[str, Dict], access_token=None) -> None:
"""Adds a mapbox layer to the map.
Args:
layer (str | dict): Layer to add. Can be "basic", "streets", "outdoors", "light", "dark", "satellite", or "satellite-streets". See https://plotly.com/python/mapbox-layers/ and https://docs.mapbox.com/mapbox-gl-js/style-spec/
access_token (str, optional): The Mapbox Access token. It can be set as an environment variable "MAPBOX_TOKEN". Defaults to None.
"""
if access_token is None:
access_token = os.environ.get("MAPBOX_TOKEN")
self.update_layout(
mapbox_style=style, mapbox_layers=[], mapbox_accesstoken=access_token
)
add_mosaic_layer(self, url, titiler_endpoint=None, name='Mosaic Layer', attribution='', opacity=1.0, **kwargs)
¶
Adds a STAC TileLayer to the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
url |
str |
HTTP URL to a MosaicJSON. |
required |
titiler_endpoint |
str |
Titiler endpoint, e.g., "https://titiler.xyz". Defaults to None. |
None |
name |
str |
The layer name to use for the layer. Defaults to 'Mosaic Layer'. |
'Mosaic Layer' |
attribution |
str |
The attribution to use. Defaults to ''. |
'' |
opacity |
float |
The opacity of the layer. Defaults to 1. |
1.0 |
Source code in leafmap/plotlymap.py
def add_mosaic_layer(
self,
url: str,
titiler_endpoint: Optional[str] = None,
name: Optional[str] = "Mosaic Layer",
attribution: Optional[str] = "",
opacity: Optional[float] = 1.0,
**kwargs,
) -> None:
"""Adds a STAC TileLayer to the map.
Args:
url (str): HTTP URL to a MosaicJSON.
titiler_endpoint (str, optional): Titiler endpoint, e.g., "https://titiler.xyz". Defaults to None.
name (str, optional): The layer name to use for the layer. Defaults to 'Mosaic Layer'.
attribution (str, optional): The attribution to use. Defaults to ''.
opacity (float, optional): The opacity of the layer. Defaults to 1.
"""
tile_url = mosaic_tile(url, titiler_endpoint, **kwargs)
center = mosaic_info(url, titiler_endpoint)["center"]
self.add_tile_layer(tile_url, name, attribution, opacity)
self.set_center(lon=center[0], lat=center[1], zoom=10)
add_planet_by_month(self, year=2016, month=1, api_key=None, token_name='PLANET_API_KEY', name=None, attribution='', opacity=1.0)
¶
Adds Planet global mosaic by month to the map. To get a Planet API key, see https://developers.planet.com/quickstart/apis/
Parameters:
Name | Type | Description | Default |
---|---|---|---|
year |
int |
The year of Planet global mosaic, must be >=2016. Defaults to 2016. |
2016 |
month |
int |
The month of Planet global mosaic, must be 1-12. Defaults to 1. |
1 |
api_key |
str |
The Planet API key. Defaults to None. |
None |
token_name |
str |
The environment variable name of the API key. Defaults to "PLANET_API_KEY". |
'PLANET_API_KEY' |
name |
str |
Name of the layer. Defaults to 'TileLayer'. |
None |
attribution |
str |
The attribution to use. Defaults to "". |
'' |
opacity |
float |
The opacity of the layer. Defaults to 1. |
1.0 |
Source code in leafmap/plotlymap.py
def add_planet_by_month(
self,
year: Optional[int] = 2016,
month: Optional[int] = 1,
api_key: Optional[str] = None,
token_name: Optional[str] = "PLANET_API_KEY",
name: Optional[str] = None,
attribution: str = "",
opacity: Optional[float] = 1.0,
) -> None:
"""Adds Planet global mosaic by month to the map. To get a Planet API key, see https://developers.planet.com/quickstart/apis/
Args:
year (int, optional): The year of Planet global mosaic, must be >=2016. Defaults to 2016.
month (int, optional): The month of Planet global mosaic, must be 1-12. Defaults to 1.
api_key (str, optional): The Planet API key. Defaults to None.
token_name (str, optional): The environment variable name of the API key. Defaults to "PLANET_API_KEY".
name (str, optional): Name of the layer. Defaults to 'TileLayer'.
attribution (str): The attribution to use. Defaults to "".
opacity (float, optional): The opacity of the layer. Defaults to 1.
"""
if name is None:
name = str(year) + "-" + str(month).zfill(2)
tile_url = planet_by_month(year, month, api_key, token_name)
self.add_tile_layer(
tile_url, name=name, attribution=attribution, opacity=opacity
)
add_planet_by_quarter(self, year=2016, quarter=1, api_key=None, token_name='PLANET_API_KEY', name=None, attribution='', opacity=1.0)
¶
Adds Planet global mosaic by month to the map. To get a Planet API key, see https://developers.planet.com/quickstart/apis/
Parameters:
Name | Type | Description | Default |
---|---|---|---|
year |
int |
The year of Planet global mosaic, must be >=2016. Defaults to 2016. |
2016 |
quarter |
int |
The quarter of Planet global mosaic, must be 1-4. Defaults to 1. |
1 |
api_key |
str |
The Planet API key. Defaults to None. |
None |
token_name |
str |
The environment variable name of the API key. Defaults to "PLANET_API_KEY". |
'PLANET_API_KEY' |
name |
str |
Name of the layer. Defaults to 'TileLayer'. |
None |
attribution |
str |
The attribution to use. Defaults to "". |
'' |
opacity |
float |
The opacity of the layer. Defaults to 1. |
1.0 |
Source code in leafmap/plotlymap.py
def add_planet_by_quarter(
self,
year: Optional[int] = 2016,
quarter: Optional[int] = 1,
api_key: Optional[str] = None,
token_name: Optional[str] = "PLANET_API_KEY",
name: Optional[str] = None,
attribution: str = "",
opacity: Optional[float] = 1.0,
) -> None:
"""Adds Planet global mosaic by month to the map. To get a Planet API key, see https://developers.planet.com/quickstart/apis/
Args:
year (int, optional): The year of Planet global mosaic, must be >=2016. Defaults to 2016.
quarter (int, optional): The quarter of Planet global mosaic, must be 1-4. Defaults to 1.
api_key (str, optional): The Planet API key. Defaults to None.
token_name (str, optional): The environment variable name of the API key. Defaults to "PLANET_API_KEY".
name (str, optional): Name of the layer. Defaults to 'TileLayer'.
attribution (str): The attribution to use. Defaults to "".
opacity (float, optional): The opacity of the layer. Defaults to 1.
"""
if name is None:
name = str(year) + "-" + "q" + str(quarter)
tile_url = planet_by_quarter(year, quarter, api_key, token_name)
self.add_tile_layer(
tile_url, name=name, attribution=attribution, opacity=opacity
)
add_scatter_plot_demo(self, **kwargs)
¶
Adds a scatter plot to the map.
Source code in leafmap/plotlymap.py
def add_scatter_plot_demo(self, **kwargs) -> None:
"""Adds a scatter plot to the map."""
lons = np.random.random(1000) * 360.0
lats = np.random.random(1000) * 180.0 - 90.0
z = np.random.random(1000) * 50.0
self.add_scattermapbox(
lon=lons, lat=lats, marker={"color": z}, name="Random points", **kwargs
)
add_stac_layer(self, url=None, collection=None, items=None, assets=None, bands=None, titiler_endpoint=None, name='STAC Layer', attribution='', opacity=1.0, **kwargs)
¶
Adds a STAC TileLayer to the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
url |
str |
HTTP URL to a STAC item, e.g., https://canada-spot-ortho.s3.amazonaws.com/canada_spot_orthoimages/canada_spot5_orthoimages/S5_2007/S5_11055_6057_20070622/S5_11055_6057_20070622.json |
None |
collection |
str |
The Microsoft Planetary Computer STAC collection ID, e.g., landsat-8-c2-l2. |
None |
items |
str |
The Microsoft Planetary Computer STAC item ID, e.g., LC08_L2SP_047027_20201204_02_T1. |
None |
assets |
str | list |
The Microsoft Planetary Computer STAC asset ID, e.g., ["SR_B7", "SR_B5", "SR_B4"]. |
None |
bands |
list |
A list of band names, e.g., ["SR_B7", "SR_B5", "SR_B4"] |
None |
titiler_endpoint |
str |
Titiler endpoint, e.g., "https://titiler.xyz", "planetary-computer", "pc". Defaults to None. |
None |
name |
str |
The layer name to use for the layer. Defaults to 'STAC Layer'. |
'STAC Layer' |
attribution |
str |
The attribution to use. Defaults to ''. |
'' |
opacity |
float |
The opacity of the layer. Defaults to 1. |
1.0 |
Source code in leafmap/plotlymap.py
def add_stac_layer(
self,
url: str = None,
collection: str = None,
items: str = None,
assets: str = None,
bands: List = None,
titiler_endpoint: Optional[str] = None,
name: Optional[str] = "STAC Layer",
attribution: Optional[str] = "",
opacity: Optional[float] = 1.0,
**kwargs,
) -> None:
"""Adds a STAC TileLayer to the map.
Args:
url (str): HTTP URL to a STAC item, e.g., https://canada-spot-ortho.s3.amazonaws.com/canada_spot_orthoimages/canada_spot5_orthoimages/S5_2007/S5_11055_6057_20070622/S5_11055_6057_20070622.json
collection (str): The Microsoft Planetary Computer STAC collection ID, e.g., landsat-8-c2-l2.
items (str): The Microsoft Planetary Computer STAC item ID, e.g., LC08_L2SP_047027_20201204_02_T1.
assets (str | list): The Microsoft Planetary Computer STAC asset ID, e.g., ["SR_B7", "SR_B5", "SR_B4"].
bands (list): A list of band names, e.g., ["SR_B7", "SR_B5", "SR_B4"]
titiler_endpoint (str, optional): Titiler endpoint, e.g., "https://titiler.xyz", "planetary-computer", "pc". Defaults to None.
name (str, optional): The layer name to use for the layer. Defaults to 'STAC Layer'.
attribution (str, optional): The attribution to use. Defaults to ''.
opacity (float, optional): The opacity of the layer. Defaults to 1.
"""
tile_url = stac_tile(
url, collection, item, assets, bands, titiler_endpoint, **kwargs
)
center = stac_center(url, collection, item, titiler_endpoint)
self.add_tile_layer(tile_url, name, attribution, opacity)
self.set_center(lon=center[0], lat=center[1], zoom=10)
add_tile_layer(self, url, name='TileLayer', attribution='', opacity=1.0, **kwargs)
¶
Adds a TileLayer to the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
url |
str |
The URL of the tile layer. |
required |
name |
str |
Name of the layer. Defaults to 'TileLayer'. |
'TileLayer' |
attribution |
str |
The attribution to use. Defaults to "". |
'' |
opacity |
float |
The opacity of the layer. Defaults to 1. |
1.0 |
Source code in leafmap/plotlymap.py
def add_tile_layer(
self,
url: str,
name: Optional[str] = "TileLayer",
attribution: Optional[str] = "",
opacity: Optional[float] = 1.0,
**kwargs,
) -> None:
"""Adds a TileLayer to the map.
Args:
url (str): The URL of the tile layer.
name (str, optional): Name of the layer. Defaults to 'TileLayer'.
attribution (str): The attribution to use. Defaults to "".
opacity (float, optional): The opacity of the layer. Defaults to 1.
"""
layer = {
"below": "traces",
"sourcetype": "raster",
"sourceattribution": attribution,
"source": [url],
"opacity": opacity,
"name": name,
}
layers = list(self.layout.mapbox.layers) + [layer]
self.update_layout(mapbox_layers=layers)
clear_controls(self)
¶
Removes all controls from the map.
Source code in leafmap/plotlymap.py
def clear_controls(self):
"""Removes all controls from the map."""
config = {
"scrollZoom": True,
"displayModeBar": False,
"editable": True,
"showLink": False,
"displaylogo": False,
}
self.show(toolbar=False, config=config)
clear_layers(self, clear_basemap=False)
¶
Clears all layers from the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
clear_basemap |
bool |
If True, clears the basemap. Defaults to False. |
False |
Source code in leafmap/plotlymap.py
def clear_layers(self, clear_basemap: Optional[bool] = False):
"""Clears all layers from the map.
Args:
clear_basemap (bool, optional): If True, clears the basemap. Defaults to False.
"""
if clear_basemap:
self.data = []
else:
if len(self.data) > 1:
self.data = self.data[:1]
find_layer_index(self, name)
¶
Finds the index of a layer.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
name |
str |
Name of the layer to find. |
required |
Returns:
Type | Description |
---|---|
int |
Index of the layer. |
Source code in leafmap/plotlymap.py
def find_layer_index(self, name: str) -> int:
"""Finds the index of a layer.
Args:
name (str): Name of the layer to find.
Returns:
int: Index of the layer.
"""
for i, layer in enumerate(self.data):
if layer.name == name:
return i
for i, layer in enumerate(self.layout.mapbox.layers):
if layer["name"] == name:
return i
return None
get_data_layers(self)
¶
Returns a dictionary of data layers in the map.
Returns:
Type | Description |
---|---|
dict |
A dictionary of data layers in the map. |
Source code in leafmap/plotlymap.py
def get_data_layers(self) -> Dict:
"""Returns a dictionary of data layers in the map.
Returns:
dict: A dictionary of data layers in the map.
"""
layers = {}
for layer in self.data:
if layer.name is not None and layer.name != "trace 0":
layers[layer.name] = layer
return layers
get_layers(self)
¶
Returns a dictionary of all layers in the map.
Returns:
Type | Description |
---|---|
dict |
A dictionary of all layers in the map. |
Source code in leafmap/plotlymap.py
def get_layers(self) -> Dict:
"""Returns a dictionary of all layers in the map.
Returns:
dict: A dictionary of all layers in the map.
"""
layers = {}
for layer in self.layout.mapbox.layers:
if layer["name"] is not None:
layers[layer["name"]] = layer
for layer in self.data:
if layer.name is not None and layer.name != "trace 0":
layers[layer.name] = layer
return layers
get_tile_layers(self)
¶
Returns a dictionary of tile layers in the map.
Returns:
Type | Description |
---|---|
dict |
A dictionary of tile layers in the map. |
Source code in leafmap/plotlymap.py
def get_tile_layers(self) -> Dict:
"""Returns a dictionary of tile layers in the map.
Returns:
dict: A dictionary of tile layers in the map.
"""
layers = {}
for layer in self.layout.mapbox.layers:
if layer["name"] is not None:
layers[layer["name"]] = layer
return layers
remove_basemap(self, name)
¶
Removes a basemap from the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
name |
str |
Name of the basemap to remove. |
required |
Source code in leafmap/plotlymap.py
def remove_basemap(self, name: str) -> None:
"""Removes a basemap from the map.
Args:
name (str): Name of the basemap to remove.
"""
layers = list(self.layout.mapbox.layers)
layers = [layer for layer in layers if layer["name"] != name]
self.layout.mapbox.layers = layers
remove_controls(self, controls)
¶
Removes controls to the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
controls |
list |
List of controls to remove, e.g., ["zoomin", "zoomout", "toimage", "pan", "resetview"]. See https://bit.ly/3Jk7wkb |
required |
Source code in leafmap/plotlymap.py
def remove_controls(self, controls: List):
"""Removes controls to the map.
Args:
controls (list): List of controls to remove, e.g., ["zoomin", "zoomout", "toimage", "pan", "resetview"]. See https://bit.ly/3Jk7wkb
"""
if isinstance(controls, str):
controls = [controls]
elif not isinstance(controls, list):
raise ValueError(
"Controls must be a string or a list of strings. See https://bit.ly/3Jk7wkb"
)
self.update_layout(modebar_remove=controls)
remove_layer(self, name)
¶
Removes a layer from the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
name |
str |
Name of the layer to remove. |
required |
Source code in leafmap/plotlymap.py
def remove_layer(self, name: str) -> None:
"""Removes a layer from the map.
Args:
name (str): Name of the layer to remove.
"""
if name in self.get_data_layers():
self.data = [layer for layer in self.data if layer.name != name]
elif name in self.get_tile_layers():
self.layout.mapbox.layers = [
layer for layer in self.layout.mapbox.layers if layer["name"] != name
]
save(self, file, format=None, width=None, height=None, scale=None, **kwargs)
¶
Convert a map to a static image and write it to a file or writeable object
Parameters:
Name | Type | Description | Default |
---|---|---|---|
file |
str |
A string representing a local file path or a writeable object (e.g. a pathlib.Path object or an open file descriptor) |
required |
format |
str |
The desired image format. One of png, jpg, jpeg, webp, svg, pdf, eps. Defaults to None. |
None |
width |
int |
The width of the exported image in layout pixels. If the |
None |
height |
int |
The height of the exported image in layout pixels. If the |
None |
scale |
int |
The scale factor to use when exporting the figure. A scale factor larger than 1.0 will increase the image resolution with respect to the figure's layout pixel dimensions. Whereas as scale factor of less than 1.0 will decrease the image resolution.. Defaults to None. |
None |
Source code in leafmap/plotlymap.py
def save(
self,
file: str,
format: Optional[str] = None,
width: Optional[int] = None,
height: Optional[int] = None,
scale: Optional[int] = None,
**kwargs,
) -> None:
"""Convert a map to a static image and write it to a file or writeable object
Args:
file (str): A string representing a local file path or a writeable object (e.g. a pathlib.Path object or an open file descriptor)
format (str, optional): The desired image format. One of png, jpg, jpeg, webp, svg, pdf, eps. Defaults to None.
width (int, optional): The width of the exported image in layout pixels. If the `scale` property is 1.0, this will also be the width of the exported image in physical pixels.. Defaults to None.
height (int, optional): The height of the exported image in layout pixels. If the `scale` property is 1.0, this will also be the height of the exported image in physical pixels.. Defaults to None.
scale (int, optional): The scale factor to use when exporting the figure. A scale factor larger than 1.0 will increase the image resolution with respect to the figure's layout pixel dimensions. Whereas as scale factor of less than 1.0 will decrease the image resolution.. Defaults to None.
"""
self.write_image(
file, format=format, width=width, height=height, scale=scale, **kwargs
)
set_center(self, lat, lon, zoom=None)
¶
Sets the center of the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
lat |
float |
Latitude. |
required |
lon |
float |
Longitude. |
required |
zoom |
int |
Zoom level of the map. Defaults to None. |
None |
Source code in leafmap/plotlymap.py
def set_center(self, lat: float, lon: float, zoom: Optional[float] = None) -> None:
"""Sets the center of the map.
Args:
lat (float): Latitude.
lon (float): Longitude.
zoom (int, optional): Zoom level of the map. Defaults to None.
"""
self.update_layout(
mapbox=dict(
center=dict(lat=lat, lon=lon),
zoom=zoom if zoom is not None else self.layout.mapbox.zoom,
)
)
set_layer_opacity(self, name, opacity=1)
¶
Sets the visibility of a layer.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
name |
str |
Name of the layer to set. |
required |
opacity |
float |
Opacity of the layer. Defaults to 1. |
1 |
Source code in leafmap/plotlymap.py
def set_layer_opacity(self, name: str, opacity: Optional[float] = 1) -> None:
"""Sets the visibility of a layer.
Args:
name (str): Name of the layer to set.
opacity (float, optional): Opacity of the layer. Defaults to 1.
"""
if name in self.get_tile_layers():
index = self.find_layer_index(name)
self.layout.mapbox.layers[index].opacity = opacity
elif name in self.get_data_layers():
index = self.find_layer_index(name)
layer = self.data[index]
if hasattr(layer, "opacity"):
layer.opacity = opacity
elif hasattr(layer, "marker"):
layer.marker.opacity = opacity
else:
print(f"Layer {name} not found.")
set_layer_visibility(self, name, show=True)
¶
Sets the visibility of a layer.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
name |
str |
Name of the layer to set. |
required |
show |
bool |
If True, shows the layer. Defaults to True. |
True |
Source code in leafmap/plotlymap.py
def set_layer_visibility(self, name: str, show: Optional[bool] = True) -> None:
"""Sets the visibility of a layer.
Args:
name (str): Name of the layer to set.
show (bool, optional): If True, shows the layer. Defaults to True.
"""
if name in self.get_tile_layers():
index = self.find_layer_index(name)
self.layout.mapbox.layers[index].visible = show
elif name in self.get_data_layers():
index = self.find_layer_index(name)
self.data[index].visible = show
else:
print(f"Layer {name} not found.")
show(self, toolbar=True, map_min_width='91%', map_max_width='98%', refresh=False, **kwargs)
¶
Shows the map.
Parameters:
Name | Type | Description | Default |
---|---|---|---|
toolbar |
bool |
Whether to show the toolbar. Defaults to True. |
True |
map_min_width |
str |
The minimum width of the map. Defaults to '91%'. |
'91%' |
map_max_width |
str |
The maximum width of the map. Defaults to '98%'. |
'98%' |
refresh |
bool |
Whether to refresh the map when the map is resized. Defaults to False. |
False |
Returns:
Type | Description |
---|---|
Canvas |
Map canvas. |
Source code in leafmap/plotlymap.py
def show(
self,
toolbar: Optional[bool] = True,
map_min_width: Optional[str] = "91%",
map_max_width: Optional[str] = "98%",
refresh: Optional[bool] = False,
**kwargs,
) -> None:
"""Shows the map.
Args:
toolbar (bool, optional): Whether to show the toolbar. Defaults to True.
map_min_width (str, optional): The minimum width of the map. Defaults to '91%'.
map_max_width (str, optional): The maximum width of the map. Defaults to '98%'.
refresh (bool, optional): Whether to refresh the map when the map is resized. Defaults to False.
Returns:
Canvas: Map canvas.
"""
if not toolbar:
super().show(**kwargs)
else:
canvas = Canvas(
self,
map_min_width=map_min_width,
map_max_width=map_max_width,
map_refresh=refresh,
)
return canvas.canvas
fix_widget_error()
¶
Fix FigureWidget - 'mapbox._derived' Value Error. Adopted from: https://github.com/plotly/plotly.py/issues/2570#issuecomment-738735816
Source code in leafmap/plotlymap.py
def fix_widget_error() -> None:
"""
Fix FigureWidget - 'mapbox._derived' Value Error.
Adopted from: https://github.com/plotly/plotly.py/issues/2570#issuecomment-738735816
"""
import shutil
import pkg_resources
pkg_dir = os.path.dirname(pkg_resources.resource_filename("plotly", "plotly.py"))
basedatatypesPath = os.path.join(pkg_dir, "basedatatypes.py")
backup_file = basedatatypesPath.replace(".py", "_bk.py")
shutil.copyfile(basedatatypesPath, backup_file)
# read basedatatypes.py
with open(basedatatypesPath, "r") as f:
lines = f.read()
find = "if not BaseFigure._is_key_path_compatible(key_path_str, self.layout):"
replace = """if not BaseFigure._is_key_path_compatible(key_path_str, self.layout):
if key_path_str == "mapbox._derived":
return"""
# add new text
lines = lines.replace(find, replace)
# overwrite old 'basedatatypes.py'
with open(basedatatypesPath, "w") as f:
f.write(lines)