mirror of
https://github.com/excalidraw/excalidraw.git
synced 2025-10-24 08:24:32 +02:00
Compare commits
14 Commits
mobile_tex
...
updatescen
Author | SHA1 | Date | |
---|---|---|---|
![]() |
eb206cc932 | ||
![]() |
16c287c848 | ||
![]() |
78024873e5 | ||
![]() |
4e41bd9dbb | ||
![]() |
edc23b854f | ||
![]() |
4843c49556 | ||
![]() |
d565413082 | ||
![]() |
dcda7184d0 | ||
![]() |
8d413670c8 | ||
![]() |
f774452124 | ||
![]() |
db4ed1ecb1 | ||
![]() |
489f45b910 | ||
![]() |
a17be085b0 | ||
![]() |
4e07a608d3 |
26
.github/workflows/autorelease-excalidraw.yml
vendored
26
.github/workflows/autorelease-excalidraw.yml
vendored
@@ -1,26 +0,0 @@
|
|||||||
name: Auto release @excalidraw/excalidraw-next
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- master
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
Auto-release-excalidraw-next:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v2
|
|
||||||
with:
|
|
||||||
fetch-depth: 2
|
|
||||||
- name: Setup Node.js 14.x
|
|
||||||
uses: actions/setup-node@v2
|
|
||||||
with:
|
|
||||||
node-version: 14.x
|
|
||||||
- name: Set up publish access
|
|
||||||
run: |
|
|
||||||
npm config set //registry.npmjs.org/:_authToken ${NPM_TOKEN}
|
|
||||||
env:
|
|
||||||
NPM_TOKEN: ${{ secrets.NPM_TOKEN }}
|
|
||||||
- name: Auto release
|
|
||||||
run: |
|
|
||||||
yarn autorelease
|
|
1
.gitignore
vendored
1
.gitignore
vendored
@@ -20,4 +20,3 @@ package-lock.json
|
|||||||
static
|
static
|
||||||
yarn-debug.log*
|
yarn-debug.log*
|
||||||
yarn-error.log*
|
yarn-error.log*
|
||||||
src/packages/excalidraw/types
|
|
||||||
|
14
README.md
14
README.md
@@ -102,20 +102,6 @@ These instructions will get you a copy of the project up and running on your loc
|
|||||||
git clone https://github.com/excalidraw/excalidraw.git
|
git clone https://github.com/excalidraw/excalidraw.git
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Install the dependencies
|
|
||||||
|
|
||||||
```bash
|
|
||||||
yarn
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Start the server
|
|
||||||
|
|
||||||
```bash
|
|
||||||
yarn start
|
|
||||||
```
|
|
||||||
|
|
||||||
Now you can open [http://localhost:3000](http://localhost:3000) and start coding in your favorite code editor.
|
|
||||||
|
|
||||||
#### Commands
|
#### Commands
|
||||||
|
|
||||||
| Command | Description |
|
| Command | Description |
|
||||||
|
40
package.json
40
package.json
@@ -19,35 +19,34 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@sentry/browser": "6.2.5",
|
"@sentry/browser": "6.2.2",
|
||||||
"@sentry/integrations": "6.2.5",
|
"@sentry/integrations": "6.2.1",
|
||||||
"@testing-library/jest-dom": "5.11.10",
|
"@testing-library/jest-dom": "5.11.9",
|
||||||
"@testing-library/react": "11.2.6",
|
"@testing-library/react": "11.2.5",
|
||||||
"@types/jest": "26.0.22",
|
"@types/jest": "26.0.20",
|
||||||
"@types/react": "17.0.3",
|
"@types/react": "17.0.2",
|
||||||
"@types/react-dom": "17.0.3",
|
"@types/react-dom": "17.0.1",
|
||||||
"@types/socket.io-client": "1.4.36",
|
"@types/socket.io-client": "1.4.36",
|
||||||
"browser-fs-access": "0.16.4",
|
"browser-fs-access": "0.14.2",
|
||||||
"clsx": "1.1.1",
|
"clsx": "1.1.1",
|
||||||
"firebase": "8.3.3",
|
"firebase": "8.2.10",
|
||||||
"i18next-browser-languagedetector": "6.1.0",
|
"i18next-browser-languagedetector": "6.0.1",
|
||||||
"lodash.throttle": "4.1.1",
|
"lodash.throttle": "4.1.1",
|
||||||
"nanoid": "3.1.22",
|
"nanoid": "3.1.21",
|
||||||
"open-color": "1.8.0",
|
"open-color": "1.8.0",
|
||||||
"pako": "1.0.11",
|
"pako": "1.0.11",
|
||||||
"perfect-freehand": "0.4.7",
|
|
||||||
"png-chunk-text": "1.0.0",
|
"png-chunk-text": "1.0.0",
|
||||||
"png-chunks-encode": "1.0.0",
|
"png-chunks-encode": "1.0.0",
|
||||||
"png-chunks-extract": "1.0.0",
|
"png-chunks-extract": "1.0.0",
|
||||||
"points-on-curve": "0.2.0",
|
"points-on-curve": "0.2.0",
|
||||||
"pwacompat": "2.0.17",
|
"pwacompat": "2.0.17",
|
||||||
"react": "17.0.2",
|
"react": "17.0.1",
|
||||||
"react-dom": "17.0.2",
|
"react-dom": "17.0.1",
|
||||||
"react-scripts": "4.0.3",
|
"react-scripts": "4.0.3",
|
||||||
"roughjs": "4.4.1",
|
"roughjs": "4.3.1",
|
||||||
"sass": "1.32.10",
|
"sass": "1.32.8",
|
||||||
"socket.io-client": "2.3.1",
|
"socket.io-client": "2.3.1",
|
||||||
"typescript": "4.2.4"
|
"typescript": "4.2.3"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@excalidraw/eslint-config": "1.0.0",
|
"@excalidraw/eslint-config": "1.0.0",
|
||||||
@@ -55,9 +54,9 @@
|
|||||||
"@types/lodash.throttle": "4.1.6",
|
"@types/lodash.throttle": "4.1.6",
|
||||||
"@types/pako": "1.0.1",
|
"@types/pako": "1.0.1",
|
||||||
"@types/resize-observer-browser": "0.1.5",
|
"@types/resize-observer-browser": "0.1.5",
|
||||||
"eslint-config-prettier": "8.3.0",
|
"eslint-config-prettier": "8.1.0",
|
||||||
"eslint-plugin-prettier": "3.3.1",
|
"eslint-plugin-prettier": "3.3.1",
|
||||||
"firebase-tools": "9.9.0",
|
"firebase-tools": "9.6.1",
|
||||||
"husky": "4.3.8",
|
"husky": "4.3.8",
|
||||||
"jest-canvas-mock": "2.3.1",
|
"jest-canvas-mock": "2.3.1",
|
||||||
"lint-staged": "10.5.4",
|
"lint-staged": "10.5.4",
|
||||||
@@ -104,7 +103,6 @@
|
|||||||
"test:other": "yarn prettier --list-different",
|
"test:other": "yarn prettier --list-different",
|
||||||
"test:typecheck": "tsc",
|
"test:typecheck": "tsc",
|
||||||
"test:update": "yarn test:app --updateSnapshot --watchAll=false",
|
"test:update": "yarn test:app --updateSnapshot --watchAll=false",
|
||||||
"test": "yarn test:app",
|
"test": "yarn test:app"
|
||||||
"autorelease": "node scripts/autorelease.js"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -51,7 +51,8 @@
|
|||||||
name="twitter:description"
|
name="twitter:description"
|
||||||
content="Excalidraw is a whiteboard tool that lets you easily sketch diagrams that have a hand-drawn feel to them."
|
content="Excalidraw is a whiteboard tool that lets you easily sketch diagrams that have a hand-drawn feel to them."
|
||||||
/>
|
/>
|
||||||
|
<!-- OG tags require absolute url for images -->
|
||||||
|
<meta name="twitter:image" content="https://excalidraw.com/og-image.png" />
|
||||||
<link rel="shortcut icon" href="favicon.ico" type="image/x-icon" />
|
<link rel="shortcut icon" href="favicon.ico" type="image/x-icon" />
|
||||||
|
|
||||||
<!-- Excalidraw version -->
|
<!-- Excalidraw version -->
|
||||||
@@ -87,8 +88,6 @@
|
|||||||
<link rel="stylesheet" href="fonts.css" type="text/css" />
|
<link rel="stylesheet" href="fonts.css" type="text/css" />
|
||||||
<script>
|
<script>
|
||||||
window.EXCALIDRAW_ASSET_PATH = "/";
|
window.EXCALIDRAW_ASSET_PATH = "/";
|
||||||
// setting this so that libraries installation reuses this window tab.
|
|
||||||
window.name = "_excalidraw";
|
|
||||||
</script>
|
</script>
|
||||||
<% if (process.env.REACT_APP_GOOGLE_ANALYTICS_ID) { %>
|
<% if (process.env.REACT_APP_GOOGLE_ANALYTICS_ID) { %>
|
||||||
<script
|
<script
|
||||||
@@ -107,16 +106,15 @@
|
|||||||
|
|
||||||
<!-- FIXME: remove this when we update CRA (fix SW caching) -->
|
<!-- FIXME: remove this when we update CRA (fix SW caching) -->
|
||||||
<style>
|
<style>
|
||||||
body,
|
body {
|
||||||
html {
|
|
||||||
margin: 0;
|
margin: 0;
|
||||||
--ui-font: system-ui, BlinkMacSystemFont, -apple-system, Segoe UI,
|
--ui-font: system-ui, BlinkMacSystemFont, -apple-system, Segoe UI,
|
||||||
Roboto, Helvetica, Arial, sans-serif;
|
Roboto, Helvetica, Arial, sans-serif;
|
||||||
font-family: var(--ui-font);
|
font-family: var(--ui-font);
|
||||||
-webkit-text-size-adjust: 100%;
|
-webkit-text-size-adjust: 100%;
|
||||||
|
|
||||||
width: 100%;
|
width: 100vw;
|
||||||
height: 100%;
|
height: 100vh;
|
||||||
}
|
}
|
||||||
|
|
||||||
.visually-hidden {
|
.visually-hidden {
|
||||||
@@ -126,7 +124,6 @@
|
|||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
clip: rect(1px, 1px, 1px, 1px);
|
clip: rect(1px, 1px, 1px, 1px);
|
||||||
white-space: nowrap; /* added line */
|
white-space: nowrap; /* added line */
|
||||||
user-select: none;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.LoadingMessage {
|
.LoadingMessage {
|
||||||
@@ -149,24 +146,6 @@
|
|||||||
color: var(--popup-text-color);
|
color: var(--popup-text-color);
|
||||||
font-size: 1.3em;
|
font-size: 1.3em;
|
||||||
}
|
}
|
||||||
#root {
|
|
||||||
height: 100%;
|
|
||||||
-webkit-touch-callout: none;
|
|
||||||
-webkit-user-select: none;
|
|
||||||
-khtml-user-select: none;
|
|
||||||
-moz-user-select: none;
|
|
||||||
-ms-user-select: none;
|
|
||||||
user-select: none;
|
|
||||||
|
|
||||||
@media screen and (min-width: 1200px) {
|
|
||||||
-webkit-touch-callout: default;
|
|
||||||
-webkit-user-select: auto;
|
|
||||||
-khtml-user-select: auto;
|
|
||||||
-moz-user-select: auto;
|
|
||||||
-ms-user-select: auto;
|
|
||||||
user-select: auto;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
|
@@ -39,37 +39,5 @@
|
|||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
"screenshots": [
|
|
||||||
{
|
|
||||||
"src": "/screenshots/virtual-whiteboard.png",
|
|
||||||
"type": "image/png",
|
|
||||||
"sizes": "462x945"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"src": "/screenshots/wireframe.png",
|
|
||||||
"type": "image/png",
|
|
||||||
"sizes": "462x945"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"src": "/screenshots/illustration.png",
|
|
||||||
"type": "image/png",
|
|
||||||
"sizes": "462x945"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"src": "/screenshots/shapes.png",
|
|
||||||
"type": "image/png",
|
|
||||||
"sizes": "462x945"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"src": "/screenshots/collaboration.png",
|
|
||||||
"type": "image/png",
|
|
||||||
"sizes": "462x945"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"src": "/screenshots/export.png",
|
|
||||||
"type": "image/png",
|
|
||||||
"sizes": "462x945"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
|
Binary file not shown.
Before Width: | Height: | Size: 28 KiB |
Binary file not shown.
Before Width: | Height: | Size: 25 KiB |
Binary file not shown.
Before Width: | Height: | Size: 47 KiB |
Binary file not shown.
Before Width: | Height: | Size: 25 KiB |
Binary file not shown.
Before Width: | Height: | Size: 27 KiB |
Binary file not shown.
Before Width: | Height: | Size: 27 KiB |
@@ -1,51 +0,0 @@
|
|||||||
const fs = require("fs");
|
|
||||||
const { exec, execSync } = require("child_process");
|
|
||||||
|
|
||||||
const excalidrawDir = `${__dirname}/../src/packages/excalidraw`;
|
|
||||||
const excalidrawPackage = `${excalidrawDir}/package.json`;
|
|
||||||
const pkg = require(excalidrawPackage);
|
|
||||||
|
|
||||||
const getShortCommitHash = () => {
|
|
||||||
return execSync("git rev-parse --short HEAD").toString().trim();
|
|
||||||
};
|
|
||||||
|
|
||||||
const publish = () => {
|
|
||||||
try {
|
|
||||||
execSync(`yarn --frozen-lockfile`);
|
|
||||||
execSync(`yarn --frozen-lockfile`, { cwd: excalidrawDir });
|
|
||||||
execSync(`yarn run build:umd`, { cwd: excalidrawDir });
|
|
||||||
execSync(`yarn --cwd ${excalidrawDir} publish`);
|
|
||||||
} catch (e) {
|
|
||||||
console.error(e);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// get files changed between prev and head commit
|
|
||||||
exec(`git diff --name-only HEAD^ HEAD`, async (error, stdout, stderr) => {
|
|
||||||
if (error || stderr) {
|
|
||||||
console.error(error);
|
|
||||||
process.exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
const changedFiles = stdout.trim().split("\n");
|
|
||||||
const filesToIgnoreRegex = /src\/excalidraw-app|packages\/utils/;
|
|
||||||
|
|
||||||
const excalidrawPackageFiles = changedFiles.filter((file) => {
|
|
||||||
return file.indexOf("src") >= 0 && !filesToIgnoreRegex.test(file);
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!excalidrawPackageFiles.length) {
|
|
||||||
process.exit(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
// update package.json
|
|
||||||
pkg.version = `${pkg.version}-${getShortCommitHash()}`;
|
|
||||||
pkg.name = "@excalidraw/excalidraw-next";
|
|
||||||
fs.writeFileSync(excalidrawPackage, JSON.stringify(pkg, null, 2), "utf8");
|
|
||||||
|
|
||||||
// update readme
|
|
||||||
const data = fs.readFileSync(`${excalidrawDir}/README_NEXT.md`, "utf8");
|
|
||||||
fs.writeFileSync(`${excalidrawDir}/README.md`, data, "utf8");
|
|
||||||
|
|
||||||
publish();
|
|
||||||
});
|
|
@@ -37,7 +37,6 @@ const crowdinMap = {
|
|||||||
"uk-UA": "en-uk",
|
"uk-UA": "en-uk",
|
||||||
"zh-CN": "en-zhcn",
|
"zh-CN": "en-zhcn",
|
||||||
"zh-TW": "en-zhtw",
|
"zh-TW": "en-zhtw",
|
||||||
"lv-LV": "en-lv",
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const flags = {
|
const flags = {
|
||||||
@@ -75,7 +74,6 @@ const flags = {
|
|||||||
"uk-UA": "🇺🇦",
|
"uk-UA": "🇺🇦",
|
||||||
"zh-CN": "🇨🇳",
|
"zh-CN": "🇨🇳",
|
||||||
"zh-TW": "🇹🇼",
|
"zh-TW": "🇹🇼",
|
||||||
"lv-LV": "🇱🇻",
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const languages = {
|
const languages = {
|
||||||
@@ -113,7 +111,6 @@ const languages = {
|
|||||||
"uk-UA": "Українська",
|
"uk-UA": "Українська",
|
||||||
"zh-CN": "简体中文",
|
"zh-CN": "简体中文",
|
||||||
"zh-TW": "繁體中文",
|
"zh-TW": "繁體中文",
|
||||||
"lv-LV": "Latviešu",
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const percentages = fs.readFileSync(
|
const percentages = fs.readFileSync(
|
||||||
|
@@ -2,20 +2,18 @@ import { register } from "./register";
|
|||||||
import { getSelectedElements } from "../scene";
|
import { getSelectedElements } from "../scene";
|
||||||
import { getNonDeletedElements } from "../element";
|
import { getNonDeletedElements } from "../element";
|
||||||
import { deepCopyElement } from "../element/newElement";
|
import { deepCopyElement } from "../element/newElement";
|
||||||
|
import { Library } from "../data/library";
|
||||||
|
|
||||||
export const actionAddToLibrary = register({
|
export const actionAddToLibrary = register({
|
||||||
name: "addToLibrary",
|
name: "addToLibrary",
|
||||||
perform: (elements, appState, _, app) => {
|
perform: (elements, appState) => {
|
||||||
const selectedElements = getSelectedElements(
|
const selectedElements = getSelectedElements(
|
||||||
getNonDeletedElements(elements),
|
getNonDeletedElements(elements),
|
||||||
appState,
|
appState,
|
||||||
);
|
);
|
||||||
|
|
||||||
app.library.loadLibrary().then((items) => {
|
Library.loadLibrary().then((items) => {
|
||||||
app.library.saveLibrary([
|
Library.saveLibrary([...items, selectedElements.map(deepCopyElement)]);
|
||||||
...items,
|
|
||||||
selectedElements.map(deepCopyElement),
|
|
||||||
]);
|
|
||||||
});
|
});
|
||||||
return false;
|
return false;
|
||||||
},
|
},
|
||||||
|
@@ -3,13 +3,12 @@ import { getDefaultAppState } from "../appState";
|
|||||||
import { ColorPicker } from "../components/ColorPicker";
|
import { ColorPicker } from "../components/ColorPicker";
|
||||||
import { resetZoom, trash, zoomIn, zoomOut } from "../components/icons";
|
import { resetZoom, trash, zoomIn, zoomOut } from "../components/icons";
|
||||||
import { ToolButton } from "../components/ToolButton";
|
import { ToolButton } from "../components/ToolButton";
|
||||||
import { DarkModeToggle } from "../components/DarkModeToggle";
|
|
||||||
import { ZOOM_STEP } from "../constants";
|
import { ZOOM_STEP } from "../constants";
|
||||||
import { getCommonBounds, getNonDeletedElements } from "../element";
|
import { getCommonBounds, getNonDeletedElements } from "../element";
|
||||||
import { newElementWith } from "../element/mutateElement";
|
import { newElementWith } from "../element/mutateElement";
|
||||||
import { ExcalidrawElement } from "../element/types";
|
import { ExcalidrawElement } from "../element/types";
|
||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
import { useIsMobile } from "../components/App";
|
import useIsMobile from "../is-mobile";
|
||||||
import { CODES, KEYS } from "../keys";
|
import { CODES, KEYS } from "../keys";
|
||||||
import { getNormalizedZoom, getSelectedElements } from "../scene";
|
import { getNormalizedZoom, getSelectedElements } from "../scene";
|
||||||
import { centerScrollOn } from "../scene/scroll";
|
import { centerScrollOn } from "../scene/scroll";
|
||||||
@@ -34,7 +33,6 @@ export const actionChangeViewBackgroundColor = register({
|
|||||||
type="canvasBackground"
|
type="canvasBackground"
|
||||||
color={appState.viewBackgroundColor}
|
color={appState.viewBackgroundColor}
|
||||||
onChange={(color) => updateData(color)}
|
onChange={(color) => updateData(color)}
|
||||||
data-testid="canvas-background-picker"
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
@@ -55,6 +53,7 @@ export const actionClearCanvas = register({
|
|||||||
exportBackground: appState.exportBackground,
|
exportBackground: appState.exportBackground,
|
||||||
exportEmbedScene: appState.exportEmbedScene,
|
exportEmbedScene: appState.exportEmbedScene,
|
||||||
gridSize: appState.gridSize,
|
gridSize: appState.gridSize,
|
||||||
|
shouldAddWatermark: appState.shouldAddWatermark,
|
||||||
showStats: appState.showStats,
|
showStats: appState.showStats,
|
||||||
pasteDialog: appState.pasteDialog,
|
pasteDialog: appState.pasteDialog,
|
||||||
},
|
},
|
||||||
@@ -73,7 +72,6 @@ export const actionClearCanvas = register({
|
|||||||
updateData(null);
|
updateData(null);
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
data-testid="clear-canvas-button"
|
|
||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
});
|
});
|
||||||
@@ -260,27 +258,3 @@ export const actionZoomToFit = register({
|
|||||||
!event.altKey &&
|
!event.altKey &&
|
||||||
!event[KEYS.CTRL_OR_CMD],
|
!event[KEYS.CTRL_OR_CMD],
|
||||||
});
|
});
|
||||||
|
|
||||||
export const actionToggleTheme = register({
|
|
||||||
name: "toggleTheme",
|
|
||||||
perform: (_, appState, value) => {
|
|
||||||
return {
|
|
||||||
appState: {
|
|
||||||
...appState,
|
|
||||||
theme: value || (appState.theme === "light" ? "dark" : "light"),
|
|
||||||
},
|
|
||||||
commitToHistory: false,
|
|
||||||
};
|
|
||||||
},
|
|
||||||
PanelComponent: ({ appState, updateData }) => (
|
|
||||||
<div style={{ marginInlineStart: "0.25rem" }}>
|
|
||||||
<DarkModeToggle
|
|
||||||
value={appState.theme}
|
|
||||||
onChange={(theme) => {
|
|
||||||
updateData(theme);
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
),
|
|
||||||
keyTest: (event) => event.altKey && event.shiftKey && event.code === CODES.D,
|
|
||||||
});
|
|
||||||
|
@@ -50,6 +50,7 @@ export const actionCopyAsSvg = register({
|
|||||||
? selectedElements
|
? selectedElements
|
||||||
: getNonDeletedElements(elements),
|
: getNonDeletedElements(elements),
|
||||||
appState,
|
appState,
|
||||||
|
app.canvas,
|
||||||
appState,
|
appState,
|
||||||
);
|
);
|
||||||
return {
|
return {
|
||||||
@@ -88,6 +89,7 @@ export const actionCopyAsPng = register({
|
|||||||
? selectedElements
|
? selectedElements
|
||||||
: getNonDeletedElements(elements),
|
: getNonDeletedElements(elements),
|
||||||
appState,
|
appState,
|
||||||
|
app.canvas,
|
||||||
appState,
|
appState,
|
||||||
);
|
);
|
||||||
return {
|
return {
|
||||||
|
@@ -8,11 +8,9 @@ import { Tooltip } from "../components/Tooltip";
|
|||||||
import { DarkModeToggle, Appearence } from "../components/DarkModeToggle";
|
import { DarkModeToggle, Appearence } from "../components/DarkModeToggle";
|
||||||
import { loadFromJSON, saveAsJSON } from "../data";
|
import { loadFromJSON, saveAsJSON } from "../data";
|
||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
import { useIsMobile } from "../components/App";
|
import useIsMobile from "../is-mobile";
|
||||||
import { KEYS } from "../keys";
|
import { KEYS } from "../keys";
|
||||||
import { register } from "./register";
|
import { register } from "./register";
|
||||||
import { supported as fsSupported } from "browser-fs-access";
|
|
||||||
import { CheckboxItem } from "../components/CheckboxItem";
|
|
||||||
|
|
||||||
export const actionChangeProjectName = register({
|
export const actionChangeProjectName = register({
|
||||||
name: "changeProjectName",
|
name: "changeProjectName",
|
||||||
@@ -25,9 +23,7 @@ export const actionChangeProjectName = register({
|
|||||||
label={t("labels.fileTitle")}
|
label={t("labels.fileTitle")}
|
||||||
value={appState.name || "Unnamed"}
|
value={appState.name || "Unnamed"}
|
||||||
onChange={(name: string) => updateData(name)}
|
onChange={(name: string) => updateData(name)}
|
||||||
isNameEditable={
|
isNameEditable={typeof appProps.name === "undefined"}
|
||||||
typeof appProps.name === "undefined" && !appState.viewModeEnabled
|
|
||||||
}
|
|
||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
});
|
});
|
||||||
@@ -41,12 +37,14 @@ export const actionChangeExportBackground = register({
|
|||||||
};
|
};
|
||||||
},
|
},
|
||||||
PanelComponent: ({ appState, updateData }) => (
|
PanelComponent: ({ appState, updateData }) => (
|
||||||
<CheckboxItem
|
<label>
|
||||||
checked={appState.exportBackground}
|
<input
|
||||||
onChange={(checked) => updateData(checked)}
|
type="checkbox"
|
||||||
>
|
checked={appState.exportBackground}
|
||||||
|
onChange={(event) => updateData(event.target.checked)}
|
||||||
|
/>{" "}
|
||||||
{t("labels.withBackground")}
|
{t("labels.withBackground")}
|
||||||
</CheckboxItem>
|
</label>
|
||||||
),
|
),
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -59,15 +57,41 @@ export const actionChangeExportEmbedScene = register({
|
|||||||
};
|
};
|
||||||
},
|
},
|
||||||
PanelComponent: ({ appState, updateData }) => (
|
PanelComponent: ({ appState, updateData }) => (
|
||||||
<CheckboxItem
|
<label style={{ display: "flex" }}>
|
||||||
checked={appState.exportEmbedScene}
|
<input
|
||||||
onChange={(checked) => updateData(checked)}
|
type="checkbox"
|
||||||
>
|
checked={appState.exportEmbedScene}
|
||||||
|
onChange={(event) => updateData(event.target.checked)}
|
||||||
|
/>{" "}
|
||||||
{t("labels.exportEmbedScene")}
|
{t("labels.exportEmbedScene")}
|
||||||
<Tooltip label={t("labels.exportEmbedScene_details")} long={true}>
|
<Tooltip
|
||||||
<div className="Tooltip-icon">{questionCircle}</div>
|
label={t("labels.exportEmbedScene_details")}
|
||||||
|
position="above"
|
||||||
|
long={true}
|
||||||
|
>
|
||||||
|
<div className="TooltipIcon">{questionCircle}</div>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</CheckboxItem>
|
</label>
|
||||||
|
),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const actionChangeShouldAddWatermark = register({
|
||||||
|
name: "changeShouldAddWatermark",
|
||||||
|
perform: (_elements, appState, value) => {
|
||||||
|
return {
|
||||||
|
appState: { ...appState, shouldAddWatermark: value },
|
||||||
|
commitToHistory: false,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
PanelComponent: ({ appState, updateData }) => (
|
||||||
|
<label>
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
checked={appState.shouldAddWatermark}
|
||||||
|
onChange={(event) => updateData(event.target.checked)}
|
||||||
|
/>{" "}
|
||||||
|
{t("labels.addWatermark")}
|
||||||
|
</label>
|
||||||
),
|
),
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -103,12 +127,12 @@ export const actionSaveScene = register({
|
|||||||
event.key === KEYS.S && event[KEYS.CTRL_OR_CMD] && !event.shiftKey,
|
event.key === KEYS.S && event[KEYS.CTRL_OR_CMD] && !event.shiftKey,
|
||||||
PanelComponent: ({ updateData }) => (
|
PanelComponent: ({ updateData }) => (
|
||||||
<ToolButton
|
<ToolButton
|
||||||
type="icon"
|
type="button"
|
||||||
icon={save}
|
icon={save}
|
||||||
title={t("buttons.save")}
|
title={t("buttons.save")}
|
||||||
aria-label={t("buttons.save")}
|
aria-label={t("buttons.save")}
|
||||||
|
showAriaLabel={useIsMobile()}
|
||||||
onClick={() => updateData(null)}
|
onClick={() => updateData(null)}
|
||||||
data-testid="save-button"
|
|
||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
});
|
});
|
||||||
@@ -138,9 +162,10 @@ export const actionSaveAsScene = register({
|
|||||||
title={t("buttons.saveAs")}
|
title={t("buttons.saveAs")}
|
||||||
aria-label={t("buttons.saveAs")}
|
aria-label={t("buttons.saveAs")}
|
||||||
showAriaLabel={useIsMobile()}
|
showAriaLabel={useIsMobile()}
|
||||||
hidden={!fsSupported}
|
hidden={
|
||||||
|
!("chooseFileSystemEntries" in window || "showOpenFilePicker" in window)
|
||||||
|
}
|
||||||
onClick={() => updateData(null)}
|
onClick={() => updateData(null)}
|
||||||
data-testid="save-as-button"
|
|
||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
});
|
});
|
||||||
@@ -178,7 +203,6 @@ export const actionLoadScene = register({
|
|||||||
aria-label={t("buttons.load")}
|
aria-label={t("buttons.load")}
|
||||||
showAriaLabel={useIsMobile()}
|
showAriaLabel={useIsMobile()}
|
||||||
onClick={updateData}
|
onClick={updateData}
|
||||||
data-testid="load-button"
|
|
||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
});
|
});
|
||||||
|
@@ -18,7 +18,7 @@ import { isBindingElement } from "../element/typeChecks";
|
|||||||
|
|
||||||
export const actionFinalize = register({
|
export const actionFinalize = register({
|
||||||
name: "finalize",
|
name: "finalize",
|
||||||
perform: (elements, appState, _, { canvas, focusContainer }) => {
|
perform: (elements, appState, _, { canvas }) => {
|
||||||
if (appState.editingLinearElement) {
|
if (appState.editingLinearElement) {
|
||||||
const {
|
const {
|
||||||
elementId,
|
elementId,
|
||||||
@@ -51,19 +51,19 @@ export const actionFinalize = register({
|
|||||||
|
|
||||||
let newElements = elements;
|
let newElements = elements;
|
||||||
if (window.document.activeElement instanceof HTMLElement) {
|
if (window.document.activeElement instanceof HTMLElement) {
|
||||||
focusContainer();
|
window.document.activeElement.blur();
|
||||||
}
|
}
|
||||||
|
|
||||||
const multiPointElement = appState.multiElement
|
const multiPointElement = appState.multiElement
|
||||||
? appState.multiElement
|
? appState.multiElement
|
||||||
: appState.editingElement?.type === "freedraw"
|
: appState.editingElement?.type === "draw"
|
||||||
? appState.editingElement
|
? appState.editingElement
|
||||||
: null;
|
: null;
|
||||||
|
|
||||||
if (multiPointElement) {
|
if (multiPointElement) {
|
||||||
// pen and mouse have hover
|
// pen and mouse have hover
|
||||||
if (
|
if (
|
||||||
multiPointElement.type !== "freedraw" &&
|
multiPointElement.type !== "draw" &&
|
||||||
appState.lastPointerDownWith !== "touch"
|
appState.lastPointerDownWith !== "touch"
|
||||||
) {
|
) {
|
||||||
const { points, lastCommittedPoint } = multiPointElement;
|
const { points, lastCommittedPoint } = multiPointElement;
|
||||||
@@ -86,7 +86,7 @@ export const actionFinalize = register({
|
|||||||
const isLoop = isPathALoop(multiPointElement.points, appState.zoom.value);
|
const isLoop = isPathALoop(multiPointElement.points, appState.zoom.value);
|
||||||
if (
|
if (
|
||||||
multiPointElement.type === "line" ||
|
multiPointElement.type === "line" ||
|
||||||
multiPointElement.type === "freedraw"
|
multiPointElement.type === "draw"
|
||||||
) {
|
) {
|
||||||
if (isLoop) {
|
if (isLoop) {
|
||||||
const linePoints = multiPointElement.points;
|
const linePoints = multiPointElement.points;
|
||||||
@@ -118,24 +118,22 @@ export const actionFinalize = register({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!appState.elementLocked && appState.elementType !== "freedraw") {
|
if (!appState.elementLocked && appState.elementType !== "draw") {
|
||||||
appState.selectedElementIds[multiPointElement.id] = true;
|
appState.selectedElementIds[multiPointElement.id] = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
if (
|
||||||
(!appState.elementLocked && appState.elementType !== "freedraw") ||
|
(!appState.elementLocked && appState.elementType !== "draw") ||
|
||||||
!multiPointElement
|
!multiPointElement
|
||||||
) {
|
) {
|
||||||
resetCursor(canvas);
|
resetCursor(canvas);
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
elements: newElements,
|
elements: newElements,
|
||||||
appState: {
|
appState: {
|
||||||
...appState,
|
...appState,
|
||||||
elementType:
|
elementType:
|
||||||
(appState.elementLocked || appState.elementType === "freedraw") &&
|
(appState.elementLocked || appState.elementType === "draw") &&
|
||||||
multiPointElement
|
multiPointElement
|
||||||
? appState.elementType
|
? appState.elementType
|
||||||
: "selection",
|
: "selection",
|
||||||
@@ -147,14 +145,14 @@ export const actionFinalize = register({
|
|||||||
selectedElementIds:
|
selectedElementIds:
|
||||||
multiPointElement &&
|
multiPointElement &&
|
||||||
!appState.elementLocked &&
|
!appState.elementLocked &&
|
||||||
appState.elementType !== "freedraw"
|
appState.elementType !== "draw"
|
||||||
? {
|
? {
|
||||||
...appState.selectedElementIds,
|
...appState.selectedElementIds,
|
||||||
[multiPointElement.id]: true,
|
[multiPointElement.id]: true,
|
||||||
}
|
}
|
||||||
: appState.selectedElementIds,
|
: appState.selectedElementIds,
|
||||||
},
|
},
|
||||||
commitToHistory: appState.elementType === "freedraw",
|
commitToHistory: appState.elementType === "draw",
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
keyTest: (event, appState) =>
|
keyTest: (event, appState) =>
|
||||||
|
@@ -1,207 +0,0 @@
|
|||||||
import { register } from "./register";
|
|
||||||
import { getSelectedElements } from "../scene";
|
|
||||||
import { getElementMap, getNonDeletedElements } from "../element";
|
|
||||||
import { mutateElement } from "../element/mutateElement";
|
|
||||||
import { ExcalidrawElement, NonDeleted } from "../element/types";
|
|
||||||
import { normalizeAngle, resizeSingleElement } from "../element/resizeElements";
|
|
||||||
import { AppState } from "../types";
|
|
||||||
import { getTransformHandles } from "../element/transformHandles";
|
|
||||||
import { isFreeDrawElement, isLinearElement } from "../element/typeChecks";
|
|
||||||
import { updateBoundElements } from "../element/binding";
|
|
||||||
import { LinearElementEditor } from "../element/linearElementEditor";
|
|
||||||
|
|
||||||
const enableActionFlipHorizontal = (
|
|
||||||
elements: readonly ExcalidrawElement[],
|
|
||||||
appState: AppState,
|
|
||||||
) => {
|
|
||||||
const eligibleElements = getSelectedElements(
|
|
||||||
getNonDeletedElements(elements),
|
|
||||||
appState,
|
|
||||||
);
|
|
||||||
return eligibleElements.length === 1 && eligibleElements[0].type !== "text";
|
|
||||||
};
|
|
||||||
|
|
||||||
const enableActionFlipVertical = (
|
|
||||||
elements: readonly ExcalidrawElement[],
|
|
||||||
appState: AppState,
|
|
||||||
) => {
|
|
||||||
const eligibleElements = getSelectedElements(
|
|
||||||
getNonDeletedElements(elements),
|
|
||||||
appState,
|
|
||||||
);
|
|
||||||
return eligibleElements.length === 1;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const actionFlipHorizontal = register({
|
|
||||||
name: "flipHorizontal",
|
|
||||||
perform: (elements, appState) => {
|
|
||||||
return {
|
|
||||||
elements: flipSelectedElements(elements, appState, "horizontal"),
|
|
||||||
appState,
|
|
||||||
commitToHistory: true,
|
|
||||||
};
|
|
||||||
},
|
|
||||||
keyTest: (event) => event.shiftKey && event.code === "KeyH",
|
|
||||||
contextItemLabel: "labels.flipHorizontal",
|
|
||||||
contextItemPredicate: (elements, appState) =>
|
|
||||||
enableActionFlipHorizontal(elements, appState),
|
|
||||||
});
|
|
||||||
|
|
||||||
export const actionFlipVertical = register({
|
|
||||||
name: "flipVertical",
|
|
||||||
perform: (elements, appState) => {
|
|
||||||
return {
|
|
||||||
elements: flipSelectedElements(elements, appState, "vertical"),
|
|
||||||
appState,
|
|
||||||
commitToHistory: true,
|
|
||||||
};
|
|
||||||
},
|
|
||||||
keyTest: (event) => event.shiftKey && event.code === "KeyV",
|
|
||||||
contextItemLabel: "labels.flipVertical",
|
|
||||||
contextItemPredicate: (elements, appState) =>
|
|
||||||
enableActionFlipVertical(elements, appState),
|
|
||||||
});
|
|
||||||
|
|
||||||
const flipSelectedElements = (
|
|
||||||
elements: readonly ExcalidrawElement[],
|
|
||||||
appState: Readonly<AppState>,
|
|
||||||
flipDirection: "horizontal" | "vertical",
|
|
||||||
) => {
|
|
||||||
const selectedElements = getSelectedElements(
|
|
||||||
getNonDeletedElements(elements),
|
|
||||||
appState,
|
|
||||||
);
|
|
||||||
|
|
||||||
// remove once we allow for groups of elements to be flipped
|
|
||||||
if (selectedElements.length > 1) {
|
|
||||||
return elements;
|
|
||||||
}
|
|
||||||
|
|
||||||
const updatedElements = flipElements(
|
|
||||||
selectedElements,
|
|
||||||
appState,
|
|
||||||
flipDirection,
|
|
||||||
);
|
|
||||||
|
|
||||||
const updatedElementsMap = getElementMap(updatedElements);
|
|
||||||
|
|
||||||
return elements.map((element) => updatedElementsMap[element.id] || element);
|
|
||||||
};
|
|
||||||
|
|
||||||
const flipElements = (
|
|
||||||
elements: NonDeleted<ExcalidrawElement>[],
|
|
||||||
appState: AppState,
|
|
||||||
flipDirection: "horizontal" | "vertical",
|
|
||||||
): ExcalidrawElement[] => {
|
|
||||||
for (let i = 0; i < elements.length; i++) {
|
|
||||||
flipElement(elements[i], appState);
|
|
||||||
// If vertical flip, rotate an extra 180
|
|
||||||
if (flipDirection === "vertical") {
|
|
||||||
rotateElement(elements[i], Math.PI);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return elements;
|
|
||||||
};
|
|
||||||
|
|
||||||
const flipElement = (
|
|
||||||
element: NonDeleted<ExcalidrawElement>,
|
|
||||||
appState: AppState,
|
|
||||||
) => {
|
|
||||||
const originalX = element.x;
|
|
||||||
const originalY = element.y;
|
|
||||||
const width = element.width;
|
|
||||||
const height = element.height;
|
|
||||||
const originalAngle = normalizeAngle(element.angle);
|
|
||||||
|
|
||||||
let finalOffsetX = 0;
|
|
||||||
if (isLinearElement(element) || isFreeDrawElement(element)) {
|
|
||||||
finalOffsetX =
|
|
||||||
element.points.reduce((max, point) => Math.max(max, point[0]), 0) * 2 -
|
|
||||||
element.width;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Rotate back to zero, if necessary
|
|
||||||
mutateElement(element, {
|
|
||||||
angle: normalizeAngle(0),
|
|
||||||
});
|
|
||||||
// Flip unrotated by pulling TransformHandle to opposite side
|
|
||||||
const transformHandles = getTransformHandles(element, appState.zoom);
|
|
||||||
let usingNWHandle = true;
|
|
||||||
let newNCoordsX = 0;
|
|
||||||
let nHandle = transformHandles.nw;
|
|
||||||
if (!nHandle) {
|
|
||||||
// Use ne handle instead
|
|
||||||
usingNWHandle = false;
|
|
||||||
nHandle = transformHandles.ne;
|
|
||||||
if (!nHandle) {
|
|
||||||
mutateElement(element, {
|
|
||||||
angle: originalAngle,
|
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isLinearElement(element)) {
|
|
||||||
for (let i = 1; i < element.points.length; i++) {
|
|
||||||
LinearElementEditor.movePoint(element, i, [
|
|
||||||
-element.points[i][0],
|
|
||||||
element.points[i][1],
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
LinearElementEditor.normalizePoints(element);
|
|
||||||
} else {
|
|
||||||
// calculate new x-coord for transformation
|
|
||||||
newNCoordsX = usingNWHandle ? element.x + 2 * width : element.x - 2 * width;
|
|
||||||
resizeSingleElement(
|
|
||||||
element,
|
|
||||||
true,
|
|
||||||
element,
|
|
||||||
usingNWHandle ? "nw" : "ne",
|
|
||||||
false,
|
|
||||||
newNCoordsX,
|
|
||||||
nHandle[1],
|
|
||||||
);
|
|
||||||
// fix the size to account for handle sizes
|
|
||||||
mutateElement(element, {
|
|
||||||
width,
|
|
||||||
height,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Rotate by (360 degrees - original angle)
|
|
||||||
let angle = normalizeAngle(2 * Math.PI - originalAngle);
|
|
||||||
if (angle < 0) {
|
|
||||||
// check, probably unnecessary
|
|
||||||
angle = normalizeAngle(angle + 2 * Math.PI);
|
|
||||||
}
|
|
||||||
mutateElement(element, {
|
|
||||||
angle,
|
|
||||||
});
|
|
||||||
|
|
||||||
// Move back to original spot to appear "flipped in place"
|
|
||||||
mutateElement(element, {
|
|
||||||
x: originalX + finalOffsetX,
|
|
||||||
y: originalY,
|
|
||||||
});
|
|
||||||
|
|
||||||
updateBoundElements(element);
|
|
||||||
};
|
|
||||||
|
|
||||||
const rotateElement = (element: ExcalidrawElement, rotationAngle: number) => {
|
|
||||||
const originalX = element.x;
|
|
||||||
const originalY = element.y;
|
|
||||||
let angle = normalizeAngle(element.angle + rotationAngle);
|
|
||||||
if (angle < 0) {
|
|
||||||
// check, probably unnecessary
|
|
||||||
angle = normalizeAngle(2 * Math.PI + angle);
|
|
||||||
}
|
|
||||||
mutateElement(element, {
|
|
||||||
angle,
|
|
||||||
});
|
|
||||||
|
|
||||||
// Move back to original spot
|
|
||||||
mutateElement(element, {
|
|
||||||
x: originalX,
|
|
||||||
y: originalY,
|
|
||||||
});
|
|
||||||
};
|
|
@@ -3,7 +3,7 @@ import React from "react";
|
|||||||
import { undo, redo } from "../components/icons";
|
import { undo, redo } from "../components/icons";
|
||||||
import { ToolButton } from "../components/ToolButton";
|
import { ToolButton } from "../components/ToolButton";
|
||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
import History, { HistoryEntry } from "../history";
|
import { SceneHistory, HistoryEntry } from "../history";
|
||||||
import { ExcalidrawElement } from "../element/types";
|
import { ExcalidrawElement } from "../element/types";
|
||||||
import { AppState } from "../types";
|
import { AppState } from "../types";
|
||||||
import { isWindows, KEYS } from "../keys";
|
import { isWindows, KEYS } from "../keys";
|
||||||
@@ -59,7 +59,7 @@ const writeData = (
|
|||||||
return { commitToHistory };
|
return { commitToHistory };
|
||||||
};
|
};
|
||||||
|
|
||||||
type ActionCreator = (history: History) => Action;
|
type ActionCreator = (history: SceneHistory) => Action;
|
||||||
|
|
||||||
export const createUndoAction: ActionCreator = (history) => ({
|
export const createUndoAction: ActionCreator = (history) => ({
|
||||||
name: "undo",
|
name: "undo",
|
||||||
|
@@ -70,10 +70,7 @@ export const actionFullScreen = register({
|
|||||||
|
|
||||||
export const actionShortcuts = register({
|
export const actionShortcuts = register({
|
||||||
name: "toggleShortcuts",
|
name: "toggleShortcuts",
|
||||||
perform: (_elements, appState, _, { focusContainer }) => {
|
perform: (_elements, appState) => {
|
||||||
if (appState.showHelpDialog) {
|
|
||||||
focusContainer();
|
|
||||||
}
|
|
||||||
return {
|
return {
|
||||||
appState: {
|
appState: {
|
||||||
...appState,
|
...appState,
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import { AppState } from "../../src/types";
|
import { AppState } from "../../src/types";
|
||||||
import { ButtonIconSelect } from "../components/ButtonIconSelect";
|
import { ButtonIconSelect } from "../components/ButtonIconSelect";
|
||||||
|
import { ButtonSelect } from "../components/ButtonSelect";
|
||||||
import { ColorPicker } from "../components/ColorPicker";
|
import { ColorPicker } from "../components/ColorPicker";
|
||||||
import { IconPicker } from "../components/IconPicker";
|
import { IconPicker } from "../components/IconPicker";
|
||||||
import {
|
import {
|
||||||
@@ -20,16 +21,6 @@ import {
|
|||||||
StrokeStyleDottedIcon,
|
StrokeStyleDottedIcon,
|
||||||
StrokeStyleSolidIcon,
|
StrokeStyleSolidIcon,
|
||||||
StrokeWidthIcon,
|
StrokeWidthIcon,
|
||||||
FontSizeSmallIcon,
|
|
||||||
FontSizeMediumIcon,
|
|
||||||
FontSizeLargeIcon,
|
|
||||||
FontSizeExtraLargeIcon,
|
|
||||||
FontFamilyHandDrawnIcon,
|
|
||||||
FontFamilyNormalIcon,
|
|
||||||
FontFamilyCodeIcon,
|
|
||||||
TextAlignLeftIcon,
|
|
||||||
TextAlignCenterIcon,
|
|
||||||
TextAlignRightIcon,
|
|
||||||
} from "../components/icons";
|
} from "../components/icons";
|
||||||
import { DEFAULT_FONT_FAMILY, DEFAULT_FONT_SIZE } from "../constants";
|
import { DEFAULT_FONT_FAMILY, DEFAULT_FONT_SIZE } from "../constants";
|
||||||
import {
|
import {
|
||||||
@@ -422,29 +413,13 @@ export const actionChangeFontSize = register({
|
|||||||
PanelComponent: ({ elements, appState, updateData }) => (
|
PanelComponent: ({ elements, appState, updateData }) => (
|
||||||
<fieldset>
|
<fieldset>
|
||||||
<legend>{t("labels.fontSize")}</legend>
|
<legend>{t("labels.fontSize")}</legend>
|
||||||
<ButtonIconSelect
|
<ButtonSelect
|
||||||
group="font-size"
|
group="font-size"
|
||||||
options={[
|
options={[
|
||||||
{
|
{ value: 16, text: t("labels.small") },
|
||||||
value: 16,
|
{ value: 20, text: t("labels.medium") },
|
||||||
text: t("labels.small"),
|
{ value: 28, text: t("labels.large") },
|
||||||
icon: <FontSizeSmallIcon theme={appState.theme} />,
|
{ value: 36, text: t("labels.veryLarge") },
|
||||||
},
|
|
||||||
{
|
|
||||||
value: 20,
|
|
||||||
text: t("labels.medium"),
|
|
||||||
icon: <FontSizeMediumIcon theme={appState.theme} />,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
value: 28,
|
|
||||||
text: t("labels.large"),
|
|
||||||
icon: <FontSizeLargeIcon theme={appState.theme} />,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
value: 36,
|
|
||||||
text: t("labels.veryLarge"),
|
|
||||||
icon: <FontSizeExtraLargeIcon theme={appState.theme} />,
|
|
||||||
},
|
|
||||||
]}
|
]}
|
||||||
value={getFormValue(
|
value={getFormValue(
|
||||||
elements,
|
elements,
|
||||||
@@ -481,28 +456,16 @@ export const actionChangeFontFamily = register({
|
|||||||
};
|
};
|
||||||
},
|
},
|
||||||
PanelComponent: ({ elements, appState, updateData }) => {
|
PanelComponent: ({ elements, appState, updateData }) => {
|
||||||
const options: { value: FontFamily; text: string; icon: JSX.Element }[] = [
|
const options: { value: FontFamily; text: string }[] = [
|
||||||
{
|
{ value: 1, text: t("labels.handDrawn") },
|
||||||
value: 1,
|
{ value: 2, text: t("labels.normal") },
|
||||||
text: t("labels.handDrawn"),
|
{ value: 3, text: t("labels.code") },
|
||||||
icon: <FontFamilyHandDrawnIcon theme={appState.theme} />,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
value: 2,
|
|
||||||
text: t("labels.normal"),
|
|
||||||
icon: <FontFamilyNormalIcon theme={appState.theme} />,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
value: 3,
|
|
||||||
text: t("labels.code"),
|
|
||||||
icon: <FontFamilyCodeIcon theme={appState.theme} />,
|
|
||||||
},
|
|
||||||
];
|
];
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<fieldset>
|
<fieldset>
|
||||||
<legend>{t("labels.fontFamily")}</legend>
|
<legend>{t("labels.fontFamily")}</legend>
|
||||||
<ButtonIconSelect<FontFamily | false>
|
<ButtonSelect<FontFamily | false>
|
||||||
group="font-family"
|
group="font-family"
|
||||||
options={options}
|
options={options}
|
||||||
value={getFormValue(
|
value={getFormValue(
|
||||||
@@ -543,24 +506,12 @@ export const actionChangeTextAlign = register({
|
|||||||
PanelComponent: ({ elements, appState, updateData }) => (
|
PanelComponent: ({ elements, appState, updateData }) => (
|
||||||
<fieldset>
|
<fieldset>
|
||||||
<legend>{t("labels.textAlign")}</legend>
|
<legend>{t("labels.textAlign")}</legend>
|
||||||
<ButtonIconSelect<TextAlign | false>
|
<ButtonSelect<TextAlign | false>
|
||||||
group="text-align"
|
group="text-align"
|
||||||
options={[
|
options={[
|
||||||
{
|
{ value: "left", text: t("labels.left") },
|
||||||
value: "left",
|
{ value: "center", text: t("labels.center") },
|
||||||
text: t("labels.left"),
|
{ value: "right", text: t("labels.right") },
|
||||||
icon: <TextAlignLeftIcon theme={appState.theme} />,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
value: "center",
|
|
||||||
text: t("labels.center"),
|
|
||||||
icon: <TextAlignCenterIcon theme={appState.theme} />,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
value: "right",
|
|
||||||
text: t("labels.right"),
|
|
||||||
icon: <TextAlignRightIcon theme={appState.theme} />,
|
|
||||||
},
|
|
||||||
]}
|
]}
|
||||||
value={getFormValue(
|
value={getFormValue(
|
||||||
elements,
|
elements,
|
||||||
|
@@ -1,5 +1,4 @@
|
|||||||
import { register } from "./register";
|
import { register } from "./register";
|
||||||
import { CODES, KEYS } from "../keys";
|
|
||||||
|
|
||||||
export const actionToggleStats = register({
|
export const actionToggleStats = register({
|
||||||
name: "stats",
|
name: "stats",
|
||||||
@@ -14,6 +13,4 @@ export const actionToggleStats = register({
|
|||||||
},
|
},
|
||||||
checked: (appState) => appState.showStats,
|
checked: (appState) => appState.showStats,
|
||||||
contextItemLabel: "stats.title",
|
contextItemLabel: "stats.title",
|
||||||
keyTest: (event) =>
|
|
||||||
!event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.SLASH,
|
|
||||||
});
|
});
|
||||||
|
@@ -26,7 +26,6 @@ export {
|
|||||||
actionZoomOut,
|
actionZoomOut,
|
||||||
actionResetZoom,
|
actionResetZoom,
|
||||||
actionZoomToFit,
|
actionZoomToFit,
|
||||||
actionToggleTheme,
|
|
||||||
} from "./actionCanvas";
|
} from "./actionCanvas";
|
||||||
|
|
||||||
export { actionFinalize } from "./actionFinalize";
|
export { actionFinalize } from "./actionFinalize";
|
||||||
@@ -67,8 +66,6 @@ export {
|
|||||||
distributeVertically,
|
distributeVertically,
|
||||||
} from "./actionDistribute";
|
} from "./actionDistribute";
|
||||||
|
|
||||||
export { actionFlipHorizontal, actionFlipVertical } from "./actionFlip";
|
|
||||||
|
|
||||||
export {
|
export {
|
||||||
actionCopy,
|
actionCopy,
|
||||||
actionCut,
|
actionCut,
|
||||||
|
@@ -7,18 +7,12 @@ import {
|
|||||||
ActionResult,
|
ActionResult,
|
||||||
} from "./types";
|
} from "./types";
|
||||||
import { ExcalidrawElement } from "../element/types";
|
import { ExcalidrawElement } from "../element/types";
|
||||||
import { AppProps, AppState } from "../types";
|
import { AppState, ExcalidrawProps } from "../types";
|
||||||
import { MODES } from "../constants";
|
import { MODES } from "../constants";
|
||||||
import Library from "../data/library";
|
|
||||||
|
|
||||||
// This is the <App> component, but for now we don't care about anything but its
|
// This is the <App> component, but for now we don't care about anything but its
|
||||||
// `canvas` state.
|
// `canvas` state.
|
||||||
type App = {
|
type App = { canvas: HTMLCanvasElement | null; props: ExcalidrawProps };
|
||||||
canvas: HTMLCanvasElement | null;
|
|
||||||
focusContainer: () => void;
|
|
||||||
props: AppProps;
|
|
||||||
library: Library;
|
|
||||||
};
|
|
||||||
|
|
||||||
export class ActionManager implements ActionsManagerInterface {
|
export class ActionManager implements ActionsManagerInterface {
|
||||||
actions = {} as ActionsManagerInterface["actions"];
|
actions = {} as ActionsManagerInterface["actions"];
|
||||||
@@ -57,15 +51,11 @@ export class ActionManager implements ActionsManagerInterface {
|
|||||||
actions.forEach((action) => this.registerAction(action));
|
actions.forEach((action) => this.registerAction(action));
|
||||||
}
|
}
|
||||||
|
|
||||||
handleKeyDown(event: React.KeyboardEvent | KeyboardEvent) {
|
handleKeyDown(event: KeyboardEvent) {
|
||||||
const canvasActions = this.app.props.UIOptions.canvasActions;
|
|
||||||
const data = Object.values(this.actions)
|
const data = Object.values(this.actions)
|
||||||
.sort((a, b) => (b.keyPriority || 0) - (a.keyPriority || 0))
|
.sort((a, b) => (b.keyPriority || 0) - (a.keyPriority || 0))
|
||||||
.filter(
|
.filter(
|
||||||
(action) =>
|
(action) =>
|
||||||
(action.name in canvasActions
|
|
||||||
? canvasActions[action.name as keyof typeof canvasActions]
|
|
||||||
: true) &&
|
|
||||||
action.keyTest &&
|
action.keyTest &&
|
||||||
action.keyTest(
|
action.keyTest(
|
||||||
event,
|
event,
|
||||||
@@ -112,15 +102,7 @@ export class ActionManager implements ActionsManagerInterface {
|
|||||||
// like the user list. We can use this key to extract more
|
// like the user list. We can use this key to extract more
|
||||||
// data from app state. This is an alternative to generic prop hell!
|
// data from app state. This is an alternative to generic prop hell!
|
||||||
renderAction = (name: ActionName, id?: string) => {
|
renderAction = (name: ActionName, id?: string) => {
|
||||||
const canvasActions = this.app.props.UIOptions.canvasActions;
|
if (this.actions[name] && "PanelComponent" in this.actions[name]) {
|
||||||
|
|
||||||
if (
|
|
||||||
this.actions[name] &&
|
|
||||||
"PanelComponent" in this.actions[name] &&
|
|
||||||
(name in canvasActions
|
|
||||||
? canvasActions[name as keyof typeof canvasActions]
|
|
||||||
: true)
|
|
||||||
) {
|
|
||||||
const action = this.actions[name];
|
const action = this.actions[name];
|
||||||
const PanelComponent = action.PanelComponent!;
|
const PanelComponent = action.PanelComponent!;
|
||||||
const updateData = (formState?: any) => {
|
const updateData = (formState?: any) => {
|
||||||
|
@@ -23,9 +23,7 @@ export type ShortcutName =
|
|||||||
| "zenMode"
|
| "zenMode"
|
||||||
| "stats"
|
| "stats"
|
||||||
| "addToLibrary"
|
| "addToLibrary"
|
||||||
| "viewMode"
|
| "viewMode";
|
||||||
| "flipHorizontal"
|
|
||||||
| "flipVertical";
|
|
||||||
|
|
||||||
const shortcutMap: Record<ShortcutName, string[]> = {
|
const shortcutMap: Record<ShortcutName, string[]> = {
|
||||||
cut: [getShortcutKey("CtrlOrCmd+X")],
|
cut: [getShortcutKey("CtrlOrCmd+X")],
|
||||||
@@ -57,10 +55,8 @@ const shortcutMap: Record<ShortcutName, string[]> = {
|
|||||||
ungroup: [getShortcutKey("CtrlOrCmd+Shift+G")],
|
ungroup: [getShortcutKey("CtrlOrCmd+Shift+G")],
|
||||||
gridMode: [getShortcutKey("CtrlOrCmd+'")],
|
gridMode: [getShortcutKey("CtrlOrCmd+'")],
|
||||||
zenMode: [getShortcutKey("Alt+Z")],
|
zenMode: [getShortcutKey("Alt+Z")],
|
||||||
stats: [getShortcutKey("Alt+/")],
|
stats: [],
|
||||||
addToLibrary: [],
|
addToLibrary: [],
|
||||||
flipHorizontal: [getShortcutKey("Shift+H")],
|
|
||||||
flipVertical: [getShortcutKey("Shift+V")],
|
|
||||||
viewMode: [getShortcutKey("Alt+R")],
|
viewMode: [getShortcutKey("Alt+R")],
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@@ -1,32 +1,22 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import { ExcalidrawElement } from "../element/types";
|
import { ExcalidrawElement } from "../element/types";
|
||||||
import { AppState, ExcalidrawProps } from "../types";
|
import { AppState, ExcalidrawProps } from "../types";
|
||||||
import Library from "../data/library";
|
|
||||||
|
|
||||||
/** if false, the action should be prevented */
|
/** if false, the action should be prevented */
|
||||||
export type ActionResult =
|
export type ActionResult =
|
||||||
| {
|
| {
|
||||||
elements?: readonly ExcalidrawElement[] | null;
|
elements?: readonly ExcalidrawElement[] | null;
|
||||||
appState?: MarkOptional<
|
appState?: MarkOptional<AppState, "offsetTop" | "offsetLeft"> | null;
|
||||||
AppState,
|
|
||||||
"offsetTop" | "offsetLeft" | "width" | "height"
|
|
||||||
> | null;
|
|
||||||
commitToHistory: boolean;
|
commitToHistory: boolean;
|
||||||
syncHistory?: boolean;
|
syncHistory?: boolean;
|
||||||
}
|
}
|
||||||
| false;
|
| false;
|
||||||
|
|
||||||
type AppAPI = {
|
|
||||||
canvas: HTMLCanvasElement | null;
|
|
||||||
focusContainer(): void;
|
|
||||||
library: Library;
|
|
||||||
};
|
|
||||||
|
|
||||||
type ActionFn = (
|
type ActionFn = (
|
||||||
elements: readonly ExcalidrawElement[],
|
elements: readonly ExcalidrawElement[],
|
||||||
appState: Readonly<AppState>,
|
appState: Readonly<AppState>,
|
||||||
formData: any,
|
formData: any,
|
||||||
app: AppAPI,
|
app: { canvas: HTMLCanvasElement | null },
|
||||||
) => ActionResult | Promise<ActionResult>;
|
) => ActionResult | Promise<ActionResult>;
|
||||||
|
|
||||||
export type UpdaterFn = (res: ActionResult) => void;
|
export type UpdaterFn = (res: ActionResult) => void;
|
||||||
@@ -52,7 +42,6 @@ export type ActionName =
|
|||||||
| "changeBackgroundColor"
|
| "changeBackgroundColor"
|
||||||
| "changeFillStyle"
|
| "changeFillStyle"
|
||||||
| "changeStrokeWidth"
|
| "changeStrokeWidth"
|
||||||
| "changeStrokeShape"
|
|
||||||
| "changeSloppiness"
|
| "changeSloppiness"
|
||||||
| "changeStrokeStyle"
|
| "changeStrokeStyle"
|
||||||
| "changeArrowhead"
|
| "changeArrowhead"
|
||||||
@@ -66,6 +55,7 @@ export type ActionName =
|
|||||||
| "changeProjectName"
|
| "changeProjectName"
|
||||||
| "changeExportBackground"
|
| "changeExportBackground"
|
||||||
| "changeExportEmbedScene"
|
| "changeExportEmbedScene"
|
||||||
|
| "changeShouldAddWatermark"
|
||||||
| "saveScene"
|
| "saveScene"
|
||||||
| "saveAsScene"
|
| "saveAsScene"
|
||||||
| "loadScene"
|
| "loadScene"
|
||||||
@@ -95,11 +85,8 @@ export type ActionName =
|
|||||||
| "alignHorizontallyCentered"
|
| "alignHorizontallyCentered"
|
||||||
| "distributeHorizontally"
|
| "distributeHorizontally"
|
||||||
| "distributeVertically"
|
| "distributeVertically"
|
||||||
| "flipHorizontal"
|
|
||||||
| "flipVertical"
|
|
||||||
| "viewMode"
|
| "viewMode"
|
||||||
| "exportWithDarkMode"
|
| "exportWithDarkMode";
|
||||||
| "toggleTheme";
|
|
||||||
|
|
||||||
export interface Action {
|
export interface Action {
|
||||||
name: ActionName;
|
name: ActionName;
|
||||||
@@ -113,7 +100,7 @@ export interface Action {
|
|||||||
perform: ActionFn;
|
perform: ActionFn;
|
||||||
keyPriority?: number;
|
keyPriority?: number;
|
||||||
keyTest?: (
|
keyTest?: (
|
||||||
event: React.KeyboardEvent | KeyboardEvent,
|
event: KeyboardEvent,
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
elements: readonly ExcalidrawElement[],
|
elements: readonly ExcalidrawElement[],
|
||||||
) => boolean;
|
) => boolean;
|
||||||
@@ -128,7 +115,6 @@ export interface Action {
|
|||||||
export interface ActionsManagerInterface {
|
export interface ActionsManagerInterface {
|
||||||
actions: Record<ActionName, Action>;
|
actions: Record<ActionName, Action>;
|
||||||
registerAction: (action: Action) => void;
|
registerAction: (action: Action) => void;
|
||||||
handleKeyDown: (event: React.KeyboardEvent | KeyboardEvent) => boolean;
|
handleKeyDown: (event: KeyboardEvent) => boolean;
|
||||||
renderAction: (name: ActionName) => React.ReactElement | null;
|
renderAction: (name: ActionName) => React.ReactElement | null;
|
||||||
executeAction: (action: Action) => void;
|
|
||||||
}
|
}
|
||||||
|
@@ -10,7 +10,7 @@ import { getDateTime } from "./utils";
|
|||||||
|
|
||||||
export const getDefaultAppState = (): Omit<
|
export const getDefaultAppState = (): Omit<
|
||||||
AppState,
|
AppState,
|
||||||
"offsetTop" | "offsetLeft" | "width" | "height"
|
"offsetTop" | "offsetLeft"
|
||||||
> => {
|
> => {
|
||||||
return {
|
return {
|
||||||
theme: "light",
|
theme: "light",
|
||||||
@@ -43,6 +43,7 @@ export const getDefaultAppState = (): Omit<
|
|||||||
exportWithDarkMode: false,
|
exportWithDarkMode: false,
|
||||||
fileHandle: null,
|
fileHandle: null,
|
||||||
gridSize: null,
|
gridSize: null,
|
||||||
|
height: window.innerHeight,
|
||||||
isBindingEnabled: true,
|
isBindingEnabled: true,
|
||||||
isLibraryOpen: false,
|
isLibraryOpen: false,
|
||||||
isLoading: false,
|
isLoading: false,
|
||||||
@@ -61,6 +62,7 @@ export const getDefaultAppState = (): Omit<
|
|||||||
selectedElementIds: {},
|
selectedElementIds: {},
|
||||||
selectedGroupIds: {},
|
selectedGroupIds: {},
|
||||||
selectionElement: null,
|
selectionElement: null,
|
||||||
|
shouldAddWatermark: false,
|
||||||
shouldCacheIgnoreZoom: false,
|
shouldCacheIgnoreZoom: false,
|
||||||
showHelpDialog: false,
|
showHelpDialog: false,
|
||||||
showStats: false,
|
showStats: false,
|
||||||
@@ -68,6 +70,7 @@ export const getDefaultAppState = (): Omit<
|
|||||||
suggestedBindings: [],
|
suggestedBindings: [],
|
||||||
toastMessage: null,
|
toastMessage: null,
|
||||||
viewBackgroundColor: oc.white,
|
viewBackgroundColor: oc.white,
|
||||||
|
width: window.innerWidth,
|
||||||
zenModeEnabled: false,
|
zenModeEnabled: false,
|
||||||
zoom: { value: 1 as NormalizedZoomValue, translation: { x: 0, y: 0 } },
|
zoom: { value: 1 as NormalizedZoomValue, translation: { x: 0, y: 0 } },
|
||||||
viewModeEnabled: false,
|
viewModeEnabled: false,
|
||||||
@@ -140,6 +143,7 @@ const APP_STATE_STORAGE_CONF = (<
|
|||||||
selectedElementIds: { browser: true, export: false },
|
selectedElementIds: { browser: true, export: false },
|
||||||
selectedGroupIds: { browser: true, export: false },
|
selectedGroupIds: { browser: true, export: false },
|
||||||
selectionElement: { browser: false, export: false },
|
selectionElement: { browser: false, export: false },
|
||||||
|
shouldAddWatermark: { browser: true, export: false },
|
||||||
shouldCacheIgnoreZoom: { browser: true, export: false },
|
shouldCacheIgnoreZoom: { browser: true, export: false },
|
||||||
showHelpDialog: { browser: false, export: false },
|
showHelpDialog: { browser: false, export: false },
|
||||||
showStats: { browser: true, export: false },
|
showStats: { browser: true, export: false },
|
||||||
|
@@ -6,20 +6,16 @@ import { getSelectedElements } from "./scene";
|
|||||||
import { AppState } from "./types";
|
import { AppState } from "./types";
|
||||||
import { SVG_EXPORT_TAG } from "./scene/export";
|
import { SVG_EXPORT_TAG } from "./scene/export";
|
||||||
import { tryParseSpreadsheet, Spreadsheet, VALID_SPREADSHEET } from "./charts";
|
import { tryParseSpreadsheet, Spreadsheet, VALID_SPREADSHEET } from "./charts";
|
||||||
import { EXPORT_DATA_TYPES } from "./constants";
|
import { canvasToBlob } from "./data/blob";
|
||||||
|
|
||||||
|
const TYPE_ELEMENTS = "excalidraw/elements";
|
||||||
|
|
||||||
type ElementsClipboard = {
|
type ElementsClipboard = {
|
||||||
type: typeof EXPORT_DATA_TYPES.excalidrawClipboard;
|
type: typeof TYPE_ELEMENTS;
|
||||||
|
created: number;
|
||||||
elements: ExcalidrawElement[];
|
elements: ExcalidrawElement[];
|
||||||
};
|
};
|
||||||
|
|
||||||
export interface ClipboardData {
|
|
||||||
spreadsheet?: Spreadsheet;
|
|
||||||
elements?: readonly ExcalidrawElement[];
|
|
||||||
text?: string;
|
|
||||||
errorMessage?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
let CLIPBOARD = "";
|
let CLIPBOARD = "";
|
||||||
let PREFER_APP_CLIPBOARD = false;
|
let PREFER_APP_CLIPBOARD = false;
|
||||||
|
|
||||||
@@ -35,16 +31,8 @@ export const probablySupportsClipboardBlob =
|
|||||||
"ClipboardItem" in window &&
|
"ClipboardItem" in window &&
|
||||||
"toBlob" in HTMLCanvasElement.prototype;
|
"toBlob" in HTMLCanvasElement.prototype;
|
||||||
|
|
||||||
const clipboardContainsElements = (
|
const isElementsClipboard = (contents: any): contents is ElementsClipboard => {
|
||||||
contents: any,
|
if (contents?.type === TYPE_ELEMENTS) {
|
||||||
): contents is { elements: ExcalidrawElement[] } => {
|
|
||||||
if (
|
|
||||||
[
|
|
||||||
EXPORT_DATA_TYPES.excalidraw,
|
|
||||||
EXPORT_DATA_TYPES.excalidrawClipboard,
|
|
||||||
].includes(contents?.type) &&
|
|
||||||
Array.isArray(contents.elements)
|
|
||||||
) {
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
@@ -55,7 +43,8 @@ export const copyToClipboard = async (
|
|||||||
appState: AppState,
|
appState: AppState,
|
||||||
) => {
|
) => {
|
||||||
const contents: ElementsClipboard = {
|
const contents: ElementsClipboard = {
|
||||||
type: EXPORT_DATA_TYPES.excalidrawClipboard,
|
type: TYPE_ELEMENTS,
|
||||||
|
created: Date.now(),
|
||||||
elements: getSelectedElements(elements, appState),
|
elements: getSelectedElements(elements, appState),
|
||||||
};
|
};
|
||||||
const json = JSON.stringify(contents);
|
const json = JSON.stringify(contents);
|
||||||
@@ -116,7 +105,12 @@ const getSystemClipboard = async (
|
|||||||
*/
|
*/
|
||||||
export const parseClipboard = async (
|
export const parseClipboard = async (
|
||||||
event: ClipboardEvent | null,
|
event: ClipboardEvent | null,
|
||||||
): Promise<ClipboardData> => {
|
): Promise<{
|
||||||
|
spreadsheet?: Spreadsheet;
|
||||||
|
elements?: readonly ExcalidrawElement[];
|
||||||
|
text?: string;
|
||||||
|
errorMessage?: string;
|
||||||
|
}> => {
|
||||||
const systemClipboard = await getSystemClipboard(event);
|
const systemClipboard = await getSystemClipboard(event);
|
||||||
|
|
||||||
// if system clipboard empty, couldn't be resolved, or contains previously
|
// if system clipboard empty, couldn't be resolved, or contains previously
|
||||||
@@ -137,9 +131,15 @@ export const parseClipboard = async (
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const systemClipboardData = JSON.parse(systemClipboard);
|
const systemClipboardData = JSON.parse(systemClipboard);
|
||||||
if (clipboardContainsElements(systemClipboardData)) {
|
// system clipboard elements are newer than in-app clipboard
|
||||||
|
if (
|
||||||
|
isElementsClipboard(systemClipboardData) &&
|
||||||
|
(!appClipboardData?.created ||
|
||||||
|
appClipboardData.created < systemClipboardData.created)
|
||||||
|
) {
|
||||||
return { elements: systemClipboardData.elements };
|
return { elements: systemClipboardData.elements };
|
||||||
}
|
}
|
||||||
|
// in-app clipboard is newer than system clipboard
|
||||||
return appClipboardData;
|
return appClipboardData;
|
||||||
} catch {
|
} catch {
|
||||||
// system clipboard doesn't contain excalidraw elements → return plaintext
|
// system clipboard doesn't contain excalidraw elements → return plaintext
|
||||||
@@ -151,7 +151,8 @@ export const parseClipboard = async (
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
export const copyBlobToClipboardAsPng = async (blob: Blob) => {
|
export const copyCanvasToClipboardAsPng = async (canvas: HTMLCanvasElement) => {
|
||||||
|
const blob = await canvasToBlob(canvas);
|
||||||
await navigator.clipboard.write([
|
await navigator.clipboard.write([
|
||||||
new window.ClipboardItem({ "image/png": blob }),
|
new window.ClipboardItem({ "image/png": blob }),
|
||||||
]);
|
]);
|
||||||
|
@@ -3,14 +3,13 @@ import { ActionManager } from "../actions/manager";
|
|||||||
import { getNonDeletedElements } from "../element";
|
import { getNonDeletedElements } from "../element";
|
||||||
import { ExcalidrawElement } from "../element/types";
|
import { ExcalidrawElement } from "../element/types";
|
||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
import { useIsMobile } from "../components/App";
|
import useIsMobile from "../is-mobile";
|
||||||
import {
|
import {
|
||||||
canChangeSharpness,
|
canChangeSharpness,
|
||||||
canHaveArrowheads,
|
canHaveArrowheads,
|
||||||
getTargetElements,
|
getTargetElements,
|
||||||
hasBackground,
|
hasBackground,
|
||||||
hasStrokeStyle,
|
hasStroke,
|
||||||
hasStrokeWidth,
|
|
||||||
hasText,
|
hasText,
|
||||||
} from "../scene";
|
} from "../scene";
|
||||||
import { SHAPES } from "../shapes";
|
import { SHAPES } from "../shapes";
|
||||||
@@ -54,17 +53,10 @@ export const SelectedShapeActions = ({
|
|||||||
{showChangeBackgroundIcons && renderAction("changeBackgroundColor")}
|
{showChangeBackgroundIcons && renderAction("changeBackgroundColor")}
|
||||||
{showFillIcons && renderAction("changeFillStyle")}
|
{showFillIcons && renderAction("changeFillStyle")}
|
||||||
|
|
||||||
{(hasStrokeWidth(elementType) ||
|
{(hasStroke(elementType) ||
|
||||||
targetElements.some((element) => hasStrokeWidth(element.type))) &&
|
targetElements.some((element) => hasStroke(element.type))) && (
|
||||||
renderAction("changeStrokeWidth")}
|
|
||||||
|
|
||||||
{(elementType === "freedraw" ||
|
|
||||||
targetElements.some((element) => element.type === "freedraw")) &&
|
|
||||||
renderAction("changeStrokeShape")}
|
|
||||||
|
|
||||||
{(hasStrokeStyle(elementType) ||
|
|
||||||
targetElements.some((element) => hasStrokeStyle(element.type))) && (
|
|
||||||
<>
|
<>
|
||||||
|
{renderAction("changeStrokeWidth")}
|
||||||
{renderAction("changeStrokeStyle")}
|
{renderAction("changeStrokeStyle")}
|
||||||
{renderAction("changeSloppiness")}
|
{renderAction("changeSloppiness")}
|
||||||
</>
|
</>
|
||||||
|
File diff suppressed because it is too large
Load Diff
@@ -1,6 +1,7 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import { ActionManager } from "../actions/manager";
|
import { ActionManager } from "../actions/manager";
|
||||||
import { AppState } from "../types";
|
import { AppState } from "../types";
|
||||||
|
import { DarkModeToggle } from "./DarkModeToggle";
|
||||||
|
|
||||||
export const BackgroundPickerAndDarkModeToggle = ({
|
export const BackgroundPickerAndDarkModeToggle = ({
|
||||||
appState,
|
appState,
|
||||||
@@ -15,10 +16,14 @@ export const BackgroundPickerAndDarkModeToggle = ({
|
|||||||
}) => (
|
}) => (
|
||||||
<div style={{ display: "flex" }}>
|
<div style={{ display: "flex" }}>
|
||||||
{actionManager.renderAction("changeViewBackgroundColor")}
|
{actionManager.renderAction("changeViewBackgroundColor")}
|
||||||
{showThemeBtn && actionManager.renderAction("toggleTheme")}
|
{showThemeBtn && (
|
||||||
{appState.fileHandle && (
|
|
||||||
<div style={{ marginInlineStart: "0.25rem" }}>
|
<div style={{ marginInlineStart: "0.25rem" }}>
|
||||||
{actionManager.renderAction("saveScene")}
|
<DarkModeToggle
|
||||||
|
value={appState.theme}
|
||||||
|
onChange={(theme) => {
|
||||||
|
setAppState({ theme });
|
||||||
|
}}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
@@ -1,3 +1,4 @@
|
|||||||
|
import React from "react";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
|
|
||||||
export const ButtonIconCycle = <T extends any>({
|
export const ButtonIconCycle = <T extends any>({
|
||||||
@@ -13,11 +14,11 @@ export const ButtonIconCycle = <T extends any>({
|
|||||||
}) => {
|
}) => {
|
||||||
const current = options.find((op) => op.value === value);
|
const current = options.find((op) => op.value === value);
|
||||||
|
|
||||||
const cycle = () => {
|
function cycle() {
|
||||||
const index = options.indexOf(current!);
|
const index = options.indexOf(current!);
|
||||||
const next = (index + 1) % options.length;
|
const next = (index + 1) % options.length;
|
||||||
onChange(options[next].value);
|
onChange(options[next].value);
|
||||||
};
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<label key={group} className={clsx({ active: current!.value !== null })}>
|
<label key={group} className={clsx({ active: current!.value !== null })}>
|
||||||
|
@@ -1,53 +0,0 @@
|
|||||||
@import "../css/variables.module";
|
|
||||||
|
|
||||||
.excalidraw {
|
|
||||||
.Card {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
align-items: center;
|
|
||||||
|
|
||||||
max-width: 290px;
|
|
||||||
|
|
||||||
margin: 1em;
|
|
||||||
|
|
||||||
text-align: center;
|
|
||||||
|
|
||||||
.Card-icon {
|
|
||||||
font-size: 2.6em;
|
|
||||||
display: flex;
|
|
||||||
flex: 0 0 auto;
|
|
||||||
padding: 1.4rem;
|
|
||||||
border-radius: 50%;
|
|
||||||
background: var(--card-color);
|
|
||||||
color: $oc-white;
|
|
||||||
|
|
||||||
svg {
|
|
||||||
width: 2.8rem;
|
|
||||||
height: 2.8rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.Card-details {
|
|
||||||
font-size: 0.96em;
|
|
||||||
min-height: 90px;
|
|
||||||
padding: 0 1em;
|
|
||||||
margin-bottom: auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
& .Card-button.ToolIcon_type_button {
|
|
||||||
height: 2.5rem;
|
|
||||||
margin-top: 1em;
|
|
||||||
margin-bottom: 0.3em;
|
|
||||||
background-color: var(--card-color);
|
|
||||||
&:hover {
|
|
||||||
background-color: var(--card-color-darker);
|
|
||||||
}
|
|
||||||
&:active {
|
|
||||||
background-color: var(--card-color-darkest);
|
|
||||||
}
|
|
||||||
.ToolIcon__label {
|
|
||||||
color: $oc-white;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,20 +0,0 @@
|
|||||||
import OpenColor from "open-color";
|
|
||||||
|
|
||||||
import "./Card.scss";
|
|
||||||
|
|
||||||
export const Card: React.FC<{
|
|
||||||
color: keyof OpenColor;
|
|
||||||
}> = ({ children, color }) => {
|
|
||||||
return (
|
|
||||||
<div
|
|
||||||
className="Card"
|
|
||||||
style={{
|
|
||||||
["--card-color" as any]: OpenColor[color][7],
|
|
||||||
["--card-color-darker" as any]: OpenColor[color][8],
|
|
||||||
["--card-color-darkest" as any]: OpenColor[color][9],
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{children}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
@@ -1,89 +0,0 @@
|
|||||||
@import "../css/variables.module";
|
|
||||||
|
|
||||||
.excalidraw {
|
|
||||||
.Checkbox {
|
|
||||||
margin: 4px 0.3em;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
|
|
||||||
cursor: pointer;
|
|
||||||
user-select: none;
|
|
||||||
|
|
||||||
-webkit-tap-highlight-color: transparent;
|
|
||||||
|
|
||||||
&:hover:not(.is-checked) .Checkbox-box:not(:focus) {
|
|
||||||
box-shadow: 0 0 0 2px #{$oc-blue-4};
|
|
||||||
}
|
|
||||||
|
|
||||||
&:hover:not(.is-checked) .Checkbox-box:not(:focus) {
|
|
||||||
svg {
|
|
||||||
display: block;
|
|
||||||
opacity: 0.3;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
&:active {
|
|
||||||
.Checkbox-box {
|
|
||||||
box-shadow: 0 0 2px 1px inset #{$oc-blue-7} !important;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
.Checkbox-box {
|
|
||||||
background-color: fade-out($oc-blue-1, 0.8);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
&.is-checked {
|
|
||||||
.Checkbox-box {
|
|
||||||
background-color: #{$oc-blue-1};
|
|
||||||
svg {
|
|
||||||
display: block;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
&:hover .Checkbox-box {
|
|
||||||
background-color: #{$oc-blue-2};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.Checkbox-box {
|
|
||||||
width: 22px;
|
|
||||||
height: 22px;
|
|
||||||
padding: 0;
|
|
||||||
flex: 0 0 auto;
|
|
||||||
|
|
||||||
margin: 0 1em;
|
|
||||||
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
|
|
||||||
box-shadow: 0 0 0 2px #{$oc-blue-7};
|
|
||||||
background-color: transparent;
|
|
||||||
border-radius: 4px;
|
|
||||||
|
|
||||||
color: #{$oc-blue-7};
|
|
||||||
|
|
||||||
&:focus {
|
|
||||||
box-shadow: 0 0 0 3px #{$oc-blue-7};
|
|
||||||
}
|
|
||||||
|
|
||||||
svg {
|
|
||||||
display: none;
|
|
||||||
width: 16px;
|
|
||||||
height: 16px;
|
|
||||||
stroke-width: 3px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.Checkbox-label {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.Tooltip-icon {
|
|
||||||
width: 1em;
|
|
||||||
height: 1em;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,26 +0,0 @@
|
|||||||
import clsx from "clsx";
|
|
||||||
import { checkIcon } from "./icons";
|
|
||||||
|
|
||||||
import "./CheckboxItem.scss";
|
|
||||||
|
|
||||||
export const CheckboxItem: React.FC<{
|
|
||||||
checked: boolean;
|
|
||||||
onChange: (checked: boolean) => void;
|
|
||||||
}> = ({ children, checked, onChange }) => {
|
|
||||||
return (
|
|
||||||
<div
|
|
||||||
className={clsx("Checkbox", { "is-checked": checked })}
|
|
||||||
onClick={(event) => {
|
|
||||||
onChange(!checked);
|
|
||||||
((event.currentTarget as HTMLDivElement).querySelector(
|
|
||||||
".Checkbox-box",
|
|
||||||
) as HTMLButtonElement).focus();
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<button className="Checkbox-box" role="checkbox" aria-checked={checked}>
|
|
||||||
{checkIcon}
|
|
||||||
</button>
|
|
||||||
<div className="Checkbox-label">{children}</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
@@ -2,7 +2,7 @@ import React from "react";
|
|||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import { ToolButton } from "./ToolButton";
|
import { ToolButton } from "./ToolButton";
|
||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
import { useIsMobile } from "../components/App";
|
import useIsMobile from "../is-mobile";
|
||||||
import { users } from "./icons";
|
import { users } from "./icons";
|
||||||
|
|
||||||
import "./CollabButton.scss";
|
import "./CollabButton.scss";
|
||||||
|
@@ -160,7 +160,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.color-picker-input {
|
.color-picker-input {
|
||||||
width: 11ch; /* length of `transparent` */
|
width: 12ch; /* length of `transparent` + 1 */
|
||||||
margin: 0;
|
margin: 0;
|
||||||
font-size: 1rem;
|
font-size: 1rem;
|
||||||
background-color: var(--input-bg-color);
|
background-color: var(--input-bg-color);
|
||||||
@@ -218,7 +218,7 @@
|
|||||||
left: 2px;
|
left: 2px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include isMobile {
|
@media #{$is-mobile-query} {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -115,7 +115,6 @@ const Picker = ({
|
|||||||
onClose();
|
onClose();
|
||||||
}
|
}
|
||||||
event.nativeEvent.stopImmediatePropagation();
|
event.nativeEvent.stopImmediatePropagation();
|
||||||
event.stopPropagation();
|
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
@@ -76,7 +76,7 @@
|
|||||||
z-index: 1;
|
z-index: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include isMobile {
|
@media #{$is-mobile-query} {
|
||||||
.context-menu-option {
|
.context-menu-option {
|
||||||
display: block;
|
display: block;
|
||||||
|
|
||||||
|
@@ -32,63 +32,67 @@ const ContextMenu = ({
|
|||||||
actionManager,
|
actionManager,
|
||||||
appState,
|
appState,
|
||||||
}: ContextMenuProps) => {
|
}: ContextMenuProps) => {
|
||||||
|
const isDarkTheme = !!document
|
||||||
|
.querySelector(".excalidraw")
|
||||||
|
?.classList.contains("theme--dark");
|
||||||
return (
|
return (
|
||||||
<Popover
|
<div
|
||||||
onCloseRequest={onCloseRequest}
|
className={clsx("excalidraw", {
|
||||||
top={top}
|
"theme--dark theme--dark-background-none": isDarkTheme,
|
||||||
left={left}
|
})}
|
||||||
fitInViewport={true}
|
|
||||||
>
|
>
|
||||||
<ul
|
<Popover
|
||||||
className="context-menu"
|
onCloseRequest={onCloseRequest}
|
||||||
onContextMenu={(event) => event.preventDefault()}
|
top={top}
|
||||||
|
left={left}
|
||||||
|
fitInViewport={true}
|
||||||
>
|
>
|
||||||
{options.map((option, idx) => {
|
<ul
|
||||||
if (option === "separator") {
|
className="context-menu"
|
||||||
return <hr key={idx} className="context-menu-option-separator" />;
|
onContextMenu={(event) => event.preventDefault()}
|
||||||
}
|
>
|
||||||
|
{options.map((option, idx) => {
|
||||||
|
if (option === "separator") {
|
||||||
|
return <hr key={idx} className="context-menu-option-separator" />;
|
||||||
|
}
|
||||||
|
|
||||||
const actionName = option.name;
|
const actionName = option.name;
|
||||||
const label = option.contextItemLabel
|
const label = option.contextItemLabel
|
||||||
? t(option.contextItemLabel)
|
? t(option.contextItemLabel)
|
||||||
: "";
|
: "";
|
||||||
return (
|
return (
|
||||||
<li key={idx} data-testid={actionName} onClick={onCloseRequest}>
|
<li key={idx} data-testid={actionName} onClick={onCloseRequest}>
|
||||||
<button
|
<button
|
||||||
className={clsx("context-menu-option", {
|
className={clsx("context-menu-option", {
|
||||||
dangerous: actionName === "deleteSelectedElements",
|
dangerous: actionName === "deleteSelectedElements",
|
||||||
checkmark: option.checked?.(appState),
|
checkmark: option.checked?.(appState),
|
||||||
})}
|
})}
|
||||||
onClick={() => actionManager.executeAction(option)}
|
onClick={() => actionManager.executeAction(option)}
|
||||||
>
|
>
|
||||||
<div className="context-menu-option__label">{label}</div>
|
<div className="context-menu-option__label">{label}</div>
|
||||||
<kbd className="context-menu-option__shortcut">
|
<kbd className="context-menu-option__shortcut">
|
||||||
{actionName
|
{actionName
|
||||||
? getShortcutFromShortcutName(actionName as ShortcutName)
|
? getShortcutFromShortcutName(actionName as ShortcutName)
|
||||||
: ""}
|
: ""}
|
||||||
</kbd>
|
</kbd>
|
||||||
</button>
|
</button>
|
||||||
</li>
|
</li>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
</ul>
|
</ul>
|
||||||
</Popover>
|
</Popover>
|
||||||
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const contextMenuNodeByContainer = new WeakMap<HTMLElement, HTMLDivElement>();
|
let contextMenuNode: HTMLDivElement;
|
||||||
|
const getContextMenuNode = (): HTMLDivElement => {
|
||||||
const getContextMenuNode = (container: HTMLElement): HTMLDivElement => {
|
|
||||||
let contextMenuNode = contextMenuNodeByContainer.get(container);
|
|
||||||
if (contextMenuNode) {
|
if (contextMenuNode) {
|
||||||
return contextMenuNode;
|
return contextMenuNode;
|
||||||
}
|
}
|
||||||
contextMenuNode = document.createElement("div");
|
const div = document.createElement("div");
|
||||||
container
|
document.body.appendChild(div);
|
||||||
.querySelector(".excalidraw-contextMenuContainer")!
|
return (contextMenuNode = div);
|
||||||
.appendChild(contextMenuNode);
|
|
||||||
contextMenuNodeByContainer.set(container, contextMenuNode);
|
|
||||||
return contextMenuNode;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
type ContextMenuParams = {
|
type ContextMenuParams = {
|
||||||
@@ -97,16 +101,10 @@ type ContextMenuParams = {
|
|||||||
left: ContextMenuProps["left"];
|
left: ContextMenuProps["left"];
|
||||||
actionManager: ContextMenuProps["actionManager"];
|
actionManager: ContextMenuProps["actionManager"];
|
||||||
appState: Readonly<AppState>;
|
appState: Readonly<AppState>;
|
||||||
container: HTMLElement;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleClose = (container: HTMLElement) => {
|
const handleClose = () => {
|
||||||
const contextMenuNode = contextMenuNodeByContainer.get(container);
|
unmountComponentAtNode(getContextMenuNode());
|
||||||
if (contextMenuNode) {
|
|
||||||
unmountComponentAtNode(contextMenuNode);
|
|
||||||
contextMenuNode.remove();
|
|
||||||
contextMenuNodeByContainer.delete(container);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
@@ -123,11 +121,11 @@ export default {
|
|||||||
top={params.top}
|
top={params.top}
|
||||||
left={params.left}
|
left={params.left}
|
||||||
options={options}
|
options={options}
|
||||||
onCloseRequest={() => handleClose(params.container)}
|
onCloseRequest={handleClose}
|
||||||
actionManager={params.actionManager}
|
actionManager={params.actionManager}
|
||||||
appState={params.appState}
|
appState={params.appState}
|
||||||
/>,
|
/>,
|
||||||
getContextMenuNode(params.container),
|
getContextMenuNode(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@@ -2,7 +2,6 @@ import "./ToolIcon.scss";
|
|||||||
|
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
import { ToolButton } from "./ToolButton";
|
|
||||||
|
|
||||||
export type Appearence = "light" | "dark";
|
export type Appearence = "light" | "dark";
|
||||||
|
|
||||||
@@ -13,19 +12,31 @@ export const DarkModeToggle = (props: {
|
|||||||
onChange: (value: Appearence) => void;
|
onChange: (value: Appearence) => void;
|
||||||
title?: string;
|
title?: string;
|
||||||
}) => {
|
}) => {
|
||||||
const title =
|
const title = props.title
|
||||||
props.title ||
|
? props.title
|
||||||
(props.value === "dark" ? t("buttons.lightMode") : t("buttons.darkMode"));
|
: props.value === "dark"
|
||||||
|
? t("buttons.lightMode")
|
||||||
|
: t("buttons.darkMode");
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ToolButton
|
<label
|
||||||
type="icon"
|
className="ToolIcon ToolIcon_type_floating ToolIcon_size_M"
|
||||||
icon={props.value === "light" ? ICONS.MOON : ICONS.SUN}
|
|
||||||
title={title}
|
|
||||||
aria-label={title}
|
|
||||||
onClick={() => props.onChange(props.value === "dark" ? "light" : "dark")}
|
|
||||||
data-testid="toggle-dark-mode"
|
data-testid="toggle-dark-mode"
|
||||||
/>
|
title={title}
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
className="ToolIcon_type_checkbox ToolIcon_toggle_opaque"
|
||||||
|
type="checkbox"
|
||||||
|
onChange={(event) =>
|
||||||
|
props.onChange(event.target.checked ? "dark" : "light")
|
||||||
|
}
|
||||||
|
checked={props.value === "dark"}
|
||||||
|
aria-label={title}
|
||||||
|
/>
|
||||||
|
<div className="ToolIcon__icon">
|
||||||
|
{props.value === "light" ? ICONS.MOON : ICONS.SUN}
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@@ -31,7 +31,7 @@
|
|||||||
padding: 0 16px 16px;
|
padding: 0 16px 16px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include isMobile {
|
@media #{$is-mobile-query} {
|
||||||
.Dialog {
|
.Dialog {
|
||||||
--metric: calc(var(--space-factor) * 4);
|
--metric: calc(var(--space-factor) * 4);
|
||||||
--inset-left: #{"max(var(--metric), var(--sal))"};
|
--inset-left: #{"max(var(--metric), var(--sal))"};
|
||||||
|
@@ -1,14 +1,13 @@
|
|||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import React, { useEffect, useState } from "react";
|
import React, { useEffect } from "react";
|
||||||
import { useCallbackRefState } from "../hooks/useCallbackRefState";
|
import { useCallbackRefState } from "../hooks/useCallbackRefState";
|
||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
import { useIsMobile } from "../components/App";
|
import useIsMobile from "../is-mobile";
|
||||||
import { KEYS } from "../keys";
|
import { KEYS } from "../keys";
|
||||||
import "./Dialog.scss";
|
import "./Dialog.scss";
|
||||||
import { back, close } from "./icons";
|
import { back, close } from "./icons";
|
||||||
import { Island } from "./Island";
|
import { Island } from "./Island";
|
||||||
import { Modal } from "./Modal";
|
import { Modal } from "./Modal";
|
||||||
import { AppState } from "../types";
|
|
||||||
|
|
||||||
export const Dialog = (props: {
|
export const Dialog = (props: {
|
||||||
children: React.ReactNode;
|
children: React.ReactNode;
|
||||||
@@ -17,10 +16,8 @@ export const Dialog = (props: {
|
|||||||
onCloseRequest(): void;
|
onCloseRequest(): void;
|
||||||
title: React.ReactNode;
|
title: React.ReactNode;
|
||||||
autofocus?: boolean;
|
autofocus?: boolean;
|
||||||
theme?: AppState["theme"];
|
|
||||||
}) => {
|
}) => {
|
||||||
const [islandNode, setIslandNode] = useCallbackRefState<HTMLDivElement>();
|
const [islandNode, setIslandNode] = useCallbackRefState<HTMLDivElement>();
|
||||||
const [lastActiveElement] = useState(document.activeElement);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!islandNode) {
|
if (!islandNode) {
|
||||||
@@ -68,25 +65,19 @@ export const Dialog = (props: {
|
|||||||
return focusableElements ? Array.from(focusableElements) : [];
|
return focusableElements ? Array.from(focusableElements) : [];
|
||||||
};
|
};
|
||||||
|
|
||||||
const onClose = () => {
|
|
||||||
(lastActiveElement as HTMLElement).focus();
|
|
||||||
props.onCloseRequest();
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal
|
<Modal
|
||||||
className={clsx("Dialog", props.className)}
|
className={clsx("Dialog", props.className)}
|
||||||
labelledBy="dialog-title"
|
labelledBy="dialog-title"
|
||||||
maxWidth={props.small ? 550 : 800}
|
maxWidth={props.small ? 550 : 800}
|
||||||
onCloseRequest={onClose}
|
onCloseRequest={props.onCloseRequest}
|
||||||
theme={props.theme}
|
|
||||||
>
|
>
|
||||||
<Island ref={setIslandNode}>
|
<Island ref={setIslandNode}>
|
||||||
<h2 id="dialog-title" className="Dialog__title">
|
<h2 id="dialog-title" className="Dialog__title">
|
||||||
<span className="Dialog__titleContent">{props.title}</span>
|
<span className="Dialog__titleContent">{props.title}</span>
|
||||||
<button
|
<button
|
||||||
className="Modal__close"
|
className="Modal__close"
|
||||||
onClick={onClose}
|
onClick={props.onCloseRequest}
|
||||||
aria-label={t("buttons.close")}
|
aria-label={t("buttons.close")}
|
||||||
>
|
>
|
||||||
{useIsMobile() ? back : close}
|
{useIsMobile() ? back : close}
|
||||||
|
@@ -2,7 +2,6 @@ import React, { useState } from "react";
|
|||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
|
|
||||||
import { Dialog } from "./Dialog";
|
import { Dialog } from "./Dialog";
|
||||||
import { useExcalidrawContainer } from "./App";
|
|
||||||
|
|
||||||
export const ErrorDialog = ({
|
export const ErrorDialog = ({
|
||||||
message,
|
message,
|
||||||
@@ -12,7 +11,6 @@ export const ErrorDialog = ({
|
|||||||
onClose?: () => void;
|
onClose?: () => void;
|
||||||
}) => {
|
}) => {
|
||||||
const [modalIsShown, setModalIsShown] = useState(!!message);
|
const [modalIsShown, setModalIsShown] = useState(!!message);
|
||||||
const excalidrawContainer = useExcalidrawContainer();
|
|
||||||
|
|
||||||
const handleClose = React.useCallback(() => {
|
const handleClose = React.useCallback(() => {
|
||||||
setModalIsShown(false);
|
setModalIsShown(false);
|
||||||
@@ -20,9 +18,7 @@ export const ErrorDialog = ({
|
|||||||
if (onClose) {
|
if (onClose) {
|
||||||
onClose();
|
onClose();
|
||||||
}
|
}
|
||||||
// TODO: Fix the A11y issues so this is never needed since we should always focus on last active element
|
}, [onClose]);
|
||||||
excalidrawContainer?.focus();
|
|
||||||
}, [onClose, excalidrawContainer]);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -32,7 +28,14 @@ export const ErrorDialog = ({
|
|||||||
onCloseRequest={handleClose}
|
onCloseRequest={handleClose}
|
||||||
title={t("errorDialog.title")}
|
title={t("errorDialog.title")}
|
||||||
>
|
>
|
||||||
<div style={{ whiteSpace: "pre-wrap" }}>{message}</div>
|
<div>
|
||||||
|
{message.split("\n").map((line) => (
|
||||||
|
<>
|
||||||
|
{line}
|
||||||
|
<br />
|
||||||
|
</>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
|
@@ -28,7 +28,24 @@
|
|||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include isMobile {
|
.ExportDialog__name {
|
||||||
|
grid-column: project-name;
|
||||||
|
margin: auto;
|
||||||
|
|
||||||
|
.TextInput {
|
||||||
|
height: calc(1rem - 3px);
|
||||||
|
|
||||||
|
&--readonly {
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
&:hover {
|
||||||
|
background: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media #{$is-mobile-query} {
|
||||||
.ExportDialog {
|
.ExportDialog {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
@@ -57,62 +74,4 @@
|
|||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.ExportDialog--json {
|
|
||||||
.ExportDialog-cards {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
|
|
||||||
justify-items: center;
|
|
||||||
row-gap: 2em;
|
|
||||||
|
|
||||||
@media (max-width: 460px) {
|
|
||||||
grid-template-columns: repeat(auto-fit, minmax(240px, 1fr));
|
|
||||||
.Card-details {
|
|
||||||
min-height: 40px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.ProjectName {
|
|
||||||
width: fit-content;
|
|
||||||
margin: 1em auto;
|
|
||||||
align-items: flex-start;
|
|
||||||
flex-direction: column;
|
|
||||||
|
|
||||||
.TextInput {
|
|
||||||
width: auto;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.ProjectName-label {
|
|
||||||
margin: 0.625em 0;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
button.ExportDialog-imageExportButton {
|
|
||||||
width: 5rem;
|
|
||||||
height: 5rem;
|
|
||||||
margin: 0 0.2em;
|
|
||||||
|
|
||||||
border-radius: 1rem;
|
|
||||||
background-color: var(--button-color);
|
|
||||||
box-shadow: 0 3px 5px -1px rgb(0 0 0 / 28%), 0 6px 10px 0 rgb(0 0 0 / 14%);
|
|
||||||
|
|
||||||
font-family: Cascadia;
|
|
||||||
font-size: 1.8em;
|
|
||||||
color: $oc-white;
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
background-color: var(--button-color-darker);
|
|
||||||
}
|
|
||||||
&:active {
|
|
||||||
background-color: var(--button-color-darkest);
|
|
||||||
box-shadow: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
svg {
|
|
||||||
width: 0.9em;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@@ -6,20 +6,16 @@ import { canvasToBlob } from "../data/blob";
|
|||||||
import { NonDeletedExcalidrawElement } from "../element/types";
|
import { NonDeletedExcalidrawElement } from "../element/types";
|
||||||
import { CanvasError } from "../errors";
|
import { CanvasError } from "../errors";
|
||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
import { useIsMobile } from "./App";
|
import useIsMobile from "../is-mobile";
|
||||||
import { getSelectedElements, isSomeElementSelected } from "../scene";
|
import { getSelectedElements, isSomeElementSelected } from "../scene";
|
||||||
import { exportToCanvas, getExportSize } from "../scene/export";
|
import { exportToCanvas, getExportSize } from "../scene/export";
|
||||||
import { AppState } from "../types";
|
import { AppState } from "../types";
|
||||||
import { Dialog } from "./Dialog";
|
import { Dialog } from "./Dialog";
|
||||||
import { clipboard, exportImage } from "./icons";
|
import "./ExportDialog.scss";
|
||||||
|
import { clipboard, exportFile, link } from "./icons";
|
||||||
import Stack from "./Stack";
|
import Stack from "./Stack";
|
||||||
import { ToolButton } from "./ToolButton";
|
import { ToolButton } from "./ToolButton";
|
||||||
|
|
||||||
import "./ExportDialog.scss";
|
|
||||||
import { supported as fsSupported } from "browser-fs-access";
|
|
||||||
import OpenColor from "open-color";
|
|
||||||
import { CheckboxItem } from "./CheckboxItem";
|
|
||||||
|
|
||||||
const scales = [1, 2, 3];
|
const scales = [1, 2, 3];
|
||||||
const defaultScale = scales.includes(devicePixelRatio) ? devicePixelRatio : 1;
|
const defaultScale = scales.includes(devicePixelRatio) ? devicePixelRatio : 1;
|
||||||
|
|
||||||
@@ -56,30 +52,7 @@ export type ExportCB = (
|
|||||||
scale?: number,
|
scale?: number,
|
||||||
) => void;
|
) => void;
|
||||||
|
|
||||||
const ExportButton: React.FC<{
|
const ExportModal = ({
|
||||||
color: keyof OpenColor;
|
|
||||||
onClick: () => void;
|
|
||||||
title: string;
|
|
||||||
shade?: number;
|
|
||||||
}> = ({ children, title, onClick, color, shade = 6 }) => {
|
|
||||||
return (
|
|
||||||
<button
|
|
||||||
className="ExportDialog-imageExportButton"
|
|
||||||
style={{
|
|
||||||
["--button-color" as any]: OpenColor[color][shade],
|
|
||||||
["--button-color-darker" as any]: OpenColor[color][shade + 1],
|
|
||||||
["--button-color-darkest" as any]: OpenColor[color][shade + 2],
|
|
||||||
}}
|
|
||||||
title={title}
|
|
||||||
aria-label={title}
|
|
||||||
onClick={onClick}
|
|
||||||
>
|
|
||||||
{children}
|
|
||||||
</button>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const ImageExportModal = ({
|
|
||||||
elements,
|
elements,
|
||||||
appState,
|
appState,
|
||||||
exportPadding = 10,
|
exportPadding = 10,
|
||||||
@@ -87,6 +60,7 @@ const ImageExportModal = ({
|
|||||||
onExportToPng,
|
onExportToPng,
|
||||||
onExportToSvg,
|
onExportToSvg,
|
||||||
onExportToClipboard,
|
onExportToClipboard,
|
||||||
|
onExportToBackend,
|
||||||
}: {
|
}: {
|
||||||
appState: AppState;
|
appState: AppState;
|
||||||
elements: readonly NonDeletedExcalidrawElement[];
|
elements: readonly NonDeletedExcalidrawElement[];
|
||||||
@@ -95,13 +69,18 @@ const ImageExportModal = ({
|
|||||||
onExportToPng: ExportCB;
|
onExportToPng: ExportCB;
|
||||||
onExportToSvg: ExportCB;
|
onExportToSvg: ExportCB;
|
||||||
onExportToClipboard: ExportCB;
|
onExportToClipboard: ExportCB;
|
||||||
|
onExportToBackend?: ExportCB;
|
||||||
onCloseRequest: () => void;
|
onCloseRequest: () => void;
|
||||||
}) => {
|
}) => {
|
||||||
const someElementIsSelected = isSomeElementSelected(elements, appState);
|
const someElementIsSelected = isSomeElementSelected(elements, appState);
|
||||||
const [scale, setScale] = useState(defaultScale);
|
const [scale, setScale] = useState(defaultScale);
|
||||||
const [exportSelected, setExportSelected] = useState(someElementIsSelected);
|
const [exportSelected, setExportSelected] = useState(someElementIsSelected);
|
||||||
const previewRef = useRef<HTMLDivElement>(null);
|
const previewRef = useRef<HTMLDivElement>(null);
|
||||||
const { exportBackground, viewBackgroundColor } = appState;
|
const {
|
||||||
|
exportBackground,
|
||||||
|
viewBackgroundColor,
|
||||||
|
shouldAddWatermark,
|
||||||
|
} = appState;
|
||||||
|
|
||||||
const exportedElements = exportSelected
|
const exportedElements = exportSelected
|
||||||
? getSelectedElements(elements, appState)
|
? getSelectedElements(elements, appState)
|
||||||
@@ -122,6 +101,7 @@ const ImageExportModal = ({
|
|||||||
viewBackgroundColor,
|
viewBackgroundColor,
|
||||||
exportPadding,
|
exportPadding,
|
||||||
scale,
|
scale,
|
||||||
|
shouldAddWatermark,
|
||||||
});
|
});
|
||||||
|
|
||||||
// if converting to blob fails, there's some problem that will
|
// if converting to blob fails, there's some problem that will
|
||||||
@@ -145,6 +125,7 @@ const ImageExportModal = ({
|
|||||||
exportPadding,
|
exportPadding,
|
||||||
viewBackgroundColor,
|
viewBackgroundColor,
|
||||||
scale,
|
scale,
|
||||||
|
shouldAddWatermark,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -152,102 +133,98 @@ const ImageExportModal = ({
|
|||||||
<div className="ExportDialog__preview" ref={previewRef} />
|
<div className="ExportDialog__preview" ref={previewRef} />
|
||||||
{supportsContextFilters &&
|
{supportsContextFilters &&
|
||||||
actionManager.renderAction("exportWithDarkMode")}
|
actionManager.renderAction("exportWithDarkMode")}
|
||||||
<div style={{ display: "grid", gridTemplateColumns: "1fr" }}>
|
<Stack.Col gap={2} align="center">
|
||||||
<div
|
<div className="ExportDialog__actions">
|
||||||
style={{
|
<Stack.Row gap={2}>
|
||||||
display: "grid",
|
<ToolButton
|
||||||
gridTemplateColumns: "repeat(auto-fit, minmax(190px, 1fr))",
|
type="button"
|
||||||
// dunno why this is needed, but when the items wrap it creates
|
label="PNG"
|
||||||
// an overflow
|
title={t("buttons.exportToPng")}
|
||||||
overflow: "hidden",
|
aria-label={t("buttons.exportToPng")}
|
||||||
}}
|
onClick={() => onExportToPng(exportedElements, scale)}
|
||||||
>
|
/>
|
||||||
{actionManager.renderAction("changeExportBackground")}
|
<ToolButton
|
||||||
{someElementIsSelected && (
|
type="button"
|
||||||
<CheckboxItem
|
label="SVG"
|
||||||
checked={exportSelected}
|
title={t("buttons.exportToSvg")}
|
||||||
onChange={(checked) => setExportSelected(checked)}
|
aria-label={t("buttons.exportToSvg")}
|
||||||
>
|
onClick={() => onExportToSvg(exportedElements, scale)}
|
||||||
{t("labels.onlySelected")}
|
/>
|
||||||
</CheckboxItem>
|
{probablySupportsClipboardBlob && (
|
||||||
)}
|
|
||||||
{actionManager.renderAction("changeExportEmbedScene")}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div style={{ display: "flex", alignItems: "center", marginTop: ".6em" }}>
|
|
||||||
<Stack.Row gap={2} justifyContent={"center"}>
|
|
||||||
{scales.map((_scale) => {
|
|
||||||
const [width, height] = getExportSize(
|
|
||||||
exportedElements,
|
|
||||||
exportPadding,
|
|
||||||
_scale,
|
|
||||||
);
|
|
||||||
|
|
||||||
const scaleButtonTitle = `${t(
|
|
||||||
"buttons.scale",
|
|
||||||
)} ${_scale}x (${width}x${height})`;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<ToolButton
|
<ToolButton
|
||||||
key={_scale}
|
type="button"
|
||||||
size="s"
|
icon={clipboard}
|
||||||
type="radio"
|
title={t("buttons.copyPngToClipboard")}
|
||||||
icon={`${_scale}x`}
|
aria-label={t("buttons.copyPngToClipboard")}
|
||||||
name="export-canvas-scale"
|
onClick={() => onExportToClipboard(exportedElements, scale)}
|
||||||
title={scaleButtonTitle}
|
|
||||||
aria-label={scaleButtonTitle}
|
|
||||||
id="export-canvas-scale"
|
|
||||||
checked={_scale === scale}
|
|
||||||
onChange={() => setScale(_scale)}
|
|
||||||
/>
|
/>
|
||||||
);
|
)}
|
||||||
})}
|
{onExportToBackend && (
|
||||||
</Stack.Row>
|
<ToolButton
|
||||||
<p style={{ marginLeft: "1em", userSelect: "none" }}>Scale</p>
|
type="button"
|
||||||
</div>
|
icon={link}
|
||||||
<div
|
title={t("buttons.getShareableLink")}
|
||||||
style={{
|
aria-label={t("buttons.getShareableLink")}
|
||||||
display: "flex",
|
onClick={() => onExportToBackend(exportedElements)}
|
||||||
alignItems: "center",
|
/>
|
||||||
justifyContent: "center",
|
)}
|
||||||
margin: ".6em 0",
|
</Stack.Row>
|
||||||
}}
|
<div className="ExportDialog__name">
|
||||||
>
|
{actionManager.renderAction("changeProjectName")}
|
||||||
{!fsSupported && actionManager.renderAction("changeProjectName")}
|
</div>
|
||||||
</div>
|
<Stack.Row gap={2}>
|
||||||
<Stack.Row gap={2} justifyContent="center" style={{ margin: "2em 0" }}>
|
{scales.map((s) => {
|
||||||
<ExportButton
|
const [width, height] = getExportSize(
|
||||||
color="indigo"
|
exportedElements,
|
||||||
title={t("buttons.exportToPng")}
|
exportPadding,
|
||||||
aria-label={t("buttons.exportToPng")}
|
shouldAddWatermark,
|
||||||
onClick={() => onExportToPng(exportedElements, scale)}
|
s,
|
||||||
>
|
);
|
||||||
PNG
|
|
||||||
</ExportButton>
|
const scaleButtonTitle = `${t(
|
||||||
<ExportButton
|
"buttons.scale",
|
||||||
color="red"
|
)} ${s}x (${width}x${height})`;
|
||||||
title={t("buttons.exportToSvg")}
|
|
||||||
aria-label={t("buttons.exportToSvg")}
|
return (
|
||||||
onClick={() => onExportToSvg(exportedElements, scale)}
|
<ToolButton
|
||||||
>
|
key={s}
|
||||||
SVG
|
size="s"
|
||||||
</ExportButton>
|
type="radio"
|
||||||
{probablySupportsClipboardBlob && (
|
icon={`${s}x`}
|
||||||
<ExportButton
|
name="export-canvas-scale"
|
||||||
title={t("buttons.copyPngToClipboard")}
|
title={scaleButtonTitle}
|
||||||
onClick={() => onExportToClipboard(exportedElements, scale)}
|
aria-label={scaleButtonTitle}
|
||||||
color="gray"
|
id="export-canvas-scale"
|
||||||
shade={7}
|
checked={s === scale}
|
||||||
>
|
onChange={() => setScale(s)}
|
||||||
{clipboard}
|
/>
|
||||||
</ExportButton>
|
);
|
||||||
|
})}
|
||||||
|
</Stack.Row>
|
||||||
|
</div>
|
||||||
|
{actionManager.renderAction("changeExportBackground")}
|
||||||
|
{someElementIsSelected && (
|
||||||
|
<div>
|
||||||
|
<label>
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
checked={exportSelected}
|
||||||
|
onChange={(event) =>
|
||||||
|
setExportSelected(event.currentTarget.checked)
|
||||||
|
}
|
||||||
|
/>{" "}
|
||||||
|
{t("labels.onlySelected")}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
)}
|
)}
|
||||||
</Stack.Row>
|
{actionManager.renderAction("changeExportEmbedScene")}
|
||||||
|
{actionManager.renderAction("changeShouldAddWatermark")}
|
||||||
|
</Stack.Col>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const ImageExportDialog = ({
|
export const ExportDialog = ({
|
||||||
elements,
|
elements,
|
||||||
appState,
|
appState,
|
||||||
exportPadding = 10,
|
exportPadding = 10,
|
||||||
@@ -255,6 +232,7 @@ export const ImageExportDialog = ({
|
|||||||
onExportToPng,
|
onExportToPng,
|
||||||
onExportToSvg,
|
onExportToSvg,
|
||||||
onExportToClipboard,
|
onExportToClipboard,
|
||||||
|
onExportToBackend,
|
||||||
}: {
|
}: {
|
||||||
appState: AppState;
|
appState: AppState;
|
||||||
elements: readonly NonDeletedExcalidrawElement[];
|
elements: readonly NonDeletedExcalidrawElement[];
|
||||||
@@ -263,11 +241,14 @@ export const ImageExportDialog = ({
|
|||||||
onExportToPng: ExportCB;
|
onExportToPng: ExportCB;
|
||||||
onExportToSvg: ExportCB;
|
onExportToSvg: ExportCB;
|
||||||
onExportToClipboard: ExportCB;
|
onExportToClipboard: ExportCB;
|
||||||
|
onExportToBackend?: ExportCB;
|
||||||
}) => {
|
}) => {
|
||||||
const [modalIsShown, setModalIsShown] = useState(false);
|
const [modalIsShown, setModalIsShown] = useState(false);
|
||||||
|
const triggerButton = useRef<HTMLButtonElement>(null);
|
||||||
|
|
||||||
const handleClose = React.useCallback(() => {
|
const handleClose = React.useCallback(() => {
|
||||||
setModalIsShown(false);
|
setModalIsShown(false);
|
||||||
|
triggerButton.current?.focus();
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -276,16 +257,17 @@ export const ImageExportDialog = ({
|
|||||||
onClick={() => {
|
onClick={() => {
|
||||||
setModalIsShown(true);
|
setModalIsShown(true);
|
||||||
}}
|
}}
|
||||||
data-testid="image-export-button"
|
data-testid="export-button"
|
||||||
icon={exportImage}
|
icon={exportFile}
|
||||||
type="button"
|
type="button"
|
||||||
aria-label={t("buttons.exportImage")}
|
aria-label={t("buttons.export")}
|
||||||
showAriaLabel={useIsMobile()}
|
showAriaLabel={useIsMobile()}
|
||||||
title={t("buttons.exportImage")}
|
title={t("buttons.export")}
|
||||||
|
ref={triggerButton}
|
||||||
/>
|
/>
|
||||||
{modalIsShown && (
|
{modalIsShown && (
|
||||||
<Dialog onCloseRequest={handleClose} title={t("buttons.exportImage")}>
|
<Dialog onCloseRequest={handleClose} title={t("buttons.export")}>
|
||||||
<ImageExportModal
|
<ExportModal
|
||||||
elements={elements}
|
elements={elements}
|
||||||
appState={appState}
|
appState={appState}
|
||||||
exportPadding={exportPadding}
|
exportPadding={exportPadding}
|
||||||
@@ -293,6 +275,7 @@ export const ImageExportDialog = ({
|
|||||||
onExportToPng={onExportToPng}
|
onExportToPng={onExportToPng}
|
||||||
onExportToSvg={onExportToSvg}
|
onExportToSvg={onExportToSvg}
|
||||||
onExportToClipboard={onExportToClipboard}
|
onExportToClipboard={onExportToClipboard}
|
||||||
|
onExportToBackend={onExportToBackend}
|
||||||
onCloseRequest={handleClose}
|
onCloseRequest={handleClose}
|
||||||
/>
|
/>
|
||||||
</Dialog>
|
</Dialog>
|
@@ -1,5 +1,6 @@
|
|||||||
.excalidraw {
|
.excalidraw {
|
||||||
.FixedSideContainer {
|
.FixedSideContainer {
|
||||||
|
--margin: 0.25rem;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
pointer-events: none;
|
pointer-events: none;
|
||||||
}
|
}
|
||||||
@@ -9,9 +10,9 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.FixedSideContainer_side_top {
|
.FixedSideContainer_side_top {
|
||||||
left: var(--space-factor);
|
left: var(--margin);
|
||||||
top: var(--space-factor);
|
top: var(--margin);
|
||||||
right: var(--space-factor);
|
right: var(--margin);
|
||||||
z-index: 2;
|
z-index: 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -22,16 +23,16 @@
|
|||||||
|
|
||||||
/* TODO: if these are used, make sure to implement RTL support
|
/* TODO: if these are used, make sure to implement RTL support
|
||||||
.FixedSideContainer_side_left {
|
.FixedSideContainer_side_left {
|
||||||
left: var(--space-factor);
|
left: var(--margin);
|
||||||
top: var(--space-factor);
|
top: var(--margin);
|
||||||
bottom: var(--space-factor);
|
bottom: var(--margin);
|
||||||
z-index: 1;
|
z-index: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.FixedSideContainer_side_right {
|
.FixedSideContainer_side_right {
|
||||||
right: var(--space-factor);
|
right: var(--margin);
|
||||||
top: var(--space-factor);
|
top: var(--margin);
|
||||||
bottom: var(--space-factor);
|
bottom: var(--margin);
|
||||||
z-index: 3;
|
z-index: 3;
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
|
@@ -3,19 +3,13 @@ import React from "react";
|
|||||||
|
|
||||||
// https://github.com/tholman/github-corners
|
// https://github.com/tholman/github-corners
|
||||||
export const GitHubCorner = React.memo(
|
export const GitHubCorner = React.memo(
|
||||||
({ theme, dir }: { theme: "light" | "dark"; dir: string }) => (
|
({ theme }: { theme: "light" | "dark" }) => (
|
||||||
<svg
|
<svg
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
width="40"
|
width="40"
|
||||||
height="40"
|
height="40"
|
||||||
viewBox="0 0 250 250"
|
viewBox="0 0 250 250"
|
||||||
className="rtl-mirror"
|
className="github-corner rtl-mirror"
|
||||||
style={{
|
|
||||||
marginTop: "calc(var(--space-factor) * -1)",
|
|
||||||
[dir === "rtl"
|
|
||||||
? "marginLeft"
|
|
||||||
: "marginRight"]: "calc(var(--space-factor) * -1)",
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
<a
|
<a
|
||||||
href="https://github.com/excalidraw/excalidraw"
|
href="https://github.com/excalidraw/excalidraw"
|
||||||
@@ -25,18 +19,18 @@ export const GitHubCorner = React.memo(
|
|||||||
>
|
>
|
||||||
<path
|
<path
|
||||||
d="M0 0l115 115h15l12 27 108 108V0z"
|
d="M0 0l115 115h15l12 27 108 108V0z"
|
||||||
fill={theme === "light" ? oc.gray[6] : oc.gray[7]}
|
fill={theme === "light" ? oc.gray[6] : oc.gray[8]}
|
||||||
/>
|
/>
|
||||||
<path
|
<path
|
||||||
className="octo-arm"
|
className="octo-arm"
|
||||||
d="M128 109c-15-9-9-19-9-19 3-7 2-11 2-11-1-7 3-2 3-2 4 5 2 11 2 11-3 10 5 15 9 16"
|
d="M128 109c-15-9-9-19-9-19 3-7 2-11 2-11-1-7 3-2 3-2 4 5 2 11 2 11-3 10 5 15 9 16"
|
||||||
style={{ transformOrigin: "130px 106px" }}
|
style={{ transformOrigin: "130px 106px" }}
|
||||||
fill={theme === "light" ? oc.white : "var(--default-bg-color)"}
|
fill={theme === "light" ? oc.white : oc.black}
|
||||||
/>
|
/>
|
||||||
<path
|
<path
|
||||||
className="octo-body"
|
className="octo-body"
|
||||||
d="M115 115s4 2 5 0l14-14c3-2 6-3 8-3-8-11-15-24 2-41 5-5 10-7 16-7 1-2 3-7 12-11 0 0 5 3 7 16 4 2 8 5 12 9s7 8 9 12c14 3 17 7 17 7-4 8-9 11-11 11 0 6-2 11-7 16-16 16-30 10-41 2 0 3-1 7-5 11l-12 11c-1 1 1 5 1 5z"
|
d="M115 115s4 2 5 0l14-14c3-2 6-3 8-3-8-11-15-24 2-41 5-5 10-7 16-7 1-2 3-7 12-11 0 0 5 3 7 16 4 2 8 5 12 9s7 8 9 12c14 3 17 7 17 7-4 8-9 11-11 11 0 6-2 11-7 16-16 16-30 10-41 2 0 3-1 7-5 11l-12 11c-1 1 1 5 1 5z"
|
||||||
fill={theme === "light" ? oc.white : "var(--default-bg-color)"}
|
fill={theme === "light" ? oc.white : oc.black}
|
||||||
/>
|
/>
|
||||||
</a>
|
</a>
|
||||||
</svg>
|
</svg>
|
@@ -153,7 +153,7 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => {
|
|||||||
<Shortcut label={t("toolBar.arrow")} shortcuts={["A", "5"]} />
|
<Shortcut label={t("toolBar.arrow")} shortcuts={["A", "5"]} />
|
||||||
<Shortcut label={t("toolBar.line")} shortcuts={["P", "6"]} />
|
<Shortcut label={t("toolBar.line")} shortcuts={["P", "6"]} />
|
||||||
<Shortcut
|
<Shortcut
|
||||||
label={t("toolBar.freedraw")}
|
label={t("toolBar.draw")}
|
||||||
shortcuts={["Shift+P", "7"]}
|
shortcuts={["Shift+P", "7"]}
|
||||||
/>
|
/>
|
||||||
<Shortcut label={t("toolBar.text")} shortcuts={["T", "8"]} />
|
<Shortcut label={t("toolBar.text")} shortcuts={["T", "8"]} />
|
||||||
@@ -231,14 +231,6 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => {
|
|||||||
label={t("labels.viewMode")}
|
label={t("labels.viewMode")}
|
||||||
shortcuts={[getShortcutKey("Alt+R")]}
|
shortcuts={[getShortcutKey("Alt+R")]}
|
||||||
/>
|
/>
|
||||||
<Shortcut
|
|
||||||
label={t("labels.toggleTheme")}
|
|
||||||
shortcuts={[getShortcutKey("Alt+Shift+D")]}
|
|
||||||
/>
|
|
||||||
<Shortcut
|
|
||||||
label={t("stats.title")}
|
|
||||||
shortcuts={[getShortcutKey("Alt+/")]}
|
|
||||||
/>
|
|
||||||
</ShortcutIsland>
|
</ShortcutIsland>
|
||||||
</Column>
|
</Column>
|
||||||
<Column>
|
<Column>
|
||||||
@@ -357,14 +349,6 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => {
|
|||||||
label={t("labels.ungroup")}
|
label={t("labels.ungroup")}
|
||||||
shortcuts={[getShortcutKey("CtrlOrCmd+Shift+G")]}
|
shortcuts={[getShortcutKey("CtrlOrCmd+Shift+G")]}
|
||||||
/>
|
/>
|
||||||
<Shortcut
|
|
||||||
label={t("labels.flipHorizontal")}
|
|
||||||
shortcuts={[getShortcutKey("Shift+H")]}
|
|
||||||
/>
|
|
||||||
<Shortcut
|
|
||||||
label={t("labels.flipVertical")}
|
|
||||||
shortcuts={[getShortcutKey("Shift+V")]}
|
|
||||||
/>
|
|
||||||
</ShortcutIsland>
|
</ShortcutIsland>
|
||||||
</Column>
|
</Column>
|
||||||
</Columns>
|
</Columns>
|
||||||
|
@@ -9,13 +9,7 @@ type HelpIconProps = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const HelpIcon = (props: HelpIconProps) => (
|
export const HelpIcon = (props: HelpIconProps) => (
|
||||||
<button
|
<label title={`${props.title} — ?`} className="help-icon">
|
||||||
className="help-icon"
|
<div onClick={props.onClick}>{questionCircle}</div>
|
||||||
onClick={props.onClick}
|
</label>
|
||||||
type="button"
|
|
||||||
title={`${props.title} — ?`}
|
|
||||||
aria-label={props.title}
|
|
||||||
>
|
|
||||||
{questionCircle}
|
|
||||||
</button>
|
|
||||||
);
|
);
|
||||||
|
@@ -19,7 +19,7 @@ $wide-viewport-width: 1000px;
|
|||||||
color: $oc-gray-6;
|
color: $oc-gray-6;
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
|
|
||||||
@include isMobile {
|
@media #{$is-mobile-query} {
|
||||||
position: static;
|
position: static;
|
||||||
padding-right: 2em;
|
padding-right: 2em;
|
||||||
}
|
}
|
||||||
|
@@ -23,7 +23,7 @@ const getHints = ({ appState, elements }: Hint) => {
|
|||||||
return t("hints.linearElementMulti");
|
return t("hints.linearElementMulti");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (elementType === "freedraw") {
|
if (elementType === "draw") {
|
||||||
return t("hints.freeDraw");
|
return t("hints.freeDraw");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -111,7 +111,7 @@
|
|||||||
:root[dir="rtl"] & {
|
:root[dir="rtl"] & {
|
||||||
left: 2px;
|
left: 2px;
|
||||||
}
|
}
|
||||||
@include isMobile {
|
@media #{$is-mobile-query} {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -88,7 +88,6 @@ function Picker<T>({
|
|||||||
onClose();
|
onClose();
|
||||||
}
|
}
|
||||||
event.nativeEvent.stopImmediatePropagation();
|
event.nativeEvent.stopImmediatePropagation();
|
||||||
event.stopPropagation();
|
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
@@ -2,6 +2,7 @@
|
|||||||
.Island {
|
.Island {
|
||||||
--padding: 0;
|
--padding: 0;
|
||||||
background-color: var(--island-bg-color);
|
background-color: var(--island-bg-color);
|
||||||
|
backdrop-filter: saturate(100%) blur(10px);
|
||||||
box-shadow: var(--shadow-island);
|
box-shadow: var(--shadow-island);
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
padding: calc(var(--padding) * var(--space-factor));
|
padding: calc(var(--padding) * var(--space-factor));
|
||||||
|
@@ -1,117 +0,0 @@
|
|||||||
import React, { useState } from "react";
|
|
||||||
import { ActionsManagerInterface } from "../actions/types";
|
|
||||||
import { NonDeletedExcalidrawElement } from "../element/types";
|
|
||||||
import { t } from "../i18n";
|
|
||||||
import { useIsMobile } from "./App";
|
|
||||||
import { AppState } from "../types";
|
|
||||||
import { Dialog } from "./Dialog";
|
|
||||||
import { exportFile, exportToFileIcon, link } from "./icons";
|
|
||||||
import { ToolButton } from "./ToolButton";
|
|
||||||
import { actionSaveAsScene } from "../actions/actionExport";
|
|
||||||
import { Card } from "./Card";
|
|
||||||
|
|
||||||
import "./ExportDialog.scss";
|
|
||||||
import { supported as fsSupported } from "browser-fs-access";
|
|
||||||
|
|
||||||
export type ExportCB = (
|
|
||||||
elements: readonly NonDeletedExcalidrawElement[],
|
|
||||||
scale?: number,
|
|
||||||
) => void;
|
|
||||||
|
|
||||||
const JSONExportModal = ({
|
|
||||||
elements,
|
|
||||||
appState,
|
|
||||||
actionManager,
|
|
||||||
onExportToBackend,
|
|
||||||
}: {
|
|
||||||
appState: AppState;
|
|
||||||
elements: readonly NonDeletedExcalidrawElement[];
|
|
||||||
actionManager: ActionsManagerInterface;
|
|
||||||
onExportToBackend?: ExportCB;
|
|
||||||
onCloseRequest: () => void;
|
|
||||||
}) => {
|
|
||||||
return (
|
|
||||||
<div className="ExportDialog ExportDialog--json">
|
|
||||||
<div className="ExportDialog-cards">
|
|
||||||
<Card color="lime">
|
|
||||||
<div className="Card-icon">{exportToFileIcon}</div>
|
|
||||||
<h2>{t("exportDialog.disk_title")}</h2>
|
|
||||||
<div className="Card-details">
|
|
||||||
{t("exportDialog.disk_details")}
|
|
||||||
{!fsSupported && actionManager.renderAction("changeProjectName")}
|
|
||||||
</div>
|
|
||||||
<ToolButton
|
|
||||||
className="Card-button"
|
|
||||||
type="button"
|
|
||||||
title={t("exportDialog.disk_button")}
|
|
||||||
aria-label={t("exportDialog.disk_button")}
|
|
||||||
showAriaLabel={true}
|
|
||||||
onClick={() => {
|
|
||||||
actionManager.executeAction(actionSaveAsScene);
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</Card>
|
|
||||||
{onExportToBackend && (
|
|
||||||
<Card color="pink">
|
|
||||||
<div className="Card-icon">{link}</div>
|
|
||||||
<h2>{t("exportDialog.link_title")}</h2>
|
|
||||||
<div className="Card-details">{t("exportDialog.link_details")}</div>
|
|
||||||
<ToolButton
|
|
||||||
className="Card-button"
|
|
||||||
type="button"
|
|
||||||
title={t("exportDialog.link_button")}
|
|
||||||
aria-label={t("exportDialog.link_button")}
|
|
||||||
showAriaLabel={true}
|
|
||||||
onClick={() => onExportToBackend(elements)}
|
|
||||||
/>
|
|
||||||
</Card>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const JSONExportDialog = ({
|
|
||||||
elements,
|
|
||||||
appState,
|
|
||||||
actionManager,
|
|
||||||
onExportToBackend,
|
|
||||||
}: {
|
|
||||||
appState: AppState;
|
|
||||||
elements: readonly NonDeletedExcalidrawElement[];
|
|
||||||
actionManager: ActionsManagerInterface;
|
|
||||||
onExportToBackend?: ExportCB;
|
|
||||||
}) => {
|
|
||||||
const [modalIsShown, setModalIsShown] = useState(false);
|
|
||||||
|
|
||||||
const handleClose = React.useCallback(() => {
|
|
||||||
setModalIsShown(false);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<ToolButton
|
|
||||||
onClick={() => {
|
|
||||||
setModalIsShown(true);
|
|
||||||
}}
|
|
||||||
data-testid="json-export-button"
|
|
||||||
icon={exportFile}
|
|
||||||
type="button"
|
|
||||||
aria-label={t("buttons.export")}
|
|
||||||
showAriaLabel={useIsMobile()}
|
|
||||||
title={t("buttons.export")}
|
|
||||||
/>
|
|
||||||
{modalIsShown && (
|
|
||||||
<Dialog onCloseRequest={handleClose} title={t("buttons.export")}>
|
|
||||||
<JSONExportModal
|
|
||||||
elements={elements}
|
|
||||||
appState={appState}
|
|
||||||
actionManager={actionManager}
|
|
||||||
onExportToBackend={onExportToBackend}
|
|
||||||
onCloseRequest={handleClose}
|
|
||||||
/>
|
|
||||||
</Dialog>
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
|
@@ -40,17 +40,50 @@
|
|||||||
.layer-ui__wrapper {
|
.layer-ui__wrapper {
|
||||||
z-index: var(--zIndex-layerUI);
|
z-index: var(--zIndex-layerUI);
|
||||||
|
|
||||||
&__top-right {
|
.encrypted-icon {
|
||||||
|
position: relative;
|
||||||
|
margin-inline-start: 15px;
|
||||||
display: flex;
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
border-radius: var(--space-factor);
|
||||||
|
color: $oc-green-9;
|
||||||
|
|
||||||
|
svg {
|
||||||
|
width: 1.2rem;
|
||||||
|
height: 1.2rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&__github-corner {
|
||||||
|
top: 0;
|
||||||
|
|
||||||
|
:root[dir="ltr"] & {
|
||||||
|
right: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
:root[dir="rtl"] & {
|
||||||
|
left: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
position: absolute;
|
||||||
|
width: 40px;
|
||||||
}
|
}
|
||||||
|
|
||||||
&__footer {
|
&__footer {
|
||||||
width: 100%;
|
position: absolute;
|
||||||
|
z-index: 100;
|
||||||
|
bottom: 0;
|
||||||
|
|
||||||
&-right {
|
:root[dir="ltr"] & {
|
||||||
z-index: 100;
|
right: 0;
|
||||||
display: flex;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
:root[dir="rtl"] & {
|
||||||
|
left: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
width: 190px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.zen-mode-transition {
|
.zen-mode-transition {
|
||||||
@@ -72,16 +105,12 @@
|
|||||||
transform: translate(-999px, 0);
|
transform: translate(-999px, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
:root[dir="ltr"] &.layer-ui__wrapper__footer-left--transition-left {
|
:root[dir="ltr"] &.App-menu_bottom--transition-left {
|
||||||
transform: translate(-92px, 0);
|
transform: translate(-92px, 0);
|
||||||
}
|
}
|
||||||
:root[dir="rtl"] &.layer-ui__wrapper__footer-left--transition-left {
|
:root[dir="rtl"] &.App-menu_bottom--transition-left {
|
||||||
transform: translate(92px, 0);
|
transform: translate(92px, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
&.layer-ui__wrapper__footer-left--transition-bottom {
|
|
||||||
transform: translate(0, 92px);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.disable-zen-mode {
|
.disable-zen-mode {
|
||||||
@@ -108,16 +137,5 @@
|
|||||||
transition-delay: 0.8s;
|
transition-delay: 0.8s;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.layer-ui__wrapper__footer-center {
|
|
||||||
pointer-events: none;
|
|
||||||
& > * {
|
|
||||||
pointer-events: all;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.layer-ui__wrapper__footer-left,
|
|
||||||
.layer-ui__wrapper__footer-right {
|
|
||||||
pointer-events: all;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -10,28 +10,24 @@ import { ActionManager } from "../actions/manager";
|
|||||||
import { CLASSES } from "../constants";
|
import { CLASSES } from "../constants";
|
||||||
import { exportCanvas } from "../data";
|
import { exportCanvas } from "../data";
|
||||||
import { importLibraryFromJSON, saveLibraryAsJSON } from "../data/json";
|
import { importLibraryFromJSON, saveLibraryAsJSON } from "../data/json";
|
||||||
|
import { Library } from "../data/library";
|
||||||
import { isTextElement, showSelectedShapeActions } from "../element";
|
import { isTextElement, showSelectedShapeActions } from "../element";
|
||||||
import { NonDeletedExcalidrawElement } from "../element/types";
|
import { NonDeletedExcalidrawElement } from "../element/types";
|
||||||
import { Language, t } from "../i18n";
|
import { Language, t } from "../i18n";
|
||||||
import { useIsMobile } from "../components/App";
|
import useIsMobile from "../is-mobile";
|
||||||
import { calculateScrollCenter, getSelectedElements } from "../scene";
|
import { calculateScrollCenter, getSelectedElements } from "../scene";
|
||||||
import { ExportType } from "../scene/types";
|
import { ExportType } from "../scene/types";
|
||||||
import {
|
import { AppState, ExcalidrawProps, LibraryItem, LibraryItems } from "../types";
|
||||||
AppProps,
|
|
||||||
AppState,
|
|
||||||
ExcalidrawProps,
|
|
||||||
LibraryItem,
|
|
||||||
LibraryItems,
|
|
||||||
} from "../types";
|
|
||||||
import { muteFSAbortError } from "../utils";
|
import { muteFSAbortError } from "../utils";
|
||||||
import { SelectedShapeActions, ShapesSwitcher, ZoomActions } from "./Actions";
|
import { SelectedShapeActions, ShapesSwitcher, ZoomActions } from "./Actions";
|
||||||
import { BackgroundPickerAndDarkModeToggle } from "./BackgroundPickerAndDarkModeToggle";
|
import { BackgroundPickerAndDarkModeToggle } from "./BackgroundPickerAndDarkModeToggle";
|
||||||
import CollabButton from "./CollabButton";
|
import CollabButton from "./CollabButton";
|
||||||
import { ErrorDialog } from "./ErrorDialog";
|
import { ErrorDialog } from "./ErrorDialog";
|
||||||
import { ExportCB, ImageExportDialog } from "./ImageExportDialog";
|
import { ExportCB, ExportDialog } from "./ExportDialog";
|
||||||
import { FixedSideContainer } from "./FixedSideContainer";
|
import { FixedSideContainer } from "./FixedSideContainer";
|
||||||
|
import { GitHubCorner } from "./GitHubCorner";
|
||||||
import { HintViewer } from "./HintViewer";
|
import { HintViewer } from "./HintViewer";
|
||||||
import { exportFile, load, trash } from "./icons";
|
import { exportFile, load, shield, trash } from "./icons";
|
||||||
import { Island } from "./Island";
|
import { Island } from "./Island";
|
||||||
import "./LayerUI.scss";
|
import "./LayerUI.scss";
|
||||||
import { LibraryUnit } from "./LibraryUnit";
|
import { LibraryUnit } from "./LibraryUnit";
|
||||||
@@ -45,8 +41,6 @@ import Stack from "./Stack";
|
|||||||
import { ToolButton } from "./ToolButton";
|
import { ToolButton } from "./ToolButton";
|
||||||
import { Tooltip } from "./Tooltip";
|
import { Tooltip } from "./Tooltip";
|
||||||
import { UserList } from "./UserList";
|
import { UserList } from "./UserList";
|
||||||
import Library from "../data/library";
|
|
||||||
import { JSONExportDialog } from "./JSONExportDialog";
|
|
||||||
|
|
||||||
interface LayerUIProps {
|
interface LayerUIProps {
|
||||||
actionManager: ActionManager;
|
actionManager: ActionManager;
|
||||||
@@ -68,14 +62,9 @@ interface LayerUIProps {
|
|||||||
appState: AppState,
|
appState: AppState,
|
||||||
canvas: HTMLCanvasElement | null,
|
canvas: HTMLCanvasElement | null,
|
||||||
) => void;
|
) => void;
|
||||||
renderTopRightUI?: (isMobile: boolean, appState: AppState) => JSX.Element;
|
renderCustomFooter?: (isMobile: boolean) => JSX.Element;
|
||||||
renderCustomFooter?: (isMobile: boolean, appState: AppState) => JSX.Element;
|
|
||||||
viewModeEnabled: boolean;
|
viewModeEnabled: boolean;
|
||||||
libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"];
|
libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"];
|
||||||
UIOptions: AppProps["UIOptions"];
|
|
||||||
focusContainer: () => void;
|
|
||||||
library: Library;
|
|
||||||
id: string;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const useOnClickOutside = (
|
const useOnClickOutside = (
|
||||||
@@ -107,7 +96,7 @@ const useOnClickOutside = (
|
|||||||
};
|
};
|
||||||
|
|
||||||
const LibraryMenuItems = ({
|
const LibraryMenuItems = ({
|
||||||
libraryItems,
|
library,
|
||||||
onRemoveFromLibrary,
|
onRemoveFromLibrary,
|
||||||
onAddToLibrary,
|
onAddToLibrary,
|
||||||
onInsertShape,
|
onInsertShape,
|
||||||
@@ -115,11 +104,8 @@ const LibraryMenuItems = ({
|
|||||||
setAppState,
|
setAppState,
|
||||||
setLibraryItems,
|
setLibraryItems,
|
||||||
libraryReturnUrl,
|
libraryReturnUrl,
|
||||||
focusContainer,
|
|
||||||
library,
|
|
||||||
id,
|
|
||||||
}: {
|
}: {
|
||||||
libraryItems: LibraryItems;
|
library: LibraryItems;
|
||||||
pendingElements: LibraryItem;
|
pendingElements: LibraryItem;
|
||||||
onRemoveFromLibrary: (index: number) => void;
|
onRemoveFromLibrary: (index: number) => void;
|
||||||
onInsertShape: (elements: LibraryItem) => void;
|
onInsertShape: (elements: LibraryItem) => void;
|
||||||
@@ -127,22 +113,18 @@ const LibraryMenuItems = ({
|
|||||||
setAppState: React.Component<any, AppState>["setState"];
|
setAppState: React.Component<any, AppState>["setState"];
|
||||||
setLibraryItems: (library: LibraryItems) => void;
|
setLibraryItems: (library: LibraryItems) => void;
|
||||||
libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"];
|
libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"];
|
||||||
focusContainer: () => void;
|
|
||||||
library: Library;
|
|
||||||
id: string;
|
|
||||||
}) => {
|
}) => {
|
||||||
const isMobile = useIsMobile();
|
const isMobile = useIsMobile();
|
||||||
const numCells = libraryItems.length + (pendingElements.length > 0 ? 1 : 0);
|
const numCells = library.length + (pendingElements.length > 0 ? 1 : 0);
|
||||||
const CELLS_PER_ROW = isMobile ? 4 : 6;
|
const CELLS_PER_ROW = isMobile ? 4 : 6;
|
||||||
const numRows = Math.max(1, Math.ceil(numCells / CELLS_PER_ROW));
|
const numRows = Math.max(1, Math.ceil(numCells / CELLS_PER_ROW));
|
||||||
const rows = [];
|
const rows = [];
|
||||||
let addedPendingElements = false;
|
let addedPendingElements = false;
|
||||||
|
|
||||||
const referrer =
|
const referrer = libraryReturnUrl || window.location.origin;
|
||||||
libraryReturnUrl || window.location.origin + window.location.pathname;
|
|
||||||
|
|
||||||
rows.push(
|
rows.push(
|
||||||
<div className="layer-ui__library-header" key="library-header">
|
<div className="layer-ui__library-header">
|
||||||
<ToolButton
|
<ToolButton
|
||||||
key="import"
|
key="import"
|
||||||
type="button"
|
type="button"
|
||||||
@@ -150,11 +132,11 @@ const LibraryMenuItems = ({
|
|||||||
aria-label={t("buttons.load")}
|
aria-label={t("buttons.load")}
|
||||||
icon={load}
|
icon={load}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
importLibraryFromJSON(library)
|
importLibraryFromJSON()
|
||||||
.then(() => {
|
.then(() => {
|
||||||
// Close and then open to get the libraries updated
|
// Maybe we should close and open the menu so that the items get updated.
|
||||||
|
// But for now we just close the menu.
|
||||||
setAppState({ isLibraryOpen: false });
|
setAppState({ isLibraryOpen: false });
|
||||||
setAppState({ isLibraryOpen: true });
|
|
||||||
})
|
})
|
||||||
.catch(muteFSAbortError)
|
.catch(muteFSAbortError)
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
@@ -162,42 +144,36 @@ const LibraryMenuItems = ({
|
|||||||
});
|
});
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{!!libraryItems.length && (
|
<ToolButton
|
||||||
<>
|
key="export"
|
||||||
<ToolButton
|
type="button"
|
||||||
key="export"
|
title={t("buttons.export")}
|
||||||
type="button"
|
aria-label={t("buttons.export")}
|
||||||
title={t("buttons.export")}
|
icon={exportFile}
|
||||||
aria-label={t("buttons.export")}
|
onClick={() => {
|
||||||
icon={exportFile}
|
saveLibraryAsJSON()
|
||||||
onClick={() => {
|
.catch(muteFSAbortError)
|
||||||
saveLibraryAsJSON(library)
|
.catch((error) => {
|
||||||
.catch(muteFSAbortError)
|
setAppState({ errorMessage: error.message });
|
||||||
.catch((error) => {
|
});
|
||||||
setAppState({ errorMessage: error.message });
|
}}
|
||||||
});
|
/>
|
||||||
}}
|
<ToolButton
|
||||||
/>
|
key="reset"
|
||||||
<ToolButton
|
type="button"
|
||||||
key="reset"
|
title={t("buttons.resetLibrary")}
|
||||||
type="button"
|
aria-label={t("buttons.resetLibrary")}
|
||||||
title={t("buttons.resetLibrary")}
|
icon={trash}
|
||||||
aria-label={t("buttons.resetLibrary")}
|
onClick={() => {
|
||||||
icon={trash}
|
if (window.confirm(t("alerts.resetLibrary"))) {
|
||||||
onClick={() => {
|
Library.resetLibrary();
|
||||||
if (window.confirm(t("alerts.resetLibrary"))) {
|
setLibraryItems([]);
|
||||||
library.resetLibrary();
|
}
|
||||||
setLibraryItems([]);
|
}}
|
||||||
focusContainer();
|
/>
|
||||||
}
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
<a
|
<a
|
||||||
href={`https://libraries.excalidraw.com?target=${
|
href={`https://libraries.excalidraw.com?referrer=${referrer}`}
|
||||||
window.name || "_blank"
|
|
||||||
}&referrer=${referrer}&useHash=true&token=${id}`}
|
|
||||||
target="_excalidraw_libraries"
|
target="_excalidraw_libraries"
|
||||||
>
|
>
|
||||||
{t("labels.libraries")}
|
{t("labels.libraries")}
|
||||||
@@ -212,13 +188,13 @@ const LibraryMenuItems = ({
|
|||||||
const shouldAddPendingElements: boolean =
|
const shouldAddPendingElements: boolean =
|
||||||
pendingElements.length > 0 &&
|
pendingElements.length > 0 &&
|
||||||
!addedPendingElements &&
|
!addedPendingElements &&
|
||||||
y + x >= libraryItems.length;
|
y + x >= library.length;
|
||||||
addedPendingElements = addedPendingElements || shouldAddPendingElements;
|
addedPendingElements = addedPendingElements || shouldAddPendingElements;
|
||||||
|
|
||||||
children.push(
|
children.push(
|
||||||
<Stack.Col key={x}>
|
<Stack.Col key={x}>
|
||||||
<LibraryUnit
|
<LibraryUnit
|
||||||
elements={libraryItems[y + x]}
|
elements={library[y + x]}
|
||||||
pendingElements={
|
pendingElements={
|
||||||
shouldAddPendingElements ? pendingElements : undefined
|
shouldAddPendingElements ? pendingElements : undefined
|
||||||
}
|
}
|
||||||
@@ -226,7 +202,7 @@ const LibraryMenuItems = ({
|
|||||||
onClick={
|
onClick={
|
||||||
shouldAddPendingElements
|
shouldAddPendingElements
|
||||||
? onAddToLibrary.bind(null, pendingElements)
|
? onAddToLibrary.bind(null, pendingElements)
|
||||||
: onInsertShape.bind(null, libraryItems[y + x])
|
: onInsertShape.bind(null, library[y + x])
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
</Stack.Col>,
|
</Stack.Col>,
|
||||||
@@ -253,9 +229,6 @@ const LibraryMenu = ({
|
|||||||
onAddToLibrary,
|
onAddToLibrary,
|
||||||
setAppState,
|
setAppState,
|
||||||
libraryReturnUrl,
|
libraryReturnUrl,
|
||||||
focusContainer,
|
|
||||||
library,
|
|
||||||
id,
|
|
||||||
}: {
|
}: {
|
||||||
pendingElements: LibraryItem;
|
pendingElements: LibraryItem;
|
||||||
onClickOutside: (event: MouseEvent) => void;
|
onClickOutside: (event: MouseEvent) => void;
|
||||||
@@ -263,9 +236,6 @@ const LibraryMenu = ({
|
|||||||
onAddToLibrary: () => void;
|
onAddToLibrary: () => void;
|
||||||
setAppState: React.Component<any, AppState>["setState"];
|
setAppState: React.Component<any, AppState>["setState"];
|
||||||
libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"];
|
libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"];
|
||||||
focusContainer: () => void;
|
|
||||||
library: Library;
|
|
||||||
id: string;
|
|
||||||
}) => {
|
}) => {
|
||||||
const ref = useRef<HTMLDivElement | null>(null);
|
const ref = useRef<HTMLDivElement | null>(null);
|
||||||
useOnClickOutside(ref, (event) => {
|
useOnClickOutside(ref, (event) => {
|
||||||
@@ -291,7 +261,7 @@ const LibraryMenu = ({
|
|||||||
resolve("loading");
|
resolve("loading");
|
||||||
}, 100);
|
}, 100);
|
||||||
}),
|
}),
|
||||||
library.loadLibrary().then((items) => {
|
Library.loadLibrary().then((items) => {
|
||||||
setLibraryItems(items);
|
setLibraryItems(items);
|
||||||
setIsLoading("ready");
|
setIsLoading("ready");
|
||||||
}),
|
}),
|
||||||
@@ -303,33 +273,24 @@ const LibraryMenu = ({
|
|||||||
return () => {
|
return () => {
|
||||||
clearTimeout(loadingTimerRef.current!);
|
clearTimeout(loadingTimerRef.current!);
|
||||||
};
|
};
|
||||||
}, [library]);
|
}, []);
|
||||||
|
|
||||||
const removeFromLibrary = useCallback(
|
const removeFromLibrary = useCallback(async (indexToRemove) => {
|
||||||
async (indexToRemove) => {
|
const items = await Library.loadLibrary();
|
||||||
const items = await library.loadLibrary();
|
const nextItems = items.filter((_, index) => index !== indexToRemove);
|
||||||
const nextItems = items.filter((_, index) => index !== indexToRemove);
|
Library.saveLibrary(nextItems);
|
||||||
library.saveLibrary(nextItems).catch((error) => {
|
setLibraryItems(nextItems);
|
||||||
setLibraryItems(items);
|
}, []);
|
||||||
setAppState({ errorMessage: t("alerts.errorRemovingFromLibrary") });
|
|
||||||
});
|
|
||||||
setLibraryItems(nextItems);
|
|
||||||
},
|
|
||||||
[library, setAppState],
|
|
||||||
);
|
|
||||||
|
|
||||||
const addToLibrary = useCallback(
|
const addToLibrary = useCallback(
|
||||||
async (elements: LibraryItem) => {
|
async (elements: LibraryItem) => {
|
||||||
const items = await library.loadLibrary();
|
const items = await Library.loadLibrary();
|
||||||
const nextItems = [...items, elements];
|
const nextItems = [...items, elements];
|
||||||
onAddToLibrary();
|
onAddToLibrary();
|
||||||
library.saveLibrary(nextItems).catch((error) => {
|
Library.saveLibrary(nextItems);
|
||||||
setLibraryItems(items);
|
|
||||||
setAppState({ errorMessage: t("alerts.errorAddingToLibrary") });
|
|
||||||
});
|
|
||||||
setLibraryItems(nextItems);
|
setLibraryItems(nextItems);
|
||||||
},
|
},
|
||||||
[onAddToLibrary, library, setAppState],
|
[onAddToLibrary],
|
||||||
);
|
);
|
||||||
|
|
||||||
return loadingState === "preloading" ? null : (
|
return loadingState === "preloading" ? null : (
|
||||||
@@ -340,7 +301,7 @@ const LibraryMenu = ({
|
|||||||
</div>
|
</div>
|
||||||
) : (
|
) : (
|
||||||
<LibraryMenuItems
|
<LibraryMenuItems
|
||||||
libraryItems={libraryItems}
|
library={libraryItems}
|
||||||
onRemoveFromLibrary={removeFromLibrary}
|
onRemoveFromLibrary={removeFromLibrary}
|
||||||
onAddToLibrary={addToLibrary}
|
onAddToLibrary={addToLibrary}
|
||||||
onInsertShape={onInsertShape}
|
onInsertShape={onInsertShape}
|
||||||
@@ -348,9 +309,6 @@ const LibraryMenu = ({
|
|||||||
setAppState={setAppState}
|
setAppState={setAppState}
|
||||||
setLibraryItems={setLibraryItems}
|
setLibraryItems={setLibraryItems}
|
||||||
libraryReturnUrl={libraryReturnUrl}
|
libraryReturnUrl={libraryReturnUrl}
|
||||||
focusContainer={focusContainer}
|
|
||||||
library={library}
|
|
||||||
id={id}
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</Island>
|
</Island>
|
||||||
@@ -372,27 +330,56 @@ const LayerUI = ({
|
|||||||
toggleZenMode,
|
toggleZenMode,
|
||||||
isCollaborating,
|
isCollaborating,
|
||||||
onExportToBackend,
|
onExportToBackend,
|
||||||
renderTopRightUI,
|
|
||||||
renderCustomFooter,
|
renderCustomFooter,
|
||||||
viewModeEnabled,
|
viewModeEnabled,
|
||||||
libraryReturnUrl,
|
libraryReturnUrl,
|
||||||
UIOptions,
|
|
||||||
focusContainer,
|
|
||||||
library,
|
|
||||||
id,
|
|
||||||
}: LayerUIProps) => {
|
}: LayerUIProps) => {
|
||||||
const isMobile = useIsMobile();
|
const isMobile = useIsMobile();
|
||||||
|
|
||||||
const renderJSONExportDialog = () => {
|
const renderEncryptedIcon = () => (
|
||||||
if (!UIOptions.canvasActions.export) {
|
<a
|
||||||
return null;
|
className={clsx("encrypted-icon tooltip zen-mode-visibility", {
|
||||||
}
|
"zen-mode-visibility--hidden": zenModeEnabled,
|
||||||
|
})}
|
||||||
|
href="https://blog.excalidraw.com/end-to-end-encryption/"
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
>
|
||||||
|
<Tooltip label={t("encrypted.tooltip")} position="above" long={true}>
|
||||||
|
{shield}
|
||||||
|
</Tooltip>
|
||||||
|
</a>
|
||||||
|
);
|
||||||
|
|
||||||
|
const renderExportDialog = () => {
|
||||||
|
const createExporter = (type: ExportType): ExportCB => async (
|
||||||
|
exportedElements,
|
||||||
|
scale,
|
||||||
|
) => {
|
||||||
|
if (canvas) {
|
||||||
|
await exportCanvas(type, exportedElements, appState, canvas, {
|
||||||
|
exportBackground: appState.exportBackground,
|
||||||
|
name: appState.name,
|
||||||
|
viewBackgroundColor: appState.viewBackgroundColor,
|
||||||
|
scale,
|
||||||
|
shouldAddWatermark: appState.shouldAddWatermark,
|
||||||
|
})
|
||||||
|
.catch(muteFSAbortError)
|
||||||
|
.catch((error) => {
|
||||||
|
console.error(error);
|
||||||
|
setAppState({ errorMessage: error.message });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<JSONExportDialog
|
<ExportDialog
|
||||||
elements={elements}
|
elements={elements}
|
||||||
appState={appState}
|
appState={appState}
|
||||||
actionManager={actionManager}
|
actionManager={actionManager}
|
||||||
|
onExportToPng={createExporter("png")}
|
||||||
|
onExportToSvg={createExporter("svg")}
|
||||||
|
onExportToClipboard={createExporter("clipboard")}
|
||||||
onExportToBackend={
|
onExportToBackend={
|
||||||
onExportToBackend
|
onExportToBackend
|
||||||
? (elements) => {
|
? (elements) => {
|
||||||
@@ -405,44 +392,6 @@ const LayerUI = ({
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const renderImageExportDialog = () => {
|
|
||||||
if (!UIOptions.canvasActions.export) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
const createExporter = (type: ExportType): ExportCB => async (
|
|
||||||
exportedElements,
|
|
||||||
scale,
|
|
||||||
) => {
|
|
||||||
await exportCanvas(type, exportedElements, appState, {
|
|
||||||
exportBackground: appState.exportBackground,
|
|
||||||
name: appState.name,
|
|
||||||
viewBackgroundColor: appState.viewBackgroundColor,
|
|
||||||
scale,
|
|
||||||
})
|
|
||||||
.catch(muteFSAbortError)
|
|
||||||
.catch((error) => {
|
|
||||||
console.error(error);
|
|
||||||
setAppState({ errorMessage: error.message });
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<ImageExportDialog
|
|
||||||
elements={elements}
|
|
||||||
appState={appState}
|
|
||||||
actionManager={actionManager}
|
|
||||||
onExportToPng={createExporter("png")}
|
|
||||||
onExportToSvg={createExporter("svg")}
|
|
||||||
onExportToClipboard={createExporter("clipboard")}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const Separator = () => {
|
|
||||||
return <div style={{ width: ".625em" }} />;
|
|
||||||
};
|
|
||||||
|
|
||||||
const renderViewModeCanvasActions = () => {
|
const renderViewModeCanvasActions = () => {
|
||||||
return (
|
return (
|
||||||
<Section
|
<Section
|
||||||
@@ -456,8 +405,9 @@ const LayerUI = ({
|
|||||||
<Island padding={2} style={{ zIndex: 1 }}>
|
<Island padding={2} style={{ zIndex: 1 }}>
|
||||||
<Stack.Col gap={4}>
|
<Stack.Col gap={4}>
|
||||||
<Stack.Row gap={1} justifyContent="space-between">
|
<Stack.Row gap={1} justifyContent="space-between">
|
||||||
{renderJSONExportDialog()}
|
{actionManager.renderAction("saveScene")}
|
||||||
{renderImageExportDialog()}
|
{actionManager.renderAction("saveAsScene")}
|
||||||
|
{renderExportDialog()}
|
||||||
</Stack.Row>
|
</Stack.Row>
|
||||||
</Stack.Col>
|
</Stack.Col>
|
||||||
</Island>
|
</Island>
|
||||||
@@ -476,12 +426,11 @@ const LayerUI = ({
|
|||||||
<Island padding={2} style={{ zIndex: 1 }}>
|
<Island padding={2} style={{ zIndex: 1 }}>
|
||||||
<Stack.Col gap={4}>
|
<Stack.Col gap={4}>
|
||||||
<Stack.Row gap={1} justifyContent="space-between">
|
<Stack.Row gap={1} justifyContent="space-between">
|
||||||
{actionManager.renderAction("clearCanvas")}
|
|
||||||
<Separator />
|
|
||||||
{actionManager.renderAction("loadScene")}
|
{actionManager.renderAction("loadScene")}
|
||||||
{renderJSONExportDialog()}
|
{actionManager.renderAction("saveScene")}
|
||||||
{renderImageExportDialog()}
|
{actionManager.renderAction("saveAsScene")}
|
||||||
<Separator />
|
{renderExportDialog()}
|
||||||
|
{actionManager.renderAction("clearCanvas")}
|
||||||
{onCollabButtonClick && (
|
{onCollabButtonClick && (
|
||||||
<CollabButton
|
<CollabButton
|
||||||
isCollaborating={isCollaborating}
|
isCollaborating={isCollaborating}
|
||||||
@@ -549,9 +498,6 @@ const LayerUI = ({
|
|||||||
onAddToLibrary={deselectItems}
|
onAddToLibrary={deselectItems}
|
||||||
setAppState={setAppState}
|
setAppState={setAppState}
|
||||||
libraryReturnUrl={libraryReturnUrl}
|
libraryReturnUrl={libraryReturnUrl}
|
||||||
focusContainer={focusContainer}
|
|
||||||
library={library}
|
|
||||||
id={id}
|
|
||||||
/>
|
/>
|
||||||
) : null;
|
) : null;
|
||||||
|
|
||||||
@@ -605,30 +551,24 @@ const LayerUI = ({
|
|||||||
)}
|
)}
|
||||||
</Section>
|
</Section>
|
||||||
)}
|
)}
|
||||||
<div
|
<UserList
|
||||||
className={clsx(
|
className={clsx("zen-mode-transition", {
|
||||||
"layer-ui__wrapper__top-right zen-mode-transition",
|
"transition-right": zenModeEnabled,
|
||||||
{
|
})}
|
||||||
"transition-right": zenModeEnabled,
|
|
||||||
},
|
|
||||||
)}
|
|
||||||
>
|
>
|
||||||
<UserList>
|
{appState.collaborators.size > 0 &&
|
||||||
{appState.collaborators.size > 0 &&
|
Array.from(appState.collaborators)
|
||||||
Array.from(appState.collaborators)
|
// Collaborator is either not initialized or is actually the current user.
|
||||||
// Collaborator is either not initialized or is actually the current user.
|
.filter(([_, client]) => Object.keys(client).length !== 0)
|
||||||
.filter(([_, client]) => Object.keys(client).length !== 0)
|
.map(([clientId, client]) => (
|
||||||
.map(([clientId, client]) => (
|
<Tooltip
|
||||||
<Tooltip
|
label={client.username || "Unknown user"}
|
||||||
label={client.username || "Unknown user"}
|
key={clientId}
|
||||||
key={clientId}
|
>
|
||||||
>
|
{actionManager.renderAction("goToCollaborator", clientId)}
|
||||||
{actionManager.renderAction("goToCollaborator", clientId)}
|
</Tooltip>
|
||||||
</Tooltip>
|
))}
|
||||||
))}
|
</UserList>
|
||||||
</UserList>
|
|
||||||
{renderTopRightUI?.(isMobile, appState)}
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</FixedSideContainer>
|
</FixedSideContainer>
|
||||||
);
|
);
|
||||||
@@ -636,61 +576,61 @@ const LayerUI = ({
|
|||||||
|
|
||||||
const renderBottomAppMenu = () => {
|
const renderBottomAppMenu = () => {
|
||||||
return (
|
return (
|
||||||
<footer
|
<div
|
||||||
role="contentinfo"
|
className={clsx("App-menu App-menu_bottom zen-mode-transition", {
|
||||||
className="layer-ui__wrapper__footer App-menu App-menu_bottom"
|
"App-menu_bottom--transition-left": zenModeEnabled,
|
||||||
|
})}
|
||||||
>
|
>
|
||||||
<div
|
<Stack.Col gap={2}>
|
||||||
className={clsx(
|
<Section heading="canvasActions">
|
||||||
"layer-ui__wrapper__footer-left zen-mode-transition",
|
<Island padding={1}>
|
||||||
{
|
<ZoomActions
|
||||||
"layer-ui__wrapper__footer-left--transition-left": zenModeEnabled,
|
renderAction={actionManager.renderAction}
|
||||||
},
|
zoom={appState.zoom}
|
||||||
)}
|
/>
|
||||||
>
|
</Island>
|
||||||
<Stack.Col gap={2}>
|
{renderEncryptedIcon()}
|
||||||
<Section heading="canvasActions">
|
</Section>
|
||||||
<Island padding={1}>
|
</Stack.Col>
|
||||||
<ZoomActions
|
</div>
|
||||||
renderAction={actionManager.renderAction}
|
|
||||||
zoom={appState.zoom}
|
|
||||||
/>
|
|
||||||
</Island>
|
|
||||||
</Section>
|
|
||||||
</Stack.Col>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className={clsx(
|
|
||||||
"layer-ui__wrapper__footer-center zen-mode-transition",
|
|
||||||
{
|
|
||||||
"layer-ui__wrapper__footer-left--transition-bottom": zenModeEnabled,
|
|
||||||
},
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
{renderCustomFooter?.(false, appState)}
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className={clsx(
|
|
||||||
"layer-ui__wrapper__footer-right zen-mode-transition",
|
|
||||||
{
|
|
||||||
"transition-right disable-pointerEvents": zenModeEnabled,
|
|
||||||
},
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
{actionManager.renderAction("toggleShortcuts")}
|
|
||||||
</div>
|
|
||||||
<button
|
|
||||||
className={clsx("disable-zen-mode", {
|
|
||||||
"disable-zen-mode--visible": showExitZenModeBtn,
|
|
||||||
})}
|
|
||||||
onClick={toggleZenMode}
|
|
||||||
>
|
|
||||||
{t("buttons.exitZenMode")}
|
|
||||||
</button>
|
|
||||||
</footer>
|
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const renderGitHubCorner = () => {
|
||||||
|
return (
|
||||||
|
<aside
|
||||||
|
className={clsx(
|
||||||
|
"layer-ui__wrapper__github-corner zen-mode-transition",
|
||||||
|
{
|
||||||
|
"transition-right": zenModeEnabled,
|
||||||
|
},
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<GitHubCorner theme={appState.theme} />
|
||||||
|
</aside>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
const renderFooter = () => (
|
||||||
|
<footer role="contentinfo" className="layer-ui__wrapper__footer">
|
||||||
|
<div
|
||||||
|
className={clsx("zen-mode-transition", {
|
||||||
|
"transition-right disable-pointerEvents": zenModeEnabled,
|
||||||
|
})}
|
||||||
|
>
|
||||||
|
{renderCustomFooter?.(false)}
|
||||||
|
{actionManager.renderAction("toggleShortcuts")}
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
className={clsx("disable-zen-mode", {
|
||||||
|
"disable-zen-mode--visible": showExitZenModeBtn,
|
||||||
|
})}
|
||||||
|
onClick={toggleZenMode}
|
||||||
|
>
|
||||||
|
{t("buttons.exitZenMode")}
|
||||||
|
</button>
|
||||||
|
</footer>
|
||||||
|
);
|
||||||
|
|
||||||
const dialogs = (
|
const dialogs = (
|
||||||
<>
|
<>
|
||||||
{appState.isLoading && <LoadingMessage />}
|
{appState.isLoading && <LoadingMessage />}
|
||||||
@@ -701,11 +641,7 @@ const LayerUI = ({
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{appState.showHelpDialog && (
|
{appState.showHelpDialog && (
|
||||||
<HelpDialog
|
<HelpDialog onClose={() => setAppState({ showHelpDialog: false })} />
|
||||||
onClose={() => {
|
|
||||||
setAppState({ showHelpDialog: false });
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
)}
|
)}
|
||||||
{appState.pasteDialog.shown && (
|
{appState.pasteDialog.shown && (
|
||||||
<PasteChartDialog
|
<PasteChartDialog
|
||||||
@@ -730,8 +666,7 @@ const LayerUI = ({
|
|||||||
elements={elements}
|
elements={elements}
|
||||||
actionManager={actionManager}
|
actionManager={actionManager}
|
||||||
libraryMenu={libraryMenu}
|
libraryMenu={libraryMenu}
|
||||||
renderJSONExportDialog={renderJSONExportDialog}
|
exportButton={renderExportDialog()}
|
||||||
renderImageExportDialog={renderImageExportDialog}
|
|
||||||
setAppState={setAppState}
|
setAppState={setAppState}
|
||||||
onCollabButtonClick={onCollabButtonClick}
|
onCollabButtonClick={onCollabButtonClick}
|
||||||
onLockToggle={onLockToggle}
|
onLockToggle={onLockToggle}
|
||||||
@@ -754,6 +689,8 @@ const LayerUI = ({
|
|||||||
{dialogs}
|
{dialogs}
|
||||||
{renderFixedSideContainer()}
|
{renderFixedSideContainer()}
|
||||||
{renderBottomAppMenu()}
|
{renderBottomAppMenu()}
|
||||||
|
{renderGitHubCorner()}
|
||||||
|
{renderFooter()}
|
||||||
{appState.scrolledOutside && (
|
{appState.scrolledOutside && (
|
||||||
<button
|
<button
|
||||||
className="scroll-back-to-content"
|
className="scroll-back-to-content"
|
||||||
|
@@ -4,7 +4,7 @@ import React, { useEffect, useRef, useState } from "react";
|
|||||||
import { close } from "../components/icons";
|
import { close } from "../components/icons";
|
||||||
import { MIME_TYPES } from "../constants";
|
import { MIME_TYPES } from "../constants";
|
||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
import { useIsMobile } from "../components/App";
|
import useIsMobile from "../is-mobile";
|
||||||
import { exportToSvg } from "../scene/export";
|
import { exportToSvg } from "../scene/export";
|
||||||
import { LibraryItem } from "../types";
|
import { LibraryItem } from "../types";
|
||||||
import "./LibraryUnit.scss";
|
import "./LibraryUnit.scss";
|
||||||
@@ -39,6 +39,7 @@ export const LibraryUnit = ({
|
|||||||
const svg = exportToSvg(elementsToRender, {
|
const svg = exportToSvg(elementsToRender, {
|
||||||
exportBackground: false,
|
exportBackground: false,
|
||||||
viewBackgroundColor: oc.white,
|
viewBackgroundColor: oc.white,
|
||||||
|
shouldAddWatermark: false,
|
||||||
});
|
});
|
||||||
for (const child of ref.current!.children) {
|
for (const child of ref.current!.children) {
|
||||||
if (child.tagName !== "svg") {
|
if (child.tagName !== "svg") {
|
||||||
|
@@ -20,8 +20,7 @@ import { BackgroundPickerAndDarkModeToggle } from "./BackgroundPickerAndDarkMode
|
|||||||
type MobileMenuProps = {
|
type MobileMenuProps = {
|
||||||
appState: AppState;
|
appState: AppState;
|
||||||
actionManager: ActionManager;
|
actionManager: ActionManager;
|
||||||
renderJSONExportDialog: () => React.ReactNode;
|
exportButton: React.ReactNode;
|
||||||
renderImageExportDialog: () => React.ReactNode;
|
|
||||||
setAppState: React.Component<any, AppState>["setState"];
|
setAppState: React.Component<any, AppState>["setState"];
|
||||||
elements: readonly NonDeletedExcalidrawElement[];
|
elements: readonly NonDeletedExcalidrawElement[];
|
||||||
libraryMenu: JSX.Element | null;
|
libraryMenu: JSX.Element | null;
|
||||||
@@ -29,7 +28,7 @@ type MobileMenuProps = {
|
|||||||
onLockToggle: () => void;
|
onLockToggle: () => void;
|
||||||
canvas: HTMLCanvasElement | null;
|
canvas: HTMLCanvasElement | null;
|
||||||
isCollaborating: boolean;
|
isCollaborating: boolean;
|
||||||
renderCustomFooter?: (isMobile: boolean, appState: AppState) => JSX.Element;
|
renderCustomFooter?: (isMobile: boolean) => JSX.Element;
|
||||||
viewModeEnabled: boolean;
|
viewModeEnabled: boolean;
|
||||||
showThemeBtn: boolean;
|
showThemeBtn: boolean;
|
||||||
};
|
};
|
||||||
@@ -39,8 +38,7 @@ export const MobileMenu = ({
|
|||||||
elements,
|
elements,
|
||||||
libraryMenu,
|
libraryMenu,
|
||||||
actionManager,
|
actionManager,
|
||||||
renderJSONExportDialog,
|
exportButton,
|
||||||
renderImageExportDialog,
|
|
||||||
setAppState,
|
setAppState,
|
||||||
onCollabButtonClick,
|
onCollabButtonClick,
|
||||||
onLockToggle,
|
onLockToggle,
|
||||||
@@ -109,17 +107,19 @@ export const MobileMenu = ({
|
|||||||
if (viewModeEnabled) {
|
if (viewModeEnabled) {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{renderJSONExportDialog()}
|
{actionManager.renderAction("saveScene")}
|
||||||
{renderImageExportDialog()}
|
{actionManager.renderAction("saveAsScene")}
|
||||||
|
{exportButton}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{actionManager.renderAction("clearCanvas")}
|
|
||||||
{actionManager.renderAction("loadScene")}
|
{actionManager.renderAction("loadScene")}
|
||||||
{renderJSONExportDialog()}
|
{actionManager.renderAction("saveScene")}
|
||||||
{renderImageExportDialog()}
|
{actionManager.renderAction("saveAsScene")}
|
||||||
|
{exportButton}
|
||||||
|
{actionManager.renderAction("clearCanvas")}
|
||||||
{onCollabButtonClick && (
|
{onCollabButtonClick && (
|
||||||
<CollabButton
|
<CollabButton
|
||||||
isCollaborating={isCollaborating}
|
isCollaborating={isCollaborating}
|
||||||
@@ -155,7 +155,7 @@ export const MobileMenu = ({
|
|||||||
<div className="panelColumn">
|
<div className="panelColumn">
|
||||||
<Stack.Col gap={4}>
|
<Stack.Col gap={4}>
|
||||||
{renderCanvasActions()}
|
{renderCanvasActions()}
|
||||||
{renderCustomFooter?.(true, appState)}
|
{renderCustomFooter?.(true)}
|
||||||
{appState.collaborators.size > 0 && (
|
{appState.collaborators.size > 0 && (
|
||||||
<fieldset>
|
<fieldset>
|
||||||
<legend>{t("labels.collaborators")}</legend>
|
<legend>{t("labels.collaborators")}</legend>
|
||||||
|
@@ -26,7 +26,8 @@
|
|||||||
right: 0;
|
right: 0;
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
z-index: 1;
|
z-index: 1;
|
||||||
background-color: transparentize($oc-black, 0.3);
|
background-color: transparentize($oc-black, 0.7);
|
||||||
|
backdrop-filter: blur(2px);
|
||||||
}
|
}
|
||||||
|
|
||||||
.Modal__content {
|
.Modal__content {
|
||||||
@@ -44,17 +45,14 @@
|
|||||||
|
|
||||||
// for modals, reset blurry bg
|
// for modals, reset blurry bg
|
||||||
background: var(--island-bg-color);
|
background: var(--island-bg-color);
|
||||||
|
backdrop-filter: none;
|
||||||
|
|
||||||
border: 1px solid var(--dialog-border-color);
|
border: 1px solid var(--dialog-border-color);
|
||||||
box-shadow: 0 2px 10px transparentize($oc-black, 0.75);
|
box-shadow: 0 2px 10px transparentize($oc-black, 0.75);
|
||||||
border-radius: 6px;
|
border-radius: 6px;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
|
|
||||||
&:focus {
|
@media #{$is-mobile-query} {
|
||||||
outline: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
@include isMobile {
|
|
||||||
max-width: 100%;
|
max-width: 100%;
|
||||||
border: 0;
|
border: 0;
|
||||||
border-radius: 0;
|
border-radius: 0;
|
||||||
@@ -84,7 +82,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@include isMobile {
|
@media #{$is-mobile-query} {
|
||||||
.Modal {
|
.Modal {
|
||||||
padding: 0;
|
padding: 0;
|
||||||
}
|
}
|
||||||
|
@@ -1,11 +1,9 @@
|
|||||||
import "./Modal.scss";
|
import "./Modal.scss";
|
||||||
|
|
||||||
import React, { useState, useLayoutEffect, useRef } from "react";
|
import React, { useState, useLayoutEffect } from "react";
|
||||||
import { createPortal } from "react-dom";
|
import { createPortal } from "react-dom";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import { KEYS } from "../keys";
|
import { KEYS } from "../keys";
|
||||||
import { useExcalidrawContainer, useIsMobile } from "./App";
|
|
||||||
import { AppState } from "../types";
|
|
||||||
|
|
||||||
export const Modal = (props: {
|
export const Modal = (props: {
|
||||||
className?: string;
|
className?: string;
|
||||||
@@ -13,10 +11,8 @@ export const Modal = (props: {
|
|||||||
maxWidth?: number;
|
maxWidth?: number;
|
||||||
onCloseRequest(): void;
|
onCloseRequest(): void;
|
||||||
labelledBy: string;
|
labelledBy: string;
|
||||||
theme?: AppState["theme"];
|
|
||||||
}) => {
|
}) => {
|
||||||
const { theme = "light" } = props;
|
const modalRoot = useBodyRoot();
|
||||||
const modalRoot = useBodyRoot(theme);
|
|
||||||
|
|
||||||
if (!modalRoot) {
|
if (!modalRoot) {
|
||||||
return null;
|
return null;
|
||||||
@@ -25,7 +21,6 @@ export const Modal = (props: {
|
|||||||
const handleKeydown = (event: React.KeyboardEvent) => {
|
const handleKeydown = (event: React.KeyboardEvent) => {
|
||||||
if (event.key === KEYS.ESCAPE) {
|
if (event.key === KEYS.ESCAPE) {
|
||||||
event.nativeEvent.stopImmediatePropagation();
|
event.nativeEvent.stopImmediatePropagation();
|
||||||
event.stopPropagation();
|
|
||||||
props.onCloseRequest();
|
props.onCloseRequest();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@@ -42,7 +37,6 @@ export const Modal = (props: {
|
|||||||
<div
|
<div
|
||||||
className="Modal__content"
|
className="Modal__content"
|
||||||
style={{ "--max-width": `${props.maxWidth}px` }}
|
style={{ "--max-width": `${props.maxWidth}px` }}
|
||||||
tabIndex={0}
|
|
||||||
>
|
>
|
||||||
{props.children}
|
{props.children}
|
||||||
</div>
|
</div>
|
||||||
@@ -51,29 +45,16 @@ export const Modal = (props: {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const useBodyRoot = (theme: AppState["theme"]) => {
|
const useBodyRoot = () => {
|
||||||
const [div, setDiv] = useState<HTMLDivElement | null>(null);
|
const [div, setDiv] = useState<HTMLDivElement | null>(null);
|
||||||
|
|
||||||
const isMobile = useIsMobile();
|
|
||||||
const isMobileRef = useRef(isMobile);
|
|
||||||
isMobileRef.current = isMobile;
|
|
||||||
|
|
||||||
const excalidrawContainer = useExcalidrawContainer();
|
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
useLayoutEffect(() => {
|
||||||
if (div) {
|
const isDarkTheme = !!document
|
||||||
div.classList.toggle("excalidraw--mobile", isMobile);
|
.querySelector(".excalidraw")
|
||||||
}
|
?.classList.contains("theme--dark");
|
||||||
}, [div, isMobile]);
|
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
|
||||||
const isDarkTheme =
|
|
||||||
!!excalidrawContainer?.classList.contains("theme--dark") ||
|
|
||||||
theme === "dark";
|
|
||||||
const div = document.createElement("div");
|
const div = document.createElement("div");
|
||||||
|
|
||||||
div.classList.add("excalidraw", "excalidraw-modal-container");
|
div.classList.add("excalidraw", "excalidraw-modal-container");
|
||||||
div.classList.toggle("excalidraw--mobile", isMobileRef.current);
|
|
||||||
|
|
||||||
if (isDarkTheme) {
|
if (isDarkTheme) {
|
||||||
div.classList.add("theme--dark");
|
div.classList.add("theme--dark");
|
||||||
@@ -86,7 +67,7 @@ const useBodyRoot = (theme: AppState["theme"]) => {
|
|||||||
return () => {
|
return () => {
|
||||||
document.body.removeChild(div);
|
document.body.removeChild(div);
|
||||||
};
|
};
|
||||||
}, [excalidrawContainer, theme]);
|
}, []);
|
||||||
|
|
||||||
return div;
|
return div;
|
||||||
};
|
};
|
||||||
|
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
.excalidraw {
|
.excalidraw {
|
||||||
.PasteChartDialog {
|
.PasteChartDialog {
|
||||||
@include isMobile {
|
@media #{$is-mobile-query} {
|
||||||
.Island {
|
.Island {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
@@ -13,7 +13,7 @@
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: space-around;
|
justify-content: space-around;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
@include isMobile {
|
@media #{$is-mobile-query} {
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
}
|
}
|
||||||
|
@@ -38,6 +38,7 @@ const ChartPreviewBtn = (props: {
|
|||||||
const svg = exportToSvg(elements, {
|
const svg = exportToSvg(elements, {
|
||||||
exportBackground: false,
|
exportBackground: false,
|
||||||
viewBackgroundColor: oc.white,
|
viewBackgroundColor: oc.white,
|
||||||
|
shouldAddWatermark: false,
|
||||||
});
|
});
|
||||||
|
|
||||||
const previewNode = previewRef.current!;
|
const previewNode = previewRef.current!;
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
.excalidraw {
|
.excalidraw {
|
||||||
.popover {
|
.popover {
|
||||||
position: absolute;
|
position: fixed;
|
||||||
z-index: 10;
|
z-index: 10;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,25 +0,0 @@
|
|||||||
.ProjectName {
|
|
||||||
margin: auto;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
|
|
||||||
.TextInput {
|
|
||||||
height: calc(1rem - 3px);
|
|
||||||
width: 200px;
|
|
||||||
overflow: hidden;
|
|
||||||
text-align: center;
|
|
||||||
margin-left: 8px;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
|
|
||||||
&--readonly {
|
|
||||||
background: none;
|
|
||||||
border: none;
|
|
||||||
&:hover {
|
|
||||||
background: none;
|
|
||||||
}
|
|
||||||
width: auto;
|
|
||||||
max-width: 200px;
|
|
||||||
padding-left: 2px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,9 +1,7 @@
|
|||||||
import "./TextInput.scss";
|
import "./TextInput.scss";
|
||||||
|
|
||||||
import React, { Component } from "react";
|
import React, { Component } from "react";
|
||||||
import { focusNearestParent } from "../utils";
|
import { selectNode, removeSelection } from "../utils";
|
||||||
|
|
||||||
import "./ProjectName.scss";
|
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
value: string;
|
value: string;
|
||||||
@@ -12,19 +10,17 @@ type Props = {
|
|||||||
isNameEditable: boolean;
|
isNameEditable: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
type State = {
|
export class ProjectName extends Component<Props> {
|
||||||
fileName: string;
|
private handleFocus = (event: React.FocusEvent<HTMLElement>) => {
|
||||||
};
|
selectNode(event.currentTarget);
|
||||||
export class ProjectName extends Component<Props, State> {
|
|
||||||
state = {
|
|
||||||
fileName: this.props.value,
|
|
||||||
};
|
};
|
||||||
private handleBlur = (event: any) => {
|
|
||||||
focusNearestParent(event.target);
|
private handleBlur = (event: React.FocusEvent<HTMLElement>) => {
|
||||||
const value = event.target.value;
|
const value = event.currentTarget.innerText.trim();
|
||||||
if (value !== this.props.value) {
|
if (value !== this.props.value) {
|
||||||
this.props.onChange(value);
|
this.props.onChange(value);
|
||||||
}
|
}
|
||||||
|
removeSelection();
|
||||||
};
|
};
|
||||||
|
|
||||||
private handleKeyDown = (event: React.KeyboardEvent<HTMLElement>) => {
|
private handleKeyDown = (event: React.KeyboardEvent<HTMLElement>) => {
|
||||||
@@ -36,30 +32,39 @@ export class ProjectName extends Component<Props, State> {
|
|||||||
event.currentTarget.blur();
|
event.currentTarget.blur();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
private makeEditable = (editable: HTMLSpanElement | null) => {
|
||||||
|
if (!editable) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
editable.contentEditable = "plaintext-only";
|
||||||
|
} catch {
|
||||||
|
editable.contentEditable = "true";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
public render() {
|
public render() {
|
||||||
return (
|
return this.props.isNameEditable ? (
|
||||||
<div className="ProjectName">
|
<span
|
||||||
<label className="ProjectName-label" htmlFor="filename">
|
suppressContentEditableWarning
|
||||||
{`${this.props.label}${this.props.isNameEditable ? "" : ":"}`}
|
ref={this.makeEditable}
|
||||||
</label>
|
data-type="wysiwyg"
|
||||||
{this.props.isNameEditable ? (
|
className="TextInput"
|
||||||
<input
|
role="textbox"
|
||||||
className="TextInput"
|
aria-label={this.props.label}
|
||||||
onBlur={this.handleBlur}
|
onBlur={this.handleBlur}
|
||||||
onKeyDown={this.handleKeyDown}
|
onKeyDown={this.handleKeyDown}
|
||||||
id="filename"
|
onFocus={this.handleFocus}
|
||||||
value={this.state.fileName}
|
>
|
||||||
onChange={(event) =>
|
{this.props.value}
|
||||||
this.setState({ fileName: event.target.value })
|
</span>
|
||||||
}
|
) : (
|
||||||
/>
|
<span
|
||||||
) : (
|
className="TextInput TextInput--readonly"
|
||||||
<span className="TextInput TextInput--readonly" id="filename">
|
aria-label={this.props.label}
|
||||||
{this.props.value}
|
>
|
||||||
</span>
|
{this.props.value}
|
||||||
)}
|
</span>
|
||||||
</div>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -6,7 +6,7 @@
|
|||||||
top: 64px;
|
top: 64px;
|
||||||
right: 12px;
|
right: 12px;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
z-index: 10;
|
z-index: 999;
|
||||||
|
|
||||||
h3 {
|
h3 {
|
||||||
margin: 0 24px 8px 0;
|
margin: 0 24px 8px 0;
|
||||||
|
@@ -1,22 +1,49 @@
|
|||||||
import React from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
|
import { copyTextToSystemClipboard } from "../clipboard";
|
||||||
|
import { DEFAULT_VERSION } from "../constants";
|
||||||
import { getCommonBounds } from "../element/bounds";
|
import { getCommonBounds } from "../element/bounds";
|
||||||
import { NonDeletedExcalidrawElement } from "../element/types";
|
import { NonDeletedExcalidrawElement } from "../element/types";
|
||||||
|
import {
|
||||||
|
getElementsStorageSize,
|
||||||
|
getTotalStorageSize,
|
||||||
|
} from "../excalidraw-app/data/localStorage";
|
||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
import { useIsMobile } from "../components/App";
|
import useIsMobile from "../is-mobile";
|
||||||
import { getTargetElements } from "../scene";
|
import { getTargetElements } from "../scene";
|
||||||
import { AppState, ExcalidrawProps } from "../types";
|
import { AppState } from "../types";
|
||||||
|
import { debounce, getVersion, nFormatter } from "../utils";
|
||||||
import { close } from "./icons";
|
import { close } from "./icons";
|
||||||
import { Island } from "./Island";
|
import { Island } from "./Island";
|
||||||
import "./Stats.scss";
|
import "./Stats.scss";
|
||||||
|
|
||||||
|
type StorageSizes = { scene: number; total: number };
|
||||||
|
|
||||||
|
const getStorageSizes = debounce((cb: (sizes: StorageSizes) => void) => {
|
||||||
|
cb({
|
||||||
|
scene: getElementsStorageSize(),
|
||||||
|
total: getTotalStorageSize(),
|
||||||
|
});
|
||||||
|
}, 500);
|
||||||
|
|
||||||
export const Stats = (props: {
|
export const Stats = (props: {
|
||||||
appState: AppState;
|
appState: AppState;
|
||||||
setAppState: React.Component<any, AppState>["setState"];
|
setAppState: React.Component<any, AppState>["setState"];
|
||||||
elements: readonly NonDeletedExcalidrawElement[];
|
elements: readonly NonDeletedExcalidrawElement[];
|
||||||
onClose: () => void;
|
onClose: () => void;
|
||||||
renderCustomStats: ExcalidrawProps["renderCustomStats"];
|
|
||||||
}) => {
|
}) => {
|
||||||
const isMobile = useIsMobile();
|
const isMobile = useIsMobile();
|
||||||
|
const [storageSizes, setStorageSizes] = useState<StorageSizes>({
|
||||||
|
scene: 0,
|
||||||
|
total: 0,
|
||||||
|
});
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
getStorageSizes((sizes) => {
|
||||||
|
setStorageSizes(sizes);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
useEffect(() => () => getStorageSizes.cancel(), []);
|
||||||
|
|
||||||
const boundingBox = getCommonBounds(props.elements);
|
const boundingBox = getCommonBounds(props.elements);
|
||||||
const selectedElements = getTargetElements(props.elements, props.appState);
|
const selectedElements = getTargetElements(props.elements, props.appState);
|
||||||
@@ -26,6 +53,17 @@ export const Stats = (props: {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const version = getVersion();
|
||||||
|
let hash;
|
||||||
|
let timestamp;
|
||||||
|
|
||||||
|
if (version !== DEFAULT_VERSION) {
|
||||||
|
timestamp = version.slice(0, 16).replace("T", " ");
|
||||||
|
hash = version.slice(21);
|
||||||
|
} else {
|
||||||
|
timestamp = t("stats.versionNotAvailable");
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="Stats">
|
<div className="Stats">
|
||||||
<Island padding={2}>
|
<Island padding={2}>
|
||||||
@@ -50,7 +88,17 @@ export const Stats = (props: {
|
|||||||
<td>{t("stats.height")}</td>
|
<td>{t("stats.height")}</td>
|
||||||
<td>{Math.round(boundingBox[3]) - Math.round(boundingBox[1])}</td>
|
<td>{Math.round(boundingBox[3]) - Math.round(boundingBox[1])}</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<th colSpan={2}>{t("stats.storage")}</th>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>{t("stats.scene")}</td>
|
||||||
|
<td>{nFormatter(storageSizes.scene, 1)}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>{t("stats.total")}</td>
|
||||||
|
<td>{nFormatter(storageSizes.total, 1)}</td>
|
||||||
|
</tr>
|
||||||
{selectedElements.length === 1 && (
|
{selectedElements.length === 1 && (
|
||||||
<tr>
|
<tr>
|
||||||
<th colSpan={2}>{t("stats.element")}</th>
|
<th colSpan={2}>{t("stats.element")}</th>
|
||||||
@@ -72,17 +120,31 @@ export const Stats = (props: {
|
|||||||
<>
|
<>
|
||||||
<tr>
|
<tr>
|
||||||
<td>{"x"}</td>
|
<td>{"x"}</td>
|
||||||
<td>{Math.round(selectedBoundingBox[0])}</td>
|
<td>
|
||||||
|
{Math.round(
|
||||||
|
selectedElements.length === 1
|
||||||
|
? selectedElements[0].x
|
||||||
|
: selectedBoundingBox[0],
|
||||||
|
)}
|
||||||
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td>{"y"}</td>
|
<td>{"y"}</td>
|
||||||
<td>{Math.round(selectedBoundingBox[1])}</td>
|
<td>
|
||||||
|
{Math.round(
|
||||||
|
selectedElements.length === 1
|
||||||
|
? selectedElements[0].y
|
||||||
|
: selectedBoundingBox[1],
|
||||||
|
)}
|
||||||
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td>{t("stats.width")}</td>
|
<td>{t("stats.width")}</td>
|
||||||
<td>
|
<td>
|
||||||
{Math.round(
|
{Math.round(
|
||||||
selectedBoundingBox[2] - selectedBoundingBox[0],
|
selectedElements.length === 1
|
||||||
|
? selectedElements[0].width
|
||||||
|
: selectedBoundingBox[2] - selectedBoundingBox[0],
|
||||||
)}
|
)}
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
@@ -90,7 +152,9 @@ export const Stats = (props: {
|
|||||||
<td>{t("stats.height")}</td>
|
<td>{t("stats.height")}</td>
|
||||||
<td>
|
<td>
|
||||||
{Math.round(
|
{Math.round(
|
||||||
selectedBoundingBox[3] - selectedBoundingBox[1],
|
selectedElements.length === 1
|
||||||
|
? selectedElements[0].height
|
||||||
|
: selectedBoundingBox[3] - selectedBoundingBox[1],
|
||||||
)}
|
)}
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
@@ -106,7 +170,28 @@ export const Stats = (props: {
|
|||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
)}
|
)}
|
||||||
{props.renderCustomStats?.(props.elements, props.appState)}
|
<tr>
|
||||||
|
<th colSpan={2}>{t("stats.version")}</th>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td
|
||||||
|
colSpan={2}
|
||||||
|
style={{ textAlign: "center", cursor: "pointer" }}
|
||||||
|
onClick={async () => {
|
||||||
|
try {
|
||||||
|
await copyTextToSystemClipboard(getVersion());
|
||||||
|
props.setAppState({
|
||||||
|
toastMessage: t("toast.copyToClipboard"),
|
||||||
|
});
|
||||||
|
} catch {}
|
||||||
|
}}
|
||||||
|
title={t("stats.versionCopy")}
|
||||||
|
>
|
||||||
|
{timestamp}
|
||||||
|
<br />
|
||||||
|
{hash}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</Island>
|
</Island>
|
||||||
|
@@ -29,13 +29,9 @@ type ToolButtonProps =
|
|||||||
children?: React.ReactNode;
|
children?: React.ReactNode;
|
||||||
onClick?(): void;
|
onClick?(): void;
|
||||||
})
|
})
|
||||||
| (ToolButtonBaseProps & {
|
|
||||||
type: "icon";
|
|
||||||
children?: React.ReactNode;
|
|
||||||
onClick?(): void;
|
|
||||||
})
|
|
||||||
| (ToolButtonBaseProps & {
|
| (ToolButtonBaseProps & {
|
||||||
type: "radio";
|
type: "radio";
|
||||||
|
|
||||||
checked: boolean;
|
checked: boolean;
|
||||||
onChange?(): void;
|
onChange?(): void;
|
||||||
});
|
});
|
||||||
@@ -47,7 +43,7 @@ export const ToolButton = React.forwardRef((props: ToolButtonProps, ref) => {
|
|||||||
React.useImperativeHandle(ref, () => innerRef.current);
|
React.useImperativeHandle(ref, () => innerRef.current);
|
||||||
const sizeCn = `ToolIcon_size_${props.size || DEFAULT_SIZE}`;
|
const sizeCn = `ToolIcon_size_${props.size || DEFAULT_SIZE}`;
|
||||||
|
|
||||||
if (props.type === "button" || props.type === "icon") {
|
if (props.type === "button") {
|
||||||
return (
|
return (
|
||||||
<button
|
<button
|
||||||
className={clsx(
|
className={clsx(
|
||||||
@@ -60,7 +56,6 @@ export const ToolButton = React.forwardRef((props: ToolButtonProps, ref) => {
|
|||||||
{
|
{
|
||||||
ToolIcon: !props.hidden,
|
ToolIcon: !props.hidden,
|
||||||
"ToolIcon--selected": props.selected,
|
"ToolIcon--selected": props.selected,
|
||||||
"ToolIcon--plain": props.type === "icon",
|
|
||||||
},
|
},
|
||||||
)}
|
)}
|
||||||
data-testid={props["data-testid"]}
|
data-testid={props["data-testid"]}
|
||||||
@@ -71,16 +66,14 @@ export const ToolButton = React.forwardRef((props: ToolButtonProps, ref) => {
|
|||||||
onClick={props.onClick}
|
onClick={props.onClick}
|
||||||
ref={innerRef}
|
ref={innerRef}
|
||||||
>
|
>
|
||||||
{(props.icon || props.label) && (
|
<div className="ToolIcon__icon" aria-hidden="true">
|
||||||
<div className="ToolIcon__icon" aria-hidden="true">
|
{props.icon || props.label}
|
||||||
{props.icon || props.label}
|
{props.keyBindingLabel && (
|
||||||
{props.keyBindingLabel && (
|
<span className="ToolIcon__keybinding">
|
||||||
<span className="ToolIcon__keybinding">
|
{props.keyBindingLabel}
|
||||||
{props.keyBindingLabel}
|
</span>
|
||||||
</span>
|
)}
|
||||||
)}
|
</div>
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
{props.showAriaLabel && (
|
{props.showAriaLabel && (
|
||||||
<div className="ToolIcon__label">{props["aria-label"]}</div>
|
<div className="ToolIcon__label">{props["aria-label"]}</div>
|
||||||
)}
|
)}
|
||||||
|
@@ -11,15 +11,6 @@
|
|||||||
background-color: var(--button-gray-1);
|
background-color: var(--button-gray-1);
|
||||||
-webkit-tap-highlight-color: transparent;
|
-webkit-tap-highlight-color: transparent;
|
||||||
border-radius: var(--space-factor);
|
border-radius: var(--space-factor);
|
||||||
user-select: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.ToolIcon--plain {
|
|
||||||
background-color: transparent;
|
|
||||||
.ToolIcon__icon {
|
|
||||||
width: 2rem;
|
|
||||||
height: 2rem;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.ToolIcon__icon {
|
.ToolIcon__icon {
|
||||||
@@ -196,6 +187,17 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.TooltipIcon {
|
||||||
|
width: 0.9em;
|
||||||
|
height: 0.9em;
|
||||||
|
margin-left: 5px;
|
||||||
|
margin-top: 1px;
|
||||||
|
|
||||||
|
@media #{$is-mobile-query} {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.unlocked-icon {
|
.unlocked-icon {
|
||||||
:root[dir="ltr"] & {
|
:root[dir="ltr"] & {
|
||||||
left: 2px;
|
left: 2px;
|
||||||
|
@@ -1,39 +1,58 @@
|
|||||||
@import "../css/variables.module";
|
@import "../css/variables.module";
|
||||||
.excalidraw-tooltip {
|
|
||||||
position: absolute;
|
|
||||||
z-index: 1000;
|
|
||||||
|
|
||||||
padding: 8px;
|
|
||||||
border-radius: 6px;
|
|
||||||
box-sizing: border-box;
|
|
||||||
pointer-events: none;
|
|
||||||
word-wrap: break-word;
|
|
||||||
|
|
||||||
background: $oc-black;
|
|
||||||
|
|
||||||
line-height: 1.5;
|
|
||||||
text-align: center;
|
|
||||||
font-size: 13px;
|
|
||||||
font-weight: 500;
|
|
||||||
color: $oc-white;
|
|
||||||
|
|
||||||
display: none;
|
|
||||||
|
|
||||||
&.excalidraw-tooltip--visible {
|
|
||||||
display: block;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.excalidraw {
|
.excalidraw {
|
||||||
.Tooltip-icon {
|
.Tooltip {
|
||||||
width: 0.9em;
|
position: relative;
|
||||||
height: 0.9em;
|
}
|
||||||
margin-left: 5px;
|
|
||||||
margin-top: 1px;
|
|
||||||
display: flex;
|
|
||||||
|
|
||||||
@include isMobile {
|
.Tooltip__label {
|
||||||
display: none;
|
--arrow-size: 4px;
|
||||||
|
visibility: hidden;
|
||||||
|
background: $oc-black;
|
||||||
|
color: $oc-white;
|
||||||
|
text-align: center;
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 8px;
|
||||||
|
position: absolute;
|
||||||
|
z-index: 10;
|
||||||
|
font-size: 13px;
|
||||||
|
line-height: 1.5;
|
||||||
|
font-weight: 500;
|
||||||
|
// extra pixel offset for unknown reasons
|
||||||
|
left: calc(50% + var(--arrow-size) / 2 - 1px);
|
||||||
|
transform: translateX(-50%);
|
||||||
|
word-wrap: break-word;
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
content: "";
|
||||||
|
border: var(--arrow-size) solid transparent;
|
||||||
|
position: absolute;
|
||||||
|
left: calc(50% - var(--arrow-size));
|
||||||
|
}
|
||||||
|
|
||||||
|
&--above {
|
||||||
|
bottom: calc(100% + var(--arrow-size) + 3px);
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
border-top-color: $oc-black;
|
||||||
|
top: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&--below {
|
||||||
|
top: calc(100% + var(--arrow-size) + 3px);
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
border-bottom-color: $oc-black;
|
||||||
|
bottom: 100%;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.Tooltip:hover .Tooltip__label {
|
||||||
|
visibility: visible;
|
||||||
|
}
|
||||||
|
|
||||||
|
.Tooltip__label:hover {
|
||||||
|
visibility: visible;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,92 +1,31 @@
|
|||||||
import "./Tooltip.scss";
|
import "./Tooltip.scss";
|
||||||
|
|
||||||
import React, { useEffect } from "react";
|
import React from "react";
|
||||||
|
|
||||||
const getTooltipDiv = () => {
|
|
||||||
const existingDiv = document.querySelector<HTMLDivElement>(
|
|
||||||
".excalidraw-tooltip",
|
|
||||||
);
|
|
||||||
if (existingDiv) {
|
|
||||||
return existingDiv;
|
|
||||||
}
|
|
||||||
const div = document.createElement("div");
|
|
||||||
document.body.appendChild(div);
|
|
||||||
div.classList.add("excalidraw-tooltip");
|
|
||||||
return div;
|
|
||||||
};
|
|
||||||
|
|
||||||
const updateTooltip = (
|
|
||||||
item: HTMLDivElement,
|
|
||||||
tooltip: HTMLDivElement,
|
|
||||||
label: string,
|
|
||||||
long: boolean,
|
|
||||||
) => {
|
|
||||||
tooltip.classList.add("excalidraw-tooltip--visible");
|
|
||||||
tooltip.style.minWidth = long ? "50ch" : "10ch";
|
|
||||||
tooltip.style.maxWidth = long ? "50ch" : "15ch";
|
|
||||||
|
|
||||||
tooltip.textContent = label;
|
|
||||||
|
|
||||||
const {
|
|
||||||
x: itemX,
|
|
||||||
bottom: itemBottom,
|
|
||||||
top: itemTop,
|
|
||||||
width: itemWidth,
|
|
||||||
} = item.getBoundingClientRect();
|
|
||||||
|
|
||||||
const {
|
|
||||||
width: labelWidth,
|
|
||||||
height: labelHeight,
|
|
||||||
} = tooltip.getBoundingClientRect();
|
|
||||||
|
|
||||||
const viewportWidth = window.innerWidth;
|
|
||||||
const viewportHeight = window.innerHeight;
|
|
||||||
|
|
||||||
const margin = 5;
|
|
||||||
|
|
||||||
const left = itemX + itemWidth / 2 - labelWidth / 2;
|
|
||||||
const offsetLeft =
|
|
||||||
left + labelWidth >= viewportWidth ? left + labelWidth - viewportWidth : 0;
|
|
||||||
|
|
||||||
const top = itemBottom + margin;
|
|
||||||
const offsetTop =
|
|
||||||
top + labelHeight >= viewportHeight
|
|
||||||
? itemBottom - itemTop + labelHeight + margin * 2
|
|
||||||
: 0;
|
|
||||||
|
|
||||||
Object.assign(tooltip.style, {
|
|
||||||
top: `${top - offsetTop}px`,
|
|
||||||
left: `${left - offsetLeft}px`,
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
type TooltipProps = {
|
type TooltipProps = {
|
||||||
children: React.ReactNode;
|
children: React.ReactNode;
|
||||||
label: string;
|
label: string;
|
||||||
|
position?: "above" | "below";
|
||||||
long?: boolean;
|
long?: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const Tooltip = ({ children, label, long = false }: TooltipProps) => {
|
export const Tooltip = ({
|
||||||
useEffect(() => {
|
children,
|
||||||
return () =>
|
label,
|
||||||
getTooltipDiv().classList.remove("excalidraw-tooltip--visible");
|
position = "below",
|
||||||
}, []);
|
long = false,
|
||||||
|
}: TooltipProps) => (
|
||||||
return (
|
<div className="Tooltip">
|
||||||
<div
|
<span
|
||||||
onPointerEnter={(event) =>
|
className={
|
||||||
updateTooltip(
|
position === "above"
|
||||||
event.currentTarget as HTMLDivElement,
|
? "Tooltip__label Tooltip__label--above"
|
||||||
getTooltipDiv(),
|
: "Tooltip__label Tooltip__label--below"
|
||||||
label,
|
|
||||||
long,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
onPointerLeave={() =>
|
|
||||||
getTooltipDiv().classList.remove("excalidraw-tooltip--visible")
|
|
||||||
}
|
}
|
||||||
|
style={{ width: long ? "50ch" : "10ch" }}
|
||||||
>
|
>
|
||||||
{children}
|
{label}
|
||||||
</div>
|
</span>
|
||||||
);
|
{children}
|
||||||
};
|
</div>
|
||||||
|
);
|
||||||
|
@@ -2,8 +2,7 @@
|
|||||||
.UserList {
|
.UserList {
|
||||||
pointer-events: none;
|
pointer-events: none;
|
||||||
/*github corner*/
|
/*github corner*/
|
||||||
padding: var(--space-factor) var(--space-factor) var(--space-factor)
|
padding: var(--space-factor) 40px var(--space-factor) var(--space-factor);
|
||||||
var(--space-factor);
|
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
justify-content: flex-end;
|
justify-content: flex-end;
|
||||||
|
@@ -41,14 +41,6 @@ const createIcon = (d: string | React.ReactNode, opts: number | Opts = 512) => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const checkIcon = createIcon(
|
|
||||||
<polyline fill="none" stroke="currentColor" points="20 6 9 17 4 12" />,
|
|
||||||
{
|
|
||||||
width: 24,
|
|
||||||
height: 24,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
export const link = createIcon(
|
export const link = createIcon(
|
||||||
"M326.612 185.391c59.747 59.809 58.927 155.698.36 214.59-.11.12-.24.25-.36.37l-67.2 67.2c-59.27 59.27-155.699 59.262-214.96 0-59.27-59.26-59.27-155.7 0-214.96l37.106-37.106c9.84-9.84 26.786-3.3 27.294 10.606.648 17.722 3.826 35.527 9.69 52.721 1.986 5.822.567 12.262-3.783 16.612l-13.087 13.087c-28.026 28.026-28.905 73.66-1.155 101.96 28.024 28.579 74.086 28.749 102.325.51l67.2-67.19c28.191-28.191 28.073-73.757 0-101.83-3.701-3.694-7.429-6.564-10.341-8.569a16.037 16.037 0 0 1-6.947-12.606c-.396-10.567 3.348-21.456 11.698-29.806l21.054-21.055c5.521-5.521 14.182-6.199 20.584-1.731a152.482 152.482 0 0 1 20.522 17.197zM467.547 44.449c-59.261-59.262-155.69-59.27-214.96 0l-67.2 67.2c-.12.12-.25.25-.36.37-58.566 58.892-59.387 154.781.36 214.59a152.454 152.454 0 0 0 20.521 17.196c6.402 4.468 15.064 3.789 20.584-1.731l21.054-21.055c8.35-8.35 12.094-19.239 11.698-29.806a16.037 16.037 0 0 0-6.947-12.606c-2.912-2.005-6.64-4.875-10.341-8.569-28.073-28.073-28.191-73.639 0-101.83l67.2-67.19c28.239-28.239 74.3-28.069 102.325.51 27.75 28.3 26.872 73.934-1.155 101.96l-13.087 13.087c-4.35 4.35-5.769 10.79-3.783 16.612 5.864 17.194 9.042 34.999 9.69 52.721.509 13.906 17.454 20.446 27.294 10.606l37.106-37.106c59.271-59.259 59.271-155.699.001-214.959z",
|
"M326.612 185.391c59.747 59.809 58.927 155.698.36 214.59-.11.12-.24.25-.36.37l-67.2 67.2c-59.27 59.27-155.699 59.262-214.96 0-59.27-59.26-59.27-155.7 0-214.96l37.106-37.106c9.84-9.84 26.786-3.3 27.294 10.606.648 17.722 3.826 35.527 9.69 52.721 1.986 5.822.567 12.262-3.783 16.612l-13.087 13.087c-28.026 28.026-28.905 73.66-1.155 101.96 28.024 28.579 74.086 28.749 102.325.51l67.2-67.19c28.191-28.191 28.073-73.757 0-101.83-3.701-3.694-7.429-6.564-10.341-8.569a16.037 16.037 0 0 1-6.947-12.606c-.396-10.567 3.348-21.456 11.698-29.806l21.054-21.055c5.521-5.521 14.182-6.199 20.584-1.731a152.482 152.482 0 0 1 20.522 17.197zM467.547 44.449c-59.261-59.262-155.69-59.27-214.96 0l-67.2 67.2c-.12.12-.25.25-.36.37-58.566 58.892-59.387 154.781.36 214.59a152.454 152.454 0 0 0 20.521 17.196c6.402 4.468 15.064 3.789 20.584-1.731l21.054-21.055c8.35-8.35 12.094-19.239 11.698-29.806a16.037 16.037 0 0 0-6.947-12.606c-2.912-2.005-6.64-4.875-10.341-8.569-28.073-28.073-28.191-73.639 0-101.83l67.2-67.19c28.239-28.239 74.3-28.069 102.325.51 27.75 28.3 26.872 73.934-1.155 101.96l-13.087 13.087c-4.35 4.35-5.769 10.79-3.783 16.612 5.864 17.194 9.042 34.999 9.69 52.721.509 13.906 17.454 20.446 27.294 10.606l37.106-37.106c59.271-59.259 59.271-155.699.001-214.959z",
|
||||||
{ mirror: true },
|
{ mirror: true },
|
||||||
@@ -88,19 +80,6 @@ export const exportFile = createIcon(
|
|||||||
{ width: 576, height: 512, mirror: true },
|
{ width: 576, height: 512, mirror: true },
|
||||||
);
|
);
|
||||||
|
|
||||||
export const exportImage = createIcon(
|
|
||||||
<>
|
|
||||||
<path d="M571 308l-95.7-96.4c-10.1-10.1-27.4-3-27.4 11.3V288h-64v64h64v65.2c0 14.3 17.3 21.4 27.4 11.3L571 332c6.6-6.6 6.6-17.4 0-24zm-187 44v-64 64z" />
|
|
||||||
<path d="M384 121.941V128H256V0h6.059c6.362 0 12.471 2.53 16.97 7.029l97.941 97.941a24.01 24.01 0 017.03 16.971zM248 160c-13.2 0-24-10.8-24-24V0H24C10.745 0 0 10.745 0 24v464c0 13.255 10.745 24 24 24h336c13.255 0 24-10.745 24-24V160H248zm-135.455 16c26.51 0 48 21.49 48 48s-21.49 48-48 48-48-21.49-48-48 21.491-48 48-48zm208 240h-256l.485-48.485L104.545 328c4.686-4.686 11.799-4.201 16.485.485L160.545 368 264.06 264.485c4.686-4.686 12.284-4.686 16.971 0L320.545 304v112z" />
|
|
||||||
</>,
|
|
||||||
{ width: 576, height: 512, mirror: true },
|
|
||||||
);
|
|
||||||
|
|
||||||
export const exportToFileIcon = createIcon(
|
|
||||||
"M216 0h80c13.3 0 24 10.7 24 24v168h87.7c17.8 0 26.7 21.5 14.1 34.1L269.7 378.3c-7.5 7.5-19.8 7.5-27.3 0L90.1 226.1c-12.6-12.6-3.7-34.1 14.1-34.1H192V24c0-13.3 10.7-24 24-24zm296 376v112c0 13.3-10.7 24-24 24H24c-13.3 0-24-10.7-24-24V376c0-13.3 10.7-24 24-24h146.7l49 49c20.1 20.1 52.5 20.1 72.6 0l49-49H488c13.3 0 24 10.7 24 24zm-124 88c0-11-9-20-20-20s-20 9-20 20 9 20 20 20 20-9 20-20zm64 0c0-11-9-20-20-20s-20 9-20 20 9 20 20 20 20-9 20-20z",
|
|
||||||
{ width: 512, height: 512 },
|
|
||||||
);
|
|
||||||
|
|
||||||
export const zoomIn = createIcon(
|
export const zoomIn = createIcon(
|
||||||
"M416 208H272V64c0-17.67-14.33-32-32-32h-32c-17.67 0-32 14.33-32 32v144H32c-17.67 0-32 14.33-32 32v32c0 17.67 14.33 32 32 32h144v144c0 17.67 14.33 32 32 32h32c17.67 0 32-14.33 32-32V304h144c17.67 0 32-14.33 32-32v-32c0-17.67-14.33-32-32-32z",
|
"M416 208H272V64c0-17.67-14.33-32-32-32h-32c-17.67 0-32 14.33-32 32v144H32c-17.67 0-32 14.33-32 32v32c0 17.67 14.33 32 32 32h144v144c0 17.67 14.33 32 32 32h32c17.67 0 32-14.33 32-32V304h144c17.67 0 32-14.33 32-32v-32c0-17.67-14.33-32-32-32z",
|
||||||
{ width: 448, height: 512 },
|
{ width: 448, height: 512 },
|
||||||
@@ -144,22 +123,6 @@ export const shareIOS = createIcon(
|
|||||||
{ width: 24, height: 24 },
|
{ width: 24, height: 24 },
|
||||||
);
|
);
|
||||||
|
|
||||||
export const shareWindows = createIcon(
|
|
||||||
<>
|
|
||||||
<path
|
|
||||||
stroke="currentColor"
|
|
||||||
fill="currentColor"
|
|
||||||
d="M40 5.6v6.1l-4.1.7c-8.9 1.4-16.5 6.9-20.6 15C13 32 10.9 43 12.4 43c.4 0 2.4-1.3 4.4-3 5-3.9 12.1-7 18.2-7.7l5-.6v12.8l11.2-11.3L62.5 22 51.2 10.8 40-.5v6.1zm10.2 22.6L44 34.5v-6.8l-6.9.6c-3.9.3-9.8 1.7-13.2 3.1-3.5 1.4-6.5 2.4-6.7 2.2-.9-1 3-7.5 6.4-10.8C28 18.6 34.4 16 40.1 16c3.7 0 3.9-.1 3.9-3.2V9.5l6.2 6.3 6.3 6.2-6.3 6.2z"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
stroke="currentColor"
|
|
||||||
fill="currentColor"
|
|
||||||
d="M0 36v20h48v-6.2c0-6 0-6.1-2-4.3-1.1 1-2 2.9-2 4.2V52H4V34c0-17.3-.1-18-2-18s-2 .7-2 20z"
|
|
||||||
/>
|
|
||||||
</>,
|
|
||||||
{ width: 64, height: 64 },
|
|
||||||
);
|
|
||||||
|
|
||||||
// Icon imported form Storybook
|
// Icon imported form Storybook
|
||||||
// Storybook is licensed under MIT https://github.com/storybookjs/storybook/blob/next/LICENSE
|
// Storybook is licensed under MIT https://github.com/storybookjs/storybook/blob/next/LICENSE
|
||||||
export const resetZoom = createIcon(
|
export const resetZoom = createIcon(
|
||||||
@@ -243,12 +206,14 @@ export const SendToBackIcon = React.memo(
|
|||||||
d="M18 7.333C18 6.597 17.403 6 16.667 6H7.333C6.597 6 6 6.597 6 7.333v9.334C6 17.403 6.597 18 7.333 18h9.334c.736 0 1.333-.597 1.333-1.333V7.333z"
|
d="M18 7.333C18 6.597 17.403 6 16.667 6H7.333C6.597 6 6 6.597 6 7.333v9.334C6 17.403 6.597 18 7.333 18h9.334c.736 0 1.333-.597 1.333-1.333V7.333z"
|
||||||
fill={activeElementColor(theme)}
|
fill={activeElementColor(theme)}
|
||||||
stroke={activeElementColor(theme)}
|
stroke={activeElementColor(theme)}
|
||||||
|
strokeLinejoin="round"
|
||||||
strokeWidth="2"
|
strokeWidth="2"
|
||||||
/>
|
/>
|
||||||
<path
|
<path
|
||||||
d="M11 3a1 1 0 00-1-1H3a1 1 0 00-1 1v7a1 1 0 001 1h8V3zM22 14a1 1 0 00-1-1h-7a1 1 0 00-1 1v7a1 1 0 001 1h8v-8z"
|
d="M11 3a1 1 0 00-1-1H3a1 1 0 00-1 1v7a1 1 0 001 1h8V3zM22 14a1 1 0 00-1-1h-7a1 1 0 00-1 1v7a1 1 0 001 1h8v-8z"
|
||||||
fill={iconFillColor(theme)}
|
fill={iconFillColor(theme)}
|
||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
|
strokeLinejoin="round"
|
||||||
strokeWidth="2"
|
strokeWidth="2"
|
||||||
/>
|
/>
|
||||||
</>,
|
</>,
|
||||||
@@ -354,6 +319,7 @@ export const DistributeHorizontallyIcon = React.memo(
|
|||||||
({ theme }: { theme: "light" | "dark" }) =>
|
({ theme }: { theme: "light" | "dark" }) =>
|
||||||
createIcon(
|
createIcon(
|
||||||
<>
|
<>
|
||||||
|
<path d="M5 5V19Z" fill="black" />
|
||||||
<path
|
<path
|
||||||
d="M19 5V19M5 5V19"
|
d="M19 5V19M5 5V19"
|
||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
@@ -371,6 +337,14 @@ export const DistributeHorizontallyIcon = React.memo(
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
<svg
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
></svg>;
|
||||||
|
|
||||||
export const DistributeVerticallyIcon = React.memo(
|
export const DistributeVerticallyIcon = React.memo(
|
||||||
({ theme }: { theme: "light" | "dark" }) =>
|
({ theme }: { theme: "light" | "dark" }) =>
|
||||||
createIcon(
|
createIcon(
|
||||||
@@ -489,16 +463,42 @@ export const GroupIcon = React.memo(({ theme }: { theme: "light" | "dark" }) =>
|
|||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
strokeWidth="2"
|
strokeWidth="2"
|
||||||
/>
|
/>
|
||||||
<g
|
<rect
|
||||||
|
x="2.5"
|
||||||
|
y="2.5"
|
||||||
|
width="30"
|
||||||
|
height="30"
|
||||||
fill={handlerColor(theme)}
|
fill={handlerColor(theme)}
|
||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
strokeWidth="6"
|
strokeWidth="6"
|
||||||
>
|
/>
|
||||||
<rect x="2.5" y="2.5" width="30" height="30" />
|
<rect
|
||||||
<rect x="2.5" y="149.5" width="30" height="30" />
|
x="2.5"
|
||||||
<rect x="147.5" y="149.5" width="30" height="30" />
|
y="149.5"
|
||||||
<rect x="147.5" y="2.5" width="30" height="30" />
|
width="30"
|
||||||
</g>
|
height="30"
|
||||||
|
fill={handlerColor(theme)}
|
||||||
|
stroke={iconFillColor(theme)}
|
||||||
|
strokeWidth="6"
|
||||||
|
/>
|
||||||
|
<rect
|
||||||
|
x="147.5"
|
||||||
|
y="149.5"
|
||||||
|
width="30"
|
||||||
|
height="30"
|
||||||
|
fill={handlerColor(theme)}
|
||||||
|
stroke={iconFillColor(theme)}
|
||||||
|
strokeWidth="6"
|
||||||
|
/>
|
||||||
|
<rect
|
||||||
|
x="147.5"
|
||||||
|
y="2.5"
|
||||||
|
width="30"
|
||||||
|
height="30"
|
||||||
|
fill={handlerColor(theme)}
|
||||||
|
stroke={iconFillColor(theme)}
|
||||||
|
strokeWidth="6"
|
||||||
|
/>
|
||||||
</>,
|
</>,
|
||||||
{ width: 182, height: 182, mirror: true },
|
{ width: 182, height: 182, mirror: true },
|
||||||
),
|
),
|
||||||
@@ -520,18 +520,60 @@ export const UngroupIcon = React.memo(
|
|||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
strokeWidth="2"
|
strokeWidth="2"
|
||||||
/>
|
/>
|
||||||
<g
|
<rect
|
||||||
|
x="2.5"
|
||||||
|
y="2.5"
|
||||||
|
width="30"
|
||||||
|
height="30"
|
||||||
fill={handlerColor(theme)}
|
fill={handlerColor(theme)}
|
||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
strokeWidth="6"
|
strokeWidth="6"
|
||||||
>
|
/>
|
||||||
<rect x="2.5" y="2.5" width="30" height="30" />
|
<rect
|
||||||
<rect x="78.5" y="149.5" width="30" height="30" />
|
x="78.5"
|
||||||
<rect x="147.5" y="149.5" width="30" height="30" />
|
y="149.5"
|
||||||
<rect x="147.5" y="78.5" width="30" height="30" />
|
width="30"
|
||||||
<rect x="105.5" y="2.5" width="30" height="30" />
|
height="30"
|
||||||
<rect x="2.5" y="102.5" width="30" height="30" />
|
fill={handlerColor(theme)}
|
||||||
</g>
|
stroke={iconFillColor(theme)}
|
||||||
|
strokeWidth="6"
|
||||||
|
/>
|
||||||
|
<rect
|
||||||
|
x="147.5"
|
||||||
|
y="149.5"
|
||||||
|
width="30"
|
||||||
|
height="30"
|
||||||
|
fill={handlerColor(theme)}
|
||||||
|
stroke={iconFillColor(theme)}
|
||||||
|
strokeWidth="6"
|
||||||
|
/>
|
||||||
|
<rect
|
||||||
|
x="147.5"
|
||||||
|
y="78.5"
|
||||||
|
width="30"
|
||||||
|
height="30"
|
||||||
|
fill={handlerColor(theme)}
|
||||||
|
stroke={iconFillColor(theme)}
|
||||||
|
strokeWidth="6"
|
||||||
|
/>
|
||||||
|
<rect
|
||||||
|
x="105.5"
|
||||||
|
y="2.5"
|
||||||
|
width="30"
|
||||||
|
height="30"
|
||||||
|
fill={handlerColor(theme)}
|
||||||
|
stroke={iconFillColor(theme)}
|
||||||
|
strokeWidth="6"
|
||||||
|
/>
|
||||||
|
<rect
|
||||||
|
x="2.5"
|
||||||
|
y="102.5"
|
||||||
|
width="30"
|
||||||
|
height="30"
|
||||||
|
fill={handlerColor(theme)}
|
||||||
|
stroke={iconFillColor(theme)}
|
||||||
|
strokeWidth="6"
|
||||||
|
/>
|
||||||
</>,
|
</>,
|
||||||
{ width: 182, height: 182, mirror: true },
|
{ width: 182, height: 182, mirror: true },
|
||||||
),
|
),
|
||||||
@@ -573,10 +615,9 @@ export const StrokeWidthIcon = React.memo(
|
|||||||
({ theme, strokeWidth }: { theme: "light" | "dark"; strokeWidth: number }) =>
|
({ theme, strokeWidth }: { theme: "light" | "dark"; strokeWidth: number }) =>
|
||||||
createIcon(
|
createIcon(
|
||||||
<path
|
<path
|
||||||
d="M6 10H32"
|
d="M6 10H34"
|
||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
strokeWidth={strokeWidth}
|
strokeWidth={strokeWidth}
|
||||||
strokeLinecap="round"
|
|
||||||
fill="none"
|
fill="none"
|
||||||
/>,
|
/>,
|
||||||
{ width: 40, height: 20 },
|
{ width: 40, height: 20 },
|
||||||
@@ -591,7 +632,6 @@ export const StrokeStyleSolidIcon = React.memo(
|
|||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
strokeWidth={2}
|
strokeWidth={2}
|
||||||
fill="none"
|
fill="none"
|
||||||
strokeLinecap="round"
|
|
||||||
/>,
|
/>,
|
||||||
{
|
{
|
||||||
width: 40,
|
width: 40,
|
||||||
@@ -609,7 +649,6 @@ export const StrokeStyleDashedIcon = React.memo(
|
|||||||
strokeWidth={2.5}
|
strokeWidth={2.5}
|
||||||
strokeDasharray={"10, 8"}
|
strokeDasharray={"10, 8"}
|
||||||
fill="none"
|
fill="none"
|
||||||
strokeLinecap="round"
|
|
||||||
/>,
|
/>,
|
||||||
{ width: 40, height: 20 },
|
{ width: 40, height: 20 },
|
||||||
),
|
),
|
||||||
@@ -619,12 +658,11 @@ export const StrokeStyleDottedIcon = React.memo(
|
|||||||
({ theme }: { theme: "light" | "dark" }) =>
|
({ theme }: { theme: "light" | "dark" }) =>
|
||||||
createIcon(
|
createIcon(
|
||||||
<path
|
<path
|
||||||
d="M6 10H36"
|
d="M6 10H34"
|
||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
strokeWidth={2.5}
|
strokeWidth={2.5}
|
||||||
strokeDasharray={"2, 4.5"}
|
strokeDasharray={"4, 4"}
|
||||||
fill="none"
|
fill="none"
|
||||||
strokeLinecap="round"
|
|
||||||
/>,
|
/>,
|
||||||
{ width: 40, height: 20 },
|
{ width: 40, height: 20 },
|
||||||
),
|
),
|
||||||
@@ -637,7 +675,6 @@ export const SloppinessArchitectIcon = React.memo(
|
|||||||
d="M3.00098 16.1691C6.28774 13.9744 19.6399 2.8905 22.7215 3.00082C25.8041 3.11113 19.1158 15.5488 21.4962 16.8309C23.8757 18.1131 34.4155 11.7148 37.0001 10.6919"
|
d="M3.00098 16.1691C6.28774 13.9744 19.6399 2.8905 22.7215 3.00082C25.8041 3.11113 19.1158 15.5488 21.4962 16.8309C23.8757 18.1131 34.4155 11.7148 37.0001 10.6919"
|
||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
strokeWidth={2}
|
strokeWidth={2}
|
||||||
strokeLinecap="round"
|
|
||||||
fill="none"
|
fill="none"
|
||||||
/>,
|
/>,
|
||||||
{ width: 40, height: 20, mirror: true },
|
{ width: 40, height: 20, mirror: true },
|
||||||
@@ -651,7 +688,6 @@ export const SloppinessArtistIcon = React.memo(
|
|||||||
d="M3 17C6.68158 14.8752 16.1296 9.09849 22.0648 6.54922C28 3.99995 22.2896 13.3209 25 14C27.7104 14.6791 36.3757 9.6471 36.3757 9.6471M6.40706 15C13 11.1918 20.0468 1.51045 23.0234 3.0052C26 4.49995 20.457 12.8659 22.7285 16.4329C25 20 36.3757 13 36.3757 13"
|
d="M3 17C6.68158 14.8752 16.1296 9.09849 22.0648 6.54922C28 3.99995 22.2896 13.3209 25 14C27.7104 14.6791 36.3757 9.6471 36.3757 9.6471M6.40706 15C13 11.1918 20.0468 1.51045 23.0234 3.0052C26 4.49995 20.457 12.8659 22.7285 16.4329C25 20 36.3757 13 36.3757 13"
|
||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
strokeWidth={2}
|
strokeWidth={2}
|
||||||
strokeLinecap="round"
|
|
||||||
fill="none"
|
fill="none"
|
||||||
/>,
|
/>,
|
||||||
{ width: 40, height: 20, mirror: true },
|
{ width: 40, height: 20, mirror: true },
|
||||||
@@ -665,7 +701,6 @@ export const SloppinessCartoonistIcon = React.memo(
|
|||||||
d="M3 15.6468C6.93692 13.5378 22.5544 2.81528 26.6206 3.00242C30.6877 3.18956 25.6708 15.3346 27.4009 16.7705C29.1309 18.2055 35.4001 12.4762 37 11.6177M3.97143 10.4917C6.61158 9.24563 16.3706 2.61886 19.8104 3.01724C23.2522 3.41472 22.0773 12.2013 24.6181 12.8783C27.1598 13.5536 33.3179 8.04068 35.0571 7.07244"
|
d="M3 15.6468C6.93692 13.5378 22.5544 2.81528 26.6206 3.00242C30.6877 3.18956 25.6708 15.3346 27.4009 16.7705C29.1309 18.2055 35.4001 12.4762 37 11.6177M3.97143 10.4917C6.61158 9.24563 16.3706 2.61886 19.8104 3.01724C23.2522 3.41472 22.0773 12.2013 24.6181 12.8783C27.1598 13.5536 33.3179 8.04068 35.0571 7.07244"
|
||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
strokeWidth={2}
|
strokeWidth={2}
|
||||||
strokeLinecap="round"
|
|
||||||
fill="none"
|
fill="none"
|
||||||
/>,
|
/>,
|
||||||
{ width: 40, height: 20, mirror: true },
|
{ width: 40, height: 20, mirror: true },
|
||||||
@@ -679,7 +714,6 @@ export const EdgeSharpIcon = React.memo(
|
|||||||
d="M10 17L10 5L35 5"
|
d="M10 17L10 5L35 5"
|
||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
strokeWidth={2}
|
strokeWidth={2}
|
||||||
strokeLinecap="round"
|
|
||||||
fill="none"
|
fill="none"
|
||||||
/>,
|
/>,
|
||||||
{ width: 40, height: 20, mirror: true },
|
{ width: 40, height: 20, mirror: true },
|
||||||
@@ -693,7 +727,6 @@ export const EdgeRoundIcon = React.memo(
|
|||||||
d="M10 17V15C10 8 13 5 21 5L33.5 5"
|
d="M10 17V15C10 8 13 5 21 5L33.5 5"
|
||||||
stroke={iconFillColor(theme)}
|
stroke={iconFillColor(theme)}
|
||||||
strokeWidth={2}
|
strokeWidth={2}
|
||||||
strokeLinecap="round"
|
|
||||||
fill="none"
|
fill="none"
|
||||||
/>,
|
/>,
|
||||||
{ width: 40, height: 20, mirror: true },
|
{ width: 40, height: 20, mirror: true },
|
||||||
@@ -761,123 +794,3 @@ export const ArrowheadBarIcon = React.memo(
|
|||||||
{ width: 40, height: 20 },
|
{ width: 40, height: 20 },
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
export const FontSizeSmallIcon = React.memo(
|
|
||||||
({ theme }: { theme: "light" | "dark" }) =>
|
|
||||||
createIcon(
|
|
||||||
<path
|
|
||||||
fill={iconFillColor(theme)}
|
|
||||||
d="M 0 69.092 L 0 55.03 A 124.24 124.24 0 0 0 4.706 57.02 Q 6.826 57.863 8.708 58.5 A 53.466 53.466 0 0 0 12.231 59.571 Q 17.236 60.889 21.387 60.889 A 20.909 20.909 0 0 0 24.265 60.704 Q 25.719 60.502 26.903 60.077 A 8.649 8.649 0 0 0 29.028 58.985 Q 31.689 57.08 31.689 53.321 Q 31.689 51.221 30.518 49.585 A 10.126 10.126 0 0 0 29.282 48.177 Q 28.352 47.287 27.075 46.436 A 23.719 23.719 0 0 0 25.752 45.627 Q 23.774 44.492 20.176 42.735 A 254.44 254.44 0 0 0 17.822 41.602 Q 11.503 38.631 8.236 35.888 A 19.742 19.742 0 0 1 8.008 35.694 A 22.18 22.18 0 0 1 2.783 29.102 Q 0.83 25.342 0.83 20.313 A 22.471 22.471 0 0 1 1.733 13.778 A 17.283 17.283 0 0 1 7.251 5.42 A 21.486 21.486 0 0 1 15.177 1.272 Q 18.361 0.338 22.166 0.09 A 43.573 43.573 0 0 1 25 0 A 42.399 42.399 0 0 1 34.349 1.01 A 39.075 39.075 0 0 1 35.62 1.319 A 67.407 67.407 0 0 1 42.108 3.382 A 83.357 83.357 0 0 1 46.191 5.03 L 41.309 16.797 Q 35.596 14.453 31.86 13.526 A 30.762 30.762 0 0 0 25.417 12.612 A 28.337 28.337 0 0 0 24.512 12.598 A 14.846 14.846 0 0 0 22.022 12.793 Q 19.498 13.224 17.92 14.6 Q 15.625 16.602 15.625 19.824 Q 15.625 21.826 16.553 23.316 Q 17.48 24.805 19.507 26.197 A 18.343 18.343 0 0 0 20.659 26.912 Q 22.596 28.035 26.516 29.953 A 299.99 299.99 0 0 0 29.102 31.201 Q 37.91 35.412 41.841 39.642 A 16.553 16.553 0 0 1 42.822 40.796 A 17.675 17.675 0 0 1 46.301 49.233 A 23.517 23.517 0 0 1 46.533 52.588 A 21.581 21.581 0 0 1 45.471 59.515 A 17.733 17.733 0 0 1 39.575 67.823 Q 33.745 72.486 24.094 73.243 A 49.683 49.683 0 0 1 20.215 73.389 A 51.712 51.712 0 0 1 9.448 72.315 A 40.672 40.672 0 0 1 0 69.092 Z"
|
|
||||||
/>,
|
|
||||||
{ width: 47, height: 77 },
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
export const FontSizeMediumIcon = React.memo(
|
|
||||||
({ theme }: { theme: "light" | "dark" }) =>
|
|
||||||
createIcon(
|
|
||||||
<path
|
|
||||||
fill={iconFillColor(theme)}
|
|
||||||
d="M 44.092 71.387 L 30.225 71.387 L 13.037 15.381 L 12.598 15.381 A 1505.093 1505.093 0 0 1 12.959 22.313 Q 13.426 31.715 13.508 36.4 A 102.991 102.991 0 0 1 13.525 38.184 L 13.525 71.387 L 0 71.387 L 0 0 L 20.605 0 L 37.5 54.59 L 37.793 54.59 L 55.713 0 L 76.318 0 L 76.318 71.387 L 62.207 71.387 L 62.207 37.598 Q 62.207 35.205 62.28 32.08 A 160.703 160.703 0 0 1 62.326 30.544 Q 62.452 26.754 62.866 17.168 A 5390.536 5390.536 0 0 1 62.939 15.479 L 62.5 15.479 L 44.092 71.387 Z"
|
|
||||||
/>,
|
|
||||||
{ width: 77, height: 75 },
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
export const FontSizeLargeIcon = React.memo(
|
|
||||||
({ theme }: { theme: "light" | "dark" }) =>
|
|
||||||
createIcon(
|
|
||||||
<path
|
|
||||||
fill={iconFillColor(theme)}
|
|
||||||
d="M 44.092 71.387 L 0 71.387 L 0 0 L 15.137 0 L 15.137 58.887 L 44.092 58.887 L 44.092 71.387 Z"
|
|
||||||
/>,
|
|
||||||
{ width: 45, height: 75 },
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
export const FontSizeExtraLargeIcon = React.memo(
|
|
||||||
({ theme }: { theme: "light" | "dark" }) =>
|
|
||||||
createIcon(
|
|
||||||
<path
|
|
||||||
fill={iconFillColor(theme)}
|
|
||||||
d="M 42.578 35.4 L 66.699 71.387 L 49.414 71.387 L 32.813 44.385 L 16.211 71.387 L 0 71.387 L 23.682 34.57 L 1.514 0 L 18.213 0 L 33.594 25.684 L 48.682 0 L 64.99 0 L 42.578 35.4 Z M 119.775 71.387 L 75.684 71.387 L 75.684 0 L 90.82 0 L 90.82 58.887 L 119.775 58.887 L 119.775 71.387 Z"
|
|
||||||
/>,
|
|
||||||
{ width: 120, height: 75 },
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
export const FontFamilyHandDrawnIcon = React.memo(
|
|
||||||
({ theme }: { theme: "light" | "dark" }) =>
|
|
||||||
createIcon(
|
|
||||||
<path
|
|
||||||
fill={iconFillColor(theme)}
|
|
||||||
d="M290.74 93.24l128.02 128.02-277.99 277.99-114.14 12.6C11.35 513.54-1.56 500.62.14 485.34l12.7-114.22 277.9-277.88zm207.2-19.06l-60.11-60.11c-18.75-18.75-49.16-18.75-67.91 0l-56.55 56.55 128.02 128.02 56.55-56.55c18.75-18.76 18.75-49.16 0-67.91z"
|
|
||||||
/>,
|
|
||||||
{ width: 448, height: 512 },
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
export const FontFamilyNormalIcon = React.memo(
|
|
||||||
({ theme }: { theme: "light" | "dark" }) =>
|
|
||||||
createIcon(
|
|
||||||
<>
|
|
||||||
<path
|
|
||||||
fill={iconFillColor(theme)}
|
|
||||||
d="M 63.818 71.68 L 54.492 71.68 L 45.898 49.561 L 17.578 49.561 L 9.082 71.68 L 0 71.68 L 27.881 0 L 35.986 0 L 63.818 71.68 Z M 20.605 41.602 L 43.213 41.602 L 35.205 19.971 L 31.787 9.277 Q 30.322 15.137 28.711 19.971 L 20.605 41.602 Z"
|
|
||||||
/>
|
|
||||||
<path
|
|
||||||
fill={iconFillColor(theme)}
|
|
||||||
d="M 68.994 71.68 L 52.686 71.68 L 47.51 54.688 L 21.484 54.688 L 16.309 71.68 L 0 71.68 L 25.195 0 L 43.701 0 L 68.994 71.68 Z M 25.293 41.992 L 43.896 41.992 A 27590.463 27590.463 0 0 1 42.2 36.532 Q 36.965 19.676 35.937 16.273 A 120.932 120.932 0 0 1 35.815 15.869 A 131.65 131.65 0 0 1 35.396 14.435 Q 34.951 12.879 34.675 11.741 A 34.866 34.866 0 0 1 34.521 11.084 A 141.762 141.762 0 0 1 33.706 14.075 Q 31.482 21.957 25.293 41.992 Z"
|
|
||||||
/>
|
|
||||||
</>,
|
|
||||||
{ width: 70, height: 78 },
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
export const FontFamilyCodeIcon = React.memo(
|
|
||||||
({ theme }: { theme: "light" | "dark" }) =>
|
|
||||||
createIcon(
|
|
||||||
<>
|
|
||||||
<path
|
|
||||||
fill={iconFillColor(theme)}
|
|
||||||
d="M278.9 511.5l-61-17.7c-6.4-1.8-10-8.5-8.2-14.9L346.2 8.7c1.8-6.4 8.5-10 14.9-8.2l61 17.7c6.4 1.8 10 8.5 8.2 14.9L293.8 503.3c-1.9 6.4-8.5 10.1-14.9 8.2zm-114-112.2l43.5-46.4c4.6-4.9 4.3-12.7-.8-17.2L117 256l90.6-79.7c5.1-4.5 5.5-12.3.8-17.2l-43.5-46.4c-4.5-4.8-12.1-5.1-17-.5L3.8 247.2c-5.1 4.7-5.1 12.8 0 17.5l144.1 135.1c4.9 4.6 12.5 4.4 17-.5zm327.2.6l144.1-135.1c5.1-4.7 5.1-12.8 0-17.5L492.1 112.1c-4.8-4.5-12.4-4.3-17 .5L431.6 159c-4.6 4.9-4.3 12.7.8 17.2L523 256l-90.6 79.7c-5.1 4.5-5.5 12.3-.8 17.2l43.5 46.4c4.5 4.9 12.1 5.1 17 .6z"
|
|
||||||
/>
|
|
||||||
</>,
|
|
||||||
{ width: 640, height: 512 },
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
export const TextAlignLeftIcon = React.memo(
|
|
||||||
({ theme }: { theme: "light" | "dark" }) =>
|
|
||||||
createIcon(
|
|
||||||
<path
|
|
||||||
d="M12.83 352h262.34A12.82 12.82 0 00288 339.17v-38.34A12.82 12.82 0 00275.17 288H12.83A12.82 12.82 0 000 300.83v38.34A12.82 12.82 0 0012.83 352zm0-256h262.34A12.82 12.82 0 00288 83.17V44.83A12.82 12.82 0 00275.17 32H12.83A12.82 12.82 0 000 44.83v38.34A12.82 12.82 0 0012.83 96zM432 160H16a16 16 0 00-16 16v32a16 16 0 0016 16h416a16 16 0 0016-16v-32a16 16 0 00-16-16zm0 256H16a16 16 0 00-16 16v32a16 16 0 0016 16h416a16 16 0 0016-16v-32a16 16 0 00-16-16z"
|
|
||||||
fill={iconFillColor(theme)}
|
|
||||||
strokeLinecap="round"
|
|
||||||
/>,
|
|
||||||
{ width: 448, height: 512 },
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
export const TextAlignCenterIcon = React.memo(
|
|
||||||
({ theme }: { theme: "light" | "dark" }) =>
|
|
||||||
createIcon(
|
|
||||||
<path
|
|
||||||
d="M432 160H16a16 16 0 00-16 16v32a16 16 0 0016 16h416a16 16 0 0016-16v-32a16 16 0 00-16-16zm0 256H16a16 16 0 00-16 16v32a16 16 0 0016 16h416a16 16 0 0016-16v-32a16 16 0 00-16-16zM108.1 96h231.81A12.09 12.09 0 00352 83.9V44.09A12.09 12.09 0 00339.91 32H108.1A12.09 12.09 0 0096 44.09V83.9A12.1 12.1 0 00108.1 96zm231.81 256A12.09 12.09 0 00352 339.9v-39.81A12.09 12.09 0 00339.91 288H108.1A12.09 12.09 0 0096 300.09v39.81a12.1 12.1 0 0012.1 12.1z"
|
|
||||||
fill={iconFillColor(theme)}
|
|
||||||
/>,
|
|
||||||
{ width: 448, height: 512 },
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
export const TextAlignRightIcon = React.memo(
|
|
||||||
({ theme }: { theme: "light" | "dark" }) =>
|
|
||||||
createIcon(
|
|
||||||
<path
|
|
||||||
d="M16 224h416a16 16 0 0016-16v-32a16 16 0 00-16-16H16a16 16 0 00-16 16v32a16 16 0 0016 16zm416 192H16a16 16 0 00-16 16v32a16 16 0 0016 16h416a16 16 0 0016-16v-32a16 16 0 00-16-16zm3.17-384H172.83A12.82 12.82 0 00160 44.83v38.34A12.82 12.82 0 00172.83 96h262.34A12.82 12.82 0 00448 83.17V44.83A12.82 12.82 0 00435.17 32zm0 256H172.83A12.82 12.82 0 00160 300.83v38.34A12.82 12.82 0 00172.83 352h262.34A12.82 12.82 0 00448 339.17v-38.34A12.82 12.82 0 00435.17 288z"
|
|
||||||
fill={iconFillColor(theme)}
|
|
||||||
strokeLinecap="round"
|
|
||||||
/>,
|
|
||||||
{ width: 448, height: 512 },
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
@@ -1,6 +1,5 @@
|
|||||||
import { FontFamily } from "./element/types";
|
import { FontFamily } from "./element/types";
|
||||||
import cssVariables from "./css/variables.module.scss";
|
import cssVariables from "./css/variables.module.scss";
|
||||||
import { AppProps } from "./types";
|
|
||||||
|
|
||||||
export const APP_NAME = "Excalidraw";
|
export const APP_NAME = "Excalidraw";
|
||||||
|
|
||||||
@@ -85,17 +84,9 @@ export const MIME_TYPES = {
|
|||||||
excalidrawlib: "application/vnd.excalidrawlib+json",
|
excalidrawlib: "application/vnd.excalidrawlib+json",
|
||||||
};
|
};
|
||||||
|
|
||||||
export const EXPORT_DATA_TYPES = {
|
|
||||||
excalidraw: "excalidraw",
|
|
||||||
excalidrawClipboard: "excalidraw/clipboard",
|
|
||||||
excalidrawLibrary: "excalidrawlib",
|
|
||||||
} as const;
|
|
||||||
|
|
||||||
export const EXPORT_SOURCE = window.location.origin;
|
|
||||||
|
|
||||||
export const STORAGE_KEYS = {
|
export const STORAGE_KEYS = {
|
||||||
LOCAL_STORAGE_LIBRARY: "excalidraw-library",
|
LOCAL_STORAGE_LIBRARY: "excalidraw-library",
|
||||||
} as const;
|
};
|
||||||
|
|
||||||
// time in milliseconds
|
// time in milliseconds
|
||||||
export const TAP_TWICE_TIMEOUT = 300;
|
export const TAP_TWICE_TIMEOUT = 300;
|
||||||
@@ -104,6 +95,7 @@ export const TITLE_TIMEOUT = 10000;
|
|||||||
export const TOAST_TIMEOUT = 5000;
|
export const TOAST_TIMEOUT = 5000;
|
||||||
export const VERSION_TIMEOUT = 30000;
|
export const VERSION_TIMEOUT = 30000;
|
||||||
export const SCROLL_TIMEOUT = 100;
|
export const SCROLL_TIMEOUT = 100;
|
||||||
|
|
||||||
export const ZOOM_STEP = 0.1;
|
export const ZOOM_STEP = 0.1;
|
||||||
|
|
||||||
// Report a user inactive after IDLE_THRESHOLD milliseconds
|
// Report a user inactive after IDLE_THRESHOLD milliseconds
|
||||||
@@ -118,29 +110,3 @@ export const MODES = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const THEME_FILTER = cssVariables.themeFilter;
|
export const THEME_FILTER = cssVariables.themeFilter;
|
||||||
|
|
||||||
export const URL_QUERY_KEYS = {
|
|
||||||
addLibrary: "addLibrary",
|
|
||||||
} as const;
|
|
||||||
|
|
||||||
export const URL_HASH_KEYS = {
|
|
||||||
addLibrary: "addLibrary",
|
|
||||||
} as const;
|
|
||||||
|
|
||||||
export const DEFAULT_UI_OPTIONS: AppProps["UIOptions"] = {
|
|
||||||
canvasActions: {
|
|
||||||
changeViewBackgroundColor: true,
|
|
||||||
clearCanvas: true,
|
|
||||||
export: true,
|
|
||||||
loadScene: true,
|
|
||||||
saveAsScene: true,
|
|
||||||
saveScene: true,
|
|
||||||
theme: true,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
export const MQ_MAX_WIDTH_PORTRAIT = 730;
|
|
||||||
export const MQ_MAX_WIDTH_LANDSCAPE = 1000;
|
|
||||||
export const MQ_MAX_HEIGHT_LANDSCAPE = 500;
|
|
||||||
|
|
||||||
export const MAX_DECIMALS_FOR_SVG_EXPORT = 2;
|
|
||||||
|
@@ -5,7 +5,6 @@
|
|||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
clip: rect(1px, 1px, 1px, 1px);
|
clip: rect(1px, 1px, 1px, 1px);
|
||||||
white-space: nowrap; /* added line */
|
white-space: nowrap; /* added line */
|
||||||
user-select: none;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.LoadingMessage {
|
.LoadingMessage {
|
||||||
|
@@ -16,12 +16,6 @@
|
|||||||
bottom: 0;
|
bottom: 0;
|
||||||
left: 0;
|
left: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
height: 100%;
|
|
||||||
width: 100%;
|
|
||||||
|
|
||||||
&:focus {
|
|
||||||
outline: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
// serves 2 purposes:
|
// serves 2 purposes:
|
||||||
// 1. prevent selecting text outside the component when double-clicking or
|
// 1. prevent selecting text outside the component when double-clicking or
|
||||||
@@ -53,12 +47,6 @@
|
|||||||
z-index: var(--zIndex-canvas);
|
z-index: var(--zIndex-canvas);
|
||||||
}
|
}
|
||||||
|
|
||||||
#canvas {
|
|
||||||
// Remove the main canvas from document flow to avoid resizeObserver
|
|
||||||
// feedback loop (see https://github.com/excalidraw/excalidraw/pull/3379)
|
|
||||||
position: absolute;
|
|
||||||
}
|
|
||||||
|
|
||||||
&.theme--dark {
|
&.theme--dark {
|
||||||
// The percentage is inspired by
|
// The percentage is inspired by
|
||||||
// https://material.io/design/color/dark-theme.html#properties, which
|
// https://material.io/design/color/dark-theme.html#properties, which
|
||||||
@@ -234,8 +222,7 @@
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
svg {
|
svg {
|
||||||
width: 36px;
|
width: 36px;
|
||||||
height: 14px;
|
height: 18px;
|
||||||
padding: 2px;
|
|
||||||
opacity: 0.6;
|
opacity: 0.6;
|
||||||
}
|
}
|
||||||
&.active svg {
|
&.active svg {
|
||||||
@@ -332,8 +319,8 @@
|
|||||||
.App-menu_bottom {
|
.App-menu_bottom {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
grid-template-columns: min-content auto min-content;
|
grid-template-columns: 1fr auto 1fr;
|
||||||
grid-gap: 15px;
|
grid-gap: 4px;
|
||||||
align-items: flex-start;
|
align-items: flex-start;
|
||||||
cursor: default;
|
cursor: default;
|
||||||
pointer-events: none !important;
|
pointer-events: none !important;
|
||||||
@@ -358,6 +345,10 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.layer-ui__wrapper:not(.disable-pointerEvents) .App-menu_bottom > * {
|
||||||
|
pointer-events: all;
|
||||||
|
}
|
||||||
|
|
||||||
.App-menu_bottom > *:first-child {
|
.App-menu_bottom > *:first-child {
|
||||||
justify-self: flex-start;
|
justify-self: flex-start;
|
||||||
}
|
}
|
||||||
@@ -415,11 +406,13 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
&.dropdown-select--floating {
|
&.dropdown-select--floating {
|
||||||
|
position: absolute;
|
||||||
margin: 0.5em;
|
margin: 0.5em;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.dropdown-select__language.dropdown-select--floating {
|
.dropdown-select__language.dropdown-select--floating {
|
||||||
|
position: absolute;
|
||||||
bottom: 10px;
|
bottom: 10px;
|
||||||
|
|
||||||
:root[dir="ltr"] & {
|
:root[dir="ltr"] & {
|
||||||
@@ -455,29 +448,22 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.help-icon {
|
.help-icon {
|
||||||
|
position: absolute;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
fill: $oc-gray-6;
|
fill: $oc-gray-6;
|
||||||
|
bottom: 14px;
|
||||||
width: 1.5rem;
|
width: 1.5rem;
|
||||||
padding: 0;
|
|
||||||
margin: 0;
|
|
||||||
margin-top: 5px;
|
|
||||||
background: none;
|
|
||||||
color: var(--icon-fill-color);
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
background: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
:root[dir="ltr"] & {
|
:root[dir="ltr"] & {
|
||||||
margin-right: 14px;
|
right: 14px;
|
||||||
}
|
}
|
||||||
|
|
||||||
:root[dir="rtl"] & {
|
:root[dir="rtl"] & {
|
||||||
margin-left: 14px;
|
left: 14px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@include isMobile {
|
@media #{$is-mobile-query} {
|
||||||
aside {
|
aside {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
@@ -493,6 +479,20 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.github-corner {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
z-index: 2;
|
||||||
|
|
||||||
|
:root[dir="ltr"] & {
|
||||||
|
right: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
:root[dir="rtl"] & {
|
||||||
|
left: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.zen-mode-visibility {
|
.zen-mode-visibility {
|
||||||
visibility: visible;
|
visibility: visible;
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
|
@@ -14,12 +14,11 @@
|
|||||||
--focus-highlight-color: #{$oc-blue-2};
|
--focus-highlight-color: #{$oc-blue-2};
|
||||||
--icon-fill-color: #{$oc-black};
|
--icon-fill-color: #{$oc-black};
|
||||||
--icon-green-fill-color: #{$oc-green-9};
|
--icon-green-fill-color: #{$oc-green-9};
|
||||||
--default-bg-color: #{$oc-white};
|
|
||||||
--input-bg-color: #{$oc-white};
|
--input-bg-color: #{$oc-white};
|
||||||
--input-border-color: #{$oc-gray-3};
|
--input-border-color: #{$oc-gray-3};
|
||||||
--input-hover-bg-color: #{$oc-gray-1};
|
--input-hover-bg-color: #{$oc-gray-1};
|
||||||
--input-label-color: #{$oc-gray-7};
|
--input-label-color: #{$oc-gray-7};
|
||||||
--island-bg-color: rgba(255, 255, 255, 0.96);
|
--island-bg-color: rgba(255, 255, 255, 0.9);
|
||||||
--keybinding-color: #{$oc-gray-5};
|
--keybinding-color: #{$oc-gray-5};
|
||||||
--link-color: #{$oc-blue-7};
|
--link-color: #{$oc-blue-7};
|
||||||
--overlay-bg-color: #{transparentize($oc-white, 0.12)};
|
--overlay-bg-color: #{transparentize($oc-white, 0.12)};
|
||||||
@@ -57,12 +56,11 @@
|
|||||||
--focus-highlight-color: #{$oc-blue-6};
|
--focus-highlight-color: #{$oc-blue-6};
|
||||||
--icon-fill-color: #{$oc-gray-4};
|
--icon-fill-color: #{$oc-gray-4};
|
||||||
--icon-green-fill-color: #{$oc-green-4};
|
--icon-green-fill-color: #{$oc-green-4};
|
||||||
--default-bg-color: #121212;
|
|
||||||
--input-bg-color: #121212;
|
--input-bg-color: #121212;
|
||||||
--input-border-color: #2e2e2e;
|
--input-border-color: #2e2e2e;
|
||||||
--input-hover-bg-color: #181818;
|
--input-hover-bg-color: #181818;
|
||||||
--input-label-color: #{$oc-gray-2};
|
--input-label-color: #{$oc-gray-2};
|
||||||
--island-bg-color: rgba(30, 30, 30, 0.98);
|
--island-bg-color: #1e1e1e;
|
||||||
--keybinding-color: #{$oc-gray-6};
|
--keybinding-color: #{$oc-gray-6};
|
||||||
--overlay-bg-color: #{transparentize($oc-gray-8, 0.88)};
|
--overlay-bg-color: #{transparentize($oc-gray-8, 0.88)};
|
||||||
--popup-bg-color: #2c2c2c;
|
--popup-bg-color: #2c2c2c;
|
||||||
|
@@ -1,13 +1,10 @@
|
|||||||
@import "open-color/open-color.scss";
|
@import "open-color/open-color.scss";
|
||||||
|
|
||||||
@mixin isMobile() {
|
// keep up to date with is-mobile.tsx
|
||||||
@at-root .excalidraw--mobile#{&} {
|
$is-mobile-query: "(max-width: 600px), (max-height: 500px) and (max-width: 1000px)";
|
||||||
@content;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$theme-filter: "invert(93%) hue-rotate(180deg)";
|
$theme-filter: "invert(93%) hue-rotate(180deg)";
|
||||||
|
|
||||||
:export {
|
:export {
|
||||||
|
isMobileQuery: unquote($is-mobile-query);
|
||||||
themeFilter: unquote($theme-filter);
|
themeFilter: unquote($theme-filter);
|
||||||
}
|
}
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import { cleanAppStateForExport } from "../appState";
|
import { cleanAppStateForExport } from "../appState";
|
||||||
import { EXPORT_DATA_TYPES } from "../constants";
|
import { MIME_TYPES } from "../constants";
|
||||||
import { clearElementsForExport } from "../element";
|
import { clearElementsForExport } from "../element";
|
||||||
import { CanvasError } from "../errors";
|
import { CanvasError } from "../errors";
|
||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
@@ -7,7 +7,7 @@ import { calculateScrollCenter } from "../scene";
|
|||||||
import { AppState } from "../types";
|
import { AppState } from "../types";
|
||||||
import { isValidExcalidrawData } from "./json";
|
import { isValidExcalidrawData } from "./json";
|
||||||
import { restore } from "./restore";
|
import { restore } from "./restore";
|
||||||
import { ImportedLibraryData } from "./types";
|
import { LibraryData } from "./types";
|
||||||
|
|
||||||
const parseFileContents = async (blob: Blob | File) => {
|
const parseFileContents = async (blob: Blob | File) => {
|
||||||
let contents: string;
|
let contents: string;
|
||||||
@@ -95,7 +95,13 @@ export const loadFromBlob = async (
|
|||||||
elements: clearElementsForExport(data.elements || []),
|
elements: clearElementsForExport(data.elements || []),
|
||||||
appState: {
|
appState: {
|
||||||
theme: localAppState?.theme,
|
theme: localAppState?.theme,
|
||||||
fileHandle: (!blob.type.startsWith("image/") && blob.handle) || null,
|
fileHandle:
|
||||||
|
blob.handle &&
|
||||||
|
["application/json", MIME_TYPES.excalidraw].includes(
|
||||||
|
getMimeType(blob),
|
||||||
|
)
|
||||||
|
? blob.handle
|
||||||
|
: null,
|
||||||
...cleanAppStateForExport(data.appState || {}),
|
...cleanAppStateForExport(data.appState || {}),
|
||||||
...(localAppState
|
...(localAppState
|
||||||
? calculateScrollCenter(data.elements || [], localAppState, null)
|
? calculateScrollCenter(data.elements || [], localAppState, null)
|
||||||
@@ -114,8 +120,8 @@ export const loadFromBlob = async (
|
|||||||
|
|
||||||
export const loadLibraryFromBlob = async (blob: Blob) => {
|
export const loadLibraryFromBlob = async (blob: Blob) => {
|
||||||
const contents = await parseFileContents(blob);
|
const contents = await parseFileContents(blob);
|
||||||
const data: ImportedLibraryData = JSON.parse(contents);
|
const data: LibraryData = JSON.parse(contents);
|
||||||
if (data.type !== EXPORT_DATA_TYPES.excalidrawLibrary) {
|
if (data.type !== "excalidrawlib") {
|
||||||
throw new Error(t("alerts.couldNotLoadInvalidFile"));
|
throw new Error(t("alerts.couldNotLoadInvalidFile"));
|
||||||
}
|
}
|
||||||
return data;
|
return data;
|
||||||
|
@@ -2,7 +2,7 @@ import decodePng from "png-chunks-extract";
|
|||||||
import tEXt from "png-chunk-text";
|
import tEXt from "png-chunk-text";
|
||||||
import encodePng from "png-chunks-encode";
|
import encodePng from "png-chunks-encode";
|
||||||
import { stringToBase64, encode, decode, base64ToString } from "./encode";
|
import { stringToBase64, encode, decode, base64ToString } from "./encode";
|
||||||
import { EXPORT_DATA_TYPES, MIME_TYPES } from "../constants";
|
import { MIME_TYPES } from "../constants";
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
// PNG
|
// PNG
|
||||||
@@ -67,10 +67,7 @@ export const decodePngMetadata = async (blob: Blob) => {
|
|||||||
const encodedData = JSON.parse(metadata.text);
|
const encodedData = JSON.parse(metadata.text);
|
||||||
if (!("encoded" in encodedData)) {
|
if (!("encoded" in encodedData)) {
|
||||||
// legacy, un-encoded scene JSON
|
// legacy, un-encoded scene JSON
|
||||||
if (
|
if ("type" in encodedData && encodedData.type === "excalidraw") {
|
||||||
"type" in encodedData &&
|
|
||||||
encodedData.type === EXPORT_DATA_TYPES.excalidraw
|
|
||||||
) {
|
|
||||||
return metadata.text;
|
return metadata.text;
|
||||||
}
|
}
|
||||||
throw new Error("FAILED");
|
throw new Error("FAILED");
|
||||||
@@ -118,10 +115,7 @@ export const decodeSvgMetadata = async ({ svg }: { svg: string }) => {
|
|||||||
const encodedData = JSON.parse(json);
|
const encodedData = JSON.parse(json);
|
||||||
if (!("encoded" in encodedData)) {
|
if (!("encoded" in encodedData)) {
|
||||||
// legacy, un-encoded scene JSON
|
// legacy, un-encoded scene JSON
|
||||||
if (
|
if ("type" in encodedData && encodedData.type === "excalidraw") {
|
||||||
"type" in encodedData &&
|
|
||||||
encodedData.type === EXPORT_DATA_TYPES.excalidraw
|
|
||||||
) {
|
|
||||||
return json;
|
return json;
|
||||||
}
|
}
|
||||||
throw new Error("FAILED");
|
throw new Error("FAILED");
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import { fileSave } from "browser-fs-access";
|
import { fileSave } from "browser-fs-access";
|
||||||
import {
|
import {
|
||||||
copyBlobToClipboardAsPng,
|
copyCanvasToClipboardAsPng,
|
||||||
copyTextToSystemClipboard,
|
copyTextToSystemClipboard,
|
||||||
} from "../clipboard";
|
} from "../clipboard";
|
||||||
import { NonDeletedExcalidrawElement } from "../element/types";
|
import { NonDeletedExcalidrawElement } from "../element/types";
|
||||||
@@ -18,18 +18,21 @@ export const exportCanvas = async (
|
|||||||
type: ExportType,
|
type: ExportType,
|
||||||
elements: readonly NonDeletedExcalidrawElement[],
|
elements: readonly NonDeletedExcalidrawElement[],
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
|
canvas: HTMLCanvasElement,
|
||||||
{
|
{
|
||||||
exportBackground,
|
exportBackground,
|
||||||
exportPadding = 10,
|
exportPadding = 10,
|
||||||
viewBackgroundColor,
|
viewBackgroundColor,
|
||||||
name,
|
name,
|
||||||
scale = 1,
|
scale = 1,
|
||||||
|
shouldAddWatermark,
|
||||||
}: {
|
}: {
|
||||||
exportBackground: boolean;
|
exportBackground: boolean;
|
||||||
exportPadding?: number;
|
exportPadding?: number;
|
||||||
viewBackgroundColor: string;
|
viewBackgroundColor: string;
|
||||||
name: string;
|
name: string;
|
||||||
scale?: number;
|
scale?: number;
|
||||||
|
shouldAddWatermark: boolean;
|
||||||
},
|
},
|
||||||
) => {
|
) => {
|
||||||
if (elements.length === 0) {
|
if (elements.length === 0) {
|
||||||
@@ -42,6 +45,7 @@ export const exportCanvas = async (
|
|||||||
viewBackgroundColor,
|
viewBackgroundColor,
|
||||||
exportPadding,
|
exportPadding,
|
||||||
scale,
|
scale,
|
||||||
|
shouldAddWatermark,
|
||||||
metadata:
|
metadata:
|
||||||
appState.exportEmbedScene && type === "svg"
|
appState.exportEmbedScene && type === "svg"
|
||||||
? await (
|
? await (
|
||||||
@@ -68,14 +72,14 @@ export const exportCanvas = async (
|
|||||||
viewBackgroundColor,
|
viewBackgroundColor,
|
||||||
exportPadding,
|
exportPadding,
|
||||||
scale,
|
scale,
|
||||||
|
shouldAddWatermark,
|
||||||
});
|
});
|
||||||
tempCanvas.style.display = "none";
|
tempCanvas.style.display = "none";
|
||||||
document.body.appendChild(tempCanvas);
|
document.body.appendChild(tempCanvas);
|
||||||
let blob = await canvasToBlob(tempCanvas);
|
|
||||||
tempCanvas.remove();
|
|
||||||
|
|
||||||
if (type === "png") {
|
if (type === "png") {
|
||||||
const fileName = `${name}.png`;
|
const fileName = `${name}.png`;
|
||||||
|
let blob = await canvasToBlob(tempCanvas);
|
||||||
if (appState.exportEmbedScene) {
|
if (appState.exportEmbedScene) {
|
||||||
blob = await (
|
blob = await (
|
||||||
await import(/* webpackChunkName: "image" */ "./image")
|
await import(/* webpackChunkName: "image" */ "./image")
|
||||||
@@ -91,7 +95,7 @@ export const exportCanvas = async (
|
|||||||
});
|
});
|
||||||
} else if (type === "clipboard") {
|
} else if (type === "clipboard") {
|
||||||
try {
|
try {
|
||||||
await copyBlobToClipboardAsPng(blob);
|
await copyCanvasToClipboardAsPng(tempCanvas);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (error.name === "CANVAS_POSSIBLY_TOO_BIG") {
|
if (error.name === "CANVAS_POSSIBLY_TOO_BIG") {
|
||||||
throw error;
|
throw error;
|
||||||
@@ -99,4 +103,9 @@ export const exportCanvas = async (
|
|||||||
throw new Error(t("alerts.couldNotCopyToClipboard"));
|
throw new Error(t("alerts.couldNotCopyToClipboard"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// clean up the DOM
|
||||||
|
if (tempCanvas !== canvas) {
|
||||||
|
tempCanvas.remove();
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
@@ -1,32 +1,28 @@
|
|||||||
import { fileOpen, fileSave } from "browser-fs-access";
|
import { fileOpen, fileSave } from "browser-fs-access";
|
||||||
import { cleanAppStateForExport } from "../appState";
|
import { cleanAppStateForExport } from "../appState";
|
||||||
import { EXPORT_DATA_TYPES, EXPORT_SOURCE, MIME_TYPES } from "../constants";
|
import { MIME_TYPES } from "../constants";
|
||||||
import { clearElementsForExport } from "../element";
|
import { clearElementsForExport } from "../element";
|
||||||
import { ExcalidrawElement } from "../element/types";
|
import { ExcalidrawElement } from "../element/types";
|
||||||
import { AppState } from "../types";
|
import { AppState } from "../types";
|
||||||
import { loadFromBlob } from "./blob";
|
import { loadFromBlob } from "./blob";
|
||||||
|
import { Library } from "./library";
|
||||||
import {
|
import { ImportedDataState } from "./types";
|
||||||
ExportedDataState,
|
|
||||||
ImportedDataState,
|
|
||||||
ExportedLibraryData,
|
|
||||||
} from "./types";
|
|
||||||
import Library from "./library";
|
|
||||||
|
|
||||||
export const serializeAsJSON = (
|
export const serializeAsJSON = (
|
||||||
elements: readonly ExcalidrawElement[],
|
elements: readonly ExcalidrawElement[],
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
): string => {
|
): string =>
|
||||||
const data: ExportedDataState = {
|
JSON.stringify(
|
||||||
type: EXPORT_DATA_TYPES.excalidraw,
|
{
|
||||||
version: 2,
|
type: "excalidraw",
|
||||||
source: EXPORT_SOURCE,
|
version: 2,
|
||||||
elements: clearElementsForExport(elements),
|
source: window.location.origin,
|
||||||
appState: cleanAppStateForExport(appState),
|
elements: clearElementsForExport(elements),
|
||||||
};
|
appState: cleanAppStateForExport(appState),
|
||||||
|
},
|
||||||
return JSON.stringify(data, null, 2);
|
null,
|
||||||
};
|
2,
|
||||||
|
);
|
||||||
|
|
||||||
export const saveAsJSON = async (
|
export const saveAsJSON = async (
|
||||||
elements: readonly ExcalidrawElement[],
|
elements: readonly ExcalidrawElement[],
|
||||||
@@ -73,7 +69,7 @@ export const isValidExcalidrawData = (data?: {
|
|||||||
appState?: any;
|
appState?: any;
|
||||||
}): data is ImportedDataState => {
|
}): data is ImportedDataState => {
|
||||||
return (
|
return (
|
||||||
data?.type === EXPORT_DATA_TYPES.excalidraw &&
|
data?.type === "excalidraw" &&
|
||||||
(!data.elements ||
|
(!data.elements ||
|
||||||
(Array.isArray(data.elements) &&
|
(Array.isArray(data.elements) &&
|
||||||
(!data.appState || typeof data.appState === "object")))
|
(!data.appState || typeof data.appState === "object")))
|
||||||
@@ -84,20 +80,22 @@ export const isValidLibrary = (json: any) => {
|
|||||||
return (
|
return (
|
||||||
typeof json === "object" &&
|
typeof json === "object" &&
|
||||||
json &&
|
json &&
|
||||||
json.type === EXPORT_DATA_TYPES.excalidrawLibrary &&
|
json.type === "excalidrawlib" &&
|
||||||
json.version === 1
|
json.version === 1
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const saveLibraryAsJSON = async (library: Library) => {
|
export const saveLibraryAsJSON = async () => {
|
||||||
const libraryItems = await library.loadLibrary();
|
const library = await Library.loadLibrary();
|
||||||
const data: ExportedLibraryData = {
|
const serialized = JSON.stringify(
|
||||||
type: EXPORT_DATA_TYPES.excalidrawLibrary,
|
{
|
||||||
version: 1,
|
type: "excalidrawlib",
|
||||||
source: EXPORT_SOURCE,
|
version: 1,
|
||||||
library: libraryItems,
|
library,
|
||||||
};
|
},
|
||||||
const serialized = JSON.stringify(data, null, 2);
|
null,
|
||||||
|
2,
|
||||||
|
);
|
||||||
const fileName = "library.excalidrawlib";
|
const fileName = "library.excalidrawlib";
|
||||||
const blob = new Blob([serialized], {
|
const blob = new Blob([serialized], {
|
||||||
type: MIME_TYPES.excalidrawlib,
|
type: MIME_TYPES.excalidrawlib,
|
||||||
@@ -109,7 +107,7 @@ export const saveLibraryAsJSON = async (library: Library) => {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
export const importLibraryFromJSON = async (library: Library) => {
|
export const importLibraryFromJSON = async () => {
|
||||||
const blob = await fileOpen({
|
const blob = await fileOpen({
|
||||||
description: "Excalidraw library files",
|
description: "Excalidraw library files",
|
||||||
// ToDo: Be over-permissive until https://bugs.webkit.org/show_bug.cgi?id=34442
|
// ToDo: Be over-permissive until https://bugs.webkit.org/show_bug.cgi?id=34442
|
||||||
@@ -118,5 +116,5 @@ export const importLibraryFromJSON = async (library: Library) => {
|
|||||||
extensions: [".json", ".excalidrawlib"],
|
extensions: [".json", ".excalidrawlib"],
|
||||||
*/
|
*/
|
||||||
});
|
});
|
||||||
await library.importLibrary(blob);
|
Library.importLibrary(blob);
|
||||||
};
|
};
|
||||||
|
@@ -1,29 +1,20 @@
|
|||||||
import { loadLibraryFromBlob } from "./blob";
|
import { loadLibraryFromBlob } from "./blob";
|
||||||
import { LibraryItems, LibraryItem } from "../types";
|
import { LibraryItems, LibraryItem } from "../types";
|
||||||
import { restoreElements } from "./restore";
|
import { restoreElements } from "./restore";
|
||||||
|
import { STORAGE_KEYS } from "../constants";
|
||||||
import { getNonDeletedElements } from "../element";
|
import { getNonDeletedElements } from "../element";
|
||||||
import App from "../components/App";
|
import { NonDeleted, ExcalidrawElement } from "../element/types";
|
||||||
|
|
||||||
class Library {
|
export class Library {
|
||||||
private libraryCache: LibraryItems | null = null;
|
private static libraryCache: LibraryItems | null = null;
|
||||||
private app: App;
|
|
||||||
|
|
||||||
constructor(app: App) {
|
static resetLibrary = () => {
|
||||||
this.app = app;
|
Library.libraryCache = null;
|
||||||
}
|
localStorage.removeItem(STORAGE_KEYS.LOCAL_STORAGE_LIBRARY);
|
||||||
|
|
||||||
resetLibrary = async () => {
|
|
||||||
await this.app.props.onLibraryChange?.([]);
|
|
||||||
this.libraryCache = [];
|
|
||||||
};
|
|
||||||
|
|
||||||
restoreLibraryItem = (libraryItem: LibraryItem): LibraryItem | null => {
|
|
||||||
const elements = getNonDeletedElements(restoreElements(libraryItem));
|
|
||||||
return elements.length ? elements : null;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/** imports library (currently merges, removing duplicates) */
|
/** imports library (currently merges, removing duplicates) */
|
||||||
async importLibrary(blob: Blob) {
|
static async importLibrary(blob: Blob) {
|
||||||
const libraryFile = await loadLibraryFromBlob(blob);
|
const libraryFile = await loadLibraryFromBlob(blob);
|
||||||
if (!libraryFile || !libraryFile.library) {
|
if (!libraryFile || !libraryFile.library) {
|
||||||
return;
|
return;
|
||||||
@@ -53,41 +44,37 @@ class Library {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const existingLibraryItems = await this.loadLibrary();
|
const existingLibraryItems = await Library.loadLibrary();
|
||||||
|
|
||||||
const filtered = libraryFile.library!.reduce((acc, libraryItem) => {
|
const filtered = libraryFile.library!.reduce((acc, libraryItem) => {
|
||||||
const restoredItem = this.restoreLibraryItem(libraryItem);
|
const restored = getNonDeletedElements(restoreElements(libraryItem));
|
||||||
if (restoredItem && isUniqueitem(existingLibraryItems, restoredItem)) {
|
if (isUniqueitem(existingLibraryItems, restored)) {
|
||||||
acc.push(restoredItem);
|
acc.push(restored);
|
||||||
}
|
}
|
||||||
return acc;
|
return acc;
|
||||||
}, [] as Mutable<LibraryItems>);
|
}, [] as (readonly NonDeleted<ExcalidrawElement>[])[]);
|
||||||
|
|
||||||
await this.saveLibrary([...existingLibraryItems, ...filtered]);
|
Library.saveLibrary([...existingLibraryItems, ...filtered]);
|
||||||
}
|
}
|
||||||
|
|
||||||
loadLibrary = (): Promise<LibraryItems> => {
|
static loadLibrary = (): Promise<LibraryItems> => {
|
||||||
return new Promise(async (resolve) => {
|
return new Promise(async (resolve) => {
|
||||||
if (this.libraryCache) {
|
if (Library.libraryCache) {
|
||||||
return resolve(JSON.parse(JSON.stringify(this.libraryCache)));
|
return resolve(JSON.parse(JSON.stringify(Library.libraryCache)));
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const libraryItems = this.app.libraryItemsFromStorage;
|
const data = localStorage.getItem(STORAGE_KEYS.LOCAL_STORAGE_LIBRARY);
|
||||||
if (!libraryItems) {
|
if (!data) {
|
||||||
return resolve([]);
|
return resolve([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
const items = libraryItems.reduce((acc, item) => {
|
const items = (JSON.parse(data) as LibraryItems).map((elements) =>
|
||||||
const restoredItem = this.restoreLibraryItem(item);
|
restoreElements(elements),
|
||||||
if (restoredItem) {
|
) as Mutable<LibraryItems>;
|
||||||
acc.push(item);
|
|
||||||
}
|
|
||||||
return acc;
|
|
||||||
}, [] as Mutable<LibraryItems>);
|
|
||||||
|
|
||||||
// clone to ensure we don't mutate the cached library elements in the app
|
// clone to ensure we don't mutate the cached library elements in the app
|
||||||
this.libraryCache = JSON.parse(JSON.stringify(items));
|
Library.libraryCache = JSON.parse(JSON.stringify(items));
|
||||||
|
|
||||||
resolve(items);
|
resolve(items);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@@ -97,19 +84,17 @@ class Library {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
saveLibrary = async (items: LibraryItems) => {
|
static saveLibrary = (items: LibraryItems) => {
|
||||||
const prevLibraryItems = this.libraryCache;
|
const prevLibraryItems = Library.libraryCache;
|
||||||
try {
|
try {
|
||||||
const serializedItems = JSON.stringify(items);
|
const serializedItems = JSON.stringify(items);
|
||||||
// cache optimistically so that the app has access to the latest
|
// cache optimistically so that consumers have access to the latest
|
||||||
// immediately
|
// immediately
|
||||||
this.libraryCache = JSON.parse(serializedItems);
|
Library.libraryCache = JSON.parse(serializedItems);
|
||||||
await this.app.props.onLibraryChange?.(items);
|
localStorage.setItem(STORAGE_KEYS.LOCAL_STORAGE_LIBRARY, serializedItems);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
this.libraryCache = prevLibraryItems;
|
Library.libraryCache = prevLibraryItems;
|
||||||
throw error;
|
console.error(error);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export default Library;
|
|
||||||
|
@@ -4,7 +4,7 @@ import {
|
|||||||
ExcalidrawSelectionElement,
|
ExcalidrawSelectionElement,
|
||||||
} from "../element/types";
|
} from "../element/types";
|
||||||
import { AppState, NormalizedZoomValue } from "../types";
|
import { AppState, NormalizedZoomValue } from "../types";
|
||||||
import { ImportedDataState } from "./types";
|
import { DataState, ImportedDataState } from "./types";
|
||||||
import { isInvisiblySmallElement, getNormalizedDimensions } from "../element";
|
import { isInvisiblySmallElement, getNormalizedDimensions } from "../element";
|
||||||
import { isLinearElementType } from "../element/typeChecks";
|
import { isLinearElementType } from "../element/typeChecks";
|
||||||
import { randomId } from "../random";
|
import { randomId } from "../random";
|
||||||
@@ -15,31 +15,6 @@ import {
|
|||||||
DEFAULT_VERTICAL_ALIGN,
|
DEFAULT_VERTICAL_ALIGN,
|
||||||
} from "../constants";
|
} from "../constants";
|
||||||
import { getDefaultAppState } from "../appState";
|
import { getDefaultAppState } from "../appState";
|
||||||
import { LinearElementEditor } from "../element/linearElementEditor";
|
|
||||||
|
|
||||||
type RestoredAppState = Omit<
|
|
||||||
AppState,
|
|
||||||
"offsetTop" | "offsetLeft" | "width" | "height"
|
|
||||||
>;
|
|
||||||
|
|
||||||
export const AllowedExcalidrawElementTypes: Record<
|
|
||||||
ExcalidrawElement["type"],
|
|
||||||
true
|
|
||||||
> = {
|
|
||||||
selection: true,
|
|
||||||
text: true,
|
|
||||||
rectangle: true,
|
|
||||||
diamond: true,
|
|
||||||
ellipse: true,
|
|
||||||
line: true,
|
|
||||||
arrow: true,
|
|
||||||
freedraw: true,
|
|
||||||
};
|
|
||||||
|
|
||||||
export type RestoredDataState = {
|
|
||||||
elements: ExcalidrawElement[];
|
|
||||||
appState: RestoredAppState;
|
|
||||||
};
|
|
||||||
|
|
||||||
const getFontFamilyByName = (fontFamilyName: string): FontFamily => {
|
const getFontFamilyByName = (fontFamilyName: string): FontFamily => {
|
||||||
for (const [id, fontFamilyString] of Object.entries(FONT_FAMILY)) {
|
for (const [id, fontFamilyString] of Object.entries(FONT_FAMILY)) {
|
||||||
@@ -50,18 +25,12 @@ const getFontFamilyByName = (fontFamilyName: string): FontFamily => {
|
|||||||
return DEFAULT_FONT_FAMILY;
|
return DEFAULT_FONT_FAMILY;
|
||||||
};
|
};
|
||||||
|
|
||||||
const restoreElementWithProperties = <
|
const restoreElementWithProperties = <T extends ExcalidrawElement>(
|
||||||
T extends ExcalidrawElement,
|
|
||||||
K extends keyof Omit<
|
|
||||||
Required<T>,
|
|
||||||
Exclude<keyof ExcalidrawElement, "type" | "x" | "y">
|
|
||||||
>
|
|
||||||
>(
|
|
||||||
element: Required<T>,
|
element: Required<T>,
|
||||||
extra: Pick<T, K>,
|
extra: Omit<Required<T>, keyof ExcalidrawElement>,
|
||||||
): T => {
|
): T => {
|
||||||
const base: Pick<T, keyof ExcalidrawElement> = {
|
const base: Pick<T, keyof ExcalidrawElement> = {
|
||||||
type: (extra as Partial<T>).type || element.type,
|
type: element.type,
|
||||||
// all elements must have version > 0 so getSceneVersion() will pick up
|
// all elements must have version > 0 so getSceneVersion() will pick up
|
||||||
// newly added elements
|
// newly added elements
|
||||||
version: element.version || 1,
|
version: element.version || 1,
|
||||||
@@ -74,8 +43,8 @@ const restoreElementWithProperties = <
|
|||||||
roughness: element.roughness ?? 1,
|
roughness: element.roughness ?? 1,
|
||||||
opacity: element.opacity == null ? 100 : element.opacity,
|
opacity: element.opacity == null ? 100 : element.opacity,
|
||||||
angle: element.angle || 0,
|
angle: element.angle || 0,
|
||||||
x: (extra as Partial<T>).x ?? element.x ?? 0,
|
x: element.x || 0,
|
||||||
y: (extra as Partial<T>).y ?? element.y ?? 0,
|
y: element.y || 0,
|
||||||
strokeColor: element.strokeColor,
|
strokeColor: element.strokeColor,
|
||||||
backgroundColor: element.backgroundColor,
|
backgroundColor: element.backgroundColor,
|
||||||
width: element.width || 0,
|
width: element.width || 0,
|
||||||
@@ -118,51 +87,28 @@ const restoreElement = (
|
|||||||
textAlign: element.textAlign || DEFAULT_TEXT_ALIGN,
|
textAlign: element.textAlign || DEFAULT_TEXT_ALIGN,
|
||||||
verticalAlign: element.verticalAlign || DEFAULT_VERTICAL_ALIGN,
|
verticalAlign: element.verticalAlign || DEFAULT_VERTICAL_ALIGN,
|
||||||
});
|
});
|
||||||
case "freedraw": {
|
|
||||||
return restoreElementWithProperties(element, {
|
|
||||||
points: element.points,
|
|
||||||
lastCommittedPoint: null,
|
|
||||||
simulatePressure: element.simulatePressure,
|
|
||||||
pressures: element.pressures,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
case "line":
|
|
||||||
// @ts-ignore LEGACY type
|
|
||||||
// eslint-disable-next-line no-fallthrough
|
|
||||||
case "draw":
|
case "draw":
|
||||||
|
case "line":
|
||||||
case "arrow": {
|
case "arrow": {
|
||||||
const {
|
const {
|
||||||
startArrowhead = null,
|
startArrowhead = null,
|
||||||
endArrowhead = element.type === "arrow" ? "arrow" : null,
|
endArrowhead = element.type === "arrow" ? "arrow" : null,
|
||||||
} = element;
|
} = element;
|
||||||
|
|
||||||
let x = element.x;
|
|
||||||
let y = element.y;
|
|
||||||
let points = // migrate old arrow model to new one
|
|
||||||
!Array.isArray(element.points) || element.points.length < 2
|
|
||||||
? [
|
|
||||||
[0, 0],
|
|
||||||
[element.width, element.height],
|
|
||||||
]
|
|
||||||
: element.points;
|
|
||||||
|
|
||||||
if (points[0][0] !== 0 || points[0][1] !== 0) {
|
|
||||||
({ points, x, y } = LinearElementEditor.getNormalizedPoints(element));
|
|
||||||
}
|
|
||||||
|
|
||||||
return restoreElementWithProperties(element, {
|
return restoreElementWithProperties(element, {
|
||||||
type:
|
|
||||||
(element.type as ExcalidrawElement["type"] | "draw") === "draw"
|
|
||||||
? "line"
|
|
||||||
: element.type,
|
|
||||||
startBinding: element.startBinding,
|
startBinding: element.startBinding,
|
||||||
endBinding: element.endBinding,
|
endBinding: element.endBinding,
|
||||||
|
points:
|
||||||
|
// migrate old arrow model to new one
|
||||||
|
!Array.isArray(element.points) || element.points.length < 2
|
||||||
|
? [
|
||||||
|
[0, 0],
|
||||||
|
[element.width, element.height],
|
||||||
|
]
|
||||||
|
: element.points,
|
||||||
lastCommittedPoint: null,
|
lastCommittedPoint: null,
|
||||||
startArrowhead,
|
startArrowhead,
|
||||||
endArrowhead,
|
endArrowhead,
|
||||||
points,
|
|
||||||
x,
|
|
||||||
y,
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
// generic elements
|
// generic elements
|
||||||
@@ -198,7 +144,7 @@ export const restoreElements = (
|
|||||||
export const restoreAppState = (
|
export const restoreAppState = (
|
||||||
appState: ImportedDataState["appState"],
|
appState: ImportedDataState["appState"],
|
||||||
localAppState: Partial<AppState> | null,
|
localAppState: Partial<AppState> | null,
|
||||||
): RestoredAppState => {
|
): AppState => {
|
||||||
appState = appState || {};
|
appState = appState || {};
|
||||||
|
|
||||||
const defaultAppState = getDefaultAppState();
|
const defaultAppState = getDefaultAppState();
|
||||||
@@ -220,9 +166,8 @@ export const restoreAppState = (
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
...nextAppState,
|
...nextAppState,
|
||||||
elementType: AllowedExcalidrawElementTypes[nextAppState.elementType]
|
offsetLeft: appState.offsetLeft || 0,
|
||||||
? nextAppState.elementType
|
offsetTop: appState.offsetTop || 0,
|
||||||
: "selection",
|
|
||||||
// Migrates from previous version where appState.zoom was a number
|
// Migrates from previous version where appState.zoom was a number
|
||||||
zoom:
|
zoom:
|
||||||
typeof appState.zoom === "number"
|
typeof appState.zoom === "number"
|
||||||
@@ -243,7 +188,7 @@ export const restore = (
|
|||||||
* Supply `null` if you can't get access to it.
|
* Supply `null` if you can't get access to it.
|
||||||
*/
|
*/
|
||||||
localAppState: Partial<AppState> | null | undefined,
|
localAppState: Partial<AppState> | null | undefined,
|
||||||
): RestoredDataState => {
|
): DataState => {
|
||||||
return {
|
return {
|
||||||
elements: restoreElements(data?.elements),
|
elements: restoreElements(data?.elements),
|
||||||
appState: restoreAppState(data?.appState, localAppState || null),
|
appState: restoreAppState(data?.appState, localAppState || null),
|
||||||
|
@@ -1,30 +1,26 @@
|
|||||||
import { ExcalidrawElement } from "../element/types";
|
import { ExcalidrawElement } from "../element/types";
|
||||||
import { AppState, LibraryItems } from "../types";
|
import { AppState, LibraryItems } from "../types";
|
||||||
import type { cleanAppStateForExport } from "../appState";
|
|
||||||
|
|
||||||
export interface ExportedDataState {
|
export interface DataState {
|
||||||
type: string;
|
type?: string;
|
||||||
version: number;
|
version?: string;
|
||||||
source: string;
|
source?: string;
|
||||||
elements: readonly ExcalidrawElement[];
|
elements: readonly ExcalidrawElement[];
|
||||||
appState: ReturnType<typeof cleanAppStateForExport>;
|
appState: MarkOptional<AppState, "offsetTop" | "offsetLeft">;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ImportedDataState {
|
export interface ImportedDataState {
|
||||||
|
type?: string;
|
||||||
|
version?: string;
|
||||||
|
source?: string;
|
||||||
|
elements?: DataState["elements"] | null;
|
||||||
|
appState?: Partial<DataState["appState"]> | null;
|
||||||
|
scrollToContent?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface LibraryData {
|
||||||
type?: string;
|
type?: string;
|
||||||
version?: number;
|
version?: number;
|
||||||
source?: string;
|
source?: string;
|
||||||
elements?: readonly ExcalidrawElement[] | null;
|
library?: LibraryItems;
|
||||||
appState?: Readonly<Partial<AppState>> | null;
|
|
||||||
scrollToContent?: boolean;
|
|
||||||
libraryItems?: LibraryItems;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ExportedLibraryData {
|
|
||||||
type: string;
|
|
||||||
version: number;
|
|
||||||
source: string;
|
|
||||||
library: LibraryItems;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface ImportedLibraryData extends Partial<ExportedLibraryData> {}
|
|
||||||
|
@@ -1,9 +1,4 @@
|
|||||||
import {
|
import { ExcalidrawElement, ExcalidrawLinearElement, Arrowhead } from "./types";
|
||||||
ExcalidrawElement,
|
|
||||||
ExcalidrawLinearElement,
|
|
||||||
Arrowhead,
|
|
||||||
ExcalidrawFreeDrawElement,
|
|
||||||
} from "./types";
|
|
||||||
import { distance2d, rotate } from "../math";
|
import { distance2d, rotate } from "../math";
|
||||||
import rough from "roughjs/bin/rough";
|
import rough from "roughjs/bin/rough";
|
||||||
import { Drawable, Op } from "roughjs/bin/core";
|
import { Drawable, Op } from "roughjs/bin/core";
|
||||||
@@ -12,7 +7,7 @@ import {
|
|||||||
getShapeForElement,
|
getShapeForElement,
|
||||||
generateRoughOptions,
|
generateRoughOptions,
|
||||||
} from "../renderer/renderElement";
|
} from "../renderer/renderElement";
|
||||||
import { isFreeDrawElement, isLinearElement } from "./typeChecks";
|
import { isLinearElement } from "./typeChecks";
|
||||||
import { rescalePoints } from "../points";
|
import { rescalePoints } from "../points";
|
||||||
|
|
||||||
// x and y position of top left corner, x and y position of bottom right corner
|
// x and y position of top left corner, x and y position of bottom right corner
|
||||||
@@ -23,9 +18,7 @@ export type Bounds = readonly [number, number, number, number];
|
|||||||
export const getElementAbsoluteCoords = (
|
export const getElementAbsoluteCoords = (
|
||||||
element: ExcalidrawElement,
|
element: ExcalidrawElement,
|
||||||
): Bounds => {
|
): Bounds => {
|
||||||
if (isFreeDrawElement(element)) {
|
if (isLinearElement(element)) {
|
||||||
return getFreeDrawElementAbsoluteCoords(element);
|
|
||||||
} else if (isLinearElement(element)) {
|
|
||||||
return getLinearElementAbsoluteCoords(element);
|
return getLinearElementAbsoluteCoords(element);
|
||||||
}
|
}
|
||||||
return [
|
return [
|
||||||
@@ -127,42 +120,9 @@ const getMinMaxXYFromCurvePathOps = (
|
|||||||
return [minX, minY, maxX, maxY];
|
return [minX, minY, maxX, maxY];
|
||||||
};
|
};
|
||||||
|
|
||||||
const getBoundsFromPoints = (
|
|
||||||
points: ExcalidrawFreeDrawElement["points"],
|
|
||||||
): [number, number, number, number] => {
|
|
||||||
let minX = Infinity;
|
|
||||||
let minY = Infinity;
|
|
||||||
let maxX = -Infinity;
|
|
||||||
let maxY = -Infinity;
|
|
||||||
|
|
||||||
for (const [x, y] of points) {
|
|
||||||
minX = Math.min(minX, x);
|
|
||||||
minY = Math.min(minY, y);
|
|
||||||
maxX = Math.max(maxX, x);
|
|
||||||
maxY = Math.max(maxY, y);
|
|
||||||
}
|
|
||||||
|
|
||||||
return [minX, minY, maxX, maxY];
|
|
||||||
};
|
|
||||||
|
|
||||||
const getFreeDrawElementAbsoluteCoords = (
|
|
||||||
element: ExcalidrawFreeDrawElement,
|
|
||||||
): [number, number, number, number] => {
|
|
||||||
const [minX, minY, maxX, maxY] = getBoundsFromPoints(element.points);
|
|
||||||
|
|
||||||
return [
|
|
||||||
minX + element.x,
|
|
||||||
minY + element.y,
|
|
||||||
maxX + element.x,
|
|
||||||
maxY + element.y,
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
const getLinearElementAbsoluteCoords = (
|
const getLinearElementAbsoluteCoords = (
|
||||||
element: ExcalidrawLinearElement,
|
element: ExcalidrawLinearElement,
|
||||||
): [number, number, number, number] => {
|
): [number, number, number, number] => {
|
||||||
let coords: [number, number, number, number];
|
|
||||||
|
|
||||||
if (element.points.length < 2 || !getShapeForElement(element)) {
|
if (element.points.length < 2 || !getShapeForElement(element)) {
|
||||||
// XXX this is just a poor estimate and not very useful
|
// XXX this is just a poor estimate and not very useful
|
||||||
const { minX, minY, maxX, maxY } = element.points.reduce(
|
const { minX, minY, maxX, maxY } = element.points.reduce(
|
||||||
@@ -177,21 +137,7 @@ const getLinearElementAbsoluteCoords = (
|
|||||||
},
|
},
|
||||||
{ minX: Infinity, minY: Infinity, maxX: -Infinity, maxY: -Infinity },
|
{ minX: Infinity, minY: Infinity, maxX: -Infinity, maxY: -Infinity },
|
||||||
);
|
);
|
||||||
coords = [
|
return [
|
||||||
minX + element.x,
|
|
||||||
minY + element.y,
|
|
||||||
maxX + element.x,
|
|
||||||
maxY + element.y,
|
|
||||||
];
|
|
||||||
} else {
|
|
||||||
const shape = getShapeForElement(element) as Drawable[];
|
|
||||||
|
|
||||||
// first element is always the curve
|
|
||||||
const ops = getCurvePathOps(shape[0]);
|
|
||||||
|
|
||||||
const [minX, minY, maxX, maxY] = getMinMaxXYFromCurvePathOps(ops);
|
|
||||||
|
|
||||||
coords = [
|
|
||||||
minX + element.x,
|
minX + element.x,
|
||||||
minY + element.y,
|
minY + element.y,
|
||||||
maxX + element.x,
|
maxX + element.x,
|
||||||
@@ -199,7 +145,19 @@ const getLinearElementAbsoluteCoords = (
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
return coords;
|
const shape = getShapeForElement(element) as Drawable[];
|
||||||
|
|
||||||
|
// first element is always the curve
|
||||||
|
const ops = getCurvePathOps(shape[0]);
|
||||||
|
|
||||||
|
const [minX, minY, maxX, maxY] = getMinMaxXYFromCurvePathOps(ops);
|
||||||
|
|
||||||
|
return [
|
||||||
|
minX + element.x,
|
||||||
|
minY + element.y,
|
||||||
|
maxX + element.x,
|
||||||
|
maxY + element.y,
|
||||||
|
];
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getArrowheadPoints = (
|
export const getArrowheadPoints = (
|
||||||
@@ -260,34 +218,20 @@ export const getArrowheadPoints = (
|
|||||||
dot: 15,
|
dot: 15,
|
||||||
}[arrowhead]; // pixels (will differ for each arrowhead)
|
}[arrowhead]; // pixels (will differ for each arrowhead)
|
||||||
|
|
||||||
let length = 0;
|
const length = element.points.reduce((total, [cx, cy], idx, points) => {
|
||||||
|
const [px, py] = idx > 0 ? points[idx - 1] : [0, 0];
|
||||||
if (arrowhead === "arrow") {
|
return total + Math.hypot(cx - px, cy - py);
|
||||||
// Length for -> arrows is based on the length of the last section
|
}, 0);
|
||||||
const [cx, cy] = element.points[element.points.length - 1];
|
|
||||||
const [px, py] =
|
|
||||||
element.points.length > 1
|
|
||||||
? element.points[element.points.length - 2]
|
|
||||||
: [0, 0];
|
|
||||||
|
|
||||||
length = Math.hypot(cx - px, cy - py);
|
|
||||||
} else {
|
|
||||||
// Length for other arrowhead types is based on the total length of the line
|
|
||||||
for (let i = 0; i < element.points.length; i++) {
|
|
||||||
const [px, py] = element.points[i - 1] || [0, 0];
|
|
||||||
const [cx, cy] = element.points[i];
|
|
||||||
length += Math.hypot(cx - px, cy - py);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Scale down the arrowhead until we hit a certain size so that it doesn't look weird.
|
// Scale down the arrowhead until we hit a certain size so that it doesn't look weird.
|
||||||
// This value is selected by minimizing a minimum size with the last segment of the arrowhead
|
// This value is selected by minimizing a minimum size with the whole length of the
|
||||||
|
// arrowhead instead of last segment of the arrowhead.
|
||||||
const minSize = Math.min(size, length / 2);
|
const minSize = Math.min(size, length / 2);
|
||||||
const xs = x2 - nx * minSize;
|
const xs = x2 - nx * minSize;
|
||||||
const ys = y2 - ny * minSize;
|
const ys = y2 - ny * minSize;
|
||||||
|
|
||||||
if (arrowhead === "dot") {
|
if (arrowhead === "dot") {
|
||||||
const r = Math.hypot(ys - y2, xs - x2) + element.strokeWidth;
|
const r = Math.hypot(ys - y2, xs - x2);
|
||||||
return [x2, y2, r];
|
return [x2, y2, r];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -333,31 +277,16 @@ const getLinearElementRotatedBounds = (
|
|||||||
return getMinMaxXYFromCurvePathOps(ops, transformXY);
|
return getMinMaxXYFromCurvePathOps(ops, transformXY);
|
||||||
};
|
};
|
||||||
|
|
||||||
// We could cache this stuff
|
|
||||||
export const getElementBounds = (
|
export const getElementBounds = (
|
||||||
element: ExcalidrawElement,
|
element: ExcalidrawElement,
|
||||||
): [number, number, number, number] => {
|
): [number, number, number, number] => {
|
||||||
let bounds: [number, number, number, number];
|
|
||||||
|
|
||||||
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element);
|
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element);
|
||||||
const cx = (x1 + x2) / 2;
|
const cx = (x1 + x2) / 2;
|
||||||
const cy = (y1 + y2) / 2;
|
const cy = (y1 + y2) / 2;
|
||||||
if (isFreeDrawElement(element)) {
|
if (isLinearElement(element)) {
|
||||||
const [minX, minY, maxX, maxY] = getBoundsFromPoints(
|
return getLinearElementRotatedBounds(element, cx, cy);
|
||||||
element.points.map(([x, y]) =>
|
}
|
||||||
rotate(x, y, cx - element.x, cy - element.y, element.angle),
|
if (element.type === "diamond") {
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
return [
|
|
||||||
minX + element.x,
|
|
||||||
minY + element.y,
|
|
||||||
maxX + element.x,
|
|
||||||
maxY + element.y,
|
|
||||||
];
|
|
||||||
} else if (isLinearElement(element)) {
|
|
||||||
bounds = getLinearElementRotatedBounds(element, cx, cy);
|
|
||||||
} else if (element.type === "diamond") {
|
|
||||||
const [x11, y11] = rotate(cx, y1, cx, cy, element.angle);
|
const [x11, y11] = rotate(cx, y1, cx, cy, element.angle);
|
||||||
const [x12, y12] = rotate(cx, y2, cx, cy, element.angle);
|
const [x12, y12] = rotate(cx, y2, cx, cy, element.angle);
|
||||||
const [x22, y22] = rotate(x1, cy, cx, cy, element.angle);
|
const [x22, y22] = rotate(x1, cy, cx, cy, element.angle);
|
||||||
@@ -366,28 +295,26 @@ export const getElementBounds = (
|
|||||||
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);
|
||||||
const maxY = Math.max(y11, y12, y22, y21);
|
const maxY = Math.max(y11, y12, y22, y21);
|
||||||
bounds = [minX, minY, maxX, maxY];
|
return [minX, minY, maxX, maxY];
|
||||||
} else if (element.type === "ellipse") {
|
}
|
||||||
|
if (element.type === "ellipse") {
|
||||||
const w = (x2 - x1) / 2;
|
const w = (x2 - x1) / 2;
|
||||||
const h = (y2 - y1) / 2;
|
const h = (y2 - y1) / 2;
|
||||||
const cos = Math.cos(element.angle);
|
const cos = Math.cos(element.angle);
|
||||||
const sin = Math.sin(element.angle);
|
const sin = Math.sin(element.angle);
|
||||||
const ww = Math.hypot(w * cos, h * sin);
|
const ww = Math.hypot(w * cos, h * sin);
|
||||||
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];
|
return [cx - ww, cy - hh, cx + ww, cy + hh];
|
||||||
} else {
|
|
||||||
const [x11, y11] = rotate(x1, y1, cx, cy, element.angle);
|
|
||||||
const [x12, y12] = rotate(x1, y2, cx, cy, element.angle);
|
|
||||||
const [x22, y22] = rotate(x2, y2, cx, cy, element.angle);
|
|
||||||
const [x21, y21] = rotate(x2, y1, cx, cy, element.angle);
|
|
||||||
const minX = Math.min(x11, x12, x22, x21);
|
|
||||||
const minY = Math.min(y11, y12, y22, y21);
|
|
||||||
const maxX = Math.max(x11, x12, x22, x21);
|
|
||||||
const maxY = Math.max(y11, y12, y22, y21);
|
|
||||||
bounds = [minX, minY, maxX, maxY];
|
|
||||||
}
|
}
|
||||||
|
const [x11, y11] = rotate(x1, y1, cx, cy, element.angle);
|
||||||
return bounds;
|
const [x12, y12] = rotate(x1, y2, cx, cy, element.angle);
|
||||||
|
const [x22, y22] = rotate(x2, y2, cx, cy, element.angle);
|
||||||
|
const [x21, y21] = rotate(x2, y1, cx, cy, element.angle);
|
||||||
|
const minX = Math.min(x11, x12, x22, x21);
|
||||||
|
const minY = Math.min(y11, y12, y22, y21);
|
||||||
|
const maxX = Math.max(x11, x12, x22, x21);
|
||||||
|
const maxY = Math.max(y11, y12, y22, y21);
|
||||||
|
return [minX, minY, maxX, maxY];
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getCommonBounds = (
|
export const getCommonBounds = (
|
||||||
@@ -418,7 +345,7 @@ export const getResizedElementAbsoluteCoords = (
|
|||||||
nextWidth: number,
|
nextWidth: number,
|
||||||
nextHeight: number,
|
nextHeight: number,
|
||||||
): [number, number, number, number] => {
|
): [number, number, number, number] => {
|
||||||
if (!(isLinearElement(element) || isFreeDrawElement(element))) {
|
if (!isLinearElement(element)) {
|
||||||
return [
|
return [
|
||||||
element.x,
|
element.x,
|
||||||
element.y,
|
element.y,
|
||||||
@@ -433,29 +360,16 @@ export const getResizedElementAbsoluteCoords = (
|
|||||||
rescalePoints(1, nextHeight, element.points),
|
rescalePoints(1, nextHeight, element.points),
|
||||||
);
|
);
|
||||||
|
|
||||||
let bounds: [number, number, number, number];
|
const gen = rough.generator();
|
||||||
|
const curve =
|
||||||
if (isFreeDrawElement(element)) {
|
element.strokeSharpness === "sharp"
|
||||||
// Free Draw
|
? gen.linearPath(
|
||||||
bounds = getBoundsFromPoints(points);
|
points as [number, number][],
|
||||||
} else {
|
generateRoughOptions(element),
|
||||||
// Line
|
)
|
||||||
const gen = rough.generator();
|
: gen.curve(points as [number, number][], generateRoughOptions(element));
|
||||||
const curve =
|
const ops = getCurvePathOps(curve);
|
||||||
element.strokeSharpness === "sharp"
|
const [minX, minY, maxX, maxY] = getMinMaxXYFromCurvePathOps(ops);
|
||||||
? gen.linearPath(
|
|
||||||
points as [number, number][],
|
|
||||||
generateRoughOptions(element),
|
|
||||||
)
|
|
||||||
: gen.curve(
|
|
||||||
points as [number, number][],
|
|
||||||
generateRoughOptions(element),
|
|
||||||
);
|
|
||||||
const ops = getCurvePathOps(curve);
|
|
||||||
bounds = getMinMaxXYFromCurvePathOps(ops);
|
|
||||||
}
|
|
||||||
|
|
||||||
const [minX, minY, maxX, maxY] = bounds;
|
|
||||||
return [
|
return [
|
||||||
minX + element.x,
|
minX + element.x,
|
||||||
minY + element.y,
|
minY + element.y,
|
||||||
|
@@ -4,13 +4,7 @@ import * as GADirection from "../gadirections";
|
|||||||
import * as GALine from "../galines";
|
import * as GALine from "../galines";
|
||||||
import * as GATransform from "../gatransforms";
|
import * as GATransform from "../gatransforms";
|
||||||
|
|
||||||
import {
|
import { isPathALoop, isPointInPolygon, rotate } from "../math";
|
||||||
distance2d,
|
|
||||||
rotatePoint,
|
|
||||||
isPathALoop,
|
|
||||||
isPointInPolygon,
|
|
||||||
rotate,
|
|
||||||
} from "../math";
|
|
||||||
import { pointsOnBezierCurves } from "points-on-curve";
|
import { pointsOnBezierCurves } from "points-on-curve";
|
||||||
|
|
||||||
import {
|
import {
|
||||||
@@ -22,7 +16,6 @@ import {
|
|||||||
ExcalidrawTextElement,
|
ExcalidrawTextElement,
|
||||||
ExcalidrawEllipseElement,
|
ExcalidrawEllipseElement,
|
||||||
NonDeleted,
|
NonDeleted,
|
||||||
ExcalidrawFreeDrawElement,
|
|
||||||
} from "./types";
|
} from "./types";
|
||||||
|
|
||||||
import { getElementAbsoluteCoords, getCurvePathOps, Bounds } from "./bounds";
|
import { getElementAbsoluteCoords, getCurvePathOps, Bounds } from "./bounds";
|
||||||
@@ -37,17 +30,10 @@ const isElementDraggableFromInside = (
|
|||||||
if (element.type === "arrow") {
|
if (element.type === "arrow") {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (element.type === "freedraw") {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
const isDraggableFromInside = element.backgroundColor !== "transparent";
|
const isDraggableFromInside = element.backgroundColor !== "transparent";
|
||||||
|
if (element.type === "line" || element.type === "draw") {
|
||||||
if (element.type === "line") {
|
|
||||||
return isDraggableFromInside && isPathALoop(element.points);
|
return isDraggableFromInside && isPathALoop(element.points);
|
||||||
}
|
}
|
||||||
|
|
||||||
return isDraggableFromInside;
|
return isDraggableFromInside;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -95,7 +81,6 @@ const isHittingElementNotConsideringBoundingBox = (
|
|||||||
: isElementDraggableFromInside(element)
|
: isElementDraggableFromInside(element)
|
||||||
? isInsideCheck
|
? isInsideCheck
|
||||||
: isNearCheck;
|
: isNearCheck;
|
||||||
|
|
||||||
return hitTestPointAgainstElement({ element, point, threshold, check });
|
return hitTestPointAgainstElement({ element, point, threshold, check });
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -166,20 +151,9 @@ const hitTestPointAgainstElement = (args: HitTestArgs): boolean => {
|
|||||||
case "ellipse":
|
case "ellipse":
|
||||||
const distance = distanceToBindableElement(args.element, args.point);
|
const distance = distanceToBindableElement(args.element, args.point);
|
||||||
return args.check(distance, args.threshold);
|
return args.check(distance, args.threshold);
|
||||||
case "freedraw": {
|
|
||||||
if (
|
|
||||||
!args.check(
|
|
||||||
distanceToRectangle(args.element, args.point),
|
|
||||||
args.threshold,
|
|
||||||
)
|
|
||||||
) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return hitTestFreeDrawElement(args.element, args.point, args.threshold);
|
|
||||||
}
|
|
||||||
case "arrow":
|
case "arrow":
|
||||||
case "line":
|
case "line":
|
||||||
|
case "draw":
|
||||||
return hitTestLinear(args);
|
return hitTestLinear(args);
|
||||||
case "selection":
|
case "selection":
|
||||||
console.warn(
|
console.warn(
|
||||||
@@ -221,10 +195,7 @@ const isOutsideCheck = (distance: number, threshold: number): boolean => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const distanceToRectangle = (
|
const distanceToRectangle = (
|
||||||
element:
|
element: ExcalidrawRectangleElement | ExcalidrawTextElement,
|
||||||
| ExcalidrawRectangleElement
|
|
||||||
| ExcalidrawTextElement
|
|
||||||
| ExcalidrawFreeDrawElement,
|
|
||||||
point: Point,
|
point: Point,
|
||||||
): number => {
|
): number => {
|
||||||
const [, pointRel, hwidth, hheight] = pointRelativeToElement(element, point);
|
const [, pointRel, hwidth, hheight] = pointRelativeToElement(element, point);
|
||||||
@@ -296,71 +267,6 @@ const ellipseParamsForTest = (
|
|||||||
return [pointRel, tangent];
|
return [pointRel, tangent];
|
||||||
};
|
};
|
||||||
|
|
||||||
const hitTestFreeDrawElement = (
|
|
||||||
element: ExcalidrawFreeDrawElement,
|
|
||||||
point: Point,
|
|
||||||
threshold: number,
|
|
||||||
): boolean => {
|
|
||||||
// Check point-distance-to-line-segment for every segment in the
|
|
||||||
// element's points (its input points, not its outline points).
|
|
||||||
// This is... okay? It's plenty fast, but the GA library may
|
|
||||||
// have a faster option.
|
|
||||||
|
|
||||||
let x: number;
|
|
||||||
let y: number;
|
|
||||||
|
|
||||||
if (element.angle === 0) {
|
|
||||||
x = point[0] - element.x;
|
|
||||||
y = point[1] - element.y;
|
|
||||||
} else {
|
|
||||||
// Counter-rotate the point around center before testing
|
|
||||||
const [minX, minY, maxX, maxY] = getElementAbsoluteCoords(element);
|
|
||||||
const rotatedPoint = rotatePoint(
|
|
||||||
point,
|
|
||||||
[minX + (maxX - minX) / 2, minY + (maxY - minY) / 2],
|
|
||||||
-element.angle,
|
|
||||||
);
|
|
||||||
x = rotatedPoint[0] - element.x;
|
|
||||||
y = rotatedPoint[1] - element.y;
|
|
||||||
}
|
|
||||||
|
|
||||||
let [A, B] = element.points;
|
|
||||||
let P: readonly [number, number];
|
|
||||||
|
|
||||||
// For freedraw dots
|
|
||||||
if (element.points.length === 2) {
|
|
||||||
return (
|
|
||||||
distance2d(A[0], A[1], x, y) < threshold ||
|
|
||||||
distance2d(B[0], B[1], x, y) < threshold
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// For freedraw lines
|
|
||||||
for (let i = 1; i < element.points.length - 1; i++) {
|
|
||||||
const delta = [B[0] - A[0], B[1] - A[1]];
|
|
||||||
const length = Math.hypot(delta[1], delta[0]);
|
|
||||||
|
|
||||||
const U = [delta[0] / length, delta[1] / length];
|
|
||||||
const C = [x - A[0], y - A[1]];
|
|
||||||
const d = (C[0] * U[0] + C[1] * U[1]) / Math.hypot(U[1], U[0]);
|
|
||||||
P = [A[0] + U[0] * d, A[1] + U[1] * d];
|
|
||||||
|
|
||||||
const da = distance2d(P[0], P[1], A[0], A[1]);
|
|
||||||
const db = distance2d(P[0], P[1], B[0], B[1]);
|
|
||||||
|
|
||||||
P = db < da && da > length ? B : da < db && db > length ? A : P;
|
|
||||||
|
|
||||||
if (Math.hypot(y - P[1], x - P[0]) < threshold) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
A = B;
|
|
||||||
B = element.points[i + 1];
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
};
|
|
||||||
|
|
||||||
const hitTestLinear = (args: HitTestArgs): boolean => {
|
const hitTestLinear = (args: HitTestArgs): boolean => {
|
||||||
const { element, threshold } = args;
|
const { element, threshold } = args;
|
||||||
if (!getShapeForElement(element)) {
|
if (!getShapeForElement(element)) {
|
||||||
|
@@ -58,6 +58,13 @@ export {
|
|||||||
} from "./sizeHelpers";
|
} from "./sizeHelpers";
|
||||||
export { showSelectedShapeActions } from "./showSelectedShapeActions";
|
export { showSelectedShapeActions } from "./showSelectedShapeActions";
|
||||||
|
|
||||||
|
export const getSyncableElements = (
|
||||||
|
elements: readonly ExcalidrawElement[], // There are places in Excalidraw where synthetic invisibly small elements are added and removed.
|
||||||
|
) =>
|
||||||
|
// It's probably best to keep those local otherwise there might be a race condition that
|
||||||
|
// gets the app into an invalid state. I've never seen it happen but I'm worried about it :)
|
||||||
|
elements.filter((el) => el.isDeleted || !isInvisiblySmallElement(el));
|
||||||
|
|
||||||
export const getElementMap = (elements: readonly ExcalidrawElement[]) =>
|
export const getElementMap = (elements: readonly ExcalidrawElement[]) =>
|
||||||
elements.reduce(
|
elements.reduce(
|
||||||
(acc: { [key: string]: ExcalidrawElement }, element: ExcalidrawElement) => {
|
(acc: { [key: string]: ExcalidrawElement }, element: ExcalidrawElement) => {
|
||||||
@@ -70,11 +77,6 @@ export const getElementMap = (elements: readonly ExcalidrawElement[]) =>
|
|||||||
export const getSceneVersion = (elements: readonly ExcalidrawElement[]) =>
|
export const getSceneVersion = (elements: readonly ExcalidrawElement[]) =>
|
||||||
elements.reduce((acc, el) => acc + el.version, 0);
|
elements.reduce((acc, el) => acc + el.version, 0);
|
||||||
|
|
||||||
export const getVisibleElements = (elements: readonly ExcalidrawElement[]) =>
|
|
||||||
elements.filter(
|
|
||||||
(el) => !el.isDeleted && !isInvisiblySmallElement(el),
|
|
||||||
) as readonly NonDeletedExcalidrawElement[];
|
|
||||||
|
|
||||||
export const getNonDeletedElements = (elements: readonly ExcalidrawElement[]) =>
|
export const getNonDeletedElements = (elements: readonly ExcalidrawElement[]) =>
|
||||||
elements.filter(
|
elements.filter(
|
||||||
(element) => !element.isDeleted,
|
(element) => !element.isDeleted,
|
||||||
|
@@ -10,7 +10,7 @@ import { getElementAbsoluteCoords } from ".";
|
|||||||
import { getElementPointsCoords } from "./bounds";
|
import { getElementPointsCoords } from "./bounds";
|
||||||
import { Point, AppState } from "../types";
|
import { Point, AppState } from "../types";
|
||||||
import { mutateElement } from "./mutateElement";
|
import { mutateElement } from "./mutateElement";
|
||||||
import History from "../history";
|
import { SceneHistory } from "../history";
|
||||||
|
|
||||||
import Scene from "../scene/Scene";
|
import Scene from "../scene/Scene";
|
||||||
import {
|
import {
|
||||||
@@ -167,7 +167,7 @@ export class LinearElementEditor {
|
|||||||
event: React.PointerEvent<HTMLCanvasElement>,
|
event: React.PointerEvent<HTMLCanvasElement>,
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
setState: React.Component<any, AppState>["setState"],
|
setState: React.Component<any, AppState>["setState"],
|
||||||
history: History,
|
history: SceneHistory,
|
||||||
scenePointer: { x: number; y: number },
|
scenePointer: { x: number; y: number },
|
||||||
): {
|
): {
|
||||||
didAddPoint: boolean;
|
didAddPoint: boolean;
|
||||||
@@ -415,31 +415,26 @@ export class LinearElementEditor {
|
|||||||
return [rotatedX - element.x, rotatedY - element.y];
|
return [rotatedX - element.x, rotatedY - element.y];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// element-mutating methods
|
||||||
|
// ---------------------------------------------------------------------------
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Normalizes line points so that the start point is at [0,0]. This is
|
* Normalizes line points so that the start point is at [0,0]. This is
|
||||||
* expected in various parts of the codebase. Also returns new x/y to account
|
* expected in various parts of the codebase.
|
||||||
* for the potential normalization.
|
|
||||||
*/
|
*/
|
||||||
static getNormalizedPoints(element: ExcalidrawLinearElement) {
|
static normalizePoints(element: NonDeleted<ExcalidrawLinearElement>) {
|
||||||
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 {
|
mutateElement(element, {
|
||||||
points: points.map((point, _idx) => {
|
points: points.map((point, _idx) => {
|
||||||
return [point[0] - offsetX, point[1] - offsetY] as const;
|
return [point[0] - offsetX, point[1] - offsetY] as const;
|
||||||
}),
|
}),
|
||||||
x: element.x + offsetX,
|
x: element.x + offsetX,
|
||||||
y: element.y + offsetY,
|
y: element.y + offsetY,
|
||||||
};
|
});
|
||||||
}
|
|
||||||
|
|
||||||
// element-mutating methods
|
|
||||||
// ---------------------------------------------------------------------------
|
|
||||||
|
|
||||||
static normalizePoints(element: NonDeleted<ExcalidrawLinearElement>) {
|
|
||||||
mutateElement(element, LinearElementEditor.getNormalizedPoints(element));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static movePointByOffset(
|
static movePointByOffset(
|
||||||
|
@@ -87,41 +87,9 @@ export const mutateElement = <TElement extends Mutable<ExcalidrawElement>>(
|
|||||||
export const newElementWith = <TElement extends ExcalidrawElement>(
|
export const newElementWith = <TElement extends ExcalidrawElement>(
|
||||||
element: TElement,
|
element: TElement,
|
||||||
updates: ElementUpdate<TElement>,
|
updates: ElementUpdate<TElement>,
|
||||||
): TElement => {
|
): TElement => ({
|
||||||
let didChange = false;
|
...element,
|
||||||
for (const key in updates) {
|
...updates,
|
||||||
const value = (updates as any)[key];
|
version: element.version + 1,
|
||||||
if (typeof value !== "undefined") {
|
versionNonce: randomInteger(),
|
||||||
if (
|
});
|
||||||
(element as any)[key] === value &&
|
|
||||||
// if object, always update in case its deep prop was mutated
|
|
||||||
(typeof value !== "object" || value === null || key === "groupIds")
|
|
||||||
) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
didChange = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!didChange) {
|
|
||||||
return element;
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
...element,
|
|
||||||
...updates,
|
|
||||||
version: element.version + 1,
|
|
||||||
versionNonce: randomInteger(),
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Mutates element and updates `version` & `versionNonce`.
|
|
||||||
*
|
|
||||||
* NOTE: does not trigger re-render.
|
|
||||||
*/
|
|
||||||
export const bumpVersion = (element: Mutable<ExcalidrawElement>) => {
|
|
||||||
element.version = element.version + 1;
|
|
||||||
element.versionNonce = randomInteger();
|
|
||||||
return element;
|
|
||||||
};
|
|
||||||
|
@@ -9,7 +9,6 @@ import {
|
|||||||
GroupId,
|
GroupId,
|
||||||
VerticalAlign,
|
VerticalAlign,
|
||||||
Arrowhead,
|
Arrowhead,
|
||||||
ExcalidrawFreeDrawElement,
|
|
||||||
} from "../element/types";
|
} from "../element/types";
|
||||||
import { measureText, getFontString } from "../utils";
|
import { measureText, getFontString } from "../utils";
|
||||||
import { randomInteger, randomId } from "../random";
|
import { randomInteger, randomId } from "../random";
|
||||||
@@ -213,22 +212,6 @@ export const updateTextElement = (
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
export const newFreeDrawElement = (
|
|
||||||
opts: {
|
|
||||||
type: "freedraw";
|
|
||||||
points?: ExcalidrawFreeDrawElement["points"];
|
|
||||||
simulatePressure: boolean;
|
|
||||||
} & ElementConstructorOpts,
|
|
||||||
): NonDeleted<ExcalidrawFreeDrawElement> => {
|
|
||||||
return {
|
|
||||||
..._newElementBase<ExcalidrawFreeDrawElement>(opts.type, opts),
|
|
||||||
points: opts.points || [],
|
|
||||||
pressures: [],
|
|
||||||
simulatePressure: opts.simulatePressure,
|
|
||||||
lastCommittedPoint: null,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export const newLinearElement = (
|
export const newLinearElement = (
|
||||||
opts: {
|
opts: {
|
||||||
type: ExcalidrawLinearElement["type"];
|
type: ExcalidrawLinearElement["type"];
|
||||||
|
@@ -18,11 +18,7 @@ import {
|
|||||||
getCommonBounds,
|
getCommonBounds,
|
||||||
getResizedElementAbsoluteCoords,
|
getResizedElementAbsoluteCoords,
|
||||||
} from "./bounds";
|
} from "./bounds";
|
||||||
import {
|
import { isLinearElement, isTextElement } from "./typeChecks";
|
||||||
isFreeDrawElement,
|
|
||||||
isLinearElement,
|
|
||||||
isTextElement,
|
|
||||||
} from "./typeChecks";
|
|
||||||
import { mutateElement } from "./mutateElement";
|
import { mutateElement } from "./mutateElement";
|
||||||
import { getPerfectElementSize } from "./sizeHelpers";
|
import { getPerfectElementSize } from "./sizeHelpers";
|
||||||
import { measureText, getFontString } from "../utils";
|
import { measureText, getFontString } from "../utils";
|
||||||
@@ -35,7 +31,7 @@ import {
|
|||||||
import { PointerDownState } from "../components/App";
|
import { PointerDownState } from "../components/App";
|
||||||
import { Point } from "../types";
|
import { Point } from "../types";
|
||||||
|
|
||||||
export const normalizeAngle = (angle: number): number => {
|
const normalizeAngle = (angle: number): number => {
|
||||||
if (angle >= 2 * Math.PI) {
|
if (angle >= 2 * Math.PI) {
|
||||||
return angle - 2 * Math.PI;
|
return angle - 2 * Math.PI;
|
||||||
}
|
}
|
||||||
@@ -185,7 +181,7 @@ const getPerfectElementSizeWithRotation = (
|
|||||||
return rotate(size.width, size.height, 0, 0, -angle);
|
return rotate(size.width, size.height, 0, 0, -angle);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const reshapeSingleTwoPointElement = (
|
const reshapeSingleTwoPointElement = (
|
||||||
element: NonDeleted<ExcalidrawLinearElement>,
|
element: NonDeleted<ExcalidrawLinearElement>,
|
||||||
resizeArrowDirection: "origin" | "end",
|
resizeArrowDirection: "origin" | "end",
|
||||||
isRotateWithDiscreteAngle: boolean,
|
isRotateWithDiscreteAngle: boolean,
|
||||||
@@ -248,7 +244,7 @@ const rescalePointsInElement = (
|
|||||||
width: number,
|
width: number,
|
||||||
height: number,
|
height: number,
|
||||||
) =>
|
) =>
|
||||||
isLinearElement(element) || isFreeDrawElement(element)
|
isLinearElement(element)
|
||||||
? {
|
? {
|
||||||
points: rescalePoints(
|
points: rescalePoints(
|
||||||
0,
|
0,
|
||||||
@@ -382,7 +378,7 @@ const resizeSingleTextElement = (
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
export const resizeSingleElement = (
|
const resizeSingleElement = (
|
||||||
stateAtResizeStart: NonDeletedExcalidrawElement,
|
stateAtResizeStart: NonDeletedExcalidrawElement,
|
||||||
shouldKeepSidesRatio: boolean,
|
shouldKeepSidesRatio: boolean,
|
||||||
element: NonDeletedExcalidrawElement,
|
element: NonDeletedExcalidrawElement,
|
||||||
@@ -408,7 +404,7 @@ export const resizeSingleElement = (
|
|||||||
-stateAtResizeStart.angle,
|
-stateAtResizeStart.angle,
|
||||||
);
|
);
|
||||||
|
|
||||||
// Get bounds corners rendered on screen
|
//Get bounds corners rendered on screen
|
||||||
const [esx1, esy1, esx2, esy2] = getResizedElementAbsoluteCoords(
|
const [esx1, esy1, esx2, esy2] = getResizedElementAbsoluteCoords(
|
||||||
element,
|
element,
|
||||||
element.width,
|
element.width,
|
||||||
@@ -648,14 +644,11 @@ const resizeMultipleElements = (
|
|||||||
font = { fontSize: nextFont.size, baseline: nextFont.baseline };
|
font = { fontSize: nextFont.size, baseline: nextFont.baseline };
|
||||||
}
|
}
|
||||||
const origCoords = getElementAbsoluteCoords(element);
|
const origCoords = getElementAbsoluteCoords(element);
|
||||||
|
|
||||||
const rescaledPoints = rescalePointsInElement(element, width, height);
|
const rescaledPoints = rescalePointsInElement(element, width, height);
|
||||||
|
|
||||||
updateBoundElements(element, {
|
updateBoundElements(element, {
|
||||||
newSize: { width, height },
|
newSize: { width, height },
|
||||||
simultaneouslyUpdated: elements,
|
simultaneouslyUpdated: elements,
|
||||||
});
|
});
|
||||||
|
|
||||||
const finalCoords = getResizedElementAbsoluteCoords(
|
const finalCoords = getResizedElementAbsoluteCoords(
|
||||||
{
|
{
|
||||||
...element,
|
...element,
|
||||||
@@ -664,7 +657,6 @@ const resizeMultipleElements = (
|
|||||||
width,
|
width,
|
||||||
height,
|
height,
|
||||||
);
|
);
|
||||||
|
|
||||||
const { x, y } = getNextXY(element, origCoords, finalCoords);
|
const { x, y } = getNextXY(element, origCoords, finalCoords);
|
||||||
return [...prev, { width, height, x, y, ...rescaledPoints, ...font }];
|
return [...prev, { width, height, x, y, ...rescaledPoints, ...font }];
|
||||||
},
|
},
|
||||||
|
@@ -1,12 +1,12 @@
|
|||||||
import { ExcalidrawElement } from "./types";
|
import { ExcalidrawElement } from "./types";
|
||||||
import { mutateElement } from "./mutateElement";
|
import { mutateElement } from "./mutateElement";
|
||||||
import { isFreeDrawElement, isLinearElement } from "./typeChecks";
|
import { isLinearElement } from "./typeChecks";
|
||||||
import { SHIFT_LOCKING_ANGLE } from "../constants";
|
import { SHIFT_LOCKING_ANGLE } from "../constants";
|
||||||
|
|
||||||
export const isInvisiblySmallElement = (
|
export const isInvisiblySmallElement = (
|
||||||
element: ExcalidrawElement,
|
element: ExcalidrawElement,
|
||||||
): boolean => {
|
): boolean => {
|
||||||
if (isLinearElement(element) || isFreeDrawElement(element)) {
|
if (isLinearElement(element)) {
|
||||||
return element.points.length < 2;
|
return element.points.length < 2;
|
||||||
}
|
}
|
||||||
return element.width === 0 && element.height === 0;
|
return element.width === 0 && element.height === 0;
|
||||||
@@ -26,7 +26,7 @@ export const getPerfectElementSize = (
|
|||||||
if (
|
if (
|
||||||
elementType === "line" ||
|
elementType === "line" ||
|
||||||
elementType === "arrow" ||
|
elementType === "arrow" ||
|
||||||
elementType === "freedraw"
|
elementType === "draw"
|
||||||
) {
|
) {
|
||||||
const lockedAngle =
|
const lockedAngle =
|
||||||
Math.round(Math.atan(absHeight / absWidth) / SHIFT_LOCKING_ANGLE) *
|
Math.round(Math.atan(absHeight / absWidth) / SHIFT_LOCKING_ANGLE) *
|
||||||
|
@@ -1,169 +0,0 @@
|
|||||||
import ReactDOM from "react-dom";
|
|
||||||
import ExcalidrawApp from "../excalidraw-app";
|
|
||||||
import { render } from "../tests/test-utils";
|
|
||||||
import { Pointer, UI } from "../tests/helpers/ui";
|
|
||||||
import { KEYS } from "../keys";
|
|
||||||
|
|
||||||
// Unmount ReactDOM from root
|
|
||||||
ReactDOM.unmountComponentAtNode(document.getElementById("root")!);
|
|
||||||
|
|
||||||
const tab = " ";
|
|
||||||
|
|
||||||
describe("textWysiwyg", () => {
|
|
||||||
let textarea: HTMLTextAreaElement;
|
|
||||||
beforeEach(async () => {
|
|
||||||
await render(<ExcalidrawApp />);
|
|
||||||
|
|
||||||
const element = UI.createElement("text");
|
|
||||||
|
|
||||||
new Pointer("mouse").clickOn(element);
|
|
||||||
textarea = document.querySelector(
|
|
||||||
".excalidraw-textEditorContainer > textarea",
|
|
||||||
)!;
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should add a tab at the start of the first line", () => {
|
|
||||||
const event = new KeyboardEvent("keydown", { key: KEYS.TAB });
|
|
||||||
textarea.value = "Line#1\nLine#2";
|
|
||||||
// cursor: "|Line#1\nLine#2"
|
|
||||||
textarea.selectionStart = 0;
|
|
||||||
textarea.selectionEnd = 0;
|
|
||||||
textarea.dispatchEvent(event);
|
|
||||||
|
|
||||||
expect(textarea.value).toEqual(`${tab}Line#1\nLine#2`);
|
|
||||||
// cursor: " |Line#1\nLine#2"
|
|
||||||
expect(textarea.selectionStart).toEqual(4);
|
|
||||||
expect(textarea.selectionEnd).toEqual(4);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should add a tab at the start of the second line", () => {
|
|
||||||
const event = new KeyboardEvent("keydown", { key: KEYS.TAB });
|
|
||||||
textarea.value = "Line#1\nLine#2";
|
|
||||||
// cursor: "Line#1\nLin|e#2"
|
|
||||||
textarea.selectionStart = 10;
|
|
||||||
textarea.selectionEnd = 10;
|
|
||||||
|
|
||||||
textarea.dispatchEvent(event);
|
|
||||||
|
|
||||||
expect(textarea.value).toEqual(`Line#1\n${tab}Line#2`);
|
|
||||||
|
|
||||||
// cursor: "Line#1\n Lin|e#2"
|
|
||||||
expect(textarea.selectionStart).toEqual(14);
|
|
||||||
expect(textarea.selectionEnd).toEqual(14);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should add a tab at the start of the first and second line", () => {
|
|
||||||
const event = new KeyboardEvent("keydown", { key: KEYS.TAB });
|
|
||||||
textarea.value = "Line#1\nLine#2\nLine#3";
|
|
||||||
// cursor: "Li|ne#1\nLi|ne#2\nLine#3"
|
|
||||||
textarea.selectionStart = 2;
|
|
||||||
textarea.selectionEnd = 9;
|
|
||||||
|
|
||||||
textarea.dispatchEvent(event);
|
|
||||||
|
|
||||||
expect(textarea.value).toEqual(`${tab}Line#1\n${tab}Line#2\nLine#3`);
|
|
||||||
|
|
||||||
// cursor: " Li|ne#1\n Li|ne#2\nLine#3"
|
|
||||||
expect(textarea.selectionStart).toEqual(6);
|
|
||||||
expect(textarea.selectionEnd).toEqual(17);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should remove a tab at the start of the first line", () => {
|
|
||||||
const event = new KeyboardEvent("keydown", {
|
|
||||||
key: KEYS.TAB,
|
|
||||||
shiftKey: true,
|
|
||||||
});
|
|
||||||
textarea.value = `${tab}Line#1\nLine#2`;
|
|
||||||
// cursor: "| Line#1\nLine#2"
|
|
||||||
textarea.selectionStart = 0;
|
|
||||||
textarea.selectionEnd = 0;
|
|
||||||
|
|
||||||
textarea.dispatchEvent(event);
|
|
||||||
|
|
||||||
expect(textarea.value).toEqual(`Line#1\nLine#2`);
|
|
||||||
|
|
||||||
// cursor: "|Line#1\nLine#2"
|
|
||||||
expect(textarea.selectionStart).toEqual(0);
|
|
||||||
expect(textarea.selectionEnd).toEqual(0);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should remove a tab at the start of the second line", () => {
|
|
||||||
const event = new KeyboardEvent("keydown", {
|
|
||||||
key: KEYS.TAB,
|
|
||||||
shiftKey: true,
|
|
||||||
});
|
|
||||||
// cursor: "Line#1\n Lin|e#2"
|
|
||||||
textarea.value = `Line#1\n${tab}Line#2`;
|
|
||||||
textarea.selectionStart = 15;
|
|
||||||
textarea.selectionEnd = 15;
|
|
||||||
|
|
||||||
textarea.dispatchEvent(event);
|
|
||||||
|
|
||||||
expect(textarea.value).toEqual(`Line#1\nLine#2`);
|
|
||||||
// cursor: "Line#1\nLin|e#2"
|
|
||||||
expect(textarea.selectionStart).toEqual(11);
|
|
||||||
expect(textarea.selectionEnd).toEqual(11);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should remove a tab at the start of the first and second line", () => {
|
|
||||||
const event = new KeyboardEvent("keydown", {
|
|
||||||
key: KEYS.TAB,
|
|
||||||
shiftKey: true,
|
|
||||||
});
|
|
||||||
// cursor: " Li|ne#1\n Li|ne#2\nLine#3"
|
|
||||||
textarea.value = `${tab}Line#1\n${tab}Line#2\nLine#3`;
|
|
||||||
textarea.selectionStart = 6;
|
|
||||||
textarea.selectionEnd = 17;
|
|
||||||
|
|
||||||
textarea.dispatchEvent(event);
|
|
||||||
|
|
||||||
expect(textarea.value).toEqual(`Line#1\nLine#2\nLine#3`);
|
|
||||||
// cursor: "Li|ne#1\nLi|ne#2\nLine#3"
|
|
||||||
expect(textarea.selectionStart).toEqual(2);
|
|
||||||
expect(textarea.selectionEnd).toEqual(9);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should remove a tab at the start of the second line and cursor stay on this line", () => {
|
|
||||||
const event = new KeyboardEvent("keydown", {
|
|
||||||
key: KEYS.TAB,
|
|
||||||
shiftKey: true,
|
|
||||||
});
|
|
||||||
// cursor: "Line#1\n | Line#2"
|
|
||||||
textarea.value = `Line#1\n${tab}Line#2`;
|
|
||||||
textarea.selectionStart = 9;
|
|
||||||
textarea.selectionEnd = 9;
|
|
||||||
textarea.dispatchEvent(event);
|
|
||||||
|
|
||||||
// cursor: "Line#1\n|Line#2"
|
|
||||||
expect(textarea.selectionStart).toEqual(7);
|
|
||||||
// expect(textarea.selectionEnd).toEqual(7);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should remove partial tabs", () => {
|
|
||||||
const event = new KeyboardEvent("keydown", {
|
|
||||||
key: KEYS.TAB,
|
|
||||||
shiftKey: true,
|
|
||||||
});
|
|
||||||
// cursor: "Line#1\n Line#|2"
|
|
||||||
textarea.value = `Line#1\n Line#2`;
|
|
||||||
textarea.selectionStart = 15;
|
|
||||||
textarea.selectionEnd = 15;
|
|
||||||
textarea.dispatchEvent(event);
|
|
||||||
|
|
||||||
expect(textarea.value).toEqual(`Line#1\nLine#2`);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should remove nothing", () => {
|
|
||||||
const event = new KeyboardEvent("keydown", {
|
|
||||||
key: KEYS.TAB,
|
|
||||||
shiftKey: true,
|
|
||||||
});
|
|
||||||
// cursor: "Line#1\n Li|ne#2"
|
|
||||||
textarea.value = `Line#1\nLine#2`;
|
|
||||||
textarea.selectionStart = 9;
|
|
||||||
textarea.selectionEnd = 9;
|
|
||||||
textarea.dispatchEvent(event);
|
|
||||||
|
|
||||||
expect(textarea.value).toEqual(`Line#1\nLine#2`);
|
|
||||||
});
|
|
||||||
});
|
|
@@ -1,4 +1,4 @@
|
|||||||
import { CODES, KEYS } from "../keys";
|
import { KEYS } from "../keys";
|
||||||
import { isWritableElement, getFontString } from "../utils";
|
import { isWritableElement, getFontString } from "../utils";
|
||||||
import Scene from "../scene/Scene";
|
import Scene from "../scene/Scene";
|
||||||
import { isTextElement } from "./typeChecks";
|
import { isTextElement } from "./typeChecks";
|
||||||
@@ -43,7 +43,6 @@ export const textWysiwyg = ({
|
|||||||
getViewportCoords,
|
getViewportCoords,
|
||||||
element,
|
element,
|
||||||
canvas,
|
canvas,
|
||||||
excalidrawContainer,
|
|
||||||
}: {
|
}: {
|
||||||
id: ExcalidrawElement["id"];
|
id: ExcalidrawElement["id"];
|
||||||
appState: AppState;
|
appState: AppState;
|
||||||
@@ -52,7 +51,6 @@ export const textWysiwyg = ({
|
|||||||
getViewportCoords: (x: number, y: number) => [number, number];
|
getViewportCoords: (x: number, y: number) => [number, number];
|
||||||
element: ExcalidrawElement;
|
element: ExcalidrawElement;
|
||||||
canvas: HTMLCanvasElement | null;
|
canvas: HTMLCanvasElement | null;
|
||||||
excalidrawContainer: HTMLDivElement | null;
|
|
||||||
}) => {
|
}) => {
|
||||||
const updateWysiwygStyle = () => {
|
const updateWysiwygStyle = () => {
|
||||||
const updatedElement = Scene.getScene(element)?.getElement(id);
|
const updatedElement = Scene.getScene(element)?.getElement(id);
|
||||||
@@ -73,7 +71,7 @@ export const textWysiwyg = ({
|
|||||||
// margin-right of parent if any
|
// margin-right of parent if any
|
||||||
Number(
|
Number(
|
||||||
getComputedStyle(
|
getComputedStyle(
|
||||||
excalidrawContainer?.parentNode as Element,
|
document.querySelector(".excalidraw")!.parentNode as Element,
|
||||||
).marginRight.slice(0, -2),
|
).marginRight.slice(0, -2),
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -136,7 +134,6 @@ export const textWysiwyg = ({
|
|||||||
}
|
}
|
||||||
|
|
||||||
editable.onkeydown = (event) => {
|
editable.onkeydown = (event) => {
|
||||||
event.stopPropagation();
|
|
||||||
if (event.key === KEYS.ESCAPE) {
|
if (event.key === KEYS.ESCAPE) {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
submittedViaKeyboard = true;
|
submittedViaKeyboard = true;
|
||||||
@@ -148,118 +145,11 @@ export const textWysiwyg = ({
|
|||||||
}
|
}
|
||||||
submittedViaKeyboard = true;
|
submittedViaKeyboard = true;
|
||||||
handleSubmit();
|
handleSubmit();
|
||||||
} else if (
|
} else if (event.key === KEYS.ENTER && !event.altKey) {
|
||||||
event.key === KEYS.TAB ||
|
event.stopPropagation();
|
||||||
(event[KEYS.CTRL_OR_CMD] &&
|
|
||||||
(event.code === CODES.BRACKET_LEFT ||
|
|
||||||
event.code === CODES.BRACKET_RIGHT))
|
|
||||||
) {
|
|
||||||
event.preventDefault();
|
|
||||||
if (event.shiftKey || event.code === CODES.BRACKET_LEFT) {
|
|
||||||
outdent();
|
|
||||||
} else {
|
|
||||||
indent();
|
|
||||||
}
|
|
||||||
// We must send an input event to resize the element
|
|
||||||
editable.dispatchEvent(new Event("input"));
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const TAB_SIZE = 4;
|
|
||||||
const TAB = " ".repeat(TAB_SIZE);
|
|
||||||
const RE_LEADING_TAB = new RegExp(`^ {1,${TAB_SIZE}}`);
|
|
||||||
const indent = () => {
|
|
||||||
const { selectionStart, selectionEnd } = editable;
|
|
||||||
const linesStartIndices = getSelectedLinesStartIndices();
|
|
||||||
|
|
||||||
let value = editable.value;
|
|
||||||
linesStartIndices.forEach((startIndex) => {
|
|
||||||
const startValue = value.slice(0, startIndex);
|
|
||||||
const endValue = value.slice(startIndex);
|
|
||||||
|
|
||||||
value = `${startValue}${TAB}${endValue}`;
|
|
||||||
});
|
|
||||||
|
|
||||||
editable.value = value;
|
|
||||||
|
|
||||||
editable.selectionStart = selectionStart + TAB_SIZE;
|
|
||||||
editable.selectionEnd = selectionEnd + TAB_SIZE * linesStartIndices.length;
|
|
||||||
};
|
|
||||||
|
|
||||||
const outdent = () => {
|
|
||||||
const { selectionStart, selectionEnd } = editable;
|
|
||||||
const linesStartIndices = getSelectedLinesStartIndices();
|
|
||||||
const removedTabs: number[] = [];
|
|
||||||
|
|
||||||
let value = editable.value;
|
|
||||||
linesStartIndices.forEach((startIndex) => {
|
|
||||||
const tabMatch = value
|
|
||||||
.slice(startIndex, startIndex + TAB_SIZE)
|
|
||||||
.match(RE_LEADING_TAB);
|
|
||||||
|
|
||||||
if (tabMatch) {
|
|
||||||
const startValue = value.slice(0, startIndex);
|
|
||||||
const endValue = value.slice(startIndex + tabMatch[0].length);
|
|
||||||
|
|
||||||
// Delete a tab from the line
|
|
||||||
value = `${startValue}${endValue}`;
|
|
||||||
removedTabs.push(startIndex);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
editable.value = value;
|
|
||||||
|
|
||||||
if (removedTabs.length) {
|
|
||||||
if (selectionStart > removedTabs[removedTabs.length - 1]) {
|
|
||||||
editable.selectionStart = Math.max(
|
|
||||||
selectionStart - TAB_SIZE,
|
|
||||||
removedTabs[removedTabs.length - 1],
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
// If the cursor is before the first tab removed, ex:
|
|
||||||
// Line| #1
|
|
||||||
// Line #2
|
|
||||||
// Lin|e #3
|
|
||||||
// we should reset the selectionStart to his initial value.
|
|
||||||
editable.selectionStart = selectionStart;
|
|
||||||
}
|
|
||||||
editable.selectionEnd = Math.max(
|
|
||||||
editable.selectionStart,
|
|
||||||
selectionEnd - TAB_SIZE * removedTabs.length,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @returns indeces of start positions of selected lines, in reverse order
|
|
||||||
*/
|
|
||||||
const getSelectedLinesStartIndices = () => {
|
|
||||||
let { selectionStart, selectionEnd, value } = editable;
|
|
||||||
|
|
||||||
// chars before selectionStart on the same line
|
|
||||||
const startOffset = value.slice(0, selectionStart).match(/[^\n]*$/)![0]
|
|
||||||
.length;
|
|
||||||
// put caret at the start of the line
|
|
||||||
selectionStart = selectionStart - startOffset;
|
|
||||||
|
|
||||||
const selected = value.slice(selectionStart, selectionEnd);
|
|
||||||
|
|
||||||
return selected
|
|
||||||
.split("\n")
|
|
||||||
.reduce(
|
|
||||||
(startIndices, line, idx, lines) =>
|
|
||||||
startIndices.concat(
|
|
||||||
idx
|
|
||||||
? // curr line index is prev line's start + prev line's length + \n
|
|
||||||
startIndices[idx - 1] + lines[idx - 1].length + 1
|
|
||||||
: // first selected line
|
|
||||||
selectionStart,
|
|
||||||
),
|
|
||||||
[] as number[],
|
|
||||||
)
|
|
||||||
.reverse();
|
|
||||||
};
|
|
||||||
|
|
||||||
const stopEvent = (event: Event) => {
|
const stopEvent = (event: Event) => {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
@@ -269,14 +159,11 @@ export const textWysiwyg = ({
|
|||||||
// so that we don't need to create separate a callback for event handlers
|
// so that we don't need to create separate a callback for event handlers
|
||||||
let submittedViaKeyboard = false;
|
let submittedViaKeyboard = false;
|
||||||
const handleSubmit = () => {
|
const handleSubmit = () => {
|
||||||
// cleanup must be run before onSubmit otherwise when app blurs the wysiwyg
|
|
||||||
// it'd get stuck in an infinite loop of blur→onSubmit after we re-focus the
|
|
||||||
// wysiwyg on update
|
|
||||||
cleanup();
|
|
||||||
onSubmit({
|
onSubmit({
|
||||||
text: normalizeText(editable.value),
|
text: normalizeText(editable.value),
|
||||||
viaKeyboard: submittedViaKeyboard,
|
viaKeyboard: submittedViaKeyboard,
|
||||||
});
|
});
|
||||||
|
cleanup();
|
||||||
};
|
};
|
||||||
|
|
||||||
const cleanup = () => {
|
const cleanup = () => {
|
||||||
@@ -320,8 +207,7 @@ export const textWysiwyg = ({
|
|||||||
// prevent blur when changing properties from the menu
|
// prevent blur when changing properties from the menu
|
||||||
const onPointerDown = (event: MouseEvent) => {
|
const onPointerDown = (event: MouseEvent) => {
|
||||||
if (
|
if (
|
||||||
(event.target instanceof HTMLElement ||
|
event.target instanceof HTMLElement &&
|
||||||
event.target instanceof SVGElement) &&
|
|
||||||
event.target.closest(`.${CLASSES.SHAPE_ACTIONS_MENU}`) &&
|
event.target.closest(`.${CLASSES.SHAPE_ACTIONS_MENU}`) &&
|
||||||
!isWritableElement(event.target)
|
!isWritableElement(event.target)
|
||||||
) {
|
) {
|
||||||
@@ -365,7 +251,7 @@ export const textWysiwyg = ({
|
|||||||
passive: false,
|
passive: false,
|
||||||
capture: true,
|
capture: true,
|
||||||
});
|
});
|
||||||
excalidrawContainer
|
document
|
||||||
?.querySelector(".excalidraw-textEditorContainer")!
|
.querySelector(".excalidraw-textEditorContainer")!
|
||||||
.appendChild(editable);
|
.appendChild(editable);
|
||||||
};
|
};
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user