@excalidraw/excalidraw 0.17.1-1d71f84 → 0.17.1-2f9526d
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.
- package/CHANGELOG.md +14 -2
- package/dist/browser/dev/excalidraw-assets-dev/{blockDiagram-91b80b7a-ACFH36JV.js → blockDiagram-91b80b7a-H47FTXHA.js} +5 -5
- package/dist/browser/dev/excalidraw-assets-dev/{c4Diagram-b2a90758-QZ27YR47.js → c4Diagram-b2a90758-NNJK6GKC.js} +3 -3
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-HO2HMSK7.js → chunk-4KQVEBHW.js} +3 -3
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-USGV265L.js → chunk-53YI56GV.js} +4 -4
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-EDFX3S7X.js → chunk-A2WCJI4I.js} +3 -3
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-IX4V72YG.js → chunk-EFLPX7NE.js} +6 -6
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-MXVETLVM.js → chunk-JYIQCNWV.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-YZIOORVX.js → chunk-LVIQQW6F.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-6U7GQNJT.js → chunk-PXLO3FOU.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-AK7SWNLN.js → chunk-Q6A4M3MN.js} +9 -5
- package/dist/browser/dev/excalidraw-assets-dev/chunk-Q6A4M3MN.js.map +7 -0
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-RWZVJAQU.js → chunk-Q6NFAEKN.js} +4597 -2668
- package/dist/browser/dev/excalidraw-assets-dev/chunk-Q6NFAEKN.js.map +7 -0
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-7DACDEY3.js → chunk-TO2AW5PW.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-NJ77ZFNJ.js → chunk-VURILHLY.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-2T2GU7NF.js → chunk-ZAYGSUHF.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{chunk-Z3PH3V2B.js → chunk-ZQR5ML6Y.js} +26 -26
- package/dist/browser/dev/excalidraw-assets-dev/chunk-ZQR5ML6Y.js.map +7 -0
- package/dist/browser/dev/excalidraw-assets-dev/{classDiagram-30eddba6-QSLMH4JW.js → classDiagram-30eddba6-CUYIJICN.js} +5 -5
- package/dist/browser/dev/excalidraw-assets-dev/{classDiagram-v2-f2df5561-DY4DYQ5P.js → classDiagram-v2-f2df5561-K6WW6K73.js} +8 -8
- package/dist/browser/dev/excalidraw-assets-dev/{dist-Z46EOVOL.js → dist-6QVAH5JA.js} +37 -15
- package/dist/browser/dev/excalidraw-assets-dev/dist-6QVAH5JA.js.map +7 -0
- package/dist/browser/dev/excalidraw-assets-dev/{en-5TCZHGGJ.js → en-Y27YPU72.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{erDiagram-47591fe2-SOOJRTCB.js → erDiagram-47591fe2-XGAD7EEP.js} +4 -4
- package/dist/browser/dev/excalidraw-assets-dev/{flowDiagram-5540d9b9-AHGL4KPK.js → flowDiagram-5540d9b9-B6EOVNNO.js} +9 -9
- package/dist/browser/dev/excalidraw-assets-dev/{flowDiagram-v2-3b53844e-56LDZZWY.js → flowDiagram-v2-3b53844e-NUG24FJH.js} +9 -9
- package/dist/browser/dev/excalidraw-assets-dev/{flowchart-elk-definition-5fe447d6-27LUKRI6.js → flowchart-elk-definition-5fe447d6-25Y7PCBL.js} +5 -5
- package/dist/browser/dev/excalidraw-assets-dev/{ganttDiagram-9a3bba1f-EHGYGNG6.js → ganttDiagram-9a3bba1f-GNL6ZDTC.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{gitGraphDiagram-96e6b4ee-AJQNBDW5.js → gitGraphDiagram-96e6b4ee-HNW52NVO.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{image-OFRRV5MB.css → image-O66MQ7WQ.css} +1 -1
- package/dist/browser/dev/excalidraw-assets-dev/image-O66MQ7WQ.css.map +7 -0
- package/dist/browser/dev/excalidraw-assets-dev/{image-EDKQZH7Z.js → image-Y5X7K6KW.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{infoDiagram-bcd20f53-SWLLQVES.js → infoDiagram-bcd20f53-FWEUVFLT.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{journeyDiagram-4fe6b3dc-7UAVCWOZ.js → journeyDiagram-4fe6b3dc-RZIUI7UG.js} +3 -3
- package/dist/browser/dev/excalidraw-assets-dev/{mindmap-definition-f354de21-SROW5KGM.js → mindmap-definition-f354de21-GBVN45GU.js} +3 -3
- package/dist/browser/dev/excalidraw-assets-dev/{pieDiagram-79897490-QKCI6NCB.js → pieDiagram-79897490-ECENNII6.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{quadrantDiagram-62f64e94-LNYJZFC5.js → quadrantDiagram-62f64e94-ZMEOFVNL.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{requirementDiagram-05bf5f74-ZZD7ZHFA.js → requirementDiagram-05bf5f74-FHZSFHCR.js} +4 -4
- package/dist/browser/dev/excalidraw-assets-dev/{sankeyDiagram-97764748-L75ZZ4UM.js → sankeyDiagram-97764748-VDKIKTA6.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{sequenceDiagram-acc0e65c-6PCU7TDK.js → sequenceDiagram-acc0e65c-6JUSPVKX.js} +3 -3
- package/dist/browser/dev/excalidraw-assets-dev/{stateDiagram-0ff1cf1a-WM76WOPR.js → stateDiagram-0ff1cf1a-L3AKWENF.js} +5 -5
- package/dist/browser/dev/excalidraw-assets-dev/{stateDiagram-v2-9a9d610d-N4HZW3O2.js → stateDiagram-v2-9a9d610d-NU3GGMCH.js} +8 -8
- package/dist/browser/dev/excalidraw-assets-dev/{timeline-definition-fea2a41d-ZHGCAXGP.js → timeline-definition-fea2a41d-JGP7XCHW.js} +2 -2
- package/dist/browser/dev/excalidraw-assets-dev/{xychartDiagram-ab372869-2DLOVRAZ.js → xychartDiagram-ab372869-HLFHHF2I.js} +3 -3
- package/dist/browser/dev/index.css +72 -28
- package/dist/browser/dev/index.css.map +3 -3
- package/dist/browser/dev/index.js +2211 -1904
- package/dist/browser/dev/index.js.map +4 -4
- package/dist/browser/prod/excalidraw-assets/{blockDiagram-91b80b7a-ONPS22AM.js → blockDiagram-91b80b7a-FVCRVGN5.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{c4Diagram-b2a90758-XMIQY7ZT.js → c4Diagram-b2a90758-56CXO7GA.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{chunk-GCHQBOKV.js → chunk-635MQ3CK.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{chunk-P5M3G2RP.js → chunk-AIKXYJX3.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{chunk-E2YLWFZX.js → chunk-CR7VMNWC.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{chunk-WEYK4A2L.js → chunk-FFF2CSVG.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{chunk-R3HAIP6R.js → chunk-G4WDCSPE.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{chunk-HFOXJM22.js → chunk-HKZSHFLX.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{chunk-XIMFFJTE.js → chunk-IKCDYWMW.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{chunk-CTYINSWT.js → chunk-IZMZ6RPD.js} +2 -2
- package/dist/browser/prod/excalidraw-assets/{chunk-AHLLBBVJ.js → chunk-L5DS24G6.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/chunk-MDMKPHYD.js +55 -0
- package/dist/browser/prod/excalidraw-assets/{chunk-CQJF3C6G.js → chunk-MUNOKHUD.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{chunk-NI6SYCUG.js → chunk-QOQYOOQ4.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{chunk-I2PZFXTK.js → chunk-ZTIWFPBM.js} +21 -21
- package/dist/browser/prod/excalidraw-assets/{classDiagram-30eddba6-IEJXXCVX.js → classDiagram-30eddba6-BCUTAUMD.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{classDiagram-v2-f2df5561-7LZDSWOS.js → classDiagram-v2-f2df5561-6SOXSGQ2.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/dist-567JAXHK.js +7 -0
- package/dist/browser/prod/excalidraw-assets/{en-LROPV2RN.js → en-GSUSWMSH.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{erDiagram-47591fe2-E5V666CF.js → erDiagram-47591fe2-RE6HB7RM.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{flowDiagram-5540d9b9-GMBRCYVF.js → flowDiagram-5540d9b9-ZNJZBERW.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{flowDiagram-v2-3b53844e-Z4HUWP6B.js → flowDiagram-v2-3b53844e-LY44JLQJ.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{flowchart-elk-definition-5fe447d6-5ZCYTX5N.js → flowchart-elk-definition-5fe447d6-TMTJ6Z7O.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{ganttDiagram-9a3bba1f-WM32OMT5.js → ganttDiagram-9a3bba1f-5O6EA6LX.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{gitGraphDiagram-96e6b4ee-CAKZ2U6E.js → gitGraphDiagram-96e6b4ee-UHYNM5DI.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/image-7MVXYJUE.js +1 -0
- package/dist/browser/prod/excalidraw-assets/{infoDiagram-bcd20f53-MUIKXGC4.js → infoDiagram-bcd20f53-BP77NQEH.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{journeyDiagram-4fe6b3dc-NYRV4HK2.js → journeyDiagram-4fe6b3dc-XMGKCMES.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{mindmap-definition-f354de21-MY55DRSM.js → mindmap-definition-f354de21-ZQRRBRWF.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{pieDiagram-79897490-47L6J6L2.js → pieDiagram-79897490-IGXEC2KX.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{quadrantDiagram-62f64e94-DF5C2GDT.js → quadrantDiagram-62f64e94-WTHHDYJL.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{requirementDiagram-05bf5f74-C4IMUBDN.js → requirementDiagram-05bf5f74-MV4OFRVW.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{sankeyDiagram-97764748-YHW7EUST.js → sankeyDiagram-97764748-ZGYUHEJT.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{sequenceDiagram-acc0e65c-H3XEHT32.js → sequenceDiagram-acc0e65c-IBSENK6W.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{stateDiagram-0ff1cf1a-Z5WB6Q3P.js → stateDiagram-0ff1cf1a-DB73XNZH.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{stateDiagram-v2-9a9d610d-T7OZETQC.js → stateDiagram-v2-9a9d610d-2OOBUPNR.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{timeline-definition-fea2a41d-VVC22BWF.js → timeline-definition-fea2a41d-P3NQQVDU.js} +1 -1
- package/dist/browser/prod/excalidraw-assets/{xychartDiagram-ab372869-JAXODQF7.js → xychartDiagram-ab372869-HI3XLK3Y.js} +1 -1
- package/dist/browser/prod/index.css +1 -1
- package/dist/browser/prod/index.js +24 -24
- package/dist/dev/{en-II4GK66F.json → en-OIPCBIOA.json} +8 -4
- package/dist/dev/index.css +72 -28
- package/dist/dev/index.css.map +3 -3
- package/dist/dev/index.js +5626 -3465
- package/dist/dev/index.js.map +4 -4
- package/dist/excalidraw/actions/actionAddToLibrary.d.ts +4 -4
- package/dist/excalidraw/actions/actionAddToLibrary.js +4 -3
- package/dist/excalidraw/actions/actionAlign.d.ts +22 -22
- package/dist/excalidraw/actions/actionAlign.js +7 -6
- package/dist/excalidraw/actions/actionBoundText.d.ts +10 -10
- package/dist/excalidraw/actions/actionBoundText.js +11 -5
- package/dist/excalidraw/actions/actionCanvas.d.ts +52 -52
- package/dist/excalidraw/actions/actionCanvas.js +19 -14
- package/dist/excalidraw/actions/actionClipboard.d.ts +24 -24
- package/dist/excalidraw/actions/actionClipboard.js +14 -13
- package/dist/excalidraw/actions/actionDeleteSelected.d.ts +10 -10
- package/dist/excalidraw/actions/actionDeleteSelected.js +6 -3
- package/dist/excalidraw/actions/actionDistribute.d.ts +10 -10
- package/dist/excalidraw/actions/actionDistribute.js +3 -2
- package/dist/excalidraw/actions/actionDuplicateSelection.d.ts +7 -8
- package/dist/excalidraw/actions/actionDuplicateSelection.js +7 -3
- package/dist/excalidraw/actions/actionElementLock.d.ts +9 -9
- package/dist/excalidraw/actions/actionElementLock.js +3 -2
- package/dist/excalidraw/actions/actionExport.d.ts +33 -33
- package/dist/excalidraw/actions/actionExport.js +15 -11
- package/dist/excalidraw/actions/actionFinalize.d.ts +9 -9
- package/dist/excalidraw/actions/actionFinalize.js +9 -5
- package/dist/excalidraw/actions/actionFlip.d.ts +10 -10
- package/dist/excalidraw/actions/actionFlip.js +12 -12
- package/dist/excalidraw/actions/actionFrame.d.ts +16 -171
- package/dist/excalidraw/actions/actionFrame.js +7 -6
- package/dist/excalidraw/actions/actionGroup.d.ts +12 -322
- package/dist/excalidraw/actions/actionGroup.js +9 -11
- package/dist/excalidraw/actions/actionHistory.d.ts +4 -3
- package/dist/excalidraw/actions/actionHistory.js +27 -28
- package/dist/excalidraw/actions/actionLinearEditor.d.ts +6 -4
- package/dist/excalidraw/actions/actionLinearEditor.js +21 -5
- package/dist/excalidraw/actions/actionLink.d.ts +5 -5
- package/dist/excalidraw/actions/actionLink.js +2 -1
- package/dist/excalidraw/actions/actionMenu.d.ts +8 -8
- package/dist/excalidraw/actions/actionMenu.js +4 -3
- package/dist/excalidraw/actions/actionNavigate.d.ts +4 -4
- package/dist/excalidraw/actions/actionNavigate.js +3 -2
- package/dist/excalidraw/actions/actionProperties.d.ts +34 -34
- package/dist/excalidraw/actions/actionProperties.js +19 -14
- package/dist/excalidraw/actions/actionSelectAll.d.ts +5 -5
- package/dist/excalidraw/actions/actionSelectAll.js +2 -1
- package/dist/excalidraw/actions/actionStyles.d.ts +9 -12
- package/dist/excalidraw/actions/actionStyles.js +4 -3
- package/dist/excalidraw/actions/actionTextAutoResize.d.ts +17 -0
- package/dist/excalidraw/actions/actionTextAutoResize.js +38 -0
- package/dist/excalidraw/actions/actionToggleGridMode.d.ts +7 -5
- package/dist/excalidraw/actions/actionToggleGridMode.js +6 -2
- package/dist/excalidraw/actions/actionToggleObjectsSnapMode.d.ts +4 -4
- package/dist/excalidraw/actions/actionToggleObjectsSnapMode.js +2 -1
- package/dist/excalidraw/actions/actionToggleStats.d.ts +4 -4
- package/dist/excalidraw/actions/actionToggleStats.js +2 -1
- package/dist/excalidraw/actions/actionToggleViewMode.d.ts +4 -4
- package/dist/excalidraw/actions/actionToggleViewMode.js +2 -1
- package/dist/excalidraw/actions/actionToggleZenMode.d.ts +4 -4
- package/dist/excalidraw/actions/actionToggleZenMode.js +2 -1
- package/dist/excalidraw/actions/actionZindex.d.ts +23 -19
- package/dist/excalidraw/actions/actionZindex.js +9 -4
- package/dist/excalidraw/actions/manager.d.ts +5 -5
- package/dist/excalidraw/actions/register.d.ts +1 -1
- package/dist/excalidraw/actions/shortcuts.d.ts +2 -2
- package/dist/excalidraw/actions/shortcuts.js +1 -1
- package/dist/excalidraw/actions/types.d.ts +8 -8
- package/dist/excalidraw/align.d.ts +1 -1
- package/dist/excalidraw/analytics.js +1 -1
- package/dist/excalidraw/animated-trail.d.ts +2 -2
- package/dist/excalidraw/appState.d.ts +5 -5
- package/dist/excalidraw/change.d.ts +191 -0
- package/dist/excalidraw/change.js +894 -0
- package/dist/excalidraw/charts.d.ts +1 -1
- package/dist/excalidraw/clients.d.ts +2 -2
- package/dist/excalidraw/clients.js +1 -1
- package/dist/excalidraw/clipboard.d.ts +3 -3
- package/dist/excalidraw/colors.d.ts +1 -1
- package/dist/excalidraw/components/Actions.d.ts +3 -3
- package/dist/excalidraw/components/Actions.js +9 -6
- package/dist/excalidraw/components/App.d.ts +28 -14
- package/dist/excalidraw/components/App.js +407 -187
- package/dist/excalidraw/components/ButtonIconSelect.js +1 -1
- package/dist/excalidraw/components/CheckboxItem.js +1 -1
- package/dist/excalidraw/components/ColorPicker/ColorInput.d.ts +1 -1
- package/dist/excalidraw/components/ColorPicker/ColorInput.js +1 -1
- package/dist/excalidraw/components/ColorPicker/ColorPicker.d.ts +4 -4
- package/dist/excalidraw/components/ColorPicker/ColorPicker.js +1 -1
- package/dist/excalidraw/components/ColorPicker/Picker.d.ts +3 -3
- package/dist/excalidraw/components/ColorPicker/PickerColorList.d.ts +1 -1
- package/dist/excalidraw/components/ColorPicker/PickerHeading.d.ts +1 -1
- package/dist/excalidraw/components/ColorPicker/ShadeList.d.ts +1 -1
- package/dist/excalidraw/components/ColorPicker/TopPicks.d.ts +1 -1
- package/dist/excalidraw/components/ColorPicker/colorPickerUtils.d.ts +2 -2
- package/dist/excalidraw/components/ColorPicker/colorPickerUtils.js +1 -1
- package/dist/excalidraw/components/ColorPicker/keyboardNavHandlers.d.ts +2 -2
- package/dist/excalidraw/components/ColorPicker/keyboardNavHandlers.js +1 -1
- package/dist/excalidraw/components/CommandPalette/CommandPalette.d.ts +1 -1
- package/dist/excalidraw/components/CommandPalette/CommandPalette.js +30 -15
- package/dist/excalidraw/components/CommandPalette/defaultCommandPaletteItems.d.ts +1 -1
- package/dist/excalidraw/components/CommandPalette/types.d.ts +3 -3
- package/dist/excalidraw/components/ConfirmDialog.d.ts +1 -1
- package/dist/excalidraw/components/ContextMenu.d.ts +2 -2
- package/dist/excalidraw/components/ContextMenu.js +2 -2
- package/dist/excalidraw/components/DarkModeToggle.d.ts +1 -1
- package/dist/excalidraw/components/DarkModeToggle.js +3 -1
- package/dist/excalidraw/components/DefaultSidebar.d.ts +2 -2
- package/dist/excalidraw/components/Dialog.js +1 -1
- package/dist/excalidraw/components/DialogActionButton.d.ts +1 -1
- package/dist/excalidraw/components/EyeDropper.d.ts +2 -2
- package/dist/excalidraw/components/FollowMode/FollowMode.d.ts +1 -1
- package/dist/excalidraw/components/FollowMode/FollowMode.js +1 -1
- package/dist/excalidraw/components/HelpDialog.js +8 -6
- package/dist/excalidraw/components/HintViewer.d.ts +1 -1
- package/dist/excalidraw/components/IconPicker.js +2 -2
- package/dist/excalidraw/components/ImageExportDialog.d.ts +1 -1
- package/dist/excalidraw/components/InitializeApp.d.ts +2 -2
- package/dist/excalidraw/components/JSONExportDialog.d.ts +3 -3
- package/dist/excalidraw/components/LayerUI.d.ts +4 -4
- package/dist/excalidraw/components/LayerUI.js +2 -2
- package/dist/excalidraw/components/LibraryMenu.d.ts +2 -2
- package/dist/excalidraw/components/LibraryMenuBrowseButton.d.ts +1 -1
- package/dist/excalidraw/components/LibraryMenuControlButtons.d.ts +1 -1
- package/dist/excalidraw/components/LibraryMenuHeaderContent.d.ts +2 -2
- package/dist/excalidraw/components/LibraryMenuItems.d.ts +1 -1
- package/dist/excalidraw/components/LibraryMenuSection.d.ts +5 -4
- package/dist/excalidraw/components/LibraryUnit.d.ts +2 -2
- package/dist/excalidraw/components/LoadingMessage.d.ts +1 -1
- package/dist/excalidraw/components/MagicSettings.js +2 -2
- package/dist/excalidraw/components/MobileMenu.d.ts +3 -3
- package/dist/excalidraw/components/MobileMenu.js +1 -1
- package/dist/excalidraw/components/Modal.d.ts +1 -1
- package/dist/excalidraw/components/OverwriteConfirm/OverwriteConfirmState.d.ts +1 -1
- package/dist/excalidraw/components/PasteChartDialog.d.ts +1 -1
- package/dist/excalidraw/components/PasteChartDialog.js +1 -1
- package/dist/excalidraw/components/PublishLibrary.d.ts +1 -1
- package/dist/excalidraw/components/RadioGroup.d.ts +2 -1
- package/dist/excalidraw/components/RadioGroup.js +1 -1
- package/dist/excalidraw/components/SVGLayer.d.ts +1 -1
- package/dist/excalidraw/components/Sidebar/Sidebar.d.ts +2 -2
- package/dist/excalidraw/components/Sidebar/Sidebar.js +1 -1
- package/dist/excalidraw/components/Sidebar/SidebarTab.d.ts +1 -1
- package/dist/excalidraw/components/Sidebar/SidebarTabTrigger.d.ts +1 -1
- package/dist/excalidraw/components/Sidebar/SidebarTrigger.d.ts +1 -1
- package/dist/excalidraw/components/Sidebar/common.d.ts +1 -1
- package/dist/excalidraw/components/Stack.d.ts +2 -2
- package/dist/excalidraw/components/Stats.d.ts +2 -2
- package/dist/excalidraw/components/TTDDialog/MermaidToExcalidraw.d.ts +1 -1
- package/dist/excalidraw/components/TTDDialog/MermaidToExcalidraw.js +6 -2
- package/dist/excalidraw/components/TTDDialog/TTDDialog.js +2 -2
- package/dist/excalidraw/components/TTDDialog/TTDDialogInput.d.ts +1 -1
- package/dist/excalidraw/components/TTDDialog/TTDDialogPanel.d.ts +1 -1
- package/dist/excalidraw/components/TTDDialog/TTDDialogPanels.d.ts +1 -1
- package/dist/excalidraw/components/TTDDialog/TTDDialogTabs.d.ts +1 -1
- package/dist/excalidraw/components/TTDDialog/TTDDialogTrigger.d.ts +1 -1
- package/dist/excalidraw/components/TTDDialog/common.d.ts +4 -4
- package/dist/excalidraw/components/TextField.d.ts +1 -1
- package/dist/excalidraw/components/Toast.d.ts +1 -1
- package/dist/excalidraw/components/ToolButton.d.ts +4 -2
- package/dist/excalidraw/components/ToolButton.js +1 -1
- package/dist/excalidraw/components/Trans.d.ts +1 -1
- package/dist/excalidraw/components/UserList.d.ts +1 -1
- package/dist/excalidraw/components/canvases/InteractiveCanvas.d.ts +3 -2
- package/dist/excalidraw/components/canvases/InteractiveCanvas.js +3 -2
- package/dist/excalidraw/components/canvases/StaticCanvas.d.ts +2 -2
- package/dist/excalidraw/components/canvases/StaticCanvas.js +2 -2
- package/dist/excalidraw/components/dropdownMenu/DropdownMenuItem.js +2 -2
- package/dist/excalidraw/components/dropdownMenu/DropdownMenuItemContentRadio.d.ts +18 -0
- package/dist/excalidraw/components/dropdownMenu/DropdownMenuItemContentRadio.js +9 -0
- package/dist/excalidraw/components/footer/Footer.d.ts +2 -2
- package/dist/excalidraw/components/hyperlink/Hyperlink.d.ts +2 -2
- package/dist/excalidraw/components/hyperlink/Hyperlink.js +3 -3
- package/dist/excalidraw/components/hyperlink/helpers.d.ts +3 -3
- package/dist/excalidraw/components/hyperlink/helpers.js +2 -3
- package/dist/excalidraw/components/icons.d.ts +6 -1
- package/dist/excalidraw/components/icons.js +23 -5
- package/dist/excalidraw/components/main-menu/DefaultItems.d.ts +12 -2
- package/dist/excalidraw/components/main-menu/DefaultItems.js +38 -7
- package/dist/excalidraw/constants.d.ts +5 -5
- package/dist/excalidraw/constants.js +6 -3
- package/dist/excalidraw/context/ui-appState.d.ts +1 -1
- package/dist/excalidraw/cursor.d.ts +1 -1
- package/dist/excalidraw/data/EditorLocalStorage.d.ts +2 -2
- package/dist/excalidraw/data/blob.d.ts +5 -5
- package/dist/excalidraw/data/filesystem.d.ts +2 -1
- package/dist/excalidraw/data/index.d.ts +4 -4
- package/dist/excalidraw/data/json.d.ts +3 -3
- package/dist/excalidraw/data/library.d.ts +3 -3
- package/dist/excalidraw/data/magic.d.ts +3 -3
- package/dist/excalidraw/data/magic.js +2 -1
- package/dist/excalidraw/data/reconcile.d.ts +6 -0
- package/dist/excalidraw/data/reconcile.js +49 -0
- package/dist/excalidraw/data/resave.d.ts +2 -2
- package/dist/excalidraw/data/restore.d.ts +5 -5
- package/dist/excalidraw/data/restore.js +7 -7
- package/dist/excalidraw/data/transform.d.ts +4 -4
- package/dist/excalidraw/data/transform.js +12 -3
- package/dist/excalidraw/data/types.d.ts +3 -3
- package/dist/excalidraw/data/url.d.ts +1 -0
- package/dist/excalidraw/data/url.js +4 -1
- package/dist/excalidraw/element/ElementCanvasButtons.d.ts +1 -1
- package/dist/excalidraw/element/binding.d.ts +50 -9
- package/dist/excalidraw/element/binding.js +712 -155
- package/dist/excalidraw/element/bounds.d.ts +3 -4
- package/dist/excalidraw/element/bounds.js +0 -3
- package/dist/excalidraw/element/collision.d.ts +14 -19
- package/dist/excalidraw/element/collision.js +36 -713
- package/dist/excalidraw/element/containerCache.d.ts +1 -1
- package/dist/excalidraw/element/dragElements.d.ts +3 -3
- package/dist/excalidraw/element/embeddable.d.ts +9 -6
- package/dist/excalidraw/element/embeddable.js +98 -62
- package/dist/excalidraw/element/image.d.ts +2 -2
- package/dist/excalidraw/element/index.d.ts +2 -3
- package/dist/excalidraw/element/index.js +1 -2
- package/dist/excalidraw/element/linearElementEditor.d.ts +12 -12
- package/dist/excalidraw/element/linearElementEditor.js +7 -5
- package/dist/excalidraw/element/mutateElement.d.ts +4 -5
- package/dist/excalidraw/element/mutateElement.js +5 -3
- package/dist/excalidraw/element/newElement.d.ts +4 -9
- package/dist/excalidraw/element/newElement.js +17 -14
- package/dist/excalidraw/element/resizeElements.d.ts +4 -4
- package/dist/excalidraw/element/resizeElements.js +170 -97
- package/dist/excalidraw/element/resizeTest.d.ts +7 -7
- package/dist/excalidraw/element/resizeTest.js +53 -8
- package/dist/excalidraw/element/showSelectedShapeActions.d.ts +2 -2
- package/dist/excalidraw/element/sizeHelpers.d.ts +2 -2
- package/dist/excalidraw/element/sizeHelpers.js +3 -0
- package/dist/excalidraw/element/sortElements.d.ts +1 -1
- package/dist/excalidraw/element/textElement.d.ts +5 -6
- package/dist/excalidraw/element/textElement.js +13 -37
- package/dist/excalidraw/element/textWysiwyg.d.ts +10 -5
- package/dist/excalidraw/element/textWysiwyg.js +6 -8
- package/dist/excalidraw/element/transformHandles.d.ts +24 -6
- package/dist/excalidraw/element/transformHandles.js +22 -11
- package/dist/excalidraw/element/typeChecks.d.ts +4 -4
- package/dist/excalidraw/element/types.d.ts +33 -10
- package/dist/excalidraw/emitter.d.ts +1 -1
- package/dist/excalidraw/errors.d.ts +3 -0
- package/dist/excalidraw/errors.js +3 -0
- package/dist/excalidraw/fractionalIndex.d.ts +40 -0
- package/dist/excalidraw/fractionalIndex.js +241 -0
- package/dist/excalidraw/frame.d.ts +4 -4
- package/dist/excalidraw/gatransforms.d.ts +1 -1
- package/dist/excalidraw/gesture.d.ts +1 -1
- package/dist/excalidraw/groups.d.ts +5 -3
- package/dist/excalidraw/groups.js +17 -0
- package/dist/excalidraw/history.d.ts +35 -47
- package/dist/excalidraw/history.js +100 -167
- package/dist/excalidraw/hooks/useCreatePortalContainer.js +2 -1
- package/dist/excalidraw/hooks/useEmitter.d.ts +2 -0
- package/dist/excalidraw/hooks/useEmitter.js +13 -0
- package/dist/excalidraw/hooks/useLibraryItemSvg.d.ts +1 -1
- package/dist/excalidraw/i18n.d.ts +1 -1
- package/dist/excalidraw/index.d.ts +3 -1
- package/dist/excalidraw/index.js +2 -0
- package/dist/excalidraw/jotai.d.ts +1 -1
- package/dist/excalidraw/laser-trails.d.ts +3 -2
- package/dist/excalidraw/locales/en.json +8 -4
- package/dist/excalidraw/math.d.ts +2 -2
- package/dist/excalidraw/points.d.ts +1 -1
- package/dist/excalidraw/queue.d.ts +1 -1
- package/dist/excalidraw/renderer/helpers.d.ts +2 -2
- package/dist/excalidraw/renderer/helpers.js +2 -2
- package/dist/excalidraw/renderer/interactiveScene.d.ts +2 -2
- package/dist/excalidraw/renderer/interactiveScene.js +8 -7
- package/dist/excalidraw/renderer/renderElement.d.ts +3 -3
- package/dist/excalidraw/renderer/renderElement.js +5 -5
- package/dist/excalidraw/renderer/renderSnaps.d.ts +1 -1
- package/dist/excalidraw/renderer/renderSnaps.js +2 -1
- package/dist/excalidraw/renderer/staticScene.d.ts +1 -1
- package/dist/excalidraw/renderer/staticScene.js +14 -3
- package/dist/excalidraw/renderer/staticSvgScene.d.ts +4 -4
- package/dist/excalidraw/renderer/staticSvgScene.js +10 -0
- package/dist/excalidraw/scene/Fonts.d.ts +2 -4
- package/dist/excalidraw/scene/Fonts.js +6 -12
- package/dist/excalidraw/scene/Renderer.d.ts +4 -4
- package/dist/excalidraw/scene/Renderer.js +2 -3
- package/dist/excalidraw/scene/Scene.d.ts +19 -12
- package/dist/excalidraw/scene/Scene.js +44 -23
- package/dist/excalidraw/scene/Shape.d.ts +1 -1
- package/dist/excalidraw/scene/ShapeCache.d.ts +4 -4
- package/dist/excalidraw/scene/comparisons.d.ts +2 -2
- package/dist/excalidraw/scene/export.d.ts +2 -2
- package/dist/excalidraw/scene/export.js +6 -5
- package/dist/excalidraw/scene/scroll.d.ts +2 -2
- package/dist/excalidraw/scene/scrollbars.d.ts +3 -3
- package/dist/excalidraw/scene/selection.d.ts +2 -2
- package/dist/excalidraw/scene/types.d.ts +5 -4
- package/dist/excalidraw/scene/zoom.d.ts +1 -1
- package/dist/excalidraw/snapping.d.ts +4 -4
- package/dist/excalidraw/store.d.ts +99 -0
- package/dist/excalidraw/store.js +269 -0
- package/dist/excalidraw/types.d.ts +33 -19
- package/dist/excalidraw/utils.d.ts +11 -4
- package/dist/excalidraw/utils.js +8 -0
- package/dist/excalidraw/zindex.d.ts +4 -4
- package/dist/excalidraw/zindex.js +9 -13
- package/dist/prod/{en-II4GK66F.json → en-OIPCBIOA.json} +8 -4
- package/dist/prod/index.css +1 -1
- package/dist/prod/index.js +44 -44
- package/dist/utils/bbox.d.ts +2 -2
- package/dist/utils/collision.d.ts +4 -0
- package/dist/utils/collision.js +48 -0
- package/dist/utils/export.d.ts +2 -2
- package/dist/utils/geometry/geometry.d.ts +71 -0
- package/dist/utils/geometry/geometry.js +674 -0
- package/dist/utils/geometry/shape.d.ts +56 -0
- package/dist/utils/geometry/shape.js +168 -0
- package/dist/utils/withinBounds.d.ts +1 -1
- package/history.ts +163 -218
- package/package.json +3 -2
- package/dist/browser/dev/excalidraw-assets-dev/chunk-AK7SWNLN.js.map +0 -7
- package/dist/browser/dev/excalidraw-assets-dev/chunk-RWZVJAQU.js.map +0 -7
- package/dist/browser/dev/excalidraw-assets-dev/chunk-Z3PH3V2B.js.map +0 -7
- package/dist/browser/dev/excalidraw-assets-dev/dist-Z46EOVOL.js.map +0 -7
- package/dist/browser/dev/excalidraw-assets-dev/image-OFRRV5MB.css.map +0 -7
- package/dist/browser/prod/excalidraw-assets/chunk-LL4GORAM.js +0 -55
- package/dist/browser/prod/excalidraw-assets/dist-PIPZXALV.js +0 -6
- package/dist/browser/prod/excalidraw-assets/image-EFCJDJH3.js +0 -1
- /package/dist/browser/dev/excalidraw-assets-dev/{blockDiagram-91b80b7a-ACFH36JV.js.map → blockDiagram-91b80b7a-H47FTXHA.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{c4Diagram-b2a90758-QZ27YR47.js.map → c4Diagram-b2a90758-NNJK6GKC.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{chunk-HO2HMSK7.js.map → chunk-4KQVEBHW.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{chunk-USGV265L.js.map → chunk-53YI56GV.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{chunk-EDFX3S7X.js.map → chunk-A2WCJI4I.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{chunk-IX4V72YG.js.map → chunk-EFLPX7NE.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{chunk-MXVETLVM.js.map → chunk-JYIQCNWV.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{chunk-YZIOORVX.js.map → chunk-LVIQQW6F.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{chunk-6U7GQNJT.js.map → chunk-PXLO3FOU.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{chunk-7DACDEY3.js.map → chunk-TO2AW5PW.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{chunk-NJ77ZFNJ.js.map → chunk-VURILHLY.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{chunk-2T2GU7NF.js.map → chunk-ZAYGSUHF.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{classDiagram-30eddba6-QSLMH4JW.js.map → classDiagram-30eddba6-CUYIJICN.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{classDiagram-v2-f2df5561-DY4DYQ5P.js.map → classDiagram-v2-f2df5561-K6WW6K73.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{en-5TCZHGGJ.js.map → en-Y27YPU72.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{erDiagram-47591fe2-SOOJRTCB.js.map → erDiagram-47591fe2-XGAD7EEP.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{flowDiagram-5540d9b9-AHGL4KPK.js.map → flowDiagram-5540d9b9-B6EOVNNO.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{flowDiagram-v2-3b53844e-56LDZZWY.js.map → flowDiagram-v2-3b53844e-NUG24FJH.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{flowchart-elk-definition-5fe447d6-27LUKRI6.js.map → flowchart-elk-definition-5fe447d6-25Y7PCBL.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{ganttDiagram-9a3bba1f-EHGYGNG6.js.map → ganttDiagram-9a3bba1f-GNL6ZDTC.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{gitGraphDiagram-96e6b4ee-AJQNBDW5.js.map → gitGraphDiagram-96e6b4ee-HNW52NVO.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{image-EDKQZH7Z.js.map → image-Y5X7K6KW.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{infoDiagram-bcd20f53-SWLLQVES.js.map → infoDiagram-bcd20f53-FWEUVFLT.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{journeyDiagram-4fe6b3dc-7UAVCWOZ.js.map → journeyDiagram-4fe6b3dc-RZIUI7UG.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{mindmap-definition-f354de21-SROW5KGM.js.map → mindmap-definition-f354de21-GBVN45GU.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{pieDiagram-79897490-QKCI6NCB.js.map → pieDiagram-79897490-ECENNII6.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{quadrantDiagram-62f64e94-LNYJZFC5.js.map → quadrantDiagram-62f64e94-ZMEOFVNL.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{requirementDiagram-05bf5f74-ZZD7ZHFA.js.map → requirementDiagram-05bf5f74-FHZSFHCR.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{sankeyDiagram-97764748-L75ZZ4UM.js.map → sankeyDiagram-97764748-VDKIKTA6.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{sequenceDiagram-acc0e65c-6PCU7TDK.js.map → sequenceDiagram-acc0e65c-6JUSPVKX.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{stateDiagram-0ff1cf1a-WM76WOPR.js.map → stateDiagram-0ff1cf1a-L3AKWENF.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{stateDiagram-v2-9a9d610d-N4HZW3O2.js.map → stateDiagram-v2-9a9d610d-NU3GGMCH.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{timeline-definition-fea2a41d-ZHGCAXGP.js.map → timeline-definition-fea2a41d-JGP7XCHW.js.map} +0 -0
- /package/dist/browser/dev/excalidraw-assets-dev/{xychartDiagram-ab372869-2DLOVRAZ.js.map → xychartDiagram-ab372869-HLFHHF2I.js.map} +0 -0
|
@@ -1,6 +1,12 @@
|
|
|
1
|
+
import * as GA from "../ga";
|
|
2
|
+
import * as GAPoint from "../gapoints";
|
|
3
|
+
import * as GADirection from "../gadirections";
|
|
4
|
+
import * as GALine from "../galines";
|
|
5
|
+
import * as GATransform from "../gatransforms";
|
|
6
|
+
import { getElementAbsoluteCoords } from "./bounds";
|
|
7
|
+
import { isPointOnShape } from "../../utils/collision";
|
|
1
8
|
import { getElementAtPosition } from "../scene";
|
|
2
|
-
import { isBindableElement, isBindingElement, isLinearElement, } from "./typeChecks";
|
|
3
|
-
import { bindingBorderTest, distanceToBindableElement, maxBindingGap, determineFocusDistance, intersectElementWithLine, determineFocusPoint, } from "./collision";
|
|
9
|
+
import { isArrowElement, isBindableElement, isBindingElement, isBoundToContainer, isLinearElement, isTextElement, } from "./typeChecks";
|
|
4
10
|
import { mutateElement } from "./mutateElement";
|
|
5
11
|
import Scene from "../scene/Scene";
|
|
6
12
|
import { LinearElementEditor } from "./linearElementEditor";
|
|
@@ -40,48 +46,119 @@ const bindOrUnbindLinearElementEdge = (linearElement, bindableElement, otherEdge
|
|
|
40
46
|
boundToElementIds,
|
|
41
47
|
// Is mutated
|
|
42
48
|
unboundFromElementIds, elementsMap) => {
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
bindLinearElement(linearElement, bindableElement, startOrEnd, elementsMap);
|
|
53
|
-
boundToElementIds.add(bindableElement.id);
|
|
54
|
-
}
|
|
49
|
+
// "keep" is for method chaining convenience, a "no-op", so just bail out
|
|
50
|
+
if (bindableElement === "keep") {
|
|
51
|
+
return;
|
|
52
|
+
}
|
|
53
|
+
// null means break the bind, so nothing to consider here
|
|
54
|
+
if (bindableElement === null) {
|
|
55
|
+
const unbound = unbindLinearElement(linearElement, startOrEnd);
|
|
56
|
+
if (unbound != null) {
|
|
57
|
+
unboundFromElementIds.add(unbound);
|
|
55
58
|
}
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
59
|
+
return;
|
|
60
|
+
}
|
|
61
|
+
// While complext arrows can do anything, simple arrow with both ends trying
|
|
62
|
+
// to bind to the same bindable should not be allowed, start binding takes
|
|
63
|
+
// precedence
|
|
64
|
+
if (isLinearElementSimple(linearElement)) {
|
|
65
|
+
if (otherEdgeBindableElement == null ||
|
|
66
|
+
(otherEdgeBindableElement === "keep"
|
|
67
|
+
? // TODO: Refactor - Needlessly complex
|
|
68
|
+
!isLinearElementSimpleAndAlreadyBoundOnOppositeEdge(linearElement, bindableElement, startOrEnd)
|
|
69
|
+
: startOrEnd === "start" ||
|
|
70
|
+
otherEdgeBindableElement.id !== bindableElement.id)) {
|
|
71
|
+
bindLinearElement(linearElement, bindableElement, startOrEnd, elementsMap);
|
|
72
|
+
boundToElementIds.add(bindableElement.id);
|
|
61
73
|
}
|
|
62
74
|
}
|
|
75
|
+
else {
|
|
76
|
+
bindLinearElement(linearElement, bindableElement, startOrEnd, elementsMap);
|
|
77
|
+
boundToElementIds.add(bindableElement.id);
|
|
78
|
+
}
|
|
63
79
|
};
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
80
|
+
const getOriginalBindingIfStillCloseOfLinearElementEdge = (linearElement, edge, app) => {
|
|
81
|
+
const elementsMap = app.scene.getNonDeletedElementsMap();
|
|
82
|
+
const coors = getLinearElementEdgeCoors(linearElement, edge, elementsMap);
|
|
83
|
+
const elementId = edge === "start"
|
|
84
|
+
? linearElement.startBinding?.elementId
|
|
85
|
+
: linearElement.endBinding?.elementId;
|
|
86
|
+
if (elementId) {
|
|
87
|
+
const element = elementsMap.get(elementId);
|
|
88
|
+
if (bindingBorderTest(element, coors, app)) {
|
|
89
|
+
return element;
|
|
71
90
|
}
|
|
91
|
+
}
|
|
92
|
+
return null;
|
|
93
|
+
};
|
|
94
|
+
const getOriginalBindingsIfStillCloseToArrowEnds = (linearElement, app) => ["start", "end"].map((edge) => getOriginalBindingIfStillCloseOfLinearElementEdge(linearElement, edge, app));
|
|
95
|
+
const getBindingStrategyForDraggingArrowEndpoints = (selectedElement, isBindingEnabled, draggingPoints, app) => {
|
|
96
|
+
const startIdx = 0;
|
|
97
|
+
const endIdx = selectedElement.points.length - 1;
|
|
98
|
+
const startDragged = draggingPoints.findIndex((i) => i === startIdx) > -1;
|
|
99
|
+
const endDragged = draggingPoints.findIndex((i) => i === endIdx) > -1;
|
|
100
|
+
const start = startDragged
|
|
101
|
+
? isBindingEnabled
|
|
102
|
+
? getElligibleElementForBindingElement(selectedElement, "start", app)
|
|
103
|
+
: null // If binding is disabled and start is dragged, break all binds
|
|
104
|
+
: // We have to update the focus and gap of the binding, so let's rebind
|
|
105
|
+
getElligibleElementForBindingElement(selectedElement, "start", app);
|
|
106
|
+
const end = endDragged
|
|
107
|
+
? isBindingEnabled
|
|
108
|
+
? getElligibleElementForBindingElement(selectedElement, "end", app)
|
|
109
|
+
: null // If binding is disabled and end is dragged, break all binds
|
|
110
|
+
: // We have to update the focus and gap of the binding, so let's rebind
|
|
111
|
+
getElligibleElementForBindingElement(selectedElement, "end", app);
|
|
112
|
+
return [start, end];
|
|
113
|
+
};
|
|
114
|
+
const getBindingStrategyForDraggingArrowOrJoints = (selectedElement, app, isBindingEnabled) => {
|
|
115
|
+
const [startIsClose, endIsClose] = getOriginalBindingsIfStillCloseToArrowEnds(selectedElement, app);
|
|
116
|
+
const start = startIsClose
|
|
117
|
+
? isBindingEnabled
|
|
118
|
+
? getElligibleElementForBindingElement(selectedElement, "start", app)
|
|
119
|
+
: null
|
|
120
|
+
: null;
|
|
121
|
+
const end = endIsClose
|
|
122
|
+
? isBindingEnabled
|
|
123
|
+
? getElligibleElementForBindingElement(selectedElement, "end", app)
|
|
124
|
+
: null
|
|
125
|
+
: null;
|
|
126
|
+
return [start, end];
|
|
127
|
+
};
|
|
128
|
+
export const bindOrUnbindLinearElements = (selectedElements, app, isBindingEnabled, draggingPoints) => {
|
|
129
|
+
selectedElements.forEach((selectedElement) => {
|
|
130
|
+
const [start, end] = draggingPoints?.length
|
|
131
|
+
? // The arrow edge points are dragged (i.e. start, end)
|
|
132
|
+
getBindingStrategyForDraggingArrowEndpoints(selectedElement, isBindingEnabled, draggingPoints ?? [], app)
|
|
133
|
+
: // The arrow itself (the shaft) or the inner joins are dragged
|
|
134
|
+
getBindingStrategyForDraggingArrowOrJoints(selectedElement, app, isBindingEnabled);
|
|
135
|
+
bindOrUnbindLinearElement(selectedElement, start, end, app.scene.getNonDeletedElementsMap());
|
|
72
136
|
});
|
|
73
137
|
};
|
|
74
|
-
const
|
|
75
|
-
|
|
138
|
+
export const getSuggestedBindingsForArrows = (selectedElements, app) => {
|
|
139
|
+
// HOT PATH: Bail out if selected elements list is too large
|
|
140
|
+
if (selectedElements.length > 50) {
|
|
141
|
+
return [];
|
|
142
|
+
}
|
|
143
|
+
return (selectedElements
|
|
144
|
+
.filter(isLinearElement)
|
|
145
|
+
.flatMap((element) => getOriginalBindingsIfStillCloseToArrowEnds(element, app))
|
|
146
|
+
.filter((element) => element !== null)
|
|
147
|
+
// Filter out bind candidates which are in the
|
|
148
|
+
// same selection / group with the arrow
|
|
149
|
+
//
|
|
150
|
+
// TODO: Is it worth turning the list into a set to avoid dupes?
|
|
151
|
+
.filter((element) => selectedElements.filter((selected) => selected.id === element?.id)
|
|
152
|
+
.length === 0));
|
|
76
153
|
};
|
|
77
|
-
export const maybeBindLinearElement = (linearElement, appState,
|
|
154
|
+
export const maybeBindLinearElement = (linearElement, appState, pointerCoords, app) => {
|
|
78
155
|
if (appState.startBoundElement != null) {
|
|
79
|
-
bindLinearElement(linearElement, appState.startBoundElement, "start",
|
|
156
|
+
bindLinearElement(linearElement, appState.startBoundElement, "start", app.scene.getNonDeletedElementsMap());
|
|
80
157
|
}
|
|
81
|
-
const hoveredElement = getHoveredElementForBinding(pointerCoords,
|
|
158
|
+
const hoveredElement = getHoveredElementForBinding(pointerCoords, app);
|
|
82
159
|
if (hoveredElement != null &&
|
|
83
160
|
!isLinearElementSimpleAndAlreadyBoundOnOppositeEdge(linearElement, hoveredElement, "end")) {
|
|
84
|
-
bindLinearElement(linearElement, hoveredElement, "end",
|
|
161
|
+
bindLinearElement(linearElement, hoveredElement, "end", app.scene.getNonDeletedElementsMap());
|
|
85
162
|
}
|
|
86
163
|
};
|
|
87
164
|
export const bindLinearElement = (linearElement, hoveredElement, startOrEnd, elementsMap) => {
|
|
@@ -107,15 +184,10 @@ const isLinearElementSimpleAndAlreadyBoundOnOppositeEdge = (linearElement, binda
|
|
|
107
184
|
return isLinearElementSimpleAndAlreadyBound(linearElement, otherBinding?.elementId, bindableElement);
|
|
108
185
|
};
|
|
109
186
|
export const isLinearElementSimpleAndAlreadyBound = (linearElement, alreadyBoundToId, bindableElement) => {
|
|
110
|
-
return (alreadyBoundToId === bindableElement.id &&
|
|
111
|
-
|
|
112
|
-
export const unbindLinearElements = (elements, elementsMap) => {
|
|
113
|
-
elements.forEach((element) => {
|
|
114
|
-
if (isBindingElement(element)) {
|
|
115
|
-
bindOrUnbindLinearElement(element, null, null, elementsMap);
|
|
116
|
-
}
|
|
117
|
-
});
|
|
187
|
+
return (alreadyBoundToId === bindableElement.id &&
|
|
188
|
+
isLinearElementSimple(linearElement));
|
|
118
189
|
};
|
|
190
|
+
const isLinearElementSimple = (linearElement) => linearElement.points.length < 3;
|
|
119
191
|
const unbindLinearElement = (linearElement, startOrEnd) => {
|
|
120
192
|
const field = startOrEnd === "start" ? "startBinding" : "endBinding";
|
|
121
193
|
const binding = linearElement[field];
|
|
@@ -125,9 +197,9 @@ const unbindLinearElement = (linearElement, startOrEnd) => {
|
|
|
125
197
|
mutateElement(linearElement, { [field]: null });
|
|
126
198
|
return binding.elementId;
|
|
127
199
|
};
|
|
128
|
-
export const getHoveredElementForBinding = (pointerCoords,
|
|
129
|
-
const hoveredElement = getElementAtPosition(
|
|
130
|
-
bindingBorderTest(element, pointerCoords,
|
|
200
|
+
export const getHoveredElementForBinding = (pointerCoords, app) => {
|
|
201
|
+
const hoveredElement = getElementAtPosition(app.scene.getNonDeletedElements(), (element) => isBindableElement(element, false) &&
|
|
202
|
+
bindingBorderTest(element, pointerCoords, app));
|
|
131
203
|
return hoveredElement;
|
|
132
204
|
};
|
|
133
205
|
const calculateFocusAndGap = (linearElement, hoveredElement, startOrEnd, elementsMap) => {
|
|
@@ -147,34 +219,38 @@ const calculateFocusAndGap = (linearElement, hoveredElement, startOrEnd, element
|
|
|
147
219
|
// done before the `changedElement` is updated, and the `newSize` is passed
|
|
148
220
|
// in explicitly.
|
|
149
221
|
export const updateBoundElements = (changedElement, elementsMap, options) => {
|
|
150
|
-
const boundLinearElements = (changedElement.boundElements ?? []).filter((el) => el.type === "arrow");
|
|
151
|
-
if (boundLinearElements.length === 0) {
|
|
152
|
-
return;
|
|
153
|
-
}
|
|
154
222
|
const { newSize, simultaneouslyUpdated } = options ?? {};
|
|
155
223
|
const simultaneouslyUpdatedElementIds = getSimultaneouslyUpdatedElementIds(simultaneouslyUpdated);
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
224
|
+
if (!isBindableElement(changedElement)) {
|
|
225
|
+
return;
|
|
226
|
+
}
|
|
227
|
+
boundElementsVisitor(elementsMap, changedElement, (element) => {
|
|
228
|
+
if (!isLinearElement(element) || element.isDeleted) {
|
|
159
229
|
return;
|
|
160
230
|
}
|
|
161
|
-
const bindableElement = changedElement;
|
|
162
231
|
// In case the boundElements are stale
|
|
163
|
-
if (!doesNeedUpdate(element,
|
|
232
|
+
if (!doesNeedUpdate(element, changedElement)) {
|
|
164
233
|
return;
|
|
165
234
|
}
|
|
166
|
-
const
|
|
167
|
-
|
|
235
|
+
const bindings = {
|
|
236
|
+
startBinding: maybeCalculateNewGapWhenScaling(changedElement, element.startBinding, newSize),
|
|
237
|
+
endBinding: maybeCalculateNewGapWhenScaling(changedElement, element.endBinding, newSize),
|
|
238
|
+
};
|
|
168
239
|
// `linearElement` is being moved/scaled already, just update the binding
|
|
169
240
|
if (simultaneouslyUpdatedElementIds.has(element.id)) {
|
|
170
|
-
mutateElement(element,
|
|
241
|
+
mutateElement(element, bindings);
|
|
171
242
|
return;
|
|
172
243
|
}
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
244
|
+
bindableElementsVisitor(elementsMap, element, (bindableElement, bindingProp) => {
|
|
245
|
+
if (bindableElement &&
|
|
246
|
+
isBindableElement(bindableElement) &&
|
|
247
|
+
(bindingProp === "startBinding" || bindingProp === "endBinding")) {
|
|
248
|
+
updateBoundPoint(element, bindingProp, bindings[bindingProp], bindableElement, elementsMap);
|
|
249
|
+
}
|
|
250
|
+
});
|
|
251
|
+
const boundText = getBoundTextElement(element, elementsMap);
|
|
252
|
+
if (boundText && !boundText.isDeleted) {
|
|
253
|
+
handleBindTextResize(element, elementsMap, false);
|
|
178
254
|
}
|
|
179
255
|
});
|
|
180
256
|
};
|
|
@@ -185,22 +261,18 @@ const doesNeedUpdate = (boundElement, changedElement) => {
|
|
|
185
261
|
const getSimultaneouslyUpdatedElementIds = (simultaneouslyUpdated) => {
|
|
186
262
|
return new Set((simultaneouslyUpdated || []).map((element) => element.id));
|
|
187
263
|
};
|
|
188
|
-
const updateBoundPoint = (linearElement, startOrEnd, binding,
|
|
264
|
+
const updateBoundPoint = (linearElement, startOrEnd, binding, bindableElement, elementsMap) => {
|
|
189
265
|
if (binding == null ||
|
|
190
266
|
// We only need to update the other end if this is a 2 point line element
|
|
191
|
-
(binding.elementId !==
|
|
192
|
-
|
|
193
|
-
}
|
|
194
|
-
const bindingElement = Scene.getScene(linearElement).getElement(binding.elementId);
|
|
195
|
-
if (bindingElement == null) {
|
|
196
|
-
// We're not cleaning up after deleted elements atm., so handle this case
|
|
267
|
+
(binding.elementId !== bindableElement.id &&
|
|
268
|
+
linearElement.points.length > 2)) {
|
|
197
269
|
return;
|
|
198
270
|
}
|
|
199
|
-
const direction = startOrEnd === "
|
|
271
|
+
const direction = startOrEnd === "startBinding" ? -1 : 1;
|
|
200
272
|
const edgePointIndex = direction === -1 ? 0 : linearElement.points.length - 1;
|
|
201
273
|
const adjacentPointIndex = edgePointIndex - direction;
|
|
202
274
|
const adjacentPoint = LinearElementEditor.getPointAtIndexGlobalCoordinates(linearElement, adjacentPointIndex, elementsMap);
|
|
203
|
-
const focusPointAbsolute = determineFocusPoint(
|
|
275
|
+
const focusPointAbsolute = determineFocusPoint(bindableElement, binding.focus, adjacentPoint, elementsMap);
|
|
204
276
|
let newEdgePoint;
|
|
205
277
|
// The linear element was not originally pointing inside the bound shape,
|
|
206
278
|
// we can point directly at the focus point
|
|
@@ -208,7 +280,7 @@ const updateBoundPoint = (linearElement, startOrEnd, binding, changedElement, el
|
|
|
208
280
|
newEdgePoint = focusPointAbsolute;
|
|
209
281
|
}
|
|
210
282
|
else {
|
|
211
|
-
const intersections = intersectElementWithLine(
|
|
283
|
+
const intersections = intersectElementWithLine(bindableElement, adjacentPoint, focusPointAbsolute, binding.gap, elementsMap);
|
|
212
284
|
if (intersections.length === 0) {
|
|
213
285
|
// This should never happen, since focusPoint should always be
|
|
214
286
|
// inside the element, but just in case, bail out
|
|
@@ -224,7 +296,7 @@ const updateBoundPoint = (linearElement, startOrEnd, binding, changedElement, el
|
|
|
224
296
|
index: edgePointIndex,
|
|
225
297
|
point: LinearElementEditor.pointFromAbsoluteCoords(linearElement, newEdgePoint, elementsMap),
|
|
226
298
|
},
|
|
227
|
-
], { [startOrEnd
|
|
299
|
+
], { [startOrEnd]: binding });
|
|
228
300
|
};
|
|
229
301
|
const maybeCalculateNewGapWhenScaling = (changedElement, currentBinding, newSize) => {
|
|
230
302
|
if (currentBinding == null || newSize == null) {
|
|
@@ -236,55 +308,13 @@ const maybeCalculateNewGapWhenScaling = (changedElement, currentBinding, newSize
|
|
|
236
308
|
const newGap = Math.max(1, Math.min(maxBindingGap(changedElement, newWidth, newHeight), gap * (newWidth < newHeight ? newWidth / width : newHeight / height)));
|
|
237
309
|
return { elementId, gap: newGap, focus };
|
|
238
310
|
};
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
const includedElementIds = new Set(selectedElements.map(({ id }) => id));
|
|
242
|
-
return selectedElements.flatMap((selectedElement) => isBindingElement(selectedElement, false)
|
|
243
|
-
? getElligibleElementsForBindingElement(selectedElement, elements, elementsMap).filter((element) => !includedElementIds.has(element.id))
|
|
244
|
-
: isBindableElement(selectedElement, false)
|
|
245
|
-
? getElligibleElementsForBindableElementAndWhere(selectedElement, elementsMap).filter((binding) => !includedElementIds.has(binding[0].id))
|
|
246
|
-
: []);
|
|
247
|
-
};
|
|
248
|
-
const getElligibleElementsForBindingElement = (linearElement, elements, elementsMap) => {
|
|
249
|
-
return [
|
|
250
|
-
getElligibleElementForBindingElement(linearElement, "start", elements, elementsMap),
|
|
251
|
-
getElligibleElementForBindingElement(linearElement, "end", elements, elementsMap),
|
|
252
|
-
].filter((element) => element != null);
|
|
253
|
-
};
|
|
254
|
-
const getElligibleElementForBindingElement = (linearElement, startOrEnd, elements, elementsMap) => {
|
|
255
|
-
return getHoveredElementForBinding(getLinearElementEdgeCoors(linearElement, startOrEnd, elementsMap), elements, elementsMap);
|
|
311
|
+
const getElligibleElementForBindingElement = (linearElement, startOrEnd, app) => {
|
|
312
|
+
return getHoveredElementForBinding(getLinearElementEdgeCoors(linearElement, startOrEnd, app.scene.getNonDeletedElementsMap()), app);
|
|
256
313
|
};
|
|
257
314
|
const getLinearElementEdgeCoors = (linearElement, startOrEnd, elementsMap) => {
|
|
258
315
|
const index = startOrEnd === "start" ? 0 : -1;
|
|
259
316
|
return tupleToCoors(LinearElementEditor.getPointAtIndexGlobalCoordinates(linearElement, index, elementsMap));
|
|
260
317
|
};
|
|
261
|
-
const getElligibleElementsForBindableElementAndWhere = (bindableElement, elementsMap) => {
|
|
262
|
-
const scene = Scene.getScene(bindableElement);
|
|
263
|
-
return scene
|
|
264
|
-
.getNonDeletedElements()
|
|
265
|
-
.map((element) => {
|
|
266
|
-
if (!isBindingElement(element, false)) {
|
|
267
|
-
return null;
|
|
268
|
-
}
|
|
269
|
-
const canBindStart = isLinearElementEligibleForNewBindingByBindable(element, "start", bindableElement, elementsMap);
|
|
270
|
-
const canBindEnd = isLinearElementEligibleForNewBindingByBindable(element, "end", bindableElement, elementsMap);
|
|
271
|
-
if (!canBindStart && !canBindEnd) {
|
|
272
|
-
return null;
|
|
273
|
-
}
|
|
274
|
-
return [
|
|
275
|
-
element,
|
|
276
|
-
canBindStart && canBindEnd ? "both" : canBindStart ? "start" : "end",
|
|
277
|
-
bindableElement,
|
|
278
|
-
];
|
|
279
|
-
})
|
|
280
|
-
.filter((maybeElement) => maybeElement != null);
|
|
281
|
-
};
|
|
282
|
-
const isLinearElementEligibleForNewBindingByBindable = (linearElement, startOrEnd, bindableElement, elementsMap) => {
|
|
283
|
-
const existingBinding = linearElement[startOrEnd === "start" ? "startBinding" : "endBinding"];
|
|
284
|
-
return (existingBinding == null &&
|
|
285
|
-
!isLinearElementSimpleAndAlreadyBoundOnOppositeEdge(linearElement, bindableElement, startOrEnd) &&
|
|
286
|
-
bindingBorderTest(bindableElement, getLinearElementEdgeCoors(linearElement, startOrEnd, elementsMap), elementsMap));
|
|
287
|
-
};
|
|
288
318
|
// We need to:
|
|
289
319
|
// 1: Update elements not selected to point to duplicated elements
|
|
290
320
|
// 2: Update duplicated elements to point to other duplicated elements
|
|
@@ -364,51 +394,578 @@ const newBindingAfterDuplication = (binding, oldIdToDuplicatedId) => {
|
|
|
364
394
|
};
|
|
365
395
|
};
|
|
366
396
|
export const fixBindingsAfterDeletion = (sceneElements, deletedElements) => {
|
|
367
|
-
const
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
397
|
+
const elements = arrayToMap(sceneElements);
|
|
398
|
+
for (const element of deletedElements) {
|
|
399
|
+
BoundElement.unbindAffected(elements, element, mutateElement);
|
|
400
|
+
BindableElement.unbindAffected(elements, element, mutateElement);
|
|
401
|
+
}
|
|
402
|
+
};
|
|
403
|
+
const newBoundElements = (boundElements, idsToRemove, elementsToAdd = []) => {
|
|
404
|
+
if (!boundElements) {
|
|
405
|
+
return null;
|
|
406
|
+
}
|
|
407
|
+
const nextBoundElements = boundElements.filter((boundElement) => !idsToRemove.has(boundElement.id));
|
|
408
|
+
nextBoundElements.push(...elementsToAdd.map((x) => ({ id: x.id, type: x.type })));
|
|
409
|
+
return nextBoundElements;
|
|
410
|
+
};
|
|
411
|
+
const bindingBorderTest = (element, { x, y }, app) => {
|
|
412
|
+
const threshold = maxBindingGap(element, element.width, element.height);
|
|
413
|
+
const shape = app.getElementShape(element);
|
|
414
|
+
return isPointOnShape([x, y], shape, threshold);
|
|
415
|
+
};
|
|
416
|
+
export const maxBindingGap = (element, elementWidth, elementHeight) => {
|
|
417
|
+
// Aligns diamonds with rectangles
|
|
418
|
+
const shapeRatio = element.type === "diamond" ? 1 / Math.sqrt(2) : 1;
|
|
419
|
+
const smallerDimension = shapeRatio * Math.min(elementWidth, elementHeight);
|
|
420
|
+
// We make the bindable boundary bigger for bigger elements
|
|
421
|
+
return Math.max(16, Math.min(0.25 * smallerDimension, 32));
|
|
422
|
+
};
|
|
423
|
+
const distanceToBindableElement = (element, point, elementsMap) => {
|
|
424
|
+
switch (element.type) {
|
|
425
|
+
case "rectangle":
|
|
426
|
+
case "image":
|
|
427
|
+
case "text":
|
|
428
|
+
case "iframe":
|
|
429
|
+
case "embeddable":
|
|
430
|
+
case "frame":
|
|
431
|
+
case "magicframe":
|
|
432
|
+
return distanceToRectangle(element, point, elementsMap);
|
|
433
|
+
case "diamond":
|
|
434
|
+
return distanceToDiamond(element, point, elementsMap);
|
|
435
|
+
case "ellipse":
|
|
436
|
+
return distanceToEllipse(element, point, elementsMap);
|
|
437
|
+
}
|
|
438
|
+
};
|
|
439
|
+
const distanceToRectangle = (element, point, elementsMap) => {
|
|
440
|
+
const [, pointRel, hwidth, hheight] = pointRelativeToElement(element, point, elementsMap);
|
|
441
|
+
return Math.max(GAPoint.distanceToLine(pointRel, GALine.equation(0, 1, -hheight)), GAPoint.distanceToLine(pointRel, GALine.equation(1, 0, -hwidth)));
|
|
442
|
+
};
|
|
443
|
+
const distanceToDiamond = (element, point, elementsMap) => {
|
|
444
|
+
const [, pointRel, hwidth, hheight] = pointRelativeToElement(element, point, elementsMap);
|
|
445
|
+
const side = GALine.equation(hheight, hwidth, -hheight * hwidth);
|
|
446
|
+
return GAPoint.distanceToLine(pointRel, side);
|
|
447
|
+
};
|
|
448
|
+
const distanceToEllipse = (element, point, elementsMap) => {
|
|
449
|
+
const [pointRel, tangent] = ellipseParamsForTest(element, point, elementsMap);
|
|
450
|
+
return -GALine.sign(tangent) * GAPoint.distanceToLine(pointRel, tangent);
|
|
451
|
+
};
|
|
452
|
+
const ellipseParamsForTest = (element, point, elementsMap) => {
|
|
453
|
+
const [, pointRel, hwidth, hheight] = pointRelativeToElement(element, point, elementsMap);
|
|
454
|
+
const [px, py] = GAPoint.toTuple(pointRel);
|
|
455
|
+
// We're working in positive quadrant, so start with `t = 45deg`, `tx=cos(t)`
|
|
456
|
+
let tx = 0.707;
|
|
457
|
+
let ty = 0.707;
|
|
458
|
+
const a = hwidth;
|
|
459
|
+
const b = hheight;
|
|
460
|
+
// This is a numerical method to find the params tx, ty at which
|
|
461
|
+
// the ellipse has the closest point to the given point
|
|
462
|
+
[0, 1, 2, 3].forEach((_) => {
|
|
463
|
+
const xx = a * tx;
|
|
464
|
+
const yy = b * ty;
|
|
465
|
+
const ex = ((a * a - b * b) * tx ** 3) / a;
|
|
466
|
+
const ey = ((b * b - a * a) * ty ** 3) / b;
|
|
467
|
+
const rx = xx - ex;
|
|
468
|
+
const ry = yy - ey;
|
|
469
|
+
const qx = px - ex;
|
|
470
|
+
const qy = py - ey;
|
|
471
|
+
const r = Math.hypot(ry, rx);
|
|
472
|
+
const q = Math.hypot(qy, qx);
|
|
473
|
+
tx = Math.min(1, Math.max(0, ((qx * r) / q + ex) / a));
|
|
474
|
+
ty = Math.min(1, Math.max(0, ((qy * r) / q + ey) / b));
|
|
475
|
+
const t = Math.hypot(ty, tx);
|
|
476
|
+
tx /= t;
|
|
477
|
+
ty /= t;
|
|
386
478
|
});
|
|
387
|
-
|
|
388
|
-
|
|
389
|
-
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
479
|
+
const closestPoint = GA.point(a * tx, b * ty);
|
|
480
|
+
const tangent = GALine.orthogonalThrough(pointRel, closestPoint);
|
|
481
|
+
return [pointRel, tangent];
|
|
482
|
+
};
|
|
483
|
+
// Returns:
|
|
484
|
+
// 1. the point relative to the elements (x, y) position
|
|
485
|
+
// 2. the point relative to the element's center with positive (x, y)
|
|
486
|
+
// 3. half element width
|
|
487
|
+
// 4. half element height
|
|
488
|
+
//
|
|
489
|
+
// Note that for linear elements the (x, y) position is not at the
|
|
490
|
+
// top right corner of their boundary.
|
|
491
|
+
//
|
|
492
|
+
// Rectangles, diamonds and ellipses are symmetrical over axes,
|
|
493
|
+
// and other elements have a rectangular boundary,
|
|
494
|
+
// so we only need to perform hit tests for the positive quadrant.
|
|
495
|
+
const pointRelativeToElement = (element, pointTuple, elementsMap) => {
|
|
496
|
+
const point = GAPoint.from(pointTuple);
|
|
497
|
+
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
|
498
|
+
const center = coordsCenter(x1, y1, x2, y2);
|
|
499
|
+
// GA has angle orientation opposite to `rotate`
|
|
500
|
+
const rotate = GATransform.rotation(center, element.angle);
|
|
501
|
+
const pointRotated = GATransform.apply(rotate, point);
|
|
502
|
+
const pointRelToCenter = GA.sub(pointRotated, GADirection.from(center));
|
|
503
|
+
const pointRelToCenterAbs = GAPoint.abs(pointRelToCenter);
|
|
504
|
+
const elementPos = GA.offset(element.x, element.y);
|
|
505
|
+
const pointRelToPos = GA.sub(pointRotated, elementPos);
|
|
506
|
+
const halfWidth = (x2 - x1) / 2;
|
|
507
|
+
const halfHeight = (y2 - y1) / 2;
|
|
508
|
+
return [pointRelToPos, pointRelToCenterAbs, halfWidth, halfHeight];
|
|
509
|
+
};
|
|
510
|
+
const relativizationToElementCenter = (element, elementsMap) => {
|
|
511
|
+
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
|
512
|
+
const center = coordsCenter(x1, y1, x2, y2);
|
|
513
|
+
// GA has angle orientation opposite to `rotate`
|
|
514
|
+
const rotate = GATransform.rotation(center, element.angle);
|
|
515
|
+
const translate = GA.reverse(GATransform.translation(GADirection.from(center)));
|
|
516
|
+
return GATransform.compose(rotate, translate);
|
|
517
|
+
};
|
|
518
|
+
const coordsCenter = (x1, y1, x2, y2) => {
|
|
519
|
+
return GA.point((x1 + x2) / 2, (y1 + y2) / 2);
|
|
520
|
+
};
|
|
521
|
+
// The focus distance is the oriented ratio between the size of
|
|
522
|
+
// the `element` and the "focus image" of the element on which
|
|
523
|
+
// all focus points lie, so it's a number between -1 and 1.
|
|
524
|
+
// The line going through `a` and `b` is a tangent to the "focus image"
|
|
525
|
+
// of the element.
|
|
526
|
+
const determineFocusDistance = (element,
|
|
527
|
+
// Point on the line, in absolute coordinates
|
|
528
|
+
a,
|
|
529
|
+
// Another point on the line, in absolute coordinates (closer to element)
|
|
530
|
+
b, elementsMap) => {
|
|
531
|
+
const relateToCenter = relativizationToElementCenter(element, elementsMap);
|
|
532
|
+
const aRel = GATransform.apply(relateToCenter, GAPoint.from(a));
|
|
533
|
+
const bRel = GATransform.apply(relateToCenter, GAPoint.from(b));
|
|
534
|
+
const line = GALine.through(aRel, bRel);
|
|
535
|
+
const q = element.height / element.width;
|
|
536
|
+
const hwidth = element.width / 2;
|
|
537
|
+
const hheight = element.height / 2;
|
|
538
|
+
const n = line[2];
|
|
539
|
+
const m = line[3];
|
|
540
|
+
const c = line[1];
|
|
541
|
+
const mabs = Math.abs(m);
|
|
542
|
+
const nabs = Math.abs(n);
|
|
543
|
+
let ret;
|
|
544
|
+
switch (element.type) {
|
|
545
|
+
case "rectangle":
|
|
546
|
+
case "image":
|
|
547
|
+
case "text":
|
|
548
|
+
case "iframe":
|
|
549
|
+
case "embeddable":
|
|
550
|
+
case "frame":
|
|
551
|
+
case "magicframe":
|
|
552
|
+
ret = c / (hwidth * (nabs + q * mabs));
|
|
553
|
+
break;
|
|
554
|
+
case "diamond":
|
|
555
|
+
ret = mabs < nabs ? c / (nabs * hwidth) : c / (mabs * hheight);
|
|
556
|
+
break;
|
|
557
|
+
case "ellipse":
|
|
558
|
+
ret = c / (hwidth * Math.sqrt(n ** 2 + q ** 2 * m ** 2));
|
|
559
|
+
break;
|
|
560
|
+
}
|
|
561
|
+
return ret || 0;
|
|
562
|
+
};
|
|
563
|
+
const determineFocusPoint = (element,
|
|
564
|
+
// The oriented, relative distance from the center of `element` of the
|
|
565
|
+
// returned focusPoint
|
|
566
|
+
focus, adjecentPoint, elementsMap) => {
|
|
567
|
+
if (focus === 0) {
|
|
568
|
+
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
|
569
|
+
const center = coordsCenter(x1, y1, x2, y2);
|
|
570
|
+
return GAPoint.toTuple(center);
|
|
571
|
+
}
|
|
572
|
+
const relateToCenter = relativizationToElementCenter(element, elementsMap);
|
|
573
|
+
const adjecentPointRel = GATransform.apply(relateToCenter, GAPoint.from(adjecentPoint));
|
|
574
|
+
const reverseRelateToCenter = GA.reverse(relateToCenter);
|
|
575
|
+
let point;
|
|
576
|
+
switch (element.type) {
|
|
577
|
+
case "rectangle":
|
|
578
|
+
case "image":
|
|
579
|
+
case "text":
|
|
580
|
+
case "diamond":
|
|
581
|
+
case "iframe":
|
|
582
|
+
case "embeddable":
|
|
583
|
+
case "frame":
|
|
584
|
+
case "magicframe":
|
|
585
|
+
point = findFocusPointForRectangulars(element, focus, adjecentPointRel);
|
|
586
|
+
break;
|
|
587
|
+
case "ellipse":
|
|
588
|
+
point = findFocusPointForEllipse(element, focus, adjecentPointRel);
|
|
589
|
+
break;
|
|
590
|
+
}
|
|
591
|
+
return GAPoint.toTuple(GATransform.apply(reverseRelateToCenter, point));
|
|
592
|
+
};
|
|
593
|
+
// Returns 2 or 0 intersection points between line going through `a` and `b`
|
|
594
|
+
// and the `element`, in ascending order of distance from `a`.
|
|
595
|
+
const intersectElementWithLine = (element,
|
|
596
|
+
// Point on the line, in absolute coordinates
|
|
597
|
+
a,
|
|
598
|
+
// Another point on the line, in absolute coordinates
|
|
599
|
+
b,
|
|
600
|
+
// If given, the element is inflated by this value
|
|
601
|
+
gap = 0, elementsMap) => {
|
|
602
|
+
const relateToCenter = relativizationToElementCenter(element, elementsMap);
|
|
603
|
+
const aRel = GATransform.apply(relateToCenter, GAPoint.from(a));
|
|
604
|
+
const bRel = GATransform.apply(relateToCenter, GAPoint.from(b));
|
|
605
|
+
const line = GALine.through(aRel, bRel);
|
|
606
|
+
const reverseRelateToCenter = GA.reverse(relateToCenter);
|
|
607
|
+
const intersections = getSortedElementLineIntersections(element, line, aRel, gap);
|
|
608
|
+
return intersections.map((point) => GAPoint.toTuple(GATransform.apply(reverseRelateToCenter, point)));
|
|
609
|
+
};
|
|
610
|
+
const getSortedElementLineIntersections = (element,
|
|
611
|
+
// Relative to element center
|
|
612
|
+
line,
|
|
613
|
+
// Relative to element center
|
|
614
|
+
nearPoint, gap = 0) => {
|
|
615
|
+
let intersections;
|
|
616
|
+
switch (element.type) {
|
|
617
|
+
case "rectangle":
|
|
618
|
+
case "image":
|
|
619
|
+
case "text":
|
|
620
|
+
case "diamond":
|
|
621
|
+
case "iframe":
|
|
622
|
+
case "embeddable":
|
|
623
|
+
case "frame":
|
|
624
|
+
case "magicframe":
|
|
625
|
+
const corners = getCorners(element);
|
|
626
|
+
intersections = corners
|
|
627
|
+
.flatMap((point, i) => {
|
|
628
|
+
const edge = [point, corners[(i + 1) % 4]];
|
|
629
|
+
return intersectSegment(line, offsetSegment(edge, gap));
|
|
630
|
+
})
|
|
631
|
+
.concat(corners.flatMap((point) => getCircleIntersections(point, gap, line)));
|
|
632
|
+
break;
|
|
633
|
+
case "ellipse":
|
|
634
|
+
intersections = getEllipseIntersections(element, gap, line);
|
|
635
|
+
break;
|
|
636
|
+
}
|
|
637
|
+
if (intersections.length < 2) {
|
|
638
|
+
// Ignore the "edge" case of only intersecting with a single corner
|
|
639
|
+
return [];
|
|
640
|
+
}
|
|
641
|
+
const sortedIntersections = intersections.sort((i1, i2) => GAPoint.distance(i1, nearPoint) - GAPoint.distance(i2, nearPoint));
|
|
642
|
+
return [
|
|
643
|
+
sortedIntersections[0],
|
|
644
|
+
sortedIntersections[sortedIntersections.length - 1],
|
|
645
|
+
];
|
|
646
|
+
};
|
|
647
|
+
const getCorners = (element, scale = 1) => {
|
|
648
|
+
const hx = (scale * element.width) / 2;
|
|
649
|
+
const hy = (scale * element.height) / 2;
|
|
650
|
+
switch (element.type) {
|
|
651
|
+
case "rectangle":
|
|
652
|
+
case "image":
|
|
653
|
+
case "text":
|
|
654
|
+
case "iframe":
|
|
655
|
+
case "embeddable":
|
|
656
|
+
case "frame":
|
|
657
|
+
case "magicframe":
|
|
658
|
+
return [
|
|
659
|
+
GA.point(hx, hy),
|
|
660
|
+
GA.point(hx, -hy),
|
|
661
|
+
GA.point(-hx, -hy),
|
|
662
|
+
GA.point(-hx, hy),
|
|
663
|
+
];
|
|
664
|
+
case "diamond":
|
|
665
|
+
return [
|
|
666
|
+
GA.point(0, hy),
|
|
667
|
+
GA.point(hx, 0),
|
|
668
|
+
GA.point(0, -hy),
|
|
669
|
+
GA.point(-hx, 0),
|
|
670
|
+
];
|
|
671
|
+
}
|
|
672
|
+
};
|
|
673
|
+
// Returns intersection of `line` with `segment`, with `segment` moved by
|
|
674
|
+
// `gap` in its polar direction.
|
|
675
|
+
// If intersection coincides with second segment point returns empty array.
|
|
676
|
+
const intersectSegment = (line, segment) => {
|
|
677
|
+
const [a, b] = segment;
|
|
678
|
+
const aDist = GAPoint.distanceToLine(a, line);
|
|
679
|
+
const bDist = GAPoint.distanceToLine(b, line);
|
|
680
|
+
if (aDist * bDist >= 0) {
|
|
681
|
+
// The intersection is outside segment `(a, b)`
|
|
682
|
+
return [];
|
|
683
|
+
}
|
|
684
|
+
return [GAPoint.intersect(line, GALine.through(a, b))];
|
|
685
|
+
};
|
|
686
|
+
const offsetSegment = (segment, distance) => {
|
|
687
|
+
const [a, b] = segment;
|
|
688
|
+
const offset = GATransform.translationOrthogonal(GADirection.fromTo(a, b), distance);
|
|
689
|
+
return [GATransform.apply(offset, a), GATransform.apply(offset, b)];
|
|
690
|
+
};
|
|
691
|
+
const getEllipseIntersections = (element, gap, line) => {
|
|
692
|
+
const a = element.width / 2 + gap;
|
|
693
|
+
const b = element.height / 2 + gap;
|
|
694
|
+
const m = line[2];
|
|
695
|
+
const n = line[3];
|
|
696
|
+
const c = line[1];
|
|
697
|
+
const squares = a * a * m * m + b * b * n * n;
|
|
698
|
+
const discr = squares - c * c;
|
|
699
|
+
if (squares === 0 || discr <= 0) {
|
|
700
|
+
return [];
|
|
701
|
+
}
|
|
702
|
+
const discrRoot = Math.sqrt(discr);
|
|
703
|
+
const xn = -a * a * m * c;
|
|
704
|
+
const yn = -b * b * n * c;
|
|
705
|
+
return [
|
|
706
|
+
GA.point((xn + a * b * n * discrRoot) / squares, (yn - a * b * m * discrRoot) / squares),
|
|
707
|
+
GA.point((xn - a * b * n * discrRoot) / squares, (yn + a * b * m * discrRoot) / squares),
|
|
708
|
+
];
|
|
709
|
+
};
|
|
710
|
+
const getCircleIntersections = (center, radius, line) => {
|
|
711
|
+
if (radius === 0) {
|
|
712
|
+
return GAPoint.distanceToLine(line, center) === 0 ? [center] : [];
|
|
713
|
+
}
|
|
714
|
+
const m = line[2];
|
|
715
|
+
const n = line[3];
|
|
716
|
+
const c = line[1];
|
|
717
|
+
const [a, b] = GAPoint.toTuple(center);
|
|
718
|
+
const r = radius;
|
|
719
|
+
const squares = m * m + n * n;
|
|
720
|
+
const discr = r * r * squares - (m * a + n * b + c) ** 2;
|
|
721
|
+
if (squares === 0 || discr <= 0) {
|
|
722
|
+
return [];
|
|
723
|
+
}
|
|
724
|
+
const discrRoot = Math.sqrt(discr);
|
|
725
|
+
const xn = a * n * n - b * m * n - m * c;
|
|
726
|
+
const yn = b * m * m - a * m * n - n * c;
|
|
727
|
+
return [
|
|
728
|
+
GA.point((xn + n * discrRoot) / squares, (yn - m * discrRoot) / squares),
|
|
729
|
+
GA.point((xn - n * discrRoot) / squares, (yn + m * discrRoot) / squares),
|
|
730
|
+
];
|
|
731
|
+
};
|
|
732
|
+
// The focus point is the tangent point of the "focus image" of the
|
|
733
|
+
// `element`, where the tangent goes through `point`.
|
|
734
|
+
const findFocusPointForEllipse = (ellipse,
|
|
735
|
+
// Between -1 and 1 (not 0) the relative size of the "focus image" of
|
|
736
|
+
// the element on which the focus point lies
|
|
737
|
+
relativeDistance,
|
|
738
|
+
// The point for which we're trying to find the focus point, relative
|
|
739
|
+
// to the ellipse center.
|
|
740
|
+
point) => {
|
|
741
|
+
const relativeDistanceAbs = Math.abs(relativeDistance);
|
|
742
|
+
const a = (ellipse.width * relativeDistanceAbs) / 2;
|
|
743
|
+
const b = (ellipse.height * relativeDistanceAbs) / 2;
|
|
744
|
+
const orientation = Math.sign(relativeDistance);
|
|
745
|
+
const [px, pyo] = GAPoint.toTuple(point);
|
|
746
|
+
// The calculation below can't handle py = 0
|
|
747
|
+
const py = pyo === 0 ? 0.0001 : pyo;
|
|
748
|
+
const squares = px ** 2 * b ** 2 + py ** 2 * a ** 2;
|
|
749
|
+
// Tangent mx + ny + 1 = 0
|
|
750
|
+
const m = (-px * b ** 2 +
|
|
751
|
+
orientation * py * Math.sqrt(Math.max(0, squares - a ** 2 * b ** 2))) /
|
|
752
|
+
squares;
|
|
753
|
+
let n = (-m * px - 1) / py;
|
|
754
|
+
if (n === 0) {
|
|
755
|
+
// if zero {-0, 0}, fall back to a same-sign value in the similar range
|
|
756
|
+
n = (Object.is(n, -0) ? -1 : 1) * 0.01;
|
|
757
|
+
}
|
|
758
|
+
const x = -(a ** 2 * m) / (n ** 2 * b ** 2 + m ** 2 * a ** 2);
|
|
759
|
+
return GA.point(x, (-m * x - 1) / n);
|
|
760
|
+
};
|
|
761
|
+
const findFocusPointForRectangulars = (element,
|
|
762
|
+
// Between -1 and 1 for how far away should the focus point be relative
|
|
763
|
+
// to the size of the element. Sign determines orientation.
|
|
764
|
+
relativeDistance,
|
|
765
|
+
// The point for which we're trying to find the focus point, relative
|
|
766
|
+
// to the element center.
|
|
767
|
+
point) => {
|
|
768
|
+
const relativeDistanceAbs = Math.abs(relativeDistance);
|
|
769
|
+
const orientation = Math.sign(relativeDistance);
|
|
770
|
+
const corners = getCorners(element, relativeDistanceAbs);
|
|
771
|
+
let maxDistance = 0;
|
|
772
|
+
let tangentPoint = null;
|
|
773
|
+
corners.forEach((corner) => {
|
|
774
|
+
const distance = orientation * GALine.through(point, corner)[1];
|
|
775
|
+
if (distance > maxDistance) {
|
|
776
|
+
maxDistance = distance;
|
|
777
|
+
tangentPoint = corner;
|
|
400
778
|
}
|
|
401
779
|
});
|
|
780
|
+
return tangentPoint;
|
|
402
781
|
};
|
|
403
|
-
const
|
|
404
|
-
|
|
405
|
-
|
|
782
|
+
export const bindingProperties = new Set([
|
|
783
|
+
"boundElements",
|
|
784
|
+
"frameId",
|
|
785
|
+
"containerId",
|
|
786
|
+
"startBinding",
|
|
787
|
+
"endBinding",
|
|
788
|
+
]);
|
|
789
|
+
/**
|
|
790
|
+
* Tries to visit each bound element (does not have to be found).
|
|
791
|
+
*/
|
|
792
|
+
const boundElementsVisitor = (elements, element, visit) => {
|
|
793
|
+
if (isBindableElement(element)) {
|
|
794
|
+
// create new instance so that possible mutations won't play a role in visiting order
|
|
795
|
+
const boundElements = element.boundElements?.slice() ?? [];
|
|
796
|
+
// last added text should be the one we keep (~previous are duplicates)
|
|
797
|
+
boundElements.forEach(({ id }) => {
|
|
798
|
+
visit(elements.get(id), "boundElements", id);
|
|
799
|
+
});
|
|
406
800
|
}
|
|
407
|
-
return binding;
|
|
408
801
|
};
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
802
|
+
/**
|
|
803
|
+
* Tries to visit each bindable element (does not have to be found).
|
|
804
|
+
*/
|
|
805
|
+
const bindableElementsVisitor = (elements, element, visit) => {
|
|
806
|
+
if (element.frameId) {
|
|
807
|
+
const id = element.frameId;
|
|
808
|
+
visit(elements.get(id), "frameId", id);
|
|
809
|
+
}
|
|
810
|
+
if (isBoundToContainer(element)) {
|
|
811
|
+
const id = element.containerId;
|
|
812
|
+
visit(elements.get(id), "containerId", id);
|
|
813
|
+
}
|
|
814
|
+
if (isArrowElement(element)) {
|
|
815
|
+
if (element.startBinding) {
|
|
816
|
+
const id = element.startBinding.elementId;
|
|
817
|
+
visit(elements.get(id), "startBinding", id);
|
|
818
|
+
}
|
|
819
|
+
if (element.endBinding) {
|
|
820
|
+
const id = element.endBinding.elementId;
|
|
821
|
+
visit(elements.get(id), "endBinding", id);
|
|
822
|
+
}
|
|
412
823
|
}
|
|
413
|
-
return boundElements.filter((ele) => !deletedElementIds.has(ele.id));
|
|
414
824
|
};
|
|
825
|
+
/**
|
|
826
|
+
* Bound element containing bindings to `frameId`, `containerId`, `startBinding` or `endBinding`.
|
|
827
|
+
*/
|
|
828
|
+
export class BoundElement {
|
|
829
|
+
/**
|
|
830
|
+
* Unbind the affected non deleted bindable elements (removing element from `boundElements`).
|
|
831
|
+
* - iterates non deleted bindable elements (`containerId` | `startBinding.elementId` | `endBinding.elementId`) of the current element
|
|
832
|
+
* - prepares updates to unbind each bindable element's `boundElements` from the current element
|
|
833
|
+
*/
|
|
834
|
+
static unbindAffected(elements, boundElement, updateElementWith) {
|
|
835
|
+
if (!boundElement) {
|
|
836
|
+
return;
|
|
837
|
+
}
|
|
838
|
+
bindableElementsVisitor(elements, boundElement, (bindableElement) => {
|
|
839
|
+
// bindable element is deleted, this is fine
|
|
840
|
+
if (!bindableElement || bindableElement.isDeleted) {
|
|
841
|
+
return;
|
|
842
|
+
}
|
|
843
|
+
boundElementsVisitor(elements, bindableElement, (_, __, boundElementId) => {
|
|
844
|
+
if (boundElementId === boundElement.id) {
|
|
845
|
+
updateElementWith(bindableElement, {
|
|
846
|
+
boundElements: newBoundElements(bindableElement.boundElements, new Set([boundElementId])),
|
|
847
|
+
});
|
|
848
|
+
}
|
|
849
|
+
});
|
|
850
|
+
});
|
|
851
|
+
}
|
|
852
|
+
/**
|
|
853
|
+
* Rebind the next affected non deleted bindable elements (adding element to `boundElements`).
|
|
854
|
+
* - iterates non deleted bindable elements (`containerId` | `startBinding.elementId` | `endBinding.elementId`) of the current element
|
|
855
|
+
* - prepares updates to rebind each bindable element's `boundElements` to the current element
|
|
856
|
+
*
|
|
857
|
+
* NOTE: rebind expects that affected elements were previously unbound with `BoundElement.unbindAffected`
|
|
858
|
+
*/
|
|
859
|
+
static rebindAffected = (elements, boundElement, updateElementWith) => {
|
|
860
|
+
// don't try to rebind element that is deleted
|
|
861
|
+
if (!boundElement || boundElement.isDeleted) {
|
|
862
|
+
return;
|
|
863
|
+
}
|
|
864
|
+
bindableElementsVisitor(elements, boundElement, (bindableElement, bindingProp) => {
|
|
865
|
+
// unbind from bindable elements, as bindings from non deleted elements into deleted elements are incorrect
|
|
866
|
+
if (!bindableElement || bindableElement.isDeleted) {
|
|
867
|
+
updateElementWith(boundElement, { [bindingProp]: null });
|
|
868
|
+
return;
|
|
869
|
+
}
|
|
870
|
+
// frame bindings are unidirectional, there is nothing to rebind
|
|
871
|
+
if (bindingProp === "frameId") {
|
|
872
|
+
return;
|
|
873
|
+
}
|
|
874
|
+
if (bindableElement.boundElements?.find((x) => x.id === boundElement.id)) {
|
|
875
|
+
return;
|
|
876
|
+
}
|
|
877
|
+
if (isArrowElement(boundElement)) {
|
|
878
|
+
// rebind if not found!
|
|
879
|
+
updateElementWith(bindableElement, {
|
|
880
|
+
boundElements: newBoundElements(bindableElement.boundElements, new Set(), new Array(boundElement)),
|
|
881
|
+
});
|
|
882
|
+
}
|
|
883
|
+
if (isTextElement(boundElement)) {
|
|
884
|
+
if (!bindableElement.boundElements?.find((x) => x.type === "text")) {
|
|
885
|
+
// rebind only if there is no other text bound already
|
|
886
|
+
updateElementWith(bindableElement, {
|
|
887
|
+
boundElements: newBoundElements(bindableElement.boundElements, new Set(), new Array(boundElement)),
|
|
888
|
+
});
|
|
889
|
+
}
|
|
890
|
+
else {
|
|
891
|
+
// unbind otherwise
|
|
892
|
+
updateElementWith(boundElement, { [bindingProp]: null });
|
|
893
|
+
}
|
|
894
|
+
}
|
|
895
|
+
});
|
|
896
|
+
};
|
|
897
|
+
}
|
|
898
|
+
/**
|
|
899
|
+
* Bindable element containing bindings to `boundElements`.
|
|
900
|
+
*/
|
|
901
|
+
export class BindableElement {
|
|
902
|
+
/**
|
|
903
|
+
* Unbind the affected non deleted bound elements (resetting `containerId`, `startBinding`, `endBinding` to `null`).
|
|
904
|
+
* - iterates through non deleted `boundElements` of the current element
|
|
905
|
+
* - prepares updates to unbind each bound element from the current element
|
|
906
|
+
*/
|
|
907
|
+
static unbindAffected(elements, bindableElement, updateElementWith) {
|
|
908
|
+
if (!bindableElement) {
|
|
909
|
+
return;
|
|
910
|
+
}
|
|
911
|
+
boundElementsVisitor(elements, bindableElement, (boundElement) => {
|
|
912
|
+
// bound element is deleted, this is fine
|
|
913
|
+
if (!boundElement || boundElement.isDeleted) {
|
|
914
|
+
return;
|
|
915
|
+
}
|
|
916
|
+
bindableElementsVisitor(elements, boundElement, (_, bindingProp, bindableElementId) => {
|
|
917
|
+
// making sure there is an element to be unbound
|
|
918
|
+
if (bindableElementId === bindableElement.id) {
|
|
919
|
+
updateElementWith(boundElement, { [bindingProp]: null });
|
|
920
|
+
}
|
|
921
|
+
});
|
|
922
|
+
});
|
|
923
|
+
}
|
|
924
|
+
/**
|
|
925
|
+
* Rebind the affected non deleted bound elements (for now setting only `containerId`, as we cannot rebind arrows atm).
|
|
926
|
+
* - iterates through non deleted `boundElements` of the current element
|
|
927
|
+
* - prepares updates to rebind each bound element to the current element or unbind it from `boundElements` in case of conflicts
|
|
928
|
+
*
|
|
929
|
+
* NOTE: rebind expects that affected elements were previously unbound with `BindaleElement.unbindAffected`
|
|
930
|
+
*/
|
|
931
|
+
static rebindAffected = (elements, bindableElement, updateElementWith) => {
|
|
932
|
+
// don't try to rebind element that is deleted (i.e. updated as deleted)
|
|
933
|
+
if (!bindableElement || bindableElement.isDeleted) {
|
|
934
|
+
return;
|
|
935
|
+
}
|
|
936
|
+
boundElementsVisitor(elements, bindableElement, (boundElement, _, boundElementId) => {
|
|
937
|
+
// unbind from bindable elements, as bindings from non deleted elements into deleted elements are incorrect
|
|
938
|
+
if (!boundElement || boundElement.isDeleted) {
|
|
939
|
+
updateElementWith(bindableElement, {
|
|
940
|
+
boundElements: newBoundElements(bindableElement.boundElements, new Set([boundElementId])),
|
|
941
|
+
});
|
|
942
|
+
return;
|
|
943
|
+
}
|
|
944
|
+
if (isTextElement(boundElement)) {
|
|
945
|
+
const boundElements = bindableElement.boundElements?.slice() ?? [];
|
|
946
|
+
// check if this is the last element in the array, if not, there is an previously bound text which should be unbound
|
|
947
|
+
if (boundElements.reverse().find((x) => x.type === "text")?.id ===
|
|
948
|
+
boundElement.id) {
|
|
949
|
+
if (boundElement.containerId !== bindableElement.id) {
|
|
950
|
+
// rebind if not bound already!
|
|
951
|
+
updateElementWith(boundElement, {
|
|
952
|
+
containerId: bindableElement.id,
|
|
953
|
+
});
|
|
954
|
+
}
|
|
955
|
+
}
|
|
956
|
+
else {
|
|
957
|
+
if (boundElement.containerId !== null) {
|
|
958
|
+
// unbind if not unbound already
|
|
959
|
+
updateElementWith(boundElement, {
|
|
960
|
+
containerId: null,
|
|
961
|
+
});
|
|
962
|
+
}
|
|
963
|
+
// unbind from boundElements as the element got bound to some other element in the meantime
|
|
964
|
+
updateElementWith(bindableElement, {
|
|
965
|
+
boundElements: newBoundElements(bindableElement.boundElements, new Set([boundElement.id])),
|
|
966
|
+
});
|
|
967
|
+
}
|
|
968
|
+
}
|
|
969
|
+
});
|
|
970
|
+
};
|
|
971
|
+
}
|