@excalidraw/excalidraw 0.17.1-7391-063c61e → 0.17.1-7391-e65bf53

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.
@@ -2485,7 +2485,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
2485
2485
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
2486
2486
 
2487
2487
  "use strict";
2488
- eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"trackEvent\": () => (/* binding */ trackEvent)\n/* harmony export */ });\n// place here categories that you want to track. We want to track just a\n// small subset of categories at a given time.\nconst ALLOWED_CATEGORIES_TO_TRACK = [\"ai\"];\nconst trackEvent = (category, action, label, value) => {\n try {\n // prettier-ignore\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_AI_BACKEND\":\"http://localhost:3015\",\"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-7391-063c61e\",\"VITE_IS_EXCALIDRAW_NPM_PACKAGE\":true}).VITE_WORKER_ID // comment out to debug locally\n || \"development\" === \"production\") {\n return;\n }\n\n if (!ALLOWED_CATEGORIES_TO_TRACK.includes(category)) {\n return;\n }\n\n if (true) {\n console.info(\"trackEvent\", {\n category,\n action,\n label,\n value\n });\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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi4vLi4vYW5hbHl0aWNzLnRzLmpzIiwibWFwcGluZ3MiOiI7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDTztBQUNQO0FBQ0E7QUFDQSx5Q0FBeUMsaXVDQUFXO0FBQ3BELE9BQU8sYUFBb0I7QUFDM0I7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUEsUUFBUSxJQUF3QztBQUNoRDtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsT0FBTztBQUNQOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxPQUFPO0FBQ1A7QUFDQSxJQUFJO0FBQ0o7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vLy4uLy4uL2FuYWx5dGljcy50cz81NDIxIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIHBsYWNlIGhlcmUgY2F0ZWdvcmllcyB0aGF0IHlvdSB3YW50IHRvIHRyYWNrLiBXZSB3YW50IHRvIHRyYWNrIGp1c3QgYVxuLy8gc21hbGwgc3Vic2V0IG9mIGNhdGVnb3JpZXMgYXQgYSBnaXZlbiB0aW1lLlxuY29uc3QgQUxMT1dFRF9DQVRFR09SSUVTX1RPX1RSQUNLID0gW1wiYWlcIl07XG5leHBvcnQgY29uc3QgdHJhY2tFdmVudCA9IChjYXRlZ29yeSwgYWN0aW9uLCBsYWJlbCwgdmFsdWUpID0+IHtcbiAgdHJ5IHtcbiAgICAvLyBwcmV0dGllci1pZ25vcmVcbiAgICBpZiAodHlwZW9mIHdpbmRvdyA9PT0gXCJ1bmRlZmluZWRcIiB8fCBwcm9jZXNzLmVudi5WSVRFX1dPUktFUl9JRCAvLyBjb21tZW50IG91dCB0byBkZWJ1ZyBsb2NhbGx5XG4gICAgfHwgcHJvY2Vzcy5lbnYuTk9ERV9FTlYgPT09IFwicHJvZHVjdGlvblwiKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKCFBTExPV0VEX0NBVEVHT1JJRVNfVE9fVFJBQ0suaW5jbHVkZXMoY2F0ZWdvcnkpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKCEocHJvY2Vzcy5lbnYuTk9ERV9FTlYgPT09IFwicHJvZHVjdGlvblwiKSkge1xuICAgICAgY29uc29sZS5pbmZvKFwidHJhY2tFdmVudFwiLCB7XG4gICAgICAgIGNhdGVnb3J5LFxuICAgICAgICBhY3Rpb24sXG4gICAgICAgIGxhYmVsLFxuICAgICAgICB2YWx1ZVxuICAgICAgfSk7XG4gICAgfVxuXG4gICAgaWYgKHdpbmRvdy5zYV9ldmVudCkge1xuICAgICAgd2luZG93LnNhX2V2ZW50KGFjdGlvbiwge1xuICAgICAgICBjYXRlZ29yeSxcbiAgICAgICAgbGFiZWwsXG4gICAgICAgIHZhbHVlXG4gICAgICB9KTtcbiAgICB9XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgY29uc29sZS5lcnJvcihcImVycm9yIGR1cmluZyBhbmFseXRpY3NcIiwgZXJyb3IpO1xuICB9XG59OyJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///../../analytics.ts\n");
2488
+ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"trackEvent\": () => (/* binding */ trackEvent)\n/* harmony export */ });\n// place here categories that you want to track. We want to track just a\n// small subset of categories at a given time.\nconst ALLOWED_CATEGORIES_TO_TRACK = [\"ai\"];\nconst trackEvent = (category, action, label, value) => {\n try {\n // prettier-ignore\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_AI_BACKEND\":\"http://localhost:3015\",\"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-7391-e65bf53\",\"VITE_IS_EXCALIDRAW_NPM_PACKAGE\":true}).VITE_WORKER_ID // comment out to debug locally\n || \"development\" === \"production\") {\n return;\n }\n\n if (!ALLOWED_CATEGORIES_TO_TRACK.includes(category)) {\n return;\n }\n\n if (true) {\n console.info(\"trackEvent\", {\n category,\n action,\n label,\n value\n });\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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi4vLi4vYW5hbHl0aWNzLnRzLmpzIiwibWFwcGluZ3MiOiI7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDTztBQUNQO0FBQ0E7QUFDQSx5Q0FBeUMsaXVDQUFXO0FBQ3BELE9BQU8sYUFBb0I7QUFDM0I7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUEsUUFBUSxJQUF3QztBQUNoRDtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsT0FBTztBQUNQOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxPQUFPO0FBQ1A7QUFDQSxJQUFJO0FBQ0o7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vLy4uLy4uL2FuYWx5dGljcy50cz81NDIxIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIHBsYWNlIGhlcmUgY2F0ZWdvcmllcyB0aGF0IHlvdSB3YW50IHRvIHRyYWNrLiBXZSB3YW50IHRvIHRyYWNrIGp1c3QgYVxuLy8gc21hbGwgc3Vic2V0IG9mIGNhdGVnb3JpZXMgYXQgYSBnaXZlbiB0aW1lLlxuY29uc3QgQUxMT1dFRF9DQVRFR09SSUVTX1RPX1RSQUNLID0gW1wiYWlcIl07XG5leHBvcnQgY29uc3QgdHJhY2tFdmVudCA9IChjYXRlZ29yeSwgYWN0aW9uLCBsYWJlbCwgdmFsdWUpID0+IHtcbiAgdHJ5IHtcbiAgICAvLyBwcmV0dGllci1pZ25vcmVcbiAgICBpZiAodHlwZW9mIHdpbmRvdyA9PT0gXCJ1bmRlZmluZWRcIiB8fCBwcm9jZXNzLmVudi5WSVRFX1dPUktFUl9JRCAvLyBjb21tZW50IG91dCB0byBkZWJ1ZyBsb2NhbGx5XG4gICAgfHwgcHJvY2Vzcy5lbnYuTk9ERV9FTlYgPT09IFwicHJvZHVjdGlvblwiKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKCFBTExPV0VEX0NBVEVHT1JJRVNfVE9fVFJBQ0suaW5jbHVkZXMoY2F0ZWdvcnkpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKCEocHJvY2Vzcy5lbnYuTk9ERV9FTlYgPT09IFwicHJvZHVjdGlvblwiKSkge1xuICAgICAgY29uc29sZS5pbmZvKFwidHJhY2tFdmVudFwiLCB7XG4gICAgICAgIGNhdGVnb3J5LFxuICAgICAgICBhY3Rpb24sXG4gICAgICAgIGxhYmVsLFxuICAgICAgICB2YWx1ZVxuICAgICAgfSk7XG4gICAgfVxuXG4gICAgaWYgKHdpbmRvdy5zYV9ldmVudCkge1xuICAgICAgd2luZG93LnNhX2V2ZW50KGFjdGlvbiwge1xuICAgICAgICBjYXRlZ29yeSxcbiAgICAgICAgbGFiZWwsXG4gICAgICAgIHZhbHVlXG4gICAgICB9KTtcbiAgICB9XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgY29uc29sZS5lcnJvcihcImVycm9yIGR1cmluZyBhbmFseXRpY3NcIiwgZXJyb3IpO1xuICB9XG59OyJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///../../analytics.ts\n");
2489
2489
 
2490
2490
  /***/ }),
2491
2491
 
@@ -4157,7 +4157,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
4157
4157
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
4158
4158
 
4159
4159
  "use strict";
