From ebc34c17cae611df33f7dc3eb4139346b6bf3a93 Mon Sep 17 00:00:00 2001 From: Hans Then Date: Thu, 14 Mar 2024 11:39:55 +0100 Subject: [PATCH] Add EventHandler to layer object In combination with JsCode this makes it easier for users to add `on` method calls for event handling without extending Folium itself. The functionality was inspired by PR #1866 by @yschopfer19. The PR was not accepted yet, because of concerns with code duplication. In the approach taken in the current PR, #1866 would not be necessary anymore, as the requested changes could be added completely in client code space. --- folium/elements.py | 83 ++++++++++++++++++++++++++++++++++++++++++ folium/map.py | 4 +- tests/test_features.py | 18 +++++++++ 3 files changed, 103 insertions(+), 2 deletions(-) diff --git a/folium/elements.py b/folium/elements.py index 9c41e66fa..56965b449 100644 --- a/folium/elements.py +++ b/folium/elements.py @@ -3,6 +3,8 @@ from branca.element import CssLink, Element, Figure, JavascriptLink, MacroElement from jinja2 import Template +from folium.utilities import JsCode + class JSCSSMixin(Element): """Render links to external Javascript and CSS resources.""" @@ -46,6 +48,87 @@ def _add_link(self, name: str, url: str, default_list: List[Tuple[str, str]]): default_list.append((name, url)) +class EventTargetMixin(Element): + '''Add Event Handlers to an element. + + Examples + -------- + >>> import folium + >>> from folium.utilities import JsCode + + >>> m = folium.Map() + + >>> geo_json_data = { + ... "type": "FeatureCollection", + ... "features": [ + ... { + ... "type": "Feature", + ... "geometry": { + ... "type": "Polygon", + ... "coordinates": [ + ... [ + ... [100.0, 0.0], + ... [101.0, 0.0], + ... [101.0, 1.0], + ... [100.0, 1.0], + ... [100.0, 0.0], + ... ] + ... ], + ... }, + ... "properties": {"prop1": {"title": "Somewhere on Sumatra"}}, + ... } + ... ], + ... } + + >>> g = folium.GeoJson(geo_json_data).add_to(m) + >>> highlight = JsCode( + ... """ + ... function highlight(e) { + ... e.target.original_color = e.layer.options.color; + ... e.target.setStyle({ color: "green" }); + ... } + ... """ + ... ) + >>> reset = JsCode( + ... """ + ... function reset(e) { + ... e.target.setStyle({ color: e.target.original_color }); + ... } + ... """ + ... ) + >>> g.on(mouseover=highlight, mouseout=reset) + ''' + + def on(self, **kwargs: JsCode): + for event, handler in kwargs.items(): + self.add_child(EventHandler(event, handler)) + return self + + def render(self, **kwargs) -> None: + super().render(**kwargs) + + +class EventHandler(MacroElement): + """Render Event Handlers.""" + + _template = Template( + """ + {% macro script(this, kwargs) %} + {{ this._parent.get_name()}}.on( + {{ this.event|tojson}}, + {{ this.handler.js_code }} + ); + {% endmacro %} + """ + ) + + def __init__(self, event: str, handler: JsCode): + super().__init__() + self._name = "EventHandler" + self.event = event + self.handler = handler + + class ElementAddToElement(MacroElement): """Abstract class to add an element to another element.""" diff --git a/folium/map.py b/folium/map.py index 01ac7d2ed..fb00ee72d 100644 --- a/folium/map.py +++ b/folium/map.py @@ -10,7 +10,7 @@ from branca.element import Element, Figure, Html, MacroElement from jinja2 import Template -from folium.elements import ElementAddToElement +from folium.elements import ElementAddToElement, EventTargetMixin from folium.utilities import ( TypeBounds, TypeJsonValue, @@ -21,7 +21,7 @@ ) -class Layer(MacroElement): +class Layer(EventTargetMixin, MacroElement): """An abstract class for everything that is a Layer on the map. It will be used to define whether an object will be included in LayerControls. diff --git a/tests/test_features.py b/tests/test_features.py index c879ad119..8dd51d61b 100644 --- a/tests/test_features.py +++ b/tests/test_features.py @@ -13,6 +13,7 @@ import folium from folium import Choropleth, ClickForMarker, GeoJson, Map, Popup +from folium.utilities import JsCode @pytest.fixture @@ -283,6 +284,23 @@ def test_geojson_empty_features_with_styling(): m.get_root().render() +def test_geojson_event_handler(): + """Test that event handlers are properly generated""" + m = Map() + data = {"type": "FeatureCollection", "features": []} + geojson = GeoJson(data, style_function=lambda x: {}).add_to(m) + fn = JsCode( + """ + function f(e) { + console.log("only for testing") + } + """ + ) + geojson.on(mouseover=fn) + rendered = m.get_root().render() + assert fn.js_code in rendered + + def test_geometry_collection_get_bounds(): """Assert #1599 is fixed""" geojson_data = {