@excalidraw/excalidraw 0.17.0 → 0.17.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -2419,7 +2419,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
2419
2419
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
2420
2420
 
2421
2421
  "use strict";
2422
- eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"trackEvent\": () => (/* binding */ trackEvent)\n/* harmony export */ });\nconst trackEvent = (category, action, label, value) => {\n try {\n // place here categories that you want to track as events\n // KEEP IN MIND THE PRICING\n const ALLOWED_CATEGORIES_TO_TRACK = []; // Uncomment the next line to track locally\n // console.log(\"Track Event\", { category, action, label, value });\n\n if (typeof window === \"undefined\" || ({\"VITE_APP_BACKEND_V2_GET_URL\":\"https://json-dev.excalidraw.com/api/v2/\",\"VITE_APP_BACKEND_V2_POST_URL\":\"https://json-dev.excalidraw.com/api/v2/post/\",\"VITE_APP_LIBRARY_URL\":\"https://libraries.excalidraw.com\",\"VITE_APP_LIBRARY_BACKEND\":\"https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries\",\"VITE_APP_WS_SERVER_URL\":\"http://localhost:3002\",\"VITE_APP_PORTAL_URL\":\"\",\"VITE_APP_PLUS_LP\":\"https://plus.excalidraw.com\",\"VITE_APP_PLUS_APP\":\"https://app.excalidraw.com\",\"VITE_APP_FIREBASE_CONFIG\":\"{\\\"apiKey\\\":\\\"AIzaSyCMkxA60XIW8KbqMYL7edC4qT5l4qHX2h8\\\",\\\"authDomain\\\":\\\"excalidraw-oss-dev.firebaseapp.com\\\",\\\"projectId\\\":\\\"excalidraw-oss-dev\\\",\\\"storageBucket\\\":\\\"excalidraw-oss-dev.appspot.com\\\",\\\"messagingSenderId\\\":\\\"664559512677\\\",\\\"appId\\\":\\\"1:664559512677:web:a385181f2928d328a7aa8c\\\"}\",\"VITE_APP_DEV_ENABLE_SW\":\"\",\"VITE_APP_DEV_DISABLE_LIVE_RELOAD\":\"\",\"VITE_APP_DISABLE_TRACKING\":\"true\",\"FAST_REFRESH\":\"false\",\"VITE_APP_PORT\":\"3000\",\"VITE_APP_DEBUG_ENABLE_TEXT_CONTAINER_BOUNDING_BOX\":\"\",\"VITE_APP_COLLAPSE_OVERLAY\":\"true\",\"VITE_APP_ENABLE_ESLINT\":\"true\",\"VITE_PKG_NAME\":\"@excalidraw/excalidraw\",\"VITE_PKG_VERSION\":\"0.17.0\",\"VITE_IS_EXCALIDRAW_NPM_PACKAGE\":true}).VITE_WORKER_ID) {\n return;\n }\n\n if (!ALLOWED_CATEGORIES_TO_TRACK.includes(category)) {\n return;\n }\n\n if (window.sa_event) {\n window.sa_event(action, {\n category,\n label,\n value\n });\n }\n } catch (error) {\n console.error(\"error during analytics\", error);\n }\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi4vLi4vYW5hbHl0aWNzLnRzLmpzIiwibWFwcGluZ3MiOiI7Ozs7QUFBTztBQUNQO0FBQ0E7QUFDQTtBQUNBLDRDQUE0QztBQUM1QyxvQ0FBb0MsZ0NBQWdDOztBQUVwRSx5Q0FBeUMsc3FDQUFXO0FBQ3BEO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxPQUFPO0FBQ1A7QUFDQSxJQUFJO0FBQ0o7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vLy4uLy4uL2FuYWx5dGljcy50cz81NDIxIl0sInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjb25zdCB0cmFja0V2ZW50ID0gKGNhdGVnb3J5LCBhY3Rpb24sIGxhYmVsLCB2YWx1ZSkgPT4ge1xuICB0cnkge1xuICAgIC8vIHBsYWNlIGhlcmUgY2F0ZWdvcmllcyB0aGF0IHlvdSB3YW50IHRvIHRyYWNrIGFzIGV2ZW50c1xuICAgIC8vIEtFRVAgSU4gTUlORCBUSEUgUFJJQ0lOR1xuICAgIGNvbnN0IEFMTE9XRURfQ0FURUdPUklFU19UT19UUkFDSyA9IFtdOyAvLyBVbmNvbW1lbnQgdGhlIG5leHQgbGluZSB0byB0cmFjayBsb2NhbGx5XG4gICAgLy8gY29uc29sZS5sb2coXCJUcmFjayBFdmVudFwiLCB7IGNhdGVnb3J5LCBhY3Rpb24sIGxhYmVsLCB2YWx1ZSB9KTtcblxuICAgIGlmICh0eXBlb2Ygd2luZG93ID09PSBcInVuZGVmaW5lZFwiIHx8IHByb2Nlc3MuZW52LlZJVEVfV09SS0VSX0lEKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKCFBTExPV0VEX0NBVEVHT1JJRVNfVE9fVFJBQ0suaW5jbHVkZXMoY2F0ZWdvcnkpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKHdpbmRvdy5zYV9ldmVudCkge1xuICAgICAgd2luZG93LnNhX2V2ZW50KGFjdGlvbiwge1xuICAgICAgICBjYXRlZ29yeSxcbiAgICAgICAgbGFiZWwsXG4gICAgICAgIHZhbHVlXG4gICAgICB9KTtcbiAgICB9XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgY29uc29sZS5lcnJvcihcImVycm9yIGR1cmluZyBhbmFseXRpY3NcIiwgZXJyb3IpO1xuICB9XG59OyJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///../../analytics.ts\n");
2422
+ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"trackEvent\": () => (/* binding */ trackEvent)\n/* harmony export */ });\nconst trackEvent = (category, action, label, value) => {\n try {\n // place here categories that you want to track as events\n // KEEP IN MIND THE PRICING\n const ALLOWED_CATEGORIES_TO_TRACK = []; // Uncomment the next line to track locally\n // console.log(\"Track Event\", { category, action, label, value });\n\n if (typeof window === \"undefined\" || ({\"VITE_APP_BACKEND_V2_GET_URL\":\"https://json-dev.excalidraw.com/api/v2/\",\"VITE_APP_BACKEND_V2_POST_URL\":\"https://json-dev.excalidraw.com/api/v2/post/\",\"VITE_APP_LIBRARY_URL\":\"https://libraries.excalidraw.com\",\"VITE_APP_LIBRARY_BACKEND\":\"https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries\",\"VITE_APP_WS_SERVER_URL\":\"http://localhost:3002\",\"VITE_APP_PORTAL_URL\":\"\",\"VITE_APP_PLUS_LP\":\"https://plus.excalidraw.com\",\"VITE_APP_PLUS_APP\":\"https://app.excalidraw.com\",\"VITE_APP_FIREBASE_CONFIG\":\"{\\\"apiKey\\\":\\\"AIzaSyCMkxA60XIW8KbqMYL7edC4qT5l4qHX2h8\\\",\\\"authDomain\\\":\\\"excalidraw-oss-dev.firebaseapp.com\\\",\\\"projectId\\\":\\\"excalidraw-oss-dev\\\",\\\"storageBucket\\\":\\\"excalidraw-oss-dev.appspot.com\\\",\\\"messagingSenderId\\\":\\\"664559512677\\\",\\\"appId\\\":\\\"1:664559512677:web:a385181f2928d328a7aa8c\\\"}\",\"VITE_APP_DEV_ENABLE_SW\":\"\",\"VITE_APP_DEV_DISABLE_LIVE_RELOAD\":\"\",\"VITE_APP_DISABLE_TRACKING\":\"true\",\"FAST_REFRESH\":\"false\",\"VITE_APP_PORT\":\"3000\",\"VITE_APP_DEBUG_ENABLE_TEXT_CONTAINER_BOUNDING_BOX\":\"\",\"VITE_APP_COLLAPSE_OVERLAY\":\"true\",\"VITE_APP_ENABLE_ESLINT\":\"true\",\"VITE_PKG_NAME\":\"@excalidraw/excalidraw\",\"VITE_PKG_VERSION\":\"0.17.1\",\"VITE_IS_EXCALIDRAW_NPM_PACKAGE\":true}).VITE_WORKER_ID) {\n return;\n }\n\n if (!ALLOWED_CATEGORIES_TO_TRACK.includes(category)) {\n return;\n }\n\n if (window.sa_event) {\n window.sa_event(action, {\n category,\n label,\n value\n });\n }\n } catch (error) {\n console.error(\"error during analytics\", error);\n }\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi4vLi4vYW5hbHl0aWNzLnRzLmpzIiwibWFwcGluZ3MiOiI7Ozs7QUFBTztBQUNQO0FBQ0E7QUFDQTtBQUNBLDRDQUE0QztBQUM1QyxvQ0FBb0MsZ0NBQWdDOztBQUVwRSx5Q0FBeUMsc3FDQUFXO0FBQ3BEO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxPQUFPO0FBQ1A7QUFDQSxJQUFJO0FBQ0o7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vLy4uLy4uL2FuYWx5dGljcy50cz81NDIxIl0sInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjb25zdCB0cmFja0V2ZW50ID0gKGNhdGVnb3J5LCBhY3Rpb24sIGxhYmVsLCB2YWx1ZSkgPT4ge1xuICB0cnkge1xuICAgIC8vIHBsYWNlIGhlcmUgY2F0ZWdvcmllcyB0aGF0IHlvdSB3YW50IHRvIHRyYWNrIGFzIGV2ZW50c1xuICAgIC8vIEtFRVAgSU4gTUlORCBUSEUgUFJJQ0lOR1xuICAgIGNvbnN0IEFMTE9XRURfQ0FURUdPUklFU19UT19UUkFDSyA9IFtdOyAvLyBVbmNvbW1lbnQgdGhlIG5leHQgbGluZSB0byB0cmFjayBsb2NhbGx5XG4gICAgLy8gY29uc29sZS5sb2coXCJUcmFjayBFdmVudFwiLCB7IGNhdGVnb3J5LCBhY3Rpb24sIGxhYmVsLCB2YWx1ZSB9KTtcblxuICAgIGlmICh0eXBlb2Ygd2luZG93ID09PSBcInVuZGVmaW5lZFwiIHx8IHByb2Nlc3MuZW52LlZJVEVfV09SS0VSX0lEKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKCFBTExPV0VEX0NBVEVHT1JJRVNfVE9fVFJBQ0suaW5jbHVkZXMoY2F0ZWdvcnkpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKHdpbmRvdy5zYV9ldmVudCkge1xuICAgICAgd2luZG93LnNhX2V2ZW50KGFjdGlvbiwge1xuICAgICAgICBjYXRlZ29yeSxcbiAgICAgICAgbGFiZWwsXG4gICAgICAgIHZhbHVlXG4gICAgICB9KTtcbiAgICB9XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgY29uc29sZS5lcnJvcihcImVycm9yIGR1cmluZyBhbmFseXRpY3NcIiwgZXJyb3IpO1xuICB9XG59OyJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///../../analytics.ts\n");
2423
2423
 
2424
2424
  /***/ }),
2425
2425
 
@@ -3871,7 +3871,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
3871
3871
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
3872
3872
 
3873
3873
  "use strict";
