From 9fb0cf72b042944bcad9a26a04e221e80773a9ba Mon Sep 17 00:00:00 2001 From: pvictor Date: Wed, 11 Oct 2023 12:07:04 +0200 Subject: [PATCH] updated uPlot to 1.6.26 --- DESCRIPTION | 4 +- inst/htmlwidgets/uPlot.js | 6441 +------------------------------------ man/uPlot.Rd | 9 +- package-lock.json | 14 +- package.json | 2 +- 5 files changed, 19 insertions(+), 6451 deletions(-) diff --git a/DESCRIPTION b/DESCRIPTION index ecd7ca1..d3c84f9 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -1,6 +1,6 @@ Package: uPlot Title: Fast Interactive Charts with 'µPlot' 'JavaScript' Library -Version: 0.0.0.9100 +Version: 0.0.0.9200 Authors@R: c(person(given = "Victor", family = "Perrier", @@ -19,7 +19,7 @@ Description: An interface for 'µPlot' 'JavaScript' library in R. License: MIT + file LICENSE Encoding: UTF-8 Roxygen: list(markdown = TRUE) -RoxygenNote: 7.2.1 +RoxygenNote: 7.2.3 Imports: htmlwidgets Depends: diff --git a/inst/htmlwidgets/uPlot.js b/inst/htmlwidgets/uPlot.js index 68225d1..5866f78 100644 --- a/inst/htmlwidgets/uPlot.js +++ b/inst/htmlwidgets/uPlot.js @@ -1,6440 +1 @@ -/******/ (() => { // webpackBootstrap -/******/ "use strict"; -/******/ var __webpack_modules__ = ({ - -/***/ "./node_modules/css-loader/dist/cjs.js!./node_modules/uplot/dist/uPlot.min.css": -/*!*************************************************************************************!*\ - !*** ./node_modules/css-loader/dist/cjs.js!./node_modules/uplot/dist/uPlot.min.css ***! - \*************************************************************************************/ -/***/ ((module, __webpack_exports__, __webpack_require__) => { - -__webpack_require__.r(__webpack_exports__); -/* harmony export */ __webpack_require__.d(__webpack_exports__, { -/* harmony export */ "default": () => (__WEBPACK_DEFAULT_EXPORT__) -/* harmony export */ }); -/* harmony import */ var _css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../css-loader/dist/runtime/sourceMaps.js */ "./node_modules/css-loader/dist/runtime/sourceMaps.js"); -/* harmony import */ var _css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); -/* harmony import */ var _css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../css-loader/dist/runtime/api.js */ "./node_modules/css-loader/dist/runtime/api.js"); -/* harmony import */ var _css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); -// Imports - - -var ___CSS_LOADER_EXPORT___ = _css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); -// Module -___CSS_LOADER_EXPORT___.push([module.id, ".uplot, .uplot *, .uplot *::before, .uplot *::after {box-sizing: border-box;}.uplot {font-family: system-ui, -apple-system, \"Segoe UI\", Roboto, \"Helvetica Neue\", Arial, \"Noto Sans\", sans-serif, \"Apple Color Emoji\", \"Segoe UI Emoji\", \"Segoe UI Symbol\", \"Noto Color Emoji\";line-height: 1.5;width: min-content;}.u-title {text-align: center;font-size: 18px;font-weight: bold;}.u-wrap {position: relative;user-select: none;}.u-over, .u-under {position: absolute;}.u-under {overflow: hidden;}.uplot canvas {display: block;position: relative;width: 100%;height: 100%;}.u-axis {position: absolute;}.u-legend {font-size: 14px;margin: auto;text-align: center;}.u-inline {display: block;}.u-inline * {display: inline-block;}.u-inline tr {margin-right: 16px;}.u-legend th {font-weight: 600;}.u-legend th > * {vertical-align: middle;display: inline-block;}.u-legend .u-marker {width: 1em;height: 1em;margin-right: 4px;background-clip: padding-box !important;}.u-inline.u-live th::after {content: \":\";vertical-align: middle;}.u-inline:not(.u-live) .u-value {display: none;}.u-series > * {padding: 4px;}.u-series th {cursor: pointer;}.u-legend .u-off > * {opacity: 0.3;}.u-select {background: rgba(0,0,0,0.07);position: absolute;pointer-events: none;}.u-cursor-x, .u-cursor-y {position: absolute;left: 0;top: 0;pointer-events: none;will-change: transform;z-index: 100;}.u-hz .u-cursor-x, .u-vt .u-cursor-y {height: 100%;border-right: 1px dashed #607D8B;}.u-hz .u-cursor-y, .u-vt .u-cursor-x {width: 100%;border-bottom: 1px dashed #607D8B;}.u-cursor-pt {position: absolute;top: 0;left: 0;border-radius: 50%;border: 0 solid;pointer-events: none;will-change: transform;z-index: 100;/*this has to be !important since we set inline \"background\" shorthand */background-clip: padding-box !important;}.u-axis.u-off, .u-select.u-off, .u-cursor-x.u-off, .u-cursor-y.u-off, .u-cursor-pt.u-off {display: none;}", "",{"version":3,"sources":["webpack://./node_modules/uplot/dist/uPlot.min.css"],"names":[],"mappings":"AAAA,qDAAqD,sBAAsB,CAAC,CAAC,QAAQ,yLAAyL,CAAC,gBAAgB,CAAC,kBAAkB,CAAC,CAAC,UAAU,kBAAkB,CAAC,eAAe,CAAC,iBAAiB,CAAC,CAAC,SAAS,kBAAkB,CAAC,iBAAiB,CAAC,CAAC,mBAAmB,kBAAkB,CAAC,CAAC,UAAU,gBAAgB,CAAC,CAAC,eAAe,cAAc,CAAC,kBAAkB,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC,SAAS,kBAAkB,CAAC,CAAC,WAAW,eAAe,CAAC,YAAY,CAAC,kBAAkB,CAAC,CAAC,WAAW,cAAc,CAAC,CAAC,aAAa,qBAAqB,CAAC,CAAC,cAAc,kBAAkB,CAAC,CAAC,cAAc,gBAAgB,CAAC,CAAC,kBAAkB,sBAAsB,CAAC,qBAAqB,CAAC,CAAC,qBAAqB,UAAU,CAAC,WAAW,CAAC,iBAAiB,CAAC,uCAAuC,CAAC,CAAC,4BAA4B,YAAY,CAAC,sBAAsB,CAAC,CAAC,iCAAiC,aAAa,CAAC,CAAC,eAAe,YAAY,CAAC,CAAC,cAAc,eAAe,CAAC,CAAC,sBAAsB,YAAY,CAAC,CAAC,WAAW,4BAA4B,CAAC,kBAAkB,CAAC,oBAAoB,CAAC,CAAC,0BAA0B,kBAAkB,CAAC,OAAO,CAAC,MAAM,CAAC,oBAAoB,CAAC,sBAAsB,CAAC,YAAY,CAAC,CAAC,sCAAsC,YAAY,CAAC,gCAAgC,CAAC,CAAC,sCAAsC,WAAW,CAAC,iCAAiC,CAAC,CAAC,cAAc,kBAAkB,CAAC,MAAM,CAAC,OAAO,CAAC,kBAAkB,CAAC,eAAe,CAAC,oBAAoB,CAAC,sBAAsB,CAAC,YAAY,CAAC,wEAAwE,CAAC,uCAAuC,CAAC,CAAC,0FAA0F,aAAa,CAAC","sourcesContent":[".uplot, .uplot *, .uplot *::before, .uplot *::after {box-sizing: border-box;}.uplot {font-family: system-ui, -apple-system, \"Segoe UI\", Roboto, \"Helvetica Neue\", Arial, \"Noto Sans\", sans-serif, \"Apple Color Emoji\", \"Segoe UI Emoji\", \"Segoe UI Symbol\", \"Noto Color Emoji\";line-height: 1.5;width: min-content;}.u-title {text-align: center;font-size: 18px;font-weight: bold;}.u-wrap {position: relative;user-select: none;}.u-over, .u-under {position: absolute;}.u-under {overflow: hidden;}.uplot canvas {display: block;position: relative;width: 100%;height: 100%;}.u-axis {position: absolute;}.u-legend {font-size: 14px;margin: auto;text-align: center;}.u-inline {display: block;}.u-inline * {display: inline-block;}.u-inline tr {margin-right: 16px;}.u-legend th {font-weight: 600;}.u-legend th > * {vertical-align: middle;display: inline-block;}.u-legend .u-marker {width: 1em;height: 1em;margin-right: 4px;background-clip: padding-box !important;}.u-inline.u-live th::after {content: \":\";vertical-align: middle;}.u-inline:not(.u-live) .u-value {display: none;}.u-series > * {padding: 4px;}.u-series th {cursor: pointer;}.u-legend .u-off > * {opacity: 0.3;}.u-select {background: rgba(0,0,0,0.07);position: absolute;pointer-events: none;}.u-cursor-x, .u-cursor-y {position: absolute;left: 0;top: 0;pointer-events: none;will-change: transform;z-index: 100;}.u-hz .u-cursor-x, .u-vt .u-cursor-y {height: 100%;border-right: 1px dashed #607D8B;}.u-hz .u-cursor-y, .u-vt .u-cursor-x {width: 100%;border-bottom: 1px dashed #607D8B;}.u-cursor-pt {position: absolute;top: 0;left: 0;border-radius: 50%;border: 0 solid;pointer-events: none;will-change: transform;z-index: 100;/*this has to be !important since we set inline \"background\" shorthand */background-clip: padding-box !important;}.u-axis.u-off, .u-select.u-off, .u-cursor-x.u-off, .u-cursor-y.u-off, .u-cursor-pt.u-off {display: none;}"],"sourceRoot":""}]); -// Exports -/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); - - -/***/ }), - -/***/ "./node_modules/css-loader/dist/runtime/api.js": -/*!*****************************************************!*\ - !*** ./node_modules/css-loader/dist/runtime/api.js ***! - \*****************************************************/ -/***/ ((module) => { - - - -/* - MIT License http://www.opensource.org/licenses/mit-license.php - Author Tobias Koppers @sokra -*/ -module.exports = function (cssWithMappingToString) { - var list = []; // return the list of modules as css string - - list.toString = function toString() { - return this.map(function (item) { - var content = ""; - var needLayer = typeof item[5] !== "undefined"; - - if (item[4]) { - content += "@supports (".concat(item[4], ") {"); - } - - if (item[2]) { - content += "@media ".concat(item[2], " {"); - } - - if (needLayer) { - content += "@layer".concat(item[5].length > 0 ? " ".concat(item[5]) : "", " {"); - } - - content += cssWithMappingToString(item); - - if (needLayer) { - content += "}"; - } - - if (item[2]) { - content += "}"; - } - - if (item[4]) { - content += "}"; - } - - return content; - }).join(""); - }; // import a list of modules into the list - - - list.i = function i(modules, media, dedupe, supports, layer) { - if (typeof modules === "string") { - modules = [[null, modules, undefined]]; - } - - var alreadyImportedModules = {}; - - if (dedupe) { - for (var k = 0; k < this.length; k++) { - var id = this[k][0]; - - if (id != null) { - alreadyImportedModules[id] = true; - } - } - } - - for (var _k = 0; _k < modules.length; _k++) { - var item = [].concat(modules[_k]); - - if (dedupe && alreadyImportedModules[item[0]]) { - continue; - } - - if (typeof layer !== "undefined") { - if (typeof item[5] === "undefined") { - item[5] = layer; - } else { - item[1] = "@layer".concat(item[5].length > 0 ? " ".concat(item[5]) : "", " {").concat(item[1], "}"); - item[5] = layer; - } - } - - if (media) { - if (!item[2]) { - item[2] = media; - } else { - item[1] = "@media ".concat(item[2], " {").concat(item[1], "}"); - item[2] = media; - } - } - - if (supports) { - if (!item[4]) { - item[4] = "".concat(supports); - } else { - item[1] = "@supports (".concat(item[4], ") {").concat(item[1], "}"); - item[4] = supports; - } - } - - list.push(item); - } - }; - - return list; -}; - -/***/ }), - -/***/ "./node_modules/css-loader/dist/runtime/sourceMaps.js": -/*!************************************************************!*\ - !*** ./node_modules/css-loader/dist/runtime/sourceMaps.js ***! - \************************************************************/ -/***/ ((module) => { - - - -module.exports = function (item) { - var content = item[1]; - var cssMapping = item[3]; - - if (!cssMapping) { - return content; - } - - if (typeof btoa === "function") { - var base64 = btoa(unescape(encodeURIComponent(JSON.stringify(cssMapping)))); - var data = "sourceMappingURL=data:application/json;charset=utf-8;base64,".concat(base64); - var sourceMapping = "/*# ".concat(data, " */"); - var sourceURLs = cssMapping.sources.map(function (source) { - return "/*# sourceURL=".concat(cssMapping.sourceRoot || "").concat(source, " */"); - }); - return [content].concat(sourceURLs).concat([sourceMapping]).join("\n"); - } - - return [content].join("\n"); -}; - -/***/ }), - -/***/ "./node_modules/uplot/dist/uPlot.min.css": -/*!***********************************************!*\ - !*** ./node_modules/uplot/dist/uPlot.min.css ***! - \***********************************************/ -/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { - -__webpack_require__.r(__webpack_exports__); -/* harmony export */ __webpack_require__.d(__webpack_exports__, { -/* harmony export */ "default": () => (__WEBPACK_DEFAULT_EXPORT__) -/* harmony export */ }); -/* harmony import */ var _style_loader_dist_runtime_injectStylesIntoStyleTag_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! !../../style-loader/dist/runtime/injectStylesIntoStyleTag.js */ "./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js"); -/* harmony import */ var _style_loader_dist_runtime_injectStylesIntoStyleTag_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_style_loader_dist_runtime_injectStylesIntoStyleTag_js__WEBPACK_IMPORTED_MODULE_0__); -/* harmony import */ var _style_loader_dist_runtime_styleDomAPI_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! !../../style-loader/dist/runtime/styleDomAPI.js */ "./node_modules/style-loader/dist/runtime/styleDomAPI.js"); -/* harmony import */ var _style_loader_dist_runtime_styleDomAPI_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_style_loader_dist_runtime_styleDomAPI_js__WEBPACK_IMPORTED_MODULE_1__); -/* harmony import */ var _style_loader_dist_runtime_insertBySelector_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! !../../style-loader/dist/runtime/insertBySelector.js */ "./node_modules/style-loader/dist/runtime/insertBySelector.js"); -/* harmony import */ var _style_loader_dist_runtime_insertBySelector_js__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(_style_loader_dist_runtime_insertBySelector_js__WEBPACK_IMPORTED_MODULE_2__); -/* harmony import */ var _style_loader_dist_runtime_setAttributesWithoutAttributes_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! !../../style-loader/dist/runtime/setAttributesWithoutAttributes.js */ "./node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js"); -/* harmony import */ var _style_loader_dist_runtime_setAttributesWithoutAttributes_js__WEBPACK_IMPORTED_MODULE_3___default = /*#__PURE__*/__webpack_require__.n(_style_loader_dist_runtime_setAttributesWithoutAttributes_js__WEBPACK_IMPORTED_MODULE_3__); -/* harmony import */ var _style_loader_dist_runtime_insertStyleElement_js__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! !../../style-loader/dist/runtime/insertStyleElement.js */ "./node_modules/style-loader/dist/runtime/insertStyleElement.js"); -/* harmony import */ var _style_loader_dist_runtime_insertStyleElement_js__WEBPACK_IMPORTED_MODULE_4___default = /*#__PURE__*/__webpack_require__.n(_style_loader_dist_runtime_insertStyleElement_js__WEBPACK_IMPORTED_MODULE_4__); -/* harmony import */ var _style_loader_dist_runtime_styleTagTransform_js__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! !../../style-loader/dist/runtime/styleTagTransform.js */ "./node_modules/style-loader/dist/runtime/styleTagTransform.js"); -/* harmony import */ var _style_loader_dist_runtime_styleTagTransform_js__WEBPACK_IMPORTED_MODULE_5___default = /*#__PURE__*/__webpack_require__.n(_style_loader_dist_runtime_styleTagTransform_js__WEBPACK_IMPORTED_MODULE_5__); -/* harmony import */ var _css_loader_dist_cjs_js_uPlot_min_css__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! !!../../css-loader/dist/cjs.js!./uPlot.min.css */ "./node_modules/css-loader/dist/cjs.js!./node_modules/uplot/dist/uPlot.min.css"); - - - - - - - - - - - -var options = {}; - -options.styleTagTransform = (_style_loader_dist_runtime_styleTagTransform_js__WEBPACK_IMPORTED_MODULE_5___default()); -options.setAttributes = (_style_loader_dist_runtime_setAttributesWithoutAttributes_js__WEBPACK_IMPORTED_MODULE_3___default()); - - options.insert = _style_loader_dist_runtime_insertBySelector_js__WEBPACK_IMPORTED_MODULE_2___default().bind(null, "head"); - -options.domAPI = (_style_loader_dist_runtime_styleDomAPI_js__WEBPACK_IMPORTED_MODULE_1___default()); -options.insertStyleElement = (_style_loader_dist_runtime_insertStyleElement_js__WEBPACK_IMPORTED_MODULE_4___default()); - -var update = _style_loader_dist_runtime_injectStylesIntoStyleTag_js__WEBPACK_IMPORTED_MODULE_0___default()(_css_loader_dist_cjs_js_uPlot_min_css__WEBPACK_IMPORTED_MODULE_6__["default"], options); - - - - - /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (_css_loader_dist_cjs_js_uPlot_min_css__WEBPACK_IMPORTED_MODULE_6__["default"] && _css_loader_dist_cjs_js_uPlot_min_css__WEBPACK_IMPORTED_MODULE_6__["default"].locals ? _css_loader_dist_cjs_js_uPlot_min_css__WEBPACK_IMPORTED_MODULE_6__["default"].locals : undefined); - - -/***/ }), - -/***/ "./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js": -/*!****************************************************************************!*\ - !*** ./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js ***! - \****************************************************************************/ -/***/ ((module) => { - - - -var stylesInDOM = []; - -function getIndexByIdentifier(identifier) { - var result = -1; - - for (var i = 0; i < stylesInDOM.length; i++) { - if (stylesInDOM[i].identifier === identifier) { - result = i; - break; - } - } - - return result; -} - -function modulesToDom(list, options) { - var idCountMap = {}; - var identifiers = []; - - for (var i = 0; i < list.length; i++) { - var item = list[i]; - var id = options.base ? item[0] + options.base : item[0]; - var count = idCountMap[id] || 0; - var identifier = "".concat(id, " ").concat(count); - idCountMap[id] = count + 1; - var indexByIdentifier = getIndexByIdentifier(identifier); - var obj = { - css: item[1], - media: item[2], - sourceMap: item[3], - supports: item[4], - layer: item[5] - }; - - if (indexByIdentifier !== -1) { - stylesInDOM[indexByIdentifier].references++; - stylesInDOM[indexByIdentifier].updater(obj); - } else { - var updater = addElementStyle(obj, options); - options.byIndex = i; - stylesInDOM.splice(i, 0, { - identifier: identifier, - updater: updater, - references: 1 - }); - } - - identifiers.push(identifier); - } - - return identifiers; -} - -function addElementStyle(obj, options) { - var api = options.domAPI(options); - api.update(obj); - - var updater = function updater(newObj) { - if (newObj) { - if (newObj.css === obj.css && newObj.media === obj.media && newObj.sourceMap === obj.sourceMap && newObj.supports === obj.supports && newObj.layer === obj.layer) { - return; - } - - api.update(obj = newObj); - } else { - api.remove(); - } - }; - - return updater; -} - -module.exports = function (list, options) { - options = options || {}; - list = list || []; - var lastIdentifiers = modulesToDom(list, options); - return function update(newList) { - newList = newList || []; - - for (var i = 0; i < lastIdentifiers.length; i++) { - var identifier = lastIdentifiers[i]; - var index = getIndexByIdentifier(identifier); - stylesInDOM[index].references--; - } - - var newLastIdentifiers = modulesToDom(newList, options); - - for (var _i = 0; _i < lastIdentifiers.length; _i++) { - var _identifier = lastIdentifiers[_i]; - - var _index = getIndexByIdentifier(_identifier); - - if (stylesInDOM[_index].references === 0) { - stylesInDOM[_index].updater(); - - stylesInDOM.splice(_index, 1); - } - } - - lastIdentifiers = newLastIdentifiers; - }; -}; - -/***/ }), - -/***/ "./node_modules/style-loader/dist/runtime/insertBySelector.js": -/*!********************************************************************!*\ - !*** ./node_modules/style-loader/dist/runtime/insertBySelector.js ***! - \********************************************************************/ -/***/ ((module) => { - - - -var memo = {}; -/* istanbul ignore next */ - -function getTarget(target) { - if (typeof memo[target] === "undefined") { - var styleTarget = document.querySelector(target); // Special case to return head of iframe instead of iframe itself - - if (window.HTMLIFrameElement && styleTarget instanceof window.HTMLIFrameElement) { - try { - // This will throw an exception if access to iframe is blocked - // due to cross-origin restrictions - styleTarget = styleTarget.contentDocument.head; - } catch (e) { - // istanbul ignore next - styleTarget = null; - } - } - - memo[target] = styleTarget; - } - - return memo[target]; -} -/* istanbul ignore next */ - - -function insertBySelector(insert, style) { - var target = getTarget(insert); - - if (!target) { - throw new Error("Couldn't find a style target. This probably means that the value for the 'insert' parameter is invalid."); - } - - target.appendChild(style); -} - -module.exports = insertBySelector; - -/***/ }), - -/***/ "./node_modules/style-loader/dist/runtime/insertStyleElement.js": -/*!**********************************************************************!*\ - !*** ./node_modules/style-loader/dist/runtime/insertStyleElement.js ***! - \**********************************************************************/ -/***/ ((module) => { - - - -/* istanbul ignore next */ -function insertStyleElement(options) { - var element = document.createElement("style"); - options.setAttributes(element, options.attributes); - options.insert(element, options.options); - return element; -} - -module.exports = insertStyleElement; - -/***/ }), - -/***/ "./node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js": -/*!**********************************************************************************!*\ - !*** ./node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js ***! - \**********************************************************************************/ -/***/ ((module, __unused_webpack_exports, __webpack_require__) => { - - - -/* istanbul ignore next */ -function setAttributesWithoutAttributes(styleElement) { - var nonce = true ? __webpack_require__.nc : 0; - - if (nonce) { - styleElement.setAttribute("nonce", nonce); - } -} - -module.exports = setAttributesWithoutAttributes; - -/***/ }), - -/***/ "./node_modules/style-loader/dist/runtime/styleDomAPI.js": -/*!***************************************************************!*\ - !*** ./node_modules/style-loader/dist/runtime/styleDomAPI.js ***! - \***************************************************************/ -/***/ ((module) => { - - - -/* istanbul ignore next */ -function apply(styleElement, options, obj) { - var css = ""; - - if (obj.supports) { - css += "@supports (".concat(obj.supports, ") {"); - } - - if (obj.media) { - css += "@media ".concat(obj.media, " {"); - } - - var needLayer = typeof obj.layer !== "undefined"; - - if (needLayer) { - css += "@layer".concat(obj.layer.length > 0 ? " ".concat(obj.layer) : "", " {"); - } - - css += obj.css; - - if (needLayer) { - css += "}"; - } - - if (obj.media) { - css += "}"; - } - - if (obj.supports) { - css += "}"; - } - - var sourceMap = obj.sourceMap; - - if (sourceMap && typeof btoa !== "undefined") { - css += "\n/*# sourceMappingURL=data:application/json;base64,".concat(btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap)))), " */"); - } // For old IE - - /* istanbul ignore if */ - - - options.styleTagTransform(css, styleElement, options.options); -} - -function removeStyleElement(styleElement) { - // istanbul ignore if - if (styleElement.parentNode === null) { - return false; - } - - styleElement.parentNode.removeChild(styleElement); -} -/* istanbul ignore next */ - - -function domAPI(options) { - var styleElement = options.insertStyleElement(options); - return { - update: function update(obj) { - apply(styleElement, options, obj); - }, - remove: function remove() { - removeStyleElement(styleElement); - } - }; -} - -module.exports = domAPI; - -/***/ }), - -/***/ "./node_modules/style-loader/dist/runtime/styleTagTransform.js": -/*!*********************************************************************!*\ - !*** ./node_modules/style-loader/dist/runtime/styleTagTransform.js ***! - \*********************************************************************/ -/***/ ((module) => { - - - -/* istanbul ignore next */ -function styleTagTransform(css, styleElement) { - if (styleElement.styleSheet) { - styleElement.styleSheet.cssText = css; - } else { - while (styleElement.firstChild) { - styleElement.removeChild(styleElement.firstChild); - } - - styleElement.appendChild(document.createTextNode(css)); - } -} - -module.exports = styleTagTransform; - -/***/ }), - -/***/ "./node_modules/uplot/dist/uPlot.esm.js": -/*!**********************************************!*\ - !*** ./node_modules/uplot/dist/uPlot.esm.js ***! - \**********************************************/ -/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { - -__webpack_require__.r(__webpack_exports__); -/* harmony export */ __webpack_require__.d(__webpack_exports__, { -/* harmony export */ "default": () => (/* binding */ uPlot) -/* harmony export */ }); -/** -* Copyright (c) 2022, Leon Sorokin -* All rights reserved. (MIT Licensed) -* -* uPlot.js (μPlot) -* A small, fast chart for time series, lines, areas, ohlc & bars -* https://github.com/leeoniya/uPlot (v1.6.22) -*/ - -const FEAT_TIME = true; - -const pre = "u-"; - -const UPLOT = "uplot"; -const ORI_HZ = pre + "hz"; -const ORI_VT = pre + "vt"; -const TITLE = pre + "title"; -const WRAP = pre + "wrap"; -const UNDER = pre + "under"; -const OVER = pre + "over"; -const AXIS = pre + "axis"; -const OFF = pre + "off"; -const SELECT = pre + "select"; -const CURSOR_X = pre + "cursor-x"; -const CURSOR_Y = pre + "cursor-y"; -const CURSOR_PT = pre + "cursor-pt"; -const LEGEND = pre + "legend"; -const LEGEND_LIVE = pre + "live"; -const LEGEND_INLINE = pre + "inline"; -const LEGEND_THEAD = pre + "thead"; -const LEGEND_SERIES = pre + "series"; -const LEGEND_MARKER = pre + "marker"; -const LEGEND_LABEL = pre + "label"; -const LEGEND_VALUE = pre + "value"; - -const WIDTH = "width"; -const HEIGHT = "height"; -const TOP = "top"; -const BOTTOM = "bottom"; -const LEFT = "left"; -const RIGHT = "right"; -const hexBlack = "#000"; -const transparent = hexBlack + "0"; - -const mousemove = "mousemove"; -const mousedown = "mousedown"; -const mouseup = "mouseup"; -const mouseenter = "mouseenter"; -const mouseleave = "mouseleave"; -const dblclick = "dblclick"; -const resize = "resize"; -const scroll = "scroll"; - -const change = "change"; -const dppxchange = "dppxchange"; - -const domEnv = typeof window != 'undefined'; - -const doc = domEnv ? document : null; -const win = domEnv ? window : null; -const nav = domEnv ? navigator : null; - -let pxRatio; - -let query; - -function setPxRatio() { - let _pxRatio = devicePixelRatio; - - // during print preview, Chrome fires off these dppx queries even without changes - if (pxRatio != _pxRatio) { - pxRatio = _pxRatio; - - query && off(change, query, setPxRatio); - query = matchMedia(`(min-resolution: ${pxRatio - 0.001}dppx) and (max-resolution: ${pxRatio + 0.001}dppx)`); - on(change, query, setPxRatio); - - win.dispatchEvent(new CustomEvent(dppxchange)); - } -} - -function addClass(el, c) { - if (c != null) { - let cl = el.classList; - !cl.contains(c) && cl.add(c); - } -} - -function remClass(el, c) { - let cl = el.classList; - cl.contains(c) && cl.remove(c); -} - -function setStylePx(el, name, value) { - el.style[name] = value + "px"; -} - -function placeTag(tag, cls, targ, refEl) { - let el = doc.createElement(tag); - - if (cls != null) - addClass(el, cls); - - if (targ != null) - targ.insertBefore(el, refEl); - - return el; -} - -function placeDiv(cls, targ) { - return placeTag("div", cls, targ); -} - -const xformCache = new WeakMap(); - -function elTrans(el, xPos, yPos, xMax, yMax) { - let xform = "translate(" + xPos + "px," + yPos + "px)"; - let xformOld = xformCache.get(el); - - if (xform != xformOld) { - el.style.transform = xform; - xformCache.set(el, xform); - - if (xPos < 0 || yPos < 0 || xPos > xMax || yPos > yMax) - addClass(el, OFF); - else - remClass(el, OFF); - } -} - -const colorCache = new WeakMap(); - -function elColor(el, background, borderColor) { - let newColor = background + borderColor; - let oldColor = colorCache.get(el); - - if (newColor != oldColor) { - colorCache.set(el, newColor); - el.style.background = background; - el.style.borderColor = borderColor; - } -} - -const sizeCache = new WeakMap(); - -function elSize(el, newWid, newHgt, centered) { - let newSize = newWid + "" + newHgt; - let oldSize = sizeCache.get(el); - - if (newSize != oldSize) { - sizeCache.set(el, newSize); - el.style.height = newHgt + "px"; - el.style.width = newWid + "px"; - el.style.marginLeft = centered ? -newWid/2 + "px" : 0; - el.style.marginTop = centered ? -newHgt/2 + "px" : 0; - } -} - -const evOpts = {passive: true}; -const evOpts2 = {...evOpts, capture: true}; - -function on(ev, el, cb, capt) { - el.addEventListener(ev, cb, capt ? evOpts2 : evOpts); -} - -function off(ev, el, cb, capt) { - el.removeEventListener(ev, cb, capt ? evOpts2 : evOpts); -} - -domEnv && setPxRatio(); - -// binary search for index of closest value -function closestIdx(num, arr, lo, hi) { - let mid; - lo = lo || 0; - hi = hi || arr.length - 1; - let bitwise = hi <= 2147483647; - - while (hi - lo > 1) { - mid = bitwise ? (lo + hi) >> 1 : floor((lo + hi) / 2); - - if (arr[mid] < num) - lo = mid; - else - hi = mid; - } - - if (num - arr[lo] <= arr[hi] - num) - return lo; - - return hi; -} - -function nonNullIdx(data, _i0, _i1, dir) { - for (let i = dir == 1 ? _i0 : _i1; i >= _i0 && i <= _i1; i += dir) { - if (data[i] != null) - return i; - } - - return -1; -} - -function getMinMax(data, _i0, _i1, sorted) { -// console.log("getMinMax()"); - - let _min = inf; - let _max = -inf; - - if (sorted == 1) { - _min = data[_i0]; - _max = data[_i1]; - } - else if (sorted == -1) { - _min = data[_i1]; - _max = data[_i0]; - } - else { - for (let i = _i0; i <= _i1; i++) { - if (data[i] != null) { - _min = min(_min, data[i]); - _max = max(_max, data[i]); - } - } - } - - return [_min, _max]; -} - -function getMinMaxLog(data, _i0, _i1) { -// console.log("getMinMax()"); - - let _min = inf; - let _max = -inf; - - for (let i = _i0; i <= _i1; i++) { - if (data[i] > 0) { - _min = min(_min, data[i]); - _max = max(_max, data[i]); - } - } - - return [ - _min == inf ? 1 : _min, - _max == -inf ? 10 : _max, - ]; -} - -const _fixedTuple = [0, 0]; - -function fixIncr(minIncr, maxIncr, minExp, maxExp) { - _fixedTuple[0] = minExp < 0 ? roundDec(minIncr, -minExp) : minIncr; - _fixedTuple[1] = maxExp < 0 ? roundDec(maxIncr, -maxExp) : maxIncr; - return _fixedTuple; -} - -function rangeLog(min, max, base, fullMags) { - let minSign = sign(min); - - let logFn = base == 10 ? log10 : log2; - - if (min == max) { - if (minSign == -1) { - min *= base; - max /= base; - } - else { - min /= base; - max *= base; - } - } - - let minExp, maxExp, minMaxIncrs; - - if (fullMags) { - minExp = floor(logFn(min)); - maxExp = ceil(logFn(max)); - - minMaxIncrs = fixIncr(pow(base, minExp), pow(base, maxExp), minExp, maxExp); - - min = minMaxIncrs[0]; - max = minMaxIncrs[1]; - } - else { - minExp = floor(logFn(abs(min))); - maxExp = floor(logFn(abs(max))); - - minMaxIncrs = fixIncr(pow(base, minExp), pow(base, maxExp), minExp, maxExp); - - min = incrRoundDn(min, minMaxIncrs[0]); - max = incrRoundUp(max, minMaxIncrs[1]); - } - - return [min, max]; -} - -function rangeAsinh(min, max, base, fullMags) { - let minMax = rangeLog(min, max, base, fullMags); - - if (min == 0) - minMax[0] = 0; - - if (max == 0) - minMax[1] = 0; - - return minMax; -} - -const rangePad = 0.1; - -const autoRangePart = { - mode: 3, - pad: rangePad, -}; - -const _eqRangePart = { - pad: 0, - soft: null, - mode: 0, -}; - -const _eqRange = { - min: _eqRangePart, - max: _eqRangePart, -}; - -// this ensures that non-temporal/numeric y-axes get multiple-snapped padding added above/below -// TODO: also account for incrs when snapping to ensure top of axis gets a tick & value -function rangeNum(_min, _max, mult, extra) { - if (isObj(mult)) - return _rangeNum(_min, _max, mult); - - _eqRangePart.pad = mult; - _eqRangePart.soft = extra ? 0 : null; - _eqRangePart.mode = extra ? 3 : 0; - - return _rangeNum(_min, _max, _eqRange); -} - -// nullish coalesce -function ifNull(lh, rh) { - return lh == null ? rh : lh; -} - -// checks if given index range in an array contains a non-null value -// aka a range-bounded Array.some() -function hasData(data, idx0, idx1) { - idx0 = ifNull(idx0, 0); - idx1 = ifNull(idx1, data.length - 1); - - while (idx0 <= idx1) { - if (data[idx0] != null) - return true; - idx0++; - } - - return false; -} - -function _rangeNum(_min, _max, cfg) { - let cmin = cfg.min; - let cmax = cfg.max; - - let padMin = ifNull(cmin.pad, 0); - let padMax = ifNull(cmax.pad, 0); - - let hardMin = ifNull(cmin.hard, -inf); - let hardMax = ifNull(cmax.hard, inf); - - let softMin = ifNull(cmin.soft, inf); - let softMax = ifNull(cmax.soft, -inf); - - let softMinMode = ifNull(cmin.mode, 0); - let softMaxMode = ifNull(cmax.mode, 0); - - let delta = _max - _min; - - // this handles situations like 89.7, 89.69999999999999 - // by assuming 0.001x deltas are precision errors -// if (delta > 0 && delta < abs(_max) / 1e3) -// delta = 0; - - // treat data as flat if delta is less than 1 billionth - if (delta < 1e-9) { - delta = 0; - - // if soft mode is 2 and all vals are flat at 0, avoid the 0.1 * 1e3 fallback - // this prevents 0,0,0 from ranging to -100,100 when softMin/softMax are -1,1 - if (_min == 0 || _max == 0) { - delta = 1e-9; - - if (softMinMode == 2 && softMin != inf) - padMin = 0; - - if (softMaxMode == 2 && softMax != -inf) - padMax = 0; - } - } - - let nonZeroDelta = delta || abs(_max) || 1e3; - let mag = log10(nonZeroDelta); - let base = pow(10, floor(mag)); - - let _padMin = nonZeroDelta * (delta == 0 ? (_min == 0 ? .1 : 1) : padMin); - let _newMin = roundDec(incrRoundDn(_min - _padMin, base/10), 9); - let _softMin = _min >= softMin && (softMinMode == 1 || softMinMode == 3 && _newMin <= softMin || softMinMode == 2 && _newMin >= softMin) ? softMin : inf; - let minLim = max(hardMin, _newMin < _softMin && _min >= _softMin ? _softMin : min(_softMin, _newMin)); - - let _padMax = nonZeroDelta * (delta == 0 ? (_max == 0 ? .1 : 1) : padMax); - let _newMax = roundDec(incrRoundUp(_max + _padMax, base/10), 9); - let _softMax = _max <= softMax && (softMaxMode == 1 || softMaxMode == 3 && _newMax >= softMax || softMaxMode == 2 && _newMax <= softMax) ? softMax : -inf; - let maxLim = min(hardMax, _newMax > _softMax && _max <= _softMax ? _softMax : max(_softMax, _newMax)); - - if (minLim == maxLim && minLim == 0) - maxLim = 100; - - return [minLim, maxLim]; -} - -// alternative: https://stackoverflow.com/a/2254896 -const numFormatter = new Intl.NumberFormat(domEnv ? nav.language : 'en-US'); -const fmtNum = val => numFormatter.format(val); - -const M = Math; - -const PI = M.PI; -const abs = M.abs; -const floor = M.floor; -const round = M.round; -const ceil = M.ceil; -const min = M.min; -const max = M.max; -const pow = M.pow; -const sign = M.sign; -const log10 = M.log10; -const log2 = M.log2; -// TODO: seems like this needs to match asinh impl if the passed v is tweaked? -const sinh = (v, linthresh = 1) => M.sinh(v) * linthresh; -const asinh = (v, linthresh = 1) => M.asinh(v / linthresh); - -const inf = Infinity; - -function numIntDigits(x) { - return (log10((x ^ (x >> 31)) - (x >> 31)) | 0) + 1; -} - -function incrRound(num, incr) { - return round(num/incr)*incr; -} - -function clamp(num, _min, _max) { - return min(max(num, _min), _max); -} - -function fnOrSelf(v) { - return typeof v == "function" ? v : () => v; -} - -const retArg0 = _0 => _0; - -const retArg1 = (_0, _1) => _1; - -const retNull = _ => null; - -const retTrue = _ => true; - -const retEq = (a, b) => a == b; - -function incrRoundUp(num, incr) { - return ceil(num/incr)*incr; -} - -function incrRoundDn(num, incr) { - return floor(num/incr)*incr; -} - -function roundDec(val, dec) { - return round(val * (dec = 10**dec)) / dec; -} - -const fixedDec = new Map(); - -function guessDec(num) { - return ((""+num).split(".")[1] || "").length; -} - -function genIncrs(base, minExp, maxExp, mults) { - let incrs = []; - - let multDec = mults.map(guessDec); - - for (let exp = minExp; exp < maxExp; exp++) { - let expa = abs(exp); - let mag = roundDec(pow(base, exp), expa); - - for (let i = 0; i < mults.length; i++) { - let _incr = mults[i] * mag; - let dec = (_incr >= 0 && exp >= 0 ? 0 : expa) + (exp >= multDec[i] ? 0 : multDec[i]); - let incr = roundDec(_incr, dec); - incrs.push(incr); - fixedDec.set(incr, dec); - } - } - - return incrs; -} - -//export const assign = Object.assign; - -const EMPTY_OBJ = {}; -const EMPTY_ARR = []; - -const nullNullTuple = [null, null]; - -const isArr = Array.isArray; - -function isStr(v) { - return typeof v == 'string'; -} - -function isObj(v) { - let is = false; - - if (v != null) { - let c = v.constructor; - is = c == null || c == Object; - } - - return is; -} - -function fastIsObj(v) { - return v != null && typeof v == 'object'; -} - -const TypedArray = Object.getPrototypeOf(Uint8Array); - -function copy(o, _isObj = isObj) { - let out; - - if (isArr(o)) { - let val = o.find(v => v != null); - - if (isArr(val) || _isObj(val)) { - out = Array(o.length); - for (let i = 0; i < o.length; i++) - out[i] = copy(o[i], _isObj); - } - else - out = o.slice(); - } - else if (o instanceof TypedArray) // also (ArrayBuffer.isView(o) && !(o instanceof DataView)) - out = o.slice(); - else if (_isObj(o)) { - out = {}; - for (let k in o) - out[k] = copy(o[k], _isObj); - } - else - out = o; - - return out; -} - -function assign(targ) { - let args = arguments; - - for (let i = 1; i < args.length; i++) { - let src = args[i]; - - for (let key in src) { - if (isObj(targ[key])) - assign(targ[key], copy(src[key])); - else - targ[key] = copy(src[key]); - } - } - - return targ; -} - -// nullModes -const NULL_REMOVE = 0; // nulls are converted to undefined (e.g. for spanGaps: true) -const NULL_RETAIN = 1; // nulls are retained, with alignment artifacts set to undefined (default) -const NULL_EXPAND = 2; // nulls are expanded to include any adjacent alignment artifacts - -// sets undefined values to nulls when adjacent to existing nulls (minesweeper) -function nullExpand(yVals, nullIdxs, alignedLen) { - for (let i = 0, xi, lastNullIdx = -1; i < nullIdxs.length; i++) { - let nullIdx = nullIdxs[i]; - - if (nullIdx > lastNullIdx) { - xi = nullIdx - 1; - while (xi >= 0 && yVals[xi] == null) - yVals[xi--] = null; - - xi = nullIdx + 1; - while (xi < alignedLen && yVals[xi] == null) - yVals[lastNullIdx = xi++] = null; - } - } -} - -// nullModes is a tables-matched array indicating how to treat nulls in each series -// output is sorted ASC on the joined field (table[0]) and duplicate join values are collapsed -function join(tables, nullModes) { - let xVals = new Set(); - - for (let ti = 0; ti < tables.length; ti++) { - let t = tables[ti]; - let xs = t[0]; - let len = xs.length; - - for (let i = 0; i < len; i++) - xVals.add(xs[i]); - } - - let data = [Array.from(xVals).sort((a, b) => a - b)]; - - let alignedLen = data[0].length; - - let xIdxs = new Map(); - - for (let i = 0; i < alignedLen; i++) - xIdxs.set(data[0][i], i); - - for (let ti = 0; ti < tables.length; ti++) { - let t = tables[ti]; - let xs = t[0]; - - for (let si = 1; si < t.length; si++) { - let ys = t[si]; - - let yVals = Array(alignedLen).fill(undefined); - - let nullMode = nullModes ? nullModes[ti][si] : NULL_RETAIN; - - let nullIdxs = []; - - for (let i = 0; i < ys.length; i++) { - let yVal = ys[i]; - let alignedIdx = xIdxs.get(xs[i]); - - if (yVal === null) { - if (nullMode != NULL_REMOVE) { - yVals[alignedIdx] = yVal; - - if (nullMode == NULL_EXPAND) - nullIdxs.push(alignedIdx); - } - } - else - yVals[alignedIdx] = yVal; - } - - nullExpand(yVals, nullIdxs, alignedLen); - - data.push(yVals); - } - } - - return data; -} - -const microTask = typeof queueMicrotask == "undefined" ? fn => Promise.resolve().then(fn) : queueMicrotask; - -const months = [ - "January", - "February", - "March", - "April", - "May", - "June", - "July", - "August", - "September", - "October", - "November", - "December", -]; - -const days = [ - "Sunday", - "Monday", - "Tuesday", - "Wednesday", - "Thursday", - "Friday", - "Saturday", -]; - -function slice3(str) { - return str.slice(0, 3); -} - -const days3 = days.map(slice3); - -const months3 = months.map(slice3); - -const engNames = { - MMMM: months, - MMM: months3, - WWWW: days, - WWW: days3, -}; - -function zeroPad2(int) { - return (int < 10 ? '0' : '') + int; -} - -function zeroPad3(int) { - return (int < 10 ? '00' : int < 100 ? '0' : '') + int; -} - -/* -function suffix(int) { - let mod10 = int % 10; - - return int + ( - mod10 == 1 && int != 11 ? "st" : - mod10 == 2 && int != 12 ? "nd" : - mod10 == 3 && int != 13 ? "rd" : "th" - ); -} -*/ - -const subs = { - // 2019 - YYYY: d => d.getFullYear(), - // 19 - YY: d => (d.getFullYear()+'').slice(2), - // July - MMMM: (d, names) => names.MMMM[d.getMonth()], - // Jul - MMM: (d, names) => names.MMM[d.getMonth()], - // 07 - MM: d => zeroPad2(d.getMonth()+1), - // 7 - M: d => d.getMonth()+1, - // 09 - DD: d => zeroPad2(d.getDate()), - // 9 - D: d => d.getDate(), - // Monday - WWWW: (d, names) => names.WWWW[d.getDay()], - // Mon - WWW: (d, names) => names.WWW[d.getDay()], - // 03 - HH: d => zeroPad2(d.getHours()), - // 3 - H: d => d.getHours(), - // 9 (12hr, unpadded) - h: d => {let h = d.getHours(); return h == 0 ? 12 : h > 12 ? h - 12 : h;}, - // AM - AA: d => d.getHours() >= 12 ? 'PM' : 'AM', - // am - aa: d => d.getHours() >= 12 ? 'pm' : 'am', - // a - a: d => d.getHours() >= 12 ? 'p' : 'a', - // 09 - mm: d => zeroPad2(d.getMinutes()), - // 9 - m: d => d.getMinutes(), - // 09 - ss: d => zeroPad2(d.getSeconds()), - // 9 - s: d => d.getSeconds(), - // 374 - fff: d => zeroPad3(d.getMilliseconds()), -}; - -function fmtDate(tpl, names) { - names = names || engNames; - let parts = []; - - let R = /\{([a-z]+)\}|[^{]+/gi, m; - - while (m = R.exec(tpl)) - parts.push(m[0][0] == '{' ? subs[m[1]] : m[0]); - - return d => { - let out = ''; - - for (let i = 0; i < parts.length; i++) - out += typeof parts[i] == "string" ? parts[i] : parts[i](d, names); - - return out; - } -} - -const localTz = new Intl.DateTimeFormat().resolvedOptions().timeZone; - -// https://stackoverflow.com/questions/15141762/how-to-initialize-a-javascript-date-to-a-particular-time-zone/53652131#53652131 -function tzDate(date, tz) { - let date2; - - // perf optimization - if (tz == 'UTC' || tz == 'Etc/UTC') - date2 = new Date(+date + date.getTimezoneOffset() * 6e4); - else if (tz == localTz) - date2 = date; - else { - date2 = new Date(date.toLocaleString('en-US', {timeZone: tz})); - date2.setMilliseconds(date.getMilliseconds()); - } - - return date2; -} - -//export const series = []; - -// default formatters: - -const onlyWhole = v => v % 1 == 0; - -const allMults = [1,2,2.5,5]; - -// ...0.01, 0.02, 0.025, 0.05, 0.1, 0.2, 0.25, 0.5 -const decIncrs = genIncrs(10, -16, 0, allMults); - -// 1, 2, 2.5, 5, 10, 20, 25, 50... -const oneIncrs = genIncrs(10, 0, 16, allMults); - -// 1, 2, 5, 10, 20, 25, 50... -const wholeIncrs = oneIncrs.filter(onlyWhole); - -const numIncrs = decIncrs.concat(oneIncrs); - -const NL = "\n"; - -const yyyy = "{YYYY}"; -const NLyyyy = NL + yyyy; -const md = "{M}/{D}"; -const NLmd = NL + md; -const NLmdyy = NLmd + "/{YY}"; - -const aa = "{aa}"; -const hmm = "{h}:{mm}"; -const hmmaa = hmm + aa; -const NLhmmaa = NL + hmmaa; -const ss = ":{ss}"; - -const _ = null; - -function genTimeStuffs(ms) { - let s = ms * 1e3, - m = s * 60, - h = m * 60, - d = h * 24, - mo = d * 30, - y = d * 365; - - // min of 1e-3 prevents setting a temporal x ticks too small since Date objects cannot advance ticks smaller than 1ms - let subSecIncrs = ms == 1 ? genIncrs(10, 0, 3, allMults).filter(onlyWhole) : genIncrs(10, -3, 0, allMults); - - let timeIncrs = subSecIncrs.concat([ - // minute divisors (# of secs) - s, - s * 5, - s * 10, - s * 15, - s * 30, - // hour divisors (# of mins) - m, - m * 5, - m * 10, - m * 15, - m * 30, - // day divisors (# of hrs) - h, - h * 2, - h * 3, - h * 4, - h * 6, - h * 8, - h * 12, - // month divisors TODO: need more? - d, - d * 2, - d * 3, - d * 4, - d * 5, - d * 6, - d * 7, - d * 8, - d * 9, - d * 10, - d * 15, - // year divisors (# months, approx) - mo, - mo * 2, - mo * 3, - mo * 4, - mo * 6, - // century divisors - y, - y * 2, - y * 5, - y * 10, - y * 25, - y * 50, - y * 100, - ]); - - // [0]: minimum num secs in the tick incr - // [1]: default tick format - // [2-7]: rollover tick formats - // [8]: mode: 0: replace [1] -> [2-7], 1: concat [1] + [2-7] - const _timeAxisStamps = [ - // tick incr default year month day hour min sec mode - [y, yyyy, _, _, _, _, _, _, 1], - [d * 28, "{MMM}", NLyyyy, _, _, _, _, _, 1], - [d, md, NLyyyy, _, _, _, _, _, 1], - [h, "{h}" + aa, NLmdyy, _, NLmd, _, _, _, 1], - [m, hmmaa, NLmdyy, _, NLmd, _, _, _, 1], - [s, ss, NLmdyy + " " + hmmaa, _, NLmd + " " + hmmaa, _, NLhmmaa, _, 1], - [ms, ss + ".{fff}", NLmdyy + " " + hmmaa, _, NLmd + " " + hmmaa, _, NLhmmaa, _, 1], - ]; - - // the ensures that axis ticks, values & grid are aligned to logical temporal breakpoints and not an arbitrary timestamp - // https://www.timeanddate.com/time/dst/ - // https://www.timeanddate.com/time/dst/2019.html - // https://www.epochconverter.com/timezones - function timeAxisSplits(tzDate) { - return (self, axisIdx, scaleMin, scaleMax, foundIncr, foundSpace) => { - let splits = []; - let isYr = foundIncr >= y; - let isMo = foundIncr >= mo && foundIncr < y; - - // get the timezone-adjusted date - let minDate = tzDate(scaleMin); - let minDateTs = roundDec(minDate * ms, 3); - - // get ts of 12am (this lands us at or before the original scaleMin) - let minMin = mkDate(minDate.getFullYear(), isYr ? 0 : minDate.getMonth(), isMo || isYr ? 1 : minDate.getDate()); - let minMinTs = roundDec(minMin * ms, 3); - - if (isMo || isYr) { - let moIncr = isMo ? foundIncr / mo : 0; - let yrIncr = isYr ? foundIncr / y : 0; - // let tzOffset = scaleMin - minDateTs; // needed? - let split = minDateTs == minMinTs ? minDateTs : roundDec(mkDate(minMin.getFullYear() + yrIncr, minMin.getMonth() + moIncr, 1) * ms, 3); - let splitDate = new Date(round(split / ms)); - let baseYear = splitDate.getFullYear(); - let baseMonth = splitDate.getMonth(); - - for (let i = 0; split <= scaleMax; i++) { - let next = mkDate(baseYear + yrIncr * i, baseMonth + moIncr * i, 1); - let offs = next - tzDate(roundDec(next * ms, 3)); - - split = roundDec((+next + offs) * ms, 3); - - if (split <= scaleMax) - splits.push(split); - } - } - else { - let incr0 = foundIncr >= d ? d : foundIncr; - let tzOffset = floor(scaleMin) - floor(minDateTs); - let split = minMinTs + tzOffset + incrRoundUp(minDateTs - minMinTs, incr0); - splits.push(split); - - let date0 = tzDate(split); - - let prevHour = date0.getHours() + (date0.getMinutes() / m) + (date0.getSeconds() / h); - let incrHours = foundIncr / h; - - let minSpace = self.axes[axisIdx]._space; - let pctSpace = foundSpace / minSpace; - - while (1) { - split = roundDec(split + foundIncr, ms == 1 ? 0 : 3); - - if (split > scaleMax) - break; - - if (incrHours > 1) { - let expectedHour = floor(roundDec(prevHour + incrHours, 6)) % 24; - let splitDate = tzDate(split); - let actualHour = splitDate.getHours(); - - let dstShift = actualHour - expectedHour; - - if (dstShift > 1) - dstShift = -1; - - split -= dstShift * h; - - prevHour = (prevHour + incrHours) % 24; - - // add a tick only if it's further than 70% of the min allowed label spacing - let prevSplit = splits[splits.length - 1]; - let pctIncr = roundDec((split - prevSplit) / foundIncr, 3); - - if (pctIncr * pctSpace >= .7) - splits.push(split); - } - else - splits.push(split); - } - } - - return splits; - } - } - - return [ - timeIncrs, - _timeAxisStamps, - timeAxisSplits, - ]; -} - -const [ timeIncrsMs, _timeAxisStampsMs, timeAxisSplitsMs ] = genTimeStuffs(1); -const [ timeIncrsS, _timeAxisStampsS, timeAxisSplitsS ] = genTimeStuffs(1e-3); - -// base 2 -genIncrs(2, -53, 53, [1]); - -/* -console.log({ - decIncrs, - oneIncrs, - wholeIncrs, - numIncrs, - timeIncrs, - fixedDec, -}); -*/ - -function timeAxisStamps(stampCfg, fmtDate) { - return stampCfg.map(s => s.map((v, i) => - i == 0 || i == 8 || v == null ? v : fmtDate(i == 1 || s[8] == 0 ? v : s[1] + v) - )); -} - -// TODO: will need to accept spaces[] and pull incr into the loop when grid will be non-uniform, eg for log scales. -// currently we ignore this for months since they're *nearly* uniform and the added complexity is not worth it -function timeAxisVals(tzDate, stamps) { - return (self, splits, axisIdx, foundSpace, foundIncr) => { - let s = stamps.find(s => foundIncr >= s[0]) || stamps[stamps.length - 1]; - - // these track boundaries when a full label is needed again - let prevYear; - let prevMnth; - let prevDate; - let prevHour; - let prevMins; - let prevSecs; - - return splits.map(split => { - let date = tzDate(split); - - let newYear = date.getFullYear(); - let newMnth = date.getMonth(); - let newDate = date.getDate(); - let newHour = date.getHours(); - let newMins = date.getMinutes(); - let newSecs = date.getSeconds(); - - let stamp = ( - newYear != prevYear && s[2] || - newMnth != prevMnth && s[3] || - newDate != prevDate && s[4] || - newHour != prevHour && s[5] || - newMins != prevMins && s[6] || - newSecs != prevSecs && s[7] || - s[1] - ); - - prevYear = newYear; - prevMnth = newMnth; - prevDate = newDate; - prevHour = newHour; - prevMins = newMins; - prevSecs = newSecs; - - return stamp(date); - }); - } -} - -// for when axis.values is defined as a static fmtDate template string -function timeAxisVal(tzDate, dateTpl) { - let stamp = fmtDate(dateTpl); - return (self, splits, axisIdx, foundSpace, foundIncr) => splits.map(split => stamp(tzDate(split))); -} - -function mkDate(y, m, d) { - return new Date(y, m, d); -} - -function timeSeriesStamp(stampCfg, fmtDate) { - return fmtDate(stampCfg); -} -const _timeSeriesStamp = '{YYYY}-{MM}-{DD} {h}:{mm}{aa}'; - -function timeSeriesVal(tzDate, stamp) { - return (self, val) => stamp(tzDate(val)); -} - -function legendStroke(self, seriesIdx) { - let s = self.series[seriesIdx]; - return s.width ? s.stroke(self, seriesIdx) : s.points.width ? s.points.stroke(self, seriesIdx) : null; -} - -function legendFill(self, seriesIdx) { - return self.series[seriesIdx].fill(self, seriesIdx); -} - -const legendOpts = { - show: true, - live: true, - isolate: false, - markers: { - show: true, - width: 2, - stroke: legendStroke, - fill: legendFill, - dash: "solid", - }, - idx: null, - idxs: null, - values: [], -}; - -function cursorPointShow(self, si) { - let o = self.cursor.points; - - let pt = placeDiv(); - - let size = o.size(self, si); - setStylePx(pt, WIDTH, size); - setStylePx(pt, HEIGHT, size); - - let mar = size / -2; - setStylePx(pt, "marginLeft", mar); - setStylePx(pt, "marginTop", mar); - - let width = o.width(self, si, size); - width && setStylePx(pt, "borderWidth", width); - - return pt; -} - -function cursorPointFill(self, si) { - let sp = self.series[si].points; - return sp._fill || sp._stroke; -} - -function cursorPointStroke(self, si) { - let sp = self.series[si].points; - return sp._stroke || sp._fill; -} - -function cursorPointSize(self, si) { - let sp = self.series[si].points; - return ptDia(sp.width, 1); -} - -function dataIdx(self, seriesIdx, cursorIdx) { - return cursorIdx; -} - -const moveTuple = [0,0]; - -function cursorMove(self, mouseLeft1, mouseTop1) { - moveTuple[0] = mouseLeft1; - moveTuple[1] = mouseTop1; - return moveTuple; -} - -function filtBtn0(self, targ, handle) { - return e => { - e.button == 0 && handle(e); - }; -} - -function passThru(self, targ, handle) { - return handle; -} - -const cursorOpts = { - show: true, - x: true, - y: true, - lock: false, - move: cursorMove, - points: { - show: cursorPointShow, - size: cursorPointSize, - width: 0, - stroke: cursorPointStroke, - fill: cursorPointFill, - }, - - bind: { - mousedown: filtBtn0, - mouseup: filtBtn0, - click: filtBtn0, - dblclick: filtBtn0, - - mousemove: passThru, - mouseleave: passThru, - mouseenter: passThru, - }, - - drag: { - setScale: true, - x: true, - y: false, - dist: 0, - uni: null, - _x: false, - _y: false, - }, - - focus: { - prox: -1, - }, - - left: -10, - top: -10, - idx: null, - dataIdx, - idxs: null, -}; - -const axisLines = { - show: true, - stroke: "rgba(0,0,0,0.07)", - width: 2, -// dash: [], -}; - -const grid = assign({}, axisLines, { - filter: retArg1, -}); - -const ticks = assign({}, grid, { - size: 10, -}); - -const border = assign({}, axisLines, { - show: false, -}); - -const font = '12px system-ui, -apple-system, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji"'; -const labelFont = "bold " + font; -const lineMult = 1.5; // font-size multiplier - -const xAxisOpts = { - show: true, - scale: "x", - stroke: hexBlack, - space: 50, - gap: 5, - size: 50, - labelGap: 0, - labelSize: 30, - labelFont, - side: 2, -// class: "x-vals", -// incrs: timeIncrs, -// values: timeVals, -// filter: retArg1, - grid, - ticks, - border, - font, - rotate: 0, -}; - -const numSeriesLabel = "Value"; -const timeSeriesLabel = "Time"; - -const xSeriesOpts = { - show: true, - scale: "x", - auto: false, - sorted: 1, -// label: "Time", -// value: v => stamp(new Date(v * 1e3)), - - // internal caches - min: inf, - max: -inf, - idxs: [], -}; - -function numAxisVals(self, splits, axisIdx, foundSpace, foundIncr) { - return splits.map(v => v == null ? "" : fmtNum(v)); -} - -function numAxisSplits(self, axisIdx, scaleMin, scaleMax, foundIncr, foundSpace, forceMin) { - let splits = []; - - let numDec = fixedDec.get(foundIncr) || 0; - - scaleMin = forceMin ? scaleMin : roundDec(incrRoundUp(scaleMin, foundIncr), numDec); - - for (let val = scaleMin; val <= scaleMax; val = roundDec(val + foundIncr, numDec)) - splits.push(Object.is(val, -0) ? 0 : val); // coalesces -0 - - return splits; -} - -// this doesnt work for sin, which needs to come off from 0 independently in pos and neg dirs -function logAxisSplits(self, axisIdx, scaleMin, scaleMax, foundIncr, foundSpace, forceMin) { - const splits = []; - - const logBase = self.scales[self.axes[axisIdx].scale].log; - - const logFn = logBase == 10 ? log10 : log2; - - const exp = floor(logFn(scaleMin)); - - foundIncr = pow(logBase, exp); - - if (exp < 0) - foundIncr = roundDec(foundIncr, -exp); - - let split = scaleMin; - - do { - splits.push(split); - split = roundDec(split + foundIncr, fixedDec.get(foundIncr)); - - if (split >= foundIncr * logBase) - foundIncr = split; - - } while (split <= scaleMax); - - return splits; -} - -function asinhAxisSplits(self, axisIdx, scaleMin, scaleMax, foundIncr, foundSpace, forceMin) { - let sc = self.scales[self.axes[axisIdx].scale]; - - let linthresh = sc.asinh; - - let posSplits = scaleMax > linthresh ? logAxisSplits(self, axisIdx, max(linthresh, scaleMin), scaleMax, foundIncr) : [linthresh]; - let zero = scaleMax >= 0 && scaleMin <= 0 ? [0] : []; - let negSplits = scaleMin < -linthresh ? logAxisSplits(self, axisIdx, max(linthresh, -scaleMax), -scaleMin, foundIncr): [linthresh]; - - return negSplits.reverse().map(v => -v).concat(zero, posSplits); -} - -const RE_ALL = /./; -const RE_12357 = /[12357]/; -const RE_125 = /[125]/; -const RE_1 = /1/; - -function logAxisValsFilt(self, splits, axisIdx, foundSpace, foundIncr) { - let axis = self.axes[axisIdx]; - let scaleKey = axis.scale; - let sc = self.scales[scaleKey]; - - if (sc.distr == 3 && sc.log == 2) - return splits; - - let valToPos = self.valToPos; - - let minSpace = axis._space; - - let _10 = valToPos(10, scaleKey); - - let re = ( - valToPos(9, scaleKey) - _10 >= minSpace ? RE_ALL : - valToPos(7, scaleKey) - _10 >= minSpace ? RE_12357 : - valToPos(5, scaleKey) - _10 >= minSpace ? RE_125 : - RE_1 - ); - - return splits.map(v => ((sc.distr == 4 && v == 0) || re.test(v)) ? v : null); -} - -function numSeriesVal(self, val) { - return val == null ? "" : fmtNum(val); -} - -const yAxisOpts = { - show: true, - scale: "y", - stroke: hexBlack, - space: 30, - gap: 5, - size: 50, - labelGap: 0, - labelSize: 30, - labelFont, - side: 3, -// class: "y-vals", -// incrs: numIncrs, -// values: (vals, space) => vals, -// filter: retArg1, - grid, - ticks, - border, - font, - rotate: 0, -}; - -// takes stroke width -function ptDia(width, mult) { - let dia = 3 + (width || 1) * 2; - return roundDec(dia * mult, 3); -} - -function seriesPointsShow(self, si) { - let { scale, idxs } = self.series[0]; - let xData = self._data[0]; - let p0 = self.valToPos(xData[idxs[0]], scale, true); - let p1 = self.valToPos(xData[idxs[1]], scale, true); - let dim = abs(p1 - p0); - - let s = self.series[si]; -// const dia = ptDia(s.width, pxRatio); - let maxPts = dim / (s.points.space * pxRatio); - return idxs[1] - idxs[0] <= maxPts; -} - -const facet = { - scale: null, - auto: true, - sorted: 0, - - // internal caches - min: inf, - max: -inf, -}; - -const xySeriesOpts = { - show: true, - auto: true, - sorted: 0, - alpha: 1, - facets: [ - assign({}, facet, {scale: 'x'}), - assign({}, facet, {scale: 'y'}), - ], -}; - -const ySeriesOpts = { - scale: "y", - auto: true, - sorted: 0, - show: true, - spanGaps: false, - gaps: (self, seriesIdx, idx0, idx1, nullGaps) => nullGaps, - alpha: 1, - points: { - show: seriesPointsShow, - filter: null, - // paths: - // stroke: "#000", - // fill: "#fff", - // width: 1, - // size: 10, - }, -// label: "Value", -// value: v => v, - values: null, - - // internal caches - min: inf, - max: -inf, - idxs: [], - - path: null, - clip: null, -}; - -function clampScale(self, val, scaleMin, scaleMax, scaleKey) { -/* - if (val < 0) { - let cssHgt = self.bbox.height / pxRatio; - let absPos = self.valToPos(abs(val), scaleKey); - let fromBtm = cssHgt - absPos; - return self.posToVal(cssHgt + fromBtm, scaleKey); - } -*/ - return scaleMin / 10; -} - -const xScaleOpts = { - time: FEAT_TIME, - auto: true, - distr: 1, - log: 10, - asinh: 1, - min: null, - max: null, - dir: 1, - ori: 0, -}; - -const yScaleOpts = assign({}, xScaleOpts, { - time: false, - ori: 1, -}); - -const syncs = {}; - -function _sync(key, opts) { - let s = syncs[key]; - - if (!s) { - s = { - key, - plots: [], - sub(plot) { - s.plots.push(plot); - }, - unsub(plot) { - s.plots = s.plots.filter(c => c != plot); - }, - pub(type, self, x, y, w, h, i) { - for (let j = 0; j < s.plots.length; j++) - s.plots[j] != self && s.plots[j].pub(type, self, x, y, w, h, i); - }, - }; - - if (key != null) - syncs[key] = s; - } - - return s; -} - -const BAND_CLIP_FILL = 1 << 0; -const BAND_CLIP_STROKE = 1 << 1; - -function orient(u, seriesIdx, cb) { - const series = u.series[seriesIdx]; - const scales = u.scales; - const bbox = u.bbox; - const scaleX = u.mode == 2 ? scales[series.facets[0].scale] : scales[u.series[0].scale]; - - let dx = u._data[0], - dy = u._data[seriesIdx], - sx = scaleX, - sy = u.mode == 2 ? scales[series.facets[1].scale] : scales[series.scale], - l = bbox.left, - t = bbox.top, - w = bbox.width, - h = bbox.height, - H = u.valToPosH, - V = u.valToPosV; - - return (sx.ori == 0 - ? cb( - series, - dx, - dy, - sx, - sy, - H, - V, - l, - t, - w, - h, - moveToH, - lineToH, - rectH, - arcH, - bezierCurveToH, - ) - : cb( - series, - dx, - dy, - sx, - sy, - V, - H, - t, - l, - h, - w, - moveToV, - lineToV, - rectV, - arcV, - bezierCurveToV, - ) - ); -} - -function bandFillClipDirs(self, seriesIdx) { - let fillDir = 0; - - // 2 bits, -1 | 1 - let clipDirs = 0; - - let bands = ifNull(self.bands, EMPTY_ARR); - - for (let i = 0; i < bands.length; i++) { - let b = bands[i]; - - // is a "from" band edge - if (b.series[0] == seriesIdx) - fillDir = b.dir; - // is a "to" band edge - else if (b.series[1] == seriesIdx) { - if (b.dir == 1) - clipDirs |= 1; - else - clipDirs |= 2; - } - } - - return [ - fillDir, - ( - clipDirs == 1 ? -1 : // neg only - clipDirs == 2 ? 1 : // pos only - clipDirs == 3 ? 2 : // both - 0 // neither - ) - ]; -} - -function seriesFillTo(self, seriesIdx, dataMin, dataMax, bandFillDir) { - let scale = self.scales[self.series[seriesIdx].scale]; - - return ( - bandFillDir == -1 ? scale.min : - bandFillDir == 1 ? scale.max : - scale.distr == 3 ? ( - scale.dir == 1 ? scale.min : - scale.max - ) : 0 - ); -} - -// creates inverted band clip path (from stroke path -> yMax || yMin) -// clipDir is always inverse of fillDir -// default clip dir is upwards (1), since default band fill is downwards/fillBelowTo (-1) (highIdx -> lowIdx) -function clipBandLine(self, seriesIdx, idx0, idx1, strokePath, clipDir) { - return orient(self, seriesIdx, (series, dataX, dataY, scaleX, scaleY, valToPosX, valToPosY, xOff, yOff, xDim, yDim) => { - let pxRound = series.pxRound; - - const dir = scaleX.dir * (scaleX.ori == 0 ? 1 : -1); - const lineTo = scaleX.ori == 0 ? lineToH : lineToV; - - let frIdx, toIdx; - - if (dir == 1) { - frIdx = idx0; - toIdx = idx1; - } - else { - frIdx = idx1; - toIdx = idx0; - } - - // path start - let x0 = pxRound(valToPosX(dataX[frIdx], scaleX, xDim, xOff)); - let y0 = pxRound(valToPosY(dataY[frIdx], scaleY, yDim, yOff)); - // path end x - let x1 = pxRound(valToPosX(dataX[toIdx], scaleX, xDim, xOff)); - // upper or lower y limit - let yLimit = pxRound(valToPosY(clipDir == 1 ? scaleY.max : scaleY.min, scaleY, yDim, yOff)); - - let clip = new Path2D(strokePath); - - lineTo(clip, x1, yLimit); - lineTo(clip, x0, yLimit); - lineTo(clip, x0, y0); - - return clip; - }); -} - -function clipGaps(gaps, ori, plotLft, plotTop, plotWid, plotHgt) { - let clip = null; - - // create clip path (invert gaps and non-gaps) - if (gaps.length > 0) { - clip = new Path2D(); - - const rect = ori == 0 ? rectH : rectV; - - let prevGapEnd = plotLft; - - for (let i = 0; i < gaps.length; i++) { - let g = gaps[i]; - - if (g[1] > g[0]) { - let w = g[0] - prevGapEnd; - - w > 0 && rect(clip, prevGapEnd, plotTop, w, plotTop + plotHgt); - - prevGapEnd = g[1]; - } - } - - let w = plotLft + plotWid - prevGapEnd; - - w > 0 && rect(clip, prevGapEnd, plotTop, w, plotTop + plotHgt); - } - - return clip; -} - -function addGap(gaps, fromX, toX) { - let prevGap = gaps[gaps.length - 1]; - - if (prevGap && prevGap[0] == fromX) // TODO: gaps must be encoded at stroke widths? - prevGap[1] = toX; - else - gaps.push([fromX, toX]); -} - -function findGaps(xs, ys, idx0, idx1, dir, pixelForX, align) { - let gaps = []; - - for (let i = dir == 1 ? idx0 : idx1; i >= idx0 && i <= idx1; i += dir) { - let yVal = ys[i]; - - if (yVal === null) { - let fr = i, to = i; - - if (dir == 1) { - while (++i <= idx1 && ys[i] === null) - to = i; - } - else { - while (--i >= idx0 && ys[i] === null) - to = i; - } - - let frPx = pixelForX(xs[fr]); - let toPx = to == fr ? frPx : pixelForX(xs[to]); - - // if value adjacent to edge null is same pixel, then it's partially - // filled and gap should start at next pixel - let frPx2 = align <= 0 ? pixelForX(xs[fr-dir]) : frPx; - // if (frPx2 == frPx) - // frPx++; - // else - frPx = frPx2; - - let toPx2 = align >= 0 ? pixelForX(xs[to+dir]) : toPx; - // if (toPx2 == toPx) - // toPx--; - // else - toPx = toPx2; - - if (toPx >= frPx) - gaps.push([frPx, toPx]); // addGap - } - } - - return gaps; -} - -function pxRoundGen(pxAlign) { - return pxAlign == 0 ? retArg0 : pxAlign == 1 ? round : v => incrRound(v, pxAlign); -} - -function rect(ori) { - let moveTo = ori == 0 ? - moveToH : - moveToV; - - let arcTo = ori == 0 ? - (p, x1, y1, x2, y2, r) => { p.arcTo(x1, y1, x2, y2, r); } : - (p, y1, x1, y2, x2, r) => { p.arcTo(x1, y1, x2, y2, r); }; - - let rect = ori == 0 ? - (p, x, y, w, h) => { p.rect(x, y, w, h); } : - (p, y, x, h, w) => { p.rect(x, y, w, h); }; - - return (p, x, y, w, h, r = 0) => { - if (r == 0) - rect(p, x, y, w, h); - else { - r = min(r, w / 2, h / 2); - - // adapted from https://stackoverflow.com/questions/1255512/how-to-draw-a-rounded-rectangle-using-html-canvas/7838871#7838871 - moveTo(p, x + r, y); - arcTo(p, x + w, y, x + w, y + h, r); - arcTo(p, x + w, y + h, x, y + h, r); - arcTo(p, x, y + h, x, y, r); - arcTo(p, x, y, x + w, y, r); - p.closePath(); - } - }; -} - -// orientation-inverting canvas functions -const moveToH = (p, x, y) => { p.moveTo(x, y); }; -const moveToV = (p, y, x) => { p.moveTo(x, y); }; -const lineToH = (p, x, y) => { p.lineTo(x, y); }; -const lineToV = (p, y, x) => { p.lineTo(x, y); }; -const rectH = rect(0); -const rectV = rect(1); -const arcH = (p, x, y, r, startAngle, endAngle) => { p.arc(x, y, r, startAngle, endAngle); }; -const arcV = (p, y, x, r, startAngle, endAngle) => { p.arc(x, y, r, startAngle, endAngle); }; -const bezierCurveToH = (p, bp1x, bp1y, bp2x, bp2y, p2x, p2y) => { p.bezierCurveTo(bp1x, bp1y, bp2x, bp2y, p2x, p2y); }; -const bezierCurveToV = (p, bp1y, bp1x, bp2y, bp2x, p2y, p2x) => { p.bezierCurveTo(bp1x, bp1y, bp2x, bp2y, p2x, p2y); }; - -// TODO: drawWrap(seriesIdx, drawPoints) (save, restore, translate, clip) -function points(opts) { - return (u, seriesIdx, idx0, idx1, filtIdxs) => { - // log("drawPoints()", arguments); - - return orient(u, seriesIdx, (series, dataX, dataY, scaleX, scaleY, valToPosX, valToPosY, xOff, yOff, xDim, yDim) => { - let { pxRound, points } = series; - - let moveTo, arc; - - if (scaleX.ori == 0) { - moveTo = moveToH; - arc = arcH; - } - else { - moveTo = moveToV; - arc = arcV; - } - - const width = roundDec(points.width * pxRatio, 3); - - let rad = (points.size - points.width) / 2 * pxRatio; - let dia = roundDec(rad * 2, 3); - - let fill = new Path2D(); - let clip = new Path2D(); - - let { left: lft, top: top, width: wid, height: hgt } = u.bbox; - - rectH(clip, - lft - dia, - top - dia, - wid + dia * 2, - hgt + dia * 2, - ); - - const drawPoint = pi => { - if (dataY[pi] != null) { - let x = pxRound(valToPosX(dataX[pi], scaleX, xDim, xOff)); - let y = pxRound(valToPosY(dataY[pi], scaleY, yDim, yOff)); - - moveTo(fill, x + rad, y); - arc(fill, x, y, rad, 0, PI * 2); - } - }; - - if (filtIdxs) - filtIdxs.forEach(drawPoint); - else { - for (let pi = idx0; pi <= idx1; pi++) - drawPoint(pi); - } - - return { - stroke: width > 0 ? fill : null, - fill, - clip, - flags: BAND_CLIP_FILL | BAND_CLIP_STROKE, - }; - }); - }; -} - -function _drawAcc(lineTo) { - return (stroke, accX, minY, maxY, inY, outY) => { - if (minY != maxY) { - if (inY != minY && outY != minY) - lineTo(stroke, accX, minY); - if (inY != maxY && outY != maxY) - lineTo(stroke, accX, maxY); - - lineTo(stroke, accX, outY); - } - }; -} - -const drawAccH = _drawAcc(lineToH); -const drawAccV = _drawAcc(lineToV); - -function linear(opts) { - const alignGaps = ifNull(opts?.alignGaps, 0); - - return (u, seriesIdx, idx0, idx1) => { - return orient(u, seriesIdx, (series, dataX, dataY, scaleX, scaleY, valToPosX, valToPosY, xOff, yOff, xDim, yDim) => { - let pxRound = series.pxRound; - - let pixelForX = val => pxRound(valToPosX(val, scaleX, xDim, xOff)); - let pixelForY = val => pxRound(valToPosY(val, scaleY, yDim, yOff)); - - let lineTo, drawAcc; - - if (scaleX.ori == 0) { - lineTo = lineToH; - drawAcc = drawAccH; - } - else { - lineTo = lineToV; - drawAcc = drawAccV; - } - - const dir = scaleX.dir * (scaleX.ori == 0 ? 1 : -1); - - const _paths = {stroke: new Path2D(), fill: null, clip: null, band: null, gaps: null, flags: BAND_CLIP_FILL}; - const stroke = _paths.stroke; - - let minY = inf, - maxY = -inf, - inY, outY, drawnAtX; - - let accX = pixelForX(dataX[dir == 1 ? idx0 : idx1]); - - // data edges - let lftIdx = nonNullIdx(dataY, idx0, idx1, 1 * dir); - let rgtIdx = nonNullIdx(dataY, idx0, idx1, -1 * dir); - let lftX = pixelForX(dataX[lftIdx]); - let rgtX = pixelForX(dataX[rgtIdx]); - - for (let i = dir == 1 ? idx0 : idx1; i >= idx0 && i <= idx1; i += dir) { - let x = pixelForX(dataX[i]); - - if (x == accX) { - if (dataY[i] != null) { - outY = pixelForY(dataY[i]); - - if (minY == inf) { - lineTo(stroke, x, outY); - inY = outY; - } - - minY = min(outY, minY); - maxY = max(outY, maxY); - } - } - else { - if (minY != inf) { - drawAcc(stroke, accX, minY, maxY, inY, outY); - drawnAtX = accX; - } - - if (dataY[i] != null) { - outY = pixelForY(dataY[i]); - lineTo(stroke, x, outY); - minY = maxY = inY = outY; - } - else { - minY = inf; - maxY = -inf; - } - - accX = x; - } - } - - if (minY != inf && minY != maxY && drawnAtX != accX) - drawAcc(stroke, accX, minY, maxY, inY, outY); - - let [ bandFillDir, bandClipDir ] = bandFillClipDirs(u, seriesIdx); - - if (series.fill != null || bandFillDir != 0) { - let fill = _paths.fill = new Path2D(stroke); - - let fillToVal = series.fillTo(u, seriesIdx, series.min, series.max, bandFillDir); - let fillToY = pixelForY(fillToVal); - - lineTo(fill, rgtX, fillToY); - lineTo(fill, lftX, fillToY); - } - - if (!series.spanGaps) { - // console.time('gaps'); - let gaps = []; - - gaps.push(...findGaps(dataX, dataY, idx0, idx1, dir, pixelForX, alignGaps)); - - // console.timeEnd('gaps'); - - // console.log('gaps', JSON.stringify(gaps)); - - _paths.gaps = gaps = series.gaps(u, seriesIdx, idx0, idx1, gaps); - - _paths.clip = clipGaps(gaps, scaleX.ori, xOff, yOff, xDim, yDim); - } - - if (bandClipDir != 0) { - _paths.band = bandClipDir == 2 ? [ - clipBandLine(u, seriesIdx, idx0, idx1, stroke, -1), - clipBandLine(u, seriesIdx, idx0, idx1, stroke, 1), - ] : clipBandLine(u, seriesIdx, idx0, idx1, stroke, bandClipDir); - } - - return _paths; - }); - }; -} - -function stepped(opts) { - const align = ifNull(opts.align, 1); - // whether to draw ascenders/descenders at null/gap bondaries - const ascDesc = ifNull(opts.ascDesc, false); - - const alignGaps = ifNull(opts.alignGaps, 0); - - return (u, seriesIdx, idx0, idx1) => { - return orient(u, seriesIdx, (series, dataX, dataY, scaleX, scaleY, valToPosX, valToPosY, xOff, yOff, xDim, yDim) => { - let pxRound = series.pxRound; - - let pixelForX = val => pxRound(valToPosX(val, scaleX, xDim, xOff)); - let pixelForY = val => pxRound(valToPosY(val, scaleY, yDim, yOff)); - - let lineTo = scaleX.ori == 0 ? lineToH : lineToV; - - const _paths = {stroke: new Path2D(), fill: null, clip: null, band: null, gaps: null, flags: BAND_CLIP_FILL}; - const stroke = _paths.stroke; - - const dir = scaleX.dir * (scaleX.ori == 0 ? 1 : -1); - - idx0 = nonNullIdx(dataY, idx0, idx1, 1); - idx1 = nonNullIdx(dataY, idx0, idx1, -1); - - let prevYPos = pixelForY(dataY[dir == 1 ? idx0 : idx1]); - let firstXPos = pixelForX(dataX[dir == 1 ? idx0 : idx1]); - let prevXPos = firstXPos; - - lineTo(stroke, firstXPos, prevYPos); - - for (let i = dir == 1 ? idx0 : idx1; i >= idx0 && i <= idx1; i += dir) { - let yVal1 = dataY[i]; - - if (yVal1 == null) - continue; - - let x1 = pixelForX(dataX[i]); - let y1 = pixelForY(yVal1); - - if (align == 1) - lineTo(stroke, x1, prevYPos); - else - lineTo(stroke, prevXPos, y1); - - lineTo(stroke, x1, y1); - - prevYPos = y1; - prevXPos = x1; - } - - let [ bandFillDir, bandClipDir ] = bandFillClipDirs(u, seriesIdx); - - if (series.fill != null || bandFillDir != 0) { - let fill = _paths.fill = new Path2D(stroke); - - let fillTo = series.fillTo(u, seriesIdx, series.min, series.max, bandFillDir); - let fillToY = pixelForY(fillTo); - - lineTo(fill, prevXPos, fillToY); - lineTo(fill, firstXPos, fillToY); - } - - if (!series.spanGaps) { - // console.time('gaps'); - let gaps = []; - - gaps.push(...findGaps(dataX, dataY, idx0, idx1, dir, pixelForX, alignGaps)); - - // console.timeEnd('gaps'); - - // console.log('gaps', JSON.stringify(gaps)); - - // expand/contract clips for ascenders/descenders - let halfStroke = (series.width * pxRatio) / 2; - let startsOffset = (ascDesc || align == 1) ? halfStroke : -halfStroke; - let endsOffset = (ascDesc || align == -1) ? -halfStroke : halfStroke; - - gaps.forEach(g => { - g[0] += startsOffset; - g[1] += endsOffset; - }); - - _paths.gaps = gaps = series.gaps(u, seriesIdx, idx0, idx1, gaps); - - _paths.clip = clipGaps(gaps, scaleX.ori, xOff, yOff, xDim, yDim); - } - - if (bandClipDir != 0) { - _paths.band = bandClipDir == 2 ? [ - clipBandLine(u, seriesIdx, idx0, idx1, stroke, -1), - clipBandLine(u, seriesIdx, idx0, idx1, stroke, 1), - ] : clipBandLine(u, seriesIdx, idx0, idx1, stroke, bandClipDir); - } - - return _paths; - }); - }; -} - -function bars(opts) { - opts = opts || EMPTY_OBJ; - const size = ifNull(opts.size, [0.6, inf, 1]); - const align = opts.align || 0; - const extraGap = (opts.gap || 0) * pxRatio; - - const radius = ifNull(opts.radius, 0); - - const gapFactor = 1 - size[0]; - const maxWidth = ifNull(size[1], inf) * pxRatio; - const minWidth = ifNull(size[2], 1) * pxRatio; - - const disp = ifNull(opts.disp, EMPTY_OBJ); - const _each = ifNull(opts.each, _ => {}); - - const { fill: dispFills, stroke: dispStrokes } = disp; - - return (u, seriesIdx, idx0, idx1) => { - return orient(u, seriesIdx, (series, dataX, dataY, scaleX, scaleY, valToPosX, valToPosY, xOff, yOff, xDim, yDim) => { - let pxRound = series.pxRound; - - const _dirX = scaleX.dir * (scaleX.ori == 0 ? 1 : -1); - const _dirY = scaleY.dir * (scaleY.ori == 1 ? 1 : -1); - - let rect = scaleX.ori == 0 ? rectH : rectV; - - let each = scaleX.ori == 0 ? _each : (u, seriesIdx, i, top, lft, hgt, wid) => { - _each(u, seriesIdx, i, lft, top, wid, hgt); - }; - - let [ bandFillDir, bandClipDir ] = bandFillClipDirs(u, seriesIdx); - - // let fillToY = series.fillTo(u, seriesIdx, series.min, series.max, bandFillDir); - let fillToY = scaleY.distr == 3 ? (bandFillDir == 1 ? scaleY.max : scaleY.min) : 0; - - let y0Pos = valToPosY(fillToY, scaleY, yDim, yOff); - - // barWid is to center of stroke - let xShift, barWid; - - let strokeWidth = pxRound(series.width * pxRatio); - - let multiPath = false; - - let fillColors = null; - let fillPaths = null; - let strokeColors = null; - let strokePaths = null; - - if (dispFills != null && (strokeWidth == 0 || dispStrokes != null)) { - multiPath = true; - - fillColors = dispFills.values(u, seriesIdx, idx0, idx1); - fillPaths = new Map(); - (new Set(fillColors)).forEach(color => { - if (color != null) - fillPaths.set(color, new Path2D()); - }); - - if (strokeWidth > 0) { - strokeColors = dispStrokes.values(u, seriesIdx, idx0, idx1); - strokePaths = new Map(); - (new Set(strokeColors)).forEach(color => { - if (color != null) - strokePaths.set(color, new Path2D()); - }); - } - } - - let { x0, size } = disp; - - if (x0 != null && size != null) { - dataX = x0.values(u, seriesIdx, idx0, idx1); - - if (x0.unit == 2) - dataX = dataX.map(pct => u.posToVal(xOff + pct * xDim, scaleX.key, true)); - - // assumes uniform sizes, for now - let sizes = size.values(u, seriesIdx, idx0, idx1); - - if (size.unit == 2) - barWid = sizes[0] * xDim; - else - barWid = valToPosX(sizes[0], scaleX, xDim, xOff) - valToPosX(0, scaleX, xDim, xOff); // assumes linear scale (delta from 0) - - barWid = pxRound(barWid - strokeWidth); - - xShift = (_dirX == 1 ? -strokeWidth / 2 : barWid + strokeWidth / 2); - } - else { - let colWid = xDim; - - if (dataX.length > 1) { - // prior index with non-undefined y data - let prevIdx = null; - - // scan full dataset for smallest adjacent delta - // will not work properly for non-linear x scales, since does not do expensive valToPosX calcs till end - for (let i = 0, minDelta = Infinity; i < dataX.length; i++) { - if (dataY[i] !== undefined) { - if (prevIdx != null) { - let delta = abs(dataX[i] - dataX[prevIdx]); - - if (delta < minDelta) { - minDelta = delta; - colWid = abs(valToPosX(dataX[i], scaleX, xDim, xOff) - valToPosX(dataX[prevIdx], scaleX, xDim, xOff)); - } - } - - prevIdx = i; - } - } - } - - let gapWid = colWid * gapFactor; - - barWid = pxRound(min(maxWidth, max(minWidth, colWid - gapWid)) - strokeWidth - extraGap); - - xShift = (align == 0 ? barWid / 2 : align == _dirX ? 0 : barWid) - align * _dirX * extraGap / 2; - } - - const _paths = {stroke: null, fill: null, clip: null, band: null, gaps: null, flags: BAND_CLIP_FILL | BAND_CLIP_STROKE}; // disp, geom - - let yLimit; - - if (bandClipDir != 0) { - _paths.band = new Path2D(); - yLimit = pxRound(valToPosY(bandClipDir == 1 ? scaleY.max : scaleY.min, scaleY, yDim, yOff)); - } - - const stroke = multiPath ? null : new Path2D(); - const band = _paths.band; - - let { y0, y1 } = disp; - - let dataY0 = null; - - if (y0 != null && y1 != null) { - dataY = y1.values(u, seriesIdx, idx0, idx1); - dataY0 = y0.values(u, seriesIdx, idx0, idx1); - } - - for (let i = _dirX == 1 ? idx0 : idx1; i >= idx0 && i <= idx1; i += _dirX) { - let yVal = dataY[i]; - - // we can skip both, drawing and band clipping for alignment artifacts - if (yVal === undefined) - continue; - - /* - // interpolate upwards band clips - if (yVal == null) { - // if (hasBands) - // yVal = costlyLerp(i, idx0, idx1, _dirX, dataY); - // else - continue; - } - */ - - let xVal = scaleX.distr != 2 || disp != null ? dataX[i] : i; - - // TODO: all xPos can be pre-computed once for all series in aligned set - let xPos = valToPosX(xVal, scaleX, xDim, xOff); - let yPos = valToPosY(ifNull(yVal, fillToY), scaleY, yDim, yOff); - - if (dataY0 != null && yVal != null) - y0Pos = valToPosY(dataY0[i], scaleY, yDim, yOff); - - let lft = pxRound(xPos - xShift); - let btm = pxRound(max(yPos, y0Pos)); - let top = pxRound(min(yPos, y0Pos)); - // this includes the stroke - let barHgt = btm - top; - - let r = radius * barWid; - - if (yVal != null) { // && yVal != fillToY (0 height bar) - if (multiPath) { - if (strokeWidth > 0 && strokeColors[i] != null) - rect(strokePaths.get(strokeColors[i]), lft, top + floor(strokeWidth / 2), barWid, max(0, barHgt - strokeWidth), r); - - if (fillColors[i] != null) - rect(fillPaths.get(fillColors[i]), lft, top + floor(strokeWidth / 2), barWid, max(0, barHgt - strokeWidth), r); - } - else - rect(stroke, lft, top + floor(strokeWidth / 2), barWid, max(0, barHgt - strokeWidth), r); - - each(u, seriesIdx, i, - lft - strokeWidth / 2, - top, - barWid + strokeWidth, - barHgt, - ); - } - - if (bandClipDir != 0) { - if (_dirY * bandClipDir == 1) { - btm = top; - top = yLimit; - } - else { - top = btm; - btm = yLimit; - } - - barHgt = btm - top; - - rect(band, lft - strokeWidth / 2, top, barWid + strokeWidth, max(0, barHgt), 0); - } - } - - if (strokeWidth > 0) - _paths.stroke = multiPath ? strokePaths : stroke; - - _paths.fill = multiPath ? fillPaths : stroke; - - return _paths; - }); - }; -} - -function splineInterp(interp, opts) { - const alignGaps = ifNull(opts?.alignGaps, 0); - - return (u, seriesIdx, idx0, idx1) => { - return orient(u, seriesIdx, (series, dataX, dataY, scaleX, scaleY, valToPosX, valToPosY, xOff, yOff, xDim, yDim) => { - let pxRound = series.pxRound; - - let pixelForX = val => pxRound(valToPosX(val, scaleX, xDim, xOff)); - let pixelForY = val => pxRound(valToPosY(val, scaleY, yDim, yOff)); - - let moveTo, bezierCurveTo, lineTo; - - if (scaleX.ori == 0) { - moveTo = moveToH; - lineTo = lineToH; - bezierCurveTo = bezierCurveToH; - } - else { - moveTo = moveToV; - lineTo = lineToV; - bezierCurveTo = bezierCurveToV; - } - - const dir = scaleX.dir * (scaleX.ori == 0 ? 1 : -1); - - idx0 = nonNullIdx(dataY, idx0, idx1, 1); - idx1 = nonNullIdx(dataY, idx0, idx1, -1); - - let firstXPos = pixelForX(dataX[dir == 1 ? idx0 : idx1]); - let prevXPos = firstXPos; - - let xCoords = []; - let yCoords = []; - - for (let i = dir == 1 ? idx0 : idx1; i >= idx0 && i <= idx1; i += dir) { - let yVal = dataY[i]; - - if (yVal != null) { - let xVal = dataX[i]; - let xPos = pixelForX(xVal); - - xCoords.push(prevXPos = xPos); - yCoords.push(pixelForY(dataY[i])); - } - } - - const _paths = {stroke: interp(xCoords, yCoords, moveTo, lineTo, bezierCurveTo, pxRound), fill: null, clip: null, band: null, gaps: null, flags: BAND_CLIP_FILL}; - const stroke = _paths.stroke; - - let [ bandFillDir, bandClipDir ] = bandFillClipDirs(u, seriesIdx); - - if (series.fill != null || bandFillDir != 0) { - let fill = _paths.fill = new Path2D(stroke); - - let fillTo = series.fillTo(u, seriesIdx, series.min, series.max, bandFillDir); - let fillToY = pixelForY(fillTo); - - lineTo(fill, prevXPos, fillToY); - lineTo(fill, firstXPos, fillToY); - } - - if (!series.spanGaps) { - // console.time('gaps'); - let gaps = []; - - gaps.push(...findGaps(dataX, dataY, idx0, idx1, dir, pixelForX, alignGaps)); - - // console.timeEnd('gaps'); - - // console.log('gaps', JSON.stringify(gaps)); - - _paths.gaps = gaps = series.gaps(u, seriesIdx, idx0, idx1, gaps); - - _paths.clip = clipGaps(gaps, scaleX.ori, xOff, yOff, xDim, yDim); - } - - if (bandClipDir != 0) { - _paths.band = bandClipDir == 2 ? [ - clipBandLine(u, seriesIdx, idx0, idx1, stroke, -1), - clipBandLine(u, seriesIdx, idx0, idx1, stroke, 1), - ] : clipBandLine(u, seriesIdx, idx0, idx1, stroke, bandClipDir); - } - - return _paths; - - // if FEAT_PATHS: false in rollup.config.js - // u.ctx.save(); - // u.ctx.beginPath(); - // u.ctx.rect(u.bbox.left, u.bbox.top, u.bbox.width, u.bbox.height); - // u.ctx.clip(); - // u.ctx.strokeStyle = u.series[sidx].stroke; - // u.ctx.stroke(stroke); - // u.ctx.fillStyle = u.series[sidx].fill; - // u.ctx.fill(fill); - // u.ctx.restore(); - // return null; - }); - }; -} - -function monotoneCubic(opts) { - return splineInterp(_monotoneCubic, opts); -} - -// Monotone Cubic Spline interpolation, adapted from the Chartist.js implementation: -// https://github.com/gionkunz/chartist-js/blob/e7e78201bffe9609915e5e53cfafa29a5d6c49f9/src/scripts/interpolation.js#L240-L369 -function _monotoneCubic(xs, ys, moveTo, lineTo, bezierCurveTo, pxRound) { - const n = xs.length; - - if (n < 2) - return null; - - const path = new Path2D(); - - moveTo(path, xs[0], ys[0]); - - if (n == 2) - lineTo(path, xs[1], ys[1]); - else { - let ms = Array(n), - ds = Array(n - 1), - dys = Array(n - 1), - dxs = Array(n - 1); - - // calc deltas and derivative - for (let i = 0; i < n - 1; i++) { - dys[i] = ys[i + 1] - ys[i]; - dxs[i] = xs[i + 1] - xs[i]; - ds[i] = dys[i] / dxs[i]; - } - - // determine desired slope (m) at each point using Fritsch-Carlson method - // http://math.stackexchange.com/questions/45218/implementation-of-monotone-cubic-interpolation - ms[0] = ds[0]; - - for (let i = 1; i < n - 1; i++) { - if (ds[i] === 0 || ds[i - 1] === 0 || (ds[i - 1] > 0) !== (ds[i] > 0)) - ms[i] = 0; - else { - ms[i] = 3 * (dxs[i - 1] + dxs[i]) / ( - (2 * dxs[i] + dxs[i - 1]) / ds[i - 1] + - (dxs[i] + 2 * dxs[i - 1]) / ds[i] - ); - - if (!isFinite(ms[i])) - ms[i] = 0; - } - } - - ms[n - 1] = ds[n - 2]; - - for (let i = 0; i < n - 1; i++) { - bezierCurveTo( - path, - xs[i] + dxs[i] / 3, - ys[i] + ms[i] * dxs[i] / 3, - xs[i + 1] - dxs[i] / 3, - ys[i + 1] - ms[i + 1] * dxs[i] / 3, - xs[i + 1], - ys[i + 1], - ); - } - } - - return path; -} - -const cursorPlots = new Set(); - -function invalidateRects() { - cursorPlots.forEach(u => { - u.syncRect(true); - }); -} - -if (domEnv) { - on(resize, win, invalidateRects); - on(scroll, win, invalidateRects, true); - on(dppxchange, win, () => { uPlot.pxRatio = pxRatio; }); -} - -const linearPath = linear() ; -const pointsPath = points() ; - -function setDefaults(d, xo, yo, initY) { - let d2 = initY ? [d[0], d[1]].concat(d.slice(2)) : [d[0]].concat(d.slice(1)); - return d2.map((o, i) => setDefault(o, i, xo, yo)); -} - -function setDefaults2(d, xyo) { - return d.map((o, i) => i == 0 ? null : assign({}, xyo, o)); // todo: assign() will not merge facet arrays -} - -function setDefault(o, i, xo, yo) { - return assign({}, (i == 0 ? xo : yo), o); -} - -function snapNumX(self, dataMin, dataMax) { - return dataMin == null ? nullNullTuple : [dataMin, dataMax]; -} - -const snapTimeX = snapNumX; - -// this ensures that non-temporal/numeric y-axes get multiple-snapped padding added above/below -// TODO: also account for incrs when snapping to ensure top of axis gets a tick & value -function snapNumY(self, dataMin, dataMax) { - return dataMin == null ? nullNullTuple : rangeNum(dataMin, dataMax, rangePad, true); -} - -function snapLogY(self, dataMin, dataMax, scale) { - return dataMin == null ? nullNullTuple : rangeLog(dataMin, dataMax, self.scales[scale].log, false); -} - -const snapLogX = snapLogY; - -function snapAsinhY(self, dataMin, dataMax, scale) { - return dataMin == null ? nullNullTuple : rangeAsinh(dataMin, dataMax, self.scales[scale].log, false); -} - -const snapAsinhX = snapAsinhY; - -// dim is logical (getClientBoundingRect) pixels, not canvas pixels -function findIncr(minVal, maxVal, incrs, dim, minSpace) { - let intDigits = max(numIntDigits(minVal), numIntDigits(maxVal)); - - let delta = maxVal - minVal; - - let incrIdx = closestIdx((minSpace / dim) * delta, incrs); - - do { - let foundIncr = incrs[incrIdx]; - let foundSpace = dim * foundIncr / delta; - - if (foundSpace >= minSpace && intDigits + (foundIncr < 5 ? fixedDec.get(foundIncr) : 0) <= 17) - return [foundIncr, foundSpace]; - } while (++incrIdx < incrs.length); - - return [0, 0]; -} - -function pxRatioFont(font) { - let fontSize, fontSizeCss; - font = font.replace(/(\d+)px/, (m, p1) => (fontSize = round((fontSizeCss = +p1) * pxRatio)) + 'px'); - return [font, fontSize, fontSizeCss]; -} - -function syncFontSize(axis) { - if (axis.show) { - [axis.font, axis.labelFont].forEach(f => { - let size = roundDec(f[2] * pxRatio, 1); - f[0] = f[0].replace(/[0-9.]+px/, size + 'px'); - f[1] = size; - }); - } -} - -function uPlot(opts, data, then) { - const self = { - mode: ifNull(opts.mode, 1), - }; - - const mode = self.mode; - - // TODO: cache denoms & mins scale.cache = {r, min, } - function getValPct(val, scale) { - let _val = ( - scale.distr == 3 ? log10(val > 0 ? val : scale.clamp(self, val, scale.min, scale.max, scale.key)) : - scale.distr == 4 ? asinh(val, scale.asinh) : - val - ); - - return (_val - scale._min) / (scale._max - scale._min); - } - - function getHPos(val, scale, dim, off) { - let pct = getValPct(val, scale); - return off + dim * (scale.dir == -1 ? (1 - pct) : pct); - } - - function getVPos(val, scale, dim, off) { - let pct = getValPct(val, scale); - return off + dim * (scale.dir == -1 ? pct : (1 - pct)); - } - - function getPos(val, scale, dim, off) { - return scale.ori == 0 ? getHPos(val, scale, dim, off) : getVPos(val, scale, dim, off); - } - - self.valToPosH = getHPos; - self.valToPosV = getVPos; - - let ready = false; - self.status = 0; - - const root = self.root = placeDiv(UPLOT); - - if (opts.id != null) - root.id = opts.id; - - addClass(root, opts.class); - - if (opts.title) { - let title = placeDiv(TITLE, root); - title.textContent = opts.title; - } - - const can = placeTag("canvas"); - const ctx = self.ctx = can.getContext("2d"); - - const wrap = placeDiv(WRAP, root); - const under = self.under = placeDiv(UNDER, wrap); - wrap.appendChild(can); - const over = self.over = placeDiv(OVER, wrap); - - opts = copy(opts); - - const pxAlign = +ifNull(opts.pxAlign, 1); - - const pxRound = pxRoundGen(pxAlign); - - (opts.plugins || []).forEach(p => { - if (p.opts) - opts = p.opts(self, opts) || opts; - }); - - const ms = opts.ms || 1e-3; - - const series = self.series = mode == 1 ? - setDefaults(opts.series || [], xSeriesOpts, ySeriesOpts, false) : - setDefaults2(opts.series || [null], xySeriesOpts); - const axes = self.axes = setDefaults(opts.axes || [], xAxisOpts, yAxisOpts, true); - const scales = self.scales = {}; - const bands = self.bands = opts.bands || []; - - bands.forEach(b => { - b.fill = fnOrSelf(b.fill || null); - b.dir = ifNull(b.dir, -1); - }); - - const xScaleKey = mode == 2 ? series[1].facets[0].scale : series[0].scale; - - const drawOrderMap = { - axes: drawAxesGrid, - series: drawSeries, - }; - - const drawOrder = (opts.drawOrder || ["axes", "series"]).map(key => drawOrderMap[key]); - - function initScale(scaleKey) { - let sc = scales[scaleKey]; - - if (sc == null) { - let scaleOpts = (opts.scales || EMPTY_OBJ)[scaleKey] || EMPTY_OBJ; - - if (scaleOpts.from != null) { - // ensure parent is initialized - initScale(scaleOpts.from); - // dependent scales inherit - scales[scaleKey] = assign({}, scales[scaleOpts.from], scaleOpts, {key: scaleKey}); - } - else { - sc = scales[scaleKey] = assign({}, (scaleKey == xScaleKey ? xScaleOpts : yScaleOpts), scaleOpts); - - sc.key = scaleKey; - - let isTime = sc.time; - - let rn = sc.range; - - let rangeIsArr = isArr(rn); - - if (scaleKey != xScaleKey || (mode == 2 && !isTime)) { - // if range array has null limits, it should be auto - if (rangeIsArr && (rn[0] == null || rn[1] == null)) { - rn = { - min: rn[0] == null ? autoRangePart : { - mode: 1, - hard: rn[0], - soft: rn[0], - }, - max: rn[1] == null ? autoRangePart : { - mode: 1, - hard: rn[1], - soft: rn[1], - }, - }; - rangeIsArr = false; - } - - if (!rangeIsArr && isObj(rn)) { - let cfg = rn; - // this is similar to snapNumY - rn = (self, dataMin, dataMax) => dataMin == null ? nullNullTuple : rangeNum(dataMin, dataMax, cfg); - } - } - - sc.range = fnOrSelf(rn || (isTime ? snapTimeX : scaleKey == xScaleKey ? - (sc.distr == 3 ? snapLogX : sc.distr == 4 ? snapAsinhX : snapNumX) : - (sc.distr == 3 ? snapLogY : sc.distr == 4 ? snapAsinhY : snapNumY) - )); - - sc.auto = fnOrSelf(rangeIsArr ? false : sc.auto); - - sc.clamp = fnOrSelf(sc.clamp || clampScale); - - // caches for expensive ops like asinh() & log() - sc._min = sc._max = null; - } - } - } - - initScale("x"); - initScale("y"); - - // TODO: init scales from facets in mode: 2 - if (mode == 1) { - series.forEach(s => { - initScale(s.scale); - }); - } - - axes.forEach(a => { - initScale(a.scale); - }); - - for (let k in opts.scales) - initScale(k); - - const scaleX = scales[xScaleKey]; - - const xScaleDistr = scaleX.distr; - - let valToPosX, valToPosY; - - if (scaleX.ori == 0) { - addClass(root, ORI_HZ); - valToPosX = getHPos; - valToPosY = getVPos; - /* - updOriDims = () => { - xDimCan = plotWid; - xOffCan = plotLft; - yDimCan = plotHgt; - yOffCan = plotTop; - - xDimCss = plotWidCss; - xOffCss = plotLftCss; - yDimCss = plotHgtCss; - yOffCss = plotTopCss; - }; - */ - } - else { - addClass(root, ORI_VT); - valToPosX = getVPos; - valToPosY = getHPos; - /* - updOriDims = () => { - xDimCan = plotHgt; - xOffCan = plotTop; - yDimCan = plotWid; - yOffCan = plotLft; - - xDimCss = plotHgtCss; - xOffCss = plotTopCss; - yDimCss = plotWidCss; - yOffCss = plotLftCss; - }; - */ - } - - const pendScales = {}; - - // explicitly-set initial scales - for (let k in scales) { - let sc = scales[k]; - - if (sc.min != null || sc.max != null) { - pendScales[k] = {min: sc.min, max: sc.max}; - sc.min = sc.max = null; - } - } - -// self.tz = opts.tz || Intl.DateTimeFormat().resolvedOptions().timeZone; - const _tzDate = (opts.tzDate || (ts => new Date(round(ts / ms)))); - const _fmtDate = (opts.fmtDate || fmtDate); - - const _timeAxisSplits = (ms == 1 ? timeAxisSplitsMs(_tzDate) : timeAxisSplitsS(_tzDate)); - const _timeAxisVals = timeAxisVals(_tzDate, timeAxisStamps((ms == 1 ? _timeAxisStampsMs : _timeAxisStampsS), _fmtDate)); - const _timeSeriesVal = timeSeriesVal(_tzDate, timeSeriesStamp(_timeSeriesStamp, _fmtDate)); - - const activeIdxs = []; - - const legend = (self.legend = assign({}, legendOpts, opts.legend)); - const showLegend = legend.show; - const markers = legend.markers; - - { - legend.idxs = activeIdxs; - - markers.width = fnOrSelf(markers.width); - markers.dash = fnOrSelf(markers.dash); - markers.stroke = fnOrSelf(markers.stroke); - markers.fill = fnOrSelf(markers.fill); - } - - let legendEl; - let legendRows = []; - let legendCells = []; - let legendCols; - let multiValLegend = false; - let NULL_LEGEND_VALUES = {}; - - if (legend.live) { - const getMultiVals = series[1] ? series[1].values : null; - multiValLegend = getMultiVals != null; - legendCols = multiValLegend ? getMultiVals(self, 1, 0) : {_: 0}; - - for (let k in legendCols) - NULL_LEGEND_VALUES[k] = "--"; - } - - if (showLegend) { - legendEl = placeTag("table", LEGEND, root); - - if (multiValLegend) { - let head = placeTag("tr", LEGEND_THEAD, legendEl); - placeTag("th", null, head); - - for (var key in legendCols) - placeTag("th", LEGEND_LABEL, head).textContent = key; - } - else { - addClass(legendEl, LEGEND_INLINE); - legend.live && addClass(legendEl, LEGEND_LIVE); - } - } - - const son = {show: true}; - const soff = {show: false}; - - function initLegendRow(s, i) { - if (i == 0 && (multiValLegend || !legend.live || mode == 2)) - return nullNullTuple; - - let cells = []; - - let row = placeTag("tr", LEGEND_SERIES, legendEl, legendEl.childNodes[i]); - - addClass(row, s.class); - - if (!s.show) - addClass(row, OFF); - - let label = placeTag("th", null, row); - - if (markers.show) { - let indic = placeDiv(LEGEND_MARKER, label); - - if (i > 0) { - let width = markers.width(self, i); - - if (width) - indic.style.border = width + "px " + markers.dash(self, i) + " " + markers.stroke(self, i); - - indic.style.background = markers.fill(self, i); - } - } - - let text = placeDiv(LEGEND_LABEL, label); - text.textContent = s.label; - - if (i > 0) { - if (!markers.show) - text.style.color = s.width > 0 ? markers.stroke(self, i) : markers.fill(self, i); - - onMouse("click", label, e => { - if (cursor._lock) - return; - - let seriesIdx = series.indexOf(s); - - if ((e.ctrlKey || e.metaKey) != legend.isolate) { - // if any other series is shown, isolate this one. else show all - let isolate = series.some((s, i) => i > 0 && i != seriesIdx && s.show); - - series.forEach((s, i) => { - i > 0 && setSeries(i, isolate ? (i == seriesIdx ? son : soff) : son, true, syncOpts.setSeries); - }); - } - else - setSeries(seriesIdx, {show: !s.show}, true, syncOpts.setSeries); - }); - - if (cursorFocus) { - onMouse(mouseenter, label, e => { - if (cursor._lock) - return; - - setSeries(series.indexOf(s), FOCUS_TRUE, true, syncOpts.setSeries); - }); - } - } - - for (var key in legendCols) { - let v = placeTag("td", LEGEND_VALUE, row); - v.textContent = "--"; - cells.push(v); - } - - return [row, cells]; - } - - const mouseListeners = new Map(); - - function onMouse(ev, targ, fn) { - const targListeners = mouseListeners.get(targ) || {}; - const listener = cursor.bind[ev](self, targ, fn); - - if (listener) { - on(ev, targ, targListeners[ev] = listener); - mouseListeners.set(targ, targListeners); - } - } - - function offMouse(ev, targ, fn) { - const targListeners = mouseListeners.get(targ) || {}; - - for (let k in targListeners) { - if (ev == null || k == ev) { - off(k, targ, targListeners[k]); - delete targListeners[k]; - } - } - - if (ev == null) - mouseListeners.delete(targ); - } - - let fullWidCss = 0; - let fullHgtCss = 0; - - let plotWidCss = 0; - let plotHgtCss = 0; - - // plot margins to account for axes - let plotLftCss = 0; - let plotTopCss = 0; - - let plotLft = 0; - let plotTop = 0; - let plotWid = 0; - let plotHgt = 0; - - self.bbox = {}; - - let shouldSetScales = false; - let shouldSetSize = false; - let shouldConvergeSize = false; - let shouldSetCursor = false; - let shouldSetLegend = false; - - function _setSize(width, height, force) { - if (force || (width != self.width || height != self.height)) - calcSize(width, height); - - resetYSeries(false); - - shouldConvergeSize = true; - shouldSetSize = true; - shouldSetCursor = shouldSetLegend = cursor.left >= 0; - commit(); - } - - function calcSize(width, height) { - // log("calcSize()", arguments); - - self.width = fullWidCss = plotWidCss = width; - self.height = fullHgtCss = plotHgtCss = height; - plotLftCss = plotTopCss = 0; - - calcPlotRect(); - calcAxesRects(); - - let bb = self.bbox; - - plotLft = bb.left = incrRound(plotLftCss * pxRatio, 0.5); - plotTop = bb.top = incrRound(plotTopCss * pxRatio, 0.5); - plotWid = bb.width = incrRound(plotWidCss * pxRatio, 0.5); - plotHgt = bb.height = incrRound(plotHgtCss * pxRatio, 0.5); - - // updOriDims(); - } - - // ensures size calc convergence - const CYCLE_LIMIT = 3; - - function convergeSize() { - let converged = false; - - let cycleNum = 0; - - while (!converged) { - cycleNum++; - - let axesConverged = axesCalc(cycleNum); - let paddingConverged = paddingCalc(cycleNum); - - converged = cycleNum == CYCLE_LIMIT || (axesConverged && paddingConverged); - - if (!converged) { - calcSize(self.width, self.height); - shouldSetSize = true; - } - } - } - - function setSize({width, height}) { - _setSize(width, height); - } - - self.setSize = setSize; - - // accumulate axis offsets, reduce canvas width - function calcPlotRect() { - // easements for edge labels - let hasTopAxis = false; - let hasBtmAxis = false; - let hasRgtAxis = false; - let hasLftAxis = false; - - axes.forEach((axis, i) => { - if (axis.show && axis._show) { - let {side, _size} = axis; - let isVt = side % 2; - let labelSize = axis.label != null ? axis.labelSize : 0; - - let fullSize = _size + labelSize; - - if (fullSize > 0) { - if (isVt) { - plotWidCss -= fullSize; - - if (side == 3) { - plotLftCss += fullSize; - hasLftAxis = true; - } - else - hasRgtAxis = true; - } - else { - plotHgtCss -= fullSize; - - if (side == 0) { - plotTopCss += fullSize; - hasTopAxis = true; - } - else - hasBtmAxis = true; - } - } - } - }); - - sidesWithAxes[0] = hasTopAxis; - sidesWithAxes[1] = hasRgtAxis; - sidesWithAxes[2] = hasBtmAxis; - sidesWithAxes[3] = hasLftAxis; - - // hz padding - plotWidCss -= _padding[1] + _padding[3]; - plotLftCss += _padding[3]; - - // vt padding - plotHgtCss -= _padding[2] + _padding[0]; - plotTopCss += _padding[0]; - } - - function calcAxesRects() { - // will accum + - let off1 = plotLftCss + plotWidCss; - let off2 = plotTopCss + plotHgtCss; - // will accum - - let off3 = plotLftCss; - let off0 = plotTopCss; - - function incrOffset(side, size) { - switch (side) { - case 1: off1 += size; return off1 - size; - case 2: off2 += size; return off2 - size; - case 3: off3 -= size; return off3 + size; - case 0: off0 -= size; return off0 + size; - } - } - - axes.forEach((axis, i) => { - if (axis.show && axis._show) { - let side = axis.side; - - axis._pos = incrOffset(side, axis._size); - - if (axis.label != null) - axis._lpos = incrOffset(side, axis.labelSize); - } - }); - } - - const cursor = (self.cursor = assign({}, cursorOpts, {drag: {y: mode == 2}}, opts.cursor)); - - { - cursor.idxs = activeIdxs; - - cursor._lock = false; - - let points = cursor.points; - - points.show = fnOrSelf(points.show); - points.size = fnOrSelf(points.size); - points.stroke = fnOrSelf(points.stroke); - points.width = fnOrSelf(points.width); - points.fill = fnOrSelf(points.fill); - } - - const focus = self.focus = assign({}, opts.focus || {alpha: 0.3}, cursor.focus); - const cursorFocus = focus.prox >= 0; - - // series-intersection markers - let cursorPts = [null]; - - function initCursorPt(s, si) { - if (si > 0) { - let pt = cursor.points.show(self, si); - - if (pt) { - addClass(pt, CURSOR_PT); - addClass(pt, s.class); - elTrans(pt, -10, -10, plotWidCss, plotHgtCss); - over.insertBefore(pt, cursorPts[si]); - - return pt; - } - } - } - - function initSeries(s, i) { - if (mode == 1 || i > 0) { - let isTime = mode == 1 && scales[s.scale].time; - - let sv = s.value; - s.value = isTime ? (isStr(sv) ? timeSeriesVal(_tzDate, timeSeriesStamp(sv, _fmtDate)) : sv || _timeSeriesVal) : sv || numSeriesVal; - s.label = s.label || (isTime ? timeSeriesLabel : numSeriesLabel); - } - - if (i > 0) { - s.width = s.width == null ? 1 : s.width; - s.paths = s.paths || linearPath || retNull; - s.fillTo = fnOrSelf(s.fillTo || seriesFillTo); - s.pxAlign = +ifNull(s.pxAlign, pxAlign); - s.pxRound = pxRoundGen(s.pxAlign); - - s.stroke = fnOrSelf(s.stroke || null); - s.fill = fnOrSelf(s.fill || null); - s._stroke = s._fill = s._paths = s._focus = null; - - let _ptDia = ptDia(s.width, 1); - let points = s.points = assign({}, { - size: _ptDia, - width: max(1, _ptDia * .2), - stroke: s.stroke, - space: _ptDia * 2, - paths: pointsPath, - _stroke: null, - _fill: null, - }, s.points); - points.show = fnOrSelf(points.show); - points.filter = fnOrSelf(points.filter); - points.fill = fnOrSelf(points.fill); - points.stroke = fnOrSelf(points.stroke); - points.paths = fnOrSelf(points.paths); - points.pxAlign = s.pxAlign; - } - - if (showLegend) { - let rowCells = initLegendRow(s, i); - legendRows.splice(i, 0, rowCells[0]); - legendCells.splice(i, 0, rowCells[1]); - legend.values.push(null); // NULL_LEGEND_VALS not yet avil here :( - } - - if (cursor.show) { - activeIdxs.splice(i, 0, null); - - let pt = initCursorPt(s, i); - pt && cursorPts.splice(i, 0, pt); - } - - fire("addSeries", i); - } - - function addSeries(opts, si) { - si = si == null ? series.length : si; - - opts = setDefault(opts, si, xSeriesOpts, ySeriesOpts); - series.splice(si, 0, opts); - initSeries(series[si], si); - } - - self.addSeries = addSeries; - - function delSeries(i) { - series.splice(i, 1); - - if (showLegend) { - legend.values.splice(i, 1); - - legendCells.splice(i, 1); - let tr = legendRows.splice(i, 1)[0]; - offMouse(null, tr.firstChild); - tr.remove(); - } - - if (cursor.show) { - activeIdxs.splice(i, 1); - - cursorPts.length > 1 && cursorPts.splice(i, 1)[0].remove(); - } - - // TODO: de-init no-longer-needed scales? - - fire("delSeries", i); - } - - self.delSeries = delSeries; - - const sidesWithAxes = [false, false, false, false]; - - function initAxis(axis, i) { - axis._show = axis.show; - - if (axis.show) { - let isVt = axis.side % 2; - - let sc = scales[axis.scale]; - - // this can occur if all series specify non-default scales - if (sc == null) { - axis.scale = isVt ? series[1].scale : xScaleKey; - sc = scales[axis.scale]; - } - - // also set defaults for incrs & values based on axis distr - let isTime = sc.time; - - axis.size = fnOrSelf(axis.size); - axis.space = fnOrSelf(axis.space); - axis.rotate = fnOrSelf(axis.rotate); - axis.incrs = fnOrSelf(axis.incrs || ( sc.distr == 2 ? wholeIncrs : (isTime ? (ms == 1 ? timeIncrsMs : timeIncrsS) : numIncrs))); - axis.splits = fnOrSelf(axis.splits || (isTime && sc.distr == 1 ? _timeAxisSplits : sc.distr == 3 ? logAxisSplits : sc.distr == 4 ? asinhAxisSplits : numAxisSplits)); - - axis.stroke = fnOrSelf(axis.stroke); - axis.grid.stroke = fnOrSelf(axis.grid.stroke); - axis.ticks.stroke = fnOrSelf(axis.ticks.stroke); - axis.border.stroke = fnOrSelf(axis.border.stroke); - - let av = axis.values; - - axis.values = ( - // static array of tick values - isArr(av) && !isArr(av[0]) ? fnOrSelf(av) : - // temporal - isTime ? ( - // config array of fmtDate string tpls - isArr(av) ? - timeAxisVals(_tzDate, timeAxisStamps(av, _fmtDate)) : - // fmtDate string tpl - isStr(av) ? - timeAxisVal(_tzDate, av) : - av || _timeAxisVals - ) : av || numAxisVals - ); - - axis.filter = fnOrSelf(axis.filter || ( sc.distr >= 3 ? logAxisValsFilt : retArg1)); - - axis.font = pxRatioFont(axis.font); - axis.labelFont = pxRatioFont(axis.labelFont); - - axis._size = axis.size(self, null, i, 0); - - axis._space = - axis._rotate = - axis._incrs = - axis._found = // foundIncrSpace - axis._splits = - axis._values = null; - - if (axis._size > 0) { - sidesWithAxes[i] = true; - axis._el = placeDiv(AXIS, wrap); - } - - // debug - // axis._el.style.background = "#" + Math.floor(Math.random()*16777215).toString(16) + '80'; - } - } - - function autoPadSide(self, side, sidesWithAxes, cycleNum) { - let [hasTopAxis, hasRgtAxis, hasBtmAxis, hasLftAxis] = sidesWithAxes; - - let ori = side % 2; - let size = 0; - - if (ori == 0 && (hasLftAxis || hasRgtAxis)) - size = (side == 0 && !hasTopAxis || side == 2 && !hasBtmAxis ? round(xAxisOpts.size / 3) : 0); - if (ori == 1 && (hasTopAxis || hasBtmAxis)) - size = (side == 1 && !hasRgtAxis || side == 3 && !hasLftAxis ? round(yAxisOpts.size / 2) : 0); - - return size; - } - - const padding = self.padding = (opts.padding || [autoPadSide,autoPadSide,autoPadSide,autoPadSide]).map(p => fnOrSelf(ifNull(p, autoPadSide))); - const _padding = self._padding = padding.map((p, i) => p(self, i, sidesWithAxes, 0)); - - let dataLen; - - // rendered data window - let i0 = null; - let i1 = null; - const idxs = mode == 1 ? series[0].idxs : null; - - let data0 = null; - - let viaAutoScaleX = false; - - function setData(_data, _resetScales) { - data = _data == null ? [] : copy(_data, fastIsObj); - - if (mode == 2) { - dataLen = 0; - for (let i = 1; i < series.length; i++) - dataLen += data[i][0].length; - self.data = data = _data; - } - else { - if (data[0] == null) - data[0] = []; - - self.data = data.slice(); - - data0 = data[0]; - dataLen = data0.length; - - if (xScaleDistr == 2) { - data[0] = Array(dataLen); - for (let i = 0; i < dataLen; i++) - data[0][i] = i; - } - } - - self._data = data; - - resetYSeries(true); - - fire("setData"); - - // forces x axis tick values to re-generate when neither x scale nor y scale changes - // in ordinal mode, scale range is by index, so will not change if new data has same length, but tick values are from data - if (xScaleDistr == 2) { - shouldConvergeSize = true; - - /* or somewhat cheaper, and uglier: - if (ready) { - // logic extracted from axesCalc() - let i = 0; - let axis = axes[i]; - let _splits = axis._splits.map(i => data0[i]); - let [_incr, _space] = axis._found; - let incr = data0[_splits[1]] - data0[_splits[0]]; - axis._values = axis.values(self, axis.filter(self, _splits, i, _space, incr), i, _space, incr); - } - */ - } - - if (_resetScales !== false) { - let xsc = scaleX; - - if (xsc.auto(self, viaAutoScaleX)) - autoScaleX(); - else - _setScale(xScaleKey, xsc.min, xsc.max); - - shouldSetCursor = cursor.left >= 0; - shouldSetLegend = true; - commit(); - } - } - - self.setData = setData; - - function autoScaleX() { - viaAutoScaleX = true; - - let _min, _max; - - if (mode == 1) { - if (dataLen > 0) { - i0 = idxs[0] = 0; - i1 = idxs[1] = dataLen - 1; - - _min = data[0][i0]; - _max = data[0][i1]; - - if (xScaleDistr == 2) { - _min = i0; - _max = i1; - } - else if (dataLen == 1) { - if (xScaleDistr == 3) - [_min, _max] = rangeLog(_min, _min, scaleX.log, false); - else if (xScaleDistr == 4) - [_min, _max] = rangeAsinh(_min, _min, scaleX.log, false); - else if (scaleX.time) - _max = _min + round(86400 / ms); - else - [_min, _max] = rangeNum(_min, _max, rangePad, true); - } - } - else { - i0 = idxs[0] = _min = null; - i1 = idxs[1] = _max = null; - } - } - - _setScale(xScaleKey, _min, _max); - } - - let ctxStroke, ctxFill, ctxWidth, ctxDash, ctxJoin, ctxCap, ctxFont, ctxAlign, ctxBaseline; - let ctxAlpha; - - function setCtxStyle(stroke = transparent, width, dash = EMPTY_ARR, cap = "butt", fill = transparent, join = "round") { - if (stroke != ctxStroke) - ctx.strokeStyle = ctxStroke = stroke; - if (fill != ctxFill) - ctx.fillStyle = ctxFill = fill; - if (width != ctxWidth) - ctx.lineWidth = ctxWidth = width; - if (join != ctxJoin) - ctx.lineJoin = ctxJoin = join; - if (cap != ctxCap) - ctx.lineCap = ctxCap = cap; // (‿|‿) - if (dash != ctxDash) - ctx.setLineDash(ctxDash = dash); - } - - function setFontStyle(font, fill, align, baseline) { - if (fill != ctxFill) - ctx.fillStyle = ctxFill = fill; - if (font != ctxFont) - ctx.font = ctxFont = font; - if (align != ctxAlign) - ctx.textAlign = ctxAlign = align; - if (baseline != ctxBaseline) - ctx.textBaseline = ctxBaseline = baseline; - } - - function accScale(wsc, psc, facet, data, sorted = 0) { - if (data.length > 0 && wsc.auto(self, viaAutoScaleX) && (psc == null || psc.min == null)) { - let _i0 = ifNull(i0, 0); - let _i1 = ifNull(i1, data.length - 1); - - // only run getMinMax() for invalidated series data, else reuse - let minMax = facet.min == null ? (wsc.distr == 3 ? getMinMaxLog(data, _i0, _i1) : getMinMax(data, _i0, _i1, sorted)) : [facet.min, facet.max]; - - // initial min/max - wsc.min = min(wsc.min, facet.min = minMax[0]); - wsc.max = max(wsc.max, facet.max = minMax[1]); - } - } - - function setScales() { - // log("setScales()", arguments); - - // wip scales - let wipScales = copy(scales, fastIsObj); - - for (let k in wipScales) { - let wsc = wipScales[k]; - let psc = pendScales[k]; - - if (psc != null && psc.min != null) { - assign(wsc, psc); - - // explicitly setting the x-scale invalidates everything (acts as redraw) - if (k == xScaleKey) - resetYSeries(true); - } - else if (k != xScaleKey || mode == 2) { - if (dataLen == 0 && wsc.from == null) { - let minMax = wsc.range(self, null, null, k); - wsc.min = minMax[0]; - wsc.max = minMax[1]; - } - else { - wsc.min = inf; - wsc.max = -inf; - } - } - } - - if (dataLen > 0) { - // pre-range y-scales from y series' data values - series.forEach((s, i) => { - if (mode == 1) { - let k = s.scale; - let wsc = wipScales[k]; - let psc = pendScales[k]; - - if (i == 0) { - let minMax = wsc.range(self, wsc.min, wsc.max, k); - - wsc.min = minMax[0]; - wsc.max = minMax[1]; - - i0 = closestIdx(wsc.min, data[0]); - i1 = closestIdx(wsc.max, data[0]); - - // closest indices can be outside of view - if (data[0][i0] < wsc.min) - i0++; - if (data[0][i1] > wsc.max) - i1--; - - s.min = data0[i0]; - s.max = data0[i1]; - } - else if (s.show && s.auto) - accScale(wsc, psc, s, data[i], s.sorted); - - s.idxs[0] = i0; - s.idxs[1] = i1; - } - else { - if (i > 0) { - if (s.show && s.auto) { - // TODO: only handles, assumes and requires facets[0] / 'x' scale, and facets[1] / 'y' scale - let [ xFacet, yFacet ] = s.facets; - let xScaleKey = xFacet.scale; - let yScaleKey = yFacet.scale; - let [ xData, yData ] = data[i]; - - accScale(wipScales[xScaleKey], pendScales[xScaleKey], xFacet, xData, xFacet.sorted); - accScale(wipScales[yScaleKey], pendScales[yScaleKey], yFacet, yData, yFacet.sorted); - - // temp - s.min = yFacet.min; - s.max = yFacet.max; - } - } - } - }); - - // range independent scales - for (let k in wipScales) { - let wsc = wipScales[k]; - let psc = pendScales[k]; - - if (wsc.from == null && (psc == null || psc.min == null)) { - let minMax = wsc.range( - self, - wsc.min == inf ? null : wsc.min, - wsc.max == -inf ? null : wsc.max, - k - ); - wsc.min = minMax[0]; - wsc.max = minMax[1]; - } - } - } - - // range dependent scales - for (let k in wipScales) { - let wsc = wipScales[k]; - - if (wsc.from != null) { - let base = wipScales[wsc.from]; - - if (base.min == null) - wsc.min = wsc.max = null; - else { - let minMax = wsc.range(self, base.min, base.max, k); - wsc.min = minMax[0]; - wsc.max = minMax[1]; - } - } - } - - let changed = {}; - let anyChanged = false; - - for (let k in wipScales) { - let wsc = wipScales[k]; - let sc = scales[k]; - - if (sc.min != wsc.min || sc.max != wsc.max) { - sc.min = wsc.min; - sc.max = wsc.max; - - let distr = sc.distr; - - sc._min = distr == 3 ? log10(sc.min) : distr == 4 ? asinh(sc.min, sc.asinh) : sc.min; - sc._max = distr == 3 ? log10(sc.max) : distr == 4 ? asinh(sc.max, sc.asinh) : sc.max; - - changed[k] = anyChanged = true; - } - } - - if (anyChanged) { - // invalidate paths of all series on changed scales - series.forEach((s, i) => { - if (mode == 2) { - if (i > 0 && changed.y) - s._paths = null; - } - else { - if (changed[s.scale]) - s._paths = null; - } - }); - - for (let k in changed) { - shouldConvergeSize = true; - fire("setScale", k); - } - - if (cursor.show) - shouldSetCursor = shouldSetLegend = cursor.left >= 0; - } - - for (let k in pendScales) - pendScales[k] = null; - } - - // grabs the nearest indices with y data outside of x-scale limits - function getOuterIdxs(ydata) { - let _i0 = clamp(i0 - 1, 0, dataLen - 1); - let _i1 = clamp(i1 + 1, 0, dataLen - 1); - - while (ydata[_i0] == null && _i0 > 0) - _i0--; - - while (ydata[_i1] == null && _i1 < dataLen - 1) - _i1++; - - return [_i0, _i1]; - } - - function drawSeries() { - if (dataLen > 0) { - series.forEach((s, i) => { - if (i > 0 && s.show && s._paths == null) { - let _idxs = getOuterIdxs(data[i]); - s._paths = s.paths(self, i, _idxs[0], _idxs[1]); - } - }); - - series.forEach((s, i) => { - if (i > 0 && s.show) { - if (ctxAlpha != s.alpha) - ctx.globalAlpha = ctxAlpha = s.alpha; - - { - cacheStrokeFill(i, false); - s._paths && drawPath(i, false); - } - - { - cacheStrokeFill(i, true); - - let show = s.points.show(self, i, i0, i1); - let idxs = s.points.filter(self, i, show, s._paths ? s._paths.gaps : null); - - if (show || idxs) { - s.points._paths = s.points.paths(self, i, i0, i1, idxs); - drawPath(i, true); - } - } - - if (ctxAlpha != 1) - ctx.globalAlpha = ctxAlpha = 1; - - fire("drawSeries", i); - } - }); - } - } - - function cacheStrokeFill(si, _points) { - let s = _points ? series[si].points : series[si]; - - s._stroke = s.stroke(self, si); - s._fill = s.fill(self, si); - } - - function drawPath(si, _points) { - let s = _points ? series[si].points : series[si]; - - let strokeStyle = s._stroke; - let fillStyle = s._fill; - - let { stroke, fill, clip: gapsClip, flags } = s._paths; - let boundsClip = null; - let width = roundDec(s.width * pxRatio, 3); - let offset = (width % 2) / 2; - - if (_points && fillStyle == null) - fillStyle = width > 0 ? "#fff" : strokeStyle; - - let _pxAlign = s.pxAlign == 1; - - _pxAlign && ctx.translate(offset, offset); - - if (!_points) { - let lft = plotLft, - top = plotTop, - wid = plotWid, - hgt = plotHgt; - - let halfWid = width * pxRatio / 2; - - if (s.min == 0) - hgt += halfWid; - - if (s.max == 0) { - top -= halfWid; - hgt += halfWid; - } - - boundsClip = new Path2D(); - boundsClip.rect(lft, top, wid, hgt); - } - - // the points pathbuilder's gapsClip is its boundsClip, since points dont need gaps clipping, and bounds depend on point size - if (_points) - strokeFill(strokeStyle, width, s.dash, s.cap, fillStyle, stroke, fill, flags, gapsClip); - else - fillStroke(si, strokeStyle, width, s.dash, s.cap, fillStyle, stroke, fill, flags, boundsClip, gapsClip); - - _pxAlign && ctx.translate(-offset, -offset); - } - - function fillStroke(si, strokeStyle, lineWidth, lineDash, lineCap, fillStyle, strokePath, fillPath, flags, boundsClip, gapsClip) { - let didStrokeFill = false; - - // for all bands where this series is the top edge, create upwards clips using the bottom edges - // and apply clips + fill with band fill or dfltFill - bands.forEach((b, bi) => { - // isUpperEdge? - if (b.series[0] == si) { - let lowerEdge = series[b.series[1]]; - let lowerData = data[b.series[1]]; - - let bandClip = (lowerEdge._paths || EMPTY_OBJ).band; - - if (isArr(bandClip)) - bandClip = b.dir == 1 ? bandClip[0] : bandClip[1]; - - let gapsClip2; - - let _fillStyle = null; - - // hasLowerEdge? - if (lowerEdge.show && bandClip && hasData(lowerData, i0, i1)) { - _fillStyle = b.fill(self, bi) || fillStyle; - gapsClip2 = lowerEdge._paths.clip; - } - else - bandClip = null; - - strokeFill(strokeStyle, lineWidth, lineDash, lineCap, _fillStyle, strokePath, fillPath, flags, boundsClip, gapsClip, gapsClip2, bandClip); - - didStrokeFill = true; - } - }); - - if (!didStrokeFill) - strokeFill(strokeStyle, lineWidth, lineDash, lineCap, fillStyle, strokePath, fillPath, flags, boundsClip, gapsClip); - } - - const CLIP_FILL_STROKE = BAND_CLIP_FILL | BAND_CLIP_STROKE; - - function strokeFill(strokeStyle, lineWidth, lineDash, lineCap, fillStyle, strokePath, fillPath, flags, boundsClip, gapsClip, gapsClip2, bandClip) { - setCtxStyle(strokeStyle, lineWidth, lineDash, lineCap, fillStyle); - - if (boundsClip || gapsClip || bandClip) { - ctx.save(); - boundsClip && ctx.clip(boundsClip); - gapsClip && ctx.clip(gapsClip); - } - - if (bandClip) { - if ((flags & CLIP_FILL_STROKE) == CLIP_FILL_STROKE) { - ctx.clip(bandClip); - gapsClip2 && ctx.clip(gapsClip2); - doFill(fillStyle, fillPath); - doStroke(strokeStyle, strokePath, lineWidth); - } - else if (flags & BAND_CLIP_STROKE) { - doFill(fillStyle, fillPath); - ctx.clip(bandClip); - doStroke(strokeStyle, strokePath, lineWidth); - } - else if (flags & BAND_CLIP_FILL) { - ctx.save(); - ctx.clip(bandClip); - gapsClip2 && ctx.clip(gapsClip2); - doFill(fillStyle, fillPath); - ctx.restore(); - doStroke(strokeStyle, strokePath, lineWidth); - } - } - else { - doFill(fillStyle, fillPath); - doStroke(strokeStyle, strokePath, lineWidth); - } - - if (boundsClip || gapsClip || bandClip) - ctx.restore(); - } - - function doStroke(strokeStyle, strokePath, lineWidth) { - if (lineWidth > 0) { - if (strokePath instanceof Map) { - strokePath.forEach((strokePath, strokeStyle) => { - ctx.strokeStyle = ctxStroke = strokeStyle; - ctx.stroke(strokePath); - }); - } - else - strokePath != null && strokeStyle && ctx.stroke(strokePath); - } - } - - function doFill(fillStyle, fillPath) { - if (fillPath instanceof Map) { - fillPath.forEach((fillPath, fillStyle) => { - ctx.fillStyle = ctxFill = fillStyle; - ctx.fill(fillPath); - }); - } - else - fillPath != null && fillStyle && ctx.fill(fillPath); - } - - function getIncrSpace(axisIdx, min, max, fullDim) { - let axis = axes[axisIdx]; - - let incrSpace; - - if (fullDim <= 0) - incrSpace = [0, 0]; - else { - let minSpace = axis._space = axis.space(self, axisIdx, min, max, fullDim); - let incrs = axis._incrs = axis.incrs(self, axisIdx, min, max, fullDim, minSpace); - incrSpace = findIncr(min, max, incrs, fullDim, minSpace); - } - - return (axis._found = incrSpace); - } - - function drawOrthoLines(offs, filts, ori, side, pos0, len, width, stroke, dash, cap) { - let offset = (width % 2) / 2; - - pxAlign == 1 && ctx.translate(offset, offset); - - setCtxStyle(stroke, width, dash, cap, stroke); - - ctx.beginPath(); - - let x0, y0, x1, y1, pos1 = pos0 + (side == 0 || side == 3 ? -len : len); - - if (ori == 0) { - y0 = pos0; - y1 = pos1; - } - else { - x0 = pos0; - x1 = pos1; - } - - for (let i = 0; i < offs.length; i++) { - if (filts[i] != null) { - if (ori == 0) - x0 = x1 = offs[i]; - else - y0 = y1 = offs[i]; - - ctx.moveTo(x0, y0); - ctx.lineTo(x1, y1); - } - } - - ctx.stroke(); - - pxAlign == 1 && ctx.translate(-offset, -offset); - } - - function axesCalc(cycleNum) { - // log("axesCalc()", arguments); - - let converged = true; - - axes.forEach((axis, i) => { - if (!axis.show) - return; - - let scale = scales[axis.scale]; - - if (scale.min == null) { - if (axis._show) { - converged = false; - axis._show = false; - resetYSeries(false); - } - return; - } - else { - if (!axis._show) { - converged = false; - axis._show = true; - resetYSeries(false); - } - } - - let side = axis.side; - let ori = side % 2; - - let {min, max} = scale; // // should this toggle them ._show = false - - let [_incr, _space] = getIncrSpace(i, min, max, ori == 0 ? plotWidCss : plotHgtCss); - - if (_space == 0) - return; - - // if we're using index positions, force first tick to match passed index - let forceMin = scale.distr == 2; - - let _splits = axis._splits = axis.splits(self, i, min, max, _incr, _space, forceMin); - - // tick labels - // BOO this assumes a specific data/series - let splits = scale.distr == 2 ? _splits.map(i => data0[i]) : _splits; - let incr = scale.distr == 2 ? data0[_splits[1]] - data0[_splits[0]] : _incr; - - let values = axis._values = axis.values(self, axis.filter(self, splits, i, _space, incr), i, _space, incr); - - // rotating of labels only supported on bottom x axis - axis._rotate = side == 2 ? axis.rotate(self, values, i, _space) : 0; - - let oldSize = axis._size; - - axis._size = ceil(axis.size(self, values, i, cycleNum)); - - if (oldSize != null && axis._size != oldSize) // ready && ? - converged = false; - }); - - return converged; - } - - function paddingCalc(cycleNum) { - let converged = true; - - padding.forEach((p, i) => { - let _p = p(self, i, sidesWithAxes, cycleNum); - - if (_p != _padding[i]) - converged = false; - - _padding[i] = _p; - }); - - return converged; - } - - function drawAxesGrid() { - for (let i = 0; i < axes.length; i++) { - let axis = axes[i]; - - if (!axis.show || !axis._show) - continue; - - let side = axis.side; - let ori = side % 2; - - let x, y; - - let fillStyle = axis.stroke(self, i); - - let shiftDir = side == 0 || side == 3 ? -1 : 1; - - // axis label - if (axis.label) { - let shiftAmt = axis.labelGap * shiftDir; - let baseLpos = round((axis._lpos + shiftAmt) * pxRatio); - - setFontStyle(axis.labelFont[0], fillStyle, "center", side == 2 ? TOP : BOTTOM); - - ctx.save(); - - if (ori == 1) { - x = y = 0; - - ctx.translate( - baseLpos, - round(plotTop + plotHgt / 2), - ); - ctx.rotate((side == 3 ? -PI : PI) / 2); - - } - else { - x = round(plotLft + plotWid / 2); - y = baseLpos; - } - - ctx.fillText(axis.label, x, y); - - ctx.restore(); - } - - let [_incr, _space] = axis._found; - - if (_space == 0) - continue; - - let scale = scales[axis.scale]; - - let plotDim = ori == 0 ? plotWid : plotHgt; - let plotOff = ori == 0 ? plotLft : plotTop; - - let axisGap = round(axis.gap * pxRatio); - - let _splits = axis._splits; - - // tick labels - // BOO this assumes a specific data/series - let splits = scale.distr == 2 ? _splits.map(i => data0[i]) : _splits; - let incr = scale.distr == 2 ? data0[_splits[1]] - data0[_splits[0]] : _incr; - - let ticks = axis.ticks; - let border = axis.border; - let tickSize = ticks.show ? round(ticks.size * pxRatio) : 0; - - // rotating of labels only supported on bottom x axis - let angle = axis._rotate * -PI/180; - - let basePos = pxRound(axis._pos * pxRatio); - let shiftAmt = (tickSize + axisGap) * shiftDir; - let finalPos = basePos + shiftAmt; - y = ori == 0 ? finalPos : 0; - x = ori == 1 ? finalPos : 0; - - let font = axis.font[0]; - let textAlign = axis.align == 1 ? LEFT : - axis.align == 2 ? RIGHT : - angle > 0 ? LEFT : - angle < 0 ? RIGHT : - ori == 0 ? "center" : side == 3 ? RIGHT : LEFT; - let textBaseline = angle || - ori == 1 ? "middle" : side == 2 ? TOP : BOTTOM; - - setFontStyle(font, fillStyle, textAlign, textBaseline); - - let lineHeight = axis.font[1] * lineMult; - - let canOffs = _splits.map(val => pxRound(getPos(val, scale, plotDim, plotOff))); - - let _values = axis._values; - - for (let i = 0; i < _values.length; i++) { - let val = _values[i]; - - if (val != null) { - if (ori == 0) - x = canOffs[i]; - else - y = canOffs[i]; - - val = "" + val; - - let _parts = val.indexOf("\n") == -1 ? [val] : val.split(/\n/gm); - - for (let j = 0; j < _parts.length; j++) { - let text = _parts[j]; - - if (angle) { - ctx.save(); - ctx.translate(x, y + j * lineHeight); // can this be replaced with position math? - ctx.rotate(angle); // can this be done once? - ctx.fillText(text, 0, 0); - ctx.restore(); - } - else - ctx.fillText(text, x, y + j * lineHeight); - } - } - } - - // ticks - if (ticks.show) { - drawOrthoLines( - canOffs, - ticks.filter(self, splits, i, _space, incr), - ori, - side, - basePos, - tickSize, - roundDec(ticks.width * pxRatio, 3), - ticks.stroke(self, i), - ticks.dash, - ticks.cap, - ); - } - - // grid - let grid = axis.grid; - - if (grid.show) { - drawOrthoLines( - canOffs, - grid.filter(self, splits, i, _space, incr), - ori, - ori == 0 ? 2 : 1, - ori == 0 ? plotTop : plotLft, - ori == 0 ? plotHgt : plotWid, - roundDec(grid.width * pxRatio, 3), - grid.stroke(self, i), - grid.dash, - grid.cap, - ); - } - - if (border.show) { - drawOrthoLines( - [basePos], - [1], - ori == 0 ? 1 : 0, - ori == 0 ? 1 : 2, - ori == 1 ? plotTop : plotLft, - ori == 1 ? plotHgt : plotWid, - roundDec(border.width * pxRatio, 3), - border.stroke(self, i), - border.dash, - border.cap, - ); - } - } - - fire("drawAxes"); - } - - function resetYSeries(minMax) { - // log("resetYSeries()", arguments); - - series.forEach((s, i) => { - if (i > 0) { - s._paths = null; - - if (minMax) { - if (mode == 1) { - s.min = null; - s.max = null; - } - else { - s.facets.forEach(f => { - f.min = null; - f.max = null; - }); - } - } - } - }); - } - - let queuedCommit = false; - - function commit() { - if (!queuedCommit) { - microTask(_commit); - queuedCommit = true; - } - } - - function _commit() { - // log("_commit()", arguments); - - if (shouldSetScales) { - setScales(); - shouldSetScales = false; - } - - if (shouldConvergeSize) { - convergeSize(); - shouldConvergeSize = false; - } - - if (shouldSetSize) { - setStylePx(under, LEFT, plotLftCss); - setStylePx(under, TOP, plotTopCss); - setStylePx(under, WIDTH, plotWidCss); - setStylePx(under, HEIGHT, plotHgtCss); - - setStylePx(over, LEFT, plotLftCss); - setStylePx(over, TOP, plotTopCss); - setStylePx(over, WIDTH, plotWidCss); - setStylePx(over, HEIGHT, plotHgtCss); - - setStylePx(wrap, WIDTH, fullWidCss); - setStylePx(wrap, HEIGHT, fullHgtCss); - - // NOTE: mutating this during print preview in Chrome forces transparent - // canvas pixels to white, even when followed up with clearRect() below - can.width = round(fullWidCss * pxRatio); - can.height = round(fullHgtCss * pxRatio); - - axes.forEach(({ _el, _show, _size, _pos, side }) => { - if (_el != null) { - if (_show) { - let posOffset = (side === 3 || side === 0 ? _size : 0); - let isVt = side % 2 == 1; - - setStylePx(_el, isVt ? "left" : "top", _pos - posOffset); - setStylePx(_el, isVt ? "width" : "height", _size); - setStylePx(_el, isVt ? "top" : "left", isVt ? plotTopCss : plotLftCss); - setStylePx(_el, isVt ? "height" : "width", isVt ? plotHgtCss : plotWidCss); - - remClass(_el, OFF); - } - else - addClass(_el, OFF); - } - }); - - // invalidate ctx style cache - ctxStroke = ctxFill = ctxWidth = ctxJoin = ctxCap = ctxFont = ctxAlign = ctxBaseline = ctxDash = null; - ctxAlpha = 1; - - syncRect(true); - - fire("setSize"); - - shouldSetSize = false; - } - - if (fullWidCss > 0 && fullHgtCss > 0) { - ctx.clearRect(0, 0, can.width, can.height); - fire("drawClear"); - drawOrder.forEach(fn => fn()); - fire("draw"); - } - - // if (shouldSetSelect) { - // TODO: update .u-select metrics (if visible) - // setStylePx(selectDiv, TOP, select.top = 0); - // setStylePx(selectDiv, LEFT, select.left = 0); - // setStylePx(selectDiv, WIDTH, select.width = 0); - // setStylePx(selectDiv, HEIGHT, select.height = 0); - // shouldSetSelect = false; - // } - - if (cursor.show && shouldSetCursor) { - updateCursor(null, true, false); - shouldSetCursor = false; - } - - // if (FEAT_LEGEND && legend.show && legend.live && shouldSetLegend) {} - - if (!ready) { - ready = true; - self.status = 1; - - fire("ready"); - } - - viaAutoScaleX = false; - - queuedCommit = false; - } - - self.redraw = (rebuildPaths, recalcAxes) => { - shouldConvergeSize = recalcAxes || false; - - if (rebuildPaths !== false) - _setScale(xScaleKey, scaleX.min, scaleX.max); - else - commit(); - }; - - // redraw() => setScale('x', scales.x.min, scales.x.max); - - // explicit, never re-ranged (is this actually true? for x and y) - function setScale(key, opts) { - let sc = scales[key]; - - if (sc.from == null) { - if (dataLen == 0) { - let minMax = sc.range(self, opts.min, opts.max, key); - opts.min = minMax[0]; - opts.max = minMax[1]; - } - - if (opts.min > opts.max) { - let _min = opts.min; - opts.min = opts.max; - opts.max = _min; - } - - if (dataLen > 1 && opts.min != null && opts.max != null && opts.max - opts.min < 1e-16) - return; - - if (key == xScaleKey) { - if (sc.distr == 2 && dataLen > 0) { - opts.min = closestIdx(opts.min, data[0]); - opts.max = closestIdx(opts.max, data[0]); - - if (opts.min == opts.max) - opts.max++; - } - } - - // log("setScale()", arguments); - - pendScales[key] = opts; - - shouldSetScales = true; - commit(); - } - } - - self.setScale = setScale; - -// INTERACTION - - let xCursor; - let yCursor; - let vCursor; - let hCursor; - - // starting position before cursor.move - let rawMouseLeft0; - let rawMouseTop0; - - // starting position - let mouseLeft0; - let mouseTop0; - - // current position before cursor.move - let rawMouseLeft1; - let rawMouseTop1; - - // current position - let mouseLeft1; - let mouseTop1; - - let dragging = false; - - const drag = cursor.drag; - - let dragX = drag.x; - let dragY = drag.y; - - if (cursor.show) { - if (cursor.x) - xCursor = placeDiv(CURSOR_X, over); - if (cursor.y) - yCursor = placeDiv(CURSOR_Y, over); - - if (scaleX.ori == 0) { - vCursor = xCursor; - hCursor = yCursor; - } - else { - vCursor = yCursor; - hCursor = xCursor; - } - - mouseLeft1 = cursor.left; - mouseTop1 = cursor.top; - } - - const select = self.select = assign({ - show: true, - over: true, - left: 0, - width: 0, - top: 0, - height: 0, - }, opts.select); - - const selectDiv = select.show ? placeDiv(SELECT, select.over ? over : under) : null; - - function setSelect(opts, _fire) { - if (select.show) { - for (let prop in opts) - setStylePx(selectDiv, prop, select[prop] = opts[prop]); - - _fire !== false && fire("setSelect"); - } - } - - self.setSelect = setSelect; - - function toggleDOM(i, onOff) { - let s = series[i]; - let label = showLegend ? legendRows[i] : null; - - if (s.show) - label && remClass(label, OFF); - else { - label && addClass(label, OFF); - cursorPts.length > 1 && elTrans(cursorPts[i], -10, -10, plotWidCss, plotHgtCss); - } - } - - function _setScale(key, min, max) { - setScale(key, {min, max}); - } - - function setSeries(i, opts, _fire, _pub) { - // log("setSeries()", arguments); - - if (opts.focus != null) - setFocus(i); - - if (opts.show != null) { - series.forEach((s, si) => { - if (si > 0 && (i == si || i == null)) { - s.show = opts.show; - toggleDOM(si, opts.show); - - _setScale(mode == 2 ? s.facets[1].scale : s.scale, null, null); - commit(); - } - }); - } - - _fire !== false && fire("setSeries", i, opts); - - _pub && pubSync("setSeries", self, i, opts); - } - - self.setSeries = setSeries; - - function setBand(bi, opts) { - assign(bands[bi], opts); - } - - function addBand(opts, bi) { - opts.fill = fnOrSelf(opts.fill || null); - opts.dir = ifNull(opts.dir, -1); - bi = bi == null ? bands.length : bi; - bands.splice(bi, 0, opts); - } - - function delBand(bi) { - if (bi == null) - bands.length = 0; - else - bands.splice(bi, 1); - } - - self.addBand = addBand; - self.setBand = setBand; - self.delBand = delBand; - - function setAlpha(i, value) { - series[i].alpha = value; - - if (cursor.show && cursorPts[i]) - cursorPts[i].style.opacity = value; - - if (showLegend && legendRows[i]) - legendRows[i].style.opacity = value; - } - - // y-distance - let closestDist; - let closestSeries; - let focusedSeries; - const FOCUS_TRUE = {focus: true}; - - function setFocus(i) { - if (i != focusedSeries) { - // log("setFocus()", arguments); - - let allFocused = i == null; - - let _setAlpha = focus.alpha != 1; - - series.forEach((s, i2) => { - let isFocused = allFocused || i2 == 0 || i2 == i; - s._focus = allFocused ? null : isFocused; - _setAlpha && setAlpha(i2, isFocused ? 1 : focus.alpha); - }); - - focusedSeries = i; - _setAlpha && commit(); - } - } - - if (showLegend && cursorFocus) { - on(mouseleave, legendEl, e => { - if (cursor._lock) - return; - - if (focusedSeries != null) - setSeries(null, FOCUS_TRUE, true, syncOpts.setSeries); - }); - } - - function posToVal(pos, scale, can) { - let sc = scales[scale]; - - if (can) - pos = pos / pxRatio - (sc.ori == 1 ? plotTopCss : plotLftCss); - - let dim = plotWidCss; - - if (sc.ori == 1) { - dim = plotHgtCss; - pos = dim - pos; - } - - if (sc.dir == -1) - pos = dim - pos; - - let _min = sc._min, - _max = sc._max, - pct = pos / dim; - - let sv = _min + (_max - _min) * pct; - - let distr = sc.distr; - - return ( - distr == 3 ? pow(10, sv) : - distr == 4 ? sinh(sv, sc.asinh) : - sv - ); - } - - function closestIdxFromXpos(pos, can) { - let v = posToVal(pos, xScaleKey, can); - return closestIdx(v, data[0], i0, i1); - } - - self.valToIdx = val => closestIdx(val, data[0]); - self.posToIdx = closestIdxFromXpos; - self.posToVal = posToVal; - self.valToPos = (val, scale, can) => ( - scales[scale].ori == 0 ? - getHPos(val, scales[scale], - can ? plotWid : plotWidCss, - can ? plotLft : 0, - ) : - getVPos(val, scales[scale], - can ? plotHgt : plotHgtCss, - can ? plotTop : 0, - ) - ); - - // defers calling expensive functions - function batch(fn) { - fn(self); - commit(); - } - - self.batch = batch; - - (self.setCursor = (opts, _fire, _pub) => { - mouseLeft1 = opts.left; - mouseTop1 = opts.top; - // assign(cursor, opts); - updateCursor(null, _fire, _pub); - }); - - function setSelH(off, dim) { - setStylePx(selectDiv, LEFT, select.left = off); - setStylePx(selectDiv, WIDTH, select.width = dim); - } - - function setSelV(off, dim) { - setStylePx(selectDiv, TOP, select.top = off); - setStylePx(selectDiv, HEIGHT, select.height = dim); - } - - let setSelX = scaleX.ori == 0 ? setSelH : setSelV; - let setSelY = scaleX.ori == 1 ? setSelH : setSelV; - - function syncLegend() { - if (showLegend && legend.live) { - for (let i = mode == 2 ? 1 : 0; i < series.length; i++) { - if (i == 0 && multiValLegend) - continue; - - let vals = legend.values[i]; - - let j = 0; - - for (let k in vals) - legendCells[i][j++].firstChild.nodeValue = vals[k]; - } - } - } - - function setLegend(opts, _fire) { - if (opts != null) { - let idx = opts.idx; - - legend.idx = idx; - series.forEach((s, sidx) => { - (sidx > 0 || !multiValLegend) && setLegendValues(sidx, idx); - }); - } - - if (showLegend && legend.live) - syncLegend(); - - shouldSetLegend = false; - - _fire !== false && fire("setLegend"); - } - - self.setLegend = setLegend; - - function setLegendValues(sidx, idx) { - let val; - - if (idx == null) - val = NULL_LEGEND_VALUES; - else { - let s = series[sidx]; - let src = sidx == 0 && xScaleDistr == 2 ? data0 : data[sidx]; - val = multiValLegend ? s.values(self, sidx, idx) : {_: s.value(self, src[idx], sidx, idx)}; - } - - legend.values[sidx] = val; - } - - function updateCursor(src, _fire, _pub) { - // ts == null && log("updateCursor()", arguments); - - rawMouseLeft1 = mouseLeft1; - rawMouseTop1 = mouseTop1; - - [mouseLeft1, mouseTop1] = cursor.move(self, mouseLeft1, mouseTop1); - - if (cursor.show) { - vCursor && elTrans(vCursor, round(mouseLeft1), 0, plotWidCss, plotHgtCss); - hCursor && elTrans(hCursor, 0, round(mouseTop1), plotWidCss, plotHgtCss); - } - - let idx; - - // when zooming to an x scale range between datapoints the binary search - // for nearest min/max indices results in this condition. cheap hack :D - let noDataInRange = i0 > i1; // works for mode 1 only - - closestDist = inf; - - // TODO: extract - let xDim = scaleX.ori == 0 ? plotWidCss : plotHgtCss; - let yDim = scaleX.ori == 1 ? plotWidCss : plotHgtCss; - - // if cursor hidden, hide points & clear legend vals - if (mouseLeft1 < 0 || dataLen == 0 || noDataInRange) { - idx = null; - - for (let i = 0; i < series.length; i++) { - if (i > 0) { - cursorPts.length > 1 && elTrans(cursorPts[i], -10, -10, plotWidCss, plotHgtCss); - } - } - - if (cursorFocus) - setSeries(null, FOCUS_TRUE, true, src == null && syncOpts.setSeries); - - if (legend.live) { - activeIdxs.fill(null); - shouldSetLegend = true; - - for (let i = 0; i < series.length; i++) - legend.values[i] = NULL_LEGEND_VALUES; - } - } - else { - // let pctY = 1 - (y / rect.height); - - let mouseXPos, valAtPosX, xPos; - - if (mode == 1) { - mouseXPos = scaleX.ori == 0 ? mouseLeft1 : mouseTop1; - valAtPosX = posToVal(mouseXPos, xScaleKey); - idx = closestIdx(valAtPosX, data[0], i0, i1); - xPos = incrRoundUp(valToPosX(data[0][idx], scaleX, xDim, 0), 0.5); - } - - for (let i = mode == 2 ? 1 : 0; i < series.length; i++) { - let s = series[i]; - - let idx1 = activeIdxs[i]; - let yVal1 = mode == 1 ? data[i][idx1] : data[i][1][idx1]; - - let idx2 = cursor.dataIdx(self, i, idx, valAtPosX); - let yVal2 = mode == 1 ? data[i][idx2] : data[i][1][idx2]; - - shouldSetLegend = shouldSetLegend || yVal2 != yVal1 || idx2 != idx1; - - activeIdxs[i] = idx2; - - let xPos2 = idx2 == idx ? xPos : incrRoundUp(valToPosX(mode == 1 ? data[0][idx2] : data[i][0][idx2], scaleX, xDim, 0), 0.5); - - if (i > 0 && s.show) { - let yPos = yVal2 == null ? -10 : incrRoundUp(valToPosY(yVal2, mode == 1 ? scales[s.scale] : scales[s.facets[1].scale], yDim, 0), 0.5); - - if (yPos > 0 && mode == 1) { - let dist = abs(yPos - mouseTop1); - - if (dist <= closestDist) { - closestDist = dist; - closestSeries = i; - } - } - - let hPos, vPos; - - if (scaleX.ori == 0) { - hPos = xPos2; - vPos = yPos; - } - else { - hPos = yPos; - vPos = xPos2; - } - - if (shouldSetLegend && cursorPts.length > 1) { - elColor(cursorPts[i], cursor.points.fill(self, i), cursor.points.stroke(self, i)); - - let ptWid, ptHgt, ptLft, ptTop, - centered = true, - getBBox = cursor.points.bbox; - - if (getBBox != null) { - centered = false; - - let bbox = getBBox(self, i); - - ptLft = bbox.left; - ptTop = bbox.top; - ptWid = bbox.width; - ptHgt = bbox.height; - } - else { - ptLft = hPos; - ptTop = vPos; - ptWid = ptHgt = cursor.points.size(self, i); - } - - elSize(cursorPts[i], ptWid, ptHgt, centered); - elTrans(cursorPts[i], ptLft, ptTop, plotWidCss, plotHgtCss); - } - } - - if (legend.live) { - if (!shouldSetLegend || i == 0 && multiValLegend) - continue; - - setLegendValues(i, idx2); - } - } - } - - cursor.idx = idx; - cursor.left = mouseLeft1; - cursor.top = mouseTop1; - - if (shouldSetLegend) { - legend.idx = idx; - setLegend(); - } - - // nit: cursor.drag.setSelect is assumed always true - if (select.show && dragging) { - if (src != null) { - let [xKey, yKey] = syncOpts.scales; - let [matchXKeys, matchYKeys] = syncOpts.match; - let [xKeySrc, yKeySrc] = src.cursor.sync.scales; - - // match the dragX/dragY implicitness/explicitness of src - let sdrag = src.cursor.drag; - dragX = sdrag._x; - dragY = sdrag._y; - - if (dragX || dragY) { - let { left, top, width, height } = src.select; - - let sori = src.scales[xKey].ori; - let sPosToVal = src.posToVal; - - let sOff, sDim, sc, a, b; - - let matchingX = xKey != null && matchXKeys(xKey, xKeySrc); - let matchingY = yKey != null && matchYKeys(yKey, yKeySrc); - - if (matchingX && dragX) { - if (sori == 0) { - sOff = left; - sDim = width; - } - else { - sOff = top; - sDim = height; - } - - sc = scales[xKey]; - - a = valToPosX(sPosToVal(sOff, xKeySrc), sc, xDim, 0); - b = valToPosX(sPosToVal(sOff + sDim, xKeySrc), sc, xDim, 0); - - setSelX(min(a,b), abs(b-a)); - } - else - setSelX(0, xDim); - - if (matchingY && dragY) { - if (sori == 1) { - sOff = left; - sDim = width; - } - else { - sOff = top; - sDim = height; - } - - sc = scales[yKey]; - - a = valToPosY(sPosToVal(sOff, yKeySrc), sc, yDim, 0); - b = valToPosY(sPosToVal(sOff + sDim, yKeySrc), sc, yDim, 0); - - setSelY(min(a,b), abs(b-a)); - } - else - setSelY(0, yDim); - } - else - hideSelect(); - } - else { - let rawDX = abs(rawMouseLeft1 - rawMouseLeft0); - let rawDY = abs(rawMouseTop1 - rawMouseTop0); - - if (scaleX.ori == 1) { - let _rawDX = rawDX; - rawDX = rawDY; - rawDY = _rawDX; - } - - dragX = drag.x && rawDX >= drag.dist; - dragY = drag.y && rawDY >= drag.dist; - - let uni = drag.uni; - - if (uni != null) { - // only calc drag status if they pass the dist thresh - if (dragX && dragY) { - dragX = rawDX >= uni; - dragY = rawDY >= uni; - - // force unidirectionality when both are under uni limit - if (!dragX && !dragY) { - if (rawDY > rawDX) - dragY = true; - else - dragX = true; - } - } - } - else if (drag.x && drag.y && (dragX || dragY)) - // if omni with no uni then both dragX / dragY should be true if either is true - dragX = dragY = true; - - let p0, p1; - - if (dragX) { - if (scaleX.ori == 0) { - p0 = mouseLeft0; - p1 = mouseLeft1; - } - else { - p0 = mouseTop0; - p1 = mouseTop1; - } - - setSelX(min(p0, p1), abs(p1 - p0)); - - if (!dragY) - setSelY(0, yDim); - } - - if (dragY) { - if (scaleX.ori == 1) { - p0 = mouseLeft0; - p1 = mouseLeft1; - } - else { - p0 = mouseTop0; - p1 = mouseTop1; - } - - setSelY(min(p0, p1), abs(p1 - p0)); - - if (!dragX) - setSelX(0, xDim); - } - - // the drag didn't pass the dist requirement - if (!dragX && !dragY) { - setSelX(0, 0); - setSelY(0, 0); - } - } - } - - drag._x = dragX; - drag._y = dragY; - - if (src == null) { - if (_pub) { - if (syncKey != null) { - let [xSyncKey, ySyncKey] = syncOpts.scales; - - syncOpts.values[0] = xSyncKey != null ? posToVal(scaleX.ori == 0 ? mouseLeft1 : mouseTop1, xSyncKey) : null; - syncOpts.values[1] = ySyncKey != null ? posToVal(scaleX.ori == 1 ? mouseLeft1 : mouseTop1, ySyncKey) : null; - } - - pubSync(mousemove, self, mouseLeft1, mouseTop1, plotWidCss, plotHgtCss, idx); - } - - if (cursorFocus) { - let shouldPub = _pub && syncOpts.setSeries; - let p = focus.prox; - - if (focusedSeries == null) { - if (closestDist <= p) - setSeries(closestSeries, FOCUS_TRUE, true, shouldPub); - } - else { - if (closestDist > p) - setSeries(null, FOCUS_TRUE, true, shouldPub); - else if (closestSeries != focusedSeries) - setSeries(closestSeries, FOCUS_TRUE, true, shouldPub); - } - } - } - - ready && _fire !== false && fire("setCursor"); - } - - let rect = null; - - function syncRect(defer) { - if (defer === true) - rect = null; - else { - rect = over.getBoundingClientRect(); - fire("syncRect", rect); - } - } - - function mouseMove(e, src, _l, _t, _w, _h, _i) { - if (cursor._lock) - return; - - cacheMouse(e, src, _l, _t, _w, _h, _i, false, e != null); - - if (e != null) - updateCursor(null, true, true); - else - updateCursor(src, true, false); - } - - function cacheMouse(e, src, _l, _t, _w, _h, _i, initial, snap) { - if (rect == null) - syncRect(false); - - if (e != null) { - _l = e.clientX - rect.left; - _t = e.clientY - rect.top; - } - else { - if (_l < 0 || _t < 0) { - mouseLeft1 = -10; - mouseTop1 = -10; - return; - } - - let [xKey, yKey] = syncOpts.scales; - - let syncOptsSrc = src.cursor.sync; - let [xValSrc, yValSrc] = syncOptsSrc.values; - let [xKeySrc, yKeySrc] = syncOptsSrc.scales; - let [matchXKeys, matchYKeys] = syncOpts.match; - - let rotSrc = src.axes[0].side % 2 == 1; - - let xDim = scaleX.ori == 0 ? plotWidCss : plotHgtCss, - yDim = scaleX.ori == 1 ? plotWidCss : plotHgtCss, - _xDim = rotSrc ? _h : _w, - _yDim = rotSrc ? _w : _h, - _xPos = rotSrc ? _t : _l, - _yPos = rotSrc ? _l : _t; - - if (xKeySrc != null) - _l = matchXKeys(xKey, xKeySrc) ? getPos(xValSrc, scales[xKey], xDim, 0) : -10; - else - _l = xDim * (_xPos/_xDim); - - if (yKeySrc != null) - _t = matchYKeys(yKey, yKeySrc) ? getPos(yValSrc, scales[yKey], yDim, 0) : -10; - else - _t = yDim * (_yPos/_yDim); - - if (scaleX.ori == 1) { - let __l = _l; - _l = _t; - _t = __l; - } - } - - if (snap) { - if (_l <= 1 || _l >= plotWidCss - 1) - _l = incrRound(_l, plotWidCss); - - if (_t <= 1 || _t >= plotHgtCss - 1) - _t = incrRound(_t, plotHgtCss); - } - - if (initial) { - rawMouseLeft0 = _l; - rawMouseTop0 = _t; - - [mouseLeft0, mouseTop0] = cursor.move(self, _l, _t); - } - else { - mouseLeft1 = _l; - mouseTop1 = _t; - } - } - - const _hideProps = { - width: 0, - height: 0, - }; - - function hideSelect() { - setSelect(_hideProps, false); - } - - function mouseDown(e, src, _l, _t, _w, _h, _i) { - dragging = true; - dragX = dragY = drag._x = drag._y = false; - - cacheMouse(e, src, _l, _t, _w, _h, _i, true, false); - - if (e != null) { - onMouse(mouseup, doc, mouseUp); - pubSync(mousedown, self, mouseLeft0, mouseTop0, plotWidCss, plotHgtCss, null); - } - } - - function mouseUp(e, src, _l, _t, _w, _h, _i) { - dragging = drag._x = drag._y = false; - - cacheMouse(e, src, _l, _t, _w, _h, _i, false, true); - - let { left, top, width, height } = select; - - let hasSelect = width > 0 || height > 0; - - hasSelect && setSelect(select); - - if (drag.setScale && hasSelect) { - // if (syncKey != null) { - // dragX = drag.x; - // dragY = drag.y; - // } - - let xOff = left, - xDim = width, - yOff = top, - yDim = height; - - if (scaleX.ori == 1) { - xOff = top, - xDim = height, - yOff = left, - yDim = width; - } - - if (dragX) { - _setScale(xScaleKey, - posToVal(xOff, xScaleKey), - posToVal(xOff + xDim, xScaleKey) - ); - } - - if (dragY) { - for (let k in scales) { - let sc = scales[k]; - - if (k != xScaleKey && sc.from == null && sc.min != inf) { - _setScale(k, - posToVal(yOff + yDim, k), - posToVal(yOff, k) - ); - } - } - } - - hideSelect(); - } - else if (cursor.lock) { - cursor._lock = !cursor._lock; - - if (!cursor._lock) - updateCursor(null, true, false); - } - - if (e != null) { - offMouse(mouseup, doc); - pubSync(mouseup, self, mouseLeft1, mouseTop1, plotWidCss, plotHgtCss, null); - } - } - - function mouseLeave(e, src, _l, _t, _w, _h, _i) { - if (!cursor._lock) { - let _dragging = dragging; - - if (dragging) { - // handle case when mousemove aren't fired all the way to edges by browser - let snapH = true; - let snapV = true; - let snapProx = 10; - - let dragH, dragV; - - if (scaleX.ori == 0) { - dragH = dragX; - dragV = dragY; - } - else { - dragH = dragY; - dragV = dragX; - } - - if (dragH && dragV) { - // maybe omni corner snap - snapH = mouseLeft1 <= snapProx || mouseLeft1 >= plotWidCss - snapProx; - snapV = mouseTop1 <= snapProx || mouseTop1 >= plotHgtCss - snapProx; - } - - if (dragH && snapH) - mouseLeft1 = mouseLeft1 < mouseLeft0 ? 0 : plotWidCss; - - if (dragV && snapV) - mouseTop1 = mouseTop1 < mouseTop0 ? 0 : plotHgtCss; - - updateCursor(null, true, true); - - dragging = false; - } - - mouseLeft1 = -10; - mouseTop1 = -10; - - // passing a non-null timestamp to force sync/mousemove event - updateCursor(null, true, true); - - if (_dragging) - dragging = _dragging; - } - } - - function dblClick(e, src, _l, _t, _w, _h, _i) { - autoScaleX(); - - hideSelect(); - - if (e != null) - pubSync(dblclick, self, mouseLeft1, mouseTop1, plotWidCss, plotHgtCss, null); - } - - function syncPxRatio() { - axes.forEach(syncFontSize); - _setSize(self.width, self.height, true); - } - - on(dppxchange, win, syncPxRatio); - - // internal pub/sub - const events = {}; - - events.mousedown = mouseDown; - events.mousemove = mouseMove; - events.mouseup = mouseUp; - events.dblclick = dblClick; - events["setSeries"] = (e, src, idx, opts) => { - setSeries(idx, opts, true, false); - }; - - if (cursor.show) { - onMouse(mousedown, over, mouseDown); - onMouse(mousemove, over, mouseMove); - onMouse(mouseenter, over, syncRect); - onMouse(mouseleave, over, mouseLeave); - - onMouse(dblclick, over, dblClick); - - cursorPlots.add(self); - - self.syncRect = syncRect; - } - - // external on/off - const hooks = self.hooks = opts.hooks || {}; - - function fire(evName, a1, a2) { - if (evName in hooks) { - hooks[evName].forEach(fn => { - fn.call(null, self, a1, a2); - }); - } - } - - (opts.plugins || []).forEach(p => { - for (let evName in p.hooks) - hooks[evName] = (hooks[evName] || []).concat(p.hooks[evName]); - }); - - const syncOpts = assign({ - key: null, - setSeries: false, - filters: { - pub: retTrue, - sub: retTrue, - }, - scales: [xScaleKey, series[1] ? series[1].scale : null], - match: [retEq, retEq], - values: [null, null], - }, cursor.sync); - - (cursor.sync = syncOpts); - - const syncKey = syncOpts.key; - - const sync = _sync(syncKey); - - function pubSync(type, src, x, y, w, h, i) { - if (syncOpts.filters.pub(type, src, x, y, w, h, i)) - sync.pub(type, src, x, y, w, h, i); - } - - sync.sub(self); - - function pub(type, src, x, y, w, h, i) { - if (syncOpts.filters.sub(type, src, x, y, w, h, i)) - events[type](null, src, x, y, w, h, i); - } - - (self.pub = pub); - - function destroy() { - sync.unsub(self); - cursorPlots.delete(self); - mouseListeners.clear(); - off(dppxchange, win, syncPxRatio); - root.remove(); - fire("destroy"); - } - - self.destroy = destroy; - - function _init() { - fire("init", opts, data); - - setData(data || opts.data, false); - - if (pendScales[xScaleKey]) - setScale(xScaleKey, pendScales[xScaleKey]); - else - autoScaleX(); - - _setSize(opts.width, opts.height); - - updateCursor(null, true, false); - - setSelect(select, false); - } - - series.forEach(initSeries); - - axes.forEach(initAxis); - - if (then) { - if (then instanceof HTMLElement) { - then.appendChild(root); - _init(); - } - else - then(self, _init); - } - else - _init(); - - return self; -} - -uPlot.assign = assign; -uPlot.fmtNum = fmtNum; -uPlot.rangeNum = rangeNum; -uPlot.rangeLog = rangeLog; -uPlot.rangeAsinh = rangeAsinh; -uPlot.orient = orient; -uPlot.pxRatio = pxRatio; - -{ - uPlot.join = join; -} - -{ - uPlot.fmtDate = fmtDate; - uPlot.tzDate = tzDate; -} - -{ - uPlot.sync = _sync; -} - -{ - uPlot.addGap = addGap; - uPlot.clipGaps = clipGaps; - - let paths = uPlot.paths = { - points, - }; - - (paths.linear = linear); - (paths.stepped = stepped); - (paths.bars = bars); - (paths.spline = monotoneCubic); -} - - - - -/***/ }), - -/***/ "./srcjs/modules/stack.js": -/*!********************************!*\ - !*** ./srcjs/modules/stack.js ***! - \********************************/ -/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { - -__webpack_require__.r(__webpack_exports__); -/* harmony export */ __webpack_require__.d(__webpack_exports__, { -/* harmony export */ "stackedChart": () => (/* binding */ stackedChart) -/* harmony export */ }); -function stack(data, omit) { - let data2 = []; - let bands = []; - let d0Len = data[0].length; - let accum = Array(d0Len); - - for (let i = 0; i < d0Len; i++) accum[i] = 0; - - for (let i = 1; i < data.length; i++) - data2.push(omit(i) ? data[i] : data[i].map((v, i) => (accum[i] += +v))); - - for (let i = 1; i < data.length; i++) - !omit(i) && - bands.push({ - series: [data.findIndex((s, j) => j > i && !omit(j)), i] - }); - - bands = bands.filter(b => b.series[1] > -1); - - return { - data: [data[0]].concat(data2), - bands - }; -} - -function getOpts(title, series) { - return { - scales: { - x: { - time: true - } - }, - series - }; -} - -function getStackedOpts(title, series, data, interp) { - let opts = getOpts(title, series); - - let interped = interp ? interp(data) : data; - - let stacked = stack(interped, i => false); - opts.bands = stacked.bands; - - opts.cursor = opts.cursor || {}; - opts.cursor.dataIdx = (u, seriesIdx, closestIdx, xValue) => { - return data[seriesIdx][closestIdx] === null ? null : closestIdx; - }; - - opts.series.forEach(s => { - s.value = (u, v, si, i) => data[si][i]; - - s.points = s.points || {}; - - // scan raw unstacked data to return only real points - s.points.filter = (u, seriesIdx, show, gaps) => { - if (show) { - let pts = []; - data[seriesIdx].forEach((v, i) => { - v !== null && pts.push(i); - }); - return pts; - } - }; - }); - - // force 0 to be the sum minimum this instead of the bottom series - opts.scales.y = { - range: (u, min, max) => { - let minMax = uPlot.rangeNum(min, max, 0.1, true); - return [minMax[0], minMax[1]]; - } - }; - - // restack on toggle - opts.hooks = { - setSeries: [ - (u, i) => { - let stacked = stack(data, i => !u.series[i].show); - u.delBand(null); - stacked.bands.forEach(b => u.addBand(b)); - u.setData(stacked.data); - } - ] - }; - - return { opts, data: stacked.data }; -} - -function stack2(series) { - // for uplot data - let data = Array(series.length); - let bands = []; - - let dataLen = series[0].values.length; - - let zeroArr = Array(dataLen).fill(0); - - let stackGroups = new Map(); - let seriesStackKeys = Array(series.length); - - series.forEach((s, si) => { - let vals = s.values.slice(); - - // apply negY - if (s.negY) { - for (let i = 0; i < vals.length; i++) { - if (vals[i] !== null) vals[i] *= -1; - } - } - - if (s.stacking.mode != "none") { - let hasPos = vals.some(v => v > 0); - // derive stacking key - let stackKey = (seriesStackKeys[si] = - s.stacking.mode + s.scaleKey + s.stacking.group + (hasPos ? "+" : "-")); - let group = stackGroups.get(stackKey); - - // initialize stacking group - if (group === null) { - group = { - series: [], - acc: zeroArr.slice(), - dir: hasPos ? -1 : 1 - }; - stackGroups.set(stackKey, group); - } - - // push for bands gen - group.series.unshift(si); - - let stacked = (data[si] = Array(dataLen)); - let { acc } = group; - - for (let i = 0; i < dataLen; i++) { - let v = vals[i]; - - if (v !== null) stacked[i] = acc[i] += v; - else stacked[i] = v; // we may want to coerce to 0 here - } - } else data[si] = vals; - }); - - // re-compute by percent - series.forEach((s, si) => { - if (s.stacking.mode == "percent") { - let group = stackGroups.get(seriesStackKeys[si]); - let { acc } = group; - - // re-negatify percent - let sign = group.dir * -1; - - let stacked = data[si]; - - for (let i = 0; i < dataLen; i++) { - let v = stacked[i]; - - if (v !== null) stacked[i] = sign * (v / acc[i]); - } - } - }); - - // generate bands between adjacent group series - stackGroups.forEach(group => { - let { series, dir } = group; - let lastIdx = series.length - 1; - - series.forEach((si, i) => { - if (i != lastIdx) { - let nextIdx = series[i + 1]; - bands.push({ - // since we're not passing x series[0] for stacking, real idxs are actually +1 - series: [si + 1, nextIdx + 1], - dir - }); - } - }); - }); - - return { - data, - bands - }; -} - -function stackedChart(title, series, _data, interp, width, height, hooks, el) { - let { opts, data } = getStackedOpts(title, series, _data, interp); - opts.title = title; - opts.width = width; - opts.height = height; - opts.hooks = Object.assign(opts.hooks, hooks); - console.log(opts); - return new uPlot(opts, data, el); -} - - - -/***/ }), - -/***/ "./srcjs/modules/utils.js": -/*!********************************!*\ - !*** ./srcjs/modules/utils.js ***! - \********************************/ -/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { - -__webpack_require__.r(__webpack_exports__); -/* harmony export */ __webpack_require__.d(__webpack_exports__, { -/* harmony export */ "getWidget": () => (/* binding */ getWidget) -/* harmony export */ }); -// From Friss tuto (https://github.com/FrissAnalytics/shinyJsTutorials/blob/master/tutorials/tutorial_03.Rmd) -function getWidget(id) { - var htmlWidgetsObj = HTMLWidgets.find("#" + id); - var widgetObj; - if (typeof htmlWidgetsObj !== "undefined") { - widgetObj = htmlWidgetsObj.getWidget(); - } - return widgetObj; -} - - -/***/ }), - -/***/ "widgets": -/*!******************************!*\ - !*** external "HTMLWidgets" ***! - \******************************/ -/***/ ((module) => { - -module.exports = window["HTMLWidgets"]; - -/***/ }) - -/******/ }); -/************************************************************************/ -/******/ // The module cache -/******/ var __webpack_module_cache__ = {}; -/******/ -/******/ // The require function -/******/ function __webpack_require__(moduleId) { -/******/ // Check if module is in cache -/******/ var cachedModule = __webpack_module_cache__[moduleId]; -/******/ if (cachedModule !== undefined) { -/******/ return cachedModule.exports; -/******/ } -/******/ // Create a new module (and put it into the cache) -/******/ var module = __webpack_module_cache__[moduleId] = { -/******/ id: moduleId, -/******/ // no module.loaded needed -/******/ exports: {} -/******/ }; -/******/ -/******/ // Execute the module function -/******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__); -/******/ -/******/ // Return the exports of the module -/******/ return module.exports; -/******/ } -/******/ -/************************************************************************/ -/******/ /* webpack/runtime/compat get default export */ -/******/ (() => { -/******/ // getDefaultExport function for compatibility with non-harmony modules -/******/ __webpack_require__.n = (module) => { -/******/ var getter = module && module.__esModule ? -/******/ () => (module['default']) : -/******/ () => (module); -/******/ __webpack_require__.d(getter, { a: getter }); -/******/ return getter; -/******/ }; -/******/ })(); -/******/ -/******/ /* webpack/runtime/define property getters */ -/******/ (() => { -/******/ // define getter functions for harmony exports -/******/ __webpack_require__.d = (exports, definition) => { -/******/ for(var key in definition) { -/******/ if(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) { -/******/ Object.defineProperty(exports, key, { enumerable: true, get: definition[key] }); -/******/ } -/******/ } -/******/ }; -/******/ })(); -/******/ -/******/ /* webpack/runtime/hasOwnProperty shorthand */ -/******/ (() => { -/******/ __webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop)) -/******/ })(); -/******/ -/******/ /* webpack/runtime/make namespace object */ -/******/ (() => { -/******/ // define __esModule on exports -/******/ __webpack_require__.r = (exports) => { -/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) { -/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' }); -/******/ } -/******/ Object.defineProperty(exports, '__esModule', { value: true }); -/******/ }; -/******/ })(); -/******/ -/******/ /* webpack/runtime/nonce */ -/******/ (() => { -/******/ __webpack_require__.nc = undefined; -/******/ })(); -/******/ -/************************************************************************/ -var __webpack_exports__ = {}; -// This entry need to be wrapped in an IIFE because it need to be isolated against other modules in the chunk. -(() => { -/*!********************************!*\ - !*** ./srcjs/widgets/uPlot.js ***! - \********************************/ -__webpack_require__.r(__webpack_exports__); -/* harmony export */ __webpack_require__.d(__webpack_exports__, { -/* harmony export */ "drawPoints": () => (/* binding */ drawPoints), -/* harmony export */ "uPlot": () => (/* reexport safe */ uplot__WEBPACK_IMPORTED_MODULE_1__["default"]) -/* harmony export */ }); -/* harmony import */ var widgets__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! widgets */ "widgets"); -/* harmony import */ var widgets__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(widgets__WEBPACK_IMPORTED_MODULE_0__); -/* harmony import */ var uplot__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! uplot */ "./node_modules/uplot/dist/uPlot.esm.js"); -/* harmony import */ var uplot_dist_uPlot_min_css__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! uplot/dist/uPlot.min.css */ "./node_modules/uplot/dist/uPlot.min.css"); -/* harmony import */ var _modules_utils__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../modules/utils */ "./srcjs/modules/utils.js"); -/* harmony import */ var _modules_stack__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../modules/stack */ "./srcjs/modules/stack.js"); - - - - - - -const resizer = el => { - const func = (u, init) => { - const resizeObserver = new ResizeObserver(entries => { - for (let entry of entries) { - let adjHeight = entry.contentRect.height; - let adjWidth = entry.contentRect.width; - const extras = u.root.querySelectorAll(".u-legend, .u-title"); - for (let extra of extras) { - adjHeight -= Math.floor(extra.offsetHeight); - } - u.setSize({ width: adjWidth, height: adjHeight }); - } - }); - resizeObserver.observe(el); - el.appendChild(u.root); - init(); - }; - return func; -}; - -const drawPoints = (u, seriesIdx, idx0, idx1) => { - const size = 5 * devicePixelRatio; - - uplot__WEBPACK_IMPORTED_MODULE_1__["default"].orient( - u, - seriesIdx, - ( - series, - dataX, - dataY, - scaleX, - scaleY, - valToPosX, - valToPosY, - xOff, - yOff, - xDim, - yDim, - moveTo, - lineTo, - rect, - arc - ) => { - let d = u.data[seriesIdx]; - - u.ctx.fillStyle = series.stroke(); - - let deg360 = 2 * Math.PI; - - console.time("points"); - - // let cir = new Path2D(); - // cir.moveTo(0, 0); - // arc(cir, 0, 0, 3, 0, deg360); - - // Create transformation matrix that moves 200 points to the right - // let m = document.createElementNS('http://www.w3.org/2000/svg', 'svg').createSVGMatrix(); - // m.a = 1; m.b = 0; - // m.c = 0; m.d = 1; - // m.e = 200; m.f = 0; - - let p = new Path2D(); - - for (let i = 0; i < d[0].length; i++) { - let xVal = d[0][i]; - let yVal = d[1][i]; - - if ( - xVal >= scaleX.min && - xVal <= scaleX.max && - yVal >= scaleY.min && - yVal <= scaleY.max - ) { - let cx = valToPosX(xVal, scaleX, xDim, xOff); - let cy = valToPosY(yVal, scaleY, yDim, yOff); - - p.moveTo(cx + size / 2, cy); - // arc(p, cx, cy, 3, 0, deg360); - arc(p, cx, cy, size / 2, 0, deg360); - - // m.e = cx; - // m.f = cy; - // p.addPath(cir, m); - - // qt.add({x: cx - 1.5, y: cy - 1.5, w: 3, h: 3, sidx: seriesIdx, didx: i}); - } - } - - console.timeEnd("points"); - - u.ctx.fill(p); - } - ); - - return null; -}; - -HTMLWidgets.widget({ - name: "uPlot", - - type: "output", - - factory: function(el, width, height) { - var plot, options, data; - - return { - renderValue: function(x) { - if (typeof plot !== "undefined") { - plot.destroy(); - } - options = x.options; - options.width = width; - options.height = height; - data = x.data; - if (x.stacked) { - if (!options.hooks) - options.hooks = {}; - options.hooks.init = [ - u => { - [...u.root.querySelectorAll(".u-legend .u-series")].forEach( - (el, i) => { - if (u.series[i]._hide) { - el.style.display = "none"; - } - } - ); - } - ]; - plot = (0,_modules_stack__WEBPACK_IMPORTED_MODULE_4__.stackedChart)( - options.title, - options.series, - data, - null, - width, - height, - options.hooks, - resizer(el) - ); - } else { - plot = new uplot__WEBPACK_IMPORTED_MODULE_1__["default"](options, data, resizer(el)); - } - }, - getWidget: function() { - return plot; - }, - resize: function(width, height) {} - }; - } -}); - -if (HTMLWidgets.shinyMode) { - Shiny.addCustomMessageHandler("uplot-api", function(obj) { - var plot = _modules_utils__WEBPACK_IMPORTED_MODULE_3__.getWidget(obj.id); - if (typeof plot != "undefined") { - plot[obj.name].apply(null, obj.args); - } - }); - Shiny.addCustomMessageHandler("uplot-setData", function(obj) { - var plot = _modules_utils__WEBPACK_IMPORTED_MODULE_3__.getWidget(obj.id); - if (typeof plot != "undefined") { - plot.setData(obj.data); - } - }); -} - - - - -})(); - -var __webpack_export_target__ = window; -for(var i in __webpack_exports__) __webpack_export_target__[i] = __webpack_exports__[i]; -if(__webpack_exports__.__esModule) Object.defineProperty(__webpack_export_target__, "__esModule", { value: true }); -/******/ })() -; -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"uPlot.js","mappings":";;;;;;;;;;;;;;;;;;AAAA;AACgG;AACjB;AAC/E,8BAA8B,sEAA2B,CAAC,+EAAqC;AAC/F;AACA,+FAA+F,wBAAwB,QAAQ,wMAAwM,iBAAiB,oBAAoB,UAAU,mBAAmB,gBAAgB,mBAAmB,SAAS,mBAAmB,mBAAmB,mBAAmB,oBAAoB,UAAU,kBAAkB,eAAe,eAAe,mBAAmB,YAAY,cAAc,SAAS,oBAAoB,WAAW,gBAAgB,aAAa,oBAAoB,WAAW,gBAAgB,aAAa,uBAAuB,cAAc,oBAAoB,cAAc,kBAAkB,kBAAkB,uBAAuB,uBAAuB,qBAAqB,WAAW,YAAY,kBAAkB,yCAAyC,4BAA4B,eAAe,wBAAwB,iCAAiC,eAAe,eAAe,cAAc,cAAc,iBAAiB,sBAAsB,cAAc,WAAW,6BAA6B,mBAAmB,sBAAsB,0BAA0B,mBAAmB,QAAQ,OAAO,qBAAqB,uBAAuB,cAAc,sCAAsC,aAAa,kCAAkC,sCAAsC,YAAY,mCAAmC,cAAc,mBAAmB,OAAO,QAAQ,mBAAmB,gBAAgB,qBAAqB,uBAAuB,aAAa,oHAAoH,0FAA0F,eAAe,OAAO,0lCAA0lC,wBAAwB,QAAQ,wMAAwM,iBAAiB,oBAAoB,UAAU,mBAAmB,gBAAgB,mBAAmB,SAAS,mBAAmB,mBAAmB,mBAAmB,oBAAoB,UAAU,kBAAkB,eAAe,eAAe,mBAAmB,YAAY,cAAc,SAAS,oBAAoB,WAAW,gBAAgB,aAAa,oBAAoB,WAAW,gBAAgB,aAAa,uBAAuB,cAAc,oBAAoB,cAAc,kBAAkB,kBAAkB,uBAAuB,uBAAuB,qBAAqB,WAAW,YAAY,kBAAkB,yCAAyC,4BAA4B,eAAe,wBAAwB,iCAAiC,eAAe,eAAe,cAAc,cAAc,iBAAiB,sBAAsB,cAAc,WAAW,6BAA6B,mBAAmB,sBAAsB,0BAA0B,mBAAmB,QAAQ,OAAO,qBAAqB,uBAAuB,cAAc,sCAAsC,aAAa,kCAAkC,sCAAsC,YAAY,mCAAmC,cAAc,mBAAmB,OAAO,QAAQ,mBAAmB,gBAAgB,qBAAqB,uBAAuB,aAAa,oHAAoH,0FAA0F,eAAe,mBAAmB;AACn0J;AACA,iEAAe,uBAAuB,EAAC;;;;;;;;;;;ACP1B;;AAEb;AACA;AACA;AACA;AACA;AACA,iBAAiB;;AAEjB;AACA;AACA;AACA;;AAEA;AACA,qDAAqD;AACrD;;AAEA;AACA,gDAAgD;AAChD;;AAEA;AACA,qFAAqF;AACrF;;AAEA;;AAEA;AACA,qBAAqB;AACrB;;AAEA;AACA,qBAAqB;AACrB;;AAEA;AACA,qBAAqB;AACrB;;AAEA;AACA,KAAK;AACL,KAAK;;;AAGL;AACA;AACA;AACA;;AAEA;;AAEA;AACA,sBAAsB,iBAAiB;AACvC;;AAEA;AACA;AACA;AACA;AACA;;AAEA,qBAAqB,qBAAqB;AAC1C;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,UAAU;AACV,sFAAsF,qBAAqB;AAC3G;AACA;AACA;;AAEA;AACA;AACA;AACA,UAAU;AACV,iDAAiD,qBAAqB;AACtE;AACA;AACA;;AAEA;AACA;AACA;AACA,UAAU;AACV,sDAAsD,qBAAqB;AAC3E;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;;;;;;;;;ACrGa;;AAEb;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,uDAAuD,cAAc;AACrE;AACA;AACA;AACA,KAAK;AACL;AACA;;AAEA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACpBA,MAAqF;AACrF,MAA2E;AAC3E,MAAkF;AAClF,MAAqG;AACrG,MAA8F;AAC9F,MAA8F;AAC9F,MAA6F;AAC7F;AACA;;AAEA;;AAEA,4BAA4B,wFAAmB;AAC/C,wBAAwB,qGAAa;;AAErC,uBAAuB,0FAAa;AACpC;AACA,iBAAiB,kFAAM;AACvB,6BAA6B,yFAAkB;;AAE/C,aAAa,6FAAG,CAAC,6EAAO;;;;AAIuC;AAC/D,OAAO,iEAAe,6EAAO,IAAI,oFAAc,GAAG,oFAAc,YAAY,EAAC;;;;;;;;;;;AC1BhE;;AAEb;;AAEA;AACA;;AAEA,kBAAkB,wBAAwB;AAC1C;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA,kBAAkB,iBAAiB;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA,MAAM;AACN;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,oBAAoB,4BAA4B;AAChD;AACA;AACA;AACA;;AAEA;;AAEA,qBAAqB,6BAA6B;AAClD;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;;;;ACvGa;;AAEb;AACA;;AAEA;AACA;AACA,sDAAsD;;AAEtD;AACA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;;AAGA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;;;;;;;;;ACtCa;;AAEb;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;;;;;;;;;ACVa;;AAEb;AACA;AACA,cAAc,KAAwC,GAAG,sBAAiB,GAAG,CAAI;;AAEjF;AACA;AACA;AACA;;AAEA;;;;;;;;;;ACXa;;AAEb;AACA;AACA;;AAEA;AACA,kDAAkD;AAClD;;AAEA;AACA,0CAA0C;AAC1C;;AAEA;;AAEA;AACA,iFAAiF;AACjF;;AAEA;;AAEA;AACA,aAAa;AACb;;AAEA;AACA,aAAa;AACb;;AAEA;AACA,aAAa;AACb;;AAEA;;AAEA;AACA,yDAAyD;AACzD,IAAI;;AAEJ;;;AAGA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;;AAEA;;;;;;;;;;ACrEa;;AAEb;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;;;;;;;;;;;;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,yCAAyC,gBAAgB,6BAA6B,gBAAgB;AACtG;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,gBAAgB;AAChB,iBAAiB;;AAEjB;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA,oCAAoC,sBAAsB;AAC1D;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,UAAU;AAC9B;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA,mBAAmB,UAAU;AAC7B;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA,wBAAwB,cAAc;AACtC;AACA;;AAEA,kBAAkB,kBAAkB;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA,mBAAmB,cAAc;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA,iBAAiB,iBAAiB;AAClC;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,wBAAwB;AACxB,wBAAwB;AACxB,wBAAwB;;AAExB;AACA;AACA,uCAAuC,qBAAqB;AAC5D;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,kBAAkB,oBAAoB;AACtC;AACA;AACA;;AAEA,kBAAkB,SAAS;AAC3B;AACA;;AAEA;;AAEA;;AAEA;;AAEA,iBAAiB,gBAAgB;AACjC;;AAEA,kBAAkB,oBAAoB;AACtC;AACA;;AAEA,mBAAmB,eAAe;AAClC;;AAEA;;AAEA;;AAEA;;AAEA,mBAAmB,eAAe;AAClC;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,sBAAsB,0CAA0C;AAC3E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,YAAY,UAAU,IAAI;;AAE1B;AACA,0BAA0B;;AAE1B;AACA;;AAEA,kBAAkB,kBAAkB;AACpC;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,iDAAiD,aAAa;AAC9D;AACA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;;AAEA,kBAAkB,KAAK;AACvB;AACA,kBAAkB,EAAE,EAAE,EAAE;AACxB;AACA,0BAA0B,GAAG;;AAE7B,kBAAkB,GAAG;AACrB,kBAAkB,EAAE,EAAE,GAAG;AACzB;AACA;AACA,mBAAmB,GAAG;;AAEtB;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,IAAI;AACtB;AACA,kBAAkB,EAAE;AACpB;AACA;AACA,wBAAwB,IAAI;AAC5B;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,4CAA4C;AAC5C;AACA;AACA;AACA;;AAEA,oBAAoB,mBAAmB;AACvC;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,2BAA2B,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE,IAAI,GAAG;;AAEvD;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;;AAEF;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,EAAE;;AAEF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;;AAEF;AACA;AACA,EAAE;;AAEF;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,sBAAsB;AACtB;AACA,CAAC;;AAED,uBAAuB;AACvB;AACA,CAAC;;AAED,wBAAwB;AACxB;AACA,CAAC;;AAED;AACA;AACA,uBAAuB;;AAEvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;;AAEA,0BAA0B,iBAAiB;AAC3C,8CAA8C;;AAE9C;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA,GAAG;;AAEH;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA,OAAO,cAAc;AACrB;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,UAAU,WAAW;AAChC,WAAW,UAAU,WAAW;AAChC;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;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,4BAA4B;AAC5B;AACA;AACA,CAAC;;AAED;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA,IAAI;AACJ;AACA,oBAAoB,oBAAoB;AACxC;AACA,IAAI;AACJ;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;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;AACA;AACA;AACA;AACA;AACA;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;AACA;;AAEA;AACA;;AAEA;;AAEA,iBAAiB,kBAAkB;AACnC;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA,EAAE;AACF;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA,kBAAkB,iBAAiB;AACnC;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA,sCAAsC,wBAAwB;AAC9D;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA,6BAA6B;AAC7B;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA,8BAA8B,8BAA8B;AAC5D,8BAA8B;;AAE9B;AACA,uBAAuB,sBAAsB;AAC7C,uBAAuB;;AAEvB;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,+BAA+B;AAC/B,+BAA+B;AAC/B,+BAA+B;AAC/B,+BAA+B;AAC/B;AACA;AACA,qDAAqD;AACrD,qDAAqD;AACrD,kEAAkE;AAClE,kEAAkE;;AAElE;AACA;AACA;AACA;;AAEA;AACA,SAAS,kBAAkB;;AAE3B;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA,SAAS,+CAA+C;;AAExD;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,wBAAwB,YAAY;AACpC;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA,mBAAmB;AACnB;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,wCAAwC,wBAAwB;AAChE;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA,mBAAmB;AACnB;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA,wCAAwC,wBAAwB;AAChE;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,KAAK;;AAEL;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA,wCAAwC;;AAExC,SAAS,uCAAuC;;AAEhD;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,KAAK;;AAEL;AACA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;;AAEA,SAAS,WAAW;;AAEpB;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,0FAA0F;;AAE1F;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,0CAA0C,kBAAkB;AAC5D;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;;AAEA,mBAAmB,0GAA0G;;AAE7H;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA,SAAS,SAAS;;AAElB;;AAEA;AACA;AACA;AACA;;AAEA,0CAA0C,wBAAwB;AAClE;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA,yBAAyB;AACzB;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA,GAAG;AACH;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA,wCAAwC,wBAAwB;AAChE;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,mBAAmB;AACnB;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,kBAAkB,WAAW;AAC7B;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,kBAAkB,WAAW;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA,kBAAkB,WAAW;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA,EAAE;AACF;;AAEA;AACA;AACA;AACA,6BAA6B,0BAA0B;AACvD;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA,iDAAiD,aAAa;AAC9D;;AAEA;AACA,iBAAiB;AACjB;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,GAAG;;AAEH;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA,6CAA6C;AAC7C;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA,EAAE;;AAEF;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,EAAE;;AAEF;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA,gCAAgC,sCAAsC,cAAc;AACpF;AACA;AACA,qCAAqC;;AAErC;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA,EAAE;;AAEF;AACA;;AAEA;;AAEA;;AAEA;;AAEA;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;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA,oBAAoB;AACpB;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA,4CAA4C;AAC5C;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,4DAA4D;;AAE5D;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,eAAe;AACf,eAAe;;AAEf;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,MAAM;AACN;AACA;AACA,2BAA2B,cAAc;AACzC,IAAI;;AAEJ;AACA;AACA;AACA;;AAEA;AACA,KAAK;AACL;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,mBAAmB,cAAc;AACjC;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,SAAS,aAAa;AACtB;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,0BAA0B;AAC1B,0BAA0B;AAC1B,0BAA0B;AAC1B,0BAA0B;AAC1B;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA,GAAG;AACH;;AAEA,wCAAwC,eAAe,OAAO,cAAc;;AAE5E;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,qCAAqC,iBAAiB,WAAW;AACjE;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,oCAAoC;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,6BAA6B;AAC7B;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA,mBAAmB,mBAAmB;AACtC;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA,oBAAoB,aAAa;AACjC;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B;AAC/B;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;;AAEJ;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;;AAEJ;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;;AAEJ;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA,IAAI;AACJ;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA,QAAQ,sCAAsC;AAC9C;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA,GAAG;;AAEH;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;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;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,kBAAkB,iBAAiB;AACnC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA,QAAQ,UAAU,UAAU;;AAE5B;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA,GAAG;;AAEH;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA,GAAG;;AAEH;AACA;;AAEA;AACA,kBAAkB,iBAAiB;AACnC;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA,mBAAmB,oBAAoB;AACvC;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA,qBAAqB,mBAAmB;AACxC;;AAEA;AACA;AACA,6CAA6C;AAC7C,0BAA0B;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;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;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA,GAAG;AACH;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,mBAAmB,+BAA+B;AAClD;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,IAAI;;AAEJ;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;;AAEF;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,iBAAiB,SAAS;AAC1B;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,IAAI;AACJ;;AAEA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,sBAAsB;;AAEtB;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA,IAAI;;AAEJ;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,GAAG;AACH;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA,EAAE;;AAEF;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA,mCAAmC,mBAAmB;AACtD;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,IAAI;AACJ;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,uDAAuD;AACvD;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA,+BAA+B;;AAE/B;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,mBAAmB,mBAAmB;AACtC;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA,oBAAoB,mBAAmB;AACvC;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,mCAAmC,mBAAmB;AACtD;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA,WAAW,2BAA2B;;AAEtC;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA,QAAQ,2BAA2B;;AAEnC;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;;AAEA;AACA;AACA;AACA,EAAE;;AAEF;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,EAAE;;AAEF;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAE4B;;;;;;;;;;;;;;;ACnwK5B;AACA;AACA;AACA;AACA;;AAEA,kBAAkB,WAAW;;AAE7B,kBAAkB,iBAAiB;AACnC;;AAEA,kBAAkB,iBAAiB;AACnC;AACA;AACA;AACA,OAAO;;AAEP;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,WAAW;AACX;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA,sBAAsB,iBAAiB;AACvC;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,YAAY,MAAM;;AAElB,sBAAsB,aAAa;AACnC;;AAEA;AACA,6BAA6B;AAC7B;AACA,MAAM;AACN,GAAG;;AAEH;AACA;AACA;AACA;AACA,YAAY,MAAM;;AAElB;AACA;;AAEA;;AAEA,sBAAsB,aAAa;AACnC;;AAEA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA,UAAU,cAAc;AACxB;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA,KAAK;AACL,GAAG;;AAEH;AACA;AACA;AACA;AACA;;AAEO;AACP,QAAQ,aAAa;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;ACjMA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;ACRA;;;;;;UCAA;UACA;;UAEA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;;UAEA;UACA;;UAEA;UACA;UACA;;;;;WCtBA;WACA;WACA;WACA;WACA;WACA,iCAAiC,WAAW;WAC5C;WACA;;;;;WCPA;WACA;WACA;WACA;WACA,yCAAyC,wCAAwC;WACjF;WACA;WACA;;;;;WCPA;;;;;WCAA;WACA;WACA;WACA,uDAAuD,iBAAiB;WACxE;WACA,gDAAgD,aAAa;WAC7D;;;;;WCNA;;;;;;;;;;;;;;;;;;;;;ACAiB;AACS;AACQ;AACQ;AACM;;AAEhD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,oCAAoC;AACxD;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA,EAAE,oDAAY;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,oBAAoB;AACpB,oBAAoB;AACpB,oBAAoB;;AAEpB;;AAEA,sBAAsB,iBAAiB;AACvC;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,qBAAqB,+DAA+D;AACpF;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,4DAAY;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV,qBAAqB,6CAAK;AAC1B;AACA,OAAO;AACP;AACA;AACA,OAAO;AACP;AACA;AACA;AACA,CAAC;;AAED;AACA;AACA,eAAe,qDAAe;AAC9B;AACA;AACA;AACA,GAAG;AACH;AACA,eAAe,qDAAe;AAC9B;AACA;AACA;AACA,GAAG;AACH;;AAE6B","sources":["webpack://uplot/./node_modules/uplot/dist/uPlot.min.css","webpack://uplot/./node_modules/css-loader/dist/runtime/api.js","webpack://uplot/./node_modules/css-loader/dist/runtime/sourceMaps.js","webpack://uplot/./node_modules/uplot/dist/uPlot.min.css?f8fa","webpack://uplot/./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js","webpack://uplot/./node_modules/style-loader/dist/runtime/insertBySelector.js","webpack://uplot/./node_modules/style-loader/dist/runtime/insertStyleElement.js","webpack://uplot/./node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js","webpack://uplot/./node_modules/style-loader/dist/runtime/styleDomAPI.js","webpack://uplot/./node_modules/style-loader/dist/runtime/styleTagTransform.js","webpack://uplot/./node_modules/uplot/dist/uPlot.esm.js","webpack://uplot/./srcjs/modules/stack.js","webpack://uplot/./srcjs/modules/utils.js","webpack://uplot/external window \"HTMLWidgets\"","webpack://uplot/webpack/bootstrap","webpack://uplot/webpack/runtime/compat get default export","webpack://uplot/webpack/runtime/define property getters","webpack://uplot/webpack/runtime/hasOwnProperty shorthand","webpack://uplot/webpack/runtime/make namespace object","webpack://uplot/webpack/runtime/nonce","webpack://uplot/./srcjs/widgets/uPlot.js"],"sourcesContent":["// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../css-loader/dist/runtime/sourceMaps.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".uplot, .uplot *, .uplot *::before, .uplot *::after {box-sizing: border-box;}.uplot {font-family: system-ui, -apple-system, \\\"Segoe UI\\\", Roboto, \\\"Helvetica Neue\\\", Arial, \\\"Noto Sans\\\", sans-serif, \\\"Apple Color Emoji\\\", \\\"Segoe UI Emoji\\\", \\\"Segoe UI Symbol\\\", \\\"Noto Color Emoji\\\";line-height: 1.5;width: min-content;}.u-title {text-align: center;font-size: 18px;font-weight: bold;}.u-wrap {position: relative;user-select: none;}.u-over, .u-under {position: absolute;}.u-under {overflow: hidden;}.uplot canvas {display: block;position: relative;width: 100%;height: 100%;}.u-axis {position: absolute;}.u-legend {font-size: 14px;margin: auto;text-align: center;}.u-inline {display: block;}.u-inline * {display: inline-block;}.u-inline tr {margin-right: 16px;}.u-legend th {font-weight: 600;}.u-legend th > * {vertical-align: middle;display: inline-block;}.u-legend .u-marker {width: 1em;height: 1em;margin-right: 4px;background-clip: padding-box !important;}.u-inline.u-live th::after {content: \\\":\\\";vertical-align: middle;}.u-inline:not(.u-live) .u-value {display: none;}.u-series > * {padding: 4px;}.u-series th {cursor: pointer;}.u-legend .u-off > * {opacity: 0.3;}.u-select {background: rgba(0,0,0,0.07);position: absolute;pointer-events: none;}.u-cursor-x, .u-cursor-y {position: absolute;left: 0;top: 0;pointer-events: none;will-change: transform;z-index: 100;}.u-hz .u-cursor-x, .u-vt .u-cursor-y {height: 100%;border-right: 1px dashed #607D8B;}.u-hz .u-cursor-y, .u-vt .u-cursor-x {width: 100%;border-bottom: 1px dashed #607D8B;}.u-cursor-pt {position: absolute;top: 0;left: 0;border-radius: 50%;border: 0 solid;pointer-events: none;will-change: transform;z-index: 100;/*this has to be !important since we set inline \\\"background\\\" shorthand */background-clip: padding-box !important;}.u-axis.u-off, .u-select.u-off, .u-cursor-x.u-off, .u-cursor-y.u-off, .u-cursor-pt.u-off {display: none;}\", \"\",{\"version\":3,\"sources\":[\"webpack://./node_modules/uplot/dist/uPlot.min.css\"],\"names\":[],\"mappings\":\"AAAA,qDAAqD,sBAAsB,CAAC,CAAC,QAAQ,yLAAyL,CAAC,gBAAgB,CAAC,kBAAkB,CAAC,CAAC,UAAU,kBAAkB,CAAC,eAAe,CAAC,iBAAiB,CAAC,CAAC,SAAS,kBAAkB,CAAC,iBAAiB,CAAC,CAAC,mBAAmB,kBAAkB,CAAC,CAAC,UAAU,gBAAgB,CAAC,CAAC,eAAe,cAAc,CAAC,kBAAkB,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC,SAAS,kBAAkB,CAAC,CAAC,WAAW,eAAe,CAAC,YAAY,CAAC,kBAAkB,CAAC,CAAC,WAAW,cAAc,CAAC,CAAC,aAAa,qBAAqB,CAAC,CAAC,cAAc,kBAAkB,CAAC,CAAC,cAAc,gBAAgB,CAAC,CAAC,kBAAkB,sBAAsB,CAAC,qBAAqB,CAAC,CAAC,qBAAqB,UAAU,CAAC,WAAW,CAAC,iBAAiB,CAAC,uCAAuC,CAAC,CAAC,4BAA4B,YAAY,CAAC,sBAAsB,CAAC,CAAC,iCAAiC,aAAa,CAAC,CAAC,eAAe,YAAY,CAAC,CAAC,cAAc,eAAe,CAAC,CAAC,sBAAsB,YAAY,CAAC,CAAC,WAAW,4BAA4B,CAAC,kBAAkB,CAAC,oBAAoB,CAAC,CAAC,0BAA0B,kBAAkB,CAAC,OAAO,CAAC,MAAM,CAAC,oBAAoB,CAAC,sBAAsB,CAAC,YAAY,CAAC,CAAC,sCAAsC,YAAY,CAAC,gCAAgC,CAAC,CAAC,sCAAsC,WAAW,CAAC,iCAAiC,CAAC,CAAC,cAAc,kBAAkB,CAAC,MAAM,CAAC,OAAO,CAAC,kBAAkB,CAAC,eAAe,CAAC,oBAAoB,CAAC,sBAAsB,CAAC,YAAY,CAAC,wEAAwE,CAAC,uCAAuC,CAAC,CAAC,0FAA0F,aAAa,CAAC\",\"sourcesContent\":[\".uplot, .uplot *, .uplot *::before, .uplot *::after {box-sizing: border-box;}.uplot {font-family: system-ui, -apple-system, \\\"Segoe UI\\\", Roboto, \\\"Helvetica Neue\\\", Arial, \\\"Noto Sans\\\", sans-serif, \\\"Apple Color Emoji\\\", \\\"Segoe UI Emoji\\\", \\\"Segoe UI Symbol\\\", \\\"Noto Color Emoji\\\";line-height: 1.5;width: min-content;}.u-title {text-align: center;font-size: 18px;font-weight: bold;}.u-wrap {position: relative;user-select: none;}.u-over, .u-under {position: absolute;}.u-under {overflow: hidden;}.uplot canvas {display: block;position: relative;width: 100%;height: 100%;}.u-axis {position: absolute;}.u-legend {font-size: 14px;margin: auto;text-align: center;}.u-inline {display: block;}.u-inline * {display: inline-block;}.u-inline tr {margin-right: 16px;}.u-legend th {font-weight: 600;}.u-legend th > * {vertical-align: middle;display: inline-block;}.u-legend .u-marker {width: 1em;height: 1em;margin-right: 4px;background-clip: padding-box !important;}.u-inline.u-live th::after {content: \\\":\\\";vertical-align: middle;}.u-inline:not(.u-live) .u-value {display: none;}.u-series > * {padding: 4px;}.u-series th {cursor: pointer;}.u-legend .u-off > * {opacity: 0.3;}.u-select {background: rgba(0,0,0,0.07);position: absolute;pointer-events: none;}.u-cursor-x, .u-cursor-y {position: absolute;left: 0;top: 0;pointer-events: none;will-change: transform;z-index: 100;}.u-hz .u-cursor-x, .u-vt .u-cursor-y {height: 100%;border-right: 1px dashed #607D8B;}.u-hz .u-cursor-y, .u-vt .u-cursor-x {width: 100%;border-bottom: 1px dashed #607D8B;}.u-cursor-pt {position: absolute;top: 0;left: 0;border-radius: 50%;border: 0 solid;pointer-events: none;will-change: transform;z-index: 100;/*this has to be !important since we set inline \\\"background\\\" shorthand */background-clip: padding-box !important;}.u-axis.u-off, .u-select.u-off, .u-cursor-x.u-off, .u-cursor-y.u-off, .u-cursor-pt.u-off {display: none;}\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","\"use strict\";\n\n/*\n  MIT License http://www.opensource.org/licenses/mit-license.php\n  Author Tobias Koppers @sokra\n*/\nmodule.exports = function (cssWithMappingToString) {\n  var list = []; // return the list of modules as css string\n\n  list.toString = function toString() {\n    return this.map(function (item) {\n      var content = \"\";\n      var needLayer = typeof item[5] !== \"undefined\";\n\n      if (item[4]) {\n        content += \"@supports (\".concat(item[4], \") {\");\n      }\n\n      if (item[2]) {\n        content += \"@media \".concat(item[2], \" {\");\n      }\n\n      if (needLayer) {\n        content += \"@layer\".concat(item[5].length > 0 ? \" \".concat(item[5]) : \"\", \" {\");\n      }\n\n      content += cssWithMappingToString(item);\n\n      if (needLayer) {\n        content += \"}\";\n      }\n\n      if (item[2]) {\n        content += \"}\";\n      }\n\n      if (item[4]) {\n        content += \"}\";\n      }\n\n      return content;\n    }).join(\"\");\n  }; // import a list of modules into the list\n\n\n  list.i = function i(modules, media, dedupe, supports, layer) {\n    if (typeof modules === \"string\") {\n      modules = [[null, modules, undefined]];\n    }\n\n    var alreadyImportedModules = {};\n\n    if (dedupe) {\n      for (var k = 0; k < this.length; k++) {\n        var id = this[k][0];\n\n        if (id != null) {\n          alreadyImportedModules[id] = true;\n        }\n      }\n    }\n\n    for (var _k = 0; _k < modules.length; _k++) {\n      var item = [].concat(modules[_k]);\n\n      if (dedupe && alreadyImportedModules[item[0]]) {\n        continue;\n      }\n\n      if (typeof layer !== \"undefined\") {\n        if (typeof item[5] === \"undefined\") {\n          item[5] = layer;\n        } else {\n          item[1] = \"@layer\".concat(item[5].length > 0 ? \" \".concat(item[5]) : \"\", \" {\").concat(item[1], \"}\");\n          item[5] = layer;\n        }\n      }\n\n      if (media) {\n        if (!item[2]) {\n          item[2] = media;\n        } else {\n          item[1] = \"@media \".concat(item[2], \" {\").concat(item[1], \"}\");\n          item[2] = media;\n        }\n      }\n\n      if (supports) {\n        if (!item[4]) {\n          item[4] = \"\".concat(supports);\n        } else {\n          item[1] = \"@supports (\".concat(item[4], \") {\").concat(item[1], \"}\");\n          item[4] = supports;\n        }\n      }\n\n      list.push(item);\n    }\n  };\n\n  return list;\n};","\"use strict\";\n\nmodule.exports = function (item) {\n  var content = item[1];\n  var cssMapping = item[3];\n\n  if (!cssMapping) {\n    return content;\n  }\n\n  if (typeof btoa === \"function\") {\n    var base64 = btoa(unescape(encodeURIComponent(JSON.stringify(cssMapping))));\n    var data = \"sourceMappingURL=data:application/json;charset=utf-8;base64,\".concat(base64);\n    var sourceMapping = \"/*# \".concat(data, \" */\");\n    var sourceURLs = cssMapping.sources.map(function (source) {\n      return \"/*# sourceURL=\".concat(cssMapping.sourceRoot || \"\").concat(source, \" */\");\n    });\n    return [content].concat(sourceURLs).concat([sourceMapping]).join(\"\\n\");\n  }\n\n  return [content].join(\"\\n\");\n};","\n      import API from \"!../../style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../css-loader/dist/cjs.js!./uPlot.min.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\n\n      options.insert = insertFn.bind(null, \"head\");\n    \noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../css-loader/dist/cjs.js!./uPlot.min.css\";\n       export default content && content.locals ? content.locals : undefined;\n","\"use strict\";\n\nvar stylesInDOM = [];\n\nfunction getIndexByIdentifier(identifier) {\n  var result = -1;\n\n  for (var i = 0; i < stylesInDOM.length; i++) {\n    if (stylesInDOM[i].identifier === identifier) {\n      result = i;\n      break;\n    }\n  }\n\n  return result;\n}\n\nfunction modulesToDom(list, options) {\n  var idCountMap = {};\n  var identifiers = [];\n\n  for (var i = 0; i < list.length; i++) {\n    var item = list[i];\n    var id = options.base ? item[0] + options.base : item[0];\n    var count = idCountMap[id] || 0;\n    var identifier = \"\".concat(id, \" \").concat(count);\n    idCountMap[id] = count + 1;\n    var indexByIdentifier = getIndexByIdentifier(identifier);\n    var obj = {\n      css: item[1],\n      media: item[2],\n      sourceMap: item[3],\n      supports: item[4],\n      layer: item[5]\n    };\n\n    if (indexByIdentifier !== -1) {\n      stylesInDOM[indexByIdentifier].references++;\n      stylesInDOM[indexByIdentifier].updater(obj);\n    } else {\n      var updater = addElementStyle(obj, options);\n      options.byIndex = i;\n      stylesInDOM.splice(i, 0, {\n        identifier: identifier,\n        updater: updater,\n        references: 1\n      });\n    }\n\n    identifiers.push(identifier);\n  }\n\n  return identifiers;\n}\n\nfunction addElementStyle(obj, options) {\n  var api = options.domAPI(options);\n  api.update(obj);\n\n  var updater = function updater(newObj) {\n    if (newObj) {\n      if (newObj.css === obj.css && newObj.media === obj.media && newObj.sourceMap === obj.sourceMap && newObj.supports === obj.supports && newObj.layer === obj.layer) {\n        return;\n      }\n\n      api.update(obj = newObj);\n    } else {\n      api.remove();\n    }\n  };\n\n  return updater;\n}\n\nmodule.exports = function (list, options) {\n  options = options || {};\n  list = list || [];\n  var lastIdentifiers = modulesToDom(list, options);\n  return function update(newList) {\n    newList = newList || [];\n\n    for (var i = 0; i < lastIdentifiers.length; i++) {\n      var identifier = lastIdentifiers[i];\n      var index = getIndexByIdentifier(identifier);\n      stylesInDOM[index].references--;\n    }\n\n    var newLastIdentifiers = modulesToDom(newList, options);\n\n    for (var _i = 0; _i < lastIdentifiers.length; _i++) {\n      var _identifier = lastIdentifiers[_i];\n\n      var _index = getIndexByIdentifier(_identifier);\n\n      if (stylesInDOM[_index].references === 0) {\n        stylesInDOM[_index].updater();\n\n        stylesInDOM.splice(_index, 1);\n      }\n    }\n\n    lastIdentifiers = newLastIdentifiers;\n  };\n};","\"use strict\";\n\nvar memo = {};\n/* istanbul ignore next  */\n\nfunction getTarget(target) {\n  if (typeof memo[target] === \"undefined\") {\n    var styleTarget = document.querySelector(target); // Special case to return head of iframe instead of iframe itself\n\n    if (window.HTMLIFrameElement && styleTarget instanceof window.HTMLIFrameElement) {\n      try {\n        // This will throw an exception if access to iframe is blocked\n        // due to cross-origin restrictions\n        styleTarget = styleTarget.contentDocument.head;\n      } catch (e) {\n        // istanbul ignore next\n        styleTarget = null;\n      }\n    }\n\n    memo[target] = styleTarget;\n  }\n\n  return memo[target];\n}\n/* istanbul ignore next  */\n\n\nfunction insertBySelector(insert, style) {\n  var target = getTarget(insert);\n\n  if (!target) {\n    throw new Error(\"Couldn't find a style target. This probably means that the value for the 'insert' parameter is invalid.\");\n  }\n\n  target.appendChild(style);\n}\n\nmodule.exports = insertBySelector;","\"use strict\";\n\n/* istanbul ignore next  */\nfunction insertStyleElement(options) {\n  var element = document.createElement(\"style\");\n  options.setAttributes(element, options.attributes);\n  options.insert(element, options.options);\n  return element;\n}\n\nmodule.exports = insertStyleElement;","\"use strict\";\n\n/* istanbul ignore next  */\nfunction setAttributesWithoutAttributes(styleElement) {\n  var nonce = typeof __webpack_nonce__ !== \"undefined\" ? __webpack_nonce__ : null;\n\n  if (nonce) {\n    styleElement.setAttribute(\"nonce\", nonce);\n  }\n}\n\nmodule.exports = setAttributesWithoutAttributes;","\"use strict\";\n\n/* istanbul ignore next  */\nfunction apply(styleElement, options, obj) {\n  var css = \"\";\n\n  if (obj.supports) {\n    css += \"@supports (\".concat(obj.supports, \") {\");\n  }\n\n  if (obj.media) {\n    css += \"@media \".concat(obj.media, \" {\");\n  }\n\n  var needLayer = typeof obj.layer !== \"undefined\";\n\n  if (needLayer) {\n    css += \"@layer\".concat(obj.layer.length > 0 ? \" \".concat(obj.layer) : \"\", \" {\");\n  }\n\n  css += obj.css;\n\n  if (needLayer) {\n    css += \"}\";\n  }\n\n  if (obj.media) {\n    css += \"}\";\n  }\n\n  if (obj.supports) {\n    css += \"}\";\n  }\n\n  var sourceMap = obj.sourceMap;\n\n  if (sourceMap && typeof btoa !== \"undefined\") {\n    css += \"\\n/*# sourceMappingURL=data:application/json;base64,\".concat(btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap)))), \" */\");\n  } // For old IE\n\n  /* istanbul ignore if  */\n\n\n  options.styleTagTransform(css, styleElement, options.options);\n}\n\nfunction removeStyleElement(styleElement) {\n  // istanbul ignore if\n  if (styleElement.parentNode === null) {\n    return false;\n  }\n\n  styleElement.parentNode.removeChild(styleElement);\n}\n/* istanbul ignore next  */\n\n\nfunction domAPI(options) {\n  var styleElement = options.insertStyleElement(options);\n  return {\n    update: function update(obj) {\n      apply(styleElement, options, obj);\n    },\n    remove: function remove() {\n      removeStyleElement(styleElement);\n    }\n  };\n}\n\nmodule.exports = domAPI;","\"use strict\";\n\n/* istanbul ignore next  */\nfunction styleTagTransform(css, styleElement) {\n  if (styleElement.styleSheet) {\n    styleElement.styleSheet.cssText = css;\n  } else {\n    while (styleElement.firstChild) {\n      styleElement.removeChild(styleElement.firstChild);\n    }\n\n    styleElement.appendChild(document.createTextNode(css));\n  }\n}\n\nmodule.exports = styleTagTransform;","/**\n* Copyright (c) 2022, Leon Sorokin\n* All rights reserved. (MIT Licensed)\n*\n* uPlot.js (μPlot)\n* A small, fast chart for time series, lines, areas, ohlc & bars\n* https://github.com/leeoniya/uPlot (v1.6.22)\n*/\n\nconst FEAT_TIME          = true;\n\nconst pre = \"u-\";\n\nconst UPLOT          =       \"uplot\";\nconst ORI_HZ         = pre + \"hz\";\nconst ORI_VT         = pre + \"vt\";\nconst TITLE          = pre + \"title\";\nconst WRAP           = pre + \"wrap\";\nconst UNDER          = pre + \"under\";\nconst OVER           = pre + \"over\";\nconst AXIS           = pre + \"axis\";\nconst OFF            = pre + \"off\";\nconst SELECT         = pre + \"select\";\nconst CURSOR_X       = pre + \"cursor-x\";\nconst CURSOR_Y       = pre + \"cursor-y\";\nconst CURSOR_PT      = pre + \"cursor-pt\";\nconst LEGEND         = pre + \"legend\";\nconst LEGEND_LIVE    = pre + \"live\";\nconst LEGEND_INLINE  = pre + \"inline\";\nconst LEGEND_THEAD   = pre + \"thead\";\nconst LEGEND_SERIES  = pre + \"series\";\nconst LEGEND_MARKER  = pre + \"marker\";\nconst LEGEND_LABEL   = pre + \"label\";\nconst LEGEND_VALUE   = pre + \"value\";\n\nconst WIDTH       = \"width\";\nconst HEIGHT      = \"height\";\nconst TOP         = \"top\";\nconst BOTTOM      = \"bottom\";\nconst LEFT        = \"left\";\nconst RIGHT       = \"right\";\nconst hexBlack    = \"#000\";\nconst transparent = hexBlack + \"0\";\n\nconst mousemove   = \"mousemove\";\nconst mousedown   = \"mousedown\";\nconst mouseup     = \"mouseup\";\nconst mouseenter  = \"mouseenter\";\nconst mouseleave  = \"mouseleave\";\nconst dblclick    = \"dblclick\";\nconst resize      = \"resize\";\nconst scroll      = \"scroll\";\n\nconst change      = \"change\";\nconst dppxchange  = \"dppxchange\";\n\nconst domEnv = typeof window != 'undefined';\n\nconst doc = domEnv ? document  : null;\nconst win = domEnv ? window    : null;\nconst nav = domEnv ? navigator : null;\n\nlet pxRatio;\n\nlet query;\n\nfunction setPxRatio() {\n\tlet _pxRatio = devicePixelRatio;\n\n\t// during print preview, Chrome fires off these dppx queries even without changes\n\tif (pxRatio != _pxRatio) {\n\t\tpxRatio = _pxRatio;\n\n\t\tquery && off(change, query, setPxRatio);\n\t\tquery = matchMedia(`(min-resolution: ${pxRatio - 0.001}dppx) and (max-resolution: ${pxRatio + 0.001}dppx)`);\n\t\ton(change, query, setPxRatio);\n\n\t\twin.dispatchEvent(new CustomEvent(dppxchange));\n\t}\n}\n\nfunction addClass(el, c) {\n\tif (c != null) {\n\t\tlet cl = el.classList;\n\t\t!cl.contains(c) && cl.add(c);\n\t}\n}\n\nfunction remClass(el, c) {\n\tlet cl = el.classList;\n\tcl.contains(c) && cl.remove(c);\n}\n\nfunction setStylePx(el, name, value) {\n\tel.style[name] = value + \"px\";\n}\n\nfunction placeTag(tag, cls, targ, refEl) {\n\tlet el = doc.createElement(tag);\n\n\tif (cls != null)\n\t\taddClass(el, cls);\n\n\tif (targ != null)\n\t\ttarg.insertBefore(el, refEl);\n\n\treturn el;\n}\n\nfunction placeDiv(cls, targ) {\n\treturn placeTag(\"div\", cls, targ);\n}\n\nconst xformCache = new WeakMap();\n\nfunction elTrans(el, xPos, yPos, xMax, yMax) {\n\tlet xform = \"translate(\" + xPos + \"px,\" + yPos + \"px)\";\n\tlet xformOld = xformCache.get(el);\n\n\tif (xform != xformOld) {\n\t\tel.style.transform = xform;\n\t\txformCache.set(el, xform);\n\n\t\tif (xPos < 0 || yPos < 0 || xPos > xMax || yPos > yMax)\n\t\t\taddClass(el, OFF);\n\t\telse\n\t\t\tremClass(el, OFF);\n\t}\n}\n\nconst colorCache = new WeakMap();\n\nfunction elColor(el, background, borderColor) {\n\tlet newColor = background + borderColor;\n\tlet oldColor = colorCache.get(el);\n\n\tif (newColor != oldColor) {\n\t\tcolorCache.set(el, newColor);\n\t\tel.style.background = background;\n\t\tel.style.borderColor = borderColor;\n\t}\n}\n\nconst sizeCache = new WeakMap();\n\nfunction elSize(el, newWid, newHgt, centered) {\n\tlet newSize = newWid + \"\" + newHgt;\n\tlet oldSize = sizeCache.get(el);\n\n\tif (newSize != oldSize) {\n\t\tsizeCache.set(el, newSize);\n\t\tel.style.height = newHgt + \"px\";\n\t\tel.style.width = newWid + \"px\";\n\t\tel.style.marginLeft = centered ? -newWid/2 + \"px\" : 0;\n\t\tel.style.marginTop = centered ? -newHgt/2 + \"px\" : 0;\n\t}\n}\n\nconst evOpts = {passive: true};\nconst evOpts2 = {...evOpts, capture: true};\n\nfunction on(ev, el, cb, capt) {\n\tel.addEventListener(ev, cb, capt ? evOpts2 : evOpts);\n}\n\nfunction off(ev, el, cb, capt) {\n\tel.removeEventListener(ev, cb, capt ? evOpts2 : evOpts);\n}\n\ndomEnv && setPxRatio();\n\n// binary search for index of closest value\nfunction closestIdx(num, arr, lo, hi) {\n\tlet mid;\n\tlo = lo || 0;\n\thi = hi || arr.length - 1;\n\tlet bitwise = hi <= 2147483647;\n\n\twhile (hi - lo > 1) {\n\t\tmid = bitwise ? (lo + hi) >> 1 : floor((lo + hi) / 2);\n\n\t\tif (arr[mid] < num)\n\t\t\tlo = mid;\n\t\telse\n\t\t\thi = mid;\n\t}\n\n\tif (num - arr[lo] <= arr[hi] - num)\n\t\treturn lo;\n\n\treturn hi;\n}\n\nfunction nonNullIdx(data, _i0, _i1, dir) {\n\tfor (let i = dir == 1 ? _i0 : _i1; i >= _i0 && i <= _i1; i += dir) {\n\t\tif (data[i] != null)\n\t\t\treturn i;\n\t}\n\n\treturn -1;\n}\n\nfunction getMinMax(data, _i0, _i1, sorted) {\n//\tconsole.log(\"getMinMax()\");\n\n\tlet _min = inf;\n\tlet _max = -inf;\n\n\tif (sorted == 1) {\n\t\t_min = data[_i0];\n\t\t_max = data[_i1];\n\t}\n\telse if (sorted == -1) {\n\t\t_min = data[_i1];\n\t\t_max = data[_i0];\n\t}\n\telse {\n\t\tfor (let i = _i0; i <= _i1; i++) {\n\t\t\tif (data[i] != null) {\n\t\t\t\t_min = min(_min, data[i]);\n\t\t\t\t_max = max(_max, data[i]);\n\t\t\t}\n\t\t}\n\t}\n\n\treturn [_min, _max];\n}\n\nfunction getMinMaxLog(data, _i0, _i1) {\n//\tconsole.log(\"getMinMax()\");\n\n\tlet _min = inf;\n\tlet _max = -inf;\n\n\tfor (let i = _i0; i <= _i1; i++) {\n\t\tif (data[i] > 0) {\n\t\t\t_min = min(_min, data[i]);\n\t\t\t_max = max(_max, data[i]);\n\t\t}\n\t}\n\n\treturn [\n\t\t_min ==  inf ?  1 : _min,\n\t\t_max == -inf ? 10 : _max,\n\t];\n}\n\nconst _fixedTuple = [0, 0];\n\nfunction fixIncr(minIncr, maxIncr, minExp, maxExp) {\n\t_fixedTuple[0] = minExp < 0 ? roundDec(minIncr, -minExp) : minIncr;\n\t_fixedTuple[1] = maxExp < 0 ? roundDec(maxIncr, -maxExp) : maxIncr;\n\treturn _fixedTuple;\n}\n\nfunction rangeLog(min, max, base, fullMags) {\n\tlet minSign = sign(min);\n\n\tlet logFn = base == 10 ? log10 : log2;\n\n\tif (min == max) {\n\t\tif (minSign == -1) {\n\t\t\tmin *= base;\n\t\t\tmax /= base;\n\t\t}\n\t\telse {\n\t\t\tmin /= base;\n\t\t\tmax *= base;\n\t\t}\n\t}\n\n\tlet minExp, maxExp, minMaxIncrs;\n\n\tif (fullMags) {\n\t\tminExp = floor(logFn(min));\n\t\tmaxExp =  ceil(logFn(max));\n\n\t\tminMaxIncrs = fixIncr(pow(base, minExp), pow(base, maxExp), minExp, maxExp);\n\n\t\tmin = minMaxIncrs[0];\n\t\tmax = minMaxIncrs[1];\n\t}\n\telse {\n\t\tminExp = floor(logFn(abs(min)));\n\t\tmaxExp = floor(logFn(abs(max)));\n\n\t\tminMaxIncrs = fixIncr(pow(base, minExp), pow(base, maxExp), minExp, maxExp);\n\n\t\tmin = incrRoundDn(min, minMaxIncrs[0]);\n\t\tmax = incrRoundUp(max, minMaxIncrs[1]);\n\t}\n\n\treturn [min, max];\n}\n\nfunction rangeAsinh(min, max, base, fullMags) {\n\tlet minMax = rangeLog(min, max, base, fullMags);\n\n\tif (min == 0)\n\t\tminMax[0] = 0;\n\n\tif (max == 0)\n\t\tminMax[1] = 0;\n\n\treturn minMax;\n}\n\nconst rangePad = 0.1;\n\nconst autoRangePart = {\n\tmode: 3,\n\tpad: rangePad,\n};\n\nconst _eqRangePart = {\n\tpad:  0,\n\tsoft: null,\n\tmode: 0,\n};\n\nconst _eqRange = {\n\tmin: _eqRangePart,\n\tmax: _eqRangePart,\n};\n\n// this ensures that non-temporal/numeric y-axes get multiple-snapped padding added above/below\n// TODO: also account for incrs when snapping to ensure top of axis gets a tick & value\nfunction rangeNum(_min, _max, mult, extra) {\n\tif (isObj(mult))\n\t\treturn _rangeNum(_min, _max, mult);\n\n\t_eqRangePart.pad  = mult;\n\t_eqRangePart.soft = extra ? 0 : null;\n\t_eqRangePart.mode = extra ? 3 : 0;\n\n\treturn _rangeNum(_min, _max, _eqRange);\n}\n\n// nullish coalesce\nfunction ifNull(lh, rh) {\n\treturn lh == null ? rh : lh;\n}\n\n// checks if given index range in an array contains a non-null value\n// aka a range-bounded Array.some()\nfunction hasData(data, idx0, idx1) {\n\tidx0 = ifNull(idx0, 0);\n\tidx1 = ifNull(idx1, data.length - 1);\n\n\twhile (idx0 <= idx1) {\n\t\tif (data[idx0] != null)\n\t\t\treturn true;\n\t\tidx0++;\n\t}\n\n\treturn false;\n}\n\nfunction _rangeNum(_min, _max, cfg) {\n\tlet cmin = cfg.min;\n\tlet cmax = cfg.max;\n\n\tlet padMin = ifNull(cmin.pad, 0);\n\tlet padMax = ifNull(cmax.pad, 0);\n\n\tlet hardMin = ifNull(cmin.hard, -inf);\n\tlet hardMax = ifNull(cmax.hard,  inf);\n\n\tlet softMin = ifNull(cmin.soft,  inf);\n\tlet softMax = ifNull(cmax.soft, -inf);\n\n\tlet softMinMode = ifNull(cmin.mode, 0);\n\tlet softMaxMode = ifNull(cmax.mode, 0);\n\n\tlet delta        = _max - _min;\n\n\t// this handles situations like 89.7, 89.69999999999999\n\t// by assuming 0.001x deltas are precision errors\n//\tif (delta > 0 && delta < abs(_max) / 1e3)\n//\t\tdelta = 0;\n\n\t// treat data as flat if delta is less than 1 billionth\n\tif (delta < 1e-9) {\n\t\tdelta = 0;\n\n\t\t// if soft mode is 2 and all vals are flat at 0, avoid the 0.1 * 1e3 fallback\n\t\t// this prevents 0,0,0 from ranging to -100,100 when softMin/softMax are -1,1\n\t\tif (_min == 0 || _max == 0) {\n\t\t\tdelta = 1e-9;\n\n\t\t\tif (softMinMode == 2 && softMin != inf)\n\t\t\t\tpadMin = 0;\n\n\t\t\tif (softMaxMode == 2 && softMax != -inf)\n\t\t\t\tpadMax = 0;\n\t\t}\n\t}\n\n\tlet nonZeroDelta = delta || abs(_max) || 1e3;\n\tlet mag          = log10(nonZeroDelta);\n\tlet base         = pow(10, floor(mag));\n\n\tlet _padMin  = nonZeroDelta * (delta == 0 ? (_min == 0 ? .1 : 1) : padMin);\n\tlet _newMin  = roundDec(incrRoundDn(_min - _padMin, base/10), 9);\n\tlet _softMin = _min >= softMin && (softMinMode == 1 || softMinMode == 3 && _newMin <= softMin || softMinMode == 2 && _newMin >= softMin) ? softMin : inf;\n\tlet minLim   = max(hardMin, _newMin < _softMin && _min >= _softMin ? _softMin : min(_softMin, _newMin));\n\n\tlet _padMax  = nonZeroDelta * (delta == 0 ? (_max == 0 ? .1 : 1) : padMax);\n\tlet _newMax  = roundDec(incrRoundUp(_max + _padMax, base/10), 9);\n\tlet _softMax = _max <= softMax && (softMaxMode == 1 || softMaxMode == 3 && _newMax >= softMax || softMaxMode == 2 && _newMax <= softMax) ? softMax : -inf;\n\tlet maxLim   = min(hardMax, _newMax > _softMax && _max <= _softMax ? _softMax : max(_softMax, _newMax));\n\n\tif (minLim == maxLim && minLim == 0)\n\t\tmaxLim = 100;\n\n\treturn [minLim, maxLim];\n}\n\n// alternative: https://stackoverflow.com/a/2254896\nconst numFormatter = new Intl.NumberFormat(domEnv ? nav.language : 'en-US');\nconst fmtNum = val => numFormatter.format(val);\n\nconst M = Math;\n\nconst PI = M.PI;\nconst abs = M.abs;\nconst floor = M.floor;\nconst round = M.round;\nconst ceil = M.ceil;\nconst min = M.min;\nconst max = M.max;\nconst pow = M.pow;\nconst sign = M.sign;\nconst log10 = M.log10;\nconst log2 = M.log2;\n// TODO: seems like this needs to match asinh impl if the passed v is tweaked?\nconst sinh =  (v, linthresh = 1) => M.sinh(v) * linthresh;\nconst asinh = (v, linthresh = 1) => M.asinh(v / linthresh);\n\nconst inf = Infinity;\n\nfunction numIntDigits(x) {\n\treturn (log10((x ^ (x >> 31)) - (x >> 31)) | 0) + 1;\n}\n\nfunction incrRound(num, incr) {\n\treturn round(num/incr)*incr;\n}\n\nfunction clamp(num, _min, _max) {\n\treturn min(max(num, _min), _max);\n}\n\nfunction fnOrSelf(v) {\n\treturn typeof v == \"function\" ? v : () => v;\n}\n\nconst retArg0 = _0 => _0;\n\nconst retArg1 = (_0, _1) => _1;\n\nconst retNull = _ => null;\n\nconst retTrue = _ => true;\n\nconst retEq = (a, b) => a == b;\n\nfunction incrRoundUp(num, incr) {\n\treturn ceil(num/incr)*incr;\n}\n\nfunction incrRoundDn(num, incr) {\n\treturn floor(num/incr)*incr;\n}\n\nfunction roundDec(val, dec) {\n\treturn round(val * (dec = 10**dec)) / dec;\n}\n\nconst fixedDec = new Map();\n\nfunction guessDec(num) {\n\treturn ((\"\"+num).split(\".\")[1] || \"\").length;\n}\n\nfunction genIncrs(base, minExp, maxExp, mults) {\n\tlet incrs = [];\n\n\tlet multDec = mults.map(guessDec);\n\n\tfor (let exp = minExp; exp < maxExp; exp++) {\n\t\tlet expa = abs(exp);\n\t\tlet mag = roundDec(pow(base, exp), expa);\n\n\t\tfor (let i = 0; i < mults.length; i++) {\n\t\t\tlet _incr = mults[i] * mag;\n\t\t\tlet dec = (_incr >= 0 && exp >= 0 ? 0 : expa) + (exp >= multDec[i] ? 0 : multDec[i]);\n\t\t\tlet incr = roundDec(_incr, dec);\n\t\t\tincrs.push(incr);\n\t\t\tfixedDec.set(incr, dec);\n\t\t}\n\t}\n\n\treturn incrs;\n}\n\n//export const assign = Object.assign;\n\nconst EMPTY_OBJ = {};\nconst EMPTY_ARR = [];\n\nconst nullNullTuple = [null, null];\n\nconst isArr = Array.isArray;\n\nfunction isStr(v) {\n\treturn typeof v == 'string';\n}\n\nfunction isObj(v) {\n\tlet is = false;\n\n\tif (v != null) {\n\t\tlet c = v.constructor;\n\t\tis = c == null || c == Object;\n\t}\n\n\treturn is;\n}\n\nfunction fastIsObj(v) {\n\treturn v != null && typeof v == 'object';\n}\n\nconst TypedArray = Object.getPrototypeOf(Uint8Array);\n\nfunction copy(o, _isObj = isObj) {\n\tlet out;\n\n\tif (isArr(o)) {\n\t\tlet val = o.find(v => v != null);\n\n\t\tif (isArr(val) || _isObj(val)) {\n\t\t\tout = Array(o.length);\n\t\t\tfor (let i = 0; i < o.length; i++)\n\t\t\t\tout[i] = copy(o[i], _isObj);\n\t\t}\n\t\telse\n\t\t\tout = o.slice();\n\t}\n\telse if (o instanceof TypedArray) // also (ArrayBuffer.isView(o) && !(o instanceof DataView))\n\t\tout = o.slice();\n\telse if (_isObj(o)) {\n\t\tout = {};\n\t\tfor (let k in o)\n\t\t\tout[k] = copy(o[k], _isObj);\n\t}\n\telse\n\t\tout = o;\n\n\treturn out;\n}\n\nfunction assign(targ) {\n\tlet args = arguments;\n\n\tfor (let i = 1; i < args.length; i++) {\n\t\tlet src = args[i];\n\n\t\tfor (let key in src) {\n\t\t\tif (isObj(targ[key]))\n\t\t\t\tassign(targ[key], copy(src[key]));\n\t\t\telse\n\t\t\t\ttarg[key] = copy(src[key]);\n\t\t}\n\t}\n\n\treturn targ;\n}\n\n// nullModes\nconst NULL_REMOVE = 0;  // nulls are converted to undefined (e.g. for spanGaps: true)\nconst NULL_RETAIN = 1;  // nulls are retained, with alignment artifacts set to undefined (default)\nconst NULL_EXPAND = 2;  // nulls are expanded to include any adjacent alignment artifacts\n\n// sets undefined values to nulls when adjacent to existing nulls (minesweeper)\nfunction nullExpand(yVals, nullIdxs, alignedLen) {\n\tfor (let i = 0, xi, lastNullIdx = -1; i < nullIdxs.length; i++) {\n\t\tlet nullIdx = nullIdxs[i];\n\n\t\tif (nullIdx > lastNullIdx) {\n\t\t\txi = nullIdx - 1;\n\t\t\twhile (xi >= 0 && yVals[xi] == null)\n\t\t\t\tyVals[xi--] = null;\n\n\t\t\txi = nullIdx + 1;\n\t\t\twhile (xi < alignedLen && yVals[xi] == null)\n\t\t\t\tyVals[lastNullIdx = xi++] = null;\n\t\t}\n\t}\n}\n\n// nullModes is a tables-matched array indicating how to treat nulls in each series\n// output is sorted ASC on the joined field (table[0]) and duplicate join values are collapsed\nfunction join(tables, nullModes) {\n\tlet xVals = new Set();\n\n\tfor (let ti = 0; ti < tables.length; ti++) {\n\t\tlet t = tables[ti];\n\t\tlet xs = t[0];\n\t\tlet len = xs.length;\n\n\t\tfor (let i = 0; i < len; i++)\n\t\t\txVals.add(xs[i]);\n\t}\n\n\tlet data = [Array.from(xVals).sort((a, b) => a - b)];\n\n\tlet alignedLen = data[0].length;\n\n\tlet xIdxs = new Map();\n\n\tfor (let i = 0; i < alignedLen; i++)\n\t\txIdxs.set(data[0][i], i);\n\n\tfor (let ti = 0; ti < tables.length; ti++) {\n\t\tlet t = tables[ti];\n\t\tlet xs = t[0];\n\n\t\tfor (let si = 1; si < t.length; si++) {\n\t\t\tlet ys = t[si];\n\n\t\t\tlet yVals = Array(alignedLen).fill(undefined);\n\n\t\t\tlet nullMode = nullModes ? nullModes[ti][si] : NULL_RETAIN;\n\n\t\t\tlet nullIdxs = [];\n\n\t\t\tfor (let i = 0; i < ys.length; i++) {\n\t\t\t\tlet yVal = ys[i];\n\t\t\t\tlet alignedIdx = xIdxs.get(xs[i]);\n\n\t\t\t\tif (yVal === null) {\n\t\t\t\t\tif (nullMode != NULL_REMOVE) {\n\t\t\t\t\t\tyVals[alignedIdx] = yVal;\n\n\t\t\t\t\t\tif (nullMode == NULL_EXPAND)\n\t\t\t\t\t\t\tnullIdxs.push(alignedIdx);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t\tyVals[alignedIdx] = yVal;\n\t\t\t}\n\n\t\t\tnullExpand(yVals, nullIdxs, alignedLen);\n\n\t\t\tdata.push(yVals);\n\t\t}\n\t}\n\n\treturn data;\n}\n\nconst microTask = typeof queueMicrotask == \"undefined\" ? fn => Promise.resolve().then(fn) : queueMicrotask;\n\nconst months = [\n\t\"January\",\n\t\"February\",\n\t\"March\",\n\t\"April\",\n\t\"May\",\n\t\"June\",\n\t\"July\",\n\t\"August\",\n\t\"September\",\n\t\"October\",\n\t\"November\",\n\t\"December\",\n];\n\nconst days = [\n\t\"Sunday\",\n\t\"Monday\",\n\t\"Tuesday\",\n\t\"Wednesday\",\n\t\"Thursday\",\n\t\"Friday\",\n\t\"Saturday\",\n];\n\nfunction slice3(str) {\n\treturn str.slice(0, 3);\n}\n\nconst days3 = days.map(slice3);\n\nconst months3 = months.map(slice3);\n\nconst engNames = {\n\tMMMM: months,\n\tMMM:  months3,\n\tWWWW: days,\n\tWWW:  days3,\n};\n\nfunction zeroPad2(int) {\n\treturn (int < 10 ? '0' : '') + int;\n}\n\nfunction zeroPad3(int) {\n\treturn (int < 10 ? '00' : int < 100 ? '0' : '') + int;\n}\n\n/*\nfunction suffix(int) {\n\tlet mod10 = int % 10;\n\n\treturn int + (\n\t\tmod10 == 1 && int != 11 ? \"st\" :\n\t\tmod10 == 2 && int != 12 ? \"nd\" :\n\t\tmod10 == 3 && int != 13 ? \"rd\" : \"th\"\n\t);\n}\n*/\n\nconst subs = {\n\t// 2019\n\tYYYY:\td => d.getFullYear(),\n\t// 19\n\tYY:\t\td => (d.getFullYear()+'').slice(2),\n\t// July\n\tMMMM:\t(d, names) => names.MMMM[d.getMonth()],\n\t// Jul\n\tMMM:\t(d, names) => names.MMM[d.getMonth()],\n\t// 07\n\tMM:\t\td => zeroPad2(d.getMonth()+1),\n\t// 7\n\tM:\t\td => d.getMonth()+1,\n\t// 09\n\tDD:\t\td => zeroPad2(d.getDate()),\n\t// 9\n\tD:\t\td => d.getDate(),\n\t// Monday\n\tWWWW:\t(d, names) => names.WWWW[d.getDay()],\n\t// Mon\n\tWWW:\t(d, names) => names.WWW[d.getDay()],\n\t// 03\n\tHH:\t\td => zeroPad2(d.getHours()),\n\t// 3\n\tH:\t\td => d.getHours(),\n\t// 9 (12hr, unpadded)\n\th:\t\td => {let h = d.getHours(); return h == 0 ? 12 : h > 12 ? h - 12 : h;},\n\t// AM\n\tAA:\t\td => d.getHours() >= 12 ? 'PM' : 'AM',\n\t// am\n\taa:\t\td => d.getHours() >= 12 ? 'pm' : 'am',\n\t// a\n\ta:\t\td => d.getHours() >= 12 ? 'p' : 'a',\n\t// 09\n\tmm:\t\td => zeroPad2(d.getMinutes()),\n\t// 9\n\tm:\t\td => d.getMinutes(),\n\t// 09\n\tss:\t\td => zeroPad2(d.getSeconds()),\n\t// 9\n\ts:\t\td => d.getSeconds(),\n\t// 374\n\tfff:\td => zeroPad3(d.getMilliseconds()),\n};\n\nfunction fmtDate(tpl, names) {\n\tnames = names || engNames;\n\tlet parts = [];\n\n\tlet R = /\\{([a-z]+)\\}|[^{]+/gi, m;\n\n\twhile (m = R.exec(tpl))\n\t\tparts.push(m[0][0] == '{' ? subs[m[1]] : m[0]);\n\n\treturn d => {\n\t\tlet out = '';\n\n\t\tfor (let i = 0; i < parts.length; i++)\n\t\t\tout += typeof parts[i] == \"string\" ? parts[i] : parts[i](d, names);\n\n\t\treturn out;\n\t}\n}\n\nconst localTz = new Intl.DateTimeFormat().resolvedOptions().timeZone;\n\n// https://stackoverflow.com/questions/15141762/how-to-initialize-a-javascript-date-to-a-particular-time-zone/53652131#53652131\nfunction tzDate(date, tz) {\n\tlet date2;\n\n\t// perf optimization\n\tif (tz == 'UTC' || tz == 'Etc/UTC')\n\t\tdate2 = new Date(+date + date.getTimezoneOffset() * 6e4);\n\telse if (tz == localTz)\n\t\tdate2 = date;\n\telse {\n\t\tdate2 = new Date(date.toLocaleString('en-US', {timeZone: tz}));\n\t\tdate2.setMilliseconds(date.getMilliseconds());\n\t}\n\n\treturn date2;\n}\n\n//export const series = [];\n\n// default formatters:\n\nconst onlyWhole = v => v % 1 == 0;\n\nconst allMults = [1,2,2.5,5];\n\n// ...0.01, 0.02, 0.025, 0.05, 0.1, 0.2, 0.25, 0.5\nconst decIncrs = genIncrs(10, -16, 0, allMults);\n\n// 1, 2, 2.5, 5, 10, 20, 25, 50...\nconst oneIncrs = genIncrs(10, 0, 16, allMults);\n\n// 1, 2,      5, 10, 20, 25, 50...\nconst wholeIncrs = oneIncrs.filter(onlyWhole);\n\nconst numIncrs = decIncrs.concat(oneIncrs);\n\nconst NL = \"\\n\";\n\nconst yyyy    = \"{YYYY}\";\nconst NLyyyy  = NL + yyyy;\nconst md      = \"{M}/{D}\";\nconst NLmd    = NL + md;\nconst NLmdyy  = NLmd + \"/{YY}\";\n\nconst aa      = \"{aa}\";\nconst hmm     = \"{h}:{mm}\";\nconst hmmaa   = hmm + aa;\nconst NLhmmaa = NL + hmmaa;\nconst ss      = \":{ss}\";\n\nconst _ = null;\n\nfunction genTimeStuffs(ms) {\n\tlet\ts  = ms * 1e3,\n\t\tm  = s  * 60,\n\t\th  = m  * 60,\n\t\td  = h  * 24,\n\t\tmo = d  * 30,\n\t\ty  = d  * 365;\n\n\t// min of 1e-3 prevents setting a temporal x ticks too small since Date objects cannot advance ticks smaller than 1ms\n\tlet subSecIncrs = ms == 1 ? genIncrs(10, 0, 3, allMults).filter(onlyWhole) : genIncrs(10, -3, 0, allMults);\n\n\tlet timeIncrs = subSecIncrs.concat([\n\t\t// minute divisors (# of secs)\n\t\ts,\n\t\ts * 5,\n\t\ts * 10,\n\t\ts * 15,\n\t\ts * 30,\n\t\t// hour divisors (# of mins)\n\t\tm,\n\t\tm * 5,\n\t\tm * 10,\n\t\tm * 15,\n\t\tm * 30,\n\t\t// day divisors (# of hrs)\n\t\th,\n\t\th * 2,\n\t\th * 3,\n\t\th * 4,\n\t\th * 6,\n\t\th * 8,\n\t\th * 12,\n\t\t// month divisors TODO: need more?\n\t\td,\n\t\td * 2,\n\t\td * 3,\n\t\td * 4,\n\t\td * 5,\n\t\td * 6,\n\t\td * 7,\n\t\td * 8,\n\t\td * 9,\n\t\td * 10,\n\t\td * 15,\n\t\t// year divisors (# months, approx)\n\t\tmo,\n\t\tmo * 2,\n\t\tmo * 3,\n\t\tmo * 4,\n\t\tmo * 6,\n\t\t// century divisors\n\t\ty,\n\t\ty * 2,\n\t\ty * 5,\n\t\ty * 10,\n\t\ty * 25,\n\t\ty * 50,\n\t\ty * 100,\n\t]);\n\n\t// [0]:   minimum num secs in the tick incr\n\t// [1]:   default tick format\n\t// [2-7]: rollover tick formats\n\t// [8]:   mode: 0: replace [1] -> [2-7], 1: concat [1] + [2-7]\n\tconst _timeAxisStamps = [\n\t//   tick incr    default          year                    month   day                   hour    min       sec   mode\n\t\t[y,           yyyy,            _,                      _,      _,                    _,      _,        _,       1],\n\t\t[d * 28,      \"{MMM}\",         NLyyyy,                 _,      _,                    _,      _,        _,       1],\n\t\t[d,           md,              NLyyyy,                 _,      _,                    _,      _,        _,       1],\n\t\t[h,           \"{h}\" + aa,      NLmdyy,                 _,      NLmd,                 _,      _,        _,       1],\n\t\t[m,           hmmaa,           NLmdyy,                 _,      NLmd,                 _,      _,        _,       1],\n\t\t[s,           ss,              NLmdyy + \" \" + hmmaa,   _,      NLmd + \" \" + hmmaa,   _,      NLhmmaa,  _,       1],\n\t\t[ms,          ss + \".{fff}\",   NLmdyy + \" \" + hmmaa,   _,      NLmd + \" \" + hmmaa,   _,      NLhmmaa,  _,       1],\n\t];\n\n\t// the ensures that axis ticks, values & grid are aligned to logical temporal breakpoints and not an arbitrary timestamp\n\t// https://www.timeanddate.com/time/dst/\n\t// https://www.timeanddate.com/time/dst/2019.html\n\t// https://www.epochconverter.com/timezones\n\tfunction timeAxisSplits(tzDate) {\n\t\treturn (self, axisIdx, scaleMin, scaleMax, foundIncr, foundSpace) => {\n\t\t\tlet splits = [];\n\t\t\tlet isYr = foundIncr >= y;\n\t\t\tlet isMo = foundIncr >= mo && foundIncr < y;\n\n\t\t\t// get the timezone-adjusted date\n\t\t\tlet minDate = tzDate(scaleMin);\n\t\t\tlet minDateTs = roundDec(minDate * ms, 3);\n\n\t\t\t// get ts of 12am (this lands us at or before the original scaleMin)\n\t\t\tlet minMin = mkDate(minDate.getFullYear(), isYr ? 0 : minDate.getMonth(), isMo || isYr ? 1 : minDate.getDate());\n\t\t\tlet minMinTs = roundDec(minMin * ms, 3);\n\n\t\t\tif (isMo || isYr) {\n\t\t\t\tlet moIncr = isMo ? foundIncr / mo : 0;\n\t\t\t\tlet yrIncr = isYr ? foundIncr / y  : 0;\n\t\t\t//\tlet tzOffset = scaleMin - minDateTs;\t\t// needed?\n\t\t\t\tlet split = minDateTs == minMinTs ? minDateTs : roundDec(mkDate(minMin.getFullYear() + yrIncr, minMin.getMonth() + moIncr, 1) * ms, 3);\n\t\t\t\tlet splitDate = new Date(round(split / ms));\n\t\t\t\tlet baseYear = splitDate.getFullYear();\n\t\t\t\tlet baseMonth = splitDate.getMonth();\n\n\t\t\t\tfor (let i = 0; split <= scaleMax; i++) {\n\t\t\t\t\tlet next = mkDate(baseYear + yrIncr * i, baseMonth + moIncr * i, 1);\n\t\t\t\t\tlet offs = next - tzDate(roundDec(next * ms, 3));\n\n\t\t\t\t\tsplit = roundDec((+next + offs) * ms, 3);\n\n\t\t\t\t\tif (split <= scaleMax)\n\t\t\t\t\t\tsplits.push(split);\n\t\t\t\t}\n\t\t\t}\n\t\t\telse {\n\t\t\t\tlet incr0 = foundIncr >= d ? d : foundIncr;\n\t\t\t\tlet tzOffset = floor(scaleMin) - floor(minDateTs);\n\t\t\t\tlet split = minMinTs + tzOffset + incrRoundUp(minDateTs - minMinTs, incr0);\n\t\t\t\tsplits.push(split);\n\n\t\t\t\tlet date0 = tzDate(split);\n\n\t\t\t\tlet prevHour = date0.getHours() + (date0.getMinutes() / m) + (date0.getSeconds() / h);\n\t\t\t\tlet incrHours = foundIncr / h;\n\n\t\t\t\tlet minSpace = self.axes[axisIdx]._space;\n\t\t\t\tlet pctSpace = foundSpace / minSpace;\n\n\t\t\t\twhile (1) {\n\t\t\t\t\tsplit = roundDec(split + foundIncr, ms == 1 ? 0 : 3);\n\n\t\t\t\t\tif (split > scaleMax)\n\t\t\t\t\t\tbreak;\n\n\t\t\t\t\tif (incrHours > 1) {\n\t\t\t\t\t\tlet expectedHour = floor(roundDec(prevHour + incrHours, 6)) % 24;\n\t\t\t\t\t\tlet splitDate = tzDate(split);\n\t\t\t\t\t\tlet actualHour = splitDate.getHours();\n\n\t\t\t\t\t\tlet dstShift = actualHour - expectedHour;\n\n\t\t\t\t\t\tif (dstShift > 1)\n\t\t\t\t\t\t\tdstShift = -1;\n\n\t\t\t\t\t\tsplit -= dstShift * h;\n\n\t\t\t\t\t\tprevHour = (prevHour + incrHours) % 24;\n\n\t\t\t\t\t\t// add a tick only if it's further than 70% of the min allowed label spacing\n\t\t\t\t\t\tlet prevSplit = splits[splits.length - 1];\n\t\t\t\t\t\tlet pctIncr = roundDec((split - prevSplit) / foundIncr, 3);\n\n\t\t\t\t\t\tif (pctIncr * pctSpace >= .7)\n\t\t\t\t\t\t\tsplits.push(split);\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t\tsplits.push(split);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\treturn splits;\n\t\t}\n\t}\n\n\treturn [\n\t\ttimeIncrs,\n\t\t_timeAxisStamps,\n\t\ttimeAxisSplits,\n\t];\n}\n\nconst [ timeIncrsMs, _timeAxisStampsMs, timeAxisSplitsMs ] = genTimeStuffs(1);\nconst [ timeIncrsS,  _timeAxisStampsS,  timeAxisSplitsS  ] = genTimeStuffs(1e-3);\n\n// base 2\ngenIncrs(2, -53, 53, [1]);\n\n/*\nconsole.log({\n\tdecIncrs,\n\toneIncrs,\n\twholeIncrs,\n\tnumIncrs,\n\ttimeIncrs,\n\tfixedDec,\n});\n*/\n\nfunction timeAxisStamps(stampCfg, fmtDate) {\n\treturn stampCfg.map(s => s.map((v, i) =>\n\t\ti == 0 || i == 8 || v == null ? v : fmtDate(i == 1 || s[8] == 0 ? v : s[1] + v)\n\t));\n}\n\n// TODO: will need to accept spaces[] and pull incr into the loop when grid will be non-uniform, eg for log scales.\n// currently we ignore this for months since they're *nearly* uniform and the added complexity is not worth it\nfunction timeAxisVals(tzDate, stamps) {\n\treturn (self, splits, axisIdx, foundSpace, foundIncr) => {\n\t\tlet s = stamps.find(s => foundIncr >= s[0]) || stamps[stamps.length - 1];\n\n\t\t// these track boundaries when a full label is needed again\n\t\tlet prevYear;\n\t\tlet prevMnth;\n\t\tlet prevDate;\n\t\tlet prevHour;\n\t\tlet prevMins;\n\t\tlet prevSecs;\n\n\t\treturn splits.map(split => {\n\t\t\tlet date = tzDate(split);\n\n\t\t\tlet newYear = date.getFullYear();\n\t\t\tlet newMnth = date.getMonth();\n\t\t\tlet newDate = date.getDate();\n\t\t\tlet newHour = date.getHours();\n\t\t\tlet newMins = date.getMinutes();\n\t\t\tlet newSecs = date.getSeconds();\n\n\t\t\tlet stamp = (\n\t\t\t\tnewYear != prevYear && s[2] ||\n\t\t\t\tnewMnth != prevMnth && s[3] ||\n\t\t\t\tnewDate != prevDate && s[4] ||\n\t\t\t\tnewHour != prevHour && s[5] ||\n\t\t\t\tnewMins != prevMins && s[6] ||\n\t\t\t\tnewSecs != prevSecs && s[7] ||\n\t\t\t\t                       s[1]\n\t\t\t);\n\n\t\t\tprevYear = newYear;\n\t\t\tprevMnth = newMnth;\n\t\t\tprevDate = newDate;\n\t\t\tprevHour = newHour;\n\t\t\tprevMins = newMins;\n\t\t\tprevSecs = newSecs;\n\n\t\t\treturn stamp(date);\n\t\t});\n\t}\n}\n\n// for when axis.values is defined as a static fmtDate template string\nfunction timeAxisVal(tzDate, dateTpl) {\n\tlet stamp = fmtDate(dateTpl);\n\treturn (self, splits, axisIdx, foundSpace, foundIncr) => splits.map(split => stamp(tzDate(split)));\n}\n\nfunction mkDate(y, m, d) {\n\treturn new Date(y, m, d);\n}\n\nfunction timeSeriesStamp(stampCfg, fmtDate) {\n\treturn fmtDate(stampCfg);\n}\nconst _timeSeriesStamp = '{YYYY}-{MM}-{DD} {h}:{mm}{aa}';\n\nfunction timeSeriesVal(tzDate, stamp) {\n\treturn (self, val) => stamp(tzDate(val));\n}\n\nfunction legendStroke(self, seriesIdx) {\n\tlet s = self.series[seriesIdx];\n\treturn s.width ? s.stroke(self, seriesIdx) : s.points.width ? s.points.stroke(self, seriesIdx) : null;\n}\n\nfunction legendFill(self, seriesIdx) {\n\treturn self.series[seriesIdx].fill(self, seriesIdx);\n}\n\nconst legendOpts = {\n\tshow: true,\n\tlive: true,\n\tisolate: false,\n\tmarkers: {\n\t\tshow: true,\n\t\twidth: 2,\n\t\tstroke: legendStroke,\n\t\tfill: legendFill,\n\t\tdash: \"solid\",\n\t},\n\tidx: null,\n\tidxs: null,\n\tvalues: [],\n};\n\nfunction cursorPointShow(self, si) {\n\tlet o = self.cursor.points;\n\n\tlet pt = placeDiv();\n\n\tlet size = o.size(self, si);\n\tsetStylePx(pt, WIDTH, size);\n\tsetStylePx(pt, HEIGHT, size);\n\n\tlet mar = size / -2;\n\tsetStylePx(pt, \"marginLeft\", mar);\n\tsetStylePx(pt, \"marginTop\", mar);\n\n\tlet width = o.width(self, si, size);\n\twidth && setStylePx(pt, \"borderWidth\", width);\n\n\treturn pt;\n}\n\nfunction cursorPointFill(self, si) {\n\tlet sp = self.series[si].points;\n\treturn sp._fill || sp._stroke;\n}\n\nfunction cursorPointStroke(self, si) {\n\tlet sp = self.series[si].points;\n\treturn sp._stroke || sp._fill;\n}\n\nfunction cursorPointSize(self, si) {\n\tlet sp = self.series[si].points;\n\treturn ptDia(sp.width, 1);\n}\n\nfunction dataIdx(self, seriesIdx, cursorIdx) {\n\treturn cursorIdx;\n}\n\nconst moveTuple = [0,0];\n\nfunction cursorMove(self, mouseLeft1, mouseTop1) {\n\tmoveTuple[0] = mouseLeft1;\n\tmoveTuple[1] = mouseTop1;\n\treturn moveTuple;\n}\n\nfunction filtBtn0(self, targ, handle) {\n\treturn e => {\n\t\te.button == 0 && handle(e);\n\t};\n}\n\nfunction passThru(self, targ, handle) {\n\treturn handle;\n}\n\nconst cursorOpts = {\n\tshow: true,\n\tx: true,\n\ty: true,\n\tlock: false,\n\tmove: cursorMove,\n\tpoints: {\n\t\tshow:   cursorPointShow,\n\t\tsize:   cursorPointSize,\n\t\twidth:  0,\n\t\tstroke: cursorPointStroke,\n\t\tfill:   cursorPointFill,\n\t},\n\n\tbind: {\n\t\tmousedown:   filtBtn0,\n\t\tmouseup:     filtBtn0,\n\t\tclick:       filtBtn0,\n\t\tdblclick:    filtBtn0,\n\n\t\tmousemove:   passThru,\n\t\tmouseleave:  passThru,\n\t\tmouseenter:  passThru,\n\t},\n\n\tdrag: {\n\t\tsetScale: true,\n\t\tx: true,\n\t\ty: false,\n\t\tdist: 0,\n\t\tuni: null,\n\t\t_x: false,\n\t\t_y: false,\n\t},\n\n\tfocus: {\n\t\tprox: -1,\n\t},\n\n\tleft: -10,\n\ttop: -10,\n\tidx: null,\n\tdataIdx,\n\tidxs: null,\n};\n\nconst axisLines = {\n\tshow: true,\n\tstroke: \"rgba(0,0,0,0.07)\",\n\twidth: 2,\n//\tdash: [],\n};\n\nconst grid = assign({}, axisLines, {\n\tfilter: retArg1,\n});\n\nconst ticks = assign({}, grid, {\n\tsize: 10,\n});\n\nconst border = assign({}, axisLines, {\n\tshow: false,\n});\n\nconst font      = '12px system-ui, -apple-system, \"Segoe UI\", Roboto, \"Helvetica Neue\", Arial, \"Noto Sans\", sans-serif, \"Apple Color Emoji\", \"Segoe UI Emoji\", \"Segoe UI Symbol\", \"Noto Color Emoji\"';\nconst labelFont = \"bold \" + font;\nconst lineMult = 1.5;\t\t// font-size multiplier\n\nconst xAxisOpts = {\n\tshow: true,\n\tscale: \"x\",\n\tstroke: hexBlack,\n\tspace: 50,\n\tgap: 5,\n\tsize: 50,\n\tlabelGap: 0,\n\tlabelSize: 30,\n\tlabelFont,\n\tside: 2,\n//\tclass: \"x-vals\",\n//\tincrs: timeIncrs,\n//\tvalues: timeVals,\n//\tfilter: retArg1,\n\tgrid,\n\tticks,\n\tborder,\n\tfont,\n\trotate: 0,\n};\n\nconst numSeriesLabel = \"Value\";\nconst timeSeriesLabel = \"Time\";\n\nconst xSeriesOpts = {\n\tshow: true,\n\tscale: \"x\",\n\tauto: false,\n\tsorted: 1,\n//\tlabel: \"Time\",\n//\tvalue: v => stamp(new Date(v * 1e3)),\n\n\t// internal caches\n\tmin: inf,\n\tmax: -inf,\n\tidxs: [],\n};\n\nfunction numAxisVals(self, splits, axisIdx, foundSpace, foundIncr) {\n\treturn splits.map(v => v == null ? \"\" : fmtNum(v));\n}\n\nfunction numAxisSplits(self, axisIdx, scaleMin, scaleMax, foundIncr, foundSpace, forceMin) {\n\tlet splits = [];\n\n\tlet numDec = fixedDec.get(foundIncr) || 0;\n\n\tscaleMin = forceMin ? scaleMin : roundDec(incrRoundUp(scaleMin, foundIncr), numDec);\n\n\tfor (let val = scaleMin; val <= scaleMax; val = roundDec(val + foundIncr, numDec))\n\t\tsplits.push(Object.is(val, -0) ? 0 : val);\t\t// coalesces -0\n\n\treturn splits;\n}\n\n// this doesnt work for sin, which needs to come off from 0 independently in pos and neg dirs\nfunction logAxisSplits(self, axisIdx, scaleMin, scaleMax, foundIncr, foundSpace, forceMin) {\n\tconst splits = [];\n\n\tconst logBase = self.scales[self.axes[axisIdx].scale].log;\n\n\tconst logFn = logBase == 10 ? log10 : log2;\n\n\tconst exp = floor(logFn(scaleMin));\n\n\tfoundIncr = pow(logBase, exp);\n\n\tif (exp < 0)\n\t\tfoundIncr = roundDec(foundIncr, -exp);\n\n\tlet split = scaleMin;\n\n\tdo {\n\t\tsplits.push(split);\n\t\tsplit = roundDec(split + foundIncr, fixedDec.get(foundIncr));\n\n\t\tif (split >= foundIncr * logBase)\n\t\t\tfoundIncr = split;\n\n\t} while (split <= scaleMax);\n\n\treturn splits;\n}\n\nfunction asinhAxisSplits(self, axisIdx, scaleMin, scaleMax, foundIncr, foundSpace, forceMin) {\n\tlet sc = self.scales[self.axes[axisIdx].scale];\n\n\tlet linthresh = sc.asinh;\n\n\tlet posSplits = scaleMax > linthresh ? logAxisSplits(self, axisIdx, max(linthresh, scaleMin), scaleMax, foundIncr) : [linthresh];\n\tlet zero = scaleMax >= 0 && scaleMin <= 0 ? [0] : [];\n\tlet negSplits = scaleMin < -linthresh ? logAxisSplits(self, axisIdx, max(linthresh, -scaleMax), -scaleMin, foundIncr): [linthresh];\n\n\treturn negSplits.reverse().map(v => -v).concat(zero, posSplits);\n}\n\nconst RE_ALL   = /./;\nconst RE_12357 = /[12357]/;\nconst RE_125   = /[125]/;\nconst RE_1     = /1/;\n\nfunction logAxisValsFilt(self, splits, axisIdx, foundSpace, foundIncr) {\n\tlet axis = self.axes[axisIdx];\n\tlet scaleKey = axis.scale;\n\tlet sc = self.scales[scaleKey];\n\n\tif (sc.distr == 3 && sc.log == 2)\n\t\treturn splits;\n\n\tlet valToPos = self.valToPos;\n\n\tlet minSpace = axis._space;\n\n\tlet _10 = valToPos(10, scaleKey);\n\n\tlet re = (\n\t\tvalToPos(9, scaleKey) - _10 >= minSpace ? RE_ALL :\n\t\tvalToPos(7, scaleKey) - _10 >= minSpace ? RE_12357 :\n\t\tvalToPos(5, scaleKey) - _10 >= minSpace ? RE_125 :\n\t\tRE_1\n\t);\n\n\treturn splits.map(v => ((sc.distr == 4 && v == 0) || re.test(v)) ? v : null);\n}\n\nfunction numSeriesVal(self, val) {\n\treturn val == null ? \"\" : fmtNum(val);\n}\n\nconst yAxisOpts = {\n\tshow: true,\n\tscale: \"y\",\n\tstroke: hexBlack,\n\tspace: 30,\n\tgap: 5,\n\tsize: 50,\n\tlabelGap: 0,\n\tlabelSize: 30,\n\tlabelFont,\n\tside: 3,\n//\tclass: \"y-vals\",\n//\tincrs: numIncrs,\n//\tvalues: (vals, space) => vals,\n//\tfilter: retArg1,\n\tgrid,\n\tticks,\n\tborder,\n\tfont,\n\trotate: 0,\n};\n\n// takes stroke width\nfunction ptDia(width, mult) {\n\tlet dia = 3 + (width || 1) * 2;\n\treturn roundDec(dia * mult, 3);\n}\n\nfunction seriesPointsShow(self, si) {\n\tlet { scale, idxs } = self.series[0];\n\tlet xData = self._data[0];\n\tlet p0 = self.valToPos(xData[idxs[0]], scale, true);\n\tlet p1 = self.valToPos(xData[idxs[1]], scale, true);\n\tlet dim = abs(p1 - p0);\n\n\tlet s = self.series[si];\n//\tconst dia = ptDia(s.width, pxRatio);\n\tlet maxPts = dim / (s.points.space * pxRatio);\n\treturn idxs[1] - idxs[0] <= maxPts;\n}\n\nconst facet = {\n\tscale: null,\n\tauto: true,\n\tsorted: 0,\n\n\t// internal caches\n\tmin: inf,\n\tmax: -inf,\n};\n\nconst xySeriesOpts = {\n\tshow: true,\n\tauto: true,\n\tsorted: 0,\n\talpha: 1,\n\tfacets: [\n\t\tassign({}, facet, {scale: 'x'}),\n\t\tassign({}, facet, {scale: 'y'}),\n\t],\n};\n\nconst ySeriesOpts = {\n\tscale: \"y\",\n\tauto: true,\n\tsorted: 0,\n\tshow: true,\n\tspanGaps: false,\n\tgaps: (self, seriesIdx, idx0, idx1, nullGaps) => nullGaps,\n\talpha: 1,\n\tpoints: {\n\t\tshow: seriesPointsShow,\n\t\tfilter: null,\n\t//  paths:\n\t//\tstroke: \"#000\",\n\t//\tfill: \"#fff\",\n\t//\twidth: 1,\n\t//\tsize: 10,\n\t},\n//\tlabel: \"Value\",\n//\tvalue: v => v,\n\tvalues: null,\n\n\t// internal caches\n\tmin: inf,\n\tmax: -inf,\n\tidxs: [],\n\n\tpath: null,\n\tclip: null,\n};\n\nfunction clampScale(self, val, scaleMin, scaleMax, scaleKey) {\n/*\n\tif (val < 0) {\n\t\tlet cssHgt = self.bbox.height / pxRatio;\n\t\tlet absPos = self.valToPos(abs(val), scaleKey);\n\t\tlet fromBtm = cssHgt - absPos;\n\t\treturn self.posToVal(cssHgt + fromBtm, scaleKey);\n\t}\n*/\n\treturn scaleMin / 10;\n}\n\nconst xScaleOpts = {\n\ttime: FEAT_TIME,\n\tauto: true,\n\tdistr: 1,\n\tlog: 10,\n\tasinh: 1,\n\tmin: null,\n\tmax: null,\n\tdir: 1,\n\tori: 0,\n};\n\nconst yScaleOpts = assign({}, xScaleOpts, {\n\ttime: false,\n\tori: 1,\n});\n\nconst syncs = {};\n\nfunction _sync(key, opts) {\n\tlet s = syncs[key];\n\n\tif (!s) {\n\t\ts = {\n\t\t\tkey,\n\t\t\tplots: [],\n\t\t\tsub(plot) {\n\t\t\t\ts.plots.push(plot);\n\t\t\t},\n\t\t\tunsub(plot) {\n\t\t\t\ts.plots = s.plots.filter(c => c != plot);\n\t\t\t},\n\t\t\tpub(type, self, x, y, w, h, i) {\n\t\t\t\tfor (let j = 0; j < s.plots.length; j++)\n\t\t\t\t\ts.plots[j] != self && s.plots[j].pub(type, self, x, y, w, h, i);\n\t\t\t},\n\t\t};\n\n\t\tif (key != null)\n\t\t\tsyncs[key] = s;\n\t}\n\n\treturn s;\n}\n\nconst BAND_CLIP_FILL   = 1 << 0;\nconst BAND_CLIP_STROKE = 1 << 1;\n\nfunction orient(u, seriesIdx, cb) {\n\tconst series = u.series[seriesIdx];\n\tconst scales = u.scales;\n\tconst bbox   = u.bbox;\n\tconst scaleX = u.mode == 2 ? scales[series.facets[0].scale] : scales[u.series[0].scale];\n\n\tlet dx = u._data[0],\n\t\tdy = u._data[seriesIdx],\n\t\tsx = scaleX,\n\t\tsy = u.mode == 2 ? scales[series.facets[1].scale] : scales[series.scale],\n\t\tl = bbox.left,\n\t\tt = bbox.top,\n\t\tw = bbox.width,\n\t\th = bbox.height,\n\t\tH = u.valToPosH,\n\t\tV = u.valToPosV;\n\n\treturn (sx.ori == 0\n\t\t? cb(\n\t\t\tseries,\n\t\t\tdx,\n\t\t\tdy,\n\t\t\tsx,\n\t\t\tsy,\n\t\t\tH,\n\t\t\tV,\n\t\t\tl,\n\t\t\tt,\n\t\t\tw,\n\t\t\th,\n\t\t\tmoveToH,\n\t\t\tlineToH,\n\t\t\trectH,\n\t\t\tarcH,\n\t\t\tbezierCurveToH,\n\t\t)\n\t\t: cb(\n\t\t\tseries,\n\t\t\tdx,\n\t\t\tdy,\n\t\t\tsx,\n\t\t\tsy,\n\t\t\tV,\n\t\t\tH,\n\t\t\tt,\n\t\t\tl,\n\t\t\th,\n\t\t\tw,\n\t\t\tmoveToV,\n\t\t\tlineToV,\n\t\t\trectV,\n\t\t\tarcV,\n\t\t\tbezierCurveToV,\n\t\t)\n\t);\n}\n\nfunction bandFillClipDirs(self, seriesIdx) {\n\tlet fillDir = 0;\n\n\t// 2 bits, -1 | 1\n\tlet clipDirs = 0;\n\n\tlet bands = ifNull(self.bands, EMPTY_ARR);\n\n\tfor (let i = 0; i < bands.length; i++) {\n\t\tlet b = bands[i];\n\n\t\t// is a \"from\" band edge\n\t\tif (b.series[0] == seriesIdx)\n\t\t\tfillDir = b.dir;\n\t\t// is a \"to\" band edge\n\t\telse if (b.series[1] == seriesIdx) {\n\t\t\tif (b.dir == 1)\n\t\t\t\tclipDirs |= 1;\n\t\t\telse\n\t\t\t\tclipDirs |= 2;\n\t\t}\n\t}\n\n\treturn [\n\t\tfillDir,\n\t\t(\n\t\t\tclipDirs == 1 ? -1 : // neg only\n\t\t\tclipDirs == 2 ?  1 : // pos only\n\t\t\tclipDirs == 3 ?  2 : // both\n\t\t\t                 0   // neither\n\t\t)\n\t];\n}\n\nfunction seriesFillTo(self, seriesIdx, dataMin, dataMax, bandFillDir) {\n\tlet scale = self.scales[self.series[seriesIdx].scale];\n\n\treturn (\n\t\tbandFillDir == -1 ? scale.min :\n\t\tbandFillDir ==  1 ? scale.max :\n\t\tscale.distr ==  3 ? (\n\t\t\tscale.dir == 1 ? scale.min :\n\t\t\tscale.max\n\t\t) : 0\n\t);\n}\n\n// creates inverted band clip path (from stroke path -> yMax || yMin)\n// clipDir is always inverse of fillDir\n// default clip dir is upwards (1), since default band fill is downwards/fillBelowTo (-1) (highIdx -> lowIdx)\nfunction clipBandLine(self, seriesIdx, idx0, idx1, strokePath, clipDir) {\n\treturn orient(self, seriesIdx, (series, dataX, dataY, scaleX, scaleY, valToPosX, valToPosY, xOff, yOff, xDim, yDim) => {\n\t\tlet pxRound = series.pxRound;\n\n\t\tconst dir = scaleX.dir * (scaleX.ori == 0 ? 1 : -1);\n\t\tconst lineTo = scaleX.ori == 0 ? lineToH : lineToV;\n\n\t\tlet frIdx, toIdx;\n\n\t\tif (dir == 1) {\n\t\t\tfrIdx = idx0;\n\t\t\ttoIdx = idx1;\n\t\t}\n\t\telse {\n\t\t\tfrIdx = idx1;\n\t\t\ttoIdx = idx0;\n\t\t}\n\n\t\t// path start\n\t\tlet x0 = pxRound(valToPosX(dataX[frIdx], scaleX, xDim, xOff));\n\t\tlet y0 = pxRound(valToPosY(dataY[frIdx], scaleY, yDim, yOff));\n\t\t// path end x\n\t\tlet x1 = pxRound(valToPosX(dataX[toIdx], scaleX, xDim, xOff));\n\t\t// upper or lower y limit\n\t\tlet yLimit = pxRound(valToPosY(clipDir == 1 ? scaleY.max : scaleY.min, scaleY, yDim, yOff));\n\n\t\tlet clip = new Path2D(strokePath);\n\n\t\tlineTo(clip, x1, yLimit);\n\t\tlineTo(clip, x0, yLimit);\n\t\tlineTo(clip, x0, y0);\n\n\t\treturn clip;\n\t});\n}\n\nfunction clipGaps(gaps, ori, plotLft, plotTop, plotWid, plotHgt) {\n\tlet clip = null;\n\n\t// create clip path (invert gaps and non-gaps)\n\tif (gaps.length > 0) {\n\t\tclip = new Path2D();\n\n\t\tconst rect = ori == 0 ? rectH : rectV;\n\n\t\tlet prevGapEnd = plotLft;\n\n\t\tfor (let i = 0; i < gaps.length; i++) {\n\t\t\tlet g = gaps[i];\n\n\t\t\tif (g[1] > g[0]) {\n\t\t\t\tlet w = g[0] - prevGapEnd;\n\n\t\t\t\tw > 0 && rect(clip, prevGapEnd, plotTop, w, plotTop + plotHgt);\n\n\t\t\t\tprevGapEnd = g[1];\n\t\t\t}\n\t\t}\n\n\t\tlet w = plotLft + plotWid - prevGapEnd;\n\n\t\tw > 0 && rect(clip, prevGapEnd, plotTop, w, plotTop + plotHgt);\n\t}\n\n\treturn clip;\n}\n\nfunction addGap(gaps, fromX, toX) {\n\tlet prevGap = gaps[gaps.length - 1];\n\n\tif (prevGap && prevGap[0] == fromX)\t\t\t// TODO: gaps must be encoded at stroke widths?\n\t\tprevGap[1] = toX;\n\telse\n\t\tgaps.push([fromX, toX]);\n}\n\nfunction findGaps(xs, ys, idx0, idx1, dir, pixelForX, align) {\n\tlet gaps = [];\n\n\tfor (let i = dir == 1 ? idx0 : idx1; i >= idx0 && i <= idx1; i += dir) {\n\t\tlet yVal = ys[i];\n\n\t\tif (yVal === null) {\n\t\t\tlet fr = i, to = i;\n\n\t\t\tif (dir == 1) {\n\t\t\t\twhile (++i <= idx1 && ys[i] === null)\n\t\t\t\t\tto = i;\n\t\t\t}\n\t\t\telse {\n\t\t\t\twhile (--i >= idx0 && ys[i] === null)\n\t\t\t\t\tto = i;\n\t\t\t}\n\n\t\t\tlet frPx = pixelForX(xs[fr]);\n\t\t\tlet toPx = to == fr ? frPx : pixelForX(xs[to]);\n\n\t\t\t// if value adjacent to edge null is same pixel, then it's partially\n\t\t\t// filled and gap should start at next pixel\n\t\t\tlet frPx2 = align <= 0 ? pixelForX(xs[fr-dir]) : frPx;\n\t\t//\tif (frPx2 == frPx)\n\t\t//\t\tfrPx++;\n\t\t//\telse\n\t\t\t\tfrPx = frPx2;\n\n\t\t\tlet toPx2 = align >= 0 ? pixelForX(xs[to+dir]) : toPx;\n\t\t//\tif (toPx2 == toPx)\n\t\t//\t\ttoPx--;\n\t\t//\telse\n\t\t\t\ttoPx = toPx2;\n\n\t\t\tif (toPx >= frPx)\n\t\t\t\tgaps.push([frPx, toPx]); // addGap\n\t\t}\n\t}\n\n\treturn gaps;\n}\n\nfunction pxRoundGen(pxAlign) {\n\treturn pxAlign == 0 ? retArg0 : pxAlign == 1 ? round : v => incrRound(v, pxAlign);\n}\n\nfunction rect(ori) {\n\tlet moveTo = ori == 0 ?\n\t\tmoveToH :\n\t\tmoveToV;\n\n\tlet arcTo = ori == 0 ?\n\t\t(p, x1, y1, x2, y2, r) => { p.arcTo(x1, y1, x2, y2, r); } :\n\t\t(p, y1, x1, y2, x2, r) => { p.arcTo(x1, y1, x2, y2, r); };\n\n\tlet rect = ori == 0 ?\n\t\t(p, x, y, w, h) => { p.rect(x, y, w, h); } :\n\t\t(p, y, x, h, w) => { p.rect(x, y, w, h); };\n\n\treturn (p, x, y, w, h, r = 0) => {\n\t\tif (r == 0)\n\t\t\trect(p, x, y, w, h);\n\t\telse {\n\t\t\tr = min(r, w / 2, h / 2);\n\n\t\t\t// adapted from https://stackoverflow.com/questions/1255512/how-to-draw-a-rounded-rectangle-using-html-canvas/7838871#7838871\n\t\t\tmoveTo(p, x + r, y);\n\t\t\tarcTo(p, x + w, y, x + w, y + h, r);\n\t\t\tarcTo(p, x + w, y + h, x, y + h, r);\n\t\t\tarcTo(p, x, y + h, x, y, r);\n\t\t\tarcTo(p, x, y, x + w, y, r);\n\t\t\tp.closePath();\n\t\t}\n\t};\n}\n\n// orientation-inverting canvas functions\nconst moveToH = (p, x, y) => { p.moveTo(x, y); };\nconst moveToV = (p, y, x) => { p.moveTo(x, y); };\nconst lineToH = (p, x, y) => { p.lineTo(x, y); };\nconst lineToV = (p, y, x) => { p.lineTo(x, y); };\nconst rectH = rect(0);\nconst rectV = rect(1);\nconst arcH = (p, x, y, r, startAngle, endAngle) => { p.arc(x, y, r, startAngle, endAngle); };\nconst arcV = (p, y, x, r, startAngle, endAngle) => { p.arc(x, y, r, startAngle, endAngle); };\nconst bezierCurveToH = (p, bp1x, bp1y, bp2x, bp2y, p2x, p2y) => { p.bezierCurveTo(bp1x, bp1y, bp2x, bp2y, p2x, p2y); };\nconst bezierCurveToV = (p, bp1y, bp1x, bp2y, bp2x, p2y, p2x) => { p.bezierCurveTo(bp1x, bp1y, bp2x, bp2y, p2x, p2y); };\n\n// TODO: drawWrap(seriesIdx, drawPoints) (save, restore, translate, clip)\nfunction points(opts) {\n\treturn (u, seriesIdx, idx0, idx1, filtIdxs) => {\n\t//\tlog(\"drawPoints()\", arguments);\n\n\t\treturn orient(u, seriesIdx, (series, dataX, dataY, scaleX, scaleY, valToPosX, valToPosY, xOff, yOff, xDim, yDim) => {\n\t\t\tlet { pxRound, points } = series;\n\n\t\t\tlet moveTo, arc;\n\n\t\t\tif (scaleX.ori == 0) {\n\t\t\t\tmoveTo = moveToH;\n\t\t\t\tarc = arcH;\n\t\t\t}\n\t\t\telse {\n\t\t\t\tmoveTo = moveToV;\n\t\t\t\tarc = arcV;\n\t\t\t}\n\n\t\t\tconst width = roundDec(points.width * pxRatio, 3);\n\n\t\t\tlet rad = (points.size - points.width) / 2 * pxRatio;\n\t\t\tlet dia = roundDec(rad * 2, 3);\n\n\t\t\tlet fill = new Path2D();\n\t\t\tlet clip = new Path2D();\n\n\t\t\tlet { left: lft, top: top, width: wid, height: hgt } = u.bbox;\n\n\t\t\trectH(clip,\n\t\t\t\tlft - dia,\n\t\t\t\ttop - dia,\n\t\t\t\twid + dia * 2,\n\t\t\t\thgt + dia * 2,\n\t\t\t);\n\n\t\t\tconst drawPoint = pi => {\n\t\t\t\tif (dataY[pi] != null) {\n\t\t\t\t\tlet x = pxRound(valToPosX(dataX[pi], scaleX, xDim, xOff));\n\t\t\t\t\tlet y = pxRound(valToPosY(dataY[pi], scaleY, yDim, yOff));\n\n\t\t\t\t\tmoveTo(fill, x + rad, y);\n\t\t\t\t\tarc(fill, x, y, rad, 0, PI * 2);\n\t\t\t\t}\n\t\t\t};\n\n\t\t\tif (filtIdxs)\n\t\t\t\tfiltIdxs.forEach(drawPoint);\n\t\t\telse {\n\t\t\t\tfor (let pi = idx0; pi <= idx1; pi++)\n\t\t\t\t\tdrawPoint(pi);\n\t\t\t}\n\n\t\t\treturn {\n\t\t\t\tstroke: width > 0 ? fill : null,\n\t\t\t\tfill,\n\t\t\t\tclip,\n\t\t\t\tflags: BAND_CLIP_FILL | BAND_CLIP_STROKE,\n\t\t\t};\n\t\t});\n\t};\n}\n\nfunction _drawAcc(lineTo) {\n\treturn (stroke, accX, minY, maxY, inY, outY) => {\n\t\tif (minY != maxY) {\n\t\t\tif (inY != minY && outY != minY)\n\t\t\t\tlineTo(stroke, accX, minY);\n\t\t\tif (inY != maxY && outY != maxY)\n\t\t\t\tlineTo(stroke, accX, maxY);\n\n\t\t\tlineTo(stroke, accX, outY);\n\t\t}\n\t};\n}\n\nconst drawAccH = _drawAcc(lineToH);\nconst drawAccV = _drawAcc(lineToV);\n\nfunction linear(opts) {\n\tconst alignGaps = ifNull(opts?.alignGaps, 0);\n\n\treturn (u, seriesIdx, idx0, idx1) => {\n\t\treturn orient(u, seriesIdx, (series, dataX, dataY, scaleX, scaleY, valToPosX, valToPosY, xOff, yOff, xDim, yDim) => {\n\t\t\tlet pxRound = series.pxRound;\n\n\t\t\tlet pixelForX = val => pxRound(valToPosX(val, scaleX, xDim, xOff));\n\t\t\tlet pixelForY = val => pxRound(valToPosY(val, scaleY, yDim, yOff));\n\n\t\t\tlet lineTo, drawAcc;\n\n\t\t\tif (scaleX.ori == 0) {\n\t\t\t\tlineTo = lineToH;\n\t\t\t\tdrawAcc = drawAccH;\n\t\t\t}\n\t\t\telse {\n\t\t\t\tlineTo = lineToV;\n\t\t\t\tdrawAcc = drawAccV;\n\t\t\t}\n\n\t\t\tconst dir = scaleX.dir * (scaleX.ori == 0 ? 1 : -1);\n\n\t\t\tconst _paths = {stroke: new Path2D(), fill: null, clip: null, band: null, gaps: null, flags: BAND_CLIP_FILL};\n\t\t\tconst stroke = _paths.stroke;\n\n\t\t\tlet minY = inf,\n\t\t\t\tmaxY = -inf,\n\t\t\t\tinY, outY, drawnAtX;\n\n\t\t\tlet accX = pixelForX(dataX[dir == 1 ? idx0 : idx1]);\n\n\t\t\t// data edges\n\t\t\tlet lftIdx = nonNullIdx(dataY, idx0, idx1,  1 * dir);\n\t\t\tlet rgtIdx = nonNullIdx(dataY, idx0, idx1, -1 * dir);\n\t\t\tlet lftX   =  pixelForX(dataX[lftIdx]);\n\t\t\tlet rgtX   =  pixelForX(dataX[rgtIdx]);\n\n\t\t\tfor (let i = dir == 1 ? idx0 : idx1; i >= idx0 && i <= idx1; i += dir) {\n\t\t\t\tlet x = pixelForX(dataX[i]);\n\n\t\t\t\tif (x == accX) {\n\t\t\t\t\tif (dataY[i] != null) {\n\t\t\t\t\t\toutY = pixelForY(dataY[i]);\n\n\t\t\t\t\t\tif (minY == inf) {\n\t\t\t\t\t\t\tlineTo(stroke, x, outY);\n\t\t\t\t\t\t\tinY = outY;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tminY = min(outY, minY);\n\t\t\t\t\t\tmaxY = max(outY, maxY);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\telse {\n\t\t\t\t\tif (minY != inf) {\n\t\t\t\t\t\tdrawAcc(stroke, accX, minY, maxY, inY, outY);\n\t\t\t\t\t\tdrawnAtX = accX;\n\t\t\t\t\t}\n\n\t\t\t\t\tif (dataY[i] != null) {\n\t\t\t\t\t\toutY = pixelForY(dataY[i]);\n\t\t\t\t\t\tlineTo(stroke, x, outY);\n\t\t\t\t\t\tminY = maxY = inY = outY;\n\t\t\t\t\t}\n\t\t\t\t\telse {\n\t\t\t\t\t\tminY = inf;\n\t\t\t\t\t\tmaxY = -inf;\n\t\t\t\t\t}\n\n\t\t\t\t\taccX = x;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (minY != inf && minY != maxY && drawnAtX != accX)\n\t\t\t\tdrawAcc(stroke, accX, minY, maxY, inY, outY);\n\n\t\t\tlet [ bandFillDir, bandClipDir ] = bandFillClipDirs(u, seriesIdx);\n\n\t\t\tif (series.fill != null || bandFillDir != 0) {\n\t\t\t\tlet fill = _paths.fill = new Path2D(stroke);\n\n\t\t\t\tlet fillToVal = series.fillTo(u, seriesIdx, series.min, series.max, bandFillDir);\n\t\t\t\tlet fillToY = pixelForY(fillToVal);\n\n\t\t\t\tlineTo(fill, rgtX, fillToY);\n\t\t\t\tlineTo(fill, lftX, fillToY);\n\t\t\t}\n\n\t\t\tif (!series.spanGaps) {\n\t\t\t//\tconsole.time('gaps');\n\t\t\t\tlet gaps = [];\n\n\t\t\t\tgaps.push(...findGaps(dataX, dataY, idx0, idx1, dir, pixelForX, alignGaps));\n\n\t\t\t//\tconsole.timeEnd('gaps');\n\n\t\t\t//\tconsole.log('gaps', JSON.stringify(gaps));\n\n\t\t\t\t_paths.gaps = gaps = series.gaps(u, seriesIdx, idx0, idx1, gaps);\n\n\t\t\t\t_paths.clip = clipGaps(gaps, scaleX.ori, xOff, yOff, xDim, yDim);\n\t\t\t}\n\n\t\t\tif (bandClipDir != 0) {\n\t\t\t\t_paths.band = bandClipDir == 2 ? [\n\t\t\t\t\tclipBandLine(u, seriesIdx, idx0, idx1, stroke, -1),\n\t\t\t\t\tclipBandLine(u, seriesIdx, idx0, idx1, stroke,  1),\n\t\t\t\t] : clipBandLine(u, seriesIdx, idx0, idx1, stroke, bandClipDir);\n\t\t\t}\n\n\t\t\treturn _paths;\n\t\t});\n\t};\n}\n\nfunction stepped(opts) {\n\tconst align = ifNull(opts.align, 1);\n\t// whether to draw ascenders/descenders at null/gap bondaries\n\tconst ascDesc = ifNull(opts.ascDesc, false);\n\n\tconst alignGaps = ifNull(opts.alignGaps, 0);\n\n\treturn (u, seriesIdx, idx0, idx1) => {\n\t\treturn orient(u, seriesIdx, (series, dataX, dataY, scaleX, scaleY, valToPosX, valToPosY, xOff, yOff, xDim, yDim) => {\n\t\t\tlet pxRound = series.pxRound;\n\n\t\t\tlet pixelForX = val => pxRound(valToPosX(val, scaleX, xDim, xOff));\n\t\t\tlet pixelForY = val => pxRound(valToPosY(val, scaleY, yDim, yOff));\n\n\t\t\tlet lineTo = scaleX.ori == 0 ? lineToH : lineToV;\n\n\t\t\tconst _paths = {stroke: new Path2D(), fill: null, clip: null, band: null, gaps: null, flags: BAND_CLIP_FILL};\n\t\t\tconst stroke = _paths.stroke;\n\n\t\t\tconst dir = scaleX.dir * (scaleX.ori == 0 ? 1 : -1);\n\n\t\t\tidx0 = nonNullIdx(dataY, idx0, idx1,  1);\n\t\t\tidx1 = nonNullIdx(dataY, idx0, idx1, -1);\n\n\t\t\tlet prevYPos  = pixelForY(dataY[dir == 1 ? idx0 : idx1]);\n\t\t\tlet firstXPos = pixelForX(dataX[dir == 1 ? idx0 : idx1]);\n\t\t\tlet prevXPos = firstXPos;\n\n\t\t\tlineTo(stroke, firstXPos, prevYPos);\n\n\t\t\tfor (let i = dir == 1 ? idx0 : idx1; i >= idx0 && i <= idx1; i += dir) {\n\t\t\t\tlet yVal1 = dataY[i];\n\n\t\t\t\tif (yVal1 == null)\n\t\t\t\t\tcontinue;\n\n\t\t\t\tlet x1 = pixelForX(dataX[i]);\n\t\t\t\tlet y1 = pixelForY(yVal1);\n\n\t\t\t\tif (align == 1)\n\t\t\t\t\tlineTo(stroke, x1, prevYPos);\n\t\t\t\telse\n\t\t\t\t\tlineTo(stroke, prevXPos, y1);\n\n\t\t\t\tlineTo(stroke, x1, y1);\n\n\t\t\t\tprevYPos = y1;\n\t\t\t\tprevXPos = x1;\n\t\t\t}\n\n\t\t\tlet [ bandFillDir, bandClipDir ] = bandFillClipDirs(u, seriesIdx);\n\n\t\t\tif (series.fill != null || bandFillDir != 0) {\n\t\t\t\tlet fill = _paths.fill = new Path2D(stroke);\n\n\t\t\t\tlet fillTo = series.fillTo(u, seriesIdx, series.min, series.max, bandFillDir);\n\t\t\t\tlet fillToY = pixelForY(fillTo);\n\n\t\t\t\tlineTo(fill, prevXPos, fillToY);\n\t\t\t\tlineTo(fill, firstXPos, fillToY);\n\t\t\t}\n\n\t\t\tif (!series.spanGaps) {\n\t\t\t//\tconsole.time('gaps');\n\t\t\t\tlet gaps = [];\n\n\t\t\t\tgaps.push(...findGaps(dataX, dataY, idx0, idx1, dir, pixelForX, alignGaps));\n\n\t\t\t//\tconsole.timeEnd('gaps');\n\n\t\t\t//\tconsole.log('gaps', JSON.stringify(gaps));\n\n\t\t\t\t// expand/contract clips for ascenders/descenders\n\t\t\t\tlet halfStroke = (series.width * pxRatio) / 2;\n\t\t\t\tlet startsOffset = (ascDesc || align ==  1) ?  halfStroke : -halfStroke;\n\t\t\t\tlet endsOffset   = (ascDesc || align == -1) ? -halfStroke :  halfStroke;\n\n\t\t\t\tgaps.forEach(g => {\n\t\t\t\t\tg[0] += startsOffset;\n\t\t\t\t\tg[1] += endsOffset;\n\t\t\t\t});\n\n\t\t\t\t_paths.gaps = gaps = series.gaps(u, seriesIdx, idx0, idx1, gaps);\n\n\t\t\t\t_paths.clip = clipGaps(gaps, scaleX.ori, xOff, yOff, xDim, yDim);\n\t\t\t}\n\n\t\t\tif (bandClipDir != 0) {\n\t\t\t\t_paths.band = bandClipDir == 2 ? [\n\t\t\t\t\tclipBandLine(u, seriesIdx, idx0, idx1, stroke, -1),\n\t\t\t\t\tclipBandLine(u, seriesIdx, idx0, idx1, stroke,  1),\n\t\t\t\t] : clipBandLine(u, seriesIdx, idx0, idx1, stroke, bandClipDir);\n\t\t\t}\n\n\t\t\treturn _paths;\n\t\t});\n\t};\n}\n\nfunction bars(opts) {\n\topts = opts || EMPTY_OBJ;\n\tconst size = ifNull(opts.size, [0.6, inf, 1]);\n\tconst align = opts.align || 0;\n\tconst extraGap = (opts.gap || 0) * pxRatio;\n\n\tconst radius = ifNull(opts.radius, 0);\n\n\tconst gapFactor = 1 - size[0];\n\tconst maxWidth  = ifNull(size[1], inf) * pxRatio;\n\tconst minWidth  = ifNull(size[2], 1) * pxRatio;\n\n\tconst disp = ifNull(opts.disp, EMPTY_OBJ);\n\tconst _each = ifNull(opts.each, _ => {});\n\n\tconst { fill: dispFills, stroke: dispStrokes } = disp;\n\n\treturn (u, seriesIdx, idx0, idx1) => {\n\t\treturn orient(u, seriesIdx, (series, dataX, dataY, scaleX, scaleY, valToPosX, valToPosY, xOff, yOff, xDim, yDim) => {\n\t\t\tlet pxRound = series.pxRound;\n\n\t\t\tconst _dirX = scaleX.dir * (scaleX.ori == 0 ? 1 : -1);\n\t\t\tconst _dirY = scaleY.dir * (scaleY.ori == 1 ? 1 : -1);\n\n\t\t\tlet rect = scaleX.ori == 0 ? rectH : rectV;\n\n\t\t\tlet each = scaleX.ori == 0 ? _each : (u, seriesIdx, i, top, lft, hgt, wid) => {\n\t\t\t\t_each(u, seriesIdx, i, lft, top, wid, hgt);\n\t\t\t};\n\n\t\t\tlet [ bandFillDir, bandClipDir ] = bandFillClipDirs(u, seriesIdx);\n\n\t\t//\tlet fillToY = series.fillTo(u, seriesIdx, series.min, series.max, bandFillDir);\n\t\t\tlet fillToY = scaleY.distr == 3 ? (bandFillDir == 1 ? scaleY.max : scaleY.min) : 0;\n\n\t\t\tlet y0Pos = valToPosY(fillToY, scaleY, yDim, yOff);\n\n\t\t\t// barWid is to center of stroke\n\t\t\tlet xShift, barWid;\n\n\t\t\tlet strokeWidth = pxRound(series.width * pxRatio);\n\n\t\t\tlet multiPath = false;\n\n\t\t\tlet fillColors = null;\n\t\t\tlet fillPaths = null;\n\t\t\tlet strokeColors = null;\n\t\t\tlet strokePaths = null;\n\n\t\t\tif (dispFills != null && (strokeWidth == 0 || dispStrokes != null)) {\n\t\t\t\tmultiPath = true;\n\n\t\t\t\tfillColors = dispFills.values(u, seriesIdx, idx0, idx1);\n\t\t\t\tfillPaths = new Map();\n\t\t\t\t(new Set(fillColors)).forEach(color => {\n\t\t\t\t\tif (color != null)\n\t\t\t\t\t\tfillPaths.set(color, new Path2D());\n\t\t\t\t});\n\n\t\t\t\tif (strokeWidth > 0) {\n\t\t\t\t\tstrokeColors = dispStrokes.values(u, seriesIdx, idx0, idx1);\n\t\t\t\t\tstrokePaths = new Map();\n\t\t\t\t\t(new Set(strokeColors)).forEach(color => {\n\t\t\t\t\t\tif (color != null)\n\t\t\t\t\t\t\tstrokePaths.set(color, new Path2D());\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tlet { x0, size } = disp;\n\n\t\t\tif (x0 != null && size != null) {\n\t\t\t\tdataX = x0.values(u, seriesIdx, idx0, idx1);\n\n\t\t\t\tif (x0.unit == 2)\n\t\t\t\t\tdataX = dataX.map(pct => u.posToVal(xOff + pct * xDim, scaleX.key, true));\n\n\t\t\t\t// assumes uniform sizes, for now\n\t\t\t\tlet sizes = size.values(u, seriesIdx, idx0, idx1);\n\n\t\t\t\tif (size.unit == 2)\n\t\t\t\t\tbarWid = sizes[0] * xDim;\n\t\t\t\telse\n\t\t\t\t\tbarWid = valToPosX(sizes[0], scaleX, xDim, xOff) - valToPosX(0, scaleX, xDim, xOff); // assumes linear scale (delta from 0)\n\n\t\t\t\tbarWid = pxRound(barWid - strokeWidth);\n\n\t\t\t\txShift = (_dirX == 1 ? -strokeWidth / 2 : barWid + strokeWidth / 2);\n\t\t\t}\n\t\t\telse {\n\t\t\t\tlet colWid = xDim;\n\n\t\t\t\tif (dataX.length > 1) {\n\t\t\t\t\t// prior index with non-undefined y data\n\t\t\t\t\tlet prevIdx = null;\n\n\t\t\t\t\t// scan full dataset for smallest adjacent delta\n\t\t\t\t\t// will not work properly for non-linear x scales, since does not do expensive valToPosX calcs till end\n\t\t\t\t\tfor (let i = 0, minDelta = Infinity; i < dataX.length; i++) {\n\t\t\t\t\t\tif (dataY[i] !== undefined) {\n\t\t\t\t\t\t\tif (prevIdx != null) {\n\t\t\t\t\t\t\t\tlet delta = abs(dataX[i] - dataX[prevIdx]);\n\n\t\t\t\t\t\t\t\tif (delta < minDelta) {\n\t\t\t\t\t\t\t\t\tminDelta = delta;\n\t\t\t\t\t\t\t\t\tcolWid = abs(valToPosX(dataX[i], scaleX, xDim, xOff) - valToPosX(dataX[prevIdx], scaleX, xDim, xOff));\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tprevIdx = i;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tlet gapWid = colWid * gapFactor;\n\n\t\t\t\tbarWid = pxRound(min(maxWidth, max(minWidth, colWid - gapWid)) - strokeWidth - extraGap);\n\n\t\t\t\txShift = (align == 0 ? barWid / 2 : align == _dirX ? 0 : barWid) - align * _dirX * extraGap / 2;\n\t\t\t}\n\n\t\t\tconst _paths = {stroke: null, fill: null, clip: null, band: null, gaps: null, flags: BAND_CLIP_FILL | BAND_CLIP_STROKE};  // disp, geom\n\n\t\t\tlet yLimit;\n\n\t\t\tif (bandClipDir != 0) {\n\t\t\t\t_paths.band = new Path2D();\n\t\t\t\tyLimit = pxRound(valToPosY(bandClipDir == 1 ? scaleY.max : scaleY.min, scaleY, yDim, yOff));\n\t\t\t}\n\n\t\t\tconst stroke = multiPath ? null : new Path2D();\n\t\t\tconst band = _paths.band;\n\n\t\t\tlet { y0, y1 } = disp;\n\n\t\t\tlet dataY0 = null;\n\n\t\t\tif (y0 != null && y1 != null) {\n\t\t\t\tdataY = y1.values(u, seriesIdx, idx0, idx1);\n\t\t\t\tdataY0 = y0.values(u, seriesIdx, idx0, idx1);\n\t\t\t}\n\n\t\t\tfor (let i = _dirX == 1 ? idx0 : idx1; i >= idx0 && i <= idx1; i += _dirX) {\n\t\t\t\tlet yVal = dataY[i];\n\n\t\t\t\t// we can skip both, drawing and band clipping for alignment artifacts\n\t\t\t\tif (yVal === undefined)\n\t\t\t\t\tcontinue;\n\n\t\t\t/*\n\t\t\t\t// interpolate upwards band clips\n\t\t\t\tif (yVal == null) {\n\t\t\t\t//\tif (hasBands)\n\t\t\t\t//\t\tyVal = costlyLerp(i, idx0, idx1, _dirX, dataY);\n\t\t\t\t//\telse\n\t\t\t\t\t\tcontinue;\n\t\t\t\t}\n\t\t\t*/\n\n\t\t\t\tlet xVal = scaleX.distr != 2 || disp != null ? dataX[i] : i;\n\n\t\t\t\t// TODO: all xPos can be pre-computed once for all series in aligned set\n\t\t\t\tlet xPos = valToPosX(xVal, scaleX, xDim, xOff);\n\t\t\t\tlet yPos = valToPosY(ifNull(yVal, fillToY), scaleY, yDim, yOff);\n\n\t\t\t\tif (dataY0 != null && yVal != null)\n\t\t\t\t\ty0Pos = valToPosY(dataY0[i], scaleY, yDim, yOff);\n\n\t\t\t\tlet lft = pxRound(xPos - xShift);\n\t\t\t\tlet btm = pxRound(max(yPos, y0Pos));\n\t\t\t\tlet top = pxRound(min(yPos, y0Pos));\n\t\t\t\t// this includes the stroke\n\t\t\t\tlet barHgt = btm - top;\n\n\t\t\t\tlet r = radius * barWid;\n\n\t\t\t\tif (yVal != null) {  // && yVal != fillToY (0 height bar)\n\t\t\t\t\tif (multiPath) {\n\t\t\t\t\t\tif (strokeWidth > 0 && strokeColors[i] != null)\n\t\t\t\t\t\t\trect(strokePaths.get(strokeColors[i]), lft, top + floor(strokeWidth / 2), barWid, max(0, barHgt - strokeWidth), r);\n\n\t\t\t\t\t\tif (fillColors[i] != null)\n\t\t\t\t\t\t\trect(fillPaths.get(fillColors[i]), lft, top + floor(strokeWidth / 2), barWid, max(0, barHgt - strokeWidth), r);\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t\trect(stroke, lft, top + floor(strokeWidth / 2), barWid, max(0, barHgt - strokeWidth), r);\n\n\t\t\t\t\teach(u, seriesIdx, i,\n\t\t\t\t\t\tlft    - strokeWidth / 2,\n\t\t\t\t\t\ttop,\n\t\t\t\t\t\tbarWid + strokeWidth,\n\t\t\t\t\t\tbarHgt,\n\t\t\t\t\t);\n\t\t\t\t}\n\n\t\t\t\tif (bandClipDir != 0) {\n\t\t\t\t\tif (_dirY * bandClipDir == 1) {\n\t\t\t\t\t\tbtm = top;\n\t\t\t\t\t\ttop = yLimit;\n\t\t\t\t\t}\n\t\t\t\t\telse {\n\t\t\t\t\t\ttop = btm;\n\t\t\t\t\t\tbtm = yLimit;\n\t\t\t\t\t}\n\n\t\t\t\t\tbarHgt = btm - top;\n\n\t\t\t\t\trect(band, lft - strokeWidth / 2, top, barWid + strokeWidth, max(0, barHgt), 0);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (strokeWidth > 0)\n\t\t\t\t_paths.stroke = multiPath ? strokePaths : stroke;\n\n\t\t\t_paths.fill = multiPath ? fillPaths : stroke;\n\n\t\t\treturn _paths;\n\t\t});\n\t};\n}\n\nfunction splineInterp(interp, opts) {\n\tconst alignGaps = ifNull(opts?.alignGaps, 0);\n\n\treturn (u, seriesIdx, idx0, idx1) => {\n\t\treturn orient(u, seriesIdx, (series, dataX, dataY, scaleX, scaleY, valToPosX, valToPosY, xOff, yOff, xDim, yDim) => {\n\t\t\tlet pxRound = series.pxRound;\n\n\t\t\tlet pixelForX = val => pxRound(valToPosX(val, scaleX, xDim, xOff));\n\t\t\tlet pixelForY = val => pxRound(valToPosY(val, scaleY, yDim, yOff));\n\n\t\t\tlet moveTo, bezierCurveTo, lineTo;\n\n\t\t\tif (scaleX.ori == 0) {\n\t\t\t\tmoveTo = moveToH;\n\t\t\t\tlineTo = lineToH;\n\t\t\t\tbezierCurveTo = bezierCurveToH;\n\t\t\t}\n\t\t\telse {\n\t\t\t\tmoveTo = moveToV;\n\t\t\t\tlineTo = lineToV;\n\t\t\t\tbezierCurveTo = bezierCurveToV;\n\t\t\t}\n\n\t\t\tconst dir = scaleX.dir * (scaleX.ori == 0 ? 1 : -1);\n\n\t\t\tidx0 = nonNullIdx(dataY, idx0, idx1,  1);\n\t\t\tidx1 = nonNullIdx(dataY, idx0, idx1, -1);\n\n\t\t\tlet firstXPos = pixelForX(dataX[dir == 1 ? idx0 : idx1]);\n\t\t\tlet prevXPos = firstXPos;\n\n\t\t\tlet xCoords = [];\n\t\t\tlet yCoords = [];\n\n\t\t\tfor (let i = dir == 1 ? idx0 : idx1; i >= idx0 && i <= idx1; i += dir) {\n\t\t\t\tlet yVal = dataY[i];\n\n\t\t\t\tif (yVal != null) {\n\t\t\t\t\tlet xVal = dataX[i];\n\t\t\t\t\tlet xPos = pixelForX(xVal);\n\n\t\t\t\t\txCoords.push(prevXPos = xPos);\n\t\t\t\t\tyCoords.push(pixelForY(dataY[i]));\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst _paths = {stroke: interp(xCoords, yCoords, moveTo, lineTo, bezierCurveTo, pxRound), fill: null, clip: null, band: null, gaps: null, flags: BAND_CLIP_FILL};\n\t\t\tconst stroke = _paths.stroke;\n\n\t\t\tlet [ bandFillDir, bandClipDir ] = bandFillClipDirs(u, seriesIdx);\n\n\t\t\tif (series.fill != null || bandFillDir != 0) {\n\t\t\t\tlet fill = _paths.fill = new Path2D(stroke);\n\n\t\t\t\tlet fillTo = series.fillTo(u, seriesIdx, series.min, series.max, bandFillDir);\n\t\t\t\tlet fillToY = pixelForY(fillTo);\n\n\t\t\t\tlineTo(fill, prevXPos, fillToY);\n\t\t\t\tlineTo(fill, firstXPos, fillToY);\n\t\t\t}\n\n\t\t\tif (!series.spanGaps) {\n\t\t\t//\tconsole.time('gaps');\n\t\t\t\tlet gaps = [];\n\n\t\t\t\tgaps.push(...findGaps(dataX, dataY, idx0, idx1, dir, pixelForX, alignGaps));\n\n\t\t\t//\tconsole.timeEnd('gaps');\n\n\t\t\t//\tconsole.log('gaps', JSON.stringify(gaps));\n\n\t\t\t\t_paths.gaps = gaps = series.gaps(u, seriesIdx, idx0, idx1, gaps);\n\n\t\t\t\t_paths.clip = clipGaps(gaps, scaleX.ori, xOff, yOff, xDim, yDim);\n\t\t\t}\n\n\t\t\tif (bandClipDir != 0) {\n\t\t\t\t_paths.band = bandClipDir == 2 ? [\n\t\t\t\t\tclipBandLine(u, seriesIdx, idx0, idx1, stroke, -1),\n\t\t\t\t\tclipBandLine(u, seriesIdx, idx0, idx1, stroke,  1),\n\t\t\t\t] : clipBandLine(u, seriesIdx, idx0, idx1, stroke, bandClipDir);\n\t\t\t}\n\n\t\t\treturn _paths;\n\n\t\t\t//  if FEAT_PATHS: false in rollup.config.js\n\t\t\t//\tu.ctx.save();\n\t\t\t//\tu.ctx.beginPath();\n\t\t\t//\tu.ctx.rect(u.bbox.left, u.bbox.top, u.bbox.width, u.bbox.height);\n\t\t\t//\tu.ctx.clip();\n\t\t\t//\tu.ctx.strokeStyle = u.series[sidx].stroke;\n\t\t\t//\tu.ctx.stroke(stroke);\n\t\t\t//\tu.ctx.fillStyle = u.series[sidx].fill;\n\t\t\t//\tu.ctx.fill(fill);\n\t\t\t//\tu.ctx.restore();\n\t\t\t//\treturn null;\n\t\t});\n\t};\n}\n\nfunction monotoneCubic(opts) {\n\treturn splineInterp(_monotoneCubic, opts);\n}\n\n// Monotone Cubic Spline interpolation, adapted from the Chartist.js implementation:\n// https://github.com/gionkunz/chartist-js/blob/e7e78201bffe9609915e5e53cfafa29a5d6c49f9/src/scripts/interpolation.js#L240-L369\nfunction _monotoneCubic(xs, ys, moveTo, lineTo, bezierCurveTo, pxRound) {\n\tconst n = xs.length;\n\n\tif (n < 2)\n\t\treturn null;\n\n\tconst path = new Path2D();\n\n\tmoveTo(path, xs[0], ys[0]);\n\n\tif (n == 2)\n\t\tlineTo(path, xs[1], ys[1]);\n\telse {\n\t\tlet ms  = Array(n),\n\t\t\tds  = Array(n - 1),\n\t\t\tdys = Array(n - 1),\n\t\t\tdxs = Array(n - 1);\n\n\t\t// calc deltas and derivative\n\t\tfor (let i = 0; i < n - 1; i++) {\n\t\t\tdys[i] = ys[i + 1] - ys[i];\n\t\t\tdxs[i] = xs[i + 1] - xs[i];\n\t\t\tds[i]  = dys[i] / dxs[i];\n\t\t}\n\n\t\t// determine desired slope (m) at each point using Fritsch-Carlson method\n\t\t// http://math.stackexchange.com/questions/45218/implementation-of-monotone-cubic-interpolation\n\t\tms[0] = ds[0];\n\n\t\tfor (let i = 1; i < n - 1; i++) {\n\t\t\tif (ds[i] === 0 || ds[i - 1] === 0 || (ds[i - 1] > 0) !== (ds[i] > 0))\n\t\t\t\tms[i] = 0;\n\t\t\telse {\n\t\t\t\tms[i] = 3 * (dxs[i - 1] + dxs[i]) / (\n\t\t\t\t\t(2 * dxs[i] + dxs[i - 1]) / ds[i - 1] +\n\t\t\t\t\t(dxs[i] + 2 * dxs[i - 1]) / ds[i]\n\t\t\t\t);\n\n\t\t\t\tif (!isFinite(ms[i]))\n\t\t\t\t\tms[i] = 0;\n\t\t\t}\n\t\t}\n\n\t\tms[n - 1] = ds[n - 2];\n\n\t\tfor (let i = 0; i < n - 1; i++) {\n\t\t\tbezierCurveTo(\n\t\t\t\tpath,\n\t\t\t\txs[i] + dxs[i] / 3,\n\t\t\t\tys[i] + ms[i] * dxs[i] / 3,\n\t\t\t\txs[i + 1] - dxs[i] / 3,\n\t\t\t\tys[i + 1] - ms[i + 1] * dxs[i] / 3,\n\t\t\t\txs[i + 1],\n\t\t\t\tys[i + 1],\n\t\t\t);\n\t\t}\n\t}\n\n\treturn path;\n}\n\nconst cursorPlots = new Set();\n\nfunction invalidateRects() {\n\tcursorPlots.forEach(u => {\n\t\tu.syncRect(true);\n\t});\n}\n\nif (domEnv) {\n\ton(resize, win, invalidateRects);\n\ton(scroll, win, invalidateRects, true);\n\ton(dppxchange, win, () => { uPlot.pxRatio = pxRatio; });\n}\n\nconst linearPath = linear() ;\nconst pointsPath = points() ;\n\nfunction setDefaults(d, xo, yo, initY) {\n\tlet d2 = initY ? [d[0], d[1]].concat(d.slice(2)) : [d[0]].concat(d.slice(1));\n\treturn d2.map((o, i) => setDefault(o, i, xo, yo));\n}\n\nfunction setDefaults2(d, xyo) {\n\treturn d.map((o, i) => i == 0 ? null : assign({}, xyo, o));  // todo: assign() will not merge facet arrays\n}\n\nfunction setDefault(o, i, xo, yo) {\n\treturn assign({}, (i == 0 ? xo : yo), o);\n}\n\nfunction snapNumX(self, dataMin, dataMax) {\n\treturn dataMin == null ? nullNullTuple : [dataMin, dataMax];\n}\n\nconst snapTimeX = snapNumX;\n\n// this ensures that non-temporal/numeric y-axes get multiple-snapped padding added above/below\n// TODO: also account for incrs when snapping to ensure top of axis gets a tick & value\nfunction snapNumY(self, dataMin, dataMax) {\n\treturn dataMin == null ? nullNullTuple : rangeNum(dataMin, dataMax, rangePad, true);\n}\n\nfunction snapLogY(self, dataMin, dataMax, scale) {\n\treturn dataMin == null ? nullNullTuple : rangeLog(dataMin, dataMax, self.scales[scale].log, false);\n}\n\nconst snapLogX = snapLogY;\n\nfunction snapAsinhY(self, dataMin, dataMax, scale) {\n\treturn dataMin == null ? nullNullTuple : rangeAsinh(dataMin, dataMax, self.scales[scale].log, false);\n}\n\nconst snapAsinhX = snapAsinhY;\n\n// dim is logical (getClientBoundingRect) pixels, not canvas pixels\nfunction findIncr(minVal, maxVal, incrs, dim, minSpace) {\n\tlet intDigits = max(numIntDigits(minVal), numIntDigits(maxVal));\n\n\tlet delta = maxVal - minVal;\n\n\tlet incrIdx = closestIdx((minSpace / dim) * delta, incrs);\n\n\tdo {\n\t\tlet foundIncr = incrs[incrIdx];\n\t\tlet foundSpace = dim * foundIncr / delta;\n\n\t\tif (foundSpace >= minSpace && intDigits + (foundIncr < 5 ? fixedDec.get(foundIncr) : 0) <= 17)\n\t\t\treturn [foundIncr, foundSpace];\n\t} while (++incrIdx < incrs.length);\n\n\treturn [0, 0];\n}\n\nfunction pxRatioFont(font) {\n\tlet fontSize, fontSizeCss;\n\tfont = font.replace(/(\\d+)px/, (m, p1) => (fontSize = round((fontSizeCss = +p1) * pxRatio)) + 'px');\n\treturn [font, fontSize, fontSizeCss];\n}\n\nfunction syncFontSize(axis) {\n\tif (axis.show) {\n\t\t[axis.font, axis.labelFont].forEach(f => {\n\t\t\tlet size = roundDec(f[2] * pxRatio, 1);\n\t\t\tf[0] = f[0].replace(/[0-9.]+px/, size + 'px');\n\t\t\tf[1] = size;\n\t\t});\n\t}\n}\n\nfunction uPlot(opts, data, then) {\n\tconst self = {\n\t\tmode: ifNull(opts.mode, 1),\n\t};\n\n\tconst mode = self.mode;\n\n\t// TODO: cache denoms & mins scale.cache = {r, min, }\n\tfunction getValPct(val, scale) {\n\t\tlet _val = (\n\t\t\tscale.distr == 3 ? log10(val > 0 ? val : scale.clamp(self, val, scale.min, scale.max, scale.key)) :\n\t\t\tscale.distr == 4 ? asinh(val, scale.asinh) :\n\t\t\tval\n\t\t);\n\n\t\treturn (_val - scale._min) / (scale._max - scale._min);\n\t}\n\n\tfunction getHPos(val, scale, dim, off) {\n\t\tlet pct = getValPct(val, scale);\n\t\treturn off + dim * (scale.dir == -1 ? (1 - pct) : pct);\n\t}\n\n\tfunction getVPos(val, scale, dim, off) {\n\t\tlet pct = getValPct(val, scale);\n\t\treturn off + dim * (scale.dir == -1 ? pct : (1 - pct));\n\t}\n\n\tfunction getPos(val, scale, dim, off) {\n\t\treturn scale.ori == 0 ? getHPos(val, scale, dim, off) : getVPos(val, scale, dim, off);\n\t}\n\n\tself.valToPosH = getHPos;\n\tself.valToPosV = getVPos;\n\n\tlet ready = false;\n\tself.status = 0;\n\n\tconst root = self.root = placeDiv(UPLOT);\n\n\tif (opts.id != null)\n\t\troot.id = opts.id;\n\n\taddClass(root, opts.class);\n\n\tif (opts.title) {\n\t\tlet title = placeDiv(TITLE, root);\n\t\ttitle.textContent = opts.title;\n\t}\n\n\tconst can = placeTag(\"canvas\");\n\tconst ctx = self.ctx = can.getContext(\"2d\");\n\n\tconst wrap = placeDiv(WRAP, root);\n\tconst under = self.under = placeDiv(UNDER, wrap);\n\twrap.appendChild(can);\n\tconst over = self.over = placeDiv(OVER, wrap);\n\n\topts = copy(opts);\n\n\tconst pxAlign = +ifNull(opts.pxAlign, 1);\n\n\tconst pxRound = pxRoundGen(pxAlign);\n\n\t(opts.plugins || []).forEach(p => {\n\t\tif (p.opts)\n\t\t\topts = p.opts(self, opts) || opts;\n\t});\n\n\tconst ms = opts.ms || 1e-3;\n\n\tconst series  = self.series = mode == 1 ?\n\t\tsetDefaults(opts.series || [], xSeriesOpts, ySeriesOpts, false) :\n\t\tsetDefaults2(opts.series || [null], xySeriesOpts);\n\tconst axes    = self.axes   = setDefaults(opts.axes   || [], xAxisOpts,   yAxisOpts,    true);\n\tconst scales  = self.scales = {};\n\tconst bands   = self.bands  = opts.bands || [];\n\n\tbands.forEach(b => {\n\t\tb.fill = fnOrSelf(b.fill || null);\n\t\tb.dir = ifNull(b.dir, -1);\n\t});\n\n\tconst xScaleKey = mode == 2 ? series[1].facets[0].scale : series[0].scale;\n\n\tconst drawOrderMap = {\n\t\taxes: drawAxesGrid,\n\t\tseries: drawSeries,\n\t};\n\n\tconst drawOrder = (opts.drawOrder || [\"axes\", \"series\"]).map(key => drawOrderMap[key]);\n\n\tfunction initScale(scaleKey) {\n\t\tlet sc = scales[scaleKey];\n\n\t\tif (sc == null) {\n\t\t\tlet scaleOpts = (opts.scales || EMPTY_OBJ)[scaleKey] || EMPTY_OBJ;\n\n\t\t\tif (scaleOpts.from != null) {\n\t\t\t\t// ensure parent is initialized\n\t\t\t\tinitScale(scaleOpts.from);\n\t\t\t\t// dependent scales inherit\n\t\t\t\tscales[scaleKey] = assign({}, scales[scaleOpts.from], scaleOpts, {key: scaleKey});\n\t\t\t}\n\t\t\telse {\n\t\t\t\tsc = scales[scaleKey] = assign({}, (scaleKey == xScaleKey ? xScaleOpts : yScaleOpts), scaleOpts);\n\n\t\t\t\tsc.key = scaleKey;\n\n\t\t\t\tlet isTime = sc.time;\n\n\t\t\t\tlet rn = sc.range;\n\n\t\t\t\tlet rangeIsArr = isArr(rn);\n\n\t\t\t\tif (scaleKey != xScaleKey || (mode == 2 && !isTime)) {\n\t\t\t\t\t// if range array has null limits, it should be auto\n\t\t\t\t\tif (rangeIsArr && (rn[0] == null || rn[1] == null)) {\n\t\t\t\t\t\trn = {\n\t\t\t\t\t\t\tmin: rn[0] == null ? autoRangePart : {\n\t\t\t\t\t\t\t\tmode: 1,\n\t\t\t\t\t\t\t\thard: rn[0],\n\t\t\t\t\t\t\t\tsoft: rn[0],\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\tmax: rn[1] == null ? autoRangePart : {\n\t\t\t\t\t\t\t\tmode: 1,\n\t\t\t\t\t\t\t\thard: rn[1],\n\t\t\t\t\t\t\t\tsoft: rn[1],\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t};\n\t\t\t\t\t\trangeIsArr = false;\n\t\t\t\t\t}\n\n\t\t\t\t\tif (!rangeIsArr && isObj(rn)) {\n\t\t\t\t\t\tlet cfg = rn;\n\t\t\t\t\t\t// this is similar to snapNumY\n\t\t\t\t\t\trn = (self, dataMin, dataMax) => dataMin == null ? nullNullTuple : rangeNum(dataMin, dataMax, cfg);\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tsc.range = fnOrSelf(rn || (isTime ? snapTimeX : scaleKey == xScaleKey ?\n\t\t\t\t\t(sc.distr == 3 ? snapLogX : sc.distr == 4 ? snapAsinhX : snapNumX) :\n\t\t\t\t\t(sc.distr == 3 ? snapLogY : sc.distr == 4 ? snapAsinhY : snapNumY)\n\t\t\t\t));\n\n\t\t\t\tsc.auto = fnOrSelf(rangeIsArr ? false : sc.auto);\n\n\t\t\t\tsc.clamp = fnOrSelf(sc.clamp || clampScale);\n\n\t\t\t\t// caches for expensive ops like asinh() & log()\n\t\t\t\tsc._min = sc._max = null;\n\t\t\t}\n\t\t}\n\t}\n\n\tinitScale(\"x\");\n\tinitScale(\"y\");\n\n\t// TODO: init scales from facets in mode: 2\n\tif (mode == 1) {\n\t\tseries.forEach(s => {\n\t\t\tinitScale(s.scale);\n\t\t});\n\t}\n\n\taxes.forEach(a => {\n\t\tinitScale(a.scale);\n\t});\n\n\tfor (let k in opts.scales)\n\t\tinitScale(k);\n\n\tconst scaleX = scales[xScaleKey];\n\n\tconst xScaleDistr = scaleX.distr;\n\n\tlet valToPosX, valToPosY;\n\n\tif (scaleX.ori == 0) {\n\t\taddClass(root, ORI_HZ);\n\t\tvalToPosX = getHPos;\n\t\tvalToPosY = getVPos;\n\t\t/*\n\t\tupdOriDims = () => {\n\t\t\txDimCan = plotWid;\n\t\t\txOffCan = plotLft;\n\t\t\tyDimCan = plotHgt;\n\t\t\tyOffCan = plotTop;\n\n\t\t\txDimCss = plotWidCss;\n\t\t\txOffCss = plotLftCss;\n\t\t\tyDimCss = plotHgtCss;\n\t\t\tyOffCss = plotTopCss;\n\t\t};\n\t\t*/\n\t}\n\telse {\n\t\taddClass(root, ORI_VT);\n\t\tvalToPosX = getVPos;\n\t\tvalToPosY = getHPos;\n\t\t/*\n\t\tupdOriDims = () => {\n\t\t\txDimCan = plotHgt;\n\t\t\txOffCan = plotTop;\n\t\t\tyDimCan = plotWid;\n\t\t\tyOffCan = plotLft;\n\n\t\t\txDimCss = plotHgtCss;\n\t\t\txOffCss = plotTopCss;\n\t\t\tyDimCss = plotWidCss;\n\t\t\tyOffCss = plotLftCss;\n\t\t};\n\t\t*/\n\t}\n\n\tconst pendScales = {};\n\n\t// explicitly-set initial scales\n\tfor (let k in scales) {\n\t\tlet sc = scales[k];\n\n\t\tif (sc.min != null || sc.max != null) {\n\t\t\tpendScales[k] = {min: sc.min, max: sc.max};\n\t\t\tsc.min = sc.max = null;\n\t\t}\n\t}\n\n//\tself.tz = opts.tz || Intl.DateTimeFormat().resolvedOptions().timeZone;\n\tconst _tzDate  = (opts.tzDate || (ts => new Date(round(ts / ms))));\n\tconst _fmtDate = (opts.fmtDate || fmtDate);\n\n\tconst _timeAxisSplits = (ms == 1 ? timeAxisSplitsMs(_tzDate) : timeAxisSplitsS(_tzDate));\n\tconst _timeAxisVals   = timeAxisVals(_tzDate, timeAxisStamps((ms == 1 ? _timeAxisStampsMs : _timeAxisStampsS), _fmtDate));\n\tconst _timeSeriesVal  = timeSeriesVal(_tzDate, timeSeriesStamp(_timeSeriesStamp, _fmtDate));\n\n\tconst activeIdxs = [];\n\n\tconst legend     = (self.legend = assign({}, legendOpts, opts.legend));\n\tconst showLegend = legend.show;\n\tconst markers    = legend.markers;\n\n\t{\n\t\tlegend.idxs = activeIdxs;\n\n\t\tmarkers.width  = fnOrSelf(markers.width);\n\t\tmarkers.dash   = fnOrSelf(markers.dash);\n\t\tmarkers.stroke = fnOrSelf(markers.stroke);\n\t\tmarkers.fill   = fnOrSelf(markers.fill);\n\t}\n\n\tlet legendEl;\n\tlet legendRows = [];\n\tlet legendCells = [];\n\tlet legendCols;\n\tlet multiValLegend = false;\n\tlet NULL_LEGEND_VALUES = {};\n\n\tif (legend.live) {\n\t\tconst getMultiVals = series[1] ? series[1].values : null;\n\t\tmultiValLegend = getMultiVals != null;\n\t\tlegendCols = multiValLegend ? getMultiVals(self, 1, 0) : {_: 0};\n\n\t\tfor (let k in legendCols)\n\t\t\tNULL_LEGEND_VALUES[k] = \"--\";\n\t}\n\n\tif (showLegend) {\n\t\tlegendEl = placeTag(\"table\", LEGEND, root);\n\n\t\tif (multiValLegend) {\n\t\t\tlet head = placeTag(\"tr\", LEGEND_THEAD, legendEl);\n\t\t\tplaceTag(\"th\", null, head);\n\n\t\t\tfor (var key in legendCols)\n\t\t\t\tplaceTag(\"th\", LEGEND_LABEL, head).textContent = key;\n\t\t}\n\t\telse {\n\t\t\taddClass(legendEl, LEGEND_INLINE);\n\t\t\tlegend.live && addClass(legendEl, LEGEND_LIVE);\n\t\t}\n\t}\n\n\tconst son  = {show: true};\n\tconst soff = {show: false};\n\n\tfunction initLegendRow(s, i) {\n\t\tif (i == 0 && (multiValLegend || !legend.live || mode == 2))\n\t\t\treturn nullNullTuple;\n\n\t\tlet cells = [];\n\n\t\tlet row = placeTag(\"tr\", LEGEND_SERIES, legendEl, legendEl.childNodes[i]);\n\n\t\taddClass(row, s.class);\n\n\t\tif (!s.show)\n\t\t\taddClass(row, OFF);\n\n\t\tlet label = placeTag(\"th\", null, row);\n\n\t\tif (markers.show) {\n\t\t\tlet indic = placeDiv(LEGEND_MARKER, label);\n\n\t\t\tif (i > 0) {\n\t\t\t\tlet width  = markers.width(self, i);\n\n\t\t\t\tif (width)\n\t\t\t\t\tindic.style.border = width + \"px \" + markers.dash(self, i) + \" \" + markers.stroke(self, i);\n\n\t\t\t\tindic.style.background = markers.fill(self, i);\n\t\t\t}\n\t\t}\n\n\t\tlet text = placeDiv(LEGEND_LABEL, label);\n\t\ttext.textContent = s.label;\n\n\t\tif (i > 0) {\n\t\t\tif (!markers.show)\n\t\t\t\ttext.style.color = s.width > 0 ? markers.stroke(self, i) : markers.fill(self, i);\n\n\t\t\tonMouse(\"click\", label, e => {\n\t\t\t\tif (cursor._lock)\n\t\t\t\t\treturn;\n\n\t\t\t\tlet seriesIdx = series.indexOf(s);\n\n\t\t\t\tif ((e.ctrlKey || e.metaKey) != legend.isolate) {\n\t\t\t\t\t// if any other series is shown, isolate this one. else show all\n\t\t\t\t\tlet isolate = series.some((s, i) => i > 0 && i != seriesIdx && s.show);\n\n\t\t\t\t\tseries.forEach((s, i) => {\n\t\t\t\t\t\ti > 0 && setSeries(i, isolate ? (i == seriesIdx ? son : soff) : son, true, syncOpts.setSeries);\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t\tsetSeries(seriesIdx, {show: !s.show}, true, syncOpts.setSeries);\n\t\t\t});\n\n\t\t\tif (cursorFocus) {\n\t\t\t\tonMouse(mouseenter, label, e => {\n\t\t\t\t\tif (cursor._lock)\n\t\t\t\t\t\treturn;\n\n\t\t\t\t\tsetSeries(series.indexOf(s), FOCUS_TRUE, true, syncOpts.setSeries);\n\t\t\t\t});\n\t\t\t}\n\t\t}\n\n\t\tfor (var key in legendCols) {\n\t\t\tlet v = placeTag(\"td\", LEGEND_VALUE, row);\n\t\t\tv.textContent = \"--\";\n\t\t\tcells.push(v);\n\t\t}\n\n\t\treturn [row, cells];\n\t}\n\n\tconst mouseListeners = new Map();\n\n\tfunction onMouse(ev, targ, fn) {\n\t\tconst targListeners = mouseListeners.get(targ) || {};\n\t\tconst listener = cursor.bind[ev](self, targ, fn);\n\n\t\tif (listener) {\n\t\t\ton(ev, targ, targListeners[ev] = listener);\n\t\t\tmouseListeners.set(targ, targListeners);\n\t\t}\n\t}\n\n\tfunction offMouse(ev, targ, fn) {\n\t\tconst targListeners = mouseListeners.get(targ) || {};\n\n\t\tfor (let k in targListeners) {\n\t\t\tif (ev == null || k == ev) {\n\t\t\t\toff(k, targ, targListeners[k]);\n\t\t\t\tdelete targListeners[k];\n\t\t\t}\n\t\t}\n\n\t\tif (ev == null)\n\t\t\tmouseListeners.delete(targ);\n\t}\n\n\tlet fullWidCss = 0;\n\tlet fullHgtCss = 0;\n\n\tlet plotWidCss = 0;\n\tlet plotHgtCss = 0;\n\n\t// plot margins to account for axes\n\tlet plotLftCss = 0;\n\tlet plotTopCss = 0;\n\n\tlet plotLft = 0;\n\tlet plotTop = 0;\n\tlet plotWid = 0;\n\tlet plotHgt = 0;\n\n\tself.bbox = {};\n\n\tlet shouldSetScales = false;\n\tlet shouldSetSize = false;\n\tlet shouldConvergeSize = false;\n\tlet shouldSetCursor = false;\n\tlet shouldSetLegend = false;\n\n\tfunction _setSize(width, height, force) {\n\t\tif (force || (width != self.width || height != self.height))\n\t\t\tcalcSize(width, height);\n\n\t\tresetYSeries(false);\n\n\t\tshouldConvergeSize = true;\n\t\tshouldSetSize = true;\n\t\tshouldSetCursor = shouldSetLegend = cursor.left >= 0;\n\t\tcommit();\n\t}\n\n\tfunction calcSize(width, height) {\n\t//\tlog(\"calcSize()\", arguments);\n\n\t\tself.width  = fullWidCss = plotWidCss = width;\n\t\tself.height = fullHgtCss = plotHgtCss = height;\n\t\tplotLftCss  = plotTopCss = 0;\n\n\t\tcalcPlotRect();\n\t\tcalcAxesRects();\n\n\t\tlet bb = self.bbox;\n\n\t\tplotLft = bb.left   = incrRound(plotLftCss * pxRatio, 0.5);\n\t\tplotTop = bb.top    = incrRound(plotTopCss * pxRatio, 0.5);\n\t\tplotWid = bb.width  = incrRound(plotWidCss * pxRatio, 0.5);\n\t\tplotHgt = bb.height = incrRound(plotHgtCss * pxRatio, 0.5);\n\n\t//\tupdOriDims();\n\t}\n\n\t// ensures size calc convergence\n\tconst CYCLE_LIMIT = 3;\n\n\tfunction convergeSize() {\n\t\tlet converged = false;\n\n\t\tlet cycleNum = 0;\n\n\t\twhile (!converged) {\n\t\t\tcycleNum++;\n\n\t\t\tlet axesConverged = axesCalc(cycleNum);\n\t\t\tlet paddingConverged = paddingCalc(cycleNum);\n\n\t\t\tconverged = cycleNum == CYCLE_LIMIT || (axesConverged && paddingConverged);\n\n\t\t\tif (!converged) {\n\t\t\t\tcalcSize(self.width, self.height);\n\t\t\t\tshouldSetSize = true;\n\t\t\t}\n\t\t}\n\t}\n\n\tfunction setSize({width, height}) {\n\t\t_setSize(width, height);\n\t}\n\n\tself.setSize = setSize;\n\n\t// accumulate axis offsets, reduce canvas width\n\tfunction calcPlotRect() {\n\t\t// easements for edge labels\n\t\tlet hasTopAxis = false;\n\t\tlet hasBtmAxis = false;\n\t\tlet hasRgtAxis = false;\n\t\tlet hasLftAxis = false;\n\n\t\taxes.forEach((axis, i) => {\n\t\t\tif (axis.show && axis._show) {\n\t\t\t\tlet {side, _size} = axis;\n\t\t\t\tlet isVt = side % 2;\n\t\t\t\tlet labelSize = axis.label != null ? axis.labelSize : 0;\n\n\t\t\t\tlet fullSize = _size + labelSize;\n\n\t\t\t\tif (fullSize > 0) {\n\t\t\t\t\tif (isVt) {\n\t\t\t\t\t\tplotWidCss -= fullSize;\n\n\t\t\t\t\t\tif (side == 3) {\n\t\t\t\t\t\t\tplotLftCss += fullSize;\n\t\t\t\t\t\t\thasLftAxis = true;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\thasRgtAxis = true;\n\t\t\t\t\t}\n\t\t\t\t\telse {\n\t\t\t\t\t\tplotHgtCss -= fullSize;\n\n\t\t\t\t\t\tif (side == 0) {\n\t\t\t\t\t\t\tplotTopCss += fullSize;\n\t\t\t\t\t\t\thasTopAxis = true;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\thasBtmAxis = true;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t});\n\n\t\tsidesWithAxes[0] = hasTopAxis;\n\t\tsidesWithAxes[1] = hasRgtAxis;\n\t\tsidesWithAxes[2] = hasBtmAxis;\n\t\tsidesWithAxes[3] = hasLftAxis;\n\n\t\t// hz padding\n\t\tplotWidCss -= _padding[1] + _padding[3];\n\t\tplotLftCss += _padding[3];\n\n\t\t// vt padding\n\t\tplotHgtCss -= _padding[2] + _padding[0];\n\t\tplotTopCss += _padding[0];\n\t}\n\n\tfunction calcAxesRects() {\n\t\t// will accum +\n\t\tlet off1 = plotLftCss + plotWidCss;\n\t\tlet off2 = plotTopCss + plotHgtCss;\n\t\t// will accum -\n\t\tlet off3 = plotLftCss;\n\t\tlet off0 = plotTopCss;\n\n\t\tfunction incrOffset(side, size) {\n\t\t\tswitch (side) {\n\t\t\t\tcase 1: off1 += size; return off1 - size;\n\t\t\t\tcase 2: off2 += size; return off2 - size;\n\t\t\t\tcase 3: off3 -= size; return off3 + size;\n\t\t\t\tcase 0: off0 -= size; return off0 + size;\n\t\t\t}\n\t\t}\n\n\t\taxes.forEach((axis, i) => {\n\t\t\tif (axis.show && axis._show) {\n\t\t\t\tlet side = axis.side;\n\n\t\t\t\taxis._pos = incrOffset(side, axis._size);\n\n\t\t\t\tif (axis.label != null)\n\t\t\t\t\taxis._lpos = incrOffset(side, axis.labelSize);\n\t\t\t}\n\t\t});\n\t}\n\n\tconst cursor = (self.cursor = assign({}, cursorOpts, {drag: {y: mode == 2}}, opts.cursor));\n\n\t{\n\t\tcursor.idxs = activeIdxs;\n\n\t\tcursor._lock = false;\n\n\t\tlet points = cursor.points;\n\n\t\tpoints.show   = fnOrSelf(points.show);\n\t\tpoints.size   = fnOrSelf(points.size);\n\t\tpoints.stroke = fnOrSelf(points.stroke);\n\t\tpoints.width  = fnOrSelf(points.width);\n\t\tpoints.fill   = fnOrSelf(points.fill);\n\t}\n\n\tconst focus = self.focus = assign({}, opts.focus || {alpha: 0.3}, cursor.focus);\n\tconst cursorFocus = focus.prox >= 0;\n\n\t// series-intersection markers\n\tlet cursorPts = [null];\n\n\tfunction initCursorPt(s, si) {\n\t\tif (si > 0) {\n\t\t\tlet pt = cursor.points.show(self, si);\n\n\t\t\tif (pt) {\n\t\t\t\taddClass(pt, CURSOR_PT);\n\t\t\t\taddClass(pt, s.class);\n\t\t\t\telTrans(pt, -10, -10, plotWidCss, plotHgtCss);\n\t\t\t\tover.insertBefore(pt, cursorPts[si]);\n\n\t\t\t\treturn pt;\n\t\t\t}\n\t\t}\n\t}\n\n\tfunction initSeries(s, i) {\n\t\tif (mode == 1 || i > 0) {\n\t\t\tlet isTime = mode == 1 && scales[s.scale].time;\n\n\t\t\tlet sv = s.value;\n\t\t\ts.value = isTime ? (isStr(sv) ? timeSeriesVal(_tzDate, timeSeriesStamp(sv, _fmtDate)) : sv || _timeSeriesVal) : sv || numSeriesVal;\n\t\t\ts.label = s.label || (isTime ? timeSeriesLabel : numSeriesLabel);\n\t\t}\n\n\t\tif (i > 0) {\n\t\t\ts.width  = s.width == null ? 1 : s.width;\n\t\t\ts.paths  = s.paths || linearPath || retNull;\n\t\t\ts.fillTo = fnOrSelf(s.fillTo || seriesFillTo);\n\t\t\ts.pxAlign = +ifNull(s.pxAlign, pxAlign);\n\t\t\ts.pxRound = pxRoundGen(s.pxAlign);\n\n\t\t\ts.stroke = fnOrSelf(s.stroke || null);\n\t\t\ts.fill   = fnOrSelf(s.fill || null);\n\t\t\ts._stroke = s._fill = s._paths = s._focus = null;\n\n\t\t\tlet _ptDia = ptDia(s.width, 1);\n\t\t\tlet points = s.points = assign({}, {\n\t\t\t\tsize: _ptDia,\n\t\t\t\twidth: max(1, _ptDia * .2),\n\t\t\t\tstroke: s.stroke,\n\t\t\t\tspace: _ptDia * 2,\n\t\t\t\tpaths: pointsPath,\n\t\t\t\t_stroke: null,\n\t\t\t\t_fill: null,\n\t\t\t}, s.points);\n\t\t\tpoints.show   = fnOrSelf(points.show);\n\t\t\tpoints.filter = fnOrSelf(points.filter);\n\t\t\tpoints.fill   = fnOrSelf(points.fill);\n\t\t\tpoints.stroke = fnOrSelf(points.stroke);\n\t\t\tpoints.paths  = fnOrSelf(points.paths);\n\t\t\tpoints.pxAlign = s.pxAlign;\n\t\t}\n\n\t\tif (showLegend) {\n\t\t\tlet rowCells = initLegendRow(s, i);\n\t\t\tlegendRows.splice(i, 0, rowCells[0]);\n\t\t\tlegendCells.splice(i, 0, rowCells[1]);\n\t\t\tlegend.values.push(null);\t// NULL_LEGEND_VALS not yet avil here :(\n\t\t}\n\n\t\tif (cursor.show) {\n\t\t\tactiveIdxs.splice(i, 0, null);\n\n\t\t\tlet pt = initCursorPt(s, i);\n\t\t\tpt && cursorPts.splice(i, 0, pt);\n\t\t}\n\n\t\tfire(\"addSeries\", i);\n\t}\n\n\tfunction addSeries(opts, si) {\n\t\tsi = si == null ? series.length : si;\n\n\t\topts = setDefault(opts, si, xSeriesOpts, ySeriesOpts);\n\t\tseries.splice(si, 0, opts);\n\t\tinitSeries(series[si], si);\n\t}\n\n\tself.addSeries = addSeries;\n\n\tfunction delSeries(i) {\n\t\tseries.splice(i, 1);\n\n\t\tif (showLegend) {\n\t\t\tlegend.values.splice(i, 1);\n\n\t\t\tlegendCells.splice(i, 1);\n\t\t\tlet tr = legendRows.splice(i, 1)[0];\n\t\t\toffMouse(null, tr.firstChild);\n\t\t\ttr.remove();\n\t\t}\n\n\t\tif (cursor.show) {\n\t\t\tactiveIdxs.splice(i, 1);\n\n\t\t\tcursorPts.length > 1 && cursorPts.splice(i, 1)[0].remove();\n\t\t}\n\n\t\t// TODO: de-init no-longer-needed scales?\n\n\t\tfire(\"delSeries\", i);\n\t}\n\n\tself.delSeries = delSeries;\n\n\tconst sidesWithAxes = [false, false, false, false];\n\n\tfunction initAxis(axis, i) {\n\t\taxis._show = axis.show;\n\n\t\tif (axis.show) {\n\t\t\tlet isVt = axis.side % 2;\n\n\t\t\tlet sc = scales[axis.scale];\n\n\t\t\t// this can occur if all series specify non-default scales\n\t\t\tif (sc == null) {\n\t\t\t\taxis.scale = isVt ? series[1].scale : xScaleKey;\n\t\t\t\tsc = scales[axis.scale];\n\t\t\t}\n\n\t\t\t// also set defaults for incrs & values based on axis distr\n\t\t\tlet isTime = sc.time;\n\n\t\t\taxis.size   = fnOrSelf(axis.size);\n\t\t\taxis.space  = fnOrSelf(axis.space);\n\t\t\taxis.rotate = fnOrSelf(axis.rotate);\n\t\t\taxis.incrs  = fnOrSelf(axis.incrs  || (          sc.distr == 2 ? wholeIncrs : (isTime ? (ms == 1 ? timeIncrsMs : timeIncrsS) : numIncrs)));\n\t\t\taxis.splits = fnOrSelf(axis.splits || (isTime && sc.distr == 1 ? _timeAxisSplits : sc.distr == 3 ? logAxisSplits : sc.distr == 4 ? asinhAxisSplits : numAxisSplits));\n\n\t\t\taxis.stroke        = fnOrSelf(axis.stroke);\n\t\t\taxis.grid.stroke   = fnOrSelf(axis.grid.stroke);\n\t\t\taxis.ticks.stroke  = fnOrSelf(axis.ticks.stroke);\n\t\t\taxis.border.stroke = fnOrSelf(axis.border.stroke);\n\n\t\t\tlet av = axis.values;\n\n\t\t\taxis.values = (\n\t\t\t\t// static array of tick values\n\t\t\t\tisArr(av) && !isArr(av[0]) ? fnOrSelf(av) :\n\t\t\t\t// temporal\n\t\t\t\tisTime ? (\n\t\t\t\t\t// config array of fmtDate string tpls\n\t\t\t\t\tisArr(av) ?\n\t\t\t\t\t\ttimeAxisVals(_tzDate, timeAxisStamps(av, _fmtDate)) :\n\t\t\t\t\t// fmtDate string tpl\n\t\t\t\t\tisStr(av) ?\n\t\t\t\t\t\ttimeAxisVal(_tzDate, av) :\n\t\t\t\t\tav || _timeAxisVals\n\t\t\t\t) : av || numAxisVals\n\t\t\t);\n\n\t\t\taxis.filter = fnOrSelf(axis.filter || (          sc.distr >= 3 ? logAxisValsFilt : retArg1));\n\n\t\t\taxis.font      = pxRatioFont(axis.font);\n\t\t\taxis.labelFont = pxRatioFont(axis.labelFont);\n\n\t\t\taxis._size   = axis.size(self, null, i, 0);\n\n\t\t\taxis._space  =\n\t\t\taxis._rotate =\n\t\t\taxis._incrs  =\n\t\t\taxis._found  =\t// foundIncrSpace\n\t\t\taxis._splits =\n\t\t\taxis._values = null;\n\n\t\t\tif (axis._size > 0) {\n\t\t\t\tsidesWithAxes[i] = true;\n\t\t\t\taxis._el = placeDiv(AXIS, wrap);\n\t\t\t}\n\n\t\t\t// debug\n\t\t//\taxis._el.style.background = \"#\"  + Math.floor(Math.random()*16777215).toString(16) + '80';\n\t\t}\n\t}\n\n\tfunction autoPadSide(self, side, sidesWithAxes, cycleNum) {\n\t\tlet [hasTopAxis, hasRgtAxis, hasBtmAxis, hasLftAxis] = sidesWithAxes;\n\n\t\tlet ori = side % 2;\n\t\tlet size = 0;\n\n\t\tif (ori == 0 && (hasLftAxis || hasRgtAxis))\n\t\t\tsize = (side == 0 && !hasTopAxis || side == 2 && !hasBtmAxis ? round(xAxisOpts.size / 3) : 0);\n\t\tif (ori == 1 && (hasTopAxis || hasBtmAxis))\n\t\t\tsize = (side == 1 && !hasRgtAxis || side == 3 && !hasLftAxis ? round(yAxisOpts.size / 2) : 0);\n\n\t\treturn size;\n\t}\n\n\tconst padding = self.padding = (opts.padding || [autoPadSide,autoPadSide,autoPadSide,autoPadSide]).map(p => fnOrSelf(ifNull(p, autoPadSide)));\n\tconst _padding = self._padding = padding.map((p, i) => p(self, i, sidesWithAxes, 0));\n\n\tlet dataLen;\n\n\t// rendered data window\n\tlet i0 = null;\n\tlet i1 = null;\n\tconst idxs = mode == 1 ? series[0].idxs : null;\n\n\tlet data0 = null;\n\n\tlet viaAutoScaleX = false;\n\n\tfunction setData(_data, _resetScales) {\n\t\tdata = _data == null ? [] : copy(_data, fastIsObj);\n\n\t\tif (mode == 2) {\n\t\t\tdataLen = 0;\n\t\t\tfor (let i = 1; i < series.length; i++)\n\t\t\t\tdataLen += data[i][0].length;\n\t\t\tself.data = data = _data;\n\t\t}\n\t\telse {\n\t\t\tif (data[0] == null)\n\t\t\t\tdata[0] = [];\n\n\t\t\tself.data = data.slice();\n\n\t\t\tdata0 = data[0];\n\t\t\tdataLen = data0.length;\n\n\t\t\tif (xScaleDistr == 2) {\n\t\t\t\tdata[0] = Array(dataLen);\n\t\t\t\tfor (let i = 0; i < dataLen; i++)\n\t\t\t\t\tdata[0][i] = i;\n\t\t\t}\n\t\t}\n\n\t\tself._data = data;\n\n\t\tresetYSeries(true);\n\n\t\tfire(\"setData\");\n\n\t\t// forces x axis tick values to re-generate when neither x scale nor y scale changes\n\t\t// in ordinal mode, scale range is by index, so will not change if new data has same length, but tick values are from data\n\t\tif (xScaleDistr == 2) {\n\t\t\tshouldConvergeSize = true;\n\n\t\t\t/* or somewhat cheaper, and uglier:\n\t\t\tif (ready) {\n\t\t\t\t// logic extracted from axesCalc()\n\t\t\t\tlet i = 0;\n\t\t\t\tlet axis = axes[i];\n\t\t\t\tlet _splits = axis._splits.map(i => data0[i]);\n\t\t\t\tlet [_incr, _space] = axis._found;\n\t\t\t\tlet incr = data0[_splits[1]] - data0[_splits[0]];\n\t\t\t\taxis._values = axis.values(self, axis.filter(self, _splits, i, _space, incr), i, _space, incr);\n\t\t\t}\n\t\t\t*/\n\t\t}\n\n\t\tif (_resetScales !== false) {\n\t\t\tlet xsc = scaleX;\n\n\t\t\tif (xsc.auto(self, viaAutoScaleX))\n\t\t\t\tautoScaleX();\n\t\t\telse\n\t\t\t\t_setScale(xScaleKey, xsc.min, xsc.max);\n\n\t\t\tshouldSetCursor = cursor.left >= 0;\n\t\t\tshouldSetLegend = true;\n\t\t\tcommit();\n\t\t}\n\t}\n\n\tself.setData = setData;\n\n\tfunction autoScaleX() {\n\t\tviaAutoScaleX = true;\n\n\t\tlet _min, _max;\n\n\t\tif (mode == 1) {\n\t\t\tif (dataLen > 0) {\n\t\t\t\ti0 = idxs[0] = 0;\n\t\t\t\ti1 = idxs[1] = dataLen - 1;\n\n\t\t\t\t_min = data[0][i0];\n\t\t\t\t_max = data[0][i1];\n\n\t\t\t\tif (xScaleDistr == 2) {\n\t\t\t\t\t_min = i0;\n\t\t\t\t\t_max = i1;\n\t\t\t\t}\n\t\t\t\telse if (dataLen == 1) {\n\t\t\t\t\tif (xScaleDistr == 3)\n\t\t\t\t\t\t[_min, _max] = rangeLog(_min, _min, scaleX.log, false);\n\t\t\t\t\telse if (xScaleDistr == 4)\n\t\t\t\t\t\t[_min, _max] = rangeAsinh(_min, _min, scaleX.log, false);\n\t\t\t\t\telse if (scaleX.time)\n\t\t\t\t\t\t_max = _min + round(86400 / ms);\n\t\t\t\t\telse\n\t\t\t\t\t\t[_min, _max] = rangeNum(_min, _max, rangePad, true);\n\t\t\t\t}\n\t\t\t}\n\t\t\telse {\n\t\t\t\ti0 = idxs[0] = _min = null;\n\t\t\t\ti1 = idxs[1] = _max = null;\n\t\t\t}\n\t\t}\n\n\t\t_setScale(xScaleKey, _min, _max);\n\t}\n\n\tlet ctxStroke, ctxFill, ctxWidth, ctxDash, ctxJoin, ctxCap, ctxFont, ctxAlign, ctxBaseline;\n\tlet ctxAlpha;\n\n\tfunction setCtxStyle(stroke = transparent, width, dash = EMPTY_ARR, cap = \"butt\", fill = transparent, join = \"round\") {\n\t\tif (stroke != ctxStroke)\n\t\t\tctx.strokeStyle = ctxStroke = stroke;\n\t\tif (fill != ctxFill)\n\t\t\tctx.fillStyle = ctxFill = fill;\n\t\tif (width != ctxWidth)\n\t\t\tctx.lineWidth = ctxWidth = width;\n\t\tif (join != ctxJoin)\n\t\t\tctx.lineJoin = ctxJoin = join;\n\t\tif (cap != ctxCap)\n\t\t\tctx.lineCap = ctxCap = cap; // (‿|‿)\n\t\tif (dash != ctxDash)\n\t\t\tctx.setLineDash(ctxDash = dash);\n\t}\n\n\tfunction setFontStyle(font, fill, align, baseline) {\n\t\tif (fill != ctxFill)\n\t\t\tctx.fillStyle = ctxFill = fill;\n\t\tif (font != ctxFont)\n\t\t\tctx.font = ctxFont = font;\n\t\tif (align != ctxAlign)\n\t\t\tctx.textAlign = ctxAlign = align;\n\t\tif (baseline != ctxBaseline)\n\t\t\tctx.textBaseline = ctxBaseline = baseline;\n\t}\n\n\tfunction accScale(wsc, psc, facet, data, sorted = 0) {\n\t\tif (data.length > 0 && wsc.auto(self, viaAutoScaleX) && (psc == null || psc.min == null)) {\n\t\t\tlet _i0 = ifNull(i0, 0);\n\t\t\tlet _i1 = ifNull(i1, data.length - 1);\n\n\t\t\t// only run getMinMax() for invalidated series data, else reuse\n\t\t\tlet minMax = facet.min == null ? (wsc.distr == 3 ? getMinMaxLog(data, _i0, _i1) : getMinMax(data, _i0, _i1, sorted)) : [facet.min, facet.max];\n\n\t\t\t// initial min/max\n\t\t\twsc.min = min(wsc.min, facet.min = minMax[0]);\n\t\t\twsc.max = max(wsc.max, facet.max = minMax[1]);\n\t\t}\n\t}\n\n\tfunction setScales() {\n\t//\tlog(\"setScales()\", arguments);\n\n\t\t// wip scales\n\t\tlet wipScales = copy(scales, fastIsObj);\n\n\t\tfor (let k in wipScales) {\n\t\t\tlet wsc = wipScales[k];\n\t\t\tlet psc = pendScales[k];\n\n\t\t\tif (psc != null && psc.min != null) {\n\t\t\t\tassign(wsc, psc);\n\n\t\t\t\t// explicitly setting the x-scale invalidates everything (acts as redraw)\n\t\t\t\tif (k == xScaleKey)\n\t\t\t\t\tresetYSeries(true);\n\t\t\t}\n\t\t\telse if (k != xScaleKey || mode == 2) {\n\t\t\t\tif (dataLen == 0 && wsc.from == null) {\n\t\t\t\t\tlet minMax = wsc.range(self, null, null, k);\n\t\t\t\t\twsc.min = minMax[0];\n\t\t\t\t\twsc.max = minMax[1];\n\t\t\t\t}\n\t\t\t\telse {\n\t\t\t\t\twsc.min = inf;\n\t\t\t\t\twsc.max = -inf;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tif (dataLen > 0) {\n\t\t\t// pre-range y-scales from y series' data values\n\t\t\tseries.forEach((s, i) => {\n\t\t\t\tif (mode == 1) {\n\t\t\t\t\tlet k = s.scale;\n\t\t\t\t\tlet wsc = wipScales[k];\n\t\t\t\t\tlet psc = pendScales[k];\n\n\t\t\t\t\tif (i == 0) {\n\t\t\t\t\t\tlet minMax = wsc.range(self, wsc.min, wsc.max, k);\n\n\t\t\t\t\t\twsc.min = minMax[0];\n\t\t\t\t\t\twsc.max = minMax[1];\n\n\t\t\t\t\t\ti0 = closestIdx(wsc.min, data[0]);\n\t\t\t\t\t\ti1 = closestIdx(wsc.max, data[0]);\n\n\t\t\t\t\t\t// closest indices can be outside of view\n\t\t\t\t\t\tif (data[0][i0] < wsc.min)\n\t\t\t\t\t\t\ti0++;\n\t\t\t\t\t\tif (data[0][i1] > wsc.max)\n\t\t\t\t\t\t\ti1--;\n\n\t\t\t\t\t\ts.min = data0[i0];\n\t\t\t\t\t\ts.max = data0[i1];\n\t\t\t\t\t}\n\t\t\t\t\telse if (s.show && s.auto)\n\t\t\t\t\t\taccScale(wsc, psc, s, data[i], s.sorted);\n\n\t\t\t\t\ts.idxs[0] = i0;\n\t\t\t\t\ts.idxs[1] = i1;\n\t\t\t\t}\n\t\t\t\telse {\n\t\t\t\t\tif (i > 0) {\n\t\t\t\t\t\tif (s.show && s.auto) {\n\t\t\t\t\t\t\t// TODO: only handles, assumes and requires facets[0] / 'x' scale, and facets[1] / 'y' scale\n\t\t\t\t\t\t\tlet [ xFacet, yFacet ] = s.facets;\n\t\t\t\t\t\t\tlet xScaleKey = xFacet.scale;\n\t\t\t\t\t\t\tlet yScaleKey = yFacet.scale;\n\t\t\t\t\t\t\tlet [ xData, yData ] = data[i];\n\n\t\t\t\t\t\t\taccScale(wipScales[xScaleKey], pendScales[xScaleKey], xFacet, xData, xFacet.sorted);\n\t\t\t\t\t\t\taccScale(wipScales[yScaleKey], pendScales[yScaleKey], yFacet, yData, yFacet.sorted);\n\n\t\t\t\t\t\t\t// temp\n\t\t\t\t\t\t\ts.min = yFacet.min;\n\t\t\t\t\t\t\ts.max = yFacet.max;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t});\n\n\t\t\t// range independent scales\n\t\t\tfor (let k in wipScales) {\n\t\t\t\tlet wsc = wipScales[k];\n\t\t\t\tlet psc = pendScales[k];\n\n\t\t\t\tif (wsc.from == null && (psc == null || psc.min == null)) {\n\t\t\t\t\tlet minMax = wsc.range(\n\t\t\t\t\t\tself,\n\t\t\t\t\t\twsc.min ==  inf ? null : wsc.min,\n\t\t\t\t\t\twsc.max == -inf ? null : wsc.max,\n\t\t\t\t\t\tk\n\t\t\t\t\t);\n\t\t\t\t\twsc.min = minMax[0];\n\t\t\t\t\twsc.max = minMax[1];\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// range dependent scales\n\t\tfor (let k in wipScales) {\n\t\t\tlet wsc = wipScales[k];\n\n\t\t\tif (wsc.from != null) {\n\t\t\t\tlet base = wipScales[wsc.from];\n\n\t\t\t\tif (base.min == null)\n\t\t\t\t\twsc.min = wsc.max = null;\n\t\t\t\telse {\n\t\t\t\t\tlet minMax = wsc.range(self, base.min, base.max, k);\n\t\t\t\t\twsc.min = minMax[0];\n\t\t\t\t\twsc.max = minMax[1];\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tlet changed = {};\n\t\tlet anyChanged = false;\n\n\t\tfor (let k in wipScales) {\n\t\t\tlet wsc = wipScales[k];\n\t\t\tlet sc = scales[k];\n\n\t\t\tif (sc.min != wsc.min || sc.max != wsc.max) {\n\t\t\t\tsc.min = wsc.min;\n\t\t\t\tsc.max = wsc.max;\n\n\t\t\t\tlet distr = sc.distr;\n\n\t\t\t\tsc._min = distr == 3 ? log10(sc.min) : distr == 4 ? asinh(sc.min, sc.asinh) : sc.min;\n\t\t\t\tsc._max = distr == 3 ? log10(sc.max) : distr == 4 ? asinh(sc.max, sc.asinh) : sc.max;\n\n\t\t\t\tchanged[k] = anyChanged = true;\n\t\t\t}\n\t\t}\n\n\t\tif (anyChanged) {\n\t\t\t// invalidate paths of all series on changed scales\n\t\t\tseries.forEach((s, i) => {\n\t\t\t\tif (mode == 2) {\n\t\t\t\t\tif (i > 0 && changed.y)\n\t\t\t\t\t\ts._paths = null;\n\t\t\t\t}\n\t\t\t\telse {\n\t\t\t\t\tif (changed[s.scale])\n\t\t\t\t\t\ts._paths = null;\n\t\t\t\t}\n\t\t\t});\n\n\t\t\tfor (let k in changed) {\n\t\t\t\tshouldConvergeSize = true;\n\t\t\t\tfire(\"setScale\", k);\n\t\t\t}\n\n\t\t\tif (cursor.show)\n\t\t\t\tshouldSetCursor = shouldSetLegend = cursor.left >= 0;\n\t\t}\n\n\t\tfor (let k in pendScales)\n\t\t\tpendScales[k] = null;\n\t}\n\n\t// grabs the nearest indices with y data outside of x-scale limits\n\tfunction getOuterIdxs(ydata) {\n\t\tlet _i0 = clamp(i0 - 1, 0, dataLen - 1);\n\t\tlet _i1 = clamp(i1 + 1, 0, dataLen - 1);\n\n\t\twhile (ydata[_i0] == null && _i0 > 0)\n\t\t\t_i0--;\n\n\t\twhile (ydata[_i1] == null && _i1 < dataLen - 1)\n\t\t\t_i1++;\n\n\t\treturn [_i0, _i1];\n\t}\n\n\tfunction drawSeries() {\n\t\tif (dataLen > 0) {\n\t\t\tseries.forEach((s, i) => {\n\t\t\t\tif (i > 0 && s.show && s._paths == null) {\n\t\t\t\t\tlet _idxs = getOuterIdxs(data[i]);\n\t\t\t\t\ts._paths = s.paths(self, i, _idxs[0], _idxs[1]);\n\t\t\t\t}\n\t\t\t});\n\n\t\t\tseries.forEach((s, i) => {\n\t\t\t\tif (i > 0 && s.show) {\n\t\t\t\t\tif (ctxAlpha != s.alpha)\n\t\t\t\t\t\tctx.globalAlpha = ctxAlpha = s.alpha;\n\n\t\t\t\t\t{\n\t\t\t\t\t\tcacheStrokeFill(i, false);\n\t\t\t\t\t\ts._paths && drawPath(i, false);\n\t\t\t\t\t}\n\n\t\t\t\t\t{\n\t\t\t\t\t\tcacheStrokeFill(i, true);\n\n\t\t\t\t\t\tlet show = s.points.show(self, i, i0, i1);\n\t\t\t\t\t\tlet idxs = s.points.filter(self, i, show, s._paths ? s._paths.gaps : null);\n\n\t\t\t\t\t\tif (show || idxs) {\n\t\t\t\t\t\t\ts.points._paths = s.points.paths(self, i, i0, i1, idxs);\n\t\t\t\t\t\t\tdrawPath(i, true);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tif (ctxAlpha != 1)\n\t\t\t\t\t\tctx.globalAlpha = ctxAlpha = 1;\n\n\t\t\t\t\tfire(\"drawSeries\", i);\n\t\t\t\t}\n\t\t\t});\n\t\t}\n\t}\n\n\tfunction cacheStrokeFill(si, _points) {\n\t\tlet s = _points ? series[si].points : series[si];\n\n\t\ts._stroke = s.stroke(self, si);\n\t\ts._fill   = s.fill(self, si);\n\t}\n\n\tfunction drawPath(si, _points) {\n\t\tlet s = _points ? series[si].points : series[si];\n\n\t\tlet strokeStyle = s._stroke;\n\t\tlet fillStyle   = s._fill;\n\n\t\tlet { stroke, fill, clip: gapsClip, flags } = s._paths;\n\t\tlet boundsClip = null;\n\t\tlet width = roundDec(s.width * pxRatio, 3);\n\t\tlet offset = (width % 2) / 2;\n\n\t\tif (_points && fillStyle == null)\n\t\t\tfillStyle = width > 0 ? \"#fff\" : strokeStyle;\n\n\t\tlet _pxAlign = s.pxAlign == 1;\n\n\t\t_pxAlign && ctx.translate(offset, offset);\n\n\t\tif (!_points) {\n\t\t\tlet lft = plotLft,\n\t\t\t\ttop = plotTop,\n\t\t\t\twid = plotWid,\n\t\t\t\thgt = plotHgt;\n\n\t\t\tlet halfWid = width * pxRatio / 2;\n\n\t\t\tif (s.min == 0)\n\t\t\t\thgt += halfWid;\n\n\t\t\tif (s.max == 0) {\n\t\t\t\ttop -= halfWid;\n\t\t\t\thgt += halfWid;\n\t\t\t}\n\n\t\t\tboundsClip = new Path2D();\n\t\t\tboundsClip.rect(lft, top, wid, hgt);\n\t\t}\n\n\t\t// the points pathbuilder's gapsClip is its boundsClip, since points dont need gaps clipping, and bounds depend on point size\n\t\tif (_points)\n\t\t\tstrokeFill(strokeStyle, width, s.dash, s.cap, fillStyle, stroke, fill, flags, gapsClip);\n\t\telse\n\t\t\tfillStroke(si, strokeStyle, width, s.dash, s.cap, fillStyle, stroke, fill, flags, boundsClip, gapsClip);\n\n\t\t_pxAlign && ctx.translate(-offset, -offset);\n\t}\n\n\tfunction fillStroke(si, strokeStyle, lineWidth, lineDash, lineCap, fillStyle, strokePath, fillPath, flags, boundsClip, gapsClip) {\n\t\tlet didStrokeFill = false;\n\n\t\t// for all bands where this series is the top edge, create upwards clips using the bottom edges\n\t\t// and apply clips + fill with band fill or dfltFill\n\t\tbands.forEach((b, bi) => {\n\t\t\t// isUpperEdge?\n\t\t\tif (b.series[0] == si) {\n\t\t\t\tlet lowerEdge = series[b.series[1]];\n\t\t\t\tlet lowerData = data[b.series[1]];\n\n\t\t\t\tlet bandClip = (lowerEdge._paths || EMPTY_OBJ).band;\n\n\t\t\t\tif (isArr(bandClip))\n\t\t\t\t\tbandClip = b.dir == 1 ? bandClip[0] : bandClip[1];\n\n\t\t\t\tlet gapsClip2;\n\n\t\t\t\tlet _fillStyle = null;\n\n\t\t\t\t// hasLowerEdge?\n\t\t\t\tif (lowerEdge.show && bandClip && hasData(lowerData, i0, i1)) {\n\t\t\t\t\t_fillStyle = b.fill(self, bi) || fillStyle;\n\t\t\t\t\tgapsClip2 = lowerEdge._paths.clip;\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t\tbandClip = null;\n\n\t\t\t\tstrokeFill(strokeStyle, lineWidth, lineDash, lineCap, _fillStyle, strokePath, fillPath, flags, boundsClip, gapsClip, gapsClip2, bandClip);\n\n\t\t\t\tdidStrokeFill = true;\n\t\t\t}\n\t\t});\n\n\t\tif (!didStrokeFill)\n\t\t\tstrokeFill(strokeStyle, lineWidth, lineDash, lineCap, fillStyle, strokePath, fillPath, flags, boundsClip, gapsClip);\n\t}\n\n\tconst CLIP_FILL_STROKE = BAND_CLIP_FILL | BAND_CLIP_STROKE;\n\n\tfunction strokeFill(strokeStyle, lineWidth, lineDash, lineCap, fillStyle, strokePath, fillPath, flags, boundsClip, gapsClip, gapsClip2, bandClip) {\n\t\tsetCtxStyle(strokeStyle, lineWidth, lineDash, lineCap, fillStyle);\n\n\t\tif (boundsClip || gapsClip || bandClip) {\n\t\t\tctx.save();\n\t\t\tboundsClip && ctx.clip(boundsClip);\n\t\t\tgapsClip && ctx.clip(gapsClip);\n\t\t}\n\n\t\tif (bandClip) {\n\t\t\tif ((flags & CLIP_FILL_STROKE) == CLIP_FILL_STROKE) {\n\t\t\t\tctx.clip(bandClip);\n\t\t\t\tgapsClip2 && ctx.clip(gapsClip2);\n\t\t\t\tdoFill(fillStyle, fillPath);\n\t\t\t\tdoStroke(strokeStyle, strokePath, lineWidth);\n\t\t\t}\n\t\t\telse if (flags & BAND_CLIP_STROKE) {\n\t\t\t\tdoFill(fillStyle, fillPath);\n\t\t\t\tctx.clip(bandClip);\n\t\t\t\tdoStroke(strokeStyle, strokePath, lineWidth);\n\t\t\t}\n\t\t\telse if (flags & BAND_CLIP_FILL) {\n\t\t\t\tctx.save();\n\t\t\t\tctx.clip(bandClip);\n\t\t\t\tgapsClip2 && ctx.clip(gapsClip2);\n\t\t\t\tdoFill(fillStyle, fillPath);\n\t\t\t\tctx.restore();\n\t\t\t\tdoStroke(strokeStyle, strokePath, lineWidth);\n\t\t\t}\n\t\t}\n\t\telse {\n\t\t\tdoFill(fillStyle, fillPath);\n\t\t\tdoStroke(strokeStyle, strokePath, lineWidth);\n\t\t}\n\n\t\tif (boundsClip || gapsClip || bandClip)\n\t\t\tctx.restore();\n\t}\n\n\tfunction doStroke(strokeStyle, strokePath, lineWidth) {\n\t\tif (lineWidth > 0) {\n\t\t\tif (strokePath instanceof Map) {\n\t\t\t\tstrokePath.forEach((strokePath, strokeStyle) => {\n\t\t\t\t\tctx.strokeStyle = ctxStroke = strokeStyle;\n\t\t\t\t\tctx.stroke(strokePath);\n\t\t\t\t});\n\t\t\t}\n\t\t\telse\n\t\t\t\tstrokePath != null && strokeStyle && ctx.stroke(strokePath);\n\t\t}\n\t}\n\n\tfunction doFill(fillStyle, fillPath) {\n\t\tif (fillPath instanceof Map) {\n\t\t\tfillPath.forEach((fillPath, fillStyle) => {\n\t\t\t\tctx.fillStyle = ctxFill = fillStyle;\n\t\t\t\tctx.fill(fillPath);\n\t\t\t});\n\t\t}\n\t\telse\n\t\t\tfillPath != null && fillStyle && ctx.fill(fillPath);\n\t}\n\n\tfunction getIncrSpace(axisIdx, min, max, fullDim) {\n\t\tlet axis = axes[axisIdx];\n\n\t\tlet incrSpace;\n\n\t\tif (fullDim <= 0)\n\t\t\tincrSpace = [0, 0];\n\t\telse {\n\t\t\tlet minSpace = axis._space = axis.space(self, axisIdx, min, max, fullDim);\n\t\t\tlet incrs    = axis._incrs = axis.incrs(self, axisIdx, min, max, fullDim, minSpace);\n\t\t\tincrSpace    = findIncr(min, max, incrs, fullDim, minSpace);\n\t\t}\n\n\t\treturn (axis._found = incrSpace);\n\t}\n\n\tfunction drawOrthoLines(offs, filts, ori, side, pos0, len, width, stroke, dash, cap) {\n\t\tlet offset = (width % 2) / 2;\n\n\t\tpxAlign == 1 && ctx.translate(offset, offset);\n\n\t\tsetCtxStyle(stroke, width, dash, cap, stroke);\n\n\t\tctx.beginPath();\n\n\t\tlet x0, y0, x1, y1, pos1 = pos0 + (side == 0 || side == 3 ? -len : len);\n\n\t\tif (ori == 0) {\n\t\t\ty0 = pos0;\n\t\t\ty1 = pos1;\n\t\t}\n\t\telse {\n\t\t\tx0 = pos0;\n\t\t\tx1 = pos1;\n\t\t}\n\n\t\tfor (let i = 0; i < offs.length; i++) {\n\t\t\tif (filts[i] != null) {\n\t\t\t\tif (ori == 0)\n\t\t\t\t\tx0 = x1 = offs[i];\n\t\t\t\telse\n\t\t\t\t\ty0 = y1 = offs[i];\n\n\t\t\t\tctx.moveTo(x0, y0);\n\t\t\t\tctx.lineTo(x1, y1);\n\t\t\t}\n\t\t}\n\n\t\tctx.stroke();\n\n\t\tpxAlign == 1 && ctx.translate(-offset, -offset);\n\t}\n\n\tfunction axesCalc(cycleNum) {\n\t//\tlog(\"axesCalc()\", arguments);\n\n\t\tlet converged = true;\n\n\t\taxes.forEach((axis, i) => {\n\t\t\tif (!axis.show)\n\t\t\t\treturn;\n\n\t\t\tlet scale = scales[axis.scale];\n\n\t\t\tif (scale.min == null) {\n\t\t\t\tif (axis._show) {\n\t\t\t\t\tconverged = false;\n\t\t\t\t\taxis._show = false;\n\t\t\t\t\tresetYSeries(false);\n\t\t\t\t}\n\t\t\t\treturn;\n\t\t\t}\n\t\t\telse {\n\t\t\t\tif (!axis._show) {\n\t\t\t\t\tconverged = false;\n\t\t\t\t\taxis._show = true;\n\t\t\t\t\tresetYSeries(false);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tlet side = axis.side;\n\t\t\tlet ori = side % 2;\n\n\t\t\tlet {min, max} = scale;\t\t// \t\t// should this toggle them ._show = false\n\n\t\t\tlet [_incr, _space] = getIncrSpace(i, min, max, ori == 0 ? plotWidCss : plotHgtCss);\n\n\t\t\tif (_space == 0)\n\t\t\t\treturn;\n\n\t\t\t// if we're using index positions, force first tick to match passed index\n\t\t\tlet forceMin = scale.distr == 2;\n\n\t\t\tlet _splits = axis._splits = axis.splits(self, i, min, max, _incr, _space, forceMin);\n\n\t\t\t// tick labels\n\t\t\t// BOO this assumes a specific data/series\n\t\t\tlet splits = scale.distr == 2 ? _splits.map(i => data0[i]) : _splits;\n\t\t\tlet incr   = scale.distr == 2 ? data0[_splits[1]] - data0[_splits[0]] : _incr;\n\n\t\t\tlet values = axis._values = axis.values(self, axis.filter(self, splits, i, _space, incr), i, _space, incr);\n\n\t\t\t// rotating of labels only supported on bottom x axis\n\t\t\taxis._rotate = side == 2 ? axis.rotate(self, values, i, _space) : 0;\n\n\t\t\tlet oldSize = axis._size;\n\n\t\t\taxis._size = ceil(axis.size(self, values, i, cycleNum));\n\n\t\t\tif (oldSize != null && axis._size != oldSize)\t\t\t// ready && ?\n\t\t\t\tconverged = false;\n\t\t});\n\n\t\treturn converged;\n\t}\n\n\tfunction paddingCalc(cycleNum) {\n\t\tlet converged = true;\n\n\t\tpadding.forEach((p, i) => {\n\t\t\tlet _p = p(self, i, sidesWithAxes, cycleNum);\n\n\t\t\tif (_p != _padding[i])\n\t\t\t\tconverged = false;\n\n\t\t\t_padding[i] = _p;\n\t\t});\n\n\t\treturn converged;\n\t}\n\n\tfunction drawAxesGrid() {\n\t\tfor (let i = 0; i < axes.length; i++) {\n\t\t\tlet axis = axes[i];\n\n\t\t\tif (!axis.show || !axis._show)\n\t\t\t\tcontinue;\n\n\t\t\tlet side = axis.side;\n\t\t\tlet ori = side % 2;\n\n\t\t\tlet x, y;\n\n\t\t\tlet fillStyle = axis.stroke(self, i);\n\n\t\t\tlet shiftDir = side == 0 || side == 3 ? -1 : 1;\n\n\t\t\t// axis label\n\t\t\tif (axis.label) {\n\t\t\t\tlet shiftAmt = axis.labelGap * shiftDir;\n\t\t\t\tlet baseLpos = round((axis._lpos + shiftAmt) * pxRatio);\n\n\t\t\t\tsetFontStyle(axis.labelFont[0], fillStyle, \"center\", side == 2 ? TOP : BOTTOM);\n\n\t\t\t\tctx.save();\n\n\t\t\t\tif (ori == 1) {\n\t\t\t\t\tx = y = 0;\n\n\t\t\t\t\tctx.translate(\n\t\t\t\t\t\tbaseLpos,\n\t\t\t\t\t\tround(plotTop + plotHgt / 2),\n\t\t\t\t\t);\n\t\t\t\t\tctx.rotate((side == 3 ? -PI : PI) / 2);\n\n\t\t\t\t}\n\t\t\t\telse {\n\t\t\t\t\tx = round(plotLft + plotWid / 2);\n\t\t\t\t\ty = baseLpos;\n\t\t\t\t}\n\n\t\t\t\tctx.fillText(axis.label, x, y);\n\n\t\t\t\tctx.restore();\n\t\t\t}\n\n\t\t\tlet [_incr, _space] = axis._found;\n\n\t\t\tif (_space == 0)\n\t\t\t\tcontinue;\n\n\t\t\tlet scale = scales[axis.scale];\n\n\t\t\tlet plotDim = ori == 0 ? plotWid : plotHgt;\n\t\t\tlet plotOff = ori == 0 ? plotLft : plotTop;\n\n\t\t\tlet axisGap = round(axis.gap * pxRatio);\n\n\t\t\tlet _splits = axis._splits;\n\n\t\t\t// tick labels\n\t\t\t// BOO this assumes a specific data/series\n\t\t\tlet splits = scale.distr == 2 ? _splits.map(i => data0[i]) : _splits;\n\t\t\tlet incr   = scale.distr == 2 ? data0[_splits[1]] - data0[_splits[0]] : _incr;\n\n\t\t\tlet ticks = axis.ticks;\n\t\t\tlet border = axis.border;\n\t\t\tlet tickSize = ticks.show ? round(ticks.size * pxRatio) : 0;\n\n\t\t\t// rotating of labels only supported on bottom x axis\n\t\t\tlet angle = axis._rotate * -PI/180;\n\n\t\t\tlet basePos  = pxRound(axis._pos * pxRatio);\n\t\t\tlet shiftAmt = (tickSize + axisGap) * shiftDir;\n\t\t\tlet finalPos = basePos + shiftAmt;\n\t\t\t    y        = ori == 0 ? finalPos : 0;\n\t\t\t    x        = ori == 1 ? finalPos : 0;\n\n\t\t\tlet font         = axis.font[0];\n\t\t\tlet textAlign    = axis.align == 1 ? LEFT :\n\t\t\t                   axis.align == 2 ? RIGHT :\n\t\t\t                   angle > 0 ? LEFT :\n\t\t\t                   angle < 0 ? RIGHT :\n\t\t\t                   ori == 0 ? \"center\" : side == 3 ? RIGHT : LEFT;\n\t\t\tlet textBaseline = angle ||\n\t\t\t                   ori == 1 ? \"middle\" : side == 2 ? TOP   : BOTTOM;\n\n\t\t\tsetFontStyle(font, fillStyle, textAlign, textBaseline);\n\n\t\t\tlet lineHeight = axis.font[1] * lineMult;\n\n\t\t\tlet canOffs = _splits.map(val => pxRound(getPos(val, scale, plotDim, plotOff)));\n\n\t\t\tlet _values = axis._values;\n\n\t\t\tfor (let i = 0; i < _values.length; i++) {\n\t\t\t\tlet val = _values[i];\n\n\t\t\t\tif (val != null) {\n\t\t\t\t\tif (ori == 0)\n\t\t\t\t\t\tx = canOffs[i];\n\t\t\t\t\telse\n\t\t\t\t\t\ty = canOffs[i];\n\n\t\t\t\t\tval = \"\" + val;\n\n\t\t\t\t\tlet _parts = val.indexOf(\"\\n\") == -1 ? [val] : val.split(/\\n/gm);\n\n\t\t\t\t\tfor (let j = 0; j < _parts.length; j++) {\n\t\t\t\t\t\tlet text = _parts[j];\n\n\t\t\t\t\t\tif (angle) {\n\t\t\t\t\t\t\tctx.save();\n\t\t\t\t\t\t\tctx.translate(x, y + j * lineHeight); // can this be replaced with position math?\n\t\t\t\t\t\t\tctx.rotate(angle); // can this be done once?\n\t\t\t\t\t\t\tctx.fillText(text, 0, 0);\n\t\t\t\t\t\t\tctx.restore();\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tctx.fillText(text, x, y + j * lineHeight);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// ticks\n\t\t\tif (ticks.show) {\n\t\t\t\tdrawOrthoLines(\n\t\t\t\t\tcanOffs,\n\t\t\t\t\tticks.filter(self, splits, i, _space, incr),\n\t\t\t\t\tori,\n\t\t\t\t\tside,\n\t\t\t\t\tbasePos,\n\t\t\t\t\ttickSize,\n\t\t\t\t\troundDec(ticks.width * pxRatio, 3),\n\t\t\t\t\tticks.stroke(self, i),\n\t\t\t\t\tticks.dash,\n\t\t\t\t\tticks.cap,\n\t\t\t\t);\n\t\t\t}\n\n\t\t\t// grid\n\t\t\tlet grid = axis.grid;\n\n\t\t\tif (grid.show) {\n\t\t\t\tdrawOrthoLines(\n\t\t\t\t\tcanOffs,\n\t\t\t\t\tgrid.filter(self, splits, i, _space, incr),\n\t\t\t\t\tori,\n\t\t\t\t\tori == 0 ? 2 : 1,\n\t\t\t\t\tori == 0 ? plotTop : plotLft,\n\t\t\t\t\tori == 0 ? plotHgt : plotWid,\n\t\t\t\t\troundDec(grid.width * pxRatio, 3),\n\t\t\t\t\tgrid.stroke(self, i),\n\t\t\t\t\tgrid.dash,\n\t\t\t\t\tgrid.cap,\n\t\t\t\t);\n\t\t\t}\n\n\t\t\tif (border.show) {\n\t\t\t\tdrawOrthoLines(\n\t\t\t\t\t[basePos],\n\t\t\t\t\t[1],\n\t\t\t\t\tori == 0 ? 1 : 0,\n\t\t\t\t\tori == 0 ? 1 : 2,\n\t\t\t\t\tori == 1 ? plotTop : plotLft,\n\t\t\t\t\tori == 1 ? plotHgt : plotWid,\n\t\t\t\t\troundDec(border.width * pxRatio, 3),\n\t\t\t\t\tborder.stroke(self, i),\n\t\t\t\t\tborder.dash,\n\t\t\t\t\tborder.cap,\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\n\t\tfire(\"drawAxes\");\n\t}\n\n\tfunction resetYSeries(minMax) {\n\t//\tlog(\"resetYSeries()\", arguments);\n\n\t\tseries.forEach((s, i) => {\n\t\t\tif (i > 0) {\n\t\t\t\ts._paths = null;\n\n\t\t\t\tif (minMax) {\n\t\t\t\t\tif (mode == 1) {\n\t\t\t\t\t\ts.min = null;\n\t\t\t\t\t\ts.max = null;\n\t\t\t\t\t}\n\t\t\t\t\telse {\n\t\t\t\t\t\ts.facets.forEach(f => {\n\t\t\t\t\t\t\tf.min = null;\n\t\t\t\t\t\t\tf.max = null;\n\t\t\t\t\t\t});\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t});\n\t}\n\n\tlet queuedCommit = false;\n\n\tfunction commit() {\n\t\tif (!queuedCommit) {\n\t\t\tmicroTask(_commit);\n\t\t\tqueuedCommit = true;\n\t\t}\n\t}\n\n\tfunction _commit() {\n\t//\tlog(\"_commit()\", arguments);\n\n\t\tif (shouldSetScales) {\n\t\t\tsetScales();\n\t\t\tshouldSetScales = false;\n\t\t}\n\n\t\tif (shouldConvergeSize) {\n\t\t\tconvergeSize();\n\t\t\tshouldConvergeSize = false;\n\t\t}\n\n\t\tif (shouldSetSize) {\n\t\t\tsetStylePx(under, LEFT,   plotLftCss);\n\t\t\tsetStylePx(under, TOP,    plotTopCss);\n\t\t\tsetStylePx(under, WIDTH,  plotWidCss);\n\t\t\tsetStylePx(under, HEIGHT, plotHgtCss);\n\n\t\t\tsetStylePx(over, LEFT,    plotLftCss);\n\t\t\tsetStylePx(over, TOP,     plotTopCss);\n\t\t\tsetStylePx(over, WIDTH,   plotWidCss);\n\t\t\tsetStylePx(over, HEIGHT,  plotHgtCss);\n\n\t\t\tsetStylePx(wrap, WIDTH,   fullWidCss);\n\t\t\tsetStylePx(wrap, HEIGHT,  fullHgtCss);\n\n\t\t\t// NOTE: mutating this during print preview in Chrome forces transparent\n\t\t\t// canvas pixels to white, even when followed up with clearRect() below\n\t\t\tcan.width  = round(fullWidCss * pxRatio);\n\t\t\tcan.height = round(fullHgtCss * pxRatio);\n\n\t\t\taxes.forEach(({ _el, _show, _size, _pos, side }) => {\n\t\t\t\tif (_el != null) {\n\t\t\t\t\tif (_show) {\n\t\t\t\t\t\tlet posOffset = (side === 3 || side === 0 ? _size : 0);\n\t\t\t\t\t\tlet isVt = side % 2 == 1;\n\n\t\t\t\t\t\tsetStylePx(_el, isVt ? \"left\"   : \"top\",    _pos - posOffset);\n\t\t\t\t\t\tsetStylePx(_el, isVt ? \"width\"  : \"height\", _size);\n\t\t\t\t\t\tsetStylePx(_el, isVt ? \"top\"    : \"left\",   isVt ? plotTopCss : plotLftCss);\n\t\t\t\t\t\tsetStylePx(_el, isVt ? \"height\" : \"width\",  isVt ? plotHgtCss : plotWidCss);\n\n\t\t\t\t\t\tremClass(_el, OFF);\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t\taddClass(_el, OFF);\n\t\t\t\t}\n\t\t\t});\n\n\t\t\t// invalidate ctx style cache\n\t\t\tctxStroke = ctxFill = ctxWidth = ctxJoin = ctxCap = ctxFont = ctxAlign = ctxBaseline = ctxDash = null;\n\t\t\tctxAlpha = 1;\n\n\t\t\tsyncRect(true);\n\n\t\t\tfire(\"setSize\");\n\n\t\t\tshouldSetSize = false;\n\t\t}\n\n\t\tif (fullWidCss > 0 && fullHgtCss > 0) {\n\t\t\tctx.clearRect(0, 0, can.width, can.height);\n\t\t\tfire(\"drawClear\");\n\t\t\tdrawOrder.forEach(fn => fn());\n\t\t\tfire(\"draw\");\n\t\t}\n\n\t//\tif (shouldSetSelect) {\n\t\t// TODO: update .u-select metrics (if visible)\n\t\t//\tsetStylePx(selectDiv, TOP, select.top = 0);\n\t\t//\tsetStylePx(selectDiv, LEFT, select.left = 0);\n\t\t//\tsetStylePx(selectDiv, WIDTH, select.width = 0);\n\t\t//\tsetStylePx(selectDiv, HEIGHT, select.height = 0);\n\t\t//\tshouldSetSelect = false;\n\t//\t}\n\n\t\tif (cursor.show && shouldSetCursor) {\n\t\t\tupdateCursor(null, true, false);\n\t\t\tshouldSetCursor = false;\n\t\t}\n\n\t//\tif (FEAT_LEGEND && legend.show && legend.live && shouldSetLegend) {}\n\n\t\tif (!ready) {\n\t\t\tready = true;\n\t\t\tself.status = 1;\n\n\t\t\tfire(\"ready\");\n\t\t}\n\n\t\tviaAutoScaleX = false;\n\n\t\tqueuedCommit = false;\n\t}\n\n\tself.redraw = (rebuildPaths, recalcAxes) => {\n\t\tshouldConvergeSize = recalcAxes || false;\n\n\t\tif (rebuildPaths !== false)\n\t\t\t_setScale(xScaleKey, scaleX.min, scaleX.max);\n\t\telse\n\t\t\tcommit();\n\t};\n\n\t// redraw() => setScale('x', scales.x.min, scales.x.max);\n\n\t// explicit, never re-ranged (is this actually true? for x and y)\n\tfunction setScale(key, opts) {\n\t\tlet sc = scales[key];\n\n\t\tif (sc.from == null) {\n\t\t\tif (dataLen == 0) {\n\t\t\t\tlet minMax = sc.range(self, opts.min, opts.max, key);\n\t\t\t\topts.min = minMax[0];\n\t\t\t\topts.max = minMax[1];\n\t\t\t}\n\n\t\t\tif (opts.min > opts.max) {\n\t\t\t\tlet _min = opts.min;\n\t\t\t\topts.min = opts.max;\n\t\t\t\topts.max = _min;\n\t\t\t}\n\n\t\t\tif (dataLen > 1 && opts.min != null && opts.max != null && opts.max - opts.min < 1e-16)\n\t\t\t\treturn;\n\n\t\t\tif (key == xScaleKey) {\n\t\t\t\tif (sc.distr == 2 && dataLen > 0) {\n\t\t\t\t\topts.min = closestIdx(opts.min, data[0]);\n\t\t\t\t\topts.max = closestIdx(opts.max, data[0]);\n\n\t\t\t\t\tif (opts.min == opts.max)\n\t\t\t\t\t\topts.max++;\n\t\t\t\t}\n\t\t\t}\n\n\t\t//\tlog(\"setScale()\", arguments);\n\n\t\t\tpendScales[key] = opts;\n\n\t\t\tshouldSetScales = true;\n\t\t\tcommit();\n\t\t}\n\t}\n\n\tself.setScale = setScale;\n\n//\tINTERACTION\n\n\tlet xCursor;\n\tlet yCursor;\n\tlet vCursor;\n\tlet hCursor;\n\n\t// starting position before cursor.move\n\tlet rawMouseLeft0;\n\tlet rawMouseTop0;\n\n\t// starting position\n\tlet mouseLeft0;\n\tlet mouseTop0;\n\n\t// current position before cursor.move\n\tlet rawMouseLeft1;\n\tlet rawMouseTop1;\n\n\t// current position\n\tlet mouseLeft1;\n\tlet mouseTop1;\n\n\tlet dragging = false;\n\n\tconst drag = cursor.drag;\n\n\tlet dragX = drag.x;\n\tlet dragY = drag.y;\n\n\tif (cursor.show) {\n\t\tif (cursor.x)\n\t\t\txCursor = placeDiv(CURSOR_X, over);\n\t\tif (cursor.y)\n\t\t\tyCursor = placeDiv(CURSOR_Y, over);\n\n\t\tif (scaleX.ori == 0) {\n\t\t\tvCursor = xCursor;\n\t\t\thCursor = yCursor;\n\t\t}\n\t\telse {\n\t\t\tvCursor = yCursor;\n\t\t\thCursor = xCursor;\n\t\t}\n\n\t\tmouseLeft1 = cursor.left;\n\t\tmouseTop1 = cursor.top;\n\t}\n\n\tconst select = self.select = assign({\n\t\tshow:   true,\n\t\tover:   true,\n\t\tleft:   0,\n\t\twidth:  0,\n\t\ttop:    0,\n\t\theight: 0,\n\t}, opts.select);\n\n\tconst selectDiv = select.show ? placeDiv(SELECT, select.over ? over : under) : null;\n\n\tfunction setSelect(opts, _fire) {\n\t\tif (select.show) {\n\t\t\tfor (let prop in opts)\n\t\t\t\tsetStylePx(selectDiv, prop, select[prop] = opts[prop]);\n\n\t\t\t_fire !== false && fire(\"setSelect\");\n\t\t}\n\t}\n\n\tself.setSelect = setSelect;\n\n\tfunction toggleDOM(i, onOff) {\n\t\tlet s = series[i];\n\t\tlet label = showLegend ? legendRows[i] : null;\n\n\t\tif (s.show)\n\t\t\tlabel && remClass(label, OFF);\n\t\telse {\n\t\t\tlabel && addClass(label, OFF);\n\t\t\tcursorPts.length > 1 && elTrans(cursorPts[i], -10, -10, plotWidCss, plotHgtCss);\n\t\t}\n\t}\n\n\tfunction _setScale(key, min, max) {\n\t\tsetScale(key, {min, max});\n\t}\n\n\tfunction setSeries(i, opts, _fire, _pub) {\n\t//\tlog(\"setSeries()\", arguments);\n\n\t\tif (opts.focus != null)\n\t\t\tsetFocus(i);\n\n\t\tif (opts.show != null) {\n\t\t\tseries.forEach((s, si) => {\n\t\t\t\tif (si > 0 && (i == si || i == null)) {\n\t\t\t\t\ts.show = opts.show;\n\t\t\t\t\ttoggleDOM(si, opts.show);\n\n\t\t\t\t\t_setScale(mode == 2 ? s.facets[1].scale : s.scale, null, null);\n\t\t\t\t\tcommit();\n\t\t\t\t}\n\t\t\t});\n\t\t}\n\n\t\t_fire !== false && fire(\"setSeries\", i, opts);\n\n\t\t_pub && pubSync(\"setSeries\", self, i, opts);\n\t}\n\n\tself.setSeries = setSeries;\n\n\tfunction setBand(bi, opts) {\n\t\tassign(bands[bi], opts);\n\t}\n\n\tfunction addBand(opts, bi) {\n\t\topts.fill = fnOrSelf(opts.fill || null);\n\t\topts.dir = ifNull(opts.dir, -1);\n\t\tbi = bi == null ? bands.length : bi;\n\t\tbands.splice(bi, 0, opts);\n\t}\n\n\tfunction delBand(bi) {\n\t\tif (bi == null)\n\t\t\tbands.length = 0;\n\t\telse\n\t\t\tbands.splice(bi, 1);\n\t}\n\n\tself.addBand = addBand;\n\tself.setBand = setBand;\n\tself.delBand = delBand;\n\n\tfunction setAlpha(i, value) {\n\t\tseries[i].alpha = value;\n\n\t\tif (cursor.show && cursorPts[i])\n\t\t\tcursorPts[i].style.opacity = value;\n\n\t\tif (showLegend && legendRows[i])\n\t\t\tlegendRows[i].style.opacity = value;\n\t}\n\n\t// y-distance\n\tlet closestDist;\n\tlet closestSeries;\n\tlet focusedSeries;\n\tconst FOCUS_TRUE  = {focus: true};\n\n\tfunction setFocus(i) {\n\t\tif (i != focusedSeries) {\n\t\t//\tlog(\"setFocus()\", arguments);\n\n\t\t\tlet allFocused = i == null;\n\n\t\t\tlet _setAlpha = focus.alpha != 1;\n\n\t\t\tseries.forEach((s, i2) => {\n\t\t\t\tlet isFocused = allFocused || i2 == 0 || i2 == i;\n\t\t\t\ts._focus = allFocused ? null : isFocused;\n\t\t\t\t_setAlpha && setAlpha(i2, isFocused ? 1 : focus.alpha);\n\t\t\t});\n\n\t\t\tfocusedSeries = i;\n\t\t\t_setAlpha && commit();\n\t\t}\n\t}\n\n\tif (showLegend && cursorFocus) {\n\t\ton(mouseleave, legendEl, e => {\n\t\t\tif (cursor._lock)\n\t\t\t\treturn;\n\n\t\t\tif (focusedSeries != null)\n\t\t\t\tsetSeries(null, FOCUS_TRUE, true, syncOpts.setSeries);\n\t\t});\n\t}\n\n\tfunction posToVal(pos, scale, can) {\n\t\tlet sc = scales[scale];\n\n\t\tif (can)\n\t\t\tpos = pos / pxRatio - (sc.ori == 1 ? plotTopCss : plotLftCss);\n\n\t\tlet dim = plotWidCss;\n\n\t\tif (sc.ori == 1) {\n\t\t\tdim = plotHgtCss;\n\t\t\tpos = dim - pos;\n\t\t}\n\n\t\tif (sc.dir == -1)\n\t\t\tpos = dim - pos;\n\n\t\tlet _min = sc._min,\n\t\t\t_max = sc._max,\n\t\t\tpct = pos / dim;\n\n\t\tlet sv = _min + (_max - _min) * pct;\n\n\t\tlet distr = sc.distr;\n\n\t\treturn (\n\t\t\tdistr == 3 ? pow(10, sv) :\n\t\t\tdistr == 4 ? sinh(sv, sc.asinh) :\n\t\t\tsv\n\t\t);\n\t}\n\n\tfunction closestIdxFromXpos(pos, can) {\n\t\tlet v = posToVal(pos, xScaleKey, can);\n\t\treturn closestIdx(v, data[0], i0, i1);\n\t}\n\n\tself.valToIdx = val => closestIdx(val, data[0]);\n\tself.posToIdx = closestIdxFromXpos;\n\tself.posToVal = posToVal;\n\tself.valToPos = (val, scale, can) => (\n\t\tscales[scale].ori == 0 ?\n\t\tgetHPos(val, scales[scale],\n\t\t\tcan ? plotWid : plotWidCss,\n\t\t\tcan ? plotLft : 0,\n\t\t) :\n\t\tgetVPos(val, scales[scale],\n\t\t\tcan ? plotHgt : plotHgtCss,\n\t\t\tcan ? plotTop : 0,\n\t\t)\n\t);\n\n\t// defers calling expensive functions\n\tfunction batch(fn) {\n\t\tfn(self);\n\t\tcommit();\n\t}\n\n\tself.batch = batch;\n\n\t(self.setCursor = (opts, _fire, _pub) => {\n\t\tmouseLeft1 = opts.left;\n\t\tmouseTop1 = opts.top;\n\t//\tassign(cursor, opts);\n\t\tupdateCursor(null, _fire, _pub);\n\t});\n\n\tfunction setSelH(off, dim) {\n\t\tsetStylePx(selectDiv, LEFT,  select.left = off);\n\t\tsetStylePx(selectDiv, WIDTH, select.width = dim);\n\t}\n\n\tfunction setSelV(off, dim) {\n\t\tsetStylePx(selectDiv, TOP,    select.top = off);\n\t\tsetStylePx(selectDiv, HEIGHT, select.height = dim);\n\t}\n\n\tlet setSelX = scaleX.ori == 0 ? setSelH : setSelV;\n\tlet setSelY = scaleX.ori == 1 ? setSelH : setSelV;\n\n\tfunction syncLegend() {\n\t\tif (showLegend && legend.live) {\n\t\t\tfor (let i = mode == 2 ? 1 : 0; i < series.length; i++) {\n\t\t\t\tif (i == 0 && multiValLegend)\n\t\t\t\t\tcontinue;\n\n\t\t\t\tlet vals = legend.values[i];\n\n\t\t\t\tlet j = 0;\n\n\t\t\t\tfor (let k in vals)\n\t\t\t\t\tlegendCells[i][j++].firstChild.nodeValue = vals[k];\n\t\t\t}\n\t\t}\n\t}\n\n\tfunction setLegend(opts, _fire) {\n\t\tif (opts != null) {\n\t\t\tlet idx = opts.idx;\n\n\t\t\tlegend.idx = idx;\n\t\t\tseries.forEach((s, sidx) => {\n\t\t\t\t(sidx > 0 || !multiValLegend) && setLegendValues(sidx, idx);\n\t\t\t});\n\t\t}\n\n\t\tif (showLegend && legend.live)\n\t\t\tsyncLegend();\n\n\t\tshouldSetLegend = false;\n\n\t\t_fire !== false && fire(\"setLegend\");\n\t}\n\n\tself.setLegend = setLegend;\n\n\tfunction setLegendValues(sidx, idx) {\n\t\tlet val;\n\n\t\tif (idx == null)\n\t\t\tval = NULL_LEGEND_VALUES;\n\t\telse {\n\t\t\tlet s = series[sidx];\n\t\t\tlet src = sidx == 0 && xScaleDistr == 2 ? data0 : data[sidx];\n\t\t\tval = multiValLegend ? s.values(self, sidx, idx) : {_: s.value(self, src[idx], sidx, idx)};\n\t\t}\n\n\t\tlegend.values[sidx] = val;\n\t}\n\n\tfunction updateCursor(src, _fire, _pub) {\n\t//\tts == null && log(\"updateCursor()\", arguments);\n\n\t\trawMouseLeft1 = mouseLeft1;\n\t\trawMouseTop1 = mouseTop1;\n\n\t\t[mouseLeft1, mouseTop1] = cursor.move(self, mouseLeft1, mouseTop1);\n\n\t\tif (cursor.show) {\n\t\t\tvCursor && elTrans(vCursor, round(mouseLeft1), 0, plotWidCss, plotHgtCss);\n\t\t\thCursor && elTrans(hCursor, 0, round(mouseTop1), plotWidCss, plotHgtCss);\n\t\t}\n\n\t\tlet idx;\n\n\t\t// when zooming to an x scale range between datapoints the binary search\n\t\t// for nearest min/max indices results in this condition. cheap hack :D\n\t\tlet noDataInRange = i0 > i1; // works for mode 1 only\n\n\t\tclosestDist = inf;\n\n\t\t// TODO: extract\n\t\tlet xDim = scaleX.ori == 0 ? plotWidCss : plotHgtCss;\n\t\tlet yDim = scaleX.ori == 1 ? plotWidCss : plotHgtCss;\n\n\t\t// if cursor hidden, hide points & clear legend vals\n\t\tif (mouseLeft1 < 0 || dataLen == 0 || noDataInRange) {\n\t\t\tidx = null;\n\n\t\t\tfor (let i = 0; i < series.length; i++) {\n\t\t\t\tif (i > 0) {\n\t\t\t\t\tcursorPts.length > 1 && elTrans(cursorPts[i], -10, -10, plotWidCss, plotHgtCss);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (cursorFocus)\n\t\t\t\tsetSeries(null, FOCUS_TRUE, true, src == null && syncOpts.setSeries);\n\n\t\t\tif (legend.live) {\n\t\t\t\tactiveIdxs.fill(null);\n\t\t\t\tshouldSetLegend = true;\n\n\t\t\t\tfor (let i = 0; i < series.length; i++)\n\t\t\t\t\tlegend.values[i] = NULL_LEGEND_VALUES;\n\t\t\t}\n\t\t}\n\t\telse {\n\t\t//\tlet pctY = 1 - (y / rect.height);\n\n\t\t\tlet mouseXPos, valAtPosX, xPos;\n\n\t\t\tif (mode == 1) {\n\t\t\t\tmouseXPos = scaleX.ori == 0 ? mouseLeft1 : mouseTop1;\n\t\t\t\tvalAtPosX = posToVal(mouseXPos, xScaleKey);\n\t\t\t\tidx = closestIdx(valAtPosX, data[0], i0, i1);\n\t\t\t\txPos = incrRoundUp(valToPosX(data[0][idx], scaleX, xDim, 0), 0.5);\n\t\t\t}\n\n\t\t\tfor (let i = mode == 2 ? 1 : 0; i < series.length; i++) {\n\t\t\t\tlet s = series[i];\n\n\t\t\t\tlet idx1  = activeIdxs[i];\n\t\t\t\tlet yVal1 = mode == 1 ? data[i][idx1] : data[i][1][idx1];\n\n\t\t\t\tlet idx2  = cursor.dataIdx(self, i, idx, valAtPosX);\n\t\t\t\tlet yVal2 = mode == 1 ? data[i][idx2] : data[i][1][idx2];\n\n\t\t\t\tshouldSetLegend = shouldSetLegend || yVal2 != yVal1 || idx2 != idx1;\n\n\t\t\t\tactiveIdxs[i] = idx2;\n\n\t\t\t\tlet xPos2 = idx2 == idx ? xPos : incrRoundUp(valToPosX(mode == 1 ? data[0][idx2] : data[i][0][idx2], scaleX, xDim, 0), 0.5);\n\n\t\t\t\tif (i > 0 && s.show) {\n\t\t\t\t\tlet yPos = yVal2 == null ? -10 : incrRoundUp(valToPosY(yVal2, mode == 1 ? scales[s.scale] : scales[s.facets[1].scale], yDim, 0), 0.5);\n\n\t\t\t\t\tif (yPos > 0 && mode == 1) {\n\t\t\t\t\t\tlet dist = abs(yPos - mouseTop1);\n\n\t\t\t\t\t\tif (dist <= closestDist) {\n\t\t\t\t\t\t\tclosestDist = dist;\n\t\t\t\t\t\t\tclosestSeries = i;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tlet hPos, vPos;\n\n\t\t\t\t\tif (scaleX.ori == 0) {\n\t\t\t\t\t\thPos = xPos2;\n\t\t\t\t\t\tvPos = yPos;\n\t\t\t\t\t}\n\t\t\t\t\telse {\n\t\t\t\t\t\thPos = yPos;\n\t\t\t\t\t\tvPos = xPos2;\n\t\t\t\t\t}\n\n\t\t\t\t\tif (shouldSetLegend && cursorPts.length > 1) {\n\t\t\t\t\t\telColor(cursorPts[i], cursor.points.fill(self, i), cursor.points.stroke(self, i));\n\n\t\t\t\t\t\tlet ptWid, ptHgt, ptLft, ptTop,\n\t\t\t\t\t\t\tcentered = true,\n\t\t\t\t\t\t\tgetBBox = cursor.points.bbox;\n\n\t\t\t\t\t\tif (getBBox != null) {\n\t\t\t\t\t\t\tcentered = false;\n\n\t\t\t\t\t\t\tlet bbox = getBBox(self, i);\n\n\t\t\t\t\t\t\tptLft = bbox.left;\n\t\t\t\t\t\t\tptTop = bbox.top;\n\t\t\t\t\t\t\tptWid = bbox.width;\n\t\t\t\t\t\t\tptHgt = bbox.height;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse {\n\t\t\t\t\t\t\tptLft = hPos;\n\t\t\t\t\t\t\tptTop = vPos;\n\t\t\t\t\t\t\tptWid = ptHgt = cursor.points.size(self, i);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\telSize(cursorPts[i], ptWid, ptHgt, centered);\n\t\t\t\t\t\telTrans(cursorPts[i], ptLft, ptTop, plotWidCss, plotHgtCss);\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tif (legend.live) {\n\t\t\t\t\tif (!shouldSetLegend || i == 0 && multiValLegend)\n\t\t\t\t\t\tcontinue;\n\n\t\t\t\t\tsetLegendValues(i, idx2);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tcursor.idx = idx;\n\t\tcursor.left = mouseLeft1;\n\t\tcursor.top = mouseTop1;\n\n\t\tif (shouldSetLegend) {\n\t\t\tlegend.idx = idx;\n\t\t\tsetLegend();\n\t\t}\n\n\t\t// nit: cursor.drag.setSelect is assumed always true\n\t\tif (select.show && dragging) {\n\t\t\tif (src != null) {\n\t\t\t\tlet [xKey, yKey] = syncOpts.scales;\n\t\t\t\tlet [matchXKeys, matchYKeys] = syncOpts.match;\n\t\t\t\tlet [xKeySrc, yKeySrc] = src.cursor.sync.scales;\n\n\t\t\t\t// match the dragX/dragY implicitness/explicitness of src\n\t\t\t\tlet sdrag = src.cursor.drag;\n\t\t\t\tdragX = sdrag._x;\n\t\t\t\tdragY = sdrag._y;\n\n\t\t\t\tif (dragX || dragY) {\n\t\t\t\t\tlet { left, top, width, height } = src.select;\n\n\t\t\t\t\tlet sori = src.scales[xKey].ori;\n\t\t\t\t\tlet sPosToVal = src.posToVal;\n\n\t\t\t\t\tlet sOff, sDim, sc, a, b;\n\n\t\t\t\t\tlet matchingX = xKey != null && matchXKeys(xKey, xKeySrc);\n\t\t\t\t\tlet matchingY = yKey != null && matchYKeys(yKey, yKeySrc);\n\n\t\t\t\t\tif (matchingX && dragX) {\n\t\t\t\t\t\tif (sori == 0) {\n\t\t\t\t\t\t\tsOff = left;\n\t\t\t\t\t\t\tsDim = width;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse {\n\t\t\t\t\t\t\tsOff = top;\n\t\t\t\t\t\t\tsDim = height;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tsc = scales[xKey];\n\n\t\t\t\t\t\ta = valToPosX(sPosToVal(sOff, xKeySrc),        sc, xDim, 0);\n\t\t\t\t\t\tb = valToPosX(sPosToVal(sOff + sDim, xKeySrc), sc, xDim, 0);\n\n\t\t\t\t\t\tsetSelX(min(a,b), abs(b-a));\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t\tsetSelX(0, xDim);\n\n\t\t\t\t\tif (matchingY && dragY) {\n\t\t\t\t\t\tif (sori == 1) {\n\t\t\t\t\t\t\tsOff = left;\n\t\t\t\t\t\t\tsDim = width;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse {\n\t\t\t\t\t\t\tsOff = top;\n\t\t\t\t\t\t\tsDim = height;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tsc = scales[yKey];\n\n\t\t\t\t\t\ta = valToPosY(sPosToVal(sOff, yKeySrc),        sc, yDim, 0);\n\t\t\t\t\t\tb = valToPosY(sPosToVal(sOff + sDim, yKeySrc), sc, yDim, 0);\n\n\t\t\t\t\t\tsetSelY(min(a,b), abs(b-a));\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t\tsetSelY(0, yDim);\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t\thideSelect();\n\t\t\t}\n\t\t\telse {\n\t\t\t\tlet rawDX = abs(rawMouseLeft1 - rawMouseLeft0);\n\t\t\t\tlet rawDY = abs(rawMouseTop1 - rawMouseTop0);\n\n\t\t\t\tif (scaleX.ori == 1) {\n\t\t\t\t\tlet _rawDX = rawDX;\n\t\t\t\t\trawDX = rawDY;\n\t\t\t\t\trawDY = _rawDX;\n\t\t\t\t}\n\n\t\t\t\tdragX = drag.x && rawDX >= drag.dist;\n\t\t\t\tdragY = drag.y && rawDY >= drag.dist;\n\n\t\t\t\tlet uni = drag.uni;\n\n\t\t\t\tif (uni != null) {\n\t\t\t\t\t// only calc drag status if they pass the dist thresh\n\t\t\t\t\tif (dragX && dragY) {\n\t\t\t\t\t\tdragX = rawDX >= uni;\n\t\t\t\t\t\tdragY = rawDY >= uni;\n\n\t\t\t\t\t\t// force unidirectionality when both are under uni limit\n\t\t\t\t\t\tif (!dragX && !dragY) {\n\t\t\t\t\t\t\tif (rawDY > rawDX)\n\t\t\t\t\t\t\t\tdragY = true;\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\tdragX = true;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\telse if (drag.x && drag.y && (dragX || dragY))\n\t\t\t\t\t// if omni with no uni then both dragX / dragY should be true if either is true\n\t\t\t\t\tdragX = dragY = true;\n\n\t\t\t\tlet p0, p1;\n\n\t\t\t\tif (dragX) {\n\t\t\t\t\tif (scaleX.ori == 0) {\n\t\t\t\t\t\tp0 = mouseLeft0;\n\t\t\t\t\t\tp1 = mouseLeft1;\n\t\t\t\t\t}\n\t\t\t\t\telse {\n\t\t\t\t\t\tp0 = mouseTop0;\n\t\t\t\t\t\tp1 = mouseTop1;\n\t\t\t\t\t}\n\n\t\t\t\t\tsetSelX(min(p0, p1), abs(p1 - p0));\n\n\t\t\t\t\tif (!dragY)\n\t\t\t\t\t\tsetSelY(0, yDim);\n\t\t\t\t}\n\n\t\t\t\tif (dragY) {\n\t\t\t\t\tif (scaleX.ori == 1) {\n\t\t\t\t\t\tp0 = mouseLeft0;\n\t\t\t\t\t\tp1 = mouseLeft1;\n\t\t\t\t\t}\n\t\t\t\t\telse {\n\t\t\t\t\t\tp0 = mouseTop0;\n\t\t\t\t\t\tp1 = mouseTop1;\n\t\t\t\t\t}\n\n\t\t\t\t\tsetSelY(min(p0, p1), abs(p1 - p0));\n\n\t\t\t\t\tif (!dragX)\n\t\t\t\t\t\tsetSelX(0, xDim);\n\t\t\t\t}\n\n\t\t\t\t// the drag didn't pass the dist requirement\n\t\t\t\tif (!dragX && !dragY) {\n\t\t\t\t\tsetSelX(0, 0);\n\t\t\t\t\tsetSelY(0, 0);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tdrag._x = dragX;\n\t\tdrag._y = dragY;\n\n\t\tif (src == null) {\n\t\t\tif (_pub) {\n\t\t\t\tif (syncKey != null) {\n\t\t\t\t\tlet [xSyncKey, ySyncKey] = syncOpts.scales;\n\n\t\t\t\t\tsyncOpts.values[0] = xSyncKey != null ? posToVal(scaleX.ori == 0 ? mouseLeft1 : mouseTop1, xSyncKey) : null;\n\t\t\t\t\tsyncOpts.values[1] = ySyncKey != null ? posToVal(scaleX.ori == 1 ? mouseLeft1 : mouseTop1, ySyncKey) : null;\n\t\t\t\t}\n\n\t\t\t\tpubSync(mousemove, self, mouseLeft1, mouseTop1, plotWidCss, plotHgtCss, idx);\n\t\t\t}\n\n\t\t\tif (cursorFocus) {\n\t\t\t\tlet shouldPub = _pub && syncOpts.setSeries;\n\t\t\t\tlet p = focus.prox;\n\n\t\t\t\tif (focusedSeries == null) {\n\t\t\t\t\tif (closestDist <= p)\n\t\t\t\t\t\tsetSeries(closestSeries, FOCUS_TRUE, true, shouldPub);\n\t\t\t\t}\n\t\t\t\telse {\n\t\t\t\t\tif (closestDist > p)\n\t\t\t\t\t\tsetSeries(null, FOCUS_TRUE, true, shouldPub);\n\t\t\t\t\telse if (closestSeries != focusedSeries)\n\t\t\t\t\t\tsetSeries(closestSeries, FOCUS_TRUE, true, shouldPub);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tready && _fire !== false && fire(\"setCursor\");\n\t}\n\n\tlet rect = null;\n\n\tfunction syncRect(defer) {\n\t\tif (defer === true)\n\t\t\trect = null;\n\t\telse {\n\t\t\trect = over.getBoundingClientRect();\n\t\t\tfire(\"syncRect\", rect);\n\t\t}\n\t}\n\n\tfunction mouseMove(e, src, _l, _t, _w, _h, _i) {\n\t\tif (cursor._lock)\n\t\t\treturn;\n\n\t\tcacheMouse(e, src, _l, _t, _w, _h, _i, false, e != null);\n\n\t\tif (e != null)\n\t\t\tupdateCursor(null, true, true);\n\t\telse\n\t\t\tupdateCursor(src, true, false);\n\t}\n\n\tfunction cacheMouse(e, src, _l, _t, _w, _h, _i, initial, snap) {\n\t\tif (rect == null)\n\t\t\tsyncRect(false);\n\n\t\tif (e != null) {\n\t\t\t_l = e.clientX - rect.left;\n\t\t\t_t = e.clientY - rect.top;\n\t\t}\n\t\telse {\n\t\t\tif (_l < 0 || _t < 0) {\n\t\t\t\tmouseLeft1 = -10;\n\t\t\t\tmouseTop1 = -10;\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlet [xKey, yKey] = syncOpts.scales;\n\n\t\t\tlet syncOptsSrc = src.cursor.sync;\n\t\t\tlet [xValSrc, yValSrc] = syncOptsSrc.values;\n\t\t\tlet [xKeySrc, yKeySrc] = syncOptsSrc.scales;\n\t\t\tlet [matchXKeys, matchYKeys] = syncOpts.match;\n\n\t\t\tlet rotSrc = src.axes[0].side % 2 == 1;\n\n\t\t\tlet xDim = scaleX.ori == 0 ? plotWidCss : plotHgtCss,\n\t\t\t\tyDim = scaleX.ori == 1 ? plotWidCss : plotHgtCss,\n\t\t\t\t_xDim = rotSrc ? _h : _w,\n\t\t\t\t_yDim = rotSrc ? _w : _h,\n\t\t\t\t_xPos = rotSrc ? _t : _l,\n\t\t\t\t_yPos = rotSrc ? _l : _t;\n\n\t\t\tif (xKeySrc != null)\n\t\t\t\t_l = matchXKeys(xKey, xKeySrc) ? getPos(xValSrc, scales[xKey], xDim, 0) : -10;\n\t\t\telse\n\t\t\t\t_l = xDim * (_xPos/_xDim);\n\n\t\t\tif (yKeySrc != null)\n\t\t\t\t_t = matchYKeys(yKey, yKeySrc) ? getPos(yValSrc, scales[yKey], yDim, 0) : -10;\n\t\t\telse\n\t\t\t\t_t = yDim * (_yPos/_yDim);\n\n\t\t\tif (scaleX.ori == 1) {\n\t\t\t\tlet __l = _l;\n\t\t\t\t_l = _t;\n\t\t\t\t_t = __l;\n\t\t\t}\n\t\t}\n\n\t\tif (snap) {\n\t\t\tif (_l <= 1 || _l >= plotWidCss - 1)\n\t\t\t\t_l = incrRound(_l, plotWidCss);\n\n\t\t\tif (_t <= 1 || _t >= plotHgtCss - 1)\n\t\t\t\t_t = incrRound(_t, plotHgtCss);\n\t\t}\n\n\t\tif (initial) {\n\t\t\trawMouseLeft0 = _l;\n\t\t\trawMouseTop0 = _t;\n\n\t\t\t[mouseLeft0, mouseTop0] = cursor.move(self, _l, _t);\n\t\t}\n\t\telse {\n\t\t\tmouseLeft1 = _l;\n\t\t\tmouseTop1 = _t;\n\t\t}\n\t}\n\n\tconst _hideProps = {\n\t\twidth: 0,\n\t\theight: 0,\n\t};\n\n\tfunction hideSelect() {\n\t\tsetSelect(_hideProps, false);\n\t}\n\n\tfunction mouseDown(e, src, _l, _t, _w, _h, _i) {\n\t\tdragging = true;\n\t\tdragX = dragY = drag._x = drag._y = false;\n\n\t\tcacheMouse(e, src, _l, _t, _w, _h, _i, true, false);\n\n\t\tif (e != null) {\n\t\t\tonMouse(mouseup, doc, mouseUp);\n\t\t\tpubSync(mousedown, self, mouseLeft0, mouseTop0, plotWidCss, plotHgtCss, null);\n\t\t}\n\t}\n\n\tfunction mouseUp(e, src, _l, _t, _w, _h, _i) {\n\t\tdragging = drag._x = drag._y = false;\n\n\t\tcacheMouse(e, src, _l, _t, _w, _h, _i, false, true);\n\n\t\tlet { left, top, width, height } = select;\n\n\t\tlet hasSelect = width > 0 || height > 0;\n\n\t\thasSelect && setSelect(select);\n\n\t\tif (drag.setScale && hasSelect) {\n\t\t//\tif (syncKey != null) {\n\t\t//\t\tdragX = drag.x;\n\t\t//\t\tdragY = drag.y;\n\t\t//\t}\n\n\t\t\tlet xOff = left,\n\t\t\t\txDim = width,\n\t\t\t\tyOff = top,\n\t\t\t\tyDim = height;\n\n\t\t\tif (scaleX.ori == 1) {\n\t\t\t\txOff = top,\n\t\t\t\txDim = height,\n\t\t\t\tyOff = left,\n\t\t\t\tyDim = width;\n\t\t\t}\n\n\t\t\tif (dragX) {\n\t\t\t\t_setScale(xScaleKey,\n\t\t\t\t\tposToVal(xOff, xScaleKey),\n\t\t\t\t\tposToVal(xOff + xDim, xScaleKey)\n\t\t\t\t);\n\t\t\t}\n\n\t\t\tif (dragY) {\n\t\t\t\tfor (let k in scales) {\n\t\t\t\t\tlet sc = scales[k];\n\n\t\t\t\t\tif (k != xScaleKey && sc.from == null && sc.min != inf) {\n\t\t\t\t\t\t_setScale(k,\n\t\t\t\t\t\t\tposToVal(yOff + yDim, k),\n\t\t\t\t\t\t\tposToVal(yOff, k)\n\t\t\t\t\t\t);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\thideSelect();\n\t\t}\n\t\telse if (cursor.lock) {\n\t\t\tcursor._lock = !cursor._lock;\n\n\t\t\tif (!cursor._lock)\n\t\t\t\tupdateCursor(null, true, false);\n\t\t}\n\n\t\tif (e != null) {\n\t\t\toffMouse(mouseup, doc);\n\t\t\tpubSync(mouseup, self, mouseLeft1, mouseTop1, plotWidCss, plotHgtCss, null);\n\t\t}\n\t}\n\n\tfunction mouseLeave(e, src, _l, _t, _w, _h, _i) {\n\t\tif (!cursor._lock) {\n\t\t\tlet _dragging = dragging;\n\n\t\t\tif (dragging) {\n\t\t\t\t// handle case when mousemove aren't fired all the way to edges by browser\n\t\t\t\tlet snapH = true;\n\t\t\t\tlet snapV = true;\n\t\t\t\tlet snapProx = 10;\n\n\t\t\t\tlet dragH, dragV;\n\n\t\t\t\tif (scaleX.ori == 0) {\n\t\t\t\t\tdragH = dragX;\n\t\t\t\t\tdragV = dragY;\n\t\t\t\t}\n\t\t\t\telse {\n\t\t\t\t\tdragH = dragY;\n\t\t\t\t\tdragV = dragX;\n\t\t\t\t}\n\n\t\t\t\tif (dragH && dragV) {\n\t\t\t\t\t// maybe omni corner snap\n\t\t\t\t\tsnapH = mouseLeft1 <= snapProx || mouseLeft1 >= plotWidCss - snapProx;\n\t\t\t\t\tsnapV = mouseTop1  <= snapProx || mouseTop1  >= plotHgtCss - snapProx;\n\t\t\t\t}\n\n\t\t\t\tif (dragH && snapH)\n\t\t\t\t\tmouseLeft1 = mouseLeft1 < mouseLeft0 ? 0 : plotWidCss;\n\n\t\t\t\tif (dragV && snapV)\n\t\t\t\t\tmouseTop1 = mouseTop1 < mouseTop0 ? 0 : plotHgtCss;\n\n\t\t\t\tupdateCursor(null, true, true);\n\n\t\t\t\tdragging = false;\n\t\t\t}\n\n\t\t\tmouseLeft1 = -10;\n\t\t\tmouseTop1 = -10;\n\n\t\t\t// passing a non-null timestamp to force sync/mousemove event\n\t\t\tupdateCursor(null, true, true);\n\n\t\t\tif (_dragging)\n\t\t\t\tdragging = _dragging;\n\t\t}\n\t}\n\n\tfunction dblClick(e, src, _l, _t, _w, _h, _i) {\n\t\tautoScaleX();\n\n\t\thideSelect();\n\n\t\tif (e != null)\n\t\t\tpubSync(dblclick, self, mouseLeft1, mouseTop1, plotWidCss, plotHgtCss, null);\n\t}\n\n\tfunction syncPxRatio() {\n\t\taxes.forEach(syncFontSize);\n\t\t_setSize(self.width, self.height, true);\n\t}\n\n\ton(dppxchange, win, syncPxRatio);\n\n\t// internal pub/sub\n\tconst events = {};\n\n\tevents.mousedown = mouseDown;\n\tevents.mousemove = mouseMove;\n\tevents.mouseup = mouseUp;\n\tevents.dblclick = dblClick;\n\tevents[\"setSeries\"] = (e, src, idx, opts) => {\n\t\tsetSeries(idx, opts, true, false);\n\t};\n\n\tif (cursor.show) {\n\t\tonMouse(mousedown,  over, mouseDown);\n\t\tonMouse(mousemove,  over, mouseMove);\n\t\tonMouse(mouseenter, over, syncRect);\n\t\tonMouse(mouseleave, over, mouseLeave);\n\n\t\tonMouse(dblclick, over, dblClick);\n\n\t\tcursorPlots.add(self);\n\n\t\tself.syncRect = syncRect;\n\t}\n\n\t// external on/off\n\tconst hooks = self.hooks = opts.hooks || {};\n\n\tfunction fire(evName, a1, a2) {\n\t\tif (evName in hooks) {\n\t\t\thooks[evName].forEach(fn => {\n\t\t\t\tfn.call(null, self, a1, a2);\n\t\t\t});\n\t\t}\n\t}\n\n\t(opts.plugins || []).forEach(p => {\n\t\tfor (let evName in p.hooks)\n\t\t\thooks[evName] = (hooks[evName] || []).concat(p.hooks[evName]);\n\t});\n\n\tconst syncOpts = assign({\n\t\tkey: null,\n\t\tsetSeries: false,\n\t\tfilters: {\n\t\t\tpub: retTrue,\n\t\t\tsub: retTrue,\n\t\t},\n\t\tscales: [xScaleKey, series[1] ? series[1].scale : null],\n\t\tmatch: [retEq, retEq],\n\t\tvalues: [null, null],\n\t}, cursor.sync);\n\n\t(cursor.sync = syncOpts);\n\n\tconst syncKey = syncOpts.key;\n\n\tconst sync = _sync(syncKey);\n\n\tfunction pubSync(type, src, x, y, w, h, i) {\n\t\tif (syncOpts.filters.pub(type, src, x, y, w, h, i))\n\t\t\tsync.pub(type, src, x, y, w, h, i);\n\t}\n\n\tsync.sub(self);\n\n\tfunction pub(type, src, x, y, w, h, i) {\n\t\tif (syncOpts.filters.sub(type, src, x, y, w, h, i))\n\t\t\tevents[type](null, src, x, y, w, h, i);\n\t}\n\n\t(self.pub = pub);\n\n\tfunction destroy() {\n\t\tsync.unsub(self);\n\t\tcursorPlots.delete(self);\n\t\tmouseListeners.clear();\n\t\toff(dppxchange, win, syncPxRatio);\n\t\troot.remove();\n\t\tfire(\"destroy\");\n\t}\n\n\tself.destroy = destroy;\n\n\tfunction _init() {\n\t\tfire(\"init\", opts, data);\n\n\t\tsetData(data || opts.data, false);\n\n\t\tif (pendScales[xScaleKey])\n\t\t\tsetScale(xScaleKey, pendScales[xScaleKey]);\n\t\telse\n\t\t\tautoScaleX();\n\n\t\t_setSize(opts.width, opts.height);\n\n\t\tupdateCursor(null, true, false);\n\n\t\tsetSelect(select, false);\n\t}\n\n\tseries.forEach(initSeries);\n\n\taxes.forEach(initAxis);\n\n\tif (then) {\n\t\tif (then instanceof HTMLElement) {\n\t\t\tthen.appendChild(root);\n\t\t\t_init();\n\t\t}\n\t\telse\n\t\t\tthen(self, _init);\n\t}\n\telse\n\t\t_init();\n\n\treturn self;\n}\n\nuPlot.assign = assign;\nuPlot.fmtNum = fmtNum;\nuPlot.rangeNum = rangeNum;\nuPlot.rangeLog = rangeLog;\nuPlot.rangeAsinh = rangeAsinh;\nuPlot.orient   = orient;\nuPlot.pxRatio = pxRatio;\n\n{\n\tuPlot.join = join;\n}\n\n{\n\tuPlot.fmtDate = fmtDate;\n\tuPlot.tzDate  = tzDate;\n}\n\n{\n\tuPlot.sync = _sync;\n}\n\n{\n\tuPlot.addGap = addGap;\n\tuPlot.clipGaps = clipGaps;\n\n\tlet paths = uPlot.paths = {\n\t\tpoints,\n\t};\n\n\t(paths.linear  = linear);\n\t(paths.stepped = stepped);\n\t(paths.bars    = bars);\n\t(paths.spline  = monotoneCubic);\n}\n\nexport { uPlot as default };\n","function stack(data, omit) {\n  let data2 = [];\n  let bands = [];\n  let d0Len = data[0].length;\n  let accum = Array(d0Len);\n\n  for (let i = 0; i < d0Len; i++) accum[i] = 0;\n\n  for (let i = 1; i < data.length; i++)\n    data2.push(omit(i) ? data[i] : data[i].map((v, i) => (accum[i] += +v)));\n\n  for (let i = 1; i < data.length; i++)\n    !omit(i) &&\n      bands.push({\n        series: [data.findIndex((s, j) => j > i && !omit(j)), i]\n      });\n\n  bands = bands.filter(b => b.series[1] > -1);\n\n  return {\n    data: [data[0]].concat(data2),\n    bands\n  };\n}\n\nfunction getOpts(title, series) {\n  return {\n    scales: {\n      x: {\n        time: true\n      }\n    },\n    series\n  };\n}\n\nfunction getStackedOpts(title, series, data, interp) {\n  let opts = getOpts(title, series);\n\n  let interped = interp ? interp(data) : data;\n\n  let stacked = stack(interped, i => false);\n  opts.bands = stacked.bands;\n\n  opts.cursor = opts.cursor || {};\n  opts.cursor.dataIdx = (u, seriesIdx, closestIdx, xValue) => {\n    return data[seriesIdx][closestIdx] === null ? null : closestIdx;\n  };\n\n  opts.series.forEach(s => {\n    s.value = (u, v, si, i) => data[si][i];\n\n    s.points = s.points || {};\n\n    // scan raw unstacked data to return only real points\n    s.points.filter = (u, seriesIdx, show, gaps) => {\n      if (show) {\n        let pts = [];\n        data[seriesIdx].forEach((v, i) => {\n          v !== null && pts.push(i);\n        });\n        return pts;\n      }\n    };\n  });\n\n  // force 0 to be the sum minimum this instead of the bottom series\n  opts.scales.y = {\n    range: (u, min, max) => {\n      let minMax = uPlot.rangeNum(min, max, 0.1, true);\n      return [minMax[0], minMax[1]];\n    }\n  };\n\n  // restack on toggle\n  opts.hooks = {\n    setSeries: [\n      (u, i) => {\n        let stacked = stack(data, i => !u.series[i].show);\n        u.delBand(null);\n        stacked.bands.forEach(b => u.addBand(b));\n        u.setData(stacked.data);\n      }\n    ]\n  };\n\n  return { opts, data: stacked.data };\n}\n\nfunction stack2(series) {\n  // for uplot data\n  let data = Array(series.length);\n  let bands = [];\n\n  let dataLen = series[0].values.length;\n\n  let zeroArr = Array(dataLen).fill(0);\n\n  let stackGroups = new Map();\n  let seriesStackKeys = Array(series.length);\n\n  series.forEach((s, si) => {\n    let vals = s.values.slice();\n\n    // apply negY\n    if (s.negY) {\n      for (let i = 0; i < vals.length; i++) {\n        if (vals[i] !== null) vals[i] *= -1;\n      }\n    }\n\n    if (s.stacking.mode != \"none\") {\n      let hasPos = vals.some(v => v > 0);\n      // derive stacking key\n      let stackKey = (seriesStackKeys[si] =\n        s.stacking.mode + s.scaleKey + s.stacking.group + (hasPos ? \"+\" : \"-\"));\n      let group = stackGroups.get(stackKey);\n\n      // initialize stacking group\n      if (group === null) {\n        group = {\n          series: [],\n          acc: zeroArr.slice(),\n          dir: hasPos ? -1 : 1\n        };\n        stackGroups.set(stackKey, group);\n      }\n\n      // push for bands gen\n      group.series.unshift(si);\n\n      let stacked = (data[si] = Array(dataLen));\n      let { acc } = group;\n\n      for (let i = 0; i < dataLen; i++) {\n        let v = vals[i];\n\n        if (v !== null) stacked[i] = acc[i] += v;\n        else stacked[i] = v; // we may want to coerce to 0 here\n      }\n    } else data[si] = vals;\n  });\n\n  // re-compute by percent\n  series.forEach((s, si) => {\n    if (s.stacking.mode == \"percent\") {\n      let group = stackGroups.get(seriesStackKeys[si]);\n      let { acc } = group;\n\n      // re-negatify percent\n      let sign = group.dir * -1;\n\n      let stacked = data[si];\n\n      for (let i = 0; i < dataLen; i++) {\n        let v = stacked[i];\n\n        if (v !== null) stacked[i] = sign * (v / acc[i]);\n      }\n    }\n  });\n\n  // generate bands between adjacent group series\n  stackGroups.forEach(group => {\n    let { series, dir } = group;\n    let lastIdx = series.length - 1;\n\n    series.forEach((si, i) => {\n      if (i != lastIdx) {\n        let nextIdx = series[i + 1];\n        bands.push({\n          // since we're not passing x series[0] for stacking, real idxs are actually +1\n          series: [si + 1, nextIdx + 1],\n          dir\n        });\n      }\n    });\n  });\n\n  return {\n    data,\n    bands\n  };\n}\n\nexport function stackedChart(title, series, _data, interp, width, height, hooks, el) {\n  let { opts, data } = getStackedOpts(title, series, _data, interp);\n  opts.title = title;\n  opts.width = width;\n  opts.height = height;\n  opts.hooks = Object.assign(opts.hooks, hooks);\n  console.log(opts);\n  return new uPlot(opts, data, el);\n}\n\n","// From Friss tuto (https://github.com/FrissAnalytics/shinyJsTutorials/blob/master/tutorials/tutorial_03.Rmd)\nexport function getWidget(id) {\n  var htmlWidgetsObj = HTMLWidgets.find(\"#\" + id);\n  var widgetObj;\n  if (typeof htmlWidgetsObj !== \"undefined\") {\n    widgetObj = htmlWidgetsObj.getWidget();\n  }\n  return widgetObj;\n}\n","module.exports = window[\"HTMLWidgets\"];","// 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\tid: moduleId,\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","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\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};","__webpack_require__.nc = undefined;","import \"widgets\";\nimport uPlot from \"uplot\";\nimport \"uplot/dist/uPlot.min.css\";\nimport * as utils from \"../modules/utils\";\nimport { stackedChart } from \"../modules/stack\";\n\nconst resizer = el => {\n  const func = (u, init) => {\n    const resizeObserver = new ResizeObserver(entries => {\n      for (let entry of entries) {\n        let adjHeight = entry.contentRect.height;\n        let adjWidth = entry.contentRect.width;\n        const extras = u.root.querySelectorAll(\".u-legend, .u-title\");\n        for (let extra of extras) {\n          adjHeight -= Math.floor(extra.offsetHeight);\n        }\n        u.setSize({ width: adjWidth, height: adjHeight });\n      }\n    });\n    resizeObserver.observe(el);\n    el.appendChild(u.root);\n    init();\n  };\n  return func;\n};\n\nconst drawPoints = (u, seriesIdx, idx0, idx1) => {\n  const size = 5 * devicePixelRatio;\n\n  uPlot.orient(\n    u,\n    seriesIdx,\n    (\n      series,\n      dataX,\n      dataY,\n      scaleX,\n      scaleY,\n      valToPosX,\n      valToPosY,\n      xOff,\n      yOff,\n      xDim,\n      yDim,\n      moveTo,\n      lineTo,\n      rect,\n      arc\n    ) => {\n      let d = u.data[seriesIdx];\n\n      u.ctx.fillStyle = series.stroke();\n\n      let deg360 = 2 * Math.PI;\n\n      console.time(\"points\");\n\n      //\tlet cir = new Path2D();\n      //\tcir.moveTo(0, 0);\n      //\tarc(cir, 0, 0, 3, 0, deg360);\n\n      // Create transformation matrix that moves 200 points to the right\n      //\tlet m = document.createElementNS('http://www.w3.org/2000/svg', 'svg').createSVGMatrix();\n      //\tm.a = 1;   m.b = 0;\n      //\tm.c = 0;   m.d = 1;\n      //\tm.e = 200; m.f = 0;\n\n      let p = new Path2D();\n\n      for (let i = 0; i < d[0].length; i++) {\n        let xVal = d[0][i];\n        let yVal = d[1][i];\n\n        if (\n          xVal >= scaleX.min &&\n          xVal <= scaleX.max &&\n          yVal >= scaleY.min &&\n          yVal <= scaleY.max\n        ) {\n          let cx = valToPosX(xVal, scaleX, xDim, xOff);\n          let cy = valToPosY(yVal, scaleY, yDim, yOff);\n\n          p.moveTo(cx + size / 2, cy);\n          //\tarc(p, cx, cy, 3, 0, deg360);\n          arc(p, cx, cy, size / 2, 0, deg360);\n\n          //\tm.e = cx;\n          //\tm.f = cy;\n          //\tp.addPath(cir, m);\n\n          //\tqt.add({x: cx - 1.5, y: cy - 1.5, w: 3, h: 3, sidx: seriesIdx, didx: i});\n        }\n      }\n\n      console.timeEnd(\"points\");\n\n      u.ctx.fill(p);\n    }\n  );\n\n  return null;\n};\n\nHTMLWidgets.widget({\n  name: \"uPlot\",\n\n  type: \"output\",\n\n  factory: function(el, width, height) {\n    var plot, options, data;\n\n    return {\n      renderValue: function(x) {\n        if (typeof plot !== \"undefined\") {\n          plot.destroy();\n        }\n        options = x.options;\n        options.width = width;\n        options.height = height;\n        data = x.data;\n        if (x.stacked) {\n          if (!options.hooks)\n            options.hooks = {};\n          options.hooks.init = [\n            u => {\n              [...u.root.querySelectorAll(\".u-legend .u-series\")].forEach(\n                (el, i) => {\n                  if (u.series[i]._hide) {\n                    el.style.display = \"none\";\n                  }\n                }\n              );\n            }\n          ];\n          plot = stackedChart(\n            options.title,\n            options.series,\n            data,\n            null,\n            width,\n            height,\n            options.hooks,\n            resizer(el)\n          );\n        } else {\n          plot = new uPlot(options, data, resizer(el));\n        }\n      },\n      getWidget: function() {\n        return plot;\n      },\n      resize: function(width, height) {}\n    };\n  }\n});\n\nif (HTMLWidgets.shinyMode) {\n  Shiny.addCustomMessageHandler(\"uplot-api\", function(obj) {\n    var plot = utils.getWidget(obj.id);\n    if (typeof plot != \"undefined\") {\n      plot[obj.name].apply(null, obj.args);\n    }\n  });\n  Shiny.addCustomMessageHandler(\"uplot-setData\", function(obj) {\n    var plot = utils.getWidget(obj.id);\n    if (typeof plot != \"undefined\") {\n      plot.setData(obj.data);\n    }\n  });\n}\n\nexport { uPlot, drawPoints };\n\n"],"names":[],"sourceRoot":""} \ No newline at end of file +(()=>{"use strict";var e={350:(e,t,l)=>{l.d(t,{Z:()=>s});var n=l(81),o=l.n(n),i=l(645),r=l.n(i)()(o());r.push([e.id,'.uplot, .uplot *, .uplot *::before, .uplot *::after {box-sizing: border-box;}.uplot {font-family: system-ui, -apple-system, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";line-height: 1.5;width: min-content;}.u-title {text-align: center;font-size: 18px;font-weight: bold;}.u-wrap {position: relative;user-select: none;}.u-over, .u-under {position: absolute;}.u-under {overflow: hidden;}.uplot canvas {display: block;position: relative;width: 100%;height: 100%;}.u-axis {position: absolute;}.u-legend {font-size: 14px;margin: auto;text-align: center;}.u-inline {display: block;}.u-inline * {display: inline-block;}.u-inline tr {margin-right: 16px;}.u-legend th {font-weight: 600;}.u-legend th > * {vertical-align: middle;display: inline-block;}.u-legend .u-marker {width: 1em;height: 1em;margin-right: 4px;background-clip: padding-box !important;}.u-inline.u-live th::after {content: ":";vertical-align: middle;}.u-inline:not(.u-live) .u-value {display: none;}.u-series > * {padding: 4px;}.u-series th {cursor: pointer;}.u-legend .u-off > * {opacity: 0.3;}.u-select {background: rgba(0,0,0,0.07);position: absolute;pointer-events: none;}.u-cursor-x, .u-cursor-y {position: absolute;left: 0;top: 0;pointer-events: none;will-change: transform;}.u-hz .u-cursor-x, .u-vt .u-cursor-y {height: 100%;border-right: 1px dashed #607D8B;}.u-hz .u-cursor-y, .u-vt .u-cursor-x {width: 100%;border-bottom: 1px dashed #607D8B;}.u-cursor-pt {position: absolute;top: 0;left: 0;border-radius: 50%;border: 0 solid;pointer-events: none;will-change: transform;/*this has to be !important since we set inline "background" shorthand */background-clip: padding-box !important;}.u-axis.u-off, .u-select.u-off, .u-cursor-x.u-off, .u-cursor-y.u-off, .u-cursor-pt.u-off {display: none;}',""]);const s=r},645:e=>{e.exports=function(e){var t=[];return t.toString=function(){return this.map((function(t){var l="",n=void 0!==t[5];return t[4]&&(l+="@supports (".concat(t[4],") {")),t[2]&&(l+="@media ".concat(t[2]," {")),n&&(l+="@layer".concat(t[5].length>0?" ".concat(t[5]):""," {")),l+=e(t),n&&(l+="}"),t[2]&&(l+="}"),t[4]&&(l+="}"),l})).join("")},t.i=function(e,l,n,o,i){"string"==typeof e&&(e=[[null,e,void 0]]);var r={};if(n)for(var s=0;s0?" ".concat(c[5]):""," {").concat(c[1],"}")),c[5]=i),l&&(c[2]?(c[1]="@media ".concat(c[2]," {").concat(c[1],"}"),c[2]=l):c[2]=l),o&&(c[4]?(c[1]="@supports (".concat(c[4],") {").concat(c[1],"}"),c[4]=o):c[4]="".concat(o)),t.push(c))}},t}},81:e=>{e.exports=function(e){return e[1]}},379:e=>{var t=[];function l(e){for(var l=-1,n=0;n{var t={};e.exports=function(e,l){var n=function(e){if(void 0===t[e]){var l=document.querySelector(e);if(window.HTMLIFrameElement&&l instanceof window.HTMLIFrameElement)try{l=l.contentDocument.head}catch(e){l=null}t[e]=l}return t[e]}(e);if(!n)throw new Error("Couldn't find a style target. This probably means that the value for the 'insert' parameter is invalid.");n.appendChild(l)}},216:e=>{e.exports=function(e){var t=document.createElement("style");return e.setAttributes(t,e.attributes),e.insert(t,e.options),t}},565:(e,t,l)=>{e.exports=function(e){var t=l.nc;t&&e.setAttribute("nonce",t)}},795:e=>{e.exports=function(e){var t=e.insertStyleElement(e);return{update:function(l){!function(e,t,l){var n="";l.supports&&(n+="@supports (".concat(l.supports,") {")),l.media&&(n+="@media ".concat(l.media," {"));var o=void 0!==l.layer;o&&(n+="@layer".concat(l.layer.length>0?" ".concat(l.layer):""," {")),n+=l.css,o&&(n+="}"),l.media&&(n+="}"),l.supports&&(n+="}");var i=l.sourceMap;i&&"undefined"!=typeof btoa&&(n+="\n/*# sourceMappingURL=data:application/json;base64,".concat(btoa(unescape(encodeURIComponent(JSON.stringify(i))))," */")),t.styleTagTransform(n,e,t.options)}(t,e,l)},remove:function(){!function(e){if(null===e.parentNode)return!1;e.parentNode.removeChild(e)}(t)}}}},589:e=>{e.exports=function(e,t){if(t.styleSheet)t.styleSheet.cssText=e;else{for(;t.firstChild;)t.removeChild(t.firstChild);t.appendChild(document.createTextNode(e))}}}},t={};function l(n){var o=t[n];if(void 0!==o)return o.exports;var i=t[n]={id:n,exports:{}};return e[n](i,i.exports,l),i.exports}l.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return l.d(t,{a:t}),t},l.d=(e,t)=>{for(var n in t)l.o(t,n)&&!l.o(e,n)&&Object.defineProperty(e,n,{enumerable:!0,get:t[n]})},l.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),l.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},l.nc=void 0;var n={};(()=>{l.r(n),l.d(n,{drawPoints:()=>hn,uPlot:()=>ql}),window.HTMLWidgets;const e="u-off",t="u-label",o="width",i="height",r="top",s="bottom",u="left",a="right",c="#000",f="#0000",d="mousemove",h="mousedown",p="mouseup",m="mouseenter",g="mouseleave",x="dblclick",w="change",v="dppxchange",b="--",y="undefined"!=typeof window,k=y?document:null,_=y?window:null,M=y?navigator:null;let S,E;function T(e,t){if(null!=t){let l=e.classList;!l.contains(t)&&l.add(t)}}function z(e,t){let l=e.classList;l.contains(t)&&l.remove(t)}function P(e,t,l){e.style[t]=l+"px"}function D(e,t,l,n){let o=k.createElement(e);return null!=t&&T(o,t),null!=l&&l.insertBefore(o,n),o}function A(e,t){return D("div",e,t)}const C=new WeakMap;function W(t,l,n,o,i){let r="translate("+l+"px,"+n+"px)";r!=C.get(t)&&(t.style.transform=r,C.set(t,r),l<0||n<0||l>o||n>i?T(t,e):z(t,e))}const H=new WeakMap;function I(e,t,l){let n=t+l;n!=H.get(e)&&(H.set(e,n),e.style.background=t,e.style.borderColor=l)}const Y=new WeakMap;function R(e,t,l,n){let o=t+""+l;o!=Y.get(e)&&(Y.set(e,o),e.style.height=l+"px",e.style.width=t+"px",e.style.marginLeft=n?-t/2+"px":0,e.style.marginTop=n?-l/2+"px":0)}const O={passive:!0},j={...O,capture:!0};function L(e,t,l,n){t.addEventListener(e,l,n?j:O)}function F(e,t,l,n){t.removeEventListener(e,l,n?j:O)}function N(e,t,l,n){let o;l=l||0;let i=(n=n||t.length-1)<=2147483647;for(;n-l>1;)o=i?l+n>>1:ne((l+n)/2),t[o]=t&&o<=l;o+=n)if(null!=e[o])return o;return-1}function U(e,t,l,n){let o=ae(e),i=ae(t),r=10==l?ce:fe;e==t&&(-1==o?(e*=l,t/=l):(e/=l,t*=l));let s=1==i?ie:ne,u=(1==o?ne:ie)(r(le(e))),a=s(r(le(t))),c=ue(l,u),f=ue(l,a);return 10==l&&(u<0&&(c=Ee(c,-u)),a<0&&(f=Ee(f,-a))),n||2==l?(e=c*o,t=f*i):(e=Se(e,c),t=Me(t,f)),[e,t]}function B(e,t,l,n){let o=U(e,t,l,n);return 0==e&&(o[0]=0),0==t&&(o[1]=0),o}y&&function e(){let t=devicePixelRatio;S!=t&&(S=t,E&&F(w,E,e),E=matchMedia(`(min-resolution: ${S-.001}dppx) and (max-resolution: ${S+.001}dppx)`),L(w,E,e),_.dispatchEvent(new CustomEvent(v)))}();const V={mode:3,pad:.1},Z={pad:0,soft:null,mode:0},q={min:Z,max:Z};function J(e,t,l,n){return Ye(l)?X(e,t,l):(Z.pad=l,Z.soft=n?0:null,Z.mode=n?3:0,X(e,t,q))}function K(e,t){return null==e?t:e}function X(e,t,l){let n=l.min,o=l.max,i=K(n.pad,0),r=K(o.pad,0),s=K(n.hard,-he),u=K(o.hard,he),a=K(n.soft,he),c=K(o.soft,-he),f=K(n.mode,0),d=K(o.mode,0),h=t-e,p=ce(h),m=se(le(e),le(t)),g=ce(m),x=le(g-p);(h<1e-9||x>10)&&(h=0,0!=e&&0!=t||(h=1e-9,2==f&&a!=he&&(i=0),2==d&&c!=-he&&(r=0)));let w=h||m||1e3,v=ce(w),b=ue(10,ne(v)),y=Ee(Se(e-w*(0==h?0==e?.1:1:i),b/10),9),k=e>=a&&(1==f||3==f&&y<=a||2==f&&y>=a)?a:he,_=se(s,y=k?k:re(k,y)),M=Ee(Me(t+w*(0==h?0==t?.1:1:r),b/10),9),S=t<=c&&(1==d||3==d&&M>=c||2==d&&M<=c)?c:-he,E=re(u,M>S&&t<=S?S:se(S,M));return _==E&&0==_&&(E=100),[_,E]}const $=new Intl.NumberFormat(y?M.language:"en-US"),Q=e=>$.format(e),ee=Math,te=ee.PI,le=ee.abs,ne=ee.floor,oe=ee.round,ie=ee.ceil,re=ee.min,se=ee.max,ue=ee.pow,ae=ee.sign,ce=ee.log10,fe=ee.log2,de=(e,t=1)=>ee.asinh(e/t),he=1/0;function pe(e){return 1+(0|ce((e^e>>31)-(e>>31)))}function me(e,t,l){return re(se(e,t),l)}function ge(e){return"function"==typeof e?e:()=>e}const xe=e=>e,we=(e,t)=>t,ve=e=>null,be=e=>!0,ye=(e,t)=>e==t,ke=e=>Ee(e,14);function _e(e,t){return ke(Ee(ke(e/t))*t)}function Me(e,t){return ke(ie(ke(e/t))*t)}function Se(e,t){return ke(ne(ke(e/t))*t)}function Ee(e,t=0){if(He(e))return e;let l=10**t,n=e*l*(1+Number.EPSILON);return oe(n)/l}const Te=new Map;function ze(e){return((""+e).split(".")[1]||"").length}function Pe(e,t,l,n){let o=[],i=n.map(ze);for(let r=t;r=0&&r>=0?0:t)+(r>=i[e]?0:i[e]),a=Ee(s,u);o.push(a),Te.set(a,u)}}return o}const De={},Ae=[],Ce=[null,null],We=Array.isArray,He=Number.isInteger;function Ie(e){return"string"==typeof e}function Ye(e){let t=!1;if(null!=e){let l=e.constructor;t=null==l||l==Object}return t}function Re(e){return null!=e&&"object"==typeof e}const Oe=Object.getPrototypeOf(Uint8Array);function je(e,t=Ye){let l;if(We(e)){let n=e.find((e=>null!=e));if(We(n)||t(n)){l=Array(e.length);for(let n=0;ni){for(n=r-1;n>=0&&null==e[n];)e[n--]=null;for(n=r+1;nPromise.resolve().then(e):queueMicrotask,Ge=["January","February","March","April","May","June","July","August","September","October","November","December"],Ue=["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"];function Be(e){return e.slice(0,3)}const Ve=Ue.map(Be),Ze=Ge.map(Be),qe={MMMM:Ge,MMM:Ze,WWWW:Ue,WWW:Ve};function Je(e){return(e<10?"0":"")+e}const Ke={YYYY:e=>e.getFullYear(),YY:e=>(e.getFullYear()+"").slice(2),MMMM:(e,t)=>t.MMMM[e.getMonth()],MMM:(e,t)=>t.MMM[e.getMonth()],MM:e=>Je(e.getMonth()+1),M:e=>e.getMonth()+1,DD:e=>Je(e.getDate()),D:e=>e.getDate(),WWWW:(e,t)=>t.WWWW[e.getDay()],WWW:(e,t)=>t.WWW[e.getDay()],HH:e=>Je(e.getHours()),H:e=>e.getHours(),h:e=>{let t=e.getHours();return 0==t?12:t>12?t-12:t},AA:e=>e.getHours()>=12?"PM":"AM",aa:e=>e.getHours()>=12?"pm":"am",a:e=>e.getHours()>=12?"p":"a",mm:e=>Je(e.getMinutes()),m:e=>e.getMinutes(),ss:e=>Je(e.getSeconds()),s:e=>e.getSeconds(),fff:e=>{return((t=e.getMilliseconds())<10?"00":t<100?"0":"")+t;var t}};function Xe(e,t){t=t||qe;let l,n=[],o=/\{([a-z]+)\}|[^{]+/gi;for(;l=o.exec(e);)n.push("{"==l[0][0]?Ke[l[1]]:l[0]);return e=>{let l="";for(let o=0;oe%1==0,et=[1,2,2.5,5],tt=Pe(10,-16,0,et),lt=Pe(10,0,16,et),nt=lt.filter(Qe),ot=tt.concat(lt),it="{YYYY}",rt="\n"+it,st="{M}/{D}",ut="\n"+st,at=ut+"/{YY}",ct="{aa}",ft="{h}:{mm}"+ct,dt="\n"+ft,ht=":{ss}",pt=null;function mt(e){let t=1e3*e,l=60*t,n=60*l,o=24*n,i=30*o,r=365*o;return[(1==e?Pe(10,0,3,et).filter(Qe):Pe(10,-3,0,et)).concat([t,5*t,10*t,15*t,30*t,l,5*l,10*l,15*l,30*l,n,2*n,3*n,4*n,6*n,8*n,12*n,o,2*o,3*o,4*o,5*o,6*o,7*o,8*o,9*o,10*o,15*o,i,2*i,3*i,4*i,6*i,r,2*r,5*r,10*r,25*r,50*r,100*r]),[[r,it,pt,pt,pt,pt,pt,pt,1],[28*o,"{MMM}",rt,pt,pt,pt,pt,pt,1],[o,st,rt,pt,pt,pt,pt,pt,1],[n,"{h}"+ct,at,pt,ut,pt,pt,pt,1],[l,ft,at,pt,ut,pt,pt,pt,1],[t,ht,at+" "+ft,pt,ut+" "+ft,pt,dt,pt,1],[e,ht+".{fff}",at+" "+ft,pt,ut+" "+ft,pt,dt,pt,1]],function(t){return(s,u,a,c,f,d)=>{let h=[],p=f>=r,m=f>=i&&f=o?o:f,r=v+(ne(a)-ne(x))+Me(x-v,i);h.push(r);let p=t(r),m=p.getHours()+p.getMinutes()/l+p.getSeconds()/n,g=f/n,w=d/s.axes[u]._space;for(;r=Ee(r+f,1==e?0:3),!(r>c);)if(g>1){let e=ne(Ee(m+g,6))%24,l=t(r).getHours()-e;l>1&&(l=-1),r-=l*n,m=(m+g)%24,Ee((r-h[h.length-1])/f,3)*w>=.7&&h.push(r)}else h.push(r)}return h}}]}const[gt,xt,wt]=mt(1),[vt,bt,yt]=mt(.001);function kt(e,t){return e.map((e=>e.map(((l,n)=>0==n||8==n||null==l?l:t(1==n||0==e[8]?l:e[1]+l)))))}function _t(e,t){return(l,n,o,i,r)=>{let s,u,a,c,f,d,h=t.find((e=>r>=e[0]))||t[t.length-1];return n.map((t=>{let l=e(t),n=l.getFullYear(),o=l.getMonth(),i=l.getDate(),r=l.getHours(),p=l.getMinutes(),m=l.getSeconds(),g=n!=s&&h[2]||o!=u&&h[3]||i!=a&&h[4]||r!=c&&h[5]||p!=f&&h[6]||m!=d&&h[7]||h[1];return s=n,u=o,a=i,c=r,f=p,d=m,g(l)}))}}function Mt(e,t,l){return new Date(e,t,l)}function St(e,t){return t(e)}function Et(e,t){return(l,n,o,i)=>null==i?b:t(e(n))}Pe(2,-53,53,[1]);const Tt={show:!0,live:!0,isolate:!1,mount:()=>{},markers:{show:!0,width:2,stroke:function(e,t){let l=e.series[t];return l.width?l.stroke(e,t):l.points.width?l.points.stroke(e,t):null},fill:function(e,t){return e.series[t].fill(e,t)},dash:"solid"},idx:null,idxs:null,values:[]},zt=[0,0];function Pt(e,t,l,n=!0){return e=>{0==e.button&&(!n||e.target==t)&&l(e)}}function Dt(e,t,l,n=!0){return e=>{(!n||e.target==t)&&l(e)}}const At={show:!0,x:!0,y:!0,lock:!1,move:function(e,t,l){return zt[0]=t,zt[1]=l,zt},points:{show:function(e,t){let l=e.cursor.points,n=A(),r=l.size(e,t);P(n,o,r),P(n,i,r);let s=r/-2;P(n,"marginLeft",s),P(n,"marginTop",s);let u=l.width(e,t,r);return u&&P(n,"borderWidth",u),n},size:function(e,t){return e.series[t].points.size},width:0,stroke:function(e,t){let l=e.series[t].points;return l._stroke||l._fill},fill:function(e,t){let l=e.series[t].points;return l._fill||l._stroke}},bind:{mousedown:Pt,mouseup:Pt,click:Pt,dblclick:Pt,mousemove:Dt,mouseleave:Dt,mouseenter:Dt},drag:{setScale:!0,x:!0,y:!1,dist:0,uni:null,click:(e,t)=>{t.stopPropagation(),t.stopImmediatePropagation()},_x:!1,_y:!1},focus:{prox:-1,bias:0},left:-10,top:-10,idx:null,dataIdx:function(e,t,l){return l},idxs:null,event:null},Ct={show:!0,stroke:"rgba(0,0,0,0.07)",width:2},Wt=Le({},Ct,{filter:we}),Ht=Le({},Wt,{size:10}),It=Le({},Ct,{show:!1}),Yt='12px system-ui, -apple-system, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji"',Rt="bold "+Yt,Ot={show:!0,scale:"x",stroke:c,space:50,gap:5,size:50,labelGap:0,labelSize:30,labelFont:Rt,side:2,grid:Wt,ticks:Ht,border:It,font:Yt,lineGap:1.5,rotate:0},jt={show:!0,scale:"x",auto:!1,sorted:1,min:he,max:-he,idxs:[]};function Lt(e,t,l,n,o){return t.map((e=>null==e?"":Q(e)))}function Ft(e,t,l,n,o,i,r){let s=[],u=Te.get(o)||0;for(let e=l=r?l:Ee(Me(l,o),u);e<=n;e=Ee(e+o,u))s.push(Object.is(e,-0)?0:e);return s}function Nt(e,t,l,n,o,i,r){const s=[],u=e.scales[e.axes[t].scale].log,a=ne((10==u?ce:fe)(l));o=ue(u,a),10==u&&a<0&&(o=Ee(o,-a));let c=l;do{s.push(c),c+=o,10==u&&(c=Ee(c,Te.get(o))),c>=o*u&&(o=c)}while(c<=n);return s}function Gt(e,t,l,n,o,i,r){let s=e.scales[e.axes[t].scale].asinh,u=n>s?Nt(e,t,se(s,l),n,o):[s],a=n>=0&&l<=0?[0]:[];return(l<-s?Nt(e,t,se(s,-n),-l,o):[s]).reverse().map((e=>-e)).concat(a,u)}const Ut=/./,Bt=/[12357]/,Vt=/[125]/,Zt=/1/,qt=(e,t,l,n)=>e.map(((e,o)=>4==t&&0==e||o%n==0&&l.test(e.toExponential()[e<0?1:0])?e:null));function Jt(e,t,l,n,o){let i=e.axes[l],r=i.scale,s=e.scales[r],u=e.valToPos,a=i._space,c=u(10,r),f=u(9,r)-c>=a?Ut:u(7,r)-c>=a?Bt:u(5,r)-c>=a?Vt:Zt;if(f==Zt){let e=le(u(1,r)-c);if(eo,tl={show:!0,auto:!0,sorted:0,gaps:el,alpha:1,facets:[Le({},Qt,{scale:"x"}),Le({},Qt,{scale:"y"})]},ll={scale:"y",auto:!0,sorted:0,show:!0,spanGaps:!1,gaps:el,alpha:1,points:{show:function(e,t){let{scale:l,idxs:n}=e.series[0],o=e._data[0],i=e.valToPos(o[n[0]],l,!0),r=e.valToPos(o[n[1]],l,!0),s=le(r-i)/(e.series[t].points.space*S);return n[1]-n[0]<=s},filter:null},values:null,min:he,max:-he,idxs:[],path:null,clip:null};function nl(e,t,l,n,o){return l/10}const ol={time:!0,auto:!0,distr:1,log:10,asinh:1,min:null,max:null,dir:1,ori:0},il=Le({},ol,{time:!1,ori:1}),rl={};function sl(e,t){let l=rl[e];return l||(l={key:e,plots:[],sub(e){l.plots.push(e)},unsub(e){l.plots=l.plots.filter((t=>t!=e))},pub(e,t,n,o,i,r,s){for(let u=0;u{let m=e.pxRound;const g=s.dir*(0==s.ori?1:-1),x=0==s.ori?wl:vl;let w,v;1==g?(w=l,v=n):(w=n,v=l);let b=m(a(t[w],s,h,f)),y=m(c(r[w],u,p,d)),k=m(a(t[v],s,h,f)),_=m(c(1==i?u.max:u.min,u,p,d)),M=new Path2D(o);return x(M,k,_),x(M,b,_),x(M,b,y),M}))}function dl(e,t,l,n,o,i){let r=null;if(e.length>0){r=new Path2D;const s=0==t?bl:yl;let u=l;for(let t=0;tl[0]){let e=l[0]-u;e>0&&s(r,u,n,e,n+i),u=l[1]}}let a=l+o-u,c=10;a>0&&s(r,u,n-c/2,a,n+i+c)}return r}function hl(e,t,l,n,o,i,r){let s=[],u=e.length;for(let a=1==o?l:n;a>=l&&a<=n;a+=o)if(null===t[a]){let c=a,f=a;if(1==o)for(;++a<=n&&null===t[a];)f=a;else for(;--a>=l&&null===t[a];)f=a;let d=i(e[c]),h=f==c?d:i(e[f]),p=c-o;d=r<=0&&p>=0&&p=0&&m>=0&&m=d&&s.push([d,h])}return s}function pl(e){return 0==e?xe:1==e?oe:t=>_e(t,e)}function ml(e){let t=0==e?gl:xl,l=0==e?(e,t,l,n,o,i)=>{e.arcTo(t,l,n,o,i)}:(e,t,l,n,o,i)=>{e.arcTo(l,t,o,n,i)},n=0==e?(e,t,l,n,o)=>{e.rect(t,l,n,o)}:(e,t,l,n,o)=>{e.rect(l,t,o,n)};return(e,o,i,r,s,u=0,a=0)=>{0==u&&0==a?n(e,o,i,r,s):(u=re(u,r/2,s/2),a=re(a,r/2,s/2),t(e,o+u,i),l(e,o+r,i,o+r,i+s,u),l(e,o+r,i+s,o,i+s,a),l(e,o,i+s,o,i,a),l(e,o,i,o+r,i,u),e.closePath())}}const gl=(e,t,l)=>{e.moveTo(t,l)},xl=(e,t,l)=>{e.moveTo(l,t)},wl=(e,t,l)=>{e.lineTo(t,l)},vl=(e,t,l)=>{e.lineTo(l,t)},bl=ml(0),yl=ml(1),kl=(e,t,l,n,o,i)=>{e.arc(t,l,n,o,i)},_l=(e,t,l,n,o,i)=>{e.arc(l,t,n,o,i)},Ml=(e,t,l,n,o,i,r)=>{e.bezierCurveTo(t,l,n,o,i,r)},Sl=(e,t,l,n,o,i,r)=>{e.bezierCurveTo(l,t,o,n,r,i)};function El(e){return(e,t,l,n,o)=>ul(e,t,((t,i,r,s,u,a,c,f,d,h,p)=>{let m,g,{pxRound:x,points:w}=t;0==s.ori?(m=gl,g=kl):(m=xl,g=_l);const v=Ee(w.width*S,3);let b=(w.size-w.width)/2*S,y=Ee(2*b,3),k=new Path2D,_=new Path2D,{left:M,top:E,width:T,height:z}=e.bbox;bl(_,M-y,E-y,T+2*y,z+2*y);const P=e=>{if(null!=r[e]){let t=x(a(i[e],s,h,f)),l=x(c(r[e],u,p,d));m(k,t+b,l),g(k,t,l,b,0,2*te)}};if(o)o.forEach(P);else for(let e=l;e<=n;e++)P(e);return{stroke:v>0?k:null,fill:k,clip:_,flags:3}}))}function Tl(e){return(t,l,n,o,i,r)=>{n!=o&&(i!=n&&r!=n&&e(t,l,n),i!=o&&r!=o&&e(t,l,o),e(t,l,r))}}const zl=Tl(wl),Pl=Tl(vl);function Dl(e){const t=K(e?.alignGaps,0);return(e,l,n,o)=>ul(e,l,((i,r,s,u,a,c,f,d,h,p,m)=>{let g,x,w=i.pxRound,v=e=>w(c(e,u,p,d)),b=e=>w(f(e,a,m,h));0==u.ori?(g=wl,x=zl):(g=vl,x=Pl);const y=u.dir*(0==u.ori?1:-1),k={stroke:new Path2D,fill:null,clip:null,band:null,gaps:null,flags:1},_=k.stroke;let M,S,E,T=he,z=-he,P=v(r[1==y?n:o]),D=G(s,n,o,1*y),A=G(s,n,o,-1*y),C=v(r[D]),W=v(r[A]),H=!1;for(let e=1==y?n:o;e>=n&&e<=o;e+=y){let t=v(r[e]),l=s[e];t==P?null!=l?(S=b(l),T==he&&(g(_,t,S),M=S),T=re(S,T),z=se(S,z)):null===l&&(H=!0):(T!=he&&(x(_,P,T,z,M,S),E=P),null!=l?(S=b(l),g(_,t,S),T=z=M=S):(T=he,z=-he,null===l&&(H=!0)),P=t)}T!=he&&T!=z&&E!=P&&x(_,P,T,z,M,S);let[I,Y]=al(e,l);if(null!=i.fill||0!=I){let t=k.fill=new Path2D(_),n=b(i.fillTo(e,l,i.min,i.max,I));g(t,W,n),g(t,C,n)}if(!i.spanGaps){let a=[];H&&a.push(...hl(r,s,n,o,y,v,t)),k.gaps=a=i.gaps(e,l,n,o,a),k.clip=dl(a,u.ori,d,h,p,m)}return 0!=Y&&(k.band=2==Y?[fl(e,l,n,o,_,-1),fl(e,l,n,o,_,1)]:fl(e,l,n,o,_,Y)),k}))}function Al(e,t,l,n,o,i){const r=e.length;if(r<2)return null;const s=new Path2D;if(l(s,e[0],t[0]),2==r)n(s,e[1],t[1]);else{let l=Array(r),n=Array(r-1),i=Array(r-1),u=Array(r-1);for(let l=0;l0!=n[e]>0?l[e]=0:(l[e]=3*(u[e-1]+u[e])/((2*u[e]+u[e-1])/n[e-1]+(u[e]+2*u[e-1])/n[e]),isFinite(l[e])||(l[e]=0));l[r-1]=n[r-2];for(let n=0;n{ql.pxRatio=S})));const Hl=Dl(),Il=El();function Yl(e,t,l,n){return(n?[e[0],e[1]].concat(e.slice(2)):[e[0]].concat(e.slice(1))).map(((e,n)=>Rl(e,n,t,l)))}function Rl(e,t,l,n){return Le({},0==t?l:n,e)}function Ol(e,t,l){return null==t?Ce:[t,l]}const jl=Ol;function Ll(e,t,l){return null==t?Ce:J(t,l,.1,!0)}function Fl(e,t,l,n){return null==t?Ce:U(t,l,e.scales[n].log,!1)}const Nl=Fl;function Gl(e,t,l,n){return null==t?Ce:B(t,l,e.scales[n].log,!1)}const Ul=Gl;function Bl(e,t,l,n,o){let i=se(pe(e),pe(t)),r=t-e,s=N(o/n*r,l);do{let e=l[s],t=n*e/r;if(t>=o&&i+(e<5?Te.get(e):0)<=17)return[e,t]}while(++s(t=oe((l=+n)*S))+"px")),t,l]}function Zl(e){e.show&&[e.font,e.labelFont].forEach((e=>{let t=Ee(e[2]*S,1);e[0]=e[0].replace(/[0-9.]+px/,t+"px"),e[1]=t}))}function ql(l,n,c){const w={mode:K(l.mode,1)},y=w.mode;function M(e,t){return((3==t.distr?ce(e>0?e:t.clamp(w,e,t.min,t.max,t.key)):4==t.distr?de(e,t.asinh):e)-t._min)/(t._max-t._min)}function E(e,t,l,n){let o=M(e,t);return n+l*(-1==t.dir?1-o:o)}function C(e,t,l,n){let o=M(e,t);return n+l*(-1==t.dir?o:1-o)}function H(e,t,l,n){return 0==t.ori?E(e,t,l,n):C(e,t,l,n)}w.valToPosH=E,w.valToPosV=C;let Y=!1;w.status=0;const O=w.root=A("uplot");null!=l.id&&(O.id=l.id),T(O,l.class),l.title&&(A("u-title",O).textContent=l.title);const j=D("canvas"),G=w.ctx=j.getContext("2d"),Z=A("u-wrap",O);L("click",Z,(e=>{e.target===X&&(Tn!=_n||zn!=Mn)&&Hn.click(w,e)}),!0);const q=w.under=A("u-under",Z);Z.appendChild(j);const X=w.over=A("u-over",Z),$=+K((l=je(l)).pxAlign,1),Q=pl($);(l.plugins||[]).forEach((e=>{e.opts&&(l=e.opts(w,l)||l)}));const ne=l.ms||.001,ae=w.series=1==y?Yl(l.series||[],jt,ll,!1):(fe=l.series||[null],pe=tl,fe.map(((e,t)=>0==t?null:Le({},pe,e))));var fe,pe;const xe=w.axes=Yl(l.axes||[],Ot,$t,!0),ke=w.scales={},Se=w.bands=l.bands||[];Se.forEach((e=>{e.fill=ge(e.fill||null),e.dir=K(e.dir,-1)}));const Pe=2==y?ae[1].facets[0].scale:ae[0].scale,He={axes:function(){for(let e=0;een[e])):v,y=2==p.distr?en[v[1]]-en[v[0]]:d,k=t.ticks,_=t.border,M=k.show?oe(k.size*S):0,E=t._rotate*-te/180,T=Q(t._pos*S),z=T+(M+x)*f;n=0==i?z:0,l=1==i?z:0,rn(t.font[0],c,1==t.align?u:2==t.align?a:E>0?u:E<0?a:0==i?"center":3==o?a:u,E||1==i?"middle":2==o?r:s);let P=t.font[1]*t.lineGap,D=v.map((e=>Q(H(e,p,m,g)))),A=t._values;for(let e=0;e0&&(ae.forEach(((e,t)=>{if(t>0&&e.show&&null==e._paths){let l=2==y?[0,n[t][0].length-1]:function(e){let t=me(El-1,0,Sl-1),l=me(Tl+1,0,Sl-1);for(;null==e[t]&&t>0;)t--;for(;null==e[l]&&l{if(t>0&&e.show){Ql!=e.alpha&&(G.globalAlpha=Ql=e.alpha),un(t,!1),e._paths&&an(t,!1);{un(t,!0);let l=e._paths?e._paths.gaps:null,n=e.points.show(w,t,El,Tl,l),o=e.points.filter(w,t,n,l);(n||o)&&(e.points._paths=e.points.paths(w,t,El,Tl,o),an(t,!0))}1!=Ql&&(G.globalAlpha=Ql=1),xo("drawSeries",t)}})))}},Oe=(l.drawOrder||["axes","series"]).map((e=>He[e]));function Fe(e){let t=ke[e];if(null==t){let n=(l.scales||De)[e]||De;if(null!=n.from)Fe(n.from),ke[e]=Le({},ke[n.from],n,{key:e});else{t=ke[e]=Le({},e==Pe?ol:il,n),t.key=e;let l=t.time,o=t.range,i=We(o);if((e!=Pe||2==y&&!l)&&(!i||null!=o[0]&&null!=o[1]||(o={min:null==o[0]?V:{mode:1,hard:o[0],soft:o[0]},max:null==o[1]?V:{mode:1,hard:o[1],soft:o[1]}},i=!1),!i&&Ye(o))){let e=o;o=(t,l,n)=>null==l?Ce:J(l,n,e)}t.range=ge(o||(l?jl:e==Pe?3==t.distr?Nl:4==t.distr?Ul:Ol:3==t.distr?Fl:4==t.distr?Gl:Ll)),t.auto=ge(!i&&t.auto),t.clamp=ge(t.clamp||nl),t._min=t._max=null}}}Fe("x"),Fe("y"),1==y&&ae.forEach((e=>{Fe(e.scale)})),xe.forEach((e=>{Fe(e.scale)}));for(let e in l.scales)Fe(e);const Ge=ke[Pe],Ue=Ge.distr;let Be,Ve;0==Ge.ori?(T(O,"u-hz"),Be=E,Ve=C):(T(O,"u-vt"),Be=C,Ve=E);const Ze={};for(let e in ke){let t=ke[e];null==t.min&&null==t.max||(Ze[e]={min:t.min,max:t.max},t.min=t.max=null)}const qe=l.tzDate||(e=>new Date(oe(e/ne))),Je=l.fmtDate||Xe,Ke=1==ne?wt(qe):yt(qe),$e=_t(qe,kt(1==ne?xt:bt,Je)),Qe=Et(qe,St("{YYYY}-{MM}-{DD} {h}:{mm}{aa}",Je)),et=[],tt=w.legend=Le({},Tt,l.legend),lt=tt.show,it=tt.markers;let rt,st,ut;tt.idxs=et,it.width=ge(it.width),it.dash=ge(it.dash),it.stroke=ge(it.stroke),it.fill=ge(it.fill);let at,ct=[],ft=[],dt=!1,ht={};if(tt.live){const e=ae[1]?ae[1].values:null;dt=null!=e,at=dt?e(w,1,0):{_:0};for(let e in at)ht[e]=b}if(lt)if(rt=D("table","u-legend",O),ut=D("tbody",null,rt),tt.mount(w,rt),dt){st=D("thead",null,rt,ut);let e=D("tr",null,st);for(var pt in D("th",null,e),at)D("th",t,e).textContent=pt}else T(rt,"u-inline"),tt.live&&T(rt,"u-live");const mt={show:!0},Mt={show:!1},zt=new Map;function Pt(e,t,l,n=!0){const o=zt.get(t)||{},i=hl.bind[e](w,t,l,n);i&&(L(e,t,o[e]=i),zt.set(t,o))}function Dt(e,t,l){const n=zt.get(t)||{};for(let l in n)null!=e&&l!=e||(F(l,t,n[l]),delete n[l]);null==e&&zt.delete(t)}let Ct=0,Wt=0,Ht=0,It=0,Yt=0,Rt=0,Ut=0,Bt=0,Vt=0,Zt=0;w.bbox={};let qt=!1,Qt=!1,el=!1,rl=!1,ul=!1,al=!1;function fl(e,t,l){(l||e!=w.width||t!=w.height)&&dl(e,t),gn(!1),el=!0,Qt=!0,hl.left>=0&&(rl=al=!0),Dn()}function dl(e,t){w.width=Ct=Ht=e,w.height=Wt=It=t,Yt=Rt=0,function(){let e=!1,t=!1,l=!1,n=!1;xe.forEach(((o,i)=>{if(o.show&&o._show){let{side:i,_size:r}=o,s=i%2,u=r+(null!=o.label?o.labelSize:0);u>0&&(s?(Ht-=u,3==i?(Yt+=u,n=!0):l=!0):(It-=u,0==i?(Rt+=u,e=!0):t=!0))}})),yl[0]=e,yl[1]=l,yl[2]=t,yl[3]=n,Ht-=Ml[1]+Ml[3],Yt+=Ml[3],It-=Ml[2]+Ml[0],Rt+=Ml[0]}(),function(){let e=Yt+Ht,t=Rt+It,l=Yt,n=Rt;function o(o,i){switch(o){case 1:return e+=i,e-i;case 2:return t+=i,t-i;case 3:return l-=i,l+i;case 0:return n-=i,n+i}}xe.forEach(((e,t)=>{if(e.show&&e._show){let t=e.side;e._pos=o(t,e._size),null!=e.label&&(e._lpos=o(t,e.labelSize))}}))}();let l=w.bbox;Ut=l.left=_e(Yt*S,.5),Bt=l.top=_e(Rt*S,.5),Vt=l.width=_e(Ht*S,.5),Zt=l.height=_e(It*S,.5)}w.setSize=function({width:e,height:t}){fl(e,t)};const hl=w.cursor=Le({},At,{drag:{y:2==y}},l.cursor),ml=e=>{hl.event=e};hl.idxs=et,hl._lock=!1;let gl=hl.points;gl.show=ge(gl.show),gl.size=ge(gl.size),gl.stroke=ge(gl.stroke),gl.width=ge(gl.width),gl.fill=ge(gl.fill);const xl=w.focus=Le({},l.focus||{alpha:.3},hl.focus),wl=xl.prox>=0;let vl=[null];function bl(l,n){if(1==y||n>0){let e=1==y&&ke[l.scale].time,t=l.value;l.value=e?Ie(t)?Et(qe,St(t,Je)):t||Qe:t||Xt,l.label=l.label||(e?"Time":"Value")}if(n>0){l.width=null==l.width?1:l.width,l.paths=l.paths||Hl||ve,l.fillTo=ge(l.fillTo||cl),l.pxAlign=+K(l.pxAlign,$),l.pxRound=pl(l.pxAlign),l.stroke=ge(l.stroke||null),l.fill=ge(l.fill||null),l._stroke=l._fill=l._paths=l._focus=null;let e=Ee(1*(3+2*(se(1,l.width)||1)),3),t=l.points=Le({},{size:e,width:se(1,.2*e),stroke:l.stroke,space:2*e,paths:Il,_stroke:null,_fill:null},l.points);t.show=ge(t.show),t.filter=ge(t.filter),t.fill=ge(t.fill),t.stroke=ge(t.stroke),t.paths=ge(t.paths),t.pxAlign=l.pxAlign}if(lt){let o=function(l,n){if(0==n&&(dt||!tt.live||2==y))return Ce;let o=[],i=D("tr","u-series",ut,ut.childNodes[n]);T(i,l.class),l.show||T(i,e);let r=D("th",null,i);if(it.show){let e=A("u-marker",r);if(n>0){let t=it.width(w,n);t&&(e.style.border=t+"px "+it.dash(w,n)+" "+it.stroke(w,n)),e.style.background=it.fill(w,n)}}let s=A(t,r);for(var u in s.textContent=l.label,n>0&&(it.show||(s.style.color=l.width>0?it.stroke(w,n):it.fill(w,n)),Pt("click",r,(e=>{if(hl._lock)return;ml(e);let t=ae.indexOf(l);if((e.ctrlKey||e.metaKey)!=tt.isolate){let e=ae.some(((e,l)=>l>0&&l!=t&&e.show));ae.forEach(((l,n)=>{n>0&&Fn(n,e?n==t?mt:Mt:mt,!0,vo.setSeries)}))}else Fn(t,{show:!l.show},!0,vo.setSeries)}),!1),wl&&Pt(m,r,(e=>{hl._lock||(ml(e),Fn(ae.indexOf(l),Bn,!0,vo.setSeries))}),!1)),at){let e=D("td","u-value",i);e.textContent="--",o.push(e)}return[i,o]}(l,n);ct.splice(n,0,o[0]),ft.splice(n,0,o[1]),tt.values.push(null)}if(hl.show){et.splice(n,0,null);let e=function(e,t){if(t>0){let l=hl.points.show(w,t);if(l)return T(l,"u-cursor-pt"),T(l,e.class),W(l,-10,-10,Ht,It),X.insertBefore(l,vl[t]),l}}(l,n);e&&vl.splice(n,0,e)}xo("addSeries",n)}w.addSeries=function(e,t){t=null==t?ae.length:t,e=1==y?Rl(e,t,jt,ll):Rl(e,t,null,tl),ae.splice(t,0,e),bl(ae[t],t)},w.delSeries=function(e){if(ae.splice(e,1),lt){tt.values.splice(e,1),ft.splice(e,1);let t=ct.splice(e,1)[0];Dt(null,t.firstChild),t.remove()}hl.show&&(et.splice(e,1),vl.length>1&&vl.splice(e,1)[0].remove()),xo("delSeries",e)};const yl=[!1,!1,!1,!1];function kl(e,t,l,n){let[o,i,r,s]=l,u=t%2,a=0;return 0==u&&(s||i)&&(a=0==t&&!o||2==t&&!r?oe(Ot.size/3):0),1==u&&(o||r)&&(a=1==t&&!i||3==t&&!s?oe($t.size/2):0),a}const _l=w.padding=(l.padding||[kl,kl,kl,kl]).map((e=>ge(K(e,kl)))),Ml=w._padding=_l.map(((e,t)=>e(w,t,yl,0)));let Sl,El=null,Tl=null;const zl=1==y?ae[0].idxs:null;let Pl,Dl,Al,Wl,ql,Jl,Kl,Xl,$l,Ql,en=null,tn=!1;function ln(e,t){if(n=null==e?[]:je(e,Re),2==y){Sl=0;for(let e=1;e=0,al=!0,Dn()}}function nn(){let e,t;tn=!0,1==y&&(Sl>0?(El=zl[0]=0,Tl=zl[1]=Sl-1,e=n[0][El],t=n[0][Tl],2==Ue?(e=El,t=Tl):e==t&&(3==Ue?[e,t]=U(e,e,Ge.log,!1):4==Ue?[e,t]=B(e,e,Ge.log,!1):Ge.time?t=e+oe(86400/ne):[e,t]=J(e,t,.1,!0))):(El=zl[0]=e=null,Tl=zl[1]=t=null)),Ln(Pe,e,t)}function on(e,t,l,n,o,i){e??=f,l??=Ae,n??="butt",o??=f,i??="round",e!=Pl&&(G.strokeStyle=Pl=e),o!=Dl&&(G.fillStyle=Dl=o),t!=Al&&(G.lineWidth=Al=t),i!=ql&&(G.lineJoin=ql=i),n!=Jl&&(G.lineCap=Jl=n),l!=Wl&&G.setLineDash(Wl=l)}function rn(e,t,l,n){t!=Dl&&(G.fillStyle=Dl=t),e!=Kl&&(G.font=Kl=e),l!=Xl&&(G.textAlign=Xl=l),n!=$l&&(G.textBaseline=$l=n)}function sn(e,t,l,n,o=0){if(n.length>0&&e.auto(w,tn)&&(null==t||null==t.min)){let t=K(El,0),i=K(Tl,n.length-1),r=null==l.min?3==e.distr?function(e,t,l){let n=he,o=-he;for(let i=t;i<=l;i++){let t=e[i];null!=t&&t>0&&(to&&(o=t))}return[n==he?1:n,o==-he?10:o]}(n,t,i):function(e,t,l,n){let o=he,i=-he;if(1==n)o=e[t],i=e[l];else if(-1==n)o=e[l],i=e[t];else for(let n=t;n<=l;n++){let t=e[n];null!=t&&(ti&&(i=t))}return[o,i]}(n,t,i,o):[l.min,l.max];e.min=re(e.min,l.min=r[0]),e.max=se(e.max,l.max=r[1])}}function un(e,t){let l=t?ae[e].points:ae[e];l._stroke=l.stroke(w,e),l._fill=l.fill(w,e)}function an(e,t){let l=t?ae[e].points:ae[e],o=l._stroke,i=l._fill,{stroke:r,fill:s,clip:u,flags:a}=l._paths,c=null,f=Ee(l.width*S,3),d=f%2/2;t&&null==i&&(i=f>0?"#fff":o);let h=1==l.pxAlign&&d>0;if(h&&G.translate(d,d),!t){let e=Ut-f/2,t=Bt-f/2,l=Vt+f,n=Zt+f;c=new Path2D,c.rect(e,t,l,n)}t?cn(o,f,l.dash,l.cap,i,r,s,a,u):function(e,t,l,o,i,r,s,u,a,c,f){let d=!1;Se.forEach(((h,p)=>{if(h.series[0]==e){let e,m=ae[h.series[1]],g=n[h.series[1]],x=(m._paths||De).band;We(x)&&(x=1==h.dir?x[0]:x[1]);let v=null;m.show&&x&&function(e,t,l){for(t=K(t,0),l=K(l,e.length-1);t<=l;){if(null!=e[t])return!0;t++}return!1}(g,El,Tl)?(v=h.fill(w,p)||r,e=m._paths.clip):x=null,cn(t,l,o,i,v,s,u,a,c,f,e,x),d=!0}})),d||cn(t,l,o,i,r,s,u,a,c,f)}(e,o,f,l.dash,l.cap,i,r,s,a,c,u),h&&G.translate(-d,-d)}function cn(e,t,l,n,o,i,r,s,u,a,c,f){on(e,t,l,n,o),(u||a||f)&&(G.save(),u&&G.clip(u),a&&G.clip(a)),f?3==(3&s)?(G.clip(f),c&&G.clip(c),dn(o,r),fn(e,i,t)):2&s?(dn(o,r),G.clip(f),fn(e,i,t)):1&s&&(G.save(),G.clip(f),c&&G.clip(c),dn(o,r),G.restore(),fn(e,i,t)):(dn(o,r),fn(e,i,t)),(u||a||f)&&G.restore()}function fn(e,t,l){l>0&&(t instanceof Map?t.forEach(((e,t)=>{G.strokeStyle=Pl=t,G.stroke(e)})):null!=t&&e&&G.stroke(t))}function dn(e,t){t instanceof Map?t.forEach(((e,t)=>{G.fillStyle=Dl=t,G.fill(e)})):null!=t&&e&&G.fill(t)}function hn(e,t,l,n,o,i,r,s,u,a){let c=r%2/2;1==$&&G.translate(c,c),on(s,r,u,a,s),G.beginPath();let f,d,h,p,m=o+(0==n||3==n?-i:i);0==l?(d=o,p=m):(f=o,h=m);for(let n=0;n{if(!l.show)return;let o=ke[l.scale];if(null==o.min)return void(l._show&&(t=!1,l._show=!1,gn(!1)));l._show||(t=!1,l._show=!0,gn(!1));let i=l.side,r=i%2,{min:s,max:u}=o,[a,c]=function(e,t,l,n){let o,i=xe[e];if(n<=0)o=[0,0];else{let r=i._space=i.space(w,e,t,l,n);o=Bl(t,l,i._incrs=i.incrs(w,e,t,l,n,r),n,r)}return i._found=o}(n,s,u,0==r?Ht:It);if(0==c)return;let f=2==o.distr,d=l._splits=l.splits(w,n,s,u,a,c,f),h=2==o.distr?d.map((e=>en[e])):d,p=2==o.distr?en[d[1]]-en[d[0]]:a,m=l._values=l.values(w,l.filter(w,h,n,c,p),n,c,p);l._rotate=2==i?l.rotate(w,m,n,c):0;let g=l._size;l._size=ie(l.size(w,m,n,e)),null!=g&&l._size!=g&&(t=!1)})),t}function mn(e){let t=!0;return _l.forEach(((l,n)=>{let o=l(w,n,yl,e);o!=Ml[n]&&(t=!1),Ml[n]=o})),t}function gn(e){ae.forEach(((t,l)=>{l>0&&(t._paths=null,e&&(1==y?(t.min=null,t.max=null):t.facets.forEach((e=>{e.min=null,e.max=null}))))}))}w.setData=ln;let xn,wn,vn,bn,yn,kn,_n,Mn,Sn,En,Tn,zn,Pn=!1;function Dn(){Pn||(Ne(An),Pn=!0)}function An(){qt&&(function(){let e=je(ke,Re);for(let t in e){let l=e[t],n=Ze[t];if(null!=n&&null!=n.min)Le(l,n),t==Pe&&gn(!0);else if(t!=Pe||2==y)if(0==Sl&&null==l.from){let e=l.range(w,null,null,t);l.min=e[0],l.max=e[1]}else l.min=he,l.max=-he}if(Sl>0){ae.forEach(((t,l)=>{if(1==y){let o=t.scale,i=e[o],r=Ze[o];if(0==l){let e=i.range(w,i.min,i.max,o);i.min=e[0],i.max=e[1],El=N(i.min,n[0]),Tl=N(i.max,n[0]),Tl-El>1&&(n[0][El]i.max&&Tl--),t.min=en[El],t.max=en[Tl]}else t.show&&t.auto&&sn(i,r,t,n[l],t.sorted);t.idxs[0]=El,t.idxs[1]=Tl}else if(l>0&&t.show&&t.auto){let[o,i]=t.facets,r=o.scale,s=i.scale,[u,a]=n[l];sn(e[r],Ze[r],o,u,o.sorted),sn(e[s],Ze[s],i,a,i.sorted),t.min=i.min,t.max=i.max}}));for(let t in e){let l=e[t],n=Ze[t];if(null==l.from&&(null==n||null==n.min)){let e=l.range(w,l.min==he?null:l.min,l.max==-he?null:l.max,t);l.min=e[0],l.max=e[1]}}}for(let t in e){let l=e[t];if(null!=l.from){let n=e[l.from];if(null==n.min)l.min=l.max=null;else{let e=l.range(w,n.min,n.max,t);l.min=e[0],l.max=e[1]}}}let t={},l=!1;for(let n in e){let o=e[n],i=ke[n];if(i.min!=o.min||i.max!=o.max){i.min=o.min,i.max=o.max;let e=i.distr;i._min=3==e?ce(i.min):4==e?de(i.min,i.asinh):i.min,i._max=3==e?ce(i.max):4==e?de(i.max,i.asinh):i.max,t[n]=l=!0}}if(l){ae.forEach(((e,l)=>{2==y?l>0&&t.y&&(e._paths=null):t[e.scale]&&(e._paths=null)}));for(let e in t)el=!0,xo("setScale",e);hl.show&&hl.left>=0&&(rl=al=!0)}for(let e in Ze)Ze[e]=null}(),qt=!1),el&&(function(){let e=!1,t=0;for(;!e;){t++;let l=pn(t),n=mn(t);e=3==t||l&&n,e||(dl(w.width,w.height),Qt=!0)}}(),el=!1),Qt&&(P(q,u,Yt),P(q,r,Rt),P(q,o,Ht),P(q,i,It),P(X,u,Yt),P(X,r,Rt),P(X,o,Ht),P(X,i,It),P(Z,o,Ct),P(Z,i,Wt),j.width=oe(Ct*S),j.height=oe(Wt*S),xe.forEach((({_el:t,_show:l,_size:n,_pos:o,side:i})=>{if(null!=t)if(l){let l=i%2==1;P(t,l?"left":"top",o-(3===i||0===i?n:0)),P(t,l?"width":"height",n),P(t,l?"top":"left",l?Rt:Yt),P(t,l?"height":"width",l?It:Ht),z(t,e)}else T(t,e)})),Pl=Dl=Al=ql=Jl=Kl=Xl=$l=Wl=null,Ql=1,to(!0),xo("setSize"),Qt=!1),Ct>0&&Wt>0&&(G.clearRect(0,0,j.width,j.height),xo("drawClear"),Oe.forEach((e=>e())),xo("draw")),Rn.show&&ul&&(jn(Rn),ul=!1),hl.show&&rl&&(Qn(null,!0,!1),rl=!1),tt.show&&tt.live&&al&&(Xn(),al=!1),Y||(Y=!0,w.status=1,xo("ready")),tn=!1,Pn=!1}function Cn(e,t){let l=ke[e];if(null==l.from){if(0==Sl){let n=l.range(w,t.min,t.max,e);t.min=n[0],t.max=n[1]}if(t.min>t.max){let e=t.min;t.min=t.max,t.max=e}if(Sl>1&&null!=t.min&&null!=t.max&&t.max-t.min<1e-16)return;e==Pe&&2==l.distr&&Sl>0&&(t.min=N(t.min,n[0]),t.max=N(t.max,n[0]),t.min==t.max&&t.max++),Ze[e]=t,qt=!0,Dn()}}w.redraw=(e,t)=>{el=t||!1,!1!==e?Ln(Pe,Ge.min,Ge.max):Dn()},w.setScale=Cn;let Wn=!1;const Hn=hl.drag;let In=Hn.x,Yn=Hn.y;hl.show&&(hl.x&&(xn=A("u-cursor-x",X)),hl.y&&(wn=A("u-cursor-y",X)),0==Ge.ori?(vn=xn,bn=wn):(vn=wn,bn=xn),Tn=hl.left,zn=hl.top);const Rn=w.select=Le({show:!0,over:!0,left:0,width:0,top:0,height:0},l.select),On=Rn.show?A("u-select",Rn.over?X:q):null;function jn(e,t){if(Rn.show){for(let t in e)Rn[t]=e[t],t in oo&&P(On,t,e[t]);!1!==t&&xo("setSelect")}}function Ln(e,t,l){Cn(e,{min:t,max:l})}function Fn(t,l,n,o){null!=l.focus&&function(e){if(e!=Un){let t=null==e,l=1!=xl.alpha;ae.forEach(((n,o)=>{let i=t||0==o||o==e;n._focus=t?null:i,l&&function(e,t){ae[e].alpha=t,hl.show&&vl[e]&&(vl[e].style.opacity=t),lt&&ct[e]&&(ct[e].style.opacity=t)}(o,i?1:xl.alpha)})),Un=e,l&&Dn()}}(t),null!=l.show&&ae.forEach(((n,o)=>{o>0&&(t==o||null==t)&&(n.show=l.show,function(t,l){let n=ae[t],o=lt?ct[t]:null;n.show?o&&z(o,e):(o&&T(o,e),vl.length>1&&W(vl[t],-10,-10,Ht,It))}(o,l.show),Ln(2==y?n.facets[1].scale:n.scale,null,null),Dn())})),!1!==n&&xo("setSeries",t,l),o&&ko("setSeries",w,t,l)}let Nn,Gn,Un;w.setSelect=jn,w.setSeries=Fn,w.addBand=function(e,t){e.fill=ge(e.fill||null),e.dir=K(e.dir,-1),t=null==t?Se.length:t,Se.splice(t,0,e)},w.setBand=function(e,t){Le(Se[e],t)},w.delBand=function(e){null==e?Se.length=0:Se.splice(e,1)};const Bn={focus:!0};function Vn(e,t,l){let n=ke[t];l&&(e=e/S-(1==n.ori?Rt:Yt));let o=Ht;1==n.ori&&(o=It,e=o-e),-1==n.dir&&(e=o-e);let i=n._min,r=i+(n._max-i)*(e/o),s=n.distr;return 3==s?ue(10,r):4==s?((e,t=1)=>ee.sinh(e)*t)(r,n.asinh):r}function Zn(e,t){P(On,u,Rn.left=e),P(On,o,Rn.width=t)}function qn(e,t){P(On,r,Rn.top=e),P(On,i,Rn.height=t)}lt&&wl&&Pt(g,rt,(e=>{hl._lock||(ml(e),null!=Un&&Fn(null,Bn,!0,vo.setSeries))})),w.valToIdx=e=>N(e,n[0]),w.posToIdx=function(e,t){return N(Vn(e,Pe,t),n[0],El,Tl)},w.posToVal=Vn,w.valToPos=(e,t,l)=>0==ke[t].ori?E(e,ke[t],l?Vt:Ht,l?Ut:0):C(e,ke[t],l?Zt:It,l?Bt:0),w.batch=function(e){e(w),Dn()},w.setCursor=(e,t,l)=>{Tn=e.left,zn=e.top,Qn(null,t,l)};let Jn=0==Ge.ori?Zn:qn,Kn=1==Ge.ori?Zn:qn;function Xn(e,t){null!=e&&(e.idxs?e.idxs.forEach(((e,t)=>{et[t]=e})):void 0!==e.idx&&et.fill(e.idx),tt.idx=et[0]);for(let e=0;e0||1==y&&!dt)&&$n(e,et[e]);lt&&tt.live&&function(){if(lt&&tt.live)for(let e=2==y?1:0;eTl;Nn=he;let r=0==Ge.ori?Ht:It,s=1==Ge.ori?Ht:It;if(Tn<0||0==Sl||i){o=null;for(let e=0;e0&&vl.length>1&&W(vl[e],-10,-10,Ht,It);wl&&Fn(null,Bn,!0,null==e&&vo.setSeries),tt.live&&(et.fill(o),al=!0)}else{let e,t,l;1==y&&(e=0==Ge.ori?Tn:zn,t=Vn(e,Pe),o=N(t,n[0],El,Tl),l=Be(n[0][o],Ge,r,0));for(let e=2==y?1:0;e0&&i.show){let t,l,n=null==f?-10:Me(Ve(f,1==y?ke[i.scale]:ke[i.facets[1].scale],s,0),1);if(wl&&n>=0&&1==y){let t=le(n-zn);if(t=0?1:-1;o==(f>=0?1:-1)&&(1==o?1==l?f>=n:f<=n:1==l?f<=n:f>=n)&&(Nn=t,Gn=e)}else Nn=t,Gn=e}}if(0==Ge.ori?(t=d,l=n):(t=n,l=d),al&&vl.length>1){I(vl[e],hl.points.fill(w,e),hl.points.stroke(w,e));let n,o,i,r,s=!0,u=hl.points.bbox;if(null!=u){s=!1;let t=u(w,e);i=t.left,r=t.top,n=t.width,o=t.height}else i=t,r=l,n=o=hl.points.size(w,e);R(vl[e],n,o,s),W(vl[e],i,r,Ht,It)}}}}if(hl.idx=o,hl.left=Tn,hl.top=zn,al&&(tt.idx=o,Xn()),Rn.show&&Wn)if(null!=e){let[t,l]=vo.scales,[n,o]=vo.match,[i,u]=e.cursor.sync.scales,a=e.cursor.drag;if(In=a._x,Yn=a._y,In||Yn){let a,c,f,d,h,{left:p,top:m,width:g,height:x}=e.select,w=e.scales[t].ori,v=e.posToVal,b=null!=t&&n(t,i),y=null!=l&&o(l,u);b&&In?(0==w?(a=p,c=g):(a=m,c=x),f=ke[t],d=Be(v(a,i),f,r,0),h=Be(v(a+c,i),f,r,0),Jn(re(d,h),le(h-d))):Jn(0,r),y&&Yn?(1==w?(a=p,c=g):(a=m,c=x),f=ke[l],d=Ve(v(a,u),f,s,0),h=Ve(v(a+c,u),f,s,0),Kn(re(d,h),le(h-d))):Kn(0,s)}else io()}else{let e=le(Sn-yn),t=le(En-kn);if(1==Ge.ori){let l=e;e=t,t=l}In=Hn.x&&e>=Hn.dist,Yn=Hn.y&&t>=Hn.dist;let l,n,o=Hn.uni;null!=o?In&&Yn&&(In=e>=o,Yn=t>=o,In||Yn||(t>e?Yn=!0:In=!0)):Hn.x&&Hn.y&&(In||Yn)&&(In=Yn=!0),In&&(0==Ge.ori?(l=_n,n=Tn):(l=Mn,n=zn),Jn(re(l,n),le(n-l)),Yn||Kn(0,s)),Yn&&(1==Ge.ori?(l=_n,n=Tn):(l=Mn,n=zn),Kn(re(l,n),le(n-l)),In||Jn(0,r)),In||Yn||(Jn(0,0),Kn(0,0))}if(Hn._x=In,Hn._y=Yn,null==e){if(l){if(null!=bo){let[e,t]=vo.scales;vo.values[0]=null!=e?Vn(0==Ge.ori?Tn:zn,e):null,vo.values[1]=null!=t?Vn(1==Ge.ori?Tn:zn,t):null}ko(d,w,Tn,zn,Ht,It,o)}if(wl){let e=l&&vo.setSeries,t=xl.prox;null==Un?Nn<=t&&Fn(Gn,Bn,!0,e):Nn>t?Fn(null,Bn,!0,e):Gn!=Un&&Fn(Gn,Bn,!0,e)}}!1!==t&&xo("setCursor")}w.setLegend=Xn;let eo=null;function to(e=!1){e?eo=null:(eo=X.getBoundingClientRect(),xo("syncRect",eo))}function lo(e,t,l,n,o,i,r){hl._lock||Wn&&null!=e&&0==e.movementX&&0==e.movementY||(no(e,t,l,n,o,i,0,!1,null!=e),null!=e?Qn(null,!0,!0):Qn(t,!0,!1))}function no(e,t,l,n,o,i,r,s,u){if(null==eo&&to(!1),ml(e),null!=e)l=e.clientX-eo.left,n=e.clientY-eo.top;else{if(l<0||n<0)return Tn=-10,void(zn=-10);let[e,r]=vo.scales,s=t.cursor.sync,[u,a]=s.values,[c,f]=s.scales,[d,h]=vo.match,p=t.axes[0].side%2==1,m=0==Ge.ori?Ht:It,g=1==Ge.ori?Ht:It,x=p?i:o,w=p?o:i,v=p?n:l,b=p?l:n;if(l=null!=c?d(e,c)?H(u,ke[e],m,0):-10:m*(v/x),n=null!=f?h(r,f)?H(a,ke[r],g,0):-10:g*(b/w),1==Ge.ori){let e=l;l=n,n=e}}u&&((l<=1||l>=Ht-1)&&(l=_e(l,Ht)),(n<=1||n>=It-1)&&(n=_e(n,It))),s?(yn=l,kn=n,[_n,Mn]=hl.move(w,l,n)):(Tn=l,zn=n)}Object.defineProperty(w,"rect",{get:()=>(null==eo&&to(!1),eo)});const oo={width:0,height:0,left:0,top:0};function io(){jn(oo,!1)}let ro,so,uo,ao;function co(e,t,l,n,o,i,r){Wn=!0,In=Yn=Hn._x=Hn._y=!1,no(e,t,l,n,o,i,0,!0,!1),null!=e&&(Pt(p,k,fo,!1),ko(h,w,_n,Mn,Ht,It,null));let{left:s,top:u,width:a,height:c}=Rn;ro=s,so=u,uo=a,ao=c,io()}function fo(e,t,l,n,o,i,r){Wn=Hn._x=Hn._y=!1,no(e,t,l,n,o,i,0,!1,!0);let{left:s,top:u,width:a,height:c}=Rn,f=a>0||c>0,d=ro!=s||so!=u||uo!=a||ao!=c;if(f&&d&&jn(Rn),Hn.setScale&&f&&d){let e=s,t=a,l=u,n=c;if(1==Ge.ori&&(e=u,t=c,l=s,n=a),In&&Ln(Pe,Vn(e,Pe),Vn(e+t,Pe)),Yn)for(let e in ke){let t=ke[e];e!=Pe&&null==t.from&&t.min!=he&&Ln(e,Vn(l+n,e),Vn(l,e))}io()}else hl.lock&&(hl._lock=!hl._lock,hl._lock||Qn(null,!0,!1));null!=e&&(Dt(p,k),ko(p,w,Tn,zn,Ht,It,null))}function ho(e,t,l,n,o,i,r){hl._lock||(ml(e),nn(),io(),null!=e&&ko(x,w,Tn,zn,Ht,It,null))}function po(){xe.forEach(Zl),fl(w.width,w.height,!0)}L(v,_,po);const mo={};mo.mousedown=co,mo.mousemove=lo,mo.mouseup=fo,mo.dblclick=ho,mo.setSeries=(e,t,l,n)=>{-1!=(l=(0,vo.match[2])(w,t,l))&&Fn(l,n,!0,!1)},hl.show&&(Pt(h,X,co),Pt(d,X,lo),Pt(m,X,(e=>{ml(e),to(!1)})),Pt(g,X,(function(e,t,l,n,o,i,r){if(hl._lock)return;ml(e);let s=Wn;if(Wn){let e,t,l=!0,n=!0,o=10;0==Ge.ori?(e=In,t=Yn):(e=Yn,t=In),e&&t&&(l=Tn<=o||Tn>=Ht-o,n=zn<=o||zn>=It-o),e&&l&&(Tn=Tn<_n?0:Ht),t&&n&&(zn=zn{e.call(null,w,t,l)}))}(l.plugins||[]).forEach((e=>{for(let t in e.hooks)go[t]=(go[t]||[]).concat(e.hooks[t])}));const wo=(e,t,l)=>l,vo=Le({key:null,setSeries:!1,filters:{pub:be,sub:be},scales:[Pe,ae[1]?ae[1].scale:null],match:[ye,ye,wo],values:[null,null]},hl.sync);2==vo.match.length&&vo.match.push(wo),hl.sync=vo;const bo=vo.key,yo=sl(bo);function ko(e,t,l,n,o,i,r){vo.filters.pub(e,t,l,n,o,i,r)&&yo.pub(e,t,l,n,o,i,r)}function _o(){xo("init",l,n),ln(n||l.data,!1),Ze[Pe]?Cn(Pe,Ze[Pe]):nn(),ul=Rn.show,rl=al=!0,fl(l.width,l.height)}return yo.sub(w),w.pub=function(e,t,l,n,o,i,r){vo.filters.sub(e,t,l,n,o,i,r)&&mo[e](null,t,l,n,o,i,r)},w.destroy=function(){yo.unsub(w),Cl.delete(w),zt.clear(),F(v,_,po),O.remove(),rt?.remove(),xo("destroy")},ae.forEach(bl),xe.forEach((function(e,t){if(e._show=e.show,e.show){let l=e.side%2,n=ke[e.scale];null==n&&(e.scale=l?ae[1].scale:Pe,n=ke[e.scale]);let o=n.time;e.size=ge(e.size),e.space=ge(e.space),e.rotate=ge(e.rotate),We(e.incrs)&&e.incrs.forEach((e=>{!Te.has(e)&&Te.set(e,ze(e))})),e.incrs=ge(e.incrs||(2==n.distr?nt:o?1==ne?gt:vt:ot)),e.splits=ge(e.splits||(o&&1==n.distr?Ke:3==n.distr?Nt:4==n.distr?Gt:Ft)),e.stroke=ge(e.stroke),e.grid.stroke=ge(e.grid.stroke),e.ticks.stroke=ge(e.ticks.stroke),e.border.stroke=ge(e.border.stroke);let i=e.values;e.values=We(i)&&!We(i[0])?ge(i):o?We(i)?_t(qe,kt(i,Je)):Ie(i)?function(e,t){let l=Xe(t);return(t,n,o,i,r)=>n.map((t=>l(e(t))))}(qe,i):i||$e:i||Lt,e.filter=ge(e.filter||(n.distr>=3&&10==n.log?Jt:3==n.distr&&2==n.log?Kt:we)),e.font=Vl(e.font),e.labelFont=Vl(e.labelFont),e._size=e.size(w,null,t,0),e._space=e._rotate=e._incrs=e._found=e._splits=e._values=null,e._size>0&&(yl[t]=!0,e._el=A("u-axis",Z))}})),c?c instanceof HTMLElement?(c.appendChild(O),_o()):c(w,_o):_o(),w}ql.assign=Le,ql.fmtNum=Q,ql.rangeNum=J,ql.rangeLog=U,ql.rangeAsinh=B,ql.orient=ul,ql.pxRatio=S,ql.join=function(e,t){if(function(e){let t=e[0][0],l=t.length;for(let n=1;n=n&&null==e[o];)o--;if(o<=n)return!0;const i=se(1,ne((o-n+1)/t));for(let t=e[n],l=n+i;l<=o;l+=i){const n=e[l];if(null!=n){if(n<=t)return!1;t=n}}return!0}(t[0])||(t=function(e){let t=e[0],l=t.length,n=Array(l);for(let e=0;et[e]-t[l]));let o=[];for(let t=0;te-t))],o=n[0].length,i=new Map;for(let e=0;eul(e,i,((u,a,c,f,d,h,p,m,g,x,w)=>{let v=u.pxRound,{left:b,width:y}=e.bbox,k=e=>v(h(e,f,x,m)),_=e=>v(p(e,d,w,g)),M=0==f.ori?wl:vl;const E={stroke:new Path2D,fill:null,clip:null,band:null,gaps:null,flags:1},T=E.stroke,z=f.dir*(0==f.ori?1:-1);r=G(c,r,s,1),s=G(c,r,s,-1);let P=_(c[1==z?r:s]),D=k(a[1==z?r:s]),A=D,C=D;o&&-1==t&&(C=b,M(T,C,P)),M(T,D,P);for(let e=1==z?r:s;e>=r&&e<=s;e+=z){let l=c[e];if(null==l)continue;let n=k(a[e]),o=_(l);1==t?M(T,n,P):M(T,A,o),M(T,n,o),P=o,A=n}let W=A;o&&1==t&&(W=b+y,M(T,W,P));let[H,I]=al(e,i);if(null!=u.fill||0!=H){let t=E.fill=new Path2D(T),l=_(u.fillTo(e,i,u.min,u.max,H));M(t,W,l),M(t,C,l)}if(!u.spanGaps){let o=[];o.push(...hl(a,c,r,s,z,k,n));let d=u.width*S/2,h=l||1==t?d:-d,p=l||-1==t?-d:d;o.forEach((e=>{e[0]+=h,e[1]+=p})),E.gaps=o=u.gaps(e,i,r,s,o),E.clip=dl(o,f.ori,m,g,x,w)}return 0!=I&&(E.band=2==I?[fl(e,i,r,s,T,-1),fl(e,i,r,s,T,1)]:fl(e,i,r,s,T,I)),E}))},e.bars=function(e){const t=K((e=e||De).size,[.6,he,1]),l=e.align||0,n=(e.gap||0)*S;let o=e.radius;o=null==o?[0,0]:"number"==typeof o?[o,0]:o;const i=ge(o),r=1-t[0],s=K(t[1],he)*S,u=K(t[2],1)*S,a=K(e.disp,De),c=K(e.each,(e=>{})),{fill:f,stroke:d}=a;return(e,t,o,h)=>ul(e,t,((p,m,g,x,w,v,b,y,k,_,M)=>{let E,T,z=p.pxRound;0==x.ori?[E,T]=i(e,t):[T,E]=i(e,t);const P=x.dir*(0==x.ori?1:-1),D=w.dir*(1==w.ori?1:-1);let A,C,W=0==x.ori?bl:yl,H=0==x.ori?c:(e,t,l,n,o,i,r)=>{c(e,t,l,o,n,r,i)},[I,Y]=al(e,t),R=3==w.distr?1==I?w.max:w.min:0,O=b(R,w,M,k),j=z(p.width*S),L=!1,F=null,N=null,G=null,U=null;null==f||0!=j&&null==d||(L=!0,F=f.values(e,t,o,h),N=new Map,new Set(F).forEach((e=>{null!=e&&N.set(e,new Path2D)})),j>0&&(G=d.values(e,t,o,h),U=new Map,new Set(G).forEach((e=>{null!=e&&U.set(e,new Path2D)}))));let{x0:B,size:V}=a,Z=!0;if(null!=B&&null!=V){m=B.values(e,t,o,h),2==B.unit&&(m=m.map((t=>e.posToVal(y+t*_,x.key,!0))));let l=V.values(e,t,o,h);C=2==V.unit?l[0]*_:v(l[0],x,_,y)-v(0,x,_,y),j>=C&&(j=0),C=z(me(C-j,u,s)),A=1==P?-j/2:C+j/2}else{let e=_;if(m.length>1){let t=null;for(let l=0,n=1/0;l=C&&(j=0),t+n<5&&(z=xe),C=z(me(e-t,u,s)-j-n),A=(0==l?C/2:l==P?0:C)-l*P*n/2,C+j>e&&(Z=!1)}const q={stroke:null,fill:null,clip:null,band:null,gaps:null,flags:3};let J;0!=Y&&(q.band=new Path2D,J=z(b(1==Y?w.max:w.min,w,M,k)));const X=L?null:new Path2D,$=q.band;let{y0:Q,y1:ee}=a,te=null;null!=Q&&null!=ee&&(g=ee.values(e,t,o,h),te=Q.values(e,t,o,h));let oe=E*C,ie=T*C;for(let l=1==P?o:h;l>=o&&l<=h;l+=P){let n=g[l];if(void 0===n)continue;let o=v(2!=x.distr||null!=a?m[l]:l,x,_,y),i=b(K(n,R),w,M,k);null!=te&&null!=n&&(O=b(te[l],w,M,k));let r=z(o-A),s=z(se(i,O)),u=z(re(i,O)),c=s-u;if(null!=n){let o=n<0?ie:oe,i=n<0?oe:ie;L?(j>0&&null!=G[l]&&W(U.get(G[l]),r,u+ne(j/2),C,se(0,c-j),o,i),null!=F[l]&&W(N.get(F[l]),r,u+ne(j/2),C,se(0,c-j),o,i)):W(X,r,u+ne(j/2),C,se(0,c-j),o,i),H(e,t,l,r-j/2,u,C+j,c)}0==Y||null==n&&!Z||(D*Y==1?(s=u,u=J):(u=s,s=J),c=s-u,W($,r-j/2,u,C+j,se(0,c),0,0))}return j>0&&(q.stroke=L?U:X),q.fill=L?N:X,q}))},e.spline=function(e){return function(e,t){const l=K(t?.alignGaps,0);return(t,n,o,i)=>ul(t,n,((r,s,u,a,c,f,d,h,p,m,g)=>{let x,w,v,b=r.pxRound,y=e=>b(f(e,a,m,h)),k=e=>b(d(e,c,g,p));0==a.ori?(x=gl,v=wl,w=Ml):(x=xl,v=vl,w=Sl);const _=a.dir*(0==a.ori?1:-1);o=G(u,o,i,1),i=G(u,o,i,-1);let M=y(s[1==_?o:i]),S=M,E=[],T=[];for(let e=1==_?o:i;e>=o&&e<=i;e+=_)if(null!=u[e]){let t=y(s[e]);E.push(S=t),T.push(k(u[e]))}const z={stroke:e(E,T,x,v,w,b),fill:null,clip:null,band:null,gaps:null,flags:1},P=z.stroke;let[D,A]=al(t,n);if(null!=r.fill||0!=D){let e=z.fill=new Path2D(P),l=k(r.fillTo(t,n,r.min,r.max,D));v(e,S,l),v(e,M,l)}if(!r.spanGaps){let e=[];e.push(...hl(s,u,o,i,_,y,l)),z.gaps=e=r.gaps(t,n,o,i,e),z.clip=dl(e,a.ori,h,p,m,g)}return 0!=A&&(z.band=2==A?[fl(t,n,o,i,P,-1),fl(t,n,o,i,P,1)]:fl(t,n,o,i,P,A)),z}))}(Al,e)}}var Jl=l(379),Kl=l.n(Jl),Xl=l(795),$l=l.n(Xl),Ql=l(569),en=l.n(Ql),tn=l(565),ln=l.n(tn),nn=l(216),on=l.n(nn),rn=l(589),sn=l.n(rn),un=l(350),an={};function cn(e){var t,l=HTMLWidgets.find("#"+e);return void 0!==l&&(t=l.getWidget()),t}function fn(e,t){let l=[],n=[],o=e[0].length,i=Array(o);for(let e=0;ei[t]+=+e)));for(let l=1;ln>l&&!t(n))),l]});return n=n.filter((e=>e.series[1]>-1)),{data:[e[0]].concat(l),bands:n}}an.styleTagTransform=sn(),an.setAttributes=ln(),an.insert=en().bind(null,"head"),an.domAPI=$l(),an.insertStyleElement=on(),Kl()(un.Z,an),un.Z&&un.Z.locals&&un.Z.locals;const dn=e=>(t,l)=>{new ResizeObserver((e=>{for(let l of e){let e=l.contentRect.height,n=l.contentRect.width;const o=t.root.querySelectorAll(".u-legend, .u-title");for(let t of o)e-=Math.floor(t.offsetHeight);t.setSize({width:n,height:e})}})).observe(e),e.appendChild(t.root),l()},hn=(e,t,l,n)=>{const o=5*devicePixelRatio;return ql.orient(e,t,((l,n,i,r,s,u,a,c,f,d,h,p,m,g,x)=>{let w=e.data[t];e.ctx.fillStyle=l.stroke();let v=2*Math.PI;console.time("points");let b=new Path2D;for(let e=0;e=r.min&&t<=r.max&&l>=s.min&&l<=s.max){let e=u(t,r,d,c),n=a(l,s,h,f);b.moveTo(e+o/2,n),x(b,e,n,o/2,0,v)}}console.timeEnd("points"),e.ctx.fill(b)})),null};HTMLWidgets.widget({name:"uPlot",type:"output",factory:function(e,t,l){var n,o,i;return{renderValue:function(r){void 0!==n&&n.destroy(),(o=r.options).width=t,o.height=l,i=r.data,r.stacked?(o.hooks||(o.hooks={}),o.hooks.init=[e=>{[...e.root.querySelectorAll(".u-legend .u-series")].forEach(((t,l)=>{e.series[l]._hide&&(t.style.display="none")}))}],n=function(e,t,l,n,o,i,r,s){let{opts:u,data:a}=function(e,t,l,n){let o=function(e,t){return{scales:{x:{time:!0}},series:t}}(0,t),i=fn(n?n(l):l,(e=>!1));return o.bands=i.bands,o.cursor=o.cursor||{},o.cursor.dataIdx=(e,t,n,o)=>null===l[t][n]?null:n,o.series.forEach((e=>{e.value=(e,t,n,o)=>l[n][o],e.points=e.points||{},e.points.filter=(e,t,n,o)=>{if(n){let e=[];return l[t].forEach(((t,l)=>{null!==t&&e.push(l)})),e}}})),o.scales.y={range:(e,t,l)=>{let n=uPlot.rangeNum(t,l,.1,!0);return[n[0],n[1]]}},o.hooks={setSeries:[(e,t)=>{let n=fn(l,(t=>!e.series[t].show));e.delBand(null),n.bands.forEach((t=>e.addBand(t))),e.setData(n.data)}]},{opts:o,data:i.data}}(0,t,l,n);return u.title=e,u.width=o,u.height=i,u.hooks=Object.assign(u.hooks,r),console.log(u),new uPlot(u,a,s)}(o.title,o.series,i,null,t,l,o.hooks,dn(e))):n=new ql(o,i,dn(e))},getWidget:function(){return n},resize:function(e,t){}}}}),HTMLWidgets.shinyMode&&(Shiny.addCustomMessageHandler("uplot-api",(function(e){var t=cn(e.id);void 0!==t&&t[e.name].apply(null,e.args)})),Shiny.addCustomMessageHandler("uplot-setData",(function(e){var t=cn(e.id);void 0!==t&&t.setData(e.data)})))})();var o=window;for(var i in n)o[i]=n[i];n.__esModule&&Object.defineProperty(o,"__esModule",{value:!0})})(); \ No newline at end of file diff --git a/man/uPlot.Rd b/man/uPlot.Rd index 0cda700..e307e60 100644 --- a/man/uPlot.Rd +++ b/man/uPlot.Rd @@ -4,7 +4,14 @@ \alias{uPlot} \title{uPlot} \usage{ -uPlot(data, options, ..., width = NULL, height = NULL, elementId = NULL) +uPlot( + data, + options = list(), + ..., + width = NULL, + height = NULL, + elementId = NULL +) } \arguments{ \item{data}{Data to plot as a \code{list} (must match \code{µPlot.js} expectations) or diff --git a/package-lock.json b/package-lock.json index d07e0b5..79e9465 100644 --- a/package-lock.json +++ b/package-lock.json @@ -11,7 +11,7 @@ "devDependencies": { "css-loader": "^6.7.1", "style-loader": "^3.3.1", - "uplot": "^1.6.22", + "uplot": "^1.6.26", "webpack": "^5.73.0", "webpack-cli": "^4.10.0", "webpack-merge": "^5.8.0" @@ -1370,9 +1370,9 @@ } }, "node_modules/uplot": { - "version": "1.6.22", - "resolved": "https://registry.npmjs.org/uplot/-/uplot-1.6.22.tgz", - "integrity": "sha512-2jtSb/YHUgtmIUn0+QJjf7ggcJicb5PKe7ijBiRDTPsG/f8F/MFayZ+g6/0kATNkDyF/qQsHJDmCp6cxncg1EQ==", + "version": "1.6.26", + "resolved": "https://registry.npmjs.org/uplot/-/uplot-1.6.26.tgz", + "integrity": "sha512-qN0mveL6UsP40TnHzHAJkUQvpfA3y8zSLXtXKVlJo/sLfj2+vjan/Z3g81MCZjy/hEDUFNtnLftPmETDA4s7Rg==", "dev": true }, "node_modules/uri-js": { @@ -2543,9 +2543,9 @@ } }, "uplot": { - "version": "1.6.22", - "resolved": "https://registry.npmjs.org/uplot/-/uplot-1.6.22.tgz", - "integrity": "sha512-2jtSb/YHUgtmIUn0+QJjf7ggcJicb5PKe7ijBiRDTPsG/f8F/MFayZ+g6/0kATNkDyF/qQsHJDmCp6cxncg1EQ==", + "version": "1.6.26", + "resolved": "https://registry.npmjs.org/uplot/-/uplot-1.6.26.tgz", + "integrity": "sha512-qN0mveL6UsP40TnHzHAJkUQvpfA3y8zSLXtXKVlJo/sLfj2+vjan/Z3g81MCZjy/hEDUFNtnLftPmETDA4s7Rg==", "dev": true }, "uri-js": { diff --git a/package.json b/package.json index c81627a..7b594ed 100644 --- a/package.json +++ b/package.json @@ -24,7 +24,7 @@ "devDependencies": { "css-loader": "^6.7.1", "style-loader": "^3.3.1", - "uplot": "^1.6.22", + "uplot": "^1.6.26", "webpack": "^5.73.0", "webpack-cli": "^4.10.0", "webpack-merge": "^5.8.0"