mirror of
https://github.com/excalidraw/excalidraw.git
synced 2025-05-03 10:00:07 -04:00
Merge branch 'excalidraw:master' into master
This commit is contained in:
commit
a4c185e475
141 changed files with 6721 additions and 4149 deletions
|
@ -17,8 +17,6 @@ VITE_APP_FIREBASE_CONFIG='{"apiKey":"AIzaSyCMkxA60XIW8KbqMYL7edC4qT5l4qHX2h8","a
|
||||||
# put these in your .env.local, or make sure you don't commit!
|
# put these in your .env.local, or make sure you don't commit!
|
||||||
# must be lowercase `true` when turned on
|
# must be lowercase `true` when turned on
|
||||||
#
|
#
|
||||||
# whether to enable Service Workers in development
|
|
||||||
VITE_APP_DEV_ENABLE_SW=
|
|
||||||
# whether to disable live reload / HMR. Usuaully what you want to do when
|
# whether to disable live reload / HMR. Usuaully what you want to do when
|
||||||
# debugging Service Workers.
|
# debugging Service Workers.
|
||||||
VITE_APP_DEV_DISABLE_LIVE_RELOAD=
|
VITE_APP_DEV_DISABLE_LIVE_RELOAD=
|
||||||
|
|
|
@ -8,3 +8,4 @@ public/workbox
|
||||||
packages/excalidraw/types
|
packages/excalidraw/types
|
||||||
examples/**/public
|
examples/**/public
|
||||||
dev-dist
|
dev-dist
|
||||||
|
coverage
|
||||||
|
|
|
@ -1547,7 +1547,7 @@
|
||||||
"@docusaurus/theme-search-algolia" "2.2.0"
|
"@docusaurus/theme-search-algolia" "2.2.0"
|
||||||
"@docusaurus/types" "2.2.0"
|
"@docusaurus/types" "2.2.0"
|
||||||
|
|
||||||
"@docusaurus/react-loadable@5.5.2", "react-loadable@npm:@docusaurus/react-loadable@5.5.2":
|
"@docusaurus/react-loadable@5.5.2":
|
||||||
version "5.5.2"
|
version "5.5.2"
|
||||||
resolved "https://registry.yarnpkg.com/@docusaurus/react-loadable/-/react-loadable-5.5.2.tgz#81aae0db81ecafbdaee3651f12804580868fa6ce"
|
resolved "https://registry.yarnpkg.com/@docusaurus/react-loadable/-/react-loadable-5.5.2.tgz#81aae0db81ecafbdaee3651f12804580868fa6ce"
|
||||||
integrity sha512-A3dYjdBGuy0IGT+wyLIGIKLRE+sAk1iNk0f1HjNDysO7u8lhL4N3VEm+FAubmJbAztn94F7MxBTPmnixbiyFdQ==
|
integrity sha512-A3dYjdBGuy0IGT+wyLIGIKLRE+sAk1iNk0f1HjNDysO7u8lhL4N3VEm+FAubmJbAztn94F7MxBTPmnixbiyFdQ==
|
||||||
|
@ -2789,7 +2789,14 @@ brace-expansion@^1.1.7:
|
||||||
balanced-match "^1.0.0"
|
balanced-match "^1.0.0"
|
||||||
concat-map "0.0.1"
|
concat-map "0.0.1"
|
||||||
|
|
||||||
braces@^3.0.2, braces@~3.0.2:
|
braces@^3.0.3:
|
||||||
|
version "3.0.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.3.tgz#490332f40919452272d55a8480adc0c441358789"
|
||||||
|
integrity sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==
|
||||||
|
dependencies:
|
||||||
|
fill-range "^7.1.1"
|
||||||
|
|
||||||
|
braces@~3.0.2:
|
||||||
version "3.0.2"
|
version "3.0.2"
|
||||||
resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107"
|
resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107"
|
||||||
integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==
|
integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==
|
||||||
|
@ -4011,6 +4018,13 @@ fill-range@^7.0.1:
|
||||||
dependencies:
|
dependencies:
|
||||||
to-regex-range "^5.0.1"
|
to-regex-range "^5.0.1"
|
||||||
|
|
||||||
|
fill-range@^7.1.1:
|
||||||
|
version "7.1.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.1.1.tgz#44265d3cac07e3ea7dc247516380643754a05292"
|
||||||
|
integrity sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==
|
||||||
|
dependencies:
|
||||||
|
to-regex-range "^5.0.1"
|
||||||
|
|
||||||
finalhandler@1.2.0:
|
finalhandler@1.2.0:
|
||||||
version "1.2.0"
|
version "1.2.0"
|
||||||
resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.2.0.tgz#7d23fe5731b207b4640e4fcd00aec1f9207a7b32"
|
resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.2.0.tgz#7d23fe5731b207b4640e4fcd00aec1f9207a7b32"
|
||||||
|
@ -5207,11 +5221,11 @@ methods@~1.1.2:
|
||||||
integrity sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==
|
integrity sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==
|
||||||
|
|
||||||
micromatch@^4.0.2, micromatch@^4.0.4, micromatch@^4.0.5:
|
micromatch@^4.0.2, micromatch@^4.0.4, micromatch@^4.0.5:
|
||||||
version "4.0.5"
|
version "4.0.8"
|
||||||
resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.5.tgz#bc8999a7cbbf77cdc89f132f6e467051b49090c6"
|
resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.8.tgz#d66fa18f3a47076789320b9b1af32bd86d9fa202"
|
||||||
integrity sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==
|
integrity sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==
|
||||||
dependencies:
|
dependencies:
|
||||||
braces "^3.0.2"
|
braces "^3.0.3"
|
||||||
picomatch "^2.3.1"
|
picomatch "^2.3.1"
|
||||||
|
|
||||||
mime-db@1.52.0, "mime-db@>= 1.43.0 < 2":
|
mime-db@1.52.0, "mime-db@>= 1.43.0 < 2":
|
||||||
|
@ -6190,14 +6204,13 @@ react-dev-utils@^12.0.1:
|
||||||
strip-ansi "^6.0.1"
|
strip-ansi "^6.0.1"
|
||||||
text-table "^0.2.0"
|
text-table "^0.2.0"
|
||||||
|
|
||||||
react-dom@^17.0.2:
|
react-dom@18.2.0:
|
||||||
version "17.0.2"
|
version "18.2.0"
|
||||||
resolved "https://registry.yarnpkg.com/react-dom/-/react-dom-17.0.2.tgz#ecffb6845e3ad8dbfcdc498f0d0a939736502c23"
|
resolved "https://registry.yarnpkg.com/react-dom/-/react-dom-18.2.0.tgz#22aaf38708db2674ed9ada224ca4aa708d821e3d"
|
||||||
integrity sha512-s4h96KtLDUQlsENhMn1ar8t2bEa+q/YAtj8pPPdIjPDGBDIVNsrD9aXNWqspUe6AzKCIG0C1HZZLqLV7qpOBGA==
|
integrity sha512-6IMTriUmvsjHUjNtEDudZfuDQUoWXVxKHhlEGSk81n4YFS+r/Kl99wXiwlVXtPBtJenozv2P+hxDsw9eA7Xo6g==
|
||||||
dependencies:
|
dependencies:
|
||||||
loose-envify "^1.1.0"
|
loose-envify "^1.1.0"
|
||||||
object-assign "^4.1.1"
|
scheduler "^0.23.0"
|
||||||
scheduler "^0.20.2"
|
|
||||||
|
|
||||||
react-error-overlay@^6.0.11:
|
react-error-overlay@^6.0.11:
|
||||||
version "6.0.11"
|
version "6.0.11"
|
||||||
|
@ -6260,6 +6273,14 @@ react-loadable-ssr-addon-v5-slorber@^1.0.1:
|
||||||
dependencies:
|
dependencies:
|
||||||
"@babel/runtime" "^7.10.3"
|
"@babel/runtime" "^7.10.3"
|
||||||
|
|
||||||
|
"react-loadable@npm:@docusaurus/react-loadable@5.5.2":
|
||||||
|
version "5.5.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/@docusaurus/react-loadable/-/react-loadable-5.5.2.tgz#81aae0db81ecafbdaee3651f12804580868fa6ce"
|
||||||
|
integrity sha512-A3dYjdBGuy0IGT+wyLIGIKLRE+sAk1iNk0f1HjNDysO7u8lhL4N3VEm+FAubmJbAztn94F7MxBTPmnixbiyFdQ==
|
||||||
|
dependencies:
|
||||||
|
"@types/react" "*"
|
||||||
|
prop-types "^15.6.2"
|
||||||
|
|
||||||
react-router-config@^5.1.1:
|
react-router-config@^5.1.1:
|
||||||
version "5.1.1"
|
version "5.1.1"
|
||||||
resolved "https://registry.yarnpkg.com/react-router-config/-/react-router-config-5.1.1.tgz#0f4263d1a80c6b2dc7b9c1902c9526478194a988"
|
resolved "https://registry.yarnpkg.com/react-router-config/-/react-router-config-5.1.1.tgz#0f4263d1a80c6b2dc7b9c1902c9526478194a988"
|
||||||
|
@ -6310,13 +6331,12 @@ react-textarea-autosize@^8.3.2:
|
||||||
use-composed-ref "^1.3.0"
|
use-composed-ref "^1.3.0"
|
||||||
use-latest "^1.2.1"
|
use-latest "^1.2.1"
|
||||||
|
|
||||||
react@^17.0.2:
|
react@18.2.0:
|
||||||
version "17.0.2"
|
version "18.2.0"
|
||||||
resolved "https://registry.yarnpkg.com/react/-/react-17.0.2.tgz#d0b5cc516d29eb3eee383f75b62864cfb6800037"
|
resolved "https://registry.yarnpkg.com/react/-/react-18.2.0.tgz#555bd98592883255fa00de14f1151a917b5d77d5"
|
||||||
integrity sha512-gnhPt75i/dq/z3/6q/0asP78D0u592D5L1pd7M8P+dck6Fu/jJeL6iVVK23fptSUZj8Vjf++7wXA8UNclGQcbA==
|
integrity sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ==
|
||||||
dependencies:
|
dependencies:
|
||||||
loose-envify "^1.1.0"
|
loose-envify "^1.1.0"
|
||||||
object-assign "^4.1.1"
|
|
||||||
|
|
||||||
readable-stream@^2.0.1:
|
readable-stream@^2.0.1:
|
||||||
version "2.3.7"
|
version "2.3.7"
|
||||||
|
@ -6664,13 +6684,12 @@ sax@^1.2.4:
|
||||||
resolved "https://registry.yarnpkg.com/sax/-/sax-1.2.4.tgz#2816234e2378bddc4e5354fab5caa895df7100d9"
|
resolved "https://registry.yarnpkg.com/sax/-/sax-1.2.4.tgz#2816234e2378bddc4e5354fab5caa895df7100d9"
|
||||||
integrity sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==
|
integrity sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==
|
||||||
|
|
||||||
scheduler@^0.20.2:
|
scheduler@^0.23.0:
|
||||||
version "0.20.2"
|
version "0.23.2"
|
||||||
resolved "https://registry.yarnpkg.com/scheduler/-/scheduler-0.20.2.tgz#4baee39436e34aa93b4874bddcbf0fe8b8b50e91"
|
resolved "https://registry.yarnpkg.com/scheduler/-/scheduler-0.23.2.tgz#414ba64a3b282892e944cf2108ecc078d115cdc3"
|
||||||
integrity sha512-2eWfGgAqqWFGqtdMmcL5zCMK1U8KlXv8SQFGglL3CEtd0aDVDWgeF/YoCmvln55m5zSk3J/20hTaSBeSObsQDQ==
|
integrity sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==
|
||||||
dependencies:
|
dependencies:
|
||||||
loose-envify "^1.1.0"
|
loose-envify "^1.1.0"
|
||||||
object-assign "^4.1.1"
|
|
||||||
|
|
||||||
schema-utils@2.7.0:
|
schema-utils@2.7.0:
|
||||||
version "2.7.0"
|
version "2.7.0"
|
||||||
|
|
|
@ -31,6 +31,7 @@ export const AppMainMenu: React.FC<{
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<MainMenu.DefaultItems.CommandPalette className="highlighted" />
|
<MainMenu.DefaultItems.CommandPalette className="highlighted" />
|
||||||
|
<MainMenu.DefaultItems.SearchMenu />
|
||||||
<MainMenu.DefaultItems.Help />
|
<MainMenu.DefaultItems.Help />
|
||||||
<MainMenu.DefaultItems.ClearCanvas />
|
<MainMenu.DefaultItems.ClearCanvas />
|
||||||
<MainMenu.Separator />
|
<MainMenu.Separator />
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
import { forwardRef, useCallback, useImperativeHandle, useRef } from "react";
|
import { forwardRef, useCallback, useImperativeHandle, useRef } from "react";
|
||||||
import { type AppState } from "../../packages/excalidraw/types";
|
import { type AppState } from "../../packages/excalidraw/types";
|
||||||
import { throttleRAF } from "../../packages/excalidraw/utils";
|
import { throttleRAF } from "../../packages/excalidraw/utils";
|
||||||
import type { LineSegment } from "../../packages/utils";
|
|
||||||
import {
|
import {
|
||||||
bootstrapCanvas,
|
bootstrapCanvas,
|
||||||
getNormalizedCanvasDimensions,
|
getNormalizedCanvasDimensions,
|
||||||
|
@ -13,12 +12,16 @@ import {
|
||||||
TrashIcon,
|
TrashIcon,
|
||||||
} from "../../packages/excalidraw/components/icons";
|
} from "../../packages/excalidraw/components/icons";
|
||||||
import { STORAGE_KEYS } from "../app_constants";
|
import { STORAGE_KEYS } from "../app_constants";
|
||||||
import { isLineSegment } from "../../packages/excalidraw/element/typeChecks";
|
import {
|
||||||
|
isLineSegment,
|
||||||
|
type GlobalPoint,
|
||||||
|
type LineSegment,
|
||||||
|
} from "../../packages/math";
|
||||||
|
|
||||||
const renderLine = (
|
const renderLine = (
|
||||||
context: CanvasRenderingContext2D,
|
context: CanvasRenderingContext2D,
|
||||||
zoom: number,
|
zoom: number,
|
||||||
segment: LineSegment,
|
segment: LineSegment<GlobalPoint>,
|
||||||
color: string,
|
color: string,
|
||||||
) => {
|
) => {
|
||||||
context.save();
|
context.save();
|
||||||
|
@ -47,10 +50,15 @@ const render = (
|
||||||
context: CanvasRenderingContext2D,
|
context: CanvasRenderingContext2D,
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
) => {
|
) => {
|
||||||
frame.forEach((el) => {
|
frame.forEach((el: DebugElement) => {
|
||||||
switch (true) {
|
switch (true) {
|
||||||
case isLineSegment(el.data):
|
case isLineSegment(el.data):
|
||||||
renderLine(context, appState.zoom.value, el.data, el.color);
|
renderLine(
|
||||||
|
context,
|
||||||
|
appState.zoom.value,
|
||||||
|
el.data as LineSegment<GlobalPoint>,
|
||||||
|
el.color,
|
||||||
|
);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -20,6 +20,7 @@ import {
|
||||||
get,
|
get,
|
||||||
} from "idb-keyval";
|
} from "idb-keyval";
|
||||||
import { clearAppStateForLocalStorage } from "../../packages/excalidraw/appState";
|
import { clearAppStateForLocalStorage } from "../../packages/excalidraw/appState";
|
||||||
|
import { SEARCH_SIDEBAR } from "../../packages/excalidraw/constants";
|
||||||
import type { LibraryPersistedData } from "../../packages/excalidraw/data/library";
|
import type { LibraryPersistedData } from "../../packages/excalidraw/data/library";
|
||||||
import type { ImportedDataState } from "../../packages/excalidraw/data/types";
|
import type { ImportedDataState } from "../../packages/excalidraw/data/types";
|
||||||
import { clearElementsForLocalStorage } from "../../packages/excalidraw/element";
|
import { clearElementsForLocalStorage } from "../../packages/excalidraw/element";
|
||||||
|
@ -66,13 +67,19 @@ const saveDataStateToLocalStorage = (
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
|
const _appState = clearAppStateForLocalStorage(appState);
|
||||||
|
|
||||||
|
if (_appState.openSidebar?.name === SEARCH_SIDEBAR.name) {
|
||||||
|
_appState.openSidebar = null;
|
||||||
|
}
|
||||||
|
|
||||||
localStorage.setItem(
|
localStorage.setItem(
|
||||||
STORAGE_KEYS.LOCAL_STORAGE_ELEMENTS,
|
STORAGE_KEYS.LOCAL_STORAGE_ELEMENTS,
|
||||||
JSON.stringify(clearElementsForLocalStorage(elements)),
|
JSON.stringify(clearElementsForLocalStorage(elements)),
|
||||||
);
|
);
|
||||||
localStorage.setItem(
|
localStorage.setItem(
|
||||||
STORAGE_KEYS.LOCAL_STORAGE_APP_STATE,
|
STORAGE_KEYS.LOCAL_STORAGE_APP_STATE,
|
||||||
JSON.stringify(clearAppStateForLocalStorage(appState)),
|
JSON.stringify(_appState),
|
||||||
);
|
);
|
||||||
updateBrowserStateVersion(STORAGE_KEYS.VERSION_DATA_STATE);
|
updateBrowserStateVersion(STORAGE_KEYS.VERSION_DATA_STATE);
|
||||||
} catch (error: any) {
|
} catch (error: any) {
|
||||||
|
|
18
package.json
18
package.json
|
@ -6,6 +6,7 @@
|
||||||
"excalidraw-app",
|
"excalidraw-app",
|
||||||
"packages/excalidraw",
|
"packages/excalidraw",
|
||||||
"packages/utils",
|
"packages/utils",
|
||||||
|
"packages/math",
|
||||||
"examples/excalidraw",
|
"examples/excalidraw",
|
||||||
"examples/excalidraw/*"
|
"examples/excalidraw/*"
|
||||||
],
|
],
|
||||||
|
@ -20,8 +21,8 @@
|
||||||
"@types/react-dom": "18.2.0",
|
"@types/react-dom": "18.2.0",
|
||||||
"@types/socket.io-client": "3.0.0",
|
"@types/socket.io-client": "3.0.0",
|
||||||
"@vitejs/plugin-react": "3.1.0",
|
"@vitejs/plugin-react": "3.1.0",
|
||||||
"@vitest/coverage-v8": "0.33.0",
|
"@vitest/coverage-v8": "2.0.5",
|
||||||
"@vitest/ui": "0.32.2",
|
"@vitest/ui": "2.0.5",
|
||||||
"chai": "4.3.6",
|
"chai": "4.3.6",
|
||||||
"dotenv": "16.0.1",
|
"dotenv": "16.0.1",
|
||||||
"eslint-config-prettier": "8.5.0",
|
"eslint-config-prettier": "8.5.0",
|
||||||
|
@ -35,13 +36,13 @@
|
||||||
"prettier": "2.6.2",
|
"prettier": "2.6.2",
|
||||||
"rewire": "6.0.0",
|
"rewire": "6.0.0",
|
||||||
"typescript": "4.9.4",
|
"typescript": "4.9.4",
|
||||||
"vite": "5.0.12",
|
"vite": "5.4.2",
|
||||||
"vite-plugin-checker": "0.6.1",
|
"vite-plugin-checker": "0.7.2",
|
||||||
"vite-plugin-ejs": "1.7.0",
|
"vite-plugin-ejs": "1.7.0",
|
||||||
"vite-plugin-pwa": "0.17.4",
|
"vite-plugin-pwa": "0.17.4",
|
||||||
"vite-plugin-svgr": "2.4.0",
|
"vite-plugin-svgr": "4.2.0",
|
||||||
"vitest": "1.6.0",
|
"vitest": "2.0.5",
|
||||||
"vitest-canvas-mock": "0.3.2"
|
"vitest-canvas-mock": "0.3.3"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": "18.0.0 - 20.x.x"
|
"node": "18.0.0 - 20.x.x"
|
||||||
|
@ -82,6 +83,7 @@
|
||||||
"clean-install": "yarn rm:node_modules && yarn install"
|
"clean-install": "yarn rm:node_modules && yarn install"
|
||||||
},
|
},
|
||||||
"resolutions": {
|
"resolutions": {
|
||||||
"@types/react": "18.2.0"
|
"@types/react": "18.2.0",
|
||||||
|
"strip-ansi": "6.0.1"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -38,7 +38,7 @@ import { DEFAULT_CANVAS_BACKGROUND_PICKS } from "../colors";
|
||||||
import type { SceneBounds } from "../element/bounds";
|
import type { SceneBounds } from "../element/bounds";
|
||||||
import { setCursor } from "../cursor";
|
import { setCursor } from "../cursor";
|
||||||
import { StoreAction } from "../store";
|
import { StoreAction } from "../store";
|
||||||
import { clamp } from "../math";
|
import { clamp } from "../../math";
|
||||||
|
|
||||||
export const actionChangeViewBackgroundColor = register({
|
export const actionChangeViewBackgroundColor = register({
|
||||||
name: "changeViewBackgroundColor",
|
name: "changeViewBackgroundColor",
|
||||||
|
|
|
@ -42,20 +42,21 @@ export const actionDuplicateSelection = register({
|
||||||
perform: (elements, appState, formData, app) => {
|
perform: (elements, appState, formData, app) => {
|
||||||
// duplicate selected point(s) if editing a line
|
// duplicate selected point(s) if editing a line
|
||||||
if (appState.editingLinearElement) {
|
if (appState.editingLinearElement) {
|
||||||
const ret = LinearElementEditor.duplicateSelectedPoints(
|
// TODO: Invariants should be checked here instead of duplicateSelectedPoints()
|
||||||
|
try {
|
||||||
|
const newAppState = LinearElementEditor.duplicateSelectedPoints(
|
||||||
appState,
|
appState,
|
||||||
app.scene.getNonDeletedElementsMap(),
|
app.scene.getNonDeletedElementsMap(),
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!ret) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
elements,
|
elements,
|
||||||
appState: ret.appState,
|
appState: newAppState,
|
||||||
storeAction: StoreAction.CAPTURE,
|
storeAction: StoreAction.CAPTURE,
|
||||||
};
|
};
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|
|
@ -6,7 +6,6 @@ import { done } from "../components/icons";
|
||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
import { register } from "./register";
|
import { register } from "./register";
|
||||||
import { mutateElement } from "../element/mutateElement";
|
import { mutateElement } from "../element/mutateElement";
|
||||||
import { isPathALoop } from "../math";
|
|
||||||
import { LinearElementEditor } from "../element/linearElementEditor";
|
import { LinearElementEditor } from "../element/linearElementEditor";
|
||||||
import {
|
import {
|
||||||
maybeBindLinearElement,
|
maybeBindLinearElement,
|
||||||
|
@ -16,6 +15,8 @@ import { isBindingElement, isLinearElement } from "../element/typeChecks";
|
||||||
import type { AppState } from "../types";
|
import type { AppState } from "../types";
|
||||||
import { resetCursor } from "../cursor";
|
import { resetCursor } from "../cursor";
|
||||||
import { StoreAction } from "../store";
|
import { StoreAction } from "../store";
|
||||||
|
import { point } from "../../math";
|
||||||
|
import { isPathALoop } from "../shapes";
|
||||||
|
|
||||||
export const actionFinalize = register({
|
export const actionFinalize = register({
|
||||||
name: "finalize",
|
name: "finalize",
|
||||||
|
@ -112,10 +113,10 @@ export const actionFinalize = register({
|
||||||
const linePoints = multiPointElement.points;
|
const linePoints = multiPointElement.points;
|
||||||
const firstPoint = linePoints[0];
|
const firstPoint = linePoints[0];
|
||||||
mutateElement(multiPointElement, {
|
mutateElement(multiPointElement, {
|
||||||
points: linePoints.map((point, index) =>
|
points: linePoints.map((p, index) =>
|
||||||
index === linePoints.length - 1
|
index === linePoints.length - 1
|
||||||
? ([firstPoint[0], firstPoint[1]] as const)
|
? point(firstPoint[0], firstPoint[1])
|
||||||
: point,
|
: p,
|
||||||
),
|
),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { useEffect, useMemo, useRef, useState } from "react";
|
import { useEffect, useMemo, useRef, useState } from "react";
|
||||||
import type { AppClassProperties, AppState, Point, Primitive } from "../types";
|
import type { AppClassProperties, AppState, Primitive } from "../types";
|
||||||
import type { StoreActionType } from "../store";
|
import type { StoreActionType } from "../store";
|
||||||
import {
|
import {
|
||||||
DEFAULT_ELEMENT_BACKGROUND_COLOR_PALETTE,
|
DEFAULT_ELEMENT_BACKGROUND_COLOR_PALETTE,
|
||||||
|
@ -115,6 +115,8 @@ import {
|
||||||
} from "../element/binding";
|
} from "../element/binding";
|
||||||
import { mutateElbowArrow } from "../element/routing";
|
import { mutateElbowArrow } from "../element/routing";
|
||||||
import { LinearElementEditor } from "../element/linearElementEditor";
|
import { LinearElementEditor } from "../element/linearElementEditor";
|
||||||
|
import type { LocalPoint } from "../../math";
|
||||||
|
import { point, vector } from "../../math";
|
||||||
|
|
||||||
const FONT_SIZE_RELATIVE_INCREASE_STEP = 0.1;
|
const FONT_SIZE_RELATIVE_INCREASE_STEP = 0.1;
|
||||||
|
|
||||||
|
@ -1648,10 +1650,10 @@ export const actionChangeArrowType = register({
|
||||||
newElement,
|
newElement,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
[finalStartPoint, finalEndPoint].map(
|
[finalStartPoint, finalEndPoint].map(
|
||||||
(point) =>
|
(p): LocalPoint =>
|
||||||
[point[0] - newElement.x, point[1] - newElement.y] as Point,
|
point(p[0] - newElement.x, p[1] - newElement.y),
|
||||||
),
|
),
|
||||||
[0, 0],
|
vector(0, 0),
|
||||||
{
|
{
|
||||||
...(startElement && newElement.startBinding
|
...(startElement && newElement.startBinding
|
||||||
? {
|
? {
|
||||||
|
|
51
packages/excalidraw/actions/actionToggleSearchMenu.ts
Normal file
51
packages/excalidraw/actions/actionToggleSearchMenu.ts
Normal file
|
@ -0,0 +1,51 @@
|
||||||
|
import { KEYS } from "../keys";
|
||||||
|
import { register } from "./register";
|
||||||
|
import type { AppState } from "../types";
|
||||||
|
import { searchIcon } from "../components/icons";
|
||||||
|
import { StoreAction } from "../store";
|
||||||
|
import { CLASSES, SEARCH_SIDEBAR } from "../constants";
|
||||||
|
|
||||||
|
export const actionToggleSearchMenu = register({
|
||||||
|
name: "searchMenu",
|
||||||
|
icon: searchIcon,
|
||||||
|
keywords: ["search", "find"],
|
||||||
|
label: "search.title",
|
||||||
|
viewMode: true,
|
||||||
|
trackEvent: {
|
||||||
|
category: "search_menu",
|
||||||
|
action: "toggle",
|
||||||
|
predicate: (appState) => appState.gridModeEnabled,
|
||||||
|
},
|
||||||
|
perform(elements, appState, _, app) {
|
||||||
|
if (appState.openSidebar?.name === SEARCH_SIDEBAR.name) {
|
||||||
|
const searchInput =
|
||||||
|
app.excalidrawContainerValue.container?.querySelector<HTMLInputElement>(
|
||||||
|
`.${CLASSES.SEARCH_MENU_INPUT_WRAPPER} input`,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (searchInput?.matches(":focus")) {
|
||||||
|
return {
|
||||||
|
appState: { ...appState, openSidebar: null },
|
||||||
|
storeAction: StoreAction.NONE,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
searchInput?.focus();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
appState: {
|
||||||
|
...appState,
|
||||||
|
openSidebar: { name: SEARCH_SIDEBAR.name },
|
||||||
|
openDialog: null,
|
||||||
|
},
|
||||||
|
storeAction: StoreAction.NONE,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
checked: (appState: AppState) => appState.gridModeEnabled,
|
||||||
|
predicate: (element, appState, props) => {
|
||||||
|
return props.gridModeEnabled === undefined;
|
||||||
|
},
|
||||||
|
keyTest: (event) => event[KEYS.CTRL_OR_CMD] && event.key === KEYS.F,
|
||||||
|
});
|
|
@ -86,3 +86,5 @@ export { actionUnbindText, actionBindText } from "./actionBoundText";
|
||||||
export { actionLink } from "./actionLink";
|
export { actionLink } from "./actionLink";
|
||||||
export { actionToggleElementLock } from "./actionElementLock";
|
export { actionToggleElementLock } from "./actionElementLock";
|
||||||
export { actionToggleLinearEditor } from "./actionLinearEditor";
|
export { actionToggleLinearEditor } from "./actionLinearEditor";
|
||||||
|
|
||||||
|
export { actionToggleSearchMenu } from "./actionToggleSearchMenu";
|
||||||
|
|
|
@ -51,7 +51,8 @@ export type ShortcutName =
|
||||||
>
|
>
|
||||||
| "saveScene"
|
| "saveScene"
|
||||||
| "imageExport"
|
| "imageExport"
|
||||||
| "commandPalette";
|
| "commandPalette"
|
||||||
|
| "searchMenu";
|
||||||
|
|
||||||
const shortcutMap: Record<ShortcutName, string[]> = {
|
const shortcutMap: Record<ShortcutName, string[]> = {
|
||||||
toggleTheme: [getShortcutKey("Shift+Alt+D")],
|
toggleTheme: [getShortcutKey("Shift+Alt+D")],
|
||||||
|
@ -112,6 +113,7 @@ const shortcutMap: Record<ShortcutName, string[]> = {
|
||||||
saveFileToDisk: [getShortcutKey("CtrlOrCmd+S")],
|
saveFileToDisk: [getShortcutKey("CtrlOrCmd+S")],
|
||||||
saveToActiveFile: [getShortcutKey("CtrlOrCmd+S")],
|
saveToActiveFile: [getShortcutKey("CtrlOrCmd+S")],
|
||||||
toggleShortcuts: [getShortcutKey("?")],
|
toggleShortcuts: [getShortcutKey("?")],
|
||||||
|
searchMenu: [getShortcutKey("CtrlOrCmd+F")],
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getShortcutFromShortcutName = (name: ShortcutName, idx = 0) => {
|
export const getShortcutFromShortcutName = (name: ShortcutName, idx = 0) => {
|
||||||
|
|
|
@ -137,7 +137,8 @@ export type ActionName =
|
||||||
| "wrapTextInContainer"
|
| "wrapTextInContainer"
|
||||||
| "commandPalette"
|
| "commandPalette"
|
||||||
| "autoResize"
|
| "autoResize"
|
||||||
| "elementStats";
|
| "elementStats"
|
||||||
|
| "searchMenu";
|
||||||
|
|
||||||
export type PanelComponentProps = {
|
export type PanelComponentProps = {
|
||||||
elements: readonly ExcalidrawElement[];
|
elements: readonly ExcalidrawElement[];
|
||||||
|
@ -191,7 +192,8 @@ export interface Action {
|
||||||
| "history"
|
| "history"
|
||||||
| "menu"
|
| "menu"
|
||||||
| "collab"
|
| "collab"
|
||||||
| "hyperlink";
|
| "hyperlink"
|
||||||
|
| "search_menu";
|
||||||
action?: string;
|
action?: string;
|
||||||
predicate?: (
|
predicate?: (
|
||||||
appState: Readonly<AppState>,
|
appState: Readonly<AppState>,
|
||||||
|
|
|
@ -116,6 +116,7 @@ export const getDefaultAppState = (): Omit<
|
||||||
objectsSnapModeEnabled: false,
|
objectsSnapModeEnabled: false,
|
||||||
userToFollow: null,
|
userToFollow: null,
|
||||||
followedBy: new Set(),
|
followedBy: new Set(),
|
||||||
|
searchMatches: [],
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -236,6 +237,7 @@ const APP_STATE_STORAGE_CONF = (<
|
||||||
objectsSnapModeEnabled: { browser: true, export: false, server: false },
|
objectsSnapModeEnabled: { browser: true, export: false, server: false },
|
||||||
userToFollow: { browser: false, export: false, server: false },
|
userToFollow: { browser: false, export: false, server: false },
|
||||||
followedBy: { browser: false, export: false, server: false },
|
followedBy: { browser: false, export: false, server: false },
|
||||||
|
searchMatches: { browser: false, export: false, server: false },
|
||||||
});
|
});
|
||||||
|
|
||||||
const _clearAppStateForStorage = <
|
const _clearAppStateForStorage = <
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
import type { Radians } from "../math";
|
||||||
|
import { point } from "../math";
|
||||||
import {
|
import {
|
||||||
COLOR_PALETTE,
|
COLOR_PALETTE,
|
||||||
DEFAULT_CHART_COLOR_INDEX,
|
DEFAULT_CHART_COLOR_INDEX,
|
||||||
|
@ -203,7 +205,7 @@ const chartXLabels = (
|
||||||
x: x + index * (BAR_WIDTH + BAR_GAP) + BAR_GAP * 2,
|
x: x + index * (BAR_WIDTH + BAR_GAP) + BAR_GAP * 2,
|
||||||
y: y + BAR_GAP / 2,
|
y: y + BAR_GAP / 2,
|
||||||
width: BAR_WIDTH,
|
width: BAR_WIDTH,
|
||||||
angle: 5.87,
|
angle: 5.87 as Radians,
|
||||||
fontSize: 16,
|
fontSize: 16,
|
||||||
textAlign: "center",
|
textAlign: "center",
|
||||||
verticalAlign: "top",
|
verticalAlign: "top",
|
||||||
|
@ -258,10 +260,7 @@ const chartLines = (
|
||||||
x,
|
x,
|
||||||
y,
|
y,
|
||||||
width: chartWidth,
|
width: chartWidth,
|
||||||
points: [
|
points: [point(0, 0), point(chartWidth, 0)],
|
||||||
[0, 0],
|
|
||||||
[chartWidth, 0],
|
|
||||||
],
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const yLine = newLinearElement({
|
const yLine = newLinearElement({
|
||||||
|
@ -272,10 +271,7 @@ const chartLines = (
|
||||||
x,
|
x,
|
||||||
y,
|
y,
|
||||||
height: chartHeight,
|
height: chartHeight,
|
||||||
points: [
|
points: [point(0, 0), point(0, -chartHeight)],
|
||||||
[0, 0],
|
|
||||||
[0, -chartHeight],
|
|
||||||
],
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const maxLine = newLinearElement({
|
const maxLine = newLinearElement({
|
||||||
|
@ -288,10 +284,7 @@ const chartLines = (
|
||||||
strokeStyle: "dotted",
|
strokeStyle: "dotted",
|
||||||
width: chartWidth,
|
width: chartWidth,
|
||||||
opacity: GRID_OPACITY,
|
opacity: GRID_OPACITY,
|
||||||
points: [
|
points: [point(0, 0), point(chartWidth, 0)],
|
||||||
[0, 0],
|
|
||||||
[chartWidth, 0],
|
|
||||||
],
|
|
||||||
});
|
});
|
||||||
|
|
||||||
return [xLine, yLine, maxLine];
|
return [xLine, yLine, maxLine];
|
||||||
|
@ -448,10 +441,7 @@ const chartTypeLine = (
|
||||||
height: cy,
|
height: cy,
|
||||||
strokeStyle: "dotted",
|
strokeStyle: "dotted",
|
||||||
opacity: GRID_OPACITY,
|
opacity: GRID_OPACITY,
|
||||||
points: [
|
points: [point(0, 0), point(0, cy)],
|
||||||
[0, 0],
|
|
||||||
[0, cy],
|
|
||||||
],
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -210,12 +210,6 @@ import {
|
||||||
isElementCompletelyInViewport,
|
isElementCompletelyInViewport,
|
||||||
isElementInViewport,
|
isElementInViewport,
|
||||||
} from "../element/sizeHelpers";
|
} from "../element/sizeHelpers";
|
||||||
import {
|
|
||||||
distance2d,
|
|
||||||
getCornerRadius,
|
|
||||||
getGridPoint,
|
|
||||||
isPathALoop,
|
|
||||||
} from "../math";
|
|
||||||
import {
|
import {
|
||||||
calculateScrollCenter,
|
calculateScrollCenter,
|
||||||
getElementsWithinSelection,
|
getElementsWithinSelection,
|
||||||
|
@ -230,7 +224,13 @@ import type {
|
||||||
ScrollBars,
|
ScrollBars,
|
||||||
} from "../scene/types";
|
} from "../scene/types";
|
||||||
import { getStateForZoom } from "../scene/zoom";
|
import { getStateForZoom } from "../scene/zoom";
|
||||||
import { findShapeByKey, getBoundTextShape, getElementShape } from "../shapes";
|
import {
|
||||||
|
findShapeByKey,
|
||||||
|
getBoundTextShape,
|
||||||
|
getCornerRadius,
|
||||||
|
getElementShape,
|
||||||
|
isPathALoop,
|
||||||
|
} from "../shapes";
|
||||||
import { getSelectionBoxShape } from "../../utils/geometry/shape";
|
import { getSelectionBoxShape } from "../../utils/geometry/shape";
|
||||||
import { isPointInShape } from "../../utils/collision";
|
import { isPointInShape } from "../../utils/collision";
|
||||||
import type {
|
import type {
|
||||||
|
@ -386,6 +386,7 @@ import {
|
||||||
getReferenceSnapPoints,
|
getReferenceSnapPoints,
|
||||||
SnapCache,
|
SnapCache,
|
||||||
isGridModeEnabled,
|
isGridModeEnabled,
|
||||||
|
getGridPoint,
|
||||||
} from "../snapping";
|
} from "../snapping";
|
||||||
import { actionWrapTextInContainer } from "../actions/actionBoundText";
|
import { actionWrapTextInContainer } from "../actions/actionBoundText";
|
||||||
import BraveMeasureTextError from "./BraveMeasureTextError";
|
import BraveMeasureTextError from "./BraveMeasureTextError";
|
||||||
|
@ -439,6 +440,9 @@ import {
|
||||||
FlowChartNavigator,
|
FlowChartNavigator,
|
||||||
getLinkDirectionFromKey,
|
getLinkDirectionFromKey,
|
||||||
} from "../element/flowchart";
|
} from "../element/flowchart";
|
||||||
|
import { searchItemInFocusAtom } from "./SearchMenu";
|
||||||
|
import type { LocalPoint, Radians } from "../../math";
|
||||||
|
import { point, pointDistance, vector } from "../../math";
|
||||||
|
|
||||||
const AppContext = React.createContext<AppClassProperties>(null!);
|
const AppContext = React.createContext<AppClassProperties>(null!);
|
||||||
const AppPropsContext = React.createContext<AppProps>(null!);
|
const AppPropsContext = React.createContext<AppProps>(null!);
|
||||||
|
@ -545,6 +549,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
public scene: Scene;
|
public scene: Scene;
|
||||||
public fonts: Fonts;
|
public fonts: Fonts;
|
||||||
public renderer: Renderer;
|
public renderer: Renderer;
|
||||||
|
public visibleElements: readonly NonDeletedExcalidrawElement[];
|
||||||
private resizeObserver: ResizeObserver | undefined;
|
private resizeObserver: ResizeObserver | undefined;
|
||||||
private nearestScrollableContainer: HTMLElement | Document | undefined;
|
private nearestScrollableContainer: HTMLElement | Document | undefined;
|
||||||
public library: AppClassProperties["library"];
|
public library: AppClassProperties["library"];
|
||||||
|
@ -552,7 +557,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
public id: string;
|
public id: string;
|
||||||
private store: Store;
|
private store: Store;
|
||||||
private history: History;
|
private history: History;
|
||||||
private excalidrawContainerValue: {
|
public excalidrawContainerValue: {
|
||||||
container: HTMLDivElement | null;
|
container: HTMLDivElement | null;
|
||||||
id: string;
|
id: string;
|
||||||
};
|
};
|
||||||
|
@ -679,6 +684,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
this.canvas = document.createElement("canvas");
|
this.canvas = document.createElement("canvas");
|
||||||
this.rc = rough.canvas(this.canvas);
|
this.rc = rough.canvas(this.canvas);
|
||||||
this.renderer = new Renderer(this.scene);
|
this.renderer = new Renderer(this.scene);
|
||||||
|
this.visibleElements = [];
|
||||||
|
|
||||||
this.store = new Store();
|
this.store = new Store();
|
||||||
this.history = new History();
|
this.history = new History();
|
||||||
|
@ -1477,6 +1483,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
newElementId: this.state.newElement?.id,
|
newElementId: this.state.newElement?.id,
|
||||||
pendingImageElementId: this.state.pendingImageElementId,
|
pendingImageElementId: this.state.pendingImageElementId,
|
||||||
});
|
});
|
||||||
|
this.visibleElements = visibleElements;
|
||||||
|
|
||||||
const allElementsMap = this.scene.getNonDeletedElementsMap();
|
const allElementsMap = this.scene.getNonDeletedElementsMap();
|
||||||
|
|
||||||
|
@ -2292,6 +2299,9 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
storeAction: StoreAction.UPDATE,
|
storeAction: StoreAction.UPDATE,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// clear the shape and image cache so that any images in initialData
|
||||||
|
// can be loaded fresh
|
||||||
|
this.clearImageShapeCache();
|
||||||
// FontFaceSet loadingdone event we listen on may not always
|
// FontFaceSet loadingdone event we listen on may not always
|
||||||
// fire (looking at you Safari), so on init we manually load all
|
// fire (looking at you Safari), so on init we manually load all
|
||||||
// fonts and rerender scene text elements once done. This also
|
// fonts and rerender scene text elements once done. This also
|
||||||
|
@ -2357,6 +2367,16 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
private clearImageShapeCache(filesMap?: BinaryFiles) {
|
||||||
|
const files = filesMap ?? this.files;
|
||||||
|
this.scene.getNonDeletedElements().forEach((element) => {
|
||||||
|
if (isInitializedImageElement(element) && files[element.fileId]) {
|
||||||
|
this.imageCache.delete(element.fileId);
|
||||||
|
ShapeCache.delete(element);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
public async componentDidMount() {
|
public async componentDidMount() {
|
||||||
this.unmounted = false;
|
this.unmounted = false;
|
||||||
this.excalidrawContainerValue.container =
|
this.excalidrawContainerValue.container =
|
||||||
|
@ -3671,15 +3691,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
|
|
||||||
this.files = { ...this.files, ...Object.fromEntries(filesMap) };
|
this.files = { ...this.files, ...Object.fromEntries(filesMap) };
|
||||||
|
|
||||||
this.scene.getNonDeletedElements().forEach((element) => {
|
this.clearImageShapeCache(Object.fromEntries(filesMap));
|
||||||
if (
|
|
||||||
isInitializedImageElement(element) &&
|
|
||||||
filesMap.has(element.fileId)
|
|
||||||
) {
|
|
||||||
this.imageCache.delete(element.fileId);
|
|
||||||
ShapeCache.delete(element);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
this.scene.triggerUpdate();
|
this.scene.triggerUpdate();
|
||||||
|
|
||||||
this.addNewImagesToImageCache();
|
this.addNewImagesToImageCache();
|
||||||
|
@ -3793,7 +3805,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
private getEditorUIOffsets = (): {
|
public getEditorUIOffsets = (): {
|
||||||
top: number;
|
top: number;
|
||||||
right: number;
|
right: number;
|
||||||
bottom: number;
|
bottom: number;
|
||||||
|
@ -4844,7 +4856,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
this.getElementHitThreshold(),
|
this.getElementHitThreshold(),
|
||||||
);
|
);
|
||||||
|
|
||||||
return isPointInShape([x, y], selectionShape);
|
return isPointInShape(point(x, y), selectionShape);
|
||||||
}
|
}
|
||||||
|
|
||||||
// take bound text element into consideration for hit collision as well
|
// take bound text element into consideration for hit collision as well
|
||||||
|
@ -5035,7 +5047,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
containerId: shouldBindToContainer ? container?.id : undefined,
|
containerId: shouldBindToContainer ? container?.id : undefined,
|
||||||
groupIds: container?.groupIds ?? [],
|
groupIds: container?.groupIds ?? [],
|
||||||
lineHeight,
|
lineHeight,
|
||||||
angle: container?.angle ?? 0,
|
angle: container?.angle ?? (0 as Radians),
|
||||||
frameId: topLayerFrame ? topLayerFrame.id : null,
|
frameId: topLayerFrame ? topLayerFrame.id : null,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -5203,7 +5215,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
element,
|
element,
|
||||||
this.scene.getNonDeletedElementsMap(),
|
this.scene.getNonDeletedElementsMap(),
|
||||||
this.state,
|
this.state,
|
||||||
[scenePointer.x, scenePointer.y],
|
point(scenePointer.x, scenePointer.y),
|
||||||
this.device.editor.isMobile,
|
this.device.editor.isMobile,
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
@ -5214,11 +5226,12 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
event: React.PointerEvent<HTMLCanvasElement>,
|
event: React.PointerEvent<HTMLCanvasElement>,
|
||||||
isTouchScreen: boolean,
|
isTouchScreen: boolean,
|
||||||
) => {
|
) => {
|
||||||
const draggedDistance = distance2d(
|
const draggedDistance = pointDistance(
|
||||||
|
point(
|
||||||
this.lastPointerDownEvent!.clientX,
|
this.lastPointerDownEvent!.clientX,
|
||||||
this.lastPointerDownEvent!.clientY,
|
this.lastPointerDownEvent!.clientY,
|
||||||
this.lastPointerUpEvent!.clientX,
|
),
|
||||||
this.lastPointerUpEvent!.clientY,
|
point(this.lastPointerUpEvent!.clientX, this.lastPointerUpEvent!.clientY),
|
||||||
);
|
);
|
||||||
if (
|
if (
|
||||||
!this.hitLinkElement ||
|
!this.hitLinkElement ||
|
||||||
|
@ -5237,7 +5250,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
this.hitLinkElement,
|
this.hitLinkElement,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
this.state,
|
this.state,
|
||||||
[lastPointerDownCoords.x, lastPointerDownCoords.y],
|
point(lastPointerDownCoords.x, lastPointerDownCoords.y),
|
||||||
this.device.editor.isMobile,
|
this.device.editor.isMobile,
|
||||||
);
|
);
|
||||||
const lastPointerUpCoords = viewportCoordsToSceneCoords(
|
const lastPointerUpCoords = viewportCoordsToSceneCoords(
|
||||||
|
@ -5248,7 +5261,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
this.hitLinkElement,
|
this.hitLinkElement,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
this.state,
|
this.state,
|
||||||
[lastPointerUpCoords.x, lastPointerUpCoords.y],
|
point(lastPointerUpCoords.x, lastPointerUpCoords.y),
|
||||||
this.device.editor.isMobile,
|
this.device.editor.isMobile,
|
||||||
);
|
);
|
||||||
if (lastPointerDownHittingLinkIcon && lastPointerUpHittingLinkIcon) {
|
if (lastPointerDownHittingLinkIcon && lastPointerUpHittingLinkIcon) {
|
||||||
|
@ -5497,17 +5510,18 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
// if we haven't yet created a temp point and we're beyond commit-zone
|
// if we haven't yet created a temp point and we're beyond commit-zone
|
||||||
// threshold, add a point
|
// threshold, add a point
|
||||||
if (
|
if (
|
||||||
distance2d(
|
pointDistance(
|
||||||
scenePointerX - rx,
|
point(scenePointerX - rx, scenePointerY - ry),
|
||||||
scenePointerY - ry,
|
lastPoint,
|
||||||
lastPoint[0],
|
|
||||||
lastPoint[1],
|
|
||||||
) >= LINE_CONFIRM_THRESHOLD
|
) >= LINE_CONFIRM_THRESHOLD
|
||||||
) {
|
) {
|
||||||
mutateElement(
|
mutateElement(
|
||||||
multiElement,
|
multiElement,
|
||||||
{
|
{
|
||||||
points: [...points, [scenePointerX - rx, scenePointerY - ry]],
|
points: [
|
||||||
|
...points,
|
||||||
|
point<LocalPoint>(scenePointerX - rx, scenePointerY - ry),
|
||||||
|
],
|
||||||
},
|
},
|
||||||
false,
|
false,
|
||||||
);
|
);
|
||||||
|
@ -5519,11 +5533,9 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
} else if (
|
} else if (
|
||||||
points.length > 2 &&
|
points.length > 2 &&
|
||||||
lastCommittedPoint &&
|
lastCommittedPoint &&
|
||||||
distance2d(
|
pointDistance(
|
||||||
scenePointerX - rx,
|
point(scenePointerX - rx, scenePointerY - ry),
|
||||||
scenePointerY - ry,
|
lastCommittedPoint,
|
||||||
lastCommittedPoint[0],
|
|
||||||
lastCommittedPoint[1],
|
|
||||||
) < LINE_CONFIRM_THRESHOLD
|
) < LINE_CONFIRM_THRESHOLD
|
||||||
) {
|
) {
|
||||||
setCursor(this.interactiveCanvas, CURSOR_TYPE.POINTER);
|
setCursor(this.interactiveCanvas, CURSOR_TYPE.POINTER);
|
||||||
|
@ -5570,10 +5582,10 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
this.scene.getNonDeletedElementsMap(),
|
this.scene.getNonDeletedElementsMap(),
|
||||||
[
|
[
|
||||||
...points.slice(0, -1),
|
...points.slice(0, -1),
|
||||||
[
|
point<LocalPoint>(
|
||||||
lastCommittedX + dxFromLastCommitted,
|
lastCommittedX + dxFromLastCommitted,
|
||||||
lastCommittedY + dyFromLastCommitted,
|
lastCommittedY + dyFromLastCommitted,
|
||||||
],
|
),
|
||||||
],
|
],
|
||||||
undefined,
|
undefined,
|
||||||
undefined,
|
undefined,
|
||||||
|
@ -5589,10 +5601,10 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
{
|
{
|
||||||
points: [
|
points: [
|
||||||
...points.slice(0, -1),
|
...points.slice(0, -1),
|
||||||
[
|
point<LocalPoint>(
|
||||||
lastCommittedX + dxFromLastCommitted,
|
lastCommittedX + dxFromLastCommitted,
|
||||||
lastCommittedY + dyFromLastCommitted,
|
lastCommittedY + dyFromLastCommitted,
|
||||||
],
|
),
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
false,
|
false,
|
||||||
|
@ -5817,17 +5829,15 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const distance = distance2d(
|
const distance = pointDistance(
|
||||||
pointerDownState.lastCoords.x,
|
point(pointerDownState.lastCoords.x, pointerDownState.lastCoords.y),
|
||||||
pointerDownState.lastCoords.y,
|
point(scenePointer.x, scenePointer.y),
|
||||||
scenePointer.x,
|
|
||||||
scenePointer.y,
|
|
||||||
);
|
);
|
||||||
const threshold = this.getElementHitThreshold();
|
const threshold = this.getElementHitThreshold();
|
||||||
const point = { ...pointerDownState.lastCoords };
|
const p = { ...pointerDownState.lastCoords };
|
||||||
let samplingInterval = 0;
|
let samplingInterval = 0;
|
||||||
while (samplingInterval <= distance) {
|
while (samplingInterval <= distance) {
|
||||||
const hitElements = this.getElementsAtPosition(point.x, point.y);
|
const hitElements = this.getElementsAtPosition(p.x, p.y);
|
||||||
processElements(hitElements);
|
processElements(hitElements);
|
||||||
|
|
||||||
// Exit since we reached current point
|
// Exit since we reached current point
|
||||||
|
@ -5839,12 +5849,10 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
samplingInterval = Math.min(samplingInterval + threshold, distance);
|
samplingInterval = Math.min(samplingInterval + threshold, distance);
|
||||||
|
|
||||||
const distanceRatio = samplingInterval / distance;
|
const distanceRatio = samplingInterval / distance;
|
||||||
const nextX =
|
const nextX = (1 - distanceRatio) * p.x + distanceRatio * scenePointer.x;
|
||||||
(1 - distanceRatio) * point.x + distanceRatio * scenePointer.x;
|
const nextY = (1 - distanceRatio) * p.y + distanceRatio * scenePointer.y;
|
||||||
const nextY =
|
p.x = nextX;
|
||||||
(1 - distanceRatio) * point.y + distanceRatio * scenePointer.y;
|
p.y = nextY;
|
||||||
point.x = nextX;
|
|
||||||
point.y = nextY;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pointerDownState.lastCoords.x = scenePointer.x;
|
pointerDownState.lastCoords.x = scenePointer.x;
|
||||||
|
@ -5970,6 +5978,16 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
this.maybeCleanupAfterMissingPointerUp(event.nativeEvent);
|
this.maybeCleanupAfterMissingPointerUp(event.nativeEvent);
|
||||||
this.maybeUnfollowRemoteUser();
|
this.maybeUnfollowRemoteUser();
|
||||||
|
|
||||||
|
if (this.state.searchMatches) {
|
||||||
|
this.setState((state) => ({
|
||||||
|
searchMatches: state.searchMatches.map((searchMatch) => ({
|
||||||
|
...searchMatch,
|
||||||
|
focus: false,
|
||||||
|
})),
|
||||||
|
}));
|
||||||
|
jotaiStore.set(searchItemInFocusAtom, null);
|
||||||
|
}
|
||||||
|
|
||||||
// since contextMenu options are potentially evaluated on each render,
|
// since contextMenu options are potentially evaluated on each render,
|
||||||
// and an contextMenu action may depend on selection state, we must
|
// and an contextMenu action may depend on selection state, we must
|
||||||
// close the contextMenu before we update the selection on pointerDown
|
// close the contextMenu before we update the selection on pointerDown
|
||||||
|
@ -6325,7 +6343,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
this.hitLinkElement,
|
this.hitLinkElement,
|
||||||
this.scene.getNonDeletedElementsMap(),
|
this.scene.getNonDeletedElementsMap(),
|
||||||
this.state,
|
this.state,
|
||||||
[scenePointer.x, scenePointer.y],
|
point(scenePointer.x, scenePointer.y),
|
||||||
)
|
)
|
||||||
) {
|
) {
|
||||||
this.handleEmbeddableCenterClick(this.hitLinkElement);
|
this.handleEmbeddableCenterClick(this.hitLinkElement);
|
||||||
|
@ -6398,8 +6416,16 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
}
|
}
|
||||||
isPanning = true;
|
isPanning = true;
|
||||||
|
|
||||||
if (!this.state.editingTextElement) {
|
// due to event.preventDefault below, container wouldn't get focus
|
||||||
|
// automatically
|
||||||
|
this.focusContainer();
|
||||||
|
|
||||||
// preventing defualt while text editing messes with cursor/focus
|
// preventing defualt while text editing messes with cursor/focus
|
||||||
|
if (!this.state.editingTextElement) {
|
||||||
|
// necessary to prevent browser from scrolling the page if excalidraw
|
||||||
|
// not full-page #4489
|
||||||
|
//
|
||||||
|
// as such, the above is broken when panning canvas while in wysiwyg
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -7008,7 +7034,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
simulatePressure,
|
simulatePressure,
|
||||||
locked: false,
|
locked: false,
|
||||||
frameId: topLayerFrame ? topLayerFrame.id : null,
|
frameId: topLayerFrame ? topLayerFrame.id : null,
|
||||||
points: [[0, 0]],
|
points: [point<LocalPoint>(0, 0)],
|
||||||
pressures: simulatePressure ? [] : [event.pressure],
|
pressures: simulatePressure ? [] : [event.pressure],
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -7216,11 +7242,9 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
if (
|
if (
|
||||||
multiElement.points.length > 1 &&
|
multiElement.points.length > 1 &&
|
||||||
lastCommittedPoint &&
|
lastCommittedPoint &&
|
||||||
distance2d(
|
pointDistance(
|
||||||
pointerDownState.origin.x - rx,
|
point(pointerDownState.origin.x - rx, pointerDownState.origin.y - ry),
|
||||||
pointerDownState.origin.y - ry,
|
lastCommittedPoint,
|
||||||
lastCommittedPoint[0],
|
|
||||||
lastCommittedPoint[1],
|
|
||||||
) < LINE_CONFIRM_THRESHOLD
|
) < LINE_CONFIRM_THRESHOLD
|
||||||
) {
|
) {
|
||||||
this.actionManager.executeAction(actionFinalize);
|
this.actionManager.executeAction(actionFinalize);
|
||||||
|
@ -7321,7 +7345,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
mutateElement(element, {
|
mutateElement(element, {
|
||||||
points: [...element.points, [0, 0]],
|
points: [...element.points, point<LocalPoint>(0, 0)],
|
||||||
});
|
});
|
||||||
const boundElement = getHoveredElementForBinding(
|
const boundElement = getHoveredElementForBinding(
|
||||||
pointerDownState.origin,
|
pointerDownState.origin,
|
||||||
|
@ -7573,11 +7597,9 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
this.state.activeTool.type === "line")
|
this.state.activeTool.type === "line")
|
||||||
) {
|
) {
|
||||||
if (
|
if (
|
||||||
distance2d(
|
pointDistance(
|
||||||
pointerCoords.x,
|
point(pointerCoords.x, pointerCoords.y),
|
||||||
pointerCoords.y,
|
point(pointerDownState.origin.x, pointerDownState.origin.y),
|
||||||
pointerDownState.origin.x,
|
|
||||||
pointerDownState.origin.y,
|
|
||||||
) < DRAGGING_THRESHOLD
|
) < DRAGGING_THRESHOLD
|
||||||
) {
|
) {
|
||||||
return;
|
return;
|
||||||
|
@ -7926,7 +7948,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
mutateElement(
|
mutateElement(
|
||||||
newElement,
|
newElement,
|
||||||
{
|
{
|
||||||
points: [...points, [dx, dy]],
|
points: [...points, point<LocalPoint>(dx, dy)],
|
||||||
pressures,
|
pressures,
|
||||||
},
|
},
|
||||||
false,
|
false,
|
||||||
|
@ -7955,7 +7977,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
mutateElement(
|
mutateElement(
|
||||||
newElement,
|
newElement,
|
||||||
{
|
{
|
||||||
points: [...points, [dx, dy]],
|
points: [...points, point<LocalPoint>(dx, dy)],
|
||||||
},
|
},
|
||||||
false,
|
false,
|
||||||
);
|
);
|
||||||
|
@ -7963,8 +7985,8 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
mutateElbowArrow(
|
mutateElbowArrow(
|
||||||
newElement,
|
newElement,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
[...points.slice(0, -1), [dx, dy]],
|
[...points.slice(0, -1), point<LocalPoint>(dx, dy)],
|
||||||
[0, 0],
|
vector(0, 0),
|
||||||
undefined,
|
undefined,
|
||||||
{
|
{
|
||||||
isDragging: true,
|
isDragging: true,
|
||||||
|
@ -7975,7 +7997,7 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
mutateElement(
|
mutateElement(
|
||||||
newElement,
|
newElement,
|
||||||
{
|
{
|
||||||
points: [...points.slice(0, -1), [dx, dy]],
|
points: [...points.slice(0, -1), point<LocalPoint>(dx, dy)],
|
||||||
},
|
},
|
||||||
false,
|
false,
|
||||||
);
|
);
|
||||||
|
@ -8284,9 +8306,9 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
: [...newElement.pressures, childEvent.pressure];
|
: [...newElement.pressures, childEvent.pressure];
|
||||||
|
|
||||||
mutateElement(newElement, {
|
mutateElement(newElement, {
|
||||||
points: [...points, [dx, dy]],
|
points: [...points, point<LocalPoint>(dx, dy)],
|
||||||
pressures,
|
pressures,
|
||||||
lastCommittedPoint: [dx, dy],
|
lastCommittedPoint: point<LocalPoint>(dx, dy),
|
||||||
});
|
});
|
||||||
|
|
||||||
this.actionManager.executeAction(actionFinalize);
|
this.actionManager.executeAction(actionFinalize);
|
||||||
|
@ -8333,7 +8355,10 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
mutateElement(newElement, {
|
mutateElement(newElement, {
|
||||||
points: [
|
points: [
|
||||||
...newElement.points,
|
...newElement.points,
|
||||||
[pointerCoords.x - newElement.x, pointerCoords.y - newElement.y],
|
point<LocalPoint>(
|
||||||
|
pointerCoords.x - newElement.x,
|
||||||
|
pointerCoords.y - newElement.y,
|
||||||
|
),
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
this.setState({
|
this.setState({
|
||||||
|
@ -8643,11 +8668,9 @@ class App extends React.Component<AppProps, AppState> {
|
||||||
if (isEraserActive(this.state) && pointerStart && pointerEnd) {
|
if (isEraserActive(this.state) && pointerStart && pointerEnd) {
|
||||||
this.eraserTrail.endPath();
|
this.eraserTrail.endPath();
|
||||||
|
|
||||||
const draggedDistance = distance2d(
|
const draggedDistance = pointDistance(
|
||||||
pointerStart.clientX,
|
point(pointerStart.clientX, pointerStart.clientY),
|
||||||
pointerStart.clientY,
|
point(pointerEnd.clientX, pointerEnd.clientY),
|
||||||
pointerEnd.clientX,
|
|
||||||
pointerEnd.clientY,
|
|
||||||
);
|
);
|
||||||
|
|
||||||
if (draggedDistance === 0) {
|
if (draggedDistance === 0) {
|
||||||
|
|
|
@ -106,7 +106,7 @@ const ColorPickerPopupContent = ({
|
||||||
return (
|
return (
|
||||||
<PropertiesPopover
|
<PropertiesPopover
|
||||||
container={container}
|
container={container}
|
||||||
style={{ maxWidth: "208px" }}
|
style={{ maxWidth: "13rem" }}
|
||||||
onFocusOutside={(event) => {
|
onFocusOutside={(event) => {
|
||||||
// refocus due to eye dropper
|
// refocus due to eye dropper
|
||||||
focusPickerContent();
|
focusPickerContent();
|
||||||
|
|
|
@ -43,7 +43,11 @@ import { InlineIcon } from "../InlineIcon";
|
||||||
import { SHAPES } from "../../shapes";
|
import { SHAPES } from "../../shapes";
|
||||||
import { canChangeBackgroundColor, canChangeStrokeColor } from "../Actions";
|
import { canChangeBackgroundColor, canChangeStrokeColor } from "../Actions";
|
||||||
import { useStableCallback } from "../../hooks/useStableCallback";
|
import { useStableCallback } from "../../hooks/useStableCallback";
|
||||||
import { actionClearCanvas, actionLink } from "../../actions";
|
import {
|
||||||
|
actionClearCanvas,
|
||||||
|
actionLink,
|
||||||
|
actionToggleSearchMenu,
|
||||||
|
} from "../../actions";
|
||||||
import { jotaiStore } from "../../jotai";
|
import { jotaiStore } from "../../jotai";
|
||||||
import { activeConfirmDialogAtom } from "../ActiveConfirmDialog";
|
import { activeConfirmDialogAtom } from "../ActiveConfirmDialog";
|
||||||
import type { CommandPaletteItem } from "./types";
|
import type { CommandPaletteItem } from "./types";
|
||||||
|
@ -382,6 +386,15 @@ function CommandPaletteInner({
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: t("search.title"),
|
||||||
|
category: DEFAULT_CATEGORIES.app,
|
||||||
|
icon: searchIcon,
|
||||||
|
viewMode: true,
|
||||||
|
perform: () => {
|
||||||
|
actionManager.executeAction(actionToggleSearchMenu);
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
label: t("labels.changeStroke"),
|
label: t("labels.changeStroke"),
|
||||||
keywords: ["color", "outline"],
|
keywords: ["color", "outline"],
|
||||||
|
|
|
@ -2,7 +2,6 @@ import clsx from "clsx";
|
||||||
import { DEFAULT_SIDEBAR, LIBRARY_SIDEBAR_TAB } from "../constants";
|
import { DEFAULT_SIDEBAR, LIBRARY_SIDEBAR_TAB } from "../constants";
|
||||||
import { useTunnels } from "../context/tunnels";
|
import { useTunnels } from "../context/tunnels";
|
||||||
import { useUIAppState } from "../context/ui-appState";
|
import { useUIAppState } from "../context/ui-appState";
|
||||||
import { t } from "../i18n";
|
|
||||||
import type { MarkOptional, Merge } from "../utility-types";
|
import type { MarkOptional, Merge } from "../utility-types";
|
||||||
import { composeEventHandlers } from "../utils";
|
import { composeEventHandlers } from "../utils";
|
||||||
import { useExcalidrawSetAppState } from "./App";
|
import { useExcalidrawSetAppState } from "./App";
|
||||||
|
@ -10,6 +9,8 @@ import { withInternalFallback } from "./hoc/withInternalFallback";
|
||||||
import { LibraryMenu } from "./LibraryMenu";
|
import { LibraryMenu } from "./LibraryMenu";
|
||||||
import type { SidebarProps, SidebarTriggerProps } from "./Sidebar/common";
|
import type { SidebarProps, SidebarTriggerProps } from "./Sidebar/common";
|
||||||
import { Sidebar } from "./Sidebar/Sidebar";
|
import { Sidebar } from "./Sidebar/Sidebar";
|
||||||
|
import "../components/dropdownMenu/DropdownMenu.scss";
|
||||||
|
import { t } from "../i18n";
|
||||||
|
|
||||||
const DefaultSidebarTrigger = withInternalFallback(
|
const DefaultSidebarTrigger = withInternalFallback(
|
||||||
"DefaultSidebarTrigger",
|
"DefaultSidebarTrigger",
|
||||||
|
@ -68,8 +69,7 @@ export const DefaultSidebar = Object.assign(
|
||||||
return (
|
return (
|
||||||
<Sidebar
|
<Sidebar
|
||||||
{...rest}
|
{...rest}
|
||||||
name="default"
|
name={"default"}
|
||||||
key="default"
|
|
||||||
className={clsx("default-sidebar", className)}
|
className={clsx("default-sidebar", className)}
|
||||||
docked={docked ?? appState.defaultSidebarDockedPreference}
|
docked={docked ?? appState.defaultSidebarDockedPreference}
|
||||||
onDock={
|
onDock={
|
||||||
|
|
|
@ -288,6 +288,10 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => {
|
||||||
label={t("stats.fullTitle")}
|
label={t("stats.fullTitle")}
|
||||||
shortcuts={[getShortcutKey("Alt+/")]}
|
shortcuts={[getShortcutKey("Alt+/")]}
|
||||||
/>
|
/>
|
||||||
|
<Shortcut
|
||||||
|
label={t("search.title")}
|
||||||
|
shortcuts={[getShortcutFromShortcutName("searchMenu")]}
|
||||||
|
/>
|
||||||
<Shortcut
|
<Shortcut
|
||||||
label={t("commandPalette.title")}
|
label={t("commandPalette.title")}
|
||||||
shortcuts={
|
shortcuts={
|
||||||
|
|
|
@ -13,6 +13,7 @@ import { isEraserActive } from "../appState";
|
||||||
import "./HintViewer.scss";
|
import "./HintViewer.scss";
|
||||||
import { isNodeInFlowchart } from "../element/flowchart";
|
import { isNodeInFlowchart } from "../element/flowchart";
|
||||||
import { isGridModeEnabled } from "../snapping";
|
import { isGridModeEnabled } from "../snapping";
|
||||||
|
import { SEARCH_SIDEBAR } from "../constants";
|
||||||
|
|
||||||
interface HintViewerProps {
|
interface HintViewerProps {
|
||||||
appState: UIAppState;
|
appState: UIAppState;
|
||||||
|
@ -30,6 +31,13 @@ const getHints = ({
|
||||||
const { activeTool, isResizing, isRotating, lastPointerDownWith } = appState;
|
const { activeTool, isResizing, isRotating, lastPointerDownWith } = appState;
|
||||||
const multiMode = appState.multiElement !== null;
|
const multiMode = appState.multiElement !== null;
|
||||||
|
|
||||||
|
if (
|
||||||
|
appState.openSidebar?.name === SEARCH_SIDEBAR.name &&
|
||||||
|
appState.searchMatches?.length
|
||||||
|
) {
|
||||||
|
return t("hints.dismissSearch");
|
||||||
|
}
|
||||||
|
|
||||||
if (appState.openSidebar && !device.editor.canFitSidebar) {
|
if (appState.openSidebar && !device.editor.canFitSidebar) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@ import {
|
||||||
CLASSES,
|
CLASSES,
|
||||||
DEFAULT_SIDEBAR,
|
DEFAULT_SIDEBAR,
|
||||||
LIBRARY_SIDEBAR_WIDTH,
|
LIBRARY_SIDEBAR_WIDTH,
|
||||||
|
SEARCH_SIDEBAR,
|
||||||
TOOL_TYPE,
|
TOOL_TYPE,
|
||||||
} from "../constants";
|
} from "../constants";
|
||||||
import { showSelectedShapeActions } from "../element";
|
import { showSelectedShapeActions } from "../element";
|
||||||
|
@ -63,6 +64,7 @@ import { LaserPointerButton } from "./LaserPointerButton";
|
||||||
import { TTDDialog } from "./TTDDialog/TTDDialog";
|
import { TTDDialog } from "./TTDDialog/TTDDialog";
|
||||||
import { Stats } from "./Stats";
|
import { Stats } from "./Stats";
|
||||||
import { actionToggleStats } from "../actions";
|
import { actionToggleStats } from "../actions";
|
||||||
|
import { SearchSidebar } from "./SearchSidebar";
|
||||||
|
|
||||||
interface LayerUIProps {
|
interface LayerUIProps {
|
||||||
actionManager: ActionManager;
|
actionManager: ActionManager;
|
||||||
|
@ -99,6 +101,7 @@ const DefaultMainMenu: React.FC<{
|
||||||
{UIOptions.canvasActions.saveAsImage && (
|
{UIOptions.canvasActions.saveAsImage && (
|
||||||
<MainMenu.DefaultItems.SaveAsImage />
|
<MainMenu.DefaultItems.SaveAsImage />
|
||||||
)}
|
)}
|
||||||
|
<MainMenu.DefaultItems.SearchMenu />
|
||||||
<MainMenu.DefaultItems.Help />
|
<MainMenu.DefaultItems.Help />
|
||||||
<MainMenu.DefaultItems.ClearCanvas />
|
<MainMenu.DefaultItems.ClearCanvas />
|
||||||
<MainMenu.Separator />
|
<MainMenu.Separator />
|
||||||
|
@ -362,6 +365,10 @@ const LayerUI = ({
|
||||||
|
|
||||||
const renderSidebars = () => {
|
const renderSidebars = () => {
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{appState.openSidebar?.name === SEARCH_SIDEBAR.name && (
|
||||||
|
<SearchSidebar />
|
||||||
|
)}
|
||||||
<DefaultSidebar
|
<DefaultSidebar
|
||||||
__fallback
|
__fallback
|
||||||
onDock={(docked) => {
|
onDock={(docked) => {
|
||||||
|
@ -372,6 +379,7 @@ const LayerUI = ({
|
||||||
);
|
);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
110
packages/excalidraw/components/SearchMenu.scss
Normal file
110
packages/excalidraw/components/SearchMenu.scss
Normal file
|
@ -0,0 +1,110 @@
|
||||||
|
@import "open-color/open-color";
|
||||||
|
|
||||||
|
.excalidraw {
|
||||||
|
.layer-ui__search {
|
||||||
|
flex: 1 0 auto;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
padding: 8px 0 0 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layer-ui__search-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0 0.75rem;
|
||||||
|
.ExcTextField {
|
||||||
|
flex: 1 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ExcTextField__input {
|
||||||
|
background-color: #f5f5f9;
|
||||||
|
@at-root .excalidraw.theme--dark#{&} {
|
||||||
|
background-color: #31303b;
|
||||||
|
}
|
||||||
|
|
||||||
|
border-radius: var(--border-radius-md);
|
||||||
|
border: 0;
|
||||||
|
|
||||||
|
input::placeholder {
|
||||||
|
font-size: 0.9rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.layer-ui__search-count {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
padding: 8px 8px 0 8px;
|
||||||
|
margin: 0 0.75rem 0.25rem 0.75rem;
|
||||||
|
font-size: 0.8em;
|
||||||
|
|
||||||
|
.result-nav {
|
||||||
|
display: flex;
|
||||||
|
|
||||||
|
.result-nav-btn {
|
||||||
|
width: 36px;
|
||||||
|
height: 36px;
|
||||||
|
--button-border: transparent;
|
||||||
|
|
||||||
|
&:active {
|
||||||
|
background-color: var(--color-surface-high);
|
||||||
|
}
|
||||||
|
|
||||||
|
&:first {
|
||||||
|
margin-right: 4px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.layer-ui__search-result-container {
|
||||||
|
overflow-y: auto;
|
||||||
|
flex: 1 1 0;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
|
||||||
|
gap: 0.125rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layer-ui__result-item {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
min-height: 2rem;
|
||||||
|
flex: 0 0 auto;
|
||||||
|
padding: 0.25rem 0.75rem;
|
||||||
|
cursor: pointer;
|
||||||
|
border: 1px solid transparent;
|
||||||
|
outline: none;
|
||||||
|
|
||||||
|
margin: 0 0.75rem;
|
||||||
|
border-radius: var(--border-radius-md);
|
||||||
|
|
||||||
|
.text-icon {
|
||||||
|
width: 1rem;
|
||||||
|
height: 1rem;
|
||||||
|
margin-right: 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preview-text {
|
||||||
|
flex: 1;
|
||||||
|
max-height: 48px;
|
||||||
|
line-height: 24px;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
word-break: break-all;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: var(--color-surface-high);
|
||||||
|
}
|
||||||
|
&:active {
|
||||||
|
border-color: var(--color-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
&.active {
|
||||||
|
background-color: var(--color-surface-high);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
701
packages/excalidraw/components/SearchMenu.tsx
Normal file
701
packages/excalidraw/components/SearchMenu.tsx
Normal file
|
@ -0,0 +1,701 @@
|
||||||
|
import { Fragment, memo, useEffect, useRef, useState } from "react";
|
||||||
|
import { collapseDownIcon, upIcon, searchIcon } from "./icons";
|
||||||
|
import { TextField } from "./TextField";
|
||||||
|
import { Button } from "./Button";
|
||||||
|
import { useApp, useExcalidrawSetAppState } from "./App";
|
||||||
|
import { debounce } from "lodash";
|
||||||
|
import type { AppClassProperties } from "../types";
|
||||||
|
import { isTextElement, newTextElement } from "../element";
|
||||||
|
import type { ExcalidrawTextElement } from "../element/types";
|
||||||
|
import { measureText } from "../element/textElement";
|
||||||
|
import { addEventListener, getFontString } from "../utils";
|
||||||
|
import { KEYS } from "../keys";
|
||||||
|
import clsx from "clsx";
|
||||||
|
import { atom, useAtom } from "jotai";
|
||||||
|
import { jotaiScope } from "../jotai";
|
||||||
|
import { t } from "../i18n";
|
||||||
|
import { isElementCompletelyInViewport } from "../element/sizeHelpers";
|
||||||
|
import { randomInteger } from "../random";
|
||||||
|
import { CLASSES, EVENT } from "../constants";
|
||||||
|
import { useStable } from "../hooks/useStable";
|
||||||
|
|
||||||
|
import "./SearchMenu.scss";
|
||||||
|
|
||||||
|
const searchQueryAtom = atom<string>("");
|
||||||
|
export const searchItemInFocusAtom = atom<number | null>(null);
|
||||||
|
|
||||||
|
const SEARCH_DEBOUNCE = 350;
|
||||||
|
|
||||||
|
type SearchMatchItem = {
|
||||||
|
textElement: ExcalidrawTextElement;
|
||||||
|
searchQuery: SearchQuery;
|
||||||
|
index: number;
|
||||||
|
preview: {
|
||||||
|
indexInSearchQuery: number;
|
||||||
|
previewText: string;
|
||||||
|
moreBefore: boolean;
|
||||||
|
moreAfter: boolean;
|
||||||
|
};
|
||||||
|
matchedLines: {
|
||||||
|
offsetX: number;
|
||||||
|
offsetY: number;
|
||||||
|
width: number;
|
||||||
|
height: number;
|
||||||
|
}[];
|
||||||
|
};
|
||||||
|
|
||||||
|
type SearchMatches = {
|
||||||
|
nonce: number | null;
|
||||||
|
items: SearchMatchItem[];
|
||||||
|
};
|
||||||
|
|
||||||
|
type SearchQuery = string & { _brand: "SearchQuery" };
|
||||||
|
|
||||||
|
export const SearchMenu = () => {
|
||||||
|
const app = useApp();
|
||||||
|
const setAppState = useExcalidrawSetAppState();
|
||||||
|
|
||||||
|
const searchInputRef = useRef<HTMLInputElement>(null);
|
||||||
|
|
||||||
|
const [inputValue, setInputValue] = useAtom(searchQueryAtom, jotaiScope);
|
||||||
|
const searchQuery = inputValue.trim() as SearchQuery;
|
||||||
|
|
||||||
|
const [isSearching, setIsSearching] = useState(false);
|
||||||
|
|
||||||
|
const [searchMatches, setSearchMatches] = useState<SearchMatches>({
|
||||||
|
nonce: null,
|
||||||
|
items: [],
|
||||||
|
});
|
||||||
|
const searchedQueryRef = useRef<SearchQuery | null>(null);
|
||||||
|
const lastSceneNonceRef = useRef<number | undefined>(undefined);
|
||||||
|
|
||||||
|
const [focusIndex, setFocusIndex] = useAtom(
|
||||||
|
searchItemInFocusAtom,
|
||||||
|
jotaiScope,
|
||||||
|
);
|
||||||
|
const elementsMap = app.scene.getNonDeletedElementsMap();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (isSearching) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
searchQuery !== searchedQueryRef.current ||
|
||||||
|
app.scene.getSceneNonce() !== lastSceneNonceRef.current
|
||||||
|
) {
|
||||||
|
searchedQueryRef.current = null;
|
||||||
|
handleSearch(searchQuery, app, (matchItems, index) => {
|
||||||
|
setSearchMatches({
|
||||||
|
nonce: randomInteger(),
|
||||||
|
items: matchItems,
|
||||||
|
});
|
||||||
|
searchedQueryRef.current = searchQuery;
|
||||||
|
lastSceneNonceRef.current = app.scene.getSceneNonce();
|
||||||
|
setAppState({
|
||||||
|
searchMatches: matchItems.map((searchMatch) => ({
|
||||||
|
id: searchMatch.textElement.id,
|
||||||
|
focus: false,
|
||||||
|
matchedLines: searchMatch.matchedLines,
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [
|
||||||
|
isSearching,
|
||||||
|
searchQuery,
|
||||||
|
elementsMap,
|
||||||
|
app,
|
||||||
|
setAppState,
|
||||||
|
setFocusIndex,
|
||||||
|
lastSceneNonceRef,
|
||||||
|
]);
|
||||||
|
|
||||||
|
const goToNextItem = () => {
|
||||||
|
if (searchMatches.items.length > 0) {
|
||||||
|
setFocusIndex((focusIndex) => {
|
||||||
|
if (focusIndex === null) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (focusIndex + 1) % searchMatches.items.length;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const goToPreviousItem = () => {
|
||||||
|
if (searchMatches.items.length > 0) {
|
||||||
|
setFocusIndex((focusIndex) => {
|
||||||
|
if (focusIndex === null) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return focusIndex - 1 < 0
|
||||||
|
? searchMatches.items.length - 1
|
||||||
|
: focusIndex - 1;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setAppState((state) => {
|
||||||
|
return {
|
||||||
|
searchMatches: state.searchMatches.map((match, index) => {
|
||||||
|
if (index === focusIndex) {
|
||||||
|
return { ...match, focus: true };
|
||||||
|
}
|
||||||
|
return { ...match, focus: false };
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}, [focusIndex, setAppState]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (searchMatches.items.length > 0 && focusIndex !== null) {
|
||||||
|
const match = searchMatches.items[focusIndex];
|
||||||
|
|
||||||
|
if (match) {
|
||||||
|
const matchAsElement = newTextElement({
|
||||||
|
text: match.searchQuery,
|
||||||
|
x: match.textElement.x + (match.matchedLines[0]?.offsetX ?? 0),
|
||||||
|
y: match.textElement.y + (match.matchedLines[0]?.offsetY ?? 0),
|
||||||
|
width: match.matchedLines[0]?.width,
|
||||||
|
height: match.matchedLines[0]?.height,
|
||||||
|
});
|
||||||
|
|
||||||
|
const isTextTiny =
|
||||||
|
match.textElement.fontSize * app.state.zoom.value < 12;
|
||||||
|
|
||||||
|
if (
|
||||||
|
!isElementCompletelyInViewport(
|
||||||
|
[matchAsElement],
|
||||||
|
app.canvas.width / window.devicePixelRatio,
|
||||||
|
app.canvas.height / window.devicePixelRatio,
|
||||||
|
{
|
||||||
|
offsetLeft: app.state.offsetLeft,
|
||||||
|
offsetTop: app.state.offsetTop,
|
||||||
|
scrollX: app.state.scrollX,
|
||||||
|
scrollY: app.state.scrollY,
|
||||||
|
zoom: app.state.zoom,
|
||||||
|
},
|
||||||
|
app.scene.getNonDeletedElementsMap(),
|
||||||
|
app.getEditorUIOffsets(),
|
||||||
|
) ||
|
||||||
|
isTextTiny
|
||||||
|
) {
|
||||||
|
let zoomOptions: Parameters<AppClassProperties["scrollToContent"]>[1];
|
||||||
|
|
||||||
|
if (isTextTiny && app.state.zoom.value >= 1) {
|
||||||
|
zoomOptions = { fitToViewport: true };
|
||||||
|
} else if (isTextTiny || app.state.zoom.value > 1) {
|
||||||
|
zoomOptions = { fitToContent: true };
|
||||||
|
}
|
||||||
|
|
||||||
|
app.scrollToContent(matchAsElement, {
|
||||||
|
animate: true,
|
||||||
|
duration: 300,
|
||||||
|
...zoomOptions,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [focusIndex, searchMatches, app]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
return () => {
|
||||||
|
setFocusIndex(null);
|
||||||
|
searchedQueryRef.current = null;
|
||||||
|
lastSceneNonceRef.current = undefined;
|
||||||
|
setAppState({
|
||||||
|
searchMatches: [],
|
||||||
|
});
|
||||||
|
setIsSearching(false);
|
||||||
|
};
|
||||||
|
}, [setAppState, setFocusIndex]);
|
||||||
|
|
||||||
|
const stableState = useStable({
|
||||||
|
goToNextItem,
|
||||||
|
goToPreviousItem,
|
||||||
|
searchMatches,
|
||||||
|
});
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const eventHandler = (event: KeyboardEvent) => {
|
||||||
|
if (
|
||||||
|
event.key === KEYS.ESCAPE &&
|
||||||
|
!app.state.openDialog &&
|
||||||
|
!app.state.openPopup
|
||||||
|
) {
|
||||||
|
event.preventDefault();
|
||||||
|
event.stopPropagation();
|
||||||
|
setAppState({
|
||||||
|
openSidebar: null,
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event[KEYS.CTRL_OR_CMD] && event.key === KEYS.F) {
|
||||||
|
event.preventDefault();
|
||||||
|
event.stopPropagation();
|
||||||
|
|
||||||
|
if (!searchInputRef.current?.matches(":focus")) {
|
||||||
|
if (app.state.openDialog) {
|
||||||
|
setAppState({
|
||||||
|
openDialog: null,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
searchInputRef.current?.focus();
|
||||||
|
searchInputRef.current?.select();
|
||||||
|
} else {
|
||||||
|
setAppState({
|
||||||
|
openSidebar: null,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
event.target instanceof HTMLElement &&
|
||||||
|
event.target.closest(".layer-ui__search")
|
||||||
|
) {
|
||||||
|
if (stableState.searchMatches.items.length) {
|
||||||
|
if (event.key === KEYS.ENTER) {
|
||||||
|
event.stopPropagation();
|
||||||
|
stableState.goToNextItem();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.key === KEYS.ARROW_UP) {
|
||||||
|
event.stopPropagation();
|
||||||
|
stableState.goToPreviousItem();
|
||||||
|
} else if (event.key === KEYS.ARROW_DOWN) {
|
||||||
|
event.stopPropagation();
|
||||||
|
stableState.goToNextItem();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// `capture` needed to prevent firing on initial open from App.tsx,
|
||||||
|
// as well as to handle events before App ones
|
||||||
|
return addEventListener(window, EVENT.KEYDOWN, eventHandler, {
|
||||||
|
capture: true,
|
||||||
|
});
|
||||||
|
}, [setAppState, stableState, app]);
|
||||||
|
|
||||||
|
const matchCount = `${searchMatches.items.length} ${
|
||||||
|
searchMatches.items.length === 1
|
||||||
|
? t("search.singleResult")
|
||||||
|
: t("search.multipleResults")
|
||||||
|
}`;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="layer-ui__search">
|
||||||
|
<div className="layer-ui__search-header">
|
||||||
|
<TextField
|
||||||
|
className={CLASSES.SEARCH_MENU_INPUT_WRAPPER}
|
||||||
|
value={inputValue}
|
||||||
|
ref={searchInputRef}
|
||||||
|
placeholder={t("search.placeholder")}
|
||||||
|
icon={searchIcon}
|
||||||
|
onChange={(value) => {
|
||||||
|
setInputValue(value);
|
||||||
|
setIsSearching(true);
|
||||||
|
const searchQuery = value.trim() as SearchQuery;
|
||||||
|
handleSearch(searchQuery, app, (matchItems, index) => {
|
||||||
|
setSearchMatches({
|
||||||
|
nonce: randomInteger(),
|
||||||
|
items: matchItems,
|
||||||
|
});
|
||||||
|
setFocusIndex(index);
|
||||||
|
searchedQueryRef.current = searchQuery;
|
||||||
|
lastSceneNonceRef.current = app.scene.getSceneNonce();
|
||||||
|
setAppState({
|
||||||
|
searchMatches: matchItems.map((searchMatch) => ({
|
||||||
|
id: searchMatch.textElement.id,
|
||||||
|
focus: false,
|
||||||
|
matchedLines: searchMatch.matchedLines,
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
|
||||||
|
setIsSearching(false);
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
selectOnRender
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="layer-ui__search-count">
|
||||||
|
{searchMatches.items.length > 0 && (
|
||||||
|
<>
|
||||||
|
{focusIndex !== null && focusIndex > -1 ? (
|
||||||
|
<div>
|
||||||
|
{focusIndex + 1} / {matchCount}
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div>{matchCount}</div>
|
||||||
|
)}
|
||||||
|
<div className="result-nav">
|
||||||
|
<Button
|
||||||
|
onSelect={() => {
|
||||||
|
goToNextItem();
|
||||||
|
}}
|
||||||
|
className="result-nav-btn"
|
||||||
|
>
|
||||||
|
{collapseDownIcon}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
onSelect={() => {
|
||||||
|
goToPreviousItem();
|
||||||
|
}}
|
||||||
|
className="result-nav-btn"
|
||||||
|
>
|
||||||
|
{upIcon}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{searchMatches.items.length === 0 &&
|
||||||
|
searchQuery &&
|
||||||
|
searchedQueryRef.current && (
|
||||||
|
<div style={{ margin: "1rem auto" }}>{t("search.noMatch")}</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<MatchList
|
||||||
|
matches={searchMatches}
|
||||||
|
onItemClick={setFocusIndex}
|
||||||
|
focusIndex={focusIndex}
|
||||||
|
searchQuery={searchQuery}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const ListItem = (props: {
|
||||||
|
preview: SearchMatchItem["preview"];
|
||||||
|
searchQuery: SearchQuery;
|
||||||
|
highlighted: boolean;
|
||||||
|
onClick?: () => void;
|
||||||
|
}) => {
|
||||||
|
const preview = [
|
||||||
|
props.preview.moreBefore ? "..." : "",
|
||||||
|
props.preview.previewText.slice(0, props.preview.indexInSearchQuery),
|
||||||
|
props.preview.previewText.slice(
|
||||||
|
props.preview.indexInSearchQuery,
|
||||||
|
props.preview.indexInSearchQuery + props.searchQuery.length,
|
||||||
|
),
|
||||||
|
props.preview.previewText.slice(
|
||||||
|
props.preview.indexInSearchQuery + props.searchQuery.length,
|
||||||
|
),
|
||||||
|
props.preview.moreAfter ? "..." : "",
|
||||||
|
];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
tabIndex={-1}
|
||||||
|
className={clsx("layer-ui__result-item", {
|
||||||
|
active: props.highlighted,
|
||||||
|
})}
|
||||||
|
onClick={props.onClick}
|
||||||
|
ref={(ref) => {
|
||||||
|
if (props.highlighted) {
|
||||||
|
ref?.scrollIntoView({ behavior: "auto", block: "nearest" });
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className="preview-text">
|
||||||
|
{preview.flatMap((text, idx) => (
|
||||||
|
<Fragment key={idx}>{idx === 2 ? <b>{text}</b> : text}</Fragment>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
interface MatchListProps {
|
||||||
|
matches: SearchMatches;
|
||||||
|
onItemClick: (index: number) => void;
|
||||||
|
focusIndex: number | null;
|
||||||
|
searchQuery: SearchQuery;
|
||||||
|
}
|
||||||
|
|
||||||
|
const MatchListBase = (props: MatchListProps) => {
|
||||||
|
return (
|
||||||
|
<div className="layer-ui__search-result-container">
|
||||||
|
{props.matches.items.map((searchMatch, index) => (
|
||||||
|
<ListItem
|
||||||
|
key={searchMatch.textElement.id + searchMatch.index}
|
||||||
|
searchQuery={props.searchQuery}
|
||||||
|
preview={searchMatch.preview}
|
||||||
|
highlighted={index === props.focusIndex}
|
||||||
|
onClick={() => props.onItemClick(index)}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const areEqual = (prevProps: MatchListProps, nextProps: MatchListProps) => {
|
||||||
|
return (
|
||||||
|
prevProps.matches.nonce === nextProps.matches.nonce &&
|
||||||
|
prevProps.focusIndex === nextProps.focusIndex
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const MatchList = memo(MatchListBase, areEqual);
|
||||||
|
|
||||||
|
const getMatchPreview = (
|
||||||
|
text: string,
|
||||||
|
index: number,
|
||||||
|
searchQuery: SearchQuery,
|
||||||
|
) => {
|
||||||
|
const WORDS_BEFORE = 2;
|
||||||
|
const WORDS_AFTER = 5;
|
||||||
|
|
||||||
|
const substrBeforeQuery = text.slice(0, index);
|
||||||
|
const wordsBeforeQuery = substrBeforeQuery.split(/\s+/);
|
||||||
|
// text = "small", query = "mall", not complete before
|
||||||
|
// text = "small", query = "smal", complete before
|
||||||
|
const isQueryCompleteBefore = substrBeforeQuery.endsWith(" ");
|
||||||
|
const startWordIndex =
|
||||||
|
wordsBeforeQuery.length -
|
||||||
|
WORDS_BEFORE -
|
||||||
|
1 -
|
||||||
|
(isQueryCompleteBefore ? 0 : 1);
|
||||||
|
let wordsBeforeAsString =
|
||||||
|
wordsBeforeQuery.slice(startWordIndex <= 0 ? 0 : startWordIndex).join(" ") +
|
||||||
|
(isQueryCompleteBefore ? " " : "");
|
||||||
|
|
||||||
|
const MAX_ALLOWED_CHARS = 20;
|
||||||
|
|
||||||
|
wordsBeforeAsString =
|
||||||
|
wordsBeforeAsString.length > MAX_ALLOWED_CHARS
|
||||||
|
? wordsBeforeAsString.slice(-MAX_ALLOWED_CHARS)
|
||||||
|
: wordsBeforeAsString;
|
||||||
|
|
||||||
|
const substrAfterQuery = text.slice(index + searchQuery.length);
|
||||||
|
const wordsAfter = substrAfterQuery.split(/\s+/);
|
||||||
|
// text = "small", query = "mall", complete after
|
||||||
|
// text = "small", query = "smal", not complete after
|
||||||
|
const isQueryCompleteAfter = !substrAfterQuery.startsWith(" ");
|
||||||
|
const numberOfWordsToTake = isQueryCompleteAfter
|
||||||
|
? WORDS_AFTER + 1
|
||||||
|
: WORDS_AFTER;
|
||||||
|
const wordsAfterAsString =
|
||||||
|
(isQueryCompleteAfter ? "" : " ") +
|
||||||
|
wordsAfter.slice(0, numberOfWordsToTake).join(" ");
|
||||||
|
|
||||||
|
return {
|
||||||
|
indexInSearchQuery: wordsBeforeAsString.length,
|
||||||
|
previewText: wordsBeforeAsString + searchQuery + wordsAfterAsString,
|
||||||
|
moreBefore: startWordIndex > 0,
|
||||||
|
moreAfter: wordsAfter.length > numberOfWordsToTake,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const normalizeWrappedText = (
|
||||||
|
wrappedText: string,
|
||||||
|
originalText: string,
|
||||||
|
): string => {
|
||||||
|
const wrappedLines = wrappedText.split("\n");
|
||||||
|
const normalizedLines: string[] = [];
|
||||||
|
let originalIndex = 0;
|
||||||
|
|
||||||
|
for (let i = 0; i < wrappedLines.length; i++) {
|
||||||
|
let currentLine = wrappedLines[i];
|
||||||
|
const nextLine = wrappedLines[i + 1];
|
||||||
|
|
||||||
|
if (nextLine) {
|
||||||
|
const nextLineIndexInOriginal = originalText.indexOf(
|
||||||
|
nextLine,
|
||||||
|
originalIndex,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (nextLineIndexInOriginal > currentLine.length + originalIndex) {
|
||||||
|
let j = nextLineIndexInOriginal - (currentLine.length + originalIndex);
|
||||||
|
|
||||||
|
while (j > 0) {
|
||||||
|
currentLine += " ";
|
||||||
|
j--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
normalizedLines.push(currentLine);
|
||||||
|
originalIndex = originalIndex + currentLine.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
return normalizedLines.join("\n");
|
||||||
|
};
|
||||||
|
|
||||||
|
const getMatchedLines = (
|
||||||
|
textElement: ExcalidrawTextElement,
|
||||||
|
searchQuery: SearchQuery,
|
||||||
|
index: number,
|
||||||
|
) => {
|
||||||
|
const normalizedText = normalizeWrappedText(
|
||||||
|
textElement.text,
|
||||||
|
textElement.originalText,
|
||||||
|
);
|
||||||
|
|
||||||
|
const lines = normalizedText.split("\n");
|
||||||
|
|
||||||
|
const lineIndexRanges = [];
|
||||||
|
let currentIndex = 0;
|
||||||
|
let lineNumber = 0;
|
||||||
|
|
||||||
|
for (const line of lines) {
|
||||||
|
const startIndex = currentIndex;
|
||||||
|
const endIndex = startIndex + line.length - 1;
|
||||||
|
|
||||||
|
lineIndexRanges.push({
|
||||||
|
line,
|
||||||
|
startIndex,
|
||||||
|
endIndex,
|
||||||
|
lineNumber,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Move to the next line's start index
|
||||||
|
currentIndex = endIndex + 1;
|
||||||
|
lineNumber++;
|
||||||
|
}
|
||||||
|
|
||||||
|
let startIndex = index;
|
||||||
|
let remainingQuery = textElement.originalText.slice(
|
||||||
|
index,
|
||||||
|
index + searchQuery.length,
|
||||||
|
);
|
||||||
|
const matchedLines: {
|
||||||
|
offsetX: number;
|
||||||
|
offsetY: number;
|
||||||
|
width: number;
|
||||||
|
height: number;
|
||||||
|
}[] = [];
|
||||||
|
|
||||||
|
for (const lineIndexRange of lineIndexRanges) {
|
||||||
|
if (remainingQuery === "") {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
startIndex >= lineIndexRange.startIndex &&
|
||||||
|
startIndex <= lineIndexRange.endIndex
|
||||||
|
) {
|
||||||
|
const matchCapacity = lineIndexRange.endIndex + 1 - startIndex;
|
||||||
|
const textToStart = lineIndexRange.line.slice(
|
||||||
|
0,
|
||||||
|
startIndex - lineIndexRange.startIndex,
|
||||||
|
);
|
||||||
|
|
||||||
|
const matchedWord = remainingQuery.slice(0, matchCapacity);
|
||||||
|
remainingQuery = remainingQuery.slice(matchCapacity);
|
||||||
|
|
||||||
|
const offset = measureText(
|
||||||
|
textToStart,
|
||||||
|
getFontString(textElement),
|
||||||
|
textElement.lineHeight,
|
||||||
|
true,
|
||||||
|
);
|
||||||
|
|
||||||
|
// measureText returns a non-zero width for the empty string
|
||||||
|
// which is not what we're after here, hence the check and the correction
|
||||||
|
if (textToStart === "") {
|
||||||
|
offset.width = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (textElement.textAlign !== "left" && lineIndexRange.line.length > 0) {
|
||||||
|
const lineLength = measureText(
|
||||||
|
lineIndexRange.line,
|
||||||
|
getFontString(textElement),
|
||||||
|
textElement.lineHeight,
|
||||||
|
true,
|
||||||
|
);
|
||||||
|
|
||||||
|
const spaceToStart =
|
||||||
|
textElement.textAlign === "center"
|
||||||
|
? (textElement.width - lineLength.width) / 2
|
||||||
|
: textElement.width - lineLength.width;
|
||||||
|
offset.width += spaceToStart;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { width, height } = measureText(
|
||||||
|
matchedWord,
|
||||||
|
getFontString(textElement),
|
||||||
|
textElement.lineHeight,
|
||||||
|
);
|
||||||
|
|
||||||
|
const offsetX = offset.width;
|
||||||
|
const offsetY = lineIndexRange.lineNumber * offset.height;
|
||||||
|
|
||||||
|
matchedLines.push({
|
||||||
|
offsetX,
|
||||||
|
offsetY,
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
});
|
||||||
|
|
||||||
|
startIndex += matchCapacity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return matchedLines;
|
||||||
|
};
|
||||||
|
|
||||||
|
const escapeSpecialCharacters = (string: string) => {
|
||||||
|
return string.replace(/[.*+?^${}()|[\]\\-]/g, "\\$&");
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleSearch = debounce(
|
||||||
|
(
|
||||||
|
searchQuery: SearchQuery,
|
||||||
|
app: AppClassProperties,
|
||||||
|
cb: (matchItems: SearchMatchItem[], focusIndex: number | null) => void,
|
||||||
|
) => {
|
||||||
|
if (!searchQuery || searchQuery === "") {
|
||||||
|
cb([], null);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const elements = app.scene.getNonDeletedElements();
|
||||||
|
const texts = elements.filter((el) =>
|
||||||
|
isTextElement(el),
|
||||||
|
) as ExcalidrawTextElement[];
|
||||||
|
|
||||||
|
texts.sort((a, b) => a.y - b.y);
|
||||||
|
|
||||||
|
const matchItems: SearchMatchItem[] = [];
|
||||||
|
|
||||||
|
const regex = new RegExp(escapeSpecialCharacters(searchQuery), "gi");
|
||||||
|
|
||||||
|
for (const textEl of texts) {
|
||||||
|
let match = null;
|
||||||
|
const text = textEl.originalText;
|
||||||
|
|
||||||
|
while ((match = regex.exec(text)) !== null) {
|
||||||
|
const preview = getMatchPreview(text, match.index, searchQuery);
|
||||||
|
const matchedLines = getMatchedLines(textEl, searchQuery, match.index);
|
||||||
|
|
||||||
|
if (matchedLines.length > 0) {
|
||||||
|
matchItems.push({
|
||||||
|
textElement: textEl,
|
||||||
|
searchQuery,
|
||||||
|
preview,
|
||||||
|
index: match.index,
|
||||||
|
matchedLines,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const visibleIds = new Set(
|
||||||
|
app.visibleElements.map((visibleElement) => visibleElement.id),
|
||||||
|
);
|
||||||
|
|
||||||
|
const focusIndex =
|
||||||
|
matchItems.findIndex((matchItem) =>
|
||||||
|
visibleIds.has(matchItem.textElement.id),
|
||||||
|
) ?? null;
|
||||||
|
|
||||||
|
cb(matchItems, focusIndex);
|
||||||
|
},
|
||||||
|
SEARCH_DEBOUNCE,
|
||||||
|
);
|
29
packages/excalidraw/components/SearchSidebar.tsx
Normal file
29
packages/excalidraw/components/SearchSidebar.tsx
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
import { SEARCH_SIDEBAR } from "../constants";
|
||||||
|
import { t } from "../i18n";
|
||||||
|
import { SearchMenu } from "./SearchMenu";
|
||||||
|
import { Sidebar } from "./Sidebar/Sidebar";
|
||||||
|
|
||||||
|
export const SearchSidebar = () => {
|
||||||
|
return (
|
||||||
|
<Sidebar name={SEARCH_SIDEBAR.name} docked>
|
||||||
|
<Sidebar.Tabs>
|
||||||
|
<Sidebar.Header>
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
color: "var(--color-primary)",
|
||||||
|
fontSize: "1.2em",
|
||||||
|
fontWeight: "bold",
|
||||||
|
textOverflow: "ellipsis",
|
||||||
|
overflow: "hidden",
|
||||||
|
whiteSpace: "nowrap",
|
||||||
|
paddingRight: "1em",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{t("search.title")}
|
||||||
|
</div>
|
||||||
|
</Sidebar.Header>
|
||||||
|
<SearchMenu />
|
||||||
|
</Sidebar.Tabs>
|
||||||
|
</Sidebar>
|
||||||
|
);
|
||||||
|
};
|
|
@ -2,13 +2,14 @@ import { mutateElement } from "../../element/mutateElement";
|
||||||
import { getBoundTextElement } from "../../element/textElement";
|
import { getBoundTextElement } from "../../element/textElement";
|
||||||
import { isArrowElement, isElbowArrow } from "../../element/typeChecks";
|
import { isArrowElement, isElbowArrow } from "../../element/typeChecks";
|
||||||
import type { ExcalidrawElement } from "../../element/types";
|
import type { ExcalidrawElement } from "../../element/types";
|
||||||
import { degreeToRadian, radianToDegree } from "../../math";
|
|
||||||
import { angleIcon } from "../icons";
|
import { angleIcon } from "../icons";
|
||||||
import DragInput from "./DragInput";
|
import DragInput from "./DragInput";
|
||||||
import type { DragInputCallbackType } from "./DragInput";
|
import type { DragInputCallbackType } from "./DragInput";
|
||||||
import { getStepSizedValue, isPropertyEditable, updateBindings } from "./utils";
|
import { getStepSizedValue, isPropertyEditable, updateBindings } from "./utils";
|
||||||
import type Scene from "../../scene/Scene";
|
import type Scene from "../../scene/Scene";
|
||||||
import type { AppState } from "../../types";
|
import type { AppState } from "../../types";
|
||||||
|
import type { Degrees } from "../../../math";
|
||||||
|
import { degreesToRadians, radiansToDegrees } from "../../../math";
|
||||||
|
|
||||||
interface AngleProps {
|
interface AngleProps {
|
||||||
element: ExcalidrawElement;
|
element: ExcalidrawElement;
|
||||||
|
@ -36,7 +37,7 @@ const handleDegreeChange: DragInputCallbackType<AngleProps["property"]> = ({
|
||||||
}
|
}
|
||||||
|
|
||||||
if (nextValue !== undefined) {
|
if (nextValue !== undefined) {
|
||||||
const nextAngle = degreeToRadian(nextValue);
|
const nextAngle = degreesToRadians(nextValue as Degrees);
|
||||||
mutateElement(latestElement, {
|
mutateElement(latestElement, {
|
||||||
angle: nextAngle,
|
angle: nextAngle,
|
||||||
});
|
});
|
||||||
|
@ -51,7 +52,7 @@ const handleDegreeChange: DragInputCallbackType<AngleProps["property"]> = ({
|
||||||
}
|
}
|
||||||
|
|
||||||
const originalAngleInDegrees =
|
const originalAngleInDegrees =
|
||||||
Math.round(radianToDegree(origElement.angle) * 100) / 100;
|
Math.round(radiansToDegrees(origElement.angle) * 100) / 100;
|
||||||
const changeInDegrees = Math.round(accumulatedChange);
|
const changeInDegrees = Math.round(accumulatedChange);
|
||||||
let nextAngleInDegrees = (originalAngleInDegrees + changeInDegrees) % 360;
|
let nextAngleInDegrees = (originalAngleInDegrees + changeInDegrees) % 360;
|
||||||
if (shouldChangeByStepSize) {
|
if (shouldChangeByStepSize) {
|
||||||
|
@ -61,7 +62,7 @@ const handleDegreeChange: DragInputCallbackType<AngleProps["property"]> = ({
|
||||||
nextAngleInDegrees =
|
nextAngleInDegrees =
|
||||||
nextAngleInDegrees < 0 ? nextAngleInDegrees + 360 : nextAngleInDegrees;
|
nextAngleInDegrees < 0 ? nextAngleInDegrees + 360 : nextAngleInDegrees;
|
||||||
|
|
||||||
const nextAngle = degreeToRadian(nextAngleInDegrees);
|
const nextAngle = degreesToRadians(nextAngleInDegrees as Degrees);
|
||||||
|
|
||||||
mutateElement(latestElement, {
|
mutateElement(latestElement, {
|
||||||
angle: nextAngle,
|
angle: nextAngle,
|
||||||
|
@ -80,7 +81,7 @@ const Angle = ({ element, scene, appState, property }: AngleProps) => {
|
||||||
<DragInput
|
<DragInput
|
||||||
label="A"
|
label="A"
|
||||||
icon={angleIcon}
|
icon={angleIcon}
|
||||||
value={Math.round((radianToDegree(element.angle) % 360) * 100) / 100}
|
value={Math.round((radiansToDegrees(element.angle) % 360) * 100) / 100}
|
||||||
elements={[element]}
|
elements={[element]}
|
||||||
dragInputCallback={handleDegreeChange}
|
dragInputCallback={handleDegreeChange}
|
||||||
editable={isPropertyEditable(element, "angle")}
|
editable={isPropertyEditable(element, "angle")}
|
||||||
|
|
|
@ -3,13 +3,14 @@ import { getBoundTextElement } from "../../element/textElement";
|
||||||
import { isArrowElement } from "../../element/typeChecks";
|
import { isArrowElement } from "../../element/typeChecks";
|
||||||
import type { ExcalidrawElement } from "../../element/types";
|
import type { ExcalidrawElement } from "../../element/types";
|
||||||
import { isInGroup } from "../../groups";
|
import { isInGroup } from "../../groups";
|
||||||
import { degreeToRadian, radianToDegree } from "../../math";
|
|
||||||
import type Scene from "../../scene/Scene";
|
import type Scene from "../../scene/Scene";
|
||||||
import { angleIcon } from "../icons";
|
import { angleIcon } from "../icons";
|
||||||
import DragInput from "./DragInput";
|
import DragInput from "./DragInput";
|
||||||
import type { DragInputCallbackType } from "./DragInput";
|
import type { DragInputCallbackType } from "./DragInput";
|
||||||
import { getStepSizedValue, isPropertyEditable } from "./utils";
|
import { getStepSizedValue, isPropertyEditable } from "./utils";
|
||||||
import type { AppState } from "../../types";
|
import type { AppState } from "../../types";
|
||||||
|
import type { Degrees } from "../../../math";
|
||||||
|
import { degreesToRadians, radiansToDegrees } from "../../../math";
|
||||||
|
|
||||||
interface MultiAngleProps {
|
interface MultiAngleProps {
|
||||||
elements: readonly ExcalidrawElement[];
|
elements: readonly ExcalidrawElement[];
|
||||||
|
@ -39,7 +40,7 @@ const handleDegreeChange: DragInputCallbackType<
|
||||||
);
|
);
|
||||||
|
|
||||||
if (nextValue !== undefined) {
|
if (nextValue !== undefined) {
|
||||||
const nextAngle = degreeToRadian(nextValue);
|
const nextAngle = degreesToRadians(nextValue as Degrees);
|
||||||
|
|
||||||
for (const element of editableLatestIndividualElements) {
|
for (const element of editableLatestIndividualElements) {
|
||||||
if (!element) {
|
if (!element) {
|
||||||
|
@ -71,7 +72,7 @@ const handleDegreeChange: DragInputCallbackType<
|
||||||
}
|
}
|
||||||
const originalElement = editableOriginalIndividualElements[i];
|
const originalElement = editableOriginalIndividualElements[i];
|
||||||
const originalAngleInDegrees =
|
const originalAngleInDegrees =
|
||||||
Math.round(radianToDegree(originalElement.angle) * 100) / 100;
|
Math.round(radiansToDegrees(originalElement.angle) * 100) / 100;
|
||||||
const changeInDegrees = Math.round(accumulatedChange);
|
const changeInDegrees = Math.round(accumulatedChange);
|
||||||
let nextAngleInDegrees = (originalAngleInDegrees + changeInDegrees) % 360;
|
let nextAngleInDegrees = (originalAngleInDegrees + changeInDegrees) % 360;
|
||||||
if (shouldChangeByStepSize) {
|
if (shouldChangeByStepSize) {
|
||||||
|
@ -81,7 +82,7 @@ const handleDegreeChange: DragInputCallbackType<
|
||||||
nextAngleInDegrees =
|
nextAngleInDegrees =
|
||||||
nextAngleInDegrees < 0 ? nextAngleInDegrees + 360 : nextAngleInDegrees;
|
nextAngleInDegrees < 0 ? nextAngleInDegrees + 360 : nextAngleInDegrees;
|
||||||
|
|
||||||
const nextAngle = degreeToRadian(nextAngleInDegrees);
|
const nextAngle = degreesToRadians(nextAngleInDegrees as Degrees);
|
||||||
|
|
||||||
mutateElement(
|
mutateElement(
|
||||||
latestElement,
|
latestElement,
|
||||||
|
@ -109,7 +110,7 @@ const MultiAngle = ({
|
||||||
(el) => !isInGroup(el) && isPropertyEditable(el, "angle"),
|
(el) => !isInGroup(el) && isPropertyEditable(el, "angle"),
|
||||||
);
|
);
|
||||||
const angles = editableLatestIndividualElements.map(
|
const angles = editableLatestIndividualElements.map(
|
||||||
(el) => Math.round((radianToDegree(el.angle) % 360) * 100) / 100,
|
(el) => Math.round((radiansToDegrees(el.angle) % 360) * 100) / 100,
|
||||||
);
|
);
|
||||||
const value = new Set(angles).size === 1 ? angles[0] : "Mixed";
|
const value = new Set(angles).size === 1 ? angles[0] : "Mixed";
|
||||||
|
|
||||||
|
|
|
@ -13,13 +13,14 @@ import type {
|
||||||
NonDeletedSceneElementsMap,
|
NonDeletedSceneElementsMap,
|
||||||
} from "../../element/types";
|
} from "../../element/types";
|
||||||
import type Scene from "../../scene/Scene";
|
import type Scene from "../../scene/Scene";
|
||||||
import type { AppState, Point } from "../../types";
|
import type { AppState } from "../../types";
|
||||||
import DragInput from "./DragInput";
|
import DragInput from "./DragInput";
|
||||||
import type { DragInputCallbackType } from "./DragInput";
|
import type { DragInputCallbackType } from "./DragInput";
|
||||||
import { getAtomicUnits, getStepSizedValue, isPropertyEditable } from "./utils";
|
import { getAtomicUnits, getStepSizedValue, isPropertyEditable } from "./utils";
|
||||||
import { getElementsInAtomicUnit, resizeElement } from "./utils";
|
import { getElementsInAtomicUnit, resizeElement } from "./utils";
|
||||||
import type { AtomicUnit } from "./utils";
|
import type { AtomicUnit } from "./utils";
|
||||||
import { MIN_WIDTH_OR_HEIGHT } from "../../constants";
|
import { MIN_WIDTH_OR_HEIGHT } from "../../constants";
|
||||||
|
import { point, type GlobalPoint } from "../../../math";
|
||||||
|
|
||||||
interface MultiDimensionProps {
|
interface MultiDimensionProps {
|
||||||
property: "width" | "height";
|
property: "width" | "height";
|
||||||
|
@ -104,7 +105,7 @@ const resizeGroup = (
|
||||||
nextHeight: number,
|
nextHeight: number,
|
||||||
initialHeight: number,
|
initialHeight: number,
|
||||||
aspectRatio: number,
|
aspectRatio: number,
|
||||||
anchor: Point,
|
anchor: GlobalPoint,
|
||||||
property: MultiDimensionProps["property"],
|
property: MultiDimensionProps["property"],
|
||||||
latestElements: ExcalidrawElement[],
|
latestElements: ExcalidrawElement[],
|
||||||
originalElements: ExcalidrawElement[],
|
originalElements: ExcalidrawElement[],
|
||||||
|
@ -181,7 +182,7 @@ const handleDimensionChange: DragInputCallbackType<
|
||||||
nextHeight,
|
nextHeight,
|
||||||
initialHeight,
|
initialHeight,
|
||||||
aspectRatio,
|
aspectRatio,
|
||||||
[x1, y1],
|
point(x1, y1),
|
||||||
property,
|
property,
|
||||||
latestElements,
|
latestElements,
|
||||||
originalElements,
|
originalElements,
|
||||||
|
@ -286,7 +287,7 @@ const handleDimensionChange: DragInputCallbackType<
|
||||||
nextHeight,
|
nextHeight,
|
||||||
initialHeight,
|
initialHeight,
|
||||||
aspectRatio,
|
aspectRatio,
|
||||||
[x1, y1],
|
point(x1, y1),
|
||||||
property,
|
property,
|
||||||
latestElements,
|
latestElements,
|
||||||
originalElements,
|
originalElements,
|
||||||
|
|
|
@ -4,7 +4,6 @@ import type {
|
||||||
NonDeletedExcalidrawElement,
|
NonDeletedExcalidrawElement,
|
||||||
NonDeletedSceneElementsMap,
|
NonDeletedSceneElementsMap,
|
||||||
} from "../../element/types";
|
} from "../../element/types";
|
||||||
import { rotate } from "../../math";
|
|
||||||
import type Scene from "../../scene/Scene";
|
import type Scene from "../../scene/Scene";
|
||||||
import StatsDragInput from "./DragInput";
|
import StatsDragInput from "./DragInput";
|
||||||
import type { DragInputCallbackType } from "./DragInput";
|
import type { DragInputCallbackType } from "./DragInput";
|
||||||
|
@ -14,6 +13,7 @@ import { useMemo } from "react";
|
||||||
import { getElementsInAtomicUnit, moveElement } from "./utils";
|
import { getElementsInAtomicUnit, moveElement } from "./utils";
|
||||||
import type { AtomicUnit } from "./utils";
|
import type { AtomicUnit } from "./utils";
|
||||||
import type { AppState } from "../../types";
|
import type { AppState } from "../../types";
|
||||||
|
import { point, pointRotateRads } from "../../../math";
|
||||||
|
|
||||||
interface MultiPositionProps {
|
interface MultiPositionProps {
|
||||||
property: "x" | "y";
|
property: "x" | "y";
|
||||||
|
@ -43,11 +43,9 @@ const moveElements = (
|
||||||
origElement.x + origElement.width / 2,
|
origElement.x + origElement.width / 2,
|
||||||
origElement.y + origElement.height / 2,
|
origElement.y + origElement.height / 2,
|
||||||
];
|
];
|
||||||
const [topLeftX, topLeftY] = rotate(
|
const [topLeftX, topLeftY] = pointRotateRads(
|
||||||
origElement.x,
|
point(origElement.x, origElement.y),
|
||||||
origElement.y,
|
point(cx, cy),
|
||||||
cx,
|
|
||||||
cy,
|
|
||||||
origElement.angle,
|
origElement.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -98,11 +96,9 @@ const moveGroupTo = (
|
||||||
latestElement.y + latestElement.height / 2,
|
latestElement.y + latestElement.height / 2,
|
||||||
];
|
];
|
||||||
|
|
||||||
const [topLeftX, topLeftY] = rotate(
|
const [topLeftX, topLeftY] = pointRotateRads(
|
||||||
latestElement.x,
|
point(latestElement.x, latestElement.y),
|
||||||
latestElement.y,
|
point(cx, cy),
|
||||||
cx,
|
|
||||||
cy,
|
|
||||||
latestElement.angle,
|
latestElement.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -174,11 +170,9 @@ const handlePositionChange: DragInputCallbackType<
|
||||||
origElement.x + origElement.width / 2,
|
origElement.x + origElement.width / 2,
|
||||||
origElement.y + origElement.height / 2,
|
origElement.y + origElement.height / 2,
|
||||||
];
|
];
|
||||||
const [topLeftX, topLeftY] = rotate(
|
const [topLeftX, topLeftY] = pointRotateRads(
|
||||||
origElement.x,
|
point(origElement.x, origElement.y),
|
||||||
origElement.y,
|
point(cx, cy),
|
||||||
cx,
|
|
||||||
cy,
|
|
||||||
origElement.angle,
|
origElement.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -246,7 +240,11 @@ const MultiPosition = ({
|
||||||
const [el] = elementsInUnit;
|
const [el] = elementsInUnit;
|
||||||
const [cx, cy] = [el.x + el.width / 2, el.y + el.height / 2];
|
const [cx, cy] = [el.x + el.width / 2, el.y + el.height / 2];
|
||||||
|
|
||||||
const [topLeftX, topLeftY] = rotate(el.x, el.y, cx, cy, el.angle);
|
const [topLeftX, topLeftY] = pointRotateRads(
|
||||||
|
point(el.x, el.y),
|
||||||
|
point(cx, cy),
|
||||||
|
el.angle,
|
||||||
|
);
|
||||||
|
|
||||||
return Math.round((property === "x" ? topLeftX : topLeftY) * 100) / 100;
|
return Math.round((property === "x" ? topLeftX : topLeftY) * 100) / 100;
|
||||||
}),
|
}),
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
import type { ElementsMap, ExcalidrawElement } from "../../element/types";
|
import type { ElementsMap, ExcalidrawElement } from "../../element/types";
|
||||||
import { rotate } from "../../math";
|
|
||||||
import StatsDragInput from "./DragInput";
|
import StatsDragInput from "./DragInput";
|
||||||
import type { DragInputCallbackType } from "./DragInput";
|
import type { DragInputCallbackType } from "./DragInput";
|
||||||
import { getStepSizedValue, moveElement } from "./utils";
|
import { getStepSizedValue, moveElement } from "./utils";
|
||||||
import type Scene from "../../scene/Scene";
|
import type Scene from "../../scene/Scene";
|
||||||
import type { AppState } from "../../types";
|
import type { AppState } from "../../types";
|
||||||
|
import { point, pointRotateRads } from "../../../math";
|
||||||
|
|
||||||
interface PositionProps {
|
interface PositionProps {
|
||||||
property: "x" | "y";
|
property: "x" | "y";
|
||||||
|
@ -32,11 +32,9 @@ const handlePositionChange: DragInputCallbackType<"x" | "y"> = ({
|
||||||
origElement.x + origElement.width / 2,
|
origElement.x + origElement.width / 2,
|
||||||
origElement.y + origElement.height / 2,
|
origElement.y + origElement.height / 2,
|
||||||
];
|
];
|
||||||
const [topLeftX, topLeftY] = rotate(
|
const [topLeftX, topLeftY] = pointRotateRads(
|
||||||
origElement.x,
|
point(origElement.x, origElement.y),
|
||||||
origElement.y,
|
point(cx, cy),
|
||||||
cx,
|
|
||||||
cy,
|
|
||||||
origElement.angle,
|
origElement.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -94,11 +92,9 @@ const Position = ({
|
||||||
scene,
|
scene,
|
||||||
appState,
|
appState,
|
||||||
}: PositionProps) => {
|
}: PositionProps) => {
|
||||||
const [topLeftX, topLeftY] = rotate(
|
const [topLeftX, topLeftY] = pointRotateRads(
|
||||||
element.x,
|
point(element.x, element.y),
|
||||||
element.y,
|
point(element.x + element.width / 2, element.y + element.height / 2),
|
||||||
element.x + element.width / 2,
|
|
||||||
element.y + element.height / 2,
|
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
const value =
|
const value =
|
||||||
|
|
|
@ -19,12 +19,13 @@ import type {
|
||||||
ExcalidrawLinearElement,
|
ExcalidrawLinearElement,
|
||||||
ExcalidrawTextElement,
|
ExcalidrawTextElement,
|
||||||
} from "../../element/types";
|
} from "../../element/types";
|
||||||
import { degreeToRadian, rotate } from "../../math";
|
|
||||||
import { getTextEditor, updateTextEditor } from "../../tests/queries/dom";
|
import { getTextEditor, updateTextEditor } from "../../tests/queries/dom";
|
||||||
import { getCommonBounds, isTextElement } from "../../element";
|
import { getCommonBounds, isTextElement } from "../../element";
|
||||||
import { API } from "../../tests/helpers/api";
|
import { API } from "../../tests/helpers/api";
|
||||||
import { actionGroup } from "../../actions";
|
import { actionGroup } from "../../actions";
|
||||||
import { isInGroup } from "../../groups";
|
import { isInGroup } from "../../groups";
|
||||||
|
import type { Degrees } from "../../../math";
|
||||||
|
import { degreesToRadians, point, pointRotateRads } from "../../../math";
|
||||||
|
|
||||||
const { h } = window;
|
const { h } = window;
|
||||||
const mouse = new Pointer("mouse");
|
const mouse = new Pointer("mouse");
|
||||||
|
@ -46,7 +47,9 @@ const testInputProperty = (
|
||||||
expect(input.value).toBe(initialValue.toString());
|
expect(input.value).toBe(initialValue.toString());
|
||||||
UI.updateInput(input, String(nextValue));
|
UI.updateInput(input, String(nextValue));
|
||||||
if (property === "angle") {
|
if (property === "angle") {
|
||||||
expect(element[property]).toBe(degreeToRadian(Number(nextValue)));
|
expect(element[property]).toBe(
|
||||||
|
degreesToRadians(Number(nextValue) as Degrees),
|
||||||
|
);
|
||||||
} else if (property === "fontSize" && isTextElement(element)) {
|
} else if (property === "fontSize" && isTextElement(element)) {
|
||||||
expect(element[property]).toBe(Number(nextValue));
|
expect(element[property]).toBe(Number(nextValue));
|
||||||
} else if (property !== "fontSize") {
|
} else if (property !== "fontSize") {
|
||||||
|
@ -260,11 +263,9 @@ describe("stats for a generic element", () => {
|
||||||
rectangle.x + rectangle.width / 2,
|
rectangle.x + rectangle.width / 2,
|
||||||
rectangle.y + rectangle.height / 2,
|
rectangle.y + rectangle.height / 2,
|
||||||
];
|
];
|
||||||
const [topLeftX, topLeftY] = rotate(
|
const [topLeftX, topLeftY] = pointRotateRads(
|
||||||
rectangle.x,
|
point(rectangle.x, rectangle.y),
|
||||||
rectangle.y,
|
point(cx, cy),
|
||||||
cx,
|
|
||||||
cy,
|
|
||||||
rectangle.angle,
|
rectangle.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -281,11 +282,9 @@ describe("stats for a generic element", () => {
|
||||||
|
|
||||||
testInputProperty(rectangle, "angle", "A", 0, 45);
|
testInputProperty(rectangle, "angle", "A", 0, 45);
|
||||||
|
|
||||||
let [newTopLeftX, newTopLeftY] = rotate(
|
let [newTopLeftX, newTopLeftY] = pointRotateRads(
|
||||||
rectangle.x,
|
point(rectangle.x, rectangle.y),
|
||||||
rectangle.y,
|
point(cx, cy),
|
||||||
cx,
|
|
||||||
cy,
|
|
||||||
rectangle.angle,
|
rectangle.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -294,11 +293,9 @@ describe("stats for a generic element", () => {
|
||||||
|
|
||||||
testInputProperty(rectangle, "angle", "A", 45, 66);
|
testInputProperty(rectangle, "angle", "A", 45, 66);
|
||||||
|
|
||||||
[newTopLeftX, newTopLeftY] = rotate(
|
[newTopLeftX, newTopLeftY] = pointRotateRads(
|
||||||
rectangle.x,
|
point(rectangle.x, rectangle.y),
|
||||||
rectangle.y,
|
point(cx, cy),
|
||||||
cx,
|
|
||||||
cy,
|
|
||||||
rectangle.angle,
|
rectangle.angle,
|
||||||
);
|
);
|
||||||
expect(newTopLeftX.toString()).not.toEqual(xInput.value);
|
expect(newTopLeftX.toString()).not.toEqual(xInput.value);
|
||||||
|
@ -313,11 +310,9 @@ describe("stats for a generic element", () => {
|
||||||
rectangle.x + rectangle.width / 2,
|
rectangle.x + rectangle.width / 2,
|
||||||
rectangle.y + rectangle.height / 2,
|
rectangle.y + rectangle.height / 2,
|
||||||
];
|
];
|
||||||
const [topLeftX, topLeftY] = rotate(
|
const [topLeftX, topLeftY] = pointRotateRads(
|
||||||
rectangle.x,
|
point(rectangle.x, rectangle.y),
|
||||||
rectangle.y,
|
point(cx, cy),
|
||||||
cx,
|
|
||||||
cy,
|
|
||||||
rectangle.angle,
|
rectangle.angle,
|
||||||
);
|
);
|
||||||
testInputProperty(rectangle, "width", "W", rectangle.width, 400);
|
testInputProperty(rectangle, "width", "W", rectangle.width, 400);
|
||||||
|
@ -325,11 +320,9 @@ describe("stats for a generic element", () => {
|
||||||
rectangle.x + rectangle.width / 2,
|
rectangle.x + rectangle.width / 2,
|
||||||
rectangle.y + rectangle.height / 2,
|
rectangle.y + rectangle.height / 2,
|
||||||
];
|
];
|
||||||
let [currentTopLeftX, currentTopLeftY] = rotate(
|
let [currentTopLeftX, currentTopLeftY] = pointRotateRads(
|
||||||
rectangle.x,
|
point(rectangle.x, rectangle.y),
|
||||||
rectangle.y,
|
point(cx, cy),
|
||||||
cx,
|
|
||||||
cy,
|
|
||||||
rectangle.angle,
|
rectangle.angle,
|
||||||
);
|
);
|
||||||
expect(currentTopLeftX).toBeCloseTo(topLeftX, 4);
|
expect(currentTopLeftX).toBeCloseTo(topLeftX, 4);
|
||||||
|
@ -340,11 +333,9 @@ describe("stats for a generic element", () => {
|
||||||
rectangle.x + rectangle.width / 2,
|
rectangle.x + rectangle.width / 2,
|
||||||
rectangle.y + rectangle.height / 2,
|
rectangle.y + rectangle.height / 2,
|
||||||
];
|
];
|
||||||
[currentTopLeftX, currentTopLeftY] = rotate(
|
[currentTopLeftX, currentTopLeftY] = pointRotateRads(
|
||||||
rectangle.x,
|
point(rectangle.x, rectangle.y),
|
||||||
rectangle.y,
|
point(cx, cy),
|
||||||
cx,
|
|
||||||
cy,
|
|
||||||
rectangle.angle,
|
rectangle.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -642,7 +633,7 @@ describe("stats for multiple elements", () => {
|
||||||
|
|
||||||
UI.updateInput(angle, "40");
|
UI.updateInput(angle, "40");
|
||||||
|
|
||||||
const angleInRadian = degreeToRadian(40);
|
const angleInRadian = degreesToRadians(40 as Degrees);
|
||||||
expect(rectangle?.angle).toBeCloseTo(angleInRadian, 4);
|
expect(rectangle?.angle).toBeCloseTo(angleInRadian, 4);
|
||||||
expect(text?.angle).toBeCloseTo(angleInRadian, 4);
|
expect(text?.angle).toBeCloseTo(angleInRadian, 4);
|
||||||
expect(frame.angle).toBe(0);
|
expect(frame.angle).toBe(0);
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
import type { Radians } from "../../../math";
|
||||||
|
import { point, pointRotateRads } from "../../../math";
|
||||||
import {
|
import {
|
||||||
bindOrUnbindLinearElements,
|
bindOrUnbindLinearElements,
|
||||||
updateBoundElements,
|
updateBoundElements,
|
||||||
|
@ -30,7 +32,6 @@ import {
|
||||||
getElementsInGroup,
|
getElementsInGroup,
|
||||||
isInGroup,
|
isInGroup,
|
||||||
} from "../../groups";
|
} from "../../groups";
|
||||||
import { rotate } from "../../math";
|
|
||||||
import type Scene from "../../scene/Scene";
|
import type Scene from "../../scene/Scene";
|
||||||
import type { AppState } from "../../types";
|
import type { AppState } from "../../types";
|
||||||
import { getFontString } from "../../utils";
|
import { getFontString } from "../../utils";
|
||||||
|
@ -229,23 +230,19 @@ export const moveElement = (
|
||||||
originalElement.x + originalElement.width / 2,
|
originalElement.x + originalElement.width / 2,
|
||||||
originalElement.y + originalElement.height / 2,
|
originalElement.y + originalElement.height / 2,
|
||||||
];
|
];
|
||||||
const [topLeftX, topLeftY] = rotate(
|
const [topLeftX, topLeftY] = pointRotateRads(
|
||||||
originalElement.x,
|
point(originalElement.x, originalElement.y),
|
||||||
originalElement.y,
|
point(cx, cy),
|
||||||
cx,
|
|
||||||
cy,
|
|
||||||
originalElement.angle,
|
originalElement.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
const changeInX = newTopLeftX - topLeftX;
|
const changeInX = newTopLeftX - topLeftX;
|
||||||
const changeInY = newTopLeftY - topLeftY;
|
const changeInY = newTopLeftY - topLeftY;
|
||||||
|
|
||||||
const [x, y] = rotate(
|
const [x, y] = pointRotateRads(
|
||||||
newTopLeftX,
|
point(newTopLeftX, newTopLeftY),
|
||||||
newTopLeftY,
|
point(cx + changeInX, cy + changeInY),
|
||||||
cx + changeInX,
|
-originalElement.angle as Radians,
|
||||||
cy + changeInY,
|
|
||||||
-originalElement.angle,
|
|
||||||
);
|
);
|
||||||
|
|
||||||
mutateElement(
|
mutateElement(
|
||||||
|
|
|
@ -25,11 +25,11 @@ import type { BinaryFiles } from "../../types";
|
||||||
import { ArrowRightIcon } from "../icons";
|
import { ArrowRightIcon } from "../icons";
|
||||||
|
|
||||||
import "./TTDDialog.scss";
|
import "./TTDDialog.scss";
|
||||||
import { isFiniteNumber } from "../../utils";
|
|
||||||
import { atom, useAtom } from "jotai";
|
import { atom, useAtom } from "jotai";
|
||||||
import { trackEvent } from "../../analytics";
|
import { trackEvent } from "../../analytics";
|
||||||
import { InlineIcon } from "../InlineIcon";
|
import { InlineIcon } from "../InlineIcon";
|
||||||
import { TTDDialogSubmitShortcut } from "./TTDDialogSubmitShortcut";
|
import { TTDDialogSubmitShortcut } from "./TTDDialogSubmitShortcut";
|
||||||
|
import { isFiniteNumber } from "../../../math";
|
||||||
|
|
||||||
const MIN_PROMPT_LENGTH = 3;
|
const MIN_PROMPT_LENGTH = 3;
|
||||||
const MAX_PROMPT_LENGTH = 1000;
|
const MAX_PROMPT_LENGTH = 1000;
|
||||||
|
|
|
@ -3,16 +3,29 @@
|
||||||
.excalidraw {
|
.excalidraw {
|
||||||
--ExcTextField--color: var(--color-on-surface);
|
--ExcTextField--color: var(--color-on-surface);
|
||||||
--ExcTextField--label-color: var(--color-on-surface);
|
--ExcTextField--label-color: var(--color-on-surface);
|
||||||
--ExcTextField--background: transparent;
|
--ExcTextField--background: var(--color-surface-low);
|
||||||
--ExcTextField--readonly--background: var(--color-surface-high);
|
--ExcTextField--readonly--background: var(--color-surface-high);
|
||||||
--ExcTextField--readonly--color: var(--color-on-surface);
|
--ExcTextField--readonly--color: var(--color-on-surface);
|
||||||
--ExcTextField--border: var(--color-border-outline);
|
--ExcTextField--border: var(--color-gray-20);
|
||||||
--ExcTextField--readonly--border: var(--color-border-outline-variant);
|
--ExcTextField--readonly--border: var(--color-border-outline-variant);
|
||||||
--ExcTextField--border-hover: var(--color-brand-hover);
|
--ExcTextField--border-hover: var(--color-brand-hover);
|
||||||
--ExcTextField--border-active: var(--color-brand-active);
|
--ExcTextField--border-active: var(--color-brand-active);
|
||||||
--ExcTextField--placeholder: var(--color-border-outline-variant);
|
--ExcTextField--placeholder: var(--color-border-outline-variant);
|
||||||
|
|
||||||
.ExcTextField {
|
.ExcTextField {
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
svg {
|
||||||
|
position: absolute;
|
||||||
|
top: 50%; // 50% is not exactly in the center of the input
|
||||||
|
transform: translateY(-50%);
|
||||||
|
left: 0.75rem;
|
||||||
|
width: 1.25rem;
|
||||||
|
height: 1.25rem;
|
||||||
|
color: var(--color-gray-40);
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
|
||||||
&--fullWidth {
|
&--fullWidth {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
|
@ -37,7 +50,6 @@
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding: 0 1rem;
|
|
||||||
|
|
||||||
height: 3rem;
|
height: 3rem;
|
||||||
|
|
||||||
|
@ -45,6 +57,8 @@
|
||||||
border: 1px solid var(--ExcTextField--border);
|
border: 1px solid var(--ExcTextField--border);
|
||||||
border-radius: 0.5rem;
|
border-radius: 0.5rem;
|
||||||
|
|
||||||
|
padding: 0 0.75rem;
|
||||||
|
|
||||||
&:not(&--readonly) {
|
&:not(&--readonly) {
|
||||||
&:hover {
|
&:hover {
|
||||||
border-color: var(--ExcTextField--border-hover);
|
border-color: var(--ExcTextField--border-hover);
|
||||||
|
@ -80,10 +94,6 @@
|
||||||
|
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
|
||||||
&::placeholder {
|
|
||||||
color: var(--ExcTextField--placeholder);
|
|
||||||
}
|
|
||||||
|
|
||||||
&:not(:focus) {
|
&:not(:focus) {
|
||||||
&:hover {
|
&:hover {
|
||||||
background-color: initial;
|
background-color: initial;
|
||||||
|
@ -105,5 +115,9 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&--hasIcon .ExcTextField__input {
|
||||||
|
padding-left: 2.5rem;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,9 @@ type TextFieldProps = {
|
||||||
fullWidth?: boolean;
|
fullWidth?: boolean;
|
||||||
selectOnRender?: boolean;
|
selectOnRender?: boolean;
|
||||||
|
|
||||||
|
icon?: React.ReactNode;
|
||||||
label?: string;
|
label?: string;
|
||||||
|
className?: string;
|
||||||
placeholder?: string;
|
placeholder?: string;
|
||||||
isRedacted?: boolean;
|
isRedacted?: boolean;
|
||||||
} & ({ value: string } | { defaultValue: string });
|
} & ({ value: string } | { defaultValue: string });
|
||||||
|
@ -37,6 +39,8 @@ export const TextField = forwardRef<HTMLInputElement, TextFieldProps>(
|
||||||
selectOnRender,
|
selectOnRender,
|
||||||
onKeyDown,
|
onKeyDown,
|
||||||
isRedacted = false,
|
isRedacted = false,
|
||||||
|
icon,
|
||||||
|
className,
|
||||||
...rest
|
...rest
|
||||||
},
|
},
|
||||||
ref,
|
ref,
|
||||||
|
@ -47,6 +51,8 @@ export const TextField = forwardRef<HTMLInputElement, TextFieldProps>(
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
useLayoutEffect(() => {
|
||||||
if (selectOnRender) {
|
if (selectOnRender) {
|
||||||
|
// focusing first is needed because vitest/jsdom
|
||||||
|
innerRef.current?.focus();
|
||||||
innerRef.current?.select();
|
innerRef.current?.select();
|
||||||
}
|
}
|
||||||
}, [selectOnRender]);
|
}, [selectOnRender]);
|
||||||
|
@ -56,14 +62,16 @@ export const TextField = forwardRef<HTMLInputElement, TextFieldProps>(
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={clsx("ExcTextField", {
|
className={clsx("ExcTextField", className, {
|
||||||
"ExcTextField--fullWidth": fullWidth,
|
"ExcTextField--fullWidth": fullWidth,
|
||||||
|
"ExcTextField--hasIcon": !!icon,
|
||||||
})}
|
})}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
innerRef.current?.focus();
|
innerRef.current?.focus();
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div className="ExcTextField__label">{label}</div>
|
{icon}
|
||||||
|
{label && <div className="ExcTextField__label">{label}</div>}
|
||||||
<div
|
<div
|
||||||
className={clsx("ExcTextField__input", {
|
className={clsx("ExcTextField__input", {
|
||||||
"ExcTextField__input--readonly": readonly,
|
"ExcTextField__input--readonly": readonly,
|
||||||
|
|
|
@ -203,6 +203,7 @@ const getRelevantAppStateProps = (
|
||||||
snapLines: appState.snapLines,
|
snapLines: appState.snapLines,
|
||||||
zenModeEnabled: appState.zenModeEnabled,
|
zenModeEnabled: appState.zenModeEnabled,
|
||||||
editingTextElement: appState.editingTextElement,
|
editingTextElement: appState.editingTextElement,
|
||||||
|
searchMatches: appState.searchMatches,
|
||||||
});
|
});
|
||||||
|
|
||||||
const areEqual = (
|
const areEqual = (
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import type { AppState, ExcalidrawProps, Point, UIAppState } from "../../types";
|
import type { AppState, ExcalidrawProps, UIAppState } from "../../types";
|
||||||
import {
|
import {
|
||||||
sceneCoordsToViewportCoords,
|
sceneCoordsToViewportCoords,
|
||||||
viewportCoordsToSceneCoords,
|
viewportCoordsToSceneCoords,
|
||||||
|
@ -36,6 +36,7 @@ import { trackEvent } from "../../analytics";
|
||||||
import { useAppProps, useExcalidrawAppState } from "../App";
|
import { useAppProps, useExcalidrawAppState } from "../App";
|
||||||
import { isEmbeddableElement } from "../../element/typeChecks";
|
import { isEmbeddableElement } from "../../element/typeChecks";
|
||||||
import { getLinkHandleFromCoords } from "./helpers";
|
import { getLinkHandleFromCoords } from "./helpers";
|
||||||
|
import { point, type GlobalPoint } from "../../../math";
|
||||||
|
|
||||||
const CONTAINER_WIDTH = 320;
|
const CONTAINER_WIDTH = 320;
|
||||||
const SPACE_BOTTOM = 85;
|
const SPACE_BOTTOM = 85;
|
||||||
|
@ -176,10 +177,12 @@ export const Hyperlink = ({
|
||||||
if (timeoutId) {
|
if (timeoutId) {
|
||||||
clearTimeout(timeoutId);
|
clearTimeout(timeoutId);
|
||||||
}
|
}
|
||||||
const shouldHide = shouldHideLinkPopup(element, elementsMap, appState, [
|
const shouldHide = shouldHideLinkPopup(
|
||||||
event.clientX,
|
element,
|
||||||
event.clientY,
|
elementsMap,
|
||||||
]) as boolean;
|
appState,
|
||||||
|
point(event.clientX, event.clientY),
|
||||||
|
) as boolean;
|
||||||
if (shouldHide) {
|
if (shouldHide) {
|
||||||
timeoutId = window.setTimeout(() => {
|
timeoutId = window.setTimeout(() => {
|
||||||
setAppState({ showHyperlinkPopup: false });
|
setAppState({ showHyperlinkPopup: false });
|
||||||
|
@ -416,7 +419,7 @@ const shouldHideLinkPopup = (
|
||||||
element: NonDeletedExcalidrawElement,
|
element: NonDeletedExcalidrawElement,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
[clientX, clientY]: Point,
|
[clientX, clientY]: GlobalPoint,
|
||||||
): Boolean => {
|
): Boolean => {
|
||||||
const { x: sceneX, y: sceneY } = viewportCoordsToSceneCoords(
|
const { x: sceneX, y: sceneY } = viewportCoordsToSceneCoords(
|
||||||
{ clientX, clientY },
|
{ clientX, clientY },
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
import type { GlobalPoint, Radians } from "../../../math";
|
||||||
|
import { point, pointRotateRads } from "../../../math";
|
||||||
import { MIME_TYPES } from "../../constants";
|
import { MIME_TYPES } from "../../constants";
|
||||||
import type { Bounds } from "../../element/bounds";
|
import type { Bounds } from "../../element/bounds";
|
||||||
import { getElementAbsoluteCoords } from "../../element/bounds";
|
import { getElementAbsoluteCoords } from "../../element/bounds";
|
||||||
|
@ -6,9 +8,8 @@ import type {
|
||||||
ElementsMap,
|
ElementsMap,
|
||||||
NonDeletedExcalidrawElement,
|
NonDeletedExcalidrawElement,
|
||||||
} from "../../element/types";
|
} from "../../element/types";
|
||||||
import { rotate } from "../../math";
|
|
||||||
import { DEFAULT_LINK_SIZE } from "../../renderer/renderElement";
|
import { DEFAULT_LINK_SIZE } from "../../renderer/renderElement";
|
||||||
import type { AppState, Point, UIAppState } from "../../types";
|
import type { AppState, UIAppState } from "../../types";
|
||||||
|
|
||||||
export const EXTERNAL_LINK_IMG = document.createElement("img");
|
export const EXTERNAL_LINK_IMG = document.createElement("img");
|
||||||
EXTERNAL_LINK_IMG.src = `data:${MIME_TYPES.svg}, ${encodeURIComponent(
|
EXTERNAL_LINK_IMG.src = `data:${MIME_TYPES.svg}, ${encodeURIComponent(
|
||||||
|
@ -17,7 +18,7 @@ EXTERNAL_LINK_IMG.src = `data:${MIME_TYPES.svg}, ${encodeURIComponent(
|
||||||
|
|
||||||
export const getLinkHandleFromCoords = (
|
export const getLinkHandleFromCoords = (
|
||||||
[x1, y1, x2, y2]: Bounds,
|
[x1, y1, x2, y2]: Bounds,
|
||||||
angle: number,
|
angle: Radians,
|
||||||
appState: Pick<UIAppState, "zoom">,
|
appState: Pick<UIAppState, "zoom">,
|
||||||
): Bounds => {
|
): Bounds => {
|
||||||
const size = DEFAULT_LINK_SIZE;
|
const size = DEFAULT_LINK_SIZE;
|
||||||
|
@ -33,11 +34,9 @@ export const getLinkHandleFromCoords = (
|
||||||
const x = x2 + dashedLineMargin - centeringOffset;
|
const x = x2 + dashedLineMargin - centeringOffset;
|
||||||
const y = y1 - dashedLineMargin - linkMarginY + centeringOffset;
|
const y = y1 - dashedLineMargin - linkMarginY + centeringOffset;
|
||||||
|
|
||||||
const [rotatedX, rotatedY] = rotate(
|
const [rotatedX, rotatedY] = pointRotateRads(
|
||||||
x + linkWidth / 2,
|
point(x + linkWidth / 2, y + linkHeight / 2),
|
||||||
y + linkHeight / 2,
|
point(centerX, centerY),
|
||||||
centerX,
|
|
||||||
centerY,
|
|
||||||
angle,
|
angle,
|
||||||
);
|
);
|
||||||
return [
|
return [
|
||||||
|
@ -52,7 +51,7 @@ export const isPointHittingLinkIcon = (
|
||||||
element: NonDeletedExcalidrawElement,
|
element: NonDeletedExcalidrawElement,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
[x, y]: Point,
|
[x, y]: GlobalPoint,
|
||||||
) => {
|
) => {
|
||||||
const threshold = 4 / appState.zoom.value;
|
const threshold = 4 / appState.zoom.value;
|
||||||
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
||||||
|
@ -73,7 +72,7 @@ export const isPointHittingLink = (
|
||||||
element: NonDeletedExcalidrawElement,
|
element: NonDeletedExcalidrawElement,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
[x, y]: Point,
|
[x, y]: GlobalPoint,
|
||||||
isMobile: boolean,
|
isMobile: boolean,
|
||||||
) => {
|
) => {
|
||||||
if (!element.link || appState.selectedElementIds[element.id]) {
|
if (!element.link || appState.selectedElementIds[element.id]) {
|
||||||
|
@ -86,5 +85,5 @@ export const isPointHittingLink = (
|
||||||
) {
|
) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
return isPointHittingLinkIcon(element, elementsMap, appState, [x, y]);
|
return isPointHittingLinkIcon(element, elementsMap, appState, point(x, y));
|
||||||
};
|
};
|
||||||
|
|
|
@ -2139,3 +2139,11 @@ export const collapseUpIcon = createIcon(
|
||||||
</g>,
|
</g>,
|
||||||
tablerIconProps,
|
tablerIconProps,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
export const upIcon = createIcon(
|
||||||
|
<g>
|
||||||
|
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||||
|
<path d="M6 15l6 -6l6 6" />
|
||||||
|
</g>,
|
||||||
|
tablerIconProps,
|
||||||
|
);
|
||||||
|
|
|
@ -15,6 +15,7 @@ import {
|
||||||
LoadIcon,
|
LoadIcon,
|
||||||
MoonIcon,
|
MoonIcon,
|
||||||
save,
|
save,
|
||||||
|
searchIcon,
|
||||||
SunIcon,
|
SunIcon,
|
||||||
TrashIcon,
|
TrashIcon,
|
||||||
usersIcon,
|
usersIcon,
|
||||||
|
@ -27,6 +28,7 @@ import {
|
||||||
actionLoadScene,
|
actionLoadScene,
|
||||||
actionSaveToActiveFile,
|
actionSaveToActiveFile,
|
||||||
actionShortcuts,
|
actionShortcuts,
|
||||||
|
actionToggleSearchMenu,
|
||||||
actionToggleTheme,
|
actionToggleTheme,
|
||||||
} from "../../actions";
|
} from "../../actions";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
|
@ -40,7 +42,6 @@ import DropdownMenuItemContentRadio from "../dropdownMenu/DropdownMenuItemConten
|
||||||
import { THEME } from "../../constants";
|
import { THEME } from "../../constants";
|
||||||
import type { Theme } from "../../element/types";
|
import type { Theme } from "../../element/types";
|
||||||
import { trackEvent } from "../../analytics";
|
import { trackEvent } from "../../analytics";
|
||||||
|
|
||||||
import "./DefaultItems.scss";
|
import "./DefaultItems.scss";
|
||||||
|
|
||||||
export const LoadScene = () => {
|
export const LoadScene = () => {
|
||||||
|
@ -145,6 +146,27 @@ export const CommandPalette = (opts?: { className?: string }) => {
|
||||||
};
|
};
|
||||||
CommandPalette.displayName = "CommandPalette";
|
CommandPalette.displayName = "CommandPalette";
|
||||||
|
|
||||||
|
export const SearchMenu = (opts?: { className?: string }) => {
|
||||||
|
const { t } = useI18n();
|
||||||
|
const actionManager = useExcalidrawActionManager();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<DropdownMenuItem
|
||||||
|
icon={searchIcon}
|
||||||
|
data-testid="search-menu-button"
|
||||||
|
onSelect={() => {
|
||||||
|
actionManager.executeAction(actionToggleSearchMenu);
|
||||||
|
}}
|
||||||
|
shortcut={getShortcutFromShortcutName("searchMenu")}
|
||||||
|
aria-label={t("search.title")}
|
||||||
|
className={opts?.className}
|
||||||
|
>
|
||||||
|
{t("search.title")}
|
||||||
|
</DropdownMenuItem>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
SearchMenu.displayName = "SearchMenu";
|
||||||
|
|
||||||
export const Help = () => {
|
export const Help = () => {
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
|
|
||||||
|
|
|
@ -113,6 +113,7 @@ export const ENV = {
|
||||||
export const CLASSES = {
|
export const CLASSES = {
|
||||||
SHAPE_ACTIONS_MENU: "App-menu__left",
|
SHAPE_ACTIONS_MENU: "App-menu__left",
|
||||||
ZOOM_ACTIONS: "zoom-actions",
|
ZOOM_ACTIONS: "zoom-actions",
|
||||||
|
SEARCH_MENU_INPUT_WRAPPER: "layer-ui__search-inputWrapper",
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -382,6 +383,10 @@ export const DEFAULT_SIDEBAR = {
|
||||||
defaultTab: LIBRARY_SIDEBAR_TAB,
|
defaultTab: LIBRARY_SIDEBAR_TAB,
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
|
export const SEARCH_SIDEBAR = {
|
||||||
|
name: "search",
|
||||||
|
};
|
||||||
|
|
||||||
export const LIBRARY_DISABLED_TYPES = new Set([
|
export const LIBRARY_DISABLED_TYPES = new Set([
|
||||||
"iframe",
|
"iframe",
|
||||||
"embeddable",
|
"embeddable",
|
||||||
|
|
|
@ -387,7 +387,7 @@ body.excalidraw-cursor-resize * {
|
||||||
.App-menu__left {
|
.App-menu__left {
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
padding: 0.75rem;
|
padding: 0.75rem;
|
||||||
width: 200px;
|
width: 12.5rem;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
}
|
}
|
||||||
|
|
|
@ -144,9 +144,9 @@
|
||||||
--border-radius-md: 0.375rem;
|
--border-radius-md: 0.375rem;
|
||||||
--border-radius-lg: 0.5rem;
|
--border-radius-lg: 0.5rem;
|
||||||
|
|
||||||
--color-surface-high: hsl(244, 100%, 97%);
|
--color-surface-high: #f1f0ff;
|
||||||
--color-surface-mid: hsl(240 25% 96%);
|
--color-surface-mid: #f2f2f7;
|
||||||
--color-surface-low: hsl(240 25% 94%);
|
--color-surface-low: #ececf4;
|
||||||
--color-surface-lowest: #ffffff;
|
--color-surface-lowest: #ffffff;
|
||||||
--color-on-surface: #1b1b1f;
|
--color-on-surface: #1b1b1f;
|
||||||
--color-brand-hover: #5753d0;
|
--color-brand-hover: #5753d0;
|
||||||
|
|
|
@ -40,11 +40,7 @@ import {
|
||||||
import { getDefaultAppState } from "../appState";
|
import { getDefaultAppState } from "../appState";
|
||||||
import { LinearElementEditor } from "../element/linearElementEditor";
|
import { LinearElementEditor } from "../element/linearElementEditor";
|
||||||
import { bumpVersion } from "../element/mutateElement";
|
import { bumpVersion } from "../element/mutateElement";
|
||||||
import {
|
import { getUpdatedTimestamp, updateActiveTool } from "../utils";
|
||||||
getUpdatedTimestamp,
|
|
||||||
isFiniteNumber,
|
|
||||||
updateActiveTool,
|
|
||||||
} from "../utils";
|
|
||||||
import { arrayToMap } from "../utils";
|
import { arrayToMap } from "../utils";
|
||||||
import type { MarkOptional, Mutable } from "../utility-types";
|
import type { MarkOptional, Mutable } from "../utility-types";
|
||||||
import { detectLineHeight, getContainerElement } from "../element/textElement";
|
import { detectLineHeight, getContainerElement } from "../element/textElement";
|
||||||
|
@ -58,6 +54,8 @@ import {
|
||||||
getNormalizedGridStep,
|
getNormalizedGridStep,
|
||||||
getNormalizedZoom,
|
getNormalizedZoom,
|
||||||
} from "../scene";
|
} from "../scene";
|
||||||
|
import type { LocalPoint, Radians } from "../../math";
|
||||||
|
import { isFiniteNumber, point } from "../../math";
|
||||||
|
|
||||||
type RestoredAppState = Omit<
|
type RestoredAppState = Omit<
|
||||||
AppState,
|
AppState,
|
||||||
|
@ -152,7 +150,7 @@ const restoreElementWithProperties = <
|
||||||
roughness: element.roughness ?? DEFAULT_ELEMENT_PROPS.roughness,
|
roughness: element.roughness ?? DEFAULT_ELEMENT_PROPS.roughness,
|
||||||
opacity:
|
opacity:
|
||||||
element.opacity == null ? DEFAULT_ELEMENT_PROPS.opacity : element.opacity,
|
element.opacity == null ? DEFAULT_ELEMENT_PROPS.opacity : element.opacity,
|
||||||
angle: element.angle || 0,
|
angle: element.angle || (0 as Radians),
|
||||||
x: extra.x ?? element.x ?? 0,
|
x: extra.x ?? element.x ?? 0,
|
||||||
y: extra.y ?? element.y ?? 0,
|
y: extra.y ?? element.y ?? 0,
|
||||||
strokeColor: element.strokeColor || DEFAULT_ELEMENT_PROPS.strokeColor,
|
strokeColor: element.strokeColor || DEFAULT_ELEMENT_PROPS.strokeColor,
|
||||||
|
@ -266,10 +264,7 @@ const restoreElement = (
|
||||||
let y = element.y;
|
let y = element.y;
|
||||||
let points = // migrate old arrow model to new one
|
let points = // migrate old arrow model to new one
|
||||||
!Array.isArray(element.points) || element.points.length < 2
|
!Array.isArray(element.points) || element.points.length < 2
|
||||||
? [
|
? [point(0, 0), point(element.width, element.height)]
|
||||||
[0, 0],
|
|
||||||
[element.width, element.height],
|
|
||||||
]
|
|
||||||
: element.points;
|
: element.points;
|
||||||
|
|
||||||
if (points[0][0] !== 0 || points[0][1] !== 0) {
|
if (points[0][0] !== 0 || points[0][1] !== 0) {
|
||||||
|
@ -293,14 +288,11 @@ const restoreElement = (
|
||||||
});
|
});
|
||||||
case "arrow": {
|
case "arrow": {
|
||||||
const { startArrowhead = null, endArrowhead = "arrow" } = element;
|
const { startArrowhead = null, endArrowhead = "arrow" } = element;
|
||||||
let x = element.x;
|
let x: number | undefined = element.x;
|
||||||
let y = element.y;
|
let y: number | undefined = element.y;
|
||||||
let points = // migrate old arrow model to new one
|
let points: readonly LocalPoint[] | undefined = // migrate old arrow model to new one
|
||||||
!Array.isArray(element.points) || element.points.length < 2
|
!Array.isArray(element.points) || element.points.length < 2
|
||||||
? [
|
? [point(0, 0), point(element.width, element.height)]
|
||||||
[0, 0],
|
|
||||||
[element.width, element.height],
|
|
||||||
]
|
|
||||||
: element.points;
|
: element.points;
|
||||||
|
|
||||||
if (points[0][0] !== 0 || points[0][1] !== 0) {
|
if (points[0][0] !== 0 || points[0][1] !== 0) {
|
||||||
|
|
|
@ -2,6 +2,7 @@ import { vi } from "vitest";
|
||||||
import type { ExcalidrawElementSkeleton } from "./transform";
|
import type { ExcalidrawElementSkeleton } from "./transform";
|
||||||
import { convertToExcalidrawElements } from "./transform";
|
import { convertToExcalidrawElements } from "./transform";
|
||||||
import type { ExcalidrawArrowElement } from "../element/types";
|
import type { ExcalidrawArrowElement } from "../element/types";
|
||||||
|
import { point } from "../../math";
|
||||||
|
|
||||||
const opts = { regenerateIds: false };
|
const opts = { regenerateIds: false };
|
||||||
|
|
||||||
|
@ -911,10 +912,7 @@ describe("Test Transform", () => {
|
||||||
x: 111.262,
|
x: 111.262,
|
||||||
y: 57,
|
y: 57,
|
||||||
strokeWidth: 2,
|
strokeWidth: 2,
|
||||||
points: [
|
points: [point(0, 0), point(272.985, 0)],
|
||||||
[0, 0],
|
|
||||||
[272.985, 0],
|
|
||||||
],
|
|
||||||
label: {
|
label: {
|
||||||
text: "How are you?",
|
text: "How are you?",
|
||||||
fontSize: 20,
|
fontSize: 20,
|
||||||
|
@ -937,7 +935,7 @@ describe("Test Transform", () => {
|
||||||
x: 77.017,
|
x: 77.017,
|
||||||
y: 79,
|
y: 79,
|
||||||
strokeWidth: 2,
|
strokeWidth: 2,
|
||||||
points: [[0, 0]],
|
points: [point(0, 0)],
|
||||||
label: {
|
label: {
|
||||||
text: "Friendship",
|
text: "Friendship",
|
||||||
fontSize: 20,
|
fontSize: 20,
|
||||||
|
|
|
@ -53,6 +53,7 @@ import { randomId } from "../random";
|
||||||
import { syncInvalidIndices } from "../fractionalIndex";
|
import { syncInvalidIndices } from "../fractionalIndex";
|
||||||
import { getLineHeight } from "../fonts";
|
import { getLineHeight } from "../fonts";
|
||||||
import { isArrowElement } from "../element/typeChecks";
|
import { isArrowElement } from "../element/typeChecks";
|
||||||
|
import { point, type LocalPoint } from "../../math";
|
||||||
|
|
||||||
export type ValidLinearElement = {
|
export type ValidLinearElement = {
|
||||||
type: "arrow" | "line";
|
type: "arrow" | "line";
|
||||||
|
@ -417,7 +418,7 @@ const bindLinearElementToElement = (
|
||||||
const endPointIndex = linearElement.points.length - 1;
|
const endPointIndex = linearElement.points.length - 1;
|
||||||
const delta = 0.5;
|
const delta = 0.5;
|
||||||
|
|
||||||
const newPoints = cloneJSON(linearElement.points) as [number, number][];
|
const newPoints = cloneJSON<readonly LocalPoint[]>(linearElement.points);
|
||||||
|
|
||||||
// left to right so shift the arrow towards right
|
// left to right so shift the arrow towards right
|
||||||
if (
|
if (
|
||||||
|
@ -535,10 +536,7 @@ export const convertToExcalidrawElements = (
|
||||||
excalidrawElement = newLinearElement({
|
excalidrawElement = newLinearElement({
|
||||||
width,
|
width,
|
||||||
height,
|
height,
|
||||||
points: [
|
points: [point(0, 0), point(width, height)],
|
||||||
[0, 0],
|
|
||||||
[width, height],
|
|
||||||
],
|
|
||||||
...element,
|
...element,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -551,10 +549,7 @@ export const convertToExcalidrawElements = (
|
||||||
width,
|
width,
|
||||||
height,
|
height,
|
||||||
endArrowhead: "arrow",
|
endArrowhead: "arrow",
|
||||||
points: [
|
points: [point(0, 0), point(width, height)],
|
||||||
[0, 0],
|
|
||||||
[width, height],
|
|
||||||
],
|
|
||||||
...element,
|
...element,
|
||||||
type: "arrow",
|
type: "arrow",
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
import * as GA from "../ga";
|
import * as GA from "../../math/ga/ga";
|
||||||
import * as GAPoint from "../gapoints";
|
import * as GAPoint from "../../math/ga/gapoints";
|
||||||
import * as GADirection from "../gadirections";
|
import * as GADirection from "../../math/ga/gadirections";
|
||||||
import * as GALine from "../galines";
|
import * as GALine from "../../math/ga/galines";
|
||||||
import * as GATransform from "../gatransforms";
|
import * as GATransform from "../../math/ga/gatransforms";
|
||||||
|
|
||||||
import type {
|
import type {
|
||||||
ExcalidrawBindableElement,
|
ExcalidrawBindableElement,
|
||||||
|
@ -10,7 +10,6 @@ import type {
|
||||||
ExcalidrawRectangleElement,
|
ExcalidrawRectangleElement,
|
||||||
ExcalidrawDiamondElement,
|
ExcalidrawDiamondElement,
|
||||||
ExcalidrawEllipseElement,
|
ExcalidrawEllipseElement,
|
||||||
ExcalidrawFreeDrawElement,
|
|
||||||
ExcalidrawImageElement,
|
ExcalidrawImageElement,
|
||||||
ExcalidrawFrameLikeElement,
|
ExcalidrawFrameLikeElement,
|
||||||
ExcalidrawIframeLikeElement,
|
ExcalidrawIframeLikeElement,
|
||||||
|
@ -26,11 +25,12 @@ import type {
|
||||||
ExcalidrawElbowArrowElement,
|
ExcalidrawElbowArrowElement,
|
||||||
FixedPoint,
|
FixedPoint,
|
||||||
SceneElementsMap,
|
SceneElementsMap,
|
||||||
|
ExcalidrawRectanguloidElement,
|
||||||
} from "./types";
|
} from "./types";
|
||||||
|
|
||||||
import type { Bounds } from "./bounds";
|
import type { Bounds } from "./bounds";
|
||||||
import { getElementAbsoluteCoords } from "./bounds";
|
import { getCenterForBounds, getElementAbsoluteCoords } from "./bounds";
|
||||||
import type { AppState, Point } from "../types";
|
import type { AppState } from "../types";
|
||||||
import { isPointOnShape } from "../../utils/collision";
|
import { isPointOnShape } from "../../utils/collision";
|
||||||
import { getElementAtPosition } from "../scene";
|
import { getElementAtPosition } from "../scene";
|
||||||
import {
|
import {
|
||||||
|
@ -51,17 +51,7 @@ import { LinearElementEditor } from "./linearElementEditor";
|
||||||
import { arrayToMap, tupleToCoors } from "../utils";
|
import { arrayToMap, tupleToCoors } from "../utils";
|
||||||
import { KEYS } from "../keys";
|
import { KEYS } from "../keys";
|
||||||
import { getBoundTextElement, handleBindTextResize } from "./textElement";
|
import { getBoundTextElement, handleBindTextResize } from "./textElement";
|
||||||
import { getElementShape } from "../shapes";
|
import { aabbForElement, getElementShape, pointInsideBounds } from "../shapes";
|
||||||
import {
|
|
||||||
aabbForElement,
|
|
||||||
clamp,
|
|
||||||
distanceSq2d,
|
|
||||||
getCenterForBounds,
|
|
||||||
getCenterForElement,
|
|
||||||
pointInsideBounds,
|
|
||||||
pointToVector,
|
|
||||||
rotatePoint,
|
|
||||||
} from "../math";
|
|
||||||
import {
|
import {
|
||||||
compareHeading,
|
compareHeading,
|
||||||
HEADING_DOWN,
|
HEADING_DOWN,
|
||||||
|
@ -72,7 +62,18 @@ import {
|
||||||
vectorToHeading,
|
vectorToHeading,
|
||||||
type Heading,
|
type Heading,
|
||||||
} from "./heading";
|
} from "./heading";
|
||||||
import { segmentIntersectRectangleElement } from "../../utils/geometry/geometry";
|
import type { LocalPoint, Radians } from "../../math";
|
||||||
|
import {
|
||||||
|
lineSegment,
|
||||||
|
point,
|
||||||
|
pointRotateRads,
|
||||||
|
type GlobalPoint,
|
||||||
|
vectorFromPoint,
|
||||||
|
pointFromPair,
|
||||||
|
pointDistanceSq,
|
||||||
|
clamp,
|
||||||
|
} from "../../math";
|
||||||
|
import { segmentIntersectRectangleElement } from "../../utils/geometry/shape";
|
||||||
|
|
||||||
export type SuggestedBinding =
|
export type SuggestedBinding =
|
||||||
| NonDeleted<ExcalidrawBindableElement>
|
| NonDeleted<ExcalidrawBindableElement>
|
||||||
|
@ -649,7 +650,7 @@ export const updateBoundElements = (
|
||||||
update,
|
update,
|
||||||
): update is NonNullable<{
|
): update is NonNullable<{
|
||||||
index: number;
|
index: number;
|
||||||
point: Point;
|
point: LocalPoint;
|
||||||
isDragging?: boolean;
|
isDragging?: boolean;
|
||||||
}> => update !== null,
|
}> => update !== null,
|
||||||
);
|
);
|
||||||
|
@ -695,14 +696,14 @@ const getSimultaneouslyUpdatedElementIds = (
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getHeadingForElbowArrowSnap = (
|
export const getHeadingForElbowArrowSnap = (
|
||||||
point: Readonly<Point>,
|
p: Readonly<GlobalPoint>,
|
||||||
otherPoint: Readonly<Point>,
|
otherPoint: Readonly<GlobalPoint>,
|
||||||
bindableElement: ExcalidrawBindableElement | undefined | null,
|
bindableElement: ExcalidrawBindableElement | undefined | null,
|
||||||
aabb: Bounds | undefined | null,
|
aabb: Bounds | undefined | null,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
origPoint: Point,
|
origPoint: GlobalPoint,
|
||||||
): Heading => {
|
): Heading => {
|
||||||
const otherPointHeading = vectorToHeading(pointToVector(otherPoint, point));
|
const otherPointHeading = vectorToHeading(vectorFromPoint(otherPoint, p));
|
||||||
|
|
||||||
if (!bindableElement || !aabb) {
|
if (!bindableElement || !aabb) {
|
||||||
return otherPointHeading;
|
return otherPointHeading;
|
||||||
|
@ -716,17 +717,23 @@ export const getHeadingForElbowArrowSnap = (
|
||||||
|
|
||||||
if (!distance) {
|
if (!distance) {
|
||||||
return vectorToHeading(
|
return vectorToHeading(
|
||||||
pointToVector(point, getCenterForElement(bindableElement)),
|
vectorFromPoint(
|
||||||
|
p,
|
||||||
|
point<GlobalPoint>(
|
||||||
|
bindableElement.x + bindableElement.width / 2,
|
||||||
|
bindableElement.y + bindableElement.height / 2,
|
||||||
|
),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const pointHeading = headingForPointFromElement(bindableElement, aabb, point);
|
const pointHeading = headingForPointFromElement(bindableElement, aabb, p);
|
||||||
|
|
||||||
return pointHeading;
|
return pointHeading;
|
||||||
};
|
};
|
||||||
|
|
||||||
const getDistanceForBinding = (
|
const getDistanceForBinding = (
|
||||||
point: Readonly<Point>,
|
point: Readonly<GlobalPoint>,
|
||||||
bindableElement: ExcalidrawBindableElement,
|
bindableElement: ExcalidrawBindableElement,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
) => {
|
) => {
|
||||||
|
@ -745,89 +752,87 @@ const getDistanceForBinding = (
|
||||||
};
|
};
|
||||||
|
|
||||||
export const bindPointToSnapToElementOutline = (
|
export const bindPointToSnapToElementOutline = (
|
||||||
point: Readonly<Point>,
|
p: Readonly<GlobalPoint>,
|
||||||
otherPoint: Readonly<Point>,
|
otherPoint: Readonly<GlobalPoint>,
|
||||||
bindableElement: ExcalidrawBindableElement | undefined,
|
bindableElement: ExcalidrawBindableElement | undefined,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): Point => {
|
): GlobalPoint => {
|
||||||
const aabb = bindableElement && aabbForElement(bindableElement);
|
const aabb = bindableElement && aabbForElement(bindableElement);
|
||||||
|
|
||||||
if (bindableElement && aabb) {
|
if (bindableElement && aabb) {
|
||||||
// TODO: Dirty hacks until tangents are properly calculated
|
// TODO: Dirty hacks until tangents are properly calculated
|
||||||
const heading = headingForPointFromElement(bindableElement, aabb, point);
|
const heading = headingForPointFromElement(bindableElement, aabb, p);
|
||||||
const intersections = [
|
const intersections = [
|
||||||
...intersectElementWithLine(
|
...(intersectElementWithLine(
|
||||||
bindableElement,
|
bindableElement,
|
||||||
[point[0], point[1] - 2 * bindableElement.height],
|
point(p[0], p[1] - 2 * bindableElement.height),
|
||||||
[point[0], point[1] + 2 * bindableElement.height],
|
point(p[0], p[1] + 2 * bindableElement.height),
|
||||||
FIXED_BINDING_DISTANCE,
|
FIXED_BINDING_DISTANCE,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
),
|
) ?? []),
|
||||||
...intersectElementWithLine(
|
...(intersectElementWithLine(
|
||||||
bindableElement,
|
bindableElement,
|
||||||
[point[0] - 2 * bindableElement.width, point[1]],
|
point(p[0] - 2 * bindableElement.width, p[1]),
|
||||||
[point[0] + 2 * bindableElement.width, point[1]],
|
point(p[0] + 2 * bindableElement.width, p[1]),
|
||||||
FIXED_BINDING_DISTANCE,
|
FIXED_BINDING_DISTANCE,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
),
|
) ?? []),
|
||||||
];
|
];
|
||||||
|
|
||||||
const isVertical =
|
const isVertical =
|
||||||
compareHeading(heading, HEADING_LEFT) ||
|
compareHeading(heading, HEADING_LEFT) ||
|
||||||
compareHeading(heading, HEADING_RIGHT);
|
compareHeading(heading, HEADING_RIGHT);
|
||||||
const dist = Math.abs(
|
const dist = Math.abs(
|
||||||
distanceToBindableElement(bindableElement, point, elementsMap),
|
distanceToBindableElement(bindableElement, p, elementsMap),
|
||||||
);
|
);
|
||||||
const isInner = isVertical
|
const isInner = isVertical
|
||||||
? dist < bindableElement.width * -0.1
|
? dist < bindableElement.width * -0.1
|
||||||
: dist < bindableElement.height * -0.1;
|
: dist < bindableElement.height * -0.1;
|
||||||
|
|
||||||
intersections.sort(
|
intersections.sort((a, b) => pointDistanceSq(a, p) - pointDistanceSq(b, p));
|
||||||
(a, b) => distanceSq2d(a, point) - distanceSq2d(b, point),
|
|
||||||
);
|
|
||||||
|
|
||||||
return isInner
|
return isInner
|
||||||
? headingToMidBindPoint(otherPoint, bindableElement, aabb)
|
? headingToMidBindPoint(otherPoint, bindableElement, aabb)
|
||||||
: intersections.filter((i) =>
|
: intersections.filter((i) =>
|
||||||
isVertical
|
isVertical
|
||||||
? Math.abs(point[1] - i[1]) < 0.1
|
? Math.abs(p[1] - i[1]) < 0.1
|
||||||
: Math.abs(point[0] - i[0]) < 0.1,
|
: Math.abs(p[0] - i[0]) < 0.1,
|
||||||
)[0] ?? point;
|
)[0] ?? point;
|
||||||
}
|
}
|
||||||
|
|
||||||
return point;
|
return p;
|
||||||
};
|
};
|
||||||
|
|
||||||
const headingToMidBindPoint = (
|
const headingToMidBindPoint = (
|
||||||
point: Point,
|
p: GlobalPoint,
|
||||||
bindableElement: ExcalidrawBindableElement,
|
bindableElement: ExcalidrawBindableElement,
|
||||||
aabb: Bounds,
|
aabb: Bounds,
|
||||||
): Point => {
|
): GlobalPoint => {
|
||||||
const center = getCenterForBounds(aabb);
|
const center = getCenterForBounds(aabb);
|
||||||
const heading = vectorToHeading(pointToVector(point, center));
|
const heading = vectorToHeading(vectorFromPoint(p, center));
|
||||||
|
|
||||||
switch (true) {
|
switch (true) {
|
||||||
case compareHeading(heading, HEADING_UP):
|
case compareHeading(heading, HEADING_UP):
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[(aabb[0] + aabb[2]) / 2 + 0.1, aabb[1]],
|
point((aabb[0] + aabb[2]) / 2 + 0.1, aabb[1]),
|
||||||
center,
|
center,
|
||||||
bindableElement.angle,
|
bindableElement.angle,
|
||||||
);
|
);
|
||||||
case compareHeading(heading, HEADING_RIGHT):
|
case compareHeading(heading, HEADING_RIGHT):
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[aabb[2], (aabb[1] + aabb[3]) / 2 + 0.1],
|
point(aabb[2], (aabb[1] + aabb[3]) / 2 + 0.1),
|
||||||
center,
|
center,
|
||||||
bindableElement.angle,
|
bindableElement.angle,
|
||||||
);
|
);
|
||||||
case compareHeading(heading, HEADING_DOWN):
|
case compareHeading(heading, HEADING_DOWN):
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[(aabb[0] + aabb[2]) / 2 - 0.1, aabb[3]],
|
point((aabb[0] + aabb[2]) / 2 - 0.1, aabb[3]),
|
||||||
center,
|
center,
|
||||||
bindableElement.angle,
|
bindableElement.angle,
|
||||||
);
|
);
|
||||||
default:
|
default:
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[aabb[0], (aabb[1] + aabb[3]) / 2 - 0.1],
|
point(aabb[0], (aabb[1] + aabb[3]) / 2 - 0.1),
|
||||||
center,
|
center,
|
||||||
bindableElement.angle,
|
bindableElement.angle,
|
||||||
);
|
);
|
||||||
|
@ -836,22 +841,25 @@ const headingToMidBindPoint = (
|
||||||
|
|
||||||
export const avoidRectangularCorner = (
|
export const avoidRectangularCorner = (
|
||||||
element: ExcalidrawBindableElement,
|
element: ExcalidrawBindableElement,
|
||||||
p: Point,
|
p: GlobalPoint,
|
||||||
): Point => {
|
): GlobalPoint => {
|
||||||
const center = getCenterForElement(element);
|
const center = point<GlobalPoint>(
|
||||||
const nonRotatedPoint = rotatePoint(p, center, -element.angle);
|
element.x + element.width / 2,
|
||||||
|
element.y + element.height / 2,
|
||||||
|
);
|
||||||
|
const nonRotatedPoint = pointRotateRads(p, center, -element.angle as Radians);
|
||||||
|
|
||||||
if (nonRotatedPoint[0] < element.x && nonRotatedPoint[1] < element.y) {
|
if (nonRotatedPoint[0] < element.x && nonRotatedPoint[1] < element.y) {
|
||||||
// Top left
|
// Top left
|
||||||
if (nonRotatedPoint[1] - element.y > -FIXED_BINDING_DISTANCE) {
|
if (nonRotatedPoint[1] - element.y > -FIXED_BINDING_DISTANCE) {
|
||||||
return rotatePoint(
|
return pointRotateRads<GlobalPoint>(
|
||||||
[element.x - FIXED_BINDING_DISTANCE, element.y],
|
point(element.x - FIXED_BINDING_DISTANCE, element.y),
|
||||||
center,
|
center,
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[element.x, element.y - FIXED_BINDING_DISTANCE],
|
point(element.x, element.y - FIXED_BINDING_DISTANCE),
|
||||||
center,
|
center,
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
|
@ -861,14 +869,14 @@ export const avoidRectangularCorner = (
|
||||||
) {
|
) {
|
||||||
// Bottom left
|
// Bottom left
|
||||||
if (nonRotatedPoint[0] - element.x > -FIXED_BINDING_DISTANCE) {
|
if (nonRotatedPoint[0] - element.x > -FIXED_BINDING_DISTANCE) {
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[element.x, element.y + element.height + FIXED_BINDING_DISTANCE],
|
point(element.x, element.y + element.height + FIXED_BINDING_DISTANCE),
|
||||||
center,
|
center,
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[element.x - FIXED_BINDING_DISTANCE, element.y + element.height],
|
point(element.x - FIXED_BINDING_DISTANCE, element.y + element.height),
|
||||||
center,
|
center,
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
|
@ -881,20 +889,20 @@ export const avoidRectangularCorner = (
|
||||||
nonRotatedPoint[0] - element.x <
|
nonRotatedPoint[0] - element.x <
|
||||||
element.width + FIXED_BINDING_DISTANCE
|
element.width + FIXED_BINDING_DISTANCE
|
||||||
) {
|
) {
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[
|
point(
|
||||||
element.x + element.width,
|
element.x + element.width,
|
||||||
element.y + element.height + FIXED_BINDING_DISTANCE,
|
element.y + element.height + FIXED_BINDING_DISTANCE,
|
||||||
],
|
),
|
||||||
center,
|
center,
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[
|
point(
|
||||||
element.x + element.width + FIXED_BINDING_DISTANCE,
|
element.x + element.width + FIXED_BINDING_DISTANCE,
|
||||||
element.y + element.height,
|
element.y + element.height,
|
||||||
],
|
),
|
||||||
center,
|
center,
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
|
@ -907,14 +915,14 @@ export const avoidRectangularCorner = (
|
||||||
nonRotatedPoint[0] - element.x <
|
nonRotatedPoint[0] - element.x <
|
||||||
element.width + FIXED_BINDING_DISTANCE
|
element.width + FIXED_BINDING_DISTANCE
|
||||||
) {
|
) {
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[element.x + element.width, element.y - FIXED_BINDING_DISTANCE],
|
point(element.x + element.width, element.y - FIXED_BINDING_DISTANCE),
|
||||||
center,
|
center,
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[element.x + element.width + FIXED_BINDING_DISTANCE, element.y],
|
point(element.x + element.width + FIXED_BINDING_DISTANCE, element.y),
|
||||||
center,
|
center,
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
|
@ -925,12 +933,12 @@ export const avoidRectangularCorner = (
|
||||||
|
|
||||||
export const snapToMid = (
|
export const snapToMid = (
|
||||||
element: ExcalidrawBindableElement,
|
element: ExcalidrawBindableElement,
|
||||||
p: Point,
|
p: GlobalPoint,
|
||||||
tolerance: number = 0.05,
|
tolerance: number = 0.05,
|
||||||
): Point => {
|
): GlobalPoint => {
|
||||||
const { x, y, width, height, angle } = element;
|
const { x, y, width, height, angle } = element;
|
||||||
const center = [x + width / 2 - 0.1, y + height / 2 - 0.1] as Point;
|
const center = point<GlobalPoint>(x + width / 2 - 0.1, y + height / 2 - 0.1);
|
||||||
const nonRotated = rotatePoint(p, center, -angle);
|
const nonRotated = pointRotateRads(p, center, -angle as Radians);
|
||||||
|
|
||||||
// snap-to-center point is adaptive to element size, but we don't want to go
|
// snap-to-center point is adaptive to element size, but we don't want to go
|
||||||
// above and below certain px distance
|
// above and below certain px distance
|
||||||
|
@ -943,22 +951,30 @@ export const snapToMid = (
|
||||||
nonRotated[1] < center[1] + verticalThrehsold
|
nonRotated[1] < center[1] + verticalThrehsold
|
||||||
) {
|
) {
|
||||||
// LEFT
|
// LEFT
|
||||||
return rotatePoint([x - FIXED_BINDING_DISTANCE, center[1]], center, angle);
|
return pointRotateRads(
|
||||||
|
point(x - FIXED_BINDING_DISTANCE, center[1]),
|
||||||
|
center,
|
||||||
|
angle,
|
||||||
|
);
|
||||||
} else if (
|
} else if (
|
||||||
nonRotated[1] <= y + height / 2 &&
|
nonRotated[1] <= y + height / 2 &&
|
||||||
nonRotated[0] > center[0] - horizontalThrehsold &&
|
nonRotated[0] > center[0] - horizontalThrehsold &&
|
||||||
nonRotated[0] < center[0] + horizontalThrehsold
|
nonRotated[0] < center[0] + horizontalThrehsold
|
||||||
) {
|
) {
|
||||||
// TOP
|
// TOP
|
||||||
return rotatePoint([center[0], y - FIXED_BINDING_DISTANCE], center, angle);
|
return pointRotateRads(
|
||||||
|
point(center[0], y - FIXED_BINDING_DISTANCE),
|
||||||
|
center,
|
||||||
|
angle,
|
||||||
|
);
|
||||||
} else if (
|
} else if (
|
||||||
nonRotated[0] >= x + width / 2 &&
|
nonRotated[0] >= x + width / 2 &&
|
||||||
nonRotated[1] > center[1] - verticalThrehsold &&
|
nonRotated[1] > center[1] - verticalThrehsold &&
|
||||||
nonRotated[1] < center[1] + verticalThrehsold
|
nonRotated[1] < center[1] + verticalThrehsold
|
||||||
) {
|
) {
|
||||||
// RIGHT
|
// RIGHT
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[x + width + FIXED_BINDING_DISTANCE, center[1]],
|
point(x + width + FIXED_BINDING_DISTANCE, center[1]),
|
||||||
center,
|
center,
|
||||||
angle,
|
angle,
|
||||||
);
|
);
|
||||||
|
@ -968,8 +984,8 @@ export const snapToMid = (
|
||||||
nonRotated[0] < center[0] + horizontalThrehsold
|
nonRotated[0] < center[0] + horizontalThrehsold
|
||||||
) {
|
) {
|
||||||
// DOWN
|
// DOWN
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[center[0], y + height + FIXED_BINDING_DISTANCE],
|
point(center[0], y + height + FIXED_BINDING_DISTANCE),
|
||||||
center,
|
center,
|
||||||
angle,
|
angle,
|
||||||
);
|
);
|
||||||
|
@ -984,7 +1000,7 @@ const updateBoundPoint = (
|
||||||
binding: PointBinding | null | undefined,
|
binding: PointBinding | null | undefined,
|
||||||
bindableElement: ExcalidrawBindableElement,
|
bindableElement: ExcalidrawBindableElement,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): Point | null => {
|
): LocalPoint | null => {
|
||||||
if (
|
if (
|
||||||
binding == null ||
|
binding == null ||
|
||||||
// We only need to update the other end if this is a 2 point line element
|
// We only need to update the other end if this is a 2 point line element
|
||||||
|
@ -1006,15 +1022,15 @@ const updateBoundPoint = (
|
||||||
startOrEnd === "startBinding" ? "start" : "end",
|
startOrEnd === "startBinding" ? "start" : "end",
|
||||||
elementsMap,
|
elementsMap,
|
||||||
).fixedPoint;
|
).fixedPoint;
|
||||||
const globalMidPoint = [
|
const globalMidPoint = point<GlobalPoint>(
|
||||||
bindableElement.x + bindableElement.width / 2,
|
bindableElement.x + bindableElement.width / 2,
|
||||||
bindableElement.y + bindableElement.height / 2,
|
bindableElement.y + bindableElement.height / 2,
|
||||||
] as Point;
|
);
|
||||||
const global = [
|
const global = point<GlobalPoint>(
|
||||||
bindableElement.x + fixedPoint[0] * bindableElement.width,
|
bindableElement.x + fixedPoint[0] * bindableElement.width,
|
||||||
bindableElement.y + fixedPoint[1] * bindableElement.height,
|
bindableElement.y + fixedPoint[1] * bindableElement.height,
|
||||||
] as Point;
|
);
|
||||||
const rotatedGlobal = rotatePoint(
|
const rotatedGlobal = pointRotateRads(
|
||||||
global,
|
global,
|
||||||
globalMidPoint,
|
globalMidPoint,
|
||||||
bindableElement.angle,
|
bindableElement.angle,
|
||||||
|
@ -1040,7 +1056,7 @@ const updateBoundPoint = (
|
||||||
elementsMap,
|
elementsMap,
|
||||||
);
|
);
|
||||||
|
|
||||||
let newEdgePoint: Point;
|
let newEdgePoint: GlobalPoint;
|
||||||
|
|
||||||
// The linear element was not originally pointing inside the bound shape,
|
// The linear element was not originally pointing inside the bound shape,
|
||||||
// we can point directly at the focus point
|
// we can point directly at the focus point
|
||||||
|
@ -1054,7 +1070,7 @@ const updateBoundPoint = (
|
||||||
binding.gap,
|
binding.gap,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
);
|
);
|
||||||
if (intersections.length === 0) {
|
if (!intersections || intersections.length === 0) {
|
||||||
// This should never happen, since focusPoint should always be
|
// This should never happen, since focusPoint should always be
|
||||||
// inside the element, but just in case, bail out
|
// inside the element, but just in case, bail out
|
||||||
newEdgePoint = focusPointAbsolute;
|
newEdgePoint = focusPointAbsolute;
|
||||||
|
@ -1101,15 +1117,15 @@ export const calculateFixedPointForElbowArrowBinding = (
|
||||||
hoveredElement,
|
hoveredElement,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
);
|
);
|
||||||
const globalMidPoint = [
|
const globalMidPoint = point(
|
||||||
bounds[0] + (bounds[2] - bounds[0]) / 2,
|
bounds[0] + (bounds[2] - bounds[0]) / 2,
|
||||||
bounds[1] + (bounds[3] - bounds[1]) / 2,
|
bounds[1] + (bounds[3] - bounds[1]) / 2,
|
||||||
] as Point;
|
);
|
||||||
const nonRotatedSnappedGlobalPoint = rotatePoint(
|
const nonRotatedSnappedGlobalPoint = pointRotateRads(
|
||||||
snappedPoint,
|
snappedPoint,
|
||||||
globalMidPoint,
|
globalMidPoint,
|
||||||
-hoveredElement.angle,
|
-hoveredElement.angle as Radians,
|
||||||
) as Point;
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
fixedPoint: normalizeFixedPoint([
|
fixedPoint: normalizeFixedPoint([
|
||||||
|
@ -1320,8 +1336,9 @@ export const bindingBorderTest = (
|
||||||
const threshold = maxBindingGap(element, element.width, element.height);
|
const threshold = maxBindingGap(element, element.width, element.height);
|
||||||
const shape = getElementShape(element, elementsMap);
|
const shape = getElementShape(element, elementsMap);
|
||||||
return (
|
return (
|
||||||
isPointOnShape([x, y], shape, threshold) ||
|
isPointOnShape(point(x, y), shape, threshold) ||
|
||||||
(fullShape === true && pointInsideBounds([x, y], aabbForElement(element)))
|
(fullShape === true &&
|
||||||
|
pointInsideBounds(point(x, y), aabbForElement(element)))
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1339,7 +1356,7 @@ export const maxBindingGap = (
|
||||||
|
|
||||||
export const distanceToBindableElement = (
|
export const distanceToBindableElement = (
|
||||||
element: ExcalidrawBindableElement,
|
element: ExcalidrawBindableElement,
|
||||||
point: Point,
|
point: GlobalPoint,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): number => {
|
): number => {
|
||||||
switch (element.type) {
|
switch (element.type) {
|
||||||
|
@ -1359,19 +1376,13 @@ export const distanceToBindableElement = (
|
||||||
};
|
};
|
||||||
|
|
||||||
const distanceToRectangle = (
|
const distanceToRectangle = (
|
||||||
element:
|
element: ExcalidrawRectanguloidElement,
|
||||||
| ExcalidrawRectangleElement
|
p: GlobalPoint,
|
||||||
| ExcalidrawTextElement
|
|
||||||
| ExcalidrawFreeDrawElement
|
|
||||||
| ExcalidrawImageElement
|
|
||||||
| ExcalidrawIframeLikeElement
|
|
||||||
| ExcalidrawFrameLikeElement,
|
|
||||||
point: Point,
|
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): number => {
|
): number => {
|
||||||
const [, pointRel, hwidth, hheight] = pointRelativeToElement(
|
const [, pointRel, hwidth, hheight] = pointRelativeToElement(
|
||||||
element,
|
element,
|
||||||
point,
|
p,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
);
|
);
|
||||||
return Math.max(
|
return Math.max(
|
||||||
|
@ -1382,7 +1393,7 @@ const distanceToRectangle = (
|
||||||
|
|
||||||
const distanceToDiamond = (
|
const distanceToDiamond = (
|
||||||
element: ExcalidrawDiamondElement,
|
element: ExcalidrawDiamondElement,
|
||||||
point: Point,
|
point: GlobalPoint,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): number => {
|
): number => {
|
||||||
const [, pointRel, hwidth, hheight] = pointRelativeToElement(
|
const [, pointRel, hwidth, hheight] = pointRelativeToElement(
|
||||||
|
@ -1396,7 +1407,7 @@ const distanceToDiamond = (
|
||||||
|
|
||||||
const distanceToEllipse = (
|
const distanceToEllipse = (
|
||||||
element: ExcalidrawEllipseElement,
|
element: ExcalidrawEllipseElement,
|
||||||
point: Point,
|
point: GlobalPoint,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): number => {
|
): number => {
|
||||||
const [pointRel, tangent] = ellipseParamsForTest(element, point, elementsMap);
|
const [pointRel, tangent] = ellipseParamsForTest(element, point, elementsMap);
|
||||||
|
@ -1405,7 +1416,7 @@ const distanceToEllipse = (
|
||||||
|
|
||||||
const ellipseParamsForTest = (
|
const ellipseParamsForTest = (
|
||||||
element: ExcalidrawEllipseElement,
|
element: ExcalidrawEllipseElement,
|
||||||
point: Point,
|
point: GlobalPoint,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): [GA.Point, GA.Line] => {
|
): [GA.Point, GA.Line] => {
|
||||||
const [, pointRel, hwidth, hheight] = pointRelativeToElement(
|
const [, pointRel, hwidth, hheight] = pointRelativeToElement(
|
||||||
|
@ -1467,7 +1478,7 @@ const ellipseParamsForTest = (
|
||||||
// so we only need to perform hit tests for the positive quadrant.
|
// so we only need to perform hit tests for the positive quadrant.
|
||||||
const pointRelativeToElement = (
|
const pointRelativeToElement = (
|
||||||
element: ExcalidrawElement,
|
element: ExcalidrawElement,
|
||||||
pointTuple: Point,
|
pointTuple: GlobalPoint,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): [GA.Point, GA.Point, number, number] => {
|
): [GA.Point, GA.Point, number, number] => {
|
||||||
const point = GAPoint.from(pointTuple);
|
const point = GAPoint.from(pointTuple);
|
||||||
|
@ -1516,9 +1527,9 @@ const coordsCenter = (
|
||||||
const determineFocusDistance = (
|
const determineFocusDistance = (
|
||||||
element: ExcalidrawBindableElement,
|
element: ExcalidrawBindableElement,
|
||||||
// Point on the line, in absolute coordinates
|
// Point on the line, in absolute coordinates
|
||||||
a: Point,
|
a: GlobalPoint,
|
||||||
// Another point on the line, in absolute coordinates (closer to element)
|
// Another point on the line, in absolute coordinates (closer to element)
|
||||||
b: Point,
|
b: GlobalPoint,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): number => {
|
): number => {
|
||||||
const relateToCenter = relativizationToElementCenter(element, elementsMap);
|
const relateToCenter = relativizationToElementCenter(element, elementsMap);
|
||||||
|
@ -1559,13 +1570,13 @@ const determineFocusPoint = (
|
||||||
// The oriented, relative distance from the center of `element` of the
|
// The oriented, relative distance from the center of `element` of the
|
||||||
// returned focusPoint
|
// returned focusPoint
|
||||||
focus: number,
|
focus: number,
|
||||||
adjecentPoint: Point,
|
adjecentPoint: GlobalPoint,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): Point => {
|
): GlobalPoint => {
|
||||||
if (focus === 0) {
|
if (focus === 0) {
|
||||||
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
||||||
const center = coordsCenter(x1, y1, x2, y2);
|
const center = coordsCenter(x1, y1, x2, y2);
|
||||||
return GAPoint.toTuple(center);
|
return pointFromPair(GAPoint.toTuple(center));
|
||||||
}
|
}
|
||||||
const relateToCenter = relativizationToElementCenter(element, elementsMap);
|
const relateToCenter = relativizationToElementCenter(element, elementsMap);
|
||||||
const adjecentPointRel = GATransform.apply(
|
const adjecentPointRel = GATransform.apply(
|
||||||
|
@ -1589,7 +1600,9 @@ const determineFocusPoint = (
|
||||||
point = findFocusPointForEllipse(element, focus, adjecentPointRel);
|
point = findFocusPointForEllipse(element, focus, adjecentPointRel);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
return GAPoint.toTuple(GATransform.apply(reverseRelateToCenter, point));
|
return pointFromPair(
|
||||||
|
GAPoint.toTuple(GATransform.apply(reverseRelateToCenter, point)),
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
// Returns 2 or 0 intersection points between line going through `a` and `b`
|
// Returns 2 or 0 intersection points between line going through `a` and `b`
|
||||||
|
@ -1597,15 +1610,15 @@ const determineFocusPoint = (
|
||||||
const intersectElementWithLine = (
|
const intersectElementWithLine = (
|
||||||
element: ExcalidrawBindableElement,
|
element: ExcalidrawBindableElement,
|
||||||
// Point on the line, in absolute coordinates
|
// Point on the line, in absolute coordinates
|
||||||
a: Point,
|
a: GlobalPoint,
|
||||||
// Another point on the line, in absolute coordinates
|
// Another point on the line, in absolute coordinates
|
||||||
b: Point,
|
b: GlobalPoint,
|
||||||
// If given, the element is inflated by this value
|
// If given, the element is inflated by this value
|
||||||
gap: number = 0,
|
gap: number = 0,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): Point[] => {
|
): GlobalPoint[] | undefined => {
|
||||||
if (isRectangularElement(element)) {
|
if (isRectangularElement(element)) {
|
||||||
return segmentIntersectRectangleElement(element, [a, b], gap);
|
return segmentIntersectRectangleElement(element, lineSegment(a, b), gap);
|
||||||
}
|
}
|
||||||
|
|
||||||
const relateToCenter = relativizationToElementCenter(element, elementsMap);
|
const relateToCenter = relativizationToElementCenter(element, elementsMap);
|
||||||
|
@ -1619,8 +1632,14 @@ const intersectElementWithLine = (
|
||||||
aRel,
|
aRel,
|
||||||
gap,
|
gap,
|
||||||
);
|
);
|
||||||
return intersections.map((point) =>
|
return intersections.map(
|
||||||
|
(point) =>
|
||||||
|
pointFromPair(
|
||||||
GAPoint.toTuple(GATransform.apply(reverseRelateToCenter, point)),
|
GAPoint.toTuple(GATransform.apply(reverseRelateToCenter, point)),
|
||||||
|
),
|
||||||
|
// pointFromArray(
|
||||||
|
// ,
|
||||||
|
// ),
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -2173,12 +2192,18 @@ export class BindableElement {
|
||||||
export const getGlobalFixedPointForBindableElement = (
|
export const getGlobalFixedPointForBindableElement = (
|
||||||
fixedPointRatio: [number, number],
|
fixedPointRatio: [number, number],
|
||||||
element: ExcalidrawBindableElement,
|
element: ExcalidrawBindableElement,
|
||||||
) => {
|
): GlobalPoint => {
|
||||||
const [fixedX, fixedY] = normalizeFixedPoint(fixedPointRatio);
|
const [fixedX, fixedY] = normalizeFixedPoint(fixedPointRatio);
|
||||||
|
|
||||||
return rotatePoint(
|
return pointRotateRads(
|
||||||
[element.x + element.width * fixedX, element.y + element.height * fixedY],
|
point(
|
||||||
getCenterForElement(element),
|
element.x + element.width * fixedX,
|
||||||
|
element.y + element.height * fixedY,
|
||||||
|
),
|
||||||
|
point<GlobalPoint>(
|
||||||
|
element.x + element.width / 2,
|
||||||
|
element.y + element.height / 2,
|
||||||
|
),
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -2186,7 +2211,7 @@ export const getGlobalFixedPointForBindableElement = (
|
||||||
const getGlobalFixedPoints = (
|
const getGlobalFixedPoints = (
|
||||||
arrow: ExcalidrawElbowArrowElement,
|
arrow: ExcalidrawElbowArrowElement,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
) => {
|
): [GlobalPoint, GlobalPoint] => {
|
||||||
const startElement =
|
const startElement =
|
||||||
arrow.startBinding &&
|
arrow.startBinding &&
|
||||||
(elementsMap.get(arrow.startBinding.elementId) as
|
(elementsMap.get(arrow.startBinding.elementId) as
|
||||||
|
@ -2197,23 +2222,26 @@ const getGlobalFixedPoints = (
|
||||||
(elementsMap.get(arrow.endBinding.elementId) as
|
(elementsMap.get(arrow.endBinding.elementId) as
|
||||||
| ExcalidrawBindableElement
|
| ExcalidrawBindableElement
|
||||||
| undefined);
|
| undefined);
|
||||||
const startPoint: Point =
|
const startPoint =
|
||||||
startElement && arrow.startBinding
|
startElement && arrow.startBinding
|
||||||
? getGlobalFixedPointForBindableElement(
|
? getGlobalFixedPointForBindableElement(
|
||||||
arrow.startBinding.fixedPoint,
|
arrow.startBinding.fixedPoint,
|
||||||
startElement as ExcalidrawBindableElement,
|
startElement as ExcalidrawBindableElement,
|
||||||
)
|
)
|
||||||
: [arrow.x + arrow.points[0][0], arrow.y + arrow.points[0][1]];
|
: point<GlobalPoint>(
|
||||||
const endPoint: Point =
|
arrow.x + arrow.points[0][0],
|
||||||
|
arrow.y + arrow.points[0][1],
|
||||||
|
);
|
||||||
|
const endPoint =
|
||||||
endElement && arrow.endBinding
|
endElement && arrow.endBinding
|
||||||
? getGlobalFixedPointForBindableElement(
|
? getGlobalFixedPointForBindableElement(
|
||||||
arrow.endBinding.fixedPoint,
|
arrow.endBinding.fixedPoint,
|
||||||
endElement as ExcalidrawBindableElement,
|
endElement as ExcalidrawBindableElement,
|
||||||
)
|
)
|
||||||
: [
|
: point<GlobalPoint>(
|
||||||
arrow.x + arrow.points[arrow.points.length - 1][0],
|
arrow.x + arrow.points[arrow.points.length - 1][0],
|
||||||
arrow.y + arrow.points[arrow.points.length - 1][1],
|
arrow.y + arrow.points[arrow.points.length - 1][1],
|
||||||
];
|
);
|
||||||
|
|
||||||
return [startPoint, endPoint];
|
return [startPoint, endPoint];
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
import type { LocalPoint } from "../../math";
|
||||||
|
import { point } from "../../math";
|
||||||
import { ROUNDNESS } from "../constants";
|
import { ROUNDNESS } from "../constants";
|
||||||
import { arrayToMap } from "../utils";
|
import { arrayToMap } from "../utils";
|
||||||
import { getElementAbsoluteCoords, getElementBounds } from "./bounds";
|
import { getElementAbsoluteCoords, getElementBounds } from "./bounds";
|
||||||
|
@ -123,9 +125,9 @@ describe("getElementBounds", () => {
|
||||||
a: 0.6447741904932416,
|
a: 0.6447741904932416,
|
||||||
}),
|
}),
|
||||||
points: [
|
points: [
|
||||||
[0, 0] as [number, number],
|
point<LocalPoint>(0, 0),
|
||||||
[67.33984375, 92.48828125] as [number, number],
|
point<LocalPoint>(67.33984375, 92.48828125),
|
||||||
[-102.7890625, 52.15625] as [number, number],
|
point<LocalPoint>(-102.7890625, 52.15625),
|
||||||
],
|
],
|
||||||
} as ExcalidrawLinearElement;
|
} as ExcalidrawLinearElement;
|
||||||
|
|
||||||
|
|
|
@ -7,10 +7,10 @@ import type {
|
||||||
ExcalidrawTextElementWithContainer,
|
ExcalidrawTextElementWithContainer,
|
||||||
ElementsMap,
|
ElementsMap,
|
||||||
} from "./types";
|
} from "./types";
|
||||||
import { distance2d, rotate, rotatePoint } from "../math";
|
|
||||||
import rough from "roughjs/bin/rough";
|
import rough from "roughjs/bin/rough";
|
||||||
|
import type { Point as RoughPoint } from "roughjs/bin/geometry";
|
||||||
import type { Drawable, Op } from "roughjs/bin/core";
|
import type { Drawable, Op } from "roughjs/bin/core";
|
||||||
import type { AppState, Point } from "../types";
|
import type { AppState } from "../types";
|
||||||
import { generateRoughOptions } from "../scene/Shape";
|
import { generateRoughOptions } from "../scene/Shape";
|
||||||
import {
|
import {
|
||||||
isArrowElement,
|
isArrowElement,
|
||||||
|
@ -22,9 +22,24 @@ import {
|
||||||
import { rescalePoints } from "../points";
|
import { rescalePoints } from "../points";
|
||||||
import { getBoundTextElement, getContainerElement } from "./textElement";
|
import { getBoundTextElement, getContainerElement } from "./textElement";
|
||||||
import { LinearElementEditor } from "./linearElementEditor";
|
import { LinearElementEditor } from "./linearElementEditor";
|
||||||
import type { Mutable } from "../utility-types";
|
|
||||||
import { ShapeCache } from "../scene/ShapeCache";
|
import { ShapeCache } from "../scene/ShapeCache";
|
||||||
import { arrayToMap } from "../utils";
|
import { arrayToMap, invariant } from "../utils";
|
||||||
|
import type {
|
||||||
|
Degrees,
|
||||||
|
GlobalPoint,
|
||||||
|
LineSegment,
|
||||||
|
LocalPoint,
|
||||||
|
Radians,
|
||||||
|
} from "../../math";
|
||||||
|
import {
|
||||||
|
degreesToRadians,
|
||||||
|
lineSegment,
|
||||||
|
point,
|
||||||
|
pointDistance,
|
||||||
|
pointFromArray,
|
||||||
|
pointRotateRads,
|
||||||
|
} from "../../math";
|
||||||
|
import type { Mutable } from "../utility-types";
|
||||||
|
|
||||||
export type RectangleBox = {
|
export type RectangleBox = {
|
||||||
x: number;
|
x: number;
|
||||||
|
@ -97,7 +112,11 @@ export class ElementBounds {
|
||||||
if (isFreeDrawElement(element)) {
|
if (isFreeDrawElement(element)) {
|
||||||
const [minX, minY, maxX, maxY] = getBoundsFromPoints(
|
const [minX, minY, maxX, maxY] = getBoundsFromPoints(
|
||||||
element.points.map(([x, y]) =>
|
element.points.map(([x, y]) =>
|
||||||
rotate(x, y, cx - element.x, cy - element.y, element.angle),
|
pointRotateRads(
|
||||||
|
point(x, y),
|
||||||
|
point(cx - element.x, cy - element.y),
|
||||||
|
element.angle,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -110,10 +129,26 @@ export class ElementBounds {
|
||||||
} else if (isLinearElement(element)) {
|
} else if (isLinearElement(element)) {
|
||||||
bounds = getLinearElementRotatedBounds(element, cx, cy, elementsMap);
|
bounds = getLinearElementRotatedBounds(element, cx, cy, elementsMap);
|
||||||
} else if (element.type === "diamond") {
|
} else if (element.type === "diamond") {
|
||||||
const [x11, y11] = rotate(cx, y1, cx, cy, element.angle);
|
const [x11, y11] = pointRotateRads(
|
||||||
const [x12, y12] = rotate(cx, y2, cx, cy, element.angle);
|
point(cx, y1),
|
||||||
const [x22, y22] = rotate(x1, cy, cx, cy, element.angle);
|
point(cx, cy),
|
||||||
const [x21, y21] = rotate(x2, cy, cx, cy, element.angle);
|
element.angle,
|
||||||
|
);
|
||||||
|
const [x12, y12] = pointRotateRads(
|
||||||
|
point(cx, y2),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
|
const [x22, y22] = pointRotateRads(
|
||||||
|
point(x1, cy),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
|
const [x21, y21] = pointRotateRads(
|
||||||
|
point(x2, cy),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
const minX = Math.min(x11, x12, x22, x21);
|
const minX = Math.min(x11, x12, x22, x21);
|
||||||
const minY = Math.min(y11, y12, y22, y21);
|
const minY = Math.min(y11, y12, y22, y21);
|
||||||
const maxX = Math.max(x11, x12, x22, x21);
|
const maxX = Math.max(x11, x12, x22, x21);
|
||||||
|
@ -128,10 +163,26 @@ export class ElementBounds {
|
||||||
const hh = Math.hypot(h * cos, w * sin);
|
const hh = Math.hypot(h * cos, w * sin);
|
||||||
bounds = [cx - ww, cy - hh, cx + ww, cy + hh];
|
bounds = [cx - ww, cy - hh, cx + ww, cy + hh];
|
||||||
} else {
|
} else {
|
||||||
const [x11, y11] = rotate(x1, y1, cx, cy, element.angle);
|
const [x11, y11] = pointRotateRads(
|
||||||
const [x12, y12] = rotate(x1, y2, cx, cy, element.angle);
|
point(x1, y1),
|
||||||
const [x22, y22] = rotate(x2, y2, cx, cy, element.angle);
|
point(cx, cy),
|
||||||
const [x21, y21] = rotate(x2, y1, cx, cy, element.angle);
|
element.angle,
|
||||||
|
);
|
||||||
|
const [x12, y12] = pointRotateRads(
|
||||||
|
point(x1, y2),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
|
const [x22, y22] = pointRotateRads(
|
||||||
|
point(x2, y2),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
|
const [x21, y21] = pointRotateRads(
|
||||||
|
point(x2, y1),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
const minX = Math.min(x11, x12, x22, x21);
|
const minX = Math.min(x11, x12, x22, x21);
|
||||||
const minY = Math.min(y11, y12, y22, y21);
|
const minY = Math.min(y11, y12, y22, y21);
|
||||||
const maxX = Math.max(x11, x12, x22, x21);
|
const maxX = Math.max(x11, x12, x22, x21);
|
||||||
|
@ -165,18 +216,18 @@ export const getElementAbsoluteCoords = (
|
||||||
? getContainerElement(element, elementsMap)
|
? getContainerElement(element, elementsMap)
|
||||||
: null;
|
: null;
|
||||||
if (isArrowElement(container)) {
|
if (isArrowElement(container)) {
|
||||||
const coords = LinearElementEditor.getBoundTextElementPosition(
|
const { x, y } = LinearElementEditor.getBoundTextElementPosition(
|
||||||
container,
|
container,
|
||||||
element as ExcalidrawTextElementWithContainer,
|
element as ExcalidrawTextElementWithContainer,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
);
|
);
|
||||||
return [
|
return [
|
||||||
coords.x,
|
x,
|
||||||
coords.y,
|
y,
|
||||||
coords.x + element.width,
|
x + element.width,
|
||||||
coords.y + element.height,
|
y + element.height,
|
||||||
coords.x + element.width / 2,
|
x + element.width / 2,
|
||||||
coords.y + element.height / 2,
|
y + element.height / 2,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -198,38 +249,40 @@ export const getElementAbsoluteCoords = (
|
||||||
export const getElementLineSegments = (
|
export const getElementLineSegments = (
|
||||||
element: ExcalidrawElement,
|
element: ExcalidrawElement,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): [Point, Point][] => {
|
): LineSegment<GlobalPoint>[] => {
|
||||||
const [x1, y1, x2, y2, cx, cy] = getElementAbsoluteCoords(
|
const [x1, y1, x2, y2, cx, cy] = getElementAbsoluteCoords(
|
||||||
element,
|
element,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
);
|
);
|
||||||
|
|
||||||
const center: Point = [cx, cy];
|
const center: GlobalPoint = point(cx, cy);
|
||||||
|
|
||||||
if (isLinearElement(element) || isFreeDrawElement(element)) {
|
if (isLinearElement(element) || isFreeDrawElement(element)) {
|
||||||
const segments: [Point, Point][] = [];
|
const segments: LineSegment<GlobalPoint>[] = [];
|
||||||
|
|
||||||
let i = 0;
|
let i = 0;
|
||||||
|
|
||||||
while (i < element.points.length - 1) {
|
while (i < element.points.length - 1) {
|
||||||
segments.push([
|
segments.push(
|
||||||
rotatePoint(
|
lineSegment(
|
||||||
[
|
pointRotateRads(
|
||||||
|
point(
|
||||||
element.points[i][0] + element.x,
|
element.points[i][0] + element.x,
|
||||||
element.points[i][1] + element.y,
|
element.points[i][1] + element.y,
|
||||||
] as Point,
|
),
|
||||||
center,
|
center,
|
||||||
element.angle,
|
element.angle,
|
||||||
),
|
),
|
||||||
rotatePoint(
|
pointRotateRads(
|
||||||
[
|
point(
|
||||||
element.points[i + 1][0] + element.x,
|
element.points[i + 1][0] + element.x,
|
||||||
element.points[i + 1][1] + element.y,
|
element.points[i + 1][1] + element.y,
|
||||||
] as Point,
|
),
|
||||||
center,
|
center,
|
||||||
element.angle,
|
element.angle,
|
||||||
),
|
),
|
||||||
]);
|
),
|
||||||
|
);
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -246,40 +299,40 @@ export const getElementLineSegments = (
|
||||||
[cx, y2],
|
[cx, y2],
|
||||||
[x1, cy],
|
[x1, cy],
|
||||||
[x2, cy],
|
[x2, cy],
|
||||||
] as Point[]
|
] as GlobalPoint[]
|
||||||
).map((point) => rotatePoint(point, center, element.angle));
|
).map((point) => pointRotateRads(point, center, element.angle));
|
||||||
|
|
||||||
if (element.type === "diamond") {
|
if (element.type === "diamond") {
|
||||||
return [
|
return [
|
||||||
[n, w],
|
lineSegment(n, w),
|
||||||
[n, e],
|
lineSegment(n, e),
|
||||||
[s, w],
|
lineSegment(s, w),
|
||||||
[s, e],
|
lineSegment(s, e),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (element.type === "ellipse") {
|
if (element.type === "ellipse") {
|
||||||
return [
|
return [
|
||||||
[n, w],
|
lineSegment(n, w),
|
||||||
[n, e],
|
lineSegment(n, e),
|
||||||
[s, w],
|
lineSegment(s, w),
|
||||||
[s, e],
|
lineSegment(s, e),
|
||||||
[n, w],
|
lineSegment(n, w),
|
||||||
[n, e],
|
lineSegment(n, e),
|
||||||
[s, w],
|
lineSegment(s, w),
|
||||||
[s, e],
|
lineSegment(s, e),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
return [
|
return [
|
||||||
[nw, ne],
|
lineSegment(nw, ne),
|
||||||
[sw, se],
|
lineSegment(sw, se),
|
||||||
[nw, sw],
|
lineSegment(nw, sw),
|
||||||
[ne, se],
|
lineSegment(ne, se),
|
||||||
[nw, e],
|
lineSegment(nw, e),
|
||||||
[sw, e],
|
lineSegment(sw, e),
|
||||||
[ne, w],
|
lineSegment(ne, w),
|
||||||
[se, w],
|
lineSegment(se, w),
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -386,10 +439,10 @@ const solveQuadratic = (
|
||||||
};
|
};
|
||||||
|
|
||||||
const getCubicBezierCurveBound = (
|
const getCubicBezierCurveBound = (
|
||||||
p0: Point,
|
p0: GlobalPoint,
|
||||||
p1: Point,
|
p1: GlobalPoint,
|
||||||
p2: Point,
|
p2: GlobalPoint,
|
||||||
p3: Point,
|
p3: GlobalPoint,
|
||||||
): Bounds => {
|
): Bounds => {
|
||||||
const solX = solveQuadratic(p0[0], p1[0], p2[0], p3[0]);
|
const solX = solveQuadratic(p0[0], p1[0], p2[0], p3[0]);
|
||||||
const solY = solveQuadratic(p0[1], p1[1], p2[1], p3[1]);
|
const solY = solveQuadratic(p0[1], p1[1], p2[1], p3[1]);
|
||||||
|
@ -415,9 +468,9 @@ const getCubicBezierCurveBound = (
|
||||||
|
|
||||||
export const getMinMaxXYFromCurvePathOps = (
|
export const getMinMaxXYFromCurvePathOps = (
|
||||||
ops: Op[],
|
ops: Op[],
|
||||||
transformXY?: (x: number, y: number) => [number, number],
|
transformXY?: (p: GlobalPoint) => GlobalPoint,
|
||||||
): Bounds => {
|
): Bounds => {
|
||||||
let currentP: Point = [0, 0];
|
let currentP: GlobalPoint = point(0, 0);
|
||||||
|
|
||||||
const { minX, minY, maxX, maxY } = ops.reduce(
|
const { minX, minY, maxX, maxY } = ops.reduce(
|
||||||
(limits, { op, data }) => {
|
(limits, { op, data }) => {
|
||||||
|
@ -425,19 +478,21 @@ export const getMinMaxXYFromCurvePathOps = (
|
||||||
// move, bcurveTo, lineTo, and curveTo
|
// move, bcurveTo, lineTo, and curveTo
|
||||||
if (op === "move") {
|
if (op === "move") {
|
||||||
// change starting point
|
// change starting point
|
||||||
currentP = data as unknown as Point;
|
const p: GlobalPoint | undefined = pointFromArray(data);
|
||||||
|
invariant(p != null, "Op data is not a point");
|
||||||
|
currentP = p;
|
||||||
// move operation does not draw anything; so, it always
|
// move operation does not draw anything; so, it always
|
||||||
// returns false
|
// returns false
|
||||||
} else if (op === "bcurveTo") {
|
} else if (op === "bcurveTo") {
|
||||||
const _p1 = [data[0], data[1]] as Point;
|
const _p1 = point<GlobalPoint>(data[0], data[1]);
|
||||||
const _p2 = [data[2], data[3]] as Point;
|
const _p2 = point<GlobalPoint>(data[2], data[3]);
|
||||||
const _p3 = [data[4], data[5]] as Point;
|
const _p3 = point<GlobalPoint>(data[4], data[5]);
|
||||||
|
|
||||||
const p1 = transformXY ? transformXY(..._p1) : _p1;
|
const p1 = transformXY ? transformXY(_p1) : _p1;
|
||||||
const p2 = transformXY ? transformXY(..._p2) : _p2;
|
const p2 = transformXY ? transformXY(_p2) : _p2;
|
||||||
const p3 = transformXY ? transformXY(..._p3) : _p3;
|
const p3 = transformXY ? transformXY(_p3) : _p3;
|
||||||
|
|
||||||
const p0 = transformXY ? transformXY(...currentP) : currentP;
|
const p0 = transformXY ? transformXY(currentP) : currentP;
|
||||||
currentP = _p3;
|
currentP = _p3;
|
||||||
|
|
||||||
const [minX, minY, maxX, maxY] = getCubicBezierCurveBound(
|
const [minX, minY, maxX, maxY] = getCubicBezierCurveBound(
|
||||||
|
@ -507,14 +562,14 @@ export const getArrowheadSize = (arrowhead: Arrowhead): number => {
|
||||||
};
|
};
|
||||||
|
|
||||||
/** @returns number in degrees */
|
/** @returns number in degrees */
|
||||||
export const getArrowheadAngle = (arrowhead: Arrowhead): number => {
|
export const getArrowheadAngle = (arrowhead: Arrowhead): Degrees => {
|
||||||
switch (arrowhead) {
|
switch (arrowhead) {
|
||||||
case "bar":
|
case "bar":
|
||||||
return 90;
|
return 90 as Degrees;
|
||||||
case "arrow":
|
case "arrow":
|
||||||
return 20;
|
return 20 as Degrees;
|
||||||
default:
|
default:
|
||||||
return 25;
|
return 25 as Degrees;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -533,19 +588,24 @@ export const getArrowheadPoints = (
|
||||||
const index = position === "start" ? 1 : ops.length - 1;
|
const index = position === "start" ? 1 : ops.length - 1;
|
||||||
|
|
||||||
const data = ops[index].data;
|
const data = ops[index].data;
|
||||||
const p3 = [data[4], data[5]] as Point;
|
|
||||||
const p2 = [data[2], data[3]] as Point;
|
invariant(data.length === 6, "Op data length is not 6");
|
||||||
const p1 = [data[0], data[1]] as Point;
|
|
||||||
|
const p3 = point(data[4], data[5]);
|
||||||
|
const p2 = point(data[2], data[3]);
|
||||||
|
const p1 = point(data[0], data[1]);
|
||||||
|
|
||||||
// We need to find p0 of the bezier curve.
|
// We need to find p0 of the bezier curve.
|
||||||
// It is typically the last point of the previous
|
// It is typically the last point of the previous
|
||||||
// curve; it can also be the position of moveTo operation.
|
// curve; it can also be the position of moveTo operation.
|
||||||
const prevOp = ops[index - 1];
|
const prevOp = ops[index - 1];
|
||||||
let p0: Point = [0, 0];
|
let p0 = point(0, 0);
|
||||||
if (prevOp.op === "move") {
|
if (prevOp.op === "move") {
|
||||||
p0 = prevOp.data as unknown as Point;
|
const p = pointFromArray(prevOp.data);
|
||||||
|
invariant(p != null, "Op data is not a point");
|
||||||
|
p0 = p;
|
||||||
} else if (prevOp.op === "bcurveTo") {
|
} else if (prevOp.op === "bcurveTo") {
|
||||||
p0 = [prevOp.data[4], prevOp.data[5]];
|
p0 = point(prevOp.data[4], prevOp.data[5]);
|
||||||
}
|
}
|
||||||
|
|
||||||
// B(t) = p0 * (1-t)^3 + 3p1 * t * (1-t)^2 + 3p2 * t^2 * (1-t) + p3 * t^3
|
// B(t) = p0 * (1-t)^3 + 3p1 * t * (1-t)^2 + 3p2 * t^2 * (1-t) + p3 * t^3
|
||||||
|
@ -610,8 +670,16 @@ export const getArrowheadPoints = (
|
||||||
const angle = getArrowheadAngle(arrowhead);
|
const angle = getArrowheadAngle(arrowhead);
|
||||||
|
|
||||||
// Return points
|
// Return points
|
||||||
const [x3, y3] = rotate(xs, ys, x2, y2, (-angle * Math.PI) / 180);
|
const [x3, y3] = pointRotateRads(
|
||||||
const [x4, y4] = rotate(xs, ys, x2, y2, (angle * Math.PI) / 180);
|
point(xs, ys),
|
||||||
|
point(x2, y2),
|
||||||
|
((-angle * Math.PI) / 180) as Radians,
|
||||||
|
);
|
||||||
|
const [x4, y4] = pointRotateRads(
|
||||||
|
point(xs, ys),
|
||||||
|
point(x2, y2),
|
||||||
|
degreesToRadians(angle),
|
||||||
|
);
|
||||||
|
|
||||||
if (arrowhead === "diamond" || arrowhead === "diamond_outline") {
|
if (arrowhead === "diamond" || arrowhead === "diamond_outline") {
|
||||||
// point opposite to the arrowhead point
|
// point opposite to the arrowhead point
|
||||||
|
@ -621,12 +689,10 @@ export const getArrowheadPoints = (
|
||||||
if (position === "start") {
|
if (position === "start") {
|
||||||
const [px, py] = element.points.length > 1 ? element.points[1] : [0, 0];
|
const [px, py] = element.points.length > 1 ? element.points[1] : [0, 0];
|
||||||
|
|
||||||
[ox, oy] = rotate(
|
[ox, oy] = pointRotateRads(
|
||||||
x2 + minSize * 2,
|
point(x2 + minSize * 2, y2),
|
||||||
y2,
|
point(x2, y2),
|
||||||
x2,
|
Math.atan2(py - y2, px - x2) as Radians,
|
||||||
y2,
|
|
||||||
Math.atan2(py - y2, px - x2),
|
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
const [px, py] =
|
const [px, py] =
|
||||||
|
@ -634,12 +700,10 @@ export const getArrowheadPoints = (
|
||||||
? element.points[element.points.length - 2]
|
? element.points[element.points.length - 2]
|
||||||
: [0, 0];
|
: [0, 0];
|
||||||
|
|
||||||
[ox, oy] = rotate(
|
[ox, oy] = pointRotateRads(
|
||||||
x2 - minSize * 2,
|
point(x2 - minSize * 2, y2),
|
||||||
y2,
|
point(x2, y2),
|
||||||
x2,
|
Math.atan2(y2 - py, x2 - px) as Radians,
|
||||||
y2,
|
|
||||||
Math.atan2(y2 - py, x2 - px),
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -665,7 +729,10 @@ const generateLinearElementShape = (
|
||||||
return "linearPath";
|
return "linearPath";
|
||||||
})();
|
})();
|
||||||
|
|
||||||
return generator[method](element.points as Mutable<Point>[], options);
|
return generator[method](
|
||||||
|
element.points as Mutable<LocalPoint>[] as RoughPoint[],
|
||||||
|
options,
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const getLinearElementRotatedBounds = (
|
const getLinearElementRotatedBounds = (
|
||||||
|
@ -678,11 +745,9 @@ const getLinearElementRotatedBounds = (
|
||||||
|
|
||||||
if (element.points.length < 2) {
|
if (element.points.length < 2) {
|
||||||
const [pointX, pointY] = element.points[0];
|
const [pointX, pointY] = element.points[0];
|
||||||
const [x, y] = rotate(
|
const [x, y] = pointRotateRads(
|
||||||
element.x + pointX,
|
point(element.x + pointX, element.y + pointY),
|
||||||
element.y + pointY,
|
point(cx, cy),
|
||||||
cx,
|
|
||||||
cy,
|
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -708,8 +773,12 @@ const getLinearElementRotatedBounds = (
|
||||||
const cachedShape = ShapeCache.get(element)?.[0];
|
const cachedShape = ShapeCache.get(element)?.[0];
|
||||||
const shape = cachedShape ?? generateLinearElementShape(element);
|
const shape = cachedShape ?? generateLinearElementShape(element);
|
||||||
const ops = getCurvePathOps(shape);
|
const ops = getCurvePathOps(shape);
|
||||||
const transformXY = (x: number, y: number) =>
|
const transformXY = ([x, y]: GlobalPoint) =>
|
||||||
rotate(element.x + x, element.y + y, cx, cy, element.angle);
|
pointRotateRads<GlobalPoint>(
|
||||||
|
point(element.x + x, element.y + y),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
const res = getMinMaxXYFromCurvePathOps(ops, transformXY);
|
const res = getMinMaxXYFromCurvePathOps(ops, transformXY);
|
||||||
let coords: Bounds = [res[0], res[1], res[2], res[3]];
|
let coords: Bounds = [res[0], res[1], res[2], res[3]];
|
||||||
if (boundTextElement) {
|
if (boundTextElement) {
|
||||||
|
@ -861,7 +930,10 @@ export const getClosestElementBounds = (
|
||||||
const elementsMap = arrayToMap(elements);
|
const elementsMap = arrayToMap(elements);
|
||||||
elements.forEach((element) => {
|
elements.forEach((element) => {
|
||||||
const [x1, y1, x2, y2] = getElementBounds(element, elementsMap);
|
const [x1, y1, x2, y2] = getElementBounds(element, elementsMap);
|
||||||
const distance = distance2d((x1 + x2) / 2, (y1 + y2) / 2, from.x, from.y);
|
const distance = pointDistance(
|
||||||
|
point((x1 + x2) / 2, (y1 + y2) / 2),
|
||||||
|
point(from.x, from.y),
|
||||||
|
);
|
||||||
|
|
||||||
if (distance < minDistance) {
|
if (distance < minDistance) {
|
||||||
minDistance = distance;
|
minDistance = distance;
|
||||||
|
@ -916,3 +988,9 @@ export const getVisibleSceneBounds = ({
|
||||||
-scrollY + height / zoom.value,
|
-scrollY + height / zoom.value,
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const getCenterForBounds = (bounds: Bounds): GlobalPoint =>
|
||||||
|
point(
|
||||||
|
bounds[0] + (bounds[2] - bounds[0]) / 2,
|
||||||
|
bounds[1] + (bounds[3] - bounds[1]) / 2,
|
||||||
|
);
|
||||||
|
|
|
@ -1,14 +1,11 @@
|
||||||
import { isPathALoop, isPointWithinBounds } from "../math";
|
|
||||||
|
|
||||||
import type {
|
import type {
|
||||||
ElementsMap,
|
ElementsMap,
|
||||||
ExcalidrawElement,
|
ExcalidrawElement,
|
||||||
ExcalidrawRectangleElement,
|
ExcalidrawRectangleElement,
|
||||||
} from "./types";
|
} from "./types";
|
||||||
|
|
||||||
import { getElementBounds } from "./bounds";
|
import { getElementBounds } from "./bounds";
|
||||||
import type { FrameNameBounds } from "../types";
|
import type { FrameNameBounds } from "../types";
|
||||||
import type { Polygon, GeometricShape } from "../../utils/geometry/shape";
|
import type { GeometricShape } from "../../utils/geometry/shape";
|
||||||
import { getPolygonShape } from "../../utils/geometry/shape";
|
import { getPolygonShape } from "../../utils/geometry/shape";
|
||||||
import { isPointInShape, isPointOnShape } from "../../utils/collision";
|
import { isPointInShape, isPointOnShape } from "../../utils/collision";
|
||||||
import { isTransparent } from "../utils";
|
import { isTransparent } from "../utils";
|
||||||
|
@ -18,7 +15,9 @@ import {
|
||||||
isImageElement,
|
isImageElement,
|
||||||
isTextElement,
|
isTextElement,
|
||||||
} from "./typeChecks";
|
} from "./typeChecks";
|
||||||
import { getBoundTextShape } from "../shapes";
|
import { getBoundTextShape, isPathALoop } from "../shapes";
|
||||||
|
import type { GlobalPoint, LocalPoint, Polygon } from "../../math";
|
||||||
|
import { isPointWithinBounds, point } from "../../math";
|
||||||
|
|
||||||
export const shouldTestInside = (element: ExcalidrawElement) => {
|
export const shouldTestInside = (element: ExcalidrawElement) => {
|
||||||
if (element.type === "arrow") {
|
if (element.type === "arrow") {
|
||||||
|
@ -42,35 +41,36 @@ export const shouldTestInside = (element: ExcalidrawElement) => {
|
||||||
return isDraggableFromInside || isImageElement(element);
|
return isDraggableFromInside || isImageElement(element);
|
||||||
};
|
};
|
||||||
|
|
||||||
export type HitTestArgs = {
|
export type HitTestArgs<Point extends GlobalPoint | LocalPoint> = {
|
||||||
x: number;
|
x: number;
|
||||||
y: number;
|
y: number;
|
||||||
element: ExcalidrawElement;
|
element: ExcalidrawElement;
|
||||||
shape: GeometricShape;
|
shape: GeometricShape<Point>;
|
||||||
threshold?: number;
|
threshold?: number;
|
||||||
frameNameBound?: FrameNameBounds | null;
|
frameNameBound?: FrameNameBounds | null;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const hitElementItself = ({
|
export const hitElementItself = <Point extends GlobalPoint | LocalPoint>({
|
||||||
x,
|
x,
|
||||||
y,
|
y,
|
||||||
element,
|
element,
|
||||||
shape,
|
shape,
|
||||||
threshold = 10,
|
threshold = 10,
|
||||||
frameNameBound = null,
|
frameNameBound = null,
|
||||||
}: HitTestArgs) => {
|
}: HitTestArgs<Point>) => {
|
||||||
let hit = shouldTestInside(element)
|
let hit = shouldTestInside(element)
|
||||||
? // Since `inShape` tests STRICTLY againt the insides of a shape
|
? // Since `inShape` tests STRICTLY againt the insides of a shape
|
||||||
// we would need `onShape` as well to include the "borders"
|
// we would need `onShape` as well to include the "borders"
|
||||||
isPointInShape([x, y], shape) || isPointOnShape([x, y], shape, threshold)
|
isPointInShape(point(x, y), shape) ||
|
||||||
: isPointOnShape([x, y], shape, threshold);
|
isPointOnShape(point(x, y), shape, threshold)
|
||||||
|
: isPointOnShape(point(x, y), shape, threshold);
|
||||||
|
|
||||||
// hit test against a frame's name
|
// hit test against a frame's name
|
||||||
if (!hit && frameNameBound) {
|
if (!hit && frameNameBound) {
|
||||||
hit = isPointInShape([x, y], {
|
hit = isPointInShape(point(x, y), {
|
||||||
type: "polygon",
|
type: "polygon",
|
||||||
data: getPolygonShape(frameNameBound as ExcalidrawRectangleElement)
|
data: getPolygonShape(frameNameBound as ExcalidrawRectangleElement)
|
||||||
.data as Polygon,
|
.data as Polygon<Point>,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -89,11 +89,13 @@ export const hitElementBoundingBox = (
|
||||||
y1 -= tolerance;
|
y1 -= tolerance;
|
||||||
x2 += tolerance;
|
x2 += tolerance;
|
||||||
y2 += tolerance;
|
y2 += tolerance;
|
||||||
return isPointWithinBounds([x1, y1], [x, y], [x2, y2]);
|
return isPointWithinBounds(point(x1, y1), point(x, y), point(x2, y2));
|
||||||
};
|
};
|
||||||
|
|
||||||
export const hitElementBoundingBoxOnly = (
|
export const hitElementBoundingBoxOnly = <
|
||||||
hitArgs: HitTestArgs,
|
Point extends GlobalPoint | LocalPoint,
|
||||||
|
>(
|
||||||
|
hitArgs: HitTestArgs<Point>,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
) => {
|
) => {
|
||||||
return (
|
return (
|
||||||
|
@ -108,10 +110,10 @@ export const hitElementBoundingBoxOnly = (
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const hitElementBoundText = (
|
export const hitElementBoundText = <Point extends GlobalPoint | LocalPoint>(
|
||||||
x: number,
|
x: number,
|
||||||
y: number,
|
y: number,
|
||||||
textShape: GeometricShape | null,
|
textShape: GeometricShape<Point> | null,
|
||||||
): boolean => {
|
): boolean => {
|
||||||
return !!textShape && isPointInShape([x, y], textShape);
|
return !!textShape && isPointInShape(point(x, y), textShape);
|
||||||
};
|
};
|
||||||
|
|
|
@ -11,7 +11,6 @@ import type {
|
||||||
PointerDownState,
|
PointerDownState,
|
||||||
} from "../types";
|
} from "../types";
|
||||||
import { getBoundTextElement, getMinTextElementWidth } from "./textElement";
|
import { getBoundTextElement, getMinTextElementWidth } from "./textElement";
|
||||||
import { getGridPoint } from "../math";
|
|
||||||
import type Scene from "../scene/Scene";
|
import type Scene from "../scene/Scene";
|
||||||
import {
|
import {
|
||||||
isArrowElement,
|
isArrowElement,
|
||||||
|
@ -21,6 +20,7 @@ import {
|
||||||
} from "./typeChecks";
|
} from "./typeChecks";
|
||||||
import { getFontString } from "../utils";
|
import { getFontString } from "../utils";
|
||||||
import { TEXT_AUTOWRAP_THRESHOLD } from "../constants";
|
import { TEXT_AUTOWRAP_THRESHOLD } from "../constants";
|
||||||
|
import { getGridPoint } from "../snapping";
|
||||||
|
|
||||||
export const dragSelectedElements = (
|
export const dragSelectedElements = (
|
||||||
pointerDownState: PointerDownState,
|
pointerDownState: PointerDownState,
|
||||||
|
|
|
@ -10,7 +10,6 @@ import {
|
||||||
import { bindLinearElement } from "./binding";
|
import { bindLinearElement } from "./binding";
|
||||||
import { LinearElementEditor } from "./linearElementEditor";
|
import { LinearElementEditor } from "./linearElementEditor";
|
||||||
import { newArrowElement, newElement } from "./newElement";
|
import { newArrowElement, newElement } from "./newElement";
|
||||||
import { aabbForElement } from "../math";
|
|
||||||
import type {
|
import type {
|
||||||
ElementsMap,
|
ElementsMap,
|
||||||
ExcalidrawBindableElement,
|
ExcalidrawBindableElement,
|
||||||
|
@ -20,7 +19,7 @@ import type {
|
||||||
OrderedExcalidrawElement,
|
OrderedExcalidrawElement,
|
||||||
} from "./types";
|
} from "./types";
|
||||||
import { KEYS } from "../keys";
|
import { KEYS } from "../keys";
|
||||||
import type { AppState, PendingExcalidrawElements, Point } from "../types";
|
import type { AppState, PendingExcalidrawElements } from "../types";
|
||||||
import { mutateElement } from "./mutateElement";
|
import { mutateElement } from "./mutateElement";
|
||||||
import { elementOverlapsWithFrame, elementsAreInFrameBounds } from "../frame";
|
import { elementOverlapsWithFrame, elementsAreInFrameBounds } from "../frame";
|
||||||
import {
|
import {
|
||||||
|
@ -30,6 +29,8 @@ import {
|
||||||
isFlowchartNodeElement,
|
isFlowchartNodeElement,
|
||||||
} from "./typeChecks";
|
} from "./typeChecks";
|
||||||
import { invariant } from "../utils";
|
import { invariant } from "../utils";
|
||||||
|
import { point, type LocalPoint } from "../../math";
|
||||||
|
import { aabbForElement } from "../shapes";
|
||||||
|
|
||||||
type LinkDirection = "up" | "right" | "down" | "left";
|
type LinkDirection = "up" | "right" | "down" | "left";
|
||||||
|
|
||||||
|
@ -81,13 +82,14 @@ const getNodeRelatives = (
|
||||||
"not an ExcalidrawBindableElement",
|
"not an ExcalidrawBindableElement",
|
||||||
);
|
);
|
||||||
|
|
||||||
const edgePoint: Point =
|
const edgePoint = (
|
||||||
type === "predecessors" ? el.points[el.points.length - 1] : [0, 0];
|
type === "predecessors" ? el.points[el.points.length - 1] : [0, 0]
|
||||||
|
) as Readonly<LocalPoint>;
|
||||||
|
|
||||||
const heading = headingForPointFromElement(node, aabbForElement(node), [
|
const heading = headingForPointFromElement(node, aabbForElement(node), [
|
||||||
edgePoint[0] + el.x,
|
edgePoint[0] + el.x,
|
||||||
edgePoint[1] + el.y,
|
edgePoint[1] + el.y,
|
||||||
]);
|
] as Readonly<LocalPoint>);
|
||||||
|
|
||||||
acc.push({
|
acc.push({
|
||||||
relative,
|
relative,
|
||||||
|
@ -419,10 +421,7 @@ const createBindingArrow = (
|
||||||
strokeColor: appState.currentItemStrokeColor,
|
strokeColor: appState.currentItemStrokeColor,
|
||||||
strokeStyle: appState.currentItemStrokeStyle,
|
strokeStyle: appState.currentItemStrokeStyle,
|
||||||
strokeWidth: appState.currentItemStrokeWidth,
|
strokeWidth: appState.currentItemStrokeWidth,
|
||||||
points: [
|
points: [point(0, 0), point(endX, endY)],
|
||||||
[0, 0],
|
|
||||||
[endX, endY],
|
|
||||||
],
|
|
||||||
elbowed: true,
|
elbowed: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -1,12 +1,18 @@
|
||||||
import { lineAngle } from "../../utils/geometry/geometry";
|
import type {
|
||||||
import type { Point, Vector } from "../../utils/geometry/shape";
|
LocalPoint,
|
||||||
|
GlobalPoint,
|
||||||
|
Triangle,
|
||||||
|
Vector,
|
||||||
|
Radians,
|
||||||
|
} from "../../math";
|
||||||
import {
|
import {
|
||||||
getCenterForBounds,
|
point,
|
||||||
PointInTriangle,
|
pointRotateRads,
|
||||||
rotatePoint,
|
pointScaleFromOrigin,
|
||||||
scalePointFromOrigin,
|
radiansToDegrees,
|
||||||
} from "../math";
|
triangleIncludesPoint,
|
||||||
import type { Bounds } from "./bounds";
|
} from "../../math";
|
||||||
|
import { getCenterForBounds, type Bounds } from "./bounds";
|
||||||
import type { ExcalidrawBindableElement } from "./types";
|
import type { ExcalidrawBindableElement } from "./types";
|
||||||
|
|
||||||
export const HEADING_RIGHT = [1, 0] as Heading;
|
export const HEADING_RIGHT = [1, 0] as Heading;
|
||||||
|
@ -15,8 +21,13 @@ export const HEADING_LEFT = [-1, 0] as Heading;
|
||||||
export const HEADING_UP = [0, -1] as Heading;
|
export const HEADING_UP = [0, -1] as Heading;
|
||||||
export type Heading = [1, 0] | [0, 1] | [-1, 0] | [0, -1];
|
export type Heading = [1, 0] | [0, 1] | [-1, 0] | [0, -1];
|
||||||
|
|
||||||
export const headingForDiamond = (a: Point, b: Point) => {
|
export const headingForDiamond = <Point extends GlobalPoint | LocalPoint>(
|
||||||
const angle = lineAngle([a, b]);
|
a: Point,
|
||||||
|
b: Point,
|
||||||
|
) => {
|
||||||
|
const angle = radiansToDegrees(
|
||||||
|
Math.atan2(b[1] - a[1], b[0] - a[0]) as Radians,
|
||||||
|
);
|
||||||
if (angle >= 315 || angle < 45) {
|
if (angle >= 315 || angle < 45) {
|
||||||
return HEADING_UP;
|
return HEADING_UP;
|
||||||
} else if (angle >= 45 && angle < 135) {
|
} else if (angle >= 45 && angle < 135) {
|
||||||
|
@ -47,56 +58,58 @@ export const compareHeading = (a: Heading, b: Heading) =>
|
||||||
// Gets the heading for the point by creating a bounding box around the rotated
|
// Gets the heading for the point by creating a bounding box around the rotated
|
||||||
// close fitting bounding box, then creating 4 search cones around the center of
|
// close fitting bounding box, then creating 4 search cones around the center of
|
||||||
// the external bbox.
|
// the external bbox.
|
||||||
export const headingForPointFromElement = (
|
export const headingForPointFromElement = <
|
||||||
|
Point extends GlobalPoint | LocalPoint,
|
||||||
|
>(
|
||||||
element: Readonly<ExcalidrawBindableElement>,
|
element: Readonly<ExcalidrawBindableElement>,
|
||||||
aabb: Readonly<Bounds>,
|
aabb: Readonly<Bounds>,
|
||||||
point: Readonly<Point>,
|
p: Readonly<LocalPoint | GlobalPoint>,
|
||||||
): Heading => {
|
): Heading => {
|
||||||
const SEARCH_CONE_MULTIPLIER = 2;
|
const SEARCH_CONE_MULTIPLIER = 2;
|
||||||
|
|
||||||
const midPoint = getCenterForBounds(aabb);
|
const midPoint = getCenterForBounds(aabb);
|
||||||
|
|
||||||
if (element.type === "diamond") {
|
if (element.type === "diamond") {
|
||||||
if (point[0] < element.x) {
|
if (p[0] < element.x) {
|
||||||
return HEADING_LEFT;
|
return HEADING_LEFT;
|
||||||
} else if (point[1] < element.y) {
|
} else if (p[1] < element.y) {
|
||||||
return HEADING_UP;
|
return HEADING_UP;
|
||||||
} else if (point[0] > element.x + element.width) {
|
} else if (p[0] > element.x + element.width) {
|
||||||
return HEADING_RIGHT;
|
return HEADING_RIGHT;
|
||||||
} else if (point[1] > element.y + element.height) {
|
} else if (p[1] > element.y + element.height) {
|
||||||
return HEADING_DOWN;
|
return HEADING_DOWN;
|
||||||
}
|
}
|
||||||
|
|
||||||
const top = rotatePoint(
|
const top = pointRotateRads(
|
||||||
scalePointFromOrigin(
|
pointScaleFromOrigin(
|
||||||
[element.x + element.width / 2, element.y],
|
point(element.x + element.width / 2, element.y),
|
||||||
midPoint,
|
midPoint,
|
||||||
SEARCH_CONE_MULTIPLIER,
|
SEARCH_CONE_MULTIPLIER,
|
||||||
),
|
),
|
||||||
midPoint,
|
midPoint,
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
const right = rotatePoint(
|
const right = pointRotateRads(
|
||||||
scalePointFromOrigin(
|
pointScaleFromOrigin(
|
||||||
[element.x + element.width, element.y + element.height / 2],
|
point(element.x + element.width, element.y + element.height / 2),
|
||||||
midPoint,
|
midPoint,
|
||||||
SEARCH_CONE_MULTIPLIER,
|
SEARCH_CONE_MULTIPLIER,
|
||||||
),
|
),
|
||||||
midPoint,
|
midPoint,
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
const bottom = rotatePoint(
|
const bottom = pointRotateRads(
|
||||||
scalePointFromOrigin(
|
pointScaleFromOrigin(
|
||||||
[element.x + element.width / 2, element.y + element.height],
|
point(element.x + element.width / 2, element.y + element.height),
|
||||||
midPoint,
|
midPoint,
|
||||||
SEARCH_CONE_MULTIPLIER,
|
SEARCH_CONE_MULTIPLIER,
|
||||||
),
|
),
|
||||||
midPoint,
|
midPoint,
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
const left = rotatePoint(
|
const left = pointRotateRads(
|
||||||
scalePointFromOrigin(
|
pointScaleFromOrigin(
|
||||||
[element.x, element.y + element.height / 2],
|
point(element.x, element.y + element.height / 2),
|
||||||
midPoint,
|
midPoint,
|
||||||
SEARCH_CONE_MULTIPLIER,
|
SEARCH_CONE_MULTIPLIER,
|
||||||
),
|
),
|
||||||
|
@ -104,43 +117,62 @@ export const headingForPointFromElement = (
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
if (PointInTriangle(point, top, right, midPoint)) {
|
if (triangleIncludesPoint([top, right, midPoint] as Triangle<Point>, p)) {
|
||||||
return headingForDiamond(top, right);
|
return headingForDiamond(top, right);
|
||||||
} else if (PointInTriangle(point, right, bottom, midPoint)) {
|
} else if (
|
||||||
|
triangleIncludesPoint([right, bottom, midPoint] as Triangle<Point>, p)
|
||||||
|
) {
|
||||||
return headingForDiamond(right, bottom);
|
return headingForDiamond(right, bottom);
|
||||||
} else if (PointInTriangle(point, bottom, left, midPoint)) {
|
} else if (
|
||||||
|
triangleIncludesPoint([bottom, left, midPoint] as Triangle<Point>, p)
|
||||||
|
) {
|
||||||
return headingForDiamond(bottom, left);
|
return headingForDiamond(bottom, left);
|
||||||
}
|
}
|
||||||
|
|
||||||
return headingForDiamond(left, top);
|
return headingForDiamond(left, top);
|
||||||
}
|
}
|
||||||
|
|
||||||
const topLeft = scalePointFromOrigin(
|
const topLeft = pointScaleFromOrigin(
|
||||||
[aabb[0], aabb[1]],
|
point(aabb[0], aabb[1]),
|
||||||
midPoint,
|
midPoint,
|
||||||
SEARCH_CONE_MULTIPLIER,
|
SEARCH_CONE_MULTIPLIER,
|
||||||
);
|
) as Point;
|
||||||
const topRight = scalePointFromOrigin(
|
const topRight = pointScaleFromOrigin(
|
||||||
[aabb[2], aabb[1]],
|
point(aabb[2], aabb[1]),
|
||||||
midPoint,
|
midPoint,
|
||||||
SEARCH_CONE_MULTIPLIER,
|
SEARCH_CONE_MULTIPLIER,
|
||||||
);
|
) as Point;
|
||||||
const bottomLeft = scalePointFromOrigin(
|
const bottomLeft = pointScaleFromOrigin(
|
||||||
[aabb[0], aabb[3]],
|
point(aabb[0], aabb[3]),
|
||||||
midPoint,
|
midPoint,
|
||||||
SEARCH_CONE_MULTIPLIER,
|
SEARCH_CONE_MULTIPLIER,
|
||||||
);
|
) as Point;
|
||||||
const bottomRight = scalePointFromOrigin(
|
const bottomRight = pointScaleFromOrigin(
|
||||||
[aabb[2], aabb[3]],
|
point(aabb[2], aabb[3]),
|
||||||
midPoint,
|
midPoint,
|
||||||
SEARCH_CONE_MULTIPLIER,
|
SEARCH_CONE_MULTIPLIER,
|
||||||
);
|
) as Point;
|
||||||
|
|
||||||
return PointInTriangle(point, topLeft, topRight, midPoint)
|
return triangleIncludesPoint(
|
||||||
|
[topLeft, topRight, midPoint] as Triangle<Point>,
|
||||||
|
p,
|
||||||
|
)
|
||||||
? HEADING_UP
|
? HEADING_UP
|
||||||
: PointInTriangle(point, topRight, bottomRight, midPoint)
|
: triangleIncludesPoint(
|
||||||
|
[topRight, bottomRight, midPoint] as Triangle<Point>,
|
||||||
|
p,
|
||||||
|
)
|
||||||
? HEADING_RIGHT
|
? HEADING_RIGHT
|
||||||
: PointInTriangle(point, bottomRight, bottomLeft, midPoint)
|
: triangleIncludesPoint(
|
||||||
|
[bottomRight, bottomLeft, midPoint] as Triangle<Point>,
|
||||||
|
p,
|
||||||
|
)
|
||||||
? HEADING_DOWN
|
? HEADING_DOWN
|
||||||
: HEADING_LEFT;
|
: HEADING_LEFT;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const flipHeading = (h: Heading): Heading =>
|
||||||
|
[
|
||||||
|
h[0] === 0 ? 0 : h[0] > 0 ? -1 : 1,
|
||||||
|
h[1] === 0 ? 0 : h[1] > 0 ? -1 : 1,
|
||||||
|
] as Heading;
|
||||||
|
|
|
@ -11,19 +11,6 @@ import type {
|
||||||
FixedPointBinding,
|
FixedPointBinding,
|
||||||
SceneElementsMap,
|
SceneElementsMap,
|
||||||
} from "./types";
|
} from "./types";
|
||||||
import {
|
|
||||||
distance2d,
|
|
||||||
rotate,
|
|
||||||
isPathALoop,
|
|
||||||
getGridPoint,
|
|
||||||
rotatePoint,
|
|
||||||
centerPoint,
|
|
||||||
getControlPointsForBezierCurve,
|
|
||||||
getBezierXY,
|
|
||||||
getBezierCurveLength,
|
|
||||||
mapIntervalToBezierT,
|
|
||||||
arePointsEqual,
|
|
||||||
} from "../math";
|
|
||||||
import { getElementAbsoluteCoords, getLockedLinearCursorAlignSize } from ".";
|
import { getElementAbsoluteCoords, getLockedLinearCursorAlignSize } from ".";
|
||||||
import type { Bounds } from "./bounds";
|
import type { Bounds } from "./bounds";
|
||||||
import {
|
import {
|
||||||
|
@ -32,7 +19,6 @@ import {
|
||||||
getMinMaxXYFromCurvePathOps,
|
getMinMaxXYFromCurvePathOps,
|
||||||
} from "./bounds";
|
} from "./bounds";
|
||||||
import type {
|
import type {
|
||||||
Point,
|
|
||||||
AppState,
|
AppState,
|
||||||
PointerCoords,
|
PointerCoords,
|
||||||
InteractiveCanvasAppState,
|
InteractiveCanvasAppState,
|
||||||
|
@ -46,7 +32,7 @@ import {
|
||||||
getHoveredElementForBinding,
|
getHoveredElementForBinding,
|
||||||
isBindingEnabled,
|
isBindingEnabled,
|
||||||
} from "./binding";
|
} from "./binding";
|
||||||
import { toBrandedType, tupleToCoors } from "../utils";
|
import { invariant, toBrandedType, tupleToCoors } from "../utils";
|
||||||
import {
|
import {
|
||||||
isBindingElement,
|
isBindingElement,
|
||||||
isElbowArrow,
|
isElbowArrow,
|
||||||
|
@ -60,10 +46,29 @@ import { ShapeCache } from "../scene/ShapeCache";
|
||||||
import type { Store } from "../store";
|
import type { Store } from "../store";
|
||||||
import { mutateElbowArrow } from "./routing";
|
import { mutateElbowArrow } from "./routing";
|
||||||
import type Scene from "../scene/Scene";
|
import type Scene from "../scene/Scene";
|
||||||
|
import type { Radians } from "../../math";
|
||||||
|
import {
|
||||||
|
pointCenter,
|
||||||
|
point,
|
||||||
|
pointRotateRads,
|
||||||
|
pointsEqual,
|
||||||
|
vector,
|
||||||
|
type GlobalPoint,
|
||||||
|
type LocalPoint,
|
||||||
|
pointDistance,
|
||||||
|
} from "../../math";
|
||||||
|
import {
|
||||||
|
getBezierCurveLength,
|
||||||
|
getBezierXY,
|
||||||
|
getControlPointsForBezierCurve,
|
||||||
|
isPathALoop,
|
||||||
|
mapIntervalToBezierT,
|
||||||
|
} from "../shapes";
|
||||||
|
import { getGridPoint } from "../snapping";
|
||||||
|
|
||||||
const editorMidPointsCache: {
|
const editorMidPointsCache: {
|
||||||
version: number | null;
|
version: number | null;
|
||||||
points: (Point | null)[];
|
points: (GlobalPoint | null)[];
|
||||||
zoom: number | null;
|
zoom: number | null;
|
||||||
} = { version: null, points: [], zoom: null };
|
} = { version: null, points: [], zoom: null };
|
||||||
export class LinearElementEditor {
|
export class LinearElementEditor {
|
||||||
|
@ -80,7 +85,7 @@ export class LinearElementEditor {
|
||||||
lastClickedIsEndPoint: boolean;
|
lastClickedIsEndPoint: boolean;
|
||||||
origin: Readonly<{ x: number; y: number }> | null;
|
origin: Readonly<{ x: number; y: number }> | null;
|
||||||
segmentMidpoint: {
|
segmentMidpoint: {
|
||||||
value: Point | null;
|
value: GlobalPoint | null;
|
||||||
index: number | null;
|
index: number | null;
|
||||||
added: boolean;
|
added: boolean;
|
||||||
};
|
};
|
||||||
|
@ -88,7 +93,7 @@ export class LinearElementEditor {
|
||||||
|
|
||||||
/** whether you're dragging a point */
|
/** whether you're dragging a point */
|
||||||
public readonly isDragging: boolean;
|
public readonly isDragging: boolean;
|
||||||
public readonly lastUncommittedPoint: Point | null;
|
public readonly lastUncommittedPoint: LocalPoint | null;
|
||||||
public readonly pointerOffset: Readonly<{ x: number; y: number }>;
|
public readonly pointerOffset: Readonly<{ x: number; y: number }>;
|
||||||
public readonly startBindingElement:
|
public readonly startBindingElement:
|
||||||
| ExcalidrawBindableElement
|
| ExcalidrawBindableElement
|
||||||
|
@ -96,13 +101,13 @@ export class LinearElementEditor {
|
||||||
| "keep";
|
| "keep";
|
||||||
public readonly endBindingElement: ExcalidrawBindableElement | null | "keep";
|
public readonly endBindingElement: ExcalidrawBindableElement | null | "keep";
|
||||||
public readonly hoverPointIndex: number;
|
public readonly hoverPointIndex: number;
|
||||||
public readonly segmentMidPointHoveredCoords: Point | null;
|
public readonly segmentMidPointHoveredCoords: GlobalPoint | null;
|
||||||
|
|
||||||
constructor(element: NonDeleted<ExcalidrawLinearElement>) {
|
constructor(element: NonDeleted<ExcalidrawLinearElement>) {
|
||||||
this.elementId = element.id as string & {
|
this.elementId = element.id as string & {
|
||||||
_brand: "excalidrawLinearElementId";
|
_brand: "excalidrawLinearElementId";
|
||||||
};
|
};
|
||||||
if (!arePointsEqual(element.points[0], [0, 0])) {
|
if (!pointsEqual(element.points[0], point(0, 0))) {
|
||||||
console.error("Linear element is not normalized", Error().stack);
|
console.error("Linear element is not normalized", Error().stack);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -280,7 +285,7 @@ export class LinearElementEditor {
|
||||||
element,
|
element,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
referencePoint,
|
referencePoint,
|
||||||
[scenePointerX, scenePointerY],
|
point(scenePointerX, scenePointerY),
|
||||||
event[KEYS.CTRL_OR_CMD] ? null : app.getEffectiveGridSize(),
|
event[KEYS.CTRL_OR_CMD] ? null : app.getEffectiveGridSize(),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -289,7 +294,10 @@ export class LinearElementEditor {
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
index: selectedIndex,
|
index: selectedIndex,
|
||||||
point: [width + referencePoint[0], height + referencePoint[1]],
|
point: point(
|
||||||
|
width + referencePoint[0],
|
||||||
|
height + referencePoint[1],
|
||||||
|
),
|
||||||
isDragging: selectedIndex === lastClickedPoint,
|
isDragging: selectedIndex === lastClickedPoint,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
@ -310,7 +318,7 @@ export class LinearElementEditor {
|
||||||
LinearElementEditor.movePoints(
|
LinearElementEditor.movePoints(
|
||||||
element,
|
element,
|
||||||
selectedPointsIndices.map((pointIndex) => {
|
selectedPointsIndices.map((pointIndex) => {
|
||||||
const newPointPosition =
|
const newPointPosition: LocalPoint =
|
||||||
pointIndex === lastClickedPoint
|
pointIndex === lastClickedPoint
|
||||||
? LinearElementEditor.createPointAt(
|
? LinearElementEditor.createPointAt(
|
||||||
element,
|
element,
|
||||||
|
@ -319,10 +327,10 @@ export class LinearElementEditor {
|
||||||
scenePointerY - linearElementEditor.pointerOffset.y,
|
scenePointerY - linearElementEditor.pointerOffset.y,
|
||||||
event[KEYS.CTRL_OR_CMD] ? null : app.getEffectiveGridSize(),
|
event[KEYS.CTRL_OR_CMD] ? null : app.getEffectiveGridSize(),
|
||||||
)
|
)
|
||||||
: ([
|
: point(
|
||||||
element.points[pointIndex][0] + deltaX,
|
element.points[pointIndex][0] + deltaX,
|
||||||
element.points[pointIndex][1] + deltaY,
|
element.points[pointIndex][1] + deltaY,
|
||||||
] as const);
|
);
|
||||||
return {
|
return {
|
||||||
index: pointIndex,
|
index: pointIndex,
|
||||||
point: newPointPosition,
|
point: newPointPosition,
|
||||||
|
@ -515,7 +523,7 @@ export class LinearElementEditor {
|
||||||
);
|
);
|
||||||
|
|
||||||
let index = 0;
|
let index = 0;
|
||||||
const midpoints: (Point | null)[] = [];
|
const midpoints: (GlobalPoint | null)[] = [];
|
||||||
while (index < points.length - 1) {
|
while (index < points.length - 1) {
|
||||||
if (
|
if (
|
||||||
LinearElementEditor.isSegmentTooShort(
|
LinearElementEditor.isSegmentTooShort(
|
||||||
|
@ -549,7 +557,7 @@ export class LinearElementEditor {
|
||||||
scenePointer: { x: number; y: number },
|
scenePointer: { x: number; y: number },
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
) => {
|
): GlobalPoint | null => {
|
||||||
const { elementId } = linearElementEditor;
|
const { elementId } = linearElementEditor;
|
||||||
const element = LinearElementEditor.getElement(elementId, elementsMap);
|
const element = LinearElementEditor.getElement(elementId, elementsMap);
|
||||||
if (!element) {
|
if (!element) {
|
||||||
|
@ -579,11 +587,12 @@ export class LinearElementEditor {
|
||||||
const existingSegmentMidpointHitCoords =
|
const existingSegmentMidpointHitCoords =
|
||||||
linearElementEditor.segmentMidPointHoveredCoords;
|
linearElementEditor.segmentMidPointHoveredCoords;
|
||||||
if (existingSegmentMidpointHitCoords) {
|
if (existingSegmentMidpointHitCoords) {
|
||||||
const distance = distance2d(
|
const distance = pointDistance(
|
||||||
|
point(
|
||||||
existingSegmentMidpointHitCoords[0],
|
existingSegmentMidpointHitCoords[0],
|
||||||
existingSegmentMidpointHitCoords[1],
|
existingSegmentMidpointHitCoords[1],
|
||||||
scenePointer.x,
|
),
|
||||||
scenePointer.y,
|
point(scenePointer.x, scenePointer.y),
|
||||||
);
|
);
|
||||||
if (distance <= threshold) {
|
if (distance <= threshold) {
|
||||||
return existingSegmentMidpointHitCoords;
|
return existingSegmentMidpointHitCoords;
|
||||||
|
@ -594,11 +603,9 @@ export class LinearElementEditor {
|
||||||
LinearElementEditor.getEditorMidPoints(element, elementsMap, appState);
|
LinearElementEditor.getEditorMidPoints(element, elementsMap, appState);
|
||||||
while (index < midPoints.length) {
|
while (index < midPoints.length) {
|
||||||
if (midPoints[index] !== null) {
|
if (midPoints[index] !== null) {
|
||||||
const distance = distance2d(
|
const distance = pointDistance(
|
||||||
midPoints[index]![0],
|
point(midPoints[index]![0], midPoints[index]![1]),
|
||||||
midPoints[index]![1],
|
point(scenePointer.x, scenePointer.y),
|
||||||
scenePointer.x,
|
|
||||||
scenePointer.y,
|
|
||||||
);
|
);
|
||||||
if (distance <= threshold) {
|
if (distance <= threshold) {
|
||||||
return midPoints[index];
|
return midPoints[index];
|
||||||
|
@ -612,15 +619,13 @@ export class LinearElementEditor {
|
||||||
|
|
||||||
static isSegmentTooShort(
|
static isSegmentTooShort(
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
startPoint: Point,
|
startPoint: GlobalPoint | LocalPoint,
|
||||||
endPoint: Point,
|
endPoint: GlobalPoint | LocalPoint,
|
||||||
zoom: AppState["zoom"],
|
zoom: AppState["zoom"],
|
||||||
) {
|
) {
|
||||||
let distance = distance2d(
|
let distance = pointDistance(
|
||||||
startPoint[0],
|
point(startPoint[0], startPoint[1]),
|
||||||
startPoint[1],
|
point(endPoint[0], endPoint[1]),
|
||||||
endPoint[0],
|
|
||||||
endPoint[1],
|
|
||||||
);
|
);
|
||||||
if (element.points.length > 2 && element.roundness) {
|
if (element.points.length > 2 && element.roundness) {
|
||||||
distance = getBezierCurveLength(element, endPoint);
|
distance = getBezierCurveLength(element, endPoint);
|
||||||
|
@ -631,12 +636,12 @@ export class LinearElementEditor {
|
||||||
|
|
||||||
static getSegmentMidPoint(
|
static getSegmentMidPoint(
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
startPoint: Point,
|
startPoint: GlobalPoint,
|
||||||
endPoint: Point,
|
endPoint: GlobalPoint,
|
||||||
endPointIndex: number,
|
endPointIndex: number,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
) {
|
): GlobalPoint {
|
||||||
let segmentMidPoint = centerPoint(startPoint, endPoint);
|
let segmentMidPoint = pointCenter(startPoint, endPoint);
|
||||||
if (element.points.length > 2 && element.roundness) {
|
if (element.points.length > 2 && element.roundness) {
|
||||||
const controlPoints = getControlPointsForBezierCurve(
|
const controlPoints = getControlPointsForBezierCurve(
|
||||||
element,
|
element,
|
||||||
|
@ -649,16 +654,15 @@ export class LinearElementEditor {
|
||||||
0.5,
|
0.5,
|
||||||
);
|
);
|
||||||
|
|
||||||
const [tx, ty] = getBezierXY(
|
segmentMidPoint = LinearElementEditor.getPointGlobalCoordinates(
|
||||||
|
element,
|
||||||
|
getBezierXY(
|
||||||
controlPoints[0],
|
controlPoints[0],
|
||||||
controlPoints[1],
|
controlPoints[1],
|
||||||
controlPoints[2],
|
controlPoints[2],
|
||||||
controlPoints[3],
|
controlPoints[3],
|
||||||
t,
|
t,
|
||||||
);
|
),
|
||||||
segmentMidPoint = LinearElementEditor.getPointGlobalCoordinates(
|
|
||||||
element,
|
|
||||||
[tx, ty],
|
|
||||||
elementsMap,
|
elementsMap,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -670,7 +674,7 @@ export class LinearElementEditor {
|
||||||
static getSegmentMidPointIndex(
|
static getSegmentMidPointIndex(
|
||||||
linearElementEditor: LinearElementEditor,
|
linearElementEditor: LinearElementEditor,
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
midPoint: Point,
|
midPoint: GlobalPoint,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
) {
|
) {
|
||||||
const element = LinearElementEditor.getElement(
|
const element = LinearElementEditor.getElement(
|
||||||
|
@ -822,11 +826,12 @@ export class LinearElementEditor {
|
||||||
const cy = (y1 + y2) / 2;
|
const cy = (y1 + y2) / 2;
|
||||||
const targetPoint =
|
const targetPoint =
|
||||||
clickedPointIndex > -1 &&
|
clickedPointIndex > -1 &&
|
||||||
rotate(
|
pointRotateRads(
|
||||||
|
point(
|
||||||
element.x + element.points[clickedPointIndex][0],
|
element.x + element.points[clickedPointIndex][0],
|
||||||
element.y + element.points[clickedPointIndex][1],
|
element.y + element.points[clickedPointIndex][1],
|
||||||
cx,
|
),
|
||||||
cy,
|
point(cx, cy),
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -865,14 +870,17 @@ export class LinearElementEditor {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static arePointsEqual(point1: Point | null, point2: Point | null) {
|
static arePointsEqual<Point extends LocalPoint | GlobalPoint>(
|
||||||
|
point1: Point | null,
|
||||||
|
point2: Point | null,
|
||||||
|
) {
|
||||||
if (!point1 && !point2) {
|
if (!point1 && !point2) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (!point1 || !point2) {
|
if (!point1 || !point2) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return arePointsEqual(point1, point2);
|
return pointsEqual(point1, point2);
|
||||||
}
|
}
|
||||||
|
|
||||||
static handlePointerMove(
|
static handlePointerMove(
|
||||||
|
@ -909,7 +917,7 @@ export class LinearElementEditor {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
let newPoint: Point;
|
let newPoint: LocalPoint;
|
||||||
|
|
||||||
if (shouldRotateWithDiscreteAngle(event) && points.length >= 2) {
|
if (shouldRotateWithDiscreteAngle(event) && points.length >= 2) {
|
||||||
const lastCommittedPoint = points[points.length - 2];
|
const lastCommittedPoint = points[points.length - 2];
|
||||||
|
@ -918,14 +926,14 @@ export class LinearElementEditor {
|
||||||
element,
|
element,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
lastCommittedPoint,
|
lastCommittedPoint,
|
||||||
[scenePointerX, scenePointerY],
|
point(scenePointerX, scenePointerY),
|
||||||
event[KEYS.CTRL_OR_CMD] ? null : app.getEffectiveGridSize(),
|
event[KEYS.CTRL_OR_CMD] ? null : app.getEffectiveGridSize(),
|
||||||
);
|
);
|
||||||
|
|
||||||
newPoint = [
|
newPoint = point(
|
||||||
width + lastCommittedPoint[0],
|
width + lastCommittedPoint[0],
|
||||||
height + lastCommittedPoint[1],
|
height + lastCommittedPoint[1],
|
||||||
];
|
);
|
||||||
} else {
|
} else {
|
||||||
newPoint = LinearElementEditor.createPointAt(
|
newPoint = LinearElementEditor.createPointAt(
|
||||||
element,
|
element,
|
||||||
|
@ -965,30 +973,36 @@ export class LinearElementEditor {
|
||||||
/** scene coords */
|
/** scene coords */
|
||||||
static getPointGlobalCoordinates(
|
static getPointGlobalCoordinates(
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
point: Point,
|
p: LocalPoint,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
) {
|
): GlobalPoint {
|
||||||
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
||||||
const cx = (x1 + x2) / 2;
|
const cx = (x1 + x2) / 2;
|
||||||
const cy = (y1 + y2) / 2;
|
const cy = (y1 + y2) / 2;
|
||||||
|
|
||||||
let { x, y } = element;
|
const { x, y } = element;
|
||||||
[x, y] = rotate(x + point[0], y + point[1], cx, cy, element.angle);
|
return pointRotateRads(
|
||||||
return [x, y] as const;
|
point(x + p[0], y + p[1]),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** scene coords */
|
/** scene coords */
|
||||||
static getPointsGlobalCoordinates(
|
static getPointsGlobalCoordinates(
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): Point[] {
|
): GlobalPoint[] {
|
||||||
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
||||||
const cx = (x1 + x2) / 2;
|
const cx = (x1 + x2) / 2;
|
||||||
const cy = (y1 + y2) / 2;
|
const cy = (y1 + y2) / 2;
|
||||||
return element.points.map((point) => {
|
return element.points.map((p) => {
|
||||||
let { x, y } = element;
|
const { x, y } = element;
|
||||||
[x, y] = rotate(x + point[0], y + point[1], cx, cy, element.angle);
|
return pointRotateRads(
|
||||||
return [x, y] as const;
|
point(x + p[0], y + p[1]),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -997,7 +1011,7 @@ export class LinearElementEditor {
|
||||||
|
|
||||||
indexMaybeFromEnd: number, // -1 for last element
|
indexMaybeFromEnd: number, // -1 for last element
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): Point {
|
): GlobalPoint {
|
||||||
const index =
|
const index =
|
||||||
indexMaybeFromEnd < 0
|
indexMaybeFromEnd < 0
|
||||||
? element.points.length + indexMaybeFromEnd
|
? element.points.length + indexMaybeFromEnd
|
||||||
|
@ -1005,35 +1019,36 @@ export class LinearElementEditor {
|
||||||
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
||||||
const cx = (x1 + x2) / 2;
|
const cx = (x1 + x2) / 2;
|
||||||
const cy = (y1 + y2) / 2;
|
const cy = (y1 + y2) / 2;
|
||||||
|
const p = element.points[index];
|
||||||
const point = element.points[index];
|
|
||||||
const { x, y } = element;
|
const { x, y } = element;
|
||||||
return point
|
|
||||||
? rotate(x + point[0], y + point[1], cx, cy, element.angle)
|
return p
|
||||||
: rotate(x, y, cx, cy, element.angle);
|
? pointRotateRads(point(x + p[0], y + p[1]), point(cx, cy), element.angle)
|
||||||
|
: pointRotateRads(point(x, y), point(cx, cy), element.angle);
|
||||||
}
|
}
|
||||||
|
|
||||||
static pointFromAbsoluteCoords(
|
static pointFromAbsoluteCoords(
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
absoluteCoords: Point,
|
absoluteCoords: GlobalPoint,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): Point {
|
): LocalPoint {
|
||||||
if (isElbowArrow(element)) {
|
if (isElbowArrow(element)) {
|
||||||
// No rotation for elbow arrows
|
// No rotation for elbow arrows
|
||||||
return [absoluteCoords[0] - element.x, absoluteCoords[1] - element.y];
|
return point(
|
||||||
|
absoluteCoords[0] - element.x,
|
||||||
|
absoluteCoords[1] - element.y,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
||||||
const cx = (x1 + x2) / 2;
|
const cx = (x1 + x2) / 2;
|
||||||
const cy = (y1 + y2) / 2;
|
const cy = (y1 + y2) / 2;
|
||||||
const [x, y] = rotate(
|
const [x, y] = pointRotateRads(
|
||||||
absoluteCoords[0],
|
point(absoluteCoords[0], absoluteCoords[1]),
|
||||||
absoluteCoords[1],
|
point(cx, cy),
|
||||||
cx,
|
-element.angle as Radians,
|
||||||
cy,
|
|
||||||
-element.angle,
|
|
||||||
);
|
);
|
||||||
return [x - element.x, y - element.y];
|
return point(x - element.x, y - element.y);
|
||||||
}
|
}
|
||||||
|
|
||||||
static getPointIndexUnderCursor(
|
static getPointIndexUnderCursor(
|
||||||
|
@ -1052,9 +1067,9 @@ export class LinearElementEditor {
|
||||||
// points on the left, thus should take precedence when clicking, if they
|
// points on the left, thus should take precedence when clicking, if they
|
||||||
// overlap
|
// overlap
|
||||||
while (--idx > -1) {
|
while (--idx > -1) {
|
||||||
const point = pointHandles[idx];
|
const p = pointHandles[idx];
|
||||||
if (
|
if (
|
||||||
distance2d(x, y, point[0], point[1]) * zoom.value <
|
pointDistance(point(x, y), point(p[0], p[1])) * zoom.value <
|
||||||
// +1px to account for outline stroke
|
// +1px to account for outline stroke
|
||||||
LinearElementEditor.POINT_HANDLE_SIZE + 1
|
LinearElementEditor.POINT_HANDLE_SIZE + 1
|
||||||
) {
|
) {
|
||||||
|
@ -1070,20 +1085,18 @@ export class LinearElementEditor {
|
||||||
scenePointerX: number,
|
scenePointerX: number,
|
||||||
scenePointerY: number,
|
scenePointerY: number,
|
||||||
gridSize: NullableGridSize,
|
gridSize: NullableGridSize,
|
||||||
): Point {
|
): LocalPoint {
|
||||||
const pointerOnGrid = getGridPoint(scenePointerX, scenePointerY, gridSize);
|
const pointerOnGrid = getGridPoint(scenePointerX, scenePointerY, gridSize);
|
||||||
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
||||||
const cx = (x1 + x2) / 2;
|
const cx = (x1 + x2) / 2;
|
||||||
const cy = (y1 + y2) / 2;
|
const cy = (y1 + y2) / 2;
|
||||||
const [rotatedX, rotatedY] = rotate(
|
const [rotatedX, rotatedY] = pointRotateRads(
|
||||||
pointerOnGrid[0],
|
point(pointerOnGrid[0], pointerOnGrid[1]),
|
||||||
pointerOnGrid[1],
|
point(cx, cy),
|
||||||
cx,
|
-element.angle as Radians,
|
||||||
cy,
|
|
||||||
-element.angle,
|
|
||||||
);
|
);
|
||||||
|
|
||||||
return [rotatedX - element.x, rotatedY - element.y];
|
return point(rotatedX - element.x, rotatedY - element.y);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1091,15 +1104,19 @@ export class LinearElementEditor {
|
||||||
* expected in various parts of the codebase. Also returns new x/y to account
|
* expected in various parts of the codebase. Also returns new x/y to account
|
||||||
* for the potential normalization.
|
* for the potential normalization.
|
||||||
*/
|
*/
|
||||||
static getNormalizedPoints(element: ExcalidrawLinearElement) {
|
static getNormalizedPoints(element: ExcalidrawLinearElement): {
|
||||||
|
points: LocalPoint[];
|
||||||
|
x: number;
|
||||||
|
y: number;
|
||||||
|
} {
|
||||||
const { points } = element;
|
const { points } = element;
|
||||||
|
|
||||||
const offsetX = points[0][0];
|
const offsetX = points[0][0];
|
||||||
const offsetY = points[0][1];
|
const offsetY = points[0][1];
|
||||||
|
|
||||||
return {
|
return {
|
||||||
points: points.map((point) => {
|
points: points.map((p) => {
|
||||||
return [point[0] - offsetX, point[1] - offsetY] as const;
|
return point(p[0] - offsetX, p[1] - offsetY);
|
||||||
}),
|
}),
|
||||||
x: element.x + offsetX,
|
x: element.x + offsetX,
|
||||||
y: element.y + offsetY,
|
y: element.y + offsetY,
|
||||||
|
@ -1116,17 +1133,23 @@ export class LinearElementEditor {
|
||||||
static duplicateSelectedPoints(
|
static duplicateSelectedPoints(
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
||||||
) {
|
): AppState {
|
||||||
if (!appState.editingLinearElement) {
|
invariant(
|
||||||
return false;
|
appState.editingLinearElement,
|
||||||
}
|
"Not currently editing a linear element",
|
||||||
|
);
|
||||||
|
|
||||||
const { selectedPointsIndices, elementId } = appState.editingLinearElement;
|
const { selectedPointsIndices, elementId } = appState.editingLinearElement;
|
||||||
const element = LinearElementEditor.getElement(elementId, elementsMap);
|
const element = LinearElementEditor.getElement(elementId, elementsMap);
|
||||||
|
|
||||||
if (!element || selectedPointsIndices === null) {
|
invariant(
|
||||||
return false;
|
element,
|
||||||
}
|
"The linear element does not exist in the provided Scene",
|
||||||
|
);
|
||||||
|
invariant(
|
||||||
|
selectedPointsIndices != null,
|
||||||
|
"There are no selected points to duplicate",
|
||||||
|
);
|
||||||
|
|
||||||
const { points } = element;
|
const { points } = element;
|
||||||
|
|
||||||
|
@ -1134,9 +1157,9 @@ export class LinearElementEditor {
|
||||||
|
|
||||||
let pointAddedToEnd = false;
|
let pointAddedToEnd = false;
|
||||||
let indexCursor = -1;
|
let indexCursor = -1;
|
||||||
const nextPoints = points.reduce((acc: Point[], point, index) => {
|
const nextPoints = points.reduce((acc: LocalPoint[], p, index) => {
|
||||||
++indexCursor;
|
++indexCursor;
|
||||||
acc.push(point);
|
acc.push(p);
|
||||||
|
|
||||||
const isSelected = selectedPointsIndices.includes(index);
|
const isSelected = selectedPointsIndices.includes(index);
|
||||||
if (isSelected) {
|
if (isSelected) {
|
||||||
|
@ -1147,8 +1170,8 @@ export class LinearElementEditor {
|
||||||
}
|
}
|
||||||
acc.push(
|
acc.push(
|
||||||
nextPoint
|
nextPoint
|
||||||
? [(point[0] + nextPoint[0]) / 2, (point[1] + nextPoint[1]) / 2]
|
? point((p[0] + nextPoint[0]) / 2, (p[1] + nextPoint[1]) / 2)
|
||||||
: [point[0], point[1]],
|
: point(p[0], p[1]),
|
||||||
);
|
);
|
||||||
|
|
||||||
nextSelectedIndices.push(indexCursor + 1);
|
nextSelectedIndices.push(indexCursor + 1);
|
||||||
|
@ -1169,7 +1192,7 @@ export class LinearElementEditor {
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
index: element.points.length - 1,
|
index: element.points.length - 1,
|
||||||
point: [lastPoint[0] + 30, lastPoint[1] + 30],
|
point: point(lastPoint[0] + 30, lastPoint[1] + 30),
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
elementsMap,
|
elementsMap,
|
||||||
|
@ -1177,13 +1200,11 @@ export class LinearElementEditor {
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
appState: {
|
|
||||||
...appState,
|
...appState,
|
||||||
editingLinearElement: {
|
editingLinearElement: {
|
||||||
...appState.editingLinearElement,
|
...appState.editingLinearElement,
|
||||||
selectedPointsIndices: nextSelectedIndices,
|
selectedPointsIndices: nextSelectedIndices,
|
||||||
},
|
},
|
||||||
},
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1209,10 +1230,10 @@ export class LinearElementEditor {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const nextPoints = element.points.reduce((acc: Point[], point, idx) => {
|
const nextPoints = element.points.reduce((acc: LocalPoint[], p, idx) => {
|
||||||
if (!pointIndices.includes(idx)) {
|
if (!pointIndices.includes(idx)) {
|
||||||
acc.push(
|
acc.push(
|
||||||
!acc.length ? [0, 0] : [point[0] - offsetX, point[1] - offsetY],
|
!acc.length ? point(0, 0) : point(p[0] - offsetX, p[1] - offsetY),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return acc;
|
return acc;
|
||||||
|
@ -1229,7 +1250,7 @@ export class LinearElementEditor {
|
||||||
|
|
||||||
static addPoints(
|
static addPoints(
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
targetPoints: { point: Point }[],
|
targetPoints: { point: LocalPoint }[],
|
||||||
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
||||||
) {
|
) {
|
||||||
const offsetX = 0;
|
const offsetX = 0;
|
||||||
|
@ -1247,7 +1268,7 @@ export class LinearElementEditor {
|
||||||
|
|
||||||
static movePoints(
|
static movePoints(
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
targetPoints: { index: number; point: Point; isDragging?: boolean }[],
|
targetPoints: { index: number; point: LocalPoint; isDragging?: boolean }[],
|
||||||
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
||||||
otherUpdates?: {
|
otherUpdates?: {
|
||||||
startBinding?: PointBinding | null;
|
startBinding?: PointBinding | null;
|
||||||
|
@ -1277,11 +1298,11 @@ export class LinearElementEditor {
|
||||||
selectedOriginPoint.point[1] + points[selectedOriginPoint.index][1];
|
selectedOriginPoint.point[1] + points[selectedOriginPoint.index][1];
|
||||||
}
|
}
|
||||||
|
|
||||||
const nextPoints = points.map((point, idx) => {
|
const nextPoints: LocalPoint[] = points.map((p, idx) => {
|
||||||
const selectedPointData = targetPoints.find((p) => p.index === idx);
|
const selectedPointData = targetPoints.find((t) => t.index === idx);
|
||||||
if (selectedPointData) {
|
if (selectedPointData) {
|
||||||
if (selectedPointData.index === 0) {
|
if (selectedPointData.index === 0) {
|
||||||
return point;
|
return p;
|
||||||
}
|
}
|
||||||
|
|
||||||
const deltaX =
|
const deltaX =
|
||||||
|
@ -1289,14 +1310,9 @@ export class LinearElementEditor {
|
||||||
const deltaY =
|
const deltaY =
|
||||||
selectedPointData.point[1] - points[selectedPointData.index][1];
|
selectedPointData.point[1] - points[selectedPointData.index][1];
|
||||||
|
|
||||||
return [
|
return point(p[0] + deltaX - offsetX, p[1] + deltaY - offsetY);
|
||||||
point[0] + deltaX - offsetX,
|
|
||||||
point[1] + deltaY - offsetY,
|
|
||||||
] as const;
|
|
||||||
}
|
}
|
||||||
return offsetX || offsetY
|
return offsetX || offsetY ? point(p[0] - offsetX, p[1] - offsetY) : p;
|
||||||
? ([point[0] - offsetX, point[1] - offsetY] as const)
|
|
||||||
: point;
|
|
||||||
});
|
});
|
||||||
|
|
||||||
LinearElementEditor._updatePoints(
|
LinearElementEditor._updatePoints(
|
||||||
|
@ -1349,11 +1365,9 @@ export class LinearElementEditor {
|
||||||
}
|
}
|
||||||
|
|
||||||
const origin = linearElementEditor.pointerDownState.origin!;
|
const origin = linearElementEditor.pointerDownState.origin!;
|
||||||
const dist = distance2d(
|
const dist = pointDistance(
|
||||||
origin.x,
|
point(origin.x, origin.y),
|
||||||
origin.y,
|
point(pointerCoords.x, pointerCoords.y),
|
||||||
pointerCoords.x,
|
|
||||||
pointerCoords.y,
|
|
||||||
);
|
);
|
||||||
if (
|
if (
|
||||||
!appState.editingLinearElement &&
|
!appState.editingLinearElement &&
|
||||||
|
@ -1418,7 +1432,7 @@ export class LinearElementEditor {
|
||||||
|
|
||||||
private static _updatePoints(
|
private static _updatePoints(
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
nextPoints: readonly Point[],
|
nextPoints: readonly LocalPoint[],
|
||||||
offsetX: number,
|
offsetX: number,
|
||||||
offsetY: number,
|
offsetY: number,
|
||||||
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
||||||
|
@ -1461,7 +1475,7 @@ export class LinearElementEditor {
|
||||||
element,
|
element,
|
||||||
mergedElementsMap,
|
mergedElementsMap,
|
||||||
nextPoints,
|
nextPoints,
|
||||||
[offsetX, offsetY],
|
vector(offsetX, offsetY),
|
||||||
bindings,
|
bindings,
|
||||||
options,
|
options,
|
||||||
);
|
);
|
||||||
|
@ -1474,7 +1488,11 @@ export class LinearElementEditor {
|
||||||
const prevCenterY = (prevCoords[1] + prevCoords[3]) / 2;
|
const prevCenterY = (prevCoords[1] + prevCoords[3]) / 2;
|
||||||
const dX = prevCenterX - nextCenterX;
|
const dX = prevCenterX - nextCenterX;
|
||||||
const dY = prevCenterY - nextCenterY;
|
const dY = prevCenterY - nextCenterY;
|
||||||
const rotated = rotate(offsetX, offsetY, dX, dY, element.angle);
|
const rotated = pointRotateRads(
|
||||||
|
point(offsetX, offsetY),
|
||||||
|
point(dX, dY),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
mutateElement(element, {
|
mutateElement(element, {
|
||||||
...otherUpdates,
|
...otherUpdates,
|
||||||
points: nextPoints,
|
points: nextPoints,
|
||||||
|
@ -1487,8 +1505,8 @@ export class LinearElementEditor {
|
||||||
private static _getShiftLockedDelta(
|
private static _getShiftLockedDelta(
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
referencePoint: Point,
|
referencePoint: LocalPoint,
|
||||||
scenePointer: Point,
|
scenePointer: GlobalPoint,
|
||||||
gridSize: NullableGridSize,
|
gridSize: NullableGridSize,
|
||||||
) {
|
) {
|
||||||
const referencePointCoords = LinearElementEditor.getPointGlobalCoordinates(
|
const referencePointCoords = LinearElementEditor.getPointGlobalCoordinates(
|
||||||
|
@ -1517,7 +1535,11 @@ export class LinearElementEditor {
|
||||||
gridY,
|
gridY,
|
||||||
);
|
);
|
||||||
|
|
||||||
return rotatePoint([width, height], [0, 0], -element.angle);
|
return pointRotateRads(
|
||||||
|
point(width, height),
|
||||||
|
point(0, 0),
|
||||||
|
-element.angle as Radians,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
static getBoundTextElementPosition = (
|
static getBoundTextElementPosition = (
|
||||||
|
@ -1548,7 +1570,7 @@ export class LinearElementEditor {
|
||||||
|
|
||||||
let midSegmentMidpoint = editorMidPointsCache.points[index];
|
let midSegmentMidpoint = editorMidPointsCache.points[index];
|
||||||
if (element.points.length === 2) {
|
if (element.points.length === 2) {
|
||||||
midSegmentMidpoint = centerPoint(points[0], points[1]);
|
midSegmentMidpoint = pointCenter(points[0], points[1]);
|
||||||
}
|
}
|
||||||
if (
|
if (
|
||||||
!midSegmentMidpoint ||
|
!midSegmentMidpoint ||
|
||||||
|
@ -1585,37 +1607,38 @@ export class LinearElementEditor {
|
||||||
);
|
);
|
||||||
const boundTextX2 = boundTextX1 + boundTextElement.width;
|
const boundTextX2 = boundTextX1 + boundTextElement.width;
|
||||||
const boundTextY2 = boundTextY1 + boundTextElement.height;
|
const boundTextY2 = boundTextY1 + boundTextElement.height;
|
||||||
|
const centerPoint = point(cx, cy);
|
||||||
|
|
||||||
const topLeftRotatedPoint = rotatePoint([x1, y1], [cx, cy], element.angle);
|
const topLeftRotatedPoint = pointRotateRads(
|
||||||
const topRightRotatedPoint = rotatePoint([x2, y1], [cx, cy], element.angle);
|
point(x1, y1),
|
||||||
|
centerPoint,
|
||||||
const counterRotateBoundTextTopLeft = rotatePoint(
|
element.angle,
|
||||||
[boundTextX1, boundTextY1],
|
|
||||||
|
|
||||||
[cx, cy],
|
|
||||||
|
|
||||||
-element.angle,
|
|
||||||
);
|
);
|
||||||
const counterRotateBoundTextTopRight = rotatePoint(
|
const topRightRotatedPoint = pointRotateRads(
|
||||||
[boundTextX2, boundTextY1],
|
point(x2, y1),
|
||||||
|
centerPoint,
|
||||||
[cx, cy],
|
element.angle,
|
||||||
|
|
||||||
-element.angle,
|
|
||||||
);
|
);
|
||||||
const counterRotateBoundTextBottomLeft = rotatePoint(
|
|
||||||
[boundTextX1, boundTextY2],
|
|
||||||
|
|
||||||
[cx, cy],
|
const counterRotateBoundTextTopLeft = pointRotateRads(
|
||||||
|
point(boundTextX1, boundTextY1),
|
||||||
-element.angle,
|
centerPoint,
|
||||||
|
-element.angle as Radians,
|
||||||
);
|
);
|
||||||
const counterRotateBoundTextBottomRight = rotatePoint(
|
const counterRotateBoundTextTopRight = pointRotateRads(
|
||||||
[boundTextX2, boundTextY2],
|
point(boundTextX2, boundTextY1),
|
||||||
|
centerPoint,
|
||||||
[cx, cy],
|
-element.angle as Radians,
|
||||||
|
);
|
||||||
-element.angle,
|
const counterRotateBoundTextBottomLeft = pointRotateRads(
|
||||||
|
point(boundTextX1, boundTextY2),
|
||||||
|
centerPoint,
|
||||||
|
-element.angle as Radians,
|
||||||
|
);
|
||||||
|
const counterRotateBoundTextBottomRight = pointRotateRads(
|
||||||
|
point(boundTextX2, boundTextY2),
|
||||||
|
centerPoint,
|
||||||
|
-element.angle as Radians,
|
||||||
);
|
);
|
||||||
|
|
||||||
if (
|
if (
|
||||||
|
|
|
@ -2,7 +2,6 @@ import type { ExcalidrawElement } from "./types";
|
||||||
import Scene from "../scene/Scene";
|
import Scene from "../scene/Scene";
|
||||||
import { getSizeFromPoints } from "../points";
|
import { getSizeFromPoints } from "../points";
|
||||||
import { randomInteger } from "../random";
|
import { randomInteger } from "../random";
|
||||||
import type { Point } from "../types";
|
|
||||||
import { getUpdatedTimestamp } from "../utils";
|
import { getUpdatedTimestamp } from "../utils";
|
||||||
import type { Mutable } from "../utility-types";
|
import type { Mutable } from "../utility-types";
|
||||||
import { ShapeCache } from "../scene/ShapeCache";
|
import { ShapeCache } from "../scene/ShapeCache";
|
||||||
|
@ -59,8 +58,8 @@ export const mutateElement = <TElement extends Mutable<ExcalidrawElement>>(
|
||||||
let didChangePoints = false;
|
let didChangePoints = false;
|
||||||
let index = prevPoints.length;
|
let index = prevPoints.length;
|
||||||
while (--index) {
|
while (--index) {
|
||||||
const prevPoint: Point = prevPoints[index];
|
const prevPoint = prevPoints[index];
|
||||||
const nextPoint: Point = nextPoints[index];
|
const nextPoint = nextPoints[index];
|
||||||
if (
|
if (
|
||||||
prevPoint[0] !== nextPoint[0] ||
|
prevPoint[0] !== nextPoint[0] ||
|
||||||
prevPoint[1] !== nextPoint[1]
|
prevPoint[1] !== nextPoint[1]
|
||||||
|
|
|
@ -4,6 +4,8 @@ import { API } from "../tests/helpers/api";
|
||||||
import { FONT_FAMILY, ROUNDNESS } from "../constants";
|
import { FONT_FAMILY, ROUNDNESS } from "../constants";
|
||||||
import { isPrimitive } from "../utils";
|
import { isPrimitive } from "../utils";
|
||||||
import type { ExcalidrawLinearElement } from "./types";
|
import type { ExcalidrawLinearElement } from "./types";
|
||||||
|
import type { LocalPoint } from "../../math";
|
||||||
|
import { point } from "../../math";
|
||||||
|
|
||||||
const assertCloneObjects = (source: any, clone: any) => {
|
const assertCloneObjects = (source: any, clone: any) => {
|
||||||
for (const key in clone) {
|
for (const key in clone) {
|
||||||
|
@ -36,10 +38,7 @@ describe("duplicating single elements", () => {
|
||||||
element.__proto__ = { hello: "world" };
|
element.__proto__ = { hello: "world" };
|
||||||
|
|
||||||
mutateElement(element, {
|
mutateElement(element, {
|
||||||
points: [
|
points: [point<LocalPoint>(1, 2), point<LocalPoint>(3, 4)],
|
||||||
[1, 2],
|
|
||||||
[3, 4],
|
|
||||||
],
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const copy = duplicateElement(null, new Map(), element);
|
const copy = duplicateElement(null, new Map(), element);
|
||||||
|
|
|
@ -30,7 +30,6 @@ import { bumpVersion, newElementWith } from "./mutateElement";
|
||||||
import { getNewGroupIdsForDuplication } from "../groups";
|
import { getNewGroupIdsForDuplication } from "../groups";
|
||||||
import type { AppState } from "../types";
|
import type { AppState } from "../types";
|
||||||
import { getElementAbsoluteCoords } from ".";
|
import { getElementAbsoluteCoords } from ".";
|
||||||
import { adjustXYWithRotation } from "../math";
|
|
||||||
import { getResizedElementAbsoluteCoords } from "./bounds";
|
import { getResizedElementAbsoluteCoords } from "./bounds";
|
||||||
import {
|
import {
|
||||||
measureText,
|
measureText,
|
||||||
|
@ -48,6 +47,7 @@ import {
|
||||||
} from "../constants";
|
} from "../constants";
|
||||||
import type { MarkOptional, Merge, Mutable } from "../utility-types";
|
import type { MarkOptional, Merge, Mutable } from "../utility-types";
|
||||||
import { getLineHeight } from "../fonts";
|
import { getLineHeight } from "../fonts";
|
||||||
|
import type { Radians } from "../../math";
|
||||||
|
|
||||||
export type ElementConstructorOpts = MarkOptional<
|
export type ElementConstructorOpts = MarkOptional<
|
||||||
Omit<ExcalidrawGenericElement, "id" | "type" | "isDeleted" | "updated">,
|
Omit<ExcalidrawGenericElement, "id" | "type" | "isDeleted" | "updated">,
|
||||||
|
@ -88,7 +88,7 @@ const _newElementBase = <T extends ExcalidrawElement>(
|
||||||
opacity = DEFAULT_ELEMENT_PROPS.opacity,
|
opacity = DEFAULT_ELEMENT_PROPS.opacity,
|
||||||
width = 0,
|
width = 0,
|
||||||
height = 0,
|
height = 0,
|
||||||
angle = 0,
|
angle = 0 as Radians,
|
||||||
groupIds = [],
|
groupIds = [],
|
||||||
frameId = null,
|
frameId = null,
|
||||||
index = null,
|
index = null,
|
||||||
|
@ -348,6 +348,53 @@ const getAdjustedDimensions = (
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const adjustXYWithRotation = (
|
||||||
|
sides: {
|
||||||
|
n?: boolean;
|
||||||
|
e?: boolean;
|
||||||
|
s?: boolean;
|
||||||
|
w?: boolean;
|
||||||
|
},
|
||||||
|
x: number,
|
||||||
|
y: number,
|
||||||
|
angle: number,
|
||||||
|
deltaX1: number,
|
||||||
|
deltaY1: number,
|
||||||
|
deltaX2: number,
|
||||||
|
deltaY2: number,
|
||||||
|
): [number, number] => {
|
||||||
|
const cos = Math.cos(angle);
|
||||||
|
const sin = Math.sin(angle);
|
||||||
|
if (sides.e && sides.w) {
|
||||||
|
x += deltaX1 + deltaX2;
|
||||||
|
} else if (sides.e) {
|
||||||
|
x += deltaX1 * (1 + cos);
|
||||||
|
y += deltaX1 * sin;
|
||||||
|
x += deltaX2 * (1 - cos);
|
||||||
|
y += deltaX2 * -sin;
|
||||||
|
} else if (sides.w) {
|
||||||
|
x += deltaX1 * (1 - cos);
|
||||||
|
y += deltaX1 * -sin;
|
||||||
|
x += deltaX2 * (1 + cos);
|
||||||
|
y += deltaX2 * sin;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sides.n && sides.s) {
|
||||||
|
y += deltaY1 + deltaY2;
|
||||||
|
} else if (sides.n) {
|
||||||
|
x += deltaY1 * sin;
|
||||||
|
y += deltaY1 * (1 - cos);
|
||||||
|
x += deltaY2 * -sin;
|
||||||
|
y += deltaY2 * (1 + cos);
|
||||||
|
} else if (sides.s) {
|
||||||
|
x += deltaY1 * -sin;
|
||||||
|
y += deltaY1 * (1 + cos);
|
||||||
|
x += deltaY2 * sin;
|
||||||
|
y += deltaY2 * (1 - cos);
|
||||||
|
}
|
||||||
|
return [x, y];
|
||||||
|
};
|
||||||
|
|
||||||
export const refreshTextDimensions = (
|
export const refreshTextDimensions = (
|
||||||
textElement: ExcalidrawTextElement,
|
textElement: ExcalidrawTextElement,
|
||||||
container: ExcalidrawTextContainer | null,
|
container: ExcalidrawTextContainer | null,
|
||||||
|
|
|
@ -1,7 +1,5 @@
|
||||||
import { MIN_FONT_SIZE, SHIFT_LOCKING_ANGLE } from "../constants";
|
import { MIN_FONT_SIZE, SHIFT_LOCKING_ANGLE } from "../constants";
|
||||||
import { rescalePoints } from "../points";
|
import { rescalePoints } from "../points";
|
||||||
|
|
||||||
import { rotate, centerPoint, rotatePoint } from "../math";
|
|
||||||
import type {
|
import type {
|
||||||
ExcalidrawLinearElement,
|
ExcalidrawLinearElement,
|
||||||
ExcalidrawTextElement,
|
ExcalidrawTextElement,
|
||||||
|
@ -38,7 +36,7 @@ import type {
|
||||||
MaybeTransformHandleType,
|
MaybeTransformHandleType,
|
||||||
TransformHandleDirection,
|
TransformHandleDirection,
|
||||||
} from "./transformHandles";
|
} from "./transformHandles";
|
||||||
import type { Point, PointerDownState } from "../types";
|
import type { PointerDownState } from "../types";
|
||||||
import Scene from "../scene/Scene";
|
import Scene from "../scene/Scene";
|
||||||
import {
|
import {
|
||||||
getApproxMinLineWidth,
|
getApproxMinLineWidth,
|
||||||
|
@ -55,16 +53,15 @@ import {
|
||||||
import { LinearElementEditor } from "./linearElementEditor";
|
import { LinearElementEditor } from "./linearElementEditor";
|
||||||
import { isInGroup } from "../groups";
|
import { isInGroup } from "../groups";
|
||||||
import { mutateElbowArrow } from "./routing";
|
import { mutateElbowArrow } from "./routing";
|
||||||
|
import type { GlobalPoint } from "../../math";
|
||||||
export const normalizeAngle = (angle: number): number => {
|
import {
|
||||||
if (angle < 0) {
|
pointCenter,
|
||||||
return angle + 2 * Math.PI;
|
normalizeRadians,
|
||||||
}
|
point,
|
||||||
if (angle >= 2 * Math.PI) {
|
pointFromPair,
|
||||||
return angle - 2 * Math.PI;
|
pointRotateRads,
|
||||||
}
|
type Radians,
|
||||||
return angle;
|
} from "../../math";
|
||||||
};
|
|
||||||
|
|
||||||
// Returns true when transform (resizing/rotation) happened
|
// Returns true when transform (resizing/rotation) happened
|
||||||
export const transformElements = (
|
export const transformElements = (
|
||||||
|
@ -158,16 +155,17 @@ const rotateSingleElement = (
|
||||||
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element, elementsMap);
|
||||||
const cx = (x1 + x2) / 2;
|
const cx = (x1 + x2) / 2;
|
||||||
const cy = (y1 + y2) / 2;
|
const cy = (y1 + y2) / 2;
|
||||||
let angle: number;
|
let angle: Radians;
|
||||||
if (isFrameLikeElement(element)) {
|
if (isFrameLikeElement(element)) {
|
||||||
angle = 0;
|
angle = 0 as Radians;
|
||||||
} else {
|
} else {
|
||||||
angle = (5 * Math.PI) / 2 + Math.atan2(pointerY - cy, pointerX - cx);
|
angle = ((5 * Math.PI) / 2 +
|
||||||
|
Math.atan2(pointerY - cy, pointerX - cx)) as Radians;
|
||||||
if (shouldRotateWithDiscreteAngle) {
|
if (shouldRotateWithDiscreteAngle) {
|
||||||
angle += SHIFT_LOCKING_ANGLE / 2;
|
angle = (angle + SHIFT_LOCKING_ANGLE / 2) as Radians;
|
||||||
angle -= angle % SHIFT_LOCKING_ANGLE;
|
angle = (angle - (angle % SHIFT_LOCKING_ANGLE)) as Radians;
|
||||||
}
|
}
|
||||||
angle = normalizeAngle(angle);
|
angle = normalizeRadians(angle as Radians);
|
||||||
}
|
}
|
||||||
const boundTextElementId = getBoundTextElementId(element);
|
const boundTextElementId = getBoundTextElementId(element);
|
||||||
|
|
||||||
|
@ -240,12 +238,10 @@ const resizeSingleTextElement = (
|
||||||
elementsMap,
|
elementsMap,
|
||||||
);
|
);
|
||||||
// rotation pointer with reverse angle
|
// rotation pointer with reverse angle
|
||||||
const [rotatedX, rotatedY] = rotate(
|
const [rotatedX, rotatedY] = pointRotateRads(
|
||||||
pointerX,
|
point(pointerX, pointerY),
|
||||||
pointerY,
|
point(cx, cy),
|
||||||
cx,
|
-element.angle as Radians,
|
||||||
cy,
|
|
||||||
-element.angle,
|
|
||||||
);
|
);
|
||||||
let scaleX = 0;
|
let scaleX = 0;
|
||||||
let scaleY = 0;
|
let scaleY = 0;
|
||||||
|
@ -279,20 +275,26 @@ const resizeSingleTextElement = (
|
||||||
const startBottomRight = [x2, y2];
|
const startBottomRight = [x2, y2];
|
||||||
const startCenter = [cx, cy];
|
const startCenter = [cx, cy];
|
||||||
|
|
||||||
let newTopLeft = [x1, y1] as [number, number];
|
let newTopLeft = point<GlobalPoint>(x1, y1);
|
||||||
if (["n", "w", "nw"].includes(transformHandleType)) {
|
if (["n", "w", "nw"].includes(transformHandleType)) {
|
||||||
newTopLeft = [
|
newTopLeft = point<GlobalPoint>(
|
||||||
startBottomRight[0] - Math.abs(nextWidth),
|
startBottomRight[0] - Math.abs(nextWidth),
|
||||||
startBottomRight[1] - Math.abs(nextHeight),
|
startBottomRight[1] - Math.abs(nextHeight),
|
||||||
];
|
);
|
||||||
}
|
}
|
||||||
if (transformHandleType === "ne") {
|
if (transformHandleType === "ne") {
|
||||||
const bottomLeft = [startTopLeft[0], startBottomRight[1]];
|
const bottomLeft = [startTopLeft[0], startBottomRight[1]];
|
||||||
newTopLeft = [bottomLeft[0], bottomLeft[1] - Math.abs(nextHeight)];
|
newTopLeft = point<GlobalPoint>(
|
||||||
|
bottomLeft[0],
|
||||||
|
bottomLeft[1] - Math.abs(nextHeight),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
if (transformHandleType === "sw") {
|
if (transformHandleType === "sw") {
|
||||||
const topRight = [startBottomRight[0], startTopLeft[1]];
|
const topRight = [startBottomRight[0], startTopLeft[1]];
|
||||||
newTopLeft = [topRight[0] - Math.abs(nextWidth), topRight[1]];
|
newTopLeft = point<GlobalPoint>(
|
||||||
|
topRight[0] - Math.abs(nextWidth),
|
||||||
|
topRight[1],
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (["s", "n"].includes(transformHandleType)) {
|
if (["s", "n"].includes(transformHandleType)) {
|
||||||
|
@ -308,13 +310,17 @@ const resizeSingleTextElement = (
|
||||||
}
|
}
|
||||||
|
|
||||||
const angle = element.angle;
|
const angle = element.angle;
|
||||||
const rotatedTopLeft = rotatePoint(newTopLeft, [cx, cy], angle);
|
const rotatedTopLeft = pointRotateRads(newTopLeft, point(cx, cy), angle);
|
||||||
const newCenter: Point = [
|
const newCenter = point<GlobalPoint>(
|
||||||
newTopLeft[0] + Math.abs(nextWidth) / 2,
|
newTopLeft[0] + Math.abs(nextWidth) / 2,
|
||||||
newTopLeft[1] + Math.abs(nextHeight) / 2,
|
newTopLeft[1] + Math.abs(nextHeight) / 2,
|
||||||
];
|
);
|
||||||
const rotatedNewCenter = rotatePoint(newCenter, [cx, cy], angle);
|
const rotatedNewCenter = pointRotateRads(newCenter, point(cx, cy), angle);
|
||||||
newTopLeft = rotatePoint(rotatedTopLeft, rotatedNewCenter, -angle);
|
newTopLeft = pointRotateRads(
|
||||||
|
rotatedTopLeft,
|
||||||
|
rotatedNewCenter,
|
||||||
|
-angle as Radians,
|
||||||
|
);
|
||||||
const [nextX, nextY] = newTopLeft;
|
const [nextX, nextY] = newTopLeft;
|
||||||
|
|
||||||
mutateElement(element, {
|
mutateElement(element, {
|
||||||
|
@ -334,14 +340,14 @@ const resizeSingleTextElement = (
|
||||||
stateAtResizeStart.height,
|
stateAtResizeStart.height,
|
||||||
true,
|
true,
|
||||||
);
|
);
|
||||||
const startTopLeft: Point = [x1, y1];
|
const startTopLeft = point<GlobalPoint>(x1, y1);
|
||||||
const startBottomRight: Point = [x2, y2];
|
const startBottomRight = point<GlobalPoint>(x2, y2);
|
||||||
const startCenter: Point = centerPoint(startTopLeft, startBottomRight);
|
const startCenter = pointCenter(startTopLeft, startBottomRight);
|
||||||
|
|
||||||
const rotatedPointer = rotatePoint(
|
const rotatedPointer = pointRotateRads(
|
||||||
[pointerX, pointerY],
|
point(pointerX, pointerY),
|
||||||
startCenter,
|
startCenter,
|
||||||
-stateAtResizeStart.angle,
|
-stateAtResizeStart.angle as Radians,
|
||||||
);
|
);
|
||||||
|
|
||||||
const [esx1, , esx2] = getResizedElementAbsoluteCoords(
|
const [esx1, , esx2] = getResizedElementAbsoluteCoords(
|
||||||
|
@ -407,13 +413,21 @@ const resizeSingleTextElement = (
|
||||||
|
|
||||||
// adjust topLeft to new rotation point
|
// adjust topLeft to new rotation point
|
||||||
const angle = stateAtResizeStart.angle;
|
const angle = stateAtResizeStart.angle;
|
||||||
const rotatedTopLeft = rotatePoint(newTopLeft, startCenter, angle);
|
const rotatedTopLeft = pointRotateRads(
|
||||||
const newCenter: Point = [
|
pointFromPair(newTopLeft),
|
||||||
|
startCenter,
|
||||||
|
angle,
|
||||||
|
);
|
||||||
|
const newCenter = point(
|
||||||
newTopLeft[0] + Math.abs(newBoundsWidth) / 2,
|
newTopLeft[0] + Math.abs(newBoundsWidth) / 2,
|
||||||
newTopLeft[1] + Math.abs(newBoundsHeight) / 2,
|
newTopLeft[1] + Math.abs(newBoundsHeight) / 2,
|
||||||
];
|
);
|
||||||
const rotatedNewCenter = rotatePoint(newCenter, startCenter, angle);
|
const rotatedNewCenter = pointRotateRads(newCenter, startCenter, angle);
|
||||||
newTopLeft = rotatePoint(rotatedTopLeft, rotatedNewCenter, -angle);
|
newTopLeft = pointRotateRads(
|
||||||
|
rotatedTopLeft,
|
||||||
|
rotatedNewCenter,
|
||||||
|
-angle as Radians,
|
||||||
|
);
|
||||||
|
|
||||||
const resizedElement: Partial<ExcalidrawTextElement> = {
|
const resizedElement: Partial<ExcalidrawTextElement> = {
|
||||||
width: Math.abs(newWidth),
|
width: Math.abs(newWidth),
|
||||||
|
@ -446,15 +460,15 @@ export const resizeSingleElement = (
|
||||||
stateAtResizeStart.height,
|
stateAtResizeStart.height,
|
||||||
true,
|
true,
|
||||||
);
|
);
|
||||||
const startTopLeft: Point = [x1, y1];
|
const startTopLeft = point(x1, y1);
|
||||||
const startBottomRight: Point = [x2, y2];
|
const startBottomRight = point(x2, y2);
|
||||||
const startCenter: Point = centerPoint(startTopLeft, startBottomRight);
|
const startCenter = pointCenter(startTopLeft, startBottomRight);
|
||||||
|
|
||||||
// Calculate new dimensions based on cursor position
|
// Calculate new dimensions based on cursor position
|
||||||
const rotatedPointer = rotatePoint(
|
const rotatedPointer = pointRotateRads(
|
||||||
[pointerX, pointerY],
|
point(pointerX, pointerY),
|
||||||
startCenter,
|
startCenter,
|
||||||
-stateAtResizeStart.angle,
|
-stateAtResizeStart.angle as Radians,
|
||||||
);
|
);
|
||||||
|
|
||||||
// Get bounds corners rendered on screen
|
// Get bounds corners rendered on screen
|
||||||
|
@ -628,13 +642,21 @@ export const resizeSingleElement = (
|
||||||
|
|
||||||
// adjust topLeft to new rotation point
|
// adjust topLeft to new rotation point
|
||||||
const angle = stateAtResizeStart.angle;
|
const angle = stateAtResizeStart.angle;
|
||||||
const rotatedTopLeft = rotatePoint(newTopLeft, startCenter, angle);
|
const rotatedTopLeft = pointRotateRads(
|
||||||
const newCenter: Point = [
|
pointFromPair(newTopLeft),
|
||||||
|
startCenter,
|
||||||
|
angle,
|
||||||
|
);
|
||||||
|
const newCenter = point(
|
||||||
newTopLeft[0] + Math.abs(newBoundsWidth) / 2,
|
newTopLeft[0] + Math.abs(newBoundsWidth) / 2,
|
||||||
newTopLeft[1] + Math.abs(newBoundsHeight) / 2,
|
newTopLeft[1] + Math.abs(newBoundsHeight) / 2,
|
||||||
];
|
);
|
||||||
const rotatedNewCenter = rotatePoint(newCenter, startCenter, angle);
|
const rotatedNewCenter = pointRotateRads(newCenter, startCenter, angle);
|
||||||
newTopLeft = rotatePoint(rotatedTopLeft, rotatedNewCenter, -angle);
|
newTopLeft = pointRotateRads(
|
||||||
|
rotatedTopLeft,
|
||||||
|
rotatedNewCenter,
|
||||||
|
-angle as Radians,
|
||||||
|
);
|
||||||
|
|
||||||
// For linear elements (x,y) are the coordinates of the first drawn point not the top-left corner
|
// For linear elements (x,y) are the coordinates of the first drawn point not the top-left corner
|
||||||
// So we need to readjust (x,y) to be where the first point should be
|
// So we need to readjust (x,y) to be where the first point should be
|
||||||
|
@ -793,21 +815,21 @@ export const resizeMultipleElements = (
|
||||||
|
|
||||||
const direction = transformHandleType;
|
const direction = transformHandleType;
|
||||||
|
|
||||||
const anchorsMap: Record<TransformHandleDirection, Point> = {
|
const anchorsMap: Record<TransformHandleDirection, GlobalPoint> = {
|
||||||
ne: [minX, maxY],
|
ne: point(minX, maxY),
|
||||||
se: [minX, minY],
|
se: point(minX, minY),
|
||||||
sw: [maxX, minY],
|
sw: point(maxX, minY),
|
||||||
nw: [maxX, maxY],
|
nw: point(maxX, maxY),
|
||||||
e: [minX, minY + height / 2],
|
e: point(minX, minY + height / 2),
|
||||||
w: [maxX, minY + height / 2],
|
w: point(maxX, minY + height / 2),
|
||||||
n: [minX + width / 2, maxY],
|
n: point(minX + width / 2, maxY),
|
||||||
s: [minX + width / 2, minY],
|
s: point(minX + width / 2, minY),
|
||||||
};
|
};
|
||||||
|
|
||||||
// anchor point must be on the opposite side of the dragged selection handle
|
// anchor point must be on the opposite side of the dragged selection handle
|
||||||
// or be the center of the selection if shouldResizeFromCenter
|
// or be the center of the selection if shouldResizeFromCenter
|
||||||
const [anchorX, anchorY]: Point = shouldResizeFromCenter
|
const [anchorX, anchorY] = shouldResizeFromCenter
|
||||||
? [midX, midY]
|
? point(midX, midY)
|
||||||
: anchorsMap[direction];
|
: anchorsMap[direction];
|
||||||
|
|
||||||
const resizeFromCenterScale = shouldResizeFromCenter ? 2 : 1;
|
const resizeFromCenterScale = shouldResizeFromCenter ? 2 : 1;
|
||||||
|
@ -898,7 +920,9 @@ export const resizeMultipleElements = (
|
||||||
|
|
||||||
const width = orig.width * scaleX;
|
const width = orig.width * scaleX;
|
||||||
const height = orig.height * scaleY;
|
const height = orig.height * scaleY;
|
||||||
const angle = normalizeAngle(orig.angle * flipFactorX * flipFactorY);
|
const angle = normalizeRadians(
|
||||||
|
(orig.angle * flipFactorX * flipFactorY) as Radians,
|
||||||
|
);
|
||||||
|
|
||||||
const isLinearOrFreeDraw = isLinearElement(orig) || isFreeDrawElement(orig);
|
const isLinearOrFreeDraw = isLinearElement(orig) || isFreeDrawElement(orig);
|
||||||
const offsetX = orig.x - anchorX;
|
const offsetX = orig.x - anchorX;
|
||||||
|
@ -1029,12 +1053,10 @@ const rotateMultipleElements = (
|
||||||
const cy = (y1 + y2) / 2;
|
const cy = (y1 + y2) / 2;
|
||||||
const origAngle =
|
const origAngle =
|
||||||
originalElements.get(element.id)?.angle ?? element.angle;
|
originalElements.get(element.id)?.angle ?? element.angle;
|
||||||
const [rotatedCX, rotatedCY] = rotate(
|
const [rotatedCX, rotatedCY] = pointRotateRads(
|
||||||
cx,
|
point(cx, cy),
|
||||||
cy,
|
point(centerX, centerY),
|
||||||
centerX,
|
(centerAngle + origAngle - element.angle) as Radians,
|
||||||
centerY,
|
|
||||||
centerAngle + origAngle - element.angle,
|
|
||||||
);
|
);
|
||||||
|
|
||||||
if (isArrowElement(element) && isElbowArrow(element)) {
|
if (isArrowElement(element) && isElbowArrow(element)) {
|
||||||
|
@ -1046,7 +1068,7 @@ const rotateMultipleElements = (
|
||||||
{
|
{
|
||||||
x: element.x + (rotatedCX - cx),
|
x: element.x + (rotatedCX - cx),
|
||||||
y: element.y + (rotatedCY - cy),
|
y: element.y + (rotatedCY - cy),
|
||||||
angle: normalizeAngle(centerAngle + origAngle),
|
angle: normalizeRadians((centerAngle + origAngle) as Radians),
|
||||||
},
|
},
|
||||||
false,
|
false,
|
||||||
);
|
);
|
||||||
|
@ -1063,7 +1085,7 @@ const rotateMultipleElements = (
|
||||||
{
|
{
|
||||||
x: boundText.x + (rotatedCX - cx),
|
x: boundText.x + (rotatedCX - cx),
|
||||||
y: boundText.y + (rotatedCY - cy),
|
y: boundText.y + (rotatedCY - cy),
|
||||||
angle: normalizeAngle(centerAngle + origAngle),
|
angle: normalizeRadians((centerAngle + origAngle) as Radians),
|
||||||
},
|
},
|
||||||
false,
|
false,
|
||||||
);
|
);
|
||||||
|
@ -1086,25 +1108,43 @@ export const getResizeOffsetXY = (
|
||||||
: getCommonBounds(selectedElements);
|
: getCommonBounds(selectedElements);
|
||||||
const cx = (x1 + x2) / 2;
|
const cx = (x1 + x2) / 2;
|
||||||
const cy = (y1 + y2) / 2;
|
const cy = (y1 + y2) / 2;
|
||||||
const angle = selectedElements.length === 1 ? selectedElements[0].angle : 0;
|
const angle = (
|
||||||
[x, y] = rotate(x, y, cx, cy, -angle);
|
selectedElements.length === 1 ? selectedElements[0].angle : 0
|
||||||
|
) as Radians;
|
||||||
|
[x, y] = pointRotateRads(point(x, y), point(cx, cy), -angle as Radians);
|
||||||
switch (transformHandleType) {
|
switch (transformHandleType) {
|
||||||
case "n":
|
case "n":
|
||||||
return rotate(x - (x1 + x2) / 2, y - y1, 0, 0, angle);
|
return pointRotateRads(
|
||||||
|
point(x - (x1 + x2) / 2, y - y1),
|
||||||
|
point(0, 0),
|
||||||
|
angle,
|
||||||
|
);
|
||||||
case "s":
|
case "s":
|
||||||
return rotate(x - (x1 + x2) / 2, y - y2, 0, 0, angle);
|
return pointRotateRads(
|
||||||
|
point(x - (x1 + x2) / 2, y - y2),
|
||||||
|
point(0, 0),
|
||||||
|
angle,
|
||||||
|
);
|
||||||
case "w":
|
case "w":
|
||||||
return rotate(x - x1, y - (y1 + y2) / 2, 0, 0, angle);
|
return pointRotateRads(
|
||||||
|
point(x - x1, y - (y1 + y2) / 2),
|
||||||
|
point(0, 0),
|
||||||
|
angle,
|
||||||
|
);
|
||||||
case "e":
|
case "e":
|
||||||
return rotate(x - x2, y - (y1 + y2) / 2, 0, 0, angle);
|
return pointRotateRads(
|
||||||
|
point(x - x2, y - (y1 + y2) / 2),
|
||||||
|
point(0, 0),
|
||||||
|
angle,
|
||||||
|
);
|
||||||
case "nw":
|
case "nw":
|
||||||
return rotate(x - x1, y - y1, 0, 0, angle);
|
return pointRotateRads(point(x - x1, y - y1), point(0, 0), angle);
|
||||||
case "ne":
|
case "ne":
|
||||||
return rotate(x - x2, y - y1, 0, 0, angle);
|
return pointRotateRads(point(x - x2, y - y1), point(0, 0), angle);
|
||||||
case "sw":
|
case "sw":
|
||||||
return rotate(x - x1, y - y2, 0, 0, angle);
|
return pointRotateRads(point(x - x1, y - y2), point(0, 0), angle);
|
||||||
case "se":
|
case "se":
|
||||||
return rotate(x - x2, y - y2, 0, 0, angle);
|
return pointRotateRads(point(x - x2, y - y2), point(0, 0), angle);
|
||||||
default:
|
default:
|
||||||
return [0, 0];
|
return [0, 0];
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,13 +20,14 @@ import type { AppState, Device, Zoom } from "../types";
|
||||||
import type { Bounds } from "./bounds";
|
import type { Bounds } from "./bounds";
|
||||||
import { getElementAbsoluteCoords } from "./bounds";
|
import { getElementAbsoluteCoords } from "./bounds";
|
||||||
import { SIDE_RESIZING_THRESHOLD } from "../constants";
|
import { SIDE_RESIZING_THRESHOLD } from "../constants";
|
||||||
import {
|
|
||||||
angleToDegrees,
|
|
||||||
pointOnLine,
|
|
||||||
pointRotate,
|
|
||||||
} from "../../utils/geometry/geometry";
|
|
||||||
import type { Line, Point } from "../../utils/geometry/shape";
|
|
||||||
import { isLinearElement } from "./typeChecks";
|
import { isLinearElement } from "./typeChecks";
|
||||||
|
import type { GlobalPoint, LineSegment, LocalPoint } from "../../math";
|
||||||
|
import {
|
||||||
|
point,
|
||||||
|
pointOnLineSegment,
|
||||||
|
pointRotateRads,
|
||||||
|
type Radians,
|
||||||
|
} from "../../math";
|
||||||
|
|
||||||
const isInsideTransformHandle = (
|
const isInsideTransformHandle = (
|
||||||
transformHandle: TransformHandle,
|
transformHandle: TransformHandle,
|
||||||
|
@ -38,7 +39,7 @@ const isInsideTransformHandle = (
|
||||||
y >= transformHandle[1] &&
|
y >= transformHandle[1] &&
|
||||||
y <= transformHandle[1] + transformHandle[3];
|
y <= transformHandle[1] + transformHandle[3];
|
||||||
|
|
||||||
export const resizeTest = (
|
export const resizeTest = <Point extends GlobalPoint | LocalPoint>(
|
||||||
element: NonDeletedExcalidrawElement,
|
element: NonDeletedExcalidrawElement,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
|
@ -91,15 +92,17 @@ export const resizeTest = (
|
||||||
if (!(isLinearElement(element) && element.points.length <= 2)) {
|
if (!(isLinearElement(element) && element.points.length <= 2)) {
|
||||||
const SPACING = SIDE_RESIZING_THRESHOLD / zoom.value;
|
const SPACING = SIDE_RESIZING_THRESHOLD / zoom.value;
|
||||||
const sides = getSelectionBorders(
|
const sides = getSelectionBorders(
|
||||||
[x1 - SPACING, y1 - SPACING],
|
point(x1 - SPACING, y1 - SPACING),
|
||||||
[x2 + SPACING, y2 + SPACING],
|
point(x2 + SPACING, y2 + SPACING),
|
||||||
[cx, cy],
|
point(cx, cy),
|
||||||
angleToDegrees(element.angle),
|
element.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
for (const [dir, side] of Object.entries(sides)) {
|
for (const [dir, side] of Object.entries(sides)) {
|
||||||
// test to see if x, y are on the line segment
|
// test to see if x, y are on the line segment
|
||||||
if (pointOnLine([x, y], side as Line, SPACING)) {
|
if (
|
||||||
|
pointOnLineSegment(point(x, y), side as LineSegment<Point>, SPACING)
|
||||||
|
) {
|
||||||
return dir as TransformHandleType;
|
return dir as TransformHandleType;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -137,7 +140,9 @@ export const getElementWithTransformHandleType = (
|
||||||
}, null as { element: NonDeletedExcalidrawElement; transformHandleType: MaybeTransformHandleType } | null);
|
}, null as { element: NonDeletedExcalidrawElement; transformHandleType: MaybeTransformHandleType } | null);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getTransformHandleTypeFromCoords = (
|
export const getTransformHandleTypeFromCoords = <
|
||||||
|
Point extends GlobalPoint | LocalPoint,
|
||||||
|
>(
|
||||||
[x1, y1, x2, y2]: Bounds,
|
[x1, y1, x2, y2]: Bounds,
|
||||||
scenePointerX: number,
|
scenePointerX: number,
|
||||||
scenePointerY: number,
|
scenePointerY: number,
|
||||||
|
@ -147,7 +152,7 @@ export const getTransformHandleTypeFromCoords = (
|
||||||
): MaybeTransformHandleType => {
|
): MaybeTransformHandleType => {
|
||||||
const transformHandles = getTransformHandlesFromCoords(
|
const transformHandles = getTransformHandlesFromCoords(
|
||||||
[x1, y1, x2, y2, (x1 + x2) / 2, (y1 + y2) / 2],
|
[x1, y1, x2, y2, (x1 + x2) / 2, (y1 + y2) / 2],
|
||||||
0,
|
0 as Radians,
|
||||||
zoom,
|
zoom,
|
||||||
pointerType,
|
pointerType,
|
||||||
getOmitSidesForDevice(device),
|
getOmitSidesForDevice(device),
|
||||||
|
@ -173,15 +178,21 @@ export const getTransformHandleTypeFromCoords = (
|
||||||
const SPACING = SIDE_RESIZING_THRESHOLD / zoom.value;
|
const SPACING = SIDE_RESIZING_THRESHOLD / zoom.value;
|
||||||
|
|
||||||
const sides = getSelectionBorders(
|
const sides = getSelectionBorders(
|
||||||
[x1 - SPACING, y1 - SPACING],
|
point(x1 - SPACING, y1 - SPACING),
|
||||||
[x2 + SPACING, y2 + SPACING],
|
point(x2 + SPACING, y2 + SPACING),
|
||||||
[cx, cy],
|
point(cx, cy),
|
||||||
angleToDegrees(0),
|
0 as Radians,
|
||||||
);
|
);
|
||||||
|
|
||||||
for (const [dir, side] of Object.entries(sides)) {
|
for (const [dir, side] of Object.entries(sides)) {
|
||||||
// test to see if x, y are on the line segment
|
// test to see if x, y are on the line segment
|
||||||
if (pointOnLine([scenePointerX, scenePointerY], side as Line, SPACING)) {
|
if (
|
||||||
|
pointOnLineSegment(
|
||||||
|
point(scenePointerX, scenePointerY),
|
||||||
|
side as LineSegment<Point>,
|
||||||
|
SPACING,
|
||||||
|
)
|
||||||
|
) {
|
||||||
return dir as TransformHandleType;
|
return dir as TransformHandleType;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -248,16 +259,16 @@ export const getCursorForResizingElement = (resizingElement: {
|
||||||
return cursor ? `${cursor}-resize` : "";
|
return cursor ? `${cursor}-resize` : "";
|
||||||
};
|
};
|
||||||
|
|
||||||
const getSelectionBorders = (
|
const getSelectionBorders = <Point extends LocalPoint | GlobalPoint>(
|
||||||
[x1, y1]: Point,
|
[x1, y1]: Point,
|
||||||
[x2, y2]: Point,
|
[x2, y2]: Point,
|
||||||
center: Point,
|
center: Point,
|
||||||
angleInDegrees: number,
|
angle: Radians,
|
||||||
) => {
|
) => {
|
||||||
const topLeft = pointRotate([x1, y1], angleInDegrees, center);
|
const topLeft = pointRotateRads(point(x1, y1), center, angle);
|
||||||
const topRight = pointRotate([x2, y1], angleInDegrees, center);
|
const topRight = pointRotateRads(point(x2, y1), center, angle);
|
||||||
const bottomLeft = pointRotate([x1, y2], angleInDegrees, center);
|
const bottomLeft = pointRotateRads(point(x1, y2), center, angle);
|
||||||
const bottomRight = pointRotate([x2, y2], angleInDegrees, center);
|
const bottomRight = pointRotateRads(point(x2, y2), center, angle);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
n: [topLeft, topRight],
|
n: [topLeft, topRight],
|
||||||
|
|
|
@ -17,6 +17,7 @@ import type {
|
||||||
ExcalidrawElbowArrowElement,
|
ExcalidrawElbowArrowElement,
|
||||||
} from "./types";
|
} from "./types";
|
||||||
import { ARROW_TYPE } from "../constants";
|
import { ARROW_TYPE } from "../constants";
|
||||||
|
import { point } from "../../math";
|
||||||
|
|
||||||
const { h } = window;
|
const { h } = window;
|
||||||
|
|
||||||
|
@ -31,8 +32,8 @@ describe("elbow arrow routing", () => {
|
||||||
}) as ExcalidrawElbowArrowElement;
|
}) as ExcalidrawElbowArrowElement;
|
||||||
scene.insertElement(arrow);
|
scene.insertElement(arrow);
|
||||||
mutateElbowArrow(arrow, scene.getNonDeletedElementsMap(), [
|
mutateElbowArrow(arrow, scene.getNonDeletedElementsMap(), [
|
||||||
[-45 - arrow.x, -100.1 - arrow.y],
|
point(-45 - arrow.x, -100.1 - arrow.y),
|
||||||
[45 - arrow.x, 99.9 - arrow.y],
|
point(45 - arrow.x, 99.9 - arrow.y),
|
||||||
]);
|
]);
|
||||||
expect(arrow.points).toEqual([
|
expect(arrow.points).toEqual([
|
||||||
[0, 0],
|
[0, 0],
|
||||||
|
@ -68,10 +69,7 @@ describe("elbow arrow routing", () => {
|
||||||
y: -100.1,
|
y: -100.1,
|
||||||
width: 90,
|
width: 90,
|
||||||
height: 200,
|
height: 200,
|
||||||
points: [
|
points: [point(0, 0), point(90, 200)],
|
||||||
[0, 0],
|
|
||||||
[90, 200],
|
|
||||||
],
|
|
||||||
}) as ExcalidrawElbowArrowElement;
|
}) as ExcalidrawElbowArrowElement;
|
||||||
scene.insertElement(rectangle1);
|
scene.insertElement(rectangle1);
|
||||||
scene.insertElement(rectangle2);
|
scene.insertElement(rectangle2);
|
||||||
|
@ -83,10 +81,7 @@ describe("elbow arrow routing", () => {
|
||||||
expect(arrow.startBinding).not.toBe(null);
|
expect(arrow.startBinding).not.toBe(null);
|
||||||
expect(arrow.endBinding).not.toBe(null);
|
expect(arrow.endBinding).not.toBe(null);
|
||||||
|
|
||||||
mutateElbowArrow(arrow, elementsMap, [
|
mutateElbowArrow(arrow, elementsMap, [point(0, 0), point(90, 200)]);
|
||||||
[0, 0],
|
|
||||||
[90, 200],
|
|
||||||
]);
|
|
||||||
|
|
||||||
expect(arrow.points).toEqual([
|
expect(arrow.points).toEqual([
|
||||||
[0, 0],
|
[0, 0],
|
||||||
|
|
|
@ -1,16 +1,19 @@
|
||||||
import { cross } from "../../utils/geometry/geometry";
|
import type { Radians } from "../../math";
|
||||||
import BinaryHeap from "../binaryheap";
|
|
||||||
import {
|
import {
|
||||||
aabbForElement,
|
point,
|
||||||
arePointsEqual,
|
pointScaleFromOrigin,
|
||||||
pointInsideBounds,
|
pointTranslate,
|
||||||
pointToVector,
|
vector,
|
||||||
scalePointFromOrigin,
|
vectorCross,
|
||||||
scaleVector,
|
vectorFromPoint,
|
||||||
translatePoint,
|
vectorScale,
|
||||||
} from "../math";
|
type GlobalPoint,
|
||||||
|
type LocalPoint,
|
||||||
|
type Vector,
|
||||||
|
} from "../../math";
|
||||||
|
import BinaryHeap from "../binaryheap";
|
||||||
import { getSizeFromPoints } from "../points";
|
import { getSizeFromPoints } from "../points";
|
||||||
import type { Point } from "../types";
|
import { aabbForElement, pointInsideBounds } from "../shapes";
|
||||||
import { isAnyTrue, toBrandedType, tupleToCoors } from "../utils";
|
import { isAnyTrue, toBrandedType, tupleToCoors } from "../utils";
|
||||||
import {
|
import {
|
||||||
bindPointToSnapToElementOutline,
|
bindPointToSnapToElementOutline,
|
||||||
|
@ -25,6 +28,8 @@ import {
|
||||||
import type { Bounds } from "./bounds";
|
import type { Bounds } from "./bounds";
|
||||||
import type { Heading } from "./heading";
|
import type { Heading } from "./heading";
|
||||||
import {
|
import {
|
||||||
|
compareHeading,
|
||||||
|
flipHeading,
|
||||||
HEADING_DOWN,
|
HEADING_DOWN,
|
||||||
HEADING_LEFT,
|
HEADING_LEFT,
|
||||||
HEADING_RIGHT,
|
HEADING_RIGHT,
|
||||||
|
@ -41,6 +46,8 @@ import type {
|
||||||
} from "./types";
|
} from "./types";
|
||||||
import type { ElementsMap, ExcalidrawBindableElement } from "./types";
|
import type { ElementsMap, ExcalidrawBindableElement } from "./types";
|
||||||
|
|
||||||
|
type GridAddress = [number, number] & { _brand: "gridaddress" };
|
||||||
|
|
||||||
type Node = {
|
type Node = {
|
||||||
f: number;
|
f: number;
|
||||||
g: number;
|
g: number;
|
||||||
|
@ -48,8 +55,8 @@ type Node = {
|
||||||
closed: boolean;
|
closed: boolean;
|
||||||
visited: boolean;
|
visited: boolean;
|
||||||
parent: Node | null;
|
parent: Node | null;
|
||||||
pos: Point;
|
pos: GlobalPoint;
|
||||||
addr: [number, number];
|
addr: GridAddress;
|
||||||
};
|
};
|
||||||
|
|
||||||
type Grid = {
|
type Grid = {
|
||||||
|
@ -63,8 +70,8 @@ const BASE_PADDING = 40;
|
||||||
export const mutateElbowArrow = (
|
export const mutateElbowArrow = (
|
||||||
arrow: ExcalidrawElbowArrowElement,
|
arrow: ExcalidrawElbowArrowElement,
|
||||||
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
||||||
nextPoints: readonly Point[],
|
nextPoints: readonly LocalPoint[],
|
||||||
offset?: Point,
|
offset?: Vector,
|
||||||
otherUpdates?: {
|
otherUpdates?: {
|
||||||
startBinding?: FixedPointBinding | null;
|
startBinding?: FixedPointBinding | null;
|
||||||
endBinding?: FixedPointBinding | null;
|
endBinding?: FixedPointBinding | null;
|
||||||
|
@ -75,14 +82,20 @@ export const mutateElbowArrow = (
|
||||||
informMutation?: boolean;
|
informMutation?: boolean;
|
||||||
},
|
},
|
||||||
) => {
|
) => {
|
||||||
const origStartGlobalPoint = translatePoint(nextPoints[0], [
|
const origStartGlobalPoint: GlobalPoint = pointTranslate(
|
||||||
arrow.x + (offset ? offset[0] : 0),
|
pointTranslate<LocalPoint, GlobalPoint>(
|
||||||
arrow.y + (offset ? offset[1] : 0),
|
nextPoints[0],
|
||||||
]);
|
vector(arrow.x, arrow.y),
|
||||||
const origEndGlobalPoint = translatePoint(nextPoints[nextPoints.length - 1], [
|
),
|
||||||
arrow.x + (offset ? offset[0] : 0),
|
offset,
|
||||||
arrow.y + (offset ? offset[1] : 0),
|
);
|
||||||
]);
|
const origEndGlobalPoint: GlobalPoint = pointTranslate(
|
||||||
|
pointTranslate<LocalPoint, GlobalPoint>(
|
||||||
|
nextPoints[nextPoints.length - 1],
|
||||||
|
vector(arrow.x, arrow.y),
|
||||||
|
),
|
||||||
|
offset,
|
||||||
|
);
|
||||||
|
|
||||||
const startElement =
|
const startElement =
|
||||||
arrow.startBinding &&
|
arrow.startBinding &&
|
||||||
|
@ -275,7 +288,10 @@ export const mutateElbowArrow = (
|
||||||
);
|
);
|
||||||
|
|
||||||
if (path) {
|
if (path) {
|
||||||
const points = path.map((node) => [node.pos[0], node.pos[1]]) as Point[];
|
const points = path.map((node) => [
|
||||||
|
node.pos[0],
|
||||||
|
node.pos[1],
|
||||||
|
]) as GlobalPoint[];
|
||||||
startDongle && points.unshift(startGlobalPoint);
|
startDongle && points.unshift(startGlobalPoint);
|
||||||
endDongle && points.push(endGlobalPoint);
|
endDongle && points.push(endGlobalPoint);
|
||||||
|
|
||||||
|
@ -284,7 +300,7 @@ export const mutateElbowArrow = (
|
||||||
{
|
{
|
||||||
...otherUpdates,
|
...otherUpdates,
|
||||||
...normalizedArrowElementUpdate(simplifyElbowArrowPoints(points), 0, 0),
|
...normalizedArrowElementUpdate(simplifyElbowArrowPoints(points), 0, 0),
|
||||||
angle: 0,
|
angle: 0 as Radians,
|
||||||
},
|
},
|
||||||
options?.informMutation,
|
options?.informMutation,
|
||||||
);
|
);
|
||||||
|
@ -363,7 +379,7 @@ const astar = (
|
||||||
}
|
}
|
||||||
|
|
||||||
// Intersect
|
// Intersect
|
||||||
const neighborHalfPoint = scalePointFromOrigin(
|
const neighborHalfPoint = pointScaleFromOrigin(
|
||||||
neighbor.pos,
|
neighbor.pos,
|
||||||
current.pos,
|
current.pos,
|
||||||
0.5,
|
0.5,
|
||||||
|
@ -380,17 +396,17 @@ const astar = (
|
||||||
// We need to check if the path we have arrived at this neighbor is the shortest one we have seen yet.
|
// We need to check if the path we have arrived at this neighbor is the shortest one we have seen yet.
|
||||||
const neighborHeading = neighborIndexToHeading(i as 0 | 1 | 2 | 3);
|
const neighborHeading = neighborIndexToHeading(i as 0 | 1 | 2 | 3);
|
||||||
const previousDirection = current.parent
|
const previousDirection = current.parent
|
||||||
? vectorToHeading(pointToVector(current.pos, current.parent.pos))
|
? vectorToHeading(vectorFromPoint(current.pos, current.parent.pos))
|
||||||
: startHeading;
|
: startHeading;
|
||||||
|
|
||||||
// Do not allow going in reverse
|
// Do not allow going in reverse
|
||||||
const reverseHeading = scaleVector(previousDirection, -1);
|
const reverseHeading = flipHeading(previousDirection);
|
||||||
const neighborIsReverseRoute =
|
const neighborIsReverseRoute =
|
||||||
arePointsEqual(reverseHeading, neighborHeading) ||
|
compareHeading(reverseHeading, neighborHeading) ||
|
||||||
(arePointsEqual(start.addr, neighbor.addr) &&
|
(gridAddressesEqual(start.addr, neighbor.addr) &&
|
||||||
arePointsEqual(neighborHeading, startHeading)) ||
|
compareHeading(neighborHeading, startHeading)) ||
|
||||||
(arePointsEqual(end.addr, neighbor.addr) &&
|
(gridAddressesEqual(end.addr, neighbor.addr) &&
|
||||||
arePointsEqual(neighborHeading, endHeading));
|
compareHeading(neighborHeading, endHeading));
|
||||||
if (neighborIsReverseRoute) {
|
if (neighborIsReverseRoute) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -444,7 +460,7 @@ const pathTo = (start: Node, node: Node) => {
|
||||||
return path;
|
return path;
|
||||||
};
|
};
|
||||||
|
|
||||||
const m_dist = (a: Point, b: Point) =>
|
const m_dist = (a: GlobalPoint | LocalPoint, b: GlobalPoint | LocalPoint) =>
|
||||||
Math.abs(a[0] - b[0]) + Math.abs(a[1] - b[1]);
|
Math.abs(a[0] - b[0]) + Math.abs(a[1] - b[1]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -541,7 +557,12 @@ const generateDynamicAABBs = (
|
||||||
const cX = first[2] + (second[0] - first[2]) / 2;
|
const cX = first[2] + (second[0] - first[2]) / 2;
|
||||||
const cY = second[3] + (first[1] - second[3]) / 2;
|
const cY = second[3] + (first[1] - second[3]) / 2;
|
||||||
|
|
||||||
if (cross([a[2], a[1]], [a[0], a[3]], [endCenterX, endCenterY]) > 0) {
|
if (
|
||||||
|
vectorCross(
|
||||||
|
vector(a[2] - endCenterX, a[1] - endCenterY),
|
||||||
|
vector(a[0] - endCenterX, a[3] - endCenterY),
|
||||||
|
) > 0
|
||||||
|
) {
|
||||||
return [
|
return [
|
||||||
[first[0], first[1], cX, first[3]],
|
[first[0], first[1], cX, first[3]],
|
||||||
[cX, second[1], second[2], second[3]],
|
[cX, second[1], second[2], second[3]],
|
||||||
|
@ -557,7 +578,12 @@ const generateDynamicAABBs = (
|
||||||
const cX = first[2] + (second[0] - first[2]) / 2;
|
const cX = first[2] + (second[0] - first[2]) / 2;
|
||||||
const cY = first[3] + (second[1] - first[3]) / 2;
|
const cY = first[3] + (second[1] - first[3]) / 2;
|
||||||
|
|
||||||
if (cross([a[0], a[1]], [a[2], a[3]], [endCenterX, endCenterY]) > 0) {
|
if (
|
||||||
|
vectorCross(
|
||||||
|
vector(a[0] - endCenterX, a[1] - endCenterY),
|
||||||
|
vector(a[2] - endCenterX, a[3] - endCenterY),
|
||||||
|
) > 0
|
||||||
|
) {
|
||||||
return [
|
return [
|
||||||
[first[0], first[1], first[2], cY],
|
[first[0], first[1], first[2], cY],
|
||||||
[second[0], cY, second[2], second[3]],
|
[second[0], cY, second[2], second[3]],
|
||||||
|
@ -573,7 +599,12 @@ const generateDynamicAABBs = (
|
||||||
const cX = second[2] + (first[0] - second[2]) / 2;
|
const cX = second[2] + (first[0] - second[2]) / 2;
|
||||||
const cY = first[3] + (second[1] - first[3]) / 2;
|
const cY = first[3] + (second[1] - first[3]) / 2;
|
||||||
|
|
||||||
if (cross([a[2], a[1]], [a[0], a[3]], [endCenterX, endCenterY]) > 0) {
|
if (
|
||||||
|
vectorCross(
|
||||||
|
vector(a[2] - endCenterX, a[1] - endCenterY),
|
||||||
|
vector(a[0] - endCenterX, a[3] - endCenterY),
|
||||||
|
) > 0
|
||||||
|
) {
|
||||||
return [
|
return [
|
||||||
[cX, first[1], first[2], first[3]],
|
[cX, first[1], first[2], first[3]],
|
||||||
[second[0], second[1], cX, second[3]],
|
[second[0], second[1], cX, second[3]],
|
||||||
|
@ -589,7 +620,12 @@ const generateDynamicAABBs = (
|
||||||
const cX = second[2] + (first[0] - second[2]) / 2;
|
const cX = second[2] + (first[0] - second[2]) / 2;
|
||||||
const cY = second[3] + (first[1] - second[3]) / 2;
|
const cY = second[3] + (first[1] - second[3]) / 2;
|
||||||
|
|
||||||
if (cross([a[0], a[1]], [a[2], a[3]], [endCenterX, endCenterY]) > 0) {
|
if (
|
||||||
|
vectorCross(
|
||||||
|
vector(a[0] - endCenterX, a[1] - endCenterY),
|
||||||
|
vector(a[2] - endCenterX, a[3] - endCenterY),
|
||||||
|
) > 0
|
||||||
|
) {
|
||||||
return [
|
return [
|
||||||
[cX, first[1], first[2], first[3]],
|
[cX, first[1], first[2], first[3]],
|
||||||
[second[0], second[1], cX, second[3]],
|
[second[0], second[1], cX, second[3]],
|
||||||
|
@ -615,9 +651,9 @@ const generateDynamicAABBs = (
|
||||||
*/
|
*/
|
||||||
const calculateGrid = (
|
const calculateGrid = (
|
||||||
aabbs: Bounds[],
|
aabbs: Bounds[],
|
||||||
start: Point,
|
start: GlobalPoint,
|
||||||
startHeading: Heading,
|
startHeading: Heading,
|
||||||
end: Point,
|
end: GlobalPoint,
|
||||||
endHeading: Heading,
|
endHeading: Heading,
|
||||||
common: Bounds,
|
common: Bounds,
|
||||||
): Grid => {
|
): Grid => {
|
||||||
|
@ -662,8 +698,8 @@ const calculateGrid = (
|
||||||
closed: false,
|
closed: false,
|
||||||
visited: false,
|
visited: false,
|
||||||
parent: null,
|
parent: null,
|
||||||
addr: [col, row] as [number, number],
|
addr: [col, row] as GridAddress,
|
||||||
pos: [x, y] as Point,
|
pos: [x, y] as GlobalPoint,
|
||||||
}),
|
}),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
@ -673,17 +709,17 @@ const calculateGrid = (
|
||||||
const getDonglePosition = (
|
const getDonglePosition = (
|
||||||
bounds: Bounds,
|
bounds: Bounds,
|
||||||
heading: Heading,
|
heading: Heading,
|
||||||
point: Point,
|
p: GlobalPoint,
|
||||||
): Point => {
|
): GlobalPoint => {
|
||||||
switch (heading) {
|
switch (heading) {
|
||||||
case HEADING_UP:
|
case HEADING_UP:
|
||||||
return [point[0], bounds[1]];
|
return point(p[0], bounds[1]);
|
||||||
case HEADING_RIGHT:
|
case HEADING_RIGHT:
|
||||||
return [bounds[2], point[1]];
|
return point(bounds[2], p[1]);
|
||||||
case HEADING_DOWN:
|
case HEADING_DOWN:
|
||||||
return [point[0], bounds[3]];
|
return point(p[0], bounds[3]);
|
||||||
}
|
}
|
||||||
return [bounds[0], point[1]];
|
return point(bounds[0], p[1]);
|
||||||
};
|
};
|
||||||
|
|
||||||
const estimateSegmentCount = (
|
const estimateSegmentCount = (
|
||||||
|
@ -826,7 +862,7 @@ const gridNodeFromAddr = (
|
||||||
/**
|
/**
|
||||||
* Get node for global point on canvas (if exists)
|
* Get node for global point on canvas (if exists)
|
||||||
*/
|
*/
|
||||||
const pointToGridNode = (point: Point, grid: Grid): Node | null => {
|
const pointToGridNode = (point: GlobalPoint, grid: Grid): Node | null => {
|
||||||
for (let col = 0; col < grid.col; col++) {
|
for (let col = 0; col < grid.col; col++) {
|
||||||
for (let row = 0; row < grid.row; row++) {
|
for (let row = 0; row < grid.row; row++) {
|
||||||
const candidate = gridNodeFromAddr([col, row], grid);
|
const candidate = gridNodeFromAddr([col, row], grid);
|
||||||
|
@ -865,15 +901,24 @@ const getBindableElementForId = (
|
||||||
};
|
};
|
||||||
|
|
||||||
const normalizedArrowElementUpdate = (
|
const normalizedArrowElementUpdate = (
|
||||||
global: Point[],
|
global: GlobalPoint[],
|
||||||
externalOffsetX?: number,
|
externalOffsetX?: number,
|
||||||
externalOffsetY?: number,
|
externalOffsetY?: number,
|
||||||
) => {
|
): {
|
||||||
|
points: LocalPoint[];
|
||||||
|
x: number;
|
||||||
|
y: number;
|
||||||
|
width: number;
|
||||||
|
height: number;
|
||||||
|
} => {
|
||||||
const offsetX = global[0][0];
|
const offsetX = global[0][0];
|
||||||
const offsetY = global[0][1];
|
const offsetY = global[0][1];
|
||||||
|
|
||||||
const points = global.map(
|
const points = global.map((p) =>
|
||||||
(point) => [point[0] - offsetX, point[1] - offsetY] as const,
|
pointTranslate<GlobalPoint, LocalPoint>(
|
||||||
|
p,
|
||||||
|
vectorScale(vectorFromPoint(global[0]), -1),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
@ -885,19 +930,22 @@ const normalizedArrowElementUpdate = (
|
||||||
};
|
};
|
||||||
|
|
||||||
/// If last and current segments have the same heading, skip the middle point
|
/// If last and current segments have the same heading, skip the middle point
|
||||||
const simplifyElbowArrowPoints = (points: Point[]): Point[] =>
|
const simplifyElbowArrowPoints = (points: GlobalPoint[]): GlobalPoint[] =>
|
||||||
points
|
points
|
||||||
.slice(2)
|
.slice(2)
|
||||||
.reduce(
|
.reduce(
|
||||||
(result, point) =>
|
(result, p) =>
|
||||||
arePointsEqual(
|
compareHeading(
|
||||||
vectorToHeading(
|
vectorToHeading(
|
||||||
pointToVector(result[result.length - 1], result[result.length - 2]),
|
vectorFromPoint(
|
||||||
|
result[result.length - 1],
|
||||||
|
result[result.length - 2],
|
||||||
),
|
),
|
||||||
vectorToHeading(pointToVector(point, result[result.length - 1])),
|
),
|
||||||
|
vectorToHeading(vectorFromPoint(p, result[result.length - 1])),
|
||||||
)
|
)
|
||||||
? [...result.slice(0, -1), point]
|
? [...result.slice(0, -1), p]
|
||||||
: [...result, point],
|
: [...result, p],
|
||||||
[points[0] ?? [0, 0], points[1] ?? [1, 0]],
|
[points[0] ?? [0, 0], points[1] ?? [1, 0]],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -915,13 +963,13 @@ const neighborIndexToHeading = (idx: number): Heading => {
|
||||||
|
|
||||||
const getGlobalPoint = (
|
const getGlobalPoint = (
|
||||||
fixedPointRatio: [number, number] | undefined | null,
|
fixedPointRatio: [number, number] | undefined | null,
|
||||||
initialPoint: Point,
|
initialPoint: GlobalPoint,
|
||||||
otherPoint: Point,
|
otherPoint: GlobalPoint,
|
||||||
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
||||||
boundElement?: ExcalidrawBindableElement | null,
|
boundElement?: ExcalidrawBindableElement | null,
|
||||||
hoveredElement?: ExcalidrawBindableElement | null,
|
hoveredElement?: ExcalidrawBindableElement | null,
|
||||||
isDragging?: boolean,
|
isDragging?: boolean,
|
||||||
): Point => {
|
): GlobalPoint => {
|
||||||
if (isDragging) {
|
if (isDragging) {
|
||||||
if (hoveredElement) {
|
if (hoveredElement) {
|
||||||
const snapPoint = getSnapPoint(
|
const snapPoint = getSnapPoint(
|
||||||
|
@ -956,36 +1004,34 @@ const getGlobalPoint = (
|
||||||
};
|
};
|
||||||
|
|
||||||
const getSnapPoint = (
|
const getSnapPoint = (
|
||||||
point: Point,
|
p: GlobalPoint,
|
||||||
otherPoint: Point,
|
otherPoint: GlobalPoint,
|
||||||
element: ExcalidrawBindableElement,
|
element: ExcalidrawBindableElement,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
) =>
|
) =>
|
||||||
bindPointToSnapToElementOutline(
|
bindPointToSnapToElementOutline(
|
||||||
isRectanguloidElement(element)
|
isRectanguloidElement(element) ? avoidRectangularCorner(element, p) : p,
|
||||||
? avoidRectangularCorner(element, point)
|
|
||||||
: point,
|
|
||||||
otherPoint,
|
otherPoint,
|
||||||
element,
|
element,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
);
|
);
|
||||||
|
|
||||||
const getBindPointHeading = (
|
const getBindPointHeading = (
|
||||||
point: Point,
|
p: GlobalPoint,
|
||||||
otherPoint: Point,
|
otherPoint: GlobalPoint,
|
||||||
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
||||||
hoveredElement: ExcalidrawBindableElement | null | undefined,
|
hoveredElement: ExcalidrawBindableElement | null | undefined,
|
||||||
origPoint: Point,
|
origPoint: GlobalPoint,
|
||||||
) =>
|
) =>
|
||||||
getHeadingForElbowArrowSnap(
|
getHeadingForElbowArrowSnap(
|
||||||
point,
|
p,
|
||||||
otherPoint,
|
otherPoint,
|
||||||
hoveredElement,
|
hoveredElement,
|
||||||
hoveredElement &&
|
hoveredElement &&
|
||||||
aabbForElement(
|
aabbForElement(
|
||||||
hoveredElement,
|
hoveredElement,
|
||||||
Array(4).fill(
|
Array(4).fill(
|
||||||
distanceToBindableElement(hoveredElement, point, elementsMap),
|
distanceToBindableElement(hoveredElement, p, elementsMap),
|
||||||
) as [number, number, number, number],
|
) as [number, number, number, number],
|
||||||
),
|
),
|
||||||
elementsMap,
|
elementsMap,
|
||||||
|
@ -993,8 +1039,8 @@ const getBindPointHeading = (
|
||||||
);
|
);
|
||||||
|
|
||||||
const getHoveredElements = (
|
const getHoveredElements = (
|
||||||
origStartGlobalPoint: Point,
|
origStartGlobalPoint: GlobalPoint,
|
||||||
origEndGlobalPoint: Point,
|
origEndGlobalPoint: GlobalPoint,
|
||||||
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
elementsMap: NonDeletedSceneElementsMap | SceneElementsMap,
|
||||||
) => {
|
) => {
|
||||||
// TODO: Might be a performance bottleneck and the Map type
|
// TODO: Might be a performance bottleneck and the Map type
|
||||||
|
@ -1018,3 +1064,6 @@ const getHoveredElements = (
|
||||||
),
|
),
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const gridAddressesEqual = (a: GridAddress, b: GridAddress): boolean =>
|
||||||
|
a[0] === b[0] && a[1] === b[1];
|
||||||
|
|
|
@ -284,16 +284,17 @@ export const measureText = (
|
||||||
text: string,
|
text: string,
|
||||||
font: FontString,
|
font: FontString,
|
||||||
lineHeight: ExcalidrawTextElement["lineHeight"],
|
lineHeight: ExcalidrawTextElement["lineHeight"],
|
||||||
|
forceAdvanceWidth?: true,
|
||||||
) => {
|
) => {
|
||||||
text = text
|
const _text = text
|
||||||
.split("\n")
|
.split("\n")
|
||||||
// replace empty lines with single space because leading/trailing empty
|
// replace empty lines with single space because leading/trailing empty
|
||||||
// lines would be stripped from computation
|
// lines would be stripped from computation
|
||||||
.map((x) => x || " ")
|
.map((x) => x || " ")
|
||||||
.join("\n");
|
.join("\n");
|
||||||
const fontSize = parseFloat(font);
|
const fontSize = parseFloat(font);
|
||||||
const height = getTextHeight(text, fontSize, lineHeight);
|
const height = getTextHeight(_text, fontSize, lineHeight);
|
||||||
const width = getTextWidth(text, font);
|
const width = getTextWidth(_text, font, forceAdvanceWidth);
|
||||||
return { width, height };
|
return { width, height };
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -19,6 +19,7 @@ import type {
|
||||||
import { API } from "../tests/helpers/api";
|
import { API } from "../tests/helpers/api";
|
||||||
import { getOriginalContainerHeightFromCache } from "./containerCache";
|
import { getOriginalContainerHeightFromCache } from "./containerCache";
|
||||||
import { getTextEditor, updateTextEditor } from "../tests/queries/dom";
|
import { getTextEditor, updateTextEditor } from "../tests/queries/dom";
|
||||||
|
import { point } from "../../math";
|
||||||
|
|
||||||
// Unmount ReactDOM from root
|
// Unmount ReactDOM from root
|
||||||
ReactDOM.unmountComponentAtNode(document.getElementById("root")!);
|
ReactDOM.unmountComponentAtNode(document.getElementById("root")!);
|
||||||
|
@ -41,10 +42,7 @@ describe("textWysiwyg", () => {
|
||||||
type: "line",
|
type: "line",
|
||||||
width: 100,
|
width: 100,
|
||||||
height: 0,
|
height: 0,
|
||||||
points: [
|
points: [point(0, 0), point(100, 0)],
|
||||||
[0, 0],
|
|
||||||
[100, 0],
|
|
||||||
],
|
|
||||||
});
|
});
|
||||||
const textSize = 20;
|
const textSize = 20;
|
||||||
const text = API.createElement({
|
const text = API.createElement({
|
||||||
|
|
|
@ -7,7 +7,6 @@ import type {
|
||||||
|
|
||||||
import type { Bounds } from "./bounds";
|
import type { Bounds } from "./bounds";
|
||||||
import { getElementAbsoluteCoords } from "./bounds";
|
import { getElementAbsoluteCoords } from "./bounds";
|
||||||
import { rotate } from "../math";
|
|
||||||
import type { Device, InteractiveCanvasAppState, Zoom } from "../types";
|
import type { Device, InteractiveCanvasAppState, Zoom } from "../types";
|
||||||
import {
|
import {
|
||||||
isElbowArrow,
|
isElbowArrow,
|
||||||
|
@ -19,6 +18,8 @@ import {
|
||||||
isAndroid,
|
isAndroid,
|
||||||
isIOS,
|
isIOS,
|
||||||
} from "../constants";
|
} from "../constants";
|
||||||
|
import type { Radians } from "../../math";
|
||||||
|
import { point, pointRotateRads } from "../../math";
|
||||||
|
|
||||||
export type TransformHandleDirection =
|
export type TransformHandleDirection =
|
||||||
| "n"
|
| "n"
|
||||||
|
@ -91,9 +92,13 @@ const generateTransformHandle = (
|
||||||
height: number,
|
height: number,
|
||||||
cx: number,
|
cx: number,
|
||||||
cy: number,
|
cy: number,
|
||||||
angle: number,
|
angle: Radians,
|
||||||
): TransformHandle => {
|
): TransformHandle => {
|
||||||
const [xx, yy] = rotate(x + width / 2, y + height / 2, cx, cy, angle);
|
const [xx, yy] = pointRotateRads(
|
||||||
|
point(x + width / 2, y + height / 2),
|
||||||
|
point(cx, cy),
|
||||||
|
angle,
|
||||||
|
);
|
||||||
return [xx - width / 2, yy - height / 2, width, height];
|
return [xx - width / 2, yy - height / 2, width, height];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -119,7 +124,7 @@ export const getOmitSidesForDevice = (device: Device) => {
|
||||||
|
|
||||||
export const getTransformHandlesFromCoords = (
|
export const getTransformHandlesFromCoords = (
|
||||||
[x1, y1, x2, y2, cx, cy]: [number, number, number, number, number, number],
|
[x1, y1, x2, y2, cx, cy]: [number, number, number, number, number, number],
|
||||||
angle: number,
|
angle: Radians,
|
||||||
zoom: Zoom,
|
zoom: Zoom,
|
||||||
pointerType: PointerType,
|
pointerType: PointerType,
|
||||||
omitSides: { [T in TransformHandleType]?: boolean } = {},
|
omitSides: { [T in TransformHandleType]?: boolean } = {},
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import type { LineSegment } from "../../utils";
|
|
||||||
import { ROUNDNESS } from "../constants";
|
import { ROUNDNESS } from "../constants";
|
||||||
import type { ElementOrToolType, Point } from "../types";
|
import type { ElementOrToolType } from "../types";
|
||||||
import type { MarkNonNullable } from "../utility-types";
|
import type { MarkNonNullable } from "../utility-types";
|
||||||
import { assertNever } from "../utils";
|
import { assertNever } from "../utils";
|
||||||
import type { Bounds } from "./bounds";
|
import type { Bounds } from "./bounds";
|
||||||
|
@ -191,7 +190,8 @@ export const isRectangularElement = (
|
||||||
element.type === "iframe" ||
|
element.type === "iframe" ||
|
||||||
element.type === "embeddable" ||
|
element.type === "embeddable" ||
|
||||||
element.type === "frame" ||
|
element.type === "frame" ||
|
||||||
element.type === "magicframe")
|
element.type === "magicframe" ||
|
||||||
|
element.type === "freedraw")
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -325,10 +325,6 @@ export const isFixedPointBinding = (
|
||||||
return binding.fixedPoint != null;
|
return binding.fixedPoint != null;
|
||||||
};
|
};
|
||||||
|
|
||||||
// TODO: Move this to @excalidraw/math
|
|
||||||
export const isPoint = (point: unknown): point is Point =>
|
|
||||||
Array.isArray(point) && point.length === 2;
|
|
||||||
|
|
||||||
// TODO: Move this to @excalidraw/math
|
// TODO: Move this to @excalidraw/math
|
||||||
export const isBounds = (box: unknown): box is Bounds =>
|
export const isBounds = (box: unknown): box is Bounds =>
|
||||||
Array.isArray(box) &&
|
Array.isArray(box) &&
|
||||||
|
@ -337,10 +333,3 @@ export const isBounds = (box: unknown): box is Bounds =>
|
||||||
typeof box[1] === "number" &&
|
typeof box[1] === "number" &&
|
||||||
typeof box[2] === "number" &&
|
typeof box[2] === "number" &&
|
||||||
typeof box[3] === "number";
|
typeof box[3] === "number";
|
||||||
|
|
||||||
// TODO: Move this to @excalidraw/math
|
|
||||||
export const isLineSegment = (segment: unknown): segment is LineSegment =>
|
|
||||||
Array.isArray(segment) &&
|
|
||||||
segment.length === 2 &&
|
|
||||||
isPoint(segment[0]) &&
|
|
||||||
isPoint(segment[0]);
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import type { Point } from "../types";
|
import type { LocalPoint, Radians } from "../../math";
|
||||||
import type {
|
import type {
|
||||||
FONT_FAMILY,
|
FONT_FAMILY,
|
||||||
ROUNDNESS,
|
ROUNDNESS,
|
||||||
|
@ -49,7 +49,7 @@ type _ExcalidrawElementBase = Readonly<{
|
||||||
opacity: number;
|
opacity: number;
|
||||||
width: number;
|
width: number;
|
||||||
height: number;
|
height: number;
|
||||||
angle: number;
|
angle: Radians;
|
||||||
/** Random integer used to seed shape generation so that the roughjs shape
|
/** Random integer used to seed shape generation so that the roughjs shape
|
||||||
doesn't differ across renders. */
|
doesn't differ across renders. */
|
||||||
seed: number;
|
seed: number;
|
||||||
|
@ -175,6 +175,15 @@ export type ExcalidrawFlowchartNodeElement =
|
||||||
| ExcalidrawDiamondElement
|
| ExcalidrawDiamondElement
|
||||||
| ExcalidrawEllipseElement;
|
| ExcalidrawEllipseElement;
|
||||||
|
|
||||||
|
export type ExcalidrawRectanguloidElement =
|
||||||
|
| ExcalidrawRectangleElement
|
||||||
|
| ExcalidrawImageElement
|
||||||
|
| ExcalidrawTextElement
|
||||||
|
| ExcalidrawFreeDrawElement
|
||||||
|
| ExcalidrawIframeLikeElement
|
||||||
|
| ExcalidrawFrameLikeElement
|
||||||
|
| ExcalidrawEmbeddableElement;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ExcalidrawElement should be JSON serializable and (eventually) contain
|
* ExcalidrawElement should be JSON serializable and (eventually) contain
|
||||||
* no computed data. The list of all ExcalidrawElements should be shareable
|
* no computed data. The list of all ExcalidrawElements should be shareable
|
||||||
|
@ -283,8 +292,8 @@ export type Arrowhead =
|
||||||
export type ExcalidrawLinearElement = _ExcalidrawElementBase &
|
export type ExcalidrawLinearElement = _ExcalidrawElementBase &
|
||||||
Readonly<{
|
Readonly<{
|
||||||
type: "line" | "arrow";
|
type: "line" | "arrow";
|
||||||
points: readonly Point[];
|
points: readonly LocalPoint[];
|
||||||
lastCommittedPoint: Point | null;
|
lastCommittedPoint: LocalPoint | null;
|
||||||
startBinding: PointBinding | null;
|
startBinding: PointBinding | null;
|
||||||
endBinding: PointBinding | null;
|
endBinding: PointBinding | null;
|
||||||
startArrowhead: Arrowhead | null;
|
startArrowhead: Arrowhead | null;
|
||||||
|
@ -309,10 +318,10 @@ export type ExcalidrawElbowArrowElement = Merge<
|
||||||
export type ExcalidrawFreeDrawElement = _ExcalidrawElementBase &
|
export type ExcalidrawFreeDrawElement = _ExcalidrawElementBase &
|
||||||
Readonly<{
|
Readonly<{
|
||||||
type: "freedraw";
|
type: "freedraw";
|
||||||
points: readonly Point[];
|
points: readonly LocalPoint[];
|
||||||
pressures: readonly number[];
|
pressures: readonly number[];
|
||||||
simulatePressure: boolean;
|
simulatePressure: boolean;
|
||||||
lastCommittedPoint: Point | null;
|
lastCommittedPoint: LocalPoint | null;
|
||||||
}>;
|
}>;
|
||||||
|
|
||||||
export type FileId = string & { _brand: "FileId" };
|
export type FileId = string & { _brand: "FileId" };
|
||||||
|
|
|
@ -11,7 +11,6 @@ import type {
|
||||||
NonDeleted,
|
NonDeleted,
|
||||||
NonDeletedExcalidrawElement,
|
NonDeletedExcalidrawElement,
|
||||||
} from "./element/types";
|
} from "./element/types";
|
||||||
import { isPointWithinBounds } from "./math";
|
|
||||||
import {
|
import {
|
||||||
getBoundTextElement,
|
getBoundTextElement,
|
||||||
getContainerElement,
|
getContainerElement,
|
||||||
|
@ -30,6 +29,7 @@ import { getElementLineSegments } from "./element/bounds";
|
||||||
import { doLineSegmentsIntersect, elementsOverlappingBBox } from "../utils/";
|
import { doLineSegmentsIntersect, elementsOverlappingBBox } from "../utils/";
|
||||||
import { isFrameElement, isFrameLikeElement } from "./element/typeChecks";
|
import { isFrameElement, isFrameLikeElement } from "./element/typeChecks";
|
||||||
import type { ReadonlySetLike } from "./utility-types";
|
import type { ReadonlySetLike } from "./utility-types";
|
||||||
|
import { isPointWithinBounds, point } from "../math";
|
||||||
|
|
||||||
// --------------------------- Frame State ------------------------------------
|
// --------------------------- Frame State ------------------------------------
|
||||||
export const bindElementsToFramesAfterDuplication = (
|
export const bindElementsToFramesAfterDuplication = (
|
||||||
|
@ -159,9 +159,9 @@ export const isCursorInFrame = (
|
||||||
const [fx1, fy1, fx2, fy2] = getElementAbsoluteCoords(frame, elementsMap);
|
const [fx1, fy1, fx2, fy2] = getElementAbsoluteCoords(frame, elementsMap);
|
||||||
|
|
||||||
return isPointWithinBounds(
|
return isPointWithinBounds(
|
||||||
[fx1, fy1],
|
point(fx1, fy1),
|
||||||
[cursorCoords.x, cursorCoords.y],
|
point(cursorCoords.x, cursorCoords.y),
|
||||||
[fx2, fy2],
|
point(fx2, fy2),
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -162,6 +162,13 @@
|
||||||
"hint_emptyLibrary": "Select an item on canvas to add it here, or install a library from the public repository, below.",
|
"hint_emptyLibrary": "Select an item on canvas to add it here, or install a library from the public repository, below.",
|
||||||
"hint_emptyPrivateLibrary": "Select an item on canvas to add it here."
|
"hint_emptyPrivateLibrary": "Select an item on canvas to add it here."
|
||||||
},
|
},
|
||||||
|
"search": {
|
||||||
|
"title": "Find on canvas",
|
||||||
|
"noMatch": "No matches found...",
|
||||||
|
"singleResult": "result",
|
||||||
|
"multipleResults": "results",
|
||||||
|
"placeholder": "Find text..."
|
||||||
|
},
|
||||||
"buttons": {
|
"buttons": {
|
||||||
"clearReset": "Reset the canvas",
|
"clearReset": "Reset the canvas",
|
||||||
"exportJSON": "Export to file",
|
"exportJSON": "Export to file",
|
||||||
|
@ -297,6 +304,7 @@
|
||||||
"shapes": "Shapes"
|
"shapes": "Shapes"
|
||||||
},
|
},
|
||||||
"hints": {
|
"hints": {
|
||||||
|
"dismissSearch": "Escape to dismiss search",
|
||||||
"canvasPanning": "To move canvas, hold mouse wheel or spacebar while dragging, or use the hand tool",
|
"canvasPanning": "To move canvas, hold mouse wheel or spacebar while dragging, or use the hand tool",
|
||||||
"linearElement": "Click to start multiple points, drag for single line",
|
"linearElement": "Click to start multiple points, drag for single line",
|
||||||
"arrowTool": "Click to start multiple points, drag for single line. Press {{arrowShortcut}} again to change arrow type.",
|
"arrowTool": "Click to start multiple points, drag for single line. Press {{arrowShortcut}} again to change arrow type.",
|
||||||
|
|
|
@ -1,99 +0,0 @@
|
||||||
import {
|
|
||||||
isPointOnSymmetricArc,
|
|
||||||
rangeIntersection,
|
|
||||||
rangesOverlap,
|
|
||||||
rotate,
|
|
||||||
} from "./math";
|
|
||||||
|
|
||||||
describe("rotate", () => {
|
|
||||||
it("should rotate over (x2, y2) and return the rotated coordinates for (x1, y1)", () => {
|
|
||||||
const x1 = 10;
|
|
||||||
const y1 = 20;
|
|
||||||
const x2 = 20;
|
|
||||||
const y2 = 30;
|
|
||||||
const angle = Math.PI / 2;
|
|
||||||
const [rotatedX, rotatedY] = rotate(x1, y1, x2, y2, angle);
|
|
||||||
expect([rotatedX, rotatedY]).toEqual([30, 20]);
|
|
||||||
const res2 = rotate(rotatedX, rotatedY, x2, y2, -angle);
|
|
||||||
expect(res2).toEqual([x1, x2]);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe("range overlap", () => {
|
|
||||||
it("should overlap when range a contains range b", () => {
|
|
||||||
expect(rangesOverlap([1, 4], [2, 3])).toBe(true);
|
|
||||||
expect(rangesOverlap([1, 4], [1, 4])).toBe(true);
|
|
||||||
expect(rangesOverlap([1, 4], [1, 3])).toBe(true);
|
|
||||||
expect(rangesOverlap([1, 4], [2, 4])).toBe(true);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should overlap when range b contains range a", () => {
|
|
||||||
expect(rangesOverlap([2, 3], [1, 4])).toBe(true);
|
|
||||||
expect(rangesOverlap([1, 3], [1, 4])).toBe(true);
|
|
||||||
expect(rangesOverlap([2, 4], [1, 4])).toBe(true);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should overlap when range a and b intersect", () => {
|
|
||||||
expect(rangesOverlap([1, 4], [2, 5])).toBe(true);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe("range intersection", () => {
|
|
||||||
it("should intersect completely with itself", () => {
|
|
||||||
expect(rangeIntersection([1, 4], [1, 4])).toEqual([1, 4]);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should intersect irrespective of order", () => {
|
|
||||||
expect(rangeIntersection([1, 4], [2, 3])).toEqual([2, 3]);
|
|
||||||
expect(rangeIntersection([2, 3], [1, 4])).toEqual([2, 3]);
|
|
||||||
expect(rangeIntersection([1, 4], [3, 5])).toEqual([3, 4]);
|
|
||||||
expect(rangeIntersection([3, 5], [1, 4])).toEqual([3, 4]);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should intersect at the edge", () => {
|
|
||||||
expect(rangeIntersection([1, 4], [4, 5])).toEqual([4, 4]);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should not intersect", () => {
|
|
||||||
expect(rangeIntersection([1, 4], [5, 7])).toEqual(null);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe("point on arc", () => {
|
|
||||||
it("should detect point on simple arc", () => {
|
|
||||||
expect(
|
|
||||||
isPointOnSymmetricArc(
|
|
||||||
{
|
|
||||||
radius: 1,
|
|
||||||
startAngle: -Math.PI / 4,
|
|
||||||
endAngle: Math.PI / 4,
|
|
||||||
},
|
|
||||||
[0.92291667, 0.385],
|
|
||||||
),
|
|
||||||
).toBe(true);
|
|
||||||
});
|
|
||||||
it("should not detect point outside of a simple arc", () => {
|
|
||||||
expect(
|
|
||||||
isPointOnSymmetricArc(
|
|
||||||
{
|
|
||||||
radius: 1,
|
|
||||||
startAngle: -Math.PI / 4,
|
|
||||||
endAngle: Math.PI / 4,
|
|
||||||
},
|
|
||||||
[-0.92291667, 0.385],
|
|
||||||
),
|
|
||||||
).toBe(false);
|
|
||||||
});
|
|
||||||
it("should not detect point with good angle but incorrect radius", () => {
|
|
||||||
expect(
|
|
||||||
isPointOnSymmetricArc(
|
|
||||||
{
|
|
||||||
radius: 1,
|
|
||||||
startAngle: -Math.PI / 4,
|
|
||||||
endAngle: Math.PI / 4,
|
|
||||||
},
|
|
||||||
[-0.5, 0.5],
|
|
||||||
),
|
|
||||||
).toBe(false);
|
|
||||||
});
|
|
||||||
});
|
|
|
@ -1,715 +0,0 @@
|
||||||
import type {
|
|
||||||
NormalizedZoomValue,
|
|
||||||
NullableGridSize,
|
|
||||||
Point,
|
|
||||||
Zoom,
|
|
||||||
} from "./types";
|
|
||||||
import {
|
|
||||||
DEFAULT_ADAPTIVE_RADIUS,
|
|
||||||
LINE_CONFIRM_THRESHOLD,
|
|
||||||
DEFAULT_PROPORTIONAL_RADIUS,
|
|
||||||
ROUNDNESS,
|
|
||||||
} from "./constants";
|
|
||||||
import type {
|
|
||||||
ExcalidrawElement,
|
|
||||||
ExcalidrawLinearElement,
|
|
||||||
NonDeleted,
|
|
||||||
} from "./element/types";
|
|
||||||
import type { Bounds } from "./element/bounds";
|
|
||||||
import { getCurvePathOps } from "./element/bounds";
|
|
||||||
import type { Mutable } from "./utility-types";
|
|
||||||
import { ShapeCache } from "./scene/ShapeCache";
|
|
||||||
import type { Vector } from "../utils/geometry/shape";
|
|
||||||
|
|
||||||
export const rotate = (
|
|
||||||
// target point to rotate
|
|
||||||
x: number,
|
|
||||||
y: number,
|
|
||||||
// point to rotate against
|
|
||||||
cx: number,
|
|
||||||
cy: number,
|
|
||||||
angle: number,
|
|
||||||
): [number, number] =>
|
|
||||||
// 𝑎′𝑥=(𝑎𝑥−𝑐𝑥)cos𝜃−(𝑎𝑦−𝑐𝑦)sin𝜃+𝑐𝑥
|
|
||||||
// 𝑎′𝑦=(𝑎𝑥−𝑐𝑥)sin𝜃+(𝑎𝑦−𝑐𝑦)cos𝜃+𝑐𝑦.
|
|
||||||
// https://math.stackexchange.com/questions/2204520/how-do-i-rotate-a-line-segment-in-a-specific-point-on-the-line
|
|
||||||
[
|
|
||||||
(x - cx) * Math.cos(angle) - (y - cy) * Math.sin(angle) + cx,
|
|
||||||
(x - cx) * Math.sin(angle) + (y - cy) * Math.cos(angle) + cy,
|
|
||||||
];
|
|
||||||
|
|
||||||
export const rotatePoint = (
|
|
||||||
point: Point,
|
|
||||||
center: Point,
|
|
||||||
angle: number,
|
|
||||||
): [number, number] => rotate(point[0], point[1], center[0], center[1], angle);
|
|
||||||
|
|
||||||
export const adjustXYWithRotation = (
|
|
||||||
sides: {
|
|
||||||
n?: boolean;
|
|
||||||
e?: boolean;
|
|
||||||
s?: boolean;
|
|
||||||
w?: boolean;
|
|
||||||
},
|
|
||||||
x: number,
|
|
||||||
y: number,
|
|
||||||
angle: number,
|
|
||||||
deltaX1: number,
|
|
||||||
deltaY1: number,
|
|
||||||
deltaX2: number,
|
|
||||||
deltaY2: number,
|
|
||||||
): [number, number] => {
|
|
||||||
const cos = Math.cos(angle);
|
|
||||||
const sin = Math.sin(angle);
|
|
||||||
if (sides.e && sides.w) {
|
|
||||||
x += deltaX1 + deltaX2;
|
|
||||||
} else if (sides.e) {
|
|
||||||
x += deltaX1 * (1 + cos);
|
|
||||||
y += deltaX1 * sin;
|
|
||||||
x += deltaX2 * (1 - cos);
|
|
||||||
y += deltaX2 * -sin;
|
|
||||||
} else if (sides.w) {
|
|
||||||
x += deltaX1 * (1 - cos);
|
|
||||||
y += deltaX1 * -sin;
|
|
||||||
x += deltaX2 * (1 + cos);
|
|
||||||
y += deltaX2 * sin;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (sides.n && sides.s) {
|
|
||||||
y += deltaY1 + deltaY2;
|
|
||||||
} else if (sides.n) {
|
|
||||||
x += deltaY1 * sin;
|
|
||||||
y += deltaY1 * (1 - cos);
|
|
||||||
x += deltaY2 * -sin;
|
|
||||||
y += deltaY2 * (1 + cos);
|
|
||||||
} else if (sides.s) {
|
|
||||||
x += deltaY1 * -sin;
|
|
||||||
y += deltaY1 * (1 + cos);
|
|
||||||
x += deltaY2 * sin;
|
|
||||||
y += deltaY2 * (1 - cos);
|
|
||||||
}
|
|
||||||
return [x, y];
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getPointOnAPath = (point: Point, path: Point[]) => {
|
|
||||||
const [px, py] = point;
|
|
||||||
const [start, ...other] = path;
|
|
||||||
let [lastX, lastY] = start;
|
|
||||||
let kLine: number = 0;
|
|
||||||
let idx: number = 0;
|
|
||||||
|
|
||||||
// if any item in the array is true, it means that a point is
|
|
||||||
// on some segment of a line based path
|
|
||||||
const retVal = other.some(([x2, y2], i) => {
|
|
||||||
// we always take a line when dealing with line segments
|
|
||||||
const x1 = lastX;
|
|
||||||
const y1 = lastY;
|
|
||||||
|
|
||||||
lastX = x2;
|
|
||||||
lastY = y2;
|
|
||||||
|
|
||||||
// if a point is not within the domain of the line segment
|
|
||||||
// it is not on the line segment
|
|
||||||
if (px < x1 || px > x2) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// check if all points lie on the same line
|
|
||||||
// y1 = kx1 + b, y2 = kx2 + b
|
|
||||||
// y2 - y1 = k(x2 - x2) -> k = (y2 - y1) / (x2 - x1)
|
|
||||||
|
|
||||||
// coefficient for the line (p0, p1)
|
|
||||||
const kL = (y2 - y1) / (x2 - x1);
|
|
||||||
|
|
||||||
// coefficient for the line segment (p0, point)
|
|
||||||
const kP1 = (py - y1) / (px - x1);
|
|
||||||
|
|
||||||
// coefficient for the line segment (point, p1)
|
|
||||||
const kP2 = (py - y2) / (px - x2);
|
|
||||||
|
|
||||||
// because we are basing both lines from the same starting point
|
|
||||||
// the only option for collinearity is having same coefficients
|
|
||||||
|
|
||||||
// using it for floating point comparisons
|
|
||||||
const epsilon = 0.3;
|
|
||||||
|
|
||||||
// if coefficient is more than an arbitrary epsilon,
|
|
||||||
// these lines are nor collinear
|
|
||||||
if (Math.abs(kP1 - kL) > epsilon && Math.abs(kP2 - kL) > epsilon) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// store the coefficient because we are goint to need it
|
|
||||||
kLine = kL;
|
|
||||||
idx = i;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
|
|
||||||
// Return a coordinate that is always on the line segment
|
|
||||||
if (retVal === true) {
|
|
||||||
return { x: point[0], y: kLine * point[0], segment: idx };
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const distance2d = (x1: number, y1: number, x2: number, y2: number) => {
|
|
||||||
const xd = x2 - x1;
|
|
||||||
const yd = y2 - y1;
|
|
||||||
return Math.hypot(xd, yd);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const distanceSq2d = (p1: Point, p2: Point) => {
|
|
||||||
const xd = p2[0] - p1[0];
|
|
||||||
const yd = p2[1] - p1[1];
|
|
||||||
return xd * xd + yd * yd;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const centerPoint = (a: Point, b: Point): Point => {
|
|
||||||
return [(a[0] + b[0]) / 2, (a[1] + b[1]) / 2];
|
|
||||||
};
|
|
||||||
|
|
||||||
// Checks if the first and last point are close enough
|
|
||||||
// to be considered a loop
|
|
||||||
export const isPathALoop = (
|
|
||||||
points: ExcalidrawLinearElement["points"],
|
|
||||||
/** supply if you want the loop detection to account for current zoom */
|
|
||||||
zoomValue: Zoom["value"] = 1 as NormalizedZoomValue,
|
|
||||||
): boolean => {
|
|
||||||
if (points.length >= 3) {
|
|
||||||
const [first, last] = [points[0], points[points.length - 1]];
|
|
||||||
const distance = distance2d(first[0], first[1], last[0], last[1]);
|
|
||||||
|
|
||||||
// Adjusting LINE_CONFIRM_THRESHOLD to current zoom so that when zoomed in
|
|
||||||
// really close we make the threshold smaller, and vice versa.
|
|
||||||
return distance <= LINE_CONFIRM_THRESHOLD / zoomValue;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Draw a line from the point to the right till infiinty
|
|
||||||
// Check how many lines of the polygon does this infinite line intersects with
|
|
||||||
// If the number of intersections is odd, point is in the polygon
|
|
||||||
export const isPointInPolygon = (
|
|
||||||
points: Point[],
|
|
||||||
x: number,
|
|
||||||
y: number,
|
|
||||||
): boolean => {
|
|
||||||
const vertices = points.length;
|
|
||||||
|
|
||||||
// There must be at least 3 vertices in polygon
|
|
||||||
if (vertices < 3) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
const extreme: Point = [Number.MAX_SAFE_INTEGER, y];
|
|
||||||
const p: Point = [x, y];
|
|
||||||
let count = 0;
|
|
||||||
for (let i = 0; i < vertices; i++) {
|
|
||||||
const current = points[i];
|
|
||||||
const next = points[(i + 1) % vertices];
|
|
||||||
if (doSegmentsIntersect(current, next, p, extreme)) {
|
|
||||||
if (orderedColinearOrientation(current, p, next) === 0) {
|
|
||||||
return isPointWithinBounds(current, p, next);
|
|
||||||
}
|
|
||||||
count++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// true if count is off
|
|
||||||
return count % 2 === 1;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Returns whether `q` lies inside the segment/rectangle defined by `p` and `r`.
|
|
||||||
// This is an approximation to "does `q` lie on a segment `pr`" check.
|
|
||||||
export const isPointWithinBounds = (p: Point, q: Point, r: Point) => {
|
|
||||||
return (
|
|
||||||
q[0] <= Math.max(p[0], r[0]) &&
|
|
||||||
q[0] >= Math.min(p[0], r[0]) &&
|
|
||||||
q[1] <= Math.max(p[1], r[1]) &&
|
|
||||||
q[1] >= Math.min(p[1], r[1])
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
// For the ordered points p, q, r, return
|
|
||||||
// 0 if p, q, r are colinear
|
|
||||||
// 1 if Clockwise
|
|
||||||
// 2 if counterclickwise
|
|
||||||
const orderedColinearOrientation = (p: Point, q: Point, r: Point) => {
|
|
||||||
const val = (q[1] - p[1]) * (r[0] - q[0]) - (q[0] - p[0]) * (r[1] - q[1]);
|
|
||||||
if (val === 0) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
return val > 0 ? 1 : 2;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Check is p1q1 intersects with p2q2
|
|
||||||
const doSegmentsIntersect = (p1: Point, q1: Point, p2: Point, q2: Point) => {
|
|
||||||
const o1 = orderedColinearOrientation(p1, q1, p2);
|
|
||||||
const o2 = orderedColinearOrientation(p1, q1, q2);
|
|
||||||
const o3 = orderedColinearOrientation(p2, q2, p1);
|
|
||||||
const o4 = orderedColinearOrientation(p2, q2, q1);
|
|
||||||
|
|
||||||
if (o1 !== o2 && o3 !== o4) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// p1, q1 and p2 are colinear and p2 lies on segment p1q1
|
|
||||||
if (o1 === 0 && isPointWithinBounds(p1, p2, q1)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// p1, q1 and p2 are colinear and q2 lies on segment p1q1
|
|
||||||
if (o2 === 0 && isPointWithinBounds(p1, q2, q1)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// p2, q2 and p1 are colinear and p1 lies on segment p2q2
|
|
||||||
if (o3 === 0 && isPointWithinBounds(p2, p1, q2)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// p2, q2 and q1 are colinear and q1 lies on segment p2q2
|
|
||||||
if (o4 === 0 && isPointWithinBounds(p2, q1, q2)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
};
|
|
||||||
|
|
||||||
// TODO: Rounding this point causes some shake when free drawing
|
|
||||||
export const getGridPoint = (
|
|
||||||
x: number,
|
|
||||||
y: number,
|
|
||||||
gridSize: NullableGridSize,
|
|
||||||
): [number, number] => {
|
|
||||||
if (gridSize) {
|
|
||||||
return [
|
|
||||||
Math.round(x / gridSize) * gridSize,
|
|
||||||
Math.round(y / gridSize) * gridSize,
|
|
||||||
];
|
|
||||||
}
|
|
||||||
return [x, y];
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getCornerRadius = (x: number, element: ExcalidrawElement) => {
|
|
||||||
if (
|
|
||||||
element.roundness?.type === ROUNDNESS.PROPORTIONAL_RADIUS ||
|
|
||||||
element.roundness?.type === ROUNDNESS.LEGACY
|
|
||||||
) {
|
|
||||||
return x * DEFAULT_PROPORTIONAL_RADIUS;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (element.roundness?.type === ROUNDNESS.ADAPTIVE_RADIUS) {
|
|
||||||
const fixedRadiusSize = element.roundness?.value ?? DEFAULT_ADAPTIVE_RADIUS;
|
|
||||||
|
|
||||||
const CUTOFF_SIZE = fixedRadiusSize / DEFAULT_PROPORTIONAL_RADIUS;
|
|
||||||
|
|
||||||
if (x <= CUTOFF_SIZE) {
|
|
||||||
return x * DEFAULT_PROPORTIONAL_RADIUS;
|
|
||||||
}
|
|
||||||
|
|
||||||
return fixedRadiusSize;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getControlPointsForBezierCurve = (
|
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
|
||||||
endPoint: Point,
|
|
||||||
) => {
|
|
||||||
const shape = ShapeCache.generateElementShape(element, null);
|
|
||||||
if (!shape) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
const ops = getCurvePathOps(shape[0]);
|
|
||||||
let currentP: Mutable<Point> = [0, 0];
|
|
||||||
let index = 0;
|
|
||||||
let minDistance = Infinity;
|
|
||||||
let controlPoints: Mutable<Point>[] | null = null;
|
|
||||||
|
|
||||||
while (index < ops.length) {
|
|
||||||
const { op, data } = ops[index];
|
|
||||||
if (op === "move") {
|
|
||||||
currentP = data as unknown as Mutable<Point>;
|
|
||||||
}
|
|
||||||
if (op === "bcurveTo") {
|
|
||||||
const p0 = currentP;
|
|
||||||
const p1 = [data[0], data[1]] as Mutable<Point>;
|
|
||||||
const p2 = [data[2], data[3]] as Mutable<Point>;
|
|
||||||
const p3 = [data[4], data[5]] as Mutable<Point>;
|
|
||||||
const distance = distance2d(p3[0], p3[1], endPoint[0], endPoint[1]);
|
|
||||||
if (distance < minDistance) {
|
|
||||||
minDistance = distance;
|
|
||||||
controlPoints = [p0, p1, p2, p3];
|
|
||||||
}
|
|
||||||
currentP = p3;
|
|
||||||
}
|
|
||||||
index++;
|
|
||||||
}
|
|
||||||
|
|
||||||
return controlPoints;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getBezierXY = (
|
|
||||||
p0: Point,
|
|
||||||
p1: Point,
|
|
||||||
p2: Point,
|
|
||||||
p3: Point,
|
|
||||||
t: number,
|
|
||||||
) => {
|
|
||||||
const equation = (t: number, idx: number) =>
|
|
||||||
Math.pow(1 - t, 3) * p3[idx] +
|
|
||||||
3 * t * Math.pow(1 - t, 2) * p2[idx] +
|
|
||||||
3 * Math.pow(t, 2) * (1 - t) * p1[idx] +
|
|
||||||
p0[idx] * Math.pow(t, 3);
|
|
||||||
const tx = equation(t, 0);
|
|
||||||
const ty = equation(t, 1);
|
|
||||||
return [tx, ty];
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getPointsInBezierCurve = (
|
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
|
||||||
endPoint: Point,
|
|
||||||
) => {
|
|
||||||
const controlPoints: Mutable<Point>[] = getControlPointsForBezierCurve(
|
|
||||||
element,
|
|
||||||
endPoint,
|
|
||||||
)!;
|
|
||||||
if (!controlPoints) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
const pointsOnCurve: Mutable<Point>[] = [];
|
|
||||||
let t = 1;
|
|
||||||
// Take 20 points on curve for better accuracy
|
|
||||||
while (t > 0) {
|
|
||||||
const point = getBezierXY(
|
|
||||||
controlPoints[0],
|
|
||||||
controlPoints[1],
|
|
||||||
controlPoints[2],
|
|
||||||
controlPoints[3],
|
|
||||||
t,
|
|
||||||
);
|
|
||||||
pointsOnCurve.push([point[0], point[1]]);
|
|
||||||
t -= 0.05;
|
|
||||||
}
|
|
||||||
if (pointsOnCurve.length) {
|
|
||||||
if (arePointsEqual(pointsOnCurve.at(-1)!, endPoint)) {
|
|
||||||
pointsOnCurve.push([endPoint[0], endPoint[1]]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return pointsOnCurve;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getBezierCurveArcLengths = (
|
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
|
||||||
endPoint: Point,
|
|
||||||
) => {
|
|
||||||
const arcLengths: number[] = [];
|
|
||||||
arcLengths[0] = 0;
|
|
||||||
const points = getPointsInBezierCurve(element, endPoint);
|
|
||||||
let index = 0;
|
|
||||||
let distance = 0;
|
|
||||||
while (index < points.length - 1) {
|
|
||||||
const segmentDistance = distance2d(
|
|
||||||
points[index][0],
|
|
||||||
points[index][1],
|
|
||||||
points[index + 1][0],
|
|
||||||
points[index + 1][1],
|
|
||||||
);
|
|
||||||
distance += segmentDistance;
|
|
||||||
arcLengths.push(distance);
|
|
||||||
index++;
|
|
||||||
}
|
|
||||||
|
|
||||||
return arcLengths;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getBezierCurveLength = (
|
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
|
||||||
endPoint: Point,
|
|
||||||
) => {
|
|
||||||
const arcLengths = getBezierCurveArcLengths(element, endPoint);
|
|
||||||
return arcLengths.at(-1) as number;
|
|
||||||
};
|
|
||||||
|
|
||||||
// This maps interval to actual interval t on the curve so that when t = 0.5, its actually the point at 50% of the length
|
|
||||||
export const mapIntervalToBezierT = (
|
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
|
||||||
endPoint: Point,
|
|
||||||
interval: number, // The interval between 0 to 1 for which you want to find the point on the curve,
|
|
||||||
) => {
|
|
||||||
const arcLengths = getBezierCurveArcLengths(element, endPoint);
|
|
||||||
const pointsCount = arcLengths.length - 1;
|
|
||||||
const curveLength = arcLengths.at(-1) as number;
|
|
||||||
const targetLength = interval * curveLength;
|
|
||||||
let low = 0;
|
|
||||||
let high = pointsCount;
|
|
||||||
let index = 0;
|
|
||||||
// Doing a binary search to find the largest length that is less than the target length
|
|
||||||
while (low < high) {
|
|
||||||
index = Math.floor(low + (high - low) / 2);
|
|
||||||
if (arcLengths[index] < targetLength) {
|
|
||||||
low = index + 1;
|
|
||||||
} else {
|
|
||||||
high = index;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (arcLengths[index] > targetLength) {
|
|
||||||
index--;
|
|
||||||
}
|
|
||||||
if (arcLengths[index] === targetLength) {
|
|
||||||
return index / pointsCount;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
1 -
|
|
||||||
(index +
|
|
||||||
(targetLength - arcLengths[index]) /
|
|
||||||
(arcLengths[index + 1] - arcLengths[index])) /
|
|
||||||
pointsCount
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const arePointsEqual = (p1: Point, p2: Point) => {
|
|
||||||
return p1[0] === p2[0] && p1[1] === p2[1];
|
|
||||||
};
|
|
||||||
|
|
||||||
export const isRightAngle = (angle: number) => {
|
|
||||||
// if our angles were mathematically accurate, we could just check
|
|
||||||
//
|
|
||||||
// angle % (Math.PI / 2) === 0
|
|
||||||
//
|
|
||||||
// but since we're in floating point land, we need to round.
|
|
||||||
//
|
|
||||||
// Below, after dividing by Math.PI, a multiple of 0.5 indicates a right
|
|
||||||
// angle, which we can check with modulo after rounding.
|
|
||||||
return Math.round((angle / Math.PI) * 10000) % 5000 === 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const radianToDegree = (r: number) => {
|
|
||||||
return (r * 180) / Math.PI;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const degreeToRadian = (d: number) => {
|
|
||||||
return (d / 180) * Math.PI;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Given two ranges, return if the two ranges overlap with each other
|
|
||||||
// e.g. [1, 3] overlaps with [2, 4] while [1, 3] does not overlap with [4, 5]
|
|
||||||
export const rangesOverlap = (
|
|
||||||
[a0, a1]: [number, number],
|
|
||||||
[b0, b1]: [number, number],
|
|
||||||
) => {
|
|
||||||
if (a0 <= b0) {
|
|
||||||
return a1 >= b0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (a0 >= b0) {
|
|
||||||
return b1 >= a0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Given two ranges,return ther intersection of the two ranges if any
|
|
||||||
// e.g. the intersection of [1, 3] and [2, 4] is [2, 3]
|
|
||||||
export const rangeIntersection = (
|
|
||||||
rangeA: [number, number],
|
|
||||||
rangeB: [number, number],
|
|
||||||
): [number, number] | null => {
|
|
||||||
const rangeStart = Math.max(rangeA[0], rangeB[0]);
|
|
||||||
const rangeEnd = Math.min(rangeA[1], rangeB[1]);
|
|
||||||
|
|
||||||
if (rangeStart <= rangeEnd) {
|
|
||||||
return [rangeStart, rangeEnd];
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const isValueInRange = (value: number, min: number, max: number) => {
|
|
||||||
return value >= min && value <= max;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const translatePoint = (p: Point, v: Vector): Point => [
|
|
||||||
p[0] + v[0],
|
|
||||||
p[1] + v[1],
|
|
||||||
];
|
|
||||||
|
|
||||||
export const scaleVector = (v: Vector, scalar: number): Vector => [
|
|
||||||
v[0] * scalar,
|
|
||||||
v[1] * scalar,
|
|
||||||
];
|
|
||||||
|
|
||||||
export const pointToVector = (p: Point, origin: Point = [0, 0]): Vector => [
|
|
||||||
p[0] - origin[0],
|
|
||||||
p[1] - origin[1],
|
|
||||||
];
|
|
||||||
|
|
||||||
export const scalePointFromOrigin = (
|
|
||||||
p: Point,
|
|
||||||
mid: Point,
|
|
||||||
multiplier: number,
|
|
||||||
) => translatePoint(mid, scaleVector(pointToVector(p, mid), multiplier));
|
|
||||||
|
|
||||||
const triangleSign = (p1: Point, p2: Point, p3: Point): number =>
|
|
||||||
(p1[0] - p3[0]) * (p2[1] - p3[1]) - (p2[0] - p3[0]) * (p1[1] - p3[1]);
|
|
||||||
|
|
||||||
export const PointInTriangle = (pt: Point, v1: Point, v2: Point, v3: Point) => {
|
|
||||||
const d1 = triangleSign(pt, v1, v2);
|
|
||||||
const d2 = triangleSign(pt, v2, v3);
|
|
||||||
const d3 = triangleSign(pt, v3, v1);
|
|
||||||
|
|
||||||
const has_neg = d1 < 0 || d2 < 0 || d3 < 0;
|
|
||||||
const has_pos = d1 > 0 || d2 > 0 || d3 > 0;
|
|
||||||
|
|
||||||
return !(has_neg && has_pos);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const magnitudeSq = (vector: Vector) =>
|
|
||||||
vector[0] * vector[0] + vector[1] * vector[1];
|
|
||||||
|
|
||||||
export const magnitude = (vector: Vector) => Math.sqrt(magnitudeSq(vector));
|
|
||||||
|
|
||||||
export const normalize = (vector: Vector): Vector => {
|
|
||||||
const m = magnitude(vector);
|
|
||||||
|
|
||||||
return [vector[0] / m, vector[1] / m];
|
|
||||||
};
|
|
||||||
|
|
||||||
export const addVectors = (
|
|
||||||
vec1: Readonly<Vector>,
|
|
||||||
vec2: Readonly<Vector>,
|
|
||||||
): Vector => [vec1[0] + vec2[0], vec1[1] + vec2[1]];
|
|
||||||
|
|
||||||
export const subtractVectors = (
|
|
||||||
vec1: Readonly<Vector>,
|
|
||||||
vec2: Readonly<Vector>,
|
|
||||||
): Vector => [vec1[0] - vec2[0], vec1[1] - vec2[1]];
|
|
||||||
|
|
||||||
export const pointInsideBounds = (p: Point, bounds: Bounds): boolean =>
|
|
||||||
p[0] > bounds[0] && p[0] < bounds[2] && p[1] > bounds[1] && p[1] < bounds[3];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the axis-aligned bounding box for a given element
|
|
||||||
*/
|
|
||||||
export const aabbForElement = (
|
|
||||||
element: Readonly<ExcalidrawElement>,
|
|
||||||
offset?: [number, number, number, number],
|
|
||||||
) => {
|
|
||||||
const bbox = {
|
|
||||||
minX: element.x,
|
|
||||||
minY: element.y,
|
|
||||||
maxX: element.x + element.width,
|
|
||||||
maxY: element.y + element.height,
|
|
||||||
midX: element.x + element.width / 2,
|
|
||||||
midY: element.y + element.height / 2,
|
|
||||||
};
|
|
||||||
|
|
||||||
const center = [bbox.midX, bbox.midY] as Point;
|
|
||||||
const [topLeftX, topLeftY] = rotatePoint(
|
|
||||||
[bbox.minX, bbox.minY],
|
|
||||||
center,
|
|
||||||
element.angle,
|
|
||||||
);
|
|
||||||
const [topRightX, topRightY] = rotatePoint(
|
|
||||||
[bbox.maxX, bbox.minY],
|
|
||||||
center,
|
|
||||||
element.angle,
|
|
||||||
);
|
|
||||||
const [bottomRightX, bottomRightY] = rotatePoint(
|
|
||||||
[bbox.maxX, bbox.maxY],
|
|
||||||
center,
|
|
||||||
element.angle,
|
|
||||||
);
|
|
||||||
const [bottomLeftX, bottomLeftY] = rotatePoint(
|
|
||||||
[bbox.minX, bbox.maxY],
|
|
||||||
center,
|
|
||||||
element.angle,
|
|
||||||
);
|
|
||||||
|
|
||||||
const bounds = [
|
|
||||||
Math.min(topLeftX, topRightX, bottomRightX, bottomLeftX),
|
|
||||||
Math.min(topLeftY, topRightY, bottomRightY, bottomLeftY),
|
|
||||||
Math.max(topLeftX, topRightX, bottomRightX, bottomLeftX),
|
|
||||||
Math.max(topLeftY, topRightY, bottomRightY, bottomLeftY),
|
|
||||||
] as Bounds;
|
|
||||||
|
|
||||||
if (offset) {
|
|
||||||
const [topOffset, rightOffset, downOffset, leftOffset] = offset;
|
|
||||||
return [
|
|
||||||
bounds[0] - leftOffset,
|
|
||||||
bounds[1] - topOffset,
|
|
||||||
bounds[2] + rightOffset,
|
|
||||||
bounds[3] + downOffset,
|
|
||||||
] as Bounds;
|
|
||||||
}
|
|
||||||
|
|
||||||
return bounds;
|
|
||||||
};
|
|
||||||
|
|
||||||
type PolarCoords = [number, number];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the polar coordinates for the given carthesian point represented by
|
|
||||||
* (x, y) for the center point 0,0 where the first number returned is the radius,
|
|
||||||
* the second is the angle in radians.
|
|
||||||
*/
|
|
||||||
export const carthesian2Polar = ([x, y]: Point): PolarCoords => [
|
|
||||||
Math.hypot(x, y),
|
|
||||||
Math.atan2(y, x),
|
|
||||||
];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Angles are in radians and centered on 0, 0. Zero radians on a 1 radius circle
|
|
||||||
* corresponds to (1, 0) carthesian coordinates (point), i.e. to the "right".
|
|
||||||
*/
|
|
||||||
type SymmetricArc = { radius: number; startAngle: number; endAngle: number };
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Determines if a carthesian point lies on a symmetric arc, i.e. an arc which
|
|
||||||
* is part of a circle contour centered on 0, 0.
|
|
||||||
*/
|
|
||||||
export const isPointOnSymmetricArc = (
|
|
||||||
{ radius: arcRadius, startAngle, endAngle }: SymmetricArc,
|
|
||||||
point: Point,
|
|
||||||
): boolean => {
|
|
||||||
const [radius, angle] = carthesian2Polar(point);
|
|
||||||
|
|
||||||
return startAngle < endAngle
|
|
||||||
? Math.abs(radius - arcRadius) < 0.0000001 &&
|
|
||||||
startAngle <= angle &&
|
|
||||||
endAngle >= angle
|
|
||||||
: startAngle <= angle || endAngle >= angle;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getCenterForBounds = (bounds: Bounds): Point => [
|
|
||||||
bounds[0] + (bounds[2] - bounds[0]) / 2,
|
|
||||||
bounds[1] + (bounds[3] - bounds[1]) / 2,
|
|
||||||
];
|
|
||||||
|
|
||||||
export const getCenterForElement = (element: ExcalidrawElement): Point => [
|
|
||||||
element.x + element.width / 2,
|
|
||||||
element.y + element.height / 2,
|
|
||||||
];
|
|
||||||
|
|
||||||
export const aabbsOverlapping = (a: Bounds, b: Bounds) =>
|
|
||||||
pointInsideBounds([a[0], a[1]], b) ||
|
|
||||||
pointInsideBounds([a[2], a[1]], b) ||
|
|
||||||
pointInsideBounds([a[2], a[3]], b) ||
|
|
||||||
pointInsideBounds([a[0], a[3]], b) ||
|
|
||||||
pointInsideBounds([b[0], b[1]], a) ||
|
|
||||||
pointInsideBounds([b[2], b[1]], a) ||
|
|
||||||
pointInsideBounds([b[2], b[3]], a) ||
|
|
||||||
pointInsideBounds([b[0], b[3]], a);
|
|
||||||
|
|
||||||
export const clamp = (value: number, min: number, max: number) => {
|
|
||||||
return Math.min(Math.max(value, min), max);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const round = (value: number, precision: number) => {
|
|
||||||
const multiplier = Math.pow(10, precision);
|
|
||||||
return Math.round((value + Number.EPSILON) * multiplier) / multiplier;
|
|
||||||
};
|
|
|
@ -1,6 +1,8 @@
|
||||||
import type { Point } from "./types";
|
import { pointFromPair, type GlobalPoint, type LocalPoint } from "../math";
|
||||||
|
|
||||||
export const getSizeFromPoints = (points: readonly Point[]) => {
|
export const getSizeFromPoints = (
|
||||||
|
points: readonly (GlobalPoint | LocalPoint)[],
|
||||||
|
) => {
|
||||||
const xs = points.map((point) => point[0]);
|
const xs = points.map((point) => point[0]);
|
||||||
const ys = points.map((point) => point[1]);
|
const ys = points.map((point) => point[1]);
|
||||||
return {
|
return {
|
||||||
|
@ -10,7 +12,7 @@ export const getSizeFromPoints = (points: readonly Point[]) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
/** @arg dimension, 0 for rescaling only x, 1 for y */
|
/** @arg dimension, 0 for rescaling only x, 1 for y */
|
||||||
export const rescalePoints = (
|
export const rescalePoints = <Point extends GlobalPoint | LocalPoint>(
|
||||||
dimension: 0 | 1,
|
dimension: 0 | 1,
|
||||||
newSize: number,
|
newSize: number,
|
||||||
points: readonly Point[],
|
points: readonly Point[],
|
||||||
|
@ -31,7 +33,7 @@ export const rescalePoints = (
|
||||||
if (newCoordinate < nextMinCoordinate) {
|
if (newCoordinate < nextMinCoordinate) {
|
||||||
nextMinCoordinate = newCoordinate;
|
nextMinCoordinate = newCoordinate;
|
||||||
}
|
}
|
||||||
return newPoint as unknown as Point;
|
return newPoint as Point;
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!normalize) {
|
if (!normalize) {
|
||||||
|
@ -45,11 +47,13 @@ export const rescalePoints = (
|
||||||
|
|
||||||
const translation = minCoordinate - nextMinCoordinate;
|
const translation = minCoordinate - nextMinCoordinate;
|
||||||
|
|
||||||
const nextPoints = scaledPoints.map(
|
const nextPoints = scaledPoints.map((scaledPoint) =>
|
||||||
(scaledPoint) =>
|
pointFromPair<Point>(
|
||||||
scaledPoint.map((value, currentDimension) => {
|
scaledPoint.map((value, currentDimension) => {
|
||||||
return currentDimension === dimension ? value + translation : value;
|
return currentDimension === dimension ? value + translation : value;
|
||||||
}) as [number, number],
|
}) as [number, number],
|
||||||
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
return nextPoints;
|
return nextPoints;
|
||||||
};
|
};
|
||||||
|
|
|
@ -30,8 +30,12 @@ import {
|
||||||
shouldShowBoundingBox,
|
shouldShowBoundingBox,
|
||||||
} from "../element/transformHandles";
|
} from "../element/transformHandles";
|
||||||
import { arrayToMap, throttleRAF } from "../utils";
|
import { arrayToMap, throttleRAF } from "../utils";
|
||||||
import type { InteractiveCanvasAppState, Point } from "../types";
|
import {
|
||||||
import { DEFAULT_TRANSFORM_HANDLE_SPACING, FRAME_STYLE } from "../constants";
|
DEFAULT_TRANSFORM_HANDLE_SPACING,
|
||||||
|
FRAME_STYLE,
|
||||||
|
THEME,
|
||||||
|
} from "../constants";
|
||||||
|
import { type InteractiveCanvasAppState } from "../types";
|
||||||
|
|
||||||
import { renderSnaps } from "../renderer/renderSnaps";
|
import { renderSnaps } from "../renderer/renderSnaps";
|
||||||
|
|
||||||
|
@ -69,7 +73,8 @@ import type {
|
||||||
InteractiveSceneRenderConfig,
|
InteractiveSceneRenderConfig,
|
||||||
RenderableElementsMap,
|
RenderableElementsMap,
|
||||||
} from "../scene/types";
|
} from "../scene/types";
|
||||||
import { getCornerRadius } from "../math";
|
import type { GlobalPoint, LocalPoint, Radians } from "../../math";
|
||||||
|
import { getCornerRadius } from "../shapes";
|
||||||
|
|
||||||
const renderLinearElementPointHighlight = (
|
const renderLinearElementPointHighlight = (
|
||||||
context: CanvasRenderingContext2D,
|
context: CanvasRenderingContext2D,
|
||||||
|
@ -101,7 +106,7 @@ const renderLinearElementPointHighlight = (
|
||||||
context.restore();
|
context.restore();
|
||||||
};
|
};
|
||||||
|
|
||||||
const highlightPoint = (
|
const highlightPoint = <Point extends LocalPoint | GlobalPoint>(
|
||||||
point: Point,
|
point: Point,
|
||||||
context: CanvasRenderingContext2D,
|
context: CanvasRenderingContext2D,
|
||||||
appState: InteractiveCanvasAppState,
|
appState: InteractiveCanvasAppState,
|
||||||
|
@ -168,7 +173,7 @@ const strokeDiamondWithRotation = (
|
||||||
context.restore();
|
context.restore();
|
||||||
};
|
};
|
||||||
|
|
||||||
const renderSingleLinearPoint = (
|
const renderSingleLinearPoint = <Point extends GlobalPoint | LocalPoint>(
|
||||||
context: CanvasRenderingContext2D,
|
context: CanvasRenderingContext2D,
|
||||||
appState: InteractiveCanvasAppState,
|
appState: InteractiveCanvasAppState,
|
||||||
point: Point,
|
point: Point,
|
||||||
|
@ -499,7 +504,7 @@ const renderLinearPointHandles = (
|
||||||
element,
|
element,
|
||||||
elementsMap,
|
elementsMap,
|
||||||
appState,
|
appState,
|
||||||
).filter((midPoint) => midPoint !== null) as Point[];
|
).filter((midPoint): midPoint is GlobalPoint => midPoint !== null);
|
||||||
|
|
||||||
midPoints.forEach((segmentMidPoint) => {
|
midPoints.forEach((segmentMidPoint) => {
|
||||||
if (
|
if (
|
||||||
|
@ -931,7 +936,7 @@ const _renderInteractiveScene = ({
|
||||||
context.setLineDash(initialLineDash);
|
context.setLineDash(initialLineDash);
|
||||||
const transformHandles = getTransformHandlesFromCoords(
|
const transformHandles = getTransformHandlesFromCoords(
|
||||||
[x1, y1, x2, y2, (x1 + x2) / 2, (y1 + y2) / 2],
|
[x1, y1, x2, y2, (x1 + x2) / 2, (y1 + y2) / 2],
|
||||||
0,
|
0 as Radians,
|
||||||
appState.zoom,
|
appState.zoom,
|
||||||
"mouse",
|
"mouse",
|
||||||
isFrameSelected
|
isFrameSelected
|
||||||
|
@ -951,9 +956,48 @@ const _renderInteractiveScene = ({
|
||||||
context.restore();
|
context.restore();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
appState.searchMatches.forEach(({ id, focus, matchedLines }) => {
|
||||||
|
const element = elementsMap.get(id);
|
||||||
|
|
||||||
|
if (element && isTextElement(element)) {
|
||||||
|
const [elementX1, elementY1, , , cx, cy] = getElementAbsoluteCoords(
|
||||||
|
element,
|
||||||
|
elementsMap,
|
||||||
|
true,
|
||||||
|
);
|
||||||
|
|
||||||
|
context.save();
|
||||||
|
if (appState.theme === THEME.LIGHT) {
|
||||||
|
if (focus) {
|
||||||
|
context.fillStyle = "rgba(255, 124, 0, 0.4)";
|
||||||
|
} else {
|
||||||
|
context.fillStyle = "rgba(255, 226, 0, 0.4)";
|
||||||
|
}
|
||||||
|
} else if (focus) {
|
||||||
|
context.fillStyle = "rgba(229, 82, 0, 0.4)";
|
||||||
|
} else {
|
||||||
|
context.fillStyle = "rgba(99, 52, 0, 0.4)";
|
||||||
|
}
|
||||||
|
|
||||||
|
context.translate(appState.scrollX, appState.scrollY);
|
||||||
|
context.translate(cx, cy);
|
||||||
|
context.rotate(element.angle);
|
||||||
|
|
||||||
|
matchedLines.forEach((matchedLine) => {
|
||||||
|
context.fillRect(
|
||||||
|
elementX1 + matchedLine.offsetX - cx,
|
||||||
|
elementY1 + matchedLine.offsetY - cy,
|
||||||
|
matchedLine.width,
|
||||||
|
matchedLine.height,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
context.restore();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
renderSnaps(context, appState);
|
renderSnaps(context, appState);
|
||||||
|
|
||||||
// Reset zoom
|
|
||||||
context.restore();
|
context.restore();
|
||||||
|
|
||||||
renderRemoteCursors({
|
renderRemoteCursors({
|
||||||
|
|
|
@ -27,7 +27,6 @@ import type {
|
||||||
InteractiveCanvasRenderConfig,
|
InteractiveCanvasRenderConfig,
|
||||||
} from "../scene/types";
|
} from "../scene/types";
|
||||||
import { distance, getFontString, isRTL } from "../utils";
|
import { distance, getFontString, isRTL } from "../utils";
|
||||||
import { getCornerRadius, isRightAngle } from "../math";
|
|
||||||
import rough from "roughjs/bin/rough";
|
import rough from "roughjs/bin/rough";
|
||||||
import type {
|
import type {
|
||||||
AppState,
|
AppState,
|
||||||
|
@ -60,6 +59,8 @@ import { LinearElementEditor } from "../element/linearElementEditor";
|
||||||
import { getContainingFrame } from "../frame";
|
import { getContainingFrame } from "../frame";
|
||||||
import { ShapeCache } from "../scene/ShapeCache";
|
import { ShapeCache } from "../scene/ShapeCache";
|
||||||
import { getVerticalOffset } from "../fonts";
|
import { getVerticalOffset } from "../fonts";
|
||||||
|
import { isRightAngleRads } from "../../math";
|
||||||
|
import { getCornerRadius } from "../shapes";
|
||||||
|
|
||||||
// using a stronger invert (100% vs our regular 93%) and saturate
|
// using a stronger invert (100% vs our regular 93%) and saturate
|
||||||
// as a temp hack to make images in dark theme look closer to original
|
// as a temp hack to make images in dark theme look closer to original
|
||||||
|
@ -907,7 +908,8 @@ export const renderElement = (
|
||||||
(!element.angle ||
|
(!element.angle ||
|
||||||
// or check if angle is a right angle in which case we can still
|
// or check if angle is a right angle in which case we can still
|
||||||
// disable smoothing without adversely affecting the result
|
// disable smoothing without adversely affecting the result
|
||||||
isRightAngle(element.angle))
|
// We need less-than comparison because of FP artihmetic
|
||||||
|
isRightAngleRads(element.angle))
|
||||||
) {
|
) {
|
||||||
// Disabling smoothing makes output much sharper, especially for
|
// Disabling smoothing makes output much sharper, especially for
|
||||||
// text. Unless for non-right angles, where the aliasing is really
|
// text. Unless for non-right angles, where the aliasing is really
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
|
import { point, type GlobalPoint, type LocalPoint } from "../../math";
|
||||||
import { THEME } from "../constants";
|
import { THEME } from "../constants";
|
||||||
import type { PointSnapLine, PointerSnapLine } from "../snapping";
|
import type { PointSnapLine, PointerSnapLine } from "../snapping";
|
||||||
import type { InteractiveCanvasAppState, Point } from "../types";
|
import type { InteractiveCanvasAppState } from "../types";
|
||||||
|
|
||||||
const SNAP_COLOR_LIGHT = "#ff6b6b";
|
const SNAP_COLOR_LIGHT = "#ff6b6b";
|
||||||
const SNAP_COLOR_DARK = "#ff0000";
|
const SNAP_COLOR_DARK = "#ff0000";
|
||||||
|
@ -85,7 +86,7 @@ const drawPointerSnapLine = (
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const drawCross = (
|
const drawCross = <Point extends LocalPoint | GlobalPoint>(
|
||||||
[x, y]: Point,
|
[x, y]: Point,
|
||||||
appState: InteractiveCanvasAppState,
|
appState: InteractiveCanvasAppState,
|
||||||
context: CanvasRenderingContext2D,
|
context: CanvasRenderingContext2D,
|
||||||
|
@ -106,18 +107,18 @@ const drawCross = (
|
||||||
context.restore();
|
context.restore();
|
||||||
};
|
};
|
||||||
|
|
||||||
const drawLine = (
|
const drawLine = <Point extends LocalPoint | GlobalPoint>(
|
||||||
from: Point,
|
from: Point,
|
||||||
to: Point,
|
to: Point,
|
||||||
context: CanvasRenderingContext2D,
|
context: CanvasRenderingContext2D,
|
||||||
) => {
|
) => {
|
||||||
context.beginPath();
|
context.beginPath();
|
||||||
context.lineTo(...from);
|
context.lineTo(from[0], from[1]);
|
||||||
context.lineTo(...to);
|
context.lineTo(to[0], to[1]);
|
||||||
context.stroke();
|
context.stroke();
|
||||||
};
|
};
|
||||||
|
|
||||||
const drawGapLine = (
|
const drawGapLine = <Point extends LocalPoint | GlobalPoint>(
|
||||||
from: Point,
|
from: Point,
|
||||||
to: Point,
|
to: Point,
|
||||||
direction: "horizontal" | "vertical",
|
direction: "horizontal" | "vertical",
|
||||||
|
@ -138,24 +139,28 @@ const drawGapLine = (
|
||||||
const halfPoint = [(from[0] + to[0]) / 2, from[1]];
|
const halfPoint = [(from[0] + to[0]) / 2, from[1]];
|
||||||
// (1)
|
// (1)
|
||||||
if (!appState.zenModeEnabled) {
|
if (!appState.zenModeEnabled) {
|
||||||
drawLine([from[0], from[1] - FULL], [from[0], from[1] + FULL], context);
|
drawLine(
|
||||||
|
point(from[0], from[1] - FULL),
|
||||||
|
point(from[0], from[1] + FULL),
|
||||||
|
context,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// (3)
|
// (3)
|
||||||
drawLine(
|
drawLine(
|
||||||
[halfPoint[0] - QUARTER, halfPoint[1] - HALF],
|
point(halfPoint[0] - QUARTER, halfPoint[1] - HALF),
|
||||||
[halfPoint[0] - QUARTER, halfPoint[1] + HALF],
|
point(halfPoint[0] - QUARTER, halfPoint[1] + HALF),
|
||||||
context,
|
context,
|
||||||
);
|
);
|
||||||
drawLine(
|
drawLine(
|
||||||
[halfPoint[0] + QUARTER, halfPoint[1] - HALF],
|
point(halfPoint[0] + QUARTER, halfPoint[1] - HALF),
|
||||||
[halfPoint[0] + QUARTER, halfPoint[1] + HALF],
|
point(halfPoint[0] + QUARTER, halfPoint[1] + HALF),
|
||||||
context,
|
context,
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!appState.zenModeEnabled) {
|
if (!appState.zenModeEnabled) {
|
||||||
// (4)
|
// (4)
|
||||||
drawLine([to[0], to[1] - FULL], [to[0], to[1] + FULL], context);
|
drawLine(point(to[0], to[1] - FULL), point(to[0], to[1] + FULL), context);
|
||||||
|
|
||||||
// (2)
|
// (2)
|
||||||
drawLine(from, to, context);
|
drawLine(from, to, context);
|
||||||
|
@ -164,24 +169,28 @@ const drawGapLine = (
|
||||||
const halfPoint = [from[0], (from[1] + to[1]) / 2];
|
const halfPoint = [from[0], (from[1] + to[1]) / 2];
|
||||||
// (1)
|
// (1)
|
||||||
if (!appState.zenModeEnabled) {
|
if (!appState.zenModeEnabled) {
|
||||||
drawLine([from[0] - FULL, from[1]], [from[0] + FULL, from[1]], context);
|
drawLine(
|
||||||
|
point(from[0] - FULL, from[1]),
|
||||||
|
point(from[0] + FULL, from[1]),
|
||||||
|
context,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// (3)
|
// (3)
|
||||||
drawLine(
|
drawLine(
|
||||||
[halfPoint[0] - HALF, halfPoint[1] - QUARTER],
|
point(halfPoint[0] - HALF, halfPoint[1] - QUARTER),
|
||||||
[halfPoint[0] + HALF, halfPoint[1] - QUARTER],
|
point(halfPoint[0] + HALF, halfPoint[1] - QUARTER),
|
||||||
context,
|
context,
|
||||||
);
|
);
|
||||||
drawLine(
|
drawLine(
|
||||||
[halfPoint[0] - HALF, halfPoint[1] + QUARTER],
|
point(halfPoint[0] - HALF, halfPoint[1] + QUARTER),
|
||||||
[halfPoint[0] + HALF, halfPoint[1] + QUARTER],
|
point(halfPoint[0] + HALF, halfPoint[1] + QUARTER),
|
||||||
context,
|
context,
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!appState.zenModeEnabled) {
|
if (!appState.zenModeEnabled) {
|
||||||
// (4)
|
// (4)
|
||||||
drawLine([to[0] - FULL, to[1]], [to[0] + FULL, to[1]], context);
|
drawLine(point(to[0] - FULL, to[1]), point(to[0] + FULL, to[1]), context);
|
||||||
|
|
||||||
// (2)
|
// (2)
|
||||||
drawLine(from, to, context);
|
drawLine(from, to, context);
|
||||||
|
|
|
@ -30,13 +30,13 @@ import type {
|
||||||
NonDeletedExcalidrawElement,
|
NonDeletedExcalidrawElement,
|
||||||
} from "../element/types";
|
} from "../element/types";
|
||||||
import { getContainingFrame } from "../frame";
|
import { getContainingFrame } from "../frame";
|
||||||
import { getCornerRadius, isPathALoop } from "../math";
|
|
||||||
import { ShapeCache } from "../scene/ShapeCache";
|
import { ShapeCache } from "../scene/ShapeCache";
|
||||||
import type { RenderableElementsMap, SVGRenderConfig } from "../scene/types";
|
import type { RenderableElementsMap, SVGRenderConfig } from "../scene/types";
|
||||||
import type { AppState, BinaryFiles } from "../types";
|
import type { AppState, BinaryFiles } from "../types";
|
||||||
import { getFontFamilyString, isRTL, isTestEnv } from "../utils";
|
import { getFontFamilyString, isRTL, isTestEnv } from "../utils";
|
||||||
import { getFreeDrawSvgPath, IMAGE_INVERT_FILTER } from "./renderElement";
|
import { getFreeDrawSvgPath, IMAGE_INVERT_FILTER } from "./renderElement";
|
||||||
import { getVerticalOffset } from "../fonts";
|
import { getVerticalOffset } from "../fonts";
|
||||||
|
import { getCornerRadius, isPathALoop } from "../shapes";
|
||||||
|
|
||||||
const roughSVGDrawWithPrecision = (
|
const roughSVGDrawWithPrecision = (
|
||||||
rsvg: RoughSVG,
|
rsvg: RoughSVG,
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
import type { Point as RoughPoint } from "roughjs/bin/geometry";
|
||||||
import type { Drawable, Options } from "roughjs/bin/core";
|
import type { Drawable, Options } from "roughjs/bin/core";
|
||||||
import type { RoughGenerator } from "roughjs/bin/generator";
|
import type { RoughGenerator } from "roughjs/bin/generator";
|
||||||
import { getDiamondPoints, getArrowheadPoints } from "../element";
|
import { getDiamondPoints, getArrowheadPoints } from "../element";
|
||||||
|
@ -9,7 +10,6 @@ import type {
|
||||||
ExcalidrawLinearElement,
|
ExcalidrawLinearElement,
|
||||||
Arrowhead,
|
Arrowhead,
|
||||||
} from "../element/types";
|
} from "../element/types";
|
||||||
import { isPathALoop, getCornerRadius, distanceSq2d } from "../math";
|
|
||||||
import { generateFreeDrawShape } from "../renderer/renderElement";
|
import { generateFreeDrawShape } from "../renderer/renderElement";
|
||||||
import { isTransparent, assertNever } from "../utils";
|
import { isTransparent, assertNever } from "../utils";
|
||||||
import { simplify } from "points-on-curve";
|
import { simplify } from "points-on-curve";
|
||||||
|
@ -23,6 +23,13 @@ import {
|
||||||
} from "../element/typeChecks";
|
} from "../element/typeChecks";
|
||||||
import { canChangeRoundness } from "./comparisons";
|
import { canChangeRoundness } from "./comparisons";
|
||||||
import type { EmbedsValidationStatus } from "../types";
|
import type { EmbedsValidationStatus } from "../types";
|
||||||
|
import {
|
||||||
|
point,
|
||||||
|
pointDistance,
|
||||||
|
type GlobalPoint,
|
||||||
|
type LocalPoint,
|
||||||
|
} from "../../math";
|
||||||
|
import { getCornerRadius, isPathALoop } from "../shapes";
|
||||||
|
|
||||||
const getDashArrayDashed = (strokeWidth: number) => [8, 8 + strokeWidth];
|
const getDashArrayDashed = (strokeWidth: number) => [8, 8 + strokeWidth];
|
||||||
|
|
||||||
|
@ -399,12 +406,14 @@ export const _generateElementShape = (
|
||||||
|
|
||||||
// points array can be empty in the beginning, so it is important to add
|
// points array can be empty in the beginning, so it is important to add
|
||||||
// initial position to it
|
// initial position to it
|
||||||
const points = element.points.length ? element.points : [[0, 0]];
|
const points = element.points.length
|
||||||
|
? element.points
|
||||||
|
: [point<LocalPoint>(0, 0)];
|
||||||
|
|
||||||
if (isElbowArrow(element)) {
|
if (isElbowArrow(element)) {
|
||||||
shape = [
|
shape = [
|
||||||
generator.path(
|
generator.path(
|
||||||
generateElbowArrowShape(points as [number, number][], 16),
|
generateElbowArrowShape(points, 16),
|
||||||
generateRoughOptions(element, true),
|
generateRoughOptions(element, true),
|
||||||
),
|
),
|
||||||
];
|
];
|
||||||
|
@ -412,12 +421,16 @@ export const _generateElementShape = (
|
||||||
// curve is always the first element
|
// curve is always the first element
|
||||||
// this simplifies finding the curve for an element
|
// this simplifies finding the curve for an element
|
||||||
if (options.fill) {
|
if (options.fill) {
|
||||||
shape = [generator.polygon(points as [number, number][], options)];
|
shape = [
|
||||||
|
generator.polygon(points as unknown as RoughPoint[], options),
|
||||||
|
];
|
||||||
} else {
|
} else {
|
||||||
shape = [generator.linearPath(points as [number, number][], options)];
|
shape = [
|
||||||
|
generator.linearPath(points as unknown as RoughPoint[], options),
|
||||||
|
];
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
shape = [generator.curve(points as [number, number][], options)];
|
shape = [generator.curve(points as unknown as RoughPoint[], options)];
|
||||||
}
|
}
|
||||||
|
|
||||||
// add lines only in arrow
|
// add lines only in arrow
|
||||||
|
@ -491,8 +504,8 @@ export const _generateElementShape = (
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const generateElbowArrowShape = (
|
const generateElbowArrowShape = <Point extends GlobalPoint | LocalPoint>(
|
||||||
points: [number, number][],
|
points: readonly Point[],
|
||||||
radius: number,
|
radius: number,
|
||||||
) => {
|
) => {
|
||||||
const subpoints = [] as [number, number][];
|
const subpoints = [] as [number, number][];
|
||||||
|
@ -501,8 +514,8 @@ const generateElbowArrowShape = (
|
||||||
const next = points[i + 1];
|
const next = points[i + 1];
|
||||||
const corner = Math.min(
|
const corner = Math.min(
|
||||||
radius,
|
radius,
|
||||||
Math.sqrt(distanceSq2d(points[i], next)) / 2,
|
pointDistance(points[i], next) / 2,
|
||||||
Math.sqrt(distanceSq2d(points[i], prev)) / 2,
|
pointDistance(points[i], prev) / 2,
|
||||||
);
|
);
|
||||||
|
|
||||||
if (prev[0] < points[i][0] && prev[1] === points[i][1]) {
|
if (prev[0] < points[i][0] && prev[1] === points[i][1]) {
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
|
import { clamp, round } from "../../math";
|
||||||
import { MAX_ZOOM, MIN_ZOOM } from "../constants";
|
import { MAX_ZOOM, MIN_ZOOM } from "../constants";
|
||||||
import { clamp, round } from "../math";
|
|
||||||
import type { NormalizedZoomValue } from "../types";
|
import type { NormalizedZoomValue } from "../types";
|
||||||
|
|
||||||
export const getNormalizedZoom = (zoom: number): NormalizedZoomValue => {
|
export const getNormalizedZoom = (zoom: number): NormalizedZoomValue => {
|
||||||
|
|
|
@ -1,5 +1,16 @@
|
||||||
|
import {
|
||||||
|
isPoint,
|
||||||
|
point,
|
||||||
|
pointDistance,
|
||||||
|
pointFromPair,
|
||||||
|
pointRotateRads,
|
||||||
|
pointsEqual,
|
||||||
|
type GlobalPoint,
|
||||||
|
type LocalPoint,
|
||||||
|
} from "../math";
|
||||||
import {
|
import {
|
||||||
getClosedCurveShape,
|
getClosedCurveShape,
|
||||||
|
getCurvePathOps,
|
||||||
getCurveShape,
|
getCurveShape,
|
||||||
getEllipseShape,
|
getEllipseShape,
|
||||||
getFreedrawShape,
|
getFreedrawShape,
|
||||||
|
@ -18,13 +29,27 @@ import {
|
||||||
SelectionIcon,
|
SelectionIcon,
|
||||||
TextIcon,
|
TextIcon,
|
||||||
} from "./components/icons";
|
} from "./components/icons";
|
||||||
|
import {
|
||||||
|
DEFAULT_ADAPTIVE_RADIUS,
|
||||||
|
DEFAULT_PROPORTIONAL_RADIUS,
|
||||||
|
LINE_CONFIRM_THRESHOLD,
|
||||||
|
ROUNDNESS,
|
||||||
|
} from "./constants";
|
||||||
import { getElementAbsoluteCoords } from "./element";
|
import { getElementAbsoluteCoords } from "./element";
|
||||||
|
import type { Bounds } from "./element/bounds";
|
||||||
import { shouldTestInside } from "./element/collision";
|
import { shouldTestInside } from "./element/collision";
|
||||||
import { LinearElementEditor } from "./element/linearElementEditor";
|
import { LinearElementEditor } from "./element/linearElementEditor";
|
||||||
import { getBoundTextElement } from "./element/textElement";
|
import { getBoundTextElement } from "./element/textElement";
|
||||||
import type { ElementsMap, ExcalidrawElement } from "./element/types";
|
import type {
|
||||||
|
ElementsMap,
|
||||||
|
ExcalidrawElement,
|
||||||
|
ExcalidrawLinearElement,
|
||||||
|
NonDeleted,
|
||||||
|
} from "./element/types";
|
||||||
import { KEYS } from "./keys";
|
import { KEYS } from "./keys";
|
||||||
import { ShapeCache } from "./scene/ShapeCache";
|
import { ShapeCache } from "./scene/ShapeCache";
|
||||||
|
import type { NormalizedZoomValue, Zoom } from "./types";
|
||||||
|
import { invariant } from "./utils";
|
||||||
|
|
||||||
export const SHAPES = [
|
export const SHAPES = [
|
||||||
{
|
{
|
||||||
|
@ -116,10 +141,10 @@ export const findShapeByKey = (key: string) => {
|
||||||
* get the pure geometric shape of an excalidraw element
|
* get the pure geometric shape of an excalidraw element
|
||||||
* which is then used for hit detection
|
* which is then used for hit detection
|
||||||
*/
|
*/
|
||||||
export const getElementShape = (
|
export const getElementShape = <Point extends GlobalPoint | LocalPoint>(
|
||||||
element: ExcalidrawElement,
|
element: ExcalidrawElement,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): GeometricShape => {
|
): GeometricShape<Point> => {
|
||||||
switch (element.type) {
|
switch (element.type) {
|
||||||
case "rectangle":
|
case "rectangle":
|
||||||
case "diamond":
|
case "diamond":
|
||||||
|
@ -139,17 +164,19 @@ export const getElementShape = (
|
||||||
const [, , , , cx, cy] = getElementAbsoluteCoords(element, elementsMap);
|
const [, , , , cx, cy] = getElementAbsoluteCoords(element, elementsMap);
|
||||||
|
|
||||||
return shouldTestInside(element)
|
return shouldTestInside(element)
|
||||||
? getClosedCurveShape(
|
? getClosedCurveShape<Point>(
|
||||||
element,
|
element,
|
||||||
roughShape,
|
roughShape,
|
||||||
[element.x, element.y],
|
point<Point>(element.x, element.y),
|
||||||
element.angle,
|
element.angle,
|
||||||
[cx, cy],
|
point(cx, cy),
|
||||||
)
|
)
|
||||||
: getCurveShape(roughShape, [element.x, element.y], element.angle, [
|
: getCurveShape<Point>(
|
||||||
cx,
|
roughShape,
|
||||||
cy,
|
point<Point>(element.x, element.y),
|
||||||
]);
|
element.angle,
|
||||||
|
point(cx, cy),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
case "ellipse":
|
case "ellipse":
|
||||||
|
@ -157,15 +184,19 @@ export const getElementShape = (
|
||||||
|
|
||||||
case "freedraw": {
|
case "freedraw": {
|
||||||
const [, , , , cx, cy] = getElementAbsoluteCoords(element, elementsMap);
|
const [, , , , cx, cy] = getElementAbsoluteCoords(element, elementsMap);
|
||||||
return getFreedrawShape(element, [cx, cy], shouldTestInside(element));
|
return getFreedrawShape(
|
||||||
|
element,
|
||||||
|
point(cx, cy),
|
||||||
|
shouldTestInside(element),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getBoundTextShape = (
|
export const getBoundTextShape = <Point extends GlobalPoint | LocalPoint>(
|
||||||
element: ExcalidrawElement,
|
element: ExcalidrawElement,
|
||||||
elementsMap: ElementsMap,
|
elementsMap: ElementsMap,
|
||||||
): GeometricShape | null => {
|
): GeometricShape<Point> | null => {
|
||||||
const boundTextElement = getBoundTextElement(element, elementsMap);
|
const boundTextElement = getBoundTextElement(element, elementsMap);
|
||||||
|
|
||||||
if (boundTextElement) {
|
if (boundTextElement) {
|
||||||
|
@ -189,3 +220,274 @@ export const getBoundTextShape = (
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const getControlPointsForBezierCurve = <
|
||||||
|
P extends GlobalPoint | LocalPoint,
|
||||||
|
>(
|
||||||
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
|
endPoint: P,
|
||||||
|
) => {
|
||||||
|
const shape = ShapeCache.generateElementShape(element, null);
|
||||||
|
if (!shape) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ops = getCurvePathOps(shape[0]);
|
||||||
|
let currentP = point<P>(0, 0);
|
||||||
|
let index = 0;
|
||||||
|
let minDistance = Infinity;
|
||||||
|
let controlPoints: P[] | null = null;
|
||||||
|
|
||||||
|
while (index < ops.length) {
|
||||||
|
const { op, data } = ops[index];
|
||||||
|
if (op === "move") {
|
||||||
|
invariant(
|
||||||
|
isPoint(data),
|
||||||
|
"The returned ops is not compatible with a point",
|
||||||
|
);
|
||||||
|
currentP = pointFromPair(data);
|
||||||
|
}
|
||||||
|
if (op === "bcurveTo") {
|
||||||
|
const p0 = currentP;
|
||||||
|
const p1 = point<P>(data[0], data[1]);
|
||||||
|
const p2 = point<P>(data[2], data[3]);
|
||||||
|
const p3 = point<P>(data[4], data[5]);
|
||||||
|
const distance = pointDistance(p3, endPoint);
|
||||||
|
if (distance < minDistance) {
|
||||||
|
minDistance = distance;
|
||||||
|
controlPoints = [p0, p1, p2, p3];
|
||||||
|
}
|
||||||
|
currentP = p3;
|
||||||
|
}
|
||||||
|
index++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return controlPoints;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getBezierXY = <P extends GlobalPoint | LocalPoint>(
|
||||||
|
p0: P,
|
||||||
|
p1: P,
|
||||||
|
p2: P,
|
||||||
|
p3: P,
|
||||||
|
t: number,
|
||||||
|
): P => {
|
||||||
|
const equation = (t: number, idx: number) =>
|
||||||
|
Math.pow(1 - t, 3) * p3[idx] +
|
||||||
|
3 * t * Math.pow(1 - t, 2) * p2[idx] +
|
||||||
|
3 * Math.pow(t, 2) * (1 - t) * p1[idx] +
|
||||||
|
p0[idx] * Math.pow(t, 3);
|
||||||
|
const tx = equation(t, 0);
|
||||||
|
const ty = equation(t, 1);
|
||||||
|
return point(tx, ty);
|
||||||
|
};
|
||||||
|
|
||||||
|
const getPointsInBezierCurve = <P extends GlobalPoint | LocalPoint>(
|
||||||
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
|
endPoint: P,
|
||||||
|
) => {
|
||||||
|
const controlPoints: P[] = getControlPointsForBezierCurve(element, endPoint)!;
|
||||||
|
if (!controlPoints) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
const pointsOnCurve: P[] = [];
|
||||||
|
let t = 1;
|
||||||
|
// Take 20 points on curve for better accuracy
|
||||||
|
while (t > 0) {
|
||||||
|
const p = getBezierXY(
|
||||||
|
controlPoints[0],
|
||||||
|
controlPoints[1],
|
||||||
|
controlPoints[2],
|
||||||
|
controlPoints[3],
|
||||||
|
t,
|
||||||
|
);
|
||||||
|
pointsOnCurve.push(point(p[0], p[1]));
|
||||||
|
t -= 0.05;
|
||||||
|
}
|
||||||
|
if (pointsOnCurve.length) {
|
||||||
|
if (pointsEqual(pointsOnCurve.at(-1)!, endPoint)) {
|
||||||
|
pointsOnCurve.push(point(endPoint[0], endPoint[1]));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return pointsOnCurve;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getBezierCurveArcLengths = <P extends GlobalPoint | LocalPoint>(
|
||||||
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
|
endPoint: P,
|
||||||
|
) => {
|
||||||
|
const arcLengths: number[] = [];
|
||||||
|
arcLengths[0] = 0;
|
||||||
|
const points = getPointsInBezierCurve(element, endPoint);
|
||||||
|
let index = 0;
|
||||||
|
let distance = 0;
|
||||||
|
while (index < points.length - 1) {
|
||||||
|
const segmentDistance = pointDistance(points[index], points[index + 1]);
|
||||||
|
distance += segmentDistance;
|
||||||
|
arcLengths.push(distance);
|
||||||
|
index++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return arcLengths;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getBezierCurveLength = <P extends GlobalPoint | LocalPoint>(
|
||||||
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
|
endPoint: P,
|
||||||
|
) => {
|
||||||
|
const arcLengths = getBezierCurveArcLengths(element, endPoint);
|
||||||
|
return arcLengths.at(-1) as number;
|
||||||
|
};
|
||||||
|
|
||||||
|
// This maps interval to actual interval t on the curve so that when t = 0.5, its actually the point at 50% of the length
|
||||||
|
export const mapIntervalToBezierT = <P extends GlobalPoint | LocalPoint>(
|
||||||
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
|
endPoint: P,
|
||||||
|
interval: number, // The interval between 0 to 1 for which you want to find the point on the curve,
|
||||||
|
) => {
|
||||||
|
const arcLengths = getBezierCurveArcLengths(element, endPoint);
|
||||||
|
const pointsCount = arcLengths.length - 1;
|
||||||
|
const curveLength = arcLengths.at(-1) as number;
|
||||||
|
const targetLength = interval * curveLength;
|
||||||
|
let low = 0;
|
||||||
|
let high = pointsCount;
|
||||||
|
let index = 0;
|
||||||
|
// Doing a binary search to find the largest length that is less than the target length
|
||||||
|
while (low < high) {
|
||||||
|
index = Math.floor(low + (high - low) / 2);
|
||||||
|
if (arcLengths[index] < targetLength) {
|
||||||
|
low = index + 1;
|
||||||
|
} else {
|
||||||
|
high = index;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (arcLengths[index] > targetLength) {
|
||||||
|
index--;
|
||||||
|
}
|
||||||
|
if (arcLengths[index] === targetLength) {
|
||||||
|
return index / pointsCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
1 -
|
||||||
|
(index +
|
||||||
|
(targetLength - arcLengths[index]) /
|
||||||
|
(arcLengths[index + 1] - arcLengths[index])) /
|
||||||
|
pointsCount
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the axis-aligned bounding box for a given element
|
||||||
|
*/
|
||||||
|
export const aabbForElement = (
|
||||||
|
element: Readonly<ExcalidrawElement>,
|
||||||
|
offset?: [number, number, number, number],
|
||||||
|
) => {
|
||||||
|
const bbox = {
|
||||||
|
minX: element.x,
|
||||||
|
minY: element.y,
|
||||||
|
maxX: element.x + element.width,
|
||||||
|
maxY: element.y + element.height,
|
||||||
|
midX: element.x + element.width / 2,
|
||||||
|
midY: element.y + element.height / 2,
|
||||||
|
};
|
||||||
|
|
||||||
|
const center = point(bbox.midX, bbox.midY);
|
||||||
|
const [topLeftX, topLeftY] = pointRotateRads(
|
||||||
|
point(bbox.minX, bbox.minY),
|
||||||
|
center,
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
|
const [topRightX, topRightY] = pointRotateRads(
|
||||||
|
point(bbox.maxX, bbox.minY),
|
||||||
|
center,
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
|
const [bottomRightX, bottomRightY] = pointRotateRads(
|
||||||
|
point(bbox.maxX, bbox.maxY),
|
||||||
|
center,
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
|
const [bottomLeftX, bottomLeftY] = pointRotateRads(
|
||||||
|
point(bbox.minX, bbox.maxY),
|
||||||
|
center,
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
|
|
||||||
|
const bounds = [
|
||||||
|
Math.min(topLeftX, topRightX, bottomRightX, bottomLeftX),
|
||||||
|
Math.min(topLeftY, topRightY, bottomRightY, bottomLeftY),
|
||||||
|
Math.max(topLeftX, topRightX, bottomRightX, bottomLeftX),
|
||||||
|
Math.max(topLeftY, topRightY, bottomRightY, bottomLeftY),
|
||||||
|
] as Bounds;
|
||||||
|
|
||||||
|
if (offset) {
|
||||||
|
const [topOffset, rightOffset, downOffset, leftOffset] = offset;
|
||||||
|
return [
|
||||||
|
bounds[0] - leftOffset,
|
||||||
|
bounds[1] - topOffset,
|
||||||
|
bounds[2] + rightOffset,
|
||||||
|
bounds[3] + downOffset,
|
||||||
|
] as Bounds;
|
||||||
|
}
|
||||||
|
|
||||||
|
return bounds;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const pointInsideBounds = <P extends GlobalPoint | LocalPoint>(
|
||||||
|
p: P,
|
||||||
|
bounds: Bounds,
|
||||||
|
): boolean =>
|
||||||
|
p[0] > bounds[0] && p[0] < bounds[2] && p[1] > bounds[1] && p[1] < bounds[3];
|
||||||
|
|
||||||
|
export const aabbsOverlapping = (a: Bounds, b: Bounds) =>
|
||||||
|
pointInsideBounds(point(a[0], a[1]), b) ||
|
||||||
|
pointInsideBounds(point(a[2], a[1]), b) ||
|
||||||
|
pointInsideBounds(point(a[2], a[3]), b) ||
|
||||||
|
pointInsideBounds(point(a[0], a[3]), b) ||
|
||||||
|
pointInsideBounds(point(b[0], b[1]), a) ||
|
||||||
|
pointInsideBounds(point(b[2], b[1]), a) ||
|
||||||
|
pointInsideBounds(point(b[2], b[3]), a) ||
|
||||||
|
pointInsideBounds(point(b[0], b[3]), a);
|
||||||
|
|
||||||
|
export const getCornerRadius = (x: number, element: ExcalidrawElement) => {
|
||||||
|
if (
|
||||||
|
element.roundness?.type === ROUNDNESS.PROPORTIONAL_RADIUS ||
|
||||||
|
element.roundness?.type === ROUNDNESS.LEGACY
|
||||||
|
) {
|
||||||
|
return x * DEFAULT_PROPORTIONAL_RADIUS;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (element.roundness?.type === ROUNDNESS.ADAPTIVE_RADIUS) {
|
||||||
|
const fixedRadiusSize = element.roundness?.value ?? DEFAULT_ADAPTIVE_RADIUS;
|
||||||
|
|
||||||
|
const CUTOFF_SIZE = fixedRadiusSize / DEFAULT_PROPORTIONAL_RADIUS;
|
||||||
|
|
||||||
|
if (x <= CUTOFF_SIZE) {
|
||||||
|
return x * DEFAULT_PROPORTIONAL_RADIUS;
|
||||||
|
}
|
||||||
|
|
||||||
|
return fixedRadiusSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Checks if the first and last point are close enough
|
||||||
|
// to be considered a loop
|
||||||
|
export const isPathALoop = (
|
||||||
|
points: ExcalidrawLinearElement["points"],
|
||||||
|
/** supply if you want the loop detection to account for current zoom */
|
||||||
|
zoomValue: Zoom["value"] = 1 as NormalizedZoomValue,
|
||||||
|
): boolean => {
|
||||||
|
if (points.length >= 3) {
|
||||||
|
const [first, last] = [points[0], points[points.length - 1]];
|
||||||
|
const distance = pointDistance(first, last);
|
||||||
|
|
||||||
|
// Adjusting LINE_CONFIRM_THRESHOLD to current zoom so that when zoomed in
|
||||||
|
// really close we make the threshold smaller, and vice versa.
|
||||||
|
return distance <= LINE_CONFIRM_THRESHOLD / zoomValue;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
|
@ -1,3 +1,12 @@
|
||||||
|
import type { InclusiveRange } from "../math";
|
||||||
|
import {
|
||||||
|
point,
|
||||||
|
pointRotateRads,
|
||||||
|
rangeInclusive,
|
||||||
|
rangeIntersection,
|
||||||
|
rangesOverlap,
|
||||||
|
type GlobalPoint,
|
||||||
|
} from "../math";
|
||||||
import { TOOL_TYPE } from "./constants";
|
import { TOOL_TYPE } from "./constants";
|
||||||
import type { Bounds } from "./element/bounds";
|
import type { Bounds } from "./element/bounds";
|
||||||
import {
|
import {
|
||||||
|
@ -14,7 +23,6 @@ import type {
|
||||||
} from "./element/types";
|
} from "./element/types";
|
||||||
import { getMaximumGroups } from "./groups";
|
import { getMaximumGroups } from "./groups";
|
||||||
import { KEYS } from "./keys";
|
import { KEYS } from "./keys";
|
||||||
import { rangeIntersection, rangesOverlap, rotatePoint } from "./math";
|
|
||||||
import {
|
import {
|
||||||
getSelectedElements,
|
getSelectedElements,
|
||||||
getVisibleAndNonSelectedElements,
|
getVisibleAndNonSelectedElements,
|
||||||
|
@ -23,7 +31,7 @@ import type {
|
||||||
AppClassProperties,
|
AppClassProperties,
|
||||||
AppState,
|
AppState,
|
||||||
KeyboardModifiersObject,
|
KeyboardModifiersObject,
|
||||||
Point,
|
NullableGridSize,
|
||||||
} from "./types";
|
} from "./types";
|
||||||
|
|
||||||
const SNAP_DISTANCE = 8;
|
const SNAP_DISTANCE = 8;
|
||||||
|
@ -42,7 +50,7 @@ type Vector2D = {
|
||||||
y: number;
|
y: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
type PointPair = [Point, Point];
|
type PointPair = [GlobalPoint, GlobalPoint];
|
||||||
|
|
||||||
export type PointSnap = {
|
export type PointSnap = {
|
||||||
type: "point";
|
type: "point";
|
||||||
|
@ -62,9 +70,9 @@ export type Gap = {
|
||||||
// ↑ end side
|
// ↑ end side
|
||||||
startBounds: Bounds;
|
startBounds: Bounds;
|
||||||
endBounds: Bounds;
|
endBounds: Bounds;
|
||||||
startSide: [Point, Point];
|
startSide: [GlobalPoint, GlobalPoint];
|
||||||
endSide: [Point, Point];
|
endSide: [GlobalPoint, GlobalPoint];
|
||||||
overlap: [number, number];
|
overlap: InclusiveRange;
|
||||||
length: number;
|
length: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -88,7 +96,7 @@ export type Snaps = Snap[];
|
||||||
|
|
||||||
export type PointSnapLine = {
|
export type PointSnapLine = {
|
||||||
type: "points";
|
type: "points";
|
||||||
points: Point[];
|
points: GlobalPoint[];
|
||||||
};
|
};
|
||||||
|
|
||||||
export type PointerSnapLine = {
|
export type PointerSnapLine = {
|
||||||
|
@ -108,14 +116,14 @@ export type SnapLine = PointSnapLine | GapSnapLine | PointerSnapLine;
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
export class SnapCache {
|
export class SnapCache {
|
||||||
private static referenceSnapPoints: Point[] | null = null;
|
private static referenceSnapPoints: GlobalPoint[] | null = null;
|
||||||
|
|
||||||
private static visibleGaps: {
|
private static visibleGaps: {
|
||||||
verticalGaps: Gap[];
|
verticalGaps: Gap[];
|
||||||
horizontalGaps: Gap[];
|
horizontalGaps: Gap[];
|
||||||
} | null = null;
|
} | null = null;
|
||||||
|
|
||||||
public static setReferenceSnapPoints = (snapPoints: Point[] | null) => {
|
public static setReferenceSnapPoints = (snapPoints: GlobalPoint[] | null) => {
|
||||||
SnapCache.referenceSnapPoints = snapPoints;
|
SnapCache.referenceSnapPoints = snapPoints;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -191,8 +199,8 @@ export const getElementsCorners = (
|
||||||
omitCenter: false,
|
omitCenter: false,
|
||||||
boundingBoxCorners: false,
|
boundingBoxCorners: false,
|
||||||
},
|
},
|
||||||
): Point[] => {
|
): GlobalPoint[] => {
|
||||||
let result: Point[] = [];
|
let result: GlobalPoint[] = [];
|
||||||
|
|
||||||
if (elements.length === 1) {
|
if (elements.length === 1) {
|
||||||
const element = elements[0];
|
const element = elements[0];
|
||||||
|
@ -219,33 +227,53 @@ export const getElementsCorners = (
|
||||||
(element.type === "diamond" || element.type === "ellipse") &&
|
(element.type === "diamond" || element.type === "ellipse") &&
|
||||||
!boundingBoxCorners
|
!boundingBoxCorners
|
||||||
) {
|
) {
|
||||||
const leftMid = rotatePoint(
|
const leftMid = pointRotateRads<GlobalPoint>(
|
||||||
[x1, y1 + halfHeight],
|
point(x1, y1 + halfHeight),
|
||||||
[cx, cy],
|
point(cx, cy),
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
const topMid = rotatePoint([x1 + halfWidth, y1], [cx, cy], element.angle);
|
const topMid = pointRotateRads<GlobalPoint>(
|
||||||
const rightMid = rotatePoint(
|
point(x1 + halfWidth, y1),
|
||||||
[x2, y1 + halfHeight],
|
point(cx, cy),
|
||||||
[cx, cy],
|
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
const bottomMid = rotatePoint(
|
const rightMid = pointRotateRads<GlobalPoint>(
|
||||||
[x1 + halfWidth, y2],
|
point(x2, y1 + halfHeight),
|
||||||
[cx, cy],
|
point(cx, cy),
|
||||||
element.angle,
|
element.angle,
|
||||||
);
|
);
|
||||||
const center: Point = [cx, cy];
|
const bottomMid = pointRotateRads<GlobalPoint>(
|
||||||
|
point(x1 + halfWidth, y2),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
|
const center = point<GlobalPoint>(cx, cy);
|
||||||
|
|
||||||
result = omitCenter
|
result = omitCenter
|
||||||
? [leftMid, topMid, rightMid, bottomMid]
|
? [leftMid, topMid, rightMid, bottomMid]
|
||||||
: [leftMid, topMid, rightMid, bottomMid, center];
|
: [leftMid, topMid, rightMid, bottomMid, center];
|
||||||
} else {
|
} else {
|
||||||
const topLeft = rotatePoint([x1, y1], [cx, cy], element.angle);
|
const topLeft = pointRotateRads<GlobalPoint>(
|
||||||
const topRight = rotatePoint([x2, y1], [cx, cy], element.angle);
|
point(x1, y1),
|
||||||
const bottomLeft = rotatePoint([x1, y2], [cx, cy], element.angle);
|
point(cx, cy),
|
||||||
const bottomRight = rotatePoint([x2, y2], [cx, cy], element.angle);
|
element.angle,
|
||||||
const center: Point = [cx, cy];
|
);
|
||||||
|
const topRight = pointRotateRads<GlobalPoint>(
|
||||||
|
point(x2, y1),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
|
const bottomLeft = pointRotateRads<GlobalPoint>(
|
||||||
|
point(x1, y2),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
|
const bottomRight = pointRotateRads<GlobalPoint>(
|
||||||
|
point(x2, y2),
|
||||||
|
point(cx, cy),
|
||||||
|
element.angle,
|
||||||
|
);
|
||||||
|
const center = point<GlobalPoint>(cx, cy);
|
||||||
|
|
||||||
result = omitCenter
|
result = omitCenter
|
||||||
? [topLeft, topRight, bottomLeft, bottomRight]
|
? [topLeft, topRight, bottomLeft, bottomRight]
|
||||||
|
@ -259,18 +287,18 @@ export const getElementsCorners = (
|
||||||
const width = maxX - minX;
|
const width = maxX - minX;
|
||||||
const height = maxY - minY;
|
const height = maxY - minY;
|
||||||
|
|
||||||
const topLeft: Point = [minX, minY];
|
const topLeft = point<GlobalPoint>(minX, minY);
|
||||||
const topRight: Point = [maxX, minY];
|
const topRight = point<GlobalPoint>(maxX, minY);
|
||||||
const bottomLeft: Point = [minX, maxY];
|
const bottomLeft = point<GlobalPoint>(minX, maxY);
|
||||||
const bottomRight: Point = [maxX, maxY];
|
const bottomRight = point<GlobalPoint>(maxX, maxY);
|
||||||
const center: Point = [minX + width / 2, minY + height / 2];
|
const center = point<GlobalPoint>(minX + width / 2, minY + height / 2);
|
||||||
|
|
||||||
result = omitCenter
|
result = omitCenter
|
||||||
? [topLeft, topRight, bottomLeft, bottomRight]
|
? [topLeft, topRight, bottomLeft, bottomRight]
|
||||||
: [topLeft, topRight, bottomLeft, bottomRight, center];
|
: [topLeft, topRight, bottomLeft, bottomRight, center];
|
||||||
}
|
}
|
||||||
|
|
||||||
return result.map((point) => [round(point[0]), round(point[1])] as Point);
|
return result.map((p) => point(round(p[0]), round(p[1])));
|
||||||
};
|
};
|
||||||
|
|
||||||
const getReferenceElements = (
|
const getReferenceElements = (
|
||||||
|
@ -339,23 +367,20 @@ export const getVisibleGaps = (
|
||||||
|
|
||||||
if (
|
if (
|
||||||
startMaxX < endMinX &&
|
startMaxX < endMinX &&
|
||||||
rangesOverlap([startMinY, startMaxY], [endMinY, endMaxY])
|
rangesOverlap(
|
||||||
|
rangeInclusive(startMinY, startMaxY),
|
||||||
|
rangeInclusive(endMinY, endMaxY),
|
||||||
|
)
|
||||||
) {
|
) {
|
||||||
horizontalGaps.push({
|
horizontalGaps.push({
|
||||||
startBounds,
|
startBounds,
|
||||||
endBounds,
|
endBounds,
|
||||||
startSide: [
|
startSide: [point(startMaxX, startMinY), point(startMaxX, startMaxY)],
|
||||||
[startMaxX, startMinY],
|
endSide: [point(endMinX, endMinY), point(endMinX, endMaxY)],
|
||||||
[startMaxX, startMaxY],
|
|
||||||
],
|
|
||||||
endSide: [
|
|
||||||
[endMinX, endMinY],
|
|
||||||
[endMinX, endMaxY],
|
|
||||||
],
|
|
||||||
length: endMinX - startMaxX,
|
length: endMinX - startMaxX,
|
||||||
overlap: rangeIntersection(
|
overlap: rangeIntersection(
|
||||||
[startMinY, startMaxY],
|
rangeInclusive(startMinY, startMaxY),
|
||||||
[endMinY, endMaxY],
|
rangeInclusive(endMinY, endMaxY),
|
||||||
)!,
|
)!,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -382,23 +407,20 @@ export const getVisibleGaps = (
|
||||||
|
|
||||||
if (
|
if (
|
||||||
startMaxY < endMinY &&
|
startMaxY < endMinY &&
|
||||||
rangesOverlap([startMinX, startMaxX], [endMinX, endMaxX])
|
rangesOverlap(
|
||||||
|
rangeInclusive(startMinX, startMaxX),
|
||||||
|
rangeInclusive(endMinX, endMaxX),
|
||||||
|
)
|
||||||
) {
|
) {
|
||||||
verticalGaps.push({
|
verticalGaps.push({
|
||||||
startBounds,
|
startBounds,
|
||||||
endBounds,
|
endBounds,
|
||||||
startSide: [
|
startSide: [point(startMinX, startMaxY), point(startMaxX, startMaxY)],
|
||||||
[startMinX, startMaxY],
|
endSide: [point(endMinX, endMinY), point(endMaxX, endMinY)],
|
||||||
[startMaxX, startMaxY],
|
|
||||||
],
|
|
||||||
endSide: [
|
|
||||||
[endMinX, endMinY],
|
|
||||||
[endMaxX, endMinY],
|
|
||||||
],
|
|
||||||
length: endMinY - startMaxY,
|
length: endMinY - startMaxY,
|
||||||
overlap: rangeIntersection(
|
overlap: rangeIntersection(
|
||||||
[startMinX, startMaxX],
|
rangeInclusive(startMinX, startMaxX),
|
||||||
[endMinX, endMaxX],
|
rangeInclusive(endMinX, endMaxX),
|
||||||
)!,
|
)!,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -441,7 +463,7 @@ const getGapSnaps = (
|
||||||
const centerY = (minY + maxY) / 2;
|
const centerY = (minY + maxY) / 2;
|
||||||
|
|
||||||
for (const gap of horizontalGaps) {
|
for (const gap of horizontalGaps) {
|
||||||
if (!rangesOverlap([minY, maxY], gap.overlap)) {
|
if (!rangesOverlap(rangeInclusive(minY, maxY), gap.overlap)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -510,7 +532,7 @@ const getGapSnaps = (
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (const gap of verticalGaps) {
|
for (const gap of verticalGaps) {
|
||||||
if (!rangesOverlap([minX, maxX], gap.overlap)) {
|
if (!rangesOverlap(rangeInclusive(minX, maxX), gap.overlap)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -603,7 +625,7 @@ export const getReferenceSnapPoints = (
|
||||||
|
|
||||||
const getPointSnaps = (
|
const getPointSnaps = (
|
||||||
selectedElements: ExcalidrawElement[],
|
selectedElements: ExcalidrawElement[],
|
||||||
selectionSnapPoints: Point[],
|
selectionSnapPoints: GlobalPoint[],
|
||||||
app: AppClassProperties,
|
app: AppClassProperties,
|
||||||
event: KeyboardModifiersObject,
|
event: KeyboardModifiersObject,
|
||||||
nearestSnapsX: Snaps,
|
nearestSnapsX: Snaps,
|
||||||
|
@ -779,8 +801,8 @@ const round = (x: number) => {
|
||||||
return Math.round(x * 10 ** decimalPlaces) / 10 ** decimalPlaces;
|
return Math.round(x * 10 ** decimalPlaces) / 10 ** decimalPlaces;
|
||||||
};
|
};
|
||||||
|
|
||||||
const dedupePoints = (points: Point[]): Point[] => {
|
const dedupePoints = (points: GlobalPoint[]): GlobalPoint[] => {
|
||||||
const map = new Map<string, Point>();
|
const map = new Map<string, GlobalPoint>();
|
||||||
|
|
||||||
for (const point of points) {
|
for (const point of points) {
|
||||||
const key = point.join(",");
|
const key = point.join(",");
|
||||||
|
@ -797,8 +819,8 @@ const createPointSnapLines = (
|
||||||
nearestSnapsX: Snaps,
|
nearestSnapsX: Snaps,
|
||||||
nearestSnapsY: Snaps,
|
nearestSnapsY: Snaps,
|
||||||
): PointSnapLine[] => {
|
): PointSnapLine[] => {
|
||||||
const snapsX = {} as { [key: string]: Point[] };
|
const snapsX = {} as { [key: string]: GlobalPoint[] };
|
||||||
const snapsY = {} as { [key: string]: Point[] };
|
const snapsY = {} as { [key: string]: GlobalPoint[] };
|
||||||
|
|
||||||
if (nearestSnapsX.length > 0) {
|
if (nearestSnapsX.length > 0) {
|
||||||
for (const snap of nearestSnapsX) {
|
for (const snap of nearestSnapsX) {
|
||||||
|
@ -809,8 +831,8 @@ const createPointSnapLines = (
|
||||||
snapsX[key] = [];
|
snapsX[key] = [];
|
||||||
}
|
}
|
||||||
snapsX[key].push(
|
snapsX[key].push(
|
||||||
...snap.points.map(
|
...snap.points.map((p) =>
|
||||||
(point) => [round(point[0]), round(point[1])] as Point,
|
point<GlobalPoint>(round(p[0]), round(p[1])),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -826,8 +848,8 @@ const createPointSnapLines = (
|
||||||
snapsY[key] = [];
|
snapsY[key] = [];
|
||||||
}
|
}
|
||||||
snapsY[key].push(
|
snapsY[key].push(
|
||||||
...snap.points.map(
|
...snap.points.map((p) =>
|
||||||
(point) => [round(point[0]), round(point[1])] as Point,
|
point<GlobalPoint>(round(p[0]), round(p[1])),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -840,8 +862,8 @@ const createPointSnapLines = (
|
||||||
type: "points",
|
type: "points",
|
||||||
points: dedupePoints(
|
points: dedupePoints(
|
||||||
points
|
points
|
||||||
.map((point) => {
|
.map((p) => {
|
||||||
return [Number(key), point[1]] as Point;
|
return point<GlobalPoint>(Number(key), p[1]);
|
||||||
})
|
})
|
||||||
.sort((a, b) => a[1] - b[1]),
|
.sort((a, b) => a[1] - b[1]),
|
||||||
),
|
),
|
||||||
|
@ -853,8 +875,8 @@ const createPointSnapLines = (
|
||||||
type: "points",
|
type: "points",
|
||||||
points: dedupePoints(
|
points: dedupePoints(
|
||||||
points
|
points
|
||||||
.map((point) => {
|
.map((p) => {
|
||||||
return [point[0], Number(key)] as Point;
|
return point<GlobalPoint>(p[0], Number(key));
|
||||||
})
|
})
|
||||||
.sort((a, b) => a[0] - b[0]),
|
.sort((a, b) => a[0] - b[0]),
|
||||||
),
|
),
|
||||||
|
@ -898,12 +920,12 @@ const createGapSnapLines = (
|
||||||
const [endMinX, endMinY, endMaxX, endMaxY] = gapSnap.gap.endBounds;
|
const [endMinX, endMinY, endMaxX, endMaxY] = gapSnap.gap.endBounds;
|
||||||
|
|
||||||
const verticalIntersection = rangeIntersection(
|
const verticalIntersection = rangeIntersection(
|
||||||
[minY, maxY],
|
rangeInclusive(minY, maxY),
|
||||||
gapSnap.gap.overlap,
|
gapSnap.gap.overlap,
|
||||||
);
|
);
|
||||||
|
|
||||||
const horizontalGapIntersection = rangeIntersection(
|
const horizontalGapIntersection = rangeIntersection(
|
||||||
[minX, maxX],
|
rangeInclusive(minX, maxX),
|
||||||
gapSnap.gap.overlap,
|
gapSnap.gap.overlap,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -918,16 +940,16 @@ const createGapSnapLines = (
|
||||||
type: "gap",
|
type: "gap",
|
||||||
direction: "horizontal",
|
direction: "horizontal",
|
||||||
points: [
|
points: [
|
||||||
[gapSnap.gap.startSide[0][0], gapLineY],
|
point(gapSnap.gap.startSide[0][0], gapLineY),
|
||||||
[minX, gapLineY],
|
point(minX, gapLineY),
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: "gap",
|
type: "gap",
|
||||||
direction: "horizontal",
|
direction: "horizontal",
|
||||||
points: [
|
points: [
|
||||||
[maxX, gapLineY],
|
point(maxX, gapLineY),
|
||||||
[gapSnap.gap.endSide[0][0], gapLineY],
|
point(gapSnap.gap.endSide[0][0], gapLineY),
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
@ -944,16 +966,16 @@ const createGapSnapLines = (
|
||||||
type: "gap",
|
type: "gap",
|
||||||
direction: "vertical",
|
direction: "vertical",
|
||||||
points: [
|
points: [
|
||||||
[gapLineX, gapSnap.gap.startSide[0][1]],
|
point(gapLineX, gapSnap.gap.startSide[0][1]),
|
||||||
[gapLineX, minY],
|
point(gapLineX, minY),
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: "gap",
|
type: "gap",
|
||||||
direction: "vertical",
|
direction: "vertical",
|
||||||
points: [
|
points: [
|
||||||
[gapLineX, maxY],
|
point(gapLineX, maxY),
|
||||||
[gapLineX, gapSnap.gap.endSide[0][1]],
|
point(gapLineX, gapSnap.gap.endSide[0][1]),
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
@ -969,18 +991,12 @@ const createGapSnapLines = (
|
||||||
{
|
{
|
||||||
type: "gap",
|
type: "gap",
|
||||||
direction: "horizontal",
|
direction: "horizontal",
|
||||||
points: [
|
points: [point(startMaxX, gapLineY), point(endMinX, gapLineY)],
|
||||||
[startMaxX, gapLineY],
|
|
||||||
[endMinX, gapLineY],
|
|
||||||
],
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: "gap",
|
type: "gap",
|
||||||
direction: "horizontal",
|
direction: "horizontal",
|
||||||
points: [
|
points: [point(endMaxX, gapLineY), point(minX, gapLineY)],
|
||||||
[endMaxX, gapLineY],
|
|
||||||
[minX, gapLineY],
|
|
||||||
],
|
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -995,18 +1011,12 @@ const createGapSnapLines = (
|
||||||
{
|
{
|
||||||
type: "gap",
|
type: "gap",
|
||||||
direction: "horizontal",
|
direction: "horizontal",
|
||||||
points: [
|
points: [point(maxX, gapLineY), point(startMinX, gapLineY)],
|
||||||
[maxX, gapLineY],
|
|
||||||
[startMinX, gapLineY],
|
|
||||||
],
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: "gap",
|
type: "gap",
|
||||||
direction: "horizontal",
|
direction: "horizontal",
|
||||||
points: [
|
points: [point(startMaxX, gapLineY), point(endMinX, gapLineY)],
|
||||||
[startMaxX, gapLineY],
|
|
||||||
[endMinX, gapLineY],
|
|
||||||
],
|
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1021,18 +1031,12 @@ const createGapSnapLines = (
|
||||||
{
|
{
|
||||||
type: "gap",
|
type: "gap",
|
||||||
direction: "vertical",
|
direction: "vertical",
|
||||||
points: [
|
points: [point(gapLineX, maxY), point(gapLineX, startMinY)],
|
||||||
[gapLineX, maxY],
|
|
||||||
[gapLineX, startMinY],
|
|
||||||
],
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: "gap",
|
type: "gap",
|
||||||
direction: "vertical",
|
direction: "vertical",
|
||||||
points: [
|
points: [point(gapLineX, startMaxY), point(gapLineX, endMinY)],
|
||||||
[gapLineX, startMaxY],
|
|
||||||
[gapLineX, endMinY],
|
|
||||||
],
|
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1047,18 +1051,12 @@ const createGapSnapLines = (
|
||||||
{
|
{
|
||||||
type: "gap",
|
type: "gap",
|
||||||
direction: "vertical",
|
direction: "vertical",
|
||||||
points: [
|
points: [point(gapLineX, startMaxY), point(gapLineX, endMinY)],
|
||||||
[gapLineX, startMaxY],
|
|
||||||
[gapLineX, endMinY],
|
|
||||||
],
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: "gap",
|
type: "gap",
|
||||||
direction: "vertical",
|
direction: "vertical",
|
||||||
points: [
|
points: [point(gapLineX, endMaxY), point(gapLineX, minY)],
|
||||||
[gapLineX, endMaxY],
|
|
||||||
[gapLineX, minY],
|
|
||||||
],
|
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1071,8 +1069,8 @@ const createGapSnapLines = (
|
||||||
gapSnapLines.map((gapSnapLine) => {
|
gapSnapLines.map((gapSnapLine) => {
|
||||||
return {
|
return {
|
||||||
...gapSnapLine,
|
...gapSnapLine,
|
||||||
points: gapSnapLine.points.map(
|
points: gapSnapLine.points.map((p) =>
|
||||||
(point) => [round(point[0]), round(point[1])] as Point,
|
point(round(p[0]), round(p[1])),
|
||||||
) as PointPair,
|
) as PointPair,
|
||||||
};
|
};
|
||||||
}),
|
}),
|
||||||
|
@ -1117,40 +1115,40 @@ export const snapResizingElements = (
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const selectionSnapPoints: Point[] = [];
|
const selectionSnapPoints: GlobalPoint[] = [];
|
||||||
|
|
||||||
if (transformHandle) {
|
if (transformHandle) {
|
||||||
switch (transformHandle) {
|
switch (transformHandle) {
|
||||||
case "e": {
|
case "e": {
|
||||||
selectionSnapPoints.push([maxX, minY], [maxX, maxY]);
|
selectionSnapPoints.push(point(maxX, minY), point(maxX, maxY));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case "w": {
|
case "w": {
|
||||||
selectionSnapPoints.push([minX, minY], [minX, maxY]);
|
selectionSnapPoints.push(point(minX, minY), point(minX, maxY));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case "n": {
|
case "n": {
|
||||||
selectionSnapPoints.push([minX, minY], [maxX, minY]);
|
selectionSnapPoints.push(point(minX, minY), point(maxX, minY));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case "s": {
|
case "s": {
|
||||||
selectionSnapPoints.push([minX, maxY], [maxX, maxY]);
|
selectionSnapPoints.push(point(minX, maxY), point(maxX, maxY));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case "ne": {
|
case "ne": {
|
||||||
selectionSnapPoints.push([maxX, minY]);
|
selectionSnapPoints.push(point(maxX, minY));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case "nw": {
|
case "nw": {
|
||||||
selectionSnapPoints.push([minX, minY]);
|
selectionSnapPoints.push(point(minX, minY));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case "se": {
|
case "se": {
|
||||||
selectionSnapPoints.push([maxX, maxY]);
|
selectionSnapPoints.push(point(maxX, maxY));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case "sw": {
|
case "sw": {
|
||||||
selectionSnapPoints.push([minX, maxY]);
|
selectionSnapPoints.push(point(minX, maxY));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1192,11 +1190,11 @@ export const snapResizingElements = (
|
||||||
round(bound),
|
round(bound),
|
||||||
);
|
);
|
||||||
|
|
||||||
const corners: Point[] = [
|
const corners: GlobalPoint[] = [
|
||||||
[x1, y1],
|
point(x1, y1),
|
||||||
[x1, y2],
|
point(x1, y2),
|
||||||
[x2, y1],
|
point(x2, y1),
|
||||||
[x2, y2],
|
point(x2, y2),
|
||||||
];
|
];
|
||||||
|
|
||||||
getPointSnaps(
|
getPointSnaps(
|
||||||
|
@ -1232,8 +1230,8 @@ export const snapNewElement = (
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
const selectionSnapPoints: Point[] = [
|
const selectionSnapPoints: GlobalPoint[] = [
|
||||||
[origin.x + dragOffset.x, origin.y + dragOffset.y],
|
point(origin.x + dragOffset.x, origin.y + dragOffset.y),
|
||||||
];
|
];
|
||||||
|
|
||||||
const snapDistance = getSnapDistance(app.state.zoom.value);
|
const snapDistance = getSnapDistance(app.state.zoom.value);
|
||||||
|
@ -1333,7 +1331,7 @@ export const getSnapLinesAtPointer = (
|
||||||
|
|
||||||
verticalSnapLines.push({
|
verticalSnapLines.push({
|
||||||
type: "pointer",
|
type: "pointer",
|
||||||
points: [corner, [corner[0], pointer.y]],
|
points: [corner, point(corner[0], pointer.y)],
|
||||||
direction: "vertical",
|
direction: "vertical",
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -1349,7 +1347,7 @@ export const getSnapLinesAtPointer = (
|
||||||
|
|
||||||
horizontalSnapLines.push({
|
horizontalSnapLines.push({
|
||||||
type: "pointer",
|
type: "pointer",
|
||||||
points: [corner, [pointer.x, corner[1]]],
|
points: [corner, point(pointer.x, corner[1])],
|
||||||
direction: "horizontal",
|
direction: "horizontal",
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -1386,3 +1384,18 @@ export const isActiveToolNonLinearSnappable = (
|
||||||
activeToolType === TOOL_TYPE.text
|
activeToolType === TOOL_TYPE.text
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// TODO: Rounding this point causes some shake when free drawing
|
||||||
|
export const getGridPoint = (
|
||||||
|
x: number,
|
||||||
|
y: number,
|
||||||
|
gridSize: NullableGridSize,
|
||||||
|
): [number, number] => {
|
||||||
|
if (gridSize) {
|
||||||
|
return [
|
||||||
|
Math.round(x / gridSize) * gridSize,
|
||||||
|
Math.round(y / gridSize) * gridSize,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
return [x, y];
|
||||||
|
};
|
||||||
|
|
|
@ -866,6 +866,7 @@ exports[`contextMenu element > right-clicking on a group should select whole gro
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id1": true,
|
"id1": true,
|
||||||
|
@ -1068,6 +1069,7 @@ exports[`contextMenu element > selecting 'Add to library' in context menu adds e
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -1283,6 +1285,7 @@ exports[`contextMenu element > selecting 'Bring forward' in context menu brings
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -1613,6 +1616,7 @@ exports[`contextMenu element > selecting 'Bring to front' in context menu brings
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -1943,6 +1947,7 @@ exports[`contextMenu element > selecting 'Copy styles' in context menu copies st
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -2158,6 +2163,7 @@ exports[`contextMenu element > selecting 'Delete' in context menu deletes elemen
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -2397,6 +2403,7 @@ exports[`contextMenu element > selecting 'Duplicate' in context menu duplicates
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0_copy": true,
|
"id0_copy": true,
|
||||||
},
|
},
|
||||||
|
@ -2699,6 +2706,7 @@ exports[`contextMenu element > selecting 'Group selection' in context menu group
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id1": true,
|
"id1": true,
|
||||||
|
@ -3065,6 +3073,7 @@ exports[`contextMenu element > selecting 'Paste styles' in context menu pastes s
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -3539,6 +3548,7 @@ exports[`contextMenu element > selecting 'Send backward' in context menu sends e
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id1": true,
|
"id1": true,
|
||||||
},
|
},
|
||||||
|
@ -3861,6 +3871,7 @@ exports[`contextMenu element > selecting 'Send to back' in context menu sends el
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id1": true,
|
"id1": true,
|
||||||
},
|
},
|
||||||
|
@ -4185,6 +4196,7 @@ exports[`contextMenu element > selecting 'Ungroup selection' in context menu ung
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id1": true,
|
"id1": true,
|
||||||
|
@ -5370,6 +5382,7 @@ exports[`contextMenu element > shows 'Group selection' in context menu for multi
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id1": true,
|
"id1": true,
|
||||||
|
@ -6496,6 +6509,7 @@ exports[`contextMenu element > shows 'Ungroup selection' in context menu for gro
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id1": true,
|
"id1": true,
|
||||||
|
@ -7431,6 +7445,7 @@ exports[`contextMenu element > shows context menu for canvas > [end of test] app
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -8339,6 +8354,7 @@ exports[`contextMenu element > shows context menu for element > [end of test] ap
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -9235,6 +9251,7 @@ exports[`contextMenu element > shows context menu for element > [end of test] ap
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id1": true,
|
"id1": true,
|
||||||
},
|
},
|
||||||
|
|
|
@ -239,6 +239,55 @@ exports[`<Excalidraw/> > Test UIOptions prop > Test canvasActions > should rende
|
||||||
Ctrl+Shift+E
|
Ctrl+Shift+E
|
||||||
</div>
|
</div>
|
||||||
</button>
|
</button>
|
||||||
|
<button
|
||||||
|
aria-label="Find on canvas"
|
||||||
|
class="dropdown-menu-item dropdown-menu-item-base"
|
||||||
|
data-testid="search-menu-button"
|
||||||
|
title="Find on canvas"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="dropdown-menu-item__icon"
|
||||||
|
>
|
||||||
|
<svg
|
||||||
|
aria-hidden="true"
|
||||||
|
class=""
|
||||||
|
fill="none"
|
||||||
|
focusable="false"
|
||||||
|
role="img"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
stroke-width="2"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<g
|
||||||
|
stroke-width="1.5"
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
d="M0 0h24v24H0z"
|
||||||
|
fill="none"
|
||||||
|
stroke="none"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M10 10m-7 0a7 7 0 1 0 14 0a7 7 0 1 0 -14 0"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M21 21l-6 -6"
|
||||||
|
/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="dropdown-menu-item__text"
|
||||||
|
>
|
||||||
|
Find on canvas
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="dropdown-menu-item__shortcut"
|
||||||
|
>
|
||||||
|
Ctrl+F
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
<button
|
<button
|
||||||
aria-label="Help"
|
aria-label="Help"
|
||||||
class="dropdown-menu-item dropdown-menu-item-base"
|
class="dropdown-menu-item dropdown-menu-item-base"
|
||||||
|
|
|
@ -80,6 +80,7 @@ exports[`history > multiplayer undo/redo > conflicts in arrows and their bindabl
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id166": true,
|
"id166": true,
|
||||||
},
|
},
|
||||||
|
@ -681,6 +682,7 @@ exports[`history > multiplayer undo/redo > conflicts in arrows and their bindabl
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id161": true,
|
"id161": true,
|
||||||
},
|
},
|
||||||
|
@ -1187,6 +1189,7 @@ exports[`history > multiplayer undo/redo > conflicts in arrows and their bindabl
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -1554,6 +1557,7 @@ exports[`history > multiplayer undo/redo > conflicts in arrows and their bindabl
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -1922,6 +1926,7 @@ exports[`history > multiplayer undo/redo > conflicts in arrows and their bindabl
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -2185,6 +2190,7 @@ exports[`history > multiplayer undo/redo > conflicts in arrows and their bindabl
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id180": true,
|
"id180": true,
|
||||||
},
|
},
|
||||||
|
@ -2627,6 +2633,7 @@ exports[`history > multiplayer undo/redo > conflicts in bound text elements and
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -2925,6 +2932,7 @@ exports[`history > multiplayer undo/redo > conflicts in bound text elements and
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -3208,6 +3216,7 @@ exports[`history > multiplayer undo/redo > conflicts in bound text elements and
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -3501,6 +3510,7 @@ exports[`history > multiplayer undo/redo > conflicts in bound text elements and
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -3786,6 +3796,7 @@ exports[`history > multiplayer undo/redo > conflicts in bound text elements and
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -4020,6 +4031,7 @@ exports[`history > multiplayer undo/redo > conflicts in bound text elements and
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -4278,6 +4290,7 @@ exports[`history > multiplayer undo/redo > conflicts in bound text elements and
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -4550,6 +4563,7 @@ exports[`history > multiplayer undo/redo > conflicts in bound text elements and
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -4780,6 +4794,7 @@ exports[`history > multiplayer undo/redo > conflicts in bound text elements and
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -5010,6 +5025,7 @@ exports[`history > multiplayer undo/redo > conflicts in bound text elements and
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -5238,6 +5254,7 @@ exports[`history > multiplayer undo/redo > conflicts in bound text elements and
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -5466,6 +5483,7 @@ exports[`history > multiplayer undo/redo > conflicts in frames and their childre
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -5723,6 +5741,7 @@ exports[`history > multiplayer undo/redo > should iterate through the history wh
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -6053,6 +6072,7 @@ exports[`history > multiplayer undo/redo > should iterate through the history wh
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -6477,6 +6497,7 @@ exports[`history > multiplayer undo/redo > should iterate through the history wh
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id100": true,
|
"id100": true,
|
||||||
"id101": true,
|
"id101": true,
|
||||||
|
@ -6855,6 +6876,7 @@ exports[`history > multiplayer undo/redo > should iterate through the history wh
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id105": true,
|
"id105": true,
|
||||||
"id106": true,
|
"id106": true,
|
||||||
|
@ -7170,6 +7192,7 @@ exports[`history > multiplayer undo/redo > should iterate through the history wh
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id113": true,
|
"id113": true,
|
||||||
},
|
},
|
||||||
|
@ -7467,6 +7490,7 @@ exports[`history > multiplayer undo/redo > should iterate through the history wh
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -7695,6 +7719,7 @@ exports[`history > multiplayer undo/redo > should iterate through the history wh
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -8049,6 +8074,7 @@ exports[`history > multiplayer undo/redo > should iterate through the history wh
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -8406,6 +8432,7 @@ exports[`history > multiplayer undo/redo > should not let remote changes to inte
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id127": true,
|
"id127": true,
|
||||||
"id128": true,
|
"id128": true,
|
||||||
|
@ -8806,6 +8833,7 @@ exports[`history > multiplayer undo/redo > should not let remote changes to inte
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -9092,6 +9120,7 @@ exports[`history > multiplayer undo/redo > should not let remote changes to inte
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id124": true,
|
"id124": true,
|
||||||
},
|
},
|
||||||
|
@ -9356,6 +9385,7 @@ exports[`history > multiplayer undo/redo > should not override remote changes on
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id81": true,
|
"id81": true,
|
||||||
},
|
},
|
||||||
|
@ -9619,6 +9649,7 @@ exports[`history > multiplayer undo/redo > should not override remote changes on
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id83": true,
|
"id83": true,
|
||||||
},
|
},
|
||||||
|
@ -9852,6 +9883,7 @@ exports[`history > multiplayer undo/redo > should override remotely added groups
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -10149,6 +10181,7 @@ exports[`history > multiplayer undo/redo > should override remotely added points
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id92": true,
|
"id92": true,
|
||||||
},
|
},
|
||||||
|
@ -10488,6 +10521,7 @@ exports[`history > multiplayer undo/redo > should redistribute deltas when eleme
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -10725,6 +10759,7 @@ exports[`history > multiplayer undo/redo > should redraw arrows on undo > [end o
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -11174,6 +11209,7 @@ exports[`history > multiplayer undo/redo > should update history entries after r
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id84": true,
|
"id84": true,
|
||||||
},
|
},
|
||||||
|
@ -11427,6 +11463,7 @@ exports[`history > singleplayer undo/redo > remounting undo/redo buttons should
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -11665,6 +11702,7 @@ exports[`history > singleplayer undo/redo > should clear the redo stack on eleme
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id14": true,
|
"id14": true,
|
||||||
},
|
},
|
||||||
|
@ -11905,6 +11943,7 @@ exports[`history > singleplayer undo/redo > should create entry when selecting f
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -12308,6 +12347,7 @@ exports[`history > singleplayer undo/redo > should create new history entry on s
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": -50,
|
"scrollX": -50,
|
||||||
"scrollY": -50,
|
"scrollY": -50,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -12551,6 +12591,7 @@ exports[`history > singleplayer undo/redo > should disable undo/redo buttons whe
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id78": true,
|
"id78": true,
|
||||||
},
|
},
|
||||||
|
@ -12791,6 +12832,7 @@ exports[`history > singleplayer undo/redo > should end up with no history entry
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id19": true,
|
"id19": true,
|
||||||
},
|
},
|
||||||
|
@ -13033,6 +13075,7 @@ exports[`history > singleplayer undo/redo > should iterate through the history w
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -13277,6 +13320,7 @@ exports[`history > singleplayer undo/redo > should not clear the redo stack on s
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id8": true,
|
"id8": true,
|
||||||
},
|
},
|
||||||
|
@ -13611,6 +13655,7 @@ exports[`history > singleplayer undo/redo > should not collapse when applying co
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -13779,6 +13824,7 @@ exports[`history > singleplayer undo/redo > should not end up with history entry
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id1": true,
|
"id1": true,
|
||||||
"id2": true,
|
"id2": true,
|
||||||
|
@ -14069,6 +14115,7 @@ exports[`history > singleplayer undo/redo > should not end up with history entry
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -14334,6 +14381,7 @@ exports[`history > singleplayer undo/redo > should not override appstate changes
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id11": true,
|
"id11": true,
|
||||||
},
|
},
|
||||||
|
@ -14609,6 +14657,7 @@ exports[`history > singleplayer undo/redo > should support appstate name or view
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -14768,6 +14817,7 @@ exports[`history > singleplayer undo/redo > should support bidirectional binding
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id55": true,
|
"id55": true,
|
||||||
},
|
},
|
||||||
|
@ -15463,6 +15513,7 @@ exports[`history > singleplayer undo/redo > should support bidirectional binding
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id49": true,
|
"id49": true,
|
||||||
},
|
},
|
||||||
|
@ -16082,6 +16133,7 @@ exports[`history > singleplayer undo/redo > should support bidirectional binding
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id61": true,
|
"id61": true,
|
||||||
},
|
},
|
||||||
|
@ -16699,6 +16751,7 @@ exports[`history > singleplayer undo/redo > should support bidirectional binding
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id62": true,
|
"id62": true,
|
||||||
},
|
},
|
||||||
|
@ -17412,6 +17465,7 @@ exports[`history > singleplayer undo/redo > should support bidirectional binding
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id69": true,
|
"id69": true,
|
||||||
"id71": true,
|
"id71": true,
|
||||||
|
@ -18161,6 +18215,7 @@ exports[`history > singleplayer undo/redo > should support changes in elements'
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id39": true,
|
"id39": true,
|
||||||
"id41": true,
|
"id41": true,
|
||||||
|
@ -18634,6 +18689,7 @@ exports[`history > singleplayer undo/redo > should support duplication of groups
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id35_copy_copy": true,
|
"id35_copy_copy": true,
|
||||||
"id36_copy_copy": true,
|
"id36_copy_copy": true,
|
||||||
|
@ -19155,6 +19211,7 @@ exports[`history > singleplayer undo/redo > should support element creation, del
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -19610,6 +19667,7 @@ exports[`history > singleplayer undo/redo > should support linear element creati
|
||||||
"resizingElement": null,
|
"resizingElement": null,
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id27": true,
|
"id27": true,
|
||||||
},
|
},
|
||||||
|
|
|
@ -84,6 +84,7 @@ exports[`given element A and group of elements B and given both are selected whe
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id2": true,
|
"id2": true,
|
||||||
|
@ -495,6 +496,7 @@ exports[`given element A and group of elements B and given both are selected whe
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id1": true,
|
"id1": true,
|
||||||
},
|
},
|
||||||
|
@ -893,6 +895,7 @@ exports[`regression tests > Cmd/Ctrl-click exclusively select element under poin
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id7": true,
|
"id7": true,
|
||||||
},
|
},
|
||||||
|
@ -1434,6 +1437,7 @@ exports[`regression tests > Drags selected element when hitting only bounding bo
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -1636,6 +1640,7 @@ exports[`regression tests > adjusts z order when grouping > [end of test] appSta
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id2": true,
|
"id2": true,
|
||||||
|
@ -2007,6 +2012,7 @@ exports[`regression tests > alt-drag duplicates an element > [end of test] appSt
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -2241,6 +2247,7 @@ exports[`regression tests > arrow keys > [end of test] appState 1`] = `
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -2419,6 +2426,7 @@ exports[`regression tests > can drag element that covers another element, while
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id1": true,
|
"id1": true,
|
||||||
},
|
},
|
||||||
|
@ -2733,6 +2741,7 @@ exports[`regression tests > change the properties of a shape > [end of test] app
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -2977,6 +2986,7 @@ exports[`regression tests > click on an element and drag it > [dragged] appState
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -3216,6 +3226,7 @@ exports[`regression tests > click on an element and drag it > [end of test] appS
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -3442,6 +3453,7 @@ exports[`regression tests > click to select a shape > [end of test] appState 1`]
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -3694,6 +3706,7 @@ exports[`regression tests > click-drag to select a group > [end of test] appStat
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id1": true,
|
"id1": true,
|
||||||
|
@ -3999,6 +4012,7 @@ exports[`regression tests > deleting last but one element in editing group shoul
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id1": true,
|
"id1": true,
|
||||||
},
|
},
|
||||||
|
@ -4412,6 +4426,7 @@ exports[`regression tests > deselects group of selected elements on pointer down
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -4691,6 +4706,7 @@ exports[`regression tests > deselects group of selected elements on pointer up w
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -4939,6 +4955,7 @@ exports[`regression tests > deselects selected element on pointer down when poin
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -5145,6 +5162,7 @@ exports[`regression tests > deselects selected element, on pointer up, when clic
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -5338,6 +5356,7 @@ exports[`regression tests > double click to edit a group > [end of test] appStat
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id2": true,
|
"id2": true,
|
||||||
},
|
},
|
||||||
|
@ -5719,6 +5738,7 @@ exports[`regression tests > drags selected elements from point inside common bou
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id1": true,
|
"id1": true,
|
||||||
|
@ -6002,6 +6022,7 @@ exports[`regression tests > draw every type of shape > [end of test] appState 1`
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -6809,6 +6830,7 @@ exports[`regression tests > given a group of selected elements with an element t
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id1": true,
|
"id1": true,
|
||||||
},
|
},
|
||||||
|
@ -7134,6 +7156,7 @@ exports[`regression tests > given a selected element A and a not selected elemen
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id1": true,
|
"id1": true,
|
||||||
|
@ -7406,6 +7429,7 @@ exports[`regression tests > given selected element A with lower z-index than uns
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id1": true,
|
"id1": true,
|
||||||
},
|
},
|
||||||
|
@ -7636,6 +7660,7 @@ exports[`regression tests > given selected element A with lower z-index than uns
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -7867,6 +7892,7 @@ exports[`regression tests > key 2 selects rectangle tool > [end of test] appStat
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -8043,6 +8069,7 @@ exports[`regression tests > key 3 selects diamond tool > [end of test] appState
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -8219,6 +8246,7 @@ exports[`regression tests > key 4 selects ellipse tool > [end of test] appState
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -8395,6 +8423,7 @@ exports[`regression tests > key 5 selects arrow tool > [end of test] appState 1`
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -8613,6 +8642,7 @@ exports[`regression tests > key 6 selects line tool > [end of test] appState 1`]
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -8830,6 +8860,7 @@ exports[`regression tests > key 7 selects freedraw tool > [end of test] appState
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -9020,6 +9051,7 @@ exports[`regression tests > key a selects arrow tool > [end of test] appState 1`
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -9238,6 +9270,7 @@ exports[`regression tests > key d selects diamond tool > [end of test] appState
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -9414,6 +9447,7 @@ exports[`regression tests > key l selects line tool > [end of test] appState 1`]
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -9631,6 +9665,7 @@ exports[`regression tests > key o selects ellipse tool > [end of test] appState
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -9807,6 +9842,7 @@ exports[`regression tests > key p selects freedraw tool > [end of test] appState
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -9997,6 +10033,7 @@ exports[`regression tests > key r selects rectangle tool > [end of test] appStat
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -10177,6 +10214,7 @@ exports[`regression tests > make a group and duplicate it > [end of test] appSta
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id1": true,
|
"id1": true,
|
||||||
|
@ -10685,6 +10723,7 @@ exports[`regression tests > noop interaction after undo shouldn't create history
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id1": true,
|
"id1": true,
|
||||||
},
|
},
|
||||||
|
@ -10956,6 +10995,7 @@ exports[`regression tests > pinch-to-zoom works > [end of test] appState 1`] = `
|
||||||
"scrollX": "-6.25000",
|
"scrollX": "-6.25000",
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -11080,6 +11120,7 @@ exports[`regression tests > shift click on selected element should deselect it o
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -11276,6 +11317,7 @@ exports[`regression tests > shift-click to multiselect, then drag > [end of test
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id1": true,
|
"id1": true,
|
||||||
|
@ -11584,6 +11626,7 @@ exports[`regression tests > should group elements and ungroup them > [end of tes
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id1": true,
|
"id1": true,
|
||||||
|
@ -11993,6 +12036,7 @@ exports[`regression tests > single-clicking on a subgroup of a selected group sh
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
"id1": true,
|
"id1": true,
|
||||||
|
@ -12600,6 +12644,7 @@ exports[`regression tests > spacebar + drag scrolls the canvas > [end of test] a
|
||||||
"scrollX": 60,
|
"scrollX": 60,
|
||||||
"scrollY": 60,
|
"scrollY": 60,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -12724,6 +12769,7 @@ exports[`regression tests > supports nested groups > [end of test] appState 1`]
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id1": true,
|
"id1": true,
|
||||||
},
|
},
|
||||||
|
@ -13305,6 +13351,7 @@ exports[`regression tests > switches from group of selected elements to another
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -13638,6 +13685,7 @@ exports[`regression tests > switches selected element on pointer down > [end of
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id0": true,
|
"id0": true,
|
||||||
},
|
},
|
||||||
|
@ -13897,6 +13945,7 @@ exports[`regression tests > two-finger scroll works > [end of test] appState 1`]
|
||||||
"scrollX": 20,
|
"scrollX": 20,
|
||||||
"scrollY": "-18.53553",
|
"scrollY": "-18.53553",
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -14019,6 +14068,7 @@ exports[`regression tests > undo/redo drawing an element > [end of test] appStat
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {
|
"selectedElementIds": {
|
||||||
"id1": true,
|
"id1": true,
|
||||||
},
|
},
|
||||||
|
@ -14394,6 +14444,7 @@ exports[`regression tests > updates fontSize & fontFamily appState > [end of tes
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
@ -14519,6 +14570,7 @@ exports[`regression tests > zoom hotkeys > [end of test] appState 1`] = `
|
||||||
"scrollX": 0,
|
"scrollX": 0,
|
||||||
"scrollY": 0,
|
"scrollY": 0,
|
||||||
"scrolledOutside": false,
|
"scrolledOutside": false,
|
||||||
|
"searchMatches": [],
|
||||||
"selectedElementIds": {},
|
"selectedElementIds": {},
|
||||||
"selectedElementsAreBeingDragged": false,
|
"selectedElementsAreBeingDragged": false,
|
||||||
"selectedGroupIds": {},
|
"selectedGroupIds": {},
|
||||||
|
|
|
@ -7,6 +7,7 @@ import { API } from "./helpers/api";
|
||||||
import { KEYS } from "../keys";
|
import { KEYS } from "../keys";
|
||||||
import { actionWrapTextInContainer } from "../actions/actionBoundText";
|
import { actionWrapTextInContainer } from "../actions/actionBoundText";
|
||||||
import { arrayToMap } from "../utils";
|
import { arrayToMap } from "../utils";
|
||||||
|
import { point } from "../../math";
|
||||||
|
|
||||||
const { h } = window;
|
const { h } = window;
|
||||||
|
|
||||||
|
@ -31,12 +32,7 @@ describe("element binding", () => {
|
||||||
y: 0,
|
y: 0,
|
||||||
width: 100,
|
width: 100,
|
||||||
height: 1,
|
height: 1,
|
||||||
points: [
|
points: [point(0, 0), point(0, 0), point(100, 0), point(100, 0)],
|
||||||
[0, 0],
|
|
||||||
[0, 0],
|
|
||||||
[100, 0],
|
|
||||||
[100, 0],
|
|
||||||
],
|
|
||||||
});
|
});
|
||||||
API.setElements([rect, arrow]);
|
API.setElements([rect, arrow]);
|
||||||
expect(arrow.startBinding).toBe(null);
|
expect(arrow.startBinding).toBe(null);
|
||||||
|
@ -314,10 +310,7 @@ describe("element binding", () => {
|
||||||
const arrow1 = API.createElement({
|
const arrow1 = API.createElement({
|
||||||
type: "arrow",
|
type: "arrow",
|
||||||
id: "arrow1",
|
id: "arrow1",
|
||||||
points: [
|
points: [point(0, 0), point(0, -87.45777932247563)],
|
||||||
[0, 0],
|
|
||||||
[0, -87.45777932247563],
|
|
||||||
],
|
|
||||||
startBinding: {
|
startBinding: {
|
||||||
elementId: "rectangle1",
|
elementId: "rectangle1",
|
||||||
focus: 0.2,
|
focus: 0.2,
|
||||||
|
@ -335,10 +328,7 @@ describe("element binding", () => {
|
||||||
const arrow2 = API.createElement({
|
const arrow2 = API.createElement({
|
||||||
type: "arrow",
|
type: "arrow",
|
||||||
id: "arrow2",
|
id: "arrow2",
|
||||||
points: [
|
points: [point(0, 0), point(0, -87.45777932247563)],
|
||||||
[0, 0],
|
|
||||||
[0, -87.45777932247563],
|
|
||||||
],
|
|
||||||
startBinding: {
|
startBinding: {
|
||||||
elementId: "text1",
|
elementId: "text1",
|
||||||
focus: 0.2,
|
focus: 0.2,
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
import type { Radians } from "../../../math";
|
||||||
import { DEFAULT_FONT_FAMILY } from "../../constants";
|
import { DEFAULT_FONT_FAMILY } from "../../constants";
|
||||||
import type { ExcalidrawElement } from "../../element/types";
|
import type { ExcalidrawElement } from "../../element/types";
|
||||||
|
|
||||||
|
@ -7,7 +8,7 @@ const elementBase: Omit<ExcalidrawElement, "type"> = {
|
||||||
y: 237,
|
y: 237,
|
||||||
width: 214,
|
width: 214,
|
||||||
height: 214,
|
height: 214,
|
||||||
angle: 0,
|
angle: 0 as Radians,
|
||||||
strokeColor: "#000000",
|
strokeColor: "#000000",
|
||||||
backgroundColor: "#15aabf",
|
backgroundColor: "#15aabf",
|
||||||
fillStyle: "hachure",
|
fillStyle: "hachure",
|
||||||
|
|
|
@ -23,23 +23,24 @@ import { Excalidraw } from "../index";
|
||||||
import type { NormalizedZoomValue } from "../types";
|
import type { NormalizedZoomValue } from "../types";
|
||||||
import { ROUNDNESS } from "../constants";
|
import { ROUNDNESS } from "../constants";
|
||||||
import { vi } from "vitest";
|
import { vi } from "vitest";
|
||||||
import * as blob from "../data/blob";
|
|
||||||
import { KEYS } from "../keys";
|
import { KEYS } from "../keys";
|
||||||
import { getBoundTextElementPosition } from "../element/textElement";
|
import { getBoundTextElementPosition } from "../element/textElement";
|
||||||
import { createPasteEvent } from "../clipboard";
|
import { createPasteEvent } from "../clipboard";
|
||||||
import { arrayToMap, cloneJSON } from "../utils";
|
import { arrayToMap, cloneJSON } from "../utils";
|
||||||
|
import type { LocalPoint } from "../../math";
|
||||||
|
import { point, type Radians } from "../../math";
|
||||||
|
|
||||||
const { h } = window;
|
const { h } = window;
|
||||||
const mouse = new Pointer("mouse");
|
const mouse = new Pointer("mouse");
|
||||||
// This needs to fixed in vitest mock, as when importActual used with mock
|
|
||||||
// the tests hangs - https://github.com/vitest-dev/vitest/issues/546.
|
|
||||||
// But fortunately spying and mocking the return value of spy works :p
|
|
||||||
|
|
||||||
const resizeImageFileSpy = vi.spyOn(blob, "resizeImageFile");
|
vi.mock("../data/blob", async (actual) => {
|
||||||
const generateIdFromFileSpy = vi.spyOn(blob, "generateIdFromFile");
|
const orig: Object = await actual();
|
||||||
|
return {
|
||||||
resizeImageFileSpy.mockImplementation(async (imageFile: File) => imageFile);
|
...orig,
|
||||||
generateIdFromFileSpy.mockImplementation(async () => "fileId" as FileId);
|
resizeImageFile: (imageFile: File) => imageFile,
|
||||||
|
generateIdFromFile: () => "fileId" as FileId,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
// Unmount ReactDOM from root
|
// Unmount ReactDOM from root
|
||||||
|
@ -131,7 +132,7 @@ const createLinearElementWithCurveInsideMinMaxPoints = (
|
||||||
y: -2412.5069664197654,
|
y: -2412.5069664197654,
|
||||||
width: 1750.4888916015625,
|
width: 1750.4888916015625,
|
||||||
height: 410.51605224609375,
|
height: 410.51605224609375,
|
||||||
angle: 0,
|
angle: 0 as Radians,
|
||||||
strokeColor: "#000000",
|
strokeColor: "#000000",
|
||||||
backgroundColor: "#fa5252",
|
backgroundColor: "#fa5252",
|
||||||
fillStyle: "hachure",
|
fillStyle: "hachure",
|
||||||
|
@ -145,9 +146,9 @@ const createLinearElementWithCurveInsideMinMaxPoints = (
|
||||||
link: null,
|
link: null,
|
||||||
locked: false,
|
locked: false,
|
||||||
points: [
|
points: [
|
||||||
[0, 0],
|
point<LocalPoint>(0, 0),
|
||||||
[-922.4761962890625, 300.3277587890625],
|
point<LocalPoint>(-922.4761962890625, 300.3277587890625),
|
||||||
[828.0126953125, 410.51605224609375],
|
point<LocalPoint>(828.0126953125, 410.51605224609375),
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
@ -423,8 +424,8 @@ describe("arrow", () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it("flips a rotated arrow horizontally with line inside min/max points bounds", async () => {
|
it("flips a rotated arrow horizontally with line inside min/max points bounds", async () => {
|
||||||
const originalAngle = Math.PI / 4;
|
const originalAngle = (Math.PI / 4) as Radians;
|
||||||
const expectedAngle = (7 * Math.PI) / 4;
|
const expectedAngle = ((7 * Math.PI) / 4) as Radians;
|
||||||
const line = createLinearElementWithCurveInsideMinMaxPoints("arrow");
|
const line = createLinearElementWithCurveInsideMinMaxPoints("arrow");
|
||||||
API.setElements([line]);
|
API.setElements([line]);
|
||||||
API.setAppState({
|
API.setAppState({
|
||||||
|
@ -444,8 +445,8 @@ describe("arrow", () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it("flips a rotated arrow vertically with line inside min/max points bounds", async () => {
|
it("flips a rotated arrow vertically with line inside min/max points bounds", async () => {
|
||||||
const originalAngle = Math.PI / 4;
|
const originalAngle = (Math.PI / 4) as Radians;
|
||||||
const expectedAngle = (7 * Math.PI) / 4;
|
const expectedAngle = ((7 * Math.PI) / 4) as Radians;
|
||||||
const line = createLinearElementWithCurveInsideMinMaxPoints("arrow");
|
const line = createLinearElementWithCurveInsideMinMaxPoints("arrow");
|
||||||
API.setElements([line]);
|
API.setElements([line]);
|
||||||
API.setAppState({
|
API.setAppState({
|
||||||
|
@ -477,8 +478,8 @@ describe("arrow", () => {
|
||||||
|
|
||||||
//TODO: elements with curve outside minMax points have a wrong bounding box!!!
|
//TODO: elements with curve outside minMax points have a wrong bounding box!!!
|
||||||
it.skip("flips a rotated arrow horizontally with line outside min/max points bounds", async () => {
|
it.skip("flips a rotated arrow horizontally with line outside min/max points bounds", async () => {
|
||||||
const originalAngle = Math.PI / 4;
|
const originalAngle = (Math.PI / 4) as Radians;
|
||||||
const expectedAngle = (7 * Math.PI) / 4;
|
const expectedAngle = ((7 * Math.PI) / 4) as Radians;
|
||||||
const line = createLinearElementsWithCurveOutsideMinMaxPoints("arrow");
|
const line = createLinearElementsWithCurveOutsideMinMaxPoints("arrow");
|
||||||
API.updateElement(line, { angle: originalAngle });
|
API.updateElement(line, { angle: originalAngle });
|
||||||
API.setElements([line]);
|
API.setElements([line]);
|
||||||
|
@ -501,8 +502,8 @@ describe("arrow", () => {
|
||||||
|
|
||||||
//TODO: elements with curve outside minMax points have a wrong bounding box!!!
|
//TODO: elements with curve outside minMax points have a wrong bounding box!!!
|
||||||
it.skip("flips a rotated arrow vertically with line outside min/max points bounds", async () => {
|
it.skip("flips a rotated arrow vertically with line outside min/max points bounds", async () => {
|
||||||
const originalAngle = Math.PI / 4;
|
const originalAngle = (Math.PI / 4) as Radians;
|
||||||
const expectedAngle = (7 * Math.PI) / 4;
|
const expectedAngle = ((7 * Math.PI) / 4) as Radians;
|
||||||
const line = createLinearElementsWithCurveOutsideMinMaxPoints("arrow");
|
const line = createLinearElementsWithCurveOutsideMinMaxPoints("arrow");
|
||||||
API.updateElement(line, { angle: originalAngle });
|
API.updateElement(line, { angle: originalAngle });
|
||||||
API.setElements([line]);
|
API.setElements([line]);
|
||||||
|
@ -585,8 +586,8 @@ describe("line", () => {
|
||||||
|
|
||||||
//TODO: elements with curve outside minMax points have a wrong bounding box
|
//TODO: elements with curve outside minMax points have a wrong bounding box
|
||||||
it.skip("flips a rotated line horizontally with line outside min/max points bounds", async () => {
|
it.skip("flips a rotated line horizontally with line outside min/max points bounds", async () => {
|
||||||
const originalAngle = Math.PI / 4;
|
const originalAngle = (Math.PI / 4) as Radians;
|
||||||
const expectedAngle = (7 * Math.PI) / 4;
|
const expectedAngle = ((7 * Math.PI) / 4) as Radians;
|
||||||
const line = createLinearElementsWithCurveOutsideMinMaxPoints("line");
|
const line = createLinearElementsWithCurveOutsideMinMaxPoints("line");
|
||||||
API.updateElement(line, { angle: originalAngle });
|
API.updateElement(line, { angle: originalAngle });
|
||||||
API.setElements([line]);
|
API.setElements([line]);
|
||||||
|
@ -600,8 +601,8 @@ describe("line", () => {
|
||||||
|
|
||||||
//TODO: elements with curve outside minMax points have a wrong bounding box
|
//TODO: elements with curve outside minMax points have a wrong bounding box
|
||||||
it.skip("flips a rotated line vertically with line outside min/max points bounds", async () => {
|
it.skip("flips a rotated line vertically with line outside min/max points bounds", async () => {
|
||||||
const originalAngle = Math.PI / 4;
|
const originalAngle = (Math.PI / 4) as Radians;
|
||||||
const expectedAngle = (7 * Math.PI) / 4;
|
const expectedAngle = ((7 * Math.PI) / 4) as Radians;
|
||||||
const line = createLinearElementsWithCurveOutsideMinMaxPoints("line");
|
const line = createLinearElementsWithCurveOutsideMinMaxPoints("line");
|
||||||
API.updateElement(line, { angle: originalAngle });
|
API.updateElement(line, { angle: originalAngle });
|
||||||
API.setElements([line]);
|
API.setElements([line]);
|
||||||
|
@ -619,8 +620,8 @@ describe("line", () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it("flips a rotated line horizontally with line inside min/max points bounds", async () => {
|
it("flips a rotated line horizontally with line inside min/max points bounds", async () => {
|
||||||
const originalAngle = Math.PI / 4;
|
const originalAngle = (Math.PI / 4) as Radians;
|
||||||
const expectedAngle = (7 * Math.PI) / 4;
|
const expectedAngle = ((7 * Math.PI) / 4) as Radians;
|
||||||
const line = createLinearElementWithCurveInsideMinMaxPoints("line");
|
const line = createLinearElementWithCurveInsideMinMaxPoints("line");
|
||||||
API.setElements([line]);
|
API.setElements([line]);
|
||||||
API.setAppState({
|
API.setAppState({
|
||||||
|
@ -640,8 +641,8 @@ describe("line", () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it("flips a rotated line vertically with line inside min/max points bounds", async () => {
|
it("flips a rotated line vertically with line inside min/max points bounds", async () => {
|
||||||
const originalAngle = Math.PI / 4;
|
const originalAngle = (Math.PI / 4) as Radians;
|
||||||
const expectedAngle = (7 * Math.PI) / 4;
|
const expectedAngle = ((7 * Math.PI) / 4) as Radians;
|
||||||
const line = createLinearElementWithCurveInsideMinMaxPoints("line");
|
const line = createLinearElementWithCurveInsideMinMaxPoints("line");
|
||||||
API.setElements([line]);
|
API.setElements([line]);
|
||||||
API.setAppState({
|
API.setAppState({
|
||||||
|
@ -772,8 +773,8 @@ describe("image", () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it("flips an rotated image horizontally correctly", async () => {
|
it("flips an rotated image horizontally correctly", async () => {
|
||||||
const originalAngle = Math.PI / 4;
|
const originalAngle = (Math.PI / 4) as Radians;
|
||||||
const expectedAngle = (7 * Math.PI) / 4;
|
const expectedAngle = ((7 * Math.PI) / 4) as Radians;
|
||||||
//paste image
|
//paste image
|
||||||
await createImage();
|
await createImage();
|
||||||
await waitFor(() => {
|
await waitFor(() => {
|
||||||
|
@ -790,8 +791,8 @@ describe("image", () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it("flips an rotated image vertically correctly", async () => {
|
it("flips an rotated image vertically correctly", async () => {
|
||||||
const originalAngle = Math.PI / 4;
|
const originalAngle = (Math.PI / 4) as Radians;
|
||||||
const expectedAngle = (7 * Math.PI) / 4;
|
const expectedAngle = ((7 * Math.PI) / 4) as Radians;
|
||||||
//paste image
|
//paste image
|
||||||
await createImage();
|
await createImage();
|
||||||
await waitFor(() => {
|
await waitFor(() => {
|
||||||
|
|
|
@ -27,7 +27,7 @@ import {
|
||||||
newImageElement,
|
newImageElement,
|
||||||
newMagicFrameElement,
|
newMagicFrameElement,
|
||||||
} from "../../element/newElement";
|
} from "../../element/newElement";
|
||||||
import type { AppState, Point } from "../../types";
|
import type { AppState } from "../../types";
|
||||||
import { getSelectedElements } from "../../scene/selection";
|
import { getSelectedElements } from "../../scene/selection";
|
||||||
import { isLinearElementType } from "../../element/typeChecks";
|
import { isLinearElementType } from "../../element/typeChecks";
|
||||||
import type { Mutable } from "../../utility-types";
|
import type { Mutable } from "../../utility-types";
|
||||||
|
@ -36,6 +36,7 @@ import type App from "../../components/App";
|
||||||
import { createTestHook } from "../../components/App";
|
import { createTestHook } from "../../components/App";
|
||||||
import type { Action } from "../../actions/types";
|
import type { Action } from "../../actions/types";
|
||||||
import { mutateElement } from "../../element/mutateElement";
|
import { mutateElement } from "../../element/mutateElement";
|
||||||
|
import { point, type LocalPoint, type Radians } from "../../../math";
|
||||||
|
|
||||||
const readFile = util.promisify(fs.readFile);
|
const readFile = util.promisify(fs.readFile);
|
||||||
// so that window.h is available when App.tsx is not imported as well.
|
// so that window.h is available when App.tsx is not imported as well.
|
||||||
|
@ -75,11 +76,12 @@ export class API {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
static updateElement = (
|
// eslint-disable-next-line prettier/prettier
|
||||||
...[element, updates]: Parameters<typeof mutateElement>
|
static updateElement = <T extends ExcalidrawElement>(
|
||||||
|
...args: Parameters<typeof mutateElement<T>>
|
||||||
) => {
|
) => {
|
||||||
act(() => {
|
act(() => {
|
||||||
mutateElement(element, updates);
|
mutateElement<T>(...args);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -171,7 +173,7 @@ export class API {
|
||||||
containerId?: T extends "text"
|
containerId?: T extends "text"
|
||||||
? ExcalidrawTextElement["containerId"]
|
? ExcalidrawTextElement["containerId"]
|
||||||
: never;
|
: never;
|
||||||
points?: T extends "arrow" | "line" ? readonly Point[] : never;
|
points?: T extends "arrow" | "line" ? readonly LocalPoint[] : never;
|
||||||
locked?: boolean;
|
locked?: boolean;
|
||||||
fileId?: T extends "image" ? string : never;
|
fileId?: T extends "image" ? string : never;
|
||||||
scale?: T extends "image" ? ExcalidrawImageElement["scale"] : never;
|
scale?: T extends "image" ? ExcalidrawImageElement["scale"] : never;
|
||||||
|
@ -218,7 +220,7 @@ export class API {
|
||||||
y,
|
y,
|
||||||
frameId: rest.frameId ?? null,
|
frameId: rest.frameId ?? null,
|
||||||
index: rest.index ?? null,
|
index: rest.index ?? null,
|
||||||
angle: rest.angle ?? 0,
|
angle: (rest.angle ?? 0) as Radians,
|
||||||
strokeColor: rest.strokeColor ?? appState.currentItemStrokeColor,
|
strokeColor: rest.strokeColor ?? appState.currentItemStrokeColor,
|
||||||
backgroundColor:
|
backgroundColor:
|
||||||
rest.backgroundColor ?? appState.currentItemBackgroundColor,
|
rest.backgroundColor ?? appState.currentItemBackgroundColor,
|
||||||
|
@ -293,8 +295,8 @@ export class API {
|
||||||
height,
|
height,
|
||||||
type,
|
type,
|
||||||
points: rest.points ?? [
|
points: rest.points ?? [
|
||||||
[0, 0],
|
point<LocalPoint>(0, 0),
|
||||||
[100, 100],
|
point<LocalPoint>(100, 100),
|
||||||
],
|
],
|
||||||
elbowed: rest.elbowed ?? false,
|
elbowed: rest.elbowed ?? false,
|
||||||
});
|
});
|
||||||
|
@ -306,8 +308,8 @@ export class API {
|
||||||
height,
|
height,
|
||||||
type,
|
type,
|
||||||
points: rest.points ?? [
|
points: rest.points ?? [
|
||||||
[0, 0],
|
point<LocalPoint>(0, 0),
|
||||||
[100, 100],
|
point<LocalPoint>(100, 100),
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import type { Point, ToolType } from "../../types";
|
import type { ToolType } from "../../types";
|
||||||
import type {
|
import type {
|
||||||
ExcalidrawElement,
|
ExcalidrawElement,
|
||||||
ExcalidrawLinearElement,
|
ExcalidrawLinearElement,
|
||||||
|
@ -30,10 +30,11 @@ import {
|
||||||
isFrameLikeElement,
|
isFrameLikeElement,
|
||||||
} from "../../element/typeChecks";
|
} from "../../element/typeChecks";
|
||||||
import { getCommonBounds, getElementPointsCoords } from "../../element/bounds";
|
import { getCommonBounds, getElementPointsCoords } from "../../element/bounds";
|
||||||
import { rotatePoint } from "../../math";
|
|
||||||
import { getTextEditor } from "../queries/dom";
|
import { getTextEditor } from "../queries/dom";
|
||||||
import { arrayToMap } from "../../utils";
|
import { arrayToMap } from "../../utils";
|
||||||
import { createTestHook } from "../../components/App";
|
import { createTestHook } from "../../components/App";
|
||||||
|
import type { GlobalPoint, LocalPoint, Radians } from "../../../math";
|
||||||
|
import { point, pointRotateRads } from "../../../math";
|
||||||
|
|
||||||
// so that window.h is available when App.tsx is not imported as well.
|
// so that window.h is available when App.tsx is not imported as well.
|
||||||
createTestHook();
|
createTestHook();
|
||||||
|
@ -68,8 +69,11 @@ export class Keyboard {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
static keyDown = (key: string) => {
|
static keyDown = (
|
||||||
fireEvent.keyDown(document, {
|
key: string,
|
||||||
|
target: HTMLElement | Document | Window = document,
|
||||||
|
) => {
|
||||||
|
fireEvent.keyDown(target, {
|
||||||
key,
|
key,
|
||||||
ctrlKey,
|
ctrlKey,
|
||||||
shiftKey,
|
shiftKey,
|
||||||
|
@ -77,8 +81,11 @@ export class Keyboard {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
static keyUp = (key: string) => {
|
static keyUp = (
|
||||||
fireEvent.keyUp(document, {
|
key: string,
|
||||||
|
target: HTMLElement | Document | Window = document,
|
||||||
|
) => {
|
||||||
|
fireEvent.keyUp(target, {
|
||||||
key,
|
key,
|
||||||
ctrlKey,
|
ctrlKey,
|
||||||
shiftKey,
|
shiftKey,
|
||||||
|
@ -86,9 +93,9 @@ export class Keyboard {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
static keyPress = (key: string) => {
|
static keyPress = (key: string, target?: HTMLElement | Document | Window) => {
|
||||||
Keyboard.keyDown(key);
|
Keyboard.keyDown(key, target);
|
||||||
Keyboard.keyUp(key);
|
Keyboard.keyUp(key, target);
|
||||||
};
|
};
|
||||||
|
|
||||||
static codeDown = (code: string) => {
|
static codeDown = (code: string) => {
|
||||||
|
@ -131,27 +138,29 @@ export class Keyboard {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
const getElementPointForSelection = (element: ExcalidrawElement): Point => {
|
const getElementPointForSelection = (
|
||||||
|
element: ExcalidrawElement,
|
||||||
|
): GlobalPoint => {
|
||||||
const { x, y, width, height, angle } = element;
|
const { x, y, width, height, angle } = element;
|
||||||
const target: Point = [
|
const target = point<GlobalPoint>(
|
||||||
x +
|
x +
|
||||||
(isLinearElement(element) || isFreeDrawElement(element) ? 0 : width / 2),
|
(isLinearElement(element) || isFreeDrawElement(element) ? 0 : width / 2),
|
||||||
y,
|
y,
|
||||||
];
|
);
|
||||||
let center: Point;
|
let center: GlobalPoint;
|
||||||
|
|
||||||
if (isLinearElement(element)) {
|
if (isLinearElement(element)) {
|
||||||
const bounds = getElementPointsCoords(element, element.points);
|
const bounds = getElementPointsCoords(element, element.points);
|
||||||
center = [(bounds[0] + bounds[2]) / 2, (bounds[1] + bounds[3]) / 2];
|
center = point((bounds[0] + bounds[2]) / 2, (bounds[1] + bounds[3]) / 2);
|
||||||
} else {
|
} else {
|
||||||
center = [x + width / 2, y + height / 2];
|
center = point(x + width / 2, y + height / 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isTextElement(element)) {
|
if (isTextElement(element)) {
|
||||||
return center;
|
return center;
|
||||||
}
|
}
|
||||||
|
|
||||||
return rotatePoint(target, center, angle);
|
return pointRotateRads(target, center, angle);
|
||||||
};
|
};
|
||||||
|
|
||||||
export class Pointer {
|
export class Pointer {
|
||||||
|
@ -328,7 +337,7 @@ const transform = (
|
||||||
const isFrameSelected = elements.some(isFrameLikeElement);
|
const isFrameSelected = elements.some(isFrameLikeElement);
|
||||||
const transformHandles = getTransformHandlesFromCoords(
|
const transformHandles = getTransformHandlesFromCoords(
|
||||||
[x1, y1, x2, y2, (x1 + x2) / 2, (y1 + y2) / 2],
|
[x1, y1, x2, y2, (x1 + x2) / 2, (y1 + y2) / 2],
|
||||||
0,
|
0 as Radians,
|
||||||
h.state.zoom,
|
h.state.zoom,
|
||||||
"mouse",
|
"mouse",
|
||||||
isFrameSelected ? OMIT_SIDES_FOR_FRAME : OMIT_SIDES_FOR_MULTIPLE_ELEMENTS,
|
isFrameSelected ? OMIT_SIDES_FOR_FRAME : OMIT_SIDES_FOR_MULTIPLE_ELEMENTS,
|
||||||
|
@ -450,7 +459,7 @@ export class UI {
|
||||||
width?: number;
|
width?: number;
|
||||||
height?: number;
|
height?: number;
|
||||||
angle?: number;
|
angle?: number;
|
||||||
points?: T extends "line" | "arrow" | "freedraw" ? Point[] : never;
|
points?: T extends "line" | "arrow" | "freedraw" ? LocalPoint[] : never;
|
||||||
} = {},
|
} = {},
|
||||||
): Element<T> & {
|
): Element<T> & {
|
||||||
/** Returns the actual, current element from the elements array, instead
|
/** Returns the actual, current element from the elements array, instead
|
||||||
|
@ -459,9 +468,9 @@ export class UI {
|
||||||
} {
|
} {
|
||||||
const width = initialWidth ?? initialHeight ?? size;
|
const width = initialWidth ?? initialHeight ?? size;
|
||||||
const height = initialHeight ?? size;
|
const height = initialHeight ?? size;
|
||||||
const points: Point[] = initialPoints ?? [
|
const points: LocalPoint[] = initialPoints ?? [
|
||||||
[0, 0],
|
point(0, 0),
|
||||||
[width, height],
|
point(width, height),
|
||||||
];
|
];
|
||||||
|
|
||||||
UI.clickTool(type);
|
UI.clickTool(type);
|
||||||
|
|
|
@ -44,6 +44,8 @@ import { queryByText } from "@testing-library/react";
|
||||||
import { HistoryEntry } from "../history";
|
import { HistoryEntry } from "../history";
|
||||||
import { AppStateChange, ElementsChange } from "../change";
|
import { AppStateChange, ElementsChange } from "../change";
|
||||||
import { Snapshot, StoreAction } from "../store";
|
import { Snapshot, StoreAction } from "../store";
|
||||||
|
import type { LocalPoint, Radians } from "../../math";
|
||||||
|
import { point } from "../../math";
|
||||||
|
|
||||||
const { h } = window;
|
const { h } = window;
|
||||||
|
|
||||||
|
@ -2038,9 +2040,9 @@ describe("history", () => {
|
||||||
width: 178.9000000000001,
|
width: 178.9000000000001,
|
||||||
height: 236.10000000000002,
|
height: 236.10000000000002,
|
||||||
points: [
|
points: [
|
||||||
[0, 0],
|
point(0, 0),
|
||||||
[178.9000000000001, 0],
|
point(178.9000000000001, 0),
|
||||||
[178.9000000000001, 236.10000000000002],
|
point(178.9000000000001, 236.10000000000002),
|
||||||
],
|
],
|
||||||
startBinding: {
|
startBinding: {
|
||||||
elementId: "KPrBI4g_v9qUB1XxYLgSz",
|
elementId: "KPrBI4g_v9qUB1XxYLgSz",
|
||||||
|
@ -2156,12 +2158,12 @@ describe("history", () => {
|
||||||
elements: [
|
elements: [
|
||||||
newElementWith(h.elements[0] as ExcalidrawLinearElement, {
|
newElementWith(h.elements[0] as ExcalidrawLinearElement, {
|
||||||
points: [
|
points: [
|
||||||
[0, 0],
|
point(0, 0),
|
||||||
[5, 5],
|
point(5, 5),
|
||||||
[10, 10],
|
point(10, 10),
|
||||||
[15, 15],
|
point(15, 15),
|
||||||
[20, 20],
|
point(20, 20),
|
||||||
],
|
] as LocalPoint[],
|
||||||
}),
|
}),
|
||||||
],
|
],
|
||||||
storeAction: StoreAction.UPDATE,
|
storeAction: StoreAction.UPDATE,
|
||||||
|
@ -4003,7 +4005,7 @@ describe("history", () => {
|
||||||
newElementWith(h.elements[0], {
|
newElementWith(h.elements[0], {
|
||||||
x: 200,
|
x: 200,
|
||||||
y: 200,
|
y: 200,
|
||||||
angle: 90,
|
angle: 90 as Radians,
|
||||||
}),
|
}),
|
||||||
],
|
],
|
||||||
storeAction: StoreAction.CAPTURE,
|
storeAction: StoreAction.CAPTURE,
|
||||||
|
@ -4121,7 +4123,7 @@ describe("history", () => {
|
||||||
newElementWith(h.elements[0], {
|
newElementWith(h.elements[0], {
|
||||||
x: 205,
|
x: 205,
|
||||||
y: 205,
|
y: 205,
|
||||||
angle: 90,
|
angle: 90 as Radians,
|
||||||
}),
|
}),
|
||||||
],
|
],
|
||||||
storeAction: StoreAction.CAPTURE,
|
storeAction: StoreAction.CAPTURE,
|
||||||
|
|
|
@ -8,7 +8,6 @@ import type {
|
||||||
SceneElementsMap,
|
SceneElementsMap,
|
||||||
} from "../element/types";
|
} from "../element/types";
|
||||||
import { Excalidraw, mutateElement } from "../index";
|
import { Excalidraw, mutateElement } from "../index";
|
||||||
import { centerPoint } from "../math";
|
|
||||||
import { reseed } from "../random";
|
import { reseed } from "../random";
|
||||||
import * as StaticScene from "../renderer/staticScene";
|
import * as StaticScene from "../renderer/staticScene";
|
||||||
import * as InteractiveCanvas from "../renderer/interactiveScene";
|
import * as InteractiveCanvas from "../renderer/interactiveScene";
|
||||||
|
@ -16,7 +15,6 @@ import * as InteractiveCanvas from "../renderer/interactiveScene";
|
||||||
import { Keyboard, Pointer, UI } from "./helpers/ui";
|
import { Keyboard, Pointer, UI } from "./helpers/ui";
|
||||||
import { screen, render, fireEvent, GlobalTestState } from "./test-utils";
|
import { screen, render, fireEvent, GlobalTestState } from "./test-utils";
|
||||||
import { API } from "../tests/helpers/api";
|
import { API } from "../tests/helpers/api";
|
||||||
import type { Point } from "../types";
|
|
||||||
import { KEYS } from "../keys";
|
import { KEYS } from "../keys";
|
||||||
import { LinearElementEditor } from "../element/linearElementEditor";
|
import { LinearElementEditor } from "../element/linearElementEditor";
|
||||||
import { act, queryByTestId, queryByText } from "@testing-library/react";
|
import { act, queryByTestId, queryByText } from "@testing-library/react";
|
||||||
|
@ -29,6 +27,8 @@ import * as textElementUtils from "../element/textElement";
|
||||||
import { ROUNDNESS, VERTICAL_ALIGN } from "../constants";
|
import { ROUNDNESS, VERTICAL_ALIGN } from "../constants";
|
||||||
import { vi } from "vitest";
|
import { vi } from "vitest";
|
||||||
import { arrayToMap } from "../utils";
|
import { arrayToMap } from "../utils";
|
||||||
|
import type { GlobalPoint } from "../../math";
|
||||||
|
import { pointCenter, point } from "../../math";
|
||||||
|
|
||||||
const renderInteractiveScene = vi.spyOn(
|
const renderInteractiveScene = vi.spyOn(
|
||||||
InteractiveCanvas,
|
InteractiveCanvas,
|
||||||
|
@ -57,9 +57,9 @@ describe("Test Linear Elements", () => {
|
||||||
interactiveCanvas = container.querySelector("canvas.interactive")!;
|
interactiveCanvas = container.querySelector("canvas.interactive")!;
|
||||||
});
|
});
|
||||||
|
|
||||||
const p1: Point = [20, 20];
|
const p1 = point<GlobalPoint>(20, 20);
|
||||||
const p2: Point = [60, 20];
|
const p2 = point<GlobalPoint>(60, 20);
|
||||||
const midpoint = centerPoint(p1, p2);
|
const midpoint = pointCenter<GlobalPoint>(p1, p2);
|
||||||
const delta = 50;
|
const delta = 50;
|
||||||
const mouse = new Pointer("mouse");
|
const mouse = new Pointer("mouse");
|
||||||
|
|
||||||
|
@ -75,10 +75,7 @@ describe("Test Linear Elements", () => {
|
||||||
height: 0,
|
height: 0,
|
||||||
type,
|
type,
|
||||||
roughness,
|
roughness,
|
||||||
points: [
|
points: [point(0, 0), point(p2[0] - p1[0], p2[1] - p1[1])],
|
||||||
[0, 0],
|
|
||||||
[p2[0] - p1[0], p2[1] - p1[1]],
|
|
||||||
],
|
|
||||||
roundness,
|
roundness,
|
||||||
});
|
});
|
||||||
API.setElements([line]);
|
API.setElements([line]);
|
||||||
|
@ -102,9 +99,9 @@ describe("Test Linear Elements", () => {
|
||||||
type,
|
type,
|
||||||
roughness,
|
roughness,
|
||||||
points: [
|
points: [
|
||||||
[0, 0],
|
point(0, 0),
|
||||||
[p3[0], p3[1]],
|
point(p3[0], p3[1]),
|
||||||
[p2[0] - p1[0], p2[1] - p1[1]],
|
point(p2[0] - p1[0], p2[1] - p1[1]),
|
||||||
],
|
],
|
||||||
roundness,
|
roundness,
|
||||||
});
|
});
|
||||||
|
@ -129,7 +126,7 @@ describe("Test Linear Elements", () => {
|
||||||
expect(h.state.editingLinearElement?.elementId).toEqual(line.id);
|
expect(h.state.editingLinearElement?.elementId).toEqual(line.id);
|
||||||
};
|
};
|
||||||
|
|
||||||
const drag = (startPoint: Point, endPoint: Point) => {
|
const drag = (startPoint: GlobalPoint, endPoint: GlobalPoint) => {
|
||||||
fireEvent.pointerDown(interactiveCanvas, {
|
fireEvent.pointerDown(interactiveCanvas, {
|
||||||
clientX: startPoint[0],
|
clientX: startPoint[0],
|
||||||
clientY: startPoint[1],
|
clientY: startPoint[1],
|
||||||
|
@ -144,7 +141,7 @@ describe("Test Linear Elements", () => {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const deletePoint = (point: Point) => {
|
const deletePoint = (point: GlobalPoint) => {
|
||||||
fireEvent.pointerDown(interactiveCanvas, {
|
fireEvent.pointerDown(interactiveCanvas, {
|
||||||
clientX: point[0],
|
clientX: point[0],
|
||||||
clientY: point[1],
|
clientY: point[1],
|
||||||
|
@ -164,7 +161,7 @@ describe("Test Linear Elements", () => {
|
||||||
expect(line.points.length).toEqual(2);
|
expect(line.points.length).toEqual(2);
|
||||||
|
|
||||||
mouse.clickAt(midpoint[0], midpoint[1]);
|
mouse.clickAt(midpoint[0], midpoint[1]);
|
||||||
drag(midpoint, [midpoint[0] + 1, midpoint[1] + 1]);
|
drag(midpoint, point(midpoint[0] + 1, midpoint[1] + 1));
|
||||||
|
|
||||||
expect(line.points.length).toEqual(2);
|
expect(line.points.length).toEqual(2);
|
||||||
|
|
||||||
|
@ -172,7 +169,7 @@ describe("Test Linear Elements", () => {
|
||||||
expect(line.y).toBe(originalY);
|
expect(line.y).toBe(originalY);
|
||||||
expect(line.points.length).toEqual(2);
|
expect(line.points.length).toEqual(2);
|
||||||
|
|
||||||
drag(midpoint, [midpoint[0] + delta, midpoint[1] + delta]);
|
drag(midpoint, point(midpoint[0] + delta, midpoint[1] + delta));
|
||||||
expect(line.x).toBe(originalX);
|
expect(line.x).toBe(originalX);
|
||||||
expect(line.y).toBe(originalY);
|
expect(line.y).toBe(originalY);
|
||||||
expect(line.points.length).toEqual(3);
|
expect(line.points.length).toEqual(3);
|
||||||
|
@ -187,7 +184,7 @@ describe("Test Linear Elements", () => {
|
||||||
expect((h.elements[0] as ExcalidrawLinearElement).points.length).toEqual(2);
|
expect((h.elements[0] as ExcalidrawLinearElement).points.length).toEqual(2);
|
||||||
|
|
||||||
// drag line from midpoint
|
// drag line from midpoint
|
||||||
drag(midpoint, [midpoint[0] + delta, midpoint[1] + delta]);
|
drag(midpoint, point(midpoint[0] + delta, midpoint[1] + delta));
|
||||||
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(`9`);
|
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(`9`);
|
||||||
expect(renderStaticScene.mock.calls.length).toMatchInlineSnapshot(`7`);
|
expect(renderStaticScene.mock.calls.length).toMatchInlineSnapshot(`7`);
|
||||||
expect(line.points.length).toEqual(3);
|
expect(line.points.length).toEqual(3);
|
||||||
|
@ -251,7 +248,7 @@ describe("Test Linear Elements", () => {
|
||||||
mouse.clickAt(midpoint[0], midpoint[1]);
|
mouse.clickAt(midpoint[0], midpoint[1]);
|
||||||
expect(line.points.length).toEqual(2);
|
expect(line.points.length).toEqual(2);
|
||||||
|
|
||||||
drag(midpoint, [midpoint[0] + 1, midpoint[1] + 1]);
|
drag(midpoint, point(midpoint[0] + 1, midpoint[1] + 1));
|
||||||
expect(line.x).toBe(originalX);
|
expect(line.x).toBe(originalX);
|
||||||
expect(line.y).toBe(originalY);
|
expect(line.y).toBe(originalY);
|
||||||
expect(line.points.length).toEqual(3);
|
expect(line.points.length).toEqual(3);
|
||||||
|
@ -264,7 +261,7 @@ describe("Test Linear Elements", () => {
|
||||||
enterLineEditingMode(line);
|
enterLineEditingMode(line);
|
||||||
|
|
||||||
// drag line from midpoint
|
// drag line from midpoint
|
||||||
drag(midpoint, [midpoint[0] + delta, midpoint[1] + delta]);
|
drag(midpoint, point(midpoint[0] + delta, midpoint[1] + delta));
|
||||||
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(
|
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(
|
||||||
`12`,
|
`12`,
|
||||||
);
|
);
|
||||||
|
@ -356,10 +353,13 @@ describe("Test Linear Elements", () => {
|
||||||
h.state,
|
h.state,
|
||||||
);
|
);
|
||||||
|
|
||||||
const startPoint = centerPoint(points[0], midPoints[0] as Point);
|
const startPoint = pointCenter(points[0], midPoints[0]!);
|
||||||
const deltaX = 50;
|
const deltaX = 50;
|
||||||
const deltaY = 20;
|
const deltaY = 20;
|
||||||
const endPoint: Point = [startPoint[0] + deltaX, startPoint[1] + deltaY];
|
const endPoint = point<GlobalPoint>(
|
||||||
|
startPoint[0] + deltaX,
|
||||||
|
startPoint[1] + deltaY,
|
||||||
|
);
|
||||||
|
|
||||||
// Move the element
|
// Move the element
|
||||||
drag(startPoint, endPoint);
|
drag(startPoint, endPoint);
|
||||||
|
@ -399,8 +399,8 @@ describe("Test Linear Elements", () => {
|
||||||
// This is the expected midpoint for line with round edge
|
// This is the expected midpoint for line with round edge
|
||||||
// hence hardcoding it so if later some bug is introduced
|
// hence hardcoding it so if later some bug is introduced
|
||||||
// this will fail and we can fix it
|
// this will fail and we can fix it
|
||||||
const firstSegmentMidpoint: Point = [55, 45];
|
const firstSegmentMidpoint = point<GlobalPoint>(55, 45);
|
||||||
const lastSegmentMidpoint: Point = [75, 40];
|
const lastSegmentMidpoint = point<GlobalPoint>(75, 40);
|
||||||
|
|
||||||
let line: ExcalidrawLinearElement;
|
let line: ExcalidrawLinearElement;
|
||||||
|
|
||||||
|
@ -414,17 +414,20 @@ describe("Test Linear Elements", () => {
|
||||||
|
|
||||||
it("should allow dragging lines from midpoints in between segments", async () => {
|
it("should allow dragging lines from midpoints in between segments", async () => {
|
||||||
// drag line via first segment midpoint
|
// drag line via first segment midpoint
|
||||||
drag(firstSegmentMidpoint, [
|
drag(
|
||||||
|
firstSegmentMidpoint,
|
||||||
|
point(
|
||||||
firstSegmentMidpoint[0] + delta,
|
firstSegmentMidpoint[0] + delta,
|
||||||
firstSegmentMidpoint[1] + delta,
|
firstSegmentMidpoint[1] + delta,
|
||||||
]);
|
),
|
||||||
|
);
|
||||||
expect(line.points.length).toEqual(4);
|
expect(line.points.length).toEqual(4);
|
||||||
|
|
||||||
// drag line from last segment midpoint
|
// drag line from last segment midpoint
|
||||||
drag(lastSegmentMidpoint, [
|
drag(
|
||||||
lastSegmentMidpoint[0] + delta,
|
lastSegmentMidpoint,
|
||||||
lastSegmentMidpoint[1] + delta,
|
point(lastSegmentMidpoint[0] + delta, lastSegmentMidpoint[1] + delta),
|
||||||
]);
|
);
|
||||||
|
|
||||||
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(
|
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(
|
||||||
`16`,
|
`16`,
|
||||||
|
@ -472,10 +475,10 @@ describe("Test Linear Elements", () => {
|
||||||
h.state,
|
h.state,
|
||||||
);
|
);
|
||||||
|
|
||||||
const hitCoords: Point = [points[0][0], points[0][1]];
|
const hitCoords = point<GlobalPoint>(points[0][0], points[0][1]);
|
||||||
|
|
||||||
// Drag from first point
|
// Drag from first point
|
||||||
drag(hitCoords, [hitCoords[0] - delta, hitCoords[1] - delta]);
|
drag(hitCoords, point(hitCoords[0] - delta, hitCoords[1] - delta));
|
||||||
|
|
||||||
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(
|
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(
|
||||||
`12`,
|
`12`,
|
||||||
|
@ -513,10 +516,10 @@ describe("Test Linear Elements", () => {
|
||||||
h.state,
|
h.state,
|
||||||
);
|
);
|
||||||
|
|
||||||
const hitCoords: Point = [points[0][0], points[0][1]];
|
const hitCoords = point<GlobalPoint>(points[0][0], points[0][1]);
|
||||||
|
|
||||||
// Drag from first point
|
// Drag from first point
|
||||||
drag(hitCoords, [hitCoords[0] + delta, hitCoords[1] + delta]);
|
drag(hitCoords, point(hitCoords[0] + delta, hitCoords[1] + delta));
|
||||||
|
|
||||||
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(
|
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(
|
||||||
`12`,
|
`12`,
|
||||||
|
@ -551,10 +554,10 @@ describe("Test Linear Elements", () => {
|
||||||
);
|
);
|
||||||
|
|
||||||
// dragging line from last segment midpoint
|
// dragging line from last segment midpoint
|
||||||
drag(lastSegmentMidpoint, [
|
drag(
|
||||||
lastSegmentMidpoint[0] + 50,
|
lastSegmentMidpoint,
|
||||||
lastSegmentMidpoint[1] + 50,
|
point(lastSegmentMidpoint[0] + 50, lastSegmentMidpoint[1] + 50),
|
||||||
]);
|
);
|
||||||
expect(line.points.length).toEqual(4);
|
expect(line.points.length).toEqual(4);
|
||||||
|
|
||||||
const midPoints = LinearElementEditor.getEditorMidPoints(
|
const midPoints = LinearElementEditor.getEditorMidPoints(
|
||||||
|
@ -586,12 +589,14 @@ describe("Test Linear Elements", () => {
|
||||||
// This is the expected midpoint for line with round edge
|
// This is the expected midpoint for line with round edge
|
||||||
// hence hardcoding it so if later some bug is introduced
|
// hence hardcoding it so if later some bug is introduced
|
||||||
// this will fail and we can fix it
|
// this will fail and we can fix it
|
||||||
const firstSegmentMidpoint: Point = [
|
const firstSegmentMidpoint = point<GlobalPoint>(
|
||||||
55.9697848965255, 47.442326230998205,
|
55.9697848965255,
|
||||||
];
|
47.442326230998205,
|
||||||
const lastSegmentMidpoint: Point = [
|
);
|
||||||
76.08587175006699, 43.294165939653226,
|
const lastSegmentMidpoint = point<GlobalPoint>(
|
||||||
];
|
76.08587175006699,
|
||||||
|
43.294165939653226,
|
||||||
|
);
|
||||||
let line: ExcalidrawLinearElement;
|
let line: ExcalidrawLinearElement;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
@ -605,17 +610,20 @@ describe("Test Linear Elements", () => {
|
||||||
|
|
||||||
it("should allow dragging lines from midpoints in between segments", async () => {
|
it("should allow dragging lines from midpoints in between segments", async () => {
|
||||||
// drag line from first segment midpoint
|
// drag line from first segment midpoint
|
||||||
drag(firstSegmentMidpoint, [
|
drag(
|
||||||
|
firstSegmentMidpoint,
|
||||||
|
point(
|
||||||
firstSegmentMidpoint[0] + delta,
|
firstSegmentMidpoint[0] + delta,
|
||||||
firstSegmentMidpoint[1] + delta,
|
firstSegmentMidpoint[1] + delta,
|
||||||
]);
|
),
|
||||||
|
);
|
||||||
expect(line.points.length).toEqual(4);
|
expect(line.points.length).toEqual(4);
|
||||||
|
|
||||||
// drag line from last segment midpoint
|
// drag line from last segment midpoint
|
||||||
drag(lastSegmentMidpoint, [
|
drag(
|
||||||
lastSegmentMidpoint[0] + delta,
|
lastSegmentMidpoint,
|
||||||
lastSegmentMidpoint[1] + delta,
|
point(lastSegmentMidpoint[0] + delta, lastSegmentMidpoint[1] + delta),
|
||||||
]);
|
);
|
||||||
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(
|
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(
|
||||||
`16`,
|
`16`,
|
||||||
);
|
);
|
||||||
|
@ -661,10 +669,10 @@ describe("Test Linear Elements", () => {
|
||||||
h.state,
|
h.state,
|
||||||
);
|
);
|
||||||
|
|
||||||
const hitCoords: Point = [points[0][0], points[0][1]];
|
const hitCoords = point<GlobalPoint>(points[0][0], points[0][1]);
|
||||||
|
|
||||||
// Drag from first point
|
// Drag from first point
|
||||||
drag(hitCoords, [hitCoords[0] - delta, hitCoords[1] - delta]);
|
drag(hitCoords, point(hitCoords[0] - delta, hitCoords[1] - delta));
|
||||||
|
|
||||||
const newPoints = LinearElementEditor.getPointsGlobalCoordinates(
|
const newPoints = LinearElementEditor.getPointsGlobalCoordinates(
|
||||||
line,
|
line,
|
||||||
|
@ -709,10 +717,10 @@ describe("Test Linear Elements", () => {
|
||||||
h.state,
|
h.state,
|
||||||
);
|
);
|
||||||
|
|
||||||
const hitCoords: Point = [points[0][0], points[0][1]];
|
const hitCoords = point<GlobalPoint>(points[0][0], points[0][1]);
|
||||||
|
|
||||||
// Drag from first point
|
// Drag from first point
|
||||||
drag(hitCoords, [hitCoords[0] + delta, hitCoords[1] + delta]);
|
drag(hitCoords, point(hitCoords[0] + delta, hitCoords[1] + delta));
|
||||||
|
|
||||||
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(
|
expect(renderInteractiveScene.mock.calls.length).toMatchInlineSnapshot(
|
||||||
`12`,
|
`12`,
|
||||||
|
@ -741,10 +749,10 @@ describe("Test Linear Elements", () => {
|
||||||
it("should update all the midpoints when a point is deleted", async () => {
|
it("should update all the midpoints when a point is deleted", async () => {
|
||||||
const elementsMap = arrayToMap(h.elements);
|
const elementsMap = arrayToMap(h.elements);
|
||||||
|
|
||||||
drag(lastSegmentMidpoint, [
|
drag(
|
||||||
lastSegmentMidpoint[0] + delta,
|
lastSegmentMidpoint,
|
||||||
lastSegmentMidpoint[1] + delta,
|
point(lastSegmentMidpoint[0] + delta, lastSegmentMidpoint[1] + delta),
|
||||||
]);
|
);
|
||||||
expect(line.points.length).toEqual(4);
|
expect(line.points.length).toEqual(4);
|
||||||
|
|
||||||
const midPoints = LinearElementEditor.getEditorMidPoints(
|
const midPoints = LinearElementEditor.getEditorMidPoints(
|
||||||
|
@ -803,8 +811,11 @@ describe("Test Linear Elements", () => {
|
||||||
API.setSelectedElements([line]);
|
API.setSelectedElements([line]);
|
||||||
enterLineEditingMode(line, true);
|
enterLineEditingMode(line, true);
|
||||||
drag(
|
drag(
|
||||||
[line.points[0][0] + line.x, line.points[0][1] + line.y],
|
point(line.points[0][0] + line.x, line.points[0][1] + line.y),
|
||||||
[dragEndPositionOffset[0] + line.x, dragEndPositionOffset[1] + line.y],
|
point(
|
||||||
|
dragEndPositionOffset[0] + line.x,
|
||||||
|
dragEndPositionOffset[1] + line.y,
|
||||||
|
),
|
||||||
);
|
);
|
||||||
expect(line.points).toMatchInlineSnapshot(`
|
expect(line.points).toMatchInlineSnapshot(`
|
||||||
[
|
[
|
||||||
|
@ -916,14 +927,18 @@ describe("Test Linear Elements", () => {
|
||||||
// This is the expected midpoint for line with round edge
|
// This is the expected midpoint for line with round edge
|
||||||
// hence hardcoding it so if later some bug is introduced
|
// hence hardcoding it so if later some bug is introduced
|
||||||
// this will fail and we can fix it
|
// this will fail and we can fix it
|
||||||
const firstSegmentMidpoint: Point = [
|
const firstSegmentMidpoint = point<GlobalPoint>(
|
||||||
55.9697848965255, 47.442326230998205,
|
55.9697848965255,
|
||||||
];
|
47.442326230998205,
|
||||||
|
);
|
||||||
// drag line from first segment midpoint
|
// drag line from first segment midpoint
|
||||||
drag(firstSegmentMidpoint, [
|
drag(
|
||||||
|
firstSegmentMidpoint,
|
||||||
|
point(
|
||||||
firstSegmentMidpoint[0] + delta,
|
firstSegmentMidpoint[0] + delta,
|
||||||
firstSegmentMidpoint[1] + delta,
|
firstSegmentMidpoint[1] + delta,
|
||||||
]);
|
),
|
||||||
|
);
|
||||||
|
|
||||||
const position = LinearElementEditor.getBoundTextElementPosition(
|
const position = LinearElementEditor.getBoundTextElementPosition(
|
||||||
container,
|
container,
|
||||||
|
@ -1136,7 +1151,7 @@ describe("Test Linear Elements", () => {
|
||||||
);
|
);
|
||||||
|
|
||||||
// Drag from last point
|
// Drag from last point
|
||||||
drag(points[1], [points[1][0] + 300, points[1][1]]);
|
drag(points[1], point(points[1][0] + 300, points[1][1]));
|
||||||
|
|
||||||
expect({ width: container.width, height: container.height })
|
expect({ width: container.width, height: container.height })
|
||||||
.toMatchInlineSnapshot(`
|
.toMatchInlineSnapshot(`
|
||||||
|
@ -1335,14 +1350,14 @@ describe("Test Linear Elements", () => {
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
index: 0,
|
index: 0,
|
||||||
point: [line.points[0][0] + 10, line.points[0][1] + 10],
|
point: point(line.points[0][0] + 10, line.points[0][1] + 10),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
index: line.points.length - 1,
|
index: line.points.length - 1,
|
||||||
point: [
|
point: point(
|
||||||
line.points[line.points.length - 1][0] - 10,
|
line.points[line.points.length - 1][0] - 10,
|
||||||
line.points[line.points.length - 1][1] - 10,
|
line.points[line.points.length - 1][1] - 10,
|
||||||
],
|
),
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
new Map() as SceneElementsMap,
|
new Map() as SceneElementsMap,
|
||||||
|
|
|
@ -43,7 +43,7 @@ describe("event callbacks", () => {
|
||||||
// files
|
// files
|
||||||
{},
|
{},
|
||||||
);
|
);
|
||||||
expect(onChange.mock.lastCall[1].viewBackgroundColor).not.toBe(
|
expect(onChange.mock?.lastCall?.[1].viewBackgroundColor).not.toBe(
|
||||||
origBackgroundColor,
|
origBackgroundColor,
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
|
@ -11,7 +11,7 @@ export const getTextEditor = async (selector: string, waitForEditor = true) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
export const updateTextEditor = (
|
export const updateTextEditor = (
|
||||||
editor: HTMLTextAreaElement,
|
editor: HTMLTextAreaElement | HTMLInputElement,
|
||||||
value: string,
|
value: string,
|
||||||
) => {
|
) => {
|
||||||
fireEvent.change(editor, { target: { value } });
|
fireEvent.change(editor, { target: { value } });
|
||||||
|
|
|
@ -7,7 +7,6 @@ import type {
|
||||||
ExcalidrawFreeDrawElement,
|
ExcalidrawFreeDrawElement,
|
||||||
ExcalidrawLinearElement,
|
ExcalidrawLinearElement,
|
||||||
} from "../element/types";
|
} from "../element/types";
|
||||||
import type { Point } from "../types";
|
|
||||||
import type { Bounds } from "../element/bounds";
|
import type { Bounds } from "../element/bounds";
|
||||||
import { getElementPointsCoords } from "../element/bounds";
|
import { getElementPointsCoords } from "../element/bounds";
|
||||||
import { Excalidraw } from "../index";
|
import { Excalidraw } from "../index";
|
||||||
|
@ -16,6 +15,8 @@ import { KEYS } from "../keys";
|
||||||
import { isLinearElement } from "../element/typeChecks";
|
import { isLinearElement } from "../element/typeChecks";
|
||||||
import { LinearElementEditor } from "../element/linearElementEditor";
|
import { LinearElementEditor } from "../element/linearElementEditor";
|
||||||
import { arrayToMap } from "../utils";
|
import { arrayToMap } from "../utils";
|
||||||
|
import type { LocalPoint } from "../../math";
|
||||||
|
import { point } from "../../math";
|
||||||
|
|
||||||
ReactDOM.unmountComponentAtNode(document.getElementById("root")!);
|
ReactDOM.unmountComponentAtNode(document.getElementById("root")!);
|
||||||
|
|
||||||
|
@ -217,18 +218,13 @@ describe("generic element", () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
describe.each(["line", "freedraw"] as const)("%s element", (type) => {
|
describe.each(["line", "freedraw"] as const)("%s element", (type) => {
|
||||||
const points: Record<typeof type, Point[]> = {
|
const points: Record<typeof type, LocalPoint[]> = {
|
||||||
line: [
|
line: [point(0, 0), point(60, -20), point(20, 40), point(-40, 0)],
|
||||||
[0, 0],
|
|
||||||
[60, -20],
|
|
||||||
[20, 40],
|
|
||||||
[-40, 0],
|
|
||||||
],
|
|
||||||
freedraw: [
|
freedraw: [
|
||||||
[0, 0],
|
point(0, 0),
|
||||||
[-2.474600807561444, 41.021700699972],
|
point(-2.474600807561444, 41.021700699972),
|
||||||
[3.6627956000014024, 47.84174560617245],
|
point(3.6627956000014024, 47.84174560617245),
|
||||||
[40.495224145598115, 47.15909710753482],
|
point(40.495224145598115, 47.15909710753482),
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -296,11 +292,11 @@ describe("arrow element", () => {
|
||||||
it("resizes with a label", async () => {
|
it("resizes with a label", async () => {
|
||||||
const arrow = UI.createElement("arrow", {
|
const arrow = UI.createElement("arrow", {
|
||||||
points: [
|
points: [
|
||||||
[0, 0],
|
point(0, 0),
|
||||||
[40, 140],
|
point(40, 140),
|
||||||
[80, 60], // label's anchor
|
point(80, 60), // label's anchor
|
||||||
[180, 20],
|
point(180, 20),
|
||||||
[200, 120],
|
point(200, 120),
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
const label = await UI.editText(arrow, "Hello");
|
const label = await UI.editText(arrow, "Hello");
|
||||||
|
@ -694,24 +690,24 @@ describe("multiple selection", () => {
|
||||||
x: 60,
|
x: 60,
|
||||||
y: 40,
|
y: 40,
|
||||||
points: [
|
points: [
|
||||||
[0, 0],
|
point(0, 0),
|
||||||
[-40, 40],
|
point(-40, 40),
|
||||||
[-60, 0],
|
point(-60, 0),
|
||||||
[0, -40],
|
point(0, -40),
|
||||||
[40, 20],
|
point(40, 20),
|
||||||
[0, 40],
|
point(0, 40),
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
const freedraw = UI.createElement("freedraw", {
|
const freedraw = UI.createElement("freedraw", {
|
||||||
x: 63.56072661326618,
|
x: 63.56072661326618,
|
||||||
y: 100,
|
y: 100,
|
||||||
points: [
|
points: [
|
||||||
[0, 0],
|
point(0, 0),
|
||||||
[-43.56072661326618, 18.15048126846341],
|
point(-43.56072661326618, 18.15048126846341),
|
||||||
[-43.56072661326618, 29.041198460587566],
|
point(-43.56072661326618, 29.041198460587566),
|
||||||
[-38.115368017204105, 42.652452795512204],
|
point(-38.115368017204105, 42.652452795512204),
|
||||||
[-19.964886748740696, 66.24829266003775],
|
point(-19.964886748740696, 66.24829266003775),
|
||||||
[19.056612930986716, 77.1390098521619],
|
point(19.056612930986716, 77.1390098521619),
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -1050,13 +1046,13 @@ describe("multiple selection", () => {
|
||||||
x: 60,
|
x: 60,
|
||||||
y: 0,
|
y: 0,
|
||||||
points: [
|
points: [
|
||||||
[0, 0],
|
point(0, 0),
|
||||||
[-40, 40],
|
point(-40, 40),
|
||||||
[-20, 60],
|
point(-20, 60),
|
||||||
[20, 20],
|
point(20, 20),
|
||||||
[40, 40],
|
point(40, 40),
|
||||||
[-20, 100],
|
point(-20, 100),
|
||||||
[-60, 60],
|
point(-60, 60),
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
File diff suppressed because one or more lines are too long
145
packages/excalidraw/tests/search.test.tsx
Normal file
145
packages/excalidraw/tests/search.test.tsx
Normal file
|
@ -0,0 +1,145 @@
|
||||||
|
import React from "react";
|
||||||
|
import { act, render, waitFor } from "./test-utils";
|
||||||
|
import { Excalidraw } from "../index";
|
||||||
|
import { CLASSES, SEARCH_SIDEBAR } from "../constants";
|
||||||
|
import { Keyboard } from "./helpers/ui";
|
||||||
|
import { KEYS } from "../keys";
|
||||||
|
import { updateTextEditor } from "./queries/dom";
|
||||||
|
import { API } from "./helpers/api";
|
||||||
|
import type { ExcalidrawTextElement } from "../element/types";
|
||||||
|
|
||||||
|
const { h } = window;
|
||||||
|
|
||||||
|
const querySearchInput = async () => {
|
||||||
|
const input =
|
||||||
|
h.app.excalidrawContainerValue.container?.querySelector<HTMLInputElement>(
|
||||||
|
`.${CLASSES.SEARCH_MENU_INPUT_WRAPPER} input`,
|
||||||
|
)!;
|
||||||
|
await waitFor(() => expect(input).not.toBeNull());
|
||||||
|
return input;
|
||||||
|
};
|
||||||
|
|
||||||
|
describe("search", () => {
|
||||||
|
beforeEach(async () => {
|
||||||
|
await render(<Excalidraw handleKeyboardGlobally />);
|
||||||
|
API.setAppState({
|
||||||
|
openSidebar: null,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should toggle search on cmd+f", async () => {
|
||||||
|
expect(h.app.state.openSidebar).toBeNull();
|
||||||
|
|
||||||
|
Keyboard.withModifierKeys({ ctrl: true }, () => {
|
||||||
|
Keyboard.keyPress(KEYS.F);
|
||||||
|
});
|
||||||
|
expect(h.app.state.openSidebar).not.toBeNull();
|
||||||
|
expect(h.app.state.openSidebar?.name).toBe(SEARCH_SIDEBAR.name);
|
||||||
|
|
||||||
|
const searchInput = await querySearchInput();
|
||||||
|
expect(searchInput.matches(":focus")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should refocus search input with cmd+f when search sidebar is still open", async () => {
|
||||||
|
Keyboard.withModifierKeys({ ctrl: true }, () => {
|
||||||
|
Keyboard.keyPress(KEYS.F);
|
||||||
|
});
|
||||||
|
|
||||||
|
const searchInput =
|
||||||
|
h.app.excalidrawContainerValue.container?.querySelector<HTMLInputElement>(
|
||||||
|
`.${CLASSES.SEARCH_MENU_INPUT_WRAPPER} input`,
|
||||||
|
);
|
||||||
|
|
||||||
|
act(() => {
|
||||||
|
searchInput?.blur();
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(h.app.state.openSidebar).not.toBeNull();
|
||||||
|
expect(searchInput?.matches(":focus")).toBe(false);
|
||||||
|
|
||||||
|
Keyboard.withModifierKeys({ ctrl: true }, () => {
|
||||||
|
Keyboard.keyPress(KEYS.F);
|
||||||
|
});
|
||||||
|
expect(searchInput?.matches(":focus")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should match text and cycle through matches on Enter", async () => {
|
||||||
|
const scrollIntoViewMock = jest.fn();
|
||||||
|
window.HTMLElement.prototype.scrollIntoView = scrollIntoViewMock;
|
||||||
|
|
||||||
|
API.setElements([
|
||||||
|
API.createElement({ type: "text", text: "test one" }),
|
||||||
|
API.createElement({ type: "text", text: "test two" }),
|
||||||
|
]);
|
||||||
|
|
||||||
|
expect(h.app.state.openSidebar).toBeNull();
|
||||||
|
|
||||||
|
Keyboard.withModifierKeys({ ctrl: true }, () => {
|
||||||
|
Keyboard.keyPress(KEYS.F);
|
||||||
|
});
|
||||||
|
expect(h.app.state.openSidebar).not.toBeNull();
|
||||||
|
expect(h.app.state.openSidebar?.name).toBe(SEARCH_SIDEBAR.name);
|
||||||
|
|
||||||
|
const searchInput = await querySearchInput();
|
||||||
|
|
||||||
|
expect(searchInput.matches(":focus")).toBe(true);
|
||||||
|
|
||||||
|
updateTextEditor(searchInput, "test");
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(h.app.state.searchMatches.length).toBe(2);
|
||||||
|
expect(h.app.state.searchMatches[0].focus).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
Keyboard.keyPress(KEYS.ENTER, searchInput);
|
||||||
|
expect(h.app.state.searchMatches[0].focus).toBe(false);
|
||||||
|
expect(h.app.state.searchMatches[1].focus).toBe(true);
|
||||||
|
|
||||||
|
Keyboard.keyPress(KEYS.ENTER, searchInput);
|
||||||
|
expect(h.app.state.searchMatches[0].focus).toBe(true);
|
||||||
|
expect(h.app.state.searchMatches[1].focus).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should match text split across multiple lines", async () => {
|
||||||
|
const scrollIntoViewMock = jest.fn();
|
||||||
|
window.HTMLElement.prototype.scrollIntoView = scrollIntoViewMock;
|
||||||
|
|
||||||
|
API.setElements([
|
||||||
|
API.createElement({
|
||||||
|
type: "text",
|
||||||
|
text: "",
|
||||||
|
}),
|
||||||
|
]);
|
||||||
|
|
||||||
|
API.updateElement(h.elements[0] as ExcalidrawTextElement, {
|
||||||
|
text: "t\ne\ns\nt \nt\ne\nx\nt \ns\np\nli\nt \ni\nn\nt\no\nm\nu\nlt\ni\np\nl\ne \nli\nn\ne\ns",
|
||||||
|
originalText: "test text split into multiple lines",
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(h.app.state.openSidebar).toBeNull();
|
||||||
|
|
||||||
|
Keyboard.withModifierKeys({ ctrl: true }, () => {
|
||||||
|
Keyboard.keyPress(KEYS.F);
|
||||||
|
});
|
||||||
|
expect(h.app.state.openSidebar).not.toBeNull();
|
||||||
|
expect(h.app.state.openSidebar?.name).toBe(SEARCH_SIDEBAR.name);
|
||||||
|
|
||||||
|
const searchInput = await querySearchInput();
|
||||||
|
|
||||||
|
expect(searchInput.matches(":focus")).toBe(true);
|
||||||
|
|
||||||
|
updateTextEditor(searchInput, "test");
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(h.app.state.searchMatches.length).toBe(1);
|
||||||
|
expect(h.app.state.searchMatches[0]?.matchedLines?.length).toBe(4);
|
||||||
|
});
|
||||||
|
|
||||||
|
updateTextEditor(searchInput, "ext spli");
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(h.app.state.searchMatches.length).toBe(1);
|
||||||
|
expect(h.app.state.searchMatches[0]?.matchedLines?.length).toBe(6);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -24,7 +24,6 @@ import type {
|
||||||
ExcalidrawNonSelectionElement,
|
ExcalidrawNonSelectionElement,
|
||||||
} from "./element/types";
|
} from "./element/types";
|
||||||
import type { Action } from "./actions/types";
|
import type { Action } from "./actions/types";
|
||||||
import type { Point as RoughPoint } from "roughjs/bin/geometry";
|
|
||||||
import type { LinearElementEditor } from "./element/linearElementEditor";
|
import type { LinearElementEditor } from "./element/linearElementEditor";
|
||||||
import type { SuggestedBinding } from "./element/binding";
|
import type { SuggestedBinding } from "./element/binding";
|
||||||
import type { ImportedDataState } from "./data/types";
|
import type { ImportedDataState } from "./data/types";
|
||||||
|
@ -43,8 +42,6 @@ import type { SnapLine } from "./snapping";
|
||||||
import type { Merge, MaybePromise, ValueOf, MakeBrand } from "./utility-types";
|
import type { Merge, MaybePromise, ValueOf, MakeBrand } from "./utility-types";
|
||||||
import type { StoreActionType } from "./store";
|
import type { StoreActionType } from "./store";
|
||||||
|
|
||||||
export type Point = Readonly<RoughPoint>;
|
|
||||||
|
|
||||||
export type SocketId = string & { _brand: "SocketId" };
|
export type SocketId = string & { _brand: "SocketId" };
|
||||||
|
|
||||||
export type Collaborator = Readonly<{
|
export type Collaborator = Readonly<{
|
||||||
|
@ -201,6 +198,8 @@ export type InteractiveCanvasAppState = Readonly<
|
||||||
snapLines: AppState["snapLines"];
|
snapLines: AppState["snapLines"];
|
||||||
zenModeEnabled: AppState["zenModeEnabled"];
|
zenModeEnabled: AppState["zenModeEnabled"];
|
||||||
editingTextElement: AppState["editingTextElement"];
|
editingTextElement: AppState["editingTextElement"];
|
||||||
|
// Search matches
|
||||||
|
searchMatches: AppState["searchMatches"];
|
||||||
}
|
}
|
||||||
>;
|
>;
|
||||||
|
|
||||||
|
@ -387,8 +386,20 @@ export interface AppState {
|
||||||
userToFollow: UserToFollow | null;
|
userToFollow: UserToFollow | null;
|
||||||
/** the socket ids of the users following the current user */
|
/** the socket ids of the users following the current user */
|
||||||
followedBy: Set<SocketId>;
|
followedBy: Set<SocketId>;
|
||||||
|
searchMatches: readonly SearchMatch[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type SearchMatch = {
|
||||||
|
id: string;
|
||||||
|
focus: boolean;
|
||||||
|
matchedLines: {
|
||||||
|
offsetX: number;
|
||||||
|
offsetY: number;
|
||||||
|
width: number;
|
||||||
|
height: number;
|
||||||
|
}[];
|
||||||
|
};
|
||||||
|
|
||||||
export type UIAppState = Omit<
|
export type UIAppState = Omit<
|
||||||
AppState,
|
AppState,
|
||||||
| "suggestedBindings"
|
| "suggestedBindings"
|
||||||
|
@ -645,6 +656,9 @@ export type AppClassProperties = {
|
||||||
getEffectiveGridSize: App["getEffectiveGridSize"];
|
getEffectiveGridSize: App["getEffectiveGridSize"];
|
||||||
setPlugins: App["setPlugins"];
|
setPlugins: App["setPlugins"];
|
||||||
plugins: App["plugins"];
|
plugins: App["plugins"];
|
||||||
|
getEditorUIOffsets: App["getEditorUIOffsets"];
|
||||||
|
visibleElements: App["visibleElements"];
|
||||||
|
excalidrawContainerValue: App["excalidrawContainerValue"];
|
||||||
};
|
};
|
||||||
|
|
||||||
export type PointerDownState = Readonly<{
|
export type PointerDownState = Readonly<{
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
import { average } from "../math";
|
||||||
import { COLOR_PALETTE } from "./colors";
|
import { COLOR_PALETTE } from "./colors";
|
||||||
import type { EVENT } from "./constants";
|
import type { EVENT } from "./constants";
|
||||||
import {
|
import {
|
||||||
|
@ -992,10 +993,6 @@ export const isMemberOf = <T extends string>(
|
||||||
|
|
||||||
export const cloneJSON = <T>(obj: T): T => JSON.parse(JSON.stringify(obj));
|
export const cloneJSON = <T>(obj: T): T => JSON.parse(JSON.stringify(obj));
|
||||||
|
|
||||||
export const isFiniteNumber = (value: any): value is number => {
|
|
||||||
return typeof value === "number" && Number.isFinite(value);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const updateStable = <T extends any[] | Record<string, any>>(
|
export const updateStable = <T extends any[] | Record<string, any>>(
|
||||||
prevValue: T,
|
prevValue: T,
|
||||||
nextValue: T,
|
nextValue: T,
|
||||||
|
@ -1079,7 +1076,6 @@ export function addEventListener(
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
const average = (a: number, b: number) => (a + b) / 2;
|
|
||||||
export function getSvgPathFromStroke(points: number[][], closed = true) {
|
export function getSvgPathFromStroke(points: number[][], closed = true) {
|
||||||
const len = points.length;
|
const len = points.length;
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
|
import { isLineSegment, lineSegment, point, type GlobalPoint } from "../math";
|
||||||
import type { LineSegment } from "../utils";
|
import type { LineSegment } from "../utils";
|
||||||
import type { BoundingBox, Bounds } from "./element/bounds";
|
import type { BoundingBox, Bounds } from "./element/bounds";
|
||||||
import { isBounds, isLineSegment } from "./element/typeChecks";
|
import { isBounds } from "./element/typeChecks";
|
||||||
import type { Point } from "./types";
|
|
||||||
|
|
||||||
// The global data holder to collect the debug operations
|
// The global data holder to collect the debug operations
|
||||||
declare global {
|
declare global {
|
||||||
|
@ -15,18 +15,22 @@ declare global {
|
||||||
|
|
||||||
export type DebugElement = {
|
export type DebugElement = {
|
||||||
color: string;
|
color: string;
|
||||||
data: LineSegment;
|
data: LineSegment<GlobalPoint>;
|
||||||
permanent: boolean;
|
permanent: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const debugDrawLine = (
|
export const debugDrawLine = (
|
||||||
segment: LineSegment | LineSegment[],
|
segment: LineSegment<GlobalPoint> | LineSegment<GlobalPoint>[],
|
||||||
opts?: {
|
opts?: {
|
||||||
color?: string;
|
color?: string;
|
||||||
permanent?: boolean;
|
permanent?: boolean;
|
||||||
},
|
},
|
||||||
) => {
|
) => {
|
||||||
(isLineSegment(segment) ? [segment] : segment).forEach((data) =>
|
const segments = (
|
||||||
|
isLineSegment(segment) ? [segment] : segment
|
||||||
|
) as LineSegment<GlobalPoint>[];
|
||||||
|
|
||||||
|
segments.forEach((data) =>
|
||||||
addToCurrentFrame({
|
addToCurrentFrame({
|
||||||
color: opts?.color ?? "red",
|
color: opts?.color ?? "red",
|
||||||
data,
|
data,
|
||||||
|
@ -36,7 +40,7 @@ export const debugDrawLine = (
|
||||||
};
|
};
|
||||||
|
|
||||||
export const debugDrawPoint = (
|
export const debugDrawPoint = (
|
||||||
point: Point,
|
p: GlobalPoint,
|
||||||
opts?: {
|
opts?: {
|
||||||
color?: string;
|
color?: string;
|
||||||
permanent?: boolean;
|
permanent?: boolean;
|
||||||
|
@ -47,20 +51,20 @@ export const debugDrawPoint = (
|
||||||
const yOffset = opts?.fuzzy ? Math.random() * 3 : 0;
|
const yOffset = opts?.fuzzy ? Math.random() * 3 : 0;
|
||||||
|
|
||||||
debugDrawLine(
|
debugDrawLine(
|
||||||
[
|
lineSegment(
|
||||||
[point[0] + xOffset - 10, point[1] + yOffset - 10],
|
point<GlobalPoint>(p[0] + xOffset - 10, p[1] + yOffset - 10),
|
||||||
[point[0] + xOffset + 10, point[1] + yOffset + 10],
|
point<GlobalPoint>(p[0] + xOffset + 10, p[1] + yOffset + 10),
|
||||||
],
|
),
|
||||||
{
|
{
|
||||||
color: opts?.color ?? "cyan",
|
color: opts?.color ?? "cyan",
|
||||||
permanent: opts?.permanent,
|
permanent: opts?.permanent,
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
debugDrawLine(
|
debugDrawLine(
|
||||||
[
|
lineSegment(
|
||||||
[point[0] + xOffset - 10, point[1] + yOffset + 10],
|
point<GlobalPoint>(p[0] + xOffset - 10, p[1] + yOffset + 10),
|
||||||
[point[0] + xOffset + 10, point[1] + yOffset - 10],
|
point<GlobalPoint>(p[0] + xOffset + 10, p[1] + yOffset - 10),
|
||||||
],
|
),
|
||||||
{
|
{
|
||||||
color: opts?.color ?? "cyan",
|
color: opts?.color ?? "cyan",
|
||||||
permanent: opts?.permanent,
|
permanent: opts?.permanent,
|
||||||
|
@ -78,22 +82,22 @@ export const debugDrawBoundingBox = (
|
||||||
(Array.isArray(box) ? box : [box]).forEach((bbox) =>
|
(Array.isArray(box) ? box : [box]).forEach((bbox) =>
|
||||||
debugDrawLine(
|
debugDrawLine(
|
||||||
[
|
[
|
||||||
[
|
lineSegment(
|
||||||
[bbox.minX, bbox.minY],
|
point<GlobalPoint>(bbox.minX, bbox.minY),
|
||||||
[bbox.maxX, bbox.minY],
|
point<GlobalPoint>(bbox.maxX, bbox.minY),
|
||||||
],
|
),
|
||||||
[
|
lineSegment(
|
||||||
[bbox.maxX, bbox.minY],
|
point<GlobalPoint>(bbox.maxX, bbox.minY),
|
||||||
[bbox.maxX, bbox.maxY],
|
point<GlobalPoint>(bbox.maxX, bbox.maxY),
|
||||||
],
|
),
|
||||||
[
|
lineSegment(
|
||||||
[bbox.maxX, bbox.maxY],
|
point<GlobalPoint>(bbox.maxX, bbox.maxY),
|
||||||
[bbox.minX, bbox.maxY],
|
point<GlobalPoint>(bbox.minX, bbox.maxY),
|
||||||
],
|
),
|
||||||
[
|
lineSegment(
|
||||||
[bbox.minX, bbox.maxY],
|
point<GlobalPoint>(bbox.minX, bbox.maxY),
|
||||||
[bbox.minX, bbox.minY],
|
point<GlobalPoint>(bbox.minX, bbox.minY),
|
||||||
],
|
),
|
||||||
],
|
],
|
||||||
{
|
{
|
||||||
color: opts?.color ?? "cyan",
|
color: opts?.color ?? "cyan",
|
||||||
|
@ -113,22 +117,22 @@ export const debugDrawBounds = (
|
||||||
(isBounds(box) ? [box] : box).forEach((bbox) =>
|
(isBounds(box) ? [box] : box).forEach((bbox) =>
|
||||||
debugDrawLine(
|
debugDrawLine(
|
||||||
[
|
[
|
||||||
[
|
lineSegment(
|
||||||
[bbox[0], bbox[1]],
|
point<GlobalPoint>(bbox[0], bbox[1]),
|
||||||
[bbox[2], bbox[1]],
|
point<GlobalPoint>(bbox[2], bbox[1]),
|
||||||
],
|
),
|
||||||
[
|
lineSegment(
|
||||||
[bbox[2], bbox[1]],
|
point<GlobalPoint>(bbox[2], bbox[1]),
|
||||||
[bbox[2], bbox[3]],
|
point<GlobalPoint>(bbox[2], bbox[3]),
|
||||||
],
|
),
|
||||||
[
|
lineSegment(
|
||||||
[bbox[2], bbox[3]],
|
point<GlobalPoint>(bbox[2], bbox[3]),
|
||||||
[bbox[0], bbox[3]],
|
point<GlobalPoint>(bbox[0], bbox[3]),
|
||||||
],
|
),
|
||||||
[
|
lineSegment(
|
||||||
[bbox[0], bbox[3]],
|
point<GlobalPoint>(bbox[0], bbox[3]),
|
||||||
[bbox[0], bbox[1]],
|
point<GlobalPoint>(bbox[0], bbox[1]),
|
||||||
],
|
),
|
||||||
],
|
],
|
||||||
{
|
{
|
||||||
color: opts?.color ?? "green",
|
color: opts?.color ?? "green",
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Add a link
Reference in a new issue