3874
- eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"ElementBounds\": () => (/* binding */ ElementBounds),\n/* harmony export */ \"getArrowheadPoints\": () => (/* binding */ getArrowheadPoints),\n/* harmony export */ \"getBoundsFromPoints\": () => (/* binding */ getBoundsFromPoints),\n/* harmony export */ \"getClosestElementBounds\": () => (/* binding */ getClosestElementBounds),\n/* harmony export */ \"getCommonBoundingBox\": () => (/* binding */ getCommonBoundingBox),\n/* harmony export */ \"getCommonBounds\": () => (/* binding */ getCommonBounds),\n/* harmony export */ \"getCurvePathOps\": () => (/* binding */ getCurvePathOps),\n/* harmony export */ \"getDiamondPoints\": () => (/* binding */ getDiamondPoints),\n/* harmony export */ \"getDraggedElementsBounds\": () => (/* binding */ getDraggedElementsBounds),\n/* harmony export */ \"getElementAbsoluteCoords\": () => (/* binding */ getElementAbsoluteCoords),\n/* harmony export */ \"getElementBounds\": () => (/* binding */ getElementBounds),\n/* harmony export */ \"getElementLineSegments\": () => (/* binding */ getElementLineSegments),\n/* harmony export */ \"getElementPointsCoords\": () => (/* binding */ getElementPointsCoords),\n/* harmony export */ \"getMinMaxXYFromCurvePathOps\": () => (/* binding */ getMinMaxXYFromCurvePathOps),\n/* harmony export */ \"getRectangleBoxAbsoluteCoords\": () => (/* binding */ getRectangleBoxAbsoluteCoords),\n/* harmony export */ \"getResizedElementAbsoluteCoords\": () => (/* binding */ getResizedElementAbsoluteCoords),\n/* harmony export */ \"pointRelativeTo\": () => (/* binding */ pointRelativeTo)\n/* harmony export */ });\n/* harmony import */ var _math__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../math */ \"../../math.ts\");\n/* harmony import */ var roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! roughjs/bin/rough */ \"../../../node_modules/roughjs/bin/rough.js\");\n/* harmony import */ var _scene_Shape__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../scene/Shape */ \"../../scene/Shape.ts\");\n/* harmony import */ var _typeChecks__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./typeChecks */ \"../../element/typeChecks.ts\");\n/* harmony import */ var _points__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../points */ \"../../points.ts\");\n/* harmony import */ var _textElement__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./textElement */ \"../../element/textElement.ts\");\n/* harmony import */ var _linearElementEditor__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./linearElementEditor */ \"../../element/linearElementEditor.ts\");\n/* harmony import */ var _scene_ShapeCache__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../scene/ShapeCache */ \"../../scene/ShapeCache.ts\");\n\n\n\n\n\n\n\n\nclass ElementBounds {\n static getBounds(element) {\n const cachedBounds = ElementBounds.boundsCache.get(element);\n\n if ((cachedBounds === null || cachedBounds === void 0 ? void 0 : cachedBounds.version) && cachedBounds.version === element.version) {\n return cachedBounds.bounds;\n }\n\n const bounds = ElementBounds.calculateBounds(element);\n ElementBounds.boundsCache.set(element, {\n version: element.version,\n bounds\n });\n return bounds;\n }\n\n static calculateBounds(element) {\n let bounds;\n const [x1, y1, x2, y2, cx, cy] = getElementAbsoluteCoords(element);\n\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isFreeDrawElement)(element)) {\n const [minX, minY, maxX, maxY] = getBoundsFromPoints(element.points.map(([x, y]) => (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x, y, cx - element.x, cy - element.y, element.angle)));\n return [minX + element.x, minY + element.y, maxX + element.x, maxY + element.y];\n } else if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isLinearElement)(element)) {\n bounds = getLinearElementRotatedBounds(element, cx, cy);\n } else if (element.type === \"diamond\") {\n const [x11, y11] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(cx, y1, cx, cy, element.angle);\n const [x12, y12] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(cx, y2, cx, cy, element.angle);\n const [x22, y22] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x1, cy, cx, cy, element.angle);\n const [x21, y21] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x2, cy, cx, cy, element.angle);\n const minX = Math.min(x11, x12, x22, x21);\n const minY = Math.min(y11, y12, y22, y21);\n const maxX = Math.max(x11, x12, x22, x21);\n const maxY = Math.max(y11, y12, y22, y21);\n bounds = [minX, minY, maxX, maxY];\n } else if (element.type === \"ellipse\") {\n const w = (x2 - x1) / 2;\n const h = (y2 - y1) / 2;\n const cos = Math.cos(element.angle);\n const sin = Math.sin(element.angle);\n const ww = Math.hypot(w * cos, h * sin);\n const hh = Math.hypot(h * cos, w * sin);\n bounds = [cx - ww, cy - hh, cx + ww, cy + hh];\n } else {\n const [x11, y11] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x1, y1, cx, cy, element.angle);\n const [x12, y12] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x1, y2, cx, cy, element.angle);\n const [x22, y22] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x2, y2, cx, cy, element.angle);\n const [x21, y21] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x2, y1, cx, cy, element.angle);\n const minX = Math.min(x11, x12, x22, x21);\n const minY = Math.min(y11, y12, y22, y21);\n const maxX = Math.max(x11, x12, x22, x21);\n const maxY = Math.max(y11, y12, y22, y21);\n bounds = [minX, minY, maxX, maxY];\n }\n\n return bounds;\n }\n\n}\nElementBounds.boundsCache = new WeakMap(); // Scene -> Scene coords, but in x1,x2,y1,y2 format.\n//\n// If the element is created from right to left, the width is going to be negative\n// This set of functions retrieves the absolute position of the 4 points.\n\nconst getElementAbsoluteCoords = (element, includeBoundText = false) => {\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isFreeDrawElement)(element)) {\n return getFreeDrawElementAbsoluteCoords(element);\n } else if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isLinearElement)(element)) {\n return _linearElementEditor__WEBPACK_IMPORTED_MODULE_6__.LinearElementEditor.getElementAbsoluteCoords(element, includeBoundText);\n } else if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(element)) {\n const container = (0,_textElement__WEBPACK_IMPORTED_MODULE_5__.getContainerElement)(element);\n\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isArrowElement)(container)) {\n const coords = _linearElementEditor__WEBPACK_IMPORTED_MODULE_6__.LinearElementEditor.getBoundTextElementPosition(container, element);\n return [coords.x, coords.y, coords.x + element.width, coords.y + element.height, coords.x + element.width / 2, coords.y + element.height / 2];\n }\n }\n\n return [element.x, element.y, element.x + element.width, element.y + element.height, element.x + element.width / 2, element.y + element.height / 2];\n};\n/*\n * for a given element, `getElementLineSegments` returns line segments\n * that can be used for visual collision detection (useful for frames)\n * as opposed to bounding box collision detection\n */\n\nconst getElementLineSegments = element => {\n const [x1, y1, x2, y2, cx, cy] = getElementAbsoluteCoords(element);\n const center = [cx, cy];\n\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isLinearElement)(element) || (0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isFreeDrawElement)(element)) {\n const segments = [];\n let i = 0;\n\n while (i < element.points.length - 1) {\n segments.push([(0,_math__WEBPACK_IMPORTED_MODULE_0__.rotatePoint)([element.points[i][0] + element.x, element.points[i][1] + element.y], center, element.angle), (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotatePoint)([element.points[i + 1][0] + element.x, element.points[i + 1][1] + element.y], center, element.angle)]);\n i++;\n }\n\n return segments;\n }\n\n const [nw, ne, sw, se, n, s, w, e] = [[x1, y1], [x2, y1], [x1, y2], [x2, y2], [cx, y1], [cx, y2], [x1, cy], [x2, cy]].map(point => (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotatePoint)(point, center, element.angle));\n\n if (element.type === \"diamond\") {\n return [[n, w], [n, e], [s, w], [s, e]];\n }\n\n if (element.type === \"ellipse\") {\n return [[n, w], [n, e], [s, w], [s, e], [n, w], [n, e], [s, w], [s, e]];\n }\n\n return [[nw, ne], [sw, se], [nw, sw], [ne, se], [nw, e], [sw, e], [ne, w], [se, w]];\n};\n/**\n * Scene -> Scene coords, but in x1,x2,y1,y2 format.\n *\n * Rectangle here means any rectangular frame, not an excalidraw element.\n */\n\nconst getRectangleBoxAbsoluteCoords = boxSceneCoords => {\n return [boxSceneCoords.x, boxSceneCoords.y, boxSceneCoords.x + boxSceneCoords.width, boxSceneCoords.y + boxSceneCoords.height, boxSceneCoords.x + boxSceneCoords.width / 2, boxSceneCoords.y + boxSceneCoords.height / 2];\n};\nconst pointRelativeTo = (element, absoluteCoords) => {\n return [absoluteCoords[0] - element.x, absoluteCoords[1] - element.y];\n};\nconst getDiamondPoints = element => {\n // Here we add +1 to avoid these numbers to be 0\n // otherwise rough.js will throw an error complaining about it\n const topX = Math.floor(element.width / 2) + 1;\n const topY = 0;\n const rightX = element.width;\n const rightY = Math.floor(element.height / 2) + 1;\n const bottomX = topX;\n const bottomY = element.height;\n const leftX = 0;\n const leftY = rightY;\n return [topX, topY, rightX, rightY, bottomX, bottomY, leftX, leftY];\n};\nconst getCurvePathOps = shape => {\n for (const set of shape.sets) {\n if (set.type === \"path\") {\n return set.ops;\n }\n }\n\n return shape.sets[0].ops;\n}; // reference: https://eliot-jones.com/2019/12/cubic-bezier-curve-bounding-boxes\n\nconst getBezierValueForT = (t, p0, p1, p2, p3) => {\n const oneMinusT = 1 - t;\n return Math.pow(oneMinusT, 3) * p0 + 3 * Math.pow(oneMinusT, 2) * t * p1 + 3 * oneMinusT * Math.pow(t, 2) * p2 + Math.pow(t, 3) * p3;\n};\n\nconst solveQuadratic = (p0, p1, p2, p3) => {\n const i = p1 - p0;\n const j = p2 - p1;\n const k = p3 - p2;\n const a = 3 * i - 6 * j + 3 * k;\n const b = 6 * j - 6 * i;\n const c = 3 * i;\n const sqrtPart = b * b - 4 * a * c;\n const hasSolution = sqrtPart >= 0;\n\n if (!hasSolution) {\n return false;\n }\n\n let s1 = null;\n let s2 = null;\n let t1 = Infinity;\n let t2 = Infinity;\n\n if (a === 0) {\n t1 = t2 = -c / b;\n } else {\n t1 = (-b + Math.sqrt(sqrtPart)) / (2 * a);\n t2 = (-b - Math.sqrt(sqrtPart)) / (2 * a);\n }\n\n if (t1 >= 0 && t1 <= 1) {\n s1 = getBezierValueForT(t1, p0, p1, p2, p3);\n }\n\n if (t2 >= 0 && t2 <= 1) {\n s2 = getBezierValueForT(t2, p0, p1, p2, p3);\n }\n\n return [s1, s2];\n};\n\nconst getCubicBezierCurveBound = (p0, p1, p2, p3) => {\n const solX = solveQuadratic(p0[0], p1[0], p2[0], p3[0]);\n const solY = solveQuadratic(p0[1], p1[1], p2[1], p3[1]);\n let minX = Math.min(p0[0], p3[0]);\n let maxX = Math.max(p0[0], p3[0]);\n\n if (solX) {\n const xs = solX.filter(x => x !== null);\n minX = Math.min(minX, ...xs);\n maxX = Math.max(maxX, ...xs);\n }\n\n let minY = Math.min(p0[1], p3[1]);\n let maxY = Math.max(p0[1], p3[1]);\n\n if (solY) {\n const ys = solY.filter(y => y !== null);\n minY = Math.min(minY, ...ys);\n maxY = Math.max(maxY, ...ys);\n }\n\n return [minX, minY, maxX, maxY];\n};\n\nconst getMinMaxXYFromCurvePathOps = (ops, transformXY) => {\n let currentP = [0, 0];\n const {\n minX,\n minY,\n maxX,\n maxY\n } = ops.reduce((limits, {\n op,\n data\n }) => {\n // There are only four operation types:\n // move, bcurveTo, lineTo, and curveTo\n if (op === \"move\") {\n // change starting point\n currentP = data; // move operation does not draw anything; so, it always\n // returns false\n } else if (op === \"bcurveTo\") {\n const _p1 = [data[0], data[1]];\n const _p2 = [data[2], data[3]];\n const _p3 = [data[4], data[5]];\n const p1 = transformXY ? transformXY(..._p1) : _p1;\n const p2 = transformXY ? transformXY(..._p2) : _p2;\n const p3 = transformXY ? transformXY(..._p3) : _p3;\n const p0 = transformXY ? transformXY(...currentP) : currentP;\n currentP = _p3;\n const [minX, minY, maxX, maxY] = getCubicBezierCurveBound(p0, p1, p2, p3);\n limits.minX = Math.min(limits.minX, minX);\n limits.minY = Math.min(limits.minY, minY);\n limits.maxX = Math.max(limits.maxX, maxX);\n limits.maxY = Math.max(limits.maxY, maxY);\n } else if (op === \"lineTo\") {// TODO: Implement this\n } else if (op === \"qcurveTo\") {// TODO: Implement this\n }\n\n return limits;\n }, {\n minX: Infinity,\n minY: Infinity,\n maxX: -Infinity,\n maxY: -Infinity\n });\n return [minX, minY, maxX, maxY];\n};\nconst getBoundsFromPoints = points => {\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n\n for (const [x, y] of points) {\n minX = Math.min(minX, x);\n minY = Math.min(minY, y);\n maxX = Math.max(maxX, x);\n maxY = Math.max(maxY, y);\n }\n\n return [minX, minY, maxX, maxY];\n};\n\nconst getFreeDrawElementAbsoluteCoords = element => {\n const [minX, minY, maxX, maxY] = getBoundsFromPoints(element.points);\n const x1 = minX + element.x;\n const y1 = minY + element.y;\n const x2 = maxX + element.x;\n const y2 = maxY + element.y;\n return [x1, y1, x2, y2, (x1 + x2) / 2, (y1 + y2) / 2];\n};\n\nconst getArrowheadPoints = (element, shape, position, arrowhead) => {\n const ops = getCurvePathOps(shape[0]);\n\n if (ops.length < 1) {\n return null;\n } // The index of the bCurve operation to examine.\n\n\n const index = position === \"start\" ? 1 : ops.length - 1;\n const data = ops[index].data;\n const p3 = [data[4], data[5]];\n const p2 = [data[2], data[3]];\n const p1 = [data[0], data[1]]; // We need to find p0 of the bezier curve.\n // It is typically the last point of the previous\n // curve; it can also be the position of moveTo operation.\n\n const prevOp = ops[index - 1];\n let p0 = [0, 0];\n\n if (prevOp.op === \"move\") {\n p0 = prevOp.data;\n } else if (prevOp.op === \"bcurveTo\") {\n p0 = [prevOp.data[4], prevOp.data[5]];\n } // B(t) = p0 * (1-t)^3 + 3p1 * t * (1-t)^2 + 3p2 * t^2 * (1-t) + p3 * t^3\n\n\n const equation = (t, idx) => Math.pow(1 - t, 3) * p3[idx] + 3 * t * Math.pow(1 - t, 2) * p2[idx] + 3 * Math.pow(t, 2) * (1 - t) * p1[idx] + p0[idx] * Math.pow(t, 3); // Ee know the last point of the arrow (or the first, if start arrowhead).\n\n\n const [x2, y2] = position === \"start\" ? p0 : p3; // By using cubic bezier equation (B(t)) and the given parameters,\n // we calculate a point that is closer to the last point.\n // The value 0.3 is chosen arbitrarily and it works best for all\n // the tested cases.\n\n const [x1, y1] = [equation(0.3, 0), equation(0.3, 1)]; // Find the normalized direction vector based on the\n // previously calculated points.\n\n const distance = Math.hypot(x2 - x1, y2 - y1);\n const nx = (x2 - x1) / distance;\n const ny = (y2 - y1) / distance;\n const size = {\n arrow: 30,\n bar: 15,\n dot: 15,\n triangle: 15\n }[arrowhead]; // pixels (will differ for each arrowhead)\n\n let length = 0;\n\n if (arrowhead === \"arrow\") {\n // Length for -> arrows is based on the length of the last section\n const [cx, cy] = element.points[element.points.length - 1];\n const [px, py] = element.points.length > 1 ? element.points[element.points.length - 2] : [0, 0];\n length = Math.hypot(cx - px, cy - py);\n } else {\n // Length for other arrowhead types is based on the total length of the line\n for (let i = 0; i < element.points.length; i++) {\n const [px, py] = element.points[i - 1] || [0, 0];\n const [cx, cy] = element.points[i];\n length += Math.hypot(cx - px, cy - py);\n }\n } // Scale down the arrowhead until we hit a certain size so that it doesn't look weird.\n // This value is selected by minimizing a minimum size with the last segment of the arrowhead\n\n\n const minSize = Math.min(size, length / 2);\n const xs = x2 - nx * minSize;\n const ys = y2 - ny * minSize;\n\n if (arrowhead === \"dot\") {\n const r = Math.hypot(ys - y2, xs - x2) + element.strokeWidth;\n return [x2, y2, r];\n }\n\n const angle = {\n arrow: 20,\n bar: 90,\n triangle: 25\n }[arrowhead]; // degrees\n // Return points\n\n const [x3, y3] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(xs, ys, x2, y2, -angle * Math.PI / 180);\n const [x4, y4] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(xs, ys, x2, y2, angle * Math.PI / 180);\n return [x2, y2, x3, y3, x4, y4];\n};\n\nconst generateLinearElementShape = element => {\n const generator = roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_1__[\"default\"].generator();\n const options = (0,_scene_Shape__WEBPACK_IMPORTED_MODULE_2__.generateRoughOptions)(element);\n\n const method = (() => {\n if (element.roundness) {\n return \"curve\";\n }\n\n if (options.fill) {\n return \"polygon\";\n }\n\n return \"linearPath\";\n })();\n\n return generator[method](element.points, options);\n};\n\nconst getLinearElementRotatedBounds = (element, cx, cy) => {\n var _a;\n\n if (element.points.length < 2) {\n const [pointX, pointY] = element.points[0];\n const [x, y] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(element.x + pointX, element.y + pointY, cx, cy, element.angle);\n let coords = [x, y, x, y];\n const boundTextElement = (0,_textElement__WEBPACK_IMPORTED_MODULE_5__.getBoundTextElement)(element);\n\n if (boundTextElement) {\n const coordsWithBoundText = _linearElementEditor__WEBPACK_IMPORTED_MODULE_6__.LinearElementEditor.getMinMaxXYWithBoundText(element, [x, y, x, y], boundTextElement);\n coords = [coordsWithBoundText[0], coordsWithBoundText[1], coordsWithBoundText[2], coordsWithBoundText[3]];\n }\n\n return coords;\n } // first element is always the curve\n\n\n const cachedShape = (_a = _scene_ShapeCache__WEBPACK_IMPORTED_MODULE_7__.ShapeCache.get(element)) === null || _a === void 0 ? void 0 : _a[0];\n const shape = cachedShape !== null && cachedShape !== void 0 ? cachedShape : generateLinearElementShape(element);\n const ops = getCurvePathOps(shape);\n\n const transformXY = (x, y) => (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(element.x + x, element.y + y, cx, cy, element.angle);\n\n const res = getMinMaxXYFromCurvePathOps(ops, transformXY);\n let coords = [res[0], res[1], res[2], res[3]];\n const boundTextElement = (0,_textElement__WEBPACK_IMPORTED_MODULE_5__.getBoundTextElement)(element);\n\n if (boundTextElement) {\n const coordsWithBoundText = _linearElementEditor__WEBPACK_IMPORTED_MODULE_6__.LinearElementEditor.getMinMaxXYWithBoundText(element, coords, boundTextElement);\n coords = [coordsWithBoundText[0], coordsWithBoundText[1], coordsWithBoundText[2], coordsWithBoundText[3]];\n }\n\n return coords;\n};\n\nconst getElementBounds = element => {\n return ElementBounds.getBounds(element);\n};\nconst getCommonBounds = elements => {\n if (!elements.length) {\n return [0, 0, 0, 0];\n }\n\n let minX = Infinity;\n let maxX = -Infinity;\n let minY = Infinity;\n let maxY = -Infinity;\n elements.forEach(element => {\n const [x1, y1, x2, y2] = getElementBounds(element);\n minX = Math.min(minX, x1);\n minY = Math.min(minY, y1);\n maxX = Math.max(maxX, x2);\n maxY = Math.max(maxY, y2);\n });\n return [minX, minY, maxX, maxY];\n};\nconst getDraggedElementsBounds = (elements, dragOffset) => {\n const [minX, minY, maxX, maxY] = getCommonBounds(elements);\n return [minX + dragOffset.x, minY + dragOffset.y, maxX + dragOffset.x, maxY + dragOffset.y];\n};\nconst getResizedElementAbsoluteCoords = (element, nextWidth, nextHeight, normalizePoints) => {\n if (!((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isLinearElement)(element) || (0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isFreeDrawElement)(element))) {\n return [element.x, element.y, element.x + nextWidth, element.y + nextHeight];\n }\n\n const points = (0,_points__WEBPACK_IMPORTED_MODULE_4__.rescalePoints)(0, nextWidth, (0,_points__WEBPACK_IMPORTED_MODULE_4__.rescalePoints)(1, nextHeight, element.points, normalizePoints), normalizePoints);\n let bounds;\n\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isFreeDrawElement)(element)) {\n // Free Draw\n bounds = getBoundsFromPoints(points);\n } else {\n // Line\n const gen = roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_1__[\"default\"].generator();\n const curve = !element.roundness ? gen.linearPath(points, (0,_scene_Shape__WEBPACK_IMPORTED_MODULE_2__.generateRoughOptions)(element)) : gen.curve(points, (0,_scene_Shape__WEBPACK_IMPORTED_MODULE_2__.generateRoughOptions)(element));\n const ops = getCurvePathOps(curve);\n bounds = getMinMaxXYFromCurvePathOps(ops);\n }\n\n const [minX, minY, maxX, maxY] = bounds;\n return [minX + element.x, minY + element.y, maxX + element.x, maxY + element.y];\n};\nconst getElementPointsCoords = (element, points) => {\n // This might be computationally heavey\n const gen = roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_1__[\"default\"].generator();\n const curve = element.roundness == null ? gen.linearPath(points, (0,_scene_Shape__WEBPACK_IMPORTED_MODULE_2__.generateRoughOptions)(element)) : gen.curve(points, (0,_scene_Shape__WEBPACK_IMPORTED_MODULE_2__.generateRoughOptions)(element));\n const ops = getCurvePathOps(curve);\n const [minX, minY, maxX, maxY] = getMinMaxXYFromCurvePathOps(ops);\n return [minX + element.x, minY + element.y, maxX + element.x, maxY + element.y];\n};\nconst getClosestElementBounds = (elements, from) => {\n if (!elements.length) {\n return [0, 0, 0, 0];\n }\n\n let minDistance = Infinity;\n let closestElement = elements[0];\n elements.forEach(element => {\n const [x1, y1, x2, y2] = getElementBounds(element);\n const distance = (0,_math__WEBPACK_IMPORTED_MODULE_0__.distance2d)((x1 + x2) / 2, (y1 + y2) / 2, from.x, from.y);\n\n if (distance < minDistance) {\n minDistance = distance;\n closestElement = element;\n }\n });\n return getElementBounds(closestElement);\n};\nconst getCommonBoundingBox = elements => {\n const [minX, minY, maxX, maxY] = getCommonBounds(elements);\n return {\n minX,\n minY,\n maxX,\n maxY,\n width: maxX - minX,\n height: maxY - minY,\n midX: (minX + maxX) / 2,\n midY: (minY + maxY) / 2\n };\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,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Q0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxHQUFHO0FBQ0g7QUFDQTtBQUNPO0FBQ1A7QUFDQTtBQUNBO0FBQ087QUFDUCxRQUFRLDREQUFlLGFBQWEsOERBQWlCO0FBQ3JEO0FBQ0E7O0FBRUEsaUJBQWlCLHNEQUFhLGVBQWUsc0RBQWE7QUFDMUQ7O0FBRUEsTUFBTSw4REFBaUI7QUFDdkI7QUFDQTtBQUNBLElBQUk7QUFDSjtBQUNBLGdCQUFnQixtRUFBZTtBQUMvQiw4REFBOEQsa0VBQW9CLCtCQUErQixrRUFBb0I7QUFDckk7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNPO0FBQ1A7QUFDQSxjQUFjLG1FQUFlO0FBQzdCLG1FQUFtRSxrRUFBb0IsK0JBQStCLGtFQUFvQjtBQUMxSTtBQUNBO0FBQ0E7QUFDQTtBQUNPO0FBQ1A7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EscUJBQXFCLGlEQUFVOztBQUUvQjtBQUNBO0FBQ0E7QUFDQTtBQUNBLEdBQUc7QUFDSDtBQUNBO0FBQ087QUFDUDtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi4vLi4vZWxlbWVudC9ib3VuZHMudHM/ZmNhMCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBkaXN0YW5jZTJkLCByb3RhdGUsIHJvdGF0ZVBvaW50IH0gZnJvbSBcIi4uL21hdGhcIjtcbmltcG9ydCByb3VnaCBmcm9tIFwicm91Z2hqcy9iaW4vcm91Z2hcIjtcbmltcG9ydCB7IGdlbmVyYXRlUm91Z2hPcHRpb25zIH0gZnJvbSBcIi4uL3NjZW5lL1NoYXBlXCI7XG5pbXBvcnQgeyBpc0Fycm93RWxlbWVudCwgaXNGcmVlRHJhd0VsZW1lbnQsIGlzTGluZWFyRWxlbWVudCwgaXNUZXh0RWxlbWVudCB9IGZyb20gXCIuL3R5cGVDaGVja3NcIjtcbmltcG9ydCB7IHJlc2NhbGVQb2ludHMgfSBmcm9tIFwiLi4vcG9pbnRzXCI7XG5pbXBvcnQgeyBnZXRCb3VuZFRleHRFbGVtZW50LCBnZXRDb250YWluZXJFbGVtZW50IH0gZnJvbSBcIi4vdGV4dEVsZW1lbnRcIjtcbmltcG9ydCB7IExpbmVhckVsZW1lbnRFZGl0b3IgfSBmcm9tIFwiLi9saW5lYXJFbGVtZW50RWRpdG9yXCI7XG5pbXBvcnQgeyBTaGFwZUNhY2hlIH0gZnJvbSBcIi4uL3NjZW5lL1NoYXBlQ2FjaGVcIjtcbmV4cG9ydCBjbGFzcyBFbGVtZW50Qm91bmRzIHtcbiAgc3RhdGljIGdldEJvdW5kcyhlbGVtZW50KSB7XG4gICAgY29uc3QgY2FjaGVkQm91bmRzID0gRWxlbWVudEJvdW5kcy5ib3VuZHNDYWNoZS5nZXQoZWxlbWVudCk7XG5cbiAgICBpZiAoKGNhY2hlZEJvdW5kcyA9PT0gbnVsbCB8fCBjYWNoZWRCb3VuZHMgPT09IHZvaWQgMCA/IHZvaWQgMCA6IGNhY2hlZEJvdW5kcy52ZXJzaW9uKSAmJiBjYWNoZWRCb3VuZHMudmVyc2lvbiA9PT0gZWxlbWVudC52ZXJzaW9uKSB7XG4gICAgICByZXR1cm4gY2FjaGVkQm91bmRzLmJvdW5kcztcbiAgICB9XG5cbiAgICBjb25zdCBib3VuZHMgPSBFbGVtZW50Qm91bmRzLmNhbGN1bGF0ZUJvdW5kcyhlbGVtZW50KTtcbiAgICBFbGVtZW50Qm91bmRzLmJvdW5kc0NhY2hlLnNldChlbGVtZW50LCB7XG4gICAgICB2ZXJzaW9uOiBlbGVtZW50LnZlcnNpb24sXG4gICAgICBib3VuZHNcbiAgICB9KTtcbiAgICByZXR1cm4gYm91bmRzO1xuICB9XG5cbiAgc3RhdGljIGNhbGN1bGF0ZUJvdW5kcyhlbGVtZW50KSB7XG4gICAgbGV0IGJvdW5kcztcbiAgICBjb25zdCBbeDEsIHkxLCB4MiwgeTIsIGN4LCBjeV0gPSBnZXRFbGVtZW50QWJzb2x1dGVDb29yZHMoZWxlbWVudCk7XG5cbiAgICBpZiAoaXNGcmVlRHJhd0VsZW1lbnQoZWxlbWVudCkpIHtcbiAgICAgIGNvbnN0IFttaW5YLCBtaW5ZLCBtYXhYLCBtYXhZXSA9IGdldEJvdW5kc0Zyb21Qb2ludHMoZWxlbWVudC5wb2ludHMubWFwKChbeCwgeV0pID0+IHJvdGF0ZSh4LCB5LCBjeCAtIGVsZW1lbnQueCwgY3kgLSBlbGVtZW50LnksIGVsZW1lbnQuYW5nbGUpKSk7XG4gICAgICByZXR1cm4gW21pblggKyBlbGVtZW50LngsIG1pblkgKyBlbGVtZW50LnksIG1heFggKyBlbGVtZW50LngsIG1heFkgKyBlbGVtZW50LnldO1xuICAgIH0gZWxzZSBpZiAoaXNMaW5lYXJFbGVtZW50KGVsZW1lbnQpKSB7XG4gICAgICBib3VuZHMgPSBnZXRMaW5lYXJFbGVtZW50Um90YXRlZEJvdW5kcyhlbGVtZW50LCBjeCwgY3kpO1xuICAgIH0gZWxzZSBpZiAoZWxlbWVudC50eXBlID09PSBcImRpYW1vbmRcIikge1xuICAgICAgY29uc3QgW3gxMSwgeTExXSA9IHJvdGF0ZShjeCwgeTEsIGN4LCBjeSwgZWxlbWVudC5hbmdsZSk7XG4gICAgICBjb25zdCBbeDEyLCB5MTJdID0gcm90YXRlKGN4LCB5MiwgY3gsIGN5LCBlbGVtZW50LmFuZ2xlKTtcbiAgICAgIGNvbnN0IFt4MjIsIHkyMl0gPSByb3RhdGUoeDEsIGN5LCBjeCwgY3ksIGVsZW1lbnQuYW5nbGUpO1xuICAgICAgY29uc3QgW3gyMSwgeTIxXSA9IHJvdGF0ZSh4MiwgY3ksIGN4LCBjeSwgZWxlbWVudC5hbmdsZSk7XG4gICAgICBjb25zdCBtaW5YID0gTWF0aC5taW4oeDExLCB4MTIsIHgyMiwgeDIxKTtcbiAgICAgIGNvbnN0IG1pblkgPSBNYXRoLm1pbih5MTEsIHkxMiwgeTIyLCB5MjEpO1xuICAgICAgY29uc3QgbWF4WCA9IE1hdGgubWF4KHgxMSwgeDEyLCB4MjIsIHgyMSk7XG4gICAgICBjb25zdCBtYXhZID0gTWF0aC5tYXgoeTExLCB5MTIsIHkyMiwgeTIxKTtcbiAgICAgIGJvdW5kcyA9IFttaW5YLCBtaW5ZLCBtYXhYLCBtYXhZXTtcbiAgICB9IGVsc2UgaWYgKGVsZW1lbnQudHlwZSA9PT0gXCJlbGxpcHNlXCIpIHtcbiAgICAgIGNvbnN0IHcgPSAoeDIgLSB4MSkgLyAyO1xuICAgICAgY29uc3QgaCA9ICh5MiAtIHkxKSAvIDI7XG4gICAgICBjb25zdCBjb3MgPSBNYXRoLmNvcyhlbGVtZW50LmFuZ2xlKTtcbiAgICAgIGNvbnN0IHNpbiA9IE1hdGguc2luKGVsZW1lbnQuYW5nbGUpO1xuICAgICAgY29uc3Qgd3cgPSBNYXRoLmh5cG90KHcgKiBjb3MsIGggKiBzaW4pO1xuICAgICAgY29uc3QgaGggPSBNYXRoLmh5cG90KGggKiBjb3MsIHcgKiBzaW4pO1xuICAgICAgYm91bmRzID0gW2N4IC0gd3csIGN5IC0gaGgsIGN4ICsgd3csIGN5ICsgaGhdO1xuICAgIH0gZWxzZSB7XG4gICAgICBjb25zdCBbeDExLCB5MTFdID0gcm90YXRlKHgxLCB5MSwgY3gsIGN5LCBlbGVtZW50LmFuZ2xlKTtcbiAgICAgIGNvbnN0IFt4MTIsIHkxMl0gPSByb3RhdGUoeDEsIHkyLCBjeCwgY3ksIGVsZW1lbnQuYW5nbGUpO1xuICAgICAgY29uc3QgW3gyMiwgeTIyXSA9IHJvdGF0ZSh4MiwgeTIsIGN4LCBjeSwgZWxlbWVudC5hbmdsZSk7XG4gICAgICBjb25zdCBbeDIxLCB5MjFdID0gcm90YXRlKHgyLCB5MSwgY3gsIGN5LCBlbGVtZW50LmFuZ2xlKTtcbiAgICAgIGNvbnN0IG1pblggPSBNYXRoLm1pbih4MTEsIHgxMiwgeDIyLCB4MjEpO1xuICAgICAgY29uc3QgbWluWSA9IE1hdGgubWluKHkxMSwgeTEyLCB5MjIsIHkyMSk7XG4gICAgICBjb25zdCBtYXhYID0gTWF0aC5tYXgoeDExLCB4MTIsIHgyMiwgeDIxKTtcbiAgICAgIGNvbnN0IG1heFkgPSBNYXRoLm1heCh5MTEsIHkxMiwgeTIyLCB5MjEpO1xuICAgICAgYm91bmRzID0gW21pblgsIG1pblksIG1heFgsIG1heFldO1xuICAgIH1cblxuICAgIHJldHVybiBib3VuZHM7XG4gIH1cblxufVxuRWxlbWVudEJvdW5kcy5ib3VuZHNDYWNoZSA9IG5ldyBXZWFrTWFwKCk7IC8vIFNjZW5lIC0+IFNjZW5lIGNvb3JkcywgYnV0IGluIHgxLHgyLHkxLHkyIGZvcm1hdC5cbi8vXG4vLyBJZiB0aGUgZWxlbWVudCBpcyBjcmVhdGVkIGZyb20gcmlnaHQgdG8gbGVmdCwgdGhlIHdpZHRoIGlzIGdvaW5nIHRvIGJlIG5lZ2F0aXZlXG4vLyBUaGlzIHNldCBvZiBmdW5jdGlvbnMgcmV0cmlldmVzIHRoZSBhYnNvbHV0ZSBwb3NpdGlvbiBvZiB0aGUgNCBwb2ludHMuXG5cbmV4cG9ydCBjb25zdCBnZXRFbGVtZW50QWJzb2x1dGVDb29yZHMgPSAoZWxlbWVudCwgaW5jbHVkZUJvdW5kVGV4dCA9IGZhbHNlKSA9PiB7XG4gIGlmIChpc0ZyZWVEcmF3RWxlbWVudChlbGVtZW50KSkge1xuICAgIHJldHVybiBnZXRGcmVlRHJhd0VsZW1lbnRBYnNvbHV0ZUNvb3JkcyhlbGVtZW50KTtcbiAgfSBlbHNlIGlmIChpc0xpbmVhckVsZW1lbnQoZWxlbWVudCkpIHtcbiAgICByZXR1cm4gTGluZWFyRWxlbWVudEVkaXRvci5nZXRFbGVtZW50QWJzb2x1dGVDb29yZHMoZWxlbWVudCwgaW5jbHVkZUJvdW5kVGV4dCk7XG4gIH0gZWxzZSBpZiAoaXNUZXh0RWxlbWVudChlbGVtZW50KSkge1xuICAgIGNvbnN0IGNvbnRhaW5lciA9IGdldENvbnRhaW5lckVsZW1lbnQoZWxlbWVudCk7XG5cbiAgICBpZiAoaXNBcnJvd0VsZW1lbnQoY29udGFpbmVyKSkge1xuICAgICAgY29uc3QgY29vcmRzID0gTGluZWFyRWxlbWVudEVkaXRvci5nZXRCb3VuZFRleHRFbGVtZW50UG9zaXRpb24oY29udGFpbmVyLCBlbGVtZW50KTtcbiAgICAgIHJldHVybiBbY29vcmRzLngsIGNvb3Jkcy55LCBjb29yZHMueCArIGVsZW1lbnQud2lkdGgsIGNvb3Jkcy55ICsgZWxlbWVudC5oZWlnaHQsIGNvb3Jkcy54ICsgZWxlbWVudC53aWR0aCAvIDIsIGNvb3Jkcy55ICsgZWxlbWVudC5oZWlnaHQgLyAyXTtcbiAgICB9XG4gIH1cblxuICByZXR1cm4gW2VsZW1lbnQueCwgZWxlbWVudC55LCBlbGVtZW50LnggKyBlbGVtZW50LndpZHRoLCBlbGVtZW50LnkgKyBlbGVtZW50LmhlaWdodCwgZWxlbWVudC54ICsgZWxlbWVudC53aWR0aCAvIDIsIGVsZW1lbnQueSArIGVsZW1lbnQuaGVpZ2h0IC8gMl07XG59O1xuLypcbiAqIGZvciBhIGdpdmVuIGVsZW1lbnQsIGBnZXRFbGVtZW50TGluZVNlZ21lbnRzYCByZXR1cm5zIGxpbmUgc2VnbWVudHNcbiAqIHRoYXQgY2FuIGJlIHVzZWQgZm9yIHZpc3VhbCBjb2xsaXNpb24gZGV0ZWN0aW9uICh1c2VmdWwgZm9yIGZyYW1lcylcbiAqIGFzIG9wcG9zZWQgdG8gYm91bmRpbmcgYm94IGNvbGxpc2lvbiBkZXRlY3Rpb25cbiAqL1xuXG5leHBvcnQgY29uc3QgZ2V0RWxlbWVudExpbmVTZWdtZW50cyA9IGVsZW1lbnQgPT4ge1xuICBjb25zdCBbeDEsIHkxLCB4MiwgeTIsIGN4LCBjeV0gPSBnZXRFbGVtZW50QWJzb2x1dGVDb29yZHMoZWxlbWVudCk7XG4gIGNvbnN0IGNlbnRlciA9IFtjeCwgY3ldO1xuXG4gIGlmIChpc0xpbmVhckVsZW1lbnQoZWxlbWVudCkgfHwgaXNGcmVlRHJhd0VsZW1lbnQoZWxlbWVudCkpIHtcbiAgICBjb25zdCBzZWdtZW50cyA9IFtdO1xuICAgIGxldCBpID0gMDtcblxuICAgIHdoaWxlIChpIDwgZWxlbWVudC5wb2ludHMubGVuZ3RoIC0gMSkge1xuICAgICAgc2VnbWVudHMucHVzaChbcm90YXRlUG9pbnQoW2VsZW1lbnQucG9pbnRzW2ldWzBdICsgZWxlbWVudC54LCBlbGVtZW50LnBvaW50c1tpXVsxXSArIGVsZW1lbnQueV0sIGNlbnRlciwgZWxlbWVudC5hbmdsZSksIHJvdGF0ZVBvaW50KFtlbGVtZW50LnBvaW50c1tpICsgMV1bMF0gKyBlbGVtZW50LngsIGVsZW1lbnQucG9pbnRzW2kgKyAxXVsxXSArIGVsZW1lbnQueV0sIGNlbnRlciwgZWxlbWVudC5hbmdsZSldKTtcbiAgICAgIGkrKztcbiAgICB9XG5cbiAgICByZXR1cm4gc2VnbWVudHM7XG4gIH1cblxuICBjb25zdCBbbncsIG5lLCBzdywgc2UsIG4sIHMsIHcsIGVdID0gW1t4MSwgeTFdLCBbeDIsIHkxXSwgW3gxLCB5Ml0sIFt4MiwgeTJdLCBbY3gsIHkxXSwgW2N4LCB5Ml0sIFt4MSwgY3ldLCBbeDIsIGN5XV0ubWFwKHBvaW50ID0+IHJvdGF0ZVBvaW50KHBvaW50LCBjZW50ZXIsIGVsZW1lbnQuYW5nbGUpKTtcblxuICBpZiAoZWxlbWVudC50eXBlID09PSBcImRpYW1vbmRcIikge1xuICAgIHJldHVybiBbW24sIHddLCBbbiwgZV0sIFtzLCB3XSwgW3MsIGVdXTtcbiAgfVxuXG4gIGlmIChlbGVtZW50LnR5cGUgPT09IFwiZWxsaXBzZVwiKSB7XG4gICAgcmV0dXJuIFtbbiwgd10sIFtuLCBlXSwgW3MsIHddLCBbcywgZV0sIFtuLCB3XSwgW24sIGVdLCBbcywgd10sIFtzLCBlXV07XG4gIH1cblxuICByZXR1cm4gW1tudywgbmVdLCBbc3csIHNlXSwgW253LCBzd10sIFtuZSwgc2VdLCBbbncsIGVdLCBbc3csIGVdLCBbbmUsIHddLCBbc2UsIHddXTtcbn07XG4vKipcbiAqIFNjZW5lIC0+IFNjZW5lIGNvb3JkcywgYnV0IGluIHgxLHgyLHkxLHkyIGZvcm1hdC5cbiAqXG4gKiBSZWN0YW5nbGUgaGVyZSBtZWFucyBhbnkgcmVjdGFuZ3VsYXIgZnJhbWUsIG5vdCBhbiBleGNhbGlkcmF3IGVsZW1lbnQuXG4gKi9cblxuZXhwb3J0IGNvbnN0IGdldFJlY3RhbmdsZUJveEFic29sdXRlQ29vcmRzID0gYm94U2NlbmVDb29yZHMgPT4ge1xuICByZXR1cm4gW2JveFNjZW5lQ29vcmRzLngsIGJveFNjZW5lQ29vcmRzLnksIGJveFNjZW5lQ29vcmRzLnggKyBib3hTY2VuZUNvb3Jkcy53aWR0aCwgYm94U2NlbmVDb29yZHMueSArIGJveFNjZW5lQ29vcmRzLmhlaWdodCwgYm94U2NlbmVDb29yZHMueCArIGJveFNjZW5lQ29vcmRzLndpZHRoIC8gMiwgYm94U2NlbmVDb29yZHMueSArIGJveFNjZW5lQ29vcmRzLmhlaWdodCAvIDJdO1xufTtcbmV4cG9ydCBjb25zdCBwb2ludFJlbGF0aXZlVG8gPSAoZWxlbWVudCwgYWJzb2x1dGVDb29yZHMpID0+IHtcbiAgcmV0dXJuIFthYnNvbHV0ZUNvb3Jkc1swXSAtIGVsZW1lbnQueCwgYWJzb2x1dGVDb29yZHNbMV0gLSBlbGVtZW50LnldO1xufTtcbmV4cG9ydCBjb25zdCBnZXREaWFtb25kUG9pbnRzID0gZWxlbWVudCA9PiB7XG4gIC8vIEhlcmUgd2UgYWRkICsxIHRvIGF2b2lkIHRoZXNlIG51bWJlcnMgdG8gYmUgMFxuICAvLyBvdGhlcndpc2Ugcm91Z2guanMgd2lsbCB0aHJvdyBhbiBlcnJvciBjb21wbGFpbmluZyBhYm91dCBpdFxuICBjb25zdCB0b3BYID0gTWF0aC5mbG9vcihlbGVtZW50LndpZHRoIC8gMikgKyAxO1xuICBjb25zdCB0b3BZID0gMDtcbiAgY29uc3QgcmlnaHRYID0gZWxlbWVudC53aWR0aDtcbiAgY29uc3QgcmlnaHRZID0gTWF0aC5mbG9vcihlbGVtZW50LmhlaWdodCAvIDIpICsgMTtcbiAgY29uc3QgYm90dG9tWCA9IHRvcFg7XG4gIGNvbnN0IGJvdHRvbVkgPSBlbGVtZW50LmhlaWdodDtcbiAgY29uc3QgbGVmdFggPSAwO1xuICBjb25zdCBsZWZ0WSA9IHJpZ2h0WTtcbiAgcmV0dXJuIFt0b3BYLCB0b3BZLCByaWdodFgsIHJpZ2h0WSwgYm90dG9tWCwgYm90dG9tWSwgbGVmdFgsIGxlZnRZXTtcbn07XG5leHBvcnQgY29uc3QgZ2V0Q3VydmVQYXRoT3BzID0gc2hhcGUgPT4ge1xuICBmb3IgKGNvbnN0IHNldCBvZiBzaGFwZS5zZXRzKSB7XG4gICAgaWYgKHNldC50eXBlID09PSBcInBhdGhcIikge1xuICAgICAgcmV0dXJuIHNldC5vcHM7XG4gICAgfVxuICB9XG5cbiAgcmV0dXJuIHNoYXBlLnNldHNbMF0ub3BzO1xufTsgLy8gcmVmZXJlbmNlOiBodHRwczovL2VsaW90LWpvbmVzLmNvbS8yMDE5LzEyL2N1YmljLWJlemllci1jdXJ2ZS1ib3VuZGluZy1ib3hlc1xuXG5jb25zdCBnZXRCZXppZXJWYWx1ZUZvclQgPSAodCwgcDAsIHAxLCBwMiwgcDMpID0+IHtcbiAgY29uc3Qgb25lTWludXNUID0gMSAtIHQ7XG4gIHJldHVybiBNYXRoLnBvdyhvbmVNaW51c1QsIDMpICogcDAgKyAzICogTWF0aC5wb3cob25lTWludXNULCAyKSAqIHQgKiBwMSArIDMgKiBvbmVNaW51c1QgKiBNYXRoLnBvdyh0LCAyKSAqIHAyICsgTWF0aC5wb3codCwgMykgKiBwMztcbn07XG5cbmNvbnN0IHNvbHZlUXVhZHJhdGljID0gKHAwLCBwMSwgcDIsIHAzKSA9PiB7XG4gIGNvbnN0IGkgPSBwMSAtIHAwO1xuICBjb25zdCBqID0gcDIgLSBwMTtcbiAgY29uc3QgayA9IHAzIC0gcDI7XG4gIGNvbnN0IGEgPSAzICogaSAtIDYgKiBqICsgMyAqIGs7XG4gIGNvbnN0IGIgPSA2ICogaiAtIDYgKiBpO1xuICBjb25zdCBjID0gMyAqIGk7XG4gIGNvbnN0IHNxcnRQYXJ0ID0gYiAqIGIgLSA0ICogYSAqIGM7XG4gIGNvbnN0IGhhc1NvbHV0aW9uID0gc3FydFBhcnQgPj0gMDtcblxuICBpZiAoIWhhc1NvbHV0aW9uKSB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG5cbiAgbGV0IHMxID0gbnVsbDtcbiAgbGV0IHMyID0gbnVsbDtcbiAgbGV0IHQxID0gSW5maW5pdHk7XG4gIGxldCB0MiA9IEluZmluaXR5O1xuXG4gIGlmIChhID09PSAwKSB7XG4gICAgdDEgPSB0MiA9IC1jIC8gYjtcbiAgfSBlbHNlIHtcbiAgICB0MSA9ICgtYiArIE1hdGguc3FydChzcXJ0UGFydCkpIC8gKDIgKiBhKTtcbiAgICB0MiA9ICgtYiAtIE1hdGguc3FydChzcXJ0UGFydCkpIC8gKDIgKiBhKTtcbiAgfVxuXG4gIGlmICh0MSA+PSAwICYmIHQxIDw9IDEpIHtcbiAgICBzMSA9IGdldEJlemllclZhbHVlRm9yVCh0MSwgcDAsIHAxLCBwMiwgcDMpO1xuICB9XG5cbiAgaWYgKHQyID49IDAgJiYgdDIgPD0gMSkge1xuICAgIHMyID0gZ2V0QmV6aWVyVmFsdWVGb3JUKHQyLCBwMCwgcDEsIHAyLCBwMyk7XG4gIH1cblxuICByZXR1cm4gW3MxLCBzMl07XG59O1xuXG5jb25zdCBnZXRDdWJpY0JlemllckN1cnZlQm91bmQgPSAocDAsIHAxLCBwMiwgcDMpID0+IHtcbiAgY29uc3Qgc29sWCA9IHNvbHZlUXVhZHJhdGljKHAwWzBdLCBwMVswXSwgcDJbMF0sIHAzWzBdKTtcbiAgY29uc3Qgc29sWSA9IHNvbHZlUXVhZHJhdGljKHAwWzFdLCBwMVsxXSwgcDJbMV0sIHAzWzFdKTtcbiAgbGV0IG1pblggPSBNYXRoLm1pbihwMFswXSwgcDNbMF0pO1xuICBsZXQgbWF4WCA9IE1hdGgubWF4KHAwWzBdLCBwM1swXSk7XG5cbiAgaWYgKHNvbFgpIHtcbiAgICBjb25zdCB4cyA9IHNvbFguZmlsdGVyKHggPT4geCAhPT0gbnVsbCk7XG4gICAgbWluWCA9IE1hdGgubWluKG1pblgsIC4uLnhzKTtcbiAgICBtYXhYID0gTWF0aC5tYXgobWF4WCwgLi4ueHMpO1xuICB9XG5cbiAgbGV0IG1pblkgPSBNYXRoLm1pbihwMFsxXSwgcDNbMV0pO1xuICBsZXQgbWF4WSA9IE1hdGgubWF4KHAwWzFdLCBwM1sxXSk7XG5cbiAgaWYgKHNvbFkpIHtcbiAgICBjb25zdCB5cyA9IHNvbFkuZmlsdGVyKHkgPT4geSAhPT0gbnVsbCk7XG4gICAgbWluWSA9IE1hdGgubWluKG1pblksIC4uLnlzKTtcbiAgICBtYXhZID0gTWF0aC5tYXgobWF4WSwgLi4ueXMpO1xuICB9XG5cbiAgcmV0dXJuIFttaW5YLCBtaW5ZLCBtYXhYLCBtYXhZXTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRNaW5NYXhYWUZyb21DdXJ2ZVBhdGhPcHMgPSAob3BzLCB0cmFuc2Zvcm1YWSkgPT4ge1xuICBsZXQgY3VycmVudFAgPSBbMCwgMF07XG4gIGNvbnN0IHtcbiAgICBtaW5YLFxuICAgIG1pblksXG4gICAgbWF4WCxcbiAgICBtYXhZXG4gIH0gPSBvcHMucmVkdWNlKChsaW1pdHMsIHtcbiAgICBvcCxcbiAgICBkYXRhXG4gIH0pID0+IHtcbiAgICAvLyBUaGVyZSBhcmUgb25seSBmb3VyIG9wZXJhdGlvbiB0eXBlczpcbiAgICAvLyBtb3ZlLCBiY3VydmVUbywgbGluZVRvLCBhbmQgY3VydmVUb1xuICAgIGlmIChvcCA9PT0gXCJtb3ZlXCIpIHtcbiAgICAgIC8vIGNoYW5nZSBzdGFydGluZyBwb2ludFxuICAgICAgY3VycmVudFAgPSBkYXRhOyAvLyBtb3ZlIG9wZXJhdGlvbiBkb2VzIG5vdCBkcmF3IGFueXRoaW5nOyBzbywgaXQgYWx3YXlzXG4gICAgICAvLyByZXR1cm5zIGZhbHNlXG4gICAgfSBlbHNlIGlmIChvcCA9PT0gXCJiY3VydmVUb1wiKSB7XG4gICAgICBjb25zdCBfcDEgPSBbZGF0YVswXSwgZGF0YVsxXV07XG4gICAgICBjb25zdCBfcDIgPSBbZGF0YVsyXSwgZGF0YVszXV07XG4gICAgICBjb25zdCBfcDMgPSBbZGF0YVs0XSwgZGF0YVs1XV07XG4gICAgICBjb25zdCBwMSA9IHRyYW5zZm9ybVhZID8gdHJhbnNmb3JtWFkoLi4uX3AxKSA6IF9wMTtcbiAgICAgIGNvbnN0IHAyID0gdHJhbnNmb3JtWFkgPyB0cmFuc2Zvcm1YWSguLi5fcDIpIDogX3AyO1xuICAgICAgY29uc3QgcDMgPSB0cmFuc2Zvcm1YWSA/IHRyYW5zZm9ybVhZKC4uLl9wMykgOiBfcDM7XG4gICAgICBjb25zdCBwMCA9IHRyYW5zZm9ybVhZID8gdHJhbnNmb3JtWFkoLi4uY3VycmVudFApIDogY3VycmVudFA7XG4gICAgICBjdXJyZW50UCA9IF9wMztcbiAgICAgIGNvbnN0IFttaW5YLCBtaW5ZLCBtYXhYLCBtYXhZXSA9IGdldEN1YmljQmV6aWVyQ3VydmVCb3VuZChwMCwgcDEsIHAyLCBwMyk7XG4gICAgICBsaW1pdHMubWluWCA9IE1hdGgubWluKGxpbWl0cy5taW5YLCBtaW5YKTtcbiAgICAgIGxpbWl0cy5taW5ZID0gTWF0aC5taW4obGltaXRzLm1pblksIG1pblkpO1xuICAgICAgbGltaXRzLm1heFggPSBNYXRoLm1heChsaW1pdHMubWF4WCwgbWF4WCk7XG4gICAgICBsaW1pdHMubWF4WSA9IE1hdGgubWF4KGxpbWl0cy5tYXhZLCBtYXhZKTtcbiAgICB9IGVsc2UgaWYgKG9wID09PSBcImxpbmVUb1wiKSB7Ly8gVE9ETzogSW1wbGVtZW50IHRoaXNcbiAgICB9IGVsc2UgaWYgKG9wID09PSBcInFjdXJ2ZVRvXCIpIHsvLyBUT0RPOiBJbXBsZW1lbnQgdGhpc1xuICAgIH1cblxuICAgIHJldHVybiBsaW1pdHM7XG4gIH0sIHtcbiAgICBtaW5YOiBJbmZpbml0eSxcbiAgICBtaW5ZOiBJbmZpbml0eSxcbiAgICBtYXhYOiAtSW5maW5pdHksXG4gICAgbWF4WTogLUluZmluaXR5XG4gIH0pO1xuICByZXR1cm4gW21pblgsIG1pblksIG1heFgsIG1heFldO1xufTtcbmV4cG9ydCBjb25zdCBnZXRCb3VuZHNGcm9tUG9pbnRzID0gcG9pbnRzID0+IHtcbiAgbGV0IG1pblggPSBJbmZpbml0eTtcbiAgbGV0IG1pblkgPSBJbmZpbml0eTtcbiAgbGV0IG1heFggPSAtSW5maW5pdHk7XG4gIGxldCBtYXhZID0gLUluZmluaXR5O1xuXG4gIGZvciAoY29uc3QgW3gsIHldIG9mIHBvaW50cykge1xuICAgIG1pblggPSBNYXRoLm1pbihtaW5YLCB4KTtcbiAgICBtaW5ZID0gTWF0aC5taW4obWluWSwgeSk7XG4gICAgbWF4WCA9IE1hdGgubWF4KG1heFgsIHgpO1xuICAgIG1heFkgPSBNYXRoLm1heChtYXhZLCB5KTtcbiAgfVxuXG4gIHJldHVybiBbbWluWCwgbWluWSwgbWF4WCwgbWF4WV07XG59O1xuXG5jb25zdCBnZXRGcmVlRHJhd0VsZW1lbnRBYnNvbHV0ZUNvb3JkcyA9IGVsZW1lbnQgPT4ge1xuICBjb25zdCBbbWluWCwgbWluWSwgbWF4WCwgbWF4WV0gPSBnZXRCb3VuZHNGcm9tUG9pbnRzKGVsZW1lbnQucG9pbnRzKTtcbiAgY29uc3QgeDEgPSBtaW5YICsgZWxlbWVudC54O1xuICBjb25zdCB5MSA9IG1pblkgKyBlbGVtZW50Lnk7XG4gIGNvbnN0IHgyID0gbWF4WCArIGVsZW1lbnQueDtcbiAgY29uc3QgeTIgPSBtYXhZICsgZWxlbWVudC55O1xuICByZXR1cm4gW3gxLCB5MSwgeDIsIHkyLCAoeDEgKyB4MikgLyAyLCAoeTEgKyB5MikgLyAyXTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRBcnJvd2hlYWRQb2ludHMgPSAoZWxlbWVudCwgc2hhcGUsIHBvc2l0aW9uLCBhcnJvd2hlYWQpID0+IHtcbiAgY29uc3Qgb3BzID0gZ2V0Q3VydmVQYXRoT3BzKHNoYXBlWzBdKTtcblxuICBpZiAob3BzLmxlbmd0aCA8IDEpIHtcbiAgICByZXR1cm4gbnVsbDtcbiAgfSAvLyBUaGUgaW5kZXggb2YgdGhlIGJDdXJ2ZSBvcGVyYXRpb24gdG8gZXhhbWluZS5cblxuXG4gIGNvbnN0IGluZGV4ID0gcG9zaXRpb24gPT09IFwic3RhcnRcIiA/IDEgOiBvcHMubGVuZ3RoIC0gMTtcbiAgY29uc3QgZGF0YSA9IG9wc1tpbmRleF0uZGF0YTtcbiAgY29uc3QgcDMgPSBbZGF0YVs0XSwgZGF0YVs1XV07XG4gIGNvbnN0IHAyID0gW2RhdGFbMl0sIGRhdGFbM11dO1xuICBjb25zdCBwMSA9IFtkYXRhWzBdLCBkYXRhWzFdXTsgLy8gV2UgbmVlZCB0byBmaW5kIHAwIG9mIHRoZSBiZXppZXIgY3VydmUuXG4gIC8vIEl0IGlzIHR5cGljYWxseSB0aGUgbGFzdCBwb2ludCBvZiB0aGUgcHJldmlvdXNcbiAgLy8gY3VydmU7IGl0IGNhbiBhbHNvIGJlIHRoZSBwb3NpdGlvbiBvZiBtb3ZlVG8gb3BlcmF0aW9uLlxuXG4gIGNvbnN0IHByZXZPcCA9IG9wc1tpbmRleCAtIDFdO1xuICBsZXQgcDAgPSBbMCwgMF07XG5cbiAgaWYgKHByZXZPcC5vcCA9PT0gXCJtb3ZlXCIpIHtcbiAgICBwMCA9IHByZXZPcC5kYXRhO1xuICB9IGVsc2UgaWYgKHByZXZPcC5vcCA9PT0gXCJiY3VydmVUb1wiKSB7XG4gICAgcDAgPSBbcHJldk9wLmRhdGFbNF0sIHByZXZPcC5kYXRhWzVdXTtcbiAgfSAvLyBCKHQpID0gcDAgKiAoMS10KV4zICsgM3AxICogdCAqICgxLXQpXjIgKyAzcDIgKiB0XjIgKiAoMS10KSArIHAzICogdF4zXG5cblxuICBjb25zdCBlcXVhdGlvbiA9ICh0LCBpZHgpID0+IE1hdGgucG93KDEgLSB0LCAzKSAqIHAzW2lkeF0gKyAzICogdCAqIE1hdGgucG93KDEgLSB0LCAyKSAqIHAyW2lkeF0gKyAzICogTWF0aC5wb3codCwgMikgKiAoMSAtIHQpICogcDFbaWR4XSArIHAwW2lkeF0gKiBNYXRoLnBvdyh0LCAzKTsgLy8gRWUga25vdyB0aGUgbGFzdCBwb2ludCBvZiB0aGUgYXJyb3cgKG9yIHRoZSBmaXJzdCwgaWYgc3RhcnQgYXJyb3doZWFkKS5cblxuXG4gIGNvbnN0IFt4MiwgeTJdID0gcG9zaXRpb24gPT09IFwic3RhcnRcIiA/IHAwIDogcDM7IC8vIEJ5IHVzaW5nIGN1YmljIGJlemllciBlcXVhdGlvbiAoQih0KSkgYW5kIHRoZSBnaXZlbiBwYXJhbWV0ZXJzLFxuICAvLyB3ZSBjYWxjdWxhdGUgYSBwb2ludCB0aGF0IGlzIGNsb3NlciB0byB0aGUgbGFzdCBwb2ludC5cbiAgLy8gVGhlIHZhbHVlIDAuMyBpcyBjaG9zZW4gYXJiaXRyYXJpbHkgYW5kIGl0IHdvcmtzIGJlc3QgZm9yIGFsbFxuICAvLyB0aGUgdGVzdGVkIGNhc2VzLlxuXG4gIGNvbnN0IFt4MSwgeTFdID0gW2VxdWF0aW9uKDAuMywgMCksIGVxdWF0aW9uKDAuMywgMSldOyAvLyBGaW5kIHRoZSBub3JtYWxpemVkIGRpcmVjdGlvbiB2ZWN0b3IgYmFzZWQgb24gdGhlXG4gIC8vIHByZXZpb3VzbHkgY2FsY3VsYXRlZCBwb2ludHMuXG5cbiAgY29uc3QgZGlzdGFuY2UgPSBNYXRoLmh5cG90KHgyIC0geDEsIHkyIC0geTEpO1xuICBjb25zdCBueCA9ICh4MiAtIHgxKSAvIGRpc3RhbmNlO1xuICBjb25zdCBueSA9ICh5MiAtIHkxKSAvIGRpc3RhbmNlO1xuICBjb25zdCBzaXplID0ge1xuICAgIGFycm93OiAzMCxcbiAgICBiYXI6IDE1LFxuICAgIGRvdDogMTUsXG4gICAgdHJpYW5nbGU6IDE1XG4gIH1bYXJyb3doZWFkXTsgLy8gcGl4ZWxzICh3aWxsIGRpZmZlciBmb3IgZWFjaCBhcnJvd2hlYWQpXG5cbiAgbGV0IGxlbmd0aCA9IDA7XG5cbiAgaWYgKGFycm93aGVhZCA9PT0gXCJhcnJvd1wiKSB7XG4gICAgLy8gTGVuZ3RoIGZvciAtPiBhcnJvd3MgaXMgYmFzZWQgb24gdGhlIGxlbmd0aCBvZiB0aGUgbGFzdCBzZWN0aW9uXG4gICAgY29uc3QgW2N4LCBjeV0gPSBlbGVtZW50LnBvaW50c1tlbGVtZW50LnBvaW50cy5sZW5ndGggLSAxXTtcbiAgICBjb25zdCBbcHgsIHB5XSA9IGVsZW1lbnQucG9pbnRzLmxlbmd0aCA+IDEgPyBlbGVtZW50LnBvaW50c1tlbGVtZW50LnBvaW50cy5sZW5ndGggLSAyXSA6IFswLCAwXTtcbiAgICBsZW5ndGggPSBNYXRoLmh5cG90KGN4IC0gcHgsIGN5IC0gcHkpO1xuICB9IGVsc2Uge1xuICAgIC8vIExlbmd0aCBmb3Igb3RoZXIgYXJyb3doZWFkIHR5cGVzIGlzIGJhc2VkIG9uIHRoZSB0b3RhbCBsZW5ndGggb2YgdGhlIGxpbmVcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IGVsZW1lbnQucG9pbnRzLmxlbmd0aDsgaSsrKSB7XG4gICAgICBjb25zdCBbcHgsIHB5XSA9IGVsZW1lbnQucG9pbnRzW2kgLSAxXSB8fCBbMCwgMF07XG4gICAgICBjb25zdCBbY3gsIGN5XSA9IGVsZW1lbnQucG9pbnRzW2ldO1xuICAgICAgbGVuZ3RoICs9IE1hdGguaHlwb3QoY3ggLSBweCwgY3kgLSBweSk7XG4gICAgfVxuICB9IC8vIFNjYWxlIGRvd24gdGhlIGFycm93aGVhZCB1bnRpbCB3ZSBoaXQgYSBjZXJ0YWluIHNpemUgc28gdGhhdCBpdCBkb2Vzbid0IGxvb2sgd2VpcmQuXG4gIC8vIFRoaXMgdmFsdWUgaXMgc2VsZWN0ZWQgYnkgbWluaW1pemluZyBhIG1pbmltdW0gc2l6ZSB3aXRoIHRoZSBsYXN0IHNlZ21lbnQgb2YgdGhlIGFycm93aGVhZFxuXG5cbiAgY29uc3QgbWluU2l6ZSA9IE1hdGgubWluKHNpemUsIGxlbmd0aCAvIDIpO1xuICBjb25zdCB4cyA9IHgyIC0gbnggKiBtaW5TaXplO1xuICBjb25zdCB5cyA9IHkyIC0gbnkgKiBtaW5TaXplO1xuXG4gIGlmIChhcnJvd2hlYWQgPT09IFwiZG90XCIpIHtcbiAgICBjb25zdCByID0gTWF0aC5oeXBvdCh5cyAtIHkyLCB4cyAtIHgyKSArIGVsZW1lbnQuc3Ryb2tlV2lkdGg7XG4gICAgcmV0dXJuIFt4MiwgeTIsIHJdO1xuICB9XG5cbiAgY29uc3QgYW5nbGUgPSB7XG4gICAgYXJyb3c6IDIwLFxuICAgIGJhcjogOTAsXG4gICAgdHJpYW5nbGU6IDI1XG4gIH1bYXJyb3doZWFkXTsgLy8gZGVncmVlc1xuICAvLyBSZXR1cm4gcG9pbnRzXG5cbiAgY29uc3QgW3gzLCB5M10gPSByb3RhdGUoeHMsIHlzLCB4MiwgeTIsIC1hbmdsZSAqIE1hdGguUEkgLyAxODApO1xuICBjb25zdCBbeDQsIHk0XSA9IHJvdGF0ZSh4cywgeXMsIHgyLCB5MiwgYW5nbGUgKiBNYXRoLlBJIC8gMTgwKTtcbiAgcmV0dXJuIFt4MiwgeTIsIHgzLCB5MywgeDQsIHk0XTtcbn07XG5cbmNvbnN0IGdlbmVyYXRlTGluZWFyRWxlbWVudFNoYXBlID0gZWxlbWVudCA9PiB7XG4gIGNvbnN0IGdlbmVyYXRvciA9IHJvdWdoLmdlbmVyYXRvcigpO1xuICBjb25zdCBvcHRpb25zID0gZ2VuZXJhdGVSb3VnaE9wdGlvbnMoZWxlbWVudCk7XG5cbiAgY29uc3QgbWV0aG9kID0gKCgpID0+IHtcbiAgICBpZiAoZWxlbWVudC5yb3VuZG5lc3MpIHtcbiAgICAgIHJldHVybiBcImN1cnZlXCI7XG4gICAgfVxuXG4gICAgaWYgKG9wdGlvbnMuZmlsbCkge1xuICAgICAgcmV0dXJuIFwicG9seWdvblwiO1xuICAgIH1cblxuICAgIHJldHVybiBcImxpbmVhclBhdGhcIjtcbiAgfSkoKTtcblxuICByZXR1cm4gZ2VuZXJhdG9yW21ldGhvZF0oZWxlbWVudC5wb2ludHMsIG9wdGlvbnMpO1xufTtcblxuY29uc3QgZ2V0TGluZWFyRWxlbWVudFJvdGF0ZWRCb3VuZHMgPSAoZWxlbWVudCwgY3gsIGN5KSA9PiB7XG4gIHZhciBfYTtcblxuICBpZiAoZWxlbWVudC5wb2ludHMubGVuZ3RoIDwgMikge1xuICAgIGNvbnN0IFtwb2ludFgsIHBvaW50WV0gPSBlbGVtZW50LnBvaW50c1swXTtcbiAgICBjb25zdCBbeCwgeV0gPSByb3RhdGUoZWxlbWVudC54ICsgcG9pbnRYLCBlbGVtZW50LnkgKyBwb2ludFksIGN4LCBjeSwgZWxlbWVudC5hbmdsZSk7XG4gICAgbGV0IGNvb3JkcyA9IFt4LCB5LCB4LCB5XTtcbiAgICBjb25zdCBib3VuZFRleHRFbGVtZW50ID0gZ2V0Qm91bmRUZXh0RWxlbWVudChlbGVtZW50KTtcblxuICAgIGlmIChib3VuZFRleHRFbGVtZW50KSB7XG4gICAgICBjb25zdCBjb29yZHNXaXRoQm91bmRUZXh0ID0gTGluZWFyRWxlbWVudEVkaXRvci5nZXRNaW5NYXhYWVdpdGhCb3VuZFRleHQoZWxlbWVudCwgW3gsIHksIHgsIHldLCBib3VuZFRleHRFbGVtZW50KTtcbiAgICAgIGNvb3JkcyA9IFtjb29yZHNXaXRoQm91bmRUZXh0WzBdLCBjb29yZHNXaXRoQm91bmRUZXh0WzFdLCBjb29yZHNXaXRoQm91bmRUZXh0WzJdLCBjb29yZHNXaXRoQm91bmRUZXh0WzNdXTtcbiAgICB9XG5cbiAgICByZXR1cm4gY29vcmRzO1xuICB9IC8vIGZpcnN0IGVsZW1lbnQgaXMgYWx3YXlzIHRoZSBjdXJ2ZVxuXG5cbiAgY29uc3QgY2FjaGVkU2hhcGUgPSAoX2EgPSBTaGFwZUNhY2hlLmdldChlbGVtZW50KSkgPT09IG51bGwgfHwgX2EgPT09IHZvaWQgMCA/IHZvaWQgMCA6IF9hWzBdO1xuICBjb25zdCBzaGFwZSA9IGNhY2hlZFNoYXBlICE9PSBudWxsICYmIGNhY2hlZFNoYXBlICE9PSB2b2lkIDAgPyBjYWNoZWRTaGFwZSA6IGdlbmVyYXRlTGluZWFyRWxlbWVudFNoYXBlKGVsZW1lbnQpO1xuICBjb25zdCBvcHMgPSBnZXRDdXJ2ZVBhdGhPcHMoc2hhcGUpO1xuXG4gIGNvbnN0IHRyYW5zZm9ybVhZID0gKHgsIHkpID0+IHJvdGF0ZShlbGVtZW50LnggKyB4LCBlbGVtZW50LnkgKyB5LCBjeCwgY3ksIGVsZW1lbnQuYW5nbGUpO1xuXG4gIGNvbnN0IHJlcyA9IGdldE1pbk1heFhZRnJvbUN1cnZlUGF0aE9wcyhvcHMsIHRyYW5zZm9ybVhZKTtcbiAgbGV0IGNvb3JkcyA9IFtyZXNbMF0sIHJlc1sxXSwgcmVzWzJdLCByZXNbM11dO1xuICBjb25zdCBib3VuZFRleHRFbGVtZW50ID0gZ2V0Qm91bmRUZXh0RWxlbWVudChlbGVtZW50KTtcblxuICBpZiAoYm91bmRUZXh0RWxlbWVudCkge1xuICAgIGNvbnN0IGNvb3Jkc1dpdGhCb3VuZFRleHQgPSBMaW5lYXJFbGVtZW50RWRpdG9yLmdldE1pbk1heFhZV2l0aEJvdW5kVGV4dChlbGVtZW50LCBjb29yZHMsIGJvdW5kVGV4dEVsZW1lbnQpO1xuICAgIGNvb3JkcyA9IFtjb29yZHNXaXRoQm91bmRUZXh0WzBdLCBjb29yZHNXaXRoQm91bmRUZXh0WzFdLCBjb29yZHNXaXRoQm91bmRUZXh0WzJdLCBjb29yZHNXaXRoQm91bmRUZXh0WzNdXTtcbiAgfVxuXG4gIHJldHVybiBjb29yZHM7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0RWxlbWVudEJvdW5kcyA9IGVsZW1lbnQgPT4ge1xuICByZXR1cm4gRWxlbWVudEJvdW5kcy5nZXRCb3VuZHMoZWxlbWVudCk7XG59O1xuZXhwb3J0IGNvbnN0IGdldENvbW1vbkJvdW5kcyA9IGVsZW1lbnRzID0+IHtcbiAgaWYgKCFlbGVtZW50cy5sZW5ndGgpIHtcbiAgICByZXR1cm4gWzAsIDAsIDAsIDBdO1xuICB9XG5cbiAgbGV0IG1pblggPSBJbmZpbml0eTtcbiAgbGV0IG1heFggPSAtSW5maW5pdHk7XG4gIGxldCBtaW5ZID0gSW5maW5pdHk7XG4gIGxldCBtYXhZID0gLUluZmluaXR5O1xuICBlbGVtZW50cy5mb3JFYWNoKGVsZW1lbnQgPT4ge1xuICAgIGNvbnN0IFt4MSwgeTEsIHgyLCB5Ml0gPSBnZXRFbGVtZW50Qm91bmRzKGVsZW1lbnQpO1xuICAgIG1pblggPSBNYXRoLm1pbihtaW5YLCB4MSk7XG4gICAgbWluWSA9IE1hdGgubWluKG1pblksIHkxKTtcbiAgICBtYXhYID0gTWF0aC5tYXgobWF4WCwgeDIpO1xuICAgIG1heFkgPSBNYXRoLm1heChtYXhZLCB5Mik7XG4gIH0pO1xuICByZXR1cm4gW21pblgsIG1pblksIG1heFgsIG1heFldO1xufTtcbmV4cG9ydCBjb25zdCBnZXREcmFnZ2VkRWxlbWVudHNCb3VuZHMgPSAoZWxlbWVudHMsIGRyYWdPZmZzZXQpID0+IHtcbiAgY29uc3QgW21pblgsIG1pblksIG1heFgsIG1heFldID0gZ2V0Q29tbW9uQm91bmRzKGVsZW1lbnRzKTtcbiAgcmV0dXJuIFttaW5YICsgZHJhZ09mZnNldC54LCBtaW5ZICsgZHJhZ09mZnNldC55LCBtYXhYICsgZHJhZ09mZnNldC54LCBtYXhZICsgZHJhZ09mZnNldC55XTtcbn07XG5leHBvcnQgY29uc3QgZ2V0UmVzaXplZEVsZW1lbnRBYnNvbHV0ZUNvb3JkcyA9IChlbGVtZW50LCBuZXh0V2lkdGgsIG5leHRIZWlnaHQsIG5vcm1hbGl6ZVBvaW50cykgPT4ge1xuICBpZiAoIShpc0xpbmVhckVsZW1lbnQoZWxlbWVudCkgfHwgaXNGcmVlRHJhd0VsZW1lbnQoZWxlbWVudCkpKSB7XG4gICAgcmV0dXJuIFtlbGVtZW50LngsIGVsZW1lbnQueSwgZWxlbWVudC54ICsgbmV4dFdpZHRoLCBlbGVtZW50LnkgKyBuZXh0SGVpZ2h0XTtcbiAgfVxuXG4gIGNvbnN0IHBvaW50cyA9IHJlc2NhbGVQb2ludHMoMCwgbmV4dFdpZHRoLCByZXNjYWxlUG9pbnRzKDEsIG5leHRIZWlnaHQsIGVsZW1lbnQucG9pbnRzLCBub3JtYWxpemVQb2ludHMpLCBub3JtYWxpemVQb2ludHMpO1xuICBsZXQgYm91bmRzO1xuXG4gIGlmIChpc0ZyZWVEcmF3RWxlbWVudChlbGVtZW50KSkge1xuICAgIC8vIEZyZWUgRHJhd1xuICAgIGJvdW5kcyA9IGdldEJvdW5kc0Zyb21Qb2ludHMocG9pbnRzKTtcbiAgfSBlbHNlIHtcbiAgICAvLyBMaW5lXG4gICAgY29uc3QgZ2VuID0gcm91Z2guZ2VuZXJhdG9yKCk7XG4gICAgY29uc3QgY3VydmUgPSAhZWxlbWVudC5yb3VuZG5lc3MgPyBnZW4ubGluZWFyUGF0aChwb2ludHMsIGdlbmVyYXRlUm91Z2hPcHRpb25zKGVsZW1lbnQpKSA6IGdlbi5jdXJ2ZShwb2ludHMsIGdlbmVyYXRlUm91Z2hPcHRpb25zKGVsZW1lbnQpKTtcbiAgICBjb25zdCBvcHMgPSBnZXRDdXJ2ZVBhdGhPcHMoY3VydmUpO1xuICAgIGJvdW5kcyA9IGdldE1pbk1heFhZRnJvbUN1cnZlUGF0aE9wcyhvcHMpO1xuICB9XG5cbiAgY29uc3QgW21pblgsIG1pblksIG1heFgsIG1heFldID0gYm91bmRzO1xuICByZXR1cm4gW21pblggKyBlbGVtZW50LngsIG1pblkgKyBlbGVtZW50LnksIG1heFggKyBlbGVtZW50LngsIG1heFkgKyBlbGVtZW50LnldO1xufTtcbmV4cG9ydCBjb25zdCBnZXRFbGVtZW50UG9pbnRzQ29vcmRzID0gKGVsZW1lbnQsIHBvaW50cykgPT4ge1xuICAvLyBUaGlzIG1pZ2h0IGJlIGNvbXB1dGF0aW9uYWxseSBoZWF2ZXlcbiAgY29uc3QgZ2VuID0gcm91Z2guZ2VuZXJhdG9yKCk7XG4gIGNvbnN0IGN1cnZlID0gZWxlbWVudC5yb3VuZG5lc3MgPT0gbnVsbCA/IGdlbi5saW5lYXJQYXRoKHBvaW50cywgZ2VuZXJhdGVSb3VnaE9wdGlvbnMoZWxlbWVudCkpIDogZ2VuLmN1cnZlKHBvaW50cywgZ2VuZXJhdGVSb3VnaE9wdGlvbnMoZWxlbWVudCkpO1xuICBjb25zdCBvcHMgPSBnZXRDdXJ2ZVBhdGhPcHMoY3VydmUpO1xuICBjb25zdCBbbWluWCwgbWluWSwgbWF4WCwgbWF4WV0gPSBnZXRNaW5NYXhYWUZyb21DdXJ2ZVBhdGhPcHMob3BzKTtcbiAgcmV0dXJuIFttaW5YICsgZWxlbWVudC54LCBtaW5ZICsgZWxlbWVudC55LCBtYXhYICsgZWxlbWVudC54LCBtYXhZICsgZWxlbWVudC55XTtcbn07XG5leHBvcnQgY29uc3QgZ2V0Q2xvc2VzdEVsZW1lbnRCb3VuZHMgPSAoZWxlbWVudHMsIGZyb20pID0+IHtcbiAgaWYgKCFlbGVtZW50cy5sZW5ndGgpIHtcbiAgICByZXR1cm4gWzAsIDAsIDAsIDBdO1xuICB9XG5cbiAgbGV0IG1pbkRpc3RhbmNlID0gSW5maW5pdHk7XG4gIGxldCBjbG9zZXN0RWxlbWVudCA9IGVsZW1lbnRzWzBdO1xuICBlbGVtZW50cy5mb3JFYWNoKGVsZW1lbnQgPT4ge1xuICAgIGNvbnN0IFt4MSwgeTEsIHgyLCB5Ml0gPSBnZXRFbGVtZW50Qm91bmRzKGVsZW1lbnQpO1xuICAgIGNvbnN0IGRpc3RhbmNlID0gZGlzdGFuY2UyZCgoeDEgKyB4MikgLyAyLCAoeTEgKyB5MikgLyAyLCBmcm9tLngsIGZyb20ueSk7XG5cbiAgICBpZiAoZGlzdGFuY2UgPCBtaW5EaXN0YW5jZSkge1xuICAgICAgbWluRGlzdGFuY2UgPSBkaXN0YW5jZTtcbiAgICAgIGNsb3Nlc3RFbGVtZW50ID0gZWxlbWVudDtcbiAgICB9XG4gIH0pO1xuICByZXR1cm4gZ2V0RWxlbWVudEJvdW5kcyhjbG9zZXN0RWxlbWVudCk7XG59O1xuZXhwb3J0IGNvbnN0IGdldENvbW1vbkJvdW5kaW5nQm94ID0gZWxlbWVudHMgPT4ge1xuICBjb25zdCBbbWluWCwgbWluWSwgbWF4WCwgbWF4WV0gPSBnZXRDb21tb25Cb3VuZHMoZWxlbWVudHMpO1xuICByZXR1cm4ge1xuICAgIG1pblgsXG4gICAgbWluWSxcbiAgICBtYXhYLFxuICAgIG1heFksXG4gICAgd2lkdGg6IG1heFggLSBtaW5YLFxuICAgIGhlaWdodDogbWF4WSAtIG1pblksXG4gICAgbWlkWDogKG1pblggKyBtYXhYKSAvIDIsXG4gICAgbWlkWTogKG1pblkgKyBtYXhZKSAvIDJcbiAgfTtcbn07Il0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///../../element/bounds.ts\n");
3874
+ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"ElementBounds\": () => (/* binding */ ElementBounds),\n/* harmony export */ \"getArrowheadPoints\": () => (/* binding */ getArrowheadPoints),\n/* harmony export */ \"getBoundsFromPoints\": () => (/* binding */ getBoundsFromPoints),\n/* harmony export */ \"getClosestElementBounds\": () => (/* binding */ getClosestElementBounds),\n/* harmony export */ \"getCommonBoundingBox\": () => (/* binding */ getCommonBoundingBox),\n/* harmony export */ \"getCommonBounds\": () => (/* binding */ getCommonBounds),\n/* harmony export */ \"getCurvePathOps\": () => (/* binding */ getCurvePathOps),\n/* harmony export */ \"getDiamondPoints\": () => (/* binding */ getDiamondPoints),\n/* harmony export */ \"getDraggedElementsBounds\": () => (/* binding */ getDraggedElementsBounds),\n/* harmony export */ \"getElementAbsoluteCoords\": () => (/* binding */ getElementAbsoluteCoords),\n/* harmony export */ \"getElementBounds\": () => (/* binding */ getElementBounds),\n/* harmony export */ \"getElementLineSegments\": () => (/* binding */ getElementLineSegments),\n/* harmony export */ \"getElementPointsCoords\": () => (/* binding */ getElementPointsCoords),\n/* harmony export */ \"getMinMaxXYFromCurvePathOps\": () => (/* binding */ getMinMaxXYFromCurvePathOps),\n/* harmony export */ \"getRectangleBoxAbsoluteCoords\": () => (/* binding */ getRectangleBoxAbsoluteCoords),\n/* harmony export */ \"getResizedElementAbsoluteCoords\": () => (/* binding */ getResizedElementAbsoluteCoords),\n/* harmony export */ \"pointRelativeTo\": () => (/* binding */ pointRelativeTo)\n/* harmony export */ });\n/* harmony import */ var _math__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../math */ \"../../math.ts\");\n/* harmony import */ var roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! roughjs/bin/rough */ \"../../../node_modules/roughjs/bin/rough.js\");\n/* harmony import */ var _scene_Shape__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../scene/Shape */ \"../../scene/Shape.ts\");\n/* harmony import */ var _typeChecks__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./typeChecks */ \"../../element/typeChecks.ts\");\n/* harmony import */ var _points__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../points */ \"../../points.ts\");\n/* harmony import */ var _textElement__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./textElement */ \"../../element/textElement.ts\");\n/* harmony import */ var _linearElementEditor__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./linearElementEditor */ \"../../element/linearElementEditor.ts\");\n/* harmony import */ var _scene_ShapeCache__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../scene/ShapeCache */ \"../../scene/ShapeCache.ts\");\n\n\n\n\n\n\n\n\nclass ElementBounds {\n static getBounds(element) {\n const cachedBounds = ElementBounds.boundsCache.get(element);\n\n if ((cachedBounds === null || cachedBounds === void 0 ? void 0 : cachedBounds.version) && cachedBounds.version === element.version && // we don't invalidate cache when we update containers and not labels,\n // which is causing problems down the line. Fix TBA.\n !(0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isBoundToContainer)(element)) {\n return cachedBounds.bounds;\n }\n\n const bounds = ElementBounds.calculateBounds(element);\n ElementBounds.boundsCache.set(element, {\n version: element.version,\n bounds\n });\n return bounds;\n }\n\n static calculateBounds(element) {\n let bounds;\n const [x1, y1, x2, y2, cx, cy] = getElementAbsoluteCoords(element);\n\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isFreeDrawElement)(element)) {\n const [minX, minY, maxX, maxY] = getBoundsFromPoints(element.points.map(([x, y]) => (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x, y, cx - element.x, cy - element.y, element.angle)));\n return [minX + element.x, minY + element.y, maxX + element.x, maxY + element.y];\n } else if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isLinearElement)(element)) {\n bounds = getLinearElementRotatedBounds(element, cx, cy);\n } else if (element.type === \"diamond\") {\n const [x11, y11] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(cx, y1, cx, cy, element.angle);\n const [x12, y12] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(cx, y2, cx, cy, element.angle);\n const [x22, y22] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x1, cy, cx, cy, element.angle);\n const [x21, y21] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x2, cy, cx, cy, element.angle);\n const minX = Math.min(x11, x12, x22, x21);\n const minY = Math.min(y11, y12, y22, y21);\n const maxX = Math.max(x11, x12, x22, x21);\n const maxY = Math.max(y11, y12, y22, y21);\n bounds = [minX, minY, maxX, maxY];\n } else if (element.type === \"ellipse\") {\n const w = (x2 - x1) / 2;\n const h = (y2 - y1) / 2;\n const cos = Math.cos(element.angle);\n const sin = Math.sin(element.angle);\n const ww = Math.hypot(w * cos, h * sin);\n const hh = Math.hypot(h * cos, w * sin);\n bounds = [cx - ww, cy - hh, cx + ww, cy + hh];\n } else {\n const [x11, y11] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x1, y1, cx, cy, element.angle);\n const [x12, y12] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x1, y2, cx, cy, element.angle);\n const [x22, y22] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x2, y2, cx, cy, element.angle);\n const [x21, y21] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x2, y1, cx, cy, element.angle);\n const minX = Math.min(x11, x12, x22, x21);\n const minY = Math.min(y11, y12, y22, y21);\n const maxX = Math.max(x11, x12, x22, x21);\n const maxY = Math.max(y11, y12, y22, y21);\n bounds = [minX, minY, maxX, maxY];\n }\n\n return bounds;\n }\n\n}\nElementBounds.boundsCache = new WeakMap(); // Scene -> Scene coords, but in x1,x2,y1,y2 format.\n//\n// If the element is created from right to left, the width is going to be negative\n// This set of functions retrieves the absolute position of the 4 points.\n\nconst getElementAbsoluteCoords = (element, includeBoundText = false) => {\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isFreeDrawElement)(element)) {\n return getFreeDrawElementAbsoluteCoords(element);\n } else if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isLinearElement)(element)) {\n return _linearElementEditor__WEBPACK_IMPORTED_MODULE_6__.LinearElementEditor.getElementAbsoluteCoords(element, includeBoundText);\n } else if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(element)) {\n const container = (0,_textElement__WEBPACK_IMPORTED_MODULE_5__.getContainerElement)(element);\n\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isArrowElement)(container)) {\n const coords = _linearElementEditor__WEBPACK_IMPORTED_MODULE_6__.LinearElementEditor.getBoundTextElementPosition(container, element);\n return [coords.x, coords.y, coords.x + element.width, coords.y + element.height, coords.x + element.width / 2, coords.y + element.height / 2];\n }\n }\n\n return [element.x, element.y, element.x + element.width, element.y + element.height, element.x + element.width / 2, element.y + element.height / 2];\n};\n/*\n * for a given element, `getElementLineSegments` returns line segments\n * that can be used for visual collision detection (useful for frames)\n * as opposed to bounding box collision detection\n */\n\nconst getElementLineSegments = element => {\n const [x1, y1, x2, y2, cx, cy] = getElementAbsoluteCoords(element);\n const center = [cx, cy];\n\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isLinearElement)(element) || (0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isFreeDrawElement)(element)) {\n const segments = [];\n let i = 0;\n\n while (i < element.points.length - 1) {\n segments.push([(0,_math__WEBPACK_IMPORTED_MODULE_0__.rotatePoint)([element.points[i][0] + element.x, element.points[i][1] + element.y], center, element.angle), (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotatePoint)([element.points[i + 1][0] + element.x, element.points[i + 1][1] + element.y], center, element.angle)]);\n i++;\n }\n\n return segments;\n }\n\n const [nw, ne, sw, se, n, s, w, e] = [[x1, y1], [x2, y1], [x1, y2], [x2, y2], [cx, y1], [cx, y2], [x1, cy], [x2, cy]].map(point => (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotatePoint)(point, center, element.angle));\n\n if (element.type === \"diamond\") {\n return [[n, w], [n, e], [s, w], [s, e]];\n }\n\n if (element.type === \"ellipse\") {\n return [[n, w], [n, e], [s, w], [s, e], [n, w], [n, e], [s, w], [s, e]];\n }\n\n return [[nw, ne], [sw, se], [nw, sw], [ne, se], [nw, e], [sw, e], [ne, w], [se, w]];\n};\n/**\n * Scene -> Scene coords, but in x1,x2,y1,y2 format.\n *\n * Rectangle here means any rectangular frame, not an excalidraw element.\n */\n\nconst getRectangleBoxAbsoluteCoords = boxSceneCoords => {\n return [boxSceneCoords.x, boxSceneCoords.y, boxSceneCoords.x + boxSceneCoords.width, boxSceneCoords.y + boxSceneCoords.height, boxSceneCoords.x + boxSceneCoords.width / 2, boxSceneCoords.y + boxSceneCoords.height / 2];\n};\nconst pointRelativeTo = (element, absoluteCoords) => {\n return [absoluteCoords[0] - element.x, absoluteCoords[1] - element.y];\n};\nconst getDiamondPoints = element => {\n // Here we add +1 to avoid these numbers to be 0\n // otherwise rough.js will throw an error complaining about it\n const topX = Math.floor(element.width / 2) + 1;\n const topY = 0;\n const rightX = element.width;\n const rightY = Math.floor(element.height / 2) + 1;\n const bottomX = topX;\n const bottomY = element.height;\n const leftX = 0;\n const leftY = rightY;\n return [topX, topY, rightX, rightY, bottomX, bottomY, leftX, leftY];\n};\nconst getCurvePathOps = shape => {\n for (const set of shape.sets) {\n if (set.type === \"path\") {\n return set.ops;\n }\n }\n\n return shape.sets[0].ops;\n}; // reference: https://eliot-jones.com/2019/12/cubic-bezier-curve-bounding-boxes\n\nconst getBezierValueForT = (t, p0, p1, p2, p3) => {\n const oneMinusT = 1 - t;\n return Math.pow(oneMinusT, 3) * p0 + 3 * Math.pow(oneMinusT, 2) * t * p1 + 3 * oneMinusT * Math.pow(t, 2) * p2 + Math.pow(t, 3) * p3;\n};\n\nconst solveQuadratic = (p0, p1, p2, p3) => {\n const i = p1 - p0;\n const j = p2 - p1;\n const k = p3 - p2;\n const a = 3 * i - 6 * j + 3 * k;\n const b = 6 * j - 6 * i;\n const c = 3 * i;\n const sqrtPart = b * b - 4 * a * c;\n const hasSolution = sqrtPart >= 0;\n\n if (!hasSolution) {\n return false;\n }\n\n let s1 = null;\n let s2 = null;\n let t1 = Infinity;\n let t2 = Infinity;\n\n if (a === 0) {\n t1 = t2 = -c / b;\n } else {\n t1 = (-b + Math.sqrt(sqrtPart)) / (2 * a);\n t2 = (-b - Math.sqrt(sqrtPart)) / (2 * a);\n }\n\n if (t1 >= 0 && t1 <= 1) {\n s1 = getBezierValueForT(t1, p0, p1, p2, p3);\n }\n\n if (t2 >= 0 && t2 <= 1) {\n s2 = getBezierValueForT(t2, p0, p1, p2, p3);\n }\n\n return [s1, s2];\n};\n\nconst getCubicBezierCurveBound = (p0, p1, p2, p3) => {\n const solX = solveQuadratic(p0[0], p1[0], p2[0], p3[0]);\n const solY = solveQuadratic(p0[1], p1[1], p2[1], p3[1]);\n let minX = Math.min(p0[0], p3[0]);\n let maxX = Math.max(p0[0], p3[0]);\n\n if (solX) {\n const xs = solX.filter(x => x !== null);\n minX = Math.min(minX, ...xs);\n maxX = Math.max(maxX, ...xs);\n }\n\n let minY = Math.min(p0[1], p3[1]);\n let maxY = Math.max(p0[1], p3[1]);\n\n if (solY) {\n const ys = solY.filter(y => y !== null);\n minY = Math.min(minY, ...ys);\n maxY = Math.max(maxY, ...ys);\n }\n\n return [minX, minY, maxX, maxY];\n};\n\nconst getMinMaxXYFromCurvePathOps = (ops, transformXY) => {\n let currentP = [0, 0];\n const {\n minX,\n minY,\n maxX,\n maxY\n } = ops.reduce((limits, {\n op,\n data\n }) => {\n // There are only four operation types:\n // move, bcurveTo, lineTo, and curveTo\n if (op === \"move\") {\n // change starting point\n currentP = data; // move operation does not draw anything; so, it always\n // returns false\n } else if (op === \"bcurveTo\") {\n const _p1 = [data[0], data[1]];\n const _p2 = [data[2], data[3]];\n const _p3 = [data[4], data[5]];\n const p1 = transformXY ? transformXY(..._p1) : _p1;\n const p2 = transformXY ? transformXY(..._p2) : _p2;\n const p3 = transformXY ? transformXY(..._p3) : _p3;\n const p0 = transformXY ? transformXY(...currentP) : currentP;\n currentP = _p3;\n const [minX, minY, maxX, maxY] = getCubicBezierCurveBound(p0, p1, p2, p3);\n limits.minX = Math.min(limits.minX, minX);\n limits.minY = Math.min(limits.minY, minY);\n limits.maxX = Math.max(limits.maxX, maxX);\n limits.maxY = Math.max(limits.maxY, maxY);\n } else if (op === \"lineTo\") {// TODO: Implement this\n } else if (op === \"qcurveTo\") {// TODO: Implement this\n }\n\n return limits;\n }, {\n minX: Infinity,\n minY: Infinity,\n maxX: -Infinity,\n maxY: -Infinity\n });\n return [minX, minY, maxX, maxY];\n};\nconst getBoundsFromPoints = points => {\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n\n for (const [x, y] of points) {\n minX = Math.min(minX, x);\n minY = Math.min(minY, y);\n maxX = Math.max(maxX, x);\n maxY = Math.max(maxY, y);\n }\n\n return [minX, minY, maxX, maxY];\n};\n\nconst getFreeDrawElementAbsoluteCoords = element => {\n const [minX, minY, maxX, maxY] = getBoundsFromPoints(element.points);\n const x1 = minX + element.x;\n const y1 = minY + element.y;\n const x2 = maxX + element.x;\n const y2 = maxY + element.y;\n return [x1, y1, x2, y2, (x1 + x2) / 2, (y1 + y2) / 2];\n};\n\nconst getArrowheadPoints = (element, shape, position, arrowhead) => {\n const ops = getCurvePathOps(shape[0]);\n\n if (ops.length < 1) {\n return null;\n } // The index of the bCurve operation to examine.\n\n\n const index = position === \"start\" ? 1 : ops.length - 1;\n const data = ops[index].data;\n const p3 = [data[4], data[5]];\n const p2 = [data[2], data[3]];\n const p1 = [data[0], data[1]]; // We need to find p0 of the bezier curve.\n // It is typically the last point of the previous\n // curve; it can also be the position of moveTo operation.\n\n const prevOp = ops[index - 1];\n let p0 = [0, 0];\n\n if (prevOp.op === \"move\") {\n p0 = prevOp.data;\n } else if (prevOp.op === \"bcurveTo\") {\n p0 = [prevOp.data[4], prevOp.data[5]];\n } // B(t) = p0 * (1-t)^3 + 3p1 * t * (1-t)^2 + 3p2 * t^2 * (1-t) + p3 * t^3\n\n\n const equation = (t, idx) => Math.pow(1 - t, 3) * p3[idx] + 3 * t * Math.pow(1 - t, 2) * p2[idx] + 3 * Math.pow(t, 2) * (1 - t) * p1[idx] + p0[idx] * Math.pow(t, 3); // Ee know the last point of the arrow (or the first, if start arrowhead).\n\n\n const [x2, y2] = position === \"start\" ? p0 : p3; // By using cubic bezier equation (B(t)) and the given parameters,\n // we calculate a point that is closer to the last point.\n // The value 0.3 is chosen arbitrarily and it works best for all\n // the tested cases.\n\n const [x1, y1] = [equation(0.3, 0), equation(0.3, 1)]; // Find the normalized direction vector based on the\n // previously calculated points.\n\n const distance = Math.hypot(x2 - x1, y2 - y1);\n const nx = (x2 - x1) / distance;\n const ny = (y2 - y1) / distance;\n const size = {\n arrow: 30,\n bar: 15,\n dot: 15,\n triangle: 15\n }[arrowhead]; // pixels (will differ for each arrowhead)\n\n let length = 0;\n\n if (arrowhead === \"arrow\") {\n // Length for -> arrows is based on the length of the last section\n const [cx, cy] = element.points[element.points.length - 1];\n const [px, py] = element.points.length > 1 ? element.points[element.points.length - 2] : [0, 0];\n length = Math.hypot(cx - px, cy - py);\n } else {\n // Length for other arrowhead types is based on the total length of the line\n for (let i = 0; i < element.points.length; i++) {\n const [px, py] = element.points[i - 1] || [0, 0];\n const [cx, cy] = element.points[i];\n length += Math.hypot(cx - px, cy - py);\n }\n } // Scale down the arrowhead until we hit a certain size so that it doesn't look weird.\n // This value is selected by minimizing a minimum size with the last segment of the arrowhead\n\n\n const minSize = Math.min(size, length / 2);\n const xs = x2 - nx * minSize;\n const ys = y2 - ny * minSize;\n\n if (arrowhead === \"dot\") {\n const r = Math.hypot(ys - y2, xs - x2) + element.strokeWidth;\n return [x2, y2, r];\n }\n\n const angle = {\n arrow: 20,\n bar: 90,\n triangle: 25\n }[arrowhead]; // degrees\n // Return points\n\n const [x3, y3] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(xs, ys, x2, y2, -angle * Math.PI / 180);\n const [x4, y4] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(xs, ys, x2, y2, angle * Math.PI / 180);\n return [x2, y2, x3, y3, x4, y4];\n};\n\nconst generateLinearElementShape = element => {\n const generator = roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_1__[\"default\"].generator();\n const options = (0,_scene_Shape__WEBPACK_IMPORTED_MODULE_2__.generateRoughOptions)(element);\n\n const method = (() => {\n if (element.roundness) {\n return \"curve\";\n }\n\n if (options.fill) {\n return \"polygon\";\n }\n\n return \"linearPath\";\n })();\n\n return generator[method](element.points, options);\n};\n\nconst getLinearElementRotatedBounds = (element, cx, cy) => {\n var _a;\n\n if (element.points.length < 2) {\n const [pointX, pointY] = element.points[0];\n const [x, y] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(element.x + pointX, element.y + pointY, cx, cy, element.angle);\n let coords = [x, y, x, y];\n const boundTextElement = (0,_textElement__WEBPACK_IMPORTED_MODULE_5__.getBoundTextElement)(element);\n\n if (boundTextElement) {\n const coordsWithBoundText = _linearElementEditor__WEBPACK_IMPORTED_MODULE_6__.LinearElementEditor.getMinMaxXYWithBoundText(element, [x, y, x, y], boundTextElement);\n coords = [coordsWithBoundText[0], coordsWithBoundText[1], coordsWithBoundText[2], coordsWithBoundText[3]];\n }\n\n return coords;\n } // first element is always the curve\n\n\n const cachedShape = (_a = _scene_ShapeCache__WEBPACK_IMPORTED_MODULE_7__.ShapeCache.get(element)) === null || _a === void 0 ? void 0 : _a[0];\n const shape = cachedShape !== null && cachedShape !== void 0 ? cachedShape : generateLinearElementShape(element);\n const ops = getCurvePathOps(shape);\n\n const transformXY = (x, y) => (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(element.x + x, element.y + y, cx, cy, element.angle);\n\n const res = getMinMaxXYFromCurvePathOps(ops, transformXY);\n let coords = [res[0], res[1], res[2], res[3]];\n const boundTextElement = (0,_textElement__WEBPACK_IMPORTED_MODULE_5__.getBoundTextElement)(element);\n\n if (boundTextElement) {\n const coordsWithBoundText = _linearElementEditor__WEBPACK_IMPORTED_MODULE_6__.LinearElementEditor.getMinMaxXYWithBoundText(element, coords, boundTextElement);\n coords = [coordsWithBoundText[0], coordsWithBoundText[1], coordsWithBoundText[2], coordsWithBoundText[3]];\n }\n\n return coords;\n};\n\nconst getElementBounds = element => {\n return ElementBounds.getBounds(element);\n};\nconst getCommonBounds = elements => {\n if (!elements.length) {\n return [0, 0, 0, 0];\n }\n\n let minX = Infinity;\n let maxX = -Infinity;\n let minY = Infinity;\n let maxY = -Infinity;\n elements.forEach(element => {\n const [x1, y1, x2, y2] = getElementBounds(element);\n minX = Math.min(minX, x1);\n minY = Math.min(minY, y1);\n maxX = Math.max(maxX, x2);\n maxY = Math.max(maxY, y2);\n });\n return [minX, minY, maxX, maxY];\n};\nconst getDraggedElementsBounds = (elements, dragOffset) => {\n const [minX, minY, maxX, maxY] = getCommonBounds(elements);\n return [minX + dragOffset.x, minY + dragOffset.y, maxX + dragOffset.x, maxY + dragOffset.y];\n};\nconst getResizedElementAbsoluteCoords = (element, nextWidth, nextHeight, normalizePoints) => {\n if (!((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isLinearElement)(element) || (0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isFreeDrawElement)(element))) {\n return [element.x, element.y, element.x + nextWidth, element.y + nextHeight];\n }\n\n const points = (0,_points__WEBPACK_IMPORTED_MODULE_4__.rescalePoints)(0, nextWidth, (0,_points__WEBPACK_IMPORTED_MODULE_4__.rescalePoints)(1, nextHeight, element.points, normalizePoints), normalizePoints);\n let bounds;\n\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isFreeDrawElement)(element)) {\n // Free Draw\n bounds = getBoundsFromPoints(points);\n } else {\n // Line\n const gen = roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_1__[\"default\"].generator();\n const curve = !element.roundness ? gen.linearPath(points, (0,_scene_Shape__WEBPACK_IMPORTED_MODULE_2__.generateRoughOptions)(element)) : gen.curve(points, (0,_scene_Shape__WEBPACK_IMPORTED_MODULE_2__.generateRoughOptions)(element));\n const ops = getCurvePathOps(curve);\n bounds = getMinMaxXYFromCurvePathOps(ops);\n }\n\n const [minX, minY, maxX, maxY] = bounds;\n return [minX + element.x, minY + element.y, maxX + element.x, maxY + element.y];\n};\nconst getElementPointsCoords = (element, points) => {\n // This might be computationally heavey\n const gen = roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_1__[\"default\"].generator();\n const curve = element.roundness == null ? gen.linearPath(points, (0,_scene_Shape__WEBPACK_IMPORTED_MODULE_2__.generateRoughOptions)(element)) : gen.curve(points, (0,_scene_Shape__WEBPACK_IMPORTED_MODULE_2__.generateRoughOptions)(element));\n const ops = getCurvePathOps(curve);\n const [minX, minY, maxX, maxY] = getMinMaxXYFromCurvePathOps(ops);\n return [minX + element.x, minY + element.y, maxX + element.x, maxY + element.y];\n};\nconst getClosestElementBounds = (elements, from) => {\n if (!elements.length) {\n return [0, 0, 0, 0];\n }\n\n let minDistance = Infinity;\n let closestElement = elements[0];\n elements.forEach(element => {\n const [x1, y1, x2, y2] = getElementBounds(element);\n const distance = (0,_math__WEBPACK_IMPORTED_MODULE_0__.distance2d)((x1 + x2) / 2, (y1 + y2) / 2, from.x, from.y);\n\n if (distance < minDistance) {\n minDistance = distance;\n closestElement = element;\n }\n });\n return getElementBounds(closestElement);\n};\nconst getCommonBoundingBox = elements => {\n const [minX, minY, maxX, maxY] = getCommonBounds(elements);\n return {\n minX,\n minY,\n maxX,\n maxY,\n width: maxX - minX,\n height: maxY - minY,\n midX: (minX + maxX) / 2,\n midY: (minY + maxY) / 2\n };\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,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O0FBQ0E7QUFDTztBQUNQO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBLEdBQUc7QUFDSDtBQUNBO0FBQ087QUFDUDtBQUNBO0FBQ0E7QUFDTztBQUNQLFFBQVEsNERBQWUsYUFBYSw4REFBaUI7QUFDckQ7QUFDQTs7QUFFQSxpQkFBaUIsc0RBQWEsZUFBZSxzREFBYTtBQUMxRDs7QUFFQSxNQUFNLDhEQUFpQjtBQUN2QjtBQUNBO0FBQ0EsSUFBSTtBQUNKO0FBQ0EsZ0JBQWdCLG1FQUFlO0FBQy9CLDhEQUE4RCxrRUFBb0IsK0JBQStCLGtFQUFvQjtBQUNySTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ087QUFDUDtBQUNBLGNBQWMsbUVBQWU7QUFDN0IsbUVBQW1FLGtFQUFvQiwrQkFBK0Isa0VBQW9CO0FBQzFJO0FBQ0E7QUFDQTtBQUNBO0FBQ087QUFDUDtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQSxxQkFBcUIsaURBQVU7O0FBRS9CO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsR0FBRztBQUNIO0FBQ0E7QUFDTztBQUNQO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSIsInNvdXJjZXMiOlsid2VicGFjazovLy8uLi8uLi9lbGVtZW50L2JvdW5kcy50cz9mY2EwIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGRpc3RhbmNlMmQsIHJvdGF0ZSwgcm90YXRlUG9pbnQgfSBmcm9tIFwiLi4vbWF0aFwiO1xuaW1wb3J0IHJvdWdoIGZyb20gXCJyb3VnaGpzL2Jpbi9yb3VnaFwiO1xuaW1wb3J0IHsgZ2VuZXJhdGVSb3VnaE9wdGlvbnMgfSBmcm9tIFwiLi4vc2NlbmUvU2hhcGVcIjtcbmltcG9ydCB7IGlzQXJyb3dFbGVtZW50LCBpc0JvdW5kVG9Db250YWluZXIsIGlzRnJlZURyYXdFbGVtZW50LCBpc0xpbmVhckVsZW1lbnQsIGlzVGV4dEVsZW1lbnQgfSBmcm9tIFwiLi90eXBlQ2hlY2tzXCI7XG5pbXBvcnQgeyByZXNjYWxlUG9pbnRzIH0gZnJvbSBcIi4uL3BvaW50c1wiO1xuaW1wb3J0IHsgZ2V0Qm91bmRUZXh0RWxlbWVudCwgZ2V0Q29udGFpbmVyRWxlbWVudCB9IGZyb20gXCIuL3RleHRFbGVtZW50XCI7XG5pbXBvcnQgeyBMaW5lYXJFbGVtZW50RWRpdG9yIH0gZnJvbSBcIi4vbGluZWFyRWxlbWVudEVkaXRvclwiO1xuaW1wb3J0IHsgU2hhcGVDYWNoZSB9IGZyb20gXCIuLi9zY2VuZS9TaGFwZUNhY2hlXCI7XG5leHBvcnQgY2xhc3MgRWxlbWVudEJvdW5kcyB7XG4gIHN0YXRpYyBnZXRCb3VuZHMoZWxlbWVudCkge1xuICAgIGNvbnN0IGNhY2hlZEJvdW5kcyA9IEVsZW1lbnRCb3VuZHMuYm91bmRzQ2FjaGUuZ2V0KGVsZW1lbnQpO1xuXG4gICAgaWYgKChjYWNoZWRCb3VuZHMgPT09IG51bGwgfHwgY2FjaGVkQm91bmRzID09PSB2b2lkIDAgPyB2b2lkIDAgOiBjYWNoZWRCb3VuZHMudmVyc2lvbikgJiYgY2FjaGVkQm91bmRzLnZlcnNpb24gPT09IGVsZW1lbnQudmVyc2lvbiAmJiAvLyB3ZSBkb24ndCBpbnZhbGlkYXRlIGNhY2hlIHdoZW4gd2UgdXBkYXRlIGNvbnRhaW5lcnMgYW5kIG5vdCBsYWJlbHMsXG4gICAgLy8gd2hpY2ggaXMgY2F1c2luZyBwcm9ibGVtcyBkb3duIHRoZSBsaW5lLiBGaXggVEJBLlxuICAgICFpc0JvdW5kVG9Db250YWluZXIoZWxlbWVudCkpIHtcbiAgICAgIHJldHVybiBjYWNoZWRCb3VuZHMuYm91bmRzO1xuICAgIH1cblxuICAgIGNvbnN0IGJvdW5kcyA9IEVsZW1lbnRCb3VuZHMuY2FsY3VsYXRlQm91bmRzKGVsZW1lbnQpO1xuICAgIEVsZW1lbnRCb3VuZHMuYm91bmRzQ2FjaGUuc2V0KGVsZW1lbnQsIHtcbiAgICAgIHZlcnNpb246IGVsZW1lbnQudmVyc2lvbixcbiAgICAgIGJvdW5kc1xuICAgIH0pO1xuICAgIHJldHVybiBib3VuZHM7XG4gIH1cblxuICBzdGF0aWMgY2FsY3VsYXRlQm91bmRzKGVsZW1lbnQpIHtcbiAgICBsZXQgYm91bmRzO1xuICAgIGNvbnN0IFt4MSwgeTEsIHgyLCB5MiwgY3gsIGN5XSA9IGdldEVsZW1lbnRBYnNvbHV0ZUNvb3JkcyhlbGVtZW50KTtcblxuICAgIGlmIChpc0ZyZWVEcmF3RWxlbWVudChlbGVtZW50KSkge1xuICAgICAgY29uc3QgW21pblgsIG1pblksIG1heFgsIG1heFldID0gZ2V0Qm91bmRzRnJvbVBvaW50cyhlbGVtZW50LnBvaW50cy5tYXAoKFt4LCB5XSkgPT4gcm90YXRlKHgsIHksIGN4IC0gZWxlbWVudC54LCBjeSAtIGVsZW1lbnQueSwgZWxlbWVudC5hbmdsZSkpKTtcbiAgICAgIHJldHVybiBbbWluWCArIGVsZW1lbnQueCwgbWluWSArIGVsZW1lbnQueSwgbWF4WCArIGVsZW1lbnQueCwgbWF4WSArIGVsZW1lbnQueV07XG4gICAgfSBlbHNlIGlmIChpc0xpbmVhckVsZW1lbnQoZWxlbWVudCkpIHtcbiAgICAgIGJvdW5kcyA9IGdldExpbmVhckVsZW1lbnRSb3RhdGVkQm91bmRzKGVsZW1lbnQsIGN4LCBjeSk7XG4gICAgfSBlbHNlIGlmIChlbGVtZW50LnR5cGUgPT09IFwiZGlhbW9uZFwiKSB7XG4gICAgICBjb25zdCBbeDExLCB5MTFdID0gcm90YXRlKGN4LCB5MSwgY3gsIGN5LCBlbGVtZW50LmFuZ2xlKTtcbiAgICAgIGNvbnN0IFt4MTIsIHkxMl0gPSByb3RhdGUoY3gsIHkyLCBjeCwgY3ksIGVsZW1lbnQuYW5nbGUpO1xuICAgICAgY29uc3QgW3gyMiwgeTIyXSA9IHJvdGF0ZSh4MSwgY3ksIGN4LCBjeSwgZWxlbWVudC5hbmdsZSk7XG4gICAgICBjb25zdCBbeDIxLCB5MjFdID0gcm90YXRlKHgyLCBjeSwgY3gsIGN5LCBlbGVtZW50LmFuZ2xlKTtcbiAgICAgIGNvbnN0IG1pblggPSBNYXRoLm1pbih4MTEsIHgxMiwgeDIyLCB4MjEpO1xuICAgICAgY29uc3QgbWluWSA9IE1hdGgubWluKHkxMSwgeTEyLCB5MjIsIHkyMSk7XG4gICAgICBjb25zdCBtYXhYID0gTWF0aC5tYXgoeDExLCB4MTIsIHgyMiwgeDIxKTtcbiAgICAgIGNvbnN0IG1heFkgPSBNYXRoLm1heCh5MTEsIHkxMiwgeTIyLCB5MjEpO1xuICAgICAgYm91bmRzID0gW21pblgsIG1pblksIG1heFgsIG1heFldO1xuICAgIH0gZWxzZSBpZiAoZWxlbWVudC50eXBlID09PSBcImVsbGlwc2VcIikge1xuICAgICAgY29uc3QgdyA9ICh4MiAtIHgxKSAvIDI7XG4gICAgICBjb25zdCBoID0gKHkyIC0geTEpIC8gMjtcbiAgICAgIGNvbnN0IGNvcyA9IE1hdGguY29zKGVsZW1lbnQuYW5nbGUpO1xuICAgICAgY29uc3Qgc2luID0gTWF0aC5zaW4oZWxlbWVudC5hbmdsZSk7XG4gICAgICBjb25zdCB3dyA9IE1hdGguaHlwb3QodyAqIGNvcywgaCAqIHNpbik7XG4gICAgICBjb25zdCBoaCA9IE1hdGguaHlwb3QoaCAqIGNvcywgdyAqIHNpbik7XG4gICAgICBib3VuZHMgPSBbY3ggLSB3dywgY3kgLSBoaCwgY3ggKyB3dywgY3kgKyBoaF07XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IFt4MTEsIHkxMV0gPSByb3RhdGUoeDEsIHkxLCBjeCwgY3ksIGVsZW1lbnQuYW5nbGUpO1xuICAgICAgY29uc3QgW3gxMiwgeTEyXSA9IHJvdGF0ZSh4MSwgeTIsIGN4LCBjeSwgZWxlbWVudC5hbmdsZSk7XG4gICAgICBjb25zdCBbeDIyLCB5MjJdID0gcm90YXRlKHgyLCB5MiwgY3gsIGN5LCBlbGVtZW50LmFuZ2xlKTtcbiAgICAgIGNvbnN0IFt4MjEsIHkyMV0gPSByb3RhdGUoeDIsIHkxLCBjeCwgY3ksIGVsZW1lbnQuYW5nbGUpO1xuICAgICAgY29uc3QgbWluWCA9IE1hdGgubWluKHgxMSwgeDEyLCB4MjIsIHgyMSk7XG4gICAgICBjb25zdCBtaW5ZID0gTWF0aC5taW4oeTExLCB5MTIsIHkyMiwgeTIxKTtcbiAgICAgIGNvbnN0IG1heFggPSBNYXRoLm1heCh4MTEsIHgxMiwgeDIyLCB4MjEpO1xuICAgICAgY29uc3QgbWF4WSA9IE1hdGgubWF4KHkxMSwgeTEyLCB5MjIsIHkyMSk7XG4gICAgICBib3VuZHMgPSBbbWluWCwgbWluWSwgbWF4WCwgbWF4WV07XG4gICAgfVxuXG4gICAgcmV0dXJuIGJvdW5kcztcbiAgfVxuXG59XG5FbGVtZW50Qm91bmRzLmJvdW5kc0NhY2hlID0gbmV3IFdlYWtNYXAoKTsgLy8gU2NlbmUgLT4gU2NlbmUgY29vcmRzLCBidXQgaW4geDEseDIseTEseTIgZm9ybWF0LlxuLy9cbi8vIElmIHRoZSBlbGVtZW50IGlzIGNyZWF0ZWQgZnJvbSByaWdodCB0byBsZWZ0LCB0aGUgd2lkdGggaXMgZ29pbmcgdG8gYmUgbmVnYXRpdmVcbi8vIFRoaXMgc2V0IG9mIGZ1bmN0aW9ucyByZXRyaWV2ZXMgdGhlIGFic29sdXRlIHBvc2l0aW9uIG9mIHRoZSA0IHBvaW50cy5cblxuZXhwb3J0IGNvbnN0IGdldEVsZW1lbnRBYnNvbHV0ZUNvb3JkcyA9IChlbGVtZW50LCBpbmNsdWRlQm91bmRUZXh0ID0gZmFsc2UpID0+IHtcbiAgaWYgKGlzRnJlZURyYXdFbGVtZW50KGVsZW1lbnQpKSB7XG4gICAgcmV0dXJuIGdldEZyZWVEcmF3RWxlbWVudEFic29sdXRlQ29vcmRzKGVsZW1lbnQpO1xuICB9IGVsc2UgaWYgKGlzTGluZWFyRWxlbWVudChlbGVtZW50KSkge1xuICAgIHJldHVybiBMaW5lYXJFbGVtZW50RWRpdG9yLmdldEVsZW1lbnRBYnNvbHV0ZUNvb3JkcyhlbGVtZW50LCBpbmNsdWRlQm91bmRUZXh0KTtcbiAgfSBlbHNlIGlmIChpc1RleHRFbGVtZW50KGVsZW1lbnQpKSB7XG4gICAgY29uc3QgY29udGFpbmVyID0gZ2V0Q29udGFpbmVyRWxlbWVudChlbGVtZW50KTtcblxuICAgIGlmIChpc0Fycm93RWxlbWVudChjb250YWluZXIpKSB7XG4gICAgICBjb25zdCBjb29yZHMgPSBMaW5lYXJFbGVtZW50RWRpdG9yLmdldEJvdW5kVGV4dEVsZW1lbnRQb3NpdGlvbihjb250YWluZXIsIGVsZW1lbnQpO1xuICAgICAgcmV0dXJuIFtjb29yZHMueCwgY29vcmRzLnksIGNvb3Jkcy54ICsgZWxlbWVudC53aWR0aCwgY29vcmRzLnkgKyBlbGVtZW50LmhlaWdodCwgY29vcmRzLnggKyBlbGVtZW50LndpZHRoIC8gMiwgY29vcmRzLnkgKyBlbGVtZW50LmhlaWdodCAvIDJdO1xuICAgIH1cbiAgfVxuXG4gIHJldHVybiBbZWxlbWVudC54LCBlbGVtZW50LnksIGVsZW1lbnQueCArIGVsZW1lbnQud2lkdGgsIGVsZW1lbnQueSArIGVsZW1lbnQuaGVpZ2h0LCBlbGVtZW50LnggKyBlbGVtZW50LndpZHRoIC8gMiwgZWxlbWVudC55ICsgZWxlbWVudC5oZWlnaHQgLyAyXTtcbn07XG4vKlxuICogZm9yIGEgZ2l2ZW4gZWxlbWVudCwgYGdldEVsZW1lbnRMaW5lU2VnbWVudHNgIHJldHVybnMgbGluZSBzZWdtZW50c1xuICogdGhhdCBjYW4gYmUgdXNlZCBmb3IgdmlzdWFsIGNvbGxpc2lvbiBkZXRlY3Rpb24gKHVzZWZ1bCBmb3IgZnJhbWVzKVxuICogYXMgb3Bwb3NlZCB0byBib3VuZGluZyBib3ggY29sbGlzaW9uIGRldGVjdGlvblxuICovXG5cbmV4cG9ydCBjb25zdCBnZXRFbGVtZW50TGluZVNlZ21lbnRzID0gZWxlbWVudCA9PiB7XG4gIGNvbnN0IFt4MSwgeTEsIHgyLCB5MiwgY3gsIGN5XSA9IGdldEVsZW1lbnRBYnNvbHV0ZUNvb3JkcyhlbGVtZW50KTtcbiAgY29uc3QgY2VudGVyID0gW2N4LCBjeV07XG5cbiAgaWYgKGlzTGluZWFyRWxlbWVudChlbGVtZW50KSB8fCBpc0ZyZWVEcmF3RWxlbWVudChlbGVtZW50KSkge1xuICAgIGNvbnN0IHNlZ21lbnRzID0gW107XG4gICAgbGV0IGkgPSAwO1xuXG4gICAgd2hpbGUgKGkgPCBlbGVtZW50LnBvaW50cy5sZW5ndGggLSAxKSB7XG4gICAgICBzZWdtZW50cy5wdXNoKFtyb3RhdGVQb2ludChbZWxlbWVudC5wb2ludHNbaV1bMF0gKyBlbGVtZW50LngsIGVsZW1lbnQucG9pbnRzW2ldWzFdICsgZWxlbWVudC55XSwgY2VudGVyLCBlbGVtZW50LmFuZ2xlKSwgcm90YXRlUG9pbnQoW2VsZW1lbnQucG9pbnRzW2kgKyAxXVswXSArIGVsZW1lbnQueCwgZWxlbWVudC5wb2ludHNbaSArIDFdWzFdICsgZWxlbWVudC55XSwgY2VudGVyLCBlbGVtZW50LmFuZ2xlKV0pO1xuICAgICAgaSsrO1xuICAgIH1cblxuICAgIHJldHVybiBzZWdtZW50cztcbiAgfVxuXG4gIGNvbnN0IFtudywgbmUsIHN3LCBzZSwgbiwgcywgdywgZV0gPSBbW3gxLCB5MV0sIFt4MiwgeTFdLCBbeDEsIHkyXSwgW3gyLCB5Ml0sIFtjeCwgeTFdLCBbY3gsIHkyXSwgW3gxLCBjeV0sIFt4MiwgY3ldXS5tYXAocG9pbnQgPT4gcm90YXRlUG9pbnQocG9pbnQsIGNlbnRlciwgZWxlbWVudC5hbmdsZSkpO1xuXG4gIGlmIChlbGVtZW50LnR5cGUgPT09IFwiZGlhbW9uZFwiKSB7XG4gICAgcmV0dXJuIFtbbiwgd10sIFtuLCBlXSwgW3MsIHddLCBbcywgZV1dO1xuICB9XG5cbiAgaWYgKGVsZW1lbnQudHlwZSA9PT0gXCJlbGxpcHNlXCIpIHtcbiAgICByZXR1cm4gW1tuLCB3XSwgW24sIGVdLCBbcywgd10sIFtzLCBlXSwgW24sIHddLCBbbiwgZV0sIFtzLCB3XSwgW3MsIGVdXTtcbiAgfVxuXG4gIHJldHVybiBbW253LCBuZV0sIFtzdywgc2VdLCBbbncsIHN3XSwgW25lLCBzZV0sIFtudywgZV0sIFtzdywgZV0sIFtuZSwgd10sIFtzZSwgd11dO1xufTtcbi8qKlxuICogU2NlbmUgLT4gU2NlbmUgY29vcmRzLCBidXQgaW4geDEseDIseTEseTIgZm9ybWF0LlxuICpcbiAqIFJlY3RhbmdsZSBoZXJlIG1lYW5zIGFueSByZWN0YW5ndWxhciBmcmFtZSwgbm90IGFuIGV4Y2FsaWRyYXcgZWxlbWVudC5cbiAqL1xuXG5leHBvcnQgY29uc3QgZ2V0UmVjdGFuZ2xlQm94QWJzb2x1dGVDb29yZHMgPSBib3hTY2VuZUNvb3JkcyA9PiB7XG4gIHJldHVybiBbYm94U2NlbmVDb29yZHMueCwgYm94U2NlbmVDb29yZHMueSwgYm94U2NlbmVDb29yZHMueCArIGJveFNjZW5lQ29vcmRzLndpZHRoLCBib3hTY2VuZUNvb3Jkcy55ICsgYm94U2NlbmVDb29yZHMuaGVpZ2h0LCBib3hTY2VuZUNvb3Jkcy54ICsgYm94U2NlbmVDb29yZHMud2lkdGggLyAyLCBib3hTY2VuZUNvb3Jkcy55ICsgYm94U2NlbmVDb29yZHMuaGVpZ2h0IC8gMl07XG59O1xuZXhwb3J0IGNvbnN0IHBvaW50UmVsYXRpdmVUbyA9IChlbGVtZW50LCBhYnNvbHV0ZUNvb3JkcykgPT4ge1xuICByZXR1cm4gW2Fic29sdXRlQ29vcmRzWzBdIC0gZWxlbWVudC54LCBhYnNvbHV0ZUNvb3Jkc1sxXSAtIGVsZW1lbnQueV07XG59O1xuZXhwb3J0IGNvbnN0IGdldERpYW1vbmRQb2ludHMgPSBlbGVtZW50ID0+IHtcbiAgLy8gSGVyZSB3ZSBhZGQgKzEgdG8gYXZvaWQgdGhlc2UgbnVtYmVycyB0byBiZSAwXG4gIC8vIG90aGVyd2lzZSByb3VnaC5qcyB3aWxsIHRocm93IGFuIGVycm9yIGNvbXBsYWluaW5nIGFib3V0IGl0XG4gIGNvbnN0IHRvcFggPSBNYXRoLmZsb29yKGVsZW1lbnQud2lkdGggLyAyKSArIDE7XG4gIGNvbnN0IHRvcFkgPSAwO1xuICBjb25zdCByaWdodFggPSBlbGVtZW50LndpZHRoO1xuICBjb25zdCByaWdodFkgPSBNYXRoLmZsb29yKGVsZW1lbnQuaGVpZ2h0IC8gMikgKyAxO1xuICBjb25zdCBib3R0b21YID0gdG9wWDtcbiAgY29uc3QgYm90dG9tWSA9IGVsZW1lbnQuaGVpZ2h0O1xuICBjb25zdCBsZWZ0WCA9IDA7XG4gIGNvbnN0IGxlZnRZID0gcmlnaHRZO1xuICByZXR1cm4gW3RvcFgsIHRvcFksIHJpZ2h0WCwgcmlnaHRZLCBib3R0b21YLCBib3R0b21ZLCBsZWZ0WCwgbGVmdFldO1xufTtcbmV4cG9ydCBjb25zdCBnZXRDdXJ2ZVBhdGhPcHMgPSBzaGFwZSA9PiB7XG4gIGZvciAoY29uc3Qgc2V0IG9mIHNoYXBlLnNldHMpIHtcbiAgICBpZiAoc2V0LnR5cGUgPT09IFwicGF0aFwiKSB7XG4gICAgICByZXR1cm4gc2V0Lm9wcztcbiAgICB9XG4gIH1cblxuICByZXR1cm4gc2hhcGUuc2V0c1swXS5vcHM7XG59OyAvLyByZWZlcmVuY2U6IGh0dHBzOi8vZWxpb3Qtam9uZXMuY29tLzIwMTkvMTIvY3ViaWMtYmV6aWVyLWN1cnZlLWJvdW5kaW5nLWJveGVzXG5cbmNvbnN0IGdldEJlemllclZhbHVlRm9yVCA9ICh0LCBwMCwgcDEsIHAyLCBwMykgPT4ge1xuICBjb25zdCBvbmVNaW51c1QgPSAxIC0gdDtcbiAgcmV0dXJuIE1hdGgucG93KG9uZU1pbnVzVCwgMykgKiBwMCArIDMgKiBNYXRoLnBvdyhvbmVNaW51c1QsIDIpICogdCAqIHAxICsgMyAqIG9uZU1pbnVzVCAqIE1hdGgucG93KHQsIDIpICogcDIgKyBNYXRoLnBvdyh0LCAzKSAqIHAzO1xufTtcblxuY29uc3Qgc29sdmVRdWFkcmF0aWMgPSAocDAsIHAxLCBwMiwgcDMpID0+IHtcbiAgY29uc3QgaSA9IHAxIC0gcDA7XG4gIGNvbnN0IGogPSBwMiAtIHAxO1xuICBjb25zdCBrID0gcDMgLSBwMjtcbiAgY29uc3QgYSA9IDMgKiBpIC0gNiAqIGogKyAzICogaztcbiAgY29uc3QgYiA9IDYgKiBqIC0gNiAqIGk7XG4gIGNvbnN0IGMgPSAzICogaTtcbiAgY29uc3Qgc3FydFBhcnQgPSBiICogYiAtIDQgKiBhICogYztcbiAgY29uc3QgaGFzU29sdXRpb24gPSBzcXJ0UGFydCA+PSAwO1xuXG4gIGlmICghaGFzU29sdXRpb24pIHtcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICBsZXQgczEgPSBudWxsO1xuICBsZXQgczIgPSBudWxsO1xuICBsZXQgdDEgPSBJbmZpbml0eTtcbiAgbGV0IHQyID0gSW5maW5pdHk7XG5cbiAgaWYgKGEgPT09IDApIHtcbiAgICB0MSA9IHQyID0gLWMgLyBiO1xuICB9IGVsc2Uge1xuICAgIHQxID0gKC1iICsgTWF0aC5zcXJ0KHNxcnRQYXJ0KSkgLyAoMiAqIGEpO1xuICAgIHQyID0gKC1iIC0gTWF0aC5zcXJ0KHNxcnRQYXJ0KSkgLyAoMiAqIGEpO1xuICB9XG5cbiAgaWYgKHQxID49IDAgJiYgdDEgPD0gMSkge1xuICAgIHMxID0gZ2V0QmV6aWVyVmFsdWVGb3JUKHQxLCBwMCwgcDEsIHAyLCBwMyk7XG4gIH1cblxuICBpZiAodDIgPj0gMCAmJiB0MiA8PSAxKSB7XG4gICAgczIgPSBnZXRCZXppZXJWYWx1ZUZvclQodDIsIHAwLCBwMSwgcDIsIHAzKTtcbiAgfVxuXG4gIHJldHVybiBbczEsIHMyXTtcbn07XG5cbmNvbnN0IGdldEN1YmljQmV6aWVyQ3VydmVCb3VuZCA9IChwMCwgcDEsIHAyLCBwMykgPT4ge1xuICBjb25zdCBzb2xYID0gc29sdmVRdWFkcmF0aWMocDBbMF0sIHAxWzBdLCBwMlswXSwgcDNbMF0pO1xuICBjb25zdCBzb2xZID0gc29sdmVRdWFkcmF0aWMocDBbMV0sIHAxWzFdLCBwMlsxXSwgcDNbMV0pO1xuICBsZXQgbWluWCA9IE1hdGgubWluKHAwWzBdLCBwM1swXSk7XG4gIGxldCBtYXhYID0gTWF0aC5tYXgocDBbMF0sIHAzWzBdKTtcblxuICBpZiAoc29sWCkge1xuICAgIGNvbnN0IHhzID0gc29sWC5maWx0ZXIoeCA9PiB4ICE9PSBudWxsKTtcbiAgICBtaW5YID0gTWF0aC5taW4obWluWCwgLi4ueHMpO1xuICAgIG1heFggPSBNYXRoLm1heChtYXhYLCAuLi54cyk7XG4gIH1cblxuICBsZXQgbWluWSA9IE1hdGgubWluKHAwWzFdLCBwM1sxXSk7XG4gIGxldCBtYXhZID0gTWF0aC5tYXgocDBbMV0sIHAzWzFdKTtcblxuICBpZiAoc29sWSkge1xuICAgIGNvbnN0IHlzID0gc29sWS5maWx0ZXIoeSA9PiB5ICE9PSBudWxsKTtcbiAgICBtaW5ZID0gTWF0aC5taW4obWluWSwgLi4ueXMpO1xuICAgIG1heFkgPSBNYXRoLm1heChtYXhZLCAuLi55cyk7XG4gIH1cblxuICByZXR1cm4gW21pblgsIG1pblksIG1heFgsIG1heFldO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldE1pbk1heFhZRnJvbUN1cnZlUGF0aE9wcyA9IChvcHMsIHRyYW5zZm9ybVhZKSA9PiB7XG4gIGxldCBjdXJyZW50UCA9IFswLCAwXTtcbiAgY29uc3Qge1xuICAgIG1pblgsXG4gICAgbWluWSxcbiAgICBtYXhYLFxuICAgIG1heFlcbiAgfSA9IG9wcy5yZWR1Y2UoKGxpbWl0cywge1xuICAgIG9wLFxuICAgIGRhdGFcbiAgfSkgPT4ge1xuICAgIC8vIFRoZXJlIGFyZSBvbmx5IGZvdXIgb3BlcmF0aW9uIHR5cGVzOlxuICAgIC8vIG1vdmUsIGJjdXJ2ZVRvLCBsaW5lVG8sIGFuZCBjdXJ2ZVRvXG4gICAgaWYgKG9wID09PSBcIm1vdmVcIikge1xuICAgICAgLy8gY2hhbmdlIHN0YXJ0aW5nIHBvaW50XG4gICAgICBjdXJyZW50UCA9IGRhdGE7IC8vIG1vdmUgb3BlcmF0aW9uIGRvZXMgbm90IGRyYXcgYW55dGhpbmc7IHNvLCBpdCBhbHdheXNcbiAgICAgIC8vIHJldHVybnMgZmFsc2VcbiAgICB9IGVsc2UgaWYgKG9wID09PSBcImJjdXJ2ZVRvXCIpIHtcbiAgICAgIGNvbnN0IF9wMSA9IFtkYXRhWzBdLCBkYXRhWzFdXTtcbiAgICAgIGNvbnN0IF9wMiA9IFtkYXRhWzJdLCBkYXRhWzNdXTtcbiAgICAgIGNvbnN0IF9wMyA9IFtkYXRhWzRdLCBkYXRhWzVdXTtcbiAgICAgIGNvbnN0IHAxID0gdHJhbnNmb3JtWFkgPyB0cmFuc2Zvcm1YWSguLi5fcDEpIDogX3AxO1xuICAgICAgY29uc3QgcDIgPSB0cmFuc2Zvcm1YWSA/IHRyYW5zZm9ybVhZKC4uLl9wMikgOiBfcDI7XG4gICAgICBjb25zdCBwMyA9IHRyYW5zZm9ybVhZID8gdHJhbnNmb3JtWFkoLi4uX3AzKSA6IF9wMztcbiAgICAgIGNvbnN0IHAwID0gdHJhbnNmb3JtWFkgPyB0cmFuc2Zvcm1YWSguLi5jdXJyZW50UCkgOiBjdXJyZW50UDtcbiAgICAgIGN1cnJlbnRQID0gX3AzO1xuICAgICAgY29uc3QgW21pblgsIG1pblksIG1heFgsIG1heFldID0gZ2V0Q3ViaWNCZXppZXJDdXJ2ZUJvdW5kKHAwLCBwMSwgcDIsIHAzKTtcbiAgICAgIGxpbWl0cy5taW5YID0gTWF0aC5taW4obGltaXRzLm1pblgsIG1pblgpO1xuICAgICAgbGltaXRzLm1pblkgPSBNYXRoLm1pbihsaW1pdHMubWluWSwgbWluWSk7XG4gICAgICBsaW1pdHMubWF4WCA9IE1hdGgubWF4KGxpbWl0cy5tYXhYLCBtYXhYKTtcbiAgICAgIGxpbWl0cy5tYXhZID0gTWF0aC5tYXgobGltaXRzLm1heFksIG1heFkpO1xuICAgIH0gZWxzZSBpZiAob3AgPT09IFwibGluZVRvXCIpIHsvLyBUT0RPOiBJbXBsZW1lbnQgdGhpc1xuICAgIH0gZWxzZSBpZiAob3AgPT09IFwicWN1cnZlVG9cIikgey8vIFRPRE86IEltcGxlbWVudCB0aGlzXG4gICAgfVxuXG4gICAgcmV0dXJuIGxpbWl0cztcbiAgfSwge1xuICAgIG1pblg6IEluZmluaXR5LFxuICAgIG1pblk6IEluZmluaXR5LFxuICAgIG1heFg6IC1JbmZpbml0eSxcbiAgICBtYXhZOiAtSW5maW5pdHlcbiAgfSk7XG4gIHJldHVybiBbbWluWCwgbWluWSwgbWF4WCwgbWF4WV07XG59O1xuZXhwb3J0IGNvbnN0IGdldEJvdW5kc0Zyb21Qb2ludHMgPSBwb2ludHMgPT4ge1xuICBsZXQgbWluWCA9IEluZmluaXR5O1xuICBsZXQgbWluWSA9IEluZmluaXR5O1xuICBsZXQgbWF4WCA9IC1JbmZpbml0eTtcbiAgbGV0IG1heFkgPSAtSW5maW5pdHk7XG5cbiAgZm9yIChjb25zdCBbeCwgeV0gb2YgcG9pbnRzKSB7XG4gICAgbWluWCA9IE1hdGgubWluKG1pblgsIHgpO1xuICAgIG1pblkgPSBNYXRoLm1pbihtaW5ZLCB5KTtcbiAgICBtYXhYID0gTWF0aC5tYXgobWF4WCwgeCk7XG4gICAgbWF4WSA9IE1hdGgubWF4KG1heFksIHkpO1xuICB9XG5cbiAgcmV0dXJuIFttaW5YLCBtaW5ZLCBtYXhYLCBtYXhZXTtcbn07XG5cbmNvbnN0IGdldEZyZWVEcmF3RWxlbWVudEFic29sdXRlQ29vcmRzID0gZWxlbWVudCA9PiB7XG4gIGNvbnN0IFttaW5YLCBtaW5ZLCBtYXhYLCBtYXhZXSA9IGdldEJvdW5kc0Zyb21Qb2ludHMoZWxlbWVudC5wb2ludHMpO1xuICBjb25zdCB4MSA9IG1pblggKyBlbGVtZW50Lng7XG4gIGNvbnN0IHkxID0gbWluWSArIGVsZW1lbnQueTtcbiAgY29uc3QgeDIgPSBtYXhYICsgZWxlbWVudC54O1xuICBjb25zdCB5MiA9IG1heFkgKyBlbGVtZW50Lnk7XG4gIHJldHVybiBbeDEsIHkxLCB4MiwgeTIsICh4MSArIHgyKSAvIDIsICh5MSArIHkyKSAvIDJdO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEFycm93aGVhZFBvaW50cyA9IChlbGVtZW50LCBzaGFwZSwgcG9zaXRpb24sIGFycm93aGVhZCkgPT4ge1xuICBjb25zdCBvcHMgPSBnZXRDdXJ2ZVBhdGhPcHMoc2hhcGVbMF0pO1xuXG4gIGlmIChvcHMubGVuZ3RoIDwgMSkge1xuICAgIHJldHVybiBudWxsO1xuICB9IC8vIFRoZSBpbmRleCBvZiB0aGUgYkN1cnZlIG9wZXJhdGlvbiB0byBleGFtaW5lLlxuXG5cbiAgY29uc3QgaW5kZXggPSBwb3NpdGlvbiA9PT0gXCJzdGFydFwiID8gMSA6IG9wcy5sZW5ndGggLSAxO1xuICBjb25zdCBkYXRhID0gb3BzW2luZGV4XS5kYXRhO1xuICBjb25zdCBwMyA9IFtkYXRhWzRdLCBkYXRhWzVdXTtcbiAgY29uc3QgcDIgPSBbZGF0YVsyXSwgZGF0YVszXV07XG4gIGNvbnN0IHAxID0gW2RhdGFbMF0sIGRhdGFbMV1dOyAvLyBXZSBuZWVkIHRvIGZpbmQgcDAgb2YgdGhlIGJlemllciBjdXJ2ZS5cbiAgLy8gSXQgaXMgdHlwaWNhbGx5IHRoZSBsYXN0IHBvaW50IG9mIHRoZSBwcmV2aW91c1xuICAvLyBjdXJ2ZTsgaXQgY2FuIGFsc28gYmUgdGhlIHBvc2l0aW9uIG9mIG1vdmVUbyBvcGVyYXRpb24uXG5cbiAgY29uc3QgcHJldk9wID0gb3BzW2luZGV4IC0gMV07XG4gIGxldCBwMCA9IFswLCAwXTtcblxuICBpZiAocHJldk9wLm9wID09PSBcIm1vdmVcIikge1xuICAgIHAwID0gcHJldk9wLmRhdGE7XG4gIH0gZWxzZSBpZiAocHJldk9wLm9wID09PSBcImJjdXJ2ZVRvXCIpIHtcbiAgICBwMCA9IFtwcmV2T3AuZGF0YVs0XSwgcHJldk9wLmRhdGFbNV1dO1xuICB9IC8vIEIodCkgPSBwMCAqICgxLXQpXjMgKyAzcDEgKiB0ICogKDEtdCleMiArIDNwMiAqIHReMiAqICgxLXQpICsgcDMgKiB0XjNcblxuXG4gIGNvbnN0IGVxdWF0aW9uID0gKHQsIGlkeCkgPT4gTWF0aC5wb3coMSAtIHQsIDMpICogcDNbaWR4XSArIDMgKiB0ICogTWF0aC5wb3coMSAtIHQsIDIpICogcDJbaWR4XSArIDMgKiBNYXRoLnBvdyh0LCAyKSAqICgxIC0gdCkgKiBwMVtpZHhdICsgcDBbaWR4XSAqIE1hdGgucG93KHQsIDMpOyAvLyBFZSBrbm93IHRoZSBsYXN0IHBvaW50IG9mIHRoZSBhcnJvdyAob3IgdGhlIGZpcnN0LCBpZiBzdGFydCBhcnJvd2hlYWQpLlxuXG5cbiAgY29uc3QgW3gyLCB5Ml0gPSBwb3NpdGlvbiA9PT0gXCJzdGFydFwiID8gcDAgOiBwMzsgLy8gQnkgdXNpbmcgY3ViaWMgYmV6aWVyIGVxdWF0aW9uIChCKHQpKSBhbmQgdGhlIGdpdmVuIHBhcmFtZXRlcnMsXG4gIC8vIHdlIGNhbGN1bGF0ZSBhIHBvaW50IHRoYXQgaXMgY2xvc2VyIHRvIHRoZSBsYXN0IHBvaW50LlxuICAvLyBUaGUgdmFsdWUgMC4zIGlzIGNob3NlbiBhcmJpdHJhcmlseSBhbmQgaXQgd29ya3MgYmVzdCBmb3IgYWxsXG4gIC8vIHRoZSB0ZXN0ZWQgY2FzZXMuXG5cbiAgY29uc3QgW3gxLCB5MV0gPSBbZXF1YXRpb24oMC4zLCAwKSwgZXF1YXRpb24oMC4zLCAxKV07IC8vIEZpbmQgdGhlIG5vcm1hbGl6ZWQgZGlyZWN0aW9uIHZlY3RvciBiYXNlZCBvbiB0aGVcbiAgLy8gcHJldmlvdXNseSBjYWxjdWxhdGVkIHBvaW50cy5cblxuICBjb25zdCBkaXN0YW5jZSA9IE1hdGguaHlwb3QoeDIgLSB4MSwgeTIgLSB5MSk7XG4gIGNvbnN0IG54ID0gKHgyIC0geDEpIC8gZGlzdGFuY2U7XG4gIGNvbnN0IG55ID0gKHkyIC0geTEpIC8gZGlzdGFuY2U7XG4gIGNvbnN0IHNpemUgPSB7XG4gICAgYXJyb3c6IDMwLFxuICAgIGJhcjogMTUsXG4gICAgZG90OiAxNSxcbiAgICB0cmlhbmdsZTogMTVcbiAgfVthcnJvd2hlYWRdOyAvLyBwaXhlbHMgKHdpbGwgZGlmZmVyIGZvciBlYWNoIGFycm93aGVhZClcblxuICBsZXQgbGVuZ3RoID0gMDtcblxuICBpZiAoYXJyb3doZWFkID09PSBcImFycm93XCIpIHtcbiAgICAvLyBMZW5ndGggZm9yIC0+IGFycm93cyBpcyBiYXNlZCBvbiB0aGUgbGVuZ3RoIG9mIHRoZSBsYXN0IHNlY3Rpb25cbiAgICBjb25zdCBbY3gsIGN5XSA9IGVsZW1lbnQucG9pbnRzW2VsZW1lbnQucG9pbnRzLmxlbmd0aCAtIDFdO1xuICAgIGNvbnN0IFtweCwgcHldID0gZWxlbWVudC5wb2ludHMubGVuZ3RoID4gMSA/IGVsZW1lbnQucG9pbnRzW2VsZW1lbnQucG9pbnRzLmxlbmd0aCAtIDJdIDogWzAsIDBdO1xuICAgIGxlbmd0aCA9IE1hdGguaHlwb3QoY3ggLSBweCwgY3kgLSBweSk7XG4gIH0gZWxzZSB7XG4gICAgLy8gTGVuZ3RoIGZvciBvdGhlciBhcnJvd2hlYWQgdHlwZXMgaXMgYmFzZWQgb24gdGhlIHRvdGFsIGxlbmd0aCBvZiB0aGUgbGluZVxuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgZWxlbWVudC5wb2ludHMubGVuZ3RoOyBpKyspIHtcbiAgICAgIGNvbnN0IFtweCwgcHldID0gZWxlbWVudC5wb2ludHNbaSAtIDFdIHx8IFswLCAwXTtcbiAgICAgIGNvbnN0IFtjeCwgY3ldID0gZWxlbWVudC5wb2ludHNbaV07XG4gICAgICBsZW5ndGggKz0gTWF0aC5oeXBvdChjeCAtIHB4LCBjeSAtIHB5KTtcbiAgICB9XG4gIH0gLy8gU2NhbGUgZG93biB0aGUgYXJyb3doZWFkIHVudGlsIHdlIGhpdCBhIGNlcnRhaW4gc2l6ZSBzbyB0aGF0IGl0IGRvZXNuJ3QgbG9vayB3ZWlyZC5cbiAgLy8gVGhpcyB2YWx1ZSBpcyBzZWxlY3RlZCBieSBtaW5pbWl6aW5nIGEgbWluaW11bSBzaXplIHdpdGggdGhlIGxhc3Qgc2VnbWVudCBvZiB0aGUgYXJyb3doZWFkXG5cblxuICBjb25zdCBtaW5TaXplID0gTWF0aC5taW4oc2l6ZSwgbGVuZ3RoIC8gMik7XG4gIGNvbnN0IHhzID0geDIgLSBueCAqIG1pblNpemU7XG4gIGNvbnN0IHlzID0geTIgLSBueSAqIG1pblNpemU7XG5cbiAgaWYgKGFycm93aGVhZCA9PT0gXCJkb3RcIikge1xuICAgIGNvbnN0IHIgPSBNYXRoLmh5cG90KHlzIC0geTIsIHhzIC0geDIpICsgZWxlbWVudC5zdHJva2VXaWR0aDtcbiAgICByZXR1cm4gW3gyLCB5Miwgcl07XG4gIH1cblxuICBjb25zdCBhbmdsZSA9IHtcbiAgICBhcnJvdzogMjAsXG4gICAgYmFyOiA5MCxcbiAgICB0cmlhbmdsZTogMjVcbiAgfVthcnJvd2hlYWRdOyAvLyBkZWdyZWVzXG4gIC8vIFJldHVybiBwb2ludHNcblxuICBjb25zdCBbeDMsIHkzXSA9IHJvdGF0ZSh4cywgeXMsIHgyLCB5MiwgLWFuZ2xlICogTWF0aC5QSSAvIDE4MCk7XG4gIGNvbnN0IFt4NCwgeTRdID0gcm90YXRlKHhzLCB5cywgeDIsIHkyLCBhbmdsZSAqIE1hdGguUEkgLyAxODApO1xuICByZXR1cm4gW3gyLCB5MiwgeDMsIHkzLCB4NCwgeTRdO1xufTtcblxuY29uc3QgZ2VuZXJhdGVMaW5lYXJFbGVtZW50U2hhcGUgPSBlbGVtZW50ID0+IHtcbiAgY29uc3QgZ2VuZXJhdG9yID0gcm91Z2guZ2VuZXJhdG9yKCk7XG4gIGNvbnN0IG9wdGlvbnMgPSBnZW5lcmF0ZVJvdWdoT3B0aW9ucyhlbGVtZW50KTtcblxuICBjb25zdCBtZXRob2QgPSAoKCkgPT4ge1xuICAgIGlmIChlbGVtZW50LnJvdW5kbmVzcykge1xuICAgICAgcmV0dXJuIFwiY3VydmVcIjtcbiAgICB9XG5cbiAgICBpZiAob3B0aW9ucy5maWxsKSB7XG4gICAgICByZXR1cm4gXCJwb2x5Z29uXCI7XG4gICAgfVxuXG4gICAgcmV0dXJuIFwibGluZWFyUGF0aFwiO1xuICB9KSgpO1xuXG4gIHJldHVybiBnZW5lcmF0b3JbbWV0aG9kXShlbGVtZW50LnBvaW50cywgb3B0aW9ucyk7XG59O1xuXG5jb25zdCBnZXRMaW5lYXJFbGVtZW50Um90YXRlZEJvdW5kcyA9IChlbGVtZW50LCBjeCwgY3kpID0+IHtcbiAgdmFyIF9hO1xuXG4gIGlmIChlbGVtZW50LnBvaW50cy5sZW5ndGggPCAyKSB7XG4gICAgY29uc3QgW3BvaW50WCwgcG9pbnRZXSA9IGVsZW1lbnQucG9pbnRzWzBdO1xuICAgIGNvbnN0IFt4LCB5XSA9IHJvdGF0ZShlbGVtZW50LnggKyBwb2ludFgsIGVsZW1lbnQueSArIHBvaW50WSwgY3gsIGN5LCBlbGVtZW50LmFuZ2xlKTtcbiAgICBsZXQgY29vcmRzID0gW3gsIHksIHgsIHldO1xuICAgIGNvbnN0IGJvdW5kVGV4dEVsZW1lbnQgPSBnZXRCb3VuZFRleHRFbGVtZW50KGVsZW1lbnQpO1xuXG4gICAgaWYgKGJvdW5kVGV4dEVsZW1lbnQpIHtcbiAgICAgIGNvbnN0IGNvb3Jkc1dpdGhCb3VuZFRleHQgPSBMaW5lYXJFbGVtZW50RWRpdG9yLmdldE1pbk1heFhZV2l0aEJvdW5kVGV4dChlbGVtZW50LCBbeCwgeSwgeCwgeV0sIGJvdW5kVGV4dEVsZW1lbnQpO1xuICAgICAgY29vcmRzID0gW2Nvb3Jkc1dpdGhCb3VuZFRleHRbMF0sIGNvb3Jkc1dpdGhCb3VuZFRleHRbMV0sIGNvb3Jkc1dpdGhCb3VuZFRleHRbMl0sIGNvb3Jkc1dpdGhCb3VuZFRleHRbM11dO1xuICAgIH1cblxuICAgIHJldHVybiBjb29yZHM7XG4gIH0gLy8gZmlyc3QgZWxlbWVudCBpcyBhbHdheXMgdGhlIGN1cnZlXG5cblxuICBjb25zdCBjYWNoZWRTaGFwZSA9IChfYSA9IFNoYXBlQ2FjaGUuZ2V0KGVsZW1lbnQpKSA9PT0gbnVsbCB8fCBfYSA9PT0gdm9pZCAwID8gdm9pZCAwIDogX2FbMF07XG4gIGNvbnN0IHNoYXBlID0gY2FjaGVkU2hhcGUgIT09IG51bGwgJiYgY2FjaGVkU2hhcGUgIT09IHZvaWQgMCA/IGNhY2hlZFNoYXBlIDogZ2VuZXJhdGVMaW5lYXJFbGVtZW50U2hhcGUoZWxlbWVudCk7XG4gIGNvbnN0IG9wcyA9IGdldEN1cnZlUGF0aE9wcyhzaGFwZSk7XG5cbiAgY29uc3QgdHJhbnNmb3JtWFkgPSAoeCwgeSkgPT4gcm90YXRlKGVsZW1lbnQueCArIHgsIGVsZW1lbnQueSArIHksIGN4LCBjeSwgZWxlbWVudC5hbmdsZSk7XG5cbiAgY29uc3QgcmVzID0gZ2V0TWluTWF4WFlGcm9tQ3VydmVQYXRoT3BzKG9wcywgdHJhbnNmb3JtWFkpO1xuICBsZXQgY29vcmRzID0gW3Jlc1swXSwgcmVzWzFdLCByZXNbMl0sIHJlc1szXV07XG4gIGNvbnN0IGJvdW5kVGV4dEVsZW1lbnQgPSBnZXRCb3VuZFRleHRFbGVtZW50KGVsZW1lbnQpO1xuXG4gIGlmIChib3VuZFRleHRFbGVtZW50KSB7XG4gICAgY29uc3QgY29vcmRzV2l0aEJvdW5kVGV4dCA9IExpbmVhckVsZW1lbnRFZGl0b3IuZ2V0TWluTWF4WFlXaXRoQm91bmRUZXh0KGVsZW1lbnQsIGNvb3JkcywgYm91bmRUZXh0RWxlbWVudCk7XG4gICAgY29vcmRzID0gW2Nvb3Jkc1dpdGhCb3VuZFRleHRbMF0sIGNvb3Jkc1dpdGhCb3VuZFRleHRbMV0sIGNvb3Jkc1dpdGhCb3VuZFRleHRbMl0sIGNvb3Jkc1dpdGhCb3VuZFRleHRbM11dO1xuICB9XG5cbiAgcmV0dXJuIGNvb3Jkcztcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRFbGVtZW50Qm91bmRzID0gZWxlbWVudCA9PiB7XG4gIHJldHVybiBFbGVtZW50Qm91bmRzLmdldEJvdW5kcyhlbGVtZW50KTtcbn07XG5leHBvcnQgY29uc3QgZ2V0Q29tbW9uQm91bmRzID0gZWxlbWVudHMgPT4ge1xuICBpZiAoIWVsZW1lbnRzLmxlbmd0aCkge1xuICAgIHJldHVybiBbMCwgMCwgMCwgMF07XG4gIH1cblxuICBsZXQgbWluWCA9IEluZmluaXR5O1xuICBsZXQgbWF4WCA9IC1JbmZpbml0eTtcbiAgbGV0IG1pblkgPSBJbmZpbml0eTtcbiAgbGV0IG1heFkgPSAtSW5maW5pdHk7XG4gIGVsZW1lbnRzLmZvckVhY2goZWxlbWVudCA9PiB7XG4gICAgY29uc3QgW3gxLCB5MSwgeDIsIHkyXSA9IGdldEVsZW1lbnRCb3VuZHMoZWxlbWVudCk7XG4gICAgbWluWCA9IE1hdGgubWluKG1pblgsIHgxKTtcbiAgICBtaW5ZID0gTWF0aC5taW4obWluWSwgeTEpO1xuICAgIG1heFggPSBNYXRoLm1heChtYXhYLCB4Mik7XG4gICAgbWF4WSA9IE1hdGgubWF4KG1heFksIHkyKTtcbiAgfSk7XG4gIHJldHVybiBbbWluWCwgbWluWSwgbWF4WCwgbWF4WV07XG59O1xuZXhwb3J0IGNvbnN0IGdldERyYWdnZWRFbGVtZW50c0JvdW5kcyA9IChlbGVtZW50cywgZHJhZ09mZnNldCkgPT4ge1xuICBjb25zdCBbbWluWCwgbWluWSwgbWF4WCwgbWF4WV0gPSBnZXRDb21tb25Cb3VuZHMoZWxlbWVudHMpO1xuICByZXR1cm4gW21pblggKyBkcmFnT2Zmc2V0LngsIG1pblkgKyBkcmFnT2Zmc2V0LnksIG1heFggKyBkcmFnT2Zmc2V0LngsIG1heFkgKyBkcmFnT2Zmc2V0LnldO1xufTtcbmV4cG9ydCBjb25zdCBnZXRSZXNpemVkRWxlbWVudEFic29sdXRlQ29vcmRzID0gKGVsZW1lbnQsIG5leHRXaWR0aCwgbmV4dEhlaWdodCwgbm9ybWFsaXplUG9pbnRzKSA9PiB7XG4gIGlmICghKGlzTGluZWFyRWxlbWVudChlbGVtZW50KSB8fCBpc0ZyZWVEcmF3RWxlbWVudChlbGVtZW50KSkpIHtcbiAgICByZXR1cm4gW2VsZW1lbnQueCwgZWxlbWVudC55LCBlbGVtZW50LnggKyBuZXh0V2lkdGgsIGVsZW1lbnQueSArIG5leHRIZWlnaHRdO1xuICB9XG5cbiAgY29uc3QgcG9pbnRzID0gcmVzY2FsZVBvaW50cygwLCBuZXh0V2lkdGgsIHJlc2NhbGVQb2ludHMoMSwgbmV4dEhlaWdodCwgZWxlbWVudC5wb2ludHMsIG5vcm1hbGl6ZVBvaW50cyksIG5vcm1hbGl6ZVBvaW50cyk7XG4gIGxldCBib3VuZHM7XG5cbiAgaWYgKGlzRnJlZURyYXdFbGVtZW50KGVsZW1lbnQpKSB7XG4gICAgLy8gRnJlZSBEcmF3XG4gICAgYm91bmRzID0gZ2V0Qm91bmRzRnJvbVBvaW50cyhwb2ludHMpO1xuICB9IGVsc2Uge1xuICAgIC8vIExpbmVcbiAgICBjb25zdCBnZW4gPSByb3VnaC5nZW5lcmF0b3IoKTtcbiAgICBjb25zdCBjdXJ2ZSA9ICFlbGVtZW50LnJvdW5kbmVzcyA/IGdlbi5saW5lYXJQYXRoKHBvaW50cywgZ2VuZXJhdGVSb3VnaE9wdGlvbnMoZWxlbWVudCkpIDogZ2VuLmN1cnZlKHBvaW50cywgZ2VuZXJhdGVSb3VnaE9wdGlvbnMoZWxlbWVudCkpO1xuICAgIGNvbnN0IG9wcyA9IGdldEN1cnZlUGF0aE9wcyhjdXJ2ZSk7XG4gICAgYm91bmRzID0gZ2V0TWluTWF4WFlGcm9tQ3VydmVQYXRoT3BzKG9wcyk7XG4gIH1cblxuICBjb25zdCBbbWluWCwgbWluWSwgbWF4WCwgbWF4WV0gPSBib3VuZHM7XG4gIHJldHVybiBbbWluWCArIGVsZW1lbnQueCwgbWluWSArIGVsZW1lbnQueSwgbWF4WCArIGVsZW1lbnQueCwgbWF4WSArIGVsZW1lbnQueV07XG59O1xuZXhwb3J0IGNvbnN0IGdldEVsZW1lbnRQb2ludHNDb29yZHMgPSAoZWxlbWVudCwgcG9pbnRzKSA9PiB7XG4gIC8vIFRoaXMgbWlnaHQgYmUgY29tcHV0YXRpb25hbGx5IGhlYXZleVxuICBjb25zdCBnZW4gPSByb3VnaC5nZW5lcmF0b3IoKTtcbiAgY29uc3QgY3VydmUgPSBlbGVtZW50LnJvdW5kbmVzcyA9PSBudWxsID8gZ2VuLmxpbmVhclBhdGgocG9pbnRzLCBnZW5lcmF0ZVJvdWdoT3B0aW9ucyhlbGVtZW50KSkgOiBnZW4uY3VydmUocG9pbnRzLCBnZW5lcmF0ZVJvdWdoT3B0aW9ucyhlbGVtZW50KSk7XG4gIGNvbnN0IG9wcyA9IGdldEN1cnZlUGF0aE9wcyhjdXJ2ZSk7XG4gIGNvbnN0IFttaW5YLCBtaW5ZLCBtYXhYLCBtYXhZXSA9IGdldE1pbk1heFhZRnJvbUN1cnZlUGF0aE9wcyhvcHMpO1xuICByZXR1cm4gW21pblggKyBlbGVtZW50LngsIG1pblkgKyBlbGVtZW50LnksIG1heFggKyBlbGVtZW50LngsIG1heFkgKyBlbGVtZW50LnldO1xufTtcbmV4cG9ydCBjb25zdCBnZXRDbG9zZXN0RWxlbWVudEJvdW5kcyA9IChlbGVtZW50cywgZnJvbSkgPT4ge1xuICBpZiAoIWVsZW1lbnRzLmxlbmd0aCkge1xuICAgIHJldHVybiBbMCwgMCwgMCwgMF07XG4gIH1cblxuICBsZXQgbWluRGlzdGFuY2UgPSBJbmZpbml0eTtcbiAgbGV0IGNsb3Nlc3RFbGVtZW50ID0gZWxlbWVudHNbMF07XG4gIGVsZW1lbnRzLmZvckVhY2goZWxlbWVudCA9PiB7XG4gICAgY29uc3QgW3gxLCB5MSwgeDIsIHkyXSA9IGdldEVsZW1lbnRCb3VuZHMoZWxlbWVudCk7XG4gICAgY29uc3QgZGlzdGFuY2UgPSBkaXN0YW5jZTJkKCh4MSArIHgyKSAvIDIsICh5MSArIHkyKSAvIDIsIGZyb20ueCwgZnJvbS55KTtcblxuICAgIGlmIChkaXN0YW5jZSA8IG1pbkRpc3RhbmNlKSB7XG4gICAgICBtaW5EaXN0YW5jZSA9IGRpc3RhbmNlO1xuICAgICAgY2xvc2VzdEVsZW1lbnQgPSBlbGVtZW50O1xuICAgIH1cbiAgfSk7XG4gIHJldHVybiBnZXRFbGVtZW50Qm91bmRzKGNsb3Nlc3RFbGVtZW50KTtcbn07XG5leHBvcnQgY29uc3QgZ2V0Q29tbW9uQm91bmRpbmdCb3ggPSBlbGVtZW50cyA9PiB7XG4gIGNvbnN0IFttaW5YLCBtaW5ZLCBtYXhYLCBtYXhZXSA9IGdldENvbW1vbkJvdW5kcyhlbGVtZW50cyk7XG4gIHJldHVybiB7XG4gICAgbWluWCxcbiAgICBtaW5ZLFxuICAgIG1heFgsXG4gICAgbWF4WSxcbiAgICB3aWR0aDogbWF4WCAtIG1pblgsXG4gICAgaGVpZ2h0OiBtYXhZIC0gbWluWSxcbiAgICBtaWRYOiAobWluWCArIG1heFgpIC8gMixcbiAgICBtaWRZOiAobWluWSArIG1heFkpIC8gMlxuICB9O1xufTsiXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///../../element/bounds.ts\n");
3875
3875
 
