diff --git a/.jshintrc b/.jshintrc index 591e845d..58a1441a 100644 --- a/.jshintrc +++ b/.jshintrc @@ -36,7 +36,8 @@ "subscribeToEvents": false, "createChart": false, "clearChart": false, - "hidePotentialLayers": false + "hidePotentialLayers": false, + "URLSearchParams": false }, "strict": "implied" } diff --git a/CHANGELOG.md b/CHANGELOG.md index b6ac61b3..3400e636 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -8,6 +8,7 @@ and this project tries to adhere to [Semantic Versioning](https://semver.org/spe ### Added - coupling of duplicated map panel controls - dependabot +- key results for wind, pv ground and pv roof settings panels ### Changed - Adapt municipality label font size according to zoom level diff --git a/Makefile b/Makefile index 01b5f0ac..e7bb51f0 100644 --- a/Makefile +++ b/Makefile @@ -90,4 +90,8 @@ update_vendor_assets: cp node_modules/shepherd.js/dist/js/shepherd.* digiplan/static/vendors/shepherd/ cp node_modules/shepherd.js/dist/css/shepherd.css digiplan/static/vendors/shepherd/ + # HTMX https://htmx.org/ + rm -r digiplan/static/vendors/htmx/js/* + cp node_modules/htmx.org/dist/htmx.min.js digiplan/static/vendors/htmx/js/ + # Done diff --git a/config/settings/base.py b/config/settings/base.py index 26af00f8..59bfaf63 100644 --- a/config/settings/base.py +++ b/config/settings/base.py @@ -90,6 +90,7 @@ THIRD_PARTY_APPS = [ "rest_framework", "django_distill", + "template_partials", ] LOCAL_APPS = ["digiplan.map.apps.MapConfig", "django_oemof", "django_mapengine"] diff --git a/digiplan/map/datapackage.py b/digiplan/map/datapackage.py index 6ce5ba37..690d5d6e 100644 --- a/digiplan/map/datapackage.py +++ b/digiplan/map/datapackage.py @@ -1,17 +1,36 @@ """Read functionality for digipipe datapackage.""" import csv import json -from collections import defaultdict +from collections import defaultdict, namedtuple from pathlib import Path -from typing import Optional +from typing import Optional, Union import pandas as pd +from cache_memoize import cache_memoize from django.conf import settings from django_oemof.settings import OEMOF_DIR -from config.settings.base import DATA_DIR +from config.settings.base import DIGIPIPE_DIR from digiplan.map import config, models +Source = namedtuple("Source", ["csv_file", "column"]) + + +def get_data_from_sources(sources: Union[Source, list[Source]]) -> pd.DataFrame: + """Extract data from single or multiple sources and merge into dataframe.""" + source_files = defaultdict(list) + if isinstance(sources, Source): + source_files[sources.csv_file].append(sources.column) + else: + for source in sources: + source_files[source.csv_file].append(source.column) + + dfs = [] + for source_file, columns in source_files.items(): + source_path = Path(DIGIPIPE_DIR, "scalars", source_file) + dfs.append(pd.read_csv(source_path, usecols=columns)) + return pd.concat(dfs) + def get_employment() -> pd.DataFrame: """Return employment data.""" @@ -144,70 +163,82 @@ def get_thermal_efficiency(component: str) -> float: return pd.read_csv(sequence_filename, sep=";").iloc[:, 1] -def get_potential_values(*, per_municipality: bool = False) -> dict: +@cache_memoize(timeout=None) +def get_potential_values() -> dict: """ Calculate max_values for sliders. - Parameters - ---------- - per_municipality: bool - If set to True, potentials are not aggregated, but given per municipality - Returns ------- dict dictionary with each slider / switch and respective max_value """ - scalars = { - "wind": "potentialarea_wind_area_stats_muns.csv", - "pv_ground": "potentialarea_pv_ground_area_stats_muns.csv", - "pv_roof": "potentialarea_pv_roof_area_stats_muns.csv", - } - - areas = { - "wind": { - "wind_2018": "stp_2018_eg", - "wind_2024": "stp_2024_vr", - "wind_2027": area - if (area := models.Municipality.objects.all().values("area").aggregate(models.Sum("area"))["area__sum"]) - else 0, # to prevent None if regions are empty - }, - "pv_ground": { - "pv_soil_quality_low": "soil_quality_low_region", - "pv_soil_quality_medium": "soil_quality_medium_region", - "pv_permanent_crops": "permanent_crops_region", - }, - "pv_roof": {"pv_roof": "installable_power"}, - } - + areas = get_potential_areas() pv_density = { "pv_soil_quality_low": "pv_ground", "pv_soil_quality_medium": "pv_ground_vertical_bifacial", "pv_permanent_crops": "pv_ground_elevated", + "pv_roof": "pv_roof", } power_density = json.load(Path.open(Path(settings.DIGIPIPE_DIR, "scalars/technology_data.json")))["power_density"] potentials = {} - for profile in areas: - path = Path(DATA_DIR, "digipipe/scalars", scalars[profile]) - reader = pd.read_csv(path) - for key, value in areas[profile].items(): - if key == "wind_2027": - # Value is already calculated from region area (see above) - potentials[key] = value - else: - if per_municipality: # noqa: PLR5501 - potentials[key] = reader[value] - else: - potentials[key] = reader[value].sum() - if profile == "wind": - potentials[key] = potentials[key] * power_density["wind"] - if profile == "pv_ground": - potentials[key] = potentials[key] * power_density[pv_density[key]] + for key in areas: + if key.startswith("wind"): + potentials[key] = areas[key] * power_density["wind"] + if key.startswith("pv"): + potentials[key] = areas[key] * power_density[pv_density[key]] return potentials +@cache_memoize(timeout=None) +def get_potential_areas(technology: Optional[str] = None) -> dict: + """ + Return potential areas. + + Parameters + ---------- + technology: str + If given, potential area only for this technology is returned + + Returns + ------- + dict + Potential areas of all technologies or specified one (in sqkm) + """ + sources = { + "wind_2018": Source("potentialarea_wind_area_stats_muns.csv", "stp_2018_eg"), + "wind_2024": Source("potentialarea_wind_area_stats_muns.csv", "stp_2024_vr"), + "pv_soil_quality_low": Source("potentialarea_pv_ground_area_stats_muns.csv", "soil_quality_low_region"), + "pv_soil_quality_medium": Source("potentialarea_pv_ground_area_stats_muns.csv", "soil_quality_medium_region"), + "pv_permanent_crops": Source("potentialarea_pv_ground_area_stats_muns.csv", "permanent_crops_region"), + "pv_roof": Source("potentialarea_pv_roof_area_stats_muns.csv", "roof_area_pv_potential_sqkm"), + } + + # Add wind for 2027 directly from model data, as it is not included in datapackage + areas = { + "wind_2027": area + if (area := models.Municipality.objects.all().values("area").aggregate(models.Sum("area"))["area__sum"]) + else 0, + } + if technology is not None: + if technology == "wind_2027": + return areas["wind_2027"] + sources = {technology: sources[technology]} + + data = get_data_from_sources(sources.values()) + data = data.sum() + for index in data.index: + # Add extracted data to areas and map source columns to keys accordingly + areas[next(key for key, source in sources.items() if source.column == index)] = data[index] + + if technology is not None: + return areas[technology] + return areas + + +@cache_memoize(timeout=None) def get_full_load_hours(year: int) -> pd.Series: """Return full load hours for given year.""" full_load_hours = pd.Series( @@ -217,6 +248,7 @@ def get_full_load_hours(year: int) -> pd.Series: return full_load_hours +@cache_memoize(timeout=None) def get_capacities_from_datapackage() -> pd.DataFrame: """Return renewable capacities for given year from datapackage.""" capacities = pd.concat( @@ -256,6 +288,7 @@ def get_capacities_from_sliders(year: int) -> pd.Series: return slider_settings +@cache_memoize(timeout=None) def get_power_density(technology: Optional[str] = None) -> dict: """Return power density for technology.""" if technology: diff --git a/digiplan/map/forms.py b/digiplan/map/forms.py index 76a43513..73c4cfd9 100644 --- a/digiplan/map/forms.py +++ b/digiplan/map/forms.py @@ -1,20 +1,28 @@ -from itertools import count # noqa: D100 +"""Module containing django forms.""" +from __future__ import annotations + +from itertools import count +from typing import TYPE_CHECKING from django.forms import BooleanField, Form, IntegerField, TextInput, renderers +from django.shortcuts import reverse from django.utils.safestring import mark_safe -from django_mapengine import legend -from . import charts +from . import charts, menu from .widgets import SwitchWidget +if TYPE_CHECKING: + from django_mapengine import legend + class TemplateForm(Form): # noqa: D101 template_name = None + extra_content = {} def __str__(self) -> str: # noqa: D105 if self.template_name: renderer = renderers.get_default_renderer() - return mark_safe(renderer.render(self.template_name, {"form": self})) # noqa: S308 + return mark_safe(renderer.render(self.template_name, {"form": self, **self.extra_content})) # noqa: S308 return super().__str__() @@ -54,33 +62,36 @@ def __init__(self, parameters, additional_parameters=None, **kwargs) -> None: # super().__init__(**kwargs) self.fields = {item["name"]: item["field"] for item in self.generate_fields(parameters, additional_parameters)} - @staticmethod - def generate_fields(parameters, additional_parameters=None): # noqa: ANN001, ANN205, D102 + def get_field_attrs(self, name: str, parameters: dict) -> dict: # noqa: ARG002 + """Set up field attributes from parameters.""" + attrs = { + "class": parameters["class"], + "data-min": parameters["min"], + "data-max": parameters["max"], + "data-from": parameters["start"], + "data-grid": "true" if "grid" in parameters and parameters["grid"] else "false", + "data-has-sidepanel": "true" if "sidepanel" in parameters else "false", + "data-color": parameters["color"] if "color" in parameters else "", + } + if "to" in parameters: + attrs["data-to"] = parameters["to"] + if "step" in parameters: + attrs["data-step"] = parameters["step"] + if "from-min" in parameters: + attrs["data-from-min"] = parameters["from-min"] + if "from-max" in parameters: + attrs["data-from-max"] = parameters["from-max"] + return attrs + + def generate_fields(self, parameters: dict, additional_parameters: dict | None = None) -> dict: + """Create fields from config parameters.""" if additional_parameters is not None: charts.merge_dicts(parameters, additional_parameters) for name, item in parameters.items(): if item["type"] == "slider": - attrs = { - "class": item["class"], - "data-min": item["min"], - "data-max": item["max"], - "data-from": item["start"], - "data-grid": "true" if "grid" in item and item["grid"] else "false", - "data-has-sidepanel": "true" if "sidepanel" in item else "false", - "data-color": item["color"] if "color" in item else "", - } - if "to" in item: - attrs["data-to"] = item["to"] - if "step" in item: - attrs["data-step"] = item["step"] - if "from-min" in item: - attrs["data-from-min"] = item["from-min"] - if "from-max" in item: - attrs["data-from-max"] = item["from-max"] - field = IntegerField( label=item["label"], - widget=TextInput(attrs=attrs), + widget=TextInput(attrs=self.get_field_attrs(name, item)), help_text=item["tooltip"], required=item.get("required", True), ) @@ -103,6 +114,32 @@ def generate_fields(parameters, additional_parameters=None): # noqa: ANN001, AN class EnergyPanelForm(PanelForm): # noqa: D101 template_name = "forms/panel_energy.html" + def __init__(self, parameters, additional_parameters=None, **kwargs) -> None: # noqa: ANN001 + """Overwrite init function to add initial key results for detail panels.""" + super().__init__(parameters, additional_parameters, **kwargs) + for technology in ("wind_2018", "wind_2024", "wind_2027", "pv_ground", "pv_roof"): + # get initial slider values for wind and pv: + key_results = menu.detail_key_results( + technology, + id_s_w_6=parameters["s_w_6"]["start"], + id_s_w_7=parameters["s_w_7"]["start"], + id_s_pv_ff_3=parameters["s_pv_ff_3"]["start"], + id_s_pv_ff_4=parameters["s_pv_ff_4"]["start"], + id_s_pv_ff_5=parameters["s_pv_ff_5"]["start"], + id_s_pv_d_3=parameters["s_pv_d_3"]["start"], + ) + for key, value in key_results.items(): + self.extra_content[f"{technology}_key_result_{key}"] = value + + def get_field_attrs(self, name: str, parameters: dict) -> dict: + """Add HTMX attributes to wind and pv detail sliders.""" + detail_slider_targets = {"s_w_6": "wind_key_results_2024", "s_w_7": "wind_key_results_2027"} + attrs = super().get_field_attrs(name, parameters) + if name in detail_slider_targets: + attrs["hx-get"] = reverse("map:detail_key_results") + attrs["hx-target"] = detail_slider_targets[name] + return attrs + class HeatPanelForm(PanelForm): # noqa: D101 template_name = "forms/panel_heat.html" diff --git a/digiplan/map/menu.py b/digiplan/map/menu.py new file mode 100644 index 00000000..73cecbba --- /dev/null +++ b/digiplan/map/menu.py @@ -0,0 +1,49 @@ +"""Add calculations for menu items.""" + +from . import config, datapackage + + +def detail_key_results(technology: str, **kwargs: dict) -> dict: + """Calculate detail key results for given technology.""" + areas = datapackage.get_potential_areas() + potential_capacities = datapackage.get_potential_values() # in MW + full_load_hours = datapackage.get_full_load_hours(2045) + nominal_power_per_unit = config.TECHNOLOGY_DATA["nominal_power_per_unit"]["wind"] + + if technology.startswith("wind"): + percentage = 1 + if technology == "wind_2024": + percentage = float(kwargs["id_s_w_6"]) / float(config.ENERGY_SETTINGS_PANEL["s_w_6"]["max"]) + if technology == "wind_2027": + percentage = float(kwargs["id_s_w_7"]) / 100 + return { + "area": areas[technology] * 100 * percentage, + "turbines": potential_capacities[technology] / nominal_power_per_unit * percentage, + "energy": potential_capacities[technology] * full_load_hours["wind"] * percentage * 1e-6, + } + if technology == "pv_ground": + percentages = { + "pv_soil_quality_low": int(kwargs["id_s_pv_ff_3"]) / 100, + "pv_soil_quality_medium": int(kwargs["id_s_pv_ff_4"]) / 100, + "pv_permanent_crops": int(kwargs["id_s_pv_ff_5"]) / 100, + } + flh_mapping = { + "pv_soil_quality_low": "pv_ground", + "pv_soil_quality_medium": "pv_ground_vertical_bifacial", + "pv_permanent_crops": "pv_ground_elevated", + } + return { + "area": sum(areas[pv_type] * 100 * percentages[pv_type] for pv_type in percentages), + "energy": sum( + potential_capacities[pv_type] * full_load_hours[flh_mapping[pv_type]] * percentages[pv_type] + for pv_type in percentages + ) + * 1e-6, + } + if technology == "pv_roof": + percentage = int(kwargs["id_s_pv_d_3"]) / 100 + return { + "area": areas[technology] * 100 * percentage, + "energy": potential_capacities[technology] * full_load_hours[technology] * percentage * 1e-6, + } + raise KeyError(f"Unknown technology '{technology}'.") diff --git a/digiplan/map/urls.py b/digiplan/map/urls.py index 2a3d8b28..9bdb13f8 100644 --- a/digiplan/map/urls.py +++ b/digiplan/map/urls.py @@ -12,4 +12,5 @@ path("choropleth//", views.get_choropleth, name="choropleth"), path("popup//", views.get_popup, name="popup"), path("charts", views.get_charts, name="charts"), + path("detail_key_results", views.DetailKeyResultsView.as_view(), name="detail_key_results"), ] diff --git a/digiplan/map/views.py b/digiplan/map/views.py index ad4f073a..53528e6c 100644 --- a/digiplan/map/views.py +++ b/digiplan/map/views.py @@ -11,7 +11,7 @@ from django_mapengine import views from digiplan import __version__ -from digiplan.map import config +from digiplan.map import config, menu from . import charts, choropleths, forms, map_config, popups, utils @@ -195,3 +195,13 @@ def get_charts(request: HttpRequest) -> response.JsonResponse: return response.JsonResponse( {lookup: charts.CHARTS[lookup](simulation_id=simulation_id).render() for lookup in lookups}, ) + + +class DetailKeyResultsView(TemplateView): + """Return HTMX-partial for requested detail key results.""" + + template_name = "forms/panel_energy.html#key_results" + + def get_context_data(self, **kwargs) -> dict: # noqa: ARG002 + """Get detail key results for requested technology.""" + return {f"key_result_{key}": value for key, value in menu.detail_key_results(**self.request.GET.dict()).items()} diff --git a/digiplan/static/js/sliders.js b/digiplan/static/js/sliders.js index b5466046..640af9a9 100644 --- a/digiplan/static/js/sliders.js +++ b/digiplan/static/js/sliders.js @@ -39,19 +39,19 @@ const pvMapControl = document.getElementsByClassName("map__layers-pv")[0]; // Setup $(".js-slider.js-slider-panel.js-power-mix").ionRangeSlider({ - onChange: function (data) { + onFinish: function (data) { PubSub.publish(eventTopics.POWER_PANEL_SLIDER_CHANGE, data); }, }); $(".js-slider.js-slider-panel").ionRangeSlider({ - onChange: function (data) { + onFinish: function (data) { PubSub.publish(eventTopics.PANEL_SLIDER_CHANGE, data); }, }); $(".js-slider.js-slider-detail-panel").ionRangeSlider({ - onChange: function (data) { + onFinish: function (data) { PubSub.publish(eventTopics.DETAIL_PANEL_SLIDER_CHANGE, data); }, }); @@ -92,6 +92,7 @@ PubSub.subscribe( PubSub.subscribe(eventTopics.PANEL_SLIDER_CHANGE, hidePotentialLayers); PubSub.subscribe(eventTopics.PANEL_SLIDER_CHANGE, adaptDetailSliders); PubSub.subscribe(eventTopics.DETAIL_PANEL_SLIDER_CHANGE, adaptMainSliders); +PubSub.subscribe(eventTopics.DETAIL_PANEL_SLIDER_CHANGE, adaptDetailKeyResults); PubSub.subscribe( eventTopics.MORE_LABEL_CLICK, showOrHideSidepanelsOnMoreLabelClick, @@ -407,6 +408,65 @@ function highlightPVMapControls(msg) { return logMessage(msg); } +function adaptDetailKeyResults(msg, data) { + const slider_id = data.input[0].id; + let technology; + let target; + let url_data = {}; + + if (slider_id === "id_s_w_6") { + technology = "wind_2024"; + url_data.id_s_w_6 = data.from; + target = "wind_key_results_2024"; + } else if (slider_id === "id_s_w_7") { + technology = "wind_2027"; + url_data.id_s_w_7 = data.from; + target = "wind_key_results_2027"; + } else if ( + ["id_s_pv_ff_3", "id_s_pv_ff_4", "id_s_pv_ff_5"].includes(slider_id) + ) { + technology = "pv_ground"; + url_data.id_s_pv_ff_3 = + $("#id_s_pv_ff_3").data("ionRangeSlider").result.from; + url_data.id_s_pv_ff_4 = + $("#id_s_pv_ff_4").data("ionRangeSlider").result.from; + url_data.id_s_pv_ff_5 = + $("#id_s_pv_ff_5").data("ionRangeSlider").result.from; + target = "pv_ground_key_results"; + } else if (slider_id === "id_s_pv_d_3") { + technology = "pv_roof"; + url_data.id_s_pv_d_3 = data.from; + target = "pv_roof_key_results"; + } else { + return logMessage(msg); + } + + const query = new URLSearchParams(url_data).toString(); + let url = `/detail_key_results?technology=${technology}&${query}`; + fetch(url) + .then((response) => { + // Check if the response is successful + if (!response.ok) { + throw new Error( + `Error requesting detail key results for slider ${slider_id}`, + ); + } + // Return the response as HTML + return response.text(); + }) + .then((html) => { + // Insert the HTML into the DOM + document.getElementById(target).innerHTML = html; + }) + .catch((error) => { + console.error( + `Error requesting detail key results for slider ${slider_id}:`, + error, + ); + }); + return logMessage(msg); +} + // Helper Functions function getColorsByIds(ids) { diff --git a/digiplan/static/vendors/htmx/js/htmx.min.js b/digiplan/static/vendors/htmx/js/htmx.min.js new file mode 100644 index 00000000..d68f3c6c --- /dev/null +++ b/digiplan/static/vendors/htmx/js/htmx.min.js @@ -0,0 +1 @@ +(function(e,t){if(typeof define==="function"&&define.amd){define([],t)}else if(typeof module==="object"&&module.exports){module.exports=t()}else{e.htmx=e.htmx||t()}})(typeof self!=="undefined"?self:this,function(){return function(){"use strict";var Q={onLoad:B,process:zt,on:de,off:ge,trigger:ce,ajax:Nr,find:C,findAll:f,closest:v,values:function(e,t){var r=dr(e,t||"post");return r.values},remove:_,addClass:z,removeClass:n,toggleClass:$,takeClass:W,defineExtension:Ur,removeExtension:Fr,logAll:V,logNone:j,logger:null,config:{historyEnabled:true,historyCacheSize:10,refreshOnHistoryMiss:false,defaultSwapStyle:"innerHTML",defaultSwapDelay:0,defaultSettleDelay:20,includeIndicatorStyles:true,indicatorClass:"htmx-indicator",requestClass:"htmx-request",addedClass:"htmx-added",settlingClass:"htmx-settling",swappingClass:"htmx-swapping",allowEval:true,allowScriptTags:true,inlineScriptNonce:"",attributesToSettle:["class","style","width","height"],withCredentials:false,timeout:0,wsReconnectDelay:"full-jitter",wsBinaryType:"blob",disableSelector:"[hx-disable], [data-hx-disable]",useTemplateFragments:false,scrollBehavior:"smooth",defaultFocusScroll:false,getCacheBusterParam:false,globalViewTransitions:false,methodsThatUseUrlParams:["get"],selfRequestsOnly:false,ignoreTitle:false,scrollIntoViewOnBoost:true,triggerSpecsCache:null},parseInterval:d,_:t,createEventSource:function(e){return new EventSource(e,{withCredentials:true})},createWebSocket:function(e){var t=new WebSocket(e,[]);t.binaryType=Q.config.wsBinaryType;return t},version:"1.9.11"};var r={addTriggerHandler:Lt,bodyContains:se,canAccessLocalStorage:U,findThisElement:xe,filterValues:yr,hasAttribute:o,getAttributeValue:te,getClosestAttributeValue:ne,getClosestMatch:c,getExpressionVars:Hr,getHeaders:xr,getInputValues:dr,getInternalData:ae,getSwapSpecification:wr,getTriggerSpecs:it,getTarget:ye,makeFragment:l,mergeObjects:le,makeSettleInfo:T,oobSwap:Ee,querySelectorExt:ue,selectAndSwap:je,settleImmediately:nr,shouldCancel:ut,triggerEvent:ce,triggerErrorEvent:fe,withExtensions:R};var w=["get","post","put","delete","patch"];var i=w.map(function(e){return"[hx-"+e+"], [data-hx-"+e+"]"}).join(", ");var S=e("head"),q=e("title"),H=e("svg",true);function e(e,t=false){return new RegExp(`<${e}(\\s[^>]*>|>)([\\s\\S]*?)<\\/${e}>`,t?"gim":"im")}function d(e){if(e==undefined){return undefined}let t=NaN;if(e.slice(-2)=="ms"){t=parseFloat(e.slice(0,-2))}else if(e.slice(-1)=="s"){t=parseFloat(e.slice(0,-1))*1e3}else if(e.slice(-1)=="m"){t=parseFloat(e.slice(0,-1))*1e3*60}else{t=parseFloat(e)}return isNaN(t)?undefined:t}function ee(e,t){return e.getAttribute&&e.getAttribute(t)}function o(e,t){return e.hasAttribute&&(e.hasAttribute(t)||e.hasAttribute("data-"+t))}function te(e,t){return ee(e,t)||ee(e,"data-"+t)}function u(e){return e.parentElement}function re(){return document}function c(e,t){while(e&&!t(e)){e=u(e)}return e?e:null}function L(e,t,r){var n=te(t,r);var i=te(t,"hx-disinherit");if(e!==t&&i&&(i==="*"||i.split(" ").indexOf(r)>=0)){return"unset"}else{return n}}function ne(t,r){var n=null;c(t,function(e){return n=L(t,e,r)});if(n!=="unset"){return n}}function h(e,t){var r=e.matches||e.matchesSelector||e.msMatchesSelector||e.mozMatchesSelector||e.webkitMatchesSelector||e.oMatchesSelector;return r&&r.call(e,t)}function A(e){var t=/<([a-z][^\/\0>\x20\t\r\n\f]*)/i;var r=t.exec(e);if(r){return r[1].toLowerCase()}else{return""}}function s(e,t){var r=new DOMParser;var n=r.parseFromString(e,"text/html");var i=n.body;while(t>0){t--;i=i.firstChild}if(i==null){i=re().createDocumentFragment()}return i}function N(e){return/",0);var a=i.querySelector("template").content;if(Q.config.allowScriptTags){oe(a.querySelectorAll("script"),function(e){if(Q.config.inlineScriptNonce){e.nonce=Q.config.inlineScriptNonce}e.htmxExecuted=navigator.userAgent.indexOf("Firefox")===-1})}else{oe(a.querySelectorAll("script"),function(e){_(e)})}return a}switch(r){case"thead":case"tbody":case"tfoot":case"colgroup":case"caption":return s(""+n+"
",1);case"col":return s(""+n+"
",2);case"tr":return s(""+n+"
",2);case"td":case"th":return s(""+n+"
",3);case"script":case"style":return s("
"+n+"
",1);default:return s(n,0)}}function ie(e){if(e){e()}}function I(e,t){return Object.prototype.toString.call(e)==="[object "+t+"]"}function k(e){return I(e,"Function")}function P(e){return I(e,"Object")}function ae(e){var t="htmx-internal-data";var r=e[t];if(!r){r=e[t]={}}return r}function M(e){var t=[];if(e){for(var r=0;r=0}function se(e){if(e.getRootNode&&e.getRootNode()instanceof window.ShadowRoot){return re().body.contains(e.getRootNode().host)}else{return re().body.contains(e)}}function D(e){return e.trim().split(/\s+/)}function le(e,t){for(var r in t){if(t.hasOwnProperty(r)){e[r]=t[r]}}return e}function E(e){try{return JSON.parse(e)}catch(e){b(e);return null}}function U(){var e="htmx:localStorageTest";try{localStorage.setItem(e,e);localStorage.removeItem(e);return true}catch(e){return false}}function F(t){try{var e=new URL(t);if(e){t=e.pathname+e.search}if(!/^\/$/.test(t)){t=t.replace(/\/+$/,"")}return t}catch(e){return t}}function t(e){return Tr(re().body,function(){return eval(e)})}function B(t){var e=Q.on("htmx:load",function(e){t(e.detail.elt)});return e}function V(){Q.logger=function(e,t,r){if(console){console.log(t,e,r)}}}function j(){Q.logger=null}function C(e,t){if(t){return e.querySelector(t)}else{return C(re(),e)}}function f(e,t){if(t){return e.querySelectorAll(t)}else{return f(re(),e)}}function _(e,t){e=p(e);if(t){setTimeout(function(){_(e);e=null},t)}else{e.parentElement.removeChild(e)}}function z(e,t,r){e=p(e);if(r){setTimeout(function(){z(e,t);e=null},r)}else{e.classList&&e.classList.add(t)}}function n(e,t,r){e=p(e);if(r){setTimeout(function(){n(e,t);e=null},r)}else{if(e.classList){e.classList.remove(t);if(e.classList.length===0){e.removeAttribute("class")}}}}function $(e,t){e=p(e);e.classList.toggle(t)}function W(e,t){e=p(e);oe(e.parentElement.children,function(e){n(e,t)});z(e,t)}function v(e,t){e=p(e);if(e.closest){return e.closest(t)}else{do{if(e==null||h(e,t)){return e}}while(e=e&&u(e));return null}}function g(e,t){return e.substring(0,t.length)===t}function G(e,t){return e.substring(e.length-t.length)===t}function J(e){var t=e.trim();if(g(t,"<")&&G(t,"/>")){return t.substring(1,t.length-2)}else{return t}}function Z(e,t){if(t.indexOf("closest ")===0){return[v(e,J(t.substr(8)))]}else if(t.indexOf("find ")===0){return[C(e,J(t.substr(5)))]}else if(t==="next"){return[e.nextElementSibling]}else if(t.indexOf("next ")===0){return[K(e,J(t.substr(5)))]}else if(t==="previous"){return[e.previousElementSibling]}else if(t.indexOf("previous ")===0){return[Y(e,J(t.substr(9)))]}else if(t==="document"){return[document]}else if(t==="window"){return[window]}else if(t==="body"){return[document.body]}else{return re().querySelectorAll(J(t))}}var K=function(e,t){var r=re().querySelectorAll(t);for(var n=0;n=0;n--){var i=r[n];if(i.compareDocumentPosition(e)===Node.DOCUMENT_POSITION_FOLLOWING){return i}}};function ue(e,t){if(t){return Z(e,t)[0]}else{return Z(re().body,e)[0]}}function p(e){if(I(e,"String")){return C(e)}else{return e}}function ve(e,t,r){if(k(t)){return{target:re().body,event:e,listener:t}}else{return{target:p(e),event:t,listener:r}}}function de(t,r,n){jr(function(){var e=ve(t,r,n);e.target.addEventListener(e.event,e.listener)});var e=k(r);return e?r:n}function ge(t,r,n){jr(function(){var e=ve(t,r,n);e.target.removeEventListener(e.event,e.listener)});return k(r)?r:n}var pe=re().createElement("output");function me(e,t){var r=ne(e,t);if(r){if(r==="this"){return[xe(e,t)]}else{var n=Z(e,r);if(n.length===0){b('The selector "'+r+'" on '+t+" returned no matches!");return[pe]}else{return n}}}}function xe(e,t){return c(e,function(e){return te(e,t)!=null})}function ye(e){var t=ne(e,"hx-target");if(t){if(t==="this"){return xe(e,"hx-target")}else{return ue(e,t)}}else{var r=ae(e);if(r.boosted){return re().body}else{return e}}}function be(e){var t=Q.config.attributesToSettle;for(var r=0;r0){o=e.substr(0,e.indexOf(":"));t=e.substr(e.indexOf(":")+1,e.length)}else{o=e}var r=re().querySelectorAll(t);if(r){oe(r,function(e){var t;var r=i.cloneNode(true);t=re().createDocumentFragment();t.appendChild(r);if(!Se(o,e)){t=r}var n={shouldSwap:true,target:e,fragment:t};if(!ce(e,"htmx:oobBeforeSwap",n))return;e=n.target;if(n["shouldSwap"]){Be(o,e,e,t,a)}oe(a.elts,function(e){ce(e,"htmx:oobAfterSwap",n)})});i.parentNode.removeChild(i)}else{i.parentNode.removeChild(i);fe(re().body,"htmx:oobErrorNoTarget",{content:i})}return e}function Ce(e,t,r){var n=ne(e,"hx-select-oob");if(n){var i=n.split(",");for(var a=0;a0){var r=t.replace("'","\\'");var n=e.tagName.replace(":","\\:");var i=o.querySelector(n+"[id='"+r+"']");if(i&&i!==o){var a=e.cloneNode();we(e,i);s.tasks.push(function(){we(e,a)})}}})}function Oe(e){return function(){n(e,Q.config.addedClass);zt(e);Nt(e);qe(e);ce(e,"htmx:load")}}function qe(e){var t="[autofocus]";var r=h(e,t)?e:e.querySelector(t);if(r!=null){r.focus()}}function a(e,t,r,n){Te(e,r,n);while(r.childNodes.length>0){var i=r.firstChild;z(i,Q.config.addedClass);e.insertBefore(i,t);if(i.nodeType!==Node.TEXT_NODE&&i.nodeType!==Node.COMMENT_NODE){n.tasks.push(Oe(i))}}}function He(e,t){var r=0;while(r-1){var t=e.replace(H,"");var r=t.match(q);if(r){return r[2]}}}function je(e,t,r,n,i,a){i.title=Ve(n);var o=l(n);if(o){Ce(r,o,i);o=Fe(r,o,a);Re(o);return Be(e,r,t,o,i)}}function _e(e,t,r){var n=e.getResponseHeader(t);if(n.indexOf("{")===0){var i=E(n);for(var a in i){if(i.hasOwnProperty(a)){var o=i[a];if(!P(o)){o={value:o}}ce(r,a,o)}}}else{var s=n.split(",");for(var l=0;l0){var o=t[0];if(o==="]"){n--;if(n===0){if(a===null){i=i+"true"}t.shift();i+=")})";try{var s=Tr(e,function(){return Function(i)()},function(){return true});s.source=i;return s}catch(e){fe(re().body,"htmx:syntax:error",{error:e,source:i});return null}}}else if(o==="["){n++}if(Qe(o,a,r)){i+="(("+r+"."+o+") ? ("+r+"."+o+") : (window."+o+"))"}else{i=i+o}a=t.shift()}}}function y(e,t){var r="";while(e.length>0&&!t.test(e[0])){r+=e.shift()}return r}function tt(e){var t;if(e.length>0&&Ze.test(e[0])){e.shift();t=y(e,Ke).trim();e.shift()}else{t=y(e,x)}return t}var rt="input, textarea, select";function nt(e,t,r){var n=[];var i=Ye(t);do{y(i,Je);var a=i.length;var o=y(i,/[,\[\s]/);if(o!==""){if(o==="every"){var s={trigger:"every"};y(i,Je);s.pollInterval=d(y(i,/[,\[\s]/));y(i,Je);var l=et(e,i,"event");if(l){s.eventFilter=l}n.push(s)}else if(o.indexOf("sse:")===0){n.push({trigger:"sse",sseEvent:o.substr(4)})}else{var u={trigger:o};var l=et(e,i,"event");if(l){u.eventFilter=l}while(i.length>0&&i[0]!==","){y(i,Je);var f=i.shift();if(f==="changed"){u.changed=true}else if(f==="once"){u.once=true}else if(f==="consume"){u.consume=true}else if(f==="delay"&&i[0]===":"){i.shift();u.delay=d(y(i,x))}else if(f==="from"&&i[0]===":"){i.shift();if(Ze.test(i[0])){var c=tt(i)}else{var c=y(i,x);if(c==="closest"||c==="find"||c==="next"||c==="previous"){i.shift();var h=tt(i);if(h.length>0){c+=" "+h}}}u.from=c}else if(f==="target"&&i[0]===":"){i.shift();u.target=tt(i)}else if(f==="throttle"&&i[0]===":"){i.shift();u.throttle=d(y(i,x))}else if(f==="queue"&&i[0]===":"){i.shift();u.queue=y(i,x)}else if(f==="root"&&i[0]===":"){i.shift();u[f]=tt(i)}else if(f==="threshold"&&i[0]===":"){i.shift();u[f]=y(i,x)}else{fe(e,"htmx:syntax:error",{token:i.shift()})}}n.push(u)}}if(i.length===a){fe(e,"htmx:syntax:error",{token:i.shift()})}y(i,Je)}while(i[0]===","&&i.shift());if(r){r[t]=n}return n}function it(e){var t=te(e,"hx-trigger");var r=[];if(t){var n=Q.config.triggerSpecsCache;r=n&&n[t]||nt(e,t,n)}if(r.length>0){return r}else if(h(e,"form")){return[{trigger:"submit"}]}else if(h(e,'input[type="button"], input[type="submit"]')){return[{trigger:"click"}]}else if(h(e,rt)){return[{trigger:"change"}]}else{return[{trigger:"click"}]}}function at(e){ae(e).cancelled=true}function ot(e,t,r){var n=ae(e);n.timeout=setTimeout(function(){if(se(e)&&n.cancelled!==true){if(!ct(r,e,Wt("hx:poll:trigger",{triggerSpec:r,target:e}))){t(e)}ot(e,t,r)}},r.pollInterval)}function st(e){return location.hostname===e.hostname&&ee(e,"href")&&ee(e,"href").indexOf("#")!==0}function lt(t,r,e){if(t.tagName==="A"&&st(t)&&(t.target===""||t.target==="_self")||t.tagName==="FORM"){r.boosted=true;var n,i;if(t.tagName==="A"){n="get";i=ee(t,"href")}else{var a=ee(t,"method");n=a?a.toLowerCase():"get";if(n==="get"){}i=ee(t,"action")}e.forEach(function(e){ht(t,function(e,t){if(v(e,Q.config.disableSelector)){m(e);return}he(n,i,e,t)},r,e,true)})}}function ut(e,t){if(e.type==="submit"||e.type==="click"){if(t.tagName==="FORM"){return true}if(h(t,'input[type="submit"], button')&&v(t,"form")!==null){return true}if(t.tagName==="A"&&t.href&&(t.getAttribute("href")==="#"||t.getAttribute("href").indexOf("#")!==0)){return true}}return false}function ft(e,t){return ae(e).boosted&&e.tagName==="A"&&t.type==="click"&&(t.ctrlKey||t.metaKey)}function ct(e,t,r){var n=e.eventFilter;if(n){try{return n.call(t,r)!==true}catch(e){fe(re().body,"htmx:eventFilter:error",{error:e,source:n.source});return true}}return false}function ht(a,o,e,s,l){var u=ae(a);var t;if(s.from){t=Z(a,s.from)}else{t=[a]}if(s.changed){t.forEach(function(e){var t=ae(e);t.lastValue=e.value})}oe(t,function(n){var i=function(e){if(!se(a)){n.removeEventListener(s.trigger,i);return}if(ft(a,e)){return}if(l||ut(e,a)){e.preventDefault()}if(ct(s,a,e)){return}var t=ae(e);t.triggerSpec=s;if(t.handledFor==null){t.handledFor=[]}if(t.handledFor.indexOf(a)<0){t.handledFor.push(a);if(s.consume){e.stopPropagation()}if(s.target&&e.target){if(!h(e.target,s.target)){return}}if(s.once){if(u.triggeredOnce){return}else{u.triggeredOnce=true}}if(s.changed){var r=ae(n);if(r.lastValue===n.value){return}r.lastValue=n.value}if(u.delayed){clearTimeout(u.delayed)}if(u.throttle){return}if(s.throttle>0){if(!u.throttle){o(a,e);u.throttle=setTimeout(function(){u.throttle=null},s.throttle)}}else if(s.delay>0){u.delayed=setTimeout(function(){o(a,e)},s.delay)}else{ce(a,"htmx:trigger");o(a,e)}}};if(e.listenerInfos==null){e.listenerInfos=[]}e.listenerInfos.push({trigger:s.trigger,listener:i,on:n});n.addEventListener(s.trigger,i)})}var vt=false;var dt=null;function gt(){if(!dt){dt=function(){vt=true};window.addEventListener("scroll",dt);setInterval(function(){if(vt){vt=false;oe(re().querySelectorAll("[hx-trigger='revealed'],[data-hx-trigger='revealed']"),function(e){pt(e)})}},200)}}function pt(t){if(!o(t,"data-hx-revealed")&&X(t)){t.setAttribute("data-hx-revealed","true");var e=ae(t);if(e.initHash){ce(t,"revealed")}else{t.addEventListener("htmx:afterProcessNode",function(e){ce(t,"revealed")},{once:true})}}}function mt(e,t,r){var n=D(r);for(var i=0;i=0){var t=wt(n);setTimeout(function(){xt(s,r,n+1)},t)}};t.onopen=function(e){n=0};ae(s).webSocket=t;t.addEventListener("message",function(e){if(yt(s)){return}var t=e.data;R(s,function(e){t=e.transformResponse(t,null,s)});var r=T(s);var n=l(t);var i=M(n.children);for(var a=0;a0){ce(u,"htmx:validation:halted",i);return}t.send(JSON.stringify(l));if(ut(e,u)){e.preventDefault()}})}else{fe(u,"htmx:noWebSocketSourceError")}}function wt(e){var t=Q.config.wsReconnectDelay;if(typeof t==="function"){return t(e)}if(t==="full-jitter"){var r=Math.min(e,6);var n=1e3*Math.pow(2,r);return n*Math.random()}b('htmx.config.wsReconnectDelay must either be a function or the string "full-jitter"')}function St(e,t,r){var n=D(r);for(var i=0;i0){setTimeout(i,n)}else{i()}}function Ht(t,i,e){var a=false;oe(w,function(r){if(o(t,"hx-"+r)){var n=te(t,"hx-"+r);a=true;i.path=n;i.verb=r;e.forEach(function(e){Lt(t,e,i,function(e,t){if(v(e,Q.config.disableSelector)){m(e);return}he(r,n,e,t)})})}});return a}function Lt(n,e,t,r){if(e.sseEvent){Rt(n,r,e.sseEvent)}else if(e.trigger==="revealed"){gt();ht(n,r,t,e);pt(n)}else if(e.trigger==="intersect"){var i={};if(e.root){i.root=ue(n,e.root)}if(e.threshold){i.threshold=parseFloat(e.threshold)}var a=new IntersectionObserver(function(e){for(var t=0;t0){t.polling=true;ot(n,r,e)}else{ht(n,r,t,e)}}function At(e){if(!e.htmxExecuted&&Q.config.allowScriptTags&&(e.type==="text/javascript"||e.type==="module"||e.type==="")){var t=re().createElement("script");oe(e.attributes,function(e){t.setAttribute(e.name,e.value)});t.textContent=e.textContent;t.async=false;if(Q.config.inlineScriptNonce){t.nonce=Q.config.inlineScriptNonce}var r=e.parentElement;try{r.insertBefore(t,e)}catch(e){b(e)}finally{if(e.parentElement){e.parentElement.removeChild(e)}}}}function Nt(e){if(h(e,"script")){At(e)}oe(f(e,"script"),function(e){At(e)})}function It(e){var t=e.attributes;for(var r=0;r0){var o=n.shift();var s=o.match(/^\s*([a-zA-Z:\-\.]+:)(.*)/);if(a===0&&s){o.split(":");i=s[1].slice(0,-1);r[i]=s[2]}else{r[i]+=o}a+=Ft(o)}for(var l in r){Bt(e,l,r[l])}}}function jt(e){Ae(e);for(var t=0;tQ.config.historyCacheSize){i.shift()}while(i.length>0){try{localStorage.setItem("htmx-history-cache",JSON.stringify(i));break}catch(e){fe(re().body,"htmx:historyCacheError",{cause:e,cache:i});i.shift()}}}function Yt(e){if(!U()){return null}e=F(e);var t=E(localStorage.getItem("htmx-history-cache"))||[];for(var r=0;r=200&&this.status<400){ce(re().body,"htmx:historyCacheMissLoad",o);var e=l(this.response);e=e.querySelector("[hx-history-elt],[data-hx-history-elt]")||e;var t=Zt();var r=T(t);var n=Ve(this.response);if(n){var i=C("title");if(i){i.innerHTML=n}else{window.document.title=n}}Ue(t,e,r);nr(r.tasks);Jt=a;ce(re().body,"htmx:historyRestore",{path:a,cacheMiss:true,serverResponse:this.response})}else{fe(re().body,"htmx:historyCacheMissLoadError",o)}};e.send()}function ar(e){er();e=e||location.pathname+location.search;var t=Yt(e);if(t){var r=l(t.content);var n=Zt();var i=T(n);Ue(n,r,i);nr(i.tasks);document.title=t.title;setTimeout(function(){window.scrollTo(0,t.scroll)},0);Jt=e;ce(re().body,"htmx:historyRestore",{path:e,item:t})}else{if(Q.config.refreshOnHistoryMiss){window.location.reload(true)}else{ir(e)}}}function or(e){var t=me(e,"hx-indicator");if(t==null){t=[e]}oe(t,function(e){var t=ae(e);t.requestCount=(t.requestCount||0)+1;e.classList["add"].call(e.classList,Q.config.requestClass)});return t}function sr(e){var t=me(e,"hx-disabled-elt");if(t==null){t=[]}oe(t,function(e){var t=ae(e);t.requestCount=(t.requestCount||0)+1;e.setAttribute("disabled","")});return t}function lr(e,t){oe(e,function(e){var t=ae(e);t.requestCount=(t.requestCount||0)-1;if(t.requestCount===0){e.classList["remove"].call(e.classList,Q.config.requestClass)}});oe(t,function(e){var t=ae(e);t.requestCount=(t.requestCount||0)-1;if(t.requestCount===0){e.removeAttribute("disabled")}})}function ur(e,t){for(var r=0;r=0}function wr(e,t){var r=t?t:ne(e,"hx-swap");var n={swapStyle:ae(e).boosted?"innerHTML":Q.config.defaultSwapStyle,swapDelay:Q.config.defaultSwapDelay,settleDelay:Q.config.defaultSettleDelay};if(Q.config.scrollIntoViewOnBoost&&ae(e).boosted&&!br(e)){n["show"]="top"}if(r){var i=D(r);if(i.length>0){for(var a=0;a0?l.join(":"):null;n["scroll"]=u;n["scrollTarget"]=f}else if(o.indexOf("show:")===0){var c=o.substr(5);var l=c.split(":");var h=l.pop();var f=l.length>0?l.join(":"):null;n["show"]=h;n["showTarget"]=f}else if(o.indexOf("focus-scroll:")===0){var v=o.substr("focus-scroll:".length);n["focusScroll"]=v=="true"}else if(a==0){n["swapStyle"]=o}else{b("Unknown modifier in hx-swap: "+o)}}}}return n}function Sr(e){return ne(e,"hx-encoding")==="multipart/form-data"||h(e,"form")&&ee(e,"enctype")==="multipart/form-data"}function Er(t,r,n){var i=null;R(r,function(e){if(i==null){i=e.encodeParameters(t,n,r)}});if(i!=null){return i}else{if(Sr(r)){return mr(n)}else{return pr(n)}}}function T(e){return{tasks:[],elts:[e]}}function Cr(e,t){var r=e[0];var n=e[e.length-1];if(t.scroll){var i=null;if(t.scrollTarget){i=ue(r,t.scrollTarget)}if(t.scroll==="top"&&(r||i)){i=i||r;i.scrollTop=0}if(t.scroll==="bottom"&&(n||i)){i=i||n;i.scrollTop=i.scrollHeight}}if(t.show){var i=null;if(t.showTarget){var a=t.showTarget;if(t.showTarget==="window"){a="body"}i=ue(r,a)}if(t.show==="top"&&(r||i)){i=i||r;i.scrollIntoView({block:"start",behavior:Q.config.scrollBehavior})}if(t.show==="bottom"&&(n||i)){i=i||n;i.scrollIntoView({block:"end",behavior:Q.config.scrollBehavior})}}}function Rr(e,t,r,n){if(n==null){n={}}if(e==null){return n}var i=te(e,t);if(i){var a=i.trim();var o=r;if(a==="unset"){return null}if(a.indexOf("javascript:")===0){a=a.substr(11);o=true}else if(a.indexOf("js:")===0){a=a.substr(3);o=true}if(a.indexOf("{")!==0){a="{"+a+"}"}var s;if(o){s=Tr(e,function(){return Function("return ("+a+")")()},{})}else{s=E(a)}for(var l in s){if(s.hasOwnProperty(l)){if(n[l]==null){n[l]=s[l]}}}}return Rr(u(e),t,r,n)}function Tr(e,t,r){if(Q.config.allowEval){return t()}else{fe(e,"htmx:evalDisallowedError");return r}}function Or(e,t){return Rr(e,"hx-vars",true,t)}function qr(e,t){return Rr(e,"hx-vals",false,t)}function Hr(e){return le(Or(e),qr(e))}function Lr(t,r,n){if(n!==null){try{t.setRequestHeader(r,n)}catch(e){t.setRequestHeader(r,encodeURIComponent(n));t.setRequestHeader(r+"-URI-AutoEncoded","true")}}}function Ar(t){if(t.responseURL&&typeof URL!=="undefined"){try{var e=new URL(t.responseURL);return e.pathname+e.search}catch(e){fe(re().body,"htmx:badResponseUrl",{url:t.responseURL})}}}function O(e,t){return t.test(e.getAllResponseHeaders())}function Nr(e,t,r){e=e.toLowerCase();if(r){if(r instanceof Element||I(r,"String")){return he(e,t,null,null,{targetOverride:p(r),returnPromise:true})}else{return he(e,t,p(r.source),r.event,{handler:r.handler,headers:r.headers,values:r.values,targetOverride:p(r.target),swapOverride:r.swap,select:r.select,returnPromise:true})}}else{return he(e,t,null,null,{returnPromise:true})}}function Ir(e){var t=[];while(e){t.push(e);e=e.parentElement}return t}function kr(e,t,r){var n;var i;if(typeof URL==="function"){i=new URL(t,document.location.href);var a=document.location.origin;n=a===i.origin}else{i=t;n=g(t,document.location.origin)}if(Q.config.selfRequestsOnly){if(!n){return false}}return ce(e,"htmx:validateUrl",le({url:i,sameHost:n},r))}function he(t,r,n,i,a,e){var o=null;var s=null;a=a!=null?a:{};if(a.returnPromise&&typeof Promise!=="undefined"){var l=new Promise(function(e,t){o=e;s=t})}if(n==null){n=re().body}var M=a.handler||Mr;var X=a.select||null;if(!se(n)){ie(o);return l}var u=a.targetOverride||ye(n);if(u==null||u==pe){fe(n,"htmx:targetError",{target:te(n,"hx-target")});ie(s);return l}var f=ae(n);var c=f.lastButtonClicked;if(c){var h=ee(c,"formaction");if(h!=null){r=h}var v=ee(c,"formmethod");if(v!=null){if(v.toLowerCase()!=="dialog"){t=v}}}var d=ne(n,"hx-confirm");if(e===undefined){var D=function(e){return he(t,r,n,i,a,!!e)};var U={target:u,elt:n,path:r,verb:t,triggeringEvent:i,etc:a,issueRequest:D,question:d};if(ce(n,"htmx:confirm",U)===false){ie(o);return l}}var g=n;var p=ne(n,"hx-sync");var m=null;var x=false;if(p){var F=p.split(":");var B=F[0].trim();if(B==="this"){g=xe(n,"hx-sync")}else{g=ue(n,B)}p=(F[1]||"drop").trim();f=ae(g);if(p==="drop"&&f.xhr&&f.abortable!==true){ie(o);return l}else if(p==="abort"){if(f.xhr){ie(o);return l}else{x=true}}else if(p==="replace"){ce(g,"htmx:abort")}else if(p.indexOf("queue")===0){var V=p.split(" ");m=(V[1]||"last").trim()}}if(f.xhr){if(f.abortable){ce(g,"htmx:abort")}else{if(m==null){if(i){var y=ae(i);if(y&&y.triggerSpec&&y.triggerSpec.queue){m=y.triggerSpec.queue}}if(m==null){m="last"}}if(f.queuedRequests==null){f.queuedRequests=[]}if(m==="first"&&f.queuedRequests.length===0){f.queuedRequests.push(function(){he(t,r,n,i,a)})}else if(m==="all"){f.queuedRequests.push(function(){he(t,r,n,i,a)})}else if(m==="last"){f.queuedRequests=[];f.queuedRequests.push(function(){he(t,r,n,i,a)})}ie(o);return l}}var b=new XMLHttpRequest;f.xhr=b;f.abortable=x;var w=function(){f.xhr=null;f.abortable=false;if(f.queuedRequests!=null&&f.queuedRequests.length>0){var e=f.queuedRequests.shift();e()}};var j=ne(n,"hx-prompt");if(j){var S=prompt(j);if(S===null||!ce(n,"htmx:prompt",{prompt:S,target:u})){ie(o);w();return l}}if(d&&!e){if(!confirm(d)){ie(o);w();return l}}var E=xr(n,u,S);if(t!=="get"&&!Sr(n)){E["Content-Type"]="application/x-www-form-urlencoded"}if(a.headers){E=le(E,a.headers)}var _=dr(n,t);var C=_.errors;var R=_.values;if(a.values){R=le(R,a.values)}var z=Hr(n);var $=le(R,z);var T=yr($,n);if(Q.config.getCacheBusterParam&&t==="get"){T["org.htmx.cache-buster"]=ee(u,"id")||"true"}if(r==null||r===""){r=re().location.href}var O=Rr(n,"hx-request");var W=ae(n).boosted;var q=Q.config.methodsThatUseUrlParams.indexOf(t)>=0;var H={boosted:W,useUrlParams:q,parameters:T,unfilteredParameters:$,headers:E,target:u,verb:t,errors:C,withCredentials:a.credentials||O.credentials||Q.config.withCredentials,timeout:a.timeout||O.timeout||Q.config.timeout,path:r,triggeringEvent:i};if(!ce(n,"htmx:configRequest",H)){ie(o);w();return l}r=H.path;t=H.verb;E=H.headers;T=H.parameters;C=H.errors;q=H.useUrlParams;if(C&&C.length>0){ce(n,"htmx:validation:halted",H);ie(o);w();return l}var G=r.split("#");var J=G[0];var L=G[1];var A=r;if(q){A=J;var Z=Object.keys(T).length!==0;if(Z){if(A.indexOf("?")<0){A+="?"}else{A+="&"}A+=pr(T);if(L){A+="#"+L}}}if(!kr(n,A,H)){fe(n,"htmx:invalidPath",H);ie(s);return l}b.open(t.toUpperCase(),A,true);b.overrideMimeType("text/html");b.withCredentials=H.withCredentials;b.timeout=H.timeout;if(O.noHeaders){}else{for(var N in E){if(E.hasOwnProperty(N)){var K=E[N];Lr(b,N,K)}}}var I={xhr:b,target:u,requestConfig:H,etc:a,boosted:W,select:X,pathInfo:{requestPath:r,finalRequestPath:A,anchor:L}};b.onload=function(){try{var e=Ir(n);I.pathInfo.responsePath=Ar(b);M(n,I);lr(k,P);ce(n,"htmx:afterRequest",I);ce(n,"htmx:afterOnLoad",I);if(!se(n)){var t=null;while(e.length>0&&t==null){var r=e.shift();if(se(r)){t=r}}if(t){ce(t,"htmx:afterRequest",I);ce(t,"htmx:afterOnLoad",I)}}ie(o);w()}catch(e){fe(n,"htmx:onLoadError",le({error:e},I));throw e}};b.onerror=function(){lr(k,P);fe(n,"htmx:afterRequest",I);fe(n,"htmx:sendError",I);ie(s);w()};b.onabort=function(){lr(k,P);fe(n,"htmx:afterRequest",I);fe(n,"htmx:sendAbort",I);ie(s);w()};b.ontimeout=function(){lr(k,P);fe(n,"htmx:afterRequest",I);fe(n,"htmx:timeout",I);ie(s);w()};if(!ce(n,"htmx:beforeRequest",I)){ie(o);w();return l}var k=or(n);var P=sr(n);oe(["loadstart","loadend","progress","abort"],function(t){oe([b,b.upload],function(e){e.addEventListener(t,function(e){ce(n,"htmx:xhr:"+t,{lengthComputable:e.lengthComputable,loaded:e.loaded,total:e.total})})})});ce(n,"htmx:beforeSend",I);var Y=q?null:Er(b,n,T);b.send(Y);return l}function Pr(e,t){var r=t.xhr;var n=null;var i=null;if(O(r,/HX-Push:/i)){n=r.getResponseHeader("HX-Push");i="push"}else if(O(r,/HX-Push-Url:/i)){n=r.getResponseHeader("HX-Push-Url");i="push"}else if(O(r,/HX-Replace-Url:/i)){n=r.getResponseHeader("HX-Replace-Url");i="replace"}if(n){if(n==="false"){return{}}else{return{type:i,path:n}}}var a=t.pathInfo.finalRequestPath;var o=t.pathInfo.responsePath;var s=ne(e,"hx-push-url");var l=ne(e,"hx-replace-url");var u=ae(e).boosted;var f=null;var c=null;if(s){f="push";c=s}else if(l){f="replace";c=l}else if(u){f="push";c=o||a}if(c){if(c==="false"){return{}}if(c==="true"){c=o||a}if(t.pathInfo.anchor&&c.indexOf("#")===-1){c=c+"#"+t.pathInfo.anchor}return{type:f,path:c}}else{return{}}}function Mr(l,u){var f=u.xhr;var c=u.target;var e=u.etc;var t=u.requestConfig;var h=u.select;if(!ce(l,"htmx:beforeOnLoad",u))return;if(O(f,/HX-Trigger:/i)){_e(f,"HX-Trigger",l)}if(O(f,/HX-Location:/i)){er();var r=f.getResponseHeader("HX-Location");var v;if(r.indexOf("{")===0){v=E(r);r=v["path"];delete v["path"]}Nr("GET",r,v).then(function(){tr(r)});return}var n=O(f,/HX-Refresh:/i)&&"true"===f.getResponseHeader("HX-Refresh");if(O(f,/HX-Redirect:/i)){location.href=f.getResponseHeader("HX-Redirect");n&&location.reload();return}if(n){location.reload();return}if(O(f,/HX-Retarget:/i)){if(f.getResponseHeader("HX-Retarget")==="this"){u.target=l}else{u.target=ue(l,f.getResponseHeader("HX-Retarget"))}}var d=Pr(l,u);var i=f.status>=200&&f.status<400&&f.status!==204;var g=f.response;var a=f.status>=400;var p=Q.config.ignoreTitle;var o=le({shouldSwap:i,serverResponse:g,isError:a,ignoreTitle:p},u);if(!ce(c,"htmx:beforeSwap",o))return;c=o.target;g=o.serverResponse;a=o.isError;p=o.ignoreTitle;u.target=c;u.failed=a;u.successful=!a;if(o.shouldSwap){if(f.status===286){at(l)}R(l,function(e){g=e.transformResponse(g,f,l)});if(d.type){er()}var s=e.swapOverride;if(O(f,/HX-Reswap:/i)){s=f.getResponseHeader("HX-Reswap")}var v=wr(l,s);if(v.hasOwnProperty("ignoreTitle")){p=v.ignoreTitle}c.classList.add(Q.config.swappingClass);var m=null;var x=null;var y=function(){try{var e=document.activeElement;var t={};try{t={elt:e,start:e?e.selectionStart:null,end:e?e.selectionEnd:null}}catch(e){}var r;if(h){r=h}if(O(f,/HX-Reselect:/i)){r=f.getResponseHeader("HX-Reselect")}if(d.type){ce(re().body,"htmx:beforeHistoryUpdate",le({history:d},u));if(d.type==="push"){tr(d.path);ce(re().body,"htmx:pushedIntoHistory",{path:d.path})}else{rr(d.path);ce(re().body,"htmx:replacedInHistory",{path:d.path})}}var n=T(c);je(v.swapStyle,c,l,g,n,r);if(t.elt&&!se(t.elt)&&ee(t.elt,"id")){var i=document.getElementById(ee(t.elt,"id"));var a={preventScroll:v.focusScroll!==undefined?!v.focusScroll:!Q.config.defaultFocusScroll};if(i){if(t.start&&i.setSelectionRange){try{i.setSelectionRange(t.start,t.end)}catch(e){}}i.focus(a)}}c.classList.remove(Q.config.swappingClass);oe(n.elts,function(e){if(e.classList){e.classList.add(Q.config.settlingClass)}ce(e,"htmx:afterSwap",u)});if(O(f,/HX-Trigger-After-Swap:/i)){var o=l;if(!se(l)){o=re().body}_e(f,"HX-Trigger-After-Swap",o)}var s=function(){oe(n.tasks,function(e){e.call()});oe(n.elts,function(e){if(e.classList){e.classList.remove(Q.config.settlingClass)}ce(e,"htmx:afterSettle",u)});if(u.pathInfo.anchor){var e=re().getElementById(u.pathInfo.anchor);if(e){e.scrollIntoView({block:"start",behavior:"auto"})}}if(n.title&&!p){var t=C("title");if(t){t.innerHTML=n.title}else{window.document.title=n.title}}Cr(n.elts,v);if(O(f,/HX-Trigger-After-Settle:/i)){var r=l;if(!se(l)){r=re().body}_e(f,"HX-Trigger-After-Settle",r)}ie(m)};if(v.settleDelay>0){setTimeout(s,v.settleDelay)}else{s()}}catch(e){fe(l,"htmx:swapError",u);ie(x);throw e}};var b=Q.config.globalViewTransitions;if(v.hasOwnProperty("transition")){b=v.transition}if(b&&ce(l,"htmx:beforeTransition",u)&&typeof Promise!=="undefined"&&document.startViewTransition){var w=new Promise(function(e,t){m=e;x=t});var S=y;y=function(){document.startViewTransition(function(){S();return w})}}if(v.swapDelay>0){setTimeout(y,v.swapDelay)}else{y()}}if(a){fe(l,"htmx:responseError",le({error:"Response Status Error Code "+f.status+" from "+u.pathInfo.requestPath},u))}}var Xr={};function Dr(){return{init:function(e){return null},onEvent:function(e,t){return true},transformResponse:function(e,t,r){return e},isInlineSwap:function(e){return false},handleSwap:function(e,t,r,n){return false},encodeParameters:function(e,t,r){return null}}}function Ur(e,t){if(t.init){t.init(r)}Xr[e]=le(Dr(),t)}function Fr(e){delete Xr[e]}function Br(e,r,n){if(e==undefined){return r}if(r==undefined){r=[]}if(n==undefined){n=[]}var t=te(e,"hx-ext");if(t){oe(t.split(","),function(e){e=e.replace(/ /g,"");if(e.slice(0,7)=="ignore:"){n.push(e.slice(7));return}if(n.indexOf(e)<0){var t=Xr[e];if(t&&r.indexOf(t)<0){r.push(t)}}})}return Br(u(e),r,n)}var Vr=false;re().addEventListener("DOMContentLoaded",function(){Vr=true});function jr(e){if(Vr||re().readyState==="complete"){e()}else{re().addEventListener("DOMContentLoaded",e)}}function _r(){if(Q.config.includeIndicatorStyles!==false){re().head.insertAdjacentHTML("beforeend","")}}function zr(){var e=re().querySelector('meta[name="htmx-config"]');if(e){return E(e.content)}else{return null}}function $r(){var e=zr();if(e){Q.config=le(Q.config,e)}}jr(function(){$r();_r();var e=re().body;zt(e);var t=re().querySelectorAll("[hx-trigger='restored'],[data-hx-trigger='restored']");e.addEventListener("htmx:abort",function(e){var t=e.target;var r=ae(t);if(r&&r.xhr){r.xhr.abort()}});const r=window.onpopstate?window.onpopstate.bind(window):null;window.onpopstate=function(e){if(e.state&&e.state.htmx){ar();oe(t,function(e){ce(e,"htmx:restored",{document:re(),triggerEvent:ce})})}else{if(r){r(e)}}};setTimeout(function(){ce(e,"htmx:load",{});e=null},0)});return Q}()}); \ No newline at end of file diff --git a/digiplan/templates/base.html b/digiplan/templates/base.html index 1283d4b7..f7da12d3 100644 --- a/digiplan/templates/base.html +++ b/digiplan/templates/base.html @@ -63,6 +63,7 @@ + {% endcompress %} {% compress js %} diff --git a/digiplan/templates/forms/panel_energy.html b/digiplan/templates/forms/panel_energy.html index a55b6b37..588c807c 100644 --- a/digiplan/templates/forms/panel_energy.html +++ b/digiplan/templates/forms/panel_energy.html @@ -1,5 +1,49 @@ {% load i18n %} {% load static %} +{% load partials %} + +{% partialdef key_results %} +
+
+
+ + {{ key_result_area | floatformat:0 }} ha + +
+
+ + {% trans "werden genutzt" %} + +
+
+ {% if key_result_turbines %} +
+
+ + {{ key_result_turbines | floatformat:0 }} + +
+
+ {% trans "Anlagen werden errichtet" %} +
+
+ {% endif %} +
+
+ + {{ key_result_energy | floatformat:1 }} TWh + +
+
+ + {% trans "können gewonnen werden" %} + +
+
+
+{% endpartialdef %} + +

{% trans "Generation" %}

@@ -68,43 +112,10 @@

{% trans "Generation" %}

-
-
-
-
- - ... ha - -
-
- - {% trans "werden genutzt" %} - -
-
-
-
- - ... - -
-
- {% trans "Anlagen werden errichtet" %} -
-
-
-
- - ... - -
-
- - {% trans "können gewonnen werden" %} - -
-
-
+
+ {% with key_result_area=wind_2018_key_result_area key_result_turbines=wind_2018_key_result_turbines key_result_energy=wind_2018_key_result_energy %} + {% partial key_results %} + {% endwith %}
@@ -147,43 +158,10 @@

{% trans "Generation" %}

-
-
-
-
- - ... ha - -
-
- - {% trans "werden genutzt" %} - -
-
-
-
- - ... - -
-
- {% trans "Anlagen werden errichtet" %} -
-
-
-
- - ... TWh - -
-
- - {% trans "können gewonnen werden" %} - -
-
-
+
+ {% with key_result_area=wind_2024_key_result_area key_result_turbines=wind_2024_key_result_turbines key_result_energy=wind_2024_key_result_energy %} + {% partial key_results %} + {% endwith %}
@@ -238,43 +216,10 @@

{% trans "Generation" %}

-
-
-
-
- - ... ha - -
-
- - {% trans "werden genutzt" %} - -
-
-
-
- - ... - -
-
- {% trans "Anlagen werden errichtet" %} -
-
-
-
- - ... TWh - -
-
- - {% trans "können gewonnen werden" %} - -
-
-
+
+ {% with key_result_area=wind_2027_key_result_area key_result_turbines=wind_2027_key_result_turbines key_result_energy=wind_2027_key_result_energy %} + {% partial key_results %} + {% endwith %}
@@ -359,33 +304,10 @@

{% trans "Generation" %}

-
-
-
-
- - ... ha - -
-
- - {% trans "werden genutzt" %} - -
-
-
-
- - ... TWh - -
-
- - {% trans "können gewonnen werden" %} - -
-
-
+
+ {% with key_result_area=pv_ground_key_result_area key_result_energy=pv_ground_key_result_energy %} + {% partial key_results %} + {% endwith %}
@@ -427,33 +349,10 @@

{% trans "Generation" %}

-
-
-
-
- - ... ha - -
-
- - {% trans "werden genutzt" %} - -
-
-
-
- - ... TWh - -
-
- - {% trans "können gewonnen werden" %} - -
-
-
+
+ {% with key_result_area=pv_roof_key_result_area key_result_energy=pv_roof_key_result_energy %} + {% partial key_results %} + {% endwith %}
diff --git a/package-lock.json b/package-lock.json index 9e6b4a91..a63c755b 100644 --- a/package-lock.json +++ b/package-lock.json @@ -9,6 +9,7 @@ "@popperjs/core": "^2.11.8", "bootstrap": "^5.2.1", "echarts": "^5.4.2", + "htmx.org": "^1.9.11", "ion-rangeslider": "^2.3.1", "jquery": "^3.6.1", "maplibre-gl": "^4.0.0", @@ -283,6 +284,11 @@ "node": ">=6" } }, + "node_modules/htmx.org": { + "version": "1.9.11", + "resolved": "https://registry.npmjs.org/htmx.org/-/htmx.org-1.9.11.tgz", + "integrity": "sha512-WlVuICn8dfNOOgYmdYzYG8zSnP3++AdHkMHooQAzGZObWpVXYathpz/I37ycF4zikR6YduzfCvEcxk20JkIUsw==" + }, "node_modules/ieee754": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", diff --git a/package.json b/package.json index 8a0299dc..c11feedd 100644 --- a/package.json +++ b/package.json @@ -3,6 +3,7 @@ "@popperjs/core": "^2.11.8", "bootstrap": "^5.2.1", "echarts": "^5.4.2", + "htmx.org": "^1.9.11", "ion-rangeslider": "^2.3.1", "jquery": "^3.6.1", "maplibre-gl": "^4.0.0", diff --git a/poetry.lock b/poetry.lock index a727c955..6dc2be50 100644 --- a/poetry.lock +++ b/poetry.lock @@ -963,6 +963,20 @@ files = [ [package.dependencies] django = "*" +[[package]] +name = "django-cache-memoize" +version = "0.2.0" +description = "Django utility for a memoization decorator that uses the Django cache framework." +optional = false +python-versions = ">=3.8" +files = [ + {file = "django-cache-memoize-0.2.0.tar.gz", hash = "sha256:79950a027ba40e4aff4efed587b76036bf5ba1f59329d7b158797b832be72ca6"}, + {file = "django_cache_memoize-0.2.0-py3-none-any.whl", hash = "sha256:a6bfd112da699d1fa85955a1e15b7c48ee25e58044398958e269678db10736f3"}, +] + +[package.extras] +dev = ["black", "flake8", "therapist", "tox", "twine"] + [[package]] name = "django-compressor" version = "4.4" @@ -1175,6 +1189,24 @@ files = [ Django = ">=2.2" redis = ">=3.0.0" +[[package]] +name = "django-template-partials" +version = "23.4" +description = "django-template-partials" +optional = false +python-versions = "*" +files = [ + {file = "django-template-partials-23.4.tar.gz", hash = "sha256:f762b0b7b2222462df0845f0556792640b769eb832eae218a0e7dadd4e5606cc"}, + {file = "django_template_partials-23.4-py2.py3-none-any.whl", hash = "sha256:d83d9c2d2836be769919e9aaf394d5feb1ac86e1187083030398308070122fca"}, +] + +[package.dependencies] +Django = "*" + +[package.extras] +docs = ["Sphinx"] +tests = ["coverage", "django_coverage_plugin"] + [[package]] name = "djangorestframework" version = "3.15.1" @@ -3788,4 +3820,4 @@ testing = ["big-O", "jaraco.functools", "jaraco.itertools", "more-itertools", "p [metadata] lock-version = "2.0" python-versions = ">=3.9,<3.12" -content-hash = "79ed8b7549efce9000f41b17c4a28f561f047d842f8960642727018b151b834d" +content-hash = "c598d0d2ed1130eb6716ea03611be35d3c849a30041ea0c283b4fa25cbb06af8" diff --git a/pyproject.toml b/pyproject.toml index f39ad363..f727786c 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -40,6 +40,8 @@ django-oemof = {git = "https://github.com/rl-institut/django-oemof.git", tag="v0 django-mapengine = "1.3.0" geojson = "^3.0.1" oemof-network = "0.5.0a1" +django-template-partials = "^23.4" +django-cache-memoize = "^0.2.0" [tool.poetry.group.dev.dependencies] Werkzeug = "^2.0.1" # https://github.com/pallets/werkzeug