mirror of
				https://github.com/excalidraw/excalidraw.git
				synced 2025-10-26 16:34:22 +01:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			export-deb
			...
			close_canv
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | c96b0404ba | 
							
								
								
									
										5
									
								
								.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.env
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| REACT_APP_BACKEND_V1_GET_URL=https://json.excalidraw.com/api/v1/ | ||||
| REACT_APP_BACKEND_V2_GET_URL=https://json.excalidraw.com/api/v2/ | ||||
| REACT_APP_BACKEND_V2_POST_URL=https://json.excalidraw.com/api/v2/post/ | ||||
| REACT_APP_SOCKET_SERVER_URL=https://portal.excalidraw.com | ||||
| REACT_APP_FIREBASE_CONFIG='{"apiKey":"AIzaSyAd15pYlMci_xIp9ko6wkEsDzAAA0Dn0RU","authDomain":"excalidraw-room-persistence.firebaseapp.com","databaseURL":"https://excalidraw-room-persistence.firebaseio.com","projectId":"excalidraw-room-persistence","storageBucket":"excalidraw-room-persistence.appspot.com","messagingSenderId":"654800341332","appId":"1:654800341332:web:4a692de832b55bd57ce0c1"}' | ||||
| @@ -1,13 +0,0 @@ | ||||
| REACT_APP_BACKEND_V2_GET_URL=https://json-dev.excalidraw.com/api/v2/ | ||||
| REACT_APP_BACKEND_V2_POST_URL=https://json-dev.excalidraw.com/api/v2/post/ | ||||
|  | ||||
| REACT_APP_LIBRARY_URL=https://libraries.excalidraw.com | ||||
| REACT_APP_LIBRARY_BACKEND=https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries | ||||
|  | ||||
| # collaboration WebSocket server (https://github.com/excalidraw/excalidraw-room) | ||||
| REACT_APP_WS_SERVER_URL=http://localhost:3002 | ||||
|  | ||||
| # set this only if using the collaboration workflow we use on excalidraw.com | ||||
| REACT_APP_PORTAL_URL= | ||||
|  | ||||
| REACT_APP_FIREBASE_CONFIG='{"apiKey":"AIzaSyCMkxA60XIW8KbqMYL7edC4qT5l4qHX2h8","authDomain":"excalidraw-oss-dev.firebaseapp.com","projectId":"excalidraw-oss-dev","storageBucket":"excalidraw-oss-dev.appspot.com","messagingSenderId":"664559512677","appId":"1:664559512677:web:a385181f2928d328a7aa8c"}' | ||||
| @@ -1,15 +1 @@ | ||||
| REACT_APP_BACKEND_V2_GET_URL=https://json.excalidraw.com/api/v2/ | ||||
| REACT_APP_BACKEND_V2_POST_URL=https://json.excalidraw.com/api/v2/post/ | ||||
|  | ||||
| REACT_APP_LIBRARY_URL=https://libraries.excalidraw.com | ||||
| REACT_APP_LIBRARY_BACKEND=https://us-central1-excalidraw-room-persistence.cloudfunctions.net/libraries | ||||
|  | ||||
| REACT_APP_PORTAL_URL=https://portal.excalidraw.com | ||||
| # Fill to set socket server URL used for collaboration. | ||||
| # Meant for forks only: excalidraw.com uses custom REACT_APP_PORTAL_URL flow | ||||
| REACT_APP_WS_SERVER_URL= | ||||
|  | ||||
| REACT_APP_FIREBASE_CONFIG='{"apiKey":"AIzaSyAd15pYlMci_xIp9ko6wkEsDzAAA0Dn0RU","authDomain":"excalidraw-room-persistence.firebaseapp.com","databaseURL":"https://excalidraw-room-persistence.firebaseio.com","projectId":"excalidraw-room-persistence","storageBucket":"excalidraw-room-persistence.appspot.com","messagingSenderId":"654800341332","appId":"1:654800341332:web:4a692de832b55bd57ce0c1"}' | ||||
|  | ||||
| # production-only vars | ||||
| REACT_APP_GOOGLE_ANALYTICS_ID=UA-387204-13 | ||||
|   | ||||
| @@ -5,4 +5,3 @@ package-lock.json | ||||
| firebase/ | ||||
| dist/ | ||||
| public/workbox | ||||
| src/packages/excalidraw/types | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| { | ||||
|   "extends": ["@excalidraw/eslint-config", "react-app"], | ||||
|   "rules": { | ||||
|     "import/no-anonymous-default-export": "off", | ||||
|     "no-restricted-globals": "off" | ||||
|     "import/no-anonymous-default-export": "off" | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										3
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,7 +10,6 @@ updates: | ||||
|       - lipis | ||||
|     assignees: | ||||
|       - lipis | ||||
|     open-pull-requests-limit: 20 | ||||
|  | ||||
|   - package-ecosystem: npm | ||||
|     directory: /src/packages/excalidraw/ | ||||
| @@ -22,7 +21,6 @@ updates: | ||||
|       - ad1992 | ||||
|     assignees: | ||||
|       - ad1992 | ||||
|     open-pull-requests-limit: 20 | ||||
|  | ||||
|   - package-ecosystem: npm | ||||
|     directory: /src/packages/utils/ | ||||
| @@ -34,4 +32,3 @@ updates: | ||||
|       - ad1992 | ||||
|     assignees: | ||||
|       - ad1992 | ||||
|     open-pull-requests-limit: 20 | ||||
|   | ||||
							
								
								
									
										1
									
								
								.github/workflows/autorelease-excalidraw.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/autorelease-excalidraw.yml
									
									
									
									
										vendored
									
									
								
							| @@ -23,5 +23,4 @@ jobs: | ||||
|           NPM_TOKEN: ${{ secrets.NPM_TOKEN }} | ||||
|       - name: Auto release | ||||
|         run: | | ||||
|           yarn add @actions/core | ||||
|           yarn autorelease | ||||
|   | ||||
							
								
								
									
										55
									
								
								.github/workflows/autorelease-preview.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										55
									
								
								.github/workflows/autorelease-preview.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,55 +0,0 @@ | ||||
| name: Auto release preview @excalidraw/excalidraw-preview | ||||
| on: | ||||
|   issue_comment: | ||||
|     types: [created, edited] | ||||
|  | ||||
| jobs: | ||||
|   Auto-release-excalidraw-preview: | ||||
|     name: Auto release preview | ||||
|     if: github.event.comment.body == '@excalibot release package' && github.event.issue.pull_request | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: React to release comment | ||||
|         uses: peter-evans/create-or-update-comment@v1 | ||||
|         with: | ||||
|           token: ${{ secrets.PUSH_TRANSLATIONS_COVERAGE_PAT }} | ||||
|           comment-id: ${{ github.event.comment.id }} | ||||
|           reactions: "+1" | ||||
|       - name: Get PR SHA | ||||
|         id: sha | ||||
|         uses: actions/github-script@v4 | ||||
|         with: | ||||
|           result-encoding: string | ||||
|           script: | | ||||
|             const { owner, repo, number } = context.issue; | ||||
|             const pr = await github.pulls.get({ | ||||
|               owner, | ||||
|               repo, | ||||
|               pull_number: number, | ||||
|             }); | ||||
|             return pr.data.head.sha | ||||
|       - uses: actions/checkout@v2 | ||||
|         with: | ||||
|           ref: ${{ steps.sha.outputs.result }} | ||||
|           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 preview | ||||
|         id: "autorelease" | ||||
|         run: | | ||||
|           yarn add @actions/core | ||||
|           yarn autorelease preview ${{ github.event.issue.number }} | ||||
|       - name: Post comment post release | ||||
|         if: always() | ||||
|         uses: peter-evans/create-or-update-comment@v1 | ||||
|         with: | ||||
|           token: ${{ secrets.PUSH_TRANSLATIONS_COVERAGE_PAT }} | ||||
|           issue-number: ${{ github.event.issue.number }} | ||||
|           body: "@${{ github.event.comment.user.login }} ${{ steps.autorelease.outputs.result }}" | ||||
							
								
								
									
										29
									
								
								.github/workflows/build-packages.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								.github/workflows/build-packages.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| name: Build packages | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - master | ||||
|   pull_request: | ||||
|  | ||||
| jobs: | ||||
|   packages: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v2 | ||||
|       - name: Setup Node.js 14.x | ||||
|         uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 14.x | ||||
|       - name: Install dependencies | ||||
|         run: | | ||||
|           yarn --frozen-lockfile | ||||
|           yarn --cwd src/packages/excalidraw | ||||
|           yarn --cwd src/packages/utils | ||||
|       - name: Build @excalidraw/excalidraw | ||||
|         run: | | ||||
|           yarn --cwd src/packages/excalidraw run pack | ||||
|       - name: Build @excalidraw/utils | ||||
|         run: | | ||||
|           yarn --cwd src/packages/utils run pack | ||||
							
								
								
									
										2
									
								
								.github/workflows/publish-docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/publish-docker.yml
									
									
									
									
										vendored
									
									
								
							| @@ -11,7 +11,7 @@ jobs: | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v2 | ||||
|       - uses: docker/build-push-action@v2 | ||||
|       - uses: docker/build-push-action@v1 | ||||
|         with: | ||||
|           username: ${{ secrets.DOCKER_USERNAME }} | ||||
|           password: ${{ secrets.DOCKER_PASSWORD }} | ||||
|   | ||||
							
								
								
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -5,11 +5,9 @@ | ||||
| .env.test.local | ||||
| .envrc | ||||
| .eslintcache | ||||
| .history | ||||
| .idea | ||||
| .vercel | ||||
| .vscode | ||||
| .yarn | ||||
| *.log | ||||
| *.tgz | ||||
| build | ||||
| @@ -23,7 +21,3 @@ static | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
| src/packages/excalidraw/types | ||||
| src/packages/excalidraw/example/public/bundle.js | ||||
| src/packages/excalidraw/example/public/excalidraw-assets-dev | ||||
| src/packages/excalidraw/example/public/excalidraw.development.js | ||||
|  | ||||
|   | ||||
| @@ -1,2 +0,0 @@ | ||||
| #!/bin/sh | ||||
| yarn lint-staged | ||||
| @@ -10,7 +10,7 @@ ARG NODE_ENV=production | ||||
| COPY . . | ||||
| RUN yarn build:app:docker | ||||
|  | ||||
| FROM nginx:1.21-alpine | ||||
| FROM nginx:1.17-alpine | ||||
|  | ||||
| COPY --from=build /opt/node_app/build /usr/share/nginx/html | ||||
|  | ||||
|   | ||||
							
								
								
									
										55
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										55
									
								
								README.md
									
									
									
									
									
								
							| @@ -32,10 +32,6 @@ Last but not least, we're thankful to these companies for offering their service | ||||
|  | ||||
| [](https://vercel.com) [](https://sentry.io) [](https://crowdin.com) | ||||
|  | ||||
| ## Who's integrating Excalidraw | ||||
|  | ||||
| [Google Cloud](https://googlecloudcheatsheet.withgoogle.com/architecture) • [Meta](https://meta.com/) • [CodeSandbox](https://codesandbox.io/) • [Obsidian Excalidraw](https://github.com/zsviczian/obsidian-excalidraw-plugin) • [Replit](https://replit.com/) • [Slite](https://slite.com/) • [Notion](https://notion.so/) • [HackerRank](https://www.hackerrank.com/) • | ||||
|  | ||||
| ## Documentation | ||||
|  | ||||
| ### Shortcuts | ||||
| @@ -74,8 +70,6 @@ The first set of digits is the room. This is visible from the server that’s go | ||||
|  | ||||
| The second set of digits is the encryption key. The Excalidraw server doesn’t know about it. This is what all the participants use to encrypt/decrypt the messages. | ||||
|  | ||||
| > Note: Please ensure that the encryption key is 22 characters long. | ||||
|  | ||||
| ## Shape libraries | ||||
|  | ||||
| Find a growing list of libraries containing assets for your drawings at [libraries.excalidraw.com](https://libraries.excalidraw.com). | ||||
| @@ -99,7 +93,7 @@ These instructions will get you a copy of the project up and running on your loc | ||||
| #### Requirements | ||||
|  | ||||
| - [Node.js](https://nodejs.org/en/) | ||||
| - [Yarn](https://yarnpkg.com/getting-started/install) (v1 or v2.4.2+) | ||||
| - [Yarn](https://yarnpkg.com/getting-started/install) | ||||
| - [Git](https://git-scm.com/downloads) | ||||
|  | ||||
| #### Clone the repo | ||||
| @@ -122,47 +116,16 @@ yarn start | ||||
|  | ||||
| Now you can open [http://localhost:3000](http://localhost:3000) and start coding in your favorite code editor. | ||||
|  | ||||
| #### Collaboration | ||||
|  | ||||
| For collaboration, you will need to set up [collab server](https://github.com/excalidraw/excalidraw-room) in local. | ||||
|  | ||||
| #### Commands | ||||
|  | ||||
| ##### Install the dependencies | ||||
|  | ||||
| ``` | ||||
| yarn | ||||
| ``` | ||||
|  | ||||
| ##### Run the project | ||||
|  | ||||
| ``` | ||||
| yarn start | ||||
| ``` | ||||
|  | ||||
| ##### Reformat all files with Prettier | ||||
|  | ||||
| ``` | ||||
| yarn fix | ||||
| ``` | ||||
|  | ||||
| ##### Run tests | ||||
|  | ||||
| ``` | ||||
| yarn test | ||||
| ``` | ||||
|  | ||||
| ##### Update test snapshots | ||||
|  | ||||
| ``` | ||||
| yarn test:update | ||||
| ``` | ||||
|  | ||||
| ##### Test for formatting with Prettier | ||||
|  | ||||
| ``` | ||||
| yarn test:code | ||||
| ``` | ||||
| | Command            | Description                       | | ||||
| | ------------------ | --------------------------------- | | ||||
| | `yarn`             | Install the dependencies          | | ||||
| | `yarn start`       | Run the project                   | | ||||
| | `yarn fix`         | Reformat all files with Prettier  | | ||||
| | `yarn test`        | Run tests                         | | ||||
| | `yarn test:update` | Update test snapshots             | | ||||
| | `yarn test:code`   | Test for formatting with Prettier | | ||||
|  | ||||
| #### Docker Compose | ||||
|  | ||||
|   | ||||
| @@ -2,8 +2,5 @@ | ||||
|   "firestore": { | ||||
|     "rules": "firestore.rules", | ||||
|     "indexes": "firestore.indexes.json" | ||||
|   }, | ||||
|   "storage": { | ||||
|     "rules": "storage.rules" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,11 +0,0 @@ | ||||
| rules_version = '2'; | ||||
| service firebase.storage { | ||||
|   match /b/{bucket}/o { | ||||
|     match /{files}/rooms/{room}/{file} { | ||||
|     	allow get, write: if true; | ||||
|     } | ||||
|     match /{files}/shareLinks/{shareLink}/{file} { | ||||
|     	allow get, write: if true; | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										55
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										55
									
								
								package.json
									
									
									
									
									
								
							| @@ -21,27 +21,21 @@ | ||||
|   "dependencies": { | ||||
|     "@sentry/browser": "6.2.5", | ||||
|     "@sentry/integrations": "6.2.5", | ||||
|     "@testing-library/jest-dom": "5.16.2", | ||||
|     "@testing-library/react": "12.1.2", | ||||
|     "@tldraw/vec": "1.4.3", | ||||
|     "@types/jest": "27.4.0", | ||||
|     "@types/pica": "5.1.3", | ||||
|     "@types/react": "17.0.39", | ||||
|     "@types/react-dom": "17.0.11", | ||||
|     "@testing-library/jest-dom": "5.11.10", | ||||
|     "@testing-library/react": "11.2.6", | ||||
|     "@types/jest": "26.0.22", | ||||
|     "@types/react": "17.0.3", | ||||
|     "@types/react-dom": "17.0.3", | ||||
|     "@types/socket.io-client": "1.4.36", | ||||
|     "browser-fs-access": "0.24.1", | ||||
|     "browser-fs-access": "0.16.4", | ||||
|     "clsx": "1.1.1", | ||||
|     "fake-indexeddb": "3.1.7", | ||||
|     "firebase": "8.3.3", | ||||
|     "i18next-browser-languagedetector": "6.1.2", | ||||
|     "idb-keyval": "6.0.3", | ||||
|     "image-blob-reduce": "3.0.1", | ||||
|     "jotai": "1.6.4", | ||||
|     "i18next-browser-languagedetector": "6.1.0", | ||||
|     "lodash.throttle": "4.1.1", | ||||
|     "nanoid": "3.1.32", | ||||
|     "open-color": "1.9.1", | ||||
|     "nanoid": "3.1.22", | ||||
|     "open-color": "1.8.0", | ||||
|     "pako": "1.0.11", | ||||
|     "perfect-freehand": "1.0.16", | ||||
|     "perfect-freehand": "0.4.7", | ||||
|     "png-chunk-text": "1.0.0", | ||||
|     "png-chunks-encode": "1.0.0", | ||||
|     "png-chunks-extract": "1.0.0", | ||||
| @@ -50,36 +44,36 @@ | ||||
|     "react": "17.0.2", | ||||
|     "react-dom": "17.0.2", | ||||
|     "react-scripts": "4.0.3", | ||||
|     "roughjs": "4.5.2", | ||||
|     "sass": "1.49.7", | ||||
|     "roughjs": "4.4.1", | ||||
|     "sass": "1.32.10", | ||||
|     "socket.io-client": "2.3.1", | ||||
|     "typescript": "4.5.5" | ||||
|     "typescript": "4.2.4" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@excalidraw/eslint-config": "1.0.0", | ||||
|     "@excalidraw/prettier-config": "1.0.2", | ||||
|     "@types/chai": "4.3.0", | ||||
|     "@types/lodash.throttle": "4.1.6", | ||||
|     "@types/pako": "1.0.3", | ||||
|     "@types/resize-observer-browser": "0.1.6", | ||||
|     "chai": "4.3.6", | ||||
|     "dotenv": "10.0.0", | ||||
|     "@types/pako": "1.0.1", | ||||
|     "@types/resize-observer-browser": "0.1.5", | ||||
|     "eslint-config-prettier": "8.3.0", | ||||
|     "eslint-plugin-prettier": "3.3.1", | ||||
|     "husky": "7.0.4", | ||||
|     "firebase-tools": "9.9.0", | ||||
|     "husky": "4.3.8", | ||||
|     "jest-canvas-mock": "2.3.1", | ||||
|     "lint-staged": "12.3.7", | ||||
|     "lint-staged": "10.5.4", | ||||
|     "pepjs": "0.5.3", | ||||
|     "prettier": "2.5.1", | ||||
|     "prettier": "2.2.1", | ||||
|     "rewire": "5.0.0" | ||||
|   }, | ||||
|   "resolutions": { | ||||
|     "@typescript-eslint/typescript-estree": "5.10.2" | ||||
|   }, | ||||
|   "engines": { | ||||
|     "node": ">=14.0.0" | ||||
|   }, | ||||
|   "homepage": ".", | ||||
|   "husky": { | ||||
|     "hooks": { | ||||
|       "pre-commit": "lint-staged" | ||||
|     } | ||||
|   }, | ||||
|   "jest": { | ||||
|     "transformIgnorePatterns": [ | ||||
|       "node_modules/(?!(roughjs|points-on-curve|path-data-parser|points-on-path|browser-fs-access)/)" | ||||
| @@ -101,7 +95,6 @@ | ||||
|     "fix": "yarn fix:other && yarn fix:code", | ||||
|     "locales-coverage": "node scripts/build-locales-coverage.js", | ||||
|     "locales-coverage:description": "node scripts/locales-coverage-description.js", | ||||
|     "prepare": "husky install", | ||||
|     "prettier": "prettier \"**/*.{css,scss,json,md,html,yml}\" --ignore-path=.eslintignore", | ||||
|     "start": "react-scripts start", | ||||
|     "test:all": "yarn test:typecheck && yarn test:code && yarn test:other && yarn test:app --watchAll=false", | ||||
|   | ||||
										
											Binary file not shown.
										
									
								
							| @@ -72,6 +72,12 @@ | ||||
|       crossorigin="anonymous" | ||||
|     /> | ||||
|  | ||||
|     <link | ||||
|       href="%REACT_APP_SOCKET_SERVER_URL%/socket.io" | ||||
|       rel="preconnect" | ||||
|       crossorigin="anonymous" | ||||
|     /> | ||||
|  | ||||
|     <link | ||||
|       rel="manifest" | ||||
|       href="manifest.json" | ||||
| @@ -111,7 +117,6 @@ | ||||
|  | ||||
|         width: 100%; | ||||
|         height: 100%; | ||||
|         overflow: hidden; | ||||
|       } | ||||
|  | ||||
|       .visually-hidden { | ||||
| @@ -124,6 +129,26 @@ | ||||
|         user-select: none; | ||||
|       } | ||||
|  | ||||
|       .LoadingMessage { | ||||
|         position: absolute; | ||||
|         top: 0; | ||||
|         right: 0; | ||||
|         bottom: 0; | ||||
|         left: 0; | ||||
|         z-index: 999; | ||||
|         display: flex; | ||||
|         align-items: center; | ||||
|         justify-content: center; | ||||
|         pointer-events: none; | ||||
|       } | ||||
|  | ||||
|       .LoadingMessage span { | ||||
|         background-color: var(--button-gray-1); | ||||
|         border-radius: 5px; | ||||
|         padding: 0.8em 1.2em; | ||||
|         color: var(--popup-text-color); | ||||
|         font-size: 1.3em; | ||||
|       } | ||||
|       #root { | ||||
|         height: 100%; | ||||
|         -webkit-touch-callout: none; | ||||
| @@ -132,10 +157,8 @@ | ||||
|         -moz-user-select: none; | ||||
|         -ms-user-select: none; | ||||
|         user-select: none; | ||||
|       } | ||||
|  | ||||
|       @media screen and (min-width: 1200px) { | ||||
|         #root { | ||||
|         @media screen and (min-width: 1200px) { | ||||
|           -webkit-touch-callout: default; | ||||
|           -webkit-user-select: auto; | ||||
|           -khtml-user-select: auto; | ||||
| @@ -152,6 +175,10 @@ | ||||
|     <header> | ||||
|       <h1 class="visually-hidden">Excalidraw</h1> | ||||
|     </header> | ||||
|     <div id="root"></div> | ||||
|     <div id="root"> | ||||
|       <div class="LoadingMessage"> | ||||
|         <span>Loading scene...</span> | ||||
|       </div> | ||||
|     </div> | ||||
|   </body> | ||||
| </html> | ||||
|   | ||||
| @@ -26,6 +26,7 @@ | ||||
|       } | ||||
|     } | ||||
|   ], | ||||
|   "capture_links": "new_client", | ||||
|   "share_target": { | ||||
|     "action": "/web-share-target", | ||||
|     "method": "POST", | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| const fs = require("fs"); | ||||
| const { exec, execSync } = require("child_process"); | ||||
| const core = require("@actions/core"); | ||||
|  | ||||
| const excalidrawDir = `${__dirname}/../src/packages/excalidraw`; | ||||
| const excalidrawPackage = `${excalidrawDir}/package.json`; | ||||
| @@ -16,62 +15,37 @@ const publish = () => { | ||||
|     execSync(`yarn --frozen-lockfile`, { cwd: excalidrawDir }); | ||||
|     execSync(`yarn run build:umd`, { cwd: excalidrawDir }); | ||||
|     execSync(`yarn --cwd ${excalidrawDir} publish`); | ||||
|     console.info("Published 🎉"); | ||||
|     core.setOutput( | ||||
|       "result", | ||||
|       `**Preview version has been shipped** :rocket: | ||||
|     You can use [@excalidraw/excalidraw-preview@${pkg.version}](https://www.npmjs.com/package/@excalidraw/excalidraw-preview/v/${pkg.version}) for testing!`, | ||||
|     ); | ||||
|   } catch (error) { | ||||
|     core.setOutput("result", "package couldn't be published :warning:!"); | ||||
|     console.error(error); | ||||
|     process.exit(1); | ||||
|   } 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); | ||||
|     core.setOutput("result", ":warning: Package couldn't be published!"); | ||||
|     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 || file.indexOf("package.json")) >= 0 && | ||||
|       !filesToIgnoreRegex.test(file) | ||||
|     ); | ||||
|     return file.indexOf("src") >= 0 && !filesToIgnoreRegex.test(file); | ||||
|   }); | ||||
|  | ||||
|   if (!excalidrawPackageFiles.length) { | ||||
|     console.info("Skipping release as no valid diff found"); | ||||
|     core.setOutput("result", "Skipping release as no valid diff found"); | ||||
|     process.exit(0); | ||||
|   } | ||||
|  | ||||
|   // update package.json | ||||
|   pkg.version = `${pkg.version}-${getShortCommitHash()}`; | ||||
|   pkg.name = "@excalidraw/excalidraw-next"; | ||||
|   let version = `${pkg.version}-${getShortCommitHash()}`; | ||||
|  | ||||
|   // update readme | ||||
|   let data = fs.readFileSync(`${excalidrawDir}/README_NEXT.md`, "utf8"); | ||||
|  | ||||
|   const isPreview = process.argv.slice(2)[0] === "preview"; | ||||
|   if (isPreview) { | ||||
|     // use pullNumber-commithash as the version for preview | ||||
|     const pullRequestNumber = process.argv.slice(3)[0]; | ||||
|     version = `${pkg.version}-${pullRequestNumber}-${getShortCommitHash()}`; | ||||
|     // replace "excalidraw-next" with "excalidraw-preview" | ||||
|     pkg.name = "@excalidraw/excalidraw-preview"; | ||||
|     data = data.replace(/excalidraw-next/g, "excalidraw-preview"); | ||||
|     data = data.trim(); | ||||
|   } | ||||
|   pkg.version = version; | ||||
|  | ||||
|   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"); | ||||
|   console.info("Publish in progress..."); | ||||
|  | ||||
|   publish(); | ||||
| }); | ||||
|   | ||||
| @@ -1,16 +1,11 @@ | ||||
| const { readdirSync, writeFileSync } = require("fs"); | ||||
| const files = readdirSync(`${__dirname}/../src/locales`); | ||||
|  | ||||
| const flatten = (object = {}, result = {}, extraKey = "") => { | ||||
|   for (const key in object) { | ||||
|     if (typeof object[key] !== "object") { | ||||
|       result[extraKey + key] = object[key]; | ||||
|     } else { | ||||
|       flatten(object[key], result, `${extraKey}${key}.`); | ||||
|     } | ||||
|   } | ||||
|   return result; | ||||
| }; | ||||
| const flatten = (object) => | ||||
|   Object.keys(object).reduce( | ||||
|     (initial, current) => ({ ...initial, ...object[current] }), | ||||
|     {}, | ||||
|   ); | ||||
|  | ||||
| const locales = files.filter( | ||||
|   (file) => file !== "README.md" && file !== "percentages.json", | ||||
| @@ -24,8 +19,10 @@ for (let index = 0; index < locales.length; index++) { | ||||
|  | ||||
|   const allKeys = Object.keys(data); | ||||
|   const translatedKeys = allKeys.filter((item) => data[item] !== ""); | ||||
|   const percentage = Math.floor((100 * translatedKeys.length) / allKeys.length); | ||||
|   percentages[currentLocale.replace(".json", "")] = percentage; | ||||
|  | ||||
|   const percentage = (100 * translatedKeys.length) / allKeys.length; | ||||
|  | ||||
|   percentages[currentLocale.replace(".json", "")] = parseInt(percentage); | ||||
| } | ||||
|  | ||||
| writeFileSync( | ||||
|   | ||||
| @@ -5,13 +5,10 @@ const THRESSHOLD = 85; | ||||
| const crowdinMap = { | ||||
|   "ar-SA": "en-ar", | ||||
|   "bg-BG": "en-bg", | ||||
|   "bn-BD": "en-bn", | ||||
|   "ca-ES": "en-ca", | ||||
|   "da-DK": "en-da", | ||||
|   "de-DE": "en-de", | ||||
|   "el-GR": "en-el", | ||||
|   "es-ES": "en-es", | ||||
|   "eu-ES": "en-eu", | ||||
|   "fa-IR": "en-fa", | ||||
|   "fi-FI": "en-fi", | ||||
|   "fr-FR": "en-fr", | ||||
| @@ -34,28 +31,19 @@ const crowdinMap = { | ||||
|   "pt-PT": "en-pt", | ||||
|   "ro-RO": "en-ro", | ||||
|   "ru-RU": "en-ru", | ||||
|   "si-LK": "en-silk", | ||||
|   "sk-SK": "en-sk", | ||||
|   "sv-SE": "en-sv", | ||||
|   "ta-IN": "en-ta", | ||||
|   "tr-TR": "en-tr", | ||||
|   "uk-UA": "en-uk", | ||||
|   "zh-CN": "en-zhcn", | ||||
|   "zh-HK": "en-zhhk", | ||||
|   "zh-TW": "en-zhtw", | ||||
|   "lt-LT": "en-lt", | ||||
|   "lv-LV": "en-lv", | ||||
|   "cs-CZ": "en-cs", | ||||
|   "kk-KZ": "en-kk", | ||||
| }; | ||||
|  | ||||
| const flags = { | ||||
|   "ar-SA": "🇸🇦", | ||||
|   "bg-BG": "🇧🇬", | ||||
|   "bn-BD": "🇧🇩", | ||||
|   "ca-ES": "🏳", | ||||
|   "cs-CZ": "🇨🇿", | ||||
|   "da-DK": "🇩🇰", | ||||
|   "de-DE": "🇩🇪", | ||||
|   "el-GR": "🇬🇷", | ||||
|   "es-ES": "🇪🇸", | ||||
| @@ -69,10 +57,7 @@ const flags = { | ||||
|   "it-IT": "🇮🇹", | ||||
|   "ja-JP": "🇯🇵", | ||||
|   "kab-KAB": "🏳", | ||||
|   "kk-KZ": "🇰🇿", | ||||
|   "ko-KR": "🇰🇷", | ||||
|   "lt-LT": "🇱🇹", | ||||
|   "lv-LV": "🇱🇻", | ||||
|   "my-MM": "🇲🇲", | ||||
|   "nb-NO": "🇳🇴", | ||||
|   "nl-NL": "🇳🇱", | ||||
| @@ -84,28 +69,22 @@ const flags = { | ||||
|   "pt-PT": "🇵🇹", | ||||
|   "ro-RO": "🇷🇴", | ||||
|   "ru-RU": "🇷🇺", | ||||
|   "si-LK": "🇱🇰", | ||||
|   "sk-SK": "🇸🇰", | ||||
|   "sv-SE": "🇸🇪", | ||||
|   "ta-IN": "🇮🇳", | ||||
|   "tr-TR": "🇹🇷", | ||||
|   "uk-UA": "🇺🇦", | ||||
|   "zh-CN": "🇨🇳", | ||||
|   "zh-HK": "🇭🇰", | ||||
|   "zh-TW": "🇹🇼", | ||||
|   "lv-LV": "🇱🇻", | ||||
| }; | ||||
|  | ||||
| const languages = { | ||||
|   "ar-SA": "العربية", | ||||
|   "bg-BG": "Български", | ||||
|   "bn-BD": "Bengali", | ||||
|   "ca-ES": "Català", | ||||
|   "cs-CZ": "Česky", | ||||
|   "da-DK": "Dansk", | ||||
|   "de-DE": "Deutsch", | ||||
|   "el-GR": "Ελληνικά", | ||||
|   "es-ES": "Español", | ||||
|   "eu-ES": "Euskara", | ||||
|   "fa-IR": "فارسی", | ||||
|   "fi-FI": "Suomi", | ||||
|   "fr-FR": "Français", | ||||
| @@ -116,10 +95,7 @@ const languages = { | ||||
|   "it-IT": "Italiano", | ||||
|   "ja-JP": "日本語", | ||||
|   "kab-KAB": "Taqbaylit", | ||||
|   "kk-KZ": "Қазақ тілі", | ||||
|   "ko-KR": "한국어", | ||||
|   "lt-LT": "Lietuvių", | ||||
|   "lv-LV": "Latviešu", | ||||
|   "my-MM": "Burmese", | ||||
|   "nb-NO": "Norsk bokmål", | ||||
|   "nl-NL": "Nederlands", | ||||
| @@ -131,15 +107,13 @@ const languages = { | ||||
|   "pt-PT": "Português", | ||||
|   "ro-RO": "Română", | ||||
|   "ru-RU": "Русский", | ||||
|   "si-LK": "සිංහල", | ||||
|   "sk-SK": "Slovenčina", | ||||
|   "sv-SE": "Svenska", | ||||
|   "ta-IN": "Tamil", | ||||
|   "tr-TR": "Türkçe", | ||||
|   "uk-UA": "Українська", | ||||
|   "zh-CN": "简体中文", | ||||
|   "zh-HK": "繁體中文 (香港)", | ||||
|   "zh-TW": "繁體中文", | ||||
|   "lv-LV": "Latviešu", | ||||
| }; | ||||
|  | ||||
| const percentages = fs.readFileSync( | ||||
|   | ||||
| @@ -1,39 +0,0 @@ | ||||
| const fs = require("fs"); | ||||
| const util = require("util"); | ||||
| const exec = util.promisify(require("child_process").exec); | ||||
| const updateReadme = require("./updateReadme"); | ||||
| const updateChangelog = require("./updateChangelog"); | ||||
|  | ||||
| const excalidrawDir = `${__dirname}/../src/packages/excalidraw`; | ||||
| const excalidrawPackage = `${excalidrawDir}/package.json`; | ||||
|  | ||||
| const updatePackageVersion = (nextVersion) => { | ||||
|   const pkg = require(excalidrawPackage); | ||||
|   pkg.version = nextVersion; | ||||
|   const content = `${JSON.stringify(pkg, null, 2)}\n`; | ||||
|   fs.writeFileSync(excalidrawPackage, content, "utf-8"); | ||||
| }; | ||||
|  | ||||
| const release = async (nextVersion) => { | ||||
|   try { | ||||
|     updateReadme(); | ||||
|     await updateChangelog(nextVersion); | ||||
|     updatePackageVersion(nextVersion); | ||||
|     await exec(`git add -u`); | ||||
|     await exec( | ||||
|       `git commit -m "docs: release @excalidraw/excalidraw@${nextVersion}  🎉"`, | ||||
|     ); | ||||
|     /* eslint-disable no-console */ | ||||
|     console.log("Done!"); | ||||
|   } catch (error) { | ||||
|     console.error(error); | ||||
|     process.exit(1); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const nextVersion = process.argv.slice(2)[0]; | ||||
| if (!nextVersion) { | ||||
|   console.error("Pass the next version to release!"); | ||||
|   process.exit(1); | ||||
| } | ||||
| release(nextVersion); | ||||
| @@ -1,104 +0,0 @@ | ||||
| const fs = require("fs"); | ||||
| const util = require("util"); | ||||
| const exec = util.promisify(require("child_process").exec); | ||||
|  | ||||
| const excalidrawDir = `${__dirname}/../src/packages/excalidraw`; | ||||
| const excalidrawPackage = `${excalidrawDir}/package.json`; | ||||
| const pkg = require(excalidrawPackage); | ||||
| const lastVersion = pkg.version; | ||||
| const existingChangeLog = fs.readFileSync( | ||||
|   `${excalidrawDir}/CHANGELOG.md`, | ||||
|   "utf8", | ||||
| ); | ||||
|  | ||||
| const supportedTypes = ["feat", "fix", "style", "refactor", "perf", "build"]; | ||||
| const headerForType = { | ||||
|   feat: "Features", | ||||
|   fix: "Fixes", | ||||
|   style: "Styles", | ||||
|   refactor: " Refactor", | ||||
|   perf: "Performance", | ||||
|   build: "Build", | ||||
| }; | ||||
| const badCommits = []; | ||||
| const getCommitHashForLastVersion = async () => { | ||||
|   try { | ||||
|     const commitMessage = `"release @excalidraw/excalidraw@${lastVersion}"`; | ||||
|     const { stdout } = await exec( | ||||
|       `git log --format=format:"%H" --grep=${commitMessage}`, | ||||
|     ); | ||||
|     return stdout; | ||||
|   } catch (error) { | ||||
|     console.error(error); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const getLibraryCommitsSinceLastRelease = async () => { | ||||
|   const commitHash = await getCommitHashForLastVersion(); | ||||
|   const { stdout } = await exec( | ||||
|     `git log --pretty=format:%s ${commitHash}...master`, | ||||
|   ); | ||||
|   const commitsSinceLastRelease = stdout.split("\n"); | ||||
|   const commitList = {}; | ||||
|   supportedTypes.forEach((type) => { | ||||
|     commitList[type] = []; | ||||
|   }); | ||||
|  | ||||
|   commitsSinceLastRelease.forEach((commit) => { | ||||
|     const indexOfColon = commit.indexOf(":"); | ||||
|     const type = commit.slice(0, indexOfColon); | ||||
|     if (!supportedTypes.includes(type)) { | ||||
|       return; | ||||
|     } | ||||
|     const messageWithoutType = commit.slice(indexOfColon + 1).trim(); | ||||
|     const messageWithCapitalizeFirst = | ||||
|       messageWithoutType.charAt(0).toUpperCase() + messageWithoutType.slice(1); | ||||
|     const prMatch = commit.match(/\(#([0-9]*)\)/); | ||||
|     if (prMatch) { | ||||
|       const prNumber = prMatch[1]; | ||||
|  | ||||
|       // return if the changelog already contains the pr number which would happen for package updates | ||||
|       if (existingChangeLog.includes(prNumber)) { | ||||
|         return; | ||||
|       } | ||||
|       const prMarkdown = `[#${prNumber}](https://github.com/excalidraw/excalidraw/pull/${prNumber})`; | ||||
|       const messageWithPRLink = messageWithCapitalizeFirst.replace( | ||||
|         /\(#[0-9]*\)/, | ||||
|         prMarkdown, | ||||
|       ); | ||||
|       commitList[type].push(messageWithPRLink); | ||||
|     } else { | ||||
|       badCommits.push(commit); | ||||
|       commitList[type].push(messageWithCapitalizeFirst); | ||||
|     } | ||||
|   }); | ||||
|   console.info("Bad commits:", badCommits); | ||||
|   return commitList; | ||||
| }; | ||||
|  | ||||
| const updateChangelog = async (nextVersion) => { | ||||
|   const commitList = await getLibraryCommitsSinceLastRelease(); | ||||
|   let changelogForLibrary = | ||||
|     "## Excalidraw Library\n\n**_This section lists the updates made to the excalidraw library and will not affect the integration._**\n\n"; | ||||
|   supportedTypes.forEach((type) => { | ||||
|     if (commitList[type].length) { | ||||
|       changelogForLibrary += `### ${headerForType[type]}\n\n`; | ||||
|       const commits = commitList[type]; | ||||
|       commits.forEach((commit) => { | ||||
|         changelogForLibrary += `- ${commit}\n\n`; | ||||
|       }); | ||||
|     } | ||||
|   }); | ||||
|   changelogForLibrary += "---\n"; | ||||
|   const lastVersionIndex = existingChangeLog.indexOf(`## ${lastVersion}`); | ||||
|   let updatedContent = | ||||
|     existingChangeLog.slice(0, lastVersionIndex) + | ||||
|     changelogForLibrary + | ||||
|     existingChangeLog.slice(lastVersionIndex); | ||||
|   const currentDate = new Date().toISOString().slice(0, 10); | ||||
|   const newVersion = `## ${nextVersion} (${currentDate})`; | ||||
|   updatedContent = updatedContent.replace(`## Unreleased`, newVersion); | ||||
|   fs.writeFileSync(`${excalidrawDir}/CHANGELOG.md`, updatedContent, "utf8"); | ||||
| }; | ||||
|  | ||||
| module.exports = updateChangelog; | ||||
| @@ -1,27 +0,0 @@ | ||||
| const fs = require("fs"); | ||||
|  | ||||
| const updateReadme = () => { | ||||
|   const excalidrawDir = `${__dirname}/../src/packages/excalidraw`; | ||||
|   let data = fs.readFileSync(`${excalidrawDir}/README_NEXT.md`, "utf8"); | ||||
|  | ||||
|   // remove note for unstable release | ||||
|   data = data.replace( | ||||
|     /<!-- unstable-readme-start-->[\s\S]*?<!-- unstable-readme-end-->/, | ||||
|     "", | ||||
|   ); | ||||
|  | ||||
|   // replace "excalidraw-next" with "excalidraw" | ||||
|   data = data.replace(/excalidraw-next/g, "excalidraw"); | ||||
|   data = data.trim(); | ||||
|  | ||||
|   const demoIndex = data.indexOf("### Demo"); | ||||
|   const excalidrawNextNote = | ||||
|     "#### Note\n\n**If you don't want to wait for the next stable release and try out the unreleased changes you can use [@excalidraw/excalidraw-next](https://www.npmjs.com/package/@excalidraw/excalidraw-next).**\n\n"; | ||||
|   // Add excalidraw next note to try out for unreleased changes | ||||
|   data = data.slice(0, demoIndex) + excalidrawNextNote + data.slice(demoIndex); | ||||
|  | ||||
|   // update readme | ||||
|   fs.writeFileSync(`${excalidrawDir}/README.md`, data, "utf8"); | ||||
| }; | ||||
|  | ||||
| module.exports = updateReadme; | ||||
| @@ -2,59 +2,22 @@ import { register } from "./register"; | ||||
| import { getSelectedElements } from "../scene"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import { deepCopyElement } from "../element/newElement"; | ||||
| import { randomId } from "../random"; | ||||
| import { t } from "../i18n"; | ||||
|  | ||||
| export const actionAddToLibrary = register({ | ||||
|   name: "addToLibrary", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState, _, app) => { | ||||
|     const selectedElements = getSelectedElements( | ||||
|       getNonDeletedElements(elements), | ||||
|       appState, | ||||
|       true, | ||||
|     ); | ||||
|     if (selectedElements.some((element) => element.type === "image")) { | ||||
|       return { | ||||
|         commitToHistory: false, | ||||
|         appState: { | ||||
|           ...appState, | ||||
|           errorMessage: "Support for adding images to the library coming soon!", | ||||
|         }, | ||||
|       }; | ||||
|     } | ||||
|  | ||||
|     return app.library | ||||
|       .loadLibrary() | ||||
|       .then((items) => { | ||||
|         return app.library.saveLibrary([ | ||||
|           { | ||||
|             id: randomId(), | ||||
|             status: "unpublished", | ||||
|             elements: selectedElements.map(deepCopyElement), | ||||
|             created: Date.now(), | ||||
|           }, | ||||
|           ...items, | ||||
|         ]); | ||||
|       }) | ||||
|       .then(() => { | ||||
|         return { | ||||
|           commitToHistory: false, | ||||
|           appState: { | ||||
|             ...appState, | ||||
|             toastMessage: t("toast.addedToLibrary"), | ||||
|           }, | ||||
|         }; | ||||
|       }) | ||||
|       .catch((error) => { | ||||
|         return { | ||||
|           commitToHistory: false, | ||||
|           appState: { | ||||
|             ...appState, | ||||
|             errorMessage: error.message, | ||||
|           }, | ||||
|         }; | ||||
|       }); | ||||
|     app.library.loadLibrary().then((items) => { | ||||
|       app.library.saveLibrary([ | ||||
|         ...items, | ||||
|         selectedElements.map(deepCopyElement), | ||||
|       ]); | ||||
|     }); | ||||
|     return false; | ||||
|   }, | ||||
|   contextItemLabel: "labels.addToLibrary", | ||||
| }); | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import React from "react"; | ||||
| import { alignElements, Alignment } from "../align"; | ||||
| import { | ||||
|   AlignBottomIcon, | ||||
| @@ -8,13 +9,13 @@ import { | ||||
|   CenterVerticallyIcon, | ||||
| } from "../components/icons"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import { getElementMap, getNonDeletedElements } from "../element"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { t } from "../i18n"; | ||||
| import { KEYS } from "../keys"; | ||||
| import { getSelectedElements, isSomeElementSelected } from "../scene"; | ||||
| import { AppState } from "../types"; | ||||
| import { arrayToMap, getShortcutKey } from "../utils"; | ||||
| import { getShortcutKey } from "../utils"; | ||||
| import { register } from "./register"; | ||||
|  | ||||
| const enableActionGroup = ( | ||||
| @@ -34,16 +35,13 @@ const alignSelectedElements = ( | ||||
|  | ||||
|   const updatedElements = alignElements(selectedElements, alignment); | ||||
|  | ||||
|   const updatedElementsMap = arrayToMap(updatedElements); | ||||
|   const updatedElementsMap = getElementMap(updatedElements); | ||||
|  | ||||
|   return elements.map( | ||||
|     (element) => updatedElementsMap.get(element.id) || element, | ||||
|   ); | ||||
|   return elements.map((element) => updatedElementsMap[element.id] || element); | ||||
| }; | ||||
|  | ||||
| export const actionAlignTop = register({ | ||||
|   name: "alignTop", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       appState, | ||||
| @@ -73,7 +71,6 @@ export const actionAlignTop = register({ | ||||
|  | ||||
| export const actionAlignBottom = register({ | ||||
|   name: "alignBottom", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       appState, | ||||
| @@ -103,7 +100,6 @@ export const actionAlignBottom = register({ | ||||
|  | ||||
| export const actionAlignLeft = register({ | ||||
|   name: "alignLeft", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       appState, | ||||
| @@ -133,8 +129,6 @@ export const actionAlignLeft = register({ | ||||
|  | ||||
| export const actionAlignRight = register({ | ||||
|   name: "alignRight", | ||||
|   trackEvent: { category: "element" }, | ||||
|  | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       appState, | ||||
| @@ -164,8 +158,6 @@ export const actionAlignRight = register({ | ||||
|  | ||||
| export const actionAlignVerticallyCentered = register({ | ||||
|   name: "alignVerticallyCentered", | ||||
|   trackEvent: { category: "element" }, | ||||
|  | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       appState, | ||||
| @@ -191,7 +183,6 @@ export const actionAlignVerticallyCentered = register({ | ||||
|  | ||||
| export const actionAlignHorizontallyCentered = register({ | ||||
|   name: "alignHorizontallyCentered", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       appState, | ||||
|   | ||||
| @@ -1,136 +0,0 @@ | ||||
| import { VERTICAL_ALIGN } from "../constants"; | ||||
| import { getNonDeletedElements, isTextElement } from "../element"; | ||||
| import { mutateElement } from "../element/mutateElement"; | ||||
| import { | ||||
|   getBoundTextElement, | ||||
|   measureText, | ||||
|   redrawTextBoundingBox, | ||||
| } from "../element/textElement"; | ||||
| import { | ||||
|   hasBoundTextElement, | ||||
|   isTextBindableContainer, | ||||
| } from "../element/typeChecks"; | ||||
| import { | ||||
|   ExcalidrawTextContainer, | ||||
|   ExcalidrawTextElement, | ||||
| } from "../element/types"; | ||||
| import { getSelectedElements } from "../scene"; | ||||
| import { getFontString } from "../utils"; | ||||
| import { register } from "./register"; | ||||
|  | ||||
| export const actionUnbindText = register({ | ||||
|   name: "unbindText", | ||||
|   contextItemLabel: "labels.unbindText", | ||||
|   trackEvent: { category: "element" }, | ||||
|   contextItemPredicate: (elements, appState) => { | ||||
|     const selectedElements = getSelectedElements(elements, appState); | ||||
|     return selectedElements.some((element) => hasBoundTextElement(element)); | ||||
|   }, | ||||
|   perform: (elements, appState) => { | ||||
|     const selectedElements = getSelectedElements( | ||||
|       getNonDeletedElements(elements), | ||||
|       appState, | ||||
|     ); | ||||
|     selectedElements.forEach((element) => { | ||||
|       const boundTextElement = getBoundTextElement(element); | ||||
|       if (boundTextElement) { | ||||
|         const { width, height, baseline } = measureText( | ||||
|           boundTextElement.originalText, | ||||
|           getFontString(boundTextElement), | ||||
|         ); | ||||
|         mutateElement(boundTextElement as ExcalidrawTextElement, { | ||||
|           containerId: null, | ||||
|           width, | ||||
|           height, | ||||
|           baseline, | ||||
|           text: boundTextElement.originalText, | ||||
|         }); | ||||
|         mutateElement(element, { | ||||
|           boundElements: element.boundElements?.filter( | ||||
|             (ele) => ele.id !== boundTextElement.id, | ||||
|           ), | ||||
|         }); | ||||
|       } | ||||
|     }); | ||||
|     return { | ||||
|       elements, | ||||
|       appState, | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
| }); | ||||
|  | ||||
| export const actionBindText = register({ | ||||
|   name: "bindText", | ||||
|   contextItemLabel: "labels.bindText", | ||||
|   trackEvent: { category: "element" }, | ||||
|   contextItemPredicate: (elements, appState) => { | ||||
|     const selectedElements = getSelectedElements(elements, appState); | ||||
|  | ||||
|     if (selectedElements.length === 2) { | ||||
|       const textElement = | ||||
|         isTextElement(selectedElements[0]) || | ||||
|         isTextElement(selectedElements[1]); | ||||
|  | ||||
|       let bindingContainer; | ||||
|       if (isTextBindableContainer(selectedElements[0])) { | ||||
|         bindingContainer = selectedElements[0]; | ||||
|       } else if (isTextBindableContainer(selectedElements[1])) { | ||||
|         bindingContainer = selectedElements[1]; | ||||
|       } | ||||
|       if ( | ||||
|         textElement && | ||||
|         bindingContainer && | ||||
|         getBoundTextElement(bindingContainer) === null | ||||
|       ) { | ||||
|         return true; | ||||
|       } | ||||
|     } | ||||
|     return false; | ||||
|   }, | ||||
|   perform: (elements, appState) => { | ||||
|     const selectedElements = getSelectedElements( | ||||
|       getNonDeletedElements(elements), | ||||
|       appState, | ||||
|     ); | ||||
|  | ||||
|     let textElement: ExcalidrawTextElement; | ||||
|     let container: ExcalidrawTextContainer; | ||||
|  | ||||
|     if ( | ||||
|       isTextElement(selectedElements[0]) && | ||||
|       isTextBindableContainer(selectedElements[1]) | ||||
|     ) { | ||||
|       textElement = selectedElements[0]; | ||||
|       container = selectedElements[1]; | ||||
|     } else { | ||||
|       textElement = selectedElements[1] as ExcalidrawTextElement; | ||||
|       container = selectedElements[0] as ExcalidrawTextContainer; | ||||
|     } | ||||
|     mutateElement(textElement, { | ||||
|       containerId: container.id, | ||||
|       verticalAlign: VERTICAL_ALIGN.MIDDLE, | ||||
|     }); | ||||
|     mutateElement(container, { | ||||
|       boundElements: (container.boundElements || []).concat({ | ||||
|         type: "text", | ||||
|         id: textElement.id, | ||||
|       }), | ||||
|     }); | ||||
|     redrawTextBoundingBox(textElement, container); | ||||
|     const updatedElements = elements.slice(); | ||||
|     const textElementIndex = updatedElements.findIndex( | ||||
|       (ele) => ele.id === textElement.id, | ||||
|     ); | ||||
|     updatedElements.splice(textElementIndex, 1); | ||||
|     const containerIndex = updatedElements.findIndex( | ||||
|       (ele) => ele.id === container.id, | ||||
|     ); | ||||
|     updatedElements.splice(containerIndex + 1, 0, textElement); | ||||
|     return { | ||||
|       elements: updatedElements, | ||||
|       appState: { ...appState, selectedElementIds: { [container.id]: true } }, | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
| }); | ||||
| @@ -1,48 +1,40 @@ | ||||
| import React from "react"; | ||||
| import { getDefaultAppState } from "../appState"; | ||||
| import { ColorPicker } from "../components/ColorPicker"; | ||||
| import { eraser, zoomIn, zoomOut } from "../components/icons"; | ||||
| import { resetZoom, trash, zoomIn, zoomOut } from "../components/icons"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import { DarkModeToggle } from "../components/DarkModeToggle"; | ||||
| import { THEME, ZOOM_STEP } from "../constants"; | ||||
| import { ZOOM_STEP } from "../constants"; | ||||
| import { getCommonBounds, getNonDeletedElements } from "../element"; | ||||
| import { newElementWith } from "../element/mutateElement"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { t } from "../i18n"; | ||||
| import { useIsMobile } from "../components/App"; | ||||
| import { CODES, KEYS } from "../keys"; | ||||
| import { getNormalizedZoom, getSelectedElements } from "../scene"; | ||||
| import { centerScrollOn } from "../scene/scroll"; | ||||
| import { getStateForZoom } from "../scene/zoom"; | ||||
| import { getNewZoom } from "../scene/zoom"; | ||||
| import { AppState, NormalizedZoomValue } from "../types"; | ||||
| import { getShortcutKey } from "../utils"; | ||||
| import { register } from "./register"; | ||||
| import { Tooltip } from "../components/Tooltip"; | ||||
| import { newElementWith } from "../element/mutateElement"; | ||||
| import { getDefaultAppState, isEraserActive } from "../appState"; | ||||
| import ClearCanvas from "../components/ClearCanvas"; | ||||
| import clsx from "clsx"; | ||||
|  | ||||
| export const actionChangeViewBackgroundColor = register({ | ||||
|   name: "changeViewBackgroundColor", | ||||
|   trackEvent: false, | ||||
|   perform: (_, appState, value) => { | ||||
|     return { | ||||
|       appState: { ...appState, ...value }, | ||||
|       commitToHistory: !!value.viewBackgroundColor, | ||||
|       appState: { ...appState, viewBackgroundColor: value }, | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ elements, appState, updateData }) => { | ||||
|   PanelComponent: ({ appState, updateData }) => { | ||||
|     return ( | ||||
|       <div style={{ position: "relative" }}> | ||||
|         <ColorPicker | ||||
|           label={t("labels.canvasBackground")} | ||||
|           type="canvasBackground" | ||||
|           color={appState.viewBackgroundColor} | ||||
|           onChange={(color) => updateData({ viewBackgroundColor: color })} | ||||
|           isActive={appState.openPopup === "canvasColorPicker"} | ||||
|           setActive={(active) => | ||||
|             updateData({ openPopup: active ? "canvasColorPicker" : null }) | ||||
|           } | ||||
|           onChange={(color) => updateData(color)} | ||||
|           data-testid="canvas-background-picker" | ||||
|           elements={elements} | ||||
|           appState={appState} | ||||
|         /> | ||||
|       </div> | ||||
|     ); | ||||
| @@ -51,51 +43,55 @@ export const actionChangeViewBackgroundColor = register({ | ||||
|  | ||||
| export const actionClearCanvas = register({ | ||||
|   name: "clearCanvas", | ||||
|   trackEvent: { category: "canvas" }, | ||||
|   perform: (elements, appState, _, app) => { | ||||
|     app.imageCache.clear(); | ||||
|   perform: (elements, appState: AppState) => { | ||||
|     return { | ||||
|       elements: elements.map((element) => | ||||
|         newElementWith(element, { isDeleted: true }), | ||||
|       ), | ||||
|       appState: { | ||||
|         ...getDefaultAppState(), | ||||
|         files: {}, | ||||
|         theme: appState.theme, | ||||
|         penMode: appState.penMode, | ||||
|         penDetected: appState.penDetected, | ||||
|         elementLocked: appState.elementLocked, | ||||
|         exportBackground: appState.exportBackground, | ||||
|         exportEmbedScene: appState.exportEmbedScene, | ||||
|         gridSize: appState.gridSize, | ||||
|         shouldAddWatermark: appState.shouldAddWatermark, | ||||
|         showStats: appState.showStats, | ||||
|         pasteDialog: appState.pasteDialog, | ||||
|         activeTool: | ||||
|           appState.activeTool.type === "image" | ||||
|             ? { ...appState.activeTool, type: "selection" } | ||||
|             : appState.activeTool, | ||||
|       }, | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
|  | ||||
|   PanelComponent: ({ updateData }) => <ClearCanvas onConfirm={updateData} />, | ||||
|   PanelComponent: ({ updateData }) => ( | ||||
|     <ToolButton | ||||
|       type="button" | ||||
|       icon={trash} | ||||
|       title={t("buttons.clearReset")} | ||||
|       aria-label={t("buttons.clearReset")} | ||||
|       showAriaLabel={useIsMobile()} | ||||
|       onClick={() => { | ||||
|         if (window.confirm(t("alerts.clearReset"))) { | ||||
|           updateData(null); | ||||
|         } | ||||
|       }} | ||||
|       data-testid="clear-canvas-button" | ||||
|     /> | ||||
|   ), | ||||
| }); | ||||
|  | ||||
| export const actionZoomIn = register({ | ||||
|   name: "zoomIn", | ||||
|   trackEvent: { category: "canvas" }, | ||||
|   perform: (_elements, appState, _, app) => { | ||||
|   perform: (_elements, appState) => { | ||||
|     const zoom = getNewZoom( | ||||
|       getNormalizedZoom(appState.zoom.value + ZOOM_STEP), | ||||
|       appState.zoom, | ||||
|       { left: appState.offsetLeft, top: appState.offsetTop }, | ||||
|       { x: appState.width / 2, y: appState.height / 2 }, | ||||
|     ); | ||||
|     return { | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         ...getStateForZoom( | ||||
|           { | ||||
|             viewportX: appState.width / 2 + appState.offsetLeft, | ||||
|             viewportY: appState.height / 2 + appState.offsetTop, | ||||
|             nextZoom: getNormalizedZoom(appState.zoom.value + ZOOM_STEP), | ||||
|           }, | ||||
|           appState, | ||||
|         ), | ||||
|         zoom, | ||||
|       }, | ||||
|       commitToHistory: false, | ||||
|     }; | ||||
| @@ -109,7 +105,6 @@ export const actionZoomIn = register({ | ||||
|       onClick={() => { | ||||
|         updateData(null); | ||||
|       }} | ||||
|       size="small" | ||||
|     /> | ||||
|   ), | ||||
|   keyTest: (event) => | ||||
| @@ -119,19 +114,18 @@ export const actionZoomIn = register({ | ||||
|  | ||||
| export const actionZoomOut = register({ | ||||
|   name: "zoomOut", | ||||
|   trackEvent: { category: "canvas" }, | ||||
|   perform: (_elements, appState, _, app) => { | ||||
|   perform: (_elements, appState) => { | ||||
|     const zoom = getNewZoom( | ||||
|       getNormalizedZoom(appState.zoom.value - ZOOM_STEP), | ||||
|       appState.zoom, | ||||
|       { left: appState.offsetLeft, top: appState.offsetTop }, | ||||
|       { x: appState.width / 2, y: appState.height / 2 }, | ||||
|     ); | ||||
|  | ||||
|     return { | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         ...getStateForZoom( | ||||
|           { | ||||
|             viewportX: appState.width / 2 + appState.offsetLeft, | ||||
|             viewportY: appState.height / 2 + appState.offsetTop, | ||||
|             nextZoom: getNormalizedZoom(appState.zoom.value - ZOOM_STEP), | ||||
|           }, | ||||
|           appState, | ||||
|         ), | ||||
|         zoom, | ||||
|       }, | ||||
|       commitToHistory: false, | ||||
|     }; | ||||
| @@ -145,7 +139,6 @@ export const actionZoomOut = register({ | ||||
|       onClick={() => { | ||||
|         updateData(null); | ||||
|       }} | ||||
|       size="small" | ||||
|     /> | ||||
|   ), | ||||
|   keyTest: (event) => | ||||
| @@ -155,38 +148,33 @@ export const actionZoomOut = register({ | ||||
|  | ||||
| export const actionResetZoom = register({ | ||||
|   name: "resetZoom", | ||||
|   trackEvent: { category: "canvas" }, | ||||
|   perform: (_elements, appState, _, app) => { | ||||
|   perform: (_elements, appState) => { | ||||
|     return { | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         ...getStateForZoom( | ||||
|         zoom: getNewZoom( | ||||
|           1 as NormalizedZoomValue, | ||||
|           appState.zoom, | ||||
|           { left: appState.offsetLeft, top: appState.offsetTop }, | ||||
|           { | ||||
|             viewportX: appState.width / 2 + appState.offsetLeft, | ||||
|             viewportY: appState.height / 2 + appState.offsetTop, | ||||
|             nextZoom: getNormalizedZoom(1), | ||||
|             x: appState.width / 2, | ||||
|             y: appState.height / 2, | ||||
|           }, | ||||
|           appState, | ||||
|         ), | ||||
|       }, | ||||
|       commitToHistory: false, | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ updateData, appState }) => ( | ||||
|     <Tooltip label={t("buttons.resetZoom")} style={{ height: "100%" }}> | ||||
|       <ToolButton | ||||
|         type="button" | ||||
|         className="reset-zoom-button" | ||||
|         title={t("buttons.resetZoom")} | ||||
|         aria-label={t("buttons.resetZoom")} | ||||
|         onClick={() => { | ||||
|           updateData(null); | ||||
|         }} | ||||
|         size="small" | ||||
|       > | ||||
|         {(appState.zoom.value * 100).toFixed(0)}% | ||||
|       </ToolButton> | ||||
|     </Tooltip> | ||||
|   PanelComponent: ({ updateData }) => ( | ||||
|     <ToolButton | ||||
|       type="button" | ||||
|       icon={resetZoom} | ||||
|       title={t("buttons.resetZoom")} | ||||
|       aria-label={t("buttons.resetZoom")} | ||||
|       onClick={() => { | ||||
|         updateData(null); | ||||
|       }} | ||||
|     /> | ||||
|   ), | ||||
|   keyTest: (event) => | ||||
|     (event.code === CODES.ZERO || event.code === CODES.NUM_ZERO) && | ||||
| @@ -225,12 +213,14 @@ const zoomToFitElements = ( | ||||
|       ? getCommonBounds(selectedElements) | ||||
|       : getCommonBounds(nonDeletedElements); | ||||
|  | ||||
|   const newZoom = { | ||||
|     value: zoomValueToFitBoundsOnViewport(commonBounds, { | ||||
|       width: appState.width, | ||||
|       height: appState.height, | ||||
|     }), | ||||
|   }; | ||||
|   const zoomValue = zoomValueToFitBoundsOnViewport(commonBounds, { | ||||
|     width: appState.width, | ||||
|     height: appState.height, | ||||
|   }); | ||||
|   const newZoom = getNewZoom(zoomValue, appState.zoom, { | ||||
|     left: appState.offsetLeft, | ||||
|     top: appState.offsetTop, | ||||
|   }); | ||||
|  | ||||
|   const [x1, y1, x2, y2] = commonBounds; | ||||
|   const centerX = (x1 + x2) / 2; | ||||
| @@ -254,7 +244,6 @@ const zoomToFitElements = ( | ||||
|  | ||||
| export const actionZoomToSelected = register({ | ||||
|   name: "zoomToSelection", | ||||
|   trackEvent: { category: "canvas" }, | ||||
|   perform: (elements, appState) => zoomToFitElements(elements, appState, true), | ||||
|   keyTest: (event) => | ||||
|     event.code === CODES.TWO && | ||||
| @@ -265,7 +254,6 @@ export const actionZoomToSelected = register({ | ||||
|  | ||||
| export const actionZoomToFit = register({ | ||||
|   name: "zoomToFit", | ||||
|   trackEvent: { category: "canvas" }, | ||||
|   perform: (elements, appState) => zoomToFitElements(elements, appState, false), | ||||
|   keyTest: (event) => | ||||
|     event.code === CODES.ONE && | ||||
| @@ -276,13 +264,11 @@ export const actionZoomToFit = register({ | ||||
|  | ||||
| export const actionToggleTheme = register({ | ||||
|   name: "toggleTheme", | ||||
|   trackEvent: { category: "canvas" }, | ||||
|   perform: (_, appState, value) => { | ||||
|     return { | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         theme: | ||||
|           value || (appState.theme === THEME.LIGHT ? THEME.DARK : THEME.LIGHT), | ||||
|         theme: value || (appState.theme === "light" ? "dark" : "light"), | ||||
|       }, | ||||
|       commitToHistory: false, | ||||
|     }; | ||||
| @@ -299,42 +285,3 @@ export const actionToggleTheme = register({ | ||||
|   ), | ||||
|   keyTest: (event) => event.altKey && event.shiftKey && event.code === CODES.D, | ||||
| }); | ||||
|  | ||||
| export const actionErase = register({ | ||||
|   name: "eraser", | ||||
|   trackEvent: { category: "toolbar" }, | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         selectedElementIds: {}, | ||||
|         selectedGroupIds: {}, | ||||
|         activeTool: { | ||||
|           ...appState.activeTool, | ||||
|           type: isEraserActive(appState) | ||||
|             ? appState.activeTool.lastActiveToolBeforeEraser ?? "selection" | ||||
|             : "eraser", | ||||
|           lastActiveToolBeforeEraser: | ||||
|             appState.activeTool.type === "eraser" //node throws incorrect type error when using isEraserActive() | ||||
|               ? null | ||||
|               : appState.activeTool.type, | ||||
|         }, | ||||
|       }, | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
|   keyTest: (event) => event.key === KEYS.E, | ||||
|   PanelComponent: ({ elements, appState, updateData, data }) => ( | ||||
|     <ToolButton | ||||
|       type="button" | ||||
|       icon={eraser} | ||||
|       className={clsx("eraser", { active: isEraserActive(appState) })} | ||||
|       title={`${t("toolBar.eraser")}-${getShortcutKey("E")}`} | ||||
|       aria-label={t("toolBar.eraser")} | ||||
|       onClick={() => { | ||||
|         updateData(null); | ||||
|       }} | ||||
|       size={data?.size || "medium"} | ||||
|     ></ToolButton> | ||||
|   ), | ||||
| }); | ||||
|   | ||||
| @@ -1,21 +1,16 @@ | ||||
| import { CODES, KEYS } from "../keys"; | ||||
| import { register } from "./register"; | ||||
| import { | ||||
|   copyTextToSystemClipboard, | ||||
|   copyToClipboard, | ||||
|   probablySupportsClipboardWriteText, | ||||
| } from "../clipboard"; | ||||
| import { copyToClipboard } from "../clipboard"; | ||||
| import { actionDeleteSelected } from "./actionDeleteSelected"; | ||||
| import { getSelectedElements } from "../scene/selection"; | ||||
| import { exportCanvas } from "../data/index"; | ||||
| import { getNonDeletedElements, isTextElement } from "../element"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import { t } from "../i18n"; | ||||
|  | ||||
| export const actionCopy = register({ | ||||
|   name: "copy", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState, _, app) => { | ||||
|     copyToClipboard(getNonDeletedElements(elements), appState, app.files); | ||||
|   perform: (elements, appState) => { | ||||
|     copyToClipboard(getNonDeletedElements(elements), appState); | ||||
|  | ||||
|     return { | ||||
|       commitToHistory: false, | ||||
| @@ -28,10 +23,9 @@ export const actionCopy = register({ | ||||
|  | ||||
| export const actionCut = register({ | ||||
|   name: "cut", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState, data, app) => { | ||||
|     actionCopy.perform(elements, appState, data, app); | ||||
|     return actionDeleteSelected.perform(elements, appState); | ||||
|     return actionDeleteSelected.perform(elements, appState, data, app); | ||||
|   }, | ||||
|   contextItemLabel: "labels.cut", | ||||
|   keyTest: (event) => event[KEYS.CTRL_OR_CMD] && event.code === CODES.X, | ||||
| @@ -39,7 +33,6 @@ export const actionCut = register({ | ||||
|  | ||||
| export const actionCopyAsSvg = register({ | ||||
|   name: "copyAsSvg", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: async (elements, appState, _data, app) => { | ||||
|     if (!app.canvas) { | ||||
|       return { | ||||
| @@ -49,7 +42,6 @@ export const actionCopyAsSvg = register({ | ||||
|     const selectedElements = getSelectedElements( | ||||
|       getNonDeletedElements(elements), | ||||
|       appState, | ||||
|       true, | ||||
|     ); | ||||
|     try { | ||||
|       await exportCanvas( | ||||
| @@ -58,13 +50,12 @@ export const actionCopyAsSvg = register({ | ||||
|           ? selectedElements | ||||
|           : getNonDeletedElements(elements), | ||||
|         appState, | ||||
|         app.files, | ||||
|         appState, | ||||
|       ); | ||||
|       return { | ||||
|         commitToHistory: false, | ||||
|       }; | ||||
|     } catch (error: any) { | ||||
|     } catch (error) { | ||||
|       console.error(error); | ||||
|       return { | ||||
|         appState: { | ||||
| @@ -80,7 +71,6 @@ export const actionCopyAsSvg = register({ | ||||
|  | ||||
| export const actionCopyAsPng = register({ | ||||
|   name: "copyAsPng", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: async (elements, appState, _data, app) => { | ||||
|     if (!app.canvas) { | ||||
|       return { | ||||
| @@ -90,7 +80,6 @@ export const actionCopyAsPng = register({ | ||||
|     const selectedElements = getSelectedElements( | ||||
|       getNonDeletedElements(elements), | ||||
|       appState, | ||||
|       true, | ||||
|     ); | ||||
|     try { | ||||
|       await exportCanvas( | ||||
| @@ -99,7 +88,6 @@ export const actionCopyAsPng = register({ | ||||
|           ? selectedElements | ||||
|           : getNonDeletedElements(elements), | ||||
|         appState, | ||||
|         app.files, | ||||
|         appState, | ||||
|       ); | ||||
|       return { | ||||
| @@ -116,7 +104,7 @@ export const actionCopyAsPng = register({ | ||||
|         }, | ||||
|         commitToHistory: false, | ||||
|       }; | ||||
|     } catch (error: any) { | ||||
|     } catch (error) { | ||||
|       console.error(error); | ||||
|       return { | ||||
|         appState: { | ||||
| @@ -130,35 +118,3 @@ export const actionCopyAsPng = register({ | ||||
|   contextItemLabel: "labels.copyAsPng", | ||||
|   keyTest: (event) => event.code === CODES.C && event.altKey && event.shiftKey, | ||||
| }); | ||||
|  | ||||
| export const copyText = register({ | ||||
|   name: "copyText", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     const selectedElements = getSelectedElements( | ||||
|       getNonDeletedElements(elements), | ||||
|       appState, | ||||
|       true, | ||||
|     ); | ||||
|  | ||||
|     const text = selectedElements | ||||
|       .reduce((acc: string[], element) => { | ||||
|         if (isTextElement(element)) { | ||||
|           acc.push(element.text); | ||||
|         } | ||||
|         return acc; | ||||
|       }, []) | ||||
|       .join("\n\n"); | ||||
|     copyTextToSystemClipboard(text); | ||||
|     return { | ||||
|       commitToHistory: false, | ||||
|     }; | ||||
|   }, | ||||
|   contextItemPredicate: (elements, appState) => { | ||||
|     return ( | ||||
|       probablySupportsClipboardWriteText && | ||||
|       getSelectedElements(elements, appState, true).some(isTextElement) | ||||
|     ); | ||||
|   }, | ||||
|   contextItemLabel: "labels.copyText", | ||||
| }); | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import { isSomeElementSelected } from "../scene"; | ||||
| import { KEYS } from "../keys"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import React from "react"; | ||||
| import { trash } from "../components/icons"; | ||||
| import { t } from "../i18n"; | ||||
| import { register } from "./register"; | ||||
| @@ -11,7 +12,6 @@ import { newElementWith } from "../element/mutateElement"; | ||||
| import { getElementsInGroup } from "../groups"; | ||||
| import { LinearElementEditor } from "../element/linearElementEditor"; | ||||
| import { fixBindingsAfterDeletion } from "../element/binding"; | ||||
| import { isBoundToContainer } from "../element/typeChecks"; | ||||
|  | ||||
| const deleteSelectedElements = ( | ||||
|   elements: readonly ExcalidrawElement[], | ||||
| @@ -22,12 +22,6 @@ const deleteSelectedElements = ( | ||||
|       if (appState.selectedElementIds[el.id]) { | ||||
|         return newElementWith(el, { isDeleted: true }); | ||||
|       } | ||||
|       if ( | ||||
|         isBoundToContainer(el) && | ||||
|         appState.selectedElementIds[el.containerId] | ||||
|       ) { | ||||
|         return newElementWith(el, { isDeleted: true }); | ||||
|       } | ||||
|       return el; | ||||
|     }), | ||||
|     appState: { | ||||
| @@ -58,12 +52,11 @@ const handleGroupEditingState = ( | ||||
|  | ||||
| export const actionDeleteSelected = register({ | ||||
|   name: "deleteSelectedElements", | ||||
|   trackEvent: { category: "element", action: "delete" }, | ||||
|   perform: (elements, appState) => { | ||||
|     if (appState.editingLinearElement) { | ||||
|       const { | ||||
|         elementId, | ||||
|         selectedPointsIndices, | ||||
|         activePointIndex, | ||||
|         startBindingElement, | ||||
|         endBindingElement, | ||||
|       } = appState.editingLinearElement; | ||||
| @@ -73,7 +66,8 @@ export const actionDeleteSelected = register({ | ||||
|       } | ||||
|       if ( | ||||
|         // case: no point selected → delete whole element | ||||
|         selectedPointsIndices == null || | ||||
|         activePointIndex == null || | ||||
|         activePointIndex === -1 || | ||||
|         // case: deleting last remaining point | ||||
|         element.points.length < 2 | ||||
|       ) { | ||||
| @@ -93,17 +87,15 @@ export const actionDeleteSelected = register({ | ||||
|       // We cannot do this inside `movePoint` because it is also called | ||||
|       // when deleting the uncommitted point (which hasn't caused any binding) | ||||
|       const binding = { | ||||
|         startBindingElement: selectedPointsIndices?.includes(0) | ||||
|           ? null | ||||
|           : startBindingElement, | ||||
|         endBindingElement: selectedPointsIndices?.includes( | ||||
|           element.points.length - 1, | ||||
|         ) | ||||
|           ? null | ||||
|           : endBindingElement, | ||||
|         startBindingElement: | ||||
|           activePointIndex === 0 ? null : startBindingElement, | ||||
|         endBindingElement: | ||||
|           activePointIndex === element.points.length - 1 | ||||
|             ? null | ||||
|             : endBindingElement, | ||||
|       }; | ||||
|  | ||||
|       LinearElementEditor.deletePoints(element, selectedPointsIndices); | ||||
|       LinearElementEditor.movePoint(element, activePointIndex, "delete"); | ||||
|  | ||||
|       return { | ||||
|         elements, | ||||
| @@ -112,17 +104,17 @@ export const actionDeleteSelected = register({ | ||||
|           editingLinearElement: { | ||||
|             ...appState.editingLinearElement, | ||||
|             ...binding, | ||||
|             selectedPointsIndices: | ||||
|               selectedPointsIndices?.[0] > 0 | ||||
|                 ? [selectedPointsIndices[0] - 1] | ||||
|                 : [0], | ||||
|             activePointIndex: activePointIndex > 0 ? activePointIndex - 1 : 0, | ||||
|           }, | ||||
|         }, | ||||
|         commitToHistory: true, | ||||
|       }; | ||||
|     } | ||||
|     let { elements: nextElements, appState: nextAppState } = | ||||
|       deleteSelectedElements(elements, appState); | ||||
|  | ||||
|     let { | ||||
|       elements: nextElements, | ||||
|       appState: nextAppState, | ||||
|     } = deleteSelectedElements(elements, appState); | ||||
|     fixBindingsAfterDeletion( | ||||
|       nextElements, | ||||
|       elements.filter(({ id }) => appState.selectedElementIds[id]), | ||||
| @@ -134,7 +126,7 @@ export const actionDeleteSelected = register({ | ||||
|       elements: nextElements, | ||||
|       appState: { | ||||
|         ...nextAppState, | ||||
|         activeTool: { ...appState.activeTool, type: "selection" }, | ||||
|         elementType: "selection", | ||||
|         multiElement: null, | ||||
|       }, | ||||
|       commitToHistory: isSomeElementSelected( | ||||
|   | ||||
| @@ -1,16 +1,17 @@ | ||||
| import React from "react"; | ||||
| import { | ||||
|   DistributeHorizontallyIcon, | ||||
|   DistributeVerticallyIcon, | ||||
| } from "../components/icons"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import { distributeElements, Distribution } from "../disitrubte"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import { getElementMap, getNonDeletedElements } from "../element"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { t } from "../i18n"; | ||||
| import { CODES, KEYS } from "../keys"; | ||||
| import { CODES } from "../keys"; | ||||
| import { getSelectedElements, isSomeElementSelected } from "../scene"; | ||||
| import { AppState } from "../types"; | ||||
| import { arrayToMap, getShortcutKey } from "../utils"; | ||||
| import { getShortcutKey } from "../utils"; | ||||
| import { register } from "./register"; | ||||
|  | ||||
| const enableActionGroup = ( | ||||
| @@ -30,16 +31,13 @@ const distributeSelectedElements = ( | ||||
|  | ||||
|   const updatedElements = distributeElements(selectedElements, distribution); | ||||
|  | ||||
|   const updatedElementsMap = arrayToMap(updatedElements); | ||||
|   const updatedElementsMap = getElementMap(updatedElements); | ||||
|  | ||||
|   return elements.map( | ||||
|     (element) => updatedElementsMap.get(element.id) || element, | ||||
|   ); | ||||
|   return elements.map((element) => updatedElementsMap[element.id] || element); | ||||
| }; | ||||
|  | ||||
| export const distributeHorizontally = register({ | ||||
|   name: "distributeHorizontally", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       appState, | ||||
| @@ -50,8 +48,7 @@ export const distributeHorizontally = register({ | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
|   keyTest: (event) => | ||||
|     !event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.H, | ||||
|   keyTest: (event) => event.altKey && event.code === CODES.H, | ||||
|   PanelComponent: ({ elements, appState, updateData }) => ( | ||||
|     <ToolButton | ||||
|       hidden={!enableActionGroup(elements, appState)} | ||||
| @@ -69,7 +66,6 @@ export const distributeHorizontally = register({ | ||||
|  | ||||
| export const distributeVertically = register({ | ||||
|   name: "distributeVertically", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       appState, | ||||
| @@ -80,8 +76,7 @@ export const distributeVertically = register({ | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
|   keyTest: (event) => | ||||
|     !event[KEYS.CTRL_OR_CMD] && event.altKey && event.code === CODES.V, | ||||
|   keyTest: (event) => event.altKey && event.code === CODES.V, | ||||
|   PanelComponent: ({ elements, appState, updateData }) => ( | ||||
|     <ToolButton | ||||
|       hidden={!enableActionGroup(elements, appState)} | ||||
|   | ||||
| @@ -1,13 +1,15 @@ | ||||
| import React from "react"; | ||||
| import { KEYS } from "../keys"; | ||||
| import { register } from "./register"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { duplicateElement, getNonDeletedElements } from "../element"; | ||||
| import { getSelectedElements, isSomeElementSelected } from "../scene"; | ||||
| import { isSomeElementSelected } from "../scene"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import { clone } from "../components/icons"; | ||||
| import { t } from "../i18n"; | ||||
| import { arrayToMap, getShortcutKey } from "../utils"; | ||||
| import { getShortcutKey } from "../utils"; | ||||
| import { LinearElementEditor } from "../element/linearElementEditor"; | ||||
| import { mutateElement } from "../element/mutateElement"; | ||||
| import { | ||||
|   selectGroupsForSelectedElements, | ||||
|   getSelectedGroupForElement, | ||||
| @@ -17,24 +19,41 @@ import { AppState } from "../types"; | ||||
| import { fixBindingsAfterDuplication } from "../element/binding"; | ||||
| import { ActionResult } from "./types"; | ||||
| import { GRID_SIZE } from "../constants"; | ||||
| import { bindTextToShapeAfterDuplication } from "../element/textElement"; | ||||
| import { isBoundToContainer } from "../element/typeChecks"; | ||||
|  | ||||
| export const actionDuplicateSelection = register({ | ||||
|   name: "duplicateSelection", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     // duplicate selected point(s) if editing a line | ||||
|     // duplicate point if selected while editing multi-point element | ||||
|     if (appState.editingLinearElement) { | ||||
|       const ret = LinearElementEditor.duplicateSelectedPoints(appState); | ||||
|  | ||||
|       if (!ret) { | ||||
|       const { activePointIndex, elementId } = appState.editingLinearElement; | ||||
|       const element = LinearElementEditor.getElement(elementId); | ||||
|       if (!element || activePointIndex === null) { | ||||
|         return false; | ||||
|       } | ||||
|  | ||||
|       const { points } = element; | ||||
|       const selectedPoint = points[activePointIndex]; | ||||
|       const nextPoint = points[activePointIndex + 1]; | ||||
|       mutateElement(element, { | ||||
|         points: [ | ||||
|           ...points.slice(0, activePointIndex + 1), | ||||
|           nextPoint | ||||
|             ? [ | ||||
|                 (selectedPoint[0] + nextPoint[0]) / 2, | ||||
|                 (selectedPoint[1] + nextPoint[1]) / 2, | ||||
|               ] | ||||
|             : [selectedPoint[0] + 30, selectedPoint[1] + 30], | ||||
|           ...points.slice(activePointIndex + 1), | ||||
|         ], | ||||
|       }); | ||||
|       return { | ||||
|         appState: { | ||||
|           ...appState, | ||||
|           editingLinearElement: { | ||||
|             ...appState.editingLinearElement, | ||||
|             activePointIndex: activePointIndex + 1, | ||||
|           }, | ||||
|         }, | ||||
|         elements, | ||||
|         appState: ret.appState, | ||||
|         commitToHistory: true, | ||||
|       }; | ||||
|     } | ||||
| @@ -88,12 +107,9 @@ const duplicateElements = ( | ||||
|   const finalElements: ExcalidrawElement[] = []; | ||||
|  | ||||
|   let index = 0; | ||||
|   const selectedElementIds = arrayToMap( | ||||
|     getSelectedElements(elements, appState, true), | ||||
|   ); | ||||
|   while (index < elements.length) { | ||||
|     const element = elements[index]; | ||||
|     if (selectedElementIds.get(element.id)) { | ||||
|     if (appState.selectedElementIds[element.id]) { | ||||
|       if (element.groupIds.length) { | ||||
|         const groupId = getSelectedGroupForElement(appState, element); | ||||
|         // if group selected, duplicate it atomically | ||||
| @@ -115,11 +131,7 @@ const duplicateElements = ( | ||||
|     } | ||||
|     index++; | ||||
|   } | ||||
|   bindTextToShapeAfterDuplication( | ||||
|     finalElements, | ||||
|     oldElements, | ||||
|     oldIdToDuplicatedId, | ||||
|   ); | ||||
|  | ||||
|   fixBindingsAfterDuplication(finalElements, oldElements, oldIdToDuplicatedId); | ||||
|  | ||||
|   return { | ||||
| @@ -129,9 +141,7 @@ const duplicateElements = ( | ||||
|         ...appState, | ||||
|         selectedGroupIds: {}, | ||||
|         selectedElementIds: newElements.reduce((acc, element) => { | ||||
|           if (!isBoundToContainer(element)) { | ||||
|             acc[element.id] = true; | ||||
|           } | ||||
|           acc[element.id] = true; | ||||
|           return acc; | ||||
|         }, {} as any), | ||||
|       }, | ||||
|   | ||||
| @@ -1,29 +1,23 @@ | ||||
| import { load, questionCircle, saveAs } from "../components/icons"; | ||||
| import React from "react"; | ||||
| import { trackEvent } from "../analytics"; | ||||
| import { load, questionCircle, save, saveAs } from "../components/icons"; | ||||
| import { ProjectName } from "../components/ProjectName"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import "../components/ToolIcon.scss"; | ||||
| import { Tooltip } from "../components/Tooltip"; | ||||
| import { DarkModeToggle } from "../components/DarkModeToggle"; | ||||
| import { DarkModeToggle, Appearence } from "../components/DarkModeToggle"; | ||||
| import { loadFromJSON, saveAsJSON } from "../data"; | ||||
| import { resaveAsImageWithScene } from "../data/resave"; | ||||
| import { t } from "../i18n"; | ||||
| import { useDeviceType } from "../components/App"; | ||||
| import { useIsMobile } from "../components/App"; | ||||
| import { KEYS } from "../keys"; | ||||
| import { register } from "./register"; | ||||
| import { supported as fsSupported } from "browser-fs-access"; | ||||
| import { CheckboxItem } from "../components/CheckboxItem"; | ||||
| import { getExportSize } from "../scene/export"; | ||||
| import { DEFAULT_EXPORT_PADDING, EXPORT_SCALES, THEME } from "../constants"; | ||||
| import { getSelectedElements, isSomeElementSelected } from "../scene"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import { ActiveFile } from "../components/ActiveFile"; | ||||
| import { isImageFileHandle } from "../data/blob"; | ||||
| import { nativeFileSystemSupported } from "../data/filesystem"; | ||||
| import { Theme } from "../element/types"; | ||||
|  | ||||
| export const actionChangeProjectName = register({ | ||||
|   name: "changeProjectName", | ||||
|   trackEvent: false, | ||||
|   perform: (_elements, appState, value) => { | ||||
|     trackEvent("change", "title"); | ||||
|     return { appState: { ...appState, name: value }, commitToHistory: false }; | ||||
|   }, | ||||
|   PanelComponent: ({ appState, updateData, appProps }) => ( | ||||
| @@ -38,58 +32,8 @@ export const actionChangeProjectName = register({ | ||||
|   ), | ||||
| }); | ||||
|  | ||||
| export const actionChangeExportScale = register({ | ||||
|   name: "changeExportScale", | ||||
|   trackEvent: { category: "export", action: "scale" }, | ||||
|   perform: (_elements, appState, value) => { | ||||
|     return { | ||||
|       appState: { ...appState, exportScale: value }, | ||||
|       commitToHistory: false, | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ elements: allElements, appState, updateData }) => { | ||||
|     const elements = getNonDeletedElements(allElements); | ||||
|     const exportSelected = isSomeElementSelected(elements, appState); | ||||
|     const exportedElements = exportSelected | ||||
|       ? getSelectedElements(elements, appState) | ||||
|       : elements; | ||||
|  | ||||
|     return ( | ||||
|       <> | ||||
|         {EXPORT_SCALES.map((s) => { | ||||
|           const [width, height] = getExportSize( | ||||
|             exportedElements, | ||||
|             DEFAULT_EXPORT_PADDING, | ||||
|             s, | ||||
|           ); | ||||
|  | ||||
|           const scaleButtonTitle = `${t( | ||||
|             "buttons.scale", | ||||
|           )} ${s}x (${width}x${height})`; | ||||
|  | ||||
|           return ( | ||||
|             <ToolButton | ||||
|               key={s} | ||||
|               size="small" | ||||
|               type="radio" | ||||
|               icon={`${s}x`} | ||||
|               name="export-canvas-scale" | ||||
|               title={scaleButtonTitle} | ||||
|               aria-label={scaleButtonTitle} | ||||
|               id="export-canvas-scale" | ||||
|               checked={s === appState.exportScale} | ||||
|               onChange={() => updateData(s)} | ||||
|             /> | ||||
|           ); | ||||
|         })} | ||||
|       </> | ||||
|     ); | ||||
|   }, | ||||
| }); | ||||
|  | ||||
| export const actionChangeExportBackground = register({ | ||||
|   name: "changeExportBackground", | ||||
|   trackEvent: { category: "export", action: "toggleBackground" }, | ||||
|   perform: (_elements, appState, value) => { | ||||
|     return { | ||||
|       appState: { ...appState, exportBackground: value }, | ||||
| @@ -108,7 +52,6 @@ export const actionChangeExportBackground = register({ | ||||
|  | ||||
| export const actionChangeExportEmbedScene = register({ | ||||
|   name: "changeExportEmbedScene", | ||||
|   trackEvent: { category: "export", action: "embedScene" }, | ||||
|   perform: (_elements, appState, value) => { | ||||
|     return { | ||||
|       appState: { ...appState, exportEmbedScene: value }, | ||||
| @@ -122,30 +65,43 @@ export const actionChangeExportEmbedScene = register({ | ||||
|     > | ||||
|       {t("labels.exportEmbedScene")} | ||||
|       <Tooltip label={t("labels.exportEmbedScene_details")} long={true}> | ||||
|         <div className="excalidraw-tooltip-icon">{questionCircle}</div> | ||||
|         <div className="Tooltip-icon">{questionCircle}</div> | ||||
|       </Tooltip> | ||||
|     </CheckboxItem> | ||||
|   ), | ||||
| }); | ||||
|  | ||||
| export const actionSaveToActiveFile = register({ | ||||
|   name: "saveToActiveFile", | ||||
|   trackEvent: { category: "export" }, | ||||
|   perform: async (elements, appState, value, app) => { | ||||
| export const actionChangeShouldAddWatermark = register({ | ||||
|   name: "changeShouldAddWatermark", | ||||
|   perform: (_elements, appState, value) => { | ||||
|     return { | ||||
|       appState: { ...appState, shouldAddWatermark: value }, | ||||
|       commitToHistory: false, | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ appState, updateData }) => ( | ||||
|     <CheckboxItem | ||||
|       checked={appState.shouldAddWatermark} | ||||
|       onChange={(checked) => updateData(checked)} | ||||
|     > | ||||
|       {t("labels.addWatermark")} | ||||
|     </CheckboxItem> | ||||
|   ), | ||||
| }); | ||||
|  | ||||
| export const actionSaveScene = register({ | ||||
|   name: "saveScene", | ||||
|   perform: async (elements, appState, value) => { | ||||
|     const fileHandleExists = !!appState.fileHandle; | ||||
|  | ||||
|     try { | ||||
|       const { fileHandle } = isImageFileHandle(appState.fileHandle) | ||||
|         ? await resaveAsImageWithScene(elements, appState, app.files) | ||||
|         : await saveAsJSON(elements, appState, app.files); | ||||
|  | ||||
|       const { fileHandle } = await saveAsJSON(elements, appState); | ||||
|       return { | ||||
|         commitToHistory: false, | ||||
|         appState: { | ||||
|           ...appState, | ||||
|           fileHandle, | ||||
|           toastMessage: fileHandleExists | ||||
|             ? fileHandle?.name | ||||
|             ? fileHandle.name | ||||
|               ? t("toast.fileSavedToFilename").replace( | ||||
|                   "{filename}", | ||||
|                   `"${fileHandle.name}"`, | ||||
| @@ -154,44 +110,39 @@ export const actionSaveToActiveFile = register({ | ||||
|             : null, | ||||
|         }, | ||||
|       }; | ||||
|     } catch (error: any) { | ||||
|     } catch (error) { | ||||
|       if (error?.name !== "AbortError") { | ||||
|         console.error(error); | ||||
|       } else { | ||||
|         console.warn(error); | ||||
|       } | ||||
|       return { commitToHistory: false }; | ||||
|     } | ||||
|   }, | ||||
|   keyTest: (event) => | ||||
|     event.key === KEYS.S && event[KEYS.CTRL_OR_CMD] && !event.shiftKey, | ||||
|   PanelComponent: ({ updateData, appState }) => ( | ||||
|     <ActiveFile | ||||
|       onSave={() => updateData(null)} | ||||
|       fileName={appState.fileHandle?.name} | ||||
|   PanelComponent: ({ updateData }) => ( | ||||
|     <ToolButton | ||||
|       type="icon" | ||||
|       icon={save} | ||||
|       title={t("buttons.save")} | ||||
|       aria-label={t("buttons.save")} | ||||
|       onClick={() => updateData(null)} | ||||
|       data-testid="save-button" | ||||
|     /> | ||||
|   ), | ||||
| }); | ||||
|  | ||||
| export const actionSaveFileToDisk = register({ | ||||
|   name: "saveFileToDisk", | ||||
|   trackEvent: { category: "export" }, | ||||
|   perform: async (elements, appState, value, app) => { | ||||
| export const actionSaveAsScene = register({ | ||||
|   name: "saveAsScene", | ||||
|   perform: async (elements, appState, value) => { | ||||
|     try { | ||||
|       const { fileHandle } = await saveAsJSON( | ||||
|         elements, | ||||
|         { | ||||
|           ...appState, | ||||
|           fileHandle: null, | ||||
|         }, | ||||
|         app.files, | ||||
|       ); | ||||
|       const { fileHandle } = await saveAsJSON(elements, { | ||||
|         ...appState, | ||||
|         fileHandle: null, | ||||
|       }); | ||||
|       return { commitToHistory: false, appState: { ...appState, fileHandle } }; | ||||
|     } catch (error: any) { | ||||
|     } catch (error) { | ||||
|       if (error?.name !== "AbortError") { | ||||
|         console.error(error); | ||||
|       } else { | ||||
|         console.warn(error); | ||||
|       } | ||||
|       return { commitToHistory: false }; | ||||
|     } | ||||
| @@ -204,8 +155,8 @@ export const actionSaveFileToDisk = register({ | ||||
|       icon={saveAs} | ||||
|       title={t("buttons.saveAs")} | ||||
|       aria-label={t("buttons.saveAs")} | ||||
|       showAriaLabel={useDeviceType().isMobile} | ||||
|       hidden={!nativeFileSystemSupported} | ||||
|       showAriaLabel={useIsMobile()} | ||||
|       hidden={!fsSupported} | ||||
|       onClick={() => updateData(null)} | ||||
|       data-testid="save-as-button" | ||||
|     /> | ||||
| @@ -214,29 +165,24 @@ export const actionSaveFileToDisk = register({ | ||||
|  | ||||
| export const actionLoadScene = register({ | ||||
|   name: "loadScene", | ||||
|   trackEvent: { category: "export" }, | ||||
|   perform: async (elements, appState, _, app) => { | ||||
|   perform: async (elements, appState) => { | ||||
|     try { | ||||
|       const { | ||||
|         elements: loadedElements, | ||||
|         appState: loadedAppState, | ||||
|         files, | ||||
|       } = await loadFromJSON(appState, elements); | ||||
|       } = await loadFromJSON(appState); | ||||
|       return { | ||||
|         elements: loadedElements, | ||||
|         appState: loadedAppState, | ||||
|         files, | ||||
|         commitToHistory: true, | ||||
|       }; | ||||
|     } catch (error: any) { | ||||
|     } catch (error) { | ||||
|       if (error?.name === "AbortError") { | ||||
|         console.warn(error); | ||||
|         return false; | ||||
|       } | ||||
|       return { | ||||
|         elements, | ||||
|         appState: { ...appState, errorMessage: error.message }, | ||||
|         files: app.files, | ||||
|         commitToHistory: false, | ||||
|       }; | ||||
|     } | ||||
| @@ -248,7 +194,7 @@ export const actionLoadScene = register({ | ||||
|       icon={load} | ||||
|       title={t("buttons.load")} | ||||
|       aria-label={t("buttons.load")} | ||||
|       showAriaLabel={useDeviceType().isMobile} | ||||
|       showAriaLabel={useIsMobile()} | ||||
|       onClick={updateData} | ||||
|       data-testid="load-button" | ||||
|     /> | ||||
| @@ -257,7 +203,6 @@ export const actionLoadScene = register({ | ||||
|  | ||||
| export const actionExportWithDarkMode = register({ | ||||
|   name: "exportWithDarkMode", | ||||
|   trackEvent: { category: "export", action: "toggleTheme" }, | ||||
|   perform: (_elements, appState, value) => { | ||||
|     return { | ||||
|       appState: { ...appState, exportWithDarkMode: value }, | ||||
| @@ -274,9 +219,9 @@ export const actionExportWithDarkMode = register({ | ||||
|       }} | ||||
|     > | ||||
|       <DarkModeToggle | ||||
|         value={appState.exportWithDarkMode ? THEME.DARK : THEME.LIGHT} | ||||
|         onChange={(theme: Theme) => { | ||||
|           updateData(theme === THEME.DARK); | ||||
|         value={appState.exportWithDarkMode ? "dark" : "light"} | ||||
|         onChange={(theme: Appearence) => { | ||||
|           updateData(theme === "dark"); | ||||
|         }} | ||||
|         title={t("labels.toggleExportColorScheme")} | ||||
|       /> | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import { KEYS } from "../keys"; | ||||
| import { isInvisiblySmallElement } from "../element"; | ||||
| import { resetCursor } from "../utils"; | ||||
| import React from "react"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import { done } from "../components/icons"; | ||||
| import { t } from "../i18n"; | ||||
| @@ -17,11 +18,13 @@ import { isBindingElement } from "../element/typeChecks"; | ||||
|  | ||||
| export const actionFinalize = register({ | ||||
|   name: "finalize", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, _, { canvas, focusContainer }) => { | ||||
|     if (appState.editingLinearElement) { | ||||
|       const { elementId, startBindingElement, endBindingElement } = | ||||
|         appState.editingLinearElement; | ||||
|       const { | ||||
|         elementId, | ||||
|         startBindingElement, | ||||
|         endBindingElement, | ||||
|       } = appState.editingLinearElement; | ||||
|       const element = LinearElementEditor.getElement(elementId); | ||||
|  | ||||
|       if (element) { | ||||
| @@ -39,7 +42,6 @@ export const actionFinalize = register({ | ||||
|               : undefined, | ||||
|           appState: { | ||||
|             ...appState, | ||||
|             cursorButton: "up", | ||||
|             editingLinearElement: null, | ||||
|           }, | ||||
|           commitToHistory: true, | ||||
| @@ -48,11 +50,6 @@ export const actionFinalize = register({ | ||||
|     } | ||||
|  | ||||
|     let newElements = elements; | ||||
|  | ||||
|     if (appState.pendingImageElement) { | ||||
|       mutateElement(appState.pendingImageElement, { isDeleted: true }, false); | ||||
|     } | ||||
|  | ||||
|     if (window.document.activeElement instanceof HTMLElement) { | ||||
|       focusContainer(); | ||||
|     } | ||||
| @@ -121,17 +118,13 @@ export const actionFinalize = register({ | ||||
|         ); | ||||
|       } | ||||
|  | ||||
|       if ( | ||||
|         !appState.activeTool.locked && | ||||
|         appState.activeTool.type !== "freedraw" | ||||
|       ) { | ||||
|       if (!appState.elementLocked && appState.elementType !== "freedraw") { | ||||
|         appState.selectedElementIds[multiPointElement.id] = true; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if ( | ||||
|       (!appState.activeTool.locked && | ||||
|         appState.activeTool.type !== "freedraw") || | ||||
|       (!appState.elementLocked && appState.elementType !== "freedraw") || | ||||
|       !multiPointElement | ||||
|     ) { | ||||
|       resetCursor(canvas); | ||||
| @@ -141,20 +134,11 @@ export const actionFinalize = register({ | ||||
|       elements: newElements, | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         cursorButton: "up", | ||||
|         activeTool: | ||||
|           (appState.activeTool.locked || | ||||
|             appState.activeTool.type === "freedraw") && | ||||
|         elementType: | ||||
|           (appState.elementLocked || appState.elementType === "freedraw") && | ||||
|           multiPointElement | ||||
|             ? appState.activeTool | ||||
|             : { | ||||
|                 ...appState.activeTool, | ||||
|                 type: | ||||
|                   appState.activeTool.type === "eraser" && | ||||
|                   appState.activeTool.lastActiveToolBeforeEraser | ||||
|                     ? appState.activeTool.lastActiveToolBeforeEraser | ||||
|                     : "selection", | ||||
|               }, | ||||
|             ? appState.elementType | ||||
|             : "selection", | ||||
|         draggingElement: null, | ||||
|         multiElement: null, | ||||
|         editingElement: null, | ||||
| @@ -162,16 +146,15 @@ export const actionFinalize = register({ | ||||
|         suggestedBindings: [], | ||||
|         selectedElementIds: | ||||
|           multiPointElement && | ||||
|           !appState.activeTool.locked && | ||||
|           appState.activeTool.type !== "freedraw" | ||||
|           !appState.elementLocked && | ||||
|           appState.elementType !== "freedraw" | ||||
|             ? { | ||||
|                 ...appState.selectedElementIds, | ||||
|                 [multiPointElement.id]: true, | ||||
|               } | ||||
|             : appState.selectedElementIds, | ||||
|         pendingImageElement: null, | ||||
|       }, | ||||
|       commitToHistory: appState.activeTool.type === "freedraw", | ||||
|       commitToHistory: appState.elementType === "freedraw", | ||||
|     }; | ||||
|   }, | ||||
|   keyTest: (event, appState) => | ||||
| @@ -180,7 +163,7 @@ export const actionFinalize = register({ | ||||
|         (!appState.draggingElement && appState.multiElement === null))) || | ||||
|     ((event.key === KEYS.ESCAPE || event.key === KEYS.ENTER) && | ||||
|       appState.multiElement !== null), | ||||
|   PanelComponent: ({ appState, updateData, data }) => ( | ||||
|   PanelComponent: ({ appState, updateData }) => ( | ||||
|     <ToolButton | ||||
|       type="button" | ||||
|       icon={done} | ||||
| @@ -188,7 +171,6 @@ export const actionFinalize = register({ | ||||
|       aria-label={t("buttons.done")} | ||||
|       onClick={updateData} | ||||
|       visible={appState.multiElement != null} | ||||
|       size={data?.size || "medium"} | ||||
|     /> | ||||
|   ), | ||||
| }); | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import { register } from "./register"; | ||||
| import { getSelectedElements } from "../scene"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import { getElementMap, getNonDeletedElements } from "../element"; | ||||
| import { mutateElement } from "../element/mutateElement"; | ||||
| import { ExcalidrawElement, NonDeleted } from "../element/types"; | ||||
| import { normalizeAngle, resizeSingleElement } from "../element/resizeElements"; | ||||
| @@ -9,7 +9,6 @@ import { getTransformHandles } from "../element/transformHandles"; | ||||
| import { isFreeDrawElement, isLinearElement } from "../element/typeChecks"; | ||||
| import { updateBoundElements } from "../element/binding"; | ||||
| import { LinearElementEditor } from "../element/linearElementEditor"; | ||||
| import { arrayToMap } from "../utils"; | ||||
|  | ||||
| const enableActionFlipHorizontal = ( | ||||
|   elements: readonly ExcalidrawElement[], | ||||
| @@ -35,7 +34,6 @@ const enableActionFlipVertical = ( | ||||
|  | ||||
| export const actionFlipHorizontal = register({ | ||||
|   name: "flipHorizontal", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       elements: flipSelectedElements(elements, appState, "horizontal"), | ||||
| @@ -51,7 +49,6 @@ export const actionFlipHorizontal = register({ | ||||
|  | ||||
| export const actionFlipVertical = register({ | ||||
|   name: "flipVertical", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       elements: flipSelectedElements(elements, appState, "vertical"), | ||||
| @@ -86,11 +83,9 @@ const flipSelectedElements = ( | ||||
|     flipDirection, | ||||
|   ); | ||||
|  | ||||
|   const updatedElementsMap = arrayToMap(updatedElements); | ||||
|   const updatedElementsMap = getElementMap(updatedElements); | ||||
|  | ||||
|   return elements.map( | ||||
|     (element) => updatedElementsMap.get(element.id) || element, | ||||
|   ); | ||||
|   return elements.map((element) => updatedElementsMap[element.id] || element); | ||||
| }; | ||||
|  | ||||
| const flipElements = ( | ||||
| @@ -98,13 +93,13 @@ const flipElements = ( | ||||
|   appState: AppState, | ||||
|   flipDirection: "horizontal" | "vertical", | ||||
| ): ExcalidrawElement[] => { | ||||
|   elements.forEach((element) => { | ||||
|     flipElement(element, appState); | ||||
|   for (let i = 0; i < elements.length; i++) { | ||||
|     flipElement(elements[i], appState); | ||||
|     // If vertical flip, rotate an extra 180 | ||||
|     if (flipDirection === "vertical") { | ||||
|       rotateElement(element, Math.PI); | ||||
|       rotateElement(elements[i], Math.PI); | ||||
|     } | ||||
|   }); | ||||
|   } | ||||
|   return elements; | ||||
| }; | ||||
|  | ||||
| @@ -147,9 +142,10 @@ const flipElement = ( | ||||
|   } | ||||
|  | ||||
|   if (isLinearElement(element)) { | ||||
|     for (let index = 1; index < element.points.length; index++) { | ||||
|       LinearElementEditor.movePoints(element, [ | ||||
|         { index, point: [-element.points[index][0], element.points[index][1]] }, | ||||
|     for (let i = 1; i < element.points.length; i++) { | ||||
|       LinearElementEditor.movePoint(element, i, [ | ||||
|         -element.points[i][0], | ||||
|         element.points[i][1], | ||||
|       ]); | ||||
|     } | ||||
|     LinearElementEditor.normalizePoints(element); | ||||
| @@ -157,7 +153,7 @@ const flipElement = ( | ||||
|     // calculate new x-coord for transformation | ||||
|     newNCoordsX = usingNWHandle ? element.x + 2 * width : element.x - 2 * width; | ||||
|     resizeSingleElement( | ||||
|       new Map().set(element.id, element), | ||||
|       element, | ||||
|       true, | ||||
|       element, | ||||
|       usingNWHandle ? "nw" : "ne", | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import React from "react"; | ||||
| import { CODES, KEYS } from "../keys"; | ||||
| import { t } from "../i18n"; | ||||
| import { arrayToMap, getShortcutKey } from "../utils"; | ||||
| import { getShortcutKey } from "../utils"; | ||||
| import { register } from "./register"; | ||||
| import { UngroupIcon, GroupIcon } from "../components/icons"; | ||||
| import { newElementWith } from "../element/mutateElement"; | ||||
| @@ -17,9 +18,8 @@ import { | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import { randomId } from "../random"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import { ExcalidrawElement, ExcalidrawTextElement } from "../element/types"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { AppState } from "../types"; | ||||
| import { isBoundToContainer } from "../element/typeChecks"; | ||||
|  | ||||
| const allElementsInSameGroup = (elements: readonly ExcalidrawElement[]) => { | ||||
|   if (elements.length >= 2) { | ||||
| @@ -45,7 +45,6 @@ const enableActionGroup = ( | ||||
|   const selectedElements = getSelectedElements( | ||||
|     getNonDeletedElements(elements), | ||||
|     appState, | ||||
|     true, | ||||
|   ); | ||||
|   return ( | ||||
|     selectedElements.length >= 2 && !allElementsInSameGroup(selectedElements) | ||||
| @@ -54,12 +53,10 @@ const enableActionGroup = ( | ||||
|  | ||||
| export const actionGroup = register({ | ||||
|   name: "group", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     const selectedElements = getSelectedElements( | ||||
|       getNonDeletedElements(elements), | ||||
|       appState, | ||||
|       true, | ||||
|     ); | ||||
|     if (selectedElements.length < 2) { | ||||
|       // nothing to group | ||||
| @@ -87,9 +84,8 @@ export const actionGroup = register({ | ||||
|       } | ||||
|     } | ||||
|     const newGroupId = randomId(); | ||||
|     const selectElementIds = arrayToMap(selectedElements); | ||||
|     const updatedElements = elements.map((element) => { | ||||
|       if (!selectElementIds.get(element.id)) { | ||||
|       if (!appState.selectedElementIds[element.id]) { | ||||
|         return element; | ||||
|       } | ||||
|       return newElementWith(element, { | ||||
| @@ -104,8 +100,9 @@ export const actionGroup = register({ | ||||
|     // to the z order of the highest element in the layer stack | ||||
|     const elementsInGroup = getElementsInGroup(updatedElements, newGroupId); | ||||
|     const lastElementInGroup = elementsInGroup[elementsInGroup.length - 1]; | ||||
|     const lastGroupElementIndex = | ||||
|       updatedElements.lastIndexOf(lastElementInGroup); | ||||
|     const lastGroupElementIndex = updatedElements.lastIndexOf( | ||||
|       lastElementInGroup, | ||||
|     ); | ||||
|     const elementsAfterGroup = updatedElements.slice(lastGroupElementIndex + 1); | ||||
|     const elementsBeforeGroup = updatedElements | ||||
|       .slice(0, lastGroupElementIndex) | ||||
| @@ -148,18 +145,12 @@ export const actionGroup = register({ | ||||
|  | ||||
| export const actionUngroup = register({ | ||||
|   name: "ungroup", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     const groupIds = getSelectedGroupIds(appState); | ||||
|     if (groupIds.length === 0) { | ||||
|       return { appState, elements, commitToHistory: false }; | ||||
|     } | ||||
|  | ||||
|     const boundTextElementIds: ExcalidrawTextElement["id"][] = []; | ||||
|     const nextElements = elements.map((element) => { | ||||
|       if (isBoundToContainer(element)) { | ||||
|         boundTextElementIds.push(element.id); | ||||
|       } | ||||
|       const nextGroupIds = removeFromSelectedGroups( | ||||
|         element.groupIds, | ||||
|         appState.selectedGroupIds, | ||||
| @@ -171,19 +162,11 @@ export const actionUngroup = register({ | ||||
|         groupIds: nextGroupIds, | ||||
|       }); | ||||
|     }); | ||||
|  | ||||
|     const updateAppState = selectGroupsForSelectedElements( | ||||
|       { ...appState, selectedGroupIds: {} }, | ||||
|       getNonDeletedElements(nextElements), | ||||
|     ); | ||||
|  | ||||
|     // remove binded text elements from selection | ||||
|     boundTextElementIds.forEach( | ||||
|       (id) => (updateAppState.selectedElementIds[id] = false), | ||||
|     ); | ||||
|     return { | ||||
|       appState: updateAppState, | ||||
|  | ||||
|       appState: selectGroupsForSelectedElements( | ||||
|         { ...appState, selectedGroupIds: {} }, | ||||
|         getNonDeletedElements(nextElements), | ||||
|       ), | ||||
|       elements: nextElements, | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import { Action, ActionResult } from "./types"; | ||||
| import React from "react"; | ||||
| import { undo, redo } from "../components/icons"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import { t } from "../i18n"; | ||||
| @@ -6,9 +7,9 @@ import History, { HistoryEntry } from "../history"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { AppState } from "../types"; | ||||
| import { isWindows, KEYS } from "../keys"; | ||||
| import { getElementMap } from "../element"; | ||||
| import { newElementWith } from "../element/mutateElement"; | ||||
| import { fixBindingsAfterDeletion } from "../element/binding"; | ||||
| import { arrayToMap } from "../utils"; | ||||
|  | ||||
| const writeData = ( | ||||
|   prevElements: readonly ExcalidrawElement[], | ||||
| @@ -27,17 +28,17 @@ const writeData = ( | ||||
|       return { commitToHistory }; | ||||
|     } | ||||
|  | ||||
|     const prevElementMap = arrayToMap(prevElements); | ||||
|     const prevElementMap = getElementMap(prevElements); | ||||
|     const nextElements = data.elements; | ||||
|     const nextElementMap = arrayToMap(nextElements); | ||||
|     const nextElementMap = getElementMap(nextElements); | ||||
|  | ||||
|     const deletedElements = prevElements.filter( | ||||
|       (prevElement) => !nextElementMap.has(prevElement.id), | ||||
|       (prevElement) => !nextElementMap.hasOwnProperty(prevElement.id), | ||||
|     ); | ||||
|     const elements = nextElements | ||||
|       .map((nextElement) => | ||||
|         newElementWith( | ||||
|           prevElementMap.get(nextElement.id) || nextElement, | ||||
|           prevElementMap[nextElement.id] || nextElement, | ||||
|           nextElement, | ||||
|         ), | ||||
|       ) | ||||
| @@ -62,20 +63,18 @@ type ActionCreator = (history: History) => Action; | ||||
|  | ||||
| export const createUndoAction: ActionCreator = (history) => ({ | ||||
|   name: "undo", | ||||
|   trackEvent: { category: "history" }, | ||||
|   perform: (elements, appState) => | ||||
|     writeData(elements, appState, () => history.undoOnce()), | ||||
|   keyTest: (event) => | ||||
|     event[KEYS.CTRL_OR_CMD] && | ||||
|     event.key.toLowerCase() === KEYS.Z && | ||||
|     !event.shiftKey, | ||||
|   PanelComponent: ({ updateData, data }) => ( | ||||
|   PanelComponent: ({ updateData }) => ( | ||||
|     <ToolButton | ||||
|       type="button" | ||||
|       icon={undo} | ||||
|       aria-label={t("buttons.undo")} | ||||
|       onClick={updateData} | ||||
|       size={data?.size || "medium"} | ||||
|     /> | ||||
|   ), | ||||
|   commitToHistory: () => false, | ||||
| @@ -83,7 +82,6 @@ export const createUndoAction: ActionCreator = (history) => ({ | ||||
|  | ||||
| export const createRedoAction: ActionCreator = (history) => ({ | ||||
|   name: "redo", | ||||
|   trackEvent: { category: "history" }, | ||||
|   perform: (elements, appState) => | ||||
|     writeData(elements, appState, () => history.redoOnce()), | ||||
|   keyTest: (event) => | ||||
| @@ -91,13 +89,12 @@ export const createRedoAction: ActionCreator = (history) => ({ | ||||
|       event.shiftKey && | ||||
|       event.key.toLowerCase() === KEYS.Z) || | ||||
|     (isWindows && event.ctrlKey && !event.shiftKey && event.key === KEYS.Y), | ||||
|   PanelComponent: ({ updateData, data }) => ( | ||||
|   PanelComponent: ({ updateData }) => ( | ||||
|     <ToolButton | ||||
|       type="button" | ||||
|       icon={redo} | ||||
|       aria-label={t("buttons.redo")} | ||||
|       onClick={updateData} | ||||
|       size={data?.size || "medium"} | ||||
|     /> | ||||
|   ), | ||||
|   commitToHistory: () => false, | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import React from "react"; | ||||
| import { menu, palette } from "../components/icons"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import { t } from "../i18n"; | ||||
| @@ -9,7 +10,6 @@ import { HelpIcon } from "../components/HelpIcon"; | ||||
|  | ||||
| export const actionToggleCanvasMenu = register({ | ||||
|   name: "toggleCanvasMenu", | ||||
|   trackEvent: { category: "menu" }, | ||||
|   perform: (_, appState) => ({ | ||||
|     appState: { | ||||
|       ...appState, | ||||
| @@ -30,7 +30,6 @@ export const actionToggleCanvasMenu = register({ | ||||
|  | ||||
| export const actionToggleEditMenu = register({ | ||||
|   name: "toggleEditMenu", | ||||
|   trackEvent: { category: "menu" }, | ||||
|   perform: (_elements, appState) => ({ | ||||
|     appState: { | ||||
|       ...appState, | ||||
| @@ -55,7 +54,6 @@ export const actionToggleEditMenu = register({ | ||||
|  | ||||
| export const actionFullScreen = register({ | ||||
|   name: "toggleFullScreen", | ||||
|   trackEvent: { category: "canvas", predicate: (appState) => !isFullScreen() }, | ||||
|   perform: () => { | ||||
|     if (!isFullScreen()) { | ||||
|       allowFullScreen(); | ||||
| @@ -72,7 +70,6 @@ export const actionFullScreen = register({ | ||||
|  | ||||
| export const actionShortcuts = register({ | ||||
|   name: "toggleShortcuts", | ||||
|   trackEvent: { category: "menu", action: "toggleHelpDialog" }, | ||||
|   perform: (_elements, appState, _, { focusContainer }) => { | ||||
|     if (appState.showHelpDialog) { | ||||
|       focusContainer(); | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import React from "react"; | ||||
| import { getClientColors, getClientInitials } from "../clients"; | ||||
| import { Avatar } from "../components/Avatar"; | ||||
| import { centerScrollOn } from "../scene/scroll"; | ||||
| @@ -6,7 +7,6 @@ import { register } from "./register"; | ||||
|  | ||||
| export const actionGoToCollaborator = register({ | ||||
|   name: "goToCollaborator", | ||||
|   trackEvent: { category: "collab" }, | ||||
|   perform: (_elements, appState, value) => { | ||||
|     const point = value as Collaborator["pointer"]; | ||||
|     if (!point) { | ||||
| @@ -30,8 +30,8 @@ export const actionGoToCollaborator = register({ | ||||
|       commitToHistory: false, | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ appState, updateData, data }) => { | ||||
|     const clientId: string | undefined = data?.id; | ||||
|   PanelComponent: ({ appState, updateData, id }) => { | ||||
|     const clientId = id; | ||||
|     if (!clientId) { | ||||
|       return null; | ||||
|     } | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import React from "react"; | ||||
| import { AppState } from "../../src/types"; | ||||
| import { ButtonIconSelect } from "../components/ButtonIconSelect"; | ||||
| import { ColorPicker } from "../components/ColorPicker"; | ||||
| @@ -6,20 +7,12 @@ import { | ||||
|   ArrowheadArrowIcon, | ||||
|   ArrowheadBarIcon, | ||||
|   ArrowheadDotIcon, | ||||
|   ArrowheadTriangleIcon, | ||||
|   ArrowheadNoneIcon, | ||||
|   EdgeRoundIcon, | ||||
|   EdgeSharpIcon, | ||||
|   FillCrossHatchIcon, | ||||
|   FillHachureIcon, | ||||
|   FillSolidIcon, | ||||
|   FontFamilyCodeIcon, | ||||
|   FontFamilyHandDrawnIcon, | ||||
|   FontFamilyNormalIcon, | ||||
|   FontSizeExtraLargeIcon, | ||||
|   FontSizeLargeIcon, | ||||
|   FontSizeMediumIcon, | ||||
|   FontSizeSmallIcon, | ||||
|   SloppinessArchitectIcon, | ||||
|   SloppinessArtistIcon, | ||||
|   SloppinessCartoonistIcon, | ||||
| @@ -27,72 +20,52 @@ import { | ||||
|   StrokeStyleDottedIcon, | ||||
|   StrokeStyleSolidIcon, | ||||
|   StrokeWidthIcon, | ||||
|   TextAlignCenterIcon, | ||||
|   FontSizeSmallIcon, | ||||
|   FontSizeMediumIcon, | ||||
|   FontSizeLargeIcon, | ||||
|   FontSizeExtraLargeIcon, | ||||
|   FontFamilyHandDrawnIcon, | ||||
|   FontFamilyNormalIcon, | ||||
|   FontFamilyCodeIcon, | ||||
|   TextAlignLeftIcon, | ||||
|   TextAlignCenterIcon, | ||||
|   TextAlignRightIcon, | ||||
|   TextAlignTopIcon, | ||||
|   TextAlignBottomIcon, | ||||
|   TextAlignMiddleIcon, | ||||
| } from "../components/icons"; | ||||
| import { | ||||
|   DEFAULT_FONT_FAMILY, | ||||
|   DEFAULT_FONT_SIZE, | ||||
|   FONT_FAMILY, | ||||
|   VERTICAL_ALIGN, | ||||
| } from "../constants"; | ||||
| import { DEFAULT_FONT_FAMILY, DEFAULT_FONT_SIZE } from "../constants"; | ||||
| import { | ||||
|   getNonDeletedElements, | ||||
|   isTextElement, | ||||
|   redrawTextBoundingBox, | ||||
| } from "../element"; | ||||
| import { mutateElement, newElementWith } from "../element/mutateElement"; | ||||
| import { | ||||
|   getBoundTextElement, | ||||
|   getContainerElement, | ||||
| } from "../element/textElement"; | ||||
| import { | ||||
|   isBoundToContainer, | ||||
|   isLinearElement, | ||||
|   isLinearElementType, | ||||
| } from "../element/typeChecks"; | ||||
| import { newElementWith } from "../element/mutateElement"; | ||||
| import { isLinearElement, isLinearElementType } from "../element/typeChecks"; | ||||
| import { | ||||
|   Arrowhead, | ||||
|   ExcalidrawElement, | ||||
|   ExcalidrawLinearElement, | ||||
|   ExcalidrawTextElement, | ||||
|   FontFamilyValues, | ||||
|   FontFamily, | ||||
|   TextAlign, | ||||
|   VerticalAlign, | ||||
| } from "../element/types"; | ||||
| import { getLanguage, t } from "../i18n"; | ||||
| import { KEYS } from "../keys"; | ||||
| import { randomInteger } from "../random"; | ||||
| import { | ||||
|   canChangeSharpness, | ||||
|   canHaveArrowheads, | ||||
|   getCommonAttributeOfSelectedElements, | ||||
|   getSelectedElements, | ||||
|   getTargetElements, | ||||
|   isSomeElementSelected, | ||||
| } from "../scene"; | ||||
| import { hasStrokeColor } from "../scene/comparisons"; | ||||
| import { arrayToMap } from "../utils"; | ||||
| import { register } from "./register"; | ||||
|  | ||||
| const FONT_SIZE_RELATIVE_INCREASE_STEP = 0.1; | ||||
|  | ||||
| const changeProperty = ( | ||||
|   elements: readonly ExcalidrawElement[], | ||||
|   appState: AppState, | ||||
|   callback: (element: ExcalidrawElement) => ExcalidrawElement, | ||||
|   includeBoundText = false, | ||||
| ) => { | ||||
|   const selectedElementIds = arrayToMap( | ||||
|     getSelectedElements(elements, appState, includeBoundText), | ||||
|   ); | ||||
|   return elements.map((element) => { | ||||
|     if ( | ||||
|       selectedElementIds.get(element.id) || | ||||
|       appState.selectedElementIds[element.id] || | ||||
|       element.id === appState.editingElement?.id | ||||
|     ) { | ||||
|       return callback(element); | ||||
| @@ -122,100 +95,17 @@ const getFormValue = function <T>( | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| const offsetElementAfterFontResize = ( | ||||
|   prevElement: ExcalidrawTextElement, | ||||
|   nextElement: ExcalidrawTextElement, | ||||
| ) => { | ||||
|   if (isBoundToContainer(nextElement)) { | ||||
|     return nextElement; | ||||
|   } | ||||
|   return mutateElement( | ||||
|     nextElement, | ||||
|     { | ||||
|       x: | ||||
|         prevElement.textAlign === "left" | ||||
|           ? prevElement.x | ||||
|           : prevElement.x + | ||||
|             (prevElement.width - nextElement.width) / | ||||
|               (prevElement.textAlign === "center" ? 2 : 1), | ||||
|       // centering vertically is non-standard, but for Excalidraw I think | ||||
|       // it makes sense | ||||
|       y: prevElement.y + (prevElement.height - nextElement.height) / 2, | ||||
|     }, | ||||
|     false, | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| const changeFontSize = ( | ||||
|   elements: readonly ExcalidrawElement[], | ||||
|   appState: AppState, | ||||
|   getNewFontSize: (element: ExcalidrawTextElement) => number, | ||||
|   fallbackValue?: ExcalidrawTextElement["fontSize"], | ||||
| ) => { | ||||
|   const newFontSizes = new Set<number>(); | ||||
|  | ||||
|   return { | ||||
|     elements: changeProperty( | ||||
|       elements, | ||||
|       appState, | ||||
|       (oldElement) => { | ||||
|         if (isTextElement(oldElement)) { | ||||
|           const newFontSize = getNewFontSize(oldElement); | ||||
|           newFontSizes.add(newFontSize); | ||||
|  | ||||
|           let newElement: ExcalidrawTextElement = newElementWith(oldElement, { | ||||
|             fontSize: newFontSize, | ||||
|           }); | ||||
|           redrawTextBoundingBox(newElement, getContainerElement(oldElement)); | ||||
|  | ||||
|           newElement = offsetElementAfterFontResize(oldElement, newElement); | ||||
|  | ||||
|           return newElement; | ||||
|         } | ||||
|  | ||||
|         return oldElement; | ||||
|       }, | ||||
|       true, | ||||
|     ), | ||||
|     appState: { | ||||
|       ...appState, | ||||
|       // update state only if we've set all select text elements to | ||||
|       // the same font size | ||||
|       currentItemFontSize: | ||||
|         newFontSizes.size === 1 | ||||
|           ? [...newFontSizes][0] | ||||
|           : fallbackValue ?? appState.currentItemFontSize, | ||||
|     }, | ||||
|     commitToHistory: true, | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| // ----------------------------------------------------------------------------- | ||||
|  | ||||
| export const actionChangeStrokeColor = register({ | ||||
|   name: "changeStrokeColor", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return { | ||||
|       ...(value.currentItemStrokeColor && { | ||||
|         elements: changeProperty( | ||||
|           elements, | ||||
|           appState, | ||||
|           (el) => { | ||||
|             return hasStrokeColor(el.type) | ||||
|               ? newElementWith(el, { | ||||
|                   strokeColor: value.currentItemStrokeColor, | ||||
|                 }) | ||||
|               : el; | ||||
|           }, | ||||
|           true, | ||||
|         ), | ||||
|       }), | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         ...value, | ||||
|       }, | ||||
|       commitToHistory: !!value.currentItemStrokeColor, | ||||
|       elements: changeProperty(elements, appState, (el) => | ||||
|         newElementWith(el, { | ||||
|           strokeColor: value, | ||||
|         }), | ||||
|       ), | ||||
|       appState: { ...appState, currentItemStrokeColor: value }, | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ elements, appState, updateData }) => ( | ||||
| @@ -230,13 +120,7 @@ export const actionChangeStrokeColor = register({ | ||||
|           (element) => element.strokeColor, | ||||
|           appState.currentItemStrokeColor, | ||||
|         )} | ||||
|         onChange={(color) => updateData({ currentItemStrokeColor: color })} | ||||
|         isActive={appState.openPopup === "strokeColorPicker"} | ||||
|         setActive={(active) => | ||||
|           updateData({ openPopup: active ? "strokeColorPicker" : null }) | ||||
|         } | ||||
|         elements={elements} | ||||
|         appState={appState} | ||||
|         onChange={updateData} | ||||
|       /> | ||||
|     </> | ||||
|   ), | ||||
| @@ -244,21 +128,15 @@ export const actionChangeStrokeColor = register({ | ||||
|  | ||||
| export const actionChangeBackgroundColor = register({ | ||||
|   name: "changeBackgroundColor", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return { | ||||
|       ...(value.currentItemBackgroundColor && { | ||||
|         elements: changeProperty(elements, appState, (el) => | ||||
|           newElementWith(el, { | ||||
|             backgroundColor: value.currentItemBackgroundColor, | ||||
|           }), | ||||
|         ), | ||||
|       }), | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         ...value, | ||||
|       }, | ||||
|       commitToHistory: !!value.currentItemBackgroundColor, | ||||
|       elements: changeProperty(elements, appState, (el) => | ||||
|         newElementWith(el, { | ||||
|           backgroundColor: value, | ||||
|         }), | ||||
|       ), | ||||
|       appState: { ...appState, currentItemBackgroundColor: value }, | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ elements, appState, updateData }) => ( | ||||
| @@ -273,13 +151,7 @@ export const actionChangeBackgroundColor = register({ | ||||
|           (element) => element.backgroundColor, | ||||
|           appState.currentItemBackgroundColor, | ||||
|         )} | ||||
|         onChange={(color) => updateData({ currentItemBackgroundColor: color })} | ||||
|         isActive={appState.openPopup === "backgroundColorPicker"} | ||||
|         setActive={(active) => | ||||
|           updateData({ openPopup: active ? "backgroundColorPicker" : null }) | ||||
|         } | ||||
|         elements={elements} | ||||
|         appState={appState} | ||||
|         onChange={updateData} | ||||
|       /> | ||||
|     </> | ||||
|   ), | ||||
| @@ -287,7 +159,6 @@ export const actionChangeBackgroundColor = register({ | ||||
|  | ||||
| export const actionChangeFillStyle = register({ | ||||
|   name: "changeFillStyle", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return { | ||||
|       elements: changeProperty(elements, appState, (el) => | ||||
| @@ -337,7 +208,6 @@ export const actionChangeFillStyle = register({ | ||||
|  | ||||
| export const actionChangeStrokeWidth = register({ | ||||
|   name: "changeStrokeWidth", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return { | ||||
|       elements: changeProperty(elements, appState, (el) => | ||||
| @@ -385,7 +255,6 @@ export const actionChangeStrokeWidth = register({ | ||||
|  | ||||
| export const actionChangeSloppiness = register({ | ||||
|   name: "changeSloppiness", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return { | ||||
|       elements: changeProperty(elements, appState, (el) => | ||||
| @@ -434,7 +303,6 @@ export const actionChangeSloppiness = register({ | ||||
|  | ||||
| export const actionChangeStrokeStyle = register({ | ||||
|   name: "changeStrokeStyle", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return { | ||||
|       elements: changeProperty(elements, appState, (el) => | ||||
| @@ -482,7 +350,6 @@ export const actionChangeStrokeStyle = register({ | ||||
|  | ||||
| export const actionChangeOpacity = register({ | ||||
|   name: "changeOpacity", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return { | ||||
|       elements: changeProperty(elements, appState, (el) => | ||||
| @@ -532,9 +399,25 @@ export const actionChangeOpacity = register({ | ||||
|  | ||||
| export const actionChangeFontSize = register({ | ||||
|   name: "changeFontSize", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return changeFontSize(elements, appState, () => value, value); | ||||
|     return { | ||||
|       elements: changeProperty(elements, appState, (el) => { | ||||
|         if (isTextElement(el)) { | ||||
|           const element: ExcalidrawTextElement = newElementWith(el, { | ||||
|             fontSize: value, | ||||
|           }); | ||||
|           redrawTextBoundingBox(element); | ||||
|           return element; | ||||
|         } | ||||
|  | ||||
|         return el; | ||||
|       }), | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         currentItemFontSize: value, | ||||
|       }, | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ elements, appState, updateData }) => ( | ||||
|     <fieldset> | ||||
| @@ -546,40 +429,27 @@ export const actionChangeFontSize = register({ | ||||
|             value: 16, | ||||
|             text: t("labels.small"), | ||||
|             icon: <FontSizeSmallIcon theme={appState.theme} />, | ||||
|             testId: "fontSize-small", | ||||
|           }, | ||||
|           { | ||||
|             value: 20, | ||||
|             text: t("labels.medium"), | ||||
|             icon: <FontSizeMediumIcon theme={appState.theme} />, | ||||
|             testId: "fontSize-medium", | ||||
|           }, | ||||
|           { | ||||
|             value: 28, | ||||
|             text: t("labels.large"), | ||||
|             icon: <FontSizeLargeIcon theme={appState.theme} />, | ||||
|             testId: "fontSize-large", | ||||
|           }, | ||||
|           { | ||||
|             value: 36, | ||||
|             text: t("labels.veryLarge"), | ||||
|             icon: <FontSizeExtraLargeIcon theme={appState.theme} />, | ||||
|             testId: "fontSize-veryLarge", | ||||
|           }, | ||||
|         ]} | ||||
|         value={getFormValue( | ||||
|           elements, | ||||
|           appState, | ||||
|           (element) => { | ||||
|             if (isTextElement(element)) { | ||||
|               return element.fontSize; | ||||
|             } | ||||
|             const boundTextElement = getBoundTextElement(element); | ||||
|             if (boundTextElement) { | ||||
|               return boundTextElement.fontSize; | ||||
|             } | ||||
|             return null; | ||||
|           }, | ||||
|           (element) => isTextElement(element) && element.fontSize, | ||||
|           appState.currentItemFontSize || DEFAULT_FONT_SIZE, | ||||
|         )} | ||||
|         onChange={(value) => updateData(value)} | ||||
| @@ -588,70 +458,21 @@ export const actionChangeFontSize = register({ | ||||
|   ), | ||||
| }); | ||||
|  | ||||
| export const actionDecreaseFontSize = register({ | ||||
|   name: "decreaseFontSize", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return changeFontSize(elements, appState, (element) => | ||||
|       Math.round( | ||||
|         // get previous value before relative increase (doesn't work fully | ||||
|         // due to rounding and float precision issues) | ||||
|         (1 / (1 + FONT_SIZE_RELATIVE_INCREASE_STEP)) * element.fontSize, | ||||
|       ), | ||||
|     ); | ||||
|   }, | ||||
|   keyTest: (event) => { | ||||
|     return ( | ||||
|       event[KEYS.CTRL_OR_CMD] && | ||||
|       event.shiftKey && | ||||
|       // KEYS.COMMA needed for MacOS | ||||
|       (event.key === KEYS.CHEVRON_LEFT || event.key === KEYS.COMMA) | ||||
|     ); | ||||
|   }, | ||||
| }); | ||||
|  | ||||
| export const actionIncreaseFontSize = register({ | ||||
|   name: "increaseFontSize", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return changeFontSize(elements, appState, (element) => | ||||
|       Math.round(element.fontSize * (1 + FONT_SIZE_RELATIVE_INCREASE_STEP)), | ||||
|     ); | ||||
|   }, | ||||
|   keyTest: (event) => { | ||||
|     return ( | ||||
|       event[KEYS.CTRL_OR_CMD] && | ||||
|       event.shiftKey && | ||||
|       // KEYS.PERIOD needed for MacOS | ||||
|       (event.key === KEYS.CHEVRON_RIGHT || event.key === KEYS.PERIOD) | ||||
|     ); | ||||
|   }, | ||||
| }); | ||||
|  | ||||
| export const actionChangeFontFamily = register({ | ||||
|   name: "changeFontFamily", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return { | ||||
|       elements: changeProperty( | ||||
|         elements, | ||||
|         appState, | ||||
|         (oldElement) => { | ||||
|           if (isTextElement(oldElement)) { | ||||
|             const newElement: ExcalidrawTextElement = newElementWith( | ||||
|               oldElement, | ||||
|               { | ||||
|                 fontFamily: value, | ||||
|               }, | ||||
|             ); | ||||
|             redrawTextBoundingBox(newElement, getContainerElement(oldElement)); | ||||
|             return newElement; | ||||
|           } | ||||
|       elements: changeProperty(elements, appState, (el) => { | ||||
|         if (isTextElement(el)) { | ||||
|           const element: ExcalidrawTextElement = newElementWith(el, { | ||||
|             fontFamily: value, | ||||
|           }); | ||||
|           redrawTextBoundingBox(element); | ||||
|           return element; | ||||
|         } | ||||
|  | ||||
|           return oldElement; | ||||
|         }, | ||||
|         true, | ||||
|       ), | ||||
|         return el; | ||||
|       }), | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         currentItemFontFamily: value, | ||||
| @@ -660,23 +481,19 @@ export const actionChangeFontFamily = register({ | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ elements, appState, updateData }) => { | ||||
|     const options: { | ||||
|       value: FontFamilyValues; | ||||
|       text: string; | ||||
|       icon: JSX.Element; | ||||
|     }[] = [ | ||||
|     const options: { value: FontFamily; text: string; icon: JSX.Element }[] = [ | ||||
|       { | ||||
|         value: FONT_FAMILY.Virgil, | ||||
|         value: 1, | ||||
|         text: t("labels.handDrawn"), | ||||
|         icon: <FontFamilyHandDrawnIcon theme={appState.theme} />, | ||||
|       }, | ||||
|       { | ||||
|         value: FONT_FAMILY.Helvetica, | ||||
|         value: 2, | ||||
|         text: t("labels.normal"), | ||||
|         icon: <FontFamilyNormalIcon theme={appState.theme} />, | ||||
|       }, | ||||
|       { | ||||
|         value: FONT_FAMILY.Cascadia, | ||||
|         value: 3, | ||||
|         text: t("labels.code"), | ||||
|         icon: <FontFamilyCodeIcon theme={appState.theme} />, | ||||
|       }, | ||||
| @@ -685,22 +502,13 @@ export const actionChangeFontFamily = register({ | ||||
|     return ( | ||||
|       <fieldset> | ||||
|         <legend>{t("labels.fontFamily")}</legend> | ||||
|         <ButtonIconSelect<FontFamilyValues | false> | ||||
|         <ButtonIconSelect<FontFamily | false> | ||||
|           group="font-family" | ||||
|           options={options} | ||||
|           value={getFormValue( | ||||
|             elements, | ||||
|             appState, | ||||
|             (element) => { | ||||
|               if (isTextElement(element)) { | ||||
|                 return element.fontFamily; | ||||
|               } | ||||
|               const boundTextElement = getBoundTextElement(element); | ||||
|               if (boundTextElement) { | ||||
|                 return boundTextElement.fontFamily; | ||||
|               } | ||||
|               return null; | ||||
|             }, | ||||
|             (element) => isTextElement(element) && element.fontFamily, | ||||
|             appState.currentItemFontFamily || DEFAULT_FONT_FAMILY, | ||||
|           )} | ||||
|           onChange={(value) => updateData(value)} | ||||
| @@ -712,26 +520,19 @@ export const actionChangeFontFamily = register({ | ||||
|  | ||||
| export const actionChangeTextAlign = register({ | ||||
|   name: "changeTextAlign", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return { | ||||
|       elements: changeProperty( | ||||
|         elements, | ||||
|         appState, | ||||
|         (oldElement) => { | ||||
|           if (isTextElement(oldElement)) { | ||||
|             const newElement: ExcalidrawTextElement = newElementWith( | ||||
|               oldElement, | ||||
|               { textAlign: value }, | ||||
|             ); | ||||
|             redrawTextBoundingBox(newElement, getContainerElement(oldElement)); | ||||
|             return newElement; | ||||
|           } | ||||
|       elements: changeProperty(elements, appState, (el) => { | ||||
|         if (isTextElement(el)) { | ||||
|           const element: ExcalidrawTextElement = newElementWith(el, { | ||||
|             textAlign: value, | ||||
|           }); | ||||
|           redrawTextBoundingBox(element); | ||||
|           return element; | ||||
|         } | ||||
|  | ||||
|           return oldElement; | ||||
|         }, | ||||
|         true, | ||||
|       ), | ||||
|         return el; | ||||
|       }), | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         currentItemTextAlign: value, | ||||
| @@ -739,121 +540,42 @@ export const actionChangeTextAlign = register({ | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ elements, appState, updateData }) => { | ||||
|     return ( | ||||
|       <fieldset> | ||||
|         <legend>{t("labels.textAlign")}</legend> | ||||
|         <ButtonIconSelect<TextAlign | false> | ||||
|           group="text-align" | ||||
|           options={[ | ||||
|             { | ||||
|               value: "left", | ||||
|               text: t("labels.left"), | ||||
|               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( | ||||
|             elements, | ||||
|             appState, | ||||
|             (element) => { | ||||
|               if (isTextElement(element)) { | ||||
|                 return element.textAlign; | ||||
|               } | ||||
|               const boundTextElement = getBoundTextElement(element); | ||||
|               if (boundTextElement) { | ||||
|                 return boundTextElement.textAlign; | ||||
|               } | ||||
|               return null; | ||||
|             }, | ||||
|             appState.currentItemTextAlign, | ||||
|           )} | ||||
|           onChange={(value) => updateData(value)} | ||||
|         /> | ||||
|       </fieldset> | ||||
|     ); | ||||
|   }, | ||||
| }); | ||||
| export const actionChangeVerticalAlign = register({ | ||||
|   name: "changeVerticalAlign", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState, value) => { | ||||
|     return { | ||||
|       elements: changeProperty( | ||||
|         elements, | ||||
|         appState, | ||||
|         (oldElement) => { | ||||
|           if (isTextElement(oldElement)) { | ||||
|             const newElement: ExcalidrawTextElement = newElementWith( | ||||
|               oldElement, | ||||
|               { verticalAlign: value }, | ||||
|             ); | ||||
|  | ||||
|             redrawTextBoundingBox(newElement, getContainerElement(oldElement)); | ||||
|             return newElement; | ||||
|           } | ||||
|  | ||||
|           return oldElement; | ||||
|         }, | ||||
|         true, | ||||
|       ), | ||||
|       appState: { | ||||
|         ...appState, | ||||
|       }, | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
|   PanelComponent: ({ elements, appState, updateData }) => { | ||||
|     return ( | ||||
|       <fieldset> | ||||
|         <ButtonIconSelect<VerticalAlign | false> | ||||
|           group="text-align" | ||||
|           options={[ | ||||
|             { | ||||
|               value: VERTICAL_ALIGN.TOP, | ||||
|               text: t("labels.alignTop"), | ||||
|               icon: <TextAlignTopIcon theme={appState.theme} />, | ||||
|             }, | ||||
|             { | ||||
|               value: VERTICAL_ALIGN.MIDDLE, | ||||
|               text: t("labels.centerVertically"), | ||||
|               icon: <TextAlignMiddleIcon theme={appState.theme} />, | ||||
|             }, | ||||
|             { | ||||
|               value: VERTICAL_ALIGN.BOTTOM, | ||||
|               text: t("labels.alignBottom"), | ||||
|               icon: <TextAlignBottomIcon theme={appState.theme} />, | ||||
|             }, | ||||
|           ]} | ||||
|           value={getFormValue(elements, appState, (element) => { | ||||
|             if (isTextElement(element) && element.containerId) { | ||||
|               return element.verticalAlign; | ||||
|             } | ||||
|             const boundTextElement = getBoundTextElement(element); | ||||
|             if (boundTextElement) { | ||||
|               return boundTextElement.verticalAlign; | ||||
|             } | ||||
|             return null; | ||||
|           })} | ||||
|           onChange={(value) => updateData(value)} | ||||
|         /> | ||||
|       </fieldset> | ||||
|     ); | ||||
|   }, | ||||
|   PanelComponent: ({ elements, appState, updateData }) => ( | ||||
|     <fieldset> | ||||
|       <legend>{t("labels.textAlign")}</legend> | ||||
|       <ButtonIconSelect<TextAlign | false> | ||||
|         group="text-align" | ||||
|         options={[ | ||||
|           { | ||||
|             value: "left", | ||||
|             text: t("labels.left"), | ||||
|             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( | ||||
|           elements, | ||||
|           appState, | ||||
|           (element) => isTextElement(element) && element.textAlign, | ||||
|           appState.currentItemTextAlign, | ||||
|         )} | ||||
|         onChange={(value) => updateData(value)} | ||||
|       /> | ||||
|     </fieldset> | ||||
|   ), | ||||
| }); | ||||
|  | ||||
| export const actionChangeSharpness = register({ | ||||
|   name: "changeSharpness", | ||||
|   trackEvent: false, | ||||
|   perform: (elements, appState, value) => { | ||||
|     const targetElements = getTargetElements( | ||||
|       getNonDeletedElements(elements), | ||||
| @@ -861,10 +583,10 @@ export const actionChangeSharpness = register({ | ||||
|     ); | ||||
|     const shouldUpdateForNonLinearElements = targetElements.length | ||||
|       ? targetElements.every((el) => !isLinearElement(el)) | ||||
|       : !isLinearElementType(appState.activeTool.type); | ||||
|       : !isLinearElementType(appState.elementType); | ||||
|     const shouldUpdateForLinearElements = targetElements.length | ||||
|       ? targetElements.every(isLinearElement) | ||||
|       : isLinearElementType(appState.activeTool.type); | ||||
|       : isLinearElementType(appState.elementType); | ||||
|     return { | ||||
|       elements: changeProperty(elements, appState, (el) => | ||||
|         newElementWith(el, { | ||||
| @@ -904,8 +626,8 @@ export const actionChangeSharpness = register({ | ||||
|           elements, | ||||
|           appState, | ||||
|           (element) => element.strokeSharpness, | ||||
|           (canChangeSharpness(appState.activeTool.type) && | ||||
|             (isLinearElementType(appState.activeTool.type) | ||||
|           (canChangeSharpness(appState.elementType) && | ||||
|             (isLinearElementType(appState.elementType) | ||||
|               ? appState.currentItemLinearStrokeSharpness | ||||
|               : appState.currentItemStrokeSharpness)) || | ||||
|             null, | ||||
| @@ -918,7 +640,6 @@ export const actionChangeSharpness = register({ | ||||
|  | ||||
| export const actionChangeArrowhead = register({ | ||||
|   name: "changeArrowhead", | ||||
|   trackEvent: false, | ||||
|   perform: ( | ||||
|     elements, | ||||
|     appState, | ||||
| @@ -989,14 +710,6 @@ export const actionChangeArrowhead = register({ | ||||
|                 icon: <ArrowheadDotIcon theme={appState.theme} flip={!isRTL} />, | ||||
|                 keyBinding: "r", | ||||
|               }, | ||||
|               { | ||||
|                 value: "triangle", | ||||
|                 text: t("labels.arrowhead_triangle"), | ||||
|                 icon: ( | ||||
|                   <ArrowheadTriangleIcon theme={appState.theme} flip={!isRTL} /> | ||||
|                 ), | ||||
|                 keyBinding: "t", | ||||
|               }, | ||||
|             ]} | ||||
|             value={getFormValue<Arrowhead | null>( | ||||
|               elements, | ||||
| @@ -1039,14 +752,6 @@ export const actionChangeArrowhead = register({ | ||||
|                 keyBinding: "r", | ||||
|                 icon: <ArrowheadDotIcon theme={appState.theme} flip={isRTL} />, | ||||
|               }, | ||||
|               { | ||||
|                 value: "triangle", | ||||
|                 text: t("labels.arrowhead_triangle"), | ||||
|                 icon: ( | ||||
|                   <ArrowheadTriangleIcon theme={appState.theme} flip={isRTL} /> | ||||
|                 ), | ||||
|                 keyBinding: "t", | ||||
|               }, | ||||
|             ]} | ||||
|             value={getFormValue<Arrowhead | null>( | ||||
|               elements, | ||||
|   | ||||
| @@ -1,11 +1,10 @@ | ||||
| import { KEYS } from "../keys"; | ||||
| import { register } from "./register"; | ||||
| import { selectGroupsForSelectedElements } from "../groups"; | ||||
| import { getNonDeletedElements, isTextElement } from "../element"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
|  | ||||
| export const actionSelectAll = register({ | ||||
|   name: "selectAll", | ||||
|   trackEvent: { category: "canvas" }, | ||||
|   perform: (elements, appState) => { | ||||
|     if (appState.editingLinearElement) { | ||||
|       return false; | ||||
| @@ -16,11 +15,7 @@ export const actionSelectAll = register({ | ||||
|           ...appState, | ||||
|           editingGroupId: null, | ||||
|           selectedElementIds: elements.reduce((map, element) => { | ||||
|             if ( | ||||
|               !element.isDeleted && | ||||
|               !(isTextElement(element) && element.containerId) && | ||||
|               element.locked === false | ||||
|             ) { | ||||
|             if (!element.isDeleted) { | ||||
|               map[element.id] = true; | ||||
|             } | ||||
|             return map; | ||||
|   | ||||
| @@ -1,71 +0,0 @@ | ||||
| import ExcalidrawApp from "../excalidraw-app"; | ||||
| import { t } from "../i18n"; | ||||
| import { CODES } from "../keys"; | ||||
| import { API } from "../tests/helpers/api"; | ||||
| import { Keyboard, Pointer, UI } from "../tests/helpers/ui"; | ||||
| import { fireEvent, render, screen } from "../tests/test-utils"; | ||||
| import { copiedStyles } from "./actionStyles"; | ||||
|  | ||||
| const { h } = window; | ||||
|  | ||||
| const mouse = new Pointer("mouse"); | ||||
|  | ||||
| describe("actionStyles", () => { | ||||
|   beforeEach(async () => { | ||||
|     await render(<ExcalidrawApp />); | ||||
|   }); | ||||
|   it("should copy & paste styles via keyboard", () => { | ||||
|     UI.clickTool("rectangle"); | ||||
|     mouse.down(10, 10); | ||||
|     mouse.up(20, 20); | ||||
|  | ||||
|     UI.clickTool("rectangle"); | ||||
|     mouse.down(10, 10); | ||||
|     mouse.up(20, 20); | ||||
|  | ||||
|     // Change some styles of second rectangle | ||||
|     UI.clickLabeledElement("Stroke"); | ||||
|     UI.clickLabeledElement(t("colors.c92a2a")); | ||||
|     UI.clickLabeledElement("Background"); | ||||
|     UI.clickLabeledElement(t("colors.e64980")); | ||||
|     // Fill style | ||||
|     fireEvent.click(screen.getByTitle("Cross-hatch")); | ||||
|     // Stroke width | ||||
|     fireEvent.click(screen.getByTitle("Bold")); | ||||
|     // Stroke style | ||||
|     fireEvent.click(screen.getByTitle("Dotted")); | ||||
|     // Roughness | ||||
|     fireEvent.click(screen.getByTitle("Cartoonist")); | ||||
|     // Opacity | ||||
|     fireEvent.change(screen.getByLabelText("Opacity"), { | ||||
|       target: { value: "60" }, | ||||
|     }); | ||||
|  | ||||
|     mouse.reset(); | ||||
|  | ||||
|     API.setSelectedElements([h.elements[1]]); | ||||
|  | ||||
|     Keyboard.withModifierKeys({ ctrl: true, alt: true }, () => { | ||||
|       Keyboard.codeDown(CODES.C); | ||||
|     }); | ||||
|     const secondRect = JSON.parse(copiedStyles); | ||||
|     expect(secondRect.id).toBe(h.elements[1].id); | ||||
|  | ||||
|     mouse.reset(); | ||||
|     // Paste styles to first rectangle | ||||
|     API.setSelectedElements([h.elements[0]]); | ||||
|     Keyboard.withModifierKeys({ ctrl: true, alt: true }, () => { | ||||
|       Keyboard.codeDown(CODES.V); | ||||
|     }); | ||||
|  | ||||
|     const firstRect = API.getSelectedElement(); | ||||
|     expect(firstRect.id).toBe(h.elements[0].id); | ||||
|     expect(firstRect.strokeColor).toBe("#c92a2a"); | ||||
|     expect(firstRect.backgroundColor).toBe("#e64980"); | ||||
|     expect(firstRect.fillStyle).toBe("cross-hatch"); | ||||
|     expect(firstRect.strokeWidth).toBe(2); // Bold: 2 | ||||
|     expect(firstRect.strokeStyle).toBe("dotted"); | ||||
|     expect(firstRect.roughness).toBe(2); // Cartoonist: 2 | ||||
|     expect(firstRect.opacity).toBe(60); | ||||
|   }); | ||||
| }); | ||||
| @@ -12,14 +12,12 @@ import { | ||||
|   DEFAULT_FONT_FAMILY, | ||||
|   DEFAULT_TEXT_ALIGN, | ||||
| } from "../constants"; | ||||
| import { getContainerElement } from "../element/textElement"; | ||||
|  | ||||
| // `copiedStyles` is exported only for tests. | ||||
| export let copiedStyles: string = "{}"; | ||||
|  | ||||
| export const actionCopyStyles = register({ | ||||
|   name: "copyStyles", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     const element = elements.find((el) => appState.selectedElementIds[el.id]); | ||||
|     if (element) { | ||||
| @@ -40,7 +38,6 @@ export const actionCopyStyles = register({ | ||||
|  | ||||
| export const actionPasteStyles = register({ | ||||
|   name: "pasteStyles", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     const pastedElement = JSON.parse(copiedStyles); | ||||
|     if (!isExcalidrawElement(pastedElement)) { | ||||
| @@ -58,14 +55,13 @@ export const actionPasteStyles = register({ | ||||
|             opacity: pastedElement?.opacity, | ||||
|             roughness: pastedElement?.roughness, | ||||
|           }); | ||||
|           if (isTextElement(newElement) && isTextElement(element)) { | ||||
|           if (isTextElement(newElement)) { | ||||
|             mutateElement(newElement, { | ||||
|               fontSize: pastedElement?.fontSize || DEFAULT_FONT_SIZE, | ||||
|               fontFamily: pastedElement?.fontFamily || DEFAULT_FONT_FAMILY, | ||||
|               textAlign: pastedElement?.textAlign || DEFAULT_TEXT_ALIGN, | ||||
|             }); | ||||
|  | ||||
|             redrawTextBoundingBox(newElement, getContainerElement(newElement)); | ||||
|             redrawTextBoundingBox(newElement); | ||||
|           } | ||||
|           return newElement; | ||||
|         } | ||||
|   | ||||
| @@ -2,14 +2,12 @@ import { CODES, KEYS } from "../keys"; | ||||
| import { register } from "./register"; | ||||
| import { GRID_SIZE } from "../constants"; | ||||
| import { AppState } from "../types"; | ||||
| import { trackEvent } from "../analytics"; | ||||
|  | ||||
| export const actionToggleGridMode = register({ | ||||
|   name: "gridMode", | ||||
|   trackEvent: { | ||||
|     category: "canvas", | ||||
|     predicate: (appState) => !appState.gridSize, | ||||
|   }, | ||||
|   perform(elements, appState) { | ||||
|     trackEvent("view", "mode", "grid"); | ||||
|     return { | ||||
|       appState: { | ||||
|         ...appState, | ||||
|   | ||||
| @@ -1,63 +0,0 @@ | ||||
| import { newElementWith } from "../element/mutateElement"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { KEYS } from "../keys"; | ||||
| import { getSelectedElements } from "../scene"; | ||||
| import { arrayToMap } from "../utils"; | ||||
| import { register } from "./register"; | ||||
|  | ||||
| export const actionToggleLock = register({ | ||||
|   name: "toggleLock", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     const selectedElements = getSelectedElements(elements, appState, true); | ||||
|  | ||||
|     if (!selectedElements.length) { | ||||
|       return false; | ||||
|     } | ||||
|  | ||||
|     const operation = getOperation(selectedElements); | ||||
|     const selectedElementsMap = arrayToMap(selectedElements); | ||||
|  | ||||
|     return { | ||||
|       elements: elements.map((element) => { | ||||
|         if (!selectedElementsMap.has(element.id)) { | ||||
|           return element; | ||||
|         } | ||||
|  | ||||
|         return newElementWith(element, { locked: operation === "lock" }); | ||||
|       }), | ||||
|       appState, | ||||
|       commitToHistory: true, | ||||
|     }; | ||||
|   }, | ||||
|   contextItemLabel: (elements, appState) => { | ||||
|     const selected = getSelectedElements(elements, appState, false); | ||||
|     if (selected.length === 1) { | ||||
|       return selected[0].locked | ||||
|         ? "labels.elementLock.unlock" | ||||
|         : "labels.elementLock.lock"; | ||||
|     } | ||||
|  | ||||
|     if (selected.length > 1) { | ||||
|       return getOperation(selected) === "lock" | ||||
|         ? "labels.elementLock.lockAll" | ||||
|         : "labels.elementLock.unlockAll"; | ||||
|     } | ||||
|  | ||||
|     throw new Error( | ||||
|       "Unexpected zero elements to lock/unlock. This should never happen.", | ||||
|     ); | ||||
|   }, | ||||
|   keyTest: (event, appState, elements) => { | ||||
|     return ( | ||||
|       event.key.toLocaleLowerCase() === KEYS.L && | ||||
|       event[KEYS.CTRL_OR_CMD] && | ||||
|       event.shiftKey && | ||||
|       getSelectedElements(elements, appState, false).length > 0 | ||||
|     ); | ||||
|   }, | ||||
| }); | ||||
|  | ||||
| const getOperation = ( | ||||
|   elements: readonly ExcalidrawElement[], | ||||
| ): "lock" | "unlock" => (elements.some((el) => !el.locked) ? "lock" : "unlock"); | ||||
| @@ -3,7 +3,6 @@ import { CODES, KEYS } from "../keys"; | ||||
|  | ||||
| export const actionToggleStats = register({ | ||||
|   name: "stats", | ||||
|   trackEvent: { category: "menu" }, | ||||
|   perform(elements, appState) { | ||||
|     return { | ||||
|       appState: { | ||||
|   | ||||
| @@ -1,17 +1,16 @@ | ||||
| import { CODES, KEYS } from "../keys"; | ||||
| import { register } from "./register"; | ||||
| import { trackEvent } from "../analytics"; | ||||
|  | ||||
| export const actionToggleViewMode = register({ | ||||
|   name: "viewMode", | ||||
|   trackEvent: { | ||||
|     category: "canvas", | ||||
|     predicate: (appState) => !appState.viewModeEnabled, | ||||
|   }, | ||||
|   perform(elements, appState) { | ||||
|     trackEvent("view", "mode", "view"); | ||||
|     return { | ||||
|       appState: { | ||||
|         ...appState, | ||||
|         viewModeEnabled: !this.checked!(appState), | ||||
|         selectedElementIds: {}, | ||||
|       }, | ||||
|       commitToHistory: false, | ||||
|     }; | ||||
|   | ||||
| @@ -1,13 +1,12 @@ | ||||
| import { CODES, KEYS } from "../keys"; | ||||
| import { register } from "./register"; | ||||
| import { trackEvent } from "../analytics"; | ||||
|  | ||||
| export const actionToggleZenMode = register({ | ||||
|   name: "zenMode", | ||||
|   trackEvent: { | ||||
|     category: "canvas", | ||||
|     predicate: (appState) => !appState.zenModeEnabled, | ||||
|   }, | ||||
|   perform(elements, appState) { | ||||
|     trackEvent("view", "mode", "zen"); | ||||
|  | ||||
|     return { | ||||
|       appState: { | ||||
|         ...appState, | ||||
|   | ||||
| @@ -18,7 +18,6 @@ import { | ||||
|  | ||||
| export const actionSendBackward = register({ | ||||
|   name: "sendBackward", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       elements: moveOneLeft(elements, appState), | ||||
| @@ -46,7 +45,6 @@ export const actionSendBackward = register({ | ||||
|  | ||||
| export const actionBringForward = register({ | ||||
|   name: "bringForward", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       elements: moveOneRight(elements, appState), | ||||
| @@ -74,7 +72,6 @@ export const actionBringForward = register({ | ||||
|  | ||||
| export const actionSendToBack = register({ | ||||
|   name: "sendToBack", | ||||
|   trackEvent: { category: "element" }, | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       elements: moveAllLeft(elements, appState), | ||||
| @@ -109,8 +106,6 @@ export const actionSendToBack = register({ | ||||
|  | ||||
| export const actionBringToFront = register({ | ||||
|   name: "bringToFront", | ||||
|   trackEvent: { category: "element" }, | ||||
|  | ||||
|   perform: (elements, appState) => { | ||||
|     return { | ||||
|       elements: moveAllRight(elements, appState), | ||||
|   | ||||
| @@ -17,7 +17,6 @@ export { | ||||
|   actionChangeFontSize, | ||||
|   actionChangeFontFamily, | ||||
|   actionChangeTextAlign, | ||||
|   actionChangeVerticalAlign, | ||||
| } from "./actionProperties"; | ||||
|  | ||||
| export { | ||||
| @@ -35,8 +34,8 @@ export { actionFinalize } from "./actionFinalize"; | ||||
| export { | ||||
|   actionChangeProjectName, | ||||
|   actionChangeExportBackground, | ||||
|   actionSaveToActiveFile, | ||||
|   actionSaveFileToDisk, | ||||
|   actionSaveScene, | ||||
|   actionSaveAsScene, | ||||
|   actionLoadScene, | ||||
| } from "./actionExport"; | ||||
|  | ||||
| @@ -75,13 +74,9 @@ export { | ||||
|   actionCut, | ||||
|   actionCopyAsPng, | ||||
|   actionCopyAsSvg, | ||||
|   copyText, | ||||
| } from "./actionClipboard"; | ||||
|  | ||||
| export { actionToggleGridMode } from "./actionToggleGridMode"; | ||||
| export { actionToggleZenMode } from "./actionToggleZenMode"; | ||||
|  | ||||
| export { actionToggleStats } from "./actionToggleStats"; | ||||
| export { actionUnbindText, actionBindText } from "./actionBoundText"; | ||||
| export { actionLink } from "../element/Hyperlink"; | ||||
| export { actionToggleLock } from "./actionToggleLock"; | ||||
|   | ||||
| @@ -1,59 +1,39 @@ | ||||
| import React from "react"; | ||||
| import { | ||||
|   Action, | ||||
|   ActionsManagerInterface, | ||||
|   UpdaterFn, | ||||
|   ActionName, | ||||
|   ActionResult, | ||||
|   PanelComponentProps, | ||||
|   ActionSource, | ||||
| } from "./types"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { AppClassProperties, AppState } from "../types"; | ||||
| import { AppProps, AppState } from "../types"; | ||||
| import { MODES } from "../constants"; | ||||
| import { trackEvent } from "../analytics"; | ||||
| import Library from "../data/library"; | ||||
|  | ||||
| const trackAction = ( | ||||
|   action: Action, | ||||
|   source: ActionSource, | ||||
|   appState: Readonly<AppState>, | ||||
|   elements: readonly ExcalidrawElement[], | ||||
|   app: AppClassProperties, | ||||
|   value: any, | ||||
| ) => { | ||||
|   if (action.trackEvent) { | ||||
|     try { | ||||
|       if (typeof action.trackEvent === "object") { | ||||
|         const shouldTrack = action.trackEvent.predicate | ||||
|           ? action.trackEvent.predicate(appState, elements, value) | ||||
|           : true; | ||||
|         if (shouldTrack) { | ||||
|           trackEvent( | ||||
|             action.trackEvent.category, | ||||
|             action.trackEvent.action || action.name, | ||||
|             `${source} (${app.deviceType.isMobile ? "mobile" : "desktop"})`, | ||||
|           ); | ||||
|         } | ||||
|       } | ||||
|     } catch (error) { | ||||
|       console.error("error while logging action:", error); | ||||
|     } | ||||
|   } | ||||
| // This is the <App> component, but for now we don't care about anything but its | ||||
| // `canvas` state. | ||||
| type App = { | ||||
|   canvas: HTMLCanvasElement | null; | ||||
|   focusContainer: () => void; | ||||
|   props: AppProps; | ||||
|   library: Library; | ||||
| }; | ||||
|  | ||||
| export class ActionManager { | ||||
|   actions = {} as Record<ActionName, Action>; | ||||
| export class ActionManager implements ActionsManagerInterface { | ||||
|   actions = {} as ActionsManagerInterface["actions"]; | ||||
|  | ||||
|   updater: (actionResult: ActionResult | Promise<ActionResult>) => void; | ||||
|  | ||||
|   getAppState: () => Readonly<AppState>; | ||||
|   getElementsIncludingDeleted: () => readonly ExcalidrawElement[]; | ||||
|   app: AppClassProperties; | ||||
|   app: App; | ||||
|  | ||||
|   constructor( | ||||
|     updater: UpdaterFn, | ||||
|     getAppState: () => AppState, | ||||
|     getElementsIncludingDeleted: () => readonly ExcalidrawElement[], | ||||
|     app: AppClassProperties, | ||||
|     app: App, | ||||
|   ) { | ||||
|     this.updater = (actionResult) => { | ||||
|       if (actionResult && "then" in actionResult) { | ||||
| @@ -94,15 +74,9 @@ export class ActionManager { | ||||
|           ), | ||||
|       ); | ||||
|  | ||||
|     if (data.length !== 1) { | ||||
|       if (data.length > 1) { | ||||
|         console.warn("Canceling as multiple actions match this shortcut", data); | ||||
|       } | ||||
|     if (data.length === 0) { | ||||
|       return false; | ||||
|     } | ||||
|  | ||||
|     const action = data[0]; | ||||
|  | ||||
|     const { viewModeEnabled } = this.getAppState(); | ||||
|     if (viewModeEnabled) { | ||||
|       if (!Object.values(MODES).includes(data[0].name)) { | ||||
| @@ -110,32 +84,34 @@ export class ActionManager { | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     const elements = this.getElementsIncludingDeleted(); | ||||
|     const appState = this.getAppState(); | ||||
|     const value = null; | ||||
|  | ||||
|     trackAction(action, "keyboard", appState, elements, this.app, null); | ||||
|  | ||||
|     event.preventDefault(); | ||||
|     event.stopPropagation(); | ||||
|     this.updater(data[0].perform(elements, appState, value, this.app)); | ||||
|     this.updater( | ||||
|       data[0].perform( | ||||
|         this.getElementsIncludingDeleted(), | ||||
|         this.getAppState(), | ||||
|         null, | ||||
|         this.app, | ||||
|       ), | ||||
|     ); | ||||
|     return true; | ||||
|   } | ||||
|  | ||||
|   executeAction(action: Action, source: ActionSource = "api") { | ||||
|     const elements = this.getElementsIncludingDeleted(); | ||||
|     const appState = this.getAppState(); | ||||
|     const value = null; | ||||
|  | ||||
|     trackAction(action, source, appState, elements, this.app, value); | ||||
|  | ||||
|     this.updater(action.perform(elements, appState, value, this.app)); | ||||
|   executeAction(action: Action) { | ||||
|     this.updater( | ||||
|       action.perform( | ||||
|         this.getElementsIncludingDeleted(), | ||||
|         this.getAppState(), | ||||
|         null, | ||||
|         this.app, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * @param data additional data sent to the PanelComponent | ||||
|    */ | ||||
|   renderAction = (name: ActionName, data?: PanelComponentProps["data"]) => { | ||||
|   // Id is an attribute that we can use to pass in data like keys. | ||||
|   // This is needed for dynamically generated action components | ||||
|   // like the user list. We can use this key to extract more | ||||
|   // data from app state. This is an alternative to generic prop hell! | ||||
|   renderAction = (name: ActionName, id?: string) => { | ||||
|     const canvasActions = this.app.props.UIOptions.canvasActions; | ||||
|  | ||||
|     if ( | ||||
| @@ -147,11 +123,7 @@ export class ActionManager { | ||||
|     ) { | ||||
|       const action = this.actions[name]; | ||||
|       const PanelComponent = action.PanelComponent!; | ||||
|       const elements = this.getElementsIncludingDeleted(); | ||||
|       const appState = this.getAppState(); | ||||
|       const updateData = (formState?: any) => { | ||||
|         trackAction(action, "ui", appState, elements, this.app, formState); | ||||
|  | ||||
|         this.updater( | ||||
|           action.perform( | ||||
|             this.getElementsIncludingDeleted(), | ||||
| @@ -167,8 +139,8 @@ export class ActionManager { | ||||
|           elements={this.getElementsIncludingDeleted()} | ||||
|           appState={this.getAppState()} | ||||
|           updateData={updateData} | ||||
|           id={id} | ||||
|           appProps={this.app.props} | ||||
|           data={data} | ||||
|         /> | ||||
|       ); | ||||
|     } | ||||
|   | ||||
| @@ -2,9 +2,7 @@ import { Action } from "./types"; | ||||
|  | ||||
| export let actions: readonly Action[] = []; | ||||
|  | ||||
| export const register = <T extends Action>(action: T) => { | ||||
| export const register = (action: Action): Action => { | ||||
|   actions = actions.concat(action); | ||||
|   return action as T & { | ||||
|     keyTest?: unknown extends T["keyTest"] ? never : T["keyTest"]; | ||||
|   }; | ||||
|   return action; | ||||
| }; | ||||
|   | ||||
| @@ -1,10 +1,8 @@ | ||||
| import { t } from "../i18n"; | ||||
| import { isDarwin } from "../keys"; | ||||
| import { getShortcutKey } from "../utils"; | ||||
| import { ActionName } from "./types"; | ||||
|  | ||||
| export type ShortcutName = SubtypeOf< | ||||
|   ActionName, | ||||
| export type ShortcutName = | ||||
|   | "cut" | ||||
|   | "copy" | ||||
|   | "paste" | ||||
| @@ -27,10 +25,7 @@ export type ShortcutName = SubtypeOf< | ||||
|   | "addToLibrary" | ||||
|   | "viewMode" | ||||
|   | "flipHorizontal" | ||||
|   | "flipVertical" | ||||
|   | "hyperlink" | ||||
|   | "toggleLock" | ||||
| >; | ||||
|   | "flipVertical"; | ||||
|  | ||||
| const shortcutMap: Record<ShortcutName, string[]> = { | ||||
|   cut: [getShortcutKey("CtrlOrCmd+X")], | ||||
| @@ -67,12 +62,10 @@ const shortcutMap: Record<ShortcutName, string[]> = { | ||||
|   flipHorizontal: [getShortcutKey("Shift+H")], | ||||
|   flipVertical: [getShortcutKey("Shift+V")], | ||||
|   viewMode: [getShortcutKey("Alt+R")], | ||||
|   hyperlink: [getShortcutKey("CtrlOrCmd+K")], | ||||
|   toggleLock: [getShortcutKey("CtrlOrCmd+Shift+L")], | ||||
| }; | ||||
|  | ||||
| export const getShortcutFromShortcutName = (name: ShortcutName) => { | ||||
|   const shortcuts = shortcutMap[name]; | ||||
|   // if multiple shortcuts available, take the first one | ||||
|   // if multiple shortcuts availiable, take the first one | ||||
|   return shortcuts && shortcuts.length > 0 ? shortcuts[0] : ""; | ||||
| }; | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| import React from "react"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { | ||||
|   AppClassProperties, | ||||
|   AppState, | ||||
|   ExcalidrawProps, | ||||
|   BinaryFiles, | ||||
| } from "../types"; | ||||
| import { ToolButtonSize } from "../components/ToolButton"; | ||||
|  | ||||
| export type ActionSource = "ui" | "keyboard" | "contextMenu" | "api"; | ||||
| import { AppState, ExcalidrawProps } from "../types"; | ||||
| import Library from "../data/library"; | ||||
|  | ||||
| /** if false, the action should be prevented */ | ||||
| export type ActionResult = | ||||
| @@ -18,18 +11,22 @@ export type ActionResult = | ||||
|         AppState, | ||||
|         "offsetTop" | "offsetLeft" | "width" | "height" | ||||
|       > | null; | ||||
|       files?: BinaryFiles | null; | ||||
|       commitToHistory: boolean; | ||||
|       syncHistory?: boolean; | ||||
|       replaceFiles?: boolean; | ||||
|     } | ||||
|   | false; | ||||
|  | ||||
| type AppAPI = { | ||||
|   canvas: HTMLCanvasElement | null; | ||||
|   focusContainer(): void; | ||||
|   library: Library; | ||||
| }; | ||||
|  | ||||
| type ActionFn = ( | ||||
|   elements: readonly ExcalidrawElement[], | ||||
|   appState: Readonly<AppState>, | ||||
|   formData: any, | ||||
|   app: AppClassProperties, | ||||
|   app: AppAPI, | ||||
| ) => ActionResult | Promise<ActionResult>; | ||||
|  | ||||
| export type UpdaterFn = (res: ActionResult) => void; | ||||
| @@ -41,7 +38,6 @@ export type ActionName = | ||||
|   | "paste" | ||||
|   | "copyAsPng" | ||||
|   | "copyAsSvg" | ||||
|   | "copyText" | ||||
|   | "sendBackward" | ||||
|   | "bringForward" | ||||
|   | "sendToBack" | ||||
| @@ -70,9 +66,9 @@ export type ActionName = | ||||
|   | "changeProjectName" | ||||
|   | "changeExportBackground" | ||||
|   | "changeExportEmbedScene" | ||||
|   | "changeExportScale" | ||||
|   | "saveToActiveFile" | ||||
|   | "saveFileToDisk" | ||||
|   | "changeShouldAddWatermark" | ||||
|   | "saveScene" | ||||
|   | "saveAsScene" | ||||
|   | "loadScene" | ||||
|   | "duplicateSelection" | ||||
|   | "deleteSelectedElements" | ||||
| @@ -85,7 +81,6 @@ export type ActionName = | ||||
|   | "zoomToSelection" | ||||
|   | "changeFontFamily" | ||||
|   | "changeTextAlign" | ||||
|   | "changeVerticalAlign" | ||||
|   | "toggleFullScreen" | ||||
|   | "toggleShortcuts" | ||||
|   | "group" | ||||
| @@ -105,26 +100,17 @@ export type ActionName = | ||||
|   | "flipVertical" | ||||
|   | "viewMode" | ||||
|   | "exportWithDarkMode" | ||||
|   | "toggleTheme" | ||||
|   | "increaseFontSize" | ||||
|   | "decreaseFontSize" | ||||
|   | "unbindText" | ||||
|   | "hyperlink" | ||||
|   | "eraser" | ||||
|   | "bindText" | ||||
|   | "toggleLock"; | ||||
|  | ||||
| export type PanelComponentProps = { | ||||
|   elements: readonly ExcalidrawElement[]; | ||||
|   appState: AppState; | ||||
|   updateData: (formData?: any) => void; | ||||
|   appProps: ExcalidrawProps; | ||||
|   data?: Partial<{ id: string; size: ToolButtonSize }>; | ||||
| }; | ||||
|   | "toggleTheme"; | ||||
|  | ||||
| export interface Action { | ||||
|   name: ActionName; | ||||
|   PanelComponent?: React.FC<PanelComponentProps>; | ||||
|   PanelComponent?: React.FC<{ | ||||
|     elements: readonly ExcalidrawElement[]; | ||||
|     appState: AppState; | ||||
|     updateData: (formData?: any) => void; | ||||
|     appProps: ExcalidrawProps; | ||||
|     id?: string; | ||||
|   }>; | ||||
|   perform: ActionFn; | ||||
|   keyPriority?: number; | ||||
|   keyTest?: ( | ||||
| @@ -132,34 +118,18 @@ export interface Action { | ||||
|     appState: AppState, | ||||
|     elements: readonly ExcalidrawElement[], | ||||
|   ) => boolean; | ||||
|   contextItemLabel?: | ||||
|     | string | ||||
|     | (( | ||||
|         elements: readonly ExcalidrawElement[], | ||||
|         appState: Readonly<AppState>, | ||||
|       ) => string); | ||||
|   contextItemLabel?: string; | ||||
|   contextItemPredicate?: ( | ||||
|     elements: readonly ExcalidrawElement[], | ||||
|     appState: AppState, | ||||
|   ) => boolean; | ||||
|   checked?: (appState: Readonly<AppState>) => boolean; | ||||
|   trackEvent: | ||||
|     | false | ||||
|     | { | ||||
|         category: | ||||
|           | "toolbar" | ||||
|           | "element" | ||||
|           | "canvas" | ||||
|           | "export" | ||||
|           | "history" | ||||
|           | "menu" | ||||
|           | "collab" | ||||
|           | "hyperlink"; | ||||
|         action?: string; | ||||
|         predicate?: ( | ||||
|           appState: Readonly<AppState>, | ||||
|           elements: readonly ExcalidrawElement[], | ||||
|           value: any, | ||||
|         ) => boolean; | ||||
|       }; | ||||
| } | ||||
|  | ||||
| export interface ActionsManagerInterface { | ||||
|   actions: Record<ActionName, Action>; | ||||
|   registerAction: (action: Action) => void; | ||||
|   handleKeyDown: (event: React.KeyboardEvent | KeyboardEvent) => boolean; | ||||
|   renderAction: (name: ActionName) => React.ReactElement | null; | ||||
|   executeAction: (action: Action) => void; | ||||
| } | ||||
|   | ||||
							
								
								
									
										37
									
								
								src/align.ts
									
									
									
									
									
								
							
							
						
						
									
										37
									
								
								src/align.ts
									
									
									
									
									
								
							| @@ -1,7 +1,13 @@ | ||||
| import { ExcalidrawElement } from "./element/types"; | ||||
| import { newElementWith } from "./element/mutateElement"; | ||||
| import { Box, getCommonBoundingBox } from "./element/bounds"; | ||||
| import { getMaximumGroups } from "./groups"; | ||||
| import { getCommonBounds } from "./element"; | ||||
|  | ||||
| interface Box { | ||||
|   minX: number; | ||||
|   minY: number; | ||||
|   maxX: number; | ||||
|   maxY: number; | ||||
| } | ||||
|  | ||||
| export interface Alignment { | ||||
|   position: "start" | "center" | "end"; | ||||
| @@ -31,6 +37,28 @@ export const alignElements = ( | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| export const getMaximumGroups = ( | ||||
|   elements: ExcalidrawElement[], | ||||
| ): ExcalidrawElement[][] => { | ||||
|   const groups: Map<String, ExcalidrawElement[]> = new Map< | ||||
|     String, | ||||
|     ExcalidrawElement[] | ||||
|   >(); | ||||
|  | ||||
|   elements.forEach((element: ExcalidrawElement) => { | ||||
|     const groupId = | ||||
|       element.groupIds.length === 0 | ||||
|         ? element.id | ||||
|         : element.groupIds[element.groupIds.length - 1]; | ||||
|  | ||||
|     const currentGroupMembers = groups.get(groupId) || []; | ||||
|  | ||||
|     groups.set(groupId, [...currentGroupMembers, element]); | ||||
|   }); | ||||
|  | ||||
|   return Array.from(groups.values()); | ||||
| }; | ||||
|  | ||||
| const calculateTranslation = ( | ||||
|   group: ExcalidrawElement[], | ||||
|   selectionBoundingBox: Box, | ||||
| @@ -60,3 +88,8 @@ const calculateTranslation = ( | ||||
|       (groupBoundingBox[min] + groupBoundingBox[max]) / 2, | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| const getCommonBoundingBox = (elements: ExcalidrawElement[]): Box => { | ||||
|   const [minX, minY, maxX, maxY] = getCommonBounds(elements); | ||||
|   return { minX, minY, maxX, maxY }; | ||||
| }; | ||||
|   | ||||
| @@ -3,20 +3,16 @@ export const trackEvent = | ||||
|   process.env?.REACT_APP_GOOGLE_ANALYTICS_ID && | ||||
|   typeof window !== "undefined" && | ||||
|   window.gtag | ||||
|     ? (category: string, action: string, label?: string, value?: number) => { | ||||
|         try { | ||||
|           window.gtag("event", action, { | ||||
|             event_category: category, | ||||
|             event_label: label, | ||||
|             value, | ||||
|           }); | ||||
|         } catch (error) { | ||||
|           console.error("error logging to ga", error); | ||||
|         } | ||||
|     ? (category: string, name: string, label?: string, value?: number) => { | ||||
|         window.gtag("event", name, { | ||||
|           event_category: category, | ||||
|           event_label: label, | ||||
|           value, | ||||
|         }); | ||||
|       } | ||||
|     : typeof process !== "undefined" && process.env?.JEST_WORKER_ID | ||||
|     ? (category: string, action: string, label?: string, value?: number) => {} | ||||
|     : (category: string, action: string, label?: string, value?: number) => { | ||||
|     ? (category: string, name: string, label?: string, value?: number) => {} | ||||
|     : (category: string, name: string, label?: string, value?: number) => { | ||||
|         // Uncomment the next line to track locally | ||||
|         // console.log("Track Event", { category, action, label, value }); | ||||
|         // console.info("Track Event", category, name, label, value); | ||||
|       }; | ||||
|   | ||||
							
								
								
									
										189
									
								
								src/appState.ts
									
									
									
									
									
								
							
							
						
						
									
										189
									
								
								src/appState.ts
									
									
									
									
									
								
							| @@ -3,23 +3,17 @@ import { | ||||
|   DEFAULT_FONT_FAMILY, | ||||
|   DEFAULT_FONT_SIZE, | ||||
|   DEFAULT_TEXT_ALIGN, | ||||
|   EXPORT_SCALES, | ||||
|   THEME, | ||||
| } from "./constants"; | ||||
| import { t } from "./i18n"; | ||||
| import { AppState, NormalizedZoomValue } from "./types"; | ||||
| import { getDateTime } from "./utils"; | ||||
|  | ||||
| const defaultExportScale = EXPORT_SCALES.includes(devicePixelRatio) | ||||
|   ? devicePixelRatio | ||||
|   : 1; | ||||
|  | ||||
| export const getDefaultAppState = (): Omit< | ||||
|   AppState, | ||||
|   "offsetTop" | "offsetLeft" | "width" | "height" | ||||
| > => { | ||||
|   return { | ||||
|     theme: THEME.LIGHT, | ||||
|     theme: "light", | ||||
|     collaborators: new Map(), | ||||
|     currentChartType: "bar", | ||||
|     currentItemBackgroundColor: "transparent", | ||||
| @@ -41,16 +35,10 @@ export const getDefaultAppState = (): Omit< | ||||
|     editingElement: null, | ||||
|     editingGroupId: null, | ||||
|     editingLinearElement: null, | ||||
|     activeTool: { | ||||
|       type: "selection", | ||||
|       locked: false, | ||||
|       lastActiveToolBeforeEraser: null, | ||||
|     }, | ||||
|     penMode: false, | ||||
|     penDetected: false, | ||||
|     elementLocked: false, | ||||
|     elementType: "selection", | ||||
|     errorMessage: null, | ||||
|     exportBackground: true, | ||||
|     exportScale: defaultExportScale, | ||||
|     exportEmbedScene: false, | ||||
|     exportWithDarkMode: false, | ||||
|     fileHandle: null, | ||||
| @@ -64,7 +52,6 @@ export const getDefaultAppState = (): Omit< | ||||
|     multiElement: null, | ||||
|     name: `${t("labels.untitled")}-${getDateTime()}`, | ||||
|     openMenu: null, | ||||
|     openPopup: null, | ||||
|     pasteDialog: { shown: false, data: null }, | ||||
|     previousSelectedElementIds: {}, | ||||
|     resizingElement: null, | ||||
| @@ -74,6 +61,7 @@ export const getDefaultAppState = (): Omit< | ||||
|     selectedElementIds: {}, | ||||
|     selectedGroupIds: {}, | ||||
|     selectionElement: null, | ||||
|     shouldAddWatermark: false, | ||||
|     shouldCacheIgnoreZoom: false, | ||||
|     showHelpDialog: false, | ||||
|     showStats: false, | ||||
| @@ -82,12 +70,8 @@ export const getDefaultAppState = (): Omit< | ||||
|     toastMessage: null, | ||||
|     viewBackgroundColor: oc.white, | ||||
|     zenModeEnabled: false, | ||||
|     zoom: { | ||||
|       value: 1 as NormalizedZoomValue, | ||||
|     }, | ||||
|     zoom: { value: 1 as NormalizedZoomValue, translation: { x: 0, y: 0 } }, | ||||
|     viewModeEnabled: false, | ||||
|     pendingImageElement: null, | ||||
|     showHyperlinkPopup: false, | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| @@ -101,88 +85,77 @@ const APP_STATE_STORAGE_CONF = (< | ||||
|     browser: boolean; | ||||
|     /** whether to keep when exporting to file/database */ | ||||
|     export: boolean; | ||||
|     /** server (shareLink/collab/...) */ | ||||
|     server: boolean; | ||||
|   }, | ||||
|   T extends Record<keyof AppState, Values>, | ||||
| >(config: { [K in keyof T]: K extends keyof AppState ? T[K] : never }) => | ||||
|   config)({ | ||||
|   theme: { browser: true, export: false, server: false }, | ||||
|   collaborators: { browser: false, export: false, server: false }, | ||||
|   currentChartType: { browser: true, export: false, server: false }, | ||||
|   currentItemBackgroundColor: { browser: true, export: false, server: false }, | ||||
|   currentItemEndArrowhead: { browser: true, export: false, server: false }, | ||||
|   currentItemFillStyle: { browser: true, export: false, server: false }, | ||||
|   currentItemFontFamily: { browser: true, export: false, server: false }, | ||||
|   currentItemFontSize: { browser: true, export: false, server: false }, | ||||
|   currentItemLinearStrokeSharpness: { | ||||
|     browser: true, | ||||
|     export: false, | ||||
|     server: false, | ||||
|   }, | ||||
|   currentItemOpacity: { browser: true, export: false, server: false }, | ||||
|   currentItemRoughness: { browser: true, export: false, server: false }, | ||||
|   currentItemStartArrowhead: { browser: true, export: false, server: false }, | ||||
|   currentItemStrokeColor: { browser: true, export: false, server: false }, | ||||
|   currentItemStrokeSharpness: { browser: true, export: false, server: false }, | ||||
|   currentItemStrokeStyle: { browser: true, export: false, server: false }, | ||||
|   currentItemStrokeWidth: { browser: true, export: false, server: false }, | ||||
|   currentItemTextAlign: { browser: true, export: false, server: false }, | ||||
|   cursorButton: { browser: true, export: false, server: false }, | ||||
|   draggingElement: { browser: false, export: false, server: false }, | ||||
|   editingElement: { browser: false, export: false, server: false }, | ||||
|   editingGroupId: { browser: true, export: false, server: false }, | ||||
|   editingLinearElement: { browser: false, export: false, server: false }, | ||||
|   activeTool: { browser: true, export: false, server: false }, | ||||
|   penMode: { browser: true, export: false, server: false }, | ||||
|   penDetected: { browser: true, export: false, server: false }, | ||||
|   errorMessage: { browser: false, export: false, server: false }, | ||||
|   exportBackground: { browser: true, export: false, server: false }, | ||||
|   exportEmbedScene: { browser: true, export: false, server: false }, | ||||
|   exportScale: { browser: true, export: false, server: false }, | ||||
|   exportWithDarkMode: { browser: true, export: false, server: false }, | ||||
|   fileHandle: { browser: false, export: false, server: false }, | ||||
|   gridSize: { browser: true, export: true, server: true }, | ||||
|   height: { browser: false, export: false, server: false }, | ||||
|   isBindingEnabled: { browser: false, export: false, server: false }, | ||||
|   isLibraryOpen: { browser: false, export: false, server: false }, | ||||
|   isLoading: { browser: false, export: false, server: false }, | ||||
|   isResizing: { browser: false, export: false, server: false }, | ||||
|   isRotating: { browser: false, export: false, server: false }, | ||||
|   lastPointerDownWith: { browser: true, export: false, server: false }, | ||||
|   multiElement: { browser: false, export: false, server: false }, | ||||
|   name: { browser: true, export: false, server: false }, | ||||
|   offsetLeft: { browser: false, export: false, server: false }, | ||||
|   offsetTop: { browser: false, export: false, server: false }, | ||||
|   openMenu: { browser: true, export: false, server: false }, | ||||
|   openPopup: { browser: false, export: false, server: false }, | ||||
|   pasteDialog: { browser: false, export: false, server: false }, | ||||
|   previousSelectedElementIds: { browser: true, export: false, server: false }, | ||||
|   resizingElement: { browser: false, export: false, server: false }, | ||||
|   scrolledOutside: { browser: true, export: false, server: false }, | ||||
|   scrollX: { browser: true, export: false, server: false }, | ||||
|   scrollY: { browser: true, export: false, server: false }, | ||||
|   selectedElementIds: { browser: true, export: false, server: false }, | ||||
|   selectedGroupIds: { browser: true, export: false, server: false }, | ||||
|   selectionElement: { browser: false, export: false, server: false }, | ||||
|   shouldCacheIgnoreZoom: { browser: true, export: false, server: false }, | ||||
|   showHelpDialog: { browser: false, export: false, server: false }, | ||||
|   showStats: { browser: true, export: false, server: false }, | ||||
|   startBoundElement: { browser: false, export: false, server: false }, | ||||
|   suggestedBindings: { browser: false, export: false, server: false }, | ||||
|   toastMessage: { browser: false, export: false, server: false }, | ||||
|   viewBackgroundColor: { browser: true, export: true, server: true }, | ||||
|   width: { browser: false, export: false, server: false }, | ||||
|   zenModeEnabled: { browser: true, export: false, server: false }, | ||||
|   zoom: { browser: true, export: false, server: false }, | ||||
|   viewModeEnabled: { browser: false, export: false, server: false }, | ||||
|   pendingImageElement: { browser: false, export: false, server: false }, | ||||
|   showHyperlinkPopup: { browser: false, export: false, server: false }, | ||||
|   T extends Record<keyof AppState, Values> | ||||
| >( | ||||
|   config: { [K in keyof T]: K extends keyof AppState ? T[K] : never }, | ||||
| ) => config)({ | ||||
|   theme: { browser: true, export: false }, | ||||
|   collaborators: { browser: false, export: false }, | ||||
|   currentChartType: { browser: true, export: false }, | ||||
|   currentItemBackgroundColor: { browser: true, export: false }, | ||||
|   currentItemEndArrowhead: { browser: true, export: false }, | ||||
|   currentItemFillStyle: { browser: true, export: false }, | ||||
|   currentItemFontFamily: { browser: true, export: false }, | ||||
|   currentItemFontSize: { browser: true, export: false }, | ||||
|   currentItemLinearStrokeSharpness: { browser: true, export: false }, | ||||
|   currentItemOpacity: { browser: true, export: false }, | ||||
|   currentItemRoughness: { browser: true, export: false }, | ||||
|   currentItemStartArrowhead: { browser: true, export: false }, | ||||
|   currentItemStrokeColor: { browser: true, export: false }, | ||||
|   currentItemStrokeSharpness: { browser: true, export: false }, | ||||
|   currentItemStrokeStyle: { browser: true, export: false }, | ||||
|   currentItemStrokeWidth: { browser: true, export: false }, | ||||
|   currentItemTextAlign: { browser: true, export: false }, | ||||
|   cursorButton: { browser: true, export: false }, | ||||
|   draggingElement: { browser: false, export: false }, | ||||
|   editingElement: { browser: false, export: false }, | ||||
|   editingGroupId: { browser: true, export: false }, | ||||
|   editingLinearElement: { browser: false, export: false }, | ||||
|   elementLocked: { browser: true, export: false }, | ||||
|   elementType: { browser: true, export: false }, | ||||
|   errorMessage: { browser: false, export: false }, | ||||
|   exportBackground: { browser: true, export: false }, | ||||
|   exportEmbedScene: { browser: true, export: false }, | ||||
|   exportWithDarkMode: { browser: true, export: false }, | ||||
|   fileHandle: { browser: false, export: false }, | ||||
|   gridSize: { browser: true, export: true }, | ||||
|   height: { browser: false, export: false }, | ||||
|   isBindingEnabled: { browser: false, export: false }, | ||||
|   isLibraryOpen: { browser: false, export: false }, | ||||
|   isLoading: { browser: false, export: false }, | ||||
|   isResizing: { browser: false, export: false }, | ||||
|   isRotating: { browser: false, export: false }, | ||||
|   lastPointerDownWith: { browser: true, export: false }, | ||||
|   multiElement: { browser: false, export: false }, | ||||
|   name: { browser: true, export: false }, | ||||
|   offsetLeft: { browser: false, export: false }, | ||||
|   offsetTop: { browser: false, export: false }, | ||||
|   openMenu: { browser: true, export: false }, | ||||
|   pasteDialog: { browser: false, export: false }, | ||||
|   previousSelectedElementIds: { browser: true, export: false }, | ||||
|   resizingElement: { browser: false, export: false }, | ||||
|   scrolledOutside: { browser: true, export: false }, | ||||
|   scrollX: { browser: true, export: false }, | ||||
|   scrollY: { browser: true, export: false }, | ||||
|   selectedElementIds: { browser: true, export: false }, | ||||
|   selectedGroupIds: { browser: true, export: false }, | ||||
|   selectionElement: { browser: false, export: false }, | ||||
|   shouldAddWatermark: { browser: true, export: false }, | ||||
|   shouldCacheIgnoreZoom: { browser: true, export: false }, | ||||
|   showHelpDialog: { browser: false, export: false }, | ||||
|   showStats: { browser: true, export: false }, | ||||
|   startBoundElement: { browser: false, export: false }, | ||||
|   suggestedBindings: { browser: false, export: false }, | ||||
|   toastMessage: { browser: false, export: false }, | ||||
|   viewBackgroundColor: { browser: true, export: true }, | ||||
|   width: { browser: false, export: false }, | ||||
|   zenModeEnabled: { browser: true, export: false }, | ||||
|   zoom: { browser: true, export: false }, | ||||
|   viewModeEnabled: { browser: false, export: false }, | ||||
| }); | ||||
|  | ||||
| const _clearAppStateForStorage = < | ||||
|   ExportType extends "export" | "browser" | "server", | ||||
| >( | ||||
| const _clearAppStateForStorage = <ExportType extends "export" | "browser">( | ||||
|   appState: Partial<AppState>, | ||||
|   exportType: ExportType, | ||||
| ) => { | ||||
| @@ -195,10 +168,8 @@ const _clearAppStateForStorage = < | ||||
|   for (const key of Object.keys(appState) as (keyof typeof appState)[]) { | ||||
|     const propConfig = APP_STATE_STORAGE_CONF[key]; | ||||
|     if (propConfig?.[exportType]) { | ||||
|       const nextValue = appState[key]; | ||||
|  | ||||
|       // https://github.com/microsoft/TypeScript/issues/31445 | ||||
|       (stateForExport as any)[key] = nextValue; | ||||
|       // @ts-ignore see https://github.com/microsoft/TypeScript/issues/31445 | ||||
|       stateForExport[key] = appState[key]; | ||||
|     } | ||||
|   } | ||||
|   return stateForExport; | ||||
| @@ -211,13 +182,3 @@ export const clearAppStateForLocalStorage = (appState: Partial<AppState>) => { | ||||
| export const cleanAppStateForExport = (appState: Partial<AppState>) => { | ||||
|   return _clearAppStateForStorage(appState, "export"); | ||||
| }; | ||||
|  | ||||
| export const clearAppStateForDatabase = (appState: Partial<AppState>) => { | ||||
|   return _clearAppStateForStorage(appState, "server"); | ||||
| }; | ||||
|  | ||||
| export const isEraserActive = ({ | ||||
|   activeTool, | ||||
| }: { | ||||
|   activeTool: AppState["activeTool"]; | ||||
| }) => activeTool.type === "eraser"; | ||||
|   | ||||
| @@ -1,10 +1,5 @@ | ||||
| import colors from "./colors"; | ||||
| import { | ||||
|   DEFAULT_FONT_FAMILY, | ||||
|   DEFAULT_FONT_SIZE, | ||||
|   ENV, | ||||
|   VERTICAL_ALIGN, | ||||
| } from "./constants"; | ||||
| import { DEFAULT_FONT_FAMILY, DEFAULT_FONT_SIZE, ENV } from "./constants"; | ||||
| import { newElement, newLinearElement, newTextElement } from "./element"; | ||||
| import { NonDeletedExcalidrawElement } from "./element/types"; | ||||
| import { randomId } from "./random"; | ||||
| @@ -108,7 +103,7 @@ const transposeCells = (cells: string[][]) => { | ||||
| }; | ||||
|  | ||||
| export const tryParseSpreadsheet = (text: string): ParseSpreadsheetResult => { | ||||
|   // Copy/paste from excel, spreadsheets, tsv, csv. | ||||
|   // Copy/paste from excel, spreadhseets, tsv, csv. | ||||
|   // For now we only accept 2 columns with an optional header | ||||
|  | ||||
|   // Check for tab separated values | ||||
| @@ -166,8 +161,7 @@ const commonProps = { | ||||
|   strokeSharpness: "sharp", | ||||
|   strokeStyle: "solid", | ||||
|   strokeWidth: 1, | ||||
|   verticalAlign: VERTICAL_ALIGN.MIDDLE, | ||||
|   locked: false, | ||||
|   verticalAlign: "middle", | ||||
| } as const; | ||||
|  | ||||
| const getChartDimentions = (spreadsheet: Spreadsheet) => { | ||||
|   | ||||
| @@ -3,23 +3,19 @@ import { | ||||
|   NonDeletedExcalidrawElement, | ||||
| } from "./element/types"; | ||||
| import { getSelectedElements } from "./scene"; | ||||
| import { AppState, BinaryFiles } from "./types"; | ||||
| import { AppState } from "./types"; | ||||
| import { SVG_EXPORT_TAG } from "./scene/export"; | ||||
| import { tryParseSpreadsheet, Spreadsheet, VALID_SPREADSHEET } from "./charts"; | ||||
| import { EXPORT_DATA_TYPES, MIME_TYPES } from "./constants"; | ||||
| import { isInitializedImageElement } from "./element/typeChecks"; | ||||
| import { isPromiseLike } from "./utils"; | ||||
| import { EXPORT_DATA_TYPES } from "./constants"; | ||||
|  | ||||
| type ElementsClipboard = { | ||||
|   type: typeof EXPORT_DATA_TYPES.excalidrawClipboard; | ||||
|   elements: ExcalidrawElement[]; | ||||
|   files: BinaryFiles | undefined; | ||||
| }; | ||||
|  | ||||
| export interface ClipboardData { | ||||
|   spreadsheet?: Spreadsheet; | ||||
|   elements?: readonly ExcalidrawElement[]; | ||||
|   files?: BinaryFiles; | ||||
|   text?: string; | ||||
|   errorMessage?: string; | ||||
| } | ||||
| @@ -41,7 +37,7 @@ export const probablySupportsClipboardBlob = | ||||
|  | ||||
| const clipboardContainsElements = ( | ||||
|   contents: any, | ||||
| ): contents is { elements: ExcalidrawElement[]; files?: BinaryFiles } => { | ||||
| ): contents is { elements: ExcalidrawElement[] } => { | ||||
|   if ( | ||||
|     [ | ||||
|       EXPORT_DATA_TYPES.excalidraw, | ||||
| @@ -57,26 +53,17 @@ const clipboardContainsElements = ( | ||||
| export const copyToClipboard = async ( | ||||
|   elements: readonly NonDeletedExcalidrawElement[], | ||||
|   appState: AppState, | ||||
|   files: BinaryFiles, | ||||
| ) => { | ||||
|   // select binded text elements when copying | ||||
|   const selectedElements = getSelectedElements(elements, appState, true); | ||||
|   const contents: ElementsClipboard = { | ||||
|     type: EXPORT_DATA_TYPES.excalidrawClipboard, | ||||
|     elements: selectedElements, | ||||
|     files: selectedElements.reduce((acc, element) => { | ||||
|       if (isInitializedImageElement(element) && files[element.fileId]) { | ||||
|         acc[element.fileId] = files[element.fileId]; | ||||
|       } | ||||
|       return acc; | ||||
|     }, {} as BinaryFiles), | ||||
|     elements: getSelectedElements(elements, appState), | ||||
|   }; | ||||
|   const json = JSON.stringify(contents); | ||||
|   CLIPBOARD = json; | ||||
|   try { | ||||
|     PREFER_APP_CLIPBOARD = false; | ||||
|     await copyTextToSystemClipboard(json); | ||||
|   } catch (error: any) { | ||||
|   } catch (error) { | ||||
|     PREFER_APP_CLIPBOARD = true; | ||||
|     console.error(error); | ||||
|   } | ||||
| @@ -89,7 +76,7 @@ const getAppClipboard = (): Partial<ElementsClipboard> => { | ||||
|  | ||||
|   try { | ||||
|     return JSON.parse(CLIPBOARD); | ||||
|   } catch (error: any) { | ||||
|   } catch (error) { | ||||
|     console.error(error); | ||||
|     return {}; | ||||
|   } | ||||
| @@ -125,7 +112,7 @@ const getSystemClipboard = async ( | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Attempts to parse clipboard. Prefers system clipboard. | ||||
|  * Attemps to parse clipboard. Prefers system clipboard. | ||||
|  */ | ||||
| export const parseClipboard = async ( | ||||
|   event: ClipboardEvent | null, | ||||
| @@ -151,10 +138,7 @@ export const parseClipboard = async ( | ||||
|   try { | ||||
|     const systemClipboardData = JSON.parse(systemClipboard); | ||||
|     if (clipboardContainsElements(systemClipboardData)) { | ||||
|       return { | ||||
|         elements: systemClipboardData.elements, | ||||
|         files: systemClipboardData.files, | ||||
|       }; | ||||
|       return { elements: systemClipboardData.elements }; | ||||
|     } | ||||
|     return appClipboardData; | ||||
|   } catch { | ||||
| @@ -167,35 +151,10 @@ export const parseClipboard = async ( | ||||
|   } | ||||
| }; | ||||
|  | ||||
| export const copyBlobToClipboardAsPng = async (blob: Blob | Promise<Blob>) => { | ||||
|   let promise; | ||||
|   try { | ||||
|     // in Safari so far we need to construct the ClipboardItem synchronously | ||||
|     // (i.e. in the same tick) otherwise browser will complain for lack of | ||||
|     // user intent. Using a Promise ClipboardItem constructor solves this. | ||||
|     // https://bugs.webkit.org/show_bug.cgi?id=222262 | ||||
|     // | ||||
|     // not await so that we can detect whether the thrown error likely relates | ||||
|     // to a lack of support for the Promise ClipboardItem constructor | ||||
|     promise = navigator.clipboard.write([ | ||||
|       new window.ClipboardItem({ | ||||
|         [MIME_TYPES.png]: blob, | ||||
|       }), | ||||
|     ]); | ||||
|   } catch (error: any) { | ||||
|     // if we're using a Promise ClipboardItem, let's try constructing | ||||
|     // with resolution value instead | ||||
|     if (isPromiseLike(blob)) { | ||||
|       await navigator.clipboard.write([ | ||||
|         new window.ClipboardItem({ | ||||
|           [MIME_TYPES.png]: await blob, | ||||
|         }), | ||||
|       ]); | ||||
|     } else { | ||||
|       throw error; | ||||
|     } | ||||
|   } | ||||
|   await promise; | ||||
| export const copyBlobToClipboardAsPng = async (blob: Blob) => { | ||||
|   await navigator.clipboard.write([ | ||||
|     new window.ClipboardItem({ "image/png": blob }), | ||||
|   ]); | ||||
| }; | ||||
|  | ||||
| export const copyTextToSystemClipboard = async (text: string | null) => { | ||||
| @@ -206,7 +165,7 @@ export const copyTextToSystemClipboard = async (text: string | null) => { | ||||
|       // not focused | ||||
|       await navigator.clipboard.writeText(text || ""); | ||||
|       copied = true; | ||||
|     } catch (error: any) { | ||||
|     } catch (error) { | ||||
|       console.error(error); | ||||
|     } | ||||
|   } | ||||
| @@ -246,7 +205,7 @@ const copyTextViaExecCommand = (text: string) => { | ||||
|     textarea.setSelectionRange(0, textarea.value.length); | ||||
|  | ||||
|     success = document.execCommand("copy"); | ||||
|   } catch (error: any) { | ||||
|   } catch (error) { | ||||
|     console.error(error); | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| import React from "react"; | ||||
| import { ActionManager } from "../actions/manager"; | ||||
| import { getNonDeletedElements } from "../element"; | ||||
| import { ExcalidrawElement, PointerType } from "../element/types"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { t } from "../i18n"; | ||||
| import { useDeviceType } from "../components/App"; | ||||
| import { useIsMobile } from "../components/App"; | ||||
| import { | ||||
|   canChangeSharpness, | ||||
|   canHaveArrowheads, | ||||
| @@ -18,79 +18,51 @@ import { AppState, Zoom } from "../types"; | ||||
| import { capitalizeString, isTransparent, setCursorForShape } from "../utils"; | ||||
| import Stack from "./Stack"; | ||||
| import { ToolButton } from "./ToolButton"; | ||||
| import { hasStrokeColor } from "../scene/comparisons"; | ||||
| import { trackEvent } from "../analytics"; | ||||
| import { hasBoundTextElement, isBoundToContainer } from "../element/typeChecks"; | ||||
|  | ||||
| export const SelectedShapeActions = ({ | ||||
|   appState, | ||||
|   elements, | ||||
|   renderAction, | ||||
|   activeTool, | ||||
|   elementType, | ||||
| }: { | ||||
|   appState: AppState; | ||||
|   elements: readonly ExcalidrawElement[]; | ||||
|   renderAction: ActionManager["renderAction"]; | ||||
|   activeTool: AppState["activeTool"]["type"]; | ||||
|   elementType: ExcalidrawElement["type"]; | ||||
| }) => { | ||||
|   const targetElements = getTargetElements( | ||||
|     getNonDeletedElements(elements), | ||||
|     appState, | ||||
|   ); | ||||
|  | ||||
|   let isSingleElementBoundContainer = false; | ||||
|   if ( | ||||
|     targetElements.length === 2 && | ||||
|     (hasBoundTextElement(targetElements[0]) || | ||||
|       hasBoundTextElement(targetElements[1])) | ||||
|   ) { | ||||
|     isSingleElementBoundContainer = true; | ||||
|   } | ||||
|   const isEditing = Boolean(appState.editingElement); | ||||
|   const deviceType = useDeviceType(); | ||||
|   const isMobile = useIsMobile(); | ||||
|   const isRTL = document.documentElement.getAttribute("dir") === "rtl"; | ||||
|  | ||||
|   const showFillIcons = | ||||
|     hasBackground(activeTool) || | ||||
|     hasBackground(elementType) || | ||||
|     targetElements.some( | ||||
|       (element) => | ||||
|         hasBackground(element.type) && !isTransparent(element.backgroundColor), | ||||
|     ); | ||||
|   const showChangeBackgroundIcons = | ||||
|     hasBackground(activeTool) || | ||||
|     hasBackground(elementType) || | ||||
|     targetElements.some((element) => hasBackground(element.type)); | ||||
|  | ||||
|   const showLinkIcon = | ||||
|     targetElements.length === 1 || isSingleElementBoundContainer; | ||||
|  | ||||
|   let commonSelectedType: string | null = targetElements[0]?.type || null; | ||||
|  | ||||
|   for (const element of targetElements) { | ||||
|     if (element.type !== commonSelectedType) { | ||||
|       commonSelectedType = null; | ||||
|       break; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     <div className="panelColumn"> | ||||
|       {((hasStrokeColor(activeTool) && | ||||
|         activeTool !== "image" && | ||||
|         commonSelectedType !== "image") || | ||||
|         targetElements.some((element) => hasStrokeColor(element.type))) && | ||||
|         renderAction("changeStrokeColor")} | ||||
|       {renderAction("changeStrokeColor")} | ||||
|       {showChangeBackgroundIcons && renderAction("changeBackgroundColor")} | ||||
|       {showFillIcons && renderAction("changeFillStyle")} | ||||
|  | ||||
|       {(hasStrokeWidth(activeTool) || | ||||
|       {(hasStrokeWidth(elementType) || | ||||
|         targetElements.some((element) => hasStrokeWidth(element.type))) && | ||||
|         renderAction("changeStrokeWidth")} | ||||
|  | ||||
|       {(activeTool === "freedraw" || | ||||
|       {(elementType === "freedraw" || | ||||
|         targetElements.some((element) => element.type === "freedraw")) && | ||||
|         renderAction("changeStrokeShape")} | ||||
|  | ||||
|       {(hasStrokeStyle(activeTool) || | ||||
|       {(hasStrokeStyle(elementType) || | ||||
|         targetElements.some((element) => hasStrokeStyle(element.type))) && ( | ||||
|         <> | ||||
|           {renderAction("changeStrokeStyle")} | ||||
| @@ -98,12 +70,12 @@ export const SelectedShapeActions = ({ | ||||
|         </> | ||||
|       )} | ||||
|  | ||||
|       {(canChangeSharpness(activeTool) || | ||||
|       {(canChangeSharpness(elementType) || | ||||
|         targetElements.some((element) => canChangeSharpness(element.type))) && ( | ||||
|         <>{renderAction("changeSharpness")}</> | ||||
|       )} | ||||
|  | ||||
|       {(hasText(activeTool) || | ||||
|       {(hasText(elementType) || | ||||
|         targetElements.some((element) => hasText(element.type))) && ( | ||||
|         <> | ||||
|           {renderAction("changeFontSize")} | ||||
| @@ -114,11 +86,7 @@ export const SelectedShapeActions = ({ | ||||
|         </> | ||||
|       )} | ||||
|  | ||||
|       {targetElements.some( | ||||
|         (element) => | ||||
|           hasBoundTextElement(element) || isBoundToContainer(element), | ||||
|       ) && renderAction("changeVerticalAlign")} | ||||
|       {(canHaveArrowheads(activeTool) || | ||||
|       {(canHaveArrowheads(elementType) || | ||||
|         targetElements.some((element) => canHaveArrowheads(element.type))) && ( | ||||
|         <>{renderAction("changeArrowhead")}</> | ||||
|       )} | ||||
| @@ -135,7 +103,7 @@ export const SelectedShapeActions = ({ | ||||
|         </div> | ||||
|       </fieldset> | ||||
|  | ||||
|       {targetElements.length > 1 && !isSingleElementBoundContainer && ( | ||||
|       {targetElements.length > 1 && ( | ||||
|         <fieldset> | ||||
|           <legend>{t("labels.align")}</legend> | ||||
|           <div className="buttonList"> | ||||
| @@ -168,15 +136,14 @@ export const SelectedShapeActions = ({ | ||||
|           </div> | ||||
|         </fieldset> | ||||
|       )} | ||||
|       {!isEditing && targetElements.length > 0 && ( | ||||
|       {!isMobile && !isEditing && targetElements.length > 0 && ( | ||||
|         <fieldset> | ||||
|           <legend>{t("labels.actions")}</legend> | ||||
|           <div className="buttonList"> | ||||
|             {!deviceType.isMobile && renderAction("duplicateSelection")} | ||||
|             {!deviceType.isMobile && renderAction("deleteSelectedElements")} | ||||
|             {renderAction("duplicateSelection")} | ||||
|             {renderAction("deleteSelectedElements")} | ||||
|             {renderAction("group")} | ||||
|             {renderAction("ungroup")} | ||||
|             {showLinkIcon && renderAction("hyperlink")} | ||||
|           </div> | ||||
|         </fieldset> | ||||
|       )} | ||||
| @@ -184,68 +151,69 @@ export const SelectedShapeActions = ({ | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| const LIBRARY_ICON = ( | ||||
|   // fa-th-large | ||||
|   <svg viewBox="0 0 512 512"> | ||||
|     <path d="M296 32h192c13.255 0 24 10.745 24 24v160c0 13.255-10.745 24-24 24H296c-13.255 0-24-10.745-24-24V56c0-13.255 10.745-24 24-24zm-80 0H24C10.745 32 0 42.745 0 56v160c0 13.255 10.745 24 24 24h192c13.255 0 24-10.745 24-24V56c0-13.255-10.745-24-24-24zM0 296v160c0 13.255 10.745 24 24 24h192c13.255 0 24-10.745 24-24V296c0-13.255-10.745-24-24-24H24c-13.255 0-24 10.745-24 24zm296 184h192c13.255 0 24-10.745 24-24V296c0-13.255-10.745-24-24-24H296c-13.255 0-24 10.745-24 24v160c0 13.255 10.745 24 24 24z" /> | ||||
|   </svg> | ||||
| ); | ||||
|  | ||||
| export const ShapesSwitcher = ({ | ||||
|   canvas, | ||||
|   activeTool, | ||||
|   elementType, | ||||
|   setAppState, | ||||
|   onImageAction, | ||||
|   appState, | ||||
|   isLibraryOpen, | ||||
| }: { | ||||
|   canvas: HTMLCanvasElement | null; | ||||
|   activeTool: AppState["activeTool"]; | ||||
|   elementType: ExcalidrawElement["type"]; | ||||
|   setAppState: React.Component<any, AppState>["setState"]; | ||||
|   onImageAction: (data: { pointerType: PointerType | null }) => void; | ||||
|   appState: AppState; | ||||
|   isLibraryOpen: boolean; | ||||
| }) => ( | ||||
|   <> | ||||
|     {SHAPES.map(({ value, icon, key }, index) => { | ||||
|       const label = t(`toolBar.${value}`); | ||||
|       const letter = key && (typeof key === "string" ? key : key[0]); | ||||
|       const shortcut = letter | ||||
|         ? `${capitalizeString(letter)} ${t("helpDialog.or")} ${index + 1}` | ||||
|         : `${index + 1}`; | ||||
|       const letter = typeof key === "string" ? key : key[0]; | ||||
|       const shortcut = `${capitalizeString(letter)} ${t("helpDialog.or")} ${ | ||||
|         index + 1 | ||||
|       }`; | ||||
|       return ( | ||||
|         <ToolButton | ||||
|           className="Shape" | ||||
|           key={value} | ||||
|           type="radio" | ||||
|           icon={icon} | ||||
|           checked={activeTool.type === value} | ||||
|           checked={elementType === value} | ||||
|           name="editor-current-shape" | ||||
|           title={`${capitalizeString(label)} — ${shortcut}`} | ||||
|           keyBindingLabel={`${index + 1}`} | ||||
|           aria-label={capitalizeString(label)} | ||||
|           aria-keyshortcuts={shortcut} | ||||
|           data-testid={value} | ||||
|           onPointerDown={({ pointerType }) => { | ||||
|             if (!appState.penDetected && pointerType === "pen") { | ||||
|               setAppState({ | ||||
|                 penDetected: true, | ||||
|                 penMode: true, | ||||
|               }); | ||||
|             } | ||||
|           }} | ||||
|           onChange={({ pointerType }) => { | ||||
|             if (appState.activeTool.type !== value) { | ||||
|               trackEvent("toolbar", value, "ui"); | ||||
|             } | ||||
|             const nextActiveTool = { ...activeTool, type: value }; | ||||
|           onChange={() => { | ||||
|             setAppState({ | ||||
|               activeTool: nextActiveTool, | ||||
|               elementType: value, | ||||
|               multiElement: null, | ||||
|               selectedElementIds: {}, | ||||
|             }); | ||||
|             setCursorForShape(canvas, { | ||||
|               ...appState, | ||||
|               activeTool: nextActiveTool, | ||||
|             }); | ||||
|             if (value === "image") { | ||||
|               onImageAction({ pointerType }); | ||||
|             } | ||||
|             setCursorForShape(canvas, value); | ||||
|             setAppState({}); | ||||
|           }} | ||||
|         /> | ||||
|       ); | ||||
|     })} | ||||
|     <ToolButton | ||||
|       className="Shape ToolIcon_type_button__library" | ||||
|       type="button" | ||||
|       icon={LIBRARY_ICON} | ||||
|       name="editor-library" | ||||
|       keyBindingLabel="9" | ||||
|       aria-keyshortcuts="9" | ||||
|       title={`${capitalizeString(t("toolBar.library"))} — 9`} | ||||
|       aria-label={capitalizeString(t("toolBar.library"))} | ||||
|       onClick={() => { | ||||
|         setAppState({ isLibraryOpen: !isLibraryOpen }); | ||||
|       }} | ||||
|     /> | ||||
|   </> | ||||
| ); | ||||
|  | ||||
| @@ -258,9 +226,12 @@ export const ZoomActions = ({ | ||||
| }) => ( | ||||
|   <Stack.Col gap={1}> | ||||
|     <Stack.Row gap={1} align="center"> | ||||
|       {renderAction("zoomOut")} | ||||
|       {renderAction("zoomIn")} | ||||
|       {renderAction("zoomOut")} | ||||
|       {renderAction("resetZoom")} | ||||
|       <div style={{ marginInlineStart: 4 }}> | ||||
|         {(zoom.value * 100).toFixed(0)}% | ||||
|       </div> | ||||
|     </Stack.Row> | ||||
|   </Stack.Col> | ||||
| ); | ||||
|   | ||||
| @@ -1,21 +0,0 @@ | ||||
| .excalidraw { | ||||
|   .ActiveFile { | ||||
|     .ActiveFile__fileName { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|  | ||||
|       span { | ||||
|         text-overflow: ellipsis; | ||||
|         overflow: hidden; | ||||
|         white-space: nowrap; | ||||
|         width: 9.3em; | ||||
|       } | ||||
|  | ||||
|       svg { | ||||
|         width: 1.15em; | ||||
|         margin-inline-end: 0.3em; | ||||
|         transform: scaleY(0.9); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,28 +0,0 @@ | ||||
| import Stack from "../components/Stack"; | ||||
| import { ToolButton } from "../components/ToolButton"; | ||||
| import { save, file } from "../components/icons"; | ||||
| import { t } from "../i18n"; | ||||
|  | ||||
| import "./ActiveFile.scss"; | ||||
|  | ||||
| type ActiveFileProps = { | ||||
|   fileName?: string; | ||||
|   onSave: () => void; | ||||
| }; | ||||
|  | ||||
| export const ActiveFile = ({ fileName, onSave }: ActiveFileProps) => ( | ||||
|   <Stack.Row className="ActiveFile" gap={1} align="center"> | ||||
|     <span className="ActiveFile__fileName"> | ||||
|       {file} | ||||
|       <span>{fileName}</span> | ||||
|     </span> | ||||
|     <ToolButton | ||||
|       type="icon" | ||||
|       icon={save} | ||||
|       title={t("buttons.save")} | ||||
|       aria-label={t("buttons.save")} | ||||
|       onClick={onSave} | ||||
|       data-testid="save-button" | ||||
|     /> | ||||
|   </Stack.Row> | ||||
| ); | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -16,5 +16,10 @@ export const BackgroundPickerAndDarkModeToggle = ({ | ||||
|   <div style={{ display: "flex" }}> | ||||
|     {actionManager.renderAction("changeViewBackgroundColor")} | ||||
|     {showThemeBtn && actionManager.renderAction("toggleTheme")} | ||||
|     {appState.fileHandle && ( | ||||
|       <div style={{ marginInlineStart: "0.25rem" }}> | ||||
|         {actionManager.renderAction("saveScene")} | ||||
|       </div> | ||||
|     )} | ||||
|   </div> | ||||
| ); | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import React from "react"; | ||||
| import clsx from "clsx"; | ||||
|  | ||||
| // TODO: It might be "clever" to add option.icon to the existing component <ButtonSelect /> | ||||
| @@ -7,7 +8,7 @@ export const ButtonIconSelect = <T extends Object>({ | ||||
|   onChange, | ||||
|   group, | ||||
| }: { | ||||
|   options: { value: T; text: string; icon: JSX.Element; testId?: string }[]; | ||||
|   options: { value: T; text: string; icon: JSX.Element }[]; | ||||
|   value: T | null; | ||||
|   onChange: (value: T) => void; | ||||
|   group: string; | ||||
| @@ -24,7 +25,6 @@ export const ButtonIconSelect = <T extends Object>({ | ||||
|           name={group} | ||||
|           onChange={() => onChange(option.value)} | ||||
|           checked={value === option.value} | ||||
|           data-testid={option.testId} | ||||
|         /> | ||||
|         {option.icon} | ||||
|       </label> | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import React from "react"; | ||||
| import clsx from "clsx"; | ||||
|  | ||||
| export const ButtonSelect = <T extends Object>({ | ||||
|   | ||||
| @@ -48,10 +48,6 @@ | ||||
|       .ToolIcon__label { | ||||
|         color: $oc-white; | ||||
|       } | ||||
|  | ||||
|       .Spinner { | ||||
|         --spinner-color: #fff; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -3,22 +3,15 @@ import OpenColor from "open-color"; | ||||
| import "./Card.scss"; | ||||
|  | ||||
| export const Card: React.FC<{ | ||||
|   color: keyof OpenColor | "primary"; | ||||
|   color: keyof OpenColor; | ||||
| }> = ({ children, color }) => { | ||||
|   return ( | ||||
|     <div | ||||
|       className="Card" | ||||
|       style={{ | ||||
|         ["--card-color" as any]: | ||||
|           color === "primary" ? "var(--color-primary)" : OpenColor[color][7], | ||||
|         ["--card-color-darker" as any]: | ||||
|           color === "primary" | ||||
|             ? "var(--color-primary-darker)" | ||||
|             : OpenColor[color][8], | ||||
|         ["--card-color-darkest" as any]: | ||||
|           color === "primary" | ||||
|             ? "var(--color-primary-darkest)" | ||||
|             : OpenColor[color][9], | ||||
|         ["--card-color" as any]: OpenColor[color][7], | ||||
|         ["--card-color-darker" as any]: OpenColor[color][8], | ||||
|         ["--card-color-darkest" as any]: OpenColor[color][9], | ||||
|       }} | ||||
|     > | ||||
|       {children} | ||||
|   | ||||
| @@ -2,20 +2,16 @@ | ||||
|  | ||||
| .excalidraw { | ||||
|   .Checkbox { | ||||
|     margin: 4px 0.3em; | ||||
|     margin: 3px 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) { | ||||
|     &:hover:not(.is-checked) .Checkbox-box { | ||||
|       box-shadow: 0 0 0 2px #{$oc-blue-4}; | ||||
|     } | ||||
|  | ||||
|     &:hover:not(.is-checked) .Checkbox-box:not(:focus) { | ||||
|       svg { | ||||
|         display: block; | ||||
|         opacity: 0.3; | ||||
| @@ -81,7 +77,7 @@ | ||||
|       align-items: center; | ||||
|     } | ||||
|  | ||||
|     .excalidraw-tooltip-icon { | ||||
|     .Tooltip-icon { | ||||
|       width: 1em; | ||||
|       height: 1em; | ||||
|     } | ||||
|   | ||||
| @@ -1,4 +1,3 @@ | ||||
| import React from "react"; | ||||
| import clsx from "clsx"; | ||||
| import { checkIcon } from "./icons"; | ||||
|  | ||||
| @@ -6,19 +5,16 @@ import "./CheckboxItem.scss"; | ||||
|  | ||||
| export const CheckboxItem: React.FC<{ | ||||
|   checked: boolean; | ||||
|   onChange: (checked: boolean, event: React.MouseEvent) => void; | ||||
|   className?: string; | ||||
| }> = ({ children, checked, onChange, className }) => { | ||||
|   onChange: (checked: boolean) => void; | ||||
| }> = ({ children, checked, onChange }) => { | ||||
|   return ( | ||||
|     <div | ||||
|       className={clsx("Checkbox", className, { "is-checked": checked })} | ||||
|       className={clsx("Checkbox", { "is-checked": checked })} | ||||
|       onClick={(event) => { | ||||
|         onChange(!checked, event); | ||||
|         ( | ||||
|           (event.currentTarget as HTMLDivElement).querySelector( | ||||
|             ".Checkbox-box", | ||||
|           ) as HTMLButtonElement | ||||
|         ).focus(); | ||||
|         onChange(!checked); | ||||
|         ((event.currentTarget as HTMLDivElement).querySelector( | ||||
|           ".Checkbox-box", | ||||
|         ) as HTMLButtonElement).focus(); | ||||
|       }} | ||||
|     > | ||||
|       <button className="Checkbox-box" role="checkbox" aria-checked={checked}> | ||||
|   | ||||
| @@ -1,43 +0,0 @@ | ||||
| import { useState } from "react"; | ||||
| import { t } from "../i18n"; | ||||
| import { useDeviceType } from "./App"; | ||||
| import { trash } from "./icons"; | ||||
| import { ToolButton } from "./ToolButton"; | ||||
|  | ||||
| import ConfirmDialog from "./ConfirmDialog"; | ||||
|  | ||||
| const ClearCanvas = ({ onConfirm }: { onConfirm: () => void }) => { | ||||
|   const [showDialog, setShowDialog] = useState(false); | ||||
|   const toggleDialog = () => { | ||||
|     setShowDialog(!showDialog); | ||||
|   }; | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <ToolButton | ||||
|         type="button" | ||||
|         icon={trash} | ||||
|         title={t("buttons.clearReset")} | ||||
|         aria-label={t("buttons.clearReset")} | ||||
|         showAriaLabel={useDeviceType().isMobile} | ||||
|         onClick={toggleDialog} | ||||
|         data-testid="clear-canvas-button" | ||||
|       /> | ||||
|  | ||||
|       {showDialog && ( | ||||
|         <ConfirmDialog | ||||
|           onConfirm={() => { | ||||
|             onConfirm(); | ||||
|             toggleDialog(); | ||||
|           }} | ||||
|           onCancel={toggleDialog} | ||||
|           title={t("clearCanvasDialog.title")} | ||||
|         > | ||||
|           <p className="clear-canvas__content"> {t("alerts.clearReset")}</p> | ||||
|         </ConfirmDialog> | ||||
|       )} | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default ClearCanvas; | ||||
| @@ -1,7 +1,8 @@ | ||||
| import React from "react"; | ||||
| import clsx from "clsx"; | ||||
| import { ToolButton } from "./ToolButton"; | ||||
| import { t } from "../i18n"; | ||||
| import { useDeviceType } from "../components/App"; | ||||
| import { useIsMobile } from "../components/App"; | ||||
| import { users } from "./icons"; | ||||
|  | ||||
| import "./CollabButton.scss"; | ||||
| @@ -26,7 +27,7 @@ const CollabButton = ({ | ||||
|         type="button" | ||||
|         title={t("labels.liveCollaboration")} | ||||
|         aria-label={t("labels.liveCollaboration")} | ||||
|         showAriaLabel={useDeviceType().isMobile} | ||||
|         showAriaLabel={useIsMobile()} | ||||
|       > | ||||
|         {collaboratorCount > 0 && ( | ||||
|           <div className="CollabButton-collaborators">{collaboratorCount}</div> | ||||
|   | ||||
| @@ -46,7 +46,7 @@ | ||||
|     top: -11px; | ||||
|   } | ||||
|  | ||||
|   .color-picker-content--default { | ||||
|   .color-picker-content { | ||||
|     padding: 0.5rem; | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(5, auto); | ||||
| @@ -59,26 +59,6 @@ | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .color-picker-content--canvas { | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     padding: 0.25rem; | ||||
|  | ||||
|     &-title { | ||||
|       color: $oc-gray-6; | ||||
|       font-size: 12px; | ||||
|       padding: 0 0.25rem; | ||||
|     } | ||||
|  | ||||
|     &-colors { | ||||
|       padding: 0.5rem 0; | ||||
|  | ||||
|       .color-picker-swatch { | ||||
|         margin: 0 0.25rem; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .color-picker-content .color-input-container { | ||||
|     grid-column: 1 / span 5; | ||||
|   } | ||||
|   | ||||
| @@ -1,59 +1,11 @@ | ||||
| import React from "react"; | ||||
| import { Popover } from "./Popover"; | ||||
| import { isTransparent } from "../utils"; | ||||
|  | ||||
| import "./ColorPicker.scss"; | ||||
| import { isArrowKey, KEYS } from "../keys"; | ||||
| import { t, getLanguage } from "../i18n"; | ||||
| import { isWritableElement } from "../utils"; | ||||
| import colors from "../colors"; | ||||
| import { ExcalidrawElement } from "../element/types"; | ||||
| import { AppState } from "../types"; | ||||
|  | ||||
| const MAX_CUSTOM_COLORS = 5; | ||||
| const MAX_DEFAULT_COLORS = 15; | ||||
|  | ||||
| export const getCustomColors = ( | ||||
|   elements: readonly ExcalidrawElement[], | ||||
|   type: "elementBackground" | "elementStroke", | ||||
| ) => { | ||||
|   const customColors: string[] = []; | ||||
|   const updatedElements = elements | ||||
|     .filter((element) => !element.isDeleted) | ||||
|     .sort((ele1, ele2) => ele2.updated - ele1.updated); | ||||
|  | ||||
|   let index = 0; | ||||
|   const elementColorTypeMap = { | ||||
|     elementBackground: "backgroundColor", | ||||
|     elementStroke: "strokeColor", | ||||
|   }; | ||||
|   const colorType = elementColorTypeMap[type] as | ||||
|     | "backgroundColor" | ||||
|     | "strokeColor"; | ||||
|   while ( | ||||
|     index < updatedElements.length && | ||||
|     customColors.length < MAX_CUSTOM_COLORS | ||||
|   ) { | ||||
|     const element = updatedElements[index]; | ||||
|  | ||||
|     if ( | ||||
|       customColors.length < MAX_CUSTOM_COLORS && | ||||
|       isCustomColor(element[colorType], type) && | ||||
|       !customColors.includes(element[colorType]) | ||||
|     ) { | ||||
|       customColors.push(element[colorType]); | ||||
|     } | ||||
|     index++; | ||||
|   } | ||||
|   return customColors; | ||||
| }; | ||||
|  | ||||
| const isCustomColor = ( | ||||
|   color: string, | ||||
|   type: "elementBackground" | "elementStroke", | ||||
| ) => { | ||||
|   return !colors[type].includes(color); | ||||
| }; | ||||
|  | ||||
| const isValidColor = (color: string) => { | ||||
|   const style = new Option().style; | ||||
| @@ -62,7 +14,7 @@ const isValidColor = (color: string) => { | ||||
| }; | ||||
|  | ||||
| const getColor = (color: string): string | null => { | ||||
|   if (isTransparent(color)) { | ||||
|   if (color === "transparent") { | ||||
|     return color; | ||||
|   } | ||||
|  | ||||
| @@ -82,7 +34,6 @@ const keyBindings = [ | ||||
|   ["1", "2", "3", "4", "5"], | ||||
|   ["q", "w", "e", "r", "t"], | ||||
|   ["a", "s", "d", "f", "g"], | ||||
|   ["z", "x", "c", "v", "b"], | ||||
| ].flat(); | ||||
|  | ||||
| const Picker = ({ | ||||
| @@ -93,7 +44,6 @@ const Picker = ({ | ||||
|   label, | ||||
|   showInput = true, | ||||
|   type, | ||||
|   elements, | ||||
| }: { | ||||
|   colors: string[]; | ||||
|   color: string | null; | ||||
| @@ -102,20 +52,12 @@ const Picker = ({ | ||||
|   label: string; | ||||
|   showInput: boolean; | ||||
|   type: "canvasBackground" | "elementBackground" | "elementStroke"; | ||||
|   elements: readonly ExcalidrawElement[]; | ||||
| }) => { | ||||
|   const firstItem = React.useRef<HTMLButtonElement>(); | ||||
|   const activeItem = React.useRef<HTMLButtonElement>(); | ||||
|   const gallery = React.useRef<HTMLDivElement>(); | ||||
|   const colorInput = React.useRef<HTMLInputElement>(); | ||||
|  | ||||
|   const [customColors] = React.useState(() => { | ||||
|     if (type === "canvasBackground") { | ||||
|       return []; | ||||
|     } | ||||
|     return getCustomColors(elements, type); | ||||
|   }); | ||||
|  | ||||
|   React.useEffect(() => { | ||||
|     // After the component is first mounted focus on first input | ||||
|     if (activeItem.current) { | ||||
| @@ -142,42 +84,23 @@ const Picker = ({ | ||||
|     } else if (isArrowKey(event.key)) { | ||||
|       const { activeElement } = document; | ||||
|       const isRTL = getLanguage().rtl; | ||||
|       let isCustom = false; | ||||
|       let index = Array.prototype.indexOf.call( | ||||
|         gallery!.current!.querySelector(".color-picker-content--default")! | ||||
|           .children, | ||||
|       const index = Array.prototype.indexOf.call( | ||||
|         gallery!.current!.children, | ||||
|         activeElement, | ||||
|       ); | ||||
|       if (index === -1) { | ||||
|         index = Array.prototype.indexOf.call( | ||||
|           gallery!.current!.querySelector( | ||||
|             ".color-picker-content--canvas-colors", | ||||
|           )!.children, | ||||
|           activeElement, | ||||
|         ); | ||||
|         if (index !== -1) { | ||||
|           isCustom = true; | ||||
|         } | ||||
|       } | ||||
|       const parentSelector = isCustom | ||||
|         ? gallery!.current!.querySelector( | ||||
|             ".color-picker-content--canvas-colors", | ||||
|           )! | ||||
|         : gallery!.current!.querySelector(".color-picker-content--default")!; | ||||
|  | ||||
|       if (index !== -1) { | ||||
|         const length = parentSelector!.children.length - (showInput ? 1 : 0); | ||||
|         const length = gallery!.current!.children.length - (showInput ? 1 : 0); | ||||
|         const nextIndex = | ||||
|           event.key === (isRTL ? KEYS.ARROW_LEFT : KEYS.ARROW_RIGHT) | ||||
|             ? (index + 1) % length | ||||
|             : event.key === (isRTL ? KEYS.ARROW_RIGHT : KEYS.ARROW_LEFT) | ||||
|             ? (length + index - 1) % length | ||||
|             : !isCustom && event.key === KEYS.ARROW_DOWN | ||||
|             : event.key === KEYS.ARROW_DOWN | ||||
|             ? (index + 5) % length | ||||
|             : !isCustom && event.key === KEYS.ARROW_UP | ||||
|             : event.key === KEYS.ARROW_UP | ||||
|             ? (length + index - 5) % length | ||||
|             : index; | ||||
|         (parentSelector!.children![nextIndex] as HTMLElement)?.focus(); | ||||
|         (gallery!.current!.children![nextIndex] as any).focus(); | ||||
|       } | ||||
|       event.preventDefault(); | ||||
|     } else if ( | ||||
| @@ -185,15 +108,7 @@ const Picker = ({ | ||||
|       !isWritableElement(event.target) | ||||
|     ) { | ||||
|       const index = keyBindings.indexOf(event.key.toLowerCase()); | ||||
|       const isCustom = index >= MAX_DEFAULT_COLORS; | ||||
|       const parentSelector = isCustom | ||||
|         ? gallery!.current!.querySelector( | ||||
|             ".color-picker-content--canvas-colors", | ||||
|           )! | ||||
|         : gallery!.current!.querySelector(".color-picker-content--default")!; | ||||
|       const actualIndex = isCustom ? index - MAX_DEFAULT_COLORS : index; | ||||
|       (parentSelector!.children![actualIndex] as HTMLElement)?.focus(); | ||||
|  | ||||
|       (gallery!.current!.children![index] as any).focus(); | ||||
|       event.preventDefault(); | ||||
|     } else if (event.key === KEYS.ESCAPE || event.key === KEYS.ENTER) { | ||||
|       event.preventDefault(); | ||||
| @@ -203,50 +118,6 @@ const Picker = ({ | ||||
|     event.stopPropagation(); | ||||
|   }; | ||||
|  | ||||
|   const renderColors = (colors: Array<string>, custom: boolean = false) => { | ||||
|     return colors.map((_color, i) => { | ||||
|       const _colorWithoutHash = _color.replace("#", ""); | ||||
|       const keyBinding = custom | ||||
|         ? keyBindings[i + MAX_DEFAULT_COLORS] | ||||
|         : keyBindings[i]; | ||||
|       const label = custom | ||||
|         ? _colorWithoutHash | ||||
|         : t(`colors.${_colorWithoutHash}`); | ||||
|       return ( | ||||
|         <button | ||||
|           className="color-picker-swatch" | ||||
|           onClick={(event) => { | ||||
|             (event.currentTarget as HTMLButtonElement).focus(); | ||||
|             onChange(_color); | ||||
|           }} | ||||
|           title={`${label}${ | ||||
|             !isTransparent(_color) ? ` (${_color})` : "" | ||||
|           } — ${keyBinding.toUpperCase()}`} | ||||
|           aria-label={label} | ||||
|           aria-keyshortcuts={keyBindings[i]} | ||||
|           style={{ color: _color }} | ||||
|           key={_color} | ||||
|           ref={(el) => { | ||||
|             if (!custom && el && i === 0) { | ||||
|               firstItem.current = el; | ||||
|             } | ||||
|             if (el && _color === color) { | ||||
|               activeItem.current = el; | ||||
|             } | ||||
|           }} | ||||
|           onFocus={() => { | ||||
|             onChange(_color); | ||||
|           }} | ||||
|         > | ||||
|           {isTransparent(_color) ? ( | ||||
|             <div className="color-picker-transparent"></div> | ||||
|           ) : undefined} | ||||
|           <span className="color-picker-keybinding">{keyBinding}</span> | ||||
|         </button> | ||||
|       ); | ||||
|     }); | ||||
|   }; | ||||
|  | ||||
|   return ( | ||||
|     <div | ||||
|       className={`color-picker color-picker-type-${type}`} | ||||
| @@ -266,20 +137,36 @@ const Picker = ({ | ||||
|         }} | ||||
|         tabIndex={0} | ||||
|       > | ||||
|         <div className="color-picker-content--default"> | ||||
|           {renderColors(colors)} | ||||
|         </div> | ||||
|         {!!customColors.length && ( | ||||
|           <div className="color-picker-content--canvas"> | ||||
|             <span className="color-picker-content--canvas-title"> | ||||
|               {t("labels.canvasColors")} | ||||
|             </span> | ||||
|             <div className="color-picker-content--canvas-colors"> | ||||
|               {renderColors(customColors, true)} | ||||
|             </div> | ||||
|           </div> | ||||
|         )} | ||||
|  | ||||
|         {colors.map((_color, i) => ( | ||||
|           <button | ||||
|             className="color-picker-swatch" | ||||
|             onClick={(event) => { | ||||
|               (event.currentTarget as HTMLButtonElement).focus(); | ||||
|               onChange(_color); | ||||
|             }} | ||||
|             title={`${_color} — ${keyBindings[i].toUpperCase()}`} | ||||
|             aria-label={_color} | ||||
|             aria-keyshortcuts={keyBindings[i]} | ||||
|             style={{ color: _color }} | ||||
|             key={_color} | ||||
|             ref={(el) => { | ||||
|               if (el && i === 0) { | ||||
|                 firstItem.current = el; | ||||
|               } | ||||
|               if (el && _color === color) { | ||||
|                 activeItem.current = el; | ||||
|               } | ||||
|             }} | ||||
|             onFocus={() => { | ||||
|               onChange(_color); | ||||
|             }} | ||||
|           > | ||||
|             {_color === "transparent" ? ( | ||||
|               <div className="color-picker-transparent"></div> | ||||
|             ) : undefined} | ||||
|             <span className="color-picker-keybinding">{keyBindings[i]}</span> | ||||
|           </button> | ||||
|         ))} | ||||
|         {showInput && ( | ||||
|           <ColorInput | ||||
|             color={color} | ||||
| @@ -351,20 +238,13 @@ export const ColorPicker = ({ | ||||
|   color, | ||||
|   onChange, | ||||
|   label, | ||||
|   isActive, | ||||
|   setActive, | ||||
|   elements, | ||||
|   appState, | ||||
| }: { | ||||
|   type: "canvasBackground" | "elementBackground" | "elementStroke"; | ||||
|   color: string | null; | ||||
|   onChange: (color: string) => void; | ||||
|   label: string; | ||||
|   isActive: boolean; | ||||
|   setActive: (active: boolean) => void; | ||||
|   elements: readonly ExcalidrawElement[]; | ||||
|   appState: AppState; | ||||
| }) => { | ||||
|   const [isActive, setActive] = React.useState(false); | ||||
|   const pickerButton = React.useRef<HTMLButtonElement>(null); | ||||
|  | ||||
|   return ( | ||||
| @@ -405,7 +285,6 @@ export const ColorPicker = ({ | ||||
|               label={label} | ||||
|               showInput={false} | ||||
|               type={type} | ||||
|               elements={elements} | ||||
|             /> | ||||
|           </Popover> | ||||
|         ) : null} | ||||
|   | ||||
| @@ -1,37 +0,0 @@ | ||||
| @import "../css/variables.module"; | ||||
|  | ||||
| .excalidraw { | ||||
|   .confirm-dialog { | ||||
|     &-buttons { | ||||
|       display: flex; | ||||
|       padding: 0.2rem 0; | ||||
|       justify-content: flex-end; | ||||
|     } | ||||
|     .ToolIcon__icon { | ||||
|       min-width: 2.5rem; | ||||
|       width: auto; | ||||
|       font-size: 1rem; | ||||
|     } | ||||
|  | ||||
|     .ToolIcon_type_button { | ||||
|       margin-left: 0.8rem; | ||||
|       padding: 0 0.5rem; | ||||
|     } | ||||
|  | ||||
|     &__content { | ||||
|       font-size: 1rem; | ||||
|     } | ||||
|  | ||||
|     &--confirm.ToolIcon_type_button { | ||||
|       background-color: $oc-red-6; | ||||
|  | ||||
|       &:hover { | ||||
|         background-color: $oc-red-8; | ||||
|       } | ||||
|  | ||||
|       .ToolIcon__icon { | ||||
|         color: $oc-white; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,52 +0,0 @@ | ||||
| import { t } from "../i18n"; | ||||
| import { Dialog, DialogProps } from "./Dialog"; | ||||
| import { ToolButton } from "./ToolButton"; | ||||
|  | ||||
| import "./ConfirmDialog.scss"; | ||||
|  | ||||
| interface Props extends Omit<DialogProps, "onCloseRequest"> { | ||||
|   onConfirm: () => void; | ||||
|   onCancel: () => void; | ||||
|   confirmText?: string; | ||||
|   cancelText?: string; | ||||
| } | ||||
| const ConfirmDialog = (props: Props) => { | ||||
|   const { | ||||
|     onConfirm, | ||||
|     onCancel, | ||||
|     children, | ||||
|     confirmText = t("buttons.confirm"), | ||||
|     cancelText = t("buttons.cancel"), | ||||
|     className = "", | ||||
|     ...rest | ||||
|   } = props; | ||||
|   return ( | ||||
|     <Dialog | ||||
|       onCloseRequest={onCancel} | ||||
|       small={true} | ||||
|       {...rest} | ||||
|       className={`confirm-dialog ${className}`} | ||||
|     > | ||||
|       {children} | ||||
|       <div className="confirm-dialog-buttons"> | ||||
|         <ToolButton | ||||
|           type="button" | ||||
|           title={cancelText} | ||||
|           aria-label={cancelText} | ||||
|           label={cancelText} | ||||
|           onClick={onCancel} | ||||
|           className="confirm-dialog--cancel" | ||||
|         /> | ||||
|         <ToolButton | ||||
|           type="button" | ||||
|           title={confirmText} | ||||
|           aria-label={confirmText} | ||||
|           label={confirmText} | ||||
|           onClick={onConfirm} | ||||
|           className="confirm-dialog--confirm" | ||||
|         /> | ||||
|       </div> | ||||
|     </Dialog> | ||||
|   ); | ||||
| }; | ||||
| export default ConfirmDialog; | ||||
| @@ -1,3 +1,4 @@ | ||||
| import React from "react"; | ||||
| import { render, unmountComponentAtNode } from "react-dom"; | ||||
| import clsx from "clsx"; | ||||
| import { Popover } from "./Popover"; | ||||
| @@ -11,7 +12,6 @@ import { | ||||
| import { Action } from "../actions/types"; | ||||
| import { ActionManager } from "../actions/manager"; | ||||
| import { AppState } from "../types"; | ||||
| import { NonDeletedExcalidrawElement } from "../element/types"; | ||||
|  | ||||
| export type ContextMenuOption = "separator" | Action; | ||||
|  | ||||
| @@ -22,7 +22,6 @@ type ContextMenuProps = { | ||||
|   left: number; | ||||
|   actionManager: ActionManager; | ||||
|   appState: Readonly<AppState>; | ||||
|   elements: readonly NonDeletedExcalidrawElement[]; | ||||
| }; | ||||
|  | ||||
| const ContextMenu = ({ | ||||
| @@ -32,7 +31,6 @@ const ContextMenu = ({ | ||||
|   left, | ||||
|   actionManager, | ||||
|   appState, | ||||
|   elements, | ||||
| }: ContextMenuProps) => { | ||||
|   return ( | ||||
|     <Popover | ||||
| @@ -40,10 +38,6 @@ const ContextMenu = ({ | ||||
|       top={top} | ||||
|       left={left} | ||||
|       fitInViewport={true} | ||||
|       offsetLeft={appState.offsetLeft} | ||||
|       offsetTop={appState.offsetTop} | ||||
|       viewportWidth={appState.width} | ||||
|       viewportHeight={appState.height} | ||||
|     > | ||||
|       <ul | ||||
|         className="context-menu" | ||||
| @@ -55,14 +49,9 @@ const ContextMenu = ({ | ||||
|           } | ||||
|  | ||||
|           const actionName = option.name; | ||||
|           let label = ""; | ||||
|           if (option.contextItemLabel) { | ||||
|             if (typeof option.contextItemLabel === "function") { | ||||
|               label = t(option.contextItemLabel(elements, appState)); | ||||
|             } else { | ||||
|               label = t(option.contextItemLabel); | ||||
|             } | ||||
|           } | ||||
|           const label = option.contextItemLabel | ||||
|             ? t(option.contextItemLabel) | ||||
|             : ""; | ||||
|           return ( | ||||
|             <li key={idx} data-testid={actionName} onClick={onCloseRequest}> | ||||
|               <button | ||||
| @@ -70,9 +59,7 @@ const ContextMenu = ({ | ||||
|                   dangerous: actionName === "deleteSelectedElements", | ||||
|                   checkmark: option.checked?.(appState), | ||||
|                 })} | ||||
|                 onClick={() => | ||||
|                   actionManager.executeAction(option, "contextMenu") | ||||
|                 } | ||||
|                 onClick={() => actionManager.executeAction(option)} | ||||
|               > | ||||
|                 <div className="context-menu-option__label">{label}</div> | ||||
|                 <kbd className="context-menu-option__shortcut"> | ||||
| @@ -111,7 +98,6 @@ type ContextMenuParams = { | ||||
|   actionManager: ContextMenuProps["actionManager"]; | ||||
|   appState: Readonly<AppState>; | ||||
|   container: HTMLElement; | ||||
|   elements: readonly NonDeletedExcalidrawElement[]; | ||||
| }; | ||||
|  | ||||
| const handleClose = (container: HTMLElement) => { | ||||
| @@ -140,7 +126,6 @@ export default { | ||||
|           onCloseRequest={() => handleClose(params.container)} | ||||
|           actionManager={params.actionManager} | ||||
|           appState={params.appState} | ||||
|           elements={params.elements} | ||||
|         />, | ||||
|         getContextMenuNode(params.container), | ||||
|       ); | ||||
|   | ||||
| @@ -1,15 +1,16 @@ | ||||
| import "./ToolIcon.scss"; | ||||
|  | ||||
| import React from "react"; | ||||
| import { t } from "../i18n"; | ||||
| import { ToolButton } from "./ToolButton"; | ||||
| import { THEME } from "../constants"; | ||||
| import { Theme } from "../element/types"; | ||||
|  | ||||
| export type Appearence = "light" | "dark"; | ||||
|  | ||||
| // We chose to use only explicit toggle and not a third option for system value, | ||||
| // but this could be added in the future. | ||||
| export const DarkModeToggle = (props: { | ||||
|   value: Theme; | ||||
|   onChange: (value: Theme) => void; | ||||
|   value: Appearence; | ||||
|   onChange: (value: Appearence) => void; | ||||
|   title?: string; | ||||
| }) => { | ||||
|   const title = | ||||
| @@ -19,12 +20,10 @@ export const DarkModeToggle = (props: { | ||||
|   return ( | ||||
|     <ToolButton | ||||
|       type="icon" | ||||
|       icon={props.value === THEME.LIGHT ? ICONS.MOON : ICONS.SUN} | ||||
|       icon={props.value === "light" ? ICONS.MOON : ICONS.SUN} | ||||
|       title={title} | ||||
|       aria-label={title} | ||||
|       onClick={() => | ||||
|         props.onChange(props.value === THEME.DARK ? THEME.LIGHT : THEME.DARK) | ||||
|       } | ||||
|       onClick={() => props.onChange(props.value === "dark" ? "light" : "dark")} | ||||
|       data-testid="toggle-dark-mode" | ||||
|     /> | ||||
|   ); | ||||
|   | ||||
| @@ -2,7 +2,7 @@ import clsx from "clsx"; | ||||
| import React, { useEffect, useState } from "react"; | ||||
| import { useCallbackRefState } from "../hooks/useCallbackRefState"; | ||||
| import { t } from "../i18n"; | ||||
| import { useExcalidrawContainer, useDeviceType } from "../components/App"; | ||||
| import { useIsMobile } from "../components/App"; | ||||
| import { KEYS } from "../keys"; | ||||
| import "./Dialog.scss"; | ||||
| import { back, close } from "./icons"; | ||||
| @@ -10,7 +10,7 @@ import { Island } from "./Island"; | ||||
| import { Modal } from "./Modal"; | ||||
| import { AppState } from "../types"; | ||||
|  | ||||
| export interface DialogProps { | ||||
| export const Dialog = (props: { | ||||
|   children: React.ReactNode; | ||||
|   className?: string; | ||||
|   small?: boolean; | ||||
| @@ -18,13 +18,9 @@ export interface DialogProps { | ||||
|   title: React.ReactNode; | ||||
|   autofocus?: boolean; | ||||
|   theme?: AppState["theme"]; | ||||
|   closeOnClickOutside?: boolean; | ||||
| } | ||||
|  | ||||
| export const Dialog = (props: DialogProps) => { | ||||
| }) => { | ||||
|   const [islandNode, setIslandNode] = useCallbackRefState<HTMLDivElement>(); | ||||
|   const [lastActiveElement] = useState(document.activeElement); | ||||
|   const { id } = useExcalidrawContainer(); | ||||
|  | ||||
|   useEffect(() => { | ||||
|     if (!islandNode) { | ||||
| @@ -84,17 +80,16 @@ export const Dialog = (props: DialogProps) => { | ||||
|       maxWidth={props.small ? 550 : 800} | ||||
|       onCloseRequest={onClose} | ||||
|       theme={props.theme} | ||||
|       closeOnClickOutside={props.closeOnClickOutside} | ||||
|     > | ||||
|       <Island ref={setIslandNode}> | ||||
|         <h2 id={`${id}-dialog-title`} className="Dialog__title"> | ||||
|         <h2 id="dialog-title" className="Dialog__title"> | ||||
|           <span className="Dialog__titleContent">{props.title}</span> | ||||
|           <button | ||||
|             className="Modal__close" | ||||
|             onClick={onClose} | ||||
|             aria-label={t("buttons.close")} | ||||
|           > | ||||
|             {useDeviceType().isMobile ? back : close} | ||||
|             {useIsMobile() ? back : close} | ||||
|           </button> | ||||
|         </h2> | ||||
|         <div className="Dialog__content">{props.children}</div> | ||||
|   | ||||
| @@ -12,7 +12,7 @@ export const ErrorDialog = ({ | ||||
|   onClose?: () => void; | ||||
| }) => { | ||||
|   const [modalIsShown, setModalIsShown] = useState(!!message); | ||||
|   const { container: excalidrawContainer } = useExcalidrawContainer(); | ||||
|   const excalidrawContainer = useExcalidrawContainer(); | ||||
|  | ||||
|   const handleClose = React.useCallback(() => { | ||||
|     setModalIsShown(false); | ||||
|   | ||||
| @@ -97,8 +97,7 @@ | ||||
|  | ||||
|     border-radius: 1rem; | ||||
|     background-color: var(--button-color); | ||||
|     box-shadow: 0 3px 5px -1px rgba(0, 0, 0, 0.28), | ||||
|       0 6px 10px 0 rgba(0, 0, 0, 0.14); | ||||
|     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; | ||||
| @@ -109,7 +108,7 @@ | ||||
|     } | ||||
|     &:active { | ||||
|       background-color: var(--button-color-darkest); | ||||
|       box-shadow: none; | ||||
|       box-shadow: 0 3px 5px -1px rgb(0 0 0 / 20%), 0 6px 10px 0 rgb(0 0 0 / 14%); | ||||
|     } | ||||
|  | ||||
|     svg { | ||||
|   | ||||
| @@ -139,7 +139,7 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => { | ||||
|         <Section title={t("helpDialog.shortcuts")}> | ||||
|           <Columns> | ||||
|             <Column> | ||||
|               <ShortcutIsland caption={t("helpDialog.tools")}> | ||||
|               <ShortcutIsland caption={t("helpDialog.shapes")}> | ||||
|                 <Shortcut | ||||
|                   label={t("toolBar.selection")} | ||||
|                   shortcuts={["V", "1"]} | ||||
| @@ -149,27 +149,14 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => { | ||||
|                   shortcuts={["R", "2"]} | ||||
|                 /> | ||||
|                 <Shortcut label={t("toolBar.diamond")} shortcuts={["D", "3"]} /> | ||||
|                 <Shortcut label={t("toolBar.ellipse")} shortcuts={["O", "4"]} /> | ||||
|                 <Shortcut label={t("toolBar.ellipse")} shortcuts={["E", "4"]} /> | ||||
|                 <Shortcut label={t("toolBar.arrow")} shortcuts={["A", "5"]} /> | ||||
|                 <Shortcut label={t("toolBar.line")} shortcuts={["P", "6"]} /> | ||||
|                 <Shortcut | ||||
|                   label={t("toolBar.freedraw")} | ||||
|                   shortcuts={["Shift + P", "X", "7"]} | ||||
|                   shortcuts={["Shift+P", "7"]} | ||||
|                 /> | ||||
|                 <Shortcut label={t("toolBar.text")} shortcuts={["T", "8"]} /> | ||||
|                 <Shortcut label={t("toolBar.image")} shortcuts={["9"]} /> | ||||
|                 <Shortcut label={t("toolBar.library")} shortcuts={["0"]} /> | ||||
|                 <Shortcut | ||||
|                   label={t("toolBar.eraser")} | ||||
|                   shortcuts={[getShortcutKey("E")]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("helpDialog.editSelectedShape")} | ||||
|                   shortcuts={[ | ||||
|                     getShortcutKey("Enter"), | ||||
|                     t("helpDialog.doubleClick"), | ||||
|                   ]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("helpDialog.textNewLine")} | ||||
|                   shortcuts={[ | ||||
| @@ -209,10 +196,6 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => { | ||||
|                   label={t("helpDialog.preventBinding")} | ||||
|                   shortcuts={[getShortcutKey("CtrlOrCmd")]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("toolBar.link")} | ||||
|                   shortcuts={[getShortcutKey("CtrlOrCmd+K")]} | ||||
|                 /> | ||||
|               </ShortcutIsland> | ||||
|               <ShortcutIsland caption={t("helpDialog.view")}> | ||||
|                 <Shortcut | ||||
| @@ -268,18 +251,6 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => { | ||||
|                   label={t("labels.multiSelect")} | ||||
|                   shortcuts={[getShortcutKey(`Shift+${t("helpDialog.click")}`)]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("helpDialog.deepSelect")} | ||||
|                   shortcuts={[ | ||||
|                     getShortcutKey(`CtrlOrCmd+${t("helpDialog.click")}`), | ||||
|                   ]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("helpDialog.deepBoxSelect")} | ||||
|                   shortcuts={[ | ||||
|                     getShortcutKey(`CtrlOrCmd+${t("helpDialog.drag")}`), | ||||
|                   ]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("labels.moveCanvas")} | ||||
|                   shortcuts={[ | ||||
| @@ -363,10 +334,6 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => { | ||||
|                     getShortcutKey(`Alt+${t("helpDialog.drag")}`), | ||||
|                   ]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("helpDialog.toggleElementLock")} | ||||
|                   shortcuts={[getShortcutKey("CtrlOrCmd+Shift+L")]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("buttons.undo")} | ||||
|                   shortcuts={[getShortcutKey("CtrlOrCmd+Z")]} | ||||
| @@ -398,22 +365,6 @@ export const HelpDialog = ({ onClose }: { onClose?: () => void }) => { | ||||
|                   label={t("labels.flipVertical")} | ||||
|                   shortcuts={[getShortcutKey("Shift+V")]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("labels.showStroke")} | ||||
|                   shortcuts={[getShortcutKey("S")]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("labels.showBackground")} | ||||
|                   shortcuts={[getShortcutKey("G")]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("labels.decreaseFontSize")} | ||||
|                   shortcuts={[getShortcutKey("CtrlOrCmd+Shift+<")]} | ||||
|                 /> | ||||
|                 <Shortcut | ||||
|                   label={t("labels.increaseFontSize")} | ||||
|                   shortcuts={[getShortcutKey("CtrlOrCmd+Shift+>")]} | ||||
|                 /> | ||||
|               </ShortcutIsland> | ||||
|             </Column> | ||||
|           </Columns> | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import React from "react"; | ||||
| import { questionCircle } from "../components/icons"; | ||||
|  | ||||
| type HelpIconProps = { | ||||
|   | ||||
| @@ -1,56 +1,37 @@ | ||||
| import React from "react"; | ||||
| import { t } from "../i18n"; | ||||
| import { NonDeletedExcalidrawElement } from "../element/types"; | ||||
| import { getSelectedElements } from "../scene"; | ||||
|  | ||||
| import "./HintViewer.scss"; | ||||
| import { AppState } from "../types"; | ||||
| import { | ||||
|   isImageElement, | ||||
|   isLinearElement, | ||||
|   isTextBindableContainer, | ||||
|   isTextElement, | ||||
| } from "../element/typeChecks"; | ||||
| import { isLinearElement } from "../element/typeChecks"; | ||||
| import { getShortcutKey } from "../utils"; | ||||
| import { isEraserActive } from "../appState"; | ||||
|  | ||||
| interface HintViewerProps { | ||||
| interface Hint { | ||||
|   appState: AppState; | ||||
|   elements: readonly NonDeletedExcalidrawElement[]; | ||||
|   isMobile: boolean; | ||||
| } | ||||
|  | ||||
| const getHints = ({ appState, elements, isMobile }: HintViewerProps) => { | ||||
|   const { activeTool, isResizing, isRotating, lastPointerDownWith } = appState; | ||||
| const getHints = ({ appState, elements }: Hint) => { | ||||
|   const { elementType, isResizing, isRotating, lastPointerDownWith } = appState; | ||||
|   const multiMode = appState.multiElement !== null; | ||||
|  | ||||
|   if (appState.isLibraryOpen) { | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   if (isEraserActive(appState)) { | ||||
|     return t("hints.eraserRevert"); | ||||
|   } | ||||
|   if (activeTool.type === "arrow" || activeTool.type === "line") { | ||||
|   if (elementType === "arrow" || elementType === "line") { | ||||
|     if (!multiMode) { | ||||
|       return t("hints.linearElement"); | ||||
|     } | ||||
|     return t("hints.linearElementMulti"); | ||||
|   } | ||||
|  | ||||
|   if (activeTool.type === "freedraw") { | ||||
|   if (elementType === "freedraw") { | ||||
|     return t("hints.freeDraw"); | ||||
|   } | ||||
|  | ||||
|   if (activeTool.type === "text") { | ||||
|   if (elementType === "text") { | ||||
|     return t("hints.text"); | ||||
|   } | ||||
|  | ||||
|   if (appState.activeTool.type === "image" && appState.pendingImageElement) { | ||||
|     return t("hints.placeImage"); | ||||
|   } | ||||
|  | ||||
|   const selectedElements = getSelectedElements(elements, appState); | ||||
|  | ||||
|   if ( | ||||
|     isResizing && | ||||
|     lastPointerDownWith === "mouse" && | ||||
| @@ -60,62 +41,29 @@ const getHints = ({ appState, elements, isMobile }: HintViewerProps) => { | ||||
|     if (isLinearElement(targetElement) && targetElement.points.length === 2) { | ||||
|       return t("hints.lockAngle"); | ||||
|     } | ||||
|     return isImageElement(targetElement) | ||||
|       ? t("hints.resizeImage") | ||||
|       : t("hints.resize"); | ||||
|     return t("hints.resize"); | ||||
|   } | ||||
|  | ||||
|   if (isRotating && lastPointerDownWith === "mouse") { | ||||
|     return t("hints.rotate"); | ||||
|   } | ||||
|  | ||||
|   if (selectedElements.length === 1 && isTextElement(selectedElements[0])) { | ||||
|     return t("hints.text_selected"); | ||||
|   } | ||||
|  | ||||
|   if (appState.editingElement && isTextElement(appState.editingElement)) { | ||||
|     return t("hints.text_editing"); | ||||
|   } | ||||
|  | ||||
|   if (activeTool.type === "selection") { | ||||
|     if ( | ||||
|       appState.draggingElement?.type === "selection" && | ||||
|       !appState.editingElement && | ||||
|       !appState.editingLinearElement | ||||
|     ) { | ||||
|       return t("hints.deepBoxSelect"); | ||||
|     } | ||||
|     if (!selectedElements.length && !isMobile) { | ||||
|       return t("hints.canvasPanning"); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (selectedElements.length === 1) { | ||||
|     if (isLinearElement(selectedElements[0])) { | ||||
|       if (appState.editingLinearElement) { | ||||
|         return appState.editingLinearElement.selectedPointsIndices | ||||
|           ? t("hints.lineEditor_pointSelected") | ||||
|           : t("hints.lineEditor_nothingSelected"); | ||||
|       } | ||||
|       return t("hints.lineEditor_info"); | ||||
|     } | ||||
|     if (isTextBindableContainer(selectedElements[0])) { | ||||
|       return t("hints.bindTextToElement"); | ||||
|   if (selectedElements.length === 1 && isLinearElement(selectedElements[0])) { | ||||
|     if (appState.editingLinearElement) { | ||||
|       return appState.editingLinearElement.activePointIndex | ||||
|         ? t("hints.lineEditor_pointSelected") | ||||
|         : t("hints.lineEditor_nothingSelected"); | ||||
|     } | ||||
|     return t("hints.lineEditor_info"); | ||||
|   } | ||||
|  | ||||
|   return null; | ||||
| }; | ||||
|  | ||||
| export const HintViewer = ({ | ||||
|   appState, | ||||
|   elements, | ||||
|   isMobile, | ||||
| }: HintViewerProps) => { | ||||
| export const HintViewer = ({ appState, elements }: Hint) => { | ||||
|   let hint = getHints({ | ||||
|     appState, | ||||
|     elements, | ||||
|     isMobile, | ||||
|   }); | ||||
|   if (!hint) { | ||||
|     return null; | ||||
|   | ||||
| @@ -22,7 +22,7 @@ | ||||
|     align-items: center; | ||||
|     justify-content: center; | ||||
|  | ||||
|     &:focus-visible { | ||||
|     &:focus { | ||||
|       outline: transparent; | ||||
|       background-color: var(--button-gray-2); | ||||
|       & svg { | ||||
| @@ -90,7 +90,7 @@ | ||||
|   .picker-content { | ||||
|     padding: 0.5rem; | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(3, auto); | ||||
|     grid-auto-flow: column; | ||||
|     grid-gap: 0.5rem; | ||||
|     border-radius: 4px; | ||||
|     :root[dir="rtl"] & { | ||||
|   | ||||
| @@ -1,24 +1,27 @@ | ||||
| import React, { useEffect, useRef, useState } from "react"; | ||||
| import { render, unmountComponentAtNode } from "react-dom"; | ||||
| import { ActionsManagerInterface } from "../actions/types"; | ||||
| import { probablySupportsClipboardBlob } from "../clipboard"; | ||||
| import { canvasToBlob } from "../data/blob"; | ||||
| import { NonDeletedExcalidrawElement } from "../element/types"; | ||||
| import { CanvasError } from "../errors"; | ||||
| import { t } from "../i18n"; | ||||
| import { useDeviceType } from "./App"; | ||||
| import { useIsMobile } from "./App"; | ||||
| import { getSelectedElements, isSomeElementSelected } from "../scene"; | ||||
| import { exportToCanvas } from "../scene/export"; | ||||
| import { AppState, BinaryFiles } from "../types"; | ||||
| import { exportToCanvas, getExportSize } from "../scene/export"; | ||||
| import { AppState } from "../types"; | ||||
| import { Dialog } from "./Dialog"; | ||||
| import { clipboard, exportImage } from "./icons"; | ||||
| import Stack from "./Stack"; | ||||
| import { ToolButton } from "./ToolButton"; | ||||
|  | ||||
| import "./ExportDialog.scss"; | ||||
| import { supported as fsSupported } from "browser-fs-access"; | ||||
| import OpenColor from "open-color"; | ||||
| import { CheckboxItem } from "./CheckboxItem"; | ||||
| import { DEFAULT_EXPORT_PADDING } from "../constants"; | ||||
| import { nativeFileSystemSupported } from "../data/filesystem"; | ||||
| import { ActionManager } from "../actions/manager"; | ||||
|  | ||||
| const scales = [1, 2, 3]; | ||||
| const defaultScale = scales.includes(devicePixelRatio) ? devicePixelRatio : 1; | ||||
|  | ||||
| const supportsContextFilters = | ||||
|   "filter" in document.createElement("canvas").getContext("2d")!; | ||||
| @@ -79,8 +82,7 @@ const ExportButton: React.FC<{ | ||||
| const ImageExportModal = ({ | ||||
|   elements, | ||||
|   appState, | ||||
|   files, | ||||
|   exportPadding = DEFAULT_EXPORT_PADDING, | ||||
|   exportPadding = 10, | ||||
|   actionManager, | ||||
|   onExportToPng, | ||||
|   onExportToSvg, | ||||
| @@ -88,21 +90,25 @@ const ImageExportModal = ({ | ||||
| }: { | ||||
|   appState: AppState; | ||||
|   elements: readonly NonDeletedExcalidrawElement[]; | ||||
|   files: BinaryFiles; | ||||
|   exportPadding?: number; | ||||
|   actionManager: ActionManager; | ||||
|   actionManager: ActionsManagerInterface; | ||||
|   onExportToPng: ExportCB; | ||||
|   onExportToSvg: ExportCB; | ||||
|   onExportToClipboard: ExportCB; | ||||
|   onCloseRequest: () => void; | ||||
| }) => { | ||||
|   const someElementIsSelected = isSomeElementSelected(elements, appState); | ||||
|   const [scale, setScale] = useState(defaultScale); | ||||
|   const [exportSelected, setExportSelected] = useState(someElementIsSelected); | ||||
|   const previewRef = useRef<HTMLDivElement>(null); | ||||
|   const { exportBackground, viewBackgroundColor } = appState; | ||||
|   const { | ||||
|     exportBackground, | ||||
|     viewBackgroundColor, | ||||
|     shouldAddWatermark, | ||||
|   } = appState; | ||||
|  | ||||
|   const exportedElements = exportSelected | ||||
|     ? getSelectedElements(elements, appState, true) | ||||
|     ? getSelectedElements(elements, appState) | ||||
|     : elements; | ||||
|  | ||||
|   useEffect(() => { | ||||
| @@ -114,29 +120,37 @@ const ImageExportModal = ({ | ||||
|     if (!previewNode) { | ||||
|       return; | ||||
|     } | ||||
|     exportToCanvas(exportedElements, appState, files, { | ||||
|       exportBackground, | ||||
|       viewBackgroundColor, | ||||
|       exportPadding, | ||||
|     }) | ||||
|       .then((canvas) => { | ||||
|         // if converting to blob fails, there's some problem that will | ||||
|         // likely prevent preview and export (e.g. canvas too big) | ||||
|         return canvasToBlob(canvas).then(() => { | ||||
|           renderPreview(canvas, previewNode); | ||||
|         }); | ||||
|       }) | ||||
|       .catch((error) => { | ||||
|         console.error(error); | ||||
|         renderPreview(new CanvasError(), previewNode); | ||||
|     try { | ||||
|       const canvas = exportToCanvas(exportedElements, appState, { | ||||
|         exportBackground, | ||||
|         viewBackgroundColor, | ||||
|         exportPadding, | ||||
|         scale, | ||||
|         shouldAddWatermark, | ||||
|       }); | ||||
|  | ||||
|       // if converting to blob fails, there's some problem that will | ||||
|       // likely prevent preview and export (e.g. canvas too big) | ||||
|       canvasToBlob(canvas) | ||||
|         .then(() => { | ||||
|           renderPreview(canvas, previewNode); | ||||
|         }) | ||||
|         .catch((error) => { | ||||
|           console.error(error); | ||||
|           renderPreview(new CanvasError(), previewNode); | ||||
|         }); | ||||
|     } catch (error) { | ||||
|       console.error(error); | ||||
|       renderPreview(new CanvasError(), previewNode); | ||||
|     } | ||||
|   }, [ | ||||
|     appState, | ||||
|     files, | ||||
|     exportedElements, | ||||
|     exportBackground, | ||||
|     exportPadding, | ||||
|     viewBackgroundColor, | ||||
|     scale, | ||||
|     shouldAddWatermark, | ||||
|   ]); | ||||
|  | ||||
|   return ( | ||||
| @@ -167,8 +181,34 @@ const ImageExportModal = ({ | ||||
|         </div> | ||||
|       </div> | ||||
|       <div style={{ display: "flex", alignItems: "center", marginTop: ".6em" }}> | ||||
|         <Stack.Row gap={2}> | ||||
|           {actionManager.renderAction("changeExportScale")} | ||||
|         <Stack.Row gap={2} justifyContent={"center"}> | ||||
|           {scales.map((_scale) => { | ||||
|             const [width, height] = getExportSize( | ||||
|               exportedElements, | ||||
|               exportPadding, | ||||
|               shouldAddWatermark, | ||||
|               _scale, | ||||
|             ); | ||||
|  | ||||
|             const scaleButtonTitle = `${t( | ||||
|               "buttons.scale", | ||||
|             )} ${_scale}x (${width}x${height})`; | ||||
|  | ||||
|             return ( | ||||
|               <ToolButton | ||||
|                 key={_scale} | ||||
|                 size="s" | ||||
|                 type="radio" | ||||
|                 icon={`${_scale}x`} | ||||
|                 name="export-canvas-scale" | ||||
|                 title={scaleButtonTitle} | ||||
|                 aria-label={scaleButtonTitle} | ||||
|                 id="export-canvas-scale" | ||||
|                 checked={_scale === scale} | ||||
|                 onChange={() => setScale(_scale)} | ||||
|               /> | ||||
|             ); | ||||
|           })} | ||||
|         </Stack.Row> | ||||
|         <p style={{ marginLeft: "1em", userSelect: "none" }}>Scale</p> | ||||
|       </div> | ||||
| @@ -180,15 +220,14 @@ const ImageExportModal = ({ | ||||
|           margin: ".6em 0", | ||||
|         }} | ||||
|       > | ||||
|         {!nativeFileSystemSupported && | ||||
|           actionManager.renderAction("changeProjectName")} | ||||
|         {!fsSupported && actionManager.renderAction("changeProjectName")} | ||||
|       </div> | ||||
|       <Stack.Row gap={2} justifyContent="center" style={{ margin: "2em 0" }}> | ||||
|         <ExportButton | ||||
|           color="indigo" | ||||
|           title={t("buttons.exportToPng")} | ||||
|           aria-label={t("buttons.exportToPng")} | ||||
|           onClick={() => onExportToPng(exportedElements)} | ||||
|           onClick={() => onExportToPng(exportedElements, scale)} | ||||
|         > | ||||
|           PNG | ||||
|         </ExportButton> | ||||
| @@ -196,14 +235,14 @@ const ImageExportModal = ({ | ||||
|           color="red" | ||||
|           title={t("buttons.exportToSvg")} | ||||
|           aria-label={t("buttons.exportToSvg")} | ||||
|           onClick={() => onExportToSvg(exportedElements)} | ||||
|           onClick={() => onExportToSvg(exportedElements, scale)} | ||||
|         > | ||||
|           SVG | ||||
|         </ExportButton> | ||||
|         {probablySupportsClipboardBlob && ( | ||||
|           <ExportButton | ||||
|             title={t("buttons.copyPngToClipboard")} | ||||
|             onClick={() => onExportToClipboard(exportedElements)} | ||||
|             onClick={() => onExportToClipboard(exportedElements, scale)} | ||||
|             color="gray" | ||||
|             shade={7} | ||||
|           > | ||||
| @@ -218,8 +257,7 @@ const ImageExportModal = ({ | ||||
| export const ImageExportDialog = ({ | ||||
|   elements, | ||||
|   appState, | ||||
|   files, | ||||
|   exportPadding = DEFAULT_EXPORT_PADDING, | ||||
|   exportPadding = 10, | ||||
|   actionManager, | ||||
|   onExportToPng, | ||||
|   onExportToSvg, | ||||
| @@ -227,9 +265,8 @@ export const ImageExportDialog = ({ | ||||
| }: { | ||||
|   appState: AppState; | ||||
|   elements: readonly NonDeletedExcalidrawElement[]; | ||||
|   files: BinaryFiles; | ||||
|   exportPadding?: number; | ||||
|   actionManager: ActionManager; | ||||
|   actionManager: ActionsManagerInterface; | ||||
|   onExportToPng: ExportCB; | ||||
|   onExportToSvg: ExportCB; | ||||
|   onExportToClipboard: ExportCB; | ||||
| @@ -250,7 +287,7 @@ export const ImageExportDialog = ({ | ||||
|         icon={exportImage} | ||||
|         type="button" | ||||
|         aria-label={t("buttons.exportImage")} | ||||
|         showAriaLabel={useDeviceType().isMobile} | ||||
|         showAriaLabel={useIsMobile()} | ||||
|         title={t("buttons.exportImage")} | ||||
|       /> | ||||
|       {modalIsShown && ( | ||||
| @@ -258,7 +295,6 @@ export const ImageExportDialog = ({ | ||||
|           <ImageExportModal | ||||
|             elements={elements} | ||||
|             appState={appState} | ||||
|             files={files} | ||||
|             exportPadding={exportPadding} | ||||
|             actionManager={actionManager} | ||||
|             onExportToPng={onExportToPng} | ||||
|   | ||||
| @@ -1,25 +1,30 @@ | ||||
| import React, { useEffect, useState } from "react"; | ||||
| import React from "react"; | ||||
|  | ||||
| import { LoadingMessage } from "./LoadingMessage"; | ||||
| import { defaultLang, Language, languages, setLanguage } from "../i18n"; | ||||
|  | ||||
| interface Props { | ||||
|   langCode: Language["code"]; | ||||
|   children: React.ReactElement; | ||||
| } | ||||
| interface State { | ||||
|   isLoading: boolean; | ||||
| } | ||||
| export class InitializeApp extends React.Component<Props, State> { | ||||
|   public state: { isLoading: boolean } = { | ||||
|     isLoading: true, | ||||
|   }; | ||||
|  | ||||
| export const InitializeApp = (props: Props) => { | ||||
|   const [loading, setLoading] = useState(true); | ||||
|  | ||||
|   useEffect(() => { | ||||
|     const updateLang = async () => { | ||||
|       await setLanguage(currentLang); | ||||
|     }; | ||||
|   async componentDidMount() { | ||||
|     const currentLang = | ||||
|       languages.find((lang) => lang.code === props.langCode) || defaultLang; | ||||
|     updateLang(); | ||||
|     setLoading(false); | ||||
|   }, [props.langCode]); | ||||
|       languages.find((lang) => lang.code === this.props.langCode) || | ||||
|       defaultLang; | ||||
|     await setLanguage(currentLang); | ||||
|     this.setState({ | ||||
|       isLoading: false, | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   return loading ? <LoadingMessage /> : props.children; | ||||
| }; | ||||
|   public render() { | ||||
|     return this.state.isLoading ? <LoadingMessage /> : this.props.children; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -3,7 +3,7 @@ | ||||
|     --padding: 0; | ||||
|     background-color: var(--island-bg-color); | ||||
|     box-shadow: var(--shadow-island); | ||||
|     border-radius: var(--border-radius-lg); | ||||
|     border-radius: 4px; | ||||
|     padding: calc(var(--padding) * var(--space-factor)); | ||||
|     position: relative; | ||||
|     transition: box-shadow 0.5s ease-in-out; | ||||
|   | ||||
| @@ -1,19 +1,17 @@ | ||||
| import React, { useState } from "react"; | ||||
| import { ActionsManagerInterface } from "../actions/types"; | ||||
| import { NonDeletedExcalidrawElement } from "../element/types"; | ||||
| import { t } from "../i18n"; | ||||
| import { useDeviceType } from "./App"; | ||||
| import { AppState, ExportOpts, BinaryFiles } from "../types"; | ||||
| import { useIsMobile } from "./App"; | ||||
| import { AppState } from "../types"; | ||||
| import { Dialog } from "./Dialog"; | ||||
| import { exportFile, exportToFileIcon, link } from "./icons"; | ||||
| import { ToolButton } from "./ToolButton"; | ||||
| import { actionSaveFileToDisk } from "../actions/actionExport"; | ||||
| import { actionSaveAsScene } from "../actions/actionExport"; | ||||
| import { Card } from "./Card"; | ||||
|  | ||||
| import "./ExportDialog.scss"; | ||||
| import { nativeFileSystemSupported } from "../data/filesystem"; | ||||
| import { trackEvent } from "../analytics"; | ||||
| import { ActionManager } from "../actions/manager"; | ||||
| import { getFrame } from "../utils"; | ||||
| import { supported as fsSupported } from "browser-fs-access"; | ||||
|  | ||||
| export type ExportCB = ( | ||||
|   elements: readonly NonDeletedExcalidrawElement[], | ||||
| @@ -23,44 +21,36 @@ export type ExportCB = ( | ||||
| const JSONExportModal = ({ | ||||
|   elements, | ||||
|   appState, | ||||
|   files, | ||||
|   actionManager, | ||||
|   exportOpts, | ||||
|   canvas, | ||||
|   onExportToBackend, | ||||
| }: { | ||||
|   appState: AppState; | ||||
|   files: BinaryFiles; | ||||
|   elements: readonly NonDeletedExcalidrawElement[]; | ||||
|   actionManager: ActionManager; | ||||
|   actionManager: ActionsManagerInterface; | ||||
|   onExportToBackend?: ExportCB; | ||||
|   onCloseRequest: () => void; | ||||
|   exportOpts: ExportOpts; | ||||
|   canvas: HTMLCanvasElement | null; | ||||
| }) => { | ||||
|   const { onExportToBackend } = exportOpts; | ||||
|   return ( | ||||
|     <div className="ExportDialog ExportDialog--json"> | ||||
|       <div className="ExportDialog-cards"> | ||||
|         {exportOpts.saveFileToDisk && ( | ||||
|           <Card color="lime"> | ||||
|             <div className="Card-icon">{exportToFileIcon}</div> | ||||
|             <h2>{t("exportDialog.disk_title")}</h2> | ||||
|             <div className="Card-details"> | ||||
|               {t("exportDialog.disk_details")} | ||||
|               {!nativeFileSystemSupported && | ||||
|                 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(actionSaveFileToDisk, "ui"); | ||||
|               }} | ||||
|             /> | ||||
|           </Card> | ||||
|         )} | ||||
|         <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> | ||||
| @@ -72,15 +62,10 @@ const JSONExportModal = ({ | ||||
|               title={t("exportDialog.link_button")} | ||||
|               aria-label={t("exportDialog.link_button")} | ||||
|               showAriaLabel={true} | ||||
|               onClick={() => { | ||||
|                 onExportToBackend(elements, appState, files, canvas); | ||||
|                 trackEvent("export", "link", `ui (${getFrame()})`); | ||||
|               }} | ||||
|               onClick={() => onExportToBackend(elements)} | ||||
|             /> | ||||
|           </Card> | ||||
|         )} | ||||
|         {exportOpts.renderCustomUI && | ||||
|           exportOpts.renderCustomUI(elements, appState, files, canvas)} | ||||
|       </div> | ||||
|     </div> | ||||
|   ); | ||||
| @@ -89,17 +74,13 @@ const JSONExportModal = ({ | ||||
| export const JSONExportDialog = ({ | ||||
|   elements, | ||||
|   appState, | ||||
|   files, | ||||
|   actionManager, | ||||
|   exportOpts, | ||||
|   canvas, | ||||
|   onExportToBackend, | ||||
| }: { | ||||
|   elements: readonly NonDeletedExcalidrawElement[]; | ||||
|   appState: AppState; | ||||
|   files: BinaryFiles; | ||||
|   actionManager: ActionManager; | ||||
|   exportOpts: ExportOpts; | ||||
|   canvas: HTMLCanvasElement | null; | ||||
|   elements: readonly NonDeletedExcalidrawElement[]; | ||||
|   actionManager: ActionsManagerInterface; | ||||
|   onExportToBackend?: ExportCB; | ||||
| }) => { | ||||
|   const [modalIsShown, setModalIsShown] = useState(false); | ||||
|  | ||||
| @@ -117,7 +98,7 @@ export const JSONExportDialog = ({ | ||||
|         icon={exportFile} | ||||
|         type="button" | ||||
|         aria-label={t("buttons.export")} | ||||
|         showAriaLabel={useDeviceType().isMobile} | ||||
|         showAriaLabel={useIsMobile()} | ||||
|         title={t("buttons.export")} | ||||
|       /> | ||||
|       {modalIsShown && ( | ||||
| @@ -125,11 +106,9 @@ export const JSONExportDialog = ({ | ||||
|           <JSONExportModal | ||||
|             elements={elements} | ||||
|             appState={appState} | ||||
|             files={files} | ||||
|             actionManager={actionManager} | ||||
|             onExportToBackend={onExportToBackend} | ||||
|             onCloseRequest={handleClose} | ||||
|             exportOpts={exportOpts} | ||||
|             canvas={canvas} | ||||
|           /> | ||||
|         </Dialog> | ||||
|       )} | ||||
|   | ||||
| @@ -1,6 +1,42 @@ | ||||
| @import "open-color/open-color"; | ||||
|  | ||||
| .excalidraw { | ||||
|   .layer-ui__library { | ||||
|     margin: auto; | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|     justify-content: center; | ||||
|  | ||||
|     .layer-ui__library-header { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       width: 100%; | ||||
|       margin: 2px 0; | ||||
|  | ||||
|       button { | ||||
|         // 2px from the left to account for focus border of left-most button | ||||
|         margin: 0 2px; | ||||
|       } | ||||
|  | ||||
|       a { | ||||
|         margin-inline-start: auto; | ||||
|         // 17px for scrollbar (needed for overlay scrollbars on Big Sur?) + 1px extra | ||||
|         padding-inline-end: 18px; | ||||
|         white-space: nowrap; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .layer-ui__library-message { | ||||
|     padding: 10px 20px; | ||||
|     max-width: 200px; | ||||
|   } | ||||
|  | ||||
|   .layer-ui__library-items { | ||||
|     max-height: 50vh; | ||||
|     overflow: auto; | ||||
|   } | ||||
|  | ||||
|   .layer-ui__wrapper { | ||||
|     z-index: var(--zIndex-layerUI); | ||||
|  | ||||
| @@ -37,10 +73,10 @@ | ||||
|       } | ||||
|  | ||||
|       :root[dir="ltr"] &.layer-ui__wrapper__footer-left--transition-left { | ||||
|         transform: translate(-76px, 0); | ||||
|         transform: translate(-92px, 0); | ||||
|       } | ||||
|       :root[dir="rtl"] &.layer-ui__wrapper__footer-left--transition-left { | ||||
|         transform: translate(76px, 0); | ||||
|         transform: translate(92px, 0); | ||||
|       } | ||||
|  | ||||
|       &.layer-ui__wrapper__footer-left--transition-bottom { | ||||
| @@ -80,19 +116,8 @@ | ||||
|       } | ||||
|     } | ||||
|     .layer-ui__wrapper__footer-left, | ||||
|     .layer-ui__wrapper__footer-right, | ||||
|     .disable-zen-mode--visible { | ||||
|       pointer-events: all; | ||||
|     } | ||||
|  | ||||
|     .layer-ui__wrapper__footer-left { | ||||
|       margin-bottom: 0.2em; | ||||
|     } | ||||
|  | ||||
|     .layer-ui__wrapper__footer-right { | ||||
|       margin-top: auto; | ||||
|       margin-bottom: auto; | ||||
|       margin-inline-end: 1em; | ||||
|       pointer-events: all; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,28 @@ | ||||
| import clsx from "clsx"; | ||||
| import React, { useCallback } from "react"; | ||||
| import React, { | ||||
|   RefObject, | ||||
|   useCallback, | ||||
|   useEffect, | ||||
|   useRef, | ||||
|   useState, | ||||
| } from "react"; | ||||
| import { ActionManager } from "../actions/manager"; | ||||
| import { CLASSES } from "../constants"; | ||||
| import { exportCanvas } from "../data"; | ||||
| import { importLibraryFromJSON, saveLibraryAsJSON } from "../data/json"; | ||||
| import { isTextElement, showSelectedShapeActions } from "../element"; | ||||
| import { NonDeletedExcalidrawElement } from "../element/types"; | ||||
| import { Language, t } from "../i18n"; | ||||
| import { useIsMobile } from "../components/App"; | ||||
| import { calculateScrollCenter, getSelectedElements } from "../scene"; | ||||
| import { ExportType } from "../scene/types"; | ||||
| import { AppProps, AppState, ExcalidrawProps, BinaryFiles } from "../types"; | ||||
| import { | ||||
|   AppProps, | ||||
|   AppState, | ||||
|   ExcalidrawProps, | ||||
|   LibraryItem, | ||||
|   LibraryItems, | ||||
| } from "../types"; | ||||
| import { muteFSAbortError } from "../utils"; | ||||
| import { SelectedShapeActions, ShapesSwitcher, ZoomActions } from "./Actions"; | ||||
| import { BackgroundPickerAndDarkModeToggle } from "./BackgroundPickerAndDarkModeToggle"; | ||||
| @@ -17,38 +31,31 @@ import { ErrorDialog } from "./ErrorDialog"; | ||||
| import { ExportCB, ImageExportDialog } from "./ImageExportDialog"; | ||||
| import { FixedSideContainer } from "./FixedSideContainer"; | ||||
| import { HintViewer } from "./HintViewer"; | ||||
| import { exportFile, load, trash } from "./icons"; | ||||
| import { Island } from "./Island"; | ||||
| import "./LayerUI.scss"; | ||||
| import { LibraryUnit } from "./LibraryUnit"; | ||||
| import { LoadingMessage } from "./LoadingMessage"; | ||||
| import { LockButton } from "./LockButton"; | ||||
| import { LockIcon } from "./LockIcon"; | ||||
| import { MobileMenu } from "./MobileMenu"; | ||||
| import { PasteChartDialog } from "./PasteChartDialog"; | ||||
| import { Section } from "./Section"; | ||||
| import { HelpDialog } from "./HelpDialog"; | ||||
| import Stack from "./Stack"; | ||||
| import { ToolButton } from "./ToolButton"; | ||||
| import { Tooltip } from "./Tooltip"; | ||||
| import { UserList } from "./UserList"; | ||||
| import Library from "../data/library"; | ||||
| import { JSONExportDialog } from "./JSONExportDialog"; | ||||
| import { LibraryButton } from "./LibraryButton"; | ||||
| import { isImageFileHandle } from "../data/blob"; | ||||
| import { LibraryMenu } from "./LibraryMenu"; | ||||
|  | ||||
| import "./LayerUI.scss"; | ||||
| import "./Toolbar.scss"; | ||||
| import { PenModeButton } from "./PenModeButton"; | ||||
| import { trackEvent } from "../analytics"; | ||||
| import { useDeviceType } from "../components/App"; | ||||
|  | ||||
| interface LayerUIProps { | ||||
|   actionManager: ActionManager; | ||||
|   appState: AppState; | ||||
|   files: BinaryFiles; | ||||
|   canvas: HTMLCanvasElement | null; | ||||
|   setAppState: React.Component<any, AppState>["setState"]; | ||||
|   elements: readonly NonDeletedExcalidrawElement[]; | ||||
|   onCollabButtonClick?: () => void; | ||||
|   onLockToggle: () => void; | ||||
|   onPenModeToggle: () => void; | ||||
|   onInsertElements: (elements: readonly NonDeletedExcalidrawElement[]) => void; | ||||
|   zenModeEnabled: boolean; | ||||
|   showExitZenModeBtn: boolean; | ||||
| @@ -56,10 +63,12 @@ interface LayerUIProps { | ||||
|   toggleZenMode: () => void; | ||||
|   langCode: Language["code"]; | ||||
|   isCollaborating: boolean; | ||||
|   renderTopRightUI?: ( | ||||
|     isMobile: boolean, | ||||
|   onExportToBackend?: ( | ||||
|     exportedElements: readonly NonDeletedExcalidrawElement[], | ||||
|     appState: AppState, | ||||
|   ) => JSX.Element | null; | ||||
|     canvas: HTMLCanvasElement | null, | ||||
|   ) => void; | ||||
|   renderTopRightUI?: (isMobile: boolean, appState: AppState) => JSX.Element; | ||||
|   renderCustomFooter?: (isMobile: boolean, appState: AppState) => JSX.Element; | ||||
|   viewModeEnabled: boolean; | ||||
|   libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"]; | ||||
| @@ -67,25 +76,302 @@ interface LayerUIProps { | ||||
|   focusContainer: () => void; | ||||
|   library: Library; | ||||
|   id: string; | ||||
|   onImageAction: (data: { insertOnCanvasDirectly: boolean }) => void; | ||||
| } | ||||
|  | ||||
| const useOnClickOutside = ( | ||||
|   ref: RefObject<HTMLElement>, | ||||
|   cb: (event: MouseEvent) => void, | ||||
| ) => { | ||||
|   useEffect(() => { | ||||
|     const listener = (event: MouseEvent) => { | ||||
|       if (!ref.current) { | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       if ( | ||||
|         event.target instanceof Element && | ||||
|         (ref.current.contains(event.target) || | ||||
|           !document.body.contains(event.target)) | ||||
|       ) { | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       cb(event); | ||||
|     }; | ||||
|     document.addEventListener("pointerdown", listener, false); | ||||
|  | ||||
|     return () => { | ||||
|       document.removeEventListener("pointerdown", listener); | ||||
|     }; | ||||
|   }, [ref, cb]); | ||||
| }; | ||||
|  | ||||
| const LibraryMenuItems = ({ | ||||
|   libraryItems, | ||||
|   onRemoveFromLibrary, | ||||
|   onAddToLibrary, | ||||
|   onInsertShape, | ||||
|   pendingElements, | ||||
|   setAppState, | ||||
|   setLibraryItems, | ||||
|   libraryReturnUrl, | ||||
|   focusContainer, | ||||
|   library, | ||||
|   id, | ||||
| }: { | ||||
|   libraryItems: LibraryItems; | ||||
|   pendingElements: LibraryItem; | ||||
|   onRemoveFromLibrary: (index: number) => void; | ||||
|   onInsertShape: (elements: LibraryItem) => void; | ||||
|   onAddToLibrary: (elements: LibraryItem) => void; | ||||
|   setAppState: React.Component<any, AppState>["setState"]; | ||||
|   setLibraryItems: (library: LibraryItems) => void; | ||||
|   libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"]; | ||||
|   focusContainer: () => void; | ||||
|   library: Library; | ||||
|   id: string; | ||||
| }) => { | ||||
|   const isMobile = useIsMobile(); | ||||
|   const numCells = libraryItems.length + (pendingElements.length > 0 ? 1 : 0); | ||||
|   const CELLS_PER_ROW = isMobile ? 4 : 6; | ||||
|   const numRows = Math.max(1, Math.ceil(numCells / CELLS_PER_ROW)); | ||||
|   const rows = []; | ||||
|   let addedPendingElements = false; | ||||
|  | ||||
|   const referrer = | ||||
|     libraryReturnUrl || window.location.origin + window.location.pathname; | ||||
|  | ||||
|   rows.push( | ||||
|     <div className="layer-ui__library-header" key="library-header"> | ||||
|       <ToolButton | ||||
|         key="import" | ||||
|         type="button" | ||||
|         title={t("buttons.load")} | ||||
|         aria-label={t("buttons.load")} | ||||
|         icon={load} | ||||
|         onClick={() => { | ||||
|           importLibraryFromJSON(library) | ||||
|             .then(() => { | ||||
|               // Close and then open to get the libraries updated | ||||
|               setAppState({ isLibraryOpen: false }); | ||||
|               setAppState({ isLibraryOpen: true }); | ||||
|             }) | ||||
|             .catch(muteFSAbortError) | ||||
|             .catch((error) => { | ||||
|               setAppState({ errorMessage: error.message }); | ||||
|             }); | ||||
|         }} | ||||
|       /> | ||||
|       {!!libraryItems.length && ( | ||||
|         <> | ||||
|           <ToolButton | ||||
|             key="export" | ||||
|             type="button" | ||||
|             title={t("buttons.export")} | ||||
|             aria-label={t("buttons.export")} | ||||
|             icon={exportFile} | ||||
|             onClick={() => { | ||||
|               saveLibraryAsJSON(library) | ||||
|                 .catch(muteFSAbortError) | ||||
|                 .catch((error) => { | ||||
|                   setAppState({ errorMessage: error.message }); | ||||
|                 }); | ||||
|             }} | ||||
|           /> | ||||
|           <ToolButton | ||||
|             key="reset" | ||||
|             type="button" | ||||
|             title={t("buttons.resetLibrary")} | ||||
|             aria-label={t("buttons.resetLibrary")} | ||||
|             icon={trash} | ||||
|             onClick={() => { | ||||
|               if (window.confirm(t("alerts.resetLibrary"))) { | ||||
|                 library.resetLibrary(); | ||||
|                 setLibraryItems([]); | ||||
|                 focusContainer(); | ||||
|               } | ||||
|             }} | ||||
|           /> | ||||
|         </> | ||||
|       )} | ||||
|       <a | ||||
|         href={`https://libraries.excalidraw.com?target=${ | ||||
|           window.name || "_blank" | ||||
|         }&referrer=${referrer}&useHash=true&token=${id}`} | ||||
|         target="_excalidraw_libraries" | ||||
|       > | ||||
|         {t("labels.libraries")} | ||||
|       </a> | ||||
|     </div>, | ||||
|   ); | ||||
|  | ||||
|   for (let row = 0; row < numRows; row++) { | ||||
|     const y = CELLS_PER_ROW * row; | ||||
|     const children = []; | ||||
|     for (let x = 0; x < CELLS_PER_ROW; x++) { | ||||
|       const shouldAddPendingElements: boolean = | ||||
|         pendingElements.length > 0 && | ||||
|         !addedPendingElements && | ||||
|         y + x >= libraryItems.length; | ||||
|       addedPendingElements = addedPendingElements || shouldAddPendingElements; | ||||
|  | ||||
|       children.push( | ||||
|         <Stack.Col key={x}> | ||||
|           <LibraryUnit | ||||
|             elements={libraryItems[y + x]} | ||||
|             pendingElements={ | ||||
|               shouldAddPendingElements ? pendingElements : undefined | ||||
|             } | ||||
|             onRemoveFromLibrary={onRemoveFromLibrary.bind(null, y + x)} | ||||
|             onClick={ | ||||
|               shouldAddPendingElements | ||||
|                 ? onAddToLibrary.bind(null, pendingElements) | ||||
|                 : onInsertShape.bind(null, libraryItems[y + x]) | ||||
|             } | ||||
|           /> | ||||
|         </Stack.Col>, | ||||
|       ); | ||||
|     } | ||||
|     rows.push( | ||||
|       <Stack.Row align="center" gap={1} key={row}> | ||||
|         {children} | ||||
|       </Stack.Row>, | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     <Stack.Col align="start" gap={1} className="layer-ui__library-items"> | ||||
|       {rows} | ||||
|     </Stack.Col> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| const LibraryMenu = ({ | ||||
|   onClickOutside, | ||||
|   onInsertShape, | ||||
|   pendingElements, | ||||
|   onAddToLibrary, | ||||
|   setAppState, | ||||
|   libraryReturnUrl, | ||||
|   focusContainer, | ||||
|   library, | ||||
|   id, | ||||
| }: { | ||||
|   pendingElements: LibraryItem; | ||||
|   onClickOutside: (event: MouseEvent) => void; | ||||
|   onInsertShape: (elements: LibraryItem) => void; | ||||
|   onAddToLibrary: () => void; | ||||
|   setAppState: React.Component<any, AppState>["setState"]; | ||||
|   libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"]; | ||||
|   focusContainer: () => void; | ||||
|   library: Library; | ||||
|   id: string; | ||||
| }) => { | ||||
|   const ref = useRef<HTMLDivElement | null>(null); | ||||
|   useOnClickOutside(ref, (event) => { | ||||
|     // If click on the library icon, do nothing. | ||||
|     if ((event.target as Element).closest(".ToolIcon_type_button__library")) { | ||||
|       return; | ||||
|     } | ||||
|     onClickOutside(event); | ||||
|   }); | ||||
|  | ||||
|   const [libraryItems, setLibraryItems] = useState<LibraryItems>([]); | ||||
|  | ||||
|   const [loadingState, setIsLoading] = useState< | ||||
|     "preloading" | "loading" | "ready" | ||||
|   >("preloading"); | ||||
|  | ||||
|   const loadingTimerRef = useRef<NodeJS.Timeout | null>(null); | ||||
|  | ||||
|   useEffect(() => { | ||||
|     Promise.race([ | ||||
|       new Promise((resolve) => { | ||||
|         loadingTimerRef.current = setTimeout(() => { | ||||
|           resolve("loading"); | ||||
|         }, 100); | ||||
|       }), | ||||
|       library.loadLibrary().then((items) => { | ||||
|         setLibraryItems(items); | ||||
|         setIsLoading("ready"); | ||||
|       }), | ||||
|     ]).then((data) => { | ||||
|       if (data === "loading") { | ||||
|         setIsLoading("loading"); | ||||
|       } | ||||
|     }); | ||||
|     return () => { | ||||
|       clearTimeout(loadingTimerRef.current!); | ||||
|     }; | ||||
|   }, [library]); | ||||
|  | ||||
|   const removeFromLibrary = useCallback( | ||||
|     async (indexToRemove) => { | ||||
|       const items = await library.loadLibrary(); | ||||
|       const nextItems = items.filter((_, index) => index !== indexToRemove); | ||||
|       library.saveLibrary(nextItems).catch((error) => { | ||||
|         setLibraryItems(items); | ||||
|         setAppState({ errorMessage: t("alerts.errorRemovingFromLibrary") }); | ||||
|       }); | ||||
|       setLibraryItems(nextItems); | ||||
|     }, | ||||
|     [library, setAppState], | ||||
|   ); | ||||
|  | ||||
|   const addToLibrary = useCallback( | ||||
|     async (elements: LibraryItem) => { | ||||
|       const items = await library.loadLibrary(); | ||||
|       const nextItems = [...items, elements]; | ||||
|       onAddToLibrary(); | ||||
|       library.saveLibrary(nextItems).catch((error) => { | ||||
|         setLibraryItems(items); | ||||
|         setAppState({ errorMessage: t("alerts.errorAddingToLibrary") }); | ||||
|       }); | ||||
|       setLibraryItems(nextItems); | ||||
|     }, | ||||
|     [onAddToLibrary, library, setAppState], | ||||
|   ); | ||||
|  | ||||
|   return loadingState === "preloading" ? null : ( | ||||
|     <Island padding={1} ref={ref} className="layer-ui__library"> | ||||
|       {loadingState === "loading" ? ( | ||||
|         <div className="layer-ui__library-message"> | ||||
|           {t("labels.libraryLoadingMessage")} | ||||
|         </div> | ||||
|       ) : ( | ||||
|         <LibraryMenuItems | ||||
|           libraryItems={libraryItems} | ||||
|           onRemoveFromLibrary={removeFromLibrary} | ||||
|           onAddToLibrary={addToLibrary} | ||||
|           onInsertShape={onInsertShape} | ||||
|           pendingElements={pendingElements} | ||||
|           setAppState={setAppState} | ||||
|           setLibraryItems={setLibraryItems} | ||||
|           libraryReturnUrl={libraryReturnUrl} | ||||
|           focusContainer={focusContainer} | ||||
|           library={library} | ||||
|           id={id} | ||||
|         /> | ||||
|       )} | ||||
|     </Island> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| const LayerUI = ({ | ||||
|   actionManager, | ||||
|   appState, | ||||
|   files, | ||||
|   setAppState, | ||||
|   canvas, | ||||
|   elements, | ||||
|   onCollabButtonClick, | ||||
|   onLockToggle, | ||||
|   onPenModeToggle, | ||||
|   onInsertElements, | ||||
|   zenModeEnabled, | ||||
|   showExitZenModeBtn, | ||||
|   showThemeBtn, | ||||
|   toggleZenMode, | ||||
|   isCollaborating, | ||||
|   onExportToBackend, | ||||
|   renderTopRightUI, | ||||
|   renderCustomFooter, | ||||
|   viewModeEnabled, | ||||
| @@ -94,9 +380,8 @@ const LayerUI = ({ | ||||
|   focusContainer, | ||||
|   library, | ||||
|   id, | ||||
|   onImageAction, | ||||
| }: LayerUIProps) => { | ||||
|   const deviceType = useDeviceType(); | ||||
|   const isMobile = useIsMobile(); | ||||
|  | ||||
|   const renderJSONExportDialog = () => { | ||||
|     if (!UIOptions.canvasActions.export) { | ||||
| @@ -107,54 +392,46 @@ const LayerUI = ({ | ||||
|       <JSONExportDialog | ||||
|         elements={elements} | ||||
|         appState={appState} | ||||
|         files={files} | ||||
|         actionManager={actionManager} | ||||
|         exportOpts={UIOptions.canvasActions.export} | ||||
|         canvas={canvas} | ||||
|         onExportToBackend={ | ||||
|           onExportToBackend | ||||
|             ? (elements) => { | ||||
|                 onExportToBackend && | ||||
|                   onExportToBackend(elements, appState, canvas); | ||||
|               } | ||||
|             : undefined | ||||
|         } | ||||
|       /> | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   const renderImageExportDialog = () => { | ||||
|     if (!UIOptions.canvasActions.saveAsImage) { | ||||
|     if (!UIOptions.canvasActions.export) { | ||||
|       return null; | ||||
|     } | ||||
|  | ||||
|     const createExporter = | ||||
|       (type: ExportType): ExportCB => | ||||
|       async (exportedElements) => { | ||||
|         trackEvent("export", type, "ui"); | ||||
|         const fileHandle = await exportCanvas( | ||||
|           type, | ||||
|           exportedElements, | ||||
|           appState, | ||||
|           files, | ||||
|           { | ||||
|             exportBackground: appState.exportBackground, | ||||
|             name: appState.name, | ||||
|             viewBackgroundColor: appState.viewBackgroundColor, | ||||
|           }, | ||||
|         ) | ||||
|           .catch(muteFSAbortError) | ||||
|           .catch((error) => { | ||||
|             console.error(error); | ||||
|             setAppState({ errorMessage: error.message }); | ||||
|           }); | ||||
|  | ||||
|         if ( | ||||
|           appState.exportEmbedScene && | ||||
|           fileHandle && | ||||
|           isImageFileHandle(fileHandle) | ||||
|         ) { | ||||
|           setAppState({ fileHandle }); | ||||
|         } | ||||
|       }; | ||||
|     const createExporter = (type: ExportType): ExportCB => async ( | ||||
|       exportedElements, | ||||
|       scale, | ||||
|     ) => { | ||||
|       await exportCanvas(type, exportedElements, appState, { | ||||
|         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 ( | ||||
|       <ImageExportDialog | ||||
|         elements={elements} | ||||
|         appState={appState} | ||||
|         files={files} | ||||
|         actionManager={actionManager} | ||||
|         onExportToPng={createExporter("png")} | ||||
|         onExportToSvg={createExporter("svg")} | ||||
| @@ -188,7 +465,6 @@ const LayerUI = ({ | ||||
|       </Section> | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   const renderCanvasActions = () => ( | ||||
|     <Section | ||||
|       heading="canvasActions" | ||||
| @@ -221,9 +497,6 @@ const LayerUI = ({ | ||||
|             setAppState={setAppState} | ||||
|             showThemeBtn={showThemeBtn} | ||||
|           /> | ||||
|           {appState.fileHandle && ( | ||||
|             <>{actionManager.renderAction("saveToActiveFile")}</> | ||||
|           )} | ||||
|         </Stack.Col> | ||||
|       </Island> | ||||
|     </Section> | ||||
| @@ -240,31 +513,27 @@ const LayerUI = ({ | ||||
|         className={CLASSES.SHAPE_ACTIONS_MENU} | ||||
|         padding={2} | ||||
|         style={{ | ||||
|           // we want to make sure this doesn't overflow so subtracting 200 | ||||
|           // we want to make sure this doesn't overflow so substracting 200 | ||||
|           // which is approximately height of zoom footer and top left menu items with some buffer | ||||
|           // if active file name is displayed, subtracting 248 to account for its height | ||||
|           maxHeight: `${appState.height - (appState.fileHandle ? 248 : 200)}px`, | ||||
|           maxHeight: `${appState.height - 200}px`, | ||||
|         }} | ||||
|       > | ||||
|         <SelectedShapeActions | ||||
|           appState={appState} | ||||
|           elements={elements} | ||||
|           renderAction={actionManager.renderAction} | ||||
|           activeTool={appState.activeTool.type} | ||||
|           elementType={appState.elementType} | ||||
|         /> | ||||
|       </Island> | ||||
|     </Section> | ||||
|   ); | ||||
|  | ||||
|   const closeLibrary = useCallback(() => { | ||||
|     const isDialogOpen = !!document.querySelector(".Dialog"); | ||||
|  | ||||
|     // Prevent closing if any dialog is open | ||||
|     if (isDialogOpen) { | ||||
|       return; | ||||
|     } | ||||
|     setAppState({ isLibraryOpen: false }); | ||||
|   }, [setAppState]); | ||||
|   const closeLibrary = useCallback( | ||||
|     (event) => { | ||||
|       setAppState({ isLibraryOpen: false }); | ||||
|     }, | ||||
|     [setAppState], | ||||
|   ); | ||||
|  | ||||
|   const deselectItems = useCallback(() => { | ||||
|     setAppState({ | ||||
| @@ -275,18 +544,15 @@ const LayerUI = ({ | ||||
|  | ||||
|   const libraryMenu = appState.isLibraryOpen ? ( | ||||
|     <LibraryMenu | ||||
|       pendingElements={getSelectedElements(elements, appState, true)} | ||||
|       onClose={closeLibrary} | ||||
|       pendingElements={getSelectedElements(elements, appState)} | ||||
|       onClickOutside={closeLibrary} | ||||
|       onInsertShape={onInsertElements} | ||||
|       onAddToLibrary={deselectItems} | ||||
|       setAppState={setAppState} | ||||
|       libraryReturnUrl={libraryReturnUrl} | ||||
|       focusContainer={focusContainer} | ||||
|       library={library} | ||||
|       theme={appState.theme} | ||||
|       files={files} | ||||
|       id={id} | ||||
|       appState={appState} | ||||
|     /> | ||||
|   ) : null; | ||||
|  | ||||
| @@ -312,54 +578,27 @@ const LayerUI = ({ | ||||
|             <Section heading="shapes"> | ||||
|               {(heading) => ( | ||||
|                 <Stack.Col gap={4} align="start"> | ||||
|                   <Stack.Row | ||||
|                     gap={1} | ||||
|                     className={clsx("App-toolbar-container", { | ||||
|                       "zen-mode": zenModeEnabled, | ||||
|                     })} | ||||
|                   > | ||||
|                     <PenModeButton | ||||
|                       zenModeEnabled={zenModeEnabled} | ||||
|                       checked={appState.penMode} | ||||
|                       onChange={onPenModeToggle} | ||||
|                       title={t("toolBar.penMode")} | ||||
|                       penDetected={appState.penDetected} | ||||
|                     /> | ||||
|                     <LockButton | ||||
|                       zenModeEnabled={zenModeEnabled} | ||||
|                       checked={appState.activeTool.locked} | ||||
|                       onChange={() => onLockToggle()} | ||||
|                       title={t("toolBar.lock")} | ||||
|                     /> | ||||
|                   <Stack.Row gap={1}> | ||||
|                     <Island | ||||
|                       padding={1} | ||||
|                       className={clsx("App-toolbar", { | ||||
|                         "zen-mode": zenModeEnabled, | ||||
|                       })} | ||||
|                       className={clsx({ "zen-mode": zenModeEnabled })} | ||||
|                     > | ||||
|                       <HintViewer | ||||
|                         appState={appState} | ||||
|                         elements={elements} | ||||
|                         isMobile={deviceType.isMobile} | ||||
|                       /> | ||||
|                       <HintViewer appState={appState} elements={elements} /> | ||||
|                       {heading} | ||||
|                       <Stack.Row gap={1}> | ||||
|                         <ShapesSwitcher | ||||
|                           appState={appState} | ||||
|                           canvas={canvas} | ||||
|                           activeTool={appState.activeTool} | ||||
|                           elementType={appState.elementType} | ||||
|                           setAppState={setAppState} | ||||
|                           onImageAction={({ pointerType }) => { | ||||
|                             onImageAction({ | ||||
|                               insertOnCanvasDirectly: pointerType !== "mouse", | ||||
|                             }); | ||||
|                           }} | ||||
|                           isLibraryOpen={appState.isLibraryOpen} | ||||
|                         /> | ||||
|                       </Stack.Row> | ||||
|                     </Island> | ||||
|                     <LibraryButton | ||||
|                       appState={appState} | ||||
|                       setAppState={setAppState} | ||||
|                     <LockIcon | ||||
|                       zenModeEnabled={zenModeEnabled} | ||||
|                       checked={appState.elementLocked} | ||||
|                       onChange={onLockToggle} | ||||
|                       title={t("toolBar.lock")} | ||||
|                     /> | ||||
|                   </Stack.Row> | ||||
|                   {libraryMenu} | ||||
| @@ -385,13 +624,11 @@ const LayerUI = ({ | ||||
|                       label={client.username || "Unknown user"} | ||||
|                       key={clientId} | ||||
|                     > | ||||
|                       {actionManager.renderAction("goToCollaborator", { | ||||
|                         id: clientId, | ||||
|                       })} | ||||
|                       {actionManager.renderAction("goToCollaborator", clientId)} | ||||
|                     </Tooltip> | ||||
|                   ))} | ||||
|             </UserList> | ||||
|             {renderTopRightUI?.(deviceType.isMobile, appState)} | ||||
|             {renderTopRightUI?.(isMobile, appState)} | ||||
|           </div> | ||||
|         </div> | ||||
|       </FixedSideContainer> | ||||
| @@ -420,40 +657,6 @@ const LayerUI = ({ | ||||
|                   zoom={appState.zoom} | ||||
|                 /> | ||||
|               </Island> | ||||
|               {!viewModeEnabled && ( | ||||
|                 <> | ||||
|                   <div | ||||
|                     className={clsx("undo-redo-buttons zen-mode-transition", { | ||||
|                       "layer-ui__wrapper__footer-left--transition-bottom": | ||||
|                         zenModeEnabled, | ||||
|                     })} | ||||
|                   > | ||||
|                     {actionManager.renderAction("undo", { size: "small" })} | ||||
|                     {actionManager.renderAction("redo", { size: "small" })} | ||||
|                   </div> | ||||
|  | ||||
|                   <div | ||||
|                     className={clsx("eraser-buttons zen-mode-transition", { | ||||
|                       "layer-ui__wrapper__footer-left--transition-left": | ||||
|                         zenModeEnabled, | ||||
|                     })} | ||||
|                   > | ||||
|                     {actionManager.renderAction("eraser", { size: "small" })} | ||||
|                   </div> | ||||
|                 </> | ||||
|               )} | ||||
|               {!viewModeEnabled && | ||||
|                 appState.multiElement && | ||||
|                 deviceType.isTouchScreen && ( | ||||
|                   <div | ||||
|                     className={clsx("finalize-button zen-mode-transition", { | ||||
|                       "layer-ui__wrapper__footer-left--transition-left": | ||||
|                         zenModeEnabled, | ||||
|                     })} | ||||
|                   > | ||||
|                     {actionManager.renderAction("finalize", { size: "small" })} | ||||
|                   </div> | ||||
|                 )} | ||||
|             </Section> | ||||
|           </Stack.Col> | ||||
|         </div> | ||||
| @@ -461,8 +664,7 @@ const LayerUI = ({ | ||||
|           className={clsx( | ||||
|             "layer-ui__wrapper__footer-center zen-mode-transition", | ||||
|             { | ||||
|               "layer-ui__wrapper__footer-left--transition-bottom": | ||||
|                 zenModeEnabled, | ||||
|               "layer-ui__wrapper__footer-left--transition-bottom": zenModeEnabled, | ||||
|             }, | ||||
|           )} | ||||
|         > | ||||
| @@ -492,7 +694,7 @@ const LayerUI = ({ | ||||
|  | ||||
|   const dialogs = ( | ||||
|     <> | ||||
|       {appState.isLoading && <LoadingMessage delay={250} />} | ||||
|       {appState.isLoading && <LoadingMessage />} | ||||
|       {appState.errorMessage && ( | ||||
|         <ErrorDialog | ||||
|           message={appState.errorMessage} | ||||
| @@ -521,7 +723,7 @@ const LayerUI = ({ | ||||
|     </> | ||||
|   ); | ||||
|  | ||||
|   return deviceType.isMobile ? ( | ||||
|   return isMobile ? ( | ||||
|     <> | ||||
|       {dialogs} | ||||
|       <MobileMenu | ||||
| @@ -533,15 +735,12 @@ const LayerUI = ({ | ||||
|         renderImageExportDialog={renderImageExportDialog} | ||||
|         setAppState={setAppState} | ||||
|         onCollabButtonClick={onCollabButtonClick} | ||||
|         onLockToggle={() => onLockToggle()} | ||||
|         onPenModeToggle={onPenModeToggle} | ||||
|         onLockToggle={onLockToggle} | ||||
|         canvas={canvas} | ||||
|         isCollaborating={isCollaborating} | ||||
|         renderCustomFooter={renderCustomFooter} | ||||
|         viewModeEnabled={viewModeEnabled} | ||||
|         showThemeBtn={showThemeBtn} | ||||
|         onImageAction={onImageAction} | ||||
|         renderTopRightUI={renderTopRightUI} | ||||
|       /> | ||||
|     </> | ||||
|   ) : ( | ||||
| @@ -589,7 +788,6 @@ const areEqual = (prev: LayerUIProps, next: LayerUIProps) => { | ||||
|     prev.renderCustomFooter === next.renderCustomFooter && | ||||
|     prev.langCode === next.langCode && | ||||
|     prev.elements === next.elements && | ||||
|     prev.files === next.files && | ||||
|     keys.every((key) => prevAppState[key] === nextAppState[key]) | ||||
|   ); | ||||
| }; | ||||
|   | ||||
| @@ -1,46 +0,0 @@ | ||||
| import React from "react"; | ||||
| import clsx from "clsx"; | ||||
| import { t } from "../i18n"; | ||||
| import { AppState } from "../types"; | ||||
| import { capitalizeString } from "../utils"; | ||||
|  | ||||
| const LIBRARY_ICON = ( | ||||
|   <svg viewBox="0 0 576 512"> | ||||
|     <path | ||||
|       fill="currentColor" | ||||
|       d="M542.22 32.05c-54.8 3.11-163.72 14.43-230.96 55.59-4.64 2.84-7.27 7.89-7.27 13.17v363.87c0 11.55 12.63 18.85 23.28 13.49 69.18-34.82 169.23-44.32 218.7-46.92 16.89-.89 30.02-14.43 30.02-30.66V62.75c.01-17.71-15.35-31.74-33.77-30.7zM264.73 87.64C197.5 46.48 88.58 35.17 33.78 32.05 15.36 31.01 0 45.04 0 62.75V400.6c0 16.24 13.13 29.78 30.02 30.66 49.49 2.6 149.59 12.11 218.77 46.95 10.62 5.35 23.21-1.94 23.21-13.46V100.63c0-5.29-2.62-10.14-7.27-12.99z" | ||||
|     ></path> | ||||
|   </svg> | ||||
| ); | ||||
|  | ||||
| export const LibraryButton: React.FC<{ | ||||
|   appState: AppState; | ||||
|   setAppState: React.Component<any, AppState>["setState"]; | ||||
|   isMobile?: boolean; | ||||
| }> = ({ appState, setAppState, isMobile }) => { | ||||
|   return ( | ||||
|     <label | ||||
|       className={clsx( | ||||
|         "ToolIcon ToolIcon_type_floating ToolIcon__library", | ||||
|         `ToolIcon_size_medium`, | ||||
|         { | ||||
|           "is-mobile": isMobile, | ||||
|         }, | ||||
|       )} | ||||
|       title={`${capitalizeString(t("toolBar.library"))} — 0`} | ||||
|     > | ||||
|       <input | ||||
|         className="ToolIcon_type_checkbox" | ||||
|         type="checkbox" | ||||
|         name="editor-library" | ||||
|         onChange={(event) => { | ||||
|           setAppState({ isLibraryOpen: event.target.checked }); | ||||
|         }} | ||||
|         checked={appState.isLibraryOpen} | ||||
|         aria-label={capitalizeString(t("toolBar.library"))} | ||||
|         aria-keyshortcuts="0" | ||||
|       /> | ||||
|       <div className="ToolIcon__icon">{LIBRARY_ICON}</div> | ||||
|     </label> | ||||
|   ); | ||||
| }; | ||||
| @@ -1,64 +0,0 @@ | ||||
| @import "open-color/open-color"; | ||||
|  | ||||
| .excalidraw { | ||||
|   .layer-ui__library { | ||||
|     margin: auto; | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|     justify-content: center; | ||||
|  | ||||
|     .layer-ui__library-header { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       width: 100%; | ||||
|       margin: 2px 0; | ||||
|  | ||||
|       button { | ||||
|         // 2px from the left to account for focus border of left-most button | ||||
|         margin: 0 2px; | ||||
|       } | ||||
|  | ||||
|       a { | ||||
|         margin-inline-start: auto; | ||||
|         // 17px for scrollbar (needed for overlay scrollbars on Big Sur?) + 1px extra | ||||
|         padding-inline-end: 18px; | ||||
|         white-space: nowrap; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .layer-ui__library-message { | ||||
|     padding: 2em 4em; | ||||
|     min-width: 200px; | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     align-items: center; | ||||
|     .Spinner { | ||||
|       margin-bottom: 1em; | ||||
|     } | ||||
|     span { | ||||
|       font-size: 0.8em; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .publish-library-success { | ||||
|     .Dialog__content { | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|     } | ||||
|  | ||||
|     &-close.ToolIcon_type_button { | ||||
|       background-color: $oc-blue-6; | ||||
|       align-self: flex-end; | ||||
|       &:hover { | ||||
|         background-color: $oc-blue-8; | ||||
|       } | ||||
|       .ToolIcon__icon { | ||||
|         width: auto; | ||||
|         font-size: 1rem; | ||||
|         color: $oc-white; | ||||
|         padding: 0 0.5rem; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,329 +0,0 @@ | ||||
| import { | ||||
|   useRef, | ||||
|   useState, | ||||
|   useEffect, | ||||
|   useCallback, | ||||
|   RefObject, | ||||
|   forwardRef, | ||||
| } from "react"; | ||||
| import Library, { libraryItemsAtom } from "../data/library"; | ||||
| import { t } from "../i18n"; | ||||
| import { randomId } from "../random"; | ||||
| import { | ||||
|   LibraryItems, | ||||
|   LibraryItem, | ||||
|   AppState, | ||||
|   BinaryFiles, | ||||
|   ExcalidrawProps, | ||||
| } from "../types"; | ||||
| import { Dialog } from "./Dialog"; | ||||
| import { Island } from "./Island"; | ||||
| import PublishLibrary from "./PublishLibrary"; | ||||
| import { ToolButton } from "./ToolButton"; | ||||
|  | ||||
| import "./LibraryMenu.scss"; | ||||
| import LibraryMenuItems from "./LibraryMenuItems"; | ||||
| import { EVENT } from "../constants"; | ||||
| import { KEYS } from "../keys"; | ||||
| import { arrayToMap } from "../utils"; | ||||
| import { trackEvent } from "../analytics"; | ||||
| import { useAtom } from "jotai"; | ||||
| import { jotaiScope } from "../jotai"; | ||||
| import Spinner from "./Spinner"; | ||||
|  | ||||
| const useOnClickOutside = ( | ||||
|   ref: RefObject<HTMLElement>, | ||||
|   cb: (event: MouseEvent) => void, | ||||
| ) => { | ||||
|   useEffect(() => { | ||||
|     const listener = (event: MouseEvent) => { | ||||
|       if (!ref.current) { | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       if ( | ||||
|         event.target instanceof Element && | ||||
|         (ref.current.contains(event.target) || | ||||
|           !document.body.contains(event.target)) | ||||
|       ) { | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       cb(event); | ||||
|     }; | ||||
|     document.addEventListener("pointerdown", listener, false); | ||||
|  | ||||
|     return () => { | ||||
|       document.removeEventListener("pointerdown", listener); | ||||
|     }; | ||||
|   }, [ref, cb]); | ||||
| }; | ||||
|  | ||||
| const getSelectedItems = ( | ||||
|   libraryItems: LibraryItems, | ||||
|   selectedItems: LibraryItem["id"][], | ||||
| ) => libraryItems.filter((item) => selectedItems.includes(item.id)); | ||||
|  | ||||
| const LibraryMenuWrapper = forwardRef< | ||||
|   HTMLDivElement, | ||||
|   { children: React.ReactNode } | ||||
| >(({ children }, ref) => { | ||||
|   return ( | ||||
|     <Island padding={1} ref={ref} className="layer-ui__library"> | ||||
|       {children} | ||||
|     </Island> | ||||
|   ); | ||||
| }); | ||||
|  | ||||
| export const LibraryMenu = ({ | ||||
|   onClose, | ||||
|   onInsertShape, | ||||
|   pendingElements, | ||||
|   onAddToLibrary, | ||||
|   theme, | ||||
|   setAppState, | ||||
|   files, | ||||
|   libraryReturnUrl, | ||||
|   focusContainer, | ||||
|   library, | ||||
|   id, | ||||
|   appState, | ||||
| }: { | ||||
|   pendingElements: LibraryItem["elements"]; | ||||
|   onClose: () => void; | ||||
|   onInsertShape: (elements: LibraryItem["elements"]) => void; | ||||
|   onAddToLibrary: () => void; | ||||
|   theme: AppState["theme"]; | ||||
|   files: BinaryFiles; | ||||
|   setAppState: React.Component<any, AppState>["setState"]; | ||||
|   libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"]; | ||||
|   focusContainer: () => void; | ||||
|   library: Library; | ||||
|   id: string; | ||||
|   appState: AppState; | ||||
| }) => { | ||||
|   const ref = useRef<HTMLDivElement | null>(null); | ||||
|  | ||||
|   useOnClickOutside(ref, (event) => { | ||||
|     // If click on the library icon, do nothing. | ||||
|     if ((event.target as Element).closest(".ToolIcon__library")) { | ||||
|       return; | ||||
|     } | ||||
|     onClose(); | ||||
|   }); | ||||
|  | ||||
|   useEffect(() => { | ||||
|     const handleKeyDown = (event: KeyboardEvent) => { | ||||
|       if (event.key === KEYS.ESCAPE) { | ||||
|         onClose(); | ||||
|       } | ||||
|     }; | ||||
|     document.addEventListener(EVENT.KEYDOWN, handleKeyDown); | ||||
|     return () => { | ||||
|       document.removeEventListener(EVENT.KEYDOWN, handleKeyDown); | ||||
|     }; | ||||
|   }, [onClose]); | ||||
|  | ||||
|   const [selectedItems, setSelectedItems] = useState<LibraryItem["id"][]>([]); | ||||
|   const [showPublishLibraryDialog, setShowPublishLibraryDialog] = | ||||
|     useState(false); | ||||
|   const [publishLibSuccess, setPublishLibSuccess] = useState<null | { | ||||
|     url: string; | ||||
|     authorName: string; | ||||
|   }>(null); | ||||
|  | ||||
|   const [libraryItemsData] = useAtom(libraryItemsAtom, jotaiScope); | ||||
|  | ||||
|   const removeFromLibrary = useCallback( | ||||
|     async (libraryItems: LibraryItems) => { | ||||
|       const nextItems = libraryItems.filter( | ||||
|         (item) => !selectedItems.includes(item.id), | ||||
|       ); | ||||
|       library.saveLibrary(nextItems).catch(() => { | ||||
|         setAppState({ errorMessage: t("alerts.errorRemovingFromLibrary") }); | ||||
|       }); | ||||
|       setSelectedItems([]); | ||||
|     }, | ||||
|     [library, setAppState, selectedItems, setSelectedItems], | ||||
|   ); | ||||
|  | ||||
|   const resetLibrary = useCallback(() => { | ||||
|     library.resetLibrary(); | ||||
|     focusContainer(); | ||||
|   }, [library, focusContainer]); | ||||
|  | ||||
|   const addToLibrary = useCallback( | ||||
|     async (elements: LibraryItem["elements"], libraryItems: LibraryItems) => { | ||||
|       trackEvent("element", "addToLibrary", "ui"); | ||||
|       if (elements.some((element) => element.type === "image")) { | ||||
|         return setAppState({ | ||||
|           errorMessage: "Support for adding images to the library coming soon!", | ||||
|         }); | ||||
|       } | ||||
|       const nextItems: LibraryItems = [ | ||||
|         { | ||||
|           status: "unpublished", | ||||
|           elements, | ||||
|           id: randomId(), | ||||
|           created: Date.now(), | ||||
|         }, | ||||
|         ...libraryItems, | ||||
|       ]; | ||||
|       onAddToLibrary(); | ||||
|       library.saveLibrary(nextItems).catch(() => { | ||||
|         setAppState({ errorMessage: t("alerts.errorAddingToLibrary") }); | ||||
|       }); | ||||
|     }, | ||||
|     [onAddToLibrary, library, setAppState], | ||||
|   ); | ||||
|  | ||||
|   const renderPublishSuccess = useCallback(() => { | ||||
|     return ( | ||||
|       <Dialog | ||||
|         onCloseRequest={() => setPublishLibSuccess(null)} | ||||
|         title={t("publishSuccessDialog.title")} | ||||
|         className="publish-library-success" | ||||
|         small={true} | ||||
|       > | ||||
|         <p> | ||||
|           {t("publishSuccessDialog.content", { | ||||
|             authorName: publishLibSuccess!.authorName, | ||||
|           })}{" "} | ||||
|           <a | ||||
|             href={publishLibSuccess?.url} | ||||
|             target="_blank" | ||||
|             rel="noopener noreferrer" | ||||
|           > | ||||
|             {t("publishSuccessDialog.link")} | ||||
|           </a> | ||||
|         </p> | ||||
|         <ToolButton | ||||
|           type="button" | ||||
|           title={t("buttons.close")} | ||||
|           aria-label={t("buttons.close")} | ||||
|           label={t("buttons.close")} | ||||
|           onClick={() => setPublishLibSuccess(null)} | ||||
|           data-testid="publish-library-success-close" | ||||
|           className="publish-library-success-close" | ||||
|         /> | ||||
|       </Dialog> | ||||
|     ); | ||||
|   }, [setPublishLibSuccess, publishLibSuccess]); | ||||
|  | ||||
|   const onPublishLibSuccess = useCallback( | ||||
|     (data, libraryItems: LibraryItems) => { | ||||
|       setShowPublishLibraryDialog(false); | ||||
|       setPublishLibSuccess({ url: data.url, authorName: data.authorName }); | ||||
|       const nextLibItems = libraryItems.slice(); | ||||
|       nextLibItems.forEach((libItem) => { | ||||
|         if (selectedItems.includes(libItem.id)) { | ||||
|           libItem.status = "published"; | ||||
|         } | ||||
|       }); | ||||
|       library.saveLibrary(nextLibItems); | ||||
|     }, | ||||
|     [setShowPublishLibraryDialog, setPublishLibSuccess, selectedItems, library], | ||||
|   ); | ||||
|  | ||||
|   const [lastSelectedItem, setLastSelectedItem] = useState< | ||||
|     LibraryItem["id"] | null | ||||
|   >(null); | ||||
|  | ||||
|   if (libraryItemsData.status === "loading") { | ||||
|     return ( | ||||
|       <LibraryMenuWrapper ref={ref}> | ||||
|         <div className="layer-ui__library-message"> | ||||
|           <Spinner size="2em" /> | ||||
|           <span>{t("labels.libraryLoadingMessage")}</span> | ||||
|         </div> | ||||
|       </LibraryMenuWrapper> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     <LibraryMenuWrapper ref={ref}> | ||||
|       {showPublishLibraryDialog && ( | ||||
|         <PublishLibrary | ||||
|           onClose={() => setShowPublishLibraryDialog(false)} | ||||
|           libraryItems={getSelectedItems( | ||||
|             libraryItemsData.libraryItems, | ||||
|             selectedItems, | ||||
|           )} | ||||
|           appState={appState} | ||||
|           onSuccess={(data) => | ||||
|             onPublishLibSuccess(data, libraryItemsData.libraryItems) | ||||
|           } | ||||
|           onError={(error) => window.alert(error)} | ||||
|           updateItemsInStorage={() => | ||||
|             library.saveLibrary(libraryItemsData.libraryItems) | ||||
|           } | ||||
|           onRemove={(id: string) => | ||||
|             setSelectedItems(selectedItems.filter((_id) => _id !== id)) | ||||
|           } | ||||
|         /> | ||||
|       )} | ||||
|       {publishLibSuccess && renderPublishSuccess()} | ||||
|       <LibraryMenuItems | ||||
|         libraryItems={libraryItemsData.libraryItems} | ||||
|         onRemoveFromLibrary={() => | ||||
|           removeFromLibrary(libraryItemsData.libraryItems) | ||||
|         } | ||||
|         onAddToLibrary={(elements) => | ||||
|           addToLibrary(elements, libraryItemsData.libraryItems) | ||||
|         } | ||||
|         onInsertShape={onInsertShape} | ||||
|         pendingElements={pendingElements} | ||||
|         setAppState={setAppState} | ||||
|         libraryReturnUrl={libraryReturnUrl} | ||||
|         library={library} | ||||
|         theme={theme} | ||||
|         files={files} | ||||
|         id={id} | ||||
|         selectedItems={selectedItems} | ||||
|         onToggle={(id, event) => { | ||||
|           const shouldSelect = !selectedItems.includes(id); | ||||
|  | ||||
|           if (shouldSelect) { | ||||
|             if (event.shiftKey && lastSelectedItem) { | ||||
|               const rangeStart = libraryItemsData.libraryItems.findIndex( | ||||
|                 (item) => item.id === lastSelectedItem, | ||||
|               ); | ||||
|               const rangeEnd = libraryItemsData.libraryItems.findIndex( | ||||
|                 (item) => item.id === id, | ||||
|               ); | ||||
|  | ||||
|               if (rangeStart === -1 || rangeEnd === -1) { | ||||
|                 setSelectedItems([...selectedItems, id]); | ||||
|                 return; | ||||
|               } | ||||
|  | ||||
|               const selectedItemsMap = arrayToMap(selectedItems); | ||||
|               const nextSelectedIds = libraryItemsData.libraryItems.reduce( | ||||
|                 (acc: LibraryItem["id"][], item, idx) => { | ||||
|                   if ( | ||||
|                     (idx >= rangeStart && idx <= rangeEnd) || | ||||
|                     selectedItemsMap.has(item.id) | ||||
|                   ) { | ||||
|                     acc.push(item.id); | ||||
|                   } | ||||
|                   return acc; | ||||
|                 }, | ||||
|                 [], | ||||
|               ); | ||||
|  | ||||
|               setSelectedItems(nextSelectedIds); | ||||
|             } else { | ||||
|               setSelectedItems([...selectedItems, id]); | ||||
|             } | ||||
|             setLastSelectedItem(id); | ||||
|           } else { | ||||
|             setLastSelectedItem(null); | ||||
|             setSelectedItems(selectedItems.filter((_id) => _id !== id)); | ||||
|           } | ||||
|         }} | ||||
|         onPublish={() => setShowPublishLibraryDialog(true)} | ||||
|         resetLibrary={resetLibrary} | ||||
|       /> | ||||
|     </LibraryMenuWrapper> | ||||
|   ); | ||||
| }; | ||||
| @@ -1,102 +0,0 @@ | ||||
| @import "open-color/open-color"; | ||||
|  | ||||
| .excalidraw { | ||||
|   .library-menu-items-container { | ||||
|     .library-actions { | ||||
|       display: flex; | ||||
|  | ||||
|       button .library-actions-counter { | ||||
|         position: absolute; | ||||
|         right: 2px; | ||||
|         bottom: 2px; | ||||
|         border-radius: 50%; | ||||
|         width: 1em; | ||||
|         height: 1em; | ||||
|         padding: 1px; | ||||
|         font-size: 0.7rem; | ||||
|         background: #fff; | ||||
|       } | ||||
|  | ||||
|       &--remove { | ||||
|         background-color: $oc-red-7; | ||||
|         &:hover { | ||||
|           background-color: $oc-red-8; | ||||
|         } | ||||
|         &:active { | ||||
|           background-color: $oc-red-9; | ||||
|         } | ||||
|         svg { | ||||
|           color: $oc-white; | ||||
|         } | ||||
|         .library-actions-counter { | ||||
|           color: $oc-red-7; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       &--export { | ||||
|         background-color: $oc-lime-5; | ||||
|  | ||||
|         &:hover { | ||||
|           background-color: $oc-lime-7; | ||||
|         } | ||||
|  | ||||
|         &:active { | ||||
|           background-color: $oc-lime-8; | ||||
|         } | ||||
|         svg { | ||||
|           color: $oc-white; | ||||
|         } | ||||
|         .library-actions-counter { | ||||
|           color: $oc-lime-5; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       &--publish { | ||||
|         background-color: $oc-cyan-6; | ||||
|         &:hover { | ||||
|           background-color: $oc-cyan-7; | ||||
|         } | ||||
|         &:active { | ||||
|           background-color: $oc-cyan-9; | ||||
|         } | ||||
|         svg { | ||||
|           color: $oc-white; | ||||
|         } | ||||
|         label { | ||||
|           margin-left: -0.2em; | ||||
|           margin-right: 1.1em; | ||||
|           color: $oc-white; | ||||
|           font-size: 0.86em; | ||||
|         } | ||||
|         .library-actions-counter { | ||||
|           color: $oc-cyan-6; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       &--load { | ||||
|         background-color: $oc-blue-6; | ||||
|         &:hover { | ||||
|           background-color: $oc-blue-7; | ||||
|         } | ||||
|         &:active { | ||||
|           background-color: $oc-blue-9; | ||||
|         } | ||||
|         svg { | ||||
|           color: $oc-white; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     &__items { | ||||
|       max-height: 50vh; | ||||
|       overflow: auto; | ||||
|       margin-top: 0.5rem; | ||||
|     } | ||||
|  | ||||
|     .separator { | ||||
|       font-weight: 500; | ||||
|       font-size: 0.9rem; | ||||
|       margin: 0.6em 0.2em; | ||||
|       color: var(--text-primary-color); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,318 +0,0 @@ | ||||
| import { chunk } from "lodash"; | ||||
| import { useCallback, useState } from "react"; | ||||
| import { importLibraryFromJSON, saveLibraryAsJSON } from "../data/json"; | ||||
| import Library from "../data/library"; | ||||
| import { ExcalidrawElement, NonDeleted } from "../element/types"; | ||||
| import { t } from "../i18n"; | ||||
| import { | ||||
|   AppState, | ||||
|   BinaryFiles, | ||||
|   ExcalidrawProps, | ||||
|   LibraryItem, | ||||
|   LibraryItems, | ||||
| } from "../types"; | ||||
| import { muteFSAbortError } from "../utils"; | ||||
| import { useDeviceType } from "./App"; | ||||
| import ConfirmDialog from "./ConfirmDialog"; | ||||
| import { exportToFileIcon, load, publishIcon, trash } from "./icons"; | ||||
| import { LibraryUnit } from "./LibraryUnit"; | ||||
| import Stack from "./Stack"; | ||||
| import { ToolButton } from "./ToolButton"; | ||||
| import { Tooltip } from "./Tooltip"; | ||||
|  | ||||
| import "./LibraryMenuItems.scss"; | ||||
| import { VERSIONS } from "../constants"; | ||||
|  | ||||
| const LibraryMenuItems = ({ | ||||
|   libraryItems, | ||||
|   onRemoveFromLibrary, | ||||
|   onAddToLibrary, | ||||
|   onInsertShape, | ||||
|   pendingElements, | ||||
|   theme, | ||||
|   setAppState, | ||||
|   libraryReturnUrl, | ||||
|   library, | ||||
|   files, | ||||
|   id, | ||||
|   selectedItems, | ||||
|   onToggle, | ||||
|   onPublish, | ||||
|   resetLibrary, | ||||
| }: { | ||||
|   libraryItems: LibraryItems; | ||||
|   pendingElements: LibraryItem["elements"]; | ||||
|   onRemoveFromLibrary: () => void; | ||||
|   onInsertShape: (elements: LibraryItem["elements"]) => void; | ||||
|   onAddToLibrary: (elements: LibraryItem["elements"]) => void; | ||||
|   theme: AppState["theme"]; | ||||
|   files: BinaryFiles; | ||||
|   setAppState: React.Component<any, AppState>["setState"]; | ||||
|   libraryReturnUrl: ExcalidrawProps["libraryReturnUrl"]; | ||||
|   library: Library; | ||||
|   id: string; | ||||
|   selectedItems: LibraryItem["id"][]; | ||||
|   onToggle: (id: LibraryItem["id"], event: React.MouseEvent) => void; | ||||
|   onPublish: () => void; | ||||
|   resetLibrary: () => void; | ||||
| }) => { | ||||
|   const renderRemoveLibAlert = useCallback(() => { | ||||
|     const content = selectedItems.length | ||||
|       ? t("alerts.removeItemsFromsLibrary", { count: selectedItems.length }) | ||||
|       : t("alerts.resetLibrary"); | ||||
|     const title = selectedItems.length | ||||
|       ? t("confirmDialog.removeItemsFromLib") | ||||
|       : t("confirmDialog.resetLibrary"); | ||||
|     return ( | ||||
|       <ConfirmDialog | ||||
|         onConfirm={() => { | ||||
|           if (selectedItems.length) { | ||||
|             onRemoveFromLibrary(); | ||||
|           } else { | ||||
|             resetLibrary(); | ||||
|           } | ||||
|           setShowRemoveLibAlert(false); | ||||
|         }} | ||||
|         onCancel={() => { | ||||
|           setShowRemoveLibAlert(false); | ||||
|         }} | ||||
|         title={title} | ||||
|       > | ||||
|         <p>{content}</p> | ||||
|       </ConfirmDialog> | ||||
|     ); | ||||
|   }, [selectedItems, onRemoveFromLibrary, resetLibrary]); | ||||
|  | ||||
|   const [showRemoveLibAlert, setShowRemoveLibAlert] = useState(false); | ||||
|  | ||||
|   const isMobile = useDeviceType().isMobile; | ||||
|  | ||||
|   const renderLibraryActions = () => { | ||||
|     const itemsSelected = !!selectedItems.length; | ||||
|     const items = itemsSelected | ||||
|       ? libraryItems.filter((item) => selectedItems.includes(item.id)) | ||||
|       : libraryItems; | ||||
|     const resetLabel = itemsSelected | ||||
|       ? t("buttons.remove") | ||||
|       : t("buttons.resetLibrary"); | ||||
|     return ( | ||||
|       <div className="library-actions"> | ||||
|         {(!itemsSelected || !isMobile) && ( | ||||
|           <ToolButton | ||||
|             key="import" | ||||
|             type="button" | ||||
|             title={t("buttons.load")} | ||||
|             aria-label={t("buttons.load")} | ||||
|             icon={load} | ||||
|             onClick={() => { | ||||
|               importLibraryFromJSON(library) | ||||
|                 .catch(muteFSAbortError) | ||||
|                 .catch((error) => { | ||||
|                   setAppState({ errorMessage: error.message }); | ||||
|                 }); | ||||
|             }} | ||||
|             className="library-actions--load" | ||||
|           /> | ||||
|         )} | ||||
|         {!!items.length && ( | ||||
|           <> | ||||
|             <ToolButton | ||||
|               key="export" | ||||
|               type="button" | ||||
|               title={t("buttons.export")} | ||||
|               aria-label={t("buttons.export")} | ||||
|               icon={exportToFileIcon} | ||||
|               onClick={async () => { | ||||
|                 const libraryItems = itemsSelected | ||||
|                   ? items | ||||
|                   : await library.loadLibrary(); | ||||
|                 saveLibraryAsJSON(libraryItems) | ||||
|                   .catch(muteFSAbortError) | ||||
|                   .catch((error) => { | ||||
|                     setAppState({ errorMessage: error.message }); | ||||
|                   }); | ||||
|               }} | ||||
|               className="library-actions--export" | ||||
|             > | ||||
|               {selectedItems.length > 0 && ( | ||||
|                 <span className="library-actions-counter"> | ||||
|                   {selectedItems.length} | ||||
|                 </span> | ||||
|               )} | ||||
|             </ToolButton> | ||||
|             <ToolButton | ||||
|               key="reset" | ||||
|               type="button" | ||||
|               title={resetLabel} | ||||
|               aria-label={resetLabel} | ||||
|               icon={trash} | ||||
|               onClick={() => setShowRemoveLibAlert(true)} | ||||
|               className="library-actions--remove" | ||||
|             > | ||||
|               {selectedItems.length > 0 && ( | ||||
|                 <span className="library-actions-counter"> | ||||
|                   {selectedItems.length} | ||||
|                 </span> | ||||
|               )} | ||||
|             </ToolButton> | ||||
|           </> | ||||
|         )} | ||||
|         {itemsSelected && !isPublished && ( | ||||
|           <Tooltip label={t("hints.publishLibrary")}> | ||||
|             <ToolButton | ||||
|               type="button" | ||||
|               aria-label={t("buttons.publishLibrary")} | ||||
|               label={t("buttons.publishLibrary")} | ||||
|               icon={publishIcon} | ||||
|               className="library-actions--publish" | ||||
|               onClick={onPublish} | ||||
|             > | ||||
|               {!isMobile && <label>{t("buttons.publishLibrary")}</label>} | ||||
|               {selectedItems.length > 0 && ( | ||||
|                 <span className="library-actions-counter"> | ||||
|                   {selectedItems.length} | ||||
|                 </span> | ||||
|               )} | ||||
|             </ToolButton> | ||||
|           </Tooltip> | ||||
|         )} | ||||
|       </div> | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   const CELLS_PER_ROW = isMobile ? 4 : 6; | ||||
|  | ||||
|   const referrer = | ||||
|     libraryReturnUrl || window.location.origin + window.location.pathname; | ||||
|   const isPublished = selectedItems.some( | ||||
|     (id) => libraryItems.find((item) => item.id === id)?.status === "published", | ||||
|   ); | ||||
|  | ||||
|   const createLibraryItemCompo = (params: { | ||||
|     item: | ||||
|       | LibraryItem | ||||
|       | /* pending library item */ { | ||||
|           id: null; | ||||
|           elements: readonly NonDeleted<ExcalidrawElement>[]; | ||||
|         } | ||||
|       | null; | ||||
|     onClick?: () => void; | ||||
|     key: string; | ||||
|   }) => { | ||||
|     return ( | ||||
|       <Stack.Col key={params.key}> | ||||
|         <LibraryUnit | ||||
|           elements={params.item?.elements} | ||||
|           files={files} | ||||
|           isPending={!params.item?.id && !!params.item?.elements} | ||||
|           onClick={params.onClick || (() => {})} | ||||
|           id={params.item?.id || null} | ||||
|           selected={!!params.item?.id && selectedItems.includes(params.item.id)} | ||||
|           onToggle={(id, event) => { | ||||
|             onToggle(id, event); | ||||
|           }} | ||||
|         /> | ||||
|       </Stack.Col> | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   const renderLibrarySection = ( | ||||
|     items: ( | ||||
|       | LibraryItem | ||||
|       | /* pending library item */ { | ||||
|           id: null; | ||||
|           elements: readonly NonDeleted<ExcalidrawElement>[]; | ||||
|         } | ||||
|     )[], | ||||
|   ) => { | ||||
|     const _items = items.map((item) => { | ||||
|       if (item.id) { | ||||
|         return createLibraryItemCompo({ | ||||
|           item, | ||||
|           onClick: () => onInsertShape(item.elements), | ||||
|           key: item.id, | ||||
|         }); | ||||
|       } | ||||
|       return createLibraryItemCompo({ | ||||
|         key: "__pending__item__", | ||||
|         item, | ||||
|         onClick: () => onAddToLibrary(pendingElements), | ||||
|       }); | ||||
|     }); | ||||
|  | ||||
|     // ensure we render all empty cells if no items are present | ||||
|     let rows = chunk(_items, CELLS_PER_ROW); | ||||
|     if (!rows.length) { | ||||
|       rows = [[]]; | ||||
|     } | ||||
|  | ||||
|     return rows.map((rowItems, index, rows) => { | ||||
|       if (index === rows.length - 1) { | ||||
|         // pad row with empty cells | ||||
|         rowItems = rowItems.concat( | ||||
|           new Array(CELLS_PER_ROW - rowItems.length) | ||||
|             .fill(null) | ||||
|             .map((_, index) => { | ||||
|               return createLibraryItemCompo({ | ||||
|                 key: `empty_${index}`, | ||||
|                 item: null, | ||||
|               }); | ||||
|             }), | ||||
|         ); | ||||
|       } | ||||
|       return ( | ||||
|         <Stack.Row align="center" gap={1} key={index}> | ||||
|           {rowItems} | ||||
|         </Stack.Row> | ||||
|       ); | ||||
|     }); | ||||
|   }; | ||||
|  | ||||
|   const publishedItems = libraryItems.filter( | ||||
|     (item) => item.status === "published", | ||||
|   ); | ||||
|   const unpublishedItems = [ | ||||
|     // append pending library item | ||||
|     ...(pendingElements.length | ||||
|       ? [{ id: null, elements: pendingElements }] | ||||
|       : []), | ||||
|     ...libraryItems.filter((item) => item.status !== "published"), | ||||
|   ]; | ||||
|  | ||||
|   return ( | ||||
|     <div className="library-menu-items-container"> | ||||
|       {showRemoveLibAlert && renderRemoveLibAlert()} | ||||
|       <div className="layer-ui__library-header" key="library-header"> | ||||
|         {renderLibraryActions()} | ||||
|         <a | ||||
|           href={`${process.env.REACT_APP_LIBRARY_URL}?target=${ | ||||
|             window.name || "_blank" | ||||
|           }&referrer=${referrer}&useHash=true&token=${id}&theme=${theme}&version=${ | ||||
|             VERSIONS.excalidrawLibrary | ||||
|           }`} | ||||
|           target="_excalidraw_libraries" | ||||
|         > | ||||
|           {t("labels.libraries")} | ||||
|         </a> | ||||
|       </div> | ||||
|       <Stack.Col | ||||
|         className="library-menu-items-container__items" | ||||
|         align="start" | ||||
|         gap={1} | ||||
|       > | ||||
|         <> | ||||
|           <div className="separator">{t("labels.personalLib")}</div> | ||||
|           {renderLibrarySection(unpublishedItems)} | ||||
|         </> | ||||
|  | ||||
|         <> | ||||
|           <div className="separator">{t("labels.excalidrawLib")} </div> | ||||
|  | ||||
|           {renderLibrarySection(publishedItems)} | ||||
|         </> | ||||
|       </Stack.Col> | ||||
|     </div> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default LibraryMenuItems; | ||||
| @@ -1,5 +1,3 @@ | ||||
| @import "../css/variables.module"; | ||||
|  | ||||
| .excalidraw { | ||||
|   .library-unit { | ||||
|     align-items: center; | ||||
| @@ -9,26 +7,10 @@ | ||||
|     position: relative; | ||||
|     width: 63px; | ||||
|     height: 63px; // match width | ||||
|  | ||||
|     &--hover { | ||||
|       box-shadow: inset 0px 0px 0px 2px $oc-blue-5; | ||||
|       border-color: $oc-blue-5; | ||||
|     } | ||||
|  | ||||
|     &--selected { | ||||
|       box-shadow: inset 0px 0px 0px 2px $oc-blue-8; | ||||
|       border-color: $oc-blue-8; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   &.theme--dark .library-unit { | ||||
|     border-color: rgb(48, 48, 48); | ||||
|   } | ||||
|  | ||||
|   .library-unit__dragger { | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|     justify-content: center; | ||||
|     height: 100%; | ||||
|     width: 100%; | ||||
|   } | ||||
| @@ -40,9 +22,9 @@ | ||||
|     max-width: 100%; | ||||
|   } | ||||
|  | ||||
|   .library-unit__checkbox-container, | ||||
|   .library-unit__checkbox-container:hover, | ||||
|   .library-unit__checkbox-container:active { | ||||
|   .library-unit__removeFromLibrary, | ||||
|   .library-unit__removeFromLibrary:hover, | ||||
|   .library-unit__removeFromLibrary:active { | ||||
|     align-items: center; | ||||
|     background: none; | ||||
|     border: none; | ||||
| @@ -50,35 +32,10 @@ | ||||
|     display: flex; | ||||
|     justify-content: center; | ||||
|     margin: 0; | ||||
|     padding: 0.5rem; | ||||
|     padding: 0; | ||||
|     position: absolute; | ||||
|     left: 2rem; | ||||
|     bottom: 2rem; | ||||
|     cursor: pointer; | ||||
|  | ||||
|     input { | ||||
|       cursor: pointer; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .library-unit__checkbox { | ||||
|     position: absolute; | ||||
|     left: 2.3rem; | ||||
|     bottom: 2.3rem; | ||||
|  | ||||
|     .Checkbox-box { | ||||
|       width: 13px; | ||||
|       height: 13px; | ||||
|       border-radius: 2px; | ||||
|       margin: 0.5em 0.5em 0.2em 0.2em; | ||||
|       background-color: $oc-blue-1; | ||||
|     } | ||||
|  | ||||
|     &.Checkbox:hover { | ||||
|       .Checkbox-box { | ||||
|         background-color: $oc-blue-2; | ||||
|       } | ||||
|     } | ||||
|     right: 5px; | ||||
|     top: 5px; | ||||
|   } | ||||
|  | ||||
|   .library-unit__removeFromLibrary > svg { | ||||
| @@ -86,37 +43,29 @@ | ||||
|     width: 16px; | ||||
|   } | ||||
|  | ||||
|   .library-unit__adder { | ||||
|   .library-unit__pulse { | ||||
|     transform: scale(1); | ||||
|     animation: library-unit__adder-animation 1s ease-in infinite; | ||||
|     animation: library-unit__pulse-animation 1s ease-in infinite; | ||||
|   } | ||||
|  | ||||
|   .library-unit__adder { | ||||
|     position: absolute; | ||||
|     left: 40%; | ||||
|     top: 40%; | ||||
|     width: 2rem; | ||||
|     height: 2rem; | ||||
|     left: 50%; | ||||
|     top: 50%; | ||||
|     width: 20px; | ||||
|     height: 20px; | ||||
|     margin-left: -10px; | ||||
|     margin-top: -10px; | ||||
|     pointer-events: none; | ||||
|   } | ||||
|   .library-unit:hover .library-unit__adder { | ||||
|     fill: $oc-blue-7; | ||||
|   } | ||||
|   .library-unit:active .library-unit__adder { | ||||
|     animation: none; | ||||
|     transform: scale(0.8); | ||||
|     fill: $oc-black; | ||||
|   } | ||||
|  | ||||
|   .library-unit__active { | ||||
|     cursor: pointer; | ||||
|   } | ||||
|  | ||||
|   @keyframes library-unit__adder-animation { | ||||
|   @keyframes library-unit__pulse-animation { | ||||
|     0% { | ||||
|       transform: scale(0.85); | ||||
|       transform: scale(0.95); | ||||
|     } | ||||
|  | ||||
|     50% { | ||||
| @@ -124,7 +73,7 @@ | ||||
|     } | ||||
|  | ||||
|     100% { | ||||
|       transform: scale(0.85); | ||||
|       transform: scale(0.95); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,103 +1,82 @@ | ||||
| import clsx from "clsx"; | ||||
| import oc from "open-color"; | ||||
| import { useEffect, useRef, useState } from "react"; | ||||
| import React, { useEffect, useRef, useState } from "react"; | ||||
| import { close } from "../components/icons"; | ||||
| import { MIME_TYPES } from "../constants"; | ||||
| import { useDeviceType } from "../components/App"; | ||||
| import { t } from "../i18n"; | ||||
| import { useIsMobile } from "../components/App"; | ||||
| import { exportToSvg } from "../scene/export"; | ||||
| import { BinaryFiles, LibraryItem } from "../types"; | ||||
| import { LibraryItem } from "../types"; | ||||
| import "./LibraryUnit.scss"; | ||||
| import { CheckboxItem } from "./CheckboxItem"; | ||||
|  | ||||
| // fa-plus | ||||
| const PLUS_ICON = ( | ||||
|   <svg viewBox="0 0 1792 1792"> | ||||
|     <path | ||||
|       d="M1600 736v192c0 26.667-9.33 49.333-28 68-18.67 18.67-41.33 28-68 28h-416v416c0 26.67-9.33 49.33-28 68s-41.33 28-68 28H800c-26.667 0-49.333-9.33-68-28s-28-41.33-28-68v-416H288c-26.667 0-49.333-9.33-68-28-18.667-18.667-28-41.333-28-68V736c0-26.667 9.333-49.333 28-68s41.333-28 68-28h416V224c0-26.667 9.333-49.333 28-68s41.333-28 68-28h192c26.67 0 49.33 9.333 68 28s28 41.333 28 68v416h416c26.67 0 49.33 9.333 68 28s28 41.333 28 68Z" | ||||
|       style={{ | ||||
|         stroke: "#fff", | ||||
|         strokeWidth: 140, | ||||
|       }} | ||||
|       transform="translate(0 64)" | ||||
|       fill="currentColor" | ||||
|       d="M1600 736v192q0 40-28 68t-68 28h-416v416q0 40-28 68t-68 28h-192q-40 0-68-28t-28-68v-416h-416q-40 0-68-28t-28-68v-192q0-40 28-68t68-28h416v-416q0-40 28-68t68-28h192q40 0 68 28t28 68v416h416q40 0 68 28t28 68z" | ||||
|     /> | ||||
|   </svg> | ||||
| ); | ||||
|  | ||||
| export const LibraryUnit = ({ | ||||
|   id, | ||||
|   elements, | ||||
|   files, | ||||
|   isPending, | ||||
|   pendingElements, | ||||
|   onRemoveFromLibrary, | ||||
|   onClick, | ||||
|   selected, | ||||
|   onToggle, | ||||
| }: { | ||||
|   id: LibraryItem["id"] | /** for pending item */ null; | ||||
|   elements?: LibraryItem["elements"]; | ||||
|   files: BinaryFiles; | ||||
|   isPending?: boolean; | ||||
|   elements?: LibraryItem; | ||||
|   pendingElements?: LibraryItem; | ||||
|   onRemoveFromLibrary: () => void; | ||||
|   onClick: () => void; | ||||
|   selected: boolean; | ||||
|   onToggle: (id: string, event: React.MouseEvent) => void; | ||||
| }) => { | ||||
|   const ref = useRef<HTMLDivElement | null>(null); | ||||
|   useEffect(() => { | ||||
|     const node = ref.current; | ||||
|     if (!node) { | ||||
|     const elementsToRender = elements || pendingElements; | ||||
|     if (!elementsToRender) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     (async () => { | ||||
|       if (!elements) { | ||||
|         return; | ||||
|     const svg = exportToSvg(elementsToRender, { | ||||
|       exportBackground: false, | ||||
|       viewBackgroundColor: oc.white, | ||||
|       shouldAddWatermark: false, | ||||
|     }); | ||||
|     for (const child of ref.current!.children) { | ||||
|       if (child.tagName !== "svg") { | ||||
|         continue; | ||||
|       } | ||||
|       const svg = await exportToSvg( | ||||
|         elements, | ||||
|         { | ||||
|           exportBackground: false, | ||||
|           viewBackgroundColor: oc.white, | ||||
|         }, | ||||
|         files, | ||||
|       ); | ||||
|       node.innerHTML = svg.outerHTML; | ||||
|     })(); | ||||
|       ref.current!.removeChild(child); | ||||
|     } | ||||
|     ref.current!.appendChild(svg); | ||||
|  | ||||
|     const current = ref.current!; | ||||
|     return () => { | ||||
|       node.innerHTML = ""; | ||||
|       current.removeChild(svg); | ||||
|     }; | ||||
|   }, [elements, files]); | ||||
|   }, [elements, pendingElements]); | ||||
|  | ||||
|   const [isHovered, setIsHovered] = useState(false); | ||||
|   const isMobile = useDeviceType().isMobile; | ||||
|   const adder = isPending && ( | ||||
|   const isMobile = useIsMobile(); | ||||
|  | ||||
|   const adder = (isHovered || isMobile) && pendingElements && ( | ||||
|     <div className="library-unit__adder">{PLUS_ICON}</div> | ||||
|   ); | ||||
|  | ||||
|   return ( | ||||
|     <div | ||||
|       className={clsx("library-unit", { | ||||
|         "library-unit__active": elements, | ||||
|         "library-unit--hover": elements && isHovered, | ||||
|         "library-unit--selected": selected, | ||||
|         "library-unit__active": elements || pendingElements, | ||||
|       })} | ||||
|       onMouseEnter={() => setIsHovered(true)} | ||||
|       onMouseLeave={() => setIsHovered(false)} | ||||
|     > | ||||
|       <div | ||||
|         className={clsx("library-unit__dragger", { | ||||
|           "library-unit__pulse": !!isPending, | ||||
|           "library-unit__pulse": !!pendingElements, | ||||
|         })} | ||||
|         ref={ref} | ||||
|         draggable={!!elements} | ||||
|         onClick={ | ||||
|           !!elements || !!isPending | ||||
|             ? (event) => { | ||||
|                 if (id && event.shiftKey) { | ||||
|                   onToggle(id, event); | ||||
|                 } else { | ||||
|                   onClick(); | ||||
|                 } | ||||
|               } | ||||
|             : undefined | ||||
|         } | ||||
|         onClick={!!elements || !!pendingElements ? onClick : undefined} | ||||
|         onDragStart={(event) => { | ||||
|           setIsHovered(false); | ||||
|           event.dataTransfer.setData( | ||||
| @@ -107,12 +86,14 @@ export const LibraryUnit = ({ | ||||
|         }} | ||||
|       /> | ||||
|       {adder} | ||||
|       {id && elements && (isHovered || isMobile || selected) && ( | ||||
|         <CheckboxItem | ||||
|           checked={selected} | ||||
|           onChange={(checked, event) => onToggle(id, event)} | ||||
|           className="library-unit__checkbox" | ||||
|         /> | ||||
|       {elements && (isHovered || isMobile) && ( | ||||
|         <button | ||||
|           className="library-unit__removeFromLibrary" | ||||
|           aria-label={t("labels.removeFromLibrary")} | ||||
|           onClick={onRemoveFromLibrary} | ||||
|         > | ||||
|           {close} | ||||
|         </button> | ||||
|       )} | ||||
|     </div> | ||||
|   ); | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user