3876
3876
  /***/ }),
3877
3877
 
@@ -4344,7 +4344,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
4344
4344
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
4345
4345
 
4346
4346
  "use strict";
4347
- eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _constants__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../constants */ \"../../constants.ts\");\n\n\nif (\"development\" !== _constants__WEBPACK_IMPORTED_MODULE_0__.ENV.TEST) {\n /* eslint-disable */\n\n /* global __webpack_public_path__:writable */\n __webpack_require__.p = window.EXCALIDRAW_ASSET_PATH || `https://unpkg.com/${\"@excalidraw/excalidraw\"}@${\"0.17.0\"}/dist/`;\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9wdWJsaWNQYXRoLmpzLmpzIiwibWFwcGluZ3MiOiI7O0FBQXNDOztBQUV0QyxJQUFJLGFBQW9CLEtBQUssZ0RBQVE7QUFDckM7O0FBRUE7QUFDQSxFQUFFLHFCQUF1Qix3REFBd0Qsd0JBQXlCLENBQUMsR0FBRyxRQUE0QixDQUFDO0FBQzNJIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vLy4vcHVibGljUGF0aC5qcz82N2Y5Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEVOViB9IGZyb20gXCIuLi8uLi9jb25zdGFudHNcIjtcblxuaWYgKHByb2Nlc3MuZW52Lk5PREVfRU5WICE9PSBFTlYuVEVTVCkge1xuICAvKiBlc2xpbnQtZGlzYWJsZSAqL1xuXG4gIC8qIGdsb2JhbCBfX3dlYnBhY2tfcHVibGljX3BhdGhfXzp3cml0YWJsZSAqL1xuICBfX3dlYnBhY2tfcHVibGljX3BhdGhfXyA9IHdpbmRvdy5FWENBTElEUkFXX0FTU0VUX1BBVEggfHwgYGh0dHBzOi8vdW5wa2cuY29tLyR7cHJvY2Vzcy5lbnYuVklURV9QS0dfTkFNRX1AJHtwcm9jZXNzLmVudi5WSVRFX1BLR19WRVJTSU9OfS9kaXN0L2A7XG59Il0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./publicPath.js\n");
4347
+ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _constants__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../constants */ \"../../constants.ts\");\n\n\nif (\"development\" !== _constants__WEBPACK_IMPORTED_MODULE_0__.ENV.TEST) {\n /* eslint-disable */\n\n /* global __webpack_public_path__:writable */\n __webpack_require__.p = window.EXCALIDRAW_ASSET_PATH || `https://unpkg.com/${\"@excalidraw/excalidraw\"}@${\"0.17.1\"}/dist/`;\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9wdWJsaWNQYXRoLmpzLmpzIiwibWFwcGluZ3MiOiI7O0FBQXNDOztBQUV0QyxJQUFJLGFBQW9CLEtBQUssZ0RBQVE7QUFDckM7O0FBRUE7QUFDQSxFQUFFLHFCQUF1Qix3REFBd0Qsd0JBQXlCLENBQUMsR0FBRyxRQUE0QixDQUFDO0FBQzNJIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vLy4vcHVibGljUGF0aC5qcz82N2Y5Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEVOViB9IGZyb20gXCIuLi8uLi9jb25zdGFudHNcIjtcblxuaWYgKHByb2Nlc3MuZW52Lk5PREVfRU5WICE9PSBFTlYuVEVTVCkge1xuICAvKiBlc2xpbnQtZGlzYWJsZSAqL1xuXG4gIC8qIGdsb2JhbCBfX3dlYnBhY2tfcHVibGljX3BhdGhfXzp3cml0YWJsZSAqL1xuICBfX3dlYnBhY2tfcHVibGljX3BhdGhfXyA9IHdpbmRvdy5FWENBTElEUkFXX0FTU0VUX1BBVEggfHwgYGh0dHBzOi8vdW5wa2cuY29tLyR7cHJvY2Vzcy5lbnYuVklURV9QS0dfTkFNRX1AJHtwcm9jZXNzLmVudi5WSVRFX1BLR19WRVJTSU9OfS9kaXN0L2A7XG59Il0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./publicPath.js\n");
4348
4348
 
