A Python package for creating interactive maps with anywidget and JavaScript mapping libraries
- Free software: MIT License
- Documentation: https://anymap.gishub.org
- πΊοΈ Interactive Maps: Create beautiful, interactive maps in Jupyter notebooks
- π Bidirectional Communication: Full Python β JavaScript communication
- π± Multi-cell Support: Render maps in multiple notebook cells without conflicts
- π― MapLibre Integration: Built-in support for MapLibre GL JS
- π οΈ Extensible: Easy to add support for other mapping libraries
- π Familiar API: Similar to ipyleaflet for easy migration
pip install anymap
conda install -c conda-forge anymap
from anymap import MapLibreMap
# Create a basic map
m = MapLibreMap(
center=[37.7749, -122.4194], # San Francisco
zoom=12,
height="600px"
)
m
from anymap import MapLibreMap
# Create a map with custom settings
m = MapLibreMap(
center=[40.7128, -74.0060], # New York City
zoom=13,
height="500px",
bearing=45, # Map rotation
pitch=60 # 3D tilt
)
# Add a marker with popup
m.add_marker(
lat=40.7128,
lng=-74.0060,
popup="<h3>New York City</h3><p>The Big Apple</p>"
)
# Add GeoJSON data
geojson_data = {
"type": "FeatureCollection",
"features": [
{
"type": "Feature",
"geometry": {
"type": "Point",
"coordinates": [-74.0060, 40.7128]
},
"properties": {"name": "NYC"}
}
]
}
m.add_geojson_layer(
layer_id="cities",
geojson_data=geojson_data,
layer_type="circle",
paint={
"circle-radius": 8,
"circle-color": "#ff0000"
}
)
def handle_click(event):
lat, lng = event['lngLat']
print(f"Clicked at: {lat:.4f}, {lng:.4f}")
m.on_map_event('click', handle_click)
# Change map properties
m.set_center(51.5074, -0.1278) # London
m.set_zoom(14)
# Animate to a location
m.fly_to(48.8566, 2.3522, zoom=15) # Paris
AnyMap is designed to work seamlessly across multiple notebook cells:
# Cell 1
m = MapLibreMap(center=[0, 0], zoom=2)
m
# Cell 2 - Same map instance
m.add_marker(0, 0, popup="Origin")
# Cell 3 - Display again
m
# Add and remove layers
m.add_source("my-source", {
"type": "geojson",
"data": geojson_data
})
m.add_layer("my-layer", {
"id": "my-layer",
"type": "circle",
"source": "my-source",
"paint": {"circle-radius": 5}
})
# Remove layers
m.remove_layer("my-layer")
m.remove_source("my-source")
# Call any MapLibre GL JS method
m.call_js_method('easeTo', {
'center': [lng, lat],
'zoom': 14,
'duration': 2000
})
Check out the example notebooks in the examples/
directory:
basic_usage.ipynb
- Basic map creation and interactionadvanced_features.ipynb
- Advanced layer management and stylingmulti_cell_test.ipynb
- Multi-cell rendering tests
To set up for development:
git clone https://github.com/opengeos/anymap.git
cd anymap
pip install -e .
Run tests:
python -m unittest tests.test_anymap -v
- β MapLibre GL JS backend
- β Mapbox GL JS backend
- π² Leaflet backend
- π² OpenLayers backend
- π² DeckGL backend
- π² KeplerGL backend
- π² Cesium backend
- π² Potree backend
Contributions are welcome! Please feel free to submit a Pull Request.
This project is licensed under the MIT License - see the LICENSE file for details.