@excalidraw/excalidraw 0.16.1-4c35eba → 0.16.1-ceb637f
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.
|
@@ -2386,7 +2386,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
|
|
|
2386
2386
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
2387
2387
|
|
|
2388
2388
|
"use strict";
|
|
2389
|
-
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"trackEvent\": () => (/* binding */ trackEvent)\n/* harmony export */ });\nconst trackEvent = (category, action, label, value) => {\n try {\n // place here categories that you want to track as events\n // KEEP IN MIND THE PRICING\n const ALLOWED_CATEGORIES_TO_TRACK = []; // Uncomment the next line to track locally\n // console.log(\"Track Event\", { category, action, label, value });\n\n if (typeof window === \"undefined\" || ({\"VITE_APP_BACKEND_V2_GET_URL\":\"https://json-dev.excalidraw.com/api/v2/\",\"VITE_APP_BACKEND_V2_POST_URL\":\"https://json-dev.excalidraw.com/api/v2/post/\",\"VITE_APP_LIBRARY_URL\":\"https://libraries.excalidraw.com\",\"VITE_APP_LIBRARY_BACKEND\":\"https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries\",\"VITE_APP_WS_SERVER_URL\":\"http://localhost:3002\",\"VITE_APP_PORTAL_URL\":\"\",\"VITE_APP_PLUS_LP\":\"https://plus.excalidraw.com\",\"VITE_APP_PLUS_APP\":\"https://app.excalidraw.com\",\"VITE_APP_FIREBASE_CONFIG\":\"{\\\"apiKey\\\":\\\"AIzaSyCMkxA60XIW8KbqMYL7edC4qT5l4qHX2h8\\\",\\\"authDomain\\\":\\\"excalidraw-oss-dev.firebaseapp.com\\\",\\\"projectId\\\":\\\"excalidraw-oss-dev\\\",\\\"storageBucket\\\":\\\"excalidraw-oss-dev.appspot.com\\\",\\\"messagingSenderId\\\":\\\"664559512677\\\",\\\"appId\\\":\\\"1:664559512677:web:a385181f2928d328a7aa8c\\\"}\",\"VITE_APP_DEV_ENABLE_SW\":\"\",\"VITE_APP_DEV_DISABLE_LIVE_RELOAD\":\"\",\"VITE_APP_DISABLE_TRACKING\":\"true\",\"FAST_REFRESH\":\"false\",\"VITE_APP_PORT\":\"3000\",\"VITE_APP_DEBUG_ENABLE_TEXT_CONTAINER_BOUNDING_BOX\":\"\",\"VITE_APP_COLLAPSE_OVERLAY\":\"true\",\"VITE_APP_ENABLE_ESLINT\":\"true\",\"VITE_PKG_NAME\":\"@excalidraw/excalidraw\",\"VITE_PKG_VERSION\":\"0.16.1-
|
|
2389
|
+
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"trackEvent\": () => (/* binding */ trackEvent)\n/* harmony export */ });\nconst trackEvent = (category, action, label, value) => {\n try {\n // place here categories that you want to track as events\n // KEEP IN MIND THE PRICING\n const ALLOWED_CATEGORIES_TO_TRACK = []; // Uncomment the next line to track locally\n // console.log(\"Track Event\", { category, action, label, value });\n\n if (typeof window === \"undefined\" || ({\"VITE_APP_BACKEND_V2_GET_URL\":\"https://json-dev.excalidraw.com/api/v2/\",\"VITE_APP_BACKEND_V2_POST_URL\":\"https://json-dev.excalidraw.com/api/v2/post/\",\"VITE_APP_LIBRARY_URL\":\"https://libraries.excalidraw.com\",\"VITE_APP_LIBRARY_BACKEND\":\"https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries\",\"VITE_APP_WS_SERVER_URL\":\"http://localhost:3002\",\"VITE_APP_PORTAL_URL\":\"\",\"VITE_APP_PLUS_LP\":\"https://plus.excalidraw.com\",\"VITE_APP_PLUS_APP\":\"https://app.excalidraw.com\",\"VITE_APP_FIREBASE_CONFIG\":\"{\\\"apiKey\\\":\\\"AIzaSyCMkxA60XIW8KbqMYL7edC4qT5l4qHX2h8\\\",\\\"authDomain\\\":\\\"excalidraw-oss-dev.firebaseapp.com\\\",\\\"projectId\\\":\\\"excalidraw-oss-dev\\\",\\\"storageBucket\\\":\\\"excalidraw-oss-dev.appspot.com\\\",\\\"messagingSenderId\\\":\\\"664559512677\\\",\\\"appId\\\":\\\"1:664559512677:web:a385181f2928d328a7aa8c\\\"}\",\"VITE_APP_DEV_ENABLE_SW\":\"\",\"VITE_APP_DEV_DISABLE_LIVE_RELOAD\":\"\",\"VITE_APP_DISABLE_TRACKING\":\"true\",\"FAST_REFRESH\":\"false\",\"VITE_APP_PORT\":\"3000\",\"VITE_APP_DEBUG_ENABLE_TEXT_CONTAINER_BOUNDING_BOX\":\"\",\"VITE_APP_COLLAPSE_OVERLAY\":\"true\",\"VITE_APP_ENABLE_ESLINT\":\"true\",\"VITE_PKG_NAME\":\"@excalidraw/excalidraw\",\"VITE_PKG_VERSION\":\"0.16.1-ceb637f\",\"VITE_IS_EXCALIDRAW_NPM_PACKAGE\":true}).VITE_WORKER_ID) {\n return;\n }\n\n if (!ALLOWED_CATEGORIES_TO_TRACK.includes(category)) {\n return;\n }\n\n if (window.sa_event) {\n window.sa_event(action, {\n category,\n label,\n value\n });\n }\n } catch (error) {\n console.error(\"error during analytics\", error);\n }\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi4vLi4vYW5hbHl0aWNzLnRzLmpzIiwibWFwcGluZ3MiOiI7Ozs7QUFBTztBQUNQO0FBQ0E7QUFDQTtBQUNBLDRDQUE0QztBQUM1QyxvQ0FBb0MsZ0NBQWdDOztBQUVwRSx5Q0FBeUMsOHFDQUFXO0FBQ3BEO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxPQUFPO0FBQ1A7QUFDQSxJQUFJO0FBQ0o7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vLy4uLy4uL2FuYWx5dGljcy50cz81NjYxIl0sInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjb25zdCB0cmFja0V2ZW50ID0gKGNhdGVnb3J5LCBhY3Rpb24sIGxhYmVsLCB2YWx1ZSkgPT4ge1xuICB0cnkge1xuICAgIC8vIHBsYWNlIGhlcmUgY2F0ZWdvcmllcyB0aGF0IHlvdSB3YW50IHRvIHRyYWNrIGFzIGV2ZW50c1xuICAgIC8vIEtFRVAgSU4gTUlORCBUSEUgUFJJQ0lOR1xuICAgIGNvbnN0IEFMTE9XRURfQ0FURUdPUklFU19UT19UUkFDSyA9IFtdOyAvLyBVbmNvbW1lbnQgdGhlIG5leHQgbGluZSB0byB0cmFjayBsb2NhbGx5XG4gICAgLy8gY29uc29sZS5sb2coXCJUcmFjayBFdmVudFwiLCB7IGNhdGVnb3J5LCBhY3Rpb24sIGxhYmVsLCB2YWx1ZSB9KTtcblxuICAgIGlmICh0eXBlb2Ygd2luZG93ID09PSBcInVuZGVmaW5lZFwiIHx8IHByb2Nlc3MuZW52LlZJVEVfV09SS0VSX0lEKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKCFBTExPV0VEX0NBVEVHT1JJRVNfVE9fVFJBQ0suaW5jbHVkZXMoY2F0ZWdvcnkpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKHdpbmRvdy5zYV9ldmVudCkge1xuICAgICAgd2luZG93LnNhX2V2ZW50KGFjdGlvbiwge1xuICAgICAgICBjYXRlZ29yeSxcbiAgICAgICAgbGFiZWwsXG4gICAgICAgIHZhbHVlXG4gICAgICB9KTtcbiAgICB9XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgY29uc29sZS5lcnJvcihcImVycm9yIGR1cmluZyBhbmFseXRpY3NcIiwgZXJyb3IpO1xuICB9XG59OyJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///../../analytics.ts\n");
|
|
2390
2390
|
|
|
2391
2391
|
/***/ }),
|
|
2392
2392
|
|
|
@@ -3992,7 +3992,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
|
|
|
3992
3992
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
3993
3993
|
|
|
3994
3994
|
"use strict";
|
|
3995
|
-
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"addElementsToFrame\": () => (/* binding */ addElementsToFrame),\n/* harmony export */ \"bindElementsToFramesAfterDuplication\": () => (/* binding */ bindElementsToFramesAfterDuplication),\n/* harmony export */ \"elementOverlapsWithFrame\": () => (/* binding */ elementOverlapsWithFrame),\n/* harmony export */ \"elementsAreInFrameBounds\": () => (/* binding */ elementsAreInFrameBounds),\n/* harmony export */ \"getContainingFrame\": () => (/* binding */ getContainingFrame),\n/* harmony export */ \"getElementsCompletelyInFrame\": () => (/* binding */ getElementsCompletelyInFrame),\n/* harmony export */ \"getElementsInNewFrame\": () => (/* binding */ getElementsInNewFrame),\n/* harmony export */ \"getElementsInResizingFrame\": () => (/* binding */ getElementsInResizingFrame),\n/* harmony export */ \"getElementsIntersectingFrame\": () => (/* binding */ getElementsIntersectingFrame),\n/* harmony export */ \"getFrameElements\": () => (/* binding */ getFrameElements),\n/* harmony export */ \"getTargetFrame\": () => (/* binding */ getTargetFrame),\n/* harmony export */ \"groupByFrames\": () => (/* binding */ groupByFrames),\n/* harmony export */ \"groupsAreAtLeastIntersectingTheFrame\": () => (/* binding */ groupsAreAtLeastIntersectingTheFrame),\n/* harmony export */ \"groupsAreCompletelyOutOfFrame\": () => (/* binding */ groupsAreCompletelyOutOfFrame),\n/* harmony export */ \"isCursorInFrame\": () => (/* binding */ isCursorInFrame),\n/* harmony export */ \"isElementContainingFrame\": () => (/* binding */ isElementContainingFrame),\n/* harmony export */ \"isElementInFrame\": () => (/* binding */ isElementInFrame),\n/* harmony export */ \"omitGroupsContainingFrames\": () => (/* binding */ omitGroupsContainingFrames),\n/* harmony export */ \"removeAllElementsFromFrame\": () => (/* binding */ removeAllElementsFromFrame),\n/* harmony export */ \"removeElementsFromFrame\": () => (/* binding */ removeElementsFromFrame),\n/* harmony export */ \"replaceAllElementsInFrame\": () => (/* binding */ replaceAllElementsInFrame),\n/* harmony export */ \"updateFrameMembershipOfSelectedElements\": () => (/* binding */ updateFrameMembershipOfSelectedElements)\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 _element_textElement__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./element/textElement */ \"../../element/textElement.ts\");\n/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./utils */ \"../../utils.ts\");\n/* harmony import */ var _element_mutateElement__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./element/mutateElement */ \"../../element/mutateElement.ts\");\n/* harmony import */ var _scene__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./scene */ \"../../scene/index.ts\");\n/* harmony import */ var _zindex__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./zindex */ \"../../zindex.ts\");\n/* harmony import */ var _groups__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ./groups */ \"../../groups.ts\");\n/* harmony import */ var _scene_Scene__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ./scene/Scene */ \"../../scene/Scene.ts\");\n/* harmony import */ var _element_bounds__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! ./element/bounds */ \"../../element/bounds.ts\");\n\n\n\n\n\n\n\n\n\n\n // --------------------------- Frame State ------------------------------------\n\nconst bindElementsToFramesAfterDuplication = (nextElements, oldElements, oldIdToDuplicatedId) => {\n const nextElementMap = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.arrayToMap)(nextElements);\n\n for (const element of oldElements) {\n if (element.frameId) {\n // use its frameId to get the new frameId\n const nextElementId = oldIdToDuplicatedId.get(element.id);\n const nextFrameId = oldIdToDuplicatedId.get(element.frameId);\n\n if (nextElementId) {\n const nextElement = nextElementMap.get(nextElementId);\n\n if (nextElement) {\n (0,_element_mutateElement__WEBPACK_IMPORTED_MODULE_4__.mutateElement)(nextElement, {\n frameId: nextFrameId !== null && nextFrameId !== void 0 ? nextFrameId : element.frameId\n }, false);\n }\n }\n }\n }\n}; // --------------------------- Frame Geometry ---------------------------------\n\nclass Point {\n constructor(x, y) {\n this.x = x;\n this.y = y;\n }\n\n}\n\nclass LineSegment {\n constructor(pointA, pointB) {\n this.first = pointA;\n this.second = pointB;\n }\n\n getBoundingBox() {\n return [new Point(Math.min(this.first.x, this.second.x), Math.min(this.first.y, this.second.y)), new Point(Math.max(this.first.x, this.second.x), Math.max(this.first.y, this.second.y))];\n }\n\n} // https://martin-thoma.com/how-to-check-if-two-line-segments-intersect/\n\n\nclass FrameGeometry {\n static crossProduct(a, b) {\n return a.x * b.y - b.x * a.y;\n }\n\n static doBoundingBoxesIntersect(a, b) {\n return a[0].x <= b[1].x && a[1].x >= b[0].x && a[0].y <= b[1].y && a[1].y >= b[0].y;\n }\n\n static isPointOnLine(a, b) {\n const aTmp = new LineSegment(new Point(0, 0), new Point(a.second.x - a.first.x, a.second.y - a.first.y));\n const bTmp = new Point(b.x - a.first.x, b.y - a.first.y);\n const r = this.crossProduct(aTmp.second, bTmp);\n return Math.abs(r) < this.EPSILON;\n }\n\n static isPointRightOfLine(a, b) {\n const aTmp = new LineSegment(new Point(0, 0), new Point(a.second.x - a.first.x, a.second.y - a.first.y));\n const bTmp = new Point(b.x - a.first.x, b.y - a.first.y);\n return this.crossProduct(aTmp.second, bTmp) < 0;\n }\n\n static lineSegmentTouchesOrCrossesLine(a, b) {\n return this.isPointOnLine(a, b.first) || this.isPointOnLine(a, b.second) || (this.isPointRightOfLine(a, b.first) ? !this.isPointRightOfLine(a, b.second) : this.isPointRightOfLine(a, b.second));\n }\n\n static doLineSegmentsIntersect(a, b) {\n const aSegment = new LineSegment(new Point(a[0][0], a[0][1]), new Point(a[1][0], a[1][1]));\n const bSegment = new LineSegment(new Point(b[0][0], b[0][1]), new Point(b[1][0], b[1][1]));\n const box1 = aSegment.getBoundingBox();\n const box2 = bSegment.getBoundingBox();\n return this.doBoundingBoxesIntersect(box1, box2) && this.lineSegmentTouchesOrCrossesLine(aSegment, bSegment) && this.lineSegmentTouchesOrCrossesLine(bSegment, aSegment);\n }\n\n static isElementIntersectingFrame(element, frame) {\n const frameLineSegments = (0,_element_bounds__WEBPACK_IMPORTED_MODULE_9__.getElementLineSegments)(frame);\n const elementLineSegments = (0,_element_bounds__WEBPACK_IMPORTED_MODULE_9__.getElementLineSegments)(element);\n const intersecting = frameLineSegments.some(frameLineSegment => elementLineSegments.some(elementLineSegment => this.doLineSegmentsIntersect(frameLineSegment, elementLineSegment)));\n return intersecting;\n }\n\n}\n\nFrameGeometry.EPSILON = 0.000001;\nconst getElementsCompletelyInFrame = (elements, frame) => omitGroupsContainingFrames((0,_scene__WEBPACK_IMPORTED_MODULE_5__.getElementsWithinSelection)(elements, frame, false)).filter(element => element.type !== \"frame\" && !element.frameId || element.frameId === frame.id);\nconst isElementContainingFrame = (elements, element, frame) => {\n return (0,_scene__WEBPACK_IMPORTED_MODULE_5__.getElementsWithinSelection)(elements, element).some(e => e.id === frame.id);\n};\nconst getElementsIntersectingFrame = (elements, frame) => elements.filter(element => FrameGeometry.isElementIntersectingFrame(element, frame));\nconst elementsAreInFrameBounds = (elements, frame) => {\n const [selectionX1, selectionY1, selectionX2, selectionY2] = (0,_element__WEBPACK_IMPORTED_MODULE_0__.getElementAbsoluteCoords)(frame);\n const [elementX1, elementY1, elementX2, elementY2] = (0,_element__WEBPACK_IMPORTED_MODULE_0__.getCommonBounds)(elements);\n return selectionX1 <= elementX1 && selectionY1 <= elementY1 && selectionX2 >= elementX2 && selectionY2 >= elementY2;\n};\nconst elementOverlapsWithFrame = (element, frame) => {\n return elementsAreInFrameBounds([element], frame) || FrameGeometry.isElementIntersectingFrame(element, frame) || isElementContainingFrame([frame], element, frame);\n};\nconst isCursorInFrame = (cursorCoords, frame) => {\n const [fx1, fy1, fx2, fy2] = (0,_element__WEBPACK_IMPORTED_MODULE_0__.getElementAbsoluteCoords)(frame);\n return (0,_math__WEBPACK_IMPORTED_MODULE_1__.isPointWithinBounds)([fx1, fy1], [cursorCoords.x, cursorCoords.y], [fx2, fy2]);\n};\nconst groupsAreAtLeastIntersectingTheFrame = (elements, groupIds, frame) => {\n const elementsInGroup = groupIds.flatMap(groupId => (0,_groups__WEBPACK_IMPORTED_MODULE_7__.getElementsInGroup)(elements, groupId));\n\n if (elementsInGroup.length === 0) {\n return true;\n }\n\n return !!elementsInGroup.find(element => elementsAreInFrameBounds([element], frame) || FrameGeometry.isElementIntersectingFrame(element, frame));\n};\nconst groupsAreCompletelyOutOfFrame = (elements, groupIds, frame) => {\n const elementsInGroup = groupIds.flatMap(groupId => (0,_groups__WEBPACK_IMPORTED_MODULE_7__.getElementsInGroup)(elements, groupId));\n\n if (elementsInGroup.length === 0) {\n return true;\n }\n\n return elementsInGroup.find(element => elementsAreInFrameBounds([element], frame) || FrameGeometry.isElementIntersectingFrame(element, frame)) === undefined;\n}; // --------------------------- Frame Utils ------------------------------------\n\n/**\n * Returns a map of frameId to frame elements. Includes empty frames.\n */\n\nconst groupByFrames = elements => {\n const frameElementsMap = new Map();\n\n for (const element of elements) {\n const frameId = (0,_element__WEBPACK_IMPORTED_MODULE_0__.isFrameElement)(element) ? element.id : element.frameId;\n\n if (frameId && !frameElementsMap.has(frameId)) {\n frameElementsMap.set(frameId, getFrameElements(elements, frameId));\n }\n }\n\n return frameElementsMap;\n};\nconst getFrameElements = (allElements, frameId) => allElements.filter(element => element.frameId === frameId);\nconst getElementsInResizingFrame = (allElements, frame, appState) => {\n const prevElementsInFrame = getFrameElements(allElements, frame.id);\n const nextElementsInFrame = new Set(prevElementsInFrame);\n const elementsCompletelyInFrame = new Set([...getElementsCompletelyInFrame(allElements, frame), ...prevElementsInFrame.filter(element => isElementContainingFrame(allElements, element, frame))]);\n const elementsNotCompletelyInFrame = prevElementsInFrame.filter(element => !elementsCompletelyInFrame.has(element)); // for elements that are completely in the frame\n // if they are part of some groups, then those groups are still\n // considered to belong to the frame\n\n const groupsToKeep = new Set(Array.from(elementsCompletelyInFrame).flatMap(element => element.groupIds));\n\n for (const element of elementsNotCompletelyInFrame) {\n if (!FrameGeometry.isElementIntersectingFrame(element, frame)) {\n if (element.groupIds.length === 0) {\n nextElementsInFrame.delete(element);\n }\n } else if (element.groupIds.length > 0) {\n // group element intersects with the frame, we should keep the groups\n // that this element is part of\n for (const id of element.groupIds) {\n groupsToKeep.add(id);\n }\n }\n }\n\n for (const element of elementsNotCompletelyInFrame) {\n if (element.groupIds.length > 0) {\n let shouldRemoveElement = true;\n\n for (const id of element.groupIds) {\n if (groupsToKeep.has(id)) {\n shouldRemoveElement = false;\n }\n }\n\n if (shouldRemoveElement) {\n nextElementsInFrame.delete(element);\n }\n }\n }\n\n const individualElementsCompletelyInFrame = Array.from(elementsCompletelyInFrame).filter(element => element.groupIds.length === 0);\n\n for (const element of individualElementsCompletelyInFrame) {\n nextElementsInFrame.add(element);\n }\n\n const newGroupElementsCompletelyInFrame = Array.from(elementsCompletelyInFrame).filter(element => element.groupIds.length > 0);\n const groupIds = (0,_groups__WEBPACK_IMPORTED_MODULE_7__.selectGroupsFromGivenElements)(newGroupElementsCompletelyInFrame, appState); // new group elements\n\n for (const [id, isSelected] of Object.entries(groupIds)) {\n if (isSelected) {\n const elementsInGroup = (0,_groups__WEBPACK_IMPORTED_MODULE_7__.getElementsInGroup)(allElements, id);\n\n if (elementsAreInFrameBounds(elementsInGroup, frame)) {\n for (const element of elementsInGroup) {\n nextElementsInFrame.add(element);\n }\n }\n }\n }\n\n return [...nextElementsInFrame].filter(element => {\n return !((0,_element__WEBPACK_IMPORTED_MODULE_0__.isTextElement)(element) && element.containerId);\n });\n};\nconst getElementsInNewFrame = (allElements, frame) => {\n return omitGroupsContainingFrames(allElements, getElementsCompletelyInFrame(allElements, frame));\n};\nconst getContainingFrame = (element,\n/**\n * Optionally an elements map, in case the elements aren't in the Scene yet.\n * Takes precedence over Scene elements, even if the element exists\n * in Scene elements and not the supplied elements map.\n */\nelementsMap) => {\n var _a;\n\n if (element.frameId) {\n if (elementsMap) {\n return elementsMap.get(element.frameId) || null;\n }\n\n return ((_a = _scene_Scene__WEBPACK_IMPORTED_MODULE_8__[\"default\"].getScene(element)) === null || _a === void 0 ? void 0 : _a.getElement(element.frameId)) || null;\n }\n\n return null;\n}; // --------------------------- Frame Operations -------------------------------\n\nconst addElementsToFrame = (allElements, elementsToAdd, frame) => {\n const _elementsToAdd = [];\n\n for (const element of elementsToAdd) {\n _elementsToAdd.push(element);\n\n const boundTextElement = (0,_element_textElement__WEBPACK_IMPORTED_MODULE_2__.getBoundTextElement)(element);\n\n if (boundTextElement) {\n _elementsToAdd.push(boundTextElement);\n }\n }\n\n const allElementsIndex = allElements.reduce((acc, element, index) => {\n acc[element.id] = index;\n return acc;\n }, {});\n const frameIndex = allElementsIndex[frame.id]; // need to be calculated before the mutation below occurs\n\n const leftFrameBoundaryIndex = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.findIndex)(allElements, e => e.frameId === frame.id);\n const existingFrameChildren = allElements.filter(element => element.frameId === frame.id);\n const addedFrameChildren_left = [];\n const addedFrameChildren_right = [];\n\n for (const element of omitGroupsContainingFrames(allElements, _elementsToAdd)) {\n if (element.frameId !== frame.id && !(0,_element__WEBPACK_IMPORTED_MODULE_0__.isFrameElement)(element)) {\n if (allElementsIndex[element.id] > frameIndex) {\n addedFrameChildren_right.push(element);\n } else {\n addedFrameChildren_left.push(element);\n }\n\n (0,_element_mutateElement__WEBPACK_IMPORTED_MODULE_4__.mutateElement)(element, {\n frameId: frame.id\n }, false);\n }\n }\n\n const frameElement = allElements[frameIndex];\n const nextFrameChildren = addedFrameChildren_left.concat(existingFrameChildren).concat(addedFrameChildren_right);\n const nextFrameChildrenMap = nextFrameChildren.reduce((acc, element) => {\n acc[element.id] = true;\n return acc;\n }, {});\n const nextOtherElements_left = allElements.slice(0, leftFrameBoundaryIndex >= 0 ? leftFrameBoundaryIndex : frameIndex).filter(element => !nextFrameChildrenMap[element.id]);\n const nextOtherElement_right = allElements.slice(frameIndex + 1).filter(element => !nextFrameChildrenMap[element.id]);\n const nextElements = nextOtherElements_left.concat(nextFrameChildren).concat([frameElement]).concat(nextOtherElement_right);\n return nextElements;\n};\nconst removeElementsFromFrame = (allElements, elementsToRemove, appState) => {\n const _elementsToRemove = [];\n\n for (const element of elementsToRemove) {\n if (element.frameId) {\n _elementsToRemove.push(element);\n\n const boundTextElement = (0,_element_textElement__WEBPACK_IMPORTED_MODULE_2__.getBoundTextElement)(element);\n\n if (boundTextElement) {\n _elementsToRemove.push(boundTextElement);\n }\n }\n }\n\n for (const element of _elementsToRemove) {\n (0,_element_mutateElement__WEBPACK_IMPORTED_MODULE_4__.mutateElement)(element, {\n frameId: null\n }, false);\n }\n\n const nextElements = (0,_zindex__WEBPACK_IMPORTED_MODULE_6__.moveOneRight)(allElements, appState, Array.from(_elementsToRemove));\n return nextElements;\n};\nconst removeAllElementsFromFrame = (allElements, frame, appState) => {\n const elementsInFrame = getFrameElements(allElements, frame.id);\n return removeElementsFromFrame(allElements, elementsInFrame, appState);\n};\nconst replaceAllElementsInFrame = (allElements, nextElementsInFrame, frame, appState) => {\n return addElementsToFrame(removeAllElementsFromFrame(allElements, frame, appState), nextElementsInFrame, frame);\n};\n/** does not mutate elements, but returns new ones */\n\nconst updateFrameMembershipOfSelectedElements = (allElements, appState, app) => {\n const selectedElements = app.scene.getSelectedElements({\n selectedElementIds: appState.selectedElementIds,\n // supplying elements explicitly in case we're passed non-state elements\n elements: allElements\n });\n const elementsToFilter = new Set(selectedElements);\n\n if (appState.editingGroupId) {\n for (const element of selectedElements) {\n if (element.groupIds.length === 0) {\n elementsToFilter.add(element);\n } else {\n element.groupIds.flatMap(gid => (0,_groups__WEBPACK_IMPORTED_MODULE_7__.getElementsInGroup)(allElements, gid)).forEach(element => elementsToFilter.add(element));\n }\n }\n }\n\n const elementsToRemove = new Set();\n elementsToFilter.forEach(element => {\n if (element.frameId && !(0,_element__WEBPACK_IMPORTED_MODULE_0__.isFrameElement)(element) && !isElementInFrame(element, allElements, appState)) {\n elementsToRemove.add(element);\n }\n });\n return elementsToRemove.size > 0 ? removeElementsFromFrame(allElements, [...elementsToRemove], appState) : allElements;\n};\n/**\n * filters out elements that are inside groups that contain a frame element\n * anywhere in the group tree\n */\n\nconst omitGroupsContainingFrames = (allElements,\n/** subset of elements you want to filter. Optional perf optimization so we\n * don't have to filter all elements unnecessarily\n */\nselectedElements) => {\n const uniqueGroupIds = new Set();\n\n for (const el of selectedElements || allElements) {\n const topMostGroupId = el.groupIds[el.groupIds.length - 1];\n\n if (topMostGroupId) {\n uniqueGroupIds.add(topMostGroupId);\n }\n }\n\n const rejectedGroupIds = new Set();\n\n for (const groupId of uniqueGroupIds) {\n if ((0,_groups__WEBPACK_IMPORTED_MODULE_7__.getElementsInGroup)(allElements, groupId).some(el => (0,_element__WEBPACK_IMPORTED_MODULE_0__.isFrameElement)(el))) {\n rejectedGroupIds.add(groupId);\n }\n }\n\n return (selectedElements || allElements).filter(el => !rejectedGroupIds.has(el.groupIds[el.groupIds.length - 1]));\n};\n/**\n * depending on the appState, return target frame, which is the frame the given element\n * is going to be added to or remove from\n */\n\nconst getTargetFrame = (element, appState) => {\n const _element = (0,_element__WEBPACK_IMPORTED_MODULE_0__.isTextElement)(element) ? (0,_element_textElement__WEBPACK_IMPORTED_MODULE_2__.getContainerElement)(element) || element : element;\n\n return appState.selectedElementIds[_element.id] && appState.selectedElementsAreBeingDragged ? appState.frameToHighlight : getContainingFrame(_element);\n}; // TODO: this a huge bottleneck for large scenes, optimise\n// given an element, return if the element is in some frame\n\nconst isElementInFrame = (element, allElements, appState) => {\n const frame = getTargetFrame(element, appState);\n\n const _element = (0,_element__WEBPACK_IMPORTED_MODULE_0__.isTextElement)(element) ? (0,_element_textElement__WEBPACK_IMPORTED_MODULE_2__.getContainerElement)(element) || element : element;\n\n if (frame) {\n if (_element.groupIds.length === 0) {\n return elementOverlapsWithFrame(_element, frame);\n }\n\n const allElementsInGroup = new Set(_element.groupIds.flatMap(gid => (0,_groups__WEBPACK_IMPORTED_MODULE_7__.getElementsInGroup)(allElements, gid)));\n\n if (appState.editingGroupId && appState.selectedElementsAreBeingDragged) {\n const selectedElements = new Set((0,_scene__WEBPACK_IMPORTED_MODULE_5__.getSelectedElements)(allElements, appState));\n const editingGroupOverlapsFrame = appState.frameToHighlight !== null;\n\n if (editingGroupOverlapsFrame) {\n return true;\n }\n\n selectedElements.forEach(selectedElement => {\n allElementsInGroup.delete(selectedElement);\n });\n }\n\n for (const elementInGroup of allElementsInGroup) {\n if ((0,_element__WEBPACK_IMPORTED_MODULE_0__.isFrameElement)(elementInGroup)) {\n return false;\n }\n }\n\n for (const elementInGroup of allElementsInGroup) {\n if (elementOverlapsWithFrame(elementInGroup, frame)) {\n return true;\n }\n }\n }\n\n return false;\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"../../frame.ts.js","mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAqF;AACxC;AACoC;AACjC;AACQ;AACkB;AAC/B;AACH;AACqC;AAC3C;AACwB,CAAC;;AAEpD;AACP,yBAAyB,kDAAU;;AAEnC;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,UAAU,qEAAa;AACvB;AACA,WAAW;AACX;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,EAAE;;;AAGF;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,8BAA8B,uEAAsB;AACpD,gCAAgC,uEAAsB;AACtD;AACA;AACA;;AAEA;;AAEA;AACO,qFAAqF,kEAA0B;AAC/G;AACP,SAAS,kEAA0B;AACnC;AACO;AACA;AACP,+DAA+D,kEAAwB;AACvF,uDAAuD,yDAAe;AACtE;AACA;AACO;AACP;AACA;AACO;AACP,+BAA+B,kEAAwB;AACvD,SAAS,0DAAmB;AAC5B;AACO;AACP,sDAAsD,2DAAkB;;AAExE;AACA;AACA;;AAEA;AACA;AACO;AACP,sDAAsD,2DAAkB;;AAExE;AACA;AACA;;AAEA;AACA,GAAG;;AAEH;AACA;AACA;;AAEO;AACP;;AAEA;AACA,oBAAoB,wDAAc;;AAElC;AACA;AACA;AACA;;AAEA;AACA;AACO;AACA;AACP;AACA;AACA;AACA,uHAAuH;AACvH;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA,mBAAmB,sEAA6B,+CAA+C;;AAE/F;AACA;AACA,8BAA8B,2DAAkB;;AAEhD;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,aAAa,uDAAa;AAC1B,GAAG;AACH;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,kBAAkB,6DAAc;AAChC;;AAEA;AACA,GAAG;;AAEI;AACP;;AAEA;AACA;;AAEA,6BAA6B,yEAAmB;;AAEhD;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,GAAG,IAAI;AACP,iDAAiD;;AAEjD,iCAAiC,iDAAS;AAC1C;AACA;AACA;;AAEA;AACA,yCAAyC,wDAAc;AACvD;AACA;AACA,QAAQ;AACR;AACA;;AAEA,MAAM,qEAAa;AACnB;AACA,OAAO;AACP;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,GAAG,IAAI;AACP;AACA;AACA;AACA;AACA;AACO;AACP;;AAEA;AACA;AACA;;AAEA,+BAA+B,yEAAmB;;AAElD;AACA;AACA;AACA;AACA;;AAEA;AACA,IAAI,qEAAa;AACjB;AACA,KAAK;AACL;;AAEA,uBAAuB,qDAAY;AACnC;AACA;AACO;AACP;AACA;AACA;AACO;AACP;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA,QAAQ;AACR,wCAAwC,2DAAkB;AAC1D;AACA;AACA;;AAEA;AACA;AACA,4BAA4B,wDAAc;AAC1C;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA,QAAQ,2DAAkB,kCAAkC,wDAAc;AAC1E;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP,mBAAmB,uDAAa,YAAY,yEAAmB;;AAE/D;AACA,GAAG;AACH;;AAEO;AACP;;AAEA,mBAAmB,uDAAa,YAAY,yEAAmB;;AAE/D;AACA;AACA;AACA;;AAEA,wEAAwE,2DAAkB;;AAE1F;AACA,uCAAuC,2DAAmB;AAC1D;;AAEA;AACA;AACA;;AAEA;AACA;AACA,OAAO;AACP;;AAEA;AACA,UAAU,wDAAc;AACxB;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA","sources":["webpack:///../../frame.ts?de46"],"sourcesContent":["import { getCommonBounds, getElementAbsoluteCoords, isTextElement } from \"./element\";\nimport { isPointWithinBounds } from \"./math\";\nimport { getBoundTextElement, getContainerElement } from \"./element/textElement\";\nimport { arrayToMap, findIndex } from \"./utils\";\nimport { mutateElement } from \"./element/mutateElement\";\nimport { getElementsWithinSelection, getSelectedElements } from \"./scene\";\nimport { isFrameElement } from \"./element\";\nimport { moveOneRight } from \"./zindex\";\nimport { getElementsInGroup, selectGroupsFromGivenElements } from \"./groups\";\nimport Scene from \"./scene/Scene\";\nimport { getElementLineSegments } from \"./element/bounds\"; // --------------------------- Frame State ------------------------------------\n\nexport const bindElementsToFramesAfterDuplication = (nextElements, oldElements, oldIdToDuplicatedId) => {\n  const nextElementMap = arrayToMap(nextElements);\n\n  for (const element of oldElements) {\n    if (element.frameId) {\n      // use its frameId to get the new frameId\n      const nextElementId = oldIdToDuplicatedId.get(element.id);\n      const nextFrameId = oldIdToDuplicatedId.get(element.frameId);\n\n      if (nextElementId) {\n        const nextElement = nextElementMap.get(nextElementId);\n\n        if (nextElement) {\n          mutateElement(nextElement, {\n            frameId: nextFrameId !== null && nextFrameId !== void 0 ? nextFrameId : element.frameId\n          }, false);\n        }\n      }\n    }\n  }\n}; // --------------------------- Frame Geometry ---------------------------------\n\nclass Point {\n  constructor(x, y) {\n    this.x = x;\n    this.y = y;\n  }\n\n}\n\nclass LineSegment {\n  constructor(pointA, pointB) {\n    this.first = pointA;\n    this.second = pointB;\n  }\n\n  getBoundingBox() {\n    return [new Point(Math.min(this.first.x, this.second.x), Math.min(this.first.y, this.second.y)), new Point(Math.max(this.first.x, this.second.x), Math.max(this.first.y, this.second.y))];\n  }\n\n} // https://martin-thoma.com/how-to-check-if-two-line-segments-intersect/\n\n\nclass FrameGeometry {\n  static crossProduct(a, b) {\n    return a.x * b.y - b.x * a.y;\n  }\n\n  static doBoundingBoxesIntersect(a, b) {\n    return a[0].x <= b[1].x && a[1].x >= b[0].x && a[0].y <= b[1].y && a[1].y >= b[0].y;\n  }\n\n  static isPointOnLine(a, b) {\n    const aTmp = new LineSegment(new Point(0, 0), new Point(a.second.x - a.first.x, a.second.y - a.first.y));\n    const bTmp = new Point(b.x - a.first.x, b.y - a.first.y);\n    const r = this.crossProduct(aTmp.second, bTmp);\n    return Math.abs(r) < this.EPSILON;\n  }\n\n  static isPointRightOfLine(a, b) {\n    const aTmp = new LineSegment(new Point(0, 0), new Point(a.second.x - a.first.x, a.second.y - a.first.y));\n    const bTmp = new Point(b.x - a.first.x, b.y - a.first.y);\n    return this.crossProduct(aTmp.second, bTmp) < 0;\n  }\n\n  static lineSegmentTouchesOrCrossesLine(a, b) {\n    return this.isPointOnLine(a, b.first) || this.isPointOnLine(a, b.second) || (this.isPointRightOfLine(a, b.first) ? !this.isPointRightOfLine(a, b.second) : this.isPointRightOfLine(a, b.second));\n  }\n\n  static doLineSegmentsIntersect(a, b) {\n    const aSegment = new LineSegment(new Point(a[0][0], a[0][1]), new Point(a[1][0], a[1][1]));\n    const bSegment = new LineSegment(new Point(b[0][0], b[0][1]), new Point(b[1][0], b[1][1]));\n    const box1 = aSegment.getBoundingBox();\n    const box2 = bSegment.getBoundingBox();\n    return this.doBoundingBoxesIntersect(box1, box2) && this.lineSegmentTouchesOrCrossesLine(aSegment, bSegment) && this.lineSegmentTouchesOrCrossesLine(bSegment, aSegment);\n  }\n\n  static isElementIntersectingFrame(element, frame) {\n    const frameLineSegments = getElementLineSegments(frame);\n    const elementLineSegments = getElementLineSegments(element);\n    const intersecting = frameLineSegments.some(frameLineSegment => elementLineSegments.some(elementLineSegment => this.doLineSegmentsIntersect(frameLineSegment, elementLineSegment)));\n    return intersecting;\n  }\n\n}\n\nFrameGeometry.EPSILON = 0.000001;\nexport const getElementsCompletelyInFrame = (elements, frame) => omitGroupsContainingFrames(getElementsWithinSelection(elements, frame, false)).filter(element => element.type !== \"frame\" && !element.frameId || element.frameId === frame.id);\nexport const isElementContainingFrame = (elements, element, frame) => {\n  return getElementsWithinSelection(elements, element).some(e => e.id === frame.id);\n};\nexport const getElementsIntersectingFrame = (elements, frame) => elements.filter(element => FrameGeometry.isElementIntersectingFrame(element, frame));\nexport const elementsAreInFrameBounds = (elements, frame) => {\n  const [selectionX1, selectionY1, selectionX2, selectionY2] = getElementAbsoluteCoords(frame);\n  const [elementX1, elementY1, elementX2, elementY2] = getCommonBounds(elements);\n  return selectionX1 <= elementX1 && selectionY1 <= elementY1 && selectionX2 >= elementX2 && selectionY2 >= elementY2;\n};\nexport const elementOverlapsWithFrame = (element, frame) => {\n  return elementsAreInFrameBounds([element], frame) || FrameGeometry.isElementIntersectingFrame(element, frame) || isElementContainingFrame([frame], element, frame);\n};\nexport const isCursorInFrame = (cursorCoords, frame) => {\n  const [fx1, fy1, fx2, fy2] = getElementAbsoluteCoords(frame);\n  return isPointWithinBounds([fx1, fy1], [cursorCoords.x, cursorCoords.y], [fx2, fy2]);\n};\nexport const groupsAreAtLeastIntersectingTheFrame = (elements, groupIds, frame) => {\n  const elementsInGroup = groupIds.flatMap(groupId => getElementsInGroup(elements, groupId));\n\n  if (elementsInGroup.length === 0) {\n    return true;\n  }\n\n  return !!elementsInGroup.find(element => elementsAreInFrameBounds([element], frame) || FrameGeometry.isElementIntersectingFrame(element, frame));\n};\nexport const groupsAreCompletelyOutOfFrame = (elements, groupIds, frame) => {\n  const elementsInGroup = groupIds.flatMap(groupId => getElementsInGroup(elements, groupId));\n\n  if (elementsInGroup.length === 0) {\n    return true;\n  }\n\n  return elementsInGroup.find(element => elementsAreInFrameBounds([element], frame) || FrameGeometry.isElementIntersectingFrame(element, frame)) === undefined;\n}; // --------------------------- Frame Utils ------------------------------------\n\n/**\n * Returns a map of frameId to frame elements. Includes empty frames.\n */\n\nexport const groupByFrames = elements => {\n  const frameElementsMap = new Map();\n\n  for (const element of elements) {\n    const frameId = isFrameElement(element) ? element.id : element.frameId;\n\n    if (frameId && !frameElementsMap.has(frameId)) {\n      frameElementsMap.set(frameId, getFrameElements(elements, frameId));\n    }\n  }\n\n  return frameElementsMap;\n};\nexport const getFrameElements = (allElements, frameId) => allElements.filter(element => element.frameId === frameId);\nexport const getElementsInResizingFrame = (allElements, frame, appState) => {\n  const prevElementsInFrame = getFrameElements(allElements, frame.id);\n  const nextElementsInFrame = new Set(prevElementsInFrame);\n  const elementsCompletelyInFrame = new Set([...getElementsCompletelyInFrame(allElements, frame), ...prevElementsInFrame.filter(element => isElementContainingFrame(allElements, element, frame))]);\n  const elementsNotCompletelyInFrame = prevElementsInFrame.filter(element => !elementsCompletelyInFrame.has(element)); // for elements that are completely in the frame\n  // if they are part of some groups, then those groups are still\n  // considered to belong to the frame\n\n  const groupsToKeep = new Set(Array.from(elementsCompletelyInFrame).flatMap(element => element.groupIds));\n\n  for (const element of elementsNotCompletelyInFrame) {\n    if (!FrameGeometry.isElementIntersectingFrame(element, frame)) {\n      if (element.groupIds.length === 0) {\n        nextElementsInFrame.delete(element);\n      }\n    } else if (element.groupIds.length > 0) {\n      // group element intersects with the frame, we should keep the groups\n      // that this element is part of\n      for (const id of element.groupIds) {\n        groupsToKeep.add(id);\n      }\n    }\n  }\n\n  for (const element of elementsNotCompletelyInFrame) {\n    if (element.groupIds.length > 0) {\n      let shouldRemoveElement = true;\n\n      for (const id of element.groupIds) {\n        if (groupsToKeep.has(id)) {\n          shouldRemoveElement = false;\n        }\n      }\n\n      if (shouldRemoveElement) {\n        nextElementsInFrame.delete(element);\n      }\n    }\n  }\n\n  const individualElementsCompletelyInFrame = Array.from(elementsCompletelyInFrame).filter(element => element.groupIds.length === 0);\n\n  for (const element of individualElementsCompletelyInFrame) {\n    nextElementsInFrame.add(element);\n  }\n\n  const newGroupElementsCompletelyInFrame = Array.from(elementsCompletelyInFrame).filter(element => element.groupIds.length > 0);\n  const groupIds = selectGroupsFromGivenElements(newGroupElementsCompletelyInFrame, appState); // new group elements\n\n  for (const [id, isSelected] of Object.entries(groupIds)) {\n    if (isSelected) {\n      const elementsInGroup = getElementsInGroup(allElements, id);\n\n      if (elementsAreInFrameBounds(elementsInGroup, frame)) {\n        for (const element of elementsInGroup) {\n          nextElementsInFrame.add(element);\n        }\n      }\n    }\n  }\n\n  return [...nextElementsInFrame].filter(element => {\n    return !(isTextElement(element) && element.containerId);\n  });\n};\nexport const getElementsInNewFrame = (allElements, frame) => {\n  return omitGroupsContainingFrames(allElements, getElementsCompletelyInFrame(allElements, frame));\n};\nexport const getContainingFrame = (element,\n/**\n * Optionally an elements map, in case the elements aren't in the Scene yet.\n * Takes precedence over Scene elements, even if the element exists\n * in Scene elements and not the supplied elements map.\n */\nelementsMap) => {\n  var _a;\n\n  if (element.frameId) {\n    if (elementsMap) {\n      return elementsMap.get(element.frameId) || null;\n    }\n\n    return ((_a = Scene.getScene(element)) === null || _a === void 0 ? void 0 : _a.getElement(element.frameId)) || null;\n  }\n\n  return null;\n}; // --------------------------- Frame Operations -------------------------------\n\nexport const addElementsToFrame = (allElements, elementsToAdd, frame) => {\n  const _elementsToAdd = [];\n\n  for (const element of elementsToAdd) {\n    _elementsToAdd.push(element);\n\n    const boundTextElement = getBoundTextElement(element);\n\n    if (boundTextElement) {\n      _elementsToAdd.push(boundTextElement);\n    }\n  }\n\n  const allElementsIndex = allElements.reduce((acc, element, index) => {\n    acc[element.id] = index;\n    return acc;\n  }, {});\n  const frameIndex = allElementsIndex[frame.id]; // need to be calculated before the mutation below occurs\n\n  const leftFrameBoundaryIndex = findIndex(allElements, e => e.frameId === frame.id);\n  const existingFrameChildren = allElements.filter(element => element.frameId === frame.id);\n  const addedFrameChildren_left = [];\n  const addedFrameChildren_right = [];\n\n  for (const element of omitGroupsContainingFrames(allElements, _elementsToAdd)) {\n    if (element.frameId !== frame.id && !isFrameElement(element)) {\n      if (allElementsIndex[element.id] > frameIndex) {\n        addedFrameChildren_right.push(element);\n      } else {\n        addedFrameChildren_left.push(element);\n      }\n\n      mutateElement(element, {\n        frameId: frame.id\n      }, false);\n    }\n  }\n\n  const frameElement = allElements[frameIndex];\n  const nextFrameChildren = addedFrameChildren_left.concat(existingFrameChildren).concat(addedFrameChildren_right);\n  const nextFrameChildrenMap = nextFrameChildren.reduce((acc, element) => {\n    acc[element.id] = true;\n    return acc;\n  }, {});\n  const nextOtherElements_left = allElements.slice(0, leftFrameBoundaryIndex >= 0 ? leftFrameBoundaryIndex : frameIndex).filter(element => !nextFrameChildrenMap[element.id]);\n  const nextOtherElement_right = allElements.slice(frameIndex + 1).filter(element => !nextFrameChildrenMap[element.id]);\n  const nextElements = nextOtherElements_left.concat(nextFrameChildren).concat([frameElement]).concat(nextOtherElement_right);\n  return nextElements;\n};\nexport const removeElementsFromFrame = (allElements, elementsToRemove, appState) => {\n  const _elementsToRemove = [];\n\n  for (const element of elementsToRemove) {\n    if (element.frameId) {\n      _elementsToRemove.push(element);\n\n      const boundTextElement = getBoundTextElement(element);\n\n      if (boundTextElement) {\n        _elementsToRemove.push(boundTextElement);\n      }\n    }\n  }\n\n  for (const element of _elementsToRemove) {\n    mutateElement(element, {\n      frameId: null\n    }, false);\n  }\n\n  const nextElements = moveOneRight(allElements, appState, Array.from(_elementsToRemove));\n  return nextElements;\n};\nexport const removeAllElementsFromFrame = (allElements, frame, appState) => {\n  const elementsInFrame = getFrameElements(allElements, frame.id);\n  return removeElementsFromFrame(allElements, elementsInFrame, appState);\n};\nexport const replaceAllElementsInFrame = (allElements, nextElementsInFrame, frame, appState) => {\n  return addElementsToFrame(removeAllElementsFromFrame(allElements, frame, appState), nextElementsInFrame, frame);\n};\n/** does not mutate elements, but returns new ones */\n\nexport const updateFrameMembershipOfSelectedElements = (allElements, appState, app) => {\n  const selectedElements = app.scene.getSelectedElements({\n    selectedElementIds: appState.selectedElementIds,\n    // supplying elements explicitly in case we're passed non-state elements\n    elements: allElements\n  });\n  const elementsToFilter = new Set(selectedElements);\n\n  if (appState.editingGroupId) {\n    for (const element of selectedElements) {\n      if (element.groupIds.length === 0) {\n        elementsToFilter.add(element);\n      } else {\n        element.groupIds.flatMap(gid => getElementsInGroup(allElements, gid)).forEach(element => elementsToFilter.add(element));\n      }\n    }\n  }\n\n  const elementsToRemove = new Set();\n  elementsToFilter.forEach(element => {\n    if (element.frameId && !isFrameElement(element) && !isElementInFrame(element, allElements, appState)) {\n      elementsToRemove.add(element);\n    }\n  });\n  return elementsToRemove.size > 0 ? removeElementsFromFrame(allElements, [...elementsToRemove], appState) : allElements;\n};\n/**\n * filters out elements that are inside groups that contain a frame element\n * anywhere in the group tree\n */\n\nexport const omitGroupsContainingFrames = (allElements,\n/** subset of elements you want to filter. Optional perf optimization so we\n * don't have to filter all elements unnecessarily\n */\nselectedElements) => {\n  const uniqueGroupIds = new Set();\n\n  for (const el of selectedElements || allElements) {\n    const topMostGroupId = el.groupIds[el.groupIds.length - 1];\n\n    if (topMostGroupId) {\n      uniqueGroupIds.add(topMostGroupId);\n    }\n  }\n\n  const rejectedGroupIds = new Set();\n\n  for (const groupId of uniqueGroupIds) {\n    if (getElementsInGroup(allElements, groupId).some(el => isFrameElement(el))) {\n      rejectedGroupIds.add(groupId);\n    }\n  }\n\n  return (selectedElements || allElements).filter(el => !rejectedGroupIds.has(el.groupIds[el.groupIds.length - 1]));\n};\n/**\n * depending on the appState, return target frame, which is the frame the given element\n * is going to be added to or remove from\n */\n\nexport const getTargetFrame = (element, appState) => {\n  const _element = isTextElement(element) ? getContainerElement(element) || element : element;\n\n  return appState.selectedElementIds[_element.id] && appState.selectedElementsAreBeingDragged ? appState.frameToHighlight : getContainingFrame(_element);\n}; // TODO: this a huge bottleneck for large scenes, optimise\n// given an element, return if the element is in some frame\n\nexport const isElementInFrame = (element, allElements, appState) => {\n  const frame = getTargetFrame(element, appState);\n\n  const _element = isTextElement(element) ? getContainerElement(element) || element : element;\n\n  if (frame) {\n    if (_element.groupIds.length === 0) {\n      return elementOverlapsWithFrame(_element, frame);\n    }\n\n    const allElementsInGroup = new Set(_element.groupIds.flatMap(gid => getElementsInGroup(allElements, gid)));\n\n    if (appState.editingGroupId && appState.selectedElementsAreBeingDragged) {\n      const selectedElements = new Set(getSelectedElements(allElements, appState));\n      const editingGroupOverlapsFrame = appState.frameToHighlight !== null;\n\n      if (editingGroupOverlapsFrame) {\n        return true;\n      }\n\n      selectedElements.forEach(selectedElement => {\n        allElementsInGroup.delete(selectedElement);\n      });\n    }\n\n    for (const elementInGroup of allElementsInGroup) {\n      if (isFrameElement(elementInGroup)) {\n        return false;\n      }\n    }\n\n    for (const elementInGroup of allElementsInGroup) {\n      if (elementOverlapsWithFrame(elementInGroup, frame)) {\n        return true;\n      }\n    }\n  }\n\n  return false;\n};"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///../../frame.ts\n");
|
|
3995
|
+
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"addElementsToFrame\": () => (/* binding */ addElementsToFrame),\n/* harmony export */ \"bindElementsToFramesAfterDuplication\": () => (/* binding */ bindElementsToFramesAfterDuplication),\n/* harmony export */ \"elementOverlapsWithFrame\": () => (/* binding */ elementOverlapsWithFrame),\n/* harmony export */ \"elementsAreInFrameBounds\": () => (/* binding */ elementsAreInFrameBounds),\n/* harmony export */ \"getContainingFrame\": () => (/* binding */ getContainingFrame),\n/* harmony export */ \"getElementsCompletelyInFrame\": () => (/* binding */ getElementsCompletelyInFrame),\n/* harmony export */ \"getElementsInNewFrame\": () => (/* binding */ getElementsInNewFrame),\n/* harmony export */ \"getElementsInResizingFrame\": () => (/* binding */ getElementsInResizingFrame),\n/* harmony export */ \"getElementsIntersectingFrame\": () => (/* binding */ getElementsIntersectingFrame),\n/* harmony export */ \"getFrameElements\": () => (/* binding */ getFrameElements),\n/* harmony export */ \"getTargetFrame\": () => (/* binding */ getTargetFrame),\n/* harmony export */ \"groupByFrames\": () => (/* binding */ groupByFrames),\n/* harmony export */ \"groupsAreAtLeastIntersectingTheFrame\": () => (/* binding */ groupsAreAtLeastIntersectingTheFrame),\n/* harmony export */ \"groupsAreCompletelyOutOfFrame\": () => (/* binding */ groupsAreCompletelyOutOfFrame),\n/* harmony export */ \"isCursorInFrame\": () => (/* binding */ isCursorInFrame),\n/* harmony export */ \"isElementContainingFrame\": () => (/* binding */ isElementContainingFrame),\n/* harmony export */ \"isElementInFrame\": () => (/* binding */ isElementInFrame),\n/* harmony export */ \"omitGroupsContainingFrames\": () => (/* binding */ omitGroupsContainingFrames),\n/* harmony export */ \"removeAllElementsFromFrame\": () => (/* binding */ removeAllElementsFromFrame),\n/* harmony export */ \"removeElementsFromFrame\": () => (/* binding */ removeElementsFromFrame),\n/* harmony export */ \"replaceAllElementsInFrame\": () => (/* binding */ replaceAllElementsInFrame),\n/* harmony export */ \"updateFrameMembershipOfSelectedElements\": () => (/* binding */ updateFrameMembershipOfSelectedElements)\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 _element_textElement__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./element/textElement */ \"../../element/textElement.ts\");\n/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./utils */ \"../../utils.ts\");\n/* harmony import */ var _element_mutateElement__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./element/mutateElement */ \"../../element/mutateElement.ts\");\n/* harmony import */ var _scene__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./scene */ \"../../scene/index.ts\");\n/* harmony import */ var _zindex__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./zindex */ \"../../zindex.ts\");\n/* harmony import */ var _groups__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ./groups */ \"../../groups.ts\");\n/* harmony import */ var _scene_Scene__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ./scene/Scene */ \"../../scene/Scene.ts\");\n/* harmony import */ var _element_bounds__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! ./element/bounds */ \"../../element/bounds.ts\");\n\n\n\n\n\n\n\n\n\n\n // --------------------------- Frame State ------------------------------------\n\nconst bindElementsToFramesAfterDuplication = (nextElements, oldElements, oldIdToDuplicatedId) => {\n const nextElementMap = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.arrayToMap)(nextElements);\n\n for (const element of oldElements) {\n if (element.frameId) {\n // use its frameId to get the new frameId\n const nextElementId = oldIdToDuplicatedId.get(element.id);\n const nextFrameId = oldIdToDuplicatedId.get(element.frameId);\n\n if (nextElementId) {\n const nextElement = nextElementMap.get(nextElementId);\n\n if (nextElement) {\n (0,_element_mutateElement__WEBPACK_IMPORTED_MODULE_4__.mutateElement)(nextElement, {\n frameId: nextFrameId !== null && nextFrameId !== void 0 ? nextFrameId : element.frameId\n }, false);\n }\n }\n }\n }\n}; // --------------------------- Frame Geometry ---------------------------------\n\nclass Point {\n constructor(x, y) {\n this.x = x;\n this.y = y;\n }\n\n}\n\nclass LineSegment {\n constructor(pointA, pointB) {\n this.first = pointA;\n this.second = pointB;\n }\n\n getBoundingBox() {\n return [new Point(Math.min(this.first.x, this.second.x), Math.min(this.first.y, this.second.y)), new Point(Math.max(this.first.x, this.second.x), Math.max(this.first.y, this.second.y))];\n }\n\n} // https://martin-thoma.com/how-to-check-if-two-line-segments-intersect/\n\n\nclass FrameGeometry {\n static crossProduct(a, b) {\n return a.x * b.y - b.x * a.y;\n }\n\n static doBoundingBoxesIntersect(a, b) {\n return a[0].x <= b[1].x && a[1].x >= b[0].x && a[0].y <= b[1].y && a[1].y >= b[0].y;\n }\n\n static isPointOnLine(a, b) {\n const aTmp = new LineSegment(new Point(0, 0), new Point(a.second.x - a.first.x, a.second.y - a.first.y));\n const bTmp = new Point(b.x - a.first.x, b.y - a.first.y);\n const r = this.crossProduct(aTmp.second, bTmp);\n return Math.abs(r) < this.EPSILON;\n }\n\n static isPointRightOfLine(a, b) {\n const aTmp = new LineSegment(new Point(0, 0), new Point(a.second.x - a.first.x, a.second.y - a.first.y));\n const bTmp = new Point(b.x - a.first.x, b.y - a.first.y);\n return this.crossProduct(aTmp.second, bTmp) < 0;\n }\n\n static lineSegmentTouchesOrCrossesLine(a, b) {\n return this.isPointOnLine(a, b.first) || this.isPointOnLine(a, b.second) || (this.isPointRightOfLine(a, b.first) ? !this.isPointRightOfLine(a, b.second) : this.isPointRightOfLine(a, b.second));\n }\n\n static doLineSegmentsIntersect(a, b) {\n const aSegment = new LineSegment(new Point(a[0][0], a[0][1]), new Point(a[1][0], a[1][1]));\n const bSegment = new LineSegment(new Point(b[0][0], b[0][1]), new Point(b[1][0], b[1][1]));\n const box1 = aSegment.getBoundingBox();\n const box2 = bSegment.getBoundingBox();\n return this.doBoundingBoxesIntersect(box1, box2) && this.lineSegmentTouchesOrCrossesLine(aSegment, bSegment) && this.lineSegmentTouchesOrCrossesLine(bSegment, aSegment);\n }\n\n static isElementIntersectingFrame(element, frame) {\n const frameLineSegments = (0,_element_bounds__WEBPACK_IMPORTED_MODULE_9__.getElementLineSegments)(frame);\n const elementLineSegments = (0,_element_bounds__WEBPACK_IMPORTED_MODULE_9__.getElementLineSegments)(element);\n const intersecting = frameLineSegments.some(frameLineSegment => elementLineSegments.some(elementLineSegment => this.doLineSegmentsIntersect(frameLineSegment, elementLineSegment)));\n return intersecting;\n }\n\n}\n\nFrameGeometry.EPSILON = 0.000001;\nconst getElementsCompletelyInFrame = (elements, frame) => omitGroupsContainingFrames((0,_scene__WEBPACK_IMPORTED_MODULE_5__.getElementsWithinSelection)(elements, frame, false)).filter(element => element.type !== \"frame\" && !element.frameId || element.frameId === frame.id);\nconst isElementContainingFrame = (elements, element, frame) => {\n return (0,_scene__WEBPACK_IMPORTED_MODULE_5__.getElementsWithinSelection)(elements, element).some(e => e.id === frame.id);\n};\nconst getElementsIntersectingFrame = (elements, frame) => elements.filter(element => FrameGeometry.isElementIntersectingFrame(element, frame));\nconst elementsAreInFrameBounds = (elements, frame) => {\n const [selectionX1, selectionY1, selectionX2, selectionY2] = (0,_element__WEBPACK_IMPORTED_MODULE_0__.getElementAbsoluteCoords)(frame);\n const [elementX1, elementY1, elementX2, elementY2] = (0,_element__WEBPACK_IMPORTED_MODULE_0__.getCommonBounds)(elements);\n return selectionX1 <= elementX1 && selectionY1 <= elementY1 && selectionX2 >= elementX2 && selectionY2 >= elementY2;\n};\nconst elementOverlapsWithFrame = (element, frame) => {\n return elementsAreInFrameBounds([element], frame) || FrameGeometry.isElementIntersectingFrame(element, frame) || isElementContainingFrame([frame], element, frame);\n};\nconst isCursorInFrame = (cursorCoords, frame) => {\n const [fx1, fy1, fx2, fy2] = (0,_element__WEBPACK_IMPORTED_MODULE_0__.getElementAbsoluteCoords)(frame);\n return (0,_math__WEBPACK_IMPORTED_MODULE_1__.isPointWithinBounds)([fx1, fy1], [cursorCoords.x, cursorCoords.y], [fx2, fy2]);\n};\nconst groupsAreAtLeastIntersectingTheFrame = (elements, groupIds, frame) => {\n const elementsInGroup = groupIds.flatMap(groupId => (0,_groups__WEBPACK_IMPORTED_MODULE_7__.getElementsInGroup)(elements, groupId));\n\n if (elementsInGroup.length === 0) {\n return true;\n }\n\n return !!elementsInGroup.find(element => elementsAreInFrameBounds([element], frame) || FrameGeometry.isElementIntersectingFrame(element, frame));\n};\nconst groupsAreCompletelyOutOfFrame = (elements, groupIds, frame) => {\n const elementsInGroup = groupIds.flatMap(groupId => (0,_groups__WEBPACK_IMPORTED_MODULE_7__.getElementsInGroup)(elements, groupId));\n\n if (elementsInGroup.length === 0) {\n return true;\n }\n\n return elementsInGroup.find(element => elementsAreInFrameBounds([element], frame) || FrameGeometry.isElementIntersectingFrame(element, frame)) === undefined;\n}; // --------------------------- Frame Utils ------------------------------------\n\n/**\n * Returns a map of frameId to frame elements. Includes empty frames.\n */\n\nconst groupByFrames = elements => {\n const frameElementsMap = new Map();\n\n for (const element of elements) {\n const frameId = (0,_element__WEBPACK_IMPORTED_MODULE_0__.isFrameElement)(element) ? element.id : element.frameId;\n\n if (frameId && !frameElementsMap.has(frameId)) {\n frameElementsMap.set(frameId, getFrameElements(elements, frameId));\n }\n }\n\n return frameElementsMap;\n};\nconst getFrameElements = (allElements, frameId) => allElements.filter(element => element.frameId === frameId);\nconst getElementsInResizingFrame = (allElements, frame, appState) => {\n const prevElementsInFrame = getFrameElements(allElements, frame.id);\n const nextElementsInFrame = new Set(prevElementsInFrame);\n const elementsCompletelyInFrame = new Set([...getElementsCompletelyInFrame(allElements, frame), ...prevElementsInFrame.filter(element => isElementContainingFrame(allElements, element, frame))]);\n const elementsNotCompletelyInFrame = prevElementsInFrame.filter(element => !elementsCompletelyInFrame.has(element)); // for elements that are completely in the frame\n // if they are part of some groups, then those groups are still\n // considered to belong to the frame\n\n const groupsToKeep = new Set(Array.from(elementsCompletelyInFrame).flatMap(element => element.groupIds));\n\n for (const element of elementsNotCompletelyInFrame) {\n if (!FrameGeometry.isElementIntersectingFrame(element, frame)) {\n if (element.groupIds.length === 0) {\n nextElementsInFrame.delete(element);\n }\n } else if (element.groupIds.length > 0) {\n // group element intersects with the frame, we should keep the groups\n // that this element is part of\n for (const id of element.groupIds) {\n groupsToKeep.add(id);\n }\n }\n }\n\n for (const element of elementsNotCompletelyInFrame) {\n if (element.groupIds.length > 0) {\n let shouldRemoveElement = true;\n\n for (const id of element.groupIds) {\n if (groupsToKeep.has(id)) {\n shouldRemoveElement = false;\n }\n }\n\n if (shouldRemoveElement) {\n nextElementsInFrame.delete(element);\n }\n }\n }\n\n const individualElementsCompletelyInFrame = Array.from(elementsCompletelyInFrame).filter(element => element.groupIds.length === 0);\n\n for (const element of individualElementsCompletelyInFrame) {\n nextElementsInFrame.add(element);\n }\n\n const newGroupElementsCompletelyInFrame = Array.from(elementsCompletelyInFrame).filter(element => element.groupIds.length > 0);\n const groupIds = (0,_groups__WEBPACK_IMPORTED_MODULE_7__.selectGroupsFromGivenElements)(newGroupElementsCompletelyInFrame, appState); // new group elements\n\n for (const [id, isSelected] of Object.entries(groupIds)) {\n if (isSelected) {\n const elementsInGroup = (0,_groups__WEBPACK_IMPORTED_MODULE_7__.getElementsInGroup)(allElements, id);\n\n if (elementsAreInFrameBounds(elementsInGroup, frame)) {\n for (const element of elementsInGroup) {\n nextElementsInFrame.add(element);\n }\n }\n }\n }\n\n return [...nextElementsInFrame].filter(element => {\n return !((0,_element__WEBPACK_IMPORTED_MODULE_0__.isTextElement)(element) && element.containerId);\n });\n};\nconst getElementsInNewFrame = (allElements, frame) => {\n return omitGroupsContainingFrames(allElements, getElementsCompletelyInFrame(allElements, frame));\n};\nconst getContainingFrame = (element,\n/**\n * Optionally an elements map, in case the elements aren't in the Scene yet.\n * Takes precedence over Scene elements, even if the element exists\n * in Scene elements and not the supplied elements map.\n */\nelementsMap) => {\n var _a;\n\n if (element.frameId) {\n if (elementsMap) {\n return elementsMap.get(element.frameId) || null;\n }\n\n return ((_a = _scene_Scene__WEBPACK_IMPORTED_MODULE_8__[\"default\"].getScene(element)) === null || _a === void 0 ? void 0 : _a.getElement(element.frameId)) || null;\n }\n\n return null;\n}; // --------------------------- Frame Operations -------------------------------\n\nconst addElementsToFrame = (allElements, elementsToAdd, frame) => {\n const currTargetFrameChildrenMap = new Map(allElements.reduce((acc, element) => {\n if (element.frameId === frame.id) {\n acc.push([element.id, element]);\n }\n\n return acc;\n }, []));\n const suppliedElementsToAddSet = new Set(elementsToAdd.map(el => el.id));\n const finalElementsToAdd = []; // - add bound text elements if not already in the array\n // - filter out elements that are already in the frame\n\n for (const element of omitGroupsContainingFrames(allElements, elementsToAdd)) {\n if (!currTargetFrameChildrenMap.has(element.id)) {\n finalElementsToAdd.push(element);\n }\n\n const boundTextElement = (0,_element_textElement__WEBPACK_IMPORTED_MODULE_2__.getBoundTextElement)(element);\n\n if (boundTextElement && !suppliedElementsToAddSet.has(boundTextElement.id) && !currTargetFrameChildrenMap.has(boundTextElement.id)) {\n finalElementsToAdd.push(boundTextElement);\n }\n }\n\n const finalElementsToAddSet = new Set(finalElementsToAdd.map(el => el.id));\n const nextElements = [];\n const processedElements = new Set();\n\n for (const element of allElements) {\n if (processedElements.has(element.id)) {\n continue;\n }\n\n processedElements.add(element.id);\n\n if (finalElementsToAddSet.has(element.id) || element.frameId && element.frameId === frame.id) {\n // will be added in bulk once we process target frame\n continue;\n } // target frame\n\n\n if (element.id === frame.id) {\n const currFrameChildren = getFrameElements(allElements, frame.id);\n currFrameChildren.forEach(child => {\n processedElements.add(child.id);\n }); // console.log(currFrameChildren, finalElementsToAdd, element);\n\n nextElements.push(...currFrameChildren, ...finalElementsToAdd, element);\n continue;\n } // console.log(\"(2)\", element.frameId);\n\n\n nextElements.push(element);\n }\n\n for (const element of finalElementsToAdd) {\n (0,_element_mutateElement__WEBPACK_IMPORTED_MODULE_4__.mutateElement)(element, {\n frameId: frame.id\n }, false);\n }\n\n return nextElements;\n};\nconst removeElementsFromFrame = (allElements, elementsToRemove, appState) => {\n const _elementsToRemove = [];\n\n for (const element of elementsToRemove) {\n if (element.frameId) {\n _elementsToRemove.push(element);\n\n const boundTextElement = (0,_element_textElement__WEBPACK_IMPORTED_MODULE_2__.getBoundTextElement)(element);\n\n if (boundTextElement) {\n _elementsToRemove.push(boundTextElement);\n }\n }\n }\n\n for (const element of _elementsToRemove) {\n (0,_element_mutateElement__WEBPACK_IMPORTED_MODULE_4__.mutateElement)(element, {\n frameId: null\n }, false);\n }\n\n const nextElements = (0,_zindex__WEBPACK_IMPORTED_MODULE_6__.moveOneRight)(allElements, appState, Array.from(_elementsToRemove));\n return nextElements;\n};\nconst removeAllElementsFromFrame = (allElements, frame, appState) => {\n const elementsInFrame = getFrameElements(allElements, frame.id);\n return removeElementsFromFrame(allElements, elementsInFrame, appState);\n};\nconst replaceAllElementsInFrame = (allElements, nextElementsInFrame, frame, appState) => {\n return addElementsToFrame(removeAllElementsFromFrame(allElements, frame, appState), nextElementsInFrame, frame);\n};\n/** does not mutate elements, but returns new ones */\n\nconst updateFrameMembershipOfSelectedElements = (allElements, appState, app) => {\n const selectedElements = app.scene.getSelectedElements({\n selectedElementIds: appState.selectedElementIds,\n // supplying elements explicitly in case we're passed non-state elements\n elements: allElements\n });\n const elementsToFilter = new Set(selectedElements);\n\n if (appState.editingGroupId) {\n for (const element of selectedElements) {\n if (element.groupIds.length === 0) {\n elementsToFilter.add(element);\n } else {\n element.groupIds.flatMap(gid => (0,_groups__WEBPACK_IMPORTED_MODULE_7__.getElementsInGroup)(allElements, gid)).forEach(element => elementsToFilter.add(element));\n }\n }\n }\n\n const elementsToRemove = new Set();\n elementsToFilter.forEach(element => {\n if (element.frameId && !(0,_element__WEBPACK_IMPORTED_MODULE_0__.isFrameElement)(element) && !isElementInFrame(element, allElements, appState)) {\n elementsToRemove.add(element);\n }\n });\n return elementsToRemove.size > 0 ? removeElementsFromFrame(allElements, [...elementsToRemove], appState) : allElements;\n};\n/**\n * filters out elements that are inside groups that contain a frame element\n * anywhere in the group tree\n */\n\nconst omitGroupsContainingFrames = (allElements,\n/** subset of elements you want to filter. Optional perf optimization so we\n * don't have to filter all elements unnecessarily\n */\nselectedElements) => {\n const uniqueGroupIds = new Set();\n\n for (const el of selectedElements || allElements) {\n const topMostGroupId = el.groupIds[el.groupIds.length - 1];\n\n if (topMostGroupId) {\n uniqueGroupIds.add(topMostGroupId);\n }\n }\n\n const rejectedGroupIds = new Set();\n\n for (const groupId of uniqueGroupIds) {\n if ((0,_groups__WEBPACK_IMPORTED_MODULE_7__.getElementsInGroup)(allElements, groupId).some(el => (0,_element__WEBPACK_IMPORTED_MODULE_0__.isFrameElement)(el))) {\n rejectedGroupIds.add(groupId);\n }\n }\n\n return (selectedElements || allElements).filter(el => !rejectedGroupIds.has(el.groupIds[el.groupIds.length - 1]));\n};\n/**\n * depending on the appState, return target frame, which is the frame the given element\n * is going to be added to or remove from\n */\n\nconst getTargetFrame = (element, appState) => {\n const _element = (0,_element__WEBPACK_IMPORTED_MODULE_0__.isTextElement)(element) ? (0,_element_textElement__WEBPACK_IMPORTED_MODULE_2__.getContainerElement)(element) || element : element;\n\n return appState.selectedElementIds[_element.id] && appState.selectedElementsAreBeingDragged ? appState.frameToHighlight : getContainingFrame(_element);\n}; // TODO: this a huge bottleneck for large scenes, optimise\n// given an element, return if the element is in some frame\n\nconst isElementInFrame = (element, allElements, appState) => {\n const frame = getTargetFrame(element, appState);\n\n const _element = (0,_element__WEBPACK_IMPORTED_MODULE_0__.isTextElement)(element) ? (0,_element_textElement__WEBPACK_IMPORTED_MODULE_2__.getContainerElement)(element) || element : element;\n\n if (frame) {\n if (_element.groupIds.length === 0) {\n return elementOverlapsWithFrame(_element, frame);\n }\n\n const allElementsInGroup = new Set(_element.groupIds.flatMap(gid => (0,_groups__WEBPACK_IMPORTED_MODULE_7__.getElementsInGroup)(allElements, gid)));\n\n if (appState.editingGroupId && appState.selectedElementsAreBeingDragged) {\n const selectedElements = new Set((0,_scene__WEBPACK_IMPORTED_MODULE_5__.getSelectedElements)(allElements, appState));\n const editingGroupOverlapsFrame = appState.frameToHighlight !== null;\n\n if (editingGroupOverlapsFrame) {\n return true;\n }\n\n selectedElements.forEach(selectedElement => {\n allElementsInGroup.delete(selectedElement);\n });\n }\n\n for (const elementInGroup of allElementsInGroup) {\n if ((0,_element__WEBPACK_IMPORTED_MODULE_0__.isFrameElement)(elementInGroup)) {\n return false;\n }\n }\n\n for (const elementInGroup of allElementsInGroup) {\n if (elementOverlapsWithFrame(elementInGroup, frame)) {\n return true;\n }\n }\n }\n\n return false;\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"../../frame.ts.js","mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAqF;AACxC;AACoC;AAC5C;AACmB;AACkB;AAC/B;AACH;AACqC;AAC3C;AACwB,CAAC;;AAEpD;AACP,yBAAyB,kDAAU;;AAEnC;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,UAAU,qEAAa;AACvB;AACA,WAAW;AACX;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,EAAE;;;AAGF;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,8BAA8B,uEAAsB;AACpD,gCAAgC,uEAAsB;AACtD;AACA;AACA;;AAEA;;AAEA;AACO,qFAAqF,kEAA0B;AAC/G;AACP,SAAS,kEAA0B;AACnC;AACO;AACA;AACP,+DAA+D,kEAAwB;AACvF,uDAAuD,yDAAe;AACtE;AACA;AACO;AACP;AACA;AACO;AACP,+BAA+B,kEAAwB;AACvD,SAAS,0DAAmB;AAC5B;AACO;AACP,sDAAsD,2DAAkB;;AAExE;AACA;AACA;;AAEA;AACA;AACO;AACP,sDAAsD,2DAAkB;;AAExE;AACA;AACA;;AAEA;AACA,GAAG;;AAEH;AACA;AACA;;AAEO;AACP;;AAEA;AACA,oBAAoB,wDAAc;;AAElC;AACA;AACA;AACA;;AAEA;AACA;AACO;AACA;AACP;AACA;AACA;AACA,uHAAuH;AACvH;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA,mBAAmB,sEAA6B,+CAA+C;;AAE/F;AACA;AACA,8BAA8B,2DAAkB;;AAEhD;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,aAAa,uDAAa;AAC1B,GAAG;AACH;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,kBAAkB,6DAAc;AAChC;;AAEA;AACA,GAAG;;AAEI;AACP;AACA;AACA;AACA;;AAEA;AACA,GAAG;AACH;AACA,iCAAiC;AACjC;;AAEA;AACA;AACA;AACA;;AAEA,6BAA6B,yEAAmB;;AAEhD;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA,MAAM;;;AAGN;AACA;AACA;AACA;AACA,OAAO,GAAG;;AAEV;AACA;AACA,MAAM;;;AAGN;AACA;;AAEA;AACA,IAAI,qEAAa;AACjB;AACA,KAAK;AACL;;AAEA;AACA;AACO;AACP;;AAEA;AACA;AACA;;AAEA,+BAA+B,yEAAmB;;AAElD;AACA;AACA;AACA;AACA;;AAEA;AACA,IAAI,qEAAa;AACjB;AACA,KAAK;AACL;;AAEA,uBAAuB,qDAAY;AACnC;AACA;AACO;AACP;AACA;AACA;AACO;AACP;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA,QAAQ;AACR,wCAAwC,2DAAkB;AAC1D;AACA;AACA;;AAEA;AACA;AACA,4BAA4B,wDAAc;AAC1C;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA,QAAQ,2DAAkB,kCAAkC,wDAAc;AAC1E;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP,mBAAmB,uDAAa,YAAY,yEAAmB;;AAE/D;AACA,GAAG;AACH;;AAEO;AACP;;AAEA,mBAAmB,uDAAa,YAAY,yEAAmB;;AAE/D;AACA;AACA;AACA;;AAEA,wEAAwE,2DAAkB;;AAE1F;AACA,uCAAuC,2DAAmB;AAC1D;;AAEA;AACA;AACA;;AAEA;AACA;AACA,OAAO;AACP;;AAEA;AACA,UAAU,wDAAc;AACxB;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA","sources":["webpack:///../../frame.ts?de46"],"sourcesContent":["import { getCommonBounds, getElementAbsoluteCoords, isTextElement } from \"./element\";\nimport { isPointWithinBounds } from \"./math\";\nimport { getBoundTextElement, getContainerElement } from \"./element/textElement\";\nimport { arrayToMap } from \"./utils\";\nimport { mutateElement } from \"./element/mutateElement\";\nimport { getElementsWithinSelection, getSelectedElements } from \"./scene\";\nimport { isFrameElement } from \"./element\";\nimport { moveOneRight } from \"./zindex\";\nimport { getElementsInGroup, selectGroupsFromGivenElements } from \"./groups\";\nimport Scene from \"./scene/Scene\";\nimport { getElementLineSegments } from \"./element/bounds\"; // --------------------------- Frame State ------------------------------------\n\nexport const bindElementsToFramesAfterDuplication = (nextElements, oldElements, oldIdToDuplicatedId) => {\n  const nextElementMap = arrayToMap(nextElements);\n\n  for (const element of oldElements) {\n    if (element.frameId) {\n      // use its frameId to get the new frameId\n      const nextElementId = oldIdToDuplicatedId.get(element.id);\n      const nextFrameId = oldIdToDuplicatedId.get(element.frameId);\n\n      if (nextElementId) {\n        const nextElement = nextElementMap.get(nextElementId);\n\n        if (nextElement) {\n          mutateElement(nextElement, {\n            frameId: nextFrameId !== null && nextFrameId !== void 0 ? nextFrameId : element.frameId\n          }, false);\n        }\n      }\n    }\n  }\n}; // --------------------------- Frame Geometry ---------------------------------\n\nclass Point {\n  constructor(x, y) {\n    this.x = x;\n    this.y = y;\n  }\n\n}\n\nclass LineSegment {\n  constructor(pointA, pointB) {\n    this.first = pointA;\n    this.second = pointB;\n  }\n\n  getBoundingBox() {\n    return [new Point(Math.min(this.first.x, this.second.x), Math.min(this.first.y, this.second.y)), new Point(Math.max(this.first.x, this.second.x), Math.max(this.first.y, this.second.y))];\n  }\n\n} // https://martin-thoma.com/how-to-check-if-two-line-segments-intersect/\n\n\nclass FrameGeometry {\n  static crossProduct(a, b) {\n    return a.x * b.y - b.x * a.y;\n  }\n\n  static doBoundingBoxesIntersect(a, b) {\n    return a[0].x <= b[1].x && a[1].x >= b[0].x && a[0].y <= b[1].y && a[1].y >= b[0].y;\n  }\n\n  static isPointOnLine(a, b) {\n    const aTmp = new LineSegment(new Point(0, 0), new Point(a.second.x - a.first.x, a.second.y - a.first.y));\n    const bTmp = new Point(b.x - a.first.x, b.y - a.first.y);\n    const r = this.crossProduct(aTmp.second, bTmp);\n    return Math.abs(r) < this.EPSILON;\n  }\n\n  static isPointRightOfLine(a, b) {\n    const aTmp = new LineSegment(new Point(0, 0), new Point(a.second.x - a.first.x, a.second.y - a.first.y));\n    const bTmp = new Point(b.x - a.first.x, b.y - a.first.y);\n    return this.crossProduct(aTmp.second, bTmp) < 0;\n  }\n\n  static lineSegmentTouchesOrCrossesLine(a, b) {\n    return this.isPointOnLine(a, b.first) || this.isPointOnLine(a, b.second) || (this.isPointRightOfLine(a, b.first) ? !this.isPointRightOfLine(a, b.second) : this.isPointRightOfLine(a, b.second));\n  }\n\n  static doLineSegmentsIntersect(a, b) {\n    const aSegment = new LineSegment(new Point(a[0][0], a[0][1]), new Point(a[1][0], a[1][1]));\n    const bSegment = new LineSegment(new Point(b[0][0], b[0][1]), new Point(b[1][0], b[1][1]));\n    const box1 = aSegment.getBoundingBox();\n    const box2 = bSegment.getBoundingBox();\n    return this.doBoundingBoxesIntersect(box1, box2) && this.lineSegmentTouchesOrCrossesLine(aSegment, bSegment) && this.lineSegmentTouchesOrCrossesLine(bSegment, aSegment);\n  }\n\n  static isElementIntersectingFrame(element, frame) {\n    const frameLineSegments = getElementLineSegments(frame);\n    const elementLineSegments = getElementLineSegments(element);\n    const intersecting = frameLineSegments.some(frameLineSegment => elementLineSegments.some(elementLineSegment => this.doLineSegmentsIntersect(frameLineSegment, elementLineSegment)));\n    return intersecting;\n  }\n\n}\n\nFrameGeometry.EPSILON = 0.000001;\nexport const getElementsCompletelyInFrame = (elements, frame) => omitGroupsContainingFrames(getElementsWithinSelection(elements, frame, false)).filter(element => element.type !== \"frame\" && !element.frameId || element.frameId === frame.id);\nexport const isElementContainingFrame = (elements, element, frame) => {\n  return getElementsWithinSelection(elements, element).some(e => e.id === frame.id);\n};\nexport const getElementsIntersectingFrame = (elements, frame) => elements.filter(element => FrameGeometry.isElementIntersectingFrame(element, frame));\nexport const elementsAreInFrameBounds = (elements, frame) => {\n  const [selectionX1, selectionY1, selectionX2, selectionY2] = getElementAbsoluteCoords(frame);\n  const [elementX1, elementY1, elementX2, elementY2] = getCommonBounds(elements);\n  return selectionX1 <= elementX1 && selectionY1 <= elementY1 && selectionX2 >= elementX2 && selectionY2 >= elementY2;\n};\nexport const elementOverlapsWithFrame = (element, frame) => {\n  return elementsAreInFrameBounds([element], frame) || FrameGeometry.isElementIntersectingFrame(element, frame) || isElementContainingFrame([frame], element, frame);\n};\nexport const isCursorInFrame = (cursorCoords, frame) => {\n  const [fx1, fy1, fx2, fy2] = getElementAbsoluteCoords(frame);\n  return isPointWithinBounds([fx1, fy1], [cursorCoords.x, cursorCoords.y], [fx2, fy2]);\n};\nexport const groupsAreAtLeastIntersectingTheFrame = (elements, groupIds, frame) => {\n  const elementsInGroup = groupIds.flatMap(groupId => getElementsInGroup(elements, groupId));\n\n  if (elementsInGroup.length === 0) {\n    return true;\n  }\n\n  return !!elementsInGroup.find(element => elementsAreInFrameBounds([element], frame) || FrameGeometry.isElementIntersectingFrame(element, frame));\n};\nexport const groupsAreCompletelyOutOfFrame = (elements, groupIds, frame) => {\n  const elementsInGroup = groupIds.flatMap(groupId => getElementsInGroup(elements, groupId));\n\n  if (elementsInGroup.length === 0) {\n    return true;\n  }\n\n  return elementsInGroup.find(element => elementsAreInFrameBounds([element], frame) || FrameGeometry.isElementIntersectingFrame(element, frame)) === undefined;\n}; // --------------------------- Frame Utils ------------------------------------\n\n/**\n * Returns a map of frameId to frame elements. Includes empty frames.\n */\n\nexport const groupByFrames = elements => {\n  const frameElementsMap = new Map();\n\n  for (const element of elements) {\n    const frameId = isFrameElement(element) ? element.id : element.frameId;\n\n    if (frameId && !frameElementsMap.has(frameId)) {\n      frameElementsMap.set(frameId, getFrameElements(elements, frameId));\n    }\n  }\n\n  return frameElementsMap;\n};\nexport const getFrameElements = (allElements, frameId) => allElements.filter(element => element.frameId === frameId);\nexport const getElementsInResizingFrame = (allElements, frame, appState) => {\n  const prevElementsInFrame = getFrameElements(allElements, frame.id);\n  const nextElementsInFrame = new Set(prevElementsInFrame);\n  const elementsCompletelyInFrame = new Set([...getElementsCompletelyInFrame(allElements, frame), ...prevElementsInFrame.filter(element => isElementContainingFrame(allElements, element, frame))]);\n  const elementsNotCompletelyInFrame = prevElementsInFrame.filter(element => !elementsCompletelyInFrame.has(element)); // for elements that are completely in the frame\n  // if they are part of some groups, then those groups are still\n  // considered to belong to the frame\n\n  const groupsToKeep = new Set(Array.from(elementsCompletelyInFrame).flatMap(element => element.groupIds));\n\n  for (const element of elementsNotCompletelyInFrame) {\n    if (!FrameGeometry.isElementIntersectingFrame(element, frame)) {\n      if (element.groupIds.length === 0) {\n        nextElementsInFrame.delete(element);\n      }\n    } else if (element.groupIds.length > 0) {\n      // group element intersects with the frame, we should keep the groups\n      // that this element is part of\n      for (const id of element.groupIds) {\n        groupsToKeep.add(id);\n      }\n    }\n  }\n\n  for (const element of elementsNotCompletelyInFrame) {\n    if (element.groupIds.length > 0) {\n      let shouldRemoveElement = true;\n\n      for (const id of element.groupIds) {\n        if (groupsToKeep.has(id)) {\n          shouldRemoveElement = false;\n        }\n      }\n\n      if (shouldRemoveElement) {\n        nextElementsInFrame.delete(element);\n      }\n    }\n  }\n\n  const individualElementsCompletelyInFrame = Array.from(elementsCompletelyInFrame).filter(element => element.groupIds.length === 0);\n\n  for (const element of individualElementsCompletelyInFrame) {\n    nextElementsInFrame.add(element);\n  }\n\n  const newGroupElementsCompletelyInFrame = Array.from(elementsCompletelyInFrame).filter(element => element.groupIds.length > 0);\n  const groupIds = selectGroupsFromGivenElements(newGroupElementsCompletelyInFrame, appState); // new group elements\n\n  for (const [id, isSelected] of Object.entries(groupIds)) {\n    if (isSelected) {\n      const elementsInGroup = getElementsInGroup(allElements, id);\n\n      if (elementsAreInFrameBounds(elementsInGroup, frame)) {\n        for (const element of elementsInGroup) {\n          nextElementsInFrame.add(element);\n        }\n      }\n    }\n  }\n\n  return [...nextElementsInFrame].filter(element => {\n    return !(isTextElement(element) && element.containerId);\n  });\n};\nexport const getElementsInNewFrame = (allElements, frame) => {\n  return omitGroupsContainingFrames(allElements, getElementsCompletelyInFrame(allElements, frame));\n};\nexport const getContainingFrame = (element,\n/**\n * Optionally an elements map, in case the elements aren't in the Scene yet.\n * Takes precedence over Scene elements, even if the element exists\n * in Scene elements and not the supplied elements map.\n */\nelementsMap) => {\n  var _a;\n\n  if (element.frameId) {\n    if (elementsMap) {\n      return elementsMap.get(element.frameId) || null;\n    }\n\n    return ((_a = Scene.getScene(element)) === null || _a === void 0 ? void 0 : _a.getElement(element.frameId)) || null;\n  }\n\n  return null;\n}; // --------------------------- Frame Operations -------------------------------\n\nexport const addElementsToFrame = (allElements, elementsToAdd, frame) => {\n  const currTargetFrameChildrenMap = new Map(allElements.reduce((acc, element) => {\n    if (element.frameId === frame.id) {\n      acc.push([element.id, element]);\n    }\n\n    return acc;\n  }, []));\n  const suppliedElementsToAddSet = new Set(elementsToAdd.map(el => el.id));\n  const finalElementsToAdd = []; // - add bound text elements if not already in the array\n  // - filter out elements that are already in the frame\n\n  for (const element of omitGroupsContainingFrames(allElements, elementsToAdd)) {\n    if (!currTargetFrameChildrenMap.has(element.id)) {\n      finalElementsToAdd.push(element);\n    }\n\n    const boundTextElement = getBoundTextElement(element);\n\n    if (boundTextElement && !suppliedElementsToAddSet.has(boundTextElement.id) && !currTargetFrameChildrenMap.has(boundTextElement.id)) {\n      finalElementsToAdd.push(boundTextElement);\n    }\n  }\n\n  const finalElementsToAddSet = new Set(finalElementsToAdd.map(el => el.id));\n  const nextElements = [];\n  const processedElements = new Set();\n\n  for (const element of allElements) {\n    if (processedElements.has(element.id)) {\n      continue;\n    }\n\n    processedElements.add(element.id);\n\n    if (finalElementsToAddSet.has(element.id) || element.frameId && element.frameId === frame.id) {\n      // will be added in bulk once we process target frame\n      continue;\n    } // target frame\n\n\n    if (element.id === frame.id) {\n      const currFrameChildren = getFrameElements(allElements, frame.id);\n      currFrameChildren.forEach(child => {\n        processedElements.add(child.id);\n      }); // console.log(currFrameChildren, finalElementsToAdd, element);\n\n      nextElements.push(...currFrameChildren, ...finalElementsToAdd, element);\n      continue;\n    } // console.log(\"(2)\", element.frameId);\n\n\n    nextElements.push(element);\n  }\n\n  for (const element of finalElementsToAdd) {\n    mutateElement(element, {\n      frameId: frame.id\n    }, false);\n  }\n\n  return nextElements;\n};\nexport const removeElementsFromFrame = (allElements, elementsToRemove, appState) => {\n  const _elementsToRemove = [];\n\n  for (const element of elementsToRemove) {\n    if (element.frameId) {\n      _elementsToRemove.push(element);\n\n      const boundTextElement = getBoundTextElement(element);\n\n      if (boundTextElement) {\n        _elementsToRemove.push(boundTextElement);\n      }\n    }\n  }\n\n  for (const element of _elementsToRemove) {\n    mutateElement(element, {\n      frameId: null\n    }, false);\n  }\n\n  const nextElements = moveOneRight(allElements, appState, Array.from(_elementsToRemove));\n  return nextElements;\n};\nexport const removeAllElementsFromFrame = (allElements, frame, appState) => {\n  const elementsInFrame = getFrameElements(allElements, frame.id);\n  return removeElementsFromFrame(allElements, elementsInFrame, appState);\n};\nexport const replaceAllElementsInFrame = (allElements, nextElementsInFrame, frame, appState) => {\n  return addElementsToFrame(removeAllElementsFromFrame(allElements, frame, appState), nextElementsInFrame, frame);\n};\n/** does not mutate elements, but returns new ones */\n\nexport const updateFrameMembershipOfSelectedElements = (allElements, appState, app) => {\n  const selectedElements = app.scene.getSelectedElements({\n    selectedElementIds: appState.selectedElementIds,\n    // supplying elements explicitly in case we're passed non-state elements\n    elements: allElements\n  });\n  const elementsToFilter = new Set(selectedElements);\n\n  if (appState.editingGroupId) {\n    for (const element of selectedElements) {\n      if (element.groupIds.length === 0) {\n        elementsToFilter.add(element);\n      } else {\n        element.groupIds.flatMap(gid => getElementsInGroup(allElements, gid)).forEach(element => elementsToFilter.add(element));\n      }\n    }\n  }\n\n  const elementsToRemove = new Set();\n  elementsToFilter.forEach(element => {\n    if (element.frameId && !isFrameElement(element) && !isElementInFrame(element, allElements, appState)) {\n      elementsToRemove.add(element);\n    }\n  });\n  return elementsToRemove.size > 0 ? removeElementsFromFrame(allElements, [...elementsToRemove], appState) : allElements;\n};\n/**\n * filters out elements that are inside groups that contain a frame element\n * anywhere in the group tree\n */\n\nexport const omitGroupsContainingFrames = (allElements,\n/** subset of elements you want to filter. Optional perf optimization so we\n * don't have to filter all elements unnecessarily\n */\nselectedElements) => {\n  const uniqueGroupIds = new Set();\n\n  for (const el of selectedElements || allElements) {\n    const topMostGroupId = el.groupIds[el.groupIds.length - 1];\n\n    if (topMostGroupId) {\n      uniqueGroupIds.add(topMostGroupId);\n    }\n  }\n\n  const rejectedGroupIds = new Set();\n\n  for (const groupId of uniqueGroupIds) {\n    if (getElementsInGroup(allElements, groupId).some(el => isFrameElement(el))) {\n      rejectedGroupIds.add(groupId);\n    }\n  }\n\n  return (selectedElements || allElements).filter(el => !rejectedGroupIds.has(el.groupIds[el.groupIds.length - 1]));\n};\n/**\n * depending on the appState, return target frame, which is the frame the given element\n * is going to be added to or remove from\n */\n\nexport const getTargetFrame = (element, appState) => {\n  const _element = isTextElement(element) ? getContainerElement(element) || element : element;\n\n  return appState.selectedElementIds[_element.id] && appState.selectedElementsAreBeingDragged ? appState.frameToHighlight : getContainingFrame(_element);\n}; // TODO: this a huge bottleneck for large scenes, optimise\n// given an element, return if the element is in some frame\n\nexport const isElementInFrame = (element, allElements, appState) => {\n  const frame = getTargetFrame(element, appState);\n\n  const _element = isTextElement(element) ? getContainerElement(element) || element : element;\n\n  if (frame) {\n    if (_element.groupIds.length === 0) {\n      return elementOverlapsWithFrame(_element, frame);\n    }\n\n    const allElementsInGroup = new Set(_element.groupIds.flatMap(gid => getElementsInGroup(allElements, gid)));\n\n    if (appState.editingGroupId && appState.selectedElementsAreBeingDragged) {\n      const selectedElements = new Set(getSelectedElements(allElements, appState));\n      const editingGroupOverlapsFrame = appState.frameToHighlight !== null;\n\n      if (editingGroupOverlapsFrame) {\n        return true;\n      }\n\n      selectedElements.forEach(selectedElement => {\n        allElementsInGroup.delete(selectedElement);\n      });\n    }\n\n    for (const elementInGroup of allElementsInGroup) {\n      if (isFrameElement(elementInGroup)) {\n        return false;\n      }\n    }\n\n    for (const elementInGroup of allElementsInGroup) {\n      if (elementOverlapsWithFrame(elementInGroup, frame)) {\n        return true;\n      }\n    }\n  }\n\n  return false;\n};"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///../../frame.ts\n");
|
|
3996
3996
|
|
|
3997
3997
|
/***/ }),
|
|
3998
3998
|
|
|
@@ -4234,7 +4234,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
|
|
|
4234
4234
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
4235
4235
|
|
|
4236
4236
|
"use strict";
|
|
4237
|
-
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.16.1-
|
|
4237
|
+
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.16.1-ceb637f\"}/dist/`;\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9wdWJsaWNQYXRoLmpzLmpzIiwibWFwcGluZ3MiOiI7O0FBQXNDOztBQUV0QyxJQUFJLGFBQW9CLEtBQUssZ0RBQVE7QUFDckM7O0FBRUE7QUFDQSxFQUFFLHFCQUF1Qix3REFBd0Qsd0JBQXlCLENBQUMsR0FBRyxnQkFBNEIsQ0FBQztBQUMzSSIsInNvdXJjZXMiOlsid2VicGFjazovLy8uL3B1YmxpY1BhdGguanM/MGUzMSJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBFTlYgfSBmcm9tIFwiLi4vLi4vY29uc3RhbnRzXCI7XG5cbmlmIChwcm9jZXNzLmVudi5OT0RFX0VOViAhPT0gRU5WLlRFU1QpIHtcbiAgLyogZXNsaW50LWRpc2FibGUgKi9cblxuICAvKiBnbG9iYWwgX193ZWJwYWNrX3B1YmxpY19wYXRoX186d3JpdGFibGUgKi9cbiAgX193ZWJwYWNrX3B1YmxpY19wYXRoX18gPSB3aW5kb3cuRVhDQUxJRFJBV19BU1NFVF9QQVRIIHx8IGBodHRwczovL3VucGtnLmNvbS8ke3Byb2Nlc3MuZW52LlZJVEVfUEtHX05BTUV9QCR7cHJvY2Vzcy5lbnYuVklURV9QS0dfVkVSU0lPTn0vZGlzdC9gO1xufSJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./publicPath.js\n");
|
|
4238
4238
|
|
|
4239
4239
|
/***/ }),
|
|
4240
4240
|
|
|
@@ -4399,7 +4399,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
|
|
|
4399
4399
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
4400
4400
|
|
|
4401
4401
|
"use strict";
|
|
4402
|
-
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"SVG_EXPORT_TAG\": () => (/* binding */ SVG_EXPORT_TAG),\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 _Scene__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ./Scene */ \"../../scene/Scene.ts\");\nvar __awaiter = undefined && undefined.__awaiter || function (thisArg, _arguments, P, generator) {\n function adopt(value) {\n return value instanceof P ? value : new P(function (resolve) {\n resolve(value);\n });\n }\n\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) {\n try {\n step(generator.next(value));\n } catch (e) {\n reject(e);\n }\n }\n\n function rejected(value) {\n try {\n step(generator[\"throw\"](value));\n } catch (e) {\n reject(e);\n }\n }\n\n function step(result) {\n result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected);\n }\n\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n};\n\n\n\n\n\n\n\n\n\n\nconst SVG_EXPORT_TAG = `<!-- svg-source:excalidraw -->`;\nconst exportToCanvas = (elements, appState, files, {\n exportBackground,\n exportPadding = _constants__WEBPACK_IMPORTED_MODULE_4__.DEFAULT_EXPORT_PADDING,\n viewBackgroundColor\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 const [minX, minY, width, height] = getCanvasSize(elements, 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)(elements).map(element => element.fileId),\n files\n });\n const onlyExportingSingleFrame = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isOnlyExportingSingleFrame)(elements);\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,\n visibleElements: elements,\n scale,\n appState: Object.assign(Object.assign({}, appState), {\n viewBackgroundColor: exportBackground ? viewBackgroundColor : null,\n scrollX: -minX + (onlyExportingSingleFrame ? 0 : exportPadding),\n scrollY: -minY + (onlyExportingSingleFrame ? 0 : exportPadding),\n zoom: defaultAppState.zoom,\n shouldCacheIgnoreZoom: false,\n theme: appState.exportWithDarkMode ? \"dark\" : \"light\"\n }),\n renderConfig: {\n imageCache,\n renderGrid: false,\n isExporting: true\n }\n });\n return canvas;\n});\nconst exportToSvg = (elements, appState, files, opts) => __awaiter(void 0, void 0, void 0, function* () {\n var _a, _b, _c;\n\n const {\n exportPadding = _constants__WEBPACK_IMPORTED_MODULE_4__.DEFAULT_EXPORT_PADDING,\n viewBackgroundColor,\n exportScale = 1,\n exportEmbedScene\n } = appState;\n let metadata = \"\";\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 text: (opts === null || opts === void 0 ? void 0 : opts.serializeAsJSON) ? (_a = opts === null || opts === void 0 ? void 0 : opts.serializeAsJSON) === null || _a === void 0 ? void 0 : _a.call(opts) : (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(elements, 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 (appState.exportWithDarkMode) {\n svgRoot.setAttribute(\"filter\", _constants__WEBPACK_IMPORTED_MODULE_4__.THEME_FILTER);\n }\n\n let assetPath = \"https://excalidraw.com/\"; // Asset path needs to be determined only when using package\n\n if (true) {\n assetPath = window.EXCALIDRAW_ASSET_PATH || `https://unpkg.com/${\"@excalidraw/excalidraw\"}@${({\"VITE_APP_BACKEND_V2_GET_URL\":\"https://json-dev.excalidraw.com/api/v2/\",\"VITE_APP_BACKEND_V2_POST_URL\":\"https://json-dev.excalidraw.com/api/v2/post/\",\"VITE_APP_LIBRARY_URL\":\"https://libraries.excalidraw.com\",\"VITE_APP_LIBRARY_BACKEND\":\"https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries\",\"VITE_APP_WS_SERVER_URL\":\"http://localhost:3002\",\"VITE_APP_PORTAL_URL\":\"\",\"VITE_APP_PLUS_LP\":\"https://plus.excalidraw.com\",\"VITE_APP_PLUS_APP\":\"https://app.excalidraw.com\",\"VITE_APP_FIREBASE_CONFIG\":\"{\\\"apiKey\\\":\\\"AIzaSyCMkxA60XIW8KbqMYL7edC4qT5l4qHX2h8\\\",\\\"authDomain\\\":\\\"excalidraw-oss-dev.firebaseapp.com\\\",\\\"projectId\\\":\\\"excalidraw-oss-dev\\\",\\\"storageBucket\\\":\\\"excalidraw-oss-dev.appspot.com\\\",\\\"messagingSenderId\\\":\\\"664559512677\\\",\\\"appId\\\":\\\"1:664559512677:web:a385181f2928d328a7aa8c\\\"}\",\"VITE_APP_DEV_ENABLE_SW\":\"\",\"VITE_APP_DEV_DISABLE_LIVE_RELOAD\":\"\",\"VITE_APP_DISABLE_TRACKING\":\"true\",\"FAST_REFRESH\":\"false\",\"VITE_APP_PORT\":\"3000\",\"VITE_APP_DEBUG_ENABLE_TEXT_CONTAINER_BOUNDING_BOX\":\"\",\"VITE_APP_COLLAPSE_OVERLAY\":\"true\",\"VITE_APP_ENABLE_ESLINT\":\"true\",\"VITE_PKG_NAME\":\"@excalidraw/excalidraw\",\"VITE_PKG_VERSION\":\"0.16.1-4c35eba\",\"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 } // do not apply clipping when we're exporting the whole scene\n\n\n const isExportingWholeCanvas = ((_c = (_b = _Scene__WEBPACK_IMPORTED_MODULE_8__[\"default\"].getScene(elements[0])) === null || _b === void 0 ? void 0 : _b.getNonDeletedElements()) === null || _c === void 0 ? void 0 : _c.length) === elements.length;\n const onlyExportingSingleFrame = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isOnlyExportingSingleFrame)(elements);\n const offsetX = -minX + (onlyExportingSingleFrame ? 0 : exportPadding);\n const offsetY = -minY + (onlyExportingSingleFrame ? 0 : exportPadding);\n const exportingFrame = isExportingWholeCanvas || !onlyExportingSingleFrame ? undefined : elements.find(element => element.type === \"frame\");\n let exportingFrameClipPath = \"\";\n\n if (exportingFrame) {\n const [x1, y1, x2, y2] = (0,_element_bounds__WEBPACK_IMPORTED_MODULE_1__.getElementAbsoluteCoords)(exportingFrame);\n const cx = (x2 - x1) / 2 - (exportingFrame.x - x1);\n const cy = (y2 - y1) / 2 - (exportingFrame.y - y1);\n exportingFrameClipPath = `<clipPath id=${exportingFrame.id}>\n <rect transform=\"translate(${exportingFrame.x + offsetX} ${exportingFrame.y + offsetY}) rotate(${exportingFrame.angle} ${cx} ${cy})\"\n width=\"${exportingFrame.width}\"\n height=\"${exportingFrame.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 </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)(elements, rsvg, svgRoot, files || {}, {\n offsetX,\n offsetY,\n exportWithDarkMode: appState.exportWithDarkMode,\n exportingFrameId: (exportingFrame === null || exportingFrame === void 0 ? void 0 : exportingFrame.id) || null,\n renderEmbeddables: opts === null || opts === void 0 ? void 0 : opts.renderEmbeddables\n });\n return svgRoot;\n}); // calculate smallest area to fit the contents in\n\nconst getCanvasSize = (elements, exportPadding) => {\n // we should decide if we are exporting the whole canvas\n // if so, we are not clipping elements in the frame\n // and therefore, we should not do anything special\n var _a, _b;\n\n const isExportingWholeCanvas = ((_b = (_a = _Scene__WEBPACK_IMPORTED_MODULE_8__[\"default\"].getScene(elements[0])) === null || _a === void 0 ? void 0 : _a.getNonDeletedElements()) === null || _b === void 0 ? void 0 : _b.length) === elements.length;\n const onlyExportingSingleFrame = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isOnlyExportingSingleFrame)(elements);\n\n if (!isExportingWholeCanvas || onlyExportingSingleFrame) {\n const frames = elements.filter(element => element.type === \"frame\");\n const exportedFrameIds = frames.reduce((acc, frame) => {\n acc[frame.id] = true;\n return acc;\n }, {}); // elements in a frame do not affect the canvas size if we're not exporting\n // the whole canvas\n\n elements = elements.filter(element => {\n var _a;\n\n return !exportedFrameIds[(_a = element.frameId) !== null && _a !== void 0 ? _a : \"\"];\n });\n }\n\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) + (onlyExportingSingleFrame ? 0 : exportPadding * 2);\n const height = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.distance)(minY, maxY) + (onlyExportingSingleFrame ? 0 : 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;AACd;AACY;AAC3B;AACF;AACkC;AACrD;AACrB;AACA;AACP;AACA,kBAAkB,8DAAsB;AACxC;AACA,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD;AACA;AACA;AACA;AACA,IAAI;AACJ,0BAA0B,6DAAkB;AAC5C;AACA;AACA,IAAI,QAAQ,gEAAgB;AAC5B;AACA,aAAa,2EAA2B;AACxC;AACA,GAAG;AACH,mCAAmC,kEAA0B;AAC7D,EAAE,wEAAiB;AACnB;AACA,QAAQ,gEAAY;AACpB;AACA;AACA;AACA,4CAA4C;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA,CAAC;AACM;AACP;;AAEA;AACA,oBAAoB,8DAAsB;AAC1C;AACA;AACA;AACA,IAAI;AACJ;;AAEA;AACA;AACA,8BAA8B,uIAED;AAC7B,gNAAgN,2DAAe,gCAAgC;AAC/P,OAAO;AACP,MAAM;AACN;AACA;AACA;;AAEA,8EAA8E;;AAE9E,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,8qCAAW,aAAa;;AAE1H;AACA,4CAA4C,uBAAuB;AACnE;;AAEA,mBAAmB,UAAU;AAC7B,IAAI;;;AAGJ,8CAA8C,uDAAc;AAC5D,mCAAmC,kEAA0B;AAC7D;AACA;AACA;AACA;;AAEA;AACA,6BAA6B,yEAAwB;AACrD;AACA;AACA,6CAA6C,kBAAkB;AAC/D,yCAAyC,4BAA4B,EAAE,2BAA2B,WAAW,sBAAsB,EAAE,IAAI,EAAE,GAAG;AAC9I,mBAAmB,qBAAqB;AACxC,oBAAoB,sBAAsB;AAC1C;AACA;AACA;AACA;;AAEA;AACA,IAAI;AACJ,IAAI;AACJ;AACA;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,qCAAqC;AACvD;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA,CAAC,GAAG;;AAEJ;AACA;AACA;AACA;AACA;;AAEA,8CAA8C,uDAAc;AAC5D,mCAAmC,kEAA0B;;AAE7D;AACA;AACA;AACA;AACA;AACA,KAAK,IAAI,GAAG;AACZ;;AAEA;AACA;;AAEA;AACA,KAAK;AACL;;AAEA,mCAAmC,gEAAe;AAClD,gBAAgB,gDAAQ;AACxB,iBAAiB,gDAAQ;AACzB;AACA;;AAEO;AACP;AACA;AACA","sources":["webpack:///../../scene/export.ts?3f4e"],"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 { distance, isOnlyExportingSingleFrame } from \"../utils\";\nimport { DEFAULT_EXPORT_PADDING, SVG_NS, THEME_FILTER } from \"../constants\";\nimport { getDefaultAppState } from \"../appState\";\nimport { serializeAsJSON } from \"../data/json\";\nimport { getInitializedImageElements, updateImageCache } from \"../element/image\";\nimport Scene from \"./Scene\";\nexport const SVG_EXPORT_TAG = `<!-- svg-source:excalidraw -->`;\nexport const exportToCanvas = (elements, appState, files, {\n  exportBackground,\n  exportPadding = DEFAULT_EXPORT_PADDING,\n  viewBackgroundColor\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  const [minX, minY, width, height] = getCanvasSize(elements, 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(elements).map(element => element.fileId),\n    files\n  });\n  const onlyExportingSingleFrame = isOnlyExportingSingleFrame(elements);\n  renderStaticScene({\n    canvas,\n    rc: rough.canvas(canvas),\n    elements,\n    visibleElements: elements,\n    scale,\n    appState: Object.assign(Object.assign({}, appState), {\n      viewBackgroundColor: exportBackground ? viewBackgroundColor : null,\n      scrollX: -minX + (onlyExportingSingleFrame ? 0 : exportPadding),\n      scrollY: -minY + (onlyExportingSingleFrame ? 0 : exportPadding),\n      zoom: defaultAppState.zoom,\n      shouldCacheIgnoreZoom: false,\n      theme: appState.exportWithDarkMode ? \"dark\" : \"light\"\n    }),\n    renderConfig: {\n      imageCache,\n      renderGrid: false,\n      isExporting: true\n    }\n  });\n  return canvas;\n});\nexport const exportToSvg = (elements, appState, files, opts) => __awaiter(void 0, void 0, void 0, function* () {\n  var _a, _b, _c;\n\n  const {\n    exportPadding = DEFAULT_EXPORT_PADDING,\n    viewBackgroundColor,\n    exportScale = 1,\n    exportEmbedScene\n  } = appState;\n  let metadata = \"\";\n\n  if (exportEmbedScene) {\n    try {\n      metadata = yield (yield import(\n      /* webpackChunkName: \"image\" */\n      \"../../src/data/image\")).encodeSvgMetadata({\n        text: (opts === null || opts === void 0 ? void 0 : opts.serializeAsJSON) ? (_a = opts === null || opts === void 0 ? void 0 : opts.serializeAsJSON) === null || _a === void 0 ? void 0 : _a.call(opts) : serializeAsJSON(elements, appState, files || {}, \"local\")\n      });\n    } catch (error) {\n      console.error(error);\n    }\n  }\n\n  const [minX, minY, width, height] = getCanvasSize(elements, 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 (appState.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  } // do not apply clipping when we're exporting the whole scene\n\n\n  const isExportingWholeCanvas = ((_c = (_b = Scene.getScene(elements[0])) === null || _b === void 0 ? void 0 : _b.getNonDeletedElements()) === null || _c === void 0 ? void 0 : _c.length) === elements.length;\n  const onlyExportingSingleFrame = isOnlyExportingSingleFrame(elements);\n  const offsetX = -minX + (onlyExportingSingleFrame ? 0 : exportPadding);\n  const offsetY = -minY + (onlyExportingSingleFrame ? 0 : exportPadding);\n  const exportingFrame = isExportingWholeCanvas || !onlyExportingSingleFrame ? undefined : elements.find(element => element.type === \"frame\");\n  let exportingFrameClipPath = \"\";\n\n  if (exportingFrame) {\n    const [x1, y1, x2, y2] = getElementAbsoluteCoords(exportingFrame);\n    const cx = (x2 - x1) / 2 - (exportingFrame.x - x1);\n    const cy = (y2 - y1) / 2 - (exportingFrame.y - y1);\n    exportingFrameClipPath = `<clipPath id=${exportingFrame.id}>\n            <rect transform=\"translate(${exportingFrame.x + offsetX} ${exportingFrame.y + offsetY}) rotate(${exportingFrame.angle} ${cx} ${cy})\"\n          width=\"${exportingFrame.width}\"\n          height=\"${exportingFrame.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    </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(elements, rsvg, svgRoot, files || {}, {\n    offsetX,\n    offsetY,\n    exportWithDarkMode: appState.exportWithDarkMode,\n    exportingFrameId: (exportingFrame === null || exportingFrame === void 0 ? void 0 : exportingFrame.id) || null,\n    renderEmbeddables: opts === null || opts === void 0 ? void 0 : opts.renderEmbeddables\n  });\n  return svgRoot;\n}); // calculate smallest area to fit the contents in\n\nconst getCanvasSize = (elements, exportPadding) => {\n  // we should decide if we are exporting the whole canvas\n  // if so, we are not clipping elements in the frame\n  // and therefore, we should not do anything special\n  var _a, _b;\n\n  const isExportingWholeCanvas = ((_b = (_a = Scene.getScene(elements[0])) === null || _a === void 0 ? void 0 : _a.getNonDeletedElements()) === null || _b === void 0 ? void 0 : _b.length) === elements.length;\n  const onlyExportingSingleFrame = isOnlyExportingSingleFrame(elements);\n\n  if (!isExportingWholeCanvas || onlyExportingSingleFrame) {\n    const frames = elements.filter(element => element.type === \"frame\");\n    const exportedFrameIds = frames.reduce((acc, frame) => {\n      acc[frame.id] = true;\n      return acc;\n    }, {}); // elements in a frame do not affect the canvas size if we're not exporting\n    // the whole canvas\n\n    elements = elements.filter(element => {\n      var _a;\n\n      return !exportedFrameIds[(_a = element.frameId) !== null && _a !== void 0 ? _a : \"\"];\n    });\n  }\n\n  const [minX, minY, maxX, maxY] = getCommonBounds(elements);\n  const width = distance(minX, maxX) + (onlyExportingSingleFrame ? 0 : exportPadding * 2);\n  const height = distance(minY, maxY) + (onlyExportingSingleFrame ? 0 : 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");
|
|
4402
|
+
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"SVG_EXPORT_TAG\": () => (/* binding */ SVG_EXPORT_TAG),\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 _Scene__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ./Scene */ \"../../scene/Scene.ts\");\nvar __awaiter = undefined && undefined.__awaiter || function (thisArg, _arguments, P, generator) {\n function adopt(value) {\n return value instanceof P ? value : new P(function (resolve) {\n resolve(value);\n });\n }\n\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) {\n try {\n step(generator.next(value));\n } catch (e) {\n reject(e);\n }\n }\n\n function rejected(value) {\n try {\n step(generator[\"throw\"](value));\n } catch (e) {\n reject(e);\n }\n }\n\n function step(result) {\n result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected);\n }\n\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n};\n\n\n\n\n\n\n\n\n\n\nconst SVG_EXPORT_TAG = `<!-- svg-source:excalidraw -->`;\nconst exportToCanvas = (elements, appState, files, {\n exportBackground,\n exportPadding = _constants__WEBPACK_IMPORTED_MODULE_4__.DEFAULT_EXPORT_PADDING,\n viewBackgroundColor\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 const [minX, minY, width, height] = getCanvasSize(elements, 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)(elements).map(element => element.fileId),\n files\n });\n const onlyExportingSingleFrame = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isOnlyExportingSingleFrame)(elements);\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,\n visibleElements: elements,\n scale,\n appState: Object.assign(Object.assign({}, appState), {\n viewBackgroundColor: exportBackground ? viewBackgroundColor : null,\n scrollX: -minX + (onlyExportingSingleFrame ? 0 : exportPadding),\n scrollY: -minY + (onlyExportingSingleFrame ? 0 : exportPadding),\n zoom: defaultAppState.zoom,\n shouldCacheIgnoreZoom: false,\n theme: appState.exportWithDarkMode ? \"dark\" : \"light\"\n }),\n renderConfig: {\n imageCache,\n renderGrid: false,\n isExporting: true\n }\n });\n return canvas;\n});\nconst exportToSvg = (elements, appState, files, opts) => __awaiter(void 0, void 0, void 0, function* () {\n var _a, _b, _c;\n\n const {\n exportPadding = _constants__WEBPACK_IMPORTED_MODULE_4__.DEFAULT_EXPORT_PADDING,\n viewBackgroundColor,\n exportScale = 1,\n exportEmbedScene\n } = appState;\n let metadata = \"\";\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 text: (opts === null || opts === void 0 ? void 0 : opts.serializeAsJSON) ? (_a = opts === null || opts === void 0 ? void 0 : opts.serializeAsJSON) === null || _a === void 0 ? void 0 : _a.call(opts) : (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(elements, 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 (appState.exportWithDarkMode) {\n svgRoot.setAttribute(\"filter\", _constants__WEBPACK_IMPORTED_MODULE_4__.THEME_FILTER);\n }\n\n let assetPath = \"https://excalidraw.com/\"; // Asset path needs to be determined only when using package\n\n if (true) {\n assetPath = window.EXCALIDRAW_ASSET_PATH || `https://unpkg.com/${\"@excalidraw/excalidraw\"}@${({\"VITE_APP_BACKEND_V2_GET_URL\":\"https://json-dev.excalidraw.com/api/v2/\",\"VITE_APP_BACKEND_V2_POST_URL\":\"https://json-dev.excalidraw.com/api/v2/post/\",\"VITE_APP_LIBRARY_URL\":\"https://libraries.excalidraw.com\",\"VITE_APP_LIBRARY_BACKEND\":\"https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries\",\"VITE_APP_WS_SERVER_URL\":\"http://localhost:3002\",\"VITE_APP_PORTAL_URL\":\"\",\"VITE_APP_PLUS_LP\":\"https://plus.excalidraw.com\",\"VITE_APP_PLUS_APP\":\"https://app.excalidraw.com\",\"VITE_APP_FIREBASE_CONFIG\":\"{\\\"apiKey\\\":\\\"AIzaSyCMkxA60XIW8KbqMYL7edC4qT5l4qHX2h8\\\",\\\"authDomain\\\":\\\"excalidraw-oss-dev.firebaseapp.com\\\",\\\"projectId\\\":\\\"excalidraw-oss-dev\\\",\\\"storageBucket\\\":\\\"excalidraw-oss-dev.appspot.com\\\",\\\"messagingSenderId\\\":\\\"664559512677\\\",\\\"appId\\\":\\\"1:664559512677:web:a385181f2928d328a7aa8c\\\"}\",\"VITE_APP_DEV_ENABLE_SW\":\"\",\"VITE_APP_DEV_DISABLE_LIVE_RELOAD\":\"\",\"VITE_APP_DISABLE_TRACKING\":\"true\",\"FAST_REFRESH\":\"false\",\"VITE_APP_PORT\":\"3000\",\"VITE_APP_DEBUG_ENABLE_TEXT_CONTAINER_BOUNDING_BOX\":\"\",\"VITE_APP_COLLAPSE_OVERLAY\":\"true\",\"VITE_APP_ENABLE_ESLINT\":\"true\",\"VITE_PKG_NAME\":\"@excalidraw/excalidraw\",\"VITE_PKG_VERSION\":\"0.16.1-ceb637f\",\"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 } // do not apply clipping when we're exporting the whole scene\n\n\n const isExportingWholeCanvas = ((_c = (_b = _Scene__WEBPACK_IMPORTED_MODULE_8__[\"default\"].getScene(elements[0])) === null || _b === void 0 ? void 0 : _b.getNonDeletedElements()) === null || _c === void 0 ? void 0 : _c.length) === elements.length;\n const onlyExportingSingleFrame = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isOnlyExportingSingleFrame)(elements);\n const offsetX = -minX + (onlyExportingSingleFrame ? 0 : exportPadding);\n const offsetY = -minY + (onlyExportingSingleFrame ? 0 : exportPadding);\n const exportingFrame = isExportingWholeCanvas || !onlyExportingSingleFrame ? undefined : elements.find(element => element.type === \"frame\");\n let exportingFrameClipPath = \"\";\n\n if (exportingFrame) {\n const [x1, y1, x2, y2] = (0,_element_bounds__WEBPACK_IMPORTED_MODULE_1__.getElementAbsoluteCoords)(exportingFrame);\n const cx = (x2 - x1) / 2 - (exportingFrame.x - x1);\n const cy = (y2 - y1) / 2 - (exportingFrame.y - y1);\n exportingFrameClipPath = `<clipPath id=${exportingFrame.id}>\n <rect transform=\"translate(${exportingFrame.x + offsetX} ${exportingFrame.y + offsetY}) rotate(${exportingFrame.angle} ${cx} ${cy})\"\n width=\"${exportingFrame.width}\"\n height=\"${exportingFrame.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 </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)(elements, rsvg, svgRoot, files || {}, {\n offsetX,\n offsetY,\n exportWithDarkMode: appState.exportWithDarkMode,\n exportingFrameId: (exportingFrame === null || exportingFrame === void 0 ? void 0 : exportingFrame.id) || null,\n renderEmbeddables: opts === null || opts === void 0 ? void 0 : opts.renderEmbeddables\n });\n return svgRoot;\n}); // calculate smallest area to fit the contents in\n\nconst getCanvasSize = (elements, exportPadding) => {\n // we should decide if we are exporting the whole canvas\n // if so, we are not clipping elements in the frame\n // and therefore, we should not do anything special\n var _a, _b;\n\n const isExportingWholeCanvas = ((_b = (_a = _Scene__WEBPACK_IMPORTED_MODULE_8__[\"default\"].getScene(elements[0])) === null || _a === void 0 ? void 0 : _a.getNonDeletedElements()) === null || _b === void 0 ? void 0 : _b.length) === elements.length;\n const onlyExportingSingleFrame = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.isOnlyExportingSingleFrame)(elements);\n\n if (!isExportingWholeCanvas || onlyExportingSingleFrame) {\n const frames = elements.filter(element => element.type === \"frame\");\n const exportedFrameIds = frames.reduce((acc, frame) => {\n acc[frame.id] = true;\n return acc;\n }, {}); // elements in a frame do not affect the canvas size if we're not exporting\n // the whole canvas\n\n elements = elements.filter(element => {\n var _a;\n\n return !exportedFrameIds[(_a = element.frameId) !== null && _a !== void 0 ? _a : \"\"];\n });\n }\n\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) + (onlyExportingSingleFrame ? 0 : exportPadding * 2);\n const height = (0,_utils__WEBPACK_IMPORTED_MODULE_3__.distance)(minY, maxY) + (onlyExportingSingleFrame ? 0 : 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;AACd;AACY;AAC3B;AACF;AACkC;AACrD;AACrB;AACA;AACP;AACA,kBAAkB,8DAAsB;AACxC;AACA,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD;AACA;AACA;AACA;AACA,IAAI;AACJ,0BAA0B,6DAAkB;AAC5C;AACA;AACA,IAAI,QAAQ,gEAAgB;AAC5B;AACA,aAAa,2EAA2B;AACxC;AACA,GAAG;AACH,mCAAmC,kEAA0B;AAC7D,EAAE,wEAAiB;AACnB;AACA,QAAQ,gEAAY;AACpB;AACA;AACA;AACA,4CAA4C;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA,CAAC;AACM;AACP;;AAEA;AACA,oBAAoB,8DAAsB;AAC1C;AACA;AACA;AACA,IAAI;AACJ;;AAEA;AACA;AACA,8BAA8B,uIAED;AAC7B,gNAAgN,2DAAe,gCAAgC;AAC/P,OAAO;AACP,MAAM;AACN;AACA;AACA;;AAEA,8EAA8E;;AAE9E,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,8qCAAW,aAAa;;AAE1H;AACA,4CAA4C,uBAAuB;AACnE;;AAEA,mBAAmB,UAAU;AAC7B,IAAI;;;AAGJ,8CAA8C,uDAAc;AAC5D,mCAAmC,kEAA0B;AAC7D;AACA;AACA;AACA;;AAEA;AACA,6BAA6B,yEAAwB;AACrD;AACA;AACA,6CAA6C,kBAAkB;AAC/D,yCAAyC,4BAA4B,EAAE,2BAA2B,WAAW,sBAAsB,EAAE,IAAI,EAAE,GAAG;AAC9I,mBAAmB,qBAAqB;AACxC,oBAAoB,sBAAsB;AAC1C;AACA;AACA;AACA;;AAEA;AACA,IAAI;AACJ,IAAI;AACJ;AACA;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,qCAAqC;AACvD;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA,CAAC,GAAG;;AAEJ;AACA;AACA;AACA;AACA;;AAEA,8CAA8C,uDAAc;AAC5D,mCAAmC,kEAA0B;;AAE7D;AACA;AACA;AACA;AACA;AACA,KAAK,IAAI,GAAG;AACZ;;AAEA;AACA;;AAEA;AACA,KAAK;AACL;;AAEA,mCAAmC,gEAAe;AAClD,gBAAgB,gDAAQ;AACxB,iBAAiB,gDAAQ;AACzB;AACA;;AAEO;AACP;AACA;AACA","sources":["webpack:///../../scene/export.ts?3f4e"],"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 { distance, isOnlyExportingSingleFrame } from \"../utils\";\nimport { DEFAULT_EXPORT_PADDING, SVG_NS, THEME_FILTER } from \"../constants\";\nimport { getDefaultAppState } from \"../appState\";\nimport { serializeAsJSON } from \"../data/json\";\nimport { getInitializedImageElements, updateImageCache } from \"../element/image\";\nimport Scene from \"./Scene\";\nexport const SVG_EXPORT_TAG = `<!-- svg-source:excalidraw -->`;\nexport const exportToCanvas = (elements, appState, files, {\n  exportBackground,\n  exportPadding = DEFAULT_EXPORT_PADDING,\n  viewBackgroundColor\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  const [minX, minY, width, height] = getCanvasSize(elements, 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(elements).map(element => element.fileId),\n    files\n  });\n  const onlyExportingSingleFrame = isOnlyExportingSingleFrame(elements);\n  renderStaticScene({\n    canvas,\n    rc: rough.canvas(canvas),\n    elements,\n    visibleElements: elements,\n    scale,\n    appState: Object.assign(Object.assign({}, appState), {\n      viewBackgroundColor: exportBackground ? viewBackgroundColor : null,\n      scrollX: -minX + (onlyExportingSingleFrame ? 0 : exportPadding),\n      scrollY: -minY + (onlyExportingSingleFrame ? 0 : exportPadding),\n      zoom: defaultAppState.zoom,\n      shouldCacheIgnoreZoom: false,\n      theme: appState.exportWithDarkMode ? \"dark\" : \"light\"\n    }),\n    renderConfig: {\n      imageCache,\n      renderGrid: false,\n      isExporting: true\n    }\n  });\n  return canvas;\n});\nexport const exportToSvg = (elements, appState, files, opts) => __awaiter(void 0, void 0, void 0, function* () {\n  var _a, _b, _c;\n\n  const {\n    exportPadding = DEFAULT_EXPORT_PADDING,\n    viewBackgroundColor,\n    exportScale = 1,\n    exportEmbedScene\n  } = appState;\n  let metadata = \"\";\n\n  if (exportEmbedScene) {\n    try {\n      metadata = yield (yield import(\n      /* webpackChunkName: \"image\" */\n      \"../../src/data/image\")).encodeSvgMetadata({\n        text: (opts === null || opts === void 0 ? void 0 : opts.serializeAsJSON) ? (_a = opts === null || opts === void 0 ? void 0 : opts.serializeAsJSON) === null || _a === void 0 ? void 0 : _a.call(opts) : serializeAsJSON(elements, appState, files || {}, \"local\")\n      });\n    } catch (error) {\n      console.error(error);\n    }\n  }\n\n  const [minX, minY, width, height] = getCanvasSize(elements, 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 (appState.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  } // do not apply clipping when we're exporting the whole scene\n\n\n  const isExportingWholeCanvas = ((_c = (_b = Scene.getScene(elements[0])) === null || _b === void 0 ? void 0 : _b.getNonDeletedElements()) === null || _c === void 0 ? void 0 : _c.length) === elements.length;\n  const onlyExportingSingleFrame = isOnlyExportingSingleFrame(elements);\n  const offsetX = -minX + (onlyExportingSingleFrame ? 0 : exportPadding);\n  const offsetY = -minY + (onlyExportingSingleFrame ? 0 : exportPadding);\n  const exportingFrame = isExportingWholeCanvas || !onlyExportingSingleFrame ? undefined : elements.find(element => element.type === \"frame\");\n  let exportingFrameClipPath = \"\";\n\n  if (exportingFrame) {\n    const [x1, y1, x2, y2] = getElementAbsoluteCoords(exportingFrame);\n    const cx = (x2 - x1) / 2 - (exportingFrame.x - x1);\n    const cy = (y2 - y1) / 2 - (exportingFrame.y - y1);\n    exportingFrameClipPath = `<clipPath id=${exportingFrame.id}>\n            <rect transform=\"translate(${exportingFrame.x + offsetX} ${exportingFrame.y + offsetY}) rotate(${exportingFrame.angle} ${cx} ${cy})\"\n          width=\"${exportingFrame.width}\"\n          height=\"${exportingFrame.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    </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(elements, rsvg, svgRoot, files || {}, {\n    offsetX,\n    offsetY,\n    exportWithDarkMode: appState.exportWithDarkMode,\n    exportingFrameId: (exportingFrame === null || exportingFrame === void 0 ? void 0 : exportingFrame.id) || null,\n    renderEmbeddables: opts === null || opts === void 0 ? void 0 : opts.renderEmbeddables\n  });\n  return svgRoot;\n}); // calculate smallest area to fit the contents in\n\nconst getCanvasSize = (elements, exportPadding) => {\n  // we should decide if we are exporting the whole canvas\n  // if so, we are not clipping elements in the frame\n  // and therefore, we should not do anything special\n  var _a, _b;\n\n  const isExportingWholeCanvas = ((_b = (_a = Scene.getScene(elements[0])) === null || _a === void 0 ? void 0 : _a.getNonDeletedElements()) === null || _b === void 0 ? void 0 : _b.length) === elements.length;\n  const onlyExportingSingleFrame = isOnlyExportingSingleFrame(elements);\n\n  if (!isExportingWholeCanvas || onlyExportingSingleFrame) {\n    const frames = elements.filter(element => element.type === \"frame\");\n    const exportedFrameIds = frames.reduce((acc, frame) => {\n      acc[frame.id] = true;\n      return acc;\n    }, {}); // elements in a frame do not affect the canvas size if we're not exporting\n    // the whole canvas\n\n    elements = elements.filter(element => {\n      var _a;\n\n      return !exportedFrameIds[(_a = element.frameId) !== null && _a !== void 0 ? _a : \"\"];\n    });\n  }\n\n  const [minX, minY, maxX, maxY] = getCommonBounds(elements);\n  const width = distance(minX, maxX) + (onlyExportingSingleFrame ? 0 : exportPadding * 2);\n  const height = distance(minY, maxY) + (onlyExportingSingleFrame ? 0 : 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");
|
|
4403
4403
|
|
|
4404
4404
|
/***/ }),
|
|
4405
4405
|
|