4349
4349
  /***/ }),
4350
4350
 
@@ -4520,7 +4520,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
4520
4520
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
4521
4521
 
4522
4522
  "use strict";
4523
- eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"exportToCanvas\": () => (/* binding */ exportToCanvas),\n/* harmony export */ \"exportToSvg\": () => (/* binding */ exportToSvg),\n/* harmony export */ \"getExportSize\": () => (/* binding */ getExportSize)\n/* harmony export */ });\n/* harmony import */ var roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! roughjs/bin/rough */ \"../../../node_modules/roughjs/bin/rough.js\");\n/* harmony import */ var _element_bounds__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../element/bounds */ \"../../element/bounds.ts\");\n/* harmony import */ var _renderer_renderScene__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../renderer/renderScene */ \"../../renderer/renderScene.ts\");\n/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../utils */ \"../../utils.ts\");\n/* harmony import */ var _constants__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../constants */ \"../../constants.ts\");\n/* harmony import */ var _appState__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../appState */ \"../../appState.ts\");\n/* harmony import */ var _data_json__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ../data/json */ \"../../data/json.ts\");\n/* harmony import */ var _element_image__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../element/image */ \"../../element/image.ts\");\n/* harmony import */ var _packages_withinBounds__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ../packages/withinBounds */ \"../withinBounds.ts\");\n/* harmony import */ var _frame__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! ../frame */ \"../../frame.ts\");\n/* harmony import */ var _element__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! ../element */ \"../../element/index.ts\");\n/* harmony import */ var _element_mutateElement__WEBPACK_IMPORTED_MODULE_11__ = __webpack_require__(/*! ../element/mutateElement */ \"../../element/mutateElement.ts\");\n/* harmony import */ var _Scene__WEBPACK_IMPORTED_MODULE_12__ = __webpack_require__(/*! ./Scene */ \"../../scene/Scene.ts\");\nvar __awaiter = undefined && undefined.__awaiter || function (thisArg, _arguments, P, generator) {\n function adopt(value) {\n return value instanceof P ? value : new P(function (resolve) {\n resolve(value);\n });\n }\n\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) {\n try {\n step(generator.next(value));\n } catch (e) {\n reject(e);\n }\n }\n\n function rejected(value) {\n try {\n step(generator[\"throw\"](value));\n } catch (e) {\n reject(e);\n }\n }\n\n function step(result) {\n result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected);\n }\n\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n};\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nconst SVG_EXPORT_TAG = `<!-- svg-source:excalidraw -->`; // getContainerElement and getBoundTextElement and potentially other helpers\n// depend on `Scene` which will not be available when these pure utils are\n// called outside initialized Excalidraw editor instance or even if called\n// from inside Excalidraw if the elements were never cached by Scene (e.g.\n// for library elements).\n//\n// As such, before passing the elements down, we need to initialize a custom\n// Scene instance and assign them to it.\n//\n// FIXME This is a super hacky workaround and we'll need to rewrite this soon.\n\nconst __createSceneForElementsHack__ = elements => {\n const scene = new _Scene__WEBPACK_IMPORTED_MODULE_12__[\"default\"](); // we can't duplicate elements to regenerate ids because we need the\n // orig ids when embedding. So we do another hack of not mapping element\n // ids to Scene instances so that we don't override the editor elements\n // mapping.\n // We still need to clone the objects themselves to regen references.\n\n scene.replaceAllElements((0,_utils__WEBPACK_IMPORTED_MODULE_3__.cloneJSON)(elements), false);\n return scene;\n};\n\nconst truncateText = (element, maxWidth) => {\n if (element.width <= maxWidth) {\n return element;\n }\n\n const canvas = document.createElement(\"canvas\");\n const ctx = canvas.getContext(\"2d\");\n ctx.font = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.getFontString)({\n fontFamily: element.fontFamily,\n fontSize: element.fontSize\n });\n let text = element.text;\n const metrics = ctx.measureText(text);\n\n if (metrics.width > maxWidth) {\n // we iterate from the right, removing characters one by one instead\n // of bulding the string up. This assumes that it's more likely\n // your frame names will overflow by not that many characters\n // (if ever), so it sohuld be faster this way.\n for (let i = text.length; i > 0; i--) {\n const newText = `${text.slice(0, i)}...`;\n\n if (ctx.measureText(newText).width <= maxWidth) {\n text = newText;\n break;\n }\n }\n }\n\n return (0,_element_mutateElement__WEBPACK_IMPORTED_MODULE_11__.newElementWith)(element, {\n text,\n width: maxWidth\n });\n};\n/**\n * When exporting frames, we need to render frame labels which are currently\n * being rendered in DOM when editing. Adding the labels as regular text\n * elements seems like a simple hack. In the future we'll want to move to\n * proper canvas rendering, even within editor (instead of DOM).\n */\n\n\nconst addFrameLabelsAsTextElements = (elements, opts) => {\n const nextElements = [];\n let frameIdx = 0;\n\n for (const element of elements) {\n if ((0,_element__WEBPACK_IMPORTED_MODULE_10__.isFrameElement)(element)) {\n frameIdx++;\n let textElement = (0,_element__WEBPACK_IMPORTED_MODULE_10__.newTextElement)({\n x: element.x,\n y: element.y - _constants__WEBPACK_IMPORTED_MODULE_4__.FRAME_STYLE.nameOffsetY,\n fontFamily: _constants__WEBPACK_IMPORTED_MODULE_4__.FONT_FAMILY.Assistant,\n fontSize: _constants__WEBPACK_IMPORTED_MODULE_4__.FRAME_STYLE.nameFontSize,\n lineHeight: _constants__WEBPACK_IMPORTED_MODULE_4__.FRAME_STYLE.nameLineHeight,\n strokeColor: opts.exportWithDarkMode ? _constants__WEBPACK_IMPORTED_MODULE_4__.FRAME_STYLE.nameColorDarkTheme : _constants__WEBPACK_IMPORTED_MODULE_4__.FRAME_STYLE.nameColorLightTheme,\n text: element.name || `Frame ${frameIdx}`\n });\n textElement.y -= textElement.height;\n textElement = truncateText(textElement, element.width);\n nextElements.push(textElement);\n }\n\n nextElements.push(element);\n }\n\n return nextElements;\n};\n\nconst getFrameRenderingConfig = (exportingFrame, frameRendering) => {\n frameRendering = frameRendering || (0,_appState__WEBPACK_IMPORTED_MODULE_5__.getDefaultAppState)().frameRendering;\n return {\n enabled: exportingFrame ? true : frameRendering.enabled,\n outline: exportingFrame ? false : frameRendering.outline,\n name: exportingFrame ? false : frameRendering.name,\n clip: exportingFrame ? true : frameRendering.clip\n };\n};\n\nconst prepareElementsForRender = ({\n elements,\n exportingFrame,\n frameRendering,\n exportWithDarkMode\n}) => {\n let nextElements;\n\n if (exportingFrame) {\n nextElements = (0,_packages_withinBounds__WEBPACK_IMPORTED_MODULE_8__.elementsOverlappingBBox)({\n elements,\n bounds: exportingFrame,\n type: \"overlap\"\n });\n } else if (frameRendering.enabled && frameRendering.name) {\n nextElements = addFrameLabelsAsTextElements(elements, {\n exportWithDarkMode\n });\n } else {\n nextElements = elements;\n }\n\n return nextElements;\n};\n\nconst exportToCanvas = (elements, appState, files, {\n exportBackground,\n exportPadding = _constants__WEBPACK_IMPORTED_MODULE_4__.DEFAULT_EXPORT_PADDING,\n viewBackgroundColor,\n exportingFrame\n}, createCanvas = (width, height) => {\n const canvas = document.createElement(\"canvas\");\n canvas.width = width * appState.exportScale;\n canvas.height = height * appState.exportScale;\n return {\n canvas,\n scale: appState.exportScale\n };\n}) => __awaiter(void 0, void 0, void 0, function* () {\n var _a;\n\n const tempScene = __createSceneForElementsHack__(elements);\n\n elements = tempScene.getNonDeletedElements();\n const frameRendering = getFrameRenderingConfig(exportingFrame !== null && exportingFrame !== void 0 ? exportingFrame : null, (_a = appState.frameRendering) !== null && _a !== void 0 ? _a : null);\n const elementsForRender = prepareElementsForRender({\n elements,\n exportingFrame,\n exportWithDarkMode: appState.exportWithDarkMode,\n frameRendering\n });\n\n if (exportingFrame) {\n exportPadding = 0;\n }\n\n const [minX, minY, width, height] = getCanvasSize(exportingFrame ? [exportingFrame] : (0,_frame__WEBPACK_IMPORTED_MODULE_9__.getRootElements)(elementsForRender), exportPadding);\n const {\n canvas,\n scale = 1\n } = createCanvas(width, height);\n const defaultAppState = (0,_appState__WEBPACK_IMPORTED_MODULE_5__.getDefaultAppState)();\n const {\n imageCache\n } = yield (0,_element_image__WEBPACK_IMPORTED_MODULE_7__.updateImageCache)({\n imageCache: new Map(),\n fileIds: (0,_element_image__WEBPACK_IMPORTED_MODULE_7__.getInitializedImageElements)(elementsForRender).map(element => element.fileId),\n files\n });\n (0,_renderer_renderScene__WEBPACK_IMPORTED_MODULE_2__.renderStaticScene)({\n canvas,\n rc: roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_0__[\"default\"].canvas(canvas),\n elements: elementsForRender,\n visibleElements: elementsForRender,\n scale,\n appState: Object.assign(Object.assign({}, appState), {\n frameRendering,\n viewBackgroundColor: exportBackground ? viewBackgroundColor : null,\n scrollX: -minX + exportPadding,\n scrollY: -minY + exportPadding,\n zoom: defaultAppState.zoom,\n shouldCacheIgnoreZoom: false,\n theme: appState.exportWithDarkMode ? \"dark\" : \"light\"\n }),\n renderConfig: {\n imageCache,\n renderGrid: false,\n isExporting: true\n }\n });\n tempScene.destroy();\n return canvas;\n});\nconst exportToSvg = (elements, appState, files, opts) => __awaiter(void 0, void 0, void 0, function* () {\n var _b, _c, _d;\n\n const tempScene = __createSceneForElementsHack__(elements);\n\n elements = tempScene.getNonDeletedElements();\n const frameRendering = getFrameRenderingConfig((_b = opts === null || opts === void 0 ? void 0 : opts.exportingFrame) !== null && _b !== void 0 ? _b : null, (_c = appState.frameRendering) !== null && _c !== void 0 ? _c : null);\n let {\n exportPadding = _constants__WEBPACK_IMPORTED_MODULE_4__.DEFAULT_EXPORT_PADDING,\n exportWithDarkMode = false,\n viewBackgroundColor,\n exportScale = 1,\n exportEmbedScene\n } = appState;\n const {\n exportingFrame = null\n } = opts || {};\n const elementsForRender = prepareElementsForRender({\n elements,\n exportingFrame,\n exportWithDarkMode,\n frameRendering\n });\n\n if (exportingFrame) {\n exportPadding = 0;\n }\n\n let metadata = \"\"; // we need to serialize the \"original\" elements before we put them through\n // the tempScene hack which duplicates and regenerates ids\n\n if (exportEmbedScene) {\n try {\n metadata = yield (yield Promise.resolve(/*! import() */).then(__webpack_require__.bind(__webpack_require__, /*! ../../src/data/image */ \"../../data/image.ts\"))).encodeSvgMetadata({\n // when embedding scene, we want to embed the origionally supplied\n // elements which don't contain the temp frame labels.\n // But it also requires that the exportToSvg is being supplied with\n // only the elements that we're exporting, and no extra.\n text: (0,_data_json__WEBPACK_IMPORTED_MODULE_6__.serializeAsJSON)(elements, appState, files || {}, \"local\")\n });\n } catch (error) {\n console.error(error);\n }\n }\n\n const [minX, minY, width, height] = getCanvasSize(exportingFrame ? [exportingFrame] : (0,_frame__WEBPACK_IMPORTED_MODULE_9__.getRootElements)(elementsForRender), exportPadding); // initialize SVG root\n\n const svgRoot = document.createElementNS(_constants__WEBPACK_IMPORTED_MODULE_4__.SVG_NS, \"svg\");\n svgRoot.setAttribute(\"version\", \"1.1\");\n svgRoot.setAttribute(\"xmlns\", _constants__WEBPACK_IMPORTED_MODULE_4__.SVG_NS);\n svgRoot.setAttribute(\"viewBox\", `0 0 ${width} ${height}`);\n svgRoot.setAttribute(\"width\", `${width * exportScale}`);\n svgRoot.setAttribute(\"height\", `${height * exportScale}`);\n\n if (exportWithDarkMode) {\n svgRoot.setAttribute(\"filter\", _constants__WEBPACK_IMPORTED_MODULE_4__.THEME_FILTER);\n }\n\n let assetPath = \"https://excalidraw.com/\"; // Asset path needs to be determined only when using package\n\n if (true) {\n assetPath = window.EXCALIDRAW_ASSET_PATH || `https://unpkg.com/${\"@excalidraw/excalidraw\"}@${({\"VITE_APP_BACKEND_V2_GET_URL\":\"https://json-dev.excalidraw.com/api/v2/\",\"VITE_APP_BACKEND_V2_POST_URL\":\"https://json-dev.excalidraw.com/api/v2/post/\",\"VITE_APP_LIBRARY_URL\":\"https://libraries.excalidraw.com\",\"VITE_APP_LIBRARY_BACKEND\":\"https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries\",\"VITE_APP_WS_SERVER_URL\":\"http://localhost:3002\",\"VITE_APP_PORTAL_URL\":\"\",\"VITE_APP_PLUS_LP\":\"https://plus.excalidraw.com\",\"VITE_APP_PLUS_APP\":\"https://app.excalidraw.com\",\"VITE_APP_FIREBASE_CONFIG\":\"{\\\"apiKey\\\":\\\"AIzaSyCMkxA60XIW8KbqMYL7edC4qT5l4qHX2h8\\\",\\\"authDomain\\\":\\\"excalidraw-oss-dev.firebaseapp.com\\\",\\\"projectId\\\":\\\"excalidraw-oss-dev\\\",\\\"storageBucket\\\":\\\"excalidraw-oss-dev.appspot.com\\\",\\\"messagingSenderId\\\":\\\"664559512677\\\",\\\"appId\\\":\\\"1:664559512677:web:a385181f2928d328a7aa8c\\\"}\",\"VITE_APP_DEV_ENABLE_SW\":\"\",\"VITE_APP_DEV_DISABLE_LIVE_RELOAD\":\"\",\"VITE_APP_DISABLE_TRACKING\":\"true\",\"FAST_REFRESH\":\"false\",\"VITE_APP_PORT\":\"3000\",\"VITE_APP_DEBUG_ENABLE_TEXT_CONTAINER_BOUNDING_BOX\":\"\",\"VITE_APP_COLLAPSE_OVERLAY\":\"true\",\"VITE_APP_ENABLE_ESLINT\":\"true\",\"VITE_PKG_NAME\":\"@excalidraw/excalidraw\",\"VITE_PKG_VERSION\":\"0.17.0\",\"VITE_IS_EXCALIDRAW_NPM_PACKAGE\":true}).PKG_VERSION}`;\n\n if (assetPath === null || assetPath === void 0 ? void 0 : assetPath.startsWith(\"/\")) {\n assetPath = assetPath.replace(\"/\", `${window.location.origin}/`);\n }\n\n assetPath = `${assetPath}/dist/excalidraw-assets/`;\n }\n\n const offsetX = -minX + exportPadding;\n const offsetY = -minY + exportPadding;\n const frameElements = (0,_frame__WEBPACK_IMPORTED_MODULE_9__.getFrameElements)(elements);\n let exportingFrameClipPath = \"\";\n\n for (const frame of frameElements) {\n const [x1, y1, x2, y2] = (0,_element_bounds__WEBPACK_IMPORTED_MODULE_1__.getElementAbsoluteCoords)(frame);\n const cx = (x2 - x1) / 2 - (frame.x - x1);\n const cy = (y2 - y1) / 2 - (frame.y - y1);\n exportingFrameClipPath += `<clipPath id=${frame.id}>\n <rect transform=\"translate(${frame.x + offsetX} ${frame.y + offsetY}) rotate(${frame.angle} ${cx} ${cy})\"\n width=\"${frame.width}\"\n height=\"${frame.height}\"\n >\n </rect>\n </clipPath>`;\n }\n\n svgRoot.innerHTML = `\n ${SVG_EXPORT_TAG}\n ${metadata}\n <defs>\n <style class=\"style-fonts\">\n @font-face {\n font-family: \"Virgil\";\n src: url(\"${assetPath}Virgil.woff2\");\n }\n @font-face {\n font-family: \"Cascadia\";\n src: url(\"${assetPath}Cascadia.woff2\");\n }\n @font-face {\n font-family: \"Assistant\";\n src: url(\"${assetPath}Assistant-Regular.woff2\");\n }\n </style>\n ${exportingFrameClipPath}\n </defs>\n `; // render background rect\n\n if (appState.exportBackground && viewBackgroundColor) {\n const rect = svgRoot.ownerDocument.createElementNS(_constants__WEBPACK_IMPORTED_MODULE_4__.SVG_NS, \"rect\");\n rect.setAttribute(\"x\", \"0\");\n rect.setAttribute(\"y\", \"0\");\n rect.setAttribute(\"width\", `${width}`);\n rect.setAttribute(\"height\", `${height}`);\n rect.setAttribute(\"fill\", viewBackgroundColor);\n svgRoot.appendChild(rect);\n }\n\n const rsvg = roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_0__[\"default\"].svg(svgRoot);\n (0,_renderer_renderScene__WEBPACK_IMPORTED_MODULE_2__.renderSceneToSvg)(elementsForRender, rsvg, svgRoot, files || {}, {\n offsetX,\n offsetY,\n exportWithDarkMode,\n renderEmbeddables: (_d = opts === null || opts === void 0 ? void 0 : opts.renderEmbeddables) !== null && _d !== void 0 ? _d : false,\n frameRendering\n });\n tempScene.destroy();\n return svgRoot;\n}); // calculate smallest area to fit the contents in\n\nconst getCanvasSize = (elements, exportPadding) => {\n const [minX, minY, maxX, maxY] = (0,_element_bounds__WEBPACK_IMPORTED_MODULE_1__.getCommonBounds)(elements);\n const width = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.distance)(minX, maxX) + exportPadding * 2;\n const height = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.distance)(minY, maxY) + exportPadding * 2;\n return [minX, minY, width, height];\n};\n\nconst getExportSize = (elements, exportPadding, scale) => {\n const [,, width, height] = getCanvasSize(elements, exportPadding).map(dimension => Math.trunc(dimension * scale));\n return [width, height];\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,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LGVBQWUsNkRBQVM7QUFDeEIsRUFBRSx1RUFBZ0IsOENBQThDO0FBQ2hFO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxHQUFHO0FBQ0g7QUFDQTtBQUNBLENBQUMsR0FBRzs7QUFFSjtBQUNBLG1DQUFtQyxnRUFBZTtBQUNsRCxnQkFBZ0IsZ0RBQVE7QUFDeEIsaUJBQWlCLGdEQUFRO0FBQ3pCO0FBQ0E7O0FBRU87QUFDUDtBQUNBO0FBQ0EiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi4vLi4vc2NlbmUvZXhwb3J0LnRzP2MzNzMiXSwic291cmNlc0NvbnRlbnQiOlsidmFyIF9fYXdhaXRlciA9IHRoaXMgJiYgdGhpcy5fX2F3YWl0ZXIgfHwgZnVuY3Rpb24gKHRoaXNBcmcsIF9hcmd1bWVudHMsIFAsIGdlbmVyYXRvcikge1xuICBmdW5jdGlvbiBhZG9wdCh2YWx1ZSkge1xuICAgIHJldHVybiB2YWx1ZSBpbnN0YW5jZW9mIFAgPyB2YWx1ZSA6IG5ldyBQKGZ1bmN0aW9uIChyZXNvbHZlKSB7XG4gICAgICByZXNvbHZlKHZhbHVlKTtcbiAgICB9KTtcbiAgfVxuXG4gIHJldHVybiBuZXcgKFAgfHwgKFAgPSBQcm9taXNlKSkoZnVuY3Rpb24gKHJlc29sdmUsIHJlamVjdCkge1xuICAgIGZ1bmN0aW9uIGZ1bGZpbGxlZCh2YWx1ZSkge1xuICAgICAgdHJ5IHtcbiAgICAgICAgc3RlcChnZW5lcmF0b3IubmV4dCh2YWx1ZSkpO1xuICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgICByZWplY3QoZSk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgZnVuY3Rpb24gcmVqZWN0ZWQodmFsdWUpIHtcbiAgICAgIHRyeSB7XG4gICAgICAgIHN0ZXAoZ2VuZXJhdG9yW1widGhyb3dcIl0odmFsdWUpKTtcbiAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgcmVqZWN0KGUpO1xuICAgICAgfVxuICAgIH1cblxuICAgIGZ1bmN0aW9uIHN0ZXAocmVzdWx0KSB7XG4gICAgICByZXN1bHQuZG9uZSA/IHJlc29sdmUocmVzdWx0LnZhbHVlKSA6IGFkb3B0KHJlc3VsdC52YWx1ZSkudGhlbihmdWxmaWxsZWQsIHJlamVjdGVkKTtcbiAgICB9XG5cbiAgICBzdGVwKChnZW5lcmF0b3IgPSBnZW5lcmF0b3IuYXBwbHkodGhpc0FyZywgX2FyZ3VtZW50cyB8fCBbXSkpLm5leHQoKSk7XG4gIH0pO1xufTtcblxuaW1wb3J0IHJvdWdoIGZyb20gXCJyb3VnaGpzL2Jpbi9yb3VnaFwiO1xuaW1wb3J0IHsgZ2V0Q29tbW9uQm91bmRzLCBnZXRFbGVtZW50QWJzb2x1dGVDb29yZHMgfSBmcm9tIFwiLi4vZWxlbWVudC9ib3VuZHNcIjtcbmltcG9ydCB7IHJlbmRlclNjZW5lVG9TdmcsIHJlbmRlclN0YXRpY1NjZW5lIH0gZnJvbSBcIi4uL3JlbmRlcmVyL3JlbmRlclNjZW5lXCI7XG5pbXBvcnQgeyBjbG9uZUpTT04sIGRpc3RhbmNlLCBnZXRGb250U3RyaW5nIH0gZnJvbSBcIi4uL3V0aWxzXCI7XG5pbXBvcnQgeyBERUZBVUxUX0VYUE9SVF9QQURESU5HLCBGT05UX0ZBTUlMWSwgRlJBTUVfU1RZTEUsIFNWR19OUywgVEhFTUVfRklMVEVSIH0gZnJvbSBcIi4uL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgZ2V0RGVmYXVsdEFwcFN0YXRlIH0gZnJvbSBcIi4uL2FwcFN0YXRlXCI7XG5pbXBvcnQgeyBzZXJpYWxpemVBc0pTT04gfSBmcm9tIFwiLi4vZGF0YS9qc29uXCI7XG5pbXBvcnQgeyBnZXRJbml0aWFsaXplZEltYWdlRWxlbWVudHMsIHVwZGF0ZUltYWdlQ2FjaGUgfSBmcm9tIFwiLi4vZWxlbWVudC9pbWFnZVwiO1xuaW1wb3J0IHsgZWxlbWVudHNPdmVybGFwcGluZ0JCb3ggfSBmcm9tIFwiLi4vcGFja2FnZXMvd2l0aGluQm91bmRzXCI7XG5pbXBvcnQgeyBnZXRGcmFtZUVsZW1lbnRzLCBnZXRSb290RWxlbWVudHMgfSBmcm9tIFwiLi4vZnJhbWVcIjtcbmltcG9ydCB7IGlzRnJhbWVFbGVtZW50LCBuZXdUZXh0RWxlbWVudCB9IGZyb20gXCIuLi9lbGVtZW50XCI7XG5pbXBvcnQgeyBuZXdFbGVtZW50V2l0aCB9IGZyb20gXCIuLi9lbGVtZW50L211dGF0ZUVsZW1lbnRcIjtcbmltcG9ydCBTY2VuZSBmcm9tIFwiLi9TY2VuZVwiO1xuY29uc3QgU1ZHX0VYUE9SVF9UQUcgPSBgPCEtLSBzdmctc291cmNlOmV4Y2FsaWRyYXcgLS0+YDsgLy8gZ2V0Q29udGFpbmVyRWxlbWVudCBhbmQgZ2V0Qm91bmRUZXh0RWxlbWVudCBhbmQgcG90ZW50aWFsbHkgb3RoZXIgaGVscGVyc1xuLy8gZGVwZW5kIG9uIGBTY2VuZWAgd2hpY2ggd2lsbCBub3QgYmUgYXZhaWxhYmxlIHdoZW4gdGhlc2UgcHVyZSB1dGlscyBhcmVcbi8vIGNhbGxlZCBvdXRzaWRlIGluaXRpYWxpemVkIEV4Y2FsaWRyYXcgZWRpdG9yIGluc3RhbmNlIG9yIGV2ZW4gaWYgY2FsbGVkXG4vLyBmcm9tIGluc2lkZSBFeGNhbGlkcmF3IGlmIHRoZSBlbGVtZW50cyB3ZXJlIG5ldmVyIGNhY2hlZCBieSBTY2VuZSAoZS5nLlxuLy8gZm9yIGxpYnJhcnkgZWxlbWVudHMpLlxuLy9cbi8vIEFzIHN1Y2gsIGJlZm9yZSBwYXNzaW5nIHRoZSBlbGVtZW50cyBkb3duLCB3ZSBuZWVkIHRvIGluaXRpYWxpemUgYSBjdXN0b21cbi8vIFNjZW5lIGluc3RhbmNlIGFuZCBhc3NpZ24gdGhlbSB0byBpdC5cbi8vXG4vLyBGSVhNRSBUaGlzIGlzIGEgc3VwZXIgaGFja3kgd29ya2Fyb3VuZCBhbmQgd2UnbGwgbmVlZCB0byByZXdyaXRlIHRoaXMgc29vbi5cblxuY29uc3QgX19jcmVhdGVTY2VuZUZvckVsZW1lbnRzSGFja19fID0gZWxlbWVudHMgPT4ge1xuICBjb25zdCBzY2VuZSA9IG5ldyBTY2VuZSgpOyAvLyB3ZSBjYW4ndCBkdXBsaWNhdGUgZWxlbWVudHMgdG8gcmVnZW5lcmF0ZSBpZHMgYmVjYXVzZSB3ZSBuZWVkIHRoZVxuICAvLyBvcmlnIGlkcyB3aGVuIGVtYmVkZGluZy4gU28gd2UgZG8gYW5vdGhlciBoYWNrIG9mIG5vdCBtYXBwaW5nIGVsZW1lbnRcbiAgLy8gaWRzIHRvIFNjZW5lIGluc3RhbmNlcyBzbyB0aGF0IHdlIGRvbid0IG92ZXJyaWRlIHRoZSBlZGl0b3IgZWxlbWVudHNcbiAgLy8gbWFwcGluZy5cbiAgLy8gV2Ugc3RpbGwgbmVlZCB0byBjbG9uZSB0aGUgb2JqZWN0cyB0aGVtc2VsdmVzIHRvIHJlZ2VuIHJlZmVyZW5jZXMuXG5cbiAgc2NlbmUucmVwbGFjZUFsbEVsZW1lbnRzKGNsb25lSlNPTihlbGVtZW50cyksIGZhbHNlKTtcbiAgcmV0dXJuIHNjZW5lO1xufTtcblxuY29uc3QgdHJ1bmNhdGVUZXh0ID0gKGVsZW1lbnQsIG1heFdpZHRoKSA9PiB7XG4gIGlmIChlbGVtZW50LndpZHRoIDw9IG1heFdpZHRoKSB7XG4gICAgcmV0dXJuIGVsZW1lbnQ7XG4gIH1cblxuICBjb25zdCBjYW52YXMgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiY2FudmFzXCIpO1xuICBjb25zdCBjdHggPSBjYW52YXMuZ2V0Q29udGV4dChcIjJkXCIpO1xuICBjdHguZm9udCA9IGdldEZvbnRTdHJpbmcoe1xuICAgIGZvbnRGYW1pbHk6IGVsZW1lbnQuZm9udEZhbWlseSxcbiAgICBmb250U2l6ZTogZWxlbWVudC5mb250U2l6ZVxuICB9KTtcbiAgbGV0IHRleHQgPSBlbGVtZW50LnRleHQ7XG4gIGNvbnN0IG1ldHJpY3MgPSBjdHgubWVhc3VyZVRleHQodGV4dCk7XG5cbiAgaWYgKG1ldHJpY3Mud2lkdGggPiBtYXhXaWR0aCkge1xuICAgIC8vIHdlIGl0ZXJhdGUgZnJvbSB0aGUgcmlnaHQsIHJlbW92aW5nIGNoYXJhY3RlcnMgb25lIGJ5IG9uZSBpbnN0ZWFkXG4gICAgLy8gb2YgYnVsZGluZyB0aGUgc3RyaW5nIHVwLiBUaGlzIGFzc3VtZXMgdGhhdCBpdCdzIG1vcmUgbGlrZWx5XG4gICAgLy8geW91ciBmcmFtZSBuYW1lcyB3aWxsIG92ZXJmbG93IGJ5IG5vdCB0aGF0IG1hbnkgY2hhcmFjdGVyc1xuICAgIC8vIChpZiBldmVyKSwgc28gaXQgc29odWxkIGJlIGZhc3RlciB0aGlzIHdheS5cbiAgICBmb3IgKGxldCBpID0gdGV4dC5sZW5ndGg7IGkgPiAwOyBpLS0pIHtcbiAgICAgIGNvbnN0IG5ld1RleHQgPSBgJHt0ZXh0LnNsaWNlKDAsIGkpfS4uLmA7XG5cbiAgICAgIGlmIChjdHgubWVhc3VyZVRleHQobmV3VGV4dCkud2lkdGggPD0gbWF4V2lkdGgpIHtcbiAgICAgICAgdGV4dCA9IG5ld1RleHQ7XG4gICAgICAgIGJyZWFrO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIHJldHVybiBuZXdFbGVtZW50V2l0aChlbGVtZW50LCB7XG4gICAgdGV4dCxcbiAgICB3aWR0aDogbWF4V2lkdGhcbiAgfSk7XG59O1xuLyoqXG4gKiBXaGVuIGV4cG9ydGluZyBmcmFtZXMsIHdlIG5lZWQgdG8gcmVuZGVyIGZyYW1lIGxhYmVscyB3aGljaCBhcmUgY3VycmVudGx5XG4gKiBiZWluZyByZW5kZXJlZCBpbiBET00gd2hlbiBlZGl0aW5nLiBBZGRpbmcgdGhlIGxhYmVscyBhcyByZWd1bGFyIHRleHRcbiAqIGVsZW1lbnRzIHNlZW1zIGxpa2UgYSBzaW1wbGUgaGFjay4gSW4gdGhlIGZ1dHVyZSB3ZSdsbCB3YW50IHRvIG1vdmUgdG9cbiAqIHByb3BlciBjYW52YXMgcmVuZGVyaW5nLCBldmVuIHdpdGhpbiBlZGl0b3IgKGluc3RlYWQgb2YgRE9NKS5cbiAqL1xuXG5cbmNvbnN0IGFkZEZyYW1lTGFiZWxzQXNUZXh0RWxlbWVudHMgPSAoZWxlbWVudHMsIG9wdHMpID0+IHtcbiAgY29uc3QgbmV4dEVsZW1lbnRzID0gW107XG4gIGxldCBmcmFtZUlkeCA9IDA7XG5cbiAgZm9yIChjb25zdCBlbGVtZW50IG9mIGVsZW1lbnRzKSB7XG4gICAgaWYgKGlzRnJhbWVFbGVtZW50KGVsZW1lbnQpKSB7XG4gICAgICBmcmFtZUlkeCsrO1xuICAgICAgbGV0IHRleHRFbGVtZW50ID0gbmV3VGV4dEVsZW1lbnQoe1xuICAgICAgICB4OiBlbGVtZW50LngsXG4gICAgICAgIHk6IGVsZW1lbnQueSAtIEZSQU1FX1NUWUxFLm5hbWVPZmZzZXRZLFxuICAgICAgICBmb250RmFtaWx5OiBGT05UX0ZBTUlMWS5Bc3Npc3RhbnQsXG4gICAgICAgIGZvbnRTaXplOiBGUkFNRV9TVFlMRS5uYW1lRm9udFNpemUsXG4gICAgICAgIGxpbmVIZWlnaHQ6IEZSQU1FX1NUWUxFLm5hbWVMaW5lSGVpZ2h0LFxuICAgICAgICBzdHJva2VDb2xvcjogb3B0cy5leHBvcnRXaXRoRGFya01vZGUgPyBGUkFNRV9TVFlMRS5uYW1lQ29sb3JEYXJrVGhlbWUgOiBGUkFNRV9TVFlMRS5uYW1lQ29sb3JMaWdodFRoZW1lLFxuICAgICAgICB0ZXh0OiBlbGVtZW50Lm5hbWUgfHwgYEZyYW1lICR7ZnJhbWVJZHh9YFxuICAgICAgfSk7XG4gICAgICB0ZXh0RWxlbWVudC55IC09IHRleHRFbGVtZW50LmhlaWdodDtcbiAgICAgIHRleHRFbGVtZW50ID0gdHJ1bmNhdGVUZXh0KHRleHRFbGVtZW50LCBlbGVtZW50LndpZHRoKTtcbiAgICAgIG5leHRFbGVtZW50cy5wdXNoKHRleHRFbGVtZW50KTtcbiAgICB9XG5cbiAgICBuZXh0RWxlbWVudHMucHVzaChlbGVtZW50KTtcbiAgfVxuXG4gIHJldHVybiBuZXh0RWxlbWVudHM7XG59O1xuXG5jb25zdCBnZXRGcmFtZVJlbmRlcmluZ0NvbmZpZyA9IChleHBvcnRpbmdGcmFtZSwgZnJhbWVSZW5kZXJpbmcpID0+IHtcbiAgZnJhbWVSZW5kZXJpbmcgPSBmcmFtZVJlbmRlcmluZyB8fCBnZXREZWZhdWx0QXBwU3RhdGUoKS5mcmFtZVJlbmRlcmluZztcbiAgcmV0dXJuIHtcbiAgICBlbmFibGVkOiBleHBvcnRpbmdGcmFtZSA/IHRydWUgOiBmcmFtZVJlbmRlcmluZy5lbmFibGVkLFxuICAgIG91dGxpbmU6IGV4cG9ydGluZ0ZyYW1lID8gZmFsc2UgOiBmcmFtZVJlbmRlcmluZy5vdXRsaW5lLFxuICAgIG5hbWU6IGV4cG9ydGluZ0ZyYW1lID8gZmFsc2UgOiBmcmFtZVJlbmRlcmluZy5uYW1lLFxuICAgIGNsaXA6IGV4cG9ydGluZ0ZyYW1lID8gdHJ1ZSA6IGZyYW1lUmVuZGVyaW5nLmNsaXBcbiAgfTtcbn07XG5cbmNvbnN0IHByZXBhcmVFbGVtZW50c0ZvclJlbmRlciA9ICh7XG4gIGVsZW1lbnRzLFxuICBleHBvcnRpbmdGcmFtZSxcbiAgZnJhbWVSZW5kZXJpbmcsXG4gIGV4cG9ydFdpdGhEYXJrTW9kZVxufSkgPT4ge1xuICBsZXQgbmV4dEVsZW1lbnRzO1xuXG4gIGlmIChleHBvcnRpbmdGcmFtZSkge1xuICAgIG5leHRFbGVtZW50cyA9IGVsZW1lbnRzT3ZlcmxhcHBpbmdCQm94KHtcbiAgICAgIGVsZW1lbnRzLFxuICAgICAgYm91bmRzOiBleHBvcnRpbmdGcmFtZSxcbiAgICAgIHR5cGU6IFwib3ZlcmxhcFwiXG4gICAgfSk7XG4gIH0gZWxzZSBpZiAoZnJhbWVSZW5kZXJpbmcuZW5hYmxlZCAmJiBmcmFtZVJlbmRlcmluZy5uYW1lKSB7XG4gICAgbmV4dEVsZW1lbnRzID0gYWRkRnJhbWVMYWJlbHNBc1RleHRFbGVtZW50cyhlbGVtZW50cywge1xuICAgICAgZXhwb3J0V2l0aERhcmtNb2RlXG4gICAgfSk7XG4gIH0gZWxzZSB7XG4gICAgbmV4dEVsZW1lbnRzID0gZWxlbWVudHM7XG4gIH1cblxuICByZXR1cm4gbmV4dEVsZW1lbnRzO1xufTtcblxuZXhwb3J0IGNvbnN0IGV4cG9ydFRvQ2FudmFzID0gKGVsZW1lbnRzLCBhcHBTdGF0ZSwgZmlsZXMsIHtcbiAgZXhwb3J0QmFja2dyb3VuZCxcbiAgZXhwb3J0UGFkZGluZyA9IERFRkFVTFRfRVhQT1JUX1BBRERJTkcsXG4gIHZpZXdCYWNrZ3JvdW5kQ29sb3IsXG4gIGV4cG9ydGluZ0ZyYW1lXG59LCBjcmVhdGVDYW52YXMgPSAod2lkdGgsIGhlaWdodCkgPT4ge1xuICBjb25zdCBjYW52YXMgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiY2FudmFzXCIpO1xuICBjYW52YXMud2lkdGggPSB3aWR0aCAqIGFwcFN0YXRlLmV4cG9ydFNjYWxlO1xuICBjYW52YXMuaGVpZ2h0ID0gaGVpZ2h0ICogYXBwU3RhdGUuZXhwb3J0U2NhbGU7XG4gIHJldHVybiB7XG4gICAgY2FudmFzLFxuICAgIHNjYWxlOiBhcHBTdGF0ZS5leHBvcnRTY2FsZVxuICB9O1xufSkgPT4gX19hd2FpdGVyKHZvaWQgMCwgdm9pZCAwLCB2b2lkIDAsIGZ1bmN0aW9uKiAoKSB7XG4gIHZhciBfYTtcblxuICBjb25zdCB0ZW1wU2NlbmUgPSBfX2NyZWF0ZVNjZW5lRm9yRWxlbWVudHNIYWNrX18oZWxlbWVudHMpO1xuXG4gIGVsZW1lbnRzID0gdGVtcFNjZW5lLmdldE5vbkRlbGV0ZWRFbGVtZW50cygpO1xuICBjb25zdCBmcmFtZVJlbmRlcmluZyA9IGdldEZyYW1lUmVuZGVyaW5nQ29uZmlnKGV4cG9ydGluZ0ZyYW1lICE9PSBudWxsICYmIGV4cG9ydGluZ0ZyYW1lICE9PSB2b2lkIDAgPyBleHBvcnRpbmdGcmFtZSA6IG51bGwsIChfYSA9IGFwcFN0YXRlLmZyYW1lUmVuZGVyaW5nKSAhPT0gbnVsbCAmJiBfYSAhPT0gdm9pZCAwID8gX2EgOiBudWxsKTtcbiAgY29uc3QgZWxlbWVudHNGb3JSZW5kZXIgPSBwcmVwYXJlRWxlbWVudHNGb3JSZW5kZXIoe1xuICAgIGVsZW1lbnRzLFxuICAgIGV4cG9ydGluZ0ZyYW1lLFxuICAgIGV4cG9ydFdpdGhEYXJrTW9kZTogYXBwU3RhdGUuZXhwb3J0V2l0aERhcmtNb2RlLFxuICAgIGZyYW1lUmVuZGVyaW5nXG4gIH0pO1xuXG4gIGlmIChleHBvcnRpbmdGcmFtZSkge1xuICAgIGV4cG9ydFBhZGRpbmcgPSAwO1xuICB9XG5cbiAgY29uc3QgW21pblgsIG1pblksIHdpZHRoLCBoZWlnaHRdID0gZ2V0Q2FudmFzU2l6ZShleHBvcnRpbmdGcmFtZSA/IFtleHBvcnRpbmdGcmFtZV0gOiBnZXRSb290RWxlbWVudHMoZWxlbWVudHNGb3JSZW5kZXIpLCBleHBvcnRQYWRkaW5nKTtcbiAgY29uc3Qge1xuICAgIGNhbnZhcyxcbiAgICBzY2FsZSA9IDFcbiAgfSA9IGNyZWF0ZUNhbnZhcyh3aWR0aCwgaGVpZ2h0KTtcbiAgY29uc3QgZGVmYXVsdEFwcFN0YXRlID0gZ2V0RGVmYXVsdEFwcFN0YXRlKCk7XG4gIGNvbnN0IHtcbiAgICBpbWFnZUNhY2hlXG4gIH0gPSB5aWVsZCB1cGRhdGVJbWFnZUNhY2hlKHtcbiAgICBpbWFnZUNhY2hlOiBuZXcgTWFwKCksXG4gICAgZmlsZUlkczogZ2V0SW5pdGlhbGl6ZWRJbWFnZUVsZW1lbnRzKGVsZW1lbnRzRm9yUmVuZGVyKS5tYXAoZWxlbWVudCA9PiBlbGVtZW50LmZpbGVJZCksXG4gICAgZmlsZXNcbiAgfSk7XG4gIHJlbmRlclN0YXRpY1NjZW5lKHtcbiAgICBjYW52YXMsXG4gICAgcmM6IHJvdWdoLmNhbnZhcyhjYW52YXMpLFxuICAgIGVsZW1lbnRzOiBlbGVtZW50c0ZvclJlbmRlcixcbiAgICB2aXNpYmxlRWxlbWVudHM6IGVsZW1lbnRzRm9yUmVuZGVyLFxuICAgIHNjYWxlLFxuICAgIGFwcFN0YXRlOiBPYmplY3QuYXNzaWduKE9iamVjdC5hc3NpZ24oe30sIGFwcFN0YXRlKSwge1xuICAgICAgZnJhbWVSZW5kZXJpbmcsXG4gICAgICB2aWV3QmFja2dyb3VuZENvbG9yOiBleHBvcnRCYWNrZ3JvdW5kID8gdmlld0JhY2tncm91bmRDb2xvciA6IG51bGwsXG4gICAgICBzY3JvbGxYOiAtbWluWCArIGV4cG9ydFBhZGRpbmcsXG4gICAgICBzY3JvbGxZOiAtbWluWSArIGV4cG9ydFBhZGRpbmcsXG4gICAgICB6b29tOiBkZWZhdWx0QXBwU3RhdGUuem9vbSxcbiAgICAgIHNob3VsZENhY2hlSWdub3JlWm9vbTogZmFsc2UsXG4gICAgICB0aGVtZTogYXBwU3RhdGUuZXhwb3J0V2l0aERhcmtNb2RlID8gXCJkYXJrXCIgOiBcImxpZ2h0XCJcbiAgICB9KSxcbiAgICByZW5kZXJDb25maWc6IHtcbiAgICAgIGltYWdlQ2FjaGUsXG4gICAgICByZW5kZXJHcmlkOiBmYWxzZSxcbiAgICAgIGlzRXhwb3J0aW5nOiB0cnVlXG4gICAgfVxuICB9KTtcbiAgdGVtcFNjZW5lLmRlc3Ryb3koKTtcbiAgcmV0dXJuIGNhbnZhcztcbn0pO1xuZXhwb3J0IGNvbnN0IGV4cG9ydFRvU3ZnID0gKGVsZW1lbnRzLCBhcHBTdGF0ZSwgZmlsZXMsIG9wdHMpID0+IF9fYXdhaXRlcih2b2lkIDAsIHZvaWQgMCwgdm9pZCAwLCBmdW5jdGlvbiogKCkge1xuICB2YXIgX2IsIF9jLCBfZDtcblxuICBjb25zdCB0ZW1wU2NlbmUgPSBfX2NyZWF0ZVNjZW5lRm9yRWxlbWVudHNIYWNrX18oZWxlbWVudHMpO1xuXG4gIGVsZW1lbnRzID0gdGVtcFNjZW5lLmdldE5vbkRlbGV0ZWRFbGVtZW50cygpO1xuICBjb25zdCBmcmFtZVJlbmRlcmluZyA9IGdldEZyYW1lUmVuZGVyaW5nQ29uZmlnKChfYiA9IG9wdHMgPT09IG51bGwgfHwgb3B0cyA9PT0gdm9pZCAwID8gdm9pZCAwIDogb3B0cy5leHBvcnRpbmdGcmFtZSkgIT09IG51bGwgJiYgX2IgIT09IHZvaWQgMCA/IF9iIDogbnVsbCwgKF9jID0gYXBwU3RhdGUuZnJhbWVSZW5kZXJpbmcpICE9PSBudWxsICYmIF9jICE9PSB2b2lkIDAgPyBfYyA6IG51bGwpO1xuICBsZXQge1xuICAgIGV4cG9ydFBhZGRpbmcgPSBERUZBVUxUX0VYUE9SVF9QQURESU5HLFxuICAgIGV4cG9ydFdpdGhEYXJrTW9kZSA9IGZhbHNlLFxuICAgIHZpZXdCYWNrZ3JvdW5kQ29sb3IsXG4gICAgZXhwb3J0U2NhbGUgPSAxLFxuICAgIGV4cG9ydEVtYmVkU2NlbmVcbiAgfSA9IGFwcFN0YXRlO1xuICBjb25zdCB7XG4gICAgZXhwb3J0aW5nRnJhbWUgPSBudWxsXG4gIH0gPSBvcHRzIHx8IHt9O1xuICBjb25zdCBlbGVtZW50c0ZvclJlbmRlciA9IHByZXBhcmVFbGVtZW50c0ZvclJlbmRlcih7XG4gICAgZWxlbWVudHMsXG4gICAgZXhwb3J0aW5nRnJhbWUsXG4gICAgZXhwb3J0V2l0aERhcmtNb2RlLFxuICAgIGZyYW1lUmVuZGVyaW5nXG4gIH0pO1xuXG4gIGlmIChleHBvcnRpbmdGcmFtZSkge1xuICAgIGV4cG9ydFBhZGRpbmcgPSAwO1xuICB9XG5cbiAgbGV0IG1ldGFkYXRhID0gXCJcIjsgLy8gd2UgbmVlZCB0byBzZXJpYWxpemUgdGhlIFwib3JpZ2luYWxcIiBlbGVtZW50cyBiZWZvcmUgd2UgcHV0IHRoZW0gdGhyb3VnaFxuICAvLyB0aGUgdGVtcFNjZW5lIGhhY2sgd2hpY2ggZHVwbGljYXRlcyBhbmQgcmVnZW5lcmF0ZXMgaWRzXG5cbiAgaWYgKGV4cG9ydEVtYmVkU2NlbmUpIHtcbiAgICB0cnkge1xuICAgICAgbWV0YWRhdGEgPSB5aWVsZCAoeWllbGQgaW1wb3J0KFxuICAgICAgLyogd2VicGFja0NodW5rTmFtZTogXCJpbWFnZVwiICovXG4gICAgICBcIi4uLy4uL3NyYy9kYXRhL2ltYWdlXCIpKS5lbmNvZGVTdmdNZXRhZGF0YSh7XG4gICAgICAgIC8vIHdoZW4gZW1iZWRkaW5nIHNjZW5lLCB3ZSB3YW50IHRvIGVtYmVkIHRoZSBvcmlnaW9uYWxseSBzdXBwbGllZFxuICAgICAgICAvLyBlbGVtZW50cyB3aGljaCBkb24ndCBjb250YWluIHRoZSB0ZW1wIGZyYW1lIGxhYmVscy5cbiAgICAgICAgLy8gQnV0IGl0IGFsc28gcmVxdWlyZXMgdGhhdCB0aGUgZXhwb3J0VG9TdmcgaXMgYmVpbmcgc3VwcGxpZWQgd2l0aFxuICAgICAgICAvLyBvbmx5IHRoZSBlbGVtZW50cyB0aGF0IHdlJ3JlIGV4cG9ydGluZywgYW5kIG5vIGV4dHJhLlxuICAgICAgICB0ZXh0OiBzZXJpYWxpemVBc0pTT04oZWxlbWVudHMsIGFwcFN0YXRlLCBmaWxlcyB8fCB7fSwgXCJsb2NhbFwiKVxuICAgICAgfSk7XG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIGNvbnNvbGUuZXJyb3IoZXJyb3IpO1xuICAgIH1cbiAgfVxuXG4gIGNvbnN0IFttaW5YLCBtaW5ZLCB3aWR0aCwgaGVpZ2h0XSA9IGdldENhbnZhc1NpemUoZXhwb3J0aW5nRnJhbWUgPyBbZXhwb3J0aW5nRnJhbWVdIDogZ2V0Um9vdEVsZW1lbnRzKGVsZW1lbnRzRm9yUmVuZGVyKSwgZXhwb3J0UGFkZGluZyk7IC8vIGluaXRpYWxpemUgU1ZHIHJvb3RcblxuICBjb25zdCBzdmdSb290ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKFNWR19OUywgXCJzdmdcIik7XG4gIHN2Z1Jvb3Quc2V0QXR0cmlidXRlKFwidmVyc2lvblwiLCBcIjEuMVwiKTtcbiAgc3ZnUm9vdC5zZXRBdHRyaWJ1dGUoXCJ4bWxuc1wiLCBTVkdfTlMpO1xuICBzdmdSb290LnNldEF0dHJpYnV0ZShcInZpZXdCb3hcIiwgYDAgMCAke3dpZHRofSAke2hlaWdodH1gKTtcbiAgc3ZnUm9vdC5zZXRBdHRyaWJ1dGUoXCJ3aWR0aFwiLCBgJHt3aWR0aCAqIGV4cG9ydFNjYWxlfWApO1xuICBzdmdSb290LnNldEF0dHJpYnV0ZShcImhlaWdodFwiLCBgJHtoZWlnaHQgKiBleHBvcnRTY2FsZX1gKTtcblxuICBpZiAoZXhwb3J0V2l0aERhcmtNb2RlKSB7XG4gICAgc3ZnUm9vdC5zZXRBdHRyaWJ1dGUoXCJmaWx0ZXJcIiwgVEhFTUVfRklMVEVSKTtcbiAgfVxuXG4gIGxldCBhc3NldFBhdGggPSBcImh0dHBzOi8vZXhjYWxpZHJhdy5jb20vXCI7IC8vIEFzc2V0IHBhdGggbmVlZHMgdG8gYmUgZGV0ZXJtaW5lZCBvbmx5IHdoZW4gdXNpbmcgcGFja2FnZVxuXG4gIGlmIChwcm9jZXNzLmVudi5WSVRFX0lTX0VYQ0FMSURSQVdfTlBNX1BBQ0tBR0UpIHtcbiAgICBhc3NldFBhdGggPSB3aW5kb3cuRVhDQUxJRFJBV19BU1NFVF9QQVRIIHx8IGBodHRwczovL3VucGtnLmNvbS8ke3Byb2Nlc3MuZW52LlZJVEVfUEtHX05BTUV9QCR7cHJvY2Vzcy5lbnYuUEtHX1ZFUlNJT059YDtcblxuICAgIGlmIChhc3NldFBhdGggPT09IG51bGwgfHwgYXNzZXRQYXRoID09PSB2b2lkIDAgPyB2b2lkIDAgOiBhc3NldFBhdGguc3RhcnRzV2l0aChcIi9cIikpIHtcbiAgICAgIGFzc2V0UGF0aCA9IGFzc2V0UGF0aC5yZXBsYWNlKFwiL1wiLCBgJHt3aW5kb3cubG9jYXRpb24ub3JpZ2lufS9gKTtcbiAgICB9XG5cbiAgICBhc3NldFBhdGggPSBgJHthc3NldFBhdGh9L2Rpc3QvZXhjYWxpZHJhdy1hc3NldHMvYDtcbiAgfVxuXG4gIGNvbnN0IG9mZnNldFggPSAtbWluWCArIGV4cG9ydFBhZGRpbmc7XG4gIGNvbnN0IG9mZnNldFkgPSAtbWluWSArIGV4cG9ydFBhZGRpbmc7XG4gIGNvbnN0IGZyYW1lRWxlbWVudHMgPSBnZXRGcmFtZUVsZW1lbnRzKGVsZW1lbnRzKTtcbiAgbGV0IGV4cG9ydGluZ0ZyYW1lQ2xpcFBhdGggPSBcIlwiO1xuXG4gIGZvciAoY29uc3QgZnJhbWUgb2YgZnJhbWVFbGVtZW50cykge1xuICAgIGNvbnN0IFt4MSwgeTEsIHgyLCB5Ml0gPSBnZXRFbGVtZW50QWJzb2x1dGVDb29yZHMoZnJhbWUpO1xuICAgIGNvbnN0IGN4ID0gKHgyIC0geDEpIC8gMiAtIChmcmFtZS54IC0geDEpO1xuICAgIGNvbnN0IGN5ID0gKHkyIC0geTEpIC8gMiAtIChmcmFtZS55IC0geTEpO1xuICAgIGV4cG9ydGluZ0ZyYW1lQ2xpcFBhdGggKz0gYDxjbGlwUGF0aCBpZD0ke2ZyYW1lLmlkfT5cbiAgICAgICAgICAgIDxyZWN0IHRyYW5zZm9ybT1cInRyYW5zbGF0ZSgke2ZyYW1lLnggKyBvZmZzZXRYfSAke2ZyYW1lLnkgKyBvZmZzZXRZfSkgcm90YXRlKCR7ZnJhbWUuYW5nbGV9ICR7Y3h9ICR7Y3l9KVwiXG4gICAgICAgICAgd2lkdGg9XCIke2ZyYW1lLndpZHRofVwiXG4gICAgICAgICAgaGVpZ2h0PVwiJHtmcmFtZS5oZWlnaHR9XCJcbiAgICAgICAgICA+XG4gICAgICAgICAgPC9yZWN0PlxuICAgICAgICA8L2NsaXBQYXRoPmA7XG4gIH1cblxuICBzdmdSb290LmlubmVySFRNTCA9IGBcbiAgJHtTVkdfRVhQT1JUX1RBR31cbiAgJHttZXRhZGF0YX1cbiAgPGRlZnM+XG4gICAgPHN0eWxlIGNsYXNzPVwic3R5bGUtZm9udHNcIj5cbiAgICAgIEBmb250LWZhY2Uge1xuICAgICAgICBmb250LWZhbWlseTogXCJWaXJnaWxcIjtcbiAgICAgICAgc3JjOiB1cmwoXCIke2Fzc2V0UGF0aH1WaXJnaWwud29mZjJcIik7XG4gICAgICB9XG4gICAgICBAZm9udC1mYWNlIHtcbiAgICAgICAgZm9udC1mYW1pbHk6IFwiQ2FzY2FkaWFcIjtcbiAgICAgICAgc3JjOiB1cmwoXCIke2Fzc2V0UGF0aH1DYXNjYWRpYS53b2ZmMlwiKTtcbiAgICAgIH1cbiAgICAgIEBmb250LWZhY2Uge1xuICAgICAgICBmb250LWZhbWlseTogXCJBc3Npc3RhbnRcIjtcbiAgICAgICAgc3JjOiB1cmwoXCIke2Fzc2V0UGF0aH1Bc3Npc3RhbnQtUmVndWxhci53b2ZmMlwiKTtcbiAgICAgIH1cbiAgICA8L3N0eWxlPlxuICAgICR7ZXhwb3J0aW5nRnJhbWVDbGlwUGF0aH1cbiAgPC9kZWZzPlxuICBgOyAvLyByZW5kZXIgYmFja2dyb3VuZCByZWN0XG5cbiAgaWYgKGFwcFN0YXRlLmV4cG9ydEJhY2tncm91bmQgJiYgdmlld0JhY2tncm91bmRDb2xvcikge1xuICAgIGNvbnN0IHJlY3QgPSBzdmdSb290Lm93bmVyRG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKFNWR19OUywgXCJyZWN0XCIpO1xuICAgIHJlY3Quc2V0QXR0cmlidXRlKFwieFwiLCBcIjBcIik7XG4gICAgcmVjdC5zZXRBdHRyaWJ1dGUoXCJ5XCIsIFwiMFwiKTtcbiAgICByZWN0LnNldEF0dHJpYnV0ZShcIndpZHRoXCIsIGAke3dpZHRofWApO1xuICAgIHJlY3Quc2V0QXR0cmlidXRlKFwiaGVpZ2h0XCIsIGAke2hlaWdodH1gKTtcbiAgICByZWN0LnNldEF0dHJpYnV0ZShcImZpbGxcIiwgdmlld0JhY2tncm91bmRDb2xvcik7XG4gICAgc3ZnUm9vdC5hcHBlbmRDaGlsZChyZWN0KTtcbiAgfVxuXG4gIGNvbnN0IHJzdmcgPSByb3VnaC5zdmcoc3ZnUm9vdCk7XG4gIHJlbmRlclNjZW5lVG9TdmcoZWxlbWVudHNGb3JSZW5kZXIsIHJzdmcsIHN2Z1Jvb3QsIGZpbGVzIHx8IHt9LCB7XG4gICAgb2Zmc2V0WCxcbiAgICBvZmZzZXRZLFxuICAgIGV4cG9ydFdpdGhEYXJrTW9kZSxcbiAgICByZW5kZXJFbWJlZGRhYmxlczogKF9kID0gb3B0cyA9PT0gbnVsbCB8fCBvcHRzID09PSB2b2lkIDAgPyB2b2lkIDAgOiBvcHRzLnJlbmRlckVtYmVkZGFibGVzKSAhPT0gbnVsbCAmJiBfZCAhPT0gdm9pZCAwID8gX2QgOiBmYWxzZSxcbiAgICBmcmFtZVJlbmRlcmluZ1xuICB9KTtcbiAgdGVtcFNjZW5lLmRlc3Ryb3koKTtcbiAgcmV0dXJuIHN2Z1Jvb3Q7XG59KTsgLy8gY2FsY3VsYXRlIHNtYWxsZXN0IGFyZWEgdG8gZml0IHRoZSBjb250ZW50cyBpblxuXG5jb25zdCBnZXRDYW52YXNTaXplID0gKGVsZW1lbnRzLCBleHBvcnRQYWRkaW5nKSA9PiB7XG4gIGNvbnN0IFttaW5YLCBtaW5ZLCBtYXhYLCBtYXhZXSA9IGdldENvbW1vbkJvdW5kcyhlbGVtZW50cyk7XG4gIGNvbnN0IHdpZHRoID0gZGlzdGFuY2UobWluWCwgbWF4WCkgKyBleHBvcnRQYWRkaW5nICogMjtcbiAgY29uc3QgaGVpZ2h0ID0gZGlzdGFuY2UobWluWSwgbWF4WSkgKyBleHBvcnRQYWRkaW5nICogMjtcbiAgcmV0dXJuIFttaW5YLCBtaW5ZLCB3aWR0aCwgaGVpZ2h0XTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRFeHBvcnRTaXplID0gKGVsZW1lbnRzLCBleHBvcnRQYWRkaW5nLCBzY2FsZSkgPT4ge1xuICBjb25zdCBbLCwgd2lkdGgsIGhlaWdodF0gPSBnZXRDYW52YXNTaXplKGVsZW1lbnRzLCBleHBvcnRQYWRkaW5nKS5tYXAoZGltZW5zaW9uID0+IE1hdGgudHJ1bmMoZGltZW5zaW9uICogc2NhbGUpKTtcbiAgcmV0dXJuIFt3aWR0aCwgaGVpZ2h0XTtcbn07Il0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///../../scene/export.ts\n");
4523
+ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"exportToCanvas\": () => (/* binding */ exportToCanvas),\n/* harmony export */ \"exportToSvg\": () => (/* binding */ exportToSvg),\n/* harmony export */ \"getExportSize\": () => (/* binding */ getExportSize)\n/* harmony export */ });\n/* harmony import */ var roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! roughjs/bin/rough */ \"../../../node_modules/roughjs/bin/rough.js\");\n/* harmony import */ var _element_bounds__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../element/bounds */ \"../../element/bounds.ts\");\n/* harmony import */ var _renderer_renderScene__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../renderer/renderScene */ \"../../renderer/renderScene.ts\");\n/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../utils */ \"../../utils.ts\");\n/* harmony import */ var _constants__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../constants */ \"../../constants.ts\");\n/* harmony import */ var _appState__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../appState */ \"../../appState.ts\");\n/* harmony import */ var _data_json__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ../data/json */ \"../../data/json.ts\");\n/* harmony import */ var _element_image__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../element/image */ \"../../element/image.ts\");\n/* harmony import */ var _packages_withinBounds__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ../packages/withinBounds */ \"../withinBounds.ts\");\n/* harmony import */ var _frame__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! ../frame */ \"../../frame.ts\");\n/* harmony import */ var _element__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! ../element */ \"../../element/index.ts\");\n/* harmony import */ var _element_mutateElement__WEBPACK_IMPORTED_MODULE_11__ = __webpack_require__(/*! ../element/mutateElement */ \"../../element/mutateElement.ts\");\n/* harmony import */ var _Scene__WEBPACK_IMPORTED_MODULE_12__ = __webpack_require__(/*! ./Scene */ \"../../scene/Scene.ts\");\nvar __awaiter = undefined && undefined.__awaiter || function (thisArg, _arguments, P, generator) {\n function adopt(value) {\n return value instanceof P ? value : new P(function (resolve) {\n resolve(value);\n });\n }\n\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) {\n try {\n step(generator.next(value));\n } catch (e) {\n reject(e);\n }\n }\n\n function rejected(value) {\n try {\n step(generator[\"throw\"](value));\n } catch (e) {\n reject(e);\n }\n }\n\n function step(result) {\n result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected);\n }\n\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n};\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nconst SVG_EXPORT_TAG = `<!-- svg-source:excalidraw -->`; // getContainerElement and getBoundTextElement and potentially other helpers\n// depend on `Scene` which will not be available when these pure utils are\n// called outside initialized Excalidraw editor instance or even if called\n// from inside Excalidraw if the elements were never cached by Scene (e.g.\n// for library elements).\n//\n// As such, before passing the elements down, we need to initialize a custom\n// Scene instance and assign them to it.\n//\n// FIXME This is a super hacky workaround and we'll need to rewrite this soon.\n\nconst __createSceneForElementsHack__ = elements => {\n const scene = new _Scene__WEBPACK_IMPORTED_MODULE_12__[\"default\"](); // we can't duplicate elements to regenerate ids because we need the\n // orig ids when embedding. So we do another hack of not mapping element\n // ids to Scene instances so that we don't override the editor elements\n // mapping.\n // We still need to clone the objects themselves to regen references.\n\n scene.replaceAllElements((0,_utils__WEBPACK_IMPORTED_MODULE_3__.cloneJSON)(elements), false);\n return scene;\n};\n\nconst truncateText = (element, maxWidth) => {\n if (element.width <= maxWidth) {\n return element;\n }\n\n const canvas = document.createElement(\"canvas\");\n const ctx = canvas.getContext(\"2d\");\n ctx.font = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.getFontString)({\n fontFamily: element.fontFamily,\n fontSize: element.fontSize\n });\n let text = element.text;\n const metrics = ctx.measureText(text);\n\n if (metrics.width > maxWidth) {\n // we iterate from the right, removing characters one by one instead\n // of bulding the string up. This assumes that it's more likely\n // your frame names will overflow by not that many characters\n // (if ever), so it sohuld be faster this way.\n for (let i = text.length; i > 0; i--) {\n const newText = `${text.slice(0, i)}...`;\n\n if (ctx.measureText(newText).width <= maxWidth) {\n text = newText;\n break;\n }\n }\n }\n\n return (0,_element_mutateElement__WEBPACK_IMPORTED_MODULE_11__.newElementWith)(element, {\n text,\n width: maxWidth\n });\n};\n/**\n * When exporting frames, we need to render frame labels which are currently\n * being rendered in DOM when editing. Adding the labels as regular text\n * elements seems like a simple hack. In the future we'll want to move to\n * proper canvas rendering, even within editor (instead of DOM).\n */\n\n\nconst addFrameLabelsAsTextElements = (elements, opts) => {\n const nextElements = [];\n let frameIdx = 0;\n\n for (const element of elements) {\n if ((0,_element__WEBPACK_IMPORTED_MODULE_10__.isFrameElement)(element)) {\n frameIdx++;\n let textElement = (0,_element__WEBPACK_IMPORTED_MODULE_10__.newTextElement)({\n x: element.x,\n y: element.y - _constants__WEBPACK_IMPORTED_MODULE_4__.FRAME_STYLE.nameOffsetY,\n fontFamily: _constants__WEBPACK_IMPORTED_MODULE_4__.FONT_FAMILY.Assistant,\n fontSize: _constants__WEBPACK_IMPORTED_MODULE_4__.FRAME_STYLE.nameFontSize,\n lineHeight: _constants__WEBPACK_IMPORTED_MODULE_4__.FRAME_STYLE.nameLineHeight,\n strokeColor: opts.exportWithDarkMode ? _constants__WEBPACK_IMPORTED_MODULE_4__.FRAME_STYLE.nameColorDarkTheme : _constants__WEBPACK_IMPORTED_MODULE_4__.FRAME_STYLE.nameColorLightTheme,\n text: element.name || `Frame ${frameIdx}`\n });\n textElement.y -= textElement.height;\n textElement = truncateText(textElement, element.width);\n nextElements.push(textElement);\n }\n\n nextElements.push(element);\n }\n\n return nextElements;\n};\n\nconst getFrameRenderingConfig = (exportingFrame, frameRendering) => {\n frameRendering = frameRendering || (0,_appState__WEBPACK_IMPORTED_MODULE_5__.getDefaultAppState)().frameRendering;\n return {\n enabled: exportingFrame ? true : frameRendering.enabled,\n outline: exportingFrame ? false : frameRendering.outline,\n name: exportingFrame ? false : frameRendering.name,\n clip: exportingFrame ? true : frameRendering.clip\n };\n};\n\nconst prepareElementsForRender = ({\n elements,\n exportingFrame,\n frameRendering,\n exportWithDarkMode\n}) => {\n let nextElements;\n\n if (exportingFrame) {\n nextElements = (0,_packages_withinBounds__WEBPACK_IMPORTED_MODULE_8__.elementsOverlappingBBox)({\n elements,\n bounds: exportingFrame,\n type: \"overlap\"\n });\n } else if (frameRendering.enabled && frameRendering.name) {\n nextElements = addFrameLabelsAsTextElements(elements, {\n exportWithDarkMode\n });\n } else {\n nextElements = elements;\n }\n\n return nextElements;\n};\n\nconst exportToCanvas = (elements, appState, files, {\n exportBackground,\n exportPadding = _constants__WEBPACK_IMPORTED_MODULE_4__.DEFAULT_EXPORT_PADDING,\n viewBackgroundColor,\n exportingFrame\n}, createCanvas = (width, height) => {\n const canvas = document.createElement(\"canvas\");\n canvas.width = width * appState.exportScale;\n canvas.height = height * appState.exportScale;\n return {\n canvas,\n scale: appState.exportScale\n };\n}) => __awaiter(void 0, void 0, void 0, function* () {\n var _a;\n\n const tempScene = __createSceneForElementsHack__(elements);\n\n elements = tempScene.getNonDeletedElements();\n const frameRendering = getFrameRenderingConfig(exportingFrame !== null && exportingFrame !== void 0 ? exportingFrame : null, (_a = appState.frameRendering) !== null && _a !== void 0 ? _a : null);\n const elementsForRender = prepareElementsForRender({\n elements,\n exportingFrame,\n exportWithDarkMode: appState.exportWithDarkMode,\n frameRendering\n });\n\n if (exportingFrame) {\n exportPadding = 0;\n }\n\n const [minX, minY, width, height] = getCanvasSize(exportingFrame ? [exportingFrame] : (0,_frame__WEBPACK_IMPORTED_MODULE_9__.getRootElements)(elementsForRender), exportPadding);\n const {\n canvas,\n scale = 1\n } = createCanvas(width, height);\n const defaultAppState = (0,_appState__WEBPACK_IMPORTED_MODULE_5__.getDefaultAppState)();\n const {\n imageCache\n } = yield (0,_element_image__WEBPACK_IMPORTED_MODULE_7__.updateImageCache)({\n imageCache: new Map(),\n fileIds: (0,_element_image__WEBPACK_IMPORTED_MODULE_7__.getInitializedImageElements)(elementsForRender).map(element => element.fileId),\n files\n });\n (0,_renderer_renderScene__WEBPACK_IMPORTED_MODULE_2__.renderStaticScene)({\n canvas,\n rc: roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_0__[\"default\"].canvas(canvas),\n elements: elementsForRender,\n visibleElements: elementsForRender,\n scale,\n appState: Object.assign(Object.assign({}, appState), {\n frameRendering,\n viewBackgroundColor: exportBackground ? viewBackgroundColor : null,\n scrollX: -minX + exportPadding,\n scrollY: -minY + exportPadding,\n zoom: defaultAppState.zoom,\n shouldCacheIgnoreZoom: false,\n theme: appState.exportWithDarkMode ? \"dark\" : \"light\"\n }),\n renderConfig: {\n imageCache,\n renderGrid: false,\n isExporting: true\n }\n });\n tempScene.destroy();\n return canvas;\n});\nconst exportToSvg = (elements, appState, files, opts) => __awaiter(void 0, void 0, void 0, function* () {\n var _b, _c, _d;\n\n const tempScene = __createSceneForElementsHack__(elements);\n\n elements = tempScene.getNonDeletedElements();\n const frameRendering = getFrameRenderingConfig((_b = opts === null || opts === void 0 ? void 0 : opts.exportingFrame) !== null && _b !== void 0 ? _b : null, (_c = appState.frameRendering) !== null && _c !== void 0 ? _c : null);\n let {\n exportPadding = _constants__WEBPACK_IMPORTED_MODULE_4__.DEFAULT_EXPORT_PADDING,\n exportWithDarkMode = false,\n viewBackgroundColor,\n exportScale = 1,\n exportEmbedScene\n } = appState;\n const {\n exportingFrame = null\n } = opts || {};\n const elementsForRender = prepareElementsForRender({\n elements,\n exportingFrame,\n exportWithDarkMode,\n frameRendering\n });\n\n if (exportingFrame) {\n exportPadding = 0;\n }\n\n let metadata = \"\"; // we need to serialize the \"original\" elements before we put them through\n // the tempScene hack which duplicates and regenerates ids\n\n if (exportEmbedScene) {\n try {\n metadata = yield (yield Promise.resolve(/*! import() */).then(__webpack_require__.bind(__webpack_require__, /*! ../../src/data/image */ \"../../data/image.ts\"))).encodeSvgMetadata({\n // when embedding scene, we want to embed the origionally supplied\n // elements which don't contain the temp frame labels.\n // But it also requires that the exportToSvg is being supplied with\n // only the elements that we're exporting, and no extra.\n text: (0,_data_json__WEBPACK_IMPORTED_MODULE_6__.serializeAsJSON)(elements, appState, files || {}, \"local\")\n });\n } catch (error) {\n console.error(error);\n }\n }\n\n const [minX, minY, width, height] = getCanvasSize(exportingFrame ? [exportingFrame] : (0,_frame__WEBPACK_IMPORTED_MODULE_9__.getRootElements)(elementsForRender), exportPadding); // initialize SVG root\n\n const svgRoot = document.createElementNS(_constants__WEBPACK_IMPORTED_MODULE_4__.SVG_NS, \"svg\");\n svgRoot.setAttribute(\"version\", \"1.1\");\n svgRoot.setAttribute(\"xmlns\", _constants__WEBPACK_IMPORTED_MODULE_4__.SVG_NS);\n svgRoot.setAttribute(\"viewBox\", `0 0 ${width} ${height}`);\n svgRoot.setAttribute(\"width\", `${width * exportScale}`);\n svgRoot.setAttribute(\"height\", `${height * exportScale}`);\n\n if (exportWithDarkMode) {\n svgRoot.setAttribute(\"filter\", _constants__WEBPACK_IMPORTED_MODULE_4__.THEME_FILTER);\n }\n\n let assetPath = \"https://excalidraw.com/\"; // Asset path needs to be determined only when using package\n\n if (true) {\n assetPath = window.EXCALIDRAW_ASSET_PATH || `https://unpkg.com/${\"@excalidraw/excalidraw\"}@${({\"VITE_APP_BACKEND_V2_GET_URL\":\"https://json-dev.excalidraw.com/api/v2/\",\"VITE_APP_BACKEND_V2_POST_URL\":\"https://json-dev.excalidraw.com/api/v2/post/\",\"VITE_APP_LIBRARY_URL\":\"https://libraries.excalidraw.com\",\"VITE_APP_LIBRARY_BACKEND\":\"https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries\",\"VITE_APP_WS_SERVER_URL\":\"http://localhost:3002\",\"VITE_APP_PORTAL_URL\":\"\",\"VITE_APP_PLUS_LP\":\"https://plus.excalidraw.com\",\"VITE_APP_PLUS_APP\":\"https://app.excalidraw.com\",\"VITE_APP_FIREBASE_CONFIG\":\"{\\\"apiKey\\\":\\\"AIzaSyCMkxA60XIW8KbqMYL7edC4qT5l4qHX2h8\\\",\\\"authDomain\\\":\\\"excalidraw-oss-dev.firebaseapp.com\\\",\\\"projectId\\\":\\\"excalidraw-oss-dev\\\",\\\"storageBucket\\\":\\\"excalidraw-oss-dev.appspot.com\\\",\\\"messagingSenderId\\\":\\\"664559512677\\\",\\\"appId\\\":\\\"1:664559512677:web:a385181f2928d328a7aa8c\\\"}\",\"VITE_APP_DEV_ENABLE_SW\":\"\",\"VITE_APP_DEV_DISABLE_LIVE_RELOAD\":\"\",\"VITE_APP_DISABLE_TRACKING\":\"true\",\"FAST_REFRESH\":\"false\",\"VITE_APP_PORT\":\"3000\",\"VITE_APP_DEBUG_ENABLE_TEXT_CONTAINER_BOUNDING_BOX\":\"\",\"VITE_APP_COLLAPSE_OVERLAY\":\"true\",\"VITE_APP_ENABLE_ESLINT\":\"true\",\"VITE_PKG_NAME\":\"@excalidraw/excalidraw\",\"VITE_PKG_VERSION\":\"0.17.1\",\"VITE_IS_EXCALIDRAW_NPM_PACKAGE\":true}).PKG_VERSION}`;\n\n if (assetPath === null || assetPath === void 0 ? void 0 : assetPath.startsWith(\"/\")) {\n assetPath = assetPath.replace(\"/\", `${window.location.origin}/`);\n }\n\n assetPath = `${assetPath}/dist/excalidraw-assets/`;\n }\n\n const offsetX = -minX + exportPadding;\n const offsetY = -minY + exportPadding;\n const frameElements = (0,_frame__WEBPACK_IMPORTED_MODULE_9__.getFrameElements)(elements);\n let exportingFrameClipPath = \"\";\n\n for (const frame of frameElements) {\n const [x1, y1, x2, y2] = (0,_element_bounds__WEBPACK_IMPORTED_MODULE_1__.getElementAbsoluteCoords)(frame);\n const cx = (x2 - x1) / 2 - (frame.x - x1);\n const cy = (y2 - y1) / 2 - (frame.y - y1);\n exportingFrameClipPath += `<clipPath id=${frame.id}>\n <rect transform=\"translate(${frame.x + offsetX} ${frame.y + offsetY}) rotate(${frame.angle} ${cx} ${cy})\"\n width=\"${frame.width}\"\n height=\"${frame.height}\"\n >\n </rect>\n </clipPath>`;\n }\n\n svgRoot.innerHTML = `\n ${SVG_EXPORT_TAG}\n ${metadata}\n <defs>\n <style class=\"style-fonts\">\n @font-face {\n font-family: \"Virgil\";\n src: url(\"${assetPath}Virgil.woff2\");\n }\n @font-face {\n font-family: \"Cascadia\";\n src: url(\"${assetPath}Cascadia.woff2\");\n }\n @font-face {\n font-family: \"Assistant\";\n src: url(\"${assetPath}Assistant-Regular.woff2\");\n }\n </style>\n ${exportingFrameClipPath}\n </defs>\n `; // render background rect\n\n if (appState.exportBackground && viewBackgroundColor) {\n const rect = svgRoot.ownerDocument.createElementNS(_constants__WEBPACK_IMPORTED_MODULE_4__.SVG_NS, \"rect\");\n rect.setAttribute(\"x\", \"0\");\n rect.setAttribute(\"y\", \"0\");\n rect.setAttribute(\"width\", `${width}`);\n rect.setAttribute(\"height\", `${height}`);\n rect.setAttribute(\"fill\", viewBackgroundColor);\n svgRoot.appendChild(rect);\n }\n\n const rsvg = roughjs_bin_rough__WEBPACK_IMPORTED_MODULE_0__[\"default\"].svg(svgRoot);\n (0,_renderer_renderScene__WEBPACK_IMPORTED_MODULE_2__.renderSceneToSvg)(elementsForRender, rsvg, svgRoot, files || {}, {\n offsetX,\n offsetY,\n exportWithDarkMode,\n renderEmbeddables: (_d = opts === null || opts === void 0 ? void 0 : opts.renderEmbeddables) !== null && _d !== void 0 ? _d : false,\n frameRendering\n });\n tempScene.destroy();\n return svgRoot;\n}); // calculate smallest area to fit the contents in\n\nconst getCanvasSize = (elements, exportPadding) => {\n const [minX, minY, maxX, maxY] = (0,_element_bounds__WEBPACK_IMPORTED_MODULE_1__.getCommonBounds)(elements);\n const width = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.distance)(minX, maxX) + exportPadding * 2;\n const height = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.distance)(minY, maxY) + exportPadding * 2;\n return [minX, minY, width, height];\n};\n\nconst getExportSize = (elements, exportPadding, scale) => {\n const [,, width, height] = getCanvasSize(elements, exportPadding).map(dimension => Math.trunc(dimension * scale));\n return [width, height];\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,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LGVBQWUsNkRBQVM7QUFDeEIsRUFBRSx1RUFBZ0IsOENBQThDO0FBQ2hFO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxHQUFHO0FBQ0g7QUFDQTtBQUNBLENBQUMsR0FBRzs7QUFFSjtBQUNBLG1DQUFtQyxnRUFBZTtBQUNsRCxnQkFBZ0IsZ0RBQVE7QUFDeEIsaUJBQWlCLGdEQUFRO0FBQ3pCO0FBQ0E7O0FBRU87QUFDUDtBQUNBO0FBQ0EiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi4vLi4vc2NlbmUvZXhwb3J0LnRzP2MzNzMiXSwic291cmNlc0NvbnRlbnQiOlsidmFyIF9fYXdhaXRlciA9IHRoaXMgJiYgdGhpcy5fX2F3YWl0ZXIgfHwgZnVuY3Rpb24gKHRoaXNBcmcsIF9hcmd1bWVudHMsIFAsIGdlbmVyYXRvcikge1xuICBmdW5jdGlvbiBhZG9wdCh2YWx1ZSkge1xuICAgIHJldHVybiB2YWx1ZSBpbnN0YW5jZW9mIFAgPyB2YWx1ZSA6IG5ldyBQKGZ1bmN0aW9uIChyZXNvbHZlKSB7XG4gICAgICByZXNvbHZlKHZhbHVlKTtcbiAgICB9KTtcbiAgfVxuXG4gIHJldHVybiBuZXcgKFAgfHwgKFAgPSBQcm9taXNlKSkoZnVuY3Rpb24gKHJlc29sdmUsIHJlamVjdCkge1xuICAgIGZ1bmN0aW9uIGZ1bGZpbGxlZCh2YWx1ZSkge1xuICAgICAgdHJ5IHtcbiAgICAgICAgc3RlcChnZW5lcmF0b3IubmV4dCh2YWx1ZSkpO1xuICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgICByZWplY3QoZSk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgZnVuY3Rpb24gcmVqZWN0ZWQodmFsdWUpIHtcbiAgICAgIHRyeSB7XG4gICAgICAgIHN0ZXAoZ2VuZXJhdG9yW1widGhyb3dcIl0odmFsdWUpKTtcbiAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgcmVqZWN0KGUpO1xuICAgICAgfVxuICAgIH1cblxuICAgIGZ1bmN0aW9uIHN0ZXAocmVzdWx0KSB7XG4gICAgICByZXN1bHQuZG9uZSA/IHJlc29sdmUocmVzdWx0LnZhbHVlKSA6IGFkb3B0KHJlc3VsdC52YWx1ZSkudGhlbihmdWxmaWxsZWQsIHJlamVjdGVkKTtcbiAgICB9XG5cbiAgICBzdGVwKChnZW5lcmF0b3IgPSBnZW5lcmF0b3IuYXBwbHkodGhpc0FyZywgX2FyZ3VtZW50cyB8fCBbXSkpLm5leHQoKSk7XG4gIH0pO1xufTtcblxuaW1wb3J0IHJvdWdoIGZyb20gXCJyb3VnaGpzL2Jpbi9yb3VnaFwiO1xuaW1wb3J0IHsgZ2V0Q29tbW9uQm91bmRzLCBnZXRFbGVtZW50QWJzb2x1dGVDb29yZHMgfSBmcm9tIFwiLi4vZWxlbWVudC9ib3VuZHNcIjtcbmltcG9ydCB7IHJlbmRlclNjZW5lVG9TdmcsIHJlbmRlclN0YXRpY1NjZW5lIH0gZnJvbSBcIi4uL3JlbmRlcmVyL3JlbmRlclNjZW5lXCI7XG5pbXBvcnQgeyBjbG9uZUpTT04sIGRpc3RhbmNlLCBnZXRGb250U3RyaW5nIH0gZnJvbSBcIi4uL3V0aWxzXCI7XG5pbXBvcnQgeyBERUZBVUxUX0VYUE9SVF9QQURESU5HLCBGT05UX0ZBTUlMWSwgRlJBTUVfU1RZTEUsIFNWR19OUywgVEhFTUVfRklMVEVSIH0gZnJvbSBcIi4uL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgZ2V0RGVmYXVsdEFwcFN0YXRlIH0gZnJvbSBcIi4uL2FwcFN0YXRlXCI7XG5pbXBvcnQgeyBzZXJpYWxpemVBc0pTT04gfSBmcm9tIFwiLi4vZGF0YS9qc29uXCI7XG5pbXBvcnQgeyBnZXRJbml0aWFsaXplZEltYWdlRWxlbWVudHMsIHVwZGF0ZUltYWdlQ2FjaGUgfSBmcm9tIFwiLi4vZWxlbWVudC9pbWFnZVwiO1xuaW1wb3J0IHsgZWxlbWVudHNPdmVybGFwcGluZ0JCb3ggfSBmcm9tIFwiLi4vcGFja2FnZXMvd2l0aGluQm91bmRzXCI7XG5pbXBvcnQgeyBnZXRGcmFtZUVsZW1lbnRzLCBnZXRSb290RWxlbWVudHMgfSBmcm9tIFwiLi4vZnJhbWVcIjtcbmltcG9ydCB7IGlzRnJhbWVFbGVtZW50LCBuZXdUZXh0RWxlbWVudCB9IGZyb20gXCIuLi9lbGVtZW50XCI7XG5pbXBvcnQgeyBuZXdFbGVtZW50V2l0aCB9IGZyb20gXCIuLi9lbGVtZW50L211dGF0ZUVsZW1lbnRcIjtcbmltcG9ydCBTY2VuZSBmcm9tIFwiLi9TY2VuZVwiO1xuY29uc3QgU1ZHX0VYUE9SVF9UQUcgPSBgPCEtLSBzdmctc291cmNlOmV4Y2FsaWRyYXcgLS0+YDsgLy8gZ2V0Q29udGFpbmVyRWxlbWVudCBhbmQgZ2V0Qm91bmRUZXh0RWxlbWVudCBhbmQgcG90ZW50aWFsbHkgb3RoZXIgaGVscGVyc1xuLy8gZGVwZW5kIG9uIGBTY2VuZWAgd2hpY2ggd2lsbCBub3QgYmUgYXZhaWxhYmxlIHdoZW4gdGhlc2UgcHVyZSB1dGlscyBhcmVcbi8vIGNhbGxlZCBvdXRzaWRlIGluaXRpYWxpemVkIEV4Y2FsaWRyYXcgZWRpdG9yIGluc3RhbmNlIG9yIGV2ZW4gaWYgY2FsbGVkXG4vLyBmcm9tIGluc2lkZSBFeGNhbGlkcmF3IGlmIHRoZSBlbGVtZW50cyB3ZXJlIG5ldmVyIGNhY2hlZCBieSBTY2VuZSAoZS5nLlxuLy8gZm9yIGxpYnJhcnkgZWxlbWVudHMpLlxuLy9cbi8vIEFzIHN1Y2gsIGJlZm9yZSBwYXNzaW5nIHRoZSBlbGVtZW50cyBkb3duLCB3ZSBuZWVkIHRvIGluaXRpYWxpemUgYSBjdXN0b21cbi8vIFNjZW5lIGluc3RhbmNlIGFuZCBhc3NpZ24gdGhlbSB0byBpdC5cbi8vXG4vLyBGSVhNRSBUaGlzIGlzIGEgc3VwZXIgaGFja3kgd29ya2Fyb3VuZCBhbmQgd2UnbGwgbmVlZCB0byByZXdyaXRlIHRoaXMgc29vbi5cblxuY29uc3QgX19jcmVhdGVTY2VuZUZvckVsZW1lbnRzSGFja19fID0gZWxlbWVudHMgPT4ge1xuICBjb25zdCBzY2VuZSA9IG5ldyBTY2VuZSgpOyAvLyB3ZSBjYW4ndCBkdXBsaWNhdGUgZWxlbWVudHMgdG8gcmVnZW5lcmF0ZSBpZHMgYmVjYXVzZSB3ZSBuZWVkIHRoZVxuICAvLyBvcmlnIGlkcyB3aGVuIGVtYmVkZGluZy4gU28gd2UgZG8gYW5vdGhlciBoYWNrIG9mIG5vdCBtYXBwaW5nIGVsZW1lbnRcbiAgLy8gaWRzIHRvIFNjZW5lIGluc3RhbmNlcyBzbyB0aGF0IHdlIGRvbid0IG92ZXJyaWRlIHRoZSBlZGl0b3IgZWxlbWVudHNcbiAgLy8gbWFwcGluZy5cbiAgLy8gV2Ugc3RpbGwgbmVlZCB0byBjbG9uZSB0aGUgb2JqZWN0cyB0aGVtc2VsdmVzIHRvIHJlZ2VuIHJlZmVyZW5jZXMuXG5cbiAgc2NlbmUucmVwbGFjZUFsbEVsZW1lbnRzKGNsb25lSlNPTihlbGVtZW50cyksIGZhbHNlKTtcbiAgcmV0dXJuIHNjZW5lO1xufTtcblxuY29uc3QgdHJ1bmNhdGVUZXh0ID0gKGVsZW1lbnQsIG1heFdpZHRoKSA9PiB7XG4gIGlmIChlbGVtZW50LndpZHRoIDw9IG1heFdpZHRoKSB7XG4gICAgcmV0dXJuIGVsZW1lbnQ7XG4gIH1cblxuICBjb25zdCBjYW52YXMgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiY2FudmFzXCIpO1xuICBjb25zdCBjdHggPSBjYW52YXMuZ2V0Q29udGV4dChcIjJkXCIpO1xuICBjdHguZm9udCA9IGdldEZvbnRTdHJpbmcoe1xuICAgIGZvbnRGYW1pbHk6IGVsZW1lbnQuZm9udEZhbWlseSxcbiAgICBmb250U2l6ZTogZWxlbWVudC5mb250U2l6ZVxuICB9KTtcbiAgbGV0IHRleHQgPSBlbGVtZW50LnRleHQ7XG4gIGNvbnN0IG1ldHJpY3MgPSBjdHgubWVhc3VyZVRleHQodGV4dCk7XG5cbiAgaWYgKG1ldHJpY3Mud2lkdGggPiBtYXhXaWR0aCkge1xuICAgIC8vIHdlIGl0ZXJhdGUgZnJvbSB0aGUgcmlnaHQsIHJlbW92aW5nIGNoYXJhY3RlcnMgb25lIGJ5IG9uZSBpbnN0ZWFkXG4gICAgLy8gb2YgYnVsZGluZyB0aGUgc3RyaW5nIHVwLiBUaGlzIGFzc3VtZXMgdGhhdCBpdCdzIG1vcmUgbGlrZWx5XG4gICAgLy8geW91ciBmcmFtZSBuYW1lcyB3aWxsIG92ZXJmbG93IGJ5IG5vdCB0aGF0IG1hbnkgY2hhcmFjdGVyc1xuICAgIC8vIChpZiBldmVyKSwgc28gaXQgc29odWxkIGJlIGZhc3RlciB0aGlzIHdheS5cbiAgICBmb3IgKGxldCBpID0gdGV4dC5sZW5ndGg7IGkgPiAwOyBpLS0pIHtcbiAgICAgIGNvbnN0IG5ld1RleHQgPSBgJHt0ZXh0LnNsaWNlKDAsIGkpfS4uLmA7XG5cbiAgICAgIGlmIChjdHgubWVhc3VyZVRleHQobmV3VGV4dCkud2lkdGggPD0gbWF4V2lkdGgpIHtcbiAgICAgICAgdGV4dCA9IG5ld1RleHQ7XG4gICAgICAgIGJyZWFrO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIHJldHVybiBuZXdFbGVtZW50V2l0aChlbGVtZW50LCB7XG4gICAgdGV4dCxcbiAgICB3aWR0aDogbWF4V2lkdGhcbiAgfSk7XG59O1xuLyoqXG4gKiBXaGVuIGV4cG9ydGluZyBmcmFtZXMsIHdlIG5lZWQgdG8gcmVuZGVyIGZyYW1lIGxhYmVscyB3aGljaCBhcmUgY3VycmVudGx5XG4gKiBiZWluZyByZW5kZXJlZCBpbiBET00gd2hlbiBlZGl0aW5nLiBBZGRpbmcgdGhlIGxhYmVscyBhcyByZWd1bGFyIHRleHRcbiAqIGVsZW1lbnRzIHNlZW1zIGxpa2UgYSBzaW1wbGUgaGFjay4gSW4gdGhlIGZ1dHVyZSB3ZSdsbCB3YW50IHRvIG1vdmUgdG9cbiAqIHByb3BlciBjYW52YXMgcmVuZGVyaW5nLCBldmVuIHdpdGhpbiBlZGl0b3IgKGluc3RlYWQgb2YgRE9NKS5cbiAqL1xuXG5cbmNvbnN0IGFkZEZyYW1lTGFiZWxzQXNUZXh0RWxlbWVudHMgPSAoZWxlbWVudHMsIG9wdHMpID0+IHtcbiAgY29uc3QgbmV4dEVsZW1lbnRzID0gW107XG4gIGxldCBmcmFtZUlkeCA9IDA7XG5cbiAgZm9yIChjb25zdCBlbGVtZW50IG9mIGVsZW1lbnRzKSB7XG4gICAgaWYgKGlzRnJhbWVFbGVtZW50KGVsZW1lbnQpKSB7XG4gICAgICBmcmFtZUlkeCsrO1xuICAgICAgbGV0IHRleHRFbGVtZW50ID0gbmV3VGV4dEVsZW1lbnQoe1xuICAgICAgICB4OiBlbGVtZW50LngsXG4gICAgICAgIHk6IGVsZW1lbnQueSAtIEZSQU1FX1NUWUxFLm5hbWVPZmZzZXRZLFxuICAgICAgICBmb250RmFtaWx5OiBGT05UX0ZBTUlMWS5Bc3Npc3RhbnQsXG4gICAgICAgIGZvbnRTaXplOiBGUkFNRV9TVFlMRS5uYW1lRm9udFNpemUsXG4gICAgICAgIGxpbmVIZWlnaHQ6IEZSQU1FX1NUWUxFLm5hbWVMaW5lSGVpZ2h0LFxuICAgICAgICBzdHJva2VDb2xvcjogb3B0cy5leHBvcnRXaXRoRGFya01vZGUgPyBGUkFNRV9TVFlMRS5uYW1lQ29sb3JEYXJrVGhlbWUgOiBGUkFNRV9TVFlMRS5uYW1lQ29sb3JMaWdodFRoZW1lLFxuICAgICAgICB0ZXh0OiBlbGVtZW50Lm5hbWUgfHwgYEZyYW1lICR7ZnJhbWVJZHh9YFxuICAgICAgfSk7XG4gICAgICB0ZXh0RWxlbWVudC55IC09IHRleHRFbGVtZW50LmhlaWdodDtcbiAgICAgIHRleHRFbGVtZW50ID0gdHJ1bmNhdGVUZXh0KHRleHRFbGVtZW50LCBlbGVtZW50LndpZHRoKTtcbiAgICAgIG5leHRFbGVtZW50cy5wdXNoKHRleHRFbGVtZW50KTtcbiAgICB9XG5cbiAgICBuZXh0RWxlbWVudHMucHVzaChlbGVtZW50KTtcbiAgfVxuXG4gIHJldHVybiBuZXh0RWxlbWVudHM7XG59O1xuXG5jb25zdCBnZXRGcmFtZVJlbmRlcmluZ0NvbmZpZyA9IChleHBvcnRpbmdGcmFtZSwgZnJhbWVSZW5kZXJpbmcpID0+IHtcbiAgZnJhbWVSZW5kZXJpbmcgPSBmcmFtZVJlbmRlcmluZyB8fCBnZXREZWZhdWx0QXBwU3RhdGUoKS5mcmFtZVJlbmRlcmluZztcbiAgcmV0dXJuIHtcbiAgICBlbmFibGVkOiBleHBvcnRpbmdGcmFtZSA/IHRydWUgOiBmcmFtZVJlbmRlcmluZy5lbmFibGVkLFxuICAgIG91dGxpbmU6IGV4cG9ydGluZ0ZyYW1lID8gZmFsc2UgOiBmcmFtZVJlbmRlcmluZy5vdXRsaW5lLFxuICAgIG5hbWU6IGV4cG9ydGluZ0ZyYW1lID8gZmFsc2UgOiBmcmFtZVJlbmRlcmluZy5uYW1lLFxuICAgIGNsaXA6IGV4cG9ydGluZ0ZyYW1lID8gdHJ1ZSA6IGZyYW1lUmVuZGVyaW5nLmNsaXBcbiAgfTtcbn07XG5cbmNvbnN0IHByZXBhcmVFbGVtZW50c0ZvclJlbmRlciA9ICh7XG4gIGVsZW1lbnRzLFxuICBleHBvcnRpbmdGcmFtZSxcbiAgZnJhbWVSZW5kZXJpbmcsXG4gIGV4cG9ydFdpdGhEYXJrTW9kZVxufSkgPT4ge1xuICBsZXQgbmV4dEVsZW1lbnRzO1xuXG4gIGlmIChleHBvcnRpbmdGcmFtZSkge1xuICAgIG5leHRFbGVtZW50cyA9IGVsZW1lbnRzT3ZlcmxhcHBpbmdCQm94KHtcbiAgICAgIGVsZW1lbnRzLFxuICAgICAgYm91bmRzOiBleHBvcnRpbmdGcmFtZSxcbiAgICAgIHR5cGU6IFwib3ZlcmxhcFwiXG4gICAgfSk7XG4gIH0gZWxzZSBpZiAoZnJhbWVSZW5kZXJpbmcuZW5hYmxlZCAmJiBmcmFtZVJlbmRlcmluZy5uYW1lKSB7XG4gICAgbmV4dEVsZW1lbnRzID0gYWRkRnJhbWVMYWJlbHNBc1RleHRFbGVtZW50cyhlbGVtZW50cywge1xuICAgICAgZXhwb3J0V2l0aERhcmtNb2RlXG4gICAgfSk7XG4gIH0gZWxzZSB7XG4gICAgbmV4dEVsZW1lbnRzID0gZWxlbWVudHM7XG4gIH1cblxuICByZXR1cm4gbmV4dEVsZW1lbnRzO1xufTtcblxuZXhwb3J0IGNvbnN0IGV4cG9ydFRvQ2FudmFzID0gKGVsZW1lbnRzLCBhcHBTdGF0ZSwgZmlsZXMsIHtcbiAgZXhwb3J0QmFja2dyb3VuZCxcbiAgZXhwb3J0UGFkZGluZyA9IERFRkFVTFRfRVhQT1JUX1BBRERJTkcsXG4gIHZpZXdCYWNrZ3JvdW5kQ29sb3IsXG4gIGV4cG9ydGluZ0ZyYW1lXG59LCBjcmVhdGVDYW52YXMgPSAod2lkdGgsIGhlaWdodCkgPT4ge1xuICBjb25zdCBjYW52YXMgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiY2FudmFzXCIpO1xuICBjYW52YXMud2lkdGggPSB3aWR0aCAqIGFwcFN0YXRlLmV4cG9ydFNjYWxlO1xuICBjYW52YXMuaGVpZ2h0ID0gaGVpZ2h0ICogYXBwU3RhdGUuZXhwb3J0U2NhbGU7XG4gIHJldHVybiB7XG4gICAgY2FudmFzLFxuICAgIHNjYWxlOiBhcHBTdGF0ZS5leHBvcnRTY2FsZVxuICB9O1xufSkgPT4gX19hd2FpdGVyKHZvaWQgMCwgdm9pZCAwLCB2b2lkIDAsIGZ1bmN0aW9uKiAoKSB7XG4gIHZhciBfYTtcblxuICBjb25zdCB0ZW1wU2NlbmUgPSBfX2NyZWF0ZVNjZW5lRm9yRWxlbWVudHNIYWNrX18oZWxlbWVudHMpO1xuXG4gIGVsZW1lbnRzID0gdGVtcFNjZW5lLmdldE5vbkRlbGV0ZWRFbGVtZW50cygpO1xuICBjb25zdCBmcmFtZVJlbmRlcmluZyA9IGdldEZyYW1lUmVuZGVyaW5nQ29uZmlnKGV4cG9ydGluZ0ZyYW1lICE9PSBudWxsICYmIGV4cG9ydGluZ0ZyYW1lICE9PSB2b2lkIDAgPyBleHBvcnRpbmdGcmFtZSA6IG51bGwsIChfYSA9IGFwcFN0YXRlLmZyYW1lUmVuZGVyaW5nKSAhPT0gbnVsbCAmJiBfYSAhPT0gdm9pZCAwID8gX2EgOiBudWxsKTtcbiAgY29uc3QgZWxlbWVudHNGb3JSZW5kZXIgPSBwcmVwYXJlRWxlbWVudHNGb3JSZW5kZXIoe1xuICAgIGVsZW1lbnRzLFxuICAgIGV4cG9ydGluZ0ZyYW1lLFxuICAgIGV4cG9ydFdpdGhEYXJrTW9kZTogYXBwU3RhdGUuZXhwb3J0V2l0aERhcmtNb2RlLFxuICAgIGZyYW1lUmVuZGVyaW5nXG4gIH0pO1xuXG4gIGlmIChleHBvcnRpbmdGcmFtZSkge1xuICAgIGV4cG9ydFBhZGRpbmcgPSAwO1xuICB9XG5cbiAgY29uc3QgW21pblgsIG1pblksIHdpZHRoLCBoZWlnaHRdID0gZ2V0Q2FudmFzU2l6ZShleHBvcnRpbmdGcmFtZSA/IFtleHBvcnRpbmdGcmFtZV0gOiBnZXRSb290RWxlbWVudHMoZWxlbWVudHNGb3JSZW5kZXIpLCBleHBvcnRQYWRkaW5nKTtcbiAgY29uc3Qge1xuICAgIGNhbnZhcyxcbiAgICBzY2FsZSA9IDFcbiAgfSA9IGNyZWF0ZUNhbnZhcyh3aWR0aCwgaGVpZ2h0KTtcbiAgY29uc3QgZGVmYXVsdEFwcFN0YXRlID0gZ2V0RGVmYXVsdEFwcFN0YXRlKCk7XG4gIGNvbnN0IHtcbiAgICBpbWFnZUNhY2hlXG4gIH0gPSB5aWVsZCB1cGRhdGVJbWFnZUNhY2hlKHtcbiAgICBpbWFnZUNhY2hlOiBuZXcgTWFwKCksXG4gICAgZmlsZUlkczogZ2V0SW5pdGlhbGl6ZWRJbWFnZUVsZW1lbnRzKGVsZW1lbnRzRm9yUmVuZGVyKS5tYXAoZWxlbWVudCA9PiBlbGVtZW50LmZpbGVJZCksXG4gICAgZmlsZXNcbiAgfSk7XG4gIHJlbmRlclN0YXRpY1NjZW5lKHtcbiAgICBjYW52YXMsXG4gICAgcmM6IHJvdWdoLmNhbnZhcyhjYW52YXMpLFxuICAgIGVsZW1lbnRzOiBlbGVtZW50c0ZvclJlbmRlcixcbiAgICB2aXNpYmxlRWxlbWVudHM6IGVsZW1lbnRzRm9yUmVuZGVyLFxuICAgIHNjYWxlLFxuICAgIGFwcFN0YXRlOiBPYmplY3QuYXNzaWduKE9iamVjdC5hc3NpZ24oe30sIGFwcFN0YXRlKSwge1xuICAgICAgZnJhbWVSZW5kZXJpbmcsXG4gICAgICB2aWV3QmFja2dyb3VuZENvbG9yOiBleHBvcnRCYWNrZ3JvdW5kID8gdmlld0JhY2tncm91bmRDb2xvciA6IG51bGwsXG4gICAgICBzY3JvbGxYOiAtbWluWCArIGV4cG9ydFBhZGRpbmcsXG4gICAgICBzY3JvbGxZOiAtbWluWSArIGV4cG9ydFBhZGRpbmcsXG4gICAgICB6b29tOiBkZWZhdWx0QXBwU3RhdGUuem9vbSxcbiAgICAgIHNob3VsZENhY2hlSWdub3JlWm9vbTogZmFsc2UsXG4gICAgICB0aGVtZTogYXBwU3RhdGUuZXhwb3J0V2l0aERhcmtNb2RlID8gXCJkYXJrXCIgOiBcImxpZ2h0XCJcbiAgICB9KSxcbiAgICByZW5kZXJDb25maWc6IHtcbiAgICAgIGltYWdlQ2FjaGUsXG4gICAgICByZW5kZXJHcmlkOiBmYWxzZSxcbiAgICAgIGlzRXhwb3J0aW5nOiB0cnVlXG4gICAgfVxuICB9KTtcbiAgdGVtcFNjZW5lLmRlc3Ryb3koKTtcbiAgcmV0dXJuIGNhbnZhcztcbn0pO1xuZXhwb3J0IGNvbnN0IGV4cG9ydFRvU3ZnID0gKGVsZW1lbnRzLCBhcHBTdGF0ZSwgZmlsZXMsIG9wdHMpID0+IF9fYXdhaXRlcih2b2lkIDAsIHZvaWQgMCwgdm9pZCAwLCBmdW5jdGlvbiogKCkge1xuICB2YXIgX2IsIF9jLCBfZDtcblxuICBjb25zdCB0ZW1wU2NlbmUgPSBfX2NyZWF0ZVNjZW5lRm9yRWxlbWVudHNIYWNrX18oZWxlbWVudHMpO1xuXG4gIGVsZW1lbnRzID0gdGVtcFNjZW5lLmdldE5vbkRlbGV0ZWRFbGVtZW50cygpO1xuICBjb25zdCBmcmFtZVJlbmRlcmluZyA9IGdldEZyYW1lUmVuZGVyaW5nQ29uZmlnKChfYiA9IG9wdHMgPT09IG51bGwgfHwgb3B0cyA9PT0gdm9pZCAwID8gdm9pZCAwIDogb3B0cy5leHBvcnRpbmdGcmFtZSkgIT09IG51bGwgJiYgX2IgIT09IHZvaWQgMCA/IF9iIDogbnVsbCwgKF9jID0gYXBwU3RhdGUuZnJhbWVSZW5kZXJpbmcpICE9PSBudWxsICYmIF9jICE9PSB2b2lkIDAgPyBfYyA6IG51bGwpO1xuICBsZXQge1xuICAgIGV4cG9ydFBhZGRpbmcgPSBERUZBVUxUX0VYUE9SVF9QQURESU5HLFxuICAgIGV4cG9ydFdpdGhEYXJrTW9kZSA9IGZhbHNlLFxuICAgIHZpZXdCYWNrZ3JvdW5kQ29sb3IsXG4gICAgZXhwb3J0U2NhbGUgPSAxLFxuICAgIGV4cG9ydEVtYmVkU2NlbmVcbiAgfSA9IGFwcFN0YXRlO1xuICBjb25zdCB7XG4gICAgZXhwb3J0aW5nRnJhbWUgPSBudWxsXG4gIH0gPSBvcHRzIHx8IHt9O1xuICBjb25zdCBlbGVtZW50c0ZvclJlbmRlciA9IHByZXBhcmVFbGVtZW50c0ZvclJlbmRlcih7XG4gICAgZWxlbWVudHMsXG4gICAgZXhwb3J0aW5nRnJhbWUsXG4gICAgZXhwb3J0V2l0aERhcmtNb2RlLFxuICAgIGZyYW1lUmVuZGVyaW5nXG4gIH0pO1xuXG4gIGlmIChleHBvcnRpbmdGcmFtZSkge1xuICAgIGV4cG9ydFBhZGRpbmcgPSAwO1xuICB9XG5cbiAgbGV0IG1ldGFkYXRhID0gXCJcIjsgLy8gd2UgbmVlZCB0byBzZXJpYWxpemUgdGhlIFwib3JpZ2luYWxcIiBlbGVtZW50cyBiZWZvcmUgd2UgcHV0IHRoZW0gdGhyb3VnaFxuICAvLyB0aGUgdGVtcFNjZW5lIGhhY2sgd2hpY2ggZHVwbGljYXRlcyBhbmQgcmVnZW5lcmF0ZXMgaWRzXG5cbiAgaWYgKGV4cG9ydEVtYmVkU2NlbmUpIHtcbiAgICB0cnkge1xuICAgICAgbWV0YWRhdGEgPSB5aWVsZCAoeWllbGQgaW1wb3J0KFxuICAgICAgLyogd2VicGFja0NodW5rTmFtZTogXCJpbWFnZVwiICovXG4gICAgICBcIi4uLy4uL3NyYy9kYXRhL2ltYWdlXCIpKS5lbmNvZGVTdmdNZXRhZGF0YSh7XG4gICAgICAgIC8vIHdoZW4gZW1iZWRkaW5nIHNjZW5lLCB3ZSB3YW50IHRvIGVtYmVkIHRoZSBvcmlnaW9uYWxseSBzdXBwbGllZFxuICAgICAgICAvLyBlbGVtZW50cyB3aGljaCBkb24ndCBjb250YWluIHRoZSB0ZW1wIGZyYW1lIGxhYmVscy5cbiAgICAgICAgLy8gQnV0IGl0IGFsc28gcmVxdWlyZXMgdGhhdCB0aGUgZXhwb3J0VG9TdmcgaXMgYmVpbmcgc3VwcGxpZWQgd2l0aFxuICAgICAgICAvLyBvbmx5IHRoZSBlbGVtZW50cyB0aGF0IHdlJ3JlIGV4cG9ydGluZywgYW5kIG5vIGV4dHJhLlxuICAgICAgICB0ZXh0OiBzZXJpYWxpemVBc0pTT04oZWxlbWVudHMsIGFwcFN0YXRlLCBmaWxlcyB8fCB7fSwgXCJsb2NhbFwiKVxuICAgICAgfSk7XG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIGNvbnNvbGUuZXJyb3IoZXJyb3IpO1xuICAgIH1cbiAgfVxuXG4gIGNvbnN0IFttaW5YLCBtaW5ZLCB3aWR0aCwgaGVpZ2h0XSA9IGdldENhbnZhc1NpemUoZXhwb3J0aW5nRnJhbWUgPyBbZXhwb3J0aW5nRnJhbWVdIDogZ2V0Um9vdEVsZW1lbnRzKGVsZW1lbnRzRm9yUmVuZGVyKSwgZXhwb3J0UGFkZGluZyk7IC8vIGluaXRpYWxpemUgU1ZHIHJvb3RcblxuICBjb25zdCBzdmdSb290ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKFNWR19OUywgXCJzdmdcIik7XG4gIHN2Z1Jvb3Quc2V0QXR0cmlidXRlKFwidmVyc2lvblwiLCBcIjEuMVwiKTtcbiAgc3ZnUm9vdC5zZXRBdHRyaWJ1dGUoXCJ4bWxuc1wiLCBTVkdfTlMpO1xuICBzdmdSb290LnNldEF0dHJpYnV0ZShcInZpZXdCb3hcIiwgYDAgMCAke3dpZHRofSAke2hlaWdodH1gKTtcbiAgc3ZnUm9vdC5zZXRBdHRyaWJ1dGUoXCJ3aWR0aFwiLCBgJHt3aWR0aCAqIGV4cG9ydFNjYWxlfWApO1xuICBzdmdSb290LnNldEF0dHJpYnV0ZShcImhlaWdodFwiLCBgJHtoZWlnaHQgKiBleHBvcnRTY2FsZX1gKTtcblxuICBpZiAoZXhwb3J0V2l0aERhcmtNb2RlKSB7XG4gICAgc3ZnUm9vdC5zZXRBdHRyaWJ1dGUoXCJmaWx0ZXJcIiwgVEhFTUVfRklMVEVSKTtcbiAgfVxuXG4gIGxldCBhc3NldFBhdGggPSBcImh0dHBzOi8vZXhjYWxpZHJhdy5jb20vXCI7IC8vIEFzc2V0IHBhdGggbmVlZHMgdG8gYmUgZGV0ZXJtaW5lZCBvbmx5IHdoZW4gdXNpbmcgcGFja2FnZVxuXG4gIGlmIChwcm9jZXNzLmVudi5WSVRFX0lTX0VYQ0FMSURSQVdfTlBNX1BBQ0tBR0UpIHtcbiAgICBhc3NldFBhdGggPSB3aW5kb3cuRVhDQUxJRFJBV19BU1NFVF9QQVRIIHx8IGBodHRwczovL3VucGtnLmNvbS8ke3Byb2Nlc3MuZW52LlZJVEVfUEtHX05BTUV9QCR7cHJvY2Vzcy5lbnYuUEtHX1ZFUlNJT059YDtcblxuICAgIGlmIChhc3NldFBhdGggPT09IG51bGwgfHwgYXNzZXRQYXRoID09PSB2b2lkIDAgPyB2b2lkIDAgOiBhc3NldFBhdGguc3RhcnRzV2l0aChcIi9cIikpIHtcbiAgICAgIGFzc2V0UGF0aCA9IGFzc2V0UGF0aC5yZXBsYWNlKFwiL1wiLCBgJHt3aW5kb3cubG9jYXRpb24ub3JpZ2lufS9gKTtcbiAgICB9XG5cbiAgICBhc3NldFBhdGggPSBgJHthc3NldFBhdGh9L2Rpc3QvZXhjYWxpZHJhdy1hc3NldHMvYDtcbiAgfVxuXG4gIGNvbnN0IG9mZnNldFggPSAtbWluWCArIGV4cG9ydFBhZGRpbmc7XG4gIGNvbnN0IG9mZnNldFkgPSAtbWluWSArIGV4cG9ydFBhZGRpbmc7XG4gIGNvbnN0IGZyYW1lRWxlbWVudHMgPSBnZXRGcmFtZUVsZW1lbnRzKGVsZW1lbnRzKTtcbiAgbGV0IGV4cG9ydGluZ0ZyYW1lQ2xpcFBhdGggPSBcIlwiO1xuXG4gIGZvciAoY29uc3QgZnJhbWUgb2YgZnJhbWVFbGVtZW50cykge1xuICAgIGNvbnN0IFt4MSwgeTEsIHgyLCB5Ml0gPSBnZXRFbGVtZW50QWJzb2x1dGVDb29yZHMoZnJhbWUpO1xuICAgIGNvbnN0IGN4ID0gKHgyIC0geDEpIC8gMiAtIChmcmFtZS54IC0geDEpO1xuICAgIGNvbnN0IGN5ID0gKHkyIC0geTEpIC8gMiAtIChmcmFtZS55IC0geTEpO1xuICAgIGV4cG9ydGluZ0ZyYW1lQ2xpcFBhdGggKz0gYDxjbGlwUGF0aCBpZD0ke2ZyYW1lLmlkfT5cbiAgICAgICAgICAgIDxyZWN0IHRyYW5zZm9ybT1cInRyYW5zbGF0ZSgke2ZyYW1lLnggKyBvZmZzZXRYfSAke2ZyYW1lLnkgKyBvZmZzZXRZfSkgcm90YXRlKCR7ZnJhbWUuYW5nbGV9ICR7Y3h9ICR7Y3l9KVwiXG4gICAgICAgICAgd2lkdGg9XCIke2ZyYW1lLndpZHRofVwiXG4gICAgICAgICAgaGVpZ2h0PVwiJHtmcmFtZS5oZWlnaHR9XCJcbiAgICAgICAgICA+XG4gICAgICAgICAgPC9yZWN0PlxuICAgICAgICA8L2NsaXBQYXRoPmA7XG4gIH1cblxuICBzdmdSb290LmlubmVySFRNTCA9IGBcbiAgJHtTVkdfRVhQT1JUX1RBR31cbiAgJHttZXRhZGF0YX1cbiAgPGRlZnM+XG4gICAgPHN0eWxlIGNsYXNzPVwic3R5bGUtZm9udHNcIj5cbiAgICAgIEBmb250LWZhY2Uge1xuICAgICAgICBmb250LWZhbWlseTogXCJWaXJnaWxcIjtcbiAgICAgICAgc3JjOiB1cmwoXCIke2Fzc2V0UGF0aH1WaXJnaWwud29mZjJcIik7XG4gICAgICB9XG4gICAgICBAZm9udC1mYWNlIHtcbiAgICAgICAgZm9udC1mYW1pbHk6IFwiQ2FzY2FkaWFcIjtcbiAgICAgICAgc3JjOiB1cmwoXCIke2Fzc2V0UGF0aH1DYXNjYWRpYS53b2ZmMlwiKTtcbiAgICAgIH1cbiAgICAgIEBmb250LWZhY2Uge1xuICAgICAgICBmb250LWZhbWlseTogXCJBc3Npc3RhbnRcIjtcbiAgICAgICAgc3JjOiB1cmwoXCIke2Fzc2V0UGF0aH1Bc3Npc3RhbnQtUmVndWxhci53b2ZmMlwiKTtcbiAgICAgIH1cbiAgICA8L3N0eWxlPlxuICAgICR7ZXhwb3J0aW5nRnJhbWVDbGlwUGF0aH1cbiAgPC9kZWZzPlxuICBgOyAvLyByZW5kZXIgYmFja2dyb3VuZCByZWN0XG5cbiAgaWYgKGFwcFN0YXRlLmV4cG9ydEJhY2tncm91bmQgJiYgdmlld0JhY2tncm91bmRDb2xvcikge1xuICAgIGNvbnN0IHJlY3QgPSBzdmdSb290Lm93bmVyRG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKFNWR19OUywgXCJyZWN0XCIpO1xuICAgIHJlY3Quc2V0QXR0cmlidXRlKFwieFwiLCBcIjBcIik7XG4gICAgcmVjdC5zZXRBdHRyaWJ1dGUoXCJ5XCIsIFwiMFwiKTtcbiAgICByZWN0LnNldEF0dHJpYnV0ZShcIndpZHRoXCIsIGAke3dpZHRofWApO1xuICAgIHJlY3Quc2V0QXR0cmlidXRlKFwiaGVpZ2h0XCIsIGAke2hlaWdodH1gKTtcbiAgICByZWN0LnNldEF0dHJpYnV0ZShcImZpbGxcIiwgdmlld0JhY2tncm91bmRDb2xvcik7XG4gICAgc3ZnUm9vdC5hcHBlbmRDaGlsZChyZWN0KTtcbiAgfVxuXG4gIGNvbnN0IHJzdmcgPSByb3VnaC5zdmcoc3ZnUm9vdCk7XG4gIHJlbmRlclNjZW5lVG9TdmcoZWxlbWVudHNGb3JSZW5kZXIsIHJzdmcsIHN2Z1Jvb3QsIGZpbGVzIHx8IHt9LCB7XG4gICAgb2Zmc2V0WCxcbiAgICBvZmZzZXRZLFxuICAgIGV4cG9ydFdpdGhEYXJrTW9kZSxcbiAgICByZW5kZXJFbWJlZGRhYmxlczogKF9kID0gb3B0cyA9PT0gbnVsbCB8fCBvcHRzID09PSB2b2lkIDAgPyB2b2lkIDAgOiBvcHRzLnJlbmRlckVtYmVkZGFibGVzKSAhPT0gbnVsbCAmJiBfZCAhPT0gdm9pZCAwID8gX2QgOiBmYWxzZSxcbiAgICBmcmFtZVJlbmRlcmluZ1xuICB9KTtcbiAgdGVtcFNjZW5lLmRlc3Ryb3koKTtcbiAgcmV0dXJuIHN2Z1Jvb3Q7XG59KTsgLy8gY2FsY3VsYXRlIHNtYWxsZXN0IGFyZWEgdG8gZml0IHRoZSBjb250ZW50cyBpblxuXG5jb25zdCBnZXRDYW52YXNTaXplID0gKGVsZW1lbnRzLCBleHBvcnRQYWRkaW5nKSA9PiB7XG4gIGNvbnN0IFttaW5YLCBtaW5ZLCBtYXhYLCBtYXhZXSA9IGdldENvbW1vbkJvdW5kcyhlbGVtZW50cyk7XG4gIGNvbnN0IHdpZHRoID0gZGlzdGFuY2UobWluWCwgbWF4WCkgKyBleHBvcnRQYWRkaW5nICogMjtcbiAgY29uc3QgaGVpZ2h0ID0gZGlzdGFuY2UobWluWSwgbWF4WSkgKyBleHBvcnRQYWRkaW5nICogMjtcbiAgcmV0dXJuIFttaW5YLCBtaW5ZLCB3aWR0aCwgaGVpZ2h0XTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRFeHBvcnRTaXplID0gKGVsZW1lbnRzLCBleHBvcnRQYWRkaW5nLCBzY2FsZSkgPT4ge1xuICBjb25zdCBbLCwgd2lkdGgsIGhlaWdodF0gPSBnZXRDYW52YXNTaXplKGVsZW1lbnRzLCBleHBvcnRQYWRkaW5nKS5tYXAoZGltZW5zaW9uID0+IE1hdGgudHJ1bmMoZGltZW5zaW9uICogc2NhbGUpKTtcbiAgcmV0dXJuIFt3aWR0aCwgaGVpZ2h0XTtcbn07Il0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///../../scene/export.ts\n");
4524
4524
 