4160
- eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"ElementBounds\": () => (/* binding */ ElementBounds),\n/* harmony export */ \"getArrowheadAngle\": () => (/* binding */ getArrowheadAngle),\n/* harmony export */ \"getArrowheadPoints\": () => (/* binding */ getArrowheadPoints),\n/* harmony export */ \"getArrowheadSize\": () => (/* binding */ getArrowheadSize),\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/* harmony import */ var _scene_Scene__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ../scene/Scene */ \"../../scene/Scene.ts\");\n\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); // hack to ensure that downstream checks could retrieve element Scene\n // so as to have correctly calculated bounds\n // FIXME remove when we get rid of all the id:Scene / element:Scene mapping\n\n const shouldCache = _scene_Scene__WEBPACK_IMPORTED_MODULE_8__[\"default\"].getScene(element);\n\n if (shouldCache) {\n ElementBounds.boundsCache.set(element, {\n version: element.version,\n bounds\n });\n }\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/** @returns number in pixels */\n\n\nconst getArrowheadSize = arrowhead => {\n switch (arrowhead) {\n case \"arrow\":\n case \"uml_arrow\":\n return 25;\n\n case \"uml_diamond\":\n case \"uml_diamond_filled\":\n return 12;\n\n default:\n return 15;\n }\n};\n/** @returns number in degrees */\n\nconst getArrowheadAngle = arrowhead => {\n switch (arrowhead) {\n case \"bar\":\n return 90;\n\n case \"arrow\":\n case \"uml_arrow\":\n return 20;\n\n default:\n return 25;\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 = getArrowheadSize(arrowhead);\n let length = 0;\n {\n // Length for -> arrows is based on the length of the last section\n const [cx, cy] = position === \"end\" ? element.points[element.points.length - 1] : element.points[0];\n const [px, py] = element.points.length > 1 ? position === \"end\" ? element.points[element.points.length - 2] : element.points[1] : [0, 0];\n length = Math.hypot(cx - px, cy - py);\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 const lengthMultiplier = arrowhead === \"uml_diamond\" || arrowhead === \"uml_diamond_filled\" ? 0.25 : 0.5;\n const minSize = Math.min(size, length * lengthMultiplier);\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 = getArrowheadAngle(arrowhead); // 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\n if (arrowhead === \"uml_diamond\" || arrowhead === \"uml_diamond_filled\") {\n // point opposite to the arrowhead point\n let ox;\n let oy;\n\n if (position === \"start\") {\n const [px, py] = element.points.length > 1 ? element.points[1] : [0, 0];\n [ox, oy] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x2 + minSize * 2, y2, x2, y2, Math.atan2(py - y2, px - x2));\n } else {\n const [px, py] = element.points.length > 1 ? element.points[element.points.length - 2] : [0, 0];\n [ox, oy] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x2 - minSize * 2, y2, x2, y2, Math.atan2(y2 - py, x2 - px));\n }\n\n return [x2, y2, x3, y3, ox, oy, x4, y4];\n }\n\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,{"version":3,"file":"../../element/bounds.ts.js","mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAA0D;AACpB;AACgB;AAC+D;AAC3E;AAC+B;AACb;AACX;AACd;AAC5B;AACP;AACA;;AAEA;AACA;AACA,KAAK,+DAAkB;AACvB;AACA;;AAEA,2DAA2D;AAC3D;AACA;;AAEA,wBAAwB,6DAAc;;AAEtC;AACA;AACA;AACA;AACA,OAAO;AACP;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA,QAAQ,8DAAiB;AACzB,0FAA0F,6CAAM;AAChG;AACA,MAAM,SAAS,4DAAe;AAC9B;AACA,MAAM;AACN,yBAAyB,6CAAM;AAC/B,yBAAyB,6CAAM;AAC/B,yBAAyB,6CAAM;AAC/B,yBAAyB,6CAAM;AAC/B;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN,yBAAyB,6CAAM;AAC/B,yBAAyB,6CAAM;AAC/B,yBAAyB,6CAAM;AAC/B,yBAAyB,6CAAM;AAC/B;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,2CAA2C;AAC3C;AACA;AACA;;AAEO;AACP,MAAM,8DAAiB;AACvB;AACA,IAAI,SAAS,4DAAe;AAC5B,WAAW,8FAA4C;AACvD,IAAI,SAAS,0DAAa;AAC1B,sBAAsB,iEAAmB;;AAEzC,QAAQ,2DAAc;AACtB,qBAAqB,iGAA+C;AACpE;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;;AAEA,MAAM,4DAAe,aAAa,8DAAiB;AACnD;AACA;;AAEA;AACA,qBAAqB,kDAAW,+FAA+F,kDAAW;AAC1I;AACA;;AAEA;AACA;;AAEA,qIAAqI,kDAAW;;AAEhJ;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;;AAEA;AACA,GAAG;;AAEH;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,IAAI;AACJ;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA,uBAAuB,0CAA0C;AACjE;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAM,2BAA2B;AACjC,MAAM,6BAA6B;AACnC;;AAEA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACO;AACP;;AAEA;AACA;AACA,IAAI;;;AAGJ;AACA;AACA;AACA;AACA,iCAAiC;AACjC;AACA,YAAY;;AAEZ;AACA;;AAEA;AACA;AACA,IAAI;AACJ;AACA,IAAI;;;AAGJ,wKAAwK;;;AAGxK,mDAAmD;AACnD;AACA;AACA;;AAEA,yDAAyD;AACzD;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,8CAA8C;;AAE9C,mBAAmB,6CAAM;AACzB,mBAAmB,6CAAM;;AAEzB;AACA;AACA;AACA;;AAEA;AACA;AACA,iBAAiB,6CAAM;AACvB,MAAM;AACN;AACA,iBAAiB,6CAAM;AACvB;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA,oBAAoB,mEAAe;AACnC,kBAAkB,kEAAoB;;AAEtC;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,GAAG;;AAEH;AACA;;AAEA;AACA;;AAEA;AACA;AACA,mBAAmB,6CAAM;AACzB;AACA,6BAA6B,iEAAmB;;AAEhD;AACA,kCAAkC,8FAA4C;AAC9E;AACA;;AAEA;AACA,IAAI;;;AAGJ,4BAA4B,6DAAc;AAC1C;AACA;;AAEA,gCAAgC,6CAAM;;AAEtC;AACA;AACA,2BAA2B,iEAAmB;;AAE9C;AACA,gCAAgC,8FAA4C;AAC5E;AACA;;AAEA;AACA;;AAEO;AACP;AACA;AACO;AACP;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACO;AACP;AACA;AACA;AACO;AACP,QAAQ,4DAAe,aAAa,8DAAiB;AACrD;AACA;;AAEA,iBAAiB,sDAAa,eAAe,sDAAa;AAC1D;;AAEA,MAAM,8DAAiB;AACvB;AACA;AACA,IAAI;AACJ;AACA,gBAAgB,mEAAe;AAC/B,8DAA8D,kEAAoB,+BAA+B,kEAAoB;AACrI;AACA;AACA;;AAEA;AACA;AACA;AACO;AACP;AACA,cAAc,mEAAe;AAC7B,mEAAmE,kEAAoB,+BAA+B,kEAAoB;AAC1I;AACA;AACA;AACA;AACO;AACP;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,qBAAqB,iDAAU;;AAE/B;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sources":["webpack:///../../element/bounds.ts?fca0"],"sourcesContent":["import { distance2d, rotate, rotatePoint } from \"../math\";\nimport rough from \"roughjs/bin/rough\";\nimport { generateRoughOptions } from \"../scene/Shape\";\nimport { isArrowElement, isBoundToContainer, isFreeDrawElement, isLinearElement, isTextElement } from \"./typeChecks\";\nimport { rescalePoints } from \"../points\";\nimport { getBoundTextElement, getContainerElement } from \"./textElement\";\nimport { LinearElementEditor } from \"./linearElementEditor\";\nimport { ShapeCache } from \"../scene/ShapeCache\";\nimport Scene from \"../scene/Scene\";\nexport class 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    !isBoundToContainer(element)) {\n      return cachedBounds.bounds;\n    }\n\n    const bounds = ElementBounds.calculateBounds(element); // hack to ensure that downstream checks could retrieve element Scene\n    // so as to have correctly calculated bounds\n    // FIXME remove when we get rid of all the id:Scene / element:Scene mapping\n\n    const shouldCache = Scene.getScene(element);\n\n    if (shouldCache) {\n      ElementBounds.boundsCache.set(element, {\n        version: element.version,\n        bounds\n      });\n    }\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 (isFreeDrawElement(element)) {\n      const [minX, minY, maxX, maxY] = getBoundsFromPoints(element.points.map(([x, y]) => 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 (isLinearElement(element)) {\n      bounds = getLinearElementRotatedBounds(element, cx, cy);\n    } else if (element.type === \"diamond\") {\n      const [x11, y11] = rotate(cx, y1, cx, cy, element.angle);\n      const [x12, y12] = rotate(cx, y2, cx, cy, element.angle);\n      const [x22, y22] = rotate(x1, cy, cx, cy, element.angle);\n      const [x21, y21] = 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] = rotate(x1, y1, cx, cy, element.angle);\n      const [x12, y12] = rotate(x1, y2, cx, cy, element.angle);\n      const [x22, y22] = rotate(x2, y2, cx, cy, element.angle);\n      const [x21, y21] = 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\nexport const getElementAbsoluteCoords = (element, includeBoundText = false) => {\n  if (isFreeDrawElement(element)) {\n    return getFreeDrawElementAbsoluteCoords(element);\n  } else if (isLinearElement(element)) {\n    return LinearElementEditor.getElementAbsoluteCoords(element, includeBoundText);\n  } else if (isTextElement(element)) {\n    const container = getContainerElement(element);\n\n    if (isArrowElement(container)) {\n      const coords = 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\nexport const getElementLineSegments = element => {\n  const [x1, y1, x2, y2, cx, cy] = getElementAbsoluteCoords(element);\n  const center = [cx, cy];\n\n  if (isLinearElement(element) || isFreeDrawElement(element)) {\n    const segments = [];\n    let i = 0;\n\n    while (i < element.points.length - 1) {\n      segments.push([rotatePoint([element.points[i][0] + element.x, element.points[i][1] + element.y], center, element.angle), 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 => 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\nexport const 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};\nexport const pointRelativeTo = (element, absoluteCoords) => {\n  return [absoluteCoords[0] - element.x, absoluteCoords[1] - element.y];\n};\nexport const 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};\nexport const 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\nexport const 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};\nexport const 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/** @returns number in pixels */\n\n\nexport const getArrowheadSize = arrowhead => {\n  switch (arrowhead) {\n    case \"arrow\":\n    case \"uml_arrow\":\n      return 25;\n\n    case \"uml_diamond\":\n    case \"uml_diamond_filled\":\n      return 12;\n\n    default:\n      return 15;\n  }\n};\n/** @returns number in degrees */\n\nexport const getArrowheadAngle = arrowhead => {\n  switch (arrowhead) {\n    case \"bar\":\n      return 90;\n\n    case \"arrow\":\n    case \"uml_arrow\":\n      return 20;\n\n    default:\n      return 25;\n  }\n};\nexport const 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 = getArrowheadSize(arrowhead);\n  let length = 0;\n  {\n    // Length for -> arrows is based on the length of the last section\n    const [cx, cy] = position === \"end\" ? element.points[element.points.length - 1] : element.points[0];\n    const [px, py] = element.points.length > 1 ? position === \"end\" ? element.points[element.points.length - 2] : element.points[1] : [0, 0];\n    length = Math.hypot(cx - px, cy - py);\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  const lengthMultiplier = arrowhead === \"uml_diamond\" || arrowhead === \"uml_diamond_filled\" ? 0.25 : 0.5;\n  const minSize = Math.min(size, length * lengthMultiplier);\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 = getArrowheadAngle(arrowhead); // Return points\n\n  const [x3, y3] = rotate(xs, ys, x2, y2, -angle * Math.PI / 180);\n  const [x4, y4] = rotate(xs, ys, x2, y2, angle * Math.PI / 180);\n\n  if (arrowhead === \"uml_diamond\" || arrowhead === \"uml_diamond_filled\") {\n    // point opposite to the arrowhead point\n    let ox;\n    let oy;\n\n    if (position === \"start\") {\n      const [px, py] = element.points.length > 1 ? element.points[1] : [0, 0];\n      [ox, oy] = rotate(x2 + minSize * 2, y2, x2, y2, Math.atan2(py - y2, px - x2));\n    } else {\n      const [px, py] = element.points.length > 1 ? element.points[element.points.length - 2] : [0, 0];\n      [ox, oy] = rotate(x2 - minSize * 2, y2, x2, y2, Math.atan2(y2 - py, x2 - px));\n    }\n\n    return [x2, y2, x3, y3, ox, oy, x4, y4];\n  }\n\n  return [x2, y2, x3, y3, x4, y4];\n};\n\nconst generateLinearElementShape = element => {\n  const generator = rough.generator();\n  const options = 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] = rotate(element.x + pointX, element.y + pointY, cx, cy, element.angle);\n    let coords = [x, y, x, y];\n    const boundTextElement = getBoundTextElement(element);\n\n    if (boundTextElement) {\n      const coordsWithBoundText = 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 = 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) => 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 = getBoundTextElement(element);\n\n  if (boundTextElement) {\n    const coordsWithBoundText = LinearElementEditor.getMinMaxXYWithBoundText(element, coords, boundTextElement);\n    coords = [coordsWithBoundText[0], coordsWithBoundText[1], coordsWithBoundText[2], coordsWithBoundText[3]];\n  }\n\n  return coords;\n};\n\nexport const getElementBounds = element => {\n  return ElementBounds.getBounds(element);\n};\nexport const 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};\nexport const 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};\nexport const getResizedElementAbsoluteCoords = (element, nextWidth, nextHeight, normalizePoints) => {\n  if (!(isLinearElement(element) || isFreeDrawElement(element))) {\n    return [element.x, element.y, element.x + nextWidth, element.y + nextHeight];\n  }\n\n  const points = rescalePoints(0, nextWidth, rescalePoints(1, nextHeight, element.points, normalizePoints), normalizePoints);\n  let bounds;\n\n  if (isFreeDrawElement(element)) {\n    // Free Draw\n    bounds = getBoundsFromPoints(points);\n  } else {\n    // Line\n    const gen = rough.generator();\n    const curve = !element.roundness ? gen.linearPath(points, generateRoughOptions(element)) : gen.curve(points, 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};\nexport const getElementPointsCoords = (element, points) => {\n  // This might be computationally heavey\n  const gen = rough.generator();\n  const curve = element.roundness == null ? gen.linearPath(points, generateRoughOptions(element)) : gen.curve(points, 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};\nexport const 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 = 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};\nexport const 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};"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///../../element/bounds.ts\n");
4160
+ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"ElementBounds\": () => (/* binding */ ElementBounds),\n/* harmony export */ \"getArrowheadAngle\": () => (/* binding */ getArrowheadAngle),\n/* harmony export */ \"getArrowheadPoints\": () => (/* binding */ getArrowheadPoints),\n/* harmony export */ \"getArrowheadSize\": () => (/* binding */ getArrowheadSize),\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/* harmony import */ var _scene_Scene__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ../scene/Scene */ \"../../scene/Scene.ts\");\n\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); // hack to ensure that downstream checks could retrieve element Scene\n // so as to have correctly calculated bounds\n // FIXME remove when we get rid of all the id:Scene / element:Scene mapping\n\n const shouldCache = _scene_Scene__WEBPACK_IMPORTED_MODULE_8__[\"default\"].getScene(element);\n\n if (shouldCache) {\n ElementBounds.boundsCache.set(element, {\n version: element.version,\n bounds\n });\n }\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/** @returns number in pixels */\n\n\nconst getArrowheadSize = arrowhead => {\n switch (arrowhead) {\n case \"arrow\":\n case \"d_arrow\":\n return 25;\n\n case \"d_diamond\":\n case \"d_diamond_filled\":\n return 12;\n\n default:\n return 15;\n }\n};\n/** @returns number in degrees */\n\nconst getArrowheadAngle = arrowhead => {\n switch (arrowhead) {\n case \"bar\":\n return 90;\n\n case \"arrow\":\n case \"d_arrow\":\n return 20;\n\n default:\n return 25;\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 = getArrowheadSize(arrowhead);\n let length = 0;\n {\n // Length for -> arrows is based on the length of the last section\n const [cx, cy] = position === \"end\" ? element.points[element.points.length - 1] : element.points[0];\n const [px, py] = element.points.length > 1 ? position === \"end\" ? element.points[element.points.length - 2] : element.points[1] : [0, 0];\n length = Math.hypot(cx - px, cy - py);\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 const lengthMultiplier = arrowhead === \"d_diamond\" || arrowhead === \"d_diamond_filled\" ? 0.25 : 0.5;\n const minSize = Math.min(size, length * lengthMultiplier);\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 = getArrowheadAngle(arrowhead); // 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\n if (arrowhead === \"d_diamond\" || arrowhead === \"d_diamond_filled\") {\n // point opposite to the arrowhead point\n let ox;\n let oy;\n\n if (position === \"start\") {\n const [px, py] = element.points.length > 1 ? element.points[1] : [0, 0];\n [ox, oy] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x2 + minSize * 2, y2, x2, y2, Math.atan2(py - y2, px - x2));\n } else {\n const [px, py] = element.points.length > 1 ? element.points[element.points.length - 2] : [0, 0];\n [ox, oy] = (0,_math__WEBPACK_IMPORTED_MODULE_0__.rotate)(x2 - minSize * 2, y2, x2, y2, Math.atan2(y2 - py, x2 - px));\n }\n\n return [x2, y2, x3, y3, ox, oy, x4, y4];\n }\n\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,{"version":3,"file":"../../element/bounds.ts.js","mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAA0D;AACpB;AACgB;AAC+D;AAC3E;AAC+B;AACb;AACX;AACd;AAC5B;AACP;AACA;;AAEA;AACA;AACA,KAAK,+DAAkB;AACvB;AACA;;AAEA,2DAA2D;AAC3D;AACA;;AAEA,wBAAwB,6DAAc;;AAEtC;AACA;AACA;AACA;AACA,OAAO;AACP;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA,QAAQ,8DAAiB;AACzB,0FAA0F,6CAAM;AAChG;AACA,MAAM,SAAS,4DAAe;AAC9B;AACA,MAAM;AACN,yBAAyB,6CAAM;AAC/B,yBAAyB,6CAAM;AAC/B,yBAAyB,6CAAM;AAC/B,yBAAyB,6CAAM;AAC/B;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN,yBAAyB,6CAAM;AAC/B,yBAAyB,6CAAM;AAC/B,yBAAyB,6CAAM;AAC/B,yBAAyB,6CAAM;AAC/B;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,2CAA2C;AAC3C;AACA;AACA;;AAEO;AACP,MAAM,8DAAiB;AACvB;AACA,IAAI,SAAS,4DAAe;AAC5B,WAAW,8FAA4C;AACvD,IAAI,SAAS,0DAAa;AAC1B,sBAAsB,iEAAmB;;AAEzC,QAAQ,2DAAc;AACtB,qBAAqB,iGAA+C;AACpE;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;;AAEA,MAAM,4DAAe,aAAa,8DAAiB;AACnD;AACA;;AAEA;AACA,qBAAqB,kDAAW,+FAA+F,kDAAW;AAC1I;AACA;;AAEA;AACA;;AAEA,qIAAqI,kDAAW;;AAEhJ;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;;AAEA;AACA,GAAG;;AAEH;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,IAAI;AACJ;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA,uBAAuB,0CAA0C;AACjE;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAM,2BAA2B;AACjC,MAAM,6BAA6B;AACnC;;AAEA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACO;AACP;;AAEA;AACA;AACA,IAAI;;;AAGJ;AACA;AACA;AACA;AACA,iCAAiC;AACjC;AACA,YAAY;;AAEZ;AACA;;AAEA;AACA;AACA,IAAI;AACJ;AACA,IAAI;;;AAGJ,wKAAwK;;;AAGxK,mDAAmD;AACnD;AACA;AACA;;AAEA,yDAAyD;AACzD;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,8CAA8C;;AAE9C,mBAAmB,6CAAM;AACzB,mBAAmB,6CAAM;;AAEzB;AACA;AACA;AACA;;AAEA;AACA;AACA,iBAAiB,6CAAM;AACvB,MAAM;AACN;AACA,iBAAiB,6CAAM;AACvB;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA,oBAAoB,mEAAe;AACnC,kBAAkB,kEAAoB;;AAEtC;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,GAAG;;AAEH;AACA;;AAEA;AACA;;AAEA;AACA;AACA,mBAAmB,6CAAM;AACzB;AACA,6BAA6B,iEAAmB;;AAEhD;AACA,kCAAkC,8FAA4C;AAC9E;AACA;;AAEA;AACA,IAAI;;;AAGJ,4BAA4B,6DAAc;AAC1C;AACA;;AAEA,gCAAgC,6CAAM;;AAEtC;AACA;AACA,2BAA2B,iEAAmB;;AAE9C;AACA,gCAAgC,8FAA4C;AAC5E;AACA;;AAEA;AACA;;AAEO;AACP;AACA;AACO;AACP;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACO;AACP;AACA;AACA;AACO;AACP,QAAQ,4DAAe,aAAa,8DAAiB;AACrD;AACA;;AAEA,iBAAiB,sDAAa,eAAe,sDAAa;AAC1D;;AAEA,MAAM,8DAAiB;AACvB;AACA;AACA,IAAI;AACJ;AACA,gBAAgB,mEAAe;AAC/B,8DAA8D,kEAAoB,+BAA+B,kEAAoB;AACrI;AACA;AACA;;AAEA;AACA;AACA;AACO;AACP;AACA,cAAc,mEAAe;AAC7B,mEAAmE,kEAAoB,+BAA+B,kEAAoB;AAC1I;AACA;AACA;AACA;AACO;AACP;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,qBAAqB,iDAAU;;AAE/B;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sources":["webpack:///../../element/bounds.ts?fca0"],"sourcesContent":["import { distance2d, rotate, rotatePoint } from \"../math\";\nimport rough from \"roughjs/bin/rough\";\nimport { generateRoughOptions } from \"../scene/Shape\";\nimport { isArrowElement, isBoundToContainer, isFreeDrawElement, isLinearElement, isTextElement } from \"./typeChecks\";\nimport { rescalePoints } from \"../points\";\nimport { getBoundTextElement, getContainerElement } from \"./textElement\";\nimport { LinearElementEditor } from \"./linearElementEditor\";\nimport { ShapeCache } from \"../scene/ShapeCache\";\nimport Scene from \"../scene/Scene\";\nexport class 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    !isBoundToContainer(element)) {\n      return cachedBounds.bounds;\n    }\n\n    const bounds = ElementBounds.calculateBounds(element); // hack to ensure that downstream checks could retrieve element Scene\n    // so as to have correctly calculated bounds\n    // FIXME remove when we get rid of all the id:Scene / element:Scene mapping\n\n    const shouldCache = Scene.getScene(element);\n\n    if (shouldCache) {\n      ElementBounds.boundsCache.set(element, {\n        version: element.version,\n        bounds\n      });\n    }\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 (isFreeDrawElement(element)) {\n      const [minX, minY, maxX, maxY] = getBoundsFromPoints(element.points.map(([x, y]) => 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 (isLinearElement(element)) {\n      bounds = getLinearElementRotatedBounds(element, cx, cy);\n    } else if (element.type === \"diamond\") {\n      const [x11, y11] = rotate(cx, y1, cx, cy, element.angle);\n      const [x12, y12] = rotate(cx, y2, cx, cy, element.angle);\n      const [x22, y22] = rotate(x1, cy, cx, cy, element.angle);\n      const [x21, y21] = 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] = rotate(x1, y1, cx, cy, element.angle);\n      const [x12, y12] = rotate(x1, y2, cx, cy, element.angle);\n      const [x22, y22] = rotate(x2, y2, cx, cy, element.angle);\n      const [x21, y21] = 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\nexport const getElementAbsoluteCoords = (element, includeBoundText = false) => {\n  if (isFreeDrawElement(element)) {\n    return getFreeDrawElementAbsoluteCoords(element);\n  } else if (isLinearElement(element)) {\n    return LinearElementEditor.getElementAbsoluteCoords(element, includeBoundText);\n  } else if (isTextElement(element)) {\n    const container = getContainerElement(element);\n\n    if (isArrowElement(container)) {\n      const coords = 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\nexport const getElementLineSegments = element => {\n  const [x1, y1, x2, y2, cx, cy] = getElementAbsoluteCoords(element);\n  const center = [cx, cy];\n\n  if (isLinearElement(element) || isFreeDrawElement(element)) {\n    const segments = [];\n    let i = 0;\n\n    while (i < element.points.length - 1) {\n      segments.push([rotatePoint([element.points[i][0] + element.x, element.points[i][1] + element.y], center, element.angle), 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 => 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\nexport const 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};\nexport const pointRelativeTo = (element, absoluteCoords) => {\n  return [absoluteCoords[0] - element.x, absoluteCoords[1] - element.y];\n};\nexport const 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};\nexport const 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\nexport const 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};\nexport const 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/** @returns number in pixels */\n\n\nexport const getArrowheadSize = arrowhead => {\n  switch (arrowhead) {\n    case \"arrow\":\n    case \"d_arrow\":\n      return 25;\n\n    case \"d_diamond\":\n    case \"d_diamond_filled\":\n      return 12;\n\n    default:\n      return 15;\n  }\n};\n/** @returns number in degrees */\n\nexport const getArrowheadAngle = arrowhead => {\n  switch (arrowhead) {\n    case \"bar\":\n      return 90;\n\n    case \"arrow\":\n    case \"d_arrow\":\n      return 20;\n\n    default:\n      return 25;\n  }\n};\nexport const 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 = getArrowheadSize(arrowhead);\n  let length = 0;\n  {\n    // Length for -> arrows is based on the length of the last section\n    const [cx, cy] = position === \"end\" ? element.points[element.points.length - 1] : element.points[0];\n    const [px, py] = element.points.length > 1 ? position === \"end\" ? element.points[element.points.length - 2] : element.points[1] : [0, 0];\n    length = Math.hypot(cx - px, cy - py);\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  const lengthMultiplier = arrowhead === \"d_diamond\" || arrowhead === \"d_diamond_filled\" ? 0.25 : 0.5;\n  const minSize = Math.min(size, length * lengthMultiplier);\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 = getArrowheadAngle(arrowhead); // Return points\n\n  const [x3, y3] = rotate(xs, ys, x2, y2, -angle * Math.PI / 180);\n  const [x4, y4] = rotate(xs, ys, x2, y2, angle * Math.PI / 180);\n\n  if (arrowhead === \"d_diamond\" || arrowhead === \"d_diamond_filled\") {\n    // point opposite to the arrowhead point\n    let ox;\n    let oy;\n\n    if (position === \"start\") {\n      const [px, py] = element.points.length > 1 ? element.points[1] : [0, 0];\n      [ox, oy] = rotate(x2 + minSize * 2, y2, x2, y2, Math.atan2(py - y2, px - x2));\n    } else {\n      const [px, py] = element.points.length > 1 ? element.points[element.points.length - 2] : [0, 0];\n      [ox, oy] = rotate(x2 - minSize * 2, y2, x2, y2, Math.atan2(y2 - py, x2 - px));\n    }\n\n    return [x2, y2, x3, y3, ox, oy, x4, y4];\n  }\n\n  return [x2, y2, x3, y3, x4, y4];\n};\n\nconst generateLinearElementShape = element => {\n  const generator = rough.generator();\n  const options = 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] = rotate(element.x + pointX, element.y + pointY, cx, cy, element.angle);\n    let coords = [x, y, x, y];\n    const boundTextElement = getBoundTextElement(element);\n\n    if (boundTextElement) {\n      const coordsWithBoundText = 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 = 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) => 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 = getBoundTextElement(element);\n\n  if (boundTextElement) {\n    const coordsWithBoundText = LinearElementEditor.getMinMaxXYWithBoundText(element, coords, boundTextElement);\n    coords = [coordsWithBoundText[0], coordsWithBoundText[1], coordsWithBoundText[2], coordsWithBoundText[3]];\n  }\n\n  return coords;\n};\n\nexport const getElementBounds = element => {\n  return ElementBounds.getBounds(element);\n};\nexport const 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};\nexport const 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};\nexport const getResizedElementAbsoluteCoords = (element, nextWidth, nextHeight, normalizePoints) => {\n  if (!(isLinearElement(element) || isFreeDrawElement(element))) {\n    return [element.x, element.y, element.x + nextWidth, element.y + nextHeight];\n  }\n\n  const points = rescalePoints(0, nextWidth, rescalePoints(1, nextHeight, element.points, normalizePoints), normalizePoints);\n  let bounds;\n\n  if (isFreeDrawElement(element)) {\n    // Free Draw\n    bounds = getBoundsFromPoints(points);\n  } else {\n    // Line\n    const gen = rough.generator();\n    const curve = !element.roundness ? gen.linearPath(points, generateRoughOptions(element)) : gen.curve(points, 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};\nexport const getElementPointsCoords = (element, points) => {\n  // This might be computationally heavey\n  const gen = rough.generator();\n  const curve = element.roundness == null ? gen.linearPath(points, generateRoughOptions(element)) : gen.curve(points, 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};\nexport const 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 = 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};\nexport const 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};"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///../../element/bounds.ts\n");
4161
4161
 
