diff --git a/dist/gridviz.js b/dist/gridviz.js index fe77043cb..61f94ab70 100644 --- a/dist/gridviz.js +++ b/dist/gridviz.js @@ -8586,35 +8586,43 @@ class Map { * @memberof App */ defineResizeObserver() { - // listen to resize events + // Track whether the observer is currently processing a resize event + let resizePending = false + const resizeObserver = new ResizeObserver((entries) => { + if (!Array.isArray(entries) || !entries.length) return + let container = this.container - // make sure canvas has been built + + // Ensure the container has valid dimensions if (container.clientWidth > 0 && container.clientHeight > 0) { - // make sure we dont exceed loop limit first - // see: https://stackoverflow.com/questions/49384120/resizeobserver-loop-limit-exceeded - window.requestAnimationFrame(() => { - if (!Array.isArray(entries) || !entries.length) { - return - } - // update the map and canvas size - if (this.h !== container.clientHeight || this.w !== container.clientWidth) { - this.h = container.clientHeight - this.w = container.clientWidth - this.geoCanvas.h = container.clientHeight - this.geoCanvas.w = container.clientWidth - this.geoCanvas.canvas.setAttribute('width', '' + this.w) - this.geoCanvas.canvas.setAttribute('height', '' + this.h) - // offscreen canvas - this.geoCanvas.offscreenCanvas.setAttribute('width', '' + this.w) - this.geoCanvas.offscreenCanvas.setAttribute('height', '' + this.h) - this.redraw() - - //update button positions - // if (this.zoomButtons) this.zoomButtons.node.style.left = this.w - 50 + 'px' - // if (this.fullscreenButton) this.fullscreenButton.node.style.left = this.w - 50 + 'px' - } - }) + if (!resizePending) { + resizePending = true // Prevent overlapping resize triggers + + window.requestAnimationFrame(() => { + resizePending = false // Reset the flag after processing + + // Check for size changes + if (this.h !== container.clientHeight || this.w !== container.clientWidth) { + this.h = container.clientHeight + this.w = container.clientWidth + + // Update geoCanvas sizes + this.geoCanvas.h = this.h + this.geoCanvas.w = this.w + this.geoCanvas.canvas.setAttribute('width', String(this.w)) + this.geoCanvas.canvas.setAttribute('height', String(this.h)) + this.geoCanvas.offscreenCanvas.setAttribute('width', String(this.w)) + this.geoCanvas.offscreenCanvas.setAttribute('height', String(this.h)) + + this.redraw() + + // Optionally reposition UI elements + // if (this.zoomButtons) this.zoomButtons.node.style.left = this.w - 50 + 'px'; + // if (this.fullscreenButton) this.fullscreenButton.node.style.left = this.w - 50 + 'px'; + } + }) + } } }) @@ -8899,8 +8907,6 @@ class Tooltip { constructor(opts) { opts = opts || {} - /** @type {string} */ - this.div = opts.div || 'tooltip_eurostat' /** @type {string} */ this.maxWidth = opts.maxWidth || '20em' /** @type {string} */ @@ -8930,21 +8936,15 @@ class Tooltip { this.xMouseOffset = opts.xMouseOffset || 0 /** @type {HTMLElement} */ this.parentElement = opts.parentElement || document.body + /** @type {HTMLElement} */ + this.tooltipElement = opts.tooltipElement || null /** * @public * @type {import("d3-selection").Selection} */ - this.tooltip = (0,d3_selection__WEBPACK_IMPORTED_MODULE_0__["default"])('#' + this.div) - - if (this.tooltip.empty()) { - //create tooltip DOM node - // this.tooltip = select( - // '#' + this.parentElement.id && this.parentElement.id != '' - // ? '#' + this.parentElement.id - // : 'body' - // ) - this.tooltip = (0,d3_selection__WEBPACK_IMPORTED_MODULE_0__["default"])('body').append('div').attr('id', this.div) - } + this.tooltip = opts.tooltipElement + ? (0,d3_selection__WEBPACK_IMPORTED_MODULE_0__["default"])(opts.tooltipElement) // Wrap the provided HTML node in a D3 selection + : (0,d3_selection__WEBPACK_IMPORTED_MODULE_0__["default"])(this.parentElement).append('div').attr('id', 'gridviz-tooltip').attr('class', 'gridviz-tooltip') // create default element //initialise this.tooltip.style('max-width', this.maxWidth) @@ -8961,6 +8961,10 @@ class Tooltip { this.tooltip.style('opacity', '0') this.tooltip.style('text-wrap', 'nowrap') + // these placeholders are needed to prevent an infinite DOM resizeObserver loop: + this.tooltip.style('left', '0') + this.tooltip.style('top', '0') + // aria-labels (thanks to wahlatlas) this.tooltip.attr('role', 'tooltip').attr('aria-live', 'polite') } @@ -8990,14 +8994,51 @@ class Tooltip { * @param {MouseEvent} event */ setPosition(event) { + // Get the bounding rect of the parent container (map2) let parentRect = this.parentElement.getBoundingClientRect() - let x = event.pageX + this.xOffset - let y = event.pageY - this.yOffset + // Get the mouse position (relative to the parent container) + let x = event.clientX - parentRect.left + this.xOffset // Relative to parent + let y = event.clientY - parentRect.top - this.yOffset // Relative to parent + // Now, apply the position to the tooltip this.tooltip.style('left', x + 'px').style('top', y + 'px') - this.ensureTooltipInsideContainer(event, parentRect) + // Ensure the tooltip stays inside the parent container + this.ensureTooltipInsideContainer(event, parentRect, this.tooltip.node()) + } + /** + * @function ensureTooltipInsideContainer + * @description Prevents the tooltip from overflowing out of the App container (ensures that the tooltip is inside the gridviz container) + * @param {MouseEvent} event + * @param {DOMRect} parentRect + * @param {HTMLElement} tooltipNode + */ + ensureTooltipInsideContainer(event, parentRect, tooltipNode) { + let node = tooltipNode + let parentWidth = parentRect.width + let parentHeight = parentRect.height + + // Ensure tooltip doesn't go beyond the right edge + if (node.offsetLeft + node.clientWidth > parentWidth) { + let left = event.clientX - node.clientWidth - this.xOffset + node.style.left = left + 'px' + } + + // Ensure tooltip doesn't go beyond the bottom edge + if (node.offsetTop + node.clientHeight > parentHeight) { + node.style.top = parentHeight - node.clientHeight + 'px' + } + + // Ensure tooltip doesn't go above the top edge + if (node.offsetTop < 0) { + node.style.top = 0 + 'px' + } + + // Ensure tooltip doesn't go beyond the left edge + if (node.offsetLeft < 0) { + node.style.left = 0 + 'px' + } } /* @@ -9028,41 +9069,6 @@ class Tooltip { this.tooltip.attr(k, v) return this } - - /** - * @function ensureTooltipInsideContainer - * @description Prevents the tooltip from overflowing out of the App container (ensures that the tooltip is inside the gridviz container) - * @param {MouseEvent} event - * @param {DOMRect} parentRect - */ - ensureTooltipInsideContainer = function (event, parentRect) { - let node = this.tooltip.node() - let parentWidth = parentRect.width - let parentHeight = parentRect.height - - //too far right - if (node.offsetLeft > parentRect.left + parentWidth - node.clientWidth) { - let left = event.x - node.clientWidth - this.xOffset - node.style.left = left + 'px' - // check if mouse covers tooltip - if (node.offsetLeft + node.clientWidth > event.x) { - //move tooltip left so it doesnt cover mouse - let left2 = event.x - node.clientWidth - this.xOffset - node.style.left = left2 + 'px' - } - // node.style.top = node.offsetTop + config.yOffset + "px"; - } - - //too far down - if (node.offsetTop + node.clientHeight > parentRect.top + parentHeight) { - node.style.top = node.offsetTop - node.clientHeight + 'px' - } - - //too far up - if (node.offsetTop < parentRect.top) { - node.style.top = parentRect.top + this.yOffset + 'px' - } - } } @@ -16176,4 +16182,4 @@ const getParameterByName = _core_GeoCanvas_js__WEBPACK_IMPORTED_MODULE_1__.GeoCa /******/ })() ; }); -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"gridviz.js","mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD,O;;;;;;;;;;;;;;ACVe;AACf;AACA;;;;;;;;;;;;;;;;;;;;;ACFuC;AACF;AACJ;;AAEjC,wBAAwB,wDAAQ,CAAC,qDAAS;AACnC;AACA;AACA,qBAAqB,wDAAQ,CAAC,kDAAM;AAC3C,iEAAe,WAAW,EAAC;;;;;;;;;;;;;;;;;ACRY;AACE;;AAE1B;AACf;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,qDAAS;AACxB,yBAAyB,yDAAS;AAClC;AACA,IAAI;AACJ,qBAAqB,qDAAS,UAAU,sDAAU;AAClD;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,UAAU;AACV;;AAEA;AACA;AACA;;;;;;;;;;;;;;;ACvDe;AACf;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACNe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC5BuC;;AAExB,oCAAoC,qDAAS;AAC5D;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,yDAAS;AACrB,YAAY,yDAAS;AACrB;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;AC5Be;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACnBe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACrBe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACnBe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACrBe;AACf;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACnBe;AACf;AACA;;;;;;;;;;;;;;;;;;;;;;;;;ACF2B;AACU;AACV;AACU;AACM;AACC;AACD;AACN;;AAEtB;AACf,6BAA6B,mDAAO;AACpC;AACA,8BAA8B,mDAAG;AACjC,qBAAqB,mDAAG;AACxB;AACA;AACA;AACA,eAAe,mDAAG,CAAC,2DAAW;AAC9B,eAAe,mDAAG;AAClB;AACA;;AAEO,6CAA6C,kDAAM;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO,4CAA4C,kDAAM;AACzD;AACA,gDAAgD,sDAAM;AACtD,qBAAqB,wDAAQ;AAC7B,qBAAqB,wDAAQ;AAC7B;AACA;AACA;AACA;AACA,EAAE,2DAAW,2BAA2B,0DAAgB;AACxD,MAAM,wDAAQ;AACd;AACA;;;;;;;;;;;;;;;;AC9C2D;;AAE3D;AACA;AACe;AACf;AACA;AACA;;AAEA;;AAEA,oCAAoC,sDAAgB,GAAG,wDAAc;;AAErE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;ACpDuC;AACJ;;AAEpB;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP,MAAM;AACN;AACA;AACA;AACA,WAAW,uDAAO;AAClB;AACA;AACA;;AAEO,kCAAkC,qDAAS;AAClD,kBAAkB,qDAAS;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;;;;;;;;;;;;;;;;;;;;;;;;ACtC2C;;AAEpC;;AAEA;AACA;;AAEP;AACA;AACA;AACA,yBAAyB,IAAI;AAC7B,wCAAwC,IAAI,GAAG,IAAI,GAAG,IAAI;AAC1D,wCAAwC,IAAI,GAAG,IAAI,GAAG,IAAI;AAC1D,0CAA0C,IAAI,GAAG,IAAI,GAAG,IAAI,GAAG,IAAI;AACnE,0CAA0C,IAAI,GAAG,IAAI,GAAG,IAAI,GAAG,IAAI;AACnE,wCAAwC,IAAI,GAAG,IAAI,GAAG,IAAI;AAC1D,0CAA0C,IAAI,GAAG,IAAI,GAAG,IAAI,GAAG,IAAI;;AAEnE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,sDAAM;AACN;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;;AAED;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;;AAEA,sDAAM,WAAW,kDAAM;AACvB;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA,CAAC;;AAED;AACA,aAAa,YAAY,EAAE,YAAY,EAAE,YAAY;AACrD;;AAEA;AACA,aAAa,YAAY,EAAE,YAAY,EAAE,YAAY,EAAE,oDAAoD;AAC3G;;AAEA;AACA;AACA,YAAY,2BAA2B,EAAE,eAAe,IAAI,eAAe,IAAI,eAAe,EAAE,qBAAqB,EAAE,GAAG;AAC1H;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;AAEO;AACP;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,sDAAM,WAAW,kDAAM;AACvB;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,cAAc,2BAA2B,EAAE,eAAe,IAAI,qBAAqB,KAAK,qBAAqB,GAAG,qBAAqB,EAAE,GAAG;AAC1I;AACA,CAAC;;AAED;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC3YA,6BAAe,oCAAS;AACxB;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;ACT2C;AACO;AACP;;AAE3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,qBAAqB,0CAAG,OAAO,qDAAU;AACzC;AACA;AACA;AACA;AACA,qCAAqC;AACrC;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;;AAEe;AACf;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;;AAEA,sDAAM,WAAW,kDAAM,CAAC,4CAAK;AAC7B;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,0CAAG;AAClB;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;;AAED;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,iCAAiC,6CAAO;AACxC;AACA;;AAEO;AACP;AACA;;AAEO;AACP;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,gBAAgB,6CAAO;AACvB;AACA;;AAEA,sDAAM,WAAW,kDAAM,CAAC,4CAAK;AAC7B;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,CAAC;;;;;;;;;;;;;;;;AC1HM;AACA;;;;;;;;;;;;;;;ACDP,YAAY;;AAEZ;AACA,8CAA8C,KAAK,OAAO;AAC1D;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,YAAY;AACZ,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,GAAG;AACH;AACA,iBAAiB;AACjB;AACA;AACA,GAAG;AACH;AACA,mFAAmF,OAAO;AAC1F;AACA,gDAAgD,OAAO;AACvD,GAAG;AACH;AACA;AACA,oDAAoD,OAAO;AAC3D;AACA;;AAEA;AACA,sCAAsC,OAAO;AAC7C;AACA;AACA;AACA;AACA;;AAEA;AACA,mCAAmC,OAAO;AAC1C;AACA;AACA;AACA;AACA;AACA,mCAAmC,4BAA4B;AAC/D;AACA;;AAEA,iEAAe,QAAQ,EAAC;;;;;;;;;;;;;;;;;;ACnFY;AACoB;;AAExD,6BAAe,oCAAS;AACxB;AACA,kBAAkB,wDAAM,4BAA4B,mDAAO,EAAE,0DAAiB;AAC9E;AACA,qCAAqC,mDAAO,EAAE,0DAAiB;AAC/D,IAAI;AACJ;AACA;AACA;AACA;;AAEO;AACP;AACA,kBAAkB,wDAAM;AACxB;AACA,+BAA+B,mDAAO,EAAE,0DAAiB;AACzD,4BAA4B,mCAAmC;AAC/D;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;AC3BA;AACA;AACO,oBAAoB;AACpB,2BAA2B;;AAE3B;AACP;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;ACZ2B;;AAE3B,UAAU,mDAAG;;AAEN;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACVP,YAAY;AACZ,YAAY;AACZ;AACA;AACA;;AAEA;AACA,oCAAoC;AACpC;AACA,GAAG,gBAAgB;AACnB;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;;AAEA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;;AAErB;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,iCAAiC,YAAY;AAC7C;AACA;;AAEA;AACA;AACA;AACA,iCAAiC,YAAY;AAC7C;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA,OAAO;AACP,KAAK;AACL;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;ACnK2B;;AAE3B,UAAU,mDAAG;;AAEN;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACVA;AACP;AACA;;AAEO;AACP;AACA;;AAEO;AACP;AACA;;;;;;;;;;;;;;;;;;;;;ACVqD;AACxB;;AAE7B;AACA;AACA;AACA,WAAW,oDAAI;AACf;AACA,KAAK;AACL;AACA;;AAEe;AACf;AACA,eAAe,kDAAS;AACxB,SAAS,oDAAI;AACb;AACA,GAAG;AACH;;AAEO,mBAAmB,4CAAQ;AAC3B,mBAAmB,4CAAQ;;;;;;;;;;;;;;;ACrBlC;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;;;;;;;;;;;;;;;ACRA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;;;;;;;;;;;;;;;;;;ACPuC;;AAEvC;AACO;AACA;;AAEP;AACA;AACA;AACA;AACA,CAAC;;AAEc;AACf,WAAW,sDAAY;AACvB;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACjBsD;;AAEtD,6BAAe,oCAAS;AACxB,aAAa,qEAAkB;AAC/B;;;;;;;;;;;;;;;;ACJA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;AAEA;AACA;AACA;AACO;AACP,gGAAgG;AAChG;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACnBA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;;;;;;;;;ACjBA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA,KAAK;AACL;AACA;;;;;;;;;;;;;;;;;ACNsD;;AAE/C;;AAEP,6BAAe,oCAAS;AACxB,UAAU,qEAAkB;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4CAA4C,qEAAkB,gCAAgC;AAC9F;;;;;;;;;;;;;;;;ACfsD;;AAEtD,6BAAe,oCAAS;AACxB,UAAU,qEAAkB;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACVA;AACA;;AAEe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA,uDAAuD;;AAEhD;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;AC9CA;AACA,6BAAe,oCAAS;AACxB,kDAAkD,OAAO;AACzD;AACA,6BAA6B;AAC7B,sCAAsC,QAAQ;AAC9C,sCAAsC,oBAAoB;AAC1D;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACV+C;AACM;AACN;;AAE/C,iEAAe;AACf;AACA;AACA;AACA,OAAO,yDAAa;AACpB;AACA;AACA;AACA;AACA,iBAAiB,6DAAa;AAC9B,OAAO,yDAAa;AACpB,OAAO,4DAAgB;AACvB;AACA;AACA,CAAC,EAAC;;;;;;;;;;;;;;;AClBF,6BAAe,oCAAS;AACxB;AACA;;;;;;;;;;;;;;;;;;;;;;;ACFqC;AACM;AACM;AACE;AACV;AACE;AACU;AAChB;;AAErC;AACA;;AAEA,6BAAe,oCAAS;AACxB,gFAAgF,oDAAQ,GAAG,2DAAW;AACtG;AACA;AACA;AACA,iDAAiD,oDAAQ,GAAG,8DAAc;AAC1E;AACA;AACA;;AAEA;AACA,gBAAgB,+DAAe;;AAE/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,cAAc,uDAAW;;AAEzB;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,qBAAqB,uDAAW;AAChC;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,QAAQ;AACR;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA,0BAA0B,0DAAU;;AAEpC;AACA;;AAEA;AACA;AACA,mDAAmD,gEAAc;;AAEjE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA,uEAAuE;AACvE,uEAAuE;AACvE,sIAAsI;AACtI,sEAAsE;AACtE;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,mCAAmC,+DAAe;AAClD,gDAAgD,wDAAQ;AACxD;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACnJO;AACP;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AClBiC;;AAEjC,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,gDAAK;AAChB;AACA;;;;;;;;;;;;;;;;;;ACZqC;;AAErC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA,kFAAkF,wDAAQ;AAC1F;;AAEO;AACP;AACA,0CAA0C,wDAAQ;AAClD;AACA;;AAEe;AACf;AACA,4BAA4B,wDAAQ;AACpC;;;;;;;;;;;;;;;AC5BA,iEAAe,YAAY,EAAC;;;;;;;;;;;;;;;;;ACAa;AACV;;AAEhB;AACf,UAAU,qDAAK,UAAU,oDAAQ,mBAAmB,oDAAQ;AAC5D,UAAU,qDAAK;AACf,UAAU,qDAAK;AACf,gBAAgB,qDAAK;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACfA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;ACJyC;AACV;AACY;AACD;;AAE1C,iEAAe;AACf,cAAc,gDAAK;;AAEnB;AACA,2BAA2B,6CAAQ,mBAAmB,6CAAQ;AAC9D;AACA;AACA,kBAAkB,qDAAO;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA,CAAC,IAAI,EAAC;;AAEN;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,OAAO;AACvB,cAAc,6CAAQ;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO,yBAAyB,iDAAK;AAC9B,+BAA+B,uDAAW;;;;;;;;;;;;;;;;ACtDhB;;AAEjC;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;;AAEd;AACA;;AAEA;AACA;AACA;AACA,gCAAgC;AAChC;AACA,4BAA4B;AAC5B;AACA;AACA,yCAAyC;AACzC,4BAA4B;AAC5B;AACA,MAAM,OAAO;AACb;AACA,cAAc,SAAS,sDAAM,SAAS;AACtC;AACA;AACA;;AAEA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,OAAO;AACpC;AACA,SAAS;AACT;;;;;;;;;;;;;;;;AC/DA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACzBkC;AACY;;AAE9C;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,cAAc,aAAa,sDAAM,SAAS,GAAG,aAAa,sDAAM,SAAS;AACzE,MAAM;AACN;AACA;AACA;;AAEA;AACA;AACA,iCAAiC,gCAAgC;AACjE,cAAc,sDAAsD,sDAAM,OAAO;AACjF,MAAM;AACN;AACA;AACA;;AAEA;AACA;AACA,cAAc,qDAAqD,sDAAM,OAAO;AAChF,MAAM;AACN;AACA;AACA;;AAEA;AACA;AACA;AACA,cAAc,aAAa,sDAAM,SAAS,GAAG,aAAa,sDAAM,SAAS;AACzE,MAAM;AACN;AACA;AACA;;AAEA;AACA;AACA,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO,mDAAmD,+CAAQ;AAC3D,mDAAmD,+CAAQ;;;;;;;;;;;;;;;;;AC9Df;;AAEnD;;AAEA;AACO;AACP;AACA,wBAAwB,mDAAQ,GAAG,yDAAS;AAC5C;;AAEO;AACP,4BAA4B,mDAAQ;AACpC;AACA;AACA,iEAAiE,mDAAQ;AACzE;AACA,SAAS,yDAAS;AAClB;;;;;;;;;;;;;;;ACjBA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,iEAAe;;AAEf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA,CAAC,mBAAmB,EAAC;;;;;;;;;;;;;;;ACtErB,iEAAe,WAAW,EAAC;;;;;;;;;;;;;;;;ACAoB;;AAE/C,iEAAe;AACf;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,QAAQ;;AAER;AACA;AACA;;AAEA;;AAEA;AACA,CAAC,EAAE,yDAAa,CAAC,EAAC;;;;;;;;;;;;;;;;AC3BX;AACP;AACA;AACA,gCAAgC;AAChC,+CAA+C;AAC/C;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;ACzBwE;AACpC;;AAErB;AACf;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,wCAAwC,wDAAS;AACjD;AACA;;AAEA;AACA,wDAAwD,oDAAM;AAC9D;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,gBAAgB,gDAAS;AACzB;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,SAAS,+CAAS;AAClB;;;;;;;;;;;;;;;ACxDA;AACA;AACA;AACA,cAAc;AACd;AACA;AACe;AACf;AACA;;;;;;;;;;;;;;;ACRA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACJuC;AACD;;AAEtC;AACA;AACA;AACA;AACA,mBAAmB,iDAAK,8CAA8C,iDAAK;AAC3E;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB,iBAAiB,yDAAS;AAC1B;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACxBA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACVyC;;AAEzC,6BAAe,oCAAS;AACxB;AACA;AACA,SAAS,sDAAU,2BAA2B,OAAO,sDAAU,uBAAuB,QAAQ;AAC9F;;;;;;;;;;;;;;;;ACNO;;AAEP,iEAAe;AACf;AACA;AACA;AACA;AACA;AACA,CAAC,EAAC;;;;;;;;;;;;;;;;ACRyC;;AAE3C,6BAAe,oCAAS;AACxB,UAAU,2DAAW;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACnBqD;;AAErD,6BAAe,oCAAS;AACxB;AACA,YAAY,0DAAS;AACrB,YAAY,0DAAS,eAAe,qDAAI;AACxC;;;;;;;;;;;;;;;;ACNoC;;AAEpC,6BAAe,oCAAS;AACxB,mDAAmD,uDAAO;AAC1D;AACA;AACA,GAAG;AACH;;;;;;;;;;;;;;;;ACPwC;;AAExC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB,iBAAiB,yDAAS;;AAE1B;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACxDA,6BAAe,sCAAW;AAC1B;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACLA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;AC1EA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;;;;;;;;;;;;;;;;;;ACZqC;AACA;AACC;;AAEtC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,SAAS,gBAAgB;AACzB;AACA;AACA;AACA,MAAM;AACN,qBAAqB,gDAAS;AAC9B;AACA;;AAEA;AACA,SAAS,iBAAiB;AAC1B;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,cAAc,iBAAiB;AAC/B;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,cAAc,gBAAgB;AAC9B;AACA;AACA;AACA;AACA;AACA,MAAM;AACN,qBAAqB,gDAAS;AAC9B;AACA;;AAEA;AACA,cAAc,iBAAiB;AAC/B;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA;;AAEA,2CAA2C,wDAAQ;;AAEnD,uGAAuG,OAAO;AAC9G;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA,6CAA6C,iBAAiB;AAC9D;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,eAAe,gDAAS;AACxB;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA,wBAAwB;AACxB;;;;;;;;;;;;;;;AC/HA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACJuC;;AAEvC;AACA,eAAe,sDAAW;AAC1B;;AAEA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;ACjCA,6BAAe,oCAAS;;AAExB,4DAA4D,OAAO;AACnE,+DAA+D,OAAO;AACtE;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;ACTA,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;;;;ACFiC;AACI;;AAErC,6BAAe,sCAAW;AAC1B,aAAa,gDAAS,iCAAiC,kDAAM;AAC7D;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,iCAAiC,sDAAsD;AACvF,wCAAwC,gDAAgD;AACxF,sCAAsC,8CAA8C;AACpF,yCAAyC;AACzC;;;;;;;;;;;;;;;;;ACrBiC;AACI;;AAErC,6BAAe,sCAAW;AAC1B,aAAa,gDAAS,gCAAgC,kDAAM;AAC5D;;;;;;;;;;;;;;;;;ACLqC;AACD;;AAEpC,6BAAe,oCAAS;AACxB,2CAA2C,uDAAO;;AAElD,sFAAsF,OAAO;AAC7F,6FAA6F,OAAO;AACpG;AACA;AACA;AACA;AACA;;AAEA,aAAa,gDAAS;AACtB;;;;;;;;;;;;;;;ACfA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACxB2C;AACM;AACI;AACM;AAChB;AACJ;AACE;AACF;AACA;AACE;AACA;AACF;AACA;AACE;AACF;AACA;AACE;AACF;AACA;AACE;AACM;AACF;AACN;AACA;AACE;AACA;AACE;AACA;AACA;AACF;AACA;AACN;AACY;AACA;;AAExC;;AAEA;AACP;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,UAAU,kDAAgB;AAC1B,aAAa,qDAAmB;AAChC,eAAe,uDAAqB;AACpC,kBAAkB,0DAAwB;AAC1C,UAAU,kDAAgB;AAC1B,QAAQ,gDAAc;AACtB,SAAS,iDAAe;AACxB,QAAQ,gDAAc;AACtB,QAAQ,gDAAc;AACtB,SAAS,iDAAe;AACxB;AACA,SAAS,kDAAe;AACxB,QAAQ,iDAAc;AACtB,QAAQ,iDAAc;AACtB,SAAS,kDAAe;AACxB,QAAQ,iDAAc;AACtB,QAAQ,iDAAc;AACtB,SAAS,kDAAe;AACxB,QAAQ,iDAAc;AACtB,QAAQ,iDAAc;AACtB,SAAS,kDAAe;AACxB,YAAY,qDAAkB;AAC9B,WAAW,oDAAiB;AAC5B,QAAQ,iDAAc;AACtB,QAAQ,iDAAc;AACtB,SAAS,kDAAe;AACxB,SAAS,kDAAe;AACxB,UAAU,mDAAgB;AAC1B,UAAU,mDAAgB;AAC1B,UAAU,mDAAgB;AAC1B,SAAS,kDAAe;AACxB,SAAS,kDAAe;AACxB,MAAM,+CAAY;AAClB,YAAY,qDAAkB;AAC9B,qBAAqB,qDAAkB;AACvC;;AAEA,iEAAe,SAAS,EAAC;;;;;;;;;;;;;;;;;ACzFW;AACE;;AAEtC;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB,mDAAmD,uDAAO;AAC1D,uFAAuF,wDAAQ;AAC/F;AACA;AACA,GAAG;AACH;;;;;;;;;;;;;;;ACbA,6BAAe,uCAAY;AAC3B,4DAA4D,OAAO;AACnE,+DAA+D,OAAO;AACtE;AACA;AACA;AACA;;;;;;;;;;;;;;;ACNA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC;AACrC;AACA;;;;;;;;;;;;;;;ACdA;AACA;AACA;;AAEA,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;;ACNqC;;AAErC,6BAAe,oCAAS;AACxB;;AAEA,+JAA+J,OAAO;AACtK,yHAAyH,OAAO;AAChI;AACA;AACA;AACA;AACA;;AAEA,SAAS,QAAQ;AACjB;AACA;;AAEA,aAAa,gDAAS;AACtB;;;;;;;;;;;;;;;AClBA,6BAAe,sCAAW;;AAE1B,4DAA4D,OAAO;AACnE,yDAAyD,OAAO;AAChE;AACA;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;ACVA,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;ACFA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,YAAY;AACZ,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA,8CAA8C,OAAO;AACrD;AACA;AACA,QAAQ;AACR;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,2CAA2C,OAAO;AAClD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA,8CAA8C,OAAO;AACrD,6BAA6B,OAAO;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,cAAc,OAAO;AACrB;AACA;;;;;;;;;;;;;;;AClEA,6BAAe,sCAAW;;AAE1B,6DAA6D,QAAQ;AACrE,6EAA6E,SAAS;AACtF;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;ACZA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;AC3BA;AACA;AACA;;AAEA,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;ACNA;AACA;AACA;AACA;;AAEA,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;;;ACPqC;AACC;;AAEtC,6BAAe,oCAAS;AACxB,6CAA6C,wDAAQ;;AAErD,sFAAsF,OAAO;AAC7F,gHAAgH,OAAO;AACvH;AACA;AACA;AACA;AACA;AACA;;AAEA,aAAa,gDAAS;AACtB;;;;;;;;;;;;;;;;;;AChBqC;AACL;AACY;;AAE5C;AACA;AACA,WAAW,qDAAK;AAChB;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA,gBAAgB,2DAAW;;AAE3B,0FAA0F,OAAO;AACjG,+DAA+D,OAAO;AACtE;AACA;AACA;AACA;AACA;AACA;;AAEA,aAAa,gDAAS;AACtB;;;;;;;;;;;;;;;;ACxB2C;;AAE3C;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA,wDAAwD,yDAAY;AACpE;;;;;;;;;;;;;;;;ACjB2C;;AAE3C;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA,6DAA6D,yDAAY;AACzE;;;;;;;;;;;;;;;ACjBA,6BAAe,sCAAW;AAC1B;AACA,mCAAmC;AACnC;AACA;;;;;;;;;;;;;;;;ACJqC;;AAErC,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA;;AAEA,uFAAuF,OAAO;AAC9F,yGAAyG,OAAO;AAChH;AACA;AACA;AACA;AACA;AACA;;AAEA,aAAa,gDAAS;AACtB;;AAEA;AACA;AACA;;;;;;;;;;;;;;;ACvBA,6BAAe,oCAAS;AACxB;AACA;;;;;;;;;;;;;;;;;ACFuC;;AAEvC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA,SAAS,sDAAW;AACpB;;;;;;;;;;;;;;;AClCA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACxBA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;ACNA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;ACRA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;ACJA,6BAAe,oCAAS;AACxB;AACA;AACA,2BAA2B;AAC3B;;;;;;;;;;;;;;;;ACJiC;;AAEjC,6BAAe,oCAAS;AACxB,cAAc,4CAAK;AACnB;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;;;;;;;;;;;;;;;;;ACVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sIAAsI;;AAE/H;AACP;AACA;;AAEA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;;AAEO;AACP,SAAS;AACT,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,qBAAqB;AACrB;AACA,+BAA+B;AAC/B;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;;;;;;;;;;;;;;;;AC7GiD;AACE;;AAEnD;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;;AAEA;AACA;AACA;AACA,4CAA4C,8DAAS;AACrD,mBAAmB,4DAAU;AAC7B;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;;;;;;;ACpB8B;AAC8B;AACd;AACM;;;;;;;;;;;;;;;;ACHa;;AAEjE,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA,mDAAmD,eAAe;AAClE,8BAA8B,6DAAQ,qBAAqB,2DAAM;AACjE,qBAAqB,0DAAK;AAC1B;AACA;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;ACvBuC;AACU;AACE;;AAEnD,oDAAS,uBAAuB,qDAAmB;AACnD,oDAAS,wBAAwB,sDAAoB;;;;;;;;;;;;;;;;ACLb;;AAExC,6BAAe,oCAAS;AACxB;AACA,IAAI,yDAAS;AACb,GAAG;AACH;;;;;;;;;;;;;;;;;;;ACNyD;AACR;AACV;AACV;;AAE7B;AACA;AACA;AACA;AACA,QAAQ,+CAAc;AACtB;;AAEA;AACA;AACA;AACA;AACA,oCAAoC,IAAI;AACxC;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;;AAEA,sBAAsB,4DAAU;AAChC;AACA,IAAI;AACJ,SAAS,2DAAK,oCAAoC,6CAAG;AACrD;;AAEA,4DAA4D,OAAO;AACnE,+DAA+D,OAAO;AACtE;AACA,QAAQ,mEAAQ;AAChB;AACA;AACA;;AAEA,aAAa,4DAAU;AACvB;;;;;;;;;;;;;;;;;;;ACzC+E;AACxC;AACD;AACK;;AAE3C;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB,iBAAiB,wDAAS,uCAAuC,mEAAoB,GAAG,uDAAW;AACnG;AACA,sEAAsE,qDAAU;AAChF;AACA;AACA;;;;;;;;;;;;;;;;AC7EuC;;AAEvC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA,iBAAiB,wDAAS;AAC1B;AACA;;;;;;;;;;;;;;;;AC3CwC;;AAExC;AACA;AACA,IAAI,kDAAI;AACR;AACA;;AAEA;AACA;AACA,IAAI,kDAAI;AACR;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA;AACA;AACA,QAAQ,iDAAG;AACX;;;;;;;;;;;;;;;;ACtBuC;;AAEvC;AACA;AACA,IAAI,iDAAG;AACP;AACA;;AAEA;AACA;AACA,IAAI,iDAAG;AACP;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA;AACA;AACA,QAAQ,iDAAG;AACX;;;;;;;;;;;;;;;;ACtBuC;;AAEvC;AACA;AACA;AACA,IAAI,iDAAG;AACP;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA,QAAQ,iDAAG;AACX;;;;;;;;;;;;;;;;ACfkC;;AAElC;AACA;AACA;AACA;AACA,IAAI,iDAAG;AACP;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;;;;;;;;;;;;;;;;ACbkC;;AAElC,6BAAe,sCAAW;AAC1B;AACA;AACA,kBAAkB,cAAc;AAChC,eAAe,oBAAoB;;AAEnC;AACA,qBAAqB,iDAAG;AACxB;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,KAAK;;AAEL;AACA;AACA,GAAG;AACH;;;;;;;;;;;;;;;;;AC5BqC;AACC;;AAEtC,6BAAe,oCAAS;AACxB,2CAA2C,wDAAO;;AAElD,sFAAsF,OAAO;AAC7F,6FAA6F,OAAO;AACpG;AACA;AACA;AACA;AACA;;AAEA,aAAa,iDAAU;AACvB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACfuC;AACC;AACU;AACR;AACM;AACR;AACc;AACV;AACF;AACN;AACQ;AACA;AACM;AACA;AACR;AACU;AACZ;AACU;AACE;AACV;AACJ;;AAEtC;;AAEO;AACP;AACA;AACA;AACA;AACA;;AAEe;AACf,SAAS,wDAAS;AAClB;;AAEO;AACP;AACA;;AAEA,0BAA0B,oDAAS;;AAEnC;AACA;AACA,UAAU,kDAAiB;AAC3B,aAAa,qDAAoB;AACjC;AACA;AACA,UAAU,kDAAiB;AAC3B,SAAS,iDAAgB;AACzB,aAAa,qDAAoB;AACjC,cAAc,sDAAqB;AACnC;AACA;AACA;AACA;AACA;AACA;AACA,MAAM,8CAAa;AACnB,QAAQ,gDAAe;AACvB,aAAa,qDAAoB;AACjC,SAAS,kDAAgB;AACzB,cAAc,uDAAqB;AACnC,QAAQ,iDAAe;AACvB,aAAa,sDAAoB;AACjC,UAAU,mDAAiB;AAC3B,SAAS,kDAAgB;AACzB,SAAS,kDAAgB;AACzB,YAAY,qDAAmB;AAC/B,QAAQ,iDAAe;AACvB,eAAe,wDAAsB;AACrC,OAAO,gDAAc;AACrB;AACA;;;;;;;;;;;;;;;;;;;ACxE+B;AACqD;;AAEpF,6BAAe,oCAAS;AACxB;AACA,kCAAkC,sDAAiB;AACnD,qBAAqB,gDAAK,GAAG,sDAAc;AAC3C,aAAa,oDAAK,eAAe,sDAAc;AAC/C,QAAQ,sDAAiB;AACzB;;;;;;;;;;;;;;;;ACTsC;;AAEtC,6BAAe,oCAAS;AACxB;;AAEA,gKAAgK,OAAO;AACvK,yHAAyH,OAAO;AAChI;AACA;AACA;AACA;AACA;;AAEA,SAAS,QAAQ;AACjB;AACA;;AAEA,aAAa,iDAAU;AACvB;;;;;;;;;;;;;;;;AClB6C;;AAE7C;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA,oCAAoC,8CAAI,GAAG,6CAAG;AAC9C;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA,QAAQ,iDAAG;AACX;AACA;;;;;;;;;;;;;;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACVqC;AACG;;AAExC,cAAc,uDAAQ;AACtB;;AAEO;AACA;AACA;AACA;AACA;AACA;AACA;;AAEP,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEO;AACP;AACA,2DAA2D;AAC3D;AACA;;AAEO;AACP;AACA,2DAA2D;AAC3D;AACA;;AAEO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,eAAe,+CAAK;;AAEpB;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,4CAA4C;AAC5C,sCAAsC,oDAAO;;AAE7C;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,yDAAyD;AACzD;AACA,2DAA2D;AAC3D;AACA,IAAI,oDAAO;AACX;AACA;AACA;AACA;AACA;AACA,KAAK;;AAEL;AACA;AACA;AACA;AACA,yCAAyC;AACzC;;AAEA;AACA;AACA,wBAAwB,OAAO;AAC/B;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,qCAAqC;AACrC;AACA;AACA;;;;;;;;;;;;;;;;;;ACxJsC;AACA;AACM;;AAE5C,6BAAe,oCAAS;AACxB;AACA;;AAEA,6CAA6C,wDAAQ;;AAErD,sFAAsF,OAAO;AAC7F,gHAAgH,OAAO;AACvH;AACA;AACA;AACA,QAAQ,wDAAQ,qCAAqC,iDAAG;AACxD;AACA;AACA;;AAEA,aAAa,iDAAU;AACvB;;;;;;;;;;;;;;;;;;ACrByC;AACH;AACM;;AAE5C,6BAAe,oCAAS;AACxB;AACA;;AAEA,6CAA6C,wDAAW;;AAExD,0FAA0F,OAAO;AACjG,+DAA+D,OAAO;AACtE;AACA,yFAAyF,iDAAG,wCAAwC,OAAO;AAC3I;AACA,YAAY,wDAAQ;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,aAAa,iDAAU;AACvB;;;;;;;;;;;;;;;;ACzBuC;;AAEvC,gBAAgB,oDAAS;;AAEzB,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;;;;;;ACN+E;AAC5C;AACD;AACI;AACK;;AAE3C;AACA;AACA;AACA;AACA;AACA,kBAAkB,wDAAK;AACvB,oDAAoD,wDAAK;AACzD;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,kBAAkB,wDAAK;AACvB;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,kBAAkB,wDAAK;AACvB;AACA;AACA,6EAA6E,wDAAK;AAClF;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,mBAAmB,iDAAG;AACtB;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB,yCAAyC,mEAAoB,GAAG,uDAAW;AAC3E;AACA;AACA;AACA;AACA,+CAA+C,qDAAU;AACzD;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;AC/EA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACvBsC;;AAEtC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA,qBAAqB,qDAAU;AAC/B;AACA;;;;;;;;;;;;;;;ACnBA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACvB6C;AACD;;AAE5C,6BAAe,sCAAW;AAC1B;AACA;AACA,YAAY,gDAAK;;AAEjB,4DAA4D,OAAO;AACnE,+DAA+D,OAAO;AACtE;AACA,sBAAsB,iDAAG;AACzB,QAAQ,wDAAQ;AAChB;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;;AAEA,aAAa,iDAAU;AACvB;;;;;;;;;;;;;;;;;ACvBuC;;AAEvC;AACA;AACA;AACA,mBAAmB,iDAAG;AACtB;;AAEA;AACA;AACA;AACA;AACA;AACA,yCAAyC,OAAO;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,mBAAmB,iDAAG;AACtB;;AAEA;AACA;AACA;AACA;AACA;AACA,oBAAoB,yBAAyB,4BAA4B,OAAO;AAChF;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;;AAEA;AACA,gBAAgB,iDAAG;AACnB,yCAAyC,OAAO;AAChD;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEO;AACP;;AAEA;AACA,mBAAmB,iDAAG;AACtB,2CAA2C;AAC3C,GAAG;;AAEH;AACA,WAAW,iDAAG;AACd;AACA;;;;;;;;;;;;;;;AChFA,iEAAe,YAAY,EAAC;;;;;;;;;;;;;;;ACAb;AACf;AACA;AACA;AACA;AACA,CAAC;AACD;AACA,WAAW,kDAAkD;AAC7D,kBAAkB,yDAAyD;AAC3E,aAAa,oDAAoD;AACjE,gBAAgB,uDAAuD;AACvE,QAAQ;AACR,GAAG;AACH;;;;;;;;;;;;;;;;;;;;ACb0C;AACoE;;;;;;;;;;;;;;;;ACDvG;AACP;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;;;;;;;;;;;;;;;;;ACPO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;;AAEO;;AAEP;;AAEe;AACf;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;AClDqC;AACW;AACD;AACF;AACL;AACH;AACF;AACgB;AACC;;AAEpD;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,wBAAwB,mDAAQ;AAChC;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,sCAAW;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,sDAAe;AACnC,kBAAkB,uDAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,oCAAoC,eAAe;AACnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA;AACA,uBAAuB,mDAAQ;AAC/B;AACA;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA,+CAA+C,oDAAS;AACxD;;AAEA;AACA;AACA,oEAAoE,oDAAS;AAC7E;;AAEA;AACA;AACA;;AAEA;AACA;AACA,uCAAuC,gDAAgD;AACvF,oDAAoD,8CAA8C;AAClG;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC;AAChC,mBAAmB,4BAA4B,QAAQ,oDAAS;AAChE;AACA;AACA,SAAS;AACT;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA,cAAc,wDAAM;AACpB;AACA;AACA;AACA,YAAY,iDAAS;AACrB;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,YAAY,wDAAO;;AAEnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,MAAM,wDAAS;AACf;AACA;;AAEA,IAAI,uDAAO;AACX;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,YAAY,wDAAM;AAClB,YAAY,wDAAO;AACnB;AACA;;AAEA,IAAI,mDAAW;AACf,IAAI,0DAAa;AACjB;AACA,IAAI,wDAAS;AACb;;AAEA;AACA,MAAM,uDAAO;AACb;AACA;AACA;AACA;AACA;AACA,sEAAsE,wDAAO;AAC7E;;AAEA;AACA;AACA,MAAM,gDAAU;AAChB,MAAM,uDAAO;AACb;AACA;AACA;;AAEA;AACA;AACA;AACA,aAAa,wDAAO;AACpB;AACA;AACA;;AAEA,IAAI,uDAAO;AACX,sBAAsB,wDAAM;AAC5B,SAAS,wDAAM;AACf;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,IAAI,0DAAa;AACjB,gBAAgB,OAAO;AACvB,0BAA0B,wDAAO;AACjC;AACA;AACA;AACA;;AAEA;;AAEA;AACA,iFAAiF,uBAAuB;AACxG,MAAM,wDAAS;AACf;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,IAAI,uDAAO;AACX,gBAAgB,OAAO;AACvB,0BAA0B,wDAAO;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,IAAI,0DAAa;AACjB;AACA,0CAA0C,qBAAqB;AAC/D,gBAAgB,OAAO;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,wDAAO;AACnB;AACA,kBAAkB,wDAAM;AACxB;AACA;AACA;AACA;AACA;;AAEA;AACA,0EAA0E,wDAAQ;AAClF;;AAEA;AACA,sEAAsE,wDAAQ;AAC9E;;AAEA;AACA,yEAAyE,wDAAQ;AACjF;;AAEA;AACA,sEAAsE,wDAAQ;AAC9E;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;;AC9bqC;;AAErC;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;;AAEA;AACA,gCAAgC,wDAAM;;AAEtC;AACA,uBAAuB,wDAAM;AAC7B;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AClEoC;;AAEpC;AACA;AACA;AACA;AACA;AACA;AACO,+BAA+B,8CAAM;AAC5C;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,mCAAmC,eAAe,YAAY,SAAS;AACvE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACxHoC;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACO,0BAA0B,8CAAM;AACvC;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACpGA;AACY;AACZ;AACA;AACA;AACA,cAAc,uBAAuB;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,wBAAwB;AACvC,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,gBAAgB,6EAA6E;AAC7F;AACA;AACA,+CAA+C;AAC/C;AACA;AACA;AACA,kBAAkB,wBAAwB;AAC1C;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,sCAAsC;AACxD;AACA;AACA;AACA;AACA,kBAAkB,uCAAuC;AACzD;AACA;AACA;AACA;AACA,kBAAkB,cAAc;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,0CAA0C;AACzD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,gCAAgC;AAC/C,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,oBAAoB;AACpC,eAAe,cAAc;AAC7B,iBAAiB;AACjB;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,QAAQ;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA,kBAAkB,cAAc;AAChC;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACzJA;AACY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA;AACA;AACA,kBAAkB,qCAAqC;AACvD;AACA;AACA;AACA;AACA;AACA,kBAAkB,2CAA2C;AAC7D;AACA;AACA,mBAAmB,qCAAqC;AACxD;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,oCAAoC;AACnD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;AC3DA;AACY;AACZ;AACA,gBAAgB,0DAA0D;AAC1E;AACA;AACA;AACA,cAAc,kCAAkC;AAChD;AACA,CAAqC;AACiB;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA,eAAe,mBAAmB;AAClC,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA,mBAAmB,mBAAmB;AACtC;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,OAAO;AACzB,sBAAsB;AACtB;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA,mBAAmB,iCAAiC;AACpD;AACA;AACA,mBAAmB,iCAAiC;AACpD;AACA;AACA,mBAAmB,iCAAiC;AACpD;AACA;AACA;AACA,mBAAmB,UAAU;AAC7B,wBAAwB;AACxB;AACA;AACA;AACA;AACA,qBAAqB,iDAAY;AACjC,sBAAsB,6CAAM;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA,wCAAwC;AACxC;AACA;AACA,iBAAiB;AACjB,cAAc,wDAAM;AACpB;AACA;AACA;AACA,mBAAmB,kBAAkB;AACrC;AACA,mBAAmB,kBAAkB;AACrC;AACA,mBAAmB,kBAAkB;AACrC;AACA,mBAAmB,kBAAkB;AACrC;AACA;AACA;AACA,mBAAmB,gBAAgB;AACnC;AACA;AACA;AACA,oBAAoB,wDAAwD;AAC5E,4BAA4B;AAC5B;AACA;AACA,kBAAkB,MAAM;AACxB;AACA;AACA;AACA;AACA,gBAAgB,gBAAgB;AAChC;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA;AACA,gBAAgB,gBAAgB;AAChC;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB,UAAU;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,oBAAoB;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB,QAAQ;AACzB;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB,QAAQ;AACzB;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB,QAAQ;AACzB;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB,QAAQ;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AClaA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACO,oBAAoB,kDAAQ;AACnC;AACA;AACA;AACA,eAAe,iCAAiC;AAChD,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACtBA;AACY;AACZ;AACA,CAAqC;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA,gCAAgC,wDAAM;AACtC;AACA;AACA,uBAAuB,wDAAM;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB;AACA;AACA,oBAAoB;AACpB;AACA;AACA;;;;;;;;;;;;;;;;;;;;AC3GA;AACY;AACZ;AACA;AACA,CAA0C;AACJ;AACgB;AACU;AAChE;AACA;AACqC;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,gBAAgB;AAC/B,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA,mBAAmB,mBAAmB;AACtC;AACA;AACA;AACA,kBAAkB;AAClB;AACA,6BAA6B,oDAAS;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc,wDAAM;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,SAAS;AAC3B,2BAA2B,gDAAO;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,wBAAwB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,wDAAM;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,mEAAmE;AAClF,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,MAAM;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,YAAY;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,gEAAgE,aAAa;AACjG;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,oBAAoB;AACpC,kBAAkB,gEAAgE;AAClF;AACA;AACA;AACA;AACA;AACA,6CAA6C,QAAQ;AACrD,uBAAuB,4BAA4B;AACnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,uCAAuC;AAC9D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,kBAAkB;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,+BAA+B;AACjD;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,gBAAgB,QAAQ,YAAY,MAAM;AAC1C;AACA;AACA;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA,gBAAgB,gBAAgB,cAAc,MAAM;AACpD;AACA;AACA;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA,gBAAgB,gBAAgB,cAAc,MAAM;AACpD;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,gBAAgB,QAAQ,cAAc,MAAM;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B,+DAAW;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oCAAoC,yEAAgB;AACpD;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,kBAAkB,MAAM;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;AC9eA;AACY;AACZ;AACA;AACA;AACA,6BAA6B,cAAc;AAC3C;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,gBAAgB;AAC/B,eAAe,kFAAkF;AACjG,iBAAiB,0DAA0D;AAC3E;AACA,gDAAgD;AAChD;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA,kBAAkB,qCAAqC;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,4CAA4C;AAC3D,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,gBAAgB;AAC/B,eAAe,8CAA8C;AAC7D,iBAAiB,0DAA0D;AAC3E,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;;;;;;;;;;;;;;;;AC5FA;AACY;AACZ;AACA,CAAwC;AACxC;AACA,cAAc,2GAA2G;AACzH;AACA;AACA;AACA;AACA,aAAa,gEAAgE;AAC7E;AACA;AACA;AACA;AACA;AACA;AACA;AACO,oBAAoB,kDAAQ;AACnC;AACA;AACA,gBAAgB,gHAAgH,oBAAoB,sKAAsK;AAC1T;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,qBAAqB;AACvC;AACA;AACA;AACA;AACA,kBAAkB,0DAA0D;AAC5E;AACA;AACA;AACA;AACA,kBAAkB,kDAAkD,sBAAsB;AAC1F,sDAAsD,cAAc;AACpE;AACA;AACA,kBAAkB,oBAAoB;AACtC;AACA;AACA;AACA;AACA,kBAAkB,mCAAmC;AACrD;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,kCAAkC;AACjD,eAAe,iCAAiC;AAChD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,kDAAkD,sBAAsB;AAC1F;AACA;AACA;AACA,gBAAgB,kDAAkD,sBAAsB,cAAc,MAAM;AAC5G;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB,MAAM;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,mCAAmC;AAClD,iBAAiB,MAAM;AACvB;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACjGA;AACY;;AAEZ,CAAqC;AACrC,WAAW,aAAa;;AAExB;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,QAAQ;AACvB;AACA;AACA;;AAEA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;;AAEA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,aAAa;AAChC;;AAEA;AACA;AACA,kBAAkB,kCAAkC;AACpD,uBAAuB,wDAAM;;AAE7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,wDAAM;AACjC;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;;AAEA;AACA;AACA,eAAe,YAAY;AAC3B;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,eAAe,YAAY;AAC3B,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACtLA;AACY;AACZ;AACA,gBAAgB,+DAA+D,kBAAkB,iFAAiF;AAClL;AACA,CAA8B;AACc;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACO,sBAAsB,qDAAO;AACpC;AACA,eAAe,8BAA8B;AAC7C,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,gBAAgB,sFAAsF;AACtG;AACA,+CAA+C;AAC/C;AACA;AACA;AACA;AACA,kBAAkB,2CAA2C;AAC7D;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,mDAAmD;AAClE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA,mCAAmC,oDAAG;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,yCAAyC;AACxD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC/GA;AACY;AACZ;AACA,CAA4C;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACO,qBAAqB,qDAAO;AACnC;AACA,eAAe,QAAQ;AACvB,eAAe,gBAAgB;AAC/B,iBAAiB;AACjB;AACA,4CAA4C;AAC5C;AACA;AACA;AACA;AACA,kBAAkB,2CAA2C;AAC7D;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,mDAAmD;AAClE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,yCAAyC;AACxD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACtDA;AACY;AACZ;AACA,gBAAgB,+DAA+D,kBAAkB,iFAAiF;AAClL;AACA;AACA,CAA4C;AAC5C,WAAW,2BAA2B;AACtC;AACA;AACoC;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACO,wBAAwB,qDAAO;AACtC;AACA,eAAe,8BAA8B;AAC7C,eAAe,QAAQ;AACvB,gBAAgB,uGAAuG;AACvH;AACA,mCAAmC;AACnC;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA,uCAAuC,oDAAI;AAC3C;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA,aAAa;AACb,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,yCAAyC;AACxD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,yCAAyC;AACxD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,yDAAyD;AACzE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,YAAY;AACxC,gCAAgC,YAAY;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B,6CAAG,IAAI,SAAS,EAAE,GAAG,GAAG,GAAG;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV,gDAAgD,GAAG,IAAI,GAAG;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,yBAAyB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,yCAAyC;AACxD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,mDAAmD;AACtE;AACA;AACA;AACA;AACA,mBAAmB,yCAAyC;AAC5D;AACA;AACA,kDAAkD,kCAAkC;AACpF;AACA,sDAAsD,kCAAkC;AACxF;AACA,2BAA2B,QAAQ;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD;AACA,eAAe,QAAQ;AACvB;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA,eAAe,sCAAsC;AACrD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AChSA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,8BAA8B,iDAAK;AAC1C;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA,mBAAmB,uCAAuC;AAC1D;AACA;AACA,mBAAmB,gBAAgB;AACnC;AACA;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,yBAAyB;AACxC,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,uCAAuC;AACtD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,8BAA8B;AACnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,UAAU;AACrC,+BAA+B,UAAU;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACjKA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,mCAAmC,iDAAK;AAC/C;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB,4BAA4B;AAC9C;AACA;AACA;AACA;AACA,eAAe,0CAA0C;AACzD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AChFA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,iCAAiC,iDAAK;AAC7C;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA,mBAAmB,4BAA4B;AAC/C;AACA;AACA,mBAAmB,kBAAkB;AACrC;AACA,mBAAmB,kBAAkB;AACrC;AACA,mBAAmB,kBAAkB;AACrC;AACA,mBAAmB,kBAAkB;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,uCAAuC;AACtD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACvGA;AACY;AACZ;AACA,CAAwC;AACT;AAC/B;AACA;AACA;AACA;AACA;AACO,2BAA2B,iDAAK;AACvC;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA,kBAAkB,kDAAkD;AACpE;AACA;AACA;AACA;AACA,kBAAkB,4BAA4B;AAC9C;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA,eAAe,0CAA0C;AACzD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,eAAe;AAC/C;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,iBAAiB;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,oDAAI;AACpC;AACA,wBAAwB,iBAAiB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACrNA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACO,wBAAwB,iDAAK;AACpC;AACA,eAAe,mGAAmG;AAClH,eAAe,uCAAuC;AACtD,gBAAgB,oNAAoN;AACpO;AACA,0CAA0C;AAC1C;AACA;AACA;AACA,mBAAmB,mGAAmG;AACtH;AACA;AACA,mBAAmB,uCAAuC;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA,kBAAkB,yDAAyD;AAC3E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,gDAAgD;AACnE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,+EAA+E;AAC9F,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,gCAAgC;AAC/C,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACnJA;AACY;AACZ;AACA,CAAwC;AACV;AAC9B;AACA;AACA,cAAc,oCAAoC;AAClD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,yBAAyB,iDAAK;AACrC;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA,kBAAkB,+BAA+B;AACjD;AACA;AACA;AACA;AACA,kBAAkB,+BAA+B;AACjD;AACA;AACA;AACA;AACA,kBAAkB,+BAA+B;AACjD;AACA;AACA;AACA;AACA,kBAAkB,+BAA+B;AACjD;AACA;AACA;AACA;AACA;AACA,kBAAkB,iBAAiB;AACnC;AACA;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,yBAAyB;AAC3C;AACA;AACA;AACA;AACA,kBAAkB,2BAA2B;AAC7C;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,uCAAuC;AACtD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,iBAAiB;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC,+BAA+B,6CAAG;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACjMA;AACY;AACZ;AACA,CAA0C;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACO,kCAAkC,mDAAM;AAC/C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,yBAAyB;AAC3C;AACA;AACA;AACA;AACA,kBAAkB,kCAAkC;AACpD;AACA,6CAA6C;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,QAAQ;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC5GA;AACY;AACZ;AACA,CAA0C;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,kCAAkC,mDAAM;AAC/C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA,4BAA4B,4DAA4D;AACxF;AACA,4BAA4B,4DAA4D;AACxF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,+CAA+C;AAC/D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,QAAQ;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,QAAQ;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,QAAQ;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACnGA;AACY;AACZ;AACA,CAA0C;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,0BAA0B,mDAAM;AACvC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,+CAA+C;AAC/D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oEAAoE,eAAe;AACnF;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,OAAO;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;AClHA;AACY;AACZ;AACA,CAA0C;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACO,gCAAgC,mDAAM;AAC7C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,2GAA2G;AAC5H;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,gBAAgB;AAC3B,WAAW,gBAAgB;AAC3B,WAAW,QAAQ;AACnB,eAAe;AACf;AACO,0DAA0D;AACjE;AACA,oBAAoB,yBAAyB;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;ACxFA;AACY;AACZ;AACA,CAA0C;AACF;AACV;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACO,yBAAyB,mDAAM;AACtC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,oBAAoB,8FAA8F;AAClH;AACA;AACA;AACA,oBAAoB,0CAA0C;AAC9D;AACA;AACA;AACA,oBAAoB,wCAAwC;AAC5D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,mGAAmG;AACpH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA,UAAU;AACV;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+EAA+E;AAC/E;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,gBAAgB;AAC3B,WAAW,yBAAyB;AACpC,YAAY,SAAS;AACrB,aAAa;AACb;AACO,2CAA2C;AAClD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,qDAAqD;AACjE,WAAW,GAAG;AACd,aAAa;AACb;AACO,6CAA6C;AACpD;AACA;AACA;AACA;AACA,2CAA2C,qDAAI,MAAM,oDAAG;AACxD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,iBAAiB;AAC7B,YAAY,iBAAiB;AAC7B,YAAY,SAAS;AACrB,aAAa;AACb;AACO,oDAAoD;AAC3D;AACA;AACA;AACA,mCAAmC,QAAQ;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,SAAS;AACrB,YAAY,SAAS;AACrB,aAAa;AACb;AACO,2DAA2D;AAClE;AACA;AACA;AACA;AACA,kCAAkC,QAAQ;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,0BAA0B;AACtC,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;AC/NA;AACY;AACZ;AACA,CAAqC;AACK;AAC1C;AACA;AACA;AACA;AACA;AACA;AACO,4BAA4B,mDAAM;AACzC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yDAAyD,wDAAwD;AACjH,oBAAoB,yDAAM;AAC1B;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA,2DAA2D;AAC3D,oBAAoB,wDAAM;AAC1B;AACA,iBAAiB;AACjB;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,SAAS;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,SAAS;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;;;;;;;;;;;;;;;;AC7SA;AACY;AACZ;AACA,CAAwC;AACxC;AACA,cAAc,oEAAoE;AAClF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,+BAA+B,iDAAK;AAC3C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA,kBAAkB,mFAAmF;AACrG;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,QAAQ;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,QAAQ;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,QAAQ;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;;;;;AClSA;AACY;AACZ;AACA,CAAwC;AACA;AACmC;AACN;AACrC;AAChC;AACA;AACA;AACA;AACA;AACA;AACO,8BAA8B,iDAAK;AAC1C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,wCAAwC;AAC1D;AACA;AACA;AACA,kBAAkB,wCAAwC;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,qDAAY;AACjC;AACA;AACA;AACA,YAAY,uEAAiB;AAC7B;AACA,0BAA0B,qEAAe;AACzC;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,8EAAmB;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,oDAAK;AAChC;AACA;AACA;AACA,gCAAgC,gBAAgB;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,gBAAgB;AAChD;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;ACpIA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,yBAAyB,iDAAK;AACrC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,mBAAmB,mFAAmF;AACtG;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;;ACnGA;AACY;AACZ;AACA,CAAwC;AACE;AAC1C;AACA,eAAe,6HAA6H;AAC5I;AACA;AACA;AACA;AACA;AACO,4BAA4B,iDAAK;AACxC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA;AACA,kBAAkB,SAAS;AAC3B;AACA;AACA;AACA;AACA,kBAAkB,SAAS;AAC3B;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,cAAc;AAClC,sBAAsB,oDAAS;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACxNA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,2BAA2B,iDAAK;AACvC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,iBAAiB,uBAAuB,wBAAwB;AAClF;AACA;AACA,kBAAkB,iBAAiB,uBAAuB,wBAAwB;AAClF;AACA;AACA,kBAAkB,iBAAiB,uBAAuB,wBAAwB;AAClF;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,QAAQ;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,yBAAyB;AAC9C,qBAAqB;AACrB;AACA;AACA;AACA,2BAA2B,WAAW;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA,+BAA+B,WAAW;AAC1C;AACA,2BAA2B,QAAQ;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;AC9HA;AACY;AACZ;AACA,CAA8C;AACA;AACoC;AAC1C;AACxC;AACA;AACA;AACA;AACA;AACO;AACP,+CAA+C;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,wDAAW;AAC9B;AACA,wBAAwB,wDAAW;AACnC;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA,+BAA+B,uEAAuE;AACtG;AACA;AACA;AACA;AACA,eAAe,oDAAoD;AACnE,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,4FAA6B,GAAG,0BAA0B;AAChF;AACA,wBAAwB,wDAAW,GAAG,8DAA8D;AACpG;AACA,2CAA2C,kDAAkD;AAC7F;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,iDAAK;AAChC,gBAAgB,kBAAkB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC7GA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,0BAA0B,iDAAK;AACtC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;AClGA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,6BAA6B,iDAAK;AACzC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,iDAAiD;AACnE;AACA;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;ACrGA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,0BAA0B,iDAAK;AACtC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,yCAAyC;AAC3D;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4CAA4C,sCAAsC;AAClF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;AC9MA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACO,2BAA2B,iDAAK;AACvC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,QAAQ;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,kEAAkE;AAC/F;AACA;;;;;;;;;;;;;;;;ACxGA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,kCAAkC,iDAAK;AAC9C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA,mBAAmB,uFAAuF;AAC1G;AACA;AACA,mBAAmB,uFAAuF;AAC1G;AACA;AACA,mBAAmB,+GAA+G;AAClI;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA,6BAA6B,kEAAkE;AAC/F;AACA;;;;;;;;;;;;;;;;AClJA;AACY;AACZ;AACA,CAA0C;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACO,gCAAgC,oDAAS;AAChD,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,oEAAoE;AACtF;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,uCAAuC;AAC3D,sBAAsB,oDAAS;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;AC/GA;AACY;AACZ;AACA,CAAwC;AACxC;AACA,gBAAgB,gIAAgI;AAChJ;AACA;AACA,aAAa,yCAAyC;AACtD;AACA;AACA;AACA;AACA;AACO,wBAAwB,iDAAK;AACpC,gBAAgB,QAAQ;AACxB,yBAAyB;AACzB;AACA;AACA;AACA,kBAAkB,+CAA+C;AACjE;AACA;AACA;AACA,kBAAkB,+CAA+C;AACjE;AACA;AACA;AACA,kBAAkB,+CAA+C;AACjE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,QAAQ;AACvB,eAAe,uCAAuC;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,cAAc;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA,2BAA2B,kBAAkB;AAC7C;AACA;AACA;AACA;AACA,2BAA2B,kBAAkB;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,QAAQ;AACvB,eAAe,SAAS;AACxB,eAAe,SAAS;AACxB,eAAe,SAAS;AACxB,kBAAkB;AAClB;AACA;AACA,oBAAoB,eAAe;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,kBAAkB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,kBAAkB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACnNA;AACY;AACZ;AACA,CAAwC;AACgB;AACmC;AAC3F;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,4CAA4C,iDAAK;AACxD,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,oDAAoD;AACtE;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA,oBAAoB,SAAS;AAC7B;AACA,wBAAwB,kBAAkB;AAC1C;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA,mBAAmB,iCAAiC;AACpD,uBAAuB,oGAA8B;AACrD;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,QAAQ;AAChC;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,QAAQ;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,qEAAe;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,2CAA2C;AACxE;AACA;;;;;;;;;;;;;;;;;;AC1GA;AACY;AACZ;AACA,CAAwC;AACgB;AAC6B;AACrF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,oCAAoC,iDAAK;AAChD,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA;AACA;AACA,oBAAoB,4BAA4B;AAChD;AACA;AACA;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,qEAAe;AACrC;AACA;AACA,0CAA0C,4BAA4B;AACtE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,8FAA2B;AACnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;AC7HA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,0BAA0B,iDAAK;AACtC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA,kBAAkB,mGAAmG;AACrH;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;;;ACtIA;AACY;AACZ;AACA,CAA0C;AACwC;AACA;AAClF;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,oDAAoD;AACnE,eAAe,gBAAgB;AAC/B,eAAe,gBAAgB;AAC/B,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA,+CAA+C;AAC/C;AACA;AACA;AACA;AACA,oBAAoB,kCAAkC;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA,2BAA2B,2DAAK;AAChC;AACA;AACA,wBAAwB,mBAAmB;AAC3C;AACA,8BAA8B,4FAA6B;AAC3D;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8BAA8B,oDAAS;AACvC;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACpEA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,wBAAwB,iDAAK;AACpC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,2EAA2E;AAC7F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;AACA;AACA;AACA;AACA,eAAe,gBAAgB;AAC/B,eAAe,qCAAqC;AACpD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC7GA;AACY;AACZ;AACA,CAAwC;AACxC;AACA,cAAc,wCAAwC;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,8BAA8B,iDAAK;AAC1C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA;AACA,kBAAkB,kEAAkE;AACpF;AACA,mBAAmB,kEAAkE;AACrF;AACA;AACA;AACA;AACA,kBAAkB,kEAAkE;AACpF;AACA,mBAAmB,kEAAkE;AACrF;AACA,mBAAmB,2EAA2E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,QAAQ;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,QAAQ;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;;AC7NA;AACY;;AAEZ,CAAiE;AACjC;;AAEhC;AACA;AACA;AACA;AACO;AACP;AACA;AACA,eAAe,uBAAuB;AACtC;AACA;AACA;AACA;;AAEA,uBAAuB,iEAAiB;AACxC;AACA,YAAY,4DAAY;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,4DAAY;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,mBAAmB,oDAAK;AACxB,sBAAsB;AACtB;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;;;;;;;;;;;ACrHA;AACY;;AAEZ,CAAiE;AACjC;;AAEhC;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,eAAe,GAAG;AAClB,eAAe,gBAAgB;AAC/B,gBAAgB,0BAA0B;AAC1C,eAAe,QAAQ;AACvB,eAAe,kBAAkB;AACjC;AACA;AACA,mBAAmB,uBAAuB;AAC1C;AACA;AACA;AACA;;AAEA,mBAAmB,aAAa;AAChC,wBAAwB,4DAAY;AACpC;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,qCAAqC;AACrC,8BAA8B;AAC9B,iCAAiC;AACjC;AACA;AACA,gCAAgC,mBAAmB,uCAAuC;AAC1F;AACA,aAAa;AACb;AACA,8BAA8B;;AAE9B;AACA;AACA;AACA,0DAA0D;AAC1D;AACA;AACA,sEAAsE;AACtE;AACA;AACA;AACA;AACA,wCAAwC;AACxC,qFAAqF;AACrF;AACA;AACA;AACA;AACA,wCAAwC;AACxC,gGAAgG;AAChG;AACA;AACA;AACA,kDAAkD;AAClD;AACA;AACA,qEAAqE;AACrE;AACA;AACA;AACA;AACA,4DAA4D;AAC5D,gGAAgG;AAChG;AACA,cAAc;AACd;AACA;AACA;AACA,kDAAkD;AAClD;AACA;AACA,mFAAmF;AACnF;AACA;AACA;AACA;AACA,4DAA4D;AAC5D,4GAA4G;AAC5G;AACA,cAAc;AACd;AACA,8CAA8C;AAC9C;AACA,UAAU;AACV,0CAA0C;AAC1C;;AAEA;AACA,4DAA4D,gBAAgB;AAC5E,iEAAiE,gBAAgB;AACjF;AACA;AACA;AACA,qCAAqC;AACrC,qCAAqC;AACrC,qDAAqD,OAAO,OAAO,kBAAkB;AACrF,4BAA4B,QAAQ;AACpC;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA,uBAAuB;AACvB;AACA,uBAAuB;AACvB;AACA;AACA;AACA,0BAA0B;AAC1B;AACA,2BAA2B,sBAAsB,gBAAgB,qCAAqC;AACtG;;AAEA;AACA,kGAAkG;AAClG;AACA,6DAA6D;;AAE7D;;AAEA,mBAAmB,aAAa;AAChC,wBAAwB,4DAAY;;AAEpC,mBAAmB,cAAc;AACjC,uBAAuB,iEAAiB;AACxC;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,wBAAwB,mBAAmB;AAC3C,sBAAsB,oDAAK;;AAE3B;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;;;;;;;;;;;ACrOA;AACY;;AAEZ,CAAiE;AACjC;;AAEhC;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,gBAAgB;AAC/B;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;;AAEA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,0CAA0C,mBAAmB;AAC7D;AACA;AACA,wBAAwB,mBAAmB;AAC3C;AACA;AACA;AACA,mBAAmB;AACnB;AACA,mCAAmC;AACnC;AACA,wBAAwB,mBAAmB;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B;AAC3B;AACA,+DAA+D,GAAG;AAClE;AACA,kBAAkB,QAAQ;AAC1B;AACA;;AAEA;AACA;AACA,mBAAmB,aAAa;AAChC,wBAAwB,4DAAY;;AAEpC,mBAAmB,aAAa;AAChC,wBAAwB,4DAAY;;AAEpC,mBAAmB,cAAc;AACjC,wBAAwB,iEAAiB;AACzC;;AAEA;;AAEA;AACA;;AAEA;AACA,wBAAwB,wBAAwB;AAChD,sBAAsB,oDAAK;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;ACpIA;AACY;AACZ;AACA,CAAsC;AACE;AACxC;AACA;AACA;AACA,aAAa,yBAAyB;AACtC;AACA;AACA;AACA,aAAa,yBAAyB;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa,0OAA0O;AACvP,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6CAA6C,oDAAG;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa,iIAAiI;AAC9I,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA,kBAAkB,oDAAa;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,qBAAqB;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa,mKAAmK;AAChL,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,2CAA2C;AAC9D,uBAAuB,oDAAM;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,aAAa;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa,mJAAmJ;AAChK,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc,qBAAqB;AACnC;AACA;AACA;AACA,kBAAkB,oDAAa;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,GAAG;AACd,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,gBAAgB;AAC3B;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,gBAAgB;AAC3B,WAAW,gBAAgB;AAC3B;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,yBAAyB;AACpC,WAAW,QAAQ;AACnB;AACO;AACP;AACA;AACA,oBAAoB,QAAQ;AAC5B;AACA;;;;;;;;;;;;;;;;;;;;ACzNA;AACY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,QAAQ;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,WAAW;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB,YAAY;AACrC;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACvHA;AACY;AACZ;AACA,CAA+C;AAC/C;AACO,+DAA+D;AACtE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,4EAA4E;AACnF;AACA;AACA;AACA;AACA,0DAA0D,sDAAc;AACxE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sEAAsE;AACtE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uEAAuE;AACvE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC1MA;AACY;AACZ;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,gBAAgB;AAC3B;AACA;AACA;AACO;AACP;AACA;AACA;AACA,oBAAoB,mBAAmB;AACvC;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC;AACnC,oCAAoC;AACpC;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACzDA;AACY;AACZ;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,cAAc;AACd;AACO,iDAAiD;AACxD;AACA;AACA;AACA,eAAe,uBAAuB;AACtC;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA,WAAW,uBAAuB;AAClC,YAAY,gBAAgB;AAC5B,aAAa;AACb;AACO;AACP,eAAe,mBAAmB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,uBAAuB;AAClC,WAAW,QAAQ;AACnB,YAAY,WAAW;AACvB,aAAa;AACb;AACO;AACP,eAAe,kBAAkB;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;;;;;;;UCvEA;UACA;;UAEA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;;UAEA;UACA;;UAEA;UACA;UACA;;;;;WCtBA;WACA;WACA;WACA;WACA,yCAAyC,wCAAwC;WACjF;WACA;WACA;;;;;WCPA;;;;;WCAA;WACA;WACA;WACA,uDAAuD,iBAAiB;WACxE;WACA,gDAAgD,aAAa;WAC7D;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACNA;AACY;AACZ;AACA;AACmC;AACY;AACR;AACA;AACI;AAC8B;AACzE;AACA;AACkD;AACJ;AACF;AAC5C;AACA;AACoE;AAChB;AACE;AACQ;AACR;AACN;AACI;AACJ;AAChD,WAAW,eAAe;AACsC;AACJ;AACR;AACJ;AACwB;AACgB;AACpC;AACM;AACE;AACJ;AACN;AAClD;AACA;AACgD;AACY;AACM;AACI;AACpB;AACI;AACtD;AACA;AACqD;AACgB;AACA;AAOtC;AACqD;AAC3B;AACzD;AACA,YAAY,uBAAuB;AACE;AACL;AACE;AACK;AACvC;AACA,CAA+C;AACxC,2BAA2B,yDAAS;AAC3C;AACA;AACA,CAA+C;AAC/C,sDAAmB;AACnB;AACA;AACA;AACA;AACA,CAAC","sources":["webpack://gridviz/webpack/universalModuleDefinition","webpack://gridviz/./node_modules/d3-array/src/ascending.js","webpack://gridviz/./node_modules/d3-array/src/bisect.js","webpack://gridviz/./node_modules/d3-array/src/bisector.js","webpack://gridviz/./node_modules/d3-array/src/descending.js","webpack://gridviz/./node_modules/d3-array/src/extent.js","webpack://gridviz/./node_modules/d3-array/src/greatest.js","webpack://gridviz/./node_modules/d3-array/src/max.js","webpack://gridviz/./node_modules/d3-array/src/maxIndex.js","webpack://gridviz/./node_modules/d3-array/src/min.js","webpack://gridviz/./node_modules/d3-array/src/minIndex.js","webpack://gridviz/./node_modules/d3-array/src/number.js","webpack://gridviz/./node_modules/d3-array/src/permute.js","webpack://gridviz/./node_modules/d3-array/src/quantile.js","webpack://gridviz/./node_modules/d3-array/src/quickselect.js","webpack://gridviz/./node_modules/d3-array/src/sort.js","webpack://gridviz/./node_modules/d3-color/src/color.js","webpack://gridviz/./node_modules/d3-color/src/define.js","webpack://gridviz/./node_modules/d3-color/src/lab.js","webpack://gridviz/./node_modules/d3-color/src/math.js","webpack://gridviz/./node_modules/d3-dispatch/src/dispatch.js","webpack://gridviz/./node_modules/d3-drag/src/nodrag.js","webpack://gridviz/./node_modules/d3-drag/src/noevent.js","webpack://gridviz/./node_modules/d3-dsv/src/csv.js","webpack://gridviz/./node_modules/d3-dsv/src/dsv.js","webpack://gridviz/./node_modules/d3-dsv/src/tsv.js","webpack://gridviz/./node_modules/d3-ease/src/cubic.js","webpack://gridviz/./node_modules/d3-fetch/src/dsv.js","webpack://gridviz/./node_modules/d3-fetch/src/json.js","webpack://gridviz/./node_modules/d3-fetch/src/text.js","webpack://gridviz/./node_modules/d3-format/src/defaultLocale.js","webpack://gridviz/./node_modules/d3-format/src/exponent.js","webpack://gridviz/./node_modules/d3-format/src/formatDecimal.js","webpack://gridviz/./node_modules/d3-format/src/formatGroup.js","webpack://gridviz/./node_modules/d3-format/src/formatNumerals.js","webpack://gridviz/./node_modules/d3-format/src/formatPrefixAuto.js","webpack://gridviz/./node_modules/d3-format/src/formatRounded.js","webpack://gridviz/./node_modules/d3-format/src/formatSpecifier.js","webpack://gridviz/./node_modules/d3-format/src/formatTrim.js","webpack://gridviz/./node_modules/d3-format/src/formatTypes.js","webpack://gridviz/./node_modules/d3-format/src/identity.js","webpack://gridviz/./node_modules/d3-format/src/locale.js","webpack://gridviz/./node_modules/d3-interpolate/src/basis.js","webpack://gridviz/./node_modules/d3-interpolate/src/basisClosed.js","webpack://gridviz/./node_modules/d3-interpolate/src/color.js","webpack://gridviz/./node_modules/d3-interpolate/src/constant.js","webpack://gridviz/./node_modules/d3-interpolate/src/lab.js","webpack://gridviz/./node_modules/d3-interpolate/src/number.js","webpack://gridviz/./node_modules/d3-interpolate/src/rgb.js","webpack://gridviz/./node_modules/d3-interpolate/src/string.js","webpack://gridviz/./node_modules/d3-interpolate/src/transform/decompose.js","webpack://gridviz/./node_modules/d3-interpolate/src/transform/index.js","webpack://gridviz/./node_modules/d3-interpolate/src/transform/parse.js","webpack://gridviz/./node_modules/d3-interpolate/src/zoom.js","webpack://gridviz/./node_modules/d3-random/src/defaultSource.js","webpack://gridviz/./node_modules/d3-random/src/normal.js","webpack://gridviz/./node_modules/d3-scale/src/init.js","webpack://gridviz/./node_modules/d3-scale/src/quantile.js","webpack://gridviz/./node_modules/d3-selection/src/array.js","webpack://gridviz/./node_modules/d3-selection/src/constant.js","webpack://gridviz/./node_modules/d3-selection/src/creator.js","webpack://gridviz/./node_modules/d3-selection/src/matcher.js","webpack://gridviz/./node_modules/d3-selection/src/namespace.js","webpack://gridviz/./node_modules/d3-selection/src/namespaces.js","webpack://gridviz/./node_modules/d3-selection/src/pointer.js","webpack://gridviz/./node_modules/d3-selection/src/select.js","webpack://gridviz/./node_modules/d3-selection/src/selection/append.js","webpack://gridviz/./node_modules/d3-selection/src/selection/attr.js","webpack://gridviz/./node_modules/d3-selection/src/selection/call.js","webpack://gridviz/./node_modules/d3-selection/src/selection/classed.js","webpack://gridviz/./node_modules/d3-selection/src/selection/clone.js","webpack://gridviz/./node_modules/d3-selection/src/selection/data.js","webpack://gridviz/./node_modules/d3-selection/src/selection/datum.js","webpack://gridviz/./node_modules/d3-selection/src/selection/dispatch.js","webpack://gridviz/./node_modules/d3-selection/src/selection/each.js","webpack://gridviz/./node_modules/d3-selection/src/selection/empty.js","webpack://gridviz/./node_modules/d3-selection/src/selection/enter.js","webpack://gridviz/./node_modules/d3-selection/src/selection/exit.js","webpack://gridviz/./node_modules/d3-selection/src/selection/filter.js","webpack://gridviz/./node_modules/d3-selection/src/selection/html.js","webpack://gridviz/./node_modules/d3-selection/src/selection/index.js","webpack://gridviz/./node_modules/d3-selection/src/selection/insert.js","webpack://gridviz/./node_modules/d3-selection/src/selection/iterator.js","webpack://gridviz/./node_modules/d3-selection/src/selection/join.js","webpack://gridviz/./node_modules/d3-selection/src/selection/lower.js","webpack://gridviz/./node_modules/d3-selection/src/selection/merge.js","webpack://gridviz/./node_modules/d3-selection/src/selection/node.js","webpack://gridviz/./node_modules/d3-selection/src/selection/nodes.js","webpack://gridviz/./node_modules/d3-selection/src/selection/on.js","webpack://gridviz/./node_modules/d3-selection/src/selection/order.js","webpack://gridviz/./node_modules/d3-selection/src/selection/property.js","webpack://gridviz/./node_modules/d3-selection/src/selection/raise.js","webpack://gridviz/./node_modules/d3-selection/src/selection/remove.js","webpack://gridviz/./node_modules/d3-selection/src/selection/select.js","webpack://gridviz/./node_modules/d3-selection/src/selection/selectAll.js","webpack://gridviz/./node_modules/d3-selection/src/selection/selectChild.js","webpack://gridviz/./node_modules/d3-selection/src/selection/selectChildren.js","webpack://gridviz/./node_modules/d3-selection/src/selection/size.js","webpack://gridviz/./node_modules/d3-selection/src/selection/sort.js","webpack://gridviz/./node_modules/d3-selection/src/selection/sparse.js","webpack://gridviz/./node_modules/d3-selection/src/selection/style.js","webpack://gridviz/./node_modules/d3-selection/src/selection/text.js","webpack://gridviz/./node_modules/d3-selection/src/selector.js","webpack://gridviz/./node_modules/d3-selection/src/selectorAll.js","webpack://gridviz/./node_modules/d3-selection/src/sourceEvent.js","webpack://gridviz/./node_modules/d3-selection/src/window.js","webpack://gridviz/./node_modules/d3-timer/src/timeout.js","webpack://gridviz/./node_modules/d3-timer/src/timer.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/active.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/index.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/interrupt.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/selection/index.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/selection/interrupt.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/selection/transition.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/attr.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/attrTween.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/delay.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/duration.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/ease.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/easeVarying.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/end.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/filter.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/index.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/interpolate.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/merge.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/on.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/remove.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/schedule.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/select.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/selectAll.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/selection.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/style.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/styleTween.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/text.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/textTween.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/transition.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/tween.js","webpack://gridviz/./node_modules/d3-zoom/src/constant.js","webpack://gridviz/./node_modules/d3-zoom/src/event.js","webpack://gridviz/./node_modules/d3-zoom/src/index.js","webpack://gridviz/./node_modules/d3-zoom/src/noevent.js","webpack://gridviz/./node_modules/d3-zoom/src/transform.js","webpack://gridviz/./node_modules/d3-zoom/src/zoom.js","webpack://gridviz/./src/button/Button.js","webpack://gridviz/./src/button/FullscreenButton.js","webpack://gridviz/./src/button/ZoomButtons.js","webpack://gridviz/./src/core/Dataset.js","webpack://gridviz/./src/core/Drawable.js","webpack://gridviz/./src/core/GeoCanvas.js","webpack://gridviz/./src/core/Layer.js","webpack://gridviz/./src/core/Legend.js","webpack://gridviz/./src/core/Map.js","webpack://gridviz/./src/core/MultiResolutionDataset.js","webpack://gridviz/./src/core/Style.js","webpack://gridviz/./src/core/Tooltip.js","webpack://gridviz/./src/dataset/CSVGrid.js","webpack://gridviz/./src/dataset/JSGrid.js","webpack://gridviz/./src/dataset/TiledGrid.js","webpack://gridviz/./src/layer/BackgroundLayer.js","webpack://gridviz/./src/layer/BackgroundLayerImage.js","webpack://gridviz/./src/layer/BackgroundLayerWMS.js","webpack://gridviz/./src/layer/GeoJSONLayer.js","webpack://gridviz/./src/layer/GridLayer.js","webpack://gridviz/./src/layer/LabelLayer.js","webpack://gridviz/./src/legend/ColorCategoryLegend.js","webpack://gridviz/./src/legend/ColorDiscreteLegend.js","webpack://gridviz/./src/legend/ColorLegend.js","webpack://gridviz/./src/legend/OrientationLegend.js","webpack://gridviz/./src/legend/SizeLegend.js","webpack://gridviz/./src/legend/TernaryLegend.js","webpack://gridviz/./src/style/CompositionStyle.js","webpack://gridviz/./src/style/DotDensityStyle.js","webpack://gridviz/./src/style/ImageStyle.js","webpack://gridviz/./src/style/IsoFenceStyle.js","webpack://gridviz/./src/style/JoyPlotStyle.js","webpack://gridviz/./src/style/LegoStyle.js","webpack://gridviz/./src/style/MosaicStyle.js","webpack://gridviz/./src/style/NinjaStarStyle.js","webpack://gridviz/./src/style/PillarStyle.js","webpack://gridviz/./src/style/SegmentStyle.js","webpack://gridviz/./src/style/ShapeColorSizeStyle.js","webpack://gridviz/./src/style/SideCategoryStyle.js","webpack://gridviz/./src/style/SideStyle.js","webpack://gridviz/./src/style/SquareColorCategoryWebGLStyle.js","webpack://gridviz/./src/style/SquareColorWebGLStyle.js","webpack://gridviz/./src/style/StrokeStyle.js","webpack://gridviz/./src/style/TanakaStyle.js","webpack://gridviz/./src/style/TextStyle.js","webpack://gridviz/./src/style/TimeSeriesStyle.js","webpack://gridviz/./src/utils/WebGLSquareColoring.js","webpack://gridviz/./src/utils/WebGLSquareColoringAdvanced.js","webpack://gridviz/./src/utils/WebGLSquareColoringCatAdvanced.js","webpack://gridviz/./src/utils/scale.js","webpack://gridviz/./src/utils/stretching.js","webpack://gridviz/./src/utils/ternary.js","webpack://gridviz/./src/utils/utils.js","webpack://gridviz/./src/utils/webGLUtils.js","webpack://gridviz/webpack/bootstrap","webpack://gridviz/webpack/runtime/define property getters","webpack://gridviz/webpack/runtime/hasOwnProperty shorthand","webpack://gridviz/webpack/runtime/make namespace object","webpack://gridviz/./src/index.js"],"sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine([], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"gridviz\"] = factory();\n\telse\n\t\troot[\"gridviz\"] = factory();\n})(self, () => {\nreturn ","export default function ascending(a, b) {\n  return a == null || b == null ? NaN : a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;\n}\n","import ascending from \"./ascending.js\";\nimport bisector from \"./bisector.js\";\nimport number from \"./number.js\";\n\nconst ascendingBisect = bisector(ascending);\nexport const bisectRight = ascendingBisect.right;\nexport const bisectLeft = ascendingBisect.left;\nexport const bisectCenter = bisector(number).center;\nexport default bisectRight;\n","import ascending from \"./ascending.js\";\nimport descending from \"./descending.js\";\n\nexport default function bisector(f) {\n  let compare1, compare2, delta;\n\n  // If an accessor is specified, promote it to a comparator. In this case we\n  // can test whether the search value is (self-) comparable. We can’t do this\n  // for a comparator (except for specific, known comparators) because we can’t\n  // tell if the comparator is symmetric, and an asymmetric comparator can’t be\n  // used to test whether a single value is comparable.\n  if (f.length !== 2) {\n    compare1 = ascending;\n    compare2 = (d, x) => ascending(f(d), x);\n    delta = (d, x) => f(d) - x;\n  } else {\n    compare1 = f === ascending || f === descending ? f : zero;\n    compare2 = f;\n    delta = f;\n  }\n\n  function left(a, x, lo = 0, hi = a.length) {\n    if (lo < hi) {\n      if (compare1(x, x) !== 0) return hi;\n      do {\n        const mid = (lo + hi) >>> 1;\n        if (compare2(a[mid], x) < 0) lo = mid + 1;\n        else hi = mid;\n      } while (lo < hi);\n    }\n    return lo;\n  }\n\n  function right(a, x, lo = 0, hi = a.length) {\n    if (lo < hi) {\n      if (compare1(x, x) !== 0) return hi;\n      do {\n        const mid = (lo + hi) >>> 1;\n        if (compare2(a[mid], x) <= 0) lo = mid + 1;\n        else hi = mid;\n      } while (lo < hi);\n    }\n    return lo;\n  }\n\n  function center(a, x, lo = 0, hi = a.length) {\n    const i = left(a, x, lo, hi - 1);\n    return i > lo && delta(a[i - 1], x) > -delta(a[i], x) ? i - 1 : i;\n  }\n\n  return {left, center, right};\n}\n\nfunction zero() {\n  return 0;\n}\n","export default function descending(a, b) {\n  return a == null || b == null ? NaN\n    : b < a ? -1\n    : b > a ? 1\n    : b >= a ? 0\n    : NaN;\n}\n","export default function extent(values, valueof) {\n  let min;\n  let max;\n  if (valueof === undefined) {\n    for (const value of values) {\n      if (value != null) {\n        if (min === undefined) {\n          if (value >= value) min = max = value;\n        } else {\n          if (min > value) min = value;\n          if (max < value) max = value;\n        }\n      }\n    }\n  } else {\n    let index = -1;\n    for (let value of values) {\n      if ((value = valueof(value, ++index, values)) != null) {\n        if (min === undefined) {\n          if (value >= value) min = max = value;\n        } else {\n          if (min > value) min = value;\n          if (max < value) max = value;\n        }\n      }\n    }\n  }\n  return [min, max];\n}\n","import ascending from \"./ascending.js\";\n\nexport default function greatest(values, compare = ascending) {\n  let max;\n  let defined = false;\n  if (compare.length === 1) {\n    let maxValue;\n    for (const element of values) {\n      const value = compare(element);\n      if (defined\n          ? ascending(value, maxValue) > 0\n          : ascending(value, value) === 0) {\n        max = element;\n        maxValue = value;\n        defined = true;\n      }\n    }\n  } else {\n    for (const value of values) {\n      if (defined\n          ? compare(value, max) > 0\n          : compare(value, value) === 0) {\n        max = value;\n        defined = true;\n      }\n    }\n  }\n  return max;\n}\n","export default function max(values, valueof) {\n  let max;\n  if (valueof === undefined) {\n    for (const value of values) {\n      if (value != null\n          && (max < value || (max === undefined && value >= value))) {\n        max = value;\n      }\n    }\n  } else {\n    let index = -1;\n    for (let value of values) {\n      if ((value = valueof(value, ++index, values)) != null\n          && (max < value || (max === undefined && value >= value))) {\n        max = value;\n      }\n    }\n  }\n  return max;\n}\n","export default function maxIndex(values, valueof) {\n  let max;\n  let maxIndex = -1;\n  let index = -1;\n  if (valueof === undefined) {\n    for (const value of values) {\n      ++index;\n      if (value != null\n          && (max < value || (max === undefined && value >= value))) {\n        max = value, maxIndex = index;\n      }\n    }\n  } else {\n    for (let value of values) {\n      if ((value = valueof(value, ++index, values)) != null\n          && (max < value || (max === undefined && value >= value))) {\n        max = value, maxIndex = index;\n      }\n    }\n  }\n  return maxIndex;\n}\n","export default function min(values, valueof) {\n  let min;\n  if (valueof === undefined) {\n    for (const value of values) {\n      if (value != null\n          && (min > value || (min === undefined && value >= value))) {\n        min = value;\n      }\n    }\n  } else {\n    let index = -1;\n    for (let value of values) {\n      if ((value = valueof(value, ++index, values)) != null\n          && (min > value || (min === undefined && value >= value))) {\n        min = value;\n      }\n    }\n  }\n  return min;\n}\n","export default function minIndex(values, valueof) {\n  let min;\n  let minIndex = -1;\n  let index = -1;\n  if (valueof === undefined) {\n    for (const value of values) {\n      ++index;\n      if (value != null\n          && (min > value || (min === undefined && value >= value))) {\n        min = value, minIndex = index;\n      }\n    }\n  } else {\n    for (let value of values) {\n      if ((value = valueof(value, ++index, values)) != null\n          && (min > value || (min === undefined && value >= value))) {\n        min = value, minIndex = index;\n      }\n    }\n  }\n  return minIndex;\n}\n","export default function number(x) {\n  return x === null ? NaN : +x;\n}\n\nexport function* numbers(values, valueof) {\n  if (valueof === undefined) {\n    for (let value of values) {\n      if (value != null && (value = +value) >= value) {\n        yield value;\n      }\n    }\n  } else {\n    let index = -1;\n    for (let value of values) {\n      if ((value = valueof(value, ++index, values)) != null && (value = +value) >= value) {\n        yield value;\n      }\n    }\n  }\n}\n","export default function permute(source, keys) {\n  return Array.from(keys, key => source[key]);\n}\n","import max from \"./max.js\";\nimport maxIndex from \"./maxIndex.js\";\nimport min from \"./min.js\";\nimport minIndex from \"./minIndex.js\";\nimport quickselect from \"./quickselect.js\";\nimport number, {numbers} from \"./number.js\";\nimport {ascendingDefined} from \"./sort.js\";\nimport greatest from \"./greatest.js\";\n\nexport default function quantile(values, p, valueof) {\n  values = Float64Array.from(numbers(values, valueof));\n  if (!(n = values.length) || isNaN(p = +p)) return;\n  if (p <= 0 || n < 2) return min(values);\n  if (p >= 1) return max(values);\n  var n,\n      i = (n - 1) * p,\n      i0 = Math.floor(i),\n      value0 = max(quickselect(values, i0).subarray(0, i0 + 1)),\n      value1 = min(values.subarray(i0 + 1));\n  return value0 + (value1 - value0) * (i - i0);\n}\n\nexport function quantileSorted(values, p, valueof = number) {\n  if (!(n = values.length) || isNaN(p = +p)) return;\n  if (p <= 0 || n < 2) return +valueof(values[0], 0, values);\n  if (p >= 1) return +valueof(values[n - 1], n - 1, values);\n  var n,\n      i = (n - 1) * p,\n      i0 = Math.floor(i),\n      value0 = +valueof(values[i0], i0, values),\n      value1 = +valueof(values[i0 + 1], i0 + 1, values);\n  return value0 + (value1 - value0) * (i - i0);\n}\n\nexport function quantileIndex(values, p, valueof = number) {\n  if (isNaN(p = +p)) return;\n  numbers = Float64Array.from(values, (_, i) => number(valueof(values[i], i, values)));\n  if (p <= 0) return minIndex(numbers);\n  if (p >= 1) return maxIndex(numbers);\n  var numbers,\n      index = Uint32Array.from(values, (_, i) => i),\n      j = numbers.length - 1,\n      i = Math.floor(j * p);\n  quickselect(index, i, 0, j, (i, j) => ascendingDefined(numbers[i], numbers[j]));\n  i = greatest(index.subarray(0, i + 1), (i) => numbers[i]);\n  return i >= 0 ? i : -1;\n}\n","import {ascendingDefined, compareDefined} from \"./sort.js\";\n\n// Based on https://github.com/mourner/quickselect\n// ISC license, Copyright 2018 Vladimir Agafonkin.\nexport default function quickselect(array, k, left = 0, right = Infinity, compare) {\n  k = Math.floor(k);\n  left = Math.floor(Math.max(0, left));\n  right = Math.floor(Math.min(array.length - 1, right));\n\n  if (!(left <= k && k <= right)) return array;\n\n  compare = compare === undefined ? ascendingDefined : compareDefined(compare);\n\n  while (right > left) {\n    if (right - left > 600) {\n      const n = right - left + 1;\n      const m = k - left + 1;\n      const z = Math.log(n);\n      const s = 0.5 * Math.exp(2 * z / 3);\n      const sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);\n      const newLeft = Math.max(left, Math.floor(k - m * s / n + sd));\n      const newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));\n      quickselect(array, k, newLeft, newRight, compare);\n    }\n\n    const t = array[k];\n    let i = left;\n    let j = right;\n\n    swap(array, left, k);\n    if (compare(array[right], t) > 0) swap(array, left, right);\n\n    while (i < j) {\n      swap(array, i, j), ++i, --j;\n      while (compare(array[i], t) < 0) ++i;\n      while (compare(array[j], t) > 0) --j;\n    }\n\n    if (compare(array[left], t) === 0) swap(array, left, j);\n    else ++j, swap(array, j, right);\n\n    if (j <= k) left = j + 1;\n    if (k <= j) right = j - 1;\n  }\n\n  return array;\n}\n\nfunction swap(array, i, j) {\n  const t = array[i];\n  array[i] = array[j];\n  array[j] = t;\n}\n","import ascending from \"./ascending.js\";\nimport permute from \"./permute.js\";\n\nexport default function sort(values, ...F) {\n  if (typeof values[Symbol.iterator] !== \"function\") throw new TypeError(\"values is not iterable\");\n  values = Array.from(values);\n  let [f] = F;\n  if ((f && f.length !== 2) || F.length > 1) {\n    const index = Uint32Array.from(values, (d, i) => i);\n    if (F.length > 1) {\n      F = F.map(f => values.map(f));\n      index.sort((i, j) => {\n        for (const f of F) {\n          const c = ascendingDefined(f[i], f[j]);\n          if (c) return c;\n        }\n      });\n    } else {\n      f = values.map(f);\n      index.sort((i, j) => ascendingDefined(f[i], f[j]));\n    }\n    return permute(values, index);\n  }\n  return values.sort(compareDefined(f));\n}\n\nexport function compareDefined(compare = ascending) {\n  if (compare === ascending) return ascendingDefined;\n  if (typeof compare !== \"function\") throw new TypeError(\"compare is not a function\");\n  return (a, b) => {\n    const x = compare(a, b);\n    if (x || x === 0) return x;\n    return (compare(b, b) === 0) - (compare(a, a) === 0);\n  };\n}\n\nexport function ascendingDefined(a, b) {\n  return (a == null || !(a >= a)) - (b == null || !(b >= b)) || (a < b ? -1 : a > b ? 1 : 0);\n}\n","import define, {extend} from \"./define.js\";\n\nexport function Color() {}\n\nexport var darker = 0.7;\nexport var brighter = 1 / darker;\n\nvar reI = \"\\\\s*([+-]?\\\\d+)\\\\s*\",\n    reN = \"\\\\s*([+-]?(?:\\\\d*\\\\.)?\\\\d+(?:[eE][+-]?\\\\d+)?)\\\\s*\",\n    reP = \"\\\\s*([+-]?(?:\\\\d*\\\\.)?\\\\d+(?:[eE][+-]?\\\\d+)?)%\\\\s*\",\n    reHex = /^#([0-9a-f]{3,8})$/,\n    reRgbInteger = new RegExp(`^rgb\\\\(${reI},${reI},${reI}\\\\)$`),\n    reRgbPercent = new RegExp(`^rgb\\\\(${reP},${reP},${reP}\\\\)$`),\n    reRgbaInteger = new RegExp(`^rgba\\\\(${reI},${reI},${reI},${reN}\\\\)$`),\n    reRgbaPercent = new RegExp(`^rgba\\\\(${reP},${reP},${reP},${reN}\\\\)$`),\n    reHslPercent = new RegExp(`^hsl\\\\(${reN},${reP},${reP}\\\\)$`),\n    reHslaPercent = new RegExp(`^hsla\\\\(${reN},${reP},${reP},${reN}\\\\)$`);\n\nvar named = {\n  aliceblue: 0xf0f8ff,\n  antiquewhite: 0xfaebd7,\n  aqua: 0x00ffff,\n  aquamarine: 0x7fffd4,\n  azure: 0xf0ffff,\n  beige: 0xf5f5dc,\n  bisque: 0xffe4c4,\n  black: 0x000000,\n  blanchedalmond: 0xffebcd,\n  blue: 0x0000ff,\n  blueviolet: 0x8a2be2,\n  brown: 0xa52a2a,\n  burlywood: 0xdeb887,\n  cadetblue: 0x5f9ea0,\n  chartreuse: 0x7fff00,\n  chocolate: 0xd2691e,\n  coral: 0xff7f50,\n  cornflowerblue: 0x6495ed,\n  cornsilk: 0xfff8dc,\n  crimson: 0xdc143c,\n  cyan: 0x00ffff,\n  darkblue: 0x00008b,\n  darkcyan: 0x008b8b,\n  darkgoldenrod: 0xb8860b,\n  darkgray: 0xa9a9a9,\n  darkgreen: 0x006400,\n  darkgrey: 0xa9a9a9,\n  darkkhaki: 0xbdb76b,\n  darkmagenta: 0x8b008b,\n  darkolivegreen: 0x556b2f,\n  darkorange: 0xff8c00,\n  darkorchid: 0x9932cc,\n  darkred: 0x8b0000,\n  darksalmon: 0xe9967a,\n  darkseagreen: 0x8fbc8f,\n  darkslateblue: 0x483d8b,\n  darkslategray: 0x2f4f4f,\n  darkslategrey: 0x2f4f4f,\n  darkturquoise: 0x00ced1,\n  darkviolet: 0x9400d3,\n  deeppink: 0xff1493,\n  deepskyblue: 0x00bfff,\n  dimgray: 0x696969,\n  dimgrey: 0x696969,\n  dodgerblue: 0x1e90ff,\n  firebrick: 0xb22222,\n  floralwhite: 0xfffaf0,\n  forestgreen: 0x228b22,\n  fuchsia: 0xff00ff,\n  gainsboro: 0xdcdcdc,\n  ghostwhite: 0xf8f8ff,\n  gold: 0xffd700,\n  goldenrod: 0xdaa520,\n  gray: 0x808080,\n  green: 0x008000,\n  greenyellow: 0xadff2f,\n  grey: 0x808080,\n  honeydew: 0xf0fff0,\n  hotpink: 0xff69b4,\n  indianred: 0xcd5c5c,\n  indigo: 0x4b0082,\n  ivory: 0xfffff0,\n  khaki: 0xf0e68c,\n  lavender: 0xe6e6fa,\n  lavenderblush: 0xfff0f5,\n  lawngreen: 0x7cfc00,\n  lemonchiffon: 0xfffacd,\n  lightblue: 0xadd8e6,\n  lightcoral: 0xf08080,\n  lightcyan: 0xe0ffff,\n  lightgoldenrodyellow: 0xfafad2,\n  lightgray: 0xd3d3d3,\n  lightgreen: 0x90ee90,\n  lightgrey: 0xd3d3d3,\n  lightpink: 0xffb6c1,\n  lightsalmon: 0xffa07a,\n  lightseagreen: 0x20b2aa,\n  lightskyblue: 0x87cefa,\n  lightslategray: 0x778899,\n  lightslategrey: 0x778899,\n  lightsteelblue: 0xb0c4de,\n  lightyellow: 0xffffe0,\n  lime: 0x00ff00,\n  limegreen: 0x32cd32,\n  linen: 0xfaf0e6,\n  magenta: 0xff00ff,\n  maroon: 0x800000,\n  mediumaquamarine: 0x66cdaa,\n  mediumblue: 0x0000cd,\n  mediumorchid: 0xba55d3,\n  mediumpurple: 0x9370db,\n  mediumseagreen: 0x3cb371,\n  mediumslateblue: 0x7b68ee,\n  mediumspringgreen: 0x00fa9a,\n  mediumturquoise: 0x48d1cc,\n  mediumvioletred: 0xc71585,\n  midnightblue: 0x191970,\n  mintcream: 0xf5fffa,\n  mistyrose: 0xffe4e1,\n  moccasin: 0xffe4b5,\n  navajowhite: 0xffdead,\n  navy: 0x000080,\n  oldlace: 0xfdf5e6,\n  olive: 0x808000,\n  olivedrab: 0x6b8e23,\n  orange: 0xffa500,\n  orangered: 0xff4500,\n  orchid: 0xda70d6,\n  palegoldenrod: 0xeee8aa,\n  palegreen: 0x98fb98,\n  paleturquoise: 0xafeeee,\n  palevioletred: 0xdb7093,\n  papayawhip: 0xffefd5,\n  peachpuff: 0xffdab9,\n  peru: 0xcd853f,\n  pink: 0xffc0cb,\n  plum: 0xdda0dd,\n  powderblue: 0xb0e0e6,\n  purple: 0x800080,\n  rebeccapurple: 0x663399,\n  red: 0xff0000,\n  rosybrown: 0xbc8f8f,\n  royalblue: 0x4169e1,\n  saddlebrown: 0x8b4513,\n  salmon: 0xfa8072,\n  sandybrown: 0xf4a460,\n  seagreen: 0x2e8b57,\n  seashell: 0xfff5ee,\n  sienna: 0xa0522d,\n  silver: 0xc0c0c0,\n  skyblue: 0x87ceeb,\n  slateblue: 0x6a5acd,\n  slategray: 0x708090,\n  slategrey: 0x708090,\n  snow: 0xfffafa,\n  springgreen: 0x00ff7f,\n  steelblue: 0x4682b4,\n  tan: 0xd2b48c,\n  teal: 0x008080,\n  thistle: 0xd8bfd8,\n  tomato: 0xff6347,\n  turquoise: 0x40e0d0,\n  violet: 0xee82ee,\n  wheat: 0xf5deb3,\n  white: 0xffffff,\n  whitesmoke: 0xf5f5f5,\n  yellow: 0xffff00,\n  yellowgreen: 0x9acd32\n};\n\ndefine(Color, color, {\n  copy(channels) {\n    return Object.assign(new this.constructor, this, channels);\n  },\n  displayable() {\n    return this.rgb().displayable();\n  },\n  hex: color_formatHex, // Deprecated! Use color.formatHex.\n  formatHex: color_formatHex,\n  formatHex8: color_formatHex8,\n  formatHsl: color_formatHsl,\n  formatRgb: color_formatRgb,\n  toString: color_formatRgb\n});\n\nfunction color_formatHex() {\n  return this.rgb().formatHex();\n}\n\nfunction color_formatHex8() {\n  return this.rgb().formatHex8();\n}\n\nfunction color_formatHsl() {\n  return hslConvert(this).formatHsl();\n}\n\nfunction color_formatRgb() {\n  return this.rgb().formatRgb();\n}\n\nexport default function color(format) {\n  var m, l;\n  format = (format + \"\").trim().toLowerCase();\n  return (m = reHex.exec(format)) ? (l = m[1].length, m = parseInt(m[1], 16), l === 6 ? rgbn(m) // #ff0000\n      : l === 3 ? new Rgb((m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), ((m & 0xf) << 4) | (m & 0xf), 1) // #f00\n      : l === 8 ? rgba(m >> 24 & 0xff, m >> 16 & 0xff, m >> 8 & 0xff, (m & 0xff) / 0xff) // #ff000000\n      : l === 4 ? rgba((m >> 12 & 0xf) | (m >> 8 & 0xf0), (m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), (((m & 0xf) << 4) | (m & 0xf)) / 0xff) // #f000\n      : null) // invalid hex\n      : (m = reRgbInteger.exec(format)) ? new Rgb(m[1], m[2], m[3], 1) // rgb(255, 0, 0)\n      : (m = reRgbPercent.exec(format)) ? new Rgb(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) // rgb(100%, 0%, 0%)\n      : (m = reRgbaInteger.exec(format)) ? rgba(m[1], m[2], m[3], m[4]) // rgba(255, 0, 0, 1)\n      : (m = reRgbaPercent.exec(format)) ? rgba(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) // rgb(100%, 0%, 0%, 1)\n      : (m = reHslPercent.exec(format)) ? hsla(m[1], m[2] / 100, m[3] / 100, 1) // hsl(120, 50%, 50%)\n      : (m = reHslaPercent.exec(format)) ? hsla(m[1], m[2] / 100, m[3] / 100, m[4]) // hsla(120, 50%, 50%, 1)\n      : named.hasOwnProperty(format) ? rgbn(named[format]) // eslint-disable-line no-prototype-builtins\n      : format === \"transparent\" ? new Rgb(NaN, NaN, NaN, 0)\n      : null;\n}\n\nfunction rgbn(n) {\n  return new Rgb(n >> 16 & 0xff, n >> 8 & 0xff, n & 0xff, 1);\n}\n\nfunction rgba(r, g, b, a) {\n  if (a <= 0) r = g = b = NaN;\n  return new Rgb(r, g, b, a);\n}\n\nexport function rgbConvert(o) {\n  if (!(o instanceof Color)) o = color(o);\n  if (!o) return new Rgb;\n  o = o.rgb();\n  return new Rgb(o.r, o.g, o.b, o.opacity);\n}\n\nexport function rgb(r, g, b, opacity) {\n  return arguments.length === 1 ? rgbConvert(r) : new Rgb(r, g, b, opacity == null ? 1 : opacity);\n}\n\nexport function Rgb(r, g, b, opacity) {\n  this.r = +r;\n  this.g = +g;\n  this.b = +b;\n  this.opacity = +opacity;\n}\n\ndefine(Rgb, rgb, extend(Color, {\n  brighter(k) {\n    k = k == null ? brighter : Math.pow(brighter, k);\n    return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);\n  },\n  darker(k) {\n    k = k == null ? darker : Math.pow(darker, k);\n    return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);\n  },\n  rgb() {\n    return this;\n  },\n  clamp() {\n    return new Rgb(clampi(this.r), clampi(this.g), clampi(this.b), clampa(this.opacity));\n  },\n  displayable() {\n    return (-0.5 <= this.r && this.r < 255.5)\n        && (-0.5 <= this.g && this.g < 255.5)\n        && (-0.5 <= this.b && this.b < 255.5)\n        && (0 <= this.opacity && this.opacity <= 1);\n  },\n  hex: rgb_formatHex, // Deprecated! Use color.formatHex.\n  formatHex: rgb_formatHex,\n  formatHex8: rgb_formatHex8,\n  formatRgb: rgb_formatRgb,\n  toString: rgb_formatRgb\n}));\n\nfunction rgb_formatHex() {\n  return `#${hex(this.r)}${hex(this.g)}${hex(this.b)}`;\n}\n\nfunction rgb_formatHex8() {\n  return `#${hex(this.r)}${hex(this.g)}${hex(this.b)}${hex((isNaN(this.opacity) ? 1 : this.opacity) * 255)}`;\n}\n\nfunction rgb_formatRgb() {\n  const a = clampa(this.opacity);\n  return `${a === 1 ? \"rgb(\" : \"rgba(\"}${clampi(this.r)}, ${clampi(this.g)}, ${clampi(this.b)}${a === 1 ? \")\" : `, ${a})`}`;\n}\n\nfunction clampa(opacity) {\n  return isNaN(opacity) ? 1 : Math.max(0, Math.min(1, opacity));\n}\n\nfunction clampi(value) {\n  return Math.max(0, Math.min(255, Math.round(value) || 0));\n}\n\nfunction hex(value) {\n  value = clampi(value);\n  return (value < 16 ? \"0\" : \"\") + value.toString(16);\n}\n\nfunction hsla(h, s, l, a) {\n  if (a <= 0) h = s = l = NaN;\n  else if (l <= 0 || l >= 1) h = s = NaN;\n  else if (s <= 0) h = NaN;\n  return new Hsl(h, s, l, a);\n}\n\nexport function hslConvert(o) {\n  if (o instanceof Hsl) return new Hsl(o.h, o.s, o.l, o.opacity);\n  if (!(o instanceof Color)) o = color(o);\n  if (!o) return new Hsl;\n  if (o instanceof Hsl) return o;\n  o = o.rgb();\n  var r = o.r / 255,\n      g = o.g / 255,\n      b = o.b / 255,\n      min = Math.min(r, g, b),\n      max = Math.max(r, g, b),\n      h = NaN,\n      s = max - min,\n      l = (max + min) / 2;\n  if (s) {\n    if (r === max) h = (g - b) / s + (g < b) * 6;\n    else if (g === max) h = (b - r) / s + 2;\n    else h = (r - g) / s + 4;\n    s /= l < 0.5 ? max + min : 2 - max - min;\n    h *= 60;\n  } else {\n    s = l > 0 && l < 1 ? 0 : h;\n  }\n  return new Hsl(h, s, l, o.opacity);\n}\n\nexport function hsl(h, s, l, opacity) {\n  return arguments.length === 1 ? hslConvert(h) : new Hsl(h, s, l, opacity == null ? 1 : opacity);\n}\n\nfunction Hsl(h, s, l, opacity) {\n  this.h = +h;\n  this.s = +s;\n  this.l = +l;\n  this.opacity = +opacity;\n}\n\ndefine(Hsl, hsl, extend(Color, {\n  brighter(k) {\n    k = k == null ? brighter : Math.pow(brighter, k);\n    return new Hsl(this.h, this.s, this.l * k, this.opacity);\n  },\n  darker(k) {\n    k = k == null ? darker : Math.pow(darker, k);\n    return new Hsl(this.h, this.s, this.l * k, this.opacity);\n  },\n  rgb() {\n    var h = this.h % 360 + (this.h < 0) * 360,\n        s = isNaN(h) || isNaN(this.s) ? 0 : this.s,\n        l = this.l,\n        m2 = l + (l < 0.5 ? l : 1 - l) * s,\n        m1 = 2 * l - m2;\n    return new Rgb(\n      hsl2rgb(h >= 240 ? h - 240 : h + 120, m1, m2),\n      hsl2rgb(h, m1, m2),\n      hsl2rgb(h < 120 ? h + 240 : h - 120, m1, m2),\n      this.opacity\n    );\n  },\n  clamp() {\n    return new Hsl(clamph(this.h), clampt(this.s), clampt(this.l), clampa(this.opacity));\n  },\n  displayable() {\n    return (0 <= this.s && this.s <= 1 || isNaN(this.s))\n        && (0 <= this.l && this.l <= 1)\n        && (0 <= this.opacity && this.opacity <= 1);\n  },\n  formatHsl() {\n    const a = clampa(this.opacity);\n    return `${a === 1 ? \"hsl(\" : \"hsla(\"}${clamph(this.h)}, ${clampt(this.s) * 100}%, ${clampt(this.l) * 100}%${a === 1 ? \")\" : `, ${a})`}`;\n  }\n}));\n\nfunction clamph(value) {\n  value = (value || 0) % 360;\n  return value < 0 ? value + 360 : value;\n}\n\nfunction clampt(value) {\n  return Math.max(0, Math.min(1, value || 0));\n}\n\n/* From FvD 13.37, CSS Color Module Level 3 */\nfunction hsl2rgb(h, m1, m2) {\n  return (h < 60 ? m1 + (m2 - m1) * h / 60\n      : h < 180 ? m2\n      : h < 240 ? m1 + (m2 - m1) * (240 - h) / 60\n      : m1) * 255;\n}\n","export default function(constructor, factory, prototype) {\n  constructor.prototype = factory.prototype = prototype;\n  prototype.constructor = constructor;\n}\n\nexport function extend(parent, definition) {\n  var prototype = Object.create(parent.prototype);\n  for (var key in definition) prototype[key] = definition[key];\n  return prototype;\n}\n","import define, {extend} from \"./define.js\";\nimport {Color, rgbConvert, Rgb} from \"./color.js\";\nimport {degrees, radians} from \"./math.js\";\n\n// https://observablehq.com/@mbostock/lab-and-rgb\nconst K = 18,\n    Xn = 0.96422,\n    Yn = 1,\n    Zn = 0.82521,\n    t0 = 4 / 29,\n    t1 = 6 / 29,\n    t2 = 3 * t1 * t1,\n    t3 = t1 * t1 * t1;\n\nfunction labConvert(o) {\n  if (o instanceof Lab) return new Lab(o.l, o.a, o.b, o.opacity);\n  if (o instanceof Hcl) return hcl2lab(o);\n  if (!(o instanceof Rgb)) o = rgbConvert(o);\n  var r = rgb2lrgb(o.r),\n      g = rgb2lrgb(o.g),\n      b = rgb2lrgb(o.b),\n      y = xyz2lab((0.2225045 * r + 0.7168786 * g + 0.0606169 * b) / Yn), x, z;\n  if (r === g && g === b) x = z = y; else {\n    x = xyz2lab((0.4360747 * r + 0.3850649 * g + 0.1430804 * b) / Xn);\n    z = xyz2lab((0.0139322 * r + 0.0971045 * g + 0.7141733 * b) / Zn);\n  }\n  return new Lab(116 * y - 16, 500 * (x - y), 200 * (y - z), o.opacity);\n}\n\nexport function gray(l, opacity) {\n  return new Lab(l, 0, 0, opacity == null ? 1 : opacity);\n}\n\nexport default function lab(l, a, b, opacity) {\n  return arguments.length === 1 ? labConvert(l) : new Lab(l, a, b, opacity == null ? 1 : opacity);\n}\n\nexport function Lab(l, a, b, opacity) {\n  this.l = +l;\n  this.a = +a;\n  this.b = +b;\n  this.opacity = +opacity;\n}\n\ndefine(Lab, lab, extend(Color, {\n  brighter(k) {\n    return new Lab(this.l + K * (k == null ? 1 : k), this.a, this.b, this.opacity);\n  },\n  darker(k) {\n    return new Lab(this.l - K * (k == null ? 1 : k), this.a, this.b, this.opacity);\n  },\n  rgb() {\n    var y = (this.l + 16) / 116,\n        x = isNaN(this.a) ? y : y + this.a / 500,\n        z = isNaN(this.b) ? y : y - this.b / 200;\n    x = Xn * lab2xyz(x);\n    y = Yn * lab2xyz(y);\n    z = Zn * lab2xyz(z);\n    return new Rgb(\n      lrgb2rgb( 3.1338561 * x - 1.6168667 * y - 0.4906146 * z),\n      lrgb2rgb(-0.9787684 * x + 1.9161415 * y + 0.0334540 * z),\n      lrgb2rgb( 0.0719453 * x - 0.2289914 * y + 1.4052427 * z),\n      this.opacity\n    );\n  }\n}));\n\nfunction xyz2lab(t) {\n  return t > t3 ? Math.pow(t, 1 / 3) : t / t2 + t0;\n}\n\nfunction lab2xyz(t) {\n  return t > t1 ? t * t * t : t2 * (t - t0);\n}\n\nfunction lrgb2rgb(x) {\n  return 255 * (x <= 0.0031308 ? 12.92 * x : 1.055 * Math.pow(x, 1 / 2.4) - 0.055);\n}\n\nfunction rgb2lrgb(x) {\n  return (x /= 255) <= 0.04045 ? x / 12.92 : Math.pow((x + 0.055) / 1.055, 2.4);\n}\n\nfunction hclConvert(o) {\n  if (o instanceof Hcl) return new Hcl(o.h, o.c, o.l, o.opacity);\n  if (!(o instanceof Lab)) o = labConvert(o);\n  if (o.a === 0 && o.b === 0) return new Hcl(NaN, 0 < o.l && o.l < 100 ? 0 : NaN, o.l, o.opacity);\n  var h = Math.atan2(o.b, o.a) * degrees;\n  return new Hcl(h < 0 ? h + 360 : h, Math.sqrt(o.a * o.a + o.b * o.b), o.l, o.opacity);\n}\n\nexport function lch(l, c, h, opacity) {\n  return arguments.length === 1 ? hclConvert(l) : new Hcl(h, c, l, opacity == null ? 1 : opacity);\n}\n\nexport function hcl(h, c, l, opacity) {\n  return arguments.length === 1 ? hclConvert(h) : new Hcl(h, c, l, opacity == null ? 1 : opacity);\n}\n\nexport function Hcl(h, c, l, opacity) {\n  this.h = +h;\n  this.c = +c;\n  this.l = +l;\n  this.opacity = +opacity;\n}\n\nfunction hcl2lab(o) {\n  if (isNaN(o.h)) return new Lab(o.l, 0, 0, o.opacity);\n  var h = o.h * radians;\n  return new Lab(o.l, Math.cos(h) * o.c, Math.sin(h) * o.c, o.opacity);\n}\n\ndefine(Hcl, hcl, extend(Color, {\n  brighter(k) {\n    return new Hcl(this.h, this.c, this.l + K * (k == null ? 1 : k), this.opacity);\n  },\n  darker(k) {\n    return new Hcl(this.h, this.c, this.l - K * (k == null ? 1 : k), this.opacity);\n  },\n  rgb() {\n    return hcl2lab(this).rgb();\n  }\n}));\n","export const radians = Math.PI / 180;\nexport const degrees = 180 / Math.PI;\n","var noop = {value: () => {}};\n\nfunction dispatch() {\n  for (var i = 0, n = arguments.length, _ = {}, t; i < n; ++i) {\n    if (!(t = arguments[i] + \"\") || (t in _) || /[\\s.]/.test(t)) throw new Error(\"illegal type: \" + t);\n    _[t] = [];\n  }\n  return new Dispatch(_);\n}\n\nfunction Dispatch(_) {\n  this._ = _;\n}\n\nfunction parseTypenames(typenames, types) {\n  return typenames.trim().split(/^|\\s+/).map(function(t) {\n    var name = \"\", i = t.indexOf(\".\");\n    if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);\n    if (t && !types.hasOwnProperty(t)) throw new Error(\"unknown type: \" + t);\n    return {type: t, name: name};\n  });\n}\n\nDispatch.prototype = dispatch.prototype = {\n  constructor: Dispatch,\n  on: function(typename, callback) {\n    var _ = this._,\n        T = parseTypenames(typename + \"\", _),\n        t,\n        i = -1,\n        n = T.length;\n\n    // If no callback was specified, return the callback of the given type and name.\n    if (arguments.length < 2) {\n      while (++i < n) if ((t = (typename = T[i]).type) && (t = get(_[t], typename.name))) return t;\n      return;\n    }\n\n    // If a type was specified, set the callback for the given type and name.\n    // Otherwise, if a null callback was specified, remove callbacks of the given name.\n    if (callback != null && typeof callback !== \"function\") throw new Error(\"invalid callback: \" + callback);\n    while (++i < n) {\n      if (t = (typename = T[i]).type) _[t] = set(_[t], typename.name, callback);\n      else if (callback == null) for (t in _) _[t] = set(_[t], typename.name, null);\n    }\n\n    return this;\n  },\n  copy: function() {\n    var copy = {}, _ = this._;\n    for (var t in _) copy[t] = _[t].slice();\n    return new Dispatch(copy);\n  },\n  call: function(type, that) {\n    if ((n = arguments.length - 2) > 0) for (var args = new Array(n), i = 0, n, t; i < n; ++i) args[i] = arguments[i + 2];\n    if (!this._.hasOwnProperty(type)) throw new Error(\"unknown type: \" + type);\n    for (t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);\n  },\n  apply: function(type, that, args) {\n    if (!this._.hasOwnProperty(type)) throw new Error(\"unknown type: \" + type);\n    for (var t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);\n  }\n};\n\nfunction get(type, name) {\n  for (var i = 0, n = type.length, c; i < n; ++i) {\n    if ((c = type[i]).name === name) {\n      return c.value;\n    }\n  }\n}\n\nfunction set(type, name, callback) {\n  for (var i = 0, n = type.length; i < n; ++i) {\n    if (type[i].name === name) {\n      type[i] = noop, type = type.slice(0, i).concat(type.slice(i + 1));\n      break;\n    }\n  }\n  if (callback != null) type.push({name: name, value: callback});\n  return type;\n}\n\nexport default dispatch;\n","import {select} from \"d3-selection\";\nimport noevent, {nonpassivecapture} from \"./noevent.js\";\n\nexport default function(view) {\n  var root = view.document.documentElement,\n      selection = select(view).on(\"dragstart.drag\", noevent, nonpassivecapture);\n  if (\"onselectstart\" in root) {\n    selection.on(\"selectstart.drag\", noevent, nonpassivecapture);\n  } else {\n    root.__noselect = root.style.MozUserSelect;\n    root.style.MozUserSelect = \"none\";\n  }\n}\n\nexport function yesdrag(view, noclick) {\n  var root = view.document.documentElement,\n      selection = select(view).on(\"dragstart.drag\", null);\n  if (noclick) {\n    selection.on(\"click.drag\", noevent, nonpassivecapture);\n    setTimeout(function() { selection.on(\"click.drag\", null); }, 0);\n  }\n  if (\"onselectstart\" in root) {\n    selection.on(\"selectstart.drag\", null);\n  } else {\n    root.style.MozUserSelect = root.__noselect;\n    delete root.__noselect;\n  }\n}\n","// These are typically used in conjunction with noevent to ensure that we can\n// preventDefault on the event.\nexport const nonpassive = {passive: false};\nexport const nonpassivecapture = {capture: true, passive: false};\n\nexport function nopropagation(event) {\n  event.stopImmediatePropagation();\n}\n\nexport default function(event) {\n  event.preventDefault();\n  event.stopImmediatePropagation();\n}\n","import dsv from \"./dsv.js\";\n\nvar csv = dsv(\",\");\n\nexport var csvParse = csv.parse;\nexport var csvParseRows = csv.parseRows;\nexport var csvFormat = csv.format;\nexport var csvFormatBody = csv.formatBody;\nexport var csvFormatRows = csv.formatRows;\nexport var csvFormatRow = csv.formatRow;\nexport var csvFormatValue = csv.formatValue;\n","var EOL = {},\n    EOF = {},\n    QUOTE = 34,\n    NEWLINE = 10,\n    RETURN = 13;\n\nfunction objectConverter(columns) {\n  return new Function(\"d\", \"return {\" + columns.map(function(name, i) {\n    return JSON.stringify(name) + \": d[\" + i + \"] || \\\"\\\"\";\n  }).join(\",\") + \"}\");\n}\n\nfunction customConverter(columns, f) {\n  var object = objectConverter(columns);\n  return function(row, i) {\n    return f(object(row), i, columns);\n  };\n}\n\n// Compute unique columns in order of discovery.\nfunction inferColumns(rows) {\n  var columnSet = Object.create(null),\n      columns = [];\n\n  rows.forEach(function(row) {\n    for (var column in row) {\n      if (!(column in columnSet)) {\n        columns.push(columnSet[column] = column);\n      }\n    }\n  });\n\n  return columns;\n}\n\nfunction pad(value, width) {\n  var s = value + \"\", length = s.length;\n  return length < width ? new Array(width - length + 1).join(0) + s : s;\n}\n\nfunction formatYear(year) {\n  return year < 0 ? \"-\" + pad(-year, 6)\n    : year > 9999 ? \"+\" + pad(year, 6)\n    : pad(year, 4);\n}\n\nfunction formatDate(date) {\n  var hours = date.getUTCHours(),\n      minutes = date.getUTCMinutes(),\n      seconds = date.getUTCSeconds(),\n      milliseconds = date.getUTCMilliseconds();\n  return isNaN(date) ? \"Invalid Date\"\n      : formatYear(date.getUTCFullYear(), 4) + \"-\" + pad(date.getUTCMonth() + 1, 2) + \"-\" + pad(date.getUTCDate(), 2)\n      + (milliseconds ? \"T\" + pad(hours, 2) + \":\" + pad(minutes, 2) + \":\" + pad(seconds, 2) + \".\" + pad(milliseconds, 3) + \"Z\"\n      : seconds ? \"T\" + pad(hours, 2) + \":\" + pad(minutes, 2) + \":\" + pad(seconds, 2) + \"Z\"\n      : minutes || hours ? \"T\" + pad(hours, 2) + \":\" + pad(minutes, 2) + \"Z\"\n      : \"\");\n}\n\nexport default function(delimiter) {\n  var reFormat = new RegExp(\"[\\\"\" + delimiter + \"\\n\\r]\"),\n      DELIMITER = delimiter.charCodeAt(0);\n\n  function parse(text, f) {\n    var convert, columns, rows = parseRows(text, function(row, i) {\n      if (convert) return convert(row, i - 1);\n      columns = row, convert = f ? customConverter(row, f) : objectConverter(row);\n    });\n    rows.columns = columns || [];\n    return rows;\n  }\n\n  function parseRows(text, f) {\n    var rows = [], // output rows\n        N = text.length,\n        I = 0, // current character index\n        n = 0, // current line number\n        t, // current token\n        eof = N <= 0, // current token followed by EOF?\n        eol = false; // current token followed by EOL?\n\n    // Strip the trailing newline.\n    if (text.charCodeAt(N - 1) === NEWLINE) --N;\n    if (text.charCodeAt(N - 1) === RETURN) --N;\n\n    function token() {\n      if (eof) return EOF;\n      if (eol) return eol = false, EOL;\n\n      // Unescape quotes.\n      var i, j = I, c;\n      if (text.charCodeAt(j) === QUOTE) {\n        while (I++ < N && text.charCodeAt(I) !== QUOTE || text.charCodeAt(++I) === QUOTE);\n        if ((i = I) >= N) eof = true;\n        else if ((c = text.charCodeAt(I++)) === NEWLINE) eol = true;\n        else if (c === RETURN) { eol = true; if (text.charCodeAt(I) === NEWLINE) ++I; }\n        return text.slice(j + 1, i - 1).replace(/\"\"/g, \"\\\"\");\n      }\n\n      // Find next delimiter or newline.\n      while (I < N) {\n        if ((c = text.charCodeAt(i = I++)) === NEWLINE) eol = true;\n        else if (c === RETURN) { eol = true; if (text.charCodeAt(I) === NEWLINE) ++I; }\n        else if (c !== DELIMITER) continue;\n        return text.slice(j, i);\n      }\n\n      // Return last token before EOF.\n      return eof = true, text.slice(j, N);\n    }\n\n    while ((t = token()) !== EOF) {\n      var row = [];\n      while (t !== EOL && t !== EOF) row.push(t), t = token();\n      if (f && (row = f(row, n++)) == null) continue;\n      rows.push(row);\n    }\n\n    return rows;\n  }\n\n  function preformatBody(rows, columns) {\n    return rows.map(function(row) {\n      return columns.map(function(column) {\n        return formatValue(row[column]);\n      }).join(delimiter);\n    });\n  }\n\n  function format(rows, columns) {\n    if (columns == null) columns = inferColumns(rows);\n    return [columns.map(formatValue).join(delimiter)].concat(preformatBody(rows, columns)).join(\"\\n\");\n  }\n\n  function formatBody(rows, columns) {\n    if (columns == null) columns = inferColumns(rows);\n    return preformatBody(rows, columns).join(\"\\n\");\n  }\n\n  function formatRows(rows) {\n    return rows.map(formatRow).join(\"\\n\");\n  }\n\n  function formatRow(row) {\n    return row.map(formatValue).join(delimiter);\n  }\n\n  function formatValue(value) {\n    return value == null ? \"\"\n        : value instanceof Date ? formatDate(value)\n        : reFormat.test(value += \"\") ? \"\\\"\" + value.replace(/\"/g, \"\\\"\\\"\") + \"\\\"\"\n        : value;\n  }\n\n  return {\n    parse: parse,\n    parseRows: parseRows,\n    format: format,\n    formatBody: formatBody,\n    formatRows: formatRows,\n    formatRow: formatRow,\n    formatValue: formatValue\n  };\n}\n","import dsv from \"./dsv.js\";\n\nvar tsv = dsv(\"\\t\");\n\nexport var tsvParse = tsv.parse;\nexport var tsvParseRows = tsv.parseRows;\nexport var tsvFormat = tsv.format;\nexport var tsvFormatBody = tsv.formatBody;\nexport var tsvFormatRows = tsv.formatRows;\nexport var tsvFormatRow = tsv.formatRow;\nexport var tsvFormatValue = tsv.formatValue;\n","export function cubicIn(t) {\n  return t * t * t;\n}\n\nexport function cubicOut(t) {\n  return --t * t * t + 1;\n}\n\nexport function cubicInOut(t) {\n  return ((t *= 2) <= 1 ? t * t * t : (t -= 2) * t * t + 2) / 2;\n}\n","import {csvParse, dsvFormat, tsvParse} from \"d3-dsv\";\nimport text from \"./text.js\";\n\nfunction dsvParse(parse) {\n  return function(input, init, row) {\n    if (arguments.length === 2 && typeof init === \"function\") row = init, init = undefined;\n    return text(input, init).then(function(response) {\n      return parse(response, row);\n    });\n  };\n}\n\nexport default function dsv(delimiter, input, init, row) {\n  if (arguments.length === 3 && typeof init === \"function\") row = init, init = undefined;\n  var format = dsvFormat(delimiter);\n  return text(input, init).then(function(response) {\n    return format.parse(response, row);\n  });\n}\n\nexport var csv = dsvParse(csvParse);\nexport var tsv = dsvParse(tsvParse);\n","function responseJson(response) {\n  if (!response.ok) throw new Error(response.status + \" \" + response.statusText);\n  if (response.status === 204 || response.status === 205) return;\n  return response.json();\n}\n\nexport default function(input, init) {\n  return fetch(input, init).then(responseJson);\n}\n","function responseText(response) {\n  if (!response.ok) throw new Error(response.status + \" \" + response.statusText);\n  return response.text();\n}\n\nexport default function(input, init) {\n  return fetch(input, init).then(responseText);\n}\n","import formatLocale from \"./locale.js\";\n\nvar locale;\nexport var format;\nexport var formatPrefix;\n\ndefaultLocale({\n  thousands: \",\",\n  grouping: [3],\n  currency: [\"$\", \"\"]\n});\n\nexport default function defaultLocale(definition) {\n  locale = formatLocale(definition);\n  format = locale.format;\n  formatPrefix = locale.formatPrefix;\n  return locale;\n}\n","import {formatDecimalParts} from \"./formatDecimal.js\";\n\nexport default function(x) {\n  return x = formatDecimalParts(Math.abs(x)), x ? x[1] : NaN;\n}\n","export default function(x) {\n  return Math.abs(x = Math.round(x)) >= 1e21\n      ? x.toLocaleString(\"en\").replace(/,/g, \"\")\n      : x.toString(10);\n}\n\n// Computes the decimal coefficient and exponent of the specified number x with\n// significant digits p, where x is positive and p is in [1, 21] or undefined.\n// For example, formatDecimalParts(1.23) returns [\"123\", 0].\nexport function formatDecimalParts(x, p) {\n  if ((i = (x = p ? x.toExponential(p - 1) : x.toExponential()).indexOf(\"e\")) < 0) return null; // NaN, ±Infinity\n  var i, coefficient = x.slice(0, i);\n\n  // The string returned by toExponential either has the form \\d\\.\\d+e[-+]\\d+\n  // (e.g., 1.2e+3) or the form \\de[-+]\\d+ (e.g., 1e+3).\n  return [\n    coefficient.length > 1 ? coefficient[0] + coefficient.slice(2) : coefficient,\n    +x.slice(i + 1)\n  ];\n}\n","export default function(grouping, thousands) {\n  return function(value, width) {\n    var i = value.length,\n        t = [],\n        j = 0,\n        g = grouping[0],\n        length = 0;\n\n    while (i > 0 && g > 0) {\n      if (length + g + 1 > width) g = Math.max(1, width - length);\n      t.push(value.substring(i -= g, i + g));\n      if ((length += g + 1) > width) break;\n      g = grouping[j = (j + 1) % grouping.length];\n    }\n\n    return t.reverse().join(thousands);\n  };\n}\n","export default function(numerals) {\n  return function(value) {\n    return value.replace(/[0-9]/g, function(i) {\n      return numerals[+i];\n    });\n  };\n}\n","import {formatDecimalParts} from \"./formatDecimal.js\";\n\nexport var prefixExponent;\n\nexport default function(x, p) {\n  var d = formatDecimalParts(x, p);\n  if (!d) return x + \"\";\n  var coefficient = d[0],\n      exponent = d[1],\n      i = exponent - (prefixExponent = Math.max(-8, Math.min(8, Math.floor(exponent / 3))) * 3) + 1,\n      n = coefficient.length;\n  return i === n ? coefficient\n      : i > n ? coefficient + new Array(i - n + 1).join(\"0\")\n      : i > 0 ? coefficient.slice(0, i) + \".\" + coefficient.slice(i)\n      : \"0.\" + new Array(1 - i).join(\"0\") + formatDecimalParts(x, Math.max(0, p + i - 1))[0]; // less than 1y!\n}\n","import {formatDecimalParts} from \"./formatDecimal.js\";\n\nexport default function(x, p) {\n  var d = formatDecimalParts(x, p);\n  if (!d) return x + \"\";\n  var coefficient = d[0],\n      exponent = d[1];\n  return exponent < 0 ? \"0.\" + new Array(-exponent).join(\"0\") + coefficient\n      : coefficient.length > exponent + 1 ? coefficient.slice(0, exponent + 1) + \".\" + coefficient.slice(exponent + 1)\n      : coefficient + new Array(exponent - coefficient.length + 2).join(\"0\");\n}\n","// [[fill]align][sign][symbol][0][width][,][.precision][~][type]\nvar re = /^(?:(.)?([<>=^]))?([+\\-( ])?([$#])?(0)?(\\d+)?(,)?(\\.\\d+)?(~)?([a-z%])?$/i;\n\nexport default function formatSpecifier(specifier) {\n  if (!(match = re.exec(specifier))) throw new Error(\"invalid format: \" + specifier);\n  var match;\n  return new FormatSpecifier({\n    fill: match[1],\n    align: match[2],\n    sign: match[3],\n    symbol: match[4],\n    zero: match[5],\n    width: match[6],\n    comma: match[7],\n    precision: match[8] && match[8].slice(1),\n    trim: match[9],\n    type: match[10]\n  });\n}\n\nformatSpecifier.prototype = FormatSpecifier.prototype; // instanceof\n\nexport function FormatSpecifier(specifier) {\n  this.fill = specifier.fill === undefined ? \" \" : specifier.fill + \"\";\n  this.align = specifier.align === undefined ? \">\" : specifier.align + \"\";\n  this.sign = specifier.sign === undefined ? \"-\" : specifier.sign + \"\";\n  this.symbol = specifier.symbol === undefined ? \"\" : specifier.symbol + \"\";\n  this.zero = !!specifier.zero;\n  this.width = specifier.width === undefined ? undefined : +specifier.width;\n  this.comma = !!specifier.comma;\n  this.precision = specifier.precision === undefined ? undefined : +specifier.precision;\n  this.trim = !!specifier.trim;\n  this.type = specifier.type === undefined ? \"\" : specifier.type + \"\";\n}\n\nFormatSpecifier.prototype.toString = function() {\n  return this.fill\n      + this.align\n      + this.sign\n      + this.symbol\n      + (this.zero ? \"0\" : \"\")\n      + (this.width === undefined ? \"\" : Math.max(1, this.width | 0))\n      + (this.comma ? \",\" : \"\")\n      + (this.precision === undefined ? \"\" : \".\" + Math.max(0, this.precision | 0))\n      + (this.trim ? \"~\" : \"\")\n      + this.type;\n};\n","// Trims insignificant zeros, e.g., replaces 1.2000k with 1.2k.\nexport default function(s) {\n  out: for (var n = s.length, i = 1, i0 = -1, i1; i < n; ++i) {\n    switch (s[i]) {\n      case \".\": i0 = i1 = i; break;\n      case \"0\": if (i0 === 0) i0 = i; i1 = i; break;\n      default: if (!+s[i]) break out; if (i0 > 0) i0 = 0; break;\n    }\n  }\n  return i0 > 0 ? s.slice(0, i0) + s.slice(i1 + 1) : s;\n}\n","import formatDecimal from \"./formatDecimal.js\";\nimport formatPrefixAuto from \"./formatPrefixAuto.js\";\nimport formatRounded from \"./formatRounded.js\";\n\nexport default {\n  \"%\": (x, p) => (x * 100).toFixed(p),\n  \"b\": (x) => Math.round(x).toString(2),\n  \"c\": (x) => x + \"\",\n  \"d\": formatDecimal,\n  \"e\": (x, p) => x.toExponential(p),\n  \"f\": (x, p) => x.toFixed(p),\n  \"g\": (x, p) => x.toPrecision(p),\n  \"o\": (x) => Math.round(x).toString(8),\n  \"p\": (x, p) => formatRounded(x * 100, p),\n  \"r\": formatRounded,\n  \"s\": formatPrefixAuto,\n  \"X\": (x) => Math.round(x).toString(16).toUpperCase(),\n  \"x\": (x) => Math.round(x).toString(16)\n};\n","export default function(x) {\n  return x;\n}\n","import exponent from \"./exponent.js\";\nimport formatGroup from \"./formatGroup.js\";\nimport formatNumerals from \"./formatNumerals.js\";\nimport formatSpecifier from \"./formatSpecifier.js\";\nimport formatTrim from \"./formatTrim.js\";\nimport formatTypes from \"./formatTypes.js\";\nimport {prefixExponent} from \"./formatPrefixAuto.js\";\nimport identity from \"./identity.js\";\n\nvar map = Array.prototype.map,\n    prefixes = [\"y\",\"z\",\"a\",\"f\",\"p\",\"n\",\"µ\",\"m\",\"\",\"k\",\"M\",\"G\",\"T\",\"P\",\"E\",\"Z\",\"Y\"];\n\nexport default function(locale) {\n  var group = locale.grouping === undefined || locale.thousands === undefined ? identity : formatGroup(map.call(locale.grouping, Number), locale.thousands + \"\"),\n      currencyPrefix = locale.currency === undefined ? \"\" : locale.currency[0] + \"\",\n      currencySuffix = locale.currency === undefined ? \"\" : locale.currency[1] + \"\",\n      decimal = locale.decimal === undefined ? \".\" : locale.decimal + \"\",\n      numerals = locale.numerals === undefined ? identity : formatNumerals(map.call(locale.numerals, String)),\n      percent = locale.percent === undefined ? \"%\" : locale.percent + \"\",\n      minus = locale.minus === undefined ? \"−\" : locale.minus + \"\",\n      nan = locale.nan === undefined ? \"NaN\" : locale.nan + \"\";\n\n  function newFormat(specifier) {\n    specifier = formatSpecifier(specifier);\n\n    var fill = specifier.fill,\n        align = specifier.align,\n        sign = specifier.sign,\n        symbol = specifier.symbol,\n        zero = specifier.zero,\n        width = specifier.width,\n        comma = specifier.comma,\n        precision = specifier.precision,\n        trim = specifier.trim,\n        type = specifier.type;\n\n    // The \"n\" type is an alias for \",g\".\n    if (type === \"n\") comma = true, type = \"g\";\n\n    // The \"\" type, and any invalid type, is an alias for \".12~g\".\n    else if (!formatTypes[type]) precision === undefined && (precision = 12), trim = true, type = \"g\";\n\n    // If zero fill is specified, padding goes after sign and before digits.\n    if (zero || (fill === \"0\" && align === \"=\")) zero = true, fill = \"0\", align = \"=\";\n\n    // Compute the prefix and suffix.\n    // For SI-prefix, the suffix is lazily computed.\n    var prefix = symbol === \"$\" ? currencyPrefix : symbol === \"#\" && /[boxX]/.test(type) ? \"0\" + type.toLowerCase() : \"\",\n        suffix = symbol === \"$\" ? currencySuffix : /[%p]/.test(type) ? percent : \"\";\n\n    // What format function should we use?\n    // Is this an integer type?\n    // Can this type generate exponential notation?\n    var formatType = formatTypes[type],\n        maybeSuffix = /[defgprs%]/.test(type);\n\n    // Set the default precision if not specified,\n    // or clamp the specified precision to the supported range.\n    // For significant precision, it must be in [1, 21].\n    // For fixed precision, it must be in [0, 20].\n    precision = precision === undefined ? 6\n        : /[gprs]/.test(type) ? Math.max(1, Math.min(21, precision))\n        : Math.max(0, Math.min(20, precision));\n\n    function format(value) {\n      var valuePrefix = prefix,\n          valueSuffix = suffix,\n          i, n, c;\n\n      if (type === \"c\") {\n        valueSuffix = formatType(value) + valueSuffix;\n        value = \"\";\n      } else {\n        value = +value;\n\n        // Determine the sign. -0 is not less than 0, but 1 / -0 is!\n        var valueNegative = value < 0 || 1 / value < 0;\n\n        // Perform the initial formatting.\n        value = isNaN(value) ? nan : formatType(Math.abs(value), precision);\n\n        // Trim insignificant zeros.\n        if (trim) value = formatTrim(value);\n\n        // If a negative value rounds to zero after formatting, and no explicit positive sign is requested, hide the sign.\n        if (valueNegative && +value === 0 && sign !== \"+\") valueNegative = false;\n\n        // Compute the prefix and suffix.\n        valuePrefix = (valueNegative ? (sign === \"(\" ? sign : minus) : sign === \"-\" || sign === \"(\" ? \"\" : sign) + valuePrefix;\n        valueSuffix = (type === \"s\" ? prefixes[8 + prefixExponent / 3] : \"\") + valueSuffix + (valueNegative && sign === \"(\" ? \")\" : \"\");\n\n        // Break the formatted value into the integer “value” part that can be\n        // grouped, and fractional or exponential “suffix” part that is not.\n        if (maybeSuffix) {\n          i = -1, n = value.length;\n          while (++i < n) {\n            if (c = value.charCodeAt(i), 48 > c || c > 57) {\n              valueSuffix = (c === 46 ? decimal + value.slice(i + 1) : value.slice(i)) + valueSuffix;\n              value = value.slice(0, i);\n              break;\n            }\n          }\n        }\n      }\n\n      // If the fill character is not \"0\", grouping is applied before padding.\n      if (comma && !zero) value = group(value, Infinity);\n\n      // Compute the padding.\n      var length = valuePrefix.length + value.length + valueSuffix.length,\n          padding = length < width ? new Array(width - length + 1).join(fill) : \"\";\n\n      // If the fill character is \"0\", grouping is applied after padding.\n      if (comma && zero) value = group(padding + value, padding.length ? width - valueSuffix.length : Infinity), padding = \"\";\n\n      // Reconstruct the final output based on the desired alignment.\n      switch (align) {\n        case \"<\": value = valuePrefix + value + valueSuffix + padding; break;\n        case \"=\": value = valuePrefix + padding + value + valueSuffix; break;\n        case \"^\": value = padding.slice(0, length = padding.length >> 1) + valuePrefix + value + valueSuffix + padding.slice(length); break;\n        default: value = padding + valuePrefix + value + valueSuffix; break;\n      }\n\n      return numerals(value);\n    }\n\n    format.toString = function() {\n      return specifier + \"\";\n    };\n\n    return format;\n  }\n\n  function formatPrefix(specifier, value) {\n    var f = newFormat((specifier = formatSpecifier(specifier), specifier.type = \"f\", specifier)),\n        e = Math.max(-8, Math.min(8, Math.floor(exponent(value) / 3))) * 3,\n        k = Math.pow(10, -e),\n        prefix = prefixes[8 + e / 3];\n    return function(value) {\n      return f(k * value) + prefix;\n    };\n  }\n\n  return {\n    format: newFormat,\n    formatPrefix: formatPrefix\n  };\n}\n","export function basis(t1, v0, v1, v2, v3) {\n  var t2 = t1 * t1, t3 = t2 * t1;\n  return ((1 - 3 * t1 + 3 * t2 - t3) * v0\n      + (4 - 6 * t2 + 3 * t3) * v1\n      + (1 + 3 * t1 + 3 * t2 - 3 * t3) * v2\n      + t3 * v3) / 6;\n}\n\nexport default function(values) {\n  var n = values.length - 1;\n  return function(t) {\n    var i = t <= 0 ? (t = 0) : t >= 1 ? (t = 1, n - 1) : Math.floor(t * n),\n        v1 = values[i],\n        v2 = values[i + 1],\n        v0 = i > 0 ? values[i - 1] : 2 * v1 - v2,\n        v3 = i < n - 1 ? values[i + 2] : 2 * v2 - v1;\n    return basis((t - i / n) * n, v0, v1, v2, v3);\n  };\n}\n","import {basis} from \"./basis.js\";\n\nexport default function(values) {\n  var n = values.length;\n  return function(t) {\n    var i = Math.floor(((t %= 1) < 0 ? ++t : t) * n),\n        v0 = values[(i + n - 1) % n],\n        v1 = values[i % n],\n        v2 = values[(i + 1) % n],\n        v3 = values[(i + 2) % n];\n    return basis((t - i / n) * n, v0, v1, v2, v3);\n  };\n}\n","import constant from \"./constant.js\";\n\nfunction linear(a, d) {\n  return function(t) {\n    return a + t * d;\n  };\n}\n\nfunction exponential(a, b, y) {\n  return a = Math.pow(a, y), b = Math.pow(b, y) - a, y = 1 / y, function(t) {\n    return Math.pow(a + t * b, y);\n  };\n}\n\nexport function hue(a, b) {\n  var d = b - a;\n  return d ? linear(a, d > 180 || d < -180 ? d - 360 * Math.round(d / 360) : d) : constant(isNaN(a) ? b : a);\n}\n\nexport function gamma(y) {\n  return (y = +y) === 1 ? nogamma : function(a, b) {\n    return b - a ? exponential(a, b, y) : constant(isNaN(a) ? b : a);\n  };\n}\n\nexport default function nogamma(a, b) {\n  var d = b - a;\n  return d ? linear(a, d) : constant(isNaN(a) ? b : a);\n}\n","export default x => () => x;\n","import {lab as colorLab} from \"d3-color\";\nimport color from \"./color.js\";\n\nexport default function lab(start, end) {\n  var l = color((start = colorLab(start)).l, (end = colorLab(end)).l),\n      a = color(start.a, end.a),\n      b = color(start.b, end.b),\n      opacity = color(start.opacity, end.opacity);\n  return function(t) {\n    start.l = l(t);\n    start.a = a(t);\n    start.b = b(t);\n    start.opacity = opacity(t);\n    return start + \"\";\n  };\n}\n","export default function(a, b) {\n  return a = +a, b = +b, function(t) {\n    return a * (1 - t) + b * t;\n  };\n}\n","import {rgb as colorRgb} from \"d3-color\";\nimport basis from \"./basis.js\";\nimport basisClosed from \"./basisClosed.js\";\nimport nogamma, {gamma} from \"./color.js\";\n\nexport default (function rgbGamma(y) {\n  var color = gamma(y);\n\n  function rgb(start, end) {\n    var r = color((start = colorRgb(start)).r, (end = colorRgb(end)).r),\n        g = color(start.g, end.g),\n        b = color(start.b, end.b),\n        opacity = nogamma(start.opacity, end.opacity);\n    return function(t) {\n      start.r = r(t);\n      start.g = g(t);\n      start.b = b(t);\n      start.opacity = opacity(t);\n      return start + \"\";\n    };\n  }\n\n  rgb.gamma = rgbGamma;\n\n  return rgb;\n})(1);\n\nfunction rgbSpline(spline) {\n  return function(colors) {\n    var n = colors.length,\n        r = new Array(n),\n        g = new Array(n),\n        b = new Array(n),\n        i, color;\n    for (i = 0; i < n; ++i) {\n      color = colorRgb(colors[i]);\n      r[i] = color.r || 0;\n      g[i] = color.g || 0;\n      b[i] = color.b || 0;\n    }\n    r = spline(r);\n    g = spline(g);\n    b = spline(b);\n    color.opacity = 1;\n    return function(t) {\n      color.r = r(t);\n      color.g = g(t);\n      color.b = b(t);\n      return color + \"\";\n    };\n  };\n}\n\nexport var rgbBasis = rgbSpline(basis);\nexport var rgbBasisClosed = rgbSpline(basisClosed);\n","import number from \"./number.js\";\n\nvar reA = /[-+]?(?:\\d+\\.?\\d*|\\.?\\d+)(?:[eE][-+]?\\d+)?/g,\n    reB = new RegExp(reA.source, \"g\");\n\nfunction zero(b) {\n  return function() {\n    return b;\n  };\n}\n\nfunction one(b) {\n  return function(t) {\n    return b(t) + \"\";\n  };\n}\n\nexport default function(a, b) {\n  var bi = reA.lastIndex = reB.lastIndex = 0, // scan index for next number in b\n      am, // current match in a\n      bm, // current match in b\n      bs, // string preceding current number in b, if any\n      i = -1, // index in s\n      s = [], // string constants and placeholders\n      q = []; // number interpolators\n\n  // Coerce inputs to strings.\n  a = a + \"\", b = b + \"\";\n\n  // Interpolate pairs of numbers in a & b.\n  while ((am = reA.exec(a))\n      && (bm = reB.exec(b))) {\n    if ((bs = bm.index) > bi) { // a string precedes the next number in b\n      bs = b.slice(bi, bs);\n      if (s[i]) s[i] += bs; // coalesce with previous string\n      else s[++i] = bs;\n    }\n    if ((am = am[0]) === (bm = bm[0])) { // numbers in a & b match\n      if (s[i]) s[i] += bm; // coalesce with previous string\n      else s[++i] = bm;\n    } else { // interpolate non-matching numbers\n      s[++i] = null;\n      q.push({i: i, x: number(am, bm)});\n    }\n    bi = reB.lastIndex;\n  }\n\n  // Add remains of b.\n  if (bi < b.length) {\n    bs = b.slice(bi);\n    if (s[i]) s[i] += bs; // coalesce with previous string\n    else s[++i] = bs;\n  }\n\n  // Special optimization for only a single match.\n  // Otherwise, interpolate each of the numbers and rejoin the string.\n  return s.length < 2 ? (q[0]\n      ? one(q[0].x)\n      : zero(b))\n      : (b = q.length, function(t) {\n          for (var i = 0, o; i < b; ++i) s[(o = q[i]).i] = o.x(t);\n          return s.join(\"\");\n        });\n}\n","var degrees = 180 / Math.PI;\n\nexport var identity = {\n  translateX: 0,\n  translateY: 0,\n  rotate: 0,\n  skewX: 0,\n  scaleX: 1,\n  scaleY: 1\n};\n\nexport default function(a, b, c, d, e, f) {\n  var scaleX, scaleY, skewX;\n  if (scaleX = Math.sqrt(a * a + b * b)) a /= scaleX, b /= scaleX;\n  if (skewX = a * c + b * d) c -= a * skewX, d -= b * skewX;\n  if (scaleY = Math.sqrt(c * c + d * d)) c /= scaleY, d /= scaleY, skewX /= scaleY;\n  if (a * d < b * c) a = -a, b = -b, skewX = -skewX, scaleX = -scaleX;\n  return {\n    translateX: e,\n    translateY: f,\n    rotate: Math.atan2(b, a) * degrees,\n    skewX: Math.atan(skewX) * degrees,\n    scaleX: scaleX,\n    scaleY: scaleY\n  };\n}\n","import number from \"../number.js\";\nimport {parseCss, parseSvg} from \"./parse.js\";\n\nfunction interpolateTransform(parse, pxComma, pxParen, degParen) {\n\n  function pop(s) {\n    return s.length ? s.pop() + \" \" : \"\";\n  }\n\n  function translate(xa, ya, xb, yb, s, q) {\n    if (xa !== xb || ya !== yb) {\n      var i = s.push(\"translate(\", null, pxComma, null, pxParen);\n      q.push({i: i - 4, x: number(xa, xb)}, {i: i - 2, x: number(ya, yb)});\n    } else if (xb || yb) {\n      s.push(\"translate(\" + xb + pxComma + yb + pxParen);\n    }\n  }\n\n  function rotate(a, b, s, q) {\n    if (a !== b) {\n      if (a - b > 180) b += 360; else if (b - a > 180) a += 360; // shortest path\n      q.push({i: s.push(pop(s) + \"rotate(\", null, degParen) - 2, x: number(a, b)});\n    } else if (b) {\n      s.push(pop(s) + \"rotate(\" + b + degParen);\n    }\n  }\n\n  function skewX(a, b, s, q) {\n    if (a !== b) {\n      q.push({i: s.push(pop(s) + \"skewX(\", null, degParen) - 2, x: number(a, b)});\n    } else if (b) {\n      s.push(pop(s) + \"skewX(\" + b + degParen);\n    }\n  }\n\n  function scale(xa, ya, xb, yb, s, q) {\n    if (xa !== xb || ya !== yb) {\n      var i = s.push(pop(s) + \"scale(\", null, \",\", null, \")\");\n      q.push({i: i - 4, x: number(xa, xb)}, {i: i - 2, x: number(ya, yb)});\n    } else if (xb !== 1 || yb !== 1) {\n      s.push(pop(s) + \"scale(\" + xb + \",\" + yb + \")\");\n    }\n  }\n\n  return function(a, b) {\n    var s = [], // string constants and placeholders\n        q = []; // number interpolators\n    a = parse(a), b = parse(b);\n    translate(a.translateX, a.translateY, b.translateX, b.translateY, s, q);\n    rotate(a.rotate, b.rotate, s, q);\n    skewX(a.skewX, b.skewX, s, q);\n    scale(a.scaleX, a.scaleY, b.scaleX, b.scaleY, s, q);\n    a = b = null; // gc\n    return function(t) {\n      var i = -1, n = q.length, o;\n      while (++i < n) s[(o = q[i]).i] = o.x(t);\n      return s.join(\"\");\n    };\n  };\n}\n\nexport var interpolateTransformCss = interpolateTransform(parseCss, \"px, \", \"px)\", \"deg)\");\nexport var interpolateTransformSvg = interpolateTransform(parseSvg, \", \", \")\", \")\");\n","import decompose, {identity} from \"./decompose.js\";\n\nvar svgNode;\n\n/* eslint-disable no-undef */\nexport function parseCss(value) {\n  const m = new (typeof DOMMatrix === \"function\" ? DOMMatrix : WebKitCSSMatrix)(value + \"\");\n  return m.isIdentity ? identity : decompose(m.a, m.b, m.c, m.d, m.e, m.f);\n}\n\nexport function parseSvg(value) {\n  if (value == null) return identity;\n  if (!svgNode) svgNode = document.createElementNS(\"http://www.w3.org/2000/svg\", \"g\");\n  svgNode.setAttribute(\"transform\", value);\n  if (!(value = svgNode.transform.baseVal.consolidate())) return identity;\n  value = value.matrix;\n  return decompose(value.a, value.b, value.c, value.d, value.e, value.f);\n}\n","var epsilon2 = 1e-12;\n\nfunction cosh(x) {\n  return ((x = Math.exp(x)) + 1 / x) / 2;\n}\n\nfunction sinh(x) {\n  return ((x = Math.exp(x)) - 1 / x) / 2;\n}\n\nfunction tanh(x) {\n  return ((x = Math.exp(2 * x)) - 1) / (x + 1);\n}\n\nexport default (function zoomRho(rho, rho2, rho4) {\n\n  // p0 = [ux0, uy0, w0]\n  // p1 = [ux1, uy1, w1]\n  function zoom(p0, p1) {\n    var ux0 = p0[0], uy0 = p0[1], w0 = p0[2],\n        ux1 = p1[0], uy1 = p1[1], w1 = p1[2],\n        dx = ux1 - ux0,\n        dy = uy1 - uy0,\n        d2 = dx * dx + dy * dy,\n        i,\n        S;\n\n    // Special case for u0 ≅ u1.\n    if (d2 < epsilon2) {\n      S = Math.log(w1 / w0) / rho;\n      i = function(t) {\n        return [\n          ux0 + t * dx,\n          uy0 + t * dy,\n          w0 * Math.exp(rho * t * S)\n        ];\n      }\n    }\n\n    // General case.\n    else {\n      var d1 = Math.sqrt(d2),\n          b0 = (w1 * w1 - w0 * w0 + rho4 * d2) / (2 * w0 * rho2 * d1),\n          b1 = (w1 * w1 - w0 * w0 - rho4 * d2) / (2 * w1 * rho2 * d1),\n          r0 = Math.log(Math.sqrt(b0 * b0 + 1) - b0),\n          r1 = Math.log(Math.sqrt(b1 * b1 + 1) - b1);\n      S = (r1 - r0) / rho;\n      i = function(t) {\n        var s = t * S,\n            coshr0 = cosh(r0),\n            u = w0 / (rho2 * d1) * (coshr0 * tanh(rho * s + r0) - sinh(r0));\n        return [\n          ux0 + u * dx,\n          uy0 + u * dy,\n          w0 * coshr0 / cosh(rho * s + r0)\n        ];\n      }\n    }\n\n    i.duration = S * 1000 * rho / Math.SQRT2;\n\n    return i;\n  }\n\n  zoom.rho = function(_) {\n    var _1 = Math.max(1e-3, +_), _2 = _1 * _1, _4 = _2 * _2;\n    return zoomRho(_1, _2, _4);\n  };\n\n  return zoom;\n})(Math.SQRT2, 2, 4);\n","export default Math.random;\n","import defaultSource from \"./defaultSource.js\";\n\nexport default (function sourceRandomNormal(source) {\n  function randomNormal(mu, sigma) {\n    var x, r;\n    mu = mu == null ? 0 : +mu;\n    sigma = sigma == null ? 1 : +sigma;\n    return function() {\n      var y;\n\n      // If available, use the second previously-generated uniform random.\n      if (x != null) y = x, x = null;\n\n      // Otherwise, generate a new x and y.\n      else do {\n        x = source() * 2 - 1;\n        y = source() * 2 - 1;\n        r = x * x + y * y;\n      } while (!r || r > 1);\n\n      return mu + sigma * y * Math.sqrt(-2 * Math.log(r) / r);\n    };\n  }\n\n  randomNormal.source = sourceRandomNormal;\n\n  return randomNormal;\n})(defaultSource);\n","export function initRange(domain, range) {\n  switch (arguments.length) {\n    case 0: break;\n    case 1: this.range(domain); break;\n    default: this.range(range).domain(domain); break;\n  }\n  return this;\n}\n\nexport function initInterpolator(domain, interpolator) {\n  switch (arguments.length) {\n    case 0: break;\n    case 1: {\n      if (typeof domain === \"function\") this.interpolator(domain);\n      else this.range(domain);\n      break;\n    }\n    default: {\n      this.domain(domain);\n      if (typeof interpolator === \"function\") this.interpolator(interpolator);\n      else this.range(interpolator);\n      break;\n    }\n  }\n  return this;\n}\n","import {ascending, bisect, quantileSorted as threshold} from \"d3-array\";\nimport {initRange} from \"./init.js\";\n\nexport default function quantile() {\n  var domain = [],\n      range = [],\n      thresholds = [],\n      unknown;\n\n  function rescale() {\n    var i = 0, n = Math.max(1, range.length);\n    thresholds = new Array(n - 1);\n    while (++i < n) thresholds[i - 1] = threshold(domain, i / n);\n    return scale;\n  }\n\n  function scale(x) {\n    return x == null || isNaN(x = +x) ? unknown : range[bisect(thresholds, x)];\n  }\n\n  scale.invertExtent = function(y) {\n    var i = range.indexOf(y);\n    return i < 0 ? [NaN, NaN] : [\n      i > 0 ? thresholds[i - 1] : domain[0],\n      i < thresholds.length ? thresholds[i] : domain[domain.length - 1]\n    ];\n  };\n\n  scale.domain = function(_) {\n    if (!arguments.length) return domain.slice();\n    domain = [];\n    for (let d of _) if (d != null && !isNaN(d = +d)) domain.push(d);\n    domain.sort(ascending);\n    return rescale();\n  };\n\n  scale.range = function(_) {\n    return arguments.length ? (range = Array.from(_), rescale()) : range.slice();\n  };\n\n  scale.unknown = function(_) {\n    return arguments.length ? (unknown = _, scale) : unknown;\n  };\n\n  scale.quantiles = function() {\n    return thresholds.slice();\n  };\n\n  scale.copy = function() {\n    return quantile()\n        .domain(domain)\n        .range(range)\n        .unknown(unknown);\n  };\n\n  return initRange.apply(scale, arguments);\n}\n","// Given something array like (or null), returns something that is strictly an\n// array. This is used to ensure that array-like objects passed to d3.selectAll\n// or selection.selectAll are converted into proper arrays when creating a\n// selection; we don’t ever want to create a selection backed by a live\n// HTMLCollection or NodeList. However, note that selection.selectAll will use a\n// static NodeList as a group, since it safely derived from querySelectorAll.\nexport default function array(x) {\n  return x == null ? [] : Array.isArray(x) ? x : Array.from(x);\n}\n","export default function(x) {\n  return function() {\n    return x;\n  };\n}\n","import namespace from \"./namespace.js\";\nimport {xhtml} from \"./namespaces.js\";\n\nfunction creatorInherit(name) {\n  return function() {\n    var document = this.ownerDocument,\n        uri = this.namespaceURI;\n    return uri === xhtml && document.documentElement.namespaceURI === xhtml\n        ? document.createElement(name)\n        : document.createElementNS(uri, name);\n  };\n}\n\nfunction creatorFixed(fullname) {\n  return function() {\n    return this.ownerDocument.createElementNS(fullname.space, fullname.local);\n  };\n}\n\nexport default function(name) {\n  var fullname = namespace(name);\n  return (fullname.local\n      ? creatorFixed\n      : creatorInherit)(fullname);\n}\n","export default function(selector) {\n  return function() {\n    return this.matches(selector);\n  };\n}\n\nexport function childMatcher(selector) {\n  return function(node) {\n    return node.matches(selector);\n  };\n}\n\n","import namespaces from \"./namespaces.js\";\n\nexport default function(name) {\n  var prefix = name += \"\", i = prefix.indexOf(\":\");\n  if (i >= 0 && (prefix = name.slice(0, i)) !== \"xmlns\") name = name.slice(i + 1);\n  return namespaces.hasOwnProperty(prefix) ? {space: namespaces[prefix], local: name} : name; // eslint-disable-line no-prototype-builtins\n}\n","export var xhtml = \"http://www.w3.org/1999/xhtml\";\n\nexport default {\n  svg: \"http://www.w3.org/2000/svg\",\n  xhtml: xhtml,\n  xlink: \"http://www.w3.org/1999/xlink\",\n  xml: \"http://www.w3.org/XML/1998/namespace\",\n  xmlns: \"http://www.w3.org/2000/xmlns/\"\n};\n","import sourceEvent from \"./sourceEvent.js\";\n\nexport default function(event, node) {\n  event = sourceEvent(event);\n  if (node === undefined) node = event.currentTarget;\n  if (node) {\n    var svg = node.ownerSVGElement || node;\n    if (svg.createSVGPoint) {\n      var point = svg.createSVGPoint();\n      point.x = event.clientX, point.y = event.clientY;\n      point = point.matrixTransform(node.getScreenCTM().inverse());\n      return [point.x, point.y];\n    }\n    if (node.getBoundingClientRect) {\n      var rect = node.getBoundingClientRect();\n      return [event.clientX - rect.left - node.clientLeft, event.clientY - rect.top - node.clientTop];\n    }\n  }\n  return [event.pageX, event.pageY];\n}\n","import {Selection, root} from \"./selection/index.js\";\n\nexport default function(selector) {\n  return typeof selector === \"string\"\n      ? new Selection([[document.querySelector(selector)]], [document.documentElement])\n      : new Selection([[selector]], root);\n}\n","import creator from \"../creator.js\";\n\nexport default function(name) {\n  var create = typeof name === \"function\" ? name : creator(name);\n  return this.select(function() {\n    return this.appendChild(create.apply(this, arguments));\n  });\n}\n","import namespace from \"../namespace.js\";\n\nfunction attrRemove(name) {\n  return function() {\n    this.removeAttribute(name);\n  };\n}\n\nfunction attrRemoveNS(fullname) {\n  return function() {\n    this.removeAttributeNS(fullname.space, fullname.local);\n  };\n}\n\nfunction attrConstant(name, value) {\n  return function() {\n    this.setAttribute(name, value);\n  };\n}\n\nfunction attrConstantNS(fullname, value) {\n  return function() {\n    this.setAttributeNS(fullname.space, fullname.local, value);\n  };\n}\n\nfunction attrFunction(name, value) {\n  return function() {\n    var v = value.apply(this, arguments);\n    if (v == null) this.removeAttribute(name);\n    else this.setAttribute(name, v);\n  };\n}\n\nfunction attrFunctionNS(fullname, value) {\n  return function() {\n    var v = value.apply(this, arguments);\n    if (v == null) this.removeAttributeNS(fullname.space, fullname.local);\n    else this.setAttributeNS(fullname.space, fullname.local, v);\n  };\n}\n\nexport default function(name, value) {\n  var fullname = namespace(name);\n\n  if (arguments.length < 2) {\n    var node = this.node();\n    return fullname.local\n        ? node.getAttributeNS(fullname.space, fullname.local)\n        : node.getAttribute(fullname);\n  }\n\n  return this.each((value == null\n      ? (fullname.local ? attrRemoveNS : attrRemove) : (typeof value === \"function\"\n      ? (fullname.local ? attrFunctionNS : attrFunction)\n      : (fullname.local ? attrConstantNS : attrConstant)))(fullname, value));\n}\n","export default function() {\n  var callback = arguments[0];\n  arguments[0] = this;\n  callback.apply(null, arguments);\n  return this;\n}\n","function classArray(string) {\n  return string.trim().split(/^|\\s+/);\n}\n\nfunction classList(node) {\n  return node.classList || new ClassList(node);\n}\n\nfunction ClassList(node) {\n  this._node = node;\n  this._names = classArray(node.getAttribute(\"class\") || \"\");\n}\n\nClassList.prototype = {\n  add: function(name) {\n    var i = this._names.indexOf(name);\n    if (i < 0) {\n      this._names.push(name);\n      this._node.setAttribute(\"class\", this._names.join(\" \"));\n    }\n  },\n  remove: function(name) {\n    var i = this._names.indexOf(name);\n    if (i >= 0) {\n      this._names.splice(i, 1);\n      this._node.setAttribute(\"class\", this._names.join(\" \"));\n    }\n  },\n  contains: function(name) {\n    return this._names.indexOf(name) >= 0;\n  }\n};\n\nfunction classedAdd(node, names) {\n  var list = classList(node), i = -1, n = names.length;\n  while (++i < n) list.add(names[i]);\n}\n\nfunction classedRemove(node, names) {\n  var list = classList(node), i = -1, n = names.length;\n  while (++i < n) list.remove(names[i]);\n}\n\nfunction classedTrue(names) {\n  return function() {\n    classedAdd(this, names);\n  };\n}\n\nfunction classedFalse(names) {\n  return function() {\n    classedRemove(this, names);\n  };\n}\n\nfunction classedFunction(names, value) {\n  return function() {\n    (value.apply(this, arguments) ? classedAdd : classedRemove)(this, names);\n  };\n}\n\nexport default function(name, value) {\n  var names = classArray(name + \"\");\n\n  if (arguments.length < 2) {\n    var list = classList(this.node()), i = -1, n = names.length;\n    while (++i < n) if (!list.contains(names[i])) return false;\n    return true;\n  }\n\n  return this.each((typeof value === \"function\"\n      ? classedFunction : value\n      ? classedTrue\n      : classedFalse)(names, value));\n}\n","function selection_cloneShallow() {\n  var clone = this.cloneNode(false), parent = this.parentNode;\n  return parent ? parent.insertBefore(clone, this.nextSibling) : clone;\n}\n\nfunction selection_cloneDeep() {\n  var clone = this.cloneNode(true), parent = this.parentNode;\n  return parent ? parent.insertBefore(clone, this.nextSibling) : clone;\n}\n\nexport default function(deep) {\n  return this.select(deep ? selection_cloneDeep : selection_cloneShallow);\n}\n","import {Selection} from \"./index.js\";\nimport {EnterNode} from \"./enter.js\";\nimport constant from \"../constant.js\";\n\nfunction bindIndex(parent, group, enter, update, exit, data) {\n  var i = 0,\n      node,\n      groupLength = group.length,\n      dataLength = data.length;\n\n  // Put any non-null nodes that fit into update.\n  // Put any null nodes into enter.\n  // Put any remaining data into enter.\n  for (; i < dataLength; ++i) {\n    if (node = group[i]) {\n      node.__data__ = data[i];\n      update[i] = node;\n    } else {\n      enter[i] = new EnterNode(parent, data[i]);\n    }\n  }\n\n  // Put any non-null nodes that don’t fit into exit.\n  for (; i < groupLength; ++i) {\n    if (node = group[i]) {\n      exit[i] = node;\n    }\n  }\n}\n\nfunction bindKey(parent, group, enter, update, exit, data, key) {\n  var i,\n      node,\n      nodeByKeyValue = new Map,\n      groupLength = group.length,\n      dataLength = data.length,\n      keyValues = new Array(groupLength),\n      keyValue;\n\n  // Compute the key for each node.\n  // If multiple nodes have the same key, the duplicates are added to exit.\n  for (i = 0; i < groupLength; ++i) {\n    if (node = group[i]) {\n      keyValues[i] = keyValue = key.call(node, node.__data__, i, group) + \"\";\n      if (nodeByKeyValue.has(keyValue)) {\n        exit[i] = node;\n      } else {\n        nodeByKeyValue.set(keyValue, node);\n      }\n    }\n  }\n\n  // Compute the key for each datum.\n  // If there a node associated with this key, join and add it to update.\n  // If there is not (or the key is a duplicate), add it to enter.\n  for (i = 0; i < dataLength; ++i) {\n    keyValue = key.call(parent, data[i], i, data) + \"\";\n    if (node = nodeByKeyValue.get(keyValue)) {\n      update[i] = node;\n      node.__data__ = data[i];\n      nodeByKeyValue.delete(keyValue);\n    } else {\n      enter[i] = new EnterNode(parent, data[i]);\n    }\n  }\n\n  // Add any remaining nodes that were not bound to data to exit.\n  for (i = 0; i < groupLength; ++i) {\n    if ((node = group[i]) && (nodeByKeyValue.get(keyValues[i]) === node)) {\n      exit[i] = node;\n    }\n  }\n}\n\nfunction datum(node) {\n  return node.__data__;\n}\n\nexport default function(value, key) {\n  if (!arguments.length) return Array.from(this, datum);\n\n  var bind = key ? bindKey : bindIndex,\n      parents = this._parents,\n      groups = this._groups;\n\n  if (typeof value !== \"function\") value = constant(value);\n\n  for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j = 0; j < m; ++j) {\n    var parent = parents[j],\n        group = groups[j],\n        groupLength = group.length,\n        data = arraylike(value.call(parent, parent && parent.__data__, j, parents)),\n        dataLength = data.length,\n        enterGroup = enter[j] = new Array(dataLength),\n        updateGroup = update[j] = new Array(dataLength),\n        exitGroup = exit[j] = new Array(groupLength);\n\n    bind(parent, group, enterGroup, updateGroup, exitGroup, data, key);\n\n    // Now connect the enter nodes to their following update node, such that\n    // appendChild can insert the materialized enter node before this node,\n    // rather than at the end of the parent node.\n    for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {\n      if (previous = enterGroup[i0]) {\n        if (i0 >= i1) i1 = i0 + 1;\n        while (!(next = updateGroup[i1]) && ++i1 < dataLength);\n        previous._next = next || null;\n      }\n    }\n  }\n\n  update = new Selection(update, parents);\n  update._enter = enter;\n  update._exit = exit;\n  return update;\n}\n\n// Given some data, this returns an array-like view of it: an object that\n// exposes a length property and allows numeric indexing. Note that unlike\n// selectAll, this isn’t worried about “live” collections because the resulting\n// array will only be used briefly while data is being bound. (It is possible to\n// cause the data to change while iterating by using a key function, but please\n// don’t; we’d rather avoid a gratuitous copy.)\nfunction arraylike(data) {\n  return typeof data === \"object\" && \"length\" in data\n    ? data // Array, TypedArray, NodeList, array-like\n    : Array.from(data); // Map, Set, iterable, string, or anything else\n}\n","export default function(value) {\n  return arguments.length\n      ? this.property(\"__data__\", value)\n      : this.node().__data__;\n}\n","import defaultView from \"../window.js\";\n\nfunction dispatchEvent(node, type, params) {\n  var window = defaultView(node),\n      event = window.CustomEvent;\n\n  if (typeof event === \"function\") {\n    event = new event(type, params);\n  } else {\n    event = window.document.createEvent(\"Event\");\n    if (params) event.initEvent(type, params.bubbles, params.cancelable), event.detail = params.detail;\n    else event.initEvent(type, false, false);\n  }\n\n  node.dispatchEvent(event);\n}\n\nfunction dispatchConstant(type, params) {\n  return function() {\n    return dispatchEvent(this, type, params);\n  };\n}\n\nfunction dispatchFunction(type, params) {\n  return function() {\n    return dispatchEvent(this, type, params.apply(this, arguments));\n  };\n}\n\nexport default function(type, params) {\n  return this.each((typeof params === \"function\"\n      ? dispatchFunction\n      : dispatchConstant)(type, params));\n}\n","export default function(callback) {\n\n  for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {\n    for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {\n      if (node = group[i]) callback.call(node, node.__data__, i, group);\n    }\n  }\n\n  return this;\n}\n","export default function() {\n  return !this.node();\n}\n","import sparse from \"./sparse.js\";\nimport {Selection} from \"./index.js\";\n\nexport default function() {\n  return new Selection(this._enter || this._groups.map(sparse), this._parents);\n}\n\nexport function EnterNode(parent, datum) {\n  this.ownerDocument = parent.ownerDocument;\n  this.namespaceURI = parent.namespaceURI;\n  this._next = null;\n  this._parent = parent;\n  this.__data__ = datum;\n}\n\nEnterNode.prototype = {\n  constructor: EnterNode,\n  appendChild: function(child) { return this._parent.insertBefore(child, this._next); },\n  insertBefore: function(child, next) { return this._parent.insertBefore(child, next); },\n  querySelector: function(selector) { return this._parent.querySelector(selector); },\n  querySelectorAll: function(selector) { return this._parent.querySelectorAll(selector); }\n};\n","import sparse from \"./sparse.js\";\nimport {Selection} from \"./index.js\";\n\nexport default function() {\n  return new Selection(this._exit || this._groups.map(sparse), this._parents);\n}\n","import {Selection} from \"./index.js\";\nimport matcher from \"../matcher.js\";\n\nexport default function(match) {\n  if (typeof match !== \"function\") match = matcher(match);\n\n  for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {\n      if ((node = group[i]) && match.call(node, node.__data__, i, group)) {\n        subgroup.push(node);\n      }\n    }\n  }\n\n  return new Selection(subgroups, this._parents);\n}\n","function htmlRemove() {\n  this.innerHTML = \"\";\n}\n\nfunction htmlConstant(value) {\n  return function() {\n    this.innerHTML = value;\n  };\n}\n\nfunction htmlFunction(value) {\n  return function() {\n    var v = value.apply(this, arguments);\n    this.innerHTML = v == null ? \"\" : v;\n  };\n}\n\nexport default function(value) {\n  return arguments.length\n      ? this.each(value == null\n          ? htmlRemove : (typeof value === \"function\"\n          ? htmlFunction\n          : htmlConstant)(value))\n      : this.node().innerHTML;\n}\n","import selection_select from \"./select.js\";\nimport selection_selectAll from \"./selectAll.js\";\nimport selection_selectChild from \"./selectChild.js\";\nimport selection_selectChildren from \"./selectChildren.js\";\nimport selection_filter from \"./filter.js\";\nimport selection_data from \"./data.js\";\nimport selection_enter from \"./enter.js\";\nimport selection_exit from \"./exit.js\";\nimport selection_join from \"./join.js\";\nimport selection_merge from \"./merge.js\";\nimport selection_order from \"./order.js\";\nimport selection_sort from \"./sort.js\";\nimport selection_call from \"./call.js\";\nimport selection_nodes from \"./nodes.js\";\nimport selection_node from \"./node.js\";\nimport selection_size from \"./size.js\";\nimport selection_empty from \"./empty.js\";\nimport selection_each from \"./each.js\";\nimport selection_attr from \"./attr.js\";\nimport selection_style from \"./style.js\";\nimport selection_property from \"./property.js\";\nimport selection_classed from \"./classed.js\";\nimport selection_text from \"./text.js\";\nimport selection_html from \"./html.js\";\nimport selection_raise from \"./raise.js\";\nimport selection_lower from \"./lower.js\";\nimport selection_append from \"./append.js\";\nimport selection_insert from \"./insert.js\";\nimport selection_remove from \"./remove.js\";\nimport selection_clone from \"./clone.js\";\nimport selection_datum from \"./datum.js\";\nimport selection_on from \"./on.js\";\nimport selection_dispatch from \"./dispatch.js\";\nimport selection_iterator from \"./iterator.js\";\n\nexport var root = [null];\n\nexport function Selection(groups, parents) {\n  this._groups = groups;\n  this._parents = parents;\n}\n\nfunction selection() {\n  return new Selection([[document.documentElement]], root);\n}\n\nfunction selection_selection() {\n  return this;\n}\n\nSelection.prototype = selection.prototype = {\n  constructor: Selection,\n  select: selection_select,\n  selectAll: selection_selectAll,\n  selectChild: selection_selectChild,\n  selectChildren: selection_selectChildren,\n  filter: selection_filter,\n  data: selection_data,\n  enter: selection_enter,\n  exit: selection_exit,\n  join: selection_join,\n  merge: selection_merge,\n  selection: selection_selection,\n  order: selection_order,\n  sort: selection_sort,\n  call: selection_call,\n  nodes: selection_nodes,\n  node: selection_node,\n  size: selection_size,\n  empty: selection_empty,\n  each: selection_each,\n  attr: selection_attr,\n  style: selection_style,\n  property: selection_property,\n  classed: selection_classed,\n  text: selection_text,\n  html: selection_html,\n  raise: selection_raise,\n  lower: selection_lower,\n  append: selection_append,\n  insert: selection_insert,\n  remove: selection_remove,\n  clone: selection_clone,\n  datum: selection_datum,\n  on: selection_on,\n  dispatch: selection_dispatch,\n  [Symbol.iterator]: selection_iterator\n};\n\nexport default selection;\n","import creator from \"../creator.js\";\nimport selector from \"../selector.js\";\n\nfunction constantNull() {\n  return null;\n}\n\nexport default function(name, before) {\n  var create = typeof name === \"function\" ? name : creator(name),\n      select = before == null ? constantNull : typeof before === \"function\" ? before : selector(before);\n  return this.select(function() {\n    return this.insertBefore(create.apply(this, arguments), select.apply(this, arguments) || null);\n  });\n}\n","export default function*() {\n  for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {\n    for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {\n      if (node = group[i]) yield node;\n    }\n  }\n}\n","export default function(onenter, onupdate, onexit) {\n  var enter = this.enter(), update = this, exit = this.exit();\n  if (typeof onenter === \"function\") {\n    enter = onenter(enter);\n    if (enter) enter = enter.selection();\n  } else {\n    enter = enter.append(onenter + \"\");\n  }\n  if (onupdate != null) {\n    update = onupdate(update);\n    if (update) update = update.selection();\n  }\n  if (onexit == null) exit.remove(); else onexit(exit);\n  return enter && update ? enter.merge(update).order() : update;\n}\n","function lower() {\n  if (this.previousSibling) this.parentNode.insertBefore(this, this.parentNode.firstChild);\n}\n\nexport default function() {\n  return this.each(lower);\n}\n","import {Selection} from \"./index.js\";\n\nexport default function(context) {\n  var selection = context.selection ? context.selection() : context;\n\n  for (var groups0 = this._groups, groups1 = selection._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {\n    for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {\n      if (node = group0[i] || group1[i]) {\n        merge[i] = node;\n      }\n    }\n  }\n\n  for (; j < m0; ++j) {\n    merges[j] = groups0[j];\n  }\n\n  return new Selection(merges, this._parents);\n}\n","export default function() {\n\n  for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {\n    for (var group = groups[j], i = 0, n = group.length; i < n; ++i) {\n      var node = group[i];\n      if (node) return node;\n    }\n  }\n\n  return null;\n}\n","export default function() {\n  return Array.from(this);\n}\n","function contextListener(listener) {\n  return function(event) {\n    listener.call(this, event, this.__data__);\n  };\n}\n\nfunction parseTypenames(typenames) {\n  return typenames.trim().split(/^|\\s+/).map(function(t) {\n    var name = \"\", i = t.indexOf(\".\");\n    if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);\n    return {type: t, name: name};\n  });\n}\n\nfunction onRemove(typename) {\n  return function() {\n    var on = this.__on;\n    if (!on) return;\n    for (var j = 0, i = -1, m = on.length, o; j < m; ++j) {\n      if (o = on[j], (!typename.type || o.type === typename.type) && o.name === typename.name) {\n        this.removeEventListener(o.type, o.listener, o.options);\n      } else {\n        on[++i] = o;\n      }\n    }\n    if (++i) on.length = i;\n    else delete this.__on;\n  };\n}\n\nfunction onAdd(typename, value, options) {\n  return function() {\n    var on = this.__on, o, listener = contextListener(value);\n    if (on) for (var j = 0, m = on.length; j < m; ++j) {\n      if ((o = on[j]).type === typename.type && o.name === typename.name) {\n        this.removeEventListener(o.type, o.listener, o.options);\n        this.addEventListener(o.type, o.listener = listener, o.options = options);\n        o.value = value;\n        return;\n      }\n    }\n    this.addEventListener(typename.type, listener, options);\n    o = {type: typename.type, name: typename.name, value: value, listener: listener, options: options};\n    if (!on) this.__on = [o];\n    else on.push(o);\n  };\n}\n\nexport default function(typename, value, options) {\n  var typenames = parseTypenames(typename + \"\"), i, n = typenames.length, t;\n\n  if (arguments.length < 2) {\n    var on = this.node().__on;\n    if (on) for (var j = 0, m = on.length, o; j < m; ++j) {\n      for (i = 0, o = on[j]; i < n; ++i) {\n        if ((t = typenames[i]).type === o.type && t.name === o.name) {\n          return o.value;\n        }\n      }\n    }\n    return;\n  }\n\n  on = value ? onAdd : onRemove;\n  for (i = 0; i < n; ++i) this.each(on(typenames[i], value, options));\n  return this;\n}\n","export default function() {\n\n  for (var groups = this._groups, j = -1, m = groups.length; ++j < m;) {\n    for (var group = groups[j], i = group.length - 1, next = group[i], node; --i >= 0;) {\n      if (node = group[i]) {\n        if (next && node.compareDocumentPosition(next) ^ 4) next.parentNode.insertBefore(node, next);\n        next = node;\n      }\n    }\n  }\n\n  return this;\n}\n","function propertyRemove(name) {\n  return function() {\n    delete this[name];\n  };\n}\n\nfunction propertyConstant(name, value) {\n  return function() {\n    this[name] = value;\n  };\n}\n\nfunction propertyFunction(name, value) {\n  return function() {\n    var v = value.apply(this, arguments);\n    if (v == null) delete this[name];\n    else this[name] = v;\n  };\n}\n\nexport default function(name, value) {\n  return arguments.length > 1\n      ? this.each((value == null\n          ? propertyRemove : typeof value === \"function\"\n          ? propertyFunction\n          : propertyConstant)(name, value))\n      : this.node()[name];\n}\n","function raise() {\n  if (this.nextSibling) this.parentNode.appendChild(this);\n}\n\nexport default function() {\n  return this.each(raise);\n}\n","function remove() {\n  var parent = this.parentNode;\n  if (parent) parent.removeChild(this);\n}\n\nexport default function() {\n  return this.each(remove);\n}\n","import {Selection} from \"./index.js\";\nimport selector from \"../selector.js\";\n\nexport default function(select) {\n  if (typeof select !== \"function\") select = selector(select);\n\n  for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {\n      if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {\n        if (\"__data__\" in node) subnode.__data__ = node.__data__;\n        subgroup[i] = subnode;\n      }\n    }\n  }\n\n  return new Selection(subgroups, this._parents);\n}\n","import {Selection} from \"./index.js\";\nimport array from \"../array.js\";\nimport selectorAll from \"../selectorAll.js\";\n\nfunction arrayAll(select) {\n  return function() {\n    return array(select.apply(this, arguments));\n  };\n}\n\nexport default function(select) {\n  if (typeof select === \"function\") select = arrayAll(select);\n  else select = selectorAll(select);\n\n  for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {\n      if (node = group[i]) {\n        subgroups.push(select.call(node, node.__data__, i, group));\n        parents.push(node);\n      }\n    }\n  }\n\n  return new Selection(subgroups, parents);\n}\n","import {childMatcher} from \"../matcher.js\";\n\nvar find = Array.prototype.find;\n\nfunction childFind(match) {\n  return function() {\n    return find.call(this.children, match);\n  };\n}\n\nfunction childFirst() {\n  return this.firstElementChild;\n}\n\nexport default function(match) {\n  return this.select(match == null ? childFirst\n      : childFind(typeof match === \"function\" ? match : childMatcher(match)));\n}\n","import {childMatcher} from \"../matcher.js\";\n\nvar filter = Array.prototype.filter;\n\nfunction children() {\n  return Array.from(this.children);\n}\n\nfunction childrenFilter(match) {\n  return function() {\n    return filter.call(this.children, match);\n  };\n}\n\nexport default function(match) {\n  return this.selectAll(match == null ? children\n      : childrenFilter(typeof match === \"function\" ? match : childMatcher(match)));\n}\n","export default function() {\n  let size = 0;\n  for (const node of this) ++size; // eslint-disable-line no-unused-vars\n  return size;\n}\n","import {Selection} from \"./index.js\";\n\nexport default function(compare) {\n  if (!compare) compare = ascending;\n\n  function compareNode(a, b) {\n    return a && b ? compare(a.__data__, b.__data__) : !a - !b;\n  }\n\n  for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, sortgroup = sortgroups[j] = new Array(n), node, i = 0; i < n; ++i) {\n      if (node = group[i]) {\n        sortgroup[i] = node;\n      }\n    }\n    sortgroup.sort(compareNode);\n  }\n\n  return new Selection(sortgroups, this._parents).order();\n}\n\nfunction ascending(a, b) {\n  return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;\n}\n","export default function(update) {\n  return new Array(update.length);\n}\n","import defaultView from \"../window.js\";\n\nfunction styleRemove(name) {\n  return function() {\n    this.style.removeProperty(name);\n  };\n}\n\nfunction styleConstant(name, value, priority) {\n  return function() {\n    this.style.setProperty(name, value, priority);\n  };\n}\n\nfunction styleFunction(name, value, priority) {\n  return function() {\n    var v = value.apply(this, arguments);\n    if (v == null) this.style.removeProperty(name);\n    else this.style.setProperty(name, v, priority);\n  };\n}\n\nexport default function(name, value, priority) {\n  return arguments.length > 1\n      ? this.each((value == null\n            ? styleRemove : typeof value === \"function\"\n            ? styleFunction\n            : styleConstant)(name, value, priority == null ? \"\" : priority))\n      : styleValue(this.node(), name);\n}\n\nexport function styleValue(node, name) {\n  return node.style.getPropertyValue(name)\n      || defaultView(node).getComputedStyle(node, null).getPropertyValue(name);\n}\n","function textRemove() {\n  this.textContent = \"\";\n}\n\nfunction textConstant(value) {\n  return function() {\n    this.textContent = value;\n  };\n}\n\nfunction textFunction(value) {\n  return function() {\n    var v = value.apply(this, arguments);\n    this.textContent = v == null ? \"\" : v;\n  };\n}\n\nexport default function(value) {\n  return arguments.length\n      ? this.each(value == null\n          ? textRemove : (typeof value === \"function\"\n          ? textFunction\n          : textConstant)(value))\n      : this.node().textContent;\n}\n","function none() {}\n\nexport default function(selector) {\n  return selector == null ? none : function() {\n    return this.querySelector(selector);\n  };\n}\n","function empty() {\n  return [];\n}\n\nexport default function(selector) {\n  return selector == null ? empty : function() {\n    return this.querySelectorAll(selector);\n  };\n}\n","export default function(event) {\n  let sourceEvent;\n  while (sourceEvent = event.sourceEvent) event = sourceEvent;\n  return event;\n}\n","export default function(node) {\n  return (node.ownerDocument && node.ownerDocument.defaultView) // node is a Node\n      || (node.document && node) // node is a Window\n      || node.defaultView; // node is a Document\n}\n","import {Timer} from \"./timer.js\";\n\nexport default function(callback, delay, time) {\n  var t = new Timer;\n  delay = delay == null ? 0 : +delay;\n  t.restart(elapsed => {\n    t.stop();\n    callback(elapsed + delay);\n  }, delay, time);\n  return t;\n}\n","var frame = 0, // is an animation frame pending?\n    timeout = 0, // is a timeout pending?\n    interval = 0, // are any timers active?\n    pokeDelay = 1000, // how frequently we check for clock skew\n    taskHead,\n    taskTail,\n    clockLast = 0,\n    clockNow = 0,\n    clockSkew = 0,\n    clock = typeof performance === \"object\" && performance.now ? performance : Date,\n    setFrame = typeof window === \"object\" && window.requestAnimationFrame ? window.requestAnimationFrame.bind(window) : function(f) { setTimeout(f, 17); };\n\nexport function now() {\n  return clockNow || (setFrame(clearNow), clockNow = clock.now() + clockSkew);\n}\n\nfunction clearNow() {\n  clockNow = 0;\n}\n\nexport function Timer() {\n  this._call =\n  this._time =\n  this._next = null;\n}\n\nTimer.prototype = timer.prototype = {\n  constructor: Timer,\n  restart: function(callback, delay, time) {\n    if (typeof callback !== \"function\") throw new TypeError(\"callback is not a function\");\n    time = (time == null ? now() : +time) + (delay == null ? 0 : +delay);\n    if (!this._next && taskTail !== this) {\n      if (taskTail) taskTail._next = this;\n      else taskHead = this;\n      taskTail = this;\n    }\n    this._call = callback;\n    this._time = time;\n    sleep();\n  },\n  stop: function() {\n    if (this._call) {\n      this._call = null;\n      this._time = Infinity;\n      sleep();\n    }\n  }\n};\n\nexport function timer(callback, delay, time) {\n  var t = new Timer;\n  t.restart(callback, delay, time);\n  return t;\n}\n\nexport function timerFlush() {\n  now(); // Get the current time, if not already set.\n  ++frame; // Pretend we’ve set an alarm, if we haven’t already.\n  var t = taskHead, e;\n  while (t) {\n    if ((e = clockNow - t._time) >= 0) t._call.call(undefined, e);\n    t = t._next;\n  }\n  --frame;\n}\n\nfunction wake() {\n  clockNow = (clockLast = clock.now()) + clockSkew;\n  frame = timeout = 0;\n  try {\n    timerFlush();\n  } finally {\n    frame = 0;\n    nap();\n    clockNow = 0;\n  }\n}\n\nfunction poke() {\n  var now = clock.now(), delay = now - clockLast;\n  if (delay > pokeDelay) clockSkew -= delay, clockLast = now;\n}\n\nfunction nap() {\n  var t0, t1 = taskHead, t2, time = Infinity;\n  while (t1) {\n    if (t1._call) {\n      if (time > t1._time) time = t1._time;\n      t0 = t1, t1 = t1._next;\n    } else {\n      t2 = t1._next, t1._next = null;\n      t1 = t0 ? t0._next = t2 : taskHead = t2;\n    }\n  }\n  taskTail = t0;\n  sleep(time);\n}\n\nfunction sleep(time) {\n  if (frame) return; // Soonest alarm already set, or will be.\n  if (timeout) timeout = clearTimeout(timeout);\n  var delay = time - clockNow; // Strictly less than if we recomputed clockNow.\n  if (delay > 24) {\n    if (time < Infinity) timeout = setTimeout(wake, time - clock.now() - clockSkew);\n    if (interval) interval = clearInterval(interval);\n  } else {\n    if (!interval) clockLast = clock.now(), interval = setInterval(poke, pokeDelay);\n    frame = 1, setFrame(wake);\n  }\n}\n","import {Transition} from \"./transition/index.js\";\nimport {SCHEDULED} from \"./transition/schedule.js\";\n\nvar root = [null];\n\nexport default function(node, name) {\n  var schedules = node.__transition,\n      schedule,\n      i;\n\n  if (schedules) {\n    name = name == null ? null : name + \"\";\n    for (i in schedules) {\n      if ((schedule = schedules[i]).state > SCHEDULED && schedule.name === name) {\n        return new Transition([[node]], root, name, +i);\n      }\n    }\n  }\n\n  return null;\n}\n","import \"./selection/index.js\";\nexport {default as transition} from \"./transition/index.js\";\nexport {default as active} from \"./active.js\";\nexport {default as interrupt} from \"./interrupt.js\";\n","import {STARTING, ENDING, ENDED} from \"./transition/schedule.js\";\n\nexport default function(node, name) {\n  var schedules = node.__transition,\n      schedule,\n      active,\n      empty = true,\n      i;\n\n  if (!schedules) return;\n\n  name = name == null ? null : name + \"\";\n\n  for (i in schedules) {\n    if ((schedule = schedules[i]).name !== name) { empty = false; continue; }\n    active = schedule.state > STARTING && schedule.state < ENDING;\n    schedule.state = ENDED;\n    schedule.timer.stop();\n    schedule.on.call(active ? \"interrupt\" : \"cancel\", node, node.__data__, schedule.index, schedule.group);\n    delete schedules[i];\n  }\n\n  if (empty) delete node.__transition;\n}\n","import {selection} from \"d3-selection\";\nimport selection_interrupt from \"./interrupt.js\";\nimport selection_transition from \"./transition.js\";\n\nselection.prototype.interrupt = selection_interrupt;\nselection.prototype.transition = selection_transition;\n","import interrupt from \"../interrupt.js\";\n\nexport default function(name) {\n  return this.each(function() {\n    interrupt(this, name);\n  });\n}\n","import {Transition, newId} from \"../transition/index.js\";\nimport schedule from \"../transition/schedule.js\";\nimport {easeCubicInOut} from \"d3-ease\";\nimport {now} from \"d3-timer\";\n\nvar defaultTiming = {\n  time: null, // Set on use.\n  delay: 0,\n  duration: 250,\n  ease: easeCubicInOut\n};\n\nfunction inherit(node, id) {\n  var timing;\n  while (!(timing = node.__transition) || !(timing = timing[id])) {\n    if (!(node = node.parentNode)) {\n      throw new Error(`transition ${id} not found`);\n    }\n  }\n  return timing;\n}\n\nexport default function(name) {\n  var id,\n      timing;\n\n  if (name instanceof Transition) {\n    id = name._id, name = name._name;\n  } else {\n    id = newId(), (timing = defaultTiming).time = now(), name = name == null ? null : name + \"\";\n  }\n\n  for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {\n      if (node = group[i]) {\n        schedule(node, name, id, i, group, timing || inherit(node, id));\n      }\n    }\n  }\n\n  return new Transition(groups, this._parents, name, id);\n}\n","import {interpolateTransformSvg as interpolateTransform} from \"d3-interpolate\";\nimport {namespace} from \"d3-selection\";\nimport {tweenValue} from \"./tween.js\";\nimport interpolate from \"./interpolate.js\";\n\nfunction attrRemove(name) {\n  return function() {\n    this.removeAttribute(name);\n  };\n}\n\nfunction attrRemoveNS(fullname) {\n  return function() {\n    this.removeAttributeNS(fullname.space, fullname.local);\n  };\n}\n\nfunction attrConstant(name, interpolate, value1) {\n  var string00,\n      string1 = value1 + \"\",\n      interpolate0;\n  return function() {\n    var string0 = this.getAttribute(name);\n    return string0 === string1 ? null\n        : string0 === string00 ? interpolate0\n        : interpolate0 = interpolate(string00 = string0, value1);\n  };\n}\n\nfunction attrConstantNS(fullname, interpolate, value1) {\n  var string00,\n      string1 = value1 + \"\",\n      interpolate0;\n  return function() {\n    var string0 = this.getAttributeNS(fullname.space, fullname.local);\n    return string0 === string1 ? null\n        : string0 === string00 ? interpolate0\n        : interpolate0 = interpolate(string00 = string0, value1);\n  };\n}\n\nfunction attrFunction(name, interpolate, value) {\n  var string00,\n      string10,\n      interpolate0;\n  return function() {\n    var string0, value1 = value(this), string1;\n    if (value1 == null) return void this.removeAttribute(name);\n    string0 = this.getAttribute(name);\n    string1 = value1 + \"\";\n    return string0 === string1 ? null\n        : string0 === string00 && string1 === string10 ? interpolate0\n        : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));\n  };\n}\n\nfunction attrFunctionNS(fullname, interpolate, value) {\n  var string00,\n      string10,\n      interpolate0;\n  return function() {\n    var string0, value1 = value(this), string1;\n    if (value1 == null) return void this.removeAttributeNS(fullname.space, fullname.local);\n    string0 = this.getAttributeNS(fullname.space, fullname.local);\n    string1 = value1 + \"\";\n    return string0 === string1 ? null\n        : string0 === string00 && string1 === string10 ? interpolate0\n        : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));\n  };\n}\n\nexport default function(name, value) {\n  var fullname = namespace(name), i = fullname === \"transform\" ? interpolateTransform : interpolate;\n  return this.attrTween(name, typeof value === \"function\"\n      ? (fullname.local ? attrFunctionNS : attrFunction)(fullname, i, tweenValue(this, \"attr.\" + name, value))\n      : value == null ? (fullname.local ? attrRemoveNS : attrRemove)(fullname)\n      : (fullname.local ? attrConstantNS : attrConstant)(fullname, i, value));\n}\n","import {namespace} from \"d3-selection\";\n\nfunction attrInterpolate(name, i) {\n  return function(t) {\n    this.setAttribute(name, i.call(this, t));\n  };\n}\n\nfunction attrInterpolateNS(fullname, i) {\n  return function(t) {\n    this.setAttributeNS(fullname.space, fullname.local, i.call(this, t));\n  };\n}\n\nfunction attrTweenNS(fullname, value) {\n  var t0, i0;\n  function tween() {\n    var i = value.apply(this, arguments);\n    if (i !== i0) t0 = (i0 = i) && attrInterpolateNS(fullname, i);\n    return t0;\n  }\n  tween._value = value;\n  return tween;\n}\n\nfunction attrTween(name, value) {\n  var t0, i0;\n  function tween() {\n    var i = value.apply(this, arguments);\n    if (i !== i0) t0 = (i0 = i) && attrInterpolate(name, i);\n    return t0;\n  }\n  tween._value = value;\n  return tween;\n}\n\nexport default function(name, value) {\n  var key = \"attr.\" + name;\n  if (arguments.length < 2) return (key = this.tween(key)) && key._value;\n  if (value == null) return this.tween(key, null);\n  if (typeof value !== \"function\") throw new Error;\n  var fullname = namespace(name);\n  return this.tween(key, (fullname.local ? attrTweenNS : attrTween)(fullname, value));\n}\n","import {get, init} from \"./schedule.js\";\n\nfunction delayFunction(id, value) {\n  return function() {\n    init(this, id).delay = +value.apply(this, arguments);\n  };\n}\n\nfunction delayConstant(id, value) {\n  return value = +value, function() {\n    init(this, id).delay = value;\n  };\n}\n\nexport default function(value) {\n  var id = this._id;\n\n  return arguments.length\n      ? this.each((typeof value === \"function\"\n          ? delayFunction\n          : delayConstant)(id, value))\n      : get(this.node(), id).delay;\n}\n","import {get, set} from \"./schedule.js\";\n\nfunction durationFunction(id, value) {\n  return function() {\n    set(this, id).duration = +value.apply(this, arguments);\n  };\n}\n\nfunction durationConstant(id, value) {\n  return value = +value, function() {\n    set(this, id).duration = value;\n  };\n}\n\nexport default function(value) {\n  var id = this._id;\n\n  return arguments.length\n      ? this.each((typeof value === \"function\"\n          ? durationFunction\n          : durationConstant)(id, value))\n      : get(this.node(), id).duration;\n}\n","import {get, set} from \"./schedule.js\";\n\nfunction easeConstant(id, value) {\n  if (typeof value !== \"function\") throw new Error;\n  return function() {\n    set(this, id).ease = value;\n  };\n}\n\nexport default function(value) {\n  var id = this._id;\n\n  return arguments.length\n      ? this.each(easeConstant(id, value))\n      : get(this.node(), id).ease;\n}\n","import {set} from \"./schedule.js\";\n\nfunction easeVarying(id, value) {\n  return function() {\n    var v = value.apply(this, arguments);\n    if (typeof v !== \"function\") throw new Error;\n    set(this, id).ease = v;\n  };\n}\n\nexport default function(value) {\n  if (typeof value !== \"function\") throw new Error;\n  return this.each(easeVarying(this._id, value));\n}\n","import {set} from \"./schedule.js\";\n\nexport default function() {\n  var on0, on1, that = this, id = that._id, size = that.size();\n  return new Promise(function(resolve, reject) {\n    var cancel = {value: reject},\n        end = {value: function() { if (--size === 0) resolve(); }};\n\n    that.each(function() {\n      var schedule = set(this, id),\n          on = schedule.on;\n\n      // If this node shared a dispatch with the previous node,\n      // just assign the updated shared dispatch and we’re done!\n      // Otherwise, copy-on-write.\n      if (on !== on0) {\n        on1 = (on0 = on).copy();\n        on1._.cancel.push(cancel);\n        on1._.interrupt.push(cancel);\n        on1._.end.push(end);\n      }\n\n      schedule.on = on1;\n    });\n\n    // The selection was empty, resolve end immediately\n    if (size === 0) resolve();\n  });\n}\n","import {matcher} from \"d3-selection\";\nimport {Transition} from \"./index.js\";\n\nexport default function(match) {\n  if (typeof match !== \"function\") match = matcher(match);\n\n  for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {\n      if ((node = group[i]) && match.call(node, node.__data__, i, group)) {\n        subgroup.push(node);\n      }\n    }\n  }\n\n  return new Transition(subgroups, this._parents, this._name, this._id);\n}\n","import {selection} from \"d3-selection\";\nimport transition_attr from \"./attr.js\";\nimport transition_attrTween from \"./attrTween.js\";\nimport transition_delay from \"./delay.js\";\nimport transition_duration from \"./duration.js\";\nimport transition_ease from \"./ease.js\";\nimport transition_easeVarying from \"./easeVarying.js\";\nimport transition_filter from \"./filter.js\";\nimport transition_merge from \"./merge.js\";\nimport transition_on from \"./on.js\";\nimport transition_remove from \"./remove.js\";\nimport transition_select from \"./select.js\";\nimport transition_selectAll from \"./selectAll.js\";\nimport transition_selection from \"./selection.js\";\nimport transition_style from \"./style.js\";\nimport transition_styleTween from \"./styleTween.js\";\nimport transition_text from \"./text.js\";\nimport transition_textTween from \"./textTween.js\";\nimport transition_transition from \"./transition.js\";\nimport transition_tween from \"./tween.js\";\nimport transition_end from \"./end.js\";\n\nvar id = 0;\n\nexport function Transition(groups, parents, name, id) {\n  this._groups = groups;\n  this._parents = parents;\n  this._name = name;\n  this._id = id;\n}\n\nexport default function transition(name) {\n  return selection().transition(name);\n}\n\nexport function newId() {\n  return ++id;\n}\n\nvar selection_prototype = selection.prototype;\n\nTransition.prototype = transition.prototype = {\n  constructor: Transition,\n  select: transition_select,\n  selectAll: transition_selectAll,\n  selectChild: selection_prototype.selectChild,\n  selectChildren: selection_prototype.selectChildren,\n  filter: transition_filter,\n  merge: transition_merge,\n  selection: transition_selection,\n  transition: transition_transition,\n  call: selection_prototype.call,\n  nodes: selection_prototype.nodes,\n  node: selection_prototype.node,\n  size: selection_prototype.size,\n  empty: selection_prototype.empty,\n  each: selection_prototype.each,\n  on: transition_on,\n  attr: transition_attr,\n  attrTween: transition_attrTween,\n  style: transition_style,\n  styleTween: transition_styleTween,\n  text: transition_text,\n  textTween: transition_textTween,\n  remove: transition_remove,\n  tween: transition_tween,\n  delay: transition_delay,\n  duration: transition_duration,\n  ease: transition_ease,\n  easeVarying: transition_easeVarying,\n  end: transition_end,\n  [Symbol.iterator]: selection_prototype[Symbol.iterator]\n};\n","import {color} from \"d3-color\";\nimport {interpolateNumber, interpolateRgb, interpolateString} from \"d3-interpolate\";\n\nexport default function(a, b) {\n  var c;\n  return (typeof b === \"number\" ? interpolateNumber\n      : b instanceof color ? interpolateRgb\n      : (c = color(b)) ? (b = c, interpolateRgb)\n      : interpolateString)(a, b);\n}\n","import {Transition} from \"./index.js\";\n\nexport default function(transition) {\n  if (transition._id !== this._id) throw new Error;\n\n  for (var groups0 = this._groups, groups1 = transition._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {\n    for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {\n      if (node = group0[i] || group1[i]) {\n        merge[i] = node;\n      }\n    }\n  }\n\n  for (; j < m0; ++j) {\n    merges[j] = groups0[j];\n  }\n\n  return new Transition(merges, this._parents, this._name, this._id);\n}\n","import {get, set, init} from \"./schedule.js\";\n\nfunction start(name) {\n  return (name + \"\").trim().split(/^|\\s+/).every(function(t) {\n    var i = t.indexOf(\".\");\n    if (i >= 0) t = t.slice(0, i);\n    return !t || t === \"start\";\n  });\n}\n\nfunction onFunction(id, name, listener) {\n  var on0, on1, sit = start(name) ? init : set;\n  return function() {\n    var schedule = sit(this, id),\n        on = schedule.on;\n\n    // If this node shared a dispatch with the previous node,\n    // just assign the updated shared dispatch and we’re done!\n    // Otherwise, copy-on-write.\n    if (on !== on0) (on1 = (on0 = on).copy()).on(name, listener);\n\n    schedule.on = on1;\n  };\n}\n\nexport default function(name, listener) {\n  var id = this._id;\n\n  return arguments.length < 2\n      ? get(this.node(), id).on.on(name)\n      : this.each(onFunction(id, name, listener));\n}\n","function removeFunction(id) {\n  return function() {\n    var parent = this.parentNode;\n    for (var i in this.__transition) if (+i !== id) return;\n    if (parent) parent.removeChild(this);\n  };\n}\n\nexport default function() {\n  return this.on(\"end.remove\", removeFunction(this._id));\n}\n","import {dispatch} from \"d3-dispatch\";\nimport {timer, timeout} from \"d3-timer\";\n\nvar emptyOn = dispatch(\"start\", \"end\", \"cancel\", \"interrupt\");\nvar emptyTween = [];\n\nexport var CREATED = 0;\nexport var SCHEDULED = 1;\nexport var STARTING = 2;\nexport var STARTED = 3;\nexport var RUNNING = 4;\nexport var ENDING = 5;\nexport var ENDED = 6;\n\nexport default function(node, name, id, index, group, timing) {\n  var schedules = node.__transition;\n  if (!schedules) node.__transition = {};\n  else if (id in schedules) return;\n  create(node, id, {\n    name: name,\n    index: index, // For context during callback.\n    group: group, // For context during callback.\n    on: emptyOn,\n    tween: emptyTween,\n    time: timing.time,\n    delay: timing.delay,\n    duration: timing.duration,\n    ease: timing.ease,\n    timer: null,\n    state: CREATED\n  });\n}\n\nexport function init(node, id) {\n  var schedule = get(node, id);\n  if (schedule.state > CREATED) throw new Error(\"too late; already scheduled\");\n  return schedule;\n}\n\nexport function set(node, id) {\n  var schedule = get(node, id);\n  if (schedule.state > STARTED) throw new Error(\"too late; already running\");\n  return schedule;\n}\n\nexport function get(node, id) {\n  var schedule = node.__transition;\n  if (!schedule || !(schedule = schedule[id])) throw new Error(\"transition not found\");\n  return schedule;\n}\n\nfunction create(node, id, self) {\n  var schedules = node.__transition,\n      tween;\n\n  // Initialize the self timer when the transition is created.\n  // Note the actual delay is not known until the first callback!\n  schedules[id] = self;\n  self.timer = timer(schedule, 0, self.time);\n\n  function schedule(elapsed) {\n    self.state = SCHEDULED;\n    self.timer.restart(start, self.delay, self.time);\n\n    // If the elapsed delay is less than our first sleep, start immediately.\n    if (self.delay <= elapsed) start(elapsed - self.delay);\n  }\n\n  function start(elapsed) {\n    var i, j, n, o;\n\n    // If the state is not SCHEDULED, then we previously errored on start.\n    if (self.state !== SCHEDULED) return stop();\n\n    for (i in schedules) {\n      o = schedules[i];\n      if (o.name !== self.name) continue;\n\n      // While this element already has a starting transition during this frame,\n      // defer starting an interrupting transition until that transition has a\n      // chance to tick (and possibly end); see d3/d3-transition#54!\n      if (o.state === STARTED) return timeout(start);\n\n      // Interrupt the active transition, if any.\n      if (o.state === RUNNING) {\n        o.state = ENDED;\n        o.timer.stop();\n        o.on.call(\"interrupt\", node, node.__data__, o.index, o.group);\n        delete schedules[i];\n      }\n\n      // Cancel any pre-empted transitions.\n      else if (+i < id) {\n        o.state = ENDED;\n        o.timer.stop();\n        o.on.call(\"cancel\", node, node.__data__, o.index, o.group);\n        delete schedules[i];\n      }\n    }\n\n    // Defer the first tick to end of the current frame; see d3/d3#1576.\n    // Note the transition may be canceled after start and before the first tick!\n    // Note this must be scheduled before the start event; see d3/d3-transition#16!\n    // Assuming this is successful, subsequent callbacks go straight to tick.\n    timeout(function() {\n      if (self.state === STARTED) {\n        self.state = RUNNING;\n        self.timer.restart(tick, self.delay, self.time);\n        tick(elapsed);\n      }\n    });\n\n    // Dispatch the start event.\n    // Note this must be done before the tween are initialized.\n    self.state = STARTING;\n    self.on.call(\"start\", node, node.__data__, self.index, self.group);\n    if (self.state !== STARTING) return; // interrupted\n    self.state = STARTED;\n\n    // Initialize the tween, deleting null tween.\n    tween = new Array(n = self.tween.length);\n    for (i = 0, j = -1; i < n; ++i) {\n      if (o = self.tween[i].value.call(node, node.__data__, self.index, self.group)) {\n        tween[++j] = o;\n      }\n    }\n    tween.length = j + 1;\n  }\n\n  function tick(elapsed) {\n    var t = elapsed < self.duration ? self.ease.call(null, elapsed / self.duration) : (self.timer.restart(stop), self.state = ENDING, 1),\n        i = -1,\n        n = tween.length;\n\n    while (++i < n) {\n      tween[i].call(node, t);\n    }\n\n    // Dispatch the end event.\n    if (self.state === ENDING) {\n      self.on.call(\"end\", node, node.__data__, self.index, self.group);\n      stop();\n    }\n  }\n\n  function stop() {\n    self.state = ENDED;\n    self.timer.stop();\n    delete schedules[id];\n    for (var i in schedules) return; // eslint-disable-line no-unused-vars\n    delete node.__transition;\n  }\n}\n","import {selector} from \"d3-selection\";\nimport {Transition} from \"./index.js\";\nimport schedule, {get} from \"./schedule.js\";\n\nexport default function(select) {\n  var name = this._name,\n      id = this._id;\n\n  if (typeof select !== \"function\") select = selector(select);\n\n  for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {\n      if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {\n        if (\"__data__\" in node) subnode.__data__ = node.__data__;\n        subgroup[i] = subnode;\n        schedule(subgroup[i], name, id, i, subgroup, get(node, id));\n      }\n    }\n  }\n\n  return new Transition(subgroups, this._parents, name, id);\n}\n","import {selectorAll} from \"d3-selection\";\nimport {Transition} from \"./index.js\";\nimport schedule, {get} from \"./schedule.js\";\n\nexport default function(select) {\n  var name = this._name,\n      id = this._id;\n\n  if (typeof select !== \"function\") select = selectorAll(select);\n\n  for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {\n      if (node = group[i]) {\n        for (var children = select.call(node, node.__data__, i, group), child, inherit = get(node, id), k = 0, l = children.length; k < l; ++k) {\n          if (child = children[k]) {\n            schedule(child, name, id, k, children, inherit);\n          }\n        }\n        subgroups.push(children);\n        parents.push(node);\n      }\n    }\n  }\n\n  return new Transition(subgroups, parents, name, id);\n}\n","import {selection} from \"d3-selection\";\n\nvar Selection = selection.prototype.constructor;\n\nexport default function() {\n  return new Selection(this._groups, this._parents);\n}\n","import {interpolateTransformCss as interpolateTransform} from \"d3-interpolate\";\nimport {style} from \"d3-selection\";\nimport {set} from \"./schedule.js\";\nimport {tweenValue} from \"./tween.js\";\nimport interpolate from \"./interpolate.js\";\n\nfunction styleNull(name, interpolate) {\n  var string00,\n      string10,\n      interpolate0;\n  return function() {\n    var string0 = style(this, name),\n        string1 = (this.style.removeProperty(name), style(this, name));\n    return string0 === string1 ? null\n        : string0 === string00 && string1 === string10 ? interpolate0\n        : interpolate0 = interpolate(string00 = string0, string10 = string1);\n  };\n}\n\nfunction styleRemove(name) {\n  return function() {\n    this.style.removeProperty(name);\n  };\n}\n\nfunction styleConstant(name, interpolate, value1) {\n  var string00,\n      string1 = value1 + \"\",\n      interpolate0;\n  return function() {\n    var string0 = style(this, name);\n    return string0 === string1 ? null\n        : string0 === string00 ? interpolate0\n        : interpolate0 = interpolate(string00 = string0, value1);\n  };\n}\n\nfunction styleFunction(name, interpolate, value) {\n  var string00,\n      string10,\n      interpolate0;\n  return function() {\n    var string0 = style(this, name),\n        value1 = value(this),\n        string1 = value1 + \"\";\n    if (value1 == null) string1 = value1 = (this.style.removeProperty(name), style(this, name));\n    return string0 === string1 ? null\n        : string0 === string00 && string1 === string10 ? interpolate0\n        : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));\n  };\n}\n\nfunction styleMaybeRemove(id, name) {\n  var on0, on1, listener0, key = \"style.\" + name, event = \"end.\" + key, remove;\n  return function() {\n    var schedule = set(this, id),\n        on = schedule.on,\n        listener = schedule.value[key] == null ? remove || (remove = styleRemove(name)) : undefined;\n\n    // If this node shared a dispatch with the previous node,\n    // just assign the updated shared dispatch and we’re done!\n    // Otherwise, copy-on-write.\n    if (on !== on0 || listener0 !== listener) (on1 = (on0 = on).copy()).on(event, listener0 = listener);\n\n    schedule.on = on1;\n  };\n}\n\nexport default function(name, value, priority) {\n  var i = (name += \"\") === \"transform\" ? interpolateTransform : interpolate;\n  return value == null ? this\n      .styleTween(name, styleNull(name, i))\n      .on(\"end.style.\" + name, styleRemove(name))\n    : typeof value === \"function\" ? this\n      .styleTween(name, styleFunction(name, i, tweenValue(this, \"style.\" + name, value)))\n      .each(styleMaybeRemove(this._id, name))\n    : this\n      .styleTween(name, styleConstant(name, i, value), priority)\n      .on(\"end.style.\" + name, null);\n}\n","function styleInterpolate(name, i, priority) {\n  return function(t) {\n    this.style.setProperty(name, i.call(this, t), priority);\n  };\n}\n\nfunction styleTween(name, value, priority) {\n  var t, i0;\n  function tween() {\n    var i = value.apply(this, arguments);\n    if (i !== i0) t = (i0 = i) && styleInterpolate(name, i, priority);\n    return t;\n  }\n  tween._value = value;\n  return tween;\n}\n\nexport default function(name, value, priority) {\n  var key = \"style.\" + (name += \"\");\n  if (arguments.length < 2) return (key = this.tween(key)) && key._value;\n  if (value == null) return this.tween(key, null);\n  if (typeof value !== \"function\") throw new Error;\n  return this.tween(key, styleTween(name, value, priority == null ? \"\" : priority));\n}\n","import {tweenValue} from \"./tween.js\";\n\nfunction textConstant(value) {\n  return function() {\n    this.textContent = value;\n  };\n}\n\nfunction textFunction(value) {\n  return function() {\n    var value1 = value(this);\n    this.textContent = value1 == null ? \"\" : value1;\n  };\n}\n\nexport default function(value) {\n  return this.tween(\"text\", typeof value === \"function\"\n      ? textFunction(tweenValue(this, \"text\", value))\n      : textConstant(value == null ? \"\" : value + \"\"));\n}\n","function textInterpolate(i) {\n  return function(t) {\n    this.textContent = i.call(this, t);\n  };\n}\n\nfunction textTween(value) {\n  var t0, i0;\n  function tween() {\n    var i = value.apply(this, arguments);\n    if (i !== i0) t0 = (i0 = i) && textInterpolate(i);\n    return t0;\n  }\n  tween._value = value;\n  return tween;\n}\n\nexport default function(value) {\n  var key = \"text\";\n  if (arguments.length < 1) return (key = this.tween(key)) && key._value;\n  if (value == null) return this.tween(key, null);\n  if (typeof value !== \"function\") throw new Error;\n  return this.tween(key, textTween(value));\n}\n","import {Transition, newId} from \"./index.js\";\nimport schedule, {get} from \"./schedule.js\";\n\nexport default function() {\n  var name = this._name,\n      id0 = this._id,\n      id1 = newId();\n\n  for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {\n      if (node = group[i]) {\n        var inherit = get(node, id0);\n        schedule(node, name, id1, i, group, {\n          time: inherit.time + inherit.delay + inherit.duration,\n          delay: 0,\n          duration: inherit.duration,\n          ease: inherit.ease\n        });\n      }\n    }\n  }\n\n  return new Transition(groups, this._parents, name, id1);\n}\n","import {get, set} from \"./schedule.js\";\n\nfunction tweenRemove(id, name) {\n  var tween0, tween1;\n  return function() {\n    var schedule = set(this, id),\n        tween = schedule.tween;\n\n    // If this node shared tween with the previous node,\n    // just assign the updated shared tween and we’re done!\n    // Otherwise, copy-on-write.\n    if (tween !== tween0) {\n      tween1 = tween0 = tween;\n      for (var i = 0, n = tween1.length; i < n; ++i) {\n        if (tween1[i].name === name) {\n          tween1 = tween1.slice();\n          tween1.splice(i, 1);\n          break;\n        }\n      }\n    }\n\n    schedule.tween = tween1;\n  };\n}\n\nfunction tweenFunction(id, name, value) {\n  var tween0, tween1;\n  if (typeof value !== \"function\") throw new Error;\n  return function() {\n    var schedule = set(this, id),\n        tween = schedule.tween;\n\n    // If this node shared tween with the previous node,\n    // just assign the updated shared tween and we’re done!\n    // Otherwise, copy-on-write.\n    if (tween !== tween0) {\n      tween1 = (tween0 = tween).slice();\n      for (var t = {name: name, value: value}, i = 0, n = tween1.length; i < n; ++i) {\n        if (tween1[i].name === name) {\n          tween1[i] = t;\n          break;\n        }\n      }\n      if (i === n) tween1.push(t);\n    }\n\n    schedule.tween = tween1;\n  };\n}\n\nexport default function(name, value) {\n  var id = this._id;\n\n  name += \"\";\n\n  if (arguments.length < 2) {\n    var tween = get(this.node(), id).tween;\n    for (var i = 0, n = tween.length, t; i < n; ++i) {\n      if ((t = tween[i]).name === name) {\n        return t.value;\n      }\n    }\n    return null;\n  }\n\n  return this.each((value == null ? tweenRemove : tweenFunction)(id, name, value));\n}\n\nexport function tweenValue(transition, name, value) {\n  var id = transition._id;\n\n  transition.each(function() {\n    var schedule = set(this, id);\n    (schedule.value || (schedule.value = {}))[name] = value.apply(this, arguments);\n  });\n\n  return function(node) {\n    return get(node, id).value[name];\n  };\n}\n","export default x => () => x;\n","export default function ZoomEvent(type, {\n  sourceEvent,\n  target,\n  transform,\n  dispatch\n}) {\n  Object.defineProperties(this, {\n    type: {value: type, enumerable: true, configurable: true},\n    sourceEvent: {value: sourceEvent, enumerable: true, configurable: true},\n    target: {value: target, enumerable: true, configurable: true},\n    transform: {value: transform, enumerable: true, configurable: true},\n    _: {value: dispatch}\n  });\n}\n","export {default as zoom} from \"./zoom.js\";\nexport {default as zoomTransform, identity as zoomIdentity, Transform as ZoomTransform} from \"./transform.js\";\n","export function nopropagation(event) {\n  event.stopImmediatePropagation();\n}\n\nexport default function(event) {\n  event.preventDefault();\n  event.stopImmediatePropagation();\n}\n","export function Transform(k, x, y) {\n  this.k = k;\n  this.x = x;\n  this.y = y;\n}\n\nTransform.prototype = {\n  constructor: Transform,\n  scale: function(k) {\n    return k === 1 ? this : new Transform(this.k * k, this.x, this.y);\n  },\n  translate: function(x, y) {\n    return x === 0 & y === 0 ? this : new Transform(this.k, this.x + this.k * x, this.y + this.k * y);\n  },\n  apply: function(point) {\n    return [point[0] * this.k + this.x, point[1] * this.k + this.y];\n  },\n  applyX: function(x) {\n    return x * this.k + this.x;\n  },\n  applyY: function(y) {\n    return y * this.k + this.y;\n  },\n  invert: function(location) {\n    return [(location[0] - this.x) / this.k, (location[1] - this.y) / this.k];\n  },\n  invertX: function(x) {\n    return (x - this.x) / this.k;\n  },\n  invertY: function(y) {\n    return (y - this.y) / this.k;\n  },\n  rescaleX: function(x) {\n    return x.copy().domain(x.range().map(this.invertX, this).map(x.invert, x));\n  },\n  rescaleY: function(y) {\n    return y.copy().domain(y.range().map(this.invertY, this).map(y.invert, y));\n  },\n  toString: function() {\n    return \"translate(\" + this.x + \",\" + this.y + \") scale(\" + this.k + \")\";\n  }\n};\n\nexport var identity = new Transform(1, 0, 0);\n\ntransform.prototype = Transform.prototype;\n\nexport default function transform(node) {\n  while (!node.__zoom) if (!(node = node.parentNode)) return identity;\n  return node.__zoom;\n}\n","import {dispatch} from \"d3-dispatch\";\nimport {dragDisable, dragEnable} from \"d3-drag\";\nimport {interpolateZoom} from \"d3-interpolate\";\nimport {select, pointer} from \"d3-selection\";\nimport {interrupt} from \"d3-transition\";\nimport constant from \"./constant.js\";\nimport ZoomEvent from \"./event.js\";\nimport {Transform, identity} from \"./transform.js\";\nimport noevent, {nopropagation} from \"./noevent.js\";\n\n// Ignore right-click, since that should open the context menu.\n// except for pinch-to-zoom, which is sent as a wheel+ctrlKey event\nfunction defaultFilter(event) {\n  return (!event.ctrlKey || event.type === 'wheel') && !event.button;\n}\n\nfunction defaultExtent() {\n  var e = this;\n  if (e instanceof SVGElement) {\n    e = e.ownerSVGElement || e;\n    if (e.hasAttribute(\"viewBox\")) {\n      e = e.viewBox.baseVal;\n      return [[e.x, e.y], [e.x + e.width, e.y + e.height]];\n    }\n    return [[0, 0], [e.width.baseVal.value, e.height.baseVal.value]];\n  }\n  return [[0, 0], [e.clientWidth, e.clientHeight]];\n}\n\nfunction defaultTransform() {\n  return this.__zoom || identity;\n}\n\nfunction defaultWheelDelta(event) {\n  return -event.deltaY * (event.deltaMode === 1 ? 0.05 : event.deltaMode ? 1 : 0.002) * (event.ctrlKey ? 10 : 1);\n}\n\nfunction defaultTouchable() {\n  return navigator.maxTouchPoints || (\"ontouchstart\" in this);\n}\n\nfunction defaultConstrain(transform, extent, translateExtent) {\n  var dx0 = transform.invertX(extent[0][0]) - translateExtent[0][0],\n      dx1 = transform.invertX(extent[1][0]) - translateExtent[1][0],\n      dy0 = transform.invertY(extent[0][1]) - translateExtent[0][1],\n      dy1 = transform.invertY(extent[1][1]) - translateExtent[1][1];\n  return transform.translate(\n    dx1 > dx0 ? (dx0 + dx1) / 2 : Math.min(0, dx0) || Math.max(0, dx1),\n    dy1 > dy0 ? (dy0 + dy1) / 2 : Math.min(0, dy0) || Math.max(0, dy1)\n  );\n}\n\nexport default function() {\n  var filter = defaultFilter,\n      extent = defaultExtent,\n      constrain = defaultConstrain,\n      wheelDelta = defaultWheelDelta,\n      touchable = defaultTouchable,\n      scaleExtent = [0, Infinity],\n      translateExtent = [[-Infinity, -Infinity], [Infinity, Infinity]],\n      duration = 250,\n      interpolate = interpolateZoom,\n      listeners = dispatch(\"start\", \"zoom\", \"end\"),\n      touchstarting,\n      touchfirst,\n      touchending,\n      touchDelay = 500,\n      wheelDelay = 150,\n      clickDistance2 = 0,\n      tapDistance = 10;\n\n  function zoom(selection) {\n    selection\n        .property(\"__zoom\", defaultTransform)\n        .on(\"wheel.zoom\", wheeled, {passive: false})\n        .on(\"mousedown.zoom\", mousedowned)\n        .on(\"dblclick.zoom\", dblclicked)\n      .filter(touchable)\n        .on(\"touchstart.zoom\", touchstarted)\n        .on(\"touchmove.zoom\", touchmoved)\n        .on(\"touchend.zoom touchcancel.zoom\", touchended)\n        .style(\"-webkit-tap-highlight-color\", \"rgba(0,0,0,0)\");\n  }\n\n  zoom.transform = function(collection, transform, point, event) {\n    var selection = collection.selection ? collection.selection() : collection;\n    selection.property(\"__zoom\", defaultTransform);\n    if (collection !== selection) {\n      schedule(collection, transform, point, event);\n    } else {\n      selection.interrupt().each(function() {\n        gesture(this, arguments)\n          .event(event)\n          .start()\n          .zoom(null, typeof transform === \"function\" ? transform.apply(this, arguments) : transform)\n          .end();\n      });\n    }\n  };\n\n  zoom.scaleBy = function(selection, k, p, event) {\n    zoom.scaleTo(selection, function() {\n      var k0 = this.__zoom.k,\n          k1 = typeof k === \"function\" ? k.apply(this, arguments) : k;\n      return k0 * k1;\n    }, p, event);\n  };\n\n  zoom.scaleTo = function(selection, k, p, event) {\n    zoom.transform(selection, function() {\n      var e = extent.apply(this, arguments),\n          t0 = this.__zoom,\n          p0 = p == null ? centroid(e) : typeof p === \"function\" ? p.apply(this, arguments) : p,\n          p1 = t0.invert(p0),\n          k1 = typeof k === \"function\" ? k.apply(this, arguments) : k;\n      return constrain(translate(scale(t0, k1), p0, p1), e, translateExtent);\n    }, p, event);\n  };\n\n  zoom.translateBy = function(selection, x, y, event) {\n    zoom.transform(selection, function() {\n      return constrain(this.__zoom.translate(\n        typeof x === \"function\" ? x.apply(this, arguments) : x,\n        typeof y === \"function\" ? y.apply(this, arguments) : y\n      ), extent.apply(this, arguments), translateExtent);\n    }, null, event);\n  };\n\n  zoom.translateTo = function(selection, x, y, p, event) {\n    zoom.transform(selection, function() {\n      var e = extent.apply(this, arguments),\n          t = this.__zoom,\n          p0 = p == null ? centroid(e) : typeof p === \"function\" ? p.apply(this, arguments) : p;\n      return constrain(identity.translate(p0[0], p0[1]).scale(t.k).translate(\n        typeof x === \"function\" ? -x.apply(this, arguments) : -x,\n        typeof y === \"function\" ? -y.apply(this, arguments) : -y\n      ), e, translateExtent);\n    }, p, event);\n  };\n\n  function scale(transform, k) {\n    k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], k));\n    return k === transform.k ? transform : new Transform(k, transform.x, transform.y);\n  }\n\n  function translate(transform, p0, p1) {\n    var x = p0[0] - p1[0] * transform.k, y = p0[1] - p1[1] * transform.k;\n    return x === transform.x && y === transform.y ? transform : new Transform(transform.k, x, y);\n  }\n\n  function centroid(extent) {\n    return [(+extent[0][0] + +extent[1][0]) / 2, (+extent[0][1] + +extent[1][1]) / 2];\n  }\n\n  function schedule(transition, transform, point, event) {\n    transition\n        .on(\"start.zoom\", function() { gesture(this, arguments).event(event).start(); })\n        .on(\"interrupt.zoom end.zoom\", function() { gesture(this, arguments).event(event).end(); })\n        .tween(\"zoom\", function() {\n          var that = this,\n              args = arguments,\n              g = gesture(that, args).event(event),\n              e = extent.apply(that, args),\n              p = point == null ? centroid(e) : typeof point === \"function\" ? point.apply(that, args) : point,\n              w = Math.max(e[1][0] - e[0][0], e[1][1] - e[0][1]),\n              a = that.__zoom,\n              b = typeof transform === \"function\" ? transform.apply(that, args) : transform,\n              i = interpolate(a.invert(p).concat(w / a.k), b.invert(p).concat(w / b.k));\n          return function(t) {\n            if (t === 1) t = b; // Avoid rounding error on end.\n            else { var l = i(t), k = w / l[2]; t = new Transform(k, p[0] - l[0] * k, p[1] - l[1] * k); }\n            g.zoom(null, t);\n          };\n        });\n  }\n\n  function gesture(that, args, clean) {\n    return (!clean && that.__zooming) || new Gesture(that, args);\n  }\n\n  function Gesture(that, args) {\n    this.that = that;\n    this.args = args;\n    this.active = 0;\n    this.sourceEvent = null;\n    this.extent = extent.apply(that, args);\n    this.taps = 0;\n  }\n\n  Gesture.prototype = {\n    event: function(event) {\n      if (event) this.sourceEvent = event;\n      return this;\n    },\n    start: function() {\n      if (++this.active === 1) {\n        this.that.__zooming = this;\n        this.emit(\"start\");\n      }\n      return this;\n    },\n    zoom: function(key, transform) {\n      if (this.mouse && key !== \"mouse\") this.mouse[1] = transform.invert(this.mouse[0]);\n      if (this.touch0 && key !== \"touch\") this.touch0[1] = transform.invert(this.touch0[0]);\n      if (this.touch1 && key !== \"touch\") this.touch1[1] = transform.invert(this.touch1[0]);\n      this.that.__zoom = transform;\n      this.emit(\"zoom\");\n      return this;\n    },\n    end: function() {\n      if (--this.active === 0) {\n        delete this.that.__zooming;\n        this.emit(\"end\");\n      }\n      return this;\n    },\n    emit: function(type) {\n      var d = select(this.that).datum();\n      listeners.call(\n        type,\n        this.that,\n        new ZoomEvent(type, {\n          sourceEvent: this.sourceEvent,\n          target: zoom,\n          type,\n          transform: this.that.__zoom,\n          dispatch: listeners\n        }),\n        d\n      );\n    }\n  };\n\n  function wheeled(event, ...args) {\n    if (!filter.apply(this, arguments)) return;\n    var g = gesture(this, args).event(event),\n        t = this.__zoom,\n        k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], t.k * Math.pow(2, wheelDelta.apply(this, arguments)))),\n        p = pointer(event);\n\n    // If the mouse is in the same location as before, reuse it.\n    // If there were recent wheel events, reset the wheel idle timeout.\n    if (g.wheel) {\n      if (g.mouse[0][0] !== p[0] || g.mouse[0][1] !== p[1]) {\n        g.mouse[1] = t.invert(g.mouse[0] = p);\n      }\n      clearTimeout(g.wheel);\n    }\n\n    // If this wheel event won’t trigger a transform change, ignore it.\n    else if (t.k === k) return;\n\n    // Otherwise, capture the mouse point and location at the start.\n    else {\n      g.mouse = [p, t.invert(p)];\n      interrupt(this);\n      g.start();\n    }\n\n    noevent(event);\n    g.wheel = setTimeout(wheelidled, wheelDelay);\n    g.zoom(\"mouse\", constrain(translate(scale(t, k), g.mouse[0], g.mouse[1]), g.extent, translateExtent));\n\n    function wheelidled() {\n      g.wheel = null;\n      g.end();\n    }\n  }\n\n  function mousedowned(event, ...args) {\n    if (touchending || !filter.apply(this, arguments)) return;\n    var currentTarget = event.currentTarget,\n        g = gesture(this, args, true).event(event),\n        v = select(event.view).on(\"mousemove.zoom\", mousemoved, true).on(\"mouseup.zoom\", mouseupped, true),\n        p = pointer(event, currentTarget),\n        x0 = event.clientX,\n        y0 = event.clientY;\n\n    dragDisable(event.view);\n    nopropagation(event);\n    g.mouse = [p, this.__zoom.invert(p)];\n    interrupt(this);\n    g.start();\n\n    function mousemoved(event) {\n      noevent(event);\n      if (!g.moved) {\n        var dx = event.clientX - x0, dy = event.clientY - y0;\n        g.moved = dx * dx + dy * dy > clickDistance2;\n      }\n      g.event(event)\n       .zoom(\"mouse\", constrain(translate(g.that.__zoom, g.mouse[0] = pointer(event, currentTarget), g.mouse[1]), g.extent, translateExtent));\n    }\n\n    function mouseupped(event) {\n      v.on(\"mousemove.zoom mouseup.zoom\", null);\n      dragEnable(event.view, g.moved);\n      noevent(event);\n      g.event(event).end();\n    }\n  }\n\n  function dblclicked(event, ...args) {\n    if (!filter.apply(this, arguments)) return;\n    var t0 = this.__zoom,\n        p0 = pointer(event.changedTouches ? event.changedTouches[0] : event, this),\n        p1 = t0.invert(p0),\n        k1 = t0.k * (event.shiftKey ? 0.5 : 2),\n        t1 = constrain(translate(scale(t0, k1), p0, p1), extent.apply(this, args), translateExtent);\n\n    noevent(event);\n    if (duration > 0) select(this).transition().duration(duration).call(schedule, t1, p0, event);\n    else select(this).call(zoom.transform, t1, p0, event);\n  }\n\n  function touchstarted(event, ...args) {\n    if (!filter.apply(this, arguments)) return;\n    var touches = event.touches,\n        n = touches.length,\n        g = gesture(this, args, event.changedTouches.length === n).event(event),\n        started, i, t, p;\n\n    nopropagation(event);\n    for (i = 0; i < n; ++i) {\n      t = touches[i], p = pointer(t, this);\n      p = [p, this.__zoom.invert(p), t.identifier];\n      if (!g.touch0) g.touch0 = p, started = true, g.taps = 1 + !!touchstarting;\n      else if (!g.touch1 && g.touch0[2] !== p[2]) g.touch1 = p, g.taps = 0;\n    }\n\n    if (touchstarting) touchstarting = clearTimeout(touchstarting);\n\n    if (started) {\n      if (g.taps < 2) touchfirst = p[0], touchstarting = setTimeout(function() { touchstarting = null; }, touchDelay);\n      interrupt(this);\n      g.start();\n    }\n  }\n\n  function touchmoved(event, ...args) {\n    if (!this.__zooming) return;\n    var g = gesture(this, args).event(event),\n        touches = event.changedTouches,\n        n = touches.length, i, t, p, l;\n\n    noevent(event);\n    for (i = 0; i < n; ++i) {\n      t = touches[i], p = pointer(t, this);\n      if (g.touch0 && g.touch0[2] === t.identifier) g.touch0[0] = p;\n      else if (g.touch1 && g.touch1[2] === t.identifier) g.touch1[0] = p;\n    }\n    t = g.that.__zoom;\n    if (g.touch1) {\n      var p0 = g.touch0[0], l0 = g.touch0[1],\n          p1 = g.touch1[0], l1 = g.touch1[1],\n          dp = (dp = p1[0] - p0[0]) * dp + (dp = p1[1] - p0[1]) * dp,\n          dl = (dl = l1[0] - l0[0]) * dl + (dl = l1[1] - l0[1]) * dl;\n      t = scale(t, Math.sqrt(dp / dl));\n      p = [(p0[0] + p1[0]) / 2, (p0[1] + p1[1]) / 2];\n      l = [(l0[0] + l1[0]) / 2, (l0[1] + l1[1]) / 2];\n    }\n    else if (g.touch0) p = g.touch0[0], l = g.touch0[1];\n    else return;\n\n    g.zoom(\"touch\", constrain(translate(t, p, l), g.extent, translateExtent));\n  }\n\n  function touchended(event, ...args) {\n    if (!this.__zooming) return;\n    var g = gesture(this, args).event(event),\n        touches = event.changedTouches,\n        n = touches.length, i, t;\n\n    nopropagation(event);\n    if (touchending) clearTimeout(touchending);\n    touchending = setTimeout(function() { touchending = null; }, touchDelay);\n    for (i = 0; i < n; ++i) {\n      t = touches[i];\n      if (g.touch0 && g.touch0[2] === t.identifier) delete g.touch0;\n      else if (g.touch1 && g.touch1[2] === t.identifier) delete g.touch1;\n    }\n    if (g.touch1 && !g.touch0) g.touch0 = g.touch1, delete g.touch1;\n    if (g.touch0) g.touch0[1] = this.__zoom.invert(g.touch0[0]);\n    else {\n      g.end();\n      // If this was a dbltap, reroute to the (optional) dblclick.zoom handler.\n      if (g.taps === 2) {\n        t = pointer(t, this);\n        if (Math.hypot(touchfirst[0] - t[0], touchfirst[1] - t[1]) < tapDistance) {\n          var p = select(this).on(\"dblclick.zoom\");\n          if (p) p.apply(this, arguments);\n        }\n      }\n    }\n  }\n\n  zoom.wheelDelta = function(_) {\n    return arguments.length ? (wheelDelta = typeof _ === \"function\" ? _ : constant(+_), zoom) : wheelDelta;\n  };\n\n  zoom.filter = function(_) {\n    return arguments.length ? (filter = typeof _ === \"function\" ? _ : constant(!!_), zoom) : filter;\n  };\n\n  zoom.touchable = function(_) {\n    return arguments.length ? (touchable = typeof _ === \"function\" ? _ : constant(!!_), zoom) : touchable;\n  };\n\n  zoom.extent = function(_) {\n    return arguments.length ? (extent = typeof _ === \"function\" ? _ : constant([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;\n  };\n\n  zoom.scaleExtent = function(_) {\n    return arguments.length ? (scaleExtent[0] = +_[0], scaleExtent[1] = +_[1], zoom) : [scaleExtent[0], scaleExtent[1]];\n  };\n\n  zoom.translateExtent = function(_) {\n    return arguments.length ? (translateExtent[0][0] = +_[0][0], translateExtent[1][0] = +_[1][0], translateExtent[0][1] = +_[0][1], translateExtent[1][1] = +_[1][1], zoom) : [[translateExtent[0][0], translateExtent[0][1]], [translateExtent[1][0], translateExtent[1][1]]];\n  };\n\n  zoom.constrain = function(_) {\n    return arguments.length ? (constrain = _, zoom) : constrain;\n  };\n\n  zoom.duration = function(_) {\n    return arguments.length ? (duration = +_, zoom) : duration;\n  };\n\n  zoom.interpolate = function(_) {\n    return arguments.length ? (interpolate = _, zoom) : interpolate;\n  };\n\n  zoom.on = function() {\n    var value = listeners.on.apply(listeners, arguments);\n    return value === listeners ? zoom : value;\n  };\n\n  zoom.clickDistance = function(_) {\n    return arguments.length ? (clickDistance2 = (_ = +_) * _, zoom) : Math.sqrt(clickDistance2);\n  };\n\n  zoom.tapDistance = function(_) {\n    return arguments.length ? (tapDistance = +_, zoom) : tapDistance;\n  };\n\n  return zoom;\n}\n","import { select } from 'd3-selection'\n\n/**\n * Parent class for button elements used to interact with the gridviz viewer.\n *\n * @module button\n * @author Joseph Davies, Julien Gaffuri\n */\nexport class Button {\n    /**\n     * @param {Object} opts\n     * opts.parentNode\n     * opts.id\n     * opts.title\n     * opts.class\n     * opts.onClickFunction\n     * opts.x\n     * opts.y\n     */\n    constructor(opts = {}) {\n        this.map = opts.map\n        this.parentNode = opts.parentNode || opts.map.container\n\n        // the div element\n        if (opts.id) this.div = select('#' + opts.id)\n\n        if (!this.div || this.div.empty()) {\n            this.div = select(document.createElement('div'))\n            if (opts.id) this.div.attr('id', opts.id)\n        }\n\n        if (opts.title) this.div.attr('title', opts.title)\n        if (opts.class) this.div.attr('class', opts.class)\n\n        // add events\n        if (opts.onClickFunction) this.div.on('click', opts.onClickFunction)\n\n        //set styles\n        this.style(\n            'box-shadow',\n            '0 7px 8px rgba(0,47,103,.08), 0 0 22px rgba(0,47,103,.04), 0 12px 17px rgba(0,47,103,.04), 0 -4px 4px rgba(0,47,103,.04)'\n        ) //.ecl-u-shadow-3\n        this.style('background-color', '#ffffff')\n        this.style('position', 'absolute')\n        this.style('cursor', 'pointer')\n        this.style('display', 'flex')\n        this.style('justify-content', 'center')\n        this.style('align-items', 'center')\n        this.style('width', '35px')\n        this.style('height', '30px')\n        // this.style(padding , '4px'\n\n        // append to parent\n        this.parentNode.appendChild(this.div.node())\n    }\n\n    /**\n     * Apply a style to the button div.\n     * @param {string} k\n     * @param {string} v\n     * @returns {this}\n     */\n    style(k, v) {\n        this.div.style(k, v)\n        return this\n    }\n}\n","import { Button } from './Button.js'\n\n/**\n * Button for toggling fullscreen mode\n *\n * @module button\n * @author Joseph Davies, Julien Gaffuri\n */\nexport class FullscreenButton extends Button {\n    /**\n     * @param {Object} opts\n     * opts.parentNode - the node that the button is appended to\n     * opts.canvas - the gridviz canvas\n     * opts.id\n     * opts.title - HTML title attribute\n     * opts.class - css class\n     * opts.onClickFunction\n     * opts.x - x position of the button\n     * opts.y - y position of the button\n     */\n\n    // default state\n    isFullscreen = false\n\n    constructor(opts) {\n        super(opts)\n\n        // append fullscreen icon to button container\n        this.div.node().innerHTML = `\n        <svg\n            style=\"height: 1.2rem; width: 1.2rem; fill:black; margin:0;\"\n            focusable=\"false\"\n            aria-hidden=\"true\"\n        >\n            <svg fill=\"#000000\" viewBox=\"0 0 96 96\" xmlns=\"http://www.w3.org/2000/svg\">\n            <title/>\n            <g>\n            <path d=\"M30,0H6A5.9966,5.9966,0,0,0,0,6V30a6,6,0,0,0,12,0V12H30A6,6,0,0,0,30,0Z\"/>\n            <path d=\"M90,0H66a6,6,0,0,0,0,12H84V30a6,6,0,0,0,12,0V6A5.9966,5.9966,0,0,0,90,0Z\"/>\n            <path d=\"M30,84H12V66A6,6,0,0,0,0,66V90a5.9966,5.9966,0,0,0,6,6H30a6,6,0,0,0,0-12Z\"/>\n            <path d=\"M90,60a5.9966,5.9966,0,0,0-6,6V84H66a6,6,0,0,0,0,12H90a5.9966,5.9966,0,0,0,6-6V66A5.9966,5.9966,0,0,0,90,60Z\"/>\n            </g>\n            </svg>\n        </svg>\n        `\n\n        //save initial map dimensions\n        this.defaultHeight = this.map.h\n        this.defaultWidth = this.map.w\n\n        // event handler\n        this.div.on('click', (e) => {\n            this.onClickFunction(e)\n        })\n        this.div.on('mouseover', (e) => {\n            this.style('background-color', 'lightgrey')\n        })\n        this.div.on('mouseout', (e) => {\n            this.style('background-color', '#ffffff')\n        })\n\n        //set position\n        if (opts.x) {\n            this.style('left', opts.x + 'px')\n        } else {\n            this.style('right', '10px')\n        }\n        if (opts.y) {\n            this.style('top', opts.y + 'px')\n        } else {\n            this.style('top', '90px')\n        }\n    }\n\n    onClickFunction(e) {\n        if (this.isFullscreen) {\n            this.closeFullscreen(this.map.container)\n            //resize canvas to default\n            this.map.h = this.defaultHeight\n            this.map.w = this.defaultWidth\n            this.map.geoCanvas.h = this.defaultHeight\n            this.map.geoCanvas.w = this.defaultWidth\n            this.map.geoCanvas.canvas.setAttribute('width', '' + this.defaultWidth)\n            this.map.geoCanvas.canvas.setAttribute('height', '' + this.defaultHeight)\n            this.map.redraw()\n            this.isFullscreen = false\n        } else {\n            this.openFullscreen(this.map.container)\n            //resize canvas to fullscreen\n            this.map.h = window.screen.height\n            this.map.w = window.screen.width\n            this.isFullscreen = true\n        }\n    }\n\n    /* Open fullscreen */\n    openFullscreen(elem) {\n        if (elem.requestFullscreen) {\n            elem.requestFullscreen()\n        } else if (elem.webkitRequestFullscreen) {\n            /* Safari */\n            elem.webkitRequestFullscreen()\n        } else if (elem.msRequestFullscreen) {\n            /* IE11 */\n            elem.msRequestFullscreen()\n        }\n    }\n\n    /* Close fullscreen */\n    closeFullscreen() {\n        if (document.exitFullscreen) {\n            document.exitFullscreen()\n        } else if (document.webkitExitFullscreen) {\n            /* Safari */\n            document.webkitExitFullscreen()\n        } else if (document.msExitFullscreen) {\n            /* IE11 */\n            document.msExitFullscreen()\n        }\n    }\n}\n","import { Button } from './Button.js'\r\n\r\n/**\r\n * Button for toggling fullscreen mode\r\n *\r\n * @module button\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class ZoomButtons extends Button {\r\n    /**\r\n     * @param {Object} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n\r\n        this.onZoom = opts.onZoom // custom user event handler\r\n        this.delta = opts.delta || 0.2\r\n\r\n        // Create zoom in button\r\n        this.zoomInBtn = document.createElement('a')\r\n        this.zoomInBtn.id = 'zoom-in'\r\n        this.zoomInBtn.className = 'gridviz-zoom-button'\r\n        this.zoomInBtn.title = 'Zoom in'\r\n        this.zoomInBtn.textContent = '+'\r\n        this.zoomInBtn.addEventListener('click', (e) => {\r\n            this.zoomIn(e)\r\n        })\r\n        this.zoomInBtn.addEventListener('mouseover', () => {\r\n            this.zoomInBtn.style.backgroundColor = 'lightgrey'\r\n        })\r\n        this.zoomInBtn.addEventListener('mouseout', () => {\r\n            this.zoomInBtn.style.backgroundColor = '#ffffff'\r\n        })\r\n\r\n        // Create zoom out button\r\n        this.zoomOutBtn = document.createElement('a')\r\n        this.zoomOutBtn.id = 'zoom-out'\r\n        this.zoomOutBtn.className = 'gridviz-zoom-button'\r\n        this.zoomOutBtn.title = 'Zoom out'\r\n        this.zoomOutBtn.textContent = '-'\r\n        this.zoomOutBtn.addEventListener('click', (e) => {\r\n            this.zoomOut(e)\r\n        })\r\n        this.zoomOutBtn.addEventListener('mouseover', () => {\r\n            this.zoomOutBtn.style.backgroundColor = 'lightgrey'\r\n        })\r\n        this.zoomOutBtn.addEventListener('mouseout', () => {\r\n            this.zoomOutBtn.style.backgroundColor = '#ffffff'\r\n        })\r\n\r\n        // Set common styles for buttons\r\n        const buttons = [this.zoomInBtn, this.zoomOutBtn]\r\n        buttons.forEach((btn, index) => {\r\n            btn.style.alignItems = 'center'\r\n            btn.style.justifyContent = 'center'\r\n            btn.style.display = 'flex'\r\n            btn.style.border = 'none'\r\n            btn.style.color = 'black'\r\n            btn.style.textAlign = 'center'\r\n            btn.style.textDecoration = 'none'\r\n            btn.style.padding = '4px'\r\n            btn.style.fontSize = '24px'\r\n            btn.style.fontWeight = 'bold'\r\n            btn.style.userSelect = 'none'\r\n            btn.style.backgroundColor = '#ffffff'\r\n            if (index === 0) btn.style.borderBottom = '1px solid grey' // Zoom in button only\r\n        })\r\n\r\n        // Unset parent class height and display for dual buttons\r\n        this.style('height', 'unset')\r\n        this.style('display', 'unset')\r\n\r\n        // Set position\r\n        if (opts.x) {\r\n            this.style('left', opts.x + 'px')\r\n        } else {\r\n            this.style('right', '10px')\r\n        }\r\n        if (opts.y) {\r\n            this.style('top', opts.y + 'px')\r\n        } else {\r\n            this.style('top', '10px')\r\n        }\r\n\r\n        // Append buttons to the container\r\n        this.div.node().appendChild(this.zoomInBtn)\r\n        this.div.node().appendChild(this.zoomOutBtn)\r\n    }\r\n\r\n    /* Zoom in */\r\n    zoomIn(e) {\r\n        this.map.setZoom(this.map.getZoom() * (1 - this.delta)).redraw()\r\n        if (this.onZoom) this.onZoom(e)\r\n    }\r\n\r\n    /* Zoom out */\r\n    zoomOut(e) {\r\n        this.map.setZoom(this.map.getZoom() * (1 + this.delta)).redraw()\r\n        if (this.onZoom) this.onZoom(e)\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n/**\r\n * A grid cell.\r\n * @typedef {{x: number, y: number}} Cell */\r\n\r\n/**\r\n * A dataset component, of grid cells.\r\n * @abstract\r\n *\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class Dataset {\r\n    /**\r\n     * @param {import(\"./Map.js\").Map} map The map.\r\n     * @param {string} url The URL of the dataset.\r\n     * @param {number} resolution The dataset resolution, in the CRS geographical unit.\r\n     * @param {{preprocess?:function(Cell):boolean, mixedResolution?:function(Cell):number}} opts\r\n     * @abstract\r\n     */\r\n    constructor(map, url, resolution, opts = {}) {\r\n        /**\r\n         * The map.\r\n         * @protected\r\n         * @type {import(\"./Map.js\").Map} */\r\n        this.map = map\r\n\r\n        /**\r\n         * The url of the dataset.\r\n         * @protected\r\n         * @type {string} */\r\n        this.url = url\r\n\r\n        /**\r\n         * The dataset resolution in geographical unit.\r\n         * @protected\r\n         * @type {number} */\r\n        this.resolution = resolution\r\n\r\n        /**\r\n         * In case the dataset is a dataset with cells having different resolution,\r\n         * this is the function returning the resolution of each cell.\r\n         * @protected\r\n         * @type {(function(Cell):number )| undefined } */\r\n        this.mixedResolution = opts.mixedResolution\r\n\r\n        /**\r\n         * A preprocess to run on each cell after loading. It can be used to apply some specific treatment before or compute a new column. And also to determine which cells to keep after loading.\r\n         * @type {(function(Cell):boolean )| undefined } */\r\n        this.preprocess = opts.preprocess || undefined\r\n\r\n        /** The cells within the view\r\n         * @protected\r\n         * @type {Array.<Cell>} */\r\n        this.cellsViewCache = []\r\n    }\r\n\r\n    /**\r\n     * Request data within a geographic envelope.\r\n     *\r\n     * @abstract\r\n     * @param {import(\"./GeoCanvas\").Envelope|undefined} extGeo\r\n     * @returns {this}\r\n     */\r\n    getData(extGeo = undefined) {\r\n        throw new Error('Method getData not implemented.')\r\n    }\r\n\r\n    /**\r\n     * Fill the view cache with all cells which are within a geographical envelope.\r\n     * @abstract\r\n     * @param {import(\"./GeoCanvas\").Envelope} extGeo The view geographical envelope.\r\n     * @returns {void}\r\n     */\r\n    updateViewCache(extGeo) {\r\n        throw new Error('Method updateViewCache not implemented.')\r\n    }\r\n\r\n    /**\r\n     * Get a cell under a given position, if any.\r\n     *\r\n     * @param {{x:number,y:number}} posGeo\r\n     * @param {Array.<Cell>} cells Some cells from the dataset (a subset if necessary, usually the view cache).\r\n     * @returns {Cell|undefined}\r\n     */\r\n    getCellFromPosition(posGeo, cells) {\r\n        //compute candidate cell position\r\n        /** @type {number} */\r\n        //const r = this.getResolution()\r\n        /** @type {number} */\r\n        //const cellX = r * Math.floor(posGeo.x / r)\r\n        /** @type {number} */\r\n        //const cellY = r * Math.floor(posGeo.y / r)\r\n\r\n        /*/get cell\r\n        for (const cell of cells) {\r\n            if (cell.x != cellX) continue\r\n            if (cell.y != cellY) continue\r\n            return cell\r\n        }\r\n        return undefined*/\r\n\r\n        //rare case of mixed resolution dataset\r\n        if (this.mixedResolution) {\r\n            for (const c of cells) {\r\n                /** @type {number} */\r\n                const r = +this.mixedResolution(c)\r\n                if (posGeo.x < c.x) continue\r\n                else if (c.x + r < posGeo.x) continue\r\n                else if (posGeo.y < c.y) continue\r\n                else if (c.y + r < posGeo.y) continue\r\n                else return c\r\n            }\r\n            return undefined\r\n        }\r\n\r\n        //common case\r\n\r\n        /** @type {number} */\r\n        const r = this.getResolution()\r\n        for (const cell of cells) {\r\n            if (posGeo.x < cell.x) continue\r\n            else if (cell.x + r < posGeo.x) continue\r\n            else if (posGeo.y < cell.y) continue\r\n            else if (cell.y + r < posGeo.y) continue\r\n            else return cell\r\n        }\r\n        return undefined\r\n    }\r\n\r\n    //getters and setters\r\n\r\n    /** @returns {number} */\r\n    getResolution() {\r\n        return this.resolution\r\n    }\r\n\r\n    /** @returns {Array.<Cell>} */\r\n    getViewCache() {\r\n        return this.cellsViewCache\r\n    }\r\n\r\n    /**\r\n     * Return the relevant dataset for a specified zoom.\r\n     * @param {number} z\r\n     * @param {number} minPixelsPerCell\r\n     * @returns {Dataset|undefined}\r\n     * */\r\n    getDataset(z, minPixelsPerCell) {\r\n        return this\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n/**\r\n * This is an abstract class used to group elements shared between Layer and Style classes.\r\n *\r\n * @abstract\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class Drawable {\r\n    /**\r\n     * @param {object} opts\r\n     */\r\n    constructor(opts) {\r\n        opts = opts || {}\r\n\r\n        /** A function specifying if the element should be visible or not.\r\n         * The function parameter is the zoom level.\r\n         * @type {function(number):boolean} */\r\n        this.visible = opts.visible\r\n\r\n        /** A function returning the alpha (transparency/opacity), between 0.0 (fully transparent) and 1.0 (fully opaque).\r\n         *  The function parameter is the zoom level.\r\n         * (see CanvasRenderingContext2D: globalAlpha property)\r\n         * @type {(function(number):number)|undefined} */\r\n        this.alpha = opts.alpha\r\n\r\n        /** A function returning the blend operation.\r\n         * The function parameter is the zoom level.\r\n         * (see CanvasRenderingContext2D: globalCompositeOperation property)\r\n         * @type {function(number):GlobalCompositeOperation} */\r\n        this.blendOperation = opts.blendOperation || ((z) => 'source-over')\r\n\r\n        /** @type {(function(number):string)|undefined} */\r\n        this.filterColor = opts.filterColor // (z) => \"#eee7\"\r\n    }\r\n\r\n    /**\r\n     * Draw layer filter.\r\n     *\r\n     * @param {import(\"./GeoCanvas.js\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @returns {void}\r\n     * @abstract\r\n     */\r\n    drawFilter(geoCanvas) {\r\n        //no filter: return\r\n        if (!this.filterColor) return\r\n\r\n        //get filter\r\n        const fc = this.filterColor(geoCanvas.view.z)\r\n\r\n        //no filter: return\r\n        if (!fc || fc == 'none') return\r\n\r\n        //draw filter\r\n        geoCanvas.offscreenCtx.fillStyle = fc\r\n        geoCanvas.offscreenCtx.fillRect(0, 0, geoCanvas.w, geoCanvas.h)\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n/** @typedef { {xMin: number, xMax: number, yMin: number, yMax: number} } Envelope */\r\n\r\n/**\r\n * A viewshed.\r\n * @typedef {{x: number, y: number, z: number}} View */\r\n\r\nimport { select } from 'd3-selection'\r\nimport { zoom as d3zoom, zoomIdentity } from 'd3-zoom'\r\n\r\n/**\r\n * A HTML canvas for geo data display, enhanced with zoom and pan capabilities.\r\n *\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class GeoCanvas {\r\n    /**\r\n     * @constructor\r\n     * @param {HTMLCanvasElement} canvas\r\n     * @param {number} x The x coordinate of the view\r\n     * @param {number} y The y coordinate of the view\r\n     * @param {number} z The zoom level of the view (pixel size, in ground m)\r\n     * @param {object} opts\r\n     */\r\n    constructor(canvas, x = 0, y = 0, z = 0, opts = undefined) {\r\n        this.opts = opts || {}\r\n\r\n        /** @type {HTMLCanvasElement} */\r\n        this.canvas = canvas\r\n\r\n        /** @type {number} */\r\n        this.w = this.canvas.offsetWidth\r\n        /** @type {number} */\r\n        this.h = this.canvas.offsetHeight\r\n\r\n        // Adjust canvas width and height based on device pixel ratio\r\n        //const dpr = window.devicePixelRatio || 1 // Get the device pixel ratio\r\n        //this.canvas.width = this.w * dpr // Set canvas width\r\n        //\\sthis.canvas.height = this.h * dpr // Set canvas height\r\n\r\n        // Create offscreen canvas for drawing operations\r\n        this.offscreenCanvas = document.createElement('canvas')\r\n        this.offscreenCanvas.width = this.w\r\n        this.offscreenCanvas.height = this.h\r\n\r\n        const ctx = this.canvas.getContext('2d')\r\n        const offscreenCtx = this.offscreenCanvas.getContext('2d')\r\n        if (!ctx) throw 'Impossible to create canvas 2D context'\r\n        if (!offscreenCtx) throw 'Impossible to create canvas 2D context'\r\n        /**@type {CanvasRenderingContext2D} */\r\n        this.ctx = ctx\r\n        this.offscreenCtx = offscreenCtx\r\n        //this.ctx.scale(dpr, dpr) // Scale the context\r\n\r\n        /**\r\n         * z: pixel size, in m/pix\r\n         * @type {View}  */\r\n        this.view = { x: x, y: y, z: z }\r\n\r\n        /** Background color.\r\n         * @type {string} */\r\n        this.backgroundColor = opts.backgroundColor || 'white'\r\n\r\n        /** @type {function(object|undefined):void} */\r\n        this.onZoomStartFun = opts.onZoomStartFun\r\n\r\n        /** @type {function(object|undefined):void} */\r\n        this.onZoomEndFun = opts.onZoomEndFun\r\n\r\n        /** @type {function(object|undefined):void} */\r\n        this.onZoomFun = opts.onZoomFun\r\n\r\n        //current extent\r\n        /** @type {Envelope} */\r\n        this.extGeo = { xMin: NaN, xMax: NaN, yMin: NaN, yMax: NaN }\r\n        this.updateExtentGeo()\r\n\r\n        //rely on d3 zoom for pan/zoom\r\n        if (!opts.disableZoom) {\r\n            let tP = zoomIdentity\r\n            const z = d3zoom()\r\n                //to make the zooming a bit faster\r\n                .wheelDelta((e) => -e.deltaY * (e.deltaMode === 1 ? 0.07 : e.deltaMode ? 1 : 0.004))\r\n                .on('zoom', (e) => {\r\n                    const t = e.transform\r\n                    const f = tP.k / t.k\r\n\r\n                    // Get container's bounding rect to adjust for offsets\r\n                    const containerRect = this.canvas.getBoundingClientRect()\r\n\r\n                    // Adjust for container's offset and zoom center\r\n                    const offsetX = e.sourceEvent.offsetX - containerRect.left\r\n                    const offsetY = e.sourceEvent.offsetY - containerRect.top\r\n\r\n                    if (f === 1) {\r\n                        // Pan logic\r\n                        const dx = tP.x - t.x\r\n                        const dy = tP.y - t.y\r\n                        this.pan(dx * this.view.z, -dy * this.view.z)\r\n                    } else {\r\n                        const se = e.sourceEvent\r\n\r\n                        if (se instanceof WheelEvent) {\r\n                            // Zoom at mouse position, adjusted by container offset\r\n                            this.zoom(f, this.pixToGeoX(offsetX), this.pixToGeoY(offsetY))\r\n                        } else if (se instanceof TouchEvent) {\r\n                            if (!se.targetTouches.length) return\r\n\r\n                            // Compute average position of the touches\r\n                            let tx = 0,\r\n                                ty = 0\r\n                            for (let tt of se.targetTouches) {\r\n                                tx += tt.clientX\r\n                                ty += tt.clientY\r\n                            }\r\n                            tx /= se.targetTouches.length\r\n                            ty /= se.targetTouches.length\r\n\r\n                            // Adjust for container's offset\r\n                            tx -= containerRect.left\r\n                            ty -= containerRect.top\r\n\r\n                            // Zoom at the average touch position\r\n                            this.zoom(f, this.pixToGeoX(tx), this.pixToGeoY(ty))\r\n                        }\r\n                    }\r\n                    tP = t\r\n\r\n                    if (this.onZoomFun) this.onZoomFun(e)\r\n                })\r\n                .on('start', (e) => {\r\n                    // start of zoom event\r\n                    // save the current canvas state to keep onscreen during pan/zoom before redrawing\r\n                    this.canvasSave.c = document.createElement('canvas')\r\n                    this.canvasSave.c.setAttribute('width', '' + this.w)\r\n                    this.canvasSave.c.setAttribute('height', '' + this.h)\r\n                    this.canvasSave.c.getContext('2d')?.drawImage(this.canvas, 0, 0)\r\n                    this.canvasSave.dx = 0\r\n                    this.canvasSave.dy = 0\r\n                    this.canvasSave.f = 1\r\n                    if (this.onZoomStartFun) this.onZoomStartFun(e)\r\n                })\r\n                .on('end', (e) => {\r\n                    // end of zoom event\r\n                    this.redraw()\r\n                    this.canvasSave = { c: null, dx: 0, dy: 0, f: 1 }\r\n\r\n                    if (this.onZoomEndFun) this.onZoomEndFun(e)\r\n                })\r\n            z(select(this.canvas))\r\n        }\r\n\r\n        //center extent\r\n        /** @type {number|undefined} */\r\n        this.xMin = opts.centerExtent ? opts.centerExtent[0] : undefined\r\n        /** @type {number|undefined} */\r\n        this.yMin = opts.centerExtent ? opts.centerExtent[1] : undefined\r\n        /** @type {number|undefined} */\r\n        this.xMax = opts.centerExtent ? opts.centerExtent[2] : undefined\r\n        /** @type {number|undefined} */\r\n        this.yMax = opts.centerExtent ? opts.centerExtent[3] : undefined\r\n\r\n        /** Zoom extent, to limit zoom in and out\r\n         *  @type {Array.<number>} */\r\n        this.zoomExtent = opts.zoomExtent || [0, Infinity]\r\n\r\n        /** Canvas state, to be used to avoid unnecessary redraws on zoom/pan\r\n         *  @type {{c:HTMLCanvasElement|null,dx:number,dy:number,f:number}} */\r\n        this.canvasSave = { c: null, dx: 0, dy: 0, f: 1 }\r\n    }\r\n\r\n    /** @returns {View} */\r\n    getView() {\r\n        return this.view\r\n    }\r\n\r\n    /** @param {Array.<number>} v */\r\n    setCenterExtent(v) {\r\n        this.xMin = v[0]\r\n        this.yMin = v[1]\r\n        this.xMax = v[2]\r\n        this.yMax = v[3]\r\n    }\r\n    /** @returns {Array.<number|undefined>} */\r\n    getCenterExtent() {\r\n        return [this.xMin, this.yMin, this.xMax, this.yMax]\r\n    }\r\n\r\n    /** @param {Array.<number>} v */\r\n    setZoomExtent(v) {\r\n        this.zoomExtent = v\r\n    }\r\n    /** @returns {Array.<number>} */\r\n    getZoomExtent() {\r\n        return this.zoomExtent\r\n    }\r\n\r\n    /** Initialise canvas transform with identity transformation. */\r\n    initCanvasTransform() {\r\n        this.ctx.setTransform(1, 0, 0, 1, 0, 0)\r\n        this.offscreenCtx.setTransform(1, 0, 0, 1, 0, 0)\r\n    }\r\n\r\n    /** Initialise canvas transform with geo to screen transformation, so that geo objects can be drawn directly in geo coordinates. */\r\n    setCanvasTransform() {\r\n        const k = 1 / this.view.z\r\n        const tx = -this.view.x / this.view.z + this.w * 0.5\r\n        const ty = this.view.y / this.view.z + this.h * 0.5\r\n        this.ctx.setTransform(k, 0, 0, -k, tx, ty)\r\n        this.offscreenCtx.setTransform(k, 0, 0, -k, tx, ty)\r\n    }\r\n\r\n    /** Get the transformation matrix to webGL screen coordinates, within [-1,1]*[-1,1] */\r\n    getWebGLTransform() {\r\n        const kx = 2.0 / (this.w * this.view.z)\r\n        const ky = 2.0 / (this.h * this.view.z)\r\n        return [kx, 0.0, 0.0, 0.0, ky, 0.0, -kx * this.view.x, -ky * this.view.y, 1.0]\r\n    }\r\n\r\n    /** The function specifying how to draw the map. */\r\n    redraw() {\r\n        throw new Error('Method redraw not implemented.')\r\n    }\r\n\r\n    /**\r\n     * Clear. To be used before a redraw for example.\r\n     * @param {string} color\r\n     */\r\n    clear(color = 'white') {\r\n        if (this.opts.transparentBackground) {\r\n            this.ctx.clearRect(0, 0, this.w, this.h)\r\n            this.offscreenCtx.clearRect(0, 0, this.w, this.h)\r\n        } else {\r\n            if (this.ctx) this.ctx.fillStyle = color\r\n            if (this.offscreenCtx) this.offscreenCtx.fillStyle = color\r\n            this.ctx.fillRect(0, 0, this.w, this.h)\r\n            this.offscreenCtx.fillRect(0, 0, this.w, this.h)\r\n        }\r\n    }\r\n\r\n    /**\r\n     * @param {number} dxGeo\r\n     * @param {number} dyGeo\r\n     */\r\n    pan(dxGeo = 0, dyGeo = 0) {\r\n        //ensures x/y extent\r\n        if (this.xMin != undefined && this.view.x + dxGeo < this.xMin) dxGeo = this.xMin - this.view.x\r\n        if (this.yMin != undefined && this.view.y + dyGeo < this.yMin) dyGeo = this.yMin - this.view.y\r\n        if (this.xMax != undefined && this.view.x + dxGeo > this.xMax) dxGeo = this.xMax - this.view.x\r\n        if (this.yMax != undefined && this.view.y + dyGeo > this.yMax) dyGeo = this.yMax - this.view.y\r\n\r\n        //pan\r\n        this.view.x += dxGeo\r\n        this.view.y += dyGeo\r\n        this.updateExtentGeo()\r\n\r\n        if (this.canvasSave.c) {\r\n            const scale = 1 / this.view.z\r\n\r\n            // Update saved canvas offset\r\n            this.canvasSave.dx -= dxGeo * scale\r\n            this.canvasSave.dy += dyGeo * scale\r\n\r\n            // clear canvas\r\n            this.clear(this.backgroundColor)\r\n\r\n            // this doesnt work on mobile https://github.com/eurostat/gridviz/issues/98\r\n            //this.ctx.drawImage(this.canvasSave.c, this.canvasSave.dx, this.canvasSave.dy)\r\n            this.offscreenCtx.drawImage(this.canvasSave.c, this.canvasSave.dx, this.canvasSave.dy)\r\n\r\n            // Render the offscreen canvas to the visible context\r\n            // this.clear(this.backgroundColor)\r\n            this.ctx.drawImage(this.offscreenCtx.canvas, 0, 0)\r\n        } else {\r\n            console.log('no canvas save')\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Zoom.\r\n     * @param {number} f The zoom factor, within ]0, Infinity]. 1 is for no change. <1 to zoom-in, >1 to zoom-out.\r\n     * @param {number} xGeo The x geo position fixed in the screen.\r\n     * @param {number} yGeo The y geo position fixed in the screen.\r\n     */\r\n    zoom(f = 1, xGeo = this.view.x, yGeo = this.view.y) {\r\n        //TODO force geo extend to remain\r\n\r\n        //trying to zoom in/out beyond limit\r\n        if (this.zoomExtent[0] == this.view.z && f <= 1) return\r\n        if (this.zoomExtent[1] == this.view.z && f >= 1) return\r\n\r\n        //ensure zoom extent preserved\r\n        const newZf = f * this.view.z\r\n        if (newZf < this.zoomExtent[0]) f = this.zoomExtent[0] / this.view.z\r\n        if (newZf > this.zoomExtent[1]) f = this.zoomExtent[1] / this.view.z\r\n\r\n        this.view.z *= f\r\n\r\n        //compute pan\r\n        let dxGeo = (xGeo - this.view.x) * (1 - f)\r\n        let dyGeo = (yGeo - this.view.y) * (1 - f)\r\n\r\n        //ensures x/y extent\r\n        if (this.xMin != undefined && this.view.x + dxGeo < this.xMin) dxGeo = this.xMin - this.view.x\r\n        if (this.yMin != undefined && this.view.y + dyGeo < this.yMin) dyGeo = this.yMin - this.view.y\r\n        if (this.xMax != undefined && this.view.x + dxGeo > this.xMax) dxGeo = this.xMax - this.view.x\r\n        if (this.yMax != undefined && this.view.y + dyGeo > this.yMax) dyGeo = this.yMax - this.view.y\r\n\r\n        //pan\r\n        this.view.x += dxGeo\r\n        this.view.y += dyGeo\r\n        this.updateExtentGeo()\r\n\r\n        // zoom in on the current canvas state\r\n        if (this.canvasSave.c) {\r\n            this.clear(this.backgroundColor)\r\n            this.canvasSave.f /= f\r\n            this.canvasSave.dx = this.geoToPixX(xGeo) * (1 - this.canvasSave.f)\r\n            this.canvasSave.dy = this.geoToPixY(yGeo) * (1 - this.canvasSave.f)\r\n            this.offscreenCtx.drawImage(\r\n                this.canvasSave.c,\r\n                this.canvasSave.dx,\r\n                this.canvasSave.dy,\r\n                this.canvasSave.f * this.canvasSave.c.width,\r\n                this.canvasSave.f * this.canvasSave.c.height\r\n            )\r\n            this.ctx.drawImage(\r\n                this.offscreenCanvas, // Use offscreen canvas as the source\r\n                0,\r\n                0, // Position the offscreen canvas at the top-left corner of the main canvas\r\n                this.canvas.width, // The width of the visible canvas\r\n                this.canvas.height // The height of the visible canvas\r\n            )\r\n        }\r\n    }\r\n\r\n    /**\r\n     * @param {number} marginPx\r\n     * @returns {Envelope} The envelope of the view, in geo coordinates.\r\n     */\r\n    updateExtentGeo(marginPx = 20) {\r\n        this.extGeo = {\r\n            xMin: this.pixToGeoX(-marginPx),\r\n            xMax: this.pixToGeoX(this.w + marginPx),\r\n            yMin: this.pixToGeoY(this.h + marginPx),\r\n            yMax: this.pixToGeoY(-marginPx),\r\n        }\r\n        return this.extGeo\r\n    }\r\n\r\n    /**\r\n     * Check if the object has to be drawn\r\n     *\r\n     * @param {{x:number,y:number}} obj\r\n     */\r\n    toDraw(obj) {\r\n        if (obj.x < this.extGeo.xMin) return false\r\n        if (obj.x > this.extGeo.xMax) return false\r\n        if (obj.y < this.extGeo.yMin) return false\r\n        if (obj.y > this.extGeo.yMax) return false\r\n        return true\r\n    }\r\n\r\n    //conversion functions\r\n    /**\r\n     * @param {number} xGeo Geo x coordinate, in m.\r\n     * @returns {number} Screen x coordinate, in pix.\r\n     */\r\n    geoToPixX(xGeo) {\r\n        return (xGeo - this.view.x) / this.view.z + this.w * 0.5\r\n    }\r\n    /**\r\n     * @param {number} yGeo Geo y coordinate, in m.\r\n     * @returns {number} Screen y coordinate, in pix.\r\n     */\r\n    geoToPixY(yGeo) {\r\n        return -(yGeo - this.view.y) / this.view.z + this.h * 0.5\r\n    }\r\n    /**\r\n     * @param {number} x Screen x coordinate, in pix.\r\n     * @returns {number} Geo x coordinate, in m.\r\n     */\r\n    pixToGeoX(x) {\r\n        return (x - this.w * 0.5) * this.view.z + this.view.x\r\n    }\r\n    /**\r\n     * @param {number} y Screen y coordinate, in pix.\r\n     * @returns {number} Geo y coordinate, in m.\r\n     */\r\n    pixToGeoY(y) {\r\n        return -(y - this.h * 0.5) * this.view.z + this.view.y\r\n    }\r\n\r\n    /** Get x,y,z elements from URL and assign them to the view. */\r\n    setViewFromURL() {\r\n        const x = GeoCanvas.getParameterByName('x'),\r\n            y = GeoCanvas.getParameterByName('y'),\r\n            z = GeoCanvas.getParameterByName('z')\r\n        if (x != null && x != undefined && !isNaN(+x)) this.view.x = +x\r\n        if (y != null && y != undefined && !isNaN(+y)) this.view.y = +y\r\n        if (z != null && z != undefined && !isNaN(+z)) this.view.z = +z\r\n    }\r\n\r\n    /**\r\n     * Get a URL parameter by name.\r\n     *\r\n     * @param {string} name\r\n     * @returns {string | null}\r\n     */\r\n    static getParameterByName(name) {\r\n        name = name.replace(/[\\[]/, '\\\\[').replace(/[\\]]/, '\\\\]')\r\n        var regex = new RegExp('[\\\\?&]' + name + '=([^&#]*)'),\r\n            results = regex.exec(location.search)\r\n        return !results ? null : decodeURIComponent(results[1].replace(/\\+/g, ' '))\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Drawable } from './Drawable.js'\r\n\r\n/**\r\n * @module core\r\n * @abstract\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class Layer extends Drawable {\r\n    /**\r\n     * Draw layer.\r\n     *\r\n     * @param {import(\"./GeoCanvas\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @param {object} legend\r\n     * @returns {void}\r\n     * @abstract\r\n     */\r\n    draw(geoCanvas, legend = undefined) {\r\n        throw new Error('Method draw not implemented.')\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { select } from 'd3-selection'\r\n\r\n/**\r\n * A legend container.\r\n *\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class Legend {\r\n    /**\r\n     * @param {Object} opts\r\n     */\r\n    constructor(opts) {\r\n        opts = opts || {}\r\n\r\n        /** @type {string} */\r\n        this.id = opts.id\r\n\r\n        //TODO stop using it. Use style method below instead.\r\n\r\n        /** @type {number} @deprecated */\r\n        this.top = opts.top\r\n        /** @type {number} @deprecated */\r\n        this.bottom = opts.bottom\r\n        /** @type {number} @deprecated */\r\n        this.left = opts.left\r\n        /** @type {number} @deprecated */\r\n        this.right = opts.right\r\n        /** @type {string} @deprecated */\r\n        this.background = opts.background || 'none'\r\n        /** @type {string} @deprecated */\r\n        this.padding = opts.padding || '5px'\r\n        /** @type {string} @deprecated */\r\n        this.border = opts.border || '0px'\r\n        /** @type {string} @deprecated */\r\n        this['border-radius'] = opts['border-radius'] || 'none'\r\n        /** @type {string} @deprecated */\r\n        this['box-shadow'] = opts['box-shadow'] || 'none'\r\n        /** @type {string} @deprecated */\r\n        this['font-family'] = opts['font-family'] || 'Helvetica, Arial, sans-serif'\r\n        /** @type {string} @deprecated */\r\n        //this.width = opts.width\r\n        /** @type {string} @deprecated */\r\n        //this.height = opts.height\r\n\r\n        //the div element\r\n        if (this.id) this.div = select('#' + this.id)\r\n\r\n        if (!this.div || this.div.empty()) {\r\n            this.div = select(document.createElement('div'))\r\n            if (this.id) this.div.attr('id', this.id)\r\n        }\r\n\r\n        //set style\r\n        this.div.style('background', this.background)\r\n        this.div.style('padding', this.padding)\r\n        this.div.style('border', this.border)\r\n        this.div.style('border-radius', this['border-radius'])\r\n        this.div.style('box-shadow', this['box-shadow'])\r\n        this.div.style('font-family', this['font-family'])\r\n\r\n        //if (this.width) this.div.style('width', this.width)\r\n        //if (this.height) this.div.style('height', this.height)\r\n\r\n        //title\r\n        this.title = opts.title\r\n        this.titleFontSize = opts.titleFontSize || '0.8em'\r\n        this.titleFontWeight = opts.titleFontWeight || 'bold'\r\n\r\n        //label\r\n        this.labelFontSize = opts.labelFontSize || '0.8em'\r\n        this.labelUnitText = opts.labelUnitText || ''\r\n        this.labelFormat = opts.labelFormat\r\n    }\r\n\r\n    makeTitle() {\r\n        if (!this.title) return\r\n        this.div\r\n            .append('div')\r\n            .style('font-size', this.titleFontSize)\r\n            .style('font-weight', this.titleFontWeight)\r\n            .style('margin-bottom', '7px')\r\n            .text(this.title)\r\n    }\r\n\r\n    /**\r\n     * Apply a style to the legend div.\r\n     * @param {string} k\r\n     * @param {string} v\r\n     * @returns {this|string}\r\n     */\r\n    style(k, v) {\r\n        if (arguments.length == 1) return this.div.style(k)\r\n        this.div.style(k, v)\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * @param {Object} opts\r\n     * @abstract\r\n     */\r\n    update(opts = {}) {\r\n        console.error('Legend update not implemented yet.')\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n// internal imports\r\nimport { GeoCanvas } from './GeoCanvas.js'\r\nimport { Tooltip } from './Tooltip.js'\r\nimport { ZoomButtons } from '../button/ZoomButtons.js'\r\nimport { FullscreenButton } from '../button/FullscreenButton.js'\r\n\r\n// external imports\r\nimport { select } from 'd3-selection'\r\n\r\n/**\r\n * A gridviz application.\r\n *\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class Map {\r\n    /**\r\n     * @param {HTMLDivElement} container\r\n     * @param {object} opts\r\n     */\r\n    constructor(container, opts) {\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * The layers.\r\n         * @type {Array.<import(\"./Layer.js\").Layer>}\r\n         * */\r\n        this.layers = opts.layers || []\r\n\r\n        //get container element\r\n        this.container = container || document.getElementById('gridviz')\r\n        if (!this.container) {\r\n            console.error('Cannot find gridviz container element.')\r\n            return\r\n        }\r\n\r\n        //https://css-tricks.com/absolute-positioning-inside-relative-positioning/\r\n        this.container.style.position = 'relative' // container element must have relative positioning\r\n\r\n        //set dimensions\r\n        /** @type {number} */\r\n        this.w = opts.w || this.container.offsetWidth\r\n        /** @type {number} */\r\n        this.h = opts.h || this.container.offsetHeight\r\n\r\n        // Create the main canvas (for rendering to screen)\r\n        /** @type {HTMLCanvasElement} */\r\n        this._canvas = opts.canvas || this.initialiseCanvas()\r\n\r\n        /**  Initialize GeoCanvas\r\n         * @type {GeoCanvas}\r\n         * @private */\r\n        this.geoCanvas = new GeoCanvas(this._canvas, opts.x, opts.y, opts.z, opts)\r\n\r\n        this.geoCanvas.redraw = () => {\r\n            this.redraw()\r\n        }\r\n\r\n        // legend div\r\n        this.legend = opts.legendContainer\r\n            ? select(opts.legendContainer) // Wrap the provided HTML node in a D3 selection\r\n            : null\r\n        if (!this.legend) this.initialiseLegend()\r\n\r\n        //tooltip\r\n\r\n        // set App container as default parent element for tooltip\r\n        if (!opts.tooltip) opts.tooltip = {}\r\n        if (!opts.tooltip.parentElement) opts.tooltip.parentElement = this.container\r\n\r\n        /**\r\n         * @private\r\n         * @type {Tooltip} */\r\n        this.tooltip = new Tooltip(opts.tooltip)\r\n\r\n        // add event listeners to container\r\n        this.mouseOverHandler = (e) => this.focusCell(e)\r\n        this.mouseMoveHandler = (e) => this.focusCell(e)\r\n        this.mouseOutHandler = (e) => this.tooltip.hide()\r\n        this.geoCanvas.canvas.addEventListener('mouseover', this.mouseOverHandler)\r\n        this.geoCanvas.canvas.addEventListener('mousemove', this.mouseMoveHandler)\r\n        this.geoCanvas.canvas.addEventListener('mouseout', this.mouseOutHandler)\r\n\r\n        // listen for resize events on the App's container and handle them\r\n        this.defineResizeObserver()\r\n\r\n        // add extra logic to onZoomStartFun\r\n        this.geoCanvas.onZoomStartFun = (e) => {\r\n            if (opts.onZoomStartFun) opts.onZoomStartFun(e)\r\n            this.tooltip.hide()\r\n        }\r\n\r\n        //for mouse over\r\n        /**\r\n         * @private\r\n         * @type {HTMLCanvasElement|null} */\r\n        this.canvasSave = null\r\n\r\n        this.selectionRectangleColor = opts.selectionRectangleColor || '#FF6347'\r\n        this.selectionRectangleWidthPix = opts.selectionRectangleWidthPix || (() => 4) //(r,z) => {}\r\n\r\n        // transparent background (e.g. leaflet) 'red painting' fix\r\n        this.transparentBackground = opts.transparentBackground\r\n\r\n        //set default globalCompositeOperation\r\n        this.defaultGlobalCompositeOperation =\r\n            opts.defaultGlobalCompositeOperation || this.geoCanvas.ctx.globalCompositeOperation\r\n    }\r\n\r\n    /**\r\n     * @protected\r\n     * @returns {HTMLCanvasElement}\r\n     */\r\n    initialiseCanvas() {\r\n        const canvas = document.createElement('canvas')\r\n        canvas.setAttribute('width', '' + this.w)\r\n        canvas.setAttribute('height', '' + this.h)\r\n        this.container.appendChild(canvas)\r\n        return canvas\r\n    }\r\n\r\n    initialiseLegend() {\r\n        this.legend = select(this.container)\r\n            .append('div') // Create a new container\r\n            .attr('id', 'gridviz-legend')\r\n            .style('position', 'absolute')\r\n            .style('width', 'auto')\r\n            .style('height', 'auto')\r\n            .style('background', '#FFFFFF')\r\n            //.style(\"padding\", this.padding)\r\n            .style('border', '0px')\r\n            //.style('border-radius', '5px')\r\n            .style('box-shadow', '3px 3px 3px grey, -3px -3px 3px #ddd')\r\n            .style('font-family', 'Helvetica, Arial, sans-serif')\r\n            .style('bottom', '15px')\r\n            .style('right', '15px')\r\n        //hide\r\n        //.style(\"visibility\", \"hidden\")\r\n    }\r\n\r\n    /**\r\n     * Set/get layer stack.\r\n     *\r\n     * @param {undefined|import(\"./Layer.js\").Layer|import(\"./Layer.js\").Layer[]} layers\r\n     * @returns { this | import(\"./Layer.js\").Layer[] }\r\n     */\r\n    layers_(layers) {\r\n        if (arguments.length === 0) return this.layers\r\n        if (arguments.length === 1)\r\n            if (Array.isArray(layers)) this.layers = layers\r\n            else this.layers = [layers]\r\n        else this.layers = arguments\r\n        return this\r\n    }\r\n\r\n    /** @returns {this} */\r\n    redraw() {\r\n        //remove legend elements\r\n        if (this.legend) this.legend.selectAll('*').remove()\r\n\r\n        //clear\r\n        this.geoCanvas.initCanvasTransform()\r\n        this.geoCanvas.clear(this.geoCanvas.backgroundColor)\r\n\r\n        const z = this.geoCanvas.view.z\r\n        this.updateExtentGeo()\r\n\r\n        //go through the layers\r\n        for (const layer of this.layers) {\r\n            //check if layer is visible\r\n            if (layer.visible && !layer.visible(z)) continue\r\n\r\n            //set layer alpha and blend mode\r\n            this.geoCanvas.offscreenCtx.globalAlpha = layer.alpha ? layer.alpha(z) : 1.0\r\n            if (layer.blendOperation) this.geoCanvas.offscreenCtx.globalCompositeOperation = layer.blendOperation(z)\r\n\r\n            //set affin transform to draw with geographical coordinates\r\n            this.geoCanvas.setCanvasTransform()\r\n\r\n            //draw layer\r\n            layer.draw(this.geoCanvas, this.legend)\r\n\r\n            //draw layer filter\r\n            if (layer.filterColor) layer.drawFilter(this.geoCanvas)\r\n\r\n            //restore default alpha and blend operation\r\n            this.geoCanvas.offscreenCtx.globalAlpha = 1.0\r\n            this.geoCanvas.offscreenCtx.globalCompositeOperation = this.defaultGlobalCompositeOperation\r\n        }\r\n\r\n        // one drawImage call: draw the offscreen canvas to the main canvas\r\n        this.geoCanvas.initCanvasTransform()\r\n        this.geoCanvas.ctx.drawImage(this.geoCanvas.offscreenCanvas, 0, 0)\r\n\r\n        this.canvasSave = null\r\n\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * @param {number} marginPx\r\n     * @returns {import('./GeoCanvas.js').Envelope}\r\n     * @public\r\n     */\r\n    updateExtentGeo(marginPx = 20) {\r\n        return this.geoCanvas.updateExtentGeo(marginPx)\r\n    }\r\n\r\n    /** @param {MouseEvent} e */\r\n    focusCell(e) {\r\n        //compute mouse geo position\r\n        const mousePositionGeo = {\r\n            x: this.geoCanvas.pixToGeoX(e.offsetX + this.tooltip.xMouseOffset),\r\n            y: this.geoCanvas.pixToGeoY(e.offsetY + this.tooltip.yMouseOffset),\r\n        }\r\n        /** @type {{cell:import('./Dataset.js').Cell,html:string,resolution:number} | undefined} */\r\n        const focus = this.getCellFocusInfo(mousePositionGeo)\r\n\r\n        // transparent background (e.g. leaflet) 'red painting' fix\r\n        if (this.transparentBackground) {\r\n            if (focus) {\r\n                this.tooltip.html(focus.html)\r\n                this.tooltip.setPosition(e)\r\n                this.tooltip.show()\r\n            } else {\r\n                this.tooltip.hide()\r\n            }\r\n            this.canvasSave = document.createElement('canvas')\r\n            this.canvasSave.setAttribute('width', '' + this.w)\r\n            this.canvasSave.setAttribute('height', '' + this.h)\r\n            this.canvasSave.getContext('2d')?.drawImage(this.geoCanvas.canvas, 0, 0)\r\n            this.geoCanvas.initCanvasTransform()\r\n            return\r\n        }\r\n\r\n        if (focus) {\r\n            this.tooltip.html(focus.html)\r\n            this.tooltip.setPosition(e)\r\n            this.tooltip.show()\r\n\r\n            //show cell position as a rectangle\r\n            if (!this.canvasSave) {\r\n                this.canvasSave = document.createElement('canvas')\r\n                this.canvasSave.setAttribute('width', '' + this.w)\r\n                this.canvasSave.setAttribute('height', '' + this.h)\r\n                this.canvasSave.getContext('2d')?.drawImage(this.geoCanvas.offscreenCanvas, 0, 0)\r\n            } else {\r\n                this.geoCanvas.offscreenCtx.drawImage(this.canvasSave, 0, 0)\r\n            }\r\n\r\n            //draw image saved + draw rectangle\r\n            const rectWPix = this.selectionRectangleWidthPix\r\n                ? this.selectionRectangleWidthPix(focus.resolution, this.geoCanvas.view.z)\r\n                : 4\r\n            this.geoCanvas.initCanvasTransform()\r\n            const ctx = this.geoCanvas.offscreenCtx\r\n            ctx.strokeStyle = this.selectionRectangleColor\r\n            ctx.lineWidth = rectWPix\r\n            ctx.beginPath()\r\n\r\n            ctx.rect(\r\n                this.geoCanvas.geoToPixX(focus.cell.x) - rectWPix / 2,\r\n                this.geoCanvas.geoToPixY(focus.cell.y) + rectWPix / 2,\r\n                focus.resolution / this.geoCanvas.view.z + rectWPix,\r\n                -focus.resolution / this.geoCanvas.view.z - rectWPix\r\n            )\r\n            ctx.stroke()\r\n            this.geoCanvas.ctx.drawImage(this.geoCanvas.offscreenCanvas, 0, 0)\r\n        } else {\r\n            this.tooltip.hide()\r\n            if (this.canvasSave) this.geoCanvas.ctx.drawImage(this.canvasSave, 0, 0)\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Return the cell HTML info at a given geo position.\r\n     * This is usefull for user interactions, to show this info where the user clicks for example.\r\n     *\r\n     * @param {{x:number,y:number}} posGeo\r\n     * @returns {{cell:import('./Dataset.js').Cell,html:string,resolution:number} | undefined}\r\n     * @protected\r\n     */\r\n    getCellFocusInfo(posGeo) {\r\n        //go through the layers, starting from top\r\n        const z = this.geoCanvas.view.z\r\n        for (let i = this.layers.length - 1; i >= 0; i--) {\r\n            /** @type {import(\"./Layer.js\").Layer} */\r\n            const layer = this.layers[i]\r\n            if (layer.visible && !layer.visible(z)) continue\r\n            if (layer.cellInfoHTML === 'none') continue // this is necessary in order to not show tooltips for layers 'on top' (e.g. population circles on top of squares)\r\n            if (!layer.cellInfoHTML) continue\r\n            if (!layer.getDataset) continue\r\n            const dsc = layer.getDataset(z)\r\n            if (!dsc) continue\r\n\r\n            //get cell at mouse position\r\n            /** @type {import('./Dataset.js').Cell|undefined} */\r\n            const cell = dsc.getCellFromPosition(posGeo, dsc.getViewCache())\r\n            //console.log(cell, dsc.resolution)\r\n            if (!cell) return undefined\r\n\r\n            //rare case for a dataset with mixed resolutions\r\n            if (dsc.mixedResolution) {\r\n                const r = +dsc.mixedResolution(cell)\r\n                const html = layer.cellInfoHTML(cell, r)\r\n                if (!html) return undefined\r\n                return { cell: cell, html: html, resolution: r }\r\n            }\r\n\r\n            const html = layer.cellInfoHTML(cell, dsc.getResolution())\r\n            if (!html) return undefined\r\n            return { cell: cell, html: html, resolution: dsc.getResolution() }\r\n        }\r\n    }\r\n\r\n    /**\r\n     * @param {number} x\r\n     * @param {number} y\r\n     * @param {number|undefined} z\r\n     */\r\n    setView(x, y, z = undefined) {\r\n        this.geoCanvas.view.x = x\r\n        this.geoCanvas.view.y = y\r\n        if (z != undefined) this.geoCanvas.view.z = z\r\n        return this\r\n    }\r\n\r\n    /** @returns {import('./GeoCanvas.js').View} */\r\n    getView() {\r\n        return this.geoCanvas.view\r\n    }\r\n\r\n    /** @returns {number} */\r\n    getZoom() {\r\n        return this.geoCanvas.view.z\r\n    }\r\n    /** @param {number} z @returns {this} */\r\n    setZoom(z) {\r\n        this.geoCanvas.view.z = z\r\n        return this\r\n    }\r\n\r\n    /** @returns {Array.<number|undefined>} */\r\n    getCenterExtent() {\r\n        return this.geoCanvas.getCenterExtent()\r\n    }\r\n    /** @param {Array.<number>} val @returns {this} */\r\n    setCenterExtent(val) {\r\n        this.geoCanvas.setCenterExtent(val)\r\n        return this\r\n    }\r\n\r\n    /** @returns {Array.<number>} */\r\n    getZoomExtent() {\r\n        return this.geoCanvas.getZoomExtent()\r\n    }\r\n    /** @param {Array.<number>} val @returns {this} */\r\n    setZoomExtent(val) {\r\n        this.geoCanvas.setZoomExtent(val)\r\n        return this\r\n    }\r\n\r\n    /** @returns {string} */\r\n    getBackgroundColor() {\r\n        return this.geoCanvas.backgroundColor\r\n    }\r\n    /** @param {string} val @returns {this} */\r\n    setBackgroundColor(val) {\r\n        this.geoCanvas.backgroundColor = val\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * Adds a set of zoom buttons to the map\r\n     *\r\n     * @param {object} opts\r\n     * @returns {this}\r\n     */\r\n    addZoomButtons(opts) {\r\n        // * opts.id\r\n        // * opts.onZoom - custom event handler function\r\n        // * opts.x\r\n        // * opts.y\r\n        // * opts.delta - zoom delta applied on each click\r\n\r\n        this.zoomButtons = new ZoomButtons({\r\n            map: this,\r\n            id: opts?.id || 'gridviz-zoom-buttons-' + this.container.id,\r\n            class: opts?.class,\r\n            x: opts?.x,\r\n            y: opts?.y,\r\n            onZoom: opts?.onZoom,\r\n            delta: opts?.delta || 0.2,\r\n        })\r\n\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * Adds a fullscreen toggle button to the app\r\n     *\r\n     * @param {object} opts\r\n     * @returns {this}\r\n     */\r\n    addFullscreenButton(opts) {\r\n        // * opts.map - the gridviz map\r\n        // * opts.id\r\n        // * opts.x\r\n        // * opts.y\r\n\r\n        this.fullscreenButton = new FullscreenButton({\r\n            map: this,\r\n            id: opts?.id || 'gridviz-fullscreen-button',\r\n            class: opts?.class,\r\n            x: opts?.x,\r\n            y: opts?.y,\r\n        })\r\n\r\n        return this\r\n    }\r\n\r\n    /** @returns {this} */\r\n    setViewFromURL() {\r\n        this.geoCanvas.setViewFromURL()\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * @description Add a resize event observer to the Apps container and update the canvas accordingly\r\n     * @memberof App\r\n     */\r\n    defineResizeObserver() {\r\n        // listen to resize events\r\n        const resizeObserver = new ResizeObserver((entries) => {\r\n            let container = this.container\r\n            // make sure canvas has been built\r\n            if (container.clientWidth > 0 && container.clientHeight > 0) {\r\n                // make sure we dont exceed loop limit first\r\n                // see: https://stackoverflow.com/questions/49384120/resizeobserver-loop-limit-exceeded\r\n                window.requestAnimationFrame(() => {\r\n                    if (!Array.isArray(entries) || !entries.length) {\r\n                        return\r\n                    }\r\n                    // update the map and canvas size\r\n                    if (this.h !== container.clientHeight || this.w !== container.clientWidth) {\r\n                        this.h = container.clientHeight\r\n                        this.w = container.clientWidth\r\n                        this.geoCanvas.h = container.clientHeight\r\n                        this.geoCanvas.w = container.clientWidth\r\n                        this.geoCanvas.canvas.setAttribute('width', '' + this.w)\r\n                        this.geoCanvas.canvas.setAttribute('height', '' + this.h)\r\n                        // offscreen canvas\r\n                        this.geoCanvas.offscreenCanvas.setAttribute('width', '' + this.w)\r\n                        this.geoCanvas.offscreenCanvas.setAttribute('height', '' + this.h)\r\n                        this.redraw()\r\n\r\n                        //update button positions\r\n                        // if (this.zoomButtons) this.zoomButtons.node.style.left = this.w - 50 + 'px'\r\n                        // if (this.fullscreenButton) this.fullscreenButton.node.style.left = this.w - 50 + 'px'\r\n                    }\r\n                })\r\n            }\r\n        })\r\n\r\n        resizeObserver.observe(this.container)\r\n    }\r\n\r\n    /**\r\n     * @description Destroy the map and it's event listeners\r\n     * This should significantly reduce the memory used when creating and destroying gridviz map instances (for example in leaflet-gridviz)\r\n     * @memberof App\r\n     */\r\n    destroy() {\r\n        // clear layers\r\n        this.layers = []\r\n        this.bgLayers = []\r\n\r\n        // remove event listeners from container\r\n        this.container.removeEventListener('mouseover', this.mouseOverHandler)\r\n        this.container.removeEventListener('mousemove', this.mouseMoveHandler)\r\n        this.container.removeEventListener('mouseout', this.mouseOutHandler)\r\n\r\n        // remove canvas\r\n        this.geoCanvas.canvas.remove()\r\n\r\n        // remove legend\r\n        this.legend?.remove()\r\n\r\n        // remove tooltip\r\n        this.tooltip.tooltip?.remove()\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n/**\r\n * A multi resolution dataset of grid cells.\r\n * It consists of different {@link Dataset}s for each resolution.\r\n *\r\n * @abstract\r\n *\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class MultiResolutionDataset {\r\n    /**\r\n     * @param {Array.<number>} resolutions The resolutions of the datasets, in CRS geographical unit.\r\n     * @param {Array.<import(\"./Dataset\").Dataset>|function(number):import(\"./Dataset\").Dataset} datasets The datasets list, one per resolution. Or a function that returns a dataset from a resolution value.\r\n     * @param { {preprocess?:function(import(\"./Dataset\").Cell):boolean} } opts Options. preprocess: A function to apply on each dataset cell to prepare its values. Can be used also to select cells to keep.\r\n     */\r\n    constructor(resolutions, datasets, opts = {}) {\r\n        opts = opts || {}\r\n\r\n        /** The resolutions of the datasets, in CRS geographical unit.\r\n         * @type {Array.<number>} */\r\n        this.resolutions = resolutions\r\n\r\n        /** The datasets. If the list is not explictely defined, build it from the resolution list with the function\r\n         * @type {Array.<import(\"./Dataset\").Dataset>} */\r\n        this.datasets = typeof datasets === 'function' ? this.resolutions.map(datasets) : datasets\r\n\r\n        //there must be as many datasets as resolutions\r\n        if (this.datasets.length > 1 && this.datasets.length != this.resolutions.length)\r\n            throw new Error(\r\n                'Uncompatible number of datasets and resolutions: ' +\r\n                    this.datasets.length +\r\n                    ' ' +\r\n                    this.resolutions.length\r\n            )\r\n\r\n        //set dataset preprocesses if specified\r\n        if (opts.preprocess) this.setPrepocesses(opts.preprocess)\r\n    }\r\n\r\n    /**\r\n     * Return the relevant dataset for a specified zoom.\r\n     * @param {number} z\r\n     * @param {number} minPixelsPerCell\r\n     * @returns {import(\"./Dataset\").Dataset|undefined}\r\n     * */\r\n    getDataset(z, minPixelsPerCell) {\r\n        //special case whith single dataset\r\n        if (this.datasets.length == 1) return this.datasets[0]\r\n\r\n        const rs = this.resolutions\r\n        let i = 0\r\n        let z_ = rs[i] / minPixelsPerCell\r\n        while (z_ < z && i < rs.length) {\r\n            i++\r\n            z_ = rs[i] / minPixelsPerCell\r\n        }\r\n        //if (i == 0) return this.dataset.datasets[0];\r\n        //return this.dataset.datasets[i - 1];\r\n        if (i == rs.length) return this.datasets[rs.length - 1]\r\n        return this.datasets[i]\r\n    }\r\n\r\n    /**\r\n     * Set a preprocess function for all datasets.\r\n     * This is a function applied on each cell after it has been loaded.\r\n     *\r\n     * @param {function(import(\"./Dataset\").Cell):boolean} preprocess\r\n     * @returns {this}\r\n     */\r\n    setPrepocesses(preprocess) {\r\n        for (let ds of this.datasets) ds.preprocess = preprocess\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * A function to ease the creation of multi resolution datasets.\r\n     *\r\n     * @param {Array.<number>} resolutions The resolutions of the datasets, in CRS geographical unit\r\n     * @param {function(number):import(\"./Dataset\").Dataset} resToDataset Function returning a dataset from a resolution\r\n     * @param { {preprocess?:function(import(\"./Dataset\").Cell):boolean} } opts Options. preprocess: A function to apply on each dataset cell to prepare its values\r\n     * @returns {MultiResolutionDataset}\r\n     */\r\n    /*static make(resolutions, resToDataset, opts) {\r\n        //make datasets\r\n        const dsc = []\r\n        for (const res of resolutions) dsc.push(resToDataset(res))\r\n        //make multi resolution dataset\r\n        return new MultiResolutionDataset(dsc, resolutions, opts)\r\n    }*/\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Drawable } from './Drawable.js'\r\n\r\n/** @typedef {\"square\"|\"circle\"|\"diamond\"|\"donut\"|\"triangle_up\"|\"triangle_down\"|\"triangle_left\"|\"triangle_right\"|\"none\"} Shape */\r\n\r\n/**\r\n * viewScale type\r\n * Returns an object from a list of cells,\r\n * @typedef {function(Array.<import('./Dataset.js').Cell>,number, number):*} ViewScale */\r\n\r\n/**\r\n * A style, to show a grid dataset.\r\n *\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class Style extends Drawable {\r\n    /**\r\n     * @abstract\r\n     * @param {{filter?:function(import('./Dataset').Cell):boolean, offset?:function(import('./Dataset').Cell, number, number):{dx:number,dy:number}, visible?:function(number):boolean,alpha?:function(number):number,blendOperation?:function(number):GlobalCompositeOperation,drawFun?:function,viewScale?:ViewScale}} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * @type {ViewScale|undefined} */\r\n        this.viewScale = opts.viewScale\r\n\r\n        /** A filter function to apply to the cell list, to filter out some cells not to be drawn (such as for example the cells with value=0).\r\n         * @protected\r\n         * @type {(function(import('./Dataset').Cell):boolean) | undefined} */\r\n        this.filter = opts.filter || undefined\r\n\r\n        /** An offset. This is to alter the position of all symbols in a given direction. In geographical unit.\r\n         * @protected\r\n         * @type {function(import('./Dataset').Cell,number,number):{dx:number,dy:number}} */\r\n        this.offset = opts.offset || ((c, r, z) => ({ dx: 0, dy: 0 }))\r\n\r\n        /** A draw function for the style.\r\n         * @type {function|undefined} */\r\n        this.drawFun = opts.drawFun\r\n\r\n        /**\r\n         * @public\r\n         * @type {Array.<import(\"./Legend\").Legend>} */\r\n        this.legends = []\r\n    }\r\n\r\n    /**\r\n     * Draw cells.\r\n     *\r\n     * @param {Array.<import('./Dataset').Cell>} cells The cells to draw.\r\n     * @param {import(\"./GeoCanvas\").GeoCanvas} geoCanvas The canvas where to draw them.\r\n     * @param {number} resolution Their resolution (in geographic unit)\r\n     * @abstract\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        if (this.drawFun) this.drawFun(cells, geoCanvas, resolution)\r\n        else throw new Error('Method draw not implemented.')\r\n    }\r\n\r\n    //getters and setters\r\n\r\n    /** @returns {function(import('./Dataset').Cell,number,number):{dx:number,dy:number}} */\r\n    getOffset() {\r\n        return this.offset\r\n    }\r\n    /** @param {function(import('./Dataset').Cell,number,number):{dx:number,dy:number}} val @returns {this} */\r\n    setOffset(val) {\r\n        this.offset = val\r\n        return this\r\n    }\r\n\r\n    /** Update legends of the style, if any\r\n     * @param {object} opts\r\n     * @returns {this} */\r\n    updateLegends(opts) {\r\n        Style.updateLegendsRecursive(this.legends, opts)\r\n        return this\r\n    }\r\n\r\n    /** @private */\r\n    static updateLegendsRecursive(lg, opts) {\r\n        if (Array.isArray(lg)) for (const lg_ of lg) this.updateLegendsRecursive(lg_, opts)\r\n        else lg.update(opts)\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"./Legend\").Legend>} legends\r\n     * @returns {this} */\r\n    addLegends(legends) {\r\n        for (let legend of legends) this.legends.push(legend)\r\n        return this\r\n    }\r\n}\r\n","//@ts-check\n'use strict'\n\nimport { select } from 'd3-selection'\n//import { transition } from \"d3-transition\";\n\n/**\n * A generic class to make a tooltip.\n * It is a div element, which can be moved under the mouse pointer and filled with some information in html.\n * @module core\n */\nexport class Tooltip {\n    /**\n     * @param {object} opts\n     */\n    constructor(opts) {\n        opts = opts || {}\n\n        /** @type {string} */\n        this.div = opts.div || 'tooltip_eurostat'\n        /** @type {string} */\n        this.maxWidth = opts.maxWidth || '20em'\n        /** @type {string} */\n        this.fontSize = opts.fontSize || '1.2em'\n        /** @type {string} */\n        this.background = opts.background || 'white'\n        /** @type {string} */\n        this.padding = opts.padding || '5px'\n        /** @type {string} */\n        this.border = opts.border || '0px'\n        /** @type {string} */\n        this['border-radius'] = opts['border-radius'] || '0px'\n        /** @type {string} */\n        this['box-shadow'] = opts['box-shadow'] || '5px 5px 5px grey'\n        /** @type {string} */\n        this['font-family'] = opts['font-family'] || 'Helvetica, Arial, sans-serif'\n\n        /** @type {number} */\n        this.transitionDuration = opts.transitionDuration || 100\n        /** @type {number} */\n        this.xOffset = opts.xOffset || 30\n        /** @type {number} */\n        this.yOffset = opts.yOffset || 20\n        /** @type {number} */ // e.g. to prevent mouse cursor covering cell being highlighted\n        this.yMouseOffset = opts.yMouseOffset || 0\n        /** @type {number} */\n        this.xMouseOffset = opts.xMouseOffset || 0\n        /** @type {HTMLElement} */\n        this.parentElement = opts.parentElement || document.body\n\n        /**\n         * @public\n         * @type {import(\"d3-selection\").Selection} */\n        this.tooltip = select('#' + this.div)\n\n        if (this.tooltip.empty()) {\n            //create tooltip DOM node\n            // this.tooltip = select(\n            //     '#' + this.parentElement.id && this.parentElement.id != ''\n            //         ? '#' + this.parentElement.id\n            //         : 'body'\n            // )\n            this.tooltip = select('body').append('div').attr('id', this.div)\n        }\n\n        //initialise\n        this.tooltip.style('max-width', this.maxWidth)\n        this.tooltip.style('overflow', 'hidden')\n        this.tooltip.style('font-size', this.fontSize)\n        this.tooltip.style('background', this.background)\n        this.tooltip.style('padding', this.padding)\n        this.tooltip.style('border', this.border)\n        this.tooltip.style('border-radius', this['border-radius'])\n        this.tooltip.style('box-shadow', this['box-shadow'])\n        this.tooltip.style('font-family', this['font-family'])\n        this.tooltip.style('position', 'absolute')\n        this.tooltip.style('pointer-events', 'none')\n        this.tooltip.style('opacity', '0')\n        this.tooltip.style('text-wrap', 'nowrap')\n\n        // aria-labels (thanks to wahlatlas)\n        this.tooltip.attr('role', 'tooltip').attr('aria-live', 'polite')\n    }\n\n    /** Show the tooltip */\n    show() {\n        // @ts-ignore\n        this.tooltip.transition().duration(this.transitionDuration).style('opacity', 1)\n    }\n\n    /** Hide the tooltip */\n    hide() {\n        // @ts-ignore\n        this.tooltip.transition().duration(this.transitionDuration).style('opacity', 0)\n    }\n\n    /**\n     * Set the content of the tooltip.\n     * @param {string} html\n     */\n    html(html) {\n        this.tooltip.html(html)\n    }\n\n    /**\n     * Set the position of the tooltip at the mouse event position.\n     * @param {MouseEvent} event\n     */\n    setPosition(event) {\n        let parentRect = this.parentElement.getBoundingClientRect()\n\n        let x = event.pageX + this.xOffset\n        let y = event.pageY - this.yOffset\n\n        this.tooltip.style('left', x + 'px').style('top', y + 'px')\n\n        this.ensureTooltipInsideContainer(event, parentRect)\n    }\n\n    /*\n\tmy.mouseover = function (event, html) {\n\t\tif (html) my.html(html);\n\t\tmy.setPosition(event);\n\t\tmy.show()\n\t\t//this.ensureTooltipInsideContainer();\n\t};\n\t\n\tmy.mousemove = function (event) {\n\t\tmy.setPosition(event);\n\t\t//this.ensureTooltipInsideContainer();\n\t};\n\t\n\tmy.mouseout = function () {\n\t\tmy.hide();\n\t};*/\n\n    style(k, v) {\n        if (arguments.length == 1) return this.tooltip.style(k)\n        this.tooltip.style(k, v)\n        return this\n    }\n\n    attr(k, v) {\n        if (arguments.length == 1) return this.tooltip.attr(k)\n        this.tooltip.attr(k, v)\n        return this\n    }\n\n    /**\n     * @function ensureTooltipInsideContainer\n     * @description Prevents the tooltip from overflowing out of the App container (ensures that the tooltip is inside the gridviz container)\n     * @param {MouseEvent} event\n     * @param {DOMRect} parentRect\n     */\n    ensureTooltipInsideContainer = function (event, parentRect) {\n        let node = this.tooltip.node()\n        let parentWidth = parentRect.width\n        let parentHeight = parentRect.height\n\n        //too far right\n        if (node.offsetLeft > parentRect.left + parentWidth - node.clientWidth) {\n            let left = event.x - node.clientWidth - this.xOffset\n            node.style.left = left + 'px'\n            // check if mouse covers tooltip\n            if (node.offsetLeft + node.clientWidth > event.x) {\n                //move tooltip left so it doesnt cover mouse\n                let left2 = event.x - node.clientWidth - this.xOffset\n                node.style.left = left2 + 'px'\n            }\n            // node.style.top = node.offsetTop + config.yOffset + \"px\";\n        }\n\n        //too far down\n        if (node.offsetTop + node.clientHeight > parentRect.top + parentHeight) {\n            node.style.top = node.offsetTop - node.clientHeight + 'px'\n        }\n\n        //too far up\n        if (node.offsetTop < parentRect.top) {\n            node.style.top = parentRect.top + this.yOffset + 'px'\n        }\n    }\n}\n","//@ts-check\r\n'use strict'\r\n\r\n/** @typedef {{ dims: object, crs: string, tileSizeCell: number, originPoint: {x:number,y:number}, resolutionGeo: number, tilingBounds:import(\"../core/GeoCanvas.js\").Envelope }} GridInfo */\r\n\r\nimport { dsv } from 'd3-fetch'\r\nimport { Dataset } from '../core/Dataset.js'\r\n\r\n/**\r\n * A dataset composed of a single CSV file (not tiled).\r\n *\r\n * @module dataset\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class CSVGrid extends Dataset {\r\n    /**\r\n     * @param {import(\"../core/Map.js\").Map} map The map.\r\n     * @param {string} url The URL of the dataset.\r\n     * @param {number} resolution The dataset resolution in geographical unit.\r\n     * @param {{preprocess?:(function(import(\"../core/Dataset.js\").Cell):boolean),delimiter?:string}} opts\r\n     */\r\n    constructor(map, url, resolution, opts = {}) {\r\n        super(map, url, resolution, opts)\r\n\r\n        /**\r\n         * @private\r\n         * @type {Array.<import(\"../core/Dataset.js\").Cell>} */\r\n        this.cells = []\r\n\r\n        /**\r\n         * @private\r\n         * @type {string} */\r\n        this.delimiter = opts.delimiter || ','\r\n\r\n        /**\r\n         * @type {string}\r\n         * @private  */\r\n        this.infoLoadingStatus = 'notLoaded'\r\n\r\n        //get data\r\n        this.getData(undefined)\r\n    }\r\n\r\n    /**\r\n     * Request data within a geographic envelope.\r\n     * @param {import(\"../core/GeoCanvas.js\").Envelope|undefined} e\r\n     */\r\n    getData(e) {\r\n        //check if data already loaded\r\n        if (this.infoLoadingStatus != 'notLoaded') return this\r\n\r\n        //load data\r\n        this.infoLoadingStatus = 'loading'\r\n        ;(async () => {\r\n            try {\r\n                const data = await dsv(this.delimiter, this.url)\r\n\r\n                //convert coordinates in numbers\r\n                for (const c of data) {\r\n                    c.x = +c.x\r\n                    c.y = +c.y\r\n                }\r\n\r\n                //preprocess/filter\r\n                if (this.preprocess) {\r\n                    this.cells = []\r\n                    for (const c of data) {\r\n                        const b = this.preprocess(c)\r\n                        if (b == false) continue\r\n                        this.cells.push(c)\r\n                    }\r\n                } else {\r\n                    this.cells = data\r\n                }\r\n\r\n                //TODO check if redraw is necessary\r\n                //that is if the dataset belongs to a layer which is visible at the current zoom level\r\n\r\n                //redraw map\r\n                if (this.map) this.map.redraw()\r\n\r\n                this.infoLoadingStatus = 'loaded'\r\n            } catch (error) {\r\n                //mark as failed\r\n                this.infoLoadingStatus = 'failed'\r\n                this.cells = []\r\n            }\r\n        })()\r\n\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * Fill the view cache with all cells which are within a geographical envelope.\r\n     *\r\n     * @param {import(\"../core/GeoCanvas.js\").Envelope} extGeo\r\n     * @returns {void}\r\n     */\r\n    updateViewCache(extGeo) {\r\n        //data not loaded yet\r\n        if (!this.cells) return\r\n\r\n        this.cellsViewCache = []\r\n        for (const cell of this.cells) {\r\n            if (+cell.x + this.resolution < extGeo.xMin) continue\r\n            if (+cell.x - this.resolution > extGeo.xMax) continue\r\n            if (+cell.y + this.resolution < extGeo.yMin) continue\r\n            if (+cell.y - this.resolution > extGeo.yMax) continue\r\n            this.cellsViewCache.push(cell)\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Dataset } from '../core/Dataset.js'\r\n\r\n/**\r\n * A dataset composed of cells defined in javascript, or loaded outside of gridviz map.\r\n *\r\n * @module dataset\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class JSGrid extends Dataset {\r\n    /**\r\n     * @param {number} resolution The dataset resolution in geographical unit.\r\n     * @param {Array.<Object>} cells The cells.\r\n     * @param {} opts\r\n     */\r\n    constructor(resolution, cells, opts = {}) {\r\n        super(undefined, '', resolution, opts)\r\n\r\n        /**\r\n         * @private\r\n         * @type {Array.<import('../core/Dataset.js').Cell>} */\r\n        this.cells = cells || []\r\n    }\r\n\r\n    /**\r\n     * Request data within a geographic envelope.\r\n     *\r\n     * @param {import(\"../core/GeoCanvas.js\").Envelope|undefined} e\r\n     */\r\n    getData(e) {\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * Fill the view cache with all cells which are within a geographical envelope.\r\n     *\r\n     * @param {import(\"../core/GeoCanvas.js\").Envelope} extGeo\r\n     * @returns {void}\r\n     */\r\n    updateViewCache(extGeo) {\r\n        //data not loaded yet\r\n        if (!this.cells) return\r\n\r\n        this.cellsViewCache = []\r\n        for (const cell of this.cells) {\r\n            if (+cell.x + this.resolution < extGeo.xMin) continue\r\n            if (+cell.x - this.resolution > extGeo.xMax) continue\r\n            if (+cell.y + this.resolution < extGeo.yMin) continue\r\n            if (+cell.y - this.resolution > extGeo.yMax) continue\r\n            this.cellsViewCache.push(cell)\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n/** @typedef {{ dims: object, crs: string, tileSizeCell: number, originPoint: {x:number,y:number}, resolutionGeo: number, tilingBounds:import(\"../core/GeoCanvas.js\").Envelope }} GridInfo */\r\n\r\n// internal\r\nimport { Dataset } from '../core/Dataset.js'\r\n//import { monitor, monitorDuration } from '../utils/Utils.js'\r\n\r\n// external\r\nimport { json, csv } from 'd3-fetch'\r\n\r\n/**\r\n * A tiled dataset, composed of CSV tiles.\r\n *\r\n * @module dataset\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class TiledGrid extends Dataset {\r\n    /**\r\n     * @param {import(\"../core/Map.js\").Map} map The map.\r\n     * @param {string} url The URL of the dataset.\r\n     * @param {{preprocess?:(function(import(\"../core/Dataset.js\").Cell):boolean), onlyDrawWhenAllTilesReady:boolean}} opts\r\n     */\r\n    constructor(map, url, opts = {}) {\r\n        super(map, url, 0, opts)\r\n        this.onlyDrawWhenAllTilesReady = opts.onlyDrawWhenAllTilesReady || false\r\n        /**\r\n         * The grid info object, from the info.json file.\r\n         *  @type {GridInfo | undefined}\r\n         * @private\r\n         *  */\r\n        this.info = undefined\r\n\r\n        /**\r\n         * @type {string}\r\n         * @private  */\r\n        this.infoLoadingStatus = 'notLoaded'\r\n\r\n        /**\r\n         * The cache of the loaded tiles. It is double indexed: by xT and then yT.\r\n         * Example: this.cache[xT][yT] returns the tile at [xT][yT] location.\r\n         *\r\n         * @type {object}\r\n         * */\r\n        this.cache = {}\r\n\r\n        //launch loading\r\n        this.loadInfo()\r\n    }\r\n\r\n    /**\r\n     * Load the info.json from the url.\r\n     * @returns this\r\n     */\r\n    loadInfo() {\r\n        if (!this.info && this.infoLoadingStatus === 'notLoaded') {\r\n            ;(async () => {\r\n                try {\r\n                    const data = await json(this.url + 'info.json')\r\n                    this.info = data\r\n                    this.resolution = data.resolutionGeo\r\n                    this.infoLoadingStatus = 'loaded'\r\n                    this.map.redraw()\r\n                } catch (error) {\r\n                    //mark as failed\r\n                    this.infoLoadingStatus = 'failed'\r\n                }\r\n            })()\r\n        } else if (this.infoLoadingStatus === 'loaded' || this.infoLoadingStatus === 'failed')\r\n            this.map.redraw()\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * Compute a tiling envelope from a geographical envelope.\r\n     * This is the function to use to know which tiles to download for a geographical view.\r\n     *\r\n     * @param {import(\"../core/GeoCanvas.js\").Envelope} e\r\n     * @returns {import(\"../core/GeoCanvas.js\").Envelope|undefined}\r\n     */\r\n    getTilingEnvelope(e) {\r\n        if (!this.info) {\r\n            this.loadInfo()\r\n            return\r\n        }\r\n\r\n        const po = this.info.originPoint,\r\n            r = this.info.resolutionGeo,\r\n            s = this.info.tileSizeCell\r\n\r\n        return {\r\n            xMin: Math.floor((e.xMin - po.x) / (r * s)),\r\n            xMax: Math.floor((e.xMax - po.x) / (r * s)),\r\n            yMin: Math.floor((e.yMin - po.y) / (r * s)),\r\n            yMax: Math.floor((e.yMax - po.y) / (r * s)),\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Request data within a geographic envelope.\r\n     *\r\n     * @param {import('../core/GeoCanvas.js').Envelope} extGeo\r\n     * @returns {this}\r\n     */\r\n    async getData(extGeo) {\r\n        if (!this.info) return this\r\n\r\n        // Get the tiling envelope and check bounds\r\n        const tb = this.getTilingEnvelope(extGeo)\r\n        if (!tb) return this\r\n\r\n        const { xMin: gbXMin, xMax: gbXMax, yMin: gbYMin, yMax: gbYMax } = this.info.tilingBounds\r\n\r\n        const xMin = Math.max(tb.xMin, gbXMin)\r\n        const xMax = Math.min(tb.xMax, gbXMax)\r\n        const yMin = Math.max(tb.yMin, gbYMin)\r\n        const yMax = Math.min(tb.yMax, gbYMax)\r\n\r\n        // Count the total number of tiles\r\n        const totalTiles = (xMax - xMin + 1) * (yMax - yMin + 1)\r\n\r\n        let processedTiles = 0 // Counter for loaded tiles\r\n        const tilePromises = []\r\n\r\n        // Iterate over tiles within bounds\r\n        for (let xT = xMin; xT <= xMax; xT++) {\r\n            for (let yT = yMin; yT <= yMax; yT++) {\r\n                if (!this.cache[xT]) this.cache[xT] = {}\r\n\r\n                // Skip already loaded or loading tiles\r\n                if (this.cache[xT][yT]) {\r\n                    ++processedTiles\r\n                    continue\r\n                }\r\n\r\n                // Mark tile as loading\r\n                this.cache[xT][yT] = 'loading'\r\n\r\n                // Prepare the tile request and store it in tilePromises\r\n                tilePromises.push(\r\n                    this.loadTile(xT, yT)\r\n                        .then((tile) => {\r\n                            // Before we increment processedTiles, check if it's the last tile\r\n                            const isLastTile = ++processedTiles === totalTiles\r\n\r\n                            // Store the tile in the cache\r\n                            this.cache[xT][yT] = tile\r\n\r\n                            // Pass the isLastTile argument to checkAndRedraw\r\n                            this.checkAndRedraw(tile, isLastTile)\r\n                        })\r\n                        .catch(() => {\r\n                            // Mark as failed but allow processing to continue\r\n                            this.cache[xT][yT] = 'failed'\r\n                            ++processedTiles\r\n                        })\r\n                )\r\n            }\r\n        }\r\n\r\n        // Handle all tile promises, including failures\r\n        await Promise.allSettled(tilePromises)\r\n        return this\r\n    }\r\n\r\n    async loadTile(xT, yT) {\r\n        try {\r\n            const data = await csv(`${this.url}${xT}/${yT}.csv`)\r\n\r\n            // Preprocess or filter the data if needed\r\n            const cells = this.preprocess ? data.filter((cell) => this.preprocess(cell) !== false) : data\r\n\r\n            if (!this.info) throw new Error('Tile info unknown')\r\n\r\n            return getGridTile(cells, xT, yT, this.info)\r\n        } catch (error) {\r\n            console.warn(`Failed to load tile ${xT}, ${yT}:`, error)\r\n            throw error\r\n        }\r\n    }\r\n\r\n    checkAndRedraw(tile, isLastTile) {\r\n        // Check if any visible layer depends on this dataset\r\n        //check if redraw is really needed, that is if:\r\n        // 1. the dataset belongs to a layer which is visible at the current zoom level\r\n        let needsRedraw = false\r\n        //go through the layers\r\n        const z = this.map.getZoom()\r\n        for (const lay of this.map.layers) {\r\n            if (lay.visible && !lay.visible(z)) continue\r\n            if (!lay.getDataset) continue\r\n            if (lay.getDataset(z) != this) continue\r\n            //found one layer. No need to seek more.\r\n            needsRedraw = true\r\n            break\r\n        }\r\n\r\n        if (!needsRedraw) return\r\n\r\n        // Check if tile intersects the current view\r\n        const env = this.map.updateExtentGeo()\r\n        const { xMin, xMax, yMin, yMax } = tile.extGeo\r\n        if (env.xMax <= xMin || env.xMin >= xMax || env.yMax <= yMin || env.yMin >= yMax) return\r\n\r\n        // Trigger redraw\r\n        if (this.onlyDrawWhenAllTilesReady) {\r\n            if (isLastTile) {\r\n                this.map.redraw()\r\n            }\r\n        } else {\r\n            this.map.redraw()\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Fill the view cache with all cells which are within a geographical envelope.\r\n     * @abstract\r\n     * @param {import(\"../core/GeoCanvas.js\").Envelope} extGeo\r\n     * @returns {void}\r\n     */\r\n    updateViewCache(extGeo) {\r\n        //\r\n        this.cellsViewCache = []\r\n\r\n        //check if info has been loaded\r\n        if (!this.info) return\r\n\r\n        //tiles within the scope\r\n        /** @type {import(\"../core/GeoCanvas.js\").Envelope|undefined} */\r\n        const tb = this.getTilingEnvelope(extGeo)\r\n        if (!tb) return\r\n\r\n        //grid bounds\r\n        /** @type {import(\"../core/GeoCanvas.js\").Envelope} */\r\n        const gb = this.info.tilingBounds\r\n\r\n        for (let xT = Math.max(tb.xMin, gb.xMin); xT <= Math.min(tb.xMax, gb.xMax); xT++) {\r\n            if (!this.cache[xT]) continue\r\n            for (let yT = Math.max(tb.yMin, gb.yMin); yT <= Math.min(tb.yMax, gb.yMax); yT++) {\r\n                //get tile\r\n                /** @type {object} */\r\n                const tile = this.cache[xT][yT]\r\n                if (!tile || typeof tile === 'string') continue\r\n\r\n                //get cells\r\n                //this.cellsViewCache = this.cellsViewCache.concat(tile.cells)\r\n\r\n                for (const cell of tile.cells) {\r\n                    if (+cell.x + this.resolution < extGeo.xMin) continue\r\n                    if (+cell.x - this.resolution > extGeo.xMax) continue\r\n                    if (+cell.y + this.resolution < extGeo.yMin) continue\r\n                    if (+cell.y - this.resolution > extGeo.yMax) continue\r\n                    this.cellsViewCache.push(cell)\r\n                }\r\n            }\r\n        }\r\n    }\r\n}\r\n\r\nfunction getGridTile(cells, xT, yT, gridInfo) {\r\n    const tile = {}\r\n\r\n    /** @type {Array.<import(\"../core/Dataset\").Cell>} */\r\n    tile.cells = cells\r\n    /** @type {number} */\r\n    tile.x = xT\r\n    /** @type {number} */\r\n    tile.y = yT\r\n\r\n    const r = gridInfo.resolutionGeo\r\n    const s = gridInfo.tileSizeCell\r\n\r\n    /** @type {import(\"../core/GeoCanvas\").Envelope} */\r\n    tile.extGeo = {\r\n        xMin: gridInfo.originPoint.x + r * s * tile.x,\r\n        xMax: gridInfo.originPoint.x + r * s * (tile.x + 1),\r\n        yMin: gridInfo.originPoint.y + r * s * tile.y,\r\n        yMax: gridInfo.originPoint.y + r * s * (tile.y + 1),\r\n    }\r\n\r\n    //convert cell coordinates into geographical coordinates\r\n    for (let cell of tile.cells) {\r\n        cell.x = tile.extGeo.xMin + cell.x * r\r\n        cell.y = tile.extGeo.yMin + cell.y * r\r\n    }\r\n\r\n    return tile\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Layer } from '../core/Layer.js'\r\n\r\n/**\r\n *\r\n * A map background layer in \"Slippy map\" XYZ standard.\r\n * See https://wiki.openstreetmap.org/wiki/Slippy_map_tilenames\r\n * https://www.maptiler.com/google-maps-coordinates-tile-bounds-projection/#6/27.88/44.48\r\n *\r\n * @module layer\r\n * @author Julien Gaffuri\r\n */\r\nexport class BackgroundLayer extends Layer {\r\n    /**\r\n     * @param {object} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** The image cache, indexed by z/y/x */\r\n        this.cache = {}\r\n\r\n        /**\r\n         * @type {string} */\r\n        this.url = opts.url\r\n        /** @type {function(number,number,number):string} */\r\n        this.urlFun = opts.urlFun || ((x, y, z) => this.url + z + '/' + x + '/' + y + '.png')\r\n\r\n        /** @type {Array.<number>} */\r\n        this.resolutions = opts.resolutions\r\n        if (!this.resolutions || this.resolutions.length == 0)\r\n            throw new Error('No resolutions provided for background layer')\r\n\r\n        /** @type {number} */\r\n        this.nbPix = opts.nbPix || 256\r\n        /** CRS coordinates of top left corner\r\n         * @type {Array.<number>} */\r\n        this.origin = opts.origin || [0, 0]\r\n        /** @type {number} */\r\n        this.z0 = opts.z0 || 0\r\n    }\r\n\r\n    /**\r\n     * Get z/x/y cache data.\r\n     * @param {number} z\r\n     * @param {number} x\r\n     * @param {number} y\r\n     * @returns {HTMLImageElement|string|undefined}\r\n     * @private\r\n     */\r\n    get(z, x, y) {\r\n        let d = this.cache[z]\r\n        if (!d) return\r\n        d = d[x]\r\n        if (!d) return\r\n        return d[y]\r\n    }\r\n\r\n    /**\r\n     * Get z/x/y cache data.\r\n     * @param {HTMLImageElement|string} img\r\n     * @param {number} z\r\n     * @param {number} x\r\n     * @param {number} y\r\n     * @returns\r\n     * @private\r\n     */\r\n    put(img, z, x, y) {\r\n        if (!this.cache[z]) this.cache[z] = {}\r\n        if (!this.cache[z][x]) this.cache[z][x] = {}\r\n        this.cache[z][x][y] = img\r\n    }\r\n\r\n    /**\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @returns {void}\r\n     */\r\n    draw(geoCanvas) {\r\n        if (!this.resolutions || this.resolutions.length == 0) {\r\n            console.error('No resolutions provided for background layer')\r\n            return\r\n        }\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const x0 = this.origin[0],\r\n            y0 = this.origin[1]\r\n\r\n        //get zoom level and resolution\r\n        let z_ = 0\r\n        for (z_ = 0; z_ < this.resolutions.length; z_++) if (this.resolutions[z_] < z) break\r\n        z_ -= 1\r\n        z_ = Math.max(0, z_)\r\n        z_ = Math.min(z_, this.resolutions.length - 1)\r\n        //console.log(this.resolutions.length, z)\r\n        const res = this.resolutions[z_]\r\n\r\n        z_ += this.z0\r\n\r\n        const sizeG = this.nbPix * res\r\n        const size = sizeG / z\r\n\r\n        //get tile numbers\r\n        const xGeoToTMS = (x) => Math.ceil((x - x0) / sizeG)\r\n        const yGeoToTMS = (y) => Math.ceil(-(y - y0) / sizeG)\r\n        const xMin = xGeoToTMS(geoCanvas.extGeo.xMin) - 1\r\n        const xMax = xGeoToTMS(geoCanvas.extGeo.xMax)\r\n        const yMax = yGeoToTMS(geoCanvas.extGeo.yMin)\r\n        const yMin = yGeoToTMS(geoCanvas.extGeo.yMax) - 1\r\n\r\n        //handle images\r\n        for (let x = xMin; x < xMax; x++) {\r\n            for (let y = yMin; y < yMax; y++) {\r\n                //get image\r\n                let img = this.get(z_, x, y)\r\n\r\n                //load image\r\n                if (!img) {\r\n                    const img = new Image()\r\n                    this.put(img, z_, x, y)\r\n                    img.onload = () => {\r\n                        geoCanvas.redraw()\r\n                    }\r\n                    img.onerror = () => {\r\n                        //case when no image\r\n                        this.put('failed', z_, x, y)\r\n                    }\r\n                    img.src = this.urlFun(x, y, z_)\r\n                    continue\r\n                }\r\n\r\n                //case when no image\r\n                if (img === 'failed') continue\r\n                if (!(img instanceof HTMLImageElement)) {\r\n                    console.log(img)\r\n                    continue\r\n                }\r\n                if (img.width == 0 || img.height == 0) continue\r\n\r\n                //draw image\r\n                const xGeo = x0 + x * sizeG\r\n                const yGeo = y0 - y * sizeG\r\n                try {\r\n                    geoCanvas.initCanvasTransform()\r\n                    geoCanvas.offscreenCtx.drawImage(\r\n                        img,\r\n                        geoCanvas.geoToPixX(xGeo),\r\n                        geoCanvas.geoToPixY(yGeo),\r\n                        size,\r\n                        size\r\n                    )\r\n                    //cg.ctx.drawImage(img, xGeo, yGeo, sizeG, -sizeG)\r\n                } catch (error) {\r\n                    console.error(error)\r\n                }\r\n            }\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Layer } from '../core/Layer.js'\r\n\r\n/**\r\n *\r\n * A map background layer composed of a single image file, geolocated.\r\n *\r\n * @module layer\r\n * @author Julien Gaffuri\r\n */\r\nexport class BackgroundLayerImage extends Layer {\r\n    /**\r\n     * @param {object} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** The image file URL\r\n         * @type {string} */\r\n        this.url = opts.url\r\n\r\n        /** The image left coordinate\r\n         * @type {number} */\r\n        this.xMin = opts.xMin || 0\r\n        /** The image top coordinate\r\n         *  @type {number} */\r\n        this.yMax = opts.yMax || 0\r\n\r\n        /** The image width, in geo unit\r\n         * @type {number} */\r\n        this.width = opts.width || 20000\r\n        /** The image height, in geo unit\r\n         * @type {number} */\r\n        this.height = opts.height || 20000\r\n\r\n        /** The image object\r\n         * @type {HTMLImageElement|undefined} */\r\n        this.img = undefined\r\n    }\r\n\r\n    /**\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @returns {void}\r\n     */\r\n    draw(geoCanvas) {\r\n        //update map extent\r\n        //geoCanvas.updateExtentGeo(0)\r\n\r\n        if (this.img) {\r\n            //the image was already downloaded: draw it\r\n\r\n            //compute screen coordinates and size ratio\r\n            const x = geoCanvas.geoToPixX(this.xMin)\r\n            const y = geoCanvas.geoToPixY(this.yMax)\r\n            const z = geoCanvas.getView().z\r\n\r\n            //draw image\r\n            geoCanvas.initCanvasTransform()\r\n            geoCanvas.offscreenCtx.drawImage(this.img, x, y, this.width / z, this.height / z)\r\n        } else {\r\n            //retrieve image\r\n\r\n            if (!this.img) {\r\n                this.img = new Image()\r\n                this.img.onload = () => {\r\n                    geoCanvas.redraw()\r\n                }\r\n                this.img.onerror = () => {\r\n                    //case when no image\r\n                    console.warn('Could not retrieve background image from', this.url)\r\n                }\r\n            }\r\n\r\n            //set URL to launch the download\r\n            this.img.src = this.url\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Layer } from '../core/Layer.js'\r\n\r\n/**\r\n *\r\n * A map WMS background layer.\r\n *\r\n * @module layer\r\n * @author Julien Gaffuri\r\n */\r\nexport class BackgroundLayerWMS extends Layer {\r\n    /**\r\n     * @param {object} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * @type {string} */\r\n        this.url = opts.url\r\n\r\n        /** @type {HTMLImageElement|undefined} */\r\n        this.img = undefined\r\n\r\n        /** @type {number|undefined} */\r\n        this.xMin = undefined\r\n        /** @type {number|undefined} */\r\n        this.xMax = undefined\r\n        /** @type {number|undefined} */\r\n        this.yMin = undefined\r\n        /** @type {number|undefined} */\r\n        this.yMax = undefined\r\n    }\r\n\r\n    /** Check if the view has moved and a new image needs to be retrieved.\r\n     * @private */\r\n    hasMoved(extGeo) {\r\n        if (extGeo.xMin != this.xMin) return true\r\n        else if (extGeo.xMax != this.xMax) return true\r\n        else if (extGeo.yMin != this.yMin) return true\r\n        else if (extGeo.yMax != this.yMax) return true\r\n        else return false\r\n    }\r\n\r\n    /**\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @returns {void}\r\n     */\r\n    draw(geoCanvas) {\r\n        //update map extent\r\n        geoCanvas.updateExtentGeo(0)\r\n\r\n        if (!this.hasMoved(geoCanvas.extGeo) && this.img) {\r\n            //the map did not move and the image was already downloaded: draw the image\r\n            geoCanvas.initCanvasTransform()\r\n            geoCanvas.offscreenCtx.drawImage(this.img, 0, 0, geoCanvas.w, geoCanvas.h)\r\n        } else {\r\n            //the map moved: retrieve new image\r\n\r\n            //\r\n            this.xMin = geoCanvas.extGeo.xMin\r\n            this.xMax = geoCanvas.extGeo.xMax\r\n            this.yMin = geoCanvas.extGeo.yMin\r\n            this.yMax = geoCanvas.extGeo.yMax\r\n\r\n            //build WMS URL\r\n            const url = []\r\n            url.push(this.url)\r\n            url.push('&width=')\r\n            url.push(geoCanvas.w)\r\n            url.push('&height=')\r\n            url.push(geoCanvas.h)\r\n            //bbox: xmin ymin xmax ymax\r\n            url.push('&bbox=')\r\n            url.push(geoCanvas.extGeo.xMin)\r\n            url.push(',')\r\n            url.push(geoCanvas.extGeo.yMin)\r\n            url.push(',')\r\n            url.push(geoCanvas.extGeo.xMax)\r\n            url.push(',')\r\n            url.push(geoCanvas.extGeo.yMax)\r\n\r\n            const urlS = url.join('')\r\n            //console.log(urlS)\r\n\r\n            if (!this.img) {\r\n                this.img = new Image()\r\n                this.img.onload = () => {\r\n                    geoCanvas.redraw()\r\n                }\r\n                this.img.onerror = () => {\r\n                    //case when no image\r\n                    console.warn('Could not retrieve WMS background image from', urlS)\r\n                }\r\n            }\r\n\r\n            //set URL to launch the download\r\n            this.img.src = urlS\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Layer } from '../core/Layer.js'\r\nimport { json } from 'd3-fetch'\r\n\r\n/**\r\n * @module layer\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class GeoJSONLayer extends Layer {\r\n    /**\r\n     * @param {object} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * @private\r\n         * @type {string} */\r\n        this.url = opts.url\r\n\r\n        /**\r\n         * A preprocess to run on each feature after loading.\r\n         * It can be used to apply some specific treatment before, format the label data, project coordinates, etc.\r\n         * Return false if the label should not be kept.\r\n         * @private\r\n         * @type {function(object):boolean} */\r\n        this.preprocess = opts.preprocess\r\n\r\n        //for points\r\n        /**\r\n         * @private\r\n         * @type {function(object,number):string} */\r\n        this.shape = opts.shape || ((f, z) => 'circle')\r\n        /**\r\n         * In pixel\r\n         * @private\r\n         * @type {function(object,number):number} */\r\n        this.size = opts.size || ((f, z) => 10)\r\n        /**\r\n         * @private\r\n         * @type {function(object,number):string} */\r\n        this.strokeStyle = opts.strokeStyle || ((f, z) => 'red')\r\n        /**\r\n         * @private\r\n         * @type {function(object,number):string} */\r\n        this.fillStyle = opts.fillStyle || ((f, z) => 'black')\r\n        /**\r\n         * In pixel\r\n         * @private\r\n         * @type {function(object,number):number} */\r\n        this.lineWidth = opts.lineWidth || ((f, z) => 2)\r\n\r\n        //for lines\r\n\r\n        /**\r\n         * @private\r\n         * @type {function(object,number):string} */\r\n        this.color = opts.color || ((f, z) => 'gray')\r\n        /**\r\n         * In pixel\r\n         * @private\r\n         * @type {function(object,number):number} */\r\n        this.width = opts.width || ((f, z) => 2)\r\n        /**\r\n         * @private\r\n         * @type {function(object,number):Array.<number>|undefined} */\r\n        this.lineDash = opts.lineDash || ((f, z) => undefined)\r\n\r\n        /**\r\n         * @private\r\n         * @type {Array.<object> | undefined} */\r\n        this.fs = undefined\r\n\r\n        /**\r\n         * @private\r\n         * @type {string} */\r\n        this.loadingStatus = 'notLoaded'\r\n    }\r\n\r\n    /**\r\n     * Draw the layer.\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @returns {void}\r\n     */\r\n    draw(geoCanvas) {\r\n        //load data, if not done yet.\r\n        if (!this.fs) {\r\n            this.load(geoCanvas.redraw)\r\n            return\r\n        }\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        for (const f of this.fs) {\r\n            const gt = f.geometry.type\r\n\r\n            if (gt == 'Point') {\r\n                const c = f.geometry.coordinates\r\n\r\n                //get style parameters for the point feature\r\n                const shape = this.shape(f, z)\r\n                if (!shape || shape == 'none') continue\r\n                const size = this.size(f, z) * z\r\n                if (!size) continue\r\n                const strokeStyle = this.strokeStyle(f, z)\r\n                const fillStyle = this.fillStyle(f, z)\r\n                const lineWidth = this.lineWidth(f, z) * z\r\n\r\n                //set canvas drawing parameters\r\n                if (strokeStyle) ctx.strokeStyle = strokeStyle\r\n                if (fillStyle) ctx.fillStyle = fillStyle\r\n                if (lineWidth) ctx.lineWidth = lineWidth\r\n\r\n                if (shape == 'circle') {\r\n                    //draw circle - fill and stroke\r\n                    ctx.beginPath()\r\n                    ctx.arc(c[0], c[1], size / 2, 0, 2 * Math.PI, false)\r\n                    if (fillStyle) ctx.fill()\r\n                    if (strokeStyle && lineWidth) ctx.stroke()\r\n                } else if (shape == 'square') {\r\n                    //draw square - fill and stroke\r\n                    ctx.beginPath()\r\n                    ctx.rect(c[0] - size / 2, c[1] - size / 2, size, size)\r\n                    if (fillStyle) ctx.fill()\r\n                    if (strokeStyle && lineWidth) ctx.stroke()\r\n                } else {\r\n                    console.error('Unexpected shape for point geojson: ' + shape)\r\n                }\r\n            } else if (gt == 'LineString') {\r\n                const cs = f.geometry.coordinates\r\n                if (cs.length < 2) continue\r\n\r\n                //set color\r\n                const col = this.color(f, z)\r\n                if (!col || col == 'none') continue\r\n                ctx.strokeStyle = col\r\n\r\n                //set linewidth\r\n                const wP = this.width(f, z)\r\n                if (!wP || wP < 0) continue\r\n                ctx.lineWidth = wP * z\r\n\r\n                //set line dash\r\n                const ldP = this.lineDash(f, z)\r\n                if (ldP) ctx.setLineDash(ldP)\r\n\r\n                //draw line\r\n                ctx.beginPath()\r\n                ctx.moveTo(cs[0][0], cs[0][1])\r\n                for (let i = 1; i < cs.length; i++) ctx.lineTo(cs[i][0], cs[i][1])\r\n                ctx.stroke()\r\n            } else {\r\n                console.log('Unsupported geometry type in GeoJSONLayer: ' + gt)\r\n            }\r\n        }\r\n\r\n        //...\r\n        ctx.setLineDash([])\r\n    }\r\n\r\n    /**\r\n     * Load data for labels, from URL this.url\r\n     * @param {function():void} callback\r\n     * @private\r\n     */\r\n    async load(callback) {\r\n        if (!this.url) {\r\n            console.log('Failed loading boundaries: No URL specified. ' + this.url)\r\n            this.loadingStatus = 'failed'\r\n            this.labels = []\r\n            return\r\n        }\r\n\r\n        //check if data already loaded\r\n        if (this.loadingStatus != 'notLoaded') return\r\n\r\n        //load data\r\n        this.loadingStatus = 'loading'\r\n\r\n        try {\r\n            const data_ = await json(this.url)\r\n\r\n            /** @type { Array.<object> } */\r\n            const data = data_.features\r\n\r\n            //preprocess/filter\r\n            if (this.preprocess) {\r\n                this.fs = []\r\n                for (const c of data) {\r\n                    const b = this.preprocess(c)\r\n                    if (b == false) continue\r\n                    this.fs.push(c)\r\n                }\r\n            } else {\r\n                //store labels\r\n                this.fs = data\r\n            }\r\n\r\n            this.loadingStatus = 'loaded'\r\n\r\n            //redraw\r\n            if (callback) callback()\r\n        } catch (error) {\r\n            console.log('Failed loading boundaries from ' + this.url)\r\n            this.fs = []\r\n            this.loadingStatus = 'failed'\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Layer } from '../core/Layer.js'\r\n\r\n/**\r\n * A layer, which specifies a dataset to be shown with specified styles.\r\n *\r\n * @module layer\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class GridLayer extends Layer {\r\n    /**\r\n     * @param {import(\"../core/Dataset\").Dataset|import(\"../core/MultiResolutionDataset\").MultiResolutionDataset} dataset The dataset to show.\r\n     * @param {Array.<import(\"../core/Style\").Style>} styles The styles, ordered in drawing order.\r\n     * @param {{visible?:function(number):boolean,alpha?:function(number):number,blendOperation?:function(number):GlobalCompositeOperation,minPixelsPerCell?:number,cellInfoHTML?:function(import(\"../core/Dataset\").Cell):string}} opts\r\n     */\r\n    constructor(dataset, styles, opts = {}) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** @type {import(\"../core/Dataset\").Dataset|import(\"../core/MultiResolutionDataset\").MultiResolutionDataset} */\r\n        this.dataset = dataset\r\n\r\n        /** @type {Array.<import(\"../core/Style\").Style>} */\r\n        this.styles = styles\r\n\r\n        /**\r\n         * This parameter is used when the dataset is a MultiResolutionDataset.\r\n         * It defines the minimum number of pixels a grid cell should have to select the dataset to display based on its resolution.\r\n         * A low value, means that the map will be more detailled (smaller cells).\r\n         * A high value, means that the map will be less detailled (larger cells).\r\n         * This value should be higher than 1, otherwise it means a grid cell is smaller than the screen resolution.\r\n         * For more complex cell representations that require some more map space, this value should be higher.\r\n         * @type {number} */\r\n        this.minPixelsPerCell = opts.minPixelsPerCell || 3\r\n\r\n        /**\r\n         * The function returning cell information as HTML.\r\n         * This is typically used for tooltip information.\r\n         * @type {function(import(\"../core/Dataset\").Cell, number):string} */\r\n        this.cellInfoHTML = opts.cellInfoHTML || GridLayer.defaultCellInfoHTML\r\n    }\r\n\r\n    /** */\r\n    draw(geoCanvas, legend) {\r\n        //get zoom level\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get layer dataset component\r\n        /** @type {import('../core/Dataset.js').Dataset|undefined} */\r\n        const dsc = this.getDataset(z)\r\n        if (!dsc) return\r\n\r\n        //launch data download, if necessary\r\n        dsc.getData(geoCanvas.extGeo)\r\n\r\n        //update dataset view cache\r\n        dsc.updateViewCache(geoCanvas.extGeo)\r\n\r\n        //draw cells, style by style\r\n        for (const s of this.styles) {\r\n            //check if style is visible\r\n            if (s.visible && !s.visible(z)) continue\r\n\r\n            //set style alpha and blend mode\r\n            //TODO: multiply by layer alpha ?\r\n            ctx.globalAlpha = s.alpha ? s.alpha(z) : 1.0\r\n            if (s.blendOperation) ctx.globalCompositeOperation = s.blendOperation(z)\r\n\r\n            //set affin transform to draw with geographical coordinates\r\n            geoCanvas.setCanvasTransform()\r\n\r\n            //draw with style\r\n            s.draw(dsc.getViewCache(), geoCanvas, dsc.getResolution())\r\n\r\n            //draw style filter\r\n            if (s.filterColor) s.drawFilter(geoCanvas)\r\n        }\r\n\r\n        //add legend element\r\n        if (legend) {\r\n            for (const s of this.styles) {\r\n                //check if style is visible\r\n                if (s.visible && !s.visible(z)) continue\r\n                GridLayer.addLegends(legend, s.legends)\r\n\r\n                //case for styles of styles, like kernel smoothing\r\n                //TODO do better\r\n                if (s['styles']) {\r\n                    for (const s2 of s['styles']) {\r\n                        if (s2.visible && !s2.visible(z)) continue\r\n                        GridLayer.addLegends(legend, s2.legends)\r\n                    }\r\n                }\r\n            }\r\n        }\r\n    }\r\n\r\n    /** @private */\r\n    static addLegends(legendComp, lg) {\r\n        if (Array.isArray(lg)) for (const lg_ of lg) this.addLegends(legendComp, lg_)\r\n        else legendComp.node().append(lg.div.node())\r\n    }\r\n\r\n    /**\r\n     * Return the relevant dataset component for a specified zoom.\r\n     *\r\n     * @param {number} z\r\n     * @returns {import(\"../core/Dataset\").Dataset|undefined}\r\n     * */\r\n    getDataset(z) {\r\n        return this.dataset.getDataset(z, this.minPixelsPerCell)\r\n    }\r\n\r\n    /**\r\n     * Set/get style stack.\r\n     *\r\n     * @param {undefined|import(\"../core/Style\").Style|Array.<import(\"../core/Style\").Style>} styles\r\n     * @returns { this | Array.<import(\"../core/Style\").Style> }\r\n     */\r\n    styles_(styles) {\r\n        if (arguments.length === 0) return this.styles\r\n        if (arguments.length === 1)\r\n            if (Array.isArray(styles)) this.styles = styles\r\n            else this.styles = [styles]\r\n        else this.styles = arguments\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * The default function returning cell information as HTML.\r\n     * This is typically used for tooltip information.\r\n     *\r\n     * @param {import(\"../core/Dataset\").Cell} cell\r\n     * @returns {string}\r\n     */\r\n    static defaultCellInfoHTML(cell) {\r\n        const buf = []\r\n        for (const key of Object.keys(cell)) {\r\n            if (key === 'x') continue\r\n            if (key === 'y') continue\r\n            buf.push('<b>', key, '</b>', ' : ', cell[key], '<br>')\r\n        }\r\n        return buf.join('')\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Layer } from '../core/Layer.js'\r\nimport { csv } from 'd3-fetch'\r\n\r\n/** A label. The name is the text to show. (x,y) are the coordinates in the same CRS as the grid.\r\n * @typedef {{name: string, x:number, y:number }} Label */\r\n\r\n/**\r\n * A (generic) layer for placename labels, to be shown on top of the grid layers.\r\n * The input is a CSV file with the position (x, y) of the labels and name + some other info on the label importance.\r\n * If the label data is not in the expected format or in the same CRS as the grid, it can be corrected with the \"preprocess\" function.\r\n * The selection of the label, their style (font, weight, etc.) and color can be specified depending on their importance and the zoom level.\r\n *\r\n * @module layer\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class LabelLayer extends Layer {\r\n    /**\r\n     * @param {object} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * The URL of the label data, as CSV file.\r\n         * The file should contain the information for each label such as the text, the position and other information for the display of the label according to the zoom level.\r\n         * If necessary, this data can be reformated with the 'preprocess' parameter.\r\n         * @private\r\n         * @type {string} */\r\n        this.url = opts.url\r\n\r\n        /** Specify if and how a label should be drawn, depending on its importance and the zoom level.\r\n         * @private\r\n         * @type {function(Label,number):string} */\r\n        this.style = opts.style || (() => 'bold 1em Arial')\r\n\r\n        /** Specify the label color, depending on its importance and the zoom level.\r\n         * @private\r\n         * @type {function(Label,number):string} */\r\n        this.color = opts.color || (opts.dark ? () => '#ddd' : () => '#222')\r\n\r\n        /** Specify the label halo color, depending on its importance and the zoom level.\r\n         * @private\r\n         * @type {function(Label,number):string} */\r\n        this.haloColor = opts.haloColor || (opts.dark ? () => '#000000BB' : () => '#FFFFFFBB')\r\n\r\n        /** Specify the label halo width, depending on its importance and the zoom level.\r\n         * @private\r\n         * @type {function(Label,number):number} */\r\n        this.haloWidth = opts.haloWidth || (() => 4)\r\n\r\n        /** The anchor where to draw the text, from label position. See HTML-canvas textAlign property.\r\n         * \"left\" || \"right\" || \"center\" || \"start\" || \"end\"\r\n         * @private\r\n         * @type {CanvasTextAlign} */\r\n        this.textAlign = opts.textAlign || 'start'\r\n\r\n        /**\r\n         * @private\r\n         * @type {Array.<number>} */\r\n        this.offsetPix = opts.offsetPix || [5, 5]\r\n\r\n        /**\r\n         * A preprocess to run on each label after loading.\r\n         * It can be used to apply some specific treatment before, format the label data, project coordinates, etc.\r\n         * Return false if the label should not be kept.\r\n         * @private\r\n         * @type {function(Label):boolean} */\r\n        this.preprocess = opts.preprocess\r\n\r\n        /**\r\n         * @private\r\n         * @type {Array.<Label> | undefined} */\r\n        this.labels = undefined\r\n\r\n        /**\r\n         * @private\r\n         * @type {string} */\r\n        this.loadingStatus = 'notLoaded'\r\n    }\r\n\r\n    /**\r\n     * Draw the label layer.\r\n     *\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @returns {void}\r\n     */\r\n    draw(geoCanvas) {\r\n        //load labels, if not done yet.\r\n        if (!this.labels) {\r\n            this.load(geoCanvas.redraw)\r\n            return\r\n        }\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //text align\r\n        ctx.textAlign = this.textAlign || 'start'\r\n\r\n        //line join and cap\r\n        ctx.lineJoin = 'bevel' //|| \"round\" || \"miter\";\r\n        ctx.lineCap = 'butt' //|| \"round\" || \"square\";\r\n\r\n        //draw in pix coordinates\r\n        geoCanvas.initCanvasTransform()\r\n\r\n        //draw labels, one by one\r\n        for (const lb of this.labels) {\r\n            //get label style\r\n            const st = this.style(lb, z)\r\n            if (!st) continue\r\n            ctx.font = st\r\n\r\n            //check label within the view, to be drawn\r\n            if (!geoCanvas.toDraw(lb)) continue\r\n\r\n            //position\r\n            const xP = geoCanvas.geoToPixX(lb.x) + this.offsetPix[0]\r\n            const yP = geoCanvas.geoToPixY(lb.y) - this.offsetPix[1]\r\n\r\n            //label stroke, for the halo\r\n            if (this.haloColor && this.haloWidth) {\r\n                const hc = this.haloColor(lb, z)\r\n                const hw = this.haloWidth(lb, z)\r\n                if (hc && hw && hw > 0) {\r\n                    ctx.strokeStyle = hc\r\n                    ctx.lineWidth = hw\r\n                    ctx.strokeText(lb.name, xP, yP)\r\n                }\r\n            }\r\n\r\n            //label fill\r\n            if (this.color) {\r\n                const col = this.color(lb, z)\r\n                if (col) {\r\n                    ctx.fillStyle = col\r\n                    ctx.fillText(lb.name, xP, yP)\r\n                }\r\n            }\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Load data for labels, from URL this.url\r\n     * @param {function():void} callback\r\n     * @private\r\n     */\r\n    async load(callback) {\r\n        if (!this.url) {\r\n            console.log('Failed loading labels: No URL specified. ' + this.url)\r\n            this.loadingStatus = 'failed'\r\n            this.labels = []\r\n            return\r\n        }\r\n\r\n        //check if data already loaded\r\n        if (this.loadingStatus != 'notLoaded') return\r\n\r\n        //load data\r\n        this.loadingStatus = 'loading'\r\n\r\n        try {\r\n            /** @type { Array.<Label> } */\r\n            const data = await csv(this.url)\r\n\r\n            //preprocess/filter\r\n            if (this.preprocess) {\r\n                this.labels = []\r\n                for (const c of data) {\r\n                    const b = this.preprocess(c)\r\n                    if (b == false) continue\r\n                    this.labels.push(c)\r\n                }\r\n            } else {\r\n                //store labels\r\n                this.labels = data\r\n            }\r\n\r\n            this.loadingStatus = 'loaded'\r\n\r\n            //redraw\r\n            if (callback) callback()\r\n        } catch (error) {\r\n            console.log('Failed loading labels from ' + this.url)\r\n            this.labels = []\r\n            this.loadingStatus = 'failed'\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Legend } from '../core/Legend.js'\r\n\r\n/**\r\n * A legend element for color categrories.\r\n *\r\n * @module legend\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class ColorCategoryLegend extends Legend {\r\n    /** @param {Object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        //col/categories array, in display order\r\n        /**\r\n         * @private\r\n         * @type {Array.<[string,string]>} */\r\n        this.colorLabel = opts.colorLabel || [['gray', '-']]\r\n\r\n        /**\r\n         * @private\r\n         * @type {import(\"../core/Style.js\").Shape} */\r\n        this.shape = opts.shape || 'circle'\r\n        this.dimension = opts.dimension || { r: 8 }\r\n        this.strokeColor = opts.strokeColor || 'gray'\r\n        this.strokeWidth = opts.strokeWidth || 1\r\n    }\r\n\r\n    /**\r\n     */\r\n    update() {\r\n        //clear\r\n        this.div.selectAll('*').remove()\r\n\r\n        //title\r\n        this.makeTitle()\r\n\r\n        //categories\r\n        const nb = this.colorLabel.length\r\n        if (nb == 0) return\r\n\r\n        for (let i = 0; i < nb; i++) {\r\n            const cat = this.colorLabel[i]\r\n\r\n            //make div for category\r\n            const d = this.div.append('div')\r\n            //to enable vertical centering\r\n            //.style(\"position\", \"relative\")\r\n\r\n            const sw = this.strokeWidth\r\n\r\n            //draw graphic element: box / circle\r\n            if (this.shape === 'square') {\r\n                const h = this.dimension.h || 15\r\n                const w = this.dimension.w || 20\r\n                d.append('div')\r\n                    .style('display', 'inline')\r\n\r\n                    .append('svg')\r\n                    .attr('width', w + 2 * sw)\r\n                    .attr('height', h + 2 * sw)\r\n\r\n                    .append('rect')\r\n                    .attr('x', sw)\r\n                    .attr('y', sw)\r\n                    .attr('width', w)\r\n                    .attr('height', h)\r\n                    .style('fill', cat[0])\r\n                    .style('stroke', this.strokeColor)\r\n                    .style('stroke-width', this.strokeWidth)\r\n            } else if (this.shape === 'circle') {\r\n                const r = this.dimension.r || 8\r\n                const h = 2 * r + 2 * sw\r\n                d.append('div')\r\n                    .style('display', 'inline')\r\n\r\n                    .append('svg')\r\n                    .attr('width', h)\r\n                    .attr('height', h)\r\n\r\n                    .append('circle')\r\n                    .attr('cx', r + sw)\r\n                    .attr('cy', r + sw)\r\n                    .attr('r', r)\r\n                    .style('fill', cat[0])\r\n                    .style('stroke', this.strokeColor)\r\n                    .style('stroke-width', this.strokeWidth)\r\n            } else {\r\n                throw new Error('Unexpected shape:' + this.shape)\r\n            }\r\n\r\n            //write label text\r\n            d.append('div')\r\n                //show on right of graphic\r\n                .style('display', 'inline')\r\n\r\n                //center vertically\r\n                //.style(\"position\", \"absolute\").style(\"top\", \"0\").style(\"bottom\", \"0\")\r\n\r\n                .style('padding-left', '5px')\r\n                .style('font-size', this.labelFontSize)\r\n                .text(cat[1])\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Legend } from '../core/Legend.js'\r\n\r\n/**\r\n * A legend element for discrete color style.\r\n * Inspiration: https://observablehq.com/@d3/color-legend\r\n *\r\n * @module legend\r\n * @author Julien Gaffuri\r\n */\r\nexport class ColorDiscreteLegend extends Legend {\r\n    /** @param {Object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** @private @type {function(import('../core/Style').ViewScale):Array.<string>} */\r\n        this.colors = opts.colors\r\n        /** @private @type {function(import('../core/Style').ViewScale):Array.<number>} */\r\n        this.breaks = opts.breaks\r\n\r\n        this.width = opts.width || 300\r\n        this.height = opts.height || 15\r\n\r\n        this.tickSize = opts.tickSize || 3\r\n\r\n        //label\r\n        this.invert = opts.invert\r\n    }\r\n\r\n    /**\r\n     * @param {{viewScale:import('../core/Style').ViewScale} } opts\r\n     */\r\n    update(opts) {\r\n        //clear\r\n        this.div.selectAll('*').remove()\r\n\r\n        //title\r\n        this.makeTitle()\r\n\r\n        //get colors and breaks\r\n        const colors = this.colors(opts.viewScale)\r\n        const breaks = this.breaks(opts.viewScale)\r\n        if (!breaks) return\r\n\r\n        //classes\r\n        const nb = colors.length\r\n        if (nb == 0) return\r\n        const w = this.width / nb\r\n\r\n        //make svg element\r\n        const svg = this.div\r\n            .append('svg')\r\n            .attr('width', this.width)\r\n            .attr('height', this.height + this.tickSize + 2 + 10)\r\n\r\n        //draw graphic elements\r\n        for (let i = 0; i < nb; i++) {\r\n            svg.append('rect')\r\n                .attr('x', i * w)\r\n                .attr('y', 0)\r\n                .attr('width', w)\r\n                .attr('height', this.height)\r\n                .style('fill', colors[i])\r\n        }\r\n\r\n        //tick line\r\n        for (let i = 1; i < nb; i++) {\r\n            svg.append('line')\r\n                .attr('x1', w * i)\r\n                .attr('y1', 0)\r\n                .attr('x2', w * i)\r\n                .attr('y2', this.height + this.tickSize)\r\n                .style('stroke', 'black')\r\n        }\r\n\r\n        //labels\r\n        for (let i = 1; i < nb; i++) {\r\n            let label = breaks[i - 1]\r\n            if (isNaN(label) || label == undefined) continue\r\n            if (this.labelFormat) label = this.labelFormat(label)\r\n\r\n            //label\r\n            svg.append('text')\r\n                .attr('id', 'ticklabel_' + i)\r\n                .attr('x', w * i)\r\n                .attr('y', this.height + this.tickSize + 2)\r\n                .style('font-size', this.labelFontSize)\r\n                //.style(\"font-weight\", \"bold\")\r\n                //.style(\"font-family\", \"Arial\")\r\n                .style('text-anchor', 'middle')\r\n                .style('alignment-baseline', 'top')\r\n                .style('dominant-baseline', 'hanging')\r\n                .style('pointer-events', 'none')\r\n                .text(label)\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Legend } from '../core/Legend.js'\r\n\r\n/**\r\n * A legend element for continuous color style.\r\n * Inspiration: https://observablehq.com/@d3/color-legend\r\n *\r\n * @module legend\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class ColorLegend extends Legend {\r\n    /** @param {Object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        //a function [0,1]->color for continuous colors.\r\n        //it can take as second argument the viewscale.\r\n        this.colorScale = opts.colorScale\r\n\r\n        //function (t[0,1]) -> value (for label text)\r\n        //it can take as second argument the viewscale.\r\n        this.textScale = opts.textScale || ((t) => t)\r\n\r\n        this.margin = opts.margin || 5\r\n\r\n        //replace with labels ?\r\n        this.tickSize = opts.tickSize || 6\r\n        this.ticks = opts.ticks || Math.floor(this.width / 50)\r\n        this.tickFormat = opts.tickFormat\r\n        this.tickUnit = opts.tickUnit\r\n\r\n        this.fontSize = opts.fontSize || '0.8em'\r\n        this.invert = opts.invert\r\n\r\n        this.width = opts.width || 300\r\n        this.height = opts.height || 15\r\n    }\r\n\r\n    /**\r\n     * @param {{viewScale:import('../core/Style').ViewScale} } opts\r\n     */\r\n    update(opts) {\r\n        //clear\r\n        this.div.selectAll('*').remove()\r\n\r\n        //title\r\n        this.makeTitle()\r\n\r\n        const svgW = this.width + 2 * this.margin\r\n        const svgH = this.height + this.margin + this.tickSize + 10\r\n        const svg = this.div.append('svg').attr('width', svgW).attr('height', svgH)\r\n        //  <rect width=\"300\" height=\"100\" style=\"fill:rgb(0,0,255);stroke-width:3;stroke:rgb(0,0,0)\" />\r\n\r\n        const g = svg.append('g').attr('transform', 'translate(' + this.margin + ' ' + 0 + ')')\r\n\r\n        //draw color bar\r\n        const w = this.width,\r\n            h = this.height\r\n        const step = 5\r\n        for (let i = 0; i < w; i += step) {\r\n            let t = i / (w - 1)\r\n            if (this.invert) t = 1 - t\r\n            g.append('rect')\r\n                .attr('x', i)\r\n                .attr('y', 0)\r\n                .attr('width', step)\r\n                .attr('height', h)\r\n                .style('fill', this.colorScale(t, opts.viewScale))\r\n        }\r\n\r\n        for (let i = 0; i < this.ticks; i++) {\r\n            let t = i / (this.ticks - 1)\r\n\r\n            //tick line\r\n            g.append('line')\r\n                .attr('x1', w * t)\r\n                .attr('y1', 0)\r\n                .attr('x2', w * t)\r\n                .attr('y2', h + this.tickSize)\r\n                .style('stroke', 'black')\r\n\r\n            //prepare tick label\r\n            g.append('text')\r\n                .attr('id', 'ticklabel_' + i)\r\n                .attr('x', w * t)\r\n                .attr('y', h + this.tickSize + 2)\r\n                .style('font-size', this.fontSize)\r\n                //.style(\"font-weight\", \"bold\")\r\n                //.style(\"font-family\", \"Arial\")\r\n                .style('text-anchor', i == 0 ? 'start' : i == this.ticks - 1 ? 'end' : 'middle')\r\n                .style('alignment-baseline', 'top')\r\n                .style('dominant-baseline', 'hanging')\r\n                .style('pointer-events', 'none')\r\n            //.text(\"-\")\r\n        }\r\n\r\n        //update tick labels\r\n\r\n        //label text format\r\n        const f = this.tickFormat && this.tickFormat != 'text' ? this.tickFormat : (v) => v\r\n        for (let i = 0; i < this.ticks; i++) {\r\n            let t = i / (this.ticks - 1)\r\n\r\n            const v = this.textScale(t, opts.viewScale)\r\n            const text = (v ? f(v) : '0') + (this.tickUnit ? this.tickUnit : '')\r\n            if (text == undefined) continue\r\n\r\n            //tick label\r\n            this.div.select('#' + 'ticklabel_' + i).text(text)\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Legend } from '../core/Legend.js'\r\n\r\n/**\r\n * A legend element for segment orientation.\r\n *\r\n * @module legend\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class OrientationLegend extends Legend {\r\n    /** @param {Object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        //orientation\r\n        this.orientation = opts.orientation || 0\r\n        //color\r\n        this.color = opts.color || ((resolution, z, viewScale) => 'gray')\r\n        //width\r\n        this.width = opts.width || ((resolution, z, viewScale) => 3 * z)\r\n        //length\r\n        this.length = opts.length || ((resolution, z, viewScale) => resolution)\r\n\r\n        //label\r\n        this.label = opts.label || '-'\r\n    }\r\n\r\n    /**\r\n     * @param {{ style: import(\"../style/SegmentStyle.js\").SegmentStyle, resolution: number, z: number, viewScale:object }} opts\r\n     */\r\n    update(opts) {\r\n        //clear\r\n        this.div.selectAll('*').remove()\r\n\r\n        //title\r\n        this.makeTitle()\r\n\r\n        const d = this.div.append('div')\r\n\r\n        //compute segment color, width and length\r\n        const color = this.color(opts.resolution, opts.z, opts.viewScale)\r\n        const widthPix = this.width(opts.resolution, opts.z, opts.viewScale) / opts.z\r\n        const lengthPix = this.length(opts.resolution, opts.z, opts.viewScale) / opts.z\r\n\r\n        //draw SVG segment\r\n        const svgS = Math.max(lengthPix, widthPix)\r\n        const svg = d.append('svg').attr('width', svgS).attr('height', svgS).style('', 'inline-block')\r\n\r\n        const cos = Math.cos((-this.orientation * Math.PI) / 180)\r\n        const sin = Math.sin((-this.orientation * Math.PI) / 180)\r\n        const dc = svgS * 0.5,\r\n            l2 = lengthPix * 0.5\r\n        svg.append('line')\r\n            .attr('x1', dc - cos * l2)\r\n            .attr('y1', dc - sin * l2)\r\n            .attr('x2', dc + cos * l2)\r\n            .attr('y2', dc + sin * l2)\r\n            .style('stroke', color)\r\n            .style('stroke-width', widthPix)\r\n\r\n        //label\r\n        d.append('div')\r\n            .style('display', 'inline')\r\n            .style('padding-left', '5px')\r\n            .style('font-size', this.labelFontSize)\r\n            .text(this.label + (this.labelUnitText ? ' ' : '') + this.labelUnitText)\r\n    }\r\n}\r\n\r\n/**\r\n *\r\n * @param {Array.<number>} orientations\r\n * @param {Array.<string>} labels\r\n * @param {object} opts\r\n * @returns  { Array.<OrientationLegend> }\r\n */\r\nexport function orientationLegend(orientations, labels, opts = {}) {\r\n    const legends = []\r\n    for (let i = 0; i < orientations.length; i++) {\r\n        opts.title = i == 0 ? opts.title : undefined\r\n        opts.orientation = orientations[i]\r\n        opts.label = labels[i]\r\n        legends.push(new OrientationLegend(opts))\r\n    }\r\n    return legends\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Legend } from '../core/Legend.js'\r\nimport { nice } from '../utils/utils.js'\r\nimport { max } from 'd3-array'\r\n\r\n/**\r\n * A legend element for proportional symbols.\r\n *\r\n * @module legend\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class SizeLegend extends Legend {\r\n    /** @param {Object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the text label, from the view scale and list of cells, resolution and zoom\r\n         *  @type { function(object, Array.<import('../core/Dataset.js').Cell>, number, number):(number|string) } */\r\n        this.label = opts.label || undefined\r\n\r\n        /** A function returning the size of the legend symbol, in geo UoM, from the viewscale, resolution and zoom\r\n         *  @type { function(object, number, number):number } */\r\n        this.size = opts.size || undefined\r\n\r\n        //symbol\r\n        /**  @type {(import(\"../core/Style\").Shape)|\"line\"} */\r\n        this.shape = opts.shape || 'circle'\r\n\r\n        //general case\r\n        this.fillColor = opts.fillColor || 'none'\r\n        this.strokeColor = opts.strokeColor || 'gray'\r\n        this.strokeWidth = opts.strokeWidth || 1\r\n\r\n        //for line shape\r\n        //TODO this.orientation = opts.orientation || 0\r\n        this.color = opts.color || 'gray'\r\n        this.length = opts.length || ((resolution, z, viewScale) => resolution)\r\n    }\r\n\r\n    /**\r\n     * @param {{ viewScale:object, resolution: number, z:number, cells:Array.<import('../core/Dataset.js').Cell> }} opts\r\n     */\r\n    update(opts) {\r\n        //clear\r\n        this.div.selectAll('*').remove()\r\n\r\n        //title\r\n        this.makeTitle()\r\n\r\n        //get label. May not be a number (!)\r\n        let label = this.label(opts.viewScale, opts.cells, opts.resolution, opts.z)\r\n\r\n        //compute size of symbol, in pix\r\n        let sizePix\r\n        if (this.size) sizePix = this.size(opts.viewScale, opts.resolution, opts.z) / opts.z\r\n        else sizePix = opts.viewScale(+label) / opts.z\r\n        if (!sizePix) return\r\n\r\n        //format label, if specified and possible\r\n        if (this.labelFormat && !isNaN(+label)) label = this.labelFormat(label)\r\n\r\n        const d = this.div.append('div')\r\n        //to enable vertical centering\r\n        //.style(\"position\", \"relative\")\r\n\r\n        //default svg construction, for square and circle\r\n        const svg = () =>\r\n            d\r\n                .append('svg')\r\n                .attr('width', sizePix + this.strokeWidth + 2)\r\n                .attr('height', sizePix + this.strokeWidth + 2)\r\n                .style('', 'inline-block')\r\n\r\n        if (this.shape === 'square') {\r\n            svg()\r\n                .append('rect')\r\n                .attr('x', 0)\r\n                .attr('y', 0)\r\n                .attr('width', sizePix)\r\n                .attr('height', sizePix)\r\n                .style('fill', this.fillColor)\r\n                .style('stroke', this.strokeColor)\r\n                .style('stroke-width', this.strokeWidth)\r\n        } else if (this.shape === 'circle') {\r\n            // <circle cx=\"50\" cy=\"50\" r=\"40\" stroke=\"black\" stroke-width=\"3\" fill=\"red\" />\r\n            const r = (sizePix + this.strokeWidth) * 0.5\r\n            svg()\r\n                .append('circle')\r\n                .attr('cx', r + 1)\r\n                .attr('cy', r + 1)\r\n                .attr('r', r)\r\n                .style('fill', this.fillColor)\r\n                .style('stroke', this.strokeColor)\r\n                .style('stroke-width', this.strokeWidth)\r\n        } else if (this.shape === 'donut') {\r\n            //TODO\r\n        } else if (this.shape === 'diamond') {\r\n            //TODO\r\n        } else if (this.shape === 'line') {\r\n            //get segment length\r\n            let lengthPix = this.length\r\n                ? this.length(opts.resolution, opts.z, opts.viewScale)\r\n                : opts.resolution\r\n            lengthPix /= opts.z\r\n\r\n            const svg = d\r\n                .append('svg')\r\n                .attr('width', lengthPix)\r\n                .attr('height', sizePix)\r\n                .style('', 'inline-block')\r\n\r\n            //TODO orientation\r\n            //<line x1=\"0\" y1=\"0\" x2=\"200\" y2=\"200\" style=\"stroke:rgb(255,0,0);stroke-width:2\" />\r\n            svg.append('line')\r\n                .attr('x1', 0)\r\n                .attr('y1', sizePix / 2)\r\n                .attr('x2', lengthPix)\r\n                .attr('y2', sizePix / 2)\r\n                .style('stroke', this.color)\r\n                .style('stroke-width', sizePix)\r\n        } else {\r\n            throw new Error('Unexpected shape:' + this.shape)\r\n        }\r\n\r\n        //label\r\n        d.append('div')\r\n            .style('display', 'inline')\r\n            .style('padding-left', '5px')\r\n            .style('font-size', this.labelFontSize)\r\n            .text(label + (this.labelUnitText ? ' ' : '') + this.labelUnitText)\r\n    }\r\n}\r\n\r\n/**\r\n * @param {Array.<number>} values\r\n * @param {function(number):number} size\r\n * @param { object } opts\r\n * @returns {Array.<SizeLegend>}\r\n */\r\nexport function sizeLegend(values, size, opts = {}) {\r\n    const legends = []\r\n    for (let value of values) {\r\n        opts.title = value == values[0] ? opts.title : undefined\r\n        opts.size = () => size(value)\r\n        opts.label = () => value\r\n        legends.push(new SizeLegend(opts))\r\n    }\r\n    return legends\r\n}\r\n\r\n/**\r\n * @param { function(import('../core/Dataset.js').Cell):number } value\r\n * @param {*} opts\r\n * @returns {Array.<SizeLegend>}\r\n */\r\nexport function sizeLegendViewScale(value, opts = {}) {\r\n    const k = opts.k || [0.9, 0.5, 0.2, 0.05]\r\n    const legends = []\r\n    for (let k_ of k) {\r\n        opts.title = k_ == k[0] ? opts.title : undefined\r\n        opts.label = (viewScale, cells) => nice(k_ * max(cells, value))\r\n        legends.push(new SizeLegend(opts))\r\n    }\r\n    return legends\r\n}\r\n\r\n/**\r\n * A function which return a stack of size legends for a discrete classification.\r\n *\r\n * @param { Array.<number> } breaks\r\n * @param { Array.<number> } sizes\r\n * @param { object } opts\r\n * @returns {Array.<SizeLegend>}\r\n */\r\nexport function sizeDiscreteLegend(breaks, sizes, opts = {}) {\r\n    const f = opts.labelFormat || ((x) => x)\r\n    const labelText = opts.labelText || defaultLabelText(f)\r\n    const legends = []\r\n    for (let i = sizes.length - 1; i >= 0; i--) {\r\n        opts.title = i == sizes.length - 1 ? opts.title : undefined\r\n        opts.size = () => sizes[i]\r\n        opts.label = () => labelText(breaks[i - 1], breaks[i])\r\n        legends.push(new SizeLegend(opts))\r\n    }\r\n    return legends\r\n}\r\n\r\n/**\r\n * A function which return a stack of size legends for a discrete classification using a viewscale.\r\n * @param { number } classNumber\r\n * @param { object } opts\r\n * @returns {Array.<SizeLegend>}\r\n */\r\nexport function sizeDiscreteViewScaleLegend(classNumber, opts = {}) {\r\n    const f = opts.labelFormat || ((x) => x)\r\n    const labelText = opts.labelText || defaultLabelText(f)\r\n    const legends = []\r\n    const viewScaleFun = opts.viewScaleFun || ((t) => t) //TODO do it differently? At sizelegend level !\r\n    for (let i = classNumber - 1; i >= 0; i--) {\r\n        opts.title = i == classNumber - 1 ? opts.title : undefined\r\n        opts.size = (viewScale) => viewScaleFun(viewScale).values[i]\r\n        opts.label = (viewScale) =>\r\n            labelText(viewScaleFun(viewScale).breaks[i - 1], viewScaleFun(viewScale).breaks[i])\r\n        legends.push(new SizeLegend(opts))\r\n    }\r\n    return legends\r\n}\r\n\r\n/**\r\n * A function that returns a function to format laberls for discrete scale legends.\r\n * @param { function(number):string } format\r\n * @returns { function(number|undefined, number|undefined): string }\r\n */\r\nfunction defaultLabelText(format) {\r\n    return (v0, v1) => {\r\n        if (v0 == undefined && v1 == undefined) return ''\r\n        if (v1 == undefined) return '> ' + format(v0)\r\n        if (v0 == undefined) return '< ' + format(v1)\r\n        return format(v0) + ' - ' + format(v1)\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { select } from 'd3-selection'\r\nimport { Legend } from '../core/Legend.js'\r\n\r\n/**\r\n *\r\n * @module legend\r\n * @author Julien Gaffuri\r\n */\r\nexport class TernaryLegend extends Legend {\r\n    /** @param {Object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        //classifier\r\n        this.classifier = opts.classifier\r\n\r\n        this.width = opts.width || 150\r\n        this.selectionColor = this.selectionColor || 'red'\r\n        this.tooltip = opts.tooltip\r\n        this.texts = opts.texts\r\n\r\n        this.leftText = opts.leftText || 'Category 0'\r\n        this.topText = opts.topText || 'Category 1'\r\n        this.rightText = opts.rightText || 'Category 2'\r\n\r\n        this.centerCoefficient = opts.centerCoefficient || this.classifier.centerCoefficient\r\n    }\r\n\r\n    /**\r\n     * @param {{} } opts\r\n     */\r\n    update(opts) {\r\n        //clear\r\n        this.div.selectAll('*').remove()\r\n\r\n        //title\r\n        this.makeTitle()\r\n\r\n        const sqrt3over2 = 0.866025\r\n        const w = this.width,\r\n            h = w * sqrt3over2\r\n        const classifier = this.classifier\r\n        const selectionColor = this.selectionColor\r\n        const selectionStrokeWidth = 0\r\n        const tt = this.tooltip\r\n        const texts = this.texts || {}\r\n\r\n        const padding = 2\r\n        const fontSize = 12\r\n\r\n        //make svg element\r\n        const svg = this.div\r\n            .append('svg')\r\n            .attr('width', w + selectionStrokeWidth)\r\n            .attr('height', h + 4 * padding + 2 * fontSize)\r\n\r\n        //top label\r\n        svg.append('text')\r\n            .attr('x', w / 2)\r\n            .attr('y', padding + fontSize)\r\n            .text(this.topText)\r\n            .attr('font-size', fontSize)\r\n            .attr('text-anchor', 'middle')\r\n        //left label\r\n        svg.append('text')\r\n            .attr('x', 0)\r\n            .attr('y', 3 * padding + 2 * fontSize + h)\r\n            .text(this.leftText)\r\n            .attr('font-size', fontSize)\r\n            .attr('text-anchor', 'start')\r\n        //right label\r\n        svg.append('text')\r\n            .attr('x', w)\r\n            .attr('y', 3 * padding + 2 * fontSize + h)\r\n            .text(this.rightText)\r\n            .attr('font-size', fontSize)\r\n            .attr('text-anchor', 'end')\r\n\r\n        //triangle group\r\n        const g = svg\r\n            .append('g')\r\n            .attr(\r\n                'transform',\r\n                'translate(' +\r\n                    selectionStrokeWidth / 2 +\r\n                    ' ' +\r\n                    (selectionStrokeWidth / 2 + (2 * padding + fontSize)) +\r\n                    ')'\r\n            )\r\n\r\n        //common function for triangle patches\r\n        const setAttributes = (elt, color, text) => {\r\n            //elt.raise();\r\n            elt.attr('fill', color)\r\n                //.attr(\"stroke\", colorOver)\r\n                //.attr(\"stroke-width\", 0)\r\n                //.attr(\"stroke-linejoin\", \"round\")\r\n                .on('mouseover', function (e) {\r\n                    /*this.parentNode.appendChild(this); select(this).attr(\"stroke-width\", selectionStrokeWidth);*/\r\n                    select(this).attr('fill', selectionColor)\r\n                    if (!tt || !text) return\r\n                    tt.html(text)\r\n                    tt.setPosition(e)\r\n                    tt.show()\r\n                })\r\n                .on('mouseout', function () {\r\n                    /*select(this).attr(\"stroke-width\", 0);*/\r\n                    select(this).attr('fill', color)\r\n                    if (tt) tt.hide()\r\n                })\r\n            if (tt && text)\r\n                elt.on('mousemove', function (e) {\r\n                    tt.setPosition(e)\r\n                })\r\n        }\r\n\r\n        //const [c0, c1, c2] = classifier.center\r\n\r\n        //trapezium s0\r\n        const t0 = g\r\n            .append('polygon')\r\n            .attr(\r\n                'points',\r\n                '0,' +\r\n                    h +\r\n                    ' ' +\r\n                    w / 3 +\r\n                    ',' +\r\n                    h +\r\n                    ' ' +\r\n                    w / 2 +\r\n                    ',' +\r\n                    (h * 2) / 3 +\r\n                    ' ' +\r\n                    w / 6 +\r\n                    ',' +\r\n                    (h * 2) / 3\r\n            )\r\n        setAttributes(t0, classifier.colors[0], texts['0'])\r\n        //trapezium s1\r\n        const t1 = g\r\n            .append('polygon')\r\n            .attr(\r\n                'points',\r\n                w / 2 +\r\n                    ',0 ' +\r\n                    (w * 2) / 3 +\r\n                    ',' +\r\n                    h / 3 +\r\n                    ' ' +\r\n                    w / 2 +\r\n                    ',' +\r\n                    (h * 2) / 3 +\r\n                    ' ' +\r\n                    w / 3 +\r\n                    ',' +\r\n                    h / 3\r\n            )\r\n        setAttributes(t1, classifier.colors[1], texts['1'])\r\n        //trapezium s2\r\n        const t2 = g\r\n            .append('polygon')\r\n            .attr(\r\n                'points',\r\n                w +\r\n                    ',' +\r\n                    h +\r\n                    ' ' +\r\n                    (w * 5) / 6 +\r\n                    ',' +\r\n                    (2 * h) / 3 +\r\n                    ' ' +\r\n                    w / 2 +\r\n                    ',' +\r\n                    (h * 2) / 3 +\r\n                    ' ' +\r\n                    (w * 2) / 3 +\r\n                    ',' +\r\n                    h\r\n            )\r\n        setAttributes(t2, classifier.colors[2], texts['2'])\r\n        //triangle s0\r\n        const t0_ = g\r\n            .append('polygon')\r\n            .attr(\r\n                'points',\r\n                w / 2 +\r\n                    ',' +\r\n                    (h * 2) / 3 +\r\n                    ' ' +\r\n                    (w * 5) / 6 +\r\n                    ',' +\r\n                    (h * 2) / 3 +\r\n                    ' ' +\r\n                    (w * 2) / 3 +\r\n                    ',' +\r\n                    h / 3\r\n            )\r\n        setAttributes(t0_, classifier.mixColors[0], texts['m12'])\r\n        //triangle s1\r\n        const t1_ = g\r\n            .append('polygon')\r\n            .attr('points', w / 2 + ',' + (h * 2) / 3 + ' ' + w / 3 + ',' + h + ' ' + (w * 2) / 3 + ',' + h)\r\n        setAttributes(t1_, classifier.mixColors[1], texts['m02'])\r\n        //triangle s2\r\n        const t2_ = g\r\n            .append('polygon')\r\n            .attr(\r\n                'points',\r\n                w / 2 + ',' + (h * 2) / 3 + ' ' + w / 6 + ',' + (h * 2) / 3 + ' ' + w / 3 + ',' + h / 3\r\n            )\r\n        setAttributes(t2_, classifier.mixColors[2], texts['m01'])\r\n\r\n        //center\r\n        if (this.centerCoefficient) {\r\n            //TODO make it an hexagon !\r\n            const center = g\r\n                .append('circle')\r\n                .attr('cx', w / 2)\r\n                .attr('cy', (h * 2) / 3)\r\n                .attr('r', (this.centerCoefficient * h) / 3)\r\n            setAttributes(center, classifier.centerColor, texts['center'])\r\n        }\r\n\r\n        /*\r\n        let middle, left, top, right, left_, bottom_, right_\r\n        if (!this.real) {\r\n\r\n            //0 left triangle\r\n            left = g.append('polygon')\r\n                .attr('points', \"0,\" + h + \" \" + (w / 3) + \",\" + h + \" \" + (w / 6) + \",\" + (2 * h / 3))\r\n            //1 top triangle\r\n            top = g.append('polygon')\r\n                .attr('points', (w / 3) + \",\" + (h / 3) + \" \" + (w * 2 / 3) + \",\" + (h / 3) + \" \" + (w / 2) + \",0\")\r\n            //2 right triangle\r\n            right = g.append('polygon')\r\n                .attr('points', (w * 2 / 3) + \",\" + h + \" \" + w + \",\" + h + \" \" + (w * 5 / 6) + \",\" + (2 * h / 3))\r\n            //middle triangle\r\n            middle = g.append('polygon')\r\n                .attr('points', (w / 2) + \",\" + (h / 3) + \" \" + (w / 4) + \",\" + (h * 5 / 6) + \" \" + (3 * w / 4) + \",\" + (h * 5 / 6))\r\n            //01 left trapezium\r\n            left_ = g.append('polygon')\r\n                .attr('points', (w / 6) + \",\" + (h * 2 / 3) + \" \" + (w / 4) + \",\" + (h * 5 / 6) + \" \" + (w / 2) + \",\" + (h / 3) + \" \" + (w / 3) + \",\" + (h / 3))\r\n            //02 bottom trapezium\r\n            bottom_ = g.append('polygon')\r\n                .attr('points', (w / 3) + \",\" + (h) + \" \" + (2 * w / 3) + \",\" + (h) + \" \" + (w * 3 / 4) + \",\" + (h * 5 / 6) + \" \" + (w / 4) + \",\" + (h * 5 / 6))\r\n            //12 right trapezium\r\n            right_ = g.append('polygon')\r\n                .attr('points', (w / 2) + \",\" + (h / 3) + \" \" + (w * 3 / 4) + \",\" + (h * 5 / 6) + \" \" + (w * 5 / 6) + \",\" + (h * 2 / 3) + \" \" + (w * 2 / 3) + \",\" + (h / 3))\r\n\r\n        } else {\r\n\r\n            //middle triangle\r\n            middle = g.append('polygon')\r\n                .attr('points', (w / 2) + \",0 0,\" + h + \" \" + w + \",\" + h)\r\n\r\n            //draw trapezium\r\n            //draw large trapezium first\r\n            for (let i_ = 2; i_ >= 0; i_--) {\r\n                const i = this.classifier.lowIndex[i_]\r\n                const r = this.classifier.lowThreshold[i]\r\n                if (i == 2)\r\n                    //01 left trapezium\r\n                    left_ = g.append('polygon')\r\n                        .attr('points', w / 2 + \",0 0,\" + h + \" \" + w * r + \",\" + h + \" \" + w * (1 + r) / 2 + \",\" + r * h)\r\n                else if (i == 1)\r\n                    //02 bottom trapezium\r\n                    bottom_ = g.append('polygon')\r\n                        .attr('points', \"0,\" + h + \" \" + w + \",\" + h + \" \" + w * (1 - r / 2) + \",\" + h * (1 - r) + \" \" + r * w / 2 + \",\" + h * (1 - r))\r\n                else\r\n                    //12 right trapezium\r\n                    right_ = g.append('polygon')\r\n                        .attr('points', w + \",\" + h + \" \" + w / 2 + \",0 \" + w * (1 - r) / 2 + \",\" + h * r + \" \" + w * (1 - r) + \",\" + h)\r\n            }\r\n\r\n            //draw triangles\r\n            //draw large triangles first\r\n            for (let i_ = 2; i_ >= 0; i_--) {\r\n                const i = this.classifier.highIndex[i_]\r\n                const r = this.classifier.highThreshold[i]\r\n\r\n                if (i == 2)\r\n                    //2 right triangle\r\n                    right = g.append('polygon')\r\n                        .attr('points', w + \",\" + h + \" \" + w * r + \",\" + h + \" \" + w * (1 + r) / 2 + \",\" + h * r)\r\n                else if (i == 1)\r\n                    //1 top triangle\r\n                    top = g.append('polygon')\r\n                        .attr('points', (w / 2) + \",0 \" + w * r / 2 + \",\" + h * (1 - r) + \" \" + w * (1 - r / 2) + \",\" + h * (1 - r))\r\n                else\r\n                    //0 left triangle\r\n                    left = g.append('polygon')\r\n                        .attr('points', \"0,\" + h + \" \" + w * (1 - r) + \",\" + h + \" \" + w * (1 - r) / 2 + \",\" + h * r)\r\n            }\r\n\r\n        }*/\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/** @typedef {\"flag\"|\"piechart\"|\"ring\"|\"segment\"|\"radar\"|\"agepyramid\"|\"halftone\"} CompositionType */\r\n\r\n/**\r\n * A style showing the composition of a total in different categories, with different color hues.\r\n * It consists of a symbol with different parts, whose size reflect the proportion of the corresponding category.\r\n * For a list of supported symbols, @see CompositionType\r\n * The symbol can be scaled depending on the cell importance.\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class CompositionStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * The dictionary (string -> color) which give the color of each category.\r\n         * @type {object} */\r\n        this.color = opts.color\r\n\r\n        /**\r\n         * A function returning the type of decomposition symbol of a cell, @see CompositionType\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number, number,object):CompositionType} */\r\n        this.type = opts.type || (() => 'flag') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the size of a cell in geographical unit.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.size = opts.size || ((c, r) => r) //(c,r,z,vs) => {}\r\n\r\n        /** For style types with stripes (flag, segment), the orientation of the stripes (0 for horizontal, other for vertical).\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number,number,object):number} */\r\n        this.stripesOrientation = opts.stripesOrientation || (() => 0) //(c,r,z,vs) => ...\r\n\r\n        /** The function specifying an offset angle for a radar, halftone or pie chart style.\r\n         * The angle is specified in degree. The rotation is anti-clockwise.\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number,number,object):number} */\r\n        this.offsetAngle = opts.offsetAngle || (() => 0) //(c,r,z,vs) => ...\r\n\r\n        /** The function specifying the height of the age pyramid, in geo unit.\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number,number,object):number} */\r\n        this.agePyramidHeight = opts.agePyramidHeight || ((c, r) => r) //(c,r,z,vs) => ...\r\n\r\n        /** For pie chart, this is parameter for internal radius, so that the pie chart looks like a donut.\r\n         * 0 for normal pie charts, 0.5 to empty half of the radius.\r\n         * @type {number} */\r\n        this.pieChartInternalRadiusFactor = opts.pieChartInternalRadiusFactor || 0\r\n    }\r\n\r\n    /**\r\n     * Draw cells as squares depending on their value.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //nb categories - used for radar and agepyramid\r\n        const nbCat = Object.entries(this.color).length\r\n\r\n        //draw calls\r\n        for (let cell of cells) {\r\n            //size\r\n            const sG = this.size ? this.size(cell, resolution, z, viewScale) : resolution\r\n            if (!sG) continue\r\n\r\n            //get offset\r\n            const offset = this.offset(cell, resolution, z)\r\n\r\n            //get symbol type\r\n            const type_ = this.type ? this.type(cell, resolution, z, viewScale) : 'flag'\r\n\r\n            //compute center position\r\n            const xc = cell.x + offset.dx + (type_ === 'agepyramid' ? 0 : resolution * 0.5)\r\n            const yc = cell.y + offset.dy + (type_ === 'agepyramid' ? 0 : resolution * 0.5)\r\n\r\n            //compute offset angle, when relevant\r\n            const offAng = this.offsetAngle\r\n                ? (this.offsetAngle(cell, resolution, z, viewScale) * Math.PI) / 180\r\n                : 0\r\n\r\n            if (type_ === 'agepyramid' || type_ === 'radar' || type_ === 'halftone') {\r\n                //get cell category max value\r\n                let maxVal = -Infinity\r\n                for (let key of Object.keys(this.color)) {\r\n                    const v = +cell[key]\r\n                    if (v > maxVal) maxVal = v\r\n                }\r\n\r\n                //cumul\r\n                let cumul = 0\r\n                if (type_ === 'agepyramid' && this.agePyramidHeight)\r\n                    cumul = (resolution - this.agePyramidHeight(cell, resolution, z, viewScale)) / 2\r\n                if (type_ === 'radar' || type_ === 'halftone') cumul = Math.PI / 2 + offAng\r\n\r\n                //compute the increment, which is the value to increment the cumul for each category\r\n                const incr =\r\n                    type_ === 'agepyramid'\r\n                        ? (this.agePyramidHeight\r\n                              ? this.agePyramidHeight(cell, resolution, z, viewScale)\r\n                              : resolution) / nbCat\r\n                        : type_ === 'radar' || type_ === 'halftone'\r\n                        ? (2 * Math.PI) / nbCat\r\n                        : undefined\r\n                if (incr === undefined) throw new Error('Unexpected symbol type:' + type_)\r\n\r\n                for (let [column, color] of Object.entries(this.color)) {\r\n                    if (type_ === 'agepyramid') {\r\n                        //set category color\r\n                        ctx.fillStyle = color\r\n\r\n                        //get category value\r\n                        const val = cell[column]\r\n\r\n                        //compute category length - in geo\r\n                        /** @type {number} */\r\n                        const wG = (sG * val) / maxVal\r\n\r\n                        //draw bar\r\n                        ctx.fillRect(xc + (resolution - wG) / 2, yc + cumul, wG, incr)\r\n\r\n                        //next height\r\n                        cumul += incr\r\n                    } else if (type_ === 'radar') {\r\n                        //set category color\r\n                        ctx.fillStyle = color\r\n\r\n                        //get categroy value\r\n                        const val = cell[column]\r\n\r\n                        //compute category radius - in geo\r\n                        /** @type {number} */\r\n                        //const rG = this.radius(val, r, stat, cellStat, z)\r\n                        const rG = (sG / 2) * Math.sqrt(val / maxVal)\r\n\r\n                        //draw angular sector\r\n                        ctx.beginPath()\r\n                        ctx.moveTo(xc, yc)\r\n                        ctx.arc(xc, yc, rG, cumul - incr, cumul)\r\n                        ctx.lineTo(xc, yc)\r\n                        ctx.fill()\r\n\r\n                        //next angular sector\r\n                        cumul += incr\r\n                    } else if (type_ === 'halftone') {\r\n                        //set category color\r\n                        ctx.fillStyle = color\r\n\r\n                        //get categroy value\r\n                        const val = cell[column]\r\n\r\n                        //compute category radius - in geo\r\n                        /** @type {number} */\r\n                        const rG = sG * 0.333 * Math.sqrt(val / maxVal)\r\n\r\n                        //draw circle\r\n                        ctx.beginPath()\r\n                        ctx.arc(\r\n                            xc + resolution * 0.25 * Math.cos(cumul),\r\n                            yc + resolution * 0.25 * Math.sin(cumul),\r\n                            rG,\r\n                            0,\r\n                            2 * Math.PI\r\n                        )\r\n                        ctx.fill()\r\n\r\n                        //next angular sector\r\n                        cumul += incr\r\n                    } else {\r\n                        throw new Error('Unexpected symbol type:' + type_)\r\n                    }\r\n                }\r\n            } else {\r\n                //compute total\r\n                let total = 0\r\n                for (let column of Object.keys(this.color)) {\r\n                    const v = +cell[column]\r\n                    if (!v) continue\r\n                    total += v\r\n                }\r\n                if (!total || isNaN(total)) continue\r\n\r\n                //draw decomposition symbol\r\n                let cumul = 0\r\n                const d = resolution * (1 - sG / resolution) * 0.5\r\n                const ori = this.stripesOrientation(cell, resolution, z, viewScale)\r\n\r\n                for (let [column, color] of Object.entries(this.color)) {\r\n                    //get share\r\n                    const share = cell[column] / total\r\n                    if (!share || isNaN(share)) continue\r\n\r\n                    //set color\r\n                    ctx.fillStyle = color\r\n\r\n                    //draw symbol part\r\n                    if (type_ === 'flag') {\r\n                        //draw flag stripe\r\n                        if (ori == 0) {\r\n                            //horizontal\r\n                            ctx.fillRect(\r\n                                cell.x + d + offset.dx,\r\n                                cell.y + d + cumul * sG + offset.dy,\r\n                                sG,\r\n                                share * sG\r\n                            )\r\n                        } else {\r\n                            //vertical\r\n                            ctx.fillRect(\r\n                                cell.x + d + cumul * sG + offset.dx,\r\n                                cell.y + d + offset.dy,\r\n                                share * sG,\r\n                                sG\r\n                            )\r\n                        }\r\n                    } else if (type_ === 'piechart') {\r\n                        //draw pie chart angular sector\r\n\r\n                        //compute angles\r\n                        const a1 = cumul * 2 * Math.PI\r\n                        const a2 = (cumul + share) * 2 * Math.PI\r\n\r\n                        //draw\r\n                        ctx.beginPath()\r\n                        ctx.moveTo(xc, yc)\r\n                        ctx.arc(xc, yc, sG * 0.5, a1 + offAng, a2 + offAng)\r\n                        if (this.pieChartInternalRadiusFactor)\r\n                            ctx.arc(\r\n                                xc,\r\n                                yc,\r\n                                sG * 0.5 * this.pieChartInternalRadiusFactor,\r\n                                a1 + offAng,\r\n                                a2 + offAng,\r\n                                true\r\n                            )\r\n                        ctx.closePath()\r\n                        ctx.fill()\r\n                    } else if (type_ === 'ring') {\r\n                        //draw ring\r\n                        ctx.beginPath()\r\n                        ctx.arc(xc, yc, Math.sqrt(1 - cumul) * sG * 0.5, 0, 2 * Math.PI)\r\n                        ctx.fill()\r\n                    } else if (type_ === 'segment') {\r\n                        //draw segment sections\r\n                        const wG = (sG * sG) / resolution\r\n                        if (ori == 0) {\r\n                            //horizontal\r\n                            ctx.fillRect(\r\n                                cell.x + offset.dx,\r\n                                cell.y + (resolution - wG) / 2 + cumul * wG + offset.dy,\r\n                                resolution,\r\n                                share * wG\r\n                            )\r\n                        } else {\r\n                            //vertical\r\n                            ctx.fillRect(\r\n                                cell.x + cumul * resolution + offset.dx,\r\n                                cell.y + (resolution - wG) / 2 + offset.dy,\r\n                                share * resolution,\r\n                                wG\r\n                            )\r\n                        }\r\n                    } else {\r\n                        throw new Error('Unexpected symbol type:' + type_)\r\n                    }\r\n\r\n                    cumul += share\r\n                }\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\nimport { randomNormal } from 'd3-random'\r\nimport { checkWebGLSupport, makeWebGLCanvas } from '../utils/webGLUtils.js'\r\nimport { WebGLSquareColoring } from '../utils/WebGLSquareColoring.js'\r\nimport { color } from 'd3-color'\r\n\r\n/**\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class DotDensityStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the number of dots for a cell value.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.dotNumber = opts.dotNumber || ((cell, resolution) => resolution / 100) //(c,r,z,vs) => {}\r\n\r\n        /** The color of the dots. Same color for all dots within a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => '#FF5733') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the size of the dots, in geo unit. Same size for all cells.\r\n         * @type {function(number, number,object):number} */\r\n        this.dotSize = opts.dotSize || ((resolution, z) => 1.5 * z) //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the sigma of the dots distribution. Same value for all cells.\r\n         * @type {function(number, number,object):number} */\r\n        this.sigma = opts.sigma || ((resolution, z) => resolution / 2) //(c,r,z,vs) => {}\r\n    }\r\n\r\n    /**\r\n     * Draw cells as text.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //get size\r\n        const sGeo = this.dotSize ? this.dotSize(resolution, z, viewScale) : z\r\n\r\n        //make random function\r\n        const sig = this.sigma ? this.sigma(resolution, z, viewScale) : resolution * 0.4\r\n        const rand = randomNormal(0, sig)\r\n\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        if (checkWebGLSupport()) {\r\n            //create canvas and webgl renderer\r\n            const cvWGL = makeWebGLCanvas(geoCanvas.w + '', geoCanvas.h + '')\r\n            if (!cvWGL) {\r\n                console.error('No webGL')\r\n                return\r\n            }\r\n\r\n            //create webGL program\r\n            const prog = new WebGLSquareColoring(cvWGL.gl, sGeo / z)\r\n\r\n            const r2 = resolution / 2\r\n\r\n            for (let cell of cells) {\r\n                //get color\r\n                const col = this.color(cell, resolution, z, viewScale)\r\n                if (!col || col === 'none') continue\r\n\r\n                //number of dots\r\n                const dotNumber = this.dotNumber(cell, resolution, z, viewScale)\r\n\r\n                //get offset\r\n                const offset = this.offset(cell, resolution, z)\r\n\r\n                //cell center\r\n                const cx = cell.x + offset.dx + r2\r\n                const cy = cell.y + offset.dy + r2\r\n\r\n                //convert color\r\n                const cc = color(col)\r\n                if (!cc) return\r\n\r\n                //random points\r\n                for (let i = 0; i <= dotNumber; i++)\r\n                    prog.addPointData2(cx + rand(), cy + rand(), cc.r, cc.g, cc.b, cc.opacity)\r\n            }\r\n\r\n            //draw\r\n            prog.draw(geoCanvas.getWebGLTransform())\r\n\r\n            //draw in canvas geo\r\n            geoCanvas.initCanvasTransform()\r\n            ctx.drawImage(cvWGL.canvas, 0, 0)\r\n        } else {\r\n            for (let cell of cells) {\r\n                //get color\r\n                const col = this.color(cell, resolution, z, viewScale)\r\n                if (!col || col === 'none') continue\r\n                //set color\r\n                ctx.fillStyle = col\r\n\r\n                //number of dots\r\n                const dotNumber = this.dotNumber(cell, resolution, z, viewScale)\r\n\r\n                //get offset\r\n                const offset = this.offset(cell, resolution, z)\r\n\r\n                //draw random dots\r\n                const cx = cell.x + offset.dx + resolution / 2,\r\n                    cy = cell.y + offset.dy + resolution / 2\r\n                for (let i = 0; i <= dotNumber; i++) {\r\n                    ctx.fillRect(cx + rand(), cy + rand(), sGeo, sGeo)\r\n                }\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class ImageStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the image URL of a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.image = opts.image || (() => '') //(c,r,z,vs) => {}\r\n\r\n        /** The image size in ground meters\r\n         *  @type {function(import('../core/Dataset.js').Cell, number, number, object):number}        */\r\n        this.size = opts.size || ((cell, resolution) => resolution)\r\n\r\n        /** Dictionnary of preloaded images. url -> image\r\n         * @private\r\n         * @type {object} */\r\n        this.cache = {}\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     * @override\r\n     */\r\n    async draw(cells, geoCanvas, resolution) {\r\n        //\r\n        const z = geoCanvas.view.z,\r\n            resolutionPix = resolution / z\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //draw in screen coordinates\r\n        geoCanvas.initCanvasTransform()\r\n\r\n        //\r\n        for (let cell of cells) {\r\n            //get cell image url\r\n            const url = this.image(cell, resolution, z, viewScale)\r\n            if (!url) continue\r\n\r\n            //size and position values\r\n            let sizePix = this.size(cell, resolution, z, viewScale) / z\r\n            if (!sizePix) continue\r\n\r\n            //get image from cache\r\n            const image = this.cache[url]\r\n\r\n            //loading, keep waiting\r\n            if (image == 'loading') return\r\n            //no image: load it\r\n            else if (!image) {\r\n                //tag as loading\r\n                this.cache[url] = 'loading'\r\n\r\n                //define image\r\n                const img = new Image()\r\n                img.onload = () => {\r\n                    //store image data in cache and redraw\r\n                    this.cache[url] = img\r\n                    geoCanvas.redraw()\r\n                }\r\n                img.onerror = () => {\r\n                    //case when no image\r\n                    console.warn('Could not retrieve image from', url)\r\n                }\r\n                //set URL to launch the download\r\n                img.src = url\r\n            } else {\r\n                //draw image\r\n                const d = (resolutionPix - sizePix) / 2\r\n                try {\r\n                    geoCanvas.offscreenCtx.drawImage(\r\n                        image,\r\n                        geoCanvas.geoToPixX(cell.x) + d,\r\n                        geoCanvas.geoToPixY(cell.y) + d - resolutionPix,\r\n                        sizePix,\r\n                        sizePix\r\n                    )\r\n                } catch (error) {\r\n                    console.error(error)\r\n                }\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\nimport { SideStyle } from './SideStyle.js'\r\n\r\n/** @typedef {{x:number,y:number,or:\"v\"|\"h\",c1:import('../core/Dataset.js').Cell|undefined,c2:import('../core/Dataset.js').Cell|undefined}} Side */\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class IsoFenceStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * The dictionary (string -> color) which give the color of each category.\r\n         * @type {object} */\r\n        this.color = opts.color\r\n\r\n        /** A function returning the height of a cell in geographical unit.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.height = opts.height || ((cell, resolution, z, viewScale) => resolution * 0.4)\r\n\r\n        /** The perspective angle, in degree, within [-180,180], from [O,x] axis.\r\n         * @type {number} */\r\n        this.angle = opts.angle != undefined ? opts.angle : 50\r\n\r\n        /** A function returning the corner line stroke style.\r\n         * @type {function(import('../core/Dataset.js').Cell,number,number,number):string} */\r\n        this.cornerLineStrokeColor = opts.cornerLineStrokeColor || ((c, r, z, angle) => '#999')\r\n\r\n        /** A function returning the corner line width.\r\n         * @type {function(import('../core/Dataset.js').Cell,number,number,number):number} */\r\n        this.cornerLineWidth = opts.cornerLineWidth || ((c, r, z, angle) => (angle % 90 == 0 ? 0 : 0.8 * z))\r\n\r\n        /**\r\n         * Show vertical cross-sections.\r\n         * @type {boolean} */\r\n        this.sVert = opts.sVert != undefined ? opts.sVert : true\r\n\r\n        /**\r\n         * Show horizontal cross-sections.\r\n         * @type {boolean} */\r\n        this.sHor = opts.sHor != undefined ? opts.sHor : true\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     * @override\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //nb categories - used for radar and agepyramid\r\n        const cats = Object.keys(this.color)\r\n\r\n        //half resolution\r\n        const r2 = resolution / 2\r\n\r\n        //get offset\r\n        // @ts-ignore\r\n        const offset = this.offset(undefined, resolution, z),\r\n            dx = offset.dx,\r\n            dy = offset.dy\r\n\r\n        //make sides\r\n        /**  @type {Array.<Side>} */\r\n        const sides = SideStyle.buildSides(\r\n            cells,\r\n            resolution,\r\n            this.angle % 180 != 90 && this.sVert,\r\n            this.angle % 180 != 0 && this.sHor\r\n        )\r\n\r\n        //\r\n        if (sides.length == 0) return\r\n\r\n        //angle in radians\r\n        const aRad = (this.angle * Math.PI) / 180,\r\n            cos = Math.cos(aRad),\r\n            sin = Math.sin(aRad)\r\n\r\n        //sort sides so that the back ones are drawn first. This depends on the angle.\r\n        //depending on distance to the reference corner point\r\n        const xCorner = Math.abs(this.angle) < 90 ? geoCanvas.extGeo.xMin : geoCanvas.extGeo.xMax\r\n        const yCorner = this.angle < 0 ? geoCanvas.extGeo.yMax : geoCanvas.extGeo.yMin\r\n        sides.sort(\r\n            (s1, s2) =>\r\n                Math.hypot(s2.x - xCorner, s2.y - yCorner) - Math.hypot(s1.x - xCorner, s1.y - yCorner)\r\n        )\r\n\r\n        //prepare function to draw corner line for a cell *c*\r\n        const drawCornerLine = (cell) => {\r\n            if (!cell) return\r\n            //line style\r\n            const lw = this.cornerLineWidth ? this.cornerLineWidth(cell, resolution, z, this.angle) : 0.8 * z\r\n            if (lw == 0) return\r\n            ctx.strokeStyle = this.cornerLineStrokeColor\r\n                ? this.cornerLineStrokeColor(cell, resolution, z, this.angle)\r\n                : '#333'\r\n            ctx.lineWidth = lw\r\n\r\n            //height - in geo\r\n            const hG = this.height(cell, resolution, z, viewScale)\r\n\r\n            //draw line\r\n            ctx.beginPath()\r\n            ctx.moveTo(cell.x + r2 + dx, cell.y + r2 + dy)\r\n            ctx.lineTo(cell.x + r2 + hG * cos + dx, cell.y + r2 + hG * sin + dy)\r\n            ctx.stroke()\r\n        }\r\n\r\n        //draw sides\r\n        ctx.lineCap = 'round'\r\n        for (let side of sides) {\r\n            const c1 = side.c1,\r\n                c2 = side.c2,\r\n                x = side.x,\r\n                y = side.y\r\n\r\n            //heights - in geo\r\n            const hG1 = c1 ? this.height(c1, resolution, z, viewScale) : 0,\r\n                hG2 = c2 ? this.height(c2, resolution, z, viewScale) : 0\r\n\r\n            //compute totals for both cells\r\n            const total1 = computeTotal(c1, cats),\r\n                total2 = computeTotal(c2, cats)\r\n            if (total1 == 0 && total2 == 0) continue\r\n\r\n            let cumul1 = 0,\r\n                cumul2 = 0\r\n            for (let [column, color] of Object.entries(this.color)) {\r\n                //draw stripe of side s and category column\r\n\r\n                //get values for both cells\r\n                let v1 = c1 ? +c1[column] : 0\r\n                let v2 = c2 ? +c2[column] : 0\r\n                if (v1 == 0 && v2 == 0) continue\r\n\r\n                //compute heights\r\n                const h1 = (hG1 * cumul1) / total1 || 0\r\n                const h1n = (hG1 * (cumul1 + v1)) / total1 || 0\r\n                const h2 = (hG2 * cumul2) / total2 || 0\r\n                const h2n = (hG2 * (cumul2 + v2)) / total2 || 0\r\n\r\n                //make path\r\n                ctx.beginPath()\r\n                if (side.or == 'h') {\r\n                    //horizontal side - vertical section\r\n                    //bottom left\r\n                    ctx.moveTo(x + h1 * cos + dx, y - r2 + h1 * sin + dy)\r\n                    //top left\r\n                    ctx.lineTo(x + h2 * cos + dx, y + r2 + h2 * sin + dy)\r\n                    //top right\r\n                    ctx.lineTo(x + h2n * cos + dx, y + r2 + h2n * sin + dy)\r\n                    //bottom right\r\n                    ctx.lineTo(x + h1n * cos + dx, y - r2 + h1n * sin + dy)\r\n                } else {\r\n                    //vertical side - horizontal section\r\n                    //bottom left\r\n                    ctx.moveTo(x - r2 + h1 * cos + dx, y + h1 * sin + dy)\r\n                    //bottom right\r\n                    ctx.lineTo(x + r2 + h2 * cos + dx, y + h2 * sin + dy)\r\n                    //top right\r\n                    ctx.lineTo(x + r2 + h2n * cos + dx, y + h2n * sin + dy)\r\n                    //top left\r\n                    ctx.lineTo(x - r2 + h1n * cos + dx, y + h1n * sin + dy)\r\n                }\r\n                //cg.ctx.closePath()\r\n\r\n                //fill\r\n                ctx.fillStyle = color\r\n                ctx.fill()\r\n\r\n                cumul1 += v1\r\n                cumul2 += v2\r\n\r\n                //TODO draw only one line\r\n                //draw corner line\r\n                //if (side.or == \"h\") {\r\n                drawCornerLine(c1)\r\n                drawCornerLine(c2)\r\n                //if (this.angle > 0 && side.or == \"h\") drawCornerLine(c2)\r\n                //else drawCornerLine(c2)\r\n                //}\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n\r\nconst computeTotal = (cell, categories) => {\r\n    if (!cell) return 0\r\n    let total = 0\r\n    for (let column of categories) {\r\n        const v = cell[column]\r\n        if (!v) continue\r\n        total += +v\r\n    }\r\n    return total || 0\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class JoyPlotStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the height of a cell in geographical unit.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.height = opts.height || ((c, r) => r * Math.random()) //(c,r,z,vs) => {}\r\n\r\n        /**\r\n         * @type {function(number,{min:number, max:number},number,number):string} */\r\n        this.lineColor = opts.lineColor || ((y, ys, r, z) => '#BBB')\r\n        /**\r\n         * @type {function(number,{min:number, max:number},number,number):number} */\r\n        this.lineWidth = opts.lineWidth || ((y, ys, r, z) => z)\r\n        /**\r\n         * @type {function(number,{min:number, max:number},number,number):string} */\r\n        this.fillColor = opts.fillColor || ((y, ys, r, z) => '#c08c5968')\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     * @override\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //index cells by y and x\r\n        /**  @type {object} */\r\n        const ind = {}\r\n        for (const cell of cells) {\r\n            let row = ind[cell.y]\r\n            if (!row) {\r\n                row = {}\r\n                ind[cell.y] = row\r\n            }\r\n            row[cell.x] = this.height(cell, resolution, z, viewScale)\r\n        }\r\n\r\n        //compute extent\r\n        const e = geoCanvas.extGeo\r\n        if (!e) return\r\n        const xMin = Math.floor(e.xMin / resolution) * resolution\r\n        const xMax = Math.floor(e.xMax / resolution) * resolution\r\n        const yMin = Math.floor(e.yMin / resolution) * resolution\r\n        const yMax = Math.floor(e.yMax / resolution) * resolution\r\n\r\n        /**  @type {{min:number, max:number}} */\r\n        const ys = { min: yMin, max: yMax }\r\n\r\n        //draw lines, row by row, stating from the top\r\n        ctx.lineJoin = 'round'\r\n        for (let y = yMax; y >= yMin; y -= resolution) {\r\n            //get row\r\n            const row = ind[y]\r\n\r\n            //no row\r\n            if (!row) continue\r\n\r\n            //place first point\r\n            ctx.beginPath()\r\n            ctx.moveTo(xMin - resolution / 2, y)\r\n\r\n            //store the previous height\r\n            /** @type {number|undefined} */\r\n            let hG_\r\n\r\n            //go through the line cells\r\n            for (let x = xMin; x <= xMax; x += resolution) {\r\n                //get column value\r\n                /** @type {number} */\r\n                let hG = row[x]\r\n                if (!hG) hG = 0\r\n\r\n                if (hG || hG_) {\r\n                    //draw line only when at least one of both values is non-null\r\n                    //TODO test bezierCurveTo\r\n                    ctx.lineTo(x + resolution / 2, y + hG)\r\n                } else {\r\n                    //else move the point\r\n                    ctx.moveTo(x + resolution / 2, y)\r\n                }\r\n                //store the previous value\r\n                hG_ = hG\r\n            }\r\n\r\n            //last point\r\n            if (hG_) ctx.lineTo(xMax + resolution / 2, y)\r\n\r\n            //draw fill\r\n            const fc = this.fillColor(y, ys, resolution, z)\r\n            if (fc && fc != 'none') {\r\n                ctx.fillStyle = fc\r\n                ctx.fill()\r\n            }\r\n\r\n            //draw line\r\n            const lc = this.lineColor(y, ys, resolution, z)\r\n            const lw = this.lineWidth(y, ys, resolution, z)\r\n            if (lc && lc != 'none' && lw > 0) {\r\n                ctx.strokeStyle = lc\r\n                ctx.lineWidth = lw\r\n                ctx.stroke()\r\n            }\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { TanakaStyle } from './TanakaStyle.js'\r\nimport { StrokeStyle } from './StrokeStyle.js'\r\nimport { SquareColorCategoryWebGLStyle } from './SquareColorCategoryWebGLStyle.js'\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class LegoStyle {\r\n    static get(value, breaks, colors, opts = {}) {\r\n        opts = opts || {}\r\n\r\n        //the colors\r\n        //http://www.jennyscrayoncollection.com/2021/06/all-current-lego-colors.html\r\n        //https://leonawicz.github.io/legocolors/reference/figures/README-plot-1.png\r\n        /*opts.colors = opts.colors || [\r\n            '#00852b', //darker green\r\n            '#afd246', //light green\r\n            '#fac80a', //dark yellow\r\n            '#bb805a', //brown\r\n            '#d67923', //mostard\r\n            '#cb4e29', //redish\r\n            '#b40000', //red\r\n            '#720012', //dark red\r\n            //\"purple\",\r\n            //\"#eee\" //whithe\r\n        ]*/\r\n\r\n        opts.colDark = opts.colDark || '#333'\r\n        opts.colBright = opts.colBright || '#aaa'\r\n        opts.widthFactor = opts.widthFactor || 0.12\r\n\r\n        //reuse tanaka as basis\r\n        const ts = TanakaStyle.get(value, breaks, colors, opts)\r\n        //style to show limits between pieces\r\n        const sst = new StrokeStyle({\r\n            strokeColor: () => '#666',\r\n            strokeWidth: (c, r, z) => 0.2 * z,\r\n            filter: opts.filter,\r\n        })\r\n\r\n        return [\r\n            ts[0],\r\n            sst,\r\n            ts[1],\r\n            new LegoTopStyle({ colDark: opts.colDark, colBright: opts.colBright, filter: opts.filter }),\r\n        ]\r\n    }\r\n\r\n    /**\r\n     * @param {function(import('../core/Dataset.js').Cell):string} code\r\n     * @param {object} color\r\n     * @param {object} opts\r\n     * @returns {Array.<Style>}\r\n     */\r\n    static getCategory(code, color, opts) {\r\n        opts = opts || {}\r\n\r\n        opts.colDark = opts.colDark || '#333'\r\n        opts.colBright = opts.colBright || '#aaa'\r\n\r\n        //\r\n        const s = new SquareColorCategoryWebGLStyle({ code: code, color: color })\r\n        //style to show limits between pieces\r\n        const sst = new StrokeStyle({ strokeColor: () => '#666', strokeWidth: (c, r, z) => 0.2 * z })\r\n\r\n        return [s, sst, new LegoTopStyle({ colDark: opts.colDark, colBright: opts.colBright })]\r\n    }\r\n}\r\n\r\n/**\r\n * A style to draw top circle of lego bricks.\r\n */\r\nclass LegoTopStyle extends Style {\r\n    /** @param {object|undefined} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n        this.colDark = opts.colDark || '#333'\r\n        this.colBright = opts.colBright || '#aaa'\r\n    }\r\n\r\n    draw(cells, geoCanvas, r) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        ctx.lineWidth = 0.6 * geoCanvas.view.z\r\n\r\n        //dark part\r\n        ctx.strokeStyle = this.colDark\r\n        for (let c of cells) {\r\n            ctx.beginPath()\r\n            ctx.arc(c.x + r * 0.5, c.y + r * 0.5, r * 0.55 * 0.5, Math.PI / 4, -Math.PI * (3 / 4), true)\r\n            ctx.stroke()\r\n        }\r\n\r\n        //bright part\r\n        ctx.strokeStyle = this.colBright\r\n        for (let c of cells) {\r\n            ctx.beginPath()\r\n            ctx.arc(c.x + r * 0.5, c.y + r * 0.5, r * 0.55 * 0.5, Math.PI / 4, -Math.PI * (3 / 4), false)\r\n            ctx.stroke()\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class MosaicStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the color of the cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => '#EA6BAC') //(c,r,z,vs) => {}\r\n\r\n        /** The mosaic factor, within [0,0.5]. Set to 0 for no mosaic effect. Set to 0.5 for strong mosaic effect.\r\n         * @type {number} */\r\n        this.mosaicFactor = opts.mosaicFactor || 0.15\r\n\r\n        /** The mosaic shadow factor, within [0,0.5]. Set to 0 for no mosaic shadow. Set to 0.5 for strong mosaic shadow.\r\n         * @type {number} */\r\n        this.shadowFactor = opts.shadowFactor || 0.2\r\n\r\n        /** The mosaic shadow color.\r\n         * @type {string} */\r\n        this.shadowColor = opts.shadowColor || '#555'\r\n    }\r\n\r\n    /**\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //set stroke style, for shadow\r\n        ctx.strokeStyle = this.shadowColor\r\n        ctx.lineWidth = this.shadowFactor * resolution\r\n        ctx.lineJoin = 'round'\r\n        ctx.lineCap = 'butt'\r\n\r\n        //function to compute position mosaic effect\r\n        const d = resolution * this.mosaicFactor\r\n        const mosaic = () => {\r\n            return { x: Math.random() * d, y: Math.random() * d }\r\n        }\r\n\r\n        for (let cell of cells) {\r\n            //set fill color\r\n            const col = this.color ? this.color(cell, resolution, z, viewScale) : undefined\r\n            if (!col || col === 'none') continue\r\n            ctx.fillStyle = col\r\n\r\n            //get offset\r\n            const offset = this.offset(cell, resolution, z)\r\n\r\n            //compute position mosaic effect\r\n            const ll = mosaic(),\r\n                ul = mosaic(),\r\n                lr = mosaic(),\r\n                ur = mosaic()\r\n\r\n            //stroke\r\n            if (this.shadowFactor > 0) {\r\n                ctx.beginPath()\r\n                ctx.moveTo(cell.x + offset.dx + ll.x, cell.y + offset.dy + ll.y)\r\n                ctx.lineTo(cell.x + offset.dx + resolution - lr.x, cell.y + offset.dy + lr.y)\r\n                ctx.lineTo(cell.x + offset.dx + resolution - ur.x, cell.y + offset.dy + resolution - ur.y)\r\n                ctx.stroke()\r\n            }\r\n\r\n            //fill\r\n\r\n            ctx.beginPath()\r\n            ctx.moveTo(cell.x + offset.dx + ll.x, cell.y + offset.dy + ll.y)\r\n            ctx.lineTo(cell.x + offset.dx + resolution - lr.x, cell.y + offset.dy + lr.y)\r\n            ctx.lineTo(cell.x + offset.dx + resolution - ur.x, cell.y + offset.dy + resolution - ur.y)\r\n            ctx.lineTo(cell.x + offset.dx + ul.x, cell.y + offset.dy + resolution - ul.y)\r\n            ctx.fill()\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class NinjaStarStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the color of the cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => '#EA6BAC') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the size of a cell, within [0,1]:\r\n         *  - 0, nothing shown\r\n         *  - 1, entire square\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.size = opts.size || ((cell, resolution) => resolution) //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the shape.\r\n         * @type {function(import(\"../core/Dataset\").Cell):string} */\r\n        this.shape = opts.shape || (() => 'o')\r\n    }\r\n\r\n    /**\r\n     *\r\n     * @param {Array.<import('../core/Dataset.js').Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        const r2 = resolution * 0.5\r\n        for (let cell of cells) {\r\n            //color\r\n            const col = this.color ? this.color(cell, resolution, z, viewScale) : undefined\r\n            if (!col || col === 'none') continue\r\n            ctx.fillStyle = col\r\n\r\n            //size - in geo unit\r\n            let k = this.size(cell, resolution, z, viewScale)\r\n            k = k < 0 ? 0 : k > 1 ? 1 : k\r\n            const sG2 = k * r2\r\n\r\n            //shape\r\n            const shape = this.shape ? this.shape(cell) : 'o'\r\n            if (shape === 'none') continue\r\n\r\n            //get offset\r\n            //TODO use\r\n            //const offset = this.offset(cell, r, z)\r\n\r\n            //center position\r\n            const cx = cell.x + r2\r\n            const cy = cell.y + r2\r\n\r\n            if (shape === 'p') {\r\n                ctx.beginPath()\r\n                ctx.moveTo(cx, cy + r2)\r\n                ctx.lineTo(cx + sG2, cy + sG2)\r\n                ctx.lineTo(cx + r2, cy)\r\n                ctx.lineTo(cx + sG2, cy - sG2)\r\n                ctx.lineTo(cx, cy - r2)\r\n                ctx.lineTo(cx - sG2, cy - sG2)\r\n                ctx.lineTo(cx - r2, cy)\r\n                ctx.lineTo(cx - sG2, cy + sG2)\r\n                ctx.fill()\r\n            } else if (shape === 'o') {\r\n                ctx.beginPath()\r\n                ctx.moveTo(cx, cy + sG2)\r\n                ctx.lineTo(cx + r2, cy + r2)\r\n                ctx.lineTo(cx + sG2, cy)\r\n                ctx.lineTo(cx + r2, cy - r2)\r\n                ctx.lineTo(cx, cy - sG2)\r\n                ctx.lineTo(cx - r2, cy - r2)\r\n                ctx.lineTo(cx - sG2, cy)\r\n                ctx.lineTo(cx - r2, cy + r2)\r\n                ctx.fill()\r\n            } else {\r\n                throw new Error('Unexpected shape:' + shape)\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class PillarStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the height of the line representing a cell, in geo unit\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.height = opts.height\r\n\r\n        /** A function returning the color of the line representing a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => '#c08c59') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the width of the line representing a cell, in geo unit\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.width = opts.width || ((cell, resolution) => 0.5 * resolution)\r\n\r\n        /** A function returning the width of the line representing a cell, in geo unit\r\n         * @type {function(number, number,object):boolean} */\r\n        this.simple = opts.simple || (() => false)\r\n\r\n        /** @type {number} */\r\n        this.viewHeightFactor = opts.viewHeightFactor || 1.5\r\n        //0,0 is the center\r\n        /** @type {number} */\r\n        this.viewSX = opts.viewSX == undefined ? 0 : opts.viewSX\r\n        /** @type {number} */\r\n        this.viewSY = opts.viewSY == undefined ? -0.5 : opts.viewSY\r\n\r\n        //TODO replace with sun location ?\r\n        /** @type {number} */\r\n        this.shadowDirection =\r\n            opts.shadowDirection == undefined ? (-40.3 * Math.PI) / 180.0 : opts.shadowDirection\r\n        /** @type {number} */\r\n        this.shadowFactor = opts.shadowFactor || 0.3\r\n        /** @type {string} */\r\n        this.shadowColor = opts.shadowColor || '#00000033'\r\n\r\n        /** @type {string} */\r\n        this.outlineCol = opts.outlineCol || '#FFFFFF'\r\n        /** @type {number} */\r\n        this.outlineWidthPix = opts.outlineWidthPix == undefined ? 0.5 : opts.outlineWidthPix\r\n    }\r\n\r\n    /**\r\n     * Draw cells as segments.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //get view center geo position\r\n        const cvx = geoCanvas.view.x + this.viewSX * geoCanvas.w * z\r\n        const cvy = geoCanvas.view.y + this.viewSY * geoCanvas.h * z\r\n        //get view height\r\n        const H = this.viewHeightFactor * (geoCanvas.w + geoCanvas.h) * 0.5 * z\r\n\r\n        //sort cells by y and x\r\n        //const distToViewCenter = (c) => { const dx = cvx - c.x, dy = cvy - c.y; return Math.sqrt(dx * dx + dy * dy) }\r\n        cells.sort((c1, c2) => 100000000 * (c2.y - c1.y) + c1.x - c2.x)\r\n\r\n        //get simple information\r\n        const simple = this.simple(resolution, z, viewScale)\r\n\r\n        ctx.lineCap = simple ? 'butt' : 'round'\r\n\r\n        //draw shadows\r\n        ctx.strokeStyle = this.shadowColor\r\n        ctx.fillStyle = this.shadowColor\r\n        for (let cell of cells) {\r\n            //width\r\n            /** @type {number|undefined} */\r\n            const wG = this.width ? this.width(cell, resolution, z, viewScale) : undefined\r\n            if (!wG || wG < 0) continue\r\n\r\n            //height\r\n            /** @type {number|undefined} */\r\n            const hG = this.height ? this.height(cell, resolution, z, viewScale) : undefined\r\n            if (!hG || hG < 0) continue\r\n\r\n            //get offset\r\n            //TODO use that\r\n            //const offset = this.offset(c, resolution, z)\r\n\r\n            //set width\r\n            ctx.lineWidth = wG\r\n\r\n            //compute cell center postition\r\n            const cx = cell.x + resolution / 2\r\n            const cy = cell.y + resolution / 2\r\n            const ls = hG * this.shadowFactor\r\n\r\n            //draw segment\r\n            ctx.beginPath()\r\n            ctx.moveTo(cx, cy)\r\n            ctx.lineTo(cx + ls * Math.cos(this.shadowDirection), cy + ls * Math.sin(this.shadowDirection))\r\n            ctx.stroke()\r\n\r\n            /*\r\n            if (this.simple) {\r\n                //draw base circle\r\n                cg.ctx.beginPath();\r\n                cg.ctx.arc(\r\n                    cx, cy,\r\n                    wG * 0.5,\r\n                    0, 2 * Math.PI, false);\r\n                //cg.ctx.stroke();\r\n                cg.ctx.fill();\r\n            }*/\r\n        }\r\n\r\n        //draw pillars\r\n        for (let cell of cells) {\r\n            //color\r\n            /** @type {string|undefined} */\r\n            const col = this.color ? this.color(cell, resolution, z, viewScale) : undefined\r\n            if (!col) continue\r\n\r\n            //width\r\n            /** @type {number|undefined} */\r\n            const wG = this.width ? this.width(cell, resolution, z, viewScale) : undefined\r\n            if (!wG || wG < 0) continue\r\n\r\n            //height\r\n            /** @type {number|undefined} */\r\n            const hG = this.height ? this.height(cell, resolution, z, viewScale) : undefined\r\n            if (!hG || hG < 0) continue\r\n\r\n            //get offset\r\n            //TODO use that\r\n            //const offset = this.offset(c, resolution, z)\r\n\r\n            //compute cell center postition\r\n            const cx = cell.x + resolution / 2\r\n            const cy = cell.y + resolution / 2\r\n\r\n            //compute angle\r\n            const dx = cx - cvx,\r\n                dy = cy - cvy\r\n            const a = Math.atan2(dy, dx)\r\n            const D = Math.sqrt(dx * dx + dy * dy)\r\n            const d = (D * hG) / (H - hG)\r\n\r\n            if (simple) {\r\n                //draw segment\r\n                ctx.strokeStyle = col\r\n                ctx.lineWidth = wG\r\n                ctx.beginPath()\r\n                ctx.moveTo(cx, cy)\r\n                ctx.lineTo(cx + d * Math.cos(a), cy + d * Math.sin(a))\r\n                ctx.stroke()\r\n            } else {\r\n                //draw background segment\r\n                ctx.strokeStyle = this.outlineCol\r\n                ctx.lineWidth = wG + 2 * this.outlineWidthPix * z\r\n                ctx.beginPath()\r\n                ctx.moveTo(cx, cy)\r\n                ctx.lineTo(cx + d * Math.cos(a), cy + d * Math.sin(a))\r\n                ctx.stroke()\r\n\r\n                //draw segment\r\n                ctx.strokeStyle = col\r\n                ctx.lineWidth = wG\r\n                ctx.beginPath()\r\n                ctx.moveTo(cx, cy)\r\n                ctx.lineTo(cx + d * Math.cos(a), cy + d * Math.sin(a))\r\n                ctx.stroke()\r\n\r\n                //draw top circle\r\n                ctx.strokeStyle = this.outlineCol\r\n                //cg.ctx.fillStyle = \"#c08c59\"\r\n                ctx.lineWidth = this.outlineWidthPix * z\r\n                ctx.beginPath()\r\n                ctx.arc(cx + d * Math.cos(a), cy + d * Math.sin(a), wG * 0.5, 0, 2 * Math.PI, false)\r\n                ctx.stroke()\r\n                //cg.ctx.fill();\r\n            }\r\n        }\r\n\r\n        //in case...\r\n        ctx.lineCap = 'butt'\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * A style where each cell is represented by a segment whose length, width, color and orientation can vary according to statistical values.\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class SegmentStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the color of the cell segment.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => '#EA6BAC') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the width of the segment representing a cell, in geo unit\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.width = opts.width || ((cell, resolution) => resolution * 0.1) //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the length of the segment representing a cell, in geo unit\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.length = opts.length || ((cell, resolution) => resolution * 0.9) //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the orientation (in degrees) of the segment representing a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.orientation = opts.orientation || (() => 180 * Math.random()) //(c,r,z,vs) => {}\r\n    }\r\n\r\n    /**\r\n     * Draw cells as segments.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //\r\n        ctx.lineCap = 'butt'\r\n\r\n        //conversion factor degree -> radian\r\n        const f = Math.PI / 180\r\n\r\n        for (let cell of cells) {\r\n            //color\r\n            /** @type {string|undefined} */\r\n            const col = this.color ? this.color(cell, resolution, z, viewScale) : undefined\r\n            if (!col) continue\r\n\r\n            //width\r\n            /** @type {number|undefined} */\r\n            const wG = this.width ? this.width(cell, resolution, z, viewScale) : undefined\r\n            if (!wG || wG < 0) continue\r\n\r\n            //length\r\n            /** @type {number|undefined} */\r\n            const lG = this.length ? this.length(cell, resolution, z, viewScale) : undefined\r\n            if (!lG || lG < 0) continue\r\n\r\n            //orientation (in radian)\r\n            /** @type {number} */\r\n            const or = this.orientation(cell, resolution, z, viewScale) * f\r\n            if (or === undefined || isNaN(or)) continue\r\n\r\n            //get offset\r\n            const offset = this.offset(cell, resolution, z)\r\n\r\n            //set color and width\r\n            ctx.strokeStyle = col\r\n            ctx.lineWidth = wG\r\n\r\n            //compute segment center postition\r\n            const cx = cell.x + resolution / 2 + offset.dx\r\n            const cy = cell.y + resolution / 2 + offset.dy\r\n\r\n            //compute segment direction\r\n            const dx = 0.5 * Math.cos(or) * lG\r\n            const dy = 0.5 * Math.sin(or) * lG\r\n\r\n            //draw segment\r\n            ctx.beginPath()\r\n            ctx.moveTo(cx - dx, cy - dy)\r\n            ctx.lineTo(cx + dx, cy + dy)\r\n            ctx.stroke()\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ viewScale: viewScale, resolution: resolution, z: z, cells: cells })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * A very generic style that shows grid cells with specific color, size and shape.\r\n * It can be used to show variables as cell colors, cell size, cell shape, or any combination of the three visual variables.\r\n *\r\n * @module style\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class ShapeColorSizeStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n        /** @type {(function(import('../core/Dataset.js').Cell, number, number, object):string) | string} */\r\n        this.color = opts.color || '#EA6BAC'\r\n\r\n        /** @type {(function(import('../core/Dataset.js').Cell, number, number, object):number) | number} */\r\n        this.size = opts.size || ((cell, resolution) => resolution)\r\n\r\n        /** @type {(function(import(\"../core/Dataset.js\").Cell,number, number,object):import(\"../core/Style.js\").Shape) | string} */\r\n        this.shape = opts.shape || 'square'\r\n    }\r\n\r\n    /**\r\n     * Draw cells as squares, with various colors and sizes.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells - The grid cells to draw.\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas - The canvas to draw on.\r\n     * @param {number} resolution - Resolution of the grid.\r\n     * @override\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //zoom\r\n        const z = geoCanvas.view.z\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        const r2 = resolution * 0.5\r\n\r\n        // Precompute if color, size, and shape are functions, for efficiency\r\n        const isColorFunction = typeof this.color === 'function'\r\n        const isSizeFunction = typeof this.size === 'function'\r\n        const isShapeFunction = typeof this.shape === 'function'\r\n\r\n        const defaultColor = this.color || 'black'\r\n        const defaultSize = this.size || resolution\r\n        const defaultShape = this.shape || 'square'\r\n\r\n        // Optimized\r\n        const colorFunction = isColorFunction ? this.color : null\r\n        const sizeFunction = isSizeFunction ? this.size : null\r\n        const shapeFunction = isShapeFunction ? this.shape : null\r\n\r\n        for (let c of cells) {\r\n            // Determine color\r\n            //@ts-ignore\r\n            const col = colorFunction ? colorFunction(c, resolution, z, viewScale) : defaultColor\r\n            if (!col || col === 'none') continue\r\n\r\n            // Determine size\r\n            //@ts-ignore\r\n            const size = sizeFunction ? sizeFunction(c, resolution, z, viewScale) : defaultSize\r\n            if (!size) continue\r\n\r\n            // Determine shape\r\n            //@ts-ignore\r\n            const shape = shapeFunction ? shapeFunction(c, resolution, z, viewScale) : defaultShape\r\n            if (shape === 'none') continue\r\n\r\n            //get offset\r\n            const offset = this.offset(c, resolution, z)\r\n\r\n            //get context\r\n            const ctx = geoCanvas.offscreenCtx\r\n            ctx.fillStyle = col\r\n            if (shape === 'square') {\r\n                //draw square\r\n                const d = resolution * (1 - size / resolution) * 0.5\r\n                ctx.fillRect(c.x + d + offset.dx, c.y + d + offset.dy, size, size)\r\n            } else if (shape === 'circle') {\r\n                //draw circle\r\n                ctx.beginPath()\r\n                ctx.arc(c.x + r2 + offset.dx, c.y + r2 + offset.dy, size * 0.5, 0, 2 * Math.PI, false)\r\n                ctx.fill()\r\n            } else if (shape === 'donut') {\r\n                //draw donut\r\n                const xc = c.x + r2 + offset.dx,\r\n                    yc = c.y + r2 + offset.dy\r\n                ctx.beginPath()\r\n                ctx.moveTo(xc, yc)\r\n                ctx.arc(xc, yc, r2, 0, 2 * Math.PI)\r\n                ctx.arc(xc, yc, (1 - size / resolution) * r2, 0, 2 * Math.PI, true)\r\n                ctx.closePath()\r\n                ctx.fill()\r\n            } else if (shape === 'diamond') {\r\n                const s2 = size * 0.5\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x + r2 - s2, c.y + r2)\r\n                ctx.lineTo(c.x + r2, c.y + r2 + s2)\r\n                ctx.lineTo(c.x + r2 + s2, c.y + r2)\r\n                ctx.lineTo(c.x + r2, c.y + r2 - s2)\r\n                ctx.fill()\r\n            } else if (shape === 'triangle_up') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x - dr2, c.y - dr2)\r\n                ctx.lineTo(c.x + r2, c.y + resolution + dr2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y - dr2)\r\n                ctx.fill()\r\n            } else if (shape === 'triangle_down') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x - dr2, c.y + resolution + dr2)\r\n                ctx.lineTo(c.x + r2, c.y - dr2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y + resolution + dr2)\r\n                ctx.fill()\r\n            } else if (shape === 'triangle_left') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x + resolution + dr2, c.y + resolution + dr2)\r\n                ctx.lineTo(c.x - dr2, c.y + r2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y - dr2)\r\n                ctx.fill()\r\n            } else if (shape === 'triangle_right') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x - dr2, c.y - dr2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y + r2)\r\n                ctx.lineTo(c.x - dr2, c.y + resolution + dr2)\r\n                ctx.fill()\r\n            } else {\r\n                throw new Error('Unexpected shape:' + shape)\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ viewScale: viewScale, resolution: resolution, z: z, cells: cells })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { SideStyle } from './SideStyle.js'\r\n\r\n/**\r\n * A style to show the sides of grid cells based on their different categories.\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class SideCategoryStyle extends SideStyle {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the category code of a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number):string} */\r\n        this.code = opts.code\r\n\r\n        /**\r\n         * The dictionary (string -> color) which give the color of each category.\r\n         * @type {object} */\r\n        this.color = opts.color\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //build sides\r\n\r\n        /**  @type {Array.<import('./SideStyle.js').Side>} */\r\n        const sides = SideStyle.buildSides(cells, resolution)\r\n        if (sides.length == 0) return\r\n\r\n        //get side view scale\r\n        const viewScale = this.viewScale ? this.viewScale(sides, resolution, z) : undefined\r\n\r\n        //draw sides\r\n\r\n        ctx.lineCap = 'butt'\r\n        const r2 = resolution * 0.5\r\n        for (let side of sides) {\r\n            //get category codes for both cells\r\n            const code1 = side.c1 ? this.code(side.c1, resolution, z) : undefined\r\n            const code2 = side.c2 ? this.code(side.c2, resolution, z) : undefined\r\n            if (code1 == code2) continue\r\n\r\n            //width\r\n            /** @type {number|undefined} */\r\n            const wG = this.width ? this.width(side, resolution, z, viewScale) : undefined\r\n            if (!wG || wG <= 0) continue\r\n            const w2 = wG * 0.5\r\n\r\n            //set width\r\n            ctx.lineWidth = wG\r\n\r\n            //draw segment with correct orientation\r\n            if (side.or === 'h') {\r\n                //top line\r\n                if (code2) {\r\n                    ctx.beginPath()\r\n                    ctx.strokeStyle = this.color[code2]\r\n                    ctx.moveTo(side.x - r2, side.y + w2)\r\n                    ctx.lineTo(side.x + r2, side.y + w2)\r\n                    ctx.stroke()\r\n                }\r\n\r\n                //bottom line\r\n                if (code1) {\r\n                    ctx.beginPath()\r\n                    ctx.strokeStyle = this.color[code1]\r\n                    ctx.moveTo(side.x - r2, side.y - w2)\r\n                    ctx.lineTo(side.x + r2, side.y - w2)\r\n                    ctx.stroke()\r\n                }\r\n            } else {\r\n                //right line\r\n                if (code2) {\r\n                    ctx.beginPath()\r\n                    ctx.strokeStyle = this.color[code2]\r\n                    ctx.moveTo(side.x + w2, side.y - r2)\r\n                    ctx.lineTo(side.x + w2, side.y + r2)\r\n                    ctx.stroke()\r\n                }\r\n\r\n                //left line\r\n                if (code1) {\r\n                    ctx.beginPath()\r\n                    ctx.strokeStyle = this.color[code1]\r\n                    ctx.moveTo(side.x - w2, side.y - r2)\r\n                    ctx.lineTo(side.x - w2, side.y + r2)\r\n                    ctx.stroke()\r\n                }\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/** @typedef {{ x:number, y:number, or:\"v\"|\"h\", c1:(import('../core/Dataset').Cell)|undefined, c2:(import('../core/Dataset').Cell)|undefined }} Side */\r\n\r\n/**\r\n * @typedef {function(Array.<Side>,number, number):*} SideViewScale */\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class SideStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts = {}) {\r\n        super(opts)\r\n\r\n        /** A function returning the color of a cell side.\r\n         * @type {function(Side, number, number, object):string} */\r\n        this.color = opts.color || ((side, resolution, z, sideViewScale) => '#EA6BAC')\r\n\r\n        /** A function returning the width of a cell side, in geo unit\r\n         * @type {function(Side, number, number, object):number} */\r\n        this.width = opts.width || ((side, resolution, z, sideViewScale) => resolution / 5)\r\n\r\n        /** A function returning the length of a cell side, in geo unit\r\n         * @type {function(Side, number, number, object):number} */\r\n        this.length = opts.length || ((side, resolution, z, sideViewScale) => resolution)\r\n\r\n        /** Set to A or true so that the side is drawn as a diamond */\r\n        this.diamond = opts.diamond\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {number} resolution\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //build sides\r\n\r\n        /**  @type {Array.<Side>} */\r\n        const sides = SideStyle.buildSides(cells, resolution)\r\n        if (sides.length == 0) return\r\n\r\n        //get side view scale\r\n        const viewScale = this.viewScale ? this.viewScale(sides, resolution, z) : undefined\r\n\r\n        //draw sides\r\n\r\n        ctx.lineCap = 'butt'\r\n        const r2 = resolution * 0.5\r\n        for (let side of sides) {\r\n            //color\r\n            /** @type {string|undefined} */\r\n            const col = this.color ? this.color(side, resolution, z, viewScale) : undefined\r\n            if (!col || col == 'none') continue\r\n\r\n            if (this.diamond) {\r\n                //set color\r\n                ctx.fillStyle = col\r\n\r\n                //draw diamond\r\n                const x = side.x,\r\n                    y = side.y\r\n                ctx.beginPath()\r\n                ctx.moveTo(x - r2, y)\r\n                ctx.lineTo(x, y + r2)\r\n                ctx.lineTo(x + r2, y)\r\n                ctx.lineTo(x, y - r2)\r\n                ctx.closePath()\r\n                ctx.fill()\r\n            } else {\r\n                //width\r\n                /** @type {number|undefined} */\r\n                const wG = this.width ? this.width(side, resolution, z, viewScale) : undefined\r\n                if (!wG || wG <= 0) continue\r\n\r\n                //length\r\n                /** @type {number|undefined} */\r\n                const lG = this.length ? this.length(side, resolution, z, viewScale) : undefined\r\n                if (!lG || lG <= 0) continue\r\n                const lG2 = lG * 0.5\r\n\r\n                //set width\r\n                ctx.lineWidth = wG\r\n                //set color\r\n                ctx.strokeStyle = col\r\n\r\n                //draw segment with correct orientation\r\n                const x = side.x,\r\n                    y = side.y\r\n                ctx.beginPath()\r\n                if (side.or === 'v') {\r\n                    ctx.moveTo(x, y - lG2)\r\n                    ctx.lineTo(x, y + lG2)\r\n                } else {\r\n                    ctx.moveTo(x - lG2, y)\r\n                    ctx.lineTo(x + lG2, y)\r\n                }\r\n                ctx.stroke()\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n\r\n    /**\r\n     *\r\n     * @param {Array.<import('../core/Dataset').Cell>} cells The cells to use to build the sides.\r\n     * @param {number} resolution The cells resolution\r\n     * @param {boolean} withHorizontal Set to true to build horizontal sides, false otherwise.\r\n     * @param {boolean} withVertical Set to true to build vertical sides, false otherwise.\r\n     * @param {boolean} center Set to true so that the side coordinate are those of its center point rather than its left/bottom point (the side x,y coordinates are those of the left point for horizontal sides, and of the bottom point for vertical sides)\r\n     * @returns { Array.<Side> }\r\n     */\r\n    static buildSides(cells, resolution, withHorizontal = true, withVertical = true, center = true) {\r\n        /** @type { Array.<Side> } */\r\n        const sides = []\r\n\r\n        const r2 = center ? resolution / 2 : 0\r\n\r\n        //make horizontal sides\r\n        //sort cells by x and y\r\n        cells.sort((c1, c2) => (c2.x == c1.x ? c1.y - c2.y : c1.x - c2.x))\r\n        let c1 = cells[0]\r\n        for (let i = 1; i < cells.length; i++) {\r\n            let c2 = cells[i]\r\n\r\n            if (c1.y + resolution == c2.y && c1.x == c2.x)\r\n                //cells in same column and touch along horizontal side\r\n                //make shared side\r\n                sides.push({\r\n                    or: 'h',\r\n                    x: c1.x + r2,\r\n                    y: c2.y,\r\n                    c1: c1,\r\n                    c2: c2,\r\n                })\r\n            else {\r\n                //cells do not touch along horizontal side\r\n                //make two sides: top one for c1, bottom for c2\r\n                sides.push({\r\n                    or: 'h',\r\n                    x: c1.x + r2,\r\n                    y: c1.y + resolution,\r\n                    c1: c1,\r\n                    c2: undefined,\r\n                })\r\n                sides.push({\r\n                    or: 'h',\r\n                    x: c2.x + r2,\r\n                    y: c2.y,\r\n                    c1: undefined,\r\n                    c2: c2,\r\n                })\r\n            }\r\n\r\n            c1 = c2\r\n        }\r\n\r\n        //make vertical sides\r\n        //sort cells by y and x\r\n        cells.sort((c1, c2) => (c2.y == c1.y ? c1.x - c2.x : c1.y - c2.y))\r\n        c1 = cells[0]\r\n        for (let i = 1; i < cells.length; i++) {\r\n            let c2 = cells[i]\r\n\r\n            if (c1.x + resolution == c2.x && c1.y == c2.y)\r\n                //cells in same row and touch along vertical side\r\n                //make shared side\r\n                sides.push({\r\n                    or: 'v',\r\n                    x: c1.x + resolution,\r\n                    y: c1.y + r2,\r\n                    c1: c1,\r\n                    c2: c2,\r\n                })\r\n            else {\r\n                //cells do not touch along vertical side\r\n                //make two sides: right one for c1, left for c2\r\n                sides.push({\r\n                    or: 'v',\r\n                    x: c1.x + resolution,\r\n                    y: c1.y + r2,\r\n                    c1: c1,\r\n                    c2: undefined,\r\n                })\r\n                sides.push({\r\n                    or: 'v',\r\n                    x: c2.x,\r\n                    y: c2.y + r2,\r\n                    c1: undefined,\r\n                    c2: c2,\r\n                })\r\n            }\r\n\r\n            c1 = c2\r\n        }\r\n        return sides\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\nimport { makeWebGLCanvas } from '../utils/webGLUtils.js'\r\nimport { WebGLSquareColoringCatAdvanced } from '../utils/WebGLSquareColoringCatAdvanced.js'\r\n\r\n/**\r\n * Style based on webGL\r\n * To show cells as colored squares, from categories.\r\n * All cells are drawn as squares, with the same size\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class SquareColorCategoryWebGLStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * A function returning the category code of the cell, for coloring.\r\n         * @type {function(import('../core/Dataset.js').Cell):string} */\r\n        this.code = opts.code\r\n\r\n        /**\r\n         * The dictionary (code -> color) which gives the color of each category code.\r\n         * @type {object} */\r\n        opts.color = opts.color || undefined\r\n\r\n        /** @type { Array.<string> } */\r\n        const codes = Object.keys(opts.color)\r\n\r\n        /** @type { object } @private */\r\n        this.catToI = {}\r\n        for (let i = 0; i < codes.length; i++) this.catToI[codes[i]] = i + ''\r\n\r\n        /** @type { Array.<string> } @private */\r\n        this.colors = []\r\n        for (const code of codes) this.colors.push(opts.color['' + code])\r\n\r\n        /**\r\n         * A function returning the size of the cells, in geographical unit. All cells have the same size.\r\n         * @type {function(number,number):number} */\r\n        this.size = opts.size // (resolution, z) => ...\r\n\r\n        /**\r\n         * @private\r\n         * @type { WebGLSquareColoringCatAdvanced } */\r\n        this.wgp = new WebGLSquareColoringCatAdvanced(this.colors)\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n\r\n        //add vertice and fragment data\r\n        const r2 = resolution / 2\r\n        let c,\r\n            nb = cells.length\r\n        const verticesBuffer = []\r\n        const iBuffer = []\r\n        for (let i = 0; i < nb; i++) {\r\n            c = cells[i]\r\n            const cat = this.code(c)\r\n            if (cat == undefined) {\r\n                console.log('Unexpected category: ' + cat)\r\n                continue\r\n            }\r\n            /** @type {number} */\r\n            const i_ = this.catToI[cat]\r\n            if (isNaN(+i_)) {\r\n                console.log('Unexpected category index: ' + cat + ' ' + i_)\r\n                continue\r\n            }\r\n            verticesBuffer.push(c.x + r2, c.y + r2)\r\n            iBuffer.push(+i_)\r\n        }\r\n\r\n        //create canvas and webgl renderer\r\n        const cvWGL = makeWebGLCanvas(geoCanvas.w + '', geoCanvas.h + '')\r\n        if (!cvWGL) {\r\n            console.error('No webGL')\r\n            return\r\n        }\r\n\r\n        //draw\r\n        const sizeGeo = this.size ? this.size(resolution, z) : resolution + 0.2 * z\r\n        this.wgp.draw(cvWGL.gl, verticesBuffer, iBuffer, geoCanvas.getWebGLTransform(), sizeGeo / z)\r\n\r\n        //draw in canvas geo\r\n        geoCanvas.initCanvasTransform()\r\n        geoCanvas.offscreenCtx.drawImage(cvWGL.canvas, 0, 0)\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\nimport { makeWebGLCanvas } from '../utils/webGLUtils.js'\r\nimport { WebGLSquareColoringAdvanced } from '../utils/WebGLSquareColoringAdvanced.js'\r\n\r\n/**\r\n * Style based on webGL\r\n * To show cells as colored squares, with computation of the colors on GPU side (faster than JavaScript side).\r\n * Alls squares with the same size\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class SquareColorWebGLStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * A function returning a t value (within [0,1]) for a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell,number,number,object):number} */\r\n        this.tFun = opts.tFun //(c,r,z,vs) => {}\r\n\r\n        /**\r\n         * Distribution stretching method.\r\n         * The stretching is performed on GPU side (fragment shader).\r\n         * @type {{ fun:string, alpha:number }} */\r\n        this.stretching = opts.stretching\r\n\r\n        /**\r\n         * The sample of the color ramp.\r\n         * The color is computed on GPU side (fragment shader) based on those values (linear interpolation).\r\n         * @type {Array.<string>} */\r\n        this.colors =\r\n            opts.colors ||\r\n            [\r\n                'rgb(158, 1, 66)',\r\n                'rgb(248, 142, 83)',\r\n                'rgb(251, 248, 176)',\r\n                'rgb(137, 207, 165)',\r\n                'rgb(94, 79, 162)',\r\n            ].reverse()\r\n        if (opts.color)\r\n            this.colors = [\r\n                opts.color(0),\r\n                opts.color(0.2),\r\n                opts.color(0.4),\r\n                opts.color(0.6),\r\n                opts.color(0.8),\r\n                opts.color(1),\r\n            ]\r\n\r\n        /**\r\n         * Define the opacity of the style, within [0,1].\r\n         * If this opacity is defined, the individual color opacity will be ignored.\r\n         * @type {function(number,number):number} */\r\n        this.opacity = opts.opacity // (r,z) => ...\r\n\r\n        /**\r\n         * A function returning the size of the cells, in geographical unit. All cells have the same size.\r\n         * @type {function(number,number):number} */\r\n        this.size = opts.size // (resolution, z) => ...\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //create canvas and webgl renderer\r\n        //for opacity control, see: https://webglfundamentals.org/webgl/lessons/webgl-and-alpha.html\r\n        const cvWGL = makeWebGLCanvas(\r\n            geoCanvas.w + '',\r\n            geoCanvas.h + '',\r\n            this.opacity != undefined ? { premultipliedAlpha: false } : undefined\r\n        )\r\n        if (!cvWGL) {\r\n            console.error('No webGL')\r\n            return\r\n        }\r\n\r\n        //add vertice and fragment data\r\n        const r2 = resolution / 2\r\n        const verticesBuffer = []\r\n        const tBuffer = []\r\n        for (let cell of cells) {\r\n            const t = this.tFun(cell, resolution, z, viewScale)\r\n            if (t == null || t == undefined) continue\r\n            verticesBuffer.push(cell.x + r2, cell.y + r2)\r\n            tBuffer.push(t > 1 ? 1 : t < 0 ? 0 : t)\r\n        }\r\n\r\n        //compute pixel size\r\n        const sizeGeo = this.size ? this.size(resolution, z) : resolution + 0.2 * z\r\n\r\n        //compute opacity\r\n        const op = this.opacity ? this.opacity(resolution, z) : undefined\r\n\r\n        //\r\n        const wgp = new WebGLSquareColoringAdvanced(cvWGL.gl, this.colors, this.stretching, sizeGeo / z, op)\r\n\r\n        //draw\r\n        wgp.draw(verticesBuffer, tBuffer, geoCanvas.getWebGLTransform())\r\n\r\n        // draw in canvas geo\r\n        // NOTE: drawing each tile this way is very inefficient. WebGL is best used with fewer, heavier/larger draw calls.\r\n        geoCanvas.initCanvasTransform()\r\n        geoCanvas.offscreenCtx.drawImage(cvWGL.canvas, 0, 0)\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class StrokeStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the color of the cell.\r\n         * @type {function(import('../core/Dataset.js').Cell,number,number,object):string} */\r\n        this.strokeColor = opts.strokeColor || (() => '#666') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the size of a cell in geographical unit.\r\n         * @type {function(import('../core/Dataset.js').Cell,number,number,object):number} */\r\n        this.size = opts.size || ((cell, resolution) => resolution) //(c,r,z,vs) => {}\r\n\r\n        /** The stroke line width in geographical unit.\r\n         * @type {function(import('../core/Dataset.js').Cell,number,number,object):number} */\r\n        this.strokeWidth = opts.strokeWidth || ((cell, resolution, z) => z * 1.5) //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the shape of a cell.\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number,number,object):import(\"../core/Style.js\").Shape} */\r\n        this.shape = opts.shape || (() => 'square') //(c,r,z,vs) => {}\r\n    }\r\n\r\n    /**\r\n     * Draw cells as squares, with various colors and size.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        const r2 = resolution * 0.5\r\n        for (let c of cells) {\r\n            //color\r\n            const col = this.strokeColor ? this.strokeColor(c, resolution, z, viewScale) : undefined\r\n            if (!col || col === 'none') continue\r\n            ctx.strokeStyle = col\r\n\r\n            //size - in geo unit\r\n            const size = this.size ? this.size(c, resolution, z, viewScale) : resolution\r\n\r\n            //width\r\n            const wi = this.strokeWidth ? this.strokeWidth(c, resolution, z, viewScale) : 1 * z\r\n            if (!wi || wi <= 0) continue\r\n            ctx.lineWidth = wi\r\n\r\n            //shape\r\n            const shape = this.shape ? this.shape(c, resolution, z, viewScale) : 'square'\r\n            if (shape === 'none') continue\r\n\r\n            //get offset\r\n            const offset = this.offset(c, resolution, z)\r\n\r\n            if (shape === 'square') {\r\n                //draw square\r\n                const d = resolution * (1 - size / resolution) * 0.5\r\n                ctx.beginPath()\r\n                ctx.rect(c.x + d + offset.dx, c.y + d + offset.dy, size, size)\r\n                ctx.stroke()\r\n            } else if (shape === 'circle') {\r\n                //draw circle\r\n                ctx.beginPath()\r\n                ctx.arc(c.x + r2 + offset.dx, c.y + r2 + offset.dy, size * 0.5, 0, 2 * Math.PI, false)\r\n                ctx.stroke()\r\n            } else if (shape === 'diamond') {\r\n                const s2 = size * 0.5\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x + r2 - s2, c.y + r2)\r\n                ctx.lineTo(c.x + r2, c.y + r2 + s2)\r\n                ctx.lineTo(c.x + r2 + s2, c.y + r2)\r\n                ctx.lineTo(c.x + r2, c.y + r2 - s2)\r\n                ctx.lineTo(c.x + r2 - s2, c.y + r2)\r\n                ctx.stroke()\r\n            } else if (shape === 'donut') {\r\n                console.error('Not implemented')\r\n            } else if (shape === 'triangle_up') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x - dr2, c.y - dr2)\r\n                ctx.lineTo(c.x + r2, c.y + resolution + dr2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y - dr2)\r\n                ctx.closePath()\r\n                ctx.stroke()\r\n            } else if (shape === 'triangle_down') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x - dr2, c.y + resolution + dr2)\r\n                ctx.lineTo(c.x + r2, c.y - dr2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y + resolution + dr2)\r\n                ctx.closePath()\r\n                ctx.stroke()\r\n            } else if (shape === 'triangle_left') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x + resolution + dr2, c.y + resolution + dr2)\r\n                ctx.lineTo(c.x - dr2, c.y + r2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y - dr2)\r\n                ctx.closePath()\r\n                ctx.stroke()\r\n            } else if (shape === 'triangle_right') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x - dr2, c.y - dr2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y + r2)\r\n                ctx.lineTo(c.x - dr2, c.y + resolution + dr2)\r\n                ctx.closePath()\r\n                ctx.stroke()\r\n            } else {\r\n                throw new Error('Unexpected shape:' + shape)\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { SideStyle } from './SideStyle.js'\r\nimport { SquareColorCategoryWebGLStyle } from './SquareColorCategoryWebGLStyle.js'\r\nimport { classifier as clFun, colorClassifier as cclFun } from '../utils/scale.js'\r\n\r\n/**\r\n * @see https://manifold.net/doc/mfd9/example__tanaka_contours.htm\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class TanakaStyle {\r\n    /**\r\n     * @param {function(import('../core/Dataset.js').Cell):number} value Function that returns the value of a cell\r\n     * @param {Array.<number>} breaks The break values\r\n     * @param {Array.<string>} colors The colors, one more than the break values\r\n     * @param {object} opts\r\n     * @returns {Array.<import(\"../core/Style\").Style>}\r\n     */\r\n    static get(value, breaks, colors, opts = {}) {\r\n        //shadow colors\r\n        opts.colorDark = opts.colorDark || '#111'\r\n        opts.colorBright = opts.colorBright || '#ddd'\r\n\r\n        /** @type { function(number, number):number } */\r\n        opts.width =\r\n            opts.width ||\r\n            ((sideValue, resolution, z) => {\r\n                const minWG = 1 * z\r\n                const maxWG = 4 * z\r\n                const step = (maxWG - minWG) / 3\r\n                return Math.min(minWG + (sideValue - 1) * step, maxWG)\r\n            })\r\n\r\n        //make classifier\r\n        const classifier = clFun(breaks)\r\n        //make colors table\r\n        const colorsDict = {}\r\n        for (let i = 0; i < colors.length; i++) colorsDict[i + ''] = colors[i]\r\n\r\n        const cellStyle = new SquareColorCategoryWebGLStyle({\r\n            code: (cell) => classifier(value(cell)),\r\n            color: colorsDict,\r\n        })\r\n\r\n        const getSideValue = (side) => {\r\n            const cl1 = side.c1 ? classifier(value(side.c1)) : -1\r\n            const cl2 = side.c2 ? classifier(value(side.c2)) : -1\r\n            return cl1 - cl2\r\n        }\r\n\r\n        /** The side style, for the shadow effect */\r\n        const sideStyle = new SideStyle({\r\n            //white or black, depending on orientation and value\r\n            color: (side) => {\r\n                const v = getSideValue(side)\r\n                if (v === 0) return\r\n                if (side.or === 'v') return v < 0 ? opts.colorBright : opts.colorDark\r\n                return v < 0 ? opts.colorDark : opts.colorBright\r\n            },\r\n            //width depends on the value, that is the number of classes of difference\r\n            width: (side, resolution, z) => opts.width(Math.abs(getSideValue(side)), resolution, z),\r\n        })\r\n\r\n        return [cellStyle, sideStyle]\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class TextStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the text of a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.text = opts.text || (() => 'X') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the color of the cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => 'black') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the font size of a cell in geo unit.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number,object):number} */\r\n        this.fontSize = opts.fontSize || ((cell, resolution) => resolution) //(c,r,z,vs) => {}\r\n\r\n        /** The text font family.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.fontFamily = opts.fontFamily || (() => 'Arial')\r\n\r\n        /** The text font weight.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.fontWeight = opts.fontWeight || (() => 'bold')\r\n    }\r\n\r\n    /**\r\n     * Draw cells as text.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //draw with HTML canvas\r\n        //in screen coordinates\r\n        geoCanvas.initCanvasTransform()\r\n\r\n        for (let cell of cells) {\r\n            //get cell text\r\n            const text = this.text ? this.text(cell, resolution, z, viewScale) : undefined\r\n            if (text == undefined || text == null || text + '' === '') continue\r\n\r\n            //color\r\n            const col = this.color ? this.color(cell, resolution, z, viewScale) : undefined\r\n            if (!col) continue\r\n            ctx.fillStyle = col\r\n\r\n            //font size\r\n            //size - in pixel unit\r\n            const fontSizePix = this.fontSize(cell, resolution, z, viewScale) / z\r\n            if (!fontSizePix) continue\r\n\r\n            //set font\r\n            const fontFamily = this.fontFamily ? this.fontFamily(cell, resolution, z, viewScale) : 'Arial'\r\n            const fontWeight = this.fontWeight ? this.fontWeight(cell, resolution, z, viewScale) : 'bold'\r\n            ctx.font = fontWeight + ' ' + fontSizePix + 'px ' + fontFamily\r\n\r\n            //get offset\r\n            const offset = this.offset(cell, resolution, z)\r\n\r\n            //text position\r\n            ctx.textAlign = 'center'\r\n            const tx = geoCanvas.geoToPixX(cell.x + resolution * 0.5 + offset.dx)\r\n            const ty = geoCanvas.geoToPixY(cell.y + resolution * 0.5 + offset.dy) + fontSizePix * 0.3 //it should be 0.5 but 0.3 seems to work better\r\n\r\n            //draw the text\r\n            ctx.fillText(text, tx, ty)\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n\r\n    /**\r\n     * Build a function [0,1]->string for characters legend\r\n     *\r\n     * @param {Array.<string>} chars\r\n     * @param {(function(number):number)|undefined} scale\r\n     * @returns {function(number):string}\r\n     */\r\n    static textScale(chars, scale = undefined) {\r\n        const nb = chars.length\r\n        return (t) => {\r\n            if (scale) t = scale(t)\r\n            if (t == 0) return ''\r\n            if (t >= 1) return chars[nb - 1]\r\n            return chars[Math.floor(t * nb)]\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/** @typedef {\"first\"|\"bottom\"|\"center\"|\"top\"|\"last\"} AnchorModeYEnum */\r\n\r\n/**\r\n * Show cell as timeseries chart\r\n * Can be used for sparkline map of https://datagistips.hypotheses.org/488\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class TimeSeriesStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** The columns of the time series, ordered in chronological order.\r\n         * @type {Array.<string>} */\r\n        this.ts = opts.ts\r\n\r\n        /** A function specifying when a value should be considered as \"no data\" and thus not ignored. The line will have a break at these values.\r\n         * @type {function(string):boolean} */\r\n        this.noData = opts.noData || ((v) => v === undefined || v == '' || v === null || isNaN(+v))\r\n\r\n        //x\r\n        /** in geo unit\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number,number):number} */\r\n        this.offsetX = opts.offsetX || ((c, r, z) => 0)\r\n        /** @type {function(import(\"../core/Dataset.js\").Cell,number,number):number} */\r\n        this.width = opts.width || ((c, r, z) => r)\r\n\r\n        //y\r\n        /** in geo unit\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number,number):number} */\r\n        this.offsetY = opts.offsetY || ((c, r, z) => 0)\r\n        /** @type {function(import(\"../core/Dataset.js\").Cell,number,number):number} */\r\n        this.height = opts.height || ((c, r, z) => r)\r\n        /** @type {function(import(\"../core/Dataset.js\").Cell,number,number):AnchorModeYEnum} */\r\n        this.anchorModeY = opts.anchorModeY || ((c, r, z) => 'center')\r\n\r\n        /** A function returning the width of the line, in geo unit\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.lineWidth = opts.lineWidth || ((v, r, s, z) => 1.5 * z)\r\n\r\n        /** A function returning the color of the chart.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => 'black') //(c,r,z,vs) => {}\r\n    }\r\n\r\n    /**\r\n     * Draw cells as text.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //compute cell amplitude\r\n        const getAmplitude = (c) => {\r\n            let min, max\r\n            for (let t of this.ts) {\r\n                const val = c[t]\r\n                if (val == undefined) continue\r\n                if (min == undefined || val < min) min = val\r\n                if (max == undefined || val > max) max = val\r\n            }\r\n            if (min == undefined) return undefined\r\n            return max - min\r\n        }\r\n\r\n        //compute max amplitude\r\n        let ampMax\r\n        for (let c of cells) {\r\n            const amp = getAmplitude(c)\r\n            if (amp == undefined) continue\r\n            if (ampMax == undefined || amp > ampMax) ampMax = amp\r\n        }\r\n        if (!ampMax) return\r\n\r\n        const nb = this.ts.length\r\n\r\n        ctx.lineCap = 'butt'\r\n        for (let c of cells) {\r\n            //line width\r\n            /** @type {number|undefined} */\r\n            const wG = this.lineWidth ? this.lineWidth(c, resolution, z, viewScale) : undefined\r\n            if (!wG || wG < 0) continue\r\n\r\n            //line color\r\n            /** @type {string|undefined} */\r\n            const col = this.color ? this.color(c, resolution, z, viewScale) : undefined\r\n            if (!col) continue\r\n\r\n            //x\r\n            const offX = this.offsetX ? this.offsetX(c, resolution, z) : 0\r\n            if (offX == undefined || isNaN(offX)) continue\r\n            const w = this.width ? this.width(c, resolution, z) : resolution\r\n            if (w == undefined || isNaN(w)) continue\r\n\r\n            //y\r\n            const offY = this.offsetY ? this.offsetY(c, resolution, z) : 0\r\n            if (offY == undefined || isNaN(offY)) continue\r\n            const h = this.height ? this.height(c, resolution, z) : resolution\r\n            if (h == undefined || isNaN(h)) continue\r\n            const anchY = this.anchorModeY ? this.anchorModeY(c, resolution, z) : 'center'\r\n            if (!anchY) continue\r\n\r\n            ctx.lineWidth = wG\r\n            ctx.strokeStyle = col\r\n\r\n            //compute anchor Y figures\r\n            let val0, y0\r\n            if (anchY === 'first') {\r\n                //get first value\r\n                val0 = c[this.ts[0]]\r\n                y0 = 0\r\n            } else if (anchY === 'last') {\r\n                //get last value\r\n                val0 = c[this.ts[this.ts.length - 1]]\r\n                y0 = 0\r\n            } else if (anchY === 'bottom') {\r\n                //get min\r\n                for (let t of this.ts) {\r\n                    const val = +c[t]\r\n                    if (val == undefined) continue\r\n                    if (val0 == undefined || val < val0) val0 = val\r\n                }\r\n                y0 = 0\r\n            } else if (anchY === 'top') {\r\n                //get max\r\n                for (let t of this.ts) {\r\n                    const val = +c[t]\r\n                    if (val == undefined) continue\r\n                    if (val0 == undefined || val > val0) val0 = val\r\n                }\r\n                y0 = resolution\r\n            } else if (anchY === 'center') {\r\n                //get min and max\r\n                let min, max\r\n                for (let t of this.ts) {\r\n                    const val = c[t]\r\n                    if (val == undefined) continue\r\n                    if (min == undefined || val < min) min = val\r\n                    if (max == undefined || val > max) max = val\r\n                }\r\n                val0 = (+max + +min) * 0.5\r\n                y0 = resolution / 2\r\n            } else {\r\n                console.log('Unexpected anchorModeY: ' + anchY)\r\n                continue\r\n            }\r\n\r\n            /*/draw line\r\n            if (val0 == undefined || isNaN(val0)) continue\r\n            cg.ctx.beginPath()\r\n            const sX = w / (nb - 1)\r\n            for (let i = 0; i < nb; i++) {\r\n                const val = c[this.ts[i]]\r\n                if (val == undefined || isNaN(val)) break\r\n                if (i == 0)\r\n                    cg.ctx.moveTo(c.x + i * sX + offX, c.y + y0 + (val - val0) * h / ampMax + offY)\r\n                else\r\n                    cg.ctx.lineTo(c.x + i * sX + offX, c.y + y0 + (val - val0) * h / ampMax + offY)\r\n            }\r\n            cg.ctx.stroke()*/\r\n\r\n            //draw line, segment by segment\r\n            const sX = w / (nb - 1)\r\n\r\n            //handle first point\r\n            let v0 = c[this.ts[0]]\r\n            if (!this.noData(v0)) {\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x + offX, c.y + y0 + ((v0 - val0) * h) / ampMax + offY)\r\n            }\r\n            //console.log(v0, isNaN(v0))\r\n\r\n            let v1\r\n            for (let i = 1; i < nb; i++) {\r\n                v1 = c[this.ts[i]]\r\n\r\n                //draw segment from v0 to v1\r\n\r\n                //both points 'no data'\r\n                if (this.noData(v0) && this.noData(v1)) {\r\n                    //second point 'no data'\r\n                } else if (!this.noData(v0) && this.noData(v1)) {\r\n                    ctx.stroke()\r\n\r\n                    //first point 'no data'\r\n                } else if (this.noData(v0) && !this.noData(v1)) {\r\n                    ctx.beginPath()\r\n                    ctx.moveTo(c.x + i * sX + offX, c.y + y0 + ((v1 - val0) * h) / ampMax + offY)\r\n\r\n                    //both points have data: trace line\r\n                } else {\r\n                    ctx.lineTo(c.x + i * sX + offX, c.y + y0 + ((v1 - val0) * h) / ampMax + offY)\r\n                    //if it is the last point, stroke\r\n                    if (i == nb - 1) ctx.stroke()\r\n                }\r\n                v0 = v1\r\n            }\r\n        }\r\n\r\n        //update legend, if any\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\n'use strict'\n\nimport { initShaderProgram, createShader } from './webGLUtils.js'\nimport { color } from 'd3-color'\n\n/**\n * Everything to easily draw colored squares with webGL.\n * All the same size, but different fill color.\n */\nexport class WebGLSquareColoring {\n    /**\n     *\n     * @param {WebGLRenderingContext} gl\n     */\n    constructor(gl, sizePix) {\n        this.gl = gl\n        this.sizePix = sizePix || 10.0\n\n        this.program = initShaderProgram(\n            gl,\n            createShader(\n                gl,\n                gl.VERTEX_SHADER,\n                `\n            attribute vec2 pos;\n            uniform float sizePix;\n            uniform mat3 mat;\n            attribute vec4 color;\n            varying vec4 vColor;\n            void main() {\n              gl_Position = vec4(mat * vec3(pos, 1.0), 1.0);\n              gl_PointSize = sizePix;\n              vColor = color;\n            }\n          `\n            ),\n            createShader(\n                gl,\n                gl.FRAGMENT_SHADER,\n                `\n            precision mediump float;\n            varying vec4 vColor;\n            void main(void) {\n                vec4 vColor_ = vColor / 255.0;\n                vColor_[3] = 255.0 * vColor_[3];\n                gl_FragColor = vColor_;\n            }`\n            )\n        )\n        gl.useProgram(this.program)\n\n        //buffer data\n        this.verticesBuffer = []\n        this.colorsBuffer = []\n    }\n\n    /** Add data to vertices/size/color buffers for color squares drawing */\n    addPointData(xC, yC, col) {\n        //convert color\n        const cc = color(col)\n        //const cc = {r:45,g:87,b:98,opacity:0.9}\n        if (!cc) return\n\n        //vertices\n        this.verticesBuffer.push(xC, yC)\n        //color\n        this.colorsBuffer.push(cc.r, cc.g, cc.b, cc.opacity)\n    }\n\n    addPointData2(xC, yC, r, g, b, opacity) {\n        //vertices\n        this.verticesBuffer.push(xC, yC)\n        //color\n        this.colorsBuffer.push(r, g, b, opacity)\n    }\n\n    /**  */\n    draw(transfoMat) {\n        const gl = this.gl\n\n        //vertice data\n        gl.bindBuffer(gl.ARRAY_BUFFER, gl.createBuffer())\n        gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(this.verticesBuffer), gl.STATIC_DRAW)\n        const position = gl.getAttribLocation(this.program, 'pos')\n        gl.vertexAttribPointer(\n            position,\n            2, //numComponents\n            gl.FLOAT, //type\n            false, //normalise\n            0, //stride\n            0 //offset\n        )\n        gl.enableVertexAttribArray(position)\n\n        //color data\n        gl.bindBuffer(gl.ARRAY_BUFFER, gl.createBuffer())\n        gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(this.colorsBuffer), gl.STATIC_DRAW)\n        var color = gl.getAttribLocation(this.program, 'color')\n        gl.vertexAttribPointer(color, 4, gl.FLOAT, false, 0, 0)\n        gl.enableVertexAttribArray(color)\n\n        //sizePix\n        gl.uniform1f(gl.getUniformLocation(this.program, 'sizePix'), 1.0 * this.sizePix)\n\n        //transformation\n        gl.uniformMatrix3fv(gl.getUniformLocation(this.program, 'mat'), false, new Float32Array(transfoMat))\n\n        // Enable the depth test\n        //gl.enable(gl.DEPTH_TEST);\n        // Clear the color buffer bit\n        gl.clear(gl.COLOR_BUFFER_BIT)\n        // Set the view port\n        //gl.viewport(0, 0, cg.w, cg.h);\n\n        gl.drawArrays(gl.POINTS, 0, this.verticesBuffer.length / 2)\n    }\n}\n","//@ts-check\n'use strict'\n\nimport { initShaderProgram, createShader } from './webGLUtils.js'\nimport { color } from 'd3-color'\n\n/**\n * Everything to easily draw colored squares with webGL.\n * All the same size, but different fill color.\n * The color interpolation is computed in the fragment shader program, by the GPU, thus it is less flexible but faster.\n */\nexport class WebGLSquareColoringAdvanced {\n    //see:\n    //https://webglfundamentals.org/webgl/lessons/fr/webgl-shaders-and-glsl.html#les-uniforms-dans-les-shaders-de-vertex\n    //https://thebookofshaders.com/glossary/?search=mix\n    //https://thebookofshaders.com/06/\n    //https://thebookofshaders.com/glossary/\n\n    /**\n     *\n     * @param {*} gl\n     * @param {Array.<String>} colors\n     * @param {{fun:string,alpha:number}} stretching\n     * @param {number} sizePix\n     * @param {number|undefined} globalOpacity\n     */\n    constructor(gl, colors, stretching, sizePix = 10, globalOpacity = undefined) {\n        /** @type {WebGLRenderingContext} */\n        this.gl = gl\n        //gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, true);\n        //gl.blendFunc(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA);\n        //gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA);\n\n        /** @type {WebGLShader} */\n        const vShader = createShader(\n            gl,\n            gl.VERTEX_SHADER,\n            `\n        attribute vec2 pos;\n        uniform float sizePix;\n        uniform mat3 mat;\n\n        attribute float t;\n        varying float vt;\n\n        void main() {\n          gl_Position = vec4(mat * vec3(pos, 1.0), 1.0);\n          gl_PointSize = sizePix;\n          vt = t;\n        }\n      `\n        )\n\n        //prepare fragment shader code\n        //declare the uniform and other variables\n        let fshString =\n            '' +\n            'precision mediump float;\\n' +\n            'varying float vt;\\n' +\n            'uniform float alpha;\\n' +\n            (() => {\n                const out = []\n                for (let i = 0; i < colors.length; i++) out.push('uniform vec4 c' + i + ';\\n')\n                return out.join('')\n            })() +\n            //start the main function, apply the stretching of t\n            'void main(void) {\\n'\n\n        if (stretching) {\n            if (stretching.fun == 'pow')\n                //sPow = (t, alpha = 3) => Math.pow(t, alpha);\n                fshString += '   float t = pow(vt, alpha);\\n'\n            else if (stretching.fun == 'powInv')\n                //sPowRev = (t, alpha = 3) => 1 - Math.pow(1 - t, 1 / alpha);\n                fshString += '   float t = 1.0-pow(1.0-vt, 1.0/alpha);\\n'\n            else if (stretching.fun == 'exp')\n                //sExp = (t, alpha = 3) => alpha == 0 ? t : (Math.exp(t * alpha) - 1) / (Math.exp(alpha) - 1);\n                fshString +=\n                    stretching.alpha == 0\n                        ? `float t = vt;`\n                        : '   float t = (exp(vt * alpha) - 1.0) / (exp(alpha) - 1.0);\\n'\n            else if (stretching.fun == 'log')\n                //sExpRev = (t, alpha = 3) => alpha == 0 ? t : 1 - (1 / alpha) * Math.log(Math.exp(alpha) * (1 - t) + t);\n                fshString +=\n                    stretching.alpha == 0\n                        ? `float t = vt;`\n                        : '   float t = 1.0 - (1.0 / alpha) * log(exp(alpha) * (1.0 - vt) + vt);\\n'\n            else if (stretching.fun == 'circle') {\n                if (stretching.alpha == 0)\n                    //if (alpha == 0) return t;\n                    fshString += '   float t = vt;\\n'\n                else if (stretching.alpha == 1)\n                    // if (alpha == 1) return Math.sqrt(2 * t - t * t);\n                    fshString += '   float t = sqrt(vt * (2.0 - vt));\\n'\n                else {\n                    //const a = alpha / (1 - alpha);\n                    //return Math.sqrt(1 / (a * a) + t * (2 / a + 2 - t)) - 1 / a;\n                    fshString +=\n                        '   float a = alpha / (1.0 - alpha);\\n' +\n                        '   float t = sqrt(1.0 / (a * a) + vt * ( 2.0/a + 2.0 - vt )) - 1.0 / a;\\n'\n                }\n            } else if (stretching.fun == 'circleInv') {\n                // 1 - sCircleLow(1 - t, alpha)\n                if (stretching.alpha == 0)\n                    //if (alpha == 0) return t;\n                    fshString += '   float t = vt;\\n'\n                else if (stretching.alpha == 1)\n                    // if (alpha == 1) return Math.sqrt(2 * t - t * t);\n                    fshString += '   float t = 1.0 - sqrt((1.0 - vt) * (1.0 + vt));\\n'\n                else {\n                    //const a = alpha / (1 - alpha);\n                    //return Math.sqrt(1 / (a * a) + (2 * t) / a + 2 * t - t * t) - 1 / a;\n                    fshString +=\n                        '   float a = alpha / (1.0 - alpha);\\n' +\n                        '   float t = 1.0 - sqrt(1.0 / (a * a) + (1.0-vt) * ( 2.0/a + 1.0 + vt )) + 1.0 / a;\\n'\n                }\n            } else {\n                console.error('Unexpected stretching function code: ' + stretching.fun)\n                fshString += '   float t = vt;\\n'\n            }\n        } else {\n            fshString += '   float t = vt;\\n'\n        }\n\n        //choose initial and final colors, and adjust t value\n        if (colors.length == 1) fshString += '   vec4 cI=c0;\\n   vec4 cF=c0;\\n'\n        else if (colors.length == 2) fshString += '   vec4 cI=c0;\\n   vec4 cF=c1;\\n'\n        else {\n            const nb = colors.length - 1\n            const nbs = nb + '.0'\n            fshString += '   vec4 cI;\\n'\n            fshString += '   vec4 cF;\\n'\n            fshString += '   if(t<1.0/' + nbs + ') { cI=c0; cF=c1; t=t*' + nbs + '; }\\n'\n            for (let i = 2; i < nb; i++)\n                fshString +=\n                    '   else if(t<' +\n                    i +\n                    '.0/' +\n                    nbs +\n                    ') { cI=c' +\n                    (i - 1) +\n                    '; cF=c' +\n                    i +\n                    '; t=' +\n                    nbs +\n                    '*t-' +\n                    (i - 1) +\n                    '.0; }\\n'\n            fshString +=\n                '   else { cI=c' + (nb - 1) + '; cF=c' + nb + '; t=' + nbs + '*t-' + (nb - 1) + '.0; }\\n'\n        }\n\n        //one single color\n        if (colors.length == 1) fshString += '   gl_FragColor = vec4(c0[0], c0[1], c0[2], c0[3]);}\\n'\n        //set interpolated color, between initial and final one\n        else fshString += '   gl_FragColor = mix(cI, cF, t);}\\n'\n\n        //console.log(fshString)\n\n        /** @type {WebGLShader} */\n        const fShader = createShader(gl, gl.FRAGMENT_SHADER, fshString)\n\n        /** @type {WebGLProgram} */\n        this.program = initShaderProgram(gl, vShader, fShader)\n        gl.useProgram(this.program)\n\n        //set uniforms\n\n        //sizePix\n        //TODO: bug here. Seems to be limited to some threshold value (around 250).\n        gl.uniform1f(gl.getUniformLocation(this.program, 'sizePix'), 1.0 * sizePix)\n\n        //stretching alpha factor\n        gl.uniform1f(gl.getUniformLocation(this.program, 'alpha'), stretching ? 1.0 * stretching.alpha : 0.0)\n\n        //colors\n        for (let i = 0; i < colors.length; i++) {\n            const c = color(colors[i])\n\n            let opacity = c.opacity\n            if (c.opacity == 1 && globalOpacity != undefined) opacity = globalOpacity\n\n            gl.uniform4fv(gl.getUniformLocation(this.program, 'c' + i), [\n                +c.r / 255.0,\n                +c.g / 255.0,\n                +c.b / 255.0,\n                +opacity,\n            ])\n        }\n    }\n\n    /**  */\n    draw(verticesBuffer, tBuffer, transfoMat) {\n        const gl = this.gl\n        const program = this.program\n\n        //vertice data\n        gl.bindBuffer(gl.ARRAY_BUFFER, gl.createBuffer())\n        gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(verticesBuffer), gl.STATIC_DRAW)\n        const position = gl.getAttribLocation(program, 'pos')\n        gl.vertexAttribPointer(\n            position,\n            2, //numComponents\n            gl.FLOAT, //type\n            false, //normalise\n            0, //stride\n            0 //offset\n        )\n        gl.enableVertexAttribArray(position)\n\n        //t data\n        gl.bindBuffer(gl.ARRAY_BUFFER, gl.createBuffer())\n        gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(tBuffer), gl.STATIC_DRAW)\n        const t = gl.getAttribLocation(program, 't')\n        gl.vertexAttribPointer(t, 1, gl.FLOAT, false, 0, 0)\n        gl.enableVertexAttribArray(t)\n\n        //transformation\n        gl.uniformMatrix3fv(gl.getUniformLocation(program, 'mat'), false, new Float32Array(transfoMat))\n\n        // Enable the depth test\n        //gl.enable(gl.DEPTH_TEST);\n        // Clear the color buffer bit\n        gl.clear(gl.COLOR_BUFFER_BIT)\n        // Set the view port\n        //gl.viewport(0, 0, cg.w, cg.h);\n\n        gl.drawArrays(gl.POINTS, 0, verticesBuffer.length / 2)\n    }\n}\n","//@ts-check\n'use strict'\n\nimport { initShaderProgram, createShader } from './webGLUtils.js'\nimport { color } from 'd3-color'\n\n/**\n * Everything to easily draw colored squares with webGL.\n * All the same size, but different fill color.\n * Color based on categories.\n */\nexport class WebGLSquareColoringCatAdvanced {\n    /**\n     * @param {Array.<string>} colors\n     */\n    constructor(colors) {\n        /**\n         * @type {Array.<string>} */\n        this.colors = colors\n\n        /** Vector shader program\n         * @type {string} */\n        this.vshString = `\n        attribute vec2 pos;\n        uniform float sizePix;\n        uniform mat3 mat;\n\n        attribute float i;\n        varying float vi;\n\n        void main() {\n          gl_Position = vec4(mat * vec3(pos, 1.0), 1.0);\n          gl_PointSize = sizePix;\n          vi = i;\n        }\n        `\n\n        //prepare fragment shader code\n        //declare the uniform and other variables\n        const out = []\n        out.push('precision mediump float;\\nvarying float vi;\\n')\n        //add color uniforms\n        out.push('uniform vec4')\n        for (let i = 0; i < colors.length; i++) {\n            if (i > 0) out.push(',')\n            out.push(' c' + i)\n        }\n        out.push(';\\n')\n        //start the main function\n        out.push('void main(void) {\\n')\n        //choose color i\n        for (let i = 0; i < colors.length; i++) {\n            if (i > 0) out.push('else ')\n            out.push('if(vi==')\n            out.push(i)\n            out.push('.0) gl_FragColor = vec4(c')\n            out.push(i)\n            out.push('[0], c')\n            out.push(i)\n            out.push('[1], c')\n            out.push(i)\n            out.push('[2], c')\n            out.push(i)\n            out.push('[3]);\\n')\n        }\n        out.push('else gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);\\n}')\n        /** Fragment shader program\n         * @type {string} */\n        this.fshString = out.join('')\n    }\n\n    /**  */\n    draw(gl, verticesBuffer, iBuffer, transfoMat, sizePix = 10) {\n        /** @type {WebGLShader} */\n        const vShader = createShader(gl, gl.VERTEX_SHADER, this.vshString)\n\n        /** @type {WebGLShader} */\n        const fShader = createShader(gl, gl.FRAGMENT_SHADER, this.fshString)\n\n        /** @type {WebGLProgram} */\n        const program = initShaderProgram(gl, vShader, fShader)\n        gl.useProgram(program)\n\n        //set uniforms\n\n        //sizePix\n        gl.uniform1f(gl.getUniformLocation(program, 'sizePix'), 1.0 * sizePix)\n\n        //colors\n        for (let i = 0; i < this.colors.length; i++) {\n            const c = color(this.colors[i])\n            gl.uniform4fv(gl.getUniformLocation(program, 'c' + i), [\n                +c.r / 255.0,\n                +c.g / 255.0,\n                +c.b / 255.0,\n                +c.opacity,\n            ])\n        }\n\n        //vertice data\n        gl.bindBuffer(gl.ARRAY_BUFFER, gl.createBuffer())\n        gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(verticesBuffer), gl.STATIC_DRAW)\n        const position = gl.getAttribLocation(program, 'pos')\n        gl.vertexAttribPointer(\n            position,\n            2, //numComponents\n            gl.FLOAT, //type\n            false, //normalise\n            0, //stride\n            0 //offset\n        )\n        gl.enableVertexAttribArray(position)\n\n        //i data\n        gl.bindBuffer(gl.ARRAY_BUFFER, gl.createBuffer())\n        gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(iBuffer), gl.STATIC_DRAW)\n        const i = gl.getAttribLocation(program, 'i')\n        gl.vertexAttribPointer(i, 1, gl.FLOAT, false, 0, 0)\n        gl.enableVertexAttribArray(i)\n\n        //transformation\n        gl.uniformMatrix3fv(gl.getUniformLocation(program, 'mat'), false, new Float32Array(transfoMat))\n\n        // Enable the depth test\n        //gl.enable(gl.DEPTH_TEST);\n        // Clear the color buffer bit\n        gl.clear(gl.COLOR_BUFFER_BIT)\n        // Set the view port\n        //gl.viewport(0, 0, cg.w, cg.h);\n\n        gl.drawArrays(gl.POINTS, 0, verticesBuffer.length / 2)\n    }\n}\n","//@ts-check\r\n'use strict'\r\n\r\nimport { extent, max } from 'd3-array'\r\nimport { scaleQuantile } from 'd3-scale'\r\n\r\n/**\r\n * A scale is simply a function that map a domain to a range.\r\n * @typedef {function(number):number} Scale */\r\n\r\n/**\r\n * A scale whose range is a color (string).\r\n * @typedef {function(number):string} ColorScale */\r\n\r\n/**\r\n * @module utils\r\n */\r\n\r\n/**\r\n * Generic function for view scale - continuous or discrete\r\n *\r\n * @param {{ valueFunction:function(import(\"../core/Dataset\").Cell):number, minValue?:number, minSizePix?:number, maxSizeFactor?:number, range?:[number, number], domain?:[number, number], stretching?:function(number):number, classNumber?:number}} opts\r\n * @returns {function(Array.<import(\"../core/Dataset\").Cell>):Scale}\r\n */\r\nexport const viewScale = (opts) => {\r\n    const valueFunction = opts.valueFunction\r\n    const minValue = opts.minValue || 0\r\n    const minSizePix = opts.minSizePix || 0\r\n    const maxSizeFactor = opts.maxSizeFactor || 1\r\n    const stretching = opts.stretching\r\n    const range_ = opts.range\r\n    const domain_ = opts.domain\r\n    const classNumber = opts.classNumber\r\n    return (cells, resolution, z) => {\r\n        const domain = domain_ || [minValue, max(cells, valueFunction)]\r\n        const range = range_ || [minSizePix * z, resolution * maxSizeFactor]\r\n        const domainSize = domain[1] - domain[0],\r\n            domainMin = domain[0]\r\n        const rangeSize = range[1] - range[0],\r\n            rangeMin = range[0]\r\n        return (t) => {\r\n            //scale to [0,1]\r\n            t = (t - domainMin) / domainSize\r\n            //stretch\r\n            if (stretching) t = stretching(t)\r\n            //classify\r\n            if (classNumber) t = t == 1 ? 1 : Math.floor(t * classNumber) / (classNumber - 1)\r\n            //scale to range\r\n            return rangeMin + t * rangeSize\r\n        }\r\n    }\r\n}\r\n\r\n/**\r\n * Generic function for view scale - quantile\r\n *\r\n * @param {{ valueFunction:function(import(\"../core/Dataset\").Cell):number, classNumber?:number, minSizePix?:number, maxSizeFactor?:number }} opts\r\n * @returns {function(Array.<import(\"../core/Dataset\").Cell>):Scale}\r\n */\r\nexport const viewScaleQuantile = (opts) => {\r\n    const valueFunction = opts.valueFunction\r\n    const classNumber = opts.classNumber || 12\r\n    const minSizePix = opts.minSizePix || 1\r\n    const maxSizeFactor = opts.maxSizeFactor || 1\r\n    const scale = scaleQuantile()\r\n    return (cells, resolution, z) => {\r\n        scale.domain(cells.map(valueFunction))\r\n        const minSizeGeo = minSizePix * z,\r\n            maxSizeGeo = resolution * maxSizeFactor\r\n        scale.range(\r\n            Array.from(\r\n                { length: classNumber },\r\n                (_, i) => minSizeGeo + (i * (maxSizeGeo - minSizeGeo)) / (classNumber - 1)\r\n            )\r\n        )\r\n        scale.breaks = scale.quantiles()\r\n        scale.values = scale.range()\r\n        return scale\r\n    }\r\n}\r\n\r\n/**\r\n * Generic function for color view scale - continuous or discrete\r\n *\r\n * @param {{ valueFunction:function(import(\"../core/Dataset\").Cell):number, colorScale?:function(number):string, colors?:Array.<string>, stretching?:function(number):number }} opts\r\n * @returns {function(Array.<import(\"../core/Dataset\").Cell>):ColorScale}\r\n */\r\nexport const viewScaleColor = (opts) => {\r\n    const valueFunction = opts.valueFunction\r\n    const stretching = opts.stretching\r\n    let colorScale = opts.colorScale || (() => 'purple')\r\n\r\n    //discrete colors case: build continuous color scale from discrete ones.\r\n    const nbClass = opts.colors?.length\r\n    if (opts.colors && nbClass)\r\n        colorScale = (t) => opts.colors[t == 1 ? nbClass - 1 : Math.floor(t * nbClass)]\r\n\r\n    return (cells) => {\r\n        if (cells.length == 0 || !cells) return\r\n        /** @type {[undefined, undefined] | [number, number]} */\r\n        const domain = extent(cells, valueFunction)\r\n        if (domain[0] == undefined) return\r\n        const domainSize = domain[1] - domain[0]\r\n        const scale = (t) => {\r\n            //scale to [0,1]\r\n            t = (t - domain[0]) / domainSize\r\n            //stretch\r\n            if (stretching) t = stretching(t)\r\n            return colorScale(t)\r\n        }\r\n        //function that return the domain value from the [0,1] range.\r\n        scale.invert = (t) => {\r\n            if (stretching) t = stretching.invert(t)\r\n            return domain[0] + t * domainSize\r\n        }\r\n        //discrete colors: return the breaks\r\n        if (opts.colors && nbClass) {\r\n            scale.breaks = []\r\n            for (let i = 1; i < nbClass; i++) scale.breaks.push(scale.invert(i / nbClass))\r\n        }\r\n\r\n        return scale\r\n    }\r\n}\r\n\r\n/**\r\n * Generic function for color view scale - quantile\r\n *\r\n * @param {{ valueFunction:function(import(\"../core/Dataset\").Cell):number, classNumber?:number, colors?:Array.<string>, colorScale?:function(number):string }} opts\r\n * @returns {function(Array.<import(\"../core/Dataset\").Cell>):ColorScale}\r\n */\r\nexport const viewScaleColorQuantile = (opts) => {\r\n    const valueFunction = opts.valueFunction\r\n    const classNumber = opts.classNumber || 12\r\n\r\n    let colors = opts.colors\r\n    if (opts.colorScale) colors = discreteColors(opts.colorScale, classNumber)\r\n    colors =\r\n        colors ||\r\n        Array.from(\r\n            { length: classNumber },\r\n            (_, i) => 'rgb(' + Math.floor((255 * i) / (classNumber - 1)) + ',150,150)'\r\n        )\r\n\r\n    const scale = scaleQuantile().range(colors)\r\n    return (cells) => {\r\n        scale.domain(cells.map(valueFunction))\r\n        scale.breaks = scale.quantiles()\r\n        scale.colors = colors\r\n        return scale\r\n    }\r\n}\r\n\r\n/**\r\n * Combine view scale functions\r\n *\r\n * @param {*} obj\r\n * @returns {function}\r\n */\r\nexport const viewScaleCombination = (obj) => {\r\n    //obj: prop and a function to call\r\n    return (cells, resolution, z) => {\r\n        const out = {}\r\n        for (const p in obj) {\r\n            out[p] = obj[p](cells, resolution, z)\r\n        }\r\n        return out\r\n    }\r\n}\r\n\r\n/**\r\n * Return a classifier function from break values.\r\n * The classifier function returns the class id (from 0 to breaks.length) from a value to classifiy.\r\n * @param {Array.<number>} breaks the breaks\r\n */\r\nexport function classifier(breaks) {\r\n    const bl = breaks.length\r\n    const classifier = (value) => {\r\n        let i = 0\r\n        while (i < bl) {\r\n            const break_ = breaks[i]\r\n            if (value <= break_) return i\r\n            i++\r\n        }\r\n        return i\r\n    }\r\n    classifier.breaks = breaks\r\n    return classifier\r\n}\r\n\r\n/**\r\n * Return a color classifier function from break values.\r\n * The classifier function returns the color from a value to classifiy.\r\n * There should be one color more than break values.\r\n * @param {Array.<number>} breaks the breaks\r\n * @param {Array.<string>} colors the colors\r\n */\r\nexport function colorClassifier(breaks, colors) {\r\n    const classifier_ = classifier(breaks)\r\n    const colorClassifier = (value) => colors[classifier_(value)]\r\n    colorClassifier.breaks = breaks\r\n    colorClassifier.colors = colors\r\n    return colorClassifier\r\n}\r\n\r\n/**\r\n * Make array of colors from a colorScale.\r\n * It is a kind of sampling, or un-interpolation\r\n *\r\n * @param {function(number):string} colorScale\r\n * @param {number} nb\r\n */\r\nexport function discreteColors(colorScale, nb) {\r\n    if (nb == 1) return [colorScale(0.5)]\r\n    const out = []\r\n    for (let i = 0; i < nb; i++) out.push(colorScale(i / (nb - 1)))\r\n    return out\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n/**\r\n * @module utils\r\n */\r\n\r\n//TODO invert for circular\r\n//TODO use Math.sqrt\r\n//TODO validate\r\n\r\n/**\r\n * Some function [0,1]->[0,1] to stretch range of values.\r\n * @see https://github.com/eurostat/gridviz/blob/master/docs/reference.md#stretching\r\n * @see https://observablehq.com/@jgaffuri/stretching\r\n */\r\n\r\n//identity function\r\nconst identity = (t) => t\r\nidentity.invert = identity\r\n\r\n/**\r\n * @param {number} base\r\n * @returns {function(number):number}\r\n */\r\nexport const exponentialScale = (base = 3) => {\r\n    if (base == 0) return identity\r\n    const a = Math.exp(base) - 1\r\n    const f = (t) => (Math.exp(t * base) - 1) / a\r\n    f.invert = (t) => Math.log(a * t + 1) / base\r\n    return f\r\n}\r\n\r\n/**\r\n * @param {number} base\r\n * @returns {function(number):number}\r\n */\r\nexport const logarithmicScale = (base = 3) => {\r\n    if (base == 0) return identity\r\n    const a = Math.exp(base),\r\n        b = 1 - a\r\n    const f = (t) => 1 - Math.log(a + t * b) / base\r\n    f.invert = (t) => (Math.exp((1 - t) * base) - a) / b\r\n    return f\r\n}\r\n\r\n/**\r\n * @param {number} exponent\r\n * @returns {function(number):number}\r\n */\r\nexport const powerScale = (exponent = 3) => {\r\n    if (exponent == 1) return identity\r\n    //TODO if (exponent == 0.5) return Math.sqrt\r\n    const f = (t) => Math.pow(t, exponent)\r\n    const a = 1 / exponent\r\n    f.invert = (t) => Math.pow(t, a)\r\n    return f\r\n}\r\n\r\n/**\r\n * @param {number} exponent\r\n * @returns {function(number):number}\r\n */\r\nexport const powerInverseScale = (exponent = 3) => {\r\n    if (exponent == 1) return identity\r\n    //TODO if (exponent == 2) return t => 1 - Math.sqrt(1 - t)\r\n    const a = 1 / exponent\r\n    const f = (t) => 1 - Math.pow(1 - t, a)\r\n    f.invert = (t) => 1 - Math.pow(1 - t, exponent)\r\n    return f\r\n}\r\n\r\n/**\r\n * @param {number} circularity\r\n * @returns {function(number):number}\r\n */\r\nexport const circularScale = (circularity = 0.8) => {\r\n    if (circularity == 0) return identity\r\n    if (circularity == 1) return (t) => Math.sqrt(t * (2 - t))\r\n    else {\r\n        const a = circularity / (1 - circularity)\r\n        return (t) => Math.sqrt(1 / (a * a) + t * (2 / a + 2 - t)) - 1 / a\r\n    }\r\n}\r\n\r\n/**\r\n * @param {number} circularity\r\n * @returns {function(number):number}\r\n */\r\nexport const circularInverseScale = (circularity = 0.8) => {\r\n    if (circularity == 0) return identity\r\n    const f = circularScale(circularity)\r\n    return (t) => 1 - f(1 - t)\r\n}\r\n\r\n//test\r\n/*\r\nconst test = (f, fun, a, err = 1e-12) => {\r\n    for (let t = 0; t <= 1; t += 1 / 50) {\r\n        const er = t - f.invert(f(t))\r\n        if (Math.abs(er) < err) continue\r\n        console.log(fun, a, er)\r\n    }\r\n}\r\n\r\nfor (let fun of [powerScale, powerInverseScale])\r\n    for (let exp = -30; exp <= 50; exp += 1) {\r\n        if (exp == 0) continue\r\n        const f = fun(exp)\r\n        test(f, fun, exp)\r\n    }\r\n\r\n\r\nfor (let fun of [exponentialScale, logarithmicScale])\r\n    for (let base = -20; base <= 20; base += 1) {\r\n        //if (exp == 0) continue\r\n        const f = fun(base)\r\n        test(f, fun, base, 1e-10)\r\n    }\r\n*/\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { interpolateLab } from 'd3-interpolate'\r\n\r\nexport const ternaryClassifier = (properties, totalFunction, opts = {}) => {\r\n    //the three properties\r\n    const p0 = properties[0],\r\n        p1 = properties[1],\r\n        p2 = properties[2]\r\n\r\n    //the classifier center point. sum must be equal to 1\r\n    const [c0, c1, c2] = opts.center || [1 / 3, 1 / 3, 1 / 3]\r\n\r\n    //parameter to decide wether to use mixed classes m0, m1, m2.\r\n    const withMixedClasses = opts.withMixedClasses != undefined ? opts.withMixedClasses : true\r\n\r\n    //paramerter decide wether to use a central class, and the size of this central class.\r\n    //set to 0 or undefined for not showing any central class. Set to 1 for a central class that contains the mix classes\r\n    const cc = opts.centerCoefficient ? 1 - opts.centerCoefficient : undefined\r\n\r\n    //the output classifier method\r\n    const fun = (c) => {\r\n        //get total\r\n        const tot = totalFunction(c)\r\n        if (!tot) return undefined\r\n        //compute shares\r\n        const [s0, s1, s2] = [+c[p0] / tot, +c[p1] / tot, +c[p2] / tot]\r\n\r\n        //class 0\r\n        if (s0 >= c0 && s1 <= c1 && s2 <= c2) {\r\n            //central class near class 0\r\n            if (cc != undefined && (s2 - c2) * (c1 - cc * c1) >= (s1 - cc * c1) * (cc * c2 - c2))\r\n                return 'center'\r\n            return '0'\r\n        }\r\n        //class 1\r\n        if (s0 <= c0 && s1 >= c1 && s2 <= c2) {\r\n            //central class near class 1\r\n            if (cc != undefined && (s2 - c2) * (c0 - cc * c0) >= (s0 - cc * c0) * (cc * c2 - c2))\r\n                return 'center'\r\n            return '1'\r\n        }\r\n        //class 2\r\n        if (s0 <= c0 && s1 <= c1 && s2 >= c2) {\r\n            //central class near class 2\r\n            if (cc != undefined && (s1 - c1) * (c0 - cc * c0) >= (s0 - cc * c0) * (cc * c1 - c1))\r\n                return 'center'\r\n            return '2'\r\n        }\r\n        //middle class 0 - intersection class 1 and 2\r\n        if (s0 <= c0 && s1 >= c1 && s2 >= c2) {\r\n            //central class\r\n            if (cc != undefined && s0 > cc * c0) return 'center'\r\n            if (withMixedClasses) return 'm12'\r\n            return s1 > s2 ? '1' : '2'\r\n        }\r\n        //middle class 1 - intersection class 0 and 1\r\n        if (s0 >= c0 && s1 <= c1 && s2 >= c2) {\r\n            //central class\r\n            if (cc != undefined && s1 > cc * c1) return 'center'\r\n            if (withMixedClasses) return 'm02'\r\n            return s0 > s2 ? '0' : '2'\r\n        }\r\n        //middle class 2 - intersection class 0 and 1\r\n        if (s0 >= c0 && s1 >= c1 && s2 <= c2) {\r\n            //central class\r\n            if (cc != undefined && s2 > cc * c2) return 'center'\r\n            if (withMixedClasses) return 'm01'\r\n            return s1 > s0 ? '1' : '0'\r\n        }\r\n        //should not happen\r\n        return 'unknown'\r\n    }\r\n\r\n    //attach information to output function\r\n    fun.center = [c0, c1, c2]\r\n    fun.centerCoefficient = opts.centerCoefficient\r\n\r\n    return fun\r\n}\r\n\r\nexport const ternaryColorClassifier = (properties, totalFunction, colors, opts = {}) => {\r\n    //the three colors\r\n    const [color0, color1, color2] = colors || ['red', 'green', 'blue']\r\n\r\n    //the color interpolation function\r\n    const colorInterpolation = opts.colorInterpolation || interpolateLab\r\n\r\n    //parameter to decide wether to use mixed classes.\r\n    const withMixedClasses = opts.withMixedClasses != undefined ? opts.withMixedClasses : true\r\n    //https://d3js.org/d3-interpolate/color\r\n    const mixColorFunction = (color1, color2) => colorInterpolation(color1, color2)(0.5)\r\n    //the colors corresponding to the mixed classes\r\n    const [mixColor0, mixColor1, mixColor2] =\r\n        opts.mixedColors || withMixedClasses\r\n            ? [\r\n                  mixColorFunction(color1, color2),\r\n                  mixColorFunction(color0, color2),\r\n                  mixColorFunction(color0, color1),\r\n              ]\r\n            : []\r\n\r\n    //the central color, used for the central class, if any. The central class is the class of relatively balanced values, around the center point\r\n    const centerColor =\r\n        opts.centerColor || colorInterpolation(mixColorFunction(color0, color1), color2)(0.333)\r\n\r\n    //make classifier\r\n    const classifier = ternaryClassifier(properties, totalFunction, opts)\r\n\r\n    //the output color classifier method\r\n    const fun = (c) => {\r\n        const cla = classifier(c)\r\n        if (cla == '0') return color0\r\n        if (cla == '1') return color1\r\n        if (cla == '2') return color2\r\n        if (cla == 'm12') return mixColor0\r\n        if (cla == 'm02') return mixColor1\r\n        if (cla == 'm01') return mixColor2\r\n        if (cla == 'center') return centerColor\r\n        return opts.defaultColor || 'black'\r\n    }\r\n    fun.center = classifier.center\r\n    fun.centerCoefficient = opts.centerCoefficient\r\n    fun.colors = [color0, color1, color2]\r\n    fun.mixColors = [mixColor0, mixColor1, mixColor2]\r\n    fun.centerColor = centerColor\r\n    fun.classifier = classifier\r\n\r\n    return fun\r\n}\r\n\r\n/*\r\nconst orderedIndexesDec = arr => [...arr.keys()].sort((a, b) => arr[b] - arr[a]);\r\nconst orderedIndexesInc = arr => [...arr.keys()].sort((a, b) => arr[a] - arr[b]);\r\n\r\nexport const ternaryClassifier = (properties, totalFunction, opts = {}) => {\r\n    const lowThreshold = opts.lowThreshold || [1 / 3, 1 / 3, 1 / 3]\r\n    const highThreshold = opts.highThreshold || [2 / 3, 2 / 3, 2 / 3]\r\n    const colors = opts.colors || [\"red\", \"green\", \"blue\"]\r\n    const colorInterpolation = opts.colorInterpolation || interpolateLab\r\n\r\n    //https://d3js.org/d3-interpolate/color\r\n    const middleColorFunction = (color1, color2) => colorInterpolation(color1, color2)(0.5)\r\n    const middleColors = opts.middleColors || [middleColorFunction(colors[1], colors[2]), middleColorFunction(colors[0], colors[2]), middleColorFunction(colors[0], colors[1])]\r\n    const centerColor = opts.centerColor || colorInterpolation(middleColors[2], colors[2])(0.333)\r\n\r\n    const high_ = orderedIndexesDec(highThreshold)\r\n    const low_ = orderedIndexesInc(lowThreshold)\r\n\r\n    const p0 = properties[0], p1 = properties[1], p2 = properties[2]\r\n    const fun = c => {\r\n        //get total\r\n        const tot = totalFunction(c)\r\n        if (!tot) return undefined\r\n        //compute shares\r\n        const shares = [+c[p0] / tot, +c[p1] / tot, +c[p2] / tot]\r\n        //return colors\r\n        //start first with the extreme high (triangles): from the larger value (small triangle) to the lower value (large triangle)\r\n        for (let i of high_) if (shares[i] > highThreshold[i]) return colors[i]\r\n        //then draw the extreme low (trapeziums): from the lower value (small trapeziums) to the larger values (large trapezium)\r\n        for (let i of low_) if (shares[i] < lowThreshold[i]) return middleColors[i]\r\n        //else central color\r\n        return centerColor\r\n    }\r\n    fun.lowThreshold = lowThreshold\r\n    fun.highThreshold = highThreshold\r\n    fun.colors = colors\r\n    fun.middleColors = middleColors\r\n    fun.centerColor = centerColor\r\n    fun.lowIndex = low_\r\n    fun.highIndex = high_\r\n    return fun\r\n}\r\n*/\r\n\r\n/*\r\nexport const ternaryClassifier3 = (properties, totalFunction, opts = {}) => {\r\n    const [a01, a12, a20] = opts.thresholds || [1/3, 1/3, 1/3]\r\n    const [c0, c1, c2] = opts.colors || [\"red\", \"green\", \"blue\"]\r\n    const centerColor = opts.centerColor || \"gray\"\r\n\r\n    const fff = a => a == 1 ? Infinity : a / (1 - a)\r\n    const c01 = fff(a01), c12 = fff(a12), c20 = fff(a20)\r\n\r\n    const p0 = properties[0], p1 = properties[1], p2 = properties[2]\r\n    const fun = c => {\r\n        //get total\r\n        const tot = totalFunction(c)\r\n        if (!tot) return undefined\r\n        //compute shares\r\n        const [s0, s1, s2] = [+c[p0] / tot, +c[p1] / tot, +c[p2] / tot]\r\n        //return colors\r\n        if (s0 * c01 > s1 && s0 > s2 * c20) return c0\r\n        else if (s1 > s0 * c01 && s1 * c12 > s2) return c1\r\n        else if (s2 * c20 > s0 && s2 > s1 * c12) return c2\r\n        else return centerColor\r\n    }\r\n    fun.colors = [c0, c1, c2]\r\n    fun.centerColor = centerColor\r\n    return fun\r\n}\r\n*/\r\n","//@ts-check\r\n'use strict'\r\n\r\n/**\r\n * Get the class id from a value and class break values\r\n *\r\n * @param {number} v the value\r\n * @param {Array.<number>} breaks the breaks\r\n * @returns The class id, from 0 to breaks.length\r\n * @deprecated use getClassifier instead.\r\n */\r\nexport function getClass(v, breaks) {\r\n    if (!breaks) return\r\n    if (breaks.length == 0) return 0\r\n    if (v <= breaks[0]) return 0\r\n    for (let i = 1; i < breaks.length; i++) if (breaks[i - 1] < v && v <= breaks[i]) return i\r\n    return breaks.length\r\n}\r\n\r\n//take 'nice' value (power of ten, or multiple)\r\nexport function nice(v, multiples = [8, 6, 5, 4, 2.5, 2]) {\r\n    //compute bigger power of ten below\r\n    const v_ = Math.pow(10, Math.floor(Math.log10(v)))\r\n    for (let multiple of multiples) if (v_ * multiple <= v) return v_ * multiple\r\n    return v_\r\n}\r\n\r\n/*\r\n//no longer used\r\nexport function loadImage(src) {\r\n    return new Promise((resolve, reject) => {\r\n        const img = new Image();\r\n        img.onload = function () { resolve(img); };\r\n        img.onerror = function () { reject(new Error('Error loading image')); };\r\n        img.src = src;\r\n    });\r\n}\r\n*/\r\n\r\n/*\r\nexport let monitor = false\r\n\r\nlet previousDate\r\nexport function monitorDuration(message) {\r\n    const nowDate = Date.now()\r\n\r\n    //first call\r\n    if (!previousDate) {\r\n        previousDate = nowDate\r\n        console.log(previousDate, message)\r\n        return\r\n    }\r\n\r\n    const d = nowDate - previousDate\r\n    previousDate = nowDate\r\n    console.log(d, message)\r\n}\r\n*/\r\n","//@ts-check\r\n'use strict'\r\n\r\n/**\r\n * @param {string} width\r\n * @param {string} height\r\n * @param {object} opts\r\n * @returns {{canvas:HTMLCanvasElement, gl:WebGLRenderingContext}}\r\n */\r\nexport function makeWebGLCanvas(width, height, opts = {}) {\r\n    const canvas = document.createElement('canvas')\r\n    canvas.setAttribute('width', width)\r\n    canvas.setAttribute('height', height)\r\n    /** @type {WebGLRenderingContext} */\r\n    const gl = canvas.getContext('webgl', opts)\r\n    if (!gl) {\r\n        throw new Error('Unable to initialize WebGL. Your browser or machine may not support it.')\r\n    }\r\n    return { canvas: canvas, gl: gl }\r\n}\r\n\r\n/**\r\n * Initialize a shader program, so WebGL knows how to draw our data\r\n *\r\n * @param {WebGLRenderingContext} gl\r\n * @param  {...WebGLShader} shaders\r\n * @returns {WebGLProgram}\r\n */\r\nexport function initShaderProgram(gl, ...shaders) {\r\n    /** @type {WebGLProgram|null} */\r\n    const program = gl.createProgram()\r\n    if (program == null) throw new Error('Cannot create webGL program')\r\n    for (const shader of shaders) gl.attachShader(program, shader)\r\n    gl.linkProgram(program)\r\n    if (gl.getProgramParameter(program, gl.LINK_STATUS)) return program\r\n    throw new Error(gl.getProgramInfoLog(program) || 'Cannot create webGL program (2)')\r\n}\r\n\r\n/**\r\n * Creates a shader of the given type, uploads the source and compiles it.\r\n *\r\n * @param {WebGLRenderingContext} gl\r\n * @param {number} type\r\n * @param  {...string} sources\r\n * @returns {WebGLShader}\r\n */\r\nexport function createShader(gl, type, ...sources) {\r\n    /** @type {WebGLShader|null} */\r\n    const shader = gl.createShader(type)\r\n    if (shader == null) throw new Error('Cannot create webGL shader')\r\n    gl.shaderSource(shader, sources.join('\\n'))\r\n    gl.compileShader(shader)\r\n    if (gl.getShaderParameter(shader, gl.COMPILE_STATUS)) return shader\r\n    throw new Error(gl.getShaderInfoLog(shader) || 'Cannot create webGL shader (2)')\r\n}\r\n\r\n/**\r\n * Check if webGL is supported\r\n *\r\n * @returns {boolean}\r\n */\r\nexport function checkWebGLSupport() {\r\n    try {\r\n        const canvas = document.createElement('canvas')\r\n        return !!(\r\n            !!window.WebGLRenderingContext &&\r\n            (canvas.getContext('webgl') || canvas.getContext('experimental-webgl'))\r\n        )\r\n    } catch (err) {\r\n        return false\r\n    }\r\n}\r\n","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","//@ts-check\r\n'use strict'\r\n\r\n// the application\r\nexport { Map } from './core/Map.js'\r\nexport { GeoCanvas } from './core/GeoCanvas.js'\r\nexport { Style } from './core/Style.js'\r\nexport { Layer } from './core/Layer.js'\r\nexport { Dataset } from './core/Dataset.js'\r\nexport { MultiResolutionDataset } from './core/MultiResolutionDataset.js'\r\n\r\n// export dataset types\r\nexport { TiledGrid } from './dataset/TiledGrid.js'\r\nexport { CSVGrid } from './dataset/CSVGrid.js'\r\nexport { JSGrid } from './dataset/JSGrid.js'\r\n\r\n// export styles\r\nexport { ShapeColorSizeStyle } from './style/ShapeColorSizeStyle.js'\r\nexport { StrokeStyle } from './style/StrokeStyle.js'\r\nexport { JoyPlotStyle } from './style/JoyPlotStyle.js'\r\nexport { CompositionStyle } from './style/CompositionStyle.js'\r\nexport { SegmentStyle } from './style/SegmentStyle.js'\r\nexport { TextStyle } from './style/TextStyle.js'\r\nexport { PillarStyle } from './style/PillarStyle.js'\r\nexport { SideStyle } from './style/SideStyle.js'\r\n//export { ContourStyle } from './style/ContourStyle.js'\r\nexport { SideCategoryStyle } from './style/SideCategoryStyle.js'\r\nexport { DotDensityStyle } from './style/DotDensityStyle.js'\r\nexport { TanakaStyle } from './style/TanakaStyle.js'\r\nexport { LegoStyle } from './style/LegoStyle.js'\r\nexport { SquareColorWebGLStyle } from './style/SquareColorWebGLStyle.js'\r\nexport { SquareColorCategoryWebGLStyle } from './style/SquareColorCategoryWebGLStyle.js'\r\nexport { MosaicStyle } from './style/MosaicStyle.js'\r\nexport { NinjaStarStyle } from './style/NinjaStarStyle.js'\r\nexport { TimeSeriesStyle } from './style/TimeSeriesStyle.js'\r\nexport { IsoFenceStyle } from './style/IsoFenceStyle.js'\r\nexport { ImageStyle } from './style/ImageStyle.js'\r\n\r\n// export additional layers\r\nexport { GridLayer } from './layer/GridLayer.js'\r\nexport { BackgroundLayer } from './layer/BackgroundLayer.js'\r\nexport { BackgroundLayerWMS } from './layer/BackgroundLayerWMS.js'\r\nexport { BackgroundLayerImage } from './layer/BackgroundLayerImage.js'\r\nexport { LabelLayer } from './layer/LabelLayer.js'\r\nexport { GeoJSONLayer } from './layer/GeoJSONLayer.js'\r\n\r\n// export legends\r\nexport { ColorLegend } from './legend/ColorLegend.js'\r\nexport { ColorDiscreteLegend } from './legend/ColorDiscreteLegend.js'\r\nexport { ColorCategoryLegend } from './legend/ColorCategoryLegend.js'\r\nexport {\r\n    SizeLegend,\r\n    sizeLegend,\r\n    sizeLegendViewScale,\r\n    sizeDiscreteLegend,\r\n    sizeDiscreteViewScaleLegend,\r\n} from './legend/SizeLegend.js'\r\nexport { OrientationLegend, orientationLegend } from './legend/OrientationLegend.js'\r\nexport { TernaryLegend } from './legend/TernaryLegend.js'\r\n\r\n// export { goToStraight, zoomTo } from \"./utils/zoomUtils\"\r\nexport * from './utils/stretching.js'\r\nexport * from './utils/scale.js'\r\nexport * from './utils/ternary.js'\r\nexport { nice } from './utils/utils.js'\r\n\r\nimport { GeoCanvas } from './core/GeoCanvas.js'\r\nexport const getParameterByName = GeoCanvas.getParameterByName\r\n\r\n// set default d3 locale\r\nimport { formatDefaultLocale } from 'd3-format'\r\nformatDefaultLocale({\r\n    decimal: '.',\r\n    thousands: ' ',\r\n    grouping: [3],\r\n    currency: ['', '€'],\r\n})\r\n"],"names":[],"sourceRoot":""} \ No newline at end of file +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"gridviz.js","mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD,O;;;;;;;;;;;;;;ACVe;AACf;AACA;;;;;;;;;;;;;;;;;;;;;ACFuC;AACF;AACJ;;AAEjC,wBAAwB,wDAAQ,CAAC,qDAAS;AACnC;AACA;AACA,qBAAqB,wDAAQ,CAAC,kDAAM;AAC3C,iEAAe,WAAW,EAAC;;;;;;;;;;;;;;;;;ACRY;AACE;;AAE1B;AACf;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,qDAAS;AACxB,yBAAyB,yDAAS;AAClC;AACA,IAAI;AACJ,qBAAqB,qDAAS,UAAU,sDAAU;AAClD;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,UAAU;AACV;;AAEA;AACA;AACA;;;;;;;;;;;;;;;ACvDe;AACf;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACNe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC5BuC;;AAExB,oCAAoC,qDAAS;AAC5D;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,yDAAS;AACrB,YAAY,yDAAS;AACrB;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;AC5Be;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACnBe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACrBe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACnBe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACrBe;AACf;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACnBe;AACf;AACA;;;;;;;;;;;;;;;;;;;;;;;;;ACF2B;AACU;AACV;AACU;AACM;AACC;AACD;AACN;;AAEtB;AACf,6BAA6B,mDAAO;AACpC;AACA,8BAA8B,mDAAG;AACjC,qBAAqB,mDAAG;AACxB;AACA;AACA;AACA,eAAe,mDAAG,CAAC,2DAAW;AAC9B,eAAe,mDAAG;AAClB;AACA;;AAEO,6CAA6C,kDAAM;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO,4CAA4C,kDAAM;AACzD;AACA,gDAAgD,sDAAM;AACtD,qBAAqB,wDAAQ;AAC7B,qBAAqB,wDAAQ;AAC7B;AACA;AACA;AACA;AACA,EAAE,2DAAW,2BAA2B,0DAAgB;AACxD,MAAM,wDAAQ;AACd;AACA;;;;;;;;;;;;;;;;AC9C2D;;AAE3D;AACA;AACe;AACf;AACA;AACA;;AAEA;;AAEA,oCAAoC,sDAAgB,GAAG,wDAAc;;AAErE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;ACpDuC;AACJ;;AAEpB;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP,MAAM;AACN;AACA;AACA;AACA,WAAW,uDAAO;AAClB;AACA;AACA;;AAEO,kCAAkC,qDAAS;AAClD,kBAAkB,qDAAS;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;;;;;;;;;;;;;;;;;;;;;;;;ACtC2C;;AAEpC;;AAEA;AACA;;AAEP;AACA;AACA;AACA,yBAAyB,IAAI;AAC7B,wCAAwC,IAAI,GAAG,IAAI,GAAG,IAAI;AAC1D,wCAAwC,IAAI,GAAG,IAAI,GAAG,IAAI;AAC1D,0CAA0C,IAAI,GAAG,IAAI,GAAG,IAAI,GAAG,IAAI;AACnE,0CAA0C,IAAI,GAAG,IAAI,GAAG,IAAI,GAAG,IAAI;AACnE,wCAAwC,IAAI,GAAG,IAAI,GAAG,IAAI;AAC1D,0CAA0C,IAAI,GAAG,IAAI,GAAG,IAAI,GAAG,IAAI;;AAEnE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,sDAAM;AACN;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;;AAED;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;;AAEA,sDAAM,WAAW,kDAAM;AACvB;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA,CAAC;;AAED;AACA,aAAa,YAAY,EAAE,YAAY,EAAE,YAAY;AACrD;;AAEA;AACA,aAAa,YAAY,EAAE,YAAY,EAAE,YAAY,EAAE,oDAAoD;AAC3G;;AAEA;AACA;AACA,YAAY,2BAA2B,EAAE,eAAe,IAAI,eAAe,IAAI,eAAe,EAAE,qBAAqB,EAAE,GAAG;AAC1H;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;AAEO;AACP;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,sDAAM,WAAW,kDAAM;AACvB;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,cAAc,2BAA2B,EAAE,eAAe,IAAI,qBAAqB,KAAK,qBAAqB,GAAG,qBAAqB,EAAE,GAAG;AAC1I;AACA,CAAC;;AAED;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC3YA,6BAAe,oCAAS;AACxB;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;ACT2C;AACO;AACP;;AAE3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,qBAAqB,0CAAG,OAAO,qDAAU;AACzC;AACA;AACA;AACA;AACA,qCAAqC;AACrC;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;;AAEe;AACf;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;;AAEA,sDAAM,WAAW,kDAAM,CAAC,4CAAK;AAC7B;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,0CAAG;AAClB;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;;AAED;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,iCAAiC,6CAAO;AACxC;AACA;;AAEO;AACP;AACA;;AAEO;AACP;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,gBAAgB,6CAAO;AACvB;AACA;;AAEA,sDAAM,WAAW,kDAAM,CAAC,4CAAK;AAC7B;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,CAAC;;;;;;;;;;;;;;;;AC1HM;AACA;;;;;;;;;;;;;;;ACDP,YAAY;;AAEZ;AACA,8CAA8C,KAAK,OAAO;AAC1D;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,YAAY;AACZ,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,GAAG;AACH;AACA,iBAAiB;AACjB;AACA;AACA,GAAG;AACH;AACA,mFAAmF,OAAO;AAC1F;AACA,gDAAgD,OAAO;AACvD,GAAG;AACH;AACA;AACA,oDAAoD,OAAO;AAC3D;AACA;;AAEA;AACA,sCAAsC,OAAO;AAC7C;AACA;AACA;AACA;AACA;;AAEA;AACA,mCAAmC,OAAO;AAC1C;AACA;AACA;AACA;AACA;AACA,mCAAmC,4BAA4B;AAC/D;AACA;;AAEA,iEAAe,QAAQ,EAAC;;;;;;;;;;;;;;;;;;ACnFY;AACoB;;AAExD,6BAAe,oCAAS;AACxB;AACA,kBAAkB,wDAAM,4BAA4B,mDAAO,EAAE,0DAAiB;AAC9E;AACA,qCAAqC,mDAAO,EAAE,0DAAiB;AAC/D,IAAI;AACJ;AACA;AACA;AACA;;AAEO;AACP;AACA,kBAAkB,wDAAM;AACxB;AACA,+BAA+B,mDAAO,EAAE,0DAAiB;AACzD,4BAA4B,mCAAmC;AAC/D;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;AC3BA;AACA;AACO,oBAAoB;AACpB,2BAA2B;;AAE3B;AACP;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;ACZ2B;;AAE3B,UAAU,mDAAG;;AAEN;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACVP,YAAY;AACZ,YAAY;AACZ;AACA;AACA;;AAEA;AACA,oCAAoC;AACpC;AACA,GAAG,gBAAgB;AACnB;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;;AAEA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;;AAErB;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,iCAAiC,YAAY;AAC7C;AACA;;AAEA;AACA;AACA;AACA,iCAAiC,YAAY;AAC7C;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA,OAAO;AACP,KAAK;AACL;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;ACnK2B;;AAE3B,UAAU,mDAAG;;AAEN;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACVA;AACP;AACA;;AAEO;AACP;AACA;;AAEO;AACP;AACA;;;;;;;;;;;;;;;;;;;;;ACVqD;AACxB;;AAE7B;AACA;AACA;AACA,WAAW,oDAAI;AACf;AACA,KAAK;AACL;AACA;;AAEe;AACf;AACA,eAAe,kDAAS;AACxB,SAAS,oDAAI;AACb;AACA,GAAG;AACH;;AAEO,mBAAmB,4CAAQ;AAC3B,mBAAmB,4CAAQ;;;;;;;;;;;;;;;ACrBlC;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;;;;;;;;;;;;;;;ACRA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;;;;;;;;;;;;;;;;;;ACPuC;;AAEvC;AACO;AACA;;AAEP;AACA;AACA;AACA;AACA,CAAC;;AAEc;AACf,WAAW,sDAAY;AACvB;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACjBsD;;AAEtD,6BAAe,oCAAS;AACxB,aAAa,qEAAkB;AAC/B;;;;;;;;;;;;;;;;ACJA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;AAEA;AACA;AACA;AACO;AACP,gGAAgG;AAChG;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACnBA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;;;;;;;;;ACjBA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA,KAAK;AACL;AACA;;;;;;;;;;;;;;;;;ACNsD;;AAE/C;;AAEP,6BAAe,oCAAS;AACxB,UAAU,qEAAkB;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4CAA4C,qEAAkB,gCAAgC;AAC9F;;;;;;;;;;;;;;;;ACfsD;;AAEtD,6BAAe,oCAAS;AACxB,UAAU,qEAAkB;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACVA;AACA;;AAEe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA,uDAAuD;;AAEhD;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;AC9CA;AACA,6BAAe,oCAAS;AACxB,kDAAkD,OAAO;AACzD;AACA,6BAA6B;AAC7B,sCAAsC,QAAQ;AAC9C,sCAAsC,oBAAoB;AAC1D;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACV+C;AACM;AACN;;AAE/C,iEAAe;AACf;AACA;AACA;AACA,OAAO,yDAAa;AACpB;AACA;AACA;AACA;AACA,iBAAiB,6DAAa;AAC9B,OAAO,yDAAa;AACpB,OAAO,4DAAgB;AACvB;AACA;AACA,CAAC,EAAC;;;;;;;;;;;;;;;AClBF,6BAAe,oCAAS;AACxB;AACA;;;;;;;;;;;;;;;;;;;;;;;ACFqC;AACM;AACM;AACE;AACV;AACE;AACU;AAChB;;AAErC;AACA;;AAEA,6BAAe,oCAAS;AACxB,gFAAgF,oDAAQ,GAAG,2DAAW;AACtG;AACA;AACA;AACA,iDAAiD,oDAAQ,GAAG,8DAAc;AAC1E;AACA;AACA;;AAEA;AACA,gBAAgB,+DAAe;;AAE/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,cAAc,uDAAW;;AAEzB;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,qBAAqB,uDAAW;AAChC;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,QAAQ;AACR;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA,0BAA0B,0DAAU;;AAEpC;AACA;;AAEA;AACA;AACA,mDAAmD,gEAAc;;AAEjE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA,uEAAuE;AACvE,uEAAuE;AACvE,sIAAsI;AACtI,sEAAsE;AACtE;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,mCAAmC,+DAAe;AAClD,gDAAgD,wDAAQ;AACxD;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACnJO;AACP;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AClBiC;;AAEjC,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,gDAAK;AAChB;AACA;;;;;;;;;;;;;;;;;;ACZqC;;AAErC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA,kFAAkF,wDAAQ;AAC1F;;AAEO;AACP;AACA,0CAA0C,wDAAQ;AAClD;AACA;;AAEe;AACf;AACA,4BAA4B,wDAAQ;AACpC;;;;;;;;;;;;;;;AC5BA,iEAAe,YAAY,EAAC;;;;;;;;;;;;;;;;;ACAa;AACV;;AAEhB;AACf,UAAU,qDAAK,UAAU,oDAAQ,mBAAmB,oDAAQ;AAC5D,UAAU,qDAAK;AACf,UAAU,qDAAK;AACf,gBAAgB,qDAAK;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACfA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;ACJyC;AACV;AACY;AACD;;AAE1C,iEAAe;AACf,cAAc,gDAAK;;AAEnB;AACA,2BAA2B,6CAAQ,mBAAmB,6CAAQ;AAC9D;AACA;AACA,kBAAkB,qDAAO;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA,CAAC,IAAI,EAAC;;AAEN;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,OAAO;AACvB,cAAc,6CAAQ;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO,yBAAyB,iDAAK;AAC9B,+BAA+B,uDAAW;;;;;;;;;;;;;;;;ACtDhB;;AAEjC;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;;AAEd;AACA;;AAEA;AACA;AACA;AACA,gCAAgC;AAChC;AACA,4BAA4B;AAC5B;AACA;AACA,yCAAyC;AACzC,4BAA4B;AAC5B;AACA,MAAM,OAAO;AACb;AACA,cAAc,SAAS,sDAAM,SAAS;AACtC;AACA;AACA;;AAEA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,OAAO;AACpC;AACA,SAAS;AACT;;;;;;;;;;;;;;;;AC/DA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACzBkC;AACY;;AAE9C;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,cAAc,aAAa,sDAAM,SAAS,GAAG,aAAa,sDAAM,SAAS;AACzE,MAAM;AACN;AACA;AACA;;AAEA;AACA;AACA,iCAAiC,gCAAgC;AACjE,cAAc,sDAAsD,sDAAM,OAAO;AACjF,MAAM;AACN;AACA;AACA;;AAEA;AACA;AACA,cAAc,qDAAqD,sDAAM,OAAO;AAChF,MAAM;AACN;AACA;AACA;;AAEA;AACA;AACA;AACA,cAAc,aAAa,sDAAM,SAAS,GAAG,aAAa,sDAAM,SAAS;AACzE,MAAM;AACN;AACA;AACA;;AAEA;AACA;AACA,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO,mDAAmD,+CAAQ;AAC3D,mDAAmD,+CAAQ;;;;;;;;;;;;;;;;;AC9Df;;AAEnD;;AAEA;AACO;AACP;AACA,wBAAwB,mDAAQ,GAAG,yDAAS;AAC5C;;AAEO;AACP,4BAA4B,mDAAQ;AACpC;AACA;AACA,iEAAiE,mDAAQ;AACzE;AACA,SAAS,yDAAS;AAClB;;;;;;;;;;;;;;;ACjBA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,iEAAe;;AAEf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA,CAAC,mBAAmB,EAAC;;;;;;;;;;;;;;;ACtErB,iEAAe,WAAW,EAAC;;;;;;;;;;;;;;;;ACAoB;;AAE/C,iEAAe;AACf;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,QAAQ;;AAER;AACA;AACA;;AAEA;;AAEA;AACA,CAAC,EAAE,yDAAa,CAAC,EAAC;;;;;;;;;;;;;;;;AC3BX;AACP;AACA;AACA,gCAAgC;AAChC,+CAA+C;AAC/C;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;ACzBwE;AACpC;;AAErB;AACf;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,wCAAwC,wDAAS;AACjD;AACA;;AAEA;AACA,wDAAwD,oDAAM;AAC9D;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,gBAAgB,gDAAS;AACzB;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,SAAS,+CAAS;AAClB;;;;;;;;;;;;;;;ACxDA;AACA;AACA;AACA,cAAc;AACd;AACA;AACe;AACf;AACA;;;;;;;;;;;;;;;ACRA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACJuC;AACD;;AAEtC;AACA;AACA;AACA;AACA,mBAAmB,iDAAK,8CAA8C,iDAAK;AAC3E;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB,iBAAiB,yDAAS;AAC1B;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACxBA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACVyC;;AAEzC,6BAAe,oCAAS;AACxB;AACA;AACA,SAAS,sDAAU,2BAA2B,OAAO,sDAAU,uBAAuB,QAAQ;AAC9F;;;;;;;;;;;;;;;;ACNO;;AAEP,iEAAe;AACf;AACA;AACA;AACA;AACA;AACA,CAAC,EAAC;;;;;;;;;;;;;;;;ACRyC;;AAE3C,6BAAe,oCAAS;AACxB,UAAU,2DAAW;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACnBqD;;AAErD,6BAAe,oCAAS;AACxB;AACA,YAAY,0DAAS;AACrB,YAAY,0DAAS,eAAe,qDAAI;AACxC;;;;;;;;;;;;;;;;ACNoC;;AAEpC,6BAAe,oCAAS;AACxB,mDAAmD,uDAAO;AAC1D;AACA;AACA,GAAG;AACH;;;;;;;;;;;;;;;;ACPwC;;AAExC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB,iBAAiB,yDAAS;;AAE1B;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACxDA,6BAAe,sCAAW;AAC1B;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACLA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;AC1EA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;;;;;;;;;;;;;;;;;;ACZqC;AACA;AACC;;AAEtC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,SAAS,gBAAgB;AACzB;AACA;AACA;AACA,MAAM;AACN,qBAAqB,gDAAS;AAC9B;AACA;;AAEA;AACA,SAAS,iBAAiB;AAC1B;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,cAAc,iBAAiB;AAC/B;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,cAAc,gBAAgB;AAC9B;AACA;AACA;AACA;AACA;AACA,MAAM;AACN,qBAAqB,gDAAS;AAC9B;AACA;;AAEA;AACA,cAAc,iBAAiB;AAC/B;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA;;AAEA,2CAA2C,wDAAQ;;AAEnD,uGAAuG,OAAO;AAC9G;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA,6CAA6C,iBAAiB;AAC9D;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,eAAe,gDAAS;AACxB;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA,wBAAwB;AACxB;;;;;;;;;;;;;;;AC/HA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACJuC;;AAEvC;AACA,eAAe,sDAAW;AAC1B;;AAEA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;ACjCA,6BAAe,oCAAS;;AAExB,4DAA4D,OAAO;AACnE,+DAA+D,OAAO;AACtE;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;ACTA,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;;;;ACFiC;AACI;;AAErC,6BAAe,sCAAW;AAC1B,aAAa,gDAAS,iCAAiC,kDAAM;AAC7D;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,iCAAiC,sDAAsD;AACvF,wCAAwC,gDAAgD;AACxF,sCAAsC,8CAA8C;AACpF,yCAAyC;AACzC;;;;;;;;;;;;;;;;;ACrBiC;AACI;;AAErC,6BAAe,sCAAW;AAC1B,aAAa,gDAAS,gCAAgC,kDAAM;AAC5D;;;;;;;;;;;;;;;;;ACLqC;AACD;;AAEpC,6BAAe,oCAAS;AACxB,2CAA2C,uDAAO;;AAElD,sFAAsF,OAAO;AAC7F,6FAA6F,OAAO;AACpG;AACA;AACA;AACA;AACA;;AAEA,aAAa,gDAAS;AACtB;;;;;;;;;;;;;;;ACfA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACxB2C;AACM;AACI;AACM;AAChB;AACJ;AACE;AACF;AACA;AACE;AACA;AACF;AACA;AACE;AACF;AACA;AACE;AACF;AACA;AACE;AACM;AACF;AACN;AACA;AACE;AACA;AACE;AACA;AACA;AACF;AACA;AACN;AACY;AACA;;AAExC;;AAEA;AACP;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,UAAU,kDAAgB;AAC1B,aAAa,qDAAmB;AAChC,eAAe,uDAAqB;AACpC,kBAAkB,0DAAwB;AAC1C,UAAU,kDAAgB;AAC1B,QAAQ,gDAAc;AACtB,SAAS,iDAAe;AACxB,QAAQ,gDAAc;AACtB,QAAQ,gDAAc;AACtB,SAAS,iDAAe;AACxB;AACA,SAAS,kDAAe;AACxB,QAAQ,iDAAc;AACtB,QAAQ,iDAAc;AACtB,SAAS,kDAAe;AACxB,QAAQ,iDAAc;AACtB,QAAQ,iDAAc;AACtB,SAAS,kDAAe;AACxB,QAAQ,iDAAc;AACtB,QAAQ,iDAAc;AACtB,SAAS,kDAAe;AACxB,YAAY,qDAAkB;AAC9B,WAAW,oDAAiB;AAC5B,QAAQ,iDAAc;AACtB,QAAQ,iDAAc;AACtB,SAAS,kDAAe;AACxB,SAAS,kDAAe;AACxB,UAAU,mDAAgB;AAC1B,UAAU,mDAAgB;AAC1B,UAAU,mDAAgB;AAC1B,SAAS,kDAAe;AACxB,SAAS,kDAAe;AACxB,MAAM,+CAAY;AAClB,YAAY,qDAAkB;AAC9B,qBAAqB,qDAAkB;AACvC;;AAEA,iEAAe,SAAS,EAAC;;;;;;;;;;;;;;;;;ACzFW;AACE;;AAEtC;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB,mDAAmD,uDAAO;AAC1D,uFAAuF,wDAAQ;AAC/F;AACA;AACA,GAAG;AACH;;;;;;;;;;;;;;;ACbA,6BAAe,uCAAY;AAC3B,4DAA4D,OAAO;AACnE,+DAA+D,OAAO;AACtE;AACA;AACA;AACA;;;;;;;;;;;;;;;ACNA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC;AACrC;AACA;;;;;;;;;;;;;;;ACdA;AACA;AACA;;AAEA,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;;ACNqC;;AAErC,6BAAe,oCAAS;AACxB;;AAEA,+JAA+J,OAAO;AACtK,yHAAyH,OAAO;AAChI;AACA;AACA;AACA;AACA;;AAEA,SAAS,QAAQ;AACjB;AACA;;AAEA,aAAa,gDAAS;AACtB;;;;;;;;;;;;;;;AClBA,6BAAe,sCAAW;;AAE1B,4DAA4D,OAAO;AACnE,yDAAyD,OAAO;AAChE;AACA;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;ACVA,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;ACFA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,YAAY;AACZ,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA,8CAA8C,OAAO;AACrD;AACA;AACA,QAAQ;AACR;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,2CAA2C,OAAO;AAClD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA,8CAA8C,OAAO;AACrD,6BAA6B,OAAO;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,cAAc,OAAO;AACrB;AACA;;;;;;;;;;;;;;;AClEA,6BAAe,sCAAW;;AAE1B,6DAA6D,QAAQ;AACrE,6EAA6E,SAAS;AACtF;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;ACZA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;AC3BA;AACA;AACA;;AAEA,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;ACNA;AACA;AACA;AACA;;AAEA,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;;;ACPqC;AACC;;AAEtC,6BAAe,oCAAS;AACxB,6CAA6C,wDAAQ;;AAErD,sFAAsF,OAAO;AAC7F,gHAAgH,OAAO;AACvH;AACA;AACA;AACA;AACA;AACA;;AAEA,aAAa,gDAAS;AACtB;;;;;;;;;;;;;;;;;;AChBqC;AACL;AACY;;AAE5C;AACA;AACA,WAAW,qDAAK;AAChB;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA,gBAAgB,2DAAW;;AAE3B,0FAA0F,OAAO;AACjG,+DAA+D,OAAO;AACtE;AACA;AACA;AACA;AACA;AACA;;AAEA,aAAa,gDAAS;AACtB;;;;;;;;;;;;;;;;ACxB2C;;AAE3C;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA,wDAAwD,yDAAY;AACpE;;;;;;;;;;;;;;;;ACjB2C;;AAE3C;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA,6DAA6D,yDAAY;AACzE;;;;;;;;;;;;;;;ACjBA,6BAAe,sCAAW;AAC1B;AACA,mCAAmC;AACnC;AACA;;;;;;;;;;;;;;;;ACJqC;;AAErC,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA;;AAEA,uFAAuF,OAAO;AAC9F,yGAAyG,OAAO;AAChH;AACA;AACA;AACA;AACA;AACA;;AAEA,aAAa,gDAAS;AACtB;;AAEA;AACA;AACA;;;;;;;;;;;;;;;ACvBA,6BAAe,oCAAS;AACxB;AACA;;;;;;;;;;;;;;;;;ACFuC;;AAEvC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA,SAAS,sDAAW;AACpB;;;;;;;;;;;;;;;AClCA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACxBA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;ACNA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;ACRA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;;;;;;;;;;;;;;;ACJA,6BAAe,oCAAS;AACxB;AACA;AACA,2BAA2B;AAC3B;;;;;;;;;;;;;;;;ACJiC;;AAEjC,6BAAe,oCAAS;AACxB,cAAc,4CAAK;AACnB;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;;;;;;;;;;;;;;;;;ACVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sIAAsI;;AAE/H;AACP;AACA;;AAEA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;;AAEO;AACP,SAAS;AACT,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,qBAAqB;AACrB;AACA,+BAA+B;AAC/B;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;;;;;;;;;;;;;;;;AC7GiD;AACE;;AAEnD;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;;AAEA;AACA;AACA;AACA,4CAA4C,8DAAS;AACrD,mBAAmB,4DAAU;AAC7B;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;;;;;;;ACpB8B;AAC8B;AACd;AACM;;;;;;;;;;;;;;;;ACHa;;AAEjE,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA,mDAAmD,eAAe;AAClE,8BAA8B,6DAAQ,qBAAqB,2DAAM;AACjE,qBAAqB,0DAAK;AAC1B;AACA;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;ACvBuC;AACU;AACE;;AAEnD,oDAAS,uBAAuB,qDAAmB;AACnD,oDAAS,wBAAwB,sDAAoB;;;;;;;;;;;;;;;;ACLb;;AAExC,6BAAe,oCAAS;AACxB;AACA,IAAI,yDAAS;AACb,GAAG;AACH;;;;;;;;;;;;;;;;;;;ACNyD;AACR;AACV;AACV;;AAE7B;AACA;AACA;AACA;AACA,QAAQ,+CAAc;AACtB;;AAEA;AACA;AACA;AACA;AACA,oCAAoC,IAAI;AACxC;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;;AAEA,sBAAsB,4DAAU;AAChC;AACA,IAAI;AACJ,SAAS,2DAAK,oCAAoC,6CAAG;AACrD;;AAEA,4DAA4D,OAAO;AACnE,+DAA+D,OAAO;AACtE;AACA,QAAQ,mEAAQ;AAChB;AACA;AACA;;AAEA,aAAa,4DAAU;AACvB;;;;;;;;;;;;;;;;;;;ACzC+E;AACxC;AACD;AACK;;AAE3C;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB,iBAAiB,wDAAS,uCAAuC,mEAAoB,GAAG,uDAAW;AACnG;AACA,sEAAsE,qDAAU;AAChF;AACA;AACA;;;;;;;;;;;;;;;;AC7EuC;;AAEvC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA,iBAAiB,wDAAS;AAC1B;AACA;;;;;;;;;;;;;;;;AC3CwC;;AAExC;AACA;AACA,IAAI,kDAAI;AACR;AACA;;AAEA;AACA;AACA,IAAI,kDAAI;AACR;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA;AACA;AACA,QAAQ,iDAAG;AACX;;;;;;;;;;;;;;;;ACtBuC;;AAEvC;AACA;AACA,IAAI,iDAAG;AACP;AACA;;AAEA;AACA;AACA,IAAI,iDAAG;AACP;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA;AACA;AACA,QAAQ,iDAAG;AACX;;;;;;;;;;;;;;;;ACtBuC;;AAEvC;AACA;AACA;AACA,IAAI,iDAAG;AACP;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA;AACA,QAAQ,iDAAG;AACX;;;;;;;;;;;;;;;;ACfkC;;AAElC;AACA;AACA;AACA;AACA,IAAI,iDAAG;AACP;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;;;;;;;;;;;;;;;;ACbkC;;AAElC,6BAAe,sCAAW;AAC1B;AACA;AACA,kBAAkB,cAAc;AAChC,eAAe,oBAAoB;;AAEnC;AACA,qBAAqB,iDAAG;AACxB;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,KAAK;;AAEL;AACA;AACA,GAAG;AACH;;;;;;;;;;;;;;;;;AC5BqC;AACC;;AAEtC,6BAAe,oCAAS;AACxB,2CAA2C,wDAAO;;AAElD,sFAAsF,OAAO;AAC7F,6FAA6F,OAAO;AACpG;AACA;AACA;AACA;AACA;;AAEA,aAAa,iDAAU;AACvB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACfuC;AACC;AACU;AACR;AACM;AACR;AACc;AACV;AACF;AACN;AACQ;AACA;AACM;AACA;AACR;AACU;AACZ;AACU;AACE;AACV;AACJ;;AAEtC;;AAEO;AACP;AACA;AACA;AACA;AACA;;AAEe;AACf,SAAS,wDAAS;AAClB;;AAEO;AACP;AACA;;AAEA,0BAA0B,oDAAS;;AAEnC;AACA;AACA,UAAU,kDAAiB;AAC3B,aAAa,qDAAoB;AACjC;AACA;AACA,UAAU,kDAAiB;AAC3B,SAAS,iDAAgB;AACzB,aAAa,qDAAoB;AACjC,cAAc,sDAAqB;AACnC;AACA;AACA;AACA;AACA;AACA;AACA,MAAM,8CAAa;AACnB,QAAQ,gDAAe;AACvB,aAAa,qDAAoB;AACjC,SAAS,kDAAgB;AACzB,cAAc,uDAAqB;AACnC,QAAQ,iDAAe;AACvB,aAAa,sDAAoB;AACjC,UAAU,mDAAiB;AAC3B,SAAS,kDAAgB;AACzB,SAAS,kDAAgB;AACzB,YAAY,qDAAmB;AAC/B,QAAQ,iDAAe;AACvB,eAAe,wDAAsB;AACrC,OAAO,gDAAc;AACrB;AACA;;;;;;;;;;;;;;;;;;;ACxE+B;AACqD;;AAEpF,6BAAe,oCAAS;AACxB;AACA,kCAAkC,sDAAiB;AACnD,qBAAqB,gDAAK,GAAG,sDAAc;AAC3C,aAAa,oDAAK,eAAe,sDAAc;AAC/C,QAAQ,sDAAiB;AACzB;;;;;;;;;;;;;;;;ACTsC;;AAEtC,6BAAe,oCAAS;AACxB;;AAEA,gKAAgK,OAAO;AACvK,yHAAyH,OAAO;AAChI;AACA;AACA;AACA;AACA;;AAEA,SAAS,QAAQ;AACjB;AACA;;AAEA,aAAa,iDAAU;AACvB;;;;;;;;;;;;;;;;AClB6C;;AAE7C;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA,oCAAoC,8CAAI,GAAG,6CAAG;AAC9C;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;AACA,QAAQ,iDAAG;AACX;AACA;;;;;;;;;;;;;;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACVqC;AACG;;AAExC,cAAc,uDAAQ;AACtB;;AAEO;AACA;AACA;AACA;AACA;AACA;AACA;;AAEP,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEO;AACP;AACA,2DAA2D;AAC3D;AACA;;AAEO;AACP;AACA,2DAA2D;AAC3D;AACA;;AAEO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,eAAe,+CAAK;;AAEpB;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,4CAA4C;AAC5C,sCAAsC,oDAAO;;AAE7C;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,yDAAyD;AACzD;AACA,2DAA2D;AAC3D;AACA,IAAI,oDAAO;AACX;AACA;AACA;AACA;AACA;AACA,KAAK;;AAEL;AACA;AACA;AACA;AACA,yCAAyC;AACzC;;AAEA;AACA;AACA,wBAAwB,OAAO;AAC/B;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,qCAAqC;AACrC;AACA;AACA;;;;;;;;;;;;;;;;;;ACxJsC;AACA;AACM;;AAE5C,6BAAe,oCAAS;AACxB;AACA;;AAEA,6CAA6C,wDAAQ;;AAErD,sFAAsF,OAAO;AAC7F,gHAAgH,OAAO;AACvH;AACA;AACA;AACA,QAAQ,wDAAQ,qCAAqC,iDAAG;AACxD;AACA;AACA;;AAEA,aAAa,iDAAU;AACvB;;;;;;;;;;;;;;;;;;ACrByC;AACH;AACM;;AAE5C,6BAAe,oCAAS;AACxB;AACA;;AAEA,6CAA6C,wDAAW;;AAExD,0FAA0F,OAAO;AACjG,+DAA+D,OAAO;AACtE;AACA,yFAAyF,iDAAG,wCAAwC,OAAO;AAC3I;AACA,YAAY,wDAAQ;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,aAAa,iDAAU;AACvB;;;;;;;;;;;;;;;;ACzBuC;;AAEvC,gBAAgB,oDAAS;;AAEzB,6BAAe,sCAAW;AAC1B;AACA;;;;;;;;;;;;;;;;;;;;ACN+E;AAC5C;AACD;AACI;AACK;;AAE3C;AACA;AACA;AACA;AACA;AACA,kBAAkB,wDAAK;AACvB,oDAAoD,wDAAK;AACzD;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,kBAAkB,wDAAK;AACvB;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,kBAAkB,wDAAK;AACvB;AACA;AACA,6EAA6E,wDAAK;AAClF;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,mBAAmB,iDAAG;AACtB;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB,yCAAyC,mEAAoB,GAAG,uDAAW;AAC3E;AACA;AACA;AACA;AACA,+CAA+C,qDAAU;AACzD;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;AC/EA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACvBsC;;AAEtC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA,qBAAqB,qDAAU;AAC/B;AACA;;;;;;;;;;;;;;;ACnBA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACvB6C;AACD;;AAE5C,6BAAe,sCAAW;AAC1B;AACA;AACA,YAAY,gDAAK;;AAEjB,4DAA4D,OAAO;AACnE,+DAA+D,OAAO;AACtE;AACA,sBAAsB,iDAAG;AACzB,QAAQ,wDAAQ;AAChB;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;;AAEA,aAAa,iDAAU;AACvB;;;;;;;;;;;;;;;;;ACvBuC;;AAEvC;AACA;AACA;AACA,mBAAmB,iDAAG;AACtB;;AAEA;AACA;AACA;AACA;AACA;AACA,yCAAyC,OAAO;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,mBAAmB,iDAAG;AACtB;;AAEA;AACA;AACA;AACA;AACA;AACA,oBAAoB,yBAAyB,4BAA4B,OAAO;AAChF;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,6BAAe,oCAAS;AACxB;;AAEA;;AAEA;AACA,gBAAgB,iDAAG;AACnB,yCAAyC,OAAO;AAChD;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEO;AACP;;AAEA;AACA,mBAAmB,iDAAG;AACtB,2CAA2C;AAC3C,GAAG;;AAEH;AACA,WAAW,iDAAG;AACd;AACA;;;;;;;;;;;;;;;AChFA,iEAAe,YAAY,EAAC;;;;;;;;;;;;;;;ACAb;AACf;AACA;AACA;AACA;AACA,CAAC;AACD;AACA,WAAW,kDAAkD;AAC7D,kBAAkB,yDAAyD;AAC3E,aAAa,oDAAoD;AACjE,gBAAgB,uDAAuD;AACvE,QAAQ;AACR,GAAG;AACH;;;;;;;;;;;;;;;;;;;;ACb0C;AACoE;;;;;;;;;;;;;;;;ACDvG;AACP;AACA;;AAEA,6BAAe,oCAAS;AACxB;AACA;AACA;;;;;;;;;;;;;;;;;ACPO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;;AAEO;;AAEP;;AAEe;AACf;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;AClDqC;AACW;AACD;AACF;AACL;AACH;AACF;AACgB;AACC;;AAEpD;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,wBAAwB,mDAAQ;AAChC;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6BAAe,sCAAW;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,sDAAe;AACnC,kBAAkB,uDAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,oCAAoC,eAAe;AACnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA;AACA,uBAAuB,mDAAQ;AAC/B;AACA;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA,+CAA+C,oDAAS;AACxD;;AAEA;AACA;AACA,oEAAoE,oDAAS;AAC7E;;AAEA;AACA;AACA;;AAEA;AACA;AACA,uCAAuC,gDAAgD;AACvF,oDAAoD,8CAA8C;AAClG;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC;AAChC,mBAAmB,4BAA4B,QAAQ,oDAAS;AAChE;AACA;AACA,SAAS;AACT;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA,cAAc,wDAAM;AACpB;AACA;AACA;AACA,YAAY,iDAAS;AACrB;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,YAAY,wDAAO;;AAEnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,MAAM,wDAAS;AACf;AACA;;AAEA,IAAI,uDAAO;AACX;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,YAAY,wDAAM;AAClB,YAAY,wDAAO;AACnB;AACA;;AAEA,IAAI,mDAAW;AACf,IAAI,0DAAa;AACjB;AACA,IAAI,wDAAS;AACb;;AAEA;AACA,MAAM,uDAAO;AACb;AACA;AACA;AACA;AACA;AACA,sEAAsE,wDAAO;AAC7E;;AAEA;AACA;AACA,MAAM,gDAAU;AAChB,MAAM,uDAAO;AACb;AACA;AACA;;AAEA;AACA;AACA;AACA,aAAa,wDAAO;AACpB;AACA;AACA;;AAEA,IAAI,uDAAO;AACX,sBAAsB,wDAAM;AAC5B,SAAS,wDAAM;AACf;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,IAAI,0DAAa;AACjB,gBAAgB,OAAO;AACvB,0BAA0B,wDAAO;AACjC;AACA;AACA;AACA;;AAEA;;AAEA;AACA,iFAAiF,uBAAuB;AACxG,MAAM,wDAAS;AACf;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,IAAI,uDAAO;AACX,gBAAgB,OAAO;AACvB,0BAA0B,wDAAO;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,IAAI,0DAAa;AACjB;AACA,0CAA0C,qBAAqB;AAC/D,gBAAgB,OAAO;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,wDAAO;AACnB;AACA,kBAAkB,wDAAM;AACxB;AACA;AACA;AACA;AACA;;AAEA;AACA,0EAA0E,wDAAQ;AAClF;;AAEA;AACA,sEAAsE,wDAAQ;AAC9E;;AAEA;AACA,yEAAyE,wDAAQ;AACjF;;AAEA;AACA,sEAAsE,wDAAQ;AAC9E;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;;AC9bqC;;AAErC;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;;AAEA;AACA,gCAAgC,wDAAM;;AAEtC;AACA,uBAAuB,wDAAM;AAC7B;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AClEoC;;AAEpC;AACA;AACA;AACA;AACA;AACA;AACO,+BAA+B,8CAAM;AAC5C;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,mCAAmC,eAAe,YAAY,SAAS;AACvE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;;AAET;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACxHoC;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACO,0BAA0B,8CAAM;AACvC;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACpGA;AACY;AACZ;AACA;AACA;AACA,cAAc,uBAAuB;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,wBAAwB;AACvC,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,gBAAgB,6EAA6E;AAC7F;AACA;AACA,+CAA+C;AAC/C;AACA;AACA;AACA,kBAAkB,wBAAwB;AAC1C;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,sCAAsC;AACxD;AACA;AACA;AACA;AACA,kBAAkB,uCAAuC;AACzD;AACA;AACA;AACA;AACA,kBAAkB,cAAc;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,0CAA0C;AACzD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,gCAAgC;AAC/C,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,oBAAoB;AACpC,eAAe,cAAc;AAC7B,iBAAiB;AACjB;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,QAAQ;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA,kBAAkB,cAAc;AAChC;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACzJA;AACY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA;AACA;AACA,kBAAkB,qCAAqC;AACvD;AACA;AACA;AACA;AACA;AACA,kBAAkB,2CAA2C;AAC7D;AACA;AACA,mBAAmB,qCAAqC;AACxD;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,oCAAoC;AACnD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;AC3DA;AACY;AACZ;AACA,gBAAgB,0DAA0D;AAC1E;AACA;AACA;AACA,cAAc,kCAAkC;AAChD;AACA,CAAqC;AACiB;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA,eAAe,mBAAmB;AAClC,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA,mBAAmB,mBAAmB;AACtC;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,OAAO;AACzB,sBAAsB;AACtB;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA,mBAAmB,iCAAiC;AACpD;AACA;AACA,mBAAmB,iCAAiC;AACpD;AACA;AACA,mBAAmB,iCAAiC;AACpD;AACA;AACA;AACA,mBAAmB,UAAU;AAC7B,wBAAwB;AACxB;AACA;AACA;AACA;AACA,qBAAqB,iDAAY;AACjC,sBAAsB,6CAAM;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA,wCAAwC;AACxC;AACA;AACA,iBAAiB;AACjB,cAAc,wDAAM;AACpB;AACA;AACA;AACA,mBAAmB,kBAAkB;AACrC;AACA,mBAAmB,kBAAkB;AACrC;AACA,mBAAmB,kBAAkB;AACrC;AACA,mBAAmB,kBAAkB;AACrC;AACA;AACA;AACA,mBAAmB,gBAAgB;AACnC;AACA;AACA;AACA,oBAAoB,wDAAwD;AAC5E,4BAA4B;AAC5B;AACA;AACA,kBAAkB,MAAM;AACxB;AACA;AACA;AACA;AACA,gBAAgB,gBAAgB;AAChC;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA;AACA,gBAAgB,gBAAgB;AAChC;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB,UAAU;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,oBAAoB;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB,QAAQ;AACzB;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB,QAAQ;AACzB;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB,QAAQ;AACzB;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB,QAAQ;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AClaA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACO,oBAAoB,kDAAQ;AACnC;AACA;AACA;AACA,eAAe,iCAAiC;AAChD,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACtBA;AACY;AACZ;AACA,CAAqC;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA,gCAAgC,wDAAM;AACtC;AACA;AACA,uBAAuB,wDAAM;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB;AACA;AACA,oBAAoB;AACpB;AACA;AACA;;;;;;;;;;;;;;;;;;;;AC3GA;AACY;AACZ;AACA;AACA,CAA0C;AACJ;AACgB;AACU;AAChE;AACA;AACqC;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,gBAAgB;AAC/B,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA,mBAAmB,mBAAmB;AACtC;AACA;AACA;AACA,kBAAkB;AAClB;AACA,6BAA6B,oDAAS;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc,wDAAM;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,SAAS;AAC3B,2BAA2B,gDAAO;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,wBAAwB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,wDAAM;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,mEAAmE;AAClF,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,MAAM;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,YAAY;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,gEAAgE,aAAa;AACjG;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,oBAAoB;AACpC,kBAAkB,gEAAgE;AAClF;AACA;AACA;AACA;AACA;AACA,6CAA6C,QAAQ;AACrD,uBAAuB,4BAA4B;AACnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,uCAAuC;AAC9D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,kBAAkB;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,+BAA+B;AACjD;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,gBAAgB,QAAQ,YAAY,MAAM;AAC1C;AACA;AACA;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA,gBAAgB,gBAAgB,cAAc,MAAM;AACpD;AACA;AACA;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA,gBAAgB,gBAAgB,cAAc,MAAM;AACpD;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,gBAAgB,QAAQ,cAAc,MAAM;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B,+DAAW;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oCAAoC,yEAAgB;AACpD;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,kBAAkB,MAAM;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;ACtfA;AACY;AACZ;AACA;AACA;AACA,6BAA6B,cAAc;AAC3C;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,gBAAgB;AAC/B,eAAe,kFAAkF;AACjG,iBAAiB,0DAA0D;AAC3E;AACA,gDAAgD;AAChD;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA,kBAAkB,qCAAqC;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,4CAA4C;AAC3D,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,gBAAgB;AAC/B,eAAe,8CAA8C;AAC7D,iBAAiB,0DAA0D;AAC3E,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;;;;;;;;;;;;;;;;AC5FA;AACY;AACZ;AACA,CAAwC;AACxC;AACA,cAAc,2GAA2G;AACzH;AACA;AACA;AACA;AACA,aAAa,gEAAgE;AAC7E;AACA;AACA;AACA;AACA;AACA;AACA;AACO,oBAAoB,kDAAQ;AACnC;AACA;AACA,gBAAgB,gHAAgH,oBAAoB,sKAAsK;AAC1T;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,qBAAqB;AACvC;AACA;AACA;AACA;AACA,kBAAkB,0DAA0D;AAC5E;AACA;AACA;AACA;AACA,kBAAkB,kDAAkD,sBAAsB;AAC1F,sDAAsD,cAAc;AACpE;AACA;AACA,kBAAkB,oBAAoB;AACtC;AACA;AACA;AACA;AACA,kBAAkB,mCAAmC;AACrD;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,kCAAkC;AACjD,eAAe,iCAAiC;AAChD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,kDAAkD,sBAAsB;AAC1F;AACA;AACA;AACA,gBAAgB,kDAAkD,sBAAsB,cAAc,MAAM;AAC5G;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB,MAAM;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,mCAAmC;AAClD,iBAAiB,MAAM;AACvB;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACjGA;AACY;;AAEZ,CAAqC;AACrC,WAAW,aAAa;;AAExB;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,QAAQ;AACvB;AACA;AACA;;AAEA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;;AAEA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,aAAa;AAChC;AACA,mBAAmB,aAAa;AAChC;;AAEA;AACA;AACA,kBAAkB,kCAAkC;AACpD;AACA,cAAc,wDAAM;AACpB,cAAc,wDAAM;;AAEpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;;AAEA;AACA;AACA,eAAe,YAAY;AAC3B;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,YAAY;AAC3B,eAAe,SAAS;AACxB,eAAe,aAAa;AAC5B;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACpLA;AACY;AACZ;AACA,gBAAgB,+DAA+D,kBAAkB,iFAAiF;AAClL;AACA,CAA8B;AACc;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACO,sBAAsB,qDAAO;AACpC;AACA,eAAe,8BAA8B;AAC7C,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,gBAAgB,sFAAsF;AACtG;AACA,+CAA+C;AAC/C;AACA;AACA;AACA;AACA,kBAAkB,2CAA2C;AAC7D;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,mDAAmD;AAClE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA,mCAAmC,oDAAG;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,yCAAyC;AACxD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC/GA;AACY;AACZ;AACA,CAA4C;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACO,qBAAqB,qDAAO;AACnC;AACA,eAAe,QAAQ;AACvB,eAAe,gBAAgB;AAC/B,iBAAiB;AACjB;AACA,4CAA4C;AAC5C;AACA;AACA;AACA;AACA,kBAAkB,2CAA2C;AAC7D;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,mDAAmD;AAClE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,yCAAyC;AACxD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACtDA;AACY;AACZ;AACA,gBAAgB,+DAA+D,kBAAkB,iFAAiF;AAClL;AACA;AACA,CAA4C;AAC5C,WAAW,2BAA2B;AACtC;AACA;AACoC;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACO,wBAAwB,qDAAO;AACtC;AACA,eAAe,8BAA8B;AAC7C,eAAe,QAAQ;AACvB,gBAAgB,uGAAuG;AACvH;AACA,mCAAmC;AACnC;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA,uCAAuC,oDAAI;AAC3C;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA,aAAa;AACb,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,yCAAyC;AACxD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,yCAAyC;AACxD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,yDAAyD;AACzE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,YAAY;AACxC,gCAAgC,YAAY;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B,6CAAG,IAAI,SAAS,EAAE,GAAG,GAAG,GAAG;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV,gDAAgD,GAAG,IAAI,GAAG;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,yBAAyB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,yCAAyC;AACxD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,mDAAmD;AACtE;AACA;AACA;AACA;AACA,mBAAmB,yCAAyC;AAC5D;AACA;AACA,kDAAkD,kCAAkC;AACpF;AACA,sDAAsD,kCAAkC;AACxF;AACA,2BAA2B,QAAQ;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD;AACA,eAAe,QAAQ;AACvB;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA,eAAe,sCAAsC;AACrD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AChSA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,8BAA8B,iDAAK;AAC1C;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA,mBAAmB,uCAAuC;AAC1D;AACA;AACA,mBAAmB,gBAAgB;AACnC;AACA;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,yBAAyB;AACxC,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,uCAAuC;AACtD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,8BAA8B;AACnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,UAAU;AACrC,+BAA+B,UAAU;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACjKA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,mCAAmC,iDAAK;AAC/C;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB,4BAA4B;AAC9C;AACA;AACA;AACA;AACA,eAAe,0CAA0C;AACzD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AChFA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,iCAAiC,iDAAK;AAC7C;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA,mBAAmB,4BAA4B;AAC/C;AACA;AACA,mBAAmB,kBAAkB;AACrC;AACA,mBAAmB,kBAAkB;AACrC;AACA,mBAAmB,kBAAkB;AACrC;AACA,mBAAmB,kBAAkB;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,uCAAuC;AACtD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACvGA;AACY;AACZ;AACA,CAAwC;AACT;AAC/B;AACA;AACA;AACA;AACA;AACO,2BAA2B,iDAAK;AACvC;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA,kBAAkB,kDAAkD;AACpE;AACA;AACA;AACA;AACA,kBAAkB,4BAA4B;AAC9C;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA,eAAe,0CAA0C;AACzD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,eAAe;AAC/C;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,iBAAiB;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,oDAAI;AACpC;AACA,wBAAwB,iBAAiB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACrNA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACO,wBAAwB,iDAAK;AACpC;AACA,eAAe,mGAAmG;AAClH,eAAe,uCAAuC;AACtD,gBAAgB,oNAAoN;AACpO;AACA,0CAA0C;AAC1C;AACA;AACA;AACA,mBAAmB,mGAAmG;AACtH;AACA;AACA,mBAAmB,uCAAuC;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA,kBAAkB,yDAAyD;AAC3E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,gDAAgD;AACnE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,+EAA+E;AAC9F,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,gCAAgC;AAC/C,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACnJA;AACY;AACZ;AACA,CAAwC;AACV;AAC9B;AACA;AACA,cAAc,oCAAoC;AAClD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,yBAAyB,iDAAK;AACrC;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA,kBAAkB,+BAA+B;AACjD;AACA;AACA;AACA;AACA,kBAAkB,+BAA+B;AACjD;AACA;AACA;AACA;AACA,kBAAkB,+BAA+B;AACjD;AACA;AACA;AACA;AACA,kBAAkB,+BAA+B;AACjD;AACA;AACA;AACA;AACA;AACA,kBAAkB,iBAAiB;AACnC;AACA;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,yBAAyB;AAC3C;AACA;AACA;AACA;AACA,kBAAkB,2BAA2B;AAC7C;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,uCAAuC;AACtD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,iBAAiB;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC,+BAA+B,6CAAG;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACjMA;AACY;AACZ;AACA,CAA0C;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACO,kCAAkC,mDAAM;AAC/C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,yBAAyB;AAC3C;AACA;AACA;AACA;AACA,kBAAkB,kCAAkC;AACpD;AACA,6CAA6C;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,QAAQ;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC5GA;AACY;AACZ;AACA,CAA0C;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,kCAAkC,mDAAM;AAC/C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA,4BAA4B,4DAA4D;AACxF;AACA,4BAA4B,4DAA4D;AACxF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,+CAA+C;AAC/D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,QAAQ;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,QAAQ;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,QAAQ;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACnGA;AACY;AACZ;AACA,CAA0C;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,0BAA0B,mDAAM;AACvC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,+CAA+C;AAC/D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oEAAoE,eAAe;AACnF;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,OAAO;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;AClHA;AACY;AACZ;AACA,CAA0C;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACO,gCAAgC,mDAAM;AAC7C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,2GAA2G;AAC5H;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,gBAAgB;AAC3B,WAAW,gBAAgB;AAC3B,WAAW,QAAQ;AACnB,eAAe;AACf;AACO,0DAA0D;AACjE;AACA,oBAAoB,yBAAyB;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;ACxFA;AACY;AACZ;AACA,CAA0C;AACF;AACV;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACO,yBAAyB,mDAAM;AACtC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,oBAAoB,8FAA8F;AAClH;AACA;AACA;AACA,oBAAoB,0CAA0C;AAC9D;AACA;AACA;AACA,oBAAoB,wCAAwC;AAC5D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,mGAAmG;AACpH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA,UAAU;AACV;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+EAA+E;AAC/E;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,gBAAgB;AAC3B,WAAW,yBAAyB;AACpC,YAAY,SAAS;AACrB,aAAa;AACb;AACO,2CAA2C;AAClD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,qDAAqD;AACjE,WAAW,GAAG;AACd,aAAa;AACb;AACO,6CAA6C;AACpD;AACA;AACA;AACA;AACA,2CAA2C,qDAAI,MAAM,oDAAG;AACxD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,iBAAiB;AAC7B,YAAY,iBAAiB;AAC7B,YAAY,SAAS;AACrB,aAAa;AACb;AACO,oDAAoD;AAC3D;AACA;AACA;AACA,mCAAmC,QAAQ;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,SAAS;AACrB,YAAY,SAAS;AACrB,aAAa;AACb;AACO,2DAA2D;AAClE;AACA;AACA;AACA;AACA,kCAAkC,QAAQ;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,0BAA0B;AACtC,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;AC/NA;AACY;AACZ;AACA,CAAqC;AACK;AAC1C;AACA;AACA;AACA;AACA;AACA;AACO,4BAA4B,mDAAM;AACzC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yDAAyD,wDAAwD;AACjH,oBAAoB,yDAAM;AAC1B;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA,2DAA2D;AAC3D,oBAAoB,wDAAM;AAC1B;AACA,iBAAiB;AACjB;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,SAAS;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,SAAS;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;;;;;;;;;;;;;;;;AC7SA;AACY;AACZ;AACA,CAAwC;AACxC;AACA,cAAc,oEAAoE;AAClF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,+BAA+B,iDAAK;AAC3C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA,kBAAkB,mFAAmF;AACrG;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,QAAQ;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,QAAQ;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,QAAQ;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;;;;;AClSA;AACY;AACZ;AACA,CAAwC;AACA;AACmC;AACN;AACrC;AAChC;AACA;AACA;AACA;AACA;AACA;AACO,8BAA8B,iDAAK;AAC1C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,wCAAwC;AAC1D;AACA;AACA;AACA,kBAAkB,wCAAwC;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,qDAAY;AACjC;AACA;AACA;AACA,YAAY,uEAAiB;AAC7B;AACA,0BAA0B,qEAAe;AACzC;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,8EAAmB;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,oDAAK;AAChC;AACA;AACA;AACA,gCAAgC,gBAAgB;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,gBAAgB;AAChD;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;ACpIA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,yBAAyB,iDAAK;AACrC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,mBAAmB,mFAAmF;AACtG;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;;ACnGA;AACY;AACZ;AACA,CAAwC;AACE;AAC1C;AACA,eAAe,6HAA6H;AAC5I;AACA;AACA;AACA;AACA;AACO,4BAA4B,iDAAK;AACxC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA;AACA,kBAAkB,SAAS;AAC3B;AACA;AACA;AACA;AACA,kBAAkB,SAAS;AAC3B;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,cAAc;AAClC,sBAAsB,oDAAS;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACxNA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,2BAA2B,iDAAK;AACvC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,iBAAiB,uBAAuB,wBAAwB;AAClF;AACA;AACA,kBAAkB,iBAAiB,uBAAuB,wBAAwB;AAClF;AACA;AACA,kBAAkB,iBAAiB,uBAAuB,wBAAwB;AAClF;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,QAAQ;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,yBAAyB;AAC9C,qBAAqB;AACrB;AACA;AACA;AACA,2BAA2B,WAAW;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA,+BAA+B,WAAW;AAC1C;AACA,2BAA2B,QAAQ;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;AC9HA;AACY;AACZ;AACA,CAA8C;AACA;AACoC;AAC1C;AACxC;AACA;AACA;AACA;AACA;AACO;AACP,+CAA+C;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,wDAAW;AAC9B;AACA,wBAAwB,wDAAW;AACnC;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA,+BAA+B,uEAAuE;AACtG;AACA;AACA;AACA;AACA,eAAe,oDAAoD;AACnE,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,4FAA6B,GAAG,0BAA0B;AAChF;AACA,wBAAwB,wDAAW,GAAG,8DAA8D;AACpG;AACA,2CAA2C,kDAAkD;AAC7F;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,iDAAK;AAChC,gBAAgB,kBAAkB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC7GA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,0BAA0B,iDAAK;AACtC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;AClGA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,6BAA6B,iDAAK;AACzC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,iDAAiD;AACnE;AACA;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;ACrGA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,0BAA0B,iDAAK;AACtC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,yCAAyC;AAC3D;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA,mBAAmB,QAAQ;AAC3B;AACA,mBAAmB,QAAQ;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4CAA4C,sCAAsC;AAClF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;AC9MA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACO,2BAA2B,iDAAK;AACvC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,QAAQ;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,kEAAkE;AAC/F;AACA;;;;;;;;;;;;;;;;ACxGA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,kCAAkC,iDAAK;AAC9C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA,mBAAmB,uFAAuF;AAC1G;AACA;AACA,mBAAmB,uFAAuF;AAC1G;AACA;AACA,mBAAmB,+GAA+G;AAClI;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA,6BAA6B,kEAAkE;AAC/F;AACA;;;;;;;;;;;;;;;;AClJA;AACY;AACZ;AACA,CAA0C;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACO,gCAAgC,oDAAS;AAChD,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,oEAAoE;AACtF;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,uCAAuC;AAC3D,sBAAsB,oDAAS;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;AC/GA;AACY;AACZ;AACA,CAAwC;AACxC;AACA,gBAAgB,gIAAgI;AAChJ;AACA;AACA,aAAa,yCAAyC;AACtD;AACA;AACA;AACA;AACA;AACO,wBAAwB,iDAAK;AACpC,gBAAgB,QAAQ;AACxB,yBAAyB;AACzB;AACA;AACA;AACA,kBAAkB,+CAA+C;AACjE;AACA;AACA;AACA,kBAAkB,+CAA+C;AACjE;AACA;AACA;AACA,kBAAkB,+CAA+C;AACjE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,QAAQ;AACvB,eAAe,uCAAuC;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,cAAc;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA,2BAA2B,kBAAkB;AAC7C;AACA;AACA;AACA;AACA,2BAA2B,kBAAkB;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,QAAQ;AACvB,eAAe,SAAS;AACxB,eAAe,SAAS;AACxB,eAAe,SAAS;AACxB,kBAAkB;AAClB;AACA;AACA,oBAAoB,eAAe;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,kBAAkB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,kBAAkB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACnNA;AACY;AACZ;AACA,CAAwC;AACgB;AACmC;AAC3F;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,4CAA4C,iDAAK;AACxD,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,oDAAoD;AACtE;AACA;AACA;AACA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA,oBAAoB,SAAS;AAC7B;AACA,wBAAwB,kBAAkB;AAC1C;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA,mBAAmB,iCAAiC;AACpD,uBAAuB,oGAA8B;AACrD;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,QAAQ;AAChC;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,QAAQ;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,qEAAe;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,2CAA2C;AACxE;AACA;;;;;;;;;;;;;;;;;;AC1GA;AACY;AACZ;AACA,CAAwC;AACgB;AAC6B;AACrF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,oCAAoC,iDAAK;AAChD,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA;AACA;AACA,oBAAoB,4BAA4B;AAChD;AACA;AACA;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA,kBAAkB,gCAAgC;AAClD;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,qEAAe;AACrC;AACA;AACA,0CAA0C,4BAA4B;AACtE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,8FAA2B;AACnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;AC7HA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,0BAA0B,iDAAK;AACtC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA,kBAAkB,yEAAyE;AAC3F;AACA;AACA;AACA,kBAAkB,mGAAmG;AACrH;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;;;ACtIA;AACY;AACZ;AACA,CAA0C;AACwC;AACA;AAClF;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,oDAAoD;AACnE,eAAe,gBAAgB;AAC/B,eAAe,gBAAgB;AAC/B,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA,+CAA+C;AAC/C;AACA;AACA;AACA;AACA,oBAAoB,kCAAkC;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA,2BAA2B,2DAAK;AAChC;AACA;AACA,wBAAwB,mBAAmB;AAC3C;AACA,8BAA8B,4FAA6B;AAC3D;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8BAA8B,oDAAS;AACvC;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACpEA;AACY;AACZ;AACA,CAAwC;AACxC;AACA;AACA;AACA;AACA;AACO,wBAAwB,iDAAK;AACpC,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,2EAA2E;AAC7F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,wCAAwC;AACvD,eAAe,uCAAuC;AACtD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;AACA;AACA;AACA;AACA,eAAe,gBAAgB;AAC/B,eAAe,qCAAqC;AACpD,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC7GA;AACY;AACZ;AACA,CAAwC;AACxC;AACA,cAAc,wCAAwC;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,8BAA8B,iDAAK;AAC1C,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA;AACA,kBAAkB,kEAAkE;AACpF;AACA,mBAAmB,kEAAkE;AACrF;AACA;AACA;AACA;AACA,kBAAkB,kEAAkE;AACpF;AACA,mBAAmB,kEAAkE;AACrF;AACA,mBAAmB,2EAA2E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA,kBAAkB,4EAA4E;AAC9F;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,2CAA2C;AAC1D,eAAe,0CAA0C;AACzD,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,QAAQ;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,QAAQ;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,iEAAiE;AAC9F;AACA;;;;;;;;;;;;;;;;;AC7NA;AACY;;AAEZ,CAAiE;AACjC;;AAEhC;AACA;AACA;AACA;AACO;AACP;AACA;AACA,eAAe,uBAAuB;AACtC;AACA;AACA;AACA;;AAEA,uBAAuB,iEAAiB;AACxC;AACA,YAAY,4DAAY;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,4DAAY;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,mBAAmB,oDAAK;AACxB,sBAAsB;AACtB;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;;;;;;;;;;;ACrHA;AACY;;AAEZ,CAAiE;AACjC;;AAEhC;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,eAAe,GAAG;AAClB,eAAe,gBAAgB;AAC/B,gBAAgB,0BAA0B;AAC1C,eAAe,QAAQ;AACvB,eAAe,kBAAkB;AACjC;AACA;AACA,mBAAmB,uBAAuB;AAC1C;AACA;AACA;AACA;;AAEA,mBAAmB,aAAa;AAChC,wBAAwB,4DAAY;AACpC;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,qCAAqC;AACrC,8BAA8B;AAC9B,iCAAiC;AACjC;AACA;AACA,gCAAgC,mBAAmB,uCAAuC;AAC1F;AACA,aAAa;AACb;AACA,8BAA8B;;AAE9B;AACA;AACA;AACA,0DAA0D;AAC1D;AACA;AACA,sEAAsE;AACtE;AACA;AACA;AACA;AACA,wCAAwC;AACxC,qFAAqF;AACrF;AACA;AACA;AACA;AACA,wCAAwC;AACxC,gGAAgG;AAChG;AACA;AACA;AACA,kDAAkD;AAClD;AACA;AACA,qEAAqE;AACrE;AACA;AACA;AACA;AACA,4DAA4D;AAC5D,gGAAgG;AAChG;AACA,cAAc;AACd;AACA;AACA;AACA,kDAAkD;AAClD;AACA;AACA,mFAAmF;AACnF;AACA;AACA;AACA;AACA,4DAA4D;AAC5D,4GAA4G;AAC5G;AACA,cAAc;AACd;AACA,8CAA8C;AAC9C;AACA,UAAU;AACV,0CAA0C;AAC1C;;AAEA;AACA,4DAA4D,gBAAgB;AAC5E,iEAAiE,gBAAgB;AACjF;AACA;AACA;AACA,qCAAqC;AACrC,qCAAqC;AACrC,qDAAqD,OAAO,OAAO,kBAAkB;AACrF,4BAA4B,QAAQ;AACpC;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA,uBAAuB;AACvB;AACA,uBAAuB;AACvB;AACA;AACA;AACA,0BAA0B;AAC1B;AACA,2BAA2B,sBAAsB,gBAAgB,qCAAqC;AACtG;;AAEA;AACA,kGAAkG;AAClG;AACA,6DAA6D;;AAE7D;;AAEA,mBAAmB,aAAa;AAChC,wBAAwB,4DAAY;;AAEpC,mBAAmB,cAAc;AACjC,uBAAuB,iEAAiB;AACxC;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,wBAAwB,mBAAmB;AAC3C,sBAAsB,oDAAK;;AAE3B;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;;;;;;;;;;;ACrOA;AACY;;AAEZ,CAAiE;AACjC;;AAEhC;AACA;AACA;AACA;AACA;AACO;AACP;AACA,eAAe,gBAAgB;AAC/B;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;;AAEA;AACA,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,0CAA0C,mBAAmB;AAC7D;AACA;AACA,wBAAwB,mBAAmB;AAC3C;AACA;AACA;AACA,mBAAmB;AACnB;AACA,mCAAmC;AACnC;AACA,wBAAwB,mBAAmB;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B;AAC3B;AACA,+DAA+D,GAAG;AAClE;AACA,kBAAkB,QAAQ;AAC1B;AACA;;AAEA;AACA;AACA,mBAAmB,aAAa;AAChC,wBAAwB,4DAAY;;AAEpC,mBAAmB,aAAa;AAChC,wBAAwB,4DAAY;;AAEpC,mBAAmB,cAAc;AACjC,wBAAwB,iEAAiB;AACzC;;AAEA;;AAEA;AACA;;AAEA;AACA,wBAAwB,wBAAwB;AAChD,sBAAsB,oDAAK;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;ACpIA;AACY;AACZ;AACA,CAAsC;AACE;AACxC;AACA;AACA;AACA,aAAa,yBAAyB;AACtC;AACA;AACA;AACA,aAAa,yBAAyB;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa,0OAA0O;AACvP,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6CAA6C,oDAAG;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa,iIAAiI;AAC9I,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA,kBAAkB,oDAAa;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,qBAAqB;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa,mKAAmK;AAChL,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,2CAA2C;AAC9D,uBAAuB,oDAAM;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,aAAa;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa,mJAAmJ;AAChK,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc,qBAAqB;AACnC;AACA;AACA;AACA,kBAAkB,oDAAa;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,GAAG;AACd,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,gBAAgB;AAC3B;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,gBAAgB;AAC3B,WAAW,gBAAgB;AAC3B;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,yBAAyB;AACpC,WAAW,QAAQ;AACnB;AACO;AACP;AACA;AACA,oBAAoB,QAAQ;AAC5B;AACA;;;;;;;;;;;;;;;;;;;;ACzNA;AACY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,QAAQ;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,WAAW;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB,YAAY;AACrC;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACvHA;AACY;AACZ;AACA,CAA+C;AAC/C;AACO,+DAA+D;AACtE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,4EAA4E;AACnF;AACA;AACA;AACA;AACA,0DAA0D,sDAAc;AACxE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sEAAsE;AACtE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uEAAuE;AACvE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC1MA;AACY;AACZ;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,gBAAgB;AAC3B;AACA;AACA;AACO;AACP;AACA;AACA;AACA,oBAAoB,mBAAmB;AACvC;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC;AACnC,oCAAoC;AACpC;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACzDA;AACY;AACZ;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,cAAc;AACd;AACO,iDAAiD;AACxD;AACA;AACA;AACA,eAAe,uBAAuB;AACtC;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA,WAAW,uBAAuB;AAClC,YAAY,gBAAgB;AAC5B,aAAa;AACb;AACO;AACP,eAAe,mBAAmB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,uBAAuB;AAClC,WAAW,QAAQ;AACnB,YAAY,WAAW;AACvB,aAAa;AACb;AACO;AACP,eAAe,kBAAkB;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;;;;;;;UCvEA;UACA;;UAEA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;;UAEA;UACA;;UAEA;UACA;UACA;;;;;WCtBA;WACA;WACA;WACA;WACA,yCAAyC,wCAAwC;WACjF;WACA;WACA;;;;;WCPA;;;;;WCAA;WACA;WACA;WACA,uDAAuD,iBAAiB;WACxE;WACA,gDAAgD,aAAa;WAC7D;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACNA;AACY;AACZ;AACA;AACmC;AACY;AACR;AACA;AACI;AAC8B;AACzE;AACA;AACkD;AACJ;AACF;AAC5C;AACA;AACoE;AAChB;AACE;AACQ;AACR;AACN;AACI;AACJ;AAChD,WAAW,eAAe;AACsC;AACJ;AACR;AACJ;AACwB;AACgB;AACpC;AACM;AACE;AACJ;AACN;AAClD;AACA;AACgD;AACY;AACM;AACI;AACpB;AACI;AACtD;AACA;AACqD;AACgB;AACA;AAOtC;AACqD;AAC3B;AACzD;AACA,YAAY,uBAAuB;AACE;AACL;AACE;AACK;AACvC;AACA,CAA+C;AACxC,2BAA2B,yDAAS;AAC3C;AACA;AACA,CAA+C;AAC/C,sDAAmB;AACnB;AACA;AACA;AACA;AACA,CAAC","sources":["webpack://gridviz/webpack/universalModuleDefinition","webpack://gridviz/./node_modules/d3-array/src/ascending.js","webpack://gridviz/./node_modules/d3-array/src/bisect.js","webpack://gridviz/./node_modules/d3-array/src/bisector.js","webpack://gridviz/./node_modules/d3-array/src/descending.js","webpack://gridviz/./node_modules/d3-array/src/extent.js","webpack://gridviz/./node_modules/d3-array/src/greatest.js","webpack://gridviz/./node_modules/d3-array/src/max.js","webpack://gridviz/./node_modules/d3-array/src/maxIndex.js","webpack://gridviz/./node_modules/d3-array/src/min.js","webpack://gridviz/./node_modules/d3-array/src/minIndex.js","webpack://gridviz/./node_modules/d3-array/src/number.js","webpack://gridviz/./node_modules/d3-array/src/permute.js","webpack://gridviz/./node_modules/d3-array/src/quantile.js","webpack://gridviz/./node_modules/d3-array/src/quickselect.js","webpack://gridviz/./node_modules/d3-array/src/sort.js","webpack://gridviz/./node_modules/d3-color/src/color.js","webpack://gridviz/./node_modules/d3-color/src/define.js","webpack://gridviz/./node_modules/d3-color/src/lab.js","webpack://gridviz/./node_modules/d3-color/src/math.js","webpack://gridviz/./node_modules/d3-dispatch/src/dispatch.js","webpack://gridviz/./node_modules/d3-drag/src/nodrag.js","webpack://gridviz/./node_modules/d3-drag/src/noevent.js","webpack://gridviz/./node_modules/d3-dsv/src/csv.js","webpack://gridviz/./node_modules/d3-dsv/src/dsv.js","webpack://gridviz/./node_modules/d3-dsv/src/tsv.js","webpack://gridviz/./node_modules/d3-ease/src/cubic.js","webpack://gridviz/./node_modules/d3-fetch/src/dsv.js","webpack://gridviz/./node_modules/d3-fetch/src/json.js","webpack://gridviz/./node_modules/d3-fetch/src/text.js","webpack://gridviz/./node_modules/d3-format/src/defaultLocale.js","webpack://gridviz/./node_modules/d3-format/src/exponent.js","webpack://gridviz/./node_modules/d3-format/src/formatDecimal.js","webpack://gridviz/./node_modules/d3-format/src/formatGroup.js","webpack://gridviz/./node_modules/d3-format/src/formatNumerals.js","webpack://gridviz/./node_modules/d3-format/src/formatPrefixAuto.js","webpack://gridviz/./node_modules/d3-format/src/formatRounded.js","webpack://gridviz/./node_modules/d3-format/src/formatSpecifier.js","webpack://gridviz/./node_modules/d3-format/src/formatTrim.js","webpack://gridviz/./node_modules/d3-format/src/formatTypes.js","webpack://gridviz/./node_modules/d3-format/src/identity.js","webpack://gridviz/./node_modules/d3-format/src/locale.js","webpack://gridviz/./node_modules/d3-interpolate/src/basis.js","webpack://gridviz/./node_modules/d3-interpolate/src/basisClosed.js","webpack://gridviz/./node_modules/d3-interpolate/src/color.js","webpack://gridviz/./node_modules/d3-interpolate/src/constant.js","webpack://gridviz/./node_modules/d3-interpolate/src/lab.js","webpack://gridviz/./node_modules/d3-interpolate/src/number.js","webpack://gridviz/./node_modules/d3-interpolate/src/rgb.js","webpack://gridviz/./node_modules/d3-interpolate/src/string.js","webpack://gridviz/./node_modules/d3-interpolate/src/transform/decompose.js","webpack://gridviz/./node_modules/d3-interpolate/src/transform/index.js","webpack://gridviz/./node_modules/d3-interpolate/src/transform/parse.js","webpack://gridviz/./node_modules/d3-interpolate/src/zoom.js","webpack://gridviz/./node_modules/d3-random/src/defaultSource.js","webpack://gridviz/./node_modules/d3-random/src/normal.js","webpack://gridviz/./node_modules/d3-scale/src/init.js","webpack://gridviz/./node_modules/d3-scale/src/quantile.js","webpack://gridviz/./node_modules/d3-selection/src/array.js","webpack://gridviz/./node_modules/d3-selection/src/constant.js","webpack://gridviz/./node_modules/d3-selection/src/creator.js","webpack://gridviz/./node_modules/d3-selection/src/matcher.js","webpack://gridviz/./node_modules/d3-selection/src/namespace.js","webpack://gridviz/./node_modules/d3-selection/src/namespaces.js","webpack://gridviz/./node_modules/d3-selection/src/pointer.js","webpack://gridviz/./node_modules/d3-selection/src/select.js","webpack://gridviz/./node_modules/d3-selection/src/selection/append.js","webpack://gridviz/./node_modules/d3-selection/src/selection/attr.js","webpack://gridviz/./node_modules/d3-selection/src/selection/call.js","webpack://gridviz/./node_modules/d3-selection/src/selection/classed.js","webpack://gridviz/./node_modules/d3-selection/src/selection/clone.js","webpack://gridviz/./node_modules/d3-selection/src/selection/data.js","webpack://gridviz/./node_modules/d3-selection/src/selection/datum.js","webpack://gridviz/./node_modules/d3-selection/src/selection/dispatch.js","webpack://gridviz/./node_modules/d3-selection/src/selection/each.js","webpack://gridviz/./node_modules/d3-selection/src/selection/empty.js","webpack://gridviz/./node_modules/d3-selection/src/selection/enter.js","webpack://gridviz/./node_modules/d3-selection/src/selection/exit.js","webpack://gridviz/./node_modules/d3-selection/src/selection/filter.js","webpack://gridviz/./node_modules/d3-selection/src/selection/html.js","webpack://gridviz/./node_modules/d3-selection/src/selection/index.js","webpack://gridviz/./node_modules/d3-selection/src/selection/insert.js","webpack://gridviz/./node_modules/d3-selection/src/selection/iterator.js","webpack://gridviz/./node_modules/d3-selection/src/selection/join.js","webpack://gridviz/./node_modules/d3-selection/src/selection/lower.js","webpack://gridviz/./node_modules/d3-selection/src/selection/merge.js","webpack://gridviz/./node_modules/d3-selection/src/selection/node.js","webpack://gridviz/./node_modules/d3-selection/src/selection/nodes.js","webpack://gridviz/./node_modules/d3-selection/src/selection/on.js","webpack://gridviz/./node_modules/d3-selection/src/selection/order.js","webpack://gridviz/./node_modules/d3-selection/src/selection/property.js","webpack://gridviz/./node_modules/d3-selection/src/selection/raise.js","webpack://gridviz/./node_modules/d3-selection/src/selection/remove.js","webpack://gridviz/./node_modules/d3-selection/src/selection/select.js","webpack://gridviz/./node_modules/d3-selection/src/selection/selectAll.js","webpack://gridviz/./node_modules/d3-selection/src/selection/selectChild.js","webpack://gridviz/./node_modules/d3-selection/src/selection/selectChildren.js","webpack://gridviz/./node_modules/d3-selection/src/selection/size.js","webpack://gridviz/./node_modules/d3-selection/src/selection/sort.js","webpack://gridviz/./node_modules/d3-selection/src/selection/sparse.js","webpack://gridviz/./node_modules/d3-selection/src/selection/style.js","webpack://gridviz/./node_modules/d3-selection/src/selection/text.js","webpack://gridviz/./node_modules/d3-selection/src/selector.js","webpack://gridviz/./node_modules/d3-selection/src/selectorAll.js","webpack://gridviz/./node_modules/d3-selection/src/sourceEvent.js","webpack://gridviz/./node_modules/d3-selection/src/window.js","webpack://gridviz/./node_modules/d3-timer/src/timeout.js","webpack://gridviz/./node_modules/d3-timer/src/timer.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/active.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/index.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/interrupt.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/selection/index.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/selection/interrupt.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/selection/transition.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/attr.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/attrTween.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/delay.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/duration.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/ease.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/easeVarying.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/end.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/filter.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/index.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/interpolate.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/merge.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/on.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/remove.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/schedule.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/select.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/selectAll.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/selection.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/style.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/styleTween.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/text.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/textTween.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/transition.js","webpack://gridviz/./node_modules/d3-zoom/node_modules/d3-transition/src/transition/tween.js","webpack://gridviz/./node_modules/d3-zoom/src/constant.js","webpack://gridviz/./node_modules/d3-zoom/src/event.js","webpack://gridviz/./node_modules/d3-zoom/src/index.js","webpack://gridviz/./node_modules/d3-zoom/src/noevent.js","webpack://gridviz/./node_modules/d3-zoom/src/transform.js","webpack://gridviz/./node_modules/d3-zoom/src/zoom.js","webpack://gridviz/./src/button/Button.js","webpack://gridviz/./src/button/FullscreenButton.js","webpack://gridviz/./src/button/ZoomButtons.js","webpack://gridviz/./src/core/Dataset.js","webpack://gridviz/./src/core/Drawable.js","webpack://gridviz/./src/core/GeoCanvas.js","webpack://gridviz/./src/core/Layer.js","webpack://gridviz/./src/core/Legend.js","webpack://gridviz/./src/core/Map.js","webpack://gridviz/./src/core/MultiResolutionDataset.js","webpack://gridviz/./src/core/Style.js","webpack://gridviz/./src/core/Tooltip.js","webpack://gridviz/./src/dataset/CSVGrid.js","webpack://gridviz/./src/dataset/JSGrid.js","webpack://gridviz/./src/dataset/TiledGrid.js","webpack://gridviz/./src/layer/BackgroundLayer.js","webpack://gridviz/./src/layer/BackgroundLayerImage.js","webpack://gridviz/./src/layer/BackgroundLayerWMS.js","webpack://gridviz/./src/layer/GeoJSONLayer.js","webpack://gridviz/./src/layer/GridLayer.js","webpack://gridviz/./src/layer/LabelLayer.js","webpack://gridviz/./src/legend/ColorCategoryLegend.js","webpack://gridviz/./src/legend/ColorDiscreteLegend.js","webpack://gridviz/./src/legend/ColorLegend.js","webpack://gridviz/./src/legend/OrientationLegend.js","webpack://gridviz/./src/legend/SizeLegend.js","webpack://gridviz/./src/legend/TernaryLegend.js","webpack://gridviz/./src/style/CompositionStyle.js","webpack://gridviz/./src/style/DotDensityStyle.js","webpack://gridviz/./src/style/ImageStyle.js","webpack://gridviz/./src/style/IsoFenceStyle.js","webpack://gridviz/./src/style/JoyPlotStyle.js","webpack://gridviz/./src/style/LegoStyle.js","webpack://gridviz/./src/style/MosaicStyle.js","webpack://gridviz/./src/style/NinjaStarStyle.js","webpack://gridviz/./src/style/PillarStyle.js","webpack://gridviz/./src/style/SegmentStyle.js","webpack://gridviz/./src/style/ShapeColorSizeStyle.js","webpack://gridviz/./src/style/SideCategoryStyle.js","webpack://gridviz/./src/style/SideStyle.js","webpack://gridviz/./src/style/SquareColorCategoryWebGLStyle.js","webpack://gridviz/./src/style/SquareColorWebGLStyle.js","webpack://gridviz/./src/style/StrokeStyle.js","webpack://gridviz/./src/style/TanakaStyle.js","webpack://gridviz/./src/style/TextStyle.js","webpack://gridviz/./src/style/TimeSeriesStyle.js","webpack://gridviz/./src/utils/WebGLSquareColoring.js","webpack://gridviz/./src/utils/WebGLSquareColoringAdvanced.js","webpack://gridviz/./src/utils/WebGLSquareColoringCatAdvanced.js","webpack://gridviz/./src/utils/scale.js","webpack://gridviz/./src/utils/stretching.js","webpack://gridviz/./src/utils/ternary.js","webpack://gridviz/./src/utils/utils.js","webpack://gridviz/./src/utils/webGLUtils.js","webpack://gridviz/webpack/bootstrap","webpack://gridviz/webpack/runtime/define property getters","webpack://gridviz/webpack/runtime/hasOwnProperty shorthand","webpack://gridviz/webpack/runtime/make namespace object","webpack://gridviz/./src/index.js"],"sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine([], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"gridviz\"] = factory();\n\telse\n\t\troot[\"gridviz\"] = factory();\n})(self, () => {\nreturn ","export default function ascending(a, b) {\n  return a == null || b == null ? NaN : a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;\n}\n","import ascending from \"./ascending.js\";\nimport bisector from \"./bisector.js\";\nimport number from \"./number.js\";\n\nconst ascendingBisect = bisector(ascending);\nexport const bisectRight = ascendingBisect.right;\nexport const bisectLeft = ascendingBisect.left;\nexport const bisectCenter = bisector(number).center;\nexport default bisectRight;\n","import ascending from \"./ascending.js\";\nimport descending from \"./descending.js\";\n\nexport default function bisector(f) {\n  let compare1, compare2, delta;\n\n  // If an accessor is specified, promote it to a comparator. In this case we\n  // can test whether the search value is (self-) comparable. We can’t do this\n  // for a comparator (except for specific, known comparators) because we can’t\n  // tell if the comparator is symmetric, and an asymmetric comparator can’t be\n  // used to test whether a single value is comparable.\n  if (f.length !== 2) {\n    compare1 = ascending;\n    compare2 = (d, x) => ascending(f(d), x);\n    delta = (d, x) => f(d) - x;\n  } else {\n    compare1 = f === ascending || f === descending ? f : zero;\n    compare2 = f;\n    delta = f;\n  }\n\n  function left(a, x, lo = 0, hi = a.length) {\n    if (lo < hi) {\n      if (compare1(x, x) !== 0) return hi;\n      do {\n        const mid = (lo + hi) >>> 1;\n        if (compare2(a[mid], x) < 0) lo = mid + 1;\n        else hi = mid;\n      } while (lo < hi);\n    }\n    return lo;\n  }\n\n  function right(a, x, lo = 0, hi = a.length) {\n    if (lo < hi) {\n      if (compare1(x, x) !== 0) return hi;\n      do {\n        const mid = (lo + hi) >>> 1;\n        if (compare2(a[mid], x) <= 0) lo = mid + 1;\n        else hi = mid;\n      } while (lo < hi);\n    }\n    return lo;\n  }\n\n  function center(a, x, lo = 0, hi = a.length) {\n    const i = left(a, x, lo, hi - 1);\n    return i > lo && delta(a[i - 1], x) > -delta(a[i], x) ? i - 1 : i;\n  }\n\n  return {left, center, right};\n}\n\nfunction zero() {\n  return 0;\n}\n","export default function descending(a, b) {\n  return a == null || b == null ? NaN\n    : b < a ? -1\n    : b > a ? 1\n    : b >= a ? 0\n    : NaN;\n}\n","export default function extent(values, valueof) {\n  let min;\n  let max;\n  if (valueof === undefined) {\n    for (const value of values) {\n      if (value != null) {\n        if (min === undefined) {\n          if (value >= value) min = max = value;\n        } else {\n          if (min > value) min = value;\n          if (max < value) max = value;\n        }\n      }\n    }\n  } else {\n    let index = -1;\n    for (let value of values) {\n      if ((value = valueof(value, ++index, values)) != null) {\n        if (min === undefined) {\n          if (value >= value) min = max = value;\n        } else {\n          if (min > value) min = value;\n          if (max < value) max = value;\n        }\n      }\n    }\n  }\n  return [min, max];\n}\n","import ascending from \"./ascending.js\";\n\nexport default function greatest(values, compare = ascending) {\n  let max;\n  let defined = false;\n  if (compare.length === 1) {\n    let maxValue;\n    for (const element of values) {\n      const value = compare(element);\n      if (defined\n          ? ascending(value, maxValue) > 0\n          : ascending(value, value) === 0) {\n        max = element;\n        maxValue = value;\n        defined = true;\n      }\n    }\n  } else {\n    for (const value of values) {\n      if (defined\n          ? compare(value, max) > 0\n          : compare(value, value) === 0) {\n        max = value;\n        defined = true;\n      }\n    }\n  }\n  return max;\n}\n","export default function max(values, valueof) {\n  let max;\n  if (valueof === undefined) {\n    for (const value of values) {\n      if (value != null\n          && (max < value || (max === undefined && value >= value))) {\n        max = value;\n      }\n    }\n  } else {\n    let index = -1;\n    for (let value of values) {\n      if ((value = valueof(value, ++index, values)) != null\n          && (max < value || (max === undefined && value >= value))) {\n        max = value;\n      }\n    }\n  }\n  return max;\n}\n","export default function maxIndex(values, valueof) {\n  let max;\n  let maxIndex = -1;\n  let index = -1;\n  if (valueof === undefined) {\n    for (const value of values) {\n      ++index;\n      if (value != null\n          && (max < value || (max === undefined && value >= value))) {\n        max = value, maxIndex = index;\n      }\n    }\n  } else {\n    for (let value of values) {\n      if ((value = valueof(value, ++index, values)) != null\n          && (max < value || (max === undefined && value >= value))) {\n        max = value, maxIndex = index;\n      }\n    }\n  }\n  return maxIndex;\n}\n","export default function min(values, valueof) {\n  let min;\n  if (valueof === undefined) {\n    for (const value of values) {\n      if (value != null\n          && (min > value || (min === undefined && value >= value))) {\n        min = value;\n      }\n    }\n  } else {\n    let index = -1;\n    for (let value of values) {\n      if ((value = valueof(value, ++index, values)) != null\n          && (min > value || (min === undefined && value >= value))) {\n        min = value;\n      }\n    }\n  }\n  return min;\n}\n","export default function minIndex(values, valueof) {\n  let min;\n  let minIndex = -1;\n  let index = -1;\n  if (valueof === undefined) {\n    for (const value of values) {\n      ++index;\n      if (value != null\n          && (min > value || (min === undefined && value >= value))) {\n        min = value, minIndex = index;\n      }\n    }\n  } else {\n    for (let value of values) {\n      if ((value = valueof(value, ++index, values)) != null\n          && (min > value || (min === undefined && value >= value))) {\n        min = value, minIndex = index;\n      }\n    }\n  }\n  return minIndex;\n}\n","export default function number(x) {\n  return x === null ? NaN : +x;\n}\n\nexport function* numbers(values, valueof) {\n  if (valueof === undefined) {\n    for (let value of values) {\n      if (value != null && (value = +value) >= value) {\n        yield value;\n      }\n    }\n  } else {\n    let index = -1;\n    for (let value of values) {\n      if ((value = valueof(value, ++index, values)) != null && (value = +value) >= value) {\n        yield value;\n      }\n    }\n  }\n}\n","export default function permute(source, keys) {\n  return Array.from(keys, key => source[key]);\n}\n","import max from \"./max.js\";\nimport maxIndex from \"./maxIndex.js\";\nimport min from \"./min.js\";\nimport minIndex from \"./minIndex.js\";\nimport quickselect from \"./quickselect.js\";\nimport number, {numbers} from \"./number.js\";\nimport {ascendingDefined} from \"./sort.js\";\nimport greatest from \"./greatest.js\";\n\nexport default function quantile(values, p, valueof) {\n  values = Float64Array.from(numbers(values, valueof));\n  if (!(n = values.length) || isNaN(p = +p)) return;\n  if (p <= 0 || n < 2) return min(values);\n  if (p >= 1) return max(values);\n  var n,\n      i = (n - 1) * p,\n      i0 = Math.floor(i),\n      value0 = max(quickselect(values, i0).subarray(0, i0 + 1)),\n      value1 = min(values.subarray(i0 + 1));\n  return value0 + (value1 - value0) * (i - i0);\n}\n\nexport function quantileSorted(values, p, valueof = number) {\n  if (!(n = values.length) || isNaN(p = +p)) return;\n  if (p <= 0 || n < 2) return +valueof(values[0], 0, values);\n  if (p >= 1) return +valueof(values[n - 1], n - 1, values);\n  var n,\n      i = (n - 1) * p,\n      i0 = Math.floor(i),\n      value0 = +valueof(values[i0], i0, values),\n      value1 = +valueof(values[i0 + 1], i0 + 1, values);\n  return value0 + (value1 - value0) * (i - i0);\n}\n\nexport function quantileIndex(values, p, valueof = number) {\n  if (isNaN(p = +p)) return;\n  numbers = Float64Array.from(values, (_, i) => number(valueof(values[i], i, values)));\n  if (p <= 0) return minIndex(numbers);\n  if (p >= 1) return maxIndex(numbers);\n  var numbers,\n      index = Uint32Array.from(values, (_, i) => i),\n      j = numbers.length - 1,\n      i = Math.floor(j * p);\n  quickselect(index, i, 0, j, (i, j) => ascendingDefined(numbers[i], numbers[j]));\n  i = greatest(index.subarray(0, i + 1), (i) => numbers[i]);\n  return i >= 0 ? i : -1;\n}\n","import {ascendingDefined, compareDefined} from \"./sort.js\";\n\n// Based on https://github.com/mourner/quickselect\n// ISC license, Copyright 2018 Vladimir Agafonkin.\nexport default function quickselect(array, k, left = 0, right = Infinity, compare) {\n  k = Math.floor(k);\n  left = Math.floor(Math.max(0, left));\n  right = Math.floor(Math.min(array.length - 1, right));\n\n  if (!(left <= k && k <= right)) return array;\n\n  compare = compare === undefined ? ascendingDefined : compareDefined(compare);\n\n  while (right > left) {\n    if (right - left > 600) {\n      const n = right - left + 1;\n      const m = k - left + 1;\n      const z = Math.log(n);\n      const s = 0.5 * Math.exp(2 * z / 3);\n      const sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);\n      const newLeft = Math.max(left, Math.floor(k - m * s / n + sd));\n      const newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));\n      quickselect(array, k, newLeft, newRight, compare);\n    }\n\n    const t = array[k];\n    let i = left;\n    let j = right;\n\n    swap(array, left, k);\n    if (compare(array[right], t) > 0) swap(array, left, right);\n\n    while (i < j) {\n      swap(array, i, j), ++i, --j;\n      while (compare(array[i], t) < 0) ++i;\n      while (compare(array[j], t) > 0) --j;\n    }\n\n    if (compare(array[left], t) === 0) swap(array, left, j);\n    else ++j, swap(array, j, right);\n\n    if (j <= k) left = j + 1;\n    if (k <= j) right = j - 1;\n  }\n\n  return array;\n}\n\nfunction swap(array, i, j) {\n  const t = array[i];\n  array[i] = array[j];\n  array[j] = t;\n}\n","import ascending from \"./ascending.js\";\nimport permute from \"./permute.js\";\n\nexport default function sort(values, ...F) {\n  if (typeof values[Symbol.iterator] !== \"function\") throw new TypeError(\"values is not iterable\");\n  values = Array.from(values);\n  let [f] = F;\n  if ((f && f.length !== 2) || F.length > 1) {\n    const index = Uint32Array.from(values, (d, i) => i);\n    if (F.length > 1) {\n      F = F.map(f => values.map(f));\n      index.sort((i, j) => {\n        for (const f of F) {\n          const c = ascendingDefined(f[i], f[j]);\n          if (c) return c;\n        }\n      });\n    } else {\n      f = values.map(f);\n      index.sort((i, j) => ascendingDefined(f[i], f[j]));\n    }\n    return permute(values, index);\n  }\n  return values.sort(compareDefined(f));\n}\n\nexport function compareDefined(compare = ascending) {\n  if (compare === ascending) return ascendingDefined;\n  if (typeof compare !== \"function\") throw new TypeError(\"compare is not a function\");\n  return (a, b) => {\n    const x = compare(a, b);\n    if (x || x === 0) return x;\n    return (compare(b, b) === 0) - (compare(a, a) === 0);\n  };\n}\n\nexport function ascendingDefined(a, b) {\n  return (a == null || !(a >= a)) - (b == null || !(b >= b)) || (a < b ? -1 : a > b ? 1 : 0);\n}\n","import define, {extend} from \"./define.js\";\n\nexport function Color() {}\n\nexport var darker = 0.7;\nexport var brighter = 1 / darker;\n\nvar reI = \"\\\\s*([+-]?\\\\d+)\\\\s*\",\n    reN = \"\\\\s*([+-]?(?:\\\\d*\\\\.)?\\\\d+(?:[eE][+-]?\\\\d+)?)\\\\s*\",\n    reP = \"\\\\s*([+-]?(?:\\\\d*\\\\.)?\\\\d+(?:[eE][+-]?\\\\d+)?)%\\\\s*\",\n    reHex = /^#([0-9a-f]{3,8})$/,\n    reRgbInteger = new RegExp(`^rgb\\\\(${reI},${reI},${reI}\\\\)$`),\n    reRgbPercent = new RegExp(`^rgb\\\\(${reP},${reP},${reP}\\\\)$`),\n    reRgbaInteger = new RegExp(`^rgba\\\\(${reI},${reI},${reI},${reN}\\\\)$`),\n    reRgbaPercent = new RegExp(`^rgba\\\\(${reP},${reP},${reP},${reN}\\\\)$`),\n    reHslPercent = new RegExp(`^hsl\\\\(${reN},${reP},${reP}\\\\)$`),\n    reHslaPercent = new RegExp(`^hsla\\\\(${reN},${reP},${reP},${reN}\\\\)$`);\n\nvar named = {\n  aliceblue: 0xf0f8ff,\n  antiquewhite: 0xfaebd7,\n  aqua: 0x00ffff,\n  aquamarine: 0x7fffd4,\n  azure: 0xf0ffff,\n  beige: 0xf5f5dc,\n  bisque: 0xffe4c4,\n  black: 0x000000,\n  blanchedalmond: 0xffebcd,\n  blue: 0x0000ff,\n  blueviolet: 0x8a2be2,\n  brown: 0xa52a2a,\n  burlywood: 0xdeb887,\n  cadetblue: 0x5f9ea0,\n  chartreuse: 0x7fff00,\n  chocolate: 0xd2691e,\n  coral: 0xff7f50,\n  cornflowerblue: 0x6495ed,\n  cornsilk: 0xfff8dc,\n  crimson: 0xdc143c,\n  cyan: 0x00ffff,\n  darkblue: 0x00008b,\n  darkcyan: 0x008b8b,\n  darkgoldenrod: 0xb8860b,\n  darkgray: 0xa9a9a9,\n  darkgreen: 0x006400,\n  darkgrey: 0xa9a9a9,\n  darkkhaki: 0xbdb76b,\n  darkmagenta: 0x8b008b,\n  darkolivegreen: 0x556b2f,\n  darkorange: 0xff8c00,\n  darkorchid: 0x9932cc,\n  darkred: 0x8b0000,\n  darksalmon: 0xe9967a,\n  darkseagreen: 0x8fbc8f,\n  darkslateblue: 0x483d8b,\n  darkslategray: 0x2f4f4f,\n  darkslategrey: 0x2f4f4f,\n  darkturquoise: 0x00ced1,\n  darkviolet: 0x9400d3,\n  deeppink: 0xff1493,\n  deepskyblue: 0x00bfff,\n  dimgray: 0x696969,\n  dimgrey: 0x696969,\n  dodgerblue: 0x1e90ff,\n  firebrick: 0xb22222,\n  floralwhite: 0xfffaf0,\n  forestgreen: 0x228b22,\n  fuchsia: 0xff00ff,\n  gainsboro: 0xdcdcdc,\n  ghostwhite: 0xf8f8ff,\n  gold: 0xffd700,\n  goldenrod: 0xdaa520,\n  gray: 0x808080,\n  green: 0x008000,\n  greenyellow: 0xadff2f,\n  grey: 0x808080,\n  honeydew: 0xf0fff0,\n  hotpink: 0xff69b4,\n  indianred: 0xcd5c5c,\n  indigo: 0x4b0082,\n  ivory: 0xfffff0,\n  khaki: 0xf0e68c,\n  lavender: 0xe6e6fa,\n  lavenderblush: 0xfff0f5,\n  lawngreen: 0x7cfc00,\n  lemonchiffon: 0xfffacd,\n  lightblue: 0xadd8e6,\n  lightcoral: 0xf08080,\n  lightcyan: 0xe0ffff,\n  lightgoldenrodyellow: 0xfafad2,\n  lightgray: 0xd3d3d3,\n  lightgreen: 0x90ee90,\n  lightgrey: 0xd3d3d3,\n  lightpink: 0xffb6c1,\n  lightsalmon: 0xffa07a,\n  lightseagreen: 0x20b2aa,\n  lightskyblue: 0x87cefa,\n  lightslategray: 0x778899,\n  lightslategrey: 0x778899,\n  lightsteelblue: 0xb0c4de,\n  lightyellow: 0xffffe0,\n  lime: 0x00ff00,\n  limegreen: 0x32cd32,\n  linen: 0xfaf0e6,\n  magenta: 0xff00ff,\n  maroon: 0x800000,\n  mediumaquamarine: 0x66cdaa,\n  mediumblue: 0x0000cd,\n  mediumorchid: 0xba55d3,\n  mediumpurple: 0x9370db,\n  mediumseagreen: 0x3cb371,\n  mediumslateblue: 0x7b68ee,\n  mediumspringgreen: 0x00fa9a,\n  mediumturquoise: 0x48d1cc,\n  mediumvioletred: 0xc71585,\n  midnightblue: 0x191970,\n  mintcream: 0xf5fffa,\n  mistyrose: 0xffe4e1,\n  moccasin: 0xffe4b5,\n  navajowhite: 0xffdead,\n  navy: 0x000080,\n  oldlace: 0xfdf5e6,\n  olive: 0x808000,\n  olivedrab: 0x6b8e23,\n  orange: 0xffa500,\n  orangered: 0xff4500,\n  orchid: 0xda70d6,\n  palegoldenrod: 0xeee8aa,\n  palegreen: 0x98fb98,\n  paleturquoise: 0xafeeee,\n  palevioletred: 0xdb7093,\n  papayawhip: 0xffefd5,\n  peachpuff: 0xffdab9,\n  peru: 0xcd853f,\n  pink: 0xffc0cb,\n  plum: 0xdda0dd,\n  powderblue: 0xb0e0e6,\n  purple: 0x800080,\n  rebeccapurple: 0x663399,\n  red: 0xff0000,\n  rosybrown: 0xbc8f8f,\n  royalblue: 0x4169e1,\n  saddlebrown: 0x8b4513,\n  salmon: 0xfa8072,\n  sandybrown: 0xf4a460,\n  seagreen: 0x2e8b57,\n  seashell: 0xfff5ee,\n  sienna: 0xa0522d,\n  silver: 0xc0c0c0,\n  skyblue: 0x87ceeb,\n  slateblue: 0x6a5acd,\n  slategray: 0x708090,\n  slategrey: 0x708090,\n  snow: 0xfffafa,\n  springgreen: 0x00ff7f,\n  steelblue: 0x4682b4,\n  tan: 0xd2b48c,\n  teal: 0x008080,\n  thistle: 0xd8bfd8,\n  tomato: 0xff6347,\n  turquoise: 0x40e0d0,\n  violet: 0xee82ee,\n  wheat: 0xf5deb3,\n  white: 0xffffff,\n  whitesmoke: 0xf5f5f5,\n  yellow: 0xffff00,\n  yellowgreen: 0x9acd32\n};\n\ndefine(Color, color, {\n  copy(channels) {\n    return Object.assign(new this.constructor, this, channels);\n  },\n  displayable() {\n    return this.rgb().displayable();\n  },\n  hex: color_formatHex, // Deprecated! Use color.formatHex.\n  formatHex: color_formatHex,\n  formatHex8: color_formatHex8,\n  formatHsl: color_formatHsl,\n  formatRgb: color_formatRgb,\n  toString: color_formatRgb\n});\n\nfunction color_formatHex() {\n  return this.rgb().formatHex();\n}\n\nfunction color_formatHex8() {\n  return this.rgb().formatHex8();\n}\n\nfunction color_formatHsl() {\n  return hslConvert(this).formatHsl();\n}\n\nfunction color_formatRgb() {\n  return this.rgb().formatRgb();\n}\n\nexport default function color(format) {\n  var m, l;\n  format = (format + \"\").trim().toLowerCase();\n  return (m = reHex.exec(format)) ? (l = m[1].length, m = parseInt(m[1], 16), l === 6 ? rgbn(m) // #ff0000\n      : l === 3 ? new Rgb((m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), ((m & 0xf) << 4) | (m & 0xf), 1) // #f00\n      : l === 8 ? rgba(m >> 24 & 0xff, m >> 16 & 0xff, m >> 8 & 0xff, (m & 0xff) / 0xff) // #ff000000\n      : l === 4 ? rgba((m >> 12 & 0xf) | (m >> 8 & 0xf0), (m >> 8 & 0xf) | (m >> 4 & 0xf0), (m >> 4 & 0xf) | (m & 0xf0), (((m & 0xf) << 4) | (m & 0xf)) / 0xff) // #f000\n      : null) // invalid hex\n      : (m = reRgbInteger.exec(format)) ? new Rgb(m[1], m[2], m[3], 1) // rgb(255, 0, 0)\n      : (m = reRgbPercent.exec(format)) ? new Rgb(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) // rgb(100%, 0%, 0%)\n      : (m = reRgbaInteger.exec(format)) ? rgba(m[1], m[2], m[3], m[4]) // rgba(255, 0, 0, 1)\n      : (m = reRgbaPercent.exec(format)) ? rgba(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) // rgb(100%, 0%, 0%, 1)\n      : (m = reHslPercent.exec(format)) ? hsla(m[1], m[2] / 100, m[3] / 100, 1) // hsl(120, 50%, 50%)\n      : (m = reHslaPercent.exec(format)) ? hsla(m[1], m[2] / 100, m[3] / 100, m[4]) // hsla(120, 50%, 50%, 1)\n      : named.hasOwnProperty(format) ? rgbn(named[format]) // eslint-disable-line no-prototype-builtins\n      : format === \"transparent\" ? new Rgb(NaN, NaN, NaN, 0)\n      : null;\n}\n\nfunction rgbn(n) {\n  return new Rgb(n >> 16 & 0xff, n >> 8 & 0xff, n & 0xff, 1);\n}\n\nfunction rgba(r, g, b, a) {\n  if (a <= 0) r = g = b = NaN;\n  return new Rgb(r, g, b, a);\n}\n\nexport function rgbConvert(o) {\n  if (!(o instanceof Color)) o = color(o);\n  if (!o) return new Rgb;\n  o = o.rgb();\n  return new Rgb(o.r, o.g, o.b, o.opacity);\n}\n\nexport function rgb(r, g, b, opacity) {\n  return arguments.length === 1 ? rgbConvert(r) : new Rgb(r, g, b, opacity == null ? 1 : opacity);\n}\n\nexport function Rgb(r, g, b, opacity) {\n  this.r = +r;\n  this.g = +g;\n  this.b = +b;\n  this.opacity = +opacity;\n}\n\ndefine(Rgb, rgb, extend(Color, {\n  brighter(k) {\n    k = k == null ? brighter : Math.pow(brighter, k);\n    return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);\n  },\n  darker(k) {\n    k = k == null ? darker : Math.pow(darker, k);\n    return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);\n  },\n  rgb() {\n    return this;\n  },\n  clamp() {\n    return new Rgb(clampi(this.r), clampi(this.g), clampi(this.b), clampa(this.opacity));\n  },\n  displayable() {\n    return (-0.5 <= this.r && this.r < 255.5)\n        && (-0.5 <= this.g && this.g < 255.5)\n        && (-0.5 <= this.b && this.b < 255.5)\n        && (0 <= this.opacity && this.opacity <= 1);\n  },\n  hex: rgb_formatHex, // Deprecated! Use color.formatHex.\n  formatHex: rgb_formatHex,\n  formatHex8: rgb_formatHex8,\n  formatRgb: rgb_formatRgb,\n  toString: rgb_formatRgb\n}));\n\nfunction rgb_formatHex() {\n  return `#${hex(this.r)}${hex(this.g)}${hex(this.b)}`;\n}\n\nfunction rgb_formatHex8() {\n  return `#${hex(this.r)}${hex(this.g)}${hex(this.b)}${hex((isNaN(this.opacity) ? 1 : this.opacity) * 255)}`;\n}\n\nfunction rgb_formatRgb() {\n  const a = clampa(this.opacity);\n  return `${a === 1 ? \"rgb(\" : \"rgba(\"}${clampi(this.r)}, ${clampi(this.g)}, ${clampi(this.b)}${a === 1 ? \")\" : `, ${a})`}`;\n}\n\nfunction clampa(opacity) {\n  return isNaN(opacity) ? 1 : Math.max(0, Math.min(1, opacity));\n}\n\nfunction clampi(value) {\n  return Math.max(0, Math.min(255, Math.round(value) || 0));\n}\n\nfunction hex(value) {\n  value = clampi(value);\n  return (value < 16 ? \"0\" : \"\") + value.toString(16);\n}\n\nfunction hsla(h, s, l, a) {\n  if (a <= 0) h = s = l = NaN;\n  else if (l <= 0 || l >= 1) h = s = NaN;\n  else if (s <= 0) h = NaN;\n  return new Hsl(h, s, l, a);\n}\n\nexport function hslConvert(o) {\n  if (o instanceof Hsl) return new Hsl(o.h, o.s, o.l, o.opacity);\n  if (!(o instanceof Color)) o = color(o);\n  if (!o) return new Hsl;\n  if (o instanceof Hsl) return o;\n  o = o.rgb();\n  var r = o.r / 255,\n      g = o.g / 255,\n      b = o.b / 255,\n      min = Math.min(r, g, b),\n      max = Math.max(r, g, b),\n      h = NaN,\n      s = max - min,\n      l = (max + min) / 2;\n  if (s) {\n    if (r === max) h = (g - b) / s + (g < b) * 6;\n    else if (g === max) h = (b - r) / s + 2;\n    else h = (r - g) / s + 4;\n    s /= l < 0.5 ? max + min : 2 - max - min;\n    h *= 60;\n  } else {\n    s = l > 0 && l < 1 ? 0 : h;\n  }\n  return new Hsl(h, s, l, o.opacity);\n}\n\nexport function hsl(h, s, l, opacity) {\n  return arguments.length === 1 ? hslConvert(h) : new Hsl(h, s, l, opacity == null ? 1 : opacity);\n}\n\nfunction Hsl(h, s, l, opacity) {\n  this.h = +h;\n  this.s = +s;\n  this.l = +l;\n  this.opacity = +opacity;\n}\n\ndefine(Hsl, hsl, extend(Color, {\n  brighter(k) {\n    k = k == null ? brighter : Math.pow(brighter, k);\n    return new Hsl(this.h, this.s, this.l * k, this.opacity);\n  },\n  darker(k) {\n    k = k == null ? darker : Math.pow(darker, k);\n    return new Hsl(this.h, this.s, this.l * k, this.opacity);\n  },\n  rgb() {\n    var h = this.h % 360 + (this.h < 0) * 360,\n        s = isNaN(h) || isNaN(this.s) ? 0 : this.s,\n        l = this.l,\n        m2 = l + (l < 0.5 ? l : 1 - l) * s,\n        m1 = 2 * l - m2;\n    return new Rgb(\n      hsl2rgb(h >= 240 ? h - 240 : h + 120, m1, m2),\n      hsl2rgb(h, m1, m2),\n      hsl2rgb(h < 120 ? h + 240 : h - 120, m1, m2),\n      this.opacity\n    );\n  },\n  clamp() {\n    return new Hsl(clamph(this.h), clampt(this.s), clampt(this.l), clampa(this.opacity));\n  },\n  displayable() {\n    return (0 <= this.s && this.s <= 1 || isNaN(this.s))\n        && (0 <= this.l && this.l <= 1)\n        && (0 <= this.opacity && this.opacity <= 1);\n  },\n  formatHsl() {\n    const a = clampa(this.opacity);\n    return `${a === 1 ? \"hsl(\" : \"hsla(\"}${clamph(this.h)}, ${clampt(this.s) * 100}%, ${clampt(this.l) * 100}%${a === 1 ? \")\" : `, ${a})`}`;\n  }\n}));\n\nfunction clamph(value) {\n  value = (value || 0) % 360;\n  return value < 0 ? value + 360 : value;\n}\n\nfunction clampt(value) {\n  return Math.max(0, Math.min(1, value || 0));\n}\n\n/* From FvD 13.37, CSS Color Module Level 3 */\nfunction hsl2rgb(h, m1, m2) {\n  return (h < 60 ? m1 + (m2 - m1) * h / 60\n      : h < 180 ? m2\n      : h < 240 ? m1 + (m2 - m1) * (240 - h) / 60\n      : m1) * 255;\n}\n","export default function(constructor, factory, prototype) {\n  constructor.prototype = factory.prototype = prototype;\n  prototype.constructor = constructor;\n}\n\nexport function extend(parent, definition) {\n  var prototype = Object.create(parent.prototype);\n  for (var key in definition) prototype[key] = definition[key];\n  return prototype;\n}\n","import define, {extend} from \"./define.js\";\nimport {Color, rgbConvert, Rgb} from \"./color.js\";\nimport {degrees, radians} from \"./math.js\";\n\n// https://observablehq.com/@mbostock/lab-and-rgb\nconst K = 18,\n    Xn = 0.96422,\n    Yn = 1,\n    Zn = 0.82521,\n    t0 = 4 / 29,\n    t1 = 6 / 29,\n    t2 = 3 * t1 * t1,\n    t3 = t1 * t1 * t1;\n\nfunction labConvert(o) {\n  if (o instanceof Lab) return new Lab(o.l, o.a, o.b, o.opacity);\n  if (o instanceof Hcl) return hcl2lab(o);\n  if (!(o instanceof Rgb)) o = rgbConvert(o);\n  var r = rgb2lrgb(o.r),\n      g = rgb2lrgb(o.g),\n      b = rgb2lrgb(o.b),\n      y = xyz2lab((0.2225045 * r + 0.7168786 * g + 0.0606169 * b) / Yn), x, z;\n  if (r === g && g === b) x = z = y; else {\n    x = xyz2lab((0.4360747 * r + 0.3850649 * g + 0.1430804 * b) / Xn);\n    z = xyz2lab((0.0139322 * r + 0.0971045 * g + 0.7141733 * b) / Zn);\n  }\n  return new Lab(116 * y - 16, 500 * (x - y), 200 * (y - z), o.opacity);\n}\n\nexport function gray(l, opacity) {\n  return new Lab(l, 0, 0, opacity == null ? 1 : opacity);\n}\n\nexport default function lab(l, a, b, opacity) {\n  return arguments.length === 1 ? labConvert(l) : new Lab(l, a, b, opacity == null ? 1 : opacity);\n}\n\nexport function Lab(l, a, b, opacity) {\n  this.l = +l;\n  this.a = +a;\n  this.b = +b;\n  this.opacity = +opacity;\n}\n\ndefine(Lab, lab, extend(Color, {\n  brighter(k) {\n    return new Lab(this.l + K * (k == null ? 1 : k), this.a, this.b, this.opacity);\n  },\n  darker(k) {\n    return new Lab(this.l - K * (k == null ? 1 : k), this.a, this.b, this.opacity);\n  },\n  rgb() {\n    var y = (this.l + 16) / 116,\n        x = isNaN(this.a) ? y : y + this.a / 500,\n        z = isNaN(this.b) ? y : y - this.b / 200;\n    x = Xn * lab2xyz(x);\n    y = Yn * lab2xyz(y);\n    z = Zn * lab2xyz(z);\n    return new Rgb(\n      lrgb2rgb( 3.1338561 * x - 1.6168667 * y - 0.4906146 * z),\n      lrgb2rgb(-0.9787684 * x + 1.9161415 * y + 0.0334540 * z),\n      lrgb2rgb( 0.0719453 * x - 0.2289914 * y + 1.4052427 * z),\n      this.opacity\n    );\n  }\n}));\n\nfunction xyz2lab(t) {\n  return t > t3 ? Math.pow(t, 1 / 3) : t / t2 + t0;\n}\n\nfunction lab2xyz(t) {\n  return t > t1 ? t * t * t : t2 * (t - t0);\n}\n\nfunction lrgb2rgb(x) {\n  return 255 * (x <= 0.0031308 ? 12.92 * x : 1.055 * Math.pow(x, 1 / 2.4) - 0.055);\n}\n\nfunction rgb2lrgb(x) {\n  return (x /= 255) <= 0.04045 ? x / 12.92 : Math.pow((x + 0.055) / 1.055, 2.4);\n}\n\nfunction hclConvert(o) {\n  if (o instanceof Hcl) return new Hcl(o.h, o.c, o.l, o.opacity);\n  if (!(o instanceof Lab)) o = labConvert(o);\n  if (o.a === 0 && o.b === 0) return new Hcl(NaN, 0 < o.l && o.l < 100 ? 0 : NaN, o.l, o.opacity);\n  var h = Math.atan2(o.b, o.a) * degrees;\n  return new Hcl(h < 0 ? h + 360 : h, Math.sqrt(o.a * o.a + o.b * o.b), o.l, o.opacity);\n}\n\nexport function lch(l, c, h, opacity) {\n  return arguments.length === 1 ? hclConvert(l) : new Hcl(h, c, l, opacity == null ? 1 : opacity);\n}\n\nexport function hcl(h, c, l, opacity) {\n  return arguments.length === 1 ? hclConvert(h) : new Hcl(h, c, l, opacity == null ? 1 : opacity);\n}\n\nexport function Hcl(h, c, l, opacity) {\n  this.h = +h;\n  this.c = +c;\n  this.l = +l;\n  this.opacity = +opacity;\n}\n\nfunction hcl2lab(o) {\n  if (isNaN(o.h)) return new Lab(o.l, 0, 0, o.opacity);\n  var h = o.h * radians;\n  return new Lab(o.l, Math.cos(h) * o.c, Math.sin(h) * o.c, o.opacity);\n}\n\ndefine(Hcl, hcl, extend(Color, {\n  brighter(k) {\n    return new Hcl(this.h, this.c, this.l + K * (k == null ? 1 : k), this.opacity);\n  },\n  darker(k) {\n    return new Hcl(this.h, this.c, this.l - K * (k == null ? 1 : k), this.opacity);\n  },\n  rgb() {\n    return hcl2lab(this).rgb();\n  }\n}));\n","export const radians = Math.PI / 180;\nexport const degrees = 180 / Math.PI;\n","var noop = {value: () => {}};\n\nfunction dispatch() {\n  for (var i = 0, n = arguments.length, _ = {}, t; i < n; ++i) {\n    if (!(t = arguments[i] + \"\") || (t in _) || /[\\s.]/.test(t)) throw new Error(\"illegal type: \" + t);\n    _[t] = [];\n  }\n  return new Dispatch(_);\n}\n\nfunction Dispatch(_) {\n  this._ = _;\n}\n\nfunction parseTypenames(typenames, types) {\n  return typenames.trim().split(/^|\\s+/).map(function(t) {\n    var name = \"\", i = t.indexOf(\".\");\n    if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);\n    if (t && !types.hasOwnProperty(t)) throw new Error(\"unknown type: \" + t);\n    return {type: t, name: name};\n  });\n}\n\nDispatch.prototype = dispatch.prototype = {\n  constructor: Dispatch,\n  on: function(typename, callback) {\n    var _ = this._,\n        T = parseTypenames(typename + \"\", _),\n        t,\n        i = -1,\n        n = T.length;\n\n    // If no callback was specified, return the callback of the given type and name.\n    if (arguments.length < 2) {\n      while (++i < n) if ((t = (typename = T[i]).type) && (t = get(_[t], typename.name))) return t;\n      return;\n    }\n\n    // If a type was specified, set the callback for the given type and name.\n    // Otherwise, if a null callback was specified, remove callbacks of the given name.\n    if (callback != null && typeof callback !== \"function\") throw new Error(\"invalid callback: \" + callback);\n    while (++i < n) {\n      if (t = (typename = T[i]).type) _[t] = set(_[t], typename.name, callback);\n      else if (callback == null) for (t in _) _[t] = set(_[t], typename.name, null);\n    }\n\n    return this;\n  },\n  copy: function() {\n    var copy = {}, _ = this._;\n    for (var t in _) copy[t] = _[t].slice();\n    return new Dispatch(copy);\n  },\n  call: function(type, that) {\n    if ((n = arguments.length - 2) > 0) for (var args = new Array(n), i = 0, n, t; i < n; ++i) args[i] = arguments[i + 2];\n    if (!this._.hasOwnProperty(type)) throw new Error(\"unknown type: \" + type);\n    for (t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);\n  },\n  apply: function(type, that, args) {\n    if (!this._.hasOwnProperty(type)) throw new Error(\"unknown type: \" + type);\n    for (var t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);\n  }\n};\n\nfunction get(type, name) {\n  for (var i = 0, n = type.length, c; i < n; ++i) {\n    if ((c = type[i]).name === name) {\n      return c.value;\n    }\n  }\n}\n\nfunction set(type, name, callback) {\n  for (var i = 0, n = type.length; i < n; ++i) {\n    if (type[i].name === name) {\n      type[i] = noop, type = type.slice(0, i).concat(type.slice(i + 1));\n      break;\n    }\n  }\n  if (callback != null) type.push({name: name, value: callback});\n  return type;\n}\n\nexport default dispatch;\n","import {select} from \"d3-selection\";\nimport noevent, {nonpassivecapture} from \"./noevent.js\";\n\nexport default function(view) {\n  var root = view.document.documentElement,\n      selection = select(view).on(\"dragstart.drag\", noevent, nonpassivecapture);\n  if (\"onselectstart\" in root) {\n    selection.on(\"selectstart.drag\", noevent, nonpassivecapture);\n  } else {\n    root.__noselect = root.style.MozUserSelect;\n    root.style.MozUserSelect = \"none\";\n  }\n}\n\nexport function yesdrag(view, noclick) {\n  var root = view.document.documentElement,\n      selection = select(view).on(\"dragstart.drag\", null);\n  if (noclick) {\n    selection.on(\"click.drag\", noevent, nonpassivecapture);\n    setTimeout(function() { selection.on(\"click.drag\", null); }, 0);\n  }\n  if (\"onselectstart\" in root) {\n    selection.on(\"selectstart.drag\", null);\n  } else {\n    root.style.MozUserSelect = root.__noselect;\n    delete root.__noselect;\n  }\n}\n","// These are typically used in conjunction with noevent to ensure that we can\n// preventDefault on the event.\nexport const nonpassive = {passive: false};\nexport const nonpassivecapture = {capture: true, passive: false};\n\nexport function nopropagation(event) {\n  event.stopImmediatePropagation();\n}\n\nexport default function(event) {\n  event.preventDefault();\n  event.stopImmediatePropagation();\n}\n","import dsv from \"./dsv.js\";\n\nvar csv = dsv(\",\");\n\nexport var csvParse = csv.parse;\nexport var csvParseRows = csv.parseRows;\nexport var csvFormat = csv.format;\nexport var csvFormatBody = csv.formatBody;\nexport var csvFormatRows = csv.formatRows;\nexport var csvFormatRow = csv.formatRow;\nexport var csvFormatValue = csv.formatValue;\n","var EOL = {},\n    EOF = {},\n    QUOTE = 34,\n    NEWLINE = 10,\n    RETURN = 13;\n\nfunction objectConverter(columns) {\n  return new Function(\"d\", \"return {\" + columns.map(function(name, i) {\n    return JSON.stringify(name) + \": d[\" + i + \"] || \\\"\\\"\";\n  }).join(\",\") + \"}\");\n}\n\nfunction customConverter(columns, f) {\n  var object = objectConverter(columns);\n  return function(row, i) {\n    return f(object(row), i, columns);\n  };\n}\n\n// Compute unique columns in order of discovery.\nfunction inferColumns(rows) {\n  var columnSet = Object.create(null),\n      columns = [];\n\n  rows.forEach(function(row) {\n    for (var column in row) {\n      if (!(column in columnSet)) {\n        columns.push(columnSet[column] = column);\n      }\n    }\n  });\n\n  return columns;\n}\n\nfunction pad(value, width) {\n  var s = value + \"\", length = s.length;\n  return length < width ? new Array(width - length + 1).join(0) + s : s;\n}\n\nfunction formatYear(year) {\n  return year < 0 ? \"-\" + pad(-year, 6)\n    : year > 9999 ? \"+\" + pad(year, 6)\n    : pad(year, 4);\n}\n\nfunction formatDate(date) {\n  var hours = date.getUTCHours(),\n      minutes = date.getUTCMinutes(),\n      seconds = date.getUTCSeconds(),\n      milliseconds = date.getUTCMilliseconds();\n  return isNaN(date) ? \"Invalid Date\"\n      : formatYear(date.getUTCFullYear(), 4) + \"-\" + pad(date.getUTCMonth() + 1, 2) + \"-\" + pad(date.getUTCDate(), 2)\n      + (milliseconds ? \"T\" + pad(hours, 2) + \":\" + pad(minutes, 2) + \":\" + pad(seconds, 2) + \".\" + pad(milliseconds, 3) + \"Z\"\n      : seconds ? \"T\" + pad(hours, 2) + \":\" + pad(minutes, 2) + \":\" + pad(seconds, 2) + \"Z\"\n      : minutes || hours ? \"T\" + pad(hours, 2) + \":\" + pad(minutes, 2) + \"Z\"\n      : \"\");\n}\n\nexport default function(delimiter) {\n  var reFormat = new RegExp(\"[\\\"\" + delimiter + \"\\n\\r]\"),\n      DELIMITER = delimiter.charCodeAt(0);\n\n  function parse(text, f) {\n    var convert, columns, rows = parseRows(text, function(row, i) {\n      if (convert) return convert(row, i - 1);\n      columns = row, convert = f ? customConverter(row, f) : objectConverter(row);\n    });\n    rows.columns = columns || [];\n    return rows;\n  }\n\n  function parseRows(text, f) {\n    var rows = [], // output rows\n        N = text.length,\n        I = 0, // current character index\n        n = 0, // current line number\n        t, // current token\n        eof = N <= 0, // current token followed by EOF?\n        eol = false; // current token followed by EOL?\n\n    // Strip the trailing newline.\n    if (text.charCodeAt(N - 1) === NEWLINE) --N;\n    if (text.charCodeAt(N - 1) === RETURN) --N;\n\n    function token() {\n      if (eof) return EOF;\n      if (eol) return eol = false, EOL;\n\n      // Unescape quotes.\n      var i, j = I, c;\n      if (text.charCodeAt(j) === QUOTE) {\n        while (I++ < N && text.charCodeAt(I) !== QUOTE || text.charCodeAt(++I) === QUOTE);\n        if ((i = I) >= N) eof = true;\n        else if ((c = text.charCodeAt(I++)) === NEWLINE) eol = true;\n        else if (c === RETURN) { eol = true; if (text.charCodeAt(I) === NEWLINE) ++I; }\n        return text.slice(j + 1, i - 1).replace(/\"\"/g, \"\\\"\");\n      }\n\n      // Find next delimiter or newline.\n      while (I < N) {\n        if ((c = text.charCodeAt(i = I++)) === NEWLINE) eol = true;\n        else if (c === RETURN) { eol = true; if (text.charCodeAt(I) === NEWLINE) ++I; }\n        else if (c !== DELIMITER) continue;\n        return text.slice(j, i);\n      }\n\n      // Return last token before EOF.\n      return eof = true, text.slice(j, N);\n    }\n\n    while ((t = token()) !== EOF) {\n      var row = [];\n      while (t !== EOL && t !== EOF) row.push(t), t = token();\n      if (f && (row = f(row, n++)) == null) continue;\n      rows.push(row);\n    }\n\n    return rows;\n  }\n\n  function preformatBody(rows, columns) {\n    return rows.map(function(row) {\n      return columns.map(function(column) {\n        return formatValue(row[column]);\n      }).join(delimiter);\n    });\n  }\n\n  function format(rows, columns) {\n    if (columns == null) columns = inferColumns(rows);\n    return [columns.map(formatValue).join(delimiter)].concat(preformatBody(rows, columns)).join(\"\\n\");\n  }\n\n  function formatBody(rows, columns) {\n    if (columns == null) columns = inferColumns(rows);\n    return preformatBody(rows, columns).join(\"\\n\");\n  }\n\n  function formatRows(rows) {\n    return rows.map(formatRow).join(\"\\n\");\n  }\n\n  function formatRow(row) {\n    return row.map(formatValue).join(delimiter);\n  }\n\n  function formatValue(value) {\n    return value == null ? \"\"\n        : value instanceof Date ? formatDate(value)\n        : reFormat.test(value += \"\") ? \"\\\"\" + value.replace(/\"/g, \"\\\"\\\"\") + \"\\\"\"\n        : value;\n  }\n\n  return {\n    parse: parse,\n    parseRows: parseRows,\n    format: format,\n    formatBody: formatBody,\n    formatRows: formatRows,\n    formatRow: formatRow,\n    formatValue: formatValue\n  };\n}\n","import dsv from \"./dsv.js\";\n\nvar tsv = dsv(\"\\t\");\n\nexport var tsvParse = tsv.parse;\nexport var tsvParseRows = tsv.parseRows;\nexport var tsvFormat = tsv.format;\nexport var tsvFormatBody = tsv.formatBody;\nexport var tsvFormatRows = tsv.formatRows;\nexport var tsvFormatRow = tsv.formatRow;\nexport var tsvFormatValue = tsv.formatValue;\n","export function cubicIn(t) {\n  return t * t * t;\n}\n\nexport function cubicOut(t) {\n  return --t * t * t + 1;\n}\n\nexport function cubicInOut(t) {\n  return ((t *= 2) <= 1 ? t * t * t : (t -= 2) * t * t + 2) / 2;\n}\n","import {csvParse, dsvFormat, tsvParse} from \"d3-dsv\";\nimport text from \"./text.js\";\n\nfunction dsvParse(parse) {\n  return function(input, init, row) {\n    if (arguments.length === 2 && typeof init === \"function\") row = init, init = undefined;\n    return text(input, init).then(function(response) {\n      return parse(response, row);\n    });\n  };\n}\n\nexport default function dsv(delimiter, input, init, row) {\n  if (arguments.length === 3 && typeof init === \"function\") row = init, init = undefined;\n  var format = dsvFormat(delimiter);\n  return text(input, init).then(function(response) {\n    return format.parse(response, row);\n  });\n}\n\nexport var csv = dsvParse(csvParse);\nexport var tsv = dsvParse(tsvParse);\n","function responseJson(response) {\n  if (!response.ok) throw new Error(response.status + \" \" + response.statusText);\n  if (response.status === 204 || response.status === 205) return;\n  return response.json();\n}\n\nexport default function(input, init) {\n  return fetch(input, init).then(responseJson);\n}\n","function responseText(response) {\n  if (!response.ok) throw new Error(response.status + \" \" + response.statusText);\n  return response.text();\n}\n\nexport default function(input, init) {\n  return fetch(input, init).then(responseText);\n}\n","import formatLocale from \"./locale.js\";\n\nvar locale;\nexport var format;\nexport var formatPrefix;\n\ndefaultLocale({\n  thousands: \",\",\n  grouping: [3],\n  currency: [\"$\", \"\"]\n});\n\nexport default function defaultLocale(definition) {\n  locale = formatLocale(definition);\n  format = locale.format;\n  formatPrefix = locale.formatPrefix;\n  return locale;\n}\n","import {formatDecimalParts} from \"./formatDecimal.js\";\n\nexport default function(x) {\n  return x = formatDecimalParts(Math.abs(x)), x ? x[1] : NaN;\n}\n","export default function(x) {\n  return Math.abs(x = Math.round(x)) >= 1e21\n      ? x.toLocaleString(\"en\").replace(/,/g, \"\")\n      : x.toString(10);\n}\n\n// Computes the decimal coefficient and exponent of the specified number x with\n// significant digits p, where x is positive and p is in [1, 21] or undefined.\n// For example, formatDecimalParts(1.23) returns [\"123\", 0].\nexport function formatDecimalParts(x, p) {\n  if ((i = (x = p ? x.toExponential(p - 1) : x.toExponential()).indexOf(\"e\")) < 0) return null; // NaN, ±Infinity\n  var i, coefficient = x.slice(0, i);\n\n  // The string returned by toExponential either has the form \\d\\.\\d+e[-+]\\d+\n  // (e.g., 1.2e+3) or the form \\de[-+]\\d+ (e.g., 1e+3).\n  return [\n    coefficient.length > 1 ? coefficient[0] + coefficient.slice(2) : coefficient,\n    +x.slice(i + 1)\n  ];\n}\n","export default function(grouping, thousands) {\n  return function(value, width) {\n    var i = value.length,\n        t = [],\n        j = 0,\n        g = grouping[0],\n        length = 0;\n\n    while (i > 0 && g > 0) {\n      if (length + g + 1 > width) g = Math.max(1, width - length);\n      t.push(value.substring(i -= g, i + g));\n      if ((length += g + 1) > width) break;\n      g = grouping[j = (j + 1) % grouping.length];\n    }\n\n    return t.reverse().join(thousands);\n  };\n}\n","export default function(numerals) {\n  return function(value) {\n    return value.replace(/[0-9]/g, function(i) {\n      return numerals[+i];\n    });\n  };\n}\n","import {formatDecimalParts} from \"./formatDecimal.js\";\n\nexport var prefixExponent;\n\nexport default function(x, p) {\n  var d = formatDecimalParts(x, p);\n  if (!d) return x + \"\";\n  var coefficient = d[0],\n      exponent = d[1],\n      i = exponent - (prefixExponent = Math.max(-8, Math.min(8, Math.floor(exponent / 3))) * 3) + 1,\n      n = coefficient.length;\n  return i === n ? coefficient\n      : i > n ? coefficient + new Array(i - n + 1).join(\"0\")\n      : i > 0 ? coefficient.slice(0, i) + \".\" + coefficient.slice(i)\n      : \"0.\" + new Array(1 - i).join(\"0\") + formatDecimalParts(x, Math.max(0, p + i - 1))[0]; // less than 1y!\n}\n","import {formatDecimalParts} from \"./formatDecimal.js\";\n\nexport default function(x, p) {\n  var d = formatDecimalParts(x, p);\n  if (!d) return x + \"\";\n  var coefficient = d[0],\n      exponent = d[1];\n  return exponent < 0 ? \"0.\" + new Array(-exponent).join(\"0\") + coefficient\n      : coefficient.length > exponent + 1 ? coefficient.slice(0, exponent + 1) + \".\" + coefficient.slice(exponent + 1)\n      : coefficient + new Array(exponent - coefficient.length + 2).join(\"0\");\n}\n","// [[fill]align][sign][symbol][0][width][,][.precision][~][type]\nvar re = /^(?:(.)?([<>=^]))?([+\\-( ])?([$#])?(0)?(\\d+)?(,)?(\\.\\d+)?(~)?([a-z%])?$/i;\n\nexport default function formatSpecifier(specifier) {\n  if (!(match = re.exec(specifier))) throw new Error(\"invalid format: \" + specifier);\n  var match;\n  return new FormatSpecifier({\n    fill: match[1],\n    align: match[2],\n    sign: match[3],\n    symbol: match[4],\n    zero: match[5],\n    width: match[6],\n    comma: match[7],\n    precision: match[8] && match[8].slice(1),\n    trim: match[9],\n    type: match[10]\n  });\n}\n\nformatSpecifier.prototype = FormatSpecifier.prototype; // instanceof\n\nexport function FormatSpecifier(specifier) {\n  this.fill = specifier.fill === undefined ? \" \" : specifier.fill + \"\";\n  this.align = specifier.align === undefined ? \">\" : specifier.align + \"\";\n  this.sign = specifier.sign === undefined ? \"-\" : specifier.sign + \"\";\n  this.symbol = specifier.symbol === undefined ? \"\" : specifier.symbol + \"\";\n  this.zero = !!specifier.zero;\n  this.width = specifier.width === undefined ? undefined : +specifier.width;\n  this.comma = !!specifier.comma;\n  this.precision = specifier.precision === undefined ? undefined : +specifier.precision;\n  this.trim = !!specifier.trim;\n  this.type = specifier.type === undefined ? \"\" : specifier.type + \"\";\n}\n\nFormatSpecifier.prototype.toString = function() {\n  return this.fill\n      + this.align\n      + this.sign\n      + this.symbol\n      + (this.zero ? \"0\" : \"\")\n      + (this.width === undefined ? \"\" : Math.max(1, this.width | 0))\n      + (this.comma ? \",\" : \"\")\n      + (this.precision === undefined ? \"\" : \".\" + Math.max(0, this.precision | 0))\n      + (this.trim ? \"~\" : \"\")\n      + this.type;\n};\n","// Trims insignificant zeros, e.g., replaces 1.2000k with 1.2k.\nexport default function(s) {\n  out: for (var n = s.length, i = 1, i0 = -1, i1; i < n; ++i) {\n    switch (s[i]) {\n      case \".\": i0 = i1 = i; break;\n      case \"0\": if (i0 === 0) i0 = i; i1 = i; break;\n      default: if (!+s[i]) break out; if (i0 > 0) i0 = 0; break;\n    }\n  }\n  return i0 > 0 ? s.slice(0, i0) + s.slice(i1 + 1) : s;\n}\n","import formatDecimal from \"./formatDecimal.js\";\nimport formatPrefixAuto from \"./formatPrefixAuto.js\";\nimport formatRounded from \"./formatRounded.js\";\n\nexport default {\n  \"%\": (x, p) => (x * 100).toFixed(p),\n  \"b\": (x) => Math.round(x).toString(2),\n  \"c\": (x) => x + \"\",\n  \"d\": formatDecimal,\n  \"e\": (x, p) => x.toExponential(p),\n  \"f\": (x, p) => x.toFixed(p),\n  \"g\": (x, p) => x.toPrecision(p),\n  \"o\": (x) => Math.round(x).toString(8),\n  \"p\": (x, p) => formatRounded(x * 100, p),\n  \"r\": formatRounded,\n  \"s\": formatPrefixAuto,\n  \"X\": (x) => Math.round(x).toString(16).toUpperCase(),\n  \"x\": (x) => Math.round(x).toString(16)\n};\n","export default function(x) {\n  return x;\n}\n","import exponent from \"./exponent.js\";\nimport formatGroup from \"./formatGroup.js\";\nimport formatNumerals from \"./formatNumerals.js\";\nimport formatSpecifier from \"./formatSpecifier.js\";\nimport formatTrim from \"./formatTrim.js\";\nimport formatTypes from \"./formatTypes.js\";\nimport {prefixExponent} from \"./formatPrefixAuto.js\";\nimport identity from \"./identity.js\";\n\nvar map = Array.prototype.map,\n    prefixes = [\"y\",\"z\",\"a\",\"f\",\"p\",\"n\",\"µ\",\"m\",\"\",\"k\",\"M\",\"G\",\"T\",\"P\",\"E\",\"Z\",\"Y\"];\n\nexport default function(locale) {\n  var group = locale.grouping === undefined || locale.thousands === undefined ? identity : formatGroup(map.call(locale.grouping, Number), locale.thousands + \"\"),\n      currencyPrefix = locale.currency === undefined ? \"\" : locale.currency[0] + \"\",\n      currencySuffix = locale.currency === undefined ? \"\" : locale.currency[1] + \"\",\n      decimal = locale.decimal === undefined ? \".\" : locale.decimal + \"\",\n      numerals = locale.numerals === undefined ? identity : formatNumerals(map.call(locale.numerals, String)),\n      percent = locale.percent === undefined ? \"%\" : locale.percent + \"\",\n      minus = locale.minus === undefined ? \"−\" : locale.minus + \"\",\n      nan = locale.nan === undefined ? \"NaN\" : locale.nan + \"\";\n\n  function newFormat(specifier) {\n    specifier = formatSpecifier(specifier);\n\n    var fill = specifier.fill,\n        align = specifier.align,\n        sign = specifier.sign,\n        symbol = specifier.symbol,\n        zero = specifier.zero,\n        width = specifier.width,\n        comma = specifier.comma,\n        precision = specifier.precision,\n        trim = specifier.trim,\n        type = specifier.type;\n\n    // The \"n\" type is an alias for \",g\".\n    if (type === \"n\") comma = true, type = \"g\";\n\n    // The \"\" type, and any invalid type, is an alias for \".12~g\".\n    else if (!formatTypes[type]) precision === undefined && (precision = 12), trim = true, type = \"g\";\n\n    // If zero fill is specified, padding goes after sign and before digits.\n    if (zero || (fill === \"0\" && align === \"=\")) zero = true, fill = \"0\", align = \"=\";\n\n    // Compute the prefix and suffix.\n    // For SI-prefix, the suffix is lazily computed.\n    var prefix = symbol === \"$\" ? currencyPrefix : symbol === \"#\" && /[boxX]/.test(type) ? \"0\" + type.toLowerCase() : \"\",\n        suffix = symbol === \"$\" ? currencySuffix : /[%p]/.test(type) ? percent : \"\";\n\n    // What format function should we use?\n    // Is this an integer type?\n    // Can this type generate exponential notation?\n    var formatType = formatTypes[type],\n        maybeSuffix = /[defgprs%]/.test(type);\n\n    // Set the default precision if not specified,\n    // or clamp the specified precision to the supported range.\n    // For significant precision, it must be in [1, 21].\n    // For fixed precision, it must be in [0, 20].\n    precision = precision === undefined ? 6\n        : /[gprs]/.test(type) ? Math.max(1, Math.min(21, precision))\n        : Math.max(0, Math.min(20, precision));\n\n    function format(value) {\n      var valuePrefix = prefix,\n          valueSuffix = suffix,\n          i, n, c;\n\n      if (type === \"c\") {\n        valueSuffix = formatType(value) + valueSuffix;\n        value = \"\";\n      } else {\n        value = +value;\n\n        // Determine the sign. -0 is not less than 0, but 1 / -0 is!\n        var valueNegative = value < 0 || 1 / value < 0;\n\n        // Perform the initial formatting.\n        value = isNaN(value) ? nan : formatType(Math.abs(value), precision);\n\n        // Trim insignificant zeros.\n        if (trim) value = formatTrim(value);\n\n        // If a negative value rounds to zero after formatting, and no explicit positive sign is requested, hide the sign.\n        if (valueNegative && +value === 0 && sign !== \"+\") valueNegative = false;\n\n        // Compute the prefix and suffix.\n        valuePrefix = (valueNegative ? (sign === \"(\" ? sign : minus) : sign === \"-\" || sign === \"(\" ? \"\" : sign) + valuePrefix;\n        valueSuffix = (type === \"s\" ? prefixes[8 + prefixExponent / 3] : \"\") + valueSuffix + (valueNegative && sign === \"(\" ? \")\" : \"\");\n\n        // Break the formatted value into the integer “value” part that can be\n        // grouped, and fractional or exponential “suffix” part that is not.\n        if (maybeSuffix) {\n          i = -1, n = value.length;\n          while (++i < n) {\n            if (c = value.charCodeAt(i), 48 > c || c > 57) {\n              valueSuffix = (c === 46 ? decimal + value.slice(i + 1) : value.slice(i)) + valueSuffix;\n              value = value.slice(0, i);\n              break;\n            }\n          }\n        }\n      }\n\n      // If the fill character is not \"0\", grouping is applied before padding.\n      if (comma && !zero) value = group(value, Infinity);\n\n      // Compute the padding.\n      var length = valuePrefix.length + value.length + valueSuffix.length,\n          padding = length < width ? new Array(width - length + 1).join(fill) : \"\";\n\n      // If the fill character is \"0\", grouping is applied after padding.\n      if (comma && zero) value = group(padding + value, padding.length ? width - valueSuffix.length : Infinity), padding = \"\";\n\n      // Reconstruct the final output based on the desired alignment.\n      switch (align) {\n        case \"<\": value = valuePrefix + value + valueSuffix + padding; break;\n        case \"=\": value = valuePrefix + padding + value + valueSuffix; break;\n        case \"^\": value = padding.slice(0, length = padding.length >> 1) + valuePrefix + value + valueSuffix + padding.slice(length); break;\n        default: value = padding + valuePrefix + value + valueSuffix; break;\n      }\n\n      return numerals(value);\n    }\n\n    format.toString = function() {\n      return specifier + \"\";\n    };\n\n    return format;\n  }\n\n  function formatPrefix(specifier, value) {\n    var f = newFormat((specifier = formatSpecifier(specifier), specifier.type = \"f\", specifier)),\n        e = Math.max(-8, Math.min(8, Math.floor(exponent(value) / 3))) * 3,\n        k = Math.pow(10, -e),\n        prefix = prefixes[8 + e / 3];\n    return function(value) {\n      return f(k * value) + prefix;\n    };\n  }\n\n  return {\n    format: newFormat,\n    formatPrefix: formatPrefix\n  };\n}\n","export function basis(t1, v0, v1, v2, v3) {\n  var t2 = t1 * t1, t3 = t2 * t1;\n  return ((1 - 3 * t1 + 3 * t2 - t3) * v0\n      + (4 - 6 * t2 + 3 * t3) * v1\n      + (1 + 3 * t1 + 3 * t2 - 3 * t3) * v2\n      + t3 * v3) / 6;\n}\n\nexport default function(values) {\n  var n = values.length - 1;\n  return function(t) {\n    var i = t <= 0 ? (t = 0) : t >= 1 ? (t = 1, n - 1) : Math.floor(t * n),\n        v1 = values[i],\n        v2 = values[i + 1],\n        v0 = i > 0 ? values[i - 1] : 2 * v1 - v2,\n        v3 = i < n - 1 ? values[i + 2] : 2 * v2 - v1;\n    return basis((t - i / n) * n, v0, v1, v2, v3);\n  };\n}\n","import {basis} from \"./basis.js\";\n\nexport default function(values) {\n  var n = values.length;\n  return function(t) {\n    var i = Math.floor(((t %= 1) < 0 ? ++t : t) * n),\n        v0 = values[(i + n - 1) % n],\n        v1 = values[i % n],\n        v2 = values[(i + 1) % n],\n        v3 = values[(i + 2) % n];\n    return basis((t - i / n) * n, v0, v1, v2, v3);\n  };\n}\n","import constant from \"./constant.js\";\n\nfunction linear(a, d) {\n  return function(t) {\n    return a + t * d;\n  };\n}\n\nfunction exponential(a, b, y) {\n  return a = Math.pow(a, y), b = Math.pow(b, y) - a, y = 1 / y, function(t) {\n    return Math.pow(a + t * b, y);\n  };\n}\n\nexport function hue(a, b) {\n  var d = b - a;\n  return d ? linear(a, d > 180 || d < -180 ? d - 360 * Math.round(d / 360) : d) : constant(isNaN(a) ? b : a);\n}\n\nexport function gamma(y) {\n  return (y = +y) === 1 ? nogamma : function(a, b) {\n    return b - a ? exponential(a, b, y) : constant(isNaN(a) ? b : a);\n  };\n}\n\nexport default function nogamma(a, b) {\n  var d = b - a;\n  return d ? linear(a, d) : constant(isNaN(a) ? b : a);\n}\n","export default x => () => x;\n","import {lab as colorLab} from \"d3-color\";\nimport color from \"./color.js\";\n\nexport default function lab(start, end) {\n  var l = color((start = colorLab(start)).l, (end = colorLab(end)).l),\n      a = color(start.a, end.a),\n      b = color(start.b, end.b),\n      opacity = color(start.opacity, end.opacity);\n  return function(t) {\n    start.l = l(t);\n    start.a = a(t);\n    start.b = b(t);\n    start.opacity = opacity(t);\n    return start + \"\";\n  };\n}\n","export default function(a, b) {\n  return a = +a, b = +b, function(t) {\n    return a * (1 - t) + b * t;\n  };\n}\n","import {rgb as colorRgb} from \"d3-color\";\nimport basis from \"./basis.js\";\nimport basisClosed from \"./basisClosed.js\";\nimport nogamma, {gamma} from \"./color.js\";\n\nexport default (function rgbGamma(y) {\n  var color = gamma(y);\n\n  function rgb(start, end) {\n    var r = color((start = colorRgb(start)).r, (end = colorRgb(end)).r),\n        g = color(start.g, end.g),\n        b = color(start.b, end.b),\n        opacity = nogamma(start.opacity, end.opacity);\n    return function(t) {\n      start.r = r(t);\n      start.g = g(t);\n      start.b = b(t);\n      start.opacity = opacity(t);\n      return start + \"\";\n    };\n  }\n\n  rgb.gamma = rgbGamma;\n\n  return rgb;\n})(1);\n\nfunction rgbSpline(spline) {\n  return function(colors) {\n    var n = colors.length,\n        r = new Array(n),\n        g = new Array(n),\n        b = new Array(n),\n        i, color;\n    for (i = 0; i < n; ++i) {\n      color = colorRgb(colors[i]);\n      r[i] = color.r || 0;\n      g[i] = color.g || 0;\n      b[i] = color.b || 0;\n    }\n    r = spline(r);\n    g = spline(g);\n    b = spline(b);\n    color.opacity = 1;\n    return function(t) {\n      color.r = r(t);\n      color.g = g(t);\n      color.b = b(t);\n      return color + \"\";\n    };\n  };\n}\n\nexport var rgbBasis = rgbSpline(basis);\nexport var rgbBasisClosed = rgbSpline(basisClosed);\n","import number from \"./number.js\";\n\nvar reA = /[-+]?(?:\\d+\\.?\\d*|\\.?\\d+)(?:[eE][-+]?\\d+)?/g,\n    reB = new RegExp(reA.source, \"g\");\n\nfunction zero(b) {\n  return function() {\n    return b;\n  };\n}\n\nfunction one(b) {\n  return function(t) {\n    return b(t) + \"\";\n  };\n}\n\nexport default function(a, b) {\n  var bi = reA.lastIndex = reB.lastIndex = 0, // scan index for next number in b\n      am, // current match in a\n      bm, // current match in b\n      bs, // string preceding current number in b, if any\n      i = -1, // index in s\n      s = [], // string constants and placeholders\n      q = []; // number interpolators\n\n  // Coerce inputs to strings.\n  a = a + \"\", b = b + \"\";\n\n  // Interpolate pairs of numbers in a & b.\n  while ((am = reA.exec(a))\n      && (bm = reB.exec(b))) {\n    if ((bs = bm.index) > bi) { // a string precedes the next number in b\n      bs = b.slice(bi, bs);\n      if (s[i]) s[i] += bs; // coalesce with previous string\n      else s[++i] = bs;\n    }\n    if ((am = am[0]) === (bm = bm[0])) { // numbers in a & b match\n      if (s[i]) s[i] += bm; // coalesce with previous string\n      else s[++i] = bm;\n    } else { // interpolate non-matching numbers\n      s[++i] = null;\n      q.push({i: i, x: number(am, bm)});\n    }\n    bi = reB.lastIndex;\n  }\n\n  // Add remains of b.\n  if (bi < b.length) {\n    bs = b.slice(bi);\n    if (s[i]) s[i] += bs; // coalesce with previous string\n    else s[++i] = bs;\n  }\n\n  // Special optimization for only a single match.\n  // Otherwise, interpolate each of the numbers and rejoin the string.\n  return s.length < 2 ? (q[0]\n      ? one(q[0].x)\n      : zero(b))\n      : (b = q.length, function(t) {\n          for (var i = 0, o; i < b; ++i) s[(o = q[i]).i] = o.x(t);\n          return s.join(\"\");\n        });\n}\n","var degrees = 180 / Math.PI;\n\nexport var identity = {\n  translateX: 0,\n  translateY: 0,\n  rotate: 0,\n  skewX: 0,\n  scaleX: 1,\n  scaleY: 1\n};\n\nexport default function(a, b, c, d, e, f) {\n  var scaleX, scaleY, skewX;\n  if (scaleX = Math.sqrt(a * a + b * b)) a /= scaleX, b /= scaleX;\n  if (skewX = a * c + b * d) c -= a * skewX, d -= b * skewX;\n  if (scaleY = Math.sqrt(c * c + d * d)) c /= scaleY, d /= scaleY, skewX /= scaleY;\n  if (a * d < b * c) a = -a, b = -b, skewX = -skewX, scaleX = -scaleX;\n  return {\n    translateX: e,\n    translateY: f,\n    rotate: Math.atan2(b, a) * degrees,\n    skewX: Math.atan(skewX) * degrees,\n    scaleX: scaleX,\n    scaleY: scaleY\n  };\n}\n","import number from \"../number.js\";\nimport {parseCss, parseSvg} from \"./parse.js\";\n\nfunction interpolateTransform(parse, pxComma, pxParen, degParen) {\n\n  function pop(s) {\n    return s.length ? s.pop() + \" \" : \"\";\n  }\n\n  function translate(xa, ya, xb, yb, s, q) {\n    if (xa !== xb || ya !== yb) {\n      var i = s.push(\"translate(\", null, pxComma, null, pxParen);\n      q.push({i: i - 4, x: number(xa, xb)}, {i: i - 2, x: number(ya, yb)});\n    } else if (xb || yb) {\n      s.push(\"translate(\" + xb + pxComma + yb + pxParen);\n    }\n  }\n\n  function rotate(a, b, s, q) {\n    if (a !== b) {\n      if (a - b > 180) b += 360; else if (b - a > 180) a += 360; // shortest path\n      q.push({i: s.push(pop(s) + \"rotate(\", null, degParen) - 2, x: number(a, b)});\n    } else if (b) {\n      s.push(pop(s) + \"rotate(\" + b + degParen);\n    }\n  }\n\n  function skewX(a, b, s, q) {\n    if (a !== b) {\n      q.push({i: s.push(pop(s) + \"skewX(\", null, degParen) - 2, x: number(a, b)});\n    } else if (b) {\n      s.push(pop(s) + \"skewX(\" + b + degParen);\n    }\n  }\n\n  function scale(xa, ya, xb, yb, s, q) {\n    if (xa !== xb || ya !== yb) {\n      var i = s.push(pop(s) + \"scale(\", null, \",\", null, \")\");\n      q.push({i: i - 4, x: number(xa, xb)}, {i: i - 2, x: number(ya, yb)});\n    } else if (xb !== 1 || yb !== 1) {\n      s.push(pop(s) + \"scale(\" + xb + \",\" + yb + \")\");\n    }\n  }\n\n  return function(a, b) {\n    var s = [], // string constants and placeholders\n        q = []; // number interpolators\n    a = parse(a), b = parse(b);\n    translate(a.translateX, a.translateY, b.translateX, b.translateY, s, q);\n    rotate(a.rotate, b.rotate, s, q);\n    skewX(a.skewX, b.skewX, s, q);\n    scale(a.scaleX, a.scaleY, b.scaleX, b.scaleY, s, q);\n    a = b = null; // gc\n    return function(t) {\n      var i = -1, n = q.length, o;\n      while (++i < n) s[(o = q[i]).i] = o.x(t);\n      return s.join(\"\");\n    };\n  };\n}\n\nexport var interpolateTransformCss = interpolateTransform(parseCss, \"px, \", \"px)\", \"deg)\");\nexport var interpolateTransformSvg = interpolateTransform(parseSvg, \", \", \")\", \")\");\n","import decompose, {identity} from \"./decompose.js\";\n\nvar svgNode;\n\n/* eslint-disable no-undef */\nexport function parseCss(value) {\n  const m = new (typeof DOMMatrix === \"function\" ? DOMMatrix : WebKitCSSMatrix)(value + \"\");\n  return m.isIdentity ? identity : decompose(m.a, m.b, m.c, m.d, m.e, m.f);\n}\n\nexport function parseSvg(value) {\n  if (value == null) return identity;\n  if (!svgNode) svgNode = document.createElementNS(\"http://www.w3.org/2000/svg\", \"g\");\n  svgNode.setAttribute(\"transform\", value);\n  if (!(value = svgNode.transform.baseVal.consolidate())) return identity;\n  value = value.matrix;\n  return decompose(value.a, value.b, value.c, value.d, value.e, value.f);\n}\n","var epsilon2 = 1e-12;\n\nfunction cosh(x) {\n  return ((x = Math.exp(x)) + 1 / x) / 2;\n}\n\nfunction sinh(x) {\n  return ((x = Math.exp(x)) - 1 / x) / 2;\n}\n\nfunction tanh(x) {\n  return ((x = Math.exp(2 * x)) - 1) / (x + 1);\n}\n\nexport default (function zoomRho(rho, rho2, rho4) {\n\n  // p0 = [ux0, uy0, w0]\n  // p1 = [ux1, uy1, w1]\n  function zoom(p0, p1) {\n    var ux0 = p0[0], uy0 = p0[1], w0 = p0[2],\n        ux1 = p1[0], uy1 = p1[1], w1 = p1[2],\n        dx = ux1 - ux0,\n        dy = uy1 - uy0,\n        d2 = dx * dx + dy * dy,\n        i,\n        S;\n\n    // Special case for u0 ≅ u1.\n    if (d2 < epsilon2) {\n      S = Math.log(w1 / w0) / rho;\n      i = function(t) {\n        return [\n          ux0 + t * dx,\n          uy0 + t * dy,\n          w0 * Math.exp(rho * t * S)\n        ];\n      }\n    }\n\n    // General case.\n    else {\n      var d1 = Math.sqrt(d2),\n          b0 = (w1 * w1 - w0 * w0 + rho4 * d2) / (2 * w0 * rho2 * d1),\n          b1 = (w1 * w1 - w0 * w0 - rho4 * d2) / (2 * w1 * rho2 * d1),\n          r0 = Math.log(Math.sqrt(b0 * b0 + 1) - b0),\n          r1 = Math.log(Math.sqrt(b1 * b1 + 1) - b1);\n      S = (r1 - r0) / rho;\n      i = function(t) {\n        var s = t * S,\n            coshr0 = cosh(r0),\n            u = w0 / (rho2 * d1) * (coshr0 * tanh(rho * s + r0) - sinh(r0));\n        return [\n          ux0 + u * dx,\n          uy0 + u * dy,\n          w0 * coshr0 / cosh(rho * s + r0)\n        ];\n      }\n    }\n\n    i.duration = S * 1000 * rho / Math.SQRT2;\n\n    return i;\n  }\n\n  zoom.rho = function(_) {\n    var _1 = Math.max(1e-3, +_), _2 = _1 * _1, _4 = _2 * _2;\n    return zoomRho(_1, _2, _4);\n  };\n\n  return zoom;\n})(Math.SQRT2, 2, 4);\n","export default Math.random;\n","import defaultSource from \"./defaultSource.js\";\n\nexport default (function sourceRandomNormal(source) {\n  function randomNormal(mu, sigma) {\n    var x, r;\n    mu = mu == null ? 0 : +mu;\n    sigma = sigma == null ? 1 : +sigma;\n    return function() {\n      var y;\n\n      // If available, use the second previously-generated uniform random.\n      if (x != null) y = x, x = null;\n\n      // Otherwise, generate a new x and y.\n      else do {\n        x = source() * 2 - 1;\n        y = source() * 2 - 1;\n        r = x * x + y * y;\n      } while (!r || r > 1);\n\n      return mu + sigma * y * Math.sqrt(-2 * Math.log(r) / r);\n    };\n  }\n\n  randomNormal.source = sourceRandomNormal;\n\n  return randomNormal;\n})(defaultSource);\n","export function initRange(domain, range) {\n  switch (arguments.length) {\n    case 0: break;\n    case 1: this.range(domain); break;\n    default: this.range(range).domain(domain); break;\n  }\n  return this;\n}\n\nexport function initInterpolator(domain, interpolator) {\n  switch (arguments.length) {\n    case 0: break;\n    case 1: {\n      if (typeof domain === \"function\") this.interpolator(domain);\n      else this.range(domain);\n      break;\n    }\n    default: {\n      this.domain(domain);\n      if (typeof interpolator === \"function\") this.interpolator(interpolator);\n      else this.range(interpolator);\n      break;\n    }\n  }\n  return this;\n}\n","import {ascending, bisect, quantileSorted as threshold} from \"d3-array\";\nimport {initRange} from \"./init.js\";\n\nexport default function quantile() {\n  var domain = [],\n      range = [],\n      thresholds = [],\n      unknown;\n\n  function rescale() {\n    var i = 0, n = Math.max(1, range.length);\n    thresholds = new Array(n - 1);\n    while (++i < n) thresholds[i - 1] = threshold(domain, i / n);\n    return scale;\n  }\n\n  function scale(x) {\n    return x == null || isNaN(x = +x) ? unknown : range[bisect(thresholds, x)];\n  }\n\n  scale.invertExtent = function(y) {\n    var i = range.indexOf(y);\n    return i < 0 ? [NaN, NaN] : [\n      i > 0 ? thresholds[i - 1] : domain[0],\n      i < thresholds.length ? thresholds[i] : domain[domain.length - 1]\n    ];\n  };\n\n  scale.domain = function(_) {\n    if (!arguments.length) return domain.slice();\n    domain = [];\n    for (let d of _) if (d != null && !isNaN(d = +d)) domain.push(d);\n    domain.sort(ascending);\n    return rescale();\n  };\n\n  scale.range = function(_) {\n    return arguments.length ? (range = Array.from(_), rescale()) : range.slice();\n  };\n\n  scale.unknown = function(_) {\n    return arguments.length ? (unknown = _, scale) : unknown;\n  };\n\n  scale.quantiles = function() {\n    return thresholds.slice();\n  };\n\n  scale.copy = function() {\n    return quantile()\n        .domain(domain)\n        .range(range)\n        .unknown(unknown);\n  };\n\n  return initRange.apply(scale, arguments);\n}\n","// Given something array like (or null), returns something that is strictly an\n// array. This is used to ensure that array-like objects passed to d3.selectAll\n// or selection.selectAll are converted into proper arrays when creating a\n// selection; we don’t ever want to create a selection backed by a live\n// HTMLCollection or NodeList. However, note that selection.selectAll will use a\n// static NodeList as a group, since it safely derived from querySelectorAll.\nexport default function array(x) {\n  return x == null ? [] : Array.isArray(x) ? x : Array.from(x);\n}\n","export default function(x) {\n  return function() {\n    return x;\n  };\n}\n","import namespace from \"./namespace.js\";\nimport {xhtml} from \"./namespaces.js\";\n\nfunction creatorInherit(name) {\n  return function() {\n    var document = this.ownerDocument,\n        uri = this.namespaceURI;\n    return uri === xhtml && document.documentElement.namespaceURI === xhtml\n        ? document.createElement(name)\n        : document.createElementNS(uri, name);\n  };\n}\n\nfunction creatorFixed(fullname) {\n  return function() {\n    return this.ownerDocument.createElementNS(fullname.space, fullname.local);\n  };\n}\n\nexport default function(name) {\n  var fullname = namespace(name);\n  return (fullname.local\n      ? creatorFixed\n      : creatorInherit)(fullname);\n}\n","export default function(selector) {\n  return function() {\n    return this.matches(selector);\n  };\n}\n\nexport function childMatcher(selector) {\n  return function(node) {\n    return node.matches(selector);\n  };\n}\n\n","import namespaces from \"./namespaces.js\";\n\nexport default function(name) {\n  var prefix = name += \"\", i = prefix.indexOf(\":\");\n  if (i >= 0 && (prefix = name.slice(0, i)) !== \"xmlns\") name = name.slice(i + 1);\n  return namespaces.hasOwnProperty(prefix) ? {space: namespaces[prefix], local: name} : name; // eslint-disable-line no-prototype-builtins\n}\n","export var xhtml = \"http://www.w3.org/1999/xhtml\";\n\nexport default {\n  svg: \"http://www.w3.org/2000/svg\",\n  xhtml: xhtml,\n  xlink: \"http://www.w3.org/1999/xlink\",\n  xml: \"http://www.w3.org/XML/1998/namespace\",\n  xmlns: \"http://www.w3.org/2000/xmlns/\"\n};\n","import sourceEvent from \"./sourceEvent.js\";\n\nexport default function(event, node) {\n  event = sourceEvent(event);\n  if (node === undefined) node = event.currentTarget;\n  if (node) {\n    var svg = node.ownerSVGElement || node;\n    if (svg.createSVGPoint) {\n      var point = svg.createSVGPoint();\n      point.x = event.clientX, point.y = event.clientY;\n      point = point.matrixTransform(node.getScreenCTM().inverse());\n      return [point.x, point.y];\n    }\n    if (node.getBoundingClientRect) {\n      var rect = node.getBoundingClientRect();\n      return [event.clientX - rect.left - node.clientLeft, event.clientY - rect.top - node.clientTop];\n    }\n  }\n  return [event.pageX, event.pageY];\n}\n","import {Selection, root} from \"./selection/index.js\";\n\nexport default function(selector) {\n  return typeof selector === \"string\"\n      ? new Selection([[document.querySelector(selector)]], [document.documentElement])\n      : new Selection([[selector]], root);\n}\n","import creator from \"../creator.js\";\n\nexport default function(name) {\n  var create = typeof name === \"function\" ? name : creator(name);\n  return this.select(function() {\n    return this.appendChild(create.apply(this, arguments));\n  });\n}\n","import namespace from \"../namespace.js\";\n\nfunction attrRemove(name) {\n  return function() {\n    this.removeAttribute(name);\n  };\n}\n\nfunction attrRemoveNS(fullname) {\n  return function() {\n    this.removeAttributeNS(fullname.space, fullname.local);\n  };\n}\n\nfunction attrConstant(name, value) {\n  return function() {\n    this.setAttribute(name, value);\n  };\n}\n\nfunction attrConstantNS(fullname, value) {\n  return function() {\n    this.setAttributeNS(fullname.space, fullname.local, value);\n  };\n}\n\nfunction attrFunction(name, value) {\n  return function() {\n    var v = value.apply(this, arguments);\n    if (v == null) this.removeAttribute(name);\n    else this.setAttribute(name, v);\n  };\n}\n\nfunction attrFunctionNS(fullname, value) {\n  return function() {\n    var v = value.apply(this, arguments);\n    if (v == null) this.removeAttributeNS(fullname.space, fullname.local);\n    else this.setAttributeNS(fullname.space, fullname.local, v);\n  };\n}\n\nexport default function(name, value) {\n  var fullname = namespace(name);\n\n  if (arguments.length < 2) {\n    var node = this.node();\n    return fullname.local\n        ? node.getAttributeNS(fullname.space, fullname.local)\n        : node.getAttribute(fullname);\n  }\n\n  return this.each((value == null\n      ? (fullname.local ? attrRemoveNS : attrRemove) : (typeof value === \"function\"\n      ? (fullname.local ? attrFunctionNS : attrFunction)\n      : (fullname.local ? attrConstantNS : attrConstant)))(fullname, value));\n}\n","export default function() {\n  var callback = arguments[0];\n  arguments[0] = this;\n  callback.apply(null, arguments);\n  return this;\n}\n","function classArray(string) {\n  return string.trim().split(/^|\\s+/);\n}\n\nfunction classList(node) {\n  return node.classList || new ClassList(node);\n}\n\nfunction ClassList(node) {\n  this._node = node;\n  this._names = classArray(node.getAttribute(\"class\") || \"\");\n}\n\nClassList.prototype = {\n  add: function(name) {\n    var i = this._names.indexOf(name);\n    if (i < 0) {\n      this._names.push(name);\n      this._node.setAttribute(\"class\", this._names.join(\" \"));\n    }\n  },\n  remove: function(name) {\n    var i = this._names.indexOf(name);\n    if (i >= 0) {\n      this._names.splice(i, 1);\n      this._node.setAttribute(\"class\", this._names.join(\" \"));\n    }\n  },\n  contains: function(name) {\n    return this._names.indexOf(name) >= 0;\n  }\n};\n\nfunction classedAdd(node, names) {\n  var list = classList(node), i = -1, n = names.length;\n  while (++i < n) list.add(names[i]);\n}\n\nfunction classedRemove(node, names) {\n  var list = classList(node), i = -1, n = names.length;\n  while (++i < n) list.remove(names[i]);\n}\n\nfunction classedTrue(names) {\n  return function() {\n    classedAdd(this, names);\n  };\n}\n\nfunction classedFalse(names) {\n  return function() {\n    classedRemove(this, names);\n  };\n}\n\nfunction classedFunction(names, value) {\n  return function() {\n    (value.apply(this, arguments) ? classedAdd : classedRemove)(this, names);\n  };\n}\n\nexport default function(name, value) {\n  var names = classArray(name + \"\");\n\n  if (arguments.length < 2) {\n    var list = classList(this.node()), i = -1, n = names.length;\n    while (++i < n) if (!list.contains(names[i])) return false;\n    return true;\n  }\n\n  return this.each((typeof value === \"function\"\n      ? classedFunction : value\n      ? classedTrue\n      : classedFalse)(names, value));\n}\n","function selection_cloneShallow() {\n  var clone = this.cloneNode(false), parent = this.parentNode;\n  return parent ? parent.insertBefore(clone, this.nextSibling) : clone;\n}\n\nfunction selection_cloneDeep() {\n  var clone = this.cloneNode(true), parent = this.parentNode;\n  return parent ? parent.insertBefore(clone, this.nextSibling) : clone;\n}\n\nexport default function(deep) {\n  return this.select(deep ? selection_cloneDeep : selection_cloneShallow);\n}\n","import {Selection} from \"./index.js\";\nimport {EnterNode} from \"./enter.js\";\nimport constant from \"../constant.js\";\n\nfunction bindIndex(parent, group, enter, update, exit, data) {\n  var i = 0,\n      node,\n      groupLength = group.length,\n      dataLength = data.length;\n\n  // Put any non-null nodes that fit into update.\n  // Put any null nodes into enter.\n  // Put any remaining data into enter.\n  for (; i < dataLength; ++i) {\n    if (node = group[i]) {\n      node.__data__ = data[i];\n      update[i] = node;\n    } else {\n      enter[i] = new EnterNode(parent, data[i]);\n    }\n  }\n\n  // Put any non-null nodes that don’t fit into exit.\n  for (; i < groupLength; ++i) {\n    if (node = group[i]) {\n      exit[i] = node;\n    }\n  }\n}\n\nfunction bindKey(parent, group, enter, update, exit, data, key) {\n  var i,\n      node,\n      nodeByKeyValue = new Map,\n      groupLength = group.length,\n      dataLength = data.length,\n      keyValues = new Array(groupLength),\n      keyValue;\n\n  // Compute the key for each node.\n  // If multiple nodes have the same key, the duplicates are added to exit.\n  for (i = 0; i < groupLength; ++i) {\n    if (node = group[i]) {\n      keyValues[i] = keyValue = key.call(node, node.__data__, i, group) + \"\";\n      if (nodeByKeyValue.has(keyValue)) {\n        exit[i] = node;\n      } else {\n        nodeByKeyValue.set(keyValue, node);\n      }\n    }\n  }\n\n  // Compute the key for each datum.\n  // If there a node associated with this key, join and add it to update.\n  // If there is not (or the key is a duplicate), add it to enter.\n  for (i = 0; i < dataLength; ++i) {\n    keyValue = key.call(parent, data[i], i, data) + \"\";\n    if (node = nodeByKeyValue.get(keyValue)) {\n      update[i] = node;\n      node.__data__ = data[i];\n      nodeByKeyValue.delete(keyValue);\n    } else {\n      enter[i] = new EnterNode(parent, data[i]);\n    }\n  }\n\n  // Add any remaining nodes that were not bound to data to exit.\n  for (i = 0; i < groupLength; ++i) {\n    if ((node = group[i]) && (nodeByKeyValue.get(keyValues[i]) === node)) {\n      exit[i] = node;\n    }\n  }\n}\n\nfunction datum(node) {\n  return node.__data__;\n}\n\nexport default function(value, key) {\n  if (!arguments.length) return Array.from(this, datum);\n\n  var bind = key ? bindKey : bindIndex,\n      parents = this._parents,\n      groups = this._groups;\n\n  if (typeof value !== \"function\") value = constant(value);\n\n  for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j = 0; j < m; ++j) {\n    var parent = parents[j],\n        group = groups[j],\n        groupLength = group.length,\n        data = arraylike(value.call(parent, parent && parent.__data__, j, parents)),\n        dataLength = data.length,\n        enterGroup = enter[j] = new Array(dataLength),\n        updateGroup = update[j] = new Array(dataLength),\n        exitGroup = exit[j] = new Array(groupLength);\n\n    bind(parent, group, enterGroup, updateGroup, exitGroup, data, key);\n\n    // Now connect the enter nodes to their following update node, such that\n    // appendChild can insert the materialized enter node before this node,\n    // rather than at the end of the parent node.\n    for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {\n      if (previous = enterGroup[i0]) {\n        if (i0 >= i1) i1 = i0 + 1;\n        while (!(next = updateGroup[i1]) && ++i1 < dataLength);\n        previous._next = next || null;\n      }\n    }\n  }\n\n  update = new Selection(update, parents);\n  update._enter = enter;\n  update._exit = exit;\n  return update;\n}\n\n// Given some data, this returns an array-like view of it: an object that\n// exposes a length property and allows numeric indexing. Note that unlike\n// selectAll, this isn’t worried about “live” collections because the resulting\n// array will only be used briefly while data is being bound. (It is possible to\n// cause the data to change while iterating by using a key function, but please\n// don’t; we’d rather avoid a gratuitous copy.)\nfunction arraylike(data) {\n  return typeof data === \"object\" && \"length\" in data\n    ? data // Array, TypedArray, NodeList, array-like\n    : Array.from(data); // Map, Set, iterable, string, or anything else\n}\n","export default function(value) {\n  return arguments.length\n      ? this.property(\"__data__\", value)\n      : this.node().__data__;\n}\n","import defaultView from \"../window.js\";\n\nfunction dispatchEvent(node, type, params) {\n  var window = defaultView(node),\n      event = window.CustomEvent;\n\n  if (typeof event === \"function\") {\n    event = new event(type, params);\n  } else {\n    event = window.document.createEvent(\"Event\");\n    if (params) event.initEvent(type, params.bubbles, params.cancelable), event.detail = params.detail;\n    else event.initEvent(type, false, false);\n  }\n\n  node.dispatchEvent(event);\n}\n\nfunction dispatchConstant(type, params) {\n  return function() {\n    return dispatchEvent(this, type, params);\n  };\n}\n\nfunction dispatchFunction(type, params) {\n  return function() {\n    return dispatchEvent(this, type, params.apply(this, arguments));\n  };\n}\n\nexport default function(type, params) {\n  return this.each((typeof params === \"function\"\n      ? dispatchFunction\n      : dispatchConstant)(type, params));\n}\n","export default function(callback) {\n\n  for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {\n    for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {\n      if (node = group[i]) callback.call(node, node.__data__, i, group);\n    }\n  }\n\n  return this;\n}\n","export default function() {\n  return !this.node();\n}\n","import sparse from \"./sparse.js\";\nimport {Selection} from \"./index.js\";\n\nexport default function() {\n  return new Selection(this._enter || this._groups.map(sparse), this._parents);\n}\n\nexport function EnterNode(parent, datum) {\n  this.ownerDocument = parent.ownerDocument;\n  this.namespaceURI = parent.namespaceURI;\n  this._next = null;\n  this._parent = parent;\n  this.__data__ = datum;\n}\n\nEnterNode.prototype = {\n  constructor: EnterNode,\n  appendChild: function(child) { return this._parent.insertBefore(child, this._next); },\n  insertBefore: function(child, next) { return this._parent.insertBefore(child, next); },\n  querySelector: function(selector) { return this._parent.querySelector(selector); },\n  querySelectorAll: function(selector) { return this._parent.querySelectorAll(selector); }\n};\n","import sparse from \"./sparse.js\";\nimport {Selection} from \"./index.js\";\n\nexport default function() {\n  return new Selection(this._exit || this._groups.map(sparse), this._parents);\n}\n","import {Selection} from \"./index.js\";\nimport matcher from \"../matcher.js\";\n\nexport default function(match) {\n  if (typeof match !== \"function\") match = matcher(match);\n\n  for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {\n      if ((node = group[i]) && match.call(node, node.__data__, i, group)) {\n        subgroup.push(node);\n      }\n    }\n  }\n\n  return new Selection(subgroups, this._parents);\n}\n","function htmlRemove() {\n  this.innerHTML = \"\";\n}\n\nfunction htmlConstant(value) {\n  return function() {\n    this.innerHTML = value;\n  };\n}\n\nfunction htmlFunction(value) {\n  return function() {\n    var v = value.apply(this, arguments);\n    this.innerHTML = v == null ? \"\" : v;\n  };\n}\n\nexport default function(value) {\n  return arguments.length\n      ? this.each(value == null\n          ? htmlRemove : (typeof value === \"function\"\n          ? htmlFunction\n          : htmlConstant)(value))\n      : this.node().innerHTML;\n}\n","import selection_select from \"./select.js\";\nimport selection_selectAll from \"./selectAll.js\";\nimport selection_selectChild from \"./selectChild.js\";\nimport selection_selectChildren from \"./selectChildren.js\";\nimport selection_filter from \"./filter.js\";\nimport selection_data from \"./data.js\";\nimport selection_enter from \"./enter.js\";\nimport selection_exit from \"./exit.js\";\nimport selection_join from \"./join.js\";\nimport selection_merge from \"./merge.js\";\nimport selection_order from \"./order.js\";\nimport selection_sort from \"./sort.js\";\nimport selection_call from \"./call.js\";\nimport selection_nodes from \"./nodes.js\";\nimport selection_node from \"./node.js\";\nimport selection_size from \"./size.js\";\nimport selection_empty from \"./empty.js\";\nimport selection_each from \"./each.js\";\nimport selection_attr from \"./attr.js\";\nimport selection_style from \"./style.js\";\nimport selection_property from \"./property.js\";\nimport selection_classed from \"./classed.js\";\nimport selection_text from \"./text.js\";\nimport selection_html from \"./html.js\";\nimport selection_raise from \"./raise.js\";\nimport selection_lower from \"./lower.js\";\nimport selection_append from \"./append.js\";\nimport selection_insert from \"./insert.js\";\nimport selection_remove from \"./remove.js\";\nimport selection_clone from \"./clone.js\";\nimport selection_datum from \"./datum.js\";\nimport selection_on from \"./on.js\";\nimport selection_dispatch from \"./dispatch.js\";\nimport selection_iterator from \"./iterator.js\";\n\nexport var root = [null];\n\nexport function Selection(groups, parents) {\n  this._groups = groups;\n  this._parents = parents;\n}\n\nfunction selection() {\n  return new Selection([[document.documentElement]], root);\n}\n\nfunction selection_selection() {\n  return this;\n}\n\nSelection.prototype = selection.prototype = {\n  constructor: Selection,\n  select: selection_select,\n  selectAll: selection_selectAll,\n  selectChild: selection_selectChild,\n  selectChildren: selection_selectChildren,\n  filter: selection_filter,\n  data: selection_data,\n  enter: selection_enter,\n  exit: selection_exit,\n  join: selection_join,\n  merge: selection_merge,\n  selection: selection_selection,\n  order: selection_order,\n  sort: selection_sort,\n  call: selection_call,\n  nodes: selection_nodes,\n  node: selection_node,\n  size: selection_size,\n  empty: selection_empty,\n  each: selection_each,\n  attr: selection_attr,\n  style: selection_style,\n  property: selection_property,\n  classed: selection_classed,\n  text: selection_text,\n  html: selection_html,\n  raise: selection_raise,\n  lower: selection_lower,\n  append: selection_append,\n  insert: selection_insert,\n  remove: selection_remove,\n  clone: selection_clone,\n  datum: selection_datum,\n  on: selection_on,\n  dispatch: selection_dispatch,\n  [Symbol.iterator]: selection_iterator\n};\n\nexport default selection;\n","import creator from \"../creator.js\";\nimport selector from \"../selector.js\";\n\nfunction constantNull() {\n  return null;\n}\n\nexport default function(name, before) {\n  var create = typeof name === \"function\" ? name : creator(name),\n      select = before == null ? constantNull : typeof before === \"function\" ? before : selector(before);\n  return this.select(function() {\n    return this.insertBefore(create.apply(this, arguments), select.apply(this, arguments) || null);\n  });\n}\n","export default function*() {\n  for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {\n    for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {\n      if (node = group[i]) yield node;\n    }\n  }\n}\n","export default function(onenter, onupdate, onexit) {\n  var enter = this.enter(), update = this, exit = this.exit();\n  if (typeof onenter === \"function\") {\n    enter = onenter(enter);\n    if (enter) enter = enter.selection();\n  } else {\n    enter = enter.append(onenter + \"\");\n  }\n  if (onupdate != null) {\n    update = onupdate(update);\n    if (update) update = update.selection();\n  }\n  if (onexit == null) exit.remove(); else onexit(exit);\n  return enter && update ? enter.merge(update).order() : update;\n}\n","function lower() {\n  if (this.previousSibling) this.parentNode.insertBefore(this, this.parentNode.firstChild);\n}\n\nexport default function() {\n  return this.each(lower);\n}\n","import {Selection} from \"./index.js\";\n\nexport default function(context) {\n  var selection = context.selection ? context.selection() : context;\n\n  for (var groups0 = this._groups, groups1 = selection._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {\n    for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {\n      if (node = group0[i] || group1[i]) {\n        merge[i] = node;\n      }\n    }\n  }\n\n  for (; j < m0; ++j) {\n    merges[j] = groups0[j];\n  }\n\n  return new Selection(merges, this._parents);\n}\n","export default function() {\n\n  for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {\n    for (var group = groups[j], i = 0, n = group.length; i < n; ++i) {\n      var node = group[i];\n      if (node) return node;\n    }\n  }\n\n  return null;\n}\n","export default function() {\n  return Array.from(this);\n}\n","function contextListener(listener) {\n  return function(event) {\n    listener.call(this, event, this.__data__);\n  };\n}\n\nfunction parseTypenames(typenames) {\n  return typenames.trim().split(/^|\\s+/).map(function(t) {\n    var name = \"\", i = t.indexOf(\".\");\n    if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);\n    return {type: t, name: name};\n  });\n}\n\nfunction onRemove(typename) {\n  return function() {\n    var on = this.__on;\n    if (!on) return;\n    for (var j = 0, i = -1, m = on.length, o; j < m; ++j) {\n      if (o = on[j], (!typename.type || o.type === typename.type) && o.name === typename.name) {\n        this.removeEventListener(o.type, o.listener, o.options);\n      } else {\n        on[++i] = o;\n      }\n    }\n    if (++i) on.length = i;\n    else delete this.__on;\n  };\n}\n\nfunction onAdd(typename, value, options) {\n  return function() {\n    var on = this.__on, o, listener = contextListener(value);\n    if (on) for (var j = 0, m = on.length; j < m; ++j) {\n      if ((o = on[j]).type === typename.type && o.name === typename.name) {\n        this.removeEventListener(o.type, o.listener, o.options);\n        this.addEventListener(o.type, o.listener = listener, o.options = options);\n        o.value = value;\n        return;\n      }\n    }\n    this.addEventListener(typename.type, listener, options);\n    o = {type: typename.type, name: typename.name, value: value, listener: listener, options: options};\n    if (!on) this.__on = [o];\n    else on.push(o);\n  };\n}\n\nexport default function(typename, value, options) {\n  var typenames = parseTypenames(typename + \"\"), i, n = typenames.length, t;\n\n  if (arguments.length < 2) {\n    var on = this.node().__on;\n    if (on) for (var j = 0, m = on.length, o; j < m; ++j) {\n      for (i = 0, o = on[j]; i < n; ++i) {\n        if ((t = typenames[i]).type === o.type && t.name === o.name) {\n          return o.value;\n        }\n      }\n    }\n    return;\n  }\n\n  on = value ? onAdd : onRemove;\n  for (i = 0; i < n; ++i) this.each(on(typenames[i], value, options));\n  return this;\n}\n","export default function() {\n\n  for (var groups = this._groups, j = -1, m = groups.length; ++j < m;) {\n    for (var group = groups[j], i = group.length - 1, next = group[i], node; --i >= 0;) {\n      if (node = group[i]) {\n        if (next && node.compareDocumentPosition(next) ^ 4) next.parentNode.insertBefore(node, next);\n        next = node;\n      }\n    }\n  }\n\n  return this;\n}\n","function propertyRemove(name) {\n  return function() {\n    delete this[name];\n  };\n}\n\nfunction propertyConstant(name, value) {\n  return function() {\n    this[name] = value;\n  };\n}\n\nfunction propertyFunction(name, value) {\n  return function() {\n    var v = value.apply(this, arguments);\n    if (v == null) delete this[name];\n    else this[name] = v;\n  };\n}\n\nexport default function(name, value) {\n  return arguments.length > 1\n      ? this.each((value == null\n          ? propertyRemove : typeof value === \"function\"\n          ? propertyFunction\n          : propertyConstant)(name, value))\n      : this.node()[name];\n}\n","function raise() {\n  if (this.nextSibling) this.parentNode.appendChild(this);\n}\n\nexport default function() {\n  return this.each(raise);\n}\n","function remove() {\n  var parent = this.parentNode;\n  if (parent) parent.removeChild(this);\n}\n\nexport default function() {\n  return this.each(remove);\n}\n","import {Selection} from \"./index.js\";\nimport selector from \"../selector.js\";\n\nexport default function(select) {\n  if (typeof select !== \"function\") select = selector(select);\n\n  for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {\n      if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {\n        if (\"__data__\" in node) subnode.__data__ = node.__data__;\n        subgroup[i] = subnode;\n      }\n    }\n  }\n\n  return new Selection(subgroups, this._parents);\n}\n","import {Selection} from \"./index.js\";\nimport array from \"../array.js\";\nimport selectorAll from \"../selectorAll.js\";\n\nfunction arrayAll(select) {\n  return function() {\n    return array(select.apply(this, arguments));\n  };\n}\n\nexport default function(select) {\n  if (typeof select === \"function\") select = arrayAll(select);\n  else select = selectorAll(select);\n\n  for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {\n      if (node = group[i]) {\n        subgroups.push(select.call(node, node.__data__, i, group));\n        parents.push(node);\n      }\n    }\n  }\n\n  return new Selection(subgroups, parents);\n}\n","import {childMatcher} from \"../matcher.js\";\n\nvar find = Array.prototype.find;\n\nfunction childFind(match) {\n  return function() {\n    return find.call(this.children, match);\n  };\n}\n\nfunction childFirst() {\n  return this.firstElementChild;\n}\n\nexport default function(match) {\n  return this.select(match == null ? childFirst\n      : childFind(typeof match === \"function\" ? match : childMatcher(match)));\n}\n","import {childMatcher} from \"../matcher.js\";\n\nvar filter = Array.prototype.filter;\n\nfunction children() {\n  return Array.from(this.children);\n}\n\nfunction childrenFilter(match) {\n  return function() {\n    return filter.call(this.children, match);\n  };\n}\n\nexport default function(match) {\n  return this.selectAll(match == null ? children\n      : childrenFilter(typeof match === \"function\" ? match : childMatcher(match)));\n}\n","export default function() {\n  let size = 0;\n  for (const node of this) ++size; // eslint-disable-line no-unused-vars\n  return size;\n}\n","import {Selection} from \"./index.js\";\n\nexport default function(compare) {\n  if (!compare) compare = ascending;\n\n  function compareNode(a, b) {\n    return a && b ? compare(a.__data__, b.__data__) : !a - !b;\n  }\n\n  for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, sortgroup = sortgroups[j] = new Array(n), node, i = 0; i < n; ++i) {\n      if (node = group[i]) {\n        sortgroup[i] = node;\n      }\n    }\n    sortgroup.sort(compareNode);\n  }\n\n  return new Selection(sortgroups, this._parents).order();\n}\n\nfunction ascending(a, b) {\n  return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;\n}\n","export default function(update) {\n  return new Array(update.length);\n}\n","import defaultView from \"../window.js\";\n\nfunction styleRemove(name) {\n  return function() {\n    this.style.removeProperty(name);\n  };\n}\n\nfunction styleConstant(name, value, priority) {\n  return function() {\n    this.style.setProperty(name, value, priority);\n  };\n}\n\nfunction styleFunction(name, value, priority) {\n  return function() {\n    var v = value.apply(this, arguments);\n    if (v == null) this.style.removeProperty(name);\n    else this.style.setProperty(name, v, priority);\n  };\n}\n\nexport default function(name, value, priority) {\n  return arguments.length > 1\n      ? this.each((value == null\n            ? styleRemove : typeof value === \"function\"\n            ? styleFunction\n            : styleConstant)(name, value, priority == null ? \"\" : priority))\n      : styleValue(this.node(), name);\n}\n\nexport function styleValue(node, name) {\n  return node.style.getPropertyValue(name)\n      || defaultView(node).getComputedStyle(node, null).getPropertyValue(name);\n}\n","function textRemove() {\n  this.textContent = \"\";\n}\n\nfunction textConstant(value) {\n  return function() {\n    this.textContent = value;\n  };\n}\n\nfunction textFunction(value) {\n  return function() {\n    var v = value.apply(this, arguments);\n    this.textContent = v == null ? \"\" : v;\n  };\n}\n\nexport default function(value) {\n  return arguments.length\n      ? this.each(value == null\n          ? textRemove : (typeof value === \"function\"\n          ? textFunction\n          : textConstant)(value))\n      : this.node().textContent;\n}\n","function none() {}\n\nexport default function(selector) {\n  return selector == null ? none : function() {\n    return this.querySelector(selector);\n  };\n}\n","function empty() {\n  return [];\n}\n\nexport default function(selector) {\n  return selector == null ? empty : function() {\n    return this.querySelectorAll(selector);\n  };\n}\n","export default function(event) {\n  let sourceEvent;\n  while (sourceEvent = event.sourceEvent) event = sourceEvent;\n  return event;\n}\n","export default function(node) {\n  return (node.ownerDocument && node.ownerDocument.defaultView) // node is a Node\n      || (node.document && node) // node is a Window\n      || node.defaultView; // node is a Document\n}\n","import {Timer} from \"./timer.js\";\n\nexport default function(callback, delay, time) {\n  var t = new Timer;\n  delay = delay == null ? 0 : +delay;\n  t.restart(elapsed => {\n    t.stop();\n    callback(elapsed + delay);\n  }, delay, time);\n  return t;\n}\n","var frame = 0, // is an animation frame pending?\n    timeout = 0, // is a timeout pending?\n    interval = 0, // are any timers active?\n    pokeDelay = 1000, // how frequently we check for clock skew\n    taskHead,\n    taskTail,\n    clockLast = 0,\n    clockNow = 0,\n    clockSkew = 0,\n    clock = typeof performance === \"object\" && performance.now ? performance : Date,\n    setFrame = typeof window === \"object\" && window.requestAnimationFrame ? window.requestAnimationFrame.bind(window) : function(f) { setTimeout(f, 17); };\n\nexport function now() {\n  return clockNow || (setFrame(clearNow), clockNow = clock.now() + clockSkew);\n}\n\nfunction clearNow() {\n  clockNow = 0;\n}\n\nexport function Timer() {\n  this._call =\n  this._time =\n  this._next = null;\n}\n\nTimer.prototype = timer.prototype = {\n  constructor: Timer,\n  restart: function(callback, delay, time) {\n    if (typeof callback !== \"function\") throw new TypeError(\"callback is not a function\");\n    time = (time == null ? now() : +time) + (delay == null ? 0 : +delay);\n    if (!this._next && taskTail !== this) {\n      if (taskTail) taskTail._next = this;\n      else taskHead = this;\n      taskTail = this;\n    }\n    this._call = callback;\n    this._time = time;\n    sleep();\n  },\n  stop: function() {\n    if (this._call) {\n      this._call = null;\n      this._time = Infinity;\n      sleep();\n    }\n  }\n};\n\nexport function timer(callback, delay, time) {\n  var t = new Timer;\n  t.restart(callback, delay, time);\n  return t;\n}\n\nexport function timerFlush() {\n  now(); // Get the current time, if not already set.\n  ++frame; // Pretend we’ve set an alarm, if we haven’t already.\n  var t = taskHead, e;\n  while (t) {\n    if ((e = clockNow - t._time) >= 0) t._call.call(undefined, e);\n    t = t._next;\n  }\n  --frame;\n}\n\nfunction wake() {\n  clockNow = (clockLast = clock.now()) + clockSkew;\n  frame = timeout = 0;\n  try {\n    timerFlush();\n  } finally {\n    frame = 0;\n    nap();\n    clockNow = 0;\n  }\n}\n\nfunction poke() {\n  var now = clock.now(), delay = now - clockLast;\n  if (delay > pokeDelay) clockSkew -= delay, clockLast = now;\n}\n\nfunction nap() {\n  var t0, t1 = taskHead, t2, time = Infinity;\n  while (t1) {\n    if (t1._call) {\n      if (time > t1._time) time = t1._time;\n      t0 = t1, t1 = t1._next;\n    } else {\n      t2 = t1._next, t1._next = null;\n      t1 = t0 ? t0._next = t2 : taskHead = t2;\n    }\n  }\n  taskTail = t0;\n  sleep(time);\n}\n\nfunction sleep(time) {\n  if (frame) return; // Soonest alarm already set, or will be.\n  if (timeout) timeout = clearTimeout(timeout);\n  var delay = time - clockNow; // Strictly less than if we recomputed clockNow.\n  if (delay > 24) {\n    if (time < Infinity) timeout = setTimeout(wake, time - clock.now() - clockSkew);\n    if (interval) interval = clearInterval(interval);\n  } else {\n    if (!interval) clockLast = clock.now(), interval = setInterval(poke, pokeDelay);\n    frame = 1, setFrame(wake);\n  }\n}\n","import {Transition} from \"./transition/index.js\";\nimport {SCHEDULED} from \"./transition/schedule.js\";\n\nvar root = [null];\n\nexport default function(node, name) {\n  var schedules = node.__transition,\n      schedule,\n      i;\n\n  if (schedules) {\n    name = name == null ? null : name + \"\";\n    for (i in schedules) {\n      if ((schedule = schedules[i]).state > SCHEDULED && schedule.name === name) {\n        return new Transition([[node]], root, name, +i);\n      }\n    }\n  }\n\n  return null;\n}\n","import \"./selection/index.js\";\nexport {default as transition} from \"./transition/index.js\";\nexport {default as active} from \"./active.js\";\nexport {default as interrupt} from \"./interrupt.js\";\n","import {STARTING, ENDING, ENDED} from \"./transition/schedule.js\";\n\nexport default function(node, name) {\n  var schedules = node.__transition,\n      schedule,\n      active,\n      empty = true,\n      i;\n\n  if (!schedules) return;\n\n  name = name == null ? null : name + \"\";\n\n  for (i in schedules) {\n    if ((schedule = schedules[i]).name !== name) { empty = false; continue; }\n    active = schedule.state > STARTING && schedule.state < ENDING;\n    schedule.state = ENDED;\n    schedule.timer.stop();\n    schedule.on.call(active ? \"interrupt\" : \"cancel\", node, node.__data__, schedule.index, schedule.group);\n    delete schedules[i];\n  }\n\n  if (empty) delete node.__transition;\n}\n","import {selection} from \"d3-selection\";\nimport selection_interrupt from \"./interrupt.js\";\nimport selection_transition from \"./transition.js\";\n\nselection.prototype.interrupt = selection_interrupt;\nselection.prototype.transition = selection_transition;\n","import interrupt from \"../interrupt.js\";\n\nexport default function(name) {\n  return this.each(function() {\n    interrupt(this, name);\n  });\n}\n","import {Transition, newId} from \"../transition/index.js\";\nimport schedule from \"../transition/schedule.js\";\nimport {easeCubicInOut} from \"d3-ease\";\nimport {now} from \"d3-timer\";\n\nvar defaultTiming = {\n  time: null, // Set on use.\n  delay: 0,\n  duration: 250,\n  ease: easeCubicInOut\n};\n\nfunction inherit(node, id) {\n  var timing;\n  while (!(timing = node.__transition) || !(timing = timing[id])) {\n    if (!(node = node.parentNode)) {\n      throw new Error(`transition ${id} not found`);\n    }\n  }\n  return timing;\n}\n\nexport default function(name) {\n  var id,\n      timing;\n\n  if (name instanceof Transition) {\n    id = name._id, name = name._name;\n  } else {\n    id = newId(), (timing = defaultTiming).time = now(), name = name == null ? null : name + \"\";\n  }\n\n  for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {\n      if (node = group[i]) {\n        schedule(node, name, id, i, group, timing || inherit(node, id));\n      }\n    }\n  }\n\n  return new Transition(groups, this._parents, name, id);\n}\n","import {interpolateTransformSvg as interpolateTransform} from \"d3-interpolate\";\nimport {namespace} from \"d3-selection\";\nimport {tweenValue} from \"./tween.js\";\nimport interpolate from \"./interpolate.js\";\n\nfunction attrRemove(name) {\n  return function() {\n    this.removeAttribute(name);\n  };\n}\n\nfunction attrRemoveNS(fullname) {\n  return function() {\n    this.removeAttributeNS(fullname.space, fullname.local);\n  };\n}\n\nfunction attrConstant(name, interpolate, value1) {\n  var string00,\n      string1 = value1 + \"\",\n      interpolate0;\n  return function() {\n    var string0 = this.getAttribute(name);\n    return string0 === string1 ? null\n        : string0 === string00 ? interpolate0\n        : interpolate0 = interpolate(string00 = string0, value1);\n  };\n}\n\nfunction attrConstantNS(fullname, interpolate, value1) {\n  var string00,\n      string1 = value1 + \"\",\n      interpolate0;\n  return function() {\n    var string0 = this.getAttributeNS(fullname.space, fullname.local);\n    return string0 === string1 ? null\n        : string0 === string00 ? interpolate0\n        : interpolate0 = interpolate(string00 = string0, value1);\n  };\n}\n\nfunction attrFunction(name, interpolate, value) {\n  var string00,\n      string10,\n      interpolate0;\n  return function() {\n    var string0, value1 = value(this), string1;\n    if (value1 == null) return void this.removeAttribute(name);\n    string0 = this.getAttribute(name);\n    string1 = value1 + \"\";\n    return string0 === string1 ? null\n        : string0 === string00 && string1 === string10 ? interpolate0\n        : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));\n  };\n}\n\nfunction attrFunctionNS(fullname, interpolate, value) {\n  var string00,\n      string10,\n      interpolate0;\n  return function() {\n    var string0, value1 = value(this), string1;\n    if (value1 == null) return void this.removeAttributeNS(fullname.space, fullname.local);\n    string0 = this.getAttributeNS(fullname.space, fullname.local);\n    string1 = value1 + \"\";\n    return string0 === string1 ? null\n        : string0 === string00 && string1 === string10 ? interpolate0\n        : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));\n  };\n}\n\nexport default function(name, value) {\n  var fullname = namespace(name), i = fullname === \"transform\" ? interpolateTransform : interpolate;\n  return this.attrTween(name, typeof value === \"function\"\n      ? (fullname.local ? attrFunctionNS : attrFunction)(fullname, i, tweenValue(this, \"attr.\" + name, value))\n      : value == null ? (fullname.local ? attrRemoveNS : attrRemove)(fullname)\n      : (fullname.local ? attrConstantNS : attrConstant)(fullname, i, value));\n}\n","import {namespace} from \"d3-selection\";\n\nfunction attrInterpolate(name, i) {\n  return function(t) {\n    this.setAttribute(name, i.call(this, t));\n  };\n}\n\nfunction attrInterpolateNS(fullname, i) {\n  return function(t) {\n    this.setAttributeNS(fullname.space, fullname.local, i.call(this, t));\n  };\n}\n\nfunction attrTweenNS(fullname, value) {\n  var t0, i0;\n  function tween() {\n    var i = value.apply(this, arguments);\n    if (i !== i0) t0 = (i0 = i) && attrInterpolateNS(fullname, i);\n    return t0;\n  }\n  tween._value = value;\n  return tween;\n}\n\nfunction attrTween(name, value) {\n  var t0, i0;\n  function tween() {\n    var i = value.apply(this, arguments);\n    if (i !== i0) t0 = (i0 = i) && attrInterpolate(name, i);\n    return t0;\n  }\n  tween._value = value;\n  return tween;\n}\n\nexport default function(name, value) {\n  var key = \"attr.\" + name;\n  if (arguments.length < 2) return (key = this.tween(key)) && key._value;\n  if (value == null) return this.tween(key, null);\n  if (typeof value !== \"function\") throw new Error;\n  var fullname = namespace(name);\n  return this.tween(key, (fullname.local ? attrTweenNS : attrTween)(fullname, value));\n}\n","import {get, init} from \"./schedule.js\";\n\nfunction delayFunction(id, value) {\n  return function() {\n    init(this, id).delay = +value.apply(this, arguments);\n  };\n}\n\nfunction delayConstant(id, value) {\n  return value = +value, function() {\n    init(this, id).delay = value;\n  };\n}\n\nexport default function(value) {\n  var id = this._id;\n\n  return arguments.length\n      ? this.each((typeof value === \"function\"\n          ? delayFunction\n          : delayConstant)(id, value))\n      : get(this.node(), id).delay;\n}\n","import {get, set} from \"./schedule.js\";\n\nfunction durationFunction(id, value) {\n  return function() {\n    set(this, id).duration = +value.apply(this, arguments);\n  };\n}\n\nfunction durationConstant(id, value) {\n  return value = +value, function() {\n    set(this, id).duration = value;\n  };\n}\n\nexport default function(value) {\n  var id = this._id;\n\n  return arguments.length\n      ? this.each((typeof value === \"function\"\n          ? durationFunction\n          : durationConstant)(id, value))\n      : get(this.node(), id).duration;\n}\n","import {get, set} from \"./schedule.js\";\n\nfunction easeConstant(id, value) {\n  if (typeof value !== \"function\") throw new Error;\n  return function() {\n    set(this, id).ease = value;\n  };\n}\n\nexport default function(value) {\n  var id = this._id;\n\n  return arguments.length\n      ? this.each(easeConstant(id, value))\n      : get(this.node(), id).ease;\n}\n","import {set} from \"./schedule.js\";\n\nfunction easeVarying(id, value) {\n  return function() {\n    var v = value.apply(this, arguments);\n    if (typeof v !== \"function\") throw new Error;\n    set(this, id).ease = v;\n  };\n}\n\nexport default function(value) {\n  if (typeof value !== \"function\") throw new Error;\n  return this.each(easeVarying(this._id, value));\n}\n","import {set} from \"./schedule.js\";\n\nexport default function() {\n  var on0, on1, that = this, id = that._id, size = that.size();\n  return new Promise(function(resolve, reject) {\n    var cancel = {value: reject},\n        end = {value: function() { if (--size === 0) resolve(); }};\n\n    that.each(function() {\n      var schedule = set(this, id),\n          on = schedule.on;\n\n      // If this node shared a dispatch with the previous node,\n      // just assign the updated shared dispatch and we’re done!\n      // Otherwise, copy-on-write.\n      if (on !== on0) {\n        on1 = (on0 = on).copy();\n        on1._.cancel.push(cancel);\n        on1._.interrupt.push(cancel);\n        on1._.end.push(end);\n      }\n\n      schedule.on = on1;\n    });\n\n    // The selection was empty, resolve end immediately\n    if (size === 0) resolve();\n  });\n}\n","import {matcher} from \"d3-selection\";\nimport {Transition} from \"./index.js\";\n\nexport default function(match) {\n  if (typeof match !== \"function\") match = matcher(match);\n\n  for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {\n      if ((node = group[i]) && match.call(node, node.__data__, i, group)) {\n        subgroup.push(node);\n      }\n    }\n  }\n\n  return new Transition(subgroups, this._parents, this._name, this._id);\n}\n","import {selection} from \"d3-selection\";\nimport transition_attr from \"./attr.js\";\nimport transition_attrTween from \"./attrTween.js\";\nimport transition_delay from \"./delay.js\";\nimport transition_duration from \"./duration.js\";\nimport transition_ease from \"./ease.js\";\nimport transition_easeVarying from \"./easeVarying.js\";\nimport transition_filter from \"./filter.js\";\nimport transition_merge from \"./merge.js\";\nimport transition_on from \"./on.js\";\nimport transition_remove from \"./remove.js\";\nimport transition_select from \"./select.js\";\nimport transition_selectAll from \"./selectAll.js\";\nimport transition_selection from \"./selection.js\";\nimport transition_style from \"./style.js\";\nimport transition_styleTween from \"./styleTween.js\";\nimport transition_text from \"./text.js\";\nimport transition_textTween from \"./textTween.js\";\nimport transition_transition from \"./transition.js\";\nimport transition_tween from \"./tween.js\";\nimport transition_end from \"./end.js\";\n\nvar id = 0;\n\nexport function Transition(groups, parents, name, id) {\n  this._groups = groups;\n  this._parents = parents;\n  this._name = name;\n  this._id = id;\n}\n\nexport default function transition(name) {\n  return selection().transition(name);\n}\n\nexport function newId() {\n  return ++id;\n}\n\nvar selection_prototype = selection.prototype;\n\nTransition.prototype = transition.prototype = {\n  constructor: Transition,\n  select: transition_select,\n  selectAll: transition_selectAll,\n  selectChild: selection_prototype.selectChild,\n  selectChildren: selection_prototype.selectChildren,\n  filter: transition_filter,\n  merge: transition_merge,\n  selection: transition_selection,\n  transition: transition_transition,\n  call: selection_prototype.call,\n  nodes: selection_prototype.nodes,\n  node: selection_prototype.node,\n  size: selection_prototype.size,\n  empty: selection_prototype.empty,\n  each: selection_prototype.each,\n  on: transition_on,\n  attr: transition_attr,\n  attrTween: transition_attrTween,\n  style: transition_style,\n  styleTween: transition_styleTween,\n  text: transition_text,\n  textTween: transition_textTween,\n  remove: transition_remove,\n  tween: transition_tween,\n  delay: transition_delay,\n  duration: transition_duration,\n  ease: transition_ease,\n  easeVarying: transition_easeVarying,\n  end: transition_end,\n  [Symbol.iterator]: selection_prototype[Symbol.iterator]\n};\n","import {color} from \"d3-color\";\nimport {interpolateNumber, interpolateRgb, interpolateString} from \"d3-interpolate\";\n\nexport default function(a, b) {\n  var c;\n  return (typeof b === \"number\" ? interpolateNumber\n      : b instanceof color ? interpolateRgb\n      : (c = color(b)) ? (b = c, interpolateRgb)\n      : interpolateString)(a, b);\n}\n","import {Transition} from \"./index.js\";\n\nexport default function(transition) {\n  if (transition._id !== this._id) throw new Error;\n\n  for (var groups0 = this._groups, groups1 = transition._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {\n    for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {\n      if (node = group0[i] || group1[i]) {\n        merge[i] = node;\n      }\n    }\n  }\n\n  for (; j < m0; ++j) {\n    merges[j] = groups0[j];\n  }\n\n  return new Transition(merges, this._parents, this._name, this._id);\n}\n","import {get, set, init} from \"./schedule.js\";\n\nfunction start(name) {\n  return (name + \"\").trim().split(/^|\\s+/).every(function(t) {\n    var i = t.indexOf(\".\");\n    if (i >= 0) t = t.slice(0, i);\n    return !t || t === \"start\";\n  });\n}\n\nfunction onFunction(id, name, listener) {\n  var on0, on1, sit = start(name) ? init : set;\n  return function() {\n    var schedule = sit(this, id),\n        on = schedule.on;\n\n    // If this node shared a dispatch with the previous node,\n    // just assign the updated shared dispatch and we’re done!\n    // Otherwise, copy-on-write.\n    if (on !== on0) (on1 = (on0 = on).copy()).on(name, listener);\n\n    schedule.on = on1;\n  };\n}\n\nexport default function(name, listener) {\n  var id = this._id;\n\n  return arguments.length < 2\n      ? get(this.node(), id).on.on(name)\n      : this.each(onFunction(id, name, listener));\n}\n","function removeFunction(id) {\n  return function() {\n    var parent = this.parentNode;\n    for (var i in this.__transition) if (+i !== id) return;\n    if (parent) parent.removeChild(this);\n  };\n}\n\nexport default function() {\n  return this.on(\"end.remove\", removeFunction(this._id));\n}\n","import {dispatch} from \"d3-dispatch\";\nimport {timer, timeout} from \"d3-timer\";\n\nvar emptyOn = dispatch(\"start\", \"end\", \"cancel\", \"interrupt\");\nvar emptyTween = [];\n\nexport var CREATED = 0;\nexport var SCHEDULED = 1;\nexport var STARTING = 2;\nexport var STARTED = 3;\nexport var RUNNING = 4;\nexport var ENDING = 5;\nexport var ENDED = 6;\n\nexport default function(node, name, id, index, group, timing) {\n  var schedules = node.__transition;\n  if (!schedules) node.__transition = {};\n  else if (id in schedules) return;\n  create(node, id, {\n    name: name,\n    index: index, // For context during callback.\n    group: group, // For context during callback.\n    on: emptyOn,\n    tween: emptyTween,\n    time: timing.time,\n    delay: timing.delay,\n    duration: timing.duration,\n    ease: timing.ease,\n    timer: null,\n    state: CREATED\n  });\n}\n\nexport function init(node, id) {\n  var schedule = get(node, id);\n  if (schedule.state > CREATED) throw new Error(\"too late; already scheduled\");\n  return schedule;\n}\n\nexport function set(node, id) {\n  var schedule = get(node, id);\n  if (schedule.state > STARTED) throw new Error(\"too late; already running\");\n  return schedule;\n}\n\nexport function get(node, id) {\n  var schedule = node.__transition;\n  if (!schedule || !(schedule = schedule[id])) throw new Error(\"transition not found\");\n  return schedule;\n}\n\nfunction create(node, id, self) {\n  var schedules = node.__transition,\n      tween;\n\n  // Initialize the self timer when the transition is created.\n  // Note the actual delay is not known until the first callback!\n  schedules[id] = self;\n  self.timer = timer(schedule, 0, self.time);\n\n  function schedule(elapsed) {\n    self.state = SCHEDULED;\n    self.timer.restart(start, self.delay, self.time);\n\n    // If the elapsed delay is less than our first sleep, start immediately.\n    if (self.delay <= elapsed) start(elapsed - self.delay);\n  }\n\n  function start(elapsed) {\n    var i, j, n, o;\n\n    // If the state is not SCHEDULED, then we previously errored on start.\n    if (self.state !== SCHEDULED) return stop();\n\n    for (i in schedules) {\n      o = schedules[i];\n      if (o.name !== self.name) continue;\n\n      // While this element already has a starting transition during this frame,\n      // defer starting an interrupting transition until that transition has a\n      // chance to tick (and possibly end); see d3/d3-transition#54!\n      if (o.state === STARTED) return timeout(start);\n\n      // Interrupt the active transition, if any.\n      if (o.state === RUNNING) {\n        o.state = ENDED;\n        o.timer.stop();\n        o.on.call(\"interrupt\", node, node.__data__, o.index, o.group);\n        delete schedules[i];\n      }\n\n      // Cancel any pre-empted transitions.\n      else if (+i < id) {\n        o.state = ENDED;\n        o.timer.stop();\n        o.on.call(\"cancel\", node, node.__data__, o.index, o.group);\n        delete schedules[i];\n      }\n    }\n\n    // Defer the first tick to end of the current frame; see d3/d3#1576.\n    // Note the transition may be canceled after start and before the first tick!\n    // Note this must be scheduled before the start event; see d3/d3-transition#16!\n    // Assuming this is successful, subsequent callbacks go straight to tick.\n    timeout(function() {\n      if (self.state === STARTED) {\n        self.state = RUNNING;\n        self.timer.restart(tick, self.delay, self.time);\n        tick(elapsed);\n      }\n    });\n\n    // Dispatch the start event.\n    // Note this must be done before the tween are initialized.\n    self.state = STARTING;\n    self.on.call(\"start\", node, node.__data__, self.index, self.group);\n    if (self.state !== STARTING) return; // interrupted\n    self.state = STARTED;\n\n    // Initialize the tween, deleting null tween.\n    tween = new Array(n = self.tween.length);\n    for (i = 0, j = -1; i < n; ++i) {\n      if (o = self.tween[i].value.call(node, node.__data__, self.index, self.group)) {\n        tween[++j] = o;\n      }\n    }\n    tween.length = j + 1;\n  }\n\n  function tick(elapsed) {\n    var t = elapsed < self.duration ? self.ease.call(null, elapsed / self.duration) : (self.timer.restart(stop), self.state = ENDING, 1),\n        i = -1,\n        n = tween.length;\n\n    while (++i < n) {\n      tween[i].call(node, t);\n    }\n\n    // Dispatch the end event.\n    if (self.state === ENDING) {\n      self.on.call(\"end\", node, node.__data__, self.index, self.group);\n      stop();\n    }\n  }\n\n  function stop() {\n    self.state = ENDED;\n    self.timer.stop();\n    delete schedules[id];\n    for (var i in schedules) return; // eslint-disable-line no-unused-vars\n    delete node.__transition;\n  }\n}\n","import {selector} from \"d3-selection\";\nimport {Transition} from \"./index.js\";\nimport schedule, {get} from \"./schedule.js\";\n\nexport default function(select) {\n  var name = this._name,\n      id = this._id;\n\n  if (typeof select !== \"function\") select = selector(select);\n\n  for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {\n      if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {\n        if (\"__data__\" in node) subnode.__data__ = node.__data__;\n        subgroup[i] = subnode;\n        schedule(subgroup[i], name, id, i, subgroup, get(node, id));\n      }\n    }\n  }\n\n  return new Transition(subgroups, this._parents, name, id);\n}\n","import {selectorAll} from \"d3-selection\";\nimport {Transition} from \"./index.js\";\nimport schedule, {get} from \"./schedule.js\";\n\nexport default function(select) {\n  var name = this._name,\n      id = this._id;\n\n  if (typeof select !== \"function\") select = selectorAll(select);\n\n  for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {\n      if (node = group[i]) {\n        for (var children = select.call(node, node.__data__, i, group), child, inherit = get(node, id), k = 0, l = children.length; k < l; ++k) {\n          if (child = children[k]) {\n            schedule(child, name, id, k, children, inherit);\n          }\n        }\n        subgroups.push(children);\n        parents.push(node);\n      }\n    }\n  }\n\n  return new Transition(subgroups, parents, name, id);\n}\n","import {selection} from \"d3-selection\";\n\nvar Selection = selection.prototype.constructor;\n\nexport default function() {\n  return new Selection(this._groups, this._parents);\n}\n","import {interpolateTransformCss as interpolateTransform} from \"d3-interpolate\";\nimport {style} from \"d3-selection\";\nimport {set} from \"./schedule.js\";\nimport {tweenValue} from \"./tween.js\";\nimport interpolate from \"./interpolate.js\";\n\nfunction styleNull(name, interpolate) {\n  var string00,\n      string10,\n      interpolate0;\n  return function() {\n    var string0 = style(this, name),\n        string1 = (this.style.removeProperty(name), style(this, name));\n    return string0 === string1 ? null\n        : string0 === string00 && string1 === string10 ? interpolate0\n        : interpolate0 = interpolate(string00 = string0, string10 = string1);\n  };\n}\n\nfunction styleRemove(name) {\n  return function() {\n    this.style.removeProperty(name);\n  };\n}\n\nfunction styleConstant(name, interpolate, value1) {\n  var string00,\n      string1 = value1 + \"\",\n      interpolate0;\n  return function() {\n    var string0 = style(this, name);\n    return string0 === string1 ? null\n        : string0 === string00 ? interpolate0\n        : interpolate0 = interpolate(string00 = string0, value1);\n  };\n}\n\nfunction styleFunction(name, interpolate, value) {\n  var string00,\n      string10,\n      interpolate0;\n  return function() {\n    var string0 = style(this, name),\n        value1 = value(this),\n        string1 = value1 + \"\";\n    if (value1 == null) string1 = value1 = (this.style.removeProperty(name), style(this, name));\n    return string0 === string1 ? null\n        : string0 === string00 && string1 === string10 ? interpolate0\n        : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));\n  };\n}\n\nfunction styleMaybeRemove(id, name) {\n  var on0, on1, listener0, key = \"style.\" + name, event = \"end.\" + key, remove;\n  return function() {\n    var schedule = set(this, id),\n        on = schedule.on,\n        listener = schedule.value[key] == null ? remove || (remove = styleRemove(name)) : undefined;\n\n    // If this node shared a dispatch with the previous node,\n    // just assign the updated shared dispatch and we’re done!\n    // Otherwise, copy-on-write.\n    if (on !== on0 || listener0 !== listener) (on1 = (on0 = on).copy()).on(event, listener0 = listener);\n\n    schedule.on = on1;\n  };\n}\n\nexport default function(name, value, priority) {\n  var i = (name += \"\") === \"transform\" ? interpolateTransform : interpolate;\n  return value == null ? this\n      .styleTween(name, styleNull(name, i))\n      .on(\"end.style.\" + name, styleRemove(name))\n    : typeof value === \"function\" ? this\n      .styleTween(name, styleFunction(name, i, tweenValue(this, \"style.\" + name, value)))\n      .each(styleMaybeRemove(this._id, name))\n    : this\n      .styleTween(name, styleConstant(name, i, value), priority)\n      .on(\"end.style.\" + name, null);\n}\n","function styleInterpolate(name, i, priority) {\n  return function(t) {\n    this.style.setProperty(name, i.call(this, t), priority);\n  };\n}\n\nfunction styleTween(name, value, priority) {\n  var t, i0;\n  function tween() {\n    var i = value.apply(this, arguments);\n    if (i !== i0) t = (i0 = i) && styleInterpolate(name, i, priority);\n    return t;\n  }\n  tween._value = value;\n  return tween;\n}\n\nexport default function(name, value, priority) {\n  var key = \"style.\" + (name += \"\");\n  if (arguments.length < 2) return (key = this.tween(key)) && key._value;\n  if (value == null) return this.tween(key, null);\n  if (typeof value !== \"function\") throw new Error;\n  return this.tween(key, styleTween(name, value, priority == null ? \"\" : priority));\n}\n","import {tweenValue} from \"./tween.js\";\n\nfunction textConstant(value) {\n  return function() {\n    this.textContent = value;\n  };\n}\n\nfunction textFunction(value) {\n  return function() {\n    var value1 = value(this);\n    this.textContent = value1 == null ? \"\" : value1;\n  };\n}\n\nexport default function(value) {\n  return this.tween(\"text\", typeof value === \"function\"\n      ? textFunction(tweenValue(this, \"text\", value))\n      : textConstant(value == null ? \"\" : value + \"\"));\n}\n","function textInterpolate(i) {\n  return function(t) {\n    this.textContent = i.call(this, t);\n  };\n}\n\nfunction textTween(value) {\n  var t0, i0;\n  function tween() {\n    var i = value.apply(this, arguments);\n    if (i !== i0) t0 = (i0 = i) && textInterpolate(i);\n    return t0;\n  }\n  tween._value = value;\n  return tween;\n}\n\nexport default function(value) {\n  var key = \"text\";\n  if (arguments.length < 1) return (key = this.tween(key)) && key._value;\n  if (value == null) return this.tween(key, null);\n  if (typeof value !== \"function\") throw new Error;\n  return this.tween(key, textTween(value));\n}\n","import {Transition, newId} from \"./index.js\";\nimport schedule, {get} from \"./schedule.js\";\n\nexport default function() {\n  var name = this._name,\n      id0 = this._id,\n      id1 = newId();\n\n  for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {\n    for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {\n      if (node = group[i]) {\n        var inherit = get(node, id0);\n        schedule(node, name, id1, i, group, {\n          time: inherit.time + inherit.delay + inherit.duration,\n          delay: 0,\n          duration: inherit.duration,\n          ease: inherit.ease\n        });\n      }\n    }\n  }\n\n  return new Transition(groups, this._parents, name, id1);\n}\n","import {get, set} from \"./schedule.js\";\n\nfunction tweenRemove(id, name) {\n  var tween0, tween1;\n  return function() {\n    var schedule = set(this, id),\n        tween = schedule.tween;\n\n    // If this node shared tween with the previous node,\n    // just assign the updated shared tween and we’re done!\n    // Otherwise, copy-on-write.\n    if (tween !== tween0) {\n      tween1 = tween0 = tween;\n      for (var i = 0, n = tween1.length; i < n; ++i) {\n        if (tween1[i].name === name) {\n          tween1 = tween1.slice();\n          tween1.splice(i, 1);\n          break;\n        }\n      }\n    }\n\n    schedule.tween = tween1;\n  };\n}\n\nfunction tweenFunction(id, name, value) {\n  var tween0, tween1;\n  if (typeof value !== \"function\") throw new Error;\n  return function() {\n    var schedule = set(this, id),\n        tween = schedule.tween;\n\n    // If this node shared tween with the previous node,\n    // just assign the updated shared tween and we’re done!\n    // Otherwise, copy-on-write.\n    if (tween !== tween0) {\n      tween1 = (tween0 = tween).slice();\n      for (var t = {name: name, value: value}, i = 0, n = tween1.length; i < n; ++i) {\n        if (tween1[i].name === name) {\n          tween1[i] = t;\n          break;\n        }\n      }\n      if (i === n) tween1.push(t);\n    }\n\n    schedule.tween = tween1;\n  };\n}\n\nexport default function(name, value) {\n  var id = this._id;\n\n  name += \"\";\n\n  if (arguments.length < 2) {\n    var tween = get(this.node(), id).tween;\n    for (var i = 0, n = tween.length, t; i < n; ++i) {\n      if ((t = tween[i]).name === name) {\n        return t.value;\n      }\n    }\n    return null;\n  }\n\n  return this.each((value == null ? tweenRemove : tweenFunction)(id, name, value));\n}\n\nexport function tweenValue(transition, name, value) {\n  var id = transition._id;\n\n  transition.each(function() {\n    var schedule = set(this, id);\n    (schedule.value || (schedule.value = {}))[name] = value.apply(this, arguments);\n  });\n\n  return function(node) {\n    return get(node, id).value[name];\n  };\n}\n","export default x => () => x;\n","export default function ZoomEvent(type, {\n  sourceEvent,\n  target,\n  transform,\n  dispatch\n}) {\n  Object.defineProperties(this, {\n    type: {value: type, enumerable: true, configurable: true},\n    sourceEvent: {value: sourceEvent, enumerable: true, configurable: true},\n    target: {value: target, enumerable: true, configurable: true},\n    transform: {value: transform, enumerable: true, configurable: true},\n    _: {value: dispatch}\n  });\n}\n","export {default as zoom} from \"./zoom.js\";\nexport {default as zoomTransform, identity as zoomIdentity, Transform as ZoomTransform} from \"./transform.js\";\n","export function nopropagation(event) {\n  event.stopImmediatePropagation();\n}\n\nexport default function(event) {\n  event.preventDefault();\n  event.stopImmediatePropagation();\n}\n","export function Transform(k, x, y) {\n  this.k = k;\n  this.x = x;\n  this.y = y;\n}\n\nTransform.prototype = {\n  constructor: Transform,\n  scale: function(k) {\n    return k === 1 ? this : new Transform(this.k * k, this.x, this.y);\n  },\n  translate: function(x, y) {\n    return x === 0 & y === 0 ? this : new Transform(this.k, this.x + this.k * x, this.y + this.k * y);\n  },\n  apply: function(point) {\n    return [point[0] * this.k + this.x, point[1] * this.k + this.y];\n  },\n  applyX: function(x) {\n    return x * this.k + this.x;\n  },\n  applyY: function(y) {\n    return y * this.k + this.y;\n  },\n  invert: function(location) {\n    return [(location[0] - this.x) / this.k, (location[1] - this.y) / this.k];\n  },\n  invertX: function(x) {\n    return (x - this.x) / this.k;\n  },\n  invertY: function(y) {\n    return (y - this.y) / this.k;\n  },\n  rescaleX: function(x) {\n    return x.copy().domain(x.range().map(this.invertX, this).map(x.invert, x));\n  },\n  rescaleY: function(y) {\n    return y.copy().domain(y.range().map(this.invertY, this).map(y.invert, y));\n  },\n  toString: function() {\n    return \"translate(\" + this.x + \",\" + this.y + \") scale(\" + this.k + \")\";\n  }\n};\n\nexport var identity = new Transform(1, 0, 0);\n\ntransform.prototype = Transform.prototype;\n\nexport default function transform(node) {\n  while (!node.__zoom) if (!(node = node.parentNode)) return identity;\n  return node.__zoom;\n}\n","import {dispatch} from \"d3-dispatch\";\nimport {dragDisable, dragEnable} from \"d3-drag\";\nimport {interpolateZoom} from \"d3-interpolate\";\nimport {select, pointer} from \"d3-selection\";\nimport {interrupt} from \"d3-transition\";\nimport constant from \"./constant.js\";\nimport ZoomEvent from \"./event.js\";\nimport {Transform, identity} from \"./transform.js\";\nimport noevent, {nopropagation} from \"./noevent.js\";\n\n// Ignore right-click, since that should open the context menu.\n// except for pinch-to-zoom, which is sent as a wheel+ctrlKey event\nfunction defaultFilter(event) {\n  return (!event.ctrlKey || event.type === 'wheel') && !event.button;\n}\n\nfunction defaultExtent() {\n  var e = this;\n  if (e instanceof SVGElement) {\n    e = e.ownerSVGElement || e;\n    if (e.hasAttribute(\"viewBox\")) {\n      e = e.viewBox.baseVal;\n      return [[e.x, e.y], [e.x + e.width, e.y + e.height]];\n    }\n    return [[0, 0], [e.width.baseVal.value, e.height.baseVal.value]];\n  }\n  return [[0, 0], [e.clientWidth, e.clientHeight]];\n}\n\nfunction defaultTransform() {\n  return this.__zoom || identity;\n}\n\nfunction defaultWheelDelta(event) {\n  return -event.deltaY * (event.deltaMode === 1 ? 0.05 : event.deltaMode ? 1 : 0.002) * (event.ctrlKey ? 10 : 1);\n}\n\nfunction defaultTouchable() {\n  return navigator.maxTouchPoints || (\"ontouchstart\" in this);\n}\n\nfunction defaultConstrain(transform, extent, translateExtent) {\n  var dx0 = transform.invertX(extent[0][0]) - translateExtent[0][0],\n      dx1 = transform.invertX(extent[1][0]) - translateExtent[1][0],\n      dy0 = transform.invertY(extent[0][1]) - translateExtent[0][1],\n      dy1 = transform.invertY(extent[1][1]) - translateExtent[1][1];\n  return transform.translate(\n    dx1 > dx0 ? (dx0 + dx1) / 2 : Math.min(0, dx0) || Math.max(0, dx1),\n    dy1 > dy0 ? (dy0 + dy1) / 2 : Math.min(0, dy0) || Math.max(0, dy1)\n  );\n}\n\nexport default function() {\n  var filter = defaultFilter,\n      extent = defaultExtent,\n      constrain = defaultConstrain,\n      wheelDelta = defaultWheelDelta,\n      touchable = defaultTouchable,\n      scaleExtent = [0, Infinity],\n      translateExtent = [[-Infinity, -Infinity], [Infinity, Infinity]],\n      duration = 250,\n      interpolate = interpolateZoom,\n      listeners = dispatch(\"start\", \"zoom\", \"end\"),\n      touchstarting,\n      touchfirst,\n      touchending,\n      touchDelay = 500,\n      wheelDelay = 150,\n      clickDistance2 = 0,\n      tapDistance = 10;\n\n  function zoom(selection) {\n    selection\n        .property(\"__zoom\", defaultTransform)\n        .on(\"wheel.zoom\", wheeled, {passive: false})\n        .on(\"mousedown.zoom\", mousedowned)\n        .on(\"dblclick.zoom\", dblclicked)\n      .filter(touchable)\n        .on(\"touchstart.zoom\", touchstarted)\n        .on(\"touchmove.zoom\", touchmoved)\n        .on(\"touchend.zoom touchcancel.zoom\", touchended)\n        .style(\"-webkit-tap-highlight-color\", \"rgba(0,0,0,0)\");\n  }\n\n  zoom.transform = function(collection, transform, point, event) {\n    var selection = collection.selection ? collection.selection() : collection;\n    selection.property(\"__zoom\", defaultTransform);\n    if (collection !== selection) {\n      schedule(collection, transform, point, event);\n    } else {\n      selection.interrupt().each(function() {\n        gesture(this, arguments)\n          .event(event)\n          .start()\n          .zoom(null, typeof transform === \"function\" ? transform.apply(this, arguments) : transform)\n          .end();\n      });\n    }\n  };\n\n  zoom.scaleBy = function(selection, k, p, event) {\n    zoom.scaleTo(selection, function() {\n      var k0 = this.__zoom.k,\n          k1 = typeof k === \"function\" ? k.apply(this, arguments) : k;\n      return k0 * k1;\n    }, p, event);\n  };\n\n  zoom.scaleTo = function(selection, k, p, event) {\n    zoom.transform(selection, function() {\n      var e = extent.apply(this, arguments),\n          t0 = this.__zoom,\n          p0 = p == null ? centroid(e) : typeof p === \"function\" ? p.apply(this, arguments) : p,\n          p1 = t0.invert(p0),\n          k1 = typeof k === \"function\" ? k.apply(this, arguments) : k;\n      return constrain(translate(scale(t0, k1), p0, p1), e, translateExtent);\n    }, p, event);\n  };\n\n  zoom.translateBy = function(selection, x, y, event) {\n    zoom.transform(selection, function() {\n      return constrain(this.__zoom.translate(\n        typeof x === \"function\" ? x.apply(this, arguments) : x,\n        typeof y === \"function\" ? y.apply(this, arguments) : y\n      ), extent.apply(this, arguments), translateExtent);\n    }, null, event);\n  };\n\n  zoom.translateTo = function(selection, x, y, p, event) {\n    zoom.transform(selection, function() {\n      var e = extent.apply(this, arguments),\n          t = this.__zoom,\n          p0 = p == null ? centroid(e) : typeof p === \"function\" ? p.apply(this, arguments) : p;\n      return constrain(identity.translate(p0[0], p0[1]).scale(t.k).translate(\n        typeof x === \"function\" ? -x.apply(this, arguments) : -x,\n        typeof y === \"function\" ? -y.apply(this, arguments) : -y\n      ), e, translateExtent);\n    }, p, event);\n  };\n\n  function scale(transform, k) {\n    k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], k));\n    return k === transform.k ? transform : new Transform(k, transform.x, transform.y);\n  }\n\n  function translate(transform, p0, p1) {\n    var x = p0[0] - p1[0] * transform.k, y = p0[1] - p1[1] * transform.k;\n    return x === transform.x && y === transform.y ? transform : new Transform(transform.k, x, y);\n  }\n\n  function centroid(extent) {\n    return [(+extent[0][0] + +extent[1][0]) / 2, (+extent[0][1] + +extent[1][1]) / 2];\n  }\n\n  function schedule(transition, transform, point, event) {\n    transition\n        .on(\"start.zoom\", function() { gesture(this, arguments).event(event).start(); })\n        .on(\"interrupt.zoom end.zoom\", function() { gesture(this, arguments).event(event).end(); })\n        .tween(\"zoom\", function() {\n          var that = this,\n              args = arguments,\n              g = gesture(that, args).event(event),\n              e = extent.apply(that, args),\n              p = point == null ? centroid(e) : typeof point === \"function\" ? point.apply(that, args) : point,\n              w = Math.max(e[1][0] - e[0][0], e[1][1] - e[0][1]),\n              a = that.__zoom,\n              b = typeof transform === \"function\" ? transform.apply(that, args) : transform,\n              i = interpolate(a.invert(p).concat(w / a.k), b.invert(p).concat(w / b.k));\n          return function(t) {\n            if (t === 1) t = b; // Avoid rounding error on end.\n            else { var l = i(t), k = w / l[2]; t = new Transform(k, p[0] - l[0] * k, p[1] - l[1] * k); }\n            g.zoom(null, t);\n          };\n        });\n  }\n\n  function gesture(that, args, clean) {\n    return (!clean && that.__zooming) || new Gesture(that, args);\n  }\n\n  function Gesture(that, args) {\n    this.that = that;\n    this.args = args;\n    this.active = 0;\n    this.sourceEvent = null;\n    this.extent = extent.apply(that, args);\n    this.taps = 0;\n  }\n\n  Gesture.prototype = {\n    event: function(event) {\n      if (event) this.sourceEvent = event;\n      return this;\n    },\n    start: function() {\n      if (++this.active === 1) {\n        this.that.__zooming = this;\n        this.emit(\"start\");\n      }\n      return this;\n    },\n    zoom: function(key, transform) {\n      if (this.mouse && key !== \"mouse\") this.mouse[1] = transform.invert(this.mouse[0]);\n      if (this.touch0 && key !== \"touch\") this.touch0[1] = transform.invert(this.touch0[0]);\n      if (this.touch1 && key !== \"touch\") this.touch1[1] = transform.invert(this.touch1[0]);\n      this.that.__zoom = transform;\n      this.emit(\"zoom\");\n      return this;\n    },\n    end: function() {\n      if (--this.active === 0) {\n        delete this.that.__zooming;\n        this.emit(\"end\");\n      }\n      return this;\n    },\n    emit: function(type) {\n      var d = select(this.that).datum();\n      listeners.call(\n        type,\n        this.that,\n        new ZoomEvent(type, {\n          sourceEvent: this.sourceEvent,\n          target: zoom,\n          type,\n          transform: this.that.__zoom,\n          dispatch: listeners\n        }),\n        d\n      );\n    }\n  };\n\n  function wheeled(event, ...args) {\n    if (!filter.apply(this, arguments)) return;\n    var g = gesture(this, args).event(event),\n        t = this.__zoom,\n        k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], t.k * Math.pow(2, wheelDelta.apply(this, arguments)))),\n        p = pointer(event);\n\n    // If the mouse is in the same location as before, reuse it.\n    // If there were recent wheel events, reset the wheel idle timeout.\n    if (g.wheel) {\n      if (g.mouse[0][0] !== p[0] || g.mouse[0][1] !== p[1]) {\n        g.mouse[1] = t.invert(g.mouse[0] = p);\n      }\n      clearTimeout(g.wheel);\n    }\n\n    // If this wheel event won’t trigger a transform change, ignore it.\n    else if (t.k === k) return;\n\n    // Otherwise, capture the mouse point and location at the start.\n    else {\n      g.mouse = [p, t.invert(p)];\n      interrupt(this);\n      g.start();\n    }\n\n    noevent(event);\n    g.wheel = setTimeout(wheelidled, wheelDelay);\n    g.zoom(\"mouse\", constrain(translate(scale(t, k), g.mouse[0], g.mouse[1]), g.extent, translateExtent));\n\n    function wheelidled() {\n      g.wheel = null;\n      g.end();\n    }\n  }\n\n  function mousedowned(event, ...args) {\n    if (touchending || !filter.apply(this, arguments)) return;\n    var currentTarget = event.currentTarget,\n        g = gesture(this, args, true).event(event),\n        v = select(event.view).on(\"mousemove.zoom\", mousemoved, true).on(\"mouseup.zoom\", mouseupped, true),\n        p = pointer(event, currentTarget),\n        x0 = event.clientX,\n        y0 = event.clientY;\n\n    dragDisable(event.view);\n    nopropagation(event);\n    g.mouse = [p, this.__zoom.invert(p)];\n    interrupt(this);\n    g.start();\n\n    function mousemoved(event) {\n      noevent(event);\n      if (!g.moved) {\n        var dx = event.clientX - x0, dy = event.clientY - y0;\n        g.moved = dx * dx + dy * dy > clickDistance2;\n      }\n      g.event(event)\n       .zoom(\"mouse\", constrain(translate(g.that.__zoom, g.mouse[0] = pointer(event, currentTarget), g.mouse[1]), g.extent, translateExtent));\n    }\n\n    function mouseupped(event) {\n      v.on(\"mousemove.zoom mouseup.zoom\", null);\n      dragEnable(event.view, g.moved);\n      noevent(event);\n      g.event(event).end();\n    }\n  }\n\n  function dblclicked(event, ...args) {\n    if (!filter.apply(this, arguments)) return;\n    var t0 = this.__zoom,\n        p0 = pointer(event.changedTouches ? event.changedTouches[0] : event, this),\n        p1 = t0.invert(p0),\n        k1 = t0.k * (event.shiftKey ? 0.5 : 2),\n        t1 = constrain(translate(scale(t0, k1), p0, p1), extent.apply(this, args), translateExtent);\n\n    noevent(event);\n    if (duration > 0) select(this).transition().duration(duration).call(schedule, t1, p0, event);\n    else select(this).call(zoom.transform, t1, p0, event);\n  }\n\n  function touchstarted(event, ...args) {\n    if (!filter.apply(this, arguments)) return;\n    var touches = event.touches,\n        n = touches.length,\n        g = gesture(this, args, event.changedTouches.length === n).event(event),\n        started, i, t, p;\n\n    nopropagation(event);\n    for (i = 0; i < n; ++i) {\n      t = touches[i], p = pointer(t, this);\n      p = [p, this.__zoom.invert(p), t.identifier];\n      if (!g.touch0) g.touch0 = p, started = true, g.taps = 1 + !!touchstarting;\n      else if (!g.touch1 && g.touch0[2] !== p[2]) g.touch1 = p, g.taps = 0;\n    }\n\n    if (touchstarting) touchstarting = clearTimeout(touchstarting);\n\n    if (started) {\n      if (g.taps < 2) touchfirst = p[0], touchstarting = setTimeout(function() { touchstarting = null; }, touchDelay);\n      interrupt(this);\n      g.start();\n    }\n  }\n\n  function touchmoved(event, ...args) {\n    if (!this.__zooming) return;\n    var g = gesture(this, args).event(event),\n        touches = event.changedTouches,\n        n = touches.length, i, t, p, l;\n\n    noevent(event);\n    for (i = 0; i < n; ++i) {\n      t = touches[i], p = pointer(t, this);\n      if (g.touch0 && g.touch0[2] === t.identifier) g.touch0[0] = p;\n      else if (g.touch1 && g.touch1[2] === t.identifier) g.touch1[0] = p;\n    }\n    t = g.that.__zoom;\n    if (g.touch1) {\n      var p0 = g.touch0[0], l0 = g.touch0[1],\n          p1 = g.touch1[0], l1 = g.touch1[1],\n          dp = (dp = p1[0] - p0[0]) * dp + (dp = p1[1] - p0[1]) * dp,\n          dl = (dl = l1[0] - l0[0]) * dl + (dl = l1[1] - l0[1]) * dl;\n      t = scale(t, Math.sqrt(dp / dl));\n      p = [(p0[0] + p1[0]) / 2, (p0[1] + p1[1]) / 2];\n      l = [(l0[0] + l1[0]) / 2, (l0[1] + l1[1]) / 2];\n    }\n    else if (g.touch0) p = g.touch0[0], l = g.touch0[1];\n    else return;\n\n    g.zoom(\"touch\", constrain(translate(t, p, l), g.extent, translateExtent));\n  }\n\n  function touchended(event, ...args) {\n    if (!this.__zooming) return;\n    var g = gesture(this, args).event(event),\n        touches = event.changedTouches,\n        n = touches.length, i, t;\n\n    nopropagation(event);\n    if (touchending) clearTimeout(touchending);\n    touchending = setTimeout(function() { touchending = null; }, touchDelay);\n    for (i = 0; i < n; ++i) {\n      t = touches[i];\n      if (g.touch0 && g.touch0[2] === t.identifier) delete g.touch0;\n      else if (g.touch1 && g.touch1[2] === t.identifier) delete g.touch1;\n    }\n    if (g.touch1 && !g.touch0) g.touch0 = g.touch1, delete g.touch1;\n    if (g.touch0) g.touch0[1] = this.__zoom.invert(g.touch0[0]);\n    else {\n      g.end();\n      // If this was a dbltap, reroute to the (optional) dblclick.zoom handler.\n      if (g.taps === 2) {\n        t = pointer(t, this);\n        if (Math.hypot(touchfirst[0] - t[0], touchfirst[1] - t[1]) < tapDistance) {\n          var p = select(this).on(\"dblclick.zoom\");\n          if (p) p.apply(this, arguments);\n        }\n      }\n    }\n  }\n\n  zoom.wheelDelta = function(_) {\n    return arguments.length ? (wheelDelta = typeof _ === \"function\" ? _ : constant(+_), zoom) : wheelDelta;\n  };\n\n  zoom.filter = function(_) {\n    return arguments.length ? (filter = typeof _ === \"function\" ? _ : constant(!!_), zoom) : filter;\n  };\n\n  zoom.touchable = function(_) {\n    return arguments.length ? (touchable = typeof _ === \"function\" ? _ : constant(!!_), zoom) : touchable;\n  };\n\n  zoom.extent = function(_) {\n    return arguments.length ? (extent = typeof _ === \"function\" ? _ : constant([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;\n  };\n\n  zoom.scaleExtent = function(_) {\n    return arguments.length ? (scaleExtent[0] = +_[0], scaleExtent[1] = +_[1], zoom) : [scaleExtent[0], scaleExtent[1]];\n  };\n\n  zoom.translateExtent = function(_) {\n    return arguments.length ? (translateExtent[0][0] = +_[0][0], translateExtent[1][0] = +_[1][0], translateExtent[0][1] = +_[0][1], translateExtent[1][1] = +_[1][1], zoom) : [[translateExtent[0][0], translateExtent[0][1]], [translateExtent[1][0], translateExtent[1][1]]];\n  };\n\n  zoom.constrain = function(_) {\n    return arguments.length ? (constrain = _, zoom) : constrain;\n  };\n\n  zoom.duration = function(_) {\n    return arguments.length ? (duration = +_, zoom) : duration;\n  };\n\n  zoom.interpolate = function(_) {\n    return arguments.length ? (interpolate = _, zoom) : interpolate;\n  };\n\n  zoom.on = function() {\n    var value = listeners.on.apply(listeners, arguments);\n    return value === listeners ? zoom : value;\n  };\n\n  zoom.clickDistance = function(_) {\n    return arguments.length ? (clickDistance2 = (_ = +_) * _, zoom) : Math.sqrt(clickDistance2);\n  };\n\n  zoom.tapDistance = function(_) {\n    return arguments.length ? (tapDistance = +_, zoom) : tapDistance;\n  };\n\n  return zoom;\n}\n","import { select } from 'd3-selection'\n\n/**\n * Parent class for button elements used to interact with the gridviz viewer.\n *\n * @module button\n * @author Joseph Davies, Julien Gaffuri\n */\nexport class Button {\n    /**\n     * @param {Object} opts\n     * opts.parentNode\n     * opts.id\n     * opts.title\n     * opts.class\n     * opts.onClickFunction\n     * opts.x\n     * opts.y\n     */\n    constructor(opts = {}) {\n        this.map = opts.map\n        this.parentNode = opts.parentNode || opts.map.container\n\n        // the div element\n        if (opts.id) this.div = select('#' + opts.id)\n\n        if (!this.div || this.div.empty()) {\n            this.div = select(document.createElement('div'))\n            if (opts.id) this.div.attr('id', opts.id)\n        }\n\n        if (opts.title) this.div.attr('title', opts.title)\n        if (opts.class) this.div.attr('class', opts.class)\n\n        // add events\n        if (opts.onClickFunction) this.div.on('click', opts.onClickFunction)\n\n        //set styles\n        this.style(\n            'box-shadow',\n            '0 7px 8px rgba(0,47,103,.08), 0 0 22px rgba(0,47,103,.04), 0 12px 17px rgba(0,47,103,.04), 0 -4px 4px rgba(0,47,103,.04)'\n        ) //.ecl-u-shadow-3\n        this.style('background-color', '#ffffff')\n        this.style('position', 'absolute')\n        this.style('cursor', 'pointer')\n        this.style('display', 'flex')\n        this.style('justify-content', 'center')\n        this.style('align-items', 'center')\n        this.style('width', '35px')\n        this.style('height', '30px')\n        // this.style(padding , '4px'\n\n        // append to parent\n        this.parentNode.appendChild(this.div.node())\n    }\n\n    /**\n     * Apply a style to the button div.\n     * @param {string} k\n     * @param {string} v\n     * @returns {this}\n     */\n    style(k, v) {\n        this.div.style(k, v)\n        return this\n    }\n}\n","import { Button } from './Button.js'\n\n/**\n * Button for toggling fullscreen mode\n *\n * @module button\n * @author Joseph Davies, Julien Gaffuri\n */\nexport class FullscreenButton extends Button {\n    /**\n     * @param {Object} opts\n     * opts.parentNode - the node that the button is appended to\n     * opts.canvas - the gridviz canvas\n     * opts.id\n     * opts.title - HTML title attribute\n     * opts.class - css class\n     * opts.onClickFunction\n     * opts.x - x position of the button\n     * opts.y - y position of the button\n     */\n\n    // default state\n    isFullscreen = false\n\n    constructor(opts) {\n        super(opts)\n\n        // append fullscreen icon to button container\n        this.div.node().innerHTML = `\n        <svg\n            style=\"height: 1.2rem; width: 1.2rem; fill:black; margin:0;\"\n            focusable=\"false\"\n            aria-hidden=\"true\"\n        >\n            <svg fill=\"#000000\" viewBox=\"0 0 96 96\" xmlns=\"http://www.w3.org/2000/svg\">\n            <title/>\n            <g>\n            <path d=\"M30,0H6A5.9966,5.9966,0,0,0,0,6V30a6,6,0,0,0,12,0V12H30A6,6,0,0,0,30,0Z\"/>\n            <path d=\"M90,0H66a6,6,0,0,0,0,12H84V30a6,6,0,0,0,12,0V6A5.9966,5.9966,0,0,0,90,0Z\"/>\n            <path d=\"M30,84H12V66A6,6,0,0,0,0,66V90a5.9966,5.9966,0,0,0,6,6H30a6,6,0,0,0,0-12Z\"/>\n            <path d=\"M90,60a5.9966,5.9966,0,0,0-6,6V84H66a6,6,0,0,0,0,12H90a5.9966,5.9966,0,0,0,6-6V66A5.9966,5.9966,0,0,0,90,60Z\"/>\n            </g>\n            </svg>\n        </svg>\n        `\n\n        //save initial map dimensions\n        this.defaultHeight = this.map.h\n        this.defaultWidth = this.map.w\n\n        // event handler\n        this.div.on('click', (e) => {\n            this.onClickFunction(e)\n        })\n        this.div.on('mouseover', (e) => {\n            this.style('background-color', 'lightgrey')\n        })\n        this.div.on('mouseout', (e) => {\n            this.style('background-color', '#ffffff')\n        })\n\n        //set position\n        if (opts.x) {\n            this.style('left', opts.x + 'px')\n        } else {\n            this.style('right', '10px')\n        }\n        if (opts.y) {\n            this.style('top', opts.y + 'px')\n        } else {\n            this.style('top', '90px')\n        }\n    }\n\n    onClickFunction(e) {\n        if (this.isFullscreen) {\n            this.closeFullscreen(this.map.container)\n            //resize canvas to default\n            this.map.h = this.defaultHeight\n            this.map.w = this.defaultWidth\n            this.map.geoCanvas.h = this.defaultHeight\n            this.map.geoCanvas.w = this.defaultWidth\n            this.map.geoCanvas.canvas.setAttribute('width', '' + this.defaultWidth)\n            this.map.geoCanvas.canvas.setAttribute('height', '' + this.defaultHeight)\n            this.map.redraw()\n            this.isFullscreen = false\n        } else {\n            this.openFullscreen(this.map.container)\n            //resize canvas to fullscreen\n            this.map.h = window.screen.height\n            this.map.w = window.screen.width\n            this.isFullscreen = true\n        }\n    }\n\n    /* Open fullscreen */\n    openFullscreen(elem) {\n        if (elem.requestFullscreen) {\n            elem.requestFullscreen()\n        } else if (elem.webkitRequestFullscreen) {\n            /* Safari */\n            elem.webkitRequestFullscreen()\n        } else if (elem.msRequestFullscreen) {\n            /* IE11 */\n            elem.msRequestFullscreen()\n        }\n    }\n\n    /* Close fullscreen */\n    closeFullscreen() {\n        if (document.exitFullscreen) {\n            document.exitFullscreen()\n        } else if (document.webkitExitFullscreen) {\n            /* Safari */\n            document.webkitExitFullscreen()\n        } else if (document.msExitFullscreen) {\n            /* IE11 */\n            document.msExitFullscreen()\n        }\n    }\n}\n","import { Button } from './Button.js'\r\n\r\n/**\r\n * Button for toggling fullscreen mode\r\n *\r\n * @module button\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class ZoomButtons extends Button {\r\n    /**\r\n     * @param {Object} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n\r\n        this.onZoom = opts.onZoom // custom user event handler\r\n        this.delta = opts.delta || 0.2\r\n\r\n        // Create zoom in button\r\n        this.zoomInBtn = document.createElement('a')\r\n        this.zoomInBtn.id = 'zoom-in'\r\n        this.zoomInBtn.className = 'gridviz-zoom-button'\r\n        this.zoomInBtn.title = 'Zoom in'\r\n        this.zoomInBtn.textContent = '+'\r\n        this.zoomInBtn.addEventListener('click', (e) => {\r\n            this.zoomIn(e)\r\n        })\r\n        this.zoomInBtn.addEventListener('mouseover', () => {\r\n            this.zoomInBtn.style.backgroundColor = 'lightgrey'\r\n        })\r\n        this.zoomInBtn.addEventListener('mouseout', () => {\r\n            this.zoomInBtn.style.backgroundColor = '#ffffff'\r\n        })\r\n\r\n        // Create zoom out button\r\n        this.zoomOutBtn = document.createElement('a')\r\n        this.zoomOutBtn.id = 'zoom-out'\r\n        this.zoomOutBtn.className = 'gridviz-zoom-button'\r\n        this.zoomOutBtn.title = 'Zoom out'\r\n        this.zoomOutBtn.textContent = '-'\r\n        this.zoomOutBtn.addEventListener('click', (e) => {\r\n            this.zoomOut(e)\r\n        })\r\n        this.zoomOutBtn.addEventListener('mouseover', () => {\r\n            this.zoomOutBtn.style.backgroundColor = 'lightgrey'\r\n        })\r\n        this.zoomOutBtn.addEventListener('mouseout', () => {\r\n            this.zoomOutBtn.style.backgroundColor = '#ffffff'\r\n        })\r\n\r\n        // Set common styles for buttons\r\n        const buttons = [this.zoomInBtn, this.zoomOutBtn]\r\n        buttons.forEach((btn, index) => {\r\n            btn.style.alignItems = 'center'\r\n            btn.style.justifyContent = 'center'\r\n            btn.style.display = 'flex'\r\n            btn.style.border = 'none'\r\n            btn.style.color = 'black'\r\n            btn.style.textAlign = 'center'\r\n            btn.style.textDecoration = 'none'\r\n            btn.style.padding = '4px'\r\n            btn.style.fontSize = '24px'\r\n            btn.style.fontWeight = 'bold'\r\n            btn.style.userSelect = 'none'\r\n            btn.style.backgroundColor = '#ffffff'\r\n            if (index === 0) btn.style.borderBottom = '1px solid grey' // Zoom in button only\r\n        })\r\n\r\n        // Unset parent class height and display for dual buttons\r\n        this.style('height', 'unset')\r\n        this.style('display', 'unset')\r\n\r\n        // Set position\r\n        if (opts.x) {\r\n            this.style('left', opts.x + 'px')\r\n        } else {\r\n            this.style('right', '10px')\r\n        }\r\n        if (opts.y) {\r\n            this.style('top', opts.y + 'px')\r\n        } else {\r\n            this.style('top', '10px')\r\n        }\r\n\r\n        // Append buttons to the container\r\n        this.div.node().appendChild(this.zoomInBtn)\r\n        this.div.node().appendChild(this.zoomOutBtn)\r\n    }\r\n\r\n    /* Zoom in */\r\n    zoomIn(e) {\r\n        this.map.setZoom(this.map.getZoom() * (1 - this.delta)).redraw()\r\n        if (this.onZoom) this.onZoom(e)\r\n    }\r\n\r\n    /* Zoom out */\r\n    zoomOut(e) {\r\n        this.map.setZoom(this.map.getZoom() * (1 + this.delta)).redraw()\r\n        if (this.onZoom) this.onZoom(e)\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n/**\r\n * A grid cell.\r\n * @typedef {{x: number, y: number}} Cell */\r\n\r\n/**\r\n * A dataset component, of grid cells.\r\n * @abstract\r\n *\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class Dataset {\r\n    /**\r\n     * @param {import(\"./Map.js\").Map} map The map.\r\n     * @param {string} url The URL of the dataset.\r\n     * @param {number} resolution The dataset resolution, in the CRS geographical unit.\r\n     * @param {{preprocess?:function(Cell):boolean, mixedResolution?:function(Cell):number}} opts\r\n     * @abstract\r\n     */\r\n    constructor(map, url, resolution, opts = {}) {\r\n        /**\r\n         * The map.\r\n         * @protected\r\n         * @type {import(\"./Map.js\").Map} */\r\n        this.map = map\r\n\r\n        /**\r\n         * The url of the dataset.\r\n         * @protected\r\n         * @type {string} */\r\n        this.url = url\r\n\r\n        /**\r\n         * The dataset resolution in geographical unit.\r\n         * @protected\r\n         * @type {number} */\r\n        this.resolution = resolution\r\n\r\n        /**\r\n         * In case the dataset is a dataset with cells having different resolution,\r\n         * this is the function returning the resolution of each cell.\r\n         * @protected\r\n         * @type {(function(Cell):number )| undefined } */\r\n        this.mixedResolution = opts.mixedResolution\r\n\r\n        /**\r\n         * A preprocess to run on each cell after loading. It can be used to apply some specific treatment before or compute a new column. And also to determine which cells to keep after loading.\r\n         * @type {(function(Cell):boolean )| undefined } */\r\n        this.preprocess = opts.preprocess || undefined\r\n\r\n        /** The cells within the view\r\n         * @protected\r\n         * @type {Array.<Cell>} */\r\n        this.cellsViewCache = []\r\n    }\r\n\r\n    /**\r\n     * Request data within a geographic envelope.\r\n     *\r\n     * @abstract\r\n     * @param {import(\"./GeoCanvas\").Envelope|undefined} extGeo\r\n     * @returns {this}\r\n     */\r\n    getData(extGeo = undefined) {\r\n        throw new Error('Method getData not implemented.')\r\n    }\r\n\r\n    /**\r\n     * Fill the view cache with all cells which are within a geographical envelope.\r\n     * @abstract\r\n     * @param {import(\"./GeoCanvas\").Envelope} extGeo The view geographical envelope.\r\n     * @returns {void}\r\n     */\r\n    updateViewCache(extGeo) {\r\n        throw new Error('Method updateViewCache not implemented.')\r\n    }\r\n\r\n    /**\r\n     * Get a cell under a given position, if any.\r\n     *\r\n     * @param {{x:number,y:number}} posGeo\r\n     * @param {Array.<Cell>} cells Some cells from the dataset (a subset if necessary, usually the view cache).\r\n     * @returns {Cell|undefined}\r\n     */\r\n    getCellFromPosition(posGeo, cells) {\r\n        //compute candidate cell position\r\n        /** @type {number} */\r\n        //const r = this.getResolution()\r\n        /** @type {number} */\r\n        //const cellX = r * Math.floor(posGeo.x / r)\r\n        /** @type {number} */\r\n        //const cellY = r * Math.floor(posGeo.y / r)\r\n\r\n        /*/get cell\r\n        for (const cell of cells) {\r\n            if (cell.x != cellX) continue\r\n            if (cell.y != cellY) continue\r\n            return cell\r\n        }\r\n        return undefined*/\r\n\r\n        //rare case of mixed resolution dataset\r\n        if (this.mixedResolution) {\r\n            for (const c of cells) {\r\n                /** @type {number} */\r\n                const r = +this.mixedResolution(c)\r\n                if (posGeo.x < c.x) continue\r\n                else if (c.x + r < posGeo.x) continue\r\n                else if (posGeo.y < c.y) continue\r\n                else if (c.y + r < posGeo.y) continue\r\n                else return c\r\n            }\r\n            return undefined\r\n        }\r\n\r\n        //common case\r\n\r\n        /** @type {number} */\r\n        const r = this.getResolution()\r\n        for (const cell of cells) {\r\n            if (posGeo.x < cell.x) continue\r\n            else if (cell.x + r < posGeo.x) continue\r\n            else if (posGeo.y < cell.y) continue\r\n            else if (cell.y + r < posGeo.y) continue\r\n            else return cell\r\n        }\r\n        return undefined\r\n    }\r\n\r\n    //getters and setters\r\n\r\n    /** @returns {number} */\r\n    getResolution() {\r\n        return this.resolution\r\n    }\r\n\r\n    /** @returns {Array.<Cell>} */\r\n    getViewCache() {\r\n        return this.cellsViewCache\r\n    }\r\n\r\n    /**\r\n     * Return the relevant dataset for a specified zoom.\r\n     * @param {number} z\r\n     * @param {number} minPixelsPerCell\r\n     * @returns {Dataset|undefined}\r\n     * */\r\n    getDataset(z, minPixelsPerCell) {\r\n        return this\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n/**\r\n * This is an abstract class used to group elements shared between Layer and Style classes.\r\n *\r\n * @abstract\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class Drawable {\r\n    /**\r\n     * @param {object} opts\r\n     */\r\n    constructor(opts) {\r\n        opts = opts || {}\r\n\r\n        /** A function specifying if the element should be visible or not.\r\n         * The function parameter is the zoom level.\r\n         * @type {function(number):boolean} */\r\n        this.visible = opts.visible\r\n\r\n        /** A function returning the alpha (transparency/opacity), between 0.0 (fully transparent) and 1.0 (fully opaque).\r\n         *  The function parameter is the zoom level.\r\n         * (see CanvasRenderingContext2D: globalAlpha property)\r\n         * @type {(function(number):number)|undefined} */\r\n        this.alpha = opts.alpha\r\n\r\n        /** A function returning the blend operation.\r\n         * The function parameter is the zoom level.\r\n         * (see CanvasRenderingContext2D: globalCompositeOperation property)\r\n         * @type {function(number):GlobalCompositeOperation} */\r\n        this.blendOperation = opts.blendOperation || ((z) => 'source-over')\r\n\r\n        /** @type {(function(number):string)|undefined} */\r\n        this.filterColor = opts.filterColor // (z) => \"#eee7\"\r\n    }\r\n\r\n    /**\r\n     * Draw layer filter.\r\n     *\r\n     * @param {import(\"./GeoCanvas.js\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @returns {void}\r\n     * @abstract\r\n     */\r\n    drawFilter(geoCanvas) {\r\n        //no filter: return\r\n        if (!this.filterColor) return\r\n\r\n        //get filter\r\n        const fc = this.filterColor(geoCanvas.view.z)\r\n\r\n        //no filter: return\r\n        if (!fc || fc == 'none') return\r\n\r\n        //draw filter\r\n        geoCanvas.offscreenCtx.fillStyle = fc\r\n        geoCanvas.offscreenCtx.fillRect(0, 0, geoCanvas.w, geoCanvas.h)\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n/** @typedef { {xMin: number, xMax: number, yMin: number, yMax: number} } Envelope */\r\n\r\n/**\r\n * A viewshed.\r\n * @typedef {{x: number, y: number, z: number}} View */\r\n\r\nimport { select } from 'd3-selection'\r\nimport { zoom as d3zoom, zoomIdentity } from 'd3-zoom'\r\n\r\n/**\r\n * A HTML canvas for geo data display, enhanced with zoom and pan capabilities.\r\n *\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class GeoCanvas {\r\n    /**\r\n     * @constructor\r\n     * @param {HTMLCanvasElement} canvas\r\n     * @param {number} x The x coordinate of the view\r\n     * @param {number} y The y coordinate of the view\r\n     * @param {number} z The zoom level of the view (pixel size, in ground m)\r\n     * @param {object} opts\r\n     */\r\n    constructor(canvas, x = 0, y = 0, z = 0, opts = undefined) {\r\n        this.opts = opts || {}\r\n\r\n        /** @type {HTMLCanvasElement} */\r\n        this.canvas = canvas\r\n\r\n        /** @type {number} */\r\n        this.w = this.canvas.offsetWidth\r\n        /** @type {number} */\r\n        this.h = this.canvas.offsetHeight\r\n\r\n        // Adjust canvas width and height based on device pixel ratio\r\n        //const dpr = window.devicePixelRatio || 1 // Get the device pixel ratio\r\n        //this.canvas.width = this.w * dpr // Set canvas width\r\n        //\\sthis.canvas.height = this.h * dpr // Set canvas height\r\n\r\n        // Create offscreen canvas for drawing operations\r\n        this.offscreenCanvas = document.createElement('canvas')\r\n        this.offscreenCanvas.width = this.w\r\n        this.offscreenCanvas.height = this.h\r\n\r\n        const ctx = this.canvas.getContext('2d')\r\n        const offscreenCtx = this.offscreenCanvas.getContext('2d')\r\n        if (!ctx) throw 'Impossible to create canvas 2D context'\r\n        if (!offscreenCtx) throw 'Impossible to create canvas 2D context'\r\n        /**@type {CanvasRenderingContext2D} */\r\n        this.ctx = ctx\r\n        this.offscreenCtx = offscreenCtx\r\n        //this.ctx.scale(dpr, dpr) // Scale the context\r\n\r\n        /**\r\n         * z: pixel size, in m/pix\r\n         * @type {View}  */\r\n        this.view = { x: x, y: y, z: z }\r\n\r\n        /** Background color.\r\n         * @type {string} */\r\n        this.backgroundColor = opts.backgroundColor || 'white'\r\n\r\n        /** @type {function(object|undefined):void} */\r\n        this.onZoomStartFun = opts.onZoomStartFun\r\n\r\n        /** @type {function(object|undefined):void} */\r\n        this.onZoomEndFun = opts.onZoomEndFun\r\n\r\n        /** @type {function(object|undefined):void} */\r\n        this.onZoomFun = opts.onZoomFun\r\n\r\n        //current extent\r\n        /** @type {Envelope} */\r\n        this.extGeo = { xMin: NaN, xMax: NaN, yMin: NaN, yMax: NaN }\r\n        this.updateExtentGeo()\r\n\r\n        //rely on d3 zoom for pan/zoom\r\n        if (!opts.disableZoom) {\r\n            let tP = zoomIdentity\r\n            const z = d3zoom()\r\n                //to make the zooming a bit faster\r\n                .wheelDelta((e) => -e.deltaY * (e.deltaMode === 1 ? 0.07 : e.deltaMode ? 1 : 0.004))\r\n                .on('zoom', (e) => {\r\n                    const t = e.transform\r\n                    const f = tP.k / t.k\r\n\r\n                    // Get container's bounding rect to adjust for offsets\r\n                    const containerRect = this.canvas.getBoundingClientRect()\r\n\r\n                    // Adjust for container's offset and zoom center\r\n                    const offsetX = e.sourceEvent.offsetX - containerRect.left\r\n                    const offsetY = e.sourceEvent.offsetY - containerRect.top\r\n\r\n                    if (f === 1) {\r\n                        // Pan logic\r\n                        const dx = tP.x - t.x\r\n                        const dy = tP.y - t.y\r\n                        this.pan(dx * this.view.z, -dy * this.view.z)\r\n                    } else {\r\n                        const se = e.sourceEvent\r\n\r\n                        if (se instanceof WheelEvent) {\r\n                            // Zoom at mouse position, adjusted by container offset\r\n                            this.zoom(f, this.pixToGeoX(offsetX), this.pixToGeoY(offsetY))\r\n                        } else if (se instanceof TouchEvent) {\r\n                            if (!se.targetTouches.length) return\r\n\r\n                            // Compute average position of the touches\r\n                            let tx = 0,\r\n                                ty = 0\r\n                            for (let tt of se.targetTouches) {\r\n                                tx += tt.clientX\r\n                                ty += tt.clientY\r\n                            }\r\n                            tx /= se.targetTouches.length\r\n                            ty /= se.targetTouches.length\r\n\r\n                            // Adjust for container's offset\r\n                            tx -= containerRect.left\r\n                            ty -= containerRect.top\r\n\r\n                            // Zoom at the average touch position\r\n                            this.zoom(f, this.pixToGeoX(tx), this.pixToGeoY(ty))\r\n                        }\r\n                    }\r\n                    tP = t\r\n\r\n                    if (this.onZoomFun) this.onZoomFun(e)\r\n                })\r\n                .on('start', (e) => {\r\n                    // start of zoom event\r\n                    // save the current canvas state to keep onscreen during pan/zoom before redrawing\r\n                    this.canvasSave.c = document.createElement('canvas')\r\n                    this.canvasSave.c.setAttribute('width', '' + this.w)\r\n                    this.canvasSave.c.setAttribute('height', '' + this.h)\r\n                    this.canvasSave.c.getContext('2d')?.drawImage(this.canvas, 0, 0)\r\n                    this.canvasSave.dx = 0\r\n                    this.canvasSave.dy = 0\r\n                    this.canvasSave.f = 1\r\n                    if (this.onZoomStartFun) this.onZoomStartFun(e)\r\n                })\r\n                .on('end', (e) => {\r\n                    // end of zoom event\r\n                    this.redraw()\r\n                    this.canvasSave = { c: null, dx: 0, dy: 0, f: 1 }\r\n\r\n                    if (this.onZoomEndFun) this.onZoomEndFun(e)\r\n                })\r\n            z(select(this.canvas))\r\n        }\r\n\r\n        //center extent\r\n        /** @type {number|undefined} */\r\n        this.xMin = opts.centerExtent ? opts.centerExtent[0] : undefined\r\n        /** @type {number|undefined} */\r\n        this.yMin = opts.centerExtent ? opts.centerExtent[1] : undefined\r\n        /** @type {number|undefined} */\r\n        this.xMax = opts.centerExtent ? opts.centerExtent[2] : undefined\r\n        /** @type {number|undefined} */\r\n        this.yMax = opts.centerExtent ? opts.centerExtent[3] : undefined\r\n\r\n        /** Zoom extent, to limit zoom in and out\r\n         *  @type {Array.<number>} */\r\n        this.zoomExtent = opts.zoomExtent || [0, Infinity]\r\n\r\n        /** Canvas state, to be used to avoid unnecessary redraws on zoom/pan\r\n         *  @type {{c:HTMLCanvasElement|null,dx:number,dy:number,f:number}} */\r\n        this.canvasSave = { c: null, dx: 0, dy: 0, f: 1 }\r\n    }\r\n\r\n    /** @returns {View} */\r\n    getView() {\r\n        return this.view\r\n    }\r\n\r\n    /** @param {Array.<number>} v */\r\n    setCenterExtent(v) {\r\n        this.xMin = v[0]\r\n        this.yMin = v[1]\r\n        this.xMax = v[2]\r\n        this.yMax = v[3]\r\n    }\r\n    /** @returns {Array.<number|undefined>} */\r\n    getCenterExtent() {\r\n        return [this.xMin, this.yMin, this.xMax, this.yMax]\r\n    }\r\n\r\n    /** @param {Array.<number>} v */\r\n    setZoomExtent(v) {\r\n        this.zoomExtent = v\r\n    }\r\n    /** @returns {Array.<number>} */\r\n    getZoomExtent() {\r\n        return this.zoomExtent\r\n    }\r\n\r\n    /** Initialise canvas transform with identity transformation. */\r\n    initCanvasTransform() {\r\n        this.ctx.setTransform(1, 0, 0, 1, 0, 0)\r\n        this.offscreenCtx.setTransform(1, 0, 0, 1, 0, 0)\r\n    }\r\n\r\n    /** Initialise canvas transform with geo to screen transformation, so that geo objects can be drawn directly in geo coordinates. */\r\n    setCanvasTransform() {\r\n        const k = 1 / this.view.z\r\n        const tx = -this.view.x / this.view.z + this.w * 0.5\r\n        const ty = this.view.y / this.view.z + this.h * 0.5\r\n        this.ctx.setTransform(k, 0, 0, -k, tx, ty)\r\n        this.offscreenCtx.setTransform(k, 0, 0, -k, tx, ty)\r\n    }\r\n\r\n    /** Get the transformation matrix to webGL screen coordinates, within [-1,1]*[-1,1] */\r\n    getWebGLTransform() {\r\n        const kx = 2.0 / (this.w * this.view.z)\r\n        const ky = 2.0 / (this.h * this.view.z)\r\n        return [kx, 0.0, 0.0, 0.0, ky, 0.0, -kx * this.view.x, -ky * this.view.y, 1.0]\r\n    }\r\n\r\n    /** The function specifying how to draw the map. */\r\n    redraw() {\r\n        throw new Error('Method redraw not implemented.')\r\n    }\r\n\r\n    /**\r\n     * Clear. To be used before a redraw for example.\r\n     * @param {string} color\r\n     */\r\n    clear(color = 'white') {\r\n        if (this.opts.transparentBackground) {\r\n            this.ctx.clearRect(0, 0, this.w, this.h)\r\n            this.offscreenCtx.clearRect(0, 0, this.w, this.h)\r\n        } else {\r\n            if (this.ctx) this.ctx.fillStyle = color\r\n            if (this.offscreenCtx) this.offscreenCtx.fillStyle = color\r\n            this.ctx.fillRect(0, 0, this.w, this.h)\r\n            this.offscreenCtx.fillRect(0, 0, this.w, this.h)\r\n        }\r\n    }\r\n\r\n    /**\r\n     * @param {number} dxGeo\r\n     * @param {number} dyGeo\r\n     */\r\n    pan(dxGeo = 0, dyGeo = 0) {\r\n        //ensures x/y extent\r\n        if (this.xMin != undefined && this.view.x + dxGeo < this.xMin) dxGeo = this.xMin - this.view.x\r\n        if (this.yMin != undefined && this.view.y + dyGeo < this.yMin) dyGeo = this.yMin - this.view.y\r\n        if (this.xMax != undefined && this.view.x + dxGeo > this.xMax) dxGeo = this.xMax - this.view.x\r\n        if (this.yMax != undefined && this.view.y + dyGeo > this.yMax) dyGeo = this.yMax - this.view.y\r\n\r\n        //pan\r\n        this.view.x += dxGeo\r\n        this.view.y += dyGeo\r\n        this.updateExtentGeo()\r\n\r\n        if (this.canvasSave.c) {\r\n            const scale = 1 / this.view.z\r\n\r\n            // Update saved canvas offset\r\n            this.canvasSave.dx -= dxGeo * scale\r\n            this.canvasSave.dy += dyGeo * scale\r\n\r\n            // clear canvas\r\n            this.clear(this.backgroundColor)\r\n\r\n            // this doesnt work on mobile https://github.com/eurostat/gridviz/issues/98\r\n            //this.ctx.drawImage(this.canvasSave.c, this.canvasSave.dx, this.canvasSave.dy)\r\n            this.offscreenCtx.drawImage(this.canvasSave.c, this.canvasSave.dx, this.canvasSave.dy)\r\n\r\n            // Render the offscreen canvas to the visible context\r\n            // this.clear(this.backgroundColor)\r\n            this.ctx.drawImage(this.offscreenCtx.canvas, 0, 0)\r\n        } else {\r\n            console.log('no canvas save')\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Zoom.\r\n     * @param {number} f The zoom factor, within ]0, Infinity]. 1 is for no change. <1 to zoom-in, >1 to zoom-out.\r\n     * @param {number} xGeo The x geo position fixed in the screen.\r\n     * @param {number} yGeo The y geo position fixed in the screen.\r\n     */\r\n    zoom(f = 1, xGeo = this.view.x, yGeo = this.view.y) {\r\n        //TODO force geo extend to remain\r\n\r\n        //trying to zoom in/out beyond limit\r\n        if (this.zoomExtent[0] == this.view.z && f <= 1) return\r\n        if (this.zoomExtent[1] == this.view.z && f >= 1) return\r\n\r\n        //ensure zoom extent preserved\r\n        const newZf = f * this.view.z\r\n        if (newZf < this.zoomExtent[0]) f = this.zoomExtent[0] / this.view.z\r\n        if (newZf > this.zoomExtent[1]) f = this.zoomExtent[1] / this.view.z\r\n\r\n        this.view.z *= f\r\n\r\n        //compute pan\r\n        let dxGeo = (xGeo - this.view.x) * (1 - f)\r\n        let dyGeo = (yGeo - this.view.y) * (1 - f)\r\n\r\n        //ensures x/y extent\r\n        if (this.xMin != undefined && this.view.x + dxGeo < this.xMin) dxGeo = this.xMin - this.view.x\r\n        if (this.yMin != undefined && this.view.y + dyGeo < this.yMin) dyGeo = this.yMin - this.view.y\r\n        if (this.xMax != undefined && this.view.x + dxGeo > this.xMax) dxGeo = this.xMax - this.view.x\r\n        if (this.yMax != undefined && this.view.y + dyGeo > this.yMax) dyGeo = this.yMax - this.view.y\r\n\r\n        //pan\r\n        this.view.x += dxGeo\r\n        this.view.y += dyGeo\r\n        this.updateExtentGeo()\r\n\r\n        // zoom in on the current canvas state\r\n        if (this.canvasSave.c) {\r\n            this.clear(this.backgroundColor)\r\n            this.canvasSave.f /= f\r\n            this.canvasSave.dx = this.geoToPixX(xGeo) * (1 - this.canvasSave.f)\r\n            this.canvasSave.dy = this.geoToPixY(yGeo) * (1 - this.canvasSave.f)\r\n            this.offscreenCtx.drawImage(\r\n                this.canvasSave.c,\r\n                this.canvasSave.dx,\r\n                this.canvasSave.dy,\r\n                this.canvasSave.f * this.canvasSave.c.width,\r\n                this.canvasSave.f * this.canvasSave.c.height\r\n            )\r\n            this.ctx.drawImage(\r\n                this.offscreenCanvas, // Use offscreen canvas as the source\r\n                0,\r\n                0, // Position the offscreen canvas at the top-left corner of the main canvas\r\n                this.canvas.width, // The width of the visible canvas\r\n                this.canvas.height // The height of the visible canvas\r\n            )\r\n        }\r\n    }\r\n\r\n    /**\r\n     * @param {number} marginPx\r\n     * @returns {Envelope} The envelope of the view, in geo coordinates.\r\n     */\r\n    updateExtentGeo(marginPx = 20) {\r\n        this.extGeo = {\r\n            xMin: this.pixToGeoX(-marginPx),\r\n            xMax: this.pixToGeoX(this.w + marginPx),\r\n            yMin: this.pixToGeoY(this.h + marginPx),\r\n            yMax: this.pixToGeoY(-marginPx),\r\n        }\r\n        return this.extGeo\r\n    }\r\n\r\n    /**\r\n     * Check if the object has to be drawn\r\n     *\r\n     * @param {{x:number,y:number}} obj\r\n     */\r\n    toDraw(obj) {\r\n        if (obj.x < this.extGeo.xMin) return false\r\n        if (obj.x > this.extGeo.xMax) return false\r\n        if (obj.y < this.extGeo.yMin) return false\r\n        if (obj.y > this.extGeo.yMax) return false\r\n        return true\r\n    }\r\n\r\n    //conversion functions\r\n    /**\r\n     * @param {number} xGeo Geo x coordinate, in m.\r\n     * @returns {number} Screen x coordinate, in pix.\r\n     */\r\n    geoToPixX(xGeo) {\r\n        return (xGeo - this.view.x) / this.view.z + this.w * 0.5\r\n    }\r\n    /**\r\n     * @param {number} yGeo Geo y coordinate, in m.\r\n     * @returns {number} Screen y coordinate, in pix.\r\n     */\r\n    geoToPixY(yGeo) {\r\n        return -(yGeo - this.view.y) / this.view.z + this.h * 0.5\r\n    }\r\n    /**\r\n     * @param {number} x Screen x coordinate, in pix.\r\n     * @returns {number} Geo x coordinate, in m.\r\n     */\r\n    pixToGeoX(x) {\r\n        return (x - this.w * 0.5) * this.view.z + this.view.x\r\n    }\r\n    /**\r\n     * @param {number} y Screen y coordinate, in pix.\r\n     * @returns {number} Geo y coordinate, in m.\r\n     */\r\n    pixToGeoY(y) {\r\n        return -(y - this.h * 0.5) * this.view.z + this.view.y\r\n    }\r\n\r\n    /** Get x,y,z elements from URL and assign them to the view. */\r\n    setViewFromURL() {\r\n        const x = GeoCanvas.getParameterByName('x'),\r\n            y = GeoCanvas.getParameterByName('y'),\r\n            z = GeoCanvas.getParameterByName('z')\r\n        if (x != null && x != undefined && !isNaN(+x)) this.view.x = +x\r\n        if (y != null && y != undefined && !isNaN(+y)) this.view.y = +y\r\n        if (z != null && z != undefined && !isNaN(+z)) this.view.z = +z\r\n    }\r\n\r\n    /**\r\n     * Get a URL parameter by name.\r\n     *\r\n     * @param {string} name\r\n     * @returns {string | null}\r\n     */\r\n    static getParameterByName(name) {\r\n        name = name.replace(/[\\[]/, '\\\\[').replace(/[\\]]/, '\\\\]')\r\n        var regex = new RegExp('[\\\\?&]' + name + '=([^&#]*)'),\r\n            results = regex.exec(location.search)\r\n        return !results ? null : decodeURIComponent(results[1].replace(/\\+/g, ' '))\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Drawable } from './Drawable.js'\r\n\r\n/**\r\n * @module core\r\n * @abstract\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class Layer extends Drawable {\r\n    /**\r\n     * Draw layer.\r\n     *\r\n     * @param {import(\"./GeoCanvas\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @param {object} legend\r\n     * @returns {void}\r\n     * @abstract\r\n     */\r\n    draw(geoCanvas, legend = undefined) {\r\n        throw new Error('Method draw not implemented.')\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { select } from 'd3-selection'\r\n\r\n/**\r\n * A legend container.\r\n *\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class Legend {\r\n    /**\r\n     * @param {Object} opts\r\n     */\r\n    constructor(opts) {\r\n        opts = opts || {}\r\n\r\n        /** @type {string} */\r\n        this.id = opts.id\r\n\r\n        //TODO stop using it. Use style method below instead.\r\n\r\n        /** @type {number} @deprecated */\r\n        this.top = opts.top\r\n        /** @type {number} @deprecated */\r\n        this.bottom = opts.bottom\r\n        /** @type {number} @deprecated */\r\n        this.left = opts.left\r\n        /** @type {number} @deprecated */\r\n        this.right = opts.right\r\n        /** @type {string} @deprecated */\r\n        this.background = opts.background || 'none'\r\n        /** @type {string} @deprecated */\r\n        this.padding = opts.padding || '5px'\r\n        /** @type {string} @deprecated */\r\n        this.border = opts.border || '0px'\r\n        /** @type {string} @deprecated */\r\n        this['border-radius'] = opts['border-radius'] || 'none'\r\n        /** @type {string} @deprecated */\r\n        this['box-shadow'] = opts['box-shadow'] || 'none'\r\n        /** @type {string} @deprecated */\r\n        this['font-family'] = opts['font-family'] || 'Helvetica, Arial, sans-serif'\r\n        /** @type {string} @deprecated */\r\n        //this.width = opts.width\r\n        /** @type {string} @deprecated */\r\n        //this.height = opts.height\r\n\r\n        //the div element\r\n        if (this.id) this.div = select('#' + this.id)\r\n\r\n        if (!this.div || this.div.empty()) {\r\n            this.div = select(document.createElement('div'))\r\n            if (this.id) this.div.attr('id', this.id)\r\n        }\r\n\r\n        //set style\r\n        this.div.style('background', this.background)\r\n        this.div.style('padding', this.padding)\r\n        this.div.style('border', this.border)\r\n        this.div.style('border-radius', this['border-radius'])\r\n        this.div.style('box-shadow', this['box-shadow'])\r\n        this.div.style('font-family', this['font-family'])\r\n\r\n        //if (this.width) this.div.style('width', this.width)\r\n        //if (this.height) this.div.style('height', this.height)\r\n\r\n        //title\r\n        this.title = opts.title\r\n        this.titleFontSize = opts.titleFontSize || '0.8em'\r\n        this.titleFontWeight = opts.titleFontWeight || 'bold'\r\n\r\n        //label\r\n        this.labelFontSize = opts.labelFontSize || '0.8em'\r\n        this.labelUnitText = opts.labelUnitText || ''\r\n        this.labelFormat = opts.labelFormat\r\n    }\r\n\r\n    makeTitle() {\r\n        if (!this.title) return\r\n        this.div\r\n            .append('div')\r\n            .style('font-size', this.titleFontSize)\r\n            .style('font-weight', this.titleFontWeight)\r\n            .style('margin-bottom', '7px')\r\n            .text(this.title)\r\n    }\r\n\r\n    /**\r\n     * Apply a style to the legend div.\r\n     * @param {string} k\r\n     * @param {string} v\r\n     * @returns {this|string}\r\n     */\r\n    style(k, v) {\r\n        if (arguments.length == 1) return this.div.style(k)\r\n        this.div.style(k, v)\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * @param {Object} opts\r\n     * @abstract\r\n     */\r\n    update(opts = {}) {\r\n        console.error('Legend update not implemented yet.')\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n// internal imports\r\nimport { GeoCanvas } from './GeoCanvas.js'\r\nimport { Tooltip } from './Tooltip.js'\r\nimport { ZoomButtons } from '../button/ZoomButtons.js'\r\nimport { FullscreenButton } from '../button/FullscreenButton.js'\r\n\r\n// external imports\r\nimport { select } from 'd3-selection'\r\n\r\n/**\r\n * A gridviz application.\r\n *\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class Map {\r\n    /**\r\n     * @param {HTMLDivElement} container\r\n     * @param {object} opts\r\n     */\r\n    constructor(container, opts) {\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * The layers.\r\n         * @type {Array.<import(\"./Layer.js\").Layer>}\r\n         * */\r\n        this.layers = opts.layers || []\r\n\r\n        //get container element\r\n        this.container = container || document.getElementById('gridviz')\r\n        if (!this.container) {\r\n            console.error('Cannot find gridviz container element.')\r\n            return\r\n        }\r\n\r\n        //https://css-tricks.com/absolute-positioning-inside-relative-positioning/\r\n        this.container.style.position = 'relative' // container element must have relative positioning\r\n\r\n        //set dimensions\r\n        /** @type {number} */\r\n        this.w = opts.w || this.container.offsetWidth\r\n        /** @type {number} */\r\n        this.h = opts.h || this.container.offsetHeight\r\n\r\n        // Create the main canvas (for rendering to screen)\r\n        /** @type {HTMLCanvasElement} */\r\n        this._canvas = opts.canvas || this.initialiseCanvas()\r\n\r\n        /**  Initialize GeoCanvas\r\n         * @type {GeoCanvas}\r\n         * @private */\r\n        this.geoCanvas = new GeoCanvas(this._canvas, opts.x, opts.y, opts.z, opts)\r\n\r\n        this.geoCanvas.redraw = () => {\r\n            this.redraw()\r\n        }\r\n\r\n        // legend div\r\n        this.legend = opts.legendContainer\r\n            ? select(opts.legendContainer) // Wrap the provided HTML node in a D3 selection\r\n            : null\r\n        if (!this.legend) this.initialiseLegend()\r\n\r\n        //tooltip\r\n\r\n        // set App container as default parent element for tooltip\r\n        if (!opts.tooltip) opts.tooltip = {}\r\n        if (!opts.tooltip.parentElement) opts.tooltip.parentElement = this.container\r\n\r\n        /**\r\n         * @private\r\n         * @type {Tooltip} */\r\n        this.tooltip = new Tooltip(opts.tooltip)\r\n\r\n        // add event listeners to container\r\n        this.mouseOverHandler = (e) => this.focusCell(e)\r\n        this.mouseMoveHandler = (e) => this.focusCell(e)\r\n        this.mouseOutHandler = (e) => this.tooltip.hide()\r\n        this.geoCanvas.canvas.addEventListener('mouseover', this.mouseOverHandler)\r\n        this.geoCanvas.canvas.addEventListener('mousemove', this.mouseMoveHandler)\r\n        this.geoCanvas.canvas.addEventListener('mouseout', this.mouseOutHandler)\r\n\r\n        // listen for resize events on the App's container and handle them\r\n        this.defineResizeObserver()\r\n\r\n        // add extra logic to onZoomStartFun\r\n        this.geoCanvas.onZoomStartFun = (e) => {\r\n            if (opts.onZoomStartFun) opts.onZoomStartFun(e)\r\n            this.tooltip.hide()\r\n        }\r\n\r\n        //for mouse over\r\n        /**\r\n         * @private\r\n         * @type {HTMLCanvasElement|null} */\r\n        this.canvasSave = null\r\n\r\n        this.selectionRectangleColor = opts.selectionRectangleColor || '#FF6347'\r\n        this.selectionRectangleWidthPix = opts.selectionRectangleWidthPix || (() => 4) //(r,z) => {}\r\n\r\n        // transparent background (e.g. leaflet) 'red painting' fix\r\n        this.transparentBackground = opts.transparentBackground\r\n\r\n        //set default globalCompositeOperation\r\n        this.defaultGlobalCompositeOperation =\r\n            opts.defaultGlobalCompositeOperation || this.geoCanvas.ctx.globalCompositeOperation\r\n    }\r\n\r\n    /**\r\n     * @protected\r\n     * @returns {HTMLCanvasElement}\r\n     */\r\n    initialiseCanvas() {\r\n        const canvas = document.createElement('canvas')\r\n        canvas.setAttribute('width', '' + this.w)\r\n        canvas.setAttribute('height', '' + this.h)\r\n        this.container.appendChild(canvas)\r\n        return canvas\r\n    }\r\n\r\n    initialiseLegend() {\r\n        this.legend = select(this.container)\r\n            .append('div') // Create a new container\r\n            .attr('id', 'gridviz-legend')\r\n            .style('position', 'absolute')\r\n            .style('width', 'auto')\r\n            .style('height', 'auto')\r\n            .style('background', '#FFFFFF')\r\n            //.style(\"padding\", this.padding)\r\n            .style('border', '0px')\r\n            //.style('border-radius', '5px')\r\n            .style('box-shadow', '3px 3px 3px grey, -3px -3px 3px #ddd')\r\n            .style('font-family', 'Helvetica, Arial, sans-serif')\r\n            .style('bottom', '15px')\r\n            .style('right', '15px')\r\n        //hide\r\n        //.style(\"visibility\", \"hidden\")\r\n    }\r\n\r\n    /**\r\n     * Set/get layer stack.\r\n     *\r\n     * @param {undefined|import(\"./Layer.js\").Layer|import(\"./Layer.js\").Layer[]} layers\r\n     * @returns { this | import(\"./Layer.js\").Layer[] }\r\n     */\r\n    layers_(layers) {\r\n        if (arguments.length === 0) return this.layers\r\n        if (arguments.length === 1)\r\n            if (Array.isArray(layers)) this.layers = layers\r\n            else this.layers = [layers]\r\n        else this.layers = arguments\r\n        return this\r\n    }\r\n\r\n    /** @returns {this} */\r\n    redraw() {\r\n        //remove legend elements\r\n        if (this.legend) this.legend.selectAll('*').remove()\r\n\r\n        //clear\r\n        this.geoCanvas.initCanvasTransform()\r\n        this.geoCanvas.clear(this.geoCanvas.backgroundColor)\r\n\r\n        const z = this.geoCanvas.view.z\r\n        this.updateExtentGeo()\r\n\r\n        //go through the layers\r\n        for (const layer of this.layers) {\r\n            //check if layer is visible\r\n            if (layer.visible && !layer.visible(z)) continue\r\n\r\n            //set layer alpha and blend mode\r\n            this.geoCanvas.offscreenCtx.globalAlpha = layer.alpha ? layer.alpha(z) : 1.0\r\n            if (layer.blendOperation) this.geoCanvas.offscreenCtx.globalCompositeOperation = layer.blendOperation(z)\r\n\r\n            //set affin transform to draw with geographical coordinates\r\n            this.geoCanvas.setCanvasTransform()\r\n\r\n            //draw layer\r\n            layer.draw(this.geoCanvas, this.legend)\r\n\r\n            //draw layer filter\r\n            if (layer.filterColor) layer.drawFilter(this.geoCanvas)\r\n\r\n            //restore default alpha and blend operation\r\n            this.geoCanvas.offscreenCtx.globalAlpha = 1.0\r\n            this.geoCanvas.offscreenCtx.globalCompositeOperation = this.defaultGlobalCompositeOperation\r\n        }\r\n\r\n        // one drawImage call: draw the offscreen canvas to the main canvas\r\n        this.geoCanvas.initCanvasTransform()\r\n        this.geoCanvas.ctx.drawImage(this.geoCanvas.offscreenCanvas, 0, 0)\r\n\r\n        this.canvasSave = null\r\n\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * @param {number} marginPx\r\n     * @returns {import('./GeoCanvas.js').Envelope}\r\n     * @public\r\n     */\r\n    updateExtentGeo(marginPx = 20) {\r\n        return this.geoCanvas.updateExtentGeo(marginPx)\r\n    }\r\n\r\n    /** @param {MouseEvent} e */\r\n    focusCell(e) {\r\n        //compute mouse geo position\r\n        const mousePositionGeo = {\r\n            x: this.geoCanvas.pixToGeoX(e.offsetX + this.tooltip.xMouseOffset),\r\n            y: this.geoCanvas.pixToGeoY(e.offsetY + this.tooltip.yMouseOffset),\r\n        }\r\n        /** @type {{cell:import('./Dataset.js').Cell,html:string,resolution:number} | undefined} */\r\n        const focus = this.getCellFocusInfo(mousePositionGeo)\r\n\r\n        // transparent background (e.g. leaflet) 'red painting' fix\r\n        if (this.transparentBackground) {\r\n            if (focus) {\r\n                this.tooltip.html(focus.html)\r\n                this.tooltip.setPosition(e)\r\n                this.tooltip.show()\r\n            } else {\r\n                this.tooltip.hide()\r\n            }\r\n            this.canvasSave = document.createElement('canvas')\r\n            this.canvasSave.setAttribute('width', '' + this.w)\r\n            this.canvasSave.setAttribute('height', '' + this.h)\r\n            this.canvasSave.getContext('2d')?.drawImage(this.geoCanvas.canvas, 0, 0)\r\n            this.geoCanvas.initCanvasTransform()\r\n            return\r\n        }\r\n\r\n        if (focus) {\r\n            this.tooltip.html(focus.html)\r\n            this.tooltip.setPosition(e)\r\n            this.tooltip.show()\r\n\r\n            //show cell position as a rectangle\r\n            if (!this.canvasSave) {\r\n                this.canvasSave = document.createElement('canvas')\r\n                this.canvasSave.setAttribute('width', '' + this.w)\r\n                this.canvasSave.setAttribute('height', '' + this.h)\r\n                this.canvasSave.getContext('2d')?.drawImage(this.geoCanvas.offscreenCanvas, 0, 0)\r\n            } else {\r\n                this.geoCanvas.offscreenCtx.drawImage(this.canvasSave, 0, 0)\r\n            }\r\n\r\n            //draw image saved + draw rectangle\r\n            const rectWPix = this.selectionRectangleWidthPix\r\n                ? this.selectionRectangleWidthPix(focus.resolution, this.geoCanvas.view.z)\r\n                : 4\r\n            this.geoCanvas.initCanvasTransform()\r\n            const ctx = this.geoCanvas.offscreenCtx\r\n            ctx.strokeStyle = this.selectionRectangleColor\r\n            ctx.lineWidth = rectWPix\r\n            ctx.beginPath()\r\n\r\n            ctx.rect(\r\n                this.geoCanvas.geoToPixX(focus.cell.x) - rectWPix / 2,\r\n                this.geoCanvas.geoToPixY(focus.cell.y) + rectWPix / 2,\r\n                focus.resolution / this.geoCanvas.view.z + rectWPix,\r\n                -focus.resolution / this.geoCanvas.view.z - rectWPix\r\n            )\r\n            ctx.stroke()\r\n            this.geoCanvas.ctx.drawImage(this.geoCanvas.offscreenCanvas, 0, 0)\r\n        } else {\r\n            this.tooltip.hide()\r\n            if (this.canvasSave) this.geoCanvas.ctx.drawImage(this.canvasSave, 0, 0)\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Return the cell HTML info at a given geo position.\r\n     * This is usefull for user interactions, to show this info where the user clicks for example.\r\n     *\r\n     * @param {{x:number,y:number}} posGeo\r\n     * @returns {{cell:import('./Dataset.js').Cell,html:string,resolution:number} | undefined}\r\n     * @protected\r\n     */\r\n    getCellFocusInfo(posGeo) {\r\n        //go through the layers, starting from top\r\n        const z = this.geoCanvas.view.z\r\n        for (let i = this.layers.length - 1; i >= 0; i--) {\r\n            /** @type {import(\"./Layer.js\").Layer} */\r\n            const layer = this.layers[i]\r\n            if (layer.visible && !layer.visible(z)) continue\r\n            if (layer.cellInfoHTML === 'none') continue // this is necessary in order to not show tooltips for layers 'on top' (e.g. population circles on top of squares)\r\n            if (!layer.cellInfoHTML) continue\r\n            if (!layer.getDataset) continue\r\n            const dsc = layer.getDataset(z)\r\n            if (!dsc) continue\r\n\r\n            //get cell at mouse position\r\n            /** @type {import('./Dataset.js').Cell|undefined} */\r\n            const cell = dsc.getCellFromPosition(posGeo, dsc.getViewCache())\r\n            //console.log(cell, dsc.resolution)\r\n            if (!cell) return undefined\r\n\r\n            //rare case for a dataset with mixed resolutions\r\n            if (dsc.mixedResolution) {\r\n                const r = +dsc.mixedResolution(cell)\r\n                const html = layer.cellInfoHTML(cell, r)\r\n                if (!html) return undefined\r\n                return { cell: cell, html: html, resolution: r }\r\n            }\r\n\r\n            const html = layer.cellInfoHTML(cell, dsc.getResolution())\r\n            if (!html) return undefined\r\n            return { cell: cell, html: html, resolution: dsc.getResolution() }\r\n        }\r\n    }\r\n\r\n    /**\r\n     * @param {number} x\r\n     * @param {number} y\r\n     * @param {number|undefined} z\r\n     */\r\n    setView(x, y, z = undefined) {\r\n        this.geoCanvas.view.x = x\r\n        this.geoCanvas.view.y = y\r\n        if (z != undefined) this.geoCanvas.view.z = z\r\n        return this\r\n    }\r\n\r\n    /** @returns {import('./GeoCanvas.js').View} */\r\n    getView() {\r\n        return this.geoCanvas.view\r\n    }\r\n\r\n    /** @returns {number} */\r\n    getZoom() {\r\n        return this.geoCanvas.view.z\r\n    }\r\n    /** @param {number} z @returns {this} */\r\n    setZoom(z) {\r\n        this.geoCanvas.view.z = z\r\n        return this\r\n    }\r\n\r\n    /** @returns {Array.<number|undefined>} */\r\n    getCenterExtent() {\r\n        return this.geoCanvas.getCenterExtent()\r\n    }\r\n    /** @param {Array.<number>} val @returns {this} */\r\n    setCenterExtent(val) {\r\n        this.geoCanvas.setCenterExtent(val)\r\n        return this\r\n    }\r\n\r\n    /** @returns {Array.<number>} */\r\n    getZoomExtent() {\r\n        return this.geoCanvas.getZoomExtent()\r\n    }\r\n    /** @param {Array.<number>} val @returns {this} */\r\n    setZoomExtent(val) {\r\n        this.geoCanvas.setZoomExtent(val)\r\n        return this\r\n    }\r\n\r\n    /** @returns {string} */\r\n    getBackgroundColor() {\r\n        return this.geoCanvas.backgroundColor\r\n    }\r\n    /** @param {string} val @returns {this} */\r\n    setBackgroundColor(val) {\r\n        this.geoCanvas.backgroundColor = val\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * Adds a set of zoom buttons to the map\r\n     *\r\n     * @param {object} opts\r\n     * @returns {this}\r\n     */\r\n    addZoomButtons(opts) {\r\n        // * opts.id\r\n        // * opts.onZoom - custom event handler function\r\n        // * opts.x\r\n        // * opts.y\r\n        // * opts.delta - zoom delta applied on each click\r\n\r\n        this.zoomButtons = new ZoomButtons({\r\n            map: this,\r\n            id: opts?.id || 'gridviz-zoom-buttons-' + this.container.id,\r\n            class: opts?.class,\r\n            x: opts?.x,\r\n            y: opts?.y,\r\n            onZoom: opts?.onZoom,\r\n            delta: opts?.delta || 0.2,\r\n        })\r\n\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * Adds a fullscreen toggle button to the app\r\n     *\r\n     * @param {object} opts\r\n     * @returns {this}\r\n     */\r\n    addFullscreenButton(opts) {\r\n        // * opts.map - the gridviz map\r\n        // * opts.id\r\n        // * opts.x\r\n        // * opts.y\r\n\r\n        this.fullscreenButton = new FullscreenButton({\r\n            map: this,\r\n            id: opts?.id || 'gridviz-fullscreen-button',\r\n            class: opts?.class,\r\n            x: opts?.x,\r\n            y: opts?.y,\r\n        })\r\n\r\n        return this\r\n    }\r\n\r\n    /** @returns {this} */\r\n    setViewFromURL() {\r\n        this.geoCanvas.setViewFromURL()\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * @description Add a resize event observer to the Apps container and update the canvas accordingly\r\n     * @memberof App\r\n     */\r\n    defineResizeObserver() {\r\n        // Track whether the observer is currently processing a resize event\r\n        let resizePending = false\r\n\r\n        const resizeObserver = new ResizeObserver((entries) => {\r\n            if (!Array.isArray(entries) || !entries.length) return\r\n\r\n            let container = this.container\r\n\r\n            // Ensure the container has valid dimensions\r\n            if (container.clientWidth > 0 && container.clientHeight > 0) {\r\n                if (!resizePending) {\r\n                    resizePending = true // Prevent overlapping resize triggers\r\n\r\n                    window.requestAnimationFrame(() => {\r\n                        resizePending = false // Reset the flag after processing\r\n\r\n                        // Check for size changes\r\n                        if (this.h !== container.clientHeight || this.w !== container.clientWidth) {\r\n                            this.h = container.clientHeight\r\n                            this.w = container.clientWidth\r\n\r\n                            // Update geoCanvas sizes\r\n                            this.geoCanvas.h = this.h\r\n                            this.geoCanvas.w = this.w\r\n                            this.geoCanvas.canvas.setAttribute('width', String(this.w))\r\n                            this.geoCanvas.canvas.setAttribute('height', String(this.h))\r\n                            this.geoCanvas.offscreenCanvas.setAttribute('width', String(this.w))\r\n                            this.geoCanvas.offscreenCanvas.setAttribute('height', String(this.h))\r\n\r\n                            this.redraw()\r\n\r\n                            // Optionally reposition UI elements\r\n                            // if (this.zoomButtons) this.zoomButtons.node.style.left = this.w - 50 + 'px';\r\n                            // if (this.fullscreenButton) this.fullscreenButton.node.style.left = this.w - 50 + 'px';\r\n                        }\r\n                    })\r\n                }\r\n            }\r\n        })\r\n\r\n        resizeObserver.observe(this.container)\r\n    }\r\n\r\n    /**\r\n     * @description Destroy the map and it's event listeners\r\n     * This should significantly reduce the memory used when creating and destroying gridviz map instances (for example in leaflet-gridviz)\r\n     * @memberof App\r\n     */\r\n    destroy() {\r\n        // clear layers\r\n        this.layers = []\r\n        this.bgLayers = []\r\n\r\n        // remove event listeners from container\r\n        this.container.removeEventListener('mouseover', this.mouseOverHandler)\r\n        this.container.removeEventListener('mousemove', this.mouseMoveHandler)\r\n        this.container.removeEventListener('mouseout', this.mouseOutHandler)\r\n\r\n        // remove canvas\r\n        this.geoCanvas.canvas.remove()\r\n\r\n        // remove legend\r\n        this.legend?.remove()\r\n\r\n        // remove tooltip\r\n        this.tooltip.tooltip?.remove()\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n/**\r\n * A multi resolution dataset of grid cells.\r\n * It consists of different {@link Dataset}s for each resolution.\r\n *\r\n * @abstract\r\n *\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class MultiResolutionDataset {\r\n    /**\r\n     * @param {Array.<number>} resolutions The resolutions of the datasets, in CRS geographical unit.\r\n     * @param {Array.<import(\"./Dataset\").Dataset>|function(number):import(\"./Dataset\").Dataset} datasets The datasets list, one per resolution. Or a function that returns a dataset from a resolution value.\r\n     * @param { {preprocess?:function(import(\"./Dataset\").Cell):boolean} } opts Options. preprocess: A function to apply on each dataset cell to prepare its values. Can be used also to select cells to keep.\r\n     */\r\n    constructor(resolutions, datasets, opts = {}) {\r\n        opts = opts || {}\r\n\r\n        /** The resolutions of the datasets, in CRS geographical unit.\r\n         * @type {Array.<number>} */\r\n        this.resolutions = resolutions\r\n\r\n        /** The datasets. If the list is not explictely defined, build it from the resolution list with the function\r\n         * @type {Array.<import(\"./Dataset\").Dataset>} */\r\n        this.datasets = typeof datasets === 'function' ? this.resolutions.map(datasets) : datasets\r\n\r\n        //there must be as many datasets as resolutions\r\n        if (this.datasets.length > 1 && this.datasets.length != this.resolutions.length)\r\n            throw new Error(\r\n                'Uncompatible number of datasets and resolutions: ' +\r\n                    this.datasets.length +\r\n                    ' ' +\r\n                    this.resolutions.length\r\n            )\r\n\r\n        //set dataset preprocesses if specified\r\n        if (opts.preprocess) this.setPrepocesses(opts.preprocess)\r\n    }\r\n\r\n    /**\r\n     * Return the relevant dataset for a specified zoom.\r\n     * @param {number} z\r\n     * @param {number} minPixelsPerCell\r\n     * @returns {import(\"./Dataset\").Dataset|undefined}\r\n     * */\r\n    getDataset(z, minPixelsPerCell) {\r\n        //special case whith single dataset\r\n        if (this.datasets.length == 1) return this.datasets[0]\r\n\r\n        const rs = this.resolutions\r\n        let i = 0\r\n        let z_ = rs[i] / minPixelsPerCell\r\n        while (z_ < z && i < rs.length) {\r\n            i++\r\n            z_ = rs[i] / minPixelsPerCell\r\n        }\r\n        //if (i == 0) return this.dataset.datasets[0];\r\n        //return this.dataset.datasets[i - 1];\r\n        if (i == rs.length) return this.datasets[rs.length - 1]\r\n        return this.datasets[i]\r\n    }\r\n\r\n    /**\r\n     * Set a preprocess function for all datasets.\r\n     * This is a function applied on each cell after it has been loaded.\r\n     *\r\n     * @param {function(import(\"./Dataset\").Cell):boolean} preprocess\r\n     * @returns {this}\r\n     */\r\n    setPrepocesses(preprocess) {\r\n        for (let ds of this.datasets) ds.preprocess = preprocess\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * A function to ease the creation of multi resolution datasets.\r\n     *\r\n     * @param {Array.<number>} resolutions The resolutions of the datasets, in CRS geographical unit\r\n     * @param {function(number):import(\"./Dataset\").Dataset} resToDataset Function returning a dataset from a resolution\r\n     * @param { {preprocess?:function(import(\"./Dataset\").Cell):boolean} } opts Options. preprocess: A function to apply on each dataset cell to prepare its values\r\n     * @returns {MultiResolutionDataset}\r\n     */\r\n    /*static make(resolutions, resToDataset, opts) {\r\n        //make datasets\r\n        const dsc = []\r\n        for (const res of resolutions) dsc.push(resToDataset(res))\r\n        //make multi resolution dataset\r\n        return new MultiResolutionDataset(dsc, resolutions, opts)\r\n    }*/\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Drawable } from './Drawable.js'\r\n\r\n/** @typedef {\"square\"|\"circle\"|\"diamond\"|\"donut\"|\"triangle_up\"|\"triangle_down\"|\"triangle_left\"|\"triangle_right\"|\"none\"} Shape */\r\n\r\n/**\r\n * viewScale type\r\n * Returns an object from a list of cells,\r\n * @typedef {function(Array.<import('./Dataset.js').Cell>,number, number):*} ViewScale */\r\n\r\n/**\r\n * A style, to show a grid dataset.\r\n *\r\n * @module core\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class Style extends Drawable {\r\n    /**\r\n     * @abstract\r\n     * @param {{filter?:function(import('./Dataset').Cell):boolean, offset?:function(import('./Dataset').Cell, number, number):{dx:number,dy:number}, visible?:function(number):boolean,alpha?:function(number):number,blendOperation?:function(number):GlobalCompositeOperation,drawFun?:function,viewScale?:ViewScale}} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * @type {ViewScale|undefined} */\r\n        this.viewScale = opts.viewScale\r\n\r\n        /** A filter function to apply to the cell list, to filter out some cells not to be drawn (such as for example the cells with value=0).\r\n         * @protected\r\n         * @type {(function(import('./Dataset').Cell):boolean) | undefined} */\r\n        this.filter = opts.filter || undefined\r\n\r\n        /** An offset. This is to alter the position of all symbols in a given direction. In geographical unit.\r\n         * @protected\r\n         * @type {function(import('./Dataset').Cell,number,number):{dx:number,dy:number}} */\r\n        this.offset = opts.offset || ((c, r, z) => ({ dx: 0, dy: 0 }))\r\n\r\n        /** A draw function for the style.\r\n         * @type {function|undefined} */\r\n        this.drawFun = opts.drawFun\r\n\r\n        /**\r\n         * @public\r\n         * @type {Array.<import(\"./Legend\").Legend>} */\r\n        this.legends = []\r\n    }\r\n\r\n    /**\r\n     * Draw cells.\r\n     *\r\n     * @param {Array.<import('./Dataset').Cell>} cells The cells to draw.\r\n     * @param {import(\"./GeoCanvas\").GeoCanvas} geoCanvas The canvas where to draw them.\r\n     * @param {number} resolution Their resolution (in geographic unit)\r\n     * @abstract\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        if (this.drawFun) this.drawFun(cells, geoCanvas, resolution)\r\n        else throw new Error('Method draw not implemented.')\r\n    }\r\n\r\n    //getters and setters\r\n\r\n    /** @returns {function(import('./Dataset').Cell,number,number):{dx:number,dy:number}} */\r\n    getOffset() {\r\n        return this.offset\r\n    }\r\n    /** @param {function(import('./Dataset').Cell,number,number):{dx:number,dy:number}} val @returns {this} */\r\n    setOffset(val) {\r\n        this.offset = val\r\n        return this\r\n    }\r\n\r\n    /** Update legends of the style, if any\r\n     * @param {object} opts\r\n     * @returns {this} */\r\n    updateLegends(opts) {\r\n        Style.updateLegendsRecursive(this.legends, opts)\r\n        return this\r\n    }\r\n\r\n    /** @private */\r\n    static updateLegendsRecursive(lg, opts) {\r\n        if (Array.isArray(lg)) for (const lg_ of lg) this.updateLegendsRecursive(lg_, opts)\r\n        else lg.update(opts)\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"./Legend\").Legend>} legends\r\n     * @returns {this} */\r\n    addLegends(legends) {\r\n        for (let legend of legends) this.legends.push(legend)\r\n        return this\r\n    }\r\n}\r\n","//@ts-check\n'use strict'\n\nimport { select } from 'd3-selection'\n//import { transition } from \"d3-transition\";\n\n/**\n * A generic class to make a tooltip.\n * It is a div element, which can be moved under the mouse pointer and filled with some information in html.\n * @module core\n */\nexport class Tooltip {\n    /**\n     * @param {object} opts\n     */\n    constructor(opts) {\n        opts = opts || {}\n\n        /** @type {string} */\n        this.maxWidth = opts.maxWidth || '20em'\n        /** @type {string} */\n        this.fontSize = opts.fontSize || '1.2em'\n        /** @type {string} */\n        this.background = opts.background || 'white'\n        /** @type {string} */\n        this.padding = opts.padding || '5px'\n        /** @type {string} */\n        this.border = opts.border || '0px'\n        /** @type {string} */\n        this['border-radius'] = opts['border-radius'] || '0px'\n        /** @type {string} */\n        this['box-shadow'] = opts['box-shadow'] || '5px 5px 5px grey'\n        /** @type {string} */\n        this['font-family'] = opts['font-family'] || 'Helvetica, Arial, sans-serif'\n\n        /** @type {number} */\n        this.transitionDuration = opts.transitionDuration || 100\n        /** @type {number} */\n        this.xOffset = opts.xOffset || 30\n        /** @type {number} */\n        this.yOffset = opts.yOffset || 20\n        /** @type {number} */ // e.g. to prevent mouse cursor covering cell being highlighted\n        this.yMouseOffset = opts.yMouseOffset || 0\n        /** @type {number} */\n        this.xMouseOffset = opts.xMouseOffset || 0\n        /** @type {HTMLElement} */\n        this.parentElement = opts.parentElement || document.body\n        /** @type {HTMLElement} */\n        this.tooltipElement = opts.tooltipElement || null\n\n        /**\n         * @public\n         * @type {import(\"d3-selection\").Selection} */\n        this.tooltip = opts.tooltipElement\n            ? select(opts.tooltipElement) // Wrap the provided HTML node in a D3 selection\n            : select(this.parentElement).append('div').attr('id', 'gridviz-tooltip').attr('class', 'gridviz-tooltip') // create default element\n\n        //initialise\n        this.tooltip.style('max-width', this.maxWidth)\n        this.tooltip.style('overflow', 'hidden')\n        this.tooltip.style('font-size', this.fontSize)\n        this.tooltip.style('background', this.background)\n        this.tooltip.style('padding', this.padding)\n        this.tooltip.style('border', this.border)\n        this.tooltip.style('border-radius', this['border-radius'])\n        this.tooltip.style('box-shadow', this['box-shadow'])\n        this.tooltip.style('font-family', this['font-family'])\n        this.tooltip.style('position', 'absolute')\n        this.tooltip.style('pointer-events', 'none')\n        this.tooltip.style('opacity', '0')\n        this.tooltip.style('text-wrap', 'nowrap')\n\n        // these placeholders are needed to prevent an infinite DOM resizeObserver loop:\n        this.tooltip.style('left', '0')\n        this.tooltip.style('top', '0')\n\n        // aria-labels (thanks to wahlatlas)\n        this.tooltip.attr('role', 'tooltip').attr('aria-live', 'polite')\n    }\n\n    /** Show the tooltip */\n    show() {\n        // @ts-ignore\n        this.tooltip.transition().duration(this.transitionDuration).style('opacity', 1)\n    }\n\n    /** Hide the tooltip */\n    hide() {\n        // @ts-ignore\n        this.tooltip.transition().duration(this.transitionDuration).style('opacity', 0)\n    }\n\n    /**\n     * Set the content of the tooltip.\n     * @param {string} html\n     */\n    html(html) {\n        this.tooltip.html(html)\n    }\n\n    /**\n     * Set the position of the tooltip at the mouse event position.\n     * @param {MouseEvent} event\n     */\n    setPosition(event) {\n        // Get the bounding rect of the parent container (map2)\n        let parentRect = this.parentElement.getBoundingClientRect()\n\n        // Get the mouse position (relative to the parent container)\n        let x = event.clientX - parentRect.left + this.xOffset // Relative to parent\n        let y = event.clientY - parentRect.top - this.yOffset // Relative to parent\n\n        // Now, apply the position to the tooltip\n        this.tooltip.style('left', x + 'px').style('top', y + 'px')\n\n        // Ensure the tooltip stays inside the parent container\n        this.ensureTooltipInsideContainer(event, parentRect, this.tooltip.node())\n    }\n    /**\n     * @function ensureTooltipInsideContainer\n     * @description Prevents the tooltip from overflowing out of the App container (ensures that the tooltip is inside the gridviz container)\n     * @param {MouseEvent} event\n     * @param {DOMRect} parentRect\n     * @param {HTMLElement} tooltipNode\n     */\n    ensureTooltipInsideContainer(event, parentRect, tooltipNode) {\n        let node = tooltipNode\n        let parentWidth = parentRect.width\n        let parentHeight = parentRect.height\n\n        // Ensure tooltip doesn't go beyond the right edge\n        if (node.offsetLeft + node.clientWidth > parentWidth) {\n            let left = event.clientX - node.clientWidth - this.xOffset\n            node.style.left = left + 'px'\n        }\n\n        // Ensure tooltip doesn't go beyond the bottom edge\n        if (node.offsetTop + node.clientHeight > parentHeight) {\n            node.style.top = parentHeight - node.clientHeight + 'px'\n        }\n\n        // Ensure tooltip doesn't go above the top edge\n        if (node.offsetTop < 0) {\n            node.style.top = 0 + 'px'\n        }\n\n        // Ensure tooltip doesn't go beyond the left edge\n        if (node.offsetLeft < 0) {\n            node.style.left = 0 + 'px'\n        }\n    }\n\n    /*\n\tmy.mouseover = function (event, html) {\n\t\tif (html) my.html(html);\n\t\tmy.setPosition(event);\n\t\tmy.show()\n\t\t//this.ensureTooltipInsideContainer();\n\t};\n\t\n\tmy.mousemove = function (event) {\n\t\tmy.setPosition(event);\n\t\t//this.ensureTooltipInsideContainer();\n\t};\n\t\n\tmy.mouseout = function () {\n\t\tmy.hide();\n\t};*/\n\n    style(k, v) {\n        if (arguments.length == 1) return this.tooltip.style(k)\n        this.tooltip.style(k, v)\n        return this\n    }\n\n    attr(k, v) {\n        if (arguments.length == 1) return this.tooltip.attr(k)\n        this.tooltip.attr(k, v)\n        return this\n    }\n}\n","//@ts-check\r\n'use strict'\r\n\r\n/** @typedef {{ dims: object, crs: string, tileSizeCell: number, originPoint: {x:number,y:number}, resolutionGeo: number, tilingBounds:import(\"../core/GeoCanvas.js\").Envelope }} GridInfo */\r\n\r\nimport { dsv } from 'd3-fetch'\r\nimport { Dataset } from '../core/Dataset.js'\r\n\r\n/**\r\n * A dataset composed of a single CSV file (not tiled).\r\n *\r\n * @module dataset\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class CSVGrid extends Dataset {\r\n    /**\r\n     * @param {import(\"../core/Map.js\").Map} map The map.\r\n     * @param {string} url The URL of the dataset.\r\n     * @param {number} resolution The dataset resolution in geographical unit.\r\n     * @param {{preprocess?:(function(import(\"../core/Dataset.js\").Cell):boolean),delimiter?:string}} opts\r\n     */\r\n    constructor(map, url, resolution, opts = {}) {\r\n        super(map, url, resolution, opts)\r\n\r\n        /**\r\n         * @private\r\n         * @type {Array.<import(\"../core/Dataset.js\").Cell>} */\r\n        this.cells = []\r\n\r\n        /**\r\n         * @private\r\n         * @type {string} */\r\n        this.delimiter = opts.delimiter || ','\r\n\r\n        /**\r\n         * @type {string}\r\n         * @private  */\r\n        this.infoLoadingStatus = 'notLoaded'\r\n\r\n        //get data\r\n        this.getData(undefined)\r\n    }\r\n\r\n    /**\r\n     * Request data within a geographic envelope.\r\n     * @param {import(\"../core/GeoCanvas.js\").Envelope|undefined} e\r\n     */\r\n    getData(e) {\r\n        //check if data already loaded\r\n        if (this.infoLoadingStatus != 'notLoaded') return this\r\n\r\n        //load data\r\n        this.infoLoadingStatus = 'loading'\r\n        ;(async () => {\r\n            try {\r\n                const data = await dsv(this.delimiter, this.url)\r\n\r\n                //convert coordinates in numbers\r\n                for (const c of data) {\r\n                    c.x = +c.x\r\n                    c.y = +c.y\r\n                }\r\n\r\n                //preprocess/filter\r\n                if (this.preprocess) {\r\n                    this.cells = []\r\n                    for (const c of data) {\r\n                        const b = this.preprocess(c)\r\n                        if (b == false) continue\r\n                        this.cells.push(c)\r\n                    }\r\n                } else {\r\n                    this.cells = data\r\n                }\r\n\r\n                //TODO check if redraw is necessary\r\n                //that is if the dataset belongs to a layer which is visible at the current zoom level\r\n\r\n                //redraw map\r\n                if (this.map) this.map.redraw()\r\n\r\n                this.infoLoadingStatus = 'loaded'\r\n            } catch (error) {\r\n                //mark as failed\r\n                this.infoLoadingStatus = 'failed'\r\n                this.cells = []\r\n            }\r\n        })()\r\n\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * Fill the view cache with all cells which are within a geographical envelope.\r\n     *\r\n     * @param {import(\"../core/GeoCanvas.js\").Envelope} extGeo\r\n     * @returns {void}\r\n     */\r\n    updateViewCache(extGeo) {\r\n        //data not loaded yet\r\n        if (!this.cells) return\r\n\r\n        this.cellsViewCache = []\r\n        for (const cell of this.cells) {\r\n            if (+cell.x + this.resolution < extGeo.xMin) continue\r\n            if (+cell.x - this.resolution > extGeo.xMax) continue\r\n            if (+cell.y + this.resolution < extGeo.yMin) continue\r\n            if (+cell.y - this.resolution > extGeo.yMax) continue\r\n            this.cellsViewCache.push(cell)\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Dataset } from '../core/Dataset.js'\r\n\r\n/**\r\n * A dataset composed of cells defined in javascript, or loaded outside of gridviz map.\r\n *\r\n * @module dataset\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class JSGrid extends Dataset {\r\n    /**\r\n     * @param {number} resolution The dataset resolution in geographical unit.\r\n     * @param {Array.<Object>} cells The cells.\r\n     * @param {} opts\r\n     */\r\n    constructor(resolution, cells, opts = {}) {\r\n        super(undefined, '', resolution, opts)\r\n\r\n        /**\r\n         * @private\r\n         * @type {Array.<import('../core/Dataset.js').Cell>} */\r\n        this.cells = cells || []\r\n    }\r\n\r\n    /**\r\n     * Request data within a geographic envelope.\r\n     *\r\n     * @param {import(\"../core/GeoCanvas.js\").Envelope|undefined} e\r\n     */\r\n    getData(e) {\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * Fill the view cache with all cells which are within a geographical envelope.\r\n     *\r\n     * @param {import(\"../core/GeoCanvas.js\").Envelope} extGeo\r\n     * @returns {void}\r\n     */\r\n    updateViewCache(extGeo) {\r\n        //data not loaded yet\r\n        if (!this.cells) return\r\n\r\n        this.cellsViewCache = []\r\n        for (const cell of this.cells) {\r\n            if (+cell.x + this.resolution < extGeo.xMin) continue\r\n            if (+cell.x - this.resolution > extGeo.xMax) continue\r\n            if (+cell.y + this.resolution < extGeo.yMin) continue\r\n            if (+cell.y - this.resolution > extGeo.yMax) continue\r\n            this.cellsViewCache.push(cell)\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n/** @typedef {{ dims: object, crs: string, tileSizeCell: number, originPoint: {x:number,y:number}, resolutionGeo: number, tilingBounds:import(\"../core/GeoCanvas.js\").Envelope }} GridInfo */\r\n\r\n// internal\r\nimport { Dataset } from '../core/Dataset.js'\r\n//import { monitor, monitorDuration } from '../utils/Utils.js'\r\n\r\n// external\r\nimport { json, csv } from 'd3-fetch'\r\n\r\n/**\r\n * A tiled dataset, composed of CSV tiles.\r\n *\r\n * @module dataset\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class TiledGrid extends Dataset {\r\n    /**\r\n     * @param {import(\"../core/Map.js\").Map} map The map.\r\n     * @param {string} url The URL of the dataset.\r\n     * @param {{preprocess?:(function(import(\"../core/Dataset.js\").Cell):boolean), onlyDrawWhenAllTilesReady:boolean}} opts\r\n     */\r\n    constructor(map, url, opts = {}) {\r\n        super(map, url, 0, opts)\r\n        this.onlyDrawWhenAllTilesReady = opts.onlyDrawWhenAllTilesReady || false\r\n        /**\r\n         * The grid info object, from the info.json file.\r\n         *  @type {GridInfo | undefined}\r\n         * @private\r\n         *  */\r\n        this.info = undefined\r\n\r\n        /**\r\n         * @type {string}\r\n         * @private  */\r\n        this.infoLoadingStatus = 'notLoaded'\r\n\r\n        /**\r\n         * The cache of the loaded tiles. It is double indexed: by xT and then yT.\r\n         * Example: this.cache[xT][yT] returns the tile at [xT][yT] location.\r\n         *\r\n         * @type {object}\r\n         * */\r\n        this.cache = {}\r\n\r\n        //launch loading\r\n        this.loadInfo()\r\n    }\r\n\r\n    /**\r\n     * Load the info.json from the url.\r\n     * @returns this\r\n     */\r\n    loadInfo() {\r\n        if (!this.info && this.infoLoadingStatus === 'notLoaded') {\r\n            ;(async () => {\r\n                try {\r\n                    const data = await json(this.url + 'info.json')\r\n                    this.info = data\r\n                    this.resolution = data.resolutionGeo\r\n                    this.infoLoadingStatus = 'loaded'\r\n                    this.map.redraw()\r\n                } catch (error) {\r\n                    //mark as failed\r\n                    this.infoLoadingStatus = 'failed'\r\n                }\r\n            })()\r\n        } else if (this.infoLoadingStatus === 'loaded' || this.infoLoadingStatus === 'failed')\r\n            this.map.redraw()\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * Compute a tiling envelope from a geographical envelope.\r\n     * This is the function to use to know which tiles to download for a geographical view.\r\n     *\r\n     * @param {import(\"../core/GeoCanvas.js\").Envelope} e\r\n     * @returns {import(\"../core/GeoCanvas.js\").Envelope|undefined}\r\n     */\r\n    getTilingEnvelope(e) {\r\n        if (!this.info) {\r\n            this.loadInfo()\r\n            return\r\n        }\r\n\r\n        const po = this.info.originPoint,\r\n            r = this.info.resolutionGeo,\r\n            s = this.info.tileSizeCell\r\n\r\n        return {\r\n            xMin: Math.floor((e.xMin - po.x) / (r * s)),\r\n            xMax: Math.floor((e.xMax - po.x) / (r * s)),\r\n            yMin: Math.floor((e.yMin - po.y) / (r * s)),\r\n            yMax: Math.floor((e.yMax - po.y) / (r * s)),\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Request data within a geographic envelope.\r\n     *\r\n     * @param {import('../core/GeoCanvas.js').Envelope} extGeo\r\n     * @returns {this}\r\n     */\r\n    async getData(extGeo) {\r\n        if (!this.info) return this\r\n\r\n        // Get the tiling envelope and check bounds\r\n        const tb = this.getTilingEnvelope(extGeo)\r\n        if (!tb) return this\r\n\r\n        const { xMin: gbXMin, xMax: gbXMax, yMin: gbYMin, yMax: gbYMax } = this.info.tilingBounds\r\n\r\n        const xMin = Math.max(tb.xMin, gbXMin)\r\n        const xMax = Math.min(tb.xMax, gbXMax)\r\n        const yMin = Math.max(tb.yMin, gbYMin)\r\n        const yMax = Math.min(tb.yMax, gbYMax)\r\n\r\n        // Count the total number of tiles\r\n        const totalTiles = (xMax - xMin + 1) * (yMax - yMin + 1)\r\n\r\n        let processedTiles = 0 // Counter for loaded tiles\r\n        const tilePromises = []\r\n\r\n        // Iterate over tiles within bounds\r\n        for (let xT = xMin; xT <= xMax; xT++) {\r\n            for (let yT = yMin; yT <= yMax; yT++) {\r\n                if (!this.cache[xT]) this.cache[xT] = {}\r\n\r\n                // Skip already loaded or loading tiles\r\n                if (this.cache[xT][yT]) {\r\n                    ++processedTiles\r\n                    continue\r\n                }\r\n\r\n                // Mark tile as loading\r\n                this.cache[xT][yT] = 'loading'\r\n\r\n                // Prepare the tile request and store it in tilePromises\r\n                tilePromises.push(\r\n                    this.loadTile(xT, yT)\r\n                        .then((tile) => {\r\n                            // Before we increment processedTiles, check if it's the last tile\r\n                            const isLastTile = ++processedTiles === totalTiles\r\n\r\n                            // Store the tile in the cache\r\n                            this.cache[xT][yT] = tile\r\n\r\n                            // Pass the isLastTile argument to checkAndRedraw\r\n                            this.checkAndRedraw(tile, isLastTile)\r\n                        })\r\n                        .catch(() => {\r\n                            // Mark as failed but allow processing to continue\r\n                            this.cache[xT][yT] = 'failed'\r\n                            ++processedTiles\r\n                        })\r\n                )\r\n            }\r\n        }\r\n\r\n        // Handle all tile promises, including failures\r\n        await Promise.allSettled(tilePromises)\r\n        return this\r\n    }\r\n\r\n    async loadTile(xT, yT) {\r\n        try {\r\n            const data = await csv(`${this.url}${xT}/${yT}.csv`)\r\n\r\n            // Preprocess or filter the data if needed\r\n            const cells = this.preprocess ? data.filter((cell) => this.preprocess(cell) !== false) : data\r\n\r\n            if (!this.info) throw new Error('Tile info unknown')\r\n\r\n            return getGridTile(cells, xT, yT, this.info)\r\n        } catch (error) {\r\n            console.warn(`Failed to load tile ${xT}, ${yT}:`, error)\r\n            throw error\r\n        }\r\n    }\r\n\r\n    checkAndRedraw(tile, isLastTile) {\r\n        // Check if any visible layer depends on this dataset\r\n        //check if redraw is really needed, that is if:\r\n        // 1. the dataset belongs to a layer which is visible at the current zoom level\r\n        let needsRedraw = false\r\n        //go through the layers\r\n        const z = this.map.getZoom()\r\n        for (const lay of this.map.layers) {\r\n            if (lay.visible && !lay.visible(z)) continue\r\n            if (!lay.getDataset) continue\r\n            if (lay.getDataset(z) != this) continue\r\n            //found one layer. No need to seek more.\r\n            needsRedraw = true\r\n            break\r\n        }\r\n\r\n        if (!needsRedraw) return\r\n\r\n        // Check if tile intersects the current view\r\n        const env = this.map.updateExtentGeo()\r\n        const { xMin, xMax, yMin, yMax } = tile.extGeo\r\n        if (env.xMax <= xMin || env.xMin >= xMax || env.yMax <= yMin || env.yMin >= yMax) return\r\n\r\n        // Trigger redraw\r\n        if (this.onlyDrawWhenAllTilesReady) {\r\n            if (isLastTile) {\r\n                this.map.redraw()\r\n            }\r\n        } else {\r\n            this.map.redraw()\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Fill the view cache with all cells which are within a geographical envelope.\r\n     * @abstract\r\n     * @param {import(\"../core/GeoCanvas.js\").Envelope} extGeo\r\n     * @returns {void}\r\n     */\r\n    updateViewCache(extGeo) {\r\n        //\r\n        this.cellsViewCache = []\r\n\r\n        //check if info has been loaded\r\n        if (!this.info) return\r\n\r\n        //tiles within the scope\r\n        /** @type {import(\"../core/GeoCanvas.js\").Envelope|undefined} */\r\n        const tb = this.getTilingEnvelope(extGeo)\r\n        if (!tb) return\r\n\r\n        //grid bounds\r\n        /** @type {import(\"../core/GeoCanvas.js\").Envelope} */\r\n        const gb = this.info.tilingBounds\r\n\r\n        for (let xT = Math.max(tb.xMin, gb.xMin); xT <= Math.min(tb.xMax, gb.xMax); xT++) {\r\n            if (!this.cache[xT]) continue\r\n            for (let yT = Math.max(tb.yMin, gb.yMin); yT <= Math.min(tb.yMax, gb.yMax); yT++) {\r\n                //get tile\r\n                /** @type {object} */\r\n                const tile = this.cache[xT][yT]\r\n                if (!tile || typeof tile === 'string') continue\r\n\r\n                //get cells\r\n                //this.cellsViewCache = this.cellsViewCache.concat(tile.cells)\r\n\r\n                for (const cell of tile.cells) {\r\n                    if (+cell.x + this.resolution < extGeo.xMin) continue\r\n                    if (+cell.x - this.resolution > extGeo.xMax) continue\r\n                    if (+cell.y + this.resolution < extGeo.yMin) continue\r\n                    if (+cell.y - this.resolution > extGeo.yMax) continue\r\n                    this.cellsViewCache.push(cell)\r\n                }\r\n            }\r\n        }\r\n    }\r\n}\r\n\r\nfunction getGridTile(cells, xT, yT, gridInfo) {\r\n    const tile = {}\r\n\r\n    /** @type {Array.<import(\"../core/Dataset\").Cell>} */\r\n    tile.cells = cells\r\n    /** @type {number} */\r\n    tile.x = xT\r\n    /** @type {number} */\r\n    tile.y = yT\r\n\r\n    const r = gridInfo.resolutionGeo\r\n    const s = gridInfo.tileSizeCell\r\n\r\n    /** @type {import(\"../core/GeoCanvas\").Envelope} */\r\n    tile.extGeo = {\r\n        xMin: gridInfo.originPoint.x + r * s * tile.x,\r\n        xMax: gridInfo.originPoint.x + r * s * (tile.x + 1),\r\n        yMin: gridInfo.originPoint.y + r * s * tile.y,\r\n        yMax: gridInfo.originPoint.y + r * s * (tile.y + 1),\r\n    }\r\n\r\n    //convert cell coordinates into geographical coordinates\r\n    for (let cell of tile.cells) {\r\n        cell.x = tile.extGeo.xMin + cell.x * r\r\n        cell.y = tile.extGeo.yMin + cell.y * r\r\n    }\r\n\r\n    return tile\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Layer } from '../core/Layer.js'\r\n\r\n/**\r\n *\r\n * A map background layer in \"Slippy map\" XYZ standard.\r\n * See https://wiki.openstreetmap.org/wiki/Slippy_map_tilenames\r\n * https://www.maptiler.com/google-maps-coordinates-tile-bounds-projection/#6/27.88/44.48\r\n *\r\n * @module layer\r\n * @author Julien Gaffuri\r\n */\r\nexport class BackgroundLayer extends Layer {\r\n    /**\r\n     * @param {object} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** The image cache, indexed by z/y/x */\r\n        this.cache = {}\r\n\r\n        /**\r\n         * @type {string} */\r\n        this.url = opts.url\r\n        /** @type {function(number,number,number):string} */\r\n        this.urlFun = opts.urlFun || ((x, y, z) => this.url + z + '/' + x + '/' + y + '.png')\r\n\r\n        /** @type {Array.<number>} */\r\n        this.resolutions = opts.resolutions\r\n        if (!this.resolutions || this.resolutions.length == 0)\r\n            throw new Error('No resolutions provided for background layer')\r\n\r\n        /** @type {number} */\r\n        this.nbPix = opts.nbPix || 256\r\n        /** CRS coordinates of top left corner\r\n         * @type {Array.<number>} */\r\n        this.origin = opts.origin || [0, 0]\r\n        /** @type {number} */\r\n        this.z0 = opts.z0 || 0\r\n    }\r\n\r\n    /**\r\n     * Get z/x/y cache data.\r\n     * @param {number} z\r\n     * @param {number} x\r\n     * @param {number} y\r\n     * @returns {HTMLImageElement|string|undefined}\r\n     * @private\r\n     */\r\n    get(z, x, y) {\r\n        let d = this.cache[z]\r\n        if (!d) return\r\n        d = d[x]\r\n        if (!d) return\r\n        return d[y]\r\n    }\r\n\r\n    /**\r\n     * Get z/x/y cache data.\r\n     * @param {HTMLImageElement|string} img\r\n     * @param {number} z\r\n     * @param {number} x\r\n     * @param {number} y\r\n     * @returns\r\n     * @private\r\n     */\r\n    put(img, z, x, y) {\r\n        if (!this.cache[z]) this.cache[z] = {}\r\n        if (!this.cache[z][x]) this.cache[z][x] = {}\r\n        this.cache[z][x][y] = img\r\n    }\r\n\r\n    /**\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @returns {void}\r\n     */\r\n    draw(geoCanvas) {\r\n        if (!this.resolutions || this.resolutions.length == 0) {\r\n            console.error('No resolutions provided for background layer')\r\n            return\r\n        }\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const x0 = this.origin[0],\r\n            y0 = this.origin[1]\r\n\r\n        //get zoom level and resolution\r\n        let z_ = 0\r\n        for (z_ = 0; z_ < this.resolutions.length; z_++) if (this.resolutions[z_] < z) break\r\n        z_ -= 1\r\n        z_ = Math.max(0, z_)\r\n        z_ = Math.min(z_, this.resolutions.length - 1)\r\n        //console.log(this.resolutions.length, z)\r\n        const res = this.resolutions[z_]\r\n\r\n        z_ += this.z0\r\n\r\n        const sizeG = this.nbPix * res\r\n        const size = sizeG / z\r\n\r\n        //get tile numbers\r\n        const xGeoToTMS = (x) => Math.ceil((x - x0) / sizeG)\r\n        const yGeoToTMS = (y) => Math.ceil(-(y - y0) / sizeG)\r\n        const xMin = xGeoToTMS(geoCanvas.extGeo.xMin) - 1\r\n        const xMax = xGeoToTMS(geoCanvas.extGeo.xMax)\r\n        const yMax = yGeoToTMS(geoCanvas.extGeo.yMin)\r\n        const yMin = yGeoToTMS(geoCanvas.extGeo.yMax) - 1\r\n\r\n        //handle images\r\n        for (let x = xMin; x < xMax; x++) {\r\n            for (let y = yMin; y < yMax; y++) {\r\n                //get image\r\n                let img = this.get(z_, x, y)\r\n\r\n                //load image\r\n                if (!img) {\r\n                    const img = new Image()\r\n                    this.put(img, z_, x, y)\r\n                    img.onload = () => {\r\n                        geoCanvas.redraw()\r\n                    }\r\n                    img.onerror = () => {\r\n                        //case when no image\r\n                        this.put('failed', z_, x, y)\r\n                    }\r\n                    img.src = this.urlFun(x, y, z_)\r\n                    continue\r\n                }\r\n\r\n                //case when no image\r\n                if (img === 'failed') continue\r\n                if (!(img instanceof HTMLImageElement)) {\r\n                    console.log(img)\r\n                    continue\r\n                }\r\n                if (img.width == 0 || img.height == 0) continue\r\n\r\n                //draw image\r\n                const xGeo = x0 + x * sizeG\r\n                const yGeo = y0 - y * sizeG\r\n                try {\r\n                    geoCanvas.initCanvasTransform()\r\n                    geoCanvas.offscreenCtx.drawImage(\r\n                        img,\r\n                        geoCanvas.geoToPixX(xGeo),\r\n                        geoCanvas.geoToPixY(yGeo),\r\n                        size,\r\n                        size\r\n                    )\r\n                    //cg.ctx.drawImage(img, xGeo, yGeo, sizeG, -sizeG)\r\n                } catch (error) {\r\n                    console.error(error)\r\n                }\r\n            }\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Layer } from '../core/Layer.js'\r\n\r\n/**\r\n *\r\n * A map background layer composed of a single image file, geolocated.\r\n *\r\n * @module layer\r\n * @author Julien Gaffuri\r\n */\r\nexport class BackgroundLayerImage extends Layer {\r\n    /**\r\n     * @param {object} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** The image file URL\r\n         * @type {string} */\r\n        this.url = opts.url\r\n\r\n        /** The image left coordinate\r\n         * @type {number} */\r\n        this.xMin = opts.xMin || 0\r\n        /** The image top coordinate\r\n         *  @type {number} */\r\n        this.yMax = opts.yMax || 0\r\n\r\n        /** The image width, in geo unit\r\n         * @type {number} */\r\n        this.width = opts.width || 20000\r\n        /** The image height, in geo unit\r\n         * @type {number} */\r\n        this.height = opts.height || 20000\r\n\r\n        /** The image object\r\n         * @type {HTMLImageElement|undefined} */\r\n        this.img = undefined\r\n    }\r\n\r\n    /**\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @returns {void}\r\n     */\r\n    draw(geoCanvas) {\r\n        //update map extent\r\n        //geoCanvas.updateExtentGeo(0)\r\n\r\n        if (this.img) {\r\n            //the image was already downloaded: draw it\r\n\r\n            //compute screen coordinates and size ratio\r\n            const x = geoCanvas.geoToPixX(this.xMin)\r\n            const y = geoCanvas.geoToPixY(this.yMax)\r\n            const z = geoCanvas.getView().z\r\n\r\n            //draw image\r\n            geoCanvas.initCanvasTransform()\r\n            geoCanvas.offscreenCtx.drawImage(this.img, x, y, this.width / z, this.height / z)\r\n        } else {\r\n            //retrieve image\r\n\r\n            if (!this.img) {\r\n                this.img = new Image()\r\n                this.img.onload = () => {\r\n                    geoCanvas.redraw()\r\n                }\r\n                this.img.onerror = () => {\r\n                    //case when no image\r\n                    console.warn('Could not retrieve background image from', this.url)\r\n                }\r\n            }\r\n\r\n            //set URL to launch the download\r\n            this.img.src = this.url\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Layer } from '../core/Layer.js'\r\n\r\n/**\r\n *\r\n * A map WMS background layer.\r\n *\r\n * @module layer\r\n * @author Julien Gaffuri\r\n */\r\nexport class BackgroundLayerWMS extends Layer {\r\n    /**\r\n     * @param {object} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * @type {string} */\r\n        this.url = opts.url\r\n\r\n        /** @type {HTMLImageElement|undefined} */\r\n        this.img = undefined\r\n\r\n        /** @type {number|undefined} */\r\n        this.xMin = undefined\r\n        /** @type {number|undefined} */\r\n        this.xMax = undefined\r\n        /** @type {number|undefined} */\r\n        this.yMin = undefined\r\n        /** @type {number|undefined} */\r\n        this.yMax = undefined\r\n    }\r\n\r\n    /** Check if the view has moved and a new image needs to be retrieved.\r\n     * @private */\r\n    hasMoved(extGeo) {\r\n        if (extGeo.xMin != this.xMin) return true\r\n        else if (extGeo.xMax != this.xMax) return true\r\n        else if (extGeo.yMin != this.yMin) return true\r\n        else if (extGeo.yMax != this.yMax) return true\r\n        else return false\r\n    }\r\n\r\n    /**\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @returns {void}\r\n     */\r\n    draw(geoCanvas) {\r\n        //update map extent\r\n        geoCanvas.updateExtentGeo(0)\r\n\r\n        if (!this.hasMoved(geoCanvas.extGeo) && this.img) {\r\n            //the map did not move and the image was already downloaded: draw the image\r\n            geoCanvas.initCanvasTransform()\r\n            geoCanvas.offscreenCtx.drawImage(this.img, 0, 0, geoCanvas.w, geoCanvas.h)\r\n        } else {\r\n            //the map moved: retrieve new image\r\n\r\n            //\r\n            this.xMin = geoCanvas.extGeo.xMin\r\n            this.xMax = geoCanvas.extGeo.xMax\r\n            this.yMin = geoCanvas.extGeo.yMin\r\n            this.yMax = geoCanvas.extGeo.yMax\r\n\r\n            //build WMS URL\r\n            const url = []\r\n            url.push(this.url)\r\n            url.push('&width=')\r\n            url.push(geoCanvas.w)\r\n            url.push('&height=')\r\n            url.push(geoCanvas.h)\r\n            //bbox: xmin ymin xmax ymax\r\n            url.push('&bbox=')\r\n            url.push(geoCanvas.extGeo.xMin)\r\n            url.push(',')\r\n            url.push(geoCanvas.extGeo.yMin)\r\n            url.push(',')\r\n            url.push(geoCanvas.extGeo.xMax)\r\n            url.push(',')\r\n            url.push(geoCanvas.extGeo.yMax)\r\n\r\n            const urlS = url.join('')\r\n            //console.log(urlS)\r\n\r\n            if (!this.img) {\r\n                this.img = new Image()\r\n                this.img.onload = () => {\r\n                    geoCanvas.redraw()\r\n                }\r\n                this.img.onerror = () => {\r\n                    //case when no image\r\n                    console.warn('Could not retrieve WMS background image from', urlS)\r\n                }\r\n            }\r\n\r\n            //set URL to launch the download\r\n            this.img.src = urlS\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Layer } from '../core/Layer.js'\r\nimport { json } from 'd3-fetch'\r\n\r\n/**\r\n * @module layer\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class GeoJSONLayer extends Layer {\r\n    /**\r\n     * @param {object} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * @private\r\n         * @type {string} */\r\n        this.url = opts.url\r\n\r\n        /**\r\n         * A preprocess to run on each feature after loading.\r\n         * It can be used to apply some specific treatment before, format the label data, project coordinates, etc.\r\n         * Return false if the label should not be kept.\r\n         * @private\r\n         * @type {function(object):boolean} */\r\n        this.preprocess = opts.preprocess\r\n\r\n        //for points\r\n        /**\r\n         * @private\r\n         * @type {function(object,number):string} */\r\n        this.shape = opts.shape || ((f, z) => 'circle')\r\n        /**\r\n         * In pixel\r\n         * @private\r\n         * @type {function(object,number):number} */\r\n        this.size = opts.size || ((f, z) => 10)\r\n        /**\r\n         * @private\r\n         * @type {function(object,number):string} */\r\n        this.strokeStyle = opts.strokeStyle || ((f, z) => 'red')\r\n        /**\r\n         * @private\r\n         * @type {function(object,number):string} */\r\n        this.fillStyle = opts.fillStyle || ((f, z) => 'black')\r\n        /**\r\n         * In pixel\r\n         * @private\r\n         * @type {function(object,number):number} */\r\n        this.lineWidth = opts.lineWidth || ((f, z) => 2)\r\n\r\n        //for lines\r\n\r\n        /**\r\n         * @private\r\n         * @type {function(object,number):string} */\r\n        this.color = opts.color || ((f, z) => 'gray')\r\n        /**\r\n         * In pixel\r\n         * @private\r\n         * @type {function(object,number):number} */\r\n        this.width = opts.width || ((f, z) => 2)\r\n        /**\r\n         * @private\r\n         * @type {function(object,number):Array.<number>|undefined} */\r\n        this.lineDash = opts.lineDash || ((f, z) => undefined)\r\n\r\n        /**\r\n         * @private\r\n         * @type {Array.<object> | undefined} */\r\n        this.fs = undefined\r\n\r\n        /**\r\n         * @private\r\n         * @type {string} */\r\n        this.loadingStatus = 'notLoaded'\r\n    }\r\n\r\n    /**\r\n     * Draw the layer.\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @returns {void}\r\n     */\r\n    draw(geoCanvas) {\r\n        //load data, if not done yet.\r\n        if (!this.fs) {\r\n            this.load(geoCanvas.redraw)\r\n            return\r\n        }\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        for (const f of this.fs) {\r\n            const gt = f.geometry.type\r\n\r\n            if (gt == 'Point') {\r\n                const c = f.geometry.coordinates\r\n\r\n                //get style parameters for the point feature\r\n                const shape = this.shape(f, z)\r\n                if (!shape || shape == 'none') continue\r\n                const size = this.size(f, z) * z\r\n                if (!size) continue\r\n                const strokeStyle = this.strokeStyle(f, z)\r\n                const fillStyle = this.fillStyle(f, z)\r\n                const lineWidth = this.lineWidth(f, z) * z\r\n\r\n                //set canvas drawing parameters\r\n                if (strokeStyle) ctx.strokeStyle = strokeStyle\r\n                if (fillStyle) ctx.fillStyle = fillStyle\r\n                if (lineWidth) ctx.lineWidth = lineWidth\r\n\r\n                if (shape == 'circle') {\r\n                    //draw circle - fill and stroke\r\n                    ctx.beginPath()\r\n                    ctx.arc(c[0], c[1], size / 2, 0, 2 * Math.PI, false)\r\n                    if (fillStyle) ctx.fill()\r\n                    if (strokeStyle && lineWidth) ctx.stroke()\r\n                } else if (shape == 'square') {\r\n                    //draw square - fill and stroke\r\n                    ctx.beginPath()\r\n                    ctx.rect(c[0] - size / 2, c[1] - size / 2, size, size)\r\n                    if (fillStyle) ctx.fill()\r\n                    if (strokeStyle && lineWidth) ctx.stroke()\r\n                } else {\r\n                    console.error('Unexpected shape for point geojson: ' + shape)\r\n                }\r\n            } else if (gt == 'LineString') {\r\n                const cs = f.geometry.coordinates\r\n                if (cs.length < 2) continue\r\n\r\n                //set color\r\n                const col = this.color(f, z)\r\n                if (!col || col == 'none') continue\r\n                ctx.strokeStyle = col\r\n\r\n                //set linewidth\r\n                const wP = this.width(f, z)\r\n                if (!wP || wP < 0) continue\r\n                ctx.lineWidth = wP * z\r\n\r\n                //set line dash\r\n                const ldP = this.lineDash(f, z)\r\n                if (ldP) ctx.setLineDash(ldP)\r\n\r\n                //draw line\r\n                ctx.beginPath()\r\n                ctx.moveTo(cs[0][0], cs[0][1])\r\n                for (let i = 1; i < cs.length; i++) ctx.lineTo(cs[i][0], cs[i][1])\r\n                ctx.stroke()\r\n            } else {\r\n                console.log('Unsupported geometry type in GeoJSONLayer: ' + gt)\r\n            }\r\n        }\r\n\r\n        //...\r\n        ctx.setLineDash([])\r\n    }\r\n\r\n    /**\r\n     * Load data for labels, from URL this.url\r\n     * @param {function():void} callback\r\n     * @private\r\n     */\r\n    async load(callback) {\r\n        if (!this.url) {\r\n            console.log('Failed loading boundaries: No URL specified. ' + this.url)\r\n            this.loadingStatus = 'failed'\r\n            this.labels = []\r\n            return\r\n        }\r\n\r\n        //check if data already loaded\r\n        if (this.loadingStatus != 'notLoaded') return\r\n\r\n        //load data\r\n        this.loadingStatus = 'loading'\r\n\r\n        try {\r\n            const data_ = await json(this.url)\r\n\r\n            /** @type { Array.<object> } */\r\n            const data = data_.features\r\n\r\n            //preprocess/filter\r\n            if (this.preprocess) {\r\n                this.fs = []\r\n                for (const c of data) {\r\n                    const b = this.preprocess(c)\r\n                    if (b == false) continue\r\n                    this.fs.push(c)\r\n                }\r\n            } else {\r\n                //store labels\r\n                this.fs = data\r\n            }\r\n\r\n            this.loadingStatus = 'loaded'\r\n\r\n            //redraw\r\n            if (callback) callback()\r\n        } catch (error) {\r\n            console.log('Failed loading boundaries from ' + this.url)\r\n            this.fs = []\r\n            this.loadingStatus = 'failed'\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Layer } from '../core/Layer.js'\r\n\r\n/**\r\n * A layer, which specifies a dataset to be shown with specified styles.\r\n *\r\n * @module layer\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class GridLayer extends Layer {\r\n    /**\r\n     * @param {import(\"../core/Dataset\").Dataset|import(\"../core/MultiResolutionDataset\").MultiResolutionDataset} dataset The dataset to show.\r\n     * @param {Array.<import(\"../core/Style\").Style>} styles The styles, ordered in drawing order.\r\n     * @param {{visible?:function(number):boolean,alpha?:function(number):number,blendOperation?:function(number):GlobalCompositeOperation,minPixelsPerCell?:number,cellInfoHTML?:function(import(\"../core/Dataset\").Cell):string}} opts\r\n     */\r\n    constructor(dataset, styles, opts = {}) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** @type {import(\"../core/Dataset\").Dataset|import(\"../core/MultiResolutionDataset\").MultiResolutionDataset} */\r\n        this.dataset = dataset\r\n\r\n        /** @type {Array.<import(\"../core/Style\").Style>} */\r\n        this.styles = styles\r\n\r\n        /**\r\n         * This parameter is used when the dataset is a MultiResolutionDataset.\r\n         * It defines the minimum number of pixels a grid cell should have to select the dataset to display based on its resolution.\r\n         * A low value, means that the map will be more detailled (smaller cells).\r\n         * A high value, means that the map will be less detailled (larger cells).\r\n         * This value should be higher than 1, otherwise it means a grid cell is smaller than the screen resolution.\r\n         * For more complex cell representations that require some more map space, this value should be higher.\r\n         * @type {number} */\r\n        this.minPixelsPerCell = opts.minPixelsPerCell || 3\r\n\r\n        /**\r\n         * The function returning cell information as HTML.\r\n         * This is typically used for tooltip information.\r\n         * @type {function(import(\"../core/Dataset\").Cell, number):string} */\r\n        this.cellInfoHTML = opts.cellInfoHTML || GridLayer.defaultCellInfoHTML\r\n    }\r\n\r\n    /** */\r\n    draw(geoCanvas, legend) {\r\n        //get zoom level\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get layer dataset component\r\n        /** @type {import('../core/Dataset.js').Dataset|undefined} */\r\n        const dsc = this.getDataset(z)\r\n        if (!dsc) return\r\n\r\n        //launch data download, if necessary\r\n        dsc.getData(geoCanvas.extGeo)\r\n\r\n        //update dataset view cache\r\n        dsc.updateViewCache(geoCanvas.extGeo)\r\n\r\n        //draw cells, style by style\r\n        for (const s of this.styles) {\r\n            //check if style is visible\r\n            if (s.visible && !s.visible(z)) continue\r\n\r\n            //set style alpha and blend mode\r\n            //TODO: multiply by layer alpha ?\r\n            ctx.globalAlpha = s.alpha ? s.alpha(z) : 1.0\r\n            if (s.blendOperation) ctx.globalCompositeOperation = s.blendOperation(z)\r\n\r\n            //set affin transform to draw with geographical coordinates\r\n            geoCanvas.setCanvasTransform()\r\n\r\n            //draw with style\r\n            s.draw(dsc.getViewCache(), geoCanvas, dsc.getResolution())\r\n\r\n            //draw style filter\r\n            if (s.filterColor) s.drawFilter(geoCanvas)\r\n        }\r\n\r\n        //add legend element\r\n        if (legend) {\r\n            for (const s of this.styles) {\r\n                //check if style is visible\r\n                if (s.visible && !s.visible(z)) continue\r\n                GridLayer.addLegends(legend, s.legends)\r\n\r\n                //case for styles of styles, like kernel smoothing\r\n                //TODO do better\r\n                if (s['styles']) {\r\n                    for (const s2 of s['styles']) {\r\n                        if (s2.visible && !s2.visible(z)) continue\r\n                        GridLayer.addLegends(legend, s2.legends)\r\n                    }\r\n                }\r\n            }\r\n        }\r\n    }\r\n\r\n    /** @private */\r\n    static addLegends(legendComp, lg) {\r\n        if (Array.isArray(lg)) for (const lg_ of lg) this.addLegends(legendComp, lg_)\r\n        else legendComp.node().append(lg.div.node())\r\n    }\r\n\r\n    /**\r\n     * Return the relevant dataset component for a specified zoom.\r\n     *\r\n     * @param {number} z\r\n     * @returns {import(\"../core/Dataset\").Dataset|undefined}\r\n     * */\r\n    getDataset(z) {\r\n        return this.dataset.getDataset(z, this.minPixelsPerCell)\r\n    }\r\n\r\n    /**\r\n     * Set/get style stack.\r\n     *\r\n     * @param {undefined|import(\"../core/Style\").Style|Array.<import(\"../core/Style\").Style>} styles\r\n     * @returns { this | Array.<import(\"../core/Style\").Style> }\r\n     */\r\n    styles_(styles) {\r\n        if (arguments.length === 0) return this.styles\r\n        if (arguments.length === 1)\r\n            if (Array.isArray(styles)) this.styles = styles\r\n            else this.styles = [styles]\r\n        else this.styles = arguments\r\n        return this\r\n    }\r\n\r\n    /**\r\n     * The default function returning cell information as HTML.\r\n     * This is typically used for tooltip information.\r\n     *\r\n     * @param {import(\"../core/Dataset\").Cell} cell\r\n     * @returns {string}\r\n     */\r\n    static defaultCellInfoHTML(cell) {\r\n        const buf = []\r\n        for (const key of Object.keys(cell)) {\r\n            if (key === 'x') continue\r\n            if (key === 'y') continue\r\n            buf.push('<b>', key, '</b>', ' : ', cell[key], '<br>')\r\n        }\r\n        return buf.join('')\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Layer } from '../core/Layer.js'\r\nimport { csv } from 'd3-fetch'\r\n\r\n/** A label. The name is the text to show. (x,y) are the coordinates in the same CRS as the grid.\r\n * @typedef {{name: string, x:number, y:number }} Label */\r\n\r\n/**\r\n * A (generic) layer for placename labels, to be shown on top of the grid layers.\r\n * The input is a CSV file with the position (x, y) of the labels and name + some other info on the label importance.\r\n * If the label data is not in the expected format or in the same CRS as the grid, it can be corrected with the \"preprocess\" function.\r\n * The selection of the label, their style (font, weight, etc.) and color can be specified depending on their importance and the zoom level.\r\n *\r\n * @module layer\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class LabelLayer extends Layer {\r\n    /**\r\n     * @param {object} opts\r\n     */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * The URL of the label data, as CSV file.\r\n         * The file should contain the information for each label such as the text, the position and other information for the display of the label according to the zoom level.\r\n         * If necessary, this data can be reformated with the 'preprocess' parameter.\r\n         * @private\r\n         * @type {string} */\r\n        this.url = opts.url\r\n\r\n        /** Specify if and how a label should be drawn, depending on its importance and the zoom level.\r\n         * @private\r\n         * @type {function(Label,number):string} */\r\n        this.style = opts.style || (() => 'bold 1em Arial')\r\n\r\n        /** Specify the label color, depending on its importance and the zoom level.\r\n         * @private\r\n         * @type {function(Label,number):string} */\r\n        this.color = opts.color || (opts.dark ? () => '#ddd' : () => '#222')\r\n\r\n        /** Specify the label halo color, depending on its importance and the zoom level.\r\n         * @private\r\n         * @type {function(Label,number):string} */\r\n        this.haloColor = opts.haloColor || (opts.dark ? () => '#000000BB' : () => '#FFFFFFBB')\r\n\r\n        /** Specify the label halo width, depending on its importance and the zoom level.\r\n         * @private\r\n         * @type {function(Label,number):number} */\r\n        this.haloWidth = opts.haloWidth || (() => 4)\r\n\r\n        /** The anchor where to draw the text, from label position. See HTML-canvas textAlign property.\r\n         * \"left\" || \"right\" || \"center\" || \"start\" || \"end\"\r\n         * @private\r\n         * @type {CanvasTextAlign} */\r\n        this.textAlign = opts.textAlign || 'start'\r\n\r\n        /**\r\n         * @private\r\n         * @type {Array.<number>} */\r\n        this.offsetPix = opts.offsetPix || [5, 5]\r\n\r\n        /**\r\n         * A preprocess to run on each label after loading.\r\n         * It can be used to apply some specific treatment before, format the label data, project coordinates, etc.\r\n         * Return false if the label should not be kept.\r\n         * @private\r\n         * @type {function(Label):boolean} */\r\n        this.preprocess = opts.preprocess\r\n\r\n        /**\r\n         * @private\r\n         * @type {Array.<Label> | undefined} */\r\n        this.labels = undefined\r\n\r\n        /**\r\n         * @private\r\n         * @type {string} */\r\n        this.loadingStatus = 'notLoaded'\r\n    }\r\n\r\n    /**\r\n     * Draw the label layer.\r\n     *\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas The canvas where to draw the layer.\r\n     * @returns {void}\r\n     */\r\n    draw(geoCanvas) {\r\n        //load labels, if not done yet.\r\n        if (!this.labels) {\r\n            this.load(geoCanvas.redraw)\r\n            return\r\n        }\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //text align\r\n        ctx.textAlign = this.textAlign || 'start'\r\n\r\n        //line join and cap\r\n        ctx.lineJoin = 'bevel' //|| \"round\" || \"miter\";\r\n        ctx.lineCap = 'butt' //|| \"round\" || \"square\";\r\n\r\n        //draw in pix coordinates\r\n        geoCanvas.initCanvasTransform()\r\n\r\n        //draw labels, one by one\r\n        for (const lb of this.labels) {\r\n            //get label style\r\n            const st = this.style(lb, z)\r\n            if (!st) continue\r\n            ctx.font = st\r\n\r\n            //check label within the view, to be drawn\r\n            if (!geoCanvas.toDraw(lb)) continue\r\n\r\n            //position\r\n            const xP = geoCanvas.geoToPixX(lb.x) + this.offsetPix[0]\r\n            const yP = geoCanvas.geoToPixY(lb.y) - this.offsetPix[1]\r\n\r\n            //label stroke, for the halo\r\n            if (this.haloColor && this.haloWidth) {\r\n                const hc = this.haloColor(lb, z)\r\n                const hw = this.haloWidth(lb, z)\r\n                if (hc && hw && hw > 0) {\r\n                    ctx.strokeStyle = hc\r\n                    ctx.lineWidth = hw\r\n                    ctx.strokeText(lb.name, xP, yP)\r\n                }\r\n            }\r\n\r\n            //label fill\r\n            if (this.color) {\r\n                const col = this.color(lb, z)\r\n                if (col) {\r\n                    ctx.fillStyle = col\r\n                    ctx.fillText(lb.name, xP, yP)\r\n                }\r\n            }\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Load data for labels, from URL this.url\r\n     * @param {function():void} callback\r\n     * @private\r\n     */\r\n    async load(callback) {\r\n        if (!this.url) {\r\n            console.log('Failed loading labels: No URL specified. ' + this.url)\r\n            this.loadingStatus = 'failed'\r\n            this.labels = []\r\n            return\r\n        }\r\n\r\n        //check if data already loaded\r\n        if (this.loadingStatus != 'notLoaded') return\r\n\r\n        //load data\r\n        this.loadingStatus = 'loading'\r\n\r\n        try {\r\n            /** @type { Array.<Label> } */\r\n            const data = await csv(this.url)\r\n\r\n            //preprocess/filter\r\n            if (this.preprocess) {\r\n                this.labels = []\r\n                for (const c of data) {\r\n                    const b = this.preprocess(c)\r\n                    if (b == false) continue\r\n                    this.labels.push(c)\r\n                }\r\n            } else {\r\n                //store labels\r\n                this.labels = data\r\n            }\r\n\r\n            this.loadingStatus = 'loaded'\r\n\r\n            //redraw\r\n            if (callback) callback()\r\n        } catch (error) {\r\n            console.log('Failed loading labels from ' + this.url)\r\n            this.labels = []\r\n            this.loadingStatus = 'failed'\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Legend } from '../core/Legend.js'\r\n\r\n/**\r\n * A legend element for color categrories.\r\n *\r\n * @module legend\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class ColorCategoryLegend extends Legend {\r\n    /** @param {Object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        //col/categories array, in display order\r\n        /**\r\n         * @private\r\n         * @type {Array.<[string,string]>} */\r\n        this.colorLabel = opts.colorLabel || [['gray', '-']]\r\n\r\n        /**\r\n         * @private\r\n         * @type {import(\"../core/Style.js\").Shape} */\r\n        this.shape = opts.shape || 'circle'\r\n        this.dimension = opts.dimension || { r: 8 }\r\n        this.strokeColor = opts.strokeColor || 'gray'\r\n        this.strokeWidth = opts.strokeWidth || 1\r\n    }\r\n\r\n    /**\r\n     */\r\n    update() {\r\n        //clear\r\n        this.div.selectAll('*').remove()\r\n\r\n        //title\r\n        this.makeTitle()\r\n\r\n        //categories\r\n        const nb = this.colorLabel.length\r\n        if (nb == 0) return\r\n\r\n        for (let i = 0; i < nb; i++) {\r\n            const cat = this.colorLabel[i]\r\n\r\n            //make div for category\r\n            const d = this.div.append('div')\r\n            //to enable vertical centering\r\n            //.style(\"position\", \"relative\")\r\n\r\n            const sw = this.strokeWidth\r\n\r\n            //draw graphic element: box / circle\r\n            if (this.shape === 'square') {\r\n                const h = this.dimension.h || 15\r\n                const w = this.dimension.w || 20\r\n                d.append('div')\r\n                    .style('display', 'inline')\r\n\r\n                    .append('svg')\r\n                    .attr('width', w + 2 * sw)\r\n                    .attr('height', h + 2 * sw)\r\n\r\n                    .append('rect')\r\n                    .attr('x', sw)\r\n                    .attr('y', sw)\r\n                    .attr('width', w)\r\n                    .attr('height', h)\r\n                    .style('fill', cat[0])\r\n                    .style('stroke', this.strokeColor)\r\n                    .style('stroke-width', this.strokeWidth)\r\n            } else if (this.shape === 'circle') {\r\n                const r = this.dimension.r || 8\r\n                const h = 2 * r + 2 * sw\r\n                d.append('div')\r\n                    .style('display', 'inline')\r\n\r\n                    .append('svg')\r\n                    .attr('width', h)\r\n                    .attr('height', h)\r\n\r\n                    .append('circle')\r\n                    .attr('cx', r + sw)\r\n                    .attr('cy', r + sw)\r\n                    .attr('r', r)\r\n                    .style('fill', cat[0])\r\n                    .style('stroke', this.strokeColor)\r\n                    .style('stroke-width', this.strokeWidth)\r\n            } else {\r\n                throw new Error('Unexpected shape:' + this.shape)\r\n            }\r\n\r\n            //write label text\r\n            d.append('div')\r\n                //show on right of graphic\r\n                .style('display', 'inline')\r\n\r\n                //center vertically\r\n                //.style(\"position\", \"absolute\").style(\"top\", \"0\").style(\"bottom\", \"0\")\r\n\r\n                .style('padding-left', '5px')\r\n                .style('font-size', this.labelFontSize)\r\n                .text(cat[1])\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Legend } from '../core/Legend.js'\r\n\r\n/**\r\n * A legend element for discrete color style.\r\n * Inspiration: https://observablehq.com/@d3/color-legend\r\n *\r\n * @module legend\r\n * @author Julien Gaffuri\r\n */\r\nexport class ColorDiscreteLegend extends Legend {\r\n    /** @param {Object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** @private @type {function(import('../core/Style').ViewScale):Array.<string>} */\r\n        this.colors = opts.colors\r\n        /** @private @type {function(import('../core/Style').ViewScale):Array.<number>} */\r\n        this.breaks = opts.breaks\r\n\r\n        this.width = opts.width || 300\r\n        this.height = opts.height || 15\r\n\r\n        this.tickSize = opts.tickSize || 3\r\n\r\n        //label\r\n        this.invert = opts.invert\r\n    }\r\n\r\n    /**\r\n     * @param {{viewScale:import('../core/Style').ViewScale} } opts\r\n     */\r\n    update(opts) {\r\n        //clear\r\n        this.div.selectAll('*').remove()\r\n\r\n        //title\r\n        this.makeTitle()\r\n\r\n        //get colors and breaks\r\n        const colors = this.colors(opts.viewScale)\r\n        const breaks = this.breaks(opts.viewScale)\r\n        if (!breaks) return\r\n\r\n        //classes\r\n        const nb = colors.length\r\n        if (nb == 0) return\r\n        const w = this.width / nb\r\n\r\n        //make svg element\r\n        const svg = this.div\r\n            .append('svg')\r\n            .attr('width', this.width)\r\n            .attr('height', this.height + this.tickSize + 2 + 10)\r\n\r\n        //draw graphic elements\r\n        for (let i = 0; i < nb; i++) {\r\n            svg.append('rect')\r\n                .attr('x', i * w)\r\n                .attr('y', 0)\r\n                .attr('width', w)\r\n                .attr('height', this.height)\r\n                .style('fill', colors[i])\r\n        }\r\n\r\n        //tick line\r\n        for (let i = 1; i < nb; i++) {\r\n            svg.append('line')\r\n                .attr('x1', w * i)\r\n                .attr('y1', 0)\r\n                .attr('x2', w * i)\r\n                .attr('y2', this.height + this.tickSize)\r\n                .style('stroke', 'black')\r\n        }\r\n\r\n        //labels\r\n        for (let i = 1; i < nb; i++) {\r\n            let label = breaks[i - 1]\r\n            if (isNaN(label) || label == undefined) continue\r\n            if (this.labelFormat) label = this.labelFormat(label)\r\n\r\n            //label\r\n            svg.append('text')\r\n                .attr('id', 'ticklabel_' + i)\r\n                .attr('x', w * i)\r\n                .attr('y', this.height + this.tickSize + 2)\r\n                .style('font-size', this.labelFontSize)\r\n                //.style(\"font-weight\", \"bold\")\r\n                //.style(\"font-family\", \"Arial\")\r\n                .style('text-anchor', 'middle')\r\n                .style('alignment-baseline', 'top')\r\n                .style('dominant-baseline', 'hanging')\r\n                .style('pointer-events', 'none')\r\n                .text(label)\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Legend } from '../core/Legend.js'\r\n\r\n/**\r\n * A legend element for continuous color style.\r\n * Inspiration: https://observablehq.com/@d3/color-legend\r\n *\r\n * @module legend\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class ColorLegend extends Legend {\r\n    /** @param {Object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        //a function [0,1]->color for continuous colors.\r\n        //it can take as second argument the viewscale.\r\n        this.colorScale = opts.colorScale\r\n\r\n        //function (t[0,1]) -> value (for label text)\r\n        //it can take as second argument the viewscale.\r\n        this.textScale = opts.textScale || ((t) => t)\r\n\r\n        this.margin = opts.margin || 5\r\n\r\n        //replace with labels ?\r\n        this.tickSize = opts.tickSize || 6\r\n        this.ticks = opts.ticks || Math.floor(this.width / 50)\r\n        this.tickFormat = opts.tickFormat\r\n        this.tickUnit = opts.tickUnit\r\n\r\n        this.fontSize = opts.fontSize || '0.8em'\r\n        this.invert = opts.invert\r\n\r\n        this.width = opts.width || 300\r\n        this.height = opts.height || 15\r\n    }\r\n\r\n    /**\r\n     * @param {{viewScale:import('../core/Style').ViewScale} } opts\r\n     */\r\n    update(opts) {\r\n        //clear\r\n        this.div.selectAll('*').remove()\r\n\r\n        //title\r\n        this.makeTitle()\r\n\r\n        const svgW = this.width + 2 * this.margin\r\n        const svgH = this.height + this.margin + this.tickSize + 10\r\n        const svg = this.div.append('svg').attr('width', svgW).attr('height', svgH)\r\n        //  <rect width=\"300\" height=\"100\" style=\"fill:rgb(0,0,255);stroke-width:3;stroke:rgb(0,0,0)\" />\r\n\r\n        const g = svg.append('g').attr('transform', 'translate(' + this.margin + ' ' + 0 + ')')\r\n\r\n        //draw color bar\r\n        const w = this.width,\r\n            h = this.height\r\n        const step = 5\r\n        for (let i = 0; i < w; i += step) {\r\n            let t = i / (w - 1)\r\n            if (this.invert) t = 1 - t\r\n            g.append('rect')\r\n                .attr('x', i)\r\n                .attr('y', 0)\r\n                .attr('width', step)\r\n                .attr('height', h)\r\n                .style('fill', this.colorScale(t, opts.viewScale))\r\n        }\r\n\r\n        for (let i = 0; i < this.ticks; i++) {\r\n            let t = i / (this.ticks - 1)\r\n\r\n            //tick line\r\n            g.append('line')\r\n                .attr('x1', w * t)\r\n                .attr('y1', 0)\r\n                .attr('x2', w * t)\r\n                .attr('y2', h + this.tickSize)\r\n                .style('stroke', 'black')\r\n\r\n            //prepare tick label\r\n            g.append('text')\r\n                .attr('id', 'ticklabel_' + i)\r\n                .attr('x', w * t)\r\n                .attr('y', h + this.tickSize + 2)\r\n                .style('font-size', this.fontSize)\r\n                //.style(\"font-weight\", \"bold\")\r\n                //.style(\"font-family\", \"Arial\")\r\n                .style('text-anchor', i == 0 ? 'start' : i == this.ticks - 1 ? 'end' : 'middle')\r\n                .style('alignment-baseline', 'top')\r\n                .style('dominant-baseline', 'hanging')\r\n                .style('pointer-events', 'none')\r\n            //.text(\"-\")\r\n        }\r\n\r\n        //update tick labels\r\n\r\n        //label text format\r\n        const f = this.tickFormat && this.tickFormat != 'text' ? this.tickFormat : (v) => v\r\n        for (let i = 0; i < this.ticks; i++) {\r\n            let t = i / (this.ticks - 1)\r\n\r\n            const v = this.textScale(t, opts.viewScale)\r\n            const text = (v ? f(v) : '0') + (this.tickUnit ? this.tickUnit : '')\r\n            if (text == undefined) continue\r\n\r\n            //tick label\r\n            this.div.select('#' + 'ticklabel_' + i).text(text)\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Legend } from '../core/Legend.js'\r\n\r\n/**\r\n * A legend element for segment orientation.\r\n *\r\n * @module legend\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class OrientationLegend extends Legend {\r\n    /** @param {Object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        //orientation\r\n        this.orientation = opts.orientation || 0\r\n        //color\r\n        this.color = opts.color || ((resolution, z, viewScale) => 'gray')\r\n        //width\r\n        this.width = opts.width || ((resolution, z, viewScale) => 3 * z)\r\n        //length\r\n        this.length = opts.length || ((resolution, z, viewScale) => resolution)\r\n\r\n        //label\r\n        this.label = opts.label || '-'\r\n    }\r\n\r\n    /**\r\n     * @param {{ style: import(\"../style/SegmentStyle.js\").SegmentStyle, resolution: number, z: number, viewScale:object }} opts\r\n     */\r\n    update(opts) {\r\n        //clear\r\n        this.div.selectAll('*').remove()\r\n\r\n        //title\r\n        this.makeTitle()\r\n\r\n        const d = this.div.append('div')\r\n\r\n        //compute segment color, width and length\r\n        const color = this.color(opts.resolution, opts.z, opts.viewScale)\r\n        const widthPix = this.width(opts.resolution, opts.z, opts.viewScale) / opts.z\r\n        const lengthPix = this.length(opts.resolution, opts.z, opts.viewScale) / opts.z\r\n\r\n        //draw SVG segment\r\n        const svgS = Math.max(lengthPix, widthPix)\r\n        const svg = d.append('svg').attr('width', svgS).attr('height', svgS).style('', 'inline-block')\r\n\r\n        const cos = Math.cos((-this.orientation * Math.PI) / 180)\r\n        const sin = Math.sin((-this.orientation * Math.PI) / 180)\r\n        const dc = svgS * 0.5,\r\n            l2 = lengthPix * 0.5\r\n        svg.append('line')\r\n            .attr('x1', dc - cos * l2)\r\n            .attr('y1', dc - sin * l2)\r\n            .attr('x2', dc + cos * l2)\r\n            .attr('y2', dc + sin * l2)\r\n            .style('stroke', color)\r\n            .style('stroke-width', widthPix)\r\n\r\n        //label\r\n        d.append('div')\r\n            .style('display', 'inline')\r\n            .style('padding-left', '5px')\r\n            .style('font-size', this.labelFontSize)\r\n            .text(this.label + (this.labelUnitText ? ' ' : '') + this.labelUnitText)\r\n    }\r\n}\r\n\r\n/**\r\n *\r\n * @param {Array.<number>} orientations\r\n * @param {Array.<string>} labels\r\n * @param {object} opts\r\n * @returns  { Array.<OrientationLegend> }\r\n */\r\nexport function orientationLegend(orientations, labels, opts = {}) {\r\n    const legends = []\r\n    for (let i = 0; i < orientations.length; i++) {\r\n        opts.title = i == 0 ? opts.title : undefined\r\n        opts.orientation = orientations[i]\r\n        opts.label = labels[i]\r\n        legends.push(new OrientationLegend(opts))\r\n    }\r\n    return legends\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Legend } from '../core/Legend.js'\r\nimport { nice } from '../utils/utils.js'\r\nimport { max } from 'd3-array'\r\n\r\n/**\r\n * A legend element for proportional symbols.\r\n *\r\n * @module legend\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class SizeLegend extends Legend {\r\n    /** @param {Object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the text label, from the view scale and list of cells, resolution and zoom\r\n         *  @type { function(object, Array.<import('../core/Dataset.js').Cell>, number, number):(number|string) } */\r\n        this.label = opts.label || undefined\r\n\r\n        /** A function returning the size of the legend symbol, in geo UoM, from the viewscale, resolution and zoom\r\n         *  @type { function(object, number, number):number } */\r\n        this.size = opts.size || undefined\r\n\r\n        //symbol\r\n        /**  @type {(import(\"../core/Style\").Shape)|\"line\"} */\r\n        this.shape = opts.shape || 'circle'\r\n\r\n        //general case\r\n        this.fillColor = opts.fillColor || 'none'\r\n        this.strokeColor = opts.strokeColor || 'gray'\r\n        this.strokeWidth = opts.strokeWidth || 1\r\n\r\n        //for line shape\r\n        //TODO this.orientation = opts.orientation || 0\r\n        this.color = opts.color || 'gray'\r\n        this.length = opts.length || ((resolution, z, viewScale) => resolution)\r\n    }\r\n\r\n    /**\r\n     * @param {{ viewScale:object, resolution: number, z:number, cells:Array.<import('../core/Dataset.js').Cell> }} opts\r\n     */\r\n    update(opts) {\r\n        //clear\r\n        this.div.selectAll('*').remove()\r\n\r\n        //title\r\n        this.makeTitle()\r\n\r\n        //get label. May not be a number (!)\r\n        let label = this.label(opts.viewScale, opts.cells, opts.resolution, opts.z)\r\n\r\n        //compute size of symbol, in pix\r\n        let sizePix\r\n        if (this.size) sizePix = this.size(opts.viewScale, opts.resolution, opts.z) / opts.z\r\n        else sizePix = opts.viewScale(+label) / opts.z\r\n        if (!sizePix) return\r\n\r\n        //format label, if specified and possible\r\n        if (this.labelFormat && !isNaN(+label)) label = this.labelFormat(label)\r\n\r\n        const d = this.div.append('div')\r\n        //to enable vertical centering\r\n        //.style(\"position\", \"relative\")\r\n\r\n        //default svg construction, for square and circle\r\n        const svg = () =>\r\n            d\r\n                .append('svg')\r\n                .attr('width', sizePix + this.strokeWidth + 2)\r\n                .attr('height', sizePix + this.strokeWidth + 2)\r\n                .style('', 'inline-block')\r\n\r\n        if (this.shape === 'square') {\r\n            svg()\r\n                .append('rect')\r\n                .attr('x', 0)\r\n                .attr('y', 0)\r\n                .attr('width', sizePix)\r\n                .attr('height', sizePix)\r\n                .style('fill', this.fillColor)\r\n                .style('stroke', this.strokeColor)\r\n                .style('stroke-width', this.strokeWidth)\r\n        } else if (this.shape === 'circle') {\r\n            // <circle cx=\"50\" cy=\"50\" r=\"40\" stroke=\"black\" stroke-width=\"3\" fill=\"red\" />\r\n            const r = (sizePix + this.strokeWidth) * 0.5\r\n            svg()\r\n                .append('circle')\r\n                .attr('cx', r + 1)\r\n                .attr('cy', r + 1)\r\n                .attr('r', r)\r\n                .style('fill', this.fillColor)\r\n                .style('stroke', this.strokeColor)\r\n                .style('stroke-width', this.strokeWidth)\r\n        } else if (this.shape === 'donut') {\r\n            //TODO\r\n        } else if (this.shape === 'diamond') {\r\n            //TODO\r\n        } else if (this.shape === 'line') {\r\n            //get segment length\r\n            let lengthPix = this.length\r\n                ? this.length(opts.resolution, opts.z, opts.viewScale)\r\n                : opts.resolution\r\n            lengthPix /= opts.z\r\n\r\n            const svg = d\r\n                .append('svg')\r\n                .attr('width', lengthPix)\r\n                .attr('height', sizePix)\r\n                .style('', 'inline-block')\r\n\r\n            //TODO orientation\r\n            //<line x1=\"0\" y1=\"0\" x2=\"200\" y2=\"200\" style=\"stroke:rgb(255,0,0);stroke-width:2\" />\r\n            svg.append('line')\r\n                .attr('x1', 0)\r\n                .attr('y1', sizePix / 2)\r\n                .attr('x2', lengthPix)\r\n                .attr('y2', sizePix / 2)\r\n                .style('stroke', this.color)\r\n                .style('stroke-width', sizePix)\r\n        } else {\r\n            throw new Error('Unexpected shape:' + this.shape)\r\n        }\r\n\r\n        //label\r\n        d.append('div')\r\n            .style('display', 'inline')\r\n            .style('padding-left', '5px')\r\n            .style('font-size', this.labelFontSize)\r\n            .text(label + (this.labelUnitText ? ' ' : '') + this.labelUnitText)\r\n    }\r\n}\r\n\r\n/**\r\n * @param {Array.<number>} values\r\n * @param {function(number):number} size\r\n * @param { object } opts\r\n * @returns {Array.<SizeLegend>}\r\n */\r\nexport function sizeLegend(values, size, opts = {}) {\r\n    const legends = []\r\n    for (let value of values) {\r\n        opts.title = value == values[0] ? opts.title : undefined\r\n        opts.size = () => size(value)\r\n        opts.label = () => value\r\n        legends.push(new SizeLegend(opts))\r\n    }\r\n    return legends\r\n}\r\n\r\n/**\r\n * @param { function(import('../core/Dataset.js').Cell):number } value\r\n * @param {*} opts\r\n * @returns {Array.<SizeLegend>}\r\n */\r\nexport function sizeLegendViewScale(value, opts = {}) {\r\n    const k = opts.k || [0.9, 0.5, 0.2, 0.05]\r\n    const legends = []\r\n    for (let k_ of k) {\r\n        opts.title = k_ == k[0] ? opts.title : undefined\r\n        opts.label = (viewScale, cells) => nice(k_ * max(cells, value))\r\n        legends.push(new SizeLegend(opts))\r\n    }\r\n    return legends\r\n}\r\n\r\n/**\r\n * A function which return a stack of size legends for a discrete classification.\r\n *\r\n * @param { Array.<number> } breaks\r\n * @param { Array.<number> } sizes\r\n * @param { object } opts\r\n * @returns {Array.<SizeLegend>}\r\n */\r\nexport function sizeDiscreteLegend(breaks, sizes, opts = {}) {\r\n    const f = opts.labelFormat || ((x) => x)\r\n    const labelText = opts.labelText || defaultLabelText(f)\r\n    const legends = []\r\n    for (let i = sizes.length - 1; i >= 0; i--) {\r\n        opts.title = i == sizes.length - 1 ? opts.title : undefined\r\n        opts.size = () => sizes[i]\r\n        opts.label = () => labelText(breaks[i - 1], breaks[i])\r\n        legends.push(new SizeLegend(opts))\r\n    }\r\n    return legends\r\n}\r\n\r\n/**\r\n * A function which return a stack of size legends for a discrete classification using a viewscale.\r\n * @param { number } classNumber\r\n * @param { object } opts\r\n * @returns {Array.<SizeLegend>}\r\n */\r\nexport function sizeDiscreteViewScaleLegend(classNumber, opts = {}) {\r\n    const f = opts.labelFormat || ((x) => x)\r\n    const labelText = opts.labelText || defaultLabelText(f)\r\n    const legends = []\r\n    const viewScaleFun = opts.viewScaleFun || ((t) => t) //TODO do it differently? At sizelegend level !\r\n    for (let i = classNumber - 1; i >= 0; i--) {\r\n        opts.title = i == classNumber - 1 ? opts.title : undefined\r\n        opts.size = (viewScale) => viewScaleFun(viewScale).values[i]\r\n        opts.label = (viewScale) =>\r\n            labelText(viewScaleFun(viewScale).breaks[i - 1], viewScaleFun(viewScale).breaks[i])\r\n        legends.push(new SizeLegend(opts))\r\n    }\r\n    return legends\r\n}\r\n\r\n/**\r\n * A function that returns a function to format laberls for discrete scale legends.\r\n * @param { function(number):string } format\r\n * @returns { function(number|undefined, number|undefined): string }\r\n */\r\nfunction defaultLabelText(format) {\r\n    return (v0, v1) => {\r\n        if (v0 == undefined && v1 == undefined) return ''\r\n        if (v1 == undefined) return '> ' + format(v0)\r\n        if (v0 == undefined) return '< ' + format(v1)\r\n        return format(v0) + ' - ' + format(v1)\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { select } from 'd3-selection'\r\nimport { Legend } from '../core/Legend.js'\r\n\r\n/**\r\n *\r\n * @module legend\r\n * @author Julien Gaffuri\r\n */\r\nexport class TernaryLegend extends Legend {\r\n    /** @param {Object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        //classifier\r\n        this.classifier = opts.classifier\r\n\r\n        this.width = opts.width || 150\r\n        this.selectionColor = this.selectionColor || 'red'\r\n        this.tooltip = opts.tooltip\r\n        this.texts = opts.texts\r\n\r\n        this.leftText = opts.leftText || 'Category 0'\r\n        this.topText = opts.topText || 'Category 1'\r\n        this.rightText = opts.rightText || 'Category 2'\r\n\r\n        this.centerCoefficient = opts.centerCoefficient || this.classifier.centerCoefficient\r\n    }\r\n\r\n    /**\r\n     * @param {{} } opts\r\n     */\r\n    update(opts) {\r\n        //clear\r\n        this.div.selectAll('*').remove()\r\n\r\n        //title\r\n        this.makeTitle()\r\n\r\n        const sqrt3over2 = 0.866025\r\n        const w = this.width,\r\n            h = w * sqrt3over2\r\n        const classifier = this.classifier\r\n        const selectionColor = this.selectionColor\r\n        const selectionStrokeWidth = 0\r\n        const tt = this.tooltip\r\n        const texts = this.texts || {}\r\n\r\n        const padding = 2\r\n        const fontSize = 12\r\n\r\n        //make svg element\r\n        const svg = this.div\r\n            .append('svg')\r\n            .attr('width', w + selectionStrokeWidth)\r\n            .attr('height', h + 4 * padding + 2 * fontSize)\r\n\r\n        //top label\r\n        svg.append('text')\r\n            .attr('x', w / 2)\r\n            .attr('y', padding + fontSize)\r\n            .text(this.topText)\r\n            .attr('font-size', fontSize)\r\n            .attr('text-anchor', 'middle')\r\n        //left label\r\n        svg.append('text')\r\n            .attr('x', 0)\r\n            .attr('y', 3 * padding + 2 * fontSize + h)\r\n            .text(this.leftText)\r\n            .attr('font-size', fontSize)\r\n            .attr('text-anchor', 'start')\r\n        //right label\r\n        svg.append('text')\r\n            .attr('x', w)\r\n            .attr('y', 3 * padding + 2 * fontSize + h)\r\n            .text(this.rightText)\r\n            .attr('font-size', fontSize)\r\n            .attr('text-anchor', 'end')\r\n\r\n        //triangle group\r\n        const g = svg\r\n            .append('g')\r\n            .attr(\r\n                'transform',\r\n                'translate(' +\r\n                    selectionStrokeWidth / 2 +\r\n                    ' ' +\r\n                    (selectionStrokeWidth / 2 + (2 * padding + fontSize)) +\r\n                    ')'\r\n            )\r\n\r\n        //common function for triangle patches\r\n        const setAttributes = (elt, color, text) => {\r\n            //elt.raise();\r\n            elt.attr('fill', color)\r\n                //.attr(\"stroke\", colorOver)\r\n                //.attr(\"stroke-width\", 0)\r\n                //.attr(\"stroke-linejoin\", \"round\")\r\n                .on('mouseover', function (e) {\r\n                    /*this.parentNode.appendChild(this); select(this).attr(\"stroke-width\", selectionStrokeWidth);*/\r\n                    select(this).attr('fill', selectionColor)\r\n                    if (!tt || !text) return\r\n                    tt.html(text)\r\n                    tt.setPosition(e)\r\n                    tt.show()\r\n                })\r\n                .on('mouseout', function () {\r\n                    /*select(this).attr(\"stroke-width\", 0);*/\r\n                    select(this).attr('fill', color)\r\n                    if (tt) tt.hide()\r\n                })\r\n            if (tt && text)\r\n                elt.on('mousemove', function (e) {\r\n                    tt.setPosition(e)\r\n                })\r\n        }\r\n\r\n        //const [c0, c1, c2] = classifier.center\r\n\r\n        //trapezium s0\r\n        const t0 = g\r\n            .append('polygon')\r\n            .attr(\r\n                'points',\r\n                '0,' +\r\n                    h +\r\n                    ' ' +\r\n                    w / 3 +\r\n                    ',' +\r\n                    h +\r\n                    ' ' +\r\n                    w / 2 +\r\n                    ',' +\r\n                    (h * 2) / 3 +\r\n                    ' ' +\r\n                    w / 6 +\r\n                    ',' +\r\n                    (h * 2) / 3\r\n            )\r\n        setAttributes(t0, classifier.colors[0], texts['0'])\r\n        //trapezium s1\r\n        const t1 = g\r\n            .append('polygon')\r\n            .attr(\r\n                'points',\r\n                w / 2 +\r\n                    ',0 ' +\r\n                    (w * 2) / 3 +\r\n                    ',' +\r\n                    h / 3 +\r\n                    ' ' +\r\n                    w / 2 +\r\n                    ',' +\r\n                    (h * 2) / 3 +\r\n                    ' ' +\r\n                    w / 3 +\r\n                    ',' +\r\n                    h / 3\r\n            )\r\n        setAttributes(t1, classifier.colors[1], texts['1'])\r\n        //trapezium s2\r\n        const t2 = g\r\n            .append('polygon')\r\n            .attr(\r\n                'points',\r\n                w +\r\n                    ',' +\r\n                    h +\r\n                    ' ' +\r\n                    (w * 5) / 6 +\r\n                    ',' +\r\n                    (2 * h) / 3 +\r\n                    ' ' +\r\n                    w / 2 +\r\n                    ',' +\r\n                    (h * 2) / 3 +\r\n                    ' ' +\r\n                    (w * 2) / 3 +\r\n                    ',' +\r\n                    h\r\n            )\r\n        setAttributes(t2, classifier.colors[2], texts['2'])\r\n        //triangle s0\r\n        const t0_ = g\r\n            .append('polygon')\r\n            .attr(\r\n                'points',\r\n                w / 2 +\r\n                    ',' +\r\n                    (h * 2) / 3 +\r\n                    ' ' +\r\n                    (w * 5) / 6 +\r\n                    ',' +\r\n                    (h * 2) / 3 +\r\n                    ' ' +\r\n                    (w * 2) / 3 +\r\n                    ',' +\r\n                    h / 3\r\n            )\r\n        setAttributes(t0_, classifier.mixColors[0], texts['m12'])\r\n        //triangle s1\r\n        const t1_ = g\r\n            .append('polygon')\r\n            .attr('points', w / 2 + ',' + (h * 2) / 3 + ' ' + w / 3 + ',' + h + ' ' + (w * 2) / 3 + ',' + h)\r\n        setAttributes(t1_, classifier.mixColors[1], texts['m02'])\r\n        //triangle s2\r\n        const t2_ = g\r\n            .append('polygon')\r\n            .attr(\r\n                'points',\r\n                w / 2 + ',' + (h * 2) / 3 + ' ' + w / 6 + ',' + (h * 2) / 3 + ' ' + w / 3 + ',' + h / 3\r\n            )\r\n        setAttributes(t2_, classifier.mixColors[2], texts['m01'])\r\n\r\n        //center\r\n        if (this.centerCoefficient) {\r\n            //TODO make it an hexagon !\r\n            const center = g\r\n                .append('circle')\r\n                .attr('cx', w / 2)\r\n                .attr('cy', (h * 2) / 3)\r\n                .attr('r', (this.centerCoefficient * h) / 3)\r\n            setAttributes(center, classifier.centerColor, texts['center'])\r\n        }\r\n\r\n        /*\r\n        let middle, left, top, right, left_, bottom_, right_\r\n        if (!this.real) {\r\n\r\n            //0 left triangle\r\n            left = g.append('polygon')\r\n                .attr('points', \"0,\" + h + \" \" + (w / 3) + \",\" + h + \" \" + (w / 6) + \",\" + (2 * h / 3))\r\n            //1 top triangle\r\n            top = g.append('polygon')\r\n                .attr('points', (w / 3) + \",\" + (h / 3) + \" \" + (w * 2 / 3) + \",\" + (h / 3) + \" \" + (w / 2) + \",0\")\r\n            //2 right triangle\r\n            right = g.append('polygon')\r\n                .attr('points', (w * 2 / 3) + \",\" + h + \" \" + w + \",\" + h + \" \" + (w * 5 / 6) + \",\" + (2 * h / 3))\r\n            //middle triangle\r\n            middle = g.append('polygon')\r\n                .attr('points', (w / 2) + \",\" + (h / 3) + \" \" + (w / 4) + \",\" + (h * 5 / 6) + \" \" + (3 * w / 4) + \",\" + (h * 5 / 6))\r\n            //01 left trapezium\r\n            left_ = g.append('polygon')\r\n                .attr('points', (w / 6) + \",\" + (h * 2 / 3) + \" \" + (w / 4) + \",\" + (h * 5 / 6) + \" \" + (w / 2) + \",\" + (h / 3) + \" \" + (w / 3) + \",\" + (h / 3))\r\n            //02 bottom trapezium\r\n            bottom_ = g.append('polygon')\r\n                .attr('points', (w / 3) + \",\" + (h) + \" \" + (2 * w / 3) + \",\" + (h) + \" \" + (w * 3 / 4) + \",\" + (h * 5 / 6) + \" \" + (w / 4) + \",\" + (h * 5 / 6))\r\n            //12 right trapezium\r\n            right_ = g.append('polygon')\r\n                .attr('points', (w / 2) + \",\" + (h / 3) + \" \" + (w * 3 / 4) + \",\" + (h * 5 / 6) + \" \" + (w * 5 / 6) + \",\" + (h * 2 / 3) + \" \" + (w * 2 / 3) + \",\" + (h / 3))\r\n\r\n        } else {\r\n\r\n            //middle triangle\r\n            middle = g.append('polygon')\r\n                .attr('points', (w / 2) + \",0 0,\" + h + \" \" + w + \",\" + h)\r\n\r\n            //draw trapezium\r\n            //draw large trapezium first\r\n            for (let i_ = 2; i_ >= 0; i_--) {\r\n                const i = this.classifier.lowIndex[i_]\r\n                const r = this.classifier.lowThreshold[i]\r\n                if (i == 2)\r\n                    //01 left trapezium\r\n                    left_ = g.append('polygon')\r\n                        .attr('points', w / 2 + \",0 0,\" + h + \" \" + w * r + \",\" + h + \" \" + w * (1 + r) / 2 + \",\" + r * h)\r\n                else if (i == 1)\r\n                    //02 bottom trapezium\r\n                    bottom_ = g.append('polygon')\r\n                        .attr('points', \"0,\" + h + \" \" + w + \",\" + h + \" \" + w * (1 - r / 2) + \",\" + h * (1 - r) + \" \" + r * w / 2 + \",\" + h * (1 - r))\r\n                else\r\n                    //12 right trapezium\r\n                    right_ = g.append('polygon')\r\n                        .attr('points', w + \",\" + h + \" \" + w / 2 + \",0 \" + w * (1 - r) / 2 + \",\" + h * r + \" \" + w * (1 - r) + \",\" + h)\r\n            }\r\n\r\n            //draw triangles\r\n            //draw large triangles first\r\n            for (let i_ = 2; i_ >= 0; i_--) {\r\n                const i = this.classifier.highIndex[i_]\r\n                const r = this.classifier.highThreshold[i]\r\n\r\n                if (i == 2)\r\n                    //2 right triangle\r\n                    right = g.append('polygon')\r\n                        .attr('points', w + \",\" + h + \" \" + w * r + \",\" + h + \" \" + w * (1 + r) / 2 + \",\" + h * r)\r\n                else if (i == 1)\r\n                    //1 top triangle\r\n                    top = g.append('polygon')\r\n                        .attr('points', (w / 2) + \",0 \" + w * r / 2 + \",\" + h * (1 - r) + \" \" + w * (1 - r / 2) + \",\" + h * (1 - r))\r\n                else\r\n                    //0 left triangle\r\n                    left = g.append('polygon')\r\n                        .attr('points', \"0,\" + h + \" \" + w * (1 - r) + \",\" + h + \" \" + w * (1 - r) / 2 + \",\" + h * r)\r\n            }\r\n\r\n        }*/\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/** @typedef {\"flag\"|\"piechart\"|\"ring\"|\"segment\"|\"radar\"|\"agepyramid\"|\"halftone\"} CompositionType */\r\n\r\n/**\r\n * A style showing the composition of a total in different categories, with different color hues.\r\n * It consists of a symbol with different parts, whose size reflect the proportion of the corresponding category.\r\n * For a list of supported symbols, @see CompositionType\r\n * The symbol can be scaled depending on the cell importance.\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class CompositionStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * The dictionary (string -> color) which give the color of each category.\r\n         * @type {object} */\r\n        this.color = opts.color\r\n\r\n        /**\r\n         * A function returning the type of decomposition symbol of a cell, @see CompositionType\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number, number,object):CompositionType} */\r\n        this.type = opts.type || (() => 'flag') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the size of a cell in geographical unit.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.size = opts.size || ((c, r) => r) //(c,r,z,vs) => {}\r\n\r\n        /** For style types with stripes (flag, segment), the orientation of the stripes (0 for horizontal, other for vertical).\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number,number,object):number} */\r\n        this.stripesOrientation = opts.stripesOrientation || (() => 0) //(c,r,z,vs) => ...\r\n\r\n        /** The function specifying an offset angle for a radar, halftone or pie chart style.\r\n         * The angle is specified in degree. The rotation is anti-clockwise.\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number,number,object):number} */\r\n        this.offsetAngle = opts.offsetAngle || (() => 0) //(c,r,z,vs) => ...\r\n\r\n        /** The function specifying the height of the age pyramid, in geo unit.\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number,number,object):number} */\r\n        this.agePyramidHeight = opts.agePyramidHeight || ((c, r) => r) //(c,r,z,vs) => ...\r\n\r\n        /** For pie chart, this is parameter for internal radius, so that the pie chart looks like a donut.\r\n         * 0 for normal pie charts, 0.5 to empty half of the radius.\r\n         * @type {number} */\r\n        this.pieChartInternalRadiusFactor = opts.pieChartInternalRadiusFactor || 0\r\n    }\r\n\r\n    /**\r\n     * Draw cells as squares depending on their value.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //nb categories - used for radar and agepyramid\r\n        const nbCat = Object.entries(this.color).length\r\n\r\n        //draw calls\r\n        for (let cell of cells) {\r\n            //size\r\n            const sG = this.size ? this.size(cell, resolution, z, viewScale) : resolution\r\n            if (!sG) continue\r\n\r\n            //get offset\r\n            const offset = this.offset(cell, resolution, z)\r\n\r\n            //get symbol type\r\n            const type_ = this.type ? this.type(cell, resolution, z, viewScale) : 'flag'\r\n\r\n            //compute center position\r\n            const xc = cell.x + offset.dx + (type_ === 'agepyramid' ? 0 : resolution * 0.5)\r\n            const yc = cell.y + offset.dy + (type_ === 'agepyramid' ? 0 : resolution * 0.5)\r\n\r\n            //compute offset angle, when relevant\r\n            const offAng = this.offsetAngle\r\n                ? (this.offsetAngle(cell, resolution, z, viewScale) * Math.PI) / 180\r\n                : 0\r\n\r\n            if (type_ === 'agepyramid' || type_ === 'radar' || type_ === 'halftone') {\r\n                //get cell category max value\r\n                let maxVal = -Infinity\r\n                for (let key of Object.keys(this.color)) {\r\n                    const v = +cell[key]\r\n                    if (v > maxVal) maxVal = v\r\n                }\r\n\r\n                //cumul\r\n                let cumul = 0\r\n                if (type_ === 'agepyramid' && this.agePyramidHeight)\r\n                    cumul = (resolution - this.agePyramidHeight(cell, resolution, z, viewScale)) / 2\r\n                if (type_ === 'radar' || type_ === 'halftone') cumul = Math.PI / 2 + offAng\r\n\r\n                //compute the increment, which is the value to increment the cumul for each category\r\n                const incr =\r\n                    type_ === 'agepyramid'\r\n                        ? (this.agePyramidHeight\r\n                              ? this.agePyramidHeight(cell, resolution, z, viewScale)\r\n                              : resolution) / nbCat\r\n                        : type_ === 'radar' || type_ === 'halftone'\r\n                        ? (2 * Math.PI) / nbCat\r\n                        : undefined\r\n                if (incr === undefined) throw new Error('Unexpected symbol type:' + type_)\r\n\r\n                for (let [column, color] of Object.entries(this.color)) {\r\n                    if (type_ === 'agepyramid') {\r\n                        //set category color\r\n                        ctx.fillStyle = color\r\n\r\n                        //get category value\r\n                        const val = cell[column]\r\n\r\n                        //compute category length - in geo\r\n                        /** @type {number} */\r\n                        const wG = (sG * val) / maxVal\r\n\r\n                        //draw bar\r\n                        ctx.fillRect(xc + (resolution - wG) / 2, yc + cumul, wG, incr)\r\n\r\n                        //next height\r\n                        cumul += incr\r\n                    } else if (type_ === 'radar') {\r\n                        //set category color\r\n                        ctx.fillStyle = color\r\n\r\n                        //get categroy value\r\n                        const val = cell[column]\r\n\r\n                        //compute category radius - in geo\r\n                        /** @type {number} */\r\n                        //const rG = this.radius(val, r, stat, cellStat, z)\r\n                        const rG = (sG / 2) * Math.sqrt(val / maxVal)\r\n\r\n                        //draw angular sector\r\n                        ctx.beginPath()\r\n                        ctx.moveTo(xc, yc)\r\n                        ctx.arc(xc, yc, rG, cumul - incr, cumul)\r\n                        ctx.lineTo(xc, yc)\r\n                        ctx.fill()\r\n\r\n                        //next angular sector\r\n                        cumul += incr\r\n                    } else if (type_ === 'halftone') {\r\n                        //set category color\r\n                        ctx.fillStyle = color\r\n\r\n                        //get categroy value\r\n                        const val = cell[column]\r\n\r\n                        //compute category radius - in geo\r\n                        /** @type {number} */\r\n                        const rG = sG * 0.333 * Math.sqrt(val / maxVal)\r\n\r\n                        //draw circle\r\n                        ctx.beginPath()\r\n                        ctx.arc(\r\n                            xc + resolution * 0.25 * Math.cos(cumul),\r\n                            yc + resolution * 0.25 * Math.sin(cumul),\r\n                            rG,\r\n                            0,\r\n                            2 * Math.PI\r\n                        )\r\n                        ctx.fill()\r\n\r\n                        //next angular sector\r\n                        cumul += incr\r\n                    } else {\r\n                        throw new Error('Unexpected symbol type:' + type_)\r\n                    }\r\n                }\r\n            } else {\r\n                //compute total\r\n                let total = 0\r\n                for (let column of Object.keys(this.color)) {\r\n                    const v = +cell[column]\r\n                    if (!v) continue\r\n                    total += v\r\n                }\r\n                if (!total || isNaN(total)) continue\r\n\r\n                //draw decomposition symbol\r\n                let cumul = 0\r\n                const d = resolution * (1 - sG / resolution) * 0.5\r\n                const ori = this.stripesOrientation(cell, resolution, z, viewScale)\r\n\r\n                for (let [column, color] of Object.entries(this.color)) {\r\n                    //get share\r\n                    const share = cell[column] / total\r\n                    if (!share || isNaN(share)) continue\r\n\r\n                    //set color\r\n                    ctx.fillStyle = color\r\n\r\n                    //draw symbol part\r\n                    if (type_ === 'flag') {\r\n                        //draw flag stripe\r\n                        if (ori == 0) {\r\n                            //horizontal\r\n                            ctx.fillRect(\r\n                                cell.x + d + offset.dx,\r\n                                cell.y + d + cumul * sG + offset.dy,\r\n                                sG,\r\n                                share * sG\r\n                            )\r\n                        } else {\r\n                            //vertical\r\n                            ctx.fillRect(\r\n                                cell.x + d + cumul * sG + offset.dx,\r\n                                cell.y + d + offset.dy,\r\n                                share * sG,\r\n                                sG\r\n                            )\r\n                        }\r\n                    } else if (type_ === 'piechart') {\r\n                        //draw pie chart angular sector\r\n\r\n                        //compute angles\r\n                        const a1 = cumul * 2 * Math.PI\r\n                        const a2 = (cumul + share) * 2 * Math.PI\r\n\r\n                        //draw\r\n                        ctx.beginPath()\r\n                        ctx.moveTo(xc, yc)\r\n                        ctx.arc(xc, yc, sG * 0.5, a1 + offAng, a2 + offAng)\r\n                        if (this.pieChartInternalRadiusFactor)\r\n                            ctx.arc(\r\n                                xc,\r\n                                yc,\r\n                                sG * 0.5 * this.pieChartInternalRadiusFactor,\r\n                                a1 + offAng,\r\n                                a2 + offAng,\r\n                                true\r\n                            )\r\n                        ctx.closePath()\r\n                        ctx.fill()\r\n                    } else if (type_ === 'ring') {\r\n                        //draw ring\r\n                        ctx.beginPath()\r\n                        ctx.arc(xc, yc, Math.sqrt(1 - cumul) * sG * 0.5, 0, 2 * Math.PI)\r\n                        ctx.fill()\r\n                    } else if (type_ === 'segment') {\r\n                        //draw segment sections\r\n                        const wG = (sG * sG) / resolution\r\n                        if (ori == 0) {\r\n                            //horizontal\r\n                            ctx.fillRect(\r\n                                cell.x + offset.dx,\r\n                                cell.y + (resolution - wG) / 2 + cumul * wG + offset.dy,\r\n                                resolution,\r\n                                share * wG\r\n                            )\r\n                        } else {\r\n                            //vertical\r\n                            ctx.fillRect(\r\n                                cell.x + cumul * resolution + offset.dx,\r\n                                cell.y + (resolution - wG) / 2 + offset.dy,\r\n                                share * resolution,\r\n                                wG\r\n                            )\r\n                        }\r\n                    } else {\r\n                        throw new Error('Unexpected symbol type:' + type_)\r\n                    }\r\n\r\n                    cumul += share\r\n                }\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\nimport { randomNormal } from 'd3-random'\r\nimport { checkWebGLSupport, makeWebGLCanvas } from '../utils/webGLUtils.js'\r\nimport { WebGLSquareColoring } from '../utils/WebGLSquareColoring.js'\r\nimport { color } from 'd3-color'\r\n\r\n/**\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class DotDensityStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the number of dots for a cell value.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.dotNumber = opts.dotNumber || ((cell, resolution) => resolution / 100) //(c,r,z,vs) => {}\r\n\r\n        /** The color of the dots. Same color for all dots within a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => '#FF5733') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the size of the dots, in geo unit. Same size for all cells.\r\n         * @type {function(number, number,object):number} */\r\n        this.dotSize = opts.dotSize || ((resolution, z) => 1.5 * z) //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the sigma of the dots distribution. Same value for all cells.\r\n         * @type {function(number, number,object):number} */\r\n        this.sigma = opts.sigma || ((resolution, z) => resolution / 2) //(c,r,z,vs) => {}\r\n    }\r\n\r\n    /**\r\n     * Draw cells as text.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //get size\r\n        const sGeo = this.dotSize ? this.dotSize(resolution, z, viewScale) : z\r\n\r\n        //make random function\r\n        const sig = this.sigma ? this.sigma(resolution, z, viewScale) : resolution * 0.4\r\n        const rand = randomNormal(0, sig)\r\n\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        if (checkWebGLSupport()) {\r\n            //create canvas and webgl renderer\r\n            const cvWGL = makeWebGLCanvas(geoCanvas.w + '', geoCanvas.h + '')\r\n            if (!cvWGL) {\r\n                console.error('No webGL')\r\n                return\r\n            }\r\n\r\n            //create webGL program\r\n            const prog = new WebGLSquareColoring(cvWGL.gl, sGeo / z)\r\n\r\n            const r2 = resolution / 2\r\n\r\n            for (let cell of cells) {\r\n                //get color\r\n                const col = this.color(cell, resolution, z, viewScale)\r\n                if (!col || col === 'none') continue\r\n\r\n                //number of dots\r\n                const dotNumber = this.dotNumber(cell, resolution, z, viewScale)\r\n\r\n                //get offset\r\n                const offset = this.offset(cell, resolution, z)\r\n\r\n                //cell center\r\n                const cx = cell.x + offset.dx + r2\r\n                const cy = cell.y + offset.dy + r2\r\n\r\n                //convert color\r\n                const cc = color(col)\r\n                if (!cc) return\r\n\r\n                //random points\r\n                for (let i = 0; i <= dotNumber; i++)\r\n                    prog.addPointData2(cx + rand(), cy + rand(), cc.r, cc.g, cc.b, cc.opacity)\r\n            }\r\n\r\n            //draw\r\n            prog.draw(geoCanvas.getWebGLTransform())\r\n\r\n            //draw in canvas geo\r\n            geoCanvas.initCanvasTransform()\r\n            ctx.drawImage(cvWGL.canvas, 0, 0)\r\n        } else {\r\n            for (let cell of cells) {\r\n                //get color\r\n                const col = this.color(cell, resolution, z, viewScale)\r\n                if (!col || col === 'none') continue\r\n                //set color\r\n                ctx.fillStyle = col\r\n\r\n                //number of dots\r\n                const dotNumber = this.dotNumber(cell, resolution, z, viewScale)\r\n\r\n                //get offset\r\n                const offset = this.offset(cell, resolution, z)\r\n\r\n                //draw random dots\r\n                const cx = cell.x + offset.dx + resolution / 2,\r\n                    cy = cell.y + offset.dy + resolution / 2\r\n                for (let i = 0; i <= dotNumber; i++) {\r\n                    ctx.fillRect(cx + rand(), cy + rand(), sGeo, sGeo)\r\n                }\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class ImageStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the image URL of a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.image = opts.image || (() => '') //(c,r,z,vs) => {}\r\n\r\n        /** The image size in ground meters\r\n         *  @type {function(import('../core/Dataset.js').Cell, number, number, object):number}        */\r\n        this.size = opts.size || ((cell, resolution) => resolution)\r\n\r\n        /** Dictionnary of preloaded images. url -> image\r\n         * @private\r\n         * @type {object} */\r\n        this.cache = {}\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     * @override\r\n     */\r\n    async draw(cells, geoCanvas, resolution) {\r\n        //\r\n        const z = geoCanvas.view.z,\r\n            resolutionPix = resolution / z\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //draw in screen coordinates\r\n        geoCanvas.initCanvasTransform()\r\n\r\n        //\r\n        for (let cell of cells) {\r\n            //get cell image url\r\n            const url = this.image(cell, resolution, z, viewScale)\r\n            if (!url) continue\r\n\r\n            //size and position values\r\n            let sizePix = this.size(cell, resolution, z, viewScale) / z\r\n            if (!sizePix) continue\r\n\r\n            //get image from cache\r\n            const image = this.cache[url]\r\n\r\n            //loading, keep waiting\r\n            if (image == 'loading') return\r\n            //no image: load it\r\n            else if (!image) {\r\n                //tag as loading\r\n                this.cache[url] = 'loading'\r\n\r\n                //define image\r\n                const img = new Image()\r\n                img.onload = () => {\r\n                    //store image data in cache and redraw\r\n                    this.cache[url] = img\r\n                    geoCanvas.redraw()\r\n                }\r\n                img.onerror = () => {\r\n                    //case when no image\r\n                    console.warn('Could not retrieve image from', url)\r\n                }\r\n                //set URL to launch the download\r\n                img.src = url\r\n            } else {\r\n                //draw image\r\n                const d = (resolutionPix - sizePix) / 2\r\n                try {\r\n                    geoCanvas.offscreenCtx.drawImage(\r\n                        image,\r\n                        geoCanvas.geoToPixX(cell.x) + d,\r\n                        geoCanvas.geoToPixY(cell.y) + d - resolutionPix,\r\n                        sizePix,\r\n                        sizePix\r\n                    )\r\n                } catch (error) {\r\n                    console.error(error)\r\n                }\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\nimport { SideStyle } from './SideStyle.js'\r\n\r\n/** @typedef {{x:number,y:number,or:\"v\"|\"h\",c1:import('../core/Dataset.js').Cell|undefined,c2:import('../core/Dataset.js').Cell|undefined}} Side */\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class IsoFenceStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * The dictionary (string -> color) which give the color of each category.\r\n         * @type {object} */\r\n        this.color = opts.color\r\n\r\n        /** A function returning the height of a cell in geographical unit.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.height = opts.height || ((cell, resolution, z, viewScale) => resolution * 0.4)\r\n\r\n        /** The perspective angle, in degree, within [-180,180], from [O,x] axis.\r\n         * @type {number} */\r\n        this.angle = opts.angle != undefined ? opts.angle : 50\r\n\r\n        /** A function returning the corner line stroke style.\r\n         * @type {function(import('../core/Dataset.js').Cell,number,number,number):string} */\r\n        this.cornerLineStrokeColor = opts.cornerLineStrokeColor || ((c, r, z, angle) => '#999')\r\n\r\n        /** A function returning the corner line width.\r\n         * @type {function(import('../core/Dataset.js').Cell,number,number,number):number} */\r\n        this.cornerLineWidth = opts.cornerLineWidth || ((c, r, z, angle) => (angle % 90 == 0 ? 0 : 0.8 * z))\r\n\r\n        /**\r\n         * Show vertical cross-sections.\r\n         * @type {boolean} */\r\n        this.sVert = opts.sVert != undefined ? opts.sVert : true\r\n\r\n        /**\r\n         * Show horizontal cross-sections.\r\n         * @type {boolean} */\r\n        this.sHor = opts.sHor != undefined ? opts.sHor : true\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     * @override\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //nb categories - used for radar and agepyramid\r\n        const cats = Object.keys(this.color)\r\n\r\n        //half resolution\r\n        const r2 = resolution / 2\r\n\r\n        //get offset\r\n        // @ts-ignore\r\n        const offset = this.offset(undefined, resolution, z),\r\n            dx = offset.dx,\r\n            dy = offset.dy\r\n\r\n        //make sides\r\n        /**  @type {Array.<Side>} */\r\n        const sides = SideStyle.buildSides(\r\n            cells,\r\n            resolution,\r\n            this.angle % 180 != 90 && this.sVert,\r\n            this.angle % 180 != 0 && this.sHor\r\n        )\r\n\r\n        //\r\n        if (sides.length == 0) return\r\n\r\n        //angle in radians\r\n        const aRad = (this.angle * Math.PI) / 180,\r\n            cos = Math.cos(aRad),\r\n            sin = Math.sin(aRad)\r\n\r\n        //sort sides so that the back ones are drawn first. This depends on the angle.\r\n        //depending on distance to the reference corner point\r\n        const xCorner = Math.abs(this.angle) < 90 ? geoCanvas.extGeo.xMin : geoCanvas.extGeo.xMax\r\n        const yCorner = this.angle < 0 ? geoCanvas.extGeo.yMax : geoCanvas.extGeo.yMin\r\n        sides.sort(\r\n            (s1, s2) =>\r\n                Math.hypot(s2.x - xCorner, s2.y - yCorner) - Math.hypot(s1.x - xCorner, s1.y - yCorner)\r\n        )\r\n\r\n        //prepare function to draw corner line for a cell *c*\r\n        const drawCornerLine = (cell) => {\r\n            if (!cell) return\r\n            //line style\r\n            const lw = this.cornerLineWidth ? this.cornerLineWidth(cell, resolution, z, this.angle) : 0.8 * z\r\n            if (lw == 0) return\r\n            ctx.strokeStyle = this.cornerLineStrokeColor\r\n                ? this.cornerLineStrokeColor(cell, resolution, z, this.angle)\r\n                : '#333'\r\n            ctx.lineWidth = lw\r\n\r\n            //height - in geo\r\n            const hG = this.height(cell, resolution, z, viewScale)\r\n\r\n            //draw line\r\n            ctx.beginPath()\r\n            ctx.moveTo(cell.x + r2 + dx, cell.y + r2 + dy)\r\n            ctx.lineTo(cell.x + r2 + hG * cos + dx, cell.y + r2 + hG * sin + dy)\r\n            ctx.stroke()\r\n        }\r\n\r\n        //draw sides\r\n        ctx.lineCap = 'round'\r\n        for (let side of sides) {\r\n            const c1 = side.c1,\r\n                c2 = side.c2,\r\n                x = side.x,\r\n                y = side.y\r\n\r\n            //heights - in geo\r\n            const hG1 = c1 ? this.height(c1, resolution, z, viewScale) : 0,\r\n                hG2 = c2 ? this.height(c2, resolution, z, viewScale) : 0\r\n\r\n            //compute totals for both cells\r\n            const total1 = computeTotal(c1, cats),\r\n                total2 = computeTotal(c2, cats)\r\n            if (total1 == 0 && total2 == 0) continue\r\n\r\n            let cumul1 = 0,\r\n                cumul2 = 0\r\n            for (let [column, color] of Object.entries(this.color)) {\r\n                //draw stripe of side s and category column\r\n\r\n                //get values for both cells\r\n                let v1 = c1 ? +c1[column] : 0\r\n                let v2 = c2 ? +c2[column] : 0\r\n                if (v1 == 0 && v2 == 0) continue\r\n\r\n                //compute heights\r\n                const h1 = (hG1 * cumul1) / total1 || 0\r\n                const h1n = (hG1 * (cumul1 + v1)) / total1 || 0\r\n                const h2 = (hG2 * cumul2) / total2 || 0\r\n                const h2n = (hG2 * (cumul2 + v2)) / total2 || 0\r\n\r\n                //make path\r\n                ctx.beginPath()\r\n                if (side.or == 'h') {\r\n                    //horizontal side - vertical section\r\n                    //bottom left\r\n                    ctx.moveTo(x + h1 * cos + dx, y - r2 + h1 * sin + dy)\r\n                    //top left\r\n                    ctx.lineTo(x + h2 * cos + dx, y + r2 + h2 * sin + dy)\r\n                    //top right\r\n                    ctx.lineTo(x + h2n * cos + dx, y + r2 + h2n * sin + dy)\r\n                    //bottom right\r\n                    ctx.lineTo(x + h1n * cos + dx, y - r2 + h1n * sin + dy)\r\n                } else {\r\n                    //vertical side - horizontal section\r\n                    //bottom left\r\n                    ctx.moveTo(x - r2 + h1 * cos + dx, y + h1 * sin + dy)\r\n                    //bottom right\r\n                    ctx.lineTo(x + r2 + h2 * cos + dx, y + h2 * sin + dy)\r\n                    //top right\r\n                    ctx.lineTo(x + r2 + h2n * cos + dx, y + h2n * sin + dy)\r\n                    //top left\r\n                    ctx.lineTo(x - r2 + h1n * cos + dx, y + h1n * sin + dy)\r\n                }\r\n                //cg.ctx.closePath()\r\n\r\n                //fill\r\n                ctx.fillStyle = color\r\n                ctx.fill()\r\n\r\n                cumul1 += v1\r\n                cumul2 += v2\r\n\r\n                //TODO draw only one line\r\n                //draw corner line\r\n                //if (side.or == \"h\") {\r\n                drawCornerLine(c1)\r\n                drawCornerLine(c2)\r\n                //if (this.angle > 0 && side.or == \"h\") drawCornerLine(c2)\r\n                //else drawCornerLine(c2)\r\n                //}\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n\r\nconst computeTotal = (cell, categories) => {\r\n    if (!cell) return 0\r\n    let total = 0\r\n    for (let column of categories) {\r\n        const v = cell[column]\r\n        if (!v) continue\r\n        total += +v\r\n    }\r\n    return total || 0\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class JoyPlotStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the height of a cell in geographical unit.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.height = opts.height || ((c, r) => r * Math.random()) //(c,r,z,vs) => {}\r\n\r\n        /**\r\n         * @type {function(number,{min:number, max:number},number,number):string} */\r\n        this.lineColor = opts.lineColor || ((y, ys, r, z) => '#BBB')\r\n        /**\r\n         * @type {function(number,{min:number, max:number},number,number):number} */\r\n        this.lineWidth = opts.lineWidth || ((y, ys, r, z) => z)\r\n        /**\r\n         * @type {function(number,{min:number, max:number},number,number):string} */\r\n        this.fillColor = opts.fillColor || ((y, ys, r, z) => '#c08c5968')\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     * @override\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //index cells by y and x\r\n        /**  @type {object} */\r\n        const ind = {}\r\n        for (const cell of cells) {\r\n            let row = ind[cell.y]\r\n            if (!row) {\r\n                row = {}\r\n                ind[cell.y] = row\r\n            }\r\n            row[cell.x] = this.height(cell, resolution, z, viewScale)\r\n        }\r\n\r\n        //compute extent\r\n        const e = geoCanvas.extGeo\r\n        if (!e) return\r\n        const xMin = Math.floor(e.xMin / resolution) * resolution\r\n        const xMax = Math.floor(e.xMax / resolution) * resolution\r\n        const yMin = Math.floor(e.yMin / resolution) * resolution\r\n        const yMax = Math.floor(e.yMax / resolution) * resolution\r\n\r\n        /**  @type {{min:number, max:number}} */\r\n        const ys = { min: yMin, max: yMax }\r\n\r\n        //draw lines, row by row, stating from the top\r\n        ctx.lineJoin = 'round'\r\n        for (let y = yMax; y >= yMin; y -= resolution) {\r\n            //get row\r\n            const row = ind[y]\r\n\r\n            //no row\r\n            if (!row) continue\r\n\r\n            //place first point\r\n            ctx.beginPath()\r\n            ctx.moveTo(xMin - resolution / 2, y)\r\n\r\n            //store the previous height\r\n            /** @type {number|undefined} */\r\n            let hG_\r\n\r\n            //go through the line cells\r\n            for (let x = xMin; x <= xMax; x += resolution) {\r\n                //get column value\r\n                /** @type {number} */\r\n                let hG = row[x]\r\n                if (!hG) hG = 0\r\n\r\n                if (hG || hG_) {\r\n                    //draw line only when at least one of both values is non-null\r\n                    //TODO test bezierCurveTo\r\n                    ctx.lineTo(x + resolution / 2, y + hG)\r\n                } else {\r\n                    //else move the point\r\n                    ctx.moveTo(x + resolution / 2, y)\r\n                }\r\n                //store the previous value\r\n                hG_ = hG\r\n            }\r\n\r\n            //last point\r\n            if (hG_) ctx.lineTo(xMax + resolution / 2, y)\r\n\r\n            //draw fill\r\n            const fc = this.fillColor(y, ys, resolution, z)\r\n            if (fc && fc != 'none') {\r\n                ctx.fillStyle = fc\r\n                ctx.fill()\r\n            }\r\n\r\n            //draw line\r\n            const lc = this.lineColor(y, ys, resolution, z)\r\n            const lw = this.lineWidth(y, ys, resolution, z)\r\n            if (lc && lc != 'none' && lw > 0) {\r\n                ctx.strokeStyle = lc\r\n                ctx.lineWidth = lw\r\n                ctx.stroke()\r\n            }\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { TanakaStyle } from './TanakaStyle.js'\r\nimport { StrokeStyle } from './StrokeStyle.js'\r\nimport { SquareColorCategoryWebGLStyle } from './SquareColorCategoryWebGLStyle.js'\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class LegoStyle {\r\n    static get(value, breaks, colors, opts = {}) {\r\n        opts = opts || {}\r\n\r\n        //the colors\r\n        //http://www.jennyscrayoncollection.com/2021/06/all-current-lego-colors.html\r\n        //https://leonawicz.github.io/legocolors/reference/figures/README-plot-1.png\r\n        /*opts.colors = opts.colors || [\r\n            '#00852b', //darker green\r\n            '#afd246', //light green\r\n            '#fac80a', //dark yellow\r\n            '#bb805a', //brown\r\n            '#d67923', //mostard\r\n            '#cb4e29', //redish\r\n            '#b40000', //red\r\n            '#720012', //dark red\r\n            //\"purple\",\r\n            //\"#eee\" //whithe\r\n        ]*/\r\n\r\n        opts.colDark = opts.colDark || '#333'\r\n        opts.colBright = opts.colBright || '#aaa'\r\n        opts.widthFactor = opts.widthFactor || 0.12\r\n\r\n        //reuse tanaka as basis\r\n        const ts = TanakaStyle.get(value, breaks, colors, opts)\r\n        //style to show limits between pieces\r\n        const sst = new StrokeStyle({\r\n            strokeColor: () => '#666',\r\n            strokeWidth: (c, r, z) => 0.2 * z,\r\n            filter: opts.filter,\r\n        })\r\n\r\n        return [\r\n            ts[0],\r\n            sst,\r\n            ts[1],\r\n            new LegoTopStyle({ colDark: opts.colDark, colBright: opts.colBright, filter: opts.filter }),\r\n        ]\r\n    }\r\n\r\n    /**\r\n     * @param {function(import('../core/Dataset.js').Cell):string} code\r\n     * @param {object} color\r\n     * @param {object} opts\r\n     * @returns {Array.<Style>}\r\n     */\r\n    static getCategory(code, color, opts) {\r\n        opts = opts || {}\r\n\r\n        opts.colDark = opts.colDark || '#333'\r\n        opts.colBright = opts.colBright || '#aaa'\r\n\r\n        //\r\n        const s = new SquareColorCategoryWebGLStyle({ code: code, color: color })\r\n        //style to show limits between pieces\r\n        const sst = new StrokeStyle({ strokeColor: () => '#666', strokeWidth: (c, r, z) => 0.2 * z })\r\n\r\n        return [s, sst, new LegoTopStyle({ colDark: opts.colDark, colBright: opts.colBright })]\r\n    }\r\n}\r\n\r\n/**\r\n * A style to draw top circle of lego bricks.\r\n */\r\nclass LegoTopStyle extends Style {\r\n    /** @param {object|undefined} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n        this.colDark = opts.colDark || '#333'\r\n        this.colBright = opts.colBright || '#aaa'\r\n    }\r\n\r\n    draw(cells, geoCanvas, r) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        ctx.lineWidth = 0.6 * geoCanvas.view.z\r\n\r\n        //dark part\r\n        ctx.strokeStyle = this.colDark\r\n        for (let c of cells) {\r\n            ctx.beginPath()\r\n            ctx.arc(c.x + r * 0.5, c.y + r * 0.5, r * 0.55 * 0.5, Math.PI / 4, -Math.PI * (3 / 4), true)\r\n            ctx.stroke()\r\n        }\r\n\r\n        //bright part\r\n        ctx.strokeStyle = this.colBright\r\n        for (let c of cells) {\r\n            ctx.beginPath()\r\n            ctx.arc(c.x + r * 0.5, c.y + r * 0.5, r * 0.55 * 0.5, Math.PI / 4, -Math.PI * (3 / 4), false)\r\n            ctx.stroke()\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class MosaicStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the color of the cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => '#EA6BAC') //(c,r,z,vs) => {}\r\n\r\n        /** The mosaic factor, within [0,0.5]. Set to 0 for no mosaic effect. Set to 0.5 for strong mosaic effect.\r\n         * @type {number} */\r\n        this.mosaicFactor = opts.mosaicFactor || 0.15\r\n\r\n        /** The mosaic shadow factor, within [0,0.5]. Set to 0 for no mosaic shadow. Set to 0.5 for strong mosaic shadow.\r\n         * @type {number} */\r\n        this.shadowFactor = opts.shadowFactor || 0.2\r\n\r\n        /** The mosaic shadow color.\r\n         * @type {string} */\r\n        this.shadowColor = opts.shadowColor || '#555'\r\n    }\r\n\r\n    /**\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //set stroke style, for shadow\r\n        ctx.strokeStyle = this.shadowColor\r\n        ctx.lineWidth = this.shadowFactor * resolution\r\n        ctx.lineJoin = 'round'\r\n        ctx.lineCap = 'butt'\r\n\r\n        //function to compute position mosaic effect\r\n        const d = resolution * this.mosaicFactor\r\n        const mosaic = () => {\r\n            return { x: Math.random() * d, y: Math.random() * d }\r\n        }\r\n\r\n        for (let cell of cells) {\r\n            //set fill color\r\n            const col = this.color ? this.color(cell, resolution, z, viewScale) : undefined\r\n            if (!col || col === 'none') continue\r\n            ctx.fillStyle = col\r\n\r\n            //get offset\r\n            const offset = this.offset(cell, resolution, z)\r\n\r\n            //compute position mosaic effect\r\n            const ll = mosaic(),\r\n                ul = mosaic(),\r\n                lr = mosaic(),\r\n                ur = mosaic()\r\n\r\n            //stroke\r\n            if (this.shadowFactor > 0) {\r\n                ctx.beginPath()\r\n                ctx.moveTo(cell.x + offset.dx + ll.x, cell.y + offset.dy + ll.y)\r\n                ctx.lineTo(cell.x + offset.dx + resolution - lr.x, cell.y + offset.dy + lr.y)\r\n                ctx.lineTo(cell.x + offset.dx + resolution - ur.x, cell.y + offset.dy + resolution - ur.y)\r\n                ctx.stroke()\r\n            }\r\n\r\n            //fill\r\n\r\n            ctx.beginPath()\r\n            ctx.moveTo(cell.x + offset.dx + ll.x, cell.y + offset.dy + ll.y)\r\n            ctx.lineTo(cell.x + offset.dx + resolution - lr.x, cell.y + offset.dy + lr.y)\r\n            ctx.lineTo(cell.x + offset.dx + resolution - ur.x, cell.y + offset.dy + resolution - ur.y)\r\n            ctx.lineTo(cell.x + offset.dx + ul.x, cell.y + offset.dy + resolution - ul.y)\r\n            ctx.fill()\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class NinjaStarStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the color of the cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => '#EA6BAC') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the size of a cell, within [0,1]:\r\n         *  - 0, nothing shown\r\n         *  - 1, entire square\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.size = opts.size || ((cell, resolution) => resolution) //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the shape.\r\n         * @type {function(import(\"../core/Dataset\").Cell):string} */\r\n        this.shape = opts.shape || (() => 'o')\r\n    }\r\n\r\n    /**\r\n     *\r\n     * @param {Array.<import('../core/Dataset.js').Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        const r2 = resolution * 0.5\r\n        for (let cell of cells) {\r\n            //color\r\n            const col = this.color ? this.color(cell, resolution, z, viewScale) : undefined\r\n            if (!col || col === 'none') continue\r\n            ctx.fillStyle = col\r\n\r\n            //size - in geo unit\r\n            let k = this.size(cell, resolution, z, viewScale)\r\n            k = k < 0 ? 0 : k > 1 ? 1 : k\r\n            const sG2 = k * r2\r\n\r\n            //shape\r\n            const shape = this.shape ? this.shape(cell) : 'o'\r\n            if (shape === 'none') continue\r\n\r\n            //get offset\r\n            //TODO use\r\n            //const offset = this.offset(cell, r, z)\r\n\r\n            //center position\r\n            const cx = cell.x + r2\r\n            const cy = cell.y + r2\r\n\r\n            if (shape === 'p') {\r\n                ctx.beginPath()\r\n                ctx.moveTo(cx, cy + r2)\r\n                ctx.lineTo(cx + sG2, cy + sG2)\r\n                ctx.lineTo(cx + r2, cy)\r\n                ctx.lineTo(cx + sG2, cy - sG2)\r\n                ctx.lineTo(cx, cy - r2)\r\n                ctx.lineTo(cx - sG2, cy - sG2)\r\n                ctx.lineTo(cx - r2, cy)\r\n                ctx.lineTo(cx - sG2, cy + sG2)\r\n                ctx.fill()\r\n            } else if (shape === 'o') {\r\n                ctx.beginPath()\r\n                ctx.moveTo(cx, cy + sG2)\r\n                ctx.lineTo(cx + r2, cy + r2)\r\n                ctx.lineTo(cx + sG2, cy)\r\n                ctx.lineTo(cx + r2, cy - r2)\r\n                ctx.lineTo(cx, cy - sG2)\r\n                ctx.lineTo(cx - r2, cy - r2)\r\n                ctx.lineTo(cx - sG2, cy)\r\n                ctx.lineTo(cx - r2, cy + r2)\r\n                ctx.fill()\r\n            } else {\r\n                throw new Error('Unexpected shape:' + shape)\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class PillarStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the height of the line representing a cell, in geo unit\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.height = opts.height\r\n\r\n        /** A function returning the color of the line representing a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => '#c08c59') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the width of the line representing a cell, in geo unit\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.width = opts.width || ((cell, resolution) => 0.5 * resolution)\r\n\r\n        /** A function returning the width of the line representing a cell, in geo unit\r\n         * @type {function(number, number,object):boolean} */\r\n        this.simple = opts.simple || (() => false)\r\n\r\n        /** @type {number} */\r\n        this.viewHeightFactor = opts.viewHeightFactor || 1.5\r\n        //0,0 is the center\r\n        /** @type {number} */\r\n        this.viewSX = opts.viewSX == undefined ? 0 : opts.viewSX\r\n        /** @type {number} */\r\n        this.viewSY = opts.viewSY == undefined ? -0.5 : opts.viewSY\r\n\r\n        //TODO replace with sun location ?\r\n        /** @type {number} */\r\n        this.shadowDirection =\r\n            opts.shadowDirection == undefined ? (-40.3 * Math.PI) / 180.0 : opts.shadowDirection\r\n        /** @type {number} */\r\n        this.shadowFactor = opts.shadowFactor || 0.3\r\n        /** @type {string} */\r\n        this.shadowColor = opts.shadowColor || '#00000033'\r\n\r\n        /** @type {string} */\r\n        this.outlineCol = opts.outlineCol || '#FFFFFF'\r\n        /** @type {number} */\r\n        this.outlineWidthPix = opts.outlineWidthPix == undefined ? 0.5 : opts.outlineWidthPix\r\n    }\r\n\r\n    /**\r\n     * Draw cells as segments.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //get view center geo position\r\n        const cvx = geoCanvas.view.x + this.viewSX * geoCanvas.w * z\r\n        const cvy = geoCanvas.view.y + this.viewSY * geoCanvas.h * z\r\n        //get view height\r\n        const H = this.viewHeightFactor * (geoCanvas.w + geoCanvas.h) * 0.5 * z\r\n\r\n        //sort cells by y and x\r\n        //const distToViewCenter = (c) => { const dx = cvx - c.x, dy = cvy - c.y; return Math.sqrt(dx * dx + dy * dy) }\r\n        cells.sort((c1, c2) => 100000000 * (c2.y - c1.y) + c1.x - c2.x)\r\n\r\n        //get simple information\r\n        const simple = this.simple(resolution, z, viewScale)\r\n\r\n        ctx.lineCap = simple ? 'butt' : 'round'\r\n\r\n        //draw shadows\r\n        ctx.strokeStyle = this.shadowColor\r\n        ctx.fillStyle = this.shadowColor\r\n        for (let cell of cells) {\r\n            //width\r\n            /** @type {number|undefined} */\r\n            const wG = this.width ? this.width(cell, resolution, z, viewScale) : undefined\r\n            if (!wG || wG < 0) continue\r\n\r\n            //height\r\n            /** @type {number|undefined} */\r\n            const hG = this.height ? this.height(cell, resolution, z, viewScale) : undefined\r\n            if (!hG || hG < 0) continue\r\n\r\n            //get offset\r\n            //TODO use that\r\n            //const offset = this.offset(c, resolution, z)\r\n\r\n            //set width\r\n            ctx.lineWidth = wG\r\n\r\n            //compute cell center postition\r\n            const cx = cell.x + resolution / 2\r\n            const cy = cell.y + resolution / 2\r\n            const ls = hG * this.shadowFactor\r\n\r\n            //draw segment\r\n            ctx.beginPath()\r\n            ctx.moveTo(cx, cy)\r\n            ctx.lineTo(cx + ls * Math.cos(this.shadowDirection), cy + ls * Math.sin(this.shadowDirection))\r\n            ctx.stroke()\r\n\r\n            /*\r\n            if (this.simple) {\r\n                //draw base circle\r\n                cg.ctx.beginPath();\r\n                cg.ctx.arc(\r\n                    cx, cy,\r\n                    wG * 0.5,\r\n                    0, 2 * Math.PI, false);\r\n                //cg.ctx.stroke();\r\n                cg.ctx.fill();\r\n            }*/\r\n        }\r\n\r\n        //draw pillars\r\n        for (let cell of cells) {\r\n            //color\r\n            /** @type {string|undefined} */\r\n            const col = this.color ? this.color(cell, resolution, z, viewScale) : undefined\r\n            if (!col) continue\r\n\r\n            //width\r\n            /** @type {number|undefined} */\r\n            const wG = this.width ? this.width(cell, resolution, z, viewScale) : undefined\r\n            if (!wG || wG < 0) continue\r\n\r\n            //height\r\n            /** @type {number|undefined} */\r\n            const hG = this.height ? this.height(cell, resolution, z, viewScale) : undefined\r\n            if (!hG || hG < 0) continue\r\n\r\n            //get offset\r\n            //TODO use that\r\n            //const offset = this.offset(c, resolution, z)\r\n\r\n            //compute cell center postition\r\n            const cx = cell.x + resolution / 2\r\n            const cy = cell.y + resolution / 2\r\n\r\n            //compute angle\r\n            const dx = cx - cvx,\r\n                dy = cy - cvy\r\n            const a = Math.atan2(dy, dx)\r\n            const D = Math.sqrt(dx * dx + dy * dy)\r\n            const d = (D * hG) / (H - hG)\r\n\r\n            if (simple) {\r\n                //draw segment\r\n                ctx.strokeStyle = col\r\n                ctx.lineWidth = wG\r\n                ctx.beginPath()\r\n                ctx.moveTo(cx, cy)\r\n                ctx.lineTo(cx + d * Math.cos(a), cy + d * Math.sin(a))\r\n                ctx.stroke()\r\n            } else {\r\n                //draw background segment\r\n                ctx.strokeStyle = this.outlineCol\r\n                ctx.lineWidth = wG + 2 * this.outlineWidthPix * z\r\n                ctx.beginPath()\r\n                ctx.moveTo(cx, cy)\r\n                ctx.lineTo(cx + d * Math.cos(a), cy + d * Math.sin(a))\r\n                ctx.stroke()\r\n\r\n                //draw segment\r\n                ctx.strokeStyle = col\r\n                ctx.lineWidth = wG\r\n                ctx.beginPath()\r\n                ctx.moveTo(cx, cy)\r\n                ctx.lineTo(cx + d * Math.cos(a), cy + d * Math.sin(a))\r\n                ctx.stroke()\r\n\r\n                //draw top circle\r\n                ctx.strokeStyle = this.outlineCol\r\n                //cg.ctx.fillStyle = \"#c08c59\"\r\n                ctx.lineWidth = this.outlineWidthPix * z\r\n                ctx.beginPath()\r\n                ctx.arc(cx + d * Math.cos(a), cy + d * Math.sin(a), wG * 0.5, 0, 2 * Math.PI, false)\r\n                ctx.stroke()\r\n                //cg.ctx.fill();\r\n            }\r\n        }\r\n\r\n        //in case...\r\n        ctx.lineCap = 'butt'\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * A style where each cell is represented by a segment whose length, width, color and orientation can vary according to statistical values.\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class SegmentStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the color of the cell segment.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => '#EA6BAC') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the width of the segment representing a cell, in geo unit\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.width = opts.width || ((cell, resolution) => resolution * 0.1) //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the length of the segment representing a cell, in geo unit\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.length = opts.length || ((cell, resolution) => resolution * 0.9) //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the orientation (in degrees) of the segment representing a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.orientation = opts.orientation || (() => 180 * Math.random()) //(c,r,z,vs) => {}\r\n    }\r\n\r\n    /**\r\n     * Draw cells as segments.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //\r\n        ctx.lineCap = 'butt'\r\n\r\n        //conversion factor degree -> radian\r\n        const f = Math.PI / 180\r\n\r\n        for (let cell of cells) {\r\n            //color\r\n            /** @type {string|undefined} */\r\n            const col = this.color ? this.color(cell, resolution, z, viewScale) : undefined\r\n            if (!col) continue\r\n\r\n            //width\r\n            /** @type {number|undefined} */\r\n            const wG = this.width ? this.width(cell, resolution, z, viewScale) : undefined\r\n            if (!wG || wG < 0) continue\r\n\r\n            //length\r\n            /** @type {number|undefined} */\r\n            const lG = this.length ? this.length(cell, resolution, z, viewScale) : undefined\r\n            if (!lG || lG < 0) continue\r\n\r\n            //orientation (in radian)\r\n            /** @type {number} */\r\n            const or = this.orientation(cell, resolution, z, viewScale) * f\r\n            if (or === undefined || isNaN(or)) continue\r\n\r\n            //get offset\r\n            const offset = this.offset(cell, resolution, z)\r\n\r\n            //set color and width\r\n            ctx.strokeStyle = col\r\n            ctx.lineWidth = wG\r\n\r\n            //compute segment center postition\r\n            const cx = cell.x + resolution / 2 + offset.dx\r\n            const cy = cell.y + resolution / 2 + offset.dy\r\n\r\n            //compute segment direction\r\n            const dx = 0.5 * Math.cos(or) * lG\r\n            const dy = 0.5 * Math.sin(or) * lG\r\n\r\n            //draw segment\r\n            ctx.beginPath()\r\n            ctx.moveTo(cx - dx, cy - dy)\r\n            ctx.lineTo(cx + dx, cy + dy)\r\n            ctx.stroke()\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ viewScale: viewScale, resolution: resolution, z: z, cells: cells })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * A very generic style that shows grid cells with specific color, size and shape.\r\n * It can be used to show variables as cell colors, cell size, cell shape, or any combination of the three visual variables.\r\n *\r\n * @module style\r\n * @author Joseph Davies, Julien Gaffuri\r\n */\r\nexport class ShapeColorSizeStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n        /** @type {(function(import('../core/Dataset.js').Cell, number, number, object):string) | string} */\r\n        this.color = opts.color || '#EA6BAC'\r\n\r\n        /** @type {(function(import('../core/Dataset.js').Cell, number, number, object):number) | number} */\r\n        this.size = opts.size || ((cell, resolution) => resolution)\r\n\r\n        /** @type {(function(import(\"../core/Dataset.js\").Cell,number, number,object):import(\"../core/Style.js\").Shape) | string} */\r\n        this.shape = opts.shape || 'square'\r\n    }\r\n\r\n    /**\r\n     * Draw cells as squares, with various colors and sizes.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells - The grid cells to draw.\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas - The canvas to draw on.\r\n     * @param {number} resolution - Resolution of the grid.\r\n     * @override\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //zoom\r\n        const z = geoCanvas.view.z\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        const r2 = resolution * 0.5\r\n\r\n        // Precompute if color, size, and shape are functions, for efficiency\r\n        const isColorFunction = typeof this.color === 'function'\r\n        const isSizeFunction = typeof this.size === 'function'\r\n        const isShapeFunction = typeof this.shape === 'function'\r\n\r\n        const defaultColor = this.color || 'black'\r\n        const defaultSize = this.size || resolution\r\n        const defaultShape = this.shape || 'square'\r\n\r\n        // Optimized\r\n        const colorFunction = isColorFunction ? this.color : null\r\n        const sizeFunction = isSizeFunction ? this.size : null\r\n        const shapeFunction = isShapeFunction ? this.shape : null\r\n\r\n        for (let c of cells) {\r\n            // Determine color\r\n            //@ts-ignore\r\n            const col = colorFunction ? colorFunction(c, resolution, z, viewScale) : defaultColor\r\n            if (!col || col === 'none') continue\r\n\r\n            // Determine size\r\n            //@ts-ignore\r\n            const size = sizeFunction ? sizeFunction(c, resolution, z, viewScale) : defaultSize\r\n            if (!size) continue\r\n\r\n            // Determine shape\r\n            //@ts-ignore\r\n            const shape = shapeFunction ? shapeFunction(c, resolution, z, viewScale) : defaultShape\r\n            if (shape === 'none') continue\r\n\r\n            //get offset\r\n            const offset = this.offset(c, resolution, z)\r\n\r\n            //get context\r\n            const ctx = geoCanvas.offscreenCtx\r\n            ctx.fillStyle = col\r\n            if (shape === 'square') {\r\n                //draw square\r\n                const d = resolution * (1 - size / resolution) * 0.5\r\n                ctx.fillRect(c.x + d + offset.dx, c.y + d + offset.dy, size, size)\r\n            } else if (shape === 'circle') {\r\n                //draw circle\r\n                ctx.beginPath()\r\n                ctx.arc(c.x + r2 + offset.dx, c.y + r2 + offset.dy, size * 0.5, 0, 2 * Math.PI, false)\r\n                ctx.fill()\r\n            } else if (shape === 'donut') {\r\n                //draw donut\r\n                const xc = c.x + r2 + offset.dx,\r\n                    yc = c.y + r2 + offset.dy\r\n                ctx.beginPath()\r\n                ctx.moveTo(xc, yc)\r\n                ctx.arc(xc, yc, r2, 0, 2 * Math.PI)\r\n                ctx.arc(xc, yc, (1 - size / resolution) * r2, 0, 2 * Math.PI, true)\r\n                ctx.closePath()\r\n                ctx.fill()\r\n            } else if (shape === 'diamond') {\r\n                const s2 = size * 0.5\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x + r2 - s2, c.y + r2)\r\n                ctx.lineTo(c.x + r2, c.y + r2 + s2)\r\n                ctx.lineTo(c.x + r2 + s2, c.y + r2)\r\n                ctx.lineTo(c.x + r2, c.y + r2 - s2)\r\n                ctx.fill()\r\n            } else if (shape === 'triangle_up') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x - dr2, c.y - dr2)\r\n                ctx.lineTo(c.x + r2, c.y + resolution + dr2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y - dr2)\r\n                ctx.fill()\r\n            } else if (shape === 'triangle_down') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x - dr2, c.y + resolution + dr2)\r\n                ctx.lineTo(c.x + r2, c.y - dr2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y + resolution + dr2)\r\n                ctx.fill()\r\n            } else if (shape === 'triangle_left') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x + resolution + dr2, c.y + resolution + dr2)\r\n                ctx.lineTo(c.x - dr2, c.y + r2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y - dr2)\r\n                ctx.fill()\r\n            } else if (shape === 'triangle_right') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x - dr2, c.y - dr2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y + r2)\r\n                ctx.lineTo(c.x - dr2, c.y + resolution + dr2)\r\n                ctx.fill()\r\n            } else {\r\n                throw new Error('Unexpected shape:' + shape)\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ viewScale: viewScale, resolution: resolution, z: z, cells: cells })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { SideStyle } from './SideStyle.js'\r\n\r\n/**\r\n * A style to show the sides of grid cells based on their different categories.\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class SideCategoryStyle extends SideStyle {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the category code of a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number):string} */\r\n        this.code = opts.code\r\n\r\n        /**\r\n         * The dictionary (string -> color) which give the color of each category.\r\n         * @type {object} */\r\n        this.color = opts.color\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //build sides\r\n\r\n        /**  @type {Array.<import('./SideStyle.js').Side>} */\r\n        const sides = SideStyle.buildSides(cells, resolution)\r\n        if (sides.length == 0) return\r\n\r\n        //get side view scale\r\n        const viewScale = this.viewScale ? this.viewScale(sides, resolution, z) : undefined\r\n\r\n        //draw sides\r\n\r\n        ctx.lineCap = 'butt'\r\n        const r2 = resolution * 0.5\r\n        for (let side of sides) {\r\n            //get category codes for both cells\r\n            const code1 = side.c1 ? this.code(side.c1, resolution, z) : undefined\r\n            const code2 = side.c2 ? this.code(side.c2, resolution, z) : undefined\r\n            if (code1 == code2) continue\r\n\r\n            //width\r\n            /** @type {number|undefined} */\r\n            const wG = this.width ? this.width(side, resolution, z, viewScale) : undefined\r\n            if (!wG || wG <= 0) continue\r\n            const w2 = wG * 0.5\r\n\r\n            //set width\r\n            ctx.lineWidth = wG\r\n\r\n            //draw segment with correct orientation\r\n            if (side.or === 'h') {\r\n                //top line\r\n                if (code2) {\r\n                    ctx.beginPath()\r\n                    ctx.strokeStyle = this.color[code2]\r\n                    ctx.moveTo(side.x - r2, side.y + w2)\r\n                    ctx.lineTo(side.x + r2, side.y + w2)\r\n                    ctx.stroke()\r\n                }\r\n\r\n                //bottom line\r\n                if (code1) {\r\n                    ctx.beginPath()\r\n                    ctx.strokeStyle = this.color[code1]\r\n                    ctx.moveTo(side.x - r2, side.y - w2)\r\n                    ctx.lineTo(side.x + r2, side.y - w2)\r\n                    ctx.stroke()\r\n                }\r\n            } else {\r\n                //right line\r\n                if (code2) {\r\n                    ctx.beginPath()\r\n                    ctx.strokeStyle = this.color[code2]\r\n                    ctx.moveTo(side.x + w2, side.y - r2)\r\n                    ctx.lineTo(side.x + w2, side.y + r2)\r\n                    ctx.stroke()\r\n                }\r\n\r\n                //left line\r\n                if (code1) {\r\n                    ctx.beginPath()\r\n                    ctx.strokeStyle = this.color[code1]\r\n                    ctx.moveTo(side.x - w2, side.y - r2)\r\n                    ctx.lineTo(side.x - w2, side.y + r2)\r\n                    ctx.stroke()\r\n                }\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/** @typedef {{ x:number, y:number, or:\"v\"|\"h\", c1:(import('../core/Dataset').Cell)|undefined, c2:(import('../core/Dataset').Cell)|undefined }} Side */\r\n\r\n/**\r\n * @typedef {function(Array.<Side>,number, number):*} SideViewScale */\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class SideStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts = {}) {\r\n        super(opts)\r\n\r\n        /** A function returning the color of a cell side.\r\n         * @type {function(Side, number, number, object):string} */\r\n        this.color = opts.color || ((side, resolution, z, sideViewScale) => '#EA6BAC')\r\n\r\n        /** A function returning the width of a cell side, in geo unit\r\n         * @type {function(Side, number, number, object):number} */\r\n        this.width = opts.width || ((side, resolution, z, sideViewScale) => resolution / 5)\r\n\r\n        /** A function returning the length of a cell side, in geo unit\r\n         * @type {function(Side, number, number, object):number} */\r\n        this.length = opts.length || ((side, resolution, z, sideViewScale) => resolution)\r\n\r\n        /** Set to A or true so that the side is drawn as a diamond */\r\n        this.diamond = opts.diamond\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {number} resolution\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //build sides\r\n\r\n        /**  @type {Array.<Side>} */\r\n        const sides = SideStyle.buildSides(cells, resolution)\r\n        if (sides.length == 0) return\r\n\r\n        //get side view scale\r\n        const viewScale = this.viewScale ? this.viewScale(sides, resolution, z) : undefined\r\n\r\n        //draw sides\r\n\r\n        ctx.lineCap = 'butt'\r\n        const r2 = resolution * 0.5\r\n        for (let side of sides) {\r\n            //color\r\n            /** @type {string|undefined} */\r\n            const col = this.color ? this.color(side, resolution, z, viewScale) : undefined\r\n            if (!col || col == 'none') continue\r\n\r\n            if (this.diamond) {\r\n                //set color\r\n                ctx.fillStyle = col\r\n\r\n                //draw diamond\r\n                const x = side.x,\r\n                    y = side.y\r\n                ctx.beginPath()\r\n                ctx.moveTo(x - r2, y)\r\n                ctx.lineTo(x, y + r2)\r\n                ctx.lineTo(x + r2, y)\r\n                ctx.lineTo(x, y - r2)\r\n                ctx.closePath()\r\n                ctx.fill()\r\n            } else {\r\n                //width\r\n                /** @type {number|undefined} */\r\n                const wG = this.width ? this.width(side, resolution, z, viewScale) : undefined\r\n                if (!wG || wG <= 0) continue\r\n\r\n                //length\r\n                /** @type {number|undefined} */\r\n                const lG = this.length ? this.length(side, resolution, z, viewScale) : undefined\r\n                if (!lG || lG <= 0) continue\r\n                const lG2 = lG * 0.5\r\n\r\n                //set width\r\n                ctx.lineWidth = wG\r\n                //set color\r\n                ctx.strokeStyle = col\r\n\r\n                //draw segment with correct orientation\r\n                const x = side.x,\r\n                    y = side.y\r\n                ctx.beginPath()\r\n                if (side.or === 'v') {\r\n                    ctx.moveTo(x, y - lG2)\r\n                    ctx.lineTo(x, y + lG2)\r\n                } else {\r\n                    ctx.moveTo(x - lG2, y)\r\n                    ctx.lineTo(x + lG2, y)\r\n                }\r\n                ctx.stroke()\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n\r\n    /**\r\n     *\r\n     * @param {Array.<import('../core/Dataset').Cell>} cells The cells to use to build the sides.\r\n     * @param {number} resolution The cells resolution\r\n     * @param {boolean} withHorizontal Set to true to build horizontal sides, false otherwise.\r\n     * @param {boolean} withVertical Set to true to build vertical sides, false otherwise.\r\n     * @param {boolean} center Set to true so that the side coordinate are those of its center point rather than its left/bottom point (the side x,y coordinates are those of the left point for horizontal sides, and of the bottom point for vertical sides)\r\n     * @returns { Array.<Side> }\r\n     */\r\n    static buildSides(cells, resolution, withHorizontal = true, withVertical = true, center = true) {\r\n        /** @type { Array.<Side> } */\r\n        const sides = []\r\n\r\n        const r2 = center ? resolution / 2 : 0\r\n\r\n        //make horizontal sides\r\n        //sort cells by x and y\r\n        cells.sort((c1, c2) => (c2.x == c1.x ? c1.y - c2.y : c1.x - c2.x))\r\n        let c1 = cells[0]\r\n        for (let i = 1; i < cells.length; i++) {\r\n            let c2 = cells[i]\r\n\r\n            if (c1.y + resolution == c2.y && c1.x == c2.x)\r\n                //cells in same column and touch along horizontal side\r\n                //make shared side\r\n                sides.push({\r\n                    or: 'h',\r\n                    x: c1.x + r2,\r\n                    y: c2.y,\r\n                    c1: c1,\r\n                    c2: c2,\r\n                })\r\n            else {\r\n                //cells do not touch along horizontal side\r\n                //make two sides: top one for c1, bottom for c2\r\n                sides.push({\r\n                    or: 'h',\r\n                    x: c1.x + r2,\r\n                    y: c1.y + resolution,\r\n                    c1: c1,\r\n                    c2: undefined,\r\n                })\r\n                sides.push({\r\n                    or: 'h',\r\n                    x: c2.x + r2,\r\n                    y: c2.y,\r\n                    c1: undefined,\r\n                    c2: c2,\r\n                })\r\n            }\r\n\r\n            c1 = c2\r\n        }\r\n\r\n        //make vertical sides\r\n        //sort cells by y and x\r\n        cells.sort((c1, c2) => (c2.y == c1.y ? c1.x - c2.x : c1.y - c2.y))\r\n        c1 = cells[0]\r\n        for (let i = 1; i < cells.length; i++) {\r\n            let c2 = cells[i]\r\n\r\n            if (c1.x + resolution == c2.x && c1.y == c2.y)\r\n                //cells in same row and touch along vertical side\r\n                //make shared side\r\n                sides.push({\r\n                    or: 'v',\r\n                    x: c1.x + resolution,\r\n                    y: c1.y + r2,\r\n                    c1: c1,\r\n                    c2: c2,\r\n                })\r\n            else {\r\n                //cells do not touch along vertical side\r\n                //make two sides: right one for c1, left for c2\r\n                sides.push({\r\n                    or: 'v',\r\n                    x: c1.x + resolution,\r\n                    y: c1.y + r2,\r\n                    c1: c1,\r\n                    c2: undefined,\r\n                })\r\n                sides.push({\r\n                    or: 'v',\r\n                    x: c2.x,\r\n                    y: c2.y + r2,\r\n                    c1: undefined,\r\n                    c2: c2,\r\n                })\r\n            }\r\n\r\n            c1 = c2\r\n        }\r\n        return sides\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\nimport { makeWebGLCanvas } from '../utils/webGLUtils.js'\r\nimport { WebGLSquareColoringCatAdvanced } from '../utils/WebGLSquareColoringCatAdvanced.js'\r\n\r\n/**\r\n * Style based on webGL\r\n * To show cells as colored squares, from categories.\r\n * All cells are drawn as squares, with the same size\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class SquareColorCategoryWebGLStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * A function returning the category code of the cell, for coloring.\r\n         * @type {function(import('../core/Dataset.js').Cell):string} */\r\n        this.code = opts.code\r\n\r\n        /**\r\n         * The dictionary (code -> color) which gives the color of each category code.\r\n         * @type {object} */\r\n        opts.color = opts.color || undefined\r\n\r\n        /** @type { Array.<string> } */\r\n        const codes = Object.keys(opts.color)\r\n\r\n        /** @type { object } @private */\r\n        this.catToI = {}\r\n        for (let i = 0; i < codes.length; i++) this.catToI[codes[i]] = i + ''\r\n\r\n        /** @type { Array.<string> } @private */\r\n        this.colors = []\r\n        for (const code of codes) this.colors.push(opts.color['' + code])\r\n\r\n        /**\r\n         * A function returning the size of the cells, in geographical unit. All cells have the same size.\r\n         * @type {function(number,number):number} */\r\n        this.size = opts.size // (resolution, z) => ...\r\n\r\n        /**\r\n         * @private\r\n         * @type { WebGLSquareColoringCatAdvanced } */\r\n        this.wgp = new WebGLSquareColoringCatAdvanced(this.colors)\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n\r\n        //add vertice and fragment data\r\n        const r2 = resolution / 2\r\n        let c,\r\n            nb = cells.length\r\n        const verticesBuffer = []\r\n        const iBuffer = []\r\n        for (let i = 0; i < nb; i++) {\r\n            c = cells[i]\r\n            const cat = this.code(c)\r\n            if (cat == undefined) {\r\n                console.log('Unexpected category: ' + cat)\r\n                continue\r\n            }\r\n            /** @type {number} */\r\n            const i_ = this.catToI[cat]\r\n            if (isNaN(+i_)) {\r\n                console.log('Unexpected category index: ' + cat + ' ' + i_)\r\n                continue\r\n            }\r\n            verticesBuffer.push(c.x + r2, c.y + r2)\r\n            iBuffer.push(+i_)\r\n        }\r\n\r\n        //create canvas and webgl renderer\r\n        const cvWGL = makeWebGLCanvas(geoCanvas.w + '', geoCanvas.h + '')\r\n        if (!cvWGL) {\r\n            console.error('No webGL')\r\n            return\r\n        }\r\n\r\n        //draw\r\n        const sizeGeo = this.size ? this.size(resolution, z) : resolution + 0.2 * z\r\n        this.wgp.draw(cvWGL.gl, verticesBuffer, iBuffer, geoCanvas.getWebGLTransform(), sizeGeo / z)\r\n\r\n        //draw in canvas geo\r\n        geoCanvas.initCanvasTransform()\r\n        geoCanvas.offscreenCtx.drawImage(cvWGL.canvas, 0, 0)\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\nimport { makeWebGLCanvas } from '../utils/webGLUtils.js'\r\nimport { WebGLSquareColoringAdvanced } from '../utils/WebGLSquareColoringAdvanced.js'\r\n\r\n/**\r\n * Style based on webGL\r\n * To show cells as colored squares, with computation of the colors on GPU side (faster than JavaScript side).\r\n * Alls squares with the same size\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class SquareColorWebGLStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /**\r\n         * A function returning a t value (within [0,1]) for a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell,number,number,object):number} */\r\n        this.tFun = opts.tFun //(c,r,z,vs) => {}\r\n\r\n        /**\r\n         * Distribution stretching method.\r\n         * The stretching is performed on GPU side (fragment shader).\r\n         * @type {{ fun:string, alpha:number }} */\r\n        this.stretching = opts.stretching\r\n\r\n        /**\r\n         * The sample of the color ramp.\r\n         * The color is computed on GPU side (fragment shader) based on those values (linear interpolation).\r\n         * @type {Array.<string>} */\r\n        this.colors =\r\n            opts.colors ||\r\n            [\r\n                'rgb(158, 1, 66)',\r\n                'rgb(248, 142, 83)',\r\n                'rgb(251, 248, 176)',\r\n                'rgb(137, 207, 165)',\r\n                'rgb(94, 79, 162)',\r\n            ].reverse()\r\n        if (opts.color)\r\n            this.colors = [\r\n                opts.color(0),\r\n                opts.color(0.2),\r\n                opts.color(0.4),\r\n                opts.color(0.6),\r\n                opts.color(0.8),\r\n                opts.color(1),\r\n            ]\r\n\r\n        /**\r\n         * Define the opacity of the style, within [0,1].\r\n         * If this opacity is defined, the individual color opacity will be ignored.\r\n         * @type {function(number,number):number} */\r\n        this.opacity = opts.opacity // (r,z) => ...\r\n\r\n        /**\r\n         * A function returning the size of the cells, in geographical unit. All cells have the same size.\r\n         * @type {function(number,number):number} */\r\n        this.size = opts.size // (resolution, z) => ...\r\n    }\r\n\r\n    /**\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //create canvas and webgl renderer\r\n        //for opacity control, see: https://webglfundamentals.org/webgl/lessons/webgl-and-alpha.html\r\n        const cvWGL = makeWebGLCanvas(\r\n            geoCanvas.w + '',\r\n            geoCanvas.h + '',\r\n            this.opacity != undefined ? { premultipliedAlpha: false } : undefined\r\n        )\r\n        if (!cvWGL) {\r\n            console.error('No webGL')\r\n            return\r\n        }\r\n\r\n        //add vertice and fragment data\r\n        const r2 = resolution / 2\r\n        const verticesBuffer = []\r\n        const tBuffer = []\r\n        for (let cell of cells) {\r\n            const t = this.tFun(cell, resolution, z, viewScale)\r\n            if (t == null || t == undefined) continue\r\n            verticesBuffer.push(cell.x + r2, cell.y + r2)\r\n            tBuffer.push(t > 1 ? 1 : t < 0 ? 0 : t)\r\n        }\r\n\r\n        //compute pixel size\r\n        const sizeGeo = this.size ? this.size(resolution, z) : resolution + 0.2 * z\r\n\r\n        //compute opacity\r\n        const op = this.opacity ? this.opacity(resolution, z) : undefined\r\n\r\n        //\r\n        const wgp = new WebGLSquareColoringAdvanced(cvWGL.gl, this.colors, this.stretching, sizeGeo / z, op)\r\n\r\n        //draw\r\n        wgp.draw(verticesBuffer, tBuffer, geoCanvas.getWebGLTransform())\r\n\r\n        // draw in canvas geo\r\n        // NOTE: drawing each tile this way is very inefficient. WebGL is best used with fewer, heavier/larger draw calls.\r\n        geoCanvas.initCanvasTransform()\r\n        geoCanvas.offscreenCtx.drawImage(cvWGL.canvas, 0, 0)\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class StrokeStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the color of the cell.\r\n         * @type {function(import('../core/Dataset.js').Cell,number,number,object):string} */\r\n        this.strokeColor = opts.strokeColor || (() => '#666') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the size of a cell in geographical unit.\r\n         * @type {function(import('../core/Dataset.js').Cell,number,number,object):number} */\r\n        this.size = opts.size || ((cell, resolution) => resolution) //(c,r,z,vs) => {}\r\n\r\n        /** The stroke line width in geographical unit.\r\n         * @type {function(import('../core/Dataset.js').Cell,number,number,object):number} */\r\n        this.strokeWidth = opts.strokeWidth || ((cell, resolution, z) => z * 1.5) //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the shape of a cell.\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number,number,object):import(\"../core/Style.js\").Shape} */\r\n        this.shape = opts.shape || (() => 'square') //(c,r,z,vs) => {}\r\n    }\r\n\r\n    /**\r\n     * Draw cells as squares, with various colors and size.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        const r2 = resolution * 0.5\r\n        for (let c of cells) {\r\n            //color\r\n            const col = this.strokeColor ? this.strokeColor(c, resolution, z, viewScale) : undefined\r\n            if (!col || col === 'none') continue\r\n            ctx.strokeStyle = col\r\n\r\n            //size - in geo unit\r\n            const size = this.size ? this.size(c, resolution, z, viewScale) : resolution\r\n\r\n            //width\r\n            const wi = this.strokeWidth ? this.strokeWidth(c, resolution, z, viewScale) : 1 * z\r\n            if (!wi || wi <= 0) continue\r\n            ctx.lineWidth = wi\r\n\r\n            //shape\r\n            const shape = this.shape ? this.shape(c, resolution, z, viewScale) : 'square'\r\n            if (shape === 'none') continue\r\n\r\n            //get offset\r\n            const offset = this.offset(c, resolution, z)\r\n\r\n            if (shape === 'square') {\r\n                //draw square\r\n                const d = resolution * (1 - size / resolution) * 0.5\r\n                ctx.beginPath()\r\n                ctx.rect(c.x + d + offset.dx, c.y + d + offset.dy, size, size)\r\n                ctx.stroke()\r\n            } else if (shape === 'circle') {\r\n                //draw circle\r\n                ctx.beginPath()\r\n                ctx.arc(c.x + r2 + offset.dx, c.y + r2 + offset.dy, size * 0.5, 0, 2 * Math.PI, false)\r\n                ctx.stroke()\r\n            } else if (shape === 'diamond') {\r\n                const s2 = size * 0.5\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x + r2 - s2, c.y + r2)\r\n                ctx.lineTo(c.x + r2, c.y + r2 + s2)\r\n                ctx.lineTo(c.x + r2 + s2, c.y + r2)\r\n                ctx.lineTo(c.x + r2, c.y + r2 - s2)\r\n                ctx.lineTo(c.x + r2 - s2, c.y + r2)\r\n                ctx.stroke()\r\n            } else if (shape === 'donut') {\r\n                console.error('Not implemented')\r\n            } else if (shape === 'triangle_up') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x - dr2, c.y - dr2)\r\n                ctx.lineTo(c.x + r2, c.y + resolution + dr2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y - dr2)\r\n                ctx.closePath()\r\n                ctx.stroke()\r\n            } else if (shape === 'triangle_down') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x - dr2, c.y + resolution + dr2)\r\n                ctx.lineTo(c.x + r2, c.y - dr2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y + resolution + dr2)\r\n                ctx.closePath()\r\n                ctx.stroke()\r\n            } else if (shape === 'triangle_left') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x + resolution + dr2, c.y + resolution + dr2)\r\n                ctx.lineTo(c.x - dr2, c.y + r2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y - dr2)\r\n                ctx.closePath()\r\n                ctx.stroke()\r\n            } else if (shape === 'triangle_right') {\r\n                const dr2 = (size - resolution) / 2\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x - dr2, c.y - dr2)\r\n                ctx.lineTo(c.x + resolution + dr2, c.y + r2)\r\n                ctx.lineTo(c.x - dr2, c.y + resolution + dr2)\r\n                ctx.closePath()\r\n                ctx.stroke()\r\n            } else {\r\n                throw new Error('Unexpected shape:' + shape)\r\n            }\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { SideStyle } from './SideStyle.js'\r\nimport { SquareColorCategoryWebGLStyle } from './SquareColorCategoryWebGLStyle.js'\r\nimport { classifier as clFun, colorClassifier as cclFun } from '../utils/scale.js'\r\n\r\n/**\r\n * @see https://manifold.net/doc/mfd9/example__tanaka_contours.htm\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class TanakaStyle {\r\n    /**\r\n     * @param {function(import('../core/Dataset.js').Cell):number} value Function that returns the value of a cell\r\n     * @param {Array.<number>} breaks The break values\r\n     * @param {Array.<string>} colors The colors, one more than the break values\r\n     * @param {object} opts\r\n     * @returns {Array.<import(\"../core/Style\").Style>}\r\n     */\r\n    static get(value, breaks, colors, opts = {}) {\r\n        //shadow colors\r\n        opts.colorDark = opts.colorDark || '#111'\r\n        opts.colorBright = opts.colorBright || '#ddd'\r\n\r\n        /** @type { function(number, number):number } */\r\n        opts.width =\r\n            opts.width ||\r\n            ((sideValue, resolution, z) => {\r\n                const minWG = 1 * z\r\n                const maxWG = 4 * z\r\n                const step = (maxWG - minWG) / 3\r\n                return Math.min(minWG + (sideValue - 1) * step, maxWG)\r\n            })\r\n\r\n        //make classifier\r\n        const classifier = clFun(breaks)\r\n        //make colors table\r\n        const colorsDict = {}\r\n        for (let i = 0; i < colors.length; i++) colorsDict[i + ''] = colors[i]\r\n\r\n        const cellStyle = new SquareColorCategoryWebGLStyle({\r\n            code: (cell) => classifier(value(cell)),\r\n            color: colorsDict,\r\n        })\r\n\r\n        const getSideValue = (side) => {\r\n            const cl1 = side.c1 ? classifier(value(side.c1)) : -1\r\n            const cl2 = side.c2 ? classifier(value(side.c2)) : -1\r\n            return cl1 - cl2\r\n        }\r\n\r\n        /** The side style, for the shadow effect */\r\n        const sideStyle = new SideStyle({\r\n            //white or black, depending on orientation and value\r\n            color: (side) => {\r\n                const v = getSideValue(side)\r\n                if (v === 0) return\r\n                if (side.or === 'v') return v < 0 ? opts.colorBright : opts.colorDark\r\n                return v < 0 ? opts.colorDark : opts.colorBright\r\n            },\r\n            //width depends on the value, that is the number of classes of difference\r\n            width: (side, resolution, z) => opts.width(Math.abs(getSideValue(side)), resolution, z),\r\n        })\r\n\r\n        return [cellStyle, sideStyle]\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/**\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class TextStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** A function returning the text of a cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.text = opts.text || (() => 'X') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the color of the cell.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => 'black') //(c,r,z,vs) => {}\r\n\r\n        /** A function returning the font size of a cell in geo unit.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number,object):number} */\r\n        this.fontSize = opts.fontSize || ((cell, resolution) => resolution) //(c,r,z,vs) => {}\r\n\r\n        /** The text font family.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.fontFamily = opts.fontFamily || (() => 'Arial')\r\n\r\n        /** The text font weight.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.fontWeight = opts.fontWeight || (() => 'bold')\r\n    }\r\n\r\n    /**\r\n     * Draw cells as text.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //draw with HTML canvas\r\n        //in screen coordinates\r\n        geoCanvas.initCanvasTransform()\r\n\r\n        for (let cell of cells) {\r\n            //get cell text\r\n            const text = this.text ? this.text(cell, resolution, z, viewScale) : undefined\r\n            if (text == undefined || text == null || text + '' === '') continue\r\n\r\n            //color\r\n            const col = this.color ? this.color(cell, resolution, z, viewScale) : undefined\r\n            if (!col) continue\r\n            ctx.fillStyle = col\r\n\r\n            //font size\r\n            //size - in pixel unit\r\n            const fontSizePix = this.fontSize(cell, resolution, z, viewScale) / z\r\n            if (!fontSizePix) continue\r\n\r\n            //set font\r\n            const fontFamily = this.fontFamily ? this.fontFamily(cell, resolution, z, viewScale) : 'Arial'\r\n            const fontWeight = this.fontWeight ? this.fontWeight(cell, resolution, z, viewScale) : 'bold'\r\n            ctx.font = fontWeight + ' ' + fontSizePix + 'px ' + fontFamily\r\n\r\n            //get offset\r\n            const offset = this.offset(cell, resolution, z)\r\n\r\n            //text position\r\n            ctx.textAlign = 'center'\r\n            const tx = geoCanvas.geoToPixX(cell.x + resolution * 0.5 + offset.dx)\r\n            const ty = geoCanvas.geoToPixY(cell.y + resolution * 0.5 + offset.dy) + fontSizePix * 0.3 //it should be 0.5 but 0.3 seems to work better\r\n\r\n            //draw the text\r\n            ctx.fillText(text, tx, ty)\r\n        }\r\n\r\n        //update legends\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n\r\n    /**\r\n     * Build a function [0,1]->string for characters legend\r\n     *\r\n     * @param {Array.<string>} chars\r\n     * @param {(function(number):number)|undefined} scale\r\n     * @returns {function(number):string}\r\n     */\r\n    static textScale(chars, scale = undefined) {\r\n        const nb = chars.length\r\n        return (t) => {\r\n            if (scale) t = scale(t)\r\n            if (t == 0) return ''\r\n            if (t >= 1) return chars[nb - 1]\r\n            return chars[Math.floor(t * nb)]\r\n        }\r\n    }\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { Style } from '../core/Style.js'\r\n\r\n/** @typedef {\"first\"|\"bottom\"|\"center\"|\"top\"|\"last\"} AnchorModeYEnum */\r\n\r\n/**\r\n * Show cell as timeseries chart\r\n * Can be used for sparkline map of https://datagistips.hypotheses.org/488\r\n *\r\n * @module style\r\n * @author Julien Gaffuri\r\n */\r\nexport class TimeSeriesStyle extends Style {\r\n    /** @param {object} opts */\r\n    constructor(opts) {\r\n        super(opts)\r\n        opts = opts || {}\r\n\r\n        /** The columns of the time series, ordered in chronological order.\r\n         * @type {Array.<string>} */\r\n        this.ts = opts.ts\r\n\r\n        /** A function specifying when a value should be considered as \"no data\" and thus not ignored. The line will have a break at these values.\r\n         * @type {function(string):boolean} */\r\n        this.noData = opts.noData || ((v) => v === undefined || v == '' || v === null || isNaN(+v))\r\n\r\n        //x\r\n        /** in geo unit\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number,number):number} */\r\n        this.offsetX = opts.offsetX || ((c, r, z) => 0)\r\n        /** @type {function(import(\"../core/Dataset.js\").Cell,number,number):number} */\r\n        this.width = opts.width || ((c, r, z) => r)\r\n\r\n        //y\r\n        /** in geo unit\r\n         * @type {function(import(\"../core/Dataset.js\").Cell,number,number):number} */\r\n        this.offsetY = opts.offsetY || ((c, r, z) => 0)\r\n        /** @type {function(import(\"../core/Dataset.js\").Cell,number,number):number} */\r\n        this.height = opts.height || ((c, r, z) => r)\r\n        /** @type {function(import(\"../core/Dataset.js\").Cell,number,number):AnchorModeYEnum} */\r\n        this.anchorModeY = opts.anchorModeY || ((c, r, z) => 'center')\r\n\r\n        /** A function returning the width of the line, in geo unit\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):number} */\r\n        this.lineWidth = opts.lineWidth || ((v, r, s, z) => 1.5 * z)\r\n\r\n        /** A function returning the color of the chart.\r\n         * @type {function(import('../core/Dataset.js').Cell, number, number, object):string} */\r\n        this.color = opts.color || (() => 'black') //(c,r,z,vs) => {}\r\n    }\r\n\r\n    /**\r\n     * Draw cells as text.\r\n     *\r\n     * @param {Array.<import(\"../core/Dataset.js\").Cell>} cells\r\n     * @param {import(\"../core/GeoCanvas.js\").GeoCanvas} geoCanvas\r\n     * @param {number} resolution\r\n     */\r\n    draw(cells, geoCanvas, resolution) {\r\n        //filter\r\n        if (this.filter) cells = cells.filter(this.filter)\r\n\r\n        //\r\n        const z = geoCanvas.view.z\r\n        const ctx = geoCanvas.offscreenCtx\r\n\r\n        //get view scale\r\n        const viewScale = this.viewScale ? this.viewScale(cells, resolution, z) : undefined\r\n\r\n        //compute cell amplitude\r\n        const getAmplitude = (c) => {\r\n            let min, max\r\n            for (let t of this.ts) {\r\n                const val = c[t]\r\n                if (val == undefined) continue\r\n                if (min == undefined || val < min) min = val\r\n                if (max == undefined || val > max) max = val\r\n            }\r\n            if (min == undefined) return undefined\r\n            return max - min\r\n        }\r\n\r\n        //compute max amplitude\r\n        let ampMax\r\n        for (let c of cells) {\r\n            const amp = getAmplitude(c)\r\n            if (amp == undefined) continue\r\n            if (ampMax == undefined || amp > ampMax) ampMax = amp\r\n        }\r\n        if (!ampMax) return\r\n\r\n        const nb = this.ts.length\r\n\r\n        ctx.lineCap = 'butt'\r\n        for (let c of cells) {\r\n            //line width\r\n            /** @type {number|undefined} */\r\n            const wG = this.lineWidth ? this.lineWidth(c, resolution, z, viewScale) : undefined\r\n            if (!wG || wG < 0) continue\r\n\r\n            //line color\r\n            /** @type {string|undefined} */\r\n            const col = this.color ? this.color(c, resolution, z, viewScale) : undefined\r\n            if (!col) continue\r\n\r\n            //x\r\n            const offX = this.offsetX ? this.offsetX(c, resolution, z) : 0\r\n            if (offX == undefined || isNaN(offX)) continue\r\n            const w = this.width ? this.width(c, resolution, z) : resolution\r\n            if (w == undefined || isNaN(w)) continue\r\n\r\n            //y\r\n            const offY = this.offsetY ? this.offsetY(c, resolution, z) : 0\r\n            if (offY == undefined || isNaN(offY)) continue\r\n            const h = this.height ? this.height(c, resolution, z) : resolution\r\n            if (h == undefined || isNaN(h)) continue\r\n            const anchY = this.anchorModeY ? this.anchorModeY(c, resolution, z) : 'center'\r\n            if (!anchY) continue\r\n\r\n            ctx.lineWidth = wG\r\n            ctx.strokeStyle = col\r\n\r\n            //compute anchor Y figures\r\n            let val0, y0\r\n            if (anchY === 'first') {\r\n                //get first value\r\n                val0 = c[this.ts[0]]\r\n                y0 = 0\r\n            } else if (anchY === 'last') {\r\n                //get last value\r\n                val0 = c[this.ts[this.ts.length - 1]]\r\n                y0 = 0\r\n            } else if (anchY === 'bottom') {\r\n                //get min\r\n                for (let t of this.ts) {\r\n                    const val = +c[t]\r\n                    if (val == undefined) continue\r\n                    if (val0 == undefined || val < val0) val0 = val\r\n                }\r\n                y0 = 0\r\n            } else if (anchY === 'top') {\r\n                //get max\r\n                for (let t of this.ts) {\r\n                    const val = +c[t]\r\n                    if (val == undefined) continue\r\n                    if (val0 == undefined || val > val0) val0 = val\r\n                }\r\n                y0 = resolution\r\n            } else if (anchY === 'center') {\r\n                //get min and max\r\n                let min, max\r\n                for (let t of this.ts) {\r\n                    const val = c[t]\r\n                    if (val == undefined) continue\r\n                    if (min == undefined || val < min) min = val\r\n                    if (max == undefined || val > max) max = val\r\n                }\r\n                val0 = (+max + +min) * 0.5\r\n                y0 = resolution / 2\r\n            } else {\r\n                console.log('Unexpected anchorModeY: ' + anchY)\r\n                continue\r\n            }\r\n\r\n            /*/draw line\r\n            if (val0 == undefined || isNaN(val0)) continue\r\n            cg.ctx.beginPath()\r\n            const sX = w / (nb - 1)\r\n            for (let i = 0; i < nb; i++) {\r\n                const val = c[this.ts[i]]\r\n                if (val == undefined || isNaN(val)) break\r\n                if (i == 0)\r\n                    cg.ctx.moveTo(c.x + i * sX + offX, c.y + y0 + (val - val0) * h / ampMax + offY)\r\n                else\r\n                    cg.ctx.lineTo(c.x + i * sX + offX, c.y + y0 + (val - val0) * h / ampMax + offY)\r\n            }\r\n            cg.ctx.stroke()*/\r\n\r\n            //draw line, segment by segment\r\n            const sX = w / (nb - 1)\r\n\r\n            //handle first point\r\n            let v0 = c[this.ts[0]]\r\n            if (!this.noData(v0)) {\r\n                ctx.beginPath()\r\n                ctx.moveTo(c.x + offX, c.y + y0 + ((v0 - val0) * h) / ampMax + offY)\r\n            }\r\n            //console.log(v0, isNaN(v0))\r\n\r\n            let v1\r\n            for (let i = 1; i < nb; i++) {\r\n                v1 = c[this.ts[i]]\r\n\r\n                //draw segment from v0 to v1\r\n\r\n                //both points 'no data'\r\n                if (this.noData(v0) && this.noData(v1)) {\r\n                    //second point 'no data'\r\n                } else if (!this.noData(v0) && this.noData(v1)) {\r\n                    ctx.stroke()\r\n\r\n                    //first point 'no data'\r\n                } else if (this.noData(v0) && !this.noData(v1)) {\r\n                    ctx.beginPath()\r\n                    ctx.moveTo(c.x + i * sX + offX, c.y + y0 + ((v1 - val0) * h) / ampMax + offY)\r\n\r\n                    //both points have data: trace line\r\n                } else {\r\n                    ctx.lineTo(c.x + i * sX + offX, c.y + y0 + ((v1 - val0) * h) / ampMax + offY)\r\n                    //if it is the last point, stroke\r\n                    if (i == nb - 1) ctx.stroke()\r\n                }\r\n                v0 = v1\r\n            }\r\n        }\r\n\r\n        //update legend, if any\r\n        this.updateLegends({ style: this, resolution: resolution, z: z, viewScale: viewScale })\r\n    }\r\n}\r\n","//@ts-check\n'use strict'\n\nimport { initShaderProgram, createShader } from './webGLUtils.js'\nimport { color } from 'd3-color'\n\n/**\n * Everything to easily draw colored squares with webGL.\n * All the same size, but different fill color.\n */\nexport class WebGLSquareColoring {\n    /**\n     *\n     * @param {WebGLRenderingContext} gl\n     */\n    constructor(gl, sizePix) {\n        this.gl = gl\n        this.sizePix = sizePix || 10.0\n\n        this.program = initShaderProgram(\n            gl,\n            createShader(\n                gl,\n                gl.VERTEX_SHADER,\n                `\n            attribute vec2 pos;\n            uniform float sizePix;\n            uniform mat3 mat;\n            attribute vec4 color;\n            varying vec4 vColor;\n            void main() {\n              gl_Position = vec4(mat * vec3(pos, 1.0), 1.0);\n              gl_PointSize = sizePix;\n              vColor = color;\n            }\n          `\n            ),\n            createShader(\n                gl,\n                gl.FRAGMENT_SHADER,\n                `\n            precision mediump float;\n            varying vec4 vColor;\n            void main(void) {\n                vec4 vColor_ = vColor / 255.0;\n                vColor_[3] = 255.0 * vColor_[3];\n                gl_FragColor = vColor_;\n            }`\n            )\n        )\n        gl.useProgram(this.program)\n\n        //buffer data\n        this.verticesBuffer = []\n        this.colorsBuffer = []\n    }\n\n    /** Add data to vertices/size/color buffers for color squares drawing */\n    addPointData(xC, yC, col) {\n        //convert color\n        const cc = color(col)\n        //const cc = {r:45,g:87,b:98,opacity:0.9}\n        if (!cc) return\n\n        //vertices\n        this.verticesBuffer.push(xC, yC)\n        //color\n        this.colorsBuffer.push(cc.r, cc.g, cc.b, cc.opacity)\n    }\n\n    addPointData2(xC, yC, r, g, b, opacity) {\n        //vertices\n        this.verticesBuffer.push(xC, yC)\n        //color\n        this.colorsBuffer.push(r, g, b, opacity)\n    }\n\n    /**  */\n    draw(transfoMat) {\n        const gl = this.gl\n\n        //vertice data\n        gl.bindBuffer(gl.ARRAY_BUFFER, gl.createBuffer())\n        gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(this.verticesBuffer), gl.STATIC_DRAW)\n        const position = gl.getAttribLocation(this.program, 'pos')\n        gl.vertexAttribPointer(\n            position,\n            2, //numComponents\n            gl.FLOAT, //type\n            false, //normalise\n            0, //stride\n            0 //offset\n        )\n        gl.enableVertexAttribArray(position)\n\n        //color data\n        gl.bindBuffer(gl.ARRAY_BUFFER, gl.createBuffer())\n        gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(this.colorsBuffer), gl.STATIC_DRAW)\n        var color = gl.getAttribLocation(this.program, 'color')\n        gl.vertexAttribPointer(color, 4, gl.FLOAT, false, 0, 0)\n        gl.enableVertexAttribArray(color)\n\n        //sizePix\n        gl.uniform1f(gl.getUniformLocation(this.program, 'sizePix'), 1.0 * this.sizePix)\n\n        //transformation\n        gl.uniformMatrix3fv(gl.getUniformLocation(this.program, 'mat'), false, new Float32Array(transfoMat))\n\n        // Enable the depth test\n        //gl.enable(gl.DEPTH_TEST);\n        // Clear the color buffer bit\n        gl.clear(gl.COLOR_BUFFER_BIT)\n        // Set the view port\n        //gl.viewport(0, 0, cg.w, cg.h);\n\n        gl.drawArrays(gl.POINTS, 0, this.verticesBuffer.length / 2)\n    }\n}\n","//@ts-check\n'use strict'\n\nimport { initShaderProgram, createShader } from './webGLUtils.js'\nimport { color } from 'd3-color'\n\n/**\n * Everything to easily draw colored squares with webGL.\n * All the same size, but different fill color.\n * The color interpolation is computed in the fragment shader program, by the GPU, thus it is less flexible but faster.\n */\nexport class WebGLSquareColoringAdvanced {\n    //see:\n    //https://webglfundamentals.org/webgl/lessons/fr/webgl-shaders-and-glsl.html#les-uniforms-dans-les-shaders-de-vertex\n    //https://thebookofshaders.com/glossary/?search=mix\n    //https://thebookofshaders.com/06/\n    //https://thebookofshaders.com/glossary/\n\n    /**\n     *\n     * @param {*} gl\n     * @param {Array.<String>} colors\n     * @param {{fun:string,alpha:number}} stretching\n     * @param {number} sizePix\n     * @param {number|undefined} globalOpacity\n     */\n    constructor(gl, colors, stretching, sizePix = 10, globalOpacity = undefined) {\n        /** @type {WebGLRenderingContext} */\n        this.gl = gl\n        //gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, true);\n        //gl.blendFunc(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA);\n        //gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA);\n\n        /** @type {WebGLShader} */\n        const vShader = createShader(\n            gl,\n            gl.VERTEX_SHADER,\n            `\n        attribute vec2 pos;\n        uniform float sizePix;\n        uniform mat3 mat;\n\n        attribute float t;\n        varying float vt;\n\n        void main() {\n          gl_Position = vec4(mat * vec3(pos, 1.0), 1.0);\n          gl_PointSize = sizePix;\n          vt = t;\n        }\n      `\n        )\n\n        //prepare fragment shader code\n        //declare the uniform and other variables\n        let fshString =\n            '' +\n            'precision mediump float;\\n' +\n            'varying float vt;\\n' +\n            'uniform float alpha;\\n' +\n            (() => {\n                const out = []\n                for (let i = 0; i < colors.length; i++) out.push('uniform vec4 c' + i + ';\\n')\n                return out.join('')\n            })() +\n            //start the main function, apply the stretching of t\n            'void main(void) {\\n'\n\n        if (stretching) {\n            if (stretching.fun == 'pow')\n                //sPow = (t, alpha = 3) => Math.pow(t, alpha);\n                fshString += '   float t = pow(vt, alpha);\\n'\n            else if (stretching.fun == 'powInv')\n                //sPowRev = (t, alpha = 3) => 1 - Math.pow(1 - t, 1 / alpha);\n                fshString += '   float t = 1.0-pow(1.0-vt, 1.0/alpha);\\n'\n            else if (stretching.fun == 'exp')\n                //sExp = (t, alpha = 3) => alpha == 0 ? t : (Math.exp(t * alpha) - 1) / (Math.exp(alpha) - 1);\n                fshString +=\n                    stretching.alpha == 0\n                        ? `float t = vt;`\n                        : '   float t = (exp(vt * alpha) - 1.0) / (exp(alpha) - 1.0);\\n'\n            else if (stretching.fun == 'log')\n                //sExpRev = (t, alpha = 3) => alpha == 0 ? t : 1 - (1 / alpha) * Math.log(Math.exp(alpha) * (1 - t) + t);\n                fshString +=\n                    stretching.alpha == 0\n                        ? `float t = vt;`\n                        : '   float t = 1.0 - (1.0 / alpha) * log(exp(alpha) * (1.0 - vt) + vt);\\n'\n            else if (stretching.fun == 'circle') {\n                if (stretching.alpha == 0)\n                    //if (alpha == 0) return t;\n                    fshString += '   float t = vt;\\n'\n                else if (stretching.alpha == 1)\n                    // if (alpha == 1) return Math.sqrt(2 * t - t * t);\n                    fshString += '   float t = sqrt(vt * (2.0 - vt));\\n'\n                else {\n                    //const a = alpha / (1 - alpha);\n                    //return Math.sqrt(1 / (a * a) + t * (2 / a + 2 - t)) - 1 / a;\n                    fshString +=\n                        '   float a = alpha / (1.0 - alpha);\\n' +\n                        '   float t = sqrt(1.0 / (a * a) + vt * ( 2.0/a + 2.0 - vt )) - 1.0 / a;\\n'\n                }\n            } else if (stretching.fun == 'circleInv') {\n                // 1 - sCircleLow(1 - t, alpha)\n                if (stretching.alpha == 0)\n                    //if (alpha == 0) return t;\n                    fshString += '   float t = vt;\\n'\n                else if (stretching.alpha == 1)\n                    // if (alpha == 1) return Math.sqrt(2 * t - t * t);\n                    fshString += '   float t = 1.0 - sqrt((1.0 - vt) * (1.0 + vt));\\n'\n                else {\n                    //const a = alpha / (1 - alpha);\n                    //return Math.sqrt(1 / (a * a) + (2 * t) / a + 2 * t - t * t) - 1 / a;\n                    fshString +=\n                        '   float a = alpha / (1.0 - alpha);\\n' +\n                        '   float t = 1.0 - sqrt(1.0 / (a * a) + (1.0-vt) * ( 2.0/a + 1.0 + vt )) + 1.0 / a;\\n'\n                }\n            } else {\n                console.error('Unexpected stretching function code: ' + stretching.fun)\n                fshString += '   float t = vt;\\n'\n            }\n        } else {\n            fshString += '   float t = vt;\\n'\n        }\n\n        //choose initial and final colors, and adjust t value\n        if (colors.length == 1) fshString += '   vec4 cI=c0;\\n   vec4 cF=c0;\\n'\n        else if (colors.length == 2) fshString += '   vec4 cI=c0;\\n   vec4 cF=c1;\\n'\n        else {\n            const nb = colors.length - 1\n            const nbs = nb + '.0'\n            fshString += '   vec4 cI;\\n'\n            fshString += '   vec4 cF;\\n'\n            fshString += '   if(t<1.0/' + nbs + ') { cI=c0; cF=c1; t=t*' + nbs + '; }\\n'\n            for (let i = 2; i < nb; i++)\n                fshString +=\n                    '   else if(t<' +\n                    i +\n                    '.0/' +\n                    nbs +\n                    ') { cI=c' +\n                    (i - 1) +\n                    '; cF=c' +\n                    i +\n                    '; t=' +\n                    nbs +\n                    '*t-' +\n                    (i - 1) +\n                    '.0; }\\n'\n            fshString +=\n                '   else { cI=c' + (nb - 1) + '; cF=c' + nb + '; t=' + nbs + '*t-' + (nb - 1) + '.0; }\\n'\n        }\n\n        //one single color\n        if (colors.length == 1) fshString += '   gl_FragColor = vec4(c0[0], c0[1], c0[2], c0[3]);}\\n'\n        //set interpolated color, between initial and final one\n        else fshString += '   gl_FragColor = mix(cI, cF, t);}\\n'\n\n        //console.log(fshString)\n\n        /** @type {WebGLShader} */\n        const fShader = createShader(gl, gl.FRAGMENT_SHADER, fshString)\n\n        /** @type {WebGLProgram} */\n        this.program = initShaderProgram(gl, vShader, fShader)\n        gl.useProgram(this.program)\n\n        //set uniforms\n\n        //sizePix\n        //TODO: bug here. Seems to be limited to some threshold value (around 250).\n        gl.uniform1f(gl.getUniformLocation(this.program, 'sizePix'), 1.0 * sizePix)\n\n        //stretching alpha factor\n        gl.uniform1f(gl.getUniformLocation(this.program, 'alpha'), stretching ? 1.0 * stretching.alpha : 0.0)\n\n        //colors\n        for (let i = 0; i < colors.length; i++) {\n            const c = color(colors[i])\n\n            let opacity = c.opacity\n            if (c.opacity == 1 && globalOpacity != undefined) opacity = globalOpacity\n\n            gl.uniform4fv(gl.getUniformLocation(this.program, 'c' + i), [\n                +c.r / 255.0,\n                +c.g / 255.0,\n                +c.b / 255.0,\n                +opacity,\n            ])\n        }\n    }\n\n    /**  */\n    draw(verticesBuffer, tBuffer, transfoMat) {\n        const gl = this.gl\n        const program = this.program\n\n        //vertice data\n        gl.bindBuffer(gl.ARRAY_BUFFER, gl.createBuffer())\n        gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(verticesBuffer), gl.STATIC_DRAW)\n        const position = gl.getAttribLocation(program, 'pos')\n        gl.vertexAttribPointer(\n            position,\n            2, //numComponents\n            gl.FLOAT, //type\n            false, //normalise\n            0, //stride\n            0 //offset\n        )\n        gl.enableVertexAttribArray(position)\n\n        //t data\n        gl.bindBuffer(gl.ARRAY_BUFFER, gl.createBuffer())\n        gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(tBuffer), gl.STATIC_DRAW)\n        const t = gl.getAttribLocation(program, 't')\n        gl.vertexAttribPointer(t, 1, gl.FLOAT, false, 0, 0)\n        gl.enableVertexAttribArray(t)\n\n        //transformation\n        gl.uniformMatrix3fv(gl.getUniformLocation(program, 'mat'), false, new Float32Array(transfoMat))\n\n        // Enable the depth test\n        //gl.enable(gl.DEPTH_TEST);\n        // Clear the color buffer bit\n        gl.clear(gl.COLOR_BUFFER_BIT)\n        // Set the view port\n        //gl.viewport(0, 0, cg.w, cg.h);\n\n        gl.drawArrays(gl.POINTS, 0, verticesBuffer.length / 2)\n    }\n}\n","//@ts-check\n'use strict'\n\nimport { initShaderProgram, createShader } from './webGLUtils.js'\nimport { color } from 'd3-color'\n\n/**\n * Everything to easily draw colored squares with webGL.\n * All the same size, but different fill color.\n * Color based on categories.\n */\nexport class WebGLSquareColoringCatAdvanced {\n    /**\n     * @param {Array.<string>} colors\n     */\n    constructor(colors) {\n        /**\n         * @type {Array.<string>} */\n        this.colors = colors\n\n        /** Vector shader program\n         * @type {string} */\n        this.vshString = `\n        attribute vec2 pos;\n        uniform float sizePix;\n        uniform mat3 mat;\n\n        attribute float i;\n        varying float vi;\n\n        void main() {\n          gl_Position = vec4(mat * vec3(pos, 1.0), 1.0);\n          gl_PointSize = sizePix;\n          vi = i;\n        }\n        `\n\n        //prepare fragment shader code\n        //declare the uniform and other variables\n        const out = []\n        out.push('precision mediump float;\\nvarying float vi;\\n')\n        //add color uniforms\n        out.push('uniform vec4')\n        for (let i = 0; i < colors.length; i++) {\n            if (i > 0) out.push(',')\n            out.push(' c' + i)\n        }\n        out.push(';\\n')\n        //start the main function\n        out.push('void main(void) {\\n')\n        //choose color i\n        for (let i = 0; i < colors.length; i++) {\n            if (i > 0) out.push('else ')\n            out.push('if(vi==')\n            out.push(i)\n            out.push('.0) gl_FragColor = vec4(c')\n            out.push(i)\n            out.push('[0], c')\n            out.push(i)\n            out.push('[1], c')\n            out.push(i)\n            out.push('[2], c')\n            out.push(i)\n            out.push('[3]);\\n')\n        }\n        out.push('else gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);\\n}')\n        /** Fragment shader program\n         * @type {string} */\n        this.fshString = out.join('')\n    }\n\n    /**  */\n    draw(gl, verticesBuffer, iBuffer, transfoMat, sizePix = 10) {\n        /** @type {WebGLShader} */\n        const vShader = createShader(gl, gl.VERTEX_SHADER, this.vshString)\n\n        /** @type {WebGLShader} */\n        const fShader = createShader(gl, gl.FRAGMENT_SHADER, this.fshString)\n\n        /** @type {WebGLProgram} */\n        const program = initShaderProgram(gl, vShader, fShader)\n        gl.useProgram(program)\n\n        //set uniforms\n\n        //sizePix\n        gl.uniform1f(gl.getUniformLocation(program, 'sizePix'), 1.0 * sizePix)\n\n        //colors\n        for (let i = 0; i < this.colors.length; i++) {\n            const c = color(this.colors[i])\n            gl.uniform4fv(gl.getUniformLocation(program, 'c' + i), [\n                +c.r / 255.0,\n                +c.g / 255.0,\n                +c.b / 255.0,\n                +c.opacity,\n            ])\n        }\n\n        //vertice data\n        gl.bindBuffer(gl.ARRAY_BUFFER, gl.createBuffer())\n        gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(verticesBuffer), gl.STATIC_DRAW)\n        const position = gl.getAttribLocation(program, 'pos')\n        gl.vertexAttribPointer(\n            position,\n            2, //numComponents\n            gl.FLOAT, //type\n            false, //normalise\n            0, //stride\n            0 //offset\n        )\n        gl.enableVertexAttribArray(position)\n\n        //i data\n        gl.bindBuffer(gl.ARRAY_BUFFER, gl.createBuffer())\n        gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(iBuffer), gl.STATIC_DRAW)\n        const i = gl.getAttribLocation(program, 'i')\n        gl.vertexAttribPointer(i, 1, gl.FLOAT, false, 0, 0)\n        gl.enableVertexAttribArray(i)\n\n        //transformation\n        gl.uniformMatrix3fv(gl.getUniformLocation(program, 'mat'), false, new Float32Array(transfoMat))\n\n        // Enable the depth test\n        //gl.enable(gl.DEPTH_TEST);\n        // Clear the color buffer bit\n        gl.clear(gl.COLOR_BUFFER_BIT)\n        // Set the view port\n        //gl.viewport(0, 0, cg.w, cg.h);\n\n        gl.drawArrays(gl.POINTS, 0, verticesBuffer.length / 2)\n    }\n}\n","//@ts-check\r\n'use strict'\r\n\r\nimport { extent, max } from 'd3-array'\r\nimport { scaleQuantile } from 'd3-scale'\r\n\r\n/**\r\n * A scale is simply a function that map a domain to a range.\r\n * @typedef {function(number):number} Scale */\r\n\r\n/**\r\n * A scale whose range is a color (string).\r\n * @typedef {function(number):string} ColorScale */\r\n\r\n/**\r\n * @module utils\r\n */\r\n\r\n/**\r\n * Generic function for view scale - continuous or discrete\r\n *\r\n * @param {{ valueFunction:function(import(\"../core/Dataset\").Cell):number, minValue?:number, minSizePix?:number, maxSizeFactor?:number, range?:[number, number], domain?:[number, number], stretching?:function(number):number, classNumber?:number}} opts\r\n * @returns {function(Array.<import(\"../core/Dataset\").Cell>):Scale}\r\n */\r\nexport const viewScale = (opts) => {\r\n    const valueFunction = opts.valueFunction\r\n    const minValue = opts.minValue || 0\r\n    const minSizePix = opts.minSizePix || 0\r\n    const maxSizeFactor = opts.maxSizeFactor || 1\r\n    const stretching = opts.stretching\r\n    const range_ = opts.range\r\n    const domain_ = opts.domain\r\n    const classNumber = opts.classNumber\r\n    return (cells, resolution, z) => {\r\n        const domain = domain_ || [minValue, max(cells, valueFunction)]\r\n        const range = range_ || [minSizePix * z, resolution * maxSizeFactor]\r\n        const domainSize = domain[1] - domain[0],\r\n            domainMin = domain[0]\r\n        const rangeSize = range[1] - range[0],\r\n            rangeMin = range[0]\r\n        return (t) => {\r\n            //scale to [0,1]\r\n            t = (t - domainMin) / domainSize\r\n            //stretch\r\n            if (stretching) t = stretching(t)\r\n            //classify\r\n            if (classNumber) t = t == 1 ? 1 : Math.floor(t * classNumber) / (classNumber - 1)\r\n            //scale to range\r\n            return rangeMin + t * rangeSize\r\n        }\r\n    }\r\n}\r\n\r\n/**\r\n * Generic function for view scale - quantile\r\n *\r\n * @param {{ valueFunction:function(import(\"../core/Dataset\").Cell):number, classNumber?:number, minSizePix?:number, maxSizeFactor?:number }} opts\r\n * @returns {function(Array.<import(\"../core/Dataset\").Cell>):Scale}\r\n */\r\nexport const viewScaleQuantile = (opts) => {\r\n    const valueFunction = opts.valueFunction\r\n    const classNumber = opts.classNumber || 12\r\n    const minSizePix = opts.minSizePix || 1\r\n    const maxSizeFactor = opts.maxSizeFactor || 1\r\n    const scale = scaleQuantile()\r\n    return (cells, resolution, z) => {\r\n        scale.domain(cells.map(valueFunction))\r\n        const minSizeGeo = minSizePix * z,\r\n            maxSizeGeo = resolution * maxSizeFactor\r\n        scale.range(\r\n            Array.from(\r\n                { length: classNumber },\r\n                (_, i) => minSizeGeo + (i * (maxSizeGeo - minSizeGeo)) / (classNumber - 1)\r\n            )\r\n        )\r\n        scale.breaks = scale.quantiles()\r\n        scale.values = scale.range()\r\n        return scale\r\n    }\r\n}\r\n\r\n/**\r\n * Generic function for color view scale - continuous or discrete\r\n *\r\n * @param {{ valueFunction:function(import(\"../core/Dataset\").Cell):number, colorScale?:function(number):string, colors?:Array.<string>, stretching?:function(number):number }} opts\r\n * @returns {function(Array.<import(\"../core/Dataset\").Cell>):ColorScale}\r\n */\r\nexport const viewScaleColor = (opts) => {\r\n    const valueFunction = opts.valueFunction\r\n    const stretching = opts.stretching\r\n    let colorScale = opts.colorScale || (() => 'purple')\r\n\r\n    //discrete colors case: build continuous color scale from discrete ones.\r\n    const nbClass = opts.colors?.length\r\n    if (opts.colors && nbClass)\r\n        colorScale = (t) => opts.colors[t == 1 ? nbClass - 1 : Math.floor(t * nbClass)]\r\n\r\n    return (cells) => {\r\n        if (cells.length == 0 || !cells) return\r\n        /** @type {[undefined, undefined] | [number, number]} */\r\n        const domain = extent(cells, valueFunction)\r\n        if (domain[0] == undefined) return\r\n        const domainSize = domain[1] - domain[0]\r\n        const scale = (t) => {\r\n            //scale to [0,1]\r\n            t = (t - domain[0]) / domainSize\r\n            //stretch\r\n            if (stretching) t = stretching(t)\r\n            return colorScale(t)\r\n        }\r\n        //function that return the domain value from the [0,1] range.\r\n        scale.invert = (t) => {\r\n            if (stretching) t = stretching.invert(t)\r\n            return domain[0] + t * domainSize\r\n        }\r\n        //discrete colors: return the breaks\r\n        if (opts.colors && nbClass) {\r\n            scale.breaks = []\r\n            for (let i = 1; i < nbClass; i++) scale.breaks.push(scale.invert(i / nbClass))\r\n        }\r\n\r\n        return scale\r\n    }\r\n}\r\n\r\n/**\r\n * Generic function for color view scale - quantile\r\n *\r\n * @param {{ valueFunction:function(import(\"../core/Dataset\").Cell):number, classNumber?:number, colors?:Array.<string>, colorScale?:function(number):string }} opts\r\n * @returns {function(Array.<import(\"../core/Dataset\").Cell>):ColorScale}\r\n */\r\nexport const viewScaleColorQuantile = (opts) => {\r\n    const valueFunction = opts.valueFunction\r\n    const classNumber = opts.classNumber || 12\r\n\r\n    let colors = opts.colors\r\n    if (opts.colorScale) colors = discreteColors(opts.colorScale, classNumber)\r\n    colors =\r\n        colors ||\r\n        Array.from(\r\n            { length: classNumber },\r\n            (_, i) => 'rgb(' + Math.floor((255 * i) / (classNumber - 1)) + ',150,150)'\r\n        )\r\n\r\n    const scale = scaleQuantile().range(colors)\r\n    return (cells) => {\r\n        scale.domain(cells.map(valueFunction))\r\n        scale.breaks = scale.quantiles()\r\n        scale.colors = colors\r\n        return scale\r\n    }\r\n}\r\n\r\n/**\r\n * Combine view scale functions\r\n *\r\n * @param {*} obj\r\n * @returns {function}\r\n */\r\nexport const viewScaleCombination = (obj) => {\r\n    //obj: prop and a function to call\r\n    return (cells, resolution, z) => {\r\n        const out = {}\r\n        for (const p in obj) {\r\n            out[p] = obj[p](cells, resolution, z)\r\n        }\r\n        return out\r\n    }\r\n}\r\n\r\n/**\r\n * Return a classifier function from break values.\r\n * The classifier function returns the class id (from 0 to breaks.length) from a value to classifiy.\r\n * @param {Array.<number>} breaks the breaks\r\n */\r\nexport function classifier(breaks) {\r\n    const bl = breaks.length\r\n    const classifier = (value) => {\r\n        let i = 0\r\n        while (i < bl) {\r\n            const break_ = breaks[i]\r\n            if (value <= break_) return i\r\n            i++\r\n        }\r\n        return i\r\n    }\r\n    classifier.breaks = breaks\r\n    return classifier\r\n}\r\n\r\n/**\r\n * Return a color classifier function from break values.\r\n * The classifier function returns the color from a value to classifiy.\r\n * There should be one color more than break values.\r\n * @param {Array.<number>} breaks the breaks\r\n * @param {Array.<string>} colors the colors\r\n */\r\nexport function colorClassifier(breaks, colors) {\r\n    const classifier_ = classifier(breaks)\r\n    const colorClassifier = (value) => colors[classifier_(value)]\r\n    colorClassifier.breaks = breaks\r\n    colorClassifier.colors = colors\r\n    return colorClassifier\r\n}\r\n\r\n/**\r\n * Make array of colors from a colorScale.\r\n * It is a kind of sampling, or un-interpolation\r\n *\r\n * @param {function(number):string} colorScale\r\n * @param {number} nb\r\n */\r\nexport function discreteColors(colorScale, nb) {\r\n    if (nb == 1) return [colorScale(0.5)]\r\n    const out = []\r\n    for (let i = 0; i < nb; i++) out.push(colorScale(i / (nb - 1)))\r\n    return out\r\n}\r\n","//@ts-check\r\n'use strict'\r\n\r\n/**\r\n * @module utils\r\n */\r\n\r\n//TODO invert for circular\r\n//TODO use Math.sqrt\r\n//TODO validate\r\n\r\n/**\r\n * Some function [0,1]->[0,1] to stretch range of values.\r\n * @see https://github.com/eurostat/gridviz/blob/master/docs/reference.md#stretching\r\n * @see https://observablehq.com/@jgaffuri/stretching\r\n */\r\n\r\n//identity function\r\nconst identity = (t) => t\r\nidentity.invert = identity\r\n\r\n/**\r\n * @param {number} base\r\n * @returns {function(number):number}\r\n */\r\nexport const exponentialScale = (base = 3) => {\r\n    if (base == 0) return identity\r\n    const a = Math.exp(base) - 1\r\n    const f = (t) => (Math.exp(t * base) - 1) / a\r\n    f.invert = (t) => Math.log(a * t + 1) / base\r\n    return f\r\n}\r\n\r\n/**\r\n * @param {number} base\r\n * @returns {function(number):number}\r\n */\r\nexport const logarithmicScale = (base = 3) => {\r\n    if (base == 0) return identity\r\n    const a = Math.exp(base),\r\n        b = 1 - a\r\n    const f = (t) => 1 - Math.log(a + t * b) / base\r\n    f.invert = (t) => (Math.exp((1 - t) * base) - a) / b\r\n    return f\r\n}\r\n\r\n/**\r\n * @param {number} exponent\r\n * @returns {function(number):number}\r\n */\r\nexport const powerScale = (exponent = 3) => {\r\n    if (exponent == 1) return identity\r\n    //TODO if (exponent == 0.5) return Math.sqrt\r\n    const f = (t) => Math.pow(t, exponent)\r\n    const a = 1 / exponent\r\n    f.invert = (t) => Math.pow(t, a)\r\n    return f\r\n}\r\n\r\n/**\r\n * @param {number} exponent\r\n * @returns {function(number):number}\r\n */\r\nexport const powerInverseScale = (exponent = 3) => {\r\n    if (exponent == 1) return identity\r\n    //TODO if (exponent == 2) return t => 1 - Math.sqrt(1 - t)\r\n    const a = 1 / exponent\r\n    const f = (t) => 1 - Math.pow(1 - t, a)\r\n    f.invert = (t) => 1 - Math.pow(1 - t, exponent)\r\n    return f\r\n}\r\n\r\n/**\r\n * @param {number} circularity\r\n * @returns {function(number):number}\r\n */\r\nexport const circularScale = (circularity = 0.8) => {\r\n    if (circularity == 0) return identity\r\n    if (circularity == 1) return (t) => Math.sqrt(t * (2 - t))\r\n    else {\r\n        const a = circularity / (1 - circularity)\r\n        return (t) => Math.sqrt(1 / (a * a) + t * (2 / a + 2 - t)) - 1 / a\r\n    }\r\n}\r\n\r\n/**\r\n * @param {number} circularity\r\n * @returns {function(number):number}\r\n */\r\nexport const circularInverseScale = (circularity = 0.8) => {\r\n    if (circularity == 0) return identity\r\n    const f = circularScale(circularity)\r\n    return (t) => 1 - f(1 - t)\r\n}\r\n\r\n//test\r\n/*\r\nconst test = (f, fun, a, err = 1e-12) => {\r\n    for (let t = 0; t <= 1; t += 1 / 50) {\r\n        const er = t - f.invert(f(t))\r\n        if (Math.abs(er) < err) continue\r\n        console.log(fun, a, er)\r\n    }\r\n}\r\n\r\nfor (let fun of [powerScale, powerInverseScale])\r\n    for (let exp = -30; exp <= 50; exp += 1) {\r\n        if (exp == 0) continue\r\n        const f = fun(exp)\r\n        test(f, fun, exp)\r\n    }\r\n\r\n\r\nfor (let fun of [exponentialScale, logarithmicScale])\r\n    for (let base = -20; base <= 20; base += 1) {\r\n        //if (exp == 0) continue\r\n        const f = fun(base)\r\n        test(f, fun, base, 1e-10)\r\n    }\r\n*/\r\n","//@ts-check\r\n'use strict'\r\n\r\nimport { interpolateLab } from 'd3-interpolate'\r\n\r\nexport const ternaryClassifier = (properties, totalFunction, opts = {}) => {\r\n    //the three properties\r\n    const p0 = properties[0],\r\n        p1 = properties[1],\r\n        p2 = properties[2]\r\n\r\n    //the classifier center point. sum must be equal to 1\r\n    const [c0, c1, c2] = opts.center || [1 / 3, 1 / 3, 1 / 3]\r\n\r\n    //parameter to decide wether to use mixed classes m0, m1, m2.\r\n    const withMixedClasses = opts.withMixedClasses != undefined ? opts.withMixedClasses : true\r\n\r\n    //paramerter decide wether to use a central class, and the size of this central class.\r\n    //set to 0 or undefined for not showing any central class. Set to 1 for a central class that contains the mix classes\r\n    const cc = opts.centerCoefficient ? 1 - opts.centerCoefficient : undefined\r\n\r\n    //the output classifier method\r\n    const fun = (c) => {\r\n        //get total\r\n        const tot = totalFunction(c)\r\n        if (!tot) return undefined\r\n        //compute shares\r\n        const [s0, s1, s2] = [+c[p0] / tot, +c[p1] / tot, +c[p2] / tot]\r\n\r\n        //class 0\r\n        if (s0 >= c0 && s1 <= c1 && s2 <= c2) {\r\n            //central class near class 0\r\n            if (cc != undefined && (s2 - c2) * (c1 - cc * c1) >= (s1 - cc * c1) * (cc * c2 - c2))\r\n                return 'center'\r\n            return '0'\r\n        }\r\n        //class 1\r\n        if (s0 <= c0 && s1 >= c1 && s2 <= c2) {\r\n            //central class near class 1\r\n            if (cc != undefined && (s2 - c2) * (c0 - cc * c0) >= (s0 - cc * c0) * (cc * c2 - c2))\r\n                return 'center'\r\n            return '1'\r\n        }\r\n        //class 2\r\n        if (s0 <= c0 && s1 <= c1 && s2 >= c2) {\r\n            //central class near class 2\r\n            if (cc != undefined && (s1 - c1) * (c0 - cc * c0) >= (s0 - cc * c0) * (cc * c1 - c1))\r\n                return 'center'\r\n            return '2'\r\n        }\r\n        //middle class 0 - intersection class 1 and 2\r\n        if (s0 <= c0 && s1 >= c1 && s2 >= c2) {\r\n            //central class\r\n            if (cc != undefined && s0 > cc * c0) return 'center'\r\n            if (withMixedClasses) return 'm12'\r\n            return s1 > s2 ? '1' : '2'\r\n        }\r\n        //middle class 1 - intersection class 0 and 1\r\n        if (s0 >= c0 && s1 <= c1 && s2 >= c2) {\r\n            //central class\r\n            if (cc != undefined && s1 > cc * c1) return 'center'\r\n            if (withMixedClasses) return 'm02'\r\n            return s0 > s2 ? '0' : '2'\r\n        }\r\n        //middle class 2 - intersection class 0 and 1\r\n        if (s0 >= c0 && s1 >= c1 && s2 <= c2) {\r\n            //central class\r\n            if (cc != undefined && s2 > cc * c2) return 'center'\r\n            if (withMixedClasses) return 'm01'\r\n            return s1 > s0 ? '1' : '0'\r\n        }\r\n        //should not happen\r\n        return 'unknown'\r\n    }\r\n\r\n    //attach information to output function\r\n    fun.center = [c0, c1, c2]\r\n    fun.centerCoefficient = opts.centerCoefficient\r\n\r\n    return fun\r\n}\r\n\r\nexport const ternaryColorClassifier = (properties, totalFunction, colors, opts = {}) => {\r\n    //the three colors\r\n    const [color0, color1, color2] = colors || ['red', 'green', 'blue']\r\n\r\n    //the color interpolation function\r\n    const colorInterpolation = opts.colorInterpolation || interpolateLab\r\n\r\n    //parameter to decide wether to use mixed classes.\r\n    const withMixedClasses = opts.withMixedClasses != undefined ? opts.withMixedClasses : true\r\n    //https://d3js.org/d3-interpolate/color\r\n    const mixColorFunction = (color1, color2) => colorInterpolation(color1, color2)(0.5)\r\n    //the colors corresponding to the mixed classes\r\n    const [mixColor0, mixColor1, mixColor2] =\r\n        opts.mixedColors || withMixedClasses\r\n            ? [\r\n                  mixColorFunction(color1, color2),\r\n                  mixColorFunction(color0, color2),\r\n                  mixColorFunction(color0, color1),\r\n              ]\r\n            : []\r\n\r\n    //the central color, used for the central class, if any. The central class is the class of relatively balanced values, around the center point\r\n    const centerColor =\r\n        opts.centerColor || colorInterpolation(mixColorFunction(color0, color1), color2)(0.333)\r\n\r\n    //make classifier\r\n    const classifier = ternaryClassifier(properties, totalFunction, opts)\r\n\r\n    //the output color classifier method\r\n    const fun = (c) => {\r\n        const cla = classifier(c)\r\n        if (cla == '0') return color0\r\n        if (cla == '1') return color1\r\n        if (cla == '2') return color2\r\n        if (cla == 'm12') return mixColor0\r\n        if (cla == 'm02') return mixColor1\r\n        if (cla == 'm01') return mixColor2\r\n        if (cla == 'center') return centerColor\r\n        return opts.defaultColor || 'black'\r\n    }\r\n    fun.center = classifier.center\r\n    fun.centerCoefficient = opts.centerCoefficient\r\n    fun.colors = [color0, color1, color2]\r\n    fun.mixColors = [mixColor0, mixColor1, mixColor2]\r\n    fun.centerColor = centerColor\r\n    fun.classifier = classifier\r\n\r\n    return fun\r\n}\r\n\r\n/*\r\nconst orderedIndexesDec = arr => [...arr.keys()].sort((a, b) => arr[b] - arr[a]);\r\nconst orderedIndexesInc = arr => [...arr.keys()].sort((a, b) => arr[a] - arr[b]);\r\n\r\nexport const ternaryClassifier = (properties, totalFunction, opts = {}) => {\r\n    const lowThreshold = opts.lowThreshold || [1 / 3, 1 / 3, 1 / 3]\r\n    const highThreshold = opts.highThreshold || [2 / 3, 2 / 3, 2 / 3]\r\n    const colors = opts.colors || [\"red\", \"green\", \"blue\"]\r\n    const colorInterpolation = opts.colorInterpolation || interpolateLab\r\n\r\n    //https://d3js.org/d3-interpolate/color\r\n    const middleColorFunction = (color1, color2) => colorInterpolation(color1, color2)(0.5)\r\n    const middleColors = opts.middleColors || [middleColorFunction(colors[1], colors[2]), middleColorFunction(colors[0], colors[2]), middleColorFunction(colors[0], colors[1])]\r\n    const centerColor = opts.centerColor || colorInterpolation(middleColors[2], colors[2])(0.333)\r\n\r\n    const high_ = orderedIndexesDec(highThreshold)\r\n    const low_ = orderedIndexesInc(lowThreshold)\r\n\r\n    const p0 = properties[0], p1 = properties[1], p2 = properties[2]\r\n    const fun = c => {\r\n        //get total\r\n        const tot = totalFunction(c)\r\n        if (!tot) return undefined\r\n        //compute shares\r\n        const shares = [+c[p0] / tot, +c[p1] / tot, +c[p2] / tot]\r\n        //return colors\r\n        //start first with the extreme high (triangles): from the larger value (small triangle) to the lower value (large triangle)\r\n        for (let i of high_) if (shares[i] > highThreshold[i]) return colors[i]\r\n        //then draw the extreme low (trapeziums): from the lower value (small trapeziums) to the larger values (large trapezium)\r\n        for (let i of low_) if (shares[i] < lowThreshold[i]) return middleColors[i]\r\n        //else central color\r\n        return centerColor\r\n    }\r\n    fun.lowThreshold = lowThreshold\r\n    fun.highThreshold = highThreshold\r\n    fun.colors = colors\r\n    fun.middleColors = middleColors\r\n    fun.centerColor = centerColor\r\n    fun.lowIndex = low_\r\n    fun.highIndex = high_\r\n    return fun\r\n}\r\n*/\r\n\r\n/*\r\nexport const ternaryClassifier3 = (properties, totalFunction, opts = {}) => {\r\n    const [a01, a12, a20] = opts.thresholds || [1/3, 1/3, 1/3]\r\n    const [c0, c1, c2] = opts.colors || [\"red\", \"green\", \"blue\"]\r\n    const centerColor = opts.centerColor || \"gray\"\r\n\r\n    const fff = a => a == 1 ? Infinity : a / (1 - a)\r\n    const c01 = fff(a01), c12 = fff(a12), c20 = fff(a20)\r\n\r\n    const p0 = properties[0], p1 = properties[1], p2 = properties[2]\r\n    const fun = c => {\r\n        //get total\r\n        const tot = totalFunction(c)\r\n        if (!tot) return undefined\r\n        //compute shares\r\n        const [s0, s1, s2] = [+c[p0] / tot, +c[p1] / tot, +c[p2] / tot]\r\n        //return colors\r\n        if (s0 * c01 > s1 && s0 > s2 * c20) return c0\r\n        else if (s1 > s0 * c01 && s1 * c12 > s2) return c1\r\n        else if (s2 * c20 > s0 && s2 > s1 * c12) return c2\r\n        else return centerColor\r\n    }\r\n    fun.colors = [c0, c1, c2]\r\n    fun.centerColor = centerColor\r\n    return fun\r\n}\r\n*/\r\n","//@ts-check\r\n'use strict'\r\n\r\n/**\r\n * Get the class id from a value and class break values\r\n *\r\n * @param {number} v the value\r\n * @param {Array.<number>} breaks the breaks\r\n * @returns The class id, from 0 to breaks.length\r\n * @deprecated use getClassifier instead.\r\n */\r\nexport function getClass(v, breaks) {\r\n    if (!breaks) return\r\n    if (breaks.length == 0) return 0\r\n    if (v <= breaks[0]) return 0\r\n    for (let i = 1; i < breaks.length; i++) if (breaks[i - 1] < v && v <= breaks[i]) return i\r\n    return breaks.length\r\n}\r\n\r\n//take 'nice' value (power of ten, or multiple)\r\nexport function nice(v, multiples = [8, 6, 5, 4, 2.5, 2]) {\r\n    //compute bigger power of ten below\r\n    const v_ = Math.pow(10, Math.floor(Math.log10(v)))\r\n    for (let multiple of multiples) if (v_ * multiple <= v) return v_ * multiple\r\n    return v_\r\n}\r\n\r\n/*\r\n//no longer used\r\nexport function loadImage(src) {\r\n    return new Promise((resolve, reject) => {\r\n        const img = new Image();\r\n        img.onload = function () { resolve(img); };\r\n        img.onerror = function () { reject(new Error('Error loading image')); };\r\n        img.src = src;\r\n    });\r\n}\r\n*/\r\n\r\n/*\r\nexport let monitor = false\r\n\r\nlet previousDate\r\nexport function monitorDuration(message) {\r\n    const nowDate = Date.now()\r\n\r\n    //first call\r\n    if (!previousDate) {\r\n        previousDate = nowDate\r\n        console.log(previousDate, message)\r\n        return\r\n    }\r\n\r\n    const d = nowDate - previousDate\r\n    previousDate = nowDate\r\n    console.log(d, message)\r\n}\r\n*/\r\n","//@ts-check\r\n'use strict'\r\n\r\n/**\r\n * @param {string} width\r\n * @param {string} height\r\n * @param {object} opts\r\n * @returns {{canvas:HTMLCanvasElement, gl:WebGLRenderingContext}}\r\n */\r\nexport function makeWebGLCanvas(width, height, opts = {}) {\r\n    const canvas = document.createElement('canvas')\r\n    canvas.setAttribute('width', width)\r\n    canvas.setAttribute('height', height)\r\n    /** @type {WebGLRenderingContext} */\r\n    const gl = canvas.getContext('webgl', opts)\r\n    if (!gl) {\r\n        throw new Error('Unable to initialize WebGL. Your browser or machine may not support it.')\r\n    }\r\n    return { canvas: canvas, gl: gl }\r\n}\r\n\r\n/**\r\n * Initialize a shader program, so WebGL knows how to draw our data\r\n *\r\n * @param {WebGLRenderingContext} gl\r\n * @param  {...WebGLShader} shaders\r\n * @returns {WebGLProgram}\r\n */\r\nexport function initShaderProgram(gl, ...shaders) {\r\n    /** @type {WebGLProgram|null} */\r\n    const program = gl.createProgram()\r\n    if (program == null) throw new Error('Cannot create webGL program')\r\n    for (const shader of shaders) gl.attachShader(program, shader)\r\n    gl.linkProgram(program)\r\n    if (gl.getProgramParameter(program, gl.LINK_STATUS)) return program\r\n    throw new Error(gl.getProgramInfoLog(program) || 'Cannot create webGL program (2)')\r\n}\r\n\r\n/**\r\n * Creates a shader of the given type, uploads the source and compiles it.\r\n *\r\n * @param {WebGLRenderingContext} gl\r\n * @param {number} type\r\n * @param  {...string} sources\r\n * @returns {WebGLShader}\r\n */\r\nexport function createShader(gl, type, ...sources) {\r\n    /** @type {WebGLShader|null} */\r\n    const shader = gl.createShader(type)\r\n    if (shader == null) throw new Error('Cannot create webGL shader')\r\n    gl.shaderSource(shader, sources.join('\\n'))\r\n    gl.compileShader(shader)\r\n    if (gl.getShaderParameter(shader, gl.COMPILE_STATUS)) return shader\r\n    throw new Error(gl.getShaderInfoLog(shader) || 'Cannot create webGL shader (2)')\r\n}\r\n\r\n/**\r\n * Check if webGL is supported\r\n *\r\n * @returns {boolean}\r\n */\r\nexport function checkWebGLSupport() {\r\n    try {\r\n        const canvas = document.createElement('canvas')\r\n        return !!(\r\n            !!window.WebGLRenderingContext &&\r\n            (canvas.getContext('webgl') || canvas.getContext('experimental-webgl'))\r\n        )\r\n    } catch (err) {\r\n        return false\r\n    }\r\n}\r\n","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","//@ts-check\r\n'use strict'\r\n\r\n// the application\r\nexport { Map } from './core/Map.js'\r\nexport { GeoCanvas } from './core/GeoCanvas.js'\r\nexport { Style } from './core/Style.js'\r\nexport { Layer } from './core/Layer.js'\r\nexport { Dataset } from './core/Dataset.js'\r\nexport { MultiResolutionDataset } from './core/MultiResolutionDataset.js'\r\n\r\n// export dataset types\r\nexport { TiledGrid } from './dataset/TiledGrid.js'\r\nexport { CSVGrid } from './dataset/CSVGrid.js'\r\nexport { JSGrid } from './dataset/JSGrid.js'\r\n\r\n// export styles\r\nexport { ShapeColorSizeStyle } from './style/ShapeColorSizeStyle.js'\r\nexport { StrokeStyle } from './style/StrokeStyle.js'\r\nexport { JoyPlotStyle } from './style/JoyPlotStyle.js'\r\nexport { CompositionStyle } from './style/CompositionStyle.js'\r\nexport { SegmentStyle } from './style/SegmentStyle.js'\r\nexport { TextStyle } from './style/TextStyle.js'\r\nexport { PillarStyle } from './style/PillarStyle.js'\r\nexport { SideStyle } from './style/SideStyle.js'\r\n//export { ContourStyle } from './style/ContourStyle.js'\r\nexport { SideCategoryStyle } from './style/SideCategoryStyle.js'\r\nexport { DotDensityStyle } from './style/DotDensityStyle.js'\r\nexport { TanakaStyle } from './style/TanakaStyle.js'\r\nexport { LegoStyle } from './style/LegoStyle.js'\r\nexport { SquareColorWebGLStyle } from './style/SquareColorWebGLStyle.js'\r\nexport { SquareColorCategoryWebGLStyle } from './style/SquareColorCategoryWebGLStyle.js'\r\nexport { MosaicStyle } from './style/MosaicStyle.js'\r\nexport { NinjaStarStyle } from './style/NinjaStarStyle.js'\r\nexport { TimeSeriesStyle } from './style/TimeSeriesStyle.js'\r\nexport { IsoFenceStyle } from './style/IsoFenceStyle.js'\r\nexport { ImageStyle } from './style/ImageStyle.js'\r\n\r\n// export additional layers\r\nexport { GridLayer } from './layer/GridLayer.js'\r\nexport { BackgroundLayer } from './layer/BackgroundLayer.js'\r\nexport { BackgroundLayerWMS } from './layer/BackgroundLayerWMS.js'\r\nexport { BackgroundLayerImage } from './layer/BackgroundLayerImage.js'\r\nexport { LabelLayer } from './layer/LabelLayer.js'\r\nexport { GeoJSONLayer } from './layer/GeoJSONLayer.js'\r\n\r\n// export legends\r\nexport { ColorLegend } from './legend/ColorLegend.js'\r\nexport { ColorDiscreteLegend } from './legend/ColorDiscreteLegend.js'\r\nexport { ColorCategoryLegend } from './legend/ColorCategoryLegend.js'\r\nexport {\r\n    SizeLegend,\r\n    sizeLegend,\r\n    sizeLegendViewScale,\r\n    sizeDiscreteLegend,\r\n    sizeDiscreteViewScaleLegend,\r\n} from './legend/SizeLegend.js'\r\nexport { OrientationLegend, orientationLegend } from './legend/OrientationLegend.js'\r\nexport { TernaryLegend } from './legend/TernaryLegend.js'\r\n\r\n// export { goToStraight, zoomTo } from \"./utils/zoomUtils\"\r\nexport * from './utils/stretching.js'\r\nexport * from './utils/scale.js'\r\nexport * from './utils/ternary.js'\r\nexport { nice } from './utils/utils.js'\r\n\r\nimport { GeoCanvas } from './core/GeoCanvas.js'\r\nexport const getParameterByName = GeoCanvas.getParameterByName\r\n\r\n// set default d3 locale\r\nimport { formatDefaultLocale } from 'd3-format'\r\nformatDefaultLocale({\r\n    decimal: '.',\r\n    thousands: ' ',\r\n    grouping: [3],\r\n    currency: ['', '€'],\r\n})\r\n"],"names":[],"sourceRoot":""} \ No newline at end of file diff --git a/dist/gridviz.min.js b/dist/gridviz.min.js index a32eba3ac..280bd341b 100644 --- a/dist/gridviz.min.js +++ b/dist/gridviz.min.js @@ -1,2 +1,2 @@ /*! For license information please see gridviz.min.js.LICENSE.txt */ -!function(t,e){"object"==typeof exports&&"object"==typeof module?module.exports=e():"function"==typeof define&&define.amd?define([],e):"object"==typeof exports?exports.gridviz=e():t.gridviz=e()}(self,(()=>(()=>{"use strict";var t={d:(e,r)=>{for(var n in r)t.o(r,n)&&!t.o(e,n)&&Object.defineProperty(e,n,{enumerable:!0,get:r[n]})},o:(t,e)=>Object.prototype.hasOwnProperty.call(t,e),r:t=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(t,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(t,"__esModule",{value:!0})}},e={};function r(){}function n(t){return null==t?r:function(){return this.querySelector(t)}}function o(){return[]}function i(t){return null==t?o:function(){return this.querySelectorAll(t)}}function a(t){return function(){return this.matches(t)}}function u(t){return function(e){return e.matches(t)}}t.r(e),t.d(e,{BackgroundLayer:()=>dl,BackgroundLayerImage:()=>Ml,BackgroundLayerWMS:()=>_l,CSVGrid:()=>Wo,ColorCategoryLegend:()=>Cs,ColorDiscreteLegend:()=>Ss,ColorLegend:()=>ps,CompositionStyle:()=>Ri,Dataset:()=>$n,DotDensityStyle:()=>Ya,GeoCanvas:()=>Dr,GeoJSONLayer:()=>rs,GridLayer:()=>cl,ImageStyle:()=>Kc,IsoFenceStyle:()=>Bc,JSGrid:()=>Jo,JoyPlotStyle:()=>_i,LabelLayer:()=>$l,Layer:()=>Nn,LegoStyle:()=>Bu,Map:()=>dn,MosaicStyle:()=>cc,MultiResolutionDataset:()=>Xn,NinjaStarStyle:()=>bc,OrientationLegend:()=>ef,PillarStyle:()=>la,SegmentStyle:()=>$i,ShapeColorSizeStyle:()=>ai,SideCategoryStyle:()=>ka,SideStyle:()=>ma,SizeLegend:()=>Ws,SquareColorCategoryWebGLStyle:()=>nu,SquareColorWebGLStyle:()=>Qu,StrokeStyle:()=>vi,Style:()=>Cn,TanakaStyle:()=>ku,TernaryLegend:()=>ff,TextStyle:()=>Qi,TiledGrid:()=>To,TimeSeriesStyle:()=>kc,circularInverseScale:()=>mf,circularScale:()=>bf,classifier:()=>xu,colorClassifier:()=>Su,discreteColors:()=>Ou,exponentialScale:()=>yf,getParameterByName:()=>th,logarithmicScale:()=>pf,nice:()=>Ms,orientationLegend:()=>rf,powerInverseScale:()=>df,powerScale:()=>vf,sizeDiscreteLegend:()=>Ys,sizeDiscreteViewScaleLegend:()=>Hs,sizeLegend:()=>Us,sizeLegendViewScale:()=>$s,ternaryClassifier:()=>Df,ternaryColorClassifier:()=>Gf,viewScale:()=>du,viewScaleColor:()=>mu,viewScaleColorQuantile:()=>gu,viewScaleCombination:()=>wu,viewScaleQuantile:()=>bu});var c=Array.prototype.find;function l(){return this.firstElementChild}var s=Array.prototype.filter;function f(){return Array.from(this.children)}function h(t){return new Array(t.length)}function y(t,e){this.ownerDocument=t.ownerDocument,this.namespaceURI=t.namespaceURI,this._next=null,this._parent=t,this.__data__=e}function p(t,e,r,n,o,i){for(var a,u=0,c=e.length,l=i.length;ue?1:t>=e?0:NaN}y.prototype={constructor:y,appendChild:function(t){return this._parent.insertBefore(t,this._next)},insertBefore:function(t,e){return this._parent.insertBefore(t,e)},querySelector:function(t){return this._parent.querySelector(t)},querySelectorAll:function(t){return this._parent.querySelectorAll(t)}};var g="http://www.w3.org/1999/xhtml";const w={svg:"http://www.w3.org/2000/svg",xhtml:g,xlink:"http://www.w3.org/1999/xlink",xml:"http://www.w3.org/XML/1998/namespace",xmlns:"http://www.w3.org/2000/xmlns/"};function x(t){var e=t+="",r=e.indexOf(":");return r>=0&&"xmlns"!==(e=t.slice(0,r))&&(t=t.slice(r+1)),w.hasOwnProperty(e)?{space:w[e],local:t}:t}function S(t){return function(){this.removeAttribute(t)}}function O(t){return function(){this.removeAttributeNS(t.space,t.local)}}function _(t,e){return function(){this.setAttribute(t,e)}}function j(t,e){return function(){this.setAttributeNS(t.space,t.local,e)}}function P(t,e){return function(){var r=e.apply(this,arguments);null==r?this.removeAttribute(t):this.setAttribute(t,r)}}function k(t,e){return function(){var r=e.apply(this,arguments);null==r?this.removeAttributeNS(t.space,t.local):this.setAttributeNS(t.space,t.local,r)}}function E(t){return t.ownerDocument&&t.ownerDocument.defaultView||t.document&&t||t.defaultView}function T(t){return function(){this.style.removeProperty(t)}}function C(t,e,r){return function(){this.style.setProperty(t,e,r)}}function A(t,e,r){return function(){var n=e.apply(this,arguments);null==n?this.style.removeProperty(t):this.style.setProperty(t,n,r)}}function M(t,e){return t.style.getPropertyValue(e)||E(t).getComputedStyle(t,null).getPropertyValue(e)}function z(t){return function(){delete this[t]}}function L(t,e){return function(){this[t]=e}}function R(t,e){return function(){var r=e.apply(this,arguments);null==r?delete this[t]:this[t]=r}}function I(t){return t.trim().split(/^|\s+/)}function F(t){return t.classList||new N(t)}function N(t){this._node=t,this._names=I(t.getAttribute("class")||"")}function B(t,e){for(var r=F(t),n=-1,o=e.length;++n=0&&(this._names.splice(e,1),this._node.setAttribute("class",this._names.join(" ")))},contains:function(t){return this._names.indexOf(t)>=0}};var st=[null];function ft(t,e){this._groups=t,this._parents=e}function ht(){return new ft([[document.documentElement]],st)}ft.prototype=ht.prototype={constructor:ft,select:function(t){"function"!=typeof t&&(t=n(t));for(var e=this._groups,r=e.length,o=new Array(r),i=0;i=j&&(j=_+1);!(O=x[j])&&++j=0;)(n=o[i])&&(a&&4^n.compareDocumentPosition(a)&&a.parentNode.insertBefore(n,a),a=n);return this},sort:function(t){function e(e,r){return e&&r?t(e.__data__,r.__data__):!e-!r}t||(t=m);for(var r=this._groups,n=r.length,o=new Array(n),i=0;i1?this.each((null==e?T:"function"==typeof e?A:C)(t,e,null==r?"":r)):M(this.node(),t)},property:function(t,e){return arguments.length>1?this.each((null==e?z:"function"==typeof e?R:L)(t,e)):this.node()[t]},classed:function(t,e){var r=I(t+"");if(arguments.length<2){for(var n=F(this.node()),o=-1,i=r.length;++o=0&&(e=t.slice(r+1),t=t.slice(0,r)),{type:t,name:e}}))}(t+""),a=i.length;if(!(arguments.length<2)){for(u=e?at:it,n=0;n{}};function dt(){for(var t,e=0,r=arguments.length,n={};e=0&&(e=t.slice(r+1),t=t.slice(0,r)),t&&!n.hasOwnProperty(t))throw new Error("unknown type: "+t);return{type:t,name:e}}))),a=-1,u=i.length;if(!(arguments.length<2)){if(null!=e&&"function"!=typeof e)throw new Error("invalid callback: "+e);for(;++a0)for(var r,n,o=new Array(r),i=0;i=0&&e._call.call(void 0,t),e=e._next;--Et}()}finally{Et=0,function(){for(var t,e,r=Pt,n=1/0;r;)r._call?(n>r._time&&(n=r._time),t=r,r=r._next):(e=r._next,r._next=null,r=t?t._next=e:Pt=e);kt=t,Ut(n)}(),zt=0}}function Wt(){var t=Rt.now(),e=t-Mt;e>At&&(Lt-=e,Mt=t)}function Ut(t){Et||(Tt&&(Tt=clearTimeout(Tt)),t-zt>24?(t<1/0&&(Tt=setTimeout(Gt,t-Rt.now()-Lt)),Ct&&(Ct=clearInterval(Ct))):(Ct||(Mt=Rt.now(),Ct=setInterval(Wt,At)),Et=1,It(Gt)))}function $t(t,e,r){var n=new Bt;return e=null==e?0:+e,n.restart((r=>{n.stop(),t(r+e)}),e,r),n}Bt.prototype=Dt.prototype={constructor:Bt,restart:function(t,e,r){if("function"!=typeof t)throw new TypeError("callback is not a function");r=(null==r?Ft():+r)+(null==e?0:+e),this._next||kt===this||(kt?kt._next=this:Pt=this,kt=this),this._call=t,this._time=r,Ut()},stop:function(){this._call&&(this._call=null,this._time=1/0,Ut())}};var Yt=wt("start","end","cancel","interrupt"),Ht=[],Vt=0,qt=2,Xt=3,Zt=5,Jt=6;function Kt(t,e,r,n,o,i){var a=t.__transition;if(a){if(r in a)return}else t.__transition={};!function(t,e,r){var n,o=t.__transition;function i(c){var l,s,f,h;if(1!==r.state)return u();for(l in o)if((h=o[l]).name===r.name){if(h.state===Xt)return $t(i);4===h.state?(h.state=Jt,h.timer.stop(),h.on.call("interrupt",t,t.__data__,h.index,h.group),delete o[l]):+lVt)throw new Error("too late; already scheduled");return r}function te(t,e){var r=ee(t,e);if(r.state>Xt)throw new Error("too late; already running");return r}function ee(t,e){var r=t.__transition;if(!r||!(r=r[e]))throw new Error("transition not found");return r}function re(t,e){var r,n,o,i=t.__transition,a=!0;if(i){for(o in e=null==e?null:e+"",i)(r=i[o]).name===e?(n=r.state>qt&&r.state180?e+=360:e-t>180&&(t+=360),i.push({i:r.push(o(r)+"rotate(",null,n)-2,x:ne(t,e)})):e&&r.push(o(r)+"rotate("+e+n)}(i.rotate,a.rotate,u,c),function(t,e,r,i){t!==e?i.push({i:r.push(o(r)+"skewX(",null,n)-2,x:ne(t,e)}):e&&r.push(o(r)+"skewX("+e+n)}(i.skewX,a.skewX,u,c),function(t,e,r,n,i,a){if(t!==r||e!==n){var u=i.push(o(i)+"scale(",null,",",null,")");a.push({i:u-4,x:ne(t,r)},{i:u-2,x:ne(e,n)})}else 1===r&&1===n||i.push(o(i)+"scale("+r+","+n+")")}(i.scaleX,i.scaleY,a.scaleX,a.scaleY,u,c),i=a=null,function(t){for(var e,r=-1,n=c.length;++r>8&15|e>>4&240,e>>4&15|240&e,(15&e)<<4|15&e,1):8===r?Le(e>>24&255,e>>16&255,e>>8&255,(255&e)/255):4===r?Le(e>>12&15|e>>8&240,e>>8&15|e>>4&240,e>>4&15|240&e,((15&e)<<4|15&e)/255):null):(e=Oe.exec(t))?new Fe(e[1],e[2],e[3],1):(e=_e.exec(t))?new Fe(255*e[1]/100,255*e[2]/100,255*e[3]/100,1):(e=je.exec(t))?Le(e[1],e[2],e[3],e[4]):(e=Pe.exec(t))?Le(255*e[1]/100,255*e[2]/100,255*e[3]/100,e[4]):(e=ke.exec(t))?Ue(e[1],e[2]/100,e[3]/100,1):(e=Ee.exec(t))?Ue(e[1],e[2]/100,e[3]/100,e[4]):Te.hasOwnProperty(t)?ze(Te[t]):"transparent"===t?new Fe(NaN,NaN,NaN,0):null}function ze(t){return new Fe(t>>16&255,t>>8&255,255&t,1)}function Le(t,e,r,n){return n<=0&&(t=e=r=NaN),new Fe(t,e,r,n)}function Re(t){return t instanceof de||(t=Me(t)),t?new Fe((t=t.rgb()).r,t.g,t.b,t.opacity):new Fe}function Ie(t,e,r,n){return 1===arguments.length?Re(t):new Fe(t,e,r,null==n?1:n)}function Fe(t,e,r,n){this.r=+t,this.g=+e,this.b=+r,this.opacity=+n}function Ne(){return`#${We(this.r)}${We(this.g)}${We(this.b)}`}function Be(){const t=De(this.opacity);return`${1===t?"rgb(":"rgba("}${Ge(this.r)}, ${Ge(this.g)}, ${Ge(this.b)}${1===t?")":`, ${t})`}`}function De(t){return isNaN(t)?1:Math.max(0,Math.min(1,t))}function Ge(t){return Math.max(0,Math.min(255,Math.round(t)||0))}function We(t){return((t=Ge(t))<16?"0":"")+t.toString(16)}function Ue(t,e,r,n){return n<=0?t=e=r=NaN:r<=0||r>=1?t=e=NaN:e<=0&&(t=NaN),new Ye(t,e,r,n)}function $e(t){if(t instanceof Ye)return new Ye(t.h,t.s,t.l,t.opacity);if(t instanceof de||(t=Me(t)),!t)return new Ye;if(t instanceof Ye)return t;var e=(t=t.rgb()).r/255,r=t.g/255,n=t.b/255,o=Math.min(e,r,n),i=Math.max(e,r,n),a=NaN,u=i-o,c=(i+o)/2;return u?(a=e===i?(r-n)/u+6*(r0&&c<1?0:a,new Ye(a,u,c,t.opacity)}function Ye(t,e,r,n){this.h=+t,this.s=+e,this.l=+r,this.opacity=+n}function He(t){return(t=(t||0)%360)<0?t+360:t}function Ve(t){return Math.max(0,Math.min(1,t||0))}function qe(t,e,r){return 255*(t<60?e+(r-e)*t/60:t<180?r:t<240?e+(r-e)*(240-t)/60:e)}function Xe(t,e,r,n,o){var i=t*t,a=i*t;return((1-3*t+3*i-a)*e+(4-6*i+3*a)*r+(1+3*t+3*i-3*a)*n+a*o)/6}pe(de,Me,{copy(t){return Object.assign(new this.constructor,this,t)},displayable(){return this.rgb().displayable()},hex:Ce,formatHex:Ce,formatHex8:function(){return this.rgb().formatHex8()},formatHsl:function(){return $e(this).formatHsl()},formatRgb:Ae,toString:Ae}),pe(Fe,Ie,ve(de,{brighter(t){return t=null==t?me:Math.pow(me,t),new Fe(this.r*t,this.g*t,this.b*t,this.opacity)},darker(t){return t=null==t?be:Math.pow(be,t),new Fe(this.r*t,this.g*t,this.b*t,this.opacity)},rgb(){return this},clamp(){return new Fe(Ge(this.r),Ge(this.g),Ge(this.b),De(this.opacity))},displayable(){return-.5<=this.r&&this.r<255.5&&-.5<=this.g&&this.g<255.5&&-.5<=this.b&&this.b<255.5&&0<=this.opacity&&this.opacity<=1},hex:Ne,formatHex:Ne,formatHex8:function(){return`#${We(this.r)}${We(this.g)}${We(this.b)}${We(255*(isNaN(this.opacity)?1:this.opacity))}`},formatRgb:Be,toString:Be})),pe(Ye,(function(t,e,r,n){return 1===arguments.length?$e(t):new Ye(t,e,r,null==n?1:n)}),ve(de,{brighter(t){return t=null==t?me:Math.pow(me,t),new Ye(this.h,this.s,this.l*t,this.opacity)},darker(t){return t=null==t?be:Math.pow(be,t),new Ye(this.h,this.s,this.l*t,this.opacity)},rgb(){var t=this.h%360+360*(this.h<0),e=isNaN(t)||isNaN(this.s)?0:this.s,r=this.l,n=r+(r<.5?r:1-r)*e,o=2*r-n;return new Fe(qe(t>=240?t-240:t+120,o,n),qe(t,o,n),qe(t<120?t+240:t-120,o,n),this.opacity)},clamp(){return new Ye(He(this.h),Ve(this.s),Ve(this.l),De(this.opacity))},displayable(){return(0<=this.s&&this.s<=1||isNaN(this.s))&&0<=this.l&&this.l<=1&&0<=this.opacity&&this.opacity<=1},formatHsl(){const t=De(this.opacity);return`${1===t?"hsl(":"hsla("}${He(this.h)}, ${100*Ve(this.s)}%, ${100*Ve(this.l)}%${1===t?")":`, ${t})`}`}}));const Ze=t=>()=>t;function Je(t,e){var r=e-t;return r?function(t,e){return function(r){return t+r*e}}(t,r):Ze(isNaN(t)?e:t)}const Ke=function t(e){var r=function(t){return 1==(t=+t)?Je:function(e,r){return r-e?function(t,e,r){return t=Math.pow(t,r),e=Math.pow(e,r)-t,r=1/r,function(n){return Math.pow(t+n*e,r)}}(e,r,t):Ze(isNaN(e)?r:e)}}(e);function n(t,e){var n=r((t=Ie(t)).r,(e=Ie(e)).r),o=r(t.g,e.g),i=r(t.b,e.b),a=Je(t.opacity,e.opacity);return function(e){return t.r=n(e),t.g=o(e),t.b=i(e),t.opacity=a(e),t+""}}return n.gamma=t,n}(1);function Qe(t){return function(e){var r,n,o=e.length,i=new Array(o),a=new Array(o),u=new Array(o);for(r=0;r=1?(r=1,e-1):Math.floor(r*e),o=t[n],i=t[n+1],a=n>0?t[n-1]:2*o-i,u=ni&&(o=e.slice(i,o),u[a]?u[a]+=o:u[++a]=o),(r=r[0])===(n=n[0])?u[a]?u[a]+=n:u[++a]=n:(u[++a]=null,c.push({i:a,x:ne(r,n)})),i=er.lastIndex;return i=0&&(t=t.slice(0,e)),!t||"start"===t}))}(e)?Qt:te;return function(){var a=i(this,t),u=a.on;u!==n&&(o=(n=u).copy()).on(e,r),a.on=o}}(r,t,e))},attr:function(t,e){var r=x(t),n="transform"===r?se:nr;return this.attrTween(t,"function"==typeof e?(r.local?lr:cr)(r,n,ye(this,"attr."+t,e)):null==e?(r.local?ir:or)(r):(r.local?ur:ar)(r,n,e))},attrTween:function(t,e){var r="attr."+t;if(arguments.length<2)return(r=this.tween(r))&&r._value;if(null==e)return this.tween(r,null);if("function"!=typeof e)throw new Error;var n=x(t);return this.tween(r,(n.local?sr:fr)(n,e))},style:function(t,e,r){var n="transform"==(t+="")?le:nr;return null==e?this.styleTween(t,function(t,e){var r,n,o;return function(){var i=M(this,t),a=(this.style.removeProperty(t),M(this,t));return i===a?null:i===r&&a===n?o:o=e(r=i,n=a)}}(t,n)).on("end.style."+t,br(t)):"function"==typeof e?this.styleTween(t,function(t,e,r){var n,o,i;return function(){var a=M(this,t),u=r(this),c=u+"";return null==u&&(this.style.removeProperty(t),c=u=M(this,t)),a===c?null:a===n&&c===o?i:(o=c,i=e(n=a,u))}}(t,n,ye(this,"style."+t,e))).each(function(t,e){var r,n,o,i,a="style."+e,u="end."+a;return function(){var c=te(this,t),l=c.on,s=null==c.value[a]?i||(i=br(e)):void 0;l===r&&o===s||(n=(r=l).copy()).on(u,o=s),c.on=n}}(this._id,t)):this.styleTween(t,function(t,e,r){var n,o,i=r+"";return function(){var a=M(this,t);return a===i?null:a===n?o:o=e(n=a,r)}}(t,n,e),r).on("end.style."+t,null)},styleTween:function(t,e,r){var n="style."+(t+="");if(arguments.length<2)return(n=this.tween(n))&&n._value;if(null==e)return this.tween(n,null);if("function"!=typeof e)throw new Error;return this.tween(n,function(t,e,r){var n,o;function i(){var i=e.apply(this,arguments);return i!==o&&(n=(o=i)&&function(t,e,r){return function(n){this.style.setProperty(t,e.call(this,n),r)}}(t,i,r)),n}return i._value=e,i}(t,e,null==r?"":r))},text:function(t){return this.tween("text","function"==typeof t?function(t){return function(){var e=t(this);this.textContent=null==e?"":e}}(ye(this,"text",t)):function(t){return function(){this.textContent=t}}(null==t?"":t+""))},textTween:function(t){var e="text";if(arguments.length<1)return(e=this.tween(e))&&e._value;if(null==t)return this.tween(e,null);if("function"!=typeof t)throw new Error;return this.tween(e,function(t){var e,r;function n(){var n=t.apply(this,arguments);return n!==r&&(e=(r=n)&&function(t){return function(e){this.textContent=t.call(this,e)}}(n)),e}return n._value=t,n}(t))},remove:function(){return this.on("end.remove",function(t){return function(){var e=this.parentNode;for(var r in this.__transition)if(+r!==t)return;e&&e.removeChild(this)}}(this._id))},tween:function(t,e){var r=this._id;if(t+="",arguments.length<2){for(var n,o=ee(this.node(),r).tween,i=0,a=o.length;i()=>t;function jr(t,{sourceEvent:e,target:r,transform:n,dispatch:o}){Object.defineProperties(this,{type:{value:t,enumerable:!0,configurable:!0},sourceEvent:{value:e,enumerable:!0,configurable:!0},target:{value:r,enumerable:!0,configurable:!0},transform:{value:n,enumerable:!0,configurable:!0},_:{value:o}})}function Pr(t,e,r){this.k=t,this.x=e,this.y=r}Pr.prototype={constructor:Pr,scale:function(t){return 1===t?this:new Pr(this.k*t,this.x,this.y)},translate:function(t,e){return 0===t&0===e?this:new Pr(this.k,this.x+this.k*t,this.y+this.k*e)},apply:function(t){return[t[0]*this.k+this.x,t[1]*this.k+this.y]},applyX:function(t){return t*this.k+this.x},applyY:function(t){return t*this.k+this.y},invert:function(t){return[(t[0]-this.x)/this.k,(t[1]-this.y)/this.k]},invertX:function(t){return(t-this.x)/this.k},invertY:function(t){return(t-this.y)/this.k},rescaleX:function(t){return t.copy().domain(t.range().map(this.invertX,this).map(t.invert,t))},rescaleY:function(t){return t.copy().domain(t.range().map(this.invertY,this).map(t.invert,t))},toString:function(){return"translate("+this.x+","+this.y+") scale("+this.k+")"}};var kr=new Pr(1,0,0);function Er(t){t.stopImmediatePropagation()}function Tr(t){t.preventDefault(),t.stopImmediatePropagation()}function Cr(t){return!(t.ctrlKey&&"wheel"!==t.type||t.button)}function Ar(){var t=this;return t instanceof SVGElement?(t=t.ownerSVGElement||t).hasAttribute("viewBox")?[[(t=t.viewBox.baseVal).x,t.y],[t.x+t.width,t.y+t.height]]:[[0,0],[t.width.baseVal.value,t.height.baseVal.value]]:[[0,0],[t.clientWidth,t.clientHeight]]}function Mr(){return this.__zoom||kr}function zr(t){return-t.deltaY*(1===t.deltaMode?.05:t.deltaMode?1:.002)*(t.ctrlKey?10:1)}function Lr(){return navigator.maxTouchPoints||"ontouchstart"in this}function Rr(t,e,r){var n=t.invertX(e[0][0])-r[0][0],o=t.invertX(e[1][0])-r[1][0],i=t.invertY(e[0][1])-r[0][1],a=t.invertY(e[1][1])-r[1][1];return t.translate(o>n?(n+o)/2:Math.min(0,n)||Math.max(0,o),a>i?(i+a)/2:Math.min(0,i)||Math.max(0,a))}function Ir(t){return Ir="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(t){return typeof t}:function(t){return t&&"function"==typeof Symbol&&t.constructor===Symbol&&t!==Symbol.prototype?"symbol":typeof t},Ir(t)}function Fr(t,e){(null==e||e>t.length)&&(e=t.length);for(var r=0,n=Array(e);r1&&void 0!==arguments[1]?arguments[1]:0,o=arguments.length>2&&void 0!==arguments[2]?arguments[2]:0,i=arguments.length>3&&void 0!==arguments[3]?arguments[3]:0,a=arguments.length>4&&void 0!==arguments[4]?arguments[4]:void 0;!function(t,e){if(!(t instanceof e))throw new TypeError("Cannot call a class as a function")}(this,t),this.opts=a||{},this.canvas=e,this.w=this.canvas.offsetWidth,this.h=this.canvas.offsetHeight,this.offscreenCanvas=document.createElement("canvas"),this.offscreenCanvas.width=this.w,this.offscreenCanvas.height=this.h;var u=this.canvas.getContext("2d"),c=this.offscreenCanvas.getContext("2d");if(!u)throw"Impossible to create canvas 2D context";if(!c)throw"Impossible to create canvas 2D context";if(this.ctx=u,this.offscreenCtx=c,this.view={x:n,y:o,z:i},this.backgroundColor=a.backgroundColor||"white",this.onZoomStartFun=a.onZoomStartFun,this.onZoomEndFun=a.onZoomEndFun,this.onZoomFun=a.onZoomFun,this.extGeo={xMin:NaN,xMax:NaN,yMin:NaN,yMax:NaN},this.updateExtentGeo(),!a.disableZoom){var l=kr;(function(){var t,e,r,n=Cr,o=Ar,i=Rr,a=zr,u=Lr,c=[0,1/0],l=[[-1/0,-1/0],[1/0,1/0]],s=250,f=_t,h=wt("start","zoom","end"),y=500,p=150,v=0,d=10;function b(t){t.property("__zoom",Mr).on("wheel.zoom",_,{passive:!1}).on("mousedown.zoom",j).on("dblclick.zoom",P).filter(u).on("touchstart.zoom",k).on("touchmove.zoom",E).on("touchend.zoom touchcancel.zoom",T).style("-webkit-tap-highlight-color","rgba(0,0,0,0)")}function m(t,e){return(e=Math.max(c[0],Math.min(c[1],e)))===t.k?t:new Pr(e,t.x,t.y)}function g(t,e,r){var n=e[0]-r[0]*t.k,o=e[1]-r[1]*t.k;return n===t.x&&o===t.y?t:new Pr(t.k,n,o)}function w(t){return[(+t[0][0]+ +t[1][0])/2,(+t[0][1]+ +t[1][1])/2]}function x(t,e,r,n){t.on("start.zoom",(function(){S(this,arguments).event(n).start()})).on("interrupt.zoom end.zoom",(function(){S(this,arguments).event(n).end()})).tween("zoom",(function(){var t=this,i=arguments,a=S(t,i).event(n),u=o.apply(t,i),c=null==r?w(u):"function"==typeof r?r.apply(t,i):r,l=Math.max(u[1][0]-u[0][0],u[1][1]-u[0][1]),s=t.__zoom,h="function"==typeof e?e.apply(t,i):e,y=f(s.invert(c).concat(l/s.k),h.invert(c).concat(l/h.k));return function(t){if(1===t)t=h;else{var e=y(t),r=l/e[2];t=new Pr(r,c[0]-e[0]*r,c[1]-e[1]*r)}a.zoom(null,t)}}))}function S(t,e,r){return!r&&t.__zooming||new O(t,e)}function O(t,e){this.that=t,this.args=e,this.active=0,this.sourceEvent=null,this.extent=o.apply(t,e),this.taps=0}function _(t,...e){if(n.apply(this,arguments)){var r=S(this,e).event(t),o=this.__zoom,u=Math.max(c[0],Math.min(c[1],o.k*Math.pow(2,a.apply(this,arguments)))),s=jt(t);if(r.wheel)r.mouse[0][0]===s[0]&&r.mouse[0][1]===s[1]||(r.mouse[1]=o.invert(r.mouse[0]=s)),clearTimeout(r.wheel);else{if(o.k===u)return;r.mouse=[s,o.invert(s)],re(this),r.start()}Tr(t),r.wheel=setTimeout((function(){r.wheel=null,r.end()}),p),r.zoom("mouse",i(g(m(o,u),r.mouse[0],r.mouse[1]),r.extent,l))}}function j(t,...e){if(!r&&n.apply(this,arguments)){var o=t.currentTarget,a=S(this,e,!0).event(t),u=pt(t.view).on("mousemove.zoom",(function(t){if(Tr(t),!a.moved){var e=t.clientX-s,r=t.clientY-f;a.moved=e*e+r*r>v}a.event(t).zoom("mouse",i(g(a.that.__zoom,a.mouse[0]=jt(t,o),a.mouse[1]),a.extent,l))}),!0).on("mouseup.zoom",(function(t){u.on("mousemove.zoom mouseup.zoom",null),function(t,e){var r=t.document.documentElement,n=pt(t).on("dragstart.drag",null);e&&(n.on("click.drag",St,xt),setTimeout((function(){n.on("click.drag",null)}),0)),"onselectstart"in r?n.on("selectstart.drag",null):(r.style.MozUserSelect=r.__noselect,delete r.__noselect)}(t.view,a.moved),Tr(t),a.event(t).end()}),!0),c=jt(t,o),s=t.clientX,f=t.clientY;!function(t){var e=t.document.documentElement,r=pt(t).on("dragstart.drag",St,xt);"onselectstart"in e?r.on("selectstart.drag",St,xt):(e.__noselect=e.style.MozUserSelect,e.style.MozUserSelect="none")}(t.view),Er(t),a.mouse=[c,this.__zoom.invert(c)],re(this),a.start()}}function P(t,...e){if(n.apply(this,arguments)){var r=this.__zoom,a=jt(t.changedTouches?t.changedTouches[0]:t,this),u=r.invert(a),c=r.k*(t.shiftKey?.5:2),f=i(g(m(r,c),a,u),o.apply(this,e),l);Tr(t),s>0?pt(this).transition().duration(s).call(x,f,a,t):pt(this).call(b.transform,f,a,t)}}function k(r,...o){if(n.apply(this,arguments)){var i,a,u,c,l=r.touches,s=l.length,f=S(this,o,r.changedTouches.length===s).event(r);for(Er(r),a=0;a=t.length?{done:!0}:{done:!1,value:t[n++]}},e:function(t){throw t},f:o}}throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}var i,a=!0,u=!1;return{s:function(){r=r.call(t)},n:function(){var t=r.next();return a=t.done,t},e:function(t){u=!0,i=t},f:function(){try{a||null==r.return||r.return()}finally{if(u)throw i}}}}(s.targetTouches);try{for(p.s();!(f=p.n()).done;){var v=f.value;h+=v.clientX,y+=v.clientY}}catch(t){p.e(t)}finally{p.f()}h/=s.targetTouches.length,y/=s.targetTouches.length,h-=o.left,y-=o.top,r.zoom(n,r.pixToGeoX(h),r.pixToGeoY(y))}}l=e,r.onZoomFun&&r.onZoomFun(t)})).on("start",(function(t){var e;r.canvasSave.c=document.createElement("canvas"),r.canvasSave.c.setAttribute("width",""+r.w),r.canvasSave.c.setAttribute("height",""+r.h),null===(e=r.canvasSave.c.getContext("2d"))||void 0===e||e.drawImage(r.canvas,0,0),r.canvasSave.dx=0,r.canvasSave.dy=0,r.canvasSave.f=1,r.onZoomStartFun&&r.onZoomStartFun(t)})).on("end",(function(t){r.redraw(),r.canvasSave={c:null,dx:0,dy:0,f:1},r.onZoomEndFun&&r.onZoomEndFun(t)}))(pt(this.canvas))}this.xMin=a.centerExtent?a.centerExtent[0]:void 0,this.yMin=a.centerExtent?a.centerExtent[1]:void 0,this.xMax=a.centerExtent?a.centerExtent[2]:void 0,this.yMax=a.centerExtent?a.centerExtent[3]:void 0,this.zoomExtent=a.zoomExtent||[0,1/0],this.canvasSave={c:null,dx:0,dy:0,f:1}}return e=t,r=[{key:"getView",value:function(){return this.view}},{key:"setCenterExtent",value:function(t){this.xMin=t[0],this.yMin=t[1],this.xMax=t[2],this.yMax=t[3]}},{key:"getCenterExtent",value:function(){return[this.xMin,this.yMin,this.xMax,this.yMax]}},{key:"setZoomExtent",value:function(t){this.zoomExtent=t}},{key:"getZoomExtent",value:function(){return this.zoomExtent}},{key:"initCanvasTransform",value:function(){this.ctx.setTransform(1,0,0,1,0,0),this.offscreenCtx.setTransform(1,0,0,1,0,0)}},{key:"setCanvasTransform",value:function(){var t=1/this.view.z,e=-this.view.x/this.view.z+.5*this.w,r=this.view.y/this.view.z+.5*this.h;this.ctx.setTransform(t,0,0,-t,e,r),this.offscreenCtx.setTransform(t,0,0,-t,e,r)}},{key:"getWebGLTransform",value:function(){var t=2/(this.w*this.view.z),e=2/(this.h*this.view.z);return[t,0,0,0,e,0,-t*this.view.x,-e*this.view.y,1]}},{key:"redraw",value:function(){throw new Error("Method redraw not implemented.")}},{key:"clear",value:function(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:"white";this.opts.transparentBackground?(this.ctx.clearRect(0,0,this.w,this.h),this.offscreenCtx.clearRect(0,0,this.w,this.h)):(this.ctx&&(this.ctx.fillStyle=t),this.offscreenCtx&&(this.offscreenCtx.fillStyle=t),this.ctx.fillRect(0,0,this.w,this.h),this.offscreenCtx.fillRect(0,0,this.w,this.h))}},{key:"pan",value:function(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:0,e=arguments.length>1&&void 0!==arguments[1]?arguments[1]:0;if(null!=this.xMin&&this.view.x+tthis.xMax&&(t=this.xMax-this.view.x),null!=this.yMax&&this.view.y+e>this.yMax&&(e=this.yMax-this.view.y),this.view.x+=t,this.view.y+=e,this.updateExtentGeo(),this.canvasSave.c){var r=1/this.view.z;this.canvasSave.dx-=t*r,this.canvasSave.dy+=e*r,this.clear(this.backgroundColor),this.offscreenCtx.drawImage(this.canvasSave.c,this.canvasSave.dx,this.canvasSave.dy),this.ctx.drawImage(this.offscreenCtx.canvas,0,0)}else console.log("no canvas save")}},{key:"zoom",value:function(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:1,e=arguments.length>1&&void 0!==arguments[1]?arguments[1]:this.view.x,r=arguments.length>2&&void 0!==arguments[2]?arguments[2]:this.view.y;if(!(this.zoomExtent[0]==this.view.z&&t<=1||this.zoomExtent[1]==this.view.z&&t>=1)){var n=t*this.view.z;nthis.zoomExtent[1]&&(t=this.zoomExtent[1]/this.view.z),this.view.z*=t;var o=(e-this.view.x)*(1-t),i=(r-this.view.y)*(1-t);null!=this.xMin&&this.view.x+othis.xMax&&(o=this.xMax-this.view.x),null!=this.yMax&&this.view.y+i>this.yMax&&(i=this.yMax-this.view.y),this.view.x+=o,this.view.y+=i,this.updateExtentGeo(),this.canvasSave.c&&(this.clear(this.backgroundColor),this.canvasSave.f/=t,this.canvasSave.dx=this.geoToPixX(e)*(1-this.canvasSave.f),this.canvasSave.dy=this.geoToPixY(r)*(1-this.canvasSave.f),this.offscreenCtx.drawImage(this.canvasSave.c,this.canvasSave.dx,this.canvasSave.dy,this.canvasSave.f*this.canvasSave.c.width,this.canvasSave.f*this.canvasSave.c.height),this.ctx.drawImage(this.offscreenCanvas,0,0,this.canvas.width,this.canvas.height))}}},{key:"updateExtentGeo",value:function(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:20;return this.extGeo={xMin:this.pixToGeoX(-t),xMax:this.pixToGeoX(this.w+t),yMin:this.pixToGeoY(this.h+t),yMax:this.pixToGeoY(-t)},this.extGeo}},{key:"toDraw",value:function(t){return!(t.xthis.extGeo.xMax||t.ythis.extGeo.yMax)}},{key:"geoToPixX",value:function(t){return(t-this.view.x)/this.view.z+.5*this.w}},{key:"geoToPixY",value:function(t){return-(t-this.view.y)/this.view.z+.5*this.h}},{key:"pixToGeoX",value:function(t){return(t-.5*this.w)*this.view.z+this.view.x}},{key:"pixToGeoY",value:function(t){return-(t-.5*this.h)*this.view.z+this.view.y}},{key:"setViewFromURL",value:function(){var e=t.getParameterByName("x"),r=t.getParameterByName("y"),n=t.getParameterByName("z");null==e||null==e||isNaN(+e)||(this.view.x=+e),null==r||null==r||isNaN(+r)||(this.view.y=+r),null==n||null==n||isNaN(+n)||(this.view.z=+n)}}],n=[{key:"getParameterByName",value:function(t){t=t.replace(/[\[]/,"\\[").replace(/[\]]/,"\\]");var e=new RegExp("[\\?&]"+t+"=([^&#]*)").exec(location.search);return e?decodeURIComponent(e[1].replace(/\+/g," ")):null}}],r&&Nr(e.prototype,r),n&&Nr(e,n),Object.defineProperty(e,"prototype",{writable:!1}),e;var e,r,n}();function Gr(t){return Gr="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(t){return typeof t}:function(t){return t&&"function"==typeof Symbol&&t.constructor===Symbol&&t!==Symbol.prototype?"symbol":typeof t},Gr(t)}function Wr(t,e){for(var r=0;re.left+n-r.clientWidth){var i=t.x-r.clientWidth-this.xOffset;if(r.style.left=i+"px",r.offsetLeft+r.clientWidth>t.x){var a=t.x-r.clientWidth-this.xOffset;r.style.left=a+"px"}}r.offsetTop+r.clientHeight>e.top+o&&(r.style.top=r.offsetTop-r.clientHeight+"px"),r.offsetTop0&&void 0!==arguments[0]?arguments[0]:{};!function(t,e){if(!(t instanceof e))throw new TypeError("Cannot call a class as a function")}(this,t),this.map=e.map,this.parentNode=e.parentNode||e.map.container,e.id&&(this.div=pt("#"+e.id)),this.div&&!this.div.empty()||(this.div=pt(document.createElement("div")),e.id&&this.div.attr("id",e.id)),e.title&&this.div.attr("title",e.title),e.class&&this.div.attr("class",e.class),e.onClickFunction&&this.div.on("click",e.onClickFunction),this.style("box-shadow","0 7px 8px rgba(0,47,103,.08), 0 0 22px rgba(0,47,103,.04), 0 12px 17px rgba(0,47,103,.04), 0 -4px 4px rgba(0,47,103,.04)"),this.style("background-color","#ffffff"),this.style("position","absolute"),this.style("cursor","pointer"),this.style("display","flex"),this.style("justify-content","center"),this.style("align-items","center"),this.style("width","35px"),this.style("height","30px"),this.parentNode.appendChild(this.div.node())},(e=[{key:"style",value:function(t,e){return this.div.style(t,e),this}}])&&Hr(t.prototype,e),Object.defineProperty(t,"prototype",{writable:!1}),t;var t,e}();function Xr(t){return Xr="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(t){return typeof t}:function(t){return t&&"function"==typeof Symbol&&t.constructor===Symbol&&t!==Symbol.prototype?"symbol":typeof t},Xr(t)}function Zr(t,e){for(var r=0;r