4525
4525
  /***/ }),
4526
4526
 
@@ -5693,7 +5693,7 @@ module.exports = JSON.parse('{"ar-SA":99,"az-AZ":18,"bg-BG":75,"bn-BD":55,"ca-ES
5693
5693
  /******/ // This function allow to reference async chunks
5694
5694
  /******/ __webpack_require__.u = (chunkId) => {
5695
5695
  /******/ // return url for filenames based on template
5696
- /******/ return "excalidraw-assets-dev/" + chunkId + "-" + {"locales/ar-SA-json":"f35c4f87e6e2dcf7c20a","locales/az-AZ-json":"5d70eb8bf3f20abc3bac","locales/bg-BG-json":"88cced5fd8a6d8298501","locales/bn-BD-json":"a06588c61947851c8579","locales/ca-ES-json":"9b25933a25836cc0be23","locales/cs-CZ-json":"1d6c97ea271d017058c9","locales/da-DK-json":"34a60c9843cecd71376f","locales/de-DE-json":"d13450595b795867412b","locales/el-GR-json":"c854b199f3ac07d40294","locales/es-ES-json":"6a667bfbf4ff3d0181a7","locales/eu-ES-json":"3ec11367a80491b09056","locales/fa-IR-json":"e459964936177074abe7","locales/fi-FI-json":"f7839bb6f5ad0a3c7e0d","locales/fr-FR-json":"20e8535e2675a6736c81","locales/gl-ES-json":"861f534b46c0db1fa8db","locales/he-IL-json":"504d78736487793ffccd","locales/hi-IN-json":"82d988431011c330242a","locales/hu-HU-json":"d4150250980011726fd9","locales/id-ID-json":"82e300d4fe1e87adba9b","locales/it-IT-json":"6ecc9aec005faab90f41","locales/ja-JP-json":"3c6a065f0f1303b297fa","locales/kaa-json":"983a9ccc652aa01980f3","locales/kab-KAB-json":"cac3cf66f1db5c1a6e19","locales/kk-KZ-json":"e9dd81c22419efd44478","locales/km-KH-json":"be77acee611d96d88dda","locales/ko-KR-json":"ad8e9d31d52b26f3fbc2","locales/ku-TR-json":"ed67af91ae1920f114a8","locales/lt-LT-json":"2c3d35d6fb5dbf95a27e","locales/lv-LV-json":"fa8973c231afb2ddafe9","locales/mr-IN-json":"2949146743072eb11b40","locales/my-MM-json":"4c04ffe415641f69ed75","locales/nb-NO-json":"b8d7a5b70562dacdad45","locales/nl-NL-json":"a65f82a25fe038c45d35","locales/nn-NO-json":"858fbccbc5be386977db","locales/oc-FR-json":"a68ea08272c8da3c0e3f","locales/pa-IN-json":"90a2b3775bd7d8983def","locales/pl-PL-json":"9bb55330d5aaf336646e","locales/pt-BR-json":"3635e753c1b6e5b681fa","locales/pt-PT-json":"0d3694a92e0134549086","locales/ro-RO-json":"e83fda16c860c6d0b383","locales/ru-RU-json":"ddc13261ce5d864d36a9","locales/si-LK-json":"16eb66dbfc6c55fc85b1","locales/sk-SK-json":"782ead8707f2ad0e8e4e","locales/sl-SI-json":"82e55cf7cdbdc7d7f959","locales/sv-SE-json":"2f362899d3ac4089534f","locales/ta-IN-json":"07623a485202da63a84b","locales/th-TH-json":"4e4b97f5f6e905191383","locales/tr-TR-json":"003be1cf6ebf0b787dec","locales/uk-UA-json":"ca6ea1156db2649d3e27","locales/vi-VN-json":"9a5e5fab41a1a120a916","locales/zh-CN-json":"670c28a6ae1e3ddecaa4","locales/zh-HK-json":"93b9b676d4f4b5702797","locales/zh-TW-json":"805d10b0eed9ca51b318","vendor":"24fff00bcf5a7372c15c"}[chunkId] + ".js";
5696
+ /******/ return "excalidraw-assets-dev/" + chunkId + "-" + {"locales/ar-SA-json":"f35c4f87e6e2dcf7c20a","locales/az-AZ-json":"5d70eb8bf3f20abc3bac","locales/bg-BG-json":"88cced5fd8a6d8298501","locales/bn-BD-json":"a06588c61947851c8579","locales/ca-ES-json":"9b25933a25836cc0be23","locales/cs-CZ-json":"1d6c97ea271d017058c9","locales/da-DK-json":"34a60c9843cecd71376f","locales/de-DE-json":"d13450595b795867412b","locales/el-GR-json":"c854b199f3ac07d40294","locales/es-ES-json":"6a667bfbf4ff3d0181a7","locales/eu-ES-json":"3ec11367a80491b09056","locales/fa-IR-json":"e459964936177074abe7","locales/fi-FI-json":"f7839bb6f5ad0a3c7e0d","locales/fr-FR-json":"20e8535e2675a6736c81","locales/gl-ES-json":"861f534b46c0db1fa8db","locales/he-IL-json":"504d78736487793ffccd","locales/hi-IN-json":"82d988431011c330242a","locales/hu-HU-json":"d4150250980011726fd9","locales/id-ID-json":"82e300d4fe1e87adba9b","locales/it-IT-json":"6ecc9aec005faab90f41","locales/ja-JP-json":"3c6a065f0f1303b297fa","locales/kaa-json":"983a9ccc652aa01980f3","locales/kab-KAB-json":"cac3cf66f1db5c1a6e19","locales/kk-KZ-json":"e9dd81c22419efd44478","locales/km-KH-json":"be77acee611d96d88dda","locales/ko-KR-json":"ad8e9d31d52b26f3fbc2","locales/ku-TR-json":"ed67af91ae1920f114a8","locales/lt-LT-json":"2c3d35d6fb5dbf95a27e","locales/lv-LV-json":"fa8973c231afb2ddafe9","locales/mr-IN-json":"2949146743072eb11b40","locales/my-MM-json":"4c04ffe415641f69ed75","locales/nb-NO-json":"b8d7a5b70562dacdad45","locales/nl-NL-json":"a65f82a25fe038c45d35","locales/nn-NO-json":"858fbccbc5be386977db","locales/oc-FR-json":"a68ea08272c8da3c0e3f","locales/pa-IN-json":"90a2b3775bd7d8983def","locales/pl-PL-json":"9bb55330d5aaf336646e","locales/pt-BR-json":"3635e753c1b6e5b681fa","locales/pt-PT-json":"0d3694a92e0134549086","locales/ro-RO-json":"e83fda16c860c6d0b383","locales/ru-RU-json":"ddc13261ce5d864d36a9","locales/si-LK-json":"16eb66dbfc6c55fc85b1","locales/sk-SK-json":"782ead8707f2ad0e8e4e","locales/sl-SI-json":"82e55cf7cdbdc7d7f959","locales/sv-SE-json":"2f362899d3ac4089534f","locales/ta-IN-json":"07623a485202da63a84b","locales/th-TH-json":"4e4b97f5f6e905191383","locales/tr-TR-json":"003be1cf6ebf0b787dec","locales/uk-UA-json":"ca6ea1156db2649d3e27","locales/vi-VN-json":"9a5e5fab41a1a120a916","locales/zh-CN-json":"670c28a6ae1e3ddecaa4","locales/zh-HK-json":"93b9b676d4f4b5702797","locales/zh-TW-json":"805d10b0eed9ca51b318","vendor":"d4c6c37cb71cd8cde431"}[chunkId] + ".js";
5697
5697
  /******/ };
5698
5698
  /******/ })();
5699
5699
  /******/