4162
4162
  /***/ }),
4163
4163
 
@@ -4630,7 +4630,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
4630
4630
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
4631
4631
 
4632
4632
  "use strict";
4633
- 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-7391-063c61e\"}/dist/`;\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9wdWJsaWNQYXRoLmpzLmpzIiwibWFwcGluZ3MiOiI7O0FBQXNDOztBQUV0QyxJQUFJLGFBQW9CLEtBQUssZ0RBQVE7QUFDckM7O0FBRUE7QUFDQSxFQUFFLHFCQUF1Qix3REFBd0Qsd0JBQXlCLENBQUMsR0FBRyxxQkFBNEIsQ0FBQztBQUMzSSIsInNvdXJjZXMiOlsid2VicGFjazovLy8uL3B1YmxpY1BhdGguanM/NjdmOSJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBFTlYgfSBmcm9tIFwiLi4vLi4vY29uc3RhbnRzXCI7XG5cbmlmIChwcm9jZXNzLmVudi5OT0RFX0VOViAhPT0gRU5WLlRFU1QpIHtcbiAgLyogZXNsaW50LWRpc2FibGUgKi9cblxuICAvKiBnbG9iYWwgX193ZWJwYWNrX3B1YmxpY19wYXRoX186d3JpdGFibGUgKi9cbiAgX193ZWJwYWNrX3B1YmxpY19wYXRoX18gPSB3aW5kb3cuRVhDQUxJRFJBV19BU1NFVF9QQVRIIHx8IGBodHRwczovL3VucGtnLmNvbS8ke3Byb2Nlc3MuZW52LlZJVEVfUEtHX05BTUV9QCR7cHJvY2Vzcy5lbnYuVklURV9QS0dfVkVSU0lPTn0vZGlzdC9gO1xufSJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./publicPath.js\n");
4633
+ 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-7391-e65bf53\"}/dist/`;\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9wdWJsaWNQYXRoLmpzLmpzIiwibWFwcGluZ3MiOiI7O0FBQXNDOztBQUV0QyxJQUFJLGFBQW9CLEtBQUssZ0RBQVE7QUFDckM7O0FBRUE7QUFDQSxFQUFFLHFCQUF1Qix3REFBd0Qsd0JBQXlCLENBQUMsR0FBRyxxQkFBNEIsQ0FBQztBQUMzSSIsInNvdXJjZXMiOlsid2VicGFjazovLy8uL3B1YmxpY1BhdGguanM/NjdmOSJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBFTlYgfSBmcm9tIFwiLi4vLi4vY29uc3RhbnRzXCI7XG5cbmlmIChwcm9jZXNzLmVudi5OT0RFX0VOViAhPT0gRU5WLlRFU1QpIHtcbiAgLyogZXNsaW50LWRpc2FibGUgKi9cblxuICAvKiBnbG9iYWwgX193ZWJwYWNrX3B1YmxpY19wYXRoX186d3JpdGFibGUgKi9cbiAgX193ZWJwYWNrX3B1YmxpY19wYXRoX18gPSB3aW5kb3cuRVhDQUxJRFJBV19BU1NFVF9QQVRIIHx8IGBodHRwczovL3VucGtnLmNvbS8ke3Byb2Nlc3MuZW52LlZJVEVfUEtHX05BTUV9QCR7cHJvY2Vzcy5lbnYuVklURV9QS0dfVkVSU0lPTn0vZGlzdC9gO1xufSJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./publicPath.js\n");
4634
4634
 
4635
4635
  /***/ }),
4636
4636
 
@@ -4773,7 +4773,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
4773
4773
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
4774
4774
 
4775
4775
  "use strict";
4776
- eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"_generateElementShape\": () => (/* binding */ _generateElementShape),\n/* harmony export */ \"generateRoughOptions\": () => (/* binding */ generateRoughOptions)\n/* harmony export */ });\n/* harmony import */ var _element__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../element */ \"../../element/index.ts\");\n/* harmony import */ var _math__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../math */ \"../../math.ts\");\n/* harmony import */ var _renderer_renderElement__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../renderer/renderElement */ \"../../renderer/renderElement.ts\");\n/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../utils */ \"../../utils.ts\");\n/* harmony import */ var points_on_curve__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! points-on-curve */ \"../../../node_modules/points-on-curve/lib/index.js\");\n/* harmony import */ var _constants__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../constants */ \"../../constants.ts\");\n/* harmony import */ var _element_typeChecks__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ../element/typeChecks */ \"../../element/typeChecks.ts\");\n/* harmony import */ var _comparisons__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ./comparisons */ \"../../scene/comparisons.ts\");\n\n\n\n\n\n\n\n\n\nconst getDashArrayDashed = strokeWidth => [8, 8 + strokeWidth];\n\nconst getDashArrayDotted = strokeWidth => [1.5, 6 + strokeWidth];\n\nfunction adjustRoughness(element) {\n const roughness = element.roughness;\n const maxSize = Math.max(element.width, element.height);\n const minSize = Math.min(element.width, element.height); // don't reduce roughness if\n\n if ( // both sides relatively big\n minSize >= 20 && maxSize >= 50 || // is round & both sides above 15px\n minSize >= 15 && !!element.roundness && (0,_comparisons__WEBPACK_IMPORTED_MODULE_7__.canChangeRoundness)(element.type) || // relatively long linear element\n (0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_6__.isLinearElement)(element) && maxSize >= 50) {\n return roughness;\n }\n\n return Math.min(roughness / (maxSize < 10 ? 3 : 2), 2.5);\n}\n\nconst generateRoughOptions = (element, continuousPath = false) => {\n const options = {\n seed: element.seed,\n strokeLineDash: element.strokeStyle === \"dashed\" ? getDashArrayDashed(element.strokeWidth) : element.strokeStyle === \"dotted\" ? getDashArrayDotted(element.strokeWidth) : undefined,\n // for non-solid strokes, disable multiStroke because it tends to make\n // dashes/dots overlay each other\n disableMultiStroke: element.strokeStyle !== \"solid\",\n // for non-solid strokes, increase the width a bit to make it visually\n // similar to solid strokes, because we're also disabling multiStroke\n strokeWidth: element.strokeStyle !== \"solid\" ? element.strokeWidth + 0.5 : element.strokeWidth,\n // when increasing strokeWidth, we must explicitly set fillWeight and\n // hachureGap because if not specified, roughjs uses strokeWidth to\n // calculate them (and we don't want the fills to be modified)\n fillWeight: element.strokeWidth / 2,\n hachureGap: element.strokeWidth * 4,\n roughness: adjustRoughness(element),\n stroke: element.strokeColor,\n preserveVertices: continuousPath || element.roughness < _constants__WEBPACK_IMPORTED_MODULE_5__.ROUGHNESS.cartoonist\n };\n\n switch (element.type) {\n case \"rectangle\":\n case \"iframe\":\n case \"embeddable\":\n case \"diamond\":\n case \"ellipse\":\n {\n options.fillStyle = element.fillStyle;\n options.fill = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isTransparent)(element.backgroundColor) ? undefined : element.backgroundColor;\n\n if (element.type === \"ellipse\") {\n options.curveFitting = 1;\n }\n\n return options;\n }\n\n case \"line\":\n case \"freedraw\":\n {\n if ((0,_math__WEBPACK_IMPORTED_MODULE_1__.isPathALoop)(element.points)) {\n options.fillStyle = element.fillStyle;\n options.fill = element.backgroundColor === \"transparent\" ? undefined : element.backgroundColor;\n }\n\n return options;\n }\n\n case \"arrow\":\n return options;\n\n default:\n {\n throw new Error(`Unimplemented type ${element.type}`);\n }\n }\n};\n\nconst modifyIframeLikeForRoughOptions = (element, isExporting) => {\n if ((0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_6__.isIframeLikeElement)(element) && (isExporting || (0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_6__.isEmbeddableElement)(element) && !element.validated) && (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isTransparent)(element.backgroundColor) && (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isTransparent)(element.strokeColor)) {\n return Object.assign(Object.assign({}, element), {\n roughness: 0,\n backgroundColor: \"#d3d3d3\",\n fillStyle: \"solid\"\n });\n } else if ((0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_6__.isIframeElement)(element)) {\n return Object.assign(Object.assign({}, element), {\n strokeColor: (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isTransparent)(element.strokeColor) ? \"#000000\" : element.strokeColor,\n backgroundColor: (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isTransparent)(element.backgroundColor) ? \"#f4f4f6\" : element.backgroundColor\n });\n }\n\n return element;\n};\n\nconst getArrowheadShapes = (element, shape, position, arrowhead, generator, options, canvasBackgroundColor) => {\n const arrowheadPoints = (0,_element__WEBPACK_IMPORTED_MODULE_0__.getArrowheadPoints)(element, shape, position, arrowhead);\n\n if (arrowheadPoints === null) {\n return [];\n }\n\n switch (arrowhead) {\n case \"dot\":\n {\n const [x, y, r] = arrowheadPoints;\n return [generator.circle(x, y, r, Object.assign(Object.assign({}, options), {\n fill: element.strokeColor,\n fillStyle: \"solid\",\n stroke: \"none\"\n }))];\n }\n\n case \"uml_triangle\":\n case \"triangle\":\n {\n const [x, y, x2, y2, x3, y3] = arrowheadPoints; // always use solid stroke for triangle arrowhead\n\n delete options.strokeLineDash;\n return [generator.polygon([[x, y], [x2, y2], [x3, y3], [x, y]], Object.assign(Object.assign({}, options), {\n fill: element.strokeColor,\n fillStyle: \"solid\"\n }))];\n }\n\n case \"uml_diamond\":\n case \"uml_diamond_filled\":\n {\n const [x, y, x2, y2, x3, y3, x4, y4] = arrowheadPoints; // always use solid stroke for triangle arrowhead\n\n delete options.strokeLineDash;\n return [generator.polygon([[x, y], [x2, y2], [x3, y3], [x4, y4], [x, y]], Object.assign(Object.assign({}, options), {\n // fill: element.strokeColor,\n fill: arrowhead === \"uml_diamond\" ? canvasBackgroundColor : element.strokeColor,\n fillStyle: \"solid\",\n roughness: Math.min(1, options.roughness || 0)\n }))];\n }\n\n case \"bar\":\n case \"arrow\":\n case \"uml_arrow\":\n default:\n {\n const [x2, y2, x3, y3, x4, y4] = arrowheadPoints;\n\n if (element.strokeStyle === \"dotted\") {\n // for dotted arrows caps, reduce gap to make it more legible\n const dash = getDashArrayDotted(element.strokeWidth - 1);\n options.strokeLineDash = [dash[0], dash[1] - 1];\n } else {\n // for solid/dashed, keep solid arrow cap\n delete options.strokeLineDash;\n }\n\n return [generator.line(x3, y3, x2, y2, options), generator.line(x4, y4, x2, y2, options)];\n }\n }\n};\n/**\n * Generates the roughjs shape for given element.\n *\n * Low-level. Use `ShapeCache.generateElementShape` instead.\n *\n * @private\n */\n\n\nconst _generateElementShape = (element, generator, {\n isExporting,\n canvasBackgroundColor\n}) => {\n switch (element.type) {\n case \"rectangle\":\n case \"iframe\":\n case \"embeddable\":\n {\n let shape; // this is for rendering the stroke/bg of the embeddable, especially\n // when the src url is not set\n\n if (element.roundness) {\n const w = element.width;\n const h = element.height;\n const r = (0,_math__WEBPACK_IMPORTED_MODULE_1__.getCornerRadius)(Math.min(w, h), element);\n shape = generator.path(`M ${r} 0 L ${w - r} 0 Q ${w} 0, ${w} ${r} L ${w} ${h - r} Q ${w} ${h}, ${w - r} ${h} L ${r} ${h} Q 0 ${h}, 0 ${h - r} L 0 ${r} Q 0 0, ${r} 0`, generateRoughOptions(modifyIframeLikeForRoughOptions(element, isExporting), true));\n } else {\n shape = generator.rectangle(0, 0, element.width, element.height, generateRoughOptions(modifyIframeLikeForRoughOptions(element, isExporting), false));\n }\n\n return shape;\n }\n\n case \"diamond\":\n {\n let shape;\n const [topX, topY, rightX, rightY, bottomX, bottomY, leftX, leftY] = (0,_element__WEBPACK_IMPORTED_MODULE_0__.getDiamondPoints)(element);\n\n if (element.roundness) {\n const verticalRadius = (0,_math__WEBPACK_IMPORTED_MODULE_1__.getCornerRadius)(Math.abs(topX - leftX), element);\n const horizontalRadius = (0,_math__WEBPACK_IMPORTED_MODULE_1__.getCornerRadius)(Math.abs(rightY - topY), element);\n shape = generator.path(`M ${topX + verticalRadius} ${topY + horizontalRadius} L ${rightX - verticalRadius} ${rightY - horizontalRadius}\n C ${rightX} ${rightY}, ${rightX} ${rightY}, ${rightX - verticalRadius} ${rightY + horizontalRadius}\n L ${bottomX + verticalRadius} ${bottomY - horizontalRadius}\n C ${bottomX} ${bottomY}, ${bottomX} ${bottomY}, ${bottomX - verticalRadius} ${bottomY - horizontalRadius}\n L ${leftX + verticalRadius} ${leftY + horizontalRadius}\n C ${leftX} ${leftY}, ${leftX} ${leftY}, ${leftX + verticalRadius} ${leftY - horizontalRadius}\n L ${topX - verticalRadius} ${topY + horizontalRadius}\n C ${topX} ${topY}, ${topX} ${topY}, ${topX + verticalRadius} ${topY + horizontalRadius}`, generateRoughOptions(element, true));\n } else {\n shape = generator.polygon([[topX, topY], [rightX, rightY], [bottomX, bottomY], [leftX, leftY]], generateRoughOptions(element));\n }\n\n return shape;\n }\n\n case \"ellipse\":\n {\n const shape = generator.ellipse(element.width / 2, element.height / 2, element.width, element.height, generateRoughOptions(element));\n return shape;\n }\n\n case \"line\":\n case \"arrow\":\n {\n let shape;\n const options = generateRoughOptions(element); // points array can be empty in the beginning, so it is important to add\n // initial position to it\n\n const points = element.points.length ? element.points : [[0, 0]]; // curve is always the first element\n // this simplifies finding the curve for an element\n\n if (!element.roundness) {\n if (options.fill) {\n shape = [generator.polygon(points, options)];\n } else {\n shape = [generator.linearPath(points, options)];\n }\n } else {\n shape = [generator.curve(points, options)];\n } // add lines only in arrow\n\n\n if (element.type === \"arrow\") {\n const {\n startArrowhead = null,\n endArrowhead = \"arrow\"\n } = element;\n\n if (startArrowhead !== null) {\n const shapes = getArrowheadShapes(element, shape, \"start\", startArrowhead, generator, options, canvasBackgroundColor);\n shape.push(...shapes);\n }\n\n if (endArrowhead !== null) {\n if (endArrowhead === undefined) {// Hey, we have an old arrow here!\n }\n\n const shapes = getArrowheadShapes(element, shape, \"end\", endArrowhead, generator, options, canvasBackgroundColor);\n shape.push(...shapes);\n }\n }\n\n return shape;\n }\n\n case \"freedraw\":\n {\n let shape;\n (0,_renderer_renderElement__WEBPACK_IMPORTED_MODULE_2__.generateFreeDrawShape)(element);\n\n if ((0,_math__WEBPACK_IMPORTED_MODULE_1__.isPathALoop)(element.points)) {\n // generate rough polygon to fill freedraw shape\n const simplifiedPoints = (0,points_on_curve__WEBPACK_IMPORTED_MODULE_4__.simplify)(element.points, 0.75);\n shape = generator.curve(simplifiedPoints, Object.assign(Object.assign({}, generateRoughOptions(element)), {\n stroke: \"none\"\n }));\n } else {\n shape = null;\n }\n\n return shape;\n }\n\n case \"frame\":\n case \"magicframe\":\n case \"text\":\n case \"image\":\n {\n const shape = null; // we return (and cache) `null` to make sure we don't regenerate\n // `element.canvas` on rerenders\n\n return shape;\n }\n\n default:\n {\n (0,_utils__WEBPACK_IMPORTED_MODULE_3__.assertNever)(element, `generateElementShape(): Unimplemented type ${element === null || element === void 0 ? void 0 : element.type}`);\n return null;\n }\n }\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"../../scene/Shape.ts.js","mappings":";;;;;;;;;;;;;AAAkE;AACX;AACW;AACZ;AACX;AACF;AAC0E;AAChE;;AAEnD;;AAEA;;AAEA;AACA;AACA;AACA,2DAA2D;;AAE3D;AACA;AACA,0CAA0C,gEAAkB;AAC5D,EAAE,oEAAe;AACjB;AACA;;AAEA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4DAA4D,4DAAoB;AAChF;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,qDAAa;;AAEpC;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,YAAY,kDAAW;AACvB;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA,8CAA8C,aAAa;AAC3D;AACA;AACA;;AAEA;AACA,MAAM,wEAAmB,6BAA6B,wEAAmB,oCAAoC,qDAAa,6BAA6B,qDAAa;AACpK,yCAAyC;AACzC;AACA;AACA;AACA,KAAK;AACL,IAAI,SAAS,oEAAe;AAC5B,yCAAyC;AACzC,mBAAmB,qDAAa;AAChC,uBAAuB,qDAAa;AACpC,KAAK;AACL;;AAEA;AACA;;AAEA;AACA,0BAA0B,4DAAkB;;AAE5C;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,wEAAwE;AACxE;AACA;AACA;AACA,SAAS;AACT;;AAEA;AACA;AACA;AACA,wDAAwD;;AAExD;AACA,sGAAsG;AACtG;AACA;AACA,SAAS;AACT;;AAEA;AACA;AACA;AACA,gEAAgE;;AAEhE;AACA,gHAAgH;AAChH;AACA;AACA;AACA;AACA,SAAS;AACT;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGO;AACP;AACA;AACA,CAAC;AACD;AACA;AACA;AACA;AACA;AACA,mBAAmB;AACnB;;AAEA;AACA;AACA;AACA,oBAAoB,sDAAe;AACnC,sCAAsC,GAAG,MAAM,OAAO,MAAM,GAAG,KAAK,GAAG,EAAE,GAAG,IAAI,GAAG,EAAE,OAAO,IAAI,GAAG,EAAE,EAAE,IAAI,OAAO,EAAE,GAAG,IAAI,GAAG,EAAE,GAAG,MAAM,EAAE,MAAM,OAAO,MAAM,GAAG,SAAS,GAAG;AAC7K,UAAU;AACV;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,6EAA6E,0DAAgB;;AAE7F;AACA,iCAAiC,sDAAe;AAChD,mCAAmC,sDAAe;AAClD,sCAAsC,uBAAuB,EAAE,yBAAyB,IAAI,yBAAyB,EAAE;AACvH,gBAAgB,QAAQ,EAAE,OAAO,IAAI,QAAQ,EAAE,OAAO,IAAI,yBAAyB,EAAE;AACrF,gBAAgB,0BAA0B,EAAE;AAC5C,gBAAgB,SAAS,EAAE,QAAQ,IAAI,SAAS,EAAE,QAAQ,IAAI,0BAA0B,EAAE;AAC1F,gBAAgB,wBAAwB,EAAE;AAC1C,gBAAgB,OAAO,EAAE,MAAM,IAAI,OAAO,EAAE,MAAM,IAAI,wBAAwB,EAAE;AAChF,gBAAgB,uBAAuB,EAAE;AACzC,gBAAgB,MAAM,EAAE,KAAK,IAAI,MAAM,EAAE,KAAK,IAAI,uBAAuB,EAAE,wBAAwB;AACnG,UAAU;AACV;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,uDAAuD;AACvD;;AAEA,0EAA0E;AAC1E;;AAEA;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA,UAAU;AACV;AACA,UAAU;;;AAGV;AACA;AACA;AACA;AACA,YAAY;;AAEZ;AACA;AACA;AACA;;AAEA;AACA,6CAA6C;AAC7C;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,QAAQ,8EAAqB;;AAE7B,YAAY,kDAAW;AACvB;AACA,mCAAmC,yDAAQ;AAC3C,kFAAkF;AAClF;AACA,WAAW;AACX,UAAU;AACV;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,4BAA4B;AAC5B;;AAEA;AACA;;AAEA;AACA;AACA,QAAQ,mDAAW,wDAAwD,+DAA+D;AAC1I;AACA;AACA;AACA","sources":["webpack:///../../scene/Shape.ts?a74e"],"sourcesContent":["import { getDiamondPoints, getArrowheadPoints } from \"../element\";\nimport { isPathALoop, getCornerRadius } from \"../math\";\nimport { generateFreeDrawShape } from \"../renderer/renderElement\";\nimport { isTransparent, assertNever } from \"../utils\";\nimport { simplify } from \"points-on-curve\";\nimport { ROUGHNESS } from \"../constants\";\nimport { isEmbeddableElement, isIframeElement, isIframeLikeElement, isLinearElement } from \"../element/typeChecks\";\nimport { canChangeRoundness } from \"./comparisons\";\n\nconst getDashArrayDashed = strokeWidth => [8, 8 + strokeWidth];\n\nconst getDashArrayDotted = strokeWidth => [1.5, 6 + strokeWidth];\n\nfunction adjustRoughness(element) {\n  const roughness = element.roughness;\n  const maxSize = Math.max(element.width, element.height);\n  const minSize = Math.min(element.width, element.height); // don't reduce roughness if\n\n  if ( // both sides relatively big\n  minSize >= 20 && maxSize >= 50 || // is round & both sides above 15px\n  minSize >= 15 && !!element.roundness && canChangeRoundness(element.type) || // relatively long linear element\n  isLinearElement(element) && maxSize >= 50) {\n    return roughness;\n  }\n\n  return Math.min(roughness / (maxSize < 10 ? 3 : 2), 2.5);\n}\n\nexport const generateRoughOptions = (element, continuousPath = false) => {\n  const options = {\n    seed: element.seed,\n    strokeLineDash: element.strokeStyle === \"dashed\" ? getDashArrayDashed(element.strokeWidth) : element.strokeStyle === \"dotted\" ? getDashArrayDotted(element.strokeWidth) : undefined,\n    // for non-solid strokes, disable multiStroke because it tends to make\n    // dashes/dots overlay each other\n    disableMultiStroke: element.strokeStyle !== \"solid\",\n    // for non-solid strokes, increase the width a bit to make it visually\n    // similar to solid strokes, because we're also disabling multiStroke\n    strokeWidth: element.strokeStyle !== \"solid\" ? element.strokeWidth + 0.5 : element.strokeWidth,\n    // when increasing strokeWidth, we must explicitly set fillWeight and\n    // hachureGap because if not specified, roughjs uses strokeWidth to\n    // calculate them (and we don't want the fills to be modified)\n    fillWeight: element.strokeWidth / 2,\n    hachureGap: element.strokeWidth * 4,\n    roughness: adjustRoughness(element),\n    stroke: element.strokeColor,\n    preserveVertices: continuousPath || element.roughness < ROUGHNESS.cartoonist\n  };\n\n  switch (element.type) {\n    case \"rectangle\":\n    case \"iframe\":\n    case \"embeddable\":\n    case \"diamond\":\n    case \"ellipse\":\n      {\n        options.fillStyle = element.fillStyle;\n        options.fill = isTransparent(element.backgroundColor) ? undefined : element.backgroundColor;\n\n        if (element.type === \"ellipse\") {\n          options.curveFitting = 1;\n        }\n\n        return options;\n      }\n\n    case \"line\":\n    case \"freedraw\":\n      {\n        if (isPathALoop(element.points)) {\n          options.fillStyle = element.fillStyle;\n          options.fill = element.backgroundColor === \"transparent\" ? undefined : element.backgroundColor;\n        }\n\n        return options;\n      }\n\n    case \"arrow\":\n      return options;\n\n    default:\n      {\n        throw new Error(`Unimplemented type ${element.type}`);\n      }\n  }\n};\n\nconst modifyIframeLikeForRoughOptions = (element, isExporting) => {\n  if (isIframeLikeElement(element) && (isExporting || isEmbeddableElement(element) && !element.validated) && isTransparent(element.backgroundColor) && isTransparent(element.strokeColor)) {\n    return Object.assign(Object.assign({}, element), {\n      roughness: 0,\n      backgroundColor: \"#d3d3d3\",\n      fillStyle: \"solid\"\n    });\n  } else if (isIframeElement(element)) {\n    return Object.assign(Object.assign({}, element), {\n      strokeColor: isTransparent(element.strokeColor) ? \"#000000\" : element.strokeColor,\n      backgroundColor: isTransparent(element.backgroundColor) ? \"#f4f4f6\" : element.backgroundColor\n    });\n  }\n\n  return element;\n};\n\nconst getArrowheadShapes = (element, shape, position, arrowhead, generator, options, canvasBackgroundColor) => {\n  const arrowheadPoints = getArrowheadPoints(element, shape, position, arrowhead);\n\n  if (arrowheadPoints === null) {\n    return [];\n  }\n\n  switch (arrowhead) {\n    case \"dot\":\n      {\n        const [x, y, r] = arrowheadPoints;\n        return [generator.circle(x, y, r, Object.assign(Object.assign({}, options), {\n          fill: element.strokeColor,\n          fillStyle: \"solid\",\n          stroke: \"none\"\n        }))];\n      }\n\n    case \"uml_triangle\":\n    case \"triangle\":\n      {\n        const [x, y, x2, y2, x3, y3] = arrowheadPoints; // always use solid stroke for triangle arrowhead\n\n        delete options.strokeLineDash;\n        return [generator.polygon([[x, y], [x2, y2], [x3, y3], [x, y]], Object.assign(Object.assign({}, options), {\n          fill: element.strokeColor,\n          fillStyle: \"solid\"\n        }))];\n      }\n\n    case \"uml_diamond\":\n    case \"uml_diamond_filled\":\n      {\n        const [x, y, x2, y2, x3, y3, x4, y4] = arrowheadPoints; // always use solid stroke for triangle arrowhead\n\n        delete options.strokeLineDash;\n        return [generator.polygon([[x, y], [x2, y2], [x3, y3], [x4, y4], [x, y]], Object.assign(Object.assign({}, options), {\n          // fill: element.strokeColor,\n          fill: arrowhead === \"uml_diamond\" ? canvasBackgroundColor : element.strokeColor,\n          fillStyle: \"solid\",\n          roughness: Math.min(1, options.roughness || 0)\n        }))];\n      }\n\n    case \"bar\":\n    case \"arrow\":\n    case \"uml_arrow\":\n    default:\n      {\n        const [x2, y2, x3, y3, x4, y4] = arrowheadPoints;\n\n        if (element.strokeStyle === \"dotted\") {\n          // for dotted arrows caps, reduce gap to make it more legible\n          const dash = getDashArrayDotted(element.strokeWidth - 1);\n          options.strokeLineDash = [dash[0], dash[1] - 1];\n        } else {\n          // for solid/dashed, keep solid arrow cap\n          delete options.strokeLineDash;\n        }\n\n        return [generator.line(x3, y3, x2, y2, options), generator.line(x4, y4, x2, y2, options)];\n      }\n  }\n};\n/**\n * Generates the roughjs shape for given element.\n *\n * Low-level. Use `ShapeCache.generateElementShape` instead.\n *\n * @private\n */\n\n\nexport const _generateElementShape = (element, generator, {\n  isExporting,\n  canvasBackgroundColor\n}) => {\n  switch (element.type) {\n    case \"rectangle\":\n    case \"iframe\":\n    case \"embeddable\":\n      {\n        let shape; // this is for rendering the stroke/bg of the embeddable, especially\n        // when the src url is not set\n\n        if (element.roundness) {\n          const w = element.width;\n          const h = element.height;\n          const r = getCornerRadius(Math.min(w, h), element);\n          shape = generator.path(`M ${r} 0 L ${w - r} 0 Q ${w} 0, ${w} ${r} L ${w} ${h - r} Q ${w} ${h}, ${w - r} ${h} L ${r} ${h} Q 0 ${h}, 0 ${h - r} L 0 ${r} Q 0 0, ${r} 0`, generateRoughOptions(modifyIframeLikeForRoughOptions(element, isExporting), true));\n        } else {\n          shape = generator.rectangle(0, 0, element.width, element.height, generateRoughOptions(modifyIframeLikeForRoughOptions(element, isExporting), false));\n        }\n\n        return shape;\n      }\n\n    case \"diamond\":\n      {\n        let shape;\n        const [topX, topY, rightX, rightY, bottomX, bottomY, leftX, leftY] = getDiamondPoints(element);\n\n        if (element.roundness) {\n          const verticalRadius = getCornerRadius(Math.abs(topX - leftX), element);\n          const horizontalRadius = getCornerRadius(Math.abs(rightY - topY), element);\n          shape = generator.path(`M ${topX + verticalRadius} ${topY + horizontalRadius} L ${rightX - verticalRadius} ${rightY - horizontalRadius}\n            C ${rightX} ${rightY}, ${rightX} ${rightY}, ${rightX - verticalRadius} ${rightY + horizontalRadius}\n            L ${bottomX + verticalRadius} ${bottomY - horizontalRadius}\n            C ${bottomX} ${bottomY}, ${bottomX} ${bottomY}, ${bottomX - verticalRadius} ${bottomY - horizontalRadius}\n            L ${leftX + verticalRadius} ${leftY + horizontalRadius}\n            C ${leftX} ${leftY}, ${leftX} ${leftY}, ${leftX + verticalRadius} ${leftY - horizontalRadius}\n            L ${topX - verticalRadius} ${topY + horizontalRadius}\n            C ${topX} ${topY}, ${topX} ${topY}, ${topX + verticalRadius} ${topY + horizontalRadius}`, generateRoughOptions(element, true));\n        } else {\n          shape = generator.polygon([[topX, topY], [rightX, rightY], [bottomX, bottomY], [leftX, leftY]], generateRoughOptions(element));\n        }\n\n        return shape;\n      }\n\n    case \"ellipse\":\n      {\n        const shape = generator.ellipse(element.width / 2, element.height / 2, element.width, element.height, generateRoughOptions(element));\n        return shape;\n      }\n\n    case \"line\":\n    case \"arrow\":\n      {\n        let shape;\n        const options = generateRoughOptions(element); // points array can be empty in the beginning, so it is important to add\n        // initial position to it\n\n        const points = element.points.length ? element.points : [[0, 0]]; // curve is always the first element\n        // this simplifies finding the curve for an element\n\n        if (!element.roundness) {\n          if (options.fill) {\n            shape = [generator.polygon(points, options)];\n          } else {\n            shape = [generator.linearPath(points, options)];\n          }\n        } else {\n          shape = [generator.curve(points, options)];\n        } // add lines only in arrow\n\n\n        if (element.type === \"arrow\") {\n          const {\n            startArrowhead = null,\n            endArrowhead = \"arrow\"\n          } = element;\n\n          if (startArrowhead !== null) {\n            const shapes = getArrowheadShapes(element, shape, \"start\", startArrowhead, generator, options, canvasBackgroundColor);\n            shape.push(...shapes);\n          }\n\n          if (endArrowhead !== null) {\n            if (endArrowhead === undefined) {// Hey, we have an old arrow here!\n            }\n\n            const shapes = getArrowheadShapes(element, shape, \"end\", endArrowhead, generator, options, canvasBackgroundColor);\n            shape.push(...shapes);\n          }\n        }\n\n        return shape;\n      }\n\n    case \"freedraw\":\n      {\n        let shape;\n        generateFreeDrawShape(element);\n\n        if (isPathALoop(element.points)) {\n          // generate rough polygon to fill freedraw shape\n          const simplifiedPoints = simplify(element.points, 0.75);\n          shape = generator.curve(simplifiedPoints, Object.assign(Object.assign({}, generateRoughOptions(element)), {\n            stroke: \"none\"\n          }));\n        } else {\n          shape = null;\n        }\n\n        return shape;\n      }\n\n    case \"frame\":\n    case \"magicframe\":\n    case \"text\":\n    case \"image\":\n      {\n        const shape = null; // we return (and cache) `null` to make sure we don't regenerate\n        // `element.canvas` on rerenders\n\n        return shape;\n      }\n\n    default:\n      {\n        assertNever(element, `generateElementShape(): Unimplemented type ${element === null || element === void 0 ? void 0 : element.type}`);\n        return null;\n      }\n  }\n};"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///../../scene/Shape.ts\n");
4776
+ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"_generateElementShape\": () => (/* binding */ _generateElementShape),\n/* harmony export */ \"generateRoughOptions\": () => (/* binding */ generateRoughOptions)\n/* harmony export */ });\n/* harmony import */ var _element__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../element */ \"../../element/index.ts\");\n/* harmony import */ var _math__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../math */ \"../../math.ts\");\n/* harmony import */ var _renderer_renderElement__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../renderer/renderElement */ \"../../renderer/renderElement.ts\");\n/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../utils */ \"../../utils.ts\");\n/* harmony import */ var points_on_curve__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! points-on-curve */ \"../../../node_modules/points-on-curve/lib/index.js\");\n/* harmony import */ var _constants__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../constants */ \"../../constants.ts\");\n/* harmony import */ var _element_typeChecks__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ../element/typeChecks */ \"../../element/typeChecks.ts\");\n/* harmony import */ var _comparisons__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ./comparisons */ \"../../scene/comparisons.ts\");\n\n\n\n\n\n\n\n\n\nconst getDashArrayDashed = strokeWidth => [8, 8 + strokeWidth];\n\nconst getDashArrayDotted = strokeWidth => [1.5, 6 + strokeWidth];\n\nfunction adjustRoughness(element) {\n const roughness = element.roughness;\n const maxSize = Math.max(element.width, element.height);\n const minSize = Math.min(element.width, element.height); // don't reduce roughness if\n\n if ( // both sides relatively big\n minSize >= 20 && maxSize >= 50 || // is round & both sides above 15px\n minSize >= 15 && !!element.roundness && (0,_comparisons__WEBPACK_IMPORTED_MODULE_7__.canChangeRoundness)(element.type) || // relatively long linear element\n (0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_6__.isLinearElement)(element) && maxSize >= 50) {\n return roughness;\n }\n\n return Math.min(roughness / (maxSize < 10 ? 3 : 2), 2.5);\n}\n\nconst generateRoughOptions = (element, continuousPath = false) => {\n const options = {\n seed: element.seed,\n strokeLineDash: element.strokeStyle === \"dashed\" ? getDashArrayDashed(element.strokeWidth) : element.strokeStyle === \"dotted\" ? getDashArrayDotted(element.strokeWidth) : undefined,\n // for non-solid strokes, disable multiStroke because it tends to make\n // dashes/dots overlay each other\n disableMultiStroke: element.strokeStyle !== \"solid\",\n // for non-solid strokes, increase the width a bit to make it visually\n // similar to solid strokes, because we're also disabling multiStroke\n strokeWidth: element.strokeStyle !== \"solid\" ? element.strokeWidth + 0.5 : element.strokeWidth,\n // when increasing strokeWidth, we must explicitly set fillWeight and\n // hachureGap because if not specified, roughjs uses strokeWidth to\n // calculate them (and we don't want the fills to be modified)\n fillWeight: element.strokeWidth / 2,\n hachureGap: element.strokeWidth * 4,\n roughness: adjustRoughness(element),\n stroke: element.strokeColor,\n preserveVertices: continuousPath || element.roughness < _constants__WEBPACK_IMPORTED_MODULE_5__.ROUGHNESS.cartoonist\n };\n\n switch (element.type) {\n case \"rectangle\":\n case \"iframe\":\n case \"embeddable\":\n case \"diamond\":\n case \"ellipse\":\n {\n options.fillStyle = element.fillStyle;\n options.fill = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isTransparent)(element.backgroundColor) ? undefined : element.backgroundColor;\n\n if (element.type === \"ellipse\") {\n options.curveFitting = 1;\n }\n\n return options;\n }\n\n case \"line\":\n case \"freedraw\":\n {\n if ((0,_math__WEBPACK_IMPORTED_MODULE_1__.isPathALoop)(element.points)) {\n options.fillStyle = element.fillStyle;\n options.fill = element.backgroundColor === \"transparent\" ? undefined : element.backgroundColor;\n }\n\n return options;\n }\n\n case \"arrow\":\n return options;\n\n default:\n {\n throw new Error(`Unimplemented type ${element.type}`);\n }\n }\n};\n\nconst modifyIframeLikeForRoughOptions = (element, isExporting) => {\n if ((0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_6__.isIframeLikeElement)(element) && (isExporting || (0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_6__.isEmbeddableElement)(element) && !element.validated) && (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isTransparent)(element.backgroundColor) && (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isTransparent)(element.strokeColor)) {\n return Object.assign(Object.assign({}, element), {\n roughness: 0,\n backgroundColor: \"#d3d3d3\",\n fillStyle: \"solid\"\n });\n } else if ((0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_6__.isIframeElement)(element)) {\n return Object.assign(Object.assign({}, element), {\n strokeColor: (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isTransparent)(element.strokeColor) ? \"#000000\" : element.strokeColor,\n backgroundColor: (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isTransparent)(element.backgroundColor) ? \"#f4f4f6\" : element.backgroundColor\n });\n }\n\n return element;\n};\n\nconst getArrowheadShapes = (element, shape, position, arrowhead, generator, options, canvasBackgroundColor) => {\n const arrowheadPoints = (0,_element__WEBPACK_IMPORTED_MODULE_0__.getArrowheadPoints)(element, shape, position, arrowhead);\n\n if (arrowheadPoints === null) {\n return [];\n }\n\n switch (arrowhead) {\n case \"dot\":\n {\n const [x, y, r] = arrowheadPoints;\n return [generator.circle(x, y, r, Object.assign(Object.assign({}, options), {\n fill: element.strokeColor,\n fillStyle: \"solid\",\n stroke: \"none\"\n }))];\n }\n\n case \"d_triangle_filled\":\n case \"d_triangle\":\n case \"triangle\":\n {\n const [x, y, x2, y2, x3, y3] = arrowheadPoints; // always use solid stroke for triangle arrowhead\n\n delete options.strokeLineDash;\n return [generator.polygon([[x, y], [x2, y2], [x3, y3], [x, y]], Object.assign(Object.assign({}, options), {\n fill: arrowhead === \"d_triangle\" ? canvasBackgroundColor : element.strokeColor,\n fillStyle: \"solid\",\n roughness: Math.min(1, options.roughness || 0)\n }))];\n }\n\n case \"d_diamond\":\n case \"d_diamond_filled\":\n {\n const [x, y, x2, y2, x3, y3, x4, y4] = arrowheadPoints; // always use solid stroke for triangle arrowhead\n\n delete options.strokeLineDash;\n return [generator.polygon([[x, y], [x2, y2], [x3, y3], [x4, y4], [x, y]], Object.assign(Object.assign({}, options), {\n fill: arrowhead === \"d_diamond\" ? canvasBackgroundColor : element.strokeColor,\n fillStyle: \"solid\",\n roughness: Math.min(1, options.roughness || 0)\n }))];\n }\n\n case \"bar\":\n case \"arrow\":\n case \"d_arrow\":\n default:\n {\n const [x2, y2, x3, y3, x4, y4] = arrowheadPoints;\n\n if (element.strokeStyle === \"dotted\") {\n // for dotted arrows caps, reduce gap to make it more legible\n const dash = getDashArrayDotted(element.strokeWidth - 1);\n options.strokeLineDash = [dash[0], dash[1] - 1];\n } else {\n // for solid/dashed, keep solid arrow cap\n delete options.strokeLineDash;\n }\n\n return [generator.line(x3, y3, x2, y2, options), generator.line(x4, y4, x2, y2, options)];\n }\n }\n};\n/**\n * Generates the roughjs shape for given element.\n *\n * Low-level. Use `ShapeCache.generateElementShape` instead.\n *\n * @private\n */\n\n\nconst _generateElementShape = (element, generator, {\n isExporting,\n canvasBackgroundColor\n}) => {\n switch (element.type) {\n case \"rectangle\":\n case \"iframe\":\n case \"embeddable\":\n {\n let shape; // this is for rendering the stroke/bg of the embeddable, especially\n // when the src url is not set\n\n if (element.roundness) {\n const w = element.width;\n const h = element.height;\n const r = (0,_math__WEBPACK_IMPORTED_MODULE_1__.getCornerRadius)(Math.min(w, h), element);\n shape = generator.path(`M ${r} 0 L ${w - r} 0 Q ${w} 0, ${w} ${r} L ${w} ${h - r} Q ${w} ${h}, ${w - r} ${h} L ${r} ${h} Q 0 ${h}, 0 ${h - r} L 0 ${r} Q 0 0, ${r} 0`, generateRoughOptions(modifyIframeLikeForRoughOptions(element, isExporting), true));\n } else {\n shape = generator.rectangle(0, 0, element.width, element.height, generateRoughOptions(modifyIframeLikeForRoughOptions(element, isExporting), false));\n }\n\n return shape;\n }\n\n case \"diamond\":\n {\n let shape;\n const [topX, topY, rightX, rightY, bottomX, bottomY, leftX, leftY] = (0,_element__WEBPACK_IMPORTED_MODULE_0__.getDiamondPoints)(element);\n\n if (element.roundness) {\n const verticalRadius = (0,_math__WEBPACK_IMPORTED_MODULE_1__.getCornerRadius)(Math.abs(topX - leftX), element);\n const horizontalRadius = (0,_math__WEBPACK_IMPORTED_MODULE_1__.getCornerRadius)(Math.abs(rightY - topY), element);\n shape = generator.path(`M ${topX + verticalRadius} ${topY + horizontalRadius} L ${rightX - verticalRadius} ${rightY - horizontalRadius}\n C ${rightX} ${rightY}, ${rightX} ${rightY}, ${rightX - verticalRadius} ${rightY + horizontalRadius}\n L ${bottomX + verticalRadius} ${bottomY - horizontalRadius}\n C ${bottomX} ${bottomY}, ${bottomX} ${bottomY}, ${bottomX - verticalRadius} ${bottomY - horizontalRadius}\n L ${leftX + verticalRadius} ${leftY + horizontalRadius}\n C ${leftX} ${leftY}, ${leftX} ${leftY}, ${leftX + verticalRadius} ${leftY - horizontalRadius}\n L ${topX - verticalRadius} ${topY + horizontalRadius}\n C ${topX} ${topY}, ${topX} ${topY}, ${topX + verticalRadius} ${topY + horizontalRadius}`, generateRoughOptions(element, true));\n } else {\n shape = generator.polygon([[topX, topY], [rightX, rightY], [bottomX, bottomY], [leftX, leftY]], generateRoughOptions(element));\n }\n\n return shape;\n }\n\n case \"ellipse\":\n {\n const shape = generator.ellipse(element.width / 2, element.height / 2, element.width, element.height, generateRoughOptions(element));\n return shape;\n }\n\n case \"line\":\n case \"arrow\":\n {\n let shape;\n const options = generateRoughOptions(element); // points array can be empty in the beginning, so it is important to add\n // initial position to it\n\n const points = element.points.length ? element.points : [[0, 0]]; // curve is always the first element\n // this simplifies finding the curve for an element\n\n if (!element.roundness) {\n if (options.fill) {\n shape = [generator.polygon(points, options)];\n } else {\n shape = [generator.linearPath(points, options)];\n }\n } else {\n shape = [generator.curve(points, options)];\n } // add lines only in arrow\n\n\n if (element.type === \"arrow\") {\n const {\n startArrowhead = null,\n endArrowhead = \"arrow\"\n } = element;\n\n if (startArrowhead !== null) {\n const shapes = getArrowheadShapes(element, shape, \"start\", startArrowhead, generator, options, canvasBackgroundColor);\n shape.push(...shapes);\n }\n\n if (endArrowhead !== null) {\n if (endArrowhead === undefined) {// Hey, we have an old arrow here!\n }\n\n const shapes = getArrowheadShapes(element, shape, \"end\", endArrowhead, generator, options, canvasBackgroundColor);\n shape.push(...shapes);\n }\n }\n\n return shape;\n }\n\n case \"freedraw\":\n {\n let shape;\n (0,_renderer_renderElement__WEBPACK_IMPORTED_MODULE_2__.generateFreeDrawShape)(element);\n\n if ((0,_math__WEBPACK_IMPORTED_MODULE_1__.isPathALoop)(element.points)) {\n // generate rough polygon to fill freedraw shape\n const simplifiedPoints = (0,points_on_curve__WEBPACK_IMPORTED_MODULE_4__.simplify)(element.points, 0.75);\n shape = generator.curve(simplifiedPoints, Object.assign(Object.assign({}, generateRoughOptions(element)), {\n stroke: \"none\"\n }));\n } else {\n shape = null;\n }\n\n return shape;\n }\n\n case \"frame\":\n case \"magicframe\":\n case \"text\":\n case \"image\":\n {\n const shape = null; // we return (and cache) `null` to make sure we don't regenerate\n // `element.canvas` on rerenders\n\n return shape;\n }\n\n default:\n {\n (0,_utils__WEBPACK_IMPORTED_MODULE_3__.assertNever)(element, `generateElementShape(): Unimplemented type ${element === null || element === void 0 ? void 0 : element.type}`);\n return null;\n }\n }\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"../../scene/Shape.ts.js","mappings":";;;;;;;;;;;;;AAAkE;AACX;AACW;AACZ;AACX;AACF;AAC0E;AAChE;;AAEnD;;AAEA;;AAEA;AACA;AACA;AACA,2DAA2D;;AAE3D;AACA;AACA,0CAA0C,gEAAkB;AAC5D,EAAE,oEAAe;AACjB;AACA;;AAEA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4DAA4D,4DAAoB;AAChF;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,qDAAa;;AAEpC;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,YAAY,kDAAW;AACvB;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA,8CAA8C,aAAa;AAC3D;AACA;AACA;;AAEA;AACA,MAAM,wEAAmB,6BAA6B,wEAAmB,oCAAoC,qDAAa,6BAA6B,qDAAa;AACpK,yCAAyC;AACzC;AACA;AACA;AACA,KAAK;AACL,IAAI,SAAS,oEAAe;AAC5B,yCAAyC;AACzC,mBAAmB,qDAAa;AAChC,uBAAuB,qDAAa;AACpC,KAAK;AACL;;AAEA;AACA;;AAEA;AACA,0BAA0B,4DAAkB;;AAE5C;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,wEAAwE;AACxE;AACA;AACA;AACA,SAAS;AACT;;AAEA;AACA;AACA;AACA;AACA,wDAAwD;;AAExD;AACA,sGAAsG;AACtG;AACA;AACA;AACA,SAAS;AACT;;AAEA;AACA;AACA;AACA,gEAAgE;;AAEhE;AACA,gHAAgH;AAChH;AACA;AACA;AACA,SAAS;AACT;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGO;AACP;AACA;AACA,CAAC;AACD;AACA;AACA;AACA;AACA;AACA,mBAAmB;AACnB;;AAEA;AACA;AACA;AACA,oBAAoB,sDAAe;AACnC,sCAAsC,GAAG,MAAM,OAAO,MAAM,GAAG,KAAK,GAAG,EAAE,GAAG,IAAI,GAAG,EAAE,OAAO,IAAI,GAAG,EAAE,EAAE,IAAI,OAAO,EAAE,GAAG,IAAI,GAAG,EAAE,GAAG,MAAM,EAAE,MAAM,OAAO,MAAM,GAAG,SAAS,GAAG;AAC7K,UAAU;AACV;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,6EAA6E,0DAAgB;;AAE7F;AACA,iCAAiC,sDAAe;AAChD,mCAAmC,sDAAe;AAClD,sCAAsC,uBAAuB,EAAE,yBAAyB,IAAI,yBAAyB,EAAE;AACvH,gBAAgB,QAAQ,EAAE,OAAO,IAAI,QAAQ,EAAE,OAAO,IAAI,yBAAyB,EAAE;AACrF,gBAAgB,0BAA0B,EAAE;AAC5C,gBAAgB,SAAS,EAAE,QAAQ,IAAI,SAAS,EAAE,QAAQ,IAAI,0BAA0B,EAAE;AAC1F,gBAAgB,wBAAwB,EAAE;AAC1C,gBAAgB,OAAO,EAAE,MAAM,IAAI,OAAO,EAAE,MAAM,IAAI,wBAAwB,EAAE;AAChF,gBAAgB,uBAAuB,EAAE;AACzC,gBAAgB,MAAM,EAAE,KAAK,IAAI,MAAM,EAAE,KAAK,IAAI,uBAAuB,EAAE,wBAAwB;AACnG,UAAU;AACV;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,uDAAuD;AACvD;;AAEA,0EAA0E;AAC1E;;AAEA;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA,UAAU;AACV;AACA,UAAU;;;AAGV;AACA;AACA;AACA;AACA,YAAY;;AAEZ;AACA;AACA;AACA;;AAEA;AACA,6CAA6C;AAC7C;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,QAAQ,8EAAqB;;AAE7B,YAAY,kDAAW;AACvB;AACA,mCAAmC,yDAAQ;AAC3C,kFAAkF;AAClF;AACA,WAAW;AACX,UAAU;AACV;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,4BAA4B;AAC5B;;AAEA;AACA;;AAEA;AACA;AACA,QAAQ,mDAAW,wDAAwD,+DAA+D;AAC1I;AACA;AACA;AACA","sources":["webpack:///../../scene/Shape.ts?a74e"],"sourcesContent":["import { getDiamondPoints, getArrowheadPoints } from \"../element\";\nimport { isPathALoop, getCornerRadius } from \"../math\";\nimport { generateFreeDrawShape } from \"../renderer/renderElement\";\nimport { isTransparent, assertNever } from \"../utils\";\nimport { simplify } from \"points-on-curve\";\nimport { ROUGHNESS } from \"../constants\";\nimport { isEmbeddableElement, isIframeElement, isIframeLikeElement, isLinearElement } from \"../element/typeChecks\";\nimport { canChangeRoundness } from \"./comparisons\";\n\nconst getDashArrayDashed = strokeWidth => [8, 8 + strokeWidth];\n\nconst getDashArrayDotted = strokeWidth => [1.5, 6 + strokeWidth];\n\nfunction adjustRoughness(element) {\n  const roughness = element.roughness;\n  const maxSize = Math.max(element.width, element.height);\n  const minSize = Math.min(element.width, element.height); // don't reduce roughness if\n\n  if ( // both sides relatively big\n  minSize >= 20 && maxSize >= 50 || // is round & both sides above 15px\n  minSize >= 15 && !!element.roundness && canChangeRoundness(element.type) || // relatively long linear element\n  isLinearElement(element) && maxSize >= 50) {\n    return roughness;\n  }\n\n  return Math.min(roughness / (maxSize < 10 ? 3 : 2), 2.5);\n}\n\nexport const generateRoughOptions = (element, continuousPath = false) => {\n  const options = {\n    seed: element.seed,\n    strokeLineDash: element.strokeStyle === \"dashed\" ? getDashArrayDashed(element.strokeWidth) : element.strokeStyle === \"dotted\" ? getDashArrayDotted(element.strokeWidth) : undefined,\n    // for non-solid strokes, disable multiStroke because it tends to make\n    // dashes/dots overlay each other\n    disableMultiStroke: element.strokeStyle !== \"solid\",\n    // for non-solid strokes, increase the width a bit to make it visually\n    // similar to solid strokes, because we're also disabling multiStroke\n    strokeWidth: element.strokeStyle !== \"solid\" ? element.strokeWidth + 0.5 : element.strokeWidth,\n    // when increasing strokeWidth, we must explicitly set fillWeight and\n    // hachureGap because if not specified, roughjs uses strokeWidth to\n    // calculate them (and we don't want the fills to be modified)\n    fillWeight: element.strokeWidth / 2,\n    hachureGap: element.strokeWidth * 4,\n    roughness: adjustRoughness(element),\n    stroke: element.strokeColor,\n    preserveVertices: continuousPath || element.roughness < ROUGHNESS.cartoonist\n  };\n\n  switch (element.type) {\n    case \"rectangle\":\n    case \"iframe\":\n    case \"embeddable\":\n    case \"diamond\":\n    case \"ellipse\":\n      {\n        options.fillStyle = element.fillStyle;\n        options.fill = isTransparent(element.backgroundColor) ? undefined : element.backgroundColor;\n\n        if (element.type === \"ellipse\") {\n          options.curveFitting = 1;\n        }\n\n        return options;\n      }\n\n    case \"line\":\n    case \"freedraw\":\n      {\n        if (isPathALoop(element.points)) {\n          options.fillStyle = element.fillStyle;\n          options.fill = element.backgroundColor === \"transparent\" ? undefined : element.backgroundColor;\n        }\n\n        return options;\n      }\n\n    case \"arrow\":\n      return options;\n\n    default:\n      {\n        throw new Error(`Unimplemented type ${element.type}`);\n      }\n  }\n};\n\nconst modifyIframeLikeForRoughOptions = (element, isExporting) => {\n  if (isIframeLikeElement(element) && (isExporting || isEmbeddableElement(element) && !element.validated) && isTransparent(element.backgroundColor) && isTransparent(element.strokeColor)) {\n    return Object.assign(Object.assign({}, element), {\n      roughness: 0,\n      backgroundColor: \"#d3d3d3\",\n      fillStyle: \"solid\"\n    });\n  } else if (isIframeElement(element)) {\n    return Object.assign(Object.assign({}, element), {\n      strokeColor: isTransparent(element.strokeColor) ? \"#000000\" : element.strokeColor,\n      backgroundColor: isTransparent(element.backgroundColor) ? \"#f4f4f6\" : element.backgroundColor\n    });\n  }\n\n  return element;\n};\n\nconst getArrowheadShapes = (element, shape, position, arrowhead, generator, options, canvasBackgroundColor) => {\n  const arrowheadPoints = getArrowheadPoints(element, shape, position, arrowhead);\n\n  if (arrowheadPoints === null) {\n    return [];\n  }\n\n  switch (arrowhead) {\n    case \"dot\":\n      {\n        const [x, y, r] = arrowheadPoints;\n        return [generator.circle(x, y, r, Object.assign(Object.assign({}, options), {\n          fill: element.strokeColor,\n          fillStyle: \"solid\",\n          stroke: \"none\"\n        }))];\n      }\n\n    case \"d_triangle_filled\":\n    case \"d_triangle\":\n    case \"triangle\":\n      {\n        const [x, y, x2, y2, x3, y3] = arrowheadPoints; // always use solid stroke for triangle arrowhead\n\n        delete options.strokeLineDash;\n        return [generator.polygon([[x, y], [x2, y2], [x3, y3], [x, y]], Object.assign(Object.assign({}, options), {\n          fill: arrowhead === \"d_triangle\" ? canvasBackgroundColor : element.strokeColor,\n          fillStyle: \"solid\",\n          roughness: Math.min(1, options.roughness || 0)\n        }))];\n      }\n\n    case \"d_diamond\":\n    case \"d_diamond_filled\":\n      {\n        const [x, y, x2, y2, x3, y3, x4, y4] = arrowheadPoints; // always use solid stroke for triangle arrowhead\n\n        delete options.strokeLineDash;\n        return [generator.polygon([[x, y], [x2, y2], [x3, y3], [x4, y4], [x, y]], Object.assign(Object.assign({}, options), {\n          fill: arrowhead === \"d_diamond\" ? canvasBackgroundColor : element.strokeColor,\n          fillStyle: \"solid\",\n          roughness: Math.min(1, options.roughness || 0)\n        }))];\n      }\n\n    case \"bar\":\n    case \"arrow\":\n    case \"d_arrow\":\n    default:\n      {\n        const [x2, y2, x3, y3, x4, y4] = arrowheadPoints;\n\n        if (element.strokeStyle === \"dotted\") {\n          // for dotted arrows caps, reduce gap to make it more legible\n          const dash = getDashArrayDotted(element.strokeWidth - 1);\n          options.strokeLineDash = [dash[0], dash[1] - 1];\n        } else {\n          // for solid/dashed, keep solid arrow cap\n          delete options.strokeLineDash;\n        }\n\n        return [generator.line(x3, y3, x2, y2, options), generator.line(x4, y4, x2, y2, options)];\n      }\n  }\n};\n/**\n * Generates the roughjs shape for given element.\n *\n * Low-level. Use `ShapeCache.generateElementShape` instead.\n *\n * @private\n */\n\n\nexport const _generateElementShape = (element, generator, {\n  isExporting,\n  canvasBackgroundColor\n}) => {\n  switch (element.type) {\n    case \"rectangle\":\n    case \"iframe\":\n    case \"embeddable\":\n      {\n        let shape; // this is for rendering the stroke/bg of the embeddable, especially\n        // when the src url is not set\n\n        if (element.roundness) {\n          const w = element.width;\n          const h = element.height;\n          const r = getCornerRadius(Math.min(w, h), element);\n          shape = generator.path(`M ${r} 0 L ${w - r} 0 Q ${w} 0, ${w} ${r} L ${w} ${h - r} Q ${w} ${h}, ${w - r} ${h} L ${r} ${h} Q 0 ${h}, 0 ${h - r} L 0 ${r} Q 0 0, ${r} 0`, generateRoughOptions(modifyIframeLikeForRoughOptions(element, isExporting), true));\n        } else {\n          shape = generator.rectangle(0, 0, element.width, element.height, generateRoughOptions(modifyIframeLikeForRoughOptions(element, isExporting), false));\n        }\n\n        return shape;\n      }\n\n    case \"diamond\":\n      {\n        let shape;\n        const [topX, topY, rightX, rightY, bottomX, bottomY, leftX, leftY] = getDiamondPoints(element);\n\n        if (element.roundness) {\n          const verticalRadius = getCornerRadius(Math.abs(topX - leftX), element);\n          const horizontalRadius = getCornerRadius(Math.abs(rightY - topY), element);\n          shape = generator.path(`M ${topX + verticalRadius} ${topY + horizontalRadius} L ${rightX - verticalRadius} ${rightY - horizontalRadius}\n            C ${rightX} ${rightY}, ${rightX} ${rightY}, ${rightX - verticalRadius} ${rightY + horizontalRadius}\n            L ${bottomX + verticalRadius} ${bottomY - horizontalRadius}\n            C ${bottomX} ${bottomY}, ${bottomX} ${bottomY}, ${bottomX - verticalRadius} ${bottomY - horizontalRadius}\n            L ${leftX + verticalRadius} ${leftY + horizontalRadius}\n            C ${leftX} ${leftY}, ${leftX} ${leftY}, ${leftX + verticalRadius} ${leftY - horizontalRadius}\n            L ${topX - verticalRadius} ${topY + horizontalRadius}\n            C ${topX} ${topY}, ${topX} ${topY}, ${topX + verticalRadius} ${topY + horizontalRadius}`, generateRoughOptions(element, true));\n        } else {\n          shape = generator.polygon([[topX, topY], [rightX, rightY], [bottomX, bottomY], [leftX, leftY]], generateRoughOptions(element));\n        }\n\n        return shape;\n      }\n\n    case \"ellipse\":\n      {\n        const shape = generator.ellipse(element.width / 2, element.height / 2, element.width, element.height, generateRoughOptions(element));\n        return shape;\n      }\n\n    case \"line\":\n    case \"arrow\":\n      {\n        let shape;\n        const options = generateRoughOptions(element); // points array can be empty in the beginning, so it is important to add\n        // initial position to it\n\n        const points = element.points.length ? element.points : [[0, 0]]; // curve is always the first element\n        // this simplifies finding the curve for an element\n\n        if (!element.roundness) {\n          if (options.fill) {\n            shape = [generator.polygon(points, options)];\n          } else {\n            shape = [generator.linearPath(points, options)];\n          }\n        } else {\n          shape = [generator.curve(points, options)];\n        } // add lines only in arrow\n\n\n        if (element.type === \"arrow\") {\n          const {\n            startArrowhead = null,\n            endArrowhead = \"arrow\"\n          } = element;\n\n          if (startArrowhead !== null) {\n            const shapes = getArrowheadShapes(element, shape, \"start\", startArrowhead, generator, options, canvasBackgroundColor);\n            shape.push(...shapes);\n          }\n\n          if (endArrowhead !== null) {\n            if (endArrowhead === undefined) {// Hey, we have an old arrow here!\n            }\n\n            const shapes = getArrowheadShapes(element, shape, \"end\", endArrowhead, generator, options, canvasBackgroundColor);\n            shape.push(...shapes);\n          }\n        }\n\n        return shape;\n      }\n\n    case \"freedraw\":\n      {\n        let shape;\n        generateFreeDrawShape(element);\n\n        if (isPathALoop(element.points)) {\n          // generate rough polygon to fill freedraw shape\n          const simplifiedPoints = simplify(element.points, 0.75);\n          shape = generator.curve(simplifiedPoints, Object.assign(Object.assign({}, generateRoughOptions(element)), {\n            stroke: \"none\"\n          }));\n        } else {\n          shape = null;\n        }\n\n        return shape;\n      }\n\n    case \"frame\":\n    case \"magicframe\":\n    case \"text\":\n    case \"image\":\n      {\n        const shape = null; // we return (and cache) `null` to make sure we don't regenerate\n        // `element.canvas` on rerenders\n\n        return shape;\n      }\n\n    default:\n      {\n        assertNever(element, `generateElementShape(): Unimplemented type ${element === null || element === void 0 ? void 0 : element.type}`);\n        return null;\n      }\n  }\n};"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///../../scene/Shape.ts\n");
4777
4777
 
4778
4778
  /***/ }),
4779
4779
 
@@ -4806,7 +4806,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
4806
4806
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
4807
4807
 
4808
4808
  "use strict";
4809
- 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\");\n/* harmony import */ var _element_typeChecks__WEBPACK_IMPORTED_MODULE_13__ = __webpack_require__(/*! ../element/typeChecks */ \"../../element/typeChecks.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\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 frameIndex = 0;\n let magicFrameIndex = 0;\n\n for (const element of elements) {\n if ((0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_13__.isFrameLikeElement)(element)) {\n if ((0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_13__.isFrameElement)(element)) {\n frameIndex++;\n } else {\n magicFrameIndex++;\n }\n\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: (0,_frame__WEBPACK_IMPORTED_MODULE_9__.getFrameLikeTitle)(element, (0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_13__.isFrameElement)(element) ? frameIndex : magicFrameIndex)\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 canvasBackgroundColor: viewBackgroundColor,\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_AI_BACKEND\":\"http://localhost:3015\",\"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-7391-063c61e\",\"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__.getFrameLikeElements)(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 isExporting: true,\n exportWithDarkMode,\n renderEmbeddables: (_d = opts === null || opts === void 0 ? void 0 : opts.renderEmbeddables) !== null && _d !== void 0 ? _d : false,\n frameRendering,\n canvasBackgroundColor: viewBackgroundColor\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,{"version":3,"file":"../../scene/export.ts.js","mappings":";;;;;;;;;;;;;;;;;;;;AAAA,gBAAgB,SAAI,IAAI,SAAI;AAC5B;AACA;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;;AAEA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,GAAG;AACH;;AAEsC;AACwC;AACA;AAChB;AACwC;AACrD;AACF;AACkC;AACd;AACiB;AACxC;AACc;AAC9B;AAC+C;AAC3E,yDAAyD;AACzD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,oBAAoB,+CAAK,IAAI;AAC7B;AACA;AACA;AACA;;AAEA,2BAA2B,iDAAS;AACpC;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,aAAa,qDAAa;AAC1B;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,8BAA8B,OAAO;AACrC,yBAAyB,iBAAiB;;AAE1C;AACA;AACA;AACA;AACA;AACA;;AAEA,SAAS,uEAAc;AACvB;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;;AAEA;AACA,QAAQ,wEAAkB;AAC1B,UAAU,oEAAc;AACxB;AACA,QAAQ;AACR;AACA;;AAEA,wBAAwB,yDAAc;AACtC;AACA,uBAAuB,+DAAuB;AAC9C,oBAAoB,6DAAqB;AACzC,kBAAkB,gEAAwB;AAC1C,oBAAoB,kEAA0B;AAC9C,+CAA+C,sEAA8B,GAAG,uEAA+B;AAC/G,cAAc,yDAAiB,UAAU,oEAAc;AACvD,OAAO;AACP;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA,qCAAqC,6DAAkB;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD;;AAEA;AACA,mBAAmB,+EAAuB;AAC1C;AACA;AACA;AACA,KAAK;AACL,IAAI;AACJ;AACA;AACA,KAAK;AACL,IAAI;AACJ;AACA;;AAEA;AACA;;AAEO;AACP;AACA,kBAAkB,8DAAsB;AACxC;AACA;AACA,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA;;AAEA,wFAAwF,uDAAe;AACvG;AACA;AACA;AACA,IAAI;AACJ,0BAA0B,6DAAkB;AAC5C;AACA;AACA,IAAI,QAAQ,gEAAgB;AAC5B;AACA,aAAa,2EAA2B;AACxC;AACA,GAAG;AACH,EAAE,wEAAiB;AACnB;AACA,QAAQ,gEAAY;AACpB;AACA;AACA;AACA,4CAA4C;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,CAAC;AACM;AACP;;AAEA;;AAEA;AACA;AACA;AACA,oBAAoB,8DAAsB;AAC1C;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA;;AAEA,qBAAqB;AACrB;;AAEA;AACA;AACA,8BAA8B,uIAED;AAC7B;AACA;AACA;AACA;AACA,cAAc,2DAAe,gCAAgC;AAC7D,OAAO;AACP,MAAM;AACN;AACA;AACA;;AAEA,wFAAwF,uDAAe,qCAAqC;;AAE5I,2CAA2C,8CAAM;AACjD;AACA,gCAAgC,8CAAM;AACtC,yCAAyC,OAAO,EAAE,OAAO;AACzD,mCAAmC,oBAAoB;AACvD,oCAAoC,qBAAqB;;AAEzD;AACA,mCAAmC,oDAAY;AAC/C;;AAEA,6CAA6C;;AAE7C,MAAM,IAA0C;AAChD,qEAAqE,wBAAyB,CAAC,GAAG,iuCAAW,aAAa;;AAE1H;AACA,4CAA4C,uBAAuB;AACnE;;AAEA,mBAAmB,UAAU;AAC7B;;AAEA;AACA;AACA,wBAAwB,4DAAoB;AAC5C;;AAEA;AACA,6BAA6B,yEAAwB;AACrD;AACA;AACA,8CAA8C,SAAS;AACvD,yCAAyC,mBAAmB,EAAE,kBAAkB,WAAW,aAAa,EAAE,IAAI,EAAE,GAAG;AACnH,mBAAmB,YAAY;AAC/B,oBAAoB,aAAa;AACjC;AACA;AACA;AACA;;AAEA;AACA,IAAI;AACJ,IAAI;AACJ;AACA;AACA;AACA;AACA,oBAAoB,UAAU;AAC9B;AACA;AACA;AACA,oBAAoB,UAAU;AAC9B;AACA;AACA;AACA,oBAAoB,UAAU;AAC9B;AACA;AACA,MAAM;AACN;AACA,KAAK;;AAEL;AACA,uDAAuD,8CAAM;AAC7D;AACA;AACA,kCAAkC,MAAM;AACxC,mCAAmC,OAAO;AAC1C;AACA;AACA;;AAEA,eAAe,6DAAS;AACxB,EAAE,uEAAgB,8CAA8C;AAChE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,CAAC,GAAG;;AAEJ;AACA,mCAAmC,gEAAe;AAClD,gBAAgB,gDAAQ;AACxB,iBAAiB,gDAAQ;AACzB;AACA;;AAEO;AACP;AACA;AACA","sources":["webpack:///../../scene/export.ts?c373"],"sourcesContent":["var __awaiter = this && this.__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\nimport rough from \"roughjs/bin/rough\";\nimport { getCommonBounds, getElementAbsoluteCoords } from \"../element/bounds\";\nimport { renderSceneToSvg, renderStaticScene } from \"../renderer/renderScene\";\nimport { cloneJSON, distance, getFontString } from \"../utils\";\nimport { DEFAULT_EXPORT_PADDING, FONT_FAMILY, FRAME_STYLE, SVG_NS, THEME_FILTER } from \"../constants\";\nimport { getDefaultAppState } from \"../appState\";\nimport { serializeAsJSON } from \"../data/json\";\nimport { getInitializedImageElements, updateImageCache } from \"../element/image\";\nimport { elementsOverlappingBBox } from \"../packages/withinBounds\";\nimport { getFrameLikeElements, getFrameLikeTitle, getRootElements } from \"../frame\";\nimport { newTextElement } from \"../element\";\nimport { newElementWith } from \"../element/mutateElement\";\nimport Scene from \"./Scene\";\nimport { isFrameElement, isFrameLikeElement } from \"../element/typeChecks\";\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(); // 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(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 = 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 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 frameIndex = 0;\n  let magicFrameIndex = 0;\n\n  for (const element of elements) {\n    if (isFrameLikeElement(element)) {\n      if (isFrameElement(element)) {\n        frameIndex++;\n      } else {\n        magicFrameIndex++;\n      }\n\n      let textElement = newTextElement({\n        x: element.x,\n        y: element.y - FRAME_STYLE.nameOffsetY,\n        fontFamily: FONT_FAMILY.Assistant,\n        fontSize: FRAME_STYLE.nameFontSize,\n        lineHeight: FRAME_STYLE.nameLineHeight,\n        strokeColor: opts.exportWithDarkMode ? FRAME_STYLE.nameColorDarkTheme : FRAME_STYLE.nameColorLightTheme,\n        text: getFrameLikeTitle(element, isFrameElement(element) ? frameIndex : magicFrameIndex)\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 || 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 = 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\nexport const exportToCanvas = (elements, appState, files, {\n  exportBackground,\n  exportPadding = 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] : getRootElements(elementsForRender), exportPadding);\n  const {\n    canvas,\n    scale = 1\n  } = createCanvas(width, height);\n  const defaultAppState = getDefaultAppState();\n  const {\n    imageCache\n  } = yield updateImageCache({\n    imageCache: new Map(),\n    fileIds: getInitializedImageElements(elementsForRender).map(element => element.fileId),\n    files\n  });\n  renderStaticScene({\n    canvas,\n    rc: rough.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      canvasBackgroundColor: viewBackgroundColor,\n      imageCache,\n      renderGrid: false,\n      isExporting: true\n    }\n  });\n  tempScene.destroy();\n  return canvas;\n});\nexport const 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 = 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 import(\n      /* webpackChunkName: \"image\" */\n      \"../../src/data/image\")).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: 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] : getRootElements(elementsForRender), exportPadding); // initialize SVG root\n\n  const svgRoot = document.createElementNS(SVG_NS, \"svg\");\n  svgRoot.setAttribute(\"version\", \"1.1\");\n  svgRoot.setAttribute(\"xmlns\", 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\", THEME_FILTER);\n  }\n\n  let assetPath = \"https://excalidraw.com/\"; // Asset path needs to be determined only when using package\n\n  if (process.env.VITE_IS_EXCALIDRAW_NPM_PACKAGE) {\n    assetPath = window.EXCALIDRAW_ASSET_PATH || `https://unpkg.com/${process.env.VITE_PKG_NAME}@${process.env.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 = getFrameLikeElements(elements);\n  let exportingFrameClipPath = \"\";\n\n  for (const frame of frameElements) {\n    const [x1, y1, x2, y2] = 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(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 = rough.svg(svgRoot);\n  renderSceneToSvg(elementsForRender, rsvg, svgRoot, files || {}, {\n    offsetX,\n    offsetY,\n    isExporting: true,\n    exportWithDarkMode,\n    renderEmbeddables: (_d = opts === null || opts === void 0 ? void 0 : opts.renderEmbeddables) !== null && _d !== void 0 ? _d : false,\n    frameRendering,\n    canvasBackgroundColor: viewBackgroundColor\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] = getCommonBounds(elements);\n  const width = distance(minX, maxX) + exportPadding * 2;\n  const height = distance(minY, maxY) + exportPadding * 2;\n  return [minX, minY, width, height];\n};\n\nexport const getExportSize = (elements, exportPadding, scale) => {\n  const [,, width, height] = getCanvasSize(elements, exportPadding).map(dimension => Math.trunc(dimension * scale));\n  return [width, height];\n};"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///../../scene/export.ts\n");
4809
+ 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\");\n/* harmony import */ var _element_typeChecks__WEBPACK_IMPORTED_MODULE_13__ = __webpack_require__(/*! ../element/typeChecks */ \"../../element/typeChecks.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\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 frameIndex = 0;\n let magicFrameIndex = 0;\n\n for (const element of elements) {\n if ((0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_13__.isFrameLikeElement)(element)) {\n if ((0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_13__.isFrameElement)(element)) {\n frameIndex++;\n } else {\n magicFrameIndex++;\n }\n\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: (0,_frame__WEBPACK_IMPORTED_MODULE_9__.getFrameLikeTitle)(element, (0,_element_typeChecks__WEBPACK_IMPORTED_MODULE_13__.isFrameElement)(element) ? frameIndex : magicFrameIndex)\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 canvasBackgroundColor: viewBackgroundColor,\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_AI_BACKEND\":\"http://localhost:3015\",\"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-7391-e65bf53\",\"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__.getFrameLikeElements)(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 isExporting: true,\n exportWithDarkMode,\n renderEmbeddables: (_d = opts === null || opts === void 0 ? void 0 : opts.renderEmbeddables) !== null && _d !== void 0 ? _d : false,\n frameRendering,\n canvasBackgroundColor: viewBackgroundColor\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,{"version":3,"file":"../../scene/export.ts.js","mappings":";;;;;;;;;;;;;;;;;;;;AAAA,gBAAgB,SAAI,IAAI,SAAI;AAC5B;AACA;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;;AAEA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,GAAG;AACH;;AAEsC;AACwC;AACA;AAChB;AACwC;AACrD;AACF;AACkC;AACd;AACiB;AACxC;AACc;AAC9B;AAC+C;AAC3E,yDAAyD;AACzD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,oBAAoB,+CAAK,IAAI;AAC7B;AACA;AACA;AACA;;AAEA,2BAA2B,iDAAS;AACpC;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,aAAa,qDAAa;AAC1B;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,8BAA8B,OAAO;AACrC,yBAAyB,iBAAiB;;AAE1C;AACA;AACA;AACA;AACA;AACA;;AAEA,SAAS,uEAAc;AACvB;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;;AAEA;AACA,QAAQ,wEAAkB;AAC1B,UAAU,oEAAc;AACxB;AACA,QAAQ;AACR;AACA;;AAEA,wBAAwB,yDAAc;AACtC;AACA,uBAAuB,+DAAuB;AAC9C,oBAAoB,6DAAqB;AACzC,kBAAkB,gEAAwB;AAC1C,oBAAoB,kEAA0B;AAC9C,+CAA+C,sEAA8B,GAAG,uEAA+B;AAC/G,cAAc,yDAAiB,UAAU,oEAAc;AACvD,OAAO;AACP;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA,qCAAqC,6DAAkB;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD;;AAEA;AACA,mBAAmB,+EAAuB;AAC1C;AACA;AACA;AACA,KAAK;AACL,IAAI;AACJ;AACA;AACA,KAAK;AACL,IAAI;AACJ;AACA;;AAEA;AACA;;AAEO;AACP;AACA,kBAAkB,8DAAsB;AACxC;AACA;AACA,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA;;AAEA,wFAAwF,uDAAe;AACvG;AACA;AACA;AACA,IAAI;AACJ,0BAA0B,6DAAkB;AAC5C;AACA;AACA,IAAI,QAAQ,gEAAgB;AAC5B;AACA,aAAa,2EAA2B;AACxC;AACA,GAAG;AACH,EAAE,wEAAiB;AACnB;AACA,QAAQ,gEAAY;AACpB;AACA;AACA;AACA,4CAA4C;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,CAAC;AACM;AACP;;AAEA;;AAEA;AACA;AACA;AACA,oBAAoB,8DAAsB;AAC1C;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA;;AAEA,qBAAqB;AACrB;;AAEA;AACA;AACA,8BAA8B,uIAED;AAC7B;AACA;AACA;AACA;AACA,cAAc,2DAAe,gCAAgC;AAC7D,OAAO;AACP,MAAM;AACN;AACA;AACA;;AAEA,wFAAwF,uDAAe,qCAAqC;;AAE5I,2CAA2C,8CAAM;AACjD;AACA,gCAAgC,8CAAM;AACtC,yCAAyC,OAAO,EAAE,OAAO;AACzD,mCAAmC,oBAAoB;AACvD,oCAAoC,qBAAqB;;AAEzD;AACA,mCAAmC,oDAAY;AAC/C;;AAEA,6CAA6C;;AAE7C,MAAM,IAA0C;AAChD,qEAAqE,wBAAyB,CAAC,GAAG,iuCAAW,aAAa;;AAE1H;AACA,4CAA4C,uBAAuB;AACnE;;AAEA,mBAAmB,UAAU;AAC7B;;AAEA;AACA;AACA,wBAAwB,4DAAoB;AAC5C;;AAEA;AACA,6BAA6B,yEAAwB;AACrD;AACA;AACA,8CAA8C,SAAS;AACvD,yCAAyC,mBAAmB,EAAE,kBAAkB,WAAW,aAAa,EAAE,IAAI,EAAE,GAAG;AACnH,mBAAmB,YAAY;AAC/B,oBAAoB,aAAa;AACjC;AACA;AACA;AACA;;AAEA;AACA,IAAI;AACJ,IAAI;AACJ;AACA;AACA;AACA;AACA,oBAAoB,UAAU;AAC9B;AACA;AACA;AACA,oBAAoB,UAAU;AAC9B;AACA;AACA;AACA,oBAAoB,UAAU;AAC9B;AACA;AACA,MAAM;AACN;AACA,KAAK;;AAEL;AACA,uDAAuD,8CAAM;AAC7D;AACA;AACA,kCAAkC,MAAM;AACxC,mCAAmC,OAAO;AAC1C;AACA;AACA;;AAEA,eAAe,6DAAS;AACxB,EAAE,uEAAgB,8CAA8C;AAChE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,CAAC,GAAG;;AAEJ;AACA,mCAAmC,gEAAe;AAClD,gBAAgB,gDAAQ;AACxB,iBAAiB,gDAAQ;AACzB;AACA;;AAEO;AACP;AACA;AACA","sources":["webpack:///../../scene/export.ts?c373"],"sourcesContent":["var __awaiter = this && this.__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\nimport rough from \"roughjs/bin/rough\";\nimport { getCommonBounds, getElementAbsoluteCoords } from \"../element/bounds\";\nimport { renderSceneToSvg, renderStaticScene } from \"../renderer/renderScene\";\nimport { cloneJSON, distance, getFontString } from \"../utils\";\nimport { DEFAULT_EXPORT_PADDING, FONT_FAMILY, FRAME_STYLE, SVG_NS, THEME_FILTER } from \"../constants\";\nimport { getDefaultAppState } from \"../appState\";\nimport { serializeAsJSON } from \"../data/json\";\nimport { getInitializedImageElements, updateImageCache } from \"../element/image\";\nimport { elementsOverlappingBBox } from \"../packages/withinBounds\";\nimport { getFrameLikeElements, getFrameLikeTitle, getRootElements } from \"../frame\";\nimport { newTextElement } from \"../element\";\nimport { newElementWith } from \"../element/mutateElement\";\nimport Scene from \"./Scene\";\nimport { isFrameElement, isFrameLikeElement } from \"../element/typeChecks\";\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(); // 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(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 = 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 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 frameIndex = 0;\n  let magicFrameIndex = 0;\n\n  for (const element of elements) {\n    if (isFrameLikeElement(element)) {\n      if (isFrameElement(element)) {\n        frameIndex++;\n      } else {\n        magicFrameIndex++;\n      }\n\n      let textElement = newTextElement({\n        x: element.x,\n        y: element.y - FRAME_STYLE.nameOffsetY,\n        fontFamily: FONT_FAMILY.Assistant,\n        fontSize: FRAME_STYLE.nameFontSize,\n        lineHeight: FRAME_STYLE.nameLineHeight,\n        strokeColor: opts.exportWithDarkMode ? FRAME_STYLE.nameColorDarkTheme : FRAME_STYLE.nameColorLightTheme,\n        text: getFrameLikeTitle(element, isFrameElement(element) ? frameIndex : magicFrameIndex)\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 || 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 = 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\nexport const exportToCanvas = (elements, appState, files, {\n  exportBackground,\n  exportPadding = 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] : getRootElements(elementsForRender), exportPadding);\n  const {\n    canvas,\n    scale = 1\n  } = createCanvas(width, height);\n  const defaultAppState = getDefaultAppState();\n  const {\n    imageCache\n  } = yield updateImageCache({\n    imageCache: new Map(),\n    fileIds: getInitializedImageElements(elementsForRender).map(element => element.fileId),\n    files\n  });\n  renderStaticScene({\n    canvas,\n    rc: rough.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      canvasBackgroundColor: viewBackgroundColor,\n      imageCache,\n      renderGrid: false,\n      isExporting: true\n    }\n  });\n  tempScene.destroy();\n  return canvas;\n});\nexport const 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 = 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 import(\n      /* webpackChunkName: \"image\" */\n      \"../../src/data/image\")).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: 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] : getRootElements(elementsForRender), exportPadding); // initialize SVG root\n\n  const svgRoot = document.createElementNS(SVG_NS, \"svg\");\n  svgRoot.setAttribute(\"version\", \"1.1\");\n  svgRoot.setAttribute(\"xmlns\", 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\", THEME_FILTER);\n  }\n\n  let assetPath = \"https://excalidraw.com/\"; // Asset path needs to be determined only when using package\n\n  if (process.env.VITE_IS_EXCALIDRAW_NPM_PACKAGE) {\n    assetPath = window.EXCALIDRAW_ASSET_PATH || `https://unpkg.com/${process.env.VITE_PKG_NAME}@${process.env.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 = getFrameLikeElements(elements);\n  let exportingFrameClipPath = \"\";\n\n  for (const frame of frameElements) {\n    const [x1, y1, x2, y2] = 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(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 = rough.svg(svgRoot);\n  renderSceneToSvg(elementsForRender, rsvg, svgRoot, files || {}, {\n    offsetX,\n    offsetY,\n    isExporting: true,\n    exportWithDarkMode,\n    renderEmbeddables: (_d = opts === null || opts === void 0 ? void 0 : opts.renderEmbeddables) !== null && _d !== void 0 ? _d : false,\n    frameRendering,\n    canvasBackgroundColor: viewBackgroundColor\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] = getCommonBounds(elements);\n  const width = distance(minX, maxX) + exportPadding * 2;\n  const height = distance(minY, maxY) + exportPadding * 2;\n  return [minX, minY, width, height];\n};\n\nexport const getExportSize = (elements, exportPadding, scale) => {\n  const [,, width, height] = getCanvasSize(elements, exportPadding).map(dimension => Math.trunc(dimension * scale));\n  return [width, height];\n};"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///../../scene/export.ts\n");
4810
4810
 
4811
4811
  /***/ }),
4812
4812
 
@@ -6012,7 +6012,7 @@ module.exports = JSON.parse('{"ar-SA":99,"az-AZ":18,"bg-BG":75,"bn-BD":55,"ca-ES
6012
6012
  /******/ // This function allow to reference async chunks
6013
6013
  /******/ __webpack_require__.u = (chunkId) => {
6014
6014
  /******/ // return url for filenames based on template
6015
- /******/ 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":"8fa3eb9e8749eedafb19"}[chunkId] + ".js";
6015
+ /******/ 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":"c09b2c6a1f68505f7149"}[chunkId] + ".js";
6016
6016
  /******/ };
6017
6017
  /******/ })();
6018
6018